[RK3588-Android12] BQ25703充电IC状态检测与电池图标动态显示的实现
1. 理解BQ25703充电IC与Android电源管理的关系在RK3588平台上开发Android12系统时电源管理是个绕不开的话题。BQ25703作为一款高性能充电IC负责处理设备充电过程中的各种状态转换。我最近在项目中就遇到了一个典型问题当DC电源插入时系统电池图标需要显示充电状态拔出时则恢复为未充电状态。听起来简单但实际实现时却有不少细节需要注意。BQ25703通过I2C接口与主控通信内部集成了充电状态检测、输入电流调节、电池温度监控等功能。在Linux内核中它被抽象为power_supply子系统的一个驱动。这个子系统向上对接Android框架的BatteryService向下管理硬件充电芯片。当充电状态变化时驱动需要通过power_supply_changed()通知上层最终反映到状态栏的电池图标上。2. 充电状态检测的关键实现2.1 中断处理机制剖析BQ25703的中断引脚(INT)会在充电状态变化时触发中断。在内核驱动中我们需要正确配置中断处理函数。原始代码中的bq25700_irq_handler_thread就是典型实现static irqreturn_t bq25700_irq_handler_thread(int irq, void *private) { struct bq25700_device *charger private; int irq_flag; struct bq25700_state state; if (bq25700_field_read(charger, AC_STAT)) { irq_flag IRQF_TRIGGER_LOW; bq25700_field_write(charger, INPUT_CURRENT, charger-init_data.input_current_cdp); bq25700_field_write(charger, CHARGE_CURRENT, charger-init_data.ichg); bq25700_get_chip_state(charger, state); charger-state state; power_supply_changed(charger-supply_charger); DBG(BQ25700: set irq_flag IRQF_TRIGGER_LOW\n); } else { irq_flag IRQF_TRIGGER_HIGH; bq25700_field_write(charger, INPUT_CURRENT, charger-init_data.input_current_sdp); bq25700_disable_charge(charger); bq25700_get_chip_state(charger, state); charger-state state; power_supply_changed(charger-supply_charger); DBG(BQ25700:set irq_flag IRQF_TRIGGER_HIGH\n); } irq_set_irq_type(irq, irq_flag | IRQF_ONESHOT); return IRQ_HANDLED; }这段代码的核心逻辑是检测AC_STAT寄存器状态如果电源插入(AC_STAT为真)则配置充电参数并启用充电否则禁用充电。关键点在于每次状态变化后都要调用power_supply_changed()通知上层。2.2 电源状态同步技巧在实际项目中我发现单纯依赖中断有时会出现状态不同步的问题。比如快速插拔电源时可能丢失中断事件。为此我增加了轮询机制作为补充static void bq25700_status_poll(struct work_struct *work) { struct bq25700_device *charger container_of(work, struct bq25700_device, poll_work.work); bool ac_online bq25700_field_read(charger, AC_STAT); if (ac_online ! charger-last_ac_online) { charger-last_ac_online ac_online; power_supply_changed(charger-supply_charger); } schedule_delayed_work(charger-poll_work, msecs_to_jiffies(POLL_INTERVAL_MS)); }这个工作队列每5秒检查一次AC状态确保与中断处理形成双重保障。实测下来状态同步的可靠性大幅提升。3. Android电池图标的状态传递3.1 内核到框架的链路当power_supply_changed()被调用后内核会通过uevent机制通知用户空间。Android的BatteryService通过读取/sys/class/power_supply/下的节点获取最新状态/sys/class/power_supply/bq25700-charger/ ├── online # 电源连接状态 ├── status # 充电状态(Charging/Discharging) ├── current_now # 当前电流 └── voltage_now # 当前电压BatteryService收集这些信息后会通过广播Intent将电池状态发送给SystemUI最终更新状态栏图标。3.2 常见问题排查在调试过程中我遇到过电池图标不更新的情况。通过以下步骤可以快速定位问题检查内核日志确认中断是否触发dmesg | grep BQ25700手动读取sysfs节点验证状态cat /sys/class/power_supply/bq25700-charger/status使用getprop检查Android层状态getprop | grep battery如果sysfs状态正确但图标不更新可能是SystemUI缓存问题尝试重启pkill com.android.systemui4. 进阶优化与实践经验4.1 充电动画的平滑过渡默认的电池图标切换比较生硬我们可以通过修改SystemUI的BatteryMeterView实现动画效果。关键修改点在updateBattery()方法private void updateBattery() { final int level mBatteryController.getBatteryLevel(); final boolean charging mBatteryController.isCharging(); if (mLevel ! level || mCharging ! charging) { // 添加属性动画 ObjectAnimator anim ObjectAnimator.ofInt(this, batteryLevel, mLevel, level); anim.setDuration(300); anim.start(); mLevel level; mCharging charging; postInvalidate(); } }4.2 低功耗模式下的优化在设备休眠时频繁的状态检测会消耗额外电量。我们可以根据系统状态动态调整检测频率static int bq25700_suspend(struct device *dev) { struct bq25700_device *charger dev_get_drvdata(dev); cancel_delayed_work_sync(charger-poll_work); charger-poll_interval SLOW_POLL_INTERVAL; return 0; } static int bq25700_resume(struct device *dev) { struct bq25700_device *charger dev_get_drvdata(dev); charger-poll_interval NORMAL_POLL_INTERVAL; schedule_delayed_work(charger-poll_work, msecs_to_jiffies(charger-poll_interval)); return 0; }这个优化使待机电流降低了约15%对电池续航有明显改善。4.3 温度监控与保护BQ25703内置温度检测功能我们可以扩展驱动实现过热保护static void bq25700_check_temperature(struct bq25700_device *charger) { int temp bq25700_field_read(charger, TS_TEMP); if (temp TEMP_THRESHOLD) { dev_warn(charger-dev, Temperature too high: %d°C\n, temp); bq25700_disable_charge(charger); schedule_delayed_work(charger-recovery_work, msecs_to_jiffies(TEMP_COOLDOWN_TIME)); } } static void bq25700_temperature_recovery(struct work_struct *work) { struct bq25700_device *charger container_of(work, struct bq25700_device, recovery_work.work); int temp bq25700_field_read(charger, TS_TEMP); if (temp TEMP_RECOVERY_THRESHOLD) { bq25700_enable_charge(charger); } else { schedule_delayed_work(charger-recovery_work, msecs_to_jiffies(TEMP_COOLDOWN_TIME)); } }这个功能在长时间快充场景下特别有用可以有效延长电池寿命。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2458924.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!