Linux-2.6.32.2內(nèi)核在mini2440上的移植(三)---DM9000網(wǎng)卡驅(qū)動(dòng)移植
掃描二維碼
隨時(shí)隨地手機(jī)看文章
移植環(huán)境
1,主機(jī)環(huán)境:VMare下CentOS 5.5 ,1G內(nèi)存。
2,集成開發(fā)環(huán)境:Elipse IDE
3,編譯編譯環(huán)境:arm-linux-gcc v4.4.3,arm-none-linux-gnueabi-gcc v4.5.1。
4,開發(fā)板:mini2440,2M nor flash,128M nand flash。
5,u-boot版本:u-boot-2009.08
6,linux 版本:linux-2.6.32.2
7,參考文章:
【1】嵌入式linux應(yīng)用開發(fā)完全手冊(cè),韋東山,編著。
【2】Mini2440 之Linux 移植開發(fā)實(shí)戰(zhàn)指南
【3】http://linux.chinaunix.net/techdoc/system/2009/08/24/1131864.shtml
3.1,移植DM9000 網(wǎng)卡驅(qū)動(dòng)
【1】設(shè)備資源初始化
Linux-2..6.32.2 已經(jīng)自帶了完善的DM9000 網(wǎng)卡驅(qū)動(dòng)驅(qū)動(dòng)(源代碼位置:linux-2.6.32.2/drivers/net/dm9000.c),它也是一個(gè)平臺(tái)設(shè)備,因此在目標(biāo)平臺(tái)初始化代碼中,只要填寫好相應(yīng)的結(jié)構(gòu)表即可,具體步驟如下:
(1)確認(rèn)已經(jīng)添加了驅(qū)動(dòng)所需的頭文件 dm9000.h:
用gedit打開linux-2.6.32.2/arch/arm/mach-mini2440.c,定位到55行附近,加入dm9000.h,如下所示:
#include
#include
#include
#include
#include
#include
#include
(2)填充該平臺(tái)設(shè)備的資源設(shè)置
定位到210行附近,如入下面代碼:
tatic struct s3c2410_platform_nand mini2440_nand_info = {
.tacls= 20,
.twrph0= 60,
.twrph1= 20,
.nr_sets= ARRAY_SIZE(mini2440_nand_sets),
.sets= mini2440_nand_sets,
.ignore_unset_ecc = 1,
};
/* DM9000AEP 10/100 ethernet controller *///定義DM9000 網(wǎng)卡設(shè)備的物理基地址,以便后面用到
#define MACH_MINI2440_DM9K_BASE (S3C2410_CS4 + 0x300)
//再填充該平臺(tái)設(shè)備的資源設(shè)置,以便和 DM9000 網(wǎng)卡驅(qū)動(dòng)接口配合起來(lái)
static struct resource mini2440_dm9k_resource[] = {
[0] = {
.start = MACH_MINI2440_DM9K_BASE,
.end = MACH_MINI2440_DM9K_BASE + 3,
.flags = IORESOURCE_MEM
},
[1] = {
.start = MACH_MINI2440_DM9K_BASE + 4,
.end = MACH_MINI2440_DM9K_BASE + 7,
.flags = IORESOURCE_MEM
},
[2] = {
.start = IRQ_EINT7,
.end = IRQ_EINT7,
.flags = IORESOURCE_IRQ | IORESOURCE_IRQ_HIGHEDGE,
}
};
/*
* * The DM9000 has no eeprom, and it's MAC address is set by
* * the bootloader before starting the kernel.
* */
static struct dm9000_plat_data mini2440_dm9k_pdata = {
.flags = (DM9000_PLATF_16BITONLY | DM9000_PLATF_NO_EEPROM),
};
static struct platform_device mini2440_device_eth = {
.name = "dm9000",
.id = -1,
.num_resources = ARRAY_SIZE(mini2440_dm9k_resource),
.resource = mini2440_dm9k_resource,
.dev = {
.platform_data = &mini2440_dm9k_pdata,
},
};
static struct platform_device *mini2440_devices[] __initdata = {
&s3c_device_usb,
&s3c_device_lcd,
&s3c_device_wdt,
&s3c_device_i2c0,
&s3c_device_iis,
&s3c_device_nand, //;把nand flash 設(shè)備添加到開發(fā)板的設(shè)備列表結(jié)構(gòu)
&mini2440_device_eth,//;把網(wǎng)卡平臺(tái)設(shè)備添加到開發(fā)板的設(shè)備列表結(jié)構(gòu)
};
static void __init mini2440_map_io(void)
【2】調(diào)整DM9000 所用的位寬寄存器
因?yàn)?Linux-2.6.32.2 的DM9000 網(wǎng)卡驅(qū)動(dòng)并不是專門為mini2440 準(zhǔn)備的,所以還要在其源代碼中做一些移植工作,如下步驟。
(1)打開linux-2.6.32.2/drivers/net/dm9000.c,定位到41行附近,添加2410 相關(guān)的配置定義,如下紅色部分:
#include
#include
#include
#include "dm9000.h"
#if defined(CONFIG_ARCH_S3C2410)
#include
#endif
(2) 在dm9000 設(shè)備的初始化函數(shù)中添加如下紅色部分,這里是配置DM9000 所用片選總線的時(shí)序,因?yàn)閙ini2440 目前只有一個(gè)通過(guò)總線外擴(kuò)的設(shè)備,在此設(shè)備驅(qū)動(dòng)中直接修改相關(guān)的寄存器配置會(huì)更加容易理解一些,當(dāng)然這部分也可以放到mach-mini2440.c 中。
打開linux-2.6.32.2/drivers/net/dm9000.c,定位到1555行附近,加入下面代碼:
static int __init
dm9000_init(void)
{
#if defined(CONFIG_ARCH_S3C2410)
unsigned int oldval_bwscon = *(volatile unsigned int *)S3C2410_BWSCON;
unsigned int oldval_bankcon4 = *(volatile unsigned int *)S3C2410_BANKCON4;
*((volatile unsigned int *)S3C2410_BWSCON) =
(oldval_bwscon & ~(3<<16)) | S3C2410_BWSCON_DW4_16 | S3C2410_BWSCON_WS4 | S3C2410_BWSCON_ST4;
*((volatile unsigned int *)S3C2410_BANKCON4) = 0x1f7c;
#endif
printk(KERN_INFO "%s Ethernet Driver, V%sn", CARDNAME, DRV_VERSION);
return platform_driver_register(&dm9000_driver);
}
【3】需要注意的是,本開發(fā)板所用的DM9000 網(wǎng)卡并沒(méi)有外接EEPROM 用以存儲(chǔ)MAC 地址,因此系統(tǒng)中的MAC 地址是一個(gè)“軟”地址,也就是可以通過(guò)軟件進(jìn)行修改,可以隨意改為其他值。
打開linux-2.6.32.2/drivers/net/dm9000.c,定位到1461行附近,加入下面一行代碼:
static int __devinit
dm9000_probe(struct platform_device *pdev)
{
... ...
/* try reading the node address from the attached EEPROM */
//;嘗試從EEPROM 讀取MAC 地址
for (i = 0; i < 6; i += 2)
dm9000_read_eeprom(db, i / 2, ndev->dev_addr+i);
if (!is_valid_ether_addr(ndev->dev_addr) && pdata != NULL) {
mac_src = "platform data";
memcpy(ndev->dev_addr, pdata->dev_addr, 6);
}
if (!is_valid_ether_addr(ndev->dev_addr)) {
/* try reading from mac */
mac_src = "chip";
for (i = 0; i < 6; i++)
ndev->dev_addr[i] = ior(db, i+DM9000_PAR);
}
//;使用“軟”MAC 地址: 08:90:90:90:90:90
memcpy(ndev->dev_addr, "x08x90x90x90x90x90", 6);
if (!is_valid_ether_addr(ndev->dev_addr))
dev_warn(db->dev, "%s: Invalid ethernet MAC address. Please "
"set using ifconfign", ndev->name);
platform_set_drvdata(pdev, ndev);
ret = register_netdev(ndev);
if (ret == 0)
printk(KERN_INFO "%s: dm9000%c at %p,%p IRQ %d MAC: %pM (%s)n",
ndev->name, dm9000_type_to_char(db->type),
db->io_addr, db->io_data, ndev->irq,
ndev->dev_addr, mac_src);
return 0;
out:
dev_err(db->dev, "not found (%d).n", ret);
dm9000_release_board(pdev, db);
free_netdev(ndev);
return ret;
}
實(shí)際上到此為止 DM9000 就已經(jīng)移植結(jié)束了。
3.2,編譯測(cè)試
【1】在編譯之前,需要確認(rèn)在內(nèi)核中已經(jīng)配置了網(wǎng)卡驅(qū)動(dòng)
在內(nèi)核目錄下執(zhí)行:
[root@localhost linux-2.6.32.2]# make menuconfig
在打開的配置菜單Device Drivers --->Network device support ---> Ethernet (10 or 100Mbit) --->,可以看到如下圖
DM9000 已經(jīng)被選中,這是因?yàn)長(zhǎng)inux-2.6.32.2默認(rèn)的內(nèi)核配置已經(jīng)加入了DM9000 的支持。
【2】編譯
[root@localhost linux-2.6.32.2]# make clean
[root@localhost linux-2.6.32.2]# make uImage
編譯完成后生成zImage 和 uImage
【3】進(jìn)行網(wǎng)卡驅(qū)動(dòng)測(cè)試
(1)采用友善官方已經(jīng)移植好的根文件系統(tǒng),可以從其提供的光盤映像/linux目錄下直接復(fù)制過(guò)來(lái)。
[root@localhost ~]# cd linux-test
[root@localhost linux-test]# ls
busybox-1.13.3 mkyaffs2image.tgz
busybox-1.13.3-mini2440.tgz myrootfs
busybox-1.18.4rootfs_qtopia_qt4
busybox-1.18.4.tar.bz2 rootfs_qtopia_qt4-20110304.tar.gz
linux-2.6.32.2 usr
linux-2.6.39 yaffs2
[root@localhost linux-test
(2)將其復(fù)制到宿主機(jī)/nfsboot目錄下并命名為roorfs
[r