STM32(五)—— EXTI

一、EXTI外部中断

1.1  中断系统

  • 中断:在主程序运行过程中,出现了特定的中断触发条件(中断源),使得CPU暂停当前正在运行的程序,转而去处理中断程序,处理完成后又返回原来被暂停的位置继续运行
  • 中断优先级:当有多个中断源同时申请中断时,CPU会根据中断源的轻重缓急进行裁决,优先响应更加紧急的中断源
  • 中断嵌套:当一个中断程序正在运行时,又有新的更高优先级的中断源申请中断,CPU再次暂停当前中断程序,转而去处理新的中断程序,处理完成后依次进行返回

1.2  STM32中断

  • 68个可屏蔽中断通道,包含EXTI、TIM、ADC、USART、SPI、I2C、TTC等多个外设
  • 使用NVIC统一管理中断,每个中断通道都拥有16个可编程的优先等级,可对优先级进行分组进一步设置抢占优先级响应优先级

1.3  NVIC基本结构

1.4  NVIC优先级分组

  • NVIC的中断优先级由优先级寄存器的4位(0~15)决定,这4位可以进行切分,分为高n位的抢占优先级和低4-n位的响应优先级
  • 抢占优先级高的可以中断嵌套,响应优先级高的可以优先排队,抢占优先级和响应优先级均相同的按中断号排队

 

1.5  EXTI简介

  • EXTI(Extern Interrupt)外部中断
  • EXTI可以监测指定GPIO口的电平信号,当其指定的GPIO口产生电平变化化时EXTI将立即向NVIC发出中断申请,经过NVIC裁决后即可中断CPU主程序,使CPU执行EXTI对应的中断程序
  • 支持的触发方式:上升沿/下降沿/双边沿/软件触发
  • 支持的GPIO口:所有的GPIO口,但相同的Pin口不能同时触发中断
  • 通道数:16个GPIO_Pin,外加PVD输出、RTC时钟、USB唤醒、以太网唤醒
  • 触发响应方式:中断响应(执行中断函数)/事件响应(不通向CPU,通向外设,触发其他外设)

适用于突发的,快速的,短暂的

1.6  EXTI基本结构

1.7  AFIO复用IO口

  • AFIO主要用于引脚复用功能的选择和重定义
  • 在STM32中,AFIO主要完成两个任务:复用功能引脚重映射、中断引脚选择

1.8  EXTI框图

二、对射式红外传感器计次&旋转编码器计次

2.1  AFIO相关函数介绍

//AFIO恢复缺省配置
void GPIO_AFIODeInit(void);
//锁定GPIO某个引脚的配置
void GPIO_PinLockConfig(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);
//配置AFIO的事件输出功能
void GPIO_EventOutputConfig(uint8_t GPIO_PortSource, uint8_t GPIO_PinSource);
void GPIO_EventOutputCmd(FunctionalState NewState);
//用于引脚重映射
void GPIO_PinRemapConfig(uint32_t GPIO_Remap, FunctionalState NewState);
//配置AFIO数据选择器,选择中断引脚
void GPIO_EXTILineConfig(uint8_t GPIO_PortSource, uint8_t GPIO_PinSource);
//和以太网有关
void GPIO_ETH_MediaInterfaceConfig(uint32_t GPIO_ETH_MediaInterface);

2.2  EXTI相关函数介绍

//EXTI恢复缺省配置
void EXTI_DeInit(void);
//EXTI初始化
void EXTI_Init(EXTI_InitTypeDef* EXTI_InitStruct);
//EXTI初始化函数的结构体参数初始化
void EXTI_StructInit(EXTI_InitTypeDef* EXTI_InitStruct);
//软件触发外部中断函数
void EXTI_GenerateSWInterrupt(uint32_t EXTI_Line);
//获取EXTI标志位
FlagStatus EXTI_GetFlagStatus(uint32_t EXTI_Line);
//清除EXTI标志位
void EXTI_ClearFlag(uint32_t EXTI_Line);
//获取EXTI中断标志位
ITStatus EXTI_GetITStatus(uint32_t EXTI_Line);
//清除EXTI中断标志位
void EXTI_ClearITPendingBit(uint32_t EXTI_Line);

2.3  NVIC相关函数介绍

//进行中断分组
void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup);
//初始化NVIC
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);
//设置中断向量表
void NVIC_SetVectorTable(uint32_t NVIC_VectTab, uint32_t Offset);
//系统低功耗配置
void NVIC_SystemLPConfig(uint8_t LowPowerMode, FunctionalState NewState);

中断分组一个芯片只能用一种,整个工程只用执行一次,如果在模块中分组,那么每一个模块分组应该相同

2.4  对射式红外传感器计次

CounterSensor.c

#include "stm32f10x.h"                  // Device header


uint16_t CountSensor_Count;

void CountSensor_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);  //开启AFIO的时钟
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	
	GPIO_Init(GPIOF, &GPIO_InitStructure);
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOF, GPIO_PinSource13);  //开启AFIO数据选择,连接到EXTI
	
	EXTI_InitTypeDef EXTI_InitStructure;
	EXTI_InitStructure.EXTI_Line = EXTI_Line13;  //选择中断通道
	EXTI_InitStructure.EXTI_LineCmd = ENABLE;  //选择中断通道状态
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //选择触发响应模式
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; //选择触发方式
	EXTI_Init(&EXTI_InitStructure);
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn; //在stm32f10x.h文件中查找同型号下的中断通道
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  //指定中断通道状态
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;  //设置抢占优先级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;  //设置响应优先级
	NVIC_Init(&NVIC_InitStructure);
}

uint16_t CountSensor_Get(void)
{
	return CountSensor_Count;
}

void EXTI15_10_IRQHandler(void)  //中断函数名字在启动文件里找
{
	if(EXTI_GetITStatus(EXTI_Line13) == SET) //判断中断标志
	{
		CountSensor_Count++;
		EXTI_ClearITPendingBit(EXTI_Line13);  //清除中断标志
	}
}


main.c

#include "stm32f10x.h"  // Device header
#include "Delay.h"
#include "OLED.h"
#include "CountSensor.h"

uint8_t KeyNum;


int main(void)
{
	OLED_Init();
	CountSensor_Init();
	
	OLED_ShowString(1, 1, "Count:");
	while(1)
	{
		OLED_ShowNum(1, 7, CountSensor_Get(), 5);
	}
}

2.5  旋转编码器计次

Encoder.c

#include "stm32f10x.h"                  // Device header

int16_t Encoder_Count;

void Encoder_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_15;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	
	GPIO_Init(GPIOF, &GPIO_InitStructure);
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOF, GPIO_PinSource13 | GPIO_PinSource15);
	
	EXTI_InitTypeDef EXTI_InitStructure;
	EXTI_InitStructure.EXTI_Line = EXTI_Line13 | EXTI_Line15;
	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
	EXTI_Init(&EXTI_InitStructure);
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
	NVIC_Init(&NVIC_InitStructure);
}

int16_t Encoder_Get(void)
{
	int16_t Temp;
	Temp = Encoder_Count;
	Encoder_Count = 0;
	return Temp;
}

void EXTI15_10_IRQHandler(void)
{
	if(EXTI_GetITStatus(EXTI_Line13) == SET)
	{
		if(GPIO_ReadInputDataBit(GPIOF, GPIO_Pin_15) == 0)
		{
			Encoder_Count--;
		}
		EXTI_ClearITPendingBit(EXTI_Line13);
	}
	if(EXTI_GetITStatus(EXTI_Line15) == SET)
	{
		if(GPIO_ReadInputDataBit(GPIOF, GPIO_Pin_13) == 0)
		{
			Encoder_Count++;
		}
		EXTI_ClearITPendingBit(EXTI_Line15);
	}
}


main.c

#include "stm32f10x.h"  // Device header
#include "Delay.h"
#include "OLED.h"
//#include "CountSensor.h"
#include "Encoder.h"

int16_t Num;


int main(void)
{
	OLED_Init();
	Encoder_Init();
//	CountSensor_Init();
	OLED_ShowString(1, 1, "Num:");
//	OLED_ShowString(1, 1, "Count:");
//	while(1)
//	{
//		OLED_ShowNum(1, 7, CountSensor_Get(), 5);
		Num += Encoder_Get();
		OLED_ShowNum(1, 7, Num, 5);
//	}
}

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
STM32中,外部中断的初始化可以通过以下步骤完成: 1. 配置外部中断线:根据需要选择外部中断线,例如EXTI_Line4。\[2\] 2. 初始化EXTI_InitTypeDef结构体:定义一个EXTI_InitTypeDef结构体,并设置其成员变量,包括EXTI_Line(指定中断线)、EXTI_Mode(指定是事件还是中断)、EXTI_Trigger(指定触发方式)和EXTI_LineCmd(是否使能)。\[2\] 3. 调用EXTI_Init函数:将初始化好的EXTI_InitTypeDef结构体作为参数传入EXTI_Init函数,以初始化外设EXTI寄存器。\[2\] 以下是一个外部中断初始化的示例代码: ```c EXTI_InitTypeDef EXTI_InitStructure; EXTI_InitStructure.EXTI_Line = EXTI_Line4; EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; EXTI_InitStructure.EXTI_LineCmd = ENABLE; EXTI_Init(&EXTI_InitStructure); ``` 在上述代码中,我们选择了EXTI_Line4作为外部中断线,并设置了中断模式为中断,触发方式为下降沿触发,使能外部中断。通过调用EXTI_Init函数,将配置好的EXTI_InitStructure结构体传入,即可完成外部中断的初始化。\[2\] #### 引用[.reference_title] - *1* *3* [stm32外部中断](https://blog.csdn.net/qq_62846920/article/details/124655894)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [stm32学习笔记——外部中断](https://blog.csdn.net/Zn_Hua/article/details/120780463)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值