1. 在两个工程的基础上修改,分别是正点原子ATK的 “实验55 网络通信实验” 和ST官方的“STSW-STM32070”例程
硬件是原子的explorer和SeerF4kernelV2。
两个例程的问题在于ATK例程用的外部SRAM,注释掉外部SRAM后跑不起来。而ST官方例程用的芯片是DP83848,需要手动修改为LAN8720A。
针对ST的例程,网上有对应的文档:ST 中stsw-stm32070 的网络例子——将DP83848CVV 修改为 LAN8720A
目前按照这个文档对ST例程进行修改。
2.继续在ATK的板子上尝试,关键点在于ATK的例程里自己做了内存管理。所以要放弃SRAM的使用的话,先要把内存管理去掉,把外部FSMC关掉。
在lwip的mem.c和memp.c中,按照lwip自己的方式,申请了ram_heap和memp_memory。而ATK的例程里注释掉了这两个地方,在lwip_comm.c中,从自己的内存池申请了内存分配给这两个单元。因此首先把这两个地方复原。
3.另一个自行管理的内存块在LAN8720.c中的ETH模块DMA缓冲区。
__align(4) ETH_DMADESCTypeDef *DMARxDscrTab; //以太网DMA接收描述符数据结构体指针
__align(4) ETH_DMADESCTypeDef *DMATxDscrTab; //以太网DMA发送描述符数据结构体指针
__align(4) uint8_t *Rx_Buff; //以太网底层驱动接收buffers指针
__align(4) uint8_t *Tx_Buff; //以太网底层驱动发送buffers指针
__align(4) ETH_DMADESCTypeDef DMARxDscrTab[ETH_RXBUFNB]; //以太网DMA接收描述符数据结构体指针
__align(4) ETH_DMADESCTypeDef DMATxDscrTab[ETH_TXBUFNB]; //以太网DMA发送描述符数据结构体指针
__align(4) uint8_t Rx_Buff[ETH_RX_BUF_SIZE*ETH_RXBUFNB]; //以太网底层驱动接收buffers指针
__align(4) uint8_t Tx_Buff[ETH_TX_BUF_SIZE*ETH_TXBUFNB]; //以太网底层驱动发送buffers指针
4. 完成了上面的操作后,可以把lwip_comm.c中所有的xxx_malloc()和xxx_free()等函数注释成空函数。
测试程序,仍然能够ping通网卡。至此已经还原Lwip中所有内存管理部分。
5. 在main.c中,将FSMC_SRAM_Init()注释掉,发现ping不通网卡了。
重新把函数放出来。
进入FSMC_SRAM_Init(),注释掉最后的
// FSMC_NORSRAMInit(&FSMC_NORSRAMInitStructure); //初始化FSMC配置
// FSMC_NORSRAMCmd(FSMC_Bank1_NORSRAM3, ENABLE); // 使能BANK1区域3
发现仍然能够ping通,说明问题在上边。
仔细看发现上面有:
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB|RCC_AHB1Periph_GPIOD|RCC_AHB1Periph_GPIOE|RCC_AHB1Periph_GPIOF|RCC_AHB1Periph_GPIOG, ENABLE);//使能PD,PE,PF,PG时钟
即一些IO时钟的初始化在FSMC_SRAM_Init()中做了,其他组件在进行操作的时候可能没有再重新初始化。
把这句单独拉出来放到main.c中,然后注释掉FSMC_SRAM_Init(),能够成功运行。
可以看到,函数之间的耦合会给后期调试带来麻烦。
至此,一个独立的LAN8720的测试代码已经整理出来。
====2018/3/2 更新=====
经查询,在LAN8720.c, LAN8720_Init()函数中,初始化APB1时钟时,只初始化了GPIOA, GPIOC, GPIOG。
而LAN8720的复位引脚为PD3,没有被初始化。因此加入GPIOD的时钟初始化即可。
u8 ETH_MACDMA_Config(void)
函数中的:
ETH_DeInit(); //AHB总线重启以太网
ETH_SoftwareReset(); //软件重启网络
while (ETH_GetSoftwareResetStatus() == SET);//等待软件重启网络完成
程序一直卡在while循环中。
u8 LAN8720_Init(void)
中,讲IO初始化的代码注释掉一部分:
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_7;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;
GPIO_Init(GPIOA, &GPIO_InitStructure);
在ATK的板子上模拟连接故障的情况。将代码刷入ATK的板子,程序也卡在了while循环处。
do
{
timeout++;
tmpreg = ETH->MACMIIAR;
} while ((tmpreg & ETH_MACMIIAR_MB) && (timeout < (uint32_t)PHY_READ_TO));