神农鼎
发表于 2022-12-2 18:45:29
本坛 PWM 实战型 老兄弟多,给力!!!
骏杰
发表于 2022-12-2 18:46:40
flyarm 发表于 2022-12-2 18:40
还有PWMA_CR1 = 0x01; //开始计数
这个怎么能放到前面,你都开始计数了,怎么又去配置寄 ...
这个开始计数我放到前面试过,放到最后也试过,还是没有解决问题
骏杰
发表于 2022-12-2 18:48:02
#include <STC8H.H>
#define uint unsigned int
#define uchar unsigned char
#define ulong unsigned long
ulong a2;
uint cnt;
uint shujuik()//启动捕获功能 并初始化
{
ulong shuj=0,li;
double yi=0;
// PWMA_CNTR=0x00;//清理计数器后也是不正常
PWMA_CCER1 = 0x00;
PWMA_CCMR1 = 0xfd; //CC1 为输入模式,且映射到 TI1FP1 上
PWMA_CCMR2 = 0xfe; //CC2 为输入模式,且映射到 TI1FP2 上
PWMA_CCER1 = 0x11; //使能 CC1/CC2 上的捕获功能
PWMA_CCER1 |= 0x00; //设置捕获极性为 CC1 的上升沿
PWMA_CCER1 |= 0x20; //设置捕获极性为 CC2 的下降沿
PWMA_CR1 = 0x01; //开始计数
PWMA_SR1=0x00; //清理标志位
while((PWMA_SR1&0x04)!=0x04);//等待捕获完成
PWMA_SR1=0x00; //清理标志位
shuj=PWMA_CCR2- PWMA_CCR1;//读取结果
SBUF=shuj>>8;//发送结果
while(TI==0);
TI=0;
SBUF=shuj;
while(TI==0);
TI=0;
return shuj;
}
void main()
{
//***************************************************这里是串口初始化
SCON = 0x50; //8位数据,可变波特率
AUXR |= 0x40; //定时器时钟1T模式
AUXR &= 0xFE; //串口1选择定时器1为波特率发生器
TMOD &= 0x0F; //设置定时器模式
TL1 = 0xE0; //设置定时初始值 11.0592
TH1 = 0xFE; //设置定时初始值
TR1 = 1; //定时器1开始计时
P_SW2 |= 0x80; //使能访问 XFR
P1M0 = 0x00;P1M1 = 0x00;
//***************************************************************
while(1)
{
while(RI==0);//用电脑向串口随便发送一个数据
RI=0;
shujuik();//捕获功能
PWMA_CCER1 = 0x00;//如果不注释输出的结果就不是正确的
PWMA_CR1=0x00;
}
}
骏杰
发表于 2022-12-2 18:51:31
神农鼎 发表于 2022-12-2 18:45
本坛 PWM 实战型 老兄弟多,给力!!!
我也问过STC的技术支持他们说只要使能肯定可以用说绝对不是IC的问题,我这实在没办法了,也找不到问题出在哪里
神农鼎
发表于 2022-12-2 19:00:34
老兄弟回去做饭吃了,等他吃饱了,再回来帮你
骏杰
发表于 2022-12-2 19:08:03
flyarm 发表于 2022-12-2 18:35
子函数里不要每次都 所有的重新配置; 前面那些初始化一次就行了; 你只需要循环 读标志位和计数器操作
...
只要关闭计数使用时再打开就还是那样
#include <STC8H.H>
#define uint unsigned int
#define uchar unsigned char
#define ulong unsigned long
ulong a2;
uint cnt;
uint shujuik()//启动捕获功能 并初始化
{
ulong shuj=0,li;
double yi=0;
PWMA_CR1 = 0x01; //开始计数
PWMA_SR1=0x00; //清理标志位
while((PWMA_SR1&0x04)!=0x04);//等待捕获完成
PWMA_SR1=0x00; //清理标志位
shuj=PWMA_CCR2- PWMA_CCR1;//读取结果
SBUF=shuj>>8;//发送结果
while(TI==0);
TI=0;
SBUF=shuj;
while(TI==0);
TI=0;
return shuj;
}
void main()
{
//***************************************************这里是串口初始化
SCON = 0x50; //8位数据,可变波特率
AUXR |= 0x40; //定时器时钟1T模式
AUXR &= 0xFE; //串口1选择定时器1为波特率发生器
TMOD &= 0x0F; //设置定时器模式
TL1 = 0xE0; //设置定时初始值 11.0592
TH1 = 0xFE; //设置定时初始值
TR1 = 1; //定时器1开始计时
P_SW2 |= 0x80; //使能访问 XFR
P1M0 = 0x00;P1M1 = 0x00;
//***************************************************************
PWMA_CCER1 = 0x00;
PWMA_CCMR1 = 0x01; //CC1 为输入模式,且映射到 TI1FP1 上
PWMA_CCMR2 = 0x02; //CC2 为输入模式,且映射到 TI1FP2 上
PWMA_CCER1 = 0x11; //使能 CC1/CC2 上的捕获功能
PWMA_CCER1 |= 0x00; //设置捕获极性为 CC1 的上升沿
PWMA_CCER1 |= 0x20; //设置捕获极性为 CC2 的下降沿
while(1)
{
while(RI==0);//用电脑向串口随便发送一个数据
RI=0;
shujuik();//捕获功能
PWMA_CR1=0x00;
}
}
zhp
发表于 2022-12-2 21:53:41
骏杰 发表于 2022-12-2 19:08
只要关闭计数使用时再打开就还是那样
#include
#define uint unsigned int
请问一下,你的这段代码的目的是为了捕获高电平的宽度吗?
如果是这样的话,你这段代码看似正确,但其实有个比较隐藏的问题
就是你必须保证你每次开始捕获时CC1(P1.0)口的电平都必须是低电平
也就是必须CC1的上升沿时先捕获计数值到PWMA_CCR1,然后在CC1的下降沿时捕获计数值到到PWMA_CCR2
最后再用PWMA-CCR2-PWMA_CCR1得到高电平的宽度
但实际上你无法保证每次捕获时P1.0口的电平都是低电平
如果开始捕获时,P1.0的电平已经是高电平,则当满足(PWMA_SR1&0x04)==0x04时,
只是产生了一次CC1的下降沿捕获,这时你再用PWMA-CCR2-PWMA_CCR1得到的值必然是不准确的
骏杰
发表于 2022-12-2 22:46:09
zhp 发表于 2022-12-2 21:53
请问一下,你的这段代码的目的是为了捕获高电平的宽度吗?
如果是这样的话,你这段代码看似正确,但其实 ...
你说的这个问题,我也考虑过我在前面加上while((PWMA_SR1&0x02)!=0x02)加上这一条区别不是很大
zhp
发表于 2022-12-3 09:41:13
骏杰 发表于 2022-12-2 22:46
你说的这个问题,我也考虑过我在前面加上while((PWMA_SR1&0x02)!=0x02)加上这一条区别不是很大 ...
如果仅仅只是加你说的这段还是肯定不行
如果是先捕获下降沿,后捕获到上上升沿,CC2-CC1也不是你所需要的值
zhp
发表于 2022-12-3 09:47:21
骏杰 发表于 2022-12-2 22:46
你说的这个问题,我也考虑过我在前面加上while((PWMA_SR1&0x02)!=0x02)加上这一条区别不是很大 ...
或许比较好的方式是连续判断两次下降沿,如下:while((PWMA_SR1&0x04)!=0x04);
PWMA_SR1=0x00;
while((PWMA_SR1&0x04)!=0x04);
PWMA_SR1=0x00;
完成后再执行你后面的操作