找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2093|回復(fù): 0
收起左側(cè)

STM32F103單片機2通道輸出相位任意配置的PWM輸出程序

[復(fù)制鏈接]
ID:928738 發(fā)表于 2023-3-31 14:40 | 顯示全部樓層 |閱讀模式
藍橋杯競賽的時候練習(xí)的程序,嵌入式競賽的項目中定時器的一個難點,其實也挺簡單的,主要是理解了就簡單。里面的是一個keil4工程的代碼,使用tim3實現(xiàn)同一個定時器輸出任意占空比、任意相位差的pwm信號,里面有注釋和原理分析,使用示波器驗證過。有問題可以留言提問

單片機源程序如下:
  1. #include "stm32f10x.h"
  2. #include "lcd.h"
  3. /*
  4. 兩路任意相位差、任意占空比的PWM信號實現(xiàn)
  5. */


  6. #define FRE 5000    //頻率
  7. #define DUTY 40                        //占空比
  8. #define PHASE 180                //相位差
  9. u32 TimingDelay = 0;

  10. /*
  11. ccr1 高電平時間  ccr2 低電平時間
  12. cnt 頻率對應(yīng)的計數(shù)個數(shù)值
  13. deg 兩個信號在CCR翻轉(zhuǎn)的差值  這個差值體現(xiàn)相位差
  14. */

  15. u16 ccr1,ccr2;
  16. u16 cnt;
  17. u16 deg;

  18. void Delay_Ms(u32 nTime);
  19. void nvic_config(void);
  20. void tim3_init(void);
  21. void cal_prama(void);


  22. //Main Body
  23. int main(void)
  24. {
  25.                 SysTick_Config(SystemCoreClock/1000);//1ms
  26.                 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
  27.                 cal_prama();
  28.                 nvic_config();
  29.                 tim3_init();
  30.                
  31.                 while(1);
  32. }
  33. //
  34. void Delay_Ms(u32 nTime)
  35. {
  36.                 TimingDelay = nTime;
  37.                 while(TimingDelay != 0);        
  38. }
  39. void tim3_init(void)
  40. {
  41.     TIM_TimeBaseInitTypeDef  TIM_TimeBaseInitStructure;
  42.                 TIM_OCInitTypeDef  TIM_OCInitStructure;
  43.                 GPIO_InitTypeDef  GPIO_InitStructure;
  44.         
  45.           TIM_DeInit(TIM3);
  46.                 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
  47.           RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
  48.         
  49.           GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
  50.           GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  51.           GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  52.           GPIO_Init(GPIOA,&GPIO_InitStructure);
  53.         
  54.          TIM_TimeBaseInitStructure.TIM_Prescaler = 72-1;//1us
  55.    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
  56.    TIM_TimeBaseInitStructure.TIM_Period         = 65535;//1us*65535 = 65.535ms
  57.    TIM_TimeBaseInitStructure.TIM_ClockDivision = 0;
  58.    TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
  59.    TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);

  60.                  /*
  61.                 兩個通道的CCR值初始為0,在第一次進入翻轉(zhuǎn)的時候,改變兩個通道的相位差。
  62.                 兩個通道極性設(shè)置為一樣。具體是什么電平有效取決于要求。改這里即可。
  63.                 TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Disable); 這個必須設(shè)置為DISABLE
  64.                 */
  65.                 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;//TIM3 CH1
  66.                 TIM_OCInitStructure.TIM_OutputState        = TIM_OutputState_Enable;
  67.                 TIM_OCInitStructure.TIM_Pulse = 0;   
  68.                 TIM_OCInitStructure.TIM_OCPolarity        = TIM_OCPolarity_High;
  69.                 TIM_OC1Init(TIM3,&TIM_OCInitStructure);
  70.                 TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Disable);
  71.                
  72.                 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;//TIM3 CH2
  73.                 TIM_OCInitStructure.TIM_OutputState        = TIM_OutputState_Enable;
  74.                 TIM_OCInitStructure.TIM_Pulse = 0;        
  75.                 TIM_OCInitStructure.TIM_OCPolarity        = TIM_OCPolarity_High;
  76.                 TIM_OC2Init(TIM3,&TIM_OCInitStructure);
  77.                 TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Disable);
  78.                
  79.                 TIM_ITConfig(TIM3,TIM_IT_CC1|TIM_IT_CC2, ENABLE);
  80.                 TIM_Cmd(TIM3,ENABLE);

  81. }
  82. void nvic_config(void)
  83. {
  84.                 NVIC_InitTypeDef NVIC_InitType;
  85.                 NVIC_InitType.NVIC_IRQChannel=TIM3_IRQn;
  86.                 NVIC_InitType.NVIC_IRQChannelCmd=ENABLE;
  87.                 NVIC_InitType.NVIC_IRQChannelPreemptionPriority=0x01;
  88.                 NVIC_InitType.NVIC_IRQChannelSubPriority=0x02;
  89.                 NVIC_Init(&NVIC_InitType);
  90. }
  91. void cal_prama(void)
  92. {
  93.                 cnt = 1000000/FRE;  //信號頻率對應(yīng)的計數(shù)個數(shù)
  94.                 ccr1 = (int)(cnt*DUTY/100.0); //高電平計數(shù)個數(shù)
  95.                 ccr2 = cnt-ccr1;     //低電平計數(shù)個數(shù)
  96.                 deg = cnt*(PHASE/360.0);   //設(shè)置到ccr上的差值,這個差值體現(xiàn)相位差
  97. }
  98. u16 out;
  99. void TIM3_IRQHandler(void)
  100. {
  101.                 static u8 num1=0,num2=2;
  102.                 if (TIM_GetITStatus(TIM3, TIM_IT_CC1) != RESET)
  103.                 {
  104.                                 TIM_ClearITPendingBit(TIM3, TIM_IT_CC1 );
  105.                                 out=TIM_GetCapture1(TIM3);
  106.                                 if(num1==0)
  107.                                 {
  108.                                                 num1=1;
  109.                                           TIM_SetCompare1(TIM3,out+ccr1);
  110.                                 }
  111.                                 else
  112.                                 {
  113.                                     num1=0;
  114.                                           TIM_SetCompare1(TIM3,out+ccr2);
  115.                                 }
  116.                 }
  117.                 if (TIM_GetITStatus(TIM3, TIM_IT_CC2) != RESET)
  118.                 {
  119.                                 TIM_ClearITPendingBit(TIM3, TIM_IT_CC2 );
  120.                                 out=TIM_GetCapture2(TIM3);
  121.                                 /*
  122.                                 只有在第一次的時候才需要加deg相位差值。
  123.                                 這個加了后兩者的相位就已經(jīng)改變了。
  124.                                 后面的部分不需要加deg,加了的話,頻率就改變了
  125.                                 if(num2==0)
  126.                                 {
  127.                                         num2=1;
  128.                                         TIM_SetCompare2(TIM3,out+ccr1+deg);
  129.                                 }
  130.                                 else
  131.                                 {
  132.                                         num2=0;
  133.                                         TIM_SetCompare2(TIM3,out+ccr2+deg);
  134.                                 }
  135.                                 這時候頻率就變?yōu)閒=fcnt/(ccr1+ccr2+deg*2);
  136.                                 而我們信號的頻率應(yīng)該是f=fcnt/(ccr1+ccr2);
  137.                                 因此注意點:
  138.                                                  只有第一次設(shè)置CCR時需要加上deg值。這時相位就改變了
  139.                                                  后面設(shè)置的時候,跟通道1一樣。 這樣兩個通道的頻率和占空比就是一樣的。
  140.                                                  這樣設(shè)置后就可以產(chǎn)生:相位差不一樣,占空比和頻率一樣的兩路PWM信號了。
  141.                                                 
  142.                                                  這個ccr1和ccr2是代表高低電平持續(xù)時間的個數(shù),也就是決定了頻率和占空比,
  143.                                                  因此兩路信號的設(shè)置的ccr1和ccr2應(yīng)該一樣。所以第一次設(shè)置要加上deg,后面設(shè)置
  144.                                                  跟另一通道一樣,就不要加deg了,加了后頻率就改變了。
  145.                                                 
  146.                                                 
  147.                                                  具體實現(xiàn)為下面代碼:
  148.                                 */
  149.                                 if(num2==2)
  150.                                 {
  151.                                                 num2=1;
  152.                                                 TIM_SetCompare2(TIM3,out+ccr1+deg);
  153.                                 }
  154.                                  else if(num2==0)
  155.                                 {
  156.                                                 num2=1;
  157.                                                 TIM_SetCompare2(TIM3,out+ccr1);
  158.                                 }
  159.                                 else
  160.                                 {
  161.                                                 num2=0;
  162.                                                 TIM_SetCompare2(TIM3,out+ccr2);
  163.                                 }
  164.                         
  165.                         
  166.                 }


  167. }
復(fù)制代碼

Keil代碼下載(附件只有代碼不含其他任何資料):
Keil程序.7z (183.98 KB, 下載次數(shù): 34)

評分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機教程網(wǎng)

快速回復(fù) 返回頂部 返回列表