STM8L051的硬件I2C调试

2019-04-02来源: eefocus关键字:STM8L051  硬件I2C调试

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); // DeInit

  CLK_PeripheralClockConfig(CLK_Peripheral_USART1, ENABLE); // SysClk for USART1

  SYSCFG_REMAPPinConfig(REMAP_Pin_USART1TxRxPortA, ENABLE); // Remap TX on PA2 and RX on PA3

  USART_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调试 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/2019/ic-news040243681.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:总结stm8硬件IIC主模式的寄存器设置及调试心得
下一篇:关于调试SPI、I2C、UART的记录

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

stm8L051低功耗采集电池电量ADC例程
自己记录一下,以防以后用uint  ADC_ReadValue(uchar  channel){     uchar   i;    static uint ADC_Buf[5];    uint vrefint;    ulong adcValue;    CLK_PCKENR2 = CLK_PCKENR2 | 0X01; //使能外设ADC的时钟,之后对ADC的寄存器操作有效    ADC1_CR1 = 0x01; //开启AD模块电源,使用Sing
发表于 2020-03-04
stm8l051的外部中断
stm8l051,如果打开全局中断后再去设置外部中断,在仿真情况下观察寄存器没有被设置,即使直接向中断控制寄存器写值也无效。关闭总中断后,再去设置相应的外部中断,则中断相关寄存器设置成功,程序运行正常。代码如下:disableInterrupts(); EXTI_DeInit();  EXTI_SetPinSensitivity(EXTI_Pin_6, EXTI_Trigger_Falling);  GPIO_Init( GPIOC, GPIO_Pin_6, GPIO_Mode_In_FL_IT );  ITC_SetSoftwarePriority(EXTI6_IRQn
发表于 2020-02-29
STM8L051 RTC写入时间失败
最近用STM8L051做的一个低功耗的项目,需要同步两个节点之间的时间,结果RTC没有在设定的时间点唤醒单片机,最开始以为是晶振的误差太大,偶然间发现两次唤醒的时间间隔和我设定的是一致的,于是排除晶振的原因。把节点的唤醒时间和收到其它节点发来的时间打印后之后发现有时候要经过2-3次才能正确的把接收到的时间信息写入到本节点,于是尝试在写入时间之前再次调用RTC初始化函数void RTC_Mode_Config(void){  CLK_LSEConfig(CLK_LSE_ON);                   
发表于 2020-02-29
STM8L051之低功耗停机配置问题
在做stm8L的小一个项目,由电池供电,当按键长按关机,系统进入停机模式,虽然系统运行的时候有十几毫安的电流消耗(还没使用RTC定时唤醒),但是在停机模式下电流消耗竟然还有1.33ma的电流,花了半天的时间找资料,看文档,在网上看到不少大牛能做到0.4ua,自己无论如何设置IO,外设都无补于事,停机模式下电流最低还有1ma。 最后只能将开机初始化的外设函数一个一个注释掉,包括ADC,DMA,beep,以及TIM4,最后发现只注释掉ADC的初始化函数,再进入停机模式,果然电流接近0,5ua,也就是说可能ADC模块在进入停机模式前的关闭设置不正确,之前的是这样的:void PerCLK_OFF(){  
发表于 2020-01-18
stm8L051使用库建工程
先前直接寄存器开发,不过挺麻烦的,寄存器开发stm8s103直接对着英文技术手册写代码,现在换到了stm8L的又要回头看英文文档,开发效率个人感觉对于我这个新手来说还是很低的。现在在社区下载了一份固件,看看能不能建立一个用库的工程吧。(先前简单看过别人建工程的过程,由于对IAR不熟悉,不成功,现在也就是刚刚弄到没任何错误。)1 先下载一份工程固件stsw-stm8016,固件迟些发送到资源上面,也可以到stm32/stm8社区下载。 2 建立一个iar的工程 这里建立的工程如下(我已经弄好的):   备注:app文件放我们自己编写的BSP驱动,lib放固件,还有三个文件 
发表于 2020-01-15
stm8L051使用库建工程
STM8L051之蜂鸣器beep--库函数版(没有用TIM2校准LSI)
STM8L051的蜂鸣器引脚与swin调试口共用一个引脚,虽然使用了beep就不能使用调试功能(程序下载完成后,断开调试的swin,重启即可输出),但还是可以烧写程序到MCU中的。 (**注意**可以不用设置选项位即可使用beep功能)下面是库函数的代码:void BeepInit(){  // BEEP_LSClockToTIMConnectCmd(ENABLE);   CLK_PeripheralClockConfig(CLK_Peripheral_BEEP,ENABLE);  // BEEP_LSICalibrationConfig(38000);   
发表于 2020-01-15
小广播
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2020 EEWORLD.com.cn, Inc. All rights reserved