blob: 1bd2d70fd3759160dd35e2a7d6b627128065a96a [file] [log] [blame]
Thomas Wouters477c8d52006-05-27 19:21:47 +00001import sys, os
Thomas Wouters477c8d52006-05-27 19:21:47 +00002
3# find_library(name) returns the pathname of a library, or None.
4if os.name == "nt":
Thomas Heller3eaaeb42008-05-23 17:26:46 +00005
6 def _get_build_version():
7 """Return the version of MSVC that was used to build Python.
8
9 For Python 2.3 and up, the version number is included in
10 sys.version. For earlier versions, assume the compiler is MSVC 6.
11 """
12 # This function was copied from Lib/distutils/msvccompiler.py
13 prefix = "MSC v."
14 i = sys.version.find(prefix)
15 if i == -1:
16 return 6
17 i = i + len(prefix)
18 s, rest = sys.version[i:].split(" ", 1)
19 majorVersion = int(s[:-2]) - 6
20 minorVersion = int(s[2:3]) / 10.0
21 # I don't think paths are affected by minor version in version 6
22 if majorVersion == 6:
23 minorVersion = 0
24 if majorVersion >= 6:
25 return majorVersion + minorVersion
26 # else we don't know what version of the compiler this is
27 return None
28
29 def find_msvcrt():
30 """Return the name of the VC runtime dll"""
31 version = _get_build_version()
32 if version is None:
33 # better be safe than sorry
34 return None
35 if version <= 6:
36 clibname = 'msvcrt'
37 else:
38 clibname = 'msvcr%d' % (version * 10)
39
40 # If python was built with in debug mode
41 import imp
42 if imp.get_suffixes()[0][0] == '_d.pyd':
43 clibname += 'd'
44 return clibname+'.dll'
45
Thomas Wouters477c8d52006-05-27 19:21:47 +000046 def find_library(name):
Thomas Heller3eaaeb42008-05-23 17:26:46 +000047 if name in ('c', 'm'):
48 return find_msvcrt()
Thomas Wouters477c8d52006-05-27 19:21:47 +000049 # See MSDN for the REAL search order.
50 for directory in os.environ['PATH'].split(os.pathsep):
51 fname = os.path.join(directory, name)
Thomas Heller00cfc372009-05-05 19:04:40 +000052 if os.path.isfile(fname):
Thomas Wouters477c8d52006-05-27 19:21:47 +000053 return fname
54 if fname.lower().endswith(".dll"):
55 continue
56 fname = fname + ".dll"
Thomas Heller00cfc372009-05-05 19:04:40 +000057 if os.path.isfile(fname):
Thomas Wouters477c8d52006-05-27 19:21:47 +000058 return fname
59 return None
60
61if os.name == "ce":
62 # search path according to MSDN:
63 # - absolute path specified by filename
64 # - The .exe launch directory
65 # - the Windows directory
66 # - ROM dll files (where are they?)
67 # - OEM specified search path: HKLM\Loader\SystemPath
68 def find_library(name):
69 return name
70
71if os.name == "posix" and sys.platform == "darwin":
72 from ctypes.macholib.dyld import dyld_find as _dyld_find
73 def find_library(name):
74 possible = ['lib%s.dylib' % name,
75 '%s.dylib' % name,
76 '%s.framework/%s' % (name, name)]
77 for name in possible:
78 try:
79 return _dyld_find(name)
80 except ValueError:
81 continue
82 return None
83
84elif os.name == "posix":
85 # Andreas Degert's find functions, using gcc, /sbin/ldconfig, objdump
Thomas Wouters0e3f5912006-08-11 14:57:12 +000086 import re, tempfile, errno
Thomas Wouters477c8d52006-05-27 19:21:47 +000087
88 def _findLib_gcc(name):
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +000089 expr = r'[^\(\)\s]*lib%s\.[^\(\)\s]*' % re.escape(name)
Thomas Wouters0e3f5912006-08-11 14:57:12 +000090 fdout, ccout = tempfile.mkstemp()
91 os.close(fdout)
Matthias Klose2c7e3ee2009-01-10 17:08:25 +000092 cmd = 'if type gcc >/dev/null 2>&1; then CC=gcc; elif type cc >/dev/null 2>&1; then CC=cc;else exit 10; fi;' \
Thomas Wouters0e3f5912006-08-11 14:57:12 +000093 '$CC -Wl,-t -o ' + ccout + ' 2>&1 -l' + name
Thomas Wouters477c8d52006-05-27 19:21:47 +000094 try:
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +000095 f = os.popen(cmd)
Guido van Rossumf4d4f8b2007-12-12 20:26:00 +000096 try:
97 trace = f.read()
98 finally:
Matthias Klose2c7e3ee2009-01-10 17:08:25 +000099 rv = f.close()
Thomas Wouters477c8d52006-05-27 19:21:47 +0000100 finally:
101 try:
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000102 os.unlink(ccout)
Guido van Rossumb940e112007-01-10 16:19:56 +0000103 except OSError as e:
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000104 if e.errno != errno.ENOENT:
105 raise
Matthias Klose2c7e3ee2009-01-10 17:08:25 +0000106 if rv == 10:
Benjamin Peterson17bf0822009-01-10 22:37:11 +0000107 raise OSError('gcc or cc command not found')
Thomas Wouters477c8d52006-05-27 19:21:47 +0000108 res = re.search(expr, trace)
109 if not res:
110 return None
111 return res.group(0)
112
Thomas Wouters1b7f8912007-09-19 03:06:30 +0000113
114 if sys.platform == "sunos5":
115 # use /usr/ccs/bin/dump on solaris
116 def _get_soname(f):
117 if not f:
118 return None
119 cmd = "/usr/ccs/bin/dump -Lpv 2>/dev/null " + f
Guido van Rossumf4d4f8b2007-12-12 20:26:00 +0000120 f = os.popen(cmd)
121 try:
122 data = f.read()
123 finally:
124 f.close()
125 res = re.search(r'\[.*\]\sSONAME\s+([^\s]+)', data)
Thomas Wouters1b7f8912007-09-19 03:06:30 +0000126 if not res:
127 return None
128 return res.group(1)
129 else:
130 def _get_soname(f):
131 # assuming GNU binutils / ELF
132 if not f:
133 return None
Matthias Klose2c7e3ee2009-01-10 17:08:25 +0000134 cmd = 'if ! type objdump >/dev/null 2>&1; then exit 10; fi;' \
135 "objdump -p -j .dynamic 2>/dev/null " + f
136 f = os.popen(cmd)
137 dump = f.read()
138 rv = f.close()
139 if rv == 10:
Benjamin Peterson17bf0822009-01-10 22:37:11 +0000140 raise OSError('objdump command not found')
Guido van Rossumf4d4f8b2007-12-12 20:26:00 +0000141 f = os.popen(cmd)
142 try:
143 data = f.read()
144 finally:
145 f.close()
146 res = re.search(r'\sSONAME\s+([^\s]+)', data)
Thomas Wouters1b7f8912007-09-19 03:06:30 +0000147 if not res:
148 return None
149 return res.group(1)
Thomas Wouters477c8d52006-05-27 19:21:47 +0000150
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000151 if (sys.platform.startswith("freebsd")
152 or sys.platform.startswith("openbsd")
153 or sys.platform.startswith("dragonfly")):
154
155 def _num_version(libname):
156 # "libxyz.so.MAJOR.MINOR" => [ MAJOR, MINOR ]
157 parts = libname.split(".")
158 nums = []
159 try:
160 while parts:
161 nums.insert(0, int(parts.pop()))
162 except ValueError:
163 pass
Christian Heimesa37d4c62007-12-04 23:02:19 +0000164 return nums or [ sys.maxsize ]
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000165
166 def find_library(name):
167 ename = re.escape(name)
168 expr = r':-l%s\.\S+ => \S*/(lib%s\.\S+)' % (ename, ename)
Guido van Rossumf4d4f8b2007-12-12 20:26:00 +0000169 f = os.popen('/sbin/ldconfig -r 2>/dev/null')
170 try:
171 data = f.read()
172 finally:
173 f.close()
174 res = re.findall(expr, data)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000175 if not res:
176 return _get_soname(_findLib_gcc(name))
Raymond Hettingerd4cb56d2008-01-30 02:55:10 +0000177 res.sort(key=_num_version)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000178 return res[-1]
179
180 else:
181
182 def _findLib_ldconfig(name):
183 # XXX assuming GLIBC's ldconfig (with option -p)
184 expr = r'/[^\(\)\s]*lib%s\.[^\(\)\s]*' % re.escape(name)
Guido van Rossumf4d4f8b2007-12-12 20:26:00 +0000185 f = os.popen('/sbin/ldconfig -p 2>/dev/null')
186 try:
187 data = f.read()
188 finally:
189 f.close()
190 res = re.search(expr, data)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000191 if not res:
192 # Hm, this works only for libs needed by the python executable.
193 cmd = 'ldd %s 2>/dev/null' % sys.executable
Guido van Rossumf4d4f8b2007-12-12 20:26:00 +0000194 f = os.popen(cmd)
195 try:
196 data = f.read()
197 finally:
198 f.close()
199 res = re.search(expr, data)
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000200 if not res:
201 return None
202 return res.group(0)
203
Matthias Klose2c7e3ee2009-01-10 17:08:25 +0000204 def _findSoname_ldconfig(name):
205 import struct
206 if struct.calcsize('l') == 4:
207 machine = os.uname()[4] + '-32'
208 else:
209 machine = os.uname()[4] + '-64'
210 mach_map = {
211 'x86_64-64': 'libc6,x86-64',
212 'ppc64-64': 'libc6,64bit',
213 'sparc64-64': 'libc6,64bit',
214 's390x-64': 'libc6,64bit',
215 'ia64-64': 'libc6,IA-64',
216 }
217 abi_type = mach_map.get(machine, 'libc6')
218
219 # XXX assuming GLIBC's ldconfig (with option -p)
220 expr = r'(\S+)\s+\((%s(?:, OS ABI:[^\)]*)?)\)[^/]*(/[^\(\)\s]*lib%s\.[^\(\)\s]*)' \
221 % (abi_type, re.escape(name))
Eric Smith81fd8042009-09-18 13:23:13 +0000222 f = os.popen('/sbin/ldconfig -p 2>/dev/null')
223 try:
224 data = f.read()
225 finally:
226 f.close()
227 res = re.search(expr, data)
Matthias Klose2c7e3ee2009-01-10 17:08:25 +0000228 if not res:
229 return None
230 return res.group(1)
231
Thomas Woutersfc7bb8c2007-01-15 15:49:28 +0000232 def find_library(name):
Matthias Klose2c7e3ee2009-01-10 17:08:25 +0000233 return _findSoname_ldconfig(name) or _get_soname(_findLib_gcc(name))
Thomas Wouters477c8d52006-05-27 19:21:47 +0000234
235################################################################
236# test code
237
238def test():
239 from ctypes import cdll
240 if os.name == "nt":
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000241 print(cdll.msvcrt)
242 print(cdll.load("msvcrt"))
243 print(find_library("msvcrt"))
Thomas Wouters477c8d52006-05-27 19:21:47 +0000244
245 if os.name == "posix":
246 # find and load_version
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000247 print(find_library("m"))
248 print(find_library("c"))
249 print(find_library("bz2"))
Thomas Wouters477c8d52006-05-27 19:21:47 +0000250
251 # getattr
252## print cdll.m
253## print cdll.bz2
254
255 # load
256 if sys.platform == "darwin":
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000257 print(cdll.LoadLibrary("libm.dylib"))
258 print(cdll.LoadLibrary("libcrypto.dylib"))
259 print(cdll.LoadLibrary("libSystem.dylib"))
260 print(cdll.LoadLibrary("System.framework/System"))
Thomas Wouters477c8d52006-05-27 19:21:47 +0000261 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000262 print(cdll.LoadLibrary("libm.so"))
263 print(cdll.LoadLibrary("libcrypt.so"))
264 print(find_library("crypt"))
Thomas Wouters477c8d52006-05-27 19:21:47 +0000265
266if __name__ == "__main__":
267 test()