环境:
PC: debian-7.6.0
ARM CPU: S3C2416
Linux-Kernel: 3.6.0(FriendlyARM)
U-boot: 1.3.4

一、问题来源

依据须要,在S3C2416上加入中断睡眠和唤醒功能。于是我就查查Linux支持S3C2416的睡眠模式:

cat /sys/power/state

运行完,万万没想到:居然是空的,该命令没有不论什么输出!也就是说,我的内核眼下不支持不论什么方式的睡眠。

不可能啊!之前我用S3C2440的CPU(内核版本号Linux_2_6_31)实现了中断的睡眠和唤醒,而S3C2416的Linux 3.6内核配置就是參照着Linux 2.6.31进行的配置。

Linux 2.6.31运行:

S3C2440-Linux 2.6.31:
cat /sys/power/state
返回:mem

经过各种颠倒黑白颠三倒四的尝试都失败后,灵机一闪:既然是cat后没反应,那就沿着cat /sys/power/state调用的函数从上到下,一步一步查。

指导思想有了,那就顺蔓摸瓜的进行调试。

、调试流程                                                                         

显示cat /sys/power/state运行结果的函数是state_show():

kernel/power/main.c

static ssize_t state_show(struct kobject *kobj, struct kobj_attribute *attr, char *buf)

{

char *s = buf;

printk("<0>""[#1]kernel speaking\n");

#ifdef CONFIG_SUSPEND

printk("<0>""[#2]kernel speaking\n");

int i;



for (i = 0; i < PM_SUSPEND_MAX; i++) {

if (pm_states[i] && valid_state(i))

s += sprintf(s,"%s ", pm_states[i]);

}

#endif

......

}

编译、烧写内核,运行cat命令。有[#2]kernel speaking信息说明CONFIG_SUSPEND宏是有定义的。

pm_states[]数组在同文件也有定义,那么接下来就看函数valid_state():

kernel/power/suspend.c

bool valid_state(suspend_state_t state)

{

printk("<0>""[#3]kernel speaking[suspend_ops-%d]\n", suspend_ops);

return suspend_ops && suspend_ops->valid && suspend_ops->valid(state);

}

这个函数測试后显示suspend_ops指针为0。哈哈,原因找到了,接着顺藤摸瓜。

suspend_ops指向struct platform_suspend_ops结构体,每一个成员的作用參见附录博文。suspend_ops的赋值在在函数suspend_set_ops():

kernel/power/suspend.c

void suspend_set_ops(const struct platform_suspend_ops *ops)

{

lock_system_sleep();

suspend_ops = ops;

unlock_system_sleep();

}



suspend_set_ops()的调用在函数s3c_pm_init():

arch/arm/plat-samsung/pm.c

int __init s3c_pm_init(void)

{

printk("S3C Power Management, Copyright 2004 Simtec Electronics\n");



suspend_set_ops(&s3c_pm_ops);

return 0;

}





s3c_pm_init()的调用在函数smdk_machine_init():

void __init smdk_machine_init(void)

{

......

s3c_pm_init();

}

而smdk_machine_init()是在详细的板上调用,我用的S3C2416是友善之臂的核心板,在mini2451_machine_init():

arch/arm/mach-s3c24xx/mach-mini2451.c

static void __init mini2451_machine_init(void)

{

#if defined(CONFIG_S3C24XX_SMDK)

printk("<0>""[#4]kernel speaking\n");

smdk_machine_init();

#endif

}

注意,结果出来了:[#4]kernel speaking信息没有说明CONFIG_S3C24XX_SMDK宏未定义。

而CONFIG_S3C24XX_SMDK宏是在配置内核是选定MINI2451后默认选定的,仅仅是被友善之臂凝视掉了,在当前文件夹下的Kconfig

config MACH_MINI2451
bool "MINI2451"
#select S3C24XX_SMDK
select S3C_DEV_FB
select S3C_DEV_HSMMC
select S3C_DEV_HSMMC1
select S3C_DEV_NAND
select S3C_DEV_USB_HOST
select S3C2416_SETUP_SDHCI
select WIRELESS_EXT
select WEXT_SPY
select WEXT_PRIV
select AVERAGE
help
Say Y here if you are using an FriendlyARM MINI2451

三、加入中断方式的睡眠和唤醒功能                                                

1、改动arch/arm/mach-s3c24xx/Kconfig。去掉S3C24XX_SMDK选项前的凝视

2、改动arch/arm/mach-s3c24xx/common-smdk.c,去掉s3c_pm_init之外的东东。并加入唤醒中断唤醒源

下面改动參见内核Documentation/arm/Samsung-S3C24XX/Suspend.txt

static irqreturn_t button_irq(int irq, void *pw)
{
return IRQ_HANDLED;
} void __init smdk_machine_init(void)
{ /* Configure the LEDs (even if we have no LED support)*/
#if 0
int ret = gpio_request_array(smdk_led_gpios,
ARRAY_SIZE(smdk_led_gpios));
if (!WARN_ON(ret < 0))
gpio_free_array(smdk_led_gpios, ARRAY_SIZE(smdk_led_gpios)); if (machine_is_smdk2443())
smdk_nand_info.twrph0 = 50; s3c_nand_set_platdata(&smdk_nand_info); platform_add_devices(smdk_devs, ARRAY_SIZE(smdk_devs));
#endif
request_irq(IRQ_EINT0, button_irq, IRQF_TRIGGER_FALLING, "IRQ_EINT0", NULL);
enable_irq_wake(IRQ_EINT0); s3c_pm_init();
}

3、在arch/arm/mach-s3c24xx文件夹下加入文件common-smdk.h

/* arch/arm/mach-s3c24xx/common-smdk.h
*
* Copyright (c) 2006 Simtec Electronics
* Ben Dooks <ben@simtec.co.uk>
*
* Common code for SMDK2410 and SMDK2440 boards
*
* http://www.fluff.org/ben/smdk2440/
*
* 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.
*/ extern void smdk_machine_init(void);

4、改动arch/arm/mach-s3c24xx/mach-mini2451.c,加入smdk_machine_init所在的头文件,即common-smdk.h

#include "common-smdk.h"



四、S3C2416睡眠唤醒測试                                                        

睡眠:

cat /sys/power/state

返回:mem

echo mem > /sys/power/state





睡好后,触发IRQ_EINT0中断(按键),CPU恢复之前的状态。部分截图:



为了更好的測试,能够再睡眠前执行一个输出递增数据的程序,这样能够直观的看到唤醒后的状态。





五、S3C2416睡眠唤醒的底层驱动实现                                  

S3C2416睡眠的底层实现





六、总结                                                                                        

1、灵感非常重要。

PS:“天才就是百分之中的一个的灵感加百分之99的汗水”。这仅仅是前半句,后半句是:“但那百分之中的一个的灵感,往往比百分之九十九的汗水来得重要”。

为什么从小听到的仅仅有前半句、、、不吐槽了。

2、正确的方法就是捷径。只是,歧路也能让人增长姿势。

3、Linux内核,尤其原理方面的东东,还知之甚少。路漫长爱漫长、、、



4、附件:Linux下截屏命令: import pic_name

标准linu休眠和唤醒机制分析(一)

标准linu休眠和唤醒机制分析(二)

标准linu休眠和唤醒机制分析(三)

标准linu休眠和唤醒机制分析(四)

从sys/power/state分析并实现S3C2416的睡眠和唤醒的更多相关文章

  1. /sys/power/state

    kernel/power/main.c中: /** * state - control system power state. * * show() returns what states are s ...

  2. 《linux 内核全然剖析》 sys.c 代码分析

    sys.c 代码分析 setregid /* * This is done BSD-style, with no consideration of the saved gid, except * th ...

  3. SYS/BIOS实例分析

    SYS/BIOS简介 SYS/BIOS是一个可扩展的实时内核(或者说是操作系统),其提供了许多模块化的APIs(应用程序接口),支持抢占式多线程,硬件抽象,实时分析和配置工具,其设计目的是为了最大限度 ...

  4. 各种 Java Thread State 分析

    转自:https://www.cnblogs.com/zhengyun_ustc/archive/2013/03/18/tda.html 1,线程状态为“waiting for monitor ent ...

  5. Suspend to RAM和Suspend to Idle分析,以及在HiKey上性能对比

    Linux内核suspend状态 Linux内核支持多种类型的睡眠状态,通过设置不同的模块进入低功耗模式来达到省电功能.目前存在四种模式:suspend to idle.power-on standb ...

  6. Android电源管理-休眠简要分析

    一.开篇 1.Linux 描述的电源状态 - On(on)                                                 S0 -  Working - Standb ...

  7. Android系统进程Zygote启动过程的源代码分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6768304 在Android系统中,所有的应用 ...

  8. SEAndroid安全机制中的进程安全上下文关联分析

    前面一篇文章分析了文件安全上下文关联过程.可是在SEAndroid中,除了要给文件关联安全上下文外,还须要给进程关联安全上下文.由于仅仅有当进程和文件都关联安全上下文之后,SEAndroid安全策略才 ...

  9. Android4.4 Framework分析——Zygote进程的启动过程

    Android启动过程中的第一个进程init.在启动过程中会启动两个关键的系统服务进程ServiceManager和Zygote. 本文要介绍的就是Zygote进程的启动,Zygote俗称孵化器,专门 ...

随机推荐

  1. linux 查找文件或者服务

    [root@localhost ~]# whereis mysql mysql: /usr/bin/mysql /usr/lib/mysql /usr/share/mysql /usr/share/m ...

  2. 【linux】ubuntu中上下左右键变为^[[A^[[B^[[D^[[C问题处理

    问题现象: 使用上下左右键时,结果为 ^[[A^[[B^[[D^[[C,如图: 原因在于ubuntu系统自带的 vi 不完整导致. 解决方法:安装完整的vi $ sudo apt-get instal ...

  3. 调用OpenCVSharp进行拍照

    一.核心代码: using OpenCvSharp; using System; using System.Collections.Generic; using System.IO; using Sy ...

  4. Linux中进程与线程及CPU使用率查询

    一.进程查询: ps -e -o 'pid,comm,args,pcpu,rsz,vsz,stime,user,uid' 说明:PCPU是Cpu使用率,8核最多是800. 或者 ps -aux 二.线 ...

  5. Spring MVC 实现文件的上传和下载

    前些天一位江苏经贸的学弟跟我留言问了我这样一个问题:“用什么技术来实现一般网页上文件的上传和下载?是框架还是Java中的IO流”.我回复他说:“使用Spring MVC框架可以做到这一点,因为Spri ...

  6. python模块之HTMLParser之穆雪峰的案例(理解其用法原理)

    # -*- coding: utf-8 -*- #python 27 #xiaodeng #python模块之HTMLParser之穆雪峰的案例(理解其用法原理) #http://www.cnblog ...

  7. Heroku免费版限制

    SLEEPS AFTER 30 MINS OF INACTIVITY   30分钟无人访问就休眠 Verified accounts come with a monthly pool of 1000 ...

  8. Linux-中断的本质

    更好的参考:CPU的内部的中断 学习中断是为了理解信号,因为信号即软中断. 中断不是轮询!比如最常见的在UART通信过程中(收发数据),有两种方式,一种是中断,一种是轮询.如果中断是轮询,这两者就没区 ...

  9. 清除li内a标签的float=left实现a标签在li内居中显示(ul内li不居中显示)

    写在前面: 修改cnblogs主页面菜单显示问题. 问题描述:在给主菜单添加hover样式后发现菜单内容并未居中.见图1. 网上搜索到资料其中一篇讲的可以说简明扼要了,也是伸手党的福利(点我查看原文) ...

  10. [Spring学习笔记 5 ] Spring AOP 详解1

    知识点回顾:一.IOC容器---DI依赖注入:setter注入(属性注入)/构造子注入/字段注入(注解 )/接口注入 out Spring IOC容器的使用: A.完全使用XML文件来配置容器所要管理 ...