让一个LED灯闪烁不过瘾,我们应该让这块开发板完成一点更高难度的任务:比如让两个LED灯闪烁。

……

当然了,以我们的现在使用的空循环技术,还是可以实现这点的。但是这样显得略为低端。所以我们使用一个高端点的技术:中断。还有就是会介绍一下在CMSIS里怎么使用中断。

一、电路

二、实现思路

第一个LED的闪烁还是用之前使用的空循环吧,别把世界弄得太复杂了。

第二个LED的闪烁就稍微自动化一点了:使用一个定时器,让它在到了需要切换引脚电平的时候通知我们一下。这样做的好处就是我们只需在定时器通知时关注第二个LED灯,而在其他的时候就可以忙别的事了。(比如让第一个LED闪烁。)

使用的中断源还是之前用到的RTT。RTT可以在计数器达到特定值时产生中断,这个特定的值(Alarm Value)可以通过访问RTT报警寄存器(RTT_AR)设定。然后在RTT的中断处理函数中切换LED引脚的电平,同时设定好下一次中断的条件就好了。

三、中断

在中断时,处理器会根据中断号在中断向量表查询中断服务函数(ISR)相关的信息。为此,我们需要知道RTT的中端号(3),还有中断向量表的位置,然后修改中断向量表。系统控制块(SCB)中有个“向量表偏移寄存器”(SCB_VTOR),在这个地址指向的区域里储存着一系列的向量,包括外部中断向量表。然后我们需要知道ISR相关信息在这个向量表的位置。接着修改中断向量表时需要知道它储存的只有ISR的地址,还是直接跳转至ISR的指令……(先别忙着动手)

四、main函数之前发生的事

实际上,入口点——即整个程序开始运行的入口,并不是main函数。这个入口是链接器指定的,默认情况下是_start函数。而在Atmel Studio生成的项目中,默认情况下链接器的参数有“--entry=Reset_Handler”的这么一项,意思就是指定程序入口为Reset_Handler

这个函数的实现在以下文件中:

src\ASF\sam\utils\cmsis\sam4e\source\templates\gcc\startup_sam4e.c

这个是函数也是重置时的中断处理函数。在这个函数中,进行了一系列的初始化工作,其中包括中断向量表的配置。然后在初始化C库之后,就调用main函数了。最后在main函数返回后执行一个死循环。

五、定义中断处理函数

CMSIS已经为定义好了各种ISR的函数原型,同时做好了默认的函数实现。这些函数在以下文件中实现:

src\ASF\sam\utils\cmsis\sam4e\source\templates\exceptions.c

不过默认的函数实现是“弱定义”为Dummy_Handler的别名,这个函数的实现只是一个简单的死循环。弱定义意味着我们可以很方便地在链接时覆盖默认的实现。方法就是重新定义一个具有相同签名的函数。因为默认情况下是“强定义”的,所以就会覆盖掉默认的实现。

四、CMSIS默默完成的工作

其实CMSIS已经做好很多事了。

  1. 在运行C语言编译后产生的代码时,需要堆栈来追踪函数调用情况,以及储存一些临时变量。我们编写的main函数得以成功运行的原因之一就是CMSIS已经帮我们准备好了堆栈。
  2. 然后CMSIS完成的事就是配置向量表了。CMSIS已经为定义好了各种ISR的函数原型,同时做好了默认的函数实现。不过默认的函数实现是“弱定义”的,这意味着我们可以很方便地覆盖默认的实现。然后CMSIS就会根据这些信息配置向量表,而在编写自定义的中断处理函数只需根据原型做出实现即可。
  3. 接着CMSIS会进行一系列系统初始化工作(调用SystemInit函数)。包括定义好读写Flash芯片时需要等待的时间,初始化震荡器、锁相环,设定系统主时钟等。
  4. “最后”,就是执行main函数了。

六、准备工作

现在程序已经略为复杂了,需要做些准备工作。

宏定义:

/* LED 使用的GPIO引脚 */
#define LED0_GPIO PIO_PA0
#define LED1_GPIO PIO_PD20 /* LED 闪烁的周期 */
#define LED0_OFF_MS 500
#define LED0_ON_MS 1000
#define LED1_OFF_MS 500
#define LED1_ON_MS 200

辅助函数CalcRTTNeedInc。之前为了计算经过指定时间后RTT记数器增加的值,写了几行代码。因为有多个地方要用到这个计算,所以把它抽象出来了:

inline uint32_t CalcRTTNeedInc(unsigned int ms)
{
/* 计数器加一的频率 */
const uint32_t freq = CHIP_FREQ_SLCK_RC / PRESCALE;
/* 计算延迟后,计数器需要增加的值
* need_inc = ms /1000 / (1/freq) */
return (ms * freq / 1000); }

六、RTT的中断处理

在理论上,本程序在RTT中断时切换第二个LED的引脚电平,并设置下一次中断的条件。

在文件sam4e16e.h中,已经定义好了RTT中断处理函数的原型了,只需实现即可。

需要注意的是,在中断处理函数中,需要通过读取一次RTT_SR以清除RTT的Alarm状态,否则该中断一直会被触发。

void RTT_Handler(void)
{
/* 通过读取状态寄存器清除Alarm */
uint32_t _ = RTT->RTT_SR;
uint32_t begin_rttv = ReadRTT_CRTV();
uint32_t int_gap_ms ;
uint32_t need_inc; if ((PIOD->PIO_ODSR & LED1_GPIO) == 0)
{
/* 现在引脚电平为低,LED是亮的 */
/* 灭灯 */
PIOD->PIO_SODR = LED1_GPIO;
/* 设置下次中断唤醒间隔的时间 */
int_gap_ms = LED1_OFF_MS;
}
else
{
/* 现在引脚电平为高,LED是灭的 */
/* 亮灯 */
PIOD->PIO_CODR = LED1_GPIO;
/* 设置下次中断唤醒间隔的时间 */
int_gap_ms = LED1_ON_MS;
} /* 计算并设置下一次中断的条件 */
need_inc = CalcRTTNeedInc(int_gap_ms);
RTT->RTT_AR = RTT_AR_ALMV(begin_rttv + need_inc - 1); return;
}

七、RTT初始化中断启用

如果需要启用中断,需要配置NVIC_ISERx寄存器,而且需要进行一定的计算。而CMSIS也做了相应的工作:

/* 启用中断 */
NVIC_ClearPendingIRQ(RTT_IRQn);
NVIC_EnableIRQ(RTT_IRQn);

对于RTT,配置时只需使能中断,同时设置第一次中断的条件即可。

/* 初始化 RTT */
RTT->RTT_MR = RTT_MR_RTPRES(PRESCALE)
| RTT_MR_RTTRST
| RTT_MR_ALMIEN
;
/* 计算第一次中断的时间
* 现在灯是亮的,第一次中断即在需要灯灭时
*/
RTT->RTT_AR = RTT_AR_ALMV(
ReadRTT_CRTV() + CalcRTTNeedInc(LED1_ON_MS) -1);

八、禁用看门狗

程序在运行若干秒之后,可能会看到一些不和谐的状况,比如某个LED灯不按照我们的设想快速闪动一两下。这是因为看门狗默认是开启的,而我们却从来没有“喂狗”,从而导致系统重置。现在我只需禁用看门狗即可:

WDT->WDT_MR = WDT_MR_WDDIS;

PS,完整程序代码:

这一部分完整代码放在下面,以后大概不会再在这个基础上修改了吧。

#include <sam.h>
//#include <sam4e_ek.h> /* LED 使用的GPIO引脚 */
#define LED0_GPIO PIO_PA0
#define LED1_GPIO PIO_PD20 /* LED 闪烁的周期 */
#define LED0_OFF_MS 500
#define LED0_ON_MS 200
#define LED1_OFF_MS 200
#define LED1_ON_MS 300 #define PRESCALE (1u<<10) inline uint32_t ReadRTT_CRTV(void)
{
//return (RTT->RTT_VR) & RTT_VR_CRTV_Msk;
uint32_t v1;
uint32_t v2;
while(1)
{
v1 = (RTT->RTT_VR) & RTT_VR_CRTV_Msk;
v2 = (RTT->RTT_VR) & RTT_VR_CRTV_Msk;
/* 通过连续读取两次RTT_VR的值以增加准备性 */
if (v1 == v2)
{
return v1;
}
}
} inline uint32_t CalcRTTNeedInc(unsigned int ms)
{
/* 计数器加一的频率 */
const uint32_t freq = CHIP_FREQ_SLCK_RC / PRESCALE;
/* 计算延迟后,计数器需要增加的值
* need_inc = ms /1000 / (1/freq) */
return (ms * freq / 1000);
} void Delay(unsigned int ms)
{
uint32_t begin_rttv = ReadRTT_CRTV();
uint32_t need_inc = CalcRTTNeedInc(ms);
uint32_t end_rttv = begin_rttv + need_inc; /* 等待*/
while(ReadRTT_CRTV() < end_rttv)
;
} /* RTT 中断处理函数
* 在这里主要就进行LED1引脚电平的切换了
*/
void RTT_Handler(void)
{
/* 通过读取状态寄存器清除Alarm */
uint32_t _ = RTT->RTT_SR;
uint32_t begin_rttv = ReadRTT_CRTV();
uint32_t int_gap_ms ;
uint32_t need_inc; if ((PIOD->PIO_ODSR & LED1_GPIO) == 0)
{
/* 现在引脚电平为低,LED是亮的 */
/* 灭灯 */
PIOD->PIO_SODR = LED1_GPIO;
/* 设置下次中断唤醒间隔的时间 */
int_gap_ms = LED1_OFF_MS;
}
else
{
/* 现在引脚电平为高,LED是灭的 */
/* 亮灯 */
PIOD->PIO_CODR = LED1_GPIO;
/* 设置下次中断唤醒间隔的时间 */
int_gap_ms = LED1_ON_MS;
} /* 计算并设置下一次中断的条件 */
need_inc = CalcRTTNeedInc(int_gap_ms);
RTT->RTT_AR = RTT_AR_ALMV(begin_rttv + need_inc - 1); return;
} int main(void)
{
/* 关闭看门狗 */
WDT->WDT_MR = WDT_MR_WDDIS;
/* 初始化PIO */
/* 让PIO控制器直接控制引脚 */
PIOA->PIO_PER = LED0_GPIO;
PIOD->PIO_PER = LED1_GPIO;
/* 引脚输出使能 */
PIOA->PIO_OER = LED0_GPIO;
PIOD->PIO_OER = LED1_GPIO;
/* 引脚输出写使能 */
PIOA->PIO_OWER = LED0_GPIO;
PIOD->PIO_OWER = LED1_GPIO; /* 初始化 RTT */
/* 启用中断 */
NVIC_ClearPendingIRQ(RTT_IRQn);
NVIC_EnableIRQ(RTT_IRQn);
RTT->RTT_MR = RTT_MR_RTPRES(PRESCALE)
| RTT_MR_RTTRST
| RTT_MR_ALMIEN
;
/* 计算第一次中断的时间
* 现在灯是亮的,第一次中断即在需要灯灭时
*/
RTT->RTT_AR = RTT_AR_ALMV(ReadRTT_CRTV() + CalcRTTNeedInc(LED1_ON_MS) -1); while (1) {
/* 设置PA0引脚为高电平,灯灭 */
PIOA->PIO_SODR = LED0_GPIO;
/* 延迟 */
Delay(LED0_OFF_MS); /* 设置PA0引脚为高电平,灯亮 */
PIOA->PIO_CODR = LED0_GPIO;
Delay(LED0_ON_MS);
}
return 0;
}

SAM4E单片机之旅——3、LED闪烁之定时器中断的更多相关文章

  1. SAM4E单片机之旅——23、在AS6(GCC)中使用FPU

    浮点单元(Floating Point Unit,FPU),是用于处理浮点数运算的单元. 为使用FPU,除了需要启用FPU外,还需要对编译器进行设置,以使其针对浮点运算生成特殊的指令.虽然在Atmel ...

  2. SAM4E单片机之旅——2、LED闪烁之轮询定时器

    之前我们使用空循环,达到了延迟的目的,但是这样子的延迟比较不精确.现在就使用实时定时器(RTT)来进行更为精确的计时.RTT虽然不是特别通用,在某些单片机上可能没有,但它较为简单. RTT内部有一个计 ...

  3. SAM4E单片机之旅——1、LED闪烁之空循环

    最近因为导师要写一本关于SAME4单片机的书籍,而我也作为一个嵌入式的初学者看了这本书.现在也让我写写几个小的程序,做做示例.既然写了文档之类的,就发到博客上来吧. 目前关于这芯片能参考的书籍大概就只 ...

  4. SAM4E单片机之旅——7、LED闪烁之TC中断

    RTT主要用做一个全局的定时器,而且不太通用.现在尝试使用一个更为通用的定时器进行定时:定时计数器(Timer Counter, TC). TC提供了广泛的功能,主要可以分为对输入的测量,以及波形的输 ...

  5. SAM4E单片机之旅——4、LED闪烁之PWM

    两个LED灯虽然可以闪了,但是总是需要CPU的参与.现在尝试使用一种更为自动化的方法:让脉宽调制(PWM)控制器输出具有一定周期和占空比的方波,以此控制LED灯的亮灭. 一.实现思路 依然使用蓝色和琥 ...

  6. SAM4E单片机之旅——6、LED闪烁之按钮控制

    现在试试用按钮控制LED灯……让LED在一个按钮按下时亮起:弹起时灭掉. 主要目的是学习GPIO的输入及中断. 一. 电路 图中的J39-n是几个跳线插座,位置在开发板LCD附近,往下进行前要先确保跳 ...

  7. SAM4E单片机之旅——5、LED呼吸和PWM

    PWM在高频情况下,一个很好的用处就是通过控制占空比来控制输出的功率,比如控制风扇转速.LED灯的亮度等.这次就利用PWM的中断功能,动态改变脉冲的占空比,来实现呼吸灯的效果. 一.实现思路 PWM可 ...

  8. SAM4E单片机之旅——8、UART初步

    通信还是比让LED灯闪烁实用得多的. 这次试试使用UART,实现开发版和PC间的通信.功能比较简单,就是把PC发向开发版的内容发送回去.这次主要介绍一下UART的配置,至于通信,则使用较为简单的不断查 ...

  9. SAM4E单片机之旅——24、使用DSP库求向量数量积

    DSP(Digital Signal Processing,数字信号处理)中会使用大量的数学运算.Cortex-M4中,配置了一些强大的部件,以提高DSP能力.同时CMSIS提供了一个DSP库,提供了 ...

随机推荐

  1. codeforces 900D 数论+组合+容斥原理

    问有多少个这样的数字序列 所有数的GCD等于x 并且 所有数的和等于y 题解: 非常难有思路啊 看题解后过的. 考虑序列GCD为x的倍数 即GCD = n*x 和当然都为y 这个条件不要忘了 这样我们 ...

  2. Linux System Programming 学习笔记(八) 文件和目录管理

    1. 文件和元数据 每个文件都是通过inode引用,每个inode索引节点都具有文件系统中唯一的inode number 一个inode索引节点是存储在Linux文件系统的磁盘介质上的物理对象,也是L ...

  3. 洛谷 [P2859] 摊位预定

    贪心 #include <iostream> #include <cstdio> #include <cstring> #include <algorithm ...

  4. Python实现删除文件夹内规定时间内的文件

    需求:  在测试程序的时候,程序会大批量的上传文件到规定目录,然后文件根据日期DAY新建文件夹存放,比如28号上传的文件放到  .../28/* 内,29号上传的文件放到 .../29/*内,因为需要 ...

  5. Codeforces 842C Ilya And The Tree 树上gcd

    题目链接 题意 给定一棵根为\(1\)的树.定义每个点的美丽值为根节点到它的路径上所有点的\(gcd\)值.但是对于每个点,在计算它的美丽值时,可以将这条路径上某个点的值变为\(0\)来最大化它的美丽 ...

  6. Unity3d Inspector面板实现set/get访问器

    简单说一下属性和字段的区别:字段就是成员变量,而属性确实提供给外部访问内部成员变量的接口.之所以会有属性的出现,就是为了避免外部对类的成员的直接访问,通俗的说就是OOP中的封装思想. using Un ...

  7. cmake ccmake

    下载libqrencode源码编译过程 git clone https://github.com/fukuchi/libqrencode.git 2001  mkdir build 2002  cd ...

  8. css查缺补漏2

    15.布局流程 一.确定页面的版心; 二.确定页面中的行模块,以及每个页面中的列模块 三.制作HTML结构 例:.top+.banner+(.main>.left+.right)+.footer ...

  9. 小W摆石子

    可以确定, 最后围成是 一个长方形 + 多出一列 的形状. 而且多出的那一列应该是和较短的边相邻. 贴代码. #include<iostream> #include<algorith ...

  10. 【java】RC4加密转16进制获取长度为40的不重复优惠码字符串 【未优化版本】

    需求:需要一串给各机构独有的优惠码 间接需求:固定长度.不重复.没有规律可循 实现思想如下: 1.首先获取一个UUID 2.去除UUID中的“-” 3.小写转大写 4.获取一个固定长度字符串 5.按照 ...