(此方法只适用于没有做odex优化的版本.比如eng,user需要禁止odex优化)
目前公司framework层的工作一直比较麻烦,具体做法是在linux下直接用SourceInsight编辑,然后mm编译framework模块.编辑也没个智能提示,语法检查之类的,最主要framework这么庞大的东西,编起来也麻烦.处理一个问题,往往要多次mm.浪费时间,效率低,试想在竞争如此激烈的今天,效率低下是不能容忍的.
先来看张图吧,了解一下android project的编译过程
图里没什么重点,最重要的就是java class file 会通过dx 转换为android 自己设计的dex文件. 这个是快速编译和调试的基础.
编译过程总是先得到class文件,然后转换成dex,最后再做签名对齐等操作.可以看到class才是实质,其它的都是点缀而已,如果我们修改了*.java代码,最先变动的是class文件.
打个比方,我改动了PackageManagerService.java.那么首先是根据我修改的java代码生成新的PackageManagerService.class,然后才是转换为dex格式的services.jar包.
如果我们要修改service.jar里的PackageManagerService,只要得到PackageManagerService.java修改后对应的class文件即可.是不是很简单.有了这个class文件,
之后的转dex格式,签名什么的就易如反掌了.搞过java的都知道,class是java代码的目标文件,里面存放的是java byte code.直接在虚拟机里的可执行代码.
好吧,到这里,可以新建一个java项目,目的是为了得到修改framework的class文件.现在是在eclipse下开发,所以能够用上ide的特性了.
首先会发现很多符号找不到的错误,具体一点就是没有代码里需要的架包.
不妨再看看我们普通的一个android项目吧,里面都有一个android.jar的架包,打开这个架包发现里面都是class文件(见下图).这货很明显就是一个java library jar.
我们apk代码里很多的import都是来自这里的.有了架包,那些找不到符号的错误都可以烟消云散了
说道架包,android源码编译的时候会在out/target/common/obj/JAVA_LIBRARIES目录下生成很多jar包.这里有我们需要的架包文件(文件夹就是架包名,架包都是class-full-debug.jar),用到了哪个就拷贝哪个.
这里我列一下必备的几个:
android-common_intermediates
android.policy_intermediates
core_intermediates
framework_intermediates
services_intermediates
还有注意加上sdk下面的android.jar
其它的,可以自己添加,需要什么找什么
我这里以packagemanagerservice.java举例:
先看看java项目的基本配置
src之上都是需要的架包,src下的包名与要修改的架包下的同名class路径一致.
我在PackageManagerService.java里添加了一行log代码
public ApplicationInfo getApplicationInfo(String packageName, int flags) {
// writer
synchronized (mPackages) {
Log.v(
TAG, "getApplicationInfo " + packageName
+ ": " + " add by chenxuesong");//加上了本人的大名
PackageParser.Package p = mPackages.get(packageName);
if (DEBUG_PACKAGE_INFO) Log.v(
TAG, "getApplicationInfo " + packageName
+ ": " + p);
if (p != null) {
// Note: isEnabledLP() does not apply here - always return info
return PackageParser.generateApplicationInfo(p, flags);
}
if ("android".equals(packageName)||"system".equals(packageName)) {
return mAndroidApplication;
}
if((flags & PackageManager.GET_UNINSTALLED_PACKAGES) != 0) {
return generateApplicationInfoFromSettingsLPw(packageName, flags);
}
}
return null;
}
PackageManagerService不愧是核心服务,好多class文件
替换class文件,一条命令搞定
该生成我们手机上能执行的文件了,(转化成dex格式)也是一条命令搞定
把这个生成的servicesdex.jar改名为services.jar并push到手机上,然后reboot.抓取日志,发现生效了
这里要提一下,这些架包都是需要签名的,不过可以用我之前的方式和谐掉,替换一下libdvm.so.一劳永逸.
具体可以看一下链接
framework修改无法重启