pinebuds/platform/drivers/norflash/norflash_gd25q32c.c

366 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,
};