51mcu模拟ps2键盘发送数据

发布者:传邮万里最新更新时间:2012-09-04 来源: 51hei关键字:51mcu  模拟ps2键盘  发送数据 手机看文章 扫描二维码
随时随地手机看文章

这个是程序是来自我以前开发的一个项目,其中里面的的void ps_send(uchar x) 这个函数,便可实现向电脑发送按键的数据 你只要对照 ps2 扫描码的表填入适当的参数即可

#include "STC89C51.H"
#include"keyboard.h"
#include
#define nop()  _nop_()

uchar *receive_buf;//接收来自计算机的数据

//uchar key;
unsigned char key_buf[21];
/******延时子程序********/
void delay_ms(uchar a)
{uchar i;
uchar j;

for(j=0;j0;i--);
}
 
void delay_us(void)//30us
{uchar i;

 	for(i=0;i<7;i++)
	{  _nop_();
	}
}

/************行列转换成索引号*****************/
/*
uchar code_chang(uchar ini_code)
 {uchar  i;
i=ini_code;
 switch(i)
     {case 0x01:return(1);break;
     case 0x02:return(2);break;
     case 0x04:return(3);break;
     case 0x08:return(4);break;
     case 0x10:return(5);break;	
 	 default:;
     }
 	
 	
 }
*/
//初始化按键端口
void init_key(void)
{
   
unsigned char i=0;
  
//右按键板
P2=0xff;

//左按键板
P0=0xff;

k_num_5 = 1;
k_num_4 = 1;
k_num_7 = 1;
k_num_8 = 1;
k_num_9 = 1;
//   led=0;
   for(i=0;i<21;i++)
   {
       key_buf[i]=1;
	   
   }

 
}

//读按键值并将其转换为索引号
unsigned char read_key(void)
{unsigned char i;
	key_buf[0]=k_esc; 
	key_buf[1]=k_F1; 
	key_buf[2]=k_F3; 
	key_buf[3]=k_F2;
	key_buf[4]=k_F5; 
	key_buf[5]=k_F4; 
	key_buf[6]=k_menu; 
	key_buf[7]=k_num_lock;
	
	key_buf[8]=k_num_6;
	key_buf[9]=k_num_1;
	key_buf[10]=k_num_2 ;
	key_buf[11]=k_num_3 ;
	key_buf[12]=k_num_0;
	key_buf[13]=k_del;
	key_buf[14]=k_enter;
	key_buf[15]=k_back;
	key_buf[16]=k_num_5;
	
	key_buf[17]=k_num_4;
	key_buf[18]=k_num_7;
	key_buf[19]=k_num_8;
	key_buf[20]=k_num_9;
	
	for(i=0;i<21;i++)	
 		{if(0==key_buf[i])
			{
		 	key_buf[i]=1;
		 	return i;
		 	break;
			}
  		}
}

 /********代码转换**************/
uchar change_code(uchar key)   //返回一个ASCII键值
{uchar i;
 static bit shitf_direct=0,key_flat=1;

i=key;
	if(7==i&&1==key_flat)
	{key_flat=0;
	shitf_direct=1;
	}
	else if(7==i&&0==key_flat)
	{key_flat=1;
	shitf_direct=0;
	}
	else
	{}
    if ( 1==shitf_direct)//转到方向盘(1为方向盘,0为数字盘)
	{   lock_LED =0;
	switch(i) 

	 	{
	 	case 12:return(0x2D);break;//insert
     	case 9:return(0x23);break;//end
     	case 10:return(0x28);break;//down
	 	case 11:return(0x22);break;//pagedown
	 	case 17:return(0x25);break;//left
	 	case 16:return(0x0C);break;//NC
	 	case 8:return(0x27);break;//right
	 	case 18:return(0x24);break;//home
	 	case 19:return(0x26);break;//up
	 	case 20:return(0x21);break;//pageup
	 	default:;
		}
	}
	else
       {lock_LED =1;
	   return(trdata[i]);
	   }
}
/*************键盘扫描*****************/ 
/*
void int1() interrupt 2 using 1
{//uchar scankb(void)
uchar sccode,recode;
 
 if((P0&0x0f)!=0x0f)//有键按下
   {delay_ms(10);
    if((P0&0x0f)!=0x0f)//仍然有键按下
   	  {recode=P0|0xf0; //列号保存
	    P0=0x0f;
   	  	sccode=0xfe;
	   while((sccode&0x40)!=0)//移位没完
	       {
		     P2=sccode;//行扫描开始
		   		if(P0 != 0x0F) //若在该行
		    	{
			 	P0=0x0f;
	         	P2=0; 
		        //return(code_chang(~sccode)+5*code_chang(~recode));//返回Keyword
			    key=code_chang(~sccode)+5*code_chang(~recode);break;
		     	}
	        	else 
				{sccode=(sccode<<1)|0x01;}//不在该行则扫下一行
		   }
	   }
   }
// return(0);
}

*/


/** ps/2发送数据   ***/
void ps_send(uchar x)
{ 	uchar i,temp,char_temp;
 	bit flat_check=1;
	delay_ms(3);
	temp=x;
	for(i=0;i<8;i++)
	{char_temp=temp&0x01;
		if(char_temp==0x01)
		{flat_check=!flat_check;
		}
	temp>>=1;
	}
	k_clk = 1;

	while(!k_clk);

	k_clk=1;
	k_data=1;
	if(k_clk==1)
	{delay_us();//30us
	}
	if(k_clk==1&&k_data==1)
	{k_data=0;
	nop();nop();nop();nop();nop();nop();nop();nop();nop();nop();//delay10us;
    k_clk=0;
	nop();nop();nop();nop();nop();//delay5us;
	temp=x;
		for(i=0;i<8;i++)
		{k_clk=1;
		nop();nop();nop();nop();nop();//delay5us;
		char_temp=temp&0x01;
			if(char_temp==0x01)
			{
			k_data=1;
			}
			else
			{
			k_data=0;
			}
		nop();nop();nop();nop();nop();nop();nop();nop();nop();nop();//delay10us;
		k_clk=0;
		nop();nop();nop();nop();nop();//delay5us;
		temp>>=1;
		}
	k_clk=1;
	nop();nop();nop();nop();nop();//delay5us;
	k_data=flat_check;
	nop();nop();nop();nop();nop();nop();nop();nop();nop();nop();//delay10us;
	k_clk=0;
	nop();nop();nop();nop();nop();//delay5us;
	k_clk=1;
	nop();nop();nop();nop();nop();//delay5us;
	k_data=1;
	nop();nop();nop();nop();nop();nop();nop();nop();nop();nop();//delay10us;
	k_clk=0;
	nop();nop();nop();nop();nop();//delay5us;
	k_clk=1;
	delay_us();
	k_clk=1;
	k_data=1;
	if(k_clk==1&&k_data==0)
	{
	return;//goto receive PC of data 
	}
	}
}



/*****************************************************************************************
串行通讯
采用11位带奇偶效检串口异步通信
******************************************************************************************/
/******串口发送***********/
/*
void series_sent(uchar keyword)
{uchar i,key_L;
 uint j;
 bit odd=0;
 key_L=keyword;
	for(i=0;i<8;i++)
	{odd=odd ^ key_L^7;
	key_L<<=1;
	}
SBUF=keyword;
TB8=odd;//odd=1为奇数 ,odd=0为偶数
while(TI==0);
for(j=0;j<500;j++);
   
}

*/

/****************串口中断***************************/
/*
void UART() interrupt 4 using 2
{bit odd,odd_check;
uchar i, buf;

if(RI)
{
	buf=SBUF;
	odd=RB8;
	for(i=0;i<8;i++)
	{odd_check=odd_check ^ buf ^7;
	buf<<=1;
	}
	if(odd_check!=odd)//校验出错发送错误代码0x55
	{
		
		SBUF=0x55;//错误代码0x55
		while(!TI);//等待发送完成
		TI=0;//清除发送标志
	}
	else
	{*receive_buf=SBUF;//将来自计算机的数据保存
	}
	RI=0;//清除接收标志
}
}

*/
/************************************************************
//mcu初始化
**************************************************************/
void ini(void)
{
TMOD=0X20;                /*定时器1八位自动重装*/
TL1=0XFA;TH1=0XFA;        /*12MHz,9600波特,误差8.5%*/
//SCON=0Xd8;PCON=0X10;      /*11位异步,定时器控制*/
//TR1=1;
   AUXR=0xbf;         //12T:0xbf    1T:0x40
   RCAP2H=0XFF;     //设T2为波特率发生器  12M----9600----0xFFD9
   RCAP2L=0Xd9;
   TR2=1;
// T2IE=1;
// T2RSE=1;
   ET2=1;
   TCLK=1;
   RCLK=1;
   TMOD=0x11;  //设T1为定时方式1,设T0为定时方式1
   TH1=0xec;  //5ms
   TL1=0X78;
   TH0=0xff;  //100us  ff9b
   TL0=0xaa;
   PCON=0x00;
//   SCON=0xD0;       //串行口工作方式3
//   ES=1;            //开串行口中断
   EA=1;
   ET0=1;           //T0开中断
   ET1=1;           //T1开中断
   IT0=1;  //外部中断0边沿触发, 
   EX0=1;//中断允许
   IT1=1;  // 外部中断1边沿触发?
   EX1=1;//中断允许
   TR1=1;           //启动T1
   TR0=1;           //启动T0
}

/**********看门狗***************/
void watchdog()
{
	WDT_CONTR = 0x3f;
}

/***************************/
void main(void)
{uchar chg_code,key;


ini();
init_key();
watchdog();//看门狗

	for(;;)
	{ 
//	key=scankb();//键盘扫描
    key= read_key();
	chg_code=change_code(key);//代码转换
//	series_sent(chg_code);//串口发送
	ps_send(chg_code);//PS/2发送
	watchdog();
	}

}
关键字:51mcu  模拟ps2键盘  发送数据 引用地址:51mcu模拟ps2键盘发送数据

上一篇:基于单片机的秒,分,时可调时钟的设计
下一篇:51单片机红外接收解码程序

推荐阅读最新更新时间:2024-03-16 13:07

avr的IO模拟SPI时序发送23位数据
#include iom16v.h /***********************************************************************/ #define DAT_L PORTA&=~(1 PA1) #define DAT_H PORTA|= (1 PA1) #define CLK_L PORTA&=~(1 PA0) #define CLK_H PORTA|= (1 PA0) #define ENA_L PORTA&=~(1 PA2) #define ENA_H PORTA|= (1 PA2) /*****************USART全局变量**************
[单片机]
STM32基于固件库学习笔记(6)使用DMA实现USART1发送数据
DMA简介 直接存储器存取(DMA)用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作。换而言之就是当外设有数据发送给mcu,此时可以使用DMA接收到用户定义空间(不占用cpu),接收完成在产生中断发给mcu(才占用CPU)反正一样。 当CPU和DMA同时访问相同的目标(RAM或外设)时,DMA请求会暂停CPU访问系统总线达若干个周期,总线仲裁器执行循环调度,以保证CPU至少可以得到一半的系统总线(存储器或外设)带宽。 两个DMA控制器有12个通道(DMA1有7个通道,DMA2有5个通道),每个通道专门用来管理来自于一个或多个外设
[单片机]
STM32基于固件库学习笔记(6)使用DMA实现USART1<font color='red'>发送</font><font color='red'>数据</font>
[nrf51822][SDK12.3] BLE分层设计 NUS 透传数据队列发送,提升带宽利用率
四、 透传服务,加入队列机制 /******************************************************************************** * @file biz_ble_nus.c * @author jianqiang.xue * @Version V1.0.0 * @Date 2022-03-16 * @brief ble数据透传服务(从机) https://blog.csdn.net/weixin_41572450/article/details/84036968 *****************************************
[单片机]
c51串口发送数据程序
#include reg51.h //----------------------------------------------------------------- void UART_init(); //串口初始化函数 void COM_send(void); //串口发送函数 char code str = 计算机你好\r\n ; char code str1 = hello!\r\n ; //------------------------------------------------------------------- void main(void) { unsigned char i;
[单片机]
STM32的USART发送数据时如何使用TXE和TC标志
在USART的发送端有2个寄存器,一个是程序可以看到的USART_DR寄存器(下图中阴影部分的TDR),另一个是程序看不到的移位寄存器(下图中阴影部分Transmit Shift Register)。 对应USART数据发送有两个标志,一个是TXE=发送数据寄存器空,另一个是TC=发送结束;对照下图,当TDR中的数据传送到移位寄存器后,TXE被设置,此时移位寄存器开始向TX信号线按位传输数据,但因为TDR已经变空,程序可以把下一个要发送的字节(操作USART_DR)写入TDR中,而不必等到移位寄存器中所有位发送结束,所有位发送结束时(送出停止位后)硬件会设置TC标志。 另一方面,在刚刚初始化好USART还没有发送任何数据时,也会
[单片机]
STM32的USART<font color='red'>发送</font><font color='red'>数据</font>时如何使用TXE和TC标志
奥迪使用高精度数据向驾驶员发送实时路面信息
3月8日,奥迪宣布将首次使用高精度群体数据(swarm data)改善其汽车到X服务(car-to-X)“地面危险警报”(LHI)的监测能力。这是奥迪实现更安全智能出行目标的又一进展。LHI新版本采用汽车到云端的应用程序,这种应用程序基于车轮打滑来估算摩擦系数。因此LHI新版本可以检测到路面附着力的微小变化,并将数据上传到云端进行处理,随后向驾驶员发出实时提醒,如前方路面可能有结冰或湿滑状况。 2017年起,奥迪汽车就可在发生事故、故障、交通拥堵、路面结冰以及能见度不佳时向其他车辆发出提醒,这种技术被称为car-to-X。该技术可为LHI分析数据,如电子控制系统、雨量传感器、光线传感器、雨刮器、前大灯、紧急呼叫系统和安全气囊等
[汽车电子]
奥迪使用高精度<font color='red'>数据</font>向驾驶员<font color='red'>发送</font>实时路面信息
DW1000开发笔记(四)DW1000使用轮询方式发送数据
一、移植官方示例 1. 复制官方示例文件 将官方驱动库中example下的第一个示例移植过来: 复制到之前移植的STM32CubeMX生成的工程文件中,并重命名文件为simple_tx_example.c: 将其添加到MDK工程中: 2. 修改官方示例文件 ① 修改替换头文件: ② 修改函数名,修改打印和延时函数: 添加两处打印信息: 修改完成。 3. 调用示例代码 修改main.c,移除我们之前自己添加的所有测试代码。 接着先引入外部定义: /* USER CODE BEGIN PFP */ extern int example_application_entry(void); /* USER COD
[单片机]
DW1000开发笔记(四)DW1000使用轮询方式<font color='red'>发送</font><font color='red'>数据</font>
51单片机:串口向电脑发送数据
单片机的串口配置可以使用SCT-ISP软件,方法如下,我的单片机晶振频率是11.0592Hz 试验程序如下: #include regx52.h #include intrins.h unsigned int num; //建立全局变量 void Delay1000ms() //@11.0592MHz { unsigned char i, j, k; _nop_(); i = 8; j = 1; k = 243; do { do { while (--k); } while (--j); } while (--i); } void
[单片机]
51单片机:串口向电脑<font color='red'>发送</font><font color='red'>数据</font>
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved