本文转载自:http://www.wjdiankong.cn/android%E7%B3%BB%E7%BB%9F%E7%AF%87%E4%B9%8B-%E7%BC%96%E5%86%99%E7%B3%BB%E7%BB%9F%E6%9C%8D%E5%8A%A1%E5%B9%B6%E4%B8%94%E5%B0%86%E5%85%B6%E7%BC%96%E8%AF%91%E5%88%B0%E7%B3%BB%E7%BB%9F%E6%BA%90%E7%A0%81/

在之前已经介绍了一篇关于 如何编写简单的驱动以及访问该驱动的小程序,最后将程序编译到Android内核源码中通过程序访问驱动验证是可以通过的,那么本文就继续这个知识点,把这个驱动程序通过JNI连接创建一个系统服务,提供给上层应用访问改服务功能,可以看到前一篇介绍驱动程序的功能是属于内核层的,而本文介绍的内容是Framework层的知识。

声明:本文内容参考罗升阳的书籍:《Android系统源代码情景分析》 如果想了解更详细的内容非常建议购买此书

非常感谢罗神的这本书,给我带来很多未知的知识,大神的博客地址:http://blog.csdn.net/luoshengyang

一、编写JNI层服务代码

第一步:创建JNI目录

进入到系统的JNI目录中:frameworks/base/services/jni 在这个目录中包含了系统服务的所有JNI实现的程序:

第二步:编写JNI代码

实现代码也比较简单,直接访问之前编译好的驱动即可,然后在提供给外部一个读写的方法,最后在进行JNI方法的手动注册:

第三步:修改编译脚本

在同一目录中有一个Android.mk文件,需要添加我们的这个服务,后续要将这个服务编译到源码中:

第四步:添加服务JNI功能的加载配置

上面已经编写好了我们的系统服务功能,也手动注册了一些读写方法,那么还需要把这个注册功能添加到系统的onload.cpp文件中被调用,不然系统编译之后也是找不到那个JNI方法的,而这个onload.cpp程序,是系统启动的时候去运行,内部是专门注册系统服务的JNI方法的:

第五步:编译framework层源码

上面几步已经完成了代码编写和脚本配置,下面就可以直接编译源码,把这个服务的JNI功能编译到源码中,可以直接使用mmm命令进行单独模块的编译:

mmm frameworks/base/services/jni
make snod

这样编译之后的system.img中就包含了我们定义的系统JNI服务实现逻辑,接下来我们就可以编写Java代码来访问这个JNI暴露的读写方法了。

二、编写Java层服务代码

第一步:创建服务的AIDL文件

进入系统服务的AIDL文件目录:frameworks/base/core/java/android/os 这里存放了系统所有服务的AIDL定义

关于AIDL内容,也是很简单的,提供读写方法:

第二步:编译AIDL文件

我们在使用AIDL的时候都知道,定义完文件之后,必须编译一下,生成对应的Java代码,这样后续才能使用类似于XXX.Stub类,才能在远端服务实现具体功能。这里也是一样的,所以我们得先编译这个aidl文件:

mmm frameworks/base

单独编译framework模块代码,这样就会产生对应的IFregService.Stub类了。

第三步:实现具体服务功能

上面已经编译aidl文件,生成了对应的java代码,下面如果想实现具体的功能,就必须继承Stub类,这个文件存放的目录为:frameworks/base/services/java/com/android/server 系统中所有具体服务实现都是在这个目录中:

代码实现也比较简单:

在这里会定义读写的native方法,和之前的JNI层实现的方法对应:

第四步:将服务添加到ServiceManager中

为了让上次访问到这个服务,和系统其它服务一样,咋们必须得把服务注册到ServiceManager中,而这个注册功能是在SystemServer类中,因为在系统启动的时候,Zygote进程产生的第一个进程就是system_server,在这个进程中做了系统服务的注册工作,我们在服务的同一目录中找到SystemServer.java类,在ServerThread::run函数中注册:

这样注册之后,上层应用就可以通过ServiceManager直接获取到这个服务了,就可以直接访问具体功能了。

第六步:编译Framework源码

上面都已经完成了Java层服务代码的实现了,而到这里,我们似乎已经看到了熟悉的代码了,比如服务的AIDL定义,实现和注册服务功能,和之前介绍的 Binder机制以及远程服务调用机制的知识 越来越接近了,下面在最后一步,编译源码:

mmm frameworks/base/services/java
make snod

编译之后得到system.img文件就包含了我们在Framework定义的FregService服务了,而这个服务的名称是freg,下面继续介绍如何编写一个程序来访问这个服务功能。

三、编写系统应用访问服务功能

前面已经介绍了编写Framework中的服务功能,在JNI层实现了访问驱动的native代码,然后实现了Java层代码调用这些native方法实现驱动的读写功能,并且定义了一个系统服务,包装这些功能,最后把这个服务注册到系统中,这里我们就来编写一个简单的Android系统程序,来访问这个服务功能:

第一步:创建程序项目

创建系统程序项目都是在这个目录中:packages/experimental,我们定义一个Freg项目:

这个项目结构和正常的Android程序结构一样,没什么好说的,因为这里不是依赖于IDE编译,所以咋们还得编写编译脚本Android.mk文件:

关于Android程序代码也比较简单,直接通过ServiceManager来访问这个服务即可:

第二步:编译应用程序

上面的程序创建完成之后,接下来再次编译源码,把这个应用打包到系统中:

mmm packages/experimental/Freg
make snod

编译完成之后,生成的system.img文件就包含了这个系统应用程序

第三步:启动模拟器验证程序

接下来咋们就可以启动模拟器,来运行这个小程序了:

emulator -kernel kernel/common/arch/arm/boot/zImage &

我们找到这个程序之后,打开的效果:

打开这个程序之后,我们可以进行读写操作了:

四、流程总结

到这里我们就成功了完成了手动编写一个简单的系统服务并且添加到系统中,结合之前的一篇文章内容,下面就来总结整个过程,先来一张图压压惊:

有了这张图咋们再来总结一下:

1、编写驱动
1>在 kernel/driver 目录下创建freg驱动程序
2>make menuconfig 编译驱动程序到内核源码中

2、编写Framework层服务的JNI程序
1>在 frameworks/base/services/jni 目录下创建了服务的jni代码
2>在onload.cpp中添加服务jni方法的注册逻辑
3>mmm frameworks/base/services/jni 编译jni程序到系统源码中

3、编写Framework层服务的Java程序
1>在 frameworks/base/core/java/android/os 目录下创建了服务的AIDL文件
2>mmm frameworks/base 编译AIDL文件,生成对应的Java代码
3>在 frameworks/base/services/java/com/android/server 目录中编写具体服务的实现功能
4>最后在同一目录下找到SystemServer.java文件中注册该服务(ServiceManager.addService方法)
5>mmm frameworks/base/services/java 编译java层服务

4、编写系统Android应用程序
1>在 packages/experimental 目录下创建Android项目,在程序中直接使用ServiceManager来得到远端服务即可
2>mmm packages/experimental/Freg 编译程序到系统中
3>启动模拟器验证结果 :emulator -kernel kernel/common/arch/arm/boot/zImage &

说到底,其实这个实验有的同学可能非常感兴趣,可能想立马就实验一把,但是这里现在最大的问题就是你得必须先编译过Android源码,这个是最核心的也是最基本的,然后在按照这些流程走的话就很简单了,所以作为一个Android开发者毕生还是要编译一次Android源码的。所以感兴趣的同学应该赶快动起手来编译Android源码。

项目地址下载:http://download.csdn.net/detail/jiangwei0910410003/9642835

五、总结

到这里就结束了如何手动编写系统服务并且添加到系统中的工作了,同时也暂时结束了这段时间介绍的Android系统篇的系列知识,其实我本来只是想介绍如何Hook掉系统的AMS服务拦截应用启动的知识,可是谁都想不到引出了这么一大串的知识出来,没办法我也只能慢慢的一篇一篇介绍了。

Android系统篇之—-编写系统服务并且将其编译到系统源码中【转】的更多相关文章

  1. Android系统篇之—-编写简单的驱动程序并且将其编译到内核源码中【转】

    本文转载自:大神 通过之前的一篇文章,我们了解了 Android中的Binder机制和远程服务调用 在这篇文章中主要介绍了Android中的应用在调用一些系统服务的时候的原理,那么接下来就继续来介绍一 ...

  2. 访何红辉:谈谈Android源码中的设计模式

    最近Android 6.0版本的源代码开放下载,刚好分析Android源码的技术书籍<Android源码设计模式解析与实战>上市,我们邀请到它的作者何红辉,来谈谈Android源码中的设计 ...

  3. 使用Xamarin开发手机聊天程序 -- 基础篇(大量图文讲解 step by step,附源码下载)

    如果是.NET开发人员,想学习手机应用开发(Android和iOS),Xamarin 无疑是最好的选择,编写一次,即可发布到Android和iOS平台,真是利器中的利器啊!而且,Xamarin已经被微 ...

  4. 使用Xamarin开发即时通信系统 -- 基础篇(大量图文讲解 step by step,附源码下载)...

    如果是.NET开发人员,想学习手机应用开发(Android和iOS),Xamarin 无疑是最好的选择,编写一次,即可发布到Android和iOS平台,真是利器中的利器啊!而且,Xamarin已经被微 ...

  5. Android 源码中的设计模式

    最近看了一些android的源码,发现设计模式无处不在啊!感觉有点乱,于是决定要把设计模式好好梳理一下,于是有了这篇文章. 面向对象的六大原则 单一职责原则 所谓职责是指类变化的原因.如果一个类有多于 ...

  6. Android 网络框架之Retrofit2使用详解及从源码中解析原理

    就目前来说Retrofit2使用的已相当的广泛,那么我们先来了解下两个问题: 1 . 什么是Retrofit? Retrofit是针对于Android/Java的.基于okHttp的.一种轻量级且安全 ...

  7. Eclipse与Android源码中ProGuard工具的使用

    由于工作需要,这两天和同事在研究android下面的ProGuard工具的使用,通过查看android官网对该工具的介绍以及网络上其它相关资料,再加上自己的亲手实践,算是有了一个基本了解.下面将自己的 ...

  8. 关于android源码中的APP编译时引用隐藏的API出现的问题

    今天在编译android源码中的计算器APP时发现,竟然无法使用系统隐藏的API,比如android.os.ServiceManager中的API,引用这个类时提示错误,记忆中在android源码中的 ...

  9. wemall app商城源码中android按钮的三种响应事件

    wemall-mobile是基于WeMall的android app商城,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可定制修改.本文分享wemall app商城源码中android按 ...

随机推荐

  1. js从$scope外部调用$scope内部函数,跨js调用非全局函数

    scope内部函数定义 //定位 $scope.LocateByPoint = function (x,y) { if(!x || !y) { window.alert("GPS坐标不存在, ...

  2. php rsa加密解密实例 及签名验证-自己实践

      <?php /** * Created by PhpStorm. * User: Administrator * Date: 2018/4/1 * Time: 1:50 */ //注意格式一 ...

  3. 修改Linux字符集

    1.查看操作系统中安装的locale信息 ls /usr/lib/locale 2.创建.i18n文件 在用户目录下创建.i18n文件,并添加如下内容: LANG="zh_CN.utf8&q ...

  4. ObjC消息机制

    深入浅出ObjC之消息    罗朝辉(http://blog.csdn.net/kesalin) 在入门级别的ObjC 教程中,我们常对从C++或Java 或其他面向对象语言转过来的程序员说,ObjC ...

  5. JavaWeb知识点总结一

    JavaWeb知识点总结一 常见的状态码以及其含义 一些常见HTTP状态码为: -- 服务器成功返回网页 -- 服务器不理解请求的语法 -- 请求的网页不存在 -- 服务不可用 常见HTTP状态码大全 ...

  6. TSharding源码阅读

    需要的背景知识:Spring 和Mybatis 实现原理和源码, javaassist字节码增强的使用, java及设计模式的使用 1 读取解析数据库配置文件 DataSourceScanner实现了 ...

  7. 转载 --iOS QQ第三方登实现

    我们经常会见到应用登陆的时候会有QQ,微信,微博等的第三方登陆 如图: 下面我们主要讲一下qq的第三方登陆如何实现 首先,到官网注册: http://wiki.connect.qq.com 一,下载S ...

  8. 也谈在 .NET 平台上使用 Scala 语言(续)

    而我是在 Ubuntu 操作系统中使用 Scala.NET 的,应该没有这个问题. 那么,就让我们来測试一下吧. 如今,我们添加一个 DotNet.cs 文件,例如以下所看到的: 1 2 3 4 5 ...

  9. Python PhatomJS 和Selenium动态加载页面 获取图片内容

    如果您觉得感兴趣的话,可以添加我的微信公众号:一步一步学Python![](http://images2017.cnblogs.com/blog/993869/201711/993869-201711 ...

  10. 使用google的GSON解析json格式的数据

    GSON是谷歌提供的开源库,用来解析Json格式的数据,非常好用.如果要使用GSON的话,则要先下载gson-2.2.4.jar这个文件,如果是在Android项目中使用,则在Android项目的li ...