原文:http://www.2cto.com/kf/201408/326462.html

1、概述

随着移动智能设备的快速发屏,电池的续航能力在很大情况下诱导了大众消费者的购买选择,android系统对电源管理的合理与否直接影响到电池的续航能力,而电池系统作为其中的一部分,主要用于对电池状态的监控(电池电量、电池状态及电池温度等)。下面将详细分析android的电池系统架构。

2、Android电池系统架构

Android系统中对电池的管理驱动层继承了linux下的power supply class,而在用户层则是在BatteryService.java中通过广播的方式将如下一些电池相关的属性上报给上层app使用。这些属性都是在java中声明,在jni中调用更新的。

而这些属性都是在com_android_server_BatteryService.cpp这个本地代码-jni中通过调用sys文件系统访问驱动层中电池相应的状态进行更新的。

3、BatteryService

代码路径:

frameworks/base/services/java/com/android/server/BatteryService.java

BatteryService 作为电池及充电相关的服务,主要作了如下几件事情: 监听 UEvent、读取sysfs 中的状态 、广播Intent.ACTION_BATTERY_CHANGED。

3.1、监听UEvent:

BatteryService实现了两个UEvenObserver mUEventObserver(如上代码所示)。uevent是Linux 内核用来向用户空间主动上报事件的机制,对于JAVA程序来说,只实现 UEventObserver的虚函数 onUEvent,然后注册即可。

3.2、读取sysfs电池状态

update读取sysfs文件做到同步取得电池信息,然后根据读到的状态更新BatteryService 的成员变量,并广播一个Intent来通知其它关注电源状态的组件。当kernel有power_supply事件上报时,mUEventObserver调用update()函数,然后update 调用native_update从sysfs中读取相关状态(com_android_server_BatteryService.cpp)。

3.3、广播Intent.ACTION_BATTERY_CHANGED

通过广播Intent.ACTION_BATTERY_CHANGED,将电池状态、电池电量、电池工艺等属性打包,发送给其它的使用者,也就是说,只要在app程序里监听了Intent.ACTION_BATTERY_CHANGED这个广播,就能获取到电池的各种状态属性!

4、本地代码-jni

代码路径:

frameworks/base/services/jni/com_android_server_BatteryService.cpp

这是battery用户空间的本地代码,调用sys文件系统访问驱动程序,并向上层BatteryService封装本地方法以隔离平台化的差异。

在这个文件的头部有如下定义:

这个就是底层battery驱动所生成的sys文件系统路径。

在这个文件里封装了一个本地方法:

供上层BatteryService使用。另外,在这个文件中通过GetFieldID,得到BatteryService.java类中声明的电池属性ID,如下:

关于java与c/c++互调的问题,这里就不再赘诉了,网上有很多的资料。

在这个文件里通过操作相应的sys文件系统来获取电池的各种属性值,看看如下这段代码:

POWER_SUPPLY_PATH已经在文件头部定义过了,对应路径:/sys/class/power_supply,然后遍历整个文件夹,查找各个能源供应设备的各种属性,如上选中部分是用来查找交流设备的属性的。

各能源设备属性概况如下:

/sys/class/power_supply/ac/online AC 电源连接状态

/sys/class/power_supply/usb/online USB电源连接状态

/sys/class/power_supply/battery/status 充电状态

/sys/class/power_supply/battery/health 电池状态

/sys/class/power_supply/battery/present 使用状态

/sys/class/power_supply/battery/capacity 电池 level

/sys/class/power_supply/battery/batt_vol 电池电压

/sys/class/power_supply/battery/batt_temp 电池温度

/sys/class/power_supply/battery/technology 电池技术

当供电设备的状态发生变化时,driver会更新这些文件,然后通过jni中的本地方法android_server_BatteryService_update向java层发送信息。

5、驱动

5.1、驱动概述

android系统电池部分的驱动程序,继承了传统linux系统下的Power Supply驱动程序架构,Battery驱动程序通过Power Supply驱动程序生成相应的sys文件系统,从而向用户空间提供电池各种属性的接口。Linux标准的 Power Supply驱动程序所使用的文件系统路径为:/sys/class/power_supply ,其中的每个子目录表示一种能源供应设备。

5.2、驱动头文件

Power Supply驱动程序头文件kernel/include/linux/power_supply.h,注册和注销驱动程序的函数如下:

intpower_supply_register(struct device *parent,struct power_supply *psy);

voidpower_supply_unregister(struct power_supply *psy);

structpower_supply {

constchar *name; /*设备名称*/

enumpower_supply_type type; /* 类型 */

enumpower_supply_property *properties; /* 属性指针 */

size_tnum_properties; /*属性的数目*/

char**supplied_to;

size_tnum_supplicants;

int(*get_property)(struct power_supply *psy, /*获得属性*/

enumpower_supply_property psp,

unionpower_supply_propval *val);

void(*external_power_changed)(struct power_supply *psy);

/* ...... 省略部分内容 */

};

5.3、power supply core

对应的驱动程序:power_supply

来看看power_supply_sysfs.c这个文件。这里主要是对诸如如下这些电源设备属性创建uevent!

这些uevent节点不一定都会创建,节点创建与否还和具体的电源设备驱动传进来的num_properties和properties有关,在创建uevent函数power_supply_uevent中可以很容易的看出这一点:

5.4、battery driver

目前项目(T808、T828)中所使用的电池检测与管理方式是POC ADC方式,对应的驱动文件是:

mediatek/kernel/drivers/power/battery_common.c

在此文件的probe函数里有如下内容:

可以看出,在这里将ac、usb及battery三种电源供应设备注册到了power supply core中去了,而相应的全局结构体变量ac_main、usb_main及battery_main作了如下定义:

各个电源设备所需要创建的uevent节点由这里传入的xx_props决定,相应的定义如下:

可发现:ac和usb只创建了一个online属性,上层app通过判断ac和usb的online状态便可知道当前系统是由什么设备在充电了;而battery则创建了如:status、health、present、capacity、batt_vol等等和电池相关的诸多属性,上层app通过这些电池属性uevent便可监控电池的当前工作状态了。

举例说明一下这些属性的状态改变后是如何向系统发送更新消息的,来看看ac online的状态更新。

该函数在power supply sysfs中show property的时候得到调用,而AC_ONLINE作为ac电源唯一的属性会在ac_update中得到更新:

ac_update则最终会在bat_thread_kthread中进行轮循,在这里有一个全局的BMT_status,作为整个电源供应设备的各种属性传达!另外再来看看CHARGING_CONTROL battery_charging_control这个全局的函数指针,原型定义如下:


chr_control_interface函数原型如下:

对应文件路径:

mediatek/platform/mt6572/kernel/drivers/power/charging_hw_pmic.c

charging_func函数指针数组定义如下:

可以看出通过如下的调用关系:

最终会调用到由:


这些枚举类型所一一对应的函数中去,如上面调用关系CHARGING_CMD_GET_CHARGER_TYPE,则是获取charger type,函数原型如下:

在这里通过pmic的硬件状态来获取相应的信息。

5.5、充电误差纠正

理想中的电池是没有内阻的,电池电压的消耗都在外部的负载上。

但实际情况却不是这样的,正由于电池内阻的存在,通过直接测量电池电压(ADC)的方式获得的电池电量都会存在一定的误差,不管电池是处于供电还是放电的状态,这种误差都会存在,特别是在电池电量满、电池电量空及关机充电的情况下这种误差很容易被用户察觉,从而带来不好的用户体验。

那么针对电池内阻导致的这个误差,完全可以通过数学方式进行纠正。

通过上图可知,只要知道了电池内阻,就可以很容易地纠正这种误差。但电池的内阻不是不变的,而是随着电池电量的变化而变化的,不同型号的电池,这种特性还不一样,那么要想得到比较准确的电池电量,就很有必要让电池厂提供一组完整的电池电压与电池内阻的关系表了!

充放电过程中误差的纠正代码:

mediatek/kernel/drivers/power/battery_meter.c

在oam_run中有如下代码:

函数mtk_imp_tracking就是针对充放电过程中电池内阻所产生的压降所作的一个△V的修正。

另外一个是,由于电池特性,在开关机的时候会出现电量跳变的问题,在系统中采用了将关机时的UI电量保存到RTC中,在下次开机的过程中用实际检测到的电量值与保存到RTC中的UI电量值进行比较判断,由于用户可能会更换电池,这两者之间的差值控制在了20%的范围内,也就是说:实际检测到的电量值在RTC中保存的UI电量值的20%范围内则使用保存在RTC中的UI电量值作为当前电池的电量值;如果实际检测到的电量值超过了RTC中UI电量值的20%,则认为用户更换了电池,用实际检测到的电量值作为当前电池的电量值

相应的判断代码如下:

在头文件:

mediate/custiom/mt6572/kernel/battery/battery/cust_battery_meter.h

中有如下定义

6、写在最后

关于电池电量检测的算法过程,这里不再赘述,在MTK发布的Customer_Training_Battery_Charging.pdf这个文件中有比较详细的讲述。

android电池管理系统的更多相关文章

  1. Android 电池管理系统架构总结 Android power and battery management architecture summaries

    文章目录 1 整体架构 2 设计构架 2.1 driver 2.1.1 Charger.ko 2.1.2 Battery.ko 2.2 power supply 2.2.1 基础架构 2.2.2 代码 ...

  2. android电池管理系统从上层的java到底层驱动的调用(转载)

    1.概述 随着移动智能设备的快速发屏,电池的续航能力在很大情况下诱导了大众消费者的购买选择,android系统对电源管理的合理与否直接影响到电池的续航能力,而电池系统作为其中的一部分,主要用于对电池状 ...

  3. 【转】android电池(五):电池 充电IC(PM2301)驱动分析篇

    关键词:android 电池  电量计  PL2301任务初始化宏 power_supply 中断线程化 平台信息:内核:linux2.6/linux3.0系统:android/android4.0  ...

  4. 【转】android电池(四):电池 电量计(MAX17040)驱动分析篇

    关键词:android 电池  电量计  MAX17040 任务初始化宏 power_supply 平台信息:内核:linux2.6/linux3.0系统:android/android4.0 平台: ...

  5. 【转】android 电池(三):android电池系统

    关键词:android电池系统电池系统架构 uevent power_supply驱动 平台信息: 内核:linux2.6/linux3.0系统:android/android4.0 平台:S5PV3 ...

  6. 【转】android 电池(二):android关机充电流程、充电画面显示

    关键词:android 电池关机充电 androidboot.mode charger关机充电 充电画面显示 平台信息:内核:linux2.6/linux3.0系统:android/android4. ...

  7. 【转】android 电池(一):锂电池基本原理篇

    关键词:android  电池关机充电 androidboot.mode charger 平台信息:内核:linux2.6/linux3.0系统:android/android4.0 平台:S5PV3 ...

  8. Android 电池系列

    android 电池(一):锂电池基本原理篇 android 电池(二):android关机充电流程.充电画面显示 android 电池(三):android电池系统 android电池(四):电池 ...

  9. GBT27930-2015电动汽车非车载传导式充电机与电池管理系统之间的通信协议

    本标准规定了电动汽车非车载传导式充电机(简称充电机)与电池管理系统(Battery Management System,简称BMS)之间基于控制器局域网(Control Area NetWork,简称 ...

随机推荐

  1. 自定义 Java Annotation ,读取注解值

    1. 首先是自定义注解: package cn.veji.hibernate.po; import java.lang.annotation.ElementType; import java.lang ...

  2. UI布局术语

    horizontal, vertical top, left, bottom, right UIEdgeInsets margin与padding Interpreting Values Values ...

  3. JavaScipt30(第六个案例)(主要知识点:给数字加千分号的正则)

    承接上文,这是第6个案例: 附上项目链接: https://github.com/wesbos/JavaScript30 这个主要是要实现在给定的json里匹配出搜索框里的city or state, ...

  4. 怎么让Eclipse对html和js代码自动提示

    使用eclipse自带的插件,无需另外安装插件,具体步骤如下1.打开eclipse→Windows→Preferences→Java→Editor→Content Assist修改Auto Activ ...

  5. 【webpack插件使用】在开发中快速掌握并使用Webpack构建web应用程序

    1.webpack-dev-server插件的基本使用 入门程序 const path = require('path'); // 导出一个Webpack的配置对象(通过node中的模块操作,向外暴露 ...

  6. Linux:DNS主、从、缓存服务器配置、DNS同步加密TSIG配置、DNS分离解析配置

    DNS主服务器配置(正向解析.反向解析) 正向解析:根据主机名查找对应的IP地址.当用户访问一个域名时(不考虑hosts文件等因素),正常情况会向指定的DNS主机发送递归查询请求反向解析:根据IP地址 ...

  7. CentOS7 安装、配置 Memcached

    点击访问原文地址 介绍 Memcached 是一个分布式.高性能的内存缓存系统,通过缓存内存中的数据和对象,提高和加速动态 web 应用程序的性能.它主要用于加速对数据库重度使用的站点. Memcac ...

  8. Flask蓝图基本使用

    Flask蓝图基本使用 Flask通过使用蓝图将视图函数模块化,使应用显得更加规整 比如我们的应用的视图函数包括用户相关和文章相关,那么我们可以通过建立两个py文件分别存储两类视图函数 user.py ...

  9. Batchelor Prize

    awards in fluid mechanics The Prize of $25,000 is awarded every four years to a single scientist for ...

  10. UVA 10692 Huge Mod

    Problem X Huge Mod Input: standard input Output: standard output Time Limit: 1 second The operator f ...