[AOSP] [JNI] [Android] AOSP中使用JNI

一. 简要

在Android中,JNI主要用于实现一些性能较高的功能,如图像处理、音频处理、视频处理等。同时,JNI也可以用于实现一些特殊的功能,如与硬件交互、与系统服务交互等。

二. Java层

在某个对象中添加如下代码,例如我在/frameworks/base/services/core/java/com/android/server/keepalive/KeepAliveManagerService.java中去使用native方法

public class KeepAliveManagerService extends IKeepAliveManager.Stub {private static final String TAG = "KeepAliveManagerService leilei";private final Context mContext;private final ActivityManagerService mActivityManagerService;private WindowManagerService mWindowManagerService;private boolean mSystemReady = false;**public static native int resumeNative(int v);****public static native int pauseNative(int v);****public static native String stopNative(String v);**// 应用保活方法,简单的调用了三个native方法:resumeNative,pauseNative,stopNative@Overridepublic boolean keepAliveApplicationByPackage(String packageName) throws RemoteException {Log.d(TAG, "keepAliveApplicationByPackage.packageName:"+packageName);if (TextUtils.isEmpty(packageName) || mActivityManagerService == null|| mContext == null || !mSystemReady){return false;}int s = resumeNative(1);int s1 = pauseNative(2);String s2 = stopNative("leilei");Log.d(TAG, "leilei keepAliveApplicationByPackage: onResumeNative:"+s);Log.d(TAG, "leilei keepAliveApplicationByPackage: onPauseNative:"+s1);Log.d(TAG, "leilei keepAliveApplicationByPackage: stopNative:"+s2);int curCallingUid = Binder.getCallingUid();return keepAliveApplicationByPackage(packageName,curCallingUid);}
}

至此java层的代码就写好了

三. C/C++层

  • JNI文件创建

    因为我写jni方法是需要在我的service对象里使用,所以我frameworks/base/services/core/jni/目录下创建为需要的文件,如下代码所示

    #include <string>
    #include <dlfcn.h>
    #include <pthread.h>
    #include <chrono>
    #include <thread>
    #include <jni.h>
    #include <nativehelper/JNIHelp.h>
    #include <android/binder_manager.h>
    #include <android/binder_stability.h>
    #include <android/hidl/manager/1.2/IServiceManager.h>
    #include <binder/IServiceManager.h>
    #include <hidl/HidlTransportSupport.h>
    #include <incremental_service.h>
    #include <memtrackproxy/MemtrackProxy.h>
    #include <schedulerservice/SchedulingPolicyService.h>
    #include <sensorservicehidl/SensorManager.h>
    #include <stats/StatsAidl.h>
    #include <stats/StatsHal.h>
    #include <bionic/malloc.h>
    #include <bionic/reserved_signals.h>
    #include <android-base/properties.h>
    #include <utils/Log.h>
    #include <utils/misc.h>
    #include <utils/AndroidThreads.h>
    #ifdef LOG_TAG
    #undef LOG_TAG
    #define LOG_TAG "leilei"
    #endif// jni静态注册
    /*extern "C" jstring
    Java_com_android_server_keepalive_KeepAliveManagerService_onResumeNative(JNIEnv *env, jclass thiz, jlong value) {// 进行本地处理,生成返回值std::string hello = "Hello from C++";jstring result = env->NewStringUTF(hello.c_str());return result;
    }*/// jni动态注册
    namespace android {
    static jint pauseNative(JNIEnv* env, jobject thiz, jint value){ALOGD("The leilei message is onPauseNative %d:",value);return value;
    }static jint resumeNative(JNIEnv *env, jobject thiz, jint value){ALOGD("The leilei message is onResumeNative %d:",value);return value;
    }static jstring stopNative(JNIEnv *env, jobject thiz,jstring value){const char* ptr = env->GetStringUTFChars(value, NULL);ALOGD("The leilei message is stopNative %s:",ptr);return value;
    }// 对应native的方法注册表
    static const JNINativeMethod gKeepAliveManagerMethods[] = {/* name, signature, funcPtr */{"pauseNative","(I)I",(void *)pauseNative},{"resumeNative","(I)I",(void *)resumeNative},{"stopNative","(Ljava/lang/String;)Ljava/lang/String;",(void*) stopNative},
    };// 自己实现一个跟jniRegisterNativeMethods一样的功能
    int registerNativeMethods(JNIEnv *env,std::string name,const JNINativeMethod *methods) {// 反射拿到java对象jclass klass = env->FindClass(name.c_str());if (klass == NULL) {return -1;}// 第一个参数:反射拿到的对象// 第二个参数:类中的native方法--注册表// 第三个参数:native方法对象的个数return env->RegisterNatives(klass, methods,3);
    }// onload.cpp中调用了JNI_OnLoad,然后调用了register_android_server_KeepAliveManager进行注册
    // jniRegisterNativeMethods对RegisterNatives封装了,所以可以很方便的使用,我们手动来实现一下
    // JNI_OnLoad是jni.h中的对象,只有调用JNI_OnLoad和RegisterNatives才是动态注册
    int register_android_server_KeepAliveManager(JNIEnv* env) {// return jniRegisterNativeMethods(env, "com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods, NELEM(gKeepAliveManagerMethods));return registerNativeMethods(env,"com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods);
    }
    };
    

    文件名称必须规范:com.android.server.keepalive.KeepAliveManagerService.cpp,由包名+类名组成,然后实现对应上层的native方法即可,方法名称最好相同(也可以不同,只要在注册函数的第一个参数中对应起来就行),例如下方法,三个参数中,前两个参数必须有,而且不能变—>JNIEnv和jobject:

    static jint pauseNative(JNIEnv* env, jobject thiz, jint value){...
    }
    

    然后就是开始动态注册jni方法,如下代码所示,由于aosp已经封装好了jniRegisterNativeMethods方法可以直接使用来注册jni方法,但是为了更深刻的理解,我们手动来实现registerNativeMethods

    // native方法注册表
    static const JNINativeMethod gKeepAliveManagerMethods[] = {/* name, signature, funcPtr */{"pauseNative","(I)I",(void *)pauseNative},{"resumeNative","(I)I",(void *)resumeNative},{"stopNative","(Ljava/lang/String;)Ljava/lang/String;",(void*) stopNative},
    };// 自己实现一个跟register_android_server_KeepAliveManager一样的功能
    int registerNativeMethods(JNIEnv *env,std::string name,const JNINativeMethod *methods) {jclass klass = env->FindClass(name.c_str());if (klass == NULL) {return -1;}// 第一个参数:反射拿到的对象// 第二个参数:类中的native方法--注册表// 第三个参数:native方法对象的个数return env->RegisterNatives(klass, methods,3);
    }int register_android_server_KeepAliveManager(JNIEnv* env) {// return jniRegisterNativeMethods(env, "com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods, NELEM(gKeepAliveManagerMethods));return registerNativeMethods(env,"com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods);
    }
    

    先实现一个native方法注册表,代表需要对应java层native方法,返回的对象是JNINativeMethod,该对象属于jni.h里的,结构如下

    typedef struct {const char* name;const char* signature;void*       fnPtr;
    } JNINativeMethod;
    

    第一个参数对应了java native方法的名称,第二个参数代表native方法里面的参数和返回值,第三个参数代表jni方法

    回到registerNativeMethods方法,主要就是通过env的FindClass反射获取Java对象,然后通过RegisterNatives(klass, methods,3);进行注册即可,第二个参数就是native方法gKeepAliveManagerMethods注册表

    **思考一下:**既然需要注册jni,那么调用register_android_server_KeepAliveManager函数的注册的入口又在哪里?下文分析

  • JNI文件引入和注册流程

    • android.bp引入编译

      需要让我们的jni文件参与编译,需要在frameworks/base/services/core/jni/Android.bp中添加该文件,如下所示

      cc_library_static {name: "libservices.core",defaults: ["libservices.core-libs"],cpp_std: "c++2a",cflags: ["-Wall","-Werror","-Wno-unused-parameter","-Wthread-safety","-DEGL_EGLEXT_PROTOTYPES","-DGL_GLEXT_PROTOTYPES",],srcs: ["BroadcastRadio/JavaRef.cpp","BroadcastRadio/NativeCallbackThread.cpp","BroadcastRadio/BroadcastRadioService.cpp","BroadcastRadio/Tuner.cpp","BroadcastRadio/TunerCallback.cpp","BroadcastRadio/convert.cpp","BroadcastRadio/regions.cpp","gnss/GnssConfiguration.cpp","gnss/GnssMeasurement.cpp","gnss/GnssMeasurementCallback.cpp","gnss/Utils.cpp","stats/SurfaceFlingerPuller.cpp",**"com.android.server.keepalive.KeepAliveManagerService.cpp",**include_dirs: ["frameworks/base/libs","frameworks/native/services","system/gatekeeper/include","system/memory/libmeminfo/include",],header_libs: ["bionic_libc_platform_headers",],
      }
      

      在此模块添加**“com.android.server.keepalive.KeepAliveManagerService.cpp”**,即可,模块名为libservices.core,会生成对应的so库

    • JNI注册入口声明

      在frameworks中,上文分析了如何调用jni注册native方法,但是调用注册的入口在哪里?就是通过frameworks/base/services/core/jni/onload.cpp文件进行调用的,需要在此文件中声明我们的注册入口,如下代码所示

      namespace android {
      int register_android_server_BatteryStatsService(JNIEnv* env);
      int register_android_server_ConsumerIrService(JNIEnv *env);
      int register_android_server_InputManager(JNIEnv* env);
      **int register_android_server_KeepAliveManager(JNIEnv* env);**
      int register_android_server_LightsService(JNIEnv* env);
      int register_android_server_PowerManagerService(JNIEnv* env);
      int register_android_server_PowerStatsService(JNIEnv* env);
      int register_android_server_HintManagerService(JNIEnv* env);
      int register_android_server_storage_AppFuse(JNIEnv* env);
      }
      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_broadcastradio_BroadcastRadioService(env);register_android_server_broadcastradio_Tuner(vm, env);register_android_server_PowerManagerService(env);register_android_server_PowerStatsService(env);register_android_server_HintManagerService(env);register_android_server_SerialService(env);register_android_server_InputManager(env);**register_android_server_KeepAliveManager(env);}**
      }
      

      只需要在namespace android中声明注册入口函数**register_android_server_KeepAliveManager**,此函数在我们创建的jni文件中会实现。

      然后在JNI_OnLoad函数中添加**register_android_server_KeepAliveManager(env);**,目的是为了方法可以被正确调用,以及传递了env对象(jni里的东西),再来看一遍我创建的jni文件,frameworks/base/services/core/jni/com.android.server.keepalive.KeepAliveManagerService.cpp

      ...
      int **register_android_server_KeepAliveManager**(JNIEnv* env) {// return jniRegisterNativeMethods(env, "com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods, NELEM(gKeepAliveManagerMethods));return registerNativeMethods(env,"com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods);
      }
      };
      

      **思考一下:**为什么需要在extern "C" jint JNI_OnLoad(JavaVM* vm, void* /* reserved */)方法中调用jni注册逻辑呢?这涉及到jni动态注册原理了,后面再分析,jni分为静态注册和动态注册

    • JNI注册表分析

      Andoird 中使用了一种不同传统Java JNI的方式来定义其native的函数。其中很重要区别是Andorid使用了一种Java 和 C 函数的映射表数组,并在其中描述了函数的参数和返回值。这个数组的类型是JNINativeMethod——该结构体位于jni.h中,结构如下

      typedef struct {const char* name;const char* signature;void*       fnPtr;
      } JNINativeMethod;
      

      三个参数代表着:native方法名称,签名—用字符串是描述了Java中函数的参数和返回值,jni函数对象-指向了java的native方法

      具体用法如下

      static const JNINativeMethod gKeepAliveManagerMethods[] = {/* name, signature, funcPtr */{"pauseNative","(I)I",(void *)pauseNative},{"resumeNative","(I)I",(void *)resumeNative},{"stopNative","(Ljava/lang/String;)Ljava/lang/String;",(void*) stopNative},
      };
      

      第三个参数前面必须带有(void *),这里主要分析第二个参数,()代表native方法的参数,()外面部分代表着返回值,I代表着java的int,jni的jint,具体如下

      字符c/c++类型Java类型
      Vvoidvoid
      Zjbooleanboolean
      Ijintint
      Jjlonglong
      Djdoubledouble
      Fjfloatfloat
      Bjbytebyte
      Cjcharchar
      Sjshortshort

      以上都是基本数据类,如果是数组,则用[代表,如整型数值 [I来表示,具体如下

      名称c/c++类型Java类型
      [IjintArrayint[]
      [FjfloatArrayfloat[]
      [BjbyteArraybyte[]
      [CjcharArraychar[]
      [SjshortArrayshort[]
      [DjdoubleArraydouble[]
      [JjlongArraylong[]
      [ZjbooleanArrayboolean[]

      那如果native参数中是对象呢,需要用如下方法表示—参数解释:

      // 参数解释
      "()" 中的字符表示参数,小括号后面的则代表返回值。
      "()V" 就表示native void Fun();
      "(II)V" 表示native void Func(int a, int b);参数是俩个整型。
      "(Ljava/lang/String;)Ljava/lang/String;" 就表示native Sting Func(String value);
      

      所以如果要用对象作为参数或者返回值,需要在前面加个”L”,中间是用”/" 隔开,后面跟包名和类名,以及分号即可。如果是对象数组,则在前面加个[即可

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

[强网杯 2022]factor有感

可直接私信&#xff0b;Q 3431550587 此题记录主要是他运用了几个新看见的攻击思路和拜读了一篇论文&#xff0c;所以写写。题目源码&#xff1a; #encoding:utf-8 from Crypto.Util.number import * from gmpy2 import * from random import randint from flag import flagd…

uni-app:showModal中实现弹窗中文本框输入

效果 代码 <template><view><!-- 显示弹窗的按钮 --><button click"showInputDialog">显示弹窗</button></view> </template><script> export default {methods: {showInputDialog() {uni.showModal({title: 请完成…

电影大师杂记

假期集中刷了好多书&#xff0c;游戏和电影&#xff0c;在虚拟世界里猛烈的各种闲逛&#xff0c;cyberpunk 2077到blade runner&#xff0c;到异形&#xff0c;到终结者&#xff0c;到星球大战&环太平洋&#xff0c;到工业光魔&#xff0c;还有各种编程的书。。。 hmmm&…

Spring的AOP开发-基于xml配置的AOP

目录 基于xml配置的AOP xml方式AOP快速入门 xml方式AOP配置详解 xml方式AOP原理剖析&#xff08;后面再深入理解一下&#xff09; AOP底层两种生成Proxy的方法 基于xml配置的AOP xml方式AOP快速入门 在前面我们自己编写的AOP基础代码还存在一些问题&#xff0c;主要是 被…

【数据库——MySQL】(15)存储过程、存储函数和事务处理习题及讲解

目录 1. 题目1.1 存储过程1.2 存储函数1.3 事务处理 2. 解答2.1 存储过程2.2 存储函数2.3 事务处理 1. 题目 1.1 存储过程 创建表 RandNumber &#xff1a;字段&#xff1a;id 自增长&#xff0c; data int&#xff1b; 创建存储过程向表中插入指定个数的随机数&#xff08;1-…

【软件测试】自动化测试selenium(二)

文章目录 三. 掌握Selenium常用的API使用1. webdriver API2. 操作测试对象3. 添加等待4. 打印信息5. 浏览器的操作6. 键盘事件7. 鼠标事件8. 定位一组元素9. 多层框架/窗口定位10. 下拉框处理11. 弹窗处理12. 上传文件13. 关闭浏览器14. 切换窗口15. 截图操作 三. 掌握Selenium…

基于electron25+vite4创建多窗口|vue3+electron25新开模态窗体

在写这篇文章的时候&#xff0c;查看了下electron最新稳定版本由几天前24.4.0升级到了25了&#xff0c;不得不说electron团队迭代速度之快&#xff01; 前几天有分享一篇electron24整合vite4全家桶技术构建桌面端vue3应用示例程序。 https://www.cnblogs.com/xiaoyan2017/p/17…

【雷达原理】雷达测距原理及实现方法

目录 一、雷达测距原理1.1 基本原理1.2 实现方法1.3 与距离有关的概念 二、MATLAB仿真实验2.1 应用案例2.2 MATLAB代码 一、雷达测距原理 1.1 基本原理 我们知道&#xff0c;电磁波的传播速度为光速 c c c&#xff0c;若雷达与目标之间的距离为 R R R&#xff0c;则雷达发出…

QT实现tcp服务器客户端

服务器.cpp #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//实例化一个服务器server new QTcpServer(this);// 此时&#xff0c;服务器已经成功进入监听状态…

LVGL_基础控件滑动条slider

LVGL_基础控件滑动条slider 1、创建滑动条控件 // 创建一个 slider 组件(对象)&#xff0c;他的父对象是活动屏幕对象 lv_obj_t *slider lv_slider_create(lv_scr_act()); LV_LOG_USER("lv_slider_get_value(slider) %d", lv_slider_get_value(slider));/* 设置位…

全屋灯具选购指南,如何选择合适的灯具。福州中宅装饰,福州装修

灯具装修指南 灯具就像我们家里的星星&#xff0c;在黑暗中带给我们明亮&#xff0c;可是灯具如果选择的不好&#xff0c;这个效果不仅体现不出来&#xff0c;还会让人觉得烦躁。 灯具到底该怎么选呢&#xff1f;装修灯具有哪些注意事项呢&#xff1f;给大家做了一个总结&#…

基于springboot的论坛网站

目录 前言 一、技术栈 二、系统功能介绍 用户信息管理 普通管理员管理 交流论坛 交流论坛评论 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了…

C语言qsort函数

排序qsort int int cmp(const void *a, const void *b) {return *(int *)a - *(int *)b;//先强转成int型&#xff0c;后解引用取值比较大小 }字符串数组 char a[] “hello world” //字符串数组&#xff0c;存放的是字符 int cmp(const void *a, const void *b) {return *(…

罗彻斯特大学探讨ChatGPT等人工智能将如何影响高等教育

人工智能聊天机器人ChatGPT持续引起互联网用户的热议&#xff0c;它能够回答关于各个领域的问题&#xff0c;创作歌曲、食谱&#xff0c;起草电子邮件等等。罗切斯特的教职员工和管理人员就他们如何处理 ChatGPT 以及它如何影响未来的教学和学习提出了他们的想法。 “让这项技…

快排(三种单趟排序法,递归非递归算法)

快排发明者:霍尔 (Sir Charles Antony Richard Hoare) 是一位英国计算机科学家。 计算机领域的爵士——托尼霍尔(Tony Hoare)(1934年1月11日出生),英文全称Sir Charles Antony Richard Hoare,常被称为Tony Hoare或者C. A. R. Hoare,1959年博士毕业于

idea清空缓存类

解决办法 网上有很多是让你去清空什么maven依赖&#xff0c;但假如这个项目是你不可以大刀阔斧的话 可以清空idea缓存 选择 Invalidate 开头的 然后全选 运行重启idea OK

FISCO BCOS(三十七)———FISCOBCOS应用开发,交易hash、区块高度的获取

这个需求怎么做? 交易hash的获取方式有很多,这里先介绍一种方式。 根据块高查询区块信息 https://webasedoc.readthedocs.io/zh_CN/latest/docs/WeBASE-Front/interface.html那我们如何知道现在的区块高度是多少? https://webasedoc.readthedocs.io/zh_CN/latest/docs/W…

VSCode Intellij IDEA CE 数据库连接

VSCode & Intellij IDEA CE 数据库连接 大概记一下现在正在用的几个工具/插件 VSCode VSCode 里面的工具我下载了很多&#xff0c;如果只是链接 MySQL 的话&#xff0c;可能用 Jun Han 这位大佬的 MySQL 就好了&#xff1a; 使用这个插件直接打开 .sql 文件单击运行就能…

nodejs+vue临沂特色产品销售平台elementui

从实际工作出发&#xff0c;对过去的临沂特色产品销售平台存在的问题进行分析&#xff0c;完善用户的使用体会。采用计算机系统来管理信息 提高了工作的效率。 随着信息化社会的形成和微电子技术日新月异的发展&#xff0c;临沂特色产品销售平台是针对目前临沂特色产品销售…

win10、win11彻底永久关闭自动更新的方法

win10、win11彻底永久关闭自动更新的方法 前言彻底关闭自动更新方法步骤一、禁用Windows Update服务二、在组策略里关闭Win10自动更新相关服务四、在注册表中关闭Win10自动更新 完结 前言 win系统的自动更新可谓是非常顽固&#xff0c;很多用户在网上试了各种关闭win系统自动更…