富芮坤FR801xh互补pwm参考

本文详细描述了一个C语言函数,用于在两个PWM通道间实现互补控制,通过调整频率和占空比来同步脉冲。函数`pwm_update_complementation_ctrl`处理了通道切换、延迟计时以及占空比级别的同步问题。
摘要由CSDN通过智能技术生成

接口放在driver_pwm.c
参数:两路pwm的通道、频率、占空比、占空比等级(如100级、255级等)

注意占空比等级要和pwm_init和pwm_update里面的相同

void pwm_update_complementation_ctrl(enum pwm_channel_t channel_a, uint32_t frequency_a, uint16_t high_duty_a,
                                        enum pwm_channel_t channel_b, uint32_t frequency_b, uint16_t high_duty_b, uint16_t duty_cycle_lvl)
{
    enum pwm_channel_t chl_tmp = (high_duty_a > high_duty_b)?channel_b:channel_a;
    enum pwm_channel_t chl_delay_start;
    uint32_t frequency_x,frequency_y;
    uint16_t high_duty_x,high_duty_y;
    uint32_t delay_cnt = 0;
    
    if(chl_tmp == channel_a)
    {
        chl_delay_start = channel_b;
        frequency_x = frequency_a;
        high_duty_x = high_duty_a;
        frequency_y = frequency_b;
        high_duty_y = high_duty_b;
    }
    else
    {
        chl_delay_start = channel_a;
        frequency_x = frequency_b;
        high_duty_x = high_duty_b;
        frequency_y = frequency_a;
        high_duty_y = high_duty_a;
    }
//    co_printf("chl_tmp:%d %d %d\r\n",chl_tmp,pwm_ctrl->channel[chl_tmp].cur_cnt,pwm_ctrl->channel[chl_tmp].total_cnt);
    GLOBAL_INT_DISABLE();

    pwm_ctrl->channel[chl_delay_start].ctrl.en = 0;
    pwm_ctrl->channel[chl_delay_start].ctrl.out_en = 0;
    pwm_ctrl->channel[chl_tmp].ctrl.en = 0;
    pwm_ctrl->channel[chl_tmp].ctrl.out_en = 0;
    
    pwm_init(chl_delay_start,frequency_y,high_duty_y);
    pwm_init(chl_tmp,frequency_x,high_duty_x); 
    pwm_ctrl->channel[chl_tmp].cur_cnt = 0;
    pwm_ctrl->channel[chl_delay_start].cur_cnt = 0;
    
    pwm_ctrl->channel[chl_delay_start].ctrl.out_en = 1;
    pwm_ctrl->channel[chl_delay_start].ctrl.en = 1;
    pwm_ctrl->channel[chl_tmp].ctrl.out_en = 1;
    pwm_ctrl->channel[chl_tmp].ctrl.en = 1;
    
    //delay_cnt = pwm_ctrl->channel[chl_tmp].total_cnt * ((duty_cycle_lvl-high_duty_a) - (duty_cycle_lvl-high_duty_b-high_duty_a)/2)/duty_cycle_lvl;
    delay_cnt = pwm_ctrl->channel[chl_tmp].total_cnt * (duty_cycle_lvl+high_duty_y-high_duty_x)/(2*duty_cycle_lvl);
    
    //co_printf("delay_cnt:%d %d %d\r\n",delay_cnt,pwm_ctrl->channel[chl_tmp].cur_cnt,pwm_ctrl->channel[chl_tmp].total_cnt);
    while(pwm_ctrl->channel[chl_tmp].cur_cnt < delay_cnt)
        pwm_ctrl->channel[chl_delay_start].cur_cnt = 1;
       
    GLOBAL_INT_RESTORE();
}



/**************************************以下为旧版本********************************************/
high1_duty < 50%      
high2_duty > 50%
high1_duty+high2_duty < 100%
duty_cycle_lvl = 100

delay = low1_t-(low2_t-high1_t)/2
delay = (total_t-high1_t) - (total_t-high2_t-high1_t)/2 = total_t/2 + high2_t/2 - high1_t/2
total_t = 1/period s
high2_t = high2_duty*total_t / duty_cycle_lvl 
high1_t =  high1_duty*total_t / duty_cycle_lvl 

delay = 1/(2*period) + high2_duty/(2*duty_cycle_lvl*period) -  high1_duty/(2*duty_cycle_lvl*period)

delay = (1000*1000) * (duty_cycle_lvl+high2_duty- high1_duty)/(2*duty_cycle_lvl*period)  // us


start pwm1
delay
start pwm2


uint32_t pwm_set_period = 2000; // 2k
uint8_t duty_cycle1 = 50;
uint8_t duty_cycle2 = 40;
uint16_t duty_cycle_lvl = 100;

void dev_pwm_change(void)
{
    uint16_t delay_time = 0;

    system_set_port_mux(GPIO_PORT_D, GPIO_BIT_4, PORTD4_FUNC_PWM4);
    system_set_port_mux(GPIO_PORT_D, GPIO_BIT_5, PORTD5_FUNC_PWM5);

    pwm_stop(PWM_CHANNEL_4);
    pwm_stop(PWM_CHANNEL_5);
    
    duty_cycle2 = 90-duty_cycle1;
    pwm_init(PWM_CHANNEL_4,pwm_set_period,duty_cycle1);
    pwm_init(PWM_CHANNEL_5,pwm_set_period,duty_cycle2);
    
    if(duty_cycle2>duty_cycle1)
    {
        delay_time = (1000*1000)*(duty_cycle_lvl+duty_cycle2-duty_cycle1)/(2*pwm_set_period*duty_cycle_lvl*10); // unit 10us
        
        pwm_start(PWM_CHANNEL_4);
        GLOBAL_INT_DISABLE();
        co_delay_10us(delay_time);
        pwm_start(PWM_CHANNEL_5);
        GLOBAL_INT_RESTORE();
    }
    else
    {
        delay_time = (1000*1000)*(duty_cycle_lvl+duty_cycle1-duty_cycle2)/(2*pwm_set_period*duty_cycle_lvl*10); // unit 10us
        
        pwm_start(PWM_CHANNEL_5);
        GLOBAL_INT_DISABLE();
        co_delay_10us(delay_time);
        pwm_start(PWM_CHANNEL_4);
        GLOBAL_INT_RESTORE();
    }

    co_printf("=delay=%dus\r\n",delay_time);
    if(duty_cycle1 < 80)
        duty_cycle1 += 10;
    else
        duty_cycle1 = 10;   
}

调用参考:

富芮坤fr8016ha是一款自带计数的跳绳,跳绳的计数、暂停、复位等功能都可以通过按键进行控制。以下是基于Arduino平台的参考代码: #include <LiquidCrystal_I2C.h> //LCD库 #define BUTTON_COUNT 4 //按键数量 #define BUTTON_PIN 2 //按键引脚 #define LED_PIN 13 //LED引脚 #define LED_ON_DURATION 100 //LED亮时长(ms) #define LCD_WIDTH 16 //LCD宽度(字符) #define LCD_HEIGHT 2 //LCD高度(行数) LiquidCrystal_I2C lcd(0x27, LCD_WIDTH, LCD_HEIGHT); //创建LCD对象 int buttonPins[BUTTON_COUNT] = {2, 3, 4, 5}; //按键引脚列表 int buttonStates[BUTTON_COUNT] = {HIGH, HIGH, HIGH, HIGH}; //按键状态列表 int lastButtonStates[BUTTON_COUNT] = {HIGH, HIGH, HIGH, HIGH}; //上次按键状态列表 int buttonClicked[BUTTON_COUNT] = {0, 0, 0, 0}; //按键单击状态列表 int ledState = LOW; //LED当前状态 unsigned long ledOnTime = 0; //LED亮起时间 int jumpCount = 0; //跳绳计数 bool jumpPaused = false; //跳绳暂停状态 void setup() { pinMode(LED_PIN, OUTPUT); //设置LED引脚为输出 lcd.init(); //初始化LCD lcd.backlight(); //打开背光 lcd.setCursor(0, 0); //设置光标位置 lcd.print("Jump Count: 0"); //显示初始计数 } void loop() { //读取按键状态 for (int i = 0; i < BUTTON_COUNT; i++) { lastButtonStates[i] = buttonStates[i]; buttonStates[i] = digitalRead(buttonPins[i]); if (buttonStates[i] == LOW && lastButtonStates[i] == HIGH) { buttonClicked[i] = 1; } else { buttonClicked[i] = 0; } } //按键单击处理 if (buttonClicked[0]) { //计数 jumpCount++; if (jumpPaused) { jumpPaused = false; } displayJumpCount(); flashLed(); } if (buttonClicked[1]) { //暂停/恢复 jumpPaused = !jumpPaused; displayJumpCount(); flashLed(); } if (buttonClicked[2]) { //复位 jumpCount = 0; jumpPaused = false; displayJumpCount(); flashLed(); } if (buttonClicked[3]) { //测试 testButton(); } //LED计时控制 if (ledState == HIGH && millis() - ledOnTime >= LED_ON_DURATION) { ledState = LOW; digitalWrite(LED_PIN, ledState); } //跳绳暂停判断 if (jumpPaused) { while (digitalRead(buttonPins[1]) == LOW) { //等待恢复按键 delay(50); } } delay(50); } void displayJumpCount() { lcd.setCursor(0, 0); //设置光标位置 lcd.print("Jump Count: "); //显示计数标题 lcd.print(jumpCount); //显示实际计数值 lcd.setCursor(0, 1); //设置光标位置 if (jumpPaused) { lcd.print("Jump Paused"); } else { lcd.print(" "); //清空暂停提示文本 } } void flashLed() { ledState = HIGH; digitalWrite(LED_PIN, ledState); ledOnTime = millis(); } void testButton() { for (int i = 0; i < BUTTON_COUNT; i++) { lcd.setCursor(0, i); lcd.print("\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF"); while (digitalRead(buttonPins[i]) != LOW) { delay(50); } lcd.setCursor(0, i); lcd.print(buttonPins[i]); delay(1000); } } 代码中定义了4个按键分别对应计数、暂停/恢复、复位、测试操作。其中测试操作是为了调试按键,非正式的功能。 跳绳计数直接在计数按键响应中进行,每次计数时显示计数值并闪烁LED。跳绳暂停和恢复则在暂停/恢复按键响应中进行,切换暂停状态并显示相应的文本。跳绳复位则在复位按键响应中进行,将计数清零并恢复计数。另外需要注意的是,跳绳暂停时如果用户一直按住暂停/恢复按键则应该在恢复前等待用户松开按键。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值