cpr*_*leg 3 c++ linux ubuntu gcc boost
我正在编写一个应用程序,它将在 DDS 系统中扮演订阅者的角色。该订阅者将在收到数据后将值写入共享内存位置,以便其他 GUI 应用程序可以读取这些值。
我已经编写了一个在 Windows 上编译和运行 A-OK 的应用程序版本,但我现在面临着创建一个在 Ubuntu 上运行的版本。作为 Windows 本机,这对我来说很麻烦。
为了创建 Ubuntu 版本,我已将我的代码库移植到 Ubuntu VM,现在正尝试在那里编译它,这正是我遇到问题的地方。
在大多数情况下,一切都可以正常编译,并且该过程与 Windows 上的过程非常相似。不幸的是,链接我所有生成的 .o 文件的最后一条语句失败了。一些描述如下:
我正在使用 boost 进程间库来编写一个头文件,该文件将提供共享内存读/写操作。实现此目的的 .cpp 如下:MemWriterH.cpp
#define BOOST_DATE_TIME_NO_LIB
#define CPP
#include <boost/interprocess/shared_memory_object.hpp>
#include <boost/interprocess/mapped_region.hpp>
#include <stdio.h>
#include <MemWriterH.h>
using namespace boost::interprocess;
extern "C" int writeToSharedMemory(char *data)
{
//Remove shared memory on construction and destruction
struct shm_remove
{
shm_remove()
{
//shared_memory_object::remove("MySharedMemory");
}
~shm_remove()
{
//shared_memory_object::remove("MySharedMemory");
}
}remover;
//Create a shared memory object.
shared_memory_object shm(open_or_create, "MySharedMemory", read_write);
//Set size
shm.truncate(1000);
//Map the whole shared memory in this process
mapped_region region(shm, read_write);
//Write given data to the mapped region
//std::memset(region.get_address(), 'h', region.get_size());
std::memcpy(region.get_address(), data, region.get_size());
return 0;
}
extern "C" void readFromSharedMemeory(char toWriteIn[])
{
//Open already created shared memory object.
shared_memory_object shm(open_only, "MySharedMemory", read_only);
//Map the whole shared memory in this process
mapped_region region(shm, read_only);
//Read data from shared memory into parameter
//char *mem = static_cast<char*>(region.get_address());
std::memcpy(toWriteIn, region.get_address(), region.get_size());
printf("Read data OK");
printf("\n%s", toWriteIn);
}
Run Code Online (Sandbox Code Playgroud)
及其随附的标题:MemWriterH.h
#ifdef CPP
extern "C" int writeToSharedMemory(char *);
extern "C" void readFromSharedMemeory(char *);
#else
int writeToSharedMemory( char *);
void readFromSharedMemeory(char* toWriteIn);
#endif
Run Code Online (Sandbox Code Playgroud)
然后使用以下命令将它们编译为 .o 文件:
g++ -c -I/$HOME/workspace_c/HID_LDM_DDS/HID_LDM_DDS_INCLUDES/MEM_MAPPING/ *.cpp
Run Code Online (Sandbox Code Playgroud)
在应用程序的主文件中,我有一个 #include,它允许使用上述函数。
虽然此应用程序还有各种其他元素(如下所示),但这些元素似乎都可以很好地编译、构建和链接。
这个生成的MemWriterH.o文件然后用于最终的 GCC 语句 - 我相信 - 将所有 .o 文件链接在一起以生成应用程序的可执行文件。最后的声明是:
gcc HID_LDM_DDS_SUB.o $HOME/workspace_c/HID_LDM_DDS/HID_LDM_DDS_HELPER/HID_LDM_DDS_HELPER.o $HOME/workspace_c/HID_LDM_DDS/HID_LDM_DDS_INCLUDES/CheckStatus.o $HOME/workspace_c/HID_LDM_DDS/HID_LDM_DDS_INCLUDES/MEM_MAPPING/MemWriterH.o $HOME/workspace_c/HID_LDM_DDS/HID_LDM_DDS_IDL/LDMSacDcps.o $HOME/workspace_c/HID_LDM_DDS/HID_LDM_DDS_IDL/LDMSplDcps.o $OSPL_HOME/lib/libddskernel.so $OSPL_HOME/lib/libdcpssac.so
Run Code Online (Sandbox Code Playgroud)
不幸的是,这给了我错误:
/usr/bin/ld: /home/bob/workspace_c/HID_LDM_DDS/HID_LDM_DDS_INCLUDES/MEM_MAPPING/MemWriterH.o: 未定义对符号“shm_open@@GLIBC_2.2”的引用
/lib/i386-linux-gnu/librt.so.1:添加符号时出错:命令行中缺少 DSO collect2:错误:ld 返回 1 个退出状态
在这一点上,我感到完全迷失了。.cpp 编译为 .o 没有任何抱怨,所以我不太确定从这里还能去哪里。什么是 DSO,如何将其添加到命令行?
编辑:
现在更新问题解决了。正如Sehe在下面指出的那样,定义 shm_open 的库丢失了。这包含在 -lrt 标志中,该标志应包含在被链接的 .o 文件之后(有关更多说明,请参见此处)。
然而,解决这个问题确实导致了另一个问题,即编译器在多次出现错误时抱怨:“对 std::some_function 的未定义引用”。事实证明,这是由于使用 gcc 命令链接 C 和 C++ 代码(头文件)时出现问题。添加 -lstdc++ 标志或使用 g++ 编译命令(请参阅此处的答案以了解正在发生的事情)也可以解决此问题,并且所有编译都成功。
归档时间: |
|
查看次数: |
3848 次 |
最近记录: |