STM8L051的硬件I2C调试

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

我的实验电路由两个独立的STM8L051模块组成,做一发一收。这两个模块的电路是我自己设计的,通过排针插在面包板上,如图所示。这两个芯片的硬件I2C在PC0和PC1,将他们连起来并用4.7K电阻上拉(请原谅我没有直插电阻然后用贴片凑合的无奈T T)。右边是做接收的模块,将它的串口接出好观察结果(我非常喜欢用串口调试,几乎拿到什么板子,第一件事情就是把串口先调出来)。
STM8L051的硬件I2C调试_第1张图片
首先是用库函数进行开发还是直接写寄存器编程的问题。因为懒,我个人更喜欢用库函数,这次调试也是用库函数编程。其实我感觉意法半导体的单片机(尤其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次):
STM8L051的硬件I2C调试_第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调试_第3张图片

你可能感兴趣的:(单片机应用)