u-boot移植(十二)---代码修改---支持DM9000网卡

一、准备工作

1.1 原理图

  

  CONFIG_DM9000_BASE

  片选信号是接在nGCS4引脚,若要确定网卡的基地址,则要根据片选信号的接口去确定。

  在三星2440的DATASHEET中memory control这一章的Figure 5-1. S3C2440A Memory Map after Reset 已经说明了片选4的地址,如下:

  

  只要发出的信号在 0x20000000--0x28000000 之间,就会使得片选4引脚变为低电平。所以可以确定我们网卡的基地址为0x20000000。

  CONFIG_DM9000_IO 和CONFIG_DM9000_DATA

   LADDR2 接在网卡的CMD引脚上,即CPU在LADDR2上发出高电平或低电平的时候可以去访问某个地址。这些地址称为IO地址和DATA地址。

  LADDR2 对应 100,即为4。

  之后就是位宽要确定,根据原理图,可以确定的是DM9000为16位的位宽(LDATA0~LADATA15)。那么要设置BWSCON寄存器:

  

  DW4应该置1。这一项可以更改 low_level_init 中的寄存器的值进行设置。

  再检查时序,时序在BANKCON4寄存器中设置,默认值也可以,寄存器也在low_level_init 中的寄存器的值进行设置。

  修改下Toch的值,改为2个时钟的保持时间。

  

1.2 修改参数

  u-boot 中已经有了DM9000的驱动,即dm9000x.c,现在要将DM9000的网卡驱动加进u-boot中进行编译。

  在drivers/net/Makefile文件中已经加进了dm9000x.c的编译文件,如下:

  

  现在要修改include/configs/jz2440.h文件,来支持dm9000x:

  

  改为:

  

  编译运行一次:

  

  运行成功,烧写:

  

二、代码修改

  搜索:No ethernet found.

  

  这里涉及到两个函数 eth_initialize 和 eth_init

    查看board_r.c中的init_sequence_r链表,找到网路的初始化程序。

  

  进入查看:

  

  这里调用的是 eth_initialize()函数。

  

  根据打印信息可以确定是调用的下面的那个。

  

  由代码可以看出  应该是在 eth_common_init()函数中初始化失败导致eth_devices初始化失败的。

  

  代码中调用了board_eth_init函数Jz2440.c (board\samsung\jz2440):

  

  这里只定义了 CONFIG_CS8900 并没有9000网卡,修改代码:

  

  编译运行:

  

  报告有错误:dm9000 address not set

  MAC地址未设置。

  对着打印信息查看代码,可以知道打印此条信息的代码是eth_write_hwaddr这个函数,这个函数在eth_initlize中调用:

 1 int eth_write_hwaddr(struct eth_device *dev, const char *base_name,
 2            int eth_number)
 3 {
 4     unsigned char env_enetaddr[6];
 5     int ret = 0;
 6 
 7     eth_getenv_enetaddr_by_index(base_name, eth_number, env_enetaddr);
 8 
 9     if (!is_zero_ethaddr(env_enetaddr)) {
10         if (!is_zero_ethaddr(dev->enetaddr) &&
11             memcmp(dev->enetaddr, env_enetaddr, 6)) {
12             printf("\nWarning: %s MAC addresses don't match:\n",
13                    dev->name);
14             printf("Address in SROM is         %pM\n",
15                    dev->enetaddr);
16             printf("Address in environment is  %pM\n",
17                    env_enetaddr);
18         }
19 
20         memcpy(dev->enetaddr, env_enetaddr, 6);
21     } else if (is_valid_ethaddr(dev->enetaddr)) {
22         eth_setenv_enetaddr_by_index(base_name, eth_number,
23                          dev->enetaddr);
24     } else if (is_zero_ethaddr(dev->enetaddr)) {
25 #ifdef CONFIG_NET_RANDOM_ETHADDR
26         net_random_ethaddr(dev->enetaddr);
27         printf("\nWarning: %s (eth%d) using random MAC address - %pM\n",
28                dev->name, eth_number, dev->enetaddr);
29 #else
30         printf("\nError: %s address not set.\n",
31                dev->name);
32         return -EINVAL;
33 #endif
34     }
35 
36     if (dev->write_hwaddr && !eth_mac_skip(eth_number)) {
37         if (!is_valid_ethaddr(dev->enetaddr)) {
38             printf("\nError: %s address %pM illegal value\n",
39                    dev->name, dev->enetaddr);
40             return -EINVAL;
41         }
42 
43         ret = dev->write_hwaddr(dev);
44         if (ret)
45             printf("\nWarning: %s failed to set MAC address\n",
46                    dev->name);
47     }
48 
49     return ret;
50 }

  上面定义了一个宏  CONFIG_NET_RANDOM_ETHADDR ,如果定义了此宏的话就会随机分配网卡物理地址,否则就打印错误信息,我们并不需要此宏。自己定义物理地址,从代码流程看,网卡地址是直接写进环境变量中的,然后再读取环境变量,这个时候就需要看看设置环境变量的地方了。

  环境变量的设置在 board_r.c的链表中,initr_env函数,initr_env 会调用 env_relocate() ,env_relocate() 调用 set_default_env 函数,set_default_env 函数中有一个结构体 default_environment ,这里面定义了默认的参数,进去看看一看就知道里面全部定义的是默认的环境变量参数,其中也有网络的:

  

  这里面没有物理地址的定义,所以我们可以自己定义物理地址:

  在这里加入下面的内容:

  

  在 include/configs/jz2440.h 中加入 CONFIG_ETHADDR的宏 ,这里我们可以根据自己本机上的MAC地址进行定义:

  

  

  保存编译,查看结果:

  

  已经不会再报错网卡了。进行测试看通讯是否正常。

  

  

  已经ping 通 网卡移植完成。

 

 

  

 

  

posted @ 2017-06-20 19:38  游戏进行中  阅读(1536)  评论(0编辑  收藏  举报