这个官方提供的案例,里面是做了一个变频p3.4变频输出的小程序,本人想改成固定频率的可控占空比的功能,但是看不懂他频率算法占空比算法。新人求指教,在线等啊
#define MAIN_Fosc 22118400L //定义主时钟
#define PWM_DUTY 6000 //定义PWM的周期,数值为时钟周期数,假如使用24.576MHZ的主频,则PWM频率为6000HZ。
#define PWM_HIGH_MIN 32 //限制PWM输出的最小占空比。用户请勿修改。
#define PWM_HIGH_MAX (PWM_DUTY-PWM_HIGH_MIN) //限制PWM输出的最大占空比。用户请勿修改。
typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;
sfr P1M1 = 0x91; //P1M1.n,P1M0.n =00--->Standard, 01--->push-pull 实际上1T的都一样
sfr P1M0 = 0x92; // =10--->pure input, 11--->open drain
sfr P0M1 = 0x93; //P0M1.n,P0M0.n =00--->Standard, 01--->push-pull
sfr P0M0 = 0x94; // =10--->pure input, 11--->open drain
sfr P2M1 = 0x95; //P2M1.n,P2M0.n =00--->Standard, 01--->push-pull
sfr P2M0 = 0x96; // =10--->pure input, 11--->open drain
sfr P3M1 = 0xB1; //P3M1.n,P3M0.n =00--->Standard, 01--->push-pull
sfr P3M0 = 0xB2; // =10--->pure input, 11--->open drain
sfr P4M1 = 0xB3; //P4M1.n,P4M0.n =00--->Standard, 01--->push-pull
sfr P4M0 = 0xB4; // =10--->pure input, 11--->open drain
sfr P5M1 = 0xC9; // P5M1.n,P5M0.n =00--->Standard, 01--->push-pull
sfr P5M0 = 0xCA; // =10--->pure input, 11--->open drain
sfr P6M1 = 0xCB; // P5M1.n,P5M0.n =00--->Standard, 01--->push-pull
sfr P6M0 = 0xCC; // =10--->pure input, 11--->open drain
sfr P7M1 = 0xE1;
sfr P7M0 = 0xE2;
sfr AUXR = 0x8E;
sfr INT_CLKO = 0x8F;
sbit P_PWM = P3^4; //定义PWM输出引脚。
u16 pwm; //定义PWM输出高电平的时间的变量。用户操作PWM的变量。
u16 PWM_high,PWM_low; //中间变量,用户请勿修改。
void delay_ms(u8 ms);
void LoadPWM(u16 i);
/******************** 主函数**************************/
void main(void)
{
P0M1 = 0; P0M0 = 0; //设置为准双向口
P1M1 = 0; P1M0 = 0; //设置为准双向口
P2M1 = 0; P2M0 = 0; //设置为准双向口
P3M1 = 0; P3M0 = 0; //设置为准双向口
P4M1 = 0; P4M0 = 0; //设置为准双向口
P5M1 = 0; P5M0 = 0; //设置为准双向口
P6M1 = 0; P6M0 = 0; //设置为准双向口
P7M1 = 0; P7M0 = 0; //设置为准双向口
P_PWM = 0;
P3M1 &= ~(1 << 4); //P3.4 设置为推挽输出
P3M0 |= (1 << 4);
TR1 = 0; //停止计数
ET1 = 1; //允许中断
PT1 = 1; //高优先级中断
TMOD &= ~0x30; //工作模式,0: 16位自动重装
AUXR |= 0x40; //1T
TMOD &= ~0x40; //定时
INT_CLKO |= 0x02; //输出时钟
TH1 = 0;
TL1 = 0;
TR1 = 1; //开始运行
EA = 1;
pwm = PWM_DUTY/10 ; //给PWM一个初值,这里为10%占空比
LoadPWM(pwm); //计算PWM重装值
while (1)
{
/*
while(pwm < (PWM_HIGH_MAX-8))
{
pwm += 8; //PWM逐渐加到最大
LoadPWM(pwm);
delay_ms(8);
}
while(pwm > (PWM_HIGH_MIN+8))
{
pwm -= 8; //PWM逐渐减到最小
LoadPWM(pwm);
delay_ms(8);
}
*/
LoadPWM(pwm);
}
}
//========================================================================
// 函数: void delay_ms(u8 ms)
// 描述: 延时函数。
// 参数: ms,要延时的ms数, 这里只支持1~255ms. 自动适应主时钟.
// 返回: none.
// 版本: VER1.0
// 日期: 2013-4-1
// 备注:
//========================================================================
void delay_ms(u8 ms)
{
unsigned int i;
do{
i = MAIN_Fosc / 13000;
while(--i) ;
}while(--ms);
}
/**************** 计算PWM重装值函数 *******************/
void LoadPWM(u16 i)
{
u16 j;
if(i > PWM_HIGH_MAX) i = PWM_HIGH_MAX; //如果写入大于最大占空比数据,则强制为最大占空比。
if(i < PWM_HIGH_MIN) i = PWM_HIGH_MIN; //如果写入小于最小占空比数据,则强制为最小占空比。
j = 65536UL - PWM_DUTY + i; //计算PWM低电平时间
i = 65536UL - i; //计算PWM高电平时间
EA = 0;
PWM_high = i; //装载PWM高电平时间
PWM_low = j; //装载PWM低电平时间
EA = 1;
}
/********************* Timer0中断函数************************/
void timer0_int (void) interrupt 3
{
if(P_PWM)
{
TH1 = (u8)(PWM_low >> 8); //如果是输出高电平,则装载低电平时间。
TL1 = (u8)PWM_low;
}
else
{
TH1 = (u8)(PWM_high >> 8); //如果是输出低电平,则装载高电平时间。
TL1 = (u8)PWM_high;
}
}
|