S3C2440驱动移植——SPI

PC主机:Ubuntu 10.4 和redhat 9.0

目标板:TQ2440开发板,linux内核2.6.30


本文将从原理上讲解如何进行SPI驱动移植。作者希望在移植SPI驱动之前,您已对SPI子系统有所了解。

本文的讲解将基于我所写的spi子系统分析:

基于S3C2440的嵌入式Linux驱动——SPI子系统解读(一)

基于S3C2440的嵌入式Linux驱动——SPI子系统解读(二)

基于S3C2440的嵌入式Linux驱动——SPI子系统解读(三)

基于S3C2440的嵌入式Linux驱动——SPI子系统解读(四)


1. 配置内核


2. 修改源码

修改arch/arm/mach-s3c2440/mach-smdk2440.c

增加内容:

#include

#include

static struct s3c2410_spi_info s3c2410_spi0_platdata={

      .pin_cs = S3C2410_GPG(2),

      .num_cs = 1,

      .bus_num = 0,

};

static struct spi_board_info s3c2410_spi0_board[]={

   [0] = {

             .modalias = “spidev”,

             .bus_num = 0,

             .chip_select = 0,

             .max_speed_hz = 500 * 1000,

    },

};

在smdk2440_devices[]中增加

&s3c_device_spi0,

在smdk2440_mach_init函数中增加

s3c_device_spi0.dev.platform_data = &s3c2410_spi0_platdata;

spi_register_board_info(s3c2410_spi0_board,ARRAY_SIZE(s3c2410_spi0_board));

修改完毕以后,重新编译内核,并下载内核。查看下设备节点:

[root@yj423 /dev]#ls /dev/spidev0.0
/dev/spidev0.0


3.测试

用杜邦线将MOSI和MISO管脚相连。

在内核源码中提供了SPI测试程序。源码位于Documentation/spi/spidev_test.c,编译之后,下到开发板并执行。

[root@yj423 yj423]#./spidev_test -D /dev/spidev0.0
spi mode: 0
bits per word: 8
max speed: 500000 Hz (500 KHz)

FF FF FF FF FF FF
40 00 00 00 00 95
FF FF FF FF FF FF
FF FF FF FF FF FF
FF FF FF FF FF FF
DE AD BE EF BA AD
F0 0D
NOTE:如果你只是为了移植而移植,那么后面的内容就不要看了。想知道WHY,那请继续往下看。


4.原理

首先,在第二小结中我们看到定义了两个数据结构,之所以要定义那是因为驱动程序需要用到,详见

基于S3C2440的嵌入式Linux驱动——SPI子系统解读(二)

需要这两个数据结构的地方,均用红色字体说明了。

数据s3c2410_spi0_platdata是作为平台的私有数据,因此使用了s3c_device_spi0.dev.platform_data = &s3c2410_spi0_platdata来添加该数据。

而s3c2410_spi0_board必须通过函数spi_register_board_info添加到内核中,

最重要的一个问题,这些数据结构的值是怎么给出的?

先来看看这两个数据结构:


[cpp] view plain copy print ?
  1. struct spi_board_info {  
  2.     /* the device name and module name are coupled, like platform_bus; 
  3.      * "modalias" is normally the driver name. 
  4.      * 
  5.      * platform_data goes to spi_device.dev.platform_data, 
  6.      * controller_data goes to spi_device.controller_data, 
  7.      * irq is copied too 
  8.      */  
  9.     char        modalias[32];  
  10.     const void  *platform_data;  
  11.     void        *controller_data;  
  12.     int     irq;  
  13.   
  14.     /* slower signaling on noisy or low voltage boards */  
  15.     u32     max_speed_hz;  
  16.   
  17.   
  18.     /* bus_num is board specific and matches the bus_num of some 
  19.      * spi_master that will probably be registered later. 
  20.      * 
  21.      * chip_select reflects how this chip is wired to that master; 
  22.      * it's less than num_chipselect. 
  23.      */  
  24.     u16     bus_num;  
  25.     u16     chip_select;  
  26.   
  27.     /* mode becomes spi_device.mode, and is essential for chips 
  28.      * where the default of SPI_CS_HIGH = 0 is wrong. 
  29.      */  
  30.     u8      mode;  
  31.   
  32.     /* ... may need additional spi_device chip config data here. 
  33.      * avoid stuff protocol drivers can set; but include stuff 
  34.      * needed to behave without being bound to a driver: 
  35.      *  - quirks like clock rate mattering when not selected 
  36.      */  
  37. };  
  38.   
  39. struct s3c2410_spi_info {  
  40.     int             pin_cs;    /* simple gpio cs */  
  41.     unsigned int         num_cs;    /* total chipselects */  
  42.     int             bus_num;       /* bus number to use. */  
  43.   
  44.     void (*gpio_setup)(struct s3c2410_spi_info *spi, int enable);  
  45.     void (*set_cs)(struct s3c2410_spi_info *spi, int cs, int pol);  
  46. };  
struct spi_board_info {
	/* the device name and module name are coupled, like platform_bus;
	 * "modalias" is normally the driver name.
	 *
	 * platform_data goes to spi_device.dev.platform_data,
	 * controller_data goes to spi_device.controller_data,
	 * irq is copied too
	 */
	char		modalias[32];
	const void	*platform_data;
	void		*controller_data;
	int		irq;

	/* slower signaling on noisy or low voltage boards */
	u32		max_speed_hz;


	/* bus_num is board specific and matches the bus_num of some
	 * spi_master that will probably be registered later.
	 *
	 * chip_select reflects how this chip is wired to that master;
	 * it's less than num_chipselect.
	 */
	u16		bus_num;
	u16		chip_select;

	/* mode becomes spi_device.mode, and is essential for chips
	 * where the default of SPI_CS_HIGH = 0 is wrong.
	 */
	u8		mode;

	/* ... may need additional spi_device chip config data here.
	 * avoid stuff protocol drivers can set; but include stuff
	 * needed to behave without being bound to a driver:
	 *  - quirks like clock rate mattering when not selected
	 */
};

struct s3c2410_spi_info {
    int             pin_cs;    /* simple gpio cs */
    unsigned int         num_cs;    /* total chipselects */
    int             bus_num;       /* bus number to use. */

    void (*gpio_setup)(struct s3c2410_spi_info *spi, int enable);
    void (*set_cs)(struct s3c2410_spi_info *spi, int cs, int pol);
};

4.1 s3c2410_spi_info

pin_cs:表示哪个管脚用输出片选信号,这里使用GPG2管脚,具体的管脚请参看你的datasheet。

bus_num:表示总线接口。这里使用0,表示使用SPI0控制器。S3C2440有两个SPI控制器,分别为SPI0和SPI1。

num_cs:表示该SPI控制器控制几个SPI从设备,这里为1,仅有一个设备。

s3c2410_spi_info作为平台数据,在s3c24xx_spi_probe函数中,bus_cs和bus_num将被复制给master中的相应字段。也就是说,该master为SPI0控制器,有1个从设备。

而pin_cs将由函数s3c24xx_spi_gpiocs使用,该函数使能片选信号。

[cpp] view plain copy print ?
  1. static void s3c24xx_spi_gpiocs(struct s3c2410_spi_info *spi, int cs, int pol)    
  2. {    
  3.     gpio_set_value(spi->pin_cs, pol);    
  4. }    
static void s3c24xx_spi_gpiocs(struct s3c2410_spi_info *spi, int cs, int pol)  
{  
    gpio_set_value(spi->pin_cs, pol);  
}  

4.2 spi_board_info

首先,我们知道spi_board_info作为SPI设备的板级信息,在spi_new_device函数中将该板级信息全部复制给spi_device。

modalias:将用于绑定驱动和设备。我们看下match方法:

[cpp] view plain copy print ?
  1. static int spi_match_device(struct device *dev, struct device_driver *drv)    
  2. {    
  3.     const struct spi_device *spi = to_spi_device(dev);    
  4.     
  5.     return strcmp(spi->modalias, drv->name) == 0;    
  6. }    
static int spi_match_device(struct device *dev, struct device_driver *drv)  
{  
    const struct spi_device *spi = to_spi_device(dev);  
  
    return strcmp(spi->modalias, drv->name) == 0;  
}  
而drv->name为在spi_driver中定义,如下

[cpp] view plain copy print ?
  1. static struct spi_driver spidev_spi = {    
  2.     .driver = {    
  3.         .name =        "spidev",    
  4.         .owner =    THIS_MODULE,    
  5.     },    
  6.     .probe =    spidev_probe,    
  7.     .remove =    __devexit_p(spidev_remove),    
  8.     
  9.     /* NOTE:  suspend/resume methods are not necessary here.  
  10.      * We don't do anything except pass the requests to/from  
  11.      * the underlying controller.  The refrigerator handles  
  12.      * most issues; the controller driver handles the rest.  
  13.      */    
  14. };    
static struct spi_driver spidev_spi = {  
    .driver = {  
        .name =        "spidev",  
        .owner =    THIS_MODULE,  
    },  
    .probe =    spidev_probe,  
    .remove =    __devexit_p(spidev_remove),  
  
    /* NOTE:  suspend/resume methods are not necessary here. 
     * We don't do anything except pass the requests to/from 
     * the underlying controller.  The refrigerator handles 
     * most issues; the controller driver handles the rest. 
     */  
};  
可见,drv->name为spidev,因此spi-device的modalias字段也必须为spidev,而该字段是由板级信息的spi_board_info的modaias复制得来。这就是为什么modalias必须为spidev。

bus_num:用来匹配主控制器和属于该主控制器的SPI设备。

[cpp] view plain copy print ?
  1. static void scan_boardinfo(struct spi_master *master)    
  2. {    
  3.     struct boardinfo    *bi;    
  4.     
  5.     mutex_lock(&board_lock);    
  6.     /*以board_list为链表头,遍历所有的boardinfo结构,链表由spi_register_board_info添加*/    
  7.     list_for_each_entry(bi, &board_list, list) {        
  8.         struct spi_board_info    *chip = bi->board_info;    
  9.         unsigned        n;    
  10.         /*遍历该boardinfo指向的spi_board_info数组*/    
  11.         for (n = bi->n_board_info; n > 0; n--, chip++) {    
  12.             if (chip->bus_num != master->bus_num) /*通过bus_num对spi设备和master进行匹配*/    
  13.                 continue;    
  14.             /* NOTE: this relies on spi_new_device to  
  15.              * issue diagnostics when given bogus inputs  
  16.              */    
  17.              /*执行到此,表示匹配完成,SPI设备由该SPI接口来控制,开始创建spi_device*/    
  18.             (void) spi_new_device(master, chip);    
  19.         }    
  20.     }    
  21.     mutex_unlock(&board_lock);    
  22. }    
static void scan_boardinfo(struct spi_master *master)  
{  
    struct boardinfo    *bi;  
  
    mutex_lock(&board_lock);  
    /*以board_list为链表头,遍历所有的boardinfo结构,链表由spi_register_board_info添加*/  
    list_for_each_entry(bi, &board_list, list) {      
        struct spi_board_info    *chip = bi->board_info;  
        unsigned        n;  
        /*遍历该boardinfo指向的spi_board_info数组*/  
        for (n = bi->n_board_info; n > 0; n--, chip++) {  
            if (chip->bus_num != master->bus_num) /*通过bus_num对spi设备和master进行匹配*/  
                continue;  
            /* NOTE: this relies on spi_new_device to 
             * issue diagnostics when given bogus inputs 
             */  
             /*执行到此,表示匹配完成,SPI设备由该SPI接口来控制,开始创建spi_device*/  
            (void) spi_new_device(master, chip);  
        }  
    }  
    mutex_unlock(&board_lock);  
}  
可见,master的bus_num和spi_board_info的bus_num必须相同,从上面我们又知道master的bus_num是由s3c2410_spi_info的bus_num复制而来,因此spi_board_info的bus_num的值必须和s3c2410_spi_info的bus_num相等。

chip_selest:该值表示这是该SPI接口上的第几个设备,这里是0,其实就是第一个设备。该值用于区分不同的设备。注意,该值必须小于s3c2410_spi_info的num_cs字段。

max_speed_hz:表示传输速率,这里设置的速度为500kHz。

其余的字段可以不设置。

5. 结束语

   本文给出了SPI驱动移植的步骤,并使用测试程序测试了该SPI驱动。最后,对移植的原理进行了讲解。

你可能感兴趣的:(linux,驱动)