第六章 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 提供了两种选择:QFN100和QFN68。
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 下载验证
把编译好的程序下载到开发板并复位,按下按键可以在串口查看按钮是否按下。