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

📄 7290t.h

📁 C8051F020串口与计算机通信的编程
💻 H
字号:
#ifndef zlg7290t
#define zlg7290t

#include "c8051f020.h"
#define uchar unsigned char 
#define uint  unsigned int

//void delay(uchar t);
//bit ZLG7290_Download(char addr, bit dp, bit flash, char dat);
//uchar get_key();				//返回键值
//void display(uchar hex);			//显示一位16进制

//内部晶振2M,不能太大,若要将频率提高,则需加大I2C_DELAY_VALUE的值
/*I2C*/
//模拟I2C总线的引脚定义
sbit I2C_SCL = P3^4;
sbit I2C_SDA = P3^5;
bit KEYINT;			//按键标志位,1为有键按下,需用软件清0


//定义I2C总线时钟的延时值,要根据实际情况修改,取值1~255
//SCL信号周期约为(I2C_DELAY_VALUE*4+15)个机器周期
#define I2C_DELAY_VALUE		255

//定义I2C总线停止后在下一次开始之前的等待时间,取值1~65535
//等待时间约为(I2C_STOP_WAIT_VALUE*8)个机器周期
//对于多数器件取值为1即可;但对于某些器件来说,较长的延时是必须的
#define I2C_STOP_WAIT_VALUE	120

//定义ZLG7290在I2C总线协议中的从机地址
//这是7位纯地址,不含读写位
#define ZLG7290_I2C_ADDR	0x38

//定义ZLG7290内部寄存器地址(子地址)
#define ZLG7290_SystemReg		0x00		//系统寄存器
#define ZLG7290_Key			    0x01		//键值寄存器
#define ZLG7290_RepeatCnt		0x02		//连击次数寄存器
#define ZLG7290_FunctionKey		0x03		//功能键寄存器
#define ZLG7290_CmdBuf			0x07		//命令缓冲区起始地址
#define ZLG7290_CmdBuf0			0x07		//命令缓冲区0
#define ZLG7290_CmdBuf1			0x08		//命令缓冲区1
#define ZLG7290_FlashOnOff		0x0C		//闪烁控制寄存器
#define ZLG7290_ScanNum			0x0D		//扫描位数寄存器
#define ZLG7290_DpRam			0x10		//显示缓存起始地址
#define ZLG7290_DpRam0			0x10		//显示缓存0
#define ZLG7290_DpRam1			0x11		//显示缓存1
#define ZLG7290_DpRam2			0x12		//显示缓存2
#define ZLG7290_DpRam3			0x13		//显示缓存3
#define ZLG7290_DpRam4			0x14		//显示缓存4
#define ZLG7290_DpRam5			0x15		//显示缓存5
#define ZLG7290_DpRam6			0x16		//显示缓存6
#define ZLG7290_DpRam7			0x17		//显示缓存7
#define ZLG7290_Flash(Fn) 	ZLG7290_cmd(0x70,(Fn))

unsigned char data I2C_Delay_t;

/*
宏定义:I2C_Delay()
功能:延时,模拟I2C总线专用
*/
 I2C_Delay()
{
	I2C_Delay_t = (I2C_DELAY_VALUE);
	while ( --I2C_Delay_t != 0 );
}

/*
函数:I2C_Init()
功能:I2C总线初始化,使总线处于空闲状态
说明:在main()函数的开始处,通常应当要执行一次本函数
*/
void I2C_Init()
{
	I2C_SCL = 1;
	I2C_Delay();
	I2C_SDA = 1;
	I2C_Delay();
}

/*
函数:I2C_Start()
功能:产生I2C总线的起始状态
说明:
	SCL处于高电平期间,当SDA出现下降沿时启动I2C总线
	不论SDA和SCL处于什么电平状态,本函数总能正确产生起始状态
	本函数也可以用来产生重复起始状态
	本函数执行后,I2C总线处于忙状态
*/
void I2C_Start()
{
	I2C_SDA = 1;
	I2C_Delay();
	I2C_SCL = 1;
	I2C_Delay();
	I2C_SDA = 0;
	I2C_Delay();
	I2C_SCL = 0;
	I2C_Delay();
}

/*
函数:I2C_Write()
功能:向I2C总线写1个字节的数据
参数:
	dat:要写到总线上的数据
*/
void I2C_Write(char dat)
{
	unsigned char t = 8;
	do
	{
		I2C_SDA = (bit)(dat & 0x80);
		dat <<= 1;
		I2C_SCL = 1;
		I2C_Delay();
		I2C_SCL = 0;
		I2C_Delay();
	} while ( --t != 0 );
}

/*
函数:I2C_Read()
功能:从从机读取1个字节的数据
返回:读取的一个字节数据
*/
char I2C_Read()
{
	char dat;
	unsigned char t = 8;
	I2C_SDA = 1;	//在读取数据之前,要把SDA拉高
	do
	{
		I2C_SCL = 1;
		I2C_Delay();
		dat <<= 1;
		if ( I2C_SDA ) dat |= 0x01;
		I2C_SCL = 0;
		I2C_Delay();
	} while ( --t != 0 );
	return dat;
}

/*
函数:I2C_GetAck()
功能:读取从机应答位
返回:
	0:从机应答
	1:从机非应答
说明:
	从机在收到每个字节的数据后,要产生应答位
	从机在收到最后1个字节的数据后,一般要产生非应答位
*/
bit I2C_GetAck()
{
	bit ack;
	I2C_SDA = 1;
	I2C_Delay();
	I2C_SCL = 1;
	I2C_Delay();
	ack = I2C_SDA;
	I2C_SCL = 0;
	I2C_Delay();
	return ack;
}

/*
函数:I2C_PutAck()
功能:主机产生应答位或非应答位
参数:
	ack=0:主机产生应答位
	ack=1:主机产生非应答位
说明:
	主机在接收完每一个字节的数据后,都应当产生应答位
	主机在接收完最后一个字节的数据后,应当产生非应答位
*/
void I2C_PutAck(bit ack)
{
	I2C_SDA = ack;
	I2C_Delay();
	I2C_SCL = 1;
	I2C_Delay();
	I2C_SCL = 0;
	I2C_Delay();
}

/*
函数:I2C_Stop()
功能:产生I2C总线的停止状态
说明:
	SCL处于高电平期间,当SDA出现上升沿时停止I2C总线
	不论SDA和SCL处于什么电平状态,本函数总能正确产生停止状态
	本函数执行后,I2C总线处于空闲状态
*/
void I2C_Stop()
{
	unsigned int t = I2C_STOP_WAIT_VALUE;
	I2C_SDA = 0;
	I2C_Delay();
	I2C_SCL = 1;
	I2C_Delay();
	I2C_SDA = 1;
	I2C_Delay();
	while ( --t != 0 );		//在下一次产生Start之前,要加一定的延时
}

/*
函数:I2C_Puts()
功能:I2C总线综合发送函数,向从机发送多个字节的数据
参数:
	SlaveAddr:从机地址(7位纯地址,不含读写位)
	SubAddr:从机的子地址
	SubMod:子地址模式,0-无子地址,1-单字节子地址,2-双字节子地址
	*dat:要发送的数据
	Size:数据的字节数
返回:
	0:发送成功
	1:在发送过程中出现异常
说明:
	本函数能够很好地适应所有常见的I2C器件,不论其是否有子地址
	当从机没有子地址时,参数SubAddr任意,而SubMod应当为0
*/

bit I2C_Puts
(
	unsigned char SlaveAddr,
	unsigned int SubAddr,
	unsigned char SubMod,
	char *dat,
	unsigned int Size
)
{
//定义临时变量
	unsigned char i;
	char a[3];
//检查长度
	if ( Size == 0 ) return 0;
//准备从机地址
	a[0] = (SlaveAddr << 1);
//检查子地址模式
	if ( SubMod > 2 ) SubMod = 2;
//确定子地址
	switch ( SubMod )
	{
	case 0:
		break;
	case 1:
		a[1] = (char)(SubAddr);
		break;
	case 2:
		a[1] = (char)(SubAddr >> 8);
		a[2] = (char)(SubAddr);
		break;
	default:
		break;
	}
//发送从机地址,接着发送子地址(如果有子地址的话)
	SubMod++;
	I2C_Start();
	for ( i=0; i<SubMod; i++ )
	{
		I2C_Write(a[i]);
		if ( I2C_GetAck() )
		{
			I2C_Stop();
			return 1;
		}
	}
//发送数据
	do
	{
		I2C_Write(*dat++);
		if ( I2C_GetAck() ) break;
	} while ( --Size != 0 );
//发送完毕,停止I2C总线,并返回结果
	I2C_Stop();
	if ( Size == 0 )
	{
		return 0;
	}
	else
	{
		return 1;
	}
}



/*
函数:I2C_Gets()
功能:I2C总线综合接收函数,从从机接收多个字节的数据
参数:
	SlaveAddr:从机地址(7位纯地址,不含读写位)
	SubAddr:从机的子地址
	SubMod:子地址模式,0-无子地址,1-单字节子地址,2-双字节子地址
	*dat:保存接收到的数据
	Size:数据的字节数
返回:
	0:接收成功
	1:在接收过程中出现异常
说明:
	本函数能够很好地适应所有常见的I2C器件,不论其是否有子地址
	当从机没有子地址时,参数SubAddr任意,而SubMod应当为0
*/

bit I2C_Gets
(
	unsigned char SlaveAddr,
	unsigned int SubAddr,
	unsigned char SubMod,
	char *dat,
	unsigned int Size
)
{
//定义临时变量
	unsigned char i;
	char a[3];
//检查长度
	if ( Size == 0 ) return 0;
//准备从机地址
	a[0] = (SlaveAddr << 1);
//检查子地址模式
	if ( SubMod > 2 ) SubMod = 2;
//如果是有子地址的从机,则要先发送从机地址和子地址
	if ( SubMod != 0 )
	{
	//确定子地址
		if ( SubMod == 1 )
		{
			a[1] = (char)(SubAddr);
		}
		else
		{
			a[1] = (char)(SubAddr >> 8);
			a[2] = (char)(SubAddr);
		}
	//发送从机地址,接着发送子地址
		SubMod++;
		I2C_Start();
		for ( i=0; i<SubMod; i++ )
		{
			I2C_Write(a[i]);
			if ( I2C_GetAck() )
			{
				I2C_Stop();
				return 1;
			}
		}
	}
//这里的I2C_Start()对于有子地址的从机是重复起始状态
//对于无子地址的从机则是正常的起始状态
	I2C_Start();
//发送从机地址
	I2C_Write(a[0]+1);
	if ( I2C_GetAck() )
	{
		I2C_Stop();
		return 1;
	}
//接收数据
	for (;;)
	{
		*dat++ = I2C_Read();
		if ( --Size == 0 )
		{
			I2C_PutAck(1);
			break;
		}
		I2C_PutAck(0);
	}
//接收完毕,停止I2C总线,并返回结果
	I2C_Stop();
	return 0;
}



/*7290part*/
void delay(uchar t)
{	uchar i,j;
	for(i=0;i<t;i++)
	for(j=0;j<255;j++);
}


/*
函数:ZLG7290_WriteReg()
功能:向ZLG7290 的某个内部寄存器写入数据
参数:
RegAddr:ZLG7290 的内部寄存器地址
dat:要写入的数据
返回:
0:正常
1:访问ZLG7290 时出现异常

bit ZLG7290_WriteReg(unsigned char RegAddr, char dat)
{
bit b;
b = I2C_Puts(ZLG7290_I2C_ADDR,RegAddr,1,&dat,1);
return b;
}
  */
/*
函数:ZLG7290_ReadReg()
功能:从ZLG7290 的某个内部寄存器读出数据
参数:
RegAddr:ZLG7290 的内部寄存器地址
*dat:保存读出的数据
返回:
0:正常
1:访问ZLG7290 时出现异常
*/
bit ZLG7290_ReadReg(unsigned char RegAddr, char *dat)
{
bit b;
I2C_Init();
b = I2C_Gets(ZLG7290_I2C_ADDR,RegAddr,1,dat,1);
return b;
}

/*
函数:ZLG7290_cmd()
功能:向ZLG7290发送控制命令
参数:
	cmd0:写入CmdBuf0寄存器的命令字(第1字节)
	cmd1:写入CmdBuf1寄存器的命令字(第2字节)
返回:
	0:正常
	1:访问ZLG7290时出现异常
*/

bit ZLG7290_cmd(char cmd0, char cmd1)
{
	bit b;
	char buf[2];
	buf[0] = cmd0;
	buf[1] = cmd1;
	I2C_Init();
	b = I2C_Puts(ZLG7290_I2C_ADDR,ZLG7290_CmdBuf,1,buf,2);
	return b;
}


/*
函数:ZLG7290_Download()
功能:下载数据并译码
参数:
	addr:取值0~7,显示缓存DpRam0~DpRam7的编号
	dp:是否点亮该位的小数点,0-熄灭,1-点亮
	flash:控制该位是否闪烁,0-不闪烁,1-闪烁
	dat:取值0~31,表示要显示的数据
返回:
	0:正常
	1:访问ZLG7290时出现异常
说明:
	显示数据具体的译码方式请参见ZLG7290的数据手册
*/

bit ZLG7290_Download(char addr, bit dp, bit flash, char dat)
{
	char cmd0;
	char cmd1;
	cmd0 = addr & 0x0F;
	cmd0 |= 0x60;
	cmd1 = dat & 0x1F;
	if ( dp ) cmd1 |= 0x80;
	if ( flash ) cmd1 |= 0x40;
	return ZLG7290_cmd(cmd0,cmd1);
}


uchar get_key()				//返回键值
{	unsigned char KeyValue;
	ZLG7290_ReadReg(ZLG7290_Key,&KeyValue);
	delay(50);
	return(KeyValue);
}


void displayled(uchar hex)			//显示一位16进制
{	uchar h,l;
	l=hex&0x0F;
	h=hex&0xF0;
	h=h>>4;
	ZLG7290_Download(0x0, 0, 0, l);
	delay(100);
	ZLG7290_Download(0x1, 0, 0, h); 
	delay(100);
}


#endif

⌨️ 快捷键说明

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