Android9.0源码学习-Sensor Framework

2024-06-06 22:32

本文主要是介绍Android9.0源码学习-Sensor Framework,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • FileList
  • Questions
  • SensorService初始化
  • registerListener流程分析
    • APP
    • Framework
      • Java
      • Native
        • Client
        • Server
  • How the app get sensor data
    • Server
    • BitTube
    • Client
  • Reference

在之前的学习中, Android Sensor概述介绍了常用传感器的功能,测量值的含义,测量原理等等。 Android Sensor应用介绍了app如何得到Sensor的数值。那么接下来就应该思考应用层是如何得到硬件测量的数值,传感器的数值又是如何一步一步上传至应用层的。

FileList

Framework(/frameworks/base/core/java/android/hardware/)

  • SensorManager.java:SDK接口,封装了Sensor相关的API,提供给APP层使用。
  • SystemSensorManager.java:继承于SensorManager,SensorManager的实现类,负责与系统Sensor的联系。
  • Sensor.java:代表一个Sensor类
  • SensorEvent.java:代表一个SensorEvent类
  • SensorEventListener.java:用于接收传感器事件的Interface

JNI(/framework/base/core/jni)

  • android_hardware_SensorManager.cpp:负责Java层和Native层的通信

Native Framework

  • Client(/framework/native/lib/sensor)
    • SensorManager.cpp:SensorManager的Native层,负责与SensorService的通信
    • SensorEventQueue.cpp:消息队列
    • Sensor.cpp:Sensor的native层
    • ISensorServer.cpp:SensorServer在客户端接口
    • ISensorEventConnection.cpp:SensorEventConnection在客户端接口
    • BitTube.cpp:单向字节管道,提供进程间单向数据通信功能。
  • Server(/framework/native/services/sensorservice)
    • SensorDevice.cpp:负责管理和维护系统中所有的Sensor,封装了Sensor的使能,配置,数据读取等功能。
    • SensorInterface.cpp:服务端Sensor接口
    • SensorService.cpp:是整个Android Sensor Framework最核心的模块,它实现了主要的Sensor控制流和数据流,完成Sensor的参数配置,数据分发
    • SensorEventConnection.cpp:Sensor的数据传输通道,当Client开始监听某一个Sensor时,一个对应的SensorEventConnection将会被创建,Server端在接收到Sensor数据后,通过写入到SensorEventConnection传递给Client端。

Questions

在Framework的学习中,我们需要搞清楚以下几个问题?

  • APP层的控制指令是怎样一层一层下发到Framework层和HAL层?
  • sensor采集到的数据又是怎样一层一层传到APP层?
  • Client端和Server端又是怎样交互的,数据是如何通讯的?
  • 当系统第一次创建SensorService时,SensorService做了哪些初始化的工作?

SensorService初始化

SystemServer的main函数,会调用startSensorService(),从而创建第一个SensorService实例。当SensorService第一个实例创建时,其onFirstRef()接口会被调用。

onFirstRef()做了这么以下几个工作:

  • 创建SensorDevice的实例(SensorDevice& dev(SensorDevice::getInstance());),并且是单例类
    • connectHidlService()
      • mSensors = ISensors::getService()
        • Sensors::Sensors()
          • hw_get_module()加载Sensor HAL的动态库
          • sensors_open_1()执行Sensor HAL的初始化
    • 初始化mSensorListVector<sensor_t> mSensorList
    • mSensors->activate()
  • 通过dev获取SensorList(list
  • registerSensor
  • 创建Looper(mLooper = new Looper(false)),用于enable sensor后,进行数据的接收
  • 创建SensorEventAckReceiver(mAckReceiver = new SensorEventAckReceiver(this)):用于在dispatch wake up sensor event给上层后,接收上层返回的确认ACK。
    在这里插入图片描述

registerListener流程分析

APP

下面的代码是在Android官网提供的APP获取light sensor数据的代码段,这里在获取SensorManager和Sensor后,会调用sensorManager.registerListener(this, mLight, SensorManager.SENSOR_DELAY_NORMAL);注册。

public class SensorActivity extends Activity implements SensorEventListener {private SensorManager sensorManager;private Sensor mLight;@Overridepublic final void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);sensorManager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);mLight = sensorManager.getDefaultSensor(Sensor.TYPE_LIGHT);}@Overridepublic final void onAccuracyChanged(Sensor sensor, int accuracy) {// Do something here if sensor accuracy changes.}@Overridepublic final void onSensorChanged(SensorEvent event) {// The light sensor returns a single value.// Many sensors return 3 values, one for each axis.float lux = event.values[0];// Do something with this sensor value.}@Overrideprotected void onResume() {super.onResume();sensorManager.registerListener(this, mLight, SensorManager.SENSOR_DELAY_NORMAL);}@Overrideprotected void onPause() {super.onPause();sensorManager.unregisterListener(this);}
}

Framework

Java

在java层的调用顺序为
SensorManager.java

  • registerListener()
    • registerListenerImpl()

SystemSensorManager.java

  • registerListenerImpl()

registerListenerImpl()会维护一个mSensorListeners的HashMap(HashMap<SensorEventListener, SensorEventQueue> mSensorListeners)。

  • queue.addSensor()SensorEventQueue queue
    • mActiveSensors.put(handle, true)handle = snesor.getHandle()
    • enableSensor()
      • nativeEnableSensor()到这里会通过JNI去call到Native层

/frameworks/base/core/java/android/hardware/SystemSensorManager.java

    @Overrideprotected boolean registerListenerImpl(SensorEventListener listener, Sensor sensor,int delayUs, Handler handler, int maxBatchReportLatencyUs, int reservedFlags) {if (listener == null || sensor == null) {Log.e(TAG, "sensor or listener is null");return false;}// Trigger Sensors should use the requestTriggerSensor call.if (sensor.getReportingMode() == Sensor.REPORTING_MODE_ONE_SHOT) {Log.e(TAG, "Trigger Sensors should use the requestTriggerSensor.");return false;}if (maxBatchReportLatencyUs < 0 || delayUs < 0) {Log.e(TAG, "maxBatchReportLatencyUs and delayUs should be non-negative");return false;}if (mSensorListeners.size() >= MAX_LISTENER_COUNT) {throw new IllegalStateException("register failed, "+ "the sensor listeners size has exceeded the maximum limit "+ MAX_LISTENER_COUNT);}// Invariants to preserve:// - one Looper per SensorEventListener// - one Looper per SensorEventQueue// We map SensorEventListener to a SensorEventQueue, which holds the loopersynchronized (mSensorListeners) {SensorEventQueue queue = mSensorListeners.get(listener);if (queue == null) {Looper looper = (handler != null) ? handler.getLooper() : mMainLooper;final String fullClassName =listener.getClass().getEnclosingClass() != null? listener.getClass().getEnclosingClass().getName(): listener.getClass().getName();queue = new SensorEventQueue(listener, looper, this, fullClassName);if (!queue.addSensor(sensor, delayUs, maxBatchReportLatencyUs)) {queue.dispose();return false;}mSensorListeners.put(listener, queue);return true;} else {return queue.addSensor(sensor, delayUs, maxBatchReportLatencyUs);}}}

Native

Client

通过JNI,java层的enableSensor()会call到SensorEventQueue.cpp的enableSensor(),到这里就要去通过mSensorEventConnectionconst sp<ISensorEventConnection>)去call到Server端了。

/frameworks/native/libs/sensor/SensorEventQueue.cpp

status_t SensorEventQueue::enableSensor(Sensor const* sensor, int32_t samplingPeriodUs) const {return mSensorEventConnection->enableDisable(sensor->getHandle(), true,us2ns(samplingPeriodUs), 0, 0);
}

/frameworks/native/libs/sensor/ISensorEventConnection.cpp

    virtual status_t enableDisable(int handle, bool enabled, nsecs_t samplingPeriodNs,nsecs_t maxBatchReportLatencyNs, int reservedFlags){Parcel data, reply;data.writeInterfaceToken(ISensorEventConnection::getInterfaceDescriptor());data.writeInt32(handle);data.writeInt32(enabled);data.writeInt64(samplingPeriodNs);data.writeInt64(maxBatchReportLatencyNs);data.writeInt32(reservedFlags);remote()->transact(ENABLE_DISABLE, data, &reply);return reply.readInt32();}
Server

/frameworks/native/services/sensorservice/SensorEventConnection.cpp

status_t SensorService::SensorEventConnection::enableDisable(int handle, bool enabled, nsecs_t samplingPeriodNs, nsecs_t maxBatchReportLatencyNs,int reservedFlags)
{status_t err;if (enabled) {err = mService->enable

这篇关于Android9.0源码学习-Sensor Framework的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1037409

相关文章

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

8种快速易用的Python Matplotlib数据可视化方法汇总(附源码)

《8种快速易用的PythonMatplotlib数据可视化方法汇总(附源码)》你是否曾经面对一堆复杂的数据,却不知道如何让它们变得直观易懂?别慌,Python的Matplotlib库是你数据可视化的... 目录引言1. 折线图(Line Plot)——趋势分析2. 柱状图(Bar Chart)——对比分析3

重新对Java的类加载器的学习方式

《重新对Java的类加载器的学习方式》:本文主要介绍重新对Java的类加载器的学习方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、简介1.2、符号引用和直接引用1、符号引用2、直接引用3、符号转直接的过程2、加载流程3、类加载的分类3.1、显示

Android实现一键录屏功能(附源码)

《Android实现一键录屏功能(附源码)》在Android5.0及以上版本,系统提供了MediaProjectionAPI,允许应用在用户授权下录制屏幕内容并输出到视频文件,所以本文将基于此实现一个... 目录一、项目介绍二、相关技术与原理三、系统权限与用户授权四、项目架构与流程五、环境配置与依赖六、完整

Android 12解决push framework.jar无法开机的方法小结

《Android12解决pushframework.jar无法开机的方法小结》:本文主要介绍在Android12中解决pushframework.jar无法开机的方法,包括编译指令、框架层和s... 目录1. android 编译指令1.1 framework层的编译指令1.2 替换framework.ja

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

Java 正则表达式URL 匹配与源码全解析

《Java正则表达式URL匹配与源码全解析》在Web应用开发中,我们经常需要对URL进行格式验证,今天我们结合Java的Pattern和Matcher类,深入理解正则表达式在实际应用中... 目录1.正则表达式分解:2. 添加域名匹配 (2)3. 添加路径和查询参数匹配 (3) 4. 最终优化版本5.设计思

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++