shlibs.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. #!/usr/bin/env python
  2. # -*- Mode: Python -*-
  3. # GObject-Introspection - a framework for introspecting GObject libraries
  4. # Copyright (C) 2009 Red Hat, Inc.
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License
  8. # as published by the Free Software Foundation; either version 2
  9. # of the License, or (at your option) any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with this program; if not, write to the Free Software
  18. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  19. # 02110-1301, USA.
  20. #
  21. from __future__ import absolute_import
  22. from __future__ import division
  23. from __future__ import print_function
  24. from __future__ import unicode_literals
  25. import os
  26. import platform
  27. import re
  28. import subprocess
  29. from .utils import get_libtool_command, extract_libtool_shlib
  30. from .ccompiler import CCompiler
  31. # For .la files, the situation is easy.
  32. def _resolve_libtool(options, binary, libraries):
  33. shlibs = []
  34. for library in libraries:
  35. shlib = extract_libtool_shlib(library)
  36. if shlib:
  37. shlibs.append(shlib)
  38. return shlibs
  39. # Assume ldd output is something vaguely like
  40. #
  41. # libpangoft2-1.0.so.0 => /usr/lib/libpangoft2-1.0.so.0 (0x006c1000)
  42. #
  43. # We say that if something in the output looks like libpangoft2<blah>
  44. # then the *first* such in the output is the soname. We require <blah>
  45. # to start with [^A-Za-z0-9_-] to avoid problems with libpango vs libpangoft2
  46. #
  47. # The negative lookbehind at the start is to avoid problems if someone
  48. # is crazy enough to name a library liblib<foo> when lib<foo> exists.
  49. #
  50. # Match absolute paths on OS X to conform to how libraries are usually
  51. # referenced on OS X systems.
  52. def _ldd_library_pattern(library_name):
  53. pattern = "(?<![A-Za-z0-9_-])(lib*%s[^A-Za-z0-9_-][^\s\(\)]*)"
  54. if platform.system() == 'Darwin':
  55. pattern = "([^\s]*lib*%s[^A-Za-z0-9_-][^\s\(\)]*)"
  56. return re.compile(pattern % re.escape(library_name))
  57. # This is a what we do for non-la files. We assume that we are on an
  58. # ELF-like system where ldd exists and the soname extracted with ldd is
  59. # a filename that can be opened with dlopen().
  60. #
  61. # On OS X this will need a straightforward alternate implementation
  62. # in terms of otool.
  63. #
  64. # Windows is more difficult, since there isn't always a straightforward
  65. # translation between library name (.lib) and the name of the .dll, so
  66. # extracting the dll names from the compiled app may not be sufficient.
  67. # We might need to hunt down the .lib in the compile-time path and
  68. # use that to figure out the name of the DLL.
  69. #
  70. def _resolve_non_libtool(options, binary, libraries):
  71. if not libraries:
  72. return []
  73. if platform.platform().startswith('OpenBSD'):
  74. # Hack for OpenBSD when using the ports' libtool which uses slightly
  75. # different directories to store the libraries in. So rewite binary.args[0]
  76. # by inserting '.libs/'.
  77. old_argdir = binary.args[0]
  78. new_libsdir = os.path.join(os.path.dirname(binary.args[0]), '.libs/')
  79. new_lib = new_libsdir + os.path.basename(binary.args[0])
  80. if os.path.exists(new_lib):
  81. binary.args[0] = new_lib
  82. os.putenv('LD_LIBRARY_PATH', new_libsdir)
  83. else:
  84. binary.args[0] = old_argdir
  85. if os.name == 'nt':
  86. cc = CCompiler()
  87. shlibs = cc.resolve_windows_libs(libraries, options)
  88. else:
  89. args = []
  90. libtool = get_libtool_command(options)
  91. if libtool:
  92. args.extend(libtool)
  93. args.append('--mode=execute')
  94. platform_system = platform.system()
  95. if options.ldd_wrapper:
  96. args.extend([options.ldd_wrapper, binary.args[0]])
  97. elif platform_system == 'Darwin':
  98. args.extend(['otool', '-L', binary.args[0]])
  99. else:
  100. args.extend(['ldd', binary.args[0]])
  101. proc = subprocess.Popen(args, stdout=subprocess.PIPE)
  102. patterns = {}
  103. for library in libraries:
  104. patterns[library] = _ldd_library_pattern(library)
  105. shlibs = []
  106. for line in proc.stdout:
  107. line = line.decode('ascii')
  108. for library, pattern in patterns.items():
  109. m = pattern.search(line)
  110. if m:
  111. del patterns[library]
  112. shlibs.append(m.group(1))
  113. break
  114. if len(patterns) > 0:
  115. raise SystemExit(
  116. "ERROR: can't resolve libraries to shared libraries: " +
  117. ", ".join(patterns.keys()))
  118. return shlibs
  119. # We want to resolve a set of library names (the <foo> of -l<foo>)
  120. # against a library to find the shared library name. The shared
  121. # library name is suppose to be what you pass to dlopen() (or
  122. # equivalent). And we want to do this using the libraries that 'binary'
  123. # is linking against.
  124. #
  125. def resolve_shlibs(options, binary, libraries):
  126. libtool = filter(lambda x: x.endswith(".la"), libraries)
  127. non_libtool = filter(lambda x: not x.endswith(".la"), libraries)
  128. return (_resolve_libtool(options, binary, libtool) +
  129. _resolve_non_libtool(options, binary, non_libtool))