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

谢谢!



推荐阅读
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 本文探讨如何使用 PHP 进行字符串处理,特别是如何检测一个字符串是否存在于另一个字符串中,并确定其具体位置。通过实例代码展示,帮助读者掌握这一常用功能。 ... [详细]
  • 开发笔记:2020 BJDCTF Re encode
    开发笔记:2020 BJDCTF Re encode ... [详细]
  • 本题探讨了在一个有向图中,如何根据特定规则将城市划分为若干个区域,使得每个区域内的城市之间能够相互到达,并且划分的区域数量最少。题目提供了时间限制和内存限制,要求在给定的城市和道路信息下,计算出最少需要划分的区域数量。 ... [详细]
  • 树链问题的优化解法:深度优先搜索与质因数分解
    本文介绍了一种通过深度优先搜索(DFS)和质因数分解来解决最长树链问题的方法。我们通过枚举树链上的最大公约数(GCD),将所有节点按其质因子分类,并计算每个类别的最长链,最终求得全局最长链。 ... [详细]
  • Linux环境下C语言实现定时向文件写入当前时间
    本文介绍如何在Linux系统中使用C语言编程,实现在每秒钟向指定文件中写入当前时间戳。通过此示例,读者可以了解基本的文件操作、时间处理以及循环控制。 ... [详细]
  • 在编译BSP包过程中,遇到了一个与 'gets' 函数相关的编译错误。该问题通常发生在较新的编译环境中,由于 'gets' 函数已被弃用并视为安全漏洞。本文将详细介绍如何通过修改源代码和配置文件来解决这一问题。 ... [详细]
  • Linux环境下进程间通信:深入解析信号机制
    本文详细探讨了Linux系统中信号的生命周期,从信号生成到处理函数执行完毕的全过程,并介绍了信号编程中的注意事项和常见应用实例。通过分析信号在进程中的注册、注销及处理过程,帮助读者理解如何高效利用信号进行进程间通信。 ... [详细]
  • 本文提供了一个详细的PHP用户认证和管理的代码示例,包括用户登录验证、数据库连接、错误处理等关键部分的实现。 ... [详细]
  • 本文探讨了在Git子模块目录中运行pre-commit时遇到的错误,并提供了一种通过Docker环境解决此问题的方法。 ... [详细]
  • 本文深入探讨了HTTP请求和响应对象的使用,详细介绍了如何通过响应对象向客户端发送数据、处理中文乱码问题以及常见的HTTP状态码。此外,还涵盖了文件下载、请求重定向、请求转发等高级功能。 ... [详细]
  • C#设计模式学习笔记:观察者模式解析
    本文将探讨观察者模式的基本概念、应用场景及其在C#中的实现方法。通过借鉴《Head First Design Patterns》和维基百科等资源,详细介绍该模式的工作原理,并提供具体代码示例。 ... [详细]
  • Coursera ML 机器学习
    2019独角兽企业重金招聘Python工程师标准线性回归算法计算过程CostFunction梯度下降算法多变量回归![选择特征](https:static.oschina.n ... [详细]
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 对 manual_async_fn 进行了改进,确保其能够正确处理和捕获输入的生命周期。 ... [详细]
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社区 版权所有