本文转载自:http://blog.csdn.net/xubin341719/article/details/8709838

一、电池系统结构

Android中的电池使用方式主要有三种:AC、USB、Battery 等不同的模式。在应用程序层次,通常包括了电池状态显示的功能。因此从 Android 系统的软件方面(包括驱动程序和用户空间内容)需要在一定程度上获得电池的状态,电池系统主要负责电池信息统计、显示。电池系统的架构如下所示:

自下而上, Android 的电池系统分成以下几个部分:

1、驱动程序

特定硬件平台电池的驱动程序,用 Linux的Power Supply 驱动程序,实现向用户空间提供信息。Battery 驱动程序需要通过sys文件系 统向用户空间提供接口, sys文件系统的路径是由上层的程序指定的。Linux标准的 Power Supply驱动程序 所使用的文件系统路径为:/sys/class/power_supply ,其中的每个子目录表示一种能源供应设备的名称。


Power Supply 驱动程序的头文件在 include/linux/power_supply.h中定义,注册和注销驱动程序的函数如下所示:

  1. int power_supply_register(struct device *parent,struct power_supply *psy);
  2. void power_supply_unregister(struct power_supply *psy);
  3. struct power_supply {
  4. const char *name;
  5. /* 设备名称 */
  6. enum power_supply_type type;
  7. /* 类型 */
  8. enum power_supply_property *properties;
  9. /* 属性指针 */
  10. size_t num_properties;
  11. /* 属性的数目 */
  12. char **supplied_to;
  13. size_t num_supplicants;
  14. int (*get_property)(struct power_supply *psy, /* 获得属性 */
  15. enum power_supply_property psp,
  16. union power_supply_propval *val);
  17. void (*external_power_changed)(struct power_supply *psy);
  18. /* ...... 省略部分内容 */
  19. };

Linux中驱动程序:power_supply



2、本地代码 - JNI

代码路径: frameworks/base/services/jni/com_android_server_BatteryService.cpp 这个类调用sys文件系统访问驱动程序,也同时提供了JNI的接口。

这个文件提供的方法列表如下所示:

  1. static JNINativeMethod sMethods[] = {
  2. {"native_update", "()V", (void*)android_server_BatteryService_update},
  3. };

处理的流程为根据设备类型判定设备后, 得到各个设备的相关属性,则需要得到更多得 信息。例如:果是交流或者 USB 设备,只需 要得到它们是否在线( onLine );如果是电 池设备,则需要得到更多的信息,例如状态 ( status ),健康程度( health ),容 量( capacity ),电压 ( voltage_now )等。

Linux 驱动 driver 维护着保存电池信息的一组文件 sysfs,供应用程序获取电源相关状态:

  1. #define AC_ONLINE_PATH "/sys/class/power_supply/ac/online" AC 电源连接状态
  2. #define USB_ONLINE_PATH "/sys/class/power_supply/usb/online" USB电源连接状态
  3. #define BATTERY_STATUS_PATH "/sys/class/power_supply/battery/status"充电状态
  4. #define BATTERY_HEALTH_PATH "/sys/class/power_supply/battery/health"电池状态
  5. #define BATTERY_PRESENT_PATH "/sys/class/power_supply/battery/present"使用状态
  6. #define BATTERY_CAPACITY_PATH "/sys/class/power_supply/battery/capacity"电池 level
  7. #define BATTERY_VOLTAGE_PATH "/sys/class/power_supply/battery/batt_vol"电池电压
  8. #define BATTERY_TEMPERATURE_PATH "/sys/class/power_supply/battery/batt_temp"电池温度
  9. #define BATTERY_TECHNOLOGY_PATH "/sys/class/power_supply/battery/technology"电池技术 当电池状态发生变化时,driver 会更新这些文件。传送信息到java

3 、Java 代码

代码路径:

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

frameworks/base/core/java/android/os/ : android.os :包中和Battery 相关的部分


frameworks/base/core/java/com/android/internal/os/:和Battery 相关的内部部分 BatteryService.java 通过调用, BatteryService JNI来实现com.android.server包中的 BatteryService类。BatteryManager.java中定义了一些 JAVA 应用程序层可以使用的常量。

电池系统在驱动程序层以上的部分都是Android 系统中默认的内容。在移植的过程中基本不需要改动。电池系统需要移植的部分仅有Battery驱动程序。Battery 驱动程序用Linux 标准的Power Supply驱动程序与上层的接口是sys文件系统,主要用于读取sys文件系统中的文件来获取电池相关的信息。整个系统中各部件的联系:

BatteryService 作为电池及充电相关的服务: 监听 Uevent、读取sysfs 里中的状态 、广播Intent.ACTION_BATTERY_CHANGED。

(1)、mUEventObserver

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

BatteryService只关注 power_supply 的事件,所以在构造函数注册:

(2)、update()

update读取sysfs文件做到同步取得电池信息, 然后根据读到的状态更新 BatteryService 的成员变量,并广播一个Intent来通知其它关注电源状态的 组件。

当kernel有power_supply事件上报时, mUEventObserver调用update()函数,然后update 调用native_update从sysfs中读取相关状态(com_android_server_BatteryService.cpp):

(3)、sysfs

Linux 驱动 driver 维护着保存电池信息的一组文件 sysfs,供应用程序获

取电源相关状态:

二、Uevent部分

Uevent是内核通知android有状态变化的一种方法,比如USB线插入、拔出,电池电量变化等等。其本质是内核发送(可以通过socket)一个字符串,应用层(android)接收并解释该字符串,获取相应信息。如下图所示,如果其中有信息变化,uevent触发,做出相应的数更新。



Android中的BatteryService及相关组件



1、Androiduevent架构

Android很多事件都是通过uevent跟kernel来异步通信的。其中类UEventObserver是核心。UEventObserver接收kernel的uevent信息的抽象类。

(1)、server层代码
       battery server:
       frameworks/frameworks/base/services/java/com/android/server/SystemServer.java
       frameworks/frameworks/base/services/java/com/android/server/BatteryService.java

(2)、java层代码
      frameworks/base/core/java/android/os/UEventObserver.java
(3)、JNI层代码
        frameworks/base/core/jni/android_os_UEventObserver.cpp
(4)、底层代码
         hardware/libhardware_legacy/uevent/uevent.c

读写kernel的接口socket(PF_NETLINK,SOCK_DGRAM, NETLINK_KOBJECT_UEVENT);
2、UEventObserver的使用


类UEventObserver提供了三个接口给子类来调用:
(1)、onUEvent(UEvent event): 子类必须重写这个onUEvent来处理uevent。
(2)、startObserving(Stringmatch): 启动进程,要提供一个字符串参数。
(3)、stopObserving(): 停止进程。
   例子://在BatteryService.java中

  1. mUEventObserver.startObserving("SUBSYSTEM=power_supply");
  2. private UEventObserver mUEventObserver = new UEventObserver() {
  3. @Override
  4. public void onUEvent(UEventObserver.UEvent event) {
  5. update();
  6. }
  7. };

在UEvent thread中会不停调用 update()方法,来更新电池的信息数据。


3、vold server分析
(1)、在system/vold/NetlinkManager.cpp中:

  1. if ((mSock = socket(PF_NETLINK,SOCK_DGRAM,NETLINK_KOBJECT_UEVENT)) < 0) {
  2. SLOGE("Unable to create uevent socket: %s", strerror(errno));
  3. return -1;
  4. }
  5. if (setsockopt(mSock, SOL_SOCKET, SO_RCVBUFFORCE, &sz, sizeof(sz)) < 0) {
  6. SLOGE("Unable to set uevent socket options: %s", strerror(errno));
  7. return -1;
  8. }
  9. if (bind(mSock, (struct sockaddr *) &nladdr, sizeof(nladdr)) < 0) {
  10. SLOGE("Unable to bind uevent socket: %s", strerror(errno));
  11. return -1;
  12. }


 (2)、然后在system/vold/NetlinkHandler.cpp的NetlinkHandler::onEvent中处理  

  1. void NetlinkHandler::onEvent(NetlinkEvent *evt) {
  2. VolumeManager *vm = VolumeManager::Instance();
  3. const char *subsys = evt->getSubsystem();
  4. if (!subsys) {
  5. SLOGW("No subsystem found in netlink event");
  6. return;
  7. }
  8. if (!strcmp(subsys, "block")) {
  9. vm->handleBlockEvent(evt);
  10. } else if (!strcmp(subsys, "switch")) {
  11. vm->handleSwitchEvent(evt);
  12. } else if (!strcmp(subsys, "battery")) {
  13. } else if (!strcmp(subsys, "power_supply")) {
  14. }
  15. }



(3)、在system/core/libsysutils/src/NetlinkListener.cpp中监听。
4、batteryserver分析

 java代码:frameworks/frameworks/base/services/java/com/android/server/BatteryService.java
   JNI代码: frameworks/base/services/jni/com_android_server_BatteryService.cpp

(1)、BatteryService是跑在system_process当中,在系统初始化的时候启动

如下在BatteryService.java中:

  1. Log.i(TAG, “Starting Battery Service.”);
  2. BatteryService battery = new BatteryService(context);
  3. ServiceManager.addService(“battery”, battery);


   (2)、数据来源
       BatteryService通过JNI(com_android_server_BatteryService.cpp)读取数据。
        BatteryService通过JNI注册的不仅有函数,还有变量。 如下:BatteryService是跑在system_process当中,在系统初始化的时候启动,如下在BatteryService.java中:

  1. //##############在BatteryService.java中声明的变量################
  2. private boolean mAcOnline;
  3. private boolean mUsbOnline;
  4. private int mBatteryStatus;
  5. private int mBatteryHealth;
  6. private boolean mBatteryPresent;
  7. private int mBatteryLevel;
  8. private int mBatteryVoltage;
  9. private int mBatteryTemperature;
  10. private String mBatteryTechnology;
  11. //在BatteryService.java中声明的变量,在com_android_server_BatteryService.cpp中共用,即在com_android_server_BatteryService.cpp中其实操作的也是BatteryService.java中声明的变量。
  12. gFieldIds.mAcOnline = env->GetFieldID(clazz, “mAcOnline”, “Z”);
  13. gFieldIds.mUsbOnline = env->GetFieldID(clazz, “mUsbOnline”, “Z”);
  14. gFieldIds.mBatteryStatus = env->GetFieldID(clazz, “mBatteryStatus”, “I”);
  15. gFieldIds.mBatteryHealth = env->GetFieldID(clazz, “mBatteryHealth”, “I”);
  16. gFieldIds.mBatteryPresent = env->GetFieldID(clazz, “mBatteryPresent”, “Z”);
  17. gFieldIds.mBatteryLevel = env->GetFieldID(clazz, “mBatteryLevel”, “I”);
  18. gFieldIds.mBatteryTechnology = env->GetFieldID(clazz, “mBatteryTechnology”, Ljava/lang/String;”);
  19. gFieldIds.mBatteryVoltage = env->GetFieldID(clazz, “mBatteryVoltage”, “I”);
  20. gFieldIds.mBatteryTemperature = env->GetFieldID(clazz, “mBatteryTemperature”, “I”);
  21. //上面这些变量的值,对应是从下面的文件中读取的,一只文件存储一个数值。
  22. #define AC_ONLINE_PATH “/sys/class/power_supply/ac/online”
  23. #define USB_ONLINE_PATH “/sys/class/power_supply/usb/online”
  24. #define BATTERY_STATUS_PATH “/sys/class/power_supply/battery/status”
  25. #define BATTERY_HEALTH_PATH “/sys/class/power_supply/battery/health”
  26. #define BATTERY_PRESENT_PATH “/sys/class/power_supply/battery/present”
  27. #define BATTERY_CAPACITY_PATH “/sys/class/power_supply/battery/capacity”
  28. #define BATTERY_VOLTAGE_PATH “/sys/class/power_supply/battery/batt_vol”
  29. #define BATTERY_TEMPERATURE_PATH “/sys/class/power_supply/battery/batt_temp”
  30. #define BATTERY_TECHNOLOGY_PATH “/sys/class/power_supply/battery/technology”


  (3)、数据传送
     BatteryService主动把数据传送给所关心的应用程序,所有的电池的信息数据是通过Intent传送出去的。在BatteryService.java中,Code如下:

  1. Intent intent = new Intent(Intent.ACTION_BATTERY_CHANGED);
  2. intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY);
  3. intent.putExtra(“status”, mBatteryStatus);
  4. intent.putExtra(“health”, mBatteryHealth);
  5. intent.putExtra(“present”, mBatteryPresent);
  6. intent.putExtra(“level”, mBatteryLevel);
  7. intent.putExtra(“scale”, BATTERY_SCALE);
  8. intent.putExtra(“icon-small”, icon);
  9. intent.putExtra(“plugged”, mPlugType);
  10. intent.putExtra(“voltage”, mBatteryVoltage);
  11. intent.putExtra(“temperature”, mBatteryTemperature);
  12. intent.putExtra(“technology”, mBatteryTechnology);
  13. ActivityManagerNative.broadcastStickyIntent(intent, null);

(4)、数据接收

应用如果想要接收到BatteryService发送出来的电池信息,则需要注册一个Intent为Intent.ACTION_BATTERY_CHANGED的BroadcastReceiver。


       注册方法如下:

  1. IntentFilter mIntentFilter = new IntentFilter();
  2. mIntentFilter.addAction(Intent.ACTION_BATTERY_CHANGED);
  3. registerReceiver(mIntentReceiver, mIntentFilter);
  4. private BroadcastReceiver mIntentReceiver = new BroadcastReceiver() {
  5. @Override
  6. public void onReceive(Context context, Intent intent) {
  7. // TODO Auto-generated method stub
  8. String action = intent.getAction();
  9. if (action.equals(Intent.ACTION_BATTERY_CHANGED)) {
  10. int nVoltage = intent.getIntExtra(“voltage”, 0);
  11. if(nVoltage!=0){
  12. mVoltage.setText(“V: ” + nVoltage + “mV – Success…”);
  13. }
  14. else{
  15. mVoltage.setText(“V: ” + nVoltage + “mV – fail…”);
  16. }
  17. }
  18. }
  19. };

(5)、数据更新
        电池的信息会随着时间不停变化,自然地,就需要考虑如何实时的更新电池的数据信息。在BatteryService启动的时候,会同时通过UEventObserver启动一个onUEvent Thread。每一个Process最多只能有一个onUEvent Thread,即使这个Process中有多个UEventObserver的实例。当在一个Process中,第一次Call startObserving()方法后,这个UEvent thread就启动了。而一旦这个UEvent thread启动之后,就不会停止。


       //在BatteryService.java中

  1. mUEventObserver.startObserving(“SUBSYSTEM=power_supply”);
  2. private UEventObserver mUEventObserver = new UEventObserver() {
  3. @Override
  4. public void onUEvent(UEventObserver.UEvent event) {
  5. update();
  6. }
  7. };



    在UEvent thread中会不停调用 update()方法,来更新电池的信息数据。

android 电池(三):android电池系统【转】的更多相关文章

  1. Android进阶(三)android httpClient 支持HTTPS的访问方式

    项目中Android https请求地址遇到了这个异常(无终端认证): javax.net.ssl.SSLPeerUnverifiedException: No peer certificate 是S ...

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

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

  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 电池关机充电 androidboot.mode charger关机充电 充电画面显示 平台信息:内核:linux2.6/linux3.0系统:android/android4. ...

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

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

  7. Android 电池系列

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

  8. Android 电池关机充电

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

  9. 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 代码 ...

随机推荐

  1. 定时执行线程池ScheduledExecutorService的使用

    ScheduledExecutorService progressExecutorService = Executors.newScheduledThreadPool(1); ScheduledFut ...

  2. Atitit.跨平台预定义函数 魔术方法 魔术函数 钩子函数 api兼容性草案 v2 q216  java c# php js.docx

    Atitit.跨平台预定义函数 魔术方法 魔术函数 钩子函数 api兼容性草案 v2 q216  java c# php js.docx 1.1. 预定义函数 魔术方法 魔术函数是什么1 1.2. & ...

  3. 已经mock类中引用的其它service类,但是在invoke私有方法的时候,该service类是空值

    错误原因:没有在开始测试用例的时候,初始化类的所有注解方法. 解决方法: 使用mock方法创建mock对象时,需要在测试用例执行前执行以下代码.通常, 这句代码可以放在测试基类或者@Before 中. ...

  4. ulimit的坑

    linux ulimit的若干坑 - ulimit真不是乱设的 原创 2016年11月16日 22:15:05 标签: linux 1997 soft和hard一起设置才好使 * soft nofil ...

  5. JQuery不能加载click事件的问题

    <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding= ...

  6. 转载 OS js oc相互调用(JavaScriptCore) ---js调用iOS ---js里面直接调用方法

    OS js oc相互调用(JavaScriptCore)   接着上节我们讲到的iOS调用js 下来我们使用js调用iOS js调用iOS分两种情况 一,js里面直接调用方法 二,js里面通过对象调用 ...

  7. 自定义验证----required属性

    1,required属性 - 表示字段不能为空(注意:只有用户单击“提交”按钮提交表单的时候,浏览器才会执行验证.目前HTML5不支持指定验证的时间,而且验证消息的样式和内容各个浏览器不大一样,不能修 ...

  8. Asp.Net MVC3中如何进行单元测试?

    下面我们就以一个示例演示一下如何进行单元测试? public Model.UserInfo UpdateEntity(Model.UserInfo entity) { db.UserInfo.Atta ...

  9. 九度OJ 1336:液晶屏裁剪 (GCD)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:983 解决:228 题目描述: 苏州某液晶厂一直生产a * b大小规格的液晶屏幕,由于该厂的加工工艺限制,液晶屏的边长都为整数.最近由于市场 ...

  10. Python高级入门01-property

    JAVA中存在对变量 私有化,公开,保护... 私有化时候,需要提供一个公开的get 和 set方法对外公开,让别人进行调用 python中同样存在    私有化变量定义是__是这个双下划线,eg:_ ...