diff options
author | Marvin Borner | 2020-09-21 22:00:40 +0200 |
---|---|---|
committer | Marvin Borner | 2020-09-21 22:00:40 +0200 |
commit | 0e9abba93578786e774525dfdf02c59194d3e1e5 (patch) | |
tree | e7df565030ebfc4d334ca142e005351ffd293cd6 | |
parent | 02d97dcb1ea14c963219f22aed24b297e8366d05 (diff) |
Frustration
-rw-r--r-- | kernel/drivers/ide.c | 63 | ||||
-rw-r--r-- | kernel/drivers/net.c | 122 | ||||
-rw-r--r-- | kernel/inc/ide.h | 3 | ||||
-rw-r--r-- | kernel/inc/net.h | 27 | ||||
-rw-r--r-- | libc/cpu.c | 7 |
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 @@ -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) |