详细解读android6.0SerialPort服务

简介: 详细解读android6.0SerialPort服务

上一篇博客描述了一个简单的串口应用程序和驱动程序,了解了应用程序访问串口的基本操作,如

打开串口,设置串口,写串口,读串口,关闭串口等。和Linux串口驱动的基本框架。这里将了解Android

下的串口系统框架,仍然使用上一篇博客中的tiny_serial.c作为本例的驱动,本例实现的功能不变,无

论应用程序写任何数据到串口,都能从该串口中读回。

关于Android系统服务基本框架,可以参考博客,

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() {

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 }}//代码效果参考:http://www.ezhiqi.com/bx/art_3713.html);

可以访问串口服务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操作流程。

博客中,只有服务端调用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

3 package="com.android.serialchat"

4 android:sharedUserId="android.uid.system"

5 ......

2. 添加串口节点名称

应用程序根据这个名称来打开对应的串口。

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

1 ......

2 [span style="color: rgba(0, 0, 255, 1)">string-array translatable="false" name="config_serialPorts"

3 "/dev/ttytiny0"

4

5 ......

用mmm frameworks/base/tests/SerialChat -B 编译出的SerialChat.apk push到机器之后,就可以

进行测试了。

a.点击打开SerialChat.apk:

b.输入要发送的内容

c.点击确定(打钩的位置)发送,并且接收显示在上方。

相关文章
|
5月前
|
Android开发
Android 11 添加Service服务SELinux问题
Android 11 添加Service服务SELinux问题
233 1
|
5月前
|
Java API 调度
Android系统 自定义开机广播,禁止后台服务,运行手动安装应用接收开机广播
Android系统 自定义开机广播,禁止后台服务,运行手动安装应用接收开机广播
237 0
|
4月前
|
调度 Android开发
43. 【Android教程】服务:Service
43. 【Android教程】服务:Service
43 2
|
22天前
|
JavaScript 前端开发 Android开发
让Vite+Vue3项目在Android端离线打开(不需要起服务)
让Vite+Vue3项目在Android端离线打开(不需要起服务)
|
1月前
|
调度 Android开发 UED
Android经典实战之Android 14前台服务适配
本文介绍了在Android 14中适配前台服务的关键步骤与最佳实践,包括指定服务类型、请求权限、优化用户体验及使用WorkManager等。通过遵循这些指南,确保应用在新系统上顺畅运行并提升用户体验。
51 6
|
8天前
|
安全 API 开发工具
Android平台RTMP推送|轻量级RTSP服务如何实现麦克风|扬声器声音采集切换
Android平台扬声器播放声音的采集,在无纸化同屏等场景下,意义很大,早期低版本的Android设备,是没法直接采集扬声器audio的(从Android 10开始支持),所以,如果需要采集扬声器audio,需要先做系统版本判断,添加相应的权限。
|
8天前
|
编解码 开发工具 Android开发
Android平台实现屏幕录制(屏幕投影)|音频播放采集|麦克风采集并推送RTMP或轻量级RTSP服务
Android平台屏幕采集、音频播放声音采集、麦克风采集编码打包推送到RTMP和轻量级RTSP服务的相关技术实现,做成高稳定低延迟的同屏系统,还需要有配套好的RTMP、RTSP直播播放器
|
2月前
|
数据处理 开发工具 数据安全/隐私保护
Android平台RTMP推送|轻量级RTSP服务|GB28181接入之文字、png图片水印的精进之路
本文探讨了Android平台上推流模块中添加文字与PNG水印的技术演进。自2015年起,为了满足应急指挥及安防领域的需求,逐步发展出三代水印技术:第一代为静态文字与图像水印;第二代实现了动态更新水印内容的能力,例如实时位置与时间信息;至第三代,则优化了数据传输效率,直接使用Bitmap对象传递水印数据至JNI层,减少了内存拷贝次数。这些迭代不仅提升了用户体验和技术效率,也体现了开发者追求极致与不断创新的精神。
|
2月前
|
数据采集 编解码 开发工具
Android平台实现无纸化同屏并推送RTMP或轻量级RTSP服务(毫秒级延迟)
一个好的无纸化同屏系统,需要考虑的有整体组网、分辨率、码率、实时延迟、音视频同步和连续性等各个指标,做容易,做好难
|
2月前
|
监控 开发工具 Android开发
Android平台实现RTSP拉流转发至轻量级RTSP服务
为满足Android平台上从外部RTSP摄像头拉流并提供轻量级RTSP服务的需求,利用大牛直播SDK实现了相关功能。SDK支持开始与停止拉流、音频视频数据回调处理及RTSP服务的启动与发布等操作。拉流仅需将未解码数据回调,对性能影响小。音频和视频数据经由特定接口传递给发布端进行处理。此外,SDK还提供了获取RTSP会话数量的功能。此方案适用于监控和巡检等低延迟应用场景,并支持二次水印添加等功能。
下一篇
无影云桌面