aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--kernel/drivers/ide.c63
-rw-r--r--kernel/drivers/net.c122
-rw-r--r--kernel/inc/ide.h3
-rw-r--r--kernel/inc/net.h27
-rw-r--r--libc/cpu.c7
5 files changed, 160 insertions, 62 deletions
diff --git a/kernel/drivers/ide.c b/kernel/drivers/ide.c
index eda18e9..c1c0227 100644
--- a/kernel/drivers/ide.c
+++ b/kernel/drivers/ide.c
@@ -5,51 +5,42 @@
#include <ide.h>
#include <print.h>
-int ide_wait(int check)
+int ide_stat()
{
- char r;
-
- // Wait while drive is busy. Once just ready is set, exit the loop
- while (((r = (char)inb(IDE_IO | IDE_CMD)) & (IDE_BUSY | IDE_READY)) != IDE_READY)
- ;
+ inb(IDE_IO + IDE_CMD);
+ inb(IDE_IO + IDE_CMD);
+ inb(IDE_IO + IDE_CMD);
+ inb(IDE_IO + IDE_CMD);
+ return inb(IDE_IO + IDE_CMD);
+}
- // Check for errors
- if (check && (r & (IDE_DRIVE_FAULT | IDE_ERROR)) != 0)
- return 0xF;
- return 0;
+void ide_wait()
+{
+ u8 stat;
+ do
+ stat = ide_stat();
+ while ((stat & IDE_BUSY) != 0);
}
-// TODO: Fix strange print workaround
+// TODO: Fix strange ide_read bugs
void *ide_read(void *b, u32 block)
{
- int sector_per_block = BLOCK_SIZE / SECTOR_SIZE; // 2
- int sector = block * sector_per_block;
-
- print("");
- ide_wait(0);
- print("");
- outb(IDE_IO | IDE_SECTOR_COUNT, sector_per_block); // Number of sectors
- print("");
- outb(IDE_IO | IDE_LOW, LBA_LOW(sector));
- print("");
- outb(IDE_IO | IDE_MID, LBA_MID(sector));
- print("");
- outb(IDE_IO | IDE_HIGH, LBA_HIGH(sector));
- print("");
+ int sector_count = BLOCK_SIZE / SECTOR_SIZE; // 2
+ int sector = block * sector_count;
+
+ outb(IDE_IO + IDE_SECTOR_COUNT, sector_count); // Number of sectors
+
+ outb(IDE_IO + IDE_LOW, LBA_LOW(sector));
+ outb(IDE_IO + IDE_MID, LBA_MID(sector));
+ outb(IDE_IO + IDE_HIGH, LBA_HIGH(sector));
// Slave/Master << 4 and last 4 bits
- print("");
- outb(IDE_IO | IDE_HEAD, 0xE0 | (1 << 4) | LBA_LAST(sector));
- print("");
- outb(IDE_IO | IDE_CMD, IDE_CMD_READ);
- print("");
- ide_wait(0);
- print("");
-
- // Read-only
- print("");
+ outb(IDE_IO + IDE_SELECT, 0xE0 | (1 << 4) | LBA_LAST(sector));
+
+ outb(IDE_IO + IDE_CMD, IDE_CMD_READ);
+
+ ide_wait();
insl(IDE_IO, b, BLOCK_SIZE / 4);
- print("");
return b;
}
diff --git a/kernel/drivers/net.c b/kernel/drivers/net.c
index 284b749..cd1a331 100644
--- a/kernel/drivers/net.c
+++ b/kernel/drivers/net.c
@@ -10,6 +10,69 @@
#include <pci.h>
#include <print.h>
+u16 flip_short(u16 short_int)
+{
+ u32 first_byte = *((u8 *)(&short_int));
+ u32 second_byte = *((u8 *)(&short_int) + 1);
+ return (first_byte << 8) | (second_byte);
+}
+
+u32 flip_long(u32 long_int)
+{
+ u32 first_byte = *((u8 *)(&long_int));
+ u32 second_byte = *((u8 *)(&long_int) + 1);
+ u32 third_byte = *((u8 *)(&long_int) + 2);
+ u32 fourth_byte = *((u8 *)(&long_int) + 3);
+ return (first_byte << 24) | (second_byte << 16) | (third_byte << 8) | (fourth_byte);
+}
+
+u8 flip_byte(u8 byte, int num_bits)
+{
+ u8 t = byte << (8 - num_bits);
+ return t | (byte >> num_bits);
+}
+
+u8 htonb(u8 byte, int num_bits)
+{
+ return flip_byte(byte, num_bits);
+}
+
+u8 ntohb(u8 byte, int num_bits)
+{
+ return flip_byte(byte, 8 - num_bits);
+}
+
+u16 htons(u16 hostshort)
+{
+ return flip_short(hostshort);
+}
+
+u32 htonl(u32 hostlong)
+{
+ return flip_long(hostlong);
+}
+
+u16 ntohs(u16 netshort)
+{
+ return flip_short(netshort);
+}
+
+u32 ntohl(u32 netlong)
+{
+ return flip_long(netlong);
+}
+
+void ethernet_handle_packet(struct ethernet_packet *packet)
+{
+ /* void *data = packet + sizeof(*packet); */
+ if (ntohs(packet->type) == ETHERNET_TYPE_ARP)
+ print("ARP PACKET\n");
+ else if (ntohs(packet->type) == ETHERNET_TYPE_IP)
+ print("IP PACKET\n");
+ else
+ printf("UNKNOWN PACKET %d\n", ntohs(packet->type));
+}
+
static int rtl_irq = 0;
static u8 mac[6];
static u8 *last_packet = NULL;
@@ -28,13 +91,13 @@ void rtl8139_find(u32 device, u16 vendor_id, u16 device_id, void *extra)
void rtl8139_irq_handler()
{
print("RTL INT!\n");
- u16 status = inw(rtl_iobase + 0x3E);
+ u16 status = inw(rtl_iobase + RTL_PORT_ISR);
if (!status)
return;
- outw(rtl_iobase + 0x3E, status);
+ outw(rtl_iobase + RTL_PORT_ISR, status);
if (status & 0x01 || status & 0x02) {
- while ((inb(rtl_iobase + 0x37) & 0x01) == 0) {
+ while ((inb(rtl_iobase + RTL_PORT_CMD) & 0x01) == 0) {
print("RECEIVE\n");
int offset = cur_rx % 0x2000;
@@ -60,11 +123,11 @@ void rtl8139_irq_handler()
memcpy(last_packet, buf_8, rx_size);
}
- /* rtl_enqueue(last_packet); */
+ ethernet_handle_packet((struct ethernet_packet *)last_packet);
}
cur_rx = (cur_rx + rx_size + 4 + 3) & ~3;
- outw(rtl_iobase + 0x3A, cur_rx - 16);
+ outw(rtl_iobase + RTL_PORT_RXPTR, cur_rx - 16);
}
}
}
@@ -84,6 +147,7 @@ void rtl8139_init()
irq_install_handler(rtl_irq, rtl8139_irq_handler);
u32 rtl_bar0 = pci_read_field(rtl_device_pci, PCI_BAR0, 4);
+ /* u32 rtl_bar1 = pci_read_field(rtl_device_pci, PCI_BAR1, 4); */
rtl_iobase = 0;
if (rtl_bar0 & 0x00000001)
@@ -91,30 +155,48 @@ void rtl8139_init()
// Get mac address
for (int i = 0; i < 6; ++i)
- mac[i] = inb((u16)(rtl_iobase + 0x00 + i));
+ mac[i] = inb(rtl_iobase + RTL_PORT_MAC + i);
printf("Mac address: %x:%x:%x:%x:%x:%x\n", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
// Activate
- outb((u16)(rtl_iobase + 0x52), 0x0);
+ outb(rtl_iobase + RTL_PORT_CONFIG, 0x0);
// Reset
- outb((u16)(rtl_iobase + 0x37), 0x10);
- while ((inb((u16)(rtl_iobase + 0x37)) & 0x10) != 0)
+ outb((u16)(rtl_iobase + RTL_PORT_CMD), 0x10);
+ while ((inb(rtl_iobase + RTL_PORT_CMD) & 0x10) != 0)
;
// Set receive buffer
rtl_rx_buffer = (u8 *)malloc(0x3000);
- memset(rtl_rx_buffer, 0x00, 0x3000);
- outl((u16)(rtl_iobase + 0x30), (u32)rtl_rx_buffer);
-
- // Enable ISR
- outw((u16)(rtl_iobase + 0x3C), 0x0005);
-
- // Accept packets
- outl((u16)(rtl_iobase + 0x44), 0xf | (1 << 7));
-
- // Enable receive and transmitter
- outb((u16)(rtl_iobase + 0x37), 0x0C);
+ memset(rtl_rx_buffer, 0, 0x3000);
+ outl(rtl_iobase + RTL_PORT_RBSTART, (u32)rtl_rx_buffer);
+
+ // Enable IRQs
+ outw(rtl_iobase + RTL_PORT_IMR, 0x8000 | /* PCI error */
+ 0x4000 | /* PCS timeout */
+ 0x40 | /* Rx FIFO over */
+ 0x20 | /* Rx underrun */
+ 0x10 | /* Rx overflow */
+ 0x08 | /* Tx error */
+ 0x04 | /* Tx okay */
+ 0x02 | /* Rx error */
+ 0x01 /* Rx okay */
+ );
+
+ // Configure transmit
+ outl(rtl_iobase + RTL_PORT_TCR, 0);
+
+ // Configure receive
+ outl(rtl_iobase + RTL_PORT_RCR, (0) | /* 8K receive */
+ 0x08 | /* broadcast */
+ 0x01 /* all physical */
+ );
+
+ // Enable receive and transmit
+ outb(rtl_iobase + RTL_PORT_CMD, 0x08 | 0x04);
+
+ // Reset rx statistics
+ outl(rtl_iobase + RTL_PORT_RXMISS, 0);
}
void net_install()
diff --git a/kernel/inc/ide.h b/kernel/inc/ide.h
index e88a6d7..c145760 100644
--- a/kernel/inc/ide.h
+++ b/kernel/inc/ide.h
@@ -21,7 +21,7 @@
#define IDE_LOW 0x3
#define IDE_MID 0x4
#define IDE_HIGH 0x5
-#define IDE_HEAD 0x6
+#define IDE_SELECT 0x6
#define IDE_CMD 0x7
#define IDE_ALTERNATE 0x3F6
@@ -35,7 +35,6 @@
#define IDE_CMD_READ_MUL 0xC4
#define IDE_CMD_WRITE_MUL 0xC5
-int ide_wait(int check);
void *ide_read(void *b, u32 block);
#endif
diff --git a/kernel/inc/net.h b/kernel/inc/net.h
index 11b94e5..f3876e9 100644
--- a/kernel/inc/net.h
+++ b/kernel/inc/net.h
@@ -3,9 +3,36 @@
#ifndef NET_H
#define NET_H
+#include <def.h>
+
#define RTL8139_VENDOR_ID 0x10ec
#define RTL8139_DEVICE_ID 0x8139
+#define RTL_PORT_MAC 0x00
+#define RTL_PORT_MAR 0x08
+#define RTL_PORT_TXSTAT 0x10
+#define RTL_PORT_TXBUF 0x20
+#define RTL_PORT_RBSTART 0x30
+#define RTL_PORT_CMD 0x37
+#define RTL_PORT_RXPTR 0x38
+#define RTL_PORT_RXADDR 0x3A
+#define RTL_PORT_IMR 0x3C
+#define RTL_PORT_ISR 0x3E
+#define RTL_PORT_TCR 0x40
+#define RTL_PORT_RCR 0x44
+#define RTL_PORT_RXMISS 0x4C
+#define RTL_PORT_CONFIG 0x52
+
+#define ETHERNET_TYPE_IP 0x0800
+#define ETHERNET_TYPE_ARP 0x0806
+
+struct ethernet_packet {
+ u8 dst_mac_addr[6];
+ u8 src_mac_addr[6];
+ u16 type;
+ u8 data[];
+} __attribute__((packed));
+
void net_install();
#endif
diff --git a/libc/cpu.c b/libc/cpu.c
index a9e7241..1f0b7e4 100644
--- a/libc/cpu.c
+++ b/libc/cpu.c
@@ -28,10 +28,9 @@ u32 inl(u16 port)
void insl(u16 port, void *addr, int n)
{
- __asm__ volatile("cld; rep insl"
- : "=D"(addr), "=c"(n)
- : "d"(port), "0"(addr), "1"(n)
- : "memory", "cc");
+ __asm__ volatile("rep insl" ::"c"(n), // Count
+ "d"(port), // Port #
+ "D"(addr)); // Buffer
}
void outb(u16 port, u8 data)