serial.read函数_过程调用属于面向对象吗

serial.read函数_过程调用属于面向对象吗1.电源管理的状态Android的Linux内核为系统提供了4种电源状态,内核的源码为当中的3种定义了名字和相应的宏定义,名字定义在kernel/power/suspend.c中:1234567constchar*constpm_states[PM_SUSPEND_MAX]={#ifdefCONFI…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

1. 电源管理的状态

Android的Linux内核为系统提供了4种电源状态,内核的源码为当中的3种定义了名字和相应的宏定义,名字定义在kernel/power/suspend.c中:

1

2

3

4

5

6

7

constchar*const pm_states[PM_SUSPEND_MAX]={

#ifdef CONFIG_EARLYSUSPEND

      [PM_SUSPEND_ON]        =“on”,

#endif

      [PM_SUSPEND_STANDBY]   =“standby”,

      [PM_SUSPEND_MEM] =“mem”,

};

相应的宏定义在:include/linux/suspend.h中:

1

2

3

4

5

6

typedefint __bitwise suspend_state_t;

 

#define PM_SUSPEND_ON          ((__force suspend_state_t) 0)

#define PM_SUSPEND_STANDBY     ((__force suspend_state_t) 1)

#define PM_SUSPEND_MEM         ((__force suspend_state_t) 3)

#define PM_SUSPEND_MAX         ((__force suspend_state_t) 4)

非常奇怪的是,第四种状态(disk)没有详细的定义,而是硬编码在代码中,不明确为什么会这样做,至少我如今看的版本号是这样(2.6.35),这样的就是所谓的suspend to disk或者叫hibernate。只是这不是重点,再说,眼下也非常少有Android的设备支持hibernate。

顾名思义:

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

PM_SUSPEND_STANDBY— 设备处于省电状态,但还能够接收某些事件,详细的行为取决与详细的设备;

PM_SUSPEND_MEM— suspend to memory,设备进入睡眠状态,但全部的数据还保存在内存中,仅仅有某些外部中断才干够唤醒设备;

眼下,大多数的Android设备都仅仅支持当中的两种:PM_SUSPEND_ON 和 PM_SUSPEND_MEM,所以以下的讨论说道suspend的地方,均是指PM_SUSPEND_MEM。

2. Early Suspend、Late Resume

Early Suspend和Late Resume是Android在标准Linux的基础上添加的一项特性。当用户空间的向内核请求进入suspend时,这时候会先进入early suspend状态,驱动程序能够注冊early suspend的回调函数,当进入该状态时,内核会逐一地调用这些回调函数。比如显示屏的驱动程序一般会注冊early suspend,在他的回调函数中,驱动程序会把屏幕和背光都关闭。在这样的状态下,全部的后台进程都还在活动中,该播放歌曲的播放歌曲,该下载数据的依旧在下载,仅仅是显示屏不良而已。进入early suspend状态以后,一旦全部的电源锁(wake lock)被释放,系统立即会进入真正的suspend流程,直到最后系统停止工作,等待外部事件的唤醒。

serial.read函数_过程调用属于面向对象吗

3. Android
的电源锁机制:wake lock

Android相比标准的Linux内核,在电源管理中增加了wake lock机制。一旦申请了某种类型的锁,电源管理模块将会“锁住”某一种电源状态,眼下,Android提供了两种类型的锁:

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

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

wake lock也能够设定超时,时间一到,自己主动释放该锁。

有关wake lock的代码在:kernel/power/wakelock.c中。

4. 电源状态迁移

内核启动完毕以后,电源管理系统会在sysfs文件系统中建立3个文件:

1

2

3

    /sys/power/state

    /sys/power/wake_lock

    /sys/power/wake_unlock

电源状态的迁移首先由用户空间的应用程序发起,当系统应用检測到一定时间内没实用户活动后(比如触摸屏、按键),能够向/sys/power/state文件写入对应的电源状态名称(请參考第一节内容),假设写入“mem”,将会触发内核启动suspend的流程,内核将会依照图2.1进行状态的迁移。应用程序也能够通过/sys/power/wake_lock申请一个WAKE_LOCK_SUSPEND 类型的锁,对应地,通过/sys/power/wake_unlock则能够释放一个锁。内核在进入suspend之前假设检測到某个锁没有释放,则会放弃本次的suspend过程,直到这个锁释放为止。

 

 

基于Android的Linux内核的电源管理:Early Suspend

1. 用户空间的接口
在kernel/power/main.c中,定义了一组sysfs的属性文件,当中一个定义是:
power_attr(state);
把这个宏展开后:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

staticstruct kobj_attribute state_attr={
\

 

        .attr={
                               \

 

                 .name=“state”,   \

 

                 .mode=0644,                          \

 

        },                                           \

 

        .show     =state_show,                          \

 

        .store      =state_store,                 \

 

}

我们再看看main.c的入口:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

staticint __init pm_init(void)

 

{

 

    ……

 

        power_kobj=kobject_create_and_add(“power”, NULL);

 

        if(!power_kobj)

 

                 returnENOMEM;

 

        return sysfs_create_group(power_kobj,&attr_group);

 

}

显然,该函数运行后,会在生成/sys/power文件夹,该文件夹下会建立一系列属性文件,当中一个就是/sys/power/state文件。用户空间向该文件的写入将会导致state_store被调用,读取该文件将会导致state_show函数被调用。

如今回到Android的HAL层中,查看一下代码:hardware/libhardware_legacy/power/power.c:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

 //定义写入/sys/power/state的命令字符串

 

staticconst char*off_state=“mem”;

 

staticconst char*on_state=“on”;

 

//打开/sys/power/state等属性文件,保存对应的文件描写叙述符

 

staticint

 

open_file_descriptors(constchar*const paths[])

 

{

 

    int i;

 

    for(i=0; i

 

终于,用户空间的电源管理系统会调用set_screen_state函数来触发suspend的流程,该函数实际上就是往/sys/power/state文件写入“mem”“on”命令字符串。

 

<pre lang=“c” line=“1”>int

 

set_screen_state(inton)

 

{

 

    ……

 

    initialize_fds();

 

    ……

 

    char buf[32];

 

    int len;

 

    if(on)

 

        len = snprintf(buf,sizeof(buf),“%s”, on_state);

 

    else

 

        len = snprintf(buf,sizeof(buf),“%s”, off_state);

 

    buf[sizeof(buf)1]=\0;

 

    len = write(g_fds[REQUEST_STATE], buf,len);

 

    ……

 

    return0;

 

}

2. 内核中数据结构和接口

与earlysuspend相关的数据结构和接口都在earlysuspend.h中进行了定义。

– early_suspend结构

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

struct early_suspend{

 

#ifdef CONFIG_HAS_EARLYSUSPEND

 

         structlist_head link;

 

         int level;

 

         void(*suspend)(struct early_suspend *h);

 

         void(*resume)(struct early_suspend *h);

 

#endif

 

};

希望运行early suspend的设备,他的设备驱动程序须要向电源管理系统注冊,该结构体用于向电源管理系统注冊earlysuspend/lateresume,当电源管理系统启动suspend流程时,回调函数suspend会被调用,相反,resume的最后阶段,回调函数resume会被调用,level字段用于调整该结构体在注冊链表中的位置,suspend时,level的数值越小,回调函数的被调用的时间越早,resume时则反过来。Android预先定义了3个level等级:

1

2

3

4

5

enum{

      EARLY_SUSPEND_LEVEL_BLANK_SCREEN=50,

      EARLY_SUSPEND_LEVEL_STOP_DRAWING=100,

      EARLY_SUSPEND_LEVEL_DISABLE_FB=150,

};

假设你想你的设备在FB设备被禁止之前运行他的early suspend回调,设备驱动程序应该把level值设定为小于150的某个数值,然后向系统注冊early_suspend结构。注冊和反注冊函数是:

1

2

     void register_early_suspend(struct early_suspend *handler);

     void unregister_early_suspend(struct early_suspend *handler);

early_suspend_handlers链表

全部注冊到系统中的early_suspend结构都会按level值按顺序增加到全局链表early_suspend_handlers中。

3. 工作流程

首先,我们从kernel/power/wakelock.c中的初始化函数開始:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

staticint __init wakelocks_init(void)

{

      int ret;

      int i;

    ……

      for(i=0; i&lt; ARRAY_SIZE(active_wake_locks); i++)

            INIT_LIST_HEAD(&amp;active_wake_locks[i]);

    ……

      wake_lock_init(&amp;main_wake_lock, WAKE_LOCK_SUSPEND,“main”);

      wake_lock(&amp;main_wake_lock);

      wake_lock_init(&amp;unknown_wakeup, WAKE_LOCK_SUSPEND,“unknown_wakeups”);

    ……

      ret= platform_device_register(&amp;power_device);

      ret = platform_driver_register(&amp;power_driver);

    ……

      suspend_work_queue= create_singlethread_workqueue(“suspend”);

    ……

      return0;

}

能够看到,显示初始化active_wake_locks链表数组,然后初始化而且锁住main_wake_lock,注冊平台设备power_device,这些数组、锁和power_device我们在兴许文章再讨论,这里我们关注的最后一个动作:创建了一个工作队列线程suspend_work_queue,该工作队列是earlysuspend的核心所在。

系统启动完毕后,相关的驱动程序通过register_early_suspend()函数注冊了early suspend特性,等待一段时间后,假设没实用户活动(比如按键、触控等操作),用户空间的电源管理服务终于会调用第一节提到的set_screen_state()函数,透过sysfs,进而会调用到内核中的state_store():

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

static ssize_t state_store(struct kobject*kobj,struct kobj_attribute*attr,

                     constchar*buf, size_t n)

{

#ifdef CONFIG_SUSPEND

#ifdef CONFIG_EARLYSUSPEND

      suspend_state_t state = PM_SUSPEND_ON;

#else

      suspend_state_t state = PM_SUSPEND_STANDBY;

#endif

      constchar*const*s;

#endif

      char*p;

      int len;

      int error=EINVAL;

 

      p = memchr(buf,\n, n);

      len = p ? p buf : n;

 

      /* First, check if we are requested to hibernate */

      if(len==4&amp;&amp;!strncmp(buf,“disk”, len)){

            error = hibernate();

  goto Exit;

      }

 

#ifdef CONFIG_SUSPEND

      for(s=&amp;pm_states[state]; state &lt; PM_SUSPEND_MAX; s++, state++){

            if(*s&amp;&amp; len== strlen(*s)&amp;&amp;!strncmp(buf,*s, len))

                  break;

      }

      if(state&lt; PM_SUSPEND_MAX&amp;&amp;*s)

#ifdef CONFIG_EARLYSUSPEND

            if(state== PM_SUSPEND_ON|| valid_state(state)){

                  error =0;

                  request_suspend_state(state);

            }

#else

            error = enter_state(state);

#endif

#endif

 

 Exit:

      return error? error : n;

}

看到了没,前一篇文章说过,suspend to disk做了特殊处理,这里直接比較传入的字符串,而不是使用兴许的pm_states数组,这里我不关心suspend to disk,所以略过hibernate的分析。

紧接着,通过pm_states数组,依据命令字符串查询得到请求的状态,默认情况下,Android的内核都会配置了CONFIG_EARLYSUSPEND,所以会调用request_suspend_state()函数,只是在调用该函数之前会先valid_state()一下,这给了平台相关的代码一个机会确认该平台是否支持所请求的电源状态。valid_state()的详细实现请參考内核代码树。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

void request_suspend_state(suspend_state_t new_state)

{

      unsignedlong irqflags;

      int old_sleep;

 

      spin_lock_irqsave(&amp;state_lock, irqflags);

      old_sleep = state &amp; SUSPEND_REQUESTED;

    ……

      if(!old_sleep&amp;&amp; new_state!= PM_SUSPEND_ON){

            state |= SUSPEND_REQUESTED;

            if(queue_work(suspend_work_queue,&amp;early_suspend_work))

                  pr_info(“early_suspend_work is in queue already\n);

      }elseif(old_sleep&amp;&amp; new_state== PM_SUSPEND_ON){

            state &amp;= ~SUSPEND_REQUESTED;

            wake_lock(&amp;main_wake_lock);

            if(!queue_work(suspend_work_queue,&amp;late_resume_work))

                  pr_info(“late_resume_work is in queue already\n);

      }

      requested_suspend_state = new_state;

      spin_unlock_irqrestore(&amp;state_lock, irqflags);

}

还记得前面初始化时建立的工作队列suspend_woek_queue吗?依据之前的电源状态和请求的状态, request_suspend_state()仅仅是简单地向suspend_work_queue中增加early_suspend_work或者是late_resume_work并调度他们运行。early_suspend_work的工作函数是early_suspend():

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

staticvoid early_suspend(struct work_struct*work)

 

{

 

        struct early_suspend*pos;

 

        unsignedlong irqflags;

 

        int abort=0;

 

        mutex_lock(&amp;early_suspend_lock);

 

        spin_lock_irqsave(&amp;state_lock,irqflags);

 

        if(state== SUSPEND_REQUESTED)

 

                 state |= SUSPENDED;

 

        else

 

                 abort =1;

 

        spin_unlock_irqrestore(&amp;state_lock,irqflags);

 

        if(abort){

 

        ……

 

        }

 

    ……

 

        list_for_each_entry(pos,&amp;early_suspend_handlers, link){

 

                 if(pos-&gt;suspend!= NULL){

 

                           if(debug_mask&amp;DEBUG_SUSPEND)

 

                                    printk(KERN_DEBUG“pos-&gt;suspend: %pF begin\n, pos-&gt;suspend);

 

                           pos-&gt;suspend(pos);

 

                           if(debug_mask&amp;DEBUG_SUSPEND)

 

                                    printk(KERN_DEBUG“pos-&gt;suspend: %pF finish\n, pos-&gt;suspend);

 

                 }

 

        }

 

        mutex_unlock(&amp;early_suspend_lock);

 

        if(debug_mask&amp; DEBUG_SUSPEND)

 

                 pr_info(“early_suspend:sync\n);

 

        sys_sync();

 

abort:

 

        spin_lock_irqsave(&amp;state_lock,irqflags);

 

        if(state==SUSPEND_REQUESTED_AND_SUSPENDED)

 

                 wake_unlock(&amp;main_wake_lock);

 

        spin_unlock_irqrestore(&amp;state_lock,irqflags);

 

}

最终看到啦,early_suspend()遍历early_suspend_handlers链表,从中取出各个驱动程序注冊的early_suspend结构,然后调用它的suspend回调函数。最后,释放main_wake_lock锁,至此整个earlysuspend的流程完毕。以下的序列图清晰地表明了整个调用的过程:

可是,这时整个系统仅仅是处于所谓的idle状态,cpu还在工作,后台进程也在工作中,那什么时候系统会真正地进入睡眠状态?注意到最后一句关键的调用了没有:
wake_unlock(&main_wake_lock);

serial.read函数_过程调用属于面向对象吗

转载于:https://www.cnblogs.com/hrhguanli/p/4516950.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/192067.html原文链接:https://javaforall.net

(0)
上一篇 2026年1月19日 下午9:43
下一篇 2026年1月19日 下午10:22


相关推荐

  • .Net Core HttpClient处理响应压缩「建议收藏」

    .Net Core HttpClient处理响应压缩「建议收藏」前言在上篇文章[ASP.NETCore中的响应压缩]中我们谈到了在ASP.NETCore服务端处理关于响应压缩的请求,服务端的主要工作就是根据Content-Encoding头信息判断采用哪种方式压缩并返回。之前在群里有人问道过,现在的网络带宽这么高了还有必要在服务端针对请求进行压缩吗?确实,如今分布式和负载均衡技术这么成熟,很多需要处理高并发大数据的场景都可以通过增加服务器节点来进行。但是,在资源受限的情况下,或者是还没必要为了某一个点去增加新的服务器节点的时候,我们还是要采用一些程序本身的常规处理

    2022年7月15日
    17
  • 如何用DeepSeek生成小红书文案?DeepSeek真的可以0基础30秒打造小红书爆款吗?

    如何用DeepSeek生成小红书文案?DeepSeek真的可以0基础30秒打造小红书爆款吗?

    2026年3月16日
    1
  • java helloworld源代码_java Hello world 源码执行流程详解

    java helloworld源代码_java Hello world 源码执行流程详解helloworld作为我们学习的第一个个程序,看起来很简单,但是要理解其执行的具体流程还是需要很深的功底,包括对组成原理,操作系统的理解,今天将javahelloworld进行整理一下吧!!废话不多说,先上helloworldpublicclassMain{publicstaticvoidmain(String[]args){Strings=”helloWorl…

    2022年5月28日
    62
  • 最新小龙虾OpenClaw一键安装教程,新手10分钟搞定(实测可用)

    最新小龙虾OpenClaw一键安装教程,新手10分钟搞定(实测可用)

    2026年3月13日
    1
  • 软件测试工程师认证

    软件测试工程师认证一般认证考试都要求拥有比较扎实的理论功底和丰富实际经验 我总结了一下目前软件测试工程可以参加的认证考试 其如下 1 四级软件测试工程师 它是全国计算机等级考试中四级的一类 在校大学生可以参加这个考试 在求职测试职位时可以加分不少 2 软件评测师 它属于全国计算机技术与软件专业技术资格考试 软考 中的一个中级考试 考试不设学历与资历条件 也不论年龄和专业 在职人员可以参加这个考试 可以检验一

    2026年3月18日
    2
  • 浅谈安防监控中视频图像处理技术

    浅谈安防监控中视频图像处理技术2019-08-1521:39:20随着计算机软件、硬件技术的日新月异的发展和普及,人类已经进入一个高速发展的信息化时代,人类大概有80%的信息来自图像,科学研究、技术应用中图像处理技术越来越成为不可缺少的手段。安防行业已经进入一个崭新的时代,高清、智能与大数据和云技术的广泛结合应用,成为时代新的话题。随着而来的是,视频图像处理技术也逐渐成熟起来,只有这样才能切合行业技术发展,为此,对于视频…

    2022年6月28日
    27

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号