用STM32F103和AD9833打造高精度信号发生器硬件设计、固件开发与波形优化全解析在电子工程和嵌入式开发领域信号发生器是不可或缺的基础工具。无论是测试滤波器响应、校准传感器还是验证通信协议一个稳定可靠的信号源都能显著提升开发效率。本文将带你从零开始基于STM32F103微控制器和AD9833波形发生器芯片构建一个成本低廉但性能不俗的DIY信号源解决方案。这个项目特别适合电子爱好者、高校学生以及需要快速原型验证的工程师。与市面上动辄上千元的专业设备相比我们的方案成本控制在百元以内却能实现0-12.5MHz的频率范围支持正弦波、三角波和方波输出频率分辨率可达0.1Hz。更重要的是通过完整的DIY过程你将深入理解数字频率合成(DDS)技术的工作原理掌握嵌入式系统与模拟电路的协同设计技巧。1. 硬件设计与元器件选型1.1 核心器件选择与电路设计AD9833是ADI公司推出的一款低功耗、可编程波形发生器采用DDS技术实现高精度的频率合成。其关键特性包括频率范围0到12.5MHz分辨率28位0.1Hz 25MHz参考时钟输出波形正弦波、三角波、方波供电电压2.3V至5.5V通信接口SPI兼容推荐外围电路配置元件参数/型号作用说明有源晶振25MHz ±50ppm提供系统时钟基准低通滤波器截止频率15MHz消除高频谐波失真输出放大器OPA2350增强驱动能力调节输出幅度去耦电容0.1μF陶瓷10μF钽电源噪声抑制提示AD9833对参考时钟质量敏感建议选用稳定性优于±50ppm的有源晶振。若预算允许TCXO(温补晶振)能进一步提升频率稳定性。1.2 PCB布局与焊接要点在实际制作中合理的PCB布局对信号质量至关重要电源分区数字部分(STM32)与模拟部分(AD9833)采用星型接地在电源入口处放置π型滤波器(10μF0.1μF)信号走线SPI时钟线(SCLK)尽量短且等长避免高频信号线平行走线必要时采用地线隔离焊接顺序先焊接电源相关元件测试各节点电压正常后再继续AD9833采用热风枪焊接温度控制在300℃以下// 硬件连接示意图 /* STM32F103 AD9833 PA5 --- SCLK PA6 --- SDATA PA7 --- FSYNC 3.3V --- VCC GND --- GND */2. 固件开发与驱动实现2.1 STM32CubeMX基础配置使用STM32CubeMX工具快速建立工程框架选择STM32F103C8T6型号启用SPI1接口配置为Master模式时钟分频设为8得到4.5MHz SPI时钟启用USART1用于调试输出配置两个定时器TIM2用于界面刷新(100ms周期)TIM3用于旋钮编码器检测关键配置参数// SPI初始化代码片段 hspi1.Instance SPI1; hspi1.Init.Mode SPI_MODE_MASTER; hspi1.Init.Direction SPI_DIRECTION_2LINES; hspi1.Init.DataSize SPI_DATASIZE_16BIT; hspi1.Init.CLKPolarity SPI_POLARITY_LOW; hspi1.Init.CLKPhase SPI_PHASE_1EDGE; hspi1.Init.NSS SPI_NSS_SOFT; hspi1.Init.BaudRatePrescaler SPI_BAUDRATEPRESCALER_8; hspi1.Init.FirstBit SPI_FIRSTBIT_MSB;2.2 AD9833驱动核心实现AD9833通过16位SPI接口进行控制其命令格式如下频率寄存器写入控制字0x4000 | (FREQ0_REG或FREQ1_REG)后跟28位频率值分两次14位传输相位寄存器写入控制字0xC000 | (PHASE0_REG或PHASE1_REG)后跟12位相位值优化后的驱动函数示例void AD9833_SetFrequency(uint32_t freq) { uint64_t freq_word (uint64_t)freq * 268435456ULL / 25000000; uint16_t freq_low (freq_word 0x3FFF) | 0x4000; uint16_t freq_high ((freq_word 14) 0x3FFF) | 0x4000; HAL_GPIO_WritePin(AD9833_FSYNC_GPIO_Port, AD9833_FSYNC_Pin, GPIO_PIN_RESET); HAL_SPI_Transmit(hspi1, freq_low, 1, 100); HAL_SPI_Transmit(hspi1, freq_high, 1, 100); HAL_GPIO_WritePin(AD9833_FSYNC_GPIO_Port, AD9833_FSYNC_Pin, GPIO_PIN_SET); }注意频率计算采用64位整数运算避免溢出268435456是2^28对应AD9833的28位调谐字。3. 用户界面与功能扩展3.1 旋钮编码器交互设计采用EC11旋转编码器实现频率调节支持以下操作模式短按切换频率调节步进(1Hz/10Hz/100Hz/1kHz)旋转增减当前频率值长按2秒切换波形类型防抖处理算法void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim) { static uint8_t last_state 0; uint8_t current_state (HAL_GPIO_ReadPin(ENC_A_GPIO_Port, ENC_A_Pin) 1) | HAL_GPIO_ReadPin(ENC_B_GPIO_Port, ENC_B_Pin); // 状态机实现方向判断 if((last_state 0x01 current_state 0x03) || (last_state 0x03 current_state 0x02) || (last_state 0x02 current_state 0x00) || (last_state 0x00 current_state 0x01)) { frequency step_size; // 顺时针 } else { frequency - step_size; // 逆时针 } last_state current_state; AD9833_SetFrequency(frequency); }3.2 OLED显示界面优化SSD1306 OLED显示屏可直观展示当前状态主界面布局┌─────────────────┐ │ 波形[Sine] │ │ 频率1000.0 Hz │ │ 步进10 Hz │ │ 幅度2.0 Vpp │ └─────────────────┘菜单导航设计进入设置菜单后旋转选择项目短按进入子项再次短按确认修改长按返回上级菜单显示刷新优化技巧采用局部刷新策略只更新变化区域使用预先渲染的字符位图减少计算量关键参数变化时添加视觉反馈如闪烁4. 系统测试与性能优化4.1 波形质量测试方案使用数字示波器进行系统级验证时重点关注以下指标频率精度对比设定值与实际输出频率在不同温度下测试频率漂移谐波失真测量输出信号的THD(总谐波失真)正弦波在1kHz时应1%方波特性上升时间(10%~90%)过冲和振铃现象实测数据对比表频率设定值实测频率误差幅度波动THD1 kHz999.8 Hz-0.02%±0.5%0.8%100 kHz99.97 kHz-0.03%±1.2%1.5%1 MHz0.999 MHz-0.1%±2.0%3.2%4.2 常见问题排查指南遇到输出异常时可按照以下流程排查无输出信号检查AD9833电源电压(引脚20)测量25MHz时钟是否正常(引脚21)用逻辑分析仪抓取SPI通信波形频率偏差大确认参考时钟精度检查SPI数据传输是否正确验证频率计算算法波形失真严重检查输出滤波器参数降低输出幅度看是否改善尝试断开后级电路测试# 简单的频率误差分析脚本示例 import numpy as np def calculate_error(set_freq, measured_freq): ppm (measured_freq - set_freq) / set_freq * 1e6 return round(ppm, 2) # 测试数据 frequencies { 1kHz: (1000, 999.8), 100kHz: (100000, 99970), 1MHz: (1e6, 0.999e6) } for name, (set_freq, meas_freq) in frequencies.items(): print(f{name}: {calculate_error(set_freq, meas_freq)} ppm)5. 进阶功能扩展思路5.1 幅度调制实现通过数字电位器或DAC控制输出幅度硬件方案使用MCP41010数字电位器调节运放增益或者采用AD5628等DAC生成控制电压软件控制void set_amplitude(float percent) { uint16_t dac_value (uint16_t)(percent * 4095 / 100); HAL_DAC_SetValue(hdac, DAC_CHANNEL_1, DAC_ALIGN_12B_R, dac_value); HAL_DAC_Start(hdac, DAC_CHANNEL_1); }5.2 扫频功能开发实现自动频率扫描对电路测试非常有用线性扫频void linear_sweep(uint32_t start, uint32_t stop, uint32_t step, uint32_t dwell) { for(uint32_t f start; f stop; f step) { AD9833_SetFrequency(f); HAL_Delay(dwell); } }对数扫频void log_sweep(uint32_t start, uint32_t stop, uint32_t points, uint32_t dwell) { double log_start log10(start); double log_stop log10(stop); double delta (log_stop - log_start) / points; for(uint32_t i 0; i points; i) { double freq pow(10, log_start i * delta); AD9833_SetFrequency((uint32_t)freq); HAL_Delay(dwell); } }5.3 PC端控制接口通过USB虚拟串口实现远程控制通信协议设计SET:FREQ1000 // 设置频率为1kHz SET:WAVESINE // 设置正弦波 GET?FREQ // 查询当前频率命令解析实现void parse_command(char* cmd) { char* token strtok(cmd, : ); if(strcmp(token, SET) 0) { token strtok(NULL, :); if(strcmp(token, FREQ) 0) { uint32_t freq atoi(strtok(NULL, )); AD9833_SetFrequency(freq); } // 其他参数处理... } }在完成基础功能后我发现AD9833的温度稳定性是影响长期频率精度的主要因素。通过添加简单的温度补偿算法——读取板载温度传感器数据对参考时钟频率进行微调可以将常温下的频率稳定性提升到±5ppm以内。另一个实用技巧是在输出端添加一个简单的幅度检测电路通过ADC反馈实现闭环控制这样即使负载变化输出幅度也能保持稳定。
用STM32F103和AD9833制作一个简易信号源:从电路搭建、驱动编写到波形测试全记录
发布时间:2026/5/16 9:26:47
用STM32F103和AD9833打造高精度信号发生器硬件设计、固件开发与波形优化全解析在电子工程和嵌入式开发领域信号发生器是不可或缺的基础工具。无论是测试滤波器响应、校准传感器还是验证通信协议一个稳定可靠的信号源都能显著提升开发效率。本文将带你从零开始基于STM32F103微控制器和AD9833波形发生器芯片构建一个成本低廉但性能不俗的DIY信号源解决方案。这个项目特别适合电子爱好者、高校学生以及需要快速原型验证的工程师。与市面上动辄上千元的专业设备相比我们的方案成本控制在百元以内却能实现0-12.5MHz的频率范围支持正弦波、三角波和方波输出频率分辨率可达0.1Hz。更重要的是通过完整的DIY过程你将深入理解数字频率合成(DDS)技术的工作原理掌握嵌入式系统与模拟电路的协同设计技巧。1. 硬件设计与元器件选型1.1 核心器件选择与电路设计AD9833是ADI公司推出的一款低功耗、可编程波形发生器采用DDS技术实现高精度的频率合成。其关键特性包括频率范围0到12.5MHz分辨率28位0.1Hz 25MHz参考时钟输出波形正弦波、三角波、方波供电电压2.3V至5.5V通信接口SPI兼容推荐外围电路配置元件参数/型号作用说明有源晶振25MHz ±50ppm提供系统时钟基准低通滤波器截止频率15MHz消除高频谐波失真输出放大器OPA2350增强驱动能力调节输出幅度去耦电容0.1μF陶瓷10μF钽电源噪声抑制提示AD9833对参考时钟质量敏感建议选用稳定性优于±50ppm的有源晶振。若预算允许TCXO(温补晶振)能进一步提升频率稳定性。1.2 PCB布局与焊接要点在实际制作中合理的PCB布局对信号质量至关重要电源分区数字部分(STM32)与模拟部分(AD9833)采用星型接地在电源入口处放置π型滤波器(10μF0.1μF)信号走线SPI时钟线(SCLK)尽量短且等长避免高频信号线平行走线必要时采用地线隔离焊接顺序先焊接电源相关元件测试各节点电压正常后再继续AD9833采用热风枪焊接温度控制在300℃以下// 硬件连接示意图 /* STM32F103 AD9833 PA5 --- SCLK PA6 --- SDATA PA7 --- FSYNC 3.3V --- VCC GND --- GND */2. 固件开发与驱动实现2.1 STM32CubeMX基础配置使用STM32CubeMX工具快速建立工程框架选择STM32F103C8T6型号启用SPI1接口配置为Master模式时钟分频设为8得到4.5MHz SPI时钟启用USART1用于调试输出配置两个定时器TIM2用于界面刷新(100ms周期)TIM3用于旋钮编码器检测关键配置参数// SPI初始化代码片段 hspi1.Instance SPI1; hspi1.Init.Mode SPI_MODE_MASTER; hspi1.Init.Direction SPI_DIRECTION_2LINES; hspi1.Init.DataSize SPI_DATASIZE_16BIT; hspi1.Init.CLKPolarity SPI_POLARITY_LOW; hspi1.Init.CLKPhase SPI_PHASE_1EDGE; hspi1.Init.NSS SPI_NSS_SOFT; hspi1.Init.BaudRatePrescaler SPI_BAUDRATEPRESCALER_8; hspi1.Init.FirstBit SPI_FIRSTBIT_MSB;2.2 AD9833驱动核心实现AD9833通过16位SPI接口进行控制其命令格式如下频率寄存器写入控制字0x4000 | (FREQ0_REG或FREQ1_REG)后跟28位频率值分两次14位传输相位寄存器写入控制字0xC000 | (PHASE0_REG或PHASE1_REG)后跟12位相位值优化后的驱动函数示例void AD9833_SetFrequency(uint32_t freq) { uint64_t freq_word (uint64_t)freq * 268435456ULL / 25000000; uint16_t freq_low (freq_word 0x3FFF) | 0x4000; uint16_t freq_high ((freq_word 14) 0x3FFF) | 0x4000; HAL_GPIO_WritePin(AD9833_FSYNC_GPIO_Port, AD9833_FSYNC_Pin, GPIO_PIN_RESET); HAL_SPI_Transmit(hspi1, freq_low, 1, 100); HAL_SPI_Transmit(hspi1, freq_high, 1, 100); HAL_GPIO_WritePin(AD9833_FSYNC_GPIO_Port, AD9833_FSYNC_Pin, GPIO_PIN_SET); }注意频率计算采用64位整数运算避免溢出268435456是2^28对应AD9833的28位调谐字。3. 用户界面与功能扩展3.1 旋钮编码器交互设计采用EC11旋转编码器实现频率调节支持以下操作模式短按切换频率调节步进(1Hz/10Hz/100Hz/1kHz)旋转增减当前频率值长按2秒切换波形类型防抖处理算法void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim) { static uint8_t last_state 0; uint8_t current_state (HAL_GPIO_ReadPin(ENC_A_GPIO_Port, ENC_A_Pin) 1) | HAL_GPIO_ReadPin(ENC_B_GPIO_Port, ENC_B_Pin); // 状态机实现方向判断 if((last_state 0x01 current_state 0x03) || (last_state 0x03 current_state 0x02) || (last_state 0x02 current_state 0x00) || (last_state 0x00 current_state 0x01)) { frequency step_size; // 顺时针 } else { frequency - step_size; // 逆时针 } last_state current_state; AD9833_SetFrequency(frequency); }3.2 OLED显示界面优化SSD1306 OLED显示屏可直观展示当前状态主界面布局┌─────────────────┐ │ 波形[Sine] │ │ 频率1000.0 Hz │ │ 步进10 Hz │ │ 幅度2.0 Vpp │ └─────────────────┘菜单导航设计进入设置菜单后旋转选择项目短按进入子项再次短按确认修改长按返回上级菜单显示刷新优化技巧采用局部刷新策略只更新变化区域使用预先渲染的字符位图减少计算量关键参数变化时添加视觉反馈如闪烁4. 系统测试与性能优化4.1 波形质量测试方案使用数字示波器进行系统级验证时重点关注以下指标频率精度对比设定值与实际输出频率在不同温度下测试频率漂移谐波失真测量输出信号的THD(总谐波失真)正弦波在1kHz时应1%方波特性上升时间(10%~90%)过冲和振铃现象实测数据对比表频率设定值实测频率误差幅度波动THD1 kHz999.8 Hz-0.02%±0.5%0.8%100 kHz99.97 kHz-0.03%±1.2%1.5%1 MHz0.999 MHz-0.1%±2.0%3.2%4.2 常见问题排查指南遇到输出异常时可按照以下流程排查无输出信号检查AD9833电源电压(引脚20)测量25MHz时钟是否正常(引脚21)用逻辑分析仪抓取SPI通信波形频率偏差大确认参考时钟精度检查SPI数据传输是否正确验证频率计算算法波形失真严重检查输出滤波器参数降低输出幅度看是否改善尝试断开后级电路测试# 简单的频率误差分析脚本示例 import numpy as np def calculate_error(set_freq, measured_freq): ppm (measured_freq - set_freq) / set_freq * 1e6 return round(ppm, 2) # 测试数据 frequencies { 1kHz: (1000, 999.8), 100kHz: (100000, 99970), 1MHz: (1e6, 0.999e6) } for name, (set_freq, meas_freq) in frequencies.items(): print(f{name}: {calculate_error(set_freq, meas_freq)} ppm)5. 进阶功能扩展思路5.1 幅度调制实现通过数字电位器或DAC控制输出幅度硬件方案使用MCP41010数字电位器调节运放增益或者采用AD5628等DAC生成控制电压软件控制void set_amplitude(float percent) { uint16_t dac_value (uint16_t)(percent * 4095 / 100); HAL_DAC_SetValue(hdac, DAC_CHANNEL_1, DAC_ALIGN_12B_R, dac_value); HAL_DAC_Start(hdac, DAC_CHANNEL_1); }5.2 扫频功能开发实现自动频率扫描对电路测试非常有用线性扫频void linear_sweep(uint32_t start, uint32_t stop, uint32_t step, uint32_t dwell) { for(uint32_t f start; f stop; f step) { AD9833_SetFrequency(f); HAL_Delay(dwell); } }对数扫频void log_sweep(uint32_t start, uint32_t stop, uint32_t points, uint32_t dwell) { double log_start log10(start); double log_stop log10(stop); double delta (log_stop - log_start) / points; for(uint32_t i 0; i points; i) { double freq pow(10, log_start i * delta); AD9833_SetFrequency((uint32_t)freq); HAL_Delay(dwell); } }5.3 PC端控制接口通过USB虚拟串口实现远程控制通信协议设计SET:FREQ1000 // 设置频率为1kHz SET:WAVESINE // 设置正弦波 GET?FREQ // 查询当前频率命令解析实现void parse_command(char* cmd) { char* token strtok(cmd, : ); if(strcmp(token, SET) 0) { token strtok(NULL, :); if(strcmp(token, FREQ) 0) { uint32_t freq atoi(strtok(NULL, )); AD9833_SetFrequency(freq); } // 其他参数处理... } }在完成基础功能后我发现AD9833的温度稳定性是影响长期频率精度的主要因素。通过添加简单的温度补偿算法——读取板载温度传感器数据对参考时钟频率进行微调可以将常温下的频率稳定性提升到±5ppm以内。另一个实用技巧是在输出端添加一个简单的幅度检测电路通过ADC反馈实现闭环控制这样即使负载变化输出幅度也能保持稳定。