在 Android的上层是使用 goToSleep() 这个函数让系统进入休眠的。但是这个命令为什么会让 Android 进入 Suspend Mode。 以前在做其他系统的时候, 一般都要自己手工去控制 apm_bios 这个设备的,比如使用 ioctl() 调用 apm_bios。所有的系统其原理都是差不多的。只是 Android 加入了一个封装, 使程序员可以更简单的操作, 可以不理会底层是如何操作的, 现在解析一个 goToSleep() 是如何工作的。 PowerManagerService.java 是 framework 层负责管理 PowerManager 的。goToSleep 就是在这个函数中定义的: goToSleep()-> goToSleepLocked()-> setPowerState()-> setScreenStateLocked()-> Power.setScreenState(on); // android_os_Power.cpp goToSleep() 在 PowerManagerService.java 中经过一系列的处理之后, 最终会进入 android_os_Power.cpp 文件中的 setScreenState() 这个函数中。 看一下 setScreenState() 的实现: // android_os_Power.cpp static int
setScreenState(JNIEnv *env, jobject clazz, jboolean on)
{
return set_screen_state(on);
}
而 set_screen_state() 在 power.c 文件中实现: int
set_screen_state(int on)
{
QEMU_FALLBACK(set_screen_state(on));
LOGI("*** set_screen_state %d", on); initialize_fds(); //LOGI("go_to_sleep eventTime=%lld now=%lld g_error=%s\n", eventTime,
// systemTime(), strerror(g_error));
if (g_error) return g_error; char buf[32];
int len;
if(on)
len = sprintf(buf, on_state);
else
len = sprintf(buf, off_state);
len = write(g_fds[REQUEST_STATE], buf, len);
if(len < 0) {
LOGE("Failed setting last user activity: g_error=%d\n", g_error);
}
return 0;
}
可以发现, 其最终是通过 write(g_fds[REQUEST_STATE], buf, len); 令系统进入休眠的。 通过分析这个文件的代码, g_fds[REQUEST_STATE] = open("/sys/power/state", O_RDWR); on_state = "wake", off_state = "standby", 也就是写 standby 到 /sys/power/state 就可以令系统进行休眠了。 /sys/power/state 的实现代码可以看 $(Kernel)/kernel/power/main.c 这个文件是怎么实现的。 所以,当文件系统起来的时候可以echo standby > /sys/power/state 令系统进入休眠

1, Android中支持的电源状态:

PM_SUSPEND_ON -- 设备处于全电源状态,也就是正常工作状态;

PM_SUSPEND_MEM -- suspend to memory,设备进入睡眠状态,但所有的数据还保存在内存中,只有某些外部中断才可以唤醒设备。

PM_SUSPEND_STANDBY ----- 在大部分的Android设备中均不支持。

2, Early Suspend / Late Resume

在Android中,增加了系统休眠的层次,把休眠划分为深度睡眠(sleep)和浅度睡眠(idle)。

因此当系统往 /sys/power/state 节点写入 mem (如在命令行写入: echo mem > /sys/power/state)将会使系统进入睡眠。参考HAL中:hardware/libhardware_legacy/power/power.c : set_screen_state ------> write (on / mem ) to /sys/power/state

浅度睡眠 仅仅是关掉背光,fb, sensor,触摸屏等在关屏状态下不需要使用的设备,而整个CPU和大部分外设还是正常工作的。

深度睡眠sleep 与 浅度睡眠 idle 之间切换的过程如上所示。

3, Android的 Wake Lock

Android系统提供了两种类型的锁:

WAKE_LOCK_SUSPEND -- 阻止系统进入suspend状态;

WAKE_LOCK_IDLE -- 阻止系统进入idle状态;

wake lock 可以设置超时释放,在持有wake lock一个固定时间之后自动释放。一般应用在系统正在处理一些事情的时候,防止系统进入深度睡眠而干扰了正在处理的任务; 尤其是在做了硬件唤醒的,当唤醒之后要处理一些响应,为了防止系统马上再次进入休眠,在设置一个超时锁。

如系统能否进入深度睡眠,当系统处在idle时,会不断判断是否还有WAKE_LOCK_SUSPEND

4, 电源状态切换的调试:

/sys/power/state

/sys/power/wake_lock

/sys/power/wake_unlock

1) cat /sys/power/state -----查看系统状态, echo mem > /sys/power/state ---- 将系统设置为进入休眠

2) echo "name" > /sys/power/wake_lock ----- 申请一个锁

cat /sys/power/wake_lock -----查看系统的wake lock 情况

wake_unlock 同样。

3) echo 15 > /sys/module/wakelock/parameters/debug_mask ------------

这样wakelock的驱动会把每次的wakelock操作都打印在console上,对于调试为什么suspend不下去这类的问题很有用。如下所示:

[ 1062.912297] wake_lock: mmc_delayed_work, stop expire timer
[ 1062.922395] wake_unlock: mmc_delayed_work, start expire timer, 990
[ 1062.931174] wake_lock: event0-79, start expire timer, 989
[ 1062.933710] wake_lock: event0-79, start expire timer, 989
[ 1062.939081] wake_lock: event0-79, start expire timer, 989
[ 1062.961143] wake_lock: event0-79, start expire timer, 986

更多相关文章

  1. Android欢迎页面2秒钟后自动跳转到主页面
  2. android 当系统存在多个Launcher时,如何设置开机自动进入默认的La
  3. android 命令(adb shell)进入指定模拟器或设备
  4. android修改进入工程模式
  5. android 休眠
  6. FregServer进程,发送BC_TRANSACTION,唤醒ServiceManager进程,返回BR
  7. Linux Kernel and Android(安卓)休眠与唤醒
  8. 【Android休眠】之Android对PowerKey事件的处理(2)EventHub
  9. Android(安卓)CM12.1 关机流程

随机推荐

  1. 如何实现Android桌面小部件(二)
  2. [转载]Android触屏滑动事件总结 (2011-08
  3. android 异步显示View
  4. android PreferenceActivity结合Preferen
  5. Roman Guy的Android Trick系列文章笔记
  6. Android中几种图像特效处理的小技巧,比如
  7. Android中的定位Demo
  8. 怎样在android中添加背景图片?
  9. 感谢Google Android
  10. android的各种布局