Linux内核中SPI总线驱动分析

本文涉及的产品
数据传输服务 DTS,数据同步 small 3个月
推荐场景:
数据库上云
数据传输服务 DTS,数据迁移 small 3个月
推荐场景:
MySQL数据库上云
数据传输服务 DTS,同步至 ClickHouse 1个月
简介:

本文主要有两个大的模块:一个是SPI总线驱动的分析 (研究了具体实现的过程);

另一个是SPI总线驱动的编写(不用研究具体的实现过程)。 

1 SPI概述

      SPI是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口,是Motorola首先在其MC68HCXX系列处理器上定义的。SPI接口主要应用在 EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号解码器之间。SPI是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间,提供方便。
      SPI的通信原理很简单,它以主从方式工作,这种模式通常有一个主设备和一个或多个从设备,需要4根线,事实上3根也可以。也是所有基于SPI的设备共有的,它们是SDI(数据输入),SDO(数据输出),SCLK(时钟),CS(片选)。
      MOSI(SDO):主器件数据输出,从器件数据输入。
      MISO(SDI):主器件数据输入,从器件数据输出。
      SCLK :时钟信号,由主器件产生。
      CS:从器件使能信号,由主器件控制。
      其中CS是控制芯片是否被选中的,也就是说只有片选信号为预先规定的使能信号时(高电位或低电位),对此芯片的操作才有效,这就允许在同一总线上连接多个SPI设备成为可能。需要注意的是,在具体的应用中,当一条SPI总线上连接有多个设备时,SPI本身的CS有可能被其他的GPIO脚代替,即每个设备的CS脚被连接到处理器端不同的GPIO,通过操作不同的GPIO口来控制具体的需要操作的SPI设备,减少各个SPI设备间的干扰。
      SPI是串行通讯协议,也就是说数据是一位一位从MSB或者LSB开始传输的,这就是SCK时钟线存在的原因,由SCK提供时钟脉冲,MISO、MOSI则基于此脉冲完成数据传输。 SPI支持4-32bits的串行数据传输,支持MSB和LSB,每次数据传输时当从设备的大小端发生变化时需要重新设置SPI Master的大小端。

2 Linux SPI驱动总体架构

     在2.6的linux内核中,SPI的驱动架构可以分为如下三个层次:SPI 核心层、SPI控制器驱动层和SPI设备驱动层。

     Linux 中SPI驱动代码位于drivers/spi目录。

2.1 SPI核心层

     SPI核心层是Linux的SPI核心部分,提供了核心数据结构的定义、SPI控制器驱动和设备驱动的注册、注销管理等API。其为硬件平台无关层,向下屏蔽了物理总线控制器的差异,定义了统一的访问策略和接口;其向上提供了统一的接口,以便SPI设备驱动通过总线控制器进行数据收发。

      Linux中,SPI核心层的代码位于driver/spi/ spi.c。由于该层是平台无关层,本文将不再叙述,有兴趣可以查阅相关资料。

2.2 SPI控制器驱动层

     SPI控制器驱动层,每种处理器平台都有自己的控制器驱动,属于平台移植相关层。它的职责是为系统中每条SPI总线实现相应的读写方法。在物理上,每个SPI控制器可以连接若干个SPI从设备。

      在系统开机时,SPI控制器驱动被首先装载。一个控制器驱动用于支持一条特定的SPI总线的读写。一个控制器驱动可以用数据结构struct spi_master来描述。

在include/liunx/spi/spi.h文件中,在数据结构struct spi_master定义如下: 
struct spi_master {  
    struct device   dev;  
    s16         bus_num;  
    u16         num_chipselect;  
    int         (*setup)(struct spi_device *spi);  
    int         (*transfer)(struct spi_device *spi, struct spi_message *mesg);  
    void        (*cleanup)(struct spi_device *spi);  
};  

     bus_num为该控制器对应的SPI总线号。
      num_chipselect 控制器支持的片选数量,即能支持多少个spi设备 
     setup函数是设置SPI总线的模式,时钟等的初始化函数, 针对设备设置SPI的工作时钟及数据传输模式等。spi_add_device函数中调用 
     transfer函数是实现SPI总线读写方法的函数。实现数据的双向传输,可能会睡眠

    cleanup注销时候调用

2.3 SPI设备驱动层

     SPI设备驱动层为用户接口层,其为用户提供了通过SPI总线访问具体设备的接口。
      SPI设备驱动层可以用两个模块来描述,structspi_driver和struct spi_device。
      相关的数据结构如下:

struct spi_driver {  
    int         (*probe)(struct spi_device *spi);  
    int         (*remove)(struct spi_device *spi);  
    void            (*shutdown)(struct spi_device *spi);  
    int         (*suspend)(struct spi_device *spi, pm_message_t mesg);  
    int         (*resume)(struct spi_device *spi);  
    struct device_driver    driver;  
}; 

  Driver是为device服务的,spi_driver注册时会扫描SPI bus上的设备,进行驱动和设备的绑定,probe函数用于驱动和设备匹配时被调用。从上面的结构体注释中我们可以知道SPI的通信是通过消息队列机制,而不是像I2C那样通过与从设备进行对话的方式。

struct spi_device {  
    struct device       dev;  
    struct spi_master   *master;  
    u32         max_speed_hz;  
    u8          chip_select;  
    u8          mode;    
    u8          bits_per_word;  
    int         irq;  
    void            *controller_state;  
    void            *controller_data;  
    char            modalias[32];   
}; 
 
        .modalias   = "m25p10",
        .mode   =SPI_MODE_0,   //CPOL=0, CPHA=0 此处选择具体数据传输模式
        .max_speed_hz    = 10000000, //最大的spi时钟频率
        /* Connected to SPI-0 as 1st Slave */
        .bus_num    = 0,   //设备连接在spi控制器0上
        .chip_select    = 0, //片选线号,在S5PC100的控制器驱动中没有使用它作为片选的依据,而是选择了下文controller_data里的方法。
       .controller_data = &smdk_spi0_csi[0],  
通常来说spi_device对应着SPI总线上某个特定的slave。并且spi_device封装了一个spi_master结构体。spi_device结构体包含了私有的特定的slave设备特性,包括它最大的频率,片选那个,输入输出模式等等

3 OMAP3630 SPI控制器

     OMAP3630上SPI是一个主/从的同步串行总线,这边有4个独立的SPI模块(SPI1,SPI2,SPI3,SPI4),各个模块之间的区别在于SPI1支持多达4个SPI设备,SPI2和SPI3支持2个SPI设备,而SPI4只支持1个SPI设备。

SPI控制器具有以下特征:

    1.可编程的串行时钟,包括频率,相位,极性。

    2.支持4到32位数据传输

    3.支持4通道或者单通道的从模式

    4.支持主的多通道模式

        4.1全双工/半双工

        4.2只发送/只接收/收发都支持模式

        4.3灵活的I/O端口控制

        4.4每个通道都支持DMA读写

    5.支持多个中断源的中断时间

    6.支持wake-up的电源管理

    7.内置64字节的FIFO

4 spi_device以下一系列的操作是在platform板文件中完成!

spi_device的板信息用spi_board_info结构体来描述:
struct spi_board_info {
charmodalias[SPI_NAME_SIZE];
const void*platform_data;
void*controller_data;
intirq;
u32max_speed_hz;
u16bus_num;
u16chip_select;
u8mode;
};
 
这个结构体记录了SPI外设使用的主机控制器序号、片选信号、数据比特率、SPI传输方式等构建的操作是以下的两个步骤:
1.spi_device就构建并注册 
static struct spi_board_info s3c_spi_devs[] __initdata = {
{
.modalias = "m25p10a",
.mode = SPI_MODE_0,
.max_speed_hz = 1000000,
.bus_num = 0,
.chip_select = 0,
.controller_data = &smdk_spi0_csi[SMDK_MMCSPI_CS],
},
};

2.
而这个info在init函数调用的时候会初始化:
spi_register_board_info(s3c_spi_devs,ARRAY_SIZE(s3c_spi_devs));
 

在板文件中添加spi_board_info,并在板文件的init函数中调用

spi_register_board_info(s3c_spi_devs,ARRAY_SIZE(s3c_spi_devs));//注册spi_board_info。这个代码会把spi_board_info注册到链表board_list上。spi_device封装了一个spi_master结构体,事实上spi_master的注册会在spi_register_board_info之后,spi_master注册的过程中会调用scan_boardinfo扫描board_list,找到挂接在它上面的spi设备,然后创建并注册spi_device。

 

至此spi_device就构建并注册完成了!!!!!!!!!!!!!

5 spi_driver的构建与注册

driver有几个重要的结构体:spi_driver、spi_transfer、spi_message

driver有几个重要的函数    spi_message_initspi_message_add_tailspi_sync

(1) spi_driver的构建
static struct spi_driver   m25p80_driver = { 
.driver = {
        .name   ="m25p80",
        .bus    =&spi_bus_type,
        .owner  = THIS_MODULE,
    },
    .probe  = m25p_probe,
    .remove =__devexit_p(m25p_remove),
};
(2)spi_driver的注册 
spi_register_driver(&m25p80_driver);//在有匹配的spi_device时,会调用m25p_probe

(3)实现probe操作
probe里完成了spi_transfer、spi_message的构建;
spi_message_init、spi_message_add_tail、spi_sync、spi_write_then_read函数的调用。

spi_transfer(里面集成了数据buf空间地址等信息)
spi_message(是spi_transfer的集合)的构建)
spi_message_init(初始化spi_message)
spi_message_add_tail(将新的spi_transfer添加到spi_message队列尾部)
spi_sync函数的调用(调用spi_master发送spi_message)
 spi_write_then_read(先写后读)

例如:
 
struct spi_transfer st={
 ………… 
};//填充spi_transfer  
struct spi_message meg;//定义message
spi_init_message(&meg);//初始化meg
spi_message_add_tail(&st,&meg);//将st放在message队列尾部
Spi_sync(spi_device,&meg);//将message与spi_device关联,发送meg
 
static int m25p10a_read( struct m25p10a *flash, loff_t from,   
        size_t len, char *buf )  
{  
    int r_count = 0, i;  
    struct spi_transfer st[2];  
    struct spi_message  msg;  
      
    spi_message_init( &msg );  
    memset( st, 0, sizeof(st) );  
  
    flash->cmd[0] = CMD_READ_BYTES;  
    flash->cmd[1] = from >> 16;  
    flash->cmd[2] = from >> 8;  
    flash->cmd[3] = from;  
  
    st[ 0 ].tx_buf = flash->cmd;  
    st[ 0 ].len = CMD_SZ;  
    spi_message_add_tail( &st[0], &msg );  
  
    st[ 1 ].rx_buf = buf;  
    st[ 1 ].len = len;  
    spi_message_add_tail( &st[1], &msg );  
  
    mutex_lock( &flash->lock );  
      
    /* Wait until finished previous write command. */  
    if (wait_till_ready(flash)) {  
        mutex_unlock( &flash->lock );  
        return -1;  
    }  
  
    spi_sync( flash->spi, &msg );  
    r_count = msg.actual_length - CMD_SZ;  
    printk( "in (%s): read %d bytes\n", __func__, r_count );  
    for( i = 0; i < r_count; i++ ) {  
        printk( "0x%02x\n", buf[ i ] );  
    }  
  
    mutex_unlock( &flash->lock );  

    return 0;  
}  

static int m25p10a_write( struct m25p10a *flash, loff_t to,   
        size_t len, const char *buf )  
{  
    int w_count = 0, i, page_offset;
    struct spi_transfer st[2]; 
    struct spi_message  msg;  
    write_enable( flash );  //写使能  
        spi_message_init( &msg );  
    memset( st, 0, sizeof(st) );  
  
    flash->cmd[0] = CMD_PAGE_PROGRAM;  
    flash->cmd[1] = to >> 16;  
    flash->cmd[2] = to >> 8;  
    flash->cmd[3] = to;  
  
    st[ 0 ].tx_buf = flash->cmd;  
    st[ 0 ].len = CMD_SZ;  
  //填充spi_transfer,将transfer放在队列后面
    spi_message_add_tail( &st[0], &msg );  
    st[ 1 ].tx_buf = buf;  
    st[ 1 ].len = len;  
    spi_message_add_tail( &st[1], &msg );  
        spi_sync( flash->spi, &msg );   调用spi_master发送spi_message
    
    return 0;  
} 
 
 
static int m25p10a_probe(struct spi_device *spi)   
{   
    int ret = 0;  
    struct m25p10a  *flash;  
    char buf[ 256 ];  
    flash = kzalloc( sizeof(struct m25p10a), GFP_KERNEL );  
    flash->spi = spi;  
    /* save flash as driver's private data */  
    spi_set_drvdata( spi, flash );    
    memset( buf, 0x7, 256 );  
    m25p10a_write( flash, 0, 20, buf); //0地址写入20个7  
    memset( buf, 0, 256 );  
    m25p10a_read( flash, 0, 25, buf ); //0地址读出25个数  
    return 0;   
}   
 
到目前为止,完成了SPI的驱动和应用。








相关实践学习
自建数据库迁移到云数据库
本场景将引导您将网站的自建数据库平滑迁移至云数据库RDS。通过使用RDS,您可以获得稳定、可靠和安全的企业级数据库服务,可以更加专注于发展核心业务,无需过多担心数据库的管理和维护。
Sqoop 企业级大数据迁移方案实战
Sqoop是一个用于在Hadoop和关系数据库服务器之间传输数据的工具。它用于从关系数据库(如MySQL,Oracle)导入数据到Hadoop HDFS,并从Hadoop文件系统导出到关系数据库。 本课程主要讲解了Sqoop的设计思想及原理、部署安装及配置、详细具体的使用方法技巧与实操案例、企业级任务管理等。结合日常工作实践,培养解决实际问题的能力。本课程由黑马程序员提供。
目录
相关文章
|
2月前
|
监控 Linux 开发者
理解Linux操作系统内核中物理设备驱动(phy driver)的功能。
综合来看,物理设备驱动在Linux系统中的作用是至关重要的,它通过与硬件设备的紧密配合,为上层应用提供稳定可靠的通信基础设施。开发一款优秀的物理设备驱动需要开发者具备深厚的硬件知识、熟练的编程技能以及对Linux内核架构的深入理解,以确保驱动程序能在不同的硬件平台和网络条件下都能提供最优的性能。
128 0
|
6月前
|
监控 Linux
Linux基础:文件和目录类命令分析。
总的来说,这些基础命令,像是Linux中藏匿的小矮人,每一次我们使用他们,他们就把我们的指令准确的传递给Linux,让我们的指令变为现实。所以,现在就开始你的Linux之旅,挥动你的命令之剑,探索这个充满神秘而又奇妙的世界吧!
128 19
|
7月前
|
缓存 网络协议 Linux
PCIe 以太网芯片 RTL8125B 的 spec 和 Linux driver 分析备忘
本文详细介绍了 Realtek RTL8125B PCIe 以太网芯片的规格以及在 Linux 中的驱动安装和配置方法。通过深入分析驱动源码,可以更好地理解其工作原理和优化方法。在实际应用中,合理配置和优化驱动程序可以显著提升网络性能和稳定性。希望本文能帮助您更好地使用和管理 RTL8125B,以满足各种网络应用需求。
640 33
|
7月前
|
数据管理 Linux iOS开发
Splunk Enterprise 9.4.1 (macOS, Linux, Windows) 发布 - 机器数据管理和分析
Splunk Enterprise 9.4.1 (macOS, Linux, Windows) 发布 - 机器数据管理和分析
126 0
Splunk Enterprise 9.4.1 (macOS, Linux, Windows) 发布 - 机器数据管理和分析
|
9月前
|
存储 运维 监控
Linux--深入理与解linux文件系统与日志文件分析
深入理解 Linux 文件系统和日志文件分析,对于系统管理员和运维工程师来说至关重要。文件系统管理涉及到文件的组织、存储和检索,而日志文件则记录了系统和应用的运行状态,是排查故障和维护系统的重要依据。通过掌握文件系统和日志文件的管理和分析技能,可以有效提升系统的稳定性和安全性。
211 7
|
9月前
|
监控 安全 Linux
启用Linux防火墙日志记录和分析功能
为iptables启用日志记录对于监控进出流量至关重要
278 1
|
10月前
|
缓存 算法 Linux
Linux内核中的调度策略优化分析####
本文深入探讨了Linux操作系统内核中调度策略的工作原理,分析了不同调度算法(如CFS、实时调度)在多核处理器环境下的性能表现,并提出了针对高并发场景下调度策略的优化建议。通过对比测试数据,展示了调度策略调整对于系统响应时间及吞吐量的影响,为系统管理员和开发者提供了性能调优的参考方向。 ####
|
存储 Unix Linux
浅入分析Linux
Linux 操作系统必须完成的两个主要目的 与硬件部分交互, 为包含在硬件平台上的所有底层可编程部件提供服务 为运行在计算机系统上的应用程序(即所谓的用户空间)提供执行环境 一些操作系统运行所有的用户程序都直接与硬件部分进行交互, 比如典型的MS-DOS。
1072 0
|
2月前
|
监控 Linux 网络安全
Linux命令大全:从入门到精通
日常使用的linux命令整理
626 13
|
3月前
|
Linux 网络安全 数据安全/隐私保护
使用Linux系统的mount命令挂载远程服务器的文件夹。
如此一来,你就完成了一次从你的Linux发车站到远程服务器文件夹的有趣旅行。在这个技术之旅中,你既探索了新地方,也学到了如何桥接不同系统之间的距离。
456 21