aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMarvin Borner2020-09-24 19:15:58 +0200
committerMarvin Borner2020-09-24 19:15:58 +0200
commit6dfddc3d7de0ec10eab5ac1a4c894e1ab48b116e (patch)
treea221c92fd22c8de2aa536f18fb55c297ad48043f
parent0e31b099484eb356184841f245d7d0a47fef0b4d (diff)
IPv6 networking...
-rw-r--r--kernel/drivers/net.c119
-rw-r--r--kernel/inc/net.h8
-rwxr-xr-xrun2
3 files changed, 57 insertions, 72 deletions
diff --git a/kernel/drivers/net.c b/kernel/drivers/net.c
index cd1a331..86c2fea 100644
--- a/kernel/drivers/net.c
+++ b/kernel/drivers/net.c
@@ -10,6 +10,10 @@
#include <pci.h>
#include <print.h>
+static u8 mac[6];
+static u8 *rx_buffer;
+u32 current_packet_ptr;
+
u16 flip_short(u16 short_int)
{
u32 first_byte = *((u8 *)(&short_int));
@@ -62,24 +66,45 @@ u32 ntohl(u32 netlong)
return flip_long(netlong);
}
-void ethernet_handle_packet(struct ethernet_packet *packet)
+void ethernet_handle_packet(struct ethernet_packet *packet, int len)
{
/* void *data = packet + sizeof(*packet); */
+ printf("", len);
if (ntohs(packet->type) == ETHERNET_TYPE_ARP)
print("ARP PACKET\n");
- else if (ntohs(packet->type) == ETHERNET_TYPE_IP)
- print("IP PACKET\n");
+ else if (ntohs(packet->type) == ETHERNET_TYPE_IP4)
+ print("IP4 PACKET\n");
+ else if (ntohs(packet->type) == ETHERNET_TYPE_IP6)
+ print("IP6 PACKET\n");
else
- printf("UNKNOWN PACKET %d\n", ntohs(packet->type));
+ printf("UNKNOWN PACKET %x\n", ntohs(packet->type));
}
+/**
+ * RTL8139 specific things
+ */
+
static int rtl_irq = 0;
-static u8 mac[6];
-static u8 *last_packet = NULL;
-static u8 *rtl_rx_buffer;
static u32 rtl_device_pci = 0;
static u32 rtl_iobase = 0;
-static u32 cur_rx = 0;
+
+void rtl_receive_packet()
+{
+ u16 *t = (u16 *)(rx_buffer + current_packet_ptr);
+ u16 length = *(t + 1);
+ t += 2;
+
+ void *packet = malloc(length);
+ memcpy(packet, t, length);
+ ethernet_handle_packet(packet, length);
+
+ current_packet_ptr = (current_packet_ptr + length + 4 + 3) & (~3);
+
+ if (current_packet_ptr > RX_BUF_SIZE)
+ current_packet_ptr -= RX_BUF_SIZE;
+
+ outw(rtl_iobase + RTL_PORT_RXPTR, current_packet_ptr - 0x10);
+}
void rtl8139_find(u32 device, u16 vendor_id, u16 device_id, void *extra)
{
@@ -94,42 +119,14 @@ void rtl8139_irq_handler()
u16 status = inw(rtl_iobase + RTL_PORT_ISR);
if (!status)
return;
- outw(rtl_iobase + RTL_PORT_ISR, status);
- if (status & 0x01 || status & 0x02) {
- while ((inb(rtl_iobase + RTL_PORT_CMD) & 0x01) == 0) {
- print("RECEIVE\n");
-
- int offset = cur_rx % 0x2000;
-
- u32 *buf_start = (u32 *)((u32)rtl_rx_buffer + offset);
- u32 rx_status = buf_start[0];
- u32 rx_size = rx_status >> 16;
-
- if (rx_status & (0x0020 | 0x0010 | 0x0004 | 0x0002)) {
- print("RX Error\n");
- } else {
- u8 *buf_8 = (u8 *)&(buf_start[1]);
-
- last_packet = malloc(rx_size);
-
- u32 packet_end = (u32)buf_8 + rx_size;
- if (packet_end > (u32)rtl_rx_buffer + 0x2000) {
- u32 s = ((u32)rtl_rx_buffer + 0x2000) - (u32)buf_8;
- memcpy(last_packet, buf_8, s);
- memcpy((void *)((u32)last_packet + s), rtl_rx_buffer,
- rx_size - s);
- } else {
- memcpy(last_packet, buf_8, rx_size);
- }
-
- ethernet_handle_packet((struct ethernet_packet *)last_packet);
- }
-
- cur_rx = (cur_rx + rx_size + 4 + 3) & ~3;
- outw(rtl_iobase + RTL_PORT_RXPTR, cur_rx - 16);
- }
+ if (status & RTL_TOK) {
+ printf("Sent packet\n");
+ } else if (status & RTL_ROK) {
+ rtl_receive_packet();
}
+
+ outw(rtl_iobase + RTL_PORT_ISR, 0x5);
}
void rtl8139_init()
@@ -151,7 +148,7 @@ void rtl8139_init()
rtl_iobase = 0;
if (rtl_bar0 & 0x00000001)
- rtl_iobase = rtl_bar0 & 0xFFFFFFFC;
+ rtl_iobase = rtl_bar0 & (~0x3);
// Get mac address
for (int i = 0; i < 6; ++i)
@@ -167,36 +164,18 @@ void rtl8139_init()
;
// Set receive buffer
- rtl_rx_buffer = (u8 *)malloc(0x3000);
- 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 */
- );
+ rx_buffer = (u8 *)malloc(RX_BUF_SIZE);
+ memset(rx_buffer, 0, RX_BUF_SIZE);
+ outl(rtl_iobase + RTL_PORT_RBSTART, (u32)rx_buffer);
+
+ // Set TOK and ROK
+ outw(rtl_iobase + RTL_PORT_IMR, 0x0005);
+
+ // AB+AM+APM+AAP except WRAP
+ outl(rtl_iobase + RTL_PORT_RCR, 0xf | (1 << 7));
// 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/net.h b/kernel/inc/net.h
index f3876e9..b5193ed 100644
--- a/kernel/inc/net.h
+++ b/kernel/inc/net.h
@@ -5,9 +5,14 @@
#include <def.h>
+#define RX_BUF_SIZE 0x3000
+
#define RTL8139_VENDOR_ID 0x10ec
#define RTL8139_DEVICE_ID 0x8139
+#define RTL_ROK (1 << 0)
+#define RTL_TOK (1 << 2)
+
#define RTL_PORT_MAC 0x00
#define RTL_PORT_MAR 0x08
#define RTL_PORT_TXSTAT 0x10
@@ -23,7 +28,8 @@
#define RTL_PORT_RXMISS 0x4C
#define RTL_PORT_CONFIG 0x52
-#define ETHERNET_TYPE_IP 0x0800
+#define ETHERNET_TYPE_IP4 0x0800
+#define ETHERNET_TYPE_IP6 0x86dd
#define ETHERNET_TYPE_ARP 0x0806
struct ethernet_packet {
diff --git a/run b/run
index 352d939..8e8f86d 100755
--- a/run
+++ b/run
@@ -12,7 +12,7 @@ no_ask="${2}"
qemu_with_flags() {
if [ -z "$network" ]; then network="rtl8139"; fi
if [ "$network" != "false" ] && [ "$mode" = "net" ]; then
- qemu-system-i386 -cpu max -no-reboot -vga std -rtc base=localtime -m 256M -smp 4 -net nic,model=${network},macaddr=42:42:42:42:42:42 -net bridge,br=br0 "$@"
+ qemu-system-i386 -cpu max -no-reboot -vga std -rtc base=localtime -m 256M -smp 4 -netdev tap,helper=/usr/lib/qemu/qemu-bridge-helper,id=melvix_net -device $network,netdev=melvix_net,id=melvix_nic "$@"
else
qemu-system-i386 -cpu max -no-reboot -vga std -rtc base=localtime -m 256M -smp 4 "$@"
fi