一、系统架构设计
1.1 拓扑结构
单相交流输入 → 整流桥 → Boost电感 → MOSFET → 输出电容 → 直流负载
↑
电流采样电阻
↑
STM32F103C8T6控制
1.2 控制目标
| 参数 |
目标值 |
说明 |
| 输入功率因数 |
>0.99 |
接近单位功率因数 |
| 输入电流THD |
<5% |
电流正弦化程度 |
| 输出电压纹波 |
<2% |
稳定直流输出 |
| 效率 |
>95% |
整体转换效率 |
| 输入电压范围 |
85-265V AC |
宽电压适应 |
二、控制策略
2.1 双闭环控制结构
电压外环(慢环):
输出电压采样 → 与参考电压比较 → 电压PI → 电流参考幅值
电流内环(快环):
输入电流采样 → 与电流参考比较 → 电流PI → PWM占空比
相位同步:
输入电压过零检测 → 锁相环(PLL) → 电流参考相位
2.2 数字控制算法
#define VOUT_REF 400.0f
#define VIN_FREQ 50.0f
#define PWM_FREQ 50000.0f
#define SAMPLE_FREQ 20000.0f
typedef struct {
float kp_v;
float ki_v;
float kp_i;
float ki_i;
float vout_err;
float iout_ref;
float iin_err;
float duty;
} PFC_Controller;
三、STM32F103C8T6实现代码
3.1 主程序(main.c)
#include "stm32f10x.h"
#include "pfc_control.h"
#include "adc.h"
#include "pwm.h"
#include "pll.h"
#include "protection.h"
PFC_Controller pfc_ctrl;
System_Status sys_status;
float vin_rms = 0.0f;
float vin_phase = 0.0f;
float iin_ref = 0.0f;
int main(void)
{
System_Init();
ADC_Init_PFC();
PWM_Init_PFC();
PLL_Init();
Protection_Init();
pfc_ctrl.kp_v = 0.5f;
pfc_ctrl.ki_v = 0.1f;
pfc_ctrl.kp_i = 0.8f;
pfc_ctrl.ki_i = 0.2f;
pfc_ctrl.duty = 0.5f;
PWM_Start(TIM1, CH1);
while(1)
{
System_Monitor();
if(Protection_Check() == PROTECT_FAULT)
{
PWM_Stop(TIM1, CH1);
Fault_Handler();
}
Delay_Ms(1);
}
}
void TIM2_IRQHandler(void)
{
if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
{
float vin_sample = ADC_GetVoltage(ADC_VIN);
float vout_sample = ADC_GetVoltage(ADC_VOUT);
float iin_sample = ADC_GetCurrent(ADC_IIN);
PLL_Update(vin_sample);
vin_phase = PLL_GetPhase();
vin_rms = PLL_GetRMS();
pfc_ctrl.vout_err = VOUT_REF - vout_sample;
pfc_ctrl.iout_ref += pfc_ctrl.ki_v * pfc_ctrl.vout_err;
if(pfc_ctrl.iout_ref > MAX_CURRENT_REF)
pfc_ctrl.iout_ref = MAX_CURRENT_REF;
if(pfc_ctrl.iout_ref < 0.0f)
pfc_ctrl.iout_ref = 0.0f;
iin_ref = pfc_ctrl.iout_ref * sinf(vin_phase);
pfc_ctrl.iin_err = iin_ref - iin_sample;
pfc_ctrl.duty += pfc_ctrl.kp_i * pfc_ctrl.iin_err;
pfc_ctrl.duty += pfc_ctrl.ki_i * pfc_ctrl.iin_err;
if(pfc_ctrl.duty > 0.95f) pfc_ctrl.duty = 0.95f;
if(pfc_ctrl.duty < 0.05f) pfc_ctrl.duty = 0.05f;
PWM_SetDuty(TIM1, CH1, pfc_ctrl.duty);
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
}
}
3.2 锁相环(pll.c)
#include "pll.h"
#include <math.h>
typedef struct {
float phase;
float freq;
float kp;
float ki;
float phase_error;
float freq_integral;
} PLL_TypeDef;
static PLL_TypeDef pll;
void PLL_Init(void)
{
pll.phase = 0.0f;
pll.freq = 50.0f;
pll.kp = 0.1f;
pll.ki = 0.01f;
pll.phase_error = 0.0f;
pll.freq_integral = 0.0f;
}
void PLL_Update(float vin_sample)
{
static float last_vin = 0.0f;
static uint8_t zero_cross_flag = 0;
if(last_vin < 0 && vin_sample >= 0)
{
zero_cross_flag = 1;
static uint32_t last_zero_time = 0;
uint32_t now = Get_Tick();
if(last_zero_time > 0)
{
float period = (now - last_zero_time) / 1000.0f;
pll.freq = 1.0f / period;
}
last_zero_time = now;
}
last_vin = vin_sample;
if(zero_cross_flag)
{
pll.phase = atan2f(vin_sample, 1.0f);
float grid_phase = 2.0f * PI * pll.freq * (Get_Tick() / 1000.0f);
pll.phase_error = grid_phase - pll.phase;
pll.freq_integral += pll.ki * pll.phase_error;
pll.freq = 50.0f + pll.freq_integral + pll.kp * pll.phase_error;
if(pll.freq > 55.0f) pll.freq = 55.0f;
if(pll.freq < 45.0f) pll.freq = 45.0f;
}
pll.phase += 2.0f * PI * pll.freq / SAMPLE_FREQ;
if(pll.phase > 2.0f * PI) pll.phase -= 2.0f * PI;
}
float PLL_GetPhase(void)
{
return pll.phase;
}
float PLL_GetRMS(void)
{
static float vin_sum = 0.0f;
static uint32_t sample_count = 0;
static float vin_square_sum = 0.0f;
vin_sum += vin_sample;
vin_square_sum += vin_sample * vin_sample;
sample_count++;
if(sample_count >= 1000)
{
float rms = sqrtf(vin_square_sum / sample_count);
vin_sum = 0.0f;
vin_square_sum = 0.0f;
sample_count = 0;
return rms;
}
return 0.0f;
}
3.3 PWM驱动(pwm.c)
#include "pwm.h"
void PWM_Init_PFC(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef TIM_OCInitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1 | RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
TIM_TimeBaseStructure.TIM_Period = (SystemCoreClock / PWM_FREQ) - 1;
TIM_TimeBaseStructure.TIM_Prescaler = 0;
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Disable;
TIM_OCInitStructure.TIM_Pulse = 0;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_High;
TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Reset;
TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCNIdleState_Reset;
TIM_OC1Init(TIM1, &TIM_OCInitStructure);
TIM_OC1PreloadConfig(TIM1, TIM_OCPreload_Enable);
TIM_CtrlPWMOutputs(TIM1, ENABLE);
TIM_Cmd(TIM1, ENABLE);
}
void PWM_SetDuty(TIM_TypeDef* TIMx, uint16_t channel, float duty)
{
uint16_t pulse = (uint16_t)((TIMx->ARR + 1) * duty);
switch(channel)
{
case CH1:
TIM_SetCompare1(TIMx, pulse);
break;
case CH2:
TIM_SetCompare2(TIMx, pulse);
break;
case CH3:
TIM_SetCompare3(TIMx, pulse);
break;
case CH4:
TIM_SetCompare4(TIMx, pulse);
break;
}
}
void PWM_Start(TIM_TypeDef* TIMx, uint16_t channel)
{
TIM_Cmd(TIMx, ENABLE);
}
void PWM_Stop(TIM_TypeDef* TIMx, uint16_t channel)
{
TIM_Cmd(TIMx, DISABLE);
}
3.4 ADC采样(adc.c)
#include "adc.h"
void ADC_Init_PFC(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 3;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5);
ADC_Cmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
float ADC_GetVoltage(uint8_t channel)
{
uint16_t adc_value = 0;
float voltage = 0.0f;
if(channel == ADC_VIN)
{
adc_value = ADC_GetConversionValue(ADC1);
voltage = (adc_value * 3.3f / 4095.0f) * 22.0f;
}
else if(channel == ADC_VOUT)
{
adc_value = ADC_GetConversionValue(ADC1);
voltage = (adc_value * 3.3f / 4095.0f) * 10.0f;
}
return voltage;
}
float ADC_GetCurrent(uint8_t channel)
{
uint16_t adc_value = 0;
float current = 0.0f;
if(channel == ADC_IIN)
{
adc_value = ADC_GetConversionValue(ADC1);
current = (adc_value * 3.3f / 4095.0f) / 0.2f;
}
return current;
}
3.5 保护功能(protection.c)
#include "protection.h"
#define VOUT_OVERVOLTAGE 450.0f
#define VOUT_UNDERVOLTAGE 350.0f
#define IIN_OVERCURRENT 15.0f
#define TEMP_OVERTEMP 85.0f
typedef enum {
PROTECT_NORMAL = 0,
PROTECT_OVERVOLT,
PROTECT_UNDERVOLT,
PROTECT_OVERCURR,
PROTECT_OVERTEMP,
PROTECT_FAULT
} ProtectState;
static ProtectState protect_state = PROTECT_NORMAL;
void Protection_Init(void)
{
protect_state = PROTECT_NORMAL;
}
ProtectState Protection_Check(void)
{
float vout = ADC_GetVoltage(ADC_VOUT);
float iin = ADC_GetCurrent(ADC_IIN);
float temp = Temperature_Get();
if(vout > VOUT_OVERVOLTAGE)
{
protect_state = PROTECT_OVERVOLT;
return protect_state;
}
if(vout < VOUT_UNDERVOLTAGE)
{
protect_state = PROTECT_UNDERVOLT;
return protect_state;
}
if(iin > IIN_OVERCURRENT)
{
protect_state = PROTECT_OVERCURR;
return protect_state;
}
if(temp > TEMP_OVERTEMP)
{
protect_state = PROTECT_OVERTEMP;
return protect_state;
}
protect_state = PROTECT_NORMAL;
return protect_state;
}
void Fault_Handler(void)
{
PWM_Stop(TIM1, CH1);
GPIO_SetBits(FAULT_LED_PORT, FAULT_LED_PIN);
uint8_t fault_code = (uint8_t)protect_state;
Flash_WriteFaultLog(fault_code);
while(1)
{
GPIO_ToggleBits(FAULT_LED_PORT, FAULT_LED_PIN);
Delay_Ms(500);
}
}
参考代码 整流器稳定输出电压 www.youwenfan.com/contentali/56322.html
四、硬件设计要点
4.1 关键元器件选型
| 元器件 |
型号/参数 |
说明 |
| MOSFET |
IPP65R110CFD |
650V/33A,低导通电阻 |
| 二极管 |
IDW30G65C5 |
碳化硅二极管,低反向恢复 |
| 电感 |
PQ32/20 |
1mH,铁硅铝磁芯 |
| 电容 |
450V/470μF电解 |
输出滤波电容 |
| 采样电阻 |
0.01Ω/5W |
锰铜合金电阻 |
4.2 PCB布局要点
- 功率回路最小化:减小开关噪声
- 模拟地与数字地分离:单点接地
- 电流采样走差分线:减小干扰
- 驱动信号远离模拟信号:避免串扰
- 散热设计:MOSFET和二极管加散热片
五、调试与测试
5.1 调试步骤
- 开环测试:不接控制算法,固定占空比测试
- 电压环测试:仅启用电压环,验证稳压性能
- 电流环测试:加入电流环,验证电流跟踪
- 并网测试:连接电网,验证功率因数
- 满载测试:验证额定功率下的性能
5.2 性能指标测试
| 测试项目 |
测试方法 |
合格标准 |
| 功率因数 |
功率计测量 |
>0.99 |
| THD |
示波器FFT分析 |
<5% |
| 效率 |
输入输出功率比 |
>95% |
| 稳压精度 |
负载变化测试 |
±1% |
| 动态响应 |
阶跃负载测试 |
超调<5% |
5.3 示波器观测点
CH1: 输入电压(整流后)
CH2: 输入电流
CH3: 输出电压
CH4: PWM驱动信号
六、优化建议
6.1 软件优化
- 数字滤波器:加入低通滤波器滤除高频噪声
- 前馈控制:加入输入电压前馈,提高动态响应
- 自适应参数:根据负载变化调整PI参数
- 软启动:避免启动冲击电流
6.2 硬件优化
- EMI滤波:输入端加入LC滤波器
- 缓冲电路:MOSFET并联RC吸收电路
- 温度传感器:NTC监测散热器温度
- 辅助电源:独立的控制电源
七、安全注意事项
高压危险!
- 隔离测试:使用隔离变压器进行调试
- 绝缘手套:操作时佩戴绝缘防护
- 急停按钮:设置紧急断电开关
- 接地保护:确保设备良好接地
- 逐步加压:从小电压开始测试