|
更多內(nèi)容可查看《STC51單片機入門(C語言)》,出版的紙版書名稱為《51單片機輕松入門—基于STC15W4K系列》 第24 章 2M字節(jié)Flash Rom存儲器W25Q16/W25X16 認(rèn)識Flash Rom FlashRom 是快速擦寫只讀存儲器,也就是我們常說的“閃存”,單片機程序存儲器就是Flash Rom,所謂“閃存”,就是一種非易失性的內(nèi)存,屬于EEPROM的改進產(chǎn)品。它的最大特點是必須按塊對數(shù)據(jù)進行擦寫操作,芯片容量大(W25Q16容量2M字節(jié)),而EEPROM則可以對單個字節(jié)進行操作,芯片容量較小(24c512容量65536字節(jié))。前者容量是后者容量的32倍。另外Flash Rom比EEPROM具有更高的讀寫速度,芯片價格基本一致,W25Q16零售價3.5元。FlashRom除用于共業(yè)領(lǐng)域外,在消費電子產(chǎn)品中運用也非常廣泛,比如常見的U盤,SD卡,CF卡等存儲設(shè)備都是使用的FlashRom作為核心存儲器件。 Flash Rom的分類 NOR閃存 INTEL公司首批生產(chǎn) NAND閃存 日立公司首批生產(chǎn),比NOR閃存的寫周期短90%,保存刪除數(shù)據(jù)速度都較快,廣泛用于SD卡、XD卡、SM卡、CF卡等存儲卡上。 如果用來存儲少量的數(shù)據(jù),這時NOR閃存更適合一些,而NAND閃存則是高資料密度的理想解決方案。比如單片機的程序存儲器就是Flash Rom中的NOR閃存,常見的存儲卡或U盤使用的是NAND閃存。 認(rèn)識W25X16/ W25Q16 這里以W25X16介紹為主,W25Q16兼容W25X16,并且性能更佳,W25Q16保持了與W25X系列功能與管腳的完全兼容,并增加了雙/四輸入輸出等高效功能。W25Q的時鐘頻率達(dá)到了80MHz,等效讀取(連續(xù))速率為每秒320兆位(40兆字節(jié)), 這是標(biāo)準(zhǔn)50MHz串行閃存?zhèn)鬏斔俾实牧兑陨稀Ec此同時,W25Q將每個讀指令需要的時鐘數(shù)目從40個減少到12個,進一步減少了70%的“隨機讀取”的指令負(fù)荷。W25Q16零售價3.5元。它們屬與NOR型閃,用量較多的還是貼片封裝,這樣可以減小印制板面積。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image002.jpg file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image004.jpg W25X16有16Mbit存儲位, 因為1024bit就是1K, 16Mbit =16*1024*1024=16 777 216 bit,從手冊存儲器結(jié)構(gòu)圖可以看出W25X16的最大地址值是1FFFFFH,24位地址高3位固定為0(W25X64才需用到最高3位),只有21位有效地址,2的21次方=2097152字節(jié),2097152個字節(jié)正好存儲16 777 216bit,這里的地址都是以字節(jié)為最小單位的。 W25X16分為8192頁,每頁256字節(jié),用“頁編程指令”每次就可以編程256字節(jié),用“扇區(qū)擦除指令”每次可擦除16頁,用“塊擦除指令”每次可擦除256頁,用“整片擦除指令”可一次擦除整個芯片,W25X16有512個可擦除扇區(qū)或32個可擦除塊。對于W25X16,1頁=256字節(jié),歸納一下, 1頁=256字節(jié) 1扇區(qū)=16頁=16*256字節(jié)=4096字節(jié) (W25X16有512個扇區(qū))。 1塊=256頁=256*256字節(jié)=65536字節(jié) (W25X16有32塊)。 它采用標(biāo)準(zhǔn)SPI接口與單片機進行通信,最大時鐘速率75M。W25X16引腳排列如下圖所示。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image006.jpg 說明: 5腳(DIO):在普通方式下,這個引腳是串行輸入引腳(DI),數(shù)據(jù)、地址和命令通過此引腳送到芯片內(nèi)部,在CLK引腳的上升沿捕獲。當(dāng)使用了“快讀雙輸出指令”時,這個引腳就變成了DO引腳,這種情況下,芯片就有了2個DO引腳,所以叫做雙輸出,這時芯片的通信速度相當(dāng)于翻了一倍,所以傳輸速度更快。 7腳(/HOLD):保持引腳,當(dāng)/CS片選為低電平,且HOLD為低電平時,DO引腳處于高阻態(tài),而且會忽略DIO和CLK引腳上的信號,把HOLD拉高,器件恢復(fù)正常工作,當(dāng)芯片與多個其它芯片共享單片機上的同一個SPI接口時,此引腳就顯得非常有用,通常此引腳接高電平保證芯片正常工作。 什么都不用做,直接連接就是了。w25的端口兼容5伏的。用個1117--3.3V的就能共用5V,或5V的用兩個二級管降壓后得3.6V也和3.3V很接近可以用。 W25X16內(nèi)部狀態(tài)寄存器(上電復(fù)位時,各位都被清零) BUSY忙位: 只讀位,在芯片執(zhí)行“頁編程”,“扇區(qū)擦除” 、“塊擦除”、 “芯片擦除”、 “寫狀態(tài)寄存器”指令時,該位自動置1,此時除了“讀狀態(tài)寄存器”指令,其它指令都無效,當(dāng)編程、擦除和寫狀態(tài)寄存器指令執(zhí)行完畢后,該為自動變0,表示芯片可以接收其它指令了。 WEL寫保護位: 只讀位,寫操作允許標(biāo)志位,當(dāng)執(zhí)行完寫使能指令后,該位為1表示允許寫操作,為0表示禁止寫,當(dāng)芯片掉電后或執(zhí)行寫禁止、頁編程、扇區(qū)擦除、塊擦除、芯片擦除和寫狀態(tài)寄存器命令后自動進入寫保護狀態(tài)。 BP2、BP1、BP0塊保護位: 可讀寫位,用于塊區(qū)保護,可用寫狀態(tài)寄存器命令修改這幾位,為這3位為0時,塊區(qū)無保護,當(dāng)SPR位為1或/WP腳為低時,這3位無法更改。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image008.jpg TB 底部頂部塊保護位: 可讀寫位,用于底部頂部塊保護,可用寫狀態(tài)寄存器命令修改這1位,當(dāng)這1位為0時,底部頂部塊區(qū)無保護,當(dāng)SPR位為1或/WP腳為低時,這1位無法更改。 SPR狀態(tài)寄存器保護位: 可讀寫位,意義如下表 W25X16包括15個基本指令,通過這15個基本指令與SPI總線就完全可以控制芯片,指令在/CS拉低后開始傳送,DIO引腳上數(shù)據(jù)的第一個字節(jié)就是指令碼,在CLK引腳的上升沿采集DIO數(shù)據(jù),高位在前。 指令的長度從1個字節(jié)到多個字節(jié),有時還會跟隨地址字節(jié)、數(shù)據(jù)字節(jié)、偽字節(jié),有時候還會是它們的組合,在/CS引腳的上升沿完成指令的傳輸,所有的讀指令都可以在任意時鐘位完成,而所有的寫、編程和擦除指令在一個字節(jié)的邊界后才能完成,否則,指令將不起作用,這個特征可以保護芯片不被意外寫入,當(dāng)芯片正在被編程、擦除或?qū)憼顟B(tài)寄存器的時候,除“讀狀態(tài)寄存器”指令,其它所有指令都將被忽略直到擦寫周期結(jié)束。 | 指令名稱 | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | I/O=(D6,D4,D2,D0) O=(D7,D5,D3,D1) | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |
說明:數(shù)據(jù)高位在前,帶括號的數(shù)據(jù)表示數(shù)據(jù)從DO引腳讀出。 寫使能06H: 寫使能指令將會使?fàn)顟B(tài)寄存器WEL位置位,在執(zhí)行每個“頁編程”、“扇區(qū)擦除”、“塊擦除”、“芯片擦除”和“寫狀態(tài)寄存器”命令之前,都要先置位WEL,/CS腳先拉低之后,“寫使能”指令碼06H從DIO引腳輸入,在CLK上升沿采集,然后再拉高/CS引腳。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image010.jpg 寫禁止04H: 時序與寫使能相同,執(zhí)行完“頁編程”、“扇區(qū)擦除”、“塊擦除”、“芯片擦除”和 “寫狀態(tài)寄存器”命令之后WEL位會自動變0,即自動進入寫禁止?fàn)顟B(tài)。 讀狀態(tài)寄存器05H 當(dāng)/CS拉低之后,開始把05H從DIO引腳送入芯片,在CLK的上升沿數(shù)據(jù)被芯片采集,當(dāng)芯片認(rèn)出采集到的數(shù)據(jù)是05H時,芯片就會把“狀態(tài)寄存器”的值從DO引腳輸出,數(shù)據(jù)在CLK的下降沿輸出,高位在前。 讀狀態(tài)寄存器指令在任何時候都可以用,甚至在編程、擦除、寫狀態(tài)寄存器的過程中也可以用,這樣就可從狀態(tài)寄存器的BUSY位判斷編程、擦除、寫狀態(tài)寄存器周期是否結(jié)束,從而讓我們知道芯片是否可以接收下一指令,如果/CS不被拉高,狀態(tài)寄存器的值將一直從DO腳輸出,當(dāng)/CS拉高后,該指令結(jié)束。讀狀態(tài)寄存器時序如下圖所示。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image012.jpg 寫狀態(tài)寄存器01H 在執(zhí)行寫狀態(tài)寄存器指令以前,需要先按“寫使能時序”執(zhí)行完“寫使能”指令,然后再次將/CS拉低,把01H從DIO引腳送入芯片,然后再把需要的狀態(tài)寄存器的值送入芯片,拉高/CS,指令結(jié)束,如果此時沒把/CS腳拉高,或者是拉得晚了,值將不會被寫入,指令無效。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image014.jpg 讀數(shù)據(jù)03H 讀數(shù)據(jù)指令允許讀取一個或多個字節(jié),先將/CS拉低,把03H從DIO引腳送入芯片,然后再把24位地址送入芯片,這些數(shù)據(jù)在時鐘的上升沿被芯片采集,芯片收到24位在CLK引腳的下降沿從DO引腳輸出,高位在前。當(dāng)讀完這個地址的數(shù)據(jù)后,地址自動增加,然后通過DO引腳把下一個地址的數(shù)據(jù)輸出,也就是說,只要CLK在工作,通過一條指令就可把整個芯片儲存區(qū)的數(shù)據(jù)全部讀出來,把/CS腳拉高,“讀數(shù)據(jù)”指令結(jié)束,當(dāng)芯片在執(zhí)行編程、擦除和讀狀態(tài)寄存器指令的周期內(nèi),“讀數(shù)據(jù)”指令無效。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image016.jpg 頁編程02H 在執(zhí)行頁編程指令以前,需要先擦除整個待寫入?yún)^(qū)域,保證待寫入?yún)^(qū)域全為1,然后按“寫使能時序”執(zhí)行完“寫使能”指令,再把/CS拉低,將02H從DIO引腳送入芯片,然后再把24位地址送入芯片,然后接著送要寫入的字節(jié)到芯片,在寫完數(shù)據(jù)后,把/CS拉高。寫完一頁后必須把地址改為0,不然的話,如果時鐘還在繼續(xù),地址將自動變?yōu)轫摰拈_始地址,如果寫入的字節(jié)不足256個字節(jié)的話,其它寫入的字節(jié)都是無意義的,如果寫入的字節(jié)大于256字節(jié),多余的字節(jié)加上無用的字節(jié)覆蓋剛剛寫入的256字節(jié),所以需要保證寫入的字節(jié)小于或等于256字節(jié)。如果寫入的地址處于寫保護狀態(tài),“頁編程”指令無效。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image018.jpg file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image020.jpg 扇區(qū)擦除20H 扇區(qū)擦除指令將一個扇區(qū)(4096字節(jié))擦除,擦除后扇區(qū)字節(jié)都為FFH,在執(zhí)行扇區(qū)擦除指令以前,需要先按“寫使能時序”執(zhí)行完“寫使能”指令,然后再次將/CS拉低,把20H從DIO引腳送入芯片,然后再把24位扇區(qū)地址送入芯片,然后拉高/CS,指令結(jié)束,如果此時沒及時把/CS腳拉高,指令將不起作用。如果擦除的地址處于寫保護狀態(tài),“扇區(qū)擦除”指令無效。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image022.jpg 塊擦除D8H 塊擦除指令將一個塊(65536字節(jié))擦除,擦除后扇區(qū)字節(jié)都為FFH,在執(zhí)行塊擦除指令以前,需要先按“寫使能時序”執(zhí)行完“寫使能”指令,然后再次將/CS拉低,把D8H從DIO引腳送入芯片,然后再把24位扇區(qū)地址送入芯片,然后拉高/CS,指令結(jié)束,如果此時沒及時把/CS腳拉高,指令將不起作用。如果擦除的地址處于寫保護狀態(tài),“塊擦除”指令無效。 芯片擦除C7H 芯片擦除指令將整個芯片儲存區(qū)擦除,擦除后整個芯片儲存區(qū)字節(jié)都為FFH,在執(zhí)行芯片擦除指令以前,需要先按“寫使能時序”執(zhí)行完“寫使能”指令,然后再次將/CS拉低,把C7H從DIO引腳送入芯片,然后拉高/CS,指令結(jié)束,如果此時沒及時把/CS腳拉高,指令將不起作用。任何一個塊處于寫保護狀態(tài),“塊擦除”指令無效。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image024.jpg
實例:W25Q16讀寫500個數(shù)據(jù) 電路如下圖所示,由于W25Q16工作電源是2.7-3.6V,要求IO口電壓最大值不能超過VCC+0.4V,因此使用了下面的電平轉(zhuǎn)換電路將單片機輸出的5V信號電壓轉(zhuǎn)換為W25Q16可以接受的3V信號電壓,實際運用中W25Q16的供電建議使用低壓差穩(wěn)壓芯片ASM1117-3.3。 file:///C:/DOCUME~1/ADMINI~1/LOCALS~1/Temp/msohtml1/01/clip_image002.gif 程序功能說明:單片機上電后向芯片寫入500個數(shù)據(jù),然后讀出這500個數(shù)據(jù)并通過串口發(fā)送給計算機,通過STC下載程序的串口助手即可看到單片機返回的結(jié)果。在程序發(fā)送的過程中P1.0連接的LED不斷閃爍,直到數(shù)據(jù)發(fā)送完畢。 /////////////////////////////////// MAIN.C /////////////////////////////////// // ----W25X系統(tǒng)Flash讀寫----- #include "W25Q16.H" sbit P10=P1^0; //********************************************************OK void delay500ms(void) { unsigned char i,j,k; for(i=15;i>0;i--) //注意后面沒分號 for(j=202;j>0;j--) //注意后面沒分號 for(k=81;k>0;k--); //注意后面有分號 } //***************************************//串行口初始化 OK void UartInit(void) //9600bps@11.0592MHz { PCON&= 0x7f; // 波特率不倍速 SCON= 0x50; // 8位數(shù)據(jù),可變波特率 BRT= 0xDC; // 設(shè)定獨立波特率發(fā)生器重裝值 AUXR|= 0x04; // 獨立波特率發(fā)生器時鐘為Fosc,即1T AUXR|= 0x01; // 串口1選擇獨立波特率發(fā)生器為波特率發(fā)生器 AUXR|= 0x10; // 啟動獨立波特率發(fā)生器 } //*************************************************************** void main() { unsigned int a=0; unsignedint i=0; unsignedchar xdata Send_DATA[500]; // 要發(fā)送的數(shù)據(jù) unsignedchar xdata Rec_DATA[500]; // 已接收的數(shù)據(jù) delay500ms(); // 等待W25Q16初始化完畢 //************************* i=0; for(i=0;i<500;i++) { Send_DATA=i; Rec_DATA=0; } W25X_SectorErase(0x000000); //4K擦除 //************************** SPI_Flash_Write_NoCheck(Send_DATA,0x000000,500); //寫N個數(shù) SPI_Flash_Read(Rec_DATA,0x000000,500); //讀N個數(shù) while(1) { SBUF =a; // 發(fā)送編號 while(TI==0); TI=0; SBUF =Rec_DATA[a] ; // 發(fā)送數(shù)據(jù) while(TI==0); TI=0; delay500ms(); P10=!P10; a++; if(a>=500) { while(1); } } } /////////////////////////////////// W25Q16.H /////////////////////////////////// #ifndef _W25Q16_H_ #define _W25Q16_H_ #include"stc12c5a.h" #include<intrins.h> // 包含位移和_nop_(); sbitCS = P1^4; // 片選 #defineu8 unsigned char #defineu16 unsigned int #defineu32 unsigned long #defineW25X_ReadStatus 0x05 //讀狀態(tài)寄存器 #defineW25X_WriteStatus 0x01 //寫狀態(tài)寄存器 #defineW25X_ReadDATA8 0x03 //普讀_數(shù)據(jù) #defineW25X_FastRead 0x0B //快讀_數(shù)據(jù) #defineW25X_DualOutput 0x3B //快讀_雙輸出 #defineW25X_Writepage 0x02 //寫_數(shù)據(jù)_0~255個字節(jié) #defineW25X_S_Erase 0x20 //扇區(qū)擦除4KB #defineW25X_B_Erase 0xD8 //塊區(qū)擦除64KB #defineW25X_C_Erase 0xC7 //整片格式化 #defineW25X_PowerDown 0xB9 //待機 #defineW25X_PowerON_ID 0xAB //開機或是讀ID #defineW25X_JEDEC_ID 0x9F //十六位的JEDEC_ID #defineW25X_WriteEnable 0x06 //寫允許 #defineW25X_WriteDisable 0x04 //寫禁止 voidW25X_SectorErase(unsigned long Addre24); //擦除資料圖示的4KB空間 voidSPI_Flash_Write_NoCheck(u8 * pbuf,u32 WriteAddr,u16 Len); voidSPI_Flash_Read(u8* pbuf,u32 ReadAddr,u16 Len) ; #endif /////////////////////////////////// W25Q16.C/////////////////////////////////// #include "SPI.H" #include "W25Q16.H" //*************** 寫允許(將WEL置位) **************************** OK void WriteEnable (void) { CS=0; SPI_WriteByte(W25X_WriteEnable); CS=1; } //*************** 寫禁止(將WEL清0) **************************** OK void WriteDisable (void) { CS=0; SPI_WriteByte(W25X_WriteDisable); CS=1; } // 功能:讀取W25Q16芯片的狀態(tài)。 // 返回值:狀態(tài)寄存器數(shù)據(jù)字節(jié) // 注:W25X16內(nèi)部狀態(tài)寄存器第0位=0表示空閑,0位=1表示忙。 unsigned char W25Q16_ReadStatus() { unsignedchar status=0; CS=0; SPI_WriteByte(W25X_ReadStatus); // 0x05讀取狀態(tài)的命令字 status=SPI_ReadByte(); // 讀取狀態(tài)字節(jié) CS=1; // 關(guān)閉片選 returnstatus; } // 功能:寫W25Q16芯片的狀態(tài)寄存器。 // 只有SPR、TB、BP2、BP1、BP0 (bit7、5、4、3、2)可以寫、 // 注:W25X16內(nèi)部狀態(tài)寄存器第0位=0表示空閑,0位=1表示忙。 void W25Q16_WriteStatus(unsigned char Status) { CS=0; SPI_WriteByte(W25X_WriteStatus); // 0x01讀取狀態(tài)的命令字 SPI_WriteByte(Status); // 寫入一個字節(jié) CS=1; // 關(guān)閉片選 } //SPI在一頁(0~65535)內(nèi)寫入少于256個字節(jié)的數(shù)據(jù) //在指定地址開始寫入最大256字節(jié)的數(shù)據(jù) //pbuf:數(shù)據(jù)存儲區(qū) //WriteAddr:開始寫入的地址(24bit) //Len:要寫入的字節(jié)數(shù)(最大256),該數(shù)不應(yīng)該超過該頁的剩余字節(jié)數(shù)!!! void W25X_Flash_Write_Page(u8* pbuf,u32WriteAddr,u16 Len) { u16 i; while(W25Q16_ReadStatus()&0x01); //判斷是否忙 WriteEnable(); //SET WEL CS=0; //使能器件 SPI_WriteByte(W25X_Writepage); //發(fā)送寫頁命令 SPI_WriteByte((u8)((WriteAddr)>>16)); //發(fā)送24bit地址 SPI_WriteByte((u8)((WriteAddr)>>8)); SPI_WriteByte((u8)WriteAddr); for(i=0;i<Len;i++) //循環(huán)寫數(shù) { SPI_WriteByte(*pbuf++); } CS=1; //取消片選 while(W25Q16_ReadStatus()&0x01); //等待寫入結(jié)束 } //無檢驗寫SPI FLASH //必須確保所寫的地址范圍內(nèi)的數(shù)據(jù)全部為0XFF,否則在非0XFF處寫入的數(shù)據(jù)將失敗! //具有自動換頁功能 //在指定地址開始寫入指定長度的數(shù)據(jù) //pbuf:數(shù)據(jù)存儲區(qū) //WriteAddr:開始寫入的地址(24bit) //Len:要寫入的字節(jié)數(shù)(最大65535) void SPI_Flash_Write_NoCheck(u8 * pbuf,u32WriteAddr,u16 Len) { u16 PageLen; // 頁內(nèi)寫入字節(jié)長度 PageLen=256-WriteAddr%256; // 單頁剩余的字節(jié)數(shù) (單頁剩余空間) if(Len<=PageLen) PageLen=Len; // 不大于256 個字節(jié) while(1) { W25X_Flash_Write_Page(pbuf,WriteAddr,PageLen); if(PageLen==Len)break; // 寫入結(jié)束了 else { pbuf+=PageLen; WriteAddr+=PageLen; Len-=PageLen; // 減去已經(jīng)寫入了的字節(jié)數(shù) if(Len>256)PageLen=256; // 一次可以寫入256 個字節(jié) elsePageLen=Len; // 不夠256 個字節(jié)了 } } } //讀取SPI FLASH //在指定地址開始讀取指定長度的數(shù)據(jù) //pbuf:數(shù)據(jù)存儲區(qū) //ReadAddr:開始讀取的地址(24bit) //Len:要讀取的字節(jié)數(shù)(最大65535) void SPI_Flash_Read(u8 * pbuf,u32ReadAddr,u16 Len) { u16 i; while(W25Q16_ReadStatus()&0x01); // 判斷是否忙 CS=0; // 使能器件 SPI_WriteByte(W25X_ReadDATA8); // 發(fā)送讀取命令 SPI_WriteByte((u8)((ReadAddr)>>16)); // 發(fā)送24bit地址 SPI_WriteByte((u8)((ReadAddr)>>8)); SPI_WriteByte((u8)ReadAddr); for(i=0;i<Len;i++) { *pbuf++=SPI_ReadByte(); // 讀一個字節(jié) } CS=1; // 取消片選 } //*************** 4K扇擦除************************OK //擦除一個扇區(qū) //Dst_Addr:扇區(qū)地址 0~511 for w25x16 //擦除一個扇區(qū)的最少時間:150ms void W25X_SectorErase(unsigned long Addr24) //擦除資料圖示的4KB空間 { unsignedchar Addr1; // 最低地址字節(jié) unsignedchar Addr2; // 中間地址字節(jié) unsignedchar Addr3; // 最高地址字節(jié) Addr1=Addr24; Addr24=Addr24>>8; Addr2=Addr24; Addr24=Addr24>>8; Addr3=Addr24; // 把地址拆開來 while(W25Q16_ReadStatus()&0x01); // 判斷是否忙 WriteEnable(); // 寫允許 CS=0; SPI_WriteByte(W25X_S_Erase); // 整扇擦除命令 SPI_WriteByte(Addr3); SPI_WriteByte(Addr2); SPI_WriteByte(Addr1); CS=1; while(W25Q16_ReadStatus()&0x01); // 等待擦除完成 } //*************** 塊擦除/64K頁************************* OK void W25X_BlockErase(unsigned long Addr24) //擦除資料圖示的64KB空間 { unsignedchar Addr1; // 最低地址字節(jié) unsignedchar Addr2; // 中間地址字節(jié) unsignedchar Addr3; // 最高地址字節(jié) Addr1=Addr24; Addr24=Addr24>>8; Addr2=Addr24; Addr24=Addr24>>8; Addr3=Addr24; // 把地址拆開來 while(W25Q16_ReadStatus()&0x01); // 判斷是否忙 WriteEnable(); // 寫允許 CS=0; SPI_WriteByte(W25X_B_Erase); // 整扇擦除命令 SPI_WriteByte(Addr3); SPI_WriteByte(Addr2); SPI_WriteByte(Addr1); CS=1; while(W25Q16_ReadStatus()&0x01); // 等待擦除完成 } //**************片擦除 ****************** OK // W25X16:25S W25X32:40S W25X64:40S void W25X_ChipErase(void) { while(W25Q16_ReadStatus()&0x01); // 判斷是否忙 WriteEnable(); // 寫允許 CS=0; SPI_WriteByte(W25X_C_Erase); // 整片擦除命令 CS=1; // 從CS=1時開始執(zhí)行擦除 while(W25Q16_ReadStatus()&0x01); // 等待擦除完成 } /////////////////////////////////// SPI.H/////////////////////////////////// #ifndef _SPI_H_ #define _SPI_H_ #include"stc12c5a.h" #include<intrins.h> // 包含位移和_nop_(); sbitSPI_DIO = P1^5; // 只作輸入 (單片機 TO 芯片) sbitSPI_DO = P1^6; // 輸出 (芯片 TO 單片機) sbitSPI_CLK = P1^7; // 時鐘 // 片選腳由W25Q16.H定義,W25Q16.C控制 voidSPI_WriteByte(unsigned char x); // 讀取狀態(tài)的命令字 unsignedchar SPI_ReadByte(); // 讀取狀態(tài)字節(jié) #endif /////////////////////////////////// SPI.C /////////////////////////////////// #include "SPI.H" unsigned char bdata dat; //dat是可位尋址的變量 sbit dat7=dat^7; sbit dat6=dat^6; sbit dat5=dat^5; sbit dat4=dat^4; sbit dat3=dat^3; sbit dat2=dat^2; sbit dat1=dat^1; sbit dat0=dat^0; // 取出dat的各個位 /****************************************************************** - 功能描述:IO模擬SPI,發(fā)送一個字節(jié) - 參數(shù)說明:x:要發(fā)送的字節(jié) - 注:很多情況下,SPI是需要有較高的速度的,此函數(shù)中不使用任何循環(huán) 結(jié)構(gòu),如for(;;) while等等,并且使用了位尋址就是為了提高速度 ******************************************************************/ void SPI_WriteByte(unsigned char x) { /*----這種使用循環(huán)與位運算的實現(xiàn)方式,速度要比直接用位尋址與順序執(zhí)行方式實現(xiàn)慢得多 ----因為它把大部分的時間花在了循環(huán)因子的遞增、比較與位運行上了 unDIOgnedchar i=0; for(i=0;i<8;i++) { SPI_DIO=x&(0x80>>i); SPI_CLK=0; SPI_CLK=1; } ----------------------------------*/ dat=x; // 將x的值賦給可位尋址的變量dat,以便取出各個位 SPI_DIO=dat7; // 取出第7個位,寫到數(shù)據(jù)線上 (高位在前) SPI_CLK=0; SPI_CLK=1; // 時鐘線產(chǎn)生上升沿,數(shù)據(jù)被寫入 SPI_DIO=dat6; SPI_CLK=0; SPI_CLK=1; SPI_DIO=dat5; SPI_CLK=0; SPI_CLK=1; SPI_DIO=dat4; SPI_CLK=0; SPI_CLK=1; SPI_DIO=dat3; SPI_CLK=0; SPI_CLK=1; SPI_DIO=dat2; SPI_CLK=0; SPI_CLK=1; SPI_DIO=dat1; SPI_CLK=0; SPI_CLK=1; SPI_DIO=dat0; SPI_CLK=0; SPI_CLK=1; } /****************************************************************** - 功能描述:IO模擬SPI,讀取一個字節(jié) - 返回說明:讀到的字節(jié) - 注:很多情況下,SPI是需要有較高的速度的,此函數(shù)中不使用任何循環(huán) 結(jié)構(gòu),如for(;;) while等等,并且使用了位尋址就是為了提高速度 ******************************************************************/ unsigned char SPI_ReadByte() { /*----這種使用循環(huán)與位運算的實現(xiàn)方式,速度要比直接用位尋址與順序執(zhí)行方式實現(xiàn)慢得多 ----因為它把大部分的時間花在了循環(huán)因子的遞增、比較與位運行上了 unDIOgnedchar i=0,temp=0; SPI_DIO=1; for(i=0;i<8;i++) { SPI_CLK=1; SPI_CLK=0; if(SPI_DIO)temp|=0x80>>i; } returntemp; ----------------------------------*/ SPI_DO=1; SPI_CLK=1; SPI_CLK=0; // 時鐘線產(chǎn)生下降沿,芯片輸出數(shù)據(jù)(高位在前) dat7=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat6=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat5=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat4=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat3=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat2=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat1=SPI_DO; SPI_CLK=1; SPI_CLK=0; dat0=SPI_DO; return(dat); }
|