搜索
bottom↓
回复: 12

发一个W25QXX+FATFS代码4096字节一个族

  [复制链接]

出0入20汤圆

发表于 2017-2-21 08:45:17 | 显示全部楼层 |阅读模式
DSTATUS disk_initialize (
        BYTE drv                                /* Physical drive nmuber (0..) */
){
        switch (drv){
                case 0 :
                        if(W25QXX_Init())  return STA_NOINIT;
                        return RES_OK;
                case 1 :
                                return RES_OK;          
                case 2 :
                return RES_OK;          
                case 3 :
                return RES_OK;
                default:
                return STA_NOINIT;
        }  
}

DSTATUS disk_status (
        BYTE drv                /* Physical drive nmuber (0..) */
){
        switch (drv){
                case 0 :
                        return RES_OK;
                case 1 :
                        return RES_OK;
                case 2 :
                        return RES_OK;
                default:
                        return STA_NOINIT;
        }
}
/*-----------------------------------------------------------------------*/
/* Read Sector(s)                                                        */

DRESULT disk_read (
        BYTE drv,                /* Physical drive nmuber (0..) */
        BYTE *buff,                /* Data buffer to store read data */
        DWORD sector,        /* Sector address (LBA) */
        BYTE count                /* Number of sectors to read (1..255) */
){
        if( count==0 )  return RES_PARERR;  
        switch (drv){
                case 0:
                        if(0==W25QXX_Read_Sector(sector,buff,count)) return RES_OK;
                        return RES_ERROR;   
                case 1:
                        return RES_OK;   
                default:
                        return RES_ERROR;
        }
}
/*-----------------------------------------------------------------------*/
/* Write Sector(s)                                                       */
#if _READONLY == 0
DRESULT disk_write (
        BYTE drv,                        /* Physical drive nmuber (0..) */
        const BYTE *buff,                /* Data to be written */
        DWORD sector,                /* Sector address (LBA) */
        BYTE count                        /* Number of sectors to write (1..255) */
){
        if( count==0 )  return RES_PARERR;  
        switch (drv){
                case 0:
                        if(W25QXX_Erase_Sector(sector,count)) return RES_ERROR ;
                        if(W25QXX_Write_Sector(sector,(unsigned char *)buff,count)) return RES_ERROR ;
                        return RES_OK;
                case 1:
                        return RES_OK;
                default:return RES_ERROR;
        }
}
#endif /* _READONLY */
/*-----------------------------------------------------------------------*/
/* Miscellaneous Functions                                               */

DRESULT disk_ioctl (
        BYTE drv,                /* Physical drive nmuber (0..) */
        BYTE ctrl,                /* Control code */
        void *buff                /* Buffer to send/receive control data */
){
        //DWORD nFrom,nTo;
        //char *buf = buff;

        if (drv==0){   
                        //MSD0_GetCardInfo(&SD0_CardInfo);
                switch(ctrl){
                        case CTRL_SYNC :
                        return RES_OK;       
                       
                        //扇区擦除
                        case CTRL_ERASE_SECTOR:
                                //nFrom = *((DWORD*)buff);
                                //nTo = *(((DWORD*)buff)+1);
                                //W25QXX_Erase_Sector(nFrom,nTo-nFrom+1);
                                return RES_OK;       
                       
                        case GET_BLOCK_SIZE: //要删除的扇区的块大小
                                *(DWORD*)buff =W25QXX_BLOCK_SIZE;
                                return RES_OK;       
               
               
                        case GET_SECTOR_SIZE:
                                *(DWORD*)buff = W25QXX_SECTOR_SIZE;
                                return RES_OK;       
                       
                        case GET_SECTOR_COUNT:
                                *(DWORD*)buff = W25QXX_MAX_SECTORS;
                                return RES_OK;       
                }
        }
        return RES_PARERR;
}

#ifndef __W25QXX_H
#define __W25QXX_H

#include "W25QXX.h"
//**********************************************************************************************************************
#define SPI_CONFIG                                                                                                SPI1_Config();
#define W25QXX_ReadWrite_Byte(a)                                        SPI1_ReadWriteByte(a)
//**********************************************************************************************************************
//修改状态寄存器,允许芯片存储器被写
unsigned char W25QXX_Init(void){
        W25QXX_Read_ID();
        //W25QXX_Write_StatusReg(0x02);                        //使能状态寄存器中的写存储器
        return 0;        //0k       
}
//***************************************************************
//读取SPI_FLASH的状态寄存器
//BIT7  6   5   4   3   2   1   0
//SPR   RV  TB BP2 BP1 BP0 WEL BUSY
//SPR:默认0,状态寄存器保护位,配合WP使用
//TB,BP2,BP1,BP0:FLASH区域写保护设置
//WEL:写使能锁定
//BUSY:忙标记位(1,忙;0,空闲)
//默认:0x00
unsigned char W25QXX_Read_StatusReg(void){
         unsigned char reg;  
         W25QXX_CS(0);                                                            //使能器件  
         W25QXX_ReadWrite_Byte(W25QXX_ReadStatusReg);            //发送读取状态寄存器命令   
         reg=W25QXX_ReadWrite_Byte(0Xff);                             //读取一个字节
         W25QXX_CS(1);                                                            //取消片选   
         return reg;  
}
//*************************************************************************
//写SPI_FLASH状态寄存器
//只有SPR,TB,BP2,BP1,BP0(bit 7,5,4,3,2)可以写!!!
void W25QXX_Write_StatusReg(unsigned char sr){  
        W25QXX_CS(0);    //片选
        W25QXX_ReadWrite_Byte(W25QXX_EnableWriteStatusReg);  //使能写状态寄存器命令  
        W25QXX_CS(1);    //取消片选
        W25QXX_CS(0);         //片选                        
        W25QXX_ReadWrite_Byte(W25QXX_WriteStatusReg);   //发送写取状态寄存器命令   
        W25QXX_ReadWrite_Byte(sr);                                               //写入一个字节
        W25QXX_CS(1);                                                           //取消片选                 
}  
//****************************************************************************
//SPI_FLASH写使能
//将WEL置位  
void W25QXX_Page_Program_Enable(void){
        W25QXX_CS(0);                                                            //使能器件  
        W25QXX_ReadWrite_Byte(W25QXX_WriteEnable);      //发送写使能
        W25QXX_CS(1);                                                            //取消片选               
}
//****************************************************************************
//SPI_FLASH写禁止
//将WEL清零
void W25QXX_Page_Program_Disable(void){
        W25QXX_CS(0);                                                            //使能器件  
        W25QXX_ReadWrite_Byte(W25QXX_WriteDisable);            //发送写禁止指令   
        W25QXX_CS(1);                                                            //取消片选               
}                             
//*********************************************************************************
//发送程序 COM=命令;adr=地址
void W25QXX_SendAdrOut(unsigned char com,unsigned int adr){
        W25QXX_ReadWrite_Byte        (com);                                                                                     /* 发送字节数据烧写命令        */
        W25QXX_ReadWrite_Byte        ((adr & 0xFFFFFF) >> 16);                          /* 发送3个字节的地址信息 */
        W25QXX_ReadWrite_Byte        ((adr & 0xFFFF) >> 8);
        W25QXX_ReadWrite_Byte        (        adr & 0xFF);
}
//****************************************************************************
//读取芯片ID W25QXXVF016的是 0XBF41  sst25vf064_ID=bf4b
//文帮的是128=0xEF17    w25q64==ef16  
unsigned int W25QXX_Read_ID(void){
        unsigned int ID = 0;      
        W25QXX_CS(0);   
        W25QXX_SendAdrOut(W25QXX_ManufactDeviceID,0);//发送读取ID命令
        //读取返回的16位值                              
        ID         =W25QXX_ReadWrite_Byte(0xFF)<<8;                //高8位数据
        ID |=W25QXX_ReadWrite_Byte(0xFF);              //底八位数据
        W25QXX_CS(1);                                   
        return ID;
}
//****************************************************************************
//读取SPI FLASH
//在指定地址开始读取指定长度的数据
//pBuffer:数据存储区
//ReadAddr:开始读取的地址(24bit)
//NumByteToRead:要读取的字节数(最大65535即64k)
unsigned int W25QXX_Read_Sector(unsigned int Sector,unsigned char* pBuffer,unsigned int cnt){
        unsigned int i;   
        W25QXX_CS(0);                                                    //使能器件  
        W25QXX_SendAdrOut(W25QXX_FastReadData,Sector*W25QXX_SECTOR_SIZE);                        //发送读取命令
        W25QXX_ReadWrite_Byte(0XFF);                                                                                           //发送一个哑数据!
        for(i=0;i< cnt*W25QXX_SECTOR_SIZE;i++)                pBuffer[i]=W25QXX_ReadWrite_Byte(0XFF);   //循环读数
        W25QXX_CS(1);                            //取消片选      
        return 0;
}
//****************************************************************************
//写入扇区已经擦除干净, NumByteToWrite 最好是8,16,32,64,128,256
//pBuffer:为待写数据组
//WriteAddr:所写数据的起始地址
//NumByteToWrite:所要写的数据的长度
unsigned int  W25QXX_Write_Sector(unsigned int Sector,unsigned char* pBuffer,unsigned int cnt){
        unsigned int i,j,addr;
       
        addr =Sector*W25QXX_SECTOR_SIZE;
        while(cnt--){
                j=W25QXX_SECTOR_SIZE / W25QXX_PAGE_SIZE;
                while(j--){
                        W25QXX_Page_Program_Enable();                  //SET WEL
                        W25QXX_CS(0);
                        W25QXX_SendAdrOut(W25QXX_ByteProgram,addr);                                        //发送写页命令
                        //发送待写的数据
                        for(i=0;i < W25QXX_PAGE_SIZE;i++)         W25QXX_ReadWrite_Byte(*pBuffer++);
                        addr +=W25QXX_PAGE_SIZE;
                        W25QXX_CS(1);
                        W25QXX_Wait_Busy();//等待写完成
                }
        }
        W25QXX_Page_Program_Disable();
        return 0;
}
//****************************************************************************
//擦除整个芯片
//整片擦除时间:
//W25X16:25s
//W25X32:40s
//W25X64:40s
//等待时间超长...
void W25QXX_Erase_Chip(void){                                            
        W25QXX_Page_Program_Enable();                 //SET WEL
        W25QXX_CS(0);                                                    //使能器件  
        W25QXX_ReadWrite_Byte(W25QXX_ChipErase);      //发送片擦除命令
        W25QXX_CS(1);                                                    //取消片选               
        W25QXX_Wait_Busy();                                             //等待芯片擦除结束
        W25QXX_Page_Program_Disable();
}  
//****************************************************************************
//擦除一个扇区  Sector_Dst_Addr  物理扇区地址号!!
//Dst_Addr:扇区地址 0~511 for w25x16
//擦除一个山区的最少时间:150ms
unsigned int W25QXX_Erase_Sector(unsigned int Sector_Dst_Addr,unsigned int cnt){
        unsigned int i;
       
        W25QXX_Page_Program_Enable();                                          //SET WEL           
        for(i=0;i<cnt;i++){
                W25QXX_CS(0);                                                                   //使能器件
                W25QXX_SendAdrOut(W25QXX_4KByte_BlockERASE,(Sector_Dst_Addr++)*W25QXX_SECTOR_SIZE);//发送扇区擦除指令  必须输入绝对物理地址
                W25QXX_CS(1);                                                                    //取消片选   
                if(W25QXX_Wait_Busy()) return 1;                                                                         //等待擦除完成
        }                  
        //W25QXX_Page_Program_Disable();
        return 0;
}
//****************************************************************************
//等待空闲
unsigned int W25QXX_Wait_Busy(void){
        unsigned int i=0;

        while ((W25QXX_Read_StatusReg() & 0x01)==0x01){   // 等待BUSY位清空
                        if(i++>90000) return 1; //一直忙!
        }
        return 0;
}
#endif

阿莫论坛20周年了!感谢大家的支持与爱护!!

月入3000的是反美的。收入3万是亲美的。收入30万是移民美国的。收入300万是取得绿卡后回国,教唆那些3000来反美的!

出0入20汤圆

 楼主| 发表于 2017-2-21 08:45:50 | 显示全部楼层
#ifndef __SPI_FLASH_H
#define __SPI_FLASH_H  

#include "stm32f2xx.h"
//-------------------------------------------------------------------------------------------------------
/* 宏,用于定义W25QXX的片选引脚 */
#define        W25QXX_CS_PIN                                        (GPIO_Pin_8)                          //PB9
#define W25QXX_CS(x) ((x>0)?         (GPIO_SetBits(GPIOB, W25QXX_CS_PIN)) : (GPIO_ResetBits(GPIOB, W25QXX_CS_PIN) ))//设置1,0
//********************************************************************************************************************
//4Kbytes为一个Sector
//16个扇区为1个Block
//W25QXXVF016B
//容量为2M字节,共有32个Block(块),512个Sector(扇区)
//初始化SPI FLASH的IO口
#define FLASH_ID 0XBF41
//指令表
#define W25QXX_ReadData                         0x03
#define W25QXX_FastReadData                     0x0B
#define W25QXX_4KByte_BlockERASE                0x20
#define W25QXX_32KByte_BlockErase               0x52
#define W25QXX_64KByte_BlockErase               0xD8
#define W25QXX_ChipErase                        0xC7
#define W25QXX_ByteProgram                      0x02
#define W25QXX_AAI_WordProgram                  0xAD
#define W25QXX_ReadStatusReg                    0x05
#define W25QXX_EnableWriteStatusReg             0x50
#define W25QXX_WriteStatusReg                   0x01
#define W25QXX_WriteEnable                      0x06
#define W25QXX_WriteDisable                     0x04
#define W25QXX_ManufactDeviceID                 0x90
#define W25QXX_JedecDeviceID                    0x9F
#define W25QXX_EBSY                             0x70
#define W25QXX_DBSY                             0x80

//****************************************************************************
//sst25  256 byte=page  4096 byte=sectors (16 page ==sectors)  1 blk=共有2048个扇区!
//w25q128  == 1024*16*1024/4096   =4096 扇区!
//********************************************************************************************************
#define W25QXX_SECTOR_SIZE                        (4096)                //固定的
#define W25QXX_PAGE_SIZE                                (256)                        //固定的
#define W25QXX_MAX_SECTORS                        (2048)                //芯片共有的扇区数! 2048*n个扇区!
#define W25QXX_BLOCK_SIZE                                (1)                                //固定的        w25q64=128 block
//*****************************************************************************
unsigned char W25QXX_Read_StatusReg(void);
void W25QXX_Write_StatusReg(unsigned char sr);
unsigned char W25QXX_Init(void);
unsigned int W25QXX_Erase_Sector(unsigned int Dst_Addr,unsigned int cnt);//扇区擦除
unsigned int W25QXX_Wait_Busy(void);           //等待空闲
void W25QXX_Erase_Chip(void);
unsigned int W25QXX_Read_ID(void);
unsigned int W25QXX_Write_Sector(unsigned int Sector,unsigned char* pBuffer,unsigned int cnt);
unsigned int W25QXX_Read_Sector(unsigned int Sector,unsigned char* pBuffer,unsigned int cnt);
void SPI1_Config(void);
unsigned char SPI1_ReadWriteByte(unsigned char TxData);

#endif

出0入20汤圆

 楼主| 发表于 2017-2-21 08:49:47 | 显示全部楼层
空间检测时:要按512字节检测:
        res=f_mount(0, &fatfs);
        if( res != FR_OK)  return res+20;
        pfs=&fatfs;
        res = f_getfree("/", &clust, &pfs);//必须是根目录,默认磁盘0;"/"或者"0:/"
        if ( res==FR_OK ){                                                                        // Get free space
                sum=(pfs->n_fatent - 2) * pfs->csize /2/1024;
                free=clust * pfs->csize;
                s=(float)free/2/1024;        //单位是MB
                s *=8;                                                                //现在是4096字节一个族。原来是512个字节。
        }
        f_mount(0, NULL);//卸载FATFS

出0入0汤圆

发表于 2017-2-23 14:39:59 | 显示全部楼层
给楼主点赞!!!!

出0入0汤圆

发表于 2017-2-24 11:46:57 | 显示全部楼层
谢谢分享。以后会用到的

出0入0汤圆

发表于 2017-5-3 14:05:03 | 显示全部楼层
谢谢楼主分享

出0入0汤圆

发表于 2017-5-3 15:23:26 | 显示全部楼层
楼主好人啊,之前看原子的例程,它是强制将flash设置为512字节一个簇,我也是...

出0入0汤圆

发表于 2017-5-3 16:40:08 | 显示全部楼层
一次写入4K 速度快很多

出0入0汤圆

发表于 2017-5-4 07:22:54 来自手机 | 显示全部楼层
谢谢分享。。。。。。

出0入0汤圆

发表于 2017-5-4 09:10:58 | 显示全部楼层
谢谢分享!!!!

出0入0汤圆

发表于 2017-5-4 09:34:25 | 显示全部楼层
设为4K一个簇,要开的缓存也比较大,对于RAM紧张的情境下就比较麻烦了

出0入42汤圆

发表于 2017-5-4 11:57:08 | 显示全部楼层
菜包 发表于 2017-5-4 09:34
设为4K一个簇,要开的缓存也比较大,对于RAM紧张的情境下就比较麻烦了

可以外置一个RAM芯片  解决   

出0入0汤圆

发表于 2019-5-10 22:30:26 | 显示全部楼层
最近在看这方面的东西,感谢分享。
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-4-26 16:57

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表