RT1052 内部集成了 4 个 QTMR 定时器,每个 QTMR 定时器又有 4 个通道.
每个通道都有独立的:
完全就是一个独立定时器的功能。
可以将QTMR 看成是一个拥有 4*4 个定时器的集合,这样光 QTMR 就拥有 16 个定时器之多
RT1052 内部有 16 个这样的通道!
RT1052 四定时器每个通道的功能包括:
1)16 位计数器(CNTR),支持向上/下计数。
2)可级联,组成 32/48/64 位计数器。
3)独立分频器(支持:1/2/4/8/16/32/64/128 分频)。
4)支持输出比较和输入捕获功能。
5)支持多种工作模式(14 种)。
6)支持输入滤波器(输入捕获时用)。
7)最大计数频率可达 150Mhz。
8)支持单次计数/连续计数。
9)比较寄存器具有预装载功能。
10)4 个通道可以同时开启计数(同步启动)。
四定时器单个通道的工作模式非常多(14 种),具体某个模式的使用方法请参考《RT1050参考手册》第 47.6.5 节。
QTMR 相关的库函数在 fsl_qtmr.c 和 fsl_qtmr.h 这两个文件中
我们使用 QTMR1 的通道 0 产生中断
使用函数 CLOCK_EnableClock 使能 QTMR1 时钟,使用方法如下:
CLOCK_EnableClock(kCLOCK_Timer1)
此函数会被 Q TMR 定时器初始化函数 QTMR_Init 调用,所以不需要我们显示的调用。
使用函数 QTMR_Init 来初始化 QTMR,此函数原型如下:
void QTMR_Init(TMR_Type * base,qtmr_channel_selection_t channel,const qtmr_config_t * config)
qtmr_config_t qtimer1_config;
QTMR_GetDefaultConfig(&qtimer1_config); //先设置为默认配置
qtimer1_config.primarySource= kQTMR_ClockDivide_128; //设置第一时钟源
QTMR_Init(TMR1, kQTMR_Channel_0, &qtimer1_config); //初始化 QTIMER
typedef enum _qtmr_channel_selection
{
kQTMR_Channel_0 = 0U, //QTMR 通道 0
kQTMR_Channel_1, //QTMR 通道 1
kQTMR_Channel_2, //QTMR 通道 2
kQTMR_Channel_3, //QTMR 通道 3
} qtmr_channel_selection_t;
typedef struct _qtmr_config
{
qtmr_primary_count_source_t primarySource; //指定第一时钟源
qtmr_input_source_t secondarySource; //指定第二时钟源
bool enableMasterMode;
bool enableExternalForce;
uint8_t faultFilterCount;
uint8_t faultFilterPeriod;
qtmr_debug_action_t debugMode;
} qtmr_config_t;
这个结构体最常用的就是前两个成员变量
typedef enum _qtmr_primary_count_source
{
kQTMR_ClockCounter0InputPin = 0,
kQTMR_ClockCounter1InputPin,
kQTMR_ClockCounter2InputPin,
kQTMR_ClockCounter3InputPin,
kQTMR_ClockCounter0Output,
kQTMR_ClockCounter1Output,
kQTMR_ClockCounter2Output,
kQTMR_ClockCounter3Output,
kQTMR_ClockDivide_1, //IPG 总线时钟 1 分频
kQTMR_ClockDivide_2, // IPG 总线时钟 2 分频
kQTMR_ClockDivide_4, // IPG 总线时钟 4 分频
kQTMR_ClockDivide_8, // IPG 总线时钟 8 分频
kQTMR_ClockDivide_16, //IPG 总线时钟 16 分频
kQTMR_ClockDivide_32, // IPG 总线时钟 32 分频
kQTMR_ClockDivide_64, // IPG 总线时钟 64 分频
kQTMR_ClockDivide_128 // IPG 总线时钟 128 分频
} qtmr_primary_count_source_t;
第一时钟源为 IPG_CLK_ROOT 的 128 分频,也就是设置为 kQTMR_ClockDivide_128。
第二时钟源的选择类似。
QTMR1 通道 0 的定时周期通过函数 QTMR_SetTimerPeriod 来设置
此函数原型如下:
void QTMR_SetTimerPeriod(TMR_Type *base, qtmr_channel_selection_t channel, uint16_t ticks)
使用函数 QTMR_EnableInterrupts 来使能 QTMR1 通道 0 的比较中断,这样当定时器计数值(CNTR0)达到我们设置的匹配比较值的时候就会产生相应中断
此函数原型如下:
void QTMR_EnableInterrupts(TMR_Type * base,qtmr_channel_selection_t channel,uint32_t mask)
typedef enum _qtmr_interrupt_enable
{
kQTMR_CompareInterruptEnable = (1U << 0), //比较中断
kQTMR_Compare1InterruptEnable = (1U << 1), //比较 1 中断
kQTMR_Compare2InterruptEnable = (1U << 2), //比较 2 中断
kQTMR_OverflowInterruptEnable = (1U << 3), //溢出中断
kQTMR_EdgeInterruptEnable = (1U << 4) //输入边沿检测中断
} qtmr_interrupt_enable_t;
我们要使能的是匹配比较中断,因此选择 kQTMR_CompareInterruptEnable。
在相关的配置完成以后,就可以使能 QTMR 了,使用函数 QTMR_StartTimer 来开启 QTMR定时器
此函数原型如下:
static inline void QTMR_StartTimer(TMR_Type * base,qtmr_channel_selection_t channel,qtmr_counting_mode_t clockSource)
第三个参数用来设置定时器的计数模式,是上升沿计数还是下降沿计数、双边沿模式。
typedef enum _qtmr_counting_mode
{
kQTMR_NoOperation = 0,
kQTMR_PriSrcRiseEdge,
kQTMR_PriSrcRiseAndFallEdge,
kQTMR_PriSrcRiseEdgeSecInpHigh,
kQTMR_QuadCountMode,
kQTMR_PriSrcRiseEdgeSecDir,
kQTMR_SecSrcTrigPriCnt,
kQTMR_CascadeCount
} qtmr_counting_mode_t;
这个参数设置的就是寄存器 CTRL0 的 CM 位,本例程中我们选择 kQTMR_PriSrcRiseEdge,也就是第一时钟源上升沿计数模式。
在定时器配置完了之后,因为要产生中断,必不可少的要设置 NVIC 相关寄存器,以使能QTMR1 的相关中断,设置方法如下:
RT1052_NVIC_SetPriority(TMR1_IRQn,6,0); //抢占优先级 6,子优先级 0
EnableIRQ(TMR1_IRQn); //是能 TMR1 中断
通过函数 QTMR_GetStatus 来获取中断状态,判断此次产生的中断来自哪个通道
调用函数 QTMR_ClearStatusFlags 来清除相应的中断标志位。
中断状态获取函数 QTMR_GetStatus 原型如下:
uint32_t QTMR_GetStatus(TMR_Type *base, qtmr_channel_selection_t channel)
其实就是读取寄存器 SCTRL0 的值
中断状态(标志位)清除函数 QTMR_ClearStatusFlags 原型如下:
void QTMR_ClearStatusFlags(TMR_Type * base,
qtmr_channel_selection_t channel,
uint32_t mask)
最后一个参数指定要清除的中断类型(中断标志位)
typedef enum _qtmr_status_flags
{
kQTMR_CompareFlag = (1U << 0), //比较标志位
kQTMR_Compare1Flag = (1U << 1), //比较 1 标志位
kQTMR_Compare2Flag = (1U << 2), //比较 2 标志位
kQTMR_OverflowFlag = (1U << 3), //溢出标志位
kQTMR_EdgeFlag = (1U << 4) //输入边沿检测标志位
} qtmr_status_flags_t;
这里要清除的即使比较标志位,所以选择kQTMR_CompareFlag