BLE芯片DA145XX系列:OTP读写、OTP Header
Dialog的BLE芯片基本都有OTP(One-Time-Programmable)存储器,即一次性写入的存储器,一旦写入之后,就无法修改。如DA14531,有32KB的OTP存储器。主要用来写入固化程序和保存一些重要的出厂参数。下面主要说明如何读写OTP存储器,以及OTPHeader的功能。
1、使用官方提供的toolbox来使用可视化界面写入OTP
可以对OTPImage(32KB) 内的逐字节信息查看,以及OTP Header(dialog官方设计的保存部分参数的区域,在OTP的结尾部分)信息
连接上芯片后,简单配置即可写入和读取OTP Image和OTPHeader

2、程序中读写OTP
Dialog提供的SDK中可以找到OTP读写的相关函数
在hw_otpc_531.h中
/** @brief OTP memory base address */
#define MEMORY_OTP_BASE (0x07F80000)
#if defined (__DA14531__)
#define MEMORY_OTP_END (0x07F88000)
#else
#define MEMORY_OTP_END (0x07F90000)
#endif
/** @brief OTP memory size (64 KB in DA14585/586 - 32KB in DA14531) */
#define MEMORY_OTP_SIZE (MEMORY_OTP_END - MEMORY_OTP_BASE)
/// ARM is a 32-bit CPU
#define CPU_WORD_SIZE (4)
/// ARM is little endian
#define CPU_LE (1)
可以看到DA14531的OTP起始地址为0x07F80000,结束地址为0x07F88000
每次最小的读写单位为4字节,小端读写
SDK提供的库函数:
读取4个字节:
/**
* @brief Read a word from OTP
* @param[in] cell_offset The offset of cell to be read in 32 bit words
* @return otp cell value
*/
uint32_t hw_otpc_word_read(uint32_t cell_offset)
{
ASSERT_CELL_OFFSET_VALID(cell_offset);
ASSERT_WARNING_OTP_CLK_ENABLED;
hw_otpc_enter_mode(HW_OTPC_MODE_READ);
return *(uint32_t *)(MEMORY_OTP_BASE + HW_OTP_CELL_SIZE * cell_offset);
}
写入4个字节:
/**
* @brief Program OTP and verify.
* @param[in] wdata The data to be programmed
* @param[in] cell_offset The offset of cell to be written in 32 bit words
* @return cell true if success or false in fail
*/
bool hw_otpc_word_prog_and_verify(uint32_t wdata, uint32_t cell_offset)
{
ASSERT_CELL_OFFSET_VALID(cell_offset);
ASSERT_WARNING_OTP_CLK_ENABLED;
hw_otpc_word_prog(wdata, cell_offset);
hw_otpc_enter_mode(HW_OTPC_MODE_PVFY);
if (wdata != *(uint32_t *)(MEMORY_OTP_BASE + HW_OTP_CELL_SIZE * cell_offset))
{
return false;
}
hw_otpc_enter_mode(HW_OTPC_MODE_RINI);
if (wdata != *(uint32_t *)(MEMORY_OTP_BASE + HW_OTP_CELL_SIZE * cell_offset))
{
return false;
}
return true;
}
比如我们要读取OTP最后1KB起始的位置,查到没有写入的地方开始写入数据:
没写入的数据读取到的都是0xFFFFFFFF,否则就是写入过,无法再次写入。
//参数区起始地址(最后1K),DA14531共32K
#define PARA_START_ADDR 31*256
//从OTP最后1K开始,逐4字节读,读到0xFFFFFFFFF为止,写入
void para_set_otp(void)
{
uint8_t i;
uint32_t addr_H,addr_L;
// Initialize OTP controller
hw_otpc_init();
hw_otpc_enter_mode(HW_OTPC_MODE_READ);
for(i=0;i<64;i++)
{
addr_H = hw_otpc_word_read(BIO_MAC_START_ADDR+2*i);
addr_L = hw_otpc_word_read(BIO_MAC_START_ADDR+2*i+1);
//找到未写入的块,写入数据
if((addr_H == 0xFFFFFFFF) && (addr_L == 0xFFFFFFFF))
{
addr_H = 0x12345678;
addr_L = 0x87654321;
//写入数据
hw_otpc_word_prog_and_verify(addr_H,PARA_START_ADDR+2*i);
hw_otpc_word_prog_and_verify(addr_L,PARA_START_ADDR+2*i+1);
break;
}
}
//关闭OTP
hw_otpc_disable();
}
需要注意官方使用的OTPheader在最后1K的末尾,不能在此处写入数据,否则部分配置功能会受到影响!读取OTPImage可以看到,DA14531的OTPHeader起始地址在0x0787ED0,总计304字节。

3、OTPHeader的作用
(1)Configuration Script,共240字节,包含预留部分。
用于配置部分寄存器参数,通过SDK启动后上电芯片会读取该区域内容,将寄存器加载为该区域内的值。
比如此处,就配置了0x50000024地址的寄存器(CLK_RC32M_REG),值为0x3D6
(2)Application Flag1、Application Flag2
全部选择YES写入后,芯片上电后会优先启用OTP内的程序写入RAM运行.
(3)Boot specific config
配置是否优先从SPI启动。
(4)Boot specific port mapping
配置boot中FLASH启动的SPI引脚映射关系。
(5)Bluetooth Device Address
蓝牙MAC地址,通过SDK做配置可以启用该地址内的值作为芯片固定的蓝牙MAC地址
DAMO开发者矩阵,由阿里巴巴达摩院和中国互联网协会联合发起,致力于探讨最前沿的技术趋势与应用成果,搭建高质量的交流与分享平台,推动技术创新与产业应用链接,围绕“人工智能与新型计算”构建开放共享的开发者生态。
更多推荐


所有评论(0)