jni注册流程

作者: Liuqc | 来源:发表于2019-12-19 21:36 被阅读0次
    System.loadLibrary("")
    
     ->Runtime.getRuntime().loadLibrary0(VMStack.getCallingClassLoader(), libname)
     
    ->private static native String nativeLoad(String filename, ClassLoader loader, String librarySearchPath); 
    
    ->Runtime.c
     Runtime_nativeLoad(JNIEnv* env, jclass ignored, jstring javaFilename, jobject javaLoader, jstring javaLibrarySearchPath)
     
    ->Opeanjdkjvm.c 
    JVM_NativeLoad(env, javaFilename, javaLoader, javaLibrarySearchPath) 
    
    ->java_vm_ext.c
    bool LoadNativeLibrary(JNIEnv* env, const std::string& path, jobject   class_loader, jstring library_path, std::string* error_msg){ 
    
      //最终会调用
      dlopen void* handle android::OpenNativeLibrary(env, runtime_-    >GetTargetSdkVersion(), path_str, class_loader, library_path); 
    
      //会调用library 是JniInvocation dlsym(handle, "JNI_Onload") 
      library->FindSymbol("JNI_OnLoad", nullptr); 
    
      JNI_OnLoadFn jni_on_load = reinterpret_cast<jni_onloadfn>(sym);
     
      int version = (*jni_on_load)(this, nullptr);
     }
    

    打开动态链接库

    void* handle=dlopen(path, RTLD_NOW)

    dlopen()是一个计算机函数,功能是以指定模式打开指定的动态链接库文件,并返回一个句柄给dlsym()的调用进程。使用dlclose()来卸载打开的库。

    功能:打开一个动态链接库,并返回动态链接库的句柄

    包含头文件:

    include <dlfcn.h>

    函数定义:

    void * dlopen( const char * pathname, int mode);

    函数描述:

    mode是打开方式,其值有多个,不同操作系统上实现的功能有所不同,在linux下,按功能可分为三类:

    1、解析方式

    RTLD_LAZY:在dlopen返回前,对于动态库中的未定义的符号不执行解析(只对函数引用有效,对于变量引用总是立即解析)。

    RTLD_NOW: 需要在dlopen返回前,解析出所有未定义符号,如果解析不出来,在dlopen会返回NULL,错误为:: undefined symbol: xxxx.......

    2、作用范围,可与解析方式通过“|”组合使用。

    RTLD_GLOBAL:动态库中定义的符号可被其后打开的其它库解析。

    RTLD_LOCAL: 与RTLD_GLOBAL作用相反,动态库中定义的符号不能被其后打开的其它库重定位。如果没有指明是RTLD_GLOBAL还是RTLD_LOCAL,则缺省为RTLD_LOCAL。

    3、作用方式

    RTLD_NODELETE: 在dlclose()期间不卸载库,并且在以后使用dlopen()重新加载库时不初始化库中的静态变量。这个flag不是POSIX-2001标准。

    RTLD_NOLOAD: 不加载库。可用于测试库是否已加载(dlopen()返回NULL说明未加载,否则说明已加载),也可用于改变已加载库的flag,如:先前加载库的flag为RTLD_LOCAL,用dlopen(RTLD_NOLOAD|RTLD_GLOBAL)后flag将变成RTLD_GLOBAL。这个flag不是POSIX-2001标准。

    RTLD_DEEPBIND:在搜索全局符号前先搜索库内的符号,避免同名符号的冲突。这个flag不是POSIX-2001标准。

    返回值:

    打开错误返回NULL

    成功,返回库引用

    编译时候要加入 -ldl (指定dl库) -rdynamic(通知链接器将所有符号添加到动态符号表中(目的是能够通过使用 dlopen 来实现向后跟踪)

    例如

    gcc test.c -o test -ldl-rdynamic

    dlsym是一个计算机函数,功能是根据动态链接库操作句柄与符号,返回符号对应的地址,不但可以获取函数地址,也可以获取变量地址

    包含头文件:

    include<dlfcn.h>

    函数定义:

    voiddlsym(voidhandle,constchar*symbol)

    函数描述:

    dlsym(dynamic library symbol)

    根据 动态链接库 操作句柄(handle)与符号(symbol),返回符号对应的地址。使用这个函数不但可以获取函数地址,也可以获取变量地址。

    handle:由dlopen打开动态链接库后返回的指针

    symbol:要求获取的函数或全局变量的名称。

    返回值:

    void* 指向函数的地址,供调用使用。

    dlsym(handle, "JNI_Onload")

    //关闭动态链接库 dlclose(handle);</jni_onloadfn>

    相关文章

      网友评论

        本文标题:jni注册流程

        本文链接:https://www.haomeiwen.com/subject/npvvnctx.html