`

[转]Android虚拟电源管理驱动

 
阅读更多

Android系统如果没有电源管理相关的驱动程序,在启动时将会提示如下错误:

I/SystemServer( 50): Starting Battery Service.
E/BatteryService( 50): Could not open '/sys/class/power_supply/ac/online'
E/BatteryService( 50): Could not open '/sys/class/power_supply/usb/online'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/present'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/capacity'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/batt_vol'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/batt_temp'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/status'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/health'
E/BatteryService( 50): Could not open '/sys/class/power_supply/battery/technology'
I/SystemServer( 50): Starting Hardware Service.

此时启动完成后将显示Poweroff界面
图1


图2


这个问题的原因很简单,因为缺少相关电源部分驱动程序,Android系统无法获取/sys/class/power_supply/下的信息,从而导致启动不能。

对于开发板,开发的初期一般没有电源管理驱动程序,那要如何解决这个问题呢?

网上流传的解决办法是通过修改Android源码中的com_android_server_BatteryService.cpp,让android系统不去去读相关信息,强行赋值为电量充足。

但是如果无法重新编译android文件系统,应该如何处理?

其实,有一个更好的解决方法,就是在Linux内核中写一个虚拟电源驱动,这样的好处是增加电源芯片后,直接在此驱动上修改即可,也不用修改Android源码了。

/sys/class # cd power_supply/
/sys/class/power_supply # ls
ac battery
/sys/class/power_supply # cd battery/
/sys/class/power_supply/battery # ls
capacity manufacturer subsystem uevent
charge_counter power technology voltage_avg
current_avg present temp
device serial_number temp_ambient
health status type
/sys/class/power_supply/battery #

驱动源码:
=========================================================================================================
/*
* Fake Battery driver for android
*
* Copyright © 2009 Rockie Cheng <
aokikyon@gmail.com >
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/

#include <linux/module.h>
#include <linux/err.h>
#include <linux/platform_device.h>
#include <linux/power_supply.h>
#include <linux/jiffies.h>
#include <linux/sched.h>

#define BAT_STAT_PRESENT 0x01
#define BAT_STAT_FULL 0x02
#define BAT_STAT_LOW 0x04
#define BAT_STAT_DESTROY 0x08
#define BAT_STAT_AC 0x10
#define BAT_STAT_CHARGING 0x20
#define BAT_STAT_DISCHARGING 0x40

#define BAT_ERR_INFOFAIL 0x02
#define BAT_ERR_OVERVOLTAGE 0x04
#define BAT_ERR_OVERTEMP 0x05
#define BAT_ERR_GAUGESTOP 0x06
#define BAT_ERR_OUT_OF_CONTROL 0x07
#define BAT_ERR_ID_FAIL 0x09
#define BAT_ERR_ACR_FAIL 0x10

#define BAT_ADDR_MFR_TYPE 0x5F

static int android_ac_get_prop(struct power_supply *psy,
enum power_supply_property psp,
union power_supply_propval *val)
{

switch (psp)
{
case POWER_SUPPLY_PROP_ONLINE:
val->intval = BAT_STAT_AC;
break;
default:
break;
}
return 0;
}

static enum power_supply_property android_ac_props[] =
{
POWER_SUPPLY_PROP_ONLINE,
};

static struct power_supply android_ac =
{
.name = "ac",
.type = POWER_SUPPLY_TYPE_MAINS,
.properties = android_ac_props,
.num_properties = ARRAY_SIZE(android_ac_props),
.get_property = android_ac_get_prop,
};

static int android_bat_get_status(union power_supply_propval *val)
{

val->intval = POWER_SUPPLY_STATUS_FULL;
return 0;
}

static int android_bat_get_health(union power_supply_propval *val)
{

val->intval = POWER_SUPPLY_HEALTH_GOOD;
return 0;
}

static int android_bat_get_mfr(union power_supply_propval *val)
{

val->strval = "Rockie";
return 0;
}

static int android_bat_get_tech(union power_supply_propval *val)
{
val->intval = POWER_SUPPLY_TECHNOLOGY_LION;
return 0;
}

static int android_bat_get_property(struct power_supply *psy,
enum power_supply_property psp,
union power_supply_propval *val)
{
int ret = 0;

switch (psp)
{
case POWER_SUPPLY_PROP_STATUS:
ret = android_bat_get_status(val);
if (ret)
return ret;
break;
case POWER_SUPPLY_PROP_PRESENT:
val->intval = BAT_STAT_PRESENT;
break;

case POWER_SUPPLY_PROP_HEALTH:
ret = android_bat_get_health(val);
break;
case POWER_SUPPLY_PROP_MANUFACTURER:
ret = android_bat_get_mfr(val);
if (ret)
return ret;
break;
case POWER_SUPPLY_PROP_TECHNOLOGY:
ret = android_bat_get_tech(val);
if (ret)
return ret;
break;
case POWER_SUPPLY_PROP_VOLTAGE_AVG:
val->intval = 3;
break;
case POWER_SUPPLY_PROP_CURRENT_AVG:
val->intval = 3;
break;
case POWER_SUPPLY_PROP_CAPACITY:
val->intval = 100;
break;
case POWER_SUPPLY_PROP_TEMP:
val->intval = 50;
break;
case POWER_SUPPLY_PROP_TEMP_AMBIENT:
val->intval = 50;
break;
case POWER_SUPPLY_PROP_CHARGE_COUNTER:
val->intval = 10;
break;
case POWER_SUPPLY_PROP_SERIAL_NUMBER:
break;
default:
ret = -EINVAL;
break;
}
return ret;
}

static enum power_supply_property android_bat_props[] = {
POWER_SUPPLY_PROP_STATUS,
POWER_SUPPLY_PROP_PRESENT,
POWER_SUPPLY_PROP_HEALTH,
POWER_SUPPLY_PROP_TECHNOLOGY,
POWER_SUPPLY_PROP_VOLTAGE_AVG,
POWER_SUPPLY_PROP_CURRENT_AVG,
POWER_SUPPLY_PROP_CAPACITY,
POWER_SUPPLY_PROP_TEMP,
POWER_SUPPLY_PROP_TEMP_AMBIENT,
POWER_SUPPLY_PROP_MANUFACTURER,
POWER_SUPPLY_PROP_SERIAL_NUMBER,
POWER_SUPPLY_PROP_CHARGE_COUNTER,
};

/*********************************************************************
* Initialisation
*********************************************************************/

static struct platform_device *bat_pdev;

static struct power_supply android_bat =
{
.properties = android_bat_props,
.num_properties = ARRAY_SIZE(android_bat_props),
.get_property = android_bat_get_property,
.use_for_apm = 1,
};

static int __init android_bat_init(void)
{
int ret = 0;

bat_pdev = platform_device_register_simple("battery", 0, NULL, 0);

ret = power_supply_register(&bat_pdev->dev, &android_ac);
if (ret)
goto ac_failed;

android_bat.name = bat_pdev->name;

ret = power_supply_register(&bat_pdev->dev, &android_bat);
if (ret)
goto battery_failed;

goto success;

power_supply_unregister(&android_bat);
battery_failed:
power_supply_unregister(&android_ac);
ac_failed:
platform_device_unregister(bat_pdev);
success:
return ret;
}

static void __exit android_bat_exit(void)
{
power_supply_unregister(&android_bat);
power_supply_unregister(&android_ac);
platform_device_unregister(bat_pdev);
}

module_init(android_bat_init);
module_exit(android_bat_exit);

MODULE_AUTHOR("Rockie Cheng < aokikyon@gmail.com >");
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Fake Battery driver for android");

==================================================================================================================
阿虚(Rockie Cheng)

本文为原创,转载请注明出处 http://hi.baidu.com/aokikyon

如有错误,欢迎指正!

分享到:
评论

相关推荐

    android手机虚拟网卡驱动Remote NDIS based Internet Sharing Device

    这个驱动使得电脑能够识别并通信与Android手机建立的虚拟网络接口。 4. **配置过程**: 配置USB tethering通常涉及以下步骤:首先在手机上开启USB调试模式,然后通过USB连接手机和电脑;接着在手机上选择USB ...

    android底层开发技术实战详解内核移植和驱动

    常见的驱动类型包括:显示驱动、触摸屏驱动、电源管理驱动、音频驱动、摄像头驱动等。驱动开发涉及驱动模型的理解、设备树的编写、中断处理、DMA传输、I/O控制等技术。 四、传感器驱动 在Android系统中,传感器驱动...

    Android驱动开发高级培训(一)

    Android驱动开发需要关注低功耗设计,如休眠和唤醒机制,以及电源状态转换。了解PMIC(电源管理集成电路)和PM子系统有助于优化电池寿命。 在图形和显示驱动方面,Android使用SurfaceFlinger作为显示服务器,处理...

    EC20 linux android驱动

    3. 网络接口管理:驱动会创建并管理虚拟网络接口(如`wwan0`),使得应用程序可以通过标准的网络API与EC20模块通信。 4. 电源管理:优化电源控制,确保在不使用时降低功耗,同时能快速恢复到活动状态。 5. 错误...

    Android系统架构及其驱动研究

    在驱动研究方面,Android系统中的驱动通常包括多个类别,如Binder驱动、WIFI驱动、蓝牙驱动、电源管理驱动等。以Binder驱动为例,它是一种基于Linux的IPC(进程间通信)机制,是Android系统中用于进程通信的关键组件...

    Android的Linux内核与驱动程序

    2. **电源管理**:Android内核实现了精细的电源管理策略,确保在不同使用场景下能有效节能。比如,CPU频率动态调整、休眠模式和待机状态的优化。 3. **文件系统**:Android使用了多种文件系统,如EXT4、F2FS(闪存...

    Android驱动开发高级培训(三)

    电源管理驱动负责监控和控制设备的功耗,确保高效能和低能耗。 4. **音频驱动**:音频编解码器驱动处理声音输入和输出,与音频硬件进行交互,提供高质量的音频体验。 5. **传感器驱动**:包括加速度计、陀螺仪、...

    android 底层驱动原理

    - **Linux内核层**:这一层包括Linux操作系统和驱动程序,提供对硬件的基本访问,例如电源管理、内存管理、进程管理等功能。它依赖于Linux 2.6内核版本,不支持Linux 2.4内核版本。这一层还包括了专门针对Android...

    Android 底层驱动原理

    6. **电源管理**:针对移动设备进行了电源管理和节能优化。 7. **多媒体支持**:增强了对多媒体格式的支持。 #### 二、Android底层驱动概述 **2.1 Android底层驱动的详细内容** Android底层驱动涵盖了从Linux内核...

    基于Android的触摸屏驱动开发.pdf

    其中,Android运行环境通过Dalvik虚拟机为每个应用程序提供独立的运行实例,而Linux内核则负责基础服务,如安全、内存管理,同时也为硬件平台提供驱动模型,确保系统的可移植性。 【Android用户输入系统框架】 在...

    USB7.1声卡驱动

    3. 功能支持:驱动程序提供了各种音效设置选项,如均衡器、环绕模式、虚拟化效果等,用户可以根据个人喜好调整。 4. 兼容性:确保USB声卡能与各种软件、游戏和多媒体应用良好配合,提供稳定的声音输出。 5. 更新与...

    Android 8.1 系统多网口实现方法

    此外,你也需要考虑电源管理,因为多网口可能会增加功耗。 在安全方面,每个接口的安全设置(如防火墙规则、访问控制列表)也需要单独管理。这可能涉及到修改`Netd`服务,使其能够处理多个接口的隔离和配置。 最后...

    Android 系统开发讲义

    开发者还需要熟悉Android特定的驱动架构,如Binder通信机制、电源管理等。 13. Android安全性机制 Android系统中包含了一系列的安全性机制,比如应用程序沙箱环境、权限管理、数据加密和安全存储等,以确保用户数据...

    USB转串口驱动程序(CP2102)

    - 安装完成后,设备管理器中应出现新的虚拟COM端口,表示驱动已成功安装。 4. **应用领域** - 开发和调试:在开发嵌入式系统时,开发者常使用CP2102来连接串口设备,如单片机、Arduino等,进行程序的下载和调试。...

    智能机驱动

    这个驱动可能包括了USB驱动、电源管理驱动、GPU驱动等,旨在确保电脑能够正确识别和操作采用MTK65xx芯片的智能设备。安装这个驱动后,用户可以连接设备进行数据同步、备份、软件更新,或者进行故障排查和刷机操作。 ...

    htC MT6573驱动

    6. **电源管理驱动**:MT6573驱动可能还包括电源管理部分,优化设备的电池使用,实现节能和延长续航。 7. **传感器驱动**:现代智能手机通常配备多种传感器,如加速度计、陀螺仪、磁力计等,这些都需要对应的驱动...

    博士BMI270开源驱动

    5. **电源管理**:考虑到BMI270的低功耗特性,驱动应支持动态电源管理,根据应用需求调整传感器的工作模式,以延长设备电池寿命。 6. **中断处理**:BMI270可能支持中断功能,驱动需要正确处理中断事件,例如当...

    android Jollen's blog

    1. **Android环境搭建**:讲解如何安装Android Studio,配置SDK,以及设置AVD(Android虚拟设备)进行模拟器测试。 2. **基本UI构建**:介绍XML布局文件的编写,包括各种视图元素如Button、TextView、EditText等,...

Global site tag (gtag.js) - Google Analytics