混合 C/C++ 代码中的结构指针

Posted

技术标签:

【中文标题】混合 C/C++ 代码中的结构指针【英文标题】:Struct pointers in mixed C / C++ code 【发布时间】:2014-12-11 08:03:23 【问题描述】:

我在从 C++ 调用 C 函数时遇到问题。特别是,我有一个具有 C 结构成员变量的类的实例。我将指向该变量的指针传递给一个 C 函数,该函数是从成员函数调用的。我观察到变量指针地址在 C 函数内部和外部都没有变化。但是,结构字段之一的地址实际上确实发生了变化。

代码大纲:

extern "C"

  #define ENAGLO
  #include <rtklib.h>


class GNSS

  protected: rtcm_t rtcm;   // A C struct type in rtklib.h.
  public: ~GNSS()  free_rtcm(&rtcm); 
  public: GNSS()  init_rtcm(&rtcm);  
  public: Load()  

    // Add some stuff to the rtcm.obs element

    // This is the problematic line. If one were to read the address
    // of rtcm.nbyte inside and outside of the function below, its
    // address is different! Thus, if the gen_rtcm3 function sets
    // rtcm->nbyte = 3, it does not persist back into C++.
    gen_rtcm3(&rtcm, 1001, 0);
  

rtklib.h 和源文件可在此处获得:RTKLIB

我在 Ubuntu 14.04 上使用 GCC 4.8.2。

更新:CMake 用于通过以下方式构建 RTKLIB 库和 C++ 代码。我查过了,如果去掉 -O3 似乎没有什么区别。

# Find all RTKLIB source files
set(RTKLIB_SOURCES 
  thirdparty/RTKLIB/src/ephemeris.c
  thirdparty/RTKLIB/src/ppp.c
  thirdparty/RTKLIB/src/preceph.c
  thirdparty/RTKLIB/src/rcvraw.c
  thirdparty/RTKLIB/src/rinex.c
  thirdparty/RTKLIB/src/sbas.c
  thirdparty/RTKLIB/src/rtkpos.c
  thirdparty/RTKLIB/src/datum.c
  thirdparty/RTKLIB/src/rtcm3.c
  thirdparty/RTKLIB/src/options.c
  thirdparty/RTKLIB/src/rtkcmn.c
  thirdparty/RTKLIB/src/rtcm2.c
  thirdparty/RTKLIB/src/stream.c
  thirdparty/RTKLIB/src/ionex.c
  thirdparty/RTKLIB/src/rcv
  thirdparty/RTKLIB/src/rcv/binex.c
  thirdparty/RTKLIB/src/rcv/rt17.c
  thirdparty/RTKLIB/src/rcv/gw10.c
  thirdparty/RTKLIB/src/rcv/nvs.c
  thirdparty/RTKLIB/src/rcv/ublox.c
  thirdparty/RTKLIB/src/rcv/septentrio.c
  thirdparty/RTKLIB/src/rcv/ss2.c
  thirdparty/RTKLIB/src/rcv/rcvlex.c
  thirdparty/RTKLIB/src/rcv/novatel.c
  thirdparty/RTKLIB/src/rcv/javad.c
  thirdparty/RTKLIB/src/rcv/skytraq.c
  thirdparty/RTKLIB/src/rcv/crescent.c
  thirdparty/RTKLIB/src/convkml.c
  thirdparty/RTKLIB/src/pntpos.c
  thirdparty/RTKLIB/src/rtcm.c
  thirdparty/RTKLIB/src/ppp_ar.c
  thirdparty/RTKLIB/src/geoid.c
  thirdparty/RTKLIB/src/rtcm3e.c
  thirdparty/RTKLIB/src/solution.c
  thirdparty/RTKLIB/src/streamsvr.c
  thirdparty/RTKLIB/src/rtklib.h
  thirdparty/RTKLIB/src/rtksvr.c
  thirdparty/RTKLIB/src/qzslex.c
  thirdparty/RTKLIB/src/lambda.c
  thirdparty/RTKLIB/src/tle.c
)

# Create a rtklib library
add_library(crates_gnss_rtklib $RTKLIB_SOURCES)
target_link_libraries(crates_gnss_rtklib    
  $LAPACK_LIBRARIES
  $BLAS_LIBRARIES
  $CMAKE_THREAD_LIBS_INIT
  $MATH_LIBRARY
  $RT_LIBRARY
)
SET_TARGET_PROPERTIES(crates_gnss_rtklib PROPERTIES COMPILE_FLAGS "-O3 -DTRACE -DENAGLO -DENAGAL -DENAQZS -DNFREQ=3")

# Crate a gazebo receiver
add_library(crates_gnss_plugin src/GNSS.cc)
target_link_libraries(crates_gnss_plugin    
  $catkin_LIBRARIES 
  $GAZEBO_LIBRARIES 
  crates_gnss_rtklib
)

# THIS IS THE LINE THAT WAS ULTIMATELY NEEDED!
#SET_TARGET_PROPERTIES(crates_gnss_plugin PROPERTIES COMPILE_FLAGS "-O3 -DTRACE -DENAGLO -DENAGAL -DENAQZS -DNFREQ=3")

【问题讨论】:

库是如何编译的? sizeof(rtcm_t) C++ 代码和 C 代码一样吗?我有一个理论,可能rtcm_t 的确切布局在您的 C++ 编译和 C 编译之间不一致(可能不同的填充、不同的成员大小或不同的结构定义(即 C lib 是版本 1,但 C++ 是使用版本 2 标头))。 打印出sizeof(rtcm_t)offsetof(rtcm_t, x)rtcm_t 中所有字段x,包括代码的C 部分和C++ 部分。一切都应该匹配。如果不匹配,则说明编译器不兼容,或编译标志不兼容,或者 C 和 C++ 的头文件集不同。 你们说得对——C 和 C++ 中分别有两种不同的 rtcm_t 大小。值——C++:243096,C:314832。至少现在我知道问题的根源了! 我终于知道这是一个编译标志问题。我已将 RTKLIB 编译到一个共享库中,该共享库具有一组与链接它的 C++ 代码不同的编译标志。这会导致不同的结构大小和运行时问题。非常感谢您的帮助。 【参考方案1】:

我终于知道这是一个编译标志问题。我已将 RTKLIB 编译到一个共享库中,该共享库具有一组与链接它的 C++ 代码不同的编译标志。这会导致不同的结构大小和运行时问题。

【讨论】:

以上是关于混合 C/C++ 代码中的结构指针的主要内容,如果未能解决你的问题,请参考以下文章

还不懂 C/C++ 指针结构体指针指针函数函数指针?赶紧看过来!

还不懂 C/C++ 指针结构体指针指针函数函数指针?赶紧看过来!

混合模式 C++/CLI 崩溃:atexit 中的堆损坏(静态析构函数注册)

是否可以编译 c/c++ 混合源代码?

Java介绍

将 C++ 类放入 C-struct 以供 C++ 函数使用(C/C++ 混合代码)