关于IIC我觉这个博客里面说的已经够清楚了

如下图所示的写操作的时序图:

其实像这种通信协议的要求是很精确的,一点点不对都可能导致在实际工程中无法读取数据。我就是被一个应答位耽误了好久,还好最后被我发现了。虽然程序不长,但是每一句话都是值得我们认真学习的,下面是我自己结合网上还有书中的程序综合的,亲测可用。最后用keil的逻辑分析仪测试了iic端口输出的波形。

iic.h

#ifndef __IIC_H
#define __IIC_H #include <reg52.h>
#include <intrins.h> sbit SCL = P2^;
sbit SDA = P2^; void iic_init(void);
void iic_start(void);
void iic_stop(void);
void iic_dalay_us(void);//iic延时专用
void iic_send_ack(bit _ack);//读取数据之后发送应答位
void iic_wait_ack(void);//写入数据之后,等待从机应答信号
void iic_write_byte(unsigned char _byte); //写一个字节
unsigned char iic_read_byte(void);//读取一个字节
void iic_write_addr_byte(unsigned char _addr,unsigned char _byte);//任意地址写一个字节
unsigned char iic_read_addr_byte(unsigned char _addr); //任意地址读一个字节 #endif

iic.c

//可以用keil自带的逻辑分析仪查看iic端口的输出波形,通过对波形的分析,判断iic协议有没有工作。
#include "iic.h"
#include "mpu6050.h" //---------5us,实际测试是8us--------
void iic_dalay_us(void)
{
unsigned char i; _nop_();//nop是1个指令周期 = 1机器周期 = 12时钟周期 = 12*1/f,对于12M晶振,1us
i = ;
while(--i); //执行一次while循环2us
} //-------起始信号,时钟高,数据高变低---------------
void iic_start(void)
{
SDA = ;
iic_dalay_us();
SCL = ;
iic_dalay_us();
SDA = ;
iic_dalay_us();
SCL = ; //钳住I2C总线,准备发送或接收数据
} //--------停止信号-------
void iic_stop(void)
{
SDA = ;
iic_dalay_us();
SCL = ;
iic_dalay_us();
SCL = ;
iic_dalay_us();
SDA = ;
iic_dalay_us();
} //-------------
void iic_init(void)
{
SCL = ;
iic_dalay_us();
SDA = ;
iic_dalay_us();
} //----读取数据完成后,发送应答位,
//----0->ack,应答,告诉从机我要继续读取下一个字节,从机收到这个信号后继续发送数据
//----1->not ack,不应答,告诉从机我不在继续接受数据,从机停止发送数据
void iic_send_ack(bit _ack)
{
SDA = _ack;//接收完成后,拉高SDA,发送非应答信号
SCL = ;
iic_dalay_us();
SCL = ; //拉低,完成应答位
iic_dalay_us();
} //-------------主机发送完成,等待应答----------------
//主机发送完一个字节的数据之后会把SDA拉高,等待从机的信号,如果从机不应答,SDA将一直拉高,此时while函数等待250us,
//如果在SCL高电平期间,SDA被从设备拉低表示从设备应答,此时while跳过,应答位结束,
void iic_wait_ack(void)
{
unsigned char i = ; SCL = ;
iic_dalay_us();
while((SDA==)&&(i<))i++;//应答时,SDA为0表示从机成功接受到数据,
SCL = ; //拉低,完成应答位
} //------------------------
//写一个字节,从高位往低位写
//主机给从机发送数据,从机是在时钟的下降沿采集SDA的数据
void iic_write_byte(unsigned char _byte)
{
unsigned char i; for(i=;i<;i++)
{
SDA = _byte&0X80; //先写高位
iic_dalay_us();
SCL = ;
iic_dalay_us();
SCL = ; //下降沿采集数据
iic_dalay_us();
_byte = _byte<<;
} SDA = ;//发送完毕后,释放数据线,检测从机应答
iic_wait_ack();//等待应答
} //----------------------------------
//任意地址写一个字节,因为我使用的时候从设备只有一个,所以不需要写入从设备地址,
//如果有多个从设备的话,可以增加一个参数,写入从设备地址
void iic_write_addr_byte(unsigned char _addr,unsigned char _byte)
{
iic_start();
iic_write_byte(SlaveAddress); //此处写入从机地址,我这里只有一个,没有在函数里面体现
iic_write_byte(_addr);
iic_write_byte(_byte);
iic_stop();
} //------------------------
//读字节,也是先读取的是高位,需要左移
//主机从从机读取数据,本质上是从机检测主机发出的时钟信号有8个上升沿
unsigned char iic_read_byte(void)
{
unsigned char i = ;
unsigned char read_byte = ; SDA = ; //先确保主机释放SDA
iic_dalay_us(); for(i = ;i<;i++)
{
SCL = ;
read_byte = (read_byte<<)|SDA ;
iic_dalay_us();
SCL = ; //拉低,采集数据
iic_dalay_us();
} return read_byte;
} //------------------
//在某一地址内读取数据
unsigned char iic_read_addr_byte(unsigned char _addr)
{
unsigned char read_data; iic_start();
iic_write_byte(SlaveAddress);//发送设备地址+写信号
iic_write_byte(_addr); //发送存储器单元
iic_start(); //必须从新启动IIC
iic_write_byte(SlaveAddress + );//发送设备地址+读信号
read_data = iic_read_byte();//读出数据
iic_send_ack();//发送非应答信号
/*
原来问题出在这里!!,如果后面跟的是iic_stop的话,就发送非应答,告诉从机不要在发数据了,注意这里没有连续读取
不然的话,从机收到0的应答位之后会一直发送数据,但是这时主机又没有接受,导致数据只能第一次发送成功,后面都收不到数据。
*/
iic_stop(); return read_data;
}

keil仿真波形:

51单片机之IIC通信原理及软件仿真的更多相关文章

  1. 4-51单片机WIFI学习(开发板51单片机自动冷启动下载原理)

    上一篇链接 http://www.cnblogs.com/yangfengwu/p/8743936.html 这一篇说一下自己板子的51单片机自动冷启动下载原理,我挥舞着键盘和鼠标,发誓要把世界写个明 ...

  2. 2-51单片机ESP8266学习-AT指令(开发板51单片机自动冷启动下载原理)

    前言:了解就行,不必深究 上一篇链接 http://www.cnblogs.com/yangfengwu/p/8720148.html 源码链接:https://pan.baidu.com/s/1wT ...

  3. ESP8266 AT指令开发(基于STC89C52单片机): 测试下诱人的程序(51单片机,8266,MQTT远程通信控制)

    前言 实现的功能,APP通过SmartConfig给Wi-Fi模块配网并绑定设备,然后通过MQTT远程控制开发板的继电器, 简而言之: 51单片机+ESP8266用AT指令实现实现MQTT,(连接的本 ...

  4. 基于51单片机IIC通信的PCF8591学习笔记

    引言 PCF8591 是单电源,低功耗8 位CMOS 数据采集器件,具有4 个模拟输入.一个输出和一个串行I2C 总线接口.3 个地址引脚A0.A1 和A2 用于编程硬件地址,允许将最多8 个器件连接 ...

  5. 基于51单片机IIC通信的AT24C02学习笔记

    引言 最近在学习几种串行通信协议,感觉收获很多,这篇文章是学习IIC总线协议的第一篇文章,以后还会再写一篇关于PCF8591 IIC通信的ADDA转换芯片的文章. 关于IIC总线 IIC 即Inter ...

  6. 基于web的IM软件通信原理分析

    关于IM(InstantMessaging)即时通信类软件(如微信,QQ),大多数都是桌面应用程序或者native应用较为流行,而网上关于原生IM或桌面IM软件类的通信原理介绍也较多,此处不再赘述.而 ...

  7. 51单片机GPIO口模拟串口通信

    51单片机GPIO口模拟串口通信 标签: bytetimer终端存储 2011-08-03 11:06 6387人阅读 评论(2) 收藏 举报 本文章已收录于:   分类: 深入C语言(20) 作者同 ...

  8. 51单片机的TXD、 RXD 既接了 232 又接了 485芯片 ,会导致通信失败!

    51单片机的TXD. RXD 既接了 232 又接了 485 ,会导致通信失败! 下面是绘制电路板用的部分电路图: 通信现象: 1.我使用了USB-232的下载模块,把它接到P4上,发现单片机只能发送 ...

  9. STM32—IIC通信(软件实现底层函数)

    使用GPIO引脚模拟SDA和SCL总线实现软件模拟IIC通信,IIC的具体通信协议层和物理层链接:IIC #ifndef __BSP_IIC_H #define __BSP_IIC_H #includ ...

随机推荐

  1. Nginx服务器之负载均衡策略

    http://www.cnblogs.com/1214804270hacker/p/9325150.html

  2. nano-sql.js的基本操作

    nano-sql是一个小而快的数据库引擎, 他支持联合查询, 分组, 事务, ORM等功能, 支持 内存, indeedDB, Local Storage, WebSQL, Level DB 注意第一 ...

  3. logback 实例

    POM : <!-- log --> <dependency> <groupId>org.slf4j</groupId> <artifactId& ...

  4. 【Python】Excel处理

    1.包导入 from openpyxl import Workbook from openpyxl import load_workbook from openpyxl.compat import r ...

  5. scala中Map和Tuple

    /** * Created by root * Description : Tuple and Map */ object MapTest { def main(args: Array[String] ...

  6. RF实现多次失败重跑结果合并的基础方法和优化方法

    实现思路:通过分次执行失败案例重跑,然后通过结果文件合并命令实现多次失败重跑结果文件的合并,并输出合并后的log和report文件: 说明:具体失败案例重跑命令和结果文件合并命令请参考本博客其他相关章 ...

  7. iOS开发--UILabel可以显示\n

    UILabel*label; //设置换行 label.lineBreakMode = UILineBreakModeWordWrap; label.numberOfLines = ; 换行符还是“\ ...

  8. SaltStack 批量分发目录

    这里演示如何将 salt-master 上的目录批量分发到多台 salt-minion,步骤如下: [root@localhost ~]$ cat /srv/salt/top.sls # 先定义入口配 ...

  9. gcc的选项

    -g: 是一个编译选项,即在源代码编译的过程中起作用,让gcc把更多调试信息(也就包括符号信息)收集起来并将存放到最终的可执行文件内. -rdynamic:   却是一个 连接选项 ,它将指示连接器把 ...

  10. codeforces水题100道 第三题 Codeforces Beta Round #47 A. Domino piling (math)

    题目链接:http://www.codeforces.com/problemset/problem/50/A题意:一个NxM的举行中最多能放多少个1x2的矩形.C++代码: #include < ...