追踪uboot下tftp命令的代码执行过程-Nagul

2024-09-03 15:32

本文主要是介绍追踪uboot下tftp命令的代码执行过程-Nagul,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、网卡驱动的添加 

网络在uboot中的启动是在uboot的第二阶段启动代码中 /lib_arm/board.c

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void start_armboot (void){}  
里面有网络初始化函数
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. eth_initialize(gd->bd);  
进入函数你会发现一系列的网卡初始化函数

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #if defined(CONFIG_MCFFEC)  
  2.     mcffec_initialize(bis);  
  3. #endif  
  4. #if defined(CONFIG_FSLDMAFEC)  
  5.     mcdmafec_initialize(bis);  
  6. #endif  
  7. #if defined(CONFIG_AT91CAP9) || defined(CONFIG_AT91SAM9260) || \  
  8.     defined(CONFIG_AT91SAM9263)  
  9.     at91sam9_eth_initialize(bis);  
  10. #endif  
  11.   
  12. #if defined(CONFIG_DRIVER_CS8900)  
  13.     cs8900_initialize(bis);  
  14. #endif  
  15.   
  16. #if defined(CONFIG_DRIVER_SMC911X)  
  17.     smc911x_initialize(bis);  
  18. #endif  

我的是smc9220的网卡,所以找到

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #if defined(CONFIG_DRIVER_SMC911X)  
  2.  smc911x_initialize(bis);  
  3. #endif  

那么需要添加这个网卡就需要定义前面的宏 CONFIG_DRIVER_SMC911X 
这个宏在/configs/unsp210.h 中定义就行了(不同型号的网卡,在移植的时候定义不同的宏就ok了)

继续跟进

smc911x_initialize(bis)每种网卡里面都会有这么一个函数,想配置不同的网卡时,在这个函数中进行注册就行了(其实不是注册,只是把设备放到了一个链表中,便于管理)
smc911x_initialize(bis)函数中关键在于这里
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. dev = (struct eth_device *) malloc (sizeof *dev);  
  2. memcpy(dev->enetaddr, bis->bi_enetaddr, 6);  
  3. sprintf(dev->name, DRIVERNAME); //设备名  
  4. dev->priv = (void *)NULL; /* this have to come before bus_to_phys() */  
  5. dev->iobase = CONFIG_DRIVER_SMC911X_BASE;  
  6. dev->init = smc911x_eth_init;    //对应底层的初始化  
  7. dev->halt = smc911x_eth_halt;  
  8. dev->send = smc911x_eth_send;     //对应底层的发生函数  
  9. dev->recv = smc911x_eth_rx;    //对应底层的接收函数  
  10. eth_register (dev); //类似于内核中注册设备一样的方法,把设备添加到链表中  
eth_register ()函数中有这么一句话
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. eth_current = eth_devices = dev;  
到了这里,我们就可以使用 eth_devices和eth_current 通信了(具体底层的函数,可在u-boot\drivers\net\目录下看到相应的.c文件,这里不再跟进了)
跟到这里,大概知道网卡驱动是如何与上面关联起来的了,那么接下来就看看TFTP命令执行的过程

二、uboot下网络命令的实现过程
那么我们在UBOOT中用的最多的命令就是update 或者 tftp 通过网络下载内核和根文件系统了
这两个命令又是如何调用网络呢?
1、首先tftp命令在uboot下的实现过程    
common/cmd_net.c中有定义
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. int do_tftpb (cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])  
  2. {  
  3.  return netboot_common (TFTP, cmdtp, argc, argv);  
  4. }  
  5.   
  6. U_BOOT_CMD(  
  7.  tftpboot,  3,  1,  do_tftpb,  
  8.  "tftpboot- boot image via network using TFTP protocol\n",  
  9.  "[loadAddress] [[hostIPaddr:]bootfilename]\n"  
  10. );  
U_BOOT_CMD就是使第一个参数的命令tftpboot执行对应的函数do_tftpb()
do_tftpb() 中netboot_common()函数中 使用NetLoop(proto)来启动tftp
NetLoop(proto_t protocol)函数在/net/net.c中
下面有根据参数协议类型来执行相应的函数,从上面的代码可以看到传递进来的是TFTP,所以会执行TFTP段代码
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. switch (protocol) {  
  2.   case TFTP:  
  3.    /* always use ARP to get server ethernet address */  
  4.    TftpStart();  
  5.    break;  
  6. 。。。  
  7. }  
接下来就是TftpStart()的事情了

2、tftp命令执行后,最先运行的函数是/net/tftp.c中TftpStart() 函数中关键就两个地方,一个收一个发
NetSetTimeout (TIMEOUT * CFG_HZ, TftpTimeout);
NetSetHandler (TftpHandler);
这两个函数就解决了收的问题了,TftpHandler函数在有数据的时候会自动的执行,毫无疑问TftpHandler里面就是接收处理函数了
最后一句TftpSend ();
发送数据是使用UDP的函数,一路跟下去你会发现最后使用的原始数据发送的(自己实现的UDP包头IP包头以太网包头)
NetSendUDPPacket(NetServerEther, TftpServerIP, TftpServerPort, TftpOurPort, len);
如果想看明白这些个数据包是如何组成的,那么必须先看TCP/IP协议卷1吧

跟代码也这么累, 先跟到这里了,不过基本上已经能够屡清楚TFTP命令的执行过程了。

这篇关于追踪uboot下tftp命令的代码执行过程-Nagul的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Redis 的 SUBSCRIBE命令详解

《Redis的SUBSCRIBE命令详解》Redis的SUBSCRIBE命令用于订阅一个或多个频道,以便接收发送到这些频道的消息,本文给大家介绍Redis的SUBSCRIBE命令,感兴趣的朋友跟随... 目录基本语法工作原理示例消息格式相关命令python 示例Redis 的 SUBSCRIBE 命令用于订

防止Linux rm命令误操作的多场景防护方案与实践

《防止Linuxrm命令误操作的多场景防护方案与实践》在Linux系统中,rm命令是删除文件和目录的高效工具,但一旦误操作,如执行rm-rf/或rm-rf/*,极易导致系统数据灾难,本文针对不同场景... 目录引言理解 rm 命令及误操作风险rm 命令基础常见误操作案例防护方案使用 rm编程 别名及安全删除

oracle 11g导入\导出(expdp impdp)之导入过程

《oracle11g导入导出(expdpimpdp)之导入过程》导出需使用SEC.DMP格式,无分号;建立expdir目录(E:/exp)并确保存在;导入在cmd下执行,需sys用户权限;若需修... 目录准备文件导入(impdp)1、建立directory2、导入语句 3、更改密码总结上一个环节,我们讲了

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

ShardingProxy读写分离之原理、配置与实践过程

《ShardingProxy读写分离之原理、配置与实践过程》ShardingProxy是ApacheShardingSphere的数据库中间件,通过三层架构实现读写分离,解决高并发场景下数据库性能瓶... 目录一、ShardingProxy技术定位与读写分离核心价值1.1 技术定位1.2 读写分离核心价值二

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本

Java Kafka消费者实现过程

《JavaKafka消费者实现过程》Kafka消费者通过KafkaConsumer类实现,核心机制包括偏移量管理、消费者组协调、批量拉取消息及多线程处理,手动提交offset确保数据可靠性,自动提交... 目录基础KafkaConsumer类分析关键代码与核心算法2.1 订阅与分区分配2.2 拉取消息2.3

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

MySQ中出现幻读问题的解决过程

《MySQ中出现幻读问题的解决过程》文章解析MySQLInnoDB通过MVCC与间隙锁机制在可重复读隔离级别下解决幻读,确保事务一致性,同时指出性能影响及乐观锁等替代方案,帮助开发者优化数据库应用... 目录一、幻读的准确定义与核心特征幻读 vs 不可重复读二、mysql隔离级别深度解析各隔离级别的实现差异