我在用stc12c5a60s2做呼吸燈設計時遇到的問題,我想設計出幾路頻率變化不一樣的pwm波形,我第一個是利用了C51里面的兩個計數器實現了呼吸燈設計,可以使三個燈做相同頻率的呼吸,代碼如下://實現三個燈的呼吸
- #include <reg52.h>
- sbit PWMOUT0 = P0^0;
- sbit PWMOUT1 = P0^1;
- sbit PWMOUT2 = P0^2;
- unsigned char HighRH = 0; //高電平重載值的高字節
- unsigned char HighRL = 0; //高電平重載值的低字節
- unsigned char LowRH = 0; //低電平重載值的高字節
- unsigned char LowRL = 0; //低電平重載值的低字節
- unsigned char T1RH = 0;
- unsigned char T1RL = 0;
- unsigned long PeriodCnt = 0;
- void ConfigTimer1(unsigned int ms);
- void ConfigPWM(unsigned int fr, unsigned char dc);
- void main()
- {
- EA = 1; //開定時器總中斷
- ConfigPWM(100, 10); //配置并啟動PWM
- ConfigTimer1(50); //用T1定時調整占空比
- while (1);
- }
- void ConfigTimer1(unsigned int ms)
- {
- unsigned long tmp; //臨時變量
- tmp = 11059200 / 12; //定時器計數頻率
- tmp = (tmp * ms) / 1000; //計算所需的計數值
- tmp = 65536 - tmp; //計算定時器重載值
- tmp = tmp + 12; //補償中斷響應延時造成的誤差
- T1RH = (unsigned char)(tmp>>8); //定時器重載值拆分為高低字節
- T1RL = (unsigned char)tmp;
- TMOD &= 0x0F; //清零T1的控制位
- TMOD |= 0x10; //配置T1為模式1
- TH1 = T1RH; //加載T1重載值
- TL1 = T1RL;
- ET1 = 1; //使能T1中斷
- TR1 = 1; //啟動T1
- }
- void ConfigPWM(unsigned int fr, unsigned char dc)
- {
- unsigned int high, low;
- PeriodCnt = (11059200/12) / fr; //計算一個周期所需的計數值
- high = (PeriodCnt*dc) / 100; //計算高電平所需的計數值
- low = PeriodCnt - high; //計算低電平所需的計數值
- high = 65536 - high + 12; //計算高電平的定時器重載值并補償中斷延時
- low = 65536 - low + 12; //計算低電平的定時器重載值并補償中斷延時
- HighRH = (unsigned char)(high>>8); //高電平重載值拆分為高低字節
- HighRL = (unsigned char)high;
- LowRH = (unsigned char)(low>>8); //低電平重載值拆分為高低字節
- LowRL = (unsigned char)low;
- TMOD &= 0xF0; //清零T0的控制位
- TMOD |= 0x01; //配置T0為模式1
- TH0 = HighRH; //加載T0重載值
- TL0 = HighRL;
- ET0 = 1; //使能T0中斷
- TR0 = 1; //啟動T0
- PWMOUT0 = 1; //輸出高電平
- PWMOUT1 = 1; //輸出高電平
- PWMOUT2 = 1; //輸出高電平
- }
- void AdjustDutyCycle(unsigned char dc) //調整占空比
- {
- unsigned int high, low;
- high = (PeriodCnt*dc)/100;
- low = PeriodCnt - high;
- high = 65536 - high + 12;
- low = 65536 - low + 12;
- HighRH = (unsigned char)(high >> 8);
- HighRL = (unsigned char)high;
- LowRH = (unsigned char)(low >> 8);
- LowRL = (unsigned char)low;
- }
- void InterruptTimer0() interrupt 1 //用定時器0,實現了燈的亮滅連續變化,實現電平的反轉
- {
- if (PWMOUT0 == 1) //當前輸出為高電平時,裝載低電平值并輸出低電平
- {
- TH0 = LowRH;
- TL0 = LowRL;
- PWMOUT0 = 0;
- PWMOUT1 = 0;
- PWMOUT2 = 0;
- }
- else //當前輸出為低電平時,裝載高電平值并輸出高電平
- {
- TH0 = HighRH;
- TL0 = HighRL;
- PWMOUT0 = 1;
- PWMOUT1 = 1;
- PWMOUT2 = 1;
- }
- }
- void InterruptTimer1() interrupt 3
- {
- unsigned char code table[13] = {
- 5, 18, 30, 41, 51, 60, 68, 75, 81, 86, 90, 93, 95
- };
- static bit dir = 0;
- static unsigned char index = 0;
- TH1 = T1RH;
- TL1 = T1RL;
- AdjustDutyCycle(table[index]);
- if(dir == 0)
- {
- index++;
- if(index >= 12)
- {
- dir = 1;
- }
- }
- else
- {
- index--;
- if(index == 0)
- {
- dir = 0;
- }
- }
- }
- 可是,我并不想看到一樣的頻率變化,我希望三個燈變化的頻率不一樣,可是上述代碼實現呼吸燈的效果是基于計數器的個數的,C51的計數器已經用完了,我只好把目光投向了stc12c5a60s,這里面不僅有四個計數器,而且有兩路PWM輸出,
- 我通過設置PCA,實現了通過按鍵可以實時改變輸出波形,
- 代碼如下:
- //利用按鍵s1對led燈的亮度調節
- #include"STC12C5A60S2.h"
- void delay(unsigned int cnt)
- {
- unsigned char i;
- for(;cnt>0;cnt--)
- for(i=0;i<250;i++);
- }
- void main()
- {
- EA = 1; //開定時器總中斷
- CCON=0; //禁止寄存器CCON中CF位的中斷
- CL=0; //PCA的16位計數器低8位
- CH=0; //PCA的16位計數器高8位
- CMOD=0x00; //選擇系統時鐘/12為計數脈沖,則PWM的頻率f=sysclk/256/12
- //PCA模塊0
- CCAP0H=0x80; //占空比控制,此時的占空比為50%,一路輸出
- CCAP0L=0x80;
- PCA_PWM0=0x00; //控制占空比的第九位為0
- CCAPM0=0x42; //允許P13作為PWM輸出
- //PCA模塊1
- CCAP1H=0xcc; //占空比控制,此時的占空比為20%,一路輸出
- CCAP1L=0xcc;
- PCA_PWM1=0x00;
- CCAPM1=0x42; //允許P13作為PWM輸出
- CR=1; //啟動PCA計數器
- while(1)
- {
- if(P10==0)
- {
- delay(200);
- while(P10==0);
- CCAP0H+=10; //占空比調節
- CCAP0L+=10;
- CCAP1H+=25; //占空比調節
- CCAP1L+=25;
- }
- }
- }
- 可是就在我想結合兩者的特點一起實現三個燈做呼吸燈,另外兩個燈通過按鍵做使其的波形不斷變化,因為這在設計上我覺得是行得通的,因為我做呼吸燈時,只涉及到51的定時器,而stc12c5a60s2很好的兼容了51的兩個定時器,這樣的話,應該是可以同時出現三個燈是呼吸燈,最后兩個燈通過按鍵調節亮度,所以我就把兩個函數粘貼在一起了,可是卻沒有出現應有的現象。
- 問題如下:只能通過按鍵對最后兩個燈實現亮度調節,而呼吸燈的現象出不來。
- 有問題的程序如下,不知道怎樣修改,還請大神解惑。
- //利用按鍵s1對led燈的亮度調節
- #include"STC12C5A60S2.h"
- sbit PWMOUT0 = P0^0;
- sbit PWMOUT1 = P0^1;
- sbit PWMOUT2 = P0^2;
- unsigned char HighRH = 0; //高電平重載值的高字節
- unsigned char HighRL = 0; //高電平重載值的低字節
- unsigned char LowRH = 0; //低電平重載值的高字節
- unsigned char LowRL = 0; //低電平重載值的低字節
- unsigned char T1RH = 0;
- unsigned char T1RL = 0;
- unsigned long PeriodCnt = 0;
- void ConfigTimer1(unsigned int ms);
- void ConfigPWM(unsigned int fr, unsigned char dc);
- void delay(unsigned int cnt)
- {
- unsigned char i;
- for(;cnt>0;cnt--)
- for(i=0;i<250;i++);
- }
- void main()
- {
- EA = 1; //開定時器總中斷
- CCON=0; //禁止寄存器CCON中CF位的中斷
- CL=0; //PCA的16位計數器低8位
- CH=0; //PCA的16位計數器高8位
- CMOD=0x00; //選擇系統時鐘/12為計數脈沖,則PWM的頻率f=sysclk/256/12
- //PCA模塊0
- CCAP0H=0x80; //占空比控制,此時的占空比為50%,一路輸出
- CCAP0L=0x80;
- PCA_PWM0=0x00; //控制占空比的第九位為0
- CCAPM0=0x42; //允許P13作為PWM輸出
- //PCA模塊1
- CCAP1H=0xcc; //占空比控制,此時的占空比為80%,一路輸出
- CCAP1L=0xcc;
- PCA_PWM1=0x00;
- CCAPM1=0x42; //允許P13作為PWM輸出
- CR=1; //啟動PCA計數器
- while(1)
- {
- ConfigPWM(100, 10); //配置并啟動PWM
- ConfigTimer1(50); //用T1定時調整占空比
- if(P10==0)
- {
- delay(200);
- while(P10==0);
- CCAP0H+=10; //占空比調節
- CCAP0L+=10;
- CCAP1H+=25; //占空比調節
- CCAP1L+=25;
- }
- }
- }
- void ConfigTimer1(unsigned int ms)
- {
- unsigned long tmp; //臨時變量
- tmp = 11059200 / 12; //定時器計數頻率
- tmp = (tmp * ms) / 1000; //計算所需的計數值
- tmp = 65536 - tmp; //計算定時器重載值
- tmp = tmp + 12; //補償中斷響應延時造成的誤差
- T1RH = (unsigned char)(tmp>>8); //定時器重載值拆分為高低字節
- T1RL = (unsigned char)tmp;
- TMOD &= 0x0F; //清零T1的控制位
- TMOD |= 0x10; //配置T1為模式1
- TH1 = T1RH; //加載T1重載值
- TL1 = T1RL;
- ET1 = 1; //使能T1中斷
- TR1 = 1; //啟動T1
- }
- void ConfigPWM(unsigned int fr, unsigned char dc)
- {
- unsigned int high, low;
- PeriodCnt = (11059200/12) / fr; //計算一個周期所需的計數值
- high = (PeriodCnt*dc) / 100; //計算高電平所需的計數值
- low = PeriodCnt - high; //計算低電平所需的計數值
- high = 65536 - high + 12; //計算高電平的定時器重載值并補償中斷延時
- low = 65536 - low + 12; //計算低電平的定時器重載值并補償中斷延時
- HighRH = (unsigned char)(high>>8); //高電平重載值拆分為高低字節
- HighRL = (unsigned char)high;
- LowRH = (unsigned char)(low>>8); //低電平重載值拆分為高低字節
- LowRL = (unsigned char)low;
- TMOD &= 0xF0; //清零T0的控制位
- TMOD |= 0x01; //配置T0為模式1
- TH0 = HighRH; //加載T0重載值
- TL0 = HighRL;
- ET0 = 1; //使能T0中斷
- TR0 = 1; //啟動T0
- PWMOUT0 = 1; //輸出高電平
- PWMOUT1 = 1; //輸出高電平
- PWMOUT2 = 1; //輸出高電平
- }
- void AdjustDutyCycle(unsigned char dc) //調整占空比
- {
- unsigned int high, low;
- high = (PeriodCnt*dc)/100;
- low = PeriodCnt - high;
- high = 65536 - high + 12;
- low = 65536 - low + 12;
- HighRH = (unsigned char)(high >> 8);
- HighRL = (unsigned char)high;
- LowRH = (unsigned char)(low >> 8);
- LowRL = (unsigned char)low;
- }
- void InterruptTimer0() interrupt 1 //用定時器0,實現了燈的亮滅連續變化,實現電平的反轉
- {
- if (PWMOUT0 == 1) //當前輸出為高電平時,裝載低電平值并輸出低電平
- {
- TH0 = LowRH;
- TL0 = LowRL;
- PWMOUT0 = 0;
- PWMOUT1 = 0;
- PWMOUT2 = 0;
- }
- else //當前輸出為低電平時,裝載高電平值并輸出高電平
- {
- TH0 = HighRH;
- TL0 = HighRL;
- PWMOUT0 = 1;
- PWMOUT1 = 1;
- PWMOUT2 = 1;
- }
- }
- void InterruptTimer1() interrupt 3
- {
- unsigned char code table[13] = {
- 5, 18, 30, 41, 51, 60, 68, 75, 81, 86, 90, 93, 95
- };
- static bit dir = 0;
- static unsigned char index = 0;
- TH1 = T1RH;
- TL1 = T1RL;
- AdjustDutyCycle(table[index]);
- if(dir == 0)
- {
- index++;
- if(index >= 12)
- {
- dir = 1;
- }
- }
- else
- {
- index--;
- if(index == 0)
- {
- dir = 0;
- }
- }
- }
復制代碼
|