上一篇博客描述了一個簡單的串口應用程序和驅動程序,瞭解了應用程序訪問串口的基本操做,如
打開串口,設置串口,寫串口,讀串口,關閉串口等。和Linux串口驅動的基本框架。這裏將瞭解Android
下的串口系統框架,仍然使用上一篇博客中的tiny_serial.c做爲本例的驅動,本例實現的功能不變,無
論應用程序寫任何數據到串口,都能從該串口中讀回。
關於Android系統服務基本框架,能夠參考http://www.cnblogs.com/hackfun/p/7612617.html博客,
Android串口(serial port)服務框架與該例子十分相似,這裏做簡單描述。固然也有些不一樣的地方,後面會分析。html
(A) 串口服務的基本框架
1. 註冊驅動
2. 註冊hal
3. 註冊JNI
4. 註冊和添加SerialService
5. 串口管理SerialManager
(B) 打開串口
(C) 設置權限
(D) 測試串口java
(A) 串口服務的基本框架
1. 註冊驅動
經過加載kernel/driver/tty/serial/tiny_serial.c驅動,生成/dev/ttytiny0節點,應用經過訪問
該節點,實現的對串口的open、read、write、close等操做。android
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提供底層驅動訪問的接口,如:app
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
添加服務:ide
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)接口ui
5. 串口管理SerialManager
frameworks/base/core/java/android/hardware/SerialManager.java中SerialManager對串口操做進一步管理,
客戶端只要實例化一個SerialManager對象,使用該對象的方法訪問串口。不過,該對象只對
openSerialPort(String name, int speed)管理。spa
(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.點擊肯定(打鉤的位置)發送,而且接收顯示在上方。