【STM32RT-Thread零基础入门】 3. PIN设备(GPIO)的使用

硬件:STM32F103ZET6、ST-LINK、usb转串口工具、4个LED灯、1个蜂鸣器、4个1k电阻、2个按键、面包板、杜邦线

文章目录

  • 前言
  • 一、PIN设备介绍
    • 1. 引脚编号获取
    • 2. 设置引脚的输入/输出模式
    • 3. 设置引脚的电平值
    • 4. 读取引脚的电平值
    • 5. 绑定引脚中断回调函数
    • 6. 脱离引脚中断回调函数
    • 7. 使能中断
  • 二、任务1:LED灯双闪控制
    • 1. 任务描述
    • 2. 代码编写
  • 三、任务2:蜂鸣器控制(查询法)
    • 1. 任务描述
    • 2. 代码编写
    • 3. 测试
  • 四、任务3:蜂鸣器控制(中断回调法)
    • 1. 任务描述
    • 2. 代码编写
    • 3. 测试
  • 五、任务4:同时实现LED闪烁和按键控制喇叭
    • 1. 任务描述
    • 2. 代码编写
    • 3. 测试
  • 总结


前言

在嵌入式系统中,GPIO是最常用的一种设备,在RT-Thread操作系统中,把GPIO命名为PIN设备。


一、PIN设备介绍

RT-Thread通过PIN设备对芯片的GPIO引脚进行管理,应用程序可以通过其提供的一组PIN设备管理接口来操作GPIO,PIN设备管理接口如表所示

接口描述
rt_pin_get()获取引脚编号
rt_pin_mode()设置引脚模式
rt_pin_write()设置引脚电平
rt_pin_read()读取引脚电平
rt_pin_attach_irq()绑定引脚中断回调函数
rt_pin_irq_enable()使能引脚中断
rt_pin_detach_irq()脱离引脚中断回调函数

1. 引脚编号获取

引脚变化和芯片的引脚号不是同一概念。RT-Thread的PIN设备驱动程序把芯片的不同引脚赋予不同的编号,操作PIN设备时,需要使用引脚编号来指定对哪个引脚进行操作。
可通过三种方法获得引脚编号:
(1)API法
利用rt_pin_get() 函数来获取引脚编号,例如获取PD9的引脚编号,则可以使用
pin_number = rt_pin_get("PD.9")
(2)宏定义法
对于STM32芯片,可以使用GET_PIN(PORTx,PIN)获取引脚编号,如硬件的PD9用于驱动LED_R_PIN,则可以把宏LED_R_PIN定义为相应的引脚编号,如下:
#define LED_R_PIN GET_PIN(D, 9)
(3)查看驱动文件
在drivers/drv_gpio.c中,pin[]数组定义了硬件平台的GPIO引脚编号,通过查看该数组,得到引脚PD9的编号为57。
具体数组代码如下:

static const struct pin_index pins[] = 
{
#if defined(GPIOA)__STM32_PIN(0 ,  A, 0 ),__STM32_PIN(1 ,  A, 1 ),__STM32_PIN(2 ,  A, 2 ),__STM32_PIN(3 ,  A, 3 ),__STM32_PIN(4 ,  A, 4 ),__STM32_PIN(5 ,  A, 5 ),__STM32_PIN(6 ,  A, 6 ),__STM32_PIN(7 ,  A, 7 ),__STM32_PIN(8 ,  A, 8 ),__STM32_PIN(9 ,  A, 9 ),__STM32_PIN(10,  A, 10),__STM32_PIN(11,  A, 11),__STM32_PIN(12,  A, 12),__STM32_PIN(13,  A, 13),__STM32_PIN(14,  A, 14),__STM32_PIN(15,  A, 15),
#if defined(GPIOB)__STM32_PIN(16,  B, 0),__STM32_PIN(17,  B, 1),__STM32_PIN(18,  B, 2),__STM32_PIN(19,  B, 3),__STM32_PIN(20,  B, 4),__STM32_PIN(21,  B, 5),__STM32_PIN(22,  B, 6),__STM32_PIN(23,  B, 7),__STM32_PIN(24,  B, 8),__STM32_PIN(25,  B, 9),__STM32_PIN(26,  B, 10),__STM32_PIN(27,  B, 11),__STM32_PIN(28,  B, 12),__STM32_PIN(29,  B, 13),__STM32_PIN(30,  B, 14),__STM32_PIN(31,  B, 15),
#if defined(GPIOC)__STM32_PIN(32,  C, 0),__STM32_PIN(33,  C, 1),__STM32_PIN(34,  C, 2),__STM32_PIN(35,  C, 3),__STM32_PIN(36,  C, 4),__STM32_PIN(37,  C, 5),__STM32_PIN(38,  C, 6),__STM32_PIN(39,  C, 7),__STM32_PIN(40,  C, 8),__STM32_PIN(41,  C, 9),__STM32_PIN(42,  C, 10),__STM32_PIN(43,  C, 11),__STM32_PIN(44,  C, 12),__STM32_PIN(45,  C, 13),__STM32_PIN(46,  C, 14),__STM32_PIN(47,  C, 15),
#if defined(GPIOD)__STM32_PIN(48,  D, 0),__STM32_PIN(49,  D, 1),__STM32_PIN(50,  D, 2),__STM32_PIN(51,  D, 3),__STM32_PIN(52,  D, 4),__STM32_PIN(53,  D, 5),__STM32_PIN(54,  D, 6),__STM32_PIN(55,  D, 7),__STM32_PIN(56,  D, 8),__STM32_PIN(57,  D, 9),__STM32_PIN(58,  D, 10),__STM32_PIN(59,  D, 11),__STM32_PIN(60,  D, 12),__STM32_PIN(61,  D, 13),__STM32_PIN(62,  D, 14),__STM32_PIN(63,  D, 15),

2. 设置引脚的输入/输出模式

宏定义描述
PIN_MODE_OUTPUT推挽输出
PIN_MODE_OUTPUT_OD开漏输出,硬件需要外接上拉电阻
PIN_MODE_INPUT输入
PIN_MODE_INPUT_PULLUP上拉输入,引脚悬空时为高电平
PIN_MODE_INPUT_PULLDOWN下拉输入,引脚悬空时为低电平
引脚可通过以下函数设置输入/输出模式:

void rt_pin_mode(rt_base_t pin, rt_base_t mode)

pin:引脚编号
mode:输入/输出模式所对应的宏

#define LED_R_PIN GET_PIN(D, 9)
rt_pin_mode(LED_R_PIN, PIN_MODE_OUTPUT);

3. 设置引脚的电平值

设置函数如下:
void rt_pin_write(rt_base_t pin, rt_base_t value)
pin:引脚编号
value:高电平为PIN_HIGH;低电平为PIN_LOW

4. 读取引脚的电平值

设置函数如下:
int rt_pin_read(rt_base_t pin)
pin:引脚编号
返回值:高电平为PIN_HIGH;低电平为PIN_LOW

5. 绑定引脚中断回调函数

引脚作为中断输入时,需要设置引脚的中断触发方式,引脚的中断触发模式有5种,RT-Thread中分别有5个宏与之对应,具体如下:

宏定义描述
PIN_IQR_MODE_RISING上升沿触发
PIN_IQR_MODE_FALLING下降沿触发
PIN_IQR_MODE_RISING_FALLING双边沿触发
PIN_IQR_MODE_HIGH_LEVEL高电平触发
PIN_IQR_MODE_LOW_LEVEL低电平触发

当引脚产生中断时,通过中断回调函数响应中断。绑定中断回调函数到引脚后,引脚有中断发生,就会执行对应的中断回调函数。

具体函数如下:
rt_err_t rt_pin_attach_irq(rt_int32_t pin, rt_uint32_t mode, void (*hdr)(void *args), void *args)
pin:引脚编号
mode:中断触发模式
hdr:中断回调函数指针,用户需要自定义这个函数
args:中断回调函数的参数,不需要设置时为RT_NULL
返回:绑定成功为RT_EOK;失败产生错误码

6. 脱离引脚中断回调函数

如不希望响应中断,或者想更换中断响应函数,则可以使用如下脱离引脚中断回调函数:
rt_err_t rt_pin_detach_irq(rt_int32_t pin)
pin:引脚编号
返回:脱离成功为RT_EOK;失败产生错误码

7. 使能中断

rt_err_t rt_pin_irq_enable(rt_base_t pin, rt_uint32_t enabled)
pin:引脚编号
enable:PIN_IRQ_ENABLE(开启),PIN_IRQ_DISABLE(关闭)
返回:使能成功为RT_EOK;失败产生错误码

二、任务1:LED灯双闪控制

1. 任务描述

控制LED1、LED2轮流闪烁。通过本任务,学习PIN设备输出功能的设置方法。

在这里插入图片描述

2. 代码编写

在项目的main.c文件中,编写如下代码:

#include <rtthread.h>
#include <rtdevice.h>
#include "drv_common.h"/* 定义左右转向灯的控制引脚 */
#define LED_L_PIN GET_PIN(D, 8)
#define LED_R_PIN GET_PIN(D, 9)int main(void)
{/* 把引脚设置为推拉输出模式 */rt_pin_mode(LED_L_PIN, PIN_MODE_OUTPUT);rt_pin_mode(LED_R_PIN, PIN_MODE_OUTPUT);while (1){rt_pin_write(LED_L_PIN, PIN_HIGH);//灭左灯rt_pin_write(LED_R_PIN, PIN_LOW);//亮右灯rt_thread_mdelay(500);//延迟500毫秒rt_pin_write(LED_L_PIN, PIN_LOW);//亮左灯rt_pin_write(LED_R_PIN, PIN_HIGH);//灭右灯rt_thread_mdelay(500);}return RT_EOK;
}

三、任务2:蜂鸣器控制(查询法)

1. 任务描述

通过按键控制蜂鸣器的开关。当按键按下时,蜂鸣器响起;当按键松开时,蜂鸣器关闭。通过本任务,学习PIN设备输入功能的使用。
在这里插入图片描述
在这里插入图片描述

2. 代码编写

#include <rtthread.h>
#include <rtdevice.h>
#include "drv_common.h"/* 定义左右转向灯的控制引脚 */
#define BEEP_PIN GET_PIN(A, 5)
#define KEY1_PIN GET_PIN(A, 0)int main(void)
{/* 把蜂鸣器引脚设置为推拉模式 */rt_pin_mode(BEEP_PIN, PIN_MODE_OUTPUT);/* 把按键引脚设置为上拉输入模式 */rt_pin_mode(KEY1_PIN, PIN_MODE_INPUT_PULLUP);while (1){if(PIN_LOW==rt_pin_read(KEY1_PIN)){//按键按下rt_thread_mdelay(20);//延时去抖if(PIN_LOW==rt_pin_read(KEY1_PIN))rt_pin_write(BEEP_PIN, PIN_LOW);//蜂鸣器响}elsert_pin_write(BEEP_PIN, PIN_HIGH);//否则,蜂鸣器不响rt_thread_mdelay(300);//每0.3秒进行一次按键扫描}return RT_EOK;
}

3. 测试

按下按键,蜂鸣器发声,松开按键,停止发声,但是经过多次尝试发现,按键检测成功率较低,所以需要利用中断进行改进。

四、任务3:蜂鸣器控制(中断回调法)

1. 任务描述

通过按键控制蜂鸣器的开关。当按键按下时,蜂鸣器响起;当按键松开时,蜂鸣器关闭。通过本任务,掌握PIN设备中断回调函数的使用方法。

2. 代码编写

#include <rtthread.h>
#include <rtdevice.h>
#include "drv_common.h"#define BEEP_PIN GET_PIN(A, 5)  //定义蜂鸣器的控制引脚
#define KEY1_PIN GET_PIN(A, 0)  //定义按键的控制引脚/* 定义中断回调函数实现 */
void beep_on(void *args)
{/* 判断按键是否按下 */if(PIN_LOW==rt_pin_read(KEY1_PIN)){/* 按键按下,驱动蜂鸣器响  */rt_pin_write(BEEP_PIN, PIN_LOW);/* 等待按键台起  */while(PIN_LOW==rt_pin_read(KEY1_PIN));/* 关闭蜂鸣器  */rt_pin_write(BEEP_PIN, PIN_HIGH);}
}
int main(void)
{rt_pin_mode(BEEP_PIN, PIN_MODE_OUTPUT); //把蜂鸣器引脚设置为推拉模式rt_pin_write(BEEP_PIN, PIN_HIGH); //初始化蜂鸣器默认状态为不响/* 把按键引脚设置为上拉输入模式 */rt_pin_mode(KEY1_PIN, PIN_MODE_INPUT_PULLUP);/* 绑定中断,下降沿触发模式,回调函数名为beep_on */rt_pin_attach_irq(KEY1_PIN, PIN_IRQ_MODE_FALLING, beep_on, RT_NULL);rt_pin_irq_enable(KEY1_PIN, PIN_IRQ_ENABLE); //使能中断return RT_EOK;
}

3. 测试

按下按键,蜂鸣器发声,松开按键,停止发声,经过多次尝试发现,按键检测成功率较高,所以中断法比查询法更具优势。

五、任务4:同时实现LED闪烁和按键控制喇叭

1. 任务描述

本任务功能为同时实现LED灯双闪功能和按键控制蜂鸣器的功能,要求两个功能不能相互影响,按键检测灵敏度要高,即每次发生按键按下的事件,程序都能成功检测该事件并开启蜂鸣器。

2. 代码编写

#include <rtthread.h>
#include <rtdevice.h>
#include "drv_common.h"#define DBG_TAG "main"
#define DBG_LVL DBG_LOG
#include <rtdbg.h>/* 定义左右转向灯的控制引脚 */
#define LED_L_PIN GET_PIN(D, 8)
#define LED_R_PIN GET_PIN(D, 9)
/* 定义蜂鸣器的控制引脚 */
#define BEEP_PIN GET_PIN(A, 5)
/* 定义按键的控制引脚 */
#define KEY1_PIN GET_PIN(A, 0)/* 定义中断回调函数 */
void beep_on(void *args)
{/* 判断按键是否按下 */if(PIN_LOW==rt_pin_read(KEY1_PIN)){rt_pin_write(BEEP_PIN, PIN_LOW); //按按键下,驱动蜂鸣器响 while(PIN_LOW==rt_pin_read(KEY1_PIN)); //等待按键台起rt_pin_write(BEEP_PIN, PIN_HIGH); //关闭蜂鸣器}
}int main(void)
{/* 把LED灯引脚设置为输出模式 */rt_pin_mode(LED_L_PIN, PIN_MODE_OUTPUT);rt_pin_mode(LED_R_PIN, PIN_MODE_OUTPUT);/* 把蜂鸣器引脚设置为推拉模式 */rt_pin_mode(BEEP_PIN, PIN_MODE_OUTPUT);/* 初始化蜂鸣器默认状态为不响 */rt_pin_write(BEEP_PIN, PIN_HIGH);/* 把按键引脚设置为上拉输入模式 */rt_pin_mode(KEY1_PIN, PIN_MODE_INPUT_PULLUP);/* 绑定中断,下降沿触发模式,回调函数名为beep_on */rt_pin_attach_irq(KEY1_PIN, PIN_IRQ_MODE_FALLING, beep_on, RT_NULL);rt_pin_irq_enable(KEY1_PIN, PIN_IRQ_ENABLE); // 使能中断 while(1){rt_pin_write(LED_L_PIN, PIN_HIGH);//亮左灯rt_pin_write(LED_R_PIN, PIN_LOW);//灭右灯rt_thread_mdelay(500);//延迟500毫秒rt_pin_write(LED_L_PIN, PIN_LOW);//灭左灯rt_pin_write(LED_R_PIN, PIN_HIGH);//亮右灯rt_thread_mdelay(500);}return RT_EOK;
}

3. 测试

测试过程如下:
(1)系统启动后,观察左右转向灯是否轮流闪烁;
(2)当按下按键时,喇叭是否发出响声;
(3)当松开按键时,喇叭是否停止发出响声;
(4)一直按住按键不松开,观察灯的闪烁情况。
测试结果:
(1)系统启动后,左右转向灯轮流闪烁;
(2)当按下按键时,喇叭发出响声;
(3)当松开按键时,喇叭停止发出响声;
(4)一直按住按键不松开,喇叭发出响声,灯停止闪烁。


总结

从任务4测试结果中,我们可以发现,按键功能影响了闪灯的功能,说明两个功能还是没有很好地解耦,依然存在相互影响的情况。
出现这种情况,主要是由于中断回调函数中存在需要长时间等待的代码,当按键一直按住不松开的时候,中断回调函数由于一直停留在等待按键松开的地方而无法退出中断处理。而中断的优先级又高于main()线程的优先级,从而导致main()线程无法得到执行。
通常,我们不应该在中断回调函数中进行长时间的处理,中断回调函数应该只做一些必要的快速处理操作,而把长时间的处理操作放到线程中实现。
关于线程和优先级的概念,我们在下一节讲述。

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

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

相关文章

Linux如何开启指定端口号

本文已收录于专栏 《运维》 目录 概念说明防火墙端口号 提供服务具体分类具体操作防火墙操作端口号操作 总结提升 概念说明 防火墙 防火墙是一种网络安全设备或软件&#xff0c;用于监控和控制网络流量&#xff0c;保护网络免受恶意攻击和未经授权的访问。防火墙可以根据预定义…

设计模式之简单工厂模式

一、概述 定义一个用于创建对象的接口&#xff0c;让子类决定实例化哪一个类。FactoryMethod使一个类的实例化延迟到其子类。 简单工厂模式&#xff1a;又叫做静态工厂方法模式&#xff0c;是由一个工厂对象决定创建出哪一种产品类的实例。 二、适用性 1.当一个类不知道它所…

wireshark入门指北

文章目录 前言安装Linux上wireshark安装 使用捕获的时候添加过滤条件抓取浏览器https内容 附录抓取非浏览器的https流量 前言 本文长期维护&#xff0c;记录使用wireshark的使用过程。 虽然有官方文档-Wireshark User’s Guide&#xff0c;但是不想去慢慢读。应用层的图形软件…

解决android studio妙明奇妙出现的模拟器

1&#xff0c;查看设备 adb devices 2&#xff0c; adb命令断开指定的设备 要断开ADB与特定设备的连接&#xff0c;可以使用以下命令&#xff1a; adb disconnect <设备ID> 将 <设备ID> 替换为您要断开连接的设备的实际ID。设备ID可以在运行 adb devices 命令…

项目中使用git vscode GitHubDesktopSetup-x64

一、使用git bash 1.使用git bash拉取gitee项目 1.在本地新建一个文件夹&#xff08;这个文件夹是用来存放从gitee上拉下来的项目的&#xff09; 2.在这个文件夹右键选择 git bash here 3.输入命令 git init (创建/初始化一个新的仓库) 4.输入命令 git remote add origin …

伪类和伪元素有何区别?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 伪类&#xff08;Pseudo-class&#xff09;⭐ 伪元素&#xff08;Pseudo-element&#xff09;⭐ 区别总结⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启航前…

【Linux】网络基础2

文章目录 网络基础21. 应用层1.1 协议1.2 HTTP 协议1.2.1 URL1.2.2 urlencode和urldecode1.2.3 HTTP协议格式1.2.4 HTTP的方法1.2.5 HTTP的状态码1.2.6 HTTP 常见的header1.2.7 最简单的HTTP服务器 2. 传输层2.1 端口号2.1.1 端口号范围划分2.1.2 认识知名端口号2.1.3 netstat2…

工作中的方法论总结

1、SMART SMART原则是目标管理的一种方法&#xff0c;通过有效地进行成员的组织与目标的制定和控制以达到更好的工作绩效。大到业务规划&#xff0c;小到个人项目开发计划都比较适用。 SMART的具体含义如下&#xff1a; S&#xff08;Specific&#xff09;&#xff1a;目标是确…

目标检测YOLOv3基于DarkNet53模型测试-笔记

目标检测YOLOv3基于DarkNet53模型测试-笔记 预测和试测结果&#xff1a; 预测代码如下所示&#xff1a; testInsects.py #YOLOv3网模型测试-单图片文件测试并显示测试结果 import time import os import paddle import numpy as np import cv2 import random from PIL impor…

ffmpeg+nginx实现rtsp协议摄像头web端播放

ffmpegnginx实现rtsp协议摄像头web端播放 环境准备准备nginx环境添加rtmp模块添加hls转发 使用ffmpeg&#xff0c;将摄像头rtsp转为rtmp并推送到nginxVLC播放验证 环境准备 nginx&#xff08;需要安装rtmp模块&#xff09;ffmpeg 6.0vlc播放器&#xff08;本地播放验证&#x…

Java内存模型

Java内存模型全称JMM&#xff08;Java Memory Model&#xff09; 内存主要有堆和栈组成 下面来一段demo代码详细讲解堆栈的作用&#xff0c;以及流程 public class Employee {private String name;private Integer age;private Department department;public Employee(){}pub…

【NLP】深入浅出全面回顾注意力机制

深入浅出全面回顾注意力机制 1. 注意力机制概述2. 举个例子&#xff1a;使用PyTorch带注意力机制的Encoder-Decoder模型3. Transformer架构回顾3.1 Transformer的顶层设计3.2 Encoder与Decoder的输入3.3 高并发长记忆的实现self-attention的矩阵计算形式多头注意力&#xff08;…