Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
drv_eth_以太网驱动
我现在用GD32F207VG做了带网口的板子,想请问一下关于网口驱动的问题?
发布于 2022-06-24 19:02:01 浏览:577
订阅该版
第一次接触这个国产芯片,在网上下载了关于这个芯片的资料,想办法把驱动文件移植过来的时候,能正常编译通过,烧写在板子上不可以运行 ```c /* * Copyright (c) 2006-2021, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2022-06-22 MoBin the first version */ /*! \file gd32f20x_enet_eval.c \brief ethernet hardware configuration \version 2015-07-15, V1.0.0, demo for GD32F20x \version 2017-06-05, V2.0.0, demo for GD32F20x \version 2018-10-31, V2.1.0, demo for GD32F20x \version 2020-09-30, V2.2.0, demo for GD32F20x \version 2021-11-01, V2.3.0, demo for GD32F20x */ /* Copyright (c) 2021, GigaDevice Semiconductor Inc. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "gd32f20x_enet.h" #include "gd32f20x_enet_eval.h" #include "main.h" static __IO uint32_t enet_init_status = 0; static void enet_gpio_config(void); static void enet_mac_dma_config(void); static void nvic_configuration(void); /*! \brief setup ethernet system(GPIOs, clocks, MAC, DMA, systick) \param[in] none \param[out] none \retval none */ void enet_system_setup(void) { nvic_configuration(); /* configure the GPIO ports for ethernet pins */ enet_gpio_config(); /* configure the ethernet MAC/DMA */ enet_mac_dma_config(); if(enet_init_status == 0) { while(1) { } } enet_interrupt_enable(ENET_DMA_INT_NIE); enet_interrupt_enable(ENET_DMA_INT_RIE); } /*! \brief configures the ethernet interface \param[in] none \param[out] none \retval none */ static void enet_mac_dma_config(void) { ErrStatus reval_state = ERROR; /* enable ethernet clock */ rcu_periph_clock_enable(RCU_ENET); rcu_periph_clock_enable(RCU_ENETTX); rcu_periph_clock_enable(RCU_ENETRX); /* reset ethernet on AHB bus */ enet_deinit(); reval_state = enet_software_reset(); if(reval_state == ERROR) { while(1) { } } #ifdef CHECKSUM_BY_HARDWARE enet_init_status = enet_init(ENET_AUTO_NEGOTIATION, ENET_AUTOCHECKSUM_DROP_FAILFRAMES, ENET_BROADCAST_FRAMES_PASS); #else enet_init_status = enet_init(ENET_AUTO_NEGOTIATION, ENET_NO_AUTOCHECKSUM, ENET_BROADCAST_FRAMES_PASS); #endif } /*! \brief configures the nested vectored interrupt controller \param[in] none \param[out] none \retval none */ static void nvic_configuration(void) { nvic_vector_table_set(NVIC_VECTTAB_FLASH, 0x0); nvic_irq_enable(ENET_IRQn, 2, 0); } /*! \brief configures the different GPIO ports \param[in] none \param[out] none \retval none */ static void enet_gpio_config(void) { rcu_periph_clock_enable(RCU_GPIOA); rcu_periph_clock_enable(RCU_GPIOB); rcu_periph_clock_enable(RCU_GPIOC); gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8); /* enable SYSCFG clock */ rcu_periph_clock_enable(RCU_AF); #ifdef MII_MODE #ifdef PHY_CLOCK_MCO /* output HXTAL clock (25MHz) on CKOUT0 pin(PA8) to clock the PHY */ rcu_ckout0_config(RCU_CKOUT0SRC_HXTAL, RCU_CKOUT0_DIV1); #endif /* PHY_CLOCK_MCO */ gpio_ethernet_phy_select(GPIO_ENET_PHY_MII); #elif defined RMII_MODE rcu_pll2_config(RCU_PLL2_MUL10); rcu_osci_on(RCU_PLL2_CK); rcu_osci_stab_wait(RCU_PLL2_CK); /* get 50MHz from CK_PLL2 on CKOUT0 pin (PA8) to clock the PHY */ rcu_ckout0_config(RCU_CKOUT0SRC_CKPLL2, RCU_CKOUT0_DIV1); gpio_ethernet_phy_select(GPIO_ENET_PHY_RMII); #endif #ifdef MII_MODE /* PA0: ETH_MII_CRS */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_0); /* PA1: ETH_RX_CLK */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PA2: ETH_MDIO */ gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* PA3: ETH_MII_COL */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_3); /* PA7: ETH_MII_RX_DV */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_7); /* PC1: ETH_MDC */ gpio_init(GPIOC, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PC2: ETH_MII_TXD2 */ gpio_init(GPIOC, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* PC3: ETH_MII_TX_CLK */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_3); /* PC4: ETH_MII_RXD0 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_4); /* PC5: ETH_MII_RXD1 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_5); /* PB0: ETH_MII_RXD2 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_0); /* PB1: ETH_MII_RXD3 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PB8: ETH_MII_TXD3 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8); /* PB10: ETH_MII_RX_ER */ gpio_init(GPIOB, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_10); /* PB11: ETH_MII_TX_EN */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_11); /* PB12: ETH_MII_TXD0 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_12); /* PB13: ETH_MII_TXD1 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_13); #elif defined RMII_MODE /* PA1: ETH_RMII_REF_CLK */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PA2: ETH_MDIO */ gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* PA7: ETH_RMII_CRS_DV */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_7); /* PC1: ETH_MDC */ gpio_init(GPIOC, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PC4: ETH_RMII_RXD0 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_4); /* PC5: ETH_RMII_RXD1 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_5); /* PB11: ETH_RMII_TX_EN */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_11); /* PB12: ETH_RMII_TXD0 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_12); /* PB13: ETH_RMII_TXD1 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_13); #endif /* MII_MODE */ } ``` ```c /*! \file gd32f20x_enet_eval.h \brief the header file of gd32f20x_enet_eval \version 2015-07-15, V1.0.0, demo for GD32F20x \version 2017-06-05, V2.0.0, demo for GD32F20x \version 2018-10-31, V2.1.0, demo for GD32F20x \version 2020-09-30, V2.2.0, demo for GD32F20x \version 2021-11-01, V2.3.0, demo for GD32F20x */ /* Copyright (c) 2021, GigaDevice Semiconductor Inc. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef GD32F20x_ENET_EVAL_H #define GD32F20x_ENET_EVAL_H #include "lwip/netif.h" #include "gd32f20x_enet.h" /* function declarations */ /* setup ethernet system(GPIOs, clocks, MAC, DMA, systick) */ void enet_system_setup(void); #endif /* GD32F20x_ENET_EVAL_H */ ``` 然后尝试用GD32F4的代码,也还是不行 ```c /* * Copyright (c) 2006-2021, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2011-11-30 aozima the first version. * 2011-12-10 aozima support dual ethernet. * 2011-12-21 aozima cleanup code. * 2012-07-13 aozima mask all GMAC MMC Interrupt. * 2012-07-20 aozima fixed mask all GMAC MMC Interrupt,and read clear. * 2012-07-20 aozima use memcpy replace byte copy. */ #include
#include
#include <> #include "lwipopts.h" #include
#include
#include
#include "gd32f20x_enet.h" #include "gd32f20x_enet_eval.h" #include "main.h" #include "gd32f20x.h" #include "synopsys_emac.h" #define ETHERNET_MAC0 ((struct rt_synopsys_eth *)(0x40020000U + 0x00008000U)) //#define EMAC_DEBUG //#define EMAC_RX_DUMP //#define EMAC_TX_DUMP #ifdef EMAC_DEBUG #define EMAC_TRACE rt_kprintf #else #define EMAC_TRACE(...) #endif #define EMAC_RXBUFNB 4 #define EMAC_TXBUFNB 2 #define EMAC_PHY_AUTO 0 #define EMAC_PHY_10MBIT 1 #define EMAC_PHY_100MBIT 2 #define MAX_ADDR_LEN 6 struct gd32_emac { /* inherit from Ethernet device */ struct eth_device parent; rt_uint8_t phy_mode; /* interface address info. */ rt_uint8_t dev_addr[MAX_ADDR_LEN]; /* hw address */ struct rt_synopsys_eth * ETHERNET_MAC; IRQn_Type ETHER_MAC_IRQ; EMAC_DMADESCTypeDef *DMATxDescToSet; EMAC_DMADESCTypeDef *DMARxDescToGet; #pragma pack(4) EMAC_DMADESCTypeDef DMARxDscrTab[EMAC_RXBUFNB]; #pragma pack(4) EMAC_DMADESCTypeDef DMATxDscrTab[EMAC_TXBUFNB]; #pragma pack(4) rt_uint8_t Rx_Buff[EMAC_RXBUFNB][EMAC_MAX_PACKET_SIZE]; #pragma pack(4) rt_uint8_t Tx_Buff[EMAC_TXBUFNB][EMAC_MAX_PACKET_SIZE]; struct rt_semaphore tx_buf_free; }; static struct gd32_emac gd32_emac_device0; /** * Initializes the DMA Tx descriptors in chain mode. */ static void EMAC_DMA_tx_desc_init(EMAC_DMADESCTypeDef *DMATxDescTab, uint8_t* TxBuff, uint32_t TxBuffCount) { uint32_t i = 0; EMAC_DMADESCTypeDef *DMATxDesc; /* Fill each DMATxDesc descriptor with the right values */ for(i=0; i < TxBuffCount; i++) { /* Get the pointer on the ith member of the Tx Desc list */ DMATxDesc = DMATxDescTab + i; /* Set Second Address Chained bit */ DMATxDesc->Status = EMAC_DMATxDesc_TCH; /* Set Buffer1 address pointer */ DMATxDesc->Buffer1Addr = (uint32_t)(&TxBuff[i*EMAC_MAX_PACKET_SIZE]); /* Initialize the next descriptor with the Next Descriptor Polling Enable */ if(i < (TxBuffCount-1)) { /* Set next descriptor address register with next descriptor base address */ DMATxDesc->Buffer2NextDescAddr = (uint32_t)(DMATxDescTab+i+1); } else { /* For last descriptor, set next descriptor address register equal to the first descriptor base address */ DMATxDesc->Buffer2NextDescAddr = (uint32_t) DMATxDescTab; } } } /** * Initializes the DMA Rx descriptors in chain mode. */ static void EMAC_DMA_rx_desc_init(EMAC_DMADESCTypeDef *DMARxDescTab, uint8_t *RxBuff, uint32_t RxBuffCount) { uint32_t i = 0; EMAC_DMADESCTypeDef *DMARxDesc; /* Fill each DMARxDesc descriptor with the right values */ for(i=0; i < RxBuffCount; i++) { /* Get the pointer on the ith member of the Rx Desc list */ DMARxDesc = DMARxDescTab+i; /* Set Own bit of the Rx descriptor Status */ DMARxDesc->Status = EMAC_DMARxDesc_OWN; /* Set Buffer1 size and Second Address Chained bit */ DMARxDesc->ControlBufferSize = EMAC_DMARxDesc_RCH | (uint32_t)EMAC_MAX_PACKET_SIZE; /* Set Buffer1 address pointer */ DMARxDesc->Buffer1Addr = (uint32_t)(&RxBuff[i*EMAC_MAX_PACKET_SIZE]); /* Initialize the next descriptor with the Next Descriptor Polling Enable */ if(i < (RxBuffCount-1)) { /* Set next descriptor address register with next descriptor base address */ DMARxDesc->Buffer2NextDescAddr = (uint32_t)(DMARxDescTab+i+1); } else { /* For last descriptor, set next descriptor address register equal to the first descriptor base address */ DMARxDesc->Buffer2NextDescAddr = (uint32_t)(DMARxDescTab); } } } static rt_err_t gd32_emac_init(rt_device_t dev) { struct gd32_emac * gd32_emac_device; struct rt_synopsys_eth * ETHERNET_MAC; gd32_emac_device = (struct gd32_emac *)dev; ETHERNET_MAC = gd32_emac_device->ETHERNET_MAC; /* Software reset */ ETHERNET_MAC->BMR |= (1<<0); /* [bit0]SWR (Software Reset) */ /* Wait for software reset */ while(ETHERNET_MAC->BMR & (1<<0)); /* Configure ETHERNET */ EMAC_init(ETHERNET_MAC, SystemCoreClock); /* mask all GMAC MMC Interrupt.*/ ETHERNET_MAC->mmc_cntl = (1<<3) | (1<<0); /* MMC Counter Freeze and reset. */ ETHERNET_MAC->mmc_intr_mask_rx = 0xFFFFFFFF; ETHERNET_MAC->mmc_intr_mask_tx = 0xFFFFFFFF; ETHERNET_MAC->mmc_ipc_intr_mask_rx = 0xFFFFFFFF; /* Enable DMA Receive interrupt (need to enable in this case Normal interrupt) */ EMAC_INT_config(ETHERNET_MAC, EMAC_DMA_INT_NIS | EMAC_DMA_INT_R | EMAC_DMA_INT_T , ENABLE); /* Initialize Tx Descriptors list: Chain Mode */ EMAC_DMA_tx_desc_init(gd32_emac_device->DMATxDscrTab, &gd32_emac_device->Tx_Buff[0][0], EMAC_TXBUFNB); gd32_emac_device->DMATxDescToSet = gd32_emac_device->DMATxDscrTab; /* Set Transmit Descriptor List Address Register */ ETHERNET_MAC->TDLAR = (uint32_t) gd32_emac_device->DMATxDescToSet; /* Initialize Rx Descriptors list: Chain Mode */ EMAC_DMA_rx_desc_init(gd32_emac_device->DMARxDscrTab, &gd32_emac_device->Rx_Buff[0][0], EMAC_RXBUFNB); gd32_emac_device->DMARxDescToGet = gd32_emac_device->DMARxDscrTab; /* Set Receive Descriptor List Address Register */ ETHERNET_MAC->RDLAR = (uint32_t) gd32_emac_device->DMARxDescToGet; /* MAC address configuration */ EMAC_MAC_Addr_config(ETHERNET_MAC, EMAC_MAC_Address0, (uint8_t*)&gd32_emac_device->dev_addr[0]); NVIC_EnableIRQ( gd32_emac_device->ETHER_MAC_IRQ ); /* Enable MAC and DMA transmission and reception */ EMAC_start(ETHERNET_MAC); return RT_EOK; } static rt_err_t gd32_emac_open(rt_device_t dev, rt_uint16_t oflag) { return RT_EOK; } static rt_err_t gd32_emac_close(rt_device_t dev) { return RT_EOK; } static rt_size_t gd32_emac_read(rt_device_t dev, rt_off_t pos, void* buffer, rt_size_t size) { rt_set_errno(-RT_ENOSYS); return 0; } static rt_size_t gd32_emac_write (rt_device_t dev, rt_off_t pos, const void* buffer, rt_size_t size) { rt_set_errno(-RT_ENOSYS); return 0; } static rt_err_t gd32_emac_control(rt_device_t dev, int cmd, void *args) { struct gd32_emac * gd32_emac_device = (struct gd32_emac *)dev; switch (cmd) { case NIOCTL_GADDR: /* get mac address */ if (args) memcpy(args, &gd32_emac_device->dev_addr[0], MAX_ADDR_LEN); else return -RT_ERROR; break; default : break; } return RT_EOK; } static void EMAC_IRQHandler(struct gd32_emac * gd32_emac_device) { rt_uint32_t status, ier; struct rt_synopsys_eth * ETHERNET_MAC; ETHERNET_MAC = gd32_emac_device->ETHERNET_MAC; /* get DMA IT status */ status = ETHERNET_MAC->SR; ier = ETHERNET_MAC->IER; /* GMAC MMC Interrupt. */ if(status & EMAC_DMA_INT_GMI) { volatile rt_uint32_t dummy; volatile rt_uint32_t * reg; EMAC_TRACE("EMAC_DMA_INT_GMI\r\n"); /* read clear all MMC interrupt. */ reg = ÐERNET_MAC->mmc_cntl; while((uint32_t)reg < (uint32_t)ÐERNET_MAC->rxicmp_err_octets) { dummy = *reg++; } } /* Normal interrupt summary. */ if(status & EMAC_DMA_INT_NIS) { rt_uint32_t nis_clear = EMAC_DMA_INT_NIS; /* [0]:Transmit Interrupt. */ if((status & ier) & EMAC_DMA_INT_T) /* packet transmission */ { rt_sem_release(&gd32_emac_device->tx_buf_free); nis_clear |= EMAC_DMA_INT_T; } /* [2]:Transmit Buffer Unavailable. */ /* [6]:Receive Interrupt. */ if((status & ier) & EMAC_DMA_INT_R) /* packet reception */ { /* a frame has been received */ eth_device_ready(&(gd32_emac_device->parent)); nis_clear |= EMAC_DMA_INT_R; } /* [14]:Early Receive Interrupt. */ EMAC_clear_pending(ETHERNET_MAC, nis_clear); } /* Abnormal interrupt summary. */ if( status & EMAC_DMA_INT_AIS) { rt_uint32_t ais_clear = EMAC_DMA_INT_AIS; /* [1]:Transmit Process Stopped. */ /* [3]:Transmit Jabber Timeout. */ /* [4]: Receive FIFO Overflow. */ /* [5]: Transmit Underflow. */ /* [7]: Receive Buffer Unavailable. */ /* [8]: Receive Process Stopped. */ /* [9]: Receive Watchdog Timeout. */ /* [10]: Early Transmit Interrupt. */ /* [13]: Fatal Bus Error. */ EMAC_clear_pending(ETHERNET_MAC, ais_clear); } } void ENET_IRQHandler(void) { /* enter interrupt */ rt_interrupt_enter(); EMAC_IRQHandler(&gd32_emac_device0); /* leave interrupt */ rt_interrupt_leave(); } /* EtherNet Device Interface */ rt_err_t gd32_emac_tx( rt_device_t dev, struct pbuf* p) { struct pbuf* q; char * to; struct gd32_emac * gd32_emac_device; struct rt_synopsys_eth * ETHERNET_MAC; gd32_emac_device = (struct gd32_emac *)dev; ETHERNET_MAC = gd32_emac_device->ETHERNET_MAC; /* get free tx buffer */ { rt_err_t result; result = rt_sem_take(&gd32_emac_device->tx_buf_free, RT_TICK_PER_SECOND/10); if (result != RT_EOK) return -RT_ERROR; } to = (char *)gd32_emac_device->DMATxDescToSet->Buffer1Addr; for (q = p; q != NULL; q = q->next) { /* Copy the frame to be sent into memory pointed by the current ETHERNET DMA Tx descriptor */ memcpy(to, q->payload, q->len); to += q->len; } #ifdef EMAC_TX_DUMP { rt_uint32_t i; rt_uint8_t *ptr = (rt_uint8_t*)(gd32_emac_device->DMATxDescToSet->Buffer1Addr); EMAC_TRACE("\r\n%c%c tx_dump:", gd32_emac_device->parent.netif->name[0], gd32_emac_device->parent.netif->name[1]); for(i=0; i
tot_len; i++) { if( (i%8) == 0 ) { EMAC_TRACE(" "); } if( (i%16) == 0 ) { EMAC_TRACE("\r\n"); } EMAC_TRACE("%02x ",*ptr); ptr++; } EMAC_TRACE("\r\ndump done!\r\n"); } #endif /* Setting the Frame Length: bits[12:0] */ gd32_emac_device->DMATxDescToSet->ControlBufferSize = (p->tot_len & EMAC_DMATxDesc_TBS1); /* Setting the last segment and first segment bits (in this case a frame is transmitted in one descriptor) */ gd32_emac_device->DMATxDescToSet->Status |= EMAC_DMATxDesc_LS | EMAC_DMATxDesc_FS; /* Enable TX Completion Interrupt */ gd32_emac_device->DMATxDescToSet->Status |= EMAC_DMATxDesc_IC; #ifdef CHECKSUM_BY_HARDWARE gd32_emac_device->DMATxDescToSet->Status |= EMAC_DMATxDesc_ChecksumTCPUDPICMPFull; /* clean ICMP checksum */ { struct eth_hdr *ethhdr = (struct eth_hdr *)(gd32_emac_device->DMATxDescToSet->Buffer1Addr); /* is IP ? */ if( ethhdr->type == htons(ETHTYPE_IP) ) { struct ip_hdr *iphdr = (struct ip_hdr *)(gd32_emac_device->DMATxDescToSet->Buffer1Addr + SIZEOF_ETH_HDR); /* is ICMP ? */ if( IPH_PROTO(iphdr) == IP_PROTO_ICMP ) { struct icmp_echo_hdr *iecho = (struct icmp_echo_hdr *)(gd32_emac_device->DMATxDescToSet->Buffer1Addr + SIZEOF_ETH_HDR + sizeof(struct ip_hdr) ); iecho->chksum = 0; } } } #endif /* Set Own bit of the Tx descriptor Status: gives the buffer back to ETHERNET DMA */ gd32_emac_device->DMATxDescToSet->Status |= EMAC_DMATxDesc_OWN; /* When Tx Buffer unavailable flag is set: clear it and resume transmission */ if ((ETHERNET_MAC->SR & EMAC_DMASR_TBUS) != (uint32_t)RESET) { /* Clear TBUS ETHERNET DMA flag */ ETHERNET_MAC->SR = EMAC_DMASR_TBUS; /* Transmit Poll Demand to resume DMA transmission*/ ETHERNET_MAC->TPDR = 0; } /* Update the ETHERNET DMA global Tx descriptor with next Tx decriptor */ /* Chained Mode */ /* Selects the next DMA Tx descriptor list for next buffer to send */ gd32_emac_device->DMATxDescToSet = (EMAC_DMADESCTypeDef*) (gd32_emac_device->DMATxDescToSet->Buffer2NextDescAddr); /* Return SUCCESS */ return RT_EOK; } /* reception a Ethernet packet. */ struct pbuf * gd32_emac_rx(rt_device_t dev) { struct pbuf* p; rt_uint32_t framelength = 0; struct gd32_emac * gd32_emac_device; struct rt_synopsys_eth * ETHERNET_MAC; gd32_emac_device = (struct gd32_emac *)dev; ETHERNET_MAC = gd32_emac_device->ETHERNET_MAC; /* init p pointer */ p = RT_NULL; /* Check if the descriptor is owned by the ETHERNET DMA (when set) or CPU (when reset) */ if(((gd32_emac_device->DMARxDescToGet->Status & EMAC_DMARxDesc_OWN) != (uint32_t)RESET)) { return p; } if (((gd32_emac_device->DMARxDescToGet->Status & EMAC_DMARxDesc_ES) == (uint32_t)RESET) && ((gd32_emac_device->DMARxDescToGet->Status & EMAC_DMARxDesc_LS) != (uint32_t)RESET) && ((gd32_emac_device->DMARxDescToGet->Status & EMAC_DMARxDesc_FS) != (uint32_t)RESET)) { /* Get the Frame Length of the received packet: substruct 4 bytes of the CRC */ framelength = ((gd32_emac_device->DMARxDescToGet->Status & EMAC_DMARxDesc_FL) >> EMAC_DMARXDESC_FRAME_LENGTHSHIFT) - 4; /* allocate buffer */ p = pbuf_alloc(PBUF_LINK, framelength, PBUF_RAM); if (p != RT_NULL) { const char * from; struct pbuf* q; from = (const char *)gd32_emac_device->DMARxDescToGet->Buffer1Addr; for (q = p; q != RT_NULL; q= q->next) { /* Copy the received frame into buffer from memory pointed by the current ETHERNET DMA Rx descriptor */ memcpy(q->payload, from, q->len); from += q->len; } #ifdef EMAC_RX_DUMP { rt_uint32_t i; rt_uint8_t *ptr = (rt_uint8_t*)(gd32_emac_device->DMARxDescToGet->Buffer1Addr); EMAC_TRACE("\r\n%c%c rx_dump:", gd32_emac_device->parent.netif->name[0], gd32_emac_device->parent.netif->name[1]); for(i=0; i
tot_len; i++) { if( (i%8) == 0 ) { EMAC_TRACE(" "); } if( (i%16) == 0 ) { EMAC_TRACE("\r\n"); } EMAC_TRACE("%02x ",*ptr); ptr++; } EMAC_TRACE("\r\ndump done!\r\n"); } #endif } } /* Set Own bit of the Rx descriptor Status: gives the buffer back to ETHERNET DMA */ gd32_emac_device->DMARxDescToGet->Status = EMAC_DMARxDesc_OWN; /* When Rx Buffer unavailable flag is set: clear it and resume reception */ if ((ETHERNET_MAC->SR & EMAC_DMASR_RBUS) != (uint32_t)RESET) { /* Clear RBUS ETHERNET DMA flag */ ETHERNET_MAC->SR = EMAC_DMASR_RBUS; /* Resume DMA reception */ ETHERNET_MAC->RPDR = 0; } /* Update the ETHERNET DMA global Rx descriptor with next Rx decriptor */ /* Chained Mode */ if((gd32_emac_device->DMARxDescToGet->ControlBufferSize & EMAC_DMARxDesc_RCH) != (uint32_t)RESET) { /* Selects the next DMA Rx descriptor list for next buffer to read */ gd32_emac_device->DMARxDescToGet = (EMAC_DMADESCTypeDef*) (gd32_emac_device->DMARxDescToGet->Buffer2NextDescAddr); } else /* Ring Mode */ { if((gd32_emac_device->DMARxDescToGet->ControlBufferSize & EMAC_DMARxDesc_RER) != (uint32_t)RESET) { /* Selects the first DMA Rx descriptor for next buffer to read: last Rx descriptor was used */ gd32_emac_device->DMARxDescToGet = (EMAC_DMADESCTypeDef*) (ETHERNET_MAC->RDLAR); } else { /* Selects the next DMA Rx descriptor list for next buffer to read */ gd32_emac_device->DMARxDescToGet = (EMAC_DMADESCTypeDef*) ((uint32_t)gd32_emac_device->DMARxDescToGet + 0x10 + ((ETHERNET_MAC->BMR & EMAC_DMABMR_DSL) >> 2)); } } return p; } /*! \brief configures the nested vectored interrupt controller \param[in] none \param[out] none \retval none */ static void nvic_configuration(void) { nvic_vector_table_set(NVIC_VECTTAB_FLASH, 0x0); nvic_irq_enable(ENET_IRQn, 2, 0); } /*! \brief configures the different GPIO ports \param[in] none \param[out] none \retval none */ static void enet_gpio_config(void) { rcu_periph_clock_enable(RCU_GPIOA); rcu_periph_clock_enable(RCU_GPIOB); rcu_periph_clock_enable(RCU_GPIOC); gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8); /* enable SYSCFG clock */ rcu_periph_clock_enable(RCU_AF); #ifdef MII_MODE #ifdef PHY_CLOCK_MCO /* output HXTAL clock (25MHz) on CKOUT0 pin(PA8) to clock the PHY */ rcu_ckout0_config(RCU_CKOUT0SRC_HXTAL, RCU_CKOUT0_DIV1); #endif /* PHY_CLOCK_MCO */ gpio_ethernet_phy_select(GPIO_ENET_PHY_MII); #elif defined RMII_MODE rcu_pll2_config(RCU_PLL2_MUL10); rcu_osci_on(RCU_PLL2_CK); rcu_osci_stab_wait(RCU_PLL2_CK); /* get 50MHz from CK_PLL2 on CKOUT0 pin (PA8) to clock the PHY */ rcu_ckout0_config(RCU_CKOUT0SRC_CKPLL2, RCU_CKOUT0_DIV1); gpio_ethernet_phy_select(GPIO_ENET_PHY_RMII); #endif #ifdef MII_MODE /* PA0: ETH_MII_CRS */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_0); /* PA1: ETH_RX_CLK */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PA2: ETH_MDIO */ gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* PA3: ETH_MII_COL */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_3); /* PA7: ETH_MII_RX_DV */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_7); /* PC1: ETH_MDC */ gpio_init(GPIOC, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PC2: ETH_MII_TXD2 */ gpio_init(GPIOC, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* PC3: ETH_MII_TX_CLK */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_3); /* PC4: ETH_MII_RXD0 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_4); /* PC5: ETH_MII_RXD1 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_5); /* PB0: ETH_MII_RXD2 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_0); /* PB1: ETH_MII_RXD3 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PB8: ETH_MII_TXD3 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8); /* PB10: ETH_MII_RX_ER */ gpio_init(GPIOB, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_10); /* PB11: ETH_MII_TX_EN */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_11); /* PB12: ETH_MII_TXD0 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_12); /* PB13: ETH_MII_TXD1 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_13); #elif defined RMII_MODE /* PA1: ETH_RMII_REF_CLK */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PA2: ETH_MDIO */ gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_2); /* PA7: ETH_RMII_CRS_DV */ gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_7); /* PC1: ETH_MDC */ gpio_init(GPIOC, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_1); /* PC4: ETH_RMII_RXD0 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_4); /* PC5: ETH_RMII_RXD1 */ gpio_init(GPIOC, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_5); /* PB11: ETH_RMII_TX_EN */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_11); /* PB12: ETH_RMII_TXD0 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_12); /* PB13: ETH_RMII_TXD1 */ gpio_init(GPIOB, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_13); #endif /* MII_MODE */ } int rt_hw_gd32_eth_init(void) { rt_kprintf("rt_gd32_eth_init...\n"); /* enable ethernet clock */ rcu_periph_clock_enable(RCU_ENET); rcu_periph_clock_enable(RCU_ENETTX); rcu_periph_clock_enable(RCU_ENETRX); nvic_configuration(); /* configure the GPIO ports for ethernet pins */ enet_gpio_config(); /* set autonegotiation mode */ gd32_emac_device0.phy_mode = EMAC_PHY_AUTO; gd32_emac_device0.ETHERNET_MAC = ETHERNET_MAC0; gd32_emac_device0.ETHER_MAC_IRQ = ENET_IRQn; // OUI 00-00-0E FUJITSU LIMITED gd32_emac_device0.dev_addr[0] = 0x00; gd32_emac_device0.dev_addr[1] = 0x00; gd32_emac_device0.dev_addr[2] = 0x0E; /* set mac address: (only for test) */ gd32_emac_device0.dev_addr[3] = 0x12; gd32_emac_device0.dev_addr[4] = 0x34; gd32_emac_device0.dev_addr[5] = 0x56; gd32_emac_device0.parent.parent.init = gd32_emac_init; gd32_emac_device0.parent.parent.open = gd32_emac_open; gd32_emac_device0.parent.parent.close = gd32_emac_close; gd32_emac_device0.parent.parent.read = gd32_emac_read; gd32_emac_device0.parent.parent.write = gd32_emac_write; gd32_emac_device0.parent.parent.control = gd32_emac_control; gd32_emac_device0.parent.parent.user_data = RT_NULL; gd32_emac_device0.parent.eth_rx = gd32_emac_rx; gd32_emac_device0.parent.eth_tx = gd32_emac_tx; /* init tx buffer free semaphore */ rt_sem_init(&gd32_emac_device0.tx_buf_free, "tx_buf0", EMAC_TXBUFNB, RT_IPC_FLAG_FIFO); eth_device_init(&(gd32_emac_device0.parent), "e0"); /* change device link status */ eth_device_linkchange(&(gd32_emac_device0.parent), RT_TRUE); return 0; } INIT_DEVICE_EXPORT(rt_hw_gd32_eth_init); ``` 麻烦大佬有时间看一下
查看更多
0
个回答
默认排序
按发布时间排序
暂无答案,快来添加答案吧
撰写答案
登录
注册新账号
关注者
0
被浏览
577
关于作者
kuailedeluojie
这家伙很懒,什么也没写!
提问
44
回答
135
被采纳
0
关注TA
发私信
相关问题
1
SPI总线挂接2个W5500以太网芯片
2
用studio开发ETH设备时drv_eth.c不参与编译
3
STM32H743 以太网驱动 问题
4
关于在drv_eth.c中如何强制PHY工作
5
如何在rt-thread工作过程中切换IP地址获取模式(静态IP或者动态IP)
6
w5500_socket错误
7
STM32H743 LAN8720A
8
rt bthread studio不支持stm32H7系列的cpu 以太网驱动
9
studio生成stm32h7工程,drv_eth.c编译不过,如何解?
10
LWIP配置一直ping不同
推荐文章
1
RT-Thread应用项目汇总
2
玩转RT-Thread系列教程
3
国产MCU移植系列教程汇总,欢迎查看!
4
机器人操作系统 (ROS2) 和 RT-Thread 通信
5
五分钟玩转RT-Thread新社区
6
【技术三千问】之《玩转ART-Pi》,看这篇就够了!干货汇总
7
关于STM32H7开发板上使用SDIO接口驱动SD卡挂载文件系统的问题总结
8
STM32的“GPU”——DMA2D实例详解
9
RT-Thread隐藏的宝藏之completion
10
【ART-PI】RT-Thread 开启RTC 与 Alarm组件
最新文章
1
RT-Thread动态模块
2
RT-Thread项目助手v0.3 | Ubuntu与MacOS平台的RT-Thread Env
3
【FRA156测评DM-MCX】- 环境配置篇
4
【基于HPM6750+RW007的网页服务器】---SD卡文件系统
5
有关RT_thread studio使用WDT的一点经验
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
USB
DMA
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
rt-smart
RTC
FAL
I2C_IIC
ESP8266
UART
cubemx
WIZnet_W5500
ota在线升级
PWM
BSP
flash
freemodbus
packages_软件包
潘多拉开发板_Pandora
GD32
定时器
ADC
flashDB
编译报错
socket
中断
rt_mq_消息队列_msg_queue
Debug
keil_MDK
SFUD
msh
ulog
C++_cpp
MicroPython
本月问答贡献
lchnu
3
个答案
2
次被采纳
张世争
1
个答案
2
次被采纳
a1012112796
9
个答案
1
次被采纳
三世执戟
8
个答案
1
次被采纳
聚散无由
5
个答案
1
次被采纳
本月文章贡献
jinchanchan
9
篇文章
13
次点赞
ssdd45555
3
篇文章
2
次点赞
聚散无由
1
篇文章
4
次点赞
rvcore
1
篇文章
1
次点赞
lvdongchina
1
篇文章
1
次点赞
回到
顶部
发布
问题
分享
好友
手机
浏览
扫码手机浏览
投诉
建议
回到
底部