#ifdef USE_W25QXX_SPI #if (_W25QXX_DEBUG==1) #include #endif #include "sys_delay.h" #include "dev_spi_sdcard.h" #include "obj_spi_w25qxx.h" #define W25QXX_DUMMY_BYTE 0xA5 SPI_HandleTypeDef hspi_w25q = {0}; w25qxx_t w25qxx; #if (_W25QXX_USE_FREERTOS==1) #define W25qxx_Delay(delay) osDelay(delay) #include "cmsis_os.h" #else #define W25qxx_Delay(delay) delay_us(delay)//sys_delay_ms(delay)// #endif void W25qxx_Cs_Init() { GPIO_InitTypeDef GPIO_InitStruct; SPIx_CS_GPIO_CLK_ENABLE(); GPIO_InitStruct.Pin = _W25QXX_CS_PIN ; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH; HAL_GPIO_Init(_W25QXX_CS_GPIO, &GPIO_InitStruct); W25QXX_CS_HIGH(); } /** * @brief SPI MSP Initialization * This function configures the hardware resources used in this example: * - Peripheral's clock enable * - Peripheral's GPIO Configuration * @param hspi: SPI handle pointer * @retval None */ void HAL_SPI_MspInit_flash(void) { GPIO_InitTypeDef GPIO_InitStruct; /*##-1- Enable peripherals and GPIO Clocks #################################*/ /* Enable GPIO TX/RX clock */ SPIx_SCK_GPIO_CLK_ENABLE(); SPIx_MISO_GPIO_CLK_ENABLE(); SPIx_MOSI_GPIO_CLK_ENABLE(); /*##-2- Configure peripheral GPIO ##########################################*/ /* SPI SCK GPIO pin configuration */ GPIO_InitStruct.Pin = SPIx_SCK_PIN; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Pull = GPIO_PULLUP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; GPIO_InitStruct.Alternate = GPIO_AF5_SPI2; HAL_GPIO_Init(SPIx_SCK_GPIO_PORT, &GPIO_InitStruct); /* SPI MISO GPIO pin configuration */ GPIO_InitStruct.Pin = SPIx_MISO_PIN; HAL_GPIO_Init(SPIx_MISO_GPIO_PORT, &GPIO_InitStruct); /* SPI MOSI GPIO pin configuration */ GPIO_InitStruct.Pin = SPIx_MOSI_PIN; HAL_GPIO_Init(SPIx_MOSI_GPIO_PORT, &GPIO_InitStruct); } void W25qxx_Spi_Init(void) { /* Enable SPI clock */ SPIx_CLK_ENABLE(); hspi_w25q.Instance = SPIx; //SP2 hspi_w25q.Init.Mode = SPI_MODE_MASTER; //设置SPI工作模式,设置为主模式 hspi_w25q.Init.Direction = SPI_DIRECTION_2LINES; //设置SPI单向或者双向的数据模式:SPI设置为双线模式 hspi_w25q.Init.DataSize = SPI_DATASIZE_8BIT; //设置SPI的数据大小:SPI发送接收8位帧结构 hspi_w25q.Init.CLKPolarity = SPI_POLARITY_HIGH; //串行同步时钟的空闲状态为高电平 hspi_w25q.Init.CLKPhase = SPI_PHASE_2EDGE; //串行同步时钟的第二个跳变沿(上升或下降)数据被采样 hspi_w25q.Init.NSS = SPI_NSS_SOFT; //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制 hspi_w25q.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_256; //定义波特率预分频的值:波特率预分频值为256 hspi_w25q.Init.FirstBit = SPI_FIRSTBIT_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始 hspi_w25q.Init.TIMode = SPI_TIMODE_DISABLE; //关闭TI模式 hspi_w25q.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE; //关闭硬件CRC校验 hspi_w25q.Init.CRCPolynomial = 7; //CRC值计算的多项式 HAL_SPI_Init(&hspi_w25q);//初始化 __HAL_SPI_ENABLE(&hspi_w25q); //使能SPI2 } //################################################################################################################### uint8_t W25qxx_Spi(uint8_t Data) { uint8_t ret; HAL_SPI_TransmitReceive(&_W25QXX_SPI,&Data,&ret,1,10); return ret; } //################################################################################################################### uint32_t W25qxx_ReadID(void) { uint32_t Temp = 0, Temp0 = 0, Temp1 = 0, Temp2 = 0; W25QXX_CS_LOW(); W25qxx_Spi(0x9F); Temp0 = W25qxx_Spi(W25QXX_DUMMY_BYTE); Temp1 = W25qxx_Spi(W25QXX_DUMMY_BYTE); Temp2 = W25qxx_Spi(W25QXX_DUMMY_BYTE); W25QXX_CS_HIGH(); Temp = (Temp0 << 16) | (Temp1 << 8) | Temp2; return Temp; } //################################################################################################################### void W25qxx_ReadUniqID(void) { W25QXX_CS_LOW(); W25qxx_Spi(0x4B); for(uint8_t i=0;i<4;i++) W25qxx_Spi(W25QXX_DUMMY_BYTE); for(uint8_t i=0;i<8;i++) w25qxx.UniqID[i] = W25qxx_Spi(W25QXX_DUMMY_BYTE); W25QXX_CS_HIGH(); } //################################################################################################################### void W25qxx_WriteEnable(void) { W25QXX_CS_LOW(); W25qxx_Spi(0x06); W25QXX_CS_HIGH(); W25qxx_Delay(1); } //################################################################################################################### void W25qxx_WriteDisable(void) { W25QXX_CS_LOW(); W25qxx_Spi(0x04); W25QXX_CS_HIGH(); W25qxx_Delay(1); } //################################################################################################################### uint8_t W25qxx_ReadStatusRegister(uint8_t SelectStatusRegister_1_2_3) { uint8_t status=0; W25QXX_CS_LOW(); if(SelectStatusRegister_1_2_3==1) { W25qxx_Spi(0x05); status=W25qxx_Spi(W25QXX_DUMMY_BYTE); w25qxx.StatusRegister1 = status; } else if(SelectStatusRegister_1_2_3==2) { W25qxx_Spi(0x35); status=W25qxx_Spi(W25QXX_DUMMY_BYTE); w25qxx.StatusRegister2 = status; } else { W25qxx_Spi(0x15); status=W25qxx_Spi(W25QXX_DUMMY_BYTE); w25qxx.StatusRegister3 = status; } W25QXX_CS_HIGH(); return status; } //################################################################################################################### void W25qxx_WriteStatusRegister(uint8_t SelectStatusRegister_1_2_3,uint8_t Data) { W25QXX_CS_LOW(); if(SelectStatusRegister_1_2_3==1) { W25qxx_Spi(0x01); w25qxx.StatusRegister1 = Data; } else if(SelectStatusRegister_1_2_3==2) { W25qxx_Spi(0x31); w25qxx.StatusRegister2 = Data; } else { W25qxx_Spi(0x11); w25qxx.StatusRegister3 = Data; } W25qxx_Spi(Data); W25QXX_CS_HIGH(); } //################################################################################################################### void W25qxx_WaitForWriteEnd(void) { W25qxx_Delay(1); W25QXX_CS_LOW(); W25qxx_Spi(0x05); do { w25qxx.StatusRegister1 = W25qxx_Spi(W25QXX_DUMMY_BYTE); W25qxx_Delay(1); } while ((w25qxx.StatusRegister1 & 0x01) == 0x01); W25QXX_CS_HIGH(); } //################################################################################################################### bool W25qxx_Init(void) { w25qxx.Lock=1; while(HAL_GetTick()<100) W25qxx_Delay(1); uint32_t id; #if (_W25QXX_DEBUG==1) printf("w25qxx Init Begin...\r\n"); #endif id=W25qxx_ReadID(); #if (_W25QXX_DEBUG==1) printf("w25qxx ID:0x%X\r\n",id); #endif switch(id&0x0000FFFF) { case 0x401A: // w25q512 w25qxx.ID=ID_W25Q512; w25qxx.BlockCount=1024; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q512\r\n"); #endif break; case 0x4019: // w25q256 w25qxx.ID=ID_W25Q256; w25qxx.BlockCount=512; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q256\r\n"); #endif break; case 0x4018: // w25q128 w25qxx.ID=ID_W25Q128; w25qxx.BlockCount=256; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q128\r\n"); #endif break; case 0x4017: // w25q64 w25qxx.ID=ID_W25Q64; w25qxx.BlockCount=128; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q64\r\n"); #endif break; case 0x4016: // w25q32 w25qxx.ID=ID_W25Q32; w25qxx.BlockCount=64; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q32\r\n"); #endif break; case 0x4015: // w25q16 w25qxx.ID=ID_W25Q16; w25qxx.BlockCount=32; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q16\r\n"); #endif break; case 0x4014: // w25q80 w25qxx.ID=ID_W25Q80; w25qxx.BlockCount=16; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q80\r\n"); #endif break; case 0x4013: // w25q40 w25qxx.ID=ID_W25Q40; w25qxx.BlockCount=8; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q40\r\n"); #endif break; case 0x4012: // w25q20 w25qxx.ID=ID_W25Q20; w25qxx.BlockCount=4; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q20\r\n"); #endif break; case 0x4011: // w25q10 w25qxx.ID=ID_W25Q10; w25qxx.BlockCount=2; #if (_W25QXX_DEBUG==1) printf("w25qxx Chip: w25q10\r\n"); #endif break; default: #if (_W25QXX_DEBUG==1) printf("w25qxx Unknown ID\r\n"); #endif w25qxx.Lock=0; return false; } w25qxx.PageSize=256; w25qxx.SectorSize=0x1000; w25qxx.SectorCount=w25qxx.BlockCount*16; w25qxx.PageCount=(w25qxx.SectorCount*w25qxx.SectorSize)/w25qxx.PageSize; w25qxx.BlockSize=w25qxx.SectorSize*16; w25qxx.CapacityInKiloByte=(w25qxx.SectorCount*w25qxx.SectorSize)/1024; W25qxx_ReadUniqID(); W25qxx_ReadStatusRegister(1); W25qxx_ReadStatusRegister(2); W25qxx_ReadStatusRegister(3); #if (_W25QXX_DEBUG==1) printf("w25qxx Page Size: %d Bytes\r\n",w25qxx.PageSize); printf("w25qxx Page Count: %d\r\n",w25qxx.PageCount); printf("w25qxx Sector Size: %d Bytes\r\n",w25qxx.SectorSize); printf("w25qxx Sector Count: %d\r\n",w25qxx.SectorCount); printf("w25qxx Block Size: %d Bytes\r\n",w25qxx.BlockSize); printf("w25qxx Block Count: %d\r\n",w25qxx.BlockCount); printf("w25qxx Capacity: %d KiloBytes\r\n",w25qxx.CapacityInKiloByte); printf("w25qxx Init Done\r\n"); #endif w25qxx.Lock=0; return true; } //################################################################################################################### void W25qxx_EraseChip(void) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; #if (_W25QXX_DEBUG==1) uint32_t StartTime=HAL_GetTick(); printf("w25qxx EraseChip Begin...\r\n"); #endif W25qxx_WriteEnable(); W25QXX_CS_LOW(); W25qxx_Spi(0xC7); W25QXX_CS_HIGH(); W25qxx_WaitForWriteEnd(); #if (_W25QXX_DEBUG==1) printf("w25qxx EraseBlock done after %d ms!\r\n",HAL_GetTick()-StartTime); #endif W25qxx_Delay(10); w25qxx.Lock=0; } //################################################################################################################### void W25qxx_EraseSector(uint32_t SectorAddr) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; #if (_W25QXX_DEBUG==1) uint32_t StartTime=HAL_GetTick(); printf("w25qxx EraseSector %d Begin...\r\n",SectorAddr); #endif W25qxx_WaitForWriteEnd(); SectorAddr = SectorAddr * w25qxx.SectorSize; W25qxx_WriteEnable(); W25QXX_CS_LOW(); W25qxx_Spi(0x20); if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((SectorAddr & 0xFF000000) >> 24); W25qxx_Spi((SectorAddr & 0xFF0000) >> 16); W25qxx_Spi((SectorAddr & 0xFF00) >> 8); W25qxx_Spi(SectorAddr & 0xFF); W25QXX_CS_HIGH(); W25qxx_WaitForWriteEnd(); #if (_W25QXX_DEBUG==1) printf("w25qxx EraseSector done after %d ms\r\n",HAL_GetTick()-StartTime); #endif W25qxx_Delay(1); w25qxx.Lock=0; } //################################################################################################################### void W25qxx_EraseBlock(uint32_t BlockAddr) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; #if (_W25QXX_DEBUG==1) printf("w25qxx EraseBlock %d Begin...\r\n",BlockAddr); W25qxx_Delay(100); uint32_t StartTime=HAL_GetTick(); #endif W25qxx_WaitForWriteEnd(); BlockAddr = BlockAddr * w25qxx.SectorSize*16; W25qxx_WriteEnable(); W25QXX_CS_LOW(); W25qxx_Spi(0xD8); if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((BlockAddr & 0xFF000000) >> 24); W25qxx_Spi((BlockAddr & 0xFF0000) >> 16); W25qxx_Spi((BlockAddr & 0xFF00) >> 8); W25qxx_Spi(BlockAddr & 0xFF); W25QXX_CS_HIGH(); W25qxx_WaitForWriteEnd(); #if (_W25QXX_DEBUG==1) printf("w25qxx EraseBlock done after %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif W25qxx_Delay(1); w25qxx.Lock=0; } //################################################################################################################### uint32_t W25qxx_PageToSector(uint32_t PageAddress) { return ((PageAddress*w25qxx.PageSize)/w25qxx.SectorSize); } //################################################################################################################### uint32_t W25qxx_PageToBlock(uint32_t PageAddress) { return ((PageAddress*w25qxx.PageSize)/w25qxx.BlockSize); } //################################################################################################################### uint32_t W25qxx_SectorToBlock(uint32_t SectorAddress) { return ((SectorAddress*w25qxx.SectorSize)/w25qxx.BlockSize); } //################################################################################################################### uint32_t W25qxx_SectorToPage(uint32_t SectorAddress) { return (SectorAddress*w25qxx.SectorSize)/w25qxx.PageSize; } //################################################################################################################### uint32_t W25qxx_BlockToPage(uint32_t BlockAddress) { return (BlockAddress*w25qxx.BlockSize)/w25qxx.PageSize; } //################################################################################################################### bool W25qxx_IsEmptyPage(uint32_t Page_Address,uint32_t OffsetInByte,uint32_t NumByteToCheck_up_to_PageSize) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; if(((NumByteToCheck_up_to_PageSize+OffsetInByte)>w25qxx.PageSize)||(NumByteToCheck_up_to_PageSize==0)) NumByteToCheck_up_to_PageSize=w25qxx.PageSize-OffsetInByte; #if (_W25QXX_DEBUG==1) printf("w25qxx CheckPage:%d, Offset:%d, Bytes:%d begin...\r\n",Page_Address,OffsetInByte,NumByteToCheck_up_to_PageSize); W25qxx_Delay(100); uint32_t StartTime=HAL_GetTick(); #endif uint8_t pBuffer[32]; uint32_t WorkAddress; uint32_t i; for(i=OffsetInByte; i=ID_W25Q256) W25qxx_Spi((WorkAddress & 0xFF000000) >> 24); W25qxx_Spi((WorkAddress & 0xFF0000) >> 16); W25qxx_Spi((WorkAddress & 0xFF00) >> 8); W25qxx_Spi(WorkAddress & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,sizeof(pBuffer),100); W25QXX_CS_HIGH(); for(uint8_t x=0;x=ID_W25Q256) W25qxx_Spi((WorkAddress & 0xFF000000) >> 24); W25qxx_Spi((WorkAddress & 0xFF0000) >> 16); W25qxx_Spi((WorkAddress & 0xFF00) >> 8); W25qxx_Spi(WorkAddress & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,1,100); W25QXX_CS_HIGH(); if(pBuffer[0]!=0xFF) goto NOT_EMPTY; } } #if (_W25QXX_DEBUG==1) printf("w25qxx CheckPage is Empty in %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif w25qxx.Lock=0; return true; NOT_EMPTY: #if (_W25QXX_DEBUG==1) printf("w25qxx CheckPage is Not Empty in %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif w25qxx.Lock=0; return false; } //################################################################################################################### bool W25qxx_IsEmptySector(uint32_t Sector_Address,uint32_t OffsetInByte,uint32_t NumByteToCheck_up_to_SectorSize) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; if((NumByteToCheck_up_to_SectorSize>w25qxx.SectorSize)||(NumByteToCheck_up_to_SectorSize==0)) NumByteToCheck_up_to_SectorSize=w25qxx.SectorSize; #if (_W25QXX_DEBUG==1) printf("w25qxx CheckSector:%d, Offset:%d, Bytes:%d begin...\r\n",Sector_Address,OffsetInByte,NumByteToCheck_up_to_SectorSize); W25qxx_Delay(100); uint32_t StartTime=HAL_GetTick(); #endif uint8_t pBuffer[32]; uint32_t WorkAddress; uint32_t i; for(i=OffsetInByte; i=ID_W25Q256) W25qxx_Spi((WorkAddress & 0xFF000000) >> 24); W25qxx_Spi((WorkAddress & 0xFF0000) >> 16); W25qxx_Spi((WorkAddress & 0xFF00) >> 8); W25qxx_Spi(WorkAddress & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,sizeof(pBuffer),100); W25QXX_CS_HIGH(); for(uint8_t x=0;x=ID_W25Q256) W25qxx_Spi((WorkAddress & 0xFF000000) >> 24); W25qxx_Spi((WorkAddress & 0xFF0000) >> 16); W25qxx_Spi((WorkAddress & 0xFF00) >> 8); W25qxx_Spi(WorkAddress & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,1,100); W25QXX_CS_HIGH(); if(pBuffer[0]!=0xFF) goto NOT_EMPTY; } } #if (_W25QXX_DEBUG==1) printf("w25qxx CheckSector is Empty in %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif w25qxx.Lock=0; return true; NOT_EMPTY: #if (_W25QXX_DEBUG==1) printf("w25qxx CheckSector is Not Empty in %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif w25qxx.Lock=0; return false; } //################################################################################################################### bool W25qxx_IsEmptyBlock(uint32_t Block_Address,uint32_t OffsetInByte,uint32_t NumByteToCheck_up_to_BlockSize) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; if((NumByteToCheck_up_to_BlockSize>w25qxx.BlockSize)||(NumByteToCheck_up_to_BlockSize==0)) NumByteToCheck_up_to_BlockSize=w25qxx.BlockSize; #if (_W25QXX_DEBUG==1) printf("w25qxx CheckBlock:%d, Offset:%d, Bytes:%d begin...\r\n",Block_Address,OffsetInByte,NumByteToCheck_up_to_BlockSize); W25qxx_Delay(100); uint32_t StartTime=HAL_GetTick(); #endif uint8_t pBuffer[32]; uint32_t WorkAddress; uint32_t i; for(i=OffsetInByte; i=ID_W25Q256) W25qxx_Spi((WorkAddress & 0xFF000000) >> 24); W25qxx_Spi((WorkAddress & 0xFF0000) >> 16); W25qxx_Spi((WorkAddress & 0xFF00) >> 8); W25qxx_Spi(WorkAddress & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,sizeof(pBuffer),100); W25QXX_CS_HIGH(); for(uint8_t x=0;x=ID_W25Q256) W25qxx_Spi((WorkAddress & 0xFF000000) >> 24); W25qxx_Spi((WorkAddress & 0xFF0000) >> 16); W25qxx_Spi((WorkAddress & 0xFF00) >> 8); W25qxx_Spi(WorkAddress & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,1,100); W25QXX_CS_HIGH(); if(pBuffer[0]!=0xFF) goto NOT_EMPTY; } } #if (_W25QXX_DEBUG==1) printf("w25qxx CheckBlock is Empty in %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif w25qxx.Lock=0; return true; NOT_EMPTY: #if (_W25QXX_DEBUG==1) printf("w25qxx CheckBlock is Not Empty in %d ms\r\n",HAL_GetTick()-StartTime); W25qxx_Delay(100); #endif w25qxx.Lock=0; return false; } //################################################################################################################### void W25qxx_WriteByte(uint8_t pBuffer, uint32_t WriteAddr_inBytes) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; #if (_W25QXX_DEBUG==1) uint32_t StartTime=HAL_GetTick(); printf("w25qxx WriteByte 0x%02X at address %d begin...",pBuffer,WriteAddr_inBytes); #endif W25qxx_WaitForWriteEnd(); W25qxx_WriteEnable(); W25QXX_CS_LOW(); W25qxx_Spi(0x02); if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((WriteAddr_inBytes & 0xFF000000) >> 24); W25qxx_Spi((WriteAddr_inBytes & 0xFF0000) >> 16); W25qxx_Spi((WriteAddr_inBytes & 0xFF00) >> 8); W25qxx_Spi(WriteAddr_inBytes & 0xFF); W25qxx_Spi(pBuffer); W25QXX_CS_HIGH(); W25qxx_WaitForWriteEnd(); #if (_W25QXX_DEBUG==1) printf("w25qxx WriteByte done after %d ms\r\n",HAL_GetTick()-StartTime); #endif w25qxx.Lock=0; } //################################################################################################################### void W25qxx_WritePage(uint8_t *pBuffer ,uint32_t Page_Address,uint32_t OffsetInByte,uint32_t NumByteToWrite_up_to_PageSize) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; if(((NumByteToWrite_up_to_PageSize+OffsetInByte)>w25qxx.PageSize)||(NumByteToWrite_up_to_PageSize==0)) NumByteToWrite_up_to_PageSize=w25qxx.PageSize-OffsetInByte; if((OffsetInByte+NumByteToWrite_up_to_PageSize) > w25qxx.PageSize) NumByteToWrite_up_to_PageSize = w25qxx.PageSize-OffsetInByte; #if (_W25QXX_DEBUG==1) printf("w25qxx WritePage:%d, Offset:%d ,Writes %d Bytes, begin...\r\n",Page_Address,OffsetInByte,NumByteToWrite_up_to_PageSize); W25qxx_Delay(100); uint32_t StartTime=HAL_GetTick(); #endif W25qxx_WaitForWriteEnd(); W25qxx_WriteEnable(); W25QXX_CS_LOW(); W25qxx_Spi(0x02); Page_Address = (Page_Address*w25qxx.PageSize)+OffsetInByte; if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((Page_Address & 0xFF000000) >> 24); W25qxx_Spi((Page_Address & 0xFF0000) >> 16); W25qxx_Spi((Page_Address & 0xFF00) >> 8); W25qxx_Spi(Page_Address&0xFF); HAL_SPI_Transmit(&_W25QXX_SPI,pBuffer,NumByteToWrite_up_to_PageSize,100); W25QXX_CS_HIGH(); W25qxx_WaitForWriteEnd(); #if (_W25QXX_DEBUG==1) StartTime = HAL_GetTick()-StartTime; for(uint32_t i=0;i2)) { printf("\r\n"); W25qxx_Delay(10); } printf("0x%02X,",pBuffer[i]); } printf("\r\n"); printf("w25qxx WritePage done after %d ms\r\n",StartTime); W25qxx_Delay(100); #endif W25qxx_Delay(1); w25qxx.Lock=0; } //################################################################################################################### void W25qxx_WriteSector(uint8_t *pBuffer ,uint32_t Sector_Address,uint32_t OffsetInByte ,uint32_t NumByteToWrite_up_to_SectorSize) { if((NumByteToWrite_up_to_SectorSize>w25qxx.SectorSize)||(NumByteToWrite_up_to_SectorSize==0)) NumByteToWrite_up_to_SectorSize=w25qxx.SectorSize; #if (_W25QXX_DEBUG==1) printf("+++w25qxx WriteSector:%d, Offset:%d ,Write %d Bytes, begin...\r\n",Sector_Address,OffsetInByte,NumByteToWrite_up_to_SectorSize); W25qxx_Delay(100); #endif if(OffsetInByte>=w25qxx.SectorSize) { #if (_W25QXX_DEBUG==1) printf("---w25qxx WriteSector Faild!\r\n"); W25qxx_Delay(100); #endif return; } uint32_t StartPage; int32_t BytesToWrite; uint32_t LocalOffset; if((OffsetInByte+NumByteToWrite_up_to_SectorSize) > w25qxx.SectorSize) BytesToWrite = w25qxx.SectorSize-OffsetInByte; else BytesToWrite = NumByteToWrite_up_to_SectorSize; StartPage = W25qxx_SectorToPage(Sector_Address)+(OffsetInByte/w25qxx.PageSize); LocalOffset = OffsetInByte%w25qxx.PageSize; do { W25qxx_WritePage(pBuffer,StartPage,LocalOffset,BytesToWrite); StartPage++; BytesToWrite-=w25qxx.PageSize-LocalOffset; pBuffer+=w25qxx.PageSize; LocalOffset=0; }while(BytesToWrite>0); #if (_W25QXX_DEBUG==1) printf("---w25qxx WriteSector Done\r\n"); W25qxx_Delay(100); #endif } //################################################################################################################### void W25qxx_WriteBlock (uint8_t* pBuffer ,uint32_t Block_Address ,uint32_t OffsetInByte ,uint32_t NumByteToWrite_up_to_BlockSize) { if((NumByteToWrite_up_to_BlockSize>w25qxx.BlockSize)||(NumByteToWrite_up_to_BlockSize==0)) NumByteToWrite_up_to_BlockSize=w25qxx.BlockSize; #if (_W25QXX_DEBUG==1) printf("+++w25qxx WriteBlock:%d, Offset:%d ,Write %d Bytes, begin...\r\n",Block_Address,OffsetInByte,NumByteToWrite_up_to_BlockSize); W25qxx_Delay(100); #endif if(OffsetInByte>=w25qxx.BlockSize) { #if (_W25QXX_DEBUG==1) printf("---w25qxx WriteBlock Faild!\r\n"); W25qxx_Delay(100); #endif return; } uint32_t StartPage; int32_t BytesToWrite; uint32_t LocalOffset; if((OffsetInByte+NumByteToWrite_up_to_BlockSize) > w25qxx.BlockSize) BytesToWrite = w25qxx.BlockSize-OffsetInByte; else BytesToWrite = NumByteToWrite_up_to_BlockSize; StartPage = W25qxx_BlockToPage(Block_Address)+(OffsetInByte/w25qxx.PageSize); LocalOffset = OffsetInByte%w25qxx.PageSize; do { W25qxx_WritePage(pBuffer,StartPage,LocalOffset,BytesToWrite); StartPage++; BytesToWrite-=w25qxx.PageSize-LocalOffset; pBuffer+=w25qxx.PageSize; LocalOffset=0; }while(BytesToWrite>0); #if (_W25QXX_DEBUG==1) printf("---w25qxx WriteBlock Done\r\n"); W25qxx_Delay(100); #endif } //################################################################################################################### void W25qxx_ReadByte(uint8_t *pBuffer,uint32_t Bytes_Address) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; #if (_W25QXX_DEBUG==1) uint32_t StartTime=HAL_GetTick(); printf("w25qxx ReadByte at address %d begin...\r\n",Bytes_Address); #endif W25QXX_CS_LOW(); W25qxx_Spi(0x0B); if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((Bytes_Address & 0xFF000000) >> 24); W25qxx_Spi((Bytes_Address & 0xFF0000) >> 16); W25qxx_Spi((Bytes_Address& 0xFF00) >> 8); W25qxx_Spi(Bytes_Address & 0xFF); W25qxx_Spi(0); *pBuffer = W25qxx_Spi(W25QXX_DUMMY_BYTE); W25QXX_CS_HIGH(); #if (_W25QXX_DEBUG==1) printf("w25qxx ReadByte 0x%02X done after %d ms\r\n",*pBuffer,HAL_GetTick()-StartTime); #endif w25qxx.Lock=0; } //################################################################################################################### void W25qxx_ReadBytes(uint8_t* pBuffer, uint32_t ReadAddr, uint32_t NumByteToRead) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; #if (_W25QXX_DEBUG==1) uint32_t StartTime=HAL_GetTick(); printf("w25qxx ReadBytes at Address:%d, %d Bytes begin...\r\n",ReadAddr,NumByteToRead); #endif W25QXX_CS_LOW(); W25qxx_Spi(0x0B); if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((ReadAddr & 0xFF000000) >> 24); W25qxx_Spi((ReadAddr & 0xFF0000) >> 16); W25qxx_Spi((ReadAddr& 0xFF00) >> 8); W25qxx_Spi(ReadAddr & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,NumByteToRead,2000); W25QXX_CS_HIGH(); #if (_W25QXX_DEBUG==1) StartTime = HAL_GetTick()-StartTime; for(uint32_t i=0;i2)) { printf("\r\n"); W25qxx_Delay(10); } printf("0x%02X,",pBuffer[i]); } printf("\r\n"); printf("w25qxx ReadBytes done after %d ms\r\n",StartTime); W25qxx_Delay(100); #endif W25qxx_Delay(1); w25qxx.Lock=0; } //################################################################################################################### void W25qxx_ReadPage(uint8_t *pBuffer,uint32_t Page_Address,uint32_t OffsetInByte,uint32_t NumByteToRead_up_to_PageSize) { while(w25qxx.Lock==1) W25qxx_Delay(1); w25qxx.Lock=1; if((NumByteToRead_up_to_PageSize>w25qxx.PageSize)||(NumByteToRead_up_to_PageSize==0)) NumByteToRead_up_to_PageSize=w25qxx.PageSize; if((OffsetInByte+NumByteToRead_up_to_PageSize) > w25qxx.PageSize) NumByteToRead_up_to_PageSize = w25qxx.PageSize-OffsetInByte; #if (_W25QXX_DEBUG==1) printf("w25qxx ReadPage:%d, Offset:%d ,Read %d Bytes, begin...\r\n",Page_Address,OffsetInByte,NumByteToRead_up_to_PageSize); W25qxx_Delay(100); uint32_t StartTime=HAL_GetTick(); #endif Page_Address = Page_Address*w25qxx.PageSize+OffsetInByte; W25QXX_CS_LOW(); W25qxx_Spi(0x0B); if(w25qxx.ID>=ID_W25Q256) W25qxx_Spi((Page_Address & 0xFF000000) >> 24); W25qxx_Spi((Page_Address & 0xFF0000) >> 16); W25qxx_Spi((Page_Address& 0xFF00) >> 8); W25qxx_Spi(Page_Address & 0xFF); W25qxx_Spi(0); HAL_SPI_Receive(&_W25QXX_SPI,pBuffer,NumByteToRead_up_to_PageSize,100); W25QXX_CS_HIGH(); #if (_W25QXX_DEBUG==1) StartTime = HAL_GetTick()-StartTime; for(uint32_t i=0;i2)) { printf("\r\n"); W25qxx_Delay(10); } printf("0x%02X,",pBuffer[i]); } printf("\r\n"); printf("w25qxx ReadPage done after %d ms\r\n",StartTime); W25qxx_Delay(100); #endif W25qxx_Delay(1); w25qxx.Lock=0; } //################################################################################################################### void W25qxx_ReadSector(uint8_t *pBuffer,uint32_t Sector_Address,uint32_t OffsetInByte,uint32_t NumByteToRead_up_to_SectorSize) { if((NumByteToRead_up_to_SectorSize>w25qxx.SectorSize)||(NumByteToRead_up_to_SectorSize==0)) NumByteToRead_up_to_SectorSize=w25qxx.SectorSize; #if (_W25QXX_DEBUG==1) printf("+++w25qxx ReadSector:%d, Offset:%d ,Read %d Bytes, begin...\r\n",Sector_Address,OffsetInByte,NumByteToRead_up_to_SectorSize); W25qxx_Delay(100); #endif if(OffsetInByte>=w25qxx.SectorSize) { #if (_W25QXX_DEBUG==1) printf("---w25qxx ReadSector Faild!\r\n"); W25qxx_Delay(100); #endif return; } uint32_t StartPage; int32_t BytesToRead; uint32_t LocalOffset; if((OffsetInByte+NumByteToRead_up_to_SectorSize) > w25qxx.SectorSize) BytesToRead = w25qxx.SectorSize-OffsetInByte; else BytesToRead = NumByteToRead_up_to_SectorSize; StartPage = W25qxx_SectorToPage(Sector_Address)+(OffsetInByte/w25qxx.PageSize); LocalOffset = OffsetInByte%w25qxx.PageSize; do { W25qxx_ReadPage(pBuffer,StartPage,LocalOffset,BytesToRead); StartPage++; BytesToRead-=w25qxx.PageSize-LocalOffset; pBuffer+=w25qxx.PageSize; LocalOffset=0; }while(BytesToRead>0); #if (_W25QXX_DEBUG==1) printf("---w25qxx ReadSector Done\r\n"); W25qxx_Delay(100); #endif } //################################################################################################################### void W25qxx_ReadBlock(uint8_t* pBuffer,uint32_t Block_Address,uint32_t OffsetInByte,uint32_t NumByteToRead_up_to_BlockSize) { if((NumByteToRead_up_to_BlockSize>w25qxx.BlockSize)||(NumByteToRead_up_to_BlockSize==0)) NumByteToRead_up_to_BlockSize=w25qxx.BlockSize; #if (_W25QXX_DEBUG==1) printf("+++w25qxx ReadBlock:%d, Offset:%d ,Read %d Bytes, begin...\r\n",Block_Address,OffsetInByte,NumByteToRead_up_to_BlockSize); W25qxx_Delay(100); #endif if(OffsetInByte>=w25qxx.BlockSize) { #if (_W25QXX_DEBUG==1) printf("w25qxx ReadBlock Faild!\r\n"); W25qxx_Delay(100); #endif return; } uint32_t StartPage; int32_t BytesToRead; uint32_t LocalOffset; if((OffsetInByte+NumByteToRead_up_to_BlockSize) > w25qxx.BlockSize) BytesToRead = w25qxx.BlockSize-OffsetInByte; else BytesToRead = NumByteToRead_up_to_BlockSize; StartPage = W25qxx_BlockToPage(Block_Address)+(OffsetInByte/w25qxx.PageSize); LocalOffset = OffsetInByte%w25qxx.PageSize; do { W25qxx_ReadPage(pBuffer,StartPage,LocalOffset,BytesToRead); StartPage++; BytesToRead-=w25qxx.PageSize-LocalOffset; pBuffer+=w25qxx.PageSize; LocalOffset=0; }while(BytesToRead>0); #if (_W25QXX_DEBUG==1) printf("---w25qxx ReadBlock Done\r\n"); W25qxx_Delay(100); #endif } #endif //-----------------------------------USE_W25QXX_SPI------------------------------------------// //###################################################################################################################