STM32f10x学习----中断和事件 后附具体操作及使用过程中遇到的问题

学习某一个东西,我们首先要了解这个东西的定义是什么,用来干什么的,怎么用,用的过程中有什么注意事项,这些都OK了,那么我们就算是基本掌握他了。

0 前言

众所周知,处理器的速度跟外围硬件设备的速度往往不在一个数量级上,因此,如果内核采取让处理器向硬件发出一个请求,然后专门等待回应的办法,显然降低内核效率。

既然硬件的响应这么慢,那么内核就应该在此期间处理其他事务,等到硬件真正完成了请求的操作之后,再回过头来对它进行处理。想要实现这种功能,轮询(polling)可能会是一种解决办法。可以让内核定期对设备的状态进行查询,然后做出相应的处理。不过这种方法很可能会让那个内核做不少无用功,因为无论硬件设备是正在忙碌着完成任务还是已经大功告成,轮询总会周期性的重复执行。更好的办法是由我们来提供一种机制,让硬件在需要的时候再向内核发出信号(变内核主动为硬件主动),这就是中断机制。

1.中断和事件

1.1概述

1.1.1 中断和事件的定义

中断:指当出现需要时,CPU暂时停止当前程序的执行转而执行处理新情况的程序和执行过程。即在程序运行过程中,系统出现了一个必须由CPU立即处理的情况,此时,CPU暂时中止程序的执行转而处理这个新的情况的过程就叫做中断。

事件:事件是中断的触发源,开放了对应的中断屏蔽位,则事件可以触发相应的中断。事件还是其它一些操作的触发源,比如 DMA,还有TIM中影子寄存器的传递与更新;而中断是不能触发这些操作的,所以要把事件与中断区分开。

简单点就是中断一定要有中断服务函数,但是事件却没有对应的函数.但是事件可以触发其他关联操作,比如触发DMA,触发ADC采样等.可以在不需要CPU干预的情况下,执行这些操作.中断则必须要CPU介入.

1.1.2 中断的类型

在cortex-M3中,将异常的类型也分为了两类

1.来自CPU的内部事件或程序执行中的事件引起的过程。如由于CPU本身故障、程序故障和请求系统服务的指令等引起的中断被称为异常。

2.由CPU以外的事件引起的中断,如I/O中断、时钟中断、控制台等则被称为中断。

Cortex‐M3 在内核水平上搭载了一个异常响应系统,支持为数众多的系统异常和外部中断。包括15个系统异常,和最多240个外部中断。具体使用了这240个中断源中的多少个,则由芯片制造商决定。除了个别异常的优先级被定死外,其它异常的优先级都是可编程的。详细的中断编号请参考《cortex-M3异常向量表》。

1.1.3 中断的处理过程

在微机系统中,对于外部中断,中断请求信号是由外部设备产生,并施加到CPU的NMI或INTR引脚上,CPU通过不断地检测NMI和INTR引脚信号来识 别是否有中断请求发生。对于内部中断,中断请求方式不需要外部施加信号激发,而是通过内部中断控制逻辑去调用。无论是外部中断还是内部中断,中断处理过程 都要经历以下步骤: 请求中断→响应中断→关闭中断→保留断点→中断源识别→保护现场→中断服务子程序→恢复现场→中断返回。

请求中断

当某一中断源需要CPU为其进行中断服务时,就输出中断请求信号,使中断控制系统的中断请求触发器置位,向CPU请求中断。系统要求中断请求信号一直保持到CPU对其进行中断响应为止。

中断响应

CPU对系统内部中断源提出的中断请求必须响应,而且自动取得中断服务子程序的入口地址,执行中断服务子程序。对于外部中断,CPU在执行当前指令的最后一个时钟周期去查询INTR引脚,若查询到中断请求信号有效,同时在系统开中断(即IF=1)的情 况下,CPU向发出中断请求的外设回送一个低电平有效的中断应答信号,作为对中断请求INTR的应答,系统自动进入中断响应周期。

关闭中断

CPU响应中断后,输出中断响应信号,自动将状态标志寄存器FR或EFR的内容压入堆栈保护起来,然后将FR或EFR中的中断标志位IF与陷阱标志位TF清零,从而自动关闭外部硬件中断。因为CPU刚进入中断时要保护现场,主要涉及堆栈操作,此时不能再响应中断,否则将造成系统混乱。

保护断点

保护断点就是将CS和IP/EIP的当前内容压入堆栈保存,以便中断处理完毕后能返回被中断的原程序继续执行,这一过程也是由CPU自动完成。

中断源识别

当系统中有多个中断源时,一旦有中断请求,CPU必须确定是哪一个中断源提出的中断请求,并由中断控制器给出中断服务子程序的入口地址,装入CS与IP/EIP两个寄存器。CPU转入相应的中断服务子程序开始执行。

保护现场

主程序和中断服务子程序都要使用CPU内部寄存器等资源,为使中断处理程序不破坏主程序中寄存器的内容,应先将断点处各寄存器的内容压入堆栈保护起来,再进入的中断处理。现场保护是由用户使用PUSH指令来实现的。

中断服务

中断服务是执行中断的主体部分,不同的中断请求,有各自不同的中断服务内容,需要根据中断源所要完成的功能,事先编写相应的中断服务子程序存入内存,等待中断请求响应后调用执行。

恢复现场

当中断处理完毕后,用户通过POP指令将保存在堆栈中的各个寄存器的内容弹出,即恢复主程序断点处寄存器的原值。

中断返回

在中断服务子程序的最后要安排一条中断返回指令IRET,执行该指令,系统自动将堆栈内保存的 IP/EIP和CS值弹出,从而恢复主程序断点处的地址值,同时还自动恢复标志寄存器FR或EFR的内容,使CPU转到被中断的程序中继续执行。

一个中断的产生,要首先经过边沿触发寄存器----》软件中断事件寄存器—》如果是软件中断就会到请求挂起寄存器,如果不是则会到事件屏蔽寄存器—》软件中断会到中断的屏蔽寄存器,不是软件中断则会到脉冲发生器—》软件中断还有最后一步就是到NVIC中断控制器,由他管理是否让CPU处理这个中断。下面是中断/事件的控制器框架图:

在这里插入图片描述

1.1.4 中断优先级管理

**Cortex-M3中定义了8个Bit用于设置中断源的优先级,用于管理中断,而STM32只选用其中的4个Bit。**在STM32的中断分组管理中STM32将中断分成5个组,分别为组0-4;同时,对每个中断设置一个抢占优先级和响应优先级。下图是中断分组的设置表:

在这里插入图片描述

当程序中设置中断分组为2时,即抢占优先级可以设置为0-3,响应优先级可以设置为0-3。

抢占优先级的级别高于响应优先级,而数值越小所代表的的优先级越高。

  1. 高优先级的抢占优先级是可以打断正在进行的低抢占优先级中断的;
  2. 抢占优先级相同的中断,高响应优先级不可以打断低响应优先级的中断;
  3. 抢占优先级相同的中断,当两个中断同时发生的情况下,哪个响应优先级高,哪个先执行;
  4. 如果两个中断的抢占优先级和响应优先级都是一样的话,则看哪个中断先发生就先执行;

2.使用示例

2.1 外部中断配置

例子: 按键中断

使用STM32库函数实现

.c文件内容

#include "key.h"
uint8_t Change_Flag=0;
void Key_GPIO_Config(void)

{		

    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA , ENABLE);  									  

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;	               //LED

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;   

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 
    
    GPIO_Init(GPIOB, &GPIO_InitStructure);	
    

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;				 
    
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; 						//PA10 上拉输入		
    GPIO_Init(GPIOA, &GPIO_InitStructure);

}

void EXTI_Key_Config(void)

{

    EXTI_InitTypeDef EXTI_InitStructure;

    GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource10);    //

    EXTI_InitStructure.EXTI_Line = EXTI_Line10;

    EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;

    EXTI_InitStructure.EXTI_Trigger = EXTI_Falling;//EXTI_Trigger_Falling; //下降沿中断

    EXTI_InitStructure.EXTI_LineCmd = ENABLE;

    EXTI_Init(&EXTI_InitStructure);
   

}

static void NVIC_Configuration(void)

{

    NVIC_InitTypeDef NVIC_InitStructure;

    /* Configure one bit for preemption priority */

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);

    NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =1;          

    NVIC_InitStructure.NVIC_IRQChannelSubPriority =0;

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

    NVIC_Init(&NVIC_InitStructure);

}


void board_init(void)
{
    NVIC_Configuration();
    EXTI_Key_Config();
    Key_GPIO_Config();
}


void EXTI15_10_IRQHandler(void)//这里为:EXTI15_10 (外部中断号的10~15都在这里实现)

{		

	 if(EXTI_GetITStatus(EXTI_Line10) != RESET)	//这里为判断相应的中断号是否进入中断,如果有多个中断的话。  
     {		 
	     Change_Flag =~Change_Flag;
         EXTI_ClearITPendingBit(EXTI_Line10);		//清中断		 

         if(Change_Flag!=0)		

             LED(ON);		

         else			

             LED(OFF); 

      }

}

.h内容

#ifndef  __KEY_H
#define  __KEY_H

#define LED(ON)      GPIO_RessetBits(GPIOB, GPIO_Pin_12)
#define LED(OFF)     GPIO_SetBits(GPIOB, GPIO_Pin_12)

void board_init(void);
#endif

3.问题总结

1.EXTI线16连接到PVD输出

​ EXTI线17连接到RTC闹钟事件

​ EXTI线18连接到USB唤醒事件

​ EXTI线19连接到以太网唤醒事件(只适用于互联型产品)

2.中断在发生以后,中断标志位必须手动清零,如果不清零,则无法进入下一次中断

中断标志相关函数:

1》EXTI_ClearFlag/EXTI_ClearITPendingBit 用来清除EXTI_Line指定的外部中断的pending bit(清除中断标识位)

​ 这两个函数一般是用在用户编写的中断处理函数的结束调用之前(在中断处理函数的最后)

​ void EXTI_ClearFlag ( uint32_t EXTI_Line )

​ void EXTI_ClearITPendingBit ( uint32_t EXTI_Line )

​ 2》EXTI_GetFlagStatus/EXTI_GetITStatus 用来获取EXTI_Line指定的外部中断的状态

​ FlagStatus EXTI_GetFlagStatus ( uint32_t EXTI_Line )

功能:检查指定的外部中断线的标志是否被置位(如果在EXTI_IMR和EXTI_EMR中允许产生该中断,则此时将产生一个中断)

注释:该函数没有检查EXTI_IMR和EXTI_EMR状态,它只是纯粹读取中断标志位的状态,但是不一定会响应中断

​ ITStatus EXTI_GetITStatus ( uint32_t EXTI_Line )

注释:除了读取中断标志位,还查看 EXT_IMR 寄存器是否对该中断进行屏蔽,在中断挂起 & 没有屏蔽的情况下就会响应中断

3.程序中有多个中断的时候要配置中断分组便于管理中断

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 设置中断分组,设定抢占优先级和响优先级的位数

设置中断分组只能在程序初始化的时候设置 ,并且只能设置一次,不然会导致中断管理混乱。

4.中断处理函数应该无返回值,无参数,应该尽量简短并且在中断处理函数中:

  1. 不能延时

  2. 不能用printf —>除非重定向到了调试串口

  3. 中断处理函数要 “短小精悍” 在中断中写while()函数不可取 可以设置一个FLAG在MAIN函数中处理

5.向量中断就是不同的中断有不同的入口地址,非向量中断就只有一个入口地址,

​ 进去了再判断中断标志来识别具体是哪个中断。向量中断实时性好,非向量中断简单

​ 向量中断控制器NVIC具有32个中断请求输入,可将其编程分为3类,FIQ,向量IRQ和非向量IRQ。

​ 向量中断------由硬件提供中断服务程序入口地址; 经常用于异常,reset;控制器直接将PC赋值

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值