当前位置: 首页 > news >正文

# 智融PD芯片I2C通信实战:从零实现高效可靠的模拟I2C驱动

🔥 智融PD芯片I2C通信实战:从零实现高效可靠的模拟I2C驱动

📋 概述

本文详细介绍了一个用于与智融PD芯片进行通信的模拟I2C驱动程序实现。该驱动通过GPIO模拟I2C协议,实现了完整的I2C通信功能,包括起始信号、停止信号、数据传输和应答检测等核心功能。

🛠️ 驱动程序基础函数

1. GPIO引脚配置

/* SDA引脚输出模式设置 */staticinlinevoidpd_sda_set_out_mode(structpd_i2c_bus_typedef*bus){gpio_init(PD_SDA_GPIO_PORT,GPIO_MODE_OUT_OD,GPIO_OSPEED_50MHZ,PD_SDA_PIN);}/* SDA引脚输入模式设置 */staticinlinevoidpd_sda_set_in_mode(structpd_i2c_bus_typedef*bus){gpio_init(PD_SDA_GPIO_PORT,GPIO_MODE_IN_FLOATING,GPIO_OSPEED_50MHZ,PD_SDA_PIN);}/* 获取SDA引脚状态 */staticinlineuint8_tget_sda(structpd_i2c_bus_typedef*bus){returngpio_input_bit_get(bus->gpiox,bus->sda_gpio_pin);}/* SDA引脚电平控制 */staticinlinevoidPD_SDA_L(structpd_i2c_bus_typedef*bus){gpio_bit_reset(bus->gpiox,bus->sda_gpio_pin);}staticinlinevoidPD_SDA_H(structpd_i2c_bus_typedef*bus){gpio_bit_set(bus->gpiox,bus->sda_gpio_pin);}/* SCL引脚电平控制 */staticinlinevoidPD_SCL_L(structpd_i2c_bus_typedef*bus){gpio_bit_reset(bus->gpiox,bus->scl_gpio_pin);}staticinlinevoidPD_SCL_H(structpd_i2c_bus_typedef*bus){gpio_bit_set(bus->gpiox,bus->scl_gpio_pin);}

IIC 起始信号

staticinlinevoidpd_i2c_start(structpd_i2c_bus_typedef*bus){pd_sda_set_out_mode(bus);PD_SDA_H(bus);bus->udelay(10);PD_SCL_H(bus);bus->udelay(10);PD_SDA_L(bus);bus->udelay(10);PD_SCL_L(bus);bus->udelay(10);}

IIC结束信号

staticinlinevoidpd_i2c_stop(structpd_i2c_bus_typedef*bus){bus->udelay(6);PD_SDA_L(bus);bus->udelay(6);PD_SCL_H(bus);bus->udelay(6);PD_SDA_H(bus);bus->udelay(6);PD_SDA_H(bus);pd_sda_set_out_mode(bus);}

IIC 重启信号

staticinlinevoidpd_i2c_restart(structpd_i2c_bus_typedef*bus){PD_SDA_H(bus);bus->udelay(6);PD_SCL_H(bus);bus->udelay(6);PD_SDA_L(bus);bus->udelay(6);PD_SCL_L(bus);bus->udelay(6);}

判断从机回复

/** * @brief pd_iic 检测响应,pd_iic * @param struct pd_i2c_bus_typedef *bus * @param NONE * @retval 返回 0 从机接收成功; 返回1,应答失败 */staticinlineuint8_tpd_i2c_check_ack(structpd_i2c_bus_typedef*bus){uint8_tack=0;uint16_tpd_chip_timeout=0;PD_SDA_H(bus);bus->udelay(5);PD_SCL_H(bus);bus->udelay(5);pd_sda_set_in_mode(bus);while(1==get_sda(bus)){pd_chip_timeout++;if(pd_chip_timeout>250){pd_i2c_stop(&pd_i2c_bus);ack=1;break;}}pd_sda_set_out_mode(bus);PD_SCL_L(bus);bus->udelay(5);returnack;}

发送检测响应

/** * @brief pd_iic 发送检测响应,iic 协议第9位为应答 * @param struct pd_i2c_bus_typedef *bus * @param ack 需要继续通信拉低0, 不需要继续通信拉高1 * @retval */staticinlinevoidpd_i2c_send_ack_Or_nack(structpd_i2c_bus_typedef*bus,intack){//int re_ack = 0;PD_SCL_L(bus);bus->udelay(10);if(ack){PD_SDA_L(bus);//继续通信}else{PD_SDA_H(bus);}bus->udelay(10);PD_SCL_H(bus);bus->udelay(10);PD_SCL_L(bus);bus->udelay(10);}

写单字节

//写单字节staticvoidpd_i2c_write_byte(structpd_i2c_bus_typedef*bus,uint8_tdata){uint8_ti=0;uint8_tdat=0;uint8_ttemps=0;dat=0x80;temps=data;for(i=0;i<8;i++){PD_SCL_L(bus);if(dat&temps)//对应位为一就发一{PD_SDA_H(bus);bus->udelay(WR_RD_DELAY_TIME);//延时一下PD_SCL_H(bus);bus->udelay(WR_RD_DELAY_TIME);//t(HIGH) Clock high period See Note 2 4.0~50 μs}else{PD_SDA_L(bus);bus->udelay(WR_RD_DELAY_TIME);//延时一下PD_SCL_H(bus);bus->udelay(WR_RD_DELAY_TIME);}dat=dat>>1;}PD_SCL_L(bus);bus->udelay(WR_RD_DELAY_TIME);PD_SDA_H(bus);//数据线拉高,为等待响应做准备bus->udelay(WR_RD_DELAY_TIME);}

读单字节

//读单字节staticuint8_tpd_i2c_read_byte(structpd_i2c_bus_typedef*bus){uint8_tmask;uint8_tdata=0;PD_SDA_H(bus);bus->udelay(10);pd_sda_set_in_mode(bus);for(mask=0x80;mask!=0;mask>>=1){PD_SCL_H(bus);bus->udelay(10);if(get_sda(bus)){data|=mask;}PD_SCL_L(bus);bus->udelay(10);}pd_sda_set_out_mode(bus);returndata;}

读寄存器

intpd_i2c_read_register(uint8_t*pd_data_buff,uint16_taddr,uint16_tlen){uint16_ti=0;// uint16_t data = 0;uint8_tsda_status=0;uint16_tstart_addr=0;// uint8_t start_temp = 0;start_addr=addr;pd_lock();//PD 读pd_i2c_start(&pd_i2c_bus);//开始pd_i2c_write_byte(&pd_i2c_bus,PD_I2C_WRITE_SLAVE_ADDR);//设备地址 0x3Cpd_i2c_bus.udelay(DLY_TIMER_1);sda_status=pd_i2c_check_ack(&pd_i2c_bus);//等待响应if(1==sda_status){pd_i2c_stop(&pd_i2c_bus);pd_i2c_stop(&pd_i2c_bus);pd_unlock();return(-1);}pd_i2c_bus.udelay(DLY_TIMER_1);pd_i2c_write_byte(&pd_i2c_bus,(uint8_t)start_addr);//数据地址pd_i2c_bus.udelay(DLY_TIMER_1);sda_status=pd_i2c_check_ack(&pd_i2c_bus);//等待响应if(1==sda_status){pd_i2c_stop(&pd_i2c_bus);pd_i2c_stop(&pd_i2c_bus);pd_unlock();return(-1);}pd_i2c_bus.udelay(DLY_TIMER_1);pd_i2c_restart(&pd_i2c_bus);//再次启动pd_i2c_write_byte(&pd_i2c_bus,PD_I2C_READ_SLAVE_ADDR);//设备地址 0x3Dsda_status=pd_i2c_check_ack(&pd_i2c_bus);//等待响应if(1==sda_status){pd_i2c_stop(&pd_i2c_bus);pd_i2c_stop(&pd_i2c_bus);pd_unlock();return(-1);}for(i=0;i<len;i++){pd_data_buff[i]=pd_i2c_read_byte(&pd_i2c_bus);//pd_i2c_bus.udelay(DLY_TIMER_2);pd_i2c_send_ack_Or_nack(&pd_i2c_bus,ACK_FLAG);}pd_i2c_bus.udelay(DLY_TIMER_1);pd_i2c_stop(&pd_i2c_bus);//停止pd_unlock();return0;}

写寄存器

/** * @brief PD 芯片写单字节 * @param register PD 芯片写寄存器 * @param p_data 数据指针 * @retval 异常返回 1,正常返回 0 * @note 软件模拟SMBUS协议 */uint8_tpd_i2c_write_register(uint8_t*pd_data_buff,uint16_taddr,uint16_tlen){uint16_ti=0;//uint16_t data = 0;uint8_tsda_status=0;uint16_tstart_addr=0;//uint8_t start_temp = 0;pd_lock();start_addr=addr;//PD 读pd_i2c_start(&pd_i2c_bus);//开始pd_i2c_write_byte(&pd_i2c_bus,PD_I2C_WRITE_SLAVE_ADDR);//设备地址 0x3Cpd_i2c_bus.udelay(DLY_TIMER_2);sda_status=pd_i2c_check_ack(&pd_i2c_bus);//等待响应if(1==sda_status){pd_unlock();returnsda_status;}pd_i2c_write_byte(&pd_i2c_bus,(uint8_t)start_addr);//数据地址pd_i2c_bus.udelay(DLY_TIMER_2);sda_status=pd_i2c_check_ack(&pd_i2c_bus);//等待响应if(1==sda_status){pd_unlock();returnsda_status;}pd_i2c_bus.udelay(DLY_TIMER_2);for(i=0;i<len;i++){pd_i2c_write_byte(&pd_i2c_bus,pd_data_buff[i]);pd_i2c_bus.udelay(DLY_TIMER_2);sda_status=pd_i2c_check_ack(&pd_i2c_bus);}pd_i2c_stop(&pd_i2c_bus);//停止pd_unlock();returnsda_status;}

🎯 结束语

通过本文的详细讲解,我们完成了智融PD芯片I2C通信驱动的完整实现。从GPIO引脚配置到I2C协议的核心时序控制,再到读写寄存器的完整流程,我们一步步构建了一个高效可靠的模拟I2C驱动。

📚 核心要点回顾

  1. GPIO配置:正确配置SDA和SCL引脚的输入输出模式是I2C通信的基础
  2. 时序控制:严格按照I2C协议的时序要求实现起始、停止、重启信号
  3. 应答机制:完善的应答检测和发送机制确保通信可靠性
  4. 读写操作:完整的读写寄存器函数封装了底层通信细节

💡 实际应用建议

  • 时序调整:根据实际硬件特性微调延时参数,优化通信速度
  • 错误处理:在生产环境中增加更完善的错误日志和重试机制
  • 性能优化:对于频繁的I2C操作,可以考虑使用DMA或硬件I2C外设

希望本文能为您的嵌入式开发工作提供实用的参考。如果在实际应用中遇到问题,欢迎在评论区交流讨论。祝您开发顺利!


本文代码已在多个实际项目中验证,具有良好的稳定性和可靠性。建议在实际使用时根据具体硬件平台进行适当调整。
`

http://www.zskr.cn/news/1535212.html

相关文章:

  • 终极指南:5分钟用Qt Material打造现代化桌面应用界面
  • 2026年6月昭通黄金回收靠谱商家筛选与变现避坑干货 - 余生黄金回收
  • 网盘直链下载助手终极指南:一键获取九大网盘真实下载地址的技术解决方案
  • AI Agent如何处理无预设流程业务?深度解析大模型自主规划的底层推理能力与架构落地实践
  • [特殊字符]️【万字深度干货】数字政府数字化转型规划信息化解决方案全解析——从国土空间规划重构到智能审批的完整技术路线(PPT)
  • VLM (4):connector
  • 伊犁2026年6月黄金回收行情解读 正规商家辨别方法 - 余生黄金回收
  • 硬盘低级格式化工具深度解析:原理、风险与实战指南
  • 2026年6月昭通卖黄金不踩坑 正规回收行情与门店实测指南 - 余生黄金回收
  • 2026武汉市家用空调-中央空调等维修安装移机加氟-本地精选指南 -欧米到家 - 欧米到家
  • 终极XXMI启动器完整指南:一站式管理6大热门游戏模组
  • Python中%运算符的真相:模运算不是取余
  • 2026年6月邢台卖黄金怎么选正规回收店实操指南 - 余生黄金回收
  • 在 Claude Code 中接入讯飞 MaaS Qwen3.6 模型
  • 360Controller背后的魔法:如何让Xbox控制器在macOS上获得新生
  • 2026年6月廊坊卖黄金防坑攻略正规回收价格明细 - 余生黄金回收
  • 合肥黄金回收:避开陷阱选对门,安心变现不踩坑 - 余生黄金回收
  • 如何在3分钟内解决iPhone USB网络共享驱动问题:终极一键安装指南
  • AI低成本服务落地:数据蒸馏、MoE路由与电价感知部署
  • Hermes Agent 国内实战生存手册:网络兼容、安装排障与Kimi深度优化
  • Codex++解锁APIKey全功能
  • Hermes Agent国内实战指南:30分钟跑通Kimi集成
  • GTA5线上小助手:一站式游戏增强平台完全指南
  • 【2026年6月】电动推拉雨棚优质企业推荐指南 - 多才菠萝
  • VBA数据结构之争:3倍效率差,90%开发者选错了
  • 2026年成都园林绿化服务公司优选榜:绿植租摆/庭院景观/绿化工程/绿植养护全覆盖 - 海棠依旧大
  • 编程思维训练:循环控制与格式化输出实现数字三角形
  • 2026天津黄金回收全攻略:多家实体门店横向评测,附详细地址与避坑指南 - 润富黄金回收
  • 终极TCP路由追踪指南:5分钟掌握tracetcp的完整使用方法
  • 如何快速上手Kimi Free API:面向开发者的完整指南