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地址

Logo

DAMO开发者矩阵,由阿里巴巴达摩院和中国互联网协会联合发起,致力于探讨最前沿的技术趋势与应用成果,搭建高质量的交流与分享平台,推动技术创新与产业应用链接,围绕“人工智能与新型计算”构建开放共享的开发者生态。

更多推荐