返回列表 发布新帖
查看: 55|回复: 7

[技术交流] APM32E030 温度传感器不同频率周期性采样问题

27

主题

10

回帖

247

积分

版主

发表于 2025-11-25 19:05:34 来自手机 | 查看全部 |阅读模式
APM32E030 内置温度传感器,尽管其精度不及外置传感器,但可满足温度趋势判断的使用需求。目前我们的项目中已启用 ADC 采样功能,其中电压、电流的采样周期较高(约 5KHz),而温度采样周期较低(仅 2Hz)。针对这种不同频率的周期性采样需求,该如何实现呢?

27

主题

10

回帖

247

积分

版主

 楼主| 发表于 2025-11-25 19:08:19 | 查看全部
APM32E030多频率ADC采样实现方案
针对APM32E030内置温度传感器与外部电压电流采样的不同频率需求,我提供几种实用的实现方案:

方案一:定时器触发 + ADC扫描模式(推荐)
这是最高效的硬件解决方案,利用APM32E030的定时器和ADC扫描功能。
  1. // 配置定义
  2. #define VOLTAGE_CHANNEL    0   // 电压采样通道
  3. #define CURRENT_CHANNEL    1   // 电流采样通道  
  4. #define TEMP_CHANNEL       16  // 内置温度传感器通道
  5. #define SAMPLE_BUFFER_SIZE 100

  6. // 采样缓冲区
  7. uint16_t voltage_buffer[SAMPLE_BUFFER_SIZE];
  8. uint16_t current_buffer[SAMPLE_BUFFER_SIZE];
  9. uint16_t temp_buffer[10];  // 温度缓冲区较小
  10. volatile uint32_t voltage_current_count = 0;
  11. volatile uint32_t temp_count = 0;
  12. volatile uint8_t temp_sampling_flag = 0;

  13. // 定时器配置
  14. void TIM_Config(void)
  15. {
  16.     TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  17.    
  18.     // 5kHz定时器 - 用于电压电流采样
  19.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
  20.     TIM_TimeBaseStructure.TIM_Period = (SystemCoreClock / 5000) - 1;  // 5kHz
  21.     TIM_TimeBaseStructure.TIM_Prescaler = 0;
  22.     TIM_TimeBaseStructure.TIM_ClockDivision = 0;
  23.     TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  24.     TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
  25.    
  26.     // 2Hz定时器 - 用于温度采样
  27.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE);
  28.     TIM_TimeBaseStructure.TIM_Period = (SystemCoreClock / 2) - 1;  // 2Hz
  29.     TIM_TimeBaseStructure.TIM_Prescaler = 0;
  30.     TIM_TimeBaseInit(TIM6, &TIM_TimeBaseStructure);
  31.    
  32.     // 启用定时器中断
  33.     TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);
  34.     TIM_ITConfig(TIM6, TIM_IT_Update, ENABLE);
  35.    
  36.     NVIC_EnableIRQ(TIM3_IRQn);
  37.     NVIC_EnableIRQ(TIM6_IRQn);
  38.    
  39.     TIM_Cmd(TIM3, ENABLE);
  40.     TIM_Cmd(TIM6, ENABLE);
  41. }

  42. // ADC配置
  43. void ADC_Config(void)
  44. {
  45.     ADC_InitTypeDef ADC_InitStructure;
  46.     GPIO_InitTypeDef GPIO_InitStructure;
  47.    
  48.     // 启用ADC时钟
  49.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
  50.    
  51.     // 配置ADC通道对应的GPIO(外部传感器)
  52.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE);
  53.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;  // PA0, PA1
  54.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
  55.     GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
  56.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  57.    
  58.     // ADC基本配置
  59.     ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
  60.     ADC_InitStructure.ADC_ScanConvMode = DISABLE;  // 单通道模式
  61.     ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;  // 单次转换
  62.     ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;
  63.     ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
  64.     ADC_InitStructure.ADC_NbrOfConversion = 1;
  65.     ADC_Init(ADC1, &ADC_InitStructure);
  66.    
  67.     // 启用温度传感器
  68.     ADC_TempSensorVrefintCmd(ENABLE);
  69.    
  70.     // 启用ADC
  71.     ADC_Cmd(ADC1, ENABLE);
  72.    
  73.     // 等待ADC稳定
  74.     Delay_ms(1);
  75. }

  76. // 定时器3中断 - 5kHz电压电流采样
  77. void TIM3_IRQHandler(void)
  78. {
  79.     if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)
  80.     {
  81.         TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
  82.         
  83.         // 交替采样电压和电流
  84.         static uint8_t vc_alternate = 0;
  85.         
  86.         if(vc_alternate == 0)
  87.         {
  88.             // 采样电压
  89.             ADC_RegularChannelConfig(ADC1, VOLTAGE_CHANNEL, 1, ADC_SampleTime_239_5Cycles);
  90.             ADC_StartConversion(ADC1);
  91.             while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);
  92.             voltage_buffer[voltage_current_count % SAMPLE_BUFFER_SIZE] = ADC_GetConversionValue(ADC1);
  93.         }
  94.         else
  95.         {
  96.             // 采样电流  
  97.             ADC_RegularChannelConfig(ADC1, CURRENT_CHANNEL, 1, ADC_SampleTime_239_5Cycles);
  98.             ADC_StartConversion(ADC1);
  99.             while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);
  100.             current_buffer[voltage_current_count % SAMPLE_BUFFER_SIZE] = ADC_GetConversionValue(ADC1);
  101.             
  102.             voltage_current_count++;
  103.         }
  104.         
  105.         vc_alternate = !vc_alternate;
  106.     }
  107. }

  108. // 定时器6中断 - 2Hz温度采样
  109. void TIM6_IRQHandler(void)
  110. {
  111.     if (TIM_GetITStatus(TIM6, TIM_IT_Update) != RESET)
  112.     {
  113.         TIM_ClearITPendingBit(TIM6, TIM_IT_Update);
  114.         temp_sampling_flag = 1;  // 设置标志,在主循环中处理
  115.     }
  116. }

  117. // 主循环中的温度采样处理
  118. void ProcessTemperatureSampling(void)
  119. {
  120.     if(temp_sampling_flag)
  121.     {
  122.         temp_sampling_flag = 0;
  123.         
  124.         // 采样温度
  125.         ADC_RegularChannelConfig(ADC1, TEMP_CHANNEL, 1, ADC_SampleTime_239_5Cycles);
  126.         ADC_StartConversion(ADC1);
  127.         while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);
  128.         uint16_t temp_raw = ADC_GetConversionValue(ADC1);
  129.         
  130.         // 温度转换(需要根据芯片手册校准)
  131.         float temperature = ConvertToTemperature(temp_raw);
  132.         
  133.         temp_buffer[temp_count % 10] = temp_raw;
  134.         temp_count++;
  135.         
  136.         // 温度数据处理
  137.         ProcessTemperatureData(temperature);
  138.     }
  139. }
复制代码

27

主题

10

回帖

247

积分

版主

 楼主| 发表于 2025-11-25 19:09:04 | 查看全部
方案二:DMA + 定时器触发(高效率方案)
如果APM32E030支持,使用DMA可以进一步提高效率:
  1. // DMA配置(如果芯片支持)
  2. void ADC_DMA_Config(void)
  3. {
  4.     DMA_InitTypeDef DMA_InitStructure;
  5.    
  6.     // 启用DMA时钟
  7.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
  8.    
  9.     // 配置DMA
  10.     DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
  11.     DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)voltage_buffer;
  12.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
  13.     DMA_InitStructure.DMA_BufferSize = SAMPLE_BUFFER_SIZE;
  14.     DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  15.     DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  16.     DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
  17.     DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
  18.     DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
  19.     DMA_InitStructure.DMA_Priority = DMA_Priority_High;
  20.     DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
  21.     DMA_Init(DMA1_Channel1, &DMA_InitStructure);
  22.    
  23.     // 启用DMA
  24.     DMA_Cmd(DMA1_Channel1, ENABLE);
  25.    
  26.     // ADC配置为DMA模式
  27.     ADC_DMACmd(ADC1, ENABLE);
  28. }
复制代码

27

主题

10

回帖

247

积分

版主

 楼主| 发表于 2025-11-25 19:10:08 | 查看全部
方案三:软件定时方案(简单实用)
如果硬件资源有限,可以使用单个定时器配合软件计数:


  1. // 软件计数方案
  2. #define VC_SAMPLE_INTERVAL  200   // 5kHz = 200us
  3. #define TEMP_SAMPLE_INTERVAL 500000  // 2Hz = 500ms

  4. void TIM_Software_Config(void)
  5. {
  6.     // 配置一个1kHz的基础定时器
  7.     TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  8.    
  9.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
  10.     TIM_TimeBaseStructure.TIM_Period = (SystemCoreClock / 1000) - 1;  // 1kHz
  11.     TIM_TimeBaseStructure.TIM_Prescaler = 0;
  12.     TIM_TimeBaseStructure.TIM_ClockDivision = 0;
  13.     TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  14.     TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
  15.    
  16.     TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);
  17.     NVIC_EnableIRQ(TIM3_IRQn);
  18.     TIM_Cmd(TIM3, ENABLE);
  19. }

  20. void TIM3_IRQHandler(void)
  21. {
  22.     static uint32_t vc_counter = 0;
  23.     static uint32_t temp_counter = 0;
  24.    
  25.     if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)
  26.     {
  27.         TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
  28.         
  29.         // 电压电流采样计数
  30.         vc_counter++;
  31.         if(vc_counter >= VC_SAMPLE_INTERVAL)
  32.         {
  33.             vc_counter = 0;
  34.             StartVoltageCurrentSampling();
  35.         }
  36.         
  37.         // 温度采样计数
  38.         temp_counter++;
  39.         if(temp_counter >= TEMP_SAMPLE_INTERVAL)
  40.         {
  41.             temp_counter = 0;
  42.             temp_sampling_flag = 1;
  43.         }
  44.     }
  45. }
复制代码


27

主题

10

回帖

247

积分

版主

 楼主| 发表于 2025-11-25 19:10:56 | 查看全部
温度传感器使用注意事项
  1. // 温度传感器校准和转换
  2. float ConvertToTemperature(uint16_t adc_value)
  3. {
  4.     // APM32E030温度传感器典型参数(参考数据手册)
  5.     const float V25 = 0.76f;    // 25°C时的电压(典型值)
  6.     const float Avg_Slope = 2.5f; // mV/°C(典型值)
  7.     const float VREF = 3.3f;    // 参考电压
  8.    
  9.     // 将ADC值转换为电压
  10.     float voltage = (adc_value * VREF) / 4095.0f;
  11.    
  12.     // 计算温度
  13.     float temperature = ((V25 - voltage) * 1000.0f / Avg_Slope) + 25.0f;
  14.    
  15.     return temperature;
  16. }
复制代码

27

主题

10

回帖

247

积分

版主

 楼主| 发表于 2025-11-25 19:14:11 | 查看全部
推荐方案
推荐使用方案一,理由如下:

硬件独立性:两个定时器独立工作,互不干扰

精确计时:硬件定时器确保采样间隔准确

低CPU开销:中断处理简单高效

易于扩展:可以方便地调整采样频率

关键优化建议
中断优先级:设置电压电流采样的中断优先级高于温度采样

缓冲区管理:使用循环缓冲区防止数据丢失

温度滤波:对温度数据进行滑动平均滤波,提高趋势判断的可靠性

错误处理:添加ADC采样超时检测等容错机制

这种架构既能满足高频电压电流采样的实时性要求,又能实现低频温度采样的趋势监测,非常适合您的应用场景。

7

主题

6

回帖

63

积分

注册会员

发表于 2025-11-25 20:10:08 来自手机 | 查看全部
看了半天,的确方案一可行,楼主很厉害啊。

2

主题

7

回帖

28

积分

新手上路

发表于 2025-11-25 20:47:24 来自手机 | 查看全部
确实有点东西。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

Microfelt微感论坛-传感器技术交流 © 2001-2025 Discuz! Team. Powered by Discuz! W1.5 粤ICP备18044996号-3|81f01e9418981fd496123c701618b320
在本版发帖
论坛管理
论坛管理 返回顶部
快速回复 返回顶部 返回列表