歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux編程 >> Linux編程

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個地方好了。

... ...

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的框架可以參考這裡:http://www.linuxidc.com/Linux/2016-12/138512.htm

這裡只列出配置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 <linux/module.h>
#include <linux/version.h>

#include <linux/init.h>

#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/interrupt.h>
#include <linux/list.h>
#include <linux/timer.h>
#include <linux/init.h>
#include <linux/serial_core.h>
#include <linux/platform_device.h>
#include <linux/irq.h>

#include <asm/gpio.h>
#include <asm/io.h>
#include <asm/arch/regs-gpio.h>

#include <linux/spi/spi.h>
#include <linux/spi/spi_bitbang.h>

#include <asm/arch/regs-spi.h>
#include <asm/arch/spi.h>
#include <asm/arch/spi-gpio.h>


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 <linux/module.h>
#include <linux/version.h>

#include <linux/init.h>

#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/interrupt.h>
#include <linux/list.h>
#include <linux/timer.h>
#include <linux/init.h>
#include <linux/serial_core.h>
#include <linux/platform_device.h>
#include <linux/irq.h>

#include <asm/gpio.h>
#include <asm/io.h>
#include <asm/arch/regs-gpio.h>

#include <linux/spi/spi.h>
#include <linux/spi/spi_bitbang.h>

#include <asm/arch/regs-spi.h>
#include <asm/arch/spi.h>
#include <asm/arch/spi-gpio.h>


/* 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時,加載驅動的實例:

謝謝!

Copyright © Linux教程網 All Rights Reserved