(虽然英特尔的论坛是一个更自然地提出这个问题的地方,但我在这里发布它是希望比英特尔完全缺乏的活动更多 - 到目前为止)
我无法创建使用 Intel Media SDK(linux 服务器)来操作 h264 视频的动态链接库,并注意到 MFX 库的设计存在问题。据我了解,程序应该链接到静态库,例如:
$ g++ .... -L/opt/intel/mediasdk/lib/lin_x64 -lmfx
但是,此libmfx.a
库似乎将所有调用委托给dlopen
ed动态库/opt/intel/mediasdk/lib64/libmfxhw64.so
。值得注意的是,静态库和动态库公开的函数名(和签名)是相同的,这有点令人困惑和危险。
虽然我不理解这种设计背后的基本原理,但它本身不应该是一个问题,当(静态)libmfx.a
包含在共享对象中时,库中的一些静态/全局初始化显然不会造成严重破坏。IE。:
+------+ +-----------+
| main | <-- | mylib.so |
+------+ | | +---------------+
| libmfx.a | (dlopen) | libmfxhw64.so |
| <------------- |
|+---------+| |+-------------+|
||MFXInit()|| || MFXInit() ||
||... || || ... ||
|| || || ||
+===========+ +===============+
上面的库可以这样组装:
$ g++ -shared -o mylib.so my1.o my2.o -lmfx
然后(动态)链接到main.o
这样:
$ g++ -o main main.o mylib.so -ldl
(请注意,附加libdl
是必要的,以允许libmfx.a
。dlopen()
libmfxhw64.so
)
不幸的是,在第一次MFXInit()
调用时,程序会导致分段错误(访问地址 0x0000400)。GDB 回溯:
#0 0x0000000000000400 in ?? ()
#1 0x00007ffff61fb4cd in MFXInit () from /opt/intel/mediasdk/lib64/libmfxhw64-p.so.1.13
#2 0x00007ffff7bd3a1f in MFX_DISP_HANDLE::LoadSelectedDLL(char const*, eMfxImplType, int, int) () from ./lib-a.so
#3 0x00007ffff7bd12b1 in MFXInit () from ./lib-a.so
#4 0x00007ffff7bd09c8 in test_mfx () at lib.c:12
#5 0x0000000000400744 in main (argc=1, argv=0x7fffffffe0d8) at main.c:8
(注意MFXInit()
at stackframe#3
是 inlibmfx.a
而 at#1
是 in libmfxhw64.so
。)
请注意,创建为静态库时不会崩溃。使用断点和反汇编程序,我设法制作了以下回溯快照,在这两种情况下都是 at ,但它们似乎遇到了不同版本的(由于重定位,绝对地址毫无意义):mylib
#1
MFXInit+424
MFXQueryVersion
#0 0x00007ffff6411980 in MFXQueryVersion () from /opt/intel/mediasdk/lib64/libmfxhw64-p.so.1.13
#1 0x00007ffff640c4cd in MFXInit () from /opt/intel/mediasdk/lib64/libmfxhw64-p.so.1.13
#2 0x000000000040484f in MFX_DISP_HANDLE::LoadSelectedDLL(char const*, eMfxImplType, int, int) ()
#3 0x00000000004020e1 in MFXInit ()
#4 0x0000000000401800 in test_mfx () at lib.c:12
#5 0x0000000000401794 in main (argc=1, argv=0x7fffffffe0e8) at main.c:8
因为静态和共享的英特尔库都公开了相同的 API 函数,所以我可以直接链接到libmfxhw64.so
胆量,但我认为绕过静态“调度程序”是没有保证的(?)
有人可以解释英特尔在上述设计背后的想法吗?规范,为什么要提供一个仅委托给.so
具有相同接口的静态库?
此外,SEGV 似乎是由libmfx.a
或中的静态/全局数据引起的libmfxhw64.so
。有没有办法在动态加载的静态/全局部分上强制执行特定的执行顺序?调试此类问题的最佳方法是什么?
在 Intel Haswell i7-4790 @3.6Ghz 上使用 Intel Media SDK R2 (ubuntu 12) 和 Intel Media SDK 2015R3-R5 (Centos 7, 1.13/1.15) 进行测试
如果您有一个有效的英特尔 MSDK 设置,请编译我的示例代码以确认问题。