9.基于STM32CubeMx软件模拟I2C通信

使用:

		MEM_Write (ReceiveString,0x01,4);
		HAL_Delay (1000);
		MEM_Read (I2CString,0x01,4);
由于24C02分为8Byte为一页,不能连续跨页写数据,如果写的数据大于8比特或者跨页了,应分为多段进行写入。且写入之间要有延时,否则后面的不能写入,如:
uint8_t SendBuf[21]="13981422143139814221";
uint8_t ReceiveBuf[21]="";

	MEM_Write(SendBuf,0x00,8);
	HAL_Delay(100);//延时一定要有。
	MEM_Write(SendBuf+8,0x08,8);
	HAL_Delay (1000);
	MEM_Read(ReceiveBuf,0x00,8);
	HAL_Delay(100);
	MEM_Read(ReceiveBuf+8,0x08,8);
	LCD_DisplayStringLine (Line3,ReceiveBuf);

驱动程序(最后两个自己编写,其余底层驱动官方提供)

#include "i2c.h"

//#define DELAY_TIME	20
#define DELAY_TIME 20

//PB6_SCL
//PB7_SDA
/**
  * @brief SDA线输入模式配置
  * @param None
  * @retval None
  */
void SDA_Input_Mode()
{
    GPIO_InitTypeDef GPIO_InitStructure = {0};

    GPIO_InitStructure.Pin = GPIO_PIN_7;
    GPIO_InitStructure.Mode = GPIO_MODE_INPUT;
    GPIO_InitStructure.Pull = GPIO_PULLUP;
    GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStructure);
}

/**
  * @brief SDA线输出模式配置
  * @param None
  * @retval None
  */
void SDA_Output_Mode()
{
    GPIO_InitTypeDef GPIO_InitStructure = {0};

    GPIO_InitStructure.Pin = GPIO_PIN_7;
    GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_OD;
    GPIO_InitStructure.Pull = GPIO_NOPULL;
    GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStructure);
}

/**
  * @brief SDA线输出一个位
  * @param val 输出的数据
  * @retval None
  */
void SDA_Output( uint16_t val )
{
    if ( val )
    {
        GPIOB->BSRR |= GPIO_PIN_7;
    }
    else
    {
        GPIOB->BRR |= GPIO_PIN_7;
    }
}

/**
  * @brief SCL线输出一个位
  * @param val 输出的数据
  * @retval None
  */
void SCL_Output( uint16_t val )
{
    if ( val )
    {
        GPIOB->BSRR |= GPIO_PIN_6;
    }
    else
    {
        GPIOB->BRR |= GPIO_PIN_6;
    }
}

/**
  * @brief SDA输入一位
  * @param None
  * @retval GPIO读入一位
  */
uint8_t SDA_Input(void)
{
	if(HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7) == GPIO_PIN_SET){
		return 1;
	}else{
		return 0;
	}
}


/**
  * @brief I2C的短暂延时
  * @param None
  * @retval None
  */
static void delay1(unsigned int n)
{
    uint32_t i;
    for ( i = 0; i < n; ++i);
}

/**
  * @brief I2C起始信号
  * @param None
  * @retval None
  */
void I2CStart(void)
{
    SDA_Output(1);
    delay1(DELAY_TIME);
    SCL_Output(1);
    delay1(DELAY_TIME);
    SDA_Output(0);
    delay1(DELAY_TIME);
    SCL_Output(0);
    delay1(DELAY_TIME);
}

/**
  * @brief I2C结束信号
  * @param None
  * @retval None
  */
void I2CStop(void)
{
    SCL_Output(0);
    delay1(DELAY_TIME);
    SDA_Output(0);
    delay1(DELAY_TIME);
    SCL_Output(1);
    delay1(DELAY_TIME);
    SDA_Output(1);
    delay1(DELAY_TIME);

}

/**
  * @brief I2C等待确认信号
  * @param None
  * @retval None
  */
unsigned char I2CWaitAck(void)
{
    unsigned short cErrTime = 5;
    SDA_Input_Mode();
    delay1(DELAY_TIME);
    SCL_Output(1);
    delay1(DELAY_TIME);
    while(SDA_Input())
    {
        cErrTime--;
        delay1(DELAY_TIME);
        if (0 == cErrTime)
        {
            SDA_Output_Mode();
            I2CStop();
            return ERROR;
        }
    }
    SDA_Output_Mode();
    SCL_Output(0);
    delay1(DELAY_TIME);
    return SUCCESS;
}

/**
  * @brief I2C发送确认信号
  * @param None
  * @retval None
  */
void I2CSendAck(void)
{
    SDA_Output(0);
    delay1(DELAY_TIME);
    delay1(DELAY_TIME);
    SCL_Output(1);
    delay1(DELAY_TIME);
    SCL_Output(0);
    delay1(DELAY_TIME);

}

/**
  * @brief I2C发送非确认信号
  * @param None
  * @retval None
  */
void I2CSendNotAck(void)
{
    SDA_Output(1);
    delay1(DELAY_TIME);
    delay1(DELAY_TIME);
    SCL_Output(1);
    delay1(DELAY_TIME);
    SCL_Output(0);
    delay1(DELAY_TIME);

}

/**
  * @brief I2C发送一个字节
  * @param cSendByte 需要发送的字节
  * @retval None
  */
void I2CSendByte(unsigned char cSendByte)
{
    unsigned char  i = 8;
    while (i--)
    {
        SCL_Output(0);
        delay1(DELAY_TIME);
        SDA_Output(cSendByte & 0x80);
        delay1(DELAY_TIME);
        cSendByte += cSendByte;
        delay1(DELAY_TIME);
        SCL_Output(1);
        delay1(DELAY_TIME);
    }
    SCL_Output(0);
    delay1(DELAY_TIME);
}

/**
  * @brief I2C接收一个字节
  * @param None
  * @retval 接收到的字节
  */
unsigned char I2CReceiveByte(void)
{
    unsigned char i = 8;
    unsigned char cR_Byte = 0;
    SDA_Input_Mode();
    while (i--)
    {
        cR_Byte += cR_Byte;
        SCL_Output(0);
        delay1(DELAY_TIME);
        delay1(DELAY_TIME);
        SCL_Output(1);
        delay1(DELAY_TIME);
        cR_Byte |=  SDA_Input();
    }
    SCL_Output(0);
    delay1(DELAY_TIME);
    SDA_Output_Mode();
    return cR_Byte;
}

//
void I2CInit(void)
{
		GPIO_InitTypeDef GPIO_InitStructure = {0};

    GPIO_InitStructure.Pin = GPIO_PIN_7 | GPIO_PIN_6;
    GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStructure.Pull = GPIO_PULLUP;
    GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStructure);
}
void MEM_Write(unsigned char *pucBuf,unsigned char ucAddr,unsigned char ucNum)
{
		I2CStart();
		I2CSendByte(0xA0);
		I2CWaitAck();
	
		I2CSendByte (ucAddr);
		I2CWaitAck();
	
		for(uint8_t i=0;i<ucNum;i++)
		{
			I2CSendByte (pucBuf[i]);
				I2CWaitAck();
		}
		I2CStop();
	
}
void MEM_Read(unsigned char *pucBuf,unsigned char ucAddr,unsigned char ucNum)
{
		I2CStart();
		I2CSendByte(0xA0);
		I2CWaitAck();
	
		I2CSendByte (ucAddr);
		I2CWaitAck();
	
		I2CStart();
		I2CSendByte(0xA1);
		I2CWaitAck();
	
		for(uint8_t i=0;i<ucNum;i++)
		{
			pucBuf[i]=I2CReceiveByte ();
			if(i==ucNum-1)
				I2CSendNotAck();
			else
				I2CSendAck();
		}
		I2CStop();
	
}

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: STM32模拟I2C驱动OLED可以通过软件模拟I2C总线来实现。使用STM32GPIO引脚模拟I2C通信的时钟线(SCL)和数据线(SDA),通过编程控制引脚的输入输出状态来模拟I2C总线的通信。 首先,需要在STM32的开发环境中配置相关的GPIO引脚作为模拟I2C总线的SCL和SDA引脚。然后,在代码中初始化这些引脚的状态,并设置为输出模式。 在实现模拟I2C驱动之前,我们需要了解OLED的通信协议。通常,OLED使用I2C协议进行通信,包括启动信号、地址字节、数据字节等。根据OLED的数据手册,我们可以编写相应的代码来模拟这些通信步骤。 在驱动程序中,需要实现I2C的启动、发送字节、接收字节、停止等操作。这些操作可以通过控制SCL和SDA引脚的电平来实现。例如,启动信号可以通过将SCL引脚置高,然后将SDA引脚从高电平切换到低电平来发送。 然后,我们可以编写一系列函数来实现模拟I2C通信。例如,写入数据的函数可以使用引脚的输入输出状态来模拟发送数据字节,读取数据的函数可以根据引脚电平模拟接收数据字节。这些函数可以在程序中被调用以实现与OLED的通信。 最后,我们可以将模拟I2C驱动与OLED的显示代码结合起来,通过模拟I2C总线来控制OLED的初始化、显示内容等操作。这样,我们就能够使用STM32来驱动模拟I2C的OLED显示屏了。 总之,通过在STM32模拟I2C总线的通信,我们可以实现驱动OLED的功能。这需要对STM32GPIO引脚进行适当配置和编程,以模拟I2C通信的时序和协议。 ### 回答2: STM32是一款广泛应用于嵌入式系统开发的微控制器芯片。而模拟I2C驱动是一种基于软件实现的I2C通信协议的方式,可以用来驱动OLED显示屏。 在使用STM32模拟I2C驱动OLED时,首先需要配置GPIO引脚作为模拟I2C的SCL和SDA线。通常,SCL线连接到芯片的时钟输入引脚,SDA线连接到芯片的数据输入/输出引脚。 然后,需要实现一些软件函数来模拟I2C通信。对于起始信号的生成,可以通过将SCL和SDA引脚置为高电平,再将SDA引脚置为低电平来实现。发送数据时,将数据按照I2C协议的要求进行处理,依次将每个数据位发送出去,并等待接收端返回的ACK信号。接收数据时,根据I2C协议的要求,接收方会返回一个ACK信号,将接收到的数据保存起来。 在驱动OLED显示屏时,可以根据显示屏的数据手册,实现针对OLED的控制命令或数据的发送函数。通过模拟I2C驱动,可以将这些命令或数据发送到OLED中,实现对显示屏内容的控制。 需要注意的是,模拟I2C驱动与硬件I2C驱动相比,由于需要通过软件对时序进行精确的控制,所以在速度和准确度上可能会有所降低。因此,在实际应用中,需根据具体要求进行优化。 总之,通过STM32模拟I2C驱动OLED显示屏,可以实现对显示内容的控制和数据的发送。这种软件实现的方式,适用于一些不支持硬件I2C接口的情况,并且灵活度较高,可以满足不同应用的需求。 ### 回答3: STM32是一种微控制器系列,它具有强大的功能和广泛的应用领域。在实现模拟I2C驱动OLED时,我们可以使用STM32GPIO软件I2C库。 首先,我们需要配置STM32GPIO引脚,用于连接到OLED的SCL(时钟)和SDA(数据)线。我们可以使用CubeMX或编写代码手动配置这些引脚。 接下来,我们需要实现软件I2C的功能。软件I2C是通过控制GPIO引脚的电平变化来模拟硬件I2C的数据传输。可以在STM32的库中找到适用于软件I2C的函数和宏。 在驱动OLED之前,我们需要确定OLED的通信协议和寄存器。通常,OLED使用I2C作为通信接口,并具有一组寄存器用于控制显示内容和属性。 为了驱动OLED,我们需要编写一些函数来发送和接收数据。发送数据时,我们将数据字节写入SDA引脚,并在SCL引脚上产生时钟脉冲。接收数据时,我们将SDA引脚配置为输入,并在SCL引脚上读取时钟脉冲。 最后,我们可以编写一些函数来初始化OLED并控制其显示内容。这些函数将使用软件I2C发送特定的命令和数据字节来设置OLED的显示模式、显示位置和显示内容。 总结起来,实现STM32模拟I2C驱动OLED需要配置GPIO引脚、实现软件I2C功能,编写发送和接收数据的函数,并控制OLED的显示内容。这样,我们就能够通过STM32来控制并显示内容在OLED上。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值