STM32 I2C从机发送数据

您所在的位置:网站首页 iic收发数据不一致 STM32 I2C从机发送数据

STM32 I2C从机发送数据

2023-09-28 04:27| 来源: 网络整理| 查看: 265

         

stm32 I2C主机发送功能在网上有很多例程,调试也很容易,从机发送数据的介绍很少,鉴于本人刚开始调试单片机程序不久,花了一整天也没调试通,最后在部门领导的帮助下,通过示波器及单步调试程序,查看寄存器状态值,终于实现了。下面主要介绍下从机发送数据注意事项。(附带例程)

1、相信使用I2C的都知道,从机是不能主动发送数据的,开始条件都是由主机生成。

 

2、根据参考文档,I2C默认是从机模式,只有在生成起始条件后,才会从从机模式自动切换到主机模式。所以作为从机模式使用时不能生成起始条件。

 

3、附图1,为从机发送器的传送时序图,截图于参考手册,图中说明部分讲述了怎样能清除事件,这点很重要,每次事件是都要去清除才可以去检测下一个事件,否则很可能检测不到其他事件,我当初就是死在这上面的。

 

4、需要的数据发送完成了,根据附图1和2可知,主机需要发送一个NACK,从机有NACK应答,当你以为这样可以的时候就大错特错了。文档中有句话,要在读倒数第二个字节之后发送,而非所有字节发送完成再发送。仔细看附图1中说明部分[EV3:TxE=1,写DR将清除该事件;移位寄存器非空。],如果不清除该事件的话,那对不起,你根本就检测不到EV3-2事件。所以在需要传输的数据传输完时,再向DR寄存器中写入一个数据后发送NACK就行了,这个数据是我们不需要的,写什么不重要,目的是为了清除事件(EV3)罢了(也可以在最后一个数据之前发送NACK,就不需要写入无用数据了,这里没有测试其他办法就不赘述了)

 

 

附图1:从发送器的传送序列图

附图2: 主接收器传送序列图

 

 

附bsp_i2c.c源码:

/**  ******************************************************************************

  * @file   bsp_i2c.c

  * @author Eason

  * @version V1.0

  * @date   2014-09-15

  * @brief  I2C应用函数接口

 ******************************************************************************

  * @attention

  *

  * 实验平台:野火 iSO-MINISTM32 开发板

  * 固件库版本V3.5

 *****************************************************************************

  */

 

#include"bsp_i2c.h"

 

/* Private typedef-----------------------------------------------------------*/

typedef enum{FAILED = 0, PASSED = !FAILED} TestStatus;

 

/* Private define------------------------------------------------------------*/

#defineI2C1_SLAVE_ADDRESS7     0x30

#defineI2C2_SLAVE_ADDRESS7     0x24

#defineBufferSize              4

#defineClockSpeed              300000

 

/* Privatevariables ---------------------------------------------------------*/

u8I2C1_Buffer_Tx[BufferSize] = {1,20,50,100};

u8I2C2_Buffer_Rx[BufferSize] = {0};

u8 TxIdx = 0,RxIdx = 0;

volatileTestStatus TransferStatus = FAILED;

 

/******************************************************************************

* FunctionName  : I2C_GPIO_Config

* Description     : Configuresthe different GPIO ports.

* Input          : None

* Output         : None

* Return         : None

******************************************************************************/

void I2C_GPIO_Config(void)

{

       GPIO_InitTypeDefGPIO_InitStructure;

 

         /* Enable peripheral clocks--------------------------------------------------*/

       /*GPIOB Periph clock enable */

       RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

       /*I2C1 and I2C2 Periph clock enable */

       RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1| RCC_APB1Periph_I2C2, ENABLE);

 

       /*Configure I2C1 pins: SCL and SDA ----------------------------------------*/

       GPIO_InitStructure.GPIO_Pin=  GPIO_Pin_6 | GPIO_Pin_7; //选择待设置的GPIO管脚

       GPIO_InitStructure.GPIO_Speed= GPIO_Speed_50MHz;      //管脚速率50MHz

       GPIO_InitStructure.GPIO_Mode= GPIO_Mode_AF_OD;  //复用开漏输出

       GPIO_Init(GPIOB,&GPIO_InitStructure);

 

       /*Configure I2C2 pins: SCL and SDA ----------------------------------------*/

       GPIO_InitStructure.GPIO_Pin= GPIO_Pin_10 | GPIO_Pin_11;

       GPIO_Init(GPIOB,&GPIO_InitStructure);

}

 

/******************************************************************************

* FunctionName  : I2C_Master_Init

* Description    : I2C Master initialize.

* Input          : None

* Output         : None

* Return         : None

******************************************************************************/

voidI2C_Master_Init(void)

{

         I2C_InitTypeDef  I2C_InitStructure;

         /* I2C1 configuration------------------------------------------------------*/

       I2C_InitStructure.I2C_Mode= I2C_Mode_I2C;

       I2C_InitStructure.I2C_DutyCycle= I2C_DutyCycle_2;

       I2C_InitStructure.I2C_OwnAddress1= I2C1_SLAVE_ADDRESS7;

       I2C_InitStructure.I2C_Ack= I2C_Ack_Enable;

       I2C_InitStructure.I2C_AcknowledgedAddress= I2C_AcknowledgedAddress_7bit;

       I2C_InitStructure.I2C_ClockSpeed= ClockSpeed;

       I2C_Init(I2C1,&I2C_InitStructure);

 

         I2C_Cmd(I2C1, ENABLE);

}

 

/******************************************************************************

* FunctionName  : I2C_Slave_Init

* Description     :I2C Slave initialize.

* Input          : None

* Output         :None

* Return         : None

******************************************************************************/

voidI2C_Slave_Init(void)

{

         I2C_InitTypeDef  I2C_InitStructure;

       /*I2C2 configuration ------------------------------------------------------*/

         I2C_InitStructure.I2C_Mode =I2C_Mode_I2C;

       I2C_InitStructure.I2C_DutyCycle= I2C_DutyCycle_2;

       I2C_InitStructure.I2C_OwnAddress1= I2C2_SLAVE_ADDRESS7;

       I2C_InitStructure.I2C_Ack= I2C_Ack_Enable;

       I2C_InitStructure.I2C_AcknowledgedAddress= I2C_AcknowledgedAddress_7bit;

       I2C_InitStructure.I2C_ClockSpeed= ClockSpeed;

       I2C_Init(I2C2,&I2C_InitStructure);

        

         I2C_Cmd(I2C2, ENABLE);

}

 

voidI2C_Mode_Init(void)

{

         I2C_GPIO_Config();

         I2C_Master_Init();

         I2C_Slave_Init();

}

/******************************************************************************

* FunctionName  : Buffercmp

* Description    : Compares two buffers.

* Input          : - pBuffer1, pBuffer2: buffers to becompared.

*                : - BufferLength: buffer'slength

* Output         : None

* Return         : PASSED: pBuffer1 identical topBuffer2

*                  FAILED: pBuffer1 differs frompBuffer2

******************************************************************************/

TestStatusBuffercmp(u8* pBuffer1, u8* pBuffer2, u16 BufferLength)

{

  while(BufferLength--)

  {

    if(*pBuffer1 != *pBuffer2)

    {

      return FAILED;

    }

   

    pBuffer1++;

    pBuffer2++;

  }

 

  return PASSED; 

}

 

/*******************************************************************************

* FunctionName  : I2C_Transmission

* Description    : Data Transmission.

* Input          : None

* Output         : None

* Return         : None

******************************************************************************/

voidI2C_Transmission(void)

{

  I2C_GenerateSTART(I2C1, ENABLE);

  while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT));  

  /* Send I2C2 slave Address for read*/

  I2C_Send7bitAddress(I2C1,I2C2_SLAVE_ADDRESS7, I2C_Direction_Receiver);

 

    /* Test on I2C2 EV1 and clear it */

  while(!I2C_CheckEvent(I2C2,I2C_EVENT_SLAVE_TRANSMITTER_ADDRESS_MATCHED));

    /* Test on I2C1 EV6 and clear it */

  while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));

 

  /* Send data */

  while (RxIdx < BufferSize)

  {

    /* Send I2C2 data */

    I2C_SendData(I2C2,I2C1_Buffer_Tx[TxIdx++]);

   /* Test on I2C1 EV7 and clear it */

         while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_RECEIVED));

         while(!I2C_CheckEvent(I2C2,I2C_EVENT_SLAVE_BYTE_TRANSMITTED)); 

    /* Store received data on I2C1 */

    I2C2_Buffer_Rx[RxIdx++] =I2C_ReceiveData(I2C1);         

  }

  I2C_SendData(I2C2, 0);  //此处写DR寄存器目的是为了清除EV3事件

 

  RxIdx = 0;

  TxIdx = 0;

  I2C_AcknowledgeConfig(I2C1,DISABLE); //主机发送NACK 

  while(!I2C_CheckEvent(I2C2,I2C_EVENT_SLAVE_ACK_FAILURE));

  while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_RECEIVED));

    /*Clear I2C2 AF flag */

  I2C_ClearFlag(I2C2, I2C_FLAG_AF);

 

  /* Send I2C1 STOP Condition */

  I2C_GenerateSTOP(I2C1, ENABLE);

 

  /* Check the corectness of written data */

  TransferStatus = Buffercmp(I2C1_Buffer_Tx,I2C2_Buffer_Rx,BufferSize);

}

 



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3