zoukankan      html  css  js  c++  java
  • I2C通信

    IIC

    原理:I2C是两线型串行总线,由时钟线SCL和数据线SDA构成的串行总线可发送和接受数据。

       I2C在传输过程中有三种类型的信号:开始信号,结束信号,应答信号

    • 空闲状态:SDA和SCL两条线同时处于高电平的时候,规定为总线是空闲信号
    • 开始信号: 当SCL为高电平的时候,SDA由高到低的跳变;是一种电平跳变时序信号
    • 停止信号:SCL为高电平期间,SDA由低到高的跳变,

    • 应答信号:发送器每发送一个字节,就在时钟脉冲9期间释放数据线,由接收器反馈一个应答信号

    应答信号为低电平时,规定为有效应答位,表示成功接收该字节;应答为高电平表示接收器接收失败

    要求在第九个时钟脉冲之前要把SDA拉低,并且确保在这个时钟高电平期间为稳定的低电平

    • 数据的有效信:在数据传输时,时钟信号为高电平期间,数据线上的数据必须保持稳定,只有在时钟线上的信号为低电平期间,数据线的高电平或者低电平状态才允许变化

        数据在SCL的上升沿到来之前就需要准备好。并在在下降沿到来的之前是稳定的

    • 数据传输:在SCL串行时钟的配合下,在SDA上逐位地串行传送每一位数据,边沿触发

    下面来点代码:

    1.用GPIO软件模拟I2C通讯过程

    1.1初始话GOPIO和IIC,最开始SDA,SCL线是拉高的

     1 void IIC_Init(void)
     2 {
     3     GPIO_InitTypeDef GPIO_Initure;
     4     
     5     __HAL_RCC_GPIOH_CLK_ENABLE(); 
     8     GPIO_Initure.Pin=GPIO_PIN_4|GPIO_PIN_5;
     9     GPIO_Initure.Mode=GPIO_MODE_OUTPUT_PP; 
    10     GPIO_Initure.Pull=GPIO_PULLUP;          
    11     GPIO_Initure.Speed=GPIO_SPEED_FAST;     
    12     HAL_GPIO_Init(GPIOH,&GPIO_Initure);
    13     
    14     IIC_SDA(1);
    15     IIC_SCL(1);  
    16 }
    //起始信号;当SCL为高电平的时候,SDA由高到低的跳变;
    1 2 void IIC_Start(void) 3 { 4 SDA_OUT(); //SDA线输出 5 IIC_SDA(1); //先都拉高 6 IIC_SCL(1); 7 delay_us(4); 8 IIC_SDA(0);//START:when CLK is high,DATA change form high to low 9 delay_us(4); 10 IIC_SCL(0); 11 }
    SCL为高电平期间,SDA由低到高的跳变
    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);
        delay_us(4);            
        IIC_SDA(1);                    
    }

    1.2接下来是读写和应答处理

    //应答等待
    u8 IIC_Wait_Ack(void)
    {
        u8 ucErrTime=0;
        SDA_IN();      // 释放SDA,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;  
    } 
    //产生应答
    void IIC_Ack(void)
    {
        IIC_SCL(0);
        SDA_OUT();
        IIC_SDA(0);
        delay_us(2);
        IIC_SCL(1);
        delay_us(2);
        IIC_SCL(0);
    }
    //不产生
    void IIC_NAck(void)
    {
        IIC_SCL(0);
        SDA_OUT();
        IIC_SDA(1);
        delay_us(2);
        IIC_SCL(1);
        delay_us(2);
        IIC_SCL(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);   
            IIC_SCL(1);
            delay_us(2); 
            IIC_SCL(0);    
            delay_us(2);
        }     
    }         
    
    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();
        else
            IIC_Ack(); 
        return receive;
    }


    //IO方向
    #define SDA_IN()  {GPIOH->MODER&=~(3<<(5*2));GPIOH->MODER|=0<<5*2;}  
    #define SDA_OUT() {GPIOH->MODER&=~(3<<(5*2));GPIOH->MODER|=1<<5*2;}
    //IO
    #define IIC_SCL(n)  (n?HAL_GPIO_WritePin(GPIOH,GPIO_PIN_4,GPIO_PIN_SET):HAL_GPIO_WritePin(GPIOH,GPIO_PIN_4,GPIO_PIN_RESET)) //SCL
    #define IIC_SDA(n)  (n?HAL_GPIO_WritePin(GPIOH,GPIO_PIN_5,GPIO_PIN_SET):HAL_GPIO_WritePin(GPIOH,GPIO_PIN_5,GPIO_PIN_RESET)) //SDA
    #define READ_SDA    HAL_GPIO_ReadPin(GPIOH,GPIO_PIN_5) 


    //初始化IIC接口
    void AT24CXX_Init(void)
    {
        IIC_Init();//IIC初始化
    }
    //在AT24CXX指定地址读出一个数据
    //ReadAddr:开始读数的地址  
    //返回值  :读到的数据
    u8 AT24CXX_ReadOneByte(u16 ReadAddr)
    {                  
        u8 temp=0;                                                                                   
        IIC_Start();  
        if(EE_TYPE>AT24C16)
        {
            IIC_Send_Byte(0XA0);       //发送写命令
            IIC_Wait_Ack();
            IIC_Send_Byte(ReadAddr>>8);//发送高地址        
        }else IIC_Send_Byte(0XA0+((ReadAddr/256)<<1));   //发送器件地址0XA0,写数据        
        IIC_Wait_Ack(); 
        IIC_Send_Byte(ReadAddr%256);   //发送低地址
        IIC_Wait_Ack();        
        IIC_Start();              
        IIC_Send_Byte(0XA1);           //进入接收模式               
        IIC_Wait_Ack();     
        temp=IIC_Read_Byte(0);           
        IIC_Stop();//产生一个停止条件        
        return temp;
    }
    //在AT24CXX指定地址写入一个数据
    //WriteAddr  :写入数据的目的地址    
    //DataToWrite:要写入的数据
    void AT24CXX_WriteOneByte(u16 WriteAddr,u8 DataToWrite)
    {                                                                                                  
        IIC_Start();  
        if(EE_TYPE>AT24C16)
        {
            IIC_Send_Byte(0XA0);        //发送写命令
            IIC_Wait_Ack();
            IIC_Send_Byte(WriteAddr>>8);//发送高地址      
        }else IIC_Send_Byte(0XA0+((WriteAddr/256)<<1));   //发送器件地址0XA0,写数据      
        IIC_Wait_Ack();       
        IIC_Send_Byte(WriteAddr%256);   //发送低地址
        IIC_Wait_Ack();                                                           
        IIC_Send_Byte(DataToWrite);     //发送字节                               
        IIC_Wait_Ack();                     
        IIC_Stop();//产生一个停止条件 
        delay_ms(10);     
    }
    //在AT24CXX里面的指定地址开始写入长度为Len的数据
    //该函数用于写入16bit或者32bit的数据.
    //WriteAddr  :开始写入的地址  
    //DataToWrite:数据数组首地址
    //Len        :要写入数据的长度2,4
    void AT24CXX_WriteLenByte(u16 WriteAddr,u32 DataToWrite,u8 Len)
    {      
        u8 t;
        for(t=0;t<Len;t++)
        {
            AT24CXX_WriteOneByte(WriteAddr+t,(DataToWrite>>(8*t))&0xff);
        }                                                    
    }
    
    //在AT24CXX里面的指定地址开始读出长度为Len的数据
    //该函数用于读出16bit或者32bit的数据.
    //ReadAddr   :开始读出的地址 
    //返回值     :数据
    //Len        :要读出数据的长度2,4
    u32 AT24CXX_ReadLenByte(u16 ReadAddr,u8 Len)
    {      
        u8 t;
        u32 temp=0;
        for(t=0;t<Len;t++)
        {
            temp<<=8;
            temp+=AT24CXX_ReadOneByte(ReadAddr+Len-t-1);                         
        }
        return temp;                                                    
    }
    //检查AT24CXX是否正常
    //这里用了24XX的最后一个地址(255)来存储标志字.
    //如果用其他24C系列,这个地址要修改
    //返回1:检测失败
    //返回0:检测成功
    u8 AT24CXX_Check(void)
    {
        u8 temp;
        temp=AT24CXX_ReadOneByte(255);//避免每次开机都写AT24CXX               
        if(temp==0X55)return 0;           
        else//排除第一次初始化的情况
        {
            AT24CXX_WriteOneByte(255,0X55);
            temp=AT24CXX_ReadOneByte(255);      
            if(temp==0X55)return 0;
        }
        return 1;                                              
    }
    
    //在AT24CXX里面的指定地址开始读出指定个数的数据
    //ReadAddr :开始读出的地址 对24c02为0~255
    //pBuffer  :数据数组首地址
    //NumToRead:要读出数据的个数
    void AT24CXX_Read(u16 ReadAddr,u8 *pBuffer,u16 NumToRead)
    {
        while(NumToRead)
        {
            *pBuffer++=AT24CXX_ReadOneByte(ReadAddr++);    
            NumToRead--;
        }
    }  
    //在AT24CXX里面的指定地址开始写入指定个数的数据
    //WriteAddr :开始写入的地址 对24c02为0~255
    //pBuffer   :数据数组首地址
    //NumToWrite:要写入数据的个数
    void AT24CXX_Write(u16 WriteAddr,u8 *pBuffer,u16 NumToWrite)
    {
        while(NumToWrite--)
        {
            AT24CXX_WriteOneByte(WriteAddr,*pBuffer);
            WriteAddr++;
            pBuffer++;
        }
    }

    上面是用软件I2C读取的AT24C02

     A0A1A2是设置器件的地址地址

    WD是写保护,接地取消写保护

    如果A1A2A3=0,读地址ADDress=0xA1,写就是ADDress=0xA0;

    写时序:  写模式(确定是读还是写),写地址(确定芯片内部的存储单元),写数据

    读操作,先确定读的地址,写模式(设置为写模式)-》  写地址(写要读的地址)-》读模式-》读数据

     下面是用51进行读写24c04,实现原理都差不多的。

    #include "24c04.h"
    #include "intrins.h"
    
    void vDelay10us(void)
    {
        unsigned char i;
        for(i=0;i<1;i++) ;
             
    
    }
    
    //启动信号开始的时候,clk = 1 data = 1  结束的时候 clk = 0 data = 0
    void vIICStart(void)
    {
    PDAT  =1 ;vDelay10us();
    
    PCLK = 1 ; vDelay10us();
    PDAT  =0 ; vDelay10us();
    PCLK = 0 ; vDelay10us();
    }
    
    
    //stop的开始条件 clk = 0  dat = 0  
    void vIICStop(void)
    {
    PDAT = 0 ;    vDelay10us(); 
    PCLK = 1 ;vDelay10us();
    PDAT = 1 ;    vDelay10us();
    
    }
    //发送数据的时候,clk从0开始,到0结束。data 最后是0 。
    void vNoACK(void)
    {
    PCLK = 0 ;vDelay10us();
    PDAT = 1 ;vDelay10us();
    PCLK = 1 ;vDelay10us();
    PCLK = 0 ;vDelay10us();
    PDAT = 0 ;vDelay10us();
    
    }
    
    //发送数据的时候,clk从0开始,到0结束。data 最后是0 。
    void vSend1Bit(void)
    {
    PCLK = 0 ;vDelay10us();
    PDAT = 1 ;vDelay10us();
    PCLK = 1 ;vDelay10us();
    PCLK = 0 ;vDelay10us();
    PDAT = 0 ;vDelay10us();
    
    }
    
    void vSend0Bit(void)
    {
    PCLK = 0 ;vDelay10us();
    PDAT = 0 ; vDelay10us();
    PCLK = 1 ; vDelay10us();
    PCLK = 0 ; vDelay10us();
    
    }
    
    
    
    void vSend1Byte(unsigned char u8Data)
    {
    unsigned char i;
     for(i = 0 ;i <8 ;i++)
     {
        if(  u8Data & 0x80 ) 
        {
              vSend1Bit();    
        }
        else
        {
             vSend0Bit();
        }       
         u8Data <<= 1 ; 
     }
    }
    
    //读取24c的ack 如果有正确的ACK 则得到1 否则得到0
    unsigned char u8ReadACK(void)
    {
    PDAT = 1 ; vDelay10us();
    PCLK = 1 ; vDelay10us();
    PCLK = 0 ; vDelay10us();
    if(    1 == PDAT)
        return 0;
        else
        return 1 ; 
    }
    
    
    void v24CWrite1Byte(unsigned char u8Address , unsigned char u8Data)
    {
                 vIICStart();
                vSend1Byte(0xA0);
                u8ReadACK();
                vSend1Byte(u8Address);
                u8ReadACK();
                vSend1Byte(u8Data);
                u8ReadACK();
                vIICStop();
    }
    
    
    unsigned char u824CRead1Byte(unsigned char u8Address)
    {
        unsigned char i,u8Temp;
    
                 vIICStart();
                vSend1Byte(0xA0);
                u8ReadACK();
                vSend1Byte(u8Address);
                u8ReadACK();
    
                vIICStart();
                vSend1Byte(0xA1);
                u8ReadACK();
    
                //读取8bit
                PDAT = 1 ; vDelay10us();
               for(i = 0 ;i<8;i++)
               {
                   u8Temp <<= 1 ;
                PCLK = 0 ; vDelay10us();
                PCLK = 1 ; vDelay10us();
                if(1 == PDAT)
                {               
                   u8Temp |= 0x01;                    
                }                            
               }   
    
               vNoACK();
               vIICStop();
            
         return u8Temp;
    
    }
  • 相关阅读:
    十年微软(Microsoft)MVP
    HTML5使用Div标签来实现表格
    C# 6.0的字典(Dictionary)的语法
    C# 6.0的属性(Property)的语法与初始值
    ASP.NET MVC使用jQuery实现Autocomplete
    The system cannot find the file specified
    Web实时通信
    实时数据显示--SignalR实例演示
    No Javascript on this page
    The SQL Server Service Broker for the current database is not enabled, and as a result query notifications are not supported.
  • 原文地址:https://www.cnblogs.com/wulianwangaxing/p/14362397.html
Copyright © 2011-2022 走看看