近学习了如何在Android 6.0上添加一个系统服务,APP如何通过新增的系统服务访问底层驱动。
在这学习过程中,收获颇多,并结合学习了《Embeded Android》--Karim Yaghmour 一书中的
Appendix B. Adding Support For New Hardware章节,受益匪浅,讲述了如何添加一个完整的系统
服务(app->framework->kernel)。尽管描述的非常详细,但是基于Android 2.3.7描述的。现在把
书中的opersys例子移植到Android 6.0上,虽然说不上复杂,但由于版本差异,难免会出现许多奇奇
怪怪的问题,甚至版本差异造成了bug出现。特以此移植记录分享学习过程。

    主要围绕以下几个步骤添加一个完整的系统服务:
(A) 添加circular-char驱动
(B) 添加opersyshw_qemu HAL
(C) 添加com_android_server_opersys_OpersysService JNI
(D) 添加IOpersysService接口
(E) 添加OpersysService
(F) 添加OpersysManager
(G) 添加系统服务
(H) 注册服务
(I) 更新API
(J) 设置权限
(K) 测试服务
(L) 添加测试APP

(A) 添加circular-char驱动
   

    circular-char是一个简单的字符设备驱动,其实现的功能就是一个简单的FIFO,APP可以通过
read、write来进行读写操作实验,即写数据到FIFO,可以从FIFO读出写入的数据。

kernel/drivers/circular-driver/circular-char.c

1 #include <linux/module.h>
  2 #include <linux/miscdevice.h>
  3 #include <linux/fs.h>
  4 #include <asm/uaccess.h>
  5 
  6 #define BUF_SIZE 200
  7 
  8 static char buf[BUF_SIZE];
  9 static char *read_ptr;
 10 static char *write_ptr;
 11 
 12 static int device_open(struct inode *inode, struct file *file)
 13 {
 14   printk("device_open called \n");
 15 
 16   return 0;
 17 }
 18 
 19 static int device_release(struct inode *inode, struct file *file)
 20 {
 21   printk("device_release called \n");
 22 
 23   return 0;
 24 }
 25 
 26 static ssize_t device_read(struct file *filp,   /* see include/linux/fs.h   */
 27                char *buffer,    /* buffer to fill with data */
 28                size_t length,   /* length of the buffer     */
 29                loff_t * offset)
 30 {
 31   int chars_read = 0;
 32 
 33   printk("device_read called \n");
 34 
 35   while(length && *read_ptr && (read_ptr != write_ptr)) {
 36     put_user(*(read_ptr++), buffer++);
 37 
 38     printk("Reading %c \n", *read_ptr);
 39 
 40     if(read_ptr >= buf + BUF_SIZE)
 41       read_ptr = buf;
 42 
 43     chars_read++;
 44     length--;
 45   }
 46 
 47   return chars_read;
 48 }
 49 
 50 static ssize_t
 51 device_write(struct file *filp, const char *buff, size_t len, loff_t * off)
 52 {
 53   int i;
 54 
 55   printk("device_write called \n");
 56 
 57   for(i = 0; i < len; i++) {
 58     get_user(*write_ptr, buff++);
 59     printk("Writing %c \n", *write_ptr);
 60     write_ptr++;
 61     if (write_ptr >= buf + BUF_SIZE)
 62       write_ptr = buf;
 63   }
 64 
 65   return len;
 66 }
 67 
 68 static struct file_operations fops = {
 69   .open = device_open,
 70   .release = device_release,
 71   .read = device_read,
 72   .write = device_write,
 73 };
 74 
 75 static struct miscdevice circ_char_misc = {
 76   .minor = MISC_DYNAMIC_MINOR,
 77   .name = "circchar",
 78   .fops = &fops,
 79 };
 80 
 81 int circ_char_enter(void)
 82 {
 83   int retval;
 84 
 85   retval = misc_register(&circ_char_misc);
 86   printk("CIRC Driver got retval %d\n", retval);
 87   printk("mmap is %08X\n", (int) fops.mmap);
 88 
 89   read_ptr = buf;
 90   write_ptr = buf;
 91 
 92   return 0;
 93 }
 94 
 95 void circ_char_exit(void)
 96 {
 97   misc_deregister(&circ_char_misc);
 98 }
 99 
100 module_init(circ_char_enter);
101 module_exit(circ_char_exit);

 

kernel/drivers/circular-driver/Kconfig

1 menuconfig DRIVER_FOR_TEST
 2     bool "Drivers for test"
 3     help
 4       Drivers for test.
 5       If unsure, say no.
 6 
 7 if DRIVER_FOR_TEST
 8 
 9 config CIRCULAR_CHAR
10     tristate "circular-char"
11     help
12       circular-char driver.
13 
14 endif

 

kernel/drivers/circular-driver/Makefile

1 obj-$(CONFIG_CIRCULAR_CHAR)     += circular-char.o

 

kernel/drivers/Kconfig

1 ......
2 source "drivers/circular-driver/Kconfig"
3 ......

 

kernel/drivers/Makefile

1 ......
2 obj-$(CONFIG_DRIVER_FOR_TEST) += circular-driver/
3 ......

 

kernel/arch/arm/configs/xxx_defconfig

......
CONFIG_DRIVER_FOR_TEST=y
CONFIG_CIRCULAR_CHAR=y
......

 

    驱动已添加到内核,编译烧录到目标板看是否加载成功:

# ls dev/circchar
    ls dev/circchar
    dev/circchar
    #echo hello > dev/circchar
    echo hello > dev/circchar
    #cat dev/circchar
    dev/circchar
    hello

   

    如果执行以上命令,输出对应得信息,则说明驱动加载成功。

 

 

(B) 添加opersyshw_qemu HAL
   

    这里添加一个opersys的HAL层,使应用和驱动分离,hal主要向应用提供open、read、write等几个
接口。

 

hardware/libhardware/tests/opersyshw/opersyshw_qemu.c

1 #define  LOG_TAG  "opersyshw_qemu"                                                                                 
  2 #include <cutils/log.h>
  3 #include <cutils/sockets.h>                                                                                        
  4 #include <sys/types.h>                                                                                             
  5 #include <sys/stat.h>                                                                                              
  6 #include <fcntl.h>
  7 #include <hardware/opersyshw.h>                                                                                    
  8 #include <malloc.h>                                                                                                
  9 
 10 #define   OPERSYSHW_DEBUG   1                                                                                      
 11 
 12 #if OPERSYSHW_DEBUG
 13 #  define D(...)   ALOGD(__VA_ARGS__)                                                                              
 14 #else
 15 #  define D(...)   ((void)0)                                                                                       
 16 #endif                                                                                                             
 17 
 18 static int fd = 0;                                                                                                 
 19 
 20 static int opersyshw__read(char* buffer, int length)                                                               
 21 {   
 22     int retval;                                                                                                    
 23     
 24     D("OPERSYS HW - read()for %d bytes called", length);                                                           
 25     
 26     retval = read(fd, buffer, length);    
 27     D("read data from driver: %s", buffer);
 28 
 29     return retval;
 30 }
 31    
 32 static int opersyshw__write(char* buffer, int length)
 33 {
 34     int retval;
 35 
 36     D("OPERSYS HW - write()for %d bytes called", length);
 37 
 38     retval = write(fd, buffer, length);
 39     D("write data to driver: %s", buffer);
 40 
 41     return retval;
 42 }
 43 
 44 static int opersyshw__close(void)
 45 {
 46     if (fd != -1) {
 47         if (!close(fd)) {
 48             return 0;
 49         }
 50     }
 51 
 52     return -1;
 53 }
 54 
 55 static int opersyshw__test(int value)
 56 {
 57     return value;
 58 }
 59 
 60 static int open_opersyshw(const struct hw_module_t* module, char const* name,
 61         struct hw_device_t** device)
 62 {
 63     struct opersyshw_device_t *dev = malloc(sizeof(struct opersyshw_device_t));
 64     if (!dev) {
 65         D("OPERSYS HW failed to malloc memory !!!");
 66         return -1;
 67     }
 68 
 69     memset(dev, 0, sizeof(*dev));
 70 
 71     dev->common.tag = HARDWARE_DEVICE_TAG;
 72     dev->common.version = 0;
 73     dev->common.module = (struct hw_module_t*)module;
 74     dev->read = opersyshw__read;
 75     dev->write = opersyshw__write;
 76     dev->close = opersyshw__close;
 77     dev->test = opersyshw__test;
 78 
 79     *device = (struct hw_device_t*) dev;
 80 
 81     fd = open("/dev/circchar", O_RDWR);
 82     if (fd < 0) {
 83         D("failed to open /dev/circchar!");
 84         return 0;
 85     }
 86 
 87     D("OPERSYS HW has been initialized");
 88 
 89     return 0;
 90 }
 91 
 92 static struct hw_module_methods_t opersyshw_module_methods = {
 93     .open = open_opersyshw,
 94 };
 95 
 96 struct hw_module_t HAL_MODULE_INFO_SYM = {
 97     .tag = HARDWARE_MODULE_TAG,
 98     .version_major = 1,
 99     .version_minor = 0,
100     .id = OPERSYSHW_HARDWARE_MODULE_ID,
101     .name = "Opersys HW Module",
102     .author = "Opersys inc.",
103     .methods = &opersyshw_module_methods,
104 };

 

hardware/libhardware/include/hardware/opersyshw.h

1 #ifndef ANDROID_OPERSYSHW_INTERFACE_H
 2 #define ANDROID_OPERSYSHW_INTERFACE_H
 3 
 4 #include <stdint.h>
 5 #include <sys/cdefs.h>
 6 #include <sys/types.h>
 7 
 8 #include <hardware/hardware.h>
 9     
10 __BEGIN_DECLS
11 
12 #define OPERSYSHW_HARDWARE_MODULE_ID "opersyshw"
13 
14 struct opersyshw_device_t {
15     struct hw_device_t common;
16 
17     int (*read)(char* buffer, int length);
18     int (*write)(char* buffer, int length);
19     int (*close)(void);
20     int (*test)(int value);
21 };
22 
23 __END_DECLS
24 
25 #endif // ANDROID_OPERSYSHW_INTERFACE_H

 

hardware/libhardware/tests/opersyshw/Android.mk

1 LOCAL_PATH := $(call my-dir)
 2     
 3 # HAL module implemenation, not prelinked and stored in
 4 # hw/<GPS_HARDWARE_MODULE_ID>.<ro.hardware>.so
 5 include $(CLEAR_VARS)
 6 LOCAL_MODULE_PATH := $(TARGET_OUT_SHARED_LIBRARIES)/hw
 7 LOCAL_CFLAGS += $(common_flags)
 8 LOCAL_LDLIBS += -llog
 9 LOCAL_C_INCLUDES := hardware/libhardware
10 LOCAL_SHARED_LIBRARIES := liblog libcutils libhardware
11 LOCAL_SRC_FILES := opersyshw_qemu.c
12 LOCAL_MODULE := opersyshw.$(TARGET_BOARD_PLATFORM)
13 LOCAL_MODULE_TAGS := optional
14 include $(BUILD_SHARED_LIBRARY)

 

    编译之后看看是否错误,是否生成.so文件,在源码根目录下:

# find ./out/ -name 'opersyshw.*.so'
    ......
    ./out/target/product/<project>/system/lib/hw/opersyshw.sc8830.so
    ......

 

    注意Android.mk中的$(TARGET_BOARD_PLATFORM),这里是sc8830,不同的平台会有差异

 

(C) 添加com_android_server_opersys_OpersysService JNI
   

    JNI接口主要是为了Java(app)调用C/C++。

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

1 #define LOG_TAG "OpersysServiceJNI"
  2 
  3 #include "jni.h"
  4 #include "JNIHelp.h"
  5 #include "android_runtime/AndroidRuntime.h"
  6 
  7 #include <utils/misc.h>
  8 #include <utils/Log.h>
  9 #include <hardware/hardware.h>
 10 #include <hardware/opersyshw.h>
 11 
 12 #include <stdio.h>
 13 
 14 namespace android
 15 {
 16 
 17 opersyshw_device_t* opersyshw_dev;
 18 
 19 static jint init_native(JNIEnv *env, jobject /* clazz */)
 20 {
 21     int err;
 22     hw_module_t* module;
 23     opersyshw_device_t* dev = NULL;
 24 
 25     //ALOGI("init_native()"); 
 26 
 27     err = hw_get_module(OPERSYSHW_HARDWARE_MODULE_ID, (hw_module_t const**)&module);
 28     if (err == 0) {
 29         if (module->methods->open(module, "", ((hw_device_t**) &dev)) != 0) {
 30             ALOGE("Can't open opersys module!!!");
 31             return 0;
 32         }
 33     } else {
 34         ALOGE("Can't get opersys module!!!");
 35         return 0;
 36     }
 37 
 38     return (jint)dev;
 39 }
 40 
 41 static void finalize_native(JNIEnv *env, jobject /* clazz */, int ptr)
 42 {
 43     opersyshw_device_t* dev = (opersyshw_device_t*)ptr;
 44 
 45     //ALOGI("finalize_native()");
 46 
 47     if (dev == NULL) {
 48         return;
 49     }
 50 
 51     dev->close();
 52 
 53     free(dev);
 54 }
 55 
 56 static int read_native(JNIEnv *env, jobject /* clazz */, int ptr, jbyteArray buffer)
 57 {
 58     opersyshw_device_t* dev = (opersyshw_device_t*)ptr;
 59     jbyte* real_byte_array;
 60     int length;
 61 
 62     //ALOGI("read_native()");
 63 
 64     real_byte_array = env->GetByteArrayElements(buffer, NULL);
 65 
 66     if (dev == NULL) {
 67         return 0;
 68     }
 69 
 70     length = dev->read((char*) real_byte_array, env->GetArrayLength(buffer));
 71 
 72     ALOGI("read data from hal: %s", (char *)real_byte_array);
 73 
 74     env->ReleaseByteArrayElements(buffer, real_byte_array, 0);
 75 
 76     return length;
 77 }
 78 
 79 static int write_native(JNIEnv *env, jobject /* clazz */, int ptr, jbyteArray buffer)
 80 {
 81     opersyshw_device_t* dev = (opersyshw_device_t*)ptr;
 82     jbyte* real_byte_array;
 83     int length;
 84 
 85     //ALOGI("write_native()");
 86 
 87     real_byte_array = env->GetByteArrayElements(buffer, NULL);
 88 
 89     if (dev == NULL) {
 90         return 0;
 91     }
 92 
 93     length = dev->write((char*) real_byte_array, env->GetArrayLength(buffer));
 94 
 95     ALOGI("write data to hal: %s", (char *)real_byte_array);
 96 
 97     env->ReleaseByteArrayElements(buffer, real_byte_array, 0);
 98 
 99     return length;
100 }
101 
102 
103 static int test_native(JNIEnv *env, jobject /* clazz */, int ptr, int value)
104 {
105     opersyshw_device_t* dev = (opersyshw_device_t*)ptr;
106 
107     if (dev == NULL) {
108         return 0;
109     }
110 
111     ALOGI("test_native()");
112 
113     return dev->test(value);
114 }
115 
116 static JNINativeMethod method_table[] = {
117     { "init_native", "()I", (void*)init_native },
118     { "finalize_native", "(I)V", (void*)finalize_native },
119     { "read_native", "(I[B)I", (void*)read_native },
120     { "write_native", "(I[B)I", (void*)write_native },
121     { "test_native", "(II)I", (void*)test_native}
122 };
123 
124 int register_android_server_opersys_OpersysService(JNIEnv *env)
125 {
126     return jniRegisterNativeMethods(env, "com/android/server/opersys/OpersysService",
127             method_table, NELEM(method_table));
128 
129 };

 

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

1 ......
 2 namespace android {
 3 ......
 4 int register_android_server_opersys_OpersysService(JNIEnv* env);
 5 ......
 6 };
 7 
 8 ....
 9 
10 extern "C" jint JNI_OnLoad(JavaVM* vm, void* /* reserved */)
11 {
12 ......
13 register_android_server_opersys_OpersysService(env);
14 ......
15 }

 

frameworks/base/services/core/jni/Android.mk

1 ......
2 LOCAL_SRC_FILES += \
3 ......
4 $(LOCAL_REL_DIR)/com_android_server_opersys_OpersysService.cpp \
5 ......

 

 

(D) 添加IOpersysService接口
   

    IOpersysService主要用于实现一个进程间通信的接口,其内部机制就是通过Binder实现进程间通信的,
即客户端与服务端(OpersysService)分别处于不同的进程中,客户端和服务端之间不能够直接相互访问,
之间必须通过Binder传递。

 

frameworks/base/core/java/android/opersys/IOpersysService.aidl

1 interface IOpersysService {
2 /**
3 * {@hide}
4 */
5 String read(int maxLength);
6 int write(String mString);
7 }

 

frameworks/base/Android.mk

1 ......
2 LOCAL_SRC_FILES += \
3 ......
4 core/java/android/opersys/IOpersysService.aidl \
5 ......

 

    其中,aidl文件主要用于生成同名的.java文件IOpersysService.java,IOpersysService.java主要实现
了一些Binder相关的设置和相关接口。
    编译后,会在out目录下生成:

out/target/common/obj/JAVA_LIBRARIES/framework_intermediates/src/core/java/android/opersys/IOpersysService.java

 


(E) 添加OpersysService
   

     OpersysService主要充当一个服务端(server),直接调用native如:

private static native int init_native();
private static native void finalize_native(int ptr);
private static native int read_native(int ptr, byte[] buffer);
private static native int write_native(int ptr, byte[] buffer);
private static native int test_native(int ptr, int value);

    这些方法对应的是frameworks/base/services/core/jni/com_android_server_opersys_OpersysService.cpp
对应的同名函数,视觉上就像Java直接调用了C/C++一样。

 

frameworks/base/services/core/java/com/android/server/opersys/OpersysService.java

1 package com.android.server.opersys;
 2 
 3 import android.content.Context;
 4 import android.os.Handler;
 5 import android.opersys.IOpersysService;
 6 import android.os.Looper;
 7 import android.os.Message;
 8 import android.os.Process;
 9 import android.util.Slog;
10 import android.os.RemoteException;
11 
12 public class OpersysService extends IOpersysService.Stub {
13     private static final String TAG = "OpersysService";
14     private Context mContext;
15     private int mNativePointer;
16 
17     public OpersysService(Context context) {
18         super();
19         mContext = context;
20         Slog.i(TAG, "Opersys Service started");
21 
22         mNativePointer = init_native();
23 
24         Slog.i(TAG, "test() returns " + test_native(mNativePointer, 20));
25     }
26 
27     protected void finalize() throws Throwable {
28         finalize_native(mNativePointer);
29         super.finalize();
30     }    
31     
32     public String read(int maxLength) throws RemoteException
33     {
34         int length;
35         byte[] buffer = new byte[maxLength];
36 
37         length = read_native(mNativePointer, buffer);
38 
39         try {
40             return new String(buffer, 0, length, "UTF-8");
41         } catch (Exception e) {
42             Slog.e(TAG, "read buffer error!");
43             return null;
44         }
45     }
46 
47     public int write(String mString) throws RemoteException
48     {
49         byte[] buffer = mString.getBytes();
50 
51         return write_native(mNativePointer, buffer);
52     }
53 
54     private static native int init_native();
55     private static native void finalize_native(int ptr);
56     private static native int read_native(int ptr, byte[] buffer);
57     private static native int write_native(int ptr, byte[] buffer);
58     private static native int test_native(int ptr, int value);
59 }

 

 

(F) 添加OpersysManager
   

    OpersysManager主要用于管理OpersysService,实例化了IOpersysService,在注册服务的
时候就是实例化了一个OpersysManager,APP(客户端)获取服务getService时也是获得这个对象,通过这个对象,APP就

可以调用该服务的相关接口(API)了
frameworks/base/core/java/android/opersys/OpersysManager.java

1 package android.opersys;
 2 
 3 import android.content.Context;
 4 import android.os.RemoteException;
 5 import android.opersys.IOpersysService;
 6 import android.util.Slog;
 7 
 8 public class OpersysManager
 9 {
10     private static final String TAG = "OpersysManager";
11 
12     public String read(int maxLength) {
13         try {
14             return mService.read(maxLength);
15         } catch (RemoteException e) {
16             Slog.e(TAG, "read error!");
17             return null;
18         }
19     }
20 
21     public int write(String mString) {
22         try {
23             return mService.write(mString);
24         } catch (RemoteException e) {
25             Slog.e(TAG, "write error!");
26             return 0;
27         }    
28     }
29 
30     public OpersysManager(Context context, IOpersysService service) {
31         mService = service;
32     }
33 
34     IOpersysService mService;
35 }

 

 

(G) 添加系统服务addService
   

    实现了OpersysService的各种接口之后,需要向SystemServer添加服务

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

1 ......
 2 import com.android.server.opersys.OpersysService;
 3     ......
 4 private void startOtherServices() {
 5     ......
 6     OpersysService opersys = null;
 7     ......
 8     if (mFactoryTestMode != FactoryTest.FACTORY_TEST_LOW_LEVEL) { 
 9         ......
10         try {
11             Slog.i(TAG, "Opersys Service");
12             opersys = new OpersysService(context);                                                             
13             Slog.i(TAG, "Add Opersys Service");
14             ServiceManager.addService(Context.OPERSYS_SERVICE, opersys);                                       
15             Slog.i(TAG, "Opersys Service Succeed!");                                                           
16         } catch (Throwable e) {
17             Slog.e(TAG, "Failure starting OpersysService Service", e);                                         
18         }
19         ......
20     }
21 }

 

frameworks/base/core/java/android/content/Context.java

1 ......
 2     @StringDef({
 3         ......
 4         OPERSYS_SERVICE,
 5         ......
 6     })
 7     ......
 8     /**
 9      * Use with {@link #getSystemService} to retrieve a
10      * {@link android.opersys.OpersysManager} for using Opersys Service.
11      *
12      * @see #getSystemService
13      */
14     public static final String OPERSYS_SERVICE = "opersys";
15     ......

 

 

(H) 注册服务

 

frameworks/base/core/java/android/app/SystemServiceRegistry.java

1 ......
 2 import android.opersys.OpersysManager;
 3 import android.opersys.IOpersysService;
 4 ......
 5 final class SystemServiceRegistry {
 6     ......
 7     static {
 8         ......
 9         registerService(Context.OPERSYS_SERVICE, OpersysManager.class,
10                 new CachedServiceFetcher<OpersysManager>() {
11             @Override
12             public OpersysManager createService(ContextImpl ctx) {
13                 IBinder b = ServiceManager.getService(Context.OPERSYS_SERVICE);
14                 IOpersysService service = IOpersysService.Stub.asInterface(b);
15                 if (service == null) {
16                     return null;
17                 }
18                 return new OpersysManager(ctx, service);
19             }});
20         ......
21     }
22     ......
23 }
24 ......

 

    到这里,主要工作都要完成了,貌似可以直接可以编译进行测试了。但是实际上还有些东西需要
设置,如编译设置,SeLinux安全设置等。

(I) 更新API
   

    在前面添加和注册了服务之后,需要更新一下API才能正常编译。

frameworks/data-binding/compiler/src/main/resources/api-versions.xml

1 ......
2 <field name="OPERSYS_SERVICE" />
3 ......

 

    在源码目录下:

# make update-api -j16

 

    编译通过后,会自动更新以下两个文件
frameworks/base/api/current.txt

......
field public static final java.lang.String OPERSYS_SERVICE = "opersys";
......
package android.opersys {

  public abstract interface IOpersysService implements android.os.IInterface {
    method public abstract int write(java.lang.String) throws android.os.RemoteException;
  }

  public static abstract class IOpersysService.Stub extends android.os.Binder implements android.opersys.IOpersysService {
    ctor public IOpersysService.Stub();
    method public android.os.IBinder asBinder();
    method public static android.opersys.IOpersysService asInterface(android.os.IBinder);
    method public boolean onTransact(int, android.os.Parcel, android.os.Parcel, int) throws android.os.RemoteException;
  }

  public class OpersysManager {
    ctor public OpersysManager(android.content.Context, android.opersys.IOpersysService);
    method public java.lang.String read(int);
    method public int write(java.lang.String);
  }

}
......

 

    frameworks/base/api/system-current.txt更新同上

 

(J) 设置权限

 

external/sepolicy/service.te

1 ......
2 type opersys_service, app_api_service, system_server_service, service_manager_type;
3 ......

 

external/sepolicy/service_contexts

1 ......
2 opersys                                   u:object_r:opersys_service:s0
3 ......

 

device/<manufacturer>/<chip>/common/sepolicy/device.te

1 ......
2 type circchar_device, dev_type;
3 ......

 

device/<manufacturer>/<chip>/common/sepolicy/file_contexts

1 ......
2 /dev/circchar        u:object_r:circchar_device:s0
3 ......

 

device/<manufacturer>/<chip>/common/sepolicy/system_server.te  

1 ......
2 allow system_server circchar_device:chr_file { open read write ioctl };
3 ......

 

system/core/rootdir/ueventd.rc

1 ......
2 /dev/circchar         0660   system     system
3 ......

 

 

(K) 测试服务

   

    完成以上步骤后,全编译一次,烧录后,可先通过命令测试一下服务是否正常启动:

# service check opersys
    service check opersys
    Service opersys:found
    # service list 
    service list
    ......
    24      opersys: [android.opersys.IOpersysService]
    ......
    # service call opersys 2 s16 "Hello, World!"
    service call opersys 2 s16 "Hello, World!"
    Result: Parcel(00000000 0000000d    '........')
    # service call opersys 1 i32 20
    service call opersys 1 i32 20
    Result: Parcel(
    0x00000000: 00000000 0000000d 00650048 006c006c '........H.e.l.l.'
    0x00000010: 002c006f 00570020 0072006f 0064006c 'o.,. .W.o.r.l.d.'
    0x00000020: 00000021                            '!...            ')

    执行以上命令得到对应的结果后,说明服务启动正常。接下来就可以在APP上使用了。


(L) 添加测试APP
   

    这个APP的功能是,当用户打开APP时,APP会获得OpersysService服务,然后向这个服务发送
“Hello Opersys”,然后从这个服务读回。

packages/apps/HelloOpersysInternal/src/com/opersys/hellointernal/HelloOpersysInternalActivity.java

1 package com.opersys.hellointernal;                                                                                 
 2                                                                                                                    
 3 import android.app.Activity;                                                                                       
 4 import android.os.Bundle;                                                                                          
 5 import android.util.Log;                                                                                           
 6 import android.os.ServiceManager;   // Will only work in AOSP                                                      
 7 //import android.opersys.IOpersysService;  // Interface "hidden" in SDK                                            
 8 import android.opersys.OpersysManager;                                                                             
 9 import android.content.Context;                                                                                    
10                                                                                                                    
11 public class HelloOpersysInternalActivity extends Activity {                                                       
12     private static final String DTAG = "HelloOpersysInternal";                                                     
13                                                                                                                    
14     /** Called when the activity is first created. */                                                              
15     @Override                                                                                                      
16     public void onCreate(Bundle savedInstanceState) {                                                              
17         super.onCreate(savedInstanceState);                                                                        
18         setContentView(R.layout.main);                                                                             
19                                                                                                                    
20         //IOpersysService om = IOpersysService.Stub.asInterface(ServiceManager.getService("opersys"));             
21                                                                                                                    
22         OpersysManager om = (OpersysManager)getSystemService(Context.OPERSYS_SERVICE);                             
23         try {                                                                                                      
24             Log.d(DTAG, "Going to write to the \"opersys\" service");                                              
25             om.write("Hello Opersys");                                                                             
26             Log.d(DTAG, "Service returned: " + om.read(20));                                                       
27         }                                                                                                          
28         catch (Exception e) {                                                                                      
29             Log.d(DTAG, "FAILED to call service");                                                                 
30             e.printStackTrace();
31         }
32     }
33 }

 

packages/apps/HelloOpersysInternal/res/layout/main.xml

1 <?xml version="1.0" encoding="utf-8"?>
 2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 3     android:layout_width="fill_parent"
 4     android:layout_height="fill_parent"
 5     android:orientation="vertical" >
 6 
 7     <TextView
 8         android:layout_width="fill_parent"
 9         android:layout_height="wrap_content"
10         android:text="@string/hello" />
11 
12 </LinearLayout>

 

packages/apps/HelloOpersysInternal/res/values/strings.xml

1 <?xml version="1.0" encoding="utf-8"?>
2 <resources>
3 
4     <string name="hello">Hello World, HelloOpersysInternalActivity!</string>
5     <string name="app_name">HelloOpersysInternal</string>
6 
7 </resources>

 

packages/apps/HelloOpersysInternal/AndroidManifest.xml

1 <?xml version="1.0" encoding="utf-8"?>
 2 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
 3     package="com.opersys.hellointernal"
 4     android:versionCode="1"
 5     android:versionName="1.0" >
 6 
 7     <uses-sdk android:minSdkVersion="10" />
 8 
 9     <application
10         android:icon="@drawable/ic_launcher"
11         android:label="@string/app_name" >
12         <activity
13             android:label="@string/app_name"
14             android:name=".HelloOpersysInternalActivity" >
15             <intent-filter >
16                 <action android:name="android.intent.action.MAIN" />
17 
18                 <category android:name="android.intent.category.LAUNCHER" />
19             </intent-filter>
20         </activity>
21     </application>
22 
23 </manifest>

 

packages/apps/HelloOpersysInternal/Android.mk

1 LOCAL_PATH:= $(call my-dir)
 2 include $(CLEAR_VARS)
 3 
 4 LOCAL_MODULE_TAGS := optional
 5     
 6 LOCAL_SRC_FILES := $(call all-java-files-under, src)
 7 
 8 LOCAL_PACKAGE_NAME := HelloOpersysInternal
 9 
10 include $(BUILD_PACKAGE)

 

vendor/sprd/open-source/common_packages.mk

1 PRODUCT_PACKAGES += \
2 ......
3 HelloOpersysInternal
4 ......

 

    到这里,已经完成了所有步骤,全编译烧录之后,打开HelloOpersysInternalAPP,再打开
logcat,可以看到以下信息:

01-01 08:07:56.137  3729  3729 D HelloOpersysInternal: Going to write to the "opersys" service

01-01 08:07:56.140  1272  3082 D opersyshw_qemu: OPERSYS HW - write()for 13 bytes called

01-01 08:07:56.140  1272  3082 D opersyshw_qemu: write data to driver: Hello Opersys

01-01 08:07:56.140  1272  3082 I OpersysServiceJNI: write data to hal: Hello Opersys

01-01 08:07:56.142  1272  3032 D opersyshw_qemu: OPERSYS HW - read()for 20 bytes called

01-01 08:07:56.142  1272  3032 D opersyshw_qemu: read data from driver: Hello Opersys

01-01 08:07:56.142  1272  3032 I OpersysServiceJNI: read data from hal: Hello Opersys

01-01 08:07:56.143  3729  3729 D HelloOpersysInternal: Service returned: Hello Opersys