14 STM32 - IIC (时序图+软件源码)

14.1 IIC简介

IIC(Inter-Integrated Circuit),中文集成电路总线,是一种串行通信总线,使用多主从架构。I2C串行总线一般有两根信号线,一根是双向的数据线SDA,另一根是时钟线SCL。所有接到I2C总线设备上的串行数据SDA都接到总线的SDA上,各设备的时钟线SCL接到总线的SCL上。主设备通过两个IO口便可以访问许多设备,因此可以节约IO口。IIC主从之间只有一根数据线,可以收数据,也可以发数据,但是不能同时收发,因此IIC属于半双工的通信模式。IIC硬件电路通常两根线接4.7K-10K电阻上拉到3.3V。结构如下图所示。

14.2 IIC时序

IIC两根线传输数据的信号分为开始信号,传输信号,应答信号,结束信号。

开始信号:SCL为高电平时,SDA由高向低跳变。

传输信号:SCL为高电平时,SDA的电平信号(整个SCL为高时,要确保SDA信号保持不变,只有SCL低电平时,SDA电平才能改变)。

应答信号:传输信号完成后,SDA在SCL低电平时输出高电平,然后修改为输入模式,待SCL为高时读取SDA电平,为高是否定应答信号,为低是肯定应答信号。

结束信号:SCL为高电平时,SDA由低向高跳变。

如下图,即完成一次8字节数据传输,发送的数据为:0xB2,(二进制: 10110110)

1: 开始信号;

2:数据改变,只有SCL为低时才能修改;

3-10:数据传输,数据为0xB2,(二进制: 10110110)

11:应答信号:高为从机确定收到数据,低为从机不确定收到数据

12: 结束信号

14.3 软件模拟源码:

c 复制代码
#include "soft_iic.h"
#include "delay.h"

//IO操作函数     
#define IIC_SCL    PBout(6) //SCL
#define IIC_SDA    PBout(7) //SDA     
#define READ_SDA   PBin(7)  //输入SDA 

//IO方向设置
#define SDA_IN()  { GPIOB->MODER&=~(3<<(7*2));GPIOB->MODER|=0<<7*2; }  //PB7输入模式
#define SDA_OUT() { GPIOB->MODER&=~(3<<(7*2));GPIOB->MODER|=1<<7*2; }  //PB7输出模式



//初始化IIC
void IICInit(void)
{
    GPIO_InitTypeDef  GPIO_InitStructure;
    
    //GPIOB8,B9初始化设置
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;   //普通输出模式
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;  //推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;    //上拉
    GPIO_Init(GPIOB, &GPIO_InitStructure);          //初始化
    IIC_SCL=1;
    IIC_SDA=1;
}

//产生IIC起始信号
static void IIC_Start(void)
{
    SDA_OUT();     //sda线输出
    IIC_SDA=1;
    IIC_SCL=1;
    DelayUs(4);
    IIC_SDA=0;    //START:when CLK is high,DATA change form high to low 
    DelayUs(4);
    IIC_SCL=0;    //钳住I2C总线,准备发送或接收数据 
}

//产生IIC停止信号
static void IIC_Stop(void)
{
    SDA_OUT();  //sda线输出
    IIC_SCL=0;
    IIC_SDA=0;  //STOP:when CLK is high DATA change form low to high
    DelayUs(4);
    IIC_SCL=1; 
    IIC_SDA=1;  //发送I2C总线结束信号
    DelayUs(4);                                   
}

//等待应答信号到来
//返回值:1,接收应答失败    0,接收应答成功
static u8 IIC_Wait_Ack(void)
{
    u8 ucErrTime=0;
    SDA_IN();      //SDA设置为输入  
    IIC_SDA=1;DelayUs(1);       
    IIC_SCL=1;DelayUs(1);     
    while(READ_SDA)
    {
        ucErrTime++;
        if(ucErrTime>250) // 超时错误
        {
            IIC_Stop();
            return 1;
        }
    }
    IIC_SCL=0; //时钟输出0        
    return 0; // 有应答返回0
}

//产生ACK应答
static void IIC_Ack(void)
{
    IIC_SCL=0;
    SDA_OUT();
    IIC_SDA=0;
    DelayUs(2);
    IIC_SCL=1;
    DelayUs(2);
    IIC_SCL=0;
}

//不产生ACK应答            
static void IIC_NAck(void)
{
    IIC_SCL=0;
    SDA_OUT();
    IIC_SDA=1;
    DelayUs(2);
    IIC_SCL=1;
    DelayUs(2);
    IIC_SCL=0;
}

static 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;
        DelayUs(2);   //对TEA5767这三个延时都是必须的
        IIC_SCL=1;
        DelayUs(2);
        IIC_SCL=0;
        DelayUs(2);
    }
}

static u8 IIC_Read_Byte(unsigned char ack)
{
    unsigned char i,receive=0;
    SDA_IN();//SDA设置为输入
    for(i=0;i<8;i++ )
    {
        IIC_SCL=0; 
        DelayUs(2);
        IIC_SCL=1;
        receive<<=1;
        if(READ_SDA)receive++;
        DelayUs(1); 
    }
    if (ack)
        IIC_Ack(); //发送ACK   
    else
        IIC_NAck(); //发送nACK
    return receive;
}

直接调用以上API函数,可以读写24C02(EEPROM)。

c 复制代码
// 读1字节内容
u8 W24C_ReadOneByte(u16 addr, u8* data)
{
    IIC_Start();
    IIC_Send_Byte(0xA0);
    if(1 == IIC_Wait_Ack())
    {
        return 1; // err
    }
    IIC_Send_Byte(addr>>8);
    if(1 == IIC_Wait_Ack())
    {
        return 2; // err
    }
    IIC_Send_Byte(addr&0x0FF);
    if(1 == IIC_Wait_Ack())
    {
        return 3; // err
    }
    IIC_Start();
    IIC_Send_Byte(0xA1);
    if(1 == IIC_Wait_Ack())
    {
        return 4; // err
    }
    data[0] = IIC_Read_Byte(0);
    IIC_Stop();
    return 0;
}


// 读n字节内容
u8 W24C_ReadByte(u16 addr, u8* buf, u16 len)
{
    u16 i;
    IIC_Start();
    IIC_Send_Byte(0xA0);
    if(1 == IIC_Wait_Ack())
    {
        return 1; // err
    }
    IIC_Send_Byte(addr>>8);
    if(1 == IIC_Wait_Ack())
    {
        return 2; // err
    }
    IIC_Send_Byte(addr&0x0FF);
    if(1 == IIC_Wait_Ack())
    {
        return 3; // err
    }
    IIC_Start();
    IIC_Send_Byte(0xA1);
    if(1 == IIC_Wait_Ack())
    {
        return 4; // err
    }
    for(i=0; i<len; i++)
    {
        buf[i] = IIC_Read_Byte(1);
    }
    IIC_Stop();
    return 0;
}

u8 W24C_WriteOneByte(u16 addr, u8 data)
{
    IIC_Start();
    IIC_Send_Byte(0xA0);
    if(1 == IIC_Wait_Ack())
    {
        return 1; // err
    }
    IIC_Send_Byte(addr>>8);
    if(1 == IIC_Wait_Ack())
    {
        return 2; // err
    }
    IIC_Send_Byte(addr&0x0FF);
    if(1 == IIC_Wait_Ack())
    {
        return 3; // err
    }
    IIC_Send_Byte(data);
    if(1 == IIC_Wait_Ack())
    {
        return 4; // err
    }
    IIC_Stop();
    DelayMs(5);
    return 0;
}


u8 W24C_WriteByteHighSpeed(u16 addr, u8* data, u8 len)
{
    u8 i;
    
    IIC_Start();
    IIC_Send_Byte(0xA0);
    if(1 == IIC_Wait_Ack())
    {
        return 1; // err
    }
    IIC_Send_Byte(addr>>8);
    if(1 == IIC_Wait_Ack())
    {
        return 2; // err
    }
    IIC_Send_Byte(addr&0x0FF);
    if(1 == IIC_Wait_Ack())
    {
        return 3; // err
    }
    for(i=0; i<len; i++)
    {
        IIC_Send_Byte(data[i]);
        IIC_Wait_Ack();
    }
    IIC_Stop();
    DelayMs(5);
    return 0;            
}


u8 W24C_WriteBytes(u16 addr, u8* data, u8 len)
{
    while(len--)
    {
        W24C_WriteOneByte(addr,*data);
        addr++;
        data++;
    }
    return 0;
}
相关推荐
Natsume171015 分钟前
嵌入式开发:GPIO、UART、SPI、I2C 驱动开发详解与实战案例
c语言·驱动开发·stm32·嵌入式硬件·mcu·架构·github
MeshddY1 小时前
(超详细)数据库项目初体验:使用C语言连接数据库完成短地址服务(本地运行版)
c语言·数据库·单片机
m0_555762901 小时前
STM32常见外设
stm32·单片机·嵌入式硬件
森焱森1 小时前
无人机三轴稳定化控制(1)____飞机的稳定控制逻辑
c语言·单片机·算法·无人机
循环过三天1 小时前
3-1 PID算法改进(积分部分)
笔记·stm32·单片机·学习·算法·pid
天天爱吃肉82183 小时前
ZigBee通信技术全解析:从协议栈到底层实现,全方位解读物联网核心无线技术
python·嵌入式硬件·物联网·servlet
东风点点吹3 小时前
STM32F103的boot跳转APP不成功问题排除
stm32·单片机·嵌入式硬件
猫猫的小茶馆5 小时前
【STM32】预分频因子(Prescaler)和重装载值(Reload Value)
c语言·stm32·单片机·嵌入式硬件·mcu·51单片机
riveting6 小时前
明远智睿H618:开启多场景智慧生活新时代
人工智能·嵌入式硬件·智能硬件·lga封装·3506
三万棵雪松7 小时前
【STM32HAL-第1讲 基础篇-单片机简介】
stm32·单片机·嵌入式硬件