android硬件访问服务程序框架和简单实例分析

2024-09-05 22:18

本文主要是介绍android硬件访问服务程序框架和简单实例分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

android硬件访问服务程序框架

  • loadLibrary 进行加载C库
  • JNI_OnLoad 注册本地方法
    – 分别调用各个硬件的函数来注册本地方法
  • SystemServer 使用本地方法(在java代码中,例如InputManagerService),对每一个硬件构造service,将其添加到系统中
  • APP 使用:获得服务 getService ;使用服务。执行service

SystemServer提供众多的服务service.
前三者属于系统进程
第四个属于用户进程
不同进程之间进行通信。所借助的是系统的binder服务。


以下属于针对上述分析

涉及的代码

\android5.0.2\frameworks\base\services\java\com\android\server\SystemServer.java
\frameworks\base\core\java\android\hardware\input\InputManager.java
\frameworks\base\services\core\java\com\android\server\input\InputManagerService.java
\frameworks\base\core\java\android\content\Context.java
\frameworks\base\core\java\android\app\ContextImpl.java
\frameworks\base\core\java\android\hardware\input\IInputManager.aidl

实例

    //LoadLibrary加载C库 System.loadLibrary("android_servers");nativeInit();//...startOtherServices(); //...//利用本地方法,构造一个service   InputManagerService  在这个java文件中嗲用native_init。从而跳转到了本地  inputManager = null;inputManager = new InputManagerService(context);//添加到系统中ServiceManager.addService(Context.INPUT_SERVICE, inputManager);//以上属于系统进程//用户进程如何使用呢?getSystemService(Context.INPUT_SERVICE);//就得到了这个服务。就可以通过service执行的一些操作

app使用

app的使用是调用getSystemService(Context.xxx来使用的)
getSystemService在Context.java中进行定义

    public abstract Object getSystemService(@ServiceName @NonNull String name);

可以看出其为抽象的基类,其实现实在ContextImpl中实现的。

 @Overridepublic Object getSystemService(String name) {ServiceFetcher fetcher = SYSTEM_SERVICE_MAP.get(name);return fetcher == null ? null : fetcher.getService(this);}

使用getSystemService获得的返回值,一般需要强转为一个Manager的类。
例如

mIm = (InputManager)Context.getSystemService(Context.INPUT_SERVICE);

问题

  • nativeInit(); 是怎么关联本地方法
  • 应用程序中自己LoadLibrary的话,另外一个app,是如何得到这个service的。其可能是通过aidl来获得这个服务的。这是和系统getSystemService不同的地方。getSystemService也是通过ServiceManager.getService来获得的。应用本身通过aidl来实现,android在编译的时候,自动实现了binder的机制。

注册本地方法

在JNI中实现
在onLoad.cpp中

android_server的实现

在framework/base/services下,
有一个Andrpid.mk
这个编译文件中,会将整个的Services目录的内容编译成两部分,

LOCAL_MODULE := services
LOCAL_MODULE:= libandroid_servers

目录下的java文件会编译为services
目录下jni下的
include (wildcard (LOCAL_PATH)/*/jni/Android.mk)
有android.mk包含的文件将会形成一个 动态库 android_service 供SystemServer进行调用。

JNI目录下的Android.mk

LOCAL_SRC_FILES += \
    $(LOCAL_REL_DIR)/com_android_server_AlarmManagerService.cpp \
    $(LOCAL_REL_DIR)/com_android_server_am_BatteryStatsService.cpp \
$(LOCAL_REL_DIR)/onload.cppLOCAL_C_INCLUDES += \
$(JNI_H_INCLUDE) \
frameworks/base/services \
frameworks/base/libs \

LOCAL_SHARED_LIBRARIES += \
    libandroid_runtime \

主要是包含一些cpp文件、头文件以及需要的动态库文件

Load_Library的入口 onload.cpp

extern "C" jint JNI_OnLoad(JavaVM* vm, void* reserved)

调用

System.loadLibrary("android_servers");

的时候,会调用到JNI_OnLoad方法。
在这个方法中,会调用到和硬件相关的jni的cpp文件

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_UsbHostManager(env);
register_android_server_VibratorService(env);
register_android_server_SystemServer(env);
register_android_server_location_GpsLocationProvider(env);
register_android_server_location_FlpHardwareProvider(env);
register_android_server_connectivity_Vpn(env);
register_android_server_AssetAtlasService(env);
register_android_server_ConsumerIrService(env);
register_android_server_BatteryStatsService(env);
register_android_server_hdmi_HdmiCecController(env);
register_android_server_tv_TvInputHal(env);
register_android_server_PersistentDataBlockService(env);
register_android_server_fingerprint_FingerprintService(env);
register_android_server_Watchdog(env);

服务的加载和被调用的过程

利用本地函数构造service是怎么样的过程呢?

答:在new InputManagerService的时候调用本地函数native_init
InputManagerService 继承IInputManager.stub
InputManagerService中提供相关的服务端的程序,供客户端调用。

在构造出service之后,利用

ServiceManager.addService(Context.INPUT_SERVICE, inputManager);

添加到ServiceManager中。

Context中InputManager的实话

在Context的实现类ContextImpl中,

registerService(INPUT_SERVICE, new StaticServiceFetcher() {public Object createStaticService() {return InputManager.getInstance();}});

此时复写的createStaticService在调用getSystemService的时候被掉用。

在app应用中,需要使用这些Context的service的时候,调用
getSystemService(Context.xxx)同时将返回值强转为一个xxxManager类型。

InputManager 中通过getInstance获得 添加到ServiceManager的服务

public static InputManager getInstance() {synchronized (InputManager.class) {if (sInstance == null) {IBinder b = ServiceManager.getService(Context.INPUT_SERVICE);sInstance = new InputManager(IInputManager.Stub.asInterface(b));}return sInstance;}}

如此一来,客户端app就能通过binder机制,调用系统提供的服务了。

ClipboardService实例

抽象类
frameworks\base\core\java\android\text\ClipboardManager.java
实现类
frameworks\base\core\java\android\content\ClipboardManager.java
ClipboardManager中获取到服务

   ClipboardManager clipboardManager =(ClipboardManager) mContext.getSystemService(Context.CLIPBOARD_SERVICE);

ClipboardManager提供的一个方法。

 clipboardManager.setText(expected);

在text目录下的ClipboardManger.java中的setText是一个抽象方法。

frameworks\base\core\java\android\text\ClipboardManager.javapublic abstract void setText(CharSequence text);

在Content下的ClipbpadrManager.java实现了这个方法。故而调用如下代码

//首先调用InputManager下的setPrimaryClippublic void setText(CharSequence text) {setPrimaryClip(ClipData.newPlainText(null, text));}//调用getService的setPrimaryClippublic void setPrimaryClip(ClipData clip) {try {if (clip != null) {clip.prepareToLeaveProcess();}getService().setPrimaryClip(clip, mContext.getOpPackageName());} catch (RemoteException e) {}}//getService 通过ServiceManager获得clipboard的IBinder对象,并转换成为IClipboard 对象static private IClipboard getService() {synchronized (sStaticLock) {if (sService != null) {return sService;}IBinder b = ServiceManager.getService("clipboard");sService = IClipboard.Stub.asInterface(b);return sService;}}// 通过aidl调用服务器部分的setPrimaryClip,这部分代码系统提供。void setPrimaryClip(in ClipData clip, String callingPackage);//aidl的实现//ClipboardServicepublic void setPrimaryClip(ClipData clip, String callingPackage) {synchronized (this) {if (clip != null && clip.getItemCount() <= 0) {throw new IllegalArgumentException("No items");}final int callingUid = Binder.getCallingUid();if (mAppOps.noteOp(AppOpsManager.OP_WRITE_CLIPBOARD, callingUid,callingPackage) != AppOpsManager.MODE_ALLOWED) {return;}checkDataOwnerLocked(clip, callingUid);final int userId = UserHandle.getUserId(callingUid);PerUserClipboard clipboard = getClipboard(userId);revokeUris(clipboard);setPrimaryClipInternal(clipboard, clip);List<UserInfo> related = getRelatedProfiles(userId);if (related != null) {int size = related.size();if (size > 1) { // Related profiles list include the current profile.boolean canCopy = false;try {canCopy = !mUm.getUserRestrictions(userId).getBoolean(UserManager.DISALLOW_CROSS_PROFILE_COPY_PASTE);} catch (RemoteException e) {Slog.e(TAG, "Remote Exception calling UserManager: " + e);}// Copy clip data to related users if allowed. If disallowed, then remove// primary clip in related users to prevent pasting stale content.if (!canCopy) {clip = null;} else {clip.fixUrisLight(userId);}for (int i = 0; i < size; i++) {int id = related.get(i).id;if (id != userId) {setPrimaryClipInternal(getClipboard(id), clip);}}}}}}

总结

  • 分析安卓系统时,需要记住android的c/s框架。SystemServer提供众多的service。这些service是由ServiceManager来进行管理的。注册时候需要使用ServiceManager.addService进行添加,客户端获取service,需要通过ServiceManager.getService来获取。
  • Context上下文是和应用程序密切相关的。应用程序可以通过Context提供的getSystemService获取系统的服务,比如INPUT_SERVICE。
    这些service的返回值通常是xxxManger 例如InputManager. 在这个xxxManager类中,调用了ServiceManager.getService。因此这个xxxManager就是在进一步的封装,提供了一些相关的方法。但是这些方法,最终还是通过获取的那个service,通过binder,调用系统提供的服务来实现的。
  • 在系统和应用之前进行传递的是通过binder方式来进行的。在java层层面试下binder机制,需要使用AIDL。在aidl中定义接口,并在系统服务中实现这些接口,通常是实现接口的内部类stub。 并在SystemService中添加到系统中。

这篇关于android硬件访问服务程序框架和简单实例分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

前端原生js实现拖拽排课效果实例

《前端原生js实现拖拽排课效果实例》:本文主要介绍如何实现一个简单的课程表拖拽功能,通过HTML、CSS和JavaScript的配合,我们实现了课程项的拖拽、放置和显示功能,文中通过实例代码介绍的... 目录1. 效果展示2. 效果分析2.1 关键点2.2 实现方法3. 代码实现3.1 html部分3.2

0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型的操作流程

《0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeekR1模型的操作流程》DeepSeekR1模型凭借其强大的自然语言处理能力,在未来具有广阔的应用前景,有望在多个领域发... 目录0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型,3步搞定一个应

redis群集简单部署过程

《redis群集简单部署过程》文章介绍了Redis,一个高性能的键值存储系统,其支持多种数据结构和命令,它还讨论了Redis的服务器端架构、数据存储和获取、协议和命令、高可用性方案、缓存机制以及监控和... 目录Redis介绍1. 基本概念2. 服务器端3. 存储和获取数据4. 协议和命令5. 高可用性6.

Android里面的Service种类以及启动方式

《Android里面的Service种类以及启动方式》Android中的Service分为前台服务和后台服务,前台服务需要亮身份牌并显示通知,后台服务则有启动方式选择,包括startService和b... 目录一句话总结:一、Service 的两种类型:1. 前台服务(必须亮身份牌)2. 后台服务(偷偷干

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

修改若依框架Token的过期时间问题

《修改若依框架Token的过期时间问题》本文介绍了如何修改若依框架中Token的过期时间,通过修改`application.yml`文件中的配置来实现,默认单位为分钟,希望此经验对大家有所帮助,也欢迎... 目录修改若依框架Token的过期时间修改Token的过期时间关闭Token的过期时js间总结修改若依

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav