追踪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

相关文章

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

作业提交过程之HDFSMapReduce

作业提交全过程详解 (1)作业提交 第1步:Client调用job.waitForCompletion方法,向整个集群提交MapReduce作业。 第2步:Client向RM申请一个作业id。 第3步:RM给Client返回该job资源的提交路径和作业id。 第4步:Client提交jar包、切片信息和配置文件到指定的资源提交路径。 第5步:Client提交完资源后,向RM申请运行MrAp

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

30常用 Maven 命令

Maven 是一个强大的项目管理和构建工具,它广泛用于 Java 项目的依赖管理、构建流程和插件集成。Maven 的命令行工具提供了大量的命令来帮助开发人员管理项目的生命周期、依赖和插件。以下是 常用 Maven 命令的使用场景及其详细解释。 1. mvn clean 使用场景:清理项目的生成目录,通常用于删除项目中自动生成的文件(如 target/ 目录)。共性规律:清理操作

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

Solr 使用Facet分组过程中与分词的矛盾解决办法

对于一般查询而言  ,  分词和存储都是必要的  .  比如  CPU  类型  ”Intel  酷睿  2  双核  P7570”,  拆分成  ”Intel”,”  酷睿  ”,”P7570”  这样一些关键字并分别索引  ,  可能提供更好的搜索体验  .  但是如果将  CPU  作为 Facet  字段  ,  最好不进行分词  .  这样就造成了矛盾  ,  解决方法

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者

linux 判断某个命令是否安装

linux 判断某个命令是否安装 if ! [ -x "$(command -v git)" ]; thenecho 'Error: git is not installed.' >&2exit 1fi

jenkins 插件执行shell命令时,提示“Command not found”处理方法

首先提示找不到“Command not found,可能我们第一反应是查看目标机器是否已支持该命令,不过如果相信能找到这里来的朋友估计遇到的跟我一样,其实目标机器是没有问题的通过一些远程工具执行shell命令是可以执行。奇怪的就是通过jenkinsSSH插件无法执行,经一番折腾各种搜索发现是jenkins没有加载/etc/profile导致。 【解决办法】: 需要在jenkins调用shell脚

Python:豆瓣电影商业数据分析-爬取全数据【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】

**爬取豆瓣电影信息,分析近年电影行业的发展情况** 本文是完整的数据分析展现,代码有完整版,包含豆瓣电影爬取的具体方式【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】   最近MBA在学习《商业数据分析》,大实训作业给了数据要进行数据分析,所以先拿豆瓣电影练练手,网络上爬取豆瓣电影TOP250较多,但对于豆瓣电影全数据的爬取教程很少,所以我自己做一版。 目