热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

enc28j60linux驱动_Linuxenc28j60网卡驱动移植(硬件spi和模拟spi)

本来想移植DM9000网卡的驱动,无奈硬件出了点问题,通过杜邦线链接开发板和DM9000网卡模块,系统上电,还没加载网卡驱动

本来想移植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个地方好了。

... ...

static int enc28j60_set_hw_macaddr(struct net_device *ndev)

{

... ...

if (!priv->hw_enable) {

if (netif_msg_drv(priv)) {

/* [cgw]: 屏蔽一下几行 */

//DECLARE_MAC_BUF(mac);

//printk(KERN_INFO DRV_NAME

//    ": %s: Setting MAC address to %s\n",

//    ndev->name, print_mac(mac, ndev->dev_addr));

}

}

... ...

}

... ...

static void dump_packet(const char *msg, int len, const char *data)

{

printk(KERN_DEBUG DRV_NAME ": %s - packet len:%d\n", msg, len);

/* [cgw]: 屏蔽一下几行 */

//print_hex_dump(KERN_DEBUG, "pk data: ", DUMP_PREFIX_OFFSET, 16, 1,

//        data, len, true);

}

... ...

static int enc28j60_net_open(struct net_device *dev)

{

... ...

if (!is_valid_ether_addr(dev->dev_addr)) {

if (netif_msg_ifup(priv)) {

/* [cgw]: 屏蔽一下几行 */

//DECLARE_MAC_BUF(mac);

//dev_err(&dev->dev, "invalid MAC address %s\n",

//    print_mac(mac, dev->dev_addr));

}

return -EADDRNOTAVAIL;

}

... ...

}

... ...

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

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

模拟spi的模式下的spi_platform_dev.c

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

static struct spi_board_info board_info[1] = {

{

.modalias = "enc28j60",    /* [cgw]: spi设备名,和设备驱动名对应 */

.bus_num = 0,                /* [cgw]: spi总线号,即spi0 */

.chip_select = 2,            /* [cgw]: spi总线上的设备号,即spi0.2 */

.max_speed_hz    = 50000,    /* [cgw]: spi时钟 */

.mode = SPI_MODE_0,          /* [cgw]: spi数据模式 */

.irq = IRQ_EINT2,

},

};

static void enc28j60_chip_select(struct s3c2410_spigpio_info *spi, int cs)

{

/* [cgw]: 选中设备号为2的spi设备 */

if (spi->board_info->chip_select == 2) {

s3c2410_gpio_cfgpin(S3C2410_GPG2, S3C2410_GPIO_OUTPUT);

/* [cgw]: 选中设备 */

if (BITBANG_CS_ACTIVE == cs) {

s3c2410_gpio_setpin(S3C2410_GPG2, 0);

/* [cgw]: 释放设备 */

} else if (BITBANG_CS_INACTIVE == cs) {

s3c2410_gpio_setpin(S3C2410_GPG2, 1);

}

}

}

/* [cgw]:  */

static struct s3c2410_spigpio_info spi_dev = {

.pin_clk = S3C2410_GPG7,

.pin_mosi = S3C2410_GPG6,

.pin_miso = S3C2410_GPG5,

.board_size = 1,                    /* [cgw]: 设置板上spi接口数量为1 */

.board_info = &board_info[0],

.chip_select = enc28j60_chip_select

};

static void spi_dev_release(struct device * dev)

{

printk("spi_dev_release! \n");

}

/* [cgw]: 分配一个平台设备 */

static struct platform_device spi_platform_dev = {

.name        = "s3c24xx-spi-gpio",        /* [cgw]: 设置平台设备名,和平台驱动名对应 */

.id          = -1,

.dev = {

.release = spi_dev_release,

.platform_data = (void *)&spi_dev,      /* [cgw]: 通过platform_data传递spi_dev给平台驱动

* 平台驱动可以访问spi_dev

*/

},

};

static int spi_dev_init(void)

{

s3c2410_gpio_cfgpin(S3C2410_GPF2, S3C2410_GPF2_EINT2);

/* [cgw]: 注册spi_platform_dev平台设备 */

platform_device_register(&spi_platform_dev);

return 0;

}

static void spi_dev_exit(void)

{

/* [cgw]: 注销spi_platform_dev平台设备 */

platform_device_unregister(&spi_platform_dev);

}

module_init(spi_dev_init);

module_exit(spi_dev_exit);

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

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

/* SPI (1) */

static struct resource s3c_spi1_resource[] = {

[0] = {

.start = S3C2410_PA_SPI + S3C2410_SPI1,

.end  = S3C2410_PA_SPI + S3C2410_SPI1 + 0x1f,

.flags = IORESOURCE_MEM,

},

[1] = {

.start = IRQ_SPI1,

.end  = IRQ_SPI1,

.flags = IORESOURCE_IRQ,

}

};

static struct spi_board_info board_info[1] = {

{

.modalias = "enc28j60",    /* [cgw]: spi设备名,和设备驱动名对应 */

.bus_num = 0,                /* [cgw]: spi总线号,即spi0 */

.chip_select = 2,            /* [cgw]: spi总线上的设备号,即spi0.2 */

.max_speed_hz    = 50000,    /* [cgw]: spi时钟 */

.mode = SPI_MODE_0,          /* [cgw]: spi数据模式 */

.irq = IRQ_EINT2,

},

};

static struct s3c2410_spi_info spi_info = {

.pin_cs = S3C2410_GPG2,    /* simple gpio cs */

.board_size = ARRAY_SIZE(board_info),

.board_info = &board_info[0],

.set_cs = NULL

};

static void spi_dev_release(struct device * dev)

{

printk("spi_dev_release! \n");

}

/* [cgw]: 分配一个平台设备 */

static struct platform_device spi_platform_dev = {

.name        = "s3c2410-spi",        /* [cgw]: 设置平台设备名,和平台驱动名对应 */

.id          = 1,

.num_resources      = ARRAY_SIZE(s3c_spi1_resource),

.resource      = s3c_spi1_resource,

.dev = {

.release = spi_dev_release,

.platform_data = &spi_info,

//.dma_mask = &s3c_device_spi1_dmamask,

//.coherent_dma_mask = 0xffffffffUL

},

};

static int spi_dev_init(void)

{

/* [cgw]: 注册spi_platform_dev平台设备 */

platform_device_register(&spi_platform_dev);

return 0;

}

static void spi_dev_exit(void)

{

/* [cgw]: 注销spi_platform_dev平台设备 */

platform_device_unregister(&spi_platform_dev);

}

module_init(spi_dev_init);

module_exit(spi_dev_exit);

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时,加载驱动的实例:

谢谢!



推荐阅读
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 本文介绍了为什么要使用多进程处理TCP服务端,多进程的好处包括可靠性高和处理大量数据时速度快。然而,多进程不能共享进程空间,因此有一些变量不能共享。文章还提供了使用多进程实现TCP服务端的代码,并对代码进行了详细注释。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • 本文介绍了在CentOS上安装Python2.7.2的详细步骤,包括下载、解压、编译和安装等操作。同时提供了一些注意事项,以及测试安装是否成功的方法。 ... [详细]
  • 本文由编程笔记#小编整理,主要介绍了关于数论相关的知识,包括数论的算法和百度百科的链接。文章还介绍了欧几里得算法、辗转相除法、gcd、lcm和扩展欧几里得算法的使用方法。此外,文章还提到了数论在求解不定方程、模线性方程和乘法逆元方面的应用。摘要长度:184字。 ... [详细]
  • 本文介绍了使用哈夫曼树实现文件压缩和解压的方法。首先对数据结构课程设计中的代码进行了分析,包括使用时间调用、常量定义和统计文件中各个字符时相关的结构体。然后讨论了哈夫曼树的实现原理和算法。最后介绍了文件压缩和解压的具体步骤,包括字符统计、构建哈夫曼树、生成编码表、编码和解码过程。通过实例演示了文件压缩和解压的效果。本文的内容对于理解哈夫曼树的实现原理和应用具有一定的参考价值。 ... [详细]
  • 本文介绍了Codeforces Round #321 (Div. 2)比赛中的问题Kefa and Dishes,通过状压和spfa算法解决了这个问题。给定一个有向图,求在不超过m步的情况下,能获得的最大权值和。点不能重复走。文章详细介绍了问题的题意、解题思路和代码实现。 ... [详细]
  • 本文介绍了一道经典的状态压缩题目——关灯问题2,并提供了解决该问题的算法思路。通过使用二进制表示灯的状态,并枚举所有可能的状态,可以求解出最少按按钮的次数,从而将所有灯关掉。本文还对状压和位运算进行了解释,并指出了该方法的适用性和局限性。 ... [详细]
  • 微软头条实习生分享深度学习自学指南
    本文介绍了一位微软头条实习生自学深度学习的经验分享,包括学习资源推荐、重要基础知识的学习要点等。作者强调了学好Python和数学基础的重要性,并提供了一些建议。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文介绍了RPC框架Thrift的安装环境变量配置与第一个实例,讲解了RPC的概念以及如何解决跨语言、c++客户端、web服务端、远程调用等需求。Thrift开发方便上手快,性能和稳定性也不错,适合初学者学习和使用。 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • 本文整理了315道Python基础题目及答案,帮助读者检验学习成果。文章介绍了学习Python的途径、Python与其他编程语言的对比、解释型和编译型编程语言的简述、Python解释器的种类和特点、位和字节的关系、以及至少5个PEP8规范。对于想要检验自己学习成果的读者,这些题目将是一个不错的选择。请注意,答案在视频中,本文不提供答案。 ... [详细]
author-avatar
吴国伟60942
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有