timer - AVR CTC 定时器频率明显不准确
问题描述
我是 AVR 设备编程的初学者,试图摆脱低效的 _ms_delay() 和 _us_delay() 阻塞功能,我一直在尝试使用内置计时器来控制基本 LED 闪烁程序的时间的程序16 位定时器上的 CTC 定时器模式。我的目标是让 LED 以 2 Hz 的频率闪烁,点亮 0.5 秒,熄灭 0.5 秒。
根据ATMega328P 数据表,CTC 输出的频率应该是 f_CTC = f_Clock/(2N(OCR1A+1),因为我的芯片是 328P Xplained mini,它的默认 CPU 速度是 16 MHz,使用上面的公式,N =64,达到我想要的频率所需的 OCR1A 值应该是 62499。考虑到所有这些,我编写了以下代码:
#include <avr/io.h>
int main(void)
{
// Setup for timer TC1 (16-bit) in CTC mode to run at 2 Hz
TCCR1A = 0x00;
OCR1A = 62499; // Sets time resolution to 0.5 s
TCCR1B = 0x0b;
TCCR1C = 0x00;
// Set pin directions
PORTD &= ~(1<<PORTD6);
DDRD |= (1<<DDD6);
while (1)
{
if(TIFR1 & (1<<OCF1A))
{
PORTD ^= (1<<PORTD6);
}
TIFR1 |= (1<<OCF1A);
}
}
但是,当我运行代码时,LED 以 1 Hz 的频率闪烁,我可以用手机计时。此外,当我将 OCR1A 更改为 31249 时,它应该将频率增加到 4 Hz,它似乎以 8 Hz 的频率闪烁,或者每秒开关 4 次。我觉得我误解了 CTC 模式的工作原理,如果有人可以简单地向我解释一下,或者我的代码有任何其他问题,我将不胜感激。
解决方案
我注意到一件事可能会导致您看到的问题。
您正在使用该行TIFR1 |= (1<<OCF1A);
来清除 OCF1A 位。您非常频繁地运行该行,因此当 OCF1A 被设置时,您的代码很可能会在 if 语句看到它被设置之前立即清除它。您无法控制该位何时设置;它可能发生在循环中的任何时候。
您应该只在验证 OCF1A 为 1 后清除它,如下所示:
if (TIFR1 & (1 << OCF1A))
{
PORTD ^= (1 << PORTD6);
TIFR1 |= (1 << OCF1A);
}
推荐阅读
- node.js - ExpressJS-Localhost 没有显示任何内容
- android - 如何从 Firebase 数据存储中获取图像 url?
- python - 定义状态常量的 Pythonic 方式
- outlook - Outlook 2016 加载项对话框高度和宽度已交换
- javascript - 在小数点前后追加值Javascript
- javascript - 无法使用传单渲染矢量切片 (.mbtiles) 文件
- javascript - 页面加载后如何从nodejs服务器向客户端发送额外的javascript代码
- android - Android 将应用程序注册为 LiveData 中的上下文
- gitlab - 将 GitLab CE 11 Omnibus 迁移到 Docker
- pandas - 基于 '%' 符号 pandas 数据框追加