android6.0 SerialPort 服務


 

 

    一篇博客描述了一個簡單的串口應用程序和驅動程序,了解了應用程序訪問串口的基本操作,如
打開串口,設置串口,寫串口,讀串口,關閉串口等。和Linux串口驅動的基本框架。這里將了解Android
下的串口系統框架,仍然使用上一篇博客中的tiny_serial.c作為本例的驅動,本例實現的功能不變,無
論應用程序寫任何數據到串口,都能從該串口中讀回。
    關於Android系統服務基本框架,可以參考http://www.cnblogs.com/hackfun/p/7612617.html博客,
Android串口(serial port)服務框架與該例子十分類似,這里作簡單描述。當然也有些不同的地方,后面會分析。

(A) 串口服務的基本框架
    1. 注冊驅動
    2. 注冊hal
    3. 注冊JNI
    4. 注冊和添加SerialService
    5. 串口管理SerialManager

(B) 打開串口
(C) 設置權限
(D) 測試串口


(A) 串口服務的基本框架
1. 注冊驅動
    通過加載kernel/driver/tty/serial/tiny_serial.c驅動,生成/dev/ttytiny0節點,應用通過訪問
該節點,實現的對串口的open、read、write、close等操作。

2. 注冊hal
    在Android串口服務中,省略了hal層,即通過JNI直接訪問驅動。

3. 注冊JNI
    通過frameworks/base/services/core/jni/com_android_server_SerialService.cpp和
frameworks/base/core/jni/android_hardware_SerialPort.cpp兩個JNI文件對驅動訪問,
為JAVA提供底層驅動訪問的接口,如:

private native ParcelFileDescriptor native_open(String path);

private native void native_open(FileDescriptor pfd, int speed) throws IOException;
private native void native_close();
private native int native_read_array(byte[] buffer, int length) throws IOException;
private native int native_read_direct(ByteBuffer buffer, int length) throws IOException;
private native void native_write_array(byte[] buffer, int length) throws IOException;
private native void native_write_direct(ByteBuffer buffer, int length) throws IOException;
private native void native_send_break();

 

4. 注冊和添加SerialService
    frameworks/base/services/core/java/com/android/server/SerialService.java的SerialService類
中提供了服務端通過JNI放問驅動的接口,如:

public String[] getSerialPorts()
public ParcelFileDescriptor openSerialPort(String path)

通過向frameworks/base/services//java/com/android/server/SystemServer.java的服務管理器ServiceManager
添加服務:

serial = new SerialService(context);
ServiceManager.addService(Context.SERIAL_SERVICE, serial);

並且,通過frameworks/base/core/java/android/app/SystemServiceRegistry.java

1 registerService(Context.SERIAL_SERVICE, SerialManager.class,
2     new CachedServiceFetcher<SerialManager>() {
3 @Override
4 public SerialManager createService(ContextImpl ctx) {
5     IBinder b = ServiceManager.getService(Context.SERIAL_SERVICE);
6     return new SerialManager(ctx, ISerialManager.Stub.asInterface(b));
7 }});

可以訪問串口服務SerialService,客戶端通過獲得SerialService,就能遠程調用getSerialPorts()和
openSerialPort(String path)接口

 

5. 串口管理SerialManager
    frameworks/base/core/java/android/hardware/SerialManager.java中SerialManager對串口操作進一步管理,
客戶端只要實例化一個SerialManager對象,使用該對象的方法訪問串口。不過,該對象只對
openSerialPort(String name, int speed)管理。

 

(B) 打開串口
    通過以上簡單分析Android串口服務框架之后,這里進一步分析串口的打開open操作流程。
    在http://www.cnblogs.com/hackfun/p/7612617.html
博客中,只有服務端調用JNI訪問底層驅動,而在串口服務中,客戶端也調用JNI直接訪問底層驅動。服務端
只負責對串口的open操作,在frameworks/base/services/core/jni/com_android_server_SerialService.cpp中

 1 static jobject android_server_SerialService_open(JNIEnv *env, jobject /* thiz */, jstring path)
 2 {
 3     const char *pathStr = env->GetStringUTFChars(path, NULL);
 4     //打開/dev/tty*節點
 5     int fd = open(pathStr, O_RDWR | O_NOCTTY);
 6     if (fd < 0) {
 7         ALOGE("could not open %s", pathStr);
 8         env->ReleaseStringUTFChars(path, pathStr);
 9         return NULL;
10     }   
11     env->ReleaseStringUTFChars(path, pathStr);
12 
13     jobject fileDescriptor = jniCreateFileDescriptor(env, fd);
14     if (fileDescriptor == NULL) {
15         return NULL;
16     }
17     //返回文件描述符,用於跨進程訪問文件
18     return env->NewObject(gParcelFileDescriptorOffsets.mClass,
19         gParcelFileDescriptorOffsets.mConstructor, fileDescriptor);
20 }

而read、write、close等相關操作在frameworks/base/core/jni/android_hardware_SerialPort.cpp中,
其中android_hardware_SerialPort_open只對串口進行相關設置,如波特,數據位等。由此看出
frameworks/base/services/core/jni/com_android_server_SerialService.cpp和
frameworks/base/core/jni/android_hardware_SerialPort.cpp可能處於兩個不同的線程中,這兩個不
同的線程對同一個文件進行訪問,需要對文件描述符進行轉換。
在frameworks/base/core/jni/android_hardware_SerialPort.cpp中

 1 static void android_hardware_SerialPort_open(JNIEnv *env, jobject thiz, jobject fileDescriptor, jint speed)
 2 {
 3     ......
 4     int fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
 5     // duplicate the file descriptor, since ParcelFileDescriptor will eventually close its copy
 6     fd = dup(fd);
 7     if (fd < 0) {
 8         jniThrowException(env, "java/io/IOException", "Could not open serial port");
 9         return;
10     }
11     ......
12 }

 

 

(C) 設置權限

device/sprd/scx35l/common/rootdir/root/ueventd.sc8830.rc

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

 

device/sprd/scx35l/common/sepolicy/file_contexts

1 ......
2 /dev/ttytiny0        u:object_r:serial_device:s0
3 ......

 

device/sprd/scx35l/common/sepolicy/system_app.te

......
allow system_app serial_device:chr_file { open read write ioctl};
......

 


(D) 測試串口
    這里引用Android6.0源碼目錄下的一個串口測試app,還要添加一些權限等設置才能正常使用。
串口測試APP源碼:frameworks/base/tests/SerialChat

1. 設置權限:

frameworks/base/tests/SerialChat/Android.mk

1 ......
2 LOCAL_CERTIFICATE := platform
3 ......

 

frameworks/base/tests/SerialChat/AndroidManifest.xml

1 ......
2 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
3     package="com.android.serialchat"
4     android:sharedUserId="android.uid.system">
5 ......

 

2. 添加串口節點名稱
    應用程序根據這個名稱來打開對應的串口。

frameworks/base/core/res/res/values/config.xml

1 ......
2 <string-array translatable="false" name="config_serialPorts">
3            <item>"/dev/ttytiny0"</item>
4 </string-array>
5 ......

    用mmm frameworks/base/tests/SerialChat -B 編譯出的SerialChat.apk push到機器之后,就可以
進行測試了。

    a.點擊打開SerialChat.apk:

       

    

    b.輸入要發送的內容

       

 

    c.點擊確定(打鈎的位置)發送,並且接收顯示在上方。

       

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM