SPI通信原理---STM32F4--HAL

SPI接口原理

SPI是一種高速全雙工同步通信,在芯片管腳上占用四根線,主要應用在EEPROM、FLASH、實時時鐘、AD轉換器,還有數字信號處理器和數字信號解碼器之間。

在這里插入圖片描述
SPI接口使用4根線通信。

  • MISO:主設備數據輸入,從設備數據輸出
  • MOSI:主設備數據輸出,從設備數據輸入
  • SCLK:時鐘信號,由主設備產生
  • CS:片選信號,由主設備控制
工作原理
  1. 主機和從機都有一個串行移位寄存器,主機通過向他的SPI串行寄存器寫入一個字節來發起一次傳輸
  2. 串行移位寄存器通過MOSI信號線將字節傳送給從機,從機將自己的串行移位寄存器中的內容通過MISO信號線返回給主機
  3. 外設的寫操作和讀操作都是同步完成的。如果只進行寫操作,主機只需忽略接收到的字節,反之,若主機要讀取從機的一個字節,就必須發送一個空字節來引發從機的傳輸
時鐘信號的相位

SPI_CR寄存器的CPOL和CPHA位,能夠組合成四種可能的時序關系,如果CPOL位為0,SCK引腳在空閑狀態保持低電平,如果CPOL=1 ,SCK引腳在空閑狀態下保持高電平。
如果CPHA=0,在串行同步時鐘的第一個跳變沿(上升或下降)數據被采樣;如果CPHA=1,在串行同步時鐘的第二個跳變沿(上升或下降)數據被采樣

在這里插入圖片描述

數據幀格式

根據SPI_CR1寄存器中的LSBFIRST位,輸出數據時可以MSB優先,也可以LSB優先
根據SPI_CR1寄存器的DFF位,每個數據幀可以是8位或是16位

程序配置過程

我們使用SPI和w25Q256通信,硬件連接為
在這里插入圖片描述

  1. 使能SPIx和IO時鐘
  2. 初始化IO口復用映射
  3. 初始化SIPx,設置SPIx工作模式
  4. 使能SPIx
  5. SPI數據傳輸

具體代碼實現

SPI_HandleTypeDef SPI5_Handler;  //SPI句柄//以下是SPI模塊的初始化代碼,配置成主機模式 						  
//SPI口初始化
//這里針是對SPI5的初始化
void SPI5_Init(void)
{SPI5_Handler.Instance=SPI5;                         //SP5SPI5_Handler.Init.Mode=SPI_MODE_MASTER;             //設置SPI工作模式,設置為主模式SPI5_Handler.Init.Direction=SPI_DIRECTION_2LINES;   //設置SPI單向或者雙向的數據模式:SPI設置為雙線模式SPI5_Handler.Init.DataSize=SPI_DATASIZE_8BIT;       //設置SPI的數據大小:SPI發送接收8位幀結構SPI5_Handler.Init.CLKPolarity=SPI_POLARITY_HIGH;    //串行同步時鐘的空閑狀態為高電平SPI5_Handler.Init.CLKPhase=SPI_PHASE_2EDGE;         //串行同步時鐘的第二個跳變沿(上升或下降)數據被采樣SPI5_Handler.Init.NSS=SPI_NSS_SOFT;                 //NSS信號由硬件(NSS管腳)還是軟件(使用SSI位)管理:內部NSS信號有SSI位控制SPI5_Handler.Init.BaudRatePrescaler=SPI_BAUDRATEPRESCALER_256;//定義波特率預分頻的值:波特率預分頻值為256SPI5_Handler.Init.FirstBit=SPI_FIRSTBIT_MSB;        //指定數據傳輸從MSB位還是LSB位開始:數據傳輸從MSB位開始SPI5_Handler.Init.TIMode=SPI_TIMODE_DISABLE;        //關閉TI模式SPI5_Handler.Init.CRCCalculation=SPI_CRCCALCULATION_DISABLE;//關閉硬件CRC校驗SPI5_Handler.Init.CRCPolynomial=7;                  //CRC值計算的多項式HAL_SPI_Init(&SPI5_Handler);//初始化__HAL_SPI_ENABLE(&SPI5_Handler);                    //使能SPI5SPI5_ReadWriteByte(0Xff);                           //啟動傳輸
}//SPI5底層驅動,時鐘使能,引腳配置
//此函數會被HAL_SPI_Init()調用
//hspi:SPI句柄
void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi)
{GPIO_InitTypeDef GPIO_Initure;__HAL_RCC_GPIOF_CLK_ENABLE();       //使能GPIOF時鐘__HAL_RCC_SPI5_CLK_ENABLE();        //使能SPI5時鐘//PF7,8,9GPIO_Initure.Pin=GPIO_PIN_7|GPIO_PIN_8|GPIO_PIN_9;GPIO_Initure.Mode=GPIO_MODE_AF_PP;              //復用推挽輸出GPIO_Initure.Pull=GPIO_PULLUP;                  //上拉GPIO_Initure.Speed=GPIO_SPEED_FAST;             //快速            GPIO_Initure.Alternate=GPIO_AF5_SPI5;           //復用為SPI5HAL_GPIO_Init(GPIOF,&GPIO_Initure);
}

先使用HAL_SPI_Init函數對SPI進行初始化,注意我們只初始化PF7、PF8、PF9,也就是SPI的SCK線,MISO線和MOSI線,CS線還沒有初始化,HAL_SPI_MspInit是HAL_SPI_Init的回調函數,我們在這里初始化GPIO以及使能。上面的代碼完成了第1到4步。接下來我們就可以進行數據傳輸了。

W25Q256

W25Q256是容量為32M字節的串行Flash芯片,它將32M的容量分為512塊(Block),每個塊大小為64K字節,每個塊又分為16個扇區(sector),每個扇區4K字節,W25Q256最小擦除單位為一個扇區,也就是每次必須擦除4K個字節。
在這里插入圖片描述

在這里插入圖片描述

W25QXX_Write函數思路
  1. 根據要寫的起始地址,確定要寫的起始區域Sector號以及在起始sector中的偏移量
  2. 根據要寫的起始地址和字節數,確定要寫的數據是否跨sector
  3. 確定好要操作的sector以及sector的地址范圍
  4. 對每一個sector,先遍歷要寫的地址區域保存的數據是不是0xFF。如果都是,就不用擦除,如果有不是0xff的區域,先讀出里面的數據,保存在緩存buffer中,然后擦除里面的數據,把這個sector要操作的數據,寫到緩存,最后一次性把緩存buffer寫到這個對應的sector中

具體代碼實現

//寫SPI FLASH  
//在指定地址開始寫入指定長度的數據
//該函數帶擦除操作!
//pBuffer:數據存儲區
//WriteAddr:開始寫入的地址(24bit)						
//NumByteToWrite:要寫入的字節數(最大65535)   
u8 W25QXX_BUFFER[4096];		 
void W25QXX_Write(u8* pBuffer,u32 WriteAddr,u16 NumByteToWrite)   
{ u32 secpos;u16 secoff;u16 secremain;	   u16 i;    u8 * W25QXX_BUF;	  W25QXX_BUF=W25QXX_BUFFER;	     secpos=WriteAddr/4096;//扇區地址  secoff=WriteAddr%4096;//在扇區內的偏移secremain=4096-secoff;//扇區剩余空間大小   //printf("ad:%X,nb:%X\r\n",WriteAddr,NumByteToWrite);//測試用if(NumByteToWrite<=secremain)secremain=NumByteToWrite;//不大于4096個字節while(1) {	W25QXX_Read(W25QXX_BUF,secpos*4096,4096);//讀出整個扇區的內容for(i=0;i<secremain;i++)//校驗數據{if(W25QXX_BUF[secoff+i]!=0XFF)break;//需要擦除  	  }if(i<secremain)//需要擦除{W25QXX_Erase_Sector(secpos);//擦除這個扇區for(i=0;i<secremain;i++)	   //復制{W25QXX_BUF[i+secoff]=pBuffer[i];	  }W25QXX_Write_NoCheck(W25QXX_BUF,secpos*4096,4096);//寫入整個扇區  }else W25QXX_Write_NoCheck(pBuffer,WriteAddr,secremain);//寫已經擦除了的,直接寫入扇區剩余區間. 				   if(NumByteToWrite==secremain)break;//寫入結束了else//寫入未結束{secpos++;//扇區地址增1secoff=0;//偏移位置為0 	 pBuffer+=secremain;  //指針偏移WriteAddr+=secremain;//寫地址偏移	   NumByteToWrite-=secremain;				//字節數遞減if(NumByteToWrite>4096)secremain=4096;	//下一個扇區還是寫不完else secremain=NumByteToWrite;			//下一個扇區可以寫完了}	 };	 
}
  1. 根據要寫的起始地址,確定要寫的起始區域Sector號以及在起始sector中的偏移量
	secpos=WriteAddr/4096;//扇區地址  secoff=WriteAddr%4096;//在扇區內的偏移

每個扇區的大小是4K字節,也就是4094,除以4096就得到扇區的地址,模4096就得到在扇區里面開始寫的地址。

  1. 根據要寫的起始地址和字節數,確定要寫的數據是否跨扇區
secremain=4096-secoff;//扇區剩余空間大小   //printf("ad:%X,nb:%X\r\n",WriteAddr,NumByteToWrite);//測試用if(NumByteToWrite<=secremain)secremain=NumByteToWrite;//不大于4096個字節

secremain是扇區剩余空間大小,NumByteToWrite是要寫入的字節數,如果NumByteToWrite<=secremain就不需要跨扇區。所以在后面有:

if(NumByteToWrite==secremain)break;//寫入結束了

不需要跨扇區,寫入結束,否則的話,就需要跨扇區,扇區號要加1,扇區偏移地址為0

secpos++;//扇區地址增1
secoff=0;//偏移位置為0 	 
  1. 對每一個sector,先遍歷要寫的地址區域保存的數據是不是0xFF。如果都是,就不用擦除,如果有不是0xff的區域,先讀出里面的數據,保存在緩存buffer中,然后擦除里面的數據,把這個sector要操作的數據,寫到緩存,最后一次性把緩存buffer寫到這個對應的sector中
		W25QXX_Read(W25QXX_BUF,secpos*4096,4096);//讀出整個扇區的內容for(i=0;i<secremain;i++)//校驗數據{if(W25QXX_BUF[secoff+i]!=0XFF)break;//需要擦除  	  }if(i<secremain)//需要擦除{W25QXX_Erase_Sector(secpos);//擦除這個扇區for(i=0;i<secremain;i++)	   //復制{W25QXX_BUF[i+secoff]=pBuffer[i];	  }W25QXX_Write_NoCheck(W25QXX_BUF,secpos*4096,4096);//寫入整個扇區  }

W25QXX_Read先保存扇區數據到W25QXX_BUF中,然后遍歷剩余扇區數據有無不等于0xFF的,如果有,則調用W25QXX_Erase_Sector擦除整個扇區,然后將要寫的數據線寫到W25QXX_BUF中,最后一次性把W25QXX_BUF緩沖寫到扇區中。

如果需要跨扇區寫數據

else//寫入未結束{secpos++;//扇區地址增1secoff=0;//偏移位置為0 	 pBuffer+=secremain;  //指針偏移WriteAddr+=secremain;//寫地址偏移	   NumByteToWrite-=secremain;				//字節數遞減if(NumByteToWrite>4096)secremain=4096;	//下一個扇區還是寫不完else secremain=NumByteToWrite;			//下一個扇區可以寫完了}	

while會一直循環,直到寫入結束了

if(NumByteToWrite==secremain)break;//寫入結束了

W25QXX_Write就是在指定地址連續寫入NumByteToWrite個字節數據

讀取FLASH數據
//讀取SPI FLASH  
//在指定地址開始讀取指定長度的數據
//pBuffer:數據存儲區
//ReadAddr:開始讀取的地址(24bit)
//NumByteToRead:要讀取的字節數(最大65535)
void W25QXX_Read(u8* pBuffer,u32 ReadAddr,u16 NumByteToRead)   
{ u16 i;   										    W25QXX_CS=0;                            //使能器件   SPI5_ReadWriteByte(W25X_ReadData);      //發送讀取命令  if(W25QXX_TYPE==W25Q256)                //如果是W25Q256的話地址為4字節的,要發送最高8位{SPI5_ReadWriteByte((u8)((ReadAddr)>>24));    }SPI5_ReadWriteByte((u8)((ReadAddr)>>16));   //發送24bit地址    SPI5_ReadWriteByte((u8)((ReadAddr)>>8));   SPI5_ReadWriteByte((u8)ReadAddr);   for(i=0;i<NumByteToRead;i++){ pBuffer[i]=SPI5_ReadWriteByte(0XFF);    //循環讀數  }W25QXX_CS=1;  				    	      
}  

W25QXX_Read從ReadAddr地址連續讀取NumByteToRead個字節數據

main函數

我們寫入數據到FALSH中,然后讀取出來在LCD上顯示

//要寫入到W25Q16的字符串數組
const u8 TEXT_Buffer[]={"Apollo STM32F4 SPI TEST"};
#define SIZE sizeof(TEXT_Buffer)	 int main(void)
{u8 key;u16 i=0;u8 datatemp[SIZE];u32 FLASH_SIZE;HAL_Init();                     //初始化HAL庫   Stm32_Clock_Init(360,25,2,8);   //設置時鐘,180Mhzdelay_init(180);                //初始化延時函數uart_init(115200);              //初始化USARTLED_Init();                     //初始化LED KEY_Init();                     //初始化按鍵SDRAM_Init();                   //初始化SDRAMLCD_Init();                     //初始化LCDW25QXX_Init();				    //W25QXX初始化POINT_COLOR=RED;LCD_ShowString(30,50,200,16,16,"Apollo STM32F4/F7"); LCD_ShowString(30,70,200,16,16,"SPI TEST");	LCD_ShowString(30,90,200,16,16,"ATOM@ALIENTEK");LCD_ShowString(30,110,200,16,16,"2016/1/16");	 		LCD_ShowString(30,130,200,16,16,"KEY1:Write  KEY0:Read");	//顯示提示信息		while(W25QXX_ReadID()!=W25Q256)								//檢測不到W25Q256{LCD_ShowString(30,150,200,16,16,"W25Q256 Check Failed!");delay_ms(500);LCD_ShowString(30,150,200,16,16,"Please Check!        ");delay_ms(500);LED0=!LED0;		//DS0閃爍}LCD_ShowString(30,150,200,16,16,"W25Q256 Ready!"); FLASH_SIZE=32*1024*1024;	//FLASH 大小為32M字節POINT_COLOR=BLUE;			//設置字體為藍色	  while(1){key=KEY_Scan(0);if(key==KEY1_PRES)//KEY1按下,寫入W25Q128{LCD_Fill(0,170,239,319,WHITE);//清除半屏    LCD_ShowString(30,170,200,16,16,"Start Write W25Q256....");W25QXX_Write((u8*)TEXT_Buffer,FLASH_SIZE-100,SIZE);		//從倒數第100個地址處開始,寫入SIZE長度的數據LCD_ShowString(30,170,200,16,16,"W25Q256 Write Finished!");	//提示傳送完成}if(key==KEY0_PRES)//KEY0按下,讀取字符串并顯示{LCD_ShowString(30,170,200,16,16,"Start Read W25Q256.... ");W25QXX_Read(datatemp,FLASH_SIZE-100,SIZE);					//從倒數第100個地址處開始,讀出SIZE個字節LCD_ShowString(30,170,200,16,16,"The Data Readed Is:   ");	//提示傳送完成LCD_ShowString(30,190,200,16,16,datatemp);					//顯示讀到的字符串} i++;delay_ms(10);if(i==20){LED0=!LED0;//提示系統正在運行	i=0;}		   }		    
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/379779.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/379779.shtml
英文地址,請注明出處:http://en.pswp.cn/news/379779.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Linux 引導管理器 grub2 使用簡介

轉自&#xff1a;杜昌彬的空間 首先向其致敬&#xff01;有改動。 grub是Linux系統即其他類unix系統的主流bootloder&#xff0c;由于grub原來版本的設計存在很大缺陷&#xff0c;與以前的grub很不相同&#xff0c;其使用和配置也發生很大變化。現在很多Linux發行版本都使用了…

pata1015_ATA / PATA的完整形式是什么?

pata1015ATA / PATA&#xff1a;高級技術附件/并行高級技術附件 (ATA/PATA: Advanced Technology Attachment/Parallel Advanced Technology Attachment) ATA is an abbreviation of Advanced Technology Attachment. ATA has existed for a long time with the name PATA. Whe…

產品

總結一下&#xff1a;  1、核心功能要做透&#xff0c;做的人家追不上&#xff0c;自己的優勢要盡量的發揮&#xff1b;  2、產品口碑要建立&#xff0c;要關注高端用戶&#xff0c;要調整自己心態&#xff1b;  3、敏捷、快&#xff0c;產品迭代要快&#xff0c;快速實現…

FreeRTOS在STM32F429上移植

準備工作 FreeRTOS系統源碼基礎工程&#xff0c;這里我們用跑馬燈實驗 1.在工程里面添加FreeRTOS源碼 在工程里面新建一個名為FreeROTS的文件夾 將FreeRTOS源碼添加到這個文件夾里面 protable里面只需留下Keil、MemMang、RVDS文件夾 2、向工程分組中添加文件 FreeRTOS_C…

C++中的指針與引用(轉)

原文地址&#xff1a;http://www.cnblogs.com/skynet/archive/2010/09/22/1832911.html寫在前面 指針和引用形式上很好區別&#xff0c;但是他們似乎有相同的功能,都能夠直接引用對象&#xff0c;對其進行直接的操作。但是什么時候使用指針&#xff1f;什么時候使用引用呢&…

實訓09.11:數據庫一些簡單操作

new Database 新建數據庫 new Table 新建表 utf-8 編碼格式 primary key 主鍵&#xff1a;特點&#xff1a;在表中是唯一的不可重復的&#xff0c;一般都是學號&#xff0c;編號 auto increment 自增&#xff0c;一般都把主鍵設置為自增 allow nul…

c語言中將整數轉換成字符串_在C語言中將ASCII字符串(char [])轉換為八進制字符串(char [])...

c語言中將整數轉換成字符串Given an ASCII string (char[]) and we have to convert it into octal string (char[]) in C. 給定一個ASCII字符串(char [])&#xff0c;我們必須在C中將其轉換為八進制字符串(char [])。 Logic: 邏輯&#xff1a; To convert an ASCII string t…

Javascript的IE和Firefox兼容性匯編收藏.txt

document.form.item 問題 現有問題&#xff1a;現有代碼中存在許多 document.formName.item("itemName") 這樣的語句&#xff0c;不能在 MF 下運行 解決方法&#xff1a;改用 document.formName.elements["elementName"] 集合類對象問題 現有問題&#xff…

FreeRTOS系統配置文件FreeRTOSConfig.h

實際使用FreeRTOS的時候&#xff0c;我們時常需要根據自己需求來配置FreeRTOS&#xff0c;而且不同架構的MCU在使用的時候配置也不同&#xff0c;FreeRTOS的系統配置文件FreeRTOSConfig.h可以完成FreeRTOS的裁剪和配置。FreeRTOSConfig.h分成兩個部分&#xff0c;一個是INCLUDE…

SQL更新多條數據

問題&#xff1a;有兩個不同的表&#xff0c;其中都有一個編號的字段&#xff0c;而且存儲的內容是相同的&#xff0c;需要將一張表中的另外一些字段依據編號去與另一個表中編號對應來更新到另一個表中。 方法&#xff1a;由于在sql中是不支持同時更新多條包含編號的數據的&…

簡單的登錄系統(java+JFrame+Mysql)

連接數據庫 package 注冊信息; import java.sql.Connection; import java.sql.DriverManager; import java.sql.Statement; public class conn_db{ Connection con; String url null; Statement stmt; public void connection() throws ClassNotFoundException{ //…

冬季止咳化痰的飲食偏方集錦

1、蘿卜蔥白風寒咳嗽 蘿卜1個,蔥白6根,生姜15克.用水三碗先將蘿卜煮熟,再放蔥白,姜,煮剩一碗湯.連渣一次服.宣肺解表,化痰止咳.治風寒咳嗽,痰多泡沫,伴畏寒,身倦酸痛等. 2、紅糖姜棗湯治傷風咳嗽 紅糖30克,鮮姜15克,紅棗30克. 以水三碗煎至過半.頓服,服后出微汗即愈. 驅風散寒.…

c語言中數組越界怎么辦_如果我們使用C語言數組中的越界索引怎么辦?

c語言中數組越界怎么辦Let’s understand first, what is index out of bounds? 首先讓我們了解一下 &#xff0c; 什么是索引超出范圍&#xff1f; Let suppose you have an array with 5 elements then the array indexing will be from 0 to 4 i.e. we can access element…

FreeRTOS任務基礎知識

任務特性 在RTOS中&#xff0c;一個實時應用可以作為一個獨立的任務&#xff0c;支持搶占&#xff0c;支持優先級&#xff0c;每個任務都有自己的堆棧&#xff0c;當任務切換時將上下文環境保存在堆棧中&#xff0c;再次調用任務時&#xff0c;取出上下文信息&#xff0c;繼續…

測試Rockey 4 Smart加密鎖的C語言代碼

測試Rockey 4 Smart加密鎖的C語言代碼 // win32Console_dog_test.cpp : Defines the entry point for the console application. /// // //測試Rockey 4 Smart加密鎖的C語言代碼 // /// #include "stdafx.h" #include <conio.h> #include "time.h" #…

C——任意一個偶數分解兩個素數

題目&#xff1a;一個偶數總能表示為兩個素數之和 以上實例運行輸出結果為&#xff1a; 請輸入一個偶數: 4 偶數4可以分解成1和3兩個素數的和 #include <stdio.h> #include <stdlib.h> int Isprimer(int n); int main() {int n,i;do{printf("請輸入一個偶數&…

c#委托調用另一窗口函數_在C#中使用委托調用成員函數

c#委托調用另一窗口函數Prerequisite: Delegates in C# 先決條件&#xff1a; C&#xff03;中的代表 We can also call a member function of a class using delegates. It is similar to static function calls, here we have to pass member function using an object on t…

Java版AVG游戲開發入門[0]——游戲模式轉換中的事件交互

Java版AVG游戲開發入門[0]——游戲模式轉換中的事件交互 示例程序下載地址&#xff1a;http://download.csdn.net/source/999273&#xff08;源碼在jar內&#xff09; AVG&#xff0c;即Adventure Game&#xff0c;可以直譯為[冒險游戲]。但是通常情況下我們說AVG是指[文字冒險…

FreeRTOS任務創建和刪除

任務創建和刪除的API函數 xTaskCreate()&#xff1a;使用動態方法創建一個任務xTaskCreateStatic()&#xff1a;使用靜態方法創建一個任務xTaskCreateRestricated()&#xff1a;創建一個使用MPU進行限制的任務&#xff0c;相關內存使用動態內存分配vTaskDelete()&#xff1a;刪…

Delphi 調試

調試&#xff1a;F9執行F8逐過程單步調試F7逐語句單步調試轉載于:https://www.cnblogs.com/JackShao/archive/2012/04/30/2476931.html