123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 |
- /* 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_get_interrupt(pcnet_device_pci);
- irq_install_handler(pcnet_irq, pcnet_irq_handler, "pcnet");
- 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);
|