文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

涨知识了,Linux的ld和ldd一起用,可以帮我们分析运行时加载库文件失败的原因

2024-11-30 02:21

关注

这里以分析 libmpi_usempif80.so 这个动态库文件,依赖哪些其他动态库,作为问题起因。

首先介绍 ld 命令如何帮我们分析mpi_usempif80 库依赖其他哪些库?mpi_usempif80 库被加载时,会从当前系统的哪些路径下搜索这些库?

#ld -lmpi_usempif08 --verbose

GNU ld (GNU Binutils for Ubuntu) 2.38
  Supported emulations:
   elf_x86_64
   elf32_x86_64
   elf_i386
   elf_iamcu
   elf_l1om
   elf_k1om
   i386pep
   i386pe
using internal linker script:
==================================================


OUTPUT_FORMAT("elf64-x86-64", "elf64-x86-64",
       "elf64-x86-64")
OUTPUT_ARCH(i386:x86-64)
ENTRY(_start)
SEARCH_DIR("=/usr/local/lib/x86_64-linux-gnu"); SEARCH_DIR("=/lib/x86_64-linux-gnu"); SEARCH_DIR("=/usr/lib/x86_64-linux-gnu"); SEARCH_DIR("=/usr/lib/x86_64-linux-gnu64"); SEARCH_DIR("=/usr/local/lib64"); SEARCH_DIR("=/lib64"); SEARCH_DIR("=/usr/lib64"); SEARCH_DIR("=/usr/local/lib"); SEARCH_DIR("=/lib"); SEARCH_DIR("=/usr/lib"); SEARCH_DIR("=/usr/x86_64-linux-gnu/lib64"); SEARCH_DIR("=/usr/x86_64-linux-gnu/lib");
SECTIONS
{
  PROVIDE (__executable_start = SEGMENT_START("text-segment", 0x400000)); . = SEGMENT_START("text-segment", 0x400000) + SIZEOF_HEADERS;
  .interp         : { *(.interp) }
  .note.gnu.build-id  : { *(.note.gnu.build-id) }
  .hash           : { *(.hash) }
  .gnu.hash       : { *(.gnu.hash) }
  .dynsym         : { *(.dynsym) }
  .dynstr         : { *(.dynstr) }
  .gnu.version    : { *(.gnu.version) }
  .gnu.version_d  : { *(.gnu.version_d) }
  .gnu.version_r  : { *(.gnu.version_r) }
  .rela.dyn       :
    {
      *(.rela.init)
      *(.rela.text .rela.text.* .rela.gnu.linkonce.t.*)
      *(.rela.fini)
      *(.rela.rodata .rela.rodata.* .rela.gnu.linkonce.r.*)
      *(.rela.data .rela.data.* .rela.gnu.linkonce.d.*)
      *(.rela.tdata .rela.tdata.* .rela.gnu.linkonce.td.*)
      *(.rela.tbss .rela.tbss.* .rela.gnu.linkonce.tb.*)
      *(.rela.ctors)
      *(.rela.dtors)
      *(.rela.got)
      *(.rela.bss .rela.bss.* .rela.gnu.linkonce.b.*)
      *(.rela.ldata .rela.ldata.* .rela.gnu.linkonce.l.*)
      *(.rela.lbss .rela.lbss.* .rela.gnu.linkonce.lb.*)
      *(.rela.lrodata .rela.lrodata.* .rela.gnu.linkonce.lr.*)
      *(.rela.ifunc)
    }
  .rela.plt       :
    {
      *(.rela.plt)
      PROVIDE_HIDDEN (__rela_iplt_start = .);
      *(.rela.iplt)
      PROVIDE_HIDDEN (__rela_iplt_end = .);
    }
  .relr.dyn : { *(.relr.dyn) }
  . = ALIGN(CONSTANT (MAXPAGESIZE));
  .init           :
  {
    KEEP (*(SORT_NONE(.init)))
  }
  .plt            : { *(.plt) *(.iplt) }
.plt.got        : { *(.plt.got) }
.plt.sec        : { *(.plt.sec) }
  .text           :
  {
    *(.text.unlikely .text.*_unlikely .text.unlikely.*)
    *(.text.exit .text.exit.*)
    *(.text.startup .text.startup.*)
    *(.text.hot .text.hot.*)
    *(SORT(.text.sorted.*))
    *(.text .stub .text.* .gnu.linkonce.t.*)
    
    *(.gnu.warning)
  }
  .fini           :
  {
    KEEP (*(SORT_NONE(.fini)))
  }
  PROVIDE (__etext = .);
  PROVIDE (_etext = .);
  PROVIDE (etext = .);
  . = ALIGN(CONSTANT (MAXPAGESIZE));
  
  . = SEGMENT_START("rodata-segment", ALIGN(CONSTANT (MAXPAGESIZE)) + (. & (CONSTANT (MAXPAGESIZE) - 1)));
  .rodata         : { *(.rodata .rodata.* .gnu.linkonce.r.*) }
  .rodata1        : { *(.rodata1) }
  .eh_frame_hdr   : { *(.eh_frame_hdr) *(.eh_frame_entry .eh_frame_entry.*) }
  .eh_frame       : ONLY_IF_RO { KEEP (*(.eh_frame)) *(.eh_frame.*) }
  .gcc_except_table   : ONLY_IF_RO { *(.gcc_except_table .gcc_except_table.*) }
  .gnu_extab   : ONLY_IF_RO { *(.gnu_extab*) }
  
  .exception_ranges   : ONLY_IF_RO { *(.exception_ranges*) }
  
  . = DATA_SEGMENT_ALIGN (CONSTANT (MAXPAGESIZE), CONSTANT (COMMONPAGESIZE));
  
  .eh_frame       : ONLY_IF_RW { KEEP (*(.eh_frame)) *(.eh_frame.*) }
  .gnu_extab      : ONLY_IF_RW { *(.gnu_extab) }
  .gcc_except_table   : ONLY_IF_RW { *(.gcc_except_table .gcc_except_table.*) }
  .exception_ranges   : ONLY_IF_RW { *(.exception_ranges*) }
  
  .tdata   :
   {
     PROVIDE_HIDDEN (__tdata_start = .);
     *(.tdata .tdata.* .gnu.linkonce.td.*)
   }
  .tbss    : { *(.tbss .tbss.* .gnu.linkonce.tb.*) *(.tcommon) }
  .preinit_array    :
  {
    PROVIDE_HIDDEN (__preinit_array_start = .);
    KEEP (*(.preinit_array))
    PROVIDE_HIDDEN (__preinit_array_end = .);
  }
  .init_array    :
  {
    PROVIDE_HIDDEN (__init_array_start = .);
    KEEP (*(SORT_BY_INIT_PRIORITY(.init_array.*) SORT_BY_INIT_PRIORITY(.ctors.*)))
    KEEP (*(.init_array EXCLUDE_FILE (*crtbegin.o *crtbegin?.o *crtend.o *crtend?.o ) .ctors))
    PROVIDE_HIDDEN (__init_array_end = .);
  }
  .fini_array    :
  {
    PROVIDE_HIDDEN (__fini_array_start = .);
    KEEP (*(SORT_BY_INIT_PRIORITY(.fini_array.*) SORT_BY_INIT_PRIORITY(.dtors.*)))
    KEEP (*(.fini_array EXCLUDE_FILE (*crtbegin.o *crtbegin?.o *crtend.o *crtend?.o ) .dtors))
    PROVIDE_HIDDEN (__fini_array_end = .);
  }
  .ctors          :
  {
    
    KEEP (*crtbegin.o(.ctors))
    KEEP (*crtbegin?.o(.ctors))
    
    KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .ctors))
    KEEP (*(SORT(.ctors.*)))
    KEEP (*(.ctors))
  }
  .dtors          :
  {
    KEEP (*crtbegin.o(.dtors))
    KEEP (*crtbegin?.o(.dtors))
    KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .dtors))
    KEEP (*(SORT(.dtors.*)))
    KEEP (*(.dtors))
  }
  .jcr            : { KEEP (*(.jcr)) }
  .data.rel.ro : { *(.data.rel.ro.local* .gnu.linkonce.d.rel.ro.local.*) *(.data.rel.ro .data.rel.ro.* .gnu.linkonce.d.rel.ro.*) }
  .dynamic        : { *(.dynamic) }
  .got            : { *(.got) *(.igot) }
  . = DATA_SEGMENT_RELRO_END (SIZEOF (.got.plt) >= 24 ? 24 : 0, .);
  .got.plt        : { *(.got.plt) *(.igot.plt) }
  .data           :
  {
    *(.data .data.* .gnu.linkonce.d.*)
    SORT(CONSTRUCTORS)
  }
  .data1          : { *(.data1) }
  _edata = .; PROVIDE (edata = .);
  . = .;
  __bss_start = .;
  .bss            :
  {
   *(.dynbss)
   *(.bss .bss.* .gnu.linkonce.b.*)
   *(COMMON)
   
   . = ALIGN(. != 0 ? 64 / 8 : 1);
  }
  .lbss   :
  {
    *(.dynlbss)
    *(.lbss .lbss.* .gnu.linkonce.lb.*)
    *(LARGE_COMMON)
  }
  . = ALIGN(64 / 8);
  . = SEGMENT_START("ldata-segment", .);
  .lrodata   ALIGN(CONSTANT (MAXPAGESIZE)) + (. & (CONSTANT (MAXPAGESIZE) - 1)) :
  {
    *(.lrodata .lrodata.* .gnu.linkonce.lr.*)
  }
  .ldata   ALIGN(CONSTANT (MAXPAGESIZE)) + (. & (CONSTANT (MAXPAGESIZE) - 1)) :
  {
    *(.ldata .ldata.* .gnu.linkonce.l.*)
    . = ALIGN(. != 0 ? 64 / 8 : 1);
  }
  . = ALIGN(64 / 8);
  _end = .; PROVIDE (end = .);
  . = DATA_SEGMENT_END (.);
  
  .stab          0 : { *(.stab) }
  .stabstr       0 : { *(.stabstr) }
  .stab.excl     0 : { *(.stab.excl) }
  .stab.exclstr  0 : { *(.stab.exclstr) }
  .stab.index    0 : { *(.stab.index) }
  .stab.indexstr 0 : { *(.stab.indexstr) }
  .comment       0 : { *(.comment) }
  .gnu.build.attributes : { *(.gnu.build.attributes .gnu.build.attributes.*) }
  
  
  .debug          0 : { *(.debug) }
  .line           0 : { *(.line) }
  
  .debug_srcinfo  0 : { *(.debug_srcinfo) }
  .debug_sfnames  0 : { *(.debug_sfnames) }
  
  .debug_aranges  0 : { *(.debug_aranges) }
  .debug_pubnames 0 : { *(.debug_pubnames) }
  
  .debug_info     0 : { *(.debug_info .gnu.linkonce.wi.*) }
  .debug_abbrev   0 : { *(.debug_abbrev) }
  .debug_line     0 : { *(.debug_line .debug_line.* .debug_line_end) }
  .debug_frame    0 : { *(.debug_frame) }
  .debug_str      0 : { *(.debug_str) }
  .debug_loc      0 : { *(.debug_loc) }
  .debug_macinfo  0 : { *(.debug_macinfo) }
  
  .debug_weaknames 0 : { *(.debug_weaknames) }
  .debug_funcnames 0 : { *(.debug_funcnames) }
  .debug_typenames 0 : { *(.debug_typenames) }
  .debug_varnames  0 : { *(.debug_varnames) }
  
  .debug_pubtypes 0 : { *(.debug_pubtypes) }
  .debug_ranges   0 : { *(.debug_ranges) }
  
  .debug_addr     0 : { *(.debug_addr) }
  .debug_line_str 0 : { *(.debug_line_str) }
  .debug_loclists 0 : { *(.debug_loclists) }
  .debug_macro    0 : { *(.debug_macro) }
  .debug_names    0 : { *(.debug_names) }
  .debug_rnglists 0 : { *(.debug_rnglists) }
  .debug_str_offsets 0 : { *(.debug_str_offsets) }
  .debug_sup      0 : { *(.debug_sup) }
  .gnu.attributes 0 : { KEEP (*(.gnu.attributes)) }
  /DISCARD/ : { *(.note.GNU-stack) *(.gnu_debuglink) *(.gnu.lto_*) }
}


==================================================
ld: mode elf_x86_64
attempt to open /usr/local/lib/x86_64-linux-gnu/libmpi_usempif08.so failed
attempt to open /usr/local/lib/x86_64-linux-gnu/libmpi_usempif08.a failed
attempt to open /lib/x86_64-linux-gnu/libmpi_usempif08.so succeeded
/lib/x86_64-linux-gnu/libmpi_usempif08.so
libmpi_mpifh.so.40 needed by /lib/x86_64-linux-gnu/libmpi_usempif08.so
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/mpi/fortran/mpif-h/.libs/libmpi_mpifh.so.40 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/.libs/libmpi_mpifh.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libmpi_mpifh.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libmpi_mpifh.so.40 failed
attempt to open /usr/local/lib/libmpi_mpifh.so.40 failed
found libmpi_mpifh.so.40 at /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40
libmpi.so.40 needed by /lib/x86_64-linux-gnu/libmpi_usempif08.so
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/mpi/fortran/mpif-h/.libs/libmpi.so.40 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/.libs/libmpi.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libmpi.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libmpi.so.40 failed
attempt to open /usr/local/lib/libmpi.so.40 failed
found libmpi.so.40 at /usr/lib/x86_64-linux-gnu/libmpi.so.40
libc.so.6 needed by /lib/x86_64-linux-gnu/libmpi_usempif08.so
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/mpi/fortran/mpif-h/.libs/libc.so.6 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/.libs/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libc.so.6 failed
attempt to open /usr/local/lib/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/fortran/gfortran/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libc.so.6 failed
attempt to open /usr/local/lib/x86_64-linux-gnu/libc.so.6 failed
attempt to open /usr/local/lib/x86_64-linux-gnu/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu64/libc.so.6 failed
attempt to open /usr/local/lib64/libc.so.6 failed
attempt to open /lib64/libc.so.6 failed
attempt to open /usr/lib64/libc.so.6 failed
attempt to open /usr/local/lib/libc.so.6 failed
attempt to open /lib/libc.so.6 failed
attempt to open /usr/lib/libc.so.6 failed
attempt to open /usr/x86_64-linux-gnu/lib64/libc.so.6 failed
attempt to open /usr/x86_64-linux-gnu/lib/libc.so.6 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/mpi/fortran/mpif-h/.libs/libc.so.6 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/.libs/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libc.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libc.so.6 failed
attempt to open /usr/local/lib/libc.so.6 failed
found libc.so.6 at /usr/lib/x86_64-linux-gnu/libc.so.6
libopen-pal.so.40 needed by /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/.libs/libopen-pal.so.40 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/opal/.libs/libopen-pal.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libopen-pal.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libopen-pal.so.40 failed
attempt to open /usr/local/lib/libopen-pal.so.40 failed
found libopen-pal.so.40 at /usr/lib/x86_64-linux-gnu/libopen-pal.so.40
libopen-rte.so.40 needed by /usr/lib/x86_64-linux-gnu/libmpi.so.40
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libopen-rte.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libopen-rte.so.40 failed
attempt to open /usr/local/lib/libopen-rte.so.40 failed
found libopen-rte.so.40 at /usr/lib/x86_64-linux-gnu/libopen-rte.so.40
libm.so.6 needed by /usr/lib/x86_64-linux-gnu/libmpi.so.40
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libm.so.6 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libm.so.6 failed
attempt to open /usr/local/lib/libm.so.6 failed
found libm.so.6 at /usr/lib/x86_64-linux-gnu/libm.so.6
libhwloc.so.15 needed by /usr/lib/x86_64-linux-gnu/libmpi.so.40
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libhwloc.so.15 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libhwloc.so.15 failed
attempt to open /usr/local/lib/libhwloc.so.15 failed
found libhwloc.so.15 at /usr/lib/x86_64-linux-gnu/libhwloc.so.15
ld-linux-x86-64.so.2 needed by /usr/lib/x86_64-linux-gnu/libc.so.6
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/ld-linux-x86-64.so.2 failed
attempt to open /usr/local/lib/ld-linux-x86-64.so.2 failed
found ld-linux-x86-64.so.2 at /usr/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2
libevent_core-2.1.so.7 needed by /usr/lib/x86_64-linux-gnu/libopen-pal.so.40
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libevent_core-2.1.so.7 failed
attempt to open /usr/local/lib/libevent_core-2.1.so.7 failed
found libevent_core-2.1.so.7 at /usr/lib/x86_64-linux-gnu/libevent_core-2.1.so.7
libevent_pthreads-2.1.so.7 needed by /usr/lib/x86_64-linux-gnu/libopen-pal.so.40
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libevent_pthreads-2.1.so.7 failed
attempt to open /usr/local/lib/libevent_pthreads-2.1.so.7 failed
found libevent_pthreads-2.1.so.7 at /usr/lib/x86_64-linux-gnu/libevent_pthreads-2.1.so.7
libz.so.1 needed by /usr/lib/x86_64-linux-gnu/libopen-rte.so.40
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libz.so.1 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libz.so.1 failed
attempt to open /usr/local/lib/libz.so.1 failed
found libz.so.1 at /usr/lib/x86_64-linux-gnu/libz.so.1
libudev.so.1 needed by /usr/lib/x86_64-linux-gnu/libhwloc.so.15
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libudev.so.1 failed
attempt to open /usr/local/lib/libudev.so.1 failed
found libudev.so.1 at /usr/lib/x86_64-linux-gnu/libudev.so.1


ld -lmpi_usempif08 --verbose | grep found
found libmpi_mpifh.so.40 at /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40
found libmpi.so.40 at /usr/lib/x86_64-linux-gnu/libmpi.so.40
ld: warning: cannot find entry symbol _start; not setting start address
found libc.so.6 at /usr/lib/x86_64-linux-gnu/libc.so.6
found libopen-pal.so.40 at /usr/lib/x86_64-linux-gnu/libopen-pal.so.40
found libopen-rte.so.40 at /usr/lib/x86_64-linux-gnu/libopen-rte.so.40
found libm.so.6 at /usr/lib/x86_64-linux-gnu/libm.so.6
found libhwloc.so.15 at /usr/lib/x86_64-linux-gnu/libhwloc.so.15
found ld-linux-x86-64.so.2 at /usr/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2
found libevent_core-2.1.so.7 at /usr/lib/x86_64-linux-gnu/libevent_core-2.1.so.7
found libevent_pthreads-2.1.so.7 at /usr/lib/x86_64-linux-gnu/libevent_pthreads-2.1.so.7
found libz.so.1 at /usr/lib/x86_64-linux-gnu/libz.so.1
found libudev.so.1 at /usr/lib/x86_64-linux-gnu/libudev.so.1

上面这段信息很长,有很多有价值的信息,如果你只想看关键部分,那么我保留其中的关键内容,来一个精华版的输出信息:

#ld -lmpi_usempif08 --verbose

GNU ld (GNU Binutils for Ubuntu) 2.38
  Supported emulations:
   elf_x86_64
……
using internal linker script:

   notice and this notice are preserved.  */
OUTPUT_FORMAT("elf64-x86-64", "elf64-x86-64",
       "elf64-x86-64")
OUTPUT_ARCH(i386:x86-64)
ENTRY(_start)  # 运行时可执行文件从代码段的 _start 位置开始运行。

# 运行时需从哪些目录找动态库文件
SEARCH_DIR("=/usr/local/lib/x86_64-linux-gnu"); SEARCH_DIR("=/lib/x86_64-linux-gnu"); SEARCH_DIR("=/usr/lib/x86_64-linux-gnu"); SEARCH_DIR("=/usr/lib/x86_64-linux-gnu64"); SEARCH_DIR("=/usr/local/lib64"); SEARCH_DIR("=/lib64"); SEARCH_DIR("=/usr/lib64"); SEARCH_DIR("=/usr/local/lib"); SEARCH_DIR("=/lib"); SEARCH_DIR("=/usr/lib"); SEARCH_DIR("=/usr/x86_64-linux-gnu/lib64"); SEARCH_DIR("=/usr/x86_64-linux-gnu/lib");

# 二进制文件有哪些的代码段、数据段
SECTIONS
{
  PROVIDE (__executable_start = SEGMENT_START("text-segment", 0x400000)); . = SEGMENT_START("text-segment", 0x400000) + SIZEOF_HEADERS;
  .interp         : { *(.interp) }
……
  .rodata         : { *(.rodata .rodata.* .gnu.linkonce.r.*) }

……
  
  # 线程本地存储区域
  .tdata   :
   {
     PROVIDE_HIDDEN (__tdata_start = .);
     *(.tdata .tdata.* .gnu.linkonce.td.*)
   }
  .tbss    : { *(.tbss .tbss.* .gnu.linkonce.tb.*) *(.tcommon) }
  .preinit_array    :
  {
    PROVIDE_HIDDEN (__preinit_array_start = .);
    KEEP (*(.preinit_array))
    PROVIDE_HIDDEN (__preinit_array_end = .);
  }
  # 数据区
  .data           :
  {
    *(.data .data.* .gnu.linkonce.d.*)
    SORT(CONSTRUCTORS)
  }
  .data1          : { *(.data1) }
  _edata = .; PROVIDE (edata = .);
  . = .;
  __bss_start = .;
  
  # BSS区
  .bss            :
  {
   ……
  }
 ……
}

# 下面是本文的重点关注部分
==================================================

# 文件1:逐个目录匹配是否找到被依赖的libmpi_usempif08.so 库文件
ld: mode elf_x86_64
attempt to open /usr/local/lib/x86_64-linux-gnu/libmpi_usempif08.so failed
attempt to open /usr/local/lib/x86_64-linux-gnu/libmpi_usempif08.a failed
attempt to open /lib/x86_64-linux-gnu/libmpi_usempif08.so succeeded
# 找到了 libmpi_usempif08.so 库文件

# 文件2:逐个目录匹配是否找到被依赖的 libmpi_mpifh.so 库文件
libmpi_mpifh.so.40 needed by /lib/x86_64-linux-gnu/libmpi_usempif08.so
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/mpi/fortran/mpif-h/.libs/libmpi_mpifh.so.40 failed
attempt to open /build/openmpi-RCkqwI/openmpi-4.1.2/debian/build-gfortran/ompi/.libs/libmpi_mpifh.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/openmpi/lib/libmpi_mpifh.so.40 failed
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libmpi_mpifh.so.40 failed
attempt to open /usr/local/lib/libmpi_mpifh.so.40 failed
found libmpi_mpifh.so.40 at /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40
# 找到了 libmpi_mpifh.so 库文件

# 找 libmpi.so.40 
libmpi.so.40 needed by /lib/x86_64-linux-gnu/libmpi_usempif08.so
……
found libmpi.so.40 at /usr/lib/x86_64-linux-gnu/libmpi.so.40

# 找 libc.so.6 
libc.so.6 needed by /lib/x86_64-linux-gnu/libmpi_usempif08.so
……
found libc.so.6 at /usr/lib/x86_64-linux-gnu/libc.so.6

# 找 libopen-pal.so.40
libopen-pal.so.40 needed by /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40
……
found libopen-pal.so.40 at /usr/lib/x86_64-linux-gnu/libopen-pal.so.40

# 找 libopen-rte.so.40
libopen-rte.so.40 needed by /usr/lib/x86_64-linux-gnu/libmpi.so.40
……
found libopen-rte.so.40 at /usr/lib/x86_64-linux-gnu/libopen-rte.so.40

# 找 libm.so.6
libm.so.6 needed by /usr/lib/x86_64-linux-gnu/libmpi.so.40
……
found libm.so.6 at /usr/lib/x86_64-linux-gnu/libm.so.6
libhwloc.so.15 needed by /usr/lib/x86_64-linux-gnu/libmpi.so.40

# 找 libhwloc.so.15
……略

# 找 libudev.so.1
libudev.so.1 needed by /usr/lib/x86_64-linux-gnu/libhwloc.so.15
attempt to open /usr/lib/x86_64-linux-gnu/libfakeroot/libudev.so.1 failed
attempt to open /usr/local/lib/libudev.so.1 failed
found libudev.so.1 at /usr/lib/x86_64-linux-gnu/libudev.so.1

进一步,我们可以用 grep 过滤只显示 mpi_usempif08 实际找到的库的信息:

#ld -lmpi_usempif08 --verbose | grep found

found libmpi_mpifh.so.40 at /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40
found libmpi.so.40 at /usr/lib/x86_64-linux-gnu/libmpi.so.40
ld: warning: cannot find entry symbol _start; not setting start address
found libc.so.6 at /usr/lib/x86_64-linux-gnu/libc.so.6
found libopen-pal.so.40 at /usr/lib/x86_64-linux-gnu/libopen-pal.so.40
found libopen-rte.so.40 at /usr/lib/x86_64-linux-gnu/libopen-rte.so.40
found libm.so.6 at /usr/lib/x86_64-linux-gnu/libm.so.6
found libhwloc.so.15 at /usr/lib/x86_64-linux-gnu/libhwloc.so.15
found ld-linux-x86-64.so.2 at /usr/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2
found libevent_core-2.1.so.7 at /usr/lib/x86_64-linux-gnu/libevent_core-2.1.so.7
found libevent_pthreads-2.1.so.7 at /usr/lib/x86_64-linux-gnu/libevent_pthreads-2.1.so.7
found libz.so.1 at /usr/lib/x86_64-linux-gnu/libz.so.1
found libudev.so.1 at /usr/lib/x86_64-linux-gnu/libudev.so.1

而ldd 与 ld是不同的。ldd是分析linux的运行库的依赖关系、ld是可执行文件链接器常用于C/C++/Fortran等语言代码生成可执行文件的链接阶段。

我们看下 使用 ldd 对同一个动态库所依赖的其他库的分析:

#ldd /usr/lib/x86_64-linux-gnu/libmpi_usempif08.so.40.30.0

 linux-vdso.so.1 (0x00007ffd2f3fd000)
 libmpi_mpifh.so.40 => /usr/lib/x86_64-linux-gnu/libmpi_mpifh.so.40 (0x00007f34366bf000)
 libmpi.so.40 => /usr/lib/x86_64-linux-gnu/libmpi.so.40 (0x00007f3436588000)
 libc.so.6 => /usr/lib/x86_64-linux-gnu/libc.so.6 (0x00007f3436200000)
 libopen-pal.so.40 => /usr/lib/x86_64-linux-gnu/libopen-pal.so.40 (0x00007f34364d5000)
 libopen-rte.so.40 => /usr/lib/x86_64-linux-gnu/libopen-rte.so.40 (0x00007f3436143000)
 libm.so.6 => /usr/lib/x86_64-linux-gnu/libm.so.6 (0x00007f343605c000)
 libhwloc.so.15 => /usr/lib/x86_64-linux-gnu/libhwloc.so.15 (0x00007f3436477000)
 /lib64/ld-linux-x86-64.so.2 (0x00007f343677c000)
 libevent_core-2.1.so.7 => /usr/lib/x86_64-linux-gnu/libevent_core-2.1.so.7 (0x00007f3436442000)
 libevent_pthreads-2.1.so.7 => /usr/lib/x86_64-linux-gnu/libevent_pthreads-2.1.so.7 (0x00007f343643d000)
 libz.so.1 => /usr/lib/x86_64-linux-gnu/libz.so.1 (0x00007f3436040000)
 libudev.so.1 => /usr/lib/x86_64-linux-gnu/libudev.so.1 (0x00007f3436016000)

这里的顺序与ld -lmpi_usempif08 --verbose | grep found 的结果是一致的。

来源:深入理解Linux内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯