340 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			340 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* os shim includes */
 | 
						||
#include "os_types.h"
 | 
						||
#include "dbg_io.h"
 | 
						||
#include "iot_io.h"
 | 
						||
#include "gpio_mtx.h"
 | 
						||
#include "pwm.h"
 | 
						||
#include "cpu.h"
 | 
						||
 | 
						||
void delay_sometime(uint32_t times)
 | 
						||
{
 | 
						||
    uint32_t loop = 0;
 | 
						||
 | 
						||
    while (loop++ < times) {
 | 
						||
        __asm volatile ("nop\n");
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
void register_print_by_addr(uint32_t addr, int num)
 | 
						||
{
 | 
						||
    volatile uint32_t *p_addr = (volatile uint32_t *)addr;
 | 
						||
    int i;
 | 
						||
 | 
						||
    for (i = 0; i < num; i++) {
 | 
						||
        iot_printf("%08x,", *p_addr);
 | 
						||
        p_addr ++;
 | 
						||
    }
 | 
						||
    iot_printf("\r\n");
 | 
						||
}
 | 
						||
 | 
						||
/*
 | 
						||
PWM 模块同步使用,PWM0为主,PWM1和PWM2 被PWM0同步,相位关系由pwm_tb_phase_set 控制
 | 
						||
*/
 | 
						||
void pwm_main2(void)
 | 
						||
{
 | 
						||
    uint8_t ch = PWM_CHANNEL_0;
 | 
						||
 | 
						||
    dbg_uart_init();
 | 
						||
    pwm_tbclk_sync(ch, false);
 | 
						||
 | 
						||
    //                  8   7
 | 
						||
    pwm_gpio_config(ch, 26, 27);
 | 
						||
 | 
						||
    pwm_tbclk_ctrl(ch, true); // 使能PWM0 的时钟
 | 
						||
    pwm_tb_phase_set(ch, 0);
 | 
						||
    pwm_tbctl_set(ch, TB_DISABLE, TB_SHADOW, TB_SYNC_CNT_ZERO,99);
 | 
						||
 | 
						||
    //pwm waveform period
 | 
						||
    pwm_tb_period_set(ch, 9999);
 | 
						||
 | 
						||
    //load from shadow select mode
 | 
						||
    pwm_cmpctl_set(ch, CC_CTR_ZERO, CC_CTR_ZERO, CC_NO_SHADOW, CC_NO_SHADOW);
 | 
						||
 | 
						||
    //action when counter equals zero or CMPA register
 | 
						||
 | 
						||
    pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_cmpa_cmpb_set(ch, 3000, 6000);
 | 
						||
 | 
						||
    pwm_cnt_mode_set(ch, TB_COUNT_UP);
 | 
						||
 | 
						||
    pwm_db_inmode_set(ch, DB_INMODE_PWMA_RAW, DB_INMODE_PWMB_RAW);
 | 
						||
    pwm_db_polsel_set(ch, DB_POLSE_NORMAL, DB_POLSE_INVERTED);    //A不反相,B反相
 | 
						||
    //pwm_db_outmode_set(ch, DB_OUTMODE_BYPASS, DB_OUTMODE_DELAYED );//A不产生死区,B产生
 | 
						||
    //pwm_db_outmode_set(ch, DB_OUTMODE_BYPASS, DB_OUTMODE_BYPASS ); //都不产生死区
 | 
						||
 | 
						||
    pwm_db_rising_delay_set(ch, 1000);
 | 
						||
    pwm_db_falling_delay_set(ch, 1000);
 | 
						||
 | 
						||
//-----------------------------------------------------------------------------
 | 
						||
 | 
						||
    ch = PWM_CHANNEL_1;
 | 
						||
 | 
						||
    //                  10   9
 | 
						||
    pwm_gpio_config(ch, 28, 29);
 | 
						||
 | 
						||
    pwm_tbclk_ctrl(ch, true); // 使能PWM1 的时钟
 | 
						||
    pwm_tb_phase_set(ch, 3333);
 | 
						||
    pwm_tbctl_set(ch, TB_ENABLE, TB_SHADOW, TB_SYNC_PWM,99);
 | 
						||
 | 
						||
    //pwm waveform period
 | 
						||
    pwm_tb_period_set(ch, 9999);
 | 
						||
 | 
						||
    //load from shadow select mode
 | 
						||
    pwm_cmpctl_set(ch, CC_CTR_ZERO, CC_CTR_ZERO, CC_NO_SHADOW, CC_NO_SHADOW);
 | 
						||
 | 
						||
    //action when counter equals zero or CMPA register
 | 
						||
 | 
						||
    pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_cmpa_cmpb_set(ch, 3000, 6000);
 | 
						||
 | 
						||
    pwm_cnt_mode_set(ch, TB_COUNT_UP);
 | 
						||
 | 
						||
    pwm_db_inmode_set(ch, DB_INMODE_PWMA_RAW, DB_INMODE_PWMB_RAW);
 | 
						||
    pwm_db_polsel_set(ch, DB_POLSE_NORMAL, DB_POLSE_INVERTED);
 | 
						||
 | 
						||
    pwm_db_outmode_set(ch, DB_OUTMODE_BYPASS, DB_OUTMODE_BYPASS );
 | 
						||
 | 
						||
    pwm_db_rising_delay_set(ch, 1000);
 | 
						||
    pwm_db_falling_delay_set(ch, 1000);
 | 
						||
 | 
						||
    //-------------------------------------------------------------------------
 | 
						||
 | 
						||
    ch = PWM_CHANNEL_2;
 | 
						||
 | 
						||
    //                  19   38
 | 
						||
    pwm_gpio_config(ch, 9, 12);
 | 
						||
 | 
						||
    pwm_tbclk_ctrl(ch, true); // 使能PWM2 的时钟
 | 
						||
    pwm_tb_phase_set(ch, 6666);
 | 
						||
    pwm_tbctl_set(ch, TB_ENABLE, TB_SHADOW, TB_SYNC_PWM,99);
 | 
						||
 | 
						||
    //pwm waveform period
 | 
						||
    pwm_tb_period_set(ch, 9999);
 | 
						||
 | 
						||
    //load from shadow select mode
 | 
						||
    pwm_cmpctl_set(ch, CC_CTR_ZERO, CC_CTR_ZERO, CC_NO_SHADOW, CC_NO_SHADOW);
 | 
						||
 | 
						||
    //action when counter equals zero or CMPA register
 | 
						||
 | 
						||
    pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_cmpa_cmpb_set(ch, 3000, 6000);
 | 
						||
 | 
						||
    pwm_cnt_mode_set(ch, TB_COUNT_UP);
 | 
						||
 | 
						||
    //选择延迟输入信号
 | 
						||
    pwm_db_inmode_set(ch, DB_INMODE_PWMB_RAW, DB_INMODE_PWMB_RAW);
 | 
						||
    pwm_db_polsel_set(ch, DB_POLSE_NORMAL, DB_POLSE_NORMAL);
 | 
						||
 | 
						||
    pwm_db_outmode_set(ch, DB_OUTMODE_BYPASS, DB_OUTMODE_BYPASS );
 | 
						||
 | 
						||
    pwm_db_rising_delay_set(ch, 100);
 | 
						||
    pwm_db_falling_delay_set(ch, 200);
 | 
						||
 | 
						||
    //-------------------------------------------------------------------------
 | 
						||
 | 
						||
    pwm_tbclk_sync(ch, true);
 | 
						||
 | 
						||
    while (1) {
 | 
						||
        delay_sometime(1000000);
 | 
						||
        delay_sometime(1000000);
 | 
						||
        delay_sometime(1000000);
 | 
						||
        delay_sometime(1000000);
 | 
						||
        delay_sometime(1000000);
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
//PWM 模块输出两路占空比可以独立调节的PWM波形,需要注意的点是在不同的
 | 
						||
//占空比组合下需要对AQ的特别设置
 | 
						||
void pwm_main3(void)
 | 
						||
{
 | 
						||
    uint8_t ch = PWM_CHANNEL_0;
 | 
						||
 | 
						||
    dbg_uart_init();
 | 
						||
    pwm_tbclk_sync(ch, false);
 | 
						||
 | 
						||
    //                  8   7
 | 
						||
    pwm_gpio_config(ch, 26, 27);
 | 
						||
 | 
						||
    pwm_tbclk_ctrl(ch, true); // 使能PWM0 的时钟
 | 
						||
    pwm_tb_phase_set(ch, 0);
 | 
						||
    pwm_tbctl_set(ch, TB_DISABLE, TB_SHADOW, TB_SYNC_CNT_ZERO, 99);
 | 
						||
 | 
						||
    //pwm waveform period
 | 
						||
    pwm_tb_period_set(ch, 9999);
 | 
						||
 | 
						||
    //load from shadow select mode
 | 
						||
    //注意这里必须都设为CC_CTR_PRD,如果为CC_CTR_ZERO 输出的波形不对,
 | 
						||
    //在B的占空比为0时,A会少一个脉冲
 | 
						||
    pwm_cmpctl_set(ch, CC_CTR_PRD, CC_CTR_PRD, CC_SHADOW, CC_SHADOW);
 | 
						||
 | 
						||
    //action when counter equals zero or CMPA register
 | 
						||
    pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_cnt_mode_set(ch, TB_COUNT_UP);
 | 
						||
 | 
						||
    // 独立输出PWM 0-100都可以,给duty_a和duty_b初值分别赋:
 | 
						||
    // 0    0       A B输出相同占空比的PWM波形
 | 
						||
    // 0    100     A 为0时B不为0
 | 
						||
    // 100  0       B 为0时A不为0
 | 
						||
    //
 | 
						||
    int duty_a = 0;
 | 
						||
    int duty_b = 10000;
 | 
						||
 | 
						||
    pwm_cmpa_cmpb_set(ch, duty_a, duty_b);
 | 
						||
 | 
						||
    pwm_tbclk_sync(ch, true);
 | 
						||
    while (1) {
 | 
						||
        //因为cntb_up的优先级高于cnta_up,所以最终起作用的是b,故应改变b的配置
 | 
						||
        if (duty_a == duty_b) {
 | 
						||
            pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
            pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_CLEAR, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
        } else if (duty_a == 0) {
 | 
						||
            pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
 | 
						||
            pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_SET, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
        } else if (duty_b == 0) {
 | 
						||
            pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_CLEAR, AQ_NOTHING, AQ_SET, AQ_NOTHING);
 | 
						||
 | 
						||
            pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
        } else {
 | 
						||
            pwm_aqctlb_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
            pwm_aqctla_set(ch, AQ_SET, AQ_NOTHING,
 | 
						||
                AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
        }
 | 
						||
        pwm_cmpa_cmpb_set(ch, duty_a, duty_b);
 | 
						||
        duty_a += 100;
 | 
						||
        if (duty_a > 10000) duty_a = 0;
 | 
						||
 | 
						||
        duty_b += 100;
 | 
						||
        if (duty_b > 10000) duty_b = 0;
 | 
						||
 | 
						||
        delay_sometime(10000);
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
int g_duty_a = 100;
 | 
						||
int g_duty_b = 200;
 | 
						||
uint8_t g_ch = PWM_CHANNEL_3;
 | 
						||
//每个周期都改变占空比
 | 
						||
uint32_t interrupt_cb(void)
 | 
						||
{
 | 
						||
    g_duty_a += 100;
 | 
						||
    if (g_duty_a > 10000) {
 | 
						||
        g_duty_a = 100;
 | 
						||
    }
 | 
						||
 | 
						||
    g_duty_b += 100;
 | 
						||
    if (g_duty_b > 10000) {
 | 
						||
        g_duty_b = 200;
 | 
						||
    }
 | 
						||
 | 
						||
    pwm_cmpa_cmpb_set(g_ch, g_duty_a, g_duty_b);
 | 
						||
 | 
						||
    return 0;
 | 
						||
}
 | 
						||
 | 
						||
// 中断测试
 | 
						||
void pwm_main4(void)
 | 
						||
{
 | 
						||
    uint32_t cpu_id = cpu_get_mhartid();
 | 
						||
    dbg_uart_init();
 | 
						||
    iot_interrupt_init(cpu_id);
 | 
						||
 | 
						||
    pwm_tbclk_sync(g_ch, false);
 | 
						||
 | 
						||
    //                           8   7
 | 
						||
    pwm_gpio_config_with_isr(g_ch, 26, 27, interrupt_cb);
 | 
						||
 | 
						||
    pwm_tbclk_ctrl(g_ch, true); // 使能PWM 的时钟
 | 
						||
    pwm_tb_phase_set(g_ch, 0);
 | 
						||
    pwm_tbctl_set(g_ch, TB_DISABLE, TB_SHADOW, TB_SYNC_CNT_ZERO,99);
 | 
						||
 | 
						||
    //pwm waveform period
 | 
						||
    pwm_tb_period_set(g_ch, 9999);
 | 
						||
 | 
						||
    //load from shadow select mode
 | 
						||
    //注意:若A或B 的ld mode被配置为 CC_CTR_FREEZE 则B的占空比保持着不再变化,
 | 
						||
    //却决于原来的状态,若是上电第一次运行,则B保持全低或者全高
 | 
						||
    pwm_cmpctl_set(g_ch, CC_CTR_PRD, CC_CTR_PRD, CC_SHADOW, CC_SHADOW);
 | 
						||
 | 
						||
    //action when counter equals zero or CMPA register
 | 
						||
    pwm_aqctlb_set(g_ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_NOTHING, AQ_NOTHING, AQ_CLEAR, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_aqctla_set(g_ch, AQ_SET, AQ_NOTHING,
 | 
						||
        AQ_CLEAR, AQ_NOTHING, AQ_NOTHING, AQ_NOTHING);
 | 
						||
 | 
						||
    pwm_cnt_mode_set(g_ch, TB_COUNT_UP);
 | 
						||
 | 
						||
    //pwm_interrupt_init(g_ch);
 | 
						||
 | 
						||
    pwm_cmpa_cmpb_set(g_ch, g_duty_a, g_duty_b);
 | 
						||
 | 
						||
    pwm_tbclk_sync(g_ch, true);
 | 
						||
 | 
						||
    while (1);
 | 
						||
}
 | 
						||
 | 
						||
//输出 h 和 l输出互补的pwm 波形
 | 
						||
void pwm_main5()
 | 
						||
{
 | 
						||
    uint8_t ch = PWM_CHANNEL_0;
 | 
						||
    float duty = 0.01;
 | 
						||
 | 
						||
    dbg_uart_init();
 | 
						||
    pwm_gpio_config(ch, 58, 59);
 | 
						||
 | 
						||
    pwm_init(ch);
 | 
						||
    pwm_start(ch);
 | 
						||
 | 
						||
    pwm_set_duty(ch, 50 * 100);
 | 
						||
    while (1) {
 | 
						||
        //pwm_set_duty(ch, duty * 100);
 | 
						||
        duty += 0.01;
 | 
						||
        if (duty > 100) {
 | 
						||
            duty = 0.01;
 | 
						||
        }
 | 
						||
 | 
						||
        delay_sometime(1000);
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
#ifdef __GNUC__
 | 
						||
 | 
						||
int main(void)
 | 
						||
{
 | 
						||
    pwm_main5();
 | 
						||
    return 0;
 | 
						||
}
 | 
						||
 | 
						||
#endif // __GCC__
 | 
						||
 |