365 lines
12 KiB
C
365 lines
12 KiB
C
/***************************************************************************
|
|
*
|
|
* Copyright 2015-2019 BES.
|
|
* All rights reserved. All unpublished rights reserved.
|
|
*
|
|
* No part of this work may be used or reproduced in any form or by any
|
|
* means, or stored in a database or retrieval system, without prior written
|
|
* permission of BES.
|
|
*
|
|
* Use of this work is governed by a license granted by BES.
|
|
* This work contains confidential and proprietary information of
|
|
* BES. which is protected by copyright, trade secret,
|
|
* trademark and other intellectual property rights.
|
|
*
|
|
****************************************************************************/
|
|
#include "plat_types.h"
|
|
#include "norflash_drv.h"
|
|
#include "hal_norflaship.h"
|
|
#include "hal_timer.h"
|
|
#include "hal_trace.h"
|
|
#include "norflash_cfg.h"
|
|
#include "norflash_gd25q32c.h"
|
|
|
|
static void gd25q32c_write_status_s8_s15(uint8_t status)
|
|
{
|
|
norflash_write_reg(GD25Q32C_CMD_WRITE_STATUS_S8_S15, &status, 1);
|
|
}
|
|
|
|
static void gd25q32c_write_status_s0_s7(uint8_t status)
|
|
{
|
|
norflash_write_reg(GD25Q32C_CMD_WRITE_STATUS_S0_S7, &status, 1);
|
|
}
|
|
|
|
static int gd25q32c_write_status(enum DRV_NORFLASH_W_STATUS_T type, uint32_t param)
|
|
{
|
|
uint8_t status_s0_s7;
|
|
uint8_t status_s8_s15;
|
|
uint32_t bp_mask = 0;
|
|
union DRV_NORFLASH_SEC_REG_CFG_T cfg;
|
|
|
|
if (type != DRV_NORFLASH_W_STATUS_INIT && type != DRV_NORFLASH_W_STATUS_QE &&
|
|
type != DRV_NORFLASH_W_STATUS_LB && type != DRV_NORFLASH_W_STATUS_BP) {
|
|
return 1;
|
|
}
|
|
|
|
if (type == DRV_NORFLASH_W_STATUS_INIT) {
|
|
gd25q32c_write_status_s0_s7(param & 0xFF);
|
|
gd25q32c_write_status_s8_s15((param >> 8) & 0xFF);
|
|
return 0;
|
|
}
|
|
|
|
if (type == DRV_NORFLASH_W_STATUS_BP) {
|
|
bp_mask = norflash_get_block_protect_mask();
|
|
status_s0_s7 = norflash_read_status_s0_s7();
|
|
status_s0_s7 = (status_s0_s7 & ~bp_mask) | (param & bp_mask);
|
|
gd25q32c_write_status_s0_s7(status_s0_s7);
|
|
if ((bp_mask & ~0xFF) == 0) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
status_s8_s15 = norflash_read_status_s8_s15();
|
|
|
|
if (type == DRV_NORFLASH_W_STATUS_QE) {
|
|
if (param) {
|
|
status_s8_s15 |= GD25Q32C_QE_BIT_MASK;
|
|
} else {
|
|
status_s8_s15 &= ~(GD25Q32C_QE_BIT_MASK);
|
|
}
|
|
} else if (type == DRV_NORFLASH_W_STATUS_BP) {
|
|
param >>= 8;
|
|
bp_mask >>= 8;
|
|
status_s8_s15 = (status_s8_s15 & ~bp_mask) | (param & bp_mask);
|
|
} else if (type == DRV_NORFLASH_W_STATUS_LB) {
|
|
cfg = norflash_get_security_register_config();
|
|
if (!cfg.s.enabled) {
|
|
return 2;
|
|
}
|
|
if (cfg.s.lb == SEC_REG_LB_S11_S13) {
|
|
if (param >= 3) {
|
|
return 3;
|
|
}
|
|
status_s8_s15 |= (STATUS_S11_LB1_BIT_MASK << param);
|
|
} else if (cfg.s.lb == SEC_REG_LB_S10) {
|
|
status_s8_s15 |= STATUS_S10_LB_BIT_MASK;
|
|
} else {
|
|
return 4;
|
|
}
|
|
}
|
|
|
|
gd25q32c_write_status_s8_s15(status_s8_s15);
|
|
|
|
return 0;
|
|
}
|
|
|
|
// ----------------------
|
|
// GigaDevice
|
|
// ----------------------
|
|
|
|
const struct NORFLASH_CFG_T gd25q32c_cfg = {
|
|
.id = { 0xC8, 0x40, 0x16, },
|
|
.speed_ratio = {
|
|
.s = {
|
|
.std_read = SPEED_RATIO_5_EIGHTH,
|
|
.others = SPEED_RATIO_5_EIGHTH,
|
|
},
|
|
},
|
|
.crm_en_bits = (1 << 5) | (0 << 4),
|
|
.crm_dis_bits = 0,
|
|
.block_protect_mask = 0x407C,
|
|
.sec_reg_cfg = {
|
|
.s = {
|
|
.enabled = true,
|
|
.base = SEC_REG_BASE_0X1000,
|
|
.size = SEC_REG_SIZE_1024,
|
|
.offset = SEC_REG_OFFSET_0X1000,
|
|
.cnt = SEC_REG_CNT_3,
|
|
.pp = SEC_REG_PP_256,
|
|
.lb = SEC_REG_LB_S11_S13,
|
|
},
|
|
},
|
|
.page_size = GD25Q32C_PAGE_SIZE,
|
|
.sector_size = GD25Q32C_SECTOR_SIZE,
|
|
.block_size = GD25Q32C_BLOCK_SIZE,
|
|
.total_size = GD25Q32C_TOTAL_SIZE,
|
|
#ifdef FLASH_HPM
|
|
.max_speed = 120 * 1000 * 1000,
|
|
#else
|
|
// No high performance mode for gd25q32e
|
|
.max_speed = 104 * 1000 * 1000,
|
|
#endif
|
|
.mode = (HAL_NORFLASH_OP_MODE_STAND_SPI |
|
|
HAL_NORFLASH_OP_MODE_FAST_SPI |
|
|
HAL_NORFLASH_OP_MODE_DUAL_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_DUAL_IO |
|
|
HAL_NORFLASH_OP_MODE_QUAD_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_QUAD_IO |
|
|
#ifdef FLASH_HPM
|
|
HAL_NORFLASH_OP_MODE_HIGH_PERFORMANCE |
|
|
#endif
|
|
HAL_NORFLASH_OP_MODE_CONTINUOUS_READ |
|
|
HAL_NORFLASH_OP_MODE_READ_WRAP |
|
|
HAL_NORFLASH_OP_MODE_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_QUAD_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_SUSPEND),
|
|
.write_status = gd25q32c_write_status,
|
|
};
|
|
|
|
// ----------------------
|
|
// Puya
|
|
// ----------------------
|
|
|
|
const struct NORFLASH_CFG_T p25q128l_cfg = {
|
|
.id = { 0x85, 0x60, 0x18, },
|
|
.speed_ratio = {
|
|
.s = {
|
|
.std_read = SPEED_RATIO_3_EIGHTH,
|
|
.others = SPEED_RATIO_8_EIGHTH,
|
|
},
|
|
},
|
|
.crm_en_bits = (1 << 5) | (0 << 4),
|
|
.crm_dis_bits = 0,
|
|
.block_protect_mask = 0x407C,
|
|
.sec_reg_cfg = {
|
|
.s = {
|
|
.enabled = true,
|
|
.base = SEC_REG_BASE_0X1000,
|
|
.size = SEC_REG_SIZE_1024,
|
|
.offset = SEC_REG_OFFSET_0X1000,
|
|
.cnt = SEC_REG_CNT_3,
|
|
.pp = SEC_REG_PP_1024,
|
|
.lb = SEC_REG_LB_S11_S13,
|
|
},
|
|
},
|
|
.page_size = GD25Q32C_PAGE_SIZE,
|
|
.sector_size = GD25Q32C_SECTOR_SIZE,
|
|
.block_size = GD25Q32C_BLOCK_SIZE,
|
|
.total_size = P25Q128L_TOTAL_SIZE,
|
|
.mode = (HAL_NORFLASH_OP_MODE_STAND_SPI |
|
|
HAL_NORFLASH_OP_MODE_FAST_SPI |
|
|
HAL_NORFLASH_OP_MODE_DUAL_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_DUAL_IO |
|
|
HAL_NORFLASH_OP_MODE_QUAD_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_QUAD_IO |
|
|
HAL_NORFLASH_OP_MODE_CONTINUOUS_READ |
|
|
HAL_NORFLASH_OP_MODE_READ_WRAP |
|
|
HAL_NORFLASH_OP_MODE_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_QUAD_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_SUSPEND),
|
|
.max_speed = 85 * 1000 * 1000,
|
|
.write_status = gd25q32c_write_status,
|
|
};
|
|
|
|
const struct NORFLASH_CFG_T p25q64l_cfg = {
|
|
.id = { 0x85, 0x60, 0x17, },
|
|
.speed_ratio = {
|
|
.s = {
|
|
.std_read = SPEED_RATIO_3_EIGHTH,
|
|
.others = SPEED_RATIO_8_EIGHTH,
|
|
},
|
|
},
|
|
.crm_en_bits = (1 << 5) | (0 << 4),
|
|
.crm_dis_bits = 0,
|
|
.block_protect_mask = 0x407C,
|
|
.sec_reg_cfg = {
|
|
.s = {
|
|
.enabled = true,
|
|
.base = SEC_REG_BASE_0X1000,
|
|
.size = SEC_REG_SIZE_1024,
|
|
.offset = SEC_REG_OFFSET_0X1000,
|
|
.cnt = SEC_REG_CNT_3,
|
|
.pp = SEC_REG_PP_1024,
|
|
.lb = SEC_REG_LB_S11_S13,
|
|
},
|
|
},
|
|
.page_size = GD25Q32C_PAGE_SIZE,
|
|
.sector_size = GD25Q32C_SECTOR_SIZE,
|
|
.block_size = GD25Q32C_BLOCK_SIZE,
|
|
.total_size = P25Q64L_TOTAL_SIZE,
|
|
.mode = (HAL_NORFLASH_OP_MODE_STAND_SPI |
|
|
HAL_NORFLASH_OP_MODE_FAST_SPI |
|
|
HAL_NORFLASH_OP_MODE_DUAL_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_DUAL_IO |
|
|
HAL_NORFLASH_OP_MODE_QUAD_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_QUAD_IO |
|
|
HAL_NORFLASH_OP_MODE_CONTINUOUS_READ |
|
|
HAL_NORFLASH_OP_MODE_READ_WRAP |
|
|
HAL_NORFLASH_OP_MODE_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_DUAL_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_QUAD_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_SUSPEND),
|
|
.max_speed = 70 * 1000 * 1000, // P25Q64L=70M, P25Q64H=120M, P25Q64U=70M/120M
|
|
.write_status = gd25q32c_write_status,
|
|
};
|
|
|
|
const struct NORFLASH_CFG_T p25q32l_cfg = {
|
|
.id = { 0x85, 0x60, 0x16, },
|
|
.speed_ratio = {
|
|
.s = {
|
|
.std_read = SPEED_RATIO_4_EIGHTH,
|
|
.others = SPEED_RATIO_8_EIGHTH,
|
|
},
|
|
},
|
|
.crm_en_bits = (1 << 5) | (0 << 4),
|
|
.crm_dis_bits = 0,
|
|
.block_protect_mask = 0x407C,
|
|
.sec_reg_cfg = {
|
|
.s = {
|
|
.enabled = true,
|
|
.base = SEC_REG_BASE_0X1000,
|
|
.size = SEC_REG_SIZE_1024,
|
|
.offset = SEC_REG_OFFSET_0X1000,
|
|
.cnt = SEC_REG_CNT_3,
|
|
.pp = SEC_REG_PP_1024,
|
|
.lb = SEC_REG_LB_S11_S13,
|
|
},
|
|
},
|
|
.page_size = GD25Q32C_PAGE_SIZE,
|
|
.sector_size = GD25Q32C_SECTOR_SIZE,
|
|
.block_size = GD25Q32C_BLOCK_SIZE,
|
|
.total_size = P25Q32L_TOTAL_SIZE,
|
|
.mode = (HAL_NORFLASH_OP_MODE_STAND_SPI |
|
|
HAL_NORFLASH_OP_MODE_FAST_SPI |
|
|
HAL_NORFLASH_OP_MODE_DUAL_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_DUAL_IO |
|
|
HAL_NORFLASH_OP_MODE_QUAD_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_QUAD_IO |
|
|
HAL_NORFLASH_OP_MODE_CONTINUOUS_READ |
|
|
HAL_NORFLASH_OP_MODE_READ_WRAP |
|
|
HAL_NORFLASH_OP_MODE_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_DUAL_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_QUAD_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_SUSPEND),
|
|
.max_speed = 62 * 1000 * 1000, // P25Q32L=62.5M, P25Q32H=104M, P25Q32U=62.5M/104M
|
|
.write_status = gd25q32c_write_status,
|
|
};
|
|
|
|
// ----------------------
|
|
// Xinxin
|
|
// ----------------------
|
|
|
|
// Additionally, the device supports JEDEC standard manufacturer and device ID and SFDP Register,
|
|
// a 64-bit Unique Serial Number and three 256-bytes Security Registers.
|
|
const struct NORFLASH_CFG_T xm25qh16c_cfg = {
|
|
.id = { 0x20, 0x40, 0x15, },
|
|
.speed_ratio = {
|
|
.s = {
|
|
.std_read = SPEED_RATIO_3_EIGHTH,
|
|
.others = SPEED_RATIO_8_EIGHTH,
|
|
},
|
|
},
|
|
.crm_en_bits = (1 << 5) | (0 << 4),
|
|
.crm_dis_bits = 0,
|
|
.block_protect_mask = 0x407C,
|
|
.sec_reg_cfg = {
|
|
.s = {
|
|
.enabled = true,
|
|
.base = SEC_REG_BASE_0X1000,
|
|
.size = SEC_REG_SIZE_256,
|
|
.offset = SEC_REG_OFFSET_0X1000,
|
|
.cnt = SEC_REG_CNT_3,
|
|
.pp = SEC_REG_PP_256,
|
|
.lb = SEC_REG_LB_S11_S13,
|
|
},
|
|
},
|
|
.page_size = GD25Q32C_PAGE_SIZE,
|
|
.sector_size = GD25Q32C_SECTOR_SIZE,
|
|
.block_size = GD25Q32C_BLOCK_SIZE,
|
|
.total_size = XM25QH16C_TOTAL_SIZE,
|
|
.max_speed = 108 * 1000 * 1000,
|
|
.mode = (HAL_NORFLASH_OP_MODE_STAND_SPI |
|
|
HAL_NORFLASH_OP_MODE_FAST_SPI |
|
|
HAL_NORFLASH_OP_MODE_DUAL_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_DUAL_IO |
|
|
HAL_NORFLASH_OP_MODE_QUAD_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_QUAD_IO |
|
|
HAL_NORFLASH_OP_MODE_CONTINUOUS_READ |
|
|
HAL_NORFLASH_OP_MODE_READ_WRAP |
|
|
HAL_NORFLASH_OP_MODE_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_QUAD_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_SUSPEND),
|
|
.write_status = gd25q32c_write_status,
|
|
};
|
|
|
|
const struct NORFLASH_CFG_T xm25qh80b_cfg = {
|
|
.id = { 0x20, 0x40, 0x14, },
|
|
.speed_ratio = {
|
|
.s = {
|
|
.std_read = SPEED_RATIO_7_EIGHTH,
|
|
.others = SPEED_RATIO_8_EIGHTH,
|
|
},
|
|
},
|
|
.crm_en_bits = (1 << 5) | (0 << 4),
|
|
.crm_dis_bits = 0,
|
|
.block_protect_mask = 0x407C,
|
|
.sec_reg_cfg = {
|
|
.s = {
|
|
.enabled = true,
|
|
.base = SEC_REG_BASE_0X1000,
|
|
.size = SEC_REG_SIZE_256,
|
|
.offset = SEC_REG_OFFSET_0X1000,
|
|
.cnt = SEC_REG_CNT_3,
|
|
.pp = SEC_REG_PP_256,
|
|
.lb = SEC_REG_LB_S11_S13,
|
|
},
|
|
},
|
|
.page_size = GD25Q32C_PAGE_SIZE,
|
|
.sector_size = GD25Q32C_SECTOR_SIZE,
|
|
.block_size = GD25Q32C_BLOCK_SIZE,
|
|
.total_size = XM25QH80B_TOTAL_SIZE,
|
|
.max_speed = 60 * 1000 * 1000, // 104M (std_read=50M or 3/8) when HFM=1 (S20 or SR3-bit4)
|
|
.mode = (HAL_NORFLASH_OP_MODE_STAND_SPI |
|
|
HAL_NORFLASH_OP_MODE_FAST_SPI |
|
|
HAL_NORFLASH_OP_MODE_DUAL_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_DUAL_IO |
|
|
HAL_NORFLASH_OP_MODE_QUAD_OUTPUT |
|
|
HAL_NORFLASH_OP_MODE_QUAD_IO |
|
|
HAL_NORFLASH_OP_MODE_CONTINUOUS_READ |
|
|
HAL_NORFLASH_OP_MODE_READ_WRAP |
|
|
HAL_NORFLASH_OP_MODE_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_QUAD_PAGE_PROGRAM |
|
|
HAL_NORFLASH_OP_MODE_SUSPEND),
|
|
.write_status = gd25q32c_write_status,
|
|
};
|
|
|