u-boot2010.03 移植篇(四)-----支持DM9000,实现tftp下载

2024-02-26 02:58

本文主要是介绍u-boot2010.03 移植篇(四)-----支持DM9000,实现tftp下载,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

----------------------------------------------------------
使用环境 
PC:     ubuntu 11.04
kernel: 2.6.32-28-generic
corss:  arm-linux-gcc 4.3.2
arm:    s3c6410
uboot:  uboot-2010-03
----------------------------------------------------------
作者:    LvApp
联系方式: 97164811@qq.com
一切版权均有作者所有,欢迎转载,请指明出处,如何修改请与本人联系,谢谢

哇塞..tftp下载,很多6410买的开发板梦寐以求需要的功能,为嘛?当然为了方便实现系统更新.一键下载之类的..想想都觉得兴奋..那就开始吧

首先.前阵子本人比较忙..所以一直没有更新.再加上.最近在看别的资料.所以在写这些内容的时候,我还得自己实战下.保证写出来没有错误吧..

这篇内容不多.因为uboot已经做的很好了..几乎不需要修改太多的东西,就能支持dm9000这款用的比较多的网卡了..

这篇文章其实有稍许超前了.因为对应的分析贴还没写好.所以这里我会适当加上一些理论补充,比如:我为什么要加这句代码到这个地方?类似于这样的问题

根据smdk6400 copy 过来的那么默认用的就是cs8900这块网卡,可以从configs/smdk6410.h这个文件中看到....这里当然不需要了..所以,果断先把关于8900的宏

全都去掉..如下:


像这样.去掉先...接着就是添加DM9000相关的宏了...这里我也贴出来..

[cpp] plain copy
  1. #define CONFIG_DM9000_NO_SROM           //此处说明,该网卡没有eeprom存储..其实dm9000是有的,只是没用,所以只好说没有了  
  2. #define CONFIG_DM9000                   //定义DM9000 回头初始化驱动,就判断这个宏了  
  3. #define CONFIG_DRIVER_DM9000            //开启DM9000驱动的代码  
  4. #define CONFIG_DM9000_BASE      0x18800300                    //这个宏是定义网卡连接到你的arm上面的地址.属于sram1地址区域  
  5. #define DM9000_IO           CONFIG_DM9000_BASE            //根据网卡手册,指定I/O DATA操作地址  
  6. #define DM9000_DATA         (CONFIG_DM9000_BASE + 0x04)  
  7. #define CONFIG_DM9000_USE_16BIT         //指明使用16位数据模式,该宏其实是为了方便知道网卡信息而已,在代码中没有半毛钱关系...网卡的16位选择,  
  8.                                         //是在内存设置的时候设置的...也就是说设定sram1内存寄存器  
  9.   
  10.   
  11. #define CONFIG_ETHADDR      00:40:5C:26:0A:5B        //这里定义一些网络相关参数,这些以后也可以直接手动设定环境变量.  
  12. #define CONFIG_NETMASK      255.255.255.0            //放在这里也是比较方面而已  
  13. #define CONFIG_IPADDR       192.168.1.123  
  14. #define CONFIG_SERVERIP     192.168.1.100  
  15. #define CONFIG_GATEWAYIP    192.168.1.1  
  16.   
  17. #define CONFIG_DM9000_DEBUG             //开启调试  

ok 网卡的配置就好了...非常简单..就是开启一些宏和设定一些相关的参数而已...

下面去添加dm9000的初始化,不然不能用,那么初始化放哪呢?网卡初始化的地方究竟是吧....在第一部分的时候分析.没有发现关于网卡初始化的地方...那么也就是说.网卡

的初始化其实是在第二部分里面完成的?没错....直接定位到start_armboot第二部分代码入口...

[cpp] plain copy
  1. #if defined(CONFIG_CMD_NET)  
  2. #if defined(CONFIG_NET_MULTI)  
  3.     puts ("Net:   ");  
  4. #endif  
  5.     eth_initialize(gd->bd);      /* 这里就是网卡的初始化函数了 */  
  6. #if defined(CONFIG_RESET_PHY_R)  
  7.     debug ("Reset Ethernet PHY\n");  
  8.     reset_phy();  
  9. #endif  

直接定位到 eth_initialize函数中...这个函数蛮长的.....
[cpp] plain copy
  1. int eth_initialize(bd_t *bis)  
  2. {  
  3.     unsigned char env_enetaddr[6];  
  4.     int eth_number = 0;  
  5.     eth_devices = NULL;  
  6.     eth_current = NULL;  
  7.   
  8.     show_boot_progress (64);  
  9. #if defined(CONFIG_MII) || defined(CONFIG_CMD_MII)  
  10.     miiphy_init();  
  11. #endif  
  12.     /* Try board-specific initialization first.  If it fails or isn't 
  13.      * present, try the cpu-specific initialization */  
  14.     if (board_eth_init(bis) < 0)        /* 注意这里有个板级初始化调用...board_eth_init(bis) */  
  15.         cpu_eth_init(bis);  
  16.   
  17. #if defined(CONFIG_DB64360) || defined(CONFIG_CPCI750)  
  18.     mv6436x_eth_initialize(bis);  
  19. #endif  
  20. #if defined(CONFIG_DB64460) || defined(CONFIG_P3Mx)  
  21.     mv6446x_eth_initialize(bis);  
  22. #endif  
  23.     if (!eth_devices) {  
  24.         puts ("No ethernet found.\n");  
  25.         show_boot_progress (-64);  
  26.     } else {  
  27.         struct eth_device *dev = eth_devices;  
  28.         char *ethprime = getenv ("ethprime");  
  29.         show_boot_progress (65);  
  30.         do {  
  31.             if (eth_number)  
  32.                 puts (", ");  
  33.             printf("%s", dev->name);  
  34.             if (ethprime && strcmp (dev->name, ethprime) == 0) {  
  35.                 eth_current = dev;  
  36.                 puts (" [PRIME]");  
  37.             }  
  38.             eth_getenv_enetaddr_by_index(eth_number, env_enetaddr);/* 获取网卡ethaddr的MAC地址 */  
  39.             if (memcmp(env_enetaddr, "\0\0\0\0\0\0", 6)) {  
  40.                 if (memcmp(dev->enetaddr, "\0\0\0\0\0\0", 6) &&  
  41.                     memcmp(dev->enetaddr, env_enetaddr, 6))  
  42.                 {  
  43.                     printf ("\nWarning: %s MAC addresses don't match:\n",  
  44.                         dev->name);  
  45.                     printf ("Address in SROM is         %pM\n",  
  46.                         dev->enetaddr);  
  47.                     printf ("Address in environment is  %pM\n",  
  48.                         env_enetaddr);  
  49.                 }  
  50.                 memcpy(dev->enetaddr, env_enetaddr, 6);  
  51.             }  
  52.             eth_number++;  
  53.             dev = dev->next;  
  54.         } while(dev != eth_devices);  
  55. #ifdef CONFIG_NET_MULTI  
  56.         /* update current ethernet name */  
  57.         if (eth_current) {  
  58.             char *act = getenv("ethact");  
  59.             if (act == NULL || strcmp(act, eth_current->name) != 0)  
  60.                 setenv("ethact", eth_current->name);  
  61.         } else  
  62.             setenv("ethact", NULL);  
  63. #endif  
  64.         putc ('\n');  
  65.     }  
  66.     return eth_number;  
  67. }  
在上面的代码中看到 board_eth_init 函数的调用,该函数是跟板级相关的网卡初始化,那么肯定在board\samsung\smdk6410\smdk6410.c中了...打开看看..果然发现这么一个接口函数
[cpp] plain copy
  1. #ifdef CONFIG_CMD_NET  
  2. int board_eth_init(bd_t *bis)  
  3. {  
  4.     int rc = 0;  
  5. #ifdef CONFIG_CS8900  
  6.     rc = cs8900_initialize(0, CONFIG_CS8900_BASE);  
  7. #endif  
  8.     return rc;  
  9. }  
  10. #endif  
而且还发现这里面有8900网卡的初始化..那应该没错了...我们就模仿它添加我们的DM9000网卡..如下:
[cpp] plain copy
  1. #ifdef CONFIG_CMD_NET  
  2. int board_eth_init(bd_t *bis)  
  3. {  
  4.     int rc = 0;  
  5. #ifdef CONFIG_CS8900  
  6.     rc = cs8900_initialize(0, CONFIG_CS8900_BASE);  
  7. #endif  
  8. #if defined(CONFIG_DM9000)  
  9.     rc = dm9000_initialize(bis);  
  10. #endif  
  11.     return rc;  
  12. }  
  13. #endif  

其实我想说,..修改好了...完工了.....真的好了....


根据添加.发现网卡也有了...网卡的一些环境变量也有了.....

下面是我用tftp功能下载uImage的截图



最后要说下..ping命令不好使的问题...ping会发出arp获得对方ip的网卡信息..接着在此发送icmp包.才能得到ping的结果...但是根据网卡调试信息

发现发送了arp之后,也获得了mac.但是没有发送icmp包...正在探查该问题所在...有人说修改timeout时间...大家可以试下...因为该bug影响不大.

所以没有深究...

今天就到了...谢谢

Finish!
Thanks a lot~


这篇关于u-boot2010.03 移植篇(四)-----支持DM9000,实现tftp下载的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/747521

相关文章

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

PyCharm中配置PyQt的实现步骤

《PyCharm中配置PyQt的实现步骤》PyCharm是JetBrains推出的一款强大的PythonIDE,结合PyQt可以进行pythion高效开发桌面GUI应用程序,本文就来介绍一下PyCha... 目录1. 安装China编程PyQt1.PyQt 核心组件2. 基础 PyQt 应用程序结构3. 使用 Q

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库