背景:拟采用bq33100超级电容管理芯片,实现自动的超级电容组的均压任务。需监控芯片的工作情况,以及电容组的均压情况。

平台:

硬件:STM32F103C8T6

通信:SMBus(低速IIC)

目标芯片:bq33100(TI)

SMbus简介:SMBus其实就是低速的IIC,和IIC协议基本相同,速度限制在10K~100KHz之间,一般应用于电源管理芯片等。

通信方式说明:这里考虑到SMBus需要限制IIC的速度,而通常使用的软件模拟IIC,我目前不清楚具体其通讯速率,而且我发现很多类似电源管理芯片的通信都是直接使用的硬件IIC,因此决定试一下硬件IIC的方式。由于STM32的硬件IIC普遍反映不是太稳定,可能会不定时卡死,这里参考野火的硬件IIC的历程中做了相应的超时处理(一旦卡死,就放弃本次的IIC通信),在实际的使用过程中,基本从来没有出现过触发这个卡死处理机制的情况(也就是说,其实可以尝试取消掉超时处理),但是这里以防万一还是选择保留。

配置代码:

void vIicConfig(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
I2C_InitTypeDef I2C_InitStructure; RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB, ENABLE );
RCC_APB1PeriphClockCmd( RCC_APB1Periph_I2C1, ENABLE ); /* GPIO.B6(IIC1_SCL) GPIO.B7(IIC1_SDA) */
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;
GPIO_Init( GPIOB, &GPIO_InitStructure ); /* can try to set to smbus mode */
I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;
I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;
I2C_InitStructure.I2C_OwnAddress1 = HUST_ADDRESS;
I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;
I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;
/* smbus require frequency between 100KHz and 10KHz -> 10K */
I2C_InitStructure.I2C_ClockSpeed = I2C_Speed; I2C_Cmd( I2C1, ENABLE );
I2C_Init( I2C1, &I2C_InitStructure ); /* enable iic acknowledge */
I2C_AcknowledgeConfig( I2C1, ENABLE );
}

IIC读数据函数封装:(含超时处理以及错误报告)

/* for bq33100 chip realtime monitoring (just read base data) */
uint8_t I2C_Hardware_BufferRead(u8* pBuffer, u8 ReadAddr, u16 NumByteToRead)
{
I2CTimeout = I2C_LONG_TIMEOUT; while(I2C_GetFlagStatus(IIC1, I2C_FLAG_BUSY))
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(9);
} /* Send START condition */
I2C_GenerateSTART(IIC1, ENABLE); I2CTimeout = I2C_SHORT_TIMEOUT;
/* Test on EV5 and clear it */
while(!I2C_CheckEvent(IIC1, I2C_EVENT_MASTER_MODE_SELECT))
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(10);
} /* Send EEPROM address for write */
I2C_Send7bitAddress(IIC1, BQ33100_DEVICE_ADDRESS, I2C_Direction_Transmitter); I2CTimeout = I2C_SHORT_TIMEOUT;
/* Test on EV6 and clear it */
while(!I2C_CheckEvent(IIC1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED))
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(11);
} /* Clear EV6 by setting again the PE bit */
I2C_Cmd(IIC1, ENABLE); /* Send the EEPROM's internal address to write to */
I2C_SendData(IIC1, ReadAddr); I2CTimeout = I2C_SHORT_TIMEOUT;
/* Test on EV8 and clear it */
while(!I2C_CheckEvent(IIC1, I2C_EVENT_MASTER_BYTE_TRANSMITTED))
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(12);
} /* Send STRAT condition a second time */
I2C_GenerateSTART(IIC1, ENABLE); I2CTimeout = I2C_SHORT_TIMEOUT;
/* Test on EV5 and clear it */
while(!I2C_CheckEvent(IIC1, I2C_EVENT_MASTER_MODE_SELECT))
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(13);
} /* Send EEPROM address for read */
I2C_Send7bitAddress(IIC1, BQ33100_DEVICE_ADDRESS, I2C_Direction_Receiver); I2CTimeout = I2C_SHORT_TIMEOUT;
/* Test on EV6 and clear it */
while(!I2C_CheckEvent(IIC1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED))
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(14);
} /* While there is data to be read */
while(NumByteToRead)
{
if(NumByteToRead == 1)
{
/* Disable Acknowledgement */
I2C_AcknowledgeConfig(IIC1, DISABLE); /* Send STOP Condition */
I2C_GenerateSTOP(IIC1, ENABLE);
} /* Test on EV7 and clear it */
I2CTimeout = I2C_LONG_TIMEOUT; while(I2C_CheckEvent(IIC1, I2C_EVENT_MASTER_BYTE_RECEIVED)==0)
{
if((I2CTimeout--) == 0) return I2C_TIMEOUT_UserCallback(3);
}
{
/* Read a byte from the EEPROM */
*pBuffer = I2C_ReceiveData(IIC1); /* Point to the next location where the byte read will be saved */
pBuffer++; /* Decrement the read bytes counter */
NumByteToRead--;
}
} /* Enable Acknowledgement to be ready for another reception */
I2C_AcknowledgeConfig(IIC1, ENABLE); return 0;
} /* hardware iic block processing */
static uint8_t I2C_TIMEOUT_UserCallback(uint8_t errorCode)
{
Error_Report(errorCode); return errorCode;
} /* report error information for debug */
static void Error_Report(uint8_t errorcode)
{
if(errorcode != 0)
{ }
}

bq33100通信函数示例:(只截取其中2个)

/* read health percent */
void bq33100_readhealthpercent(void)
{
uint8_t origin_healthpercent; bq33100.healthpercent_errorcode = I2C_Hardware_BufferRead(&origin_healthpercent,Health_ADD,1); bq33100.healthpercent = origin_healthpercent;
} /* read capacitance */
void bq33100_readcapacitance(void)
{
uint8_t temp[2];
uint16_t tmp = 0;
uint16_t origin_capacitance = 0; bq33100.capacitance_errorcode = I2C_Hardware_BufferRead(temp,Capacitance_ADD,2); tmp = (((uint16_t)temp[1] << 8) & 0xFF00); origin_capacitance = ((u16)(tmp + temp[0]) & 0xFFFF); bq33100.capacitance_mF = (float)origin_capacitance*1000;
}

——cloud over sky

——2020/3/1

STM32读取bq33100数据——硬件IIC的更多相关文章

  1. STM32读取Guidance数据——Guidance SDK

    更新记录:2019/11/14    更新STM32(F407VET6)读取Guidance数据 Github地址. 背景:想要将祖传的Guidance用于DJI A3/新固件的N3飞控.DJI已经停 ...

  2. STM32硬件IIC

    /** * @brief 写一个字节到I2C设备中 * @param * @arg pBuffer:缓冲区指针 * @arg WriteAddr:写地址 * @retval 正常返回1,异常返回0 * ...

  3. STM32 IIC双机通信—— HAL库硬件IIC版

    参考传送门 关于IIC的原理这里我就不多说了,网上有很多很好的解析,如果要看我个人对IIC的理解的话,可以点击查看,这里主要讲一下怎样利用STM32CubeMx实现IIC的通讯,经过个人实践,感觉HA ...

  4. STM32硬件IIC操作

    Stm32具有IIC接口,接口有以下主要特性 多主机功能:该模块既可做主设备也可做从设备 主设备功能 C地址检测 位/10位地址和广播呼叫 支持不同的通讯速度 状态标志: 发送器/接收器模式标志 字节 ...

  5. STM32硬件IIC (转)

    源: STM32硬件IIC

  6. STM32硬件IIC驱动设计(转)

    源: STM32硬件IIC驱动设计 参考: STM32—硬件IIC主机通信 STM32’s I2C 硬件BUG引发的血案(qzm) 解决STM32 I2C接口死锁在BUSY状态的方法讨论

  7. STM32读取HX711(AD)模块数据——压力传感器

    背景:在无人机动力系统的选型时,为了测试无人机的动力系统所能提供的最大拉力,使用压力传感装置测量拉力. 链接: 压力传感器tb链接: HX711模块是一个24位精度的AD模块. (1)https:// ...

  8. STM32读取匿名光流数据——与Guidance的光流和超声波做对比测试

    使用两个串口同时读取匿名光流和Guidance数据:用以比较两个光流的效果 Github链接:https://github.com/W-yt/YuTian_Pro/tree/master/Guidan ...

  9. STM32读取温湿度传感器DHT11和DHT21(AM2301)系列问题

    1.DHT11和DHT21传感器 这两种传感器都是奥松公司的产品,具体的传感器说明书在其官网上有(www.aosong.com). DHT11 数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合 ...

随机推荐

  1. Codeforces Round #577 (Div. 2) D. Treasure Hunting

    Codeforces Round #577 (Div. 2)  D. Treasure Hunting 这个一场div2 前面三题特别简单,这个D题的dp还是比较难的,不过题目告诉你了只能往上走,所以 ...

  2. Java λ表达式

    目录 Java Lambda表达式 1. 函数式编程思想 2. 体验Lambda的更优写法 3. Lambda表达式的标准格式 4. 练习:使用Lambda表达式(无参无返回) 5. 练习:使用Lam ...

  3. ssh框架整合时的延迟加载问题(no session问题)的分析以及解决方案

    当我们整合完三大框架,并采用hibernate的延迟加载方案时,会出现如下的异常: 现在对这个异常进行分析,如下图所示(模拟通过id查询用户信息的过程): 上图分析了为什么会出现no session的 ...

  4. 设计模式之GOF23迭代器模式

    迭代器模式Iterator /** * 自定义迭代器接口 * @author 小帆敲代码 * */public interface MyIterator {  void first();//游标置于第 ...

  5. [hdu4498]离散化,simpson求积分

    题意:,求这个函数在[0,100]上的图像的长度. 思路:采用离散化的思想,求出所有交点 ,把交点排序,把[0,100]分成若干个小区间,这样原函数在每个小区间上的图像属于某一个二次函数或者是一条直线 ...

  6. sql 取值时将字符串类型数字转化成整形

    select cast(a.Vchcode as int) as avchcode,a.ptypeid,a.assqty,unit,b.pfullname,b.standard,b.type from ...

  7. 容器技术之LXC WEB管理工具LXC WEB Panel

    前一篇博文中主要说了下,lxc容器在Linux上的简单管理,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/12901493.html:今天我们来介绍下lxc的图 ...

  8. AT命令集详解

    1.2 AT的优点. 命令简单易懂,并且采用标准串口来收发AT命令,这样对设备控制大大简化了,转换成简单串口编程了. AT命令提供了一组标准的硬件接口--串口.这个简化的硬件设计.较新的电信网络模块, ...

  9. JQuery踩过的坑,遇到就记下

    1 乱用选择器 坑人指数:200 JQuery选择器调用代价很大,反复调用效率更低.应采用缓存对象的方法或采用链式调用的方式. //错误的写法 $("#button").click ...

  10. mybatis随记

    JDBC问题:1.数据库配置信息硬编码 2.频繁创建,释放数据库连接 3.sql,设置参数,获取结果集硬编码,不通用   解决方案:1.配置文件 2.采用连接池 3.使用反射和内省   自定义持久层框 ...