当前位置:首页 > 单片机 > 单片机
[导读]/******************************************************************************************************************************************///总结: 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();
}


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

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭