SPI是“Serial Peripheral Interface” 的縮寫,是一種四線制的同步串行通信接口,用來連接微控制器、傳感器、存儲設(shè)備,SPI設(shè)備分為主設(shè)備和從設(shè)備兩種,用于通信和控制的四根線分別是:
CS 片選信號
SCK 時鐘信號
MISO 主設(shè)備的數(shù)據(jù)輸入、從設(shè)備的數(shù)據(jù)輸出腳
MOSI 主設(shè)備的數(shù)據(jù)輸出、從設(shè)備的數(shù)據(jù)輸入腳
因?yàn)樵诖蠖鄶?shù)情況下,CPU或SOC一側(cè)通常都是工作在主設(shè)備模式,所以,目前的Linux內(nèi)核版本中,只實(shí)現(xiàn)了主模式的驅(qū)動框架。
一、硬件結(jié)構(gòu)
通常,負(fù)責(zé)發(fā)出時鐘信號的設(shè)備我們稱之為主設(shè)備,另一方則作為從設(shè)備,下圖是一個SPI系統(tǒng)的硬件連接示例:
如上圖所示,主設(shè)備對應(yīng)SOC芯片中的SPI控制器,通常,一個SOC中可能存在多個SPI控制器,像上面的例子所示,SOC芯片中有3個SPI控制器。每個控制器下可以連接多個SPI從設(shè)備,每個從設(shè)備有各自獨(dú)立的CS引腳。每個從設(shè)備共享另外3個信號引腳:SCK、MISO、MOSI。任何時刻,只有一個CS引腳處于有效狀態(tài),與該有效CS引腳連接的設(shè)備此時可以與主設(shè)備(SPI控制器)通信,其它的從設(shè)備處于等待狀態(tài),并且它們的3個引腳必須處于高阻狀態(tài)。
二、工作時序
按照時鐘信號和數(shù)據(jù)信號之間的相位關(guān)系,SPI有4種工作時序模式:
我們用CPOL表示時鐘信號的初始電平的狀態(tài),CPOL為0表示時鐘信號初始狀態(tài)為低電平,為1表示時鐘信號的初始電平是高電平。另外,我們用CPHA來表示在那個時鐘沿采樣數(shù)據(jù),CPHA為0表示在首個時鐘變化沿采樣數(shù)據(jù),而CPHA為1則表示要在第二個時鐘變化沿來采樣數(shù)據(jù)。內(nèi)核用CPOL和CPHA的組合來表示當(dāng)前SPI需要的工作模式:
CPOL=0,CPHA=1 模式0
CPOL=0,CPHA=1 模式1
CPOL=1,CPHA=0 模式2
CPOL=1,CPHA=1 模式3
三、確定驅(qū)動文件
SPI作為linux里面比較小的一個子系統(tǒng),其驅(qū)動程序位于/drivers/spi/*目錄,首先,我們可以通過Makefile及Kconfig來確定我們需要看的源文件。
[plain] view plain copy#
# Makefile for kernel SPI drivers.
#
# small core, mostly translating board-specific
# config declarations into driver model code
obj-$(CONFIG_SPI_MASTER) += spi.o
obj-$(CONFIG_SPI_SPIDEV) += spidev.o
[plain] view plain copy# SPI master controller drivers (bus)
obj-$(CONFIG_SPI_BITBANG) += spi-bitbang.o
obj-$(CONFIG_SPI_IMX) += spi-imx.o
對應(yīng)的Kconfig去配置內(nèi)核
編譯生成的目標(biāo)文件如下
通過以上分析我們知道,spi驅(qū)動由三部分組成,分別是core(spi.c),master controller driver (spi_imx.c)以及SPIprotocol drivers (spidev.c)。
四、數(shù)據(jù)結(jié)構(gòu)分析
Spi驅(qū)動涉及的數(shù)據(jù)結(jié)構(gòu)主要位于/include/linux/spi.h,其中spi.c,spi-imx.c,spidev.c均用到了spi.h里的結(jié)構(gòu)體。
1.spi_master
spi_master代表一個主機(jī)控制器,此處表示imx的SPI控制器。一般不需要自己編寫spi控制器驅(qū)動,但是了解這個結(jié)構(gòu)體還是必要的。
[objc] view plain copystruct spi_master {
struct device dev; //設(shè)備模型使用
struct list_head list;
/* other than negative (== assign one dynamically), bus_num is fully
* board-specific. usually that simplifies to being SOC-specific.
* example: one SOC has three SPI controllers, numbered 0..2,
* and one board‘s schematics might show it using SPI-2. software
* would normally use bus_num=2 for that controller.
*/
s16 bus_num; //總線(或控制器)編號,imx6q有5個spi控制器,0~4
/* chipselects will be integral to many controllers; some others
* might use board-specific GPIOs.
*/
u16 num_chipselect; //片選數(shù)量,決定該控制器下面掛接多少個SPI設(shè)備,從設(shè)備的片選號不能大于這個數(shù)量
/* some SPI controllers pose alignment requirements on DMAable
* buffers; let protocol drivers know about these requirements.
*/
u16 dma_alignment;
/* spi_device.mode flags understood by this controller driver */
u16 mode_bits; //master支持的設(shè)備模式
/* bitmask of supported bits_per_word for transfers */
u32 bits_per_word_mask;
#define SPI_BPW_MASK(bits) BIT((bits) - 1)
#define SPI_BIT_MASK(bits) (((bits) == 32) ? ~0U : (BIT(bits) - 1))
#define SPI_BPW_RANGE_MASK(min, max) (SPI_BIT_MASK(max) - SPI_BIT_MASK(min - 1))
/* limits on transfer speed */
u32 min_speed_hz;
u32 max_speed_hz;
/* other constraints relevant to this driver */
u16 flags;
#define SPI_MASTER_HALF_DUPLEX BIT(0) /* can’t do full duplex */
#define SPI_MASTER_NO_RX BIT(1) /* can‘t do buffer read */
#define SPI_MASTER_NO_TX BIT(2) /* can’t do buffer write */
#define SPI_MASTER_MUST_RX BIT(3) /* requires rx */
#define SPI_MASTER_MUST_TX BIT(4) /* requires tx */
/* lock and mutex for SPI bus locking */
spinlock_t bus_lock_spinlock;
struct mutex bus_lock_mutex;
/* flag indicating that the SPI bus is locked for exclusive use */
bool bus_lock_flag;
/* Setup mode and clock, etc (spi driver may call many times)。
*
* IMPORTANT: this may be called when transfers to another
* device are active. DO NOT UPDATE SHARED REGISTERS in ways
* which could break those transfers.
*/
int (*setup)(struct spi_device *spi); //根據(jù)spi設(shè)備更新硬件配置。設(shè)置模式、時鐘等,這個需要我們自己具體實(shí)現(xiàn),主要設(shè)置SPI控制器和工作方式
/* bidirectional bulk transfers
*
* + The transfer() method may not sleep; its main role is
* just to add the message to the queue.
* + For now there‘s no remove-from-queue operation, or
* any other request management
* + To a given spi_device, message queueing is pure fifo
*
* + The master’s main job is to process its message queue,
* selecting a chip then transferring data
* + If there are multiple spi_device children, the i/o queue
* arbitration algorithm is unspecified (round robin, fifo,
* priority, reservations, preemption, etc)
*
* + Chipselect stays active during the entire message
* (unless modified by spi_transfer.cs_change != 0)。
* + The message transfers use clock and SPI mode parameters
* previously established by setup() for this device
*/
int (*transfer)(struct spi_device *spi,
struct spi_message *mesg); //添加消息到隊(duì)列的方法。這個函數(shù)不可睡眠。它的職責(zé)是安排發(fā)生的傳送并且調(diào)用注冊的回調(diào)函數(shù)complete()。這個不同的控制器要具體實(shí)現(xiàn),傳輸數(shù)據(jù)最后都要調(diào)用這個函數(shù)
/* called on release() to free memory provided by spi_master */
void (*cleanup)(struct spi_device *spi); //cleanup函數(shù)會在spidev_release函數(shù)中被調(diào)用,spidev_release被登記為spi dev的release函數(shù)。
/*
* Used to enable core support for DMA handling, if can_dma()
* exists and returns true then the transfer will be mapped
* prior to transfer_one() being called. The driver should
* not modify or store xfer and dma_tx and dma_rx must be set
* while the device is prepared.
*/
bool (*can_dma)(struct spi_master *master,
struct spi_device *spi,
struct spi_transfer *xfer);
/*
* These hooks are for drivers that want to use the generic
* master transfer queueing mechanism. If these are used, the
* transfer() function above must NOT be specified by the driver.
* Over time we expect SPI drivers to be phased over to this API.
*/
bool queued;
struct kthread_worker kworker;
struct task_struct *kworker_task;
struct kthread_work pump_messages;
spinlock_t queue_lock;
struct list_head queue;
struct spi_message *cur_msg;
bool busy;
bool running;
bool rt;
bool auto_runtime_pm;
bool cur_msg_prepared;
bool cur_msg_mapped;
struct completion xfer_completion;
size_t max_dma_len;
int (*prepare_transfer_hardware)(struct spi_master *master);
int (*transfer_one_message)(struct spi_master *master,
struct spi_message *mesg);
int (*unprepare_transfer_hardware)(struct spi_master *master);
int (*prepare_message)(struct spi_master *master,
struct spi_message *message);
int (*unprepare_message)(struct spi_master *master,
struct spi_message *message);
/*
* These hooks are for drivers that use a generic implementation
* of transfer_one_message() provied by the core.
*/
void (*set_cs)(struct spi_device *spi, bool enable);
int (*transfer_one)(struct spi_master *master, struct spi_device *spi,
struct spi_transfer *transfer);
/* gpio chip select */
int *cs_gpios;
/* DMA channels for use with core dmaengine helpers */
struct dma_chan *dma_tx;
struct dma_chan *dma_rx;
/* dummy data for full duplex devices */
void *dummy_rx;
void *dummy_tx;
};
2. spi_device
spi_device代表一個外圍spi設(shè)備,由master controller driver注冊完成后掃描BSP中注冊設(shè)備產(chǎn)生的設(shè)備鏈表并向spi_bus注冊產(chǎn)生。在內(nèi)核中,每個spi_device代表一個物理的spi設(shè)備。
[objc] view plain copystruct spi_device {
struct device dev; //設(shè)備模型使用
struct spi_master *master; //設(shè)備使用的master結(jié)構(gòu),掛在哪個主控制器下
u32 max_speed_hz; //通訊時鐘最大頻率
u8 chip_select; //片選號,每個master支持多個spi_device
u16 mode; //設(shè)備支持的模式,如片選是高or低?
#define SPI_CPHA 0x01 /* clock phase */
#define SPI_CPOL 0x02 /* clock polarity */
#define SPI_MODE_0 (0|0) /* (original MicroWire) */
#define SPI_MODE_1 (0|SPI_CPHA)
#define SPI_MODE_2 (SPI_CPOL|0)
#define SPI_MODE_3 (SPI_CPOL|SPI_CPHA)
#define SPI_CS_HIGH 0x04 /* chipselect active high? 為1時片選的有效信號是高電平*/
#define SPI_LSB_FIRST 0x08 /* per-word bits-on-wire 發(fā)送時低比特在前*/
#define SPI_3WIRE 0x10 /* SI/SO signals shared 輸入輸出信號使用同一根信號線*/
#define SPI_LOOP 0x20 /* loopback mode 回環(huán)模式*/
#define SPI_NO_CS 0x40 /* 1 dev/bus, no chipselect */
#define SPI_READY 0x80 /* slave pulls low to pause */
#define SPI_TX_DUAL 0x100 /* transmit with 2 wires */
#define SPI_TX_QUAD 0x200 /* transmit with 4 wires */
#define SPI_RX_DUAL 0x400 /* receive with 2 wires */
#define SPI_RX_QUAD 0x800 /* receive with 4 wires */
u8 bits_per_word; //每個字長的比特數(shù),默認(rèn)是8
int irq; //中斷號
void *controller_state; //控制寄存器狀態(tài)
void *controller_data;
char modalias[SPI_NAME_SIZE]; //設(shè)備驅(qū)動的名字
int cs_gpio; /* chip select gpio */
/*
* likely need more hooks for more protocol options affecting how
* the controller talks to each chip, like:
* - memory packing (12 bit samples into low bits, others zeroed)
* - priority
* - drop chipselect after each word
* - chipselect delays
* - 。。。
*/
};
由于一個SPI總線上可以有多個SPI設(shè)備,因此需要片選號來區(qū)分它們,SPI控制器根據(jù)片選號來選擇不同的片選線,從而實(shí)現(xiàn)每次只同一個設(shè)備通信。
spi_device的mode成員有兩個比特位含義很重要。SPI_CPHA選擇對數(shù)據(jù)線采樣的時機(jī),0選擇每個時鐘周期的第一個沿跳變時采樣數(shù)據(jù),1選擇第二個時鐘沿采樣數(shù)據(jù);SPI_CPOL選擇每個時鐘周期開始的極性,0表示時鐘以低電平開始,1選擇高電平開始。這兩個比特有四種組合,對應(yīng)SPI_MODE_0~SPI_MODE_3。
另一個比較重要的成員是bits_per_word。這個成員指定每次讀寫的字長,單位是比特。雖然大部分SPI接口的字長是8或者16,仍然會有一些特殊的例子。需要說明的是,如果這個成員為零的話,默認(rèn)使用8作為字長。
最后一個成員并不是設(shè)備的名字,而是需要綁定的驅(qū)動的名字。
3.spi_driver
spi_driver代表一個SPI protocol drivers,即外設(shè)驅(qū)動。
[objc] view plain copystruct spi_driver {
const struct spi_device_id *id_table; //支持的spi_device設(shè)備表
int (*probe)(struct spi_device *spi); //和spi匹配成功之后會調(diào)用這個方法。因此這個方法需要對設(shè)備和私有數(shù)據(jù)進(jìn)行初始化。
int (*remove)(struct spi_device *spi); //解除spi_device和spi_driver的綁定,釋放probe申請的資源。
void (*shutdown)(struct spi_device *spi); //關(guān)閉
int (*suspend)(struct spi_device *spi, pm_message_t mesg); //掛起
int (*resume)(struct spi_device *spi); //恢復(fù)
struct device_driver driver; //設(shè)備模型使用
};
通常對于從事Linux驅(qū)動工作人員來說,spi設(shè)備的驅(qū)動主要就是實(shí)現(xiàn)這個結(jié)構(gòu)體中的各個接口,并將之注冊到spi子系統(tǒng)中去。
4.spi_transfer
spi_transfer代表一個讀寫緩沖對,包含接收緩沖區(qū)及發(fā)送緩沖區(qū),其實(shí),spi_transfer的發(fā)送是通過構(gòu)建spi_message實(shí)現(xiàn),通過將spi_transfer中的鏈表transfer_list鏈接到spi_message中的transfers,再以spi_message形勢向底層發(fā)送數(shù)據(jù)。每個spi_transfer都可以對傳輸?shù)囊恍?a target="_blank">參數(shù)進(jìn)行設(shè)置,使得master controller按照它要求的參數(shù)進(jìn)行數(shù)據(jù)發(fā)送。
[cpp] view plain copystruct spi_transfer {
/* it‘s ok if tx_buf == rx_buf (right?)
* for MicroWire, one buffer must be null
* buffers must work with dma_*map_single() calls, unless
* spi_message.is_dma_mapped reports a pre-existing mapping
*/
const void *tx_buf; //發(fā)送緩沖區(qū),要寫入設(shè)備的數(shù)據(jù)(必須是dma_safe),或者為NULL。
void *rx_buf; //接收緩沖區(qū),要讀取的數(shù)據(jù)緩沖(必須是dma_safe),或者為NULL。
unsigned len; //緩沖區(qū)長度,tx和rx的大小(字節(jié)數(shù))。這里不是指它的和,而是各自的長度,它們總是相等的。
dma_addr_t tx_dma; //如果spi_message.is_dma_mapped是真,這個是tx的dma地址
dma_addr_t rx_dma; //如果spi_message.is_dma_mapped是真,這個是rx的dma地址
struct sg_table tx_sg;
struct sg_table rx_sg;
unsigned cs_change:1; //當(dāng)前spi_transfer發(fā)送完成之后重新片選。影響此次傳輸之后的片選。指示本次transfer結(jié)束之后是否要重新片選并調(diào)用setup改變設(shè)置。這個標(biāo)志可以減少系統(tǒng)開銷。
u8 tx_nbits;
u8 rx_nbits;
#define SPI_NBITS_SINGLE 0x01 /* 1bit transfer */
#define SPI_NBITS_DUAL 0x02 /* 2bits transfer */
#define SPI_NBITS_QUAD 0x04 /* 4bits transfer */
u8 bits_per_word; //每個字長的比特數(shù),0代表使用spi_device中的默認(rèn)值8
u16 delay_usecs; //發(fā)送完成一個spi_transfer后延時時間,此次傳輸結(jié)束和片選改變之間的延時,之后就會啟動另一個傳輸或者結(jié)束整個消息
u32 speed_hz; //通信時鐘。如果是0,使用默認(rèn)值
struct list_head transfer_list; //用于鏈接到spi_message,用來連接的雙向鏈接節(jié)點(diǎn)
};《/span》
5.spi_message
spi_message代表spi消息,由多個spi_transfer段組成。
spi_message用來原子的執(zhí)行spi_transfer表示的一串?dāng)?shù)組傳輸請求。
這個傳輸隊(duì)列是原子的,這意味著在這個消息完成之前不會有其它消息占用總線。
消息的執(zhí)行總是按照FIFO的順序。
向底層提交spi_message的代碼要負(fù)責(zé)管理它的內(nèi)存空間。未顯示初始化的內(nèi)存需要使用0來初始化。
[cpp] view plain copystruct spi_message {
struct list_head transfers; //spi_transfer鏈表隊(duì)列,此次消息的傳輸段隊(duì)列,一個消息可以包含多個傳輸段。
struct spi_device *spi; //傳輸?shù)哪康脑O(shè)備
unsigned is_dma_mapped:1; //如果為真,此次調(diào)用提供dma和cpu虛擬地址。
/* REVISIT: we might want a flag affecting the behavior of the
* last transfer 。。。 allowing things like “read 16 bit length L”
* immediately followed by “read L bytes”。 Basically imposing
* a specific message scheduling algorithm.
*
* Some controller drivers (message-at-a-time queue processing)
* could provide that as their default scheduling algorithm. But
* others (with multi-message pipelines) could need a flag to
* tell them about such special cases.
*/
/* completion is reported through a callback */
void (*complete)(void *context); //異步調(diào)用完成后的回調(diào)函數(shù)
void *context; //回調(diào)函數(shù)的參數(shù)
unsigned frame_length;
unsigned actual_length; //《span style=“font-family: Arial, Helvetica, sans-serif;”》實(shí)際傳輸?shù)臄?shù)據(jù)長度《/span》
int status; //該消息的發(fā)送結(jié)果,成功被置0,否則是一個負(fù)的錯誤碼。
/* for optional use by whatever driver currently owns the
* spi_message 。。。 between calls to spi_async and then later
* complete(), that’s the spi_master controller driver.
*/
struct list_head queue; //下面兩個成員是給擁有本消息的驅(qū)動選用的。spi_master會使用它們。自己最好不要使用。
void *state;
};
控制器驅(qū)動會先寫入tx的數(shù)據(jù),然后讀取同樣長度的數(shù)據(jù)。長度指示是len。
如果tx_buff是空指針,填充rx_buff的時候會輸出0(為了產(chǎn)生接收的時鐘),如果rx_buff是NULL,接收到的數(shù)據(jù)將被丟棄。
只有l(wèi)en長讀的數(shù)據(jù)會被輸出和接收。
輸出不完整的字長是錯誤的(比如字長為2字節(jié)的時候輸出三個字節(jié),最后一個字節(jié)湊不成一個整字)。
本地內(nèi)存中的數(shù)據(jù)總是使用本地cpu的字節(jié)序,無論spi的字節(jié)序是大段模式還是小段模式(使用SPI_LSB_FIRS)
當(dāng)spi_transfer的字長不是8bit的2次冪的整數(shù)倍,這些數(shù)據(jù)字就包含擴(kuò)展位。在spi通信驅(qū)動看來內(nèi)存中的數(shù)據(jù)總是剛好對齊的,所以rx中位定義和rx中未使用的比特位總是最高有效位。(比如13bit的字長,每個字占2字節(jié),rx和tx都應(yīng)該如此存放)
所有的spi傳輸都以使能相關(guān)的片選線為開始。一般來說片選線在本消息結(jié)束之前保持有效的狀態(tài)。驅(qū)動可以使用
spi_transfer中的cs_change成員來影響片選:
(i)如果transfer不是message的最后一個,這個標(biāo)志量可以方便的將片選線置位無效的狀態(tài)。
有時需要這種方法來告知芯片一個命令的結(jié)束并使芯片完成這一批處理任務(wù)。
(ii)當(dāng)這個trasfer是最后一個時,片選可以一直保持有效知道下一個transfer到來。
在多spi從機(jī)的總線上沒有辦法阻止其他設(shè)備接收數(shù)據(jù),這種方法可以作為一個特別的提示;開始往另一個設(shè)備傳輸信息就要先將本芯片的片選置為無效。但在其他情況下,這可以保證正確性。一些設(shè)備后面的信息依賴于前面的信息并且在一個處理序列完成后需要禁用片選線。
上面這段是翻譯的,講的不明白。
再說一下:cs_change影響此transfer完成后是否禁用片選線并調(diào)用setup改變配置。(這個標(biāo)志量就是chip select change片選改變的意思)
沒有特殊情況,一個spi_message應(yīng)該只在最后一個transfer置位該標(biāo)志量。
6.spi_board_info
spi_device的板信息用spi_board_info結(jié)構(gòu)體描述,該結(jié)構(gòu)體記錄著SPI外設(shè)使用的主機(jī)控制器序號、片選序號、數(shù)據(jù)比特率、SPI傳輸模式(即CPOL、CPHA)等。ARM Linux3.x之后的內(nèi)核在改為設(shè)備樹之后,不再需要在arch/arm/mach-xxx中編碼SPI的板級信息了,而傾向于在SPI控制器節(jié)點(diǎn)下填寫子節(jié)點(diǎn)。
[cpp] view plain copystruct 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[SPI_NAME_SIZE];
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.
*/
u16 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
*/
};
7.spi_bitbang
spi_bitbang是具體的負(fù)責(zé)信息傳輸?shù)臄?shù)據(jù)結(jié)構(gòu),它維護(hù)一個workqueue_struct,每收到一個消息,都會向其中添加一個work_struct,由內(nèi)核守護(hù)進(jìn)程在將來的某個時間調(diào)用該work_struct中的function進(jìn)行消息發(fā)送。
[cpp] view plain copystruct spi_bitbang {
spinlock_t lock;
u8 busy; //忙標(biāo)志
u8 use_dma;
u8 flags; /* extra spi-》mode support */
struct spi_master *master;
/* setup_transfer() changes clock and/or wordsize to match settings
* for this transfer; zeroes restore defaults from spi_device.
*/
int (*setup_transfer)(struct spi_device *spi,
struct spi_transfer *t); //對數(shù)據(jù)傳輸進(jìn)行設(shè)置
void (*chipselect)(struct spi_device *spi, int is_on); //控制片選
#define BITBANG_CS_ACTIVE 1 /* normally nCS, active low */
#define BITBANG_CS_INACTIVE 0
/* txrx_bufs() may handle dma mapping for transfers that don’t
* already have one (transfer.{tx,rx}_dma is zero), or use PIO
*/
int (*txrx_bufs)(struct spi_device *spi, struct spi_transfer *t); //實(shí)際的數(shù)據(jù)傳輸函數(shù)
/* txrx_word[SPI_MODE_*]() just looks like a shift register */
u32 (*txrx_word[4])(struct spi_device *spi,
unsigned nsecs,
u32 word, u8 bits);
};
評論
查看更多