单片机
名称:IIC协议 EEPROM24c02 通过串口通信存数读取数据
内容:此程序用于检测EEPROM性能,测试方法如下:写入24c02一个数据,然后在内存中改变这些数据, 掉电后主内存将失去这些信息,然后从24c02中调入这些数据。看是否与写入的相同。
电脑通过串口发送一个十六进制的数据到单片机,存储进24c02,要求断电重启后在数码管上显示上一次发送的数据。
(本例是1us机器周期,即晶振频率要小于12MHZ)
[cpp] view plain copy
#include
#include
#define _Nop() _nop_() //定义空指令
#define DataPort P0
sbit WEI=P2^7;
sbit DUAN=P2^6;
// 常,变量定义区
unsigned char code dofly_DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,
0x77,0x7c,0x39,0x5e,0x79,0x71,0x40,0x00};// 显示段码值0~F,-,全空
unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码
unsigned char TempData[8];
sbit SDA=P2^1; //模拟I2C数据传送位
sbit SCL=P2^0; //模拟I2C时钟控制位
bit ack; //应答标志位
unsigned char res;
void DelayUs2x(unsigned char t);//函数声明
void DelayMs(unsigned char t);
void Delay(unsigned int t)
{
while(t--);
}
void InitUART(void)
{
SCON=0x50;
TMOD|=0x20;
TH1=0xFD;
TR1=1;
EA=1;
}
void DelayUs2x(unsigned char t)
{
while(--t);
}
void DelayMs(unsigned char t)
{
while(t--)
{
//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);
}
}
void Display(unsigned char FirstBit,unsigned char Num)
{
unsigned char i;
for(i=0;i
DataPort=0; //清空数据,防止有交替重影
DUAN=1; //段锁存
DUAN=0;
DataPort=dofly_WeiMa[i+FirstBit]; //取位码
WEI=1; //位锁存
WEI=0;
DataPort=TempData[i]; //取显示数据,段码
DUAN=1; //段锁存
DUAN=0;
Delay(200); // 扫描间隙延时,时间太长会闪烁,
//太短会造成重影
}
}
/*------------------------------------------------
启动总线
------------------------------------------------*/
void Start_I2c()
{
SDA=1; //发送起始条件的数据信号
_Nop();
SCL=1;
_Nop(); //起始条件建立时间大于4.7us,延时
_Nop();
_Nop();
_Nop();
_Nop();
SDA=0; //发送起始信号
_Nop(); //起始条件锁定时间大于4μ
_Nop();
_Nop();
_Nop();
_Nop();
SCL=0; //钳住I2C总线,准备发送或接收数据
_Nop();
_Nop();
}
/*------------------------------------------------
结束总线
------------------------------------------------*/
void Stop_I2c()
{
SDA=0; //发送结束条件的数据信号
_Nop(); //发送结束条件的时钟信号
SCL=1; //结束条件建立时间大于4μ
_Nop();
_Nop();
_Nop();
_Nop();
_Nop();
SDA=1; //发送I2C总线结束信号
_Nop();
_Nop();
_Nop();
_Nop();
}
void SendByte(unsigned char c)
{
unsigned char BitCnt;
for(BitCnt=0;BitCnt<8;BitCnt++) //要传送的数据长度为8位
{
if((c<
_Nop();
SCL=1; //置时钟线为高,通知被控器开始接收数据位
_Nop();
_Nop(); //保证时钟高电平周期大于4μ
_Nop();
_Nop();
_Nop();
SCL=0;
}
_Nop();
_Nop();
SDA=1; //8位发送完后释放数据线,准备接收应答位
_Nop();
_Nop();
SCL=1;
_Nop();
_Nop();
_Nop();
if(SDA==1)ack=0;
else ack=1; //判断是否接收到应答信号
SCL=0;
_Nop();
_Nop();
}
unsigned char RcvByte()
{
unsigned char retc;
unsigned char BitCnt;
retc=0;
SDA=1; //置数据线为输入方式
for(BitCnt=0;BitCnt<8;BitCnt++)
{
_Nop();
SCL=0; //置时钟线为低,准备接收数据位
_Nop();
_Nop(); //时钟低电平周期大于4.7us
_Nop();
_Nop();
_Nop();
SCL=1; //置时钟线为高使数据线上数据有效
_Nop();
_Nop();
retc=retc<<1;
if(SDA==1)retc=retc+1; //读数据位,接收的数据位放入retc中
_Nop();
_Nop();
}
SCL=0;
_Nop();
_Nop();
return(retc);
}
/*----------------------------------------------------------------
应答子函数
原型: void Ack_I2c(void);
----------------------------------------------------------------*/
void Ack_I2c(void)
{
SDA=0;
_Nop();
_Nop();
_Nop();
SCL=1;
_Nop();
_Nop(); //时钟低电平周期大于4μ
_Nop();
_Nop();
_Nop();
SCL=0; //清时钟线,钳住I2C总线以便继续接收
_Nop();
_Nop();
}
/*----------------------------------------------------------------
非应答子函数
原型: void NoAck_I2c(void);
----------------------------------------------------------------*/
void NoAck_I2c(void)
{
SDA=1;
_Nop();
_Nop();
_Nop();
SCL=1;
_Nop();
_Nop(); //时钟低电平周期大于4μ
_Nop();
_Nop();
_Nop();
SCL=0; //清时钟线,钳住I2C总线以便继续接收
_Nop();
_Nop();
}
/*----------------------------------------------------------------
向无子地址器件发送字节数据函数
函数原型: bit ISendByte(unsigned char sla,ucahr c);
功能: 从启动总线到发送地址,数据,结束总线的全过程,从器件地址sla.
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit ISendByte(unsigned char sla,unsigned char c)
{
Start_I2c(); //启动总线
SendByte(sla); //发送器件地址
if(ack==0)return(0);
SendByte(c); //发送数据
if(ack==0)return(0);
Stop_I2c(); //结束总线
return(1);
}
*/
/*----------------------------------------------------------------
向有子地址器件发送多字节数据函数
函数原型: bit ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);
功能: 从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------*/
bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
unsigned char i;
Start_I2c(); //启动总线
SendByte(sla); //发送器件地址
if(ack==0)return(0);
SendByte(suba); //发送器件子地址
if(ack==0)return(0);
for(i=0;i
SendByte(*s); //发送数据
if(ack==0)return(0);
s++;
}
Stop_I2c(); //结束总线
return(1);
}
/*----------------------------------------------------------------
向无子地址器件读字节数据函数
函数原型: bit IRcvByte(unsigned char sla,ucahr *c);
功能: 从启动总线到发送地址,读数据,结束总线的全过程,从器件地
址sla,返回值在c.
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit IRcvByte(unsigned char sla,unsigned char *c)
{
Start_I2c(); //启动总线
SendByte(sla+1); //发送器件地址
if(ack==0)return(0);
*c=RcvByte(); //读取数据
NoAck_I2c(); //发送非就答位
Stop_I2c(); //结束总线
return(1);
}
*/
/*----------------------------------------------------------------
向有子地址器件读取多字节数据函数
函数原型: bit ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);
功能: 从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
如果返回1表示操作成功,否则操作有误。
注意: 使用前必须已结束总线。
----------------------------------------------------------------*/
bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
unsigned char i;
Start_I2c(); //启动总线
SendByte(sla); //发送器件地址
if(ack==0)return(0);
SendByte(suba); //发送器件子地址
if(ack==0)return(0);
Start_I2c();
SendByte(sla+1);
if(ack==0)return(0);
for(i=0;i
*s=RcvByte(); //发送数据
Ack_I2c(); //发送就答位
s++;
}
*s=RcvByte();
NoAck_I2c(); //发送非应位
Stop_I2c(); //结束总线
return(1);
}
/*------------------------------------------------
主函数
------------------------------------------------*/
void main()
{
unsigned char doflye; // 定义临时变量
unsigned char i;
IRcvStr(0xae,4,&doflye,1); //调用存储数据
TempData[0]=dofly_DuanMa[doflye/16];
TempData[1]=dofly_DuanMa[doflye%16];
InitUART();
ES=1;
while(1)
{
Display(0,2);
doflye=res;
ISendStr(0xae,4,&doflye,1); //写入24c02
}
}
void UART_SER(void) interrupt 4
{
unsigned char Temp;
// unsigned char i;
if(RI)
{
RI=0;
Temp=SBUF;
res=Temp;
TempData[0]=dofly_DuanMa[Temp/16];
TempData[1]=dofly_DuanMa[Temp%16];
}
if(TI)
TI=0;
}
文章来源:NK_test的博客
实验一 基本I/O口试验:点亮二极管
1、 试验现象:
8个二极管间隔发光。
2、 试验目的:
了解最简单的单片机程序的编写方法,了解单片机I/O口驱动二极管的方法
3、 试验任务分析:
要想让二极管按照我们的要求发光,首先要搞清楚电路的连接形式,我们先只看和这部分内容有关的电路。当JMP0跳线接在12位置时(选通二极管显示),电路如下图所示:
下面,分别把单片机各管脚功能作一简单解释
:
XTAL1和XTAL2端:
由于单片机是一种时序电路,工作的时候必须外加时钟周期,没有时钟周期,就不能执行程序代码,单片机就不能工作。
XTAL1和XTAL2即为外接时钟引脚。时钟的产生有两种方式,内部方式产生和外部方式产生。该电路是内部方式产生时钟的典型电路,内部时钟的晶振频率一般是4M~12M之间,学习板上用的是12M的晶振,外接两个谐振电容。该电容的典型值是30pf。
RST端:
RST是复位端,简单的说,单片机的复位和计算机的重启动是一样的概念。如何进行复位呢?只要在RST端加上高电平就可以了。
图示电路也是一个典型的复位电路,加电瞬间,电容两端相当于短路,RST端产生一个高电平,使得单片机复位。然后随着电容充电,RST电压慢慢下降,当降低到低电平时,单片机开始正常工作。同样,在按下按键时,RST也产生一个高电平,单片机也被复位。下载程序的时候,应该拔下RST跳线,即可断开复位电路,避免串入干扰信号。
EA/VPP:
EA端是内部程序存储器和外部程序存储器的选择端,当EA=1时,访问片内程序存储器,当EA=0时,访问片外程序存储器。对于我们的学习板来说,由于AT89S52自带8k的程序存储器,没有扩展外部程序存储器,所以应该接高电平。
VPP是引脚的第二功能,暂且不介绍了。
ALE:
地址锁存允许信号。在扩展了外部程序存储器的情况下,当单片机访问外部程序存储器时,ALE输出低8位地址的锁存信号。在本学习板上没有用到这个端子.
PROG为引脚的第二功能,暂且不介绍了,有兴趣的同学可以查询相关教材。
PSEN:
外部程序存储器读选通信号,由于在学习板上没有扩展外部程序存储器,所以这个脚也不用。
P1口:
P1口可作为通用的I/O口使用。在本电路中,P1.5口外接蜂鸣器(其余几个和本试验无关,暂不介绍)。上图可知,当P1.5输出高电平时,对应的三极管导通,蜂鸣器发声。(同学可能会问,这个功能好像和我们的试验任务没有什么关系啊,我一会在给大家解释)。
P3口:
P3口是双功能口,第一种功能和我们P1口类似,也可以作通用的I/O口使用。第二种功能和单片机的串行通信,中断等功能有关,我们暂且不介绍。以后用到相应功能在给大家介绍。
P2口:
P2也可以作为输入口或者输出口来用,在试验板上,P2口的作用在于选通数码管显示。在本例中我们不用。
P0口:
P0口在我们这个试验中扮演着重要的角色,从图上可知,发光二极管是由P0口驱动的。且慢,大家可能会发现,P0口是通过74AS244驱动发光二极管的,这是为什么呢?在这里,74AS244是个缓冲器,它的作用在于隔离单片机和外围电路,这样可以保护单片机,并且能够增强单片机的输出驱动能力。在该电路中,如果我们去掉74AS244,而直接把二极管接在P0输出口上,也是可以的,这是因为电源通过上拉电阻能够提供较大的驱动电流。
同时大家要注意,当P0口作为输出口使用的时候,它的输出级是漏级开路的形式,所以它应该外接上拉电阻,这时才能有高电平输出。(我们的板子上面用了一个排阻。漏级开路的输出级类似于ttl电路中集电极开路的输出级,大家可以参考随便一本数电教材,关于oc门的原理介绍,上面说得非常清楚。)
下面我们来看看怎样才能使得P0口驱动的8个二极管按照要求发光。我们发现,只要P0口相应的一个引脚输出低电平的时候,则对应的二极管发光。例如:欲LED1发光,只需要P0.0引脚输出相应的低电平就可以啦!
因此,如果我们需要8个二极管间隔发光,在板子上,从左至右依次为亮灭,则P0口的输出值应该是:01010101,即为55H。
好啦,分析清楚之后,我们可以开始写程序啦!
4、试验程序如下:
org 0000h ;(1)
clr p1.5 ;(2)
loop: mov p0,#55h ;(3)
ajmp loop ;(4)
end ;(5)
注释
(1)org是个伪指令,也就是说它在汇编时不产生目标代码。(大家可以在medwin环境里打开反汇编窗口看看就明白啦。)它一般出现在每段源程序或者数据块的开始,指明此语句后面的程序或者数据块的起始地址。我们编写好的程序是存放在单片机的程序存储器中的,它的可寻址空间是64k,即0000h~0ffffh。这个语句表示我们的程序从程序存储器地址为0000h单元开始存放。
(2)从电路图上面可以看到P1.5连接蜂鸣器,所以这个语句的意思是,把P1.5置零,不让蜂鸣器响。由于单片机复位以后,P1口的内容为FFH,这样蜂鸣器就会一直响。所以要把该端子置零。当然,如果你不怕吵,这句话也可以不写。
(3)给P0口赋值55H,使得8个二极管间隔发光。
(4)跳转回标记为loop的指令。(注ljmp指令也具有同样的功能,两者区别是:ajmp只能在2k字节内转移,而ljmp可以在64k字节内转移。本程序用ajmp就足够了。)
(5)这同样也是一条伪指令,告诉我们程序到这里就结束啦。
好啦,然后把这个程序进行编译,下载,你就会看到学习板上的发光二极管乖乖的按照你的指令工作了。
5、 课后练习
(1)、学习该程序中出现的知识点涉及到的理论知识,包括I/O口,程序存储器,数据存储器,和使用到的指令。
(2)、编写程序,使得发光二极管从左到右四个亮,四个不亮
来源:玩转单片机