/* * Copyright (C) 2016 TriaGnoSys GmbH * 2020 Otto-von-Guericke-Universität Magdeburg * * This file is subject to the terms and conditions of the GNU Lesser General * Public License v2.1. See the file LICENSE in the top level directory for more * details. */ /** * @ingroup cpu_stm32 * @{ * * @file * @brief Low-level ETH driver implementation * * @author Víctor Ariño * @author Marian Buschsieweke * * @} */ #include #include #include #include "bitarithm.h" #include "iolist.h" #include "mii.h" #include "mutex.h" #include "net/ethernet.h" #include "net/eui_provider.h" #include "net/netdev/eth.h" #include "periph/gpio.h" #include "timex.h" #define ENABLE_DEBUG 0 #define ENABLE_DEBUG_VERBOSE 0 #include "debug.h" #if IS_USED(MODULE_STM32_ETH_LINK_UP) #include "xtimer.h" #define STM32_ETH_LINK_UP_TIMEOUT_US (1UL * US_PER_SEC) static xtimer_t _link_status_timer; #endif /* IS_USED(MODULE_STM32_ETH_LINK_UP) */ /* Set the value of the divider with the clock configured */ #if !defined(CLOCK_CORECLOCK) || CLOCK_CORECLOCK < (20000000U) #error This peripheral requires a CORECLOCK of at least 20MHz #elif CLOCK_CORECLOCK < (35000000U) #define CLOCK_RANGE ETH_MACMIIAR_CR_Div16 #elif CLOCK_CORECLOCK < (60000000U) #define CLOCK_RANGE ETH_MACMIIAR_CR_Div26 #elif CLOCK_CORECLOCK < (100000000U) #define CLOCK_RANGE ETH_MACMIIAR_CR_Div42 #elif CLOCK_CORECLOCK < (150000000U) #define CLOCK_RANGE ETH_MACMIIAR_CR_Div62 #else /* CLOCK_CORECLOCK < (20000000U) */ #define CLOCK_RANGE ETH_MACMIIAR_CR_Div102 #endif /* CLOCK_CORECLOCK < (20000000U) */ /* Default DMA buffer setup */ #ifndef ETH_RX_DESCRIPTOR_COUNT #define ETH_RX_DESCRIPTOR_COUNT (6U) #endif #ifndef ETH_TX_DESCRIPTOR_COUNT #define ETH_TX_DESCRIPTOR_COUNT (8U) #endif #ifndef ETH_RX_BUFFER_SIZE #define ETH_RX_BUFFER_SIZE (256U) #endif /* Bitmask to extract link state */ #define LINK_STATE (0x01) /* Bitmask to extract notification state */ #define LINK_STATE_NOTIFIED (0x02) #define LINK_STATE_DOWN (0x00) #define LINK_STATE_UP (0x01) #define LINK_STATE_NOTIFIED_DOWN (LINK_STATE_NOTIFIED | LINK_STATE_DOWN) #define LINK_STATE_NOTIFIED_UP (LINK_STATE_NOTIFIED | LINK_STATE_UP) #if (ETH_RX_BUFFER_SIZE % 16) != 0 /* For compatibility with 128bit memory interfaces, the buffer size needs to * be a multiple of 16 Byte. For 64 bit memory interfaces need the size to be * a multiple of 8 Byte, for 32 bit a multiple of 4 byte is sufficient. */ #warning "ETH_RX_BUFFER_SIZE is not a multiple of 16. (See comment above.)" #endif #if ETH_RX_DESCRIPTOR_COUNT * ETH_RX_BUFFER_SIZE < 1524U #warning "Total RX buffers lower than MTU, you won't receive huge frames!" #endif #define MIN(a, b) (((a) <= (b)) ? (a) : (b)) /* Synchronization between IRQ and thread context */ mutex_t stm32_eth_tx_completed = MUTEX_INIT_LOCKED; /* Descriptors */ static edma_desc_t rx_desc[ETH_RX_DESCRIPTOR_COUNT]; static edma_desc_t tx_desc[ETH_TX_DESCRIPTOR_COUNT]; static edma_desc_t *rx_curr; static edma_desc_t *tx_curr; /* RX Buffers */ static char rx_buffer[ETH_RX_DESCRIPTOR_COUNT][ETH_RX_BUFFER_SIZE]; /* Netdev used in RIOT's API to upper layer */ netdev_t *stm32_eth_netdev; #if IS_USED(MODULE_STM32_ETH_LINK_UP) /* Used for checking the link status */ static uint8_t _link_state = LINK_STATE_DOWN; #endif /* IS_USED(MODULE_STM32_ETH_LINK_UP) */ static void _debug_tx_descriptor_info(unsigned line) { if (IS_ACTIVE(ENABLE_DEBUG) && IS_ACTIVE(ENABLE_DEBUG_VERBOSE)) { DEBUG("[stm32_eth:%u] TX descriptors:\n", line); for (unsigned i = 0; i < ETH_TX_DESCRIPTOR_COUNT; i++) { uint32_t status = tx_desc[i].status; DEBUG(" %s %u: OWN=%c, ES=%c, UF=%c, EC=%c, NC=%c, FS=%c, " "LS=%c\n", (tx_curr == tx_desc + i) ? "-->" : " ", i, (status & TX_DESC_STAT_OWN) ? '1' : '0', (status & TX_DESC_STAT_ES) ? '1' : '0', (status & TX_DESC_STAT_UF) ? '1' : '0', (status & TX_DESC_STAT_EC) ? '1' : '0', (status & TX_DESC_STAT_NC) ? '1' : '0', (status & TX_DESC_STAT_FS) ? '1' : '0', (status & TX_DESC_STAT_LS) ? '1' : '0'); } } } static inline uint32_t _len_from_rx_desc_status(uint32_t status) { /* bits 16-29 contain the frame length including 4 B frame check sequence */ return (status >> 16) & 0x3fff; } static void _debug_rx_descriptor_info(unsigned line) { if (IS_ACTIVE(ENABLE_DEBUG) && IS_ACTIVE(ENABLE_DEBUG_VERBOSE)) { DEBUG("[stm32_eth:%u] RX descriptors:\n", line); for (unsigned i = 0; i < ETH_RX_DESCRIPTOR_COUNT; i++) { uint32_t status = rx_desc[i].status; DEBUG(" %s %u: OWN=%c, FS=%c, LS=%c, ES=%c, DE=%c, FL=%lu\n", (rx_curr == rx_desc + i) ? "-->" : " ", i, (status & RX_DESC_STAT_OWN) ? '1' : '0', (status & RX_DESC_STAT_FS) ? '1' : '0', (status & RX_DESC_STAT_LS) ? '1' : '0', (status & RX_DESC_STAT_ES) ? '1' : '0', (status & RX_DESC_STAT_DE) ? '1' : '0', _len_from_rx_desc_status(status)); } } } /** * @brief Read or write a MII register * * @param[in] reg MII register to access * @param[in] value Value to write (ignored when @p write is `false`) * @param[in] write Whether to write (`true`) or read (`false`) to/from the * register * * @return The value of the MII register accessed. (This should be equal to * @p value, if @p write was `true`.) */ static uint16_t _mii_reg_transfer(unsigned reg, uint16_t value, bool write) { unsigned tmp; const uint16_t phy_addr = eth_config.phy_addr; while (ETH->MACMIIAR & ETH_MACMIIAR_MB) {} tmp = CLOCK_RANGE | ETH_MACMIIAR_MB | (((phy_addr & 0x1f) << 11) | ((reg & 0x1f) << 6)); if (write) { tmp |= ETH_MACMIIAR_MW; ETH->MACMIIDR = value; } ETH->MACMIIAR = tmp; while (ETH->MACMIIAR & ETH_MACMIIAR_MB) {} return ETH->MACMIIDR; } static inline int16_t _mii_reg_read(uint8_t reg) { return _mii_reg_transfer(reg, 0, false); } static inline void _mii_reg_write(uint8_t reg, uint16_t value) { _mii_reg_transfer(reg, value, true); } static inline bool _get_link_status(void) { return (_mii_reg_read(MII_BMSR) & MII_BMSR_LINK); } static void stm32_eth_get_addr(char *out) { unsigned t; t = ETH->MACA0HR; out[5] = (t >> 8); out[4] = (t & 0xff); t = ETH->MACA0LR; out[3] = (t >> 24); out[2] = (t >> 16); out[1] = (t >> 8); out[0] = (t & 0xff); } /** Set the mac address. The peripheral supports up to 4 MACs but only one is * implemented */ static void stm32_eth_set_addr(const uint8_t *addr) { ETH->MACA0HR &= 0xffff0000; ETH->MACA0HR |= (addr[5] << 8) | addr[4]; ETH->MACA0LR = (addr[3] << 24) | (addr[2] << 16) | (addr[1] << 8) | addr[0]; } static void _init_dma_descriptors(void) { size_t i; for (i = 0; i < ETH_RX_DESCRIPTOR_COUNT; i++) { rx_desc[i].status = RX_DESC_STAT_OWN; rx_desc[i].control = RX_DESC_CTRL_RCH | (ETH_RX_BUFFER_SIZE & 0x0fff); rx_desc[i].buffer_addr = &rx_buffer[i][0]; if ((i + 1) < ETH_RX_DESCRIPTOR_COUNT) { rx_desc[i].desc_next = &rx_desc[i + 1]; } } rx_desc[i - 1].desc_next = &rx_desc[0]; for (i = 0; i < ETH_TX_DESCRIPTOR_COUNT - 1; i++) { tx_desc[i].desc_next = &tx_desc[i + 1]; } tx_desc[ETH_RX_DESCRIPTOR_COUNT - 1].desc_next = &tx_desc[0]; rx_curr = &rx_desc[0]; tx_curr = &tx_desc[0]; ETH->DMARDLAR = (uintptr_t)rx_curr; ETH->DMATDLAR = (uintptr_t)tx_curr; } static void _reset_eth_dma(void) { /* disable DMA TX and RX */ ETH->DMAOMR &= ~(ETH_DMAOMR_ST | ETH_DMAOMR_SR); _init_dma_descriptors(); /* enable DMA TX and RX */ ETH->DMAOMR |= ETH_DMAOMR_ST | ETH_DMAOMR_SR; } static int stm32_eth_set(netdev_t *dev, netopt_t opt, const void *value, size_t max_len) { int res = -1; switch (opt) { case NETOPT_ADDRESS: assert(max_len >= ETHERNET_ADDR_LEN); stm32_eth_set_addr(value); res = ETHERNET_ADDR_LEN; break; default: res = netdev_eth_set(dev, opt, value, max_len); break; } return res; } static int stm32_eth_get(netdev_t *dev, netopt_t opt, void *value, size_t max_len) { int res = -1; switch (opt) { case NETOPT_ADDRESS: assert(max_len >= ETHERNET_ADDR_LEN); stm32_eth_get_addr(value); res = ETHERNET_ADDR_LEN; break; case NETOPT_LINK: assert(max_len == sizeof(netopt_enable_t)); { netopt_enable_t tmp = _get_link_status(); memcpy(value, &tmp, sizeof(tmp)); } res = sizeof(netopt_enable_t); break; default: res = netdev_eth_get(dev, opt, value, max_len); break; } return res; } #if IS_USED(MODULE_STM32_ETH_LINK_UP) static void _timer_cb(void *arg) { netdev_t *dev = (netdev_t *)arg; uint8_t state = LINK_STATE_DOWN; if (_get_link_status()) { state = LINK_STATE_UP; } if ((_link_state & LINK_STATE) != state) { /* link state changed, notify upper layer */ _link_state = state; dev->event_callback(dev, NETDEV_EVENT_ISR); } xtimer_set(&_link_status_timer, STM32_ETH_LINK_UP_TIMEOUT_US); } #endif /* IS_USED(MODULE_STM32_ETH_LINK_UP) */ static bool _phy_can_negotiate(void) { return (_mii_reg_read(MII_BMSR) & MII_BMSR_HAS_AN); } #if IS_USED(MODULE_STM32_ETH_AUTO) static void _complete_auto_negotiation(void) { /* first, wait until auto-negotiation really has completed */ uint16_t bmsr; do { bmsr = _mii_reg_read(MII_BMSR); if (!(bmsr & MII_BMSR_LINK)) { /* disconnected during auto-negotiation */ return; } } while (!(bmsr & MII_BMSR_AN_DONE)); DEBUG("[stm32_eth] PHY auto-negotiation completed, PHY link up\n"); /* Get current MACCR state without speed config */ uint32_t maccr = ETH->MACCR & ~(ETH_MACCR_FES | ETH_MACCR_DM); /* stupidly, there is seemingly no way to get current connection speed * and duplex mode. But we can deduce it from our advertised capabilities * and the link partner abilities */ uint16_t adv = _mii_reg_read(MII_ADVERTISE); uint16_t lpa = _mii_reg_read(MII_LPA); if ((adv & MII_ADVERTISE_100) && (lpa & MII_LPA_100)) { /* 100 Mbps */ maccr |= ETH_MACCR_FES; if ((adv & MII_ADVERTISE_100_F) && (lpa & MII_LPA_100_F)) { /* full duplex */ maccr |= ETH_MACCR_DM; } } else if ((adv & MII_ADVERTISE_10_F) && (lpa & MII_LPA_10_F)) { /* full duplex */ maccr |= ETH_MACCR_DM; } DEBUG("[stm32_eth] %s Mbps %s duplex \n", (maccr & ETH_MACCR_FES) ? "100" : "10", (maccr & ETH_MACCR_DM) ? "full" : "half"); ETH->MACCR = maccr; } #endif /* IS_USED(MODULE_STM32_ETH_AUTO) */ static void _setup_phy(void) { DEBUG("[stm32_eth] Reset PHY\n"); /* reset PHY */ _mii_reg_write(MII_BMCR, MII_BMCR_RESET); /* wait till PHY reset is completed */ while (MII_BMCR_RESET & _mii_reg_read(MII_BMCR)) {} /* check if auto-negotiation is enabled and supported */ if (IS_USED(MODULE_STM32_ETH_AUTO) && _phy_can_negotiate()) { _mii_reg_write(MII_BMCR, MII_BMCR_AN_ENABLE); DEBUG("[stm32_eth] Enabled auto-negotiation\n"); /* We'll continue link setup once auto-negotiation is done */ return; } /* Get current MACCR state without speed config */ uint32_t maccr = ETH->MACCR & ~(ETH_MACCR_FES | ETH_MACCR_DM); DEBUG("[stm32_eth] No PHY auto-negotiation disabled or unsupported\n"); /* disable auto-negotiation and force manually configured speed to be * used */ _mii_reg_write(MII_BMCR, eth_config.speed); /* configure MACCR to match PHY speed */ if (eth_config.speed & MII_BMCR_FULL_DPLX) { maccr |= ETH_MACCR_DM; } if (eth_config.speed & MII_BMCR_SPEED_100) { maccr |= ETH_MACCR_FES; } DEBUG("[stm32_eth] %s Mbps %s duplex \n", (maccr & ETH_MACCR_FES) ? "100" : "10", (maccr & ETH_MACCR_DM) ? "full" : "half"); /* Apply new duplex & speed configuration in MAC */ ETH->MACCR = maccr; } static int stm32_eth_init(netdev_t *netdev) { (void)netdev; #if IS_USED(MODULE_STM32_ETH_LINK_UP) _link_status_timer.callback = _timer_cb; _link_status_timer.arg = netdev; xtimer_set(&_link_status_timer, STM32_ETH_LINK_UP_TIMEOUT_US); #endif /* The PTP clock is initialized prior to the netdevs and will have already * initialized the common stuff, if used.*/ if (!IS_USED(MODULE_PERIPH_INIT_PTP)) { stm32_eth_common_init(); } /* set the clock divider */ while (ETH->MACMIIAR & ETH_MACMIIAR_MB) {} ETH->MACMIIAR = CLOCK_RANGE; /* ROD = Don't receive own frames in half-duplex mode * IPCO = Drop IPv4 packets carrying TCP/UDP/ICMP when checksum is invalid * APCS = Do not pass padding and CRC fields to application (CRC is checked * by hardware already) */ ETH->MACCR |= ETH_MACCR_ROD | ETH_MACCR_IPCO | ETH_MACCR_APCS; /* pass all */ //ETH->MACFFR |= ETH_MACFFR_RA; /* pass on perfect filter match and pass all multicast address matches */ ETH->MACFFR |= ETH_MACFFR_PAM; /* store forward */ ETH->DMAOMR |= (ETH_DMAOMR_RSF | ETH_DMAOMR_TSF | ETH_DMAOMR_OSF); /* configure DMA */ ETH->DMABMR = ETH_DMABMR_DA | ETH_DMABMR_AAB | ETH_DMABMR_FB | ETH_DMABMR_RDP_32Beat | ETH_DMABMR_PBL_32Beat | ETH_DMABMR_EDE; netdev_register(netdev, NETDEV_STM32_ETH, 0); eui48_t hwaddr; netdev_eui48_get(netdev, &hwaddr); stm32_eth_set_addr(hwaddr.uint8); ETH->DMAIER |= ETH_DMAIER_NISE | ETH_DMAIER_TIE | ETH_DMAIER_RIE; /* enable transmitter and receiver */ ETH->MACCR |= ETH_MACCR_TE | ETH_MACCR_RE; /* flush transmit FIFO */ ETH->DMAOMR |= ETH_DMAOMR_FTF; /* wait for FIFO flushing to complete */ while (ETH->DMAOMR & ETH_DMAOMR_FTF) { } _reset_eth_dma(); _setup_phy(); return 0; } static int stm32_eth_send(netdev_t *netdev, const struct iolist *iolist) { (void)netdev; netdev->event_callback(netdev, NETDEV_EVENT_TX_STARTED); unsigned bytes_to_send = iolist_size(iolist); /* Input must not be bigger than maximum allowed frame length */ assert(bytes_to_send <= ETHERNET_FRAME_LEN); /* This API is not thread safe, check that no other thread is sending */ assert(!(tx_desc[0].status & TX_DESC_STAT_OWN)); /* We cannot send more chunks than allocated descriptors */ assert(iolist_count(iolist) <= ETH_TX_DESCRIPTOR_COUNT); edma_desc_t *dma_iter = tx_curr; for (unsigned i = 0; iolist; iolist = iolist->iol_next, i++) { dma_iter->control = iolist->iol_len; dma_iter->buffer_addr = iolist->iol_base; uint32_t status = TX_DESC_STAT_IC | TX_DESC_STAT_TCH | TX_DESC_STAT_CIC | TX_DESC_STAT_OWN; if (!i) { /* fist chunk */ status |= TX_DESC_STAT_FS; } if (!iolist->iol_next) { /* last chunk */ status |= TX_DESC_STAT_LS; } dma_iter->status = status; dma_iter = dma_iter->desc_next; } /* start TX */ ETH->DMATPDR = 0; /* await completion */ if (IS_ACTIVE(ENABLE_DEBUG_VERBOSE)) { DEBUG("[stm32_eth] Started to send %u B via DMA\n", bytes_to_send); } mutex_lock(&stm32_eth_tx_completed); if (IS_ACTIVE(ENABLE_DEBUG_VERBOSE)) { DEBUG("[stm32_eth] TX completed\n"); } /* Error check */ _debug_tx_descriptor_info(__LINE__); int error = 0; while (1) { uint32_t status = tx_curr->status; /* The Error Summary (ES) bit is set, if any error during TX occurred */ if (status & TX_DESC_STAT_ES) { if (status & TX_DESC_STAT_EC) { DEBUG("[stm32_eth] collision in half duplex mode\n"); error = -EBUSY; } else if (status & TX_DESC_STAT_NC) { DEBUG("[stm32_eth] no carrier detected during TX\n"); error = -ENETDOWN; } else { /* don't detect underflow error here, as we trigger TX only * after all descriptors have been handed over to the DMA. * Hence, the DMA should never run out of desciprtors during * TX. */ DEBUG("[stm32_eth] unhandled error during TX\n"); error = -EIO; } _reset_eth_dma(); } tx_curr = tx_curr->desc_next; if (status & TX_DESC_STAT_LS) { break; } } netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE); if (error) { return error; } return (int)bytes_to_send; } static int get_rx_frame_size(void) { _debug_rx_descriptor_info(__LINE__); edma_desc_t *i = rx_curr; uint32_t status; while (1) { /* Wait until DMA gave up control over descriptor */ if ((status = i->status) & RX_DESC_STAT_OWN) { DEBUG("[stm32_eth] RX not completed (spurious interrupt?)\n"); return -EAGAIN; } if (status & RX_DESC_STAT_DE) { DEBUG("[stm32_eth] Overflow during RX\n"); _reset_eth_dma(); return -EOVERFLOW; } if (status & RX_DESC_STAT_ES) { DEBUG("[stm32_eth] Error during RX\n"); _reset_eth_dma(); return -EIO; } if (status & RX_DESC_STAT_LS) { break; } i = i->desc_next; } return _len_from_rx_desc_status(status) - ETHERNET_FCS_LEN; } static void drop_frame_and_update_rx_curr(void) { while (1) { uint32_t old_status = rx_curr->status; /* hand over old descriptor to DMA */ rx_curr->status = RX_DESC_STAT_OWN; rx_curr = rx_curr->desc_next; if (old_status & (RX_DESC_STAT_LS | RX_DESC_STAT_ES)) { /* reached either last DMA descriptor of frame or error ==> done */ return; } } } static void handle_lost_rx_irqs(void) { edma_desc_t *iter = rx_curr; while (1) { uint32_t status = iter->status; if (status & RX_DESC_STAT_OWN) { break; } if (status & RX_DESC_STAT_LS) { DEBUG("[stm32_eth] Lost RX IRQ, sending event to upper layer\n"); /* we use the ISR event for this, as the upper layer calls recv() * right away on an NETDEV_EVENT_RX_COMPLETE. Because there could be * potentially quite a lot of received frames in the queue, we might * risk a stack overflow if we would send an * NETDEV_EVENT_RX_COMPLETE */ netdev_trigger_event_isr(stm32_eth_netdev); break; } iter = iter->desc_next; } } static int stm32_eth_recv(netdev_t *netdev, void *_buf, size_t max_len, void *_info) { (void)netdev; netdev_eth_rx_info_t *info = _info; char *buf = _buf; /* Determine the size of received frame. The frame might span multiple * DMA buffers */ int size = get_rx_frame_size(); if (size < 0) { if (size != -EAGAIN) { DEBUG("[stm32_eth] Dropping frame due to error\n"); drop_frame_and_update_rx_curr(); } return size; } if (!buf) { if (max_len) { DEBUG("[stm32_eth] Dropping frame as requested by upper layer\n"); drop_frame_and_update_rx_curr(); } return size; } if (max_len < (size_t)size) { DEBUG("[stm32_eth] Buffer provided by upper layer is too small\n"); drop_frame_and_update_rx_curr(); return -ENOBUFS; } /* Fetch payload, collect RX timestamp from last descriptor if module periph_ptp is used, and * hand DMA descriptors back to the DMA */ size_t remain = size; while (1) { /* there can be one more DMA descriptor than needed for holding the Ethernet * payload, as the 4 byte FCS will also be stored by DMA */ if (remain) { size_t chunk = MIN(remain, ETH_RX_BUFFER_SIZE); memcpy(buf, rx_curr->buffer_addr, chunk); buf += chunk; remain -= chunk; } if (rx_curr->status & RX_DESC_STAT_LS) { /* LS bit set --> reached last DMA descriptor of this frame */ if (IS_USED(MODULE_PERIPH_PTP)) { info->timestamp = rx_curr->ts_low; info->timestamp += (uint64_t)rx_curr->ts_high * NS_PER_SEC; info->flags |= NETDEV_ETH_RX_INFO_FLAG_TIMESTAMP; } rx_curr->status = RX_DESC_STAT_OWN; rx_curr = rx_curr->desc_next; break; } rx_curr->status = RX_DESC_STAT_OWN; rx_curr = rx_curr->desc_next; } _debug_rx_descriptor_info(__LINE__); handle_lost_rx_irqs(); return size; } void stm32_eth_isr_eth_wkup(void) { cortexm_isr_end(); } static void stm32_eth_isr(netdev_t *netdev) { #if IS_USED(MODULE_STM32_ETH_LINK_UP) switch (_link_state) { case LINK_STATE_UP: DEBUG("[stm32_eth] Link UP\n"); if (IS_USED(MODULE_STM32_ETH_AUTO)) { /* Complete auto-negotiation of the link */ _complete_auto_negotiation(); } netdev->event_callback(netdev, NETDEV_EVENT_LINK_UP); _link_state = LINK_STATE_NOTIFIED_UP; return; case LINK_STATE_DOWN: DEBUG("[stm32_eth] Link DOWN\n"); netdev->event_callback(netdev, NETDEV_EVENT_LINK_DOWN); _link_state = LINK_STATE_NOTIFIED_DOWN; return; default: break; } #endif netdev->event_callback(netdev, NETDEV_EVENT_RX_COMPLETE); } static const netdev_driver_t netdev_driver_stm32f4eth = { .send = stm32_eth_send, .recv = stm32_eth_recv, .init = stm32_eth_init, .isr = stm32_eth_isr, .get = stm32_eth_get, .set = stm32_eth_set, }; void stm32_eth_netdev_setup(netdev_t *netdev) { stm32_eth_netdev = netdev; netdev->driver = &netdev_driver_stm32f4eth; }