当前位置:首页 > 单片机 > 单片机
[导读]ps2key.c/*============================================================使用1602液晶显示和PS/2键盘的示例 明浩 2004/2-------------------------------------------------http://www.cdle.net http://bbs.cdle.n

ps2key.c
/*============================================================
使用1602液晶显示和PS/2键盘的示例 明浩 2004/2
-------------------------------------------------
http://www.cdle.net http://bbs.cdle.net/
==============================================================

SMC1602A(16*2)模拟口线接线方式
连接线图:
---------------------------------------------------
|LCM-----51 | LCM-----51 | LCM------51 |
--------------------------------------------------|
|DB0-----P1.0 | DB4-----P1.4 | RW-------P2.0 |
|DB1-----P1.1 | DB5-----P1.5 | RS-------P2.1 |
|DB2-----P1.2 | DB6-----P1.6 | E--------P2.2 |
|DB3-----P1.3 | DB7-----P1.7 | VLCD接1K电阻到GND|
---------------------------------------------------

Keyboard接线
PS/2--------51
1 DATA------P3.4
3 GND
4 VCC
5 CLK-------P3.3 接在51的外部中断,触发方式为低电平

本程式源码只供学习参考,不得应用于商业用途,如有需要请联系作者。

[注:AT89x51使用12M或11.0592M晶振 实测使用11.0592M]
[Keil uV2 7.01编译运行通过 程式中没有做键盘数据的奇偶校验]

=============================================================*/
#include
#include "scancodes.h"

#define LCM_RW P2_0 //定义LCD引脚
#define LCM_RS P2_1
#define LCM_E P2_2
#define LCM_Data P1

#define Key_Data P3_4 //定义Keyboard引脚
#define Key_CLK P3_3

#define Busy 0x80 //用于检测LCM状态字中的Busy标识

void LCMInit(void);
void DisplayOneChar(unsigned char X unsigned char Y unsigned char DData);
void DisplayListChar(unsigned char X unsigned char Y unsigned char code *DData);
void Delay5Ms(void);
void Delay400Ms(void);
void Decode(unsigned char ScanCode);
void WriteDataLCM(unsigned char WDLCM);
void WriteCommandLCM(unsigned char WCLCM BuysC);

unsigned char ReadDataLCM(void);
unsigned char ReadStatusLCM(void);
unsigned char code cdle_net[] = {"-www.cdle.net--"};
unsigned char code email[] = {"pnzwzw@cdle.net"};
unsigned char code Cls[] = {" "};
static unsigned char IntNum = 0; //中断次数计数
static unsigned char KeyV; //键值
static unsigned char DisNum = 0; //显示用指针
static unsigned char Key_UP=0 Shift = 0;//Key_UP是键松开标识,Shift是Shift键按下标识
static unsigned char BF = 0; //标识是否有字符被收到

void main(void)
{
unsigned char TempCyc;

Delay400Ms(); //启动等待,等LCM讲入工作状态
LCMInit(); //LCM初始化
Delay5Ms(); //延时片刻(可不要)

DisplayListChar(0 0 cdle_net);
DisplayListChar(0 1 email);
ReadDataLCM();//测试用句无意义
for (TempCyc=0; TempCyc<10; TempCyc++)
Delay400Ms(); //延时
DisplayListChar(0 1 Cls);

IT1 = 0; //设外部中断1为低电平触发
EA = 1;
EX1 = 1; //开中断

do
{
if (BF)
Decode(KeyV);
else
EA = 1; //开中断
}
while(1);
}

//写数据
void WriteDataLCM(unsigned char WDLCM)
{
ReadStatusLCM(); //检测忙
LCM_Data = WDLCM;
LCM_RS = 1;
LCM_RW = 0;
LCM_E = 0; //若晶振速度太高能在这后加小的延时
LCM_E = 0; //延时
LCM_E = 1;
}

//写指令
void WriteCommandLCM(unsigned char WCLCM BuysC) //BuysC为0时忽略忙检测
{
if (BuysC) ReadStatusLCM(); //根据需要检测忙
LCM_Data = WCLCM;
LCM_RS = 0;
LCM_RW = 0;
LCM_E = 0;
LCM_E = 0;
LCM_E = 1;
}

//读数据
unsigned char ReadDataLCM(void)
{
LCM_RS = 1;
LCM_RW = 1;
LCM_E = 0;
LCM_E = 0;
LCM_E = 1;
return(LCM_Data);
}

//读状态
unsigned char ReadStatusLCM(void)
{
LCM_Data = 0xFF;
LCM_RS = 0;
LCM_RW = 1;
LCM_E = 0;
LCM_E = 0;
LCM_E = 1;
while (LCM_Data & Busy); //检测忙信号
return(LCM_Data);
}

void LCMInit(void) //LCM初始化
{
LCM_Data = 0;
WriteCommandLCM(0x38 0 //三次显示模式设置,不检测忙信号
Delay5Ms();
WriteCommandLCM(0x38 0 0 style="LINE-HEIGHT: 150%">WriteCommandLCM(0x38 1 //显示模式设置 开始需求每次检测忙信号
WriteCommandLCM(0x08 1 //关闭显示
WriteCommandLCM(0x01 1 //显示清屏
WriteCommandLCM(0x06 1 // 显示光标移动设置
WriteCommandLCM(0x0F 1 // 显示开及光标设置
}

//按指定位置显示一个字符
void DisplayOneChar(unsigned char X unsigned char Y unsigned char DData)
{
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
if (Y) X |= 0x40; //当要显示第二行时地址码+0x40;
X |= 0x80; //算出指令码
WriteCommandLCM(X 1 //发命令字
WriteDataLCM(DData); //发数据
}

//按指定位置显示一串字符
void DisplayListChar(unsigned char X unsigned char Y unsigned char code *DData)
{
unsigned char ListLength;

ListLength = 0;
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
while (DData[ListLength]>0x19) //若到达字串尾则退出
{
if (X <= 0xF) //X坐标应小于0xF
{
DisplayOneChar(X Y DData[ListLength]); //显示单个字符
ListLength++;
X++;
}
}
}

//5ms延时
void Delay5Ms(void)
{
unsigned int TempCyc = 5552;
while(TempCyc--);
}

//400ms延时
void Delay400Ms(void)
{
unsigned char TempCycA = 5;
unsigned int TempCycB;
while(TempCycA--)
{
TempCycB=7269;
while(TempCycB--);
};
}

void Keyboard_out(void) interrupt 2
{
if ((IntNum > 0) && (IntNum < 9))
{
KeyV = KeyV >> 1; //因键盘数据是低>>高,结合上一句所以右移一位
if (Key_Data) KeyV = KeyV | 0x80; //当键盘数据线为1时为1到最高位
}
IntNum++;
while (!Key_CLK); //等待PS/2CLK拉高

if (IntNum > 10)
{
IntNum = 0; //当中断11次后表示一帧数据收完,清变量准备下一次接收
BF = 1; //标识有字符输入完了
EA = 0; //关中断等显示完后再开中断 (注:如这里不用BF和关中断直接调Decode()则所Decode中所调用的所有函数要声明为再入函数)
}
}

void Decode(unsigned char ScanCode) //注意:如SHIFT+G为12H 34H F0H 34H F0H 12H,也就是说shift的通码+G的通码+shift的断码+G的断码
{
unsigned char TempCyc;

if (!Key_UP) //当键盘松开时
{
switch (ScanCode)
{
case 0xF0 : // 当收到0xF0,Key_UP置1表示断码开始
Key_UP = 1;
break;

case 0x12 : // 左 SHIFT
Shift = 1;
break;

case 0x59 : // 右 SHIFT
Shift = 1;
break;

default:
if (DisNum > 15)
{
DisplayListChar(0 1 Cls);//清LCD第二行
DisNum = 0;
}
if(!Shift) //如果SHIFT没按下
{
for (TempCyc = 0;(UnShifted[TempCyc][0]!=ScanCode)&&(TempCyc<59); TempCyc++); //查表显示
if (UnShifted[TempCyc][0] == ScanCode) DisplayOneChar(DisNum 1 UnShifted[TempCyc][1]);
DisNum++;
}
else //按下SHIFT
{
for(TempCyc = 0; (Shifted[TempCyc][0]!=ScanCode)&&(TempCyc<59); TempCyc++); //查表显示
if (Shifted[TempCyc][0] == ScanCode) DisplayOneChar(DisNum 1 Shifted[TempCyc][1]);
DisNum++;
}

break;
}
}
else
{
Key_UP = 0;
switch (ScanCode) //当键松开时不处理判码,如G 34H F0H 34H 那么第二个34H不会被处理
{
case 0x12 : // 左 SHIFT
Shift = 0;
break;

case 0x59 : // 右 SHIFT
Shift = 0;
break;
}
}
BF = 0; //标识字符处理完了
}

scancodes.h

unsigned char code UnShifted[59][2] = {
0x1C 'a'
0x32 'b'
0x21 'c'
0x23 'd'
0x24 'e'
0x2B 'f'
0x34 'g'
0x33 'h'
0x43 'i'
0x3B 'j'
0x42 'k'
0x4B 'l'
0x3A 'm'
0x31 'n'
0x44 'o'
0x4D 'p'
0x15 'q'
0x2D 'r'
0x1B 's'
0x2C 't'
0x3C 'u'
0x2A 'v'
0x1D 'w'
0x22 'x'
0x35 'y'
0x1A 'z'
0x45 '0'
0x16 '1'
0x1E '2'
0x26 '3'
0x25 '4'
0x2E '5'
0x36 '6'
0x3D '7'
0x3E '8'
0x46 '9'
0x0E '`'
0x4E '-'
0x55 '='
0x5D '\'
0x29 ' '
0x54 '['
0x5B ']'
0x4C ';'
0x52 '''
0x41 ' '
0x49 '.'
0x4A '/'
0x71 '.'
0x70 '0'
0x69 '1'
0x72 '2'
0x7A '3'
0x6B '4'
0x73 '5'
0x74 '6'
0x6C '7'
0x75 '8'
0x7D '9'
};

unsigned char code Shifted[59][2] = {
0x1C 'A'
0x32 'B'
0x21 'C'
0x23 'D'
0x24 'E'
0x2B 'F'
0x34 'G'
0x33 'H'
0x43 'I'
0x3B 'J'
0x42 'K'
0x4B 'L'
0x3A 'M'
0x31 'N'
0x44 'O'
0x4D 'P'
0x15 'Q'
0x2D 'R'
0x1B 'S'
0x2C 'T'
0x3C 'U'
0x2A 'V'
0x1D 'W'
0x22 'X'
0x35 'Y'
0x1A 'Z'
0x45 '0'
0x16 '1'
0x1E '2'
0x26 '3'
0x25 '4'
0x2E '5'
0x36 '6'
0x3D '7'
0x3E '8'
0x46 '9'
0x0E '~'
0x4E '_'
0x55 '+'
0x5D '|'
0x29 ' '
0x54 '{'
0x5B '}'
0x4C ':'
0x52 '"'
0x41 '<'
0x49 '>'
0x4A '?'
0x71 '.'
0x70 '0'
0x69 '1'
0x72 '2'
0x7A '3'
0x6B '4'
0x73 '5'
0x74 '6'
0x6C '7'
0x75 '8'
0x7D '9'
};


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

单片机是一种嵌入式系统,它是一块集成电路芯片,内部包含了处理器、存储器和输入输出接口等功能。

关键字: 单片机 编写程序 嵌入式

在现代电子技术的快速发展中,单片机以其高度的集成性、稳定性和可靠性,在工业自动化、智能家居、医疗设备、航空航天等诸多领域得到了广泛应用。S32单片机,作为其中的佼佼者,其引脚功能丰富多样,是实现与外部设备通信、控制、数据...

关键字: s32单片机引脚 单片机

在微控制器领域,MSP430与STM32无疑是两颗璀璨的明星。它们各自凭借其独特的技术特点和广泛的应用领域,在市场上占据了重要的位置。本文将深入解析MSP430与STM32之间的区别,探讨它们在不同应用场景下的优势和局限...

关键字: MSP430 STM32 单片机

该系列产品有助于嵌入式设计人员在更广泛的系统中轻松实现USB功能

关键字: 单片机 嵌入式设计 USB

单片机编程语言是程序员与微控制器进行交流的桥梁,它们构成了单片机系统的软件开发基石,决定着如何有效、高效地控制和管理单片机的各项资源。随着微控制器技术的不断发展,针对不同应用场景的需求,形成了丰富多样的编程语言体系。本文...

关键字: 单片机 微控制器

单片机,全称为“单片微型计算机”或“微控制器”(Microcontroller Unit,简称MCU),是一种高度集成化的电子器件,它是现代科技领域的关键组件,尤其在自动化控制、物联网、消费电子、汽车电子、工业控制等领域...

关键字: 单片机 MCU

STM32是由意法半导体公司(STMicroelectronics)推出的基于ARM Cortex-M内核的32位微控制器系列,以其高性能、低功耗、丰富的外设接口和强大的生态系统深受广大嵌入式开发者喜爱。本文将详细介绍S...

关键字: STM32 单片机

在当前的科技浪潮中,单片机作为嵌入式系统的重要组成部分,正以其强大的功能和广泛的应用领域受到越来越多行业的青睐。在众多单片机中,W79E2051以其卓越的性能和稳定的工作特性,成为市场上的明星产品。本文将深入探讨W79E...

关键字: 单片机 w79e2051单片机

单片机,又称为微控制器或微处理器,是现代电子设备中的核心部件之一。它集成了中央处理器、存储器、输入输出接口等电路,通过外部信号引脚与外部设备进行通信,实现对设备的控制和管理。本文将详细介绍单片机的外部信号引脚名称及其功能...

关键字: 单片机 微控制器 中央处理器

随着科技的飞速发展,单片机和嵌入式系统在现代电子设备中的应用越来越广泛。它们不仅提高了设备的智能化水平,还推动了各行各业的创新与发展。在单片机和嵌入式系统的开发中,编程语言的选择至关重要。本文将深入探讨单片机和嵌入式系统...

关键字: 单片机 嵌入式系统 电子设备
关闭
关闭