当前位置: 首页 > news >正文

第六章 GPIO输入——按键检测

单芯片解决方案,开启全新体验——W55MH32 高性能以太网单片机

W55MH32是WIZnet重磅推出的高性能以太网单片机,它为用户带来前所未有的集成化体验。这颗芯片将强大的组件集于一身,具体来说,一颗W55MH32内置高性能Arm® Cortex-M3核心,其主频最高可达216MHz;配备1024KB FLASH与96KB SRAM,满足存储与数据处理需求;集成TOE引擎,包含WIZnet全硬件TCP/IP协议栈、内置MAC以及PHY,拥有独立的32KB以太网收发缓存,可供8个独立硬件socket使用。如此配置,真正实现了All-in-One解决方案,为开发者提供极大便利。

在封装规格上,W55MH32 提供了两种选择:QFN100QFN68

W55MH32L采用QFN100封装版本,尺寸为12x12mm,其资源丰富,专为各种复杂工控场景设计。它拥有66个GPIO、3个ADC、12通道DMA、17个定时器、2个I2C、5个串口、2个SPI接口(其中1个带I2S接口复用)、1个CAN、1个USB2.0以及1个SDIO接口。如此丰富的外设资源,能够轻松应对工业控制中多样化的连接需求,无论是与各类传感器、执行器的通信,还是对复杂工业协议的支持,都能游刃有余,成为复杂工控领域的理想选择。 同系列还有QFN68封装的W55MH32Q版本,该版本体积更小,仅为8x8mm,成本低,适合集成度高的网关模组等场景,软件使用方法一致。更多信息和资料请进入http://www.w5500.com/网站或者私信获取。

此外,本W55MH32支持硬件加密算法单元,WIZnet还推出TOE+SSL应用,涵盖TCP SSL、HTTP SSL以及 MQTT SSL等,为网络通信安全再添保障。

为助力开发者快速上手与深入开发,基于W55MH32L这颗芯片,WIZnet精心打造了配套开发板。开发板集成WIZ-Link芯片,借助一根USB C口数据线,就能轻松实现调试、下载以及串口打印日志等功能。开发板将所有外设全部引出,拓展功能也大幅提升,便于开发者全面评估芯片性能。

若您想获取芯片和开发板的更多详细信息,包括产品特性、技术参数以及价格等,欢迎访问官方网页:http://www.w5500.com/,我们期待与您共同探索W55MH32的无限可能。

第六章 GPIO输入——按键检测

目录

第六章 GPIO输入——按键检测

1 硬件设计

2 软件设计

2.1 按键输入检测方式

2.1.1 按键扫描

2.1.2 外部中断

2.2 编程要点

2.3 代码分析

2.4 下载验证


本章参考资料:《W55MH32中文参考手册》按键检测使用到GPIO外设的基本输入功能。

1 硬件设计

按键机械触点断开、闭合时,由于触点的弹性作用,按键开关不会马上稳定接通或一下子断开, 使用按键时会产生图 按键抖动说明图 中的带波纹信号,需要用软件消抖处理滤波,不方便输入检测。本实验板连接的按键带硬件消抖功能, 见下图

它利用电容充放电的延时,消除了波纹,从而简化软件的处理,软件只需要直接检测引脚的电平即可。

从按键的原理图可知,这些按键在没有被按下的时候,GPIO引脚的输入状态为低电平(按键所在的电路不通,引脚接地),当按键按下时, GPIO引脚的输入状态为高电平(按键所在的电路导通,引脚接到电源)。只要我们检测引脚的输入电平,即可判断按键是否被按下。

若您使用的实验板按键的连接方式或引脚不一样,只需根据我们的工程修改引脚即可,程序的控制原理相同。

2 软件设计

2.1 按键输入检测方式

按键输入检测方式分为定时扫描和外部中断两种,工作原理和优缺点如下文所示。本篇我们主要讲解用外部中断的方式进行按键输入检测。

2.1.1 按键扫描

工作原理:通过 CPU 定时或不断地读取按键所连接的 GPIO 引脚电平状态,来判断按键是否被按下。比如在一个循环中,反复检测按键对应的引脚是高电平还是低电平 ,若原本是高电平,检测到变为低电平,且经过消抖处理(一般通过延时,比如 5~10ms 再次检测确认)后还是低电平,就认为按键被按下;松开时则相反。像独立按键扫描,就是单独检测每个按键引脚;矩阵按键扫描则是按行或列扫描,通过行列交叉点判断哪个按键动作 。

优点:

  • 原理简单易懂,实现起来相对容易,对于初学者友好。
  • 可灵活控制扫描频率和时机,在一些对按键检测实时性要求不高,且系统资源较为充裕的场景下,能很好地工作。

缺点:

  • 占用 CPU 资源,CPU 需要不断执行扫描程序来检测按键状态,若系统中还有其他任务,会影响整体运行效率。
  • 实时性较差,因为是定时或不断扫描,不能及时响应按键动作,存在检测延迟。

2.1.2 外部中断

工作原理:当外部事件发生,比如按键按下使连接到单片机的外部中断引脚上电平发生变化(可配置为上升沿触发、下降沿触发或双边沿触发 ),单片机的中断系统会迫使 CPU 暂停正在执行的程序,转而去执行预先编写好的中断服务程序来处理该事件,处理完后再返回原来中断的地方继续执行原程序 。以按键为例,按键连接到外部中断引脚,按下按键产生电平变化,触发中断请求,CPU 响应后进入中断服务函数处理按键事件。

优点:

  • 实时性强,能在外部事件发生的瞬间及时响应,快速处理事件,适用于对实时性要求高的场景,如工业控制中对突发信号的快速响应。
  • 节省 CPU 资源,CPU 不需要一直轮询检测按键状态,只有在外部事件触发中断时才会处理相关事务,提高了 CPU 使用效率。

缺点:

  • 配置相对复杂,需要对中断相关寄存器(如中断使能寄存器、中断优先级寄存器、中断触发方式寄存器等 )进行正确配置,对开发者要求较高。
  • 多个中断源同时请求时,需要合理处理中断优先级和中断嵌套等问题,否则可能导致系统运行异常。

2.2 编程要点

1.使能GPIO端口时钟;

2.初始化GPIO目标引脚为输入模式(浮空输入);

3.编写简单测试程序,检测按键的状态

2.3 代码分析

1. 头文件包含和宏定义

#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include "delay.h"
#include "w55mh32.h"#define GPIO_GROUP_TEST GPIOG
#define GPIO_MODE_TEST  GPIO_Mode_IPU
#define GPIO_SPEED_TEST GPIO_Speed_50MHz
#define GPIO_PIN_TEST   GPIO_Pin_5USART_TypeDef *USART_TEST = USART1;

头文件包含:包含标准库头文件 stdlib.h、string.h、stdio.h,以及自定义的 delay.h 和 w55mh32.h 头文件。

宏定义:

  • GPIO_GROUP_TEST:定义使用的 GPIO 组为 GPIOG。
  • GPIO_MODE_TEST:定义 GPIO 模式为上拉输入模式 GPIO_Mode_IPU。
  • GPIO_SPEED_TEST:定义 GPIO 速度为 50MHz。
  • GPIO_PIN_TEST:定义使用的 GPIO 引脚为 GPIO_Pin_5。
  • USART_TEST:定义使用的 USART 外设为 USART1。

2. 函数声明

void UART_Configuration(uint32_t bound);
void GPIO_Configuration(void);

声明了两个函数:UART_Configuration 用于配置串口通信,GPIO_Configuration 用于配置 GPIO 引脚和外部中断。

3. main 函数

int main(void)
{
    RCC_ClocksTypeDef clocks;
    delay_init();
    UART_Configuration(115200);
    RCC_GetClocksFreq(&clocks);    printf("\n");
    printf("SYSCLK: %3.1fMhz, HCLK: %3.1fMhz, PCLK1: %3.1fMhz, PCLK2: %3.1fMhz, ADCCLK: %3.1fMhz\n",
           (float)clocks.SYSCLK_Frequency / 1000000, (float)clocks.HCLK_Frequency / 1000000,
           (float)clocks.PCLK1_Frequency / 1000000, (float)clocks.PCLK2_Frequency / 1000000, (float)clocks.ADCCLK_Frequency / 1000000);
    printf("GPIO IO Input Test.\n");    GPIO_Configuration();    while (1)
    {
    }
}

初始化操作:

  • delay_init():初始化延时函数。
  • UART_Configuration(115200):配置串口通信,波特率为 115200。
  • RCC_GetClocksFreq(&clocks):获取系统时钟频率信息。

打印信息:打印系统时钟频率信息和测试提示信息。

GPIO 配置:调用 GPIO_Configuration() 函数配置 GPIO 引脚和外部中断。

主循环:进入一个无限循环,等待外部中断触发。

4. GPIO_Configuration 函数

void GPIO_Configuration(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;
    EXTI_InitTypeDef EXTI_InitStructure;    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOG | RCC_APB2Periph_AFIO, ENABLE);    GPIO_InitStructure.GPIO_Pin   = GPIO_PIN_TEST;
    GPIO_InitStructure.GPIO_Speed = GPIO_SPEED_TEST;
    GPIO_InitStructure.GPIO_Mode  = GPIO_MODE_TEST;
    GPIO_Init(GPIO_GROUP_TEST, &GPIO_InitStructure);    GPIO_EXTILineConfig(GPIO_PortSourceGPIOG, GPIO_PinSource5);    NVIC_InitStructure.NVIC_IRQChannel                   = EXTI9_5_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority        = 3;
    NVIC_InitStructure.NVIC_IRQChannelCmd                = ENABLE;
    NVIC_Init(&NVIC_InitStructure);    EXTI_InitStructure.EXTI_Line    = EXTI_Line5;
    EXTI_InitStructure.EXTI_Mode    = EXTI_Mode_Interrupt;
    EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
    EXTI_InitStructure.EXTI_LineCmd = ENABLE;
    EXTI_Init(&EXTI_InitStructure);
}

时钟使能:使能 GPIOG 和复用功能 I/O(AFIO)的时钟。

GPIO 初始化:配置 GPIOG 的 GPIO_Pin_5 为上拉输入模式,速度为 50MHz。

外部中断线配置:将 GPIOG 的 GPIO_Pin_5 映射到外部中断线 EXTI_Line5。

NVIC 配置:

  • 配置外部中断线 5~9 的中断通道 EXTI9_5_IRQn。
  • 设置抢占优先级为 2,子优先级为 3,并使能该中断通道。

EXTI 配置:配置外部中断线 EXTI_Line5 为中断模式,下降沿触发,并使能该中断线。

5. EXTI9_5_IRQHandler 函数

void EXTI9_5_IRQHandler(void)
{
    if (EXTI_GetITStatus(EXTI_Line5) == SET)
    {
        delay_ms(10);
        if (GPIO_ReadInputDataBit(GPIOG, GPIO_Pin_5) == Bit_RESET)
        {
            printf("The key is pressed\n");
        }
    }
    EXTI_ClearITPendingBit(EXTI_Line5);
}

中断处理:

  • 检查外部中断线 EXTI_Line5 的中断标志位是否被置位。
  • 进行 10ms 的消抖处理。
  • 读取 GPIOG 的 GPIO_Pin_5 的输入电平,如果为低电平,则打印 "The key is pressed"。
  • 清除外部中断线 EXTI_Line5 的中断标志位。

6. UART_Configuration 函数

void UART_Configuration(uint32_t bound)
{
    GPIO_InitTypeDef  GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);    GPIO_InitStructure.GPIO_Pin   = GPIO_Pin_9;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF_PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);    GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init(GPIOA, &GPIO_InitStructure);    USART_InitStructure.USART_BaudRate            = bound;
    USART_InitStructure.USART_WordLength          = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits            = USART_StopBits_1;
    USART_InitStructure.USART_Parity              = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode                = USART_Mode_Rx | USART_Mode_Tx;    USART_Init(USART_TEST, &USART_InitStructure);
    USART_Cmd(USART_TEST, ENABLE);
}

时钟使能:使能 USART1 和 GPIOA 的时钟。

GPIO 初始化:

  • 配置 GPIOA 的 GPIO_Pin_9 为复用推挽输出模式,用于 USART1 的发送引脚。
  • 配置 GPIOA 的 GPIO_Pin_10 为浮空输入模式,用于 USART1 的接收引脚。

USART 初始化:

  • 配置 USART1 的波特率、数据位、停止位、校验位、硬件流控制和工作模式。
  • 使能 USART1。

7. EXTI1_IRQHandler 函数

void EXTI1_IRQHandler(void)
{
    if (EXTI_GetITStatus(EXTI_Line1) == SET)
    {
        delay_ms(10);
        if (GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_1) == Bit_SET)
        {
            printf("The key is pressed\n");
        }
    }
    EXTI_ClearITPendingBit(EXTI_Line1);
}

中断处理:处理外部中断线 EXTI_Line1 的中断事件,与 EXTI9_5_IRQHandler 类似,但该中断在本代码中未被使用。

8. SER_PutChar 函数和 fputc 函数

int SER_PutChar(int ch)
{
    while (!USART_GetFlagStatus(USART_TEST, USART_FLAG_TC));
    USART_SendData(USART_TEST, (uint8_t)ch);    return ch;
}int fputc(int c, FILE *f)
{
    if (c == '\n')
    {
        SER_PutChar('\r');
    }
    return (SER_PutChar(c));
}

SER_PutChar 函数:将一个字符通过 USART 发送出去,等待发送完成标志位被置位。

fputc 函数:重定向标准输出函数,将换行符 \n 替换为回车符 \r 后再发送

2.4 下载验证

把编译好的程序下载到开发板并复位,按下按键可以在串口查看按钮是否按下。

http://www.xdnf.cn/news/418267.html

相关文章:

  • 工业4G路由器IR5000公交站台物联网应用解决方案
  • 游戏引擎学习第275天:将旋转和剪切传递给渲染器
  • 【Linux】简单设计libc库
  • Spring Boot之Web服务器的启动流程分析
  • Antd中Form详解:
  • Mapreduce初使用
  • 第四章 部件篇之按钮矩阵部件
  • 在Linux中使用 times函数 和 close函数 两种方式 打印进程时间。
  • 线代第二章矩阵第八节逆矩阵、解矩阵方程
  • 【计算机视觉】OpenCV项目实战:基于face_recognition库的实时人脸识别系统深度解析
  • 光谱相机的光电信号转换
  • 基于Java的家政服务平台设计与实现(代码+数据库+LW)
  • 游戏引擎学习第277天:稀疏实体系统
  • GNU Screen 曝多漏洞:本地提权与终端劫持风险浮现
  • 前端如何应对精确数字运算?用BigNumber.js解决JavaScript原生Number类型在处理大数或高精度计算时的局限性
  • SQL中联表的运用
  • OpenHarmony 开源鸿蒙南向开发——linux下使用make交叉编译第三方库——mqtt库
  • WSL 安装 Debian 12 后,Linux 如何安装 nginx ?
  • Boby家族之Smart Boby:你的智能编程助手
  • YOLOv11融合[AAAI2025]的PConv模块
  • [51单片机]---DS18B20 温度检测
  • 第六节第二部分:抽象类的应用-模板方法设计模式
  • vim 练习题
  • 苍穹外卖--新增菜品
  • 按键精灵ios脚本新增元素功能助力辅助工具开发(一)
  • 机器学习07-归一化与标准化
  • mybatis中${}和#{}的区别
  • 【RabbitMQ】工作队列和发布/订阅模式的具体实现
  • 微服务八股(自用)
  • React Native告别图标体积大手动更换慢的噩梦:让图标更新像修改文字一样简单