⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 serial.c

📁 拼音输入法用在单片机上.rar
💻 C
字号:
//************************************************************************
//文件原作者:"51单片机世界" 斑竹丁丁(聂小猛)
//由液晶汉显模块demo程序修改而成。
//*************************************************************************
//本函数集来自“51单片机世界”,作者斑竹丁丁(聂小猛)。
//主页地址http://www.mcu51.com
//串口中断服务程序,仅需做简单调用即可完成串口输入输出的处理
//出入均设有缓冲区,大小可任意设置。
//可供使用的函数名:
/*
extern char getbyte(void);//从接收缓冲区取一个byte,如不想等待则在调用前检测inbufsign是否为1。
extern putbyte(char c);//放入一个字节到发送缓冲区
extern putstring(uchar  *puts);//发送一个定义在程序存储区的字符串到串口
extern puthex(uchar c);//发送一个字节的hex码,分成两个字节发。
extern  bit   inbufsign;
#define CR putstring("\r\n")//发送一个回车换行
extern void serial_init (void);
*/

//*************************************************************************
#pragma small
#include <reg51.h>
//#include "w77e58.h"
#define  uchar  unsigned char
#define  uint   unsigned int

#define  ILEN  36 //32                /* size of serial receiving buffer */
idata    uchar inbuf[ILEN];
data     uchar idata *inlast=inbuf;   //最后由中断进入接收缓冲区的字节位置
data     uchar idata *getlast=inbuf;  //最后取走的字节位置
bit      inbufsign;                   //接收缓冲区非空标志   有=1
bit      inbufful;                    //输入缓冲区满标志  满=1
//#define  CR putstring("\r\n")        //CR=回车换行

//*****************************
/*串口2接收一个字节数据
uchar getchar(void)
{
        uchar c;
        RI_1 = 0;
        while(!RI_1);
        c= SBUF1;
        SBUF1 = c;
        RI_1 = 0;
        return c;
}
*/
/********************************************************************
程序功能:扫描4*4小键盘得到key[4][4]中的一个字符                     *
键盘安排:这是一个编辑输入的小键盘                                   *
┌┬──┐┌┬──┐┌┬──┐┌┬──┐                            *
││ 1  │┆│ 2  │┆│ 3  │┆│ 4  │                            *
││,.;?│┆│abc │┆│def │┆│ghi │                            *
│└──┤┆└──┤┆└──┤┆└──┤                            *
│    └────┴────┴────┴──┐                      *
├┬──┐├┬──┐├┬──┐├┬──┐    │                      *
││ 5  │┆│ 6  │┆│ 7  │┆│ 8  │    │                      *
││jkl │┆│mno │┆│pqrs│┆│tuv │    │                      *
│└──┤┆└──┤┆└──┤┆└──┤  │                      *
│     └────┴────┴────┴─┐│                      *
├┬──┐├┬──┐├┬──┐├┬──┐ ││                      *
││ 9  │┆│ 0  │┆│前移│┆│后移│  ││                      *
││WXYZ│┆│空格│┆│(英)│┆│(数)│  ││                      *
│└──┤┆└──┤┆└──┤┆└──┤ ││                      *
│     └────┴────┴────┴┐││                      *
├┬──┐├┬──┐├┬──┐├┬──┐│││                      *
││前页│┆│后页│┆│取消│┆│确定│││└─P1.0 行             *
││(拼)│┆│    │┆│ ← ┆││ ?  ││└──P1.1 行             *
│└──┤┆└──┤┆└──┤┆└──┤└───P1.2 行             *
│     └────┴────┴────┴────P1.3 行             *
│      ┆      ┆      └────────P1.4 列             *
│      │      └─────────────P1.5 列             *
│      └──────────────────P1.6 列             *
└───────────────────────P1.7 列             *
行:      P1.0~P1.3                                                 *
列:      P1.7~P1.4                                                 *
扫描原理:  从P1.7到P1.4逐位清0,然后检测P1.0~P1.3那个被拉低         *
参考例程:  51单片机世界->汉显液晶主页  http://www.hzlcd.com         *
           斑竹丁丁 的 DEMO程序 并得到诸位大侠的指点而作成的        *
********************************************************************/
//
uchar getchar(void)
{
        idata uchar c,c1,i,j;
        idata uchar ch1,ch2,ch3;
        bit   iskey=0;
        idata uchar  t=0;//延时
        uchar code dout[4]={0x7f,0xbf,0xdf,0xef};
        uchar code key[4][4]={{'1','5','9','s'},{'2','6','0','f'},{'3','7','e',0x08},{'4','8','d',0x0d}};
        while(1)
	{
re_scan:        for(i=0;i<4;i++)
                {
	                P1=dout[i];//01111111,0x7f
	                           //10111111,0xbf
	                           //11011111,0xdf
	                           //11101111,0xef
	                c=P1&0x0f;
	                if(c!=0x0f)
	                {
	                	if(!(c&0x01))     j=0;  //00001110
	                	else if(!(c&0x02))j=1;  //00001101
	                	else if(!(c&0x04))j=2;  //00001011
                                else if(!(c&0x08))j=3;  //00000111
				c1=(key[i][j]);
			        iskey=1;
                                if(t==220)
                                        ch1=c1;
                                if(t==230)
                                        ch2=c1;
                                if(t==240)
                                        ch3=c1;
			}
                }
                t++;
		if(t==0xff)
			if(iskey)
                        {
                                if((ch1!=ch2)||(ch2!=ch3)||(ch3!=c1))
                                {
                                        t=0;iskey=0;
                                        goto re_scan;

                                }
                                else
                                        return(c1);
                        }
	}
}

//*****************************
/*放入一个字节到发送缓冲区
putbyte(unsigned char c)
{
        SBUF=c;
        while(!TI);
        TI=0;
}
*/
//******************************
/*放n个字节数据到发送缓冲区
putbytes(uchar *outplace,uint j) 
{ 	int i;
	for(i=0;i<j;i++)
	{
		putbyte(*outplace);
		outplace++;
	}
}
*/
//***************************************
/*发送一个定义在程序存储区的字符串到串口
putstring(uchar  *puts)
{
	for(;*puts!=0;puts++)   //遇到停止符0结束
	putbyte(*puts);
}
*/
//*************************************
/*发送一个字节的hex码,分成两个字节发。
uchar code hex_[]={"0123456789ABCDEF"};
puthex(unsigned char c)
{
	int  ch;
	ch=(c>>4)&0x0f;
	putbyte(hex_[ch]);
	ch=c&0x0f;
	putbyte(hex_[ch]);
}
*/
//*************************************
//从接收缓冲区取一个byte,在调用前检测inbufsign是否为1。
char  getbyte (void)
{
	char idata c ;
	while (!inbufsign);      //缓冲区空等待
 	ES=0;
	c= *getlast;            //取数据
	getlast++;              //最后取走的数据位置加一
	inbufful=0;             //输入缓冲区的满标志清零
	if (getlast==inbuf+ILEN) getlast=inbuf; //地址到顶部回到底部
	if (getlast==inlast) inbufsign=0;       //地址相等置接收缓冲区空空标志,再取数前要检该标志
	ES=1;
	return (c);             //取回数据
}
//*****************************************
//串口中断处理

serial () interrupt 4
{
	if (RI)
	{
		RI = 0;
		if(!inbufful)
		{
			*inlast= SBUF;            //放入数据
			inlast++;                    //最后放入的位置加一
			inbufsign=1;
			if (inlast==inbuf+ILEN) inlast=inbuf;//地址到顶部回到底部
			if (inlast==getlast)	inbufful=1;  //接收缓冲区满置满标志
		}
	}
}

//*****************************
//串口初始化                0xfd=19200,0xfa=9600,0xf4=4800,0xe8=2400,0xd0=1200 110592
void serial_init (void)
{
	//用t1作波特率
	
	SCON  = 0x50;                     // mode 1: 8-bit UART, enable receiver
	TMOD |= 0x20;                      // timer 1 mode 2: 8-Bit reload
	//SCON1 = 0x70;
	//PCON |= 0x80; 
	TH1 = 0xfa;//fa,          //baud*2    /* reload value 19200 baud
	TR1   = 1;                         // timer 1 run
	//用t2作波特率
/*
	T2CON=0x30;
	RCAP2H=0xFF;
	RCAP2L=0xb8;//70=4800,b8=9600,dc=19200,ee=38400,f4=57600,fa=115200
	TR2=1;ET2=1;
	SCON=0x50;//SCON=0xd0;
*/

	ES = 1;
	REN=1;
	EA=1;
	SM2=1;          //SM2=1时收到的第9位为1才置位RI标志

	//TMOD |=0x01;  //th1 auto load 2X8,th0 1X16
	//TH0=31; TL0=0; //X 32 =1S
	//TR0=1; ET0=1;
	//putstring("com is ready!");
}
//*****************

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -