当前位置: 首页 > news >正文

从 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占空比来逼近正弦波形。具体实现分为三个关键步骤:

  1. 正弦波表生成:预先计算一个周期内的采样值
#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)); } }
  1. 定时器配置(以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); }
  1. 动态更新占空比
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操作要点

  1. 添加"Analog"类别的运算放大器(如LM358)
  2. 配置双电源供电(±5V)
  3. 连接PWM输出到滤波电路输入
  4. 添加示波器观察原始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中有效分析波形质量需要合理配置测量仪器:

  1. 示波器设置

    • 时间基准:根据信号频率调整(如1kHz正弦波建议1ms/div)
    • 触发模式:建议使用自动触发
    • 通道耦合:观察高频噪声时使用AC耦合
  2. 频谱分析仪

    • 窗函数选择:Hanning窗适合大多数情况
    • 分辨率带宽(RBW):设置为信号频率的1/10
    • 幅度刻度:对数坐标更易观察谐波分量

典型谐波分布对比

谐波次数PWM+滤波方案幅度(dB)直接DAC方案幅度(dB)
基波(1)00
2-45-60
3-50-65
5-55-70

4.2 常见问题排查指南

  • PWM方案输出失真大

    • 检查滤波电路截止频率是否匹配信号频率
    • 验证PWM基频是否足够高(建议至少10倍于目标频率)
    • 尝试增加滤波阶数或改用有源滤波
  • DAC方案输出有台阶

    • 提高采样点数(至少32点/周期)
    • 启用DAC输出缓冲
    • 在输出端添加小电容(如100pF)平滑台阶
  • Proteus仿真不收敛

    • 减小仿真步长(建议1μs或更小)
    • 检查是否有浮空引脚
    • 尝试使用"Digital"模式的PWM输出而非"Analog"
http://www.jsqmd.com/news/697837/

相关文章:

  • HEIF Utility完整指南:在Windows上轻松处理iPhone照片的实用工具
  • DeepSeek 开源 TileKernels:用 Python 写出逼近硬件极限的 GPU 内核
  • SES工程移植避坑指南:为什么你的启动文件总报错?详解Startup.s与Vector.s的正确替换姿势
  • 嵌入式C语言面试官最爱问的6个基础概念,你真的都搞懂了吗?
  • Rocky Linux 9 与Centos区别,以及软件安装dnf命令
  • 2026宜昌现代简约装修选购指南,专业公司口碑排名出炉 - myqiye
  • 开源推荐:API Relay — 大模型API中转站,多账号自动轮换+赛博朋克管理面板
  • Arduino IDE 2.0+ 库文件搬家指南:告别C盘爆满,轻松迁移Arduino15到D盘
  • Windows Cleaner终极指南:三分钟解决C盘爆红,电脑焕然一新!
  • 避坑指南:树莓派配置LIRC红外遥控最容易踩的5个坑(内核版本、设备节点、配置文件格式)
  • 构建企业内网精准时钟:AD域控NTP服务端与客户端配置实战
  • Claude Code 使用教程
  • 盘点2026年山东、湖北实力强的石英管源头厂家哪家性价比高 - 工业品牌热点
  • GLM-5.1 上线火山 Coding Plan:Opus 级编码能力,不限购真香
  • 如何让无导航PDF秒变智能文档?pdfdir一键添加专业级书签
  • CAD VBA实战:利用GetBoundingBox与GetVariable实现智能图元定位与批量标注
  • 告别卡顿!保姆级教程:在 Windows Server 2019/2022 上为 Docker 正确配置 WSL 2 后端
  • DC-DC反馈电阻取值:效率、精度与稳定性的权衡艺术
  • Element UI el-select全选功能翻车实录:我踩过的3个坑和性能优化方案
  • TileLang + TileKernels:DeepSeek 的 GPU 内核开发新范式,70 行 Python 替代 3000 行 CUDA
  • YOLO演进史 | 正负样本分配策略的“进化论”
  • 从代码到电线:手把手教你用Python和树莓派玩转RS485多设备通信(模拟I2C主从)
  • 想了解黑龙江滨沃管业克拉管,它的性价比高不高? - mypinpai
  • 终极1Fichier下载管理指南:5分钟快速上手的高效下载解决方案
  • 别再只用基础门了!用Verilog UDP为你的FPGA/ASIC验证提速(避坑指南)
  • 在F1C100s上跑GBA游戏:手把手教你用Buildroot配置SDL和编译gpsp模拟器
  • OpenCore Legacy Patcher:老Mac升级新系统的完整方案深度解析
  • 周深2026「深深的」演唱会抢票攻略|告别秒空,新手也能轻松抢到票
  • ARM SVE与SME架构:原理、启用控制与性能优化
  • LFM2.5-VL-1.6B部署教程:配合Redis缓存高频问答提升响应效率