Android Senor Framework (三)SensorService启动

2024-04-18 10:58

本文主要是介绍Android Senor Framework (三)SensorService启动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

SystemServer 启动SensorService

Zygote启动之后,调用SystemServer的main方法(调用run方法)启动系统服务;

代码路径:

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

SystemServer类中提供的run 方法中,在启动service之前,会加载本地动态库System.loadLibrary(“android_servers”)初始化本地 Native service,过程如下:

/*** The main entry point from zygote.*/
public static void main(String[] args) {new SystemServer().run();
}
public final class SystemServer {private static final String TAG = "SystemServer";private void run() {// Initialize native services.System.loadLibrary("android_servers");// Start services.try {Trace.traceBegin(Trace.TRACE_TAG_SYSTEM_SERVER, "StartServices");startBootstrapServices();startCoreServices();startOtherServices();}
...
}

在 startBootstrapServices(); 的过程的最后 会调用startSensorService(); 如下:

    private void startBootstrapServices() {
......// The sensor service needs access to package manager service, app ops// service, and permissions service, therefore we start it after them.startSensorService();}

在SystemServer类定义中,定义了本地方法startSensorService 如下:

public final class SystemServer {                                                                                                                                                                                   private static final String TAG = "SystemServer";//......private static native void startSensorService();//......
}

当SystemServer调用run 方法,会通过JNI调用到本地方法;

JNI 访问native 方法

通过对JNI的了解 System.loadLibrary(“android_servers”); 会去查找libandroid_servers.so 这个库文件;

yujixuan@yujixuan:~/prj/SC20_R06_master_0526/code/frameworks$ grep -rn libandroid_servers ./
./base/services/Android.mk:54:LOCAL_MODULE:= libandroid_servers

通过检索可知,加载该本地库,会调用在 base/services/ 下编译库文件的onload函数;即:

代码路径: ./base/services/core/jni/onload.cpp
extern "C" jint JNI_OnLoad(JavaVM* vm, void* /* reserved */)
{JNIEnv* env = NULL;jint result = -1;if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {ALOGE("GetEnv failed!");return result;}ALOG_ASSERT(env, "Could not retrieve the env!");register_android_server_ActivityManagerService(env);register_android_server_PowerManagerService(env);register_android_server_SerialService(env);register_android_server_InputApplicationHandle(env);register_android_server_InputWindowHandle(env);register_android_server_InputManager(env);register_android_server_LightsService(env);register_android_server_AlarmManagerService(env);register_android_server_UsbDeviceManager(env);register_android_server_UsbMidiDevice(env);register_android_server_UsbHostManager(env);register_android_server_vr_VrManagerService(env);register_android_server_VibratorService(env);register_android_server_SystemServer(env);
//......return JNI_VERSION_1_4;
}

由上可知,在JNI_OnLoad 中会调用添加若干个 server,没有发现sensor相关的server;

实则它是 SystemServer中,下面是register_android_server_SystemServer内容:

代码路径:./base/services/core/jni/com_android_server_SystemServer.cpp
/** JNI registration.*/
static const JNINativeMethod gMethods[] = {/* name, signature, funcPtr */{ "startSensorService", "()V", (void*) android_server_SystemServer_startSensorService },
};int register_android_server_SystemServer(JNIEnv* env)
{return jniRegisterNativeMethods(env, "com/android/server/SystemServer",gMethods, NELEM(gMethods));
}

可以看到 register_android_server_SystemServer 实际上是注册绑定了 sensor相关的 service启动方法(为啥不直接命名为sensor server,目前还不清楚);

通过method可知,systemServer调用run方法启动 startSensorService,会调用到android_server_SystemServer_startSensorService ; 以下是它的实现:

static void android_server_SystemServer_startSensorService(JNIEnv* /* env */, jobject /* clazz */) {char propBuf[PROPERTY_VALUE_MAX];property_get("system_init.startsensorservice", propBuf, "1");if (strcmp(propBuf, "1") == 0) {// Start the sensor service in a new threadcreateThreadEtc(start_sensor_service, nullptr,"StartSensorThread", PRIORITY_FOREGROUND);}
}

判断属性 “system_init.startsensorservice” 是否存在,如果存在 创建线程"StartSensorThread"(默认被配置了,还找到哪里配置的)

线程内容如下:

static int start_sensor_service(void* /*unused*/) {SensorService::instantiate();return 0;
}

该线程调用SensorService中的instantiate方法,创建了SensorService的实例;

SensorService实现

继续分析SensorService instantiate的创建,首先是看SensorService类的定义:

路径:./frameworks/native/services/sensorservice/SensorService.h
class SensorService :public BinderService<SensorService>,public BnSensorServer,protected Thread
{// nested class/struct for internal useclass SensorEventConnection;public:void cleanupConnection(SensorEventConnection* connection);status_t enable(const sp<SensorEventConnection>& connection, int handle,nsecs_t samplingPeriodNs,  nsecs_t maxBatchReportLatencyNs, int reservedFlags,const String16& opPackageName);status_t disable(const sp<SensorEventConnection>& connection, int handle);status_t setEventRate(const sp<SensorEventConnection>& connection, int handle, nsecs_t ns,const String16& opPackageName);status_t flushSensor(const sp<SensorEventConnection>& connection,const String16& opPackageName);
private:friend class BinderService<SensorService>;// nested class/struct for internal useclass SensorRecord;class SensorEventAckReceiver;struct SensorRegistrationInfostatic char const* getServiceName() ANDROID_API { return "sensorservice"; }SensorService() ANDROID_API;virtual ~SensorService();virtual void onFirstRef();// Thread interfacevirtual bool threadLoop();// ISensorServer interfacevirtual Vector<Sensor> getSensorList(const String16& opPackageName);virtual status_t dump(int fd, const Vector<String16>& args);String8 getSensorName(int handle) const;bool isVirtualSensor(int handle) const;sp<SensorInterface> getSensorInterfaceFromHandle(int handle) const;bool isWakeUpSensor(int type) const;void recordLastValueLocked(sensors_event_t const* buffer, size_t count);static void sortEventBuffer(sensors_event_t* buffer, size_t count);const Sensor& registerSensor(SensorInterface* sensor,bool isDebug = false, bool isVirtual = false);const Sensor& registerVirtualSensor(SensorInterface* sensor, bool isDebug = false);
};

删了一些相对无关的方法及属性后,还有如上内容,没有instantiate方法;

通过类的定义可以发现SensorService 继承了BinderService;BnSensorServer,Thread这三个class; instantiate 是被定义在它的父类BinderService中了;

以下是BinderService的实现:

代码路径:./frameworks/native/include/binder/BinderService.hclass BinderService
{
public:static status_t publish(bool allowIsolated = false) {sp<IServiceManager> sm(defaultServiceManager());return sm->addService(String16(SERVICE::getServiceName()),new SERVICE(), allowIsolated);}static void publishAndJoinThreadPool(bool allowIsolated = false) {publish(allowIsolated);joinThreadPool();}static void instantiate() { publish(); }
......
};

publish()中

sp sm(defaultServiceManager());

可知SensorService::instantiate()在这一过程创建了SensorService并通过addService将自己新创建的SensorService服务添加到Android服务列表里了。

SensorService有一条继承关系如下:

SensorService : public BnSensorServer
BnSensorServer : public BnInterface<ISensorServer>
BnInterface :  public BBinder
BBinder : public IBinder
IBinder : public virtual RefBase 

在Android引用计数系统里,当RefBase的子类对象被第一次强引用时自动调用其onFirstRef方法,所以当第一次使用SensorService,onFirstRef方法将被被自动回调,onFirstRef方法里会创建SensorDevice 用来与HAL层进行交互,后面内容会做分析。

note:Android系统智能指针的设计思路(轻量级指针、强指针、弱指针) 虚继承

void SensorService::onFirstRef() {ALOGD("nuSensorService starting...");SensorDevice& dev(SensorDevice::getInstance());
//......

时序图:

根据以上过程,可绘制绘制出简要的uml时序图,如下:

总结过程:

  1. 系统初始化进程加载,启动SystemServer,Zygote中会执行SystemServier main方法,导致其run方法被调用;
  2. 加载本地库文件, System.loadLibrary("android_servers"); 获取本地方法;
  3. 被加载到的JNI库文件导致JNI_Onload函数被调用;调用本地jni文件
  4. 注册本地方法jniRegisterNativeMethods 数组;
  5. 完成Java 到 C++ 函数绑定,使Java能否访问到C库中的函数;
  6. 启动startBootstrapServices();
  7. 最后调用native方法 native void startSensorService();
  8. JNI文件com_android_server_SystemServer.cpp,绑定的函数数组,由java的startSensorService方法绑定到android_server_SystemServer_startSensorService函数;
  9. C++函数中,start_sensor_service被调用;
  10. 调用SensorService的inistantiate函数(继承父类BinderService得到的);
  11. 调用publish
  12. 创建一个Serivces,通过sm->addService 来添加到android中去; sm->addService( String16(SERVICE::getServiceName()), new SERVICE(), allowIsolated); 其中sm是ServiceManager的引用:sp sm(defaultServiceManager());

这篇关于Android Senor Framework (三)SensorService启动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/914592

相关文章

MySQL数据库宕机,启动不起来,教你一招搞定!

作者介绍:老苏,10余年DBA工作运维经验,擅长Oracle、MySQL、PG、Mongodb数据库运维(如安装迁移,性能优化、故障应急处理等)公众号:老苏畅谈运维欢迎关注本人公众号,更多精彩与您分享。 MySQL数据库宕机,数据页损坏问题,启动不起来,该如何排查和解决,本文将为你说明具体的排查过程。 查看MySQL error日志 查看 MySQL error日志,排查哪个表(表空间

springboot3打包成war包,用tomcat8启动

1、在pom中,将打包类型改为war <packaging>war</packaging> 2、pom中排除SpringBoot内置的Tomcat容器并添加Tomcat依赖,用于编译和测试,         *依赖时一定设置 scope 为 provided (相当于 tomcat 依赖只在本地运行和测试的时候有效,         打包的时候会排除这个依赖)<scope>provided

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

内核启动时减少log的方式

内核引导选项 内核引导选项大体上可以分为两类:一类与设备无关、另一类与设备有关。与设备有关的引导选项多如牛毛,需要你自己阅读内核中的相应驱动程序源码以获取其能够接受的引导选项。比如,如果你想知道可以向 AHA1542 SCSI 驱动程序传递哪些引导选项,那么就查看 drivers/scsi/aha1542.c 文件,一般在前面 100 行注释里就可以找到所接受的引导选项说明。大多数选项是通过"_

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M

Linux服务器Java启动脚本

Linux服务器Java启动脚本 1、初版2、优化版本3、常用脚本仓库 本文章介绍了如何在Linux服务器上执行Java并启动jar包, 通常我们会使用nohup直接启动,但是还是需要手动停止然后再次启动, 那如何更优雅的在服务器上启动jar包呢,让我们一起探讨一下吧。 1、初版 第一个版本是常用的做法,直接使用nohup后台启动jar包, 并将日志输出到当前文件夹n

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

衡石分析平台使用手册-单机安装及启动

单机安装及启动​ 本文讲述如何在单机环境下进行 HENGSHI SENSE 安装的操作过程。 在安装前请确认网络环境,如果是隔离环境,无法连接互联网时,请先按照 离线环境安装依赖的指导进行依赖包的安装,然后按照本文的指导继续操作。如果网络环境可以连接互联网,请直接按照本文的指导进行安装。 准备工作​ 请参考安装环境文档准备安装环境。 配置用户与安装目录。 在操作前请检查您是否有 sud

SpringBoot项目是如何启动

启动步骤 概念 运行main方法,初始化SpringApplication 从spring.factories读取listener ApplicationContentInitializer运行run方法读取环境变量,配置信息创建SpringApplication上下文预初始化上下文,将启动类作为配置类进行读取调用 refresh 加载 IOC容器,加载所有的自动配置类,创建容器在这个过程