6.2、Android硬件访问服务编写系统代码

1、实现接口文件给App使用,接口文件是应用程序查询获得服务时获得

  使用AIDL(Android接口定义语言)来实现ILedService.java接口

 定义ILedService.aidl

 interface ILedService

 {

   int IedCtrl(int which,int status);

 }

 把ILedService.aidl放在已经编译好的Android系统源码中的目录:frameworks/base/core/java/android/os目录下,同时修改frameworks/base下的Android.mk文件,其就是makefile文件,其他子目录没有Android.mk文件:仿照其他添加aidl文件添加一句:

core/java/android/os/ILedService.aidl

接着在frameworks/base目录下执行:mmm . (该命令会帮我们生成ILedService.java文件,同时执行mmm命令的前提是已经成功编译了Android系统)

编译的结果会放到out目录下,进入out目录下搜索:“find -name "ILedService.java”

 .out/target/common/obj/JAVA_LIBRARIES/framework_intermediates/src/core/java/android/os目录下有个ILedService.java文件

App如果使用:ILedService iLedService;

      iLedService = ILedService .Stub.asInterface(ServiceManager.getService("led"));

      然后调用iLedService.IedCtrl();这个方法会把服务请求发给LedService.java

2、实现LedService.java(可以参考VibratorService.java振动器服务)

package com.android.server;

import android.os.ILedService;

public class LedService extends ILedService.stub{

  private static final String TAG = "LedService";

  /*call native c function to access hardware*/

  public int ledCtrl(int which ,int status) throws android.os.RemoteException//父类ILedService.stub继承接口ILedService,所以接口里面的ledCtrl要实现,可以参考生成的lLedService.java

  {

    return native_ledCtrl(which ,status);

  }

  //构造函数

  public LedService(){

    native_ledOpen();

  }

  public static native int native_ledCtrl(int which ,int status) ;//这些本地方法由JNI文件提供

  public static native int native_ledOpen() ;

  public static  native void native_ledClose() ;

3、修改SystemServer.java

  在Slog.i(TAG,"Vibrator Service")下面添加:

  Slog.i(TAG,"Led Service");

  led = new LedService(context);

  ServiceManager.addService("led",led);

4、实现com_android_server_LedService.cpp(JNI文件)(参考com_android_server_VibratorService.cpp),本例中没有实现HAL,其把HAL写到JNI中了,后面会来实现HAL

  里面注册本地方法,供LedService.java使用

  #define LOG_TAG "LedService"

  #include "jni.h"

  #include "JNIHelp.h"

  #include "android_runtime/AndroidRuntime.h"

  #include <utils/misc.h>

  #include <utils/Log.h>

  #include <hardware_legacy/vibrator.h>

  #include <stdio.h>

  //还有一些头问题

  namespace android

  {

  static jint fd;

  jint ledOpen(JNIEnv *env,jobject cls)

  {

    fd = open("/dev/leds",O_RDWR);

    ALOGI(“native ledOpen:%d”,fd);//打印

    if(fd >=0)

      return 0;

    else

      return -1;

  } 

  void ledClose(JNIEnv *env,jobject cls) 

  {

    close(fd);

  }  

  jint ledCtrl(JNIEnv *env,jobject cls,jint which,jint status)

  {

    int ret = ioctl(fd,status,which);

    return ret;

  }

  static const JNINativeMethod methods[] = {

    {"native_ledOpen","()I",(void *)ledOpen},

    {"native_ledClose","()V",(void *)ledClose},

    {"native_ledCtrl","(II)I",(void *)ledCtrl},

  };

  int register_android_server_LedService(JNIEnv *env)

  {

    return jniRegisterNativeMethods(env,"com/android/server/LedService",methods,NELEM(methods));

  }

  }

5、修改Onload.cpp

  添加:register_android_server_LedService(env);并且还要声明,参考register_android_servier_VibratorService(env)

上传各个文件:

A、按照上面添加ILedService.aidl生成ILedService.java

B、

把修改的SystemServer.java和实现的LedService.java上传到:

frameworks/base/services/java/com/android/server/SystemServer.java

frameworks/base/services/core/java/com/android/server/LedServer.java

不需要修改frameworks/base/services/core/Android.mk,其已经通过下面语句全部包含了:

LOCAL_SRC_FILES +=

        $(call all-java_files_under,java)

C、JNI:com_android_server_LedService.cpp和onload.cpp

 目录:frameworks/base/services/core/jni/onload.cpp

    frameworks/base/services/core/jni/com_android_server_LedService.cpp

 修改frameworks/base/services/core/jni/Android.mk,添加:

   $(LOCAL_REL_DIR)/com_android_server_LedService.cpp

编译:

mmm frameworks/base/services//修改的文件都在services下面,并且主要是servicess下的Android.mk把子目录下的Android.mk都一层一层包含了,同时也可以看到在services下的Android.mk里面把jni下的哪些cpp编译成了libandroid_servers,就是SystemServer.java中load的C库android_servers.so

make snod//会生成system.img,但是在Android源码顶层目录下和out/target/product/tiny4412/下都有system.img,而且两个不一样

./gen-img.sh  //执行这个后,上面两个system.img一样了,如果不执行这句必须用out目录下的system.img

设置为SD卡启动,使用miniTools烧写system.img之后,在设置为nand启动

整体程序说明:

A、SystemServer.java中的main函数调用new SystemServer().run()

B、在run()函数中System.loadLibrary("android_servers")//加载C库,这个C库对应的文件就是Onload.cpp和我们实现的一大堆com_android_server_XXX.cpp(JNI文件)

C、Onload.cpp里面有个JNI_Onload函数在库加载的时候被调用,其会调用JNI中提供的register_android_server_xxxService(env)来注册本地方法函数到服务类LedService,在LedService.java中可以看到起声明了和本地方法对应的函数

D、SystemServer.java函数接着调用startOtherServices()函数来调用ServiceManager.addService(名字,服务类LedService对象)在注册,及告诉ServiceManager进程

E、APP应用程序通过getService(名字)来获得Service

原文地址:https://www.cnblogs.com/liusiluandzhangkun/p/9125003.html