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

📄 lesson10.c

📁 基于8051的生产线记件系统
💻 C
📖 第 1 页 / 共 3 页
字号:
/************************************************/
/*单位名称:华南理工大学电信学院*/
/*模块:1位共阴数码管,CD4511驱动,使用反极法的键盘,串行EEPROM24C02
		6116RAM,红外检测头,74LS14施密特触发器,max232*/
/*创建人:景永年   日期:2006年6月11日*/
/*修改人:*/

/*功能描述:统计红外光电断续检测头上0的个数(即生产线上产品的数量),按键发声*/
/*其他说明:*/
/*版本号:*/
/*所有库函数头文件都在keil/C51/INC(lude)目录下,详细说明可以见C51.PDF和C51HLP.CHM。
reg52.h包含了各专用寄存器ACC、P0~P3、TMOD、SCON、THx、TLx、SBUF等还有SFR的可寻址位
string.h包含了字符串操作,本程序使用了字符串比较:memcmp();
intrins.h包含了固有函数,还有空操作:_nop_()
absacc.h可读存储器直接访问,如XBYTE[]

全局变量g_sec_disp、g_min_disp 、g_hr_disp 定义在外部空间0xFFFF、0xEFFF、0xDFFF上,可以直接访问此地址。xdata在64K可寻址范围内定义变量,data在片内256字节内定义变量,pdata在片外256字节内定义变量,范围大寻址慢。
直接存储器访问有两种方法:1._at_2.XBYTE[],XWORD[]

keypad_scan();键盘扫描子程序
keypad_response_dual();双按键响应子程序
keypad_response_single();单按键响应子程序
display_led();显示子程序
delay();延时
time0_init();定时器T0初始化
time1_init();定时器T1初始化
time1_serial_init();T1串行通信初始化*/
/*************************************************/

#include "reg52.h"//reg52.h包含了各专用寄存器ACCP0~P3TMODSCONTHxTLxSBUF等还有SFR的可寻址位
#include "string.h"//string.h包含了字符串操作,本程序使用了字符串比较:memcmp();
#include "intrins.h"//intrins.h包含了固有函数,还有空操作:_nop_()
#include "absacc.h"//可读存储器直接访问,如XBYTE[],
					//所有库函数头文件都在keil/C51/INC(lude)目录下,详细说明可以见C51.PDF和C51HLP.CHM
#define uchar unsigned char
#define uint unsigned int
#define _Nop() _nop_()

uchar volatile xdata g_sec_disp _at_ 0xFFFF;//全局变量g_sec_disp定义在外部空间0xFFFF上,可以直接访问此地址
uchar volatile xdata g_min_disp _at_ 0xEFFF;//xdata在64K可寻址范围内定义变量,data在片内256字节内定义变量
uchar volatile xdata g_hr_disp  _at_ 0xDFFF;//pdata在片外256字节内定义变量,范围大寻址慢。直接存储器访问有两种方法:1._at_2.XBYTE[],XWORD[]

void keypad_scan(void);
void keypad_response_dual(uchar key_number);
void keypad_response_single(uchar key_number);
void display_led(uchar hr, uchar min,uchar sec);
void delay(uchar m, uchar n);
void time0_init(void);
void time1_init(void);
void time1_serial_init(void);//T1串行通信初始化
void serial_interrupt(void);
/************************I2C start***************************************/
sbit SCL = P3^3;//emulate the I2C bus,P3.3用做时钟SCL,P3.4用做数据SDA
sbit SDA = P3^4;
bit ack;//从机应答信号
void Start_I2C(void);
void Stop_I2C(void);
void SendB(unsigned char c);
unsigned char RcvB(void);
void Ack_I2C(bit a);
//bit ISendB(unsigned char sla, unsigned char c);
//bit IRcvB(uchar sla, uchar *c); 
bit ISendStr(unsigned char sla, unsigned char suba, unsigned char * s,unsigned char no);
bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char * s,unsigned char no);
data uchar g_uc_data_send[3],g_uc_data_receive[3];//全局变量g_uc_data_send[3]用于I2C数据发送,g_uc_data_receive[3]I2C数据接收
/**********************I2C End ***************************************/

data uint g_ui_ram_addr;
data uchar g_uc_recv_serial_cnt;

data uchar g_uc_sec_cnt,g_uc_min_cnt,g_uc_hr_cnt;//88 88 88全局变量g_uc_sec_cnt,g_uc_min_cnt,g_uc_hr_cnt分别代表数码显示部分的秒位,分位和时位
struct time
{
	uchar hr;
	uchar min;
	uchar sec;
} g_st_current_time;//全局结构体g_st_current_time,当前时间。调用函数:定时器0中断服务程序
data uchar g_uc_ms_cnt;//全局变量g_uc_ms_cnt定时T0中断的次数,即0.1ms的个数。调用函数:定时器0中断服务程序
data uchar g_uc_key_number,g_uc_disp_cnt;//g_uc_key_number键号,调用函数:keypad_scan_××()。g_uc_disp_cnt显示位置控制符
data uchar g_uc_key_any_cnt,g_uc_key_no_combined,uc_keypad_response_flag;//有按键按下的次数;合成的键值;uc_keypad_response_flag=0,for keypad_response_dual;uc_keypad_response_flag=1,for keypad_response_single.
data uchar g_uc_log_number,g_uc_time_interval;//g_uc_log_number记录在线产量值;g_uc_time_interval第几个时间段(10分钟)
data uint  g_ui_timer1_interval_cnt;//一段(10分钟)的计数器,使用函数:定时器1中断服务程序
code uchar g_uc_key_number_all[12]={0xF3,0xEB,0xDB,0xBB,0xF5,0xED,0xDD,0xBD,0xF6,0xEE,0xDE,0xBE};//g_uc_key_number_all[]数组保存所有反极法键盘对应的键值
sbit beep = P1^7;//sbit定义SFR的某一位,beep驱动蜂鸣器,0电平有效

main()
{		
	g_uc_sec_cnt = 0;
	g_uc_min_cnt = 0;
	g_uc_hr_cnt = 0;
	g_st_current_time.hr = 0;
	g_st_current_time.min = 0;
	g_st_current_time.sec = 0;
	g_uc_disp_cnt = 0;
	g_uc_key_any_cnt = 0;
	uc_keypad_response_flag = 0;
	g_uc_log_number = 0;
	g_uc_time_interval = 0;

	g_uc_recv_serial_cnt = 0;
	g_ui_ram_addr = 0x7800;//first addr

	time0_init();
	time1_init();
	//ET0 = 1;
	//TR0 = 1;
	while(1)
	{
		keypad_scan();
	}	
}
/*************************************************/
/*time0_init*/
/*************************************************/
void time0_init(void)
{
	TMOD = 0x11 ;//定时中断的初始化,定时器0工作在定时器模式,定时器1工作在定时器模式
	IE = 0x80   ;//开总中断;由TR0启停,16bit(模式1)
	TH0 = 0x3C   ;//初值15536
	TL0 = 0xB0   ;//100ms=0.1s	
}
/*************************************************/
/*time1_init*/
/*************************************************/
void time1_init(void)
{
	TH1 = 0x3C   ;//初值15536
	TL1 = 0xB0   ;//100ms=0.1s
}
/*************************************************/
/*time1_init*///设置串行通信的波特率
/*************************************************/
void time1_serial_init(void)
{
	TMOD = 0x20;
	IE = 0x80;
	TH1 = 0xF3;//baud rate:2400
	TL1 = 0xF3;
	SCON = 0x50;//mode 1
	PCON = 0x80;
}
/*************************************************/
/*name:keypad_scan()
function: scan the keypad, get the key number
使用反极法检测键值,P1.7用做蜂鸣器,所以检测列(col)时,送0x87,而没有从0x07。
因为蜂鸣器‘0’有效,’1’无效。
output:uc_key_number */
/*************************************************/
void keypad_scan(void)
{
	P1 = 0XF8;	
	ACC = P1;
	ACC &= 0XF8;
	if(!(ACC == 0XF8))
	{
		delay(2,255);
		P1 = 0XF8;
		ACC = P1;
		ACC &= 0XF8;
		if(!(ACC == 0XF8))/*note:if(ACC != 0XF8) ACC XOR 0xF8;if(!(ACC == 0XF8)) the value of the ACC don't change*/
		{
			g_uc_key_number = ACC;/*get the row number*/

			P1 = 0X87;//P1.7 for beep! '0' beep;'1' not beep.
			ACC = P1;
			ACC &= 0X87;
			if(!(ACC == 0X87))//P1.7 for beep!
			{
				delay(2,255);
				P1 = 0X87;//P1.7 for beep!
				ACC = P1;
				ACC &= 0X87;//P1.7 for beep!
				if(!(ACC == 0X87))//P1.7 for beep!
				{
					g_uc_key_number |= ACC;/*combine to the key number*/
					if(uc_keypad_response_flag == 0)
					{
						keypad_response_dual(g_uc_key_number);
					}
					else
					{
						keypad_response_single(g_uc_key_number);
					}
				}
			}
		}
	}
}
/*************************************************/
/*name:keypad_response_dual()
function:response to the dual key number 
input:uc_key_number*/
/*************************************************/
data uchar uc_temp_receive[7][3];
void keypad_response_dual(uchar key_number)
{
	data uchar k,j;//k for key number;j for rotate
	data uchar uc_device_addr,uc_device_sub_addr;//
	data uchar uc_tmp_password[3];
	data uint temp;//暂存产量总和	

	while(!(ACC == 0x87))//wait for the release of the key//P1.7 for beep!
	{
		delay(2,255);
		P1 = 0x87;
		ACC = P1;
		ACC &= 0X87;
	}
	{		
		uc_keypad_response_flag = 1;//for single key press
		g_uc_key_any_cnt ++;
		for(k=0;k<=11;k++)
		{
			if(key_number == g_uc_key_number_all[k])
			  break;
		}
		if(g_uc_key_any_cnt == 1)
		{
			g_uc_key_no_combined = k << 4;
		}
		else
		{
			g_uc_key_any_cnt = 0;
			g_uc_key_no_combined += k;
			switch(g_uc_key_no_combined)
			{
				case 0xA0:/*A+0*/
						g_uc_disp_cnt = 2;//password 4位
						beep = 0;//提示用户输入密码
						delay(200,200);
						beep = 1;

						while(g_uc_disp_cnt < 6)
						{
							keypad_scan();
							switch(g_uc_disp_cnt)
							{
								case 3://键盘扫描后,g_uc_disp_cnt已经++了
									g_min_disp = 0x8F;
									g_sec_disp = 0xFF;
									break;
								case 4://密码用8888显示为不可见//需要按B确认!?
									g_min_disp = 0x88;
									g_sec_disp = 0xFF;
									break;
								case 5:
									g_sec_disp = 0x8F;
									break;
								case 6:
									g_sec_disp = 0x88;
									break;
								default:
									break;
							}
						}						
						g_uc_disp_cnt = 0;
						uc_device_addr = 0xA0;//0xA0=1010 0000B, 1010is the device number,000 is the A2A1A0 hard wired, 0 write.
						uc_device_sub_addr = 0x00;//EEPROM 首地址
						g_uc_data_send[0] = 0x00;//密码只有4位,g_uc_data_send[0]写00
						g_uc_data_send[1] = g_uc_min_cnt;//存储的数据保存在数组g_uc_data_send中
						g_uc_data_send[2] = g_uc_sec_cnt;						
						ISendStr(uc_device_addr,uc_device_sub_addr,g_uc_data_send,3);//以页方式存储						
					break;
				case 0xAB:/*A+B*/
						g_uc_disp_cnt = 2;//password 4位
						beep = 0;//提示键盘按下
						delay(50,100);
						beep = 1;
						while(g_uc_disp_cnt < 6)
						{
							keypad_scan();
							switch(g_uc_disp_cnt)
							{
								case 3://键盘扫描后,g_uc_disp_cnt已经++了
									g_min_disp = 0x8F;
									g_sec_disp = 0xFF;
									break;
								case 4://密码用8888显示为不可见//需要按B确认!?
									g_min_disp = 0x88;
									g_sec_disp = 0xFF;
									break;
								case 5:
									g_sec_disp = 0x8F;
									break;
								case 6:
									g_sec_disp = 0x88;
									break;
								default:
									break;
							}
						}
						g_uc_disp_cnt = 0;
						uc_tmp_password[0] = 0x00;
						uc_tmp_password[1] = g_uc_min_cnt;
						uc_tmp_password[2] = g_uc_sec_cnt;
						uc_device_addr = 0xA0;//0xA0=1010 0000B, 1010is the device number,000 is the A2A1A0 hard wired, 0 write.
						uc_device_sub_addr = 0x00;//EEPROM 首地址
						IRcvStr(uc_device_addr,uc_device_sub_addr,g_uc_data_receive,3);//以页方式读取//读出的数据保存在数组g_uc_data_receive中
						j = memcmp(uc_tmp_password,g_uc_data_receive,3);//读出的数据保存在数组g_uc_data_send中
						if(j == 0)
						{
							for(j = 1;j < 7;j++)//clear the data
							{
								uc_device_sub_addr = 0x00 + ((j) << 3);//EEPROM 首地址
								g_uc_data_send[0] = 0x00;
								g_uc_data_send[1] = 0x00;
								g_uc_data_send[2] = 0x00;
								ISendStr(uc_device_addr,uc_device_sub_addr,g_uc_data_send,3);//以页方式存储								
							}
							beep = 0;//提示用户密码正确,数据已经清除
							delay(200,200);
							beep = 1;
						}
						else
						{
							beep = 0;
							for(j = 0;j < 8;j++)//断续提示声,提示用户password错误
							{
								delay(50,200);
								if(beep == 0)beep = 1;
								else beep = 0;
							}
						}
					break;
				case 0xA1:
					beep = 0;//提示键盘按下
					delay(50,100);
					beep = 1;
					TR0 = 0;//并不启动
					g_uc_hr_cnt = 0xFF;//清屏
					g_uc_min_cnt = 0xFF;
					g_uc_sec_cnt = 0xFF;
					display_led(g_uc_hr_cnt ,g_uc_min_cnt ,g_uc_sec_cnt);
					g_uc_disp_cnt = 0;//设置时钟,6位
					while(g_uc_disp_cnt < 6)
					{
						keypad_scan();
					}
					g_uc_disp_cnt = 0;
					g_uc_hr_cnt %= 0x24;//adjust the hr
					g_uc_min_cnt %= 0x60;//adjust the min
					g_uc_sec_cnt %= 0x60;//adjust the sec
					display_led(g_uc_hr_cnt ,g_uc_min_cnt ,g_uc_sec_cnt);
					g_st_current_time.hr = g_uc_hr_cnt;
					g_st_current_time.min = g_uc_min_cnt;
					g_st_current_time.sec = g_uc_sec_cnt;
					break;
				case 0xA2:
					ET0 = 1;
					TR0 = 1;//启动T0

⌨️ 快捷键说明

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