SPI知識:
1)高速同步串行口。3~4線接口(CS ,CLK ,MOSI,MISO),收發(fā)獨立、可同步進行。
本文引用地址:http://m.butianyuan.cn/article/201611/322986.htm
2)SPI分為主從模式,主模式提供時鐘和片選選擇信號.
3) 模式控制:CPOL用來控制時鐘信號(clk)在空閑時候的狀態(tài);CPHA用來控制采樣時刻時CLK的邊緣動作。
CPOLCPHA模式
00CLK空閑時為低電平,CLK上升沿采樣數(shù)據(jù)。
01CLK空閑為低電平,CLK下降沿采樣數(shù)據(jù)。
10CLK空閑時為高電平,CLK上升沿采樣數(shù)據(jù)。
11CLK空閑時為高電平,CLK下降沿采樣數(shù)據(jù)。
1)SPI配置(3.01庫):
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//雙工模式
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;//SPI主模式
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; //8bit數(shù)據(jù)
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;//CLK空閑時為高電平
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;//CLK上升沿采樣,因為上升沿是第二個邊沿動作,所以也可以理解為第二個邊沿采樣
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;//片選用軟件控制
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4;//SPI頻率
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;//高位在前
SPI_InitStructure.SPI_CRCPolynomial = 7;//crc7,stm32spi帶硬件ecc
SPI_Init(SPI1, &SPI_InitStructure);
2)CS信號:
主模式下要為從設備提供片選信號,值得注意的是STM32的主頻相當較高,要提防數(shù)據(jù)沒有完全發(fā)送前拉高CS信號。
3)SPI讀寫:(非中斷模式)
a)寫一個字節(jié):
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET);
//確保發(fā)生前Buffer為空,也就是說上一次已經(jīng)發(fā)生完成
SPI_I2S_SendData(SPI1, Data);//往寄存器中寫入一個字節(jié)
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET);
//等待接受到一個字節(jié)數(shù)據(jù),為什么要這么做?加這一句的原因是為了確保這個字節(jié)已經(jīng)發(fā)送出去,因為發(fā)生和接受是并行同步進行,那就是說你發(fā)生出去一個字節(jié)意味著你收到一個字節(jié)。所以這樣判斷完全沒有問題,再說必要性,如果你不加這句你就會容易犯過早拉高CS信號的錯誤,你想想如果在SPI_I2S_SendData(SPI1, Data)后面立即拉高CS是什么后果。
SPI_I2S_ReceiveData(SPI1);//都會接收到的數(shù)據(jù),看起來沒什么必要,但以用stm32的經(jīng)驗推薦這樣做,也許會有意想不到的收獲。
SPI_Writebyte(u8 data)
{
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPI1, Data);
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET);
SPI_I2S_ReceiveData(SPI1);
}
b)讀一個字節(jié):
讀的時候要注意一個問題,因為從模式是沒法提供時鐘的,所以主模式下必須要在接收的同時提供時鐘。辦法就是發(fā)送一個字節(jié)來實現(xiàn),因為還是上面說的,發(fā)送一個字節(jié)就意味著收到一個字節(jié),代碼和寫完全一樣,只要把讀出來的字節(jié)保存即可。
u8SPI_Readbyte(u8 data)
{
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPI1, Data);
while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET);
return SPI_I2S_ReceiveData(SPI1);
}
總結(jié):上面的程序是最求穩(wěn)定而設定的,如果你對速度有要求,你可以做相應的精簡,比如讀寫直接對寄存器進行操作,另外配置SPI前要對從模式的模式了解清楚,包括從設備支持的時鐘范圍和模式(CPOL,CPHA狀態(tài))。
評論