* Reverted last commit - not a good idea to provide untested patches. With the

patch applied, the card didn't work at all anymore.
* Minor 80-column/white space cleanup.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@26879 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2008-08-08 15:37:43 +00:00
parent ae9067b364
commit 8c5620e78f

View File

@ -1,7 +1,7 @@
/* Realtek RTL8169 Family Driver
* Copyright (C) 2004 Marcus Overhagen <marcus@overhagen.de>. All rights reserved.
*
* Permission to use, copy, modify and distribute this software and its
* Permission to use, copy, modify and distribute this software and its
* documentation for any purpose and without fee is hereby granted, provided
* that the above copyright notice appear in all copies, and that both the
* copyright notice and this permission notice appear in supporting documentation.
@ -16,6 +16,7 @@
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include <KernelExport.h>
#include <Errors.h>
#include <stdlib.h>
@ -39,11 +40,11 @@ read_settings(rtl8169_device *device)
void *handle;
const char *param;
int mtu, count;
handle = load_driver_settings("rtl8169");
if (!handle)
return;
param = get_driver_parameter(handle, "mtu", "-1", "-1");
mtu = atoi(param);
if (mtu >= 50 && mtu <= 1500)
@ -126,8 +127,8 @@ phy_config(rtl8169_device *device)
// see IEE 802.3-2002 (is also uses decimal numbers when refering
// to the registers, as do we). Added a little documentation
write_phy_reg(device, 31, 0x0001); // vendor specific (enter programming mode?)
write_phy_reg(device, 21, 0x1000); // vendor specific
write_phy_reg(device, 24, 0x65c7); // vendor specific
write_phy_reg(device, 21, 0x1000); // vendor specific
write_phy_reg(device, 24, 0x65c7); // vendor specific
write_phy_reg_bit(device, 4, 11, 0); // reset T (T=toggle) bit in reg 4 (ability)
val = read_phy_reg(device, 4) & 0x0fff; // get only the message code fields
write_phy_reg(device, 4, val); // and write them back. this clears the page and makes it unformatted (see 37.2.4.3.1)
@ -171,7 +172,7 @@ phy_config(rtl8169_device *device)
write_phy_reg_bit(device, 4, 11, 1); // set toggle bit high
write_phy_reg_bit(device, 4, 11, 0); // set toggle bit low => this is a toggle
write_phy_reg(device, 31, 0x0000); // vendor specific (leave programming mode?)
}
}
write_phy_reg(device, 4, 0x01e1); // 10/100 capability
write_phy_reg(device, 9, 0x0200); // 1000 capability
@ -261,7 +262,7 @@ print_debug_info(void *cookie)
device->intTxCurrentCount,
device->intTimerTotalCount,
device->intTimerCurrentCount);
device->intTotalCountOld = device->intTotalCount;
device->intCurrentCount = 0;
device->intRxCurrentCount = 0;
@ -280,21 +281,27 @@ init_buf_desc(rtl8169_device *device)
void *rx_buf_virt, *rx_buf_phy;
int i;
device->txBufArea = alloc_mem(&tx_buf_virt, &tx_buf_phy, device->txBufferCount * FRAME_SIZE, 0, "rtl8169 tx buf");
device->rxBufArea = alloc_mem(&rx_buf_virt, &rx_buf_phy, device->rxBufferCount * FRAME_SIZE, 0, "rtl8169 rx buf");
device->txDescArea = alloc_mem(&tx_buf_desc_virt, &tx_buf_desc_phy, device->txBufferCount * sizeof(buf_desc), 0, "rtl8169 tx desc");
device->rxDescArea = alloc_mem(&rx_buf_desc_virt, &rx_buf_desc_phy, device->rxBufferCount * sizeof(buf_desc), 0, "rtl8169 rx desc");
if (device->txBufArea < B_OK || device->rxBufArea < B_OK || device->txDescArea < B_OK || device->rxDescArea < B_OK)
device->txBufArea = alloc_mem(&tx_buf_virt, &tx_buf_phy,
device->txBufferCount * FRAME_SIZE, 0, "rtl8169 tx buf");
device->rxBufArea = alloc_mem(&rx_buf_virt, &rx_buf_phy,
device->rxBufferCount * FRAME_SIZE, 0, "rtl8169 rx buf");
device->txDescArea = alloc_mem(&tx_buf_desc_virt, &tx_buf_desc_phy,
device->txBufferCount * sizeof(buf_desc), 0, "rtl8169 tx desc");
device->rxDescArea = alloc_mem(&rx_buf_desc_virt, &rx_buf_desc_phy,
device->rxBufferCount * sizeof(buf_desc), 0, "rtl8169 rx desc");
if (device->txBufArea < B_OK || device->rxBufArea < B_OK
|| device->txDescArea < B_OK || device->rxDescArea < B_OK)
return B_NO_MEMORY;
device->txDesc = (buf_desc *) tx_buf_desc_virt;
device->rxDesc = (buf_desc *) rx_buf_desc_virt;
device->txDesc = (buf_desc *)tx_buf_desc_virt;
device->rxDesc = (buf_desc *)rx_buf_desc_virt;
// setup transmit descriptors
for (i = 0; i < device->txBufferCount; i++) {
device->txBuf[i] = (char *)tx_buf_virt + (i * FRAME_SIZE);
device->txDesc[i].stat_len = TX_DESC_FS | TX_DESC_LS;
device->txDesc[i].buf_low = (uint32)((char *)tx_buf_phy + (i * FRAME_SIZE));
device->txDesc[i].buf_low
= (uint32)((char *)tx_buf_phy + (i * FRAME_SIZE));
device->txDesc[i].buf_high = 0;
}
device->txDesc[i - 1].stat_len |= TX_DESC_EOR;
@ -303,11 +310,12 @@ init_buf_desc(rtl8169_device *device)
for (i = 0; i < device->rxBufferCount; i++) {
device->rxBuf[i] = (char *)rx_buf_virt + (i * FRAME_SIZE);
device->rxDesc[i].stat_len = RX_DESC_OWN | FRAME_SIZE;
device->rxDesc[i].buf_low = (uint32)((char *)rx_buf_phy + (i * FRAME_SIZE));
device->rxDesc[i].buf_low
= (uint32)((char *)rx_buf_phy + (i * FRAME_SIZE));
device->rxDesc[i].buf_high = 0;
}
device->rxDesc[i - 1].stat_len |= RX_DESC_EOR;
write32(REG_RDSAR_LOW, (uint32)rx_buf_desc_phy);
write32(REG_RDSAR_HIGH, 0);
write32(REG_TNPDS_LOW, (uint32)tx_buf_desc_phy);
@ -350,7 +358,7 @@ rtl8169_rx_int(rtl8169_device *device)
{
int32 limit;
int32 count;
acquire_spinlock(&device->rxSpinlock);
for (count = 0, limit = device->rxFree; limit > 0; limit--) {
@ -361,7 +369,7 @@ rtl8169_rx_int(rtl8169_device *device)
}
// dprintf("rx int, rxFree %d, count %d\n", device->rxFree, count);
device->rxFree -= count;
release_spinlock(&device->rxSpinlock);
@ -381,13 +389,13 @@ rtl8169_int(void *data)
stat = read16(REG_INT_STAT);
if (stat == 0 || stat == 0xffff)
return B_UNHANDLED_INTERRUPT;
write16(REG_INT_STAT, stat);
ret = B_HANDLED_INTERRUPT;
PROFILING_ONLY(device->intTotalCount++);
PROFILING_ONLY(device->intCurrentCount++);
if (stat & INT_FOVW) {
TRACE("INT_FOVW\n");
}
@ -429,36 +437,36 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
int dev_id;
int mask;
int i;
TRACE("rtl8169_open()\n");
for (dev_id = 0; (deviceName = gDevNameList[dev_id]) != NULL; dev_id++) {
if (!strcmp(name, deviceName))
break;
}
}
if (deviceName == NULL) {
ERROR("invalid device name\n");
return B_ERROR;
}
// allow only one concurrent access
mask = 1 << dev_id;
if (atomic_or(&gOpenMask, mask) & mask)
return B_BUSY;
*cookie = device = (rtl8169_device *)malloc(sizeof(rtl8169_device));
if (!device) {
atomic_and(&gOpenMask, ~(1 << dev_id));
return B_NO_MEMORY;
}
memset(device, 0, sizeof(*device));
device->devId = dev_id;
device->pciInfo = gDevList[dev_id];
device->nonblocking = (flags & O_NONBLOCK) ? true : false;
device->closed = false;
// setup defaults
device->maxframesize = 1514; // not FRAME_SIZE
device->txBufferCount = DEFAULT_TX_BUF_COUNT;
@ -473,7 +481,7 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
device->rxFree = device->rxBufferCount;
device->rxReadySem = create_sem(0, "rtl8169 rx ready");
set_sem_owner(device->rxReadySem, B_SYSTEM_TEAM);
device->txBuf = (void **)malloc(sizeof(void *) * device->txBufferCount);
B_INITIALIZE_SPINLOCK(&device->txSpinlock);
device->txNextIndex = 0;
@ -481,18 +489,23 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
device->txUsed = 0;
device->txFreeSem = create_sem(device->txBufferCount, "rtl8169 tx free");
set_sem_owner(device->txFreeSem, B_SYSTEM_TEAM);
// enable busmaster and memory mapped access, disable io port access
val = gPci->read_pci_config(device->pciInfo->bus, device->pciInfo->device, device->pciInfo->function, PCI_command, 2);
val = gPci->read_pci_config(device->pciInfo->bus, device->pciInfo->device,
device->pciInfo->function, PCI_command, 2);
val = PCI_PCICMD_BME | PCI_PCICMD_MSE | (val & ~PCI_PCICMD_IOS);
gPci->write_pci_config(device->pciInfo->bus, device->pciInfo->device, device->pciInfo->function, PCI_command, 2, val);
gPci->write_pci_config(device->pciInfo->bus, device->pciInfo->device,
device->pciInfo->function, PCI_command, 2, val);
// adjust PCI latency timer
TRACE("changing PCI latency to 0x40\n");
gPci->write_pci_config(device->pciInfo->bus, device->pciInfo->device, device->pciInfo->function, PCI_latency, 1, 0x40);
gPci->write_pci_config(device->pciInfo->bus, device->pciInfo->device,
device->pciInfo->function, PCI_latency, 1, 0x40);
// get IRQ
device->irq = gPci->read_pci_config(device->pciInfo->bus, device->pciInfo->device, device->pciInfo->function, PCI_interrupt_line, 1);
device->irq = gPci->read_pci_config(device->pciInfo->bus,
device->pciInfo->device, device->pciInfo->function, PCI_interrupt_line,
1);
if (device->irq == 0 || device->irq == 0xff) {
ERROR("no IRQ assigned\n");
goto err;
@ -501,15 +514,17 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
TRACE("IRQ %d\n", device->irq);
// map registers into memory
val = gPci->read_pci_config(device->pciInfo->bus, device->pciInfo->device, device->pciInfo->function, 0x18, 4);
val = gPci->read_pci_config(device->pciInfo->bus, device->pciInfo->device,
device->pciInfo->function, 0x14, 4);
val &= PCI_address_memory_32_mask;
TRACE("hardware register address %p\n", (void *) val);
device->regArea = map_mem(&device->regAddr, (void *)val, 256, 0, "rtl8169 register");
device->regArea = map_mem(&device->regAddr, (void *)val, 256, 0,
"rtl8169 register");
if (device->regArea < B_OK) {
ERROR("can't map hardware registers\n");
goto err;
}
TRACE("mapped registers to %p\n", device->regAddr);
// disable receiver & transmitter XXX might be removed
@ -527,7 +542,8 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
TRACE("reset done\n");
// get MAC hardware version
device->mac_version = ((read32(REG_TX_CONFIG) & 0x7c000000) >> 25) | ((read32(REG_TX_CONFIG) & 0x00800000) >> 23);
device->mac_version = ((read32(REG_TX_CONFIG) & 0x7c000000) >> 25)
| ((read32(REG_TX_CONFIG) & 0x00800000) >> 23);
TRACE("8169 Mac Version %d\n", device->mac_version);
if (device->mac_version > 0) { // this is a RTL8169s single chip
// get PHY hardware version
@ -538,35 +554,38 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
device->phy_version = 0;
TRACE("8169 Phy Version unknown\n");
}
if (device->mac_version == 1) {
// as it's done by the BSD driver...
TRACE("Setting MAC Reg C+CR 0x82h = 0x01h\n");
write8(0x82, 0x01); // don't know what this does
TRACE("Setting PHY Reg 0x0bh = 0x00h\n");
write_phy_reg(device, 0x0b, 0x0000); // 0xb is a reserved (vendor specific register), don't know what this does
write_phy_reg(device, 0x0b, 0x0000);
// 0xb is a reserved (vendor specific register), don't know what
// this does
}
// configure PHY
phy_config(device);
dump_phy_stat(device);
print_link_status(device);
// initialize MAC address
// initialize MAC address
for (i = 0; i < 6; i++)
device->macaddr[i] = read8(i);
TRACE("MAC %02x:%02x:%02x:%02x:%02x:%02x\n",
device->macaddr[0], device->macaddr[1], device->macaddr[2],
device->macaddr[3], device->macaddr[4], device->macaddr[5]);
// setup interrupt handler
if (install_io_interrupt_handler(device->irq, rtl8169_int, device, 0) < B_OK) {
if (install_io_interrupt_handler(device->irq, rtl8169_int, device, 0)
< B_OK) {
ERROR("can't install interrupt handler\n");
goto err;
}
#ifdef PROFILING
device->intTotalCount = 0;
device->intTotalCountOld = 0;
@ -577,16 +596,19 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
device->intRxCurrentCount = 0;
device->intTxCurrentCount = 0;
device->intTimerCurrentCount = 0;
device->timer = create_timer(print_debug_info, device, 1000000, B_PERIODIC_TIMER);
device->timer = create_timer(print_debug_info, device, 1000000,
B_PERIODIC_TIMER);
#endif // PROFILING
write16(0xe0, read16(0xe0)); // write CR+ command
write16(0xe0, read16(0xe0) | 0x0003); // don't know what this does, BSD says "enable C+ Tx/Rx"
write16(0xe0, read16(0xe0) | 0x0003);
// don't know what this does, BSD says "enable C+ Tx/Rx"
if (device->mac_version == 1) {
TRACE("Setting Reg C+CR bit 3 and bit 14 to 1\n");
// bit 3 is PCI multiple read/write enable (max Tx/Rx DMA burst size setting is no longer valid then)
// bit 3 is PCI multiple read/write enable (max Tx/Rx DMA burst size
// setting is no longer valid then)
// bit 14 ??? (need more docs)
write16(0xe0, read16(0xe0) | 0x4008);
}
@ -599,36 +621,39 @@ rtl8169_open(const char *name, uint32 flags, void** cookie)
// enable receiver & transmitter
write8(REG_CR, read8(REG_CR) | CR_RE | CR_TE);
write8(REG_9346CR, 0xc0); // enable config access
write8(REG_CONFIG1, read8(REG_CONFIG1) & ~1); // disable power management
write8(REG_9346CR, 0x00); // disable config access
write8(0xec, 0x3f); // disable early transmit treshold
write16(0xda, FRAME_SIZE); // receive packet maximum size
write16(0x5c, read16(0x5c) & 0xf000); // disable early receive interrupts
write32(0x4c, 0); // RxMissed ???
// setup receive config, can only be done when receiver is enabled!
// 1024 byte FIFO treshold, 1024 DMA burst
write32(REG_RX_CONFIG, (read32(REG_RX_CONFIG) & RX_CONFIG_MASK)
| (0x6 << RC_CONFIG_RXFTH_Shift) | (0x6 << RC_CONFIG_MAXDMA_Shift)
| RX_CONFIG_AcceptBroad | RX_CONFIG_AcceptMulti | RX_CONFIG_AcceptMyPhys);
| RX_CONFIG_AcceptBroad | RX_CONFIG_AcceptMulti
| RX_CONFIG_AcceptMyPhys);
write32(0x8, 0); // multicast filter
write32(0xc, 0); // multicast filter
// setup transmit config, can only be done when transmitter is enabled!
// setup transmit config, can only be done when transmitter is enabled!
// append CRC, 1024 DMA burst
write32(REG_TX_CONFIG, (read32(REG_TX_CONFIG) & ~(0x10000 | (1 << 8))) | (0x6 << 8));
write32(REG_TX_CONFIG, (read32(REG_TX_CONFIG) & ~(0x10000 | (1 << 8)))
| (0x6 << 8));
// clear pending interrupt status
write16(REG_INT_STAT, 0xffff);
// enable used interrupts
write16(REG_INT_MASK, INT_FOVW | INT_PUN | INT_TER | INT_TOK | INT_RER | INT_ROK);
write16(REG_INT_MASK, INT_FOVW | INT_PUN | INT_TER | INT_TOK | INT_RER
| INT_ROK);
return B_OK;
@ -653,7 +678,7 @@ rtl8169_close(void* cookie)
{
rtl8169_device *device = (rtl8169_device *)cookie;
TRACE("rtl8169_close()\n");
device->closed = true;
release_sem(device->rxReadySem);
release_sem(device->txFreeSem);
@ -673,9 +698,9 @@ rtl8169_free(void* cookie)
// disable interrupts
write16(REG_INT_MASK, 0);
PROFILING_ONLY(delete_timer(device->timer));
// well...
remove_io_interrupt_handler (device->irq, rtl8169_int, device);
@ -695,27 +720,30 @@ rtl8169_free(void* cookie)
status_t
rtl8169_read(void* cookie, off_t position, void *buf, size_t* num_bytes)
rtl8169_read(void* cookie, off_t position, void *buf, size_t* numBytes)
{
rtl8169_device *device = (rtl8169_device *)cookie;
cpu_status cpu;
status_t stat;
int len;
TRACE("rtl8169_read() enter\n");
if (device->closed) {
TRACE("rtl8169_read() interrupted 1\n");
return B_INTERRUPTED;
}
retry:
stat = acquire_sem_etc(device->rxReadySem, 1, B_CAN_INTERRUPT | (device->nonblocking ? B_TIMEOUT : 0), 0);
stat = acquire_sem_etc(device->rxReadySem, 1,
B_CAN_INTERRUPT | (device->nonblocking ? B_TIMEOUT : 0), 0);
if (device->closed) {
// TRACE("rtl8169_read() interrupted 2\n"); // net_server will crash if we print this (race condition in net_server?)
// TRACE("rtl8169_read() interrupted 2\n");
// net_server will crash if we print this
// (race condition in net_server?)
return B_INTERRUPTED;
}
if (stat == B_WOULD_BLOCK) {
TRACE("rtl8169_read() would block (OK 0 bytes)\n");
*num_bytes = 0;
*numBytes = 0;
return B_OK;
}
if (stat != B_OK) {
@ -732,39 +760,41 @@ retry:
len -= 4; // remove CRC that Realtek always appends
if (len < 0)
len = 0;
if (len > (int)*num_bytes)
len = *num_bytes;
if (len > (int)*numBytes)
len = *numBytes;
memcpy(buf, device->rxBuf[device->rxNextIndex], len);
*num_bytes = len;
*numBytes = len;
cpu = disable_interrupts();
acquire_spinlock(&device->rxSpinlock);
device->rxDesc[device->rxNextIndex].stat_len = RX_DESC_OWN | FRAME_SIZE | (device->rxDesc[device->rxNextIndex].stat_len & RX_DESC_EOR);
device->rxDesc[device->rxNextIndex].stat_len = RX_DESC_OWN | FRAME_SIZE
| (device->rxDesc[device->rxNextIndex].stat_len & RX_DESC_EOR);
device->rxFree++;
release_spinlock(&device->rxSpinlock);
restore_interrupts(cpu);
device->rxNextIndex = (device->rxNextIndex + 1) % device->rxBufferCount;
TRACE("rtl8169_read() leave\n");
return B_OK;
}
status_t
rtl8169_write(void* cookie, off_t position, const void* buffer, size_t* num_bytes)
rtl8169_write(void* cookie, off_t position, const void* buffer,
size_t* numBytes)
{
rtl8169_device *device = (rtl8169_device *)cookie;
cpu_status cpu;
status_t stat;
int len;
TRACE("rtl8169_write() enter\n");
len = *num_bytes;
len = *numBytes;
if (len > FRAME_SIZE) {
TRACE("rtl8169_write() buffer too large\n");
return B_ERROR;
@ -775,14 +805,15 @@ rtl8169_write(void* cookie, off_t position, const void* buffer, size_t* num_byte
return B_INTERRUPTED;
}
retry:
stat = acquire_sem_etc(device->txFreeSem, 1, B_CAN_INTERRUPT | B_TIMEOUT, device->nonblocking ? 0 : TX_TIMEOUT);
stat = acquire_sem_etc(device->txFreeSem, 1, B_CAN_INTERRUPT | B_TIMEOUT,
device->nonblocking ? 0 : TX_TIMEOUT);
if (device->closed) {
TRACE("rtl8169_write() interrupted 2\n");
return B_INTERRUPTED;
}
if (stat == B_WOULD_BLOCK) {
TRACE("rtl8169_write() would block (OK 0 bytes)\n");
*num_bytes = 0;
*numBytes = 0;
return B_OK;
}
if (stat == B_TIMED_OUT) {
@ -793,7 +824,7 @@ retry:
TRACE("rtl8169_write() error\n");
return B_ERROR;
}
if (device->txDesc[device->txNextIndex].stat_len & TX_DESC_OWN) {
ERROR("rtl8169_write() buffer still in use\n");
goto retry;
@ -805,13 +836,15 @@ retry:
acquire_spinlock(&device->txSpinlock);
device->txUsed++;
device->txDesc[device->txNextIndex].stat_len = (device->txDesc[device->txNextIndex].stat_len & RX_DESC_EOR) | TX_DESC_OWN | TX_DESC_FS | TX_DESC_LS | len;
device->txDesc[device->txNextIndex].stat_len
= (device->txDesc[device->txNextIndex].stat_len & RX_DESC_EOR)
| TX_DESC_OWN | TX_DESC_FS | TX_DESC_LS | len;
release_spinlock(&device->txSpinlock);
restore_interrupts(cpu);
device->txNextIndex = (device->txNextIndex + 1) % device->txBufferCount;
write8(REG_TPPOLL, read8(REG_TPPOLL) | TPPOLL_NPQ); // set queue polling bit
TRACE("rtl8169_write() leave\n");
@ -828,12 +861,12 @@ rtl8169_control(void *cookie, uint32 op, void *arg, size_t len)
case ETHER_INIT:
TRACE("rtl8169_control() ETHER_INIT\n");
return B_OK;
case ETHER_GETADDR:
TRACE("rtl8169_control() ETHER_GETADDR\n");
memcpy(arg, &device->macaddr, sizeof(device->macaddr));
return B_OK;
case ETHER_NONBLOCK:
if (*(int32 *)arg) {
TRACE("non blocking mode on\n");
@ -852,20 +885,22 @@ rtl8169_control(void *cookie, uint32 op, void *arg, size_t len)
case ETHER_ADDMULTI:
TRACE("rtl8169_control() ETHER_ADDMULTI\n");
break;
case ETHER_REMMULTI:
TRACE("rtl8169_control() ETHER_REMMULTI\n");
return B_OK;
case ETHER_SETPROMISC:
if (*(int32 *)arg) {
TRACE("promiscuous mode on\n");
write32(REG_RX_CONFIG, read32(REG_RX_CONFIG) | RX_CONFIG_AcceptAllPhys);
write32(REG_RX_CONFIG, read32(REG_RX_CONFIG)
| RX_CONFIG_AcceptAllPhys);
write32(0x8, 0xffffffff); // multicast filter
write32(0xc, 0xffffffff); // multicast filter
} else {
TRACE("promiscuous mode off\n");
write32(REG_RX_CONFIG, read32(REG_RX_CONFIG) & ~RX_CONFIG_AcceptAllPhys);
write32(REG_RX_CONFIG, read32(REG_RX_CONFIG)
& ~RX_CONFIG_AcceptAllPhys);
write32(0x8, 0); // multicast filter
write32(0xc, 0); // multicast filter
}
@ -875,12 +910,12 @@ rtl8169_control(void *cookie, uint32 op, void *arg, size_t len)
TRACE("rtl8169_control() ETHER_GETFRAMESIZE, framesize = %d (MTU = %d)\n", device->maxframesize, device->maxframesize - 14);
*(uint32*)arg = device->maxframesize;
return B_OK;
default:
TRACE("rtl8169_control() Invalid command\n");
break;
}
return B_ERROR;
}