物联网|按键实验---学习I/O的输入及中断的编程|函数说明的格式|如何使用CMSIS的延时|读取通过外部中断实现按键捕获代码的实现及分析-学习笔记(14)

文章目录

  • 通过外部中断实现按键捕获代码的实现及分析
  • Tip1:函数说明的格式
  • Tip2:如何使用CMSIS的延时
    • GetTick函数原型
    • stm32f407_intr_handle.c解析
      • 中断处理函数:void EXTI4_IRQHandler
    • 调试流程
    • 软件模拟调试
  • 两种代码的比较
  • 课后作业:

通过外部中断实现按键捕获代码的实现及分析

1 代码的流程分析
2 代码的实现
库函数HAL_Init(void)分析:

HAL_StatusTypeDef HAL_Init(void)
{/* Configure Flash prefetch, Instruction cache, Data cache */
#if (INSTRUCTION_CACHE_ENABLE != 0U) //0U表示无符号整型 0 , 1U 表示无符号整型1 ~0U就是对无符号数0取反。__HAL_FLASH_INSTRUCTION_CACHE_ENABLE();
#endif /* INSTRUCTION_CACHE_ENABLE  允许指令缓存*/#if (DATA_CACHE_ENABLE != 0U)__HAL_FLASH_DATA_CACHE_ENABLE();
#endif /* DATA_CACHE_ENABLE */#if (PREFETCH_ENABLE != 0U)__HAL_FLASH_PREFETCH_BUFFER_ENABLE();
#endif /* PREFETCH_ENABLE *//* Set Interrupt Group Priority 中断优先级分组*/HAL_NVIC_SetPriorityGrouping(NVIC_PRIORITYGROUP_4);/* Use systick as time base source and configure 1ms tick (default clock after Reset is HSI) */HAL_InitTick(TICK_INT_PRIORITY); //初始化系统时钟/* Init the low level hardware 初始化底层硬件(堆栈指针)*/HAL_MspInit();  //使用HAL_Delay延时/* Return function status */return HAL_OK;
}

Tip1:函数说明的格式

/****************
*函数名:main
*函数的描述:通过中断实现按键的捕获
*输入参教:
*输出参数:
*返回值:
*图数作者:
*创建时间:
*更改说明:
*****************/

Tip2:如何使用CMSIS的延时

HAL_Delay()系统延时的步骤:
它的实现步骤如下:
1.用变量获得系统时钟源计数器的值
2.获得要延迟时间的参数值
3.比较两者大小,若时钟计数器的值大于要实现延迟的值,就会困在循环里;反之,跳出循环,延时完成。

/*** @brief此函数提供最小延迟(以毫秒为单位)对变量递增。* @note在默认实现中,SysTick计时器是基准时间的来源。*它用于在固定的时间间隔生成中断,其中uwTick是递增的。这个函数被声明为__weak,在其他情况下会被覆盖实现在用户文件。*@param Delay指定延迟时间长度,单位为毫秒。*@retval无*/
__weak void HAL_Delay(uint32_t Delay)
{uint32_t tickstart = HAL_GetTick(); //获得起始时钟uint32_t wait = Delay;/* Add a freq to guarantee minimum wait */if (wait < HAL_MAX_DELAY)   //#define HAL_MAX_DELAY      0xFFFFFFFFU=1111 1111 1111 1111 1111 1111 1111 1111{//  HAL_TICK_FREQ_1KHZ         = 1U,//  HAL_TICK_FREQ_DEFAULT      = HAL_TICK_FREQ_1KHZwait += (uint32_t)(uwTickFreq);  //作用是给wait加1。HAL_TickFreqTypeDef uwTickFreq = HAL_TICK_FREQ_DEFAULT;  /* 1KHz */}while((HAL_GetTick() - tickstart) < wait) //当前时钟-起始时钟的值小于wait(delay)就重复操作,直到计时结束{}
}

GetTick函数原型

/**调用这个函数是为了增加一个全局变量“uwTick”用作申请时基。
在默认实现中,这个变量每1ms增加一次在SysTick ISR。
这个函数被声明为__weak,在其他情况下会被覆盖实现在用户文件。
* @retval无*/
__weak void HAL_IncTick(void)
{uwTick += uwTickFreq;
}/**
* @brief提供以毫秒为单位的tick值。
这个函数被声明为__weak,在其他情况下会被覆盖实现在用户文件。
@retval tick value*/
__weak uint32_t HAL_GetTick(void)
{return uwTick;
}

对于裸机程度,没有其他的隐蔽的东西,自己好好分析还是可以理清程序运行的细节及时序关系。遵守的原则:
1、尽量不要在主程序中使用死等的延时,二、每个子程序(也可以叫任务吧)的查询频率要大于主程序运行的时间。比如:ad采样,100ms采样一次,那么,主程序一定要在100ms以内执行完毕。
死等的延时可以是us级别的,时序性较高的地方,大的延时就使用定时器。

stm32f407_intr_handle.c解析

中断处理函数:void EXTI4_IRQHandler

根据startup_stm32f407xx.s中对于handler的描述,编写相应的断点处理函数:

__Vectors       DCD     __initial_sp               ; Top of StackDCD     Reset_Handler              ; Reset HandlerDCD     NMI_Handler                ; NMI HandlerDCD     HardFault_Handler          ; Hard Fault HandlerDCD     MemManage_Handler          ; MPU Fault HandlerDCD     BusFault_Handler           ; Bus Fault HandlerDCD     UsageFault_Handler         ; Usage Fault HandlerDCD     0                          ; ReservedDCD     0                          ; ReservedDCD     0                          ; ReservedDCD     0                          ; ReservedDCD     SVC_Handler                ; SVCall HandlerDCD     DebugMon_Handler           ; Debug Monitor HandlerDCD     0                          ; ReservedDCD     PendSV_Handler             ; PendSV HandlerDCD     SysTick_Handler            ; SysTick Handler

执行函数:HAL_GPIO_EXTI_IRQHandler(KEY0_PIN),调用stm32f4xx_hal_gpio.c中的void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin),函数的定义为:

void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{/* EXTI line interrupt detected *///#define __HAL_GPIO_EXTI_GET_IT(__EXTI_LINE__) (EXTI->PR & (__EXTI_LINE__))//__EXTI_LINE__ specifies the EXTI line flag to check.if(__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != RESET) // RESET = 0U(stm32f4xx.h),表明检测到了中断{/*** @brief  Clears the EXTI's line pending bits.* @param  __EXTI_LINE__ specifies the EXTI lines to clear.*          This parameter can be any combination of GPIO_PIN_x where x can be (0..15)* @retval None*/__HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin);HAL_GPIO_EXTI_Callback(GPIO_Pin);}
}

回调函数在HAL_GPIO_EXTI_Callback在HAL中未明确实现(弱声明),需要自己构造实现函数。
原型为:

/*** @brief  EXTI line detection callbacks.* @param  GPIO_Pin Specifies the pins connected EXTI line* @retval None*/
__weak void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{/* Prevent unused argument(s) compilation warning */UNUSED(GPIO_Pin);/* NOTE: This function Should not be modified, when the callback is needed,the HAL_GPIO_EXTI_Callback could be implemented in the user file*/
}

在key.c中对回调函数进行了重写:

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{if(GPIO_Pin == KEY0_PIN) // KEY0_PIN被按下{Led_Ctrl(LED0_PIN_ROW, LED0_PIN, LED_ON); //执行点灯操作}
}

本节中uint16_t Detect_key(uint16_t key_pin)未使用。

调试流程

在中断处理函数和uwTick自加操作函数处设立中断进行调试:

void EXTI4_IRQHandler(void)
{HAL_GPIO_EXTI_IRQHandler(KEY0_PIN);
}void SysTick_Handler(void)
{HAL_IncTick(); //uwTick自加操作 uwTick += uwTickFreq;}

在这里插入图片描述
在这里插入图片描述

软件模拟调试

采用软件模拟方式,进入调试,停在了main函数底下的LED0_Init()函数:

运行出现错误:*** error 65: access violation at 0x40023830 : no ‘write’ permission
由于没有指定相应的cpu:
解决方法:

  • 1 新建cpu.ini,写入:
map 0x40000000,0x400FFFFF read write
  • 2 将cpu.ini添加入debug:
    在这里插入图片描述

两种代码的比较

中断模式对CPu占用率更小,等待时间更短,减小CPU占有率

课后作业:

1:预习缤程手册关于USART控制器的描述(第30章)
2:通过中断的方式实现一个程序,要求按下KEY1以后:让LED1间隔0.5S闪烁,并持续5S后关闭,实现代码并在板调试

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MGRE综合

实验 一、实验思路 1.先按照上图配置IP地址及环回 2.写缺省使公网可通 3.让R1、R4、R5每台路由器均成为中心站点形成全连网状结构拓扑 4.让R1成为中心站点R2R3为分支站点 5.分区域宣告ospf之后更改ospf在虚拟接口Tunnel工作方式为broadcast及让R1 当选DR 二、上虚拟机操作…

小白玩转浏览器开发者工具—F12(超详细)

目录 1、检查元素&#xff1a;&#x1f680; 2、修改样式&#xff1a;&#x1f6eb; 3、调试代码&#xff1a;&#x1f451; 4、网络分析&#xff1a;&#x1f682; 5、控制台输出&#xff1a;&#x1f681; 6、移动设备模拟&#xff1a;&#x1f3a8; 7、缓存管理&…

强大的截图软件--Snipaste

这里写目录标题 前言Snipaste贴图并置顶标注功能 下载 前言 在工作中&#xff0c;我们经常需要保存当前屏幕的图片&#xff0c;虽然系统总是会自带一些截图工具&#xff0c;但似乎用起来总是不那个顺手&#xff0c;例如我们需要对图片进行一些标注&#xff0c;或者将图片贴在屏…

浏览器自动访问打开网址的软件小工具模拟测试

用微软框架写了个浏览器自动访问和打开网址的工具&#xff0c;进行测试模拟&#xff1a; 1、获取链接方式&#xff0c;可通过API接口返回JSON链接格式&#xff0c;也可以集成到文档手动录入链接由软件进行循环运行。 2、配置一些参数&#xff1a;数量、次数、时间间隔等 看下演…

《向量数据库指南》——使用 AI原生云向量数据库Milvus Cloud的好处

目录 3. 使用 Milvus Cloud的好处 a. 高效存储和检索 b. 高度灵活、可扩展 c. 高性能、高可用 d. 易用性 e. 可靠性 3. 使用 Milvus Cloud的好处 Milvus Cloud生态系统提供了有效的数据库监控、数据迁移和数据量估算工具。如果不想要花时间和精力维护 Milvus,也可以选择…

【网络基础进阶之路】设计网络划分的实战详解

PS&#xff1a;本要求基于华为的eNSP模拟软件进行 具体要求&#xff1a; 完成步骤&#xff1a; 1、对192.168.1.0/24进行子网划分 2、对每一个路由器进行IP的配置 3、开始静态路由的书写&#xff0c;在写之前&#xff0c;我们可以先对每一个路由器写一条通向右边的缺省路由&…

vue videojs视频播放插件 动态资源

概览&#xff1a;使用vediojs第三方视频播放插件&#xff0c;视频加载有两种方式&#xff0c;第一种是html方式&#xff0c;类似于img标签的src&#xff0c;写在video标签内的<source src>&#xff1b;第二种方式是js方式&#xff0c;实例化一个videojs对象并且赋值src。…

Nginx 15分钟入门

1、反向代理和负载均衡 Nginx 反向代理 负载均衡 如上图&#xff0c;网站初期用户量较少的时候&#xff0c;一台服务器就够用&#xff0c;但是当大量用户注册&#xff0c;那么显然一台机器就不够了&#xff0c;我们把同一个项目部署在3台服务器上&#xff0c;如下图所示。那…

Nginx 高性能内存池 ----【学习笔记】

跟着这篇文章学习&#xff1a; c代码实现一个高性能内存池&#xff08;超详细版本&#xff09;_c 内存池库_linux大本营的博客-CSDN博客https://blog.csdn.net/qq_40989769/article/details/130874660以及这个视频学习&#xff1a; nginx的内存池_哔哩哔哩_bilibilihttps://w…

安科瑞智能仪表在户用光伏,工商业光伏,地面光伏,光电建筑的应用

光伏储能逆变器 应用场景 户用储能&#xff0c;小型工商业储能&#xff0c;微电网储能 功能 1.对电能参数进行采样计量和监测&#xff0c;逆变器或者能量管理系统&#xff08;EMS&#xff09;与之进行通讯&#xff0c;根据实时功率及累计电能实现防逆流、调节发电量、电池充…

项目一:基于stm32的阿里云智慧消防监控系统

若该文为原创文章&#xff0c;转载请注明原文出处。 Hi&#xff0c;大家好&#xff0c;我是忆枫&#xff0c;今天向大家介绍一个单片机项目。 一、简介 智慧消防监控系统&#xff0c;是用于检测火灾&#xff0c;温度&#xff0c;烟雾的监控系统。以 stm32单片机为核心外加 MQ…

AcWing1171. 距离(lcatarjan)

输入样例1&#xff1a; 2 2 1 2 100 1 2 2 1输出样例1&#xff1a; 100 100输入样例2&#xff1a; 3 2 1 2 10 3 1 15 1 2 3 2输出样例2&#xff1a; 10 25 #include<bits/stdc.h> using namespace std; typedef long long ll; const int N2e55; int n,m,x,y,k,r…