您的位置:首页 > 编程语言

jz2440开发板移植U-boot之修改代码支持DM9000网卡

2018-01-12 22:00 477 查看
今天我们来移植U-boot到jz2440开发板,修改代码支持DM9000网卡。查看之前写的移植记录请点击链接:点击查看之前的移植记录

现在大多数开发板都支持DM9000网卡。我们的U-boot源码里面也是有DM9000网卡的驱动程序的。文件为Dm9000x.c(drivers\net).

首先我去网卡目录的Makefile文件中搜索dm9000字符串:



由Makefile得知,如果我们定义了CONFIG_DRIVER_DM9000这个宏,那么dm9000网卡就会被编译进去(同时要去掉cs8900网卡)。

首先在smdk2440.h中找到下面的宏定义:

#define CONFIG_CS8900       /* we have a CS8900 on-board */
#define CONFIG_CS8900_BASE  0x19000300
#define CONFIG_CS8900_BUS16 /* the Linux driver does accesses as shorts */


这是cs8900网卡的定义,我们把它去掉,加上dm9000所需要的宏:

#ifdef 0
#define CONFIG_CS8900 /* we have a CS8900 on-board */ #define CONFIG_CS8900_BASE 0x19000300 #define CONFIG_CS8900_BUS16 /* the Linux driver does accesses as shorts */
#else
#define CONFIG_DRIVER_DM9000
#endif


重新编译uboot(肯定会显示有错误,不可能这么简单就弄好了,但是我们一步解决一个错误):

显示错误:

dm9000x.c:156: error: 'DM9000_DATA' undeclared (first use in this function)
dm9000x.c:156: error: (Each undeclared identifier is reported only once
dm9000x.c:156: error: for each function it appears in.)
。。。。
。。。。


显示说dm9000x.c中的56行没有定义DM9000_DATA这个宏参数。在经验不足,对驱动程序了解不多的情况下,我们没有办法知道这个宏参数是干什么的,但是我们可以仿照uboot中现有的程序中相同的地方模仿。我们在uboot源码中搜索DM9000_DATA这个字符串:

输入:grep “DM9000_DATA” * -nR (这是Linux下搜索字符串的命令)

显示很多配置文件中用到了DM9000_DATA这个参数,我们随便找一个配置文件进去看看别人是怎么配置的,假如找的是这个吧:

include/configs/at91sam9261ek.h:159:#define DM9000_DATA                 (CONFIG_DM9000_BASE + 4)


好,进去看了之后,别的是这么配置DM9000的网卡参数的:

#define CONFIG_DRIVER_DM9000
#define CONFIG_DM9000_BASE              0x04014000
#define DM9000_IO                       CONFIG_DM9000_BASE
#define DM9000_DATA                     (CONFIG_DM9000_BASE + 2)


原来是我们少定义了一些参数。那么我们来看看如何取值CONFIG_DM9000_BASE 与DM9000_DATA。

在分析之前先说一下:

DM9000属于内存类接口,想要操作内存类接口,需要知道它的1)访问地址,同时还要根据DM9000的实际情况2)设置内存控制器的时序以及位宽。

1)设置访问地址:

首先是 CONFIG_DM9000_BASE ,这个是DM9000的基地址,是CPU用来识别网卡的一个基地址。 查看DM9000网卡原理图:它的片选引脚是:nGCS4,那么我们查看2440手册,因为网卡是内存类的接口,所以我们看2440手册内存控制器那一章,找到片选信号的地址分配表:



从中可以看出:nGCS4这个片选引脚对应的初始地址是:0x20000000。那么我们就把CONFIG_DM9000_BASE这个值设为0x20000000,就会把nGCS4引脚设为低电平,从而片选到DM9000网卡设备。

然后是 DM9000_DATA的地址取值。查看原理图知,cpu只发出了LADDR2给DM9000网卡内的CMD引脚, 说明CPU只关心访问地址的第2位,所以我们把基地址加4,设置一下第二位的值:#define DM9000_DATA (CONFIG_DM9000_BASE + 4)

那么总的修改是:

#define CONFIG_DRIVER_DM9000
#define CONFIG_DM9000_BASE              0x20000000
#define DM9000_IO                       CONFIG_DM9000_BASE
#define DM9000_DATA                     (CONFIG_DM9000_BASE + 4)


那么我们已经设置好了上面说的第一步,设置访问地址,下面再看如何设置内存控制器的时序以及位宽。

2)设置内存控制器的时序以及位宽:

uboot中lowlevel_init.S中有设置内存控制器寄存器的相关的代码:

SMRDATA:
.long 0x22011110     //BWSCON
.long 0x00000700     //BANKCON0
.long 0x00000700     //BANKCON1
.long 0x00000700     //BANKCON2
.long 0x00000700     //BANKCON3
.long 0x00000700     //BANKCON4
.long 0x00000700     //BANKCON5
.long 0x00018005     //BANKCON6
.long 0x00018005     //BANKCON7
.long 0x008C04F4     // REFRESH
.long 0x000000B1     //BANKSIZE
.long 0x00000030     //MRSRB6
.long 0x00000030     //MRSRB7


.long 0x22011110 //BWSCON这个寄存器应该是设置的位宽,.long 0x00000700 //BANKCON4(nGCS4)这个应该是设置时序的寄存器

1)首先我们在2440手册中搜索BWSCON这个寄存器。在207页有设置位宽的寄存器位操作(其中19:18:17:16位对应的是BANK4,所以查看是否需要设置这四位):



DW4是设置位宽的的位,对应17:16位。我们计算.long 0x22011110 //BWSCON这个寄存器的值,用寄存器位查看器知:



17:16位对应的是01.那么查看手册知01 = 16-bit。查看原理图得知,DM9000网卡的位宽刚好是16位,所以,不需要修改了。18位的等待信号DM9000上没有,不需要设置。19位的也没有用到,不需要设置。

2)再看如何设置时序(设置BANKCON4寄存器)。找到如下图:



这里面就是设置DM9000的时序了,至于怎么设置(为什么这么设置),可以参考韦东山第二期视频移植DM9000网卡驱动程序那一节里面有详细讲过,这里不再讲解。

看看原始设置的各个位是对少:



对比之前韦东山二期视频讲解的DM9000时序设置,我们在这里将位7:6设置为01。则对应的寄存器的值为:



所以将BANKCON4的值改为0x00000740。

然后重新编译:

编译顺利通过。

将生成的镜像文件烧写到NOR FLASH,启动:



看到,显示Net: No ethernet found。那么我们去源码中搜索看是哪个地方。

源码中搜索:Net:

Board.c中:

#if defined(CONFIG_CMD_NET)
puts("Net:   ");
eth_initialize(gd->bd);


跳到:eth_initialize:

里面有一个函数board_eth_init:

我们找board_eth_init 函数(smdk2410.c,这个名字一直没有改成2440的,忘记了,暂时这么用吧,不影响):

int board_eth_init(bd_t *bis)
{
int rc = 0;
#ifdef CONFIG_CS8900
rc = cs8900_initialize(0, CONFIG_CS8900_BASE);
#endif
return rc;
}


发现没有初始化我们的DM9000网卡。我们在DM9000网卡驱动程序中找到:dm9000_initialize。应该是DM9000的初始化函数。我们在linux下搜索这个dm9000_initialize:grep “dm9000_initialize” * -nR,看看有没有类似的初始化。发现大多是这样的:

return dm9000_initialize(bis);


那么我们将上面的board_eth_init函数改成:

int board_eth_init(bd_t *bis)
{
int rc = 0;
#ifdef CONFIG_CS8900
rc = cs8900_initialize(0, CONFIG_CS8900_BASE);
#endif

#ifdef CONFIG_DRIVER_DM9000
rc = dm9000_initialize(bis);
#endif
return rc;
}


重新编译,启动:



OK啦,我们的网卡终于加好了。

现在ping一下电脑看看:

先设置ip:

set ipaddr 10.11.235.102

ping 10.11.235.146

显示

*** ERROR: `ethaddr' not set


没有设置mac地址,重启设置:

set ipaddr 192.168.1.102

set ethaddr 00:0c:29:4d:e4:f4 (可以随便设置)

ping 192.168.1.102

显示如下:



OK!显示主机alive。已经ping通了。下面我们试一下看看能不能用tftp下载。因为uboot没有集成TFTP工具,那么我们就用windows上的tftp工具进行下载内核。

如图为TFTP下载工具(注意服务器ip)



在板子上uboot上设置:

set serverip 192.168.1.102

tftp 30000000 uImage_4.3



然后启动内核:

bootm 30000000

成功启动内核。说明我们已经成功了!!!现在下载比用dnw下载要方便快捷的多。

想一起探讨以及获得各种学习资源加我(有我博客中写的代码的原稿):

qq:1126137994

微信:liu1126137994

可以共同交流关于嵌入式,操作系统,C++语言,C语言,数据结构等技术问题
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息