从 PWM 到正弦波STM32F103 波形生成的双路径实战解析在嵌入式开发中信号生成是基础却至关重要的技能。许多开发者熟悉基础的PWM输出但当需求升级到更复杂的模拟信号如正弦波时往往面临选择是用芯片内置的DAC直接输出还是通过PWM加滤波电路间接实现本文将深入探讨这两种方法的实现细节、优劣对比以及在Proteus环境下的仿真验证技巧。1. 硬件基础与方案选型STM32F103系列微控制器作为经典的Cortex-M3内核芯片其外设资源丰富但配置灵活度较高。在波形生成场景中我们需要先明确几个关键硬件特性PWM模块STM32F103通常配备多个高级定时器如TIM1和通用定时器如TIM3/TIM4支持PWM生成。以TIM3为例其特性包括16位自动重装载寄存器可编程预分频器4个独立通道CH1-CH4支持PWM模式1和模式2DAC模块并非所有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 i0; iSAMPLE_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.1kHzProteus操作要点添加Analog类别的运算放大器如LM358配置双电源供电±5V连接PWM输出到滤波电路输入添加示波器观察原始PWM和滤波后波形注意实际电路中运放选择应考虑压摆率(Slew Rate)。对于10kHz以上信号建议选用SR5V/μ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)002-45-603-50-655-55-704.2 常见问题排查指南PWM方案输出失真大检查滤波电路截止频率是否匹配信号频率验证PWM基频是否足够高建议至少10倍于目标频率尝试增加滤波阶数或改用有源滤波DAC方案输出有台阶提高采样点数至少32点/周期启用DAC输出缓冲在输出端添加小电容如100pF平滑台阶Proteus仿真不收敛减小仿真步长建议1μs或更小检查是否有浮空引脚尝试使用Digital模式的PWM输出而非Analog