项目场景:基于STM32F407实现GPIO软件模拟IIC驱动EEPROM

背景:工作中用到了EEPROM用于存储配置信息,需要对EEPROM进行读、写功能的实现。

硬件:使用的EEPROM型号为 BL24C64A-PARC,
附一下这个片子的datasheet链接:https://atta.szlcsc.com/upload/public/pdf/source/20161130/1480498805803.pdf


先分析一下这个存储芯片

图1-图2:介绍了这个存储芯片总工64K容量,即可以存储65536个比特位,共有256个存储页面,每页可以存65536/256=256个比特位,即256/8=32个字节。

图3:介绍了芯片的地址引脚,这个地址引脚为这个芯片的固定地址,如果只用一个芯片的话,地址可任意配置,但此地址在IIC通信中需要用到,我这边硬件直接给固定到地址为000,如果你用其他IIC设备,这个设备地址设置不要设置与000相同即可

图4:接下来就说一下这种容量较大的存储片的数据地址,之前大部分可能接触的都是容量较小的EEPROM芯片,这种小容量芯片8个地址位即可完整访问到该芯片的每个数据寄存器,但这款芯片8位的数据地址位已经不能满足要求了。计算一下:256页*32字节/页=8192个字节,2^13=8192,即二进制地址,需要13个数据位才能遍历完8192个数据地址。数据文档中也给我们做了说明。


好了,以上只是根据此款芯片的数据手册进行了关键必要信息的初步了解。相信学习IIC的同学,根据各种资料已经明白IIC的SDA和SCL引脚的时序要求及表达各类信号的信号时序。下面开始根据IIC总线的要求进行编程。我要实现的是软件模拟IIC的功能,废话不多说,直接干代码

@gpio_iic.h文件
#include "gpio_iic.h"
/*
*********************************************************************************************************
*   函 数 名: i2c_Delay
*   功能说明: I2C总线位延迟,最快400KHz
*   形    参:无
*   返 回 值: 无
*********************************************************************************************************
*/
static void i2c_Delay(void)
{uint8_t i;/* 可用逻辑分析仪测量I2C通讯时的频率工作条件:CPU主频168MHz ,MDK编译环境,1级优化经测试,循环次数为20~250时都能通讯正常*/for (i = 0; i < 40; i++);
}
/*
*********************************************************************************************************
*   函 数 名: i2c_Start
*   功能说明: CPU发起I2C总线启动信号
*   形    参:无
*   返 回 值: 无
*********************************************************************************************************
*/
void i2c_Start(void)
{/* 当SCL高电平时,SDA出现一个下跳沿表示I2C总线启动信号 */EEPROM_I2C_SDA_1();EEPROM_I2C_SCL_1();i2c_Delay();EEPROM_I2C_SDA_0();i2c_Delay();EEPROM_I2C_SCL_0();i2c_Delay();
}
/*
*********************************************************************************************************
*   函 数 名: i2c_Start
*   功能说明: CPU发起I2C总线停止信号
*   形    参:无
*   返 回 值: 无
*********************************************************************************************************
*/
void i2c_Stop(void)
{/* 当SCL高电平时,SDA出现一个上跳沿表示I2C总线停止信号 */EEPROM_I2C_SDA_0();EEPROM_I2C_SCL_1();i2c_Delay();EEPROM_I2C_SDA_1();
}
/*
*********************************************************************************************************
*   函 数 名: i2c_SendByte
*   功能说明: CPU向I2C总线设备发送8bit数据
*   形    参:_ucByte : 等待发送的字节
*   返 回 值: 无
*********************************************************************************************************
*/
void i2c_SendByte(uint8_t _ucByte)
{uint8_t i;/* 先发送字节的高位bit7 */for (i = 0; i < 8; i++){     if (_ucByte & 0x80){EEPROM_I2C_SDA_1();}else{EEPROM_I2C_SDA_0();}i2c_Delay();EEPROM_I2C_SCL_1();i2c_Delay();    EEPROM_I2C_SCL_0();if (i == 7){EEPROM_I2C_SDA_1(); // 释放总线}_ucByte <<= 1;  /* 左移一个bit */i2c_Delay();}
}
/*
*********************************************************************************************************
*   函 数 名: i2c_ReadByte
*   功能说明: CPU从I2C总线设备读取8bit数据
*   形    参:无
*   返 回 值: 读到的数据
*********************************************************************************************************
*/
uint8_t i2c_ReadByte(void)
{uint8_t i;uint8_t value;/* 读到第1个bit为数据的bit7 */value = 0;for (i = 0; i < 8; i++){value <<= 1;EEPROM_I2C_SCL_1();i2c_Delay();if (EEPROM_I2C_SDA_READ()){value++;}EEPROM_I2C_SCL_0();i2c_Delay();}return value;
}
/*
*********************************************************************************************************
*   函 数 名: i2c_WaitAck
*   功能说明: CPU产生一个时钟,并读取器件的ACK应答信号
*   形    参:无
*   返 回 值: 返回0表示正确应答,1表示无器件响应
*********************************************************************************************************
*/
uint8_t i2c_WaitAck(void)
{uint8_t re;EEPROM_I2C_SDA_1(); /* CPU释放SDA总线 */i2c_Delay();EEPROM_I2C_SCL_1(); /* CPU驱动SCL = 1, 此时器件会返回ACK应答 */i2c_Delay();if (EEPROM_I2C_SDA_READ()) /* CPU读取SDA口线状态 */{re = 1;}else{re = 0;}EEPROM_I2C_SCL_0();i2c_Delay();return re;
}
/*
*********************************************************************************************************
*   函 数 名: i2c_Ack
*   功能说明: CPU产生一个ACK信号
*   形    参:无
*   返 回 值: 无
*********************************************************************************************************
*/
void i2c_Ack(void)
{EEPROM_I2C_SDA_0();    /* CPU驱动SDA = 0 */i2c_Delay();EEPROM_I2C_SCL_1();  /* CPU产生1个时钟 */i2c_Delay();EEPROM_I2C_SCL_0();i2c_Delay();EEPROM_I2C_SDA_1();   /* CPU释放SDA总线 */
}
/*
*********************************************************************************************************
*   函 数 名: i2c_NAck
*   功能说明: CPU产生1个NACK信号
*   形    参:无
*   返 回 值: 无
*********************************************************************************************************
*/
void i2c_NAck(void)
{EEPROM_I2C_SDA_1();    /* CPU驱动SDA = 1 */i2c_Delay();EEPROM_I2C_SCL_1();  /* CPU产生1个时钟 */i2c_Delay();EEPROM_I2C_SCL_0();i2c_Delay();
}
/*
*********************************************************************************************************
*   函 数 名: i2c_CfgGpio
*   功能说明: 配置I2C总线的GPIO,采用模拟IO的方式实现
*   形    参:无
*   返 回 值: 无
*********************************************************************************************************
*/
void i2c_CfgGpio(void)
{GPIO_InitTypeDef GPIO_InitStructure;RCC_AHB1PeriphClockCmd(EEPROM_I2C_GPIO_CLK, ENABLE);   /* 打开GPIO时钟 */GPIO_InitStructure.GPIO_Pin = EEPROM_I2C_SCL_PIN | EEPROM_I2C_SDA_PIN;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;   GPIO_InitStructure.GPIO_OType = GPIO_OType_OD;     /* 开漏输出 */GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;GPIO_Init(EEPROM_I2C_GPIO_PORT, &GPIO_InitStructure);/* 给一个停止信号, 复位I2C总线上的所有设备到待机模式 */i2c_Stop();
}
/*
*********************************************************************************************************
*   函 数 名: i2c_CheckDevice
*   功能说明: 检测I2C总线设备,CPU向发送设备地址,然后读取设备应答来判断该设备是否存在
*   形    参:_Address:设备的I2C总线地址
*   返 回 值: 返回值 0 表示正确, 返回1表示未探测到
*********************************************************************************************************
*/
uint8_t i2c_CheckDevice(uint8_t _Address)
{uint8_t ucAck;i2c_CfgGpio();       /* 配置GPIO */i2c_Start();        /* 发送启动信号 *//* 发送设备地址+读写控制bit(0 = w, 1 = r) bit7 先传 */i2c_SendByte(_Address | EEPROM_I2C_WR);ucAck = i2c_WaitAck();   /* 检测设备的ACK应答 */i2c_Stop();         /* 发送停止信号 */return ucAck;
}

@gpio_iic.h文件
#ifndef _BSP_I2C_GPIO_H
#define _BSP_I2C_GPIO_H#include "stm32f4xx.h"
#include <inttypes.h>#define EEPROM_I2C_WR    0       /* 写控制bit */
#define EEPROM_I2C_RD   1       /* 读控制bit */#define LNB_I2C_WR  0       /* 写控制bit */
#define LNB_I2C_RD  1       /* 读控制bit *//* 定义I2C总线连接的GPIO端口, 用户只需要修改下面4行代码即可任意改变SCL和SDA的引脚 */
#define LNB_I2C_GPIO_PORT               GPIOF           /* GPIO端口 */
#define LNB_I2C_GPIO_CLK                RCC_AHB1Periph_GPIOF        /* GPIO端口时钟 */
#define LNB_I2C_SCL_PIN                 GPIO_Pin_1          /* 连接到SCL时钟线的GPIO */
#define LNB_I2C_SDA_PIN                 GPIO_Pin_0          /* 连接到SDA数据线的GPIO *//* 定义读写SCL和SDA的宏,已增加代码的可移植性和可阅读性 */
#if 1   /* 条件编译: 1 选择GPIO的库函数实现IO读写 */#define LNB_I2C_SCL_1()  GPIO_SetBits(LNB_I2C_GPIO_PORT, LNB_I2C_SCL_PIN)      /* SCL = 1 */#define LNB_I2C_SCL_0()  GPIO_ResetBits(LNB_I2C_GPIO_PORT, LNB_I2C_SCL_PIN)       /* SCL = 0 */#define LNB_I2C_SDA_1()  GPIO_SetBits(LNB_I2C_GPIO_PORT, LNB_I2C_SDA_PIN)     /* SDA = 1 */#define LNB_I2C_SDA_0()  GPIO_ResetBits(LNB_I2C_GPIO_PORT, LNB_I2C_SDA_PIN)       /* SDA = 0 */#define LNB_I2C_SDA_READ()  GPIO_ReadInputDataBit(LNB_I2C_GPIO_PORT, LNB_I2C_SDA_PIN) /* 读SDA口线状态 */
#else   /* 这个分支选择直接寄存器操作实现IO读写 *//* 注意:如下写法,在IAR最高级别优化时,会被编译器错误优化 */#define LNB_I2C_SCL_1()  LNB_I2C_GPIO_PORT->BSRRL = LNB_I2C_SCL_PIN                /* SCL = 1 */#define LNB_I2C_SCL_0()  LNB_I2C_GPIO_PORT->BSRRH = LNB_I2C_SCL_PIN               /* SCL = 0 */#define LNB_I2C_SDA_1()  LNB_I2C_GPIO_PORT->BSRRL = LNB_I2C_SDA_PIN               /* SDA = 1 */#define LNB_I2C_SDA_0()  LNB_I2C_GPIO_PORT->BSRRH = LNB_I2C_SDA_PIN               /* SDA = 0 */#define LNB_I2C_SDA_READ()  ((LNB_I2C_GPIO_PORT->IDR & LNB_I2C_SDA_PIN) != 0) /* 读SDA口线状态 */
#endif/* 定义I2C总线连接的GPIO端口, 用户只需要修改下面4行代码即可任意改变SCL和SDA的引脚 */
#define EEPROM_I2C_GPIO_PORT                GPIOB           /* GPIO端口 */
#define EEPROM_I2C_GPIO_CLK             RCC_AHB1Periph_GPIOB        /* GPIO端口时钟 */
#define EEPROM_I2C_SCL_PIN                  GPIO_Pin_6          /* 连接到SCL时钟线的GPIO */
#define EEPROM_I2C_SDA_PIN                  GPIO_Pin_7          /* 连接到SDA数据线的GPIO *//* 定义读写SCL和SDA的宏,已增加代码的可移植性和可阅读性 */
#if 1   /* 条件编译: 1 选择GPIO的库函数实现IO读写 */#define EEPROM_I2C_SCL_1()  GPIO_SetBits(EEPROM_I2C_GPIO_PORT, EEPROM_I2C_SCL_PIN)     /* SCL = 1 */#define EEPROM_I2C_SCL_0()  GPIO_ResetBits(EEPROM_I2C_GPIO_PORT, EEPROM_I2C_SCL_PIN)      /* SCL = 0 */#define EEPROM_I2C_SDA_1()  GPIO_SetBits(EEPROM_I2C_GPIO_PORT, EEPROM_I2C_SDA_PIN)        /* SDA = 1 */#define EEPROM_I2C_SDA_0()  GPIO_ResetBits(EEPROM_I2C_GPIO_PORT, EEPROM_I2C_SDA_PIN)      /* SDA = 0 */#define EEPROM_I2C_SDA_READ()  GPIO_ReadInputDataBit(EEPROM_I2C_GPIO_PORT, EEPROM_I2C_SDA_PIN)    /* 读SDA口线状态 */
#else   /* 这个分支选择直接寄存器操作实现IO读写 *//* 注意:如下写法,在IAR最高级别优化时,会被编译器错误优化 */#define EEPROM_I2C_SCL_1()  EEPROM_I2C_GPIO_PORT->BSRRL = EEPROM_I2C_SCL_PIN               /* SCL = 1 */#define EEPROM_I2C_SCL_0()  EEPROM_I2C_GPIO_PORT->BSRRH = EEPROM_I2C_SCL_PIN              /* SCL = 0 */#define EEPROM_I2C_SDA_1()  EEPROM_I2C_GPIO_PORT->BSRRL = EEPROM_I2C_SDA_PIN              /* SDA = 1 */#define EEPROM_I2C_SDA_0()  EEPROM_I2C_GPIO_PORT->BSRRH = EEPROM_I2C_SDA_PIN              /* SDA = 0 */#define EEPROM_I2C_SDA_READ()  ((EEPROM_I2C_GPIO_PORT->IDR & EEPROM_I2C_SDA_PIN) != 0)    /* 读SDA口线状态 */
#endifvoid i2c_Start(void);
void i2c_Stop(void);
void i2c_SendByte(uint8_t _ucByte);
uint8_t i2c_ReadByte(void);
uint8_t i2c_WaitAck(void);
void i2c_Ack(void);
void i2c_NAck(void);
void i2c_CfgGpio(void);
uint8_t i2c_CheckDevice(uint8_t _Address);
#endif

以上,我们完成了最基本的用GPIO引脚模拟IIC的每个信号的代码以及引脚输入输出的相关宏定义。我用的是STM32F407ZGT6的PB6和PB7引脚,分别作为IIC的SCL和SDA引脚(硬件),可根据实实际引脚进行修改,上面还有我一个其他外设使用的IIC设备,LNB开头命名的,大家可以忽略。

下面根据芯片的IIC时序要求,来实际完成通过软件IIC读写芯片的程序。
如下图所示,介绍了此存储芯片进行读写操作的的IIC时序。

Figure7:介绍的是按页写数据,首先是主机发送start起始信号,然后主机发送设备地址(注意读写位置位,读1,写0)。等待从设备应答,收到应答信号,发送数据寄存器的第一个地址,收到应答,发送数据寄存器第二个地址,再次收到应答后,直接一个字节一个字节的去发送数据,数据发送完成,给一个停止信号,表示数据写操作完成。

Figure9和Figure10:表示从芯片读取数据时的IIC时序。Figure9是读取随机的数据寄存器中的数据;Figure10是读取连续数据寄存器地址中的数据。


有了以上数据手册给出的时序,我们只需要根据上面已经写好的信号去封装相应的读写函数即可,代码的每一步基本都做了注释,根据时序来阅读一下代码,很容易理解的。

@soft_iic.c文件
#include "soft_iic.h"
#include "gpio_iic.h"
#include "usart.h"/*
*********************************************************************************************************
*   函 数 名: ee_CheckOk
*   功能说明: 判断串行EERPOM是否正常
*   形    参:无
*   返 回 值: 1 表示正常, 0 表示不正常
*********************************************************************************************************
*/
uint8_t ee_CheckOk(void)
{if (i2c_CheckDevice(EEPROM_DEV_ADDR) == 0){return 1;}else{/* 失败后,切记发送I2C总线停止信号 */i2c_Stop();      return 0;}
}/*
*********************************************************************************************************
*   函 数 名: ee_ReadBytes
*   功能说明: 从串行EEPROM指定地址处开始读取若干数据
*   形    参:_usAddress : 起始地址
*            _usSize : 数据长度,单位为字节
*            _pReadBuf : 存放读到的数据的缓冲区指针
*   返 回 值: 0 表示失败,1表示成功
*********************************************************************************************************
*/
uint8_t ee_ReadBytes(char *_pReadBuf, uint8_t _usAddress1,uint8_t _usAddress2,  uint16_t _usSize)
{uint16_t i;/* 采用串行EEPROM随即读取指令序列,连续读取若干字节 *//* 第1步:发起I2C总线启动信号 */i2c_Start();/* 第2步:发起控制字节,高7bit是地址,bit0是读写控制位,0表示写,1表示读 */i2c_SendByte(EEPROM_DEV_ADDR | EEPROM_I2C_WR); /* 此处是写指令 *//* 第3步:等待ACK */if (i2c_WaitAck() != 0){goto cmd_fail;   /* EEPROM器件无应答 */}/* 第4步:发送字节地址,24C02只有256字节,因此1个字节就够了,如果是24C04以上,那么此处需要连发多个地址 */i2c_SendByte((uint8_t)_usAddress1);/* 第5步:等待ACK */if (i2c_WaitAck() != 0){goto cmd_fail;    /* EEPROM器件无应答 */}/* 第6步:发送字节地址,24C02只有256字节,因此1个字节就够了,如果是24C04以上,那么此处需要连发多个地址 */i2c_SendByte((uint8_t)_usAddress2);/* 第7步:等待ACK */if (i2c_WaitAck() != 0){goto cmd_fail;    /* EEPROM器件无应答 */}/* 第8步:重新启动I2C总线。前面的代码的目的向EEPROM传送地址,下面开始读取数据 */i2c_Start();/* 第9步:发起控制字节,高7bit是地址,bit0是读写控制位,0表示写,1表示读 */i2c_SendByte(EEPROM_DEV_ADDR | EEPROM_I2C_RD); /* 此处是读指令 *//* 第10步:发送ACK */if (i2c_WaitAck() != 0){goto cmd_fail;  /* EEPROM器件无应答 */}  /* 第9步:循环读取数据 */for (i = 0; i < _usSize; i++){_pReadBuf[i] = i2c_ReadByte();  /* 读1个字节 *//* 每读完1个字节后,需要发送Ack, 最后一个字节不需要Ack,发Nack */if (i != _usSize - 1){i2c_Ack(); /* 中间字节读完后,CPU产生ACK信号(驱动SDA = 0) */}else{i2c_NAck();    /* 最后1个字节读完后,CPU产生NACK信号(驱动SDA = 1) */}}/* 发送I2C总线停止信号 */i2c_Stop();return 1;   /* 执行成功 */cmd_fail: /* 命令执行失败后,切记发送停止信号,避免影响I2C总线上其他设备 *//* 发送I2C总线停止信号 */i2c_Stop();return 0;
}/*
*********************************************************************************************************
*   函 数 名: ee_WriteBytes
*   功能说明: 向串行EEPROM指定地址写入若干数据,采用页写操作提高写入效率
*   形    参:_usAddress1 : 起始高字节地址_usAddress2 : 起始低字节地址
*                       _usSize : 数据长度,单位为字节
*                     _pWriteBuf : 存放读到的数据的缓冲区指针
*   返 回 值: 0 表示失败,1表示成功
*********************************************************************************************************
*/
uint8_t ee_WriteBytes(char *_pWriteBuf, uint8_t _usAddress1,uint8_t _usAddress2, uint16_t _usSize)
{uint16_t i,m;uint8_t usAddr1,usAddr2;/* 写串行EEPROM不像读操作可以连续读取很多字节,每次写操作只能在同一个page。对于24xx02,page size = 32简单的处理方法为:按字节写操作模式,没写1个字节,都发送地址为了提高连续写的效率: 本函数采用page wirte操作。*/usAddr1 = _usAddress1;usAddr2 = _usAddress2;   for (i = 0; i < _usSize; i++){/* 当发送第1个字节或是页面首地址时,需要重新发起启动信号和地址 */if ((i == 0) || (usAddr2 & (EEPROM_PAGE_SIZE - 1)) == 0){/* 第0步:发停止信号,启动内部写操作 */i2c_Stop();/* 通过检查器件应答的方式,判断内部写操作是否完成, 一般小于 10ms             CLK频率为200KHz时,查询次数为30次左右*/for (m = 0; m < 1000; m++){              /* 第1步:发起I2C总线启动信号 */i2c_Start();/* 第2步:发起控制字节,高7bit是地址,bit0是读写控制位,0表示写,1表示读 */i2c_SendByte(EEPROM_DEV_ADDR | EEPROM_I2C_WR); /* 此处是写指令 *//* 第3步:发送一个时钟,判断器件是否正确应答 */if (i2c_WaitAck() == 0){break;}}if (m  == 1000){goto cmd_fail; /* EEPROM器件写超时 */}/* 第4步:发送字节地址,24C02只有8192字节,因此2个字节就够了,如果是24C04以上,那么此处需要连发多个地址 */i2c_SendByte((uint8_t)usAddr1);/* 第5步:等待ACK */if (i2c_WaitAck() != 0){goto cmd_fail;   /* EEPROM器件无应答 */}/* 第6步:发送字节地址,24C02只有8192字节,因此2个字节就够了,如果是24C04以上,那么此处需要连发多个地址 */i2c_SendByte((uint8_t)usAddr2);/* 第7步:等待ACK */if (i2c_WaitAck() != 0){goto cmd_fail;   /* EEPROM器件无应答 */}}/* 第6步:开始写入数据 */i2c_SendByte(_pWriteBuf[i]);/* 第7步:发送ACK */if (i2c_WaitAck() != 0){goto cmd_fail; /* EEPROM器件无应答 */}usAddr2++;  /* 地址增1 */      }/* 命令执行成功,发送I2C总线停止信号 */i2c_Stop();return 1;cmd_fail: /* 命令执行失败后,切记发送停止信号,避免影响I2C总线上其他设备 *//* 发送I2C总线停止信号 */i2c_Stop();return 0;
}void ee_Erase(void)
{uint16_t i;char buf[EEPROM_SIZE];/* 填充缓冲区 */for (i = 0; i < EEPROM_SIZE; i++){buf[i] = 0xFF;}/* 写EEPROM, 起始地址 = 0,数据长度为 256 */if (ee_WriteBytes(buf, 0,0, EEPROM_SIZE) == 0){printf("擦除eeprom出错!\r\n");return;}else{printf("擦除eeprom成功!\r\n");}
}
void lnb_Erase(void)
{uint16_t i;uint8_t buf[LNB_SIZE];/* 填充缓冲区 */for (i = 0; i < LNB_SIZE; i++){buf[i] = 0xFF;}/* 写EEPROM, 起始地址 = 0,数据长度为 256 */if (lnb_WriteBytes(buf, 0, LNB_SIZE) == 0){printf("擦除eeprom出错!\r\n");return;}else{printf("擦除eeprom成功!\r\n");}
}/*--------------------------------------------------------------------------------------------------*/
static void ee_Delay(__IO uint32_t nCount)   //简单的延时函数
{for(; nCount != 0; nCount--);
}
@soft_iic.h文件
#ifndef __I2C_EE_H
#define __I2C_EE_H#include "stm32f4xx.h"/* * BL24C02 2kb = 2048bit = 2048/8 B = 256 B* 256 pages of 32 bytes each** Device Address* 1 0 1 0 A2 A1 A0 R/W* 1 0 1 0 0  0  0  0 = 0XA0* 1 0 1 0 0  0  0  1 = 0XA1 *//* AT24C01/02每页有8个字节 * AT24C04/08A/16A每页有16个字节 */#define EEPROM_DEV_ADDR            0xA0        /* 24xx02的设备地址 */
#define EEPROM_PAGE_SIZE          32     /* 24xx02的页面大小 */
#define EEPROM_SIZE               8192            /* 24xx02总容量 */#define LNB_DEV_ADDR           0x12        /* lnb的设备地址 */
#define LNB_PAGE_SIZE         1           /* lnb的页面大小 */
#define LNB_SIZE                  6           /* lnb总容量 */
static void ee_Delay(__IO uint32_t nCount);
uint8_t ee_CheckOk(void);
uint8_t lnb_CheckOk(void);
uint8_t ee_ReadBytes(char *_pReadBuf, uint8_t _usAddress1,uint8_t _usAddress2,  uint16_t _usSize);
uint8_t lnb_ReadBytes(uint8_t *_pReadBuf, uint8_t _usAddress,  uint16_t _usSize);
uint8_t ee_WriteBytes(char *_pWriteBuf, uint8_t _usAddress1,uint8_t _usAddress2, uint16_t _usSize);
uint8_t lnb_WriteBytes(uint8_t *_pWriteBuf, uint8_t _usAddress, uint16_t _usSize);
void ee_Erase(void);
void lnb_Erase(void);
uint8_t ee_Test(void);
uint8_t lnb_Test(void);
void LNB_Delay(__IO uint32_t nCount);#endif /* __I2C_EE_H */

在这里特别说明一下芯片的设备地址:
数据手册给出了设备地址:从高位到低位=0b 1 0 1 0 A2 A1 A0 R/W
其中,前面已经介绍过了,我给A2、A1、A0在硬件上拉低为0了,硬件设计相关;
所以设备地址前七位就是0b1010000+R/W位。而R/W位即前面说的读写位,读是1,写是0。
所以在进行读操作时,设备地址是0b10100001,即0xA1。
写操作时,设备地址是0b10100000,即0xA0。

  • Device Address(设备地址)
  • 1 0 1 0 A2 A1 A0 R/W
  • 1 0 1 0 0 0 0 0 = 0XA0
  • 1 0 1 0 0 0 0 1 = 0XA1

/***********************END OF FILE/

结尾:

基本上,IIC是一个标准的通信协议,一般可以直接应用上述代码。但像数据寄存器地址位只有8位的,就需要改一下上面的数据寄存器地址部分。同时,也要根据外设的实际时序去做相应修改,有的IIC设备在读写数据的应答信号处,可能略有区别,相信大家沉心静气的研究完IIC协议后,能够很好的应用这个通信协议。


还有,其实单片机的硬件IIC外设也是比较简单的,可能有如网上说的一些不稳定有问题的情况,但应付普通使用环境下,还是基本没问题的。可以使用HAL库驱动IIC硬件外设,这个版本好像很稳定。


感谢阅读,有参考野火哥的代码教程,在这里感谢一下野火哥!

IIC协议的软件模拟实现程序相关推荐

  1. 【STM32】0.96寸OLED显示屏(7针SPI协议)软件模拟SPI

    Author:AXYZdong 自动化专业 工科男 有一点思考,有一点想法,有一点理性 [自制展示]2020鼠年大吉 文章目录 概述 3.1硬件设计 3.2软件设计 3.2.1编程要点 3.2.2代码 ...

  2. PCF8591 ---AD\DA转换器的使用(IIC通信的软件模拟与应用) (by 51单片机)

    依旧是借着蓝桥杯的机会,来学习一些模块,和软件模拟一些通信方式. 今天学习的模块是PCF8591,一款AD\DA转换器,使用IIC通信方式. 芯片的内部框图如下所示: 各引脚作用如下表所示: 仍旧和之 ...

  3. STM32软件模拟IIC---读写驱动AT24Cxx

    以下内容皆是个人学习过程中的总结,记录一下整个过程,用于后期复习,如有不对之处,麻烦各位大佬指出~ (喜欢的朋友麻烦点个关注~~~ 后期还会进行持续更新) 概述 AT24C系列为美国ATMEL公司推出 ...

  4. 西门子 PLC 入门 :如何模拟PLC程序

    ​ 前言: 在本文(第 3 部分)中,您将学习如何模拟上一部分编写的 PLC 程序. 模拟 PLC 程序是我们将程序下载到工厂中的实际 PLC 并使用真实设备和设备对其进行测试之前的最后一步.另一方面 ...

  5. STM32F103单片机软件模拟IIC并读取TMP112数字温度传感器

    本文利用STM32F103系列单片机读取TMP112数字温度传感器的温度信息,TMP112数字传感器采用IIC总线协议通信.STM32自身含有硬件IIC资源,分别是PB6-->SCL.PB7-- ...

  6. 软件模拟IIC主从机

    软件模拟IIC 从机部分 从机接收部分 从机发送部分 主机部分 阻塞式发送 定时器中断方式发送 从机部分 因为项目简单,就只有数据接收,数据命令处理,显示.显示部分使用定时器中断动态扫描方式,主函数用 ...

  7. IIC软件模拟-读写EEPROM

    这里写目录标题 1.IIC简介 2. I2C 基本读写过程 2.1.主机写数据到从机 2.2.主机由从机中读数据 2.3.读和写数据 2.4.地址及数据方向 2.5.响应信号 3.软件模拟I2C 4. ...

  8. 【蓝桥杯嵌入式】【STM32】5_IICEEPROM之软件模拟IIC

    文章目录 1.原理图 2.源代码   下载工程文件:   https://gitee.com/Joseph_Cooper/blue-bridge-embedded 1.原理图 顺便指出了该器件的地址. ...

  9. 单片机软件模拟SPI接口—加深理解SPI总线协议

    单片机软件模拟SPI接口-加深理解SPI总线协议   SPI(Serial Peripheral Interfacer 串行外设接口)是摩托罗拉公司推出的一种同步串行通讯接口,用于微处理器臌控制器和外 ...

最新文章

  1. mysql select 返回列,是否可以对在mysql SELECT语句中返回列的顺序进行排序?
  2. Q45 跳跃游戏 II
  3. Cold-Staking | TPoS vs LPoS vs DPoS
  4. 每天一道LeetCode-----在给定序列中找到满足nums[i]nums[i-1]nums[i]nums[i+1]的位置,要求时间复杂度是O(logN)
  5. CF1486D Max Median
  6. Linux平台安装Clion
  7. 配置IIS Express 7.5以允许外部访问
  8. Java 实现 pdf 和 Excel 的生成及数据动态插入、导出
  9. DNS 攻击方式及攻击案例
  10. 注记字体样式选择指导
  11. Redis开发运维实践开发者设计规范之延迟考虑
  12. 一、 Vue.js简介
  13. 在我眼中的生活的苦难
  14. java 局域网聊天
  15. Linux中etc目录详解大全总汇详解
  16. CAD全称AutoCAD (全系列中文版软件+注册机下载) 2004-2020 安装视频教程
  17. 在excel中如何筛选重复数据_Excel数据筛选操作总结之数据透视表
  18. 【刷爆LeetCode】五月算法集训(14)栈
  19. 前端面试基础题总结 (必会)
  20. Fedora23安装fcitx拼音输入法

热门文章

  1. 2300套php网站源码模板 完整后台程序 整站带数据,ID41:建材自适应PC手机平板微信Seo结构送2300套PHP整站源码...
  2. 盾牌第一至七季/全集The Shield迅雷下载
  3. 大专生可以参加软件测试培训吗
  4. 2018 LinkedSee灵犀首届AIOps峰会首发Linked AIOps
  5. [Knowledge]_[apk signature签名]
  6. 保理业务系统-特点介绍一
  7. 文件从服务器剪切后传输丢失,剪切文件粘贴出错文件丢失数据恢复成功的因素...
  8. v-for循环时,我只需要点击到的元素做出相应反应,其他的元素不变
  9. javascript:定时器多次开启无法停止的问题setInterval
  10. 白鹭引擎之Scroller容器