更改.so库中的函数名称

问题描述 投票:0回答:1

我只有旧项目的1 .so文件。如何在不在项目或模块中创建相同包的情况下使用此文件?

android android-ndk
1个回答
0
投票

实际上,您不需要在.so文件中更改函数名称。您可以使用dlopen在运行时动态加载.so库,并使用dlsym为您获取指针YOUR_FUNCTION_NAME(),然后通过指针调用YOUR_FUNCTION_NAME()。要在当前项目中执行此操作,您可以创建“包装器”,如下所示:

public class OldSoHelper {
    public static native void loadOldSo();
    public static native <TYPE_OF_RESULT> runFunFromOldSo(<PARAMETERS>);
    public static native void unloadOldSo();
}

并在当前项目的相应.c/.cpp文件中(例如默认情况下为native-lib.cpp):

void *handle;
<TYPE_OF_OLD_FUNCTION> (*old_fun_wrapper)(<PARAMETERS_OF_OLD_FUNCTION>);

extern "C"
JNIEXPORT void JNICALL
Java_<YOUR_PACKAGE_NAME>_OldSoHelper_loadOldSo(JNIEnv *env, jclass type) {
     handle = dlopen("<YOUR_OLD_SO>.so", RTLD_NOW);
     old_fun_wrapper = (<TYPE_OF_OLD_FUNCTION> (*)(<PARAMETERS_OF_OLD_FUNCTION>))(dlsym(handle, "<OLD_FUNCTION_NAME_e.g._Java_com_abc_dee_Native_appInit>"));
}

extern "C"
JNIEXPORT jobject JNICALL
Java_<YOUR_PACKAGE_NAME>_OldSoHelper_runFunFromOldSo(JNIEnv *env, jclass type,
                                                      <PARAMETERS_FOR_OLD_FUNCTION>)
{   
    jclass ResultClass = env->FindClass("YOUR/PACKAGE/NAME/RESULT_CLASS");
    jobject result = ...
    jfieldID fieldId = env->GetFieldID(ResultClass, "<FIELD_NAME>", "<FILED_TYPE_LETTER>");

    <TYPE_OF_OLD_FUNCTION> res = old_fun_wrapper(<PARAMETERS_FOR_OLD_FUNCTION>);

    env->Set<TYPE>Field(result, fieldId , res.filed);

    return result;
}

extern "C"
JNIEXPORT void JNICALL
Java_<YOUR_PACKAGE_NAME>_OldSoHelper_unloadOldSo(JNIEnv *env, jclass type) {
     if (handle) {
         dlclose(handle);
     }
}

从java代码中你可以调用:

...
// when you need old .so e.g. in onCreate()
OldSoHelper.loadOldSo();
...

// when you no need to call function from old .so
<TYPE_OF_RESULT> result = OldSoHelper.runFunFromOldSo(<PARAMETERS>);

...
// when you no need old .so e.g. in onDestroy()
OldSoHelper.unloadOldSo();
...
© www.soinside.com 2019 - 2024. All rights reserved.