第三十四章 TRNG——真随机数

第三十四章 TRNG——真随机数

目录

第三十四章 TRNG——真随机数

1 TRNG概述

1.1 简介

1.2 硬件结构

1.3 真随机与伪随机的区别

1.4 工作流程

2 应用场景

3 注意事项

4 程序设计

4.1 TRNG_IntTest例程

4.1.1 执行函数TRNG_Int()

4.1.2 配置嵌套向量中断控制器

4.1.3 使能TRNG输出功能

4.1.4 使能TRNG中断

4.1.5 设置伪随机数种子

4.1.6 启动TRNG硬件

4.1.7 中断服务函数

4.1.8 主函数main()

4.1.9 下载验证

4.2 TRNG_PollingTest例程

4.2.1 执行函数TRNG_Polling()

4.2.2 下载验证

5 总结


在当今数字化时代,信息安全已成为嵌入式系统设计中不可忽视的关键因素。从物联网设备的身份认证到加密通信的密钥生成,高质量的随机数是构建安全系统的基石。W55MH32内置的真随机数生成器(TRNG)模块为开发者提供了硬件级别的随机数解决方案。

本文将通过W55MH32 TRNG的工作原理、应用场景、程序设计等方面对真随机数进行讲解。

1 TRNG概述

1.1 简介

TRNGTrue Random Number Generator)即真随机数生成器,与伪随机数生成器(PRNG)不同,其随机性来源于物理噪声,而非确定性算法。W55MH32TRNG模块利用芯片内部的物理噪声源(如热噪声、时钟抖动等)生成不可预测的随机数,适用于加密、安全认证、随机密钥生成等对随机性要求高的场景。

1.2 硬件结构

W55MH32TRNG模块主要由以下部分组成:

  • 噪声源:通常基于MOS管的热噪声或环形振荡器的抖动,产生原始随机信号。
  • 放大与整形电路:增强噪声信号并转换为可处理的数字信号。
  • 采样电路:对噪声信号进行采样,生成原始随机比特流。
  • 熵累积器:收集采样得到的熵,积累足够的随机性。
  • 随机数生成器:将累积的熵转换为可用的随机数(如32位整数)。
  • 硬件测试与校准:确保噪声源正常工作,必要时进行校准。

1.3 真随机与伪随机的区别

随机数在计算机科学中主要分为两类——真随机和伪随机,其对比如下所示:

特性

真随机数(TRNG)

伪随机数(PRNG)

随机性来源

物理噪声(热噪声、时钟抖动)

数学算法(如线性同余法)

可预测性

不可预测(基于物理现象)

理论上可预测(已知种子和算法)

周期性

无周期性

存在周期性(周期长度取决于算法)

硬件依赖

需要特定硬件模块

纯软件实现

应用场景

加密、安全认证、密码学

模拟、游戏、非安全场景随机数

1.4 工作流程

TRNG的工作流程可以分为以下几个关键步骤:

  1. 噪声采集阶段:多个环形振荡器同时工作,其输出频率的微小差异被捕获并放大。这些差异作为原始随机信号输入到后续处理电路。
  2. 数字化阶段:模拟的随机信号被采样并转换为数字比特流。采样过程通常使用高速时钟进行,确保捕获到足够的随机信息。
  3. 随机性增强阶段:原始数字信号可能存在统计偏差,需要通过算法进行后处理。W55MH32TRNG使用一种称为"异或树"的结构,将多个环形振荡器的输出进行异或运算,增强随机性并消除可能的偏差。
  4. 质量检测阶段:生成的随机数经过实时统计测试,确保其符合随机性标准。W55MH32TRNG实现了两种主要测试:
    1. 单比特频率测试:确保01的出现概率接近50%
    2. 游程测试:检测连续相同比特的长度是否符合随机分布
  5. 输出阶段:通过质量检测的随机数被存储在数据寄存器中,供CPU读取使用。当检测到质量问题时,TRNG会自动禁用输出并设置错误标志。

2 应用场景

  • 加密密钥生成:为AESRSA等加密算法生成初始密钥。
  • 安全认证:生成随机挑战值(Challenge)用于身份验证。
  • 随机数种子:为PRNG提供高质量的初始种子。
  • 安全协议:如TLS握手、VPN密钥协商等场景。
  • 游戏:需要高随机性的虚拟骰子、卡牌等应用。

3 注意事项

  1. 低功耗模式:在睡眠或停机模式下,TRNG可能停止工作,需重新初始化。
  2. 噪声源依赖性:温度、电压等环境因素可能影响噪声源强度,导致随机性波动。
  3. 验证测试:在关键应用中,建议对生成的随机数进行离线测试(如使用NIST测试工具)。
  4. 多线程安全:在RTOS环境中,访问TRNG时需加锁,避免竞争条件。

4 程序设计

4.1 TRNG_IntTest例程

TRNG_IntTest例程主要实现了基于W55MH32芯片的真随机数生成器(TRNG)中断测试功能。以下是实现过程和结果验证:

4.1.1 执行函数TRNG_Int()

TRNG的中断配置、TRNG输出使能、中断使能、随机种子设置和启动TRNG硬件主要在TRNG_Int()函数中实现:

void TRNG_Int(void)
{
    NVIC_Configuration();
    TRNG_Out(ENABLE);
    TRNG_ITConfig(ENABLE);
    TRNG_SetPseudoRandom(0X12345560);
    TRNG_Start();
}

4.1.2 配置嵌套向量中断控制器

NVIC_Configuration()为中断配置函数:

void NVIC_Configuration(void)
{
    NVIC_InitTypeDef NVIC_InitStructure;
 
    NVIC_SetPriorityGrouping(NVIC_PriorityGroup_1);
 
    NVIC_InitStructure.NVIC_IRQChannel                   = TRNG_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority        = 1;
    NVIC_InitStructure.NVIC_IRQChannelCmd                = ENABLE;
    NVIC_Init(&NVIC_InitStructure);
}
 

该函数主要进行了如下配置:

  • NVIC_PriorityGroup_1:表示使用优先级分组模式1,即:
    • 抢占优先级占1位(0-1
    • 子优先级占3位(0-7
  • 中断参数配置:
    • 中断源:TRNG_IRQn(真随机数生成器中断)
    • 双重优先级机制:抢占优先级1可以打断更低抢占优先级的中断
    • 子优先级1:相同抢占优先级时决定响应顺序
    • 使能TRNG硬件中断通道
  • NVIC初始化

4.1.3 使能TRNG输出功能

TRNG_Out()函数是TRNG初始化过程中的关键步骤,通过控制时钟来启用或禁用TRNG模块,从而间接控制随机数的生成:

void TRNG_Out(FunctionalState NewState)
{
	if (NewState != DISABLE)
	{
		RCC->RCC_SYSCFG_CONFIG = 0x01;
		SYSCFG->SYSCFG_LOCK = 0xCDED3526;
		SYSCFG->SSC_CLK_EN |= TRNG_RNG_ENABLE;
	}
	else
	{
		RCC->RCC_SYSCFG_CONFIG = 0x00;
		SYSCFG->SSC_CLK_EN &= ~TRNG_RNG_ENABLE;
	}
}

当使能TRNG输出时,首先配置SYSCFG时钟域(RCC_SYSCFG_CONFIG=0x01),然后向锁定寄存器写入特定密钥(SYSCFG_LOCK=0xCDED3526)以解锁SYSCFG寄存器,最后设置安全系统时钟使能寄存器(SSC_CLK_EN)的对应位以开启TRNG时钟。

当失能输出时,则执行相反操作:清除时钟域配置并禁用TRNG时钟。这种设计通过时钟控制间接管理TRNG模块,是低功耗和安全设计的常见做法,其中锁定机制可防止意外修改寄存器。

4.1.4 使能TRNG中断

TRNG_ITConfig()是控制TRNG中断功能的函数,主要用于启用或禁用TRNG模块的中断机制:

void TRNG_ITConfig(FunctionalState NewState)
{
	if (NewState != DISABLE)
	{
		TRNG->RNG_CSR |= TRNG_RNG_CSR_INTP_EN_Mask;
	}
	else
	{
		TRNG->RNG_CSR &= ~TRNG_RNG_CSR_INTP_EN_Mask;
	}
}

当传入参数为ENABLE使能中断时,函数通过位操作(|=)将 TRNG 控制状态寄存器(RNG_CSR)中的中断使能位(TRNG_RNG_CSR_INTP_EN_Mask)置 1,允许 TRNG 在随机数生成完成或检测到错误时触发中断。

当传入为DISABLE失能中断时,则通过位操作(&= ~)清除该位,禁用中断功能。

4.1.5 设置伪随机数种子

通过TRNG_SetPseudoRandom()函数设置伪随机种子,主要用于增强随机数生成的质量或实现特定应用场景:

void TRNG_SetPseudoRandom(uint32_t TRNG_PseudoRandom)
{
	TRNG->RNG_PN = TRNG_PseudoRandom;
}

该函数将传入的32位种子值写入RNG_PN寄存器。

4.1.6 启动TRNG硬件

TRNG_Start()为启动TRNG(真随机数生成器)的函数,通过配置特定寄存器来激活TRNG模块:

void TRNG_Start(void)
{
	TRNG->RNG_AMA &= ~TRNG_RNG_AMA_PD_ALL_Mask;
	TRNG->RNG_CSR &= ~TRNG_RNG_CSR_S128_TRNG0_Mask;
}

函数首先通过TRNG->RNG_AMA &= ~TRNG_RNG_AMA_PD_ALL_Mask来清除电源关闭位,将TRNG的模拟电路从低功耗模式唤醒,激活环形振荡器等物理随机源;接着通过TRNG->RNG_CSR &= ~TRNG_RNG_CSR_S128_TRNG0_Mask清除128位采样模式标志。这两个步骤共同完成TRNG的启动初始化,为后续生成随机数做准备。

4.1.7 中断服务函数

RNG_IRQHandler()是处理TRNG中断的函数,主要用于响应随机数生成完成事件和检测到的安全攻击事件:

void RNG_IRQHandler(void)
{
    if (TRNG_GetITStatus(TRNG_IT_RNG0_S128) == SET)
    {
        printf("Rng : %08X %08X %08X %08X \r\n", TRNG->RNG_DATA, TRNG->RNG_DATA, TRNG->RNG_DATA, TRNG->RNG_DATA);
        TRNG_ClearITPendingBit(TRNG_IT_RNG0_S128);
    }
    if (TRNG_GetITStatus(TRNG_IT_RNG0_ATTACK) == SET)
    {
        TRNG_ClearITPendingBit(TRNG_IT_RNG0_ATTACK);
    }
    NVIC_ClearPendingIRQ(TRNG_IRQn);
}

128位随机数生成完成(TRNG_IT_RNG0_S128标志置位)时,函数通过连续4次读取RNG_DATA寄存器获取128位随机数(432位值)并打印输出,随后清除中断标志;当检测到安全攻击(TRNG_IT_RNG0_ATTACK标志置位)时,清除中断标志。最后,函数清除NVIC中断挂起标志以允许后续中断。

4.1.8 主函数main()

主函数main()如下:

int main(void)
{
    RCC_ClocksTypeDef clocks;
 
    delay_init();
 
    UART_Configuration(115200);
    printf("TRNG Int Out Test.\n");
    RCC_GetClocksFreq(&clocks);
 
    printf("SYSCLK: %3.1fMhz, HCLK: %3.1fMhz, PCLK1: %3.1fMhz, PCLK2: %3.1fMhz, ADCCLK: %3.1fMhz\n",
           (float)clocks.SYSCLK_Frequency / 1000000, (float)clocks.HCLK_Frequency / 1000000,
           (float)clocks.PCLK1_Frequency / 1000000, (float)clocks.PCLK2_Frequency / 1000000, (float)clocks.ADCCLK_Frequency / 1000000);
 
 
    TRNG_Int();
    while (1);
}

程序首先初始化延时函数并配置串口通信(波特率115200),随后打印系统各时钟频率(SYSCLKHCLKPCLK1/2ADCCLK)。接着调用TRNG_Int()函数初始化TRNG,该函数会配置NVIC中断、使能TRNG输出、设置中断模式和伪随机种子并启动TRNG。最后程序进入无限循环,等待TRNG生成128位随机数时触发中断,由中断处理函数RNG_IRQHandler()读取并打印随机数。

4.1.9 下载验证

程序下载运行后,首先打印了示例名称和系统各时钟的频率,然后便不停地打印生成的随机数:

4.2 TRNG_PollingTest例程

TRNG_PollingTest例程为TRNG的轮询模式操作,与之前的中断模式不同,此函数通过主动查询方式获取随机数。

4.2.1 执行函数TRNG_Polling()

TRNG的输出使能、启动和查询打印的功能主要在TRNG_Polling()函数中实现:

void TRNG_Polling(void)
{
    uint32_t Buf[4];
    TRNG_Out(ENABLE);
    TRNG_Start();
    while (1)
    {
        if (0 == TRNG_Get(Buf))
        {
            printf("Rng : %08X %08X %08X %08X \r\n", Buf[0], Buf[1], Buf[2], Buf[3]);
            TRNG_ClearITPendingBit(TRNG_IT_RNG0_S128);
        }
    }
}

首先定义了一个4元素的32位整数数组作为缓冲区,然后调用TRNG_Out(ENABLE)使能TRNG输出,并通过TRNG_Start()启动TRNG生成随机数。随后进入无限循环,不断调用TRNG_Get()函数尝试获取随机数,当该函数返回0时表示获取成功,此时将432位随机数打印输出,并清除中断标志位(即使在轮询模式下也需清除)。

TRNG_Start()TRNG_Get()在上节内容已经讲解,且主函数仅示例名称和执行函数有所修改,其他保持一致,这里就不再赘述。

4.2.2 下载验证

程序下载运行后,首先打印了示例名称和各系统时钟频率,接着便不断打印生成的随机数:

5 总结

W55MH32TRNG模块为嵌入式系统提供了硬件级别的真随机数生成能力,是构建安全系统的重要组成部分。通过合理配置和使用,可以生成高质量的随机数,满足加密、认证等安全敏感应用的需求。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值