10

我正在尝试编写一个使用 libhdf5 的 matlab mex 函数;我的 Linux 安装提供了 libhdf5-1.8 共享库和头文件。但是,我的 Matlab 版本 r2007b 提供了 1.6 版本的 libhdf5.so。(显然,Matlab.mat文件引导 hdf5)。当我编译 mex 时,它在 Matlab 中出现了段错误。如果我将我的 libhdf5 版本降级到 1.6(不是长期选项),代码可以编译并运行良好。

问题:我该如何解决这个问题?如何告诉 mex 编译过程链接到 /usr/lib64/libhdf5.so.6 而不是 /opt/matlab/bin/glnxa64/libhdf5.so.0 ?当我尝试-Wl,-rpath-link,/usr/lib64在我的编译中使用它时,我会收到如下错误:

/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/../../../../x86_64-pc-linux-gnu/bin/ld: warning: libhdf5.so.0, needed by /opt/matlab/matlab75/bin/glnxa64/libmat.so, may conflict with libhdf5.so.6
/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/../../../../lib64/crt1.o: In function `_start':
(.text+0x20): undefined reference to `main'
collect2: ld returned 1 exit status

    mex: link of 'hdf5_read_strings.mexa64' failed.

make: *** [hdf5_read_strings.mexa64] Error 1

确认。最后的手段是下载 hdf5-1.6.5 标头的本地副本并完成它,但这不是未来的证明(Matlab 版本升级是我的未来。)。有任何想法吗?

编辑:根据 Ramashalanka 的出色建议,我

A) 调用mex -v以获取 3 个gcc命令;最后是链接器命令;

B)用 a 调用该链接器命令-v以获取collect命令;

C)调用它collect2 -v -t和其余的标志。

我的输出的相关部分:

/usr/bin/ld: mode elf_x86_64
/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/../../../../lib64/crti.o
/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/crtbeginS.o
hdf5_read_strings.o
mexversion.o
-lmx (/opt/matlab/matlab75/bin/glnxa64/libmx.so)
-lmex (/opt/matlab/matlab75/bin/glnxa64/libmex.so)
-lhdf5 (/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/../../../../lib64/libhdf5.so)
/lib64/libz.so
-lm (/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/../../../../lib64/libm.so)
-lstdc++ (/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/libstdc++.so)
-lgcc_s (/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/libgcc_s.so)
/lib64/libpthread.so.0
/lib64/libc.so.6
/lib64/ld-linux-x86-64.so.2
-lgcc_s (/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/libgcc_s.so)
/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/crtendS.o
/usr/lib/gcc/x86_64-pc-linux-gnu/4.3.4/../../../../lib64/crtn.o

所以,实际上libhdf5.sofrom/usr/lib64正在被引用。但是,我相信这会被环境变量覆盖,我LD_LIBRARY_PATH的 Matlab 版本会在运行时自动设置它,因此它可以找到自己的版本,例如libmex.so,等等。

我认为该crt_file.c示例既可以使用 b/c,也可以不使用我正在使用的函数 ( H5DOpen,在从 1.6 到 1.8 的移动中具有签名更改(是的,我正在使用-DH5_USE_16_API)),或者不太可能,b/ c 它不会击中需要 hdf5 的 Matlab 内部部件。确认。

4

2 回答 2

9

以下适用于我的系统:

  1. 安装 hdf5 版本 1.8.4(你已经这样做了:我安装了源代码并编译以确保它与我的系统兼容,我获得 gcc 版本并获得静态库 - 例如为我的系统提供的二进制文件是icc具体的)。

  2. 制作一个目标文件。你已经有了自己的文件。我h5_crtfile.c这里使用了简单的(一个好主意,首先从这个简单的文件开始寻找警告)。我改为main使用mexFunction通常的 args 并包含mex.h.

  3. 指定要显式加载的静态-lhdf51.8.4 库(没有必要的 -L 的完整路径)并且不要包含在LDFLAGS. 包括一个-t选项,这样您就可以确保没有加载动态 hdf5 库。你还需要-lz, withzlib安装。对于 darwin,我们还需要一个-bundlein LDFLAGS

    mex CFLAGS='-I/usr/local/hdf5/include' LDFLAGS='-t /usr/local/hdf5/lib/libhdf5.a -lz -bundle' h5_crtfile.c -v
    

    对于 linux,您需要一个等效的与位置无关的调用,例如fPICand maybe -shared,但我没有具有 matlab 许可证的 linux 系统,所以我无法检查:

    mex CFLAGS='-fPIC -I/usr/local/hdf5/include' LDFLAGS='-t /usr/local/hdf5/lib/libhdf5.a -lz -shared' h5_crtfile.c -v
    
  4. 运行h5_crtfilemex 文件。这在我的机器上运行没有问题。它只是执行 H5Fcreate 和 H5Fclose 以在当前目录中创建“file.h5”,当我调用时,file file.h5我得到file.h5: Hierarchical Data Format (version 5) data.

请注意,如果我在第 3 步中包含-lhdf5上述内容,那么当我尝试运行可执行文件时,matlab 会中止(因为它随后使用 matlab 的动态库,对我来说是 1.6.5 版),所以这绝对解决了我系统上的问题。

谢谢你的问题。我上面的解决方案对我来说肯定比我以前做的要容易得多。希望以上内容对您有用。

于 2010-02-02T23:59:45.723 回答
3

我接受了 Ramashalanka 的回答,因为它引导我找到了确切的解决方案,我将在此处发布只是为了完整起见:

  1. 从hdf5网站下载hdf5-1.6.5库,并将头文件安装到本地目录;
  2. 告诉 mex 在此本地目录中查找“hdf5.h”,而不是在标准位置(例如/usr/include。)
  3. 告诉 mex 编译我的代码和matlab 提供的共享库不要在.-ldfh5LDFLAGS

我使用的命令本质上是:

/opt/matlab/matlab_default/bin/mex -v CC#gcc CXX#g++ CFLAGS#"-Wall -O3 -fPIC -I./hdf5_1.6.5/src -I/usr/include -I/opt/matlab/matlab_default/extern/include" CXXFLAGS#"-Wall -O3 -fPIC -I./hdf5_1.6.5/src -I/usr/include -I/opt/matlab/matlab_default/extern/include " -O -lmwblas -largeArrayDims -L/usr/lib64 hdf5_read_strings.c /opt/matlab/matlab_default/bin/glnxa64/libhdf5.so.0

这被 mex 翻译成命令:

gcc -c -I/opt/matlab/matlab75/extern/include -DMATLAB_MEX_FILE -Wall -O3 -fPIC -I./hdf5_1.6.5/src -I/usr/include -I/opt/matlab/matlab_default/extern/include -O -DNDEBUG hdf5_read_strings.c
gcc -c -I/opt/matlab/matlab75/extern/include -DMATLAB_MEX_FILE -Wall -O3 -fPIC -I./hdf5_1.6.5/src -I/usr/include -I/opt/matlab/matlab_default/extern/include -O -DNDEBUG /opt/matlab/matlab75/extern/src/mexversion.c
gcc -O -pthread -shared -Wl,--version-script,/opt/matlab/matlab75/extern/lib/glnxa64/mexFunction.map -Wl,--no-undefined -o hdf5_read_strings.mexa64  hdf5_read_strings.o mexversion.o  -lmwblas -L/usr/lib64 /opt/matlab/matlab_default/bin/glnxa64/libhdf5.so.0 -Wl,-rpath-link,/opt/matlab/matlab_default/bin/glnxa64 -L/opt/matlab/matlab_default/bin/glnxa64 -lmx -lmex -lmat -lm -lstdc++

这个解决方案应该适用于我所有的各种目标机器,至少在我升级到 matlab r2009a 之前,我相信它使用 hdf5-1.8。感谢所有的帮助,很抱歉对此如此密集——我认为我过度致力于使用 hdf5 的打包版本,而不是一组本地头文件。

请注意,如果 Mathworks 提供了一组带有 Matlab 发行版的头文件,这一切都将是微不足道的......

于 2010-02-09T18:12:57.537 回答