329 lines
13 KiB
C
Raw Normal View History

/******************************************************************************************************************************************
* : SWM341_sfc.c
* : SWM341单片机的SFCSerial Flash Controller
* : http://www.synwit.com.cn/e/tool/gbook/?bid=1
* :
* : V1.1.0 20171025
* :
*
*
*******************************************************************************************************************************************
* @attention
*
* THE PRESENT FIRMWARE WHICH IS FOR GUIDANCE ONLY AIMS AT PROVIDING CUSTOMERS WITH CODING INFORMATION
* REGARDING THEIR PRODUCTS IN ORDER FOR THEM TO SAVE TIME. AS A RESULT, SYNWIT SHALL NOT BE HELD LIABLE
* FOR ANY DIRECT, INDIRECT OR CONSEQUENTIAL DAMAGES WITH RESPECT TO ANY CLAIMS ARISING FROM THE CONTENT
* OF SUCH FIRMWARE AND/OR THE USE MADE BY CUSTOMERS OF THE CODING INFORMATION CONTAINED HEREIN IN CONN-
* -ECTION WITH THEIR PRODUCTS.
*
* COPYRIGHT 2012 Synwit Technology
*******************************************************************************************************************************************/
#include "SWM341.h"
#include "SWM341_sfc.h"
/******************************************************************************************************************************************
* : SFC_Init()
* : SFCSerial Flash Controller
* : SFC_InitStructure * initStruct SFC初始化配置值
* :
* :
******************************************************************************************************************************************/
void SFC_Init(SFC_InitStructure * initStruct)
{
SYS->CLKEN1 |= (1 << SYS_CLKEN1_SFC_Pos);
*((__IO uint32_t *)((uint32_t )&SFC->CFG + 0x3F4)) = 7;
SFC->CFG &= ~(SFC_CFG_CLKDIV_Msk | SFC_CFG_DATA4L_RD_Msk | SFC_CFG_DATA4L_PP_Msk);
SFC->CFG |= (initStruct->ClkDiv << SFC_CFG_CLKDIV_Pos) |
(initStruct->Width_Read << SFC_CFG_DATA4L_RD_Pos) |
(initStruct->Width_PageProgram << SFC_CFG_DATA4L_PP_Pos);
SFC->CFG |= (1 << SFC_CFG_CMDWREN_Pos);
SFC->CMDAHB &= ~(SFC_CMDAHB_READ_Msk | SFC_CMDAHB_PP_Msk);
SFC->CMDAHB |= (initStruct->Cmd_Read << SFC_CMDAHB_READ_Pos) |
(initStruct->Cmd_PageProgram << SFC_CMDAHB_PP_Pos);
SFC->CFG &= ~(1 << SFC_CFG_CMDWREN_Pos);
SFC->TIM &= ~(SFC_TIM_WIP_CHK_ITV_Msk | SFC_TIM_WIP_CHK_LMT_Msk);
SFC->TIM |= ((CyclesPerUs / 10) << SFC_TIM_WIP_CHK_ITV_Pos) | //2048 * (CyclesPerUs / 10) / CyclesPerUs us = 0.2 ms
(255 << SFC_TIM_WIP_CHK_LMT_Pos);
2023-12-12 18:23:12 +08:00
if((initStruct->Width_Read == SFC_RDWIDTH_4) || (initStruct->Width_PageProgram == SFC_PPWIDTH_4))
{
if(SFC_QuadState() == 0)
SFC_QuadSwitch(1);
}
}
/******************************************************************************************************************************************
* : SFC_ReadJEDEC()
* : JEDEC ID
* :
* : uint32_t JEDEC ID
* :
******************************************************************************************************************************************/
uint32_t SFC_ReadJEDEC(void)
{
SFC->CFG &= ~SFC_CFG_CMDTYPE_Msk;
SFC->CFG |= (1 << SFC_CFG_CMDWREN_Pos) |
(2 << SFC_CFG_CMDTYPE_Pos);
SFC->CMD = SFC_CMD_READ_JEDEC;
SFC->GO = 1;
2023-12-12 18:23:12 +08:00
__DSB(); __ISB();
while(SFC->GO) __NOP();
return SFC->DATA;
}
/******************************************************************************************************************************************
* : SFC_Erase()
* : SPI Flash扇区擦除4K字节
* : uint32_t addr 4K对齐addr%4096 == 0
* uint8_t wait 1 Flash 0
* :
* :
******************************************************************************************************************************************/
void SFC_Erase(uint32_t addr, uint8_t wait)
{
SFC_EraseEx(addr, SFC_CMD_ERASE_SECTOR, wait);
}
/******************************************************************************************************************************************
* : SFC_EraseEx()
* : SPI Flash擦除
* : uint32_t addr addr == 0xFFFFFFFF
* uint8_t cmd SPI Flash支持多种大小的块的擦除使
* uint8_t wait 1 Flash 0
* :
* :
******************************************************************************************************************************************/
void SFC_EraseEx(uint32_t addr, uint8_t cmd, uint8_t wait)
{
uint8_t type = (addr == 0xFFFFFFFF) ? 5 : 7;
SFC->ADDR = addr;
SFC->CFG &= ~SFC_CFG_CMDTYPE_Msk;
SFC->CFG |= (1 << SFC_CFG_WREN_Pos) |
(1 << SFC_CFG_CMDWREN_Pos) |
(type << SFC_CFG_CMDTYPE_Pos);
SFC->CMD = cmd;
2023-12-12 18:23:12 +08:00
SFC->GO = 1;
__DSB(); __ISB();
while(SFC->GO) __NOP();
SFC->CFG &= ~SFC_CFG_WREN_Msk;
if(wait)
while(SFC_FlashBusy()) __NOP();
}
/******************************************************************************************************************************************
* : SFC_Write()
* : SPI Flash数据写入
* : uint32_t addr Flash中的地址
* uint32_t buff[] Flash中的数据
* uint32_t cnt 64
* :
* : addr/256 == (addr+(cnt-1)*4)/256
* cnt > 4 LCD_DCLK |__||__||__||__||__||__|
* cnt = 4 SFC_Write()
******************************************************************************************************************************************/
void SFC_Write(uint32_t addr, uint32_t buff[], uint32_t cnt)
{
SFC->CFG |= (1 << SFC_CFG_WREN_Pos);
for(int i = 0; i < cnt; i++)
*((volatile unsigned int *)(0x70000000+addr+i*4)) = buff[i];
while(SFC->SR & SFC_SR_BUSY_Msk) __NOP();
SFC->CFG &= ~SFC_CFG_WREN_Msk;
}
2023-12-12 18:23:12 +08:00
#define IOSPI_CS_Low() GPIO_ClrBit(GPIOD, PIN6); __NOP(); __NOP(); __NOP(); __NOP()
#define IOSPI_CS_High() __NOP(); __NOP(); __NOP(); __NOP(); GPIO_SetBit(GPIOD, PIN6)
#define IOSPI_CLK_Low() GPIO_ClrBit(GPIOD, PIN5); __NOP(); __NOP()
#define IOSPI_CLK_High() __NOP(); __NOP(); GPIO_SetBit(GPIOD, PIN5)
#define IOSPI_MOSI_Low() GPIO_ClrBit(GPIOD, PIN8)
#define IOSPI_MOSI_High() GPIO_SetBit(GPIOD, PIN8)
#define IOSPI_MISO_Value() GPIO_GetBit(GPIOD, PIN7)
static uint8_t IOSPI_ReadWrite(uint8_t data)
{
uint8_t val = 0;
for(int i = 0; i < 8; i++)
{
IOSPI_CLK_Low();
if(data & (1 << (7 - i)))
IOSPI_MOSI_High();
else
IOSPI_MOSI_Low();
IOSPI_CLK_High();
val = (val << 1) | IOSPI_MISO_Value();
}
return val;
}
/******************************************************************************************************************************************
* : SFC_GPIOWrite()
* : SFC GPIO SPI
* : uint32_t addr Flash中的地址
* uint32_t buff[] Flash中的数据
* uint32_t cnt 64
* :
* : GPIO 使 SFC 便使 SFC
******************************************************************************************************************************************/
void SFC_GPIOWrite(uint32_t addr, uint32_t buff[], uint32_t cnt)
{
IOSPI_CS_Low();
IOSPI_ReadWrite(SFC_CMD_WRITE_ENABLE);
IOSPI_CS_High();
IOSPI_CS_Low();
IOSPI_ReadWrite(SFC_CMD_PAGE_PROGRAM);
IOSPI_ReadWrite(addr >> 16);
IOSPI_ReadWrite(addr >> 8);
IOSPI_ReadWrite(addr);
for(int i = 0; i < cnt * 4; i++)
{
IOSPI_ReadWrite(((uint8_t *)buff)[i]);
}
IOSPI_CS_High();
int busy;
do {
IOSPI_CS_Low();
IOSPI_ReadWrite(SFC_CMD_READ_STATUS_REG1);
busy = IOSPI_ReadWrite(0xFF) & (1 << SFC_STATUS_REG_BUSY_Pos);
IOSPI_CS_High();
} while(busy);
}
/******************************************************************************************************************************************
* : SFC_Read()
* : SPI Flash数据读取
* : uint32_t addr Flash中的地址
* uint32_t buff[] buff指向的内存
* uint32_t cnt
* :
* :
******************************************************************************************************************************************/
void SFC_Read(uint32_t addr, uint32_t buff[], uint32_t cnt)
{
for(int i = 0; i < cnt; i++)
buff[i] = *((volatile unsigned int *)(0x70000000+addr+i*4));
}
/******************************************************************************************************************************************
* : SFC_ReadStatusReg()
* : SPI Flash读取状态寄存器
* : uint8_t cmd 使
* : uint8_t
* :
******************************************************************************************************************************************/
uint8_t SFC_ReadStatusReg(uint8_t cmd)
{
SFC->CFG &= ~SFC_CFG_CMDTYPE_Msk;
SFC->CFG |= (1 << SFC_CFG_CMDWREN_Pos) |
(1 << SFC_CFG_CMDTYPE_Pos);
SFC->CMD = cmd;
SFC->GO = 1;
2023-12-12 18:23:12 +08:00
__DSB(); __ISB();
while(SFC->GO) __NOP();
return SFC->DATA;
}
/******************************************************************************************************************************************
* : SFC_WriteStatusReg()
* : SPI Flash写入状态寄存器
* : uint8_t cmd 使
* uint16_t reg
* :
* :
******************************************************************************************************************************************/
void SFC_WriteStatusReg(uint8_t cmd, uint16_t reg)
{
SFC->CFG &= ~SFC_CFG_CMDTYPE_Msk;
SFC->CFG |= (1 << SFC_CFG_WREN_Pos) |
(1 << SFC_CFG_CMDWREN_Pos) |
(6 << SFC_CFG_CMDTYPE_Pos);
SFC->CMD = cmd;
SFC->DATA = reg;
SFC->GO = 1;
2023-12-12 18:23:12 +08:00
__DSB(); __ISB();
while(SFC->GO) __NOP();
}
/******************************************************************************************************************************************
* : SFC_QuadSwitch()
* : SPI Flash Quad模式开关
* : uint8_t on 1 Quad 0 Quad
* :
* :
******************************************************************************************************************************************/
void SFC_QuadSwitch(uint8_t on)
{
uint16_t reg = (SFC_ReadStatusReg(SFC_CMD_READ_STATUS_REG2) << 8) |
(SFC_ReadStatusReg(SFC_CMD_READ_STATUS_REG1) << 0);
if(on)
reg |= (1 << SFC_STATUS_REG_QUAD_Pos);
else
reg &=~(1 << SFC_STATUS_REG_QUAD_Pos);
SFC_WriteStatusReg(SFC_CMD_WRITE_STATUS_REG1, reg);
}
/******************************************************************************************************************************************
* : SFC_QuadState()
* : SPI Flash Quad模式开关状态查询
* :
* : 1 Quad 0 Quad
* :
******************************************************************************************************************************************/
uint8_t SFC_QuadState(void)
{
uint8_t reg = SFC_ReadStatusReg(SFC_CMD_READ_STATUS_REG2);
if(reg & (1 << (SFC_STATUS_REG_QUAD_Pos - 8)))
return 1;
else
return 0;
}
/******************************************************************************************************************************************
* : SFC_FlashBusy()
* : SPI Flash Page ProgramSector EraseBlock EraseChip Erase
* :
* : 1 Flash Erase/Write 0 Flash Erase/Write
* :
******************************************************************************************************************************************/
uint8_t SFC_FlashBusy(void)
{
uint8_t reg = SFC_ReadStatusReg(SFC_CMD_READ_STATUS_REG1);
if(reg & (1 << SFC_STATUS_REG_BUSY_Pos))
return 1;
else
return 0;
}