linux enc28j60网卡驱动移植(硬件spi和模拟spi)

本来想移植DM9000网卡的驱动,无奈硬件出了点问题,通过杜邦线链接开发板和DM9000网卡模块,系统上电,还没加载网卡驱动就直接崩溃了,找不到原因。。。刚好手上有一个enc28j60的网卡模块,于是就着手移植enc28j60的驱动。

其实移植enc28j60的驱动也十分简单,网上有现成的,只需要分配一些硬件资源即可。

由于我的内核版本老到掉牙,没有自带enc28j60的驱动,只能在网上找一个:

enc28j60.c

http://git.ti.com/ti-linux-kernel/ti-linux-kernel/blobs/7dac6f8df607929e51f4fd598d80bd009c45a9f8/drivers/net/enc28j60.c

enc28j60_hw.h

http://git.ti.com/ti-linux-kernel/ti-linux-kernel/blobs/7dac6f8df607929e51f4fd598d80bd009c45a9f8/drivers/net/enc28j60_hw.h

由于这个驱动是支持较新的内核,移植到2.6.22.6,只要改动3个地方好了。

 1 ... ...
 2
 3 static int enc28j60_set_hw_macaddr(struct net_device *ndev)
 4 {
 5     ... ...
 6
 7     if (!priv->hw_enable) {
 8         if (netif_msg_drv(priv)) {
 9             /* [cgw]: 屏蔽一下几行 */
10             //DECLARE_MAC_BUF(mac);
11             //printk(KERN_INFO DRV_NAME
12             //    ": %s: Setting MAC address to %s\n",
13             //    ndev->name, print_mac(mac, ndev->dev_addr));
14         }
15     }
16
17     ... ...
18 }
19
20 ... ...
21
22 static void dump_packet(const char *msg, int len, const char *data)
23 {
24     printk(KERN_DEBUG DRV_NAME ": %s - packet len:%d\n", msg, len);
25     /* [cgw]: 屏蔽一下几行 */
26     //print_hex_dump(KERN_DEBUG, "pk data: ", DUMP_PREFIX_OFFSET, 16, 1,
27     //        data, len, true);
28 }
29
30 ... ...
31
32 static int enc28j60_net_open(struct net_device *dev)
33 {
34     ... ...
35
36     if (!is_valid_ether_addr(dev->dev_addr)) {
37         if (netif_msg_ifup(priv)) {
38             /* [cgw]: 屏蔽一下几行 */
39             //DECLARE_MAC_BUF(mac);
40             //dev_err(&dev->dev, "invalid MAC address %s\n",
41             //    print_mac(mac, dev->dev_addr));
42         }
43         return -EADDRNOTAVAIL;
44     }
45
46     ... ...
47 }
48
49 ... ...

都是些打印相关的东西,屏蔽掉就好。

spi的框架可以参考这里:http://www.cnblogs.com/hackfun/p/6082489.html

这里只列出配置spi硬件资源的代码,只需要写一个spi_platform_dev.c文件就行了。模拟spi的模式下,spi_platform_dev.c和http://www.cnblogs.com/hackfun/p/6082489.html这里的spi_platform_dev.c文件相似,只需要增加一个外部中断入口给enc28j60用于接收中断,和更改spi的模式等。

模拟spi的模式下的spi_platform_dev.c

  1 #include <linux/module.h>
  2 #include <linux/version.h>
  3
  4 #include <linux/init.h>
  5
  6 #include <linux/kernel.h>
  7 #include <linux/types.h>
  8 #include <linux/interrupt.h>
  9 #include <linux/list.h>
 10 #include <linux/timer.h>
 11 #include <linux/init.h>
 12 #include <linux/serial_core.h>
 13 #include <linux/platform_device.h>
 14 #include <linux/irq.h>
 15
 16 #include <asm/gpio.h>
 17 #include <asm/io.h>
 18 #include <asm/arch/regs-gpio.h>
 19
 20 #include <linux/spi/spi.h>
 21 #include <linux/spi/spi_bitbang.h>
 22
 23 #include <asm/arch/regs-spi.h>
 24 #include <asm/arch/spi.h>
 25 #include <asm/arch/spi-gpio.h>
 26
 27
 28 static struct spi_board_info board_info[1] = {
 29     {
 30     .modalias = "enc28j60",    /* [cgw]: spi设备名,和设备驱动名对应 */
 31     .bus_num = 0,                 /* [cgw]: spi总线号,即spi0 */
 32     .chip_select = 2,             /* [cgw]: spi总线上的设备号,即spi0.2 */
 33     .max_speed_hz    = 50000,     /* [cgw]: spi时钟 */
 34     .mode = SPI_MODE_0,           /* [cgw]: spi数据模式 */
 35     .irq = IRQ_EINT2,
 36     },
 37 };
 38
 39
 40 static void enc28j60_chip_select(struct s3c2410_spigpio_info *spi, int cs)
 41 {
 42     /* [cgw]: 选中设备号为2的spi设备 */
 43     if (spi->board_info->chip_select == 2) {
 44         s3c2410_gpio_cfgpin(S3C2410_GPG2, S3C2410_GPIO_OUTPUT);
 45         /* [cgw]: 选中设备 */
 46         if (BITBANG_CS_ACTIVE == cs) {
 47             s3c2410_gpio_setpin(S3C2410_GPG2, 0);
 48         /* [cgw]: 释放设备 */
 49         } else if (BITBANG_CS_INACTIVE == cs) {
 50             s3c2410_gpio_setpin(S3C2410_GPG2, 1);
 51         }
 52     }
 53 }
 54
 55 /* [cgw]:  */
 56 static struct s3c2410_spigpio_info spi_dev = {
 57     .pin_clk = S3C2410_GPG7,
 58     .pin_mosi = S3C2410_GPG6,
 59     .pin_miso = S3C2410_GPG5,
 60     .board_size = 1,                    /* [cgw]: 设置板上spi接口数量为1 */
 61     .board_info = &board_info[0],
 62     .chip_select = enc28j60_chip_select
 63 };
 64
 65 static void spi_dev_release(struct device * dev)
 66 {
 67     printk("spi_dev_release! \n");
 68 }
 69
 70 /* [cgw]: 分配一个平台设备 */
 71 static struct platform_device spi_platform_dev = {
 72     .name         = "s3c24xx-spi-gpio",        /* [cgw]: 设置平台设备名,和平台驱动名对应 */
 73     .id           = -1,
 74     .dev = {
 75         .release = spi_dev_release,
 76         .platform_data = (void *)&spi_dev,      /* [cgw]: 通过platform_data传递spi_dev给平台驱动
 77                                                 * 平台驱动可以访问spi_dev
 78                                                 */
 79     },
 80 };
 81
 82
 83 static int spi_dev_init(void)
 84 {
 85     s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_EINT2);
 86
 87     /* [cgw]: 注册spi_platform_dev平台设备 */
 88     platform_device_register(&spi_platform_dev);
 89     return 0;
 90 }
 91
 92 static void spi_dev_exit(void)
 93 {
 94     /* [cgw]: 注销spi_platform_dev平台设备 */
 95     platform_device_unregister(&spi_platform_dev);
 96 }
 97
 98 module_init(spi_dev_init);
 99 module_exit(spi_dev_exit);
100
101 MODULE_LICENSE("GPL");

makefile:

KERN_DIR = /work/system/linux-2.6.22.6

all:
    make -C $(KERN_DIR) M=`pwd` modules 

clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm -rf modules.order

obj-m    += spi_platform_dev.o
obj-m    += spi_s3c24xx_gpio.o
obj-m    += spi_bitbang.o
obj-m    += enc28j60.o

硬件spi的模式下的spi_platform_dev.c

  1 #include <linux/module.h>
  2 #include <linux/version.h>
  3
  4 #include <linux/init.h>
  5
  6 #include <linux/kernel.h>
  7 #include <linux/types.h>
  8 #include <linux/interrupt.h>
  9 #include <linux/list.h>
 10 #include <linux/timer.h>
 11 #include <linux/init.h>
 12 #include <linux/serial_core.h>
 13 #include <linux/platform_device.h>
 14 #include <linux/irq.h>
 15
 16 #include <asm/gpio.h>
 17 #include <asm/io.h>
 18 #include <asm/arch/regs-gpio.h>
 19
 20 #include <linux/spi/spi.h>
 21 #include <linux/spi/spi_bitbang.h>
 22
 23 #include <asm/arch/regs-spi.h>
 24 #include <asm/arch/spi.h>
 25 #include <asm/arch/spi-gpio.h>
 26
 27
 28 /* SPI (1) */
 29
 30 static struct resource s3c_spi1_resource[] = {
 31     [0] = {
 32         .start = S3C2410_PA_SPI + S3C2410_SPI1,
 33         .end   = S3C2410_PA_SPI + S3C2410_SPI1 + 0x1f,
 34         .flags = IORESOURCE_MEM,
 35     },
 36     [1] = {
 37         .start = IRQ_SPI1,
 38         .end   = IRQ_SPI1,
 39         .flags = IORESOURCE_IRQ,
 40     }
 41
 42 };
 43
 44
 45 static struct spi_board_info board_info[1] = {
 46     {
 47     .modalias = "enc28j60",    /* [cgw]: spi设备名,和设备驱动名对应 */
 48     .bus_num = 0,                 /* [cgw]: spi总线号,即spi0 */
 49     .chip_select = 2,             /* [cgw]: spi总线上的设备号,即spi0.2 */
 50     .max_speed_hz    = 50000,     /* [cgw]: spi时钟 */
 51     .mode = SPI_MODE_0,           /* [cgw]: spi数据模式 */
 52     .irq = IRQ_EINT2,
 53     },
 54 };
 55
 56 static struct s3c2410_spi_info spi_info = {
 57     .pin_cs = S3C2410_GPG2,    /* simple gpio cs */
 58     .board_size = ARRAY_SIZE(board_info),
 59     .board_info = &board_info[0],
 60     .set_cs = NULL
 61 };
 62
 63
 64 static void spi_dev_release(struct device * dev)
 65 {
 66     printk("spi_dev_release! \n");
 67 }
 68
 69 /* [cgw]: 分配一个平台设备 */
 70 static struct platform_device spi_platform_dev = {
 71     .name         = "s3c2410-spi",        /* [cgw]: 设置平台设备名,和平台驱动名对应 */
 72     .id           = 1,
 73     .num_resources      = ARRAY_SIZE(s3c_spi1_resource),
 74     .resource      = s3c_spi1_resource,
 75     .dev = {
 76         .release = spi_dev_release,
 77         .platform_data = &spi_info,
 78         //.dma_mask = &s3c_device_spi1_dmamask,
 79         //.coherent_dma_mask = 0xffffffffUL
 80     },
 81 };
 82
 83
 84 static int spi_dev_init(void)
 85 {
 86     /* [cgw]: 注册spi_platform_dev平台设备 */
 87     platform_device_register(&spi_platform_dev);
 88     return 0;
 89 }
 90
 91 static void spi_dev_exit(void)
 92 {
 93     /* [cgw]: 注销spi_platform_dev平台设备 */
 94     platform_device_unregister(&spi_platform_dev);
 95 }
 96
 97 module_init(spi_dev_init);
 98 module_exit(spi_dev_exit);
 99
100 MODULE_LICENSE("GPL");

makefile:

KERN_DIR = /work/system/linux-2.6.22.6

all:
    make -C $(KERN_DIR) M=`pwd` modules 

clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm -rf modules.order

obj-m    += spi_platform_dev.o
obj-m    += spi_s3c24xx.o
obj-m    += spi_bitbang.o
obj-m    += enc28j60.o

加载spi平台设备时(platform_device),应注意模拟spi时应加载spi_s3c24xx_gpio.c,硬件spi时应加载spi_s3c24xx.c

如:

模拟spi:

1 # insmod spi_bitbang.ko
2 # insmod spi_platform_dev.ko
3 # insmod spi_s3c24xx_gpio.ko
4 # insmod enc28j60.ko

硬件spi:

1 # insmod spi_bitbang.ko
2 # insmod spi_platform_dev.ko
3 # insmod spi_s3c24xx.ko
4 # insmod enc28j60.ko

其中spi_bitbang.c , spi_s3c24xx_gpio.c , spi_s3c24xx.c为内核原生源文件,不需要改动。

硬件spi时,加载驱动的实例:

谢谢!

时间: 2024-12-28 22:13:48

linux enc28j60网卡驱动移植(硬件spi和模拟spi)的相关文章

linux网卡驱动移植

这里重要的是物理层PHY receiver,MAC(media access control)层,这里与软件中的协议栈不同,在硬件上MAC是PHY的下一层.DM9000A将MAC和PHY做到一起,也可以像IIS设备那样,SOC内有IIS的控制器,而声卡UDA1341放在片外.网卡当然也有这种设计,它是把PHY的下层MAC放入SOC内,片外的是PHY,当然我暂时还没见过这种的.DM9000A的输入是并行的总线,可以和CPU直接IO.而IIS那种需要通过:CPU CORE BUS->I2S控制器->

【Linux驱动】TQ2440 DM9000E网卡驱动移植(Linux-2.6.30.4)

花了一天的时间研究了一下Linux-2.6.30.4版本号内核下关于TQ2440 DM9000E的网卡驱动移植.总结一下自己的收获. 事实上.在Linux-2.6.30.4版本号内核下有关于网卡驱动,可是这个网卡驱动不是针对于 DM9000E.而是针对于 DM9000的. 因此要在此基础之上改动一些參数.而且打上必要的相关代码.这个网卡驱动移植是參照了"天嵌科技出品-Linux移植之Step By Step_V4.6"的文档.但就是这样.还是花费了将近一天的时间.事实上这里的难点在于:

Linux驱动学习之TQ2440 DM9000E网卡驱动移植(Linux-2.6.30.4)

引言 在之前的文章中,我们介绍了如何使用Scala IDE也就是eclipse中集成的Scala开发插件来进行Scala语言程序的开发,在使用了一段时间之后,发现eclipse对Scala的支持并不是很好.用户体验比较差,比如联想速度比较慢等.由于在公司一直使用的Scala开发工具是Intellij IDEA(好吧,其实我使用Scala IDE的目的就是想试一下这两个各有什么优缺点),各方面感觉还不错,所以在此介绍一下这个开发环境. Intellij IDEA是jetbrain开发的一个IDE,

AM335x(TQ335x)学习笔记——Nand&amp;&amp;网卡驱动移植

移植完成声卡驱动之后本想再接再励,移植网卡驱动,但没想到的是TI维护的内核太健壮,移植网卡驱动跟之前移植按键驱动一样简单,Nand驱动也是如此,于是,本人将Nand和网卡放在同一篇文章中介绍.介绍之前先感慨一下:TI的维护的内核真的很健壮,DTS真的很强大. 1. Nand驱动移植 阅读TQ335x的原理图可知,TQ335x的Nand连接到了GPMC上,且都是使用的相应引脚的MODE0复用功能,AM335x上上电复位后这些引脚的默认状态就处于MODE0模式,故无需进行pinmux设置,原始的DT

AM335x(TQ335x)学习笔记——Nand&amp;amp;&amp;amp;网卡驱动移植

移植完毕声卡驱动之后本想再接再励,移植网卡驱动,但没想到的是TI维护的内核太健壮,移植网卡驱动跟之前移植按键驱动一样简单,Nand驱动也是如此,于是,本人将Nand和网卡放在同一篇文章中介绍.介绍之前先感慨一下:TI的维护的内核真的非常健壮,DTS真的非常强大. 1. Nand驱动移植 阅读TQ335x的原理图可知,TQ335x的Nand连接到了GPMC上,且与DTS中默认的配置吻合,此处不做不论什么改动,详情例如以下: nandflash_pins_s0: nandflash_pins_s0

I.MX6Q(TQIMX6Q/TQE9)学习笔记——新版BSP之网卡驱动移植

由于对网卡这块不是很熟悉,误以为网卡驱动也可以简单的配置下DTS就可以正常工作了,实际移植中遇到了些问题.闲话少说,下面开始tqimx6q的网卡驱动移植. DTS编写 首先在我们的DTS中添加网卡配置,参考sabrelite的DTS,我们可以如下编写: &fec { pinctrl-names = "default"; pinctrl-0 = <&pinctrl_enet_1>; phy-mode = "rgmii"; status =

linux 编译网卡驱动

将smsc7500网卡驱动拷贝到/drive/net/usb文件夹下 拷贝ioctl_7500.h  smsc7500usbnet.c smsc7500version.h smsclan7500.h  smsc7500usbnet.h smsclan7500.c到usb文件夹下, 打开Kconfig,找到95xx,进行相应的修改. 打开Makefile ,添加相应的.o文件 在make menuconfig里面把smsc7500选上 然后make linux 编译网卡驱动

mini2440移植uboot-2008.10 (二) DM9000网卡驱动移植

还是利用 mini2440移植uboot-2008.10 (一)  修改好的代码 通过观察可以发现,mini2400使用的网卡芯片是DM9000,在uboot-2008.10源码中已经支持该芯片的驱动(drivers/net/dm9000.c),但是并不完善 具体原理还不是很清楚,现在只是做一下移植步骤的记录 1.修改uboot配置文件(include/configs/mini2440.h) 53 /* 54 * Hardware drivers 55 */ 56 #if 0 57 #defin

Linux e1000e网卡驱动

目录 识别网卡 命令行参数 附加配置 技术支持 一.识别网卡e1000e驱动支持Intel所有的GbE PCIe网卡,除了82575,82576,基于82580系列的网卡.提示:Intel(R) PRO/1000 P Dual网卡是支持e1000,但不支持e1000e,因为82546部分支持PCIe. 更多信息关于怎么识别你的网卡,去官网的 Adapter & Driver ID Guide: http://support.intel.com/support/go/network/adapter