介绍spi基本协议,以及在linux内核中spi驱动程序框架。
SPI驱动基于标准了总线设备驱动模型,因此本文仅示意分析SPI框架,驱动实现细节按照总线设备驱动模型来实现,不做过多分析。
在一些flash中经常使用SPI协议,如下,SPI一般有四个线:
根据引脚推断,有片选引脚,所以说一条SPI总线上可以挂多个设备,哪个CS被选中,则哪个设备活跃。同时有数据输入和输出引脚,因此是全双工。


根据SCK没数据时候的状态,以及上升下降沿采样,可以分为四种模式,SPI0-3。使用SPI进行通信的时候,需要根据具体的从设备手册,确定是支持哪些模式。

内核中SPI遵循总线设备驱动模型。因此驱动包含几个部分,总线的注册,控制器的注册及设备创建,驱动的注册。即总线设备驱动模型。
通过config增加SPI支持后,SPI总线被注册进内核,spi_init调用bus_register。
每个CPU有自己的SPI控制器,在设备树中描述,同时在设备树SPI控制器节点下存在该控制器下连接的设备。由控制器驱动直接创建出spi_device,spi_new_device。
那么最后只差一个spi_driver。spidev.c是一个通用的驱动,调用spi_register_driver注册驱动。
围绕以下结构体:
spi_bus_type(总线)、spi_master(控制器)、spi_device(设备)、spi_driver(驱动)
在总线设备驱动模型框架下,追踪以上结构体分配、设置过程就是SPI的框架实现。
-
- ecspi1: ecspi@02008000 {
- #address-cells = <1>;
- #size-cells = <0>;
- compatible = "fsl,imx6ul-ecspi", "fsl,imx51-ecspi";
- reg = <0x02008000 0x4000>;
- interrupts = <GIC_SPI 31 IRQ_TYPE_LEVEL_HIGH>;
- clocks = <&clks IMX6UL_CLK_ECSPI1>,
- <&clks IMX6UL_CLK_ECSPI1>;
- clock-names = "ipg", "per";
- dmas = <&sdma 3 7 1>, <&sdma 4 7 2>;
- dma-names = "rx", "tx";
- status = "disabled";
- };
-
- &ecspi1 {
- fsl,spi-num-chipselects = <1>;
- cs-gpios = <&gpio4 26 0>;
- pinctrl-names = "default";
- pinctrl-0 = <&pinctrl_ecspi1_1 &pinctrl_ecspi1_cs_1>;
- status = "okay";
-
- flash: n25q032@0 {
- #address-cells = <1>;
- #size-cells = <1>;
- compatible = "st,n25q032";
- spi-max-frequency = <20000000>;
- reg = <0>;
- };
- };





