Inside Android

Init

Overview

android/system/core/init/readme.txt  -> usage of init process function and services, including start / stop service and setprop …

Init fist process is same as Linux init.

system/core/init/readme.txt
system\core\rootdir\init.rc
system/core/init/keyword.h

Details

对于多个class的启动顺序,for example.
501 on nonencrypted
502     class_start main
503     class_start late_start

515 on charger
516     class_start charger

........

./builtins.c:619:        action_for_each_trigger("nonencrypted", action_add_queue_tail);
./init.c:1145:    action_for_each_trigger("early-init", action_add_queue_tail);
./init.c:1153:    action_for_each_trigger("init", action_add_queue_tail);
./init.c:1172:        action_for_each_trigger("charger", action_add_queue_tail);

........

 

Other's traslation

中文翻译src\system\core\init\readme.txt
#This portion come from Internet
Actions
Actions是一系列命令的命名。Actions拥有一个触发器(trigger)用来决定action何时执行。当一个action在符合触发条件被执行时,如果它还没被加入到待执行队列中的话,则加入到队列最后。

队列中的action依次执行,action中的命令也依次执行。Init在执行命令的中间处理其它活动(设备创建/销毁,property设置,进程重启)。
Actions表现形式为:
on <trigger>
   <command>
   <command>
   <command>
Services
Services是由init启动,在它们退出时重启(可选)。Service表现形式为:
service <name> <pathname> [ <argument> ]*
   <option>
   <option>
   ...
Options
Options是Services的修饰,它们影响init何时、如何运行service.
critical
     这是一个设备关键服务(device-critical service) .如果它在4分钟内退出超过4次,设备将重启并进入恢复模式。
disabled
     这个服务的级别将不会自动启动,它必须被依照服务名指定启动才可以启动。
setenv <name> <value>
     设置已启动的进程的环境变量<name>的值<value>
socket <name> <type> <perm> [ <user> [ <group> ] ]
     创建一个名为/dev/socket/<name>的unix domin socket,并传送它的fd到已启动的进程。<type>必须为"dgram"或"stream".用户和组默认为0.
user <username>
     在执行服务前改变用户名。当前默认为root.如果你的进程需要linux能力,你不能使用这个命令。你必须在还是root时请求能力,并下降到你需要的uid.
group <groupname> [ <groupname> ]*
     在执行服务前改变组。在第一个组后的组将设为进程附加组(通过setgroups()).当前默认为root.
oneshot
     在服务退出后不重启。
class <name>
     为service指定一个类别名。同样类名的所有的服务可以一起启动或停止。如果没有指定类别的服务默认为"default"类。
onrestart
       当服务重启时执行一个命令。
Triggers
     Triggers(触发器)是一个字符串,可以用来匹配某种类型的事件并执行一个action。
boot
     这是当init开始后执行的第一个触发器(当/init.conf被加载)
<name>=<value>
     当property <name>被设为指定的值<value>时触发。
device-added-<path>
device-removed-<path>
     当设备节点被添加或移除时触发。
service-exited-<name>
     当指定的服务存在时触发
Commands
exec <path> [ <argument> ]*
     Fork并执行一个程序(<path>).这将被block直到程序执行完毕。最好避免执行例如内建命令以外的程序,它可能会导致init被阻塞不动。
export <name> <value>
     设定全局环境变量<name>的值<value>,当这个命令执行后所有的进程都可以取得。
ifup <interface>
     使网络接口<interface>联机。
import <filename>
     解析一个init配置文件,扩展当前配置文件。
hostname <name>
     设置主机名
chmod <octal-mode> <path>
     改变文件访问权限
chown <owner> <group> <path>
     改变文件所属和组
class_start <serviceclass>
     当指定类别的服务没有运行,启动该类别所有的服务。
class_stop <serviceclass>
     当指定类别的服务正在运行,停止该类别所有的服务。
domainname <name>
     设置域名。
insmod <path>
     加载该路径<path>的模块
mkdir <path> [mode] [owner] [group]
     在<path>创建一个目录,可选选项:mod,owner,group.如果没有指定,目录以755权限,owner为root,group为root创建.
mount <type> <device> <dir> [ <mountoption> ]*
     尝试mount <device>到目录<dir>. <device>可以用mtd@name格式以命名指定一个mtd块设备。<mountoption>包含"ro","rw","remount","noatime".
setkey
     暂时没有
setprop <name> <value>
     设置系统property <name>的值<value>.
setrlimit <resource> <cur> <max>
     设置resource的rlimit.
start <service>
     启动一个没有运行的服务。
stop <service>
     停止一个正在运行的服务。
symlink <target> <path>
     创建一个<path>的符号链接到<target>
sysclktz <mins_west_of_gmt>
     设置系统时区(GMT为0)
trigger <event>
     触发一个事件。用于调用其它action。
write <path> <string> [ <string> ]*
     打开<path>的文件并写入一个或多个字符串。
Properties
Init会更新一些系统property以提供查看它正在干嘛。
init.action
     当前正在执行的action,如果没有则为""
init.command
     被执行的命令,如果没有则为""
init.svc.<name>
     命名为<name>的服务的状态("stopped", "running", "restarting")

 

LPM(Low power mode)

关机充电

Android original关机充电实现流程:
整体流程:bootloader判断是USB开机还是power key开机,然后通过cmdline传递给kernel,kernel会通过属性系统(设置ro.boot.mode属性)告诉init进程,最终init进程判断bootmode属性来决定启动方式,即调用normal boot 还是 charger mode
用户空间Init进程判断cmdline来设置属性
init会调用process_kernel_cmdline函数来处理cmdline,
其中import_kernel_nv函数对androidboot字符进行判断,
并设置属性ro.boot.mode为charger
import_kernel_nv函数片段如下:
     708 } else if (!strncmp(name, "androidboot.", 12) && name_len > 12) {
     709 const char *boot_prop_name = name + 12;
     710 char prop[PROP_NAME_MAX];
     711 int cnt;
     712
     713 cnt = snprintf(prop, sizeof(prop), "ro.boot.%s", boot_prop_name);
     714 if (cnt < PROP_NAME_MAX)
     715 property_set(prop, value);
     716 }

用户空间Init进程,读取属性从而加载不同启动流程
init进程(android/system/core/init/*.c) main函数中会get属性ro.boot.mode,
如果为charger,那么进入charger队列,而非boot队列
init main函数代码片段:
    1071 if (is_charger) {
    1072 action_for_each_trigger("charger", action_add_queue_tail);
    1073 } else {
    1074 action_for_each_trigger("early-boot", action_add_queue_tail);
    1075 action_for_each_trigger("boot", action_add_queue_tail);
    1076 }
 

 

 

 

input system

./native/include/android/keycodes.h   //define keycode , like HOME is “3” (adb shell input keyevent 3)

Android 用vold取代 udevd,来监听uevent

Android input debug:
toolbox: getevent
1.    adb shell getevent -i //view detailed information of /dev/input
2.    adb shell getevent //fetch all of input events
(sendevent /dev/input/eventX type code value)
sourcecode: ./core/toolbox/getevent.c
input设备节点:/dev/input/event*

键布局映射文件通常放在/system/usr/keylayout和/data/usr/keylayout
对于每一个键盘设备xxx,设置系统属性android.keylayout.xxx
键定义:键定义遵循如下格式key SCANCODE KEYCODE [FLAGS...],当扫描码是一个数字,键码定义在你描述的布局文件android.keylayout.xxx,另外可以设置相关的FLAGS:

键值映射关系文件*.kl

For example:
./system/usr/keylayout/AVRCP.kl
./system/usr/keylayout/Generic.kl
./system/usr/keylayout/Vendor_045e_Product_028e.kl
./system/usr/keylayout/Vendor_046d_Product_c216.kl
./system/usr/keylayout/Vendor_046d_Product_c294.kl
./system/usr/keylayout/Vendor_046d_Product_c299.kl
./system/usr/keylayout/Vendor_046d_Product_c532.kl
./system/usr/keylayout/Vendor_04e8_Product_7021.kl
./system/usr/keylayout/Vendor_054c_Product_0268.kl
./system/usr/keylayout/Vendor_05ac_Product_0239.kl
./system/usr/keylayout/Vendor_22b8_Product_093d.kl
./system/usr/keylayout/gpio-keys.kl
./system/usr/keylayout/qwerty.kl
./system/usr/keylayout/sci-keypad.kl


#Code analysis on Android 4.4
//framework/base/services/input/EventHub.cpp [Native]
EventHub::getEvents -> epoll_wait (不断查询/dev/input/event*事件)
<- InputReader::loopOnce
<- InputManager::initialize
<- NativeInputManager::NativeInputManager

//com_android_server_input_InputManagerService.cpp [JNI]
本文件提供对JAVA的接口,其中nativeInit会创建NativeInputManager对象实例
最终这些JNI会被InputManagerService来使用

//In java
其中InputManagerService最为SystemServer.java的一个服务线程,代码如下
    inputManager = wm.getInputManagerService();
    ServiceManager.addService(Context.INPUT_SERVICE, inputManager);

 

Property System

Android系统的全局变量的离线存储方案,bionic使用共享内存的方式来实现。

In-depth study property system including property server
Android property related source file:
bionic/libc/bionic/system_properties.cpp

on property:gsm.pnx67xx.xufs_started=1  //表示当属性gsm…为1时执行
start pnx67xx
在进程/system/bin/xufs的main方法中,设置了gsm.pnx67xx.xufs_started属性为1
property_set("gsm.pnx67xx.xufs_started", "1");

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值