toaruos/modules/pcnet.c

376 lines
10 KiB
C

/* vim: tabstop=4 shiftwidth=4 noexpandtab
* This file is part of ToaruOS and is released under the terms
* of the NCSA / University of Illinois License - see LICENSE.md
* Copyright (C) 2016-2018 K. Lange
*/
#include <kernel/module.h>
#include <kernel/logging.h>
#include <kernel/printf.h>
#include <kernel/pci.h>
#include <kernel/mem.h>
#include <kernel/pipe.h>
#include <kernel/ipv4.h>
#include <kernel/mod/net.h>
#include <toaru/list.h>
static list_t * net_queue = NULL;
static spin_lock_t net_queue_lock = { 0 };
static list_t * rx_wait;
static uint32_t pcnet_device_pci = 0x00000000;
static uint32_t pcnet_io_base = 0;
static uint32_t pcnet_mem_base = 0;
static int pcnet_irq;
static uint8_t mac[6];
static uint32_t pcnet_buffer_phys;
static uint8_t *pcnet_buffer_virt;
static uint8_t * pcnet_rx_de_start;
static uint8_t * pcnet_tx_de_start;
static uint8_t * pcnet_rx_start;
static uint8_t * pcnet_tx_start;
static uint32_t pcnet_rx_de_phys;
static uint32_t pcnet_tx_de_phys;
static uint32_t pcnet_rx_phys;
static uint32_t pcnet_tx_phys;
static int pcnet_rx_buffer_id = 0;
static int pcnet_tx_buffer_id = 0;
#define PCNET_DE_SIZE 16
#define PCNET_BUFFER_SIZE 1548
#define PCNET_RX_COUNT 32
#define PCNET_TX_COUNT 8
static void find_pcnet(uint32_t device, uint16_t vendorid, uint16_t deviceid, void * extra) {
if ((vendorid == 0x1022) && (deviceid == 0x2000)) {
*((uint32_t *)extra) = device;
}
}
static void write_rap32(uint32_t value) {
outportl(pcnet_io_base + 0x14, value);
}
static void write_rap16(uint16_t value) {
outports(pcnet_io_base + 0x12, value);
}
static uint32_t read_csr32(uint32_t csr_no) {
write_rap32(csr_no);
return inportl(pcnet_io_base + 0x10);
}
static uint16_t read_csr16(uint16_t csr_no) {
write_rap32(csr_no);
return inports(pcnet_io_base + 0x10);
}
static void write_csr32(uint32_t csr_no, uint32_t value) {
write_rap32(csr_no);
outportl(pcnet_io_base + 0x10, value);
}
static void write_csr16(uint32_t csr_no, uint16_t value) {
write_rap16(csr_no);
outports(pcnet_io_base + 0x10, value);
}
static uint32_t read_bcr32(uint32_t bcr_no) {
write_rap32(bcr_no);
return inportl(pcnet_io_base + 0x1c);
}
static void write_bcr32(uint32_t bcr_no, uint32_t value) {
write_rap32(bcr_no);
outportl(pcnet_io_base + 0x1c, value);
}
static uint32_t virt_to_phys(uint8_t * virt) {
return ((uintptr_t)virt - (uintptr_t)pcnet_buffer_virt) + pcnet_buffer_phys;
}
static int driver_owns(uint8_t * de_table, int index) {
return (de_table[PCNET_DE_SIZE * index + 7] & 0x80) == 0;
}
static int next_tx_index(int current_tx_index) {
int out = current_tx_index + 1;
if (out == PCNET_TX_COUNT) {
return 0;
}
return out;
}
static int next_rx_index(int current_rx_index) {
int out = current_rx_index + 1;
if (out == PCNET_RX_COUNT) {
return 0;
}
return out;
}
static void init_descriptor(int index, int is_tx) {
uint8_t * de_table = is_tx ? pcnet_tx_de_start : pcnet_rx_de_start;
memset(&de_table[index * PCNET_DE_SIZE], 0, PCNET_DE_SIZE);
uint32_t buf_addr = is_tx ? pcnet_tx_phys : pcnet_rx_phys;
*(uint32_t *)&de_table[index * PCNET_DE_SIZE] = buf_addr + index * PCNET_BUFFER_SIZE;
uint16_t bcnt = (uint16_t)(-PCNET_BUFFER_SIZE);
bcnt &= 0x0FFF;
bcnt |= 0xF000;
*(uint16_t *)&de_table[index * PCNET_DE_SIZE + 4] = bcnt;
if (!is_tx) {
de_table[index * PCNET_DE_SIZE + 7] = 0x80;
}
}
static void enqueue_packet(void * buffer) {
spin_lock(net_queue_lock);
list_insert(net_queue, buffer);
spin_unlock(net_queue_lock);
}
static struct ethernet_packet * dequeue_packet(void) {
while (!net_queue->length) {
sleep_on(rx_wait);
}
spin_lock(net_queue_lock);
node_t * n = list_dequeue(net_queue);
void* value = n->value;
free(n);
spin_unlock(net_queue_lock);
return value;
}
static uint8_t* pcnet_get_mac() {
return mac;
}
static void pcnet_send_packet(uint8_t* payload, size_t payload_size) {
if (!driver_owns(pcnet_tx_de_start, pcnet_tx_buffer_id)) {
/* sleep? */
debug_print(ERROR, "No transmit descriptors available. Bailing.");
return;
}
if (payload_size > PCNET_BUFFER_SIZE) {
debug_print(ERROR, "Packet too big; max is %d, got %d", PCNET_BUFFER_SIZE, payload_size);
return;
}
memcpy((void *)(pcnet_tx_start + pcnet_tx_buffer_id * PCNET_BUFFER_SIZE), payload, payload_size);
pcnet_tx_de_start[pcnet_tx_buffer_id * PCNET_DE_SIZE + 7] |= 0x3;
uint16_t bcnt = (uint16_t)(-payload_size);
bcnt &= 0x0FFF;
bcnt |= 0xF000;
*(uint16_t *)&pcnet_tx_de_start[pcnet_tx_buffer_id * PCNET_DE_SIZE + 4] = bcnt;
pcnet_tx_de_start[pcnet_tx_buffer_id * PCNET_DE_SIZE + 7] |= 0x80;
write_csr32(0, read_csr32(0) | (1 << 3));
pcnet_tx_buffer_id = next_tx_index(pcnet_tx_buffer_id);
}
static int pcnet_irq_handler(struct regs *r) {
write_csr32(0, read_csr32(0) | 0x0400);
irq_ack(pcnet_irq);
while (driver_owns(pcnet_rx_de_start, pcnet_rx_buffer_id)) {
uint16_t plen = *(uint16_t *)&pcnet_rx_de_start[pcnet_rx_buffer_id * PCNET_DE_SIZE + 8];
void * pbuf = (void *)(pcnet_rx_start + pcnet_rx_buffer_id * PCNET_BUFFER_SIZE);
void * packet = malloc(plen);
memcpy(packet, pbuf, plen);
pcnet_rx_de_start[pcnet_rx_buffer_id * PCNET_DE_SIZE + 7] = 0x80;
enqueue_packet(packet);
pcnet_rx_buffer_id = next_rx_index(pcnet_rx_buffer_id);
}
wakeup_queue(rx_wait);
return 1;
}
static void pcnet_init(void * data, char * name) {
uint16_t command_reg = pci_read_field(pcnet_device_pci, PCI_COMMAND, 4) & 0xFFFF0000;
if (command_reg & (1 << 2)) {
debug_print(NOTICE, "Bus mastering already enabled.\n");
}
command_reg |= (1 << 2);
command_reg |= (1 << 0);
pci_write_field(pcnet_device_pci, PCI_COMMAND, 4, command_reg);
pcnet_io_base = pci_read_field(pcnet_device_pci, PCI_BAR0, 4) & 0xFFFFFFF0;
pcnet_mem_base = pci_read_field(pcnet_device_pci, PCI_BAR1, 4) & 0xFFFFFFF0;
pcnet_irq = pci_read_field(pcnet_device_pci, PCI_INTERRUPT_LINE, 1);
irq_install_handler(pcnet_irq, pcnet_irq_handler);
debug_print(NOTICE, "irq line: %d", pcnet_irq);
debug_print(NOTICE, "io base: 0x%x", pcnet_io_base);
/* Read MAC from EEPROM */
mac[0] = inportb(pcnet_io_base + 0);
mac[1] = inportb(pcnet_io_base + 1);
mac[2] = inportb(pcnet_io_base + 2);
mac[3] = inportb(pcnet_io_base + 3);
mac[4] = inportb(pcnet_io_base + 4);
mac[5] = inportb(pcnet_io_base + 5);
/* Force reset */
inportl(pcnet_io_base + 0x18);
inports(pcnet_io_base + 0x14);
unsigned long s, ss;
relative_time(0, 10, &s, &ss);
sleep_until((process_t *)current_process, s, ss);
switch_task(0);
debug_print(NOTICE, "pcnet return from sleep");
/* set 32-bit mode */
outportl(pcnet_io_base + 0x10, 0);
/* SWSTYLE to 2 */
uint32_t csr58 = read_csr32(58);
csr58 &= 0xFFF0;
csr58 |= 2;
write_csr32(58, csr58);
/* ASEL enable */
uint32_t bcr2 = read_bcr32(2);
bcr2 |= 0x2;
write_bcr32(2, bcr2);
debug_print(NOTICE, "device mac %2x:%2x:%2x:%2x:%2x:%2x", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
if (!pcnet_buffer_virt) {
debug_print(ERROR, "Failed.");
return;
}
debug_print(WARNING, "phys: 0x%x, virt: 0x%x", pcnet_buffer_phys, pcnet_buffer_virt);
pcnet_rx_de_start = pcnet_buffer_virt + 28;
pcnet_tx_de_start = pcnet_rx_de_start + PCNET_RX_COUNT * PCNET_DE_SIZE;
pcnet_rx_start = pcnet_tx_de_start + PCNET_TX_COUNT * PCNET_DE_SIZE;
pcnet_tx_start = pcnet_rx_start + PCNET_RX_COUNT * PCNET_BUFFER_SIZE;
pcnet_rx_de_phys = virt_to_phys(pcnet_rx_de_start);
pcnet_tx_de_phys = virt_to_phys(pcnet_tx_de_start);
pcnet_rx_phys = virt_to_phys(pcnet_rx_start);
pcnet_tx_phys = virt_to_phys(pcnet_tx_start);
/* set up descriptors */
for (int i = 0; i < PCNET_RX_COUNT; i++) {
init_descriptor(i, 0);
}
for (int i = 0; i < PCNET_TX_COUNT; i++) {
init_descriptor(i, 1);
}
/* Set up device configuration structure */
((uint16_t *)&pcnet_buffer_virt[0])[0] = 0x0000;
pcnet_buffer_virt[2] = 5 << 4; /* RLEN << 4 */
pcnet_buffer_virt[3] = 3 << 4; /* TLEN << 4 */
pcnet_buffer_virt[4] = mac[0];
pcnet_buffer_virt[5] = mac[1];
pcnet_buffer_virt[6] = mac[2];
pcnet_buffer_virt[7] = mac[3];
pcnet_buffer_virt[8] = mac[4];
pcnet_buffer_virt[9] = mac[5];
pcnet_buffer_virt[10] = 0; /* reserved */
pcnet_buffer_virt[11] = 0; /* reserved */
pcnet_buffer_virt[12] = 0;
pcnet_buffer_virt[13] = 0;
pcnet_buffer_virt[14] = 0;
pcnet_buffer_virt[15] = 0;
pcnet_buffer_virt[16] = 0;
pcnet_buffer_virt[17] = 0;
pcnet_buffer_virt[18] = 0;
pcnet_buffer_virt[19] = 0;
((uint32_t *)&pcnet_buffer_virt[20])[0] = pcnet_rx_de_phys;
((uint32_t *)&pcnet_buffer_virt[24])[0] = pcnet_tx_de_phys;
/* Configure network */
net_queue = list_create();
rx_wait = list_create();
write_csr32(1, 0xFFFF & pcnet_buffer_phys);
write_csr32(2, 0xFFFF & (pcnet_buffer_phys >> 16));
uint32_t a = read_csr32(1);
uint32_t b = read_csr32(2);
debug_print(ERROR, "csr1 = 0x%4x csr2= 0x%4x", a, b);
uint16_t csr3 = read_csr32(3);
if (csr3 & (1 << 10)) csr3 ^= (1 << 10);
if (csr3 & (1 << 2)) csr3 ^= (1 << 2);
csr3 |= (1 << 9);
csr3 |= (1 << 8);
write_csr32(3, csr3); /* Disable interrupt on init */
write_csr32(4, read_csr32(4) | (1 << 1) | (1 << 12) | (1 << 14)); /* pad */
write_csr32(0, read_csr32(0) | (1 << 0) | (1 << 6)); /* do it */
uint64_t start_time;
asm volatile (".byte 0x0f, 0x31" : "=A" (start_time));
uint32_t status;
while (((status = read_csr32(0)) & (1 << 8)) == 0) {
uint64_t now_time;
asm volatile (".byte 0x0f, 0x31" : "=A" (now_time));
if (now_time - start_time > 0x10000) {
debug_print(ERROR, "Could not initialize PCNet card, status is 0x%4x", status);
return;
}
}
/* Start card */
uint16_t csr0 = read_csr32(0);
if (csr0 & (1 << 0)) csr0 ^= (1 << 0);
if (csr0 & (1 << 2)) csr0 ^= (1 << 2);
csr0 |= (1 << 1);
write_csr32(0, csr0);
debug_print(NOTICE, "Card start.");
init_netif_funcs(pcnet_get_mac, dequeue_packet, pcnet_send_packet, "AMD PCnet FAST II/III");
}
static int init(void) {
pci_scan(&find_pcnet, -1, &pcnet_device_pci);
if (!pcnet_device_pci) {
debug_print(WARNING, "No PCNET device found.");
return 1;
}
/* Initialize ring buffers */
debug_print(WARNING, "Request a large continuous chunk of memory.");
/* This fits 32x1548 (rx) + 8x1548 (tx) + 32x16 (rx DE) + 8x16 (tx DE) */
pcnet_buffer_virt = (void*)kvmalloc_p(0x10000, &pcnet_buffer_phys);
create_kernel_tasklet(pcnet_init, "[pcnet]", NULL);
return 0;
}
static int fini(void) {
return 0;
}
MODULE_DEF(pcnet, init, fini);
MODULE_DEPENDS(net);