51单片机-IIC实验1-AT24C02数据存储(实战1)

发布于:2024-09-18 ⋅ 阅读:(74) ⋅ 点赞:(0)

本实验主要是针对IIC的具体案例进行实战,主要利用支持IIC通信的芯片AT24C02进行与51单片机构成通信。51(AT89C52)本身不带有IIC通信,所以,我们需要给51写一个IIC时序,以便与支持IIC协议的AT24C02数据存储芯片通信。SCL低电平允许数据变化,高电平期间,主机可以读取SDA上的数据。(掉电不丢失)

        本实验是数据存储实验,即,EEPROM的IIC通信。本实验我们会分为两个模块,一个AT24C02模块和IIC模块,也就是两个.文件。其实这个说白了,就是AT24C02,相当于IIC的一个子类。AT24C02继承于IIC。所以在应用层面,我们只对AT24C02操作,如下图画一个图。

话不多说上才艺!

第一步构建IIC基本通信函数6个:(参考上一节IIC时序)

void IIC_Start(void); //起始函数
void IIC_Stop(void);//终止函数
void IIC_SendByte(unsigned char Byte);//发送一个字节
unsigned char IIC_Receive(void); //接收一个字节
void IIC_SendAck(unsigned char AckBit); //发送应答
unsigned char IIC_ReceiveAck(void); //接收应答

第二步编写AT24C02读写操作,根据AT24C02的数据手册中相关写操作时序配置相对应的寄存器(读操作一般都有返回值)(参考上一节IIC时序含AT24C02操作)

//字节写:在WORD ADDRESS处写入数据DATA
void AT24C02_WriteByte(unsigned char WordAddress,Data)
unsigned char AT24C02_ReadByte(unsigned char WordAddress) //字节读

main.c

#include <REGX52.H>
#include "LCD1602.h"
#include "AT24C02.h"
#include "Delay.h"

unsigned char KeyNum;
unsigned int Num;

void main()
{

	LCD_Init();  //初始化LCD1602
	
	LCD_ShowNum(1,1,Num,5);  //初始置位
	
	while(1)
	{	
		KeyNum = key(); //获取键值
		
		if(KeyNum == 1)  //按键1 对数据进行加操作
		{
			Num++;
			LCD_ShowNum(1,9,KeyNum,2);
			LCD_ShowNum(1,1,Num,5);
		}
		
		if(KeyNum == 2)  //按键2 对数据进行减操作
		{
			Num--;
			LCD_ShowNum(1,9,KeyNum,2);
			LCD_ShowNum(1,1,Num,5);
		}
		
		if(KeyNum == 3)  //按键3 对数据写入操作(写入到AT24C02的EEPROM中(0-255字节))
		{
			LCD_ShowNum(1,9,KeyNum,2);
			AT24C02_WriteByte(0,Num%256);
			Delay(5);
			AT24C02_WriteByte(1,Num/256);
			Delay(5);
			
			LCD_ShowString(2,1,"Write OK!");
			Delay(1000);
			LCD_ShowString(2,1,"         ");
		}
		
		if(KeyNum == 4)  //按键4 对数据读出操作,读出写入到AT24C02的EEPROM中数据
		{
			LCD_ShowNum(1,9,KeyNum,2);
			Num = AT24C02_ReadByte(1) << 8;
			Num |= AT24C02_ReadByte(0);
			LCD_ShowNum(1,1,Num,5);
			LCD_ShowString(2,1,"Read OK!");
			Delay(1000);
			LCD_ShowString(2,1,"         ");
			
		}
	
	}
}

AT24C02.c

#include <REGX52.H>
#include "IIC.h"

#define AT24C02_ADDRESS 0XA0 //1010 0000 前四位AT24C02地址不变,最后一位决定是写还是读 1:读,即接收 0:写,即发送


//仿照帧格式去写(可参考上一节IIC时序介绍)
//字节写:在WORD ADDRESS处写入数据DATA
void AT24C02_WriteByte(unsigned char WordAddress,Data)
{

	IIC_Start();  //起始信号
	IIC_SendByte(AT24C02_ADDRESS); //发送从机地址和写操作
	IIC_ReceiveAck();  //接收应答位
	IIC_SendByte(WordAddress); //字地址:指定在WORD ADDRESS处写入数据DATA
	IIC_ReceiveAck();  //接收应答位
	IIC_SendByte(Data); //写入数据到WordAddress中
	IIC_ReceiveAck(); //接收应答位
	IIC_Stop();//结束信号
	
}



//Ack : 0:表示应答 1:表示非应答
//随机读:读出在WORD ADDRESS处的数据DATA
unsigned char AT24C02_ReadByte(unsigned char WordAddress)
{
	unsigned char Data;
	
	//写操作,就是先找到要通信的从机
	IIC_Start();  //起始信号
	IIC_SendByte(AT24C02_ADDRESS); //发送从机地址和写操作
	IIC_ReceiveAck();  //接收应答位
	IIC_SendByte(WordAddress); //字地址:指定在WORD ADDRESS处写入数据DATA
	IIC_ReceiveAck();  //接收应答位
	
	//找到对应的从机之后,开始接收从机发过来的数据
	IIC_Start();  //起始信号
	IIC_SendByte(AT24C02_ADDRESS | 0X01); //发送从机地址和读操作
	IIC_ReceiveAck();  //接收应答位
	Data = IIC_ReceiveByte();  //接收一个字节的数据
	IIC_SendAck(1); 
	IIC_Stop();//结束信号
	
	
	return Data;
}


AT24C02.h

#ifndef __AT24C02_H__
#define __AT24C02_H__

void AT24C02_WriteByte(unsigned char WordAddress,Data); //字节写:在WORD ADDRESS处写入数据DATA
unsigned char AT24C02_ReadByte(unsigned char WordAddress); //随机读:读出在WORD ADDRESS处的数据DATA


#endif

IIC.c

#include <REGX52.H>


//位声明 ,两根线定义在单片机的管脚,也就是说SCL接在P2^1管脚,SDA接在P2^0管脚,
sbit IIC_SCL = P2^1;
sbit IIC_SDA = P2^0;


//IIC的6个基本函数(符合IIC时序的基本操作,软件模拟IIC,让AT24C02继承于IIC,后期直接调用即可,无需关注底层细节,其实这个就是IIC的驱动)

//起始函数
void IIC_Start(void)
{
	
	//起始条件:SCL高电平期间,SDA从高电平切换到低电平
	IIC_SDA = 1;  //先保证SDA处于高电平状态	
	IIC_SCL = 1; 
	IIC_SDA = 0;
	IIC_SCL = 0; 	
	
}

//终止函数
void IIC_Stop(void)
{
	
	
	//终止条件:SCL高电平期间,SDA从低电平切换到高电平
	IIC_SDA = 0; //先保证SDA处于低电平状态
	IIC_SCL = 1; 	
	IIC_SDA = 1;
	IIC_SCL = 0; 	
}

//发送一个字节,发送一个字节:SCL低电平期间,主机将数据位依次放到SDA线上(高位在前),
//然后拉高SCL,从机将在SCL高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,即可发送一个字节(可参考上一节内容)

void IIC_SendByte(unsigned char Byte)
{
	unsigned char i;
	for(i=0; i<8; i++)
	{
		IIC_SDA = Byte & (0X80>>i) ;//将数据位依次放到SDA线上(高位在前)&按位与
		IIC_SCL = 1;
		IIC_SCL = 0;
	}
	
}




//接收一个字节:SCL低电平期间,从机将数据位依次放到SDA线上(高位在前),
//然后拉高SCL,主机将在SCL高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,
//即可接收一个字节(主机在接收之前,需要释放SDA)

//接收一个字节数据
unsigned char IIC_ReceiveByte(void)
{
	unsigned char i, Byte = 0X00; //用于保存接收的字节
	
	IIC_SDA = 1; //主机在接收之前,需要释放SDA,终止对SDA的控制
	
	for(i=0; i<8; i++)
	{
		IIC_SCL = 1; //拉高SCL,主机将在SCL高电平期间读取数据位
		if(IIC_SDA)
		{
			Byte |= (0X80>>i);
		}
		IIC_SCL = 0;  //SCL低电平期间,从机将数据位依次放到SDA线上(高位在前)
	}
	
	
	return Byte;	
}



//发送应答:在接收完一个字节之后,主机在下一个时钟发送一位数据,数据0表示应答,数据1表示非应答 (IIC时序手册)
void IIC_SendAck(unsigned char AckBit)
{
	IIC_SDA = AckBit; //应答位,SCL低电平期间,从机将数据位依次放到SDA线上(高位在前)
	IIC_SCL = 1;  //SCL从高到底,把数据放到SDA上
	IIC_SCL = 0;
	
}


//接收应答:在发送完一个字节之后,主机在下一个时钟接收一位数据,判断从机是否应答,
//数据0表示应答,数据1表示非应答(主机在接收之前,需要释放SDA)

unsigned char IIC_ReceiveAck(void)
{
	unsigned char AckBit;
	IIC_SDA = 1; //主机在接收之前,需要释放SDA
	
	IIC_SCL = 1; //高电平期间,主机可以读取IIC上的数据位
	
	AckBit = IIC_SDA;
	
	IIC_SCL = 0;
	
	return AckBit;
}














IIC.h

#ifndef __IIC_H__
#define __IIC_H__

void IIC_Start(void); //起始函数
void IIC_Stop(void);//终止函数
void IIC_SendByte(unsigned char Byte);//发送一个字节
unsigned char IIC_ReceiveByte(void); //接收一个字节
void IIC_SendAck(unsigned char AckBit); //发送应答
unsigned char IIC_ReceiveAck(void); //接收应答



#endif

LCD1602.c

#include <REGX52.H>

//引脚配置:
sbit LCD_RS=P2^6;
sbit LCD_RW=P2^5;
sbit LCD_EN=P2^7;
#define LCD_DataPort P0

//函数定义:
/**
  * @brief  LCD1602延时函数,12MHz调用可延时1ms
  * @param  无
  * @retval 无
  */
void LCD_Delay()
{
	unsigned char i, j;

	i = 2;
	j = 239;
	do
	{
		while (--j);
	} while (--i);
}

/**
  * @brief  LCD1602写命令
  * @param  Command 要写入的命令
  * @retval 无
  */
void LCD_WriteCommand(unsigned char Command)
{
	LCD_RS=0;
	LCD_RW=0;
	LCD_DataPort=Command;
	LCD_EN=1;
	LCD_Delay();
	LCD_EN=0;
	LCD_Delay();
}

/**
  * @brief  LCD1602写数据
  * @param  Data 要写入的数据
  * @retval 无
  */
void LCD_WriteData(unsigned char Data)
{
	LCD_RS=1;
	LCD_RW=0;
	LCD_DataPort=Data;
	LCD_EN=1;
	LCD_Delay();
	LCD_EN=0;
	LCD_Delay();
}

/**
  * @brief  LCD1602设置光标位置
  * @param  Line 行位置,范围:1~2
  * @param  Column 列位置,范围:1~16
  * @retval 无
  */
void LCD_SetCursor(unsigned char Line,unsigned char Column)
{
	if(Line==1)
	{
		LCD_WriteCommand(0x80|(Column-1));
	}
	else if(Line==2)
	{
		LCD_WriteCommand(0x80|(Column-1+0x40));
	}
}

/**
  * @brief  LCD1602初始化函数
  * @param  无
  * @retval 无
  */
void LCD_Init()
{
	LCD_WriteCommand(0x38);//八位数据接口,两行显示,5*7点阵
	LCD_WriteCommand(0x0c);//显示开,光标关,闪烁关
	LCD_WriteCommand(0x06);//数据读写操作后,光标自动加一,画面不动
	LCD_WriteCommand(0x01);//光标复位,清屏
}

/**
  * @brief  在LCD1602指定位置上显示一个字符
  * @param  Line 行位置,范围:1~2
  * @param  Column 列位置,范围:1~16
  * @param  Char 要显示的字符
  * @retval 无
  */
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char)
{
	LCD_SetCursor(Line,Column);
	LCD_WriteData(Char);
}

/**
  * @brief  在LCD1602指定位置开始显示所给字符串
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  String 要显示的字符串
  * @retval 无
  */
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=0;String[i]!='\0';i++)
	{
		LCD_WriteData(String[i]);
	}
}

/**
  * @brief  返回值=X的Y次方
  */
int LCD_Pow(int X,int Y)
{
	unsigned char i;
	int Result=1;
	for(i=0;i<Y;i++)
	{
		Result*=X;
	}
	return Result;
}

/**
  * @brief  在LCD1602指定位置开始显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~65535
  * @param  Length 要显示数字的长度,范围:1~5
  * @retval 无
  */
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number/LCD_Pow(10,i-1)%10+'0');
	}
}

/**
  * @brief  在LCD1602指定位置开始以有符号十进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:-32768~32767
  * @param  Length 要显示数字的长度,范围:1~5
  * @retval 无
  */
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length)
{
	unsigned char i;
	unsigned int Number1;
	LCD_SetCursor(Line,Column);
	if(Number>=0)
	{
		LCD_WriteData('+');
		Number1=Number;
	}
	else
	{
		LCD_WriteData('-');
		Number1=-Number;
	}
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number1/LCD_Pow(10,i-1)%10+'0');
	}
}

/**
  * @brief  在LCD1602指定位置开始以十六进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~0xFFFF
  * @param  Length 要显示数字的长度,范围:1~4
  * @retval 无
  */
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i,SingleNumber;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		SingleNumber=Number/LCD_Pow(16,i-1)%16;
		if(SingleNumber<10)
		{
			LCD_WriteData(SingleNumber+'0');
		}
		else
		{
			LCD_WriteData(SingleNumber-10+'A');
		}
	}
}

/**
  * @brief  在LCD1602指定位置开始以二进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~1111 1111 1111 1111
  * @param  Length 要显示数字的长度,范围:1~16
  * @retval 无
  */
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number/LCD_Pow(2,i-1)%2+'0');
	}
}

LCD1602.h

#ifndef __LCD1602_H__
#define __LCD1602_H__

//用户调用函数:
void LCD_Init();
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char);
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String);
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length);
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);

#endif

Key.c

#include <REGX52.H>
#include "Delay.h"

/**
  * @brief  获取独立按键键码
  * @param  无
  * @retval 按下按键的键码,范围:0~4,无按键按下时返回值为0
  */
unsigned char Key()
{
	unsigned char KeyNumber=0;
	
	if(P3_1==0){Delay(20);while(P3_1==0);Delay(20);KeyNumber=1;}
	if(P3_0==0){Delay(20);while(P3_0==0);Delay(20);KeyNumber=2;}
	if(P3_2==0){Delay(20);while(P3_2==0);Delay(20);KeyNumber=3;}
	if(P3_3==0){Delay(20);while(P3_3==0);Delay(20);KeyNumber=4;}
	
	return KeyNumber;
}

Key.h

#ifndef __KEY_H__
#define __KEY_H__

unsigned char Key();

#endif

Delay.c

#include "intrins.h"

void Delay(unsigned int xms)		//@11.0592MHz
{
	unsigned char i, j;
	
	while(xms--)
	{
		_nop_();
		i = 2;
		j = 199;
		do
		{
			while (--j);
		} while (--i);
	}
}

Delay.h

#ifndef __DELAY_H__
#define __DELAY_H__



void Delay(unsigned int xms);




#endif