当前位置:首页 > 单片机 > 单片机
[导读]/******************************************************************************************************************************************///总结: SCL为高电平时, SDA上的数据才有效// 传输数据 SCL = 1;

/*
********************************************************************
********************************************************************
*/

//总结: SCL为高电平时, SDA上的数据才有效
// 传输数据 SCL = 1; SDA = 1 或 SDA = 0 要求是稳定的数据
// 启动信号 SCL = 1; SDA = 1 ---> 0
// 停止信号 SCL = 1; SDA = 0 ---> 1
// 主要用proteus调试查看每次总线上的数据
// 主要调试i2cstar() 以及i2cstop()这两个函数, 只要这两个函数正确,一般就正确了。

//
//关于地址的说明
// 无论是写入还是读出,被操作器件容量在256字节以内时,一个字节(8位)的寻址范围即可满足要求。当容量为512字节
// (2页). 1k字节(4页)和2k字节(8页)时,采用占用器件引脚地址(A2,A1,A0)的办法,将引脚地址做为页地址(占用的引脚地址线悬空).
// 当容量在4K字节以上时,存储单元地址字节将用两个字节表示(高8位在前)
//


/* 应用例子
void main( void )
{
uchar ReadBuf[6];
LCD_init();
LCD_write_string( 1, 3, " I2C test " ); //测度液晶


I2cWriteDataToAddr( 0xa0, 0x0007, 'h' ); //在器件地址为0xa0的IIC器件上的0x0007这个地址写入'h'这个字符
I2cWriteDataToAddr( 0xa0, 0x0008, 'e' );
I2cWriteDataToAddr( 0xae, 0x0007, 'b' );
I2cWriteDataToAddr( 0xae, 0x0008, 'e' );
I2cWriteDataToAddr( 0xae, 0x0009, 'i' ); //器件地址 0xae, 字节地址 0x0009, 写入 'i'

LCD_write_char( 2, 1, I2cReadDataFromAddr( 0xa0, 0x0007 ) ); //把器件地址为0xa0的0x0007这个地址上的字节数据读出来
LCD_write_char( 2, 2, I2cReadDataFromAddr( 0xa0, 0x0008 ) );
LCD_write_char( 2, 3, I2cReadDataFromAddr( 0xae, 0x0007 ) );
LCD_write_char( 2, 4, I2cReadDataFromAddr( 0xae, 0x0008 ) );
LCD_write_char( 2, 5, I2cReadDataFromAddr( 0xae, 0x0009 ) );

I2c_Write_n( 0xa0, 0x0001, "0123456789", 10 ); //连续写10个字节
LCD_write_char( 2, 0, I2cReadDataFromAddr( 0xa0, 0x0001 ) );
LCD_write_char( 2, 1, I2cReadDataFromAddr( 0xa0, 0x0002 ) );
LCD_write_char( 2, 2, I2cReadDataFromAddr( 0xa0, 0x0003 ) );
LCD_write_char( 2, 3, I2cReadDataFromAddr( 0xa0, 0x0004 ) );
LCD_write_char( 2, 4, I2cReadDataFromAddr( 0xa0, 0x0005 ) );
LCD_write_char( 2, 5, I2cReadDataFromAddr( 0xa0, 0x0006 ) );
LCD_write_char( 2, 6, I2cReadDataFromAddr( 0xa0, 0x0007 ) );
LCD_write_char( 2, 7, I2cReadDataFromAddr( 0xa0, 0x0008 ) );
LCD_write_char( 2, 8, I2cReadDataFromAddr( 0xa0, 0x0009 ) );
LCD_write_char( 2, 9, I2cReadDataFromAddr( 0xa0, 0x000a ) );

I2c_Read_n( 0xa0, 0x0002, ReadBuf, 5 ); //连续读5个字节
LCD_write_array( 1, 4, ReadBuf );

while( 1 );
}
*/


#ifndef _24C64_H_
#define _24C64_H_

#include
#include

//数据类型说明
#define uchar unsigned char
#define uint unsigned int


//5us延时宏定义
#define NOP_5 _nop_(); _nop_(); _nop_(); _nop_(); _nop_();


//管脚连接信息
#define SCL P3_6
#define SDA P3_7

//定义读出数据缓冲区
//#define I2CSIZE 16 //定义16个字节 尽量不要太大,节省空间
//uchar xdata I2cBuffer[I2CSIZE];

//注意,在写函数中加入了5ms延时,如果晶振有变,则适当修改。
void delay_5ms();

//IIC函数
void I2cStart( void );//启动
void I2cStop( void );//终止
uchar WaitAsk( void );//等待应答
void SendAsk( void );//发送应答
void SendNoAsk( void );//发送非应答
void I2cWriteByte( uchar wbyte );//写字节
void I2cWriteDataToAddr( uchar DeviceAddress, uint ByteAddress, uchar Wdata );//写字节到某地址
uchar I2cReadByte( void );//读字节
uchar I2cReadDataFromAddr( uchar DeviceAddress, uint ByteAddress );//从某器件读字节
void I2c_Write_n( uchar DeviceAddress, uint ByteAddress, uchar *Wdata, uchar n );//写n个
void I2c_Read_n( uchar DeviceAddress, uint ByteAddress, uchar *rdatabuf, uchar n );//读n个

#endif // <24c64.h>


#define _24c64_c_
#include "24c64.h"

/*
********************************************************************
** 函数名:5ms延时函数
** 注意 :
** 说明 :
**
********************************************************************
*/
void delay_5ms( void )
{
uchar i;
uchar j;
uchar t;
for( t = 0; t < 10; t++ )
{
for( j = 0; j < 15; j ++ )
{
for( i = 0; i < 21; i++ )
{;}
}
}
}

/*
********************************************************************
** 函数名:i2c启动
** 注意 :
** 说明 :
********************************************************************
*/
void I2cStart( void )
{
//scl=1时
//sda由1-->0
SDA = 1; //准备下降沿
SCL = 1;
NOP_5;
SDA = 0;
NOP_5;

SCL = 0; //一定要
//SDA = 0;
}

/*
********************************************************************
** 函数名:i2c停止
** 注意 :
** 说明 :
********************************************************************
*/
void I2cStop( void )
{
//scl=1时
//sda由0-->1
SDA = 0; //准备上升沿
SCL = 1;
NOP_5;
SDA = 1;
NOP_5;

//SCL = 0; //本来书上说要此句,但发觉加入后仿真不正确...
//SDA = 0;
}

/*
********************************************************************
** 函数名:查询应答信号
** 注意 :
** 说明 :如果有应答返回 1
********************************************************************
*/
uchar WaitAsk( void )
{
uchar askflag = 0;

SDA = 1; //SDA置为输入

SCL = 1;
NOP_5; //5us后检测

if( SDA == 0 ) //检测sda线
askflag = 1; //有应答返回 1 表示成功

SCL = 0;

return askflag;
}

/*
********************************************************************
** 函数名:发送应答信号
** 注意 :
** 说明 :SCL = 1 , SDA = 0;
********************************************************************
*/
void SendAsk( void )
{
SDA = 0;

SCL = 1;
NOP_5;
SCL = 0; //在scl为高时,sda为0

SDA = 1;
}

/*
********************************************************************
** 函数名:发送非应答信号
** 注意 :
** 说明 :SCL = 1 , SDA = 1
********************************************************************
*/
void SendNoAsk( void )
{
SDA = 1;

SCL = 1;
NOP_5;
SCL = 0; //在scl为高时, sda为1

SDA = 0;
}

/*
********************************************************************
** 函数名 :写一个字节数据
** 入口参数: 字节数据wbyte
** 注意 :
** 说明 :可以用来写地址字节,也可以用来写数据字节
********************************************************************
*/
void I2cWriteByte( uchar wbyte )
{
uchar i;
for(i = 0; i < 8; i++ )
{
if( ( wbyte & 0x80 ) == 0x80 ) //!!!! 先发送高位,再发送低位.., 在数据传输时一定要注意此处细节
SDA = 1;
else
SDA = 0; //因为数据传输时要保持数据稳定, 因此要先准备好SDA上的数据才能进行SCL的变化

SCL = 1;
NOP_5;
SCL = 0;

wbyte = wbyte << 1;
}
}

/*
********************************************************************
** 函数名 :写一个字节数据到某器件某地址
** 入口参数: 器件地址DeviceAddress 字节地址ByteAddress 要写的字节数据Wdata
** 注意 :里面加有5ms延时。
** 说明 :I2cWriteDataToAddr( 0xa0, 0x08, 'a' );
********************************************************************
*/
void I2cWriteDataToAddr( uchar DeviceAddress, uint ByteAddress, uchar Wdata )
{
I2cStart();
I2cWriteByte( DeviceAddress );
WaitAsk(); //等待应答
I2cWriteByte( ByteAddress >> 8 ); //高8位
WaitAsk();
I2cWriteByte( ByteAddress ); //低8位
WaitAsk();
I2cWriteByte( Wdata );
WaitAsk();
I2cStop();

delay_5ms(); //发觉这里要加一小段延时,如果不加,则可以在外面加。
}

/*
********************************************************************
** 函数名 :读一个字节数据
** 入口参数: 无
** 注意 :
** 说明 :
********************************************************************
*/
uchar I2cReadByte( void )
{
uchar rbyte = 0;
uchar i = 0;
for(i = 0; i < 8; i++ )
{
rbyte = rbyte << 1; //非常注意...此语句不放在循环体内最后.

SDA = 1; //SDA为输入

SCL = 1;
NOP_5;

if( SDA == 1 )
rbyte = rbyte | 0x01;

SCL = 0;
}

return rbyte;
}


/*
********************************************************************
** 函数名 :写一个字节数据到某器件某地址
** 入口参数: 器件地址DeviceAddress 字节地址ByteAddress
** 出口参数: 读到的字节数据rdata
** 注意 :
** 说明 :I2cWriteDataToAddr( 0xa0, 0x08, 'a' );
********************************************************************
*/
uchar I2cReadDataFromAddr( uchar DeviceAddress, uint ByteAddress )
{
uchar rdata;

I2cStart();
I2cWriteByte( DeviceAddress );
WaitAsk(); //等待应答
I2cWriteByte( ByteAddress >> 8 ); //高8位
WaitAsk();
I2cWriteByte( ByteAddress ); //低8位
WaitAsk();

I2cStart();
I2cWriteByte( DeviceAddress | 0x01 ); //读
WaitAsk();
rdata = I2cReadByte();
SendNoAsk(); //不发送应答信号
I2cStop();

return rdata;
}

/*
********************************************************************
** 函数名 :连续写字节数据到某器件某地址之后的好几个单元
** 入口参数: 器件地址DeviceAddress 字节地址ByteAddress 要写的字节数据取址*Wdata 字节数据的个数n
** 注意 :里面加有5ms延时。
** 说明 :I2cWriteDataToAddr( 0xa0, 0x08, "hebei is a big pig!", 20 );
********************************************************************
*/
void I2c_Write_n( uchar DeviceAddress, uint ByteAddress, uchar *Wdata, uchar n )
{
uchar i = 0;
I2cStart();
I2cWriteByte( DeviceAddress );
WaitAsk(); //等待应答
I2cWriteByte( ByteAddress >> 8 ); //高8位
WaitAsk();
I2cWriteByte( ByteAddress ); //低8位
WaitAsk();

for( i = 0; i < n; i++ )
{
I2cWriteByte( *Wdata );
WaitAsk();
Wdata++;
}

I2cStop();

delay_5ms(); //发觉这里要加一小段延时,如果不加,则可以在外面加。
}

/*
********************************************************************
** 函数名 :写一个字节数据到某器件某地址
** 入口参数: 器件地址DeviceAddress 字节地址ByteAddress
** 出口参数: 读到的字节数据rdata
** 注意 :
** 说明 :I2c_Read_n( 0xa0, 0x0003, a, 10 ) //uchar a[10];
********************************************************************
*/
void I2c_Read_n( uchar DeviceAddress, uint ByteAddress, uchar *rdatabuf, uchar n )
{
uchar i = 0;

I2cStart(); //启动总线
I2cWriteByte( DeviceAddress );
WaitAsk(); //等待应答
I2cWriteByte( ByteAddress >> 8 ); //高8位
WaitAsk();
I2cWriteByte( ByteAddress ); //低8位
WaitAsk();

I2cStart();//重新启动
I2cWriteByte( DeviceAddress | 0x01 ); //读
WaitAsk();

for( i = 0; i < n; i++ )
{
*rdatabuf = I2cReadByte();
SendAsk(); //连续发送应答信号
rdatabuf++;
}

I2cStop();
}


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭