Files

5534 lines
181 KiB
C
Executable File

/*
* halSERFLASH.c- Sigmastar
*
* Copyright (c) [2019~2020] SigmaStar Technology.
*
*
* This software is licensed under the terms of the GNU General Public
* License version 2, as published by the Free Software Foundation, and
* may be copied, distributed, and modified under those terms.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License version 2 for more details.
*
*/
#include <linux/string.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/timer.h>
#include <linux/jiffies.h>
#include <asm/io.h>
// Common Definition
#include "../include/ms_platform.h"
#include "../include/ms_msys.h"
// Internal Definition
#include "drvSERFLASH.h"
#include "drvDeviceInfo.h"
#include "regSERFLASH.h"
#include "halSERFLASH.h"
#include "registers.h"
// !!! Uranus Serial Flash Notes: !!!
// - The clock of DMA & Read via XIU operations must be < 3*CPU clock
// - The clock of DMA & Read via XIU operations are determined by only REG_ISP_CLK_SRC; other operations by REG_ISP_CLK_SRC only
// - DMA program can't run on DRAM, but in flash ONLY
// - DMA from SPI to DRAM => size/DRAM start/DRAM end must be 8-B aligned
//-------------------------------------------------------------------------------------------------
// Driver Compiler Options
//-------------------------------------------------------------------------------------------------
//-------------------------------------------------------------------------------------------------
// Local Defines
//-------------------------------------------------------------------------------------------------
#if 0
//Both read and write via IRUISP
#define CONFIG_RIUISP 1
#define XIUREAD_MODE 0 //not use
#else
//Select FSP read function
//#define CONFIG_FSP_READ_RIUOP 1
#define CONFIG_FSP_READ_BDMA 1
//#define CONFIG_FSP_READ_DIRERECT 1
//Select FSP write function
//#define CONFIG_FSP_WRITE_RIUOP 1
//#define CONFIG_FSP_WRITE_RIUOP_BRUST 1
#define CONFIG_FSP_WRITE_BDMA 1
#endif
#if defined(CONFIG_FSP_READ_BDMA) && CONFIG_FSP_READ_BDMA == 1
#include "hal_bdma.h"
#endif
#define READ_BYTE(_reg) (*(volatile MS_U8*)(_reg))
#define READ_WORD(_reg) (*(volatile MS_U16*)(_reg))
#define READ_LONG(_reg) (*(volatile MS_U32*)(_reg))
#define WRITE_BYTE(_reg, _val) {(*((volatile MS_U8*)(_reg))) = (MS_U8)(_val); }
#define WRITE_WORD(_reg, _val) {(*((volatile MS_U16*)(_reg))) = (MS_U16)(_val); }
#define WRITE_LONG(_reg, _val) {(*((volatile MS_U32*)(_reg))) = (MS_U32)(_val); }
#define WRITE_WORD_MASK(_reg, _val, _mask) {(*((volatile MS_U16*)(_reg))) = ((*((volatile MS_U16*)(_reg))) & ~(_mask)) | ((MS_U16)(_val) & (_mask)); }
//#define WRITE_WORD_MASK(_reg, _val, _mask) {*((volatile MS_U16*)(_reg)) = (*((volatile MS_U16*)(_reg))) & ~(_mask)) | ((MS_U16)(_val) & (_mask); }
// XIU_ADDR
// #define SFSH_XIU_REG32(addr) (*((volatile MS_U32 *)(_hal_isp.u32XiuBaseAddr + ((addr)<<2))))
//#define SFSH_XIU_READ32(addr) (*((volatile MS_U32 *)(_hal_isp.u32XiuBaseAddr + ((addr)<<2)))) // TODO: check AEON 32 byte access order issue
//
// ISP_CMD
// #define ISP_REG16(addr) (*((volatile MS_U16 *)(_hal_isp.u32IspBaseAddr + ((addr)<<2))))
#define ISP_READ(addr) READ_WORD(_hal_isp.u32IspBaseAddr + ((addr)<<2))
#define ISP_WRITE(addr, val) WRITE_WORD(_hal_isp.u32IspBaseAddr + ((addr)<<2), (val))
#define ISP_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32IspBaseAddr + ((addr)<<2), (val), (mask))
#define FSP_READ(addr) READ_WORD(_hal_isp.u32FspBaseAddr + ((addr)<<2))
#define FSP_WRITE(addr, val) WRITE_WORD(_hal_isp.u32FspBaseAddr + ((addr)<<2), (val))
#define FSP_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32FspBaseAddr + ((addr)<<2), (val), (mask))
#define QSPI_READ(addr) READ_WORD(_hal_isp.u32QspiBaseAddr + ((addr)<<2))
#define QSPI_WRITE(addr, val) WRITE_WORD(_hal_isp.u32QspiBaseAddr + ((addr)<<2), (val))
#define QSPI_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32QspiBaseAddr + ((addr)<<2), (val), (mask))
#define SPI_FLASH_CMD(u8FLASHCmd) ISP_WRITE(REG_ISP_SPI_COMMAND, (MS_U8)u8FLASHCmd)
#define SPI_WRITE_DATA(u8Data) ISP_WRITE(REG_ISP_SPI_WDATA, (MS_U8)u8Data)
#define SPI_READ_DATA() READ_BYTE(_hal_isp.u32IspBaseAddr + ((REG_ISP_SPI_RDATA)<<2))
//#define MHEG5_READ(addr) READ_WORD(_hal_isp.u32Mheg5BaseAddr + ((addr)<<2))
//#define MHEG5_WRITE(addr, val) WRITE_WORD((_hal_isp.u32Mheg5BaseAddr + (addr << 2)), (val))
//#define MHEG5_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32Mheg5BaseAddr + ((addr)<<2), (val), (mask))
// PIU_DMA
#define PIU_READ(addr) READ_WORD(_hal_isp.u32PiuBaseAddr + ((addr)<<2))
#define PIU_WRITE(addr, val) WRITE_WORD(_hal_isp.u32PiuBaseAddr + ((addr)<<2), (val))
#define PIU_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32PiuBaseAddr + ((addr)<<2), (val), (mask))
//// PM_SLEEP CMD.
//#define PM_READ(addr) READ_WORD(_hal_isp.u32PMBaseAddr+ ((addr)<<2))
#define PM_READ(addr) READ_WORD(BASE_REG_PMSLEEP_ADDR+ ((addr)<<2))
//#define PM_WRITE(addr, val) WRITE_WORD(_hal_isp.u32PMBaseAddr+ ((addr)<<2), (val))
//#define PM_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32PMBaseAddr+ ((addr)<<2), (val), (mask))
//#define PM_GPIO_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32PMGPIOBaseAddr+ ((addr)<<2), (val), (mask))
// CLK_GEN
//#define CLK_READ(addr) READ_WORD(_hal_isp.u32CLK0BaseAddr + ((addr)<<2))
//#define CLK_WRITE(addr, val) WRITE_WORD(_hal_isp.u32CLK0BaseAddr + ((addr)<<2), (val))
//#define CLK_WRITE_MASK(addr, val, mask) WRITE_WORD_MASK(_hal_isp.u32CLK0BaseAddr + ((addr)<<2), (val), (mask))
//MS_U32<->MS_U16
#define LOU16(u32Val) ((MS_U16)(u32Val))
#define HIU16(u32Val) ((MS_U16)((u32Val) >> 16))
//serial flash mutex wait time
#define SERFLASH_MUTEX_WAIT_TIME 3000
// Time-out system
#if !defined (MSOS_TYPE_NOS) && !defined(MSOS_TYPE_LINUX)
#define SERFLASH_SAFETY_FACTOR 10
#define SER_FLASH_TIME(_stamp, _msec) { (_stamp) = MsOS_GetSystemTime() + (_msec); }
#define SER_FLASH_EXPIRE(_stamp) ( (MsOS_GetSystemTime() > (_stamp)) ? 1 : 0 )
#else // defined (MSOS_TYPE_NOS)
#define SERFLASH_SAFETY_FACTOR 3000*30
#define SER_FLASH_TIME(_stamp) (do_gettimeofday(&_stamp))
#define SER_FLASH_EXPIRE(_stamp,_msec) (_Hal_GetMsTime(_stamp, _msec))
#endif
#define MAX_READY_WAIT_JIFFIES 40*HZ
#define CHK_NUM_WAITDONE 2000
#define SINGLE_WRITE_LENGTH 4
#define SINGLE_READ_LENGTH 8
//-------------------------------------------------------------------------------------------------
// Local Structures
//-------------------------------------------------------------------------------------------------
typedef struct
{
// MS_U32 u32XiuBaseAddr; // REG_SFSH_XIU_BASE
// MS_U32 u32Mheg5BaseAddr;
MS_U32 u32IspBaseAddr; // REG_ISP_BASE
MS_U32 u32FspBaseAddr; // REG_FSP_BASE
MS_U32 u32QspiBaseAddr; // REG_QSPI_BASE
// MS_U32 u32PiuBaseAddr; // REG_PIU_BASE
// MS_U32 u32PMBaseAddr; // REG_PM_BASE
// MS_U32 u32CLK0BaseAddr; // REG_PM_BASE
MS_U32 u32BdmaBaseAddr; // for supernova.lite
// MS_U32 u32RiuBaseAddr;
// MS_U32 u32PMGPIOBaseAddr;
} hal_isp_t;
//-------------------------------------------------------------------------------------------------
// Global Variables
//-------------------------------------------------------------------------------------------------
hal_SERFLASH_t _hal_SERFLASH;
MS_U8 _u8SERFLASHDbgLevel;
MS_BOOL _bFSPMode = 0;
MS_BOOL _bXIUMode = 0; // default XIU mode, set 0 to RIU mode
MS_BOOL bDetect = FALSE; // initial flasg : true and false
MS_BOOL _bIBPM = FALSE; // Individual Block Protect mode : true and false
MS_BOOL _bWPPullHigh = 0; // WP pin pull high or can control info
MS_BOOL _bHasEAR = FALSE; // Extended Address Register mode supported
MS_U8 _u8RegEAR = 0xFF; // Extended Address Register value
MS_U8 u8Mode;
MS_U32 pu8BDMA_phys = 0;
MS_U32 pu8BDMA_virt = 0;
MS_U32 pu8BDMA_bus = 0;
MS_U32 BASE_FLASH_OFFSET = 0;
#define BDMA_ALIGN (32)
#define BDMA_SIZE_WARNING (128*1024+BDMA_ALIGN) //print message for unresonable size
//E_QUAD_MODE;//E_FAST_MODE;
const static SPI_READ_MODE gReadMode = E_QUAD_MODE;
static MS_U32 u32BdmaSize = 64 * 1024 + BDMA_ALIGN;//default size
static MSYS_DMEM_INFO mem_info;
MS_BOOL gQuadSupport = 0; //extern on MTD
//-------------------------------------------------------------------------------------------------
// Local Variables
//-------------------------------------------------------------------------------------------------
static MS_S32 _s32SERFLASH_Mutex;
//
// Spi Clk Table (List)
//
/*
static MS_U16 _hal_ckg_spi_pm[] = {
PM_SPI_CLK_XTALI
,PM_SPI_CLK_54MHZ
,PM_SPI_CLK_86MHZ
,PM_SPI_CLK_108MHZ
};
static MS_U16 _hal_ckg_spi_nonpm[] = {
CLK0_CKG_SPI_XTALI
,CLK0_CKG_SPI_54MHZ
,CLK0_CKG_SPI_86MHZ
,CLK0_CKG_SPI_108MHZ
};
*/
static hal_isp_t _hal_isp =
{
//.u32XiuBaseAddr = BASEADDR_XIU,
//.u32Mheg5BaseAddr = BASEADDR_RIU + BK_MHEG5,
.u32IspBaseAddr = BASE_REG_ISP_ADDR,
.u32FspBaseAddr = BASE_REG_FSP_ADDR,
.u32QspiBaseAddr = BASE_REG_QSPI_ADDR,
//.u32PiuBaseAddr = BASEADDR_RIU + BK_PIU,
//.u32PMBaseAddr = BASEADDR_RIU + BK_PMSLP,
//.u32CLK0BaseAddr = BASEADDR_NonPMBankRIU + BK_CLK0,
.u32BdmaBaseAddr = BASE_REG_BDMACh0_ADDR,
// .u32RiuBaseAddr= IO_ADDRESS(MS_BASE_REG_RIU_PA),
//.u32PMGPIOBaseAddr=BASEADDR_RIU+BK_PMGPIO,
};
// For linux, thread sync is handled by mtd. So, these functions are empty.
#define MSOS_PROCESS_PRIVATE 0x00000000
#define MSOS_PROCESS_SHARED 0x00000001
//static spinlock_t _gtSpiLock;
//static MS_U32 _gtSpiFlag;
/// Suspend type
/*typedef enum
{
E_MSOS_PRIORITY, ///< Priority-order suspension
E_MSOS_FIFO, ///< FIFO-order suspension
} MsOSAttribute;
extern void *high_memory;
extern void (*_dma_cache_wback_inv)(unsigned long start, unsigned long size);*/
void (*_HAL_FSP_Write_Callback_Func)(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data);
extern MS_BOOL MS_SERFLASH_IN_INTERRUPT (void);
extern MS_S32 MS_SERFLASH_CREATE_MUTEX ( MsOSAttribute eAttribute, char *pMutexName, MS_U32 u32Flag);
extern MS_BOOL MS_SERFLASH_DELETE_MUTEX(MS_S32 s32MutexId);
extern MS_BOOL MS_SERFLASH_OBTAIN_MUTEX (MS_S32 s32MutexId, MS_U32 u32WaitMs);
extern MS_BOOL MS_SERFLASH_RELEASE_MUTEX (MS_S32 s32MutexId);
//-------------------------------------------------------------------------------------------------
// Debug Functions
//-------------------------------------------------------------------------------------------------
//-------------------------------------------------------------------------------------------------
// Local Functions
//-------------------------------------------------------------------------------------------------
static void _HAL_SPI_Rest(void);
static void _HAL_ISP_Enable(void);
static void _HAL_ISP_Disable(void);
static void _HAL_ISP_2XMode(MS_BOOL bEnable);
static MS_BOOL _HAL_SERFLASH_WaitWriteCmdRdy(void);
static MS_BOOL _HAL_SERFLASH_WaitWriteDataRdy(void);
static MS_BOOL _HAL_SERFLASH_WaitReadDataRdy(void);
static MS_BOOL _HAL_SERFLASH_WaitWriteDone(void);
#ifdef CONFIG_RIUISP
static MS_BOOL _HAL_SERFLASH_CheckWriteDone(void);
//static MS_BOOL _HAL_SERFLASH_XIURead(MS_U32 u32Addr,MS_U32 u32Size,MS_U8 * pu8Data);
static MS_BOOL _HAL_SERFLASH_RIURead(MS_U32 u32Addr,MS_U32 u32Size,MS_U8 * pu8Data);
#endif
static void _HAL_SERFLASH_ActiveFlash_Set_HW_WP(MS_BOOL bEnable);
MS_BOOL HAL_FSP_EraseChip(void);
MS_BOOL HAL_FSP_CheckWriteDone(void);
MS_BOOL HAL_FSP_ReadREMS4(MS_U8 * pu8Data, MS_U32 u32Size);
void HAL_FSP_Entry(void);
void HAL_FSP_Exit(void);
void _HAL_BDMA_INIT(U32 u32DataSize);
//-------------------------------------------------------------------------------------------------
// Debug Functions
//-------------------------------------------------------------------------------------------------
//BDMA_Result MDrv_BDMA_CopyHnd(MS_PHYADDR u32SrcAddr, MS_PHYADDR u32DstAddr, MS_U32 u32Len, BDMA_CpyType eCpyType, MS_U8 u8OpCfg)
//{
// return -1;
//}
static MS_BOOL _Hal_GetMsTime( struct timeval tPreTime, MS_U32 u32Fac)
{
MS_U32 u32NsTime = 0;
struct timeval time_st;
do_gettimeofday(&time_st);
u32NsTime = ((time_st.tv_sec - tPreTime.tv_sec) * 1000) + ((time_st.tv_usec - tPreTime.tv_usec)/1000);
if(u32NsTime > u32Fac)
return TRUE;
return FALSE;
}
//-------------------------------------------------------------------------------------------------
// check if pm51 on SPI
// @return TRUE : succeed
// @return FALSE : fail
// @note :
//-------------------------------------------------------------------------------------------------
/*static MS_BOOL _HAL_SERFLASH_Check51RunMode(void)
{
MS_U8 u8PM51RunMode;
u8PM51RunMode = PM_READ(REG_PM_CHK_51MODE);
if((u8PM51RunMode & PM_51_ON_SPI))
return FALSE;
else
return TRUE;
}*/
void HAL_SERFLASH_SelectReadMode(SPI_READ_MODE eReadMode)
{
switch(eReadMode)
{
case E_SINGLE_MODE://1-1-1 mode (SPI COMMAND is 0x03)
QSPI_WRITE_MASK(REG_ISP_SPI_MODE, SFSH_CHIP_NORMOL_ENABLE, SFSH_CHIP_FAST_MASK);
break;
case E_FAST_MODE://1-1-1 mode (SPI COMMAND is 0x0B)
QSPI_WRITE_MASK(REG_ISP_SPI_MODE, SFSH_CHIP_FAST_ENABLE, SFSH_CHIP_FAST_MASK);
break;
case E_DUAL_D_MODE: //1-1-2 mode (SPI COMMAND is 0x3B)
QSPI_WRITE_MASK(REG_ISP_SPI_MODE, SFSH_CHIP_DUALREAD_ENABLE, SFSH_CHIP_FAST_MASK);
break;
case E_DUAL_AD_MODE: //1-2-2 mode (SPI COMMAND is 0xBB)
QSPI_WRITE_MASK(REG_ISP_SPI_MODE, SFSH_CHIP_2XREAD_ENABLE, SFSH_CHIP_FAST_MASK);
break;
case E_QUAD_MODE: //1-1-4 mode (SPI COMMAND is 0x6B)
QSPI_WRITE_MASK(REG_ISP_SPI_MODE, SFSH_CHIP_4XREAD_ENABLE , SFSH_CHIP_FAST_MASK);
break;
case E_RIUISP_MODE:
default:
break;
}
}
//-------------------------------------------------------------------------------------------------
// Software reset spi_burst
// @return TRUE : succeed
// @return FALSE : fail
// @note : If no spi reset, it may cause BDMA fail.
//-------------------------------------------------------------------------------------------------
static void _HAL_SPI_Rest(void)
{
// mark for A3
#if 1
ISP_WRITE_MASK(REG_ISP_CHIP_RST, SFSH_CHIP_RESET, SFSH_CHIP_RESET_MASK);
ISP_WRITE_MASK(REG_ISP_CHIP_RST, SFSH_CHIP_NOTRESET, SFSH_CHIP_RESET_MASK);
// Call the callback function to switch back the chip selection.
if(McuChipSelectCB != NULL )
{
(*McuChipSelectCB)();
}
#endif
}
//-------------------------------------------------------------------------------------------------
// Enable RIU ISP engine
// @return TRUE : succeed
// @return FALSE : fail
// @note : If Enable ISP engine, the XIU mode does not work
//-------------------------------------------------------------------------------------------------
static void _HAL_ISP_Enable(void)
{
ISP_WRITE(REG_ISP_PASSWORD, 0xAAAA);
}
//-------------------------------------------------------------------------------------------------
// Disable RIU ISP engine
// @return TRUE : succeed
// @return FALSE : fail
// @note : If Disable ISP engine, the XIU mode works
//-------------------------------------------------------------------------------------------------
static void _HAL_ISP_Disable(void)
{
ISP_WRITE(REG_ISP_PASSWORD, 0x5555);
_HAL_SPI_Rest();
}
//-------------------------------------------------------------------------------------------------
// Enable/Disable address and data dual mode (SPI command is 0xBB)
// @return TRUE : succeed
// @return FALSE : fail
//-------------------------------------------------------------------------------------------------
static void _HAL_ISP_2XMode(MS_BOOL bEnable)
{
if(bEnable) // on 2Xmode
{
QSPI_WRITE_MASK(REG_ISP_SPI_MODE,SFSH_CHIP_2XREAD_ENABLE,SFSH_CHIP_FAST_MASK);
}
else // off 2Xmode
{
QSPI_WRITE_MASK(REG_ISP_SPI_MODE,SFSH_CHIP_FAST_ENABLE,SFSH_CHIP_FAST_MASK);
}
}
//-------------------------------------------------------------------------------------------------
// Enable/Disable address and data dual mode (SPI command is 0xBB)
// @return TRUE : succeed
// @return FALSE : fail
//-------------------------------------------------------------------------------------------------
static void _HAL_ISP_4XMode(MS_BOOL bEnable)
{
if(bEnable) // on 4Xmode
{
QSPI_WRITE_MASK(REG_ISP_SPI_MODE,SFSH_CHIP_4XREAD_ENABLE,SFSH_CHIP_FAST_MASK);
printk("[SerFlash]TODO: correct Quad mode GPIO\n");
//PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_HOLD_NOT_GPIO, PM_SPI_HOLD_GPIO_MASK);
//PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_WP_NOT_GPIO, PM_SPI_WP_GPIO_MASK);
}
else // off 4Xmode
{
QSPI_WRITE_MASK(REG_ISP_SPI_MODE,SFSH_CHIP_FAST_ENABLE,SFSH_CHIP_FAST_MASK);
}
}
//-------------------------------------------------------------------------------------------------
// Wait for SPI Write Cmd Ready
// @return TRUE : succeed
// @return FALSE : fail before timeout
//-------------------------------------------------------------------------------------------------
static MS_BOOL _HAL_SERFLASH_WaitWriteCmdRdy(void)
{
MS_BOOL bRet = FALSE;
struct timeval time_st;
SER_FLASH_TIME(time_st);
do
{
if ( (ISP_READ(REG_ISP_SPI_WR_CMDRDY) & ISP_SPI_WR_CMDRDY_MASK) == ISP_SPI_WR_CMDRDY )
{
bRet = TRUE;
break;
}
} while (!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
if (bRet == FALSE)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("Wait for SPI Write Cmd Ready fails!\n"));
}
return bRet;
}
//-------------------------------------------------------------------------------------------------
// Wait for SPI Write Data Ready
// @return TRUE : succeed
// @return FALSE : fail before timeout
//-------------------------------------------------------------------------------------------------
static MS_BOOL _HAL_SERFLASH_WaitWriteDataRdy(void)
{
MS_BOOL bRet = FALSE;
struct timeval time_st;
SER_FLASH_TIME(time_st);
do
{
if ( (ISP_READ(REG_ISP_SPI_WR_DATARDY) & ISP_SPI_WR_DATARDY_MASK) == ISP_SPI_WR_DATARDY )
{
bRet = TRUE;
break;
}
} while (!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
if (bRet == FALSE)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("Wait for SPI Write Data Ready fails!\n"));
}
return bRet;
}
//-------------------------------------------------------------------------------------------------
// Wait for SPI Read Data Ready
// @return TRUE : succeed
// @return FALSE : fail before timeout
//-------------------------------------------------------------------------------------------------
static MS_BOOL _HAL_SERFLASH_WaitReadDataRdy(void)
{
MS_BOOL bRet = FALSE;
struct timeval time_st;
SER_FLASH_TIME(time_st);
do
{
if ( (ISP_READ(REG_ISP_SPI_RD_DATARDY) & ISP_SPI_RD_DATARDY_MASK) == ISP_SPI_RD_DATARDY )
{
bRet = TRUE;
break;
}
} while (!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
if (bRet == FALSE)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("Wait for SPI Read Data Ready fails!\n"));
}
return bRet;
}
//-------------------------------------------------------------------------------------------------
// Wait for Write/Erase to be done
// @return TRUE : succeed
// @return FALSE : fail before timeout
//-------------------------------------------------------------------------------------------------
static MS_BOOL _HAL_SERFLASH_WaitWriteDone(void)
{
MS_BOOL bRet = FALSE;
struct timeval time_st;
SER_FLASH_TIME(time_st);
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
return FALSE;
}
do
{
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_RDSR); // RDSR
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
break;
}
if ( (ISP_READ(REG_ISP_SPI_RDATA) & SF_SR_WIP_MASK) == 0 ) // WIP = 0 write done
{
bRet = TRUE;
break;
}
} while (!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
if (bRet == FALSE)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("Wait for Write to be done fails!\n"));
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
return bRet;
}
#ifdef CONFIG_RIUISP
//-------------------------------------------------------------------------------------------------
// Check Write/Erase to be done
// @return TRUE : succeed
// @return FALSE : fail before timeout
//-------------------------------------------------------------------------------------------------
static MS_BOOL _HAL_SERFLASH_CheckWriteDone(void)
{
MS_BOOL bRet = FALSE;
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto _HAL_SERFLASH_CheckWriteDone_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_RDSR); // RDSR
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto _HAL_SERFLASH_CheckWriteDone_return;
}
if ( (ISP_READ(REG_ISP_SPI_RDATA) & SF_SR_WIP_MASK) == 0 ) // WIP = 0 write done
{
bRet = TRUE;
}
_HAL_SERFLASH_CheckWriteDone_return:
return bRet;
}
#endif
//-------------------------------------------------------------------------------------------------
/// Enable/Disable flash HW WP
/// @param bEnable \b IN: enable or disable HW protection
//-------------------------------------------------------------------------------------------------
static void _HAL_SERFLASH_ActiveFlash_Set_HW_WP(MS_BOOL bEnable)
{
extern void msFlash_ActiveFlash_Set_HW_WP(MS_BOOL bEnable) __attribute__ ((weak));
if (msFlash_ActiveFlash_Set_HW_WP != NULL)
{
msFlash_ActiveFlash_Set_HW_WP(bEnable);
}
else
{
/*if(FlashSetHWWPCB != NULL )
{
(*FlashSetHWWPCB)(bEnable);
}*/
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ALERT, printk("msFlash_ActiveFlash_Set_HW_WP() is not defined in this system\n"));
// ASSERT(msFlash_ActiveFlash_Set_HW_WP != NULL);
}
return;
}
#if defined (MCU_AEON)
//Aeon SPI Address is 64K bytes windows
static MS_BOOL _HAL_SetAeon_SPIMappingAddr(MS_U32 u32addr)
{
MS_U16 u16MHEGAddr = (MS_U16)((_hal_isp.u32XiuBaseAddr + u32addr) >> 16);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, 0x%08X)\n", __FUNCTION__, (int)u32addr, (int)u16MHEGAddr));
MHEG5_WRITE(REG_SPI_BASE, u16MHEGAddr);
return TRUE;
}
#endif
#ifdef CONFIG_RIUISP
static MS_BOOL _HAL_SERFLASH_RIURead(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = FALSE;
//MS_U8 *pu8ReadBuf = pu8Data;
MS_U32 u32I;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, %d, %p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
_HAL_ISP_Enable();
do{
if(_HAL_SERFLASH_WaitWriteDone()) break;
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
}while(1);
ISP_WRITE(REG_ISP_SPI_ADDR_L, LOU16(u32Addr));
ISP_WRITE(REG_ISP_SPI_ADDR_H, HIU16(u32Addr));
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_Read_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_READ); // READ // 0x0B fast Read : HW doesn't support now
for ( u32I = 0; u32I < u32Size; u32I++ )
{
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SERFLASH_Read_return;
}
*(pu8Data + u32I) = (MS_U8)ISP_READ(REG_ISP_SPI_RDATA);//SPI_READ_DATA();
}
//--- Flush OCP memory --------
// MsOS_FlushMemory();
bRet = TRUE;
HAL_SERFLASH_Read_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
#if defined (MCU_AEON)
//restore default value
_HAL_SetAeon_SPIMappingAddr(0);
#endif
return bRet;
}
/*
static MS_BOOL _HAL_SERFLASH_XIURead(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = FALSE;
MS_U32 u32I;
MS_U8 *pu8ReadBuf = pu8Data;
MS_U32 u32Value, u32AliSize;
MS_U32 u32AliAddr, u32RemSize = u32Size;
MS_U32 u32pos;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, %d, %p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
_HAL_ISP_Enable();
do{
if(_HAL_SERFLASH_WaitWriteDone()) break;
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
}while(1);
_HAL_ISP_Disable();
// 4-BYTE Aligment for 32 bit CPU Aligment
u32AliAddr = (u32Addr & 0xFFFFFFFC);
u32pos = u32AliAddr >> 2;
#if defined (MCU_AEON)
//write SPI mapping address
_HAL_SetAeon_SPIMappingAddr(u32AliAddr);
#endif
//---- Read first data for not aligment address ------
if(u32AliAddr < u32Addr)
{
u32Value = SFSH_XIU_READ32(u32pos);
u32pos++;
for(u32I = 0; (u32I < 4) && (u32RemSize > 0); u32I++)
{
if(u32AliAddr >= u32Addr)
{
*pu8ReadBuf++ = (MS_U8)(u32Value & 0xFF);
u32RemSize--;
}
u32Value >>= 8;
u32AliAddr++;
}
}
//----Read datum for aligment address------
u32AliSize = (u32RemSize & 0xFFFFFFFC);
for( u32I = 0; u32I < u32AliSize; u32I += 4)
{
#if defined (MCU_AEON)
if((u32AliAddr & 0xFFFF) == 0)
_HAL_SetAeon_SPIMappingAddr(u32AliAddr);
#endif
// only indirect mode
u32Value = SFSH_XIU_READ32(u32pos);
*pu8ReadBuf++ = ( u32Value >> 0) & 0xFF;
*pu8ReadBuf++ = ( u32Value >> 8) & 0xFF;
*pu8ReadBuf++ = ( u32Value >> 16)& 0xFF;
*pu8ReadBuf++ = ( u32Value >> 24)& 0xFF;
u32pos++;
u32AliAddr += 4;
}
//--- Read remain datum --------
if(u32RemSize > u32AliSize)
{
#if defined (MCU_AEON)
if((u32AliAddr & 0xFFFF) == 0)
_HAL_SetAeon_SPIMappingAddr(u32AliAddr);
#endif
u32Value = SFSH_XIU_READ32(u32pos);
}
while(u32RemSize > u32AliSize)
{
*pu8ReadBuf++ = (u32Value & 0xFF);
u32Value >>= 8;
u32AliSize++;
}
//--- Flush OCP memory --------
//MsOS_FlushMemory();
bRet = TRUE;
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
#if defined (MCU_AEON)
//restore default value
_HAL_SetAeon_SPIMappingAddr(0);
#endif
return bRet;
}
*/
#endif
//-------------------------------------------------------------------------------------------------
// Global Functions
//-------------------------------------------------------------------------------------------------
////////////////////////////////////////////////////////////////////////////////
/// @brief \b Function \b Name: HAL_SERFLASH_SetCKG()
/// @brief \b Function \b Description: This function is used to set ckg_spi dynamically
/// @param <IN> \b eCkgSpi : enumerate the ckg_spi
/// @param <OUT> \b NONE :
/// @param <RET> \b TRUE: Success FALSE: Fail
/// @param <GLOBAL> \b NONE :
/// @param <NOTE> \b : Please use this function carefully , and is restricted to Flash ability
////////////////////////////////////////////////////////////////////////////////
MS_BOOL HAL_SERFLASH_SetCKG(SPI_DrvCKG eCkgSpi)
{
MS_BOOL Ret = FALSE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
// // NON-PM Doman
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,CLK0_CLK_SWITCH_OFF,CLK0_CLK_SWITCH_MASK); // run @ 12M
//
// switch (eCkgSpi)
// {
// case E_SPI_XTALI:
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,_hal_ckg_spi_nonpm[0],CLK0_CKG_SPI_MASK); // set ckg_spi
// break;
//
// case E_SPI_54M:
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,_hal_ckg_spi_nonpm[1],CLK0_CKG_SPI_MASK); // set ckg_spi
// break;
//
// case E_SPI_86M:
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,_hal_ckg_spi_nonpm[2],CLK0_CKG_SPI_MASK); // set ckg_spi
// break;
//
// case E_SPI_108M:
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,_hal_ckg_spi_nonpm[3],CLK0_CKG_SPI_MASK); // set ckg_spi
// break;
//
// default:
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,_hal_ckg_spi_nonpm[1],CLK0_CKG_SPI_MASK); // set ckg_spi
// break;
// }
//
//
// CLK_WRITE_MASK(REG_CLK0_CKG_SPI,CLK0_CLK_SWITCH_ON,CLK0_CLK_SWITCH_MASK); // run @ ckg_spi
// // PM Doman
// PM_WRITE_MASK(REG_PM_CKG_SPI,PM_SPI_CLK_SWITCH_OFF,PM_SPI_CLK_SWITCH_MASK); // run @ 12M
// switch (eCkgSpi)
// {
// case E_SPI_XTALI:
// PM_WRITE_MASK(REG_PM_CKG_SPI,_hal_ckg_spi_pm[0],PM_SPI_CLK_SEL_MASK); // set ckg_spi
// break;
//
// case E_SPI_54M:
// PM_WRITE_MASK(REG_PM_CKG_SPI,_hal_ckg_spi_pm[1],PM_SPI_CLK_SEL_MASK); // set ckg_spi
// break;
//
// case E_SPI_86M:
// PM_WRITE_MASK(REG_PM_CKG_SPI,_hal_ckg_spi_pm[2],PM_SPI_CLK_SEL_MASK); // set ckg_spi
// break;
//
// case E_SPI_108M:
// PM_WRITE_MASK(REG_PM_CKG_SPI,_hal_ckg_spi_pm[3],PM_SPI_CLK_SEL_MASK); // set ckg_spi
// break;
//
// default:
// PM_WRITE_MASK(REG_PM_CKG_SPI,_hal_ckg_spi_pm[1],PM_SPI_CLK_SEL_MASK); // set ckg_spi
// break;
// }
// PM_WRITE_MASK(REG_PM_CKG_SPI,PM_SPI_CLK_SWITCH_ON,PM_SPI_CLK_SWITCH_MASK); // run @ ckg_spi
//
Ret = TRUE;
return Ret;
}
////////////////////////////////////////////////////////////////////////////////
/// @brief \b Function \b Name: HAL_SERFLASH_ClkDiv()
/// @brief \b Function \b Description: This function is used to set clock div dynamically
/// @param <IN> \b eCkgSpi : enumerate the clk_div
/// @param <OUT> \b NONE :
/// @param <RET> \b TRUE: Success FALSE: Fail
/// @param <GLOBAL> \b NONE :
/// @param <NOTE> \b : Please use this function carefully , and is restricted to Flash ability
////////////////////////////////////////////////////////////////////////////////
void HAL_SERFLASH_ClkDiv(SPI_DrvClkDiv eClkDivSpi)
{
switch (eClkDivSpi)
{
case E_SPI_DIV2:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV2);
break;
case E_SPI_DIV4:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV4);
break;
case E_SPI_DIV8:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV8);
break;
case E_SPI_DIV16:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV16);
break;
case E_SPI_DIV32:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV32);
break;
case E_SPI_DIV64:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV64);
break;
case E_SPI_DIV128:
ISP_WRITE(REG_ISP_SPI_CLKDIV,ISP_SPI_CLKDIV128);
break;
case E_SPI_ClkDiv_NOT_SUPPORT:
default:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
break;
}
}
////////////////////////////////////////////////////////////////////////////////
/// @brief \b Function \b Name: HAL_SERFLASH_SetMode()
/// @brief \b Function \b Description: This function is used to set RIU/XIU dynamically
/// @param <IN> \b bXiuRiu : Enable for XIU (Default) Disable for RIU(Optional)
/// @param <OUT> \b NONE :
/// @param <RET> \b TRUE: Success FALSE: Fail
/// @param <GLOBAL> \b NONE :
/// @param <NOTE> \b : XIU is faster than RIU, but is sensitive to ckg.
////////////////////////////////////////////////////////////////////////////////
MS_BOOL HAL_SERFLASH_SetMode(MS_BOOL bXiuRiu)
{
MS_BOOL Ret = FALSE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
_bXIUMode = bXiuRiu;
Ret = TRUE;
return Ret;
}
////////////////////////////////////////////////////////////////////////////////
/// @brief \b Function \b Name: HAL_SERFLASH_Set2XREAD()
/// @brief \b Function \b Description: This function is used to set 2XREAD dynamically
/// @param <IN> \b b2XMode : ENABLE for 2XREAD DISABLE for NORMAL
/// @param <OUT> \b NONE :
/// @param <RET> \b TRUE: Success FALSE: Fail
/// @param <GLOBAL> \b NONE :
/// @param <NOTE> \b : Please use this function carefully, and needs Flash support
////////////////////////////////////////////////////////////////////////////////
MS_BOOL HAL_SERFLASH_Set2XREAD(MS_BOOL b2XMode)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
if(!bDetect)
{
HAL_SERFLASH_DetectType();
}
MS_ASSERT(_hal_SERFLASH.b2XREAD); // check hw support or not
if(_hal_SERFLASH.b2XREAD)
{
_HAL_ISP_2XMode(b2XMode);
}
else
{
//UNUSED(b2XMode);
printk("%s This flash does not support 2XREAD!!!\n", __FUNCTION__);
}
return TRUE;
}
////////////////////////////////////////////////////////////////////////////////
/// @brief \b Function \b Name: HAL_SERFLASH_Set2XREAD()
/// @brief \b Function \b Description: This function is used to set 2XREAD dynamically
/// @param <IN> \b b2XMode : ENABLE for 2XREAD DISABLE for NORMAL
/// @param <OUT> \b NONE :
/// @param <RET> \b TRUE: Success FALSE: Fail
/// @param <GLOBAL> \b NONE :
/// @param <NOTE> \b : Please use this function carefully, and needs Flash support
////////////////////////////////////////////////////////////////////////////////
MS_BOOL HAL_SERFLASH_Set4XREAD(MS_BOOL b4XMode)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
MS_ASSERT(_hal_SERFLASH.b4XREAD); // check hw support or not
printk("[SerFlash]TODO: correct Quad mode GPIO\n");
//PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_HOLD_NOT_GPIO, PM_SPI_HOLD_GPIO_MASK);
if(_hal_SERFLASH.b4XREAD)
{
_HAL_ISP_4XMode(b4XMode);
}
else
{
//UNUSED(b4XMode);
printk("%s This flash does not support 4XREAD!!!\n", __FUNCTION__);
}
return TRUE;
}
////////////////////////////////////////////////////////////////////////////////
/// @brief \b Function \b Name: HAL_SERFLASH_ChipSelect()
/// @brief \b Function \b Description: set active flash among multi-spi flashes
/// @param <IN> \b u8FlashIndex : flash index (0 or 1)
/// @param <OUT> \b NONE :
/// @param <RET> \b TRUE: Success FALSE: Fail
/// @param <GLOBAL> \b NONE :
////////////////////////////////////////////////////////////////////////////////
MS_BOOL HAL_SERFLASH_ChipSelect(MS_U8 u8FlashIndex)
{
MS_BOOL Ret = FALSE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X)\n", __FUNCTION__, (int)u8FlashIndex));
switch (u8FlashIndex)
{
case FLASH_ID0:
QSPI_WRITE_MASK(REG_ISP_SPI_CHIP_SELE,SFSH_CHIP_SELE_EXT1,SFSH_CHIP_SELE_MASK);
Ret = TRUE;
break;
case FLASH_ID1:
QSPI_WRITE_MASK(REG_ISP_SPI_CHIP_SELE,SFSH_CHIP_SELE_EXT2,SFSH_CHIP_SELE_MASK);
Ret = TRUE;
break;
case FLASH_ID2:
QSPI_WRITE_MASK(REG_ISP_SPI_CHIP_SELE,SFSH_CHIP_SELE_EXT3,SFSH_CHIP_SELE_MASK);
Ret = TRUE;
break;
case FLASH_ID3:
//UNUSED(u8FlashIndex); //Reserved
default:
//UNUSED(u8FlashIndex); //Invalid flash ID
Ret = FALSE;
break;
}
WAIT_SFSH_CS_STAT(); // wait for chip select done
return Ret;
}
void HAL_SERFLASH_Config(void)
{
#if 0
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, 0x%08X, 0x%08X)\n", __FUNCTION__, (int)u32PMRegBaseAddr, (int)u32NonPMRegBaseAddr, (int)u32XiuBaseAddr));
_hal_isp.u32XiuBaseAddr = u32XiuBaseAddr;
_hal_isp.u32Mheg5BaseAddr = u32NonPMRegBaseAddr + BK_MHEG5;
_hal_isp.u32IspBaseAddr = u32PMRegBaseAddr + BK_ISP;
_hal_isp.u32FspBaseAddr = u32PMRegBaseAddr + BK_FSP;
_hal_isp.u32QspiBaseAddr = u32PMRegBaseAddr + BK_QSPI;
_hal_isp.u32PiuBaseAddr = u32PMRegBaseAddr + BK_PIU;
_hal_isp.u32PMBaseAddr = u32PMRegBaseAddr + BK_PMSLP;
_hal_isp.u32CLK0BaseAddr = u32NonPMRegBaseAddr + BK_CLK0;
_hal_isp.u32RiuBaseAddr = u32PMRegBaseAddr;
#endif
}
void HAL_SERFLASH_Init(void)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
_s32SERFLASH_Mutex = MS_SERFLASH_CREATE_MUTEX(E_MSOS_FIFO, "Mutex SERFLASH", MSOS_PROCESS_SHARED);
MS_ASSERT(_s32SERFLASH_Mutex >= 0);
#ifdef CONFIG_RIUISP
#ifdef MCU_AEON
ISP_WRITE(REG_ISP_SPI_CLKDIV, 1<<2); // cpu clock / div4
#else// MCU_MIPS @ 720 Mhz for T8
/* output clk=mcu_clk/SPI_DIV */
//set mcu_clk to 110Mhz
//WRITE_WORD_MASK(GET_REG_ADDR(BASE_REG_CHIPTOP_ADDR, 0x22),BITS(11:10, 2),BMASK(11:10)); // set ckg_spi
HAL_SERFLASH_ClkDiv(E_SPI_DIV8);
#endif
ISP_WRITE(REG_ISP_DEV_SEL, 0x0); //mark for A3
ISP_WRITE(REG_ISP_SPI_ENDIAN, ISP_SPI_ENDIAN_SEL);
#else
/* spi_clk determind FSP and BDMA output frequency */
/* it's set at IPL stage or clk platform default, so don't set at this time*/
// set spi_clk
//HAL_SERFLASH_SetCKG(E_SPI_54M);
_HAL_ISP_Disable();
QSPI_WRITE_MASK(REG_SPI_CS_TIME, SFSH_CS_DESEL_TWO, SFSH_CS_DESEL_MASK);
QSPI_WRITE_MASK(REG_SPI_CS_TIME, SFSH_CS_SETUP_TWO , SFSH_CS_SETUP_MASK);
QSPI_WRITE_MASK(REG_SPI_CS_TIME, SFSH_CS_HOLD_TWO , SFSH_CS_HOLD_MASK);
_HAL_BDMA_INIT(u32BdmaSize);
#endif
}
void HAL_SERFLASH_SetGPIO(MS_BOOL bSwitch)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_NOTICE, printk("%s() Chicago TODO\n", __FUNCTION__));
// MS_U32 u32PmGPIObase = _hal_isp.u32PMBaseAddr + 0x200;
//
// if(bSwitch)// The PAD of the SPI set as GPIO IN.
// {
// PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_IS_GPIO, PM_SPI_GPIO_MASK);
// PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_HOLD_IS_GPIO, PM_SPI_HOLD_GPIO_MASK);
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPICZ_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPICZ_OEN)<<2))|(BIT(0))));
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPICK_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPICK_OEN)<<2))|(BIT(0))));
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPIDI_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPIDI_OEN)<<2))|(BIT(0))));
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPIDO_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPIDO_OEN)<<2))|(BIT(0))));
// }
// else
// {
// PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_NOT_GPIO, PM_SPI_GPIO_MASK);
// PM_WRITE_MASK(REG_PM_SPI_IS_GPIO, PM_SPI_HOLD_NOT_GPIO, PM_SPI_HOLD_GPIO_MASK);
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPICZ_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPICZ_OEN)<<2))|(BIT(0))));
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPICK_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPICK_OEN)<<2))|(BIT(0))));
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPIDI_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPIDI_OEN)<<2))|(BIT(0))));
// WRITE_BYTE( u32PmGPIObase + ((REG_PM_GPIO_SPIDO_OEN)<<2), (READ_BYTE(u32PmGPIObase + ((REG_PM_GPIO_SPIDO_OEN)<<2))|(BIT(0))));
// }
}
extern hal_SERFLASH_t _hal_SERFLASH_table[];
extern ST_WRITE_PROTECT _pstWriteProtectTable_MX25L6445E[];
extern ST_WRITE_PROTECT _pstWriteProtectTable_MX25L12845E[];
MS_U32 PAL_SPI_GetClk(void)
{
MS_U16 reg;
MS_U32 u32Clk = 0;
reg = (PM_READ(0x20) >> 10) & 0xF;
printk("reg = %X\r\n", reg);
switch(reg)
{
case 1:
u32Clk = 27;
break;
case 4:
u32Clk = 54;
break;
case 5:
u32Clk = 72;
break;
case 6:
u32Clk = 86;
break;
case 7:
u32Clk = 108;
break;
default:
u32Clk = 0;
break;
}
return u32Clk;
}
MS_BOOL HAL_SERFLASH_DetectType(void)
{
#define READ_ID_SIZE 3
#define READ_REMS4_SIZE 2
MS_U8 u8FlashId[READ_ID_SIZE];
MS_U8 u8FlashREMS4[READ_REMS4_SIZE];
MS_U32 u32Index;
memset(&_hal_SERFLASH, 0, sizeof(_hal_SERFLASH));
if (HAL_SERFLASH_ReadID(u8FlashId, sizeof(u8FlashId))== TRUE)
{
if(u8FlashId[0]==0xFF && u8FlashId[1]==0xFF && u8FlashId[2]==0xFF)
{
bDetect = FALSE;
return bDetect;
}
/* find current serial flash */
for (u32Index = 0; _hal_SERFLASH_table[u32Index].u8MID != 0; u32Index++)
{
if ( (_hal_SERFLASH_table[u32Index].u8MID == u8FlashId[0])
&& (_hal_SERFLASH_table[u32Index].u8DID0 == u8FlashId[1])
&& (_hal_SERFLASH_table[u32Index].u8DID1 == u8FlashId[2])
)
{
memcpy(&_hal_SERFLASH, &(_hal_SERFLASH_table[u32Index]), sizeof(_hal_SERFLASH));
if(_hal_SERFLASH.u16FlashType == FLASH_IC_MX25L6405D||_hal_SERFLASH.u16FlashType == FLASH_IC_MX25L12805D)
{
if (HAL_SERFLASH_ReadREMS4(u8FlashREMS4,READ_REMS4_SIZE)== TRUE)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("[FSP] MXIC REMS: 0x%02X,0x%02X\n", u8FlashREMS4[0], u8FlashREMS4[1]) );
// patch : MXIC 6405D vs 6445E(MXIC 12805D vs 12845E)
if( u8FlashREMS4[0] == 0xC2)
{
if( u8FlashREMS4[1] == 0x16)
{
_hal_SERFLASH.u16FlashType = FLASH_IC_MX25L6445E;
_hal_SERFLASH.pWriteProtectTable = _pstWriteProtectTable_MX25L6445E;
_hal_SERFLASH.u16SPIMaxClk[1] = E_QUAD_MODE;
}
if( u8FlashREMS4[1] == 0x17)
{
_hal_SERFLASH.u16FlashType = FLASH_IC_MX25L12845E;
_hal_SERFLASH.pWriteProtectTable = _pstWriteProtectTable_MX25L12845E;
_hal_SERFLASH.u16SPIMaxClk[1] = E_QUAD_MODE;
}
}
}
}
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO,
printk("[FSP] Flash is detected (0x%04X, 0x%02X, 0x%02X, 0x%02X) ver1.1\n",
_hal_SERFLASH.u16FlashType,
_hal_SERFLASH.u8MID,
_hal_SERFLASH.u8DID0,
_hal_SERFLASH.u8DID1
)
);
bDetect = TRUE;
break;
}
else
{
continue;
}
}
// printf("[%x]\n",_hal_SERFLASH.u16FlashType);
if( _hal_SERFLASH.u8MID == MID_GD )
{
_hal_SERFLASH.u8WrsrBlkProtect = BITS(6:2, 0x00);
}
if(_hal_SERFLASH.u32FlashSize >= 0x2000000)
{
if((_hal_SERFLASH.u8MID == MID_MXIC)||(_hal_SERFLASH.u8MID == MID_GD)||(_hal_SERFLASH.u8MID == MID_WB))
{
_bHasEAR = TRUE; // support EAR mode
}
}
if(gReadMode == E_QUAD_MODE && _hal_SERFLASH.u8MID == MID_MXIC)
{
if(PAL_SPI_GetClk() > 54)
{
printk("MX supports QUAD mode only when CLK <= 54MHz\n");
BUG();
}
}
// If the Board uses a unknown flash type, force setting a secure flash type for booting. //FLASH_IC_MX25L6405D
if( bDetect != TRUE )
{
#if 1
memcpy(&_hal_SERFLASH, &(_hal_SERFLASH_table[0]), sizeof(_hal_SERFLASH));
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO,
printk("[FSP] Unknown flash type (0x%02X, 0x%02X, 0x%02X) and use default flash type 0x%04X\n",
_hal_SERFLASH.u8MID,
_hal_SERFLASH.u8DID0,
_hal_SERFLASH.u8DID1,
_hal_SERFLASH.u16FlashType
)
);
#else
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO,
printk("Unsupport flash type (0x%02X, 0x%02X, 0x%02X), please add flash info to serial flash driver\n",
u8FlashId[0],
u8FlashId[1],
u8FlashId[2]
)
);
MS_ASSERT(0);
#endif
bDetect = TRUE;
}
}
#ifdef CONFIG_RIUISP
ISP_WRITE(REG_ISP_DEV_SEL, ISP_DEV_SEL);
#else
// //check norflaash support to 4Xmode or 2xmode
// if( _hal_SERFLASH.u16SPIMaxClk[1]==E_QUAD_MODE && gQuadSupport==0 )
// gReadMode = E_DUAL_D_MODE;
// else
// gReadMode = _hal_SERFLASH.u16SPIMaxClk[1];
// don't overwrite default setting
// gReadMode = E_FAST_MODE;
HAL_SERFLASH_SetCKG(_hal_SERFLASH.u16SPIMaxClk[0]);
switch ( gReadMode )
{
case E_SINGLE_MODE:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("[FSP] 1-1-1 READ NORMAL MODE\n"));
break;
case E_FAST_MODE:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("[FSP] 1-1-1 FAST_READ MODE\n"));
break;
case E_DUAL_D_MODE:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("[FSP] 1-1-2 DUAL_FAST_READ MODE\n"));
break;
case E_DUAL_AD_MODE:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("[FSP] 1-2-2 2xIO_READ MODE\n"));
break;
case E_QUAD_MODE:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("[FSP] 1-1-4 QUAD_READ MODE\n"));
HAL_QUAD_Enable(1);
break;
default:
break;
}
#endif
return bDetect;
}
MS_BOOL HAL_SERFLASH_DetectSize(MS_U32 *u32FlashSize)
{
MS_BOOL Ret = FALSE;
do{
*u32FlashSize = _hal_SERFLASH.u32FlashSize;
Ret = TRUE;
}while(0);
return Ret;
}
MS_BOOL HAL_SERFLASH_EraseChip(void)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if(!_HAL_SERFLASH_WaitWriteDone())
{
goto HAL_SERFLASH_EraseChip_return;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_EraseChip_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_EraseChip_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_CE); // CHIP_ERASE
bRet = _HAL_SERFLASH_WaitWriteDone();
HAL_SERFLASH_EraseChip_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#else
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()\n", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
//MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_EraseChip();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_AddressToBlock(MS_U32 u32FlashAddr, MS_U32 *pu32BlockIndex)
{
MS_U32 u32NextAddr;
MS_BOOL bRet = FALSE;
if (_hal_SERFLASH.pSpecialBlocks == NULL)
{
*pu32BlockIndex = u32FlashAddr / SERFLASH_SECTOR_SIZE;
bRet = TRUE;
}
else
{
// TODO: review, optimize this flow
for (u32NextAddr = 0, *pu32BlockIndex = 0; *pu32BlockIndex < NUMBER_OF_SERFLASH_SECTORS; (*pu32BlockIndex)++)
{
// outside the special block
if ( *pu32BlockIndex < _hal_SERFLASH.pSpecialBlocks->u16Start
|| *pu32BlockIndex > _hal_SERFLASH.pSpecialBlocks->u16End
)
{
u32NextAddr += SERFLASH_SECTOR_SIZE; // i.e. normal block size
}
// inside the special block
else
{
u32NextAddr += _hal_SERFLASH.pSpecialBlocks->au32SizeList[*pu32BlockIndex - _hal_SERFLASH.pSpecialBlocks->u16Start];
}
if (u32NextAddr > u32FlashAddr)
{
bRet = TRUE;
break;
}
}
}
return bRet;
}
MS_BOOL HAL_SERFLASH_BlockToAddress(MS_U32 u32BlockIndex, MS_U32 *pu32FlashAddr)
{
if ( _hal_SERFLASH.pSpecialBlocks == NULL
|| u32BlockIndex <= _hal_SERFLASH.pSpecialBlocks->u16Start
)
{
*pu32FlashAddr = u32BlockIndex * SERFLASH_SECTOR_SIZE;
}
else
{
MS_U32 u32Index;
*pu32FlashAddr = _hal_SERFLASH.pSpecialBlocks->u16Start * SERFLASH_SECTOR_SIZE;
for (u32Index = _hal_SERFLASH.pSpecialBlocks->u16Start;
u32Index < u32BlockIndex && u32Index <= _hal_SERFLASH.pSpecialBlocks->u16End;
u32Index++
)
{
*pu32FlashAddr += _hal_SERFLASH.pSpecialBlocks->au32SizeList[u32Index - _hal_SERFLASH.pSpecialBlocks->u16Start];
}
if (u32BlockIndex > _hal_SERFLASH.pSpecialBlocks->u16End + 1)
{
*pu32FlashAddr += (u32BlockIndex - _hal_SERFLASH.pSpecialBlocks->u16End - 1) * SERFLASH_SECTOR_SIZE;
}
}
return TRUE;
}
MS_BOOL HAL_SERFLASH_BlockErase(MS_U32 u32StartBlock, MS_U32 u32EndBlock, MS_BOOL bWait)
{
#ifdef CONFIG_RIUISP
MS_BOOL bRet = FALSE;
MS_U32 u32I;
MS_U32 u32FlashAddr = 0;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, 0x%08X, %d)\n", __FUNCTION__, (int)u32StartBlock, (int)u32EndBlock, bWait));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
if( u32StartBlock > u32EndBlock || u32EndBlock >= _hal_SERFLASH.u32NumSec )
{
printk("%s (0x%08X, 0x%08X, %d)\n", __FUNCTION__, (int)u32StartBlock, (int)u32EndBlock, bWait);
goto HAL_SERFLASH_BlockErase_return;
}
_HAL_ISP_Enable();
if(!_HAL_SERFLASH_WaitWriteDone())
{
goto HAL_SERFLASH_BlockErase_return;
}
for( u32I = u32StartBlock; u32I <= u32EndBlock; u32I++)
{
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_64BE); // BLOCK_ERASE
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
if (HAL_SERFLASH_BlockToAddress(u32I, &u32FlashAddr) == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, HIU16(u32FlashAddr) & 0xFF);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32FlashAddr) >> 8);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32FlashAddr) & 0xFF);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x5555); // disable trigger mode
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
if(bWait == TRUE )
{
if(!_HAL_SERFLASH_WaitWriteDone())
{
printk("%s : Wait Write Done Fail!!!\n", __FUNCTION__ );
bRet = FALSE;
}
else
{
bRet = TRUE;
}
}
else
{
bRet = TRUE;
}
}
HAL_SERFLASH_BlockErase_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
#else
return HAL_FSP_BlockErase(u32StartBlock, u32EndBlock, bWait);
#endif
}
MS_BOOL HAL_SERFLASH_SectorErase(MS_U32 u32SectorAddress)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X)\n", __FUNCTION__, (int)u32SectorAddress));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s ENTRY fails!\n", __FUNCTION__));
return bRet;
}
if( u32SectorAddress > _hal_SERFLASH.u32FlashSize )
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s (0x%08X)\n", __FUNCTION__, (int)u32SectorAddress));
goto HAL_SERFLASH_BlockErase_return;
}
_HAL_ISP_Enable();
if(!_HAL_SERFLASH_WaitWriteDone())
{
goto HAL_SERFLASH_BlockErase_return;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_SE);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, HIU16(u32SectorAddress) & 0xFF);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32SectorAddress) >> 8);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32SectorAddress) & 0xFF);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_BlockErase_return;
}
bRet = TRUE;
HAL_SERFLASH_BlockErase_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x5555); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#else
bRet = HAL_FSP_SectorErase(u32SectorAddress,FLASH_ERASE_04K);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_CheckWriteDone(void)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
bRet = _HAL_SERFLASH_CheckWriteDone();
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s() = %d\n", __FUNCTION__, bRet));
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#else
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
// MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_CheckWriteDone();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_Write(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
MS_U16 u16I, u16Rem, u16WriteBytes;
MS_U8 *u8Buf = pu8Data;
MS_BOOL b2XREAD = FALSE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, %d, %p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
b2XREAD = (BIT(2) & ISP_READ(REG_ISP_SPI_MODE))? 1 : 0;
_HAL_ISP_2XMode(DISABLE);
_HAL_ISP_Enable();
if(!_HAL_SERFLASH_WaitWriteDone())
{
goto HAL_SERFLASH_Write_return;
}
u16Rem = u32Addr % SERFLASH_PAGE_SIZE;
if (u16Rem)
{
u16WriteBytes = SERFLASH_PAGE_SIZE - u16Rem;
if (u32Size < u16WriteBytes)
{
u16WriteBytes = u32Size;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
ISP_WRITE(REG_ISP_SPI_ADDR_L, LOU16(u32Addr));
ISP_WRITE(REG_ISP_SPI_ADDR_H, (MS_U8)HIU16(u32Addr));
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_PP); // PAGE_PROG
for ( u16I = 0; u16I < u16WriteBytes; u16I++ )
{
ISP_WRITE(REG_ISP_SPI_WDATA, (MS_U8)*(u8Buf + u16I));//SPI_WRITE_DATA( *(u8Buf + u16I) );
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
bRet = _HAL_SERFLASH_WaitWriteDone();
if ( bRet == TRUE )
{
u32Addr += u16WriteBytes;
u8Buf += u16WriteBytes;
u32Size -= u16WriteBytes;
}
else
{
goto HAL_SERFLASH_Write_return;
}
}
while(u32Size)
{
if( u32Size > SERFLASH_PAGE_SIZE)
{
u16WriteBytes = SERFLASH_PAGE_SIZE; //write SERFLASH_PAGE_SIZE bytes one time
}
else
{
u16WriteBytes = u32Size;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
ISP_WRITE(REG_ISP_SPI_ADDR_L, LOU16(u32Addr));
ISP_WRITE(REG_ISP_SPI_ADDR_H, (MS_U8)HIU16(u32Addr));
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_PP); // PAGE_PROG
// Improve flash write speed
if(u16WriteBytes == 256)
{
// Write 256 bytes to flash
MS_U8 u8Index = 0;
do{
ISP_WRITE(REG_ISP_SPI_WDATA, (MS_U8)*(u8Buf + u8Index));//SPI_WRITE_DATA( *(u8Buf + u8Index) );
u8Index++;
if( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
}while(u8Index != 0);
}
else
{
for ( u16I = 0; u16I < u16WriteBytes; u16I++ )
{
ISP_WRITE(REG_ISP_SPI_WDATA, (MS_U8)*(u8Buf + u16I));//SPI_WRITE_DATA( *(u8Buf + u16I) );
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_Write_return;
}
}
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
bRet = _HAL_SERFLASH_WaitWriteDone();
if ( bRet == TRUE )
{
u32Addr += u16WriteBytes;
u8Buf += u16WriteBytes;
u32Size -= u16WriteBytes;
}
else
{
goto HAL_SERFLASH_Write_return;
}
}
HAL_SERFLASH_Write_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
// restore the 2x READ setting.
HAL_SERFLASH_SelectReadMode(_hal_SERFLASH.u16SPIMaxClk[1]);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#else
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, %d, %p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
#if defined(CONFIG_FSP_WRITE_RIUOP)
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
HAL_FSP_Entry();
bRet = HAL_FSP_Write(u32Addr, u32Size, pu8Data);
HAL_FSP_Exit();
#elif defined(CONFIG_FSP_WRITE_BDMA)
HAL_FSP_Entry();
HAL_FSP_WriteExtAddrReg((u32Addr >> 24) & 0xFF);
bRet = HAL_FSP_Write_BDMA(u32Addr, u32Size, pu8Data);
HAL_FSP_Exit();
#else
#error "FSP write"
#endif
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_ReadRedirect(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
if(!BASE_FLASH_OFFSET)
{
BASE_FLASH_OFFSET = (MS_U32)ioremap(MS_SPI_ADDR, 0x2000000);
//printk("[SER flash]MS_SPI_ADDR=(0x%08X)\n",(int)BASE_FLASH_OFFSET);
}
if(BASE_FLASH_OFFSET)
{
MS_U32 u32ReadSize = 0;
if (u32Addr < 0x1000000)
{
if (_u8RegEAR) // switch back to bottom 16MB
HAL_FSP_WriteExtAddrReg(0);
u32ReadSize = u32Size > (0x1000000 - u32Addr) ? (0x1000000 - u32Addr) : u32Size;
memcpy((void *)pu8Data, (const void *)(BASE_FLASH_OFFSET+u32Addr), u32ReadSize);
u32Size = u32Size - u32ReadSize;
}
if (u32Size)
{
MS_U8 u8EAR = 0;
u32Addr += u32ReadSize;
u8EAR = u32Addr >> 24;
if (u8EAR)
HAL_FSP_WriteExtAddrReg(u8EAR);
memcpy((void *)pu8Data+u32ReadSize,(const void *)(BASE_FLASH_OFFSET+(u32Addr&0xFFFFFF)), u32Size);
}
}
return TRUE;
}
MS_BOOL HAL_SERFLASH_ReadBdma(MS_U32 u32Offset, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL Ret = FALSE;
MS_U32 u32AlignedOffset = (u32Offset) & (~0xF);
MS_U32 u32AlignedSize;
MS_U32 u32Delta;
u32Delta = u32Offset - u32AlignedOffset;
u32AlignedSize = ((u32Size + u32Delta) + 0xF) & (~0xF);
{
HalBdmaParam_t tBdmaParam;
const u8 u8DmaCh = HAL_BDMA_CH0;
_HAL_BDMA_INIT(u32AlignedSize);
tBdmaParam.ePathSel = HAL_BDMA_SPI_TO_MIU0;//(pstDmaCfg->phyaddr < ARM_MIU1_BASE_ADDR) ? (HAL_BDMA_MEM_TO_MIU0) : (HAL_BDMA_MEM_TO_MIU1);
tBdmaParam.bIntMode = 0; //0: polling mode
tBdmaParam.eDstAddrMode = HAL_BDMA_ADDR_INC;
tBdmaParam.u32TxCount = u32AlignedSize;//(u32Size + 0xF) & (~0xF);
tBdmaParam.pSrcAddr = (void*)u32AlignedOffset;//u32Offset;
tBdmaParam.pDstAddr = (void*)pu8BDMA_bus;
tBdmaParam.pfTxCbFunc = NULL;//msys_bdma_done;
tBdmaParam.u32Pattern = 0;
HalBdma_Initialize(u8DmaCh);
//DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("%s(0x%08X, %d, %p)b\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
if (HAL_BDMA_PROC_DONE != HalBdma_Transfer(u8DmaCh, &tBdmaParam)) {
//DEBUG_SER_FLASH(E_SERFLASH_DBGLV_INFO, printk("err\n"));
return FALSE;
}
Chip_Flush_MIU_Pipe();
Chip_Inv_Cache_Range(pu8BDMA_virt, u32AlignedSize);//u32Size);
//memcpy(pu8Data, (void*)pu8BDMA_virt, u32Size);
memcpy(pu8Data, (void*)pu8BDMA_virt + u32Delta, u32Size);
Ret = TRUE;
}
return TRUE;
}
MS_BOOL HAL_SERFLASH_Read(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL Ret = FALSE;
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
//MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
#ifdef CONFIG_RIUISP
/*if( _bXIUMode )
{
Ret = _HAL_SERFLASH_XIURead( u32Addr, u32Size, pu8Data);
}
else// RIU mode*/
{
Ret = _HAL_SERFLASH_RIURead( u32Addr, u32Size, pu8Data);
}
#else
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, %d, %p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
if(_hal_SERFLASH.u8MID == MID_25Q)
{
HAL_SERFLASH_SelectReadMode(_hal_SERFLASH.u16SPIMaxClk[1]);
}
else
{
HAL_SERFLASH_SelectReadMode(gReadMode);
}
#ifdef CONFIG_FSP_READ_DIRERECT
if(!BASE_FLASH_OFFSET)
{
BASE_FLASH_OFFSET = (MS_U32)ioremap(MS_SPI_ADDR, 0x2000000);
//printk("[SER flash]MS_SPI_ADDR=(0x%08X)\n",(int)BASE_FLASH_OFFSET);
}
if(BASE_FLASH_OFFSET)
{
MS_U32 u32ReadSize = 0;
if (u32Addr < 0x1000000)
{
if (_u8RegEAR) // switch back to bottom 16MB
HAL_FSP_WriteExtAddrReg(0);
u32ReadSize = u32Size > (0x1000000 - u32Addr) ? (0x1000000 - u32Addr) : u32Size;
memcpy((void *)pu8Data, (const void *)(BASE_FLASH_OFFSET+u32Addr), u32ReadSize);
u32Size = u32Size - u32ReadSize;
}
if (u32Size)
{
MS_U8 u8EAR = 0;
u32Addr += u32ReadSize;
u8EAR = u32Addr >> 24;
if (u8EAR)
HAL_FSP_WriteExtAddrReg(u8EAR);
memcpy((void *)pu8Data+u32ReadSize,(const void *)(BASE_FLASH_OFFSET+(u32Addr&0xFFFFFF)), u32Size);
}
}
Ret = TRUE;
#elif defined(CONFIG_FSP_READ_RIUOP)
Ret = HAL_FSP_Read(u32Addr, u32Size, pu8Data);
#elif defined(CONFIG_FSP_READ_BDMA)
HAL_FSP_WriteExtAddrReg((u32Addr >> 24) & 0xFF);
Ret = HAL_SERFLASH_ReadBdma(u32Addr, u32Size, pu8Data);
#else
#error "FPS READ"
#endif
if(gReadMode > E_FAST_MODE && gReadMode != E_RIUISP_MODE)
{
HAL_SERFLASH_SelectReadMode(E_FAST_MODE);
}
#endif
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return Ret;
}
EN_WP_AREA_EXISTED_RTN HAL_SERFLASH_WP_Area_Existed(MS_U32 u32UpperBound, MS_U32 u32LowerBound, MS_U8 *pu8BlockProtectBits)
{
ST_WRITE_PROTECT *pWriteProtectTable;
MS_U8 u8Index;
MS_BOOL bPartialBoundFitted;
MS_BOOL bEndOfTable;
MS_U32 u32PartialFittedLowerBound = u32UpperBound;
MS_U32 u32PartialFittedUpperBound = u32LowerBound;
if (NULL == _hal_SERFLASH.pWriteProtectTable)
{
return WP_TABLE_NOT_SUPPORT;
}
for (u8Index = 0, bEndOfTable = FALSE, bPartialBoundFitted = FALSE; FALSE == bEndOfTable; u8Index++)
{
pWriteProtectTable = &(_hal_SERFLASH.pWriteProtectTable[u8Index]);
if ( 0xFFFFFFFF == pWriteProtectTable->u32LowerBound
&& 0xFFFFFFFF == pWriteProtectTable->u32UpperBound
)
{
bEndOfTable = TRUE;
}
if ( pWriteProtectTable->u32LowerBound == u32LowerBound
&& pWriteProtectTable->u32UpperBound == u32UpperBound
)
{
*pu8BlockProtectBits &= ~pWriteProtectTable->u8BlockProtectMask;
*pu8BlockProtectBits |= pWriteProtectTable->u8BlockProtectBits;
return WP_AREA_EXACTLY_AVAILABLE;
}
else if (u32LowerBound <= pWriteProtectTable->u32LowerBound && pWriteProtectTable->u32UpperBound <= u32UpperBound)
{
//
// u32PartialFittedUpperBound & u32PartialFittedLowerBound would be initialized first time when bPartialBoundFitted == FALSE (init value)
// 1. first match: FALSE == bPartialBoundFitted
// 2. better match: (pWriteProtectTable->u32UpperBound - pWriteProtectTable->u32LowerBound) > (u32PartialFittedUpperBound - u32PartialFittedLowerBound)
//
if ( FALSE == bPartialBoundFitted
|| (pWriteProtectTable->u32UpperBound - pWriteProtectTable->u32LowerBound) > (u32PartialFittedUpperBound - u32PartialFittedLowerBound)
)
{
u32PartialFittedUpperBound = pWriteProtectTable->u32UpperBound;
u32PartialFittedLowerBound = pWriteProtectTable->u32LowerBound;
*pu8BlockProtectBits &= ~pWriteProtectTable->u8BlockProtectMask;
*pu8BlockProtectBits |= pWriteProtectTable->u8BlockProtectBits;
}
bPartialBoundFitted = TRUE;
}
}
if (TRUE == bPartialBoundFitted)
{
return WP_AREA_PARTIALLY_AVAILABLE;
}
else
{
return WP_AREA_NOT_AVAILABLE;
}
}
EN_WP_AREA_EXISTED_RTN HAL_SERFLASH_WP_Status_Existed(MS_U32* u32UpperBound, MS_U32* u32LowerBound, MS_U8 u8BlockProtectBits)
{
ST_WRITE_PROTECT *pWriteProtectTable;
MS_U8 u8Index;
MS_BOOL bEndOfTable;
if (NULL == _hal_SERFLASH.pWriteProtectTable)
{
return WP_TABLE_NOT_SUPPORT;
}
for (u8Index = 0, bEndOfTable = FALSE; FALSE == bEndOfTable; u8Index++)
{
pWriteProtectTable = &(_hal_SERFLASH.pWriteProtectTable[u8Index]);
if ( 0xFFFFFFFF == pWriteProtectTable->u32LowerBound
&& 0xFFFFFFFF == pWriteProtectTable->u32UpperBound
)
{
bEndOfTable = TRUE;
}
if((u8BlockProtectBits & pWriteProtectTable->u8BlockProtectMask) == pWriteProtectTable->u8BlockProtectBits)
{
*u32UpperBound = pWriteProtectTable->u32UpperBound;
*u32LowerBound = pWriteProtectTable->u32LowerBound;
return WP_AREA_EXACTLY_AVAILABLE;
}
}
return WP_AREA_NOT_AVAILABLE;
}
MS_BOOL HAL_SERFLASH_WriteProtect_Area(MS_BOOL bEnableAllArea, MS_U8 u8BlockProtectBits)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
MS_U8 u8Status0, u8Status1;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d, 0x%02X)\n", __FUNCTION__, bEnableAllArea, u8BlockProtectBits));
MS_ASSERT( MS_SERFLASH_IN_INTERRUPT() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(DISABLE);
udelay(bEnableAllArea ? 5 : 20); // when disable WP, delay more time
_HAL_ISP_Enable();
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WRSR); // WRSR
if (TRUE == bEnableAllArea)
{
if (_hal_SERFLASH.u8MID == MID_ATMEL)
{
ISP_WRITE(REG_ISP_SPI_WDATA, SERFLASH_WRSR_BLK_PROTECT); // SPRL 1 -> 0
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WRSR); // WRSR
}
ISP_WRITE(REG_ISP_SPI_WDATA, SF_SR_SRWD | SERFLASH_WRSR_BLK_PROTECT); // SF_SR_SRWD: SRWD Status Register Write Protect
}
else
{
if (_hal_SERFLASH.u8MID == MID_ATMEL)
{
ISP_WRITE(REG_ISP_SPI_WDATA, u8BlockProtectBits); // [4:2] or [5:2] protect blocks // SPRL 1 -> 0
// programming sector protection
{
int i;
MS_U32 u32FlashAddr;
// search write protect table
for (i = 0;
0xFFFFFFFF != _hal_SERFLASH.pWriteProtectTable[i].u32LowerBound && 0xFFFFFFFF != _hal_SERFLASH.pWriteProtectTable[i].u32UpperBound; // the end of write protect table
i++
)
{
// if found, write
if (u8BlockProtectBits == _hal_SERFLASH.pWriteProtectTable[i].u8BlockProtectBits)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("u8BlockProtectBits = 0x%X, u32LowerBound = 0x%X, u32UpperBound = 0x%X\n",
(unsigned int)u8BlockProtectBits,
(unsigned int)_hal_SERFLASH.pWriteProtectTable[i].u32LowerBound,
(unsigned int)_hal_SERFLASH.pWriteProtectTable[i].u32UpperBound
)
);
for (u32FlashAddr = 0; u32FlashAddr < _hal_SERFLASH.u32FlashSize; u32FlashAddr += _hal_SERFLASH.u32SecSize)
{
if (_hal_SERFLASH.pWriteProtectTable[i].u32LowerBound <= (u32FlashAddr + _hal_SERFLASH.u32SecSize - 1) &&
u32FlashAddr <= _hal_SERFLASH.pWriteProtectTable[i].u32UpperBound)
{
continue;
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, 0x39); // unprotect sector
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, (u32FlashAddr >> 16) & 0xFF);
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, ((MS_U16)u32FlashAddr) >> 8);
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, u32FlashAddr & 0xFF);
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
bRet = _HAL_SERFLASH_WaitWriteDone();
}
break;
}
}
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_Flash_WriteProtect_Area_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WRSR); // WRSR
}
ISP_WRITE(REG_ISP_SPI_WDATA, SF_SR_SRWD | u8BlockProtectBits); // [4:2] or [5:2] protect blocks
}
bRet = _HAL_SERFLASH_WaitWriteDone();
HAL_Flash_WriteProtect_Area_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
if ((_hal_SERFLASH.u8MID == MID_GD) || (_hal_SERFLASH.u16FlashType == FLASH_IC_S25FL032K)
|| (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q64CV) || (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q32BV))//MSB S8-S15
{
HAL_SERFLASH_ReadStatusReg(&u8Status0);
HAL_SERFLASH_ReadStatusReg2(&u8Status1);
HAL_SERFLASH_WriteStatusReg(((u8Status1 << 8)|u8Status0|0x4000));//CMP = 1
udelay(40);
}
if (bEnableAllArea)// _REVIEW_
{
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(bEnableAllArea);
}
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#else
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d, 0x%02X)\n", __FUNCTION__, bEnableAllArea, u8BlockProtectBits));
MS_ASSERT( MS_SERFLASH_IN_INTERRUPT() == FALSE );
//MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_WriteProtect_Area(bEnableAllArea, u8BlockProtectBits);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
//-------------------------------------------------------------------------------------------------
/*spasion flash DYB unlock*/
MS_BOOL HAL_SERFLASH_SPSNDYB_UNLOCK(MS_U32 u32StartBlock, MS_U32 u32EndBlock)
{
//u32StartBlock, u32EndBlock --> 0~285
MS_BOOL bRet = FALSE;
MS_U32 u32I;
MS_U32 u32FlashAddr = 0;
MS_U8 dybStatusReg=0;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG,printk("%s ENTRY \n", __FUNCTION__));
MS_ASSERT( MS_SERFLASH_IN_INTERRUPT() == FALSE );
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(DISABLE);
//MsOS_DelayTask(20); // when disable WP, delay more time
udelay(20);
_HAL_ISP_Enable();
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
for( u32I = u32StartBlock; u32I <= u32EndBlock; u32I++)
{
bRet = FALSE;
dybStatusReg=0;
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, 0xE1); // DYB write
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
if(u32I<=31)
u32FlashAddr = u32I * 4096;
else
u32FlashAddr = (u32I -30) * 64 * 1024;
ISP_WRITE(REG_ISP_SPI_WDATA, HIU16(u32FlashAddr)>>8); //MSB , 4th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, HIU16(u32FlashAddr) & 0xFF);//3th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32FlashAddr) >> 8);//2th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32FlashAddr) & 0xFF);//1th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, 0xFF);//0xFF --> unlock; 0x00-->lock
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x5555); // disable trigger mode
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333);
//read DYB status register to check the result
ISP_WRITE(REG_ISP_SPI_WDATA, 0xE0); // DYB read
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, HIU16(u32FlashAddr)>>8); //MSB , 4th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, HIU16(u32FlashAddr) & 0xFF);//3th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32FlashAddr) >> 8);//2th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, LOU16(u32FlashAddr) & 0xFF);//1th byte
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
dybStatusReg = (MS_U8)ISP_READ(REG_ISP_SPI_RDATA);//SPI_READ_DATA();
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x5555); // disable trigger mode
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
if(dybStatusReg != 0xFF )
{
bRet = FALSE;
goto HAL_SPSNFLASH_DYB_UNLOCK_return;
}
else
bRet = TRUE;
}
HAL_SPSNFLASH_DYB_UNLOCK_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
}
MS_BOOL HAL_SERFLASH_WriteProtect(MS_BOOL bEnable)
{
// Note: Temporarily don't call this function until MSTV_Tool ready
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
MS_U8 u8Status0, u8Status1;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d)\n", __FUNCTION__, bEnable));
MS_ASSERT( MS_SERFLASH_IN_INTERRUPT() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(DISABLE);
//udelay(bEnable ? 5 : 20); //mark for A3
_HAL_ISP_Enable();
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WRSR); // WRSR
if (bEnable)
{
if (_hal_SERFLASH.u8MID == MID_ATMEL)
{
ISP_WRITE(REG_ISP_SPI_WDATA, SERFLASH_WRSR_BLK_PROTECT); // SPRL 1 -> 0
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WRSR); // WRSR
}
ISP_WRITE(REG_ISP_SPI_WDATA, SF_SR_SRWD | SERFLASH_WRSR_BLK_PROTECT); // SF_SR_SRWD: SRWD Status Register Write Protect
if ((_hal_SERFLASH.u8MID == MID_GD) || (_hal_SERFLASH.u16FlashType == FLASH_IC_S25FL032K)
|| (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q64CV) || (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q32BV))//MSB S8-S15
{
if ( _HAL_SERFLASH_WaitWriteDone() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, 0x40);
}
}
else
{
if (_hal_SERFLASH.u8MID == MID_ATMEL)
{
ISP_WRITE(REG_ISP_SPI_WDATA, 0 << 2); // [4:2] or [5:2] protect blocks // SPRL 1 -> 0
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WRSR); // WRSR
}
if ((_hal_SERFLASH.u8MID == MID_GD) || (_hal_SERFLASH.u16FlashType == FLASH_IC_S25FL032K)
|| (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q64CV) || (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q32BV))//MSB S8-S15
{
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(6:2, 0x1F)); // [6:2] protect blocks
if ( _HAL_SERFLASH_WaitWriteDone() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, 0x40);
}
else
{
ISP_WRITE(REG_ISP_SPI_WDATA, SF_SR_SRWD); // [4:2] or [5:2] protect blocks
}
}
bRet = _HAL_SERFLASH_WaitWriteDone();
HAL_SERFLASH_WriteProtect_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
if ((_hal_SERFLASH.u8MID == MID_GD) || (_hal_SERFLASH.u16FlashType == FLASH_IC_S25FL032K)
|| (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q64CV) || (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q32BV))//MSB S8-S15
{
HAL_SERFLASH_ReadStatusReg(&u8Status0);
HAL_SERFLASH_ReadStatusReg2(&u8Status1);
HAL_SERFLASH_WriteStatusReg(((u8Status1 << 8)|u8Status0|0x4000));//CMP = 1
udelay(40);
}
if (bEnable) // _REVIEW_
{
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(bEnable);
}
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
#else
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d)\n", __FUNCTION__, bEnable));
MS_ASSERT( MS_SERFLASH_IN_INTERRUPT() == FALSE );
// MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_WriteProtect(bEnable);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_ReadID(MS_U8 *pu8Data, MS_U32 u32Size)
{
// HW doesn't support ReadID on MX/ST flash; use trigger mode instead.
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
MS_U32 u32I;
MS_U8 *u8ptr = pu8Data;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if(!_HAL_SERFLASH_WaitWriteDone())
{
goto HAL_SERFLASH_ReadID_return;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_ReadID_return;
}
// SFSH_RIU_REG16(REG_SFSH_SPI_COMMAND) = ISP_SPI_CMD_RDID; // RDID
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_RDID); // RDID
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadID_return;
}
for ( u32I = 0; u32I < u32Size; u32I++ )
{
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadID_return;
}
u8ptr[u32I] = ISP_READ(REG_ISP_SPI_RDATA);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk(" 0x%02X", u8ptr[u32I]));
}
bRet = TRUE;
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
HAL_SERFLASH_ReadID_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("\n"));
#else
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_ReadID(pu8Data, u32Size);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_U64 HAL_SERFLASH_ReadUID(void)
{
#define READ_UID_SIZE 8
MS_U8 u8I = 0;
MS_U8 u8ptr[READ_UID_SIZE];
MS_U8 u8Size = READ_UID_SIZE;
MS_U64 u64FlashUId = 0;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if(!_HAL_SERFLASH_WaitWriteDone())
{
goto HAL_SERFLASH_READUID_RETURN;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
// SFSH_RIU_REG16(REG_SFSH_SPI_COMMAND) = ISP_SPI_CMD_RDID; // RDID
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // enable trigger mode
if(_hal_SERFLASH.u16FlashType == FLASH_IC_EN25QH16)
{
ISP_WRITE(REG_ISP_SPI_WDATA, 0x5A); // RDUID
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
for(u8I = 0; u8I < 2; u8I++)
{
ISP_WRITE(REG_ISP_SPI_WDATA, 0x00);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
}
ISP_WRITE(REG_ISP_SPI_WDATA, 0x80); //start address
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
ISP_WRITE(REG_ISP_SPI_WDATA, 0xFF);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
}
else if( _hal_SERFLASH.u16FlashType == FLASH_IC_W25Q16)
{
ISP_WRITE(REG_ISP_SPI_WDATA, 0x4B); // RDUID
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
for(u8I = 0;u8I < 4;u8I++)
{
ISP_WRITE(REG_ISP_SPI_WDATA, 0xFF); // RDUID
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
}
}
else
{
goto HAL_SERFLASH_READUID_RETURN;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_READ); // READ
for ( u8I = 0; u8I < u8Size; u8I++ )
{
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SERFLASH_READUID_RETURN;
}
u8ptr[u8I] = ISP_READ(REG_ISP_SPI_RDATA);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s, %d 0x%02X\n",__FUNCTION__, u8I, u8ptr[u8I]));
}
for(u8I = 0;u8I < 8;u8I++)
{
u64FlashUId <<= 8;
u64FlashUId += u8ptr[u8I];
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
HAL_SERFLASH_READUID_RETURN:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("\n"));
return u64FlashUId;
}
MS_BOOL HAL_SERFLASH_ReadREMS4(MS_U8 * pu8Data, MS_U32 u32Size)
{
#ifdef CONFIG_RIUISP
MS_BOOL bRet = FALSE;
MS_U32 u32Index;
MS_U8 *u8ptr = pu8Data;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if ( !_HAL_SERFLASH_WaitWriteDone() )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
if ( !_HAL_SERFLASH_WaitWriteCmdRdy() )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_REMS4); // READ_REMS4 for new MXIC Flash
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_WDATA_DUMMY);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_WDATA_DUMMY);
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, 0x00); // if ADD is 0x00, MID first. if ADD is 0x01, DID first
if ( _HAL_SERFLASH_WaitWriteDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
for ( u32Index = 0; u32Index < u32Size; u32Index++ )
{
ISP_WRITE(REG_ISP_SPI_RDREQ, ISP_SPI_RDREQ); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadREMS4_return;
}
u8ptr[u32Index] = ISP_READ(REG_ISP_SPI_RDATA);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk(" 0x%02X", u8ptr[u32Index]));
}
bRet = TRUE;
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
HAL_SERFLASH_ReadREMS4_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
#else
return HAL_FSP_ReadREMS4(pu8Data, u32Size);
#endif
}
MS_BOOL HAL_SERFLASH_DMA(MS_U32 u32FlashStart, MS_U32 u32DRASstart, MS_U32 u32Size)
{
MS_BOOL bRet = FALSE;
#if 0
MS_U32 u32Timer;
MS_U32 u32Timeout = SERFLASH_SAFETY_FACTOR*u32Size/(108*1000/4/8);
u32Timeout=u32Timeout; //to make compiler happy
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, 0x%08X, %d)\n", __FUNCTION__, (int)u32FlashStart, (int)u32DRASstart, (int)u32Size));
// [URANUS_REV_A][OBSOLETE] // TODO: <-@@@ CHIP SPECIFIC
#if 0 // TODO: review
if (MDrv_SYS_GetChipRev() == 0x00)
{
// DMA program can't run on DRAM, but in flash ONLY
return FALSE;
}
#endif // TODO: review
// [URANUS_REV_A][OBSOLETE] // TODO: <-@@@ CHIP SPECIFIC
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
ISP_WRITE_MASK(REG_ISP_CHIP_SEL, SFSH_CHIP_SEL_RIU, SFSH_CHIP_SEL_MODE_SEL_MASK); // For DMA only
_HAL_ISP_Disable();
// SFSH_RIU_REG16(REG_SFSH_SPI_CLK_DIV) = 0x02; // 108MHz div3 (max. 50MHz for this ST flash) for FAST_READ
PIU_WRITE(REG_PIU_DMA_SIZE_L, LOU16(u32Size));
PIU_WRITE(REG_PIU_DMA_SIZE_H, HIU16(u32Size));
PIU_WRITE(REG_PIU_DMA_DRAMSTART_L, LOU16(u32DRASstart));
PIU_WRITE(REG_PIU_DMA_DRAMSTART_H, HIU16(u32DRASstart));
PIU_WRITE(REG_PIU_DMA_SPISTART_L, LOU16(u32FlashStart));
PIU_WRITE(REG_PIU_DMA_SPISTART_H, HIU16(u32FlashStart));
// SFSH_PIU_REG16(REG_SFSH_DMA_CMD) = 0 << 5; // 0: little-endian 1: big-endian
// SFSH_PIU_REG16(REG_SFSH_DMA_CMD) |= 1; // trigger
PIU_WRITE(REG_PIU_DMA_CMD, PIU_DMA_CMD_LE | PIU_DMA_CMD_FIRE); // trigger
// Wait for DMA to be done
SER_FLASH_TIME(u32Timer);
do
{
if ( (PIU_READ(REG_PIU_DMA_STATUS) & PIU_DMA_DONE_MASK) == PIU_DMA_DONE ) // finished
{
bRet = TRUE;
break;
}
} while (!SER_FLASH_EXPIRE(u32Timer,u32Timeout));
if (bRet == FALSE)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("DMA timeout!\n"));
}
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_ReadStatusReg(MS_U8 *pu8StatusReg)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
*pu8StatusReg = 0xFF;
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_RDSR); // RDSR
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
ISP_WRITE(REG_ISP_SPI_RDREQ, 0x01); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
*pu8StatusReg = ISP_READ(REG_ISP_SPI_RDATA);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk(" 0x%02X", *pu8StatusReg));
bRet = TRUE;
HAL_SERFLASH_ReadStatusReg_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("\n"));
#else
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
// MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_ReadStatusReg(pu8StatusReg);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_ReadStatusReg2(MS_U8 *pu8StatusReg)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
*pu8StatusReg = 0x00;
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if ( !_HAL_SERFLASH_WaitWriteDone() )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_RDSR2); // RDSR2
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
ISP_WRITE(REG_ISP_SPI_RDREQ, 0x01); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SERFLASH_ReadStatusReg_return;
}
*pu8StatusReg = ISP_READ(REG_ISP_SPI_RDATA);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk(" 0x%02X", *pu8StatusReg));
bRet = TRUE;
HAL_SERFLASH_ReadStatusReg_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("\n"));
#else
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
// MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_ReadStatusReg2(pu8StatusReg);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SERFLASH_WriteStatusReg(MS_U16 u16StatusReg)
{
MS_BOOL bRet = FALSE;
#ifdef CONFIG_RIUISP
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if ( !_HAL_SERFLASH_WaitWriteDone() )
{
goto HAL_SERFLASH_WriteStatusReg_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_WREN); // WREN
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_WriteStatusReg_return;
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_WRSR); // WRSR
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_WriteStatusReg_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, (MS_U8)(u16StatusReg & 0xFF )); // LSB
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_WriteStatusReg_return;
}
if((_hal_SERFLASH.u8MID == MID_GD) || (_hal_SERFLASH.u16FlashType == FLASH_IC_S25FL032K)
||(_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q64CV) || (_hal_SERFLASH.u16FlashType == FLASH_IC_W25Q32BV))
{
ISP_WRITE(REG_ISP_SPI_WDATA, (MS_U8)(u16StatusReg >> 8 )); // MSB
//printf("write_StatusReg=[%x]\n",u16StatusReg);
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_WriteStatusReg_return;
}
}
bRet = TRUE;
HAL_SERFLASH_WriteStatusReg_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("\n"));
#else
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
bRet = HAL_FSP_WriteStatusReg(u16StatusReg);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
#endif
return bRet;
}
MS_BOOL HAL_SPI_EnterIBPM(void)
{
MS_BOOL bRet = FALSE;
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
_HAL_ISP_Enable();
if ( !_HAL_SERFLASH_WaitWriteDone() )
{
goto HAL_SPI_EnableIBPM_return;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SPI_EnableIBPM_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_WPSEL); // WPSEL
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SPI_EnableIBPM_return;
}
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_RDSCUR); // Read Security Register
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SPI_EnableIBPM_return;
}
ISP_WRITE(REG_ISP_SPI_RDREQ, 0x01); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SPI_EnableIBPM_return;
}
if((ISP_READ(REG_ISP_SPI_RDATA) & BIT(7)) == BIT(7))
{
bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG,
printk("MXIC Security Register 0x%02X\n", ISP_READ(REG_ISP_SPI_RDATA)));
}
HAL_SPI_EnableIBPM_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
}
MS_BOOL HAL_SPI_SingleBlockLock(MS_PHYADDR u32FlashAddr, MS_BOOL bLock)
{
MS_BOOL bRet = FALSE;
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return bRet;
}
if ( _bIBPM != TRUE )
{
printk("%s not in Individual Block Protect Mode\n", __FUNCTION__);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
}
_HAL_ISP_Enable();
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
if( bLock )
{
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_SBLK); // Single Block Lock Protection
}
else
{
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_SBULK); // Single Block unLock Protection
}
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x10)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x08)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x00)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_SingleBlockLock_return;
}
#if defined (MS_DEBUG)
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_RDBLOCK); // Read Block Lock Status
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x10)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x08)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x00)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_SingleBlockLock_return;
}
ISP_WRITE(REG_ISP_SPI_RDREQ, 0x01); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SPI_SingleBlockLock_return;
}
if( bLock )
{
if( ISP_READ(REG_ISP_SPI_RDATA) == 0xFF )
bRet = TRUE;
}
else
{
if( ISP_READ(REG_ISP_SPI_RDATA) == 0x00 )
bRet = TRUE;
}
#else//No Ceck
bRet = TRUE;
#endif
HAL_SPI_SingleBlockLock_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
}
MS_BOOL HAL_SPI_GangBlockLock(MS_BOOL bLock)
{
MS_BOOL bRet = FALSE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d)\n", __FUNCTION__, bLock));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return bRet;
}
if ( _bIBPM != TRUE )
{
printk("%s not in Individual Block Protect Mode\n", __FUNCTION__);
return bRet;
}
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(bLock);
udelay(bLock ? 5 : 20); // when disable WP, delay more time
_HAL_ISP_Enable();
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_SPI_COMMAND, ISP_SPI_CMD_WREN); // WREN
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SERFLASH_WriteProtect_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
if( bLock )
{
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_GBLK); // Gang Block Lock Protection
}
else
{
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_GBULK); // Gang Block unLock Protection
}
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SERFLASH_WriteProtect_return;
}
bRet = TRUE;
HAL_SERFLASH_WriteProtect_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
if (bLock) // _REVIEW_
{
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(bLock);
}
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
}
MS_U8 HAL_SPI_ReadBlockStatus(MS_PHYADDR u32FlashAddr)
{
MS_U8 u8Val = 0xA5;
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return u8Val;
}
if ( _bIBPM != TRUE )
{
printk("%s not in Individual Block Protect Mode\n", __FUNCTION__);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return u8Val;
}
_HAL_ISP_Enable();
if ( !_HAL_SERFLASH_WaitWriteDone() )
{
goto HAL_SPI_ReadBlockStatus_return;
}
if ( _HAL_SERFLASH_WaitWriteCmdRdy() == FALSE )
{
goto HAL_SPI_ReadBlockStatus_return;
}
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x3333); // Enable trigger mode
ISP_WRITE(REG_ISP_SPI_WDATA, ISP_SPI_CMD_RDBLOCK); // Read Block Lock Status
if ( !_HAL_SERFLASH_WaitWriteDataRdy() )
{
goto HAL_SPI_ReadBlockStatus_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x10)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_ReadBlockStatus_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x08)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_ReadBlockStatus_return;
}
ISP_WRITE(REG_ISP_SPI_WDATA, BITS(7:0, ((u32FlashAddr >> 0x00)&0xFF)));
if(!_HAL_SERFLASH_WaitWriteDataRdy())
{
goto HAL_SPI_ReadBlockStatus_return;
}
ISP_WRITE(REG_ISP_SPI_RDREQ, 0x01); // SPI read request
if ( _HAL_SERFLASH_WaitReadDataRdy() == FALSE )
{
goto HAL_SPI_ReadBlockStatus_return;
}
u8Val = ISP_READ(REG_ISP_SPI_RDATA);
HAL_SPI_ReadBlockStatus_return:
ISP_WRITE(REG_ISP_SPI_CECLR, ISP_SPI_CECLR); // SPI CEB dis
ISP_WRITE(REG_ISP_TRIGGER_MODE, 0x2222); // disable trigger mode
_HAL_ISP_Disable();
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return u8Val;
}
/*static MS_U32 _Get_Time(void)
{
struct timespec ts;
getnstimeofday(&ts);
return ts.tv_sec* 1000+ ts.tv_nsec/1000000;
}*/
/*
//FSP command , note that it cannot be access only by PM51 if secure boot on.
MS_U8 HAL_SERFLASH_ReadStatusByFSP(void)
{
MS_U8 u8datget;
MS_BOOL bRet = FALSE;
bRet=HAL_FSP_ReadStatusReg(&u8datget);
if(!bRet)
printk("Read status failed \n");
return u8datget ;
}
void HAL_SERFLASH_ReadWordFlashByFSP(MS_U32 u32Addr, MS_U8 *pu8Buf)
{
#define HAL_FSP_READ_SIZE 4
if(!HAL_FSP_Read(u32Addr,HAL_FSP_READ_SIZE,pu8Buf))
printk("HAL_FSP_Read Fail:Addr is %x pu8Buf is :%x\n",(unsigned int)u32Addr, (unsigned int)pu8Buf);
}
void HAL_SERFLASH_CheckEmptyByFSP(MS_U32 u32Addr, MS_U32 u32ChkSize)
{
MS_U32 i;
MS_U32 u32FlashData = 0;
while(HAL_SERFLASH_ReadStatusByFSP()==0x01)
{
printk("device is busy\n");
}
for(i=0;i<u32ChkSize;i+=4)
{
HAL_SERFLASH_ReadWordFlashByFSP(u32Addr+i,(MS_U8 *)u32FlashData);
printk("[FSP Debug] fr = 0x%x\n", (int)u32FlashData);
if(u32FlashData != 0xFFFFFFFF)
{
printk("check failed in addr:%x value:%x\n",(int)(u32Addr+i), (int)u32FlashData);
//while(1);
}
}
}
static void HAL_SERFLASH_EraseByFSP(MS_U32 u32Addr, MS_U8 EraseCmd)
{
HAL_FSP_SectorErase(u32Addr,EraseCmd);
}
void HAL_SERFLASH_EraseSectorByFSP(MS_U32 u32Addr) //4 // erase 4K bytes, need to aligned in 4K boundary
{
HAL_SERFLASH_EraseByFSP(u32Addr,0x20);
}
void HAL_SERFLASH_EraseBlock32KByFSP(MS_U32 u32Addr) //4 // erase 32K bytes, need to aligned in 4K boundary
{
HAL_SERFLASH_EraseByFSP(u32Addr,0x52);
}
void HAL_SERFLASH_EraseBlock64KByFSP(MS_U32 u32Addr) //4 // erase 64K bytes, need to aligned in 4K boundary
{
HAL_SERFLASH_EraseByFSP(u32Addr,0xD8);
}
void HAL_SERFLASH_ProgramFlashByFSP(MS_U32 u32Addr, MS_U32 u32Data)
{
#define HAL_FSP_WRITE_SIZE 4
MS_U8 pu8Write_Data = NULL;
MS_U8 u8index;
for (u8index=0;u8index<HAL_FSP_WRITE_SIZE;u8index++)
{
pu8Write_Data=(MS_U8)(u32Data >>(u8index*8));
if(!HAL_FSP_Write(u32Addr,HAL_FSP_WRITE_SIZE, &pu8Write_Data))
printk("HAL_FSP_WRITE Fail:Addr is %x pu8Buf is :%x\n",(int)u32Addr, (int)pu8Write_Data);
}
}*/
static MS_BOOL _HAL_FSP_WaitDone(void)
{
MS_BOOL bRet = FALSE;
//struct timeval time_st;
unsigned long deadline;
//SER_FLASH_TIME(time_st);
deadline = jiffies + MAX_READY_WAIT_JIFFIES;
do{
if ( (FSP_READ(REG_FSP_DONE_FLAG) & REG_FSP_DONE_FLAG_MASK) == REG_FSP_DONE )
{
bRet = TRUE;
break;
}
//}while (!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR ));
} while (!time_after_eq(jiffies, deadline));
FSP_WRITE_MASK(REG_FSP_DONE_CLR,REG_FSP_CLR,REG_FSP_DONE_CLR_MASK);
return bRet;
}
MS_U8 HAL_FSP_ReadBufs(MS_U8 u8Idx)
{
switch ( u8Idx )
{
case 0: return (MS_U8)((FSP_READ(REG_FSP_RDB0) & 0x00FF) >> 0);
break;
case 1: return (MS_U8)((FSP_READ(REG_FSP_RDB1) & 0xFF00) >> 8);
break;
case 2: return (MS_U8)((FSP_READ(REG_FSP_RDB2) & 0x00FF) >> 0);
break;
case 3: return (MS_U8)((FSP_READ(REG_FSP_RDB3) & 0xFF00) >> 8);
break;
case 4: return (MS_U8)((FSP_READ(REG_FSP_RDB4) & 0x00FF) >> 0);
break;
case 5: return (MS_U8)((FSP_READ(REG_FSP_RDB5) & 0xFF00) >> 8);
break;
case 6: return (MS_U8)((FSP_READ(REG_FSP_RDB6) & 0x00FF) >> 0);
break;
case 7: return (MS_U8)((FSP_READ(REG_FSP_RDB7) & 0xFF00) >> 8);
break;
case 8: return (MS_U8)((FSP_READ(REG_FSP_RDB8) & 0x00FF) >> 0);
break;
case 9: return (MS_U8)((FSP_READ(REG_FSP_RDB9) & 0xFF00) >> 8);
break;
}
return -1;
}
void HAL_FSP_WriteBufs(MS_U8 u8Idx, MS_U8 u8Data)
{
switch ( u8Idx )
{
case 0:
FSP_WRITE_MASK(REG_FSP_WDB0,REG_FSP_WDB0_DATA(u8Data),REG_FSP_WDB0_MASK);
break;
case 1:
FSP_WRITE_MASK(REG_FSP_WDB1,REG_FSP_WDB1_DATA(u8Data),REG_FSP_WDB1_MASK);
break;
case 2:
FSP_WRITE_MASK(REG_FSP_WDB2,REG_FSP_WDB2_DATA(u8Data),REG_FSP_WDB2_MASK);
break;
case 3:
FSP_WRITE_MASK(REG_FSP_WDB3,REG_FSP_WDB3_DATA(u8Data),REG_FSP_WDB3_MASK);
break;
case 4:
FSP_WRITE_MASK(REG_FSP_WDB4,REG_FSP_WDB4_DATA(u8Data),REG_FSP_WDB4_MASK);
break;
case 5:
FSP_WRITE_MASK(REG_FSP_WDB5,REG_FSP_WDB5_DATA(u8Data),REG_FSP_WDB5_MASK);
break;
case 6:
FSP_WRITE_MASK(REG_FSP_WDB6,REG_FSP_WDB6_DATA(u8Data),REG_FSP_WDB6_MASK);
break;
case 7:
FSP_WRITE_MASK(REG_FSP_WDB7,REG_FSP_WDB7_DATA(u8Data),REG_FSP_WDB7_MASK);
break;
case 8:
FSP_WRITE_MASK(REG_FSP_WDB8,REG_FSP_WDB8_DATA(u8Data),REG_FSP_WDB8_MASK);
break;
case 9:
FSP_WRITE_MASK(REG_FSP_WDB9,REG_FSP_WDB9_DATA(u8Data),REG_FSP_WDB9_MASK);
break;
default:
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("HAL_FSP_WriteBufs fails\n"));
break;
}
}
void HAL_FSP_Entry(void)
{
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("[FSP] %s ENTRY fails!\n", __FUNCTION__);
return;
}
HAL_SERFLASH_SelectReadMode(E_SINGLE_MODE);
#if 0
if (gReadMode==E_QUAD_MODE)
{
HAL_QUAD_Enable(1);
}
#endif
}
void HAL_FSP_Exit(void)
{
//if (gReadMode==E_QUAD_MODE)
//HAL_QUAD_Enable(1);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
}
MS_BOOL HAL_FSP_EraseChip(void)
{
MS_BOOL bRet = TRUE;
//DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printf("%s()\n", __FUNCTION__));
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_CE);
HAL_FSP_WriteBufs(2,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(1),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Erase Chip Timeout !!!!\r\n");
}
return bRet;
}
static MS_BOOL _HAL_FSP_BlockErase(MS_U32 u32FlashAddr, EN_FLASH_ERASE eSize)
{
MS_BOOL bRet = TRUE;
MS_U8 u8EAR = u32FlashAddr >> 24;
// DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printf("%s(0x%08X, 0x%08X)\n", __FUNCTION__, (int)u32FlashAddr, (int)eSize));
if (u8EAR != _u8RegEAR)
{
HAL_FSP_WriteExtAddrReg(u8EAR);
}
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,eSize);
HAL_FSP_WriteBufs(2,(MS_U8)((u32FlashAddr>>0x10)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u32FlashAddr>>0x08)&0xFF));
HAL_FSP_WriteBufs(4,(MS_U8)((u32FlashAddr>>0x00)&0xFF));
HAL_FSP_WriteBufs(5,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(4),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Block Erase Timeout !!!!offset:0x%lx cmd:0x%x \r\n", u32FlashAddr, eSize);
}
return bRet;
}
MS_BOOL HAL_FSP_BlockErase(MS_U32 u32StartBlock, MS_U32 u32EndBlock, MS_BOOL bWait)
{
MS_BOOL bRet = TRUE;
MS_U32 u32Idx;
MS_U32 u32FlashAddr = 0;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08x, 0x%08x, %d)\n", __FUNCTION__, (unsigned int)u32StartBlock, (unsigned int)u32EndBlock, (int)bWait));
MS_ASSERT( u32StartBlock<=u32EndBlock && u32EndBlock<NUMBER_OF_SERFLASH_SECTORS );
HAL_FSP_Entry();
for( u32Idx = u32StartBlock; u32Idx <= u32EndBlock; u32Idx++ )
{
bRet &= HAL_SERFLASH_BlockToAddress(u32Idx, &u32FlashAddr);
bRet &= _HAL_FSP_BlockErase(u32FlashAddr,FLASH_ERASE_64K);
}
HAL_FSP_Exit();
return bRet;
}
MS_BOOL HAL_FSP_SectorErase(MS_U32 u32SectorAddress,MS_U8 u8cmd)
{
MS_BOOL bRet = TRUE;
HAL_FSP_Entry();
bRet &= _HAL_FSP_BlockErase(u32SectorAddress,u8cmd);
HAL_FSP_Exit();
return bRet;
}
MS_BOOL HAL_FSP_CheckWriteDone(void)
{
MS_BOOL bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
HAL_FSP_WriteBufs(0,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(1),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_1STCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Check Write Done Timeout !!!!\r\n");
}
return bRet;
}
MS_BOOL HAL_FSP_Write_Burst(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = TRUE;
MS_U8 u8Status = 0;
MS_U16 u16Data = 0;
MS_U32 u32I, u32J, u32K;
MS_U32 u32quotient;
MS_U32 u32remainder;
MS_U32 u32PageIdx;
MS_U32 u32PageRemainder;
//write enable
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
u16Data = FSP_READ(REG_FSP_CTRL);
u16Data &= 0xFF;
FSP_WRITE(REG_FSP_CTRL, u16Data);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP]WB Write enable command Fail!!!!\r\n");
return bRet;
}
u32PageIdx = u32Size / FLASH_PAGE_SIZE;
u32PageRemainder = u32Size % FLASH_PAGE_SIZE;
if(u32PageRemainder)
{
u32PageIdx++;
u32Size = u32PageRemainder;
}
for(u32K = 0; u32K < u32PageIdx; u32K++)
{
//write command
HAL_FSP_WriteBufs(0,SPI_CMD_PP);
HAL_FSP_WriteBufs(1,(MS_U8)((u32Addr>>0x10)&0xFF));
HAL_FSP_WriteBufs(2,(MS_U8)((u32Addr>>0x08)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u32Addr>>0x00)&0xFF));
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(4),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP]WB0 Write command Fail!!!!\r\n");
return bRet;
}
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_ENABLE_BURST);
u32quotient = (u32Size / REG_FSP_MAX_WRITEDATA_SIZE);
u32remainder = (u32Size % REG_FSP_MAX_WRITEDATA_SIZE);
if(u32remainder)
{
u32quotient++;
u32Size = u32remainder;
}
else
u32Size = REG_FSP_MAX_WRITEDATA_SIZE;
for( u32I = 0; u32I < u32quotient; u32I++ )
{
for( u32J = 0; u32J < u32Size; u32J++ )
{
HAL_FSP_WriteBufs(u32J,*(pu8Data+u32J));
}
pu8Data += u32I;
u32Size = REG_FSP_MAX_WRITEDATA_SIZE;
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(u32J), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER, REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP]WB1 Write command Fail!!!!\r\n");
return bRet;
}
}
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_DISABLE_BURST);
do
{
HAL_FSP_WriteBufs(0,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(1),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP]WB2 Write command Fail!!!!\r\n");
return bRet;
}
u8Status = HAL_FSP_ReadBufs(0);
}while(u8Status & FLASH_OIP);
u32Size = FLASH_PAGE_SIZE;
}
return bRet;
}
MS_BOOL HAL_QPI_Enable(MS_BOOL bEnable)
{
MS_BOOL bRet = TRUE;
HAL_FSP_WriteBufs(0,SPI_CMD_QPI);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] QPI Enable Timeout !!!!\r\n");
}
return bRet;
}
MS_BOOL HAL_QPI_RESET(MS_BOOL bEnable)
{
MS_BOOL bRet = TRUE;
HAL_FSP_WriteBufs(0,SPI_CMD_QPI_RST);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] QPI Reset Timeout !!!!\r\n");
}
return bRet;
}
MS_BOOL HAL_QUAD_Enable(MS_BOOL bEnable)
{
MS_BOOL bRet = TRUE;
MS_U8 u8data;
MS_U16 u16data;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("[FSP] %s %d\n", __FUNCTION__, bEnable));
if(_hal_SERFLASH.u8MID == MID_MXIC)
{
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR, 1, &u8data);
if(bEnable)
{
if((u8data & SF_SR_QUAD) == 0) {
u8data |= SF_SR_QUAD;
bRet = HAL_FSP_WriteStatusReg(u8data);
}
} else {
if((u8data & SF_SR_QUAD) != 0) {
u8data &= ~SF_SR_QUAD;
bRet = HAL_FSP_WriteStatusReg(u8data);
}
}
}
else if(_hal_SERFLASH.u8MID == MID_GD||_hal_SERFLASH.u8MID == MID_ZB)
{
MS_U8 u8status[3]={0xff,0xff,0xff};
if(bEnable)
u8data = BITS(1:1, 1);
else
u8data = BITS(1:1, 0);
bRet = HAL_FSP_WriteStatus(0x31, 1, &u8data);
bRet = HAL_FSP_ReadStatus(0x35, 1, u8status);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("[FSP] GD status2:0x%x 0x%x 0x%x\n",u8status[0], u8status[1], u8status[2]));
}
else if(_hal_SERFLASH.u8MID == MID_XMC)
{
if(_hal_SERFLASH.u8DID0 == 0x70)
{
bRet = HAL_FSP_ReadStatus(0x3A, 1, &u8data);//enter otp mode
}
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR, 1, &u8data);
if(bEnable)
{
if((u8data & SF_SR_QUAD) == 0) {
u8data |= SF_SR_QUAD;
bRet = HAL_FSP_WriteStatusReg(u8data);
}
} else {
if((u8data & SF_SR_QUAD) != 0) {
u8data &= ~SF_SR_QUAD;
bRet = HAL_FSP_WriteStatusReg(u8data);
}
}
if(_hal_SERFLASH.u8DID0 == 0x70)
{
bRet = HAL_FSP_ReadStatus(0x04, 1, &u8data);//exit otp mode
}
}
else if(_hal_SERFLASH.u8MID == MID_WB || _hal_SERFLASH.u8MID == MID_EON)
{
MS_U8 u8status[1]={0xff};
if(bEnable)
u8data = BITS(1:1, 1);
else
u8data = BITS(1:1, 0);
bRet = HAL_FSP_WriteStatus(SPI_CMD_WRSR2, 1, &u8data);
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR2, 1, u8status);
//printk("[FSP] WB u8data:0x%x status2:0x%x\n",u8data, u8status[0]);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("[FSP] WB status2:0x%x\n",u8status[0]));
}
else if(_hal_SERFLASH.u8MID == MID_XTX)
{
if(bEnable) {
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR2, 1, &u8data);
u8data |= BIT1;
u16data = (u8data << 0x08)&0xffff;
u16data |= BITS(3:0,3);
bRet = HAL_FSP_WriteStatusReg2(u16data);
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR2, 1, &u8data);
}
else
{
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR2, 1, &u8data);
u8data &= ~BIT1;
u16data = (u8data << 0x08)&0xffff;
u16data |= BITS(3:0,3);
bRet = HAL_FSP_WriteStatusReg2(u16data);
}
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR2, 1, &u8data);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("[FSP] XTX status2:0x%x\n",u8data));
}
return bRet;
}
void HAL_FSP_Write_Egine(MS_U32 u32Write_Addr, MS_U8 u8Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = TRUE;
MS_U32 u32I;
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_PP);
HAL_FSP_WriteBufs(2,(MS_U8)((u32Write_Addr>>0x10)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u32Write_Addr>>0x08)&0xFF));
HAL_FSP_WriteBufs(4,(MS_U8)((u32Write_Addr>>0x00)&0xFF));
for( u32I = 0; u32I < u8Size; u32I++ )
{
HAL_FSP_WriteBufs((5+u32I),*(pu8Data+u32I));
}
HAL_FSP_WriteBufs((5 + u8Size),SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1((4+u8Size)),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP]PP FAIL Timeout !!!!\r\n");
}
}
MS_BOOL HAL_FSP_Write(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
// MS_U32 u32I;
MS_BOOL bRet = TRUE;
MS_U32 u32J;
MS_U32 u32quotient;
MS_U8 u8BufSize = REG_FSP_WRITEDATA_SIZE;
MS_U8 u8dat_align=0;
MS_U8 u8addr_align=0;
MS_U8 u8remainder=0;
MS_U8 u8writebytes=0;
u8addr_align=((MS_U8)(u32Addr))&0xF;
u8dat_align=u8addr_align%4;
if(u8dat_align)
{
u8writebytes = 4-u8dat_align;
HAL_FSP_Write_Egine(u32Addr,u8writebytes,pu8Data);
u32Addr += u8writebytes;
pu8Data += u8writebytes;
}
u32quotient = ((u32Size-u8writebytes) / u8BufSize);
u8remainder = ((u32Size-u8writebytes) % u8BufSize);
for(u32J = 0; u32J < u32quotient; u32J++)
{
HAL_FSP_Write_Egine(u32Addr,u8BufSize,pu8Data);
u32Addr += u8BufSize;
pu8Data += u8BufSize;
}
if(u8remainder)
{
HAL_FSP_Write_Egine(u32Addr,u8remainder,pu8Data);
u32Addr += u8remainder;
pu8Data += u8remainder;
}
return bRet;
}
MS_BOOL HAL_FSP_Read(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = TRUE;
MS_U32 u32Idx, u32J;
MS_U32 u32quotient;
MS_U32 u32remainder;
MS_U8 u8BufSize = REG_FSP_READDATA_SIZE;
u32quotient = (u32Size / u8BufSize);
u32remainder = (u32Size % u8BufSize);
if(u32remainder)
{
u32quotient++;
u32Size = u32remainder;
}
else
u32Size = u8BufSize;
for(u32J = 0; u32J < u32quotient; u32J++)
{
HAL_FSP_WriteBufs(0, SPI_CMD_READ);
HAL_FSP_WriteBufs(1,(MS_U8)((u32Addr>>0x10)&0xFF));
HAL_FSP_WriteBufs(2,(MS_U8)((u32Addr>>0x08)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u32Addr>>0x00)&0xFF));
/*Lost first byte in using normal read command;Dummy Byte for Fast Read*/
//HAL_FSP_WriteBufs(4, 0x00);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(4),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(u32Size),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] RIU Read FAIL Timeout !!!!\r\n");
}
for( u32Idx = 0; u32Idx < u32Size; u32Idx++ )
*(pu8Data + u32Idx) = HAL_FSP_ReadBufs(u32Idx);
pu8Data += u32Idx;
u32Addr += u32Idx;
u32Size = u8BufSize;
}
return bRet;
}
MS_BOOL HAL_FSP_BurstRead(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = TRUE;
MS_U32 u32Idx, u32J;
MS_U32 u32quotient;
MS_U32 u32remainder;
MS_U8 u8BufSize = REG_FSP_READDATA_SIZE;
u32quotient = (u32Size / u8BufSize);
u32remainder = (u32Size % u8BufSize);
if(u32remainder)
{
u32quotient++;
u32Size = u32remainder;
}
else
u32Size = u8BufSize;
HAL_FSP_WriteBufs(0, SPI_CMD_FASTREAD);
HAL_FSP_WriteBufs(1,(MS_U8)((u32Addr>>0x10)&0xFF));
HAL_FSP_WriteBufs(2,(MS_U8)((u32Addr>>0x08)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u32Addr>>0x00)&0xFF));
HAL_FSP_WriteBufs(4, 0x00);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(5),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(u32Size),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
for(u32J = 0; u32J < u32quotient; u32J++)
{
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_ENABLE_BURST);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(0),REG_FSP_WBF_SIZE0_MASK);
if(!bRet)
{
printk("[FSP] RIU Read Burst FAIL Timeout !!!!\r\n");
}
for( u32Idx = 0; u32Idx < u32Size; u32Idx++ )
*(pu8Data + u32Idx) = HAL_FSP_ReadBufs(u32Idx);
pu8Data += u32Idx;
if(u32remainder && (u32remainder != u8BufSize))
{
u32Size = u8BufSize;
u32remainder = u8BufSize;
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(u32Size),REG_FSP_RBF_SIZE0_MASK);
}
}
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_DISABLE_BURST);
return bRet;
}
MS_BOOL HAL_FSP_WriteProtect_Area(MS_BOOL bEnableAllArea, MS_U8 u8BlockProtectBits)
{
MS_BOOL bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d)\n", __FUNCTION__, bEnableAllArea));
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(0);
//MsOS_DelayTask(bEnableAllArea ? 5 : 20);
udelay(20);
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_WRSR);
if (bEnableAllArea)
{
// SF_SR_SRWD: SRWD Status Register Write Protect
//HAL_FSP_WriteBufs(2,(MS_U8)(SF_SR_SRWD | SERFLASH_WRSR_BLK_PROTECT));
HAL_FSP_WriteBufs(2,(MS_U8)(SERFLASH_WRSR_BLK_PROTECT));
}
else
{
// [4:2] or [5:2] protect blocks
//HAL_FSP_WriteBufs(2,(SF_SR_SRWD | u8BlockProtectBits));
HAL_FSP_WriteBufs(2,(u8BlockProtectBits));
}
HAL_FSP_WriteBufs(3,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(2),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Write Protect Area FAIL Timeout !!!!\r\n");
}
if( bEnableAllArea )
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(bEnableAllArea);
return bRet;
}
MS_BOOL HAL_FSP_WriteProtect(MS_BOOL bEnable)
{
MS_BOOL bRet = TRUE;
MS_U8 u8Status;
MS_U8 u8Status2;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(%d)\n", __FUNCTION__, bEnable));
bRet = HAL_FSP_ReadStatusReg(&u8Status);
if(_hal_SERFLASH.u8MID == MID_XTX && _hal_SERFLASH.u8DID0 == 0x40 && _hal_SERFLASH.u8DID1 == 0x18)
{
HAL_FSP_ReadStatusReg2(&u8Status2);
}
if(bRet == FALSE)
return bRet;
u8Status |= SF_SR_SRWD;//checked on WB/GD/MX
//quad mode should be handled in read/write
#if 0 //device dependent
if (gReadMode==E_QUAD_MODE)
u8Status |= SF_SR_QUAD;
#endif
//clear BP bits
if(_hal_SERFLASH.u8MID == MID_GD)
{
u8Status &= ~BITS(6:2,0x1F);
}
else {//check on WB/MX
u8Status &= ~BITS(5:2,0xF);
}
if (bEnable)
u8Status |= SERFLASH_WRSR_BLK_PROTECT;
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(0);
//MsOS_DelayTask(bEnable ? 5 : 20);
udelay(20);
if(_hal_SERFLASH.u8MID == MID_XTX && _hal_SERFLASH.u8DID0 == 0x40 && _hal_SERFLASH.u8DID1 == 0x18)
{
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_WRSR);
HAL_FSP_WriteBufs(2, u8Status);
HAL_FSP_WriteBufs(3, u8Status2);
HAL_FSP_WriteBufs(4,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(3),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
}
else
{
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_WRSR);
HAL_FSP_WriteBufs(2, u8Status);
HAL_FSP_WriteBufs(3,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(2),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
}
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Write Protect FAIL Timeout !!!!\r\n");
}
if( bEnable )
_HAL_SERFLASH_ActiveFlash_Set_HW_WP(bEnable);
return bRet;
}
MS_BOOL HAL_FSP_ReadID(MS_U8 *pu8Data, MS_U32 u32Size)
{
MS_BOOL bRet = TRUE;
MS_U8 *u8ptr = pu8Data;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s() in\n", __FUNCTION__));
HAL_SERFLASH_SelectReadMode(E_SINGLE_MODE);//set normal mode(1-1-1)
HAL_FSP_WriteBufs(0, SPI_CMD_RDID);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(u32Size),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Read ID FAIL Timeout !!!!\r\n");
}
*(u8ptr + 0) = HAL_FSP_ReadBufs(0);
*(u8ptr + 1) = HAL_FSP_ReadBufs(1);
*(u8ptr + 2) = HAL_FSP_ReadBufs(2);
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s() out\n", __FUNCTION__));
return bRet;
}
MS_BOOL HAL_FSP_ReadStatusReg(MS_U8 *pu8StatusReg)
{
MS_BOOL bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
*pu8StatusReg = 0xFF;
HAL_FSP_WriteBufs(0,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(1),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Read Status FAIL Timeout !!!!\r\n");
}
*pu8StatusReg = HAL_FSP_ReadBufs(0);
return bRet;
}
MS_BOOL HAL_FSP_ReadStatusReg2(MS_U8 *pu8StatusReg)
{
MS_BOOL bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
*pu8StatusReg = 0xFF;
HAL_FSP_WriteBufs(0,SPI_CMD_RDSR2);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(1),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Read status2 FAIL Timeout !!!!\r\n");
}
*pu8StatusReg = HAL_FSP_ReadBufs(0);
return bRet;
}
MS_BOOL HAL_FSP_ReadREMS4(MS_U8 * pu8Data, MS_U32 u32Size)
{
MS_BOOL bRet = FALSE;
MS_U32 u32Index;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
MS_ASSERT( MsOS_In_Interrupt() == FALSE );
MS_ASSERT(_HAL_SERFLASH_Check51RunMode());
if (FALSE == MS_SERFLASH_OBTAIN_MUTEX(_s32SERFLASH_Mutex, SERFLASH_MUTEX_WAIT_TIME))
{
printk("%s ENTRY fails!\n", __FUNCTION__);
return FALSE;
}
HAL_SERFLASH_SelectReadMode(E_FAST_MODE);//set normal mode(1-1-1)
HAL_FSP_WriteBufs(0,ISP_SPI_CMD_REMS);
HAL_FSP_WriteBufs(1,0); //dummy
HAL_FSP_WriteBufs(2,0); //dummy
HAL_FSP_WriteBufs(3,0); // start address
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(4),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(u32Size),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Read REMS4 FAIL Timeout !!!!\r\n");
}
for( u32Index = 0; u32Index < u32Size; u32Index++ )
*(pu8Data + u32Index) = HAL_FSP_ReadBufs(u32Index);
MS_SERFLASH_RELEASE_MUTEX(_s32SERFLASH_Mutex);
return bRet;
}
MS_BOOL HAL_FSP_WriteStatusReg(MS_U16 u16StatusReg)
{
MS_BOOL bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_WRSR);
HAL_FSP_WriteBufs(2,(MS_U8)((u16StatusReg>>0x00)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u16StatusReg>>0x08)&0xFF));
HAL_FSP_WriteBufs(4,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(3),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Write Status FAIL Timeout !!!!\r\n");
}
return bRet;
}
MS_BOOL HAL_FSP_WriteStatusReg2(MS_U16 u16StatusReg)
{
MS_BOOL bRet = TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_WRSR);
HAL_FSP_WriteBufs(2,(MS_U8)((u16StatusReg>>0x00)&0xFF));
HAL_FSP_WriteBufs(3,(MS_U8)((u16StatusReg>>0x08)&0xFF));
HAL_FSP_WriteBufs(4,SPI_CMD_RDSR2);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(3),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] Write Status FAIL Timeout !!!!\r\n");
}
return bRet;
}
MS_BOOL HAL_FSP_ReadStatus(MS_U8 u8CMD, MS_U8 u8CountData, MS_U8* pu8Data)
{
MS_BOOL bRet = TRUE;
MS_U32 u32I;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
*pu8Data = 0xFF;
HAL_FSP_WriteBufs(0,u8CMD);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(0),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(u8CountData),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_OFF,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_OFF,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] ReadStatus FAIL Timeout !!!!\r\n");
}
for( u32I = 0; u32I < u8CountData; u32I++ )
{
*(pu8Data+u32I) = HAL_FSP_ReadBufs(u32I);
}
return bRet;
}
MS_BOOL HAL_FSP_WriteStatus(MS_U8 u8CMD, MS_U8 u8CountData, MS_U8* pu8Data)
{
MS_BOOL bRet = TRUE;
MS_U32 u32I;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,u8CMD);
for( u32I = 0; u32I < u8CountData; u32I++ )
{
HAL_FSP_WriteBufs((2+u32I),*(pu8Data+u32I));
}
HAL_FSP_WriteBufs(2+u8CountData,SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1((1+u8CountData)),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(1),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(1),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_ON,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD,REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] WriteStatus FAIL Timeout !!!!\r\n");
}
return bRet;
}
MS_BOOL HAL_FSP_WriteExtAddrReg(MS_U8 u8ExtAddrReg)
{
MS_BOOL bRet = TRUE;
if (!_bHasEAR)
return FALSE;
else if (u8ExtAddrReg == _u8RegEAR)
return TRUE;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s()", __FUNCTION__));
HAL_FSP_WriteBufs(0,SPI_CMD_WREN);
HAL_FSP_WriteBufs(1,SPI_CMD_WREAR);
HAL_FSP_WriteBufs(2,u8ExtAddrReg);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE0(1),REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE1(2),REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE,REG_FSP_WBF_SIZE2(0),REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE0(0),REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE1(0),REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE,REG_FSP_RBF_SIZE2(0),REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_ENABLE,REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_NRESET,REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_INT,REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_2NDCMD_ON,REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_3THCMD_OFF,REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL,REG_FSP_FSCHK_ON,REG_FSP_FSCHK_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE,REG_FSP_TRIGGER_MASK);
bRet &= _HAL_FSP_WaitDone();
if (bRet)
{
_u8RegEAR = u8ExtAddrReg;
}
return bRet;
}
#ifndef CONFIG_RIUISP
void _HAL_BDMA_INIT(U32 u32DataSize)
{
u32DataSize += BDMA_ALIGN;
if(pu8BDMA_virt != 0 && u32BdmaSize < u32DataSize)
{
int err;
err = msys_release_dmem(&mem_info);
if(u32DataSize >= BDMA_SIZE_WARNING)
{
pr_err("\n\n --->from %ld to %d\n\n", u32BdmaSize, u32DataSize);
}
if(0 != err)
{
printk("[Ser flash] Unable to free BDMA mem bus=0x%08X (err:%d)\n", (unsigned int)pu8BDMA_bus, err);
}
pu8BDMA_virt = 0;
}
if(pu8BDMA_virt != 0)
return;
mem_info.length = u32DataSize;
strcpy(mem_info.name, "BDMA_FSP_WBUFF");
if(!msys_request_dmem(&mem_info) )
{
pu8BDMA_phys = mem_info.phys;
pu8BDMA_virt = mem_info.kvirt;
pu8BDMA_bus = pu8BDMA_phys&0x1FFFFFFF;
u32BdmaSize = mem_info.length;
printk("[Ser flash] phys=0x%08x, virt=0x%08x, bus=0x%08x len:0x%lX\n", (unsigned int)pu8BDMA_phys, (unsigned int)pu8BDMA_virt, (unsigned int)pu8BDMA_bus, u32BdmaSize);
}
else
{
printk("[Ser flash] BDMA request buffer failed");
}
//printk("[Ser flash] BDMA vaddr=0x%x,paddr=0x%x bus=0x%x len=0x%x\n", (unsigned int)pu8BDMA_virt, (unsigned int)pu8BDMA_phys, (unsigned int)pu8BDMA_bus , (unsigned int)mem_info.length);
}
/*
static bool _HAL_BDMA_FlashToMem(MS_U32 u32Src_off, MS_U32 u32Dst_off, MS_U32 u32Len)
{
struct timeval time_st;
MS_U16 u16data;
MS_BOOL bRet;
//Set source and destination path
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 0x00);
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x12<<2)), 0X4035);
// Set start address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x14<<2)), (u32Src_off & 0x0000FFFF));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x15<<2)), (u32Src_off >> 16));
// Set end address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x16<<2)), (u32Dst_off & 0x0000FFFF));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x17<<2)), (u32Dst_off >> 16));
// Set Size
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x18<<2)), (u32Len & 0x0000FFFF));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x19<<2)), (u32Len >> 16));
// Trigger
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x10<<2)), 1);
SER_FLASH_TIME(time_st);
do
{
//check done
u16data = READ_WORD(_hal_isp.u32BdmaBaseAddr + ((0x11)<<2));
if(u16data & 8)
{
//clear done
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x11<<2)), 8);
bRet = TRUE;
break;
}
} while(!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
if(bRet == FALSE)
{
printk("Wait for BDMA Done fails!\n");
}
return bRet;
}
static bool _HAL_BDMA_READ(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet;
//printk("[Ser flash] %s off=0x%x, len=0x%x\n", __FUNCTION__, (unsigned int)u32Addr, (unsigned int)u32Size);
if(pu8BDMA_phys==0)
_HAL_BDMA_INIT();
if(pu8BDMA_phys==0)
return FALSE;
bRet = _HAL_BDMA_FlashToMem( u32Addr, pu8BDMA_phys&0x0FFFFFFF, u32Size);
if(bRet == TRUE)
memcpy((void *)pu8Data,(const void *) pu8BDMA_virt, u32Size);
return bRet;
}
*/
void DumpMemory(MS_U32 u32Addr, MS_U32 u32Size)
{
MS_U8* pdata = (MS_U8*)u32Addr;
printk("Address 0x%p:", (void*)u32Addr);
while( (MS_U32)pdata != (u32Addr+u32Size))
{
printk(" %02x", *pdata);
pdata++;
}
printk("\n");
}
MS_BOOL CompareMemory(MS_U32 u32Addr, MS_U32 u32Addr2, MS_U32 u32Size)
{
MS_U8* pdata = (MS_U8*)u32Addr;
MS_U8* pdata2 = (MS_U8*)u32Addr2;
while( (MS_U32)pdata != (u32Addr+u32Size))
{
if(*pdata != *pdata2)
{
DumpMemory(u32Addr, u32Size);
DumpMemory(u32Addr2, u32Size);
return FALSE;
}
pdata++,pdata2++;
}
return TRUE;
}
static MS_BOOL _HAL_QP_BDMAToFSP(MS_U32 u32Src_off, MS_U32 u32Dst_off, MS_U32 u32Len)
{
//struct timeval time_st;
MS_U16 u16data;
MS_BOOL bRet=FALSE;
unsigned long deadline;
MS_U8 u8EAR = u32Dst_off >> 24;
MS_U8 u8status;
if(u32Len>256)
return FALSE;
if((u8EAR == 0) && ((u32Dst_off + u32Len) > 0x1000000))
{
printk("[FSP] PP BDMA cross the 16MB boundary\r\n");
return FALSE;
}
if(u8EAR != _u8RegEAR)
{
HAL_FSP_WriteExtAddrReg(u8EAR);
}
_HAL_BDMA_INIT(u32Len);
if(pu8BDMA_virt == 0)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("%s, remap BDMA buffer failed)\n", __FUNCTION__));
return FALSE;
}
memcpy((void *)pu8BDMA_virt, (const void *)u32Src_off, u32Len);
Chip_Flush_Cache_Range(pu8BDMA_virt, u32Len);
// printk(" %s(0x%08X, %3d)\n", __FUNCTION__, (int)u32Dst_off, (int)u32Len);
do
{
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR, 1, &u8status);
if (!bRet)
{
break;
}
} while (u8status & SF_SR_WIP_MASK);
/*FSP*/
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
HAL_FSP_WriteBufs(0, SPI_CMD_WREN);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(1), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(0), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(0), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(0), REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] FSP(SPI_CMD_WREN) FAIL Timeout !!!!\r\n");
}
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
HAL_FSP_WriteBufs(0, SPI_CMD_QP);
HAL_FSP_WriteBufs(1, (MS_U8)((u32Dst_off>>0x10)&0xFF));
HAL_FSP_WriteBufs(2, (MS_U8)((u32Dst_off>>0x08)&0xFF));
HAL_FSP_WriteBufs(3, (MS_U8)((u32Dst_off>>0x00)&0xFF));
//HAL_FSP_WriteBufs(4, 0x00); //dummy
//HAL_FSP_WriteBufs(6, SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(4), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(0), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(0), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(0), REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] FSP FAIL Timeout !!!!\r\n");
}
//printk("%s, u32Dst_off = %ld\r\n", __FUNCTION__, u32Dst_off);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
//set FSP Outside replace
FSP_WRITE(REG_FSP_WBF_SIZE_OUTSIDE, (u32Len + 1)&0x0FFF);
//FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_REPLACED(5), REG_FSP_WBF_REPLACED_MASK);
//FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_MODE(0), REG_FSP_WBF_MODE_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_OUTSIDE_ENABLE, REG_FSP_WBF_OUTSIDE_ENABLE_MASK);
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_ENABLE_BURST);
FSP_WRITE(REG_FSP_QUAD_MODE, REG_FSP_QUAD_ENABLE);
//FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_REPLACED(5), REG_FSP_WBF_REPLACED_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(0), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(0), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(0), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(0), REG_FSP_RBF_SIZE2_MASK);
/*BDMA */
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 0x00);
//Set source and destination path
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x2<<2)), 0X2B40); //MIU to FSP(0xB)
// Set source as MIU0
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x3<<2)), 0X0000); //MIU0 as channel0
// Set start address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x4<<2)), (pu8BDMA_bus & 0x0000FFFF));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x5<<2)), (pu8BDMA_bus >> 16));
// Set end address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x6<<2)), 0x0);
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x7<<2)), 0x0);
// Set Size
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x8<<2)), (u32Len & 0x0000FFFF));
//WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x9<<2)), (u32Len >> 16));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 1); // Trigger
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
//check BDMA done
//SER_FLASH_TIME(time_st);
deadline = jiffies + MAX_READY_WAIT_JIFFIES;
do
{
//check done
u16data = READ_WORD(_hal_isp.u32BdmaBaseAddr + ((0x1)<<2));
if(u16data & 8)
{
//clear done
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x1<<2)), 8);
bRet = TRUE;
break;
}
//} while(!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
} while (!time_after_eq(jiffies, deadline));
if(bRet == FALSE)
{
printk("Wait for BDMA Done fails!\n");
}
//check FSP done
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] BDMA2FSP FAIL Timeout !!!!\r\n");
}
FSP_WRITE(REG_FSP_WBF_SIZE_OUTSIDE, 0x0);
FSP_WRITE(REG_FSP_WBF_OUTSIDE, 0x0);
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_BURST_OFF);
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_DISABLE_BURST);
FSP_WRITE(REG_FSP_QUAD_MODE, 0x0);
//FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
//CompareMemory(u32Src_off, u32Dst_off+0xF4000000, u32Len);
#if 0
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] FSP(SPI_CMD_RDSR) FAIL Timeout !!!!\r\n");
}
#endif
do
{
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR, 1, &u8status);
if (!bRet)
{
break;
}
} while (u8status & SF_SR_WIP_MASK);
//printk("[FSP] GD status2:0x%x 0x%x 0x%x\n",u8status[0], u8status[1], u8status[2]);
return bRet;
}
static MS_BOOL _HAL_4PP_BDMAToFSP(MS_U32 u32Src_off, MS_U32 u32Dst_off, MS_U32 u32Len)
{
MS_U16 u16data;
MS_BOOL bRet=FALSE;
unsigned long deadline;
MS_U8 u8EAR = u32Dst_off >> 24;
MS_U8 u8status;
if(u32Len>256)
return FALSE;
if((u8EAR == 0) && ((u32Dst_off + u32Len) > 0x1000000))
{
printk("[FSP] PP BDMA cross the 16MB boundary\r\n");
return FALSE;
}
if(u8EAR != _u8RegEAR)
{
HAL_FSP_WriteExtAddrReg(u8EAR);
}
_HAL_BDMA_INIT(u32Len);
if(pu8BDMA_virt == 0)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("%s, remap BDMA buffer failed)\n", __FUNCTION__));
return FALSE;
}
memcpy((void *)pu8BDMA_virt, (const void *)u32Src_off, u32Len);
Chip_Flush_Cache_Range(pu8BDMA_virt, u32Len);
do
{
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR, 1, &u8status);
if (!bRet)
{
break;
}
} while (u8status & SF_SR_WIP_MASK);
/*FSP*/
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
HAL_FSP_WriteBufs(0, SPI_CMD_WREN);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(1), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(0), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(0), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(0), REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] FSP(SPI_CMD_WREN) FAIL Timeout !!!!\r\n");
}
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
HAL_FSP_WriteBufs(0, SPI_CMD_4PP);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(1), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(0), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(0), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(0), REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] FSP FAIL Timeout !!!!\r\n");
}
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
HAL_FSP_WriteBufs(0, (MS_U8)((u32Dst_off>>0x10)&0xFF));
HAL_FSP_WriteBufs(1, (MS_U8)((u32Dst_off>>0x08)&0xFF));
HAL_FSP_WriteBufs(2, (MS_U8)((u32Dst_off>>0x00)&0xFF));
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(3), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(0), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(0), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(0), REG_FSP_RBF_SIZE2_MASK);
//set FSP Outside replace
FSP_WRITE(REG_FSP_WBF_SIZE_OUTSIDE, (u32Len + 1)&0x00FFFFFF);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_REPLACED(3), REG_FSP_WBF_REPLACED_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_MODE(0), REG_FSP_WBF_MODE_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_OUTSIDE_ENABLE, REG_FSP_WBF_OUTSIDE_ENABLE_MASK);
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_ENABLE_BURST);
FSP_WRITE(REG_FSP_QUAD_MODE, REG_FSP_QUAD_ENABLE);
/*BDMA */
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 0x00);
//Set source and destination path
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x2<<2)), 0X2B40); //MIU to FSP(0xB)
// Set source as MIU0
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x3<<2)), 0X0000); //MIU0 as channel0
// Set start address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x4<<2)), (pu8BDMA_bus & 0x0000FFFF));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x5<<2)), (pu8BDMA_bus >> 16));
// Set end address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x6<<2)), 0x0);
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x7<<2)), 0x0);
// Set Size
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x8<<2)), (u32Len & 0x0000FFFF));
//WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x9<<2)), (u32Len >> 16));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 1); // Trigger
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
//check BDMA done
//SER_FLASH_TIME(time_st);
deadline = jiffies + MAX_READY_WAIT_JIFFIES;
do
{
//check done
u16data = READ_WORD(_hal_isp.u32BdmaBaseAddr + ((0x1)<<2));
if(u16data & 8)
{
//clear done
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x1<<2)), 8);
bRet = TRUE;
break;
}
//} while(!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
} while (!time_after_eq(jiffies, deadline));
if(bRet == FALSE)
{
printk("Wait for BDMA Done fails!\n");
}
//check FSP done
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] BDMA2FSP FAIL Timeout !!!!\r\n");
}
FSP_WRITE(REG_FSP_WBF_SIZE_OUTSIDE, 0x0);
FSP_WRITE(REG_FSP_WBF_OUTSIDE, 0x0);
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_BURST_OFF);
QSPI_WRITE(REG_SPI_BURST_WRITE,REG_SPI_DISABLE_BURST);
FSP_WRITE(REG_FSP_QUAD_MODE, 0x0);
do
{
bRet = HAL_FSP_ReadStatus(SPI_CMD_RDSR, 1, &u8status);
if (!bRet)
{
break;
}
} while (u8status & SF_SR_WIP_MASK);
return bRet;
}
#if 0
static MS_BOOL _HAL_PP_BDMAToFSP(MS_U32 u32Src_off, MS_U32 u32Dst_off, MS_U32 u32Len)
{
//struct timeval time_st;
MS_U16 u16data;
MS_BOOL bRet=FALSE;
unsigned long deadline;
MS_U8 u8EAR = u32Dst_off >> 24;
if(u32Len>256)
return FALSE;
if((u8EAR == 0) && ((u32Dst_off + u32Len) > 0x1000000))
{
printk("[FSP] PP BDMA cross the 16MB boundary\r\n");
return FALSE;
}
if(u8EAR != _u8RegEAR)
{
HAL_FSP_WriteExtAddrReg(u8EAR);
}
_HAL_BDMA_INIT(u32Len);
if(pu8BDMA_virt == 0)
{
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_ERR, printk("%s, remap BDMA buffer failed)\n", __FUNCTION__));
return FALSE;
}
memcpy((void *)pu8BDMA_virt, (const void *)u32Src_off, u32Len);
Chip_Flush_Cache_Range(pu8BDMA_virt, u32Len);
// printk(" %s(0x%08X, %3d)\n", __FUNCTION__, (int)u32Dst_off, (int)u32Len);
/*BDMA */
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 0x00);
//Set source and destination path
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x2<<2)), 0X2B40); //MIU to FSP(0xB)
// Set source as MIU0
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x3<<2)), 0X0000); //MIU0 as channel0
// Set start address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x4<<2)), (pu8BDMA_bus & 0x0000FFFF));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x5<<2)), (pu8BDMA_bus >> 16));
// Set end address
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x6<<2)), 0x0);
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x7<<2)), 0x0);
// Set Size
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x8<<2)), (u32Len & 0x0000FFFF));
//WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x9<<2)), (u32Len >> 16));
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x0<<2)), 1); // Trigger
/*FSP*/
HAL_FSP_WriteBufs(0, SPI_CMD_WREN);
HAL_FSP_WriteBufs(1, SPI_CMD_PP);
HAL_FSP_WriteBufs(2, (MS_U8)((u32Dst_off>>0x10)&0xFF));
HAL_FSP_WriteBufs(3, (MS_U8)((u32Dst_off>>0x08)&0xFF));
HAL_FSP_WriteBufs(4, (MS_U8)((u32Dst_off>>0x00)&0xFF));
HAL_FSP_WriteBufs(5, 0x00); //dummy
HAL_FSP_WriteBufs(6, SPI_CMD_RDSR);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE0(1), REG_FSP_WBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE1(5), REG_FSP_WBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_SIZE, REG_FSP_WBF_SIZE2(1), REG_FSP_WBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE0(0), REG_FSP_RBF_SIZE0_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE1(0), REG_FSP_RBF_SIZE1_MASK);
FSP_WRITE_MASK(REG_FSP_RBF_SIZE, REG_FSP_RBF_SIZE2(1), REG_FSP_RBF_SIZE2_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_ENABLE, REG_FSP_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_NRESET, REG_FSP_RESET_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_INT, REG_FSP_INT_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_2NDCMD_ON, REG_FSP_2NDCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_3THCMD_ON, REG_FSP_3THCMD_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_3THCMD, REG_FSP_RDSR_MASK);
FSP_WRITE_MASK(REG_FSP_CTRL, REG_FSP_FSCHK_ON, REG_FSP_FSCHK_MASK);
//set FSP Outside replace
FSP_WRITE(REG_FSP_WBF_SIZE_OUTSIDE, u32Len&0x00FFFFFF);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_REPLACED(5), REG_FSP_WBF_REPLACED_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_MODE(1), REG_FSP_WBF_MODE_MASK);
FSP_WRITE_MASK(REG_FSP_WBF_OUTSIDE, REG_FSP_WBF_OUTSIDE_ENABLE, REG_FSP_WBF_OUTSIDE_ENABLE_MASK);
FSP_WRITE_MASK(REG_FSP_TRIGGER,REG_FSP_FIRE, REG_FSP_TRIGGER_MASK);
//check BDMA done
//SER_FLASH_TIME(time_st);
deadline = jiffies + MAX_READY_WAIT_JIFFIES;
do
{
//check done
u16data = READ_WORD(_hal_isp.u32BdmaBaseAddr + ((0x1)<<2));
if(u16data & 8)
{
//clear done
WRITE_WORD((_hal_isp.u32BdmaBaseAddr + (0x1<<2)), 8);
bRet = TRUE;
break;
}
//} while(!SER_FLASH_EXPIRE(time_st, SERFLASH_SAFETY_FACTOR));
} while (!time_after_eq(jiffies, deadline));
if(bRet == FALSE)
{
printk("Wait for BDMA Done fails!\n");
}
//check FSP done
bRet = _HAL_FSP_WaitDone();
if(!bRet)
{
printk("[FSP] BDMA2FSP FAIL Timeout !!!!\r\n");
}
//reset
FSP_WRITE(REG_FSP_WBF_SIZE_OUTSIDE, 0x0);
FSP_WRITE(REG_FSP_WBF_OUTSIDE, 0x0);
//CompareMemory(u32Src_off, u32Dst_off+0xF4000000, u32Len);
return bRet;
}
#endif
MS_BOOL HAL_FSP_Write_BDMA(MS_U32 u32Addr, MS_U32 u32Size, MS_U8 *pu8Data)
{
MS_BOOL bRet = FALSE;
MS_U16 u16Rem, u16WriteBytes;
//MS_U16 u16WriteBytes;
MS_U8 *u8Buf = pu8Data;
DEBUG_SER_FLASH(E_SERFLASH_DBGLV_DEBUG, printk("%s(0x%08X, %4d, %p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data));
//printk("%s(0x%08X, %d, 0x%p)\n", __FUNCTION__, (int)u32Addr, (int)u32Size, pu8Data);
#if 1
u16Rem = u32Addr % SERFLASH_PAGE_SIZE;
if (u16Rem)
{
u16WriteBytes = SERFLASH_PAGE_SIZE - u16Rem;
if (u32Size < u16WriteBytes)
{
u16WriteBytes = u32Size;
}
if(_hal_SERFLASH.u8MID == MID_MXIC)
{
bRet = _HAL_4PP_BDMAToFSP((MS_U32)u8Buf, u32Addr, u16WriteBytes);
}
else
{
bRet = _HAL_QP_BDMAToFSP((MS_U32)u8Buf, u32Addr, u16WriteBytes);
}
//bRet = _HAL_PP_BDMAToFSP((MS_U32)u8Buf, u32Addr, u16WriteBytes);
if ( bRet == TRUE )
{
u32Addr += u16WriteBytes;
u8Buf += u16WriteBytes;
u32Size -= u16WriteBytes;
}
else
{
goto HAL_SERFLASH_Write_return;
}
}
#endif
while(u32Size)
{
u16WriteBytes =(u32Size>SERFLASH_PAGE_SIZE) ? SERFLASH_PAGE_SIZE:u32Size;
//bRet = _HAL_PP_BDMAToFSP((MS_U32)u8Buf, u32Addr, u16WriteBytes);
if(_hal_SERFLASH.u8MID == MID_MXIC)
{
bRet = _HAL_4PP_BDMAToFSP((MS_U32)u8Buf, u32Addr, u16WriteBytes);
}
else
{
bRet = _HAL_QP_BDMAToFSP((MS_U32)u8Buf, u32Addr, u16WriteBytes);
}
if ( bRet == TRUE )
{
u32Addr += u16WriteBytes;
u8Buf += u16WriteBytes;
u32Size -= u16WriteBytes;
}
else
{
goto HAL_SERFLASH_Write_return;
}
}
HAL_SERFLASH_Write_return:
return bRet;
}
#endif