久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 5408|回復: 0
打印 上一主題 下一主題
收起左側

W25Q64存儲器芯片寫不進數據,STC單片機程序

[復制鏈接]
跳轉到指定樓層
樓主
ID:109865 發表于 2019-5-2 23:03 來自手機 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
有用過W25Q64的嗎,程序分享一下,忙乎快2天了,用網上下載的程序就是寫不進數據,我用的單片機是STC12C5A60S。
網上的程序是這樣的。
//////////////////////////////////  MAIN.C ///////////////////////////////////
// ----W25X系統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位數據,可變波特率
    BRT= 0xDC;         // 設定獨立波特率發生器重裝值
    AUXR|= 0x04;       // 獨立波特率發生器時鐘為Fosc,即1T
    AUXR|= 0x01;       // 串口1選擇獨立波特率發生器為波特率發生器
    AUXR|= 0x10;       // 啟動獨立波特率發生器
}
//***************************************************************  
void main()
{
    unsigned  int a=0;
    unsignedint  i=0;
    unsignedchar xdata Send_DATA[500];  // 要發送的數據
    unsignedchar xdata Rec_DATA[500];   // 已接收的數據
    delay500ms();                       // 等待W25Q16初始化完畢
    //*************************
    UartInit();                   //串行口初始化.9600bps@11.0592MHz     
    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個數
    SPI_Flash_Read(Rec_DATA,0x000000,500);   //讀N個數
    while(1)
    {      
        SBUF  =a;                // 發送編號
        while(TI==0);
        TI=0;
        SBUF  =Rec_DATA[a] ;     // 發送數據
        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      //讀狀態寄存器
    #defineW25X_WriteStatus      0x01      //寫狀態寄存器
    #defineW25X_ReadDATA8        0x03      //普讀_數據
    #defineW25X_FastRead         0x0B      //快讀_數據
    #defineW25X_DualOutput       0x3B      //快讀_雙輸出
    #defineW25X_Writepage        0x02      //寫_數據_0~255個字節
    #defineW25X_S_Erase          0x20      //扇區擦除4KB
    #defineW25X_B_Erase          0xD8      //塊區擦除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芯片的狀態。
// 返回值:狀態寄存器數據字節
// 注:W25X16內部狀態寄存器第0位=0表示空閑,0位=1表示忙。
unsigned char W25Q16_ReadStatus()
{
    unsignedchar status=0;
    CS=0;
    SPI_WriteByte(W25X_ReadStatus);   // 0x05讀取狀態的命令字
    status=SPI_ReadByte();        // 讀取狀態字節
    CS=1;                         // 關閉片選
    returnstatus;
}  
// 功能:寫W25Q16芯片的狀態寄存器。
// 只有SPR、TB、BP2、BP1、BP0 (bit7、5、4、3、2)可以寫、
// 注:W25X16內部狀態寄存器第0位=0表示空閑,0位=1表示忙。
void W25Q16_WriteStatus(unsigned char Status)
{
    CS=0;
    SPI_WriteByte(W25X_WriteStatus);  // 0x01讀取狀態的命令字
    SPI_WriteByte(Status);            // 寫入一個字節
    CS=1;                         // 關閉片選
}
//SPI在一頁(0~65535)內寫入少于256個字節的數據
//在指定地址開始寫入最大256字節的數據
//pbuf:數據存儲區
//WriteAddr:開始寫入的地址(24bit)
//Len:要寫入的字節數(最大256),該數不應該超過該頁的剩余字節數!!!  
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);      //發送寫頁命令
   SPI_WriteByte((u8)((WriteAddr)>>16)); //發送24bit地址   
   SPI_WriteByte((u8)((WriteAddr)>>8));   
   SPI_WriteByte((u8)WriteAddr);  
    for(i=0;i<Len;i++)               //循環寫數
    {
        SPI_WriteByte(*pbuf++);      
    }
    CS=1;                              //取消片選
    while(W25Q16_ReadStatus()&0x01);   //等待寫入結束   
}
//無檢驗寫SPI FLASH
//必須確保所寫的地址范圍內的數據全部為0XFF,否則在非0XFF處寫入的數據將失敗!
//具有自動換頁功能
//在指定地址開始寫入指定長度的數據
//pbuf:數據存儲區
//WriteAddr:開始寫入的地址(24bit)
//Len:要寫入的字節數(最大65535)
void SPI_Flash_Write_NoCheck(u8 * pbuf,u32WriteAddr,u16 Len)
{
    u16 PageLen;                  // 頁內寫入字節長度
    PageLen=256-WriteAddr%256;    // 單頁剩余的字節數 (單頁剩余空間)
    if(Len<=PageLen) PageLen=Len; // 不大于256 個字節
    while(1)
    {
        W25X_Flash_Write_Page(pbuf,WriteAddr,PageLen);
        if(PageLen==Len)break;   // 寫入結束了
        else
        {
            pbuf+=PageLen;
            WriteAddr+=PageLen;
            Len-=PageLen;              //  減去已經寫入了的字節數
            if(Len>256)PageLen=256;   // 一次可以寫入256 個字節
            elsePageLen=Len;          // 不夠256 個字節了
        }
    }
}
//讀取SPI FLASH  
//在指定地址開始讀取指定長度的數據
//pbuf:數據存儲區
//ReadAddr:開始讀取的地址(24bit)
//Len:要讀取的字節數(最大65535)
void SPI_Flash_Read(u8 * pbuf,u32ReadAddr,u16 Len)   
{
    u16 i;  
    while(W25Q16_ReadStatus()&0x01);      // 判斷是否忙                                                     
    CS=0;                                 // 使能器件   
    SPI_WriteByte(W25X_ReadDATA8);        // 發送讀取命令   
    SPI_WriteByte((u8)((ReadAddr)>>16));  // 發送24bit地址   
   SPI_WriteByte((u8)((ReadAddr)>>8));   
   SPI_WriteByte((u8)ReadAddr);  
   for(i=0;i<Len;i++)
    {
       *pbuf++=SPI_ReadByte();            // 讀一個字節   
    }
    CS=1;                                  // 取消片選            
}  
//*************** 4K扇擦除************************OK
//擦除一個扇區
//Dst_Addr:扇區地址 0~511 for w25x16
//擦除一個扇區的最少時間:150ms
void W25X_SectorErase(unsigned long Addr24) //擦除資料圖示的4KB空間
{
    unsignedchar Addr1;       // 最低地址字節
    unsignedchar Addr2;       // 中間地址字節
    unsignedchar Addr3;       // 最高地址字節  
    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;       // 最低地址字節
    unsignedchar Addr2;       // 中間地址字節
    unsignedchar Addr3;       // 最高地址字節  
    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時開始執行擦除
    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);      // 讀取狀態的命令字
    unsignedchar SPI_ReadByte();             // 讀取狀態字節
#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,發送一個字節
- 參數說明:x:要發送的字節
- 注:很多情況下,SPI是需要有較高的速度的,此函數中不使用任何循環
       結構,如for(;;) while等等,并且使用了位尋址就是為了提高速度
******************************************************************/
void SPI_WriteByte(unsigned char x)
{
/*----這種使用循環與位運算的實現方式,速度要比直接用位尋址與順序執行方式實現慢得多
----因為它把大部分的時間花在了循環因子的遞增、比較與位運行上了
    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個位,寫到數據線上   (高位在前)
    SPI_CLK=0;
    SPI_CLK=1;     // 時鐘線產生上升沿,數據被寫入
   
    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,讀取一個字節
- 返回說明:讀到的字節
- 注:很多情況下,SPI是需要有較高的速度的,此函數中不使用任何循環
   結構,如for(;;) while等等,并且使用了位尋址就是為了提高速度
******************************************************************/
unsigned char SPI_ReadByte()
{  
    /*----這種使用循環與位運算的實現方式,速度要比直接用位尋址與順序執行方式實現慢得多
    ----因為它把大部分的時間花在了循環因子的遞增、比較與位運行上了
    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;        // 時鐘線產生下降沿,芯片輸出數據(高位在前)
    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);
}
分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏1 分享淘帖 頂 踩
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 欧美日韩一 | 日本不卡一区二区三区在线观看 | 欧美电影一区 | 99国产精品久久久久久久 | 国产一级在线 | 在线观看国产 | 欧美一级二级三级 | 国产欧美精品在线观看 | 国产 欧美 日韩 一区 | 亚洲视频免费在线观看 | 欧美男男videos| 久久精品中文字幕 | 午夜爱爱毛片xxxx视频免费看 | 国产精品久久久久久久久免费软件 | 欧美成年网站 | 国产电影一区二区三区爱妃记 | 中文精品视频 | 欧美乱操 | 99成人 | 亚洲精品久久久久久久久久久久久 | 久色一区 | 亚洲欧美日韩精品久久亚洲区 | 久久精品国产一区二区电影 | 日本韩国电影免费观看 | 欧美成年网站 | 性视频一区 | 欧美日韩精品久久久免费观看 | 午夜爽爽爽男女免费观看影院 | 精品久久国产视频 | 亚洲第一天堂 | 在线免费观看黄a | 特级生活片 | 成人免费毛片片v | 伊人伊人 | 欧美精品免费观看二区 | 亚洲一区 | 欧洲一级毛片 | 毛片一级片| 免费毛片在线 | av在线播放免费 | 亚洲一区二区三区免费视频 |