4X4矩阵键盘扫描程序
4X4矩阵键盘扫描:
1. 4根行线的GIO均设为Output,根列线的GIO均设为Input;
2. 4根行线的GIO分别置为0111、1011、1101、1110,读逐一读取列线GIO的值,可确定是哪一个按键。
电路图例如以下:
注意:
1. 图中用作输入的GIO,一定要有一个上拉电阻。
2. 芯片中的每个引脚是否用作了GPIO口来用。需配置芯片的寄存器,使引脚当作GPIO口来使用,才会有效。
測试代码例如以下:
#define KEY_GIO_ROW_1 37
#define KEY_GIO_ROW_2 33
#define KEY_GIO_ROW_3 32
#define KEY_GIO_ROW_4 35
#define KEY_GIO_COL_1 22
#define KEY_GIO_COL_2 23
#define KEY_GIO_COL_3 24
#define KEY_GIO_COL_4 36
int scanKey()
{
int keyValue = 0;
int col1Value=0,col2Value=0,col3Value=0,col4Value=0,row1Value=0,row2Value=0,row3Value=0,row4Value=0;
static int press1=0,press2=0,press3=0,press4=0;
static int press5=0,press6=0,press7=0,press8=0;
static int press9=0,press10=0,press11=0,press12=0;
static int press13=0,press14=0,press15=0,press16=0; dm365SetGPIO(KEY_GIO_ROW_1, 0);
dm365SetGPIO(KEY_GIO_ROW_2, 1);
dm365SetGPIO(KEY_GIO_ROW_3, 1);
dm365SetGPIO(KEY_GIO_ROW_4, 1);
col1Value = dm365GetGPIO(KEY_GIO_COL_1);
col2Value = dm365GetGPIO(KEY_GIO_COL_2);
col3Value = dm365GetGPIO(KEY_GIO_COL_3);
col4Value = dm365GetGPIO(KEY_GIO_COL_4);
keyValue = col1Value | (col2Value << 1) | (col3Value << 2) | (col4Value << 3);
// printf("=1==keyValue = %x\n",keyValue);
switch(keyValue)
{
case 0x0E:
{
if(!press1)
{
press1 = 1;
printf("KEY 1\n");
}
}
break;
case 0x0D:
{
if(!press2)
{
press2 = 1;
printf("KEY 2\n");
}
}
break;
case 0x0B:
{
if(!press3)
{
press3 = 1;
printf("KEY 3\n");
}
}
break;
case 0x07:
{
if(!press4)
{
press4 = 1;
printf("KEY 4\n");
}
}
break;
default:
{
press1 = 0;
press2 = 0;
press3 = 0;
press4 = 0;
}
break;
} dm365SetGPIO(KEY_GIO_ROW_1, 1);
dm365SetGPIO(KEY_GIO_ROW_2, 0);
dm365SetGPIO(KEY_GIO_ROW_3, 1);
dm365SetGPIO(KEY_GIO_ROW_4, 1);
col1Value = dm365GetGPIO(KEY_GIO_COL_1);
col2Value = dm365GetGPIO(KEY_GIO_COL_2);
col3Value = dm365GetGPIO(KEY_GIO_COL_3);
col4Value = dm365GetGPIO(KEY_GIO_COL_4);
keyValue = col1Value | (col2Value << 1) | (col3Value << 2) | (col4Value << 3);
// printf("=2==keyValue = %x\n",keyValue);
switch(keyValue)
{
case 0x0E:
{
if(!press5)
{
press5 = 1;
printf("KEY 5\n");
}
}
break;
case 0x0D:
{
if(!press6)
{
press6 = 1;
printf("KEY 6\n");
}
}
break;
case 0x0B:
{
if(!press7)
{
press7 = 1;
printf("KEY 7\n");
}
}
break;
case 0x07:
{
if(!press8)
{
press8 = 1;
printf("KEY 8\n");
}
}
break;
default:
{
press5 = 0;
press6 = 0;
press7 = 0;
press8 = 0;
}
break;
} dm365SetGPIO(KEY_GIO_ROW_1, 1);
dm365SetGPIO(KEY_GIO_ROW_2, 1);
dm365SetGPIO(KEY_GIO_ROW_3, 0);
dm365SetGPIO(KEY_GIO_ROW_4, 1);
col1Value = dm365GetGPIO(KEY_GIO_COL_1);
col2Value = dm365GetGPIO(KEY_GIO_COL_2);
col3Value = dm365GetGPIO(KEY_GIO_COL_3);
col4Value = dm365GetGPIO(KEY_GIO_COL_4);
keyValue = col1Value | (col2Value << 1) | (col3Value << 2) | (col4Value << 3);
// printf("=3==keyValue = %x\n",keyValue);
switch(keyValue)
{
case 0x0E:
{
if(!press9)
{
press9 = 1;
printf("KEY 9\n");
}
}
break;
case 0x0D:
{
if(!press10)
{
press10 = 1;
printf("KEY 10\n");
}
}
break;
case 0x0B:
{
if(!press11)
{
press11 = 1;
printf("KEY 11\n");
}
}
break;
case 0x07:
{
if(!press12)
{
press12 = 1;
printf("KEY 12\n");
}
}
break;
default:
{
press9 = 0;
press10 = 0;
press11 = 0;
press12 = 0;
}
break;
} dm365SetGPIO(KEY_GIO_ROW_1, 1);
dm365SetGPIO(KEY_GIO_ROW_2, 1);
dm365SetGPIO(KEY_GIO_ROW_3, 1);
dm365SetGPIO(KEY_GIO_ROW_4, 0);
col1Value = dm365GetGPIO(KEY_GIO_COL_1);
col2Value = dm365GetGPIO(KEY_GIO_COL_2);
col3Value = dm365GetGPIO(KEY_GIO_COL_3);
col4Value = dm365GetGPIO(KEY_GIO_COL_4);
keyValue = col1Value | (col2Value << 1) | (col3Value << 2) | (col4Value << 3);
// printf("=4==keyValue = %x\n",keyValue);
switch(keyValue)
{
case 0x0E:
{
if(!press13)
{
press13 = 1;
printf("KEY 13\n");
}
}
break;
case 0x0D:
{
if(!press14)
{
press14 = 1;
printf("KEY 14\n");
}
}
break;
case 0x0B:
{
if(!press15)
{
press15 = 1;
printf("KEY 15\n");
}
}
break;
case 0x07:
{
if(!press16)
{
press16 = 1;
printf("KEY 16\n");
}
}
break;
default:
{
press13 = 0;
press14 = 0;
press15 = 0;
press16 = 0;
}
break;
} return keyValue;
}
void *KeyMngThread()
{
int resetValue = 1;
int resetCout = 0;
int alarmInValue = 1;
int alarmInCout = 0;
while(1)
{
resetValue = dm365GetGPIO(GIO_RESET);
if(0 == resetValue)
{
resetCout++;
}
else if(1 == resetValue)
{
resetCout = 0;
}
if(resetCout == 30)
{
resetCout = 0;
system("rm -f /mnt/nand/sysenv.cfg");
system("/bin/sync");
// System("reboot");
system("/tmp/shutdown -r now \n");
}
alarmInValue = dm365GetGPIO(GIO_ALARM_IN);
if(0 == alarmInValue)
{
dm365SetGPIO(GIO_LED,0); //control led off .
}
else if(1 == alarmInValue)
{
dm365SetGPIO(GIO_LED,1); //control led on .
} scanKey(); usleep(100000);
} }
代码中dm365SetGPIO( )里将GPIO默认设置为Output,
dm365GetGPIO( )中将GPIO默认设置为Input,
通过字符设备驱动实现应用层操作底层GPIO。
4X4矩阵键盘扫描程序的更多相关文章
- 4x4矩阵键盘 扫描程序
一:不排除第四位异常处理 uchar JuzhenkeyScan() { // P3=0xfe; // temp=P3; // while(temp!=0xfe) // { // temp=P3; / ...
- 4x4矩阵键盘扫描
4x4矩阵键盘扫描 Windows 10 IoT Core 是微软针对物联网市场的一个重要产品,与以往的Windows版本不同,是为物联网设备专门设计的,硬件也不仅仅限于x86架构,同时可以在ARM架 ...
- Win10 IoT C#开发 6 - 4x4矩阵键盘扫描
Windows 10 IoT Core 是微软针对物联网市场的一个重要产品,与以往的Windows版本不同,是为物联网设备专门设计的,硬件也不仅仅限于x86架构,同时可以在ARM架构上运行. 上一章我 ...
- 「雕爷学编程」Arduino动手做(26)——4X4矩阵键盘模块
37款传感器与模块的提法,在网络上广泛流传,其实Arduino能够兼容的传感器模块肯定是不止37种的.鉴于本人手头积累了一些传感器和模块,依照实践出真知(一定要动手做)的理念,以学习和交流为目的,这里 ...
- 基于FPGA的4x4矩阵键盘驱动调试
好久不见,因为博主最近两个月有点事情,加上接着考试,考完试也有点事情要处理,最近才稍微闲了一些,这才赶紧记录分享一篇博文.FPGA驱动4x4矩阵键盘.这个其实原理是十分简单,但是由于博主做的时候遇到了 ...
- STM32 实现 4*4 矩阵键盘扫描(HAL库、标准库 都适用)
本文实现的代码是基于STM32HAL库的基础上的,不过标准库也可以用,只是调用的库函数不同,逻辑跟配置是一样的,按我这里的逻辑来配置即可. 1.键盘原理图: 原理举例:先把 F0-F7 内部拉高,这样 ...
- stm32矩阵键盘扫描数据通过USB发送
Keyboard.c #include "keyboard.h"#include "my_usb.h"#include " ...
- MCU软件最佳实践——矩阵键盘驱动
1.矩阵键盘vs独立按键 在mcu应用开发过程中,独立按键比较常见,但是在需要的按键数比较多时,使用矩阵键盘则可以减少io占用,提高系统资源利用率.例如,某mcu项目要求有16个按钮,如果采用独立按键 ...
- 【STM32学习笔记】STM32f407 使用4*4矩阵键盘
作者:李剀 出处:https://www.cnblogs.com/kevin-nancy/ 欢迎转载,但也请保留上面这段声明.谢谢! 写在前面: 这是本人第一次开始写博客,可能写的不是很好,也请大家谅 ...
随机推荐
- 【CSWS2014 Summer School】深度问答技术及其在搜索中的应用-马艳军
Title: 深度问答技术及其在搜索中的应用 马艳军博士, 百度 Abstract: 深度问答(DeepQA)是一种基于对自然语言深度理解的智能问答技术,其核心技术涉及知识图谱建设.语义表示和计算.语 ...
- 【转】TCP/IP详解学习笔记(一)
TCP/IP详解学习笔记 这位仁兄写得太好了. http://blog.csdn.net/goodboy1881/category/204448.aspx TCP/IP详解学习笔记(13)-T ...
- iOS 事件传递及响应过程
iOS 事件传递及响应过程 -->>事件到来-->>事件分发 -->>事件响应 事件( Events) 官方文档( Events(iOS)) 是这样描写叙述的: U ...
- curl命令(测试连接命令)
curl命令是一个利用URL规则在命令行下工作的文件传输工具.它支持文件的上传和下载,所以是综合传输工具,但按传统,习惯称curl为下载工具.作为一款强力工具,curl支持包括HTTP.HTTPS.f ...
- Windows 消息机制浅析
1. Windows 的历史 中国人喜欢以史为鉴,而事实也确实是,如果你能知道一件事情的来龙去脉,往往可以更容易地理解事物为什么会表现为当前这样的现状.所以,我的介绍性开场白通常会以一段历 ...
- bss段,代码段及数据段,堆栈段的区别
bss段,代码段及数据段,堆栈段的区别 时间:2012-11-21 10:0772人阅读 BSS段:BSS段(bss segment)通常是指用来存放程序中未初始化的全局变量的一块内存区域.BSS是英 ...
- Openstack网络相关概念比较复杂,经常使人混淆,本文进行相关说明。
Openstack网络相关概念比较复杂,经常使人混淆,本文进行相关说明. 文中相关术语与缩写 英文 缩写 中文 Virtual Local Area Network VLAN 虚拟局域网 Virtua ...
- A. Dreamoon and Stairs(Codeforces Round #272)
A. Dreamoon and Stairs time limit per test 1 second memory limit per test 256 megabytes input standa ...
- 更改npm全局模块和cache默认安装位置
来源于:http://blog.csdn.net/friendan/article/details/51736231 1.因为我安装的Node.js自带了npm,所以在nodejs文件夹里面新建以下两 ...
- c:forEach标签
//varStat代表 遍历typeListDesc集合所用到的方法 <!-- stat当前迭代的第几项 --> <c:forEach var="type" it ...