4.6.1 installd介绍
在前面对PKMS构造函数分析时介绍过一个Installer类型的对象mInstaller,它通过socket和后台服务installd交互,以完成一些重要操作。这里先回顾一下PKMS中mInstaller的调用方法:
~~~
mInstaller = new Installer();//创建一个Installer对象
//对某个APK文件进行dexopt优化
mInstaller.dexopt(paths[i], Process.SYSTEM_UID,true);
//扫描完系统Package后,调用moveFiles函数
mInstaller.moveFiles();
//当存储空间不足时,调用该函数清理存储空间
mInstaller.freeCache(freeStorageSize);
~~~
Installer的种种行为都和其背后的installd有关。下面来分析installd。
1. installd概貌
installd是一个native进程,代码非常简单,其功能就是启动一个socket,然后处理来自Installer的命令,其代码如下:
installd.c
~~~
int main(const int argc, const char *argv[]) {
charbuf[BUFFER_MAX];
structsockaddr addr;
socklen_t alen;
intlsocket, s, count;
//
if (初始化全局变量,如果失败则退出) {
initialize_globals();
initialize_directories();
......
}
......
lsocket= android_get_control_socket(SOCKET_PATH);
listen(lsocket, 5);
fcntl(lsocket, F_SETFD, FD_CLOEXEC);
for (;;){
alen= sizeof(addr);
s =accept(lsocket, &addr, &alen);
fcntl(s, F_SETFD, FD_CLOEXEC);
for(;;) {
unsigned short count;
readx(s, &count, sizeof(count));
//执行installer发出的命令,具体解释见下文
execute(s, buf);
}
close(s);
}
return0;
}
~~~
installd支持的命令及参数信息都保存在数据结构cmds中,代码如下:
**installd.c**
~~~
struct cmdinfo cmds[] = {//第二个变量是参数个数,第三个参数是命令响应函数
{"ping", 0,do_ping },
{"install", 3,do_install },
{"dexopt", 3,do_dexopt },
{"movedex", 2,do_move_dex },
{"rmdex", 1,do_rm_dex },
{"remove", 2,do_remove },
{"rename", 2, do_rename },
{"freecache", 1,do_free_cache },
{"rmcache", 1,do_rm_cache },
{"protect", 2,do_protect },
{"getsize", 4,do_get_size },
{"rmuserdata", 2,do_rm_user_data },
{"movefiles", 0,do_movefiles },
{"linklib", 2,do_linklib },
{"unlinklib", 1,do_unlinklib },
{"mkuserdata", 3,do_mk_user_data },
{"rmuser", 1,do_rm_user },
};
~~~
下面来分析相关的几个命令。
2. dexOpt命令分析
PKMS在需要对一个APK或jar包做dex优化时,会发送dexopt命令给installd,相应的处理函数为do_dexopt,代码如下:
**installd.c**
~~~
static int do_dexopt(char **arg, charreply[REPLY_MAX])
{
returndexopt(arg[0], atoi(arg[1]), atoi(arg[2]));
}
~~~
**commands.c**
~~~
int dexopt(const char *apk_path, uid_t uid, intis_public)
{
structutimbuf ut;
structstat apk_stat, dex_stat;
chardex_path[PKG_PATH_MAX];
chardexopt_flags[PROPERTY_VALUE_MAX];
char*end;
int res,zip_fd=-1, odex_fd=-1;
......
//取出系统级的dexopt_flags参数
property_get("dalvik.vm.dexopt-flags", dexopt_flags,"");
strcpy(dex_path, apk_path);
end =strrchr(dex_path, '.');
if (end!= NULL) {
strcpy(end, ".odex");
if(stat(dex_path, &dex_stat) == 0) {
return 0;
}
}
//得到一个字符串,用于描述dex文件名,位于/data/dalvik-cache/下
if(create_cache_path(dex_path, apk_path)) {
return -1;
}
memset(&apk_stat, 0, sizeof(apk_stat));
stat(apk_path, &apk_stat);
zip_fd =open(apk_path, O_RDONLY, 0);
......
unlink(dex_path);
odex_fd= open(dex_path, O_RDWR | O_CREAT | O_EXCL, 0644);
......
pid_tpid;
pid =fork();
if (pid== 0) {
......//uid设置
//创建一个新进程,然后对exec dexopt进程进行dex优化
run_dexopt(zip_fd,odex_fd, apk_path, dexopt_flags);
exit(67);
} else {
//installd将等待dexopt完成优化工作
res= wait_dexopt(pid, apk_path);
......
}
......//资源清理
return-1;
}
~~~
让人大跌眼镜的是,dex优化工作竟然由installd委派给dexopt进程来实现。dex优化后会生成一个dex文件,一般位于/data/dalvik-cache/目录中。这里给出一个示例,如图4-12所示。
:-: ![](http://img.blog.csdn.net/20150803111157859?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center)
图4-12 dex文件示例
>[info] **提示** :dexopt进程由android源码/dalvik/dexopt/OptMain.cpp定义。感兴趣的读者可深入研究dex优化的工作原理。
3. movefiles命令分析
PKMS扫描完系统Package后,将发送该命令给installd,相应处理函数的代码如下:
**installd.c**
~~~
static int do_movefiles(char **arg, charreply[REPLY_MAX])
{
returnmovefiles();
}
[-->commands.c]
int movefiles()
{
DIR *d;
int dfd,subfd;
structdirent *de;
structstat s;
charbuf[PKG_PATH_MAX+1];
intbufp, bufe, bufi, readlen;
charsrcpkg[PKG_NAME_MAX];
chardstpkg[PKG_NAME_MAX];
charsrcpath[PKG_PATH_MAX];
chardstpath[PKG_PATH_MAX];
intdstuid=-1, dstgid=-1;
inthasspace;
//打开/system/etc/updatecmds/目录
d =opendir(UPDATE_COMMANDS_DIR_PREFIX);
if (d ==NULL) {
gotodone;
}
dfd =dirfd(d);
while((de = readdir(d))) {
......//解析该目录下的文件,然后执行对应操作
}
closedir(d);
done:
return0;
}
~~~
先来看/system/etc/updatecmds/目录下到底是什么文件,这里给出一个示例,如图4-13所示。
:-: ![](http://img.blog.csdn.net/20150803111217969?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center)
图4-13 movefiles示例
以图4-13中最后两行为例,movefiles将把com.google.android.gsf下的databases目录转移到com.andorid.providers.im下。从文件中的注释可知,movefiles的功能和系统升级有关。
4. doFreeCache
第3章介绍了DeviceStorageMonitorService,当系统空间不够时,DSMS会调用PKMS的freeStorageAndNotify函数进行空间清理。该工作真正的实施者是installd,相应的处理命令为do_free_cache,其代码如下:
**installd.c**
~~~
static int do_free_cache(char **arg, charreply[REPLY_MAX])
{
returnfree_cache((int64_t)atoll(arg[0]));
}
~~~
**commands.c**
~~~
int free_cache(int64_t free_size)
{
constchar *name;
int dfd,subfd;
DIR *d;
structdirent *de;
int64_tavail;
avail =disk_free();//获取当前系统的剩余空间大小
if(avail < 0) return -1;
if(avail >= free_size) return 0;
d =opendir(android_data_dir.path);//打开/data/目录
dfd =dirfd(d);
while((de = readdir(d))) {
if (de->d_type != DT_DIR) continue;
name= de->d_name;
......//略过.和..文件
subfd = openat(dfd, name, O_RDONLY | O_DIRECTORY);
//删除/data及各级子目录中的cache文件夹
delete_dir_contents_fd(subfd, "cache");
close(subfd);
......//如果剩余空间恢复正常,则返回
}
closedir(d);
return-1;//清理空间后,仍然不满足要求
}
~~~
installd的介绍就到此为止,这部分内容比较简单,读者完全可自行深入研究。
- 前言
- 第1章 搭建Android源码工作环境
- 1.1 Android系统架构
- 1.2 搭建开发环境
- 1.2.1 下载源码
- 1.2.2 编译源码
- 1.2.3 利用Eclipse调试system_process
- 1.3 本章小结
- 第2章 深入理解Java Binder和MessageQueue
- 2.1 概述
- 2.2 Java层中的Binder架构分析
- 2.2.1 Binder架构总览
- 2.2.2 初始化Java层Binder框架
- 2.2.3 addService实例分析
- 2.2.4 Java层Binder架构总结
- 2.3 心系两界的MessageQueue
- 2.3.1 MessageQueue的创建
- 2.3.2 提取消息
- 2.3.3 nativePollOnce函数分析
- 2.3.4 MessageQueue总结
- 2.4 本章小结
- 第3章 深入理解SystemServer
- 3.1 概述
- 3.2 SystemServer分析
- 3.2.1 main函数分析
- 3.2.2 Service群英会
- 3.3 EntropyService分析
- 3.4 DropBoxManagerService分析
- 3.4.1 DBMS构造函数分析
- 3.4.2 dropbox日志文件的添加
- 3.4.3 DBMS和settings数据库
- 3.5 DiskStatsService和DeviceStorageMonitorService分析
- 3.5.1 DiskStatsService分析
- 3.5.2 DeviceStorageManagerService分析
- 3.6 SamplingProfilerService分析
- 3.6.1 SamplingProfilerService构造函数分析
- 3.6.2 SamplingProfilerIntegration分析
- 3.7 ClipboardService分析
- 3.7.1 复制数据到剪贴板
- 3.7.2 从剪切板粘贴数据
- 3.7.3 CBS中的权限管理
- 3.8 本章小结
- 第4章 深入理解PackageManagerService
- 4.1 概述
- 4.2 初识PackageManagerService
- 4.3 PKMS的main函数分析
- 4.3.1 构造函数分析之前期准备工作
- 4.3.2 构造函数分析之扫描Package
- 4.3.3 构造函数分析之扫尾工作
- 4.3.4 PKMS构造函数总结
- 4.4 APK Installation分析
- 4.4.1 adb install分析
- 4.4.2 pm分析
- 4.4.3 installPackageWithVerification函数分析
- 4.4.4 APK 安装流程总结
- 4.4.5 Verification介绍
- 4.5 queryIntentActivities分析
- 4.5.1 Intent及IntentFilter介绍
- 4.5.2 Activity信息的管理
- 4.5.3 Intent 匹配查询分析
- 4.5.4 queryIntentActivities总结
- 4.6 installd及UserManager介绍
- 4.6.1 installd介绍
- 4.6.2 UserManager介绍
- 4.7 本章学习指导
- 4.8 本章小结
- 第5章 深入理解PowerManagerService
- 5.1 概述
- 5.2 初识PowerManagerService
- 5.2.1 PMS构造函数分析
- 5.2.2 init分析
- 5.2.3 systemReady分析
- 5.2.4 BootComplete处理
- 5.2.5 初识PowerManagerService总结
- 5.3 PMS WakeLock分析
- 5.3.1 WakeLock客户端分析
- 5.3.2 PMS acquireWakeLock分析
- 5.3.3 Power类及LightService类介绍
- 5.3.4 WakeLock总结
- 5.4 userActivity及Power按键处理分析
- 5.4.1 userActivity分析
- 5.4.2 Power按键处理分析
- 5.5 BatteryService及BatteryStatsService分析
- 5.5.1 BatteryService分析
- 5.5.2 BatteryStatsService分析
- 5.5.3 BatteryService及BatteryStatsService总结
- 5.6 本章学习指导
- 5.7 本章小结
- 第6章 深入理解ActivityManagerService
- 6.1 概述
- 6.2 初识ActivityManagerService
- 6.2.1 ActivityManagerService的main函数分析
- 6.2.2 AMS的 setSystemProcess分析
- 6.2.3 AMS的 installSystemProviders函数分析
- 6.2.4 AMS的 systemReady分析
- 6.2.5 初识ActivityManagerService总结
- 6.3 startActivity分析
- 6.3.1 从am说起
- 6.3.2 AMS的startActivityAndWait函数分析
- 6.3.3 startActivityLocked分析
- 6.4 Broadcast和BroadcastReceiver分析
- 6.4.1 registerReceiver流程分析
- 6.4.2 sendBroadcast流程分析
- 6.4.3 BROADCAST_INTENT_MSG消息处理函数
- 6.4.4 应用进程处理广播分析
- 6.4.5 广播处理总结
- 6.5 startService之按图索骥
- 6.5.1 Service知识介绍
- 6.5.2 startService流程图
- 6.6 AMS中的进程管理
- 6.6.1 Linux进程管理介绍
- 6.6.2 关于Android中的进程管理的介绍
- 6.6.3 AMS进程管理函数分析
- 6.6.4 AMS进程管理总结
- 6.7 App的 Crash处理
- 6.7.1 应用进程的Crash处理
- 6.7.2 AMS的handleApplicationCrash分析
- 6.7.3 AppDeathRecipient binderDied分析
- 6.7.4 App的Crash处理总结
- 6.8 本章学习指导
- 6.9 本章小结
- 第7章 深入理解ContentProvider
- 7.1 概述
- 7.2 MediaProvider的启动及创建
- 7.2.1 Context的getContentResolver函数分析
- 7.2.2 MediaStore.Image.Media的query函数分析
- 7.2.3 MediaProvider的启动及创建总结
- 7.3 SQLite创建数据库分析
- 7.3.1 SQLite及SQLiteDatabase家族
- 7.3.2 MediaProvider创建数据库分析
- 7.3.3 SQLiteDatabase创建数据库的分析总结
- 7.4 Cursor 的query函数的实现分析
- 7.4.1 提取query关键点
- 7.4.2 MediaProvider 的query分析
- 7.4.3 query关键点分析
- 7.4.4 Cursor query实现分析总结
- 7.5 Cursor close函数实现分析
- 7.5.1 客户端close的分析
- 7.5.2 服务端close的分析
- 7.5.3 finalize函数分析
- 7.5.4 Cursor close函数总结
- 7.6 ContentResolver openAssetFileDescriptor函数分析
- 7.6.1 openAssetFileDescriptor之客户端调用分析
- 7.6.2 ContentProvider的 openTypedAssetFile函数分析
- 7.6.3 跨进程传递文件描述符的探讨
- 7.6.4 openAssetFileDescriptor函数分析总结
- 7.7 本章学习指导
- 7.8 本章小结
- 第8章 深入理解ContentService和AccountManagerService
- 8.1 概述
- 8.2 数据更新通知机制分析
- 8.2.1 初识ContentService
- 8.2.2 ContentResovler 的registerContentObserver分析
- 8.2.3 ContentResolver的 notifyChange分析
- 8.2.4 数据更新通知机制总结和深入探讨
- 8.3 AccountManagerService分析
- 8.3.1 初识AccountManagerService
- 8.3.2 AccountManager addAccount分析
- 8.3.3 AccountManagerService的分析总结
- 8.4 数据同步管理SyncManager分析
- 8.4.1 初识SyncManager
- 8.4.2 ContentResolver 的requestSync分析
- 8.4.3 数据同步管理SyncManager分析总结
- 8.5 本章学习指导
- 8.6 本章小结