繁体   English   中英

当动态库使用静态库的符号时,请从静态库中打开一个动态库

[英]dlopen a dynamic library from a static library, when the dynamic library uses symbols of the static one

这个问题与dlopen静态库linux C ++中的动态库密切相关,但存在进一步的复杂性(并使用C ++代替C):

我有一个链接到静态库(.a)的应用程序,该库使用dlopen函数加载动态库(.so)。 另外,动态库调用在静态库中定义的函数。

有没有一种方法可以在不将动态库与静态库链接的情况下进行编译,反之亦然?

这是到目前为止我尝试过的内容,对相关问题的示例进行了一些修改:

app.cpp:

#include "staticlib.hpp"
#include <iostream>
int main()
{
    std::cout << "and the magic number is: " << doSomethingDynamicish() << std::endl;
    return 0;
}

staticlib.hpp:

#ifndef __STATICLIB_H__
#define __STATICLIB_H__

int doSomethingDynamicish();
int doSomethingBoring();
#endif

staticlib.cpp:

#include "staticlib.hpp"
#include "dlfcn.h"
#include <iostream>
int doSomethingDynamicish()
{
    void* handle = dlopen("./libdynlib.so",RTLD_NOW);
    if(!handle)
    {
        std::cout << "could not dlopen: " << dlerror() << std::endl;
        return 0;
    }

    typedef int(*dynamicfnc)();
    dynamicfnc func = (dynamicfnc)dlsym(handle,"GetMeANumber");
    const char* err = dlerror();
    if(err)
    {
        std::cout << "could not dlsym: " <<err << std::endl;
        return 0;
    }
    return func();
}

staticlib2.cpp:

#include "staticlib.hpp"
#include "dlfcn.h"
#include <iostream>

int doSomethingBoring()

{
    std::cout << "This function is so boring." << std::endl;
    return 0;

}

dynlib.cpp:

#include "staticlib.hpp"

extern "C" int GetMeANumber()
{
    doSomethingBoring();
    return 1337;
}

并建立:

g++ -c -o staticlib.o staticlib.cpp
g++ -c -o staticlib2.o staticlib2.cpp
ar rv libstaticlib.a staticlib.o staticlib2.o
ranlib libstaticlib.a
g++ -rdynamic -o app app.cpp libstaticlib.a -ldl 
g++ -fPIC -shared -o libdynlib.so dynlib.cpp

当我用./app运行它时

could not dlopen: ./libdynlib.so: undefined symbol: _Z17doSomethingBoringv
and the magic number is: 0

dlopen手册页

如果可执行文件与标志“ -rdynamic”(或同义,“-export-dynamic”)链接,则可执行文件中的全局符号也将用于解析动态加载的库中的引用。

这意味着要使应用程序导出其符号以供在动态库中使用,您必须将应用程序与-rdynamic标志-rdynamic


除了上述问题之外,还有另一个问题与静态库有关:问题在于,由于在主程序中未调用doSomethingBoring函数,因此未链接静态库中的目标文件staticlib2.o 。 。

答案可以在这个旧问题中找到, 该问题告诉您添加--whole-archive链接器标志:

g++ -rdynamic -o app app.cpp -L. \
    -Wl,--whole-archive -lstaticlib \
    -Wl,--no-whole-archive -ldl

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM