从 PWM 到正弦波:在 Proteus 里用 STM32F103 的 DAC 或 PWM+滤波生成波形全记录
从 PWM 到正弦波:STM32F103 波形生成的双路径实战解析
在嵌入式开发中,信号生成是基础却至关重要的技能。许多开发者熟悉基础的PWM输出,但当需求升级到更复杂的模拟信号(如正弦波)时,往往面临选择:是用芯片内置的DAC直接输出,还是通过PWM加滤波电路间接实现?本文将深入探讨这两种方法的实现细节、优劣对比以及在Proteus环境下的仿真验证技巧。
1. 硬件基础与方案选型
STM32F103系列微控制器作为经典的Cortex-M3内核芯片,其外设资源丰富但配置灵活度较高。在波形生成场景中,我们需要先明确几个关键硬件特性:
PWM模块:STM32F103通常配备多个高级定时器(如TIM1)和通用定时器(如TIM3/TIM4),支持PWM生成。以TIM3为例,其特性包括:
- 16位自动重装载寄存器
- 可编程预分频器
- 4个独立通道(CH1-CH4)
- 支持PWM模式1和模式2
DAC模块:并非所有STM32F103子型号都配备DAC。以STM32F103C8T6为例:
- 无内置DAC
- 需使用STM32F103RET6等型号才具备12位双通道DAC
- 最大转换速率约1MHz
方案对比表:
| 特性 | PWM+滤波方案 | 直接DAC方案 |
|---|---|---|
| 硬件要求 | 任何STM32F103型号 | 需特定型号支持DAC |
| 输出精度 | 依赖滤波电路设计 | 固定12位分辨率 |
| 频率上限 | 受限于PWM频率和滤波电路 | 约1MHz |
| 谐波失真 | 较高,需复杂滤波 | 较低 |
| 外围电路复杂度 | 需要RC/LPF设计 | 可直接输出 |
| 动态调整便利性 | 需同步调整PWM和滤波参数 | 直接修改DAC寄存器 |
提示:在资源受限或芯片型号固定的场景下,PWM+滤波方案更具普适性;当需要高精度波形且芯片支持时,DAC方案更优。
2. PWM生成正弦波(SPWM)全实现
2.1 正弦波表生成与PWM调制
正弦脉宽调制(SPWM)的核心思想是通过改变PWM占空比来逼近正弦波形。具体实现分为三个关键步骤:
- 正弦波表生成:预先计算一个周期内的采样值
#define SAMPLE_POINTS 256 uint16_t sinTable[SAMPLE_POINTS]; void generateSinTable(void) { for(int i=0; i<SAMPLE_POINTS; i++) { float angle = 2 * 3.1415926 * i / SAMPLE_POINTS; sinTable[i] = (uint16_t)((sin(angle) + 1) * (PWM_MAX/2)); } }- 定时器配置(以TIM3为例):
void TIM3_Configuration(uint16_t arr) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStruct; TIM_OCInitTypeDef TIM_OCInitStructure; // 时基单元配置 TIM_TimeBaseStruct.TIM_Period = arr - 1; TIM_TimeBaseStruct.TIM_Prescaler = 72 - 1; // 1MHz计数频率 TIM_TimeBaseStruct.TIM_ClockDivision = 0; TIM_TimeBaseStruct.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStruct); // PWM通道配置 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC2Init(TIM3, &TIM_OCInitStructure); TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable); TIM_Cmd(TIM3, ENABLE); }- 动态更新占空比:
uint16_t phase = 0; while(1) { TIM_SetCompare2(TIM3, sinTable[phase]); phase = (phase + 1) % SAMPLE_POINTS; Delay(10); // 控制波形频率 }2.2 滤波电路设计与Proteus实现
在Proteus中搭建二阶低通滤波电路时,关键参数计算如下:
截止频率公式:
fc = 1 / (2π√(R1*R2*C1*C2))典型元件值(针对1kHz正弦波):
- R1 = R2 = 10kΩ
- C1 = 22nF, C2 = 10nF
- 理论截止频率 ≈ 1.1kHz
Proteus操作要点:
- 添加"Analog"类别的运算放大器(如LM358)
- 配置双电源供电(±5V)
- 连接PWM输出到滤波电路输入
- 添加示波器观察原始PWM和滤波后波形
注意:实际电路中,运放选择应考虑压摆率(Slew Rate)。对于10kHz以上信号,建议选用SR>5V/μs的运放如TL082。
3. DAC直接输出方案详解
3.1 DAC初始化与波形生成
对于配备DAC的STM32F103型号,配置流程更为直接:
void DAC_Configuration(void) { DAC_InitTypeDef DAC_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; // DAC通道1对应PA4 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOA, &GPIO_InitStructure); DAC_InitStructure.DAC_Trigger = DAC_Trigger_None; DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None; DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Enable; DAC_Init(DAC_Channel_1, &DAC_InitStructure); DAC_Cmd(DAC_Channel_1, ENABLE); } void generateSineWave(void) { static uint16_t phase = 0; uint16_t dacValue = (uint16_t)(2048 * sin(2*3.1415926*phase/SAMPLE_POINTS) + 2048); DAC_SetChannel1Data(DAC_Align_12b_R, dacValue); phase = (phase + 1) % SAMPLE_POINTS; }3.2 输出质量优化技巧
- 软件校准:通过测量实际输出,建立校正表补偿非线性误差
uint16_t calibrationTable[4096]; // 12位DAC的校准表 // 在校准过程中填充校正表 void applyCalibration(uint16_t rawValue) { DAC_SetChannel1Data(DAC_Align_12b_R, calibrationTable[rawValue]); }- DMA传输:减少CPU干预,提高波形更新率
DAC_DMACmd(DAC_Channel_1, ENABLE); DMA_InitStructure.DMA_PeripheralBaseAddr = DAC_DHR12R1_ADDRESS; DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)sinTable; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST; DMA_InitStructure.DMA_BufferSize = SAMPLE_POINTS; DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; DMA_Init(DMA1_Channel3, &DMA_InitStructure); DMA_Cmd(DMA1_Channel3, ENABLE);4. Proteus仿真与波形分析
4.1 虚拟仪器配置技巧
在Proteus中有效分析波形质量需要合理配置测量仪器:
示波器设置:
- 时间基准:根据信号频率调整(如1kHz正弦波建议1ms/div)
- 触发模式:建议使用自动触发
- 通道耦合:观察高频噪声时使用AC耦合
频谱分析仪:
- 窗函数选择:Hanning窗适合大多数情况
- 分辨率带宽(RBW):设置为信号频率的1/10
- 幅度刻度:对数坐标更易观察谐波分量
典型谐波分布对比:
| 谐波次数 | PWM+滤波方案幅度(dB) | 直接DAC方案幅度(dB) |
|---|---|---|
| 基波(1) | 0 | 0 |
| 2 | -45 | -60 |
| 3 | -50 | -65 |
| 5 | -55 | -70 |
4.2 常见问题排查指南
PWM方案输出失真大:
- 检查滤波电路截止频率是否匹配信号频率
- 验证PWM基频是否足够高(建议至少10倍于目标频率)
- 尝试增加滤波阶数或改用有源滤波
DAC方案输出有台阶:
- 提高采样点数(至少32点/周期)
- 启用DAC输出缓冲
- 在输出端添加小电容(如100pF)平滑台阶
Proteus仿真不收敛:
- 减小仿真步长(建议1μs或更小)
- 检查是否有浮空引脚
- 尝试使用"Digital"模式的PWM输出而非"Analog"
