stm32——外部中断EXTI

        上回书说到定时器的级联,今天来谈谈外部中断EXTI。我使用的是STM32F103C8T6的学习板。仅供大家参考。

        什么是中断呢?中断是指计算机在执行程序的过程中,当出现某些异常情况或特殊事件(例如外部设备请求、定时时间到达、程序错误等)时,计算机暂停当前正在执行的程序,转而去处理这些异常情况或特殊事件的机制。也就是只要你触发了中断,只有等把中断的事件处理完了,才能去运行其他的程序。stm32的中断非常强大,每个外设都可以产生中断,这里呢,我就单纯的说说外部中断按键的方式如何去控制小灯的亮灭。也就是我们常说的独立按键怎么用。继续往下看。

        EXTI(External Interrupt/event controller) 是外部中断/事件控制器,管理了控制器的20个中断/事件线。每个中断/事件线都对应一个边沿检测器,可以实现输入信号的上升沿检测和下降沿的检测。EXTI可以实现对每个中断/事件线进行单独配置,可以单独配置为中断或者事件,以及触发事件的属性。

EXTI功能框图

        

中断/事件线

        EXTI一共有20个中断/事件线。每个GPIO口都可以被设置成输入线,占用EXTI0~EXTI15,也就是PX0,就对应着EXTI0,一一对应。

        我们先来看看程序,首先是开启外部中断的时候,我们需要用到两个时钟GPIOC端口(对应外部中断的端口)和AFIO端口(EXTI所需要的端口)。

void RCC_Configuration(void)
{
  	/* 时钟开启 */
  	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);
}

        接下来,配置GPIO口,NVIC中断以及EXTI初始化。这里我使用的是GPIOC的line13作为它的外部中断,对应单片机上的KEY2按键,通过控制GPIOA上PIN3的LED3小灯来观测现象。



void GPIO_Configuration(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOC, &GPIO_InitStructure);
}


void EXIT_Configuration(void)
{
	 EXTI_InitTypeDef EXTI_InitStructure;
	 GPIO_EXTILineConfig(GPIO_PortSourceGPIOC, GPIO_PinSource13);

  	EXTI_InitStructure.EXTI_Line = EXTI_Line13;
  	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
  	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
  	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
  	EXTI_Init(&EXTI_InitStructure);
}



void NVIC_Configuration(void)
{								
	NVIC_InitTypeDef NVIC_InitStructure;
	 
#ifdef  VECT_TAB_RAM  
  		
 		NVIC_SetVectorTable(NVIC_VectTab_RAM , 0x0); 
#else 	
  		
  		NVIC_SetVectorTable(NVIC_VectTab_FLASH , 0x0);   
#endif

	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);

	NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_Init(&NVIC_InitStructure);
}

        配置是配置完了接下来就是中断函数。大家可以打开自己的启动文件,来看看对应中断函数的命名,0到4是每个都有自身的一个中断函数。5~9是共用中断EXTI9_5_IRQHandler(),10~15是共用一个中断函数EXTI15_10_IRQHandler。因为我用的是GPIOC的13引脚,我使用的中断就是EXTI15_10_IRQHandler。这里要注意使用对应的GPIO口要用上GPIO_EXTILineConfig(GPIO_PortSourceGPIOC, GPIO_PinSource13);这个语句。否则无法看到对应的现象。接下来是中断程序。

void EXTI15_10_IRQHandler(void)
{
	
	if(EXTI_GetITStatus(EXTI_Line13) != RESET) 
     {
        GPIO_WriteBit(GPIOA, GPIO_Pin_3,(BitAction)((1-GPIO_ReadOutputDataBit(GPIOA,GPIO_Pin_3))));
        EXTI_ClearITPendingBit(EXTI_Line13); 
     }

}

        大家看看现象吧,应该是D3受到KEY2按键的控制。使灯发生亮灭的操作。

        谈到中断,这里就要考虑到中断优先级的问题,有两个优先级的概念,分别是抢占式优先级和响应优先级,每个中断源都需要被指定这两种优先级。具有高抢占式优先级的中断可以在具有低抢占式优先级的中断处理过程中被响应,即中断嵌套,或者说高抢占式优先级的中断可以嵌套低抢占式优先级的中断。当两个中断源的抢占式优先级相同时,这两个中断将没有嵌套关系。当一个中断到来后,如果正在处理另一个中断,这个后到来的中断就要等到前一个中断处理完之后才能被处理。如果这两个中断同时到达,则中断控制器根据它们的响应优先级高低来决定先处理哪一个;如果它们的抢占式优先级和响应优先级都相等,则根据它们在中断表中的排位顺序决定先处理哪一个。优先级是从0开始是最高的优先级,优先级越大越低。大家可以结合串口通信,设置EXTI0、EXTI1、EXTI2这三个外部中断来看看。

        这里就不过多说了给个简单的代码,中断配置和中断函数。请看代码。

void NVIC_Configuration(void)
{
    NVIC_InitTypeDef NVIC_InitStructure;

    // 如果定义了 VECT_TAB_RAM 宏
    #ifdef  VECT_TAB_RAM  
        // 将中断向量表设置为位于 RAM 中,并设置偏移量为 0
        NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); 
    // 否则
    #else  
        // 将中断向量表设置为位于 FLASH 中,并设置偏移量为 0
        NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);   
    #endif

    // 设置中断优先级分组为 2
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

    // 配置 EXTI0 中断
    NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;
    // 设置抢占优先级为 2
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
    // 设置响应优先级为 0
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    // 使能该中断通道
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    // 初始化 EXTI0 中断
    NVIC_Init(&NVIC_InitStructure);

    // 配置 EXTI1 中断
    NVIC_InitStructure.NVIC_IRQChannel = EXTI1_IRQn;
    // 设置抢占优先级为 1
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
    // 设置响应优先级为 0
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    // 使能该中断通道
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    // 初始化 EXTI1 中断
    NVIC_Init(&NVIC_InitStructure);

    // 配置 EXTI2 中断
    NVIC_InitStructure.NVIC_IRQChannel = EXTI2_IRQn;
    // 设置抢占优先级为 0
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    // 设置响应优先级为 0
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    // 使能该中断通道
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    // 初始化 EXTI2 中断
    NVIC_Init(&NVIC_InitStructure);
}

        中断函数

/*******************************************************************************
* Function Name  : EXTI0_IRQHandler
* Description    : This function handles External interrupt Line 0 request.
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
void EXTI0_IRQHandler(void)
{
	printf("\r\nEXIT0 IRQHandler enter.\r\n");
	
	EXTI_GenerateSWInterrupt(EXTI_Line1);
	
	printf("\r\nEXIT0 IRQHandler return.\r\n");
	
	EXTI_ClearFlag(EXTI_Line0);
}

/*******************************************************************************
* Function Name  : EXTI1_IRQHandler
* Description    : This function handles External interrupt Line 1 request.
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
void EXTI1_IRQHandler(void)
{
	printf("\r\nEXIT1 IRQHandler enter.\r\n");
	
	EXTI_GenerateSWInterrupt(EXTI_Line2);
	
	printf("\r\nEXIT1 IRQHandler return.\r\n");
	
	EXTI_ClearFlag(EXTI_Line1);
}

/*******************************************************************************
* Function Name  : EXTI2_IRQHandler
* Description    : This function handles External interrupt Line 2 request.
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
void EXTI2_IRQHandler(void)
{
	printf("\r\nEXIT2 IRQHandler enter.\r\n");
	
	printf("\r\nEXIT2 IRQHandler return.\r\n");
	
	EXTI_ClearFlag(EXTI_Line2);
}

        好了,今天就说这么多。

        欲知后事如何,且听下回分解。OVO......

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

YOLOV8花朵实例分割实战

原文:YOLOV8花朵实例分割实战 - 知乎 (zhihu.com) 一、代码: https://github.com/ultralytics/ultralytics​github.com/ultralytics/ultralytics 与先前几个版本相比,YOLOv8 模型更快、更准确,同时为训练模型提供统一框架,以执行以下基本任务: 目标检测;实例分割;图…

奇安信20240513笔试

题目一 解题思路 n转为字符串,如果位数为偶数,取前一半设为x,后一段为y,从x最低位开始,9,9*10,9*10*10。。。 到最高位,加x,如果x大于或等于y,加1. 位数为奇数…

武汉免费 【FPGA实战训练】 Vivado入门与设计师资课程

一.背景介绍 当今高度数字化和智能化的工业领域,对高效、灵活且可靠的技术解决方案的需求日益迫切。随着工业 4.0 时代的到来,工业生产过程正经历着前所未有的变革,从传统的机械化、自动化逐步迈向智能化和信息化。在这一背景下&…

全志A527 T527 cat /proc/cupinfo没有Serial问题

1.前言 我们有些客户是使用cpuinfo节点去获取系统的cpuid的,如下: cat /proc/cupinfo processor : 0 BogoMIPS : 48.00 Features : fp asimd evtstrm aes pmull sha1 sha2 crc32 atomics fphp asimdhp cpuid asimdrdm lrcpc dcpop asimddp CPU impleme…

代码随想录-Day51

115. 不同的子序列 给你两个字符串 s 和 t ,统计并返回在 s 的 子序列 中 t 出现的个数,结果需要对 109 7 取模。 示例 1: 输入:s “rabbbit”, t “rabbit” 输出:3 解释: 如下所示, 有 3 种可以从 …

基于 LlamaIndex、Claude-3.5 Sonnet 和 MongoDB,构建具有超级检索能力的智能体

节前,我们组织了一场算法岗技术&面试讨论会,邀请了一些互联网大厂朋友、今年参加社招和校招面试的同学。 针对大模型技术趋势、算法项目落地经验分享、新手如何入门算法岗、该如何准备面试攻略、面试常考点等热门话题进行了深入的讨论。 总结链接如…

Day1--每日一练

🍁 个人主页:爱编程的Tom💫 本篇博文收录专栏:每日一练-算法篇👉 目前其它专栏:c系列小游戏 c语言系列--万物的开始_ Java专栏等 🎉 欢迎 👍点赞✍评论⭐收藏&…

Java之父James Gosling宣布正式退休 创造无数人的饭碗

编程语言Java的创始人,被誉为“Java之父”的James Gosling,近日在社交媒体上宣布了自己正式退休的消息。Gosling表示:“我终于退休了。做了这么多年的软件工程师,现在是时候享受人生了。”他透露,在亚马逊的过去7年是非…

代码随想录算法训练营第四十七天|1143.最长公共子序列、 1035.不相交的线、53. 最大子序和、392.判断子序列

1143.最长公共子序列 题目链接:1143.最长公共子序列 文档讲解:代码随想录 状态:一开始没想明白为啥要 max(dp[i - 1][j], dp[i][j - 1]) 思路: 如果text1[i - 1] 与 text2[j - 1]相同,那么找到了一个公共元素&#xff…

GitLab介绍,以及add an SSH key

GitLab GitLab 是一个用于仓库管理系统的开源项目,现今并在国内外大中型互联网公司广泛使用。 git,gitlab,github区别 git 是一种基于命令的版本控制系统,全命令操作,没有可视化界面; gitlab 是一个基于git实现的在线代码仓库…

K8s驱逐场景以及规避方案参考 —— 筑梦之路

Pod 驱逐分为两种情况: 较安全驱逐 & 提高稳定性的良性驱逐 API 发起驱逐,典型案例:kubectl drain Node Not Ready 时,Controller Manager 发起的驱逐 有风险的驱逐 节点压力驱逐 节点磁盘空间不足、内存不足 或 Pid 不足&…

简易Qt串口助手

界面显示如下 关于串口类 初始化 设置串口号 设置波特率 打开串口 发送按钮功能实现 接收数据显示在控件中 关闭串口

Vortex GPGPU的硬件设计和代码结构分析

文章目录 前言一、GPGPU是什么?1.1 GPU和GPGPU之间的差异1.2 GPU和CPU之间的集成方式1.3 GPU包含什么(列举和VMIPS向量体系结构的差异) 二、Vortex GPGPU是什么?2.1 Vortex GPGPU的技术边界和验证环境2.2 Vortex GPGPU的指令集设计…

30万的剧本杀店 被“好色”店长玩死了

文|琥珀食酒社 作者 | 朱珀 对开店搞钱的人来讲 什么才是最苦逼的? 不是一开始生意就不行 而是刚开始好到不行 最后只剩下不行 本期投稿的主人公糊糊 就是这样的 苦逼大BOSS 30万开剧本杀店 短短几个月 从巅峰跌到谷底 被捞钱又好色的猪队友…

C++ 类和对象 拷贝构造函数

一 拷贝构造函数的概念: 拷贝构造函数是一种特殊的构造函数,用于创建一个对象是另一个对象的副本。当需要用一个已存在的对象来初始化一个新对象时,或者将对象传递给函数或从函数返回对象时,会调用拷贝构造函数。 二 拷贝构造函…

LabVIEW高能质子束流密度分布测试系统

LabVIEW平台开发的高能质子束流密度分布测试系统。该系统主要应用于电子器件的抗辐射加固试验,旨在精确测量高能质子束的密度分布,以评估电子器件在辐射环境下的性能表现和耐受能力。 系统组成与设计 硬件组成: 法拉第杯探测器:…

自动化测试高级控件交互方法:TouchAction、触屏操作、点按,双击,滑动,手势解锁!

在自动化测试领域中,TouchAction 是一种非常强大的工具,它允许我们模拟用户在设备屏幕上的各种触摸事件。这种模拟不仅限于简单的点击操作,还包括滑动、长按、多点触控等复杂的手势。 点按与双击 点按和双击是触屏设备上最基本的操作之一。…

数据库图形化管理界面应用 Navicat Premium 使用教程

经同学介绍的一个把数据库可视化的软件Navicat Premium,很好用,在这里分享一下,需要的同学可以去了解看看 一:下载并解压 链接:https://pan.baidu.com/s/1ZcDH6m7EAurAp_QmXWx81A 提取码:e5f6 解压到合…

景芯SoC训练营DFT debug

景芯训练营VIP学员在实践课上遇到个DFT C1 violation,导致check_design_rule无法通过,具体报错如下: 遇到这个问题第一反映一定是确认时钟,于是小编让学员去排查add_clock是否指定了时钟,指定的时钟位置是否正确。 景芯…

Redis原理-数据结构

Redis原理篇 1、原理篇-Redis数据结构 1.1 Redis数据结构-动态字符串 我们都知道Redis中保存的Key是字符串,value往往是字符串或者字符串的集合。可见字符串是Redis中最常用的一种数据结构。 不过Redis没有直接使用C语言中的字符串,因为C语言字符串存…