11. EPIC定时器

news2025/1/10 21:12:23

11. EPIC定时器

  • EPIT定时器简介
    • EPIT定时器结构分析
    • EPIT 定时器相关寄存器
      • EPITx_CR
      • EPITx_SR
      • EPITx_LR 加载寄存器
      • EPITx_CMPR 比较寄存器
      • EPITx_CNR 计数寄存器
    • EPIT 配置步骤
  • 例程代码编写
    • bsp_epittimer.h
    • bsp_epittimer.c
    • main.c

EPIT定时器简介

EPIT定时器是增强的周期中断定时器,主要是完成周期性中断定时。EPIT是一个 32 位定时器,在处理器几乎不用介入的情况下提供精准的定时中断,软件使能后EPIT就会开始运行。EPIT有以下特点:

  • 时钟源可选的 32 位向下计数器
  • 12 位的分频值
  • 当计数值和比较值相等的时候产生中断

EPIT定时器结构分析

在这里插入图片描述

  1. 这是一个多路选择器,用来选择 EPIT 定时器的时钟源
  2. 这是一个 12 位的分频器,负责对时钟源进行分频,12 位对应的值是 0 ~ 4095,对应 1 ~ 4096分频
  3. 经过分频的时钟进入到 EPIT 内部,在内部有 3 个重要的寄存器:计数寄存器、加载寄存器和比较寄存器。这 3 个寄存器都是 32 位的。EPIT 计数器是一个向下的计数器,也就是说给它一个初值,它就会从这个初值开始递减,直到减为 0,计数寄存器保存的就是当前的计数器。如果EPIT工作在 set-and-forget 模式下,当计数寄存器里面的值减到0,EPIT 就会重新从加载寄存器读取数值到计数寄存器里面,重新开始向下计数。比较寄存器里保存的数值用于和计数寄存器的计数值比较,如果相等就会产生一个比较事件。
  4. 比较器
  5. EPIT 可以设置引脚输出,如果设置了的话就会通过指定的引脚输出信号
  6. 产生比较中断,也就是定时中断
  7. EPIT 定时器有两种工作模式:set-and-forget 和 free-running
    set-and-forget 模式: EPITx_CR(x=1,2)寄存器的 RLD 位置为 1 的时候 EPIT 工作在此模式下,在此模式下 EPIT 的计数器从加载寄存器 EPITx_LR 中获取初始值,而不能直接向计数器寄存器写入数据。不管什么时候,只要计数器计数到 0,那么就会从加载寄存器 EPITx_LR 中重新加载数据到计数器中。
    free-running 模式: EPITx_CR 寄存器的RLD 位清零的时候 EPIT 工作在此模式下,当计数器计数到 0 后会重新从 0xFFFFFFFF 开始计数,并不是从加载计数器中获取数据。

EPIT 定时器相关寄存器

EPITx_CR

在这里插入图片描述

  • CLKSRC(bit25:24): EPIT 时钟源选择位,为 0 的时候关闭时钟源,为 1 的时候选择ipg_clk 时钟,为 2 的时候选择 ipg_clk_highfreq,为 3 的时候选择 ipg_clk_32k。本例程中,选择ipg_clk = 66MHz
  • PRESCALAR(bit15:4): 时钟源分频值,可设置范围0 ~ 4095,对应1 ~ 4096 分频
  • RLD(bit3): EPIT 工作模式,为 0 的时候工作在 free-running 模式,为 1 的时候工作在 set-and-forget 模式,本例程为 1
  • OCIEN(bit2): 比较中断使能位,为 0 的时候关闭比较中断,为 1 的时候使能比较中断,这里我们要使能比较中断
  • ENMOD(bit1): 设置计数器初始值,为 0 的时候初始值等于上次关闭 EPIT 定时器以后计数器里面的值,为 1 的时候来源于加载寄存器或者0xFFFFFFFF(由工作模式决定)
  • EN(bit0): EPIT 使能位,为 0 的时候关闭 EPIT,为 1 的时候使能 EPIT

EPITx_SR

在这里插入图片描述
OCIF: 比较中断标志位,为 0 的时候表示没有比较事件发生,为 1 的时候表示有比较事件发生。当比较事件发生以后需要手动清除此位,此位是写1清零

EPITx_LR 加载寄存器

用于保存计数器的当前值,以供下一次计数使用。当计数器每次计时达到 0 以后,会重新加载此寄存器中的值,并且计数器重新开始计数

EPITx_CMPR 比较寄存器

当计数器的值和 CMPR 寄存器的值相等的时候就会产生中断

EPITx_CNR 计数寄存器

通入时钟,每一个时钟周期它的值就会减 1

EPIT 配置步骤

  1. 设置 EPIT1 的时钟源
    设置寄存器 EPIT1_CR 寄存器的 CLKSRC(bit25:24) 位
  2. 设置分频值
    设置寄存器 EPIT1_CR 寄存器的 PRESCALAR(bit 5:4) 位
  3. 设置工作模式
    设置寄存器 EPIT1_CR 寄存器的 RLD(bit3) 位
  4. 设置计数器的初始值来源
    设置寄存器 EPIT1_CR 寄存器的 ENMOD(bit1) 位
  5. 使能比较中断
    设置寄存器 EPIT1_CR 寄存器的 OCIEN(bit2) 位
  6. 设置加载值和比较值
    设置 EPIT1_LR 寄存器的加载值和 EPIT1_CMPR 比较值
  7. EPIT1 中断设置和中断服务函数编写
    使能GIC 中对应的 EPIT1 中断,注册中断服务函数,还可以设置中断优先级,最后编写中断服务函数
  8. 使能 EPIT1 定时器
    通过寄存器 EPIT1_CR 的 EN(bit0) 位来设置

例程代码编写

bsp_epittimer.h

#pragma once
#include "imx6ul.h"
void epit1_init(unsigned int frac, unsigned int value);
void epit_irqhandler();

bsp_epittimer.c

#include "bsp_epittimer.h"
#include "bsp_int.h"
#include "bsp_led.h"

void epit1_init(unsigned int frac, unsigned int value)
{
	if(frac > 0xFFF)
		frac = 0xFFF;
	EPIT1->CR = 0; // 先清零 CR 寄存器
	// 先将 EPIT 配置好之后再使能
	// CR 寄存器:24-25位配置时钟源,选择Peripheral clock=66;4-15位配置frac分频值
	// 3位当计数器到0的话从LR重新加载数值,也就是工作在set-and-forger 模式;
	// 2位比较中断使能;1位设置计数器初始值来自LR 寄存器;0位先关闭EPIT1
	EPIT1->CR = (1<<24|frac<<4|1<<3|1<<2|1<<1);
	EPIT1->LR = value;
	EPIT1->CMPR = 0;
	
	// 使能 GIC 中对应的中断
	GIC_EnableIRQ(EPIT1_IRQn);

	// 注册中断服务函数
	system_register_irqhandler(EPIT1_IRQn, (system_irq_handler_t)epit1_irqhandler, NULL);
	EPIT1->CR |= 1<<0; // 使能EPIT1
}
void epit1_irqhandler()
{
	static unsigned char state = 0;
	state = !state;
	if(EPIT1->SR & (1<<0)) // 判断比较事件发生
	{
		led_switch(LED0, state);	// 定时器周期到,反转LED
	}
	EPIT1->SR |= 1<<0;		// 清除中断标志位
}

分频值和 value 就可以决定中断频率,公式为:EPIT 溢出时间=((分频值+1)*value)/输入时钟频率

main.c

int main()
{
	int_init();			// 初始化中断
	imx6u_clkinit();	// 初始化系统时钟
	clk_enable();		// 使能所有时钟
	led_init();			// 初始化LED
	beep_init();		// 初始化beep
	key_init();
	epit1_init(0, 66000000/2);	// 初始化EPIT1定时器,1分频,计数器值为66000000/2 也就是500ms
	while(1)
	{
		delay(500);
	}
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1196714.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

jupyter notebook中markdown改变图像大小

文章目录 &#x1f56e;原始图像&#x1f56e;改变图像大小&#x1f56e;使图像靠左 在 jupyter notebook中&#xff0c;导入的图片过大&#xff0c;想要改变图像的大小 &#x1f56e;原始图像 &#x1f56e;改变图像大小 复制小括号里面的内容到src后面&#xff0c;满足<…

【LeetCode】挑战100天 Day09(热题+面试经典150题)

【LeetCode】挑战100天 Day09&#xff08;热题面试经典150题&#xff09; 一、LeetCode介绍二、LeetCode 热题 HOT 100-112.1 题目2.2 题解 三、面试经典 150 题-113.1 题目3.2 题解 一、LeetCode介绍 LeetCode是一个在线编程网站&#xff0c;提供各种算法和数据结构的题目&…

【自定义类型:结构体】

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 前言 1. 结构体类型的声明 1.1 结构体的概念 1.2 结构的声明 ​编辑 1.3 特殊的声明 1.4 结构的自引用 2. 结构体变量的创建和初始化 3. 结构成员访问操作符 4. 结构体内…

学习linux必不可少的【yum命令】!

&#x1f4da;&#x1f4da; &#x1f3c5;我是默&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; ​​ &#x1f31f;在这里&#xff0c;我要推荐给大家我的专栏《Linux》。&#x1f3af;&#x1f3af; &#x1f680;无论你是编程小白&#xff0c;还是有一…

【JavaEE初阶】 TCP滑动窗口与流量控制和拥塞控制

文章目录 &#x1f384;为什么出现滑动窗口&#x1f38b;滑动窗口丢包问题&#x1f6a9;情况一&#xff1a;数据包已经抵达&#xff0c;ACK被丢了。&#x1f6a9;情况二&#xff1a;数据包就直接丢了 &#x1f332;流量控制&#xff08;安全机制&#xff09;&#x1f333;拥塞控…

STM32F4X SDIO(九) 例程讲解-SD卡擦除、读写

STM32F4X SDIO &#xff08;九&#xff09; 例程讲解-SD卡擦除、读写 例程讲解-SD卡擦除、读写SD卡擦除CMD32:ERASE_WR_BLK_START命令发送命令响应 CMD33:ERASE_WR_BLK_END命令发送命令响应CMD38:ERASE命令响应 CMD13:SD_CMD_SEND_STATUS命令发送命令回应 SD卡读数据CMD16:SET_…

C++ 配合图形库实现画线效果

#include<stdio.h> #include <conio.h> #include<math.h> #include <graphics.h> // 引用图形库头文件 #define N 12 int List[N][N];void draw() {for (int i 0; i < N; i) {int x 200 * cos(2 * 3.14 * i / N);int y 200 * sin(2 * 3.1…

SwiftUI 如何保证 Text 中字符数量相等的字符串显示宽度一定相同?

0. 问题现象 在 SwiftUI 中我们往往需要将内容相似的字符串展列出来给用户比较,这些字符串内容各有不同但字符数量始终是相等的,我们希望它们的显示宽度始终保持一致: 如上图所示:即使是等宽字符组成的字符串在字符数量相等时它们的显示宽度仍然可能不一致。但演示中最底部…

抖音短视频账号矩阵系统、短视频矩阵源码+无人直播源码开发可打包

抖音短视频账号矩阵系统、短视频矩阵源码无人直播源码开发可打包 矩阵系统源码主要有三种框架&#xff1a;Spring、Struts和Hibernate。Spring框架是一个全栈式的Java应用程序开发框架&#xff0c;提供了IOC容器、AOP、事务管理等功能。Struts框架是一个MVC架构的Web应用程序框…

Raft分布式一致性算法

拜占庭将军 假设多位拜占庭将军中没有叛军&#xff0c;信使的信息可靠但有可能被暗杀的情况下&#xff0c;将军们如何达成是否要进攻的一致性决定&#xff1f;解决问题的思路是&#xff0c;从多位处于平等地位的将军中选举出一位大将军&#xff0c;所有作战指令由大将军发出。…

python升级pip的时候一直失败

如图,一直提示使用 python.exe -m pip install --upgrade pip 进行升级pip,但是执行这句命令又不成功.然后综合了几篇文章以后使用了下面的命令可以升级了 python -m pip install --upgrade pip --user -i https://mirrors.aliyun.com/pypi/simple/ 主要是在推荐的语句上使用…

解决内存泄漏问题,Profiler工具的使用介绍

什么是内存泄漏 内存泄漏&#xff08;Memory Leak&#xff09;是指程序中已动态分配的堆内存由于某种原因程序未释放或无法释放&#xff0c;造成系统内存的浪费&#xff0c;导致程序运行速度减慢甚至系统崩溃等严重后果。 以上是官方针对内存泄漏的说法。说的通俗一点&#x…

高频SQL50题(基础版)-3

文章目录 主要内容一.SQL练习题1.1174-即时食物配送代码如下&#xff08;示例&#xff09;: 2.550-游戏玩法分析代码如下&#xff08;示例&#xff09;: 3.2356-每位教师所教授的科目种类的数量代码如下&#xff08;示例&#xff09;: 4.1141-查询近30天活跃用户数代码如下&…

【FPGA】Verilog:计数器 | 异步计数器 | 同步计数器 | 2位二进制计数器的实现 | 4位十进制计数器的实现

目录 Ⅰ. 实践说明 0x00 计数器&#xff08;Counter&#xff09; 0x01 异步计数器&#xff08;Asynchronous Counter&#xff09; 0x02 同步计数器&#xff08;Synchronous Counter&#xff09; Ⅱ. 实践&#xff1a;2位二进制计数器 0x00 实践说明 0x01 输出表 0x02 代…

由于找不到dll无法执行代码的解决方法,深度解析5个dll修复方法

在使用计算机的过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是“找不到dll无法执行代码”。这个错误通常发生在程序运行时&#xff0c;系统无法找到所需的动态链接库&#xff08;DLL&#xff09;文件。这个问题可能由多种原因引起&#xff0c;包括缺少…

前端如何结合mock模拟假数据

由于某人不想写后端接口&#xff0c;不想用真数据对接vue-element-admin框架&#xff0c;用以前的接口&#xff0c;改token有点点麻烦&#xff0c;所以咱试试mock.js

《算法通关村——透彻理解二叉树中序遍历的应用》

《算法通关村——透彻理解二叉树中序遍历的应用》 直接上题 108. 将有序数组转换为二叉搜索树 给你一个整数数组 nums &#xff0c;其中元素已经按 升序 排列&#xff0c;请你将其转换为一棵 高度平衡 二叉搜索树。 高度平衡 二叉树是一棵满足「每个节点的左右两个子树的高…

【电工基础】

电工基础 11.1 简介1.2 电路作用1.3 电路模型1.4 电流定义1.5 电压定义1.6 电动势1.7 电阻元件1.7.1 电阻元件定义1.7.2 电阻原件的特性1.7.31.7.4 1.81.91.10 345 1 1.1 简介 电源外部&#xff0c;正电荷移动的方向是由电源正极向电源负极方向&#xff0c;负电荷移动的方向是…

GEE:遥感影像二值化

作者:CSDN @ _养乐多_ 在Google Earth Engine(GEE)中,图像二值化是将图像中的像素值转换为二进制(0或1)的过程。这通常用于将连续的遥感图像转换为只有两个值的二值图像,以突出图像中的特定特征或区域。 结果如下图所示,将NDVI图像中,大于0.3的值设置为1(黑色),小…

VBA_MF系列技术资料1-217

MF系列VBA技术资料 为了让广大学员在VBA编程中有切实可行的思路及有效的提高自己的编程技巧&#xff0c;我参考大量的资料&#xff0c;并结合自己的经验总结了这份MF系列VBA技术综合资料&#xff0c;而且开放源码&#xff08;MF04除外&#xff09;&#xff0c;其中MF01-04属于定…