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

时间:2021-11-22 05:12:07

一、准备工作

1.1 原理图

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

  CONFIG_DM9000_BASE

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

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

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

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

  CONFIG_DM9000_IO 和CONFIG_DM9000_DATA

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

  LADDR2 对应 100,即为4。

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

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

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

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

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

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

1.2 修改参数

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

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

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

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

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

  改为:

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

  编译运行一次:

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

  运行成功,烧写:

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

二、代码修改

  搜索:No ethernet found.

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

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

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

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

  进入查看:

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

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

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

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

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

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

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

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

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

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

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

  编译运行:

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

  报告有错误:dm9000 address not set

  MAC地址未设置。

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

 int eth_write_hwaddr(struct eth_device *dev, const char *base_name,
int eth_number)
{
unsigned char env_enetaddr[];
int ret = ; eth_getenv_enetaddr_by_index(base_name, eth_number, env_enetaddr); if (!is_zero_ethaddr(env_enetaddr)) {
if (!is_zero_ethaddr(dev->enetaddr) &&
memcmp(dev->enetaddr, env_enetaddr, )) {
printf("\nWarning: %s MAC addresses don't match:\n",
dev->name);
printf("Address in SROM is %pM\n",
dev->enetaddr);
printf("Address in environment is %pM\n",
env_enetaddr);
} memcpy(dev->enetaddr, env_enetaddr, );
} else if (is_valid_ethaddr(dev->enetaddr)) {
eth_setenv_enetaddr_by_index(base_name, eth_number,
dev->enetaddr);
} else if (is_zero_ethaddr(dev->enetaddr)) {
#ifdef CONFIG_NET_RANDOM_ETHADDR
net_random_ethaddr(dev->enetaddr);
printf("\nWarning: %s (eth%d) using random MAC address - %pM\n",
dev->name, eth_number, dev->enetaddr);
#else
printf("\nError: %s address not set.\n",
dev->name);
return -EINVAL;
#endif
} if (dev->write_hwaddr && !eth_mac_skip(eth_number)) {
if (!is_valid_ethaddr(dev->enetaddr)) {
printf("\nError: %s address %pM illegal value\n",
dev->name, dev->enetaddr);
return -EINVAL;
} ret = dev->write_hwaddr(dev);
if (ret)
printf("\nWarning: %s failed to set MAC address\n",
dev->name);
} return ret;
}

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

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

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

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

  在这里加入下面的内容:

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

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

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

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

  保存编译,查看结果:

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

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

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

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

  已经ping 通 网卡移植完成。