STM32F407 模拟IIC AT24C16出坑记录

事情经过:有个项目一个模拟IIC上挂了3个外设,其中一个为AT24C02,模拟IIC用的是原子的。这种模式下使用完全没有问题,直到项目需要更换AT24C16。。。第一次找了两个供应商打样了一批AT24C16的IC卡,直接替换没有问题,认为方案替换没有问题。问题在于,当供应商批量供货的时候,懵了,全部不能读写,但是这些卡在另外的(程序一样就是主控芯片和电路不一样)一个读卡设备上却是可以的。

分析过程:

一、供应商确认前后两批AT24C16的卡都是同一个型号,就是批次不一样。???批次出问题了?但为什么同样程序的另外的卡设备可以读,是这个读卡设备的电路更稳定?那好,把不能读的设备电路改成可以读的设备一样,懵。。。 也是不行。这时怀疑的方向是这两个设备的主控芯片不一样,模拟IIC的IO初始化方式会不会不一样和是不是IIC挂的设备太多导致的。

二、基于上面一,脱离自己的产品,用原子的F407开发板试,把板载的AT24C02焊掉,把IIC飞行连接第一批AT24C16和批量的AT24C16,懵。。。 第一批的依旧可以,批量的依旧不行。。。。

陷入深深的迷惑中。。。,上逻辑分析仪、示波器、看芯片资料、问度娘。。。。反反复复的试,结果得出如下魔幻解决方案:

1、SDA SCL IO口要分开初始化 而且SDA要设置为开漏输出

按上面修改后在开发板上可以正常读写了,但是移植到产品上后

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
STM32F407模拟IIC代码可以使用软件模拟的方式实现。以下是一个简单的示例代码: ``` #include "stm32f4xx.h" #define IIC_SCL_PIN GPIO_Pin_6 #define IIC_SDA_PIN GPIO_Pin_7 #define IIC_SCL_PORT GPIOB #define IIC_SDA_PORT GPIOB // 初始化IIC总线 void IIC_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = IIC_SCL_PIN | IIC_SDA_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_OD; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(IIC_SCL_PORT, &GPIO_InitStructure); } // IIC起始信号 void IIC_Start(void) { GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); GPIO_SetBits(IIC_SDA_PORT, IIC_SDA_PIN); delay_us(4); GPIO_ResetBits(IIC_SDA_PORT, IIC_SDA_PIN); delay_us(4); GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(4); } // IIC停止信号 void IIC_Stop(void) { GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); GPIO_ResetBits(IIC_SDA_PORT, IIC_SDA_PIN); delay_us(4); GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); GPIO_SetBits(IIC_SDA_PORT, IIC_SDA_PIN); delay_us(4); } // IIC等待应答信号 // 返回: 1,接收应答失败 // 0,接收应答成功 uint8_t IIC_Wait_Ack(void) { uint8_t ucErrTime = 0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(IIC_SDA_PORT, &GPIO_InitStructure); GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(1); while (GPIO_ReadInputDataBit(IIC_SDA_PORT, IIC_SDA_PIN)) { ucErrTime++; if (ucErrTime > 250) { IIC_Stop(); return 1; } } GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_Init(IIC_SDA_PORT, &GPIO_InitStructure); return 0; } // IIC发送一个字节 void IIC_Send_Byte(uint8_t txd) { uint8_t t; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_Init(IIC_SDA_PORT, &GPIO_InitStructure); GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); for (t = 0; t < 8; t++) { GPIO_WriteBit(IIC_SDA_PORT, IIC_SDA_PIN, (txd & 0x80) >> 7); txd <<= 1; delay_us(2); GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(2); GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(2); } } // IIC读取一个字节 // ack=1,发送ACK // ack=0,发送NACK uint8_t IIC_Read_Byte(uint8_t ack) { uint8_t i, receive = 0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN; GPIO_Init(IIC_SDA_PORT, &GPIO_InitStructure); for (i = 0; i < 8; i++) { GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(2); receive <<= 1; receive |= GPIO_ReadInputDataBit(IIC_SDA_PORT, IIC_SDA_PIN); GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(1); } if (!ack) IIC_NAck(); else IIC_Ack(); return receive; } // IIC发送应答信号 void IIC_Ack(void) { GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_Init(IIC_SDA_PORT, &GPIO_InitStructure); GPIO_ResetBits(IIC_SDA_PORT, IIC_SDA_PIN); delay_us(2); GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(2); GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); } // IIC发送非应答信号 void IIC_NAck(void) { GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_Init(IIC_SDA_PORT, &GPIO_InitStructure); GPIO_SetBits(IIC_SDA_PORT, IIC_SDA_PIN); delay_us(2); GPIO_SetBits(IIC_SCL_PORT, IIC_SCL_PIN); delay_us(2); GPIO_ResetBits(IIC_SCL_PORT, IIC_SCL_PIN); } ``` 以上是一个简单的STM32F407模拟IIC代码,可以使用该代码来实现IIC通信。需要注意的是,由于是软件模拟,因此需要延时函数来模拟IIC的时序。这里的延时函数需要根据具体的系统时钟频率进行相应的调整。此外,还需根据实际的硬件连接,在代码中正确配置IIC的引脚和端口。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值