历史上的今天
返回首页

历史上的今天

今天是:2024年12月26日(星期四)

正在发生

2019年12月26日 | STM32学习笔记:【008】IIC通信协议详解

2019-12-26 来源:eefocus

什么是IIC(I2C)?

  IIC 即Inter-Integrated Circuit(集成电路总线),这种总线类型是由飞利浦半导体公司设计出来的一种简单、双向、二线制、同步串行总线。它是一种多向控制总线,也就是说多个芯片可以连接到同一总线结构下,同时每个芯片都可以作为实时数据传输的控制源。这种方式简化了信号传输总线接口。


  那么也就是说,只要收发双方同时接入SDA(双向数据线)、SCL(同步时钟线)便可以进行通信。


  I2C总线的工作速度分为 3 种模式(实际上,IIC的通信速率由SCL决定):

    S(标准模式),测量与控制场合;

    F(快速模式),速率为 400kb/s;(默认)

    Hs(高速模式),速率为 3.4Mb/s。

 

IIC接线框图

  一般情况下,SCL与SDA默认由上拉电阻拉高。这也是为了方便通信协议。

  多机连接时,为了区分不同的从机,我们会使用自定义的地址码进行区分。

   

IIC的通信状态

  IIC的通信要注意以下6个知识点:

    1.空闲状态

    2.开始信号

    3.停止信号

    4.应答信号

    5.数据的有效性

    6.数据传输

 

空闲状态:

  在IIC中规定,当SDA、SCL同时为高电平时,视为空闲状态。

  注意,这个规定是通信设备通信前的判断条件。

 

开始信号 & 停止信号:

  在IIC中规定,当SCL为高电平,且SDA从高到低的跳变时,视为数据开始传输;

  在IIC中规定,当SCL为高电平,且SDA从低到高的跳变时,视为数据停止传输;

  

 

数据有效性 & 数据传送 & 应答信号(ACK)

  数据有效性:

    在传输数据时,应保证数据在SCL的上升沿到来之前准备好,并在下降沿到来之前必须稳定。

    (由于在电路中,电平的跳变往往伴随着毛刺。)

  数据传送:

    在I2C总线上传送的每一位数据都有一个时钟脉冲相对应(或同步控制),即在SCL串行时钟的配合下,在SDA上逐位地串行传送每一位数据。

    在一般情况下,传输数据时,从数据的最高有效位开始发送。

  应答信号:

    在IIC中规定,发送方每发送1个字节(8位)后需要接收接收方发送的应答信号。

    ACK为0时,视为有效应答;ACK为1时,视为无效响应。

    总结:谁发了数据,谁就要接收一个应答信号。

/*

    https://blog.csdn.net/return_oops/article/details/80965437

*/

//使用IIC1 挂载M24C02,OLED,LM75AD,HT1382    PB6,PB7

 

#define SDA_IN()  {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)8<<28;}

#define SDA_OUT() {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)3<<28;}

 

//IO操作函数     

#define IIC_SCL    PBout(6) //SCL

#define IIC_SDA    PBout(7) //SDA     

#define READ_SDA   PBin(7)  //输入SDA 

 

//IIC所有操作函数

void IIC_Init(void);                //初始化IIC的IO口                 

void IIC_Start(void);                //发送IIC开始信号

void IIC_Stop(void);                  //发送IIC停止信号

void IIC_Send_Byte(u8 txd);            //IIC发送一个字节

u8 IIC_Read_Byte(unsigned char ack);//IIC读取一个字节

u8 IIC_Wait_Ack(void);                 //IIC等待ACK信号

void IIC_Ack(void);                    //IIC发送ACK信号

void IIC_NAck(void);                //IIC不发送ACK信号

 

void I2C_WriteByte(uint16_t addr,uint8_t data,uint8_t device_addr);

uint16_t I2C_ReadByte(uint16_t addr,uint8_t device_addr,uint8_t ByteNumToRead);//寄存器地址,器件地址,要读的字节数          



void IIC_Init(void)

{                         

    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(    RCC_APB2Periph_GPIOB, ENABLE );    

       

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ;   //推挽输出

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOB, &GPIO_InitStructure);

 

    IIC_SCL=1;

    IIC_SDA=1;

 

}

//产生IIC起始信号

void IIC_Start(void)

{

    SDA_OUT();     //sda线输出

    IIC_SDA=1;            

    IIC_SCL=1;

    delay_us(4);

     IIC_SDA=0;//START:when CLK is high,DATA change form high to low 

    delay_us(4);

    IIC_SCL=0;//钳住I2C总线,准备发送或接收数据 

}      

//产生IIC停止信号

void IIC_Stop(void)

{

    SDA_OUT();//sda线输出

    IIC_SCL=0;

    IIC_SDA=0;//STOP:when CLK is high DATA change form low to high

     delay_us(4);

    IIC_SCL=1; 

    IIC_SDA=1;//发送I2C总线结束信号

    delay_us(4);                                   

}

//等待应答信号到来

//返回值:1,接收应答失败

//        0,接收应答成功

u8 IIC_Wait_Ack(void)

{

    u8 ucErrTime=0;

    SDA_IN();      //SDA设置为输入  

    IIC_SDA=1;delay_us(1);       

    IIC_SCL=1;delay_us(1);     

    while(READ_SDA)

    {

        ucErrTime++;

        if(ucErrTime>250)

        {

            IIC_Stop();

            return 1;

        }

    }

    IIC_SCL=0;//时钟输出0        

    return 0;  

//产生ACK应答

void IIC_Ack(void)

{

    IIC_SCL=0;

    SDA_OUT();

    IIC_SDA=0;

    delay_us(2);

    IIC_SCL=1;

    delay_us(2);

    IIC_SCL=0;

}

//不产生ACK应答            

void IIC_NAck(void)

{

    IIC_SCL=0;

    SDA_OUT();

    IIC_SDA=1;

    delay_us(2);

    IIC_SCL=1;

    delay_us(2);

    IIC_SCL=0;

}                                          

//IIC发送一个字节

//返回从机有无应答

//1,有应答

//0,无应答              

void IIC_Send_Byte(u8 txd)

{                        

    u8 t;   

        SDA_OUT();         

    IIC_SCL=0;//拉低时钟开始数据传输

    for(t=0;t<8;t++)

    {              

        IIC_SDA=(txd&0x80)>>7;

        txd<<=1;       

        delay_us(2);   //对TEA5767这三个延时都是必须的

        IIC_SCL=1;

        delay_us(2); 

        IIC_SCL=0;    

        delay_us(2);

    }     

}         

//读1个字节,ack=1时,发送ACK,ack=0,发送nACK   

u8 IIC_Read_Byte(unsigned char ack)

{

    unsigned char i,receive=0;

    SDA_IN();//SDA设置为输入

  for(i=0;i<8;i++ )

    {

    IIC_SCL=0; 

    delay_us(2);

        IIC_SCL=1;

    receive<<=1;

    if(READ_SDA)receive++;   

        delay_us(1); 

  }                     

    if (!ack)

            IIC_NAck();//发送nACK

    else

            IIC_Ack(); //发送ACK   

    return receive;

}

 

void I2C_WriteByte(uint16_t addr,uint8_t data,uint8_t device_addr)

{

    IIC_Start();  

    

    if(device_addr==0xA0) //eeprom地址大于1字节

        IIC_Send_Byte(0xA0 + ((addr/256)<<1));//发送高地址

    else

        IIC_Send_Byte(device_addr);        //发器件地址

    IIC_Wait_Ack(); 

    IIC_Send_Byte(addr&0xFF);   //发送低地址

    IIC_Wait_Ack(); 

    IIC_Send_Byte(data);     //发送字节                               

        IIC_Wait_Ack();                     

        IIC_Stop();//产生一个停止条件 

    if(device_addr==0xA0) //

        delay_ms(10);

    else

        delay_us(2);

}

 

uint16_t I2C_ReadByte(uint16_t addr,uint8_t device_addr,uint8_t ByteNumToRead)  //读寄存器或读数据

{    

        uint16_t data;

        IIC_Start();  

        if(device_addr==0xA0)

            IIC_Send_Byte(0xA0 + ((addr/256)<<1));

        else

            IIC_Send_Byte(device_addr);    

        IIC_Wait_Ack();

        IIC_Send_Byte(addr&0xFF);   //发送低地址

        IIC_Wait_Ack(); 

 

        IIC_Start();      

        IIC_Send_Byte(device_addr+1);        //发器件地址

        IIC_Wait_Ack();

        if(ByteNumToRead == 1)//LM75温度数据为11bit

        {

            data=IIC_Read_Byte(0);

        }

        else

            {

                data=IIC_Read_Byte(1);

                data=(data<<8)+IIC_Read_Byte(0);

            }

        IIC_Stop();//产生一个停止条件        

        return data;

}


推荐阅读

史海拾趣

Fujitsu America公司的发展小趣事

为了更好地服务北美市场的客户,Fujitsu America实施了一系列本地化服务策略。公司不仅在当地建立了完善的销售和服务网络,还积极培养本地化的技术和服务团队,以更好地满足客户的实际需求。同时,Fujitsu America还注重与当地政府和行业协会的沟通与合作,积极参与行业标准的制定和推广工作,为公司在北美市场的长期发展奠定了坚实的基础。通过这些本地化服务策略的实施,Fujitsu America在北美市场树立了良好的品牌形象和市场地位。

永源微电子(APM)公司的发展小趣事

随着国内外市场的不断开拓和技术实力的不断提升,永源微电子开始实施全球化战略。公司积极寻求与国际知名企业的合作机会,通过技术引进和合资合作等方式,不断提升自身的国际化水平。同时,永源微电子还注重内部管理和人才培养,通过建立完善的管理体系和激励机制,吸引和留住了一批优秀的技术人才和管理人才。在全球化战略的推动下,永源微电子实现了持续稳健的发展,成为了电子行业中一颗璀璨的明星。

以上五个故事分别从不同角度描绘了永源微电子(APM)公司的发展历程,包括创立与初步发展、技术突破与产品线拓展、市场拓展与品牌建设、A轮融资与战略合作以及全球化战略与持续发展等方面。这些故事基于事实性的描述,展现了永源微电子在电子行业中的成长轨迹和发展成就。

Elpress AB公司的发展小趣事

在追求经济效益的同时,Elpress AB也注重可持续发展和环境保护。公司积极采用环保材料和绿色生产工艺,降低生产过程中的能耗和排放。同时,Elpress AB还积极参与环保公益活动,推动电子行业的绿色发展。这些举措不仅体现了Elpress AB的社会责任感,也为公司的长期发展奠定了坚实的基础。

APTA Group Inc公司的发展小趣事

APTA Group Inc深知,单打独斗在竞争激烈的电子行业中难以长久立足。因此,公司积极寻求与其他企业的战略合作。通过与知名电子产品制造商的合作,APTA得以将其技术应用于更广泛的领域,同时也借助合作伙伴的市场渠道,提高了品牌知名度和市场占有率。这种互利共赢的合作模式,为APTA的快速发展提供了有力支撑。

Geo Semiconductor Inc公司的发展小趣事
通过增强故障诊断与报警功能,及时发现并处理故障,减少因故障导致的停机时间。
EMS GmbH公司的发展小趣事

近年来,随着汽车行业向电动化、智能化方向发展,汽车转换器注塑件的需求也发生了变化。EMS GmbH公司积极应对行业挑战,加大研发投入,推动产品向智能化、绿色化方向转型。同时,公司还关注新兴领域的发展机会,如新能源汽车、自动驾驶等领域,寻求新的增长点。这些努力使EMS GmbH公司能够保持行业领先地位,并在未来市场中保持竞争力。

问答坊 | AI 解惑

Windows Embedded工控 油田开采效率高

  能源是一种极重要的资源。大家都知道,没有能源,汽车不能开,飞机不能飞,工业无法持续发展,经济提高也没有保证。   所以,开发油田的意义很大,现在咱国家原油产量在1.6亿吨左右。按现有资源的可获量以及目前开采利用情况和技术经济条件 ...…

查看全部问答>

EVC下能用skin++么?

用这个EVC做出的界面真难看,,,,我试着把skin++加进去。。 可是一加就出错:BtnTest.obj : error LNK2019: unresolved external symbol \"__declspec(dllimport) int __cdecl InitializeSkin(char *)\" (__imp_?InitializeSkin@@YAHPAD@Z) refe ...…

查看全部问答>

如何把 irp->AssociatedIrp.SystemBuffer 整个复制了?

如何把 irp->AssociatedIrp.SystemBuffer SystemBuffer整个复制了?…

查看全部问答>

CE flash播放

哪位高手有在CE上播放flash的类啊?我想在CE上播放flash.能不能借我参考一下,谢谢。…

查看全部问答>

想在wince下实现一个简单的列表界面,显示500条数据信息;同时运行一个通信协议

想在wince下实现一个简单的列表界面,显示500条数据信息;同时运行一个通信协议,  要求稳定性要高,可长期运行, 我的列表界面和 协议栈  是用两个 进程好呢,还是用 两个线程好呢 ? 看了网上一些信息, 多个线程间容易通信 ...…

查看全部问答>

硅光电池采样

我手上现有几颗硅光电池,想用它来替代光敏电阻,来对环境光做粗略采样。用万用表测了一下其输出电压最大在500mv,光源用的是节能灯。我用的单片机是mega16,采用电路如附件。不知这样的采集是否正确,希望给点意见和更好的采集方法。   …

查看全部问答>

软件朋友的商机

    用博客写了篇技术文章,想要贴图,发现它不支持,才知道博客编辑器使用起来非常不便。经多次使用发现博客编辑器至少要在以下几点加以改善。     1、要能贴图。在技术性文章中,图片说明是必不可少的,图文 ...…

查看全部问答>

请问哪位大虾知道STR736怎么用定时器来做外部脉冲计数?

                                 如题…

查看全部问答>

电源制作

哪位老兄用220V交流电做过449的产品?我要用12V,5V,3.5V, 8V. 该怎样制作?电源要稳定…

查看全部问答>