外部中断相关分析
扫描二维码
随时随地手机看文章
//======================================================//
**基于STM32的按键控制与外部中断实例详解
**为了减少重复的内容,所以将外部中断的例程与按键控制实例
放在一起学习。
一开始先在51黑论坛率先上传了本文,上传时间不分什么顺序,只是希望在学习中慢慢写下自己的思路。
//======================================================//
/********************************************************/
因为涉及的内容较多,这里大体说一下整片文章的内容分布:
1)、按键控制实例
1、按键的相关信息描述
2、按键的初始化代码实现
3、常用按键检测及控制代码(主要学习其实现思路)
2)、外部中断实例
1、外部中断的相关信息描述
2、外部中断的初始化代码实现
/********************************************************/
==================================华丽的分界线=================================
//=========================按键控制实例=============================//
==================================华丽的分界线=================================
首先,先来讲述简单的按键控制,下面来看看按键是什么东西,开发的时候按键又有哪些要注意的事情.
释义:
按键开关是一种电子开关,属于电子元器件类,使用时以满足操作力的条件向开关操作方向施压开关功能闭合接通,当撤销压力时开关即断开,其内部结构是靠金属弹片受力变化来实现通断的。
一般在开发中涉及按键的一般是按键消抖。按键消抖通常的按键所用开关为机械弹性开关,当机械触点断开、闭合时,由于机械触点的弹性作用,一个按键开关在闭合时不会马上稳定地接通,在断开时也不会一下子断开。也就是在闭合及断开的那一瞬间会有一小段的不稳定状态,即伴随着连续的抖动。按键在电路中一般也就0跟1两中状态,正常情况下,按键没有按下的情况下,按键所连的引脚状态一般都是为1。原因是一般会在引脚上加上一个上拉,然后再与按键相连,按键的另一端接地,所以在按键没有按下的情况下,引脚返回的一般都是高电平,这是电路决定的。如下图:
按键的抖动就是介于1跟0的不定状态。所以为了取得稳定状态,需要在延迟取状态值,也就是既然知道有抖动存在了,那么就把取值的时间延迟一下。还有一个东西跟抖动有关,那就是触发沿的设置,比如上升沿和下降沿的触发。上升沿是按键松开后那一瞬间,电平从0到1的过程,下降沿就与之相反,也就是按键按下的一瞬间,电平从1降为0的过程。边沿触发一般会用在外部中断的过程中。下面的内容也会讲到。
按键相对来说比较简单,按键的初始化跟LED和BEEP蜂鸣器几乎一致,不同之处也就在于按键是输入设备,所以GPIO的模式上一般选为输入模式,紧接着就要设置上拉/下拉输入模式,最后设置初始状态,这个要根据电路来间接。
//=======================key.c===============================//
/************************************************
接口:
key0引脚接在 PE4 低电平有效
key1引脚接在 PE3 低电平有效
key_up引脚接在 PA0 高电平有效
************************************************/
#include "stm32f10x.h"
#include "key.h"
#include "delay.h"
void key_Init(void)
{
RCC ->APB2ENR |= 1 << 2;//使能 PORTA 时钟
RCC ->APB2ENR |= 1 << 6;//使能 PORTE 时钟
//key_up配置
GPIOA ->CRL &= ~(15 << 0);//先设置输入模式
GPIOA ->CRL |= 1 << 3; //PA.0 下拉输入模式
GPIOA ->ODR &= ~(1 << 0); //设置下拉,即设置初始状态为低电平
//key1配置
GPIOE ->CRL &= ~(3 << 13);//先设置输入模式
GPIOE ->CRL |= 1 << 15; //PE.3 上拉输入模式
GPIOE ->ODR |= 1 << 3; //设置上拉,即设置初始状态为高电平
//key0配置
GPIOE ->CRL &= ~(3 << 17);//先设置输入模式
GPIOE ->CRL |= 1 << 19; //PE.4 上拉输入模式
GPIOE ->ODR |= 1 << 4; //设置上拉,即设置初始状态为高电平
}
/*************************************************************
**功 能:按键处理函数
**输入参数:无
**输出参数:
//mode:0,不支持连续按;1,支持连续按;
//0,没有任何按键按下
//1,KEY0 按下
//2,KEY1 按下
//3,KEY_UP 按下 即 WK_UP
**注 释:注意此函数有响应优先级,KEY0>KEY1>KEY_UP!!
***********************************************************/
u8 KEY_Scan(u8 mode)
{
static u8 key_up=1; //按键按松开标志
if(mode)key_up=1; //支持连按
if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))
{
delay_ms(10); //去抖动
key_up=0;
if(KEY0==0)return 1;
else if(KEY1==0)return 2;
else if(WK_UP==1)return 3;
}
else if(KEY0==1&&KEY1==1&&WK_UP==0)key_up=1;
return 0;// 无按键按下
}
//======================================================//
注释:按键检测函数,原理是检测按键的电平,而mode的作用就是一直把key_up置1,那么就会一直进入按键按下的检测,而不会进入按键松开的检测。
if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))这句是进行按键按下的检测,
else if(KEY0==1&&KEY1==1&&WK_UP==0)这句是进行按键松开的检测。
如果没有按键按下的话,虽然key_up为1,但是没有按键,那么就只会返回0。
==================================华丽的分界线=================================
//===========================key.h===========================//
#ifndef __KEY_H_
#define __KEY_H_
#include "stm32f10x.h"
#include "sys.h"
#define KEY0 PEin(4) //PE4
#define KEY1 PEin(3) //PE3
#define WK_UP PAin(0) //PA0 WK_UP 即 KEY_UP
#define KEY0_PRES 1 //KEY0 按下
#define KEY1_PRES 2 //KEY1 按下
#define WKUP_PRES 3 //KEY_UP 按下(即 WK_UP/KEY_UP)
extern void key_Init(void);
extern u8 KEY_Scan(u8 mode);
#endif
//======================================================//
==================================华丽的分界线=================================
//==========================main.c============================//
/*************************************************************
**功 能:按键处理函数
//1,KEY0 按下 控制LED0
//2,KEY1 按下 控制LED1
//3,KEY_UP 按下 即 WK_UP 控制蜂鸣器
***********************************************************/
#include "stm32f10x.h"
#include "sys.h"
#include "delay.h"
#include "led.h"
#include "beep.h"
#include "key.h"
int main(void)
{
u8 key=0;
Stm32_Clock_Init(9); //系统时钟设置
delay_init(72); //延时初始化
led_Init();//LED初始化
beep_Init(); //beep初始化
key_Init(); //key初始化
while(1)
{
key=KEY_Scan(0); //得到键值
if(key)
{
switch(key)
{
case WKUP_PRES: //控制蜂鸣器
GPIOB ->ODR |= (1 << 8);//PB.8 输出高,即蜂鸣器开
delay_ms(10000);//延时
GPIOB ->ODR &= ~(1 << 8);//PB.8 输出低,即蜂鸣器关
break;
case KEY1_PRES: //控制 LED1
GPIOB ->ODR &= ~ (1 << 5);//PB.5输出低电平,即点亮LED
delay_ms(10000); //延时
GPIOB ->ODR |= 1 << 5;//PB.5输出高电平,即灭掉LED
break;
case KEY0_PRES: //控制 LED0
GPIOE ->ODR &= ~(1 << 5);//PB.5输出低电平,即点亮LED
delay_ms(10000); //延时
GPIOE ->ODR |= 1 << 5;//PB.5输出高电平,即灭掉LED
break;
}
}
else delay_ms(10);
}
}
//======================================================//
==================================华丽的分界线=================================
//=========================外部中断实例=============================//
**注释:特别鸣谢CSDN博客的博主->“zzwdkxx”的博文,在外部中断的学习中直接的帮助!
博文地址:http://blog.csdn.net/zzwdkxx/article/details/9036679
==================================华丽的分界线=================================
接着下面将进行外部和中断的讲解,讲之前先对中断先做一定了解。
//============================百度搜索==========================//
简介:
中断是处理器处理外部突发事件的一个重要技术。它能使处理器在运行过程中对外部事件发出的中断请求及时地进行处理,处理完成后又立即返回断点,继续进行处理器原来的工作。引起中断的原因或者说发出中断请求的来源叫做中断源。根据中断源的不同,可以把中断分为硬件中断和软件中断两大类,而硬件中断又可以分为外部中断和内部中断 两类。
外部中断一般是由计算机外设发出的中断请指求,如:键盘中断、打印机中断、定时器中断等。外部中断是可以屏蔽的中断,也就是说,利用中断控制器可以屏蔽这些外部设备 的中断请求。
内部中断是指因硬件出错(如突然掉电、奇偶校验错等)或运算出错(除数为零、运算溢出、单步中断等)所引起的中断。内部中断是不可屏蔽的中断。
软件中断其实并不是真正的中断,它们只是可被调用执行的一般程序。
优先级:
CPU为了处理并发的中断请求,规定了中断的优先权,中断优先权由高到低的顺序是: (1)除法错、溢出中断、软件中断 (2)不可屏蔽