I2C是现代一种极为常见的低速外设通信协议,比起SPI或者UART,它最大的优势应该就是节省芯片管脚了:理论上只要地址够用,多少外设挂I2C总线上都没问题,只占两个管脚。但也因此,I2C的协议就相对复杂一些,以面对多个外设。同时,过多的外设也使得通信速率难以提升,一般只在100kbps或以下。本文不专门介绍I2C的时序和协议,而介绍我在调试STM8L051的硬件I2C的过程以及遇到的问题,和大家分享。

我的实验电路由两个独立的STM8L051模块组成,做一发一收。这两个模块的电路是我自己设计的,通过排针插在面包板上,如图所示。这两个芯片的硬件I2C在PC0和PC1,将他们连起来并用4.7K电阻上拉(请原谅我没有直插电阻然后用贴片凑合的无奈T T)。右边是做接收的模块,将它的串口接出好观察结果(我非常喜欢用串口调试,几乎拿到什么板子,第一件事情就是把串口先调出来)。

首先是用库函数进行开发还是直接写寄存器编程的问题。因为懒,我个人更喜欢用库函数,这次调试也是用库函数编程。其实我感觉意法半导体的单片机(尤其STM32)能够流行,其设计合理的库函数是一个关键原因。另外ST官方也为库函数写了大量的例程,使得参考和移植都会方便。但使用库函数其实会运行很多没必要的代码,以及各种函数调用,都会耗费时间和存储资源,在资源本身就紧张8位单片机上用库函数其实是很低效的。我一个师兄表示他在STM8上一直都是直接写寄存器。
ST官方库函数的例程中,有两个板子对通的程序,他的设计是,先进行主机发送、从机接收,然后从机发送,主机接收,主机收回后比较数据,判断传输是否有误。为了方便研究,我将例程分开,分别测试主发从收和主收从发两个过程。

一、 主机发送,从机接收

为了观看传输结果,我会事先配置串口。串口配置程序和串口输出字符串程序如下:

void USART_Config(void)
{USART_DeInit(USART1); // DeInitCLK_PeripheralClockConfig(CLK_Peripheral_USART1, ENABLE); // SysClk for USART1SYSCFG_REMAPPinConfig(REMAP_Pin_USART1TxRxPortA, ENABLE); // Remap TX on PA2 and RX on PA3USART_Init(USART1, (uint32_t)9600, USART_WordLength_8b, USART_StopBits_1, \USART_Parity_No, USART_Mode_Tx);USART_Cmd(USART1, ENABLE);
}
void UART_SendStr(char *str)
{int i = 0;for(i=0;str[i]!=0;i++){while (!(USART1->SR & 0x80));   /* wait for READY */USART_SendData8(USART1,str[i]);}
}

程序中串口被remap到了PA2和PA3,这主要是因为STM8L051芯片没有PC2和PC3,所以必须remap。波特率设为9600,只进行输出,不提供中断。

STM8L的硬件I2C在其参考手册RM0031中有详细的叙述(https://www.st.com/content/ccc/resource/technical/document/reference_manual/2e/3b/8c/8f/60/af/4b/2c/CD00218714.pdf/files/CD00218714.pdf/jcr:content/translations/en.CD00218714.pdf )。为了方便,我只实现7位地址的I2C通信。
在主发从收通信中,主机会遇到的事件包括EV5(发送完START bit)、EV6(发送完从机地址并收到ACK)、EV8(TXE,发送寄存器空,即发送了一个字节)和EV8_2(发送完成)。主机在中断中处理这些问题。I2C设置代码如下:

void I2C_Config(void)
{CLK_PeripheralClockConfig(CLK_Peripheral_I2C1, ENABLE);I2C_DeInit(I2C1);I2C_Init(I2C1, 100000, 0xA0,I2C_Mode_I2C, I2C_DutyCycle_2,I2C_Ack_Enable, I2C_AcknowledgedAddress_7bit);I2C_ITConfig(I2C1, (I2C_IT_TypeDef)(I2C_IT_EVT | I2C_IT_BUF), ENABLE);
}

要发送数据时,I2C先发送开始符号,然后等待发送完成:

I2C_GenerateSTART(I2C1, ENABLE); // Start and into Master Mode
while(NumOfBytes); // Wait for all bytes have been transmitted

主机的中断服务程序在官方样例基础上缩减:

#define SLAVE_ADDRESS 0x30
__IO uint8_t TxBuffer[32] = "Get it!\n";
__IO uint8_t NumOfBytes = 9;
__IO uint8_t Tx_Idx =0;
INTERRUPT_HANDLER(I2C1_SPI2_IRQHandler,29)
{switch (I2C_GetLastEvent(I2C1)){/* EV5 */case I2C_EVENT_MASTER_MODE_SELECT :/* Send slave Address for write */I2C_Send7bitAddress(I2C1, SLAVE_ADDRESS, I2C_Direction_Transmitter);break;/* EV6 */case I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED:if (NumOfBytes != 0){/* Send the first Data */I2C_SendData(I2C1, TxBuffer[Tx_Idx++]);/* Decrement number of bytes */NumOfBytes--;}if (NumOfBytes == 0){I2C_ITConfig(I2C1, I2C_IT_BUF, DISABLE);}break;/* EV8 */case I2C_EVENT_MASTER_BYTE_TRANSMITTING:/* Transmit Data */I2C_SendData(I2C1, TxBuffer[Tx_Idx++]);/* Decrement number of bytes */NumOfBytes--;if (NumOfBytes == 0){I2C_ITConfig(I2C1, I2C_IT_BUF, DISABLE);}break;/* EV8_2 */case I2C_EVENT_MASTER_BYTE_TRANSMITTED:/* Send STOP condition */I2C_GenerateSTOP(I2C1, ENABLE);I2C_ITConfig(I2C1, I2C_IT_EVT, DISABLE);break;default:break;}
}

从机的接收过程更为简单。从机设置时将I2C地址设置为0x30(主机向0x30发送信息),其他和主机相同,然后开启中断等待即可。从机接收过程中遇到的事件有EV1(收到主机发送的本机地址)、EV2(收到一个字节数据)和EV4(停止传输)。其中断服务程序也就是为这些事件准备的:

__IO uint8_t Slave_Buffer_Rx[32];
__IO uint8_t Rx_Idx = 0;
__IO uint16_t Event = 0x00;
uint8_t RecvFlag = 0;INTERRUPT_HANDLER(I2C1_SPI2_IRQHandler,29)
{Event = I2C_GetLastEvent(I2C1);switch (Event){/******* Slave transmitter ******//* check on EV1 */case I2C_EVENT_SLAVE_TRANSMITTER_ADDRESS_MATCHED:break;/* check on EV3 */case I2C_EVENT_SLAVE_BYTE_TRANSMITTING:break;/******* Slave receiver **********//* check on EV1*/case I2C_EVENT_SLAVE_RECEIVER_ADDRESS_MATCHED:break;/* Check on EV2*/case I2C_EVENT_SLAVE_BYTE_RECEIVED:Slave_Buffer_Rx[Rx_Idx++] = I2C_ReceiveData(I2C1);break;/* Check on EV4 */case (I2C_EVENT_SLAVE_STOP_DETECTED):/* write to CR2 to clear STOPF flag */I2C1->CR2 |= I2C_CR2_ACK;RecvFlag = 1;break;default:break;}
}

程序中,我用RecvFlag标记接收完成,主程序在RecvFlag为1时,将收到的字符串从串口发出。主机发来的是“Get it!\n”,从串口看到结果如下图所示(发送了2次):

二、主机接收,从机发送

官方例程中的主机接收代码没有用中断,我这里也如此操作,以后有时间再试试主机中断接收。主机设置代码为:

void I2C_Config(void)
{CLK_PeripheralClockConfig(CLK_Peripheral_I2C1, ENABLE);I2C_DeInit(I2C1);I2C_Init(I2C1, 100000, 0xA0,I2C_Mode_I2C, I2C_DutyCycle_2,I2C_Ack_Enable, I2C_AcknowledgedAddress_7bit);
}

在主机接收过程中,其实传输进程还是主机控制的。在开始传输后,经历EV5、EV6、EV7(主机接收到从机一个字节数据)和EV7_1(主机接收从机最后一个字节),在main()函数中运行如下代码来传输:

while (I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY));
I2C_GenerateSTART(I2C1, ENABLE); // Start and into Master Mode
/* Test on EV5 and clear it */
while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
/* Send slave Address for write */
I2C_Send7bitAddress(I2C1, SLAVE_ADDRESS, I2C_Direction_Receiver);
/* Test on EV6 and clear it */
while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));
/* While there is data to be read */
while(NumOfBytes)
{/* The last bytes need STOP but not ACK */if (NumOfBytes == 1){/* Disable Acknowledgement */I2C_AcknowledgeConfig(I2C1, DISABLE);/* Send STOP Condition */I2C_GenerateSTOP(I2C1, ENABLE);/* Poll on RxNE Flag */while ((I2C_GetFlagStatus(I2C1, I2C_FLAG_RXNE) == RESET));/* Read a byte */RxBuffer[Rx_Idx++] = I2C_ReceiveData(I2C1);/* Decrement the read bytes counter */NumOfBytes--;}/* Test on EV7 and clear it */if (I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED) ){/* Read a byte */RxBuffer[Rx_Idx++] = I2C_ReceiveData(I2C1);/* Decrement the read bytes counter */NumOfBytes--;}
}

代码中,在收到最后一个字节(NumOfBytes == 1)时将ACK Disable并发送STOP结束传输过程。

在从机发射端,依然使用中断来处理发送过程,从机设置和前一节相同。从机发射要经历EV1、EV3(TXE=1,发送寄存器空,发送了一个字节数据)和EV3_2(AF=1,未收到ACK)中断处理代码如下:

INTERRUPT_HANDLER(I2C1_SPI2_IRQHandler,29)
{
/* check on EV3_2 */
if (I2C_ReadRegister(I2C1, I2C_Register_SR2)){/* Clears SR2 register */I2C1->SR2 = 0;}
Event = I2C_GetLastEvent(I2C1);switch (Event){/******* Slave transmitter ******//* check on EV1 */case I2C_EVENT_SLAVE_TRANSMITTER_ADDRESS_MATCHED:Tx_Idx = 0;break;/* check on EV3 */case I2C_EVENT_SLAVE_BYTE_TRANSMITTING:I2C_SendData(I2C1, Slave_Buffer_Tx[Tx_Idx++]);break;/******* Slave receiver **********//* check on EV1*/case I2C_EVENT_SLAVE_RECEIVER_ADDRESS_MATCHED:break;/* Check on EV2*/case I2C_EVENT_SLAVE_BYTE_RECEIVED:break;/* Check on EV4 */case (I2C_EVENT_SLAVE_STOP_DETECTED):break;default:break;}
}

STM8L的从机发送的结束机制值得好好吐槽一下。我看到有网络上帖子说主收从发只能收一次,我之前也删掉了if (I2C_ReadRegister(I2C1, I2C_Register_SR2))这个判断,因为I2C_SR2其实是个错误寄存器,我想传输没错误的话应该就不用管它了,然后就只能传一次。直到再次读手册RM0031,看到 EV3-2: AF=1, AF is cleared by writing ‘0’ in AF bit of SR2 register.这句话,AF是Acknowledge Failure,也就是说,它其实是根据没收到ACK来判断传输结束的……将这个寄存器清零后,硬件I2C恢复初始状态。
最后验证,从机发送Got it!\n,主机收到发送到电脑上结果为:

STM8L051的硬件I2C调试相关推荐

  1. STM32L系列+ADXL345的I2C调试

    STM32的硬件I2C调试确实要比模拟麻烦很多啊,一大堆的配置,调通F系列的,本以为直接移植到L系列会很轻松,没想到问题依然很多,现直接附上STM32L系列的I2C初始化及读写函数: 1.GPIO的初 ...

  2. vb6编写dll读取dat文件_【STM32Cube_15】使用硬件I2C读取温湿度传感器数据(SHT30)...

    寻求更好的阅读体验,请移步Mculover666的个人博客: [STM32Cube_15]使用硬件I2C读取温湿度传感器数据(SHT30)​www.mculover666.cn 本篇详细的记录了如何使 ...

  3. 使用MPU6050在STM32F103C8T6中的硬件I2C数据传输

    文章以自学为主,主要用来保存学习记录,有问题大家可以互相讨论,一起进步 使用MPU6050在STM32F103C8T6中的硬件I2C数据传输 前言 一.使用的硬件平台 二.程序移植步骤 1.引入野火官 ...

  4. 基于阿里平头哥的单片机软、硬件i2C驱动oled

    基于RiskV的阿里平头哥MCU开箱文章之软硬件i2C驱动oled 12864 Risk-V简介 国产单片机及开发环境CDK iic简介 iIC代码移植详细介绍 # stm32 iic: # 向国产单 ...

  5. I2C调试遇到的问题以及解决办法---“busy死锁状态”以及“从设备地址无应答”问题

    stm32f103c8t6 cubemx 关于I2C通信不通的若干问题解决方法 i2c调试过程中出现BUSY状态 i2c调试过程中,向从设备发送地址,从设备无应答信号传给主设备 总结 i2c调试过程中 ...

  6. nRF52832之硬件I2C

    这几天一直在折腾nRF52832的硬件I2C,到了今天终于出现了成果,在此也印证了那句话:"耕耘就有收获" 52832的硬件I2C虽然官方提供了demo,但是自己对I2C通信理解的 ...

  7. STM32通信硬件 I2C

    20.1关于 I2C STM32F103系列的I²C控制器,可作为通信主机或从机,因此有四种工作模式可选择:主机发送模式.主机接收模式.从机发送模式.从机接收模式. 传输速度上,支持标准模式(Stan ...

  8. STM32HAL库学习笔记--硬件I2C读写AT24C512

    摘要:由于需要实现掉电存储功能,在无线传输模块上增加了一块EEPROM芯片,因为工程使用CubeMx配置并生成,为了方便不再使用IO口模拟I2C,而是使用f1自带的硬件I2C来实现.配置和调试过程和遇 ...

  9. TMS320F28069的硬件I2C一直卡在查询 I2caRegs.I2CSTR.bit.XRDY

    项目场景: 使用TMS320F28069的硬件I2C读取MPU6050的数据 问题描述 Uint16 I2CA_ReadBytes( Uint16 SlaveAddress, Uint16 RomAd ...

最新文章

  1. 00后的简历有多野?!!
  2. 关于base target=_self 等
  3. hotmail在outlook2007中的设置
  4. mysql的事物隔离级别
  5. 解决git push 中remote: Permission to xxxxx.git denied to xxx. fatal: unable to access xxxx 403(转)
  6. jmeter中重定向多个正则表达式_2020年jmeter技术实战续集,最新技术全栈,值得收藏
  7. Codeforces Round #256 (Div. 2)
  8. realme X50将于年前亮相:支持双模5G+120Hz刷新率屏幕
  9. 基于JAVA+SpringMVC+MYSQL的学生成绩管理系统
  10. python线程同步
  11. 周志华:AAAI 2019论文提交创纪录,达到7745篇
  12. AIX 64位内核与32位内核区别
  13. JavaScript - textarea 滚动至顶部或底部
  14. 【光学】基于matlab实现圆孔的菲涅尔衍射仿真
  15. 阿铭Linux_总览部分学习笔记20190114
  16. Identifying App Installations
  17. 计算机组成原理随堂作业,2019-华南理工-计算机组成原理-随堂作业
  18. FreeRTOS 事件标志组
  19. 雷锋微视点:腾讯电商、搜搜、微信将出门独立
  20. jumpserver版本升级(1.3.1升级到最新版)

热门文章

  1. 软件工程小项目~企业员工信息管理系统-需求分析~~
  2. 提莫队长正在待命(DP)
  3. 【阅读笔记】Deep Mutual Learning
  4. ubuntu安装软件包命令_从Ubuntu命令行搜索安装软件包
  5. 浏览器工作原理:浅析HTTP请求流程
  6. linux运维(九)
  7. 马德里的Uber司机
  8. Error:间接寻址级别不同——C++真的魔鬼
  9. NOIP模拟题 通讯 强连通分量缩点 最小树形图--朱刘算法
  10. 贝叶斯法则与虚假阳性的病例