在将vsomeip移植到Android上时,我遇到了dynamic_cast问题。
为了简化情况,我创建了一个相当简单的测试app/lib结构,如下所示:
它包含一个库和一个应用程序:
动态cast在x86和其他arm平台(例如:使用linaro )上工作得很好;但是在Android上不能工作,它总是得到nullptr。
AOSP的环境是:
以下是我的一些参考资料,但都不起作用:
有人有主意吗?非常感谢~~
完整的可运行的src可以从:Y247byBfl6JaTtjR6tYn/view?usp=共享下载。
还有粘贴在这里:
//Android.bp:可以由mm直接构建
libbase_1_srcs = [
"src_1.cpp",
"src_2.cpp",
"src_3.cpp",
"lib_1.cpp",
]
main_srcs = [
"main.cpp",
"src_1.cpp",
"src_2.cpp",
]
cc_defaults {
name: "test_cast_defaults",
cppflags: [
"-std=c++11",
"-frtti",
]
}
cc_library_shared {
name: "libbase_1",
vendor: true,
srcs: libbase_1_srcs,
defaults: [
"test_cast_defaults"
],
rtti: true,
}
cc_binary {
name: "test_cast",
srcs: main_srcs,
vendor: true,
defaults: [
"test_cast_defaults"
],
shared_libs: [
],
rtti: true,
}
//CMakeLists.txt:测试X86或其他
cmake_minimum_required(VERSION 3.1)
set(project_name "test_cast")
project(${project_name})
add_library(base_1 SHARED lib_1.cpp src_1.cpp src_2.cpp src_3.cpp)
target_include_directories(base_1 PUBLIC ${${project_name}_SOURCE_DIR}/)
add_executable(test_cast main.cpp src_1.cpp src_2.cpp)
target_include_directories(test_cast PUBLIC ${${project_name}_SOURCE_DIR}/)
target_link_libraries(test_cast PUBLIC dl)
//main.cpp
#include <dlfcn.h>
#include <iostream>
#include "lib_1.h"
#include <memory>
#define libname "libbase_1.so"
#define funcname "get_plugin"
int main(void)
{
void *handle = dlopen(libname, RTLD_LAZY | RTLD_GLOBAL);
void *func;
if (handle == nullptr) {
std::cout << "Can not find: " << libname << std::endl;
return -1;
}
func = dlsym(handle, funcname);
const char *dlsym_error = dlerror();
if (dlsym_error) {
std::cout << "dlsym err: " << dlsym_error <<std::endl;
return -1;
}
plugin_init_func func_get = reinterpret_cast<plugin_init_func>(func);
get_plugin_func func_create = (*func_get)();
auto base_1 = (*func_create)();
auto base_2 = std::dynamic_pointer_cast<Base_2>(base_1);
if (base_2) {
std::cout << "dl: dynamic cast success!" << std::endl;
} else {
std::cout << "dl: dynamic cast failed!" << std::endl;
}
return 0;
}
//lib_1.cpp
#include "lib_1.h"
get_plugin_func get_plugin(void)
{
return Final::get;
}
//lib_1.h
#ifndef __LIB_1_H__
#define __LIB_1_H__
#include "header_3.h"
typedef std::shared_ptr<Base_1> (*get_plugin_func)();
typedef get_plugin_func (*plugin_init_func)();
extern "C" {
get_plugin_func get_plugin(void);
};
#endif
//报头_1.h
#ifndef __HEADER_1__
#define __HEADER_1__
class Base_1 {
public:
virtual ~Base_1();
};
#endif
//报头_2.h
#ifndef __HEADER_2__
#define __HEADER_2__
class Base_2 {
public:
virtual ~Base_2();
};
#endif
//报头_3.h
#ifndef __HEADER_3__
#define __HEADER_3__
#include "header_1.h"
#include "header_2.h"
#include <iostream>
#include <memory>
class Final : public Base_2,
public Base_1 {
public:
static std::shared_ptr<Base_1> get(void) {
std::shared_ptr<Base_1> base_1 = std::make_shared<Final>();
auto base_2 = std::dynamic_pointer_cast<Base_2>(base_1);
if (base_2) {
std::cout << "in lib: dynamic cast success!" << std::endl;
} else {
std::cout << "in lib: dynamic cast failed!" << std::endl;
}
return base_1;
}
~Final();
};
#endif
//src_1.cpp
#include <header_1.h>
#include <iostream>
Base_1::~Base_1()
{
std::cout << "in: " << __func__ << std::endl;
}
//src_2 src
#include "header_2.h"
#include <iostream>
Base_2::~Base_2()
{
std::cout << "in: " << __func__ << std::endl;
}
//src_3 src
#include "header_3.h"
#include <iostream>
Final::~Final() {
std::cout << "in: " << __func__ << std::endl;
}
成功运行(X86):
in lib: dynamic cast success!
dl: dynamic cast success!
in: ~Final
in: ~Base_1
in: ~Base_2
失败运行(Android):
in lib: dynamic cast success!
dl: dynamic cast failed!
in: ~Final
in: ~Base_1
in: ~Base_2
发布于 2020-10-26 23:21:48
当我通过vsomeip与clang编译器一起使用CommonAPI时,我也遇到了同样的问题,只有变通办法才能帮助我修复它。这里有更多详细信息:https://github.com/nkh-lab/genivi-capi-someip-examples/blob/ndk/CMakeLists.txt
问题是,vsomeip_v3::configuration_plugin
在libvsomeip3.so
和libvsomeip3-cfg.so
中有相同的类型信息,而dynamic_pointer_cast
使用错误的类型(例如,typeinfo表中的第一个条目)。因此,我们需要更改上述库的加载顺序,然后dynamic_pointer_cast
将使用正确的库。
发布于 2020-10-23 09:17:15
(这不是一个实际的解决方案,只是一个大的评论)
我认为不应该在主可执行文件和共享库中包含相同的对象模块。以下是避免这种情况的可能方法(Makefile):
CXXFLAGS += -g -W -Wall -Wextra -fPIC
LDFLAGS += -g
clean:
rm *.o *.so main 2>/dev/null || true
main: LDFLAGS += -export-dynamic
main: LDLIBS += -ldl
main: main.o src_1.o src_2.o src_3.o
${CXX} ${LDFLAGS} -o $@ $^ ${LDLIBS}
libbase_1.so: LDFLAGS += -shared
libbase_1.so: lib_1.o
${CXX} ${LDFLAGS} -o $@ $^ ${LDLIBS}
https://stackoverflow.com/questions/64494220
复制相似问题