blob: 302293ac25fc9947761147e9e5c6caf3bfedb4ae [file] [log] [blame]
Greg Ward7c6395a2000-06-21 03:33:03 +00001"""distutils.cygwinccompiler
2
Greg Wardf34506a2000-06-29 22:57:55 +00003Provides the CygwinCCompiler class, a subclass of UnixCCompiler that
4handles the Cygwin port of the GNU C compiler to Windows. It also contains
5the Mingw32CCompiler class which handles the mingw32 port of GCC (same as
6cygwin in no-cygwin mode).
Greg Ward7c6395a2000-06-21 03:33:03 +00007"""
8
Greg Wardbf5c7092000-08-02 01:31:56 +00009# problems:
10#
11# * if you use a msvc compiled python version (1.5.2)
12# 1. you have to insert a __GNUC__ section in its config.h
13# 2. you have to generate a import library for its dll
14# - create a def-file for python??.dll
15# - create a import library using
16# dlltool --dllname python15.dll --def python15.def \
17# --output-lib libpython15.a
18#
19# see also http://starship.python.net/crew/kernr/mingw32/Notes.html
20#
Fred Drakeb94b8492001-12-06 20:51:35 +000021# * We put export_symbols in a def-file, and don't use
Greg Wardbf5c7092000-08-02 01:31:56 +000022# --export-all-symbols because it doesn't worked reliable in some
23# tested configurations. And because other windows compilers also
24# need their symbols specified this no serious problem.
25#
26# tested configurations:
Fred Drakeb94b8492001-12-06 20:51:35 +000027#
28# * cygwin gcc 2.91.57/ld 2.9.4/dllwrap 0.2.4 works
Greg Wardbf5c7092000-08-02 01:31:56 +000029# (after patching python's config.h and for C++ some other include files)
30# see also http://starship.python.net/crew/kernr/mingw32/Notes.html
Fred Drakeb94b8492001-12-06 20:51:35 +000031# * mingw32 gcc 2.95.2/ld 2.9.4/dllwrap 0.2.4 works
32# (ld doesn't support -shared, so we use dllwrap)
Greg Wardbf5c7092000-08-02 01:31:56 +000033# * cygwin gcc 2.95.2/ld 2.10.90/dllwrap 2.10.90 works now
34# - its dllwrap doesn't work, there is a bug in binutils 2.10.90
Greg Ward7483d682000-09-01 01:24:31 +000035# see also http://sources.redhat.com/ml/cygwin/2000-06/msg01274.html
Fred Drakeb94b8492001-12-06 20:51:35 +000036# - using gcc -mdll instead dllwrap doesn't work without -static because
Greg Wardbf5c7092000-08-02 01:31:56 +000037# it tries to link against dlls instead their import libraries. (If
38# it finds the dll first.)
Fred Drakeb94b8492001-12-06 20:51:35 +000039# By specifying -static we force ld to link against the import libraries,
40# this is windows standard and there are normally not the necessary symbols
Greg Wardbf5c7092000-08-02 01:31:56 +000041# in the dlls.
Fred Drakeb94b8492001-12-06 20:51:35 +000042# *** only the version of June 2000 shows these problems
Greg Wardbf5c7092000-08-02 01:31:56 +000043
Greg Ward7c6395a2000-06-21 03:33:03 +000044# created 2000/05/05, Rene Liebscher
45
46__revision__ = "$Id$"
47
Greg Wardb1dceae2000-08-13 00:43:56 +000048import os,sys,copy
Greg Ward42406482000-09-27 02:08:14 +000049from distutils.ccompiler import gen_preprocess_options, gen_lib_options
Greg Ward7c6395a2000-06-21 03:33:03 +000050from distutils.unixccompiler import UnixCCompiler
Greg Wardbf5c7092000-08-02 01:31:56 +000051from distutils.file_util import write_file
Greg Ward42406482000-09-27 02:08:14 +000052from distutils.errors import DistutilsExecError, CompileError, UnknownFileError
Jeremy Hyltoncd8a1142002-06-04 20:14:43 +000053from distutils import log
Greg Ward7c6395a2000-06-21 03:33:03 +000054
Greg Ward7c6395a2000-06-21 03:33:03 +000055class CygwinCCompiler (UnixCCompiler):
56
57 compiler_type = 'cygwin'
Greg Wardb1dceae2000-08-13 00:43:56 +000058 obj_extension = ".o"
59 static_lib_extension = ".a"
60 shared_lib_extension = ".dll"
61 static_lib_format = "lib%s%s"
62 shared_lib_format = "%s%s"
63 exe_extension = ".exe"
Fred Drakeb94b8492001-12-06 20:51:35 +000064
Jeremy Hylton1bba31d2002-06-13 17:28:18 +000065 def __init__ (self, verbose=0, dry_run=0, force=0):
Greg Ward7c6395a2000-06-21 03:33:03 +000066
67 UnixCCompiler.__init__ (self, verbose, dry_run, force)
68
Greg Warde8e9d112000-08-13 01:18:55 +000069 (status, details) = check_config_h()
70 self.debug_print("Python's GCC status: %s (details: %s)" %
71 (status, details))
72 if status is not CONFIG_H_OK:
Greg Wardbf5c7092000-08-02 01:31:56 +000073 self.warn(
Jeremy Hylton1bba31d2002-06-13 17:28:18 +000074 "Python's pyconfig.h doesn't seem to support your compiler. "
75 "Reason: %s. "
76 "Compiling may fail because of undefined preprocessor macros."
77 % details)
Fred Drakeb94b8492001-12-06 20:51:35 +000078
Jeremy Hylton1bba31d2002-06-13 17:28:18 +000079 self.gcc_version, self.ld_version, self.dllwrap_version = \
Greg Wardbf5c7092000-08-02 01:31:56 +000080 get_versions()
Greg Wardb1dceae2000-08-13 00:43:56 +000081 self.debug_print(self.compiler_type + ": gcc %s, ld %s, dllwrap %s\n" %
Fred Drakeb94b8492001-12-06 20:51:35 +000082 (self.gcc_version,
83 self.ld_version,
Greg Wardbf5c7092000-08-02 01:31:56 +000084 self.dllwrap_version) )
85
Fred Drakeb94b8492001-12-06 20:51:35 +000086 # ld_version >= "2.10.90" should also be able to use
Greg Wardbf5c7092000-08-02 01:31:56 +000087 # gcc -mdll instead of dllwrap
Fred Drakeb94b8492001-12-06 20:51:35 +000088 # Older dllwraps had own version numbers, newer ones use the
Greg Wardbf5c7092000-08-02 01:31:56 +000089 # same as the rest of binutils ( also ld )
90 # dllwrap 2.10.90 is buggy
Fred Drakeb94b8492001-12-06 20:51:35 +000091 if self.ld_version >= "2.10.90":
Greg Ward42406482000-09-27 02:08:14 +000092 self.linker_dll = "gcc"
Greg Wardbf5c7092000-08-02 01:31:56 +000093 else:
Greg Ward42406482000-09-27 02:08:14 +000094 self.linker_dll = "dllwrap"
Greg Wardbf5c7092000-08-02 01:31:56 +000095
Greg Wardf34506a2000-06-29 22:57:55 +000096 # Hard-code GCC because that's what this is all about.
97 # XXX optimization, warnings etc. should be customizable.
Greg Wardbf5c7092000-08-02 01:31:56 +000098 self.set_executables(compiler='gcc -mcygwin -O -Wall',
99 compiler_so='gcc -mcygwin -mdll -O -Wall',
100 linker_exe='gcc -mcygwin',
101 linker_so=('%s -mcygwin -mdll -static' %
Greg Ward42406482000-09-27 02:08:14 +0000102 self.linker_dll))
Greg Ward7c6395a2000-06-21 03:33:03 +0000103
Fred Drakeb94b8492001-12-06 20:51:35 +0000104 # cygwin and mingw32 need different sets of libraries
Greg Wardbf5c7092000-08-02 01:31:56 +0000105 if self.gcc_version == "2.91.57":
106 # cygwin shouldn't need msvcrt, but without the dlls will crash
107 # (gcc version 2.91.57) -- perhaps something about initialization
108 self.dll_libraries=["msvcrt"]
Fred Drakeb94b8492001-12-06 20:51:35 +0000109 self.warn(
Greg Wardbf5c7092000-08-02 01:31:56 +0000110 "Consider upgrading to a newer version of gcc")
111 else:
112 self.dll_libraries=[]
Fred Drakeb94b8492001-12-06 20:51:35 +0000113
Greg Ward7c6395a2000-06-21 03:33:03 +0000114 # __init__ ()
115
Greg Ward42406482000-09-27 02:08:14 +0000116 # not much different of the compile method in UnixCCompiler,
117 # but we have to insert some lines in the middle of it, so
118 # we put here a adapted version of it.
Fred Drakeb94b8492001-12-06 20:51:35 +0000119 # (If we would call compile() in the base class, it would do some
Greg Ward42406482000-09-27 02:08:14 +0000120 # initializations a second time, this is why all is done here.)
Jeremy Hylton1bba31d2002-06-13 17:28:18 +0000121 def compile(self, sources,
122 output_dir=None, macros=None, include_dirs=None, debug=0,
123 extra_preargs=None, extra_postargs=None, depends=None):
124
125 macros, objects, extra_postargs, pp_opts, build = \
126 self._setup_compile(output_dir, macros, include_dirs, sources,
127 depends, extra_postargs)
128 cc_args = self._get_cc_args(pp_opts, debug, extra_preargs)
Greg Ward42406482000-09-27 02:08:14 +0000129
Jeremy Hylton1bba31d2002-06-13 17:28:18 +0000130 for obj, (src, ext) in build.items():
131 if ext == '.rc' or ext == '.res':
132 # gcc needs '.res' and '.rc' compiled to object files !!!
133 try:
134 self.spawn (["windres","-i",src,"-o",obj])
135 except DistutilsExecError, msg:
136 raise CompileError, msg
137 else: # for other files use the C-compiler
138 try:
139 self.spawn (self.compiler_so + cc_args +
140 [src, '-o', obj] +
141 extra_postargs)
142 except DistutilsExecError, msg:
143 raise CompileError, msg
Greg Ward42406482000-09-27 02:08:14 +0000144
145 # Return *all* object filenames, not just the ones we just built.
146 return objects
147
Greg Ward42406482000-09-27 02:08:14 +0000148 def link (self,
149 target_desc,
150 objects,
151 output_filename,
152 output_dir=None,
153 libraries=None,
154 library_dirs=None,
155 runtime_library_dirs=None,
156 export_symbols=None,
157 debug=0,
158 extra_preargs=None,
159 extra_postargs=None,
160 build_temp=None):
Fred Drakeb94b8492001-12-06 20:51:35 +0000161
Greg Wardb1dceae2000-08-13 00:43:56 +0000162 # use separate copies, so we can modify the lists
163 extra_preargs = copy.copy(extra_preargs or [])
164 libraries = copy.copy(libraries or [])
Greg Ward42406482000-09-27 02:08:14 +0000165 objects = copy.copy(objects or [])
Fred Drakeb94b8492001-12-06 20:51:35 +0000166
Greg Wardbf5c7092000-08-02 01:31:56 +0000167 # Additional libraries
Greg Wardf34506a2000-06-29 22:57:55 +0000168 libraries.extend(self.dll_libraries)
Greg Wardbf5c7092000-08-02 01:31:56 +0000169
Greg Ward42406482000-09-27 02:08:14 +0000170 # handle export symbols by creating a def-file
171 # with executables this only works with gcc/ld as linker
172 if ((export_symbols is not None) and
173 (target_desc != self.EXECUTABLE or self.linker_dll == "gcc")):
174 # (The linker doesn't do anything if output is up-to-date.
175 # So it would probably better to check if we really need this,
Fred Drakeb94b8492001-12-06 20:51:35 +0000176 # but for this we had to insert some unchanged parts of
177 # UnixCCompiler, and this is not what we want.)
Greg Ward42406482000-09-27 02:08:14 +0000178
Fred Drakeb94b8492001-12-06 20:51:35 +0000179 # we want to put some files in the same directory as the
Greg Ward42406482000-09-27 02:08:14 +0000180 # object files are, build_temp doesn't help much
181 # where are the object files
182 temp_dir = os.path.dirname(objects[0])
183 # name of dll to give the helper files the same base name
184 (dll_name, dll_extension) = os.path.splitext(
185 os.path.basename(output_filename))
186
187 # generate the filenames for these files
Greg Wardbf5c7092000-08-02 01:31:56 +0000188 def_file = os.path.join(temp_dir, dll_name + ".def")
Greg Ward42406482000-09-27 02:08:14 +0000189 lib_file = os.path.join(temp_dir, 'lib' + dll_name + ".a")
Fred Drakeb94b8492001-12-06 20:51:35 +0000190
Greg Ward42406482000-09-27 02:08:14 +0000191 # Generate .def file
Greg Wardbf5c7092000-08-02 01:31:56 +0000192 contents = [
193 "LIBRARY %s" % os.path.basename(output_filename),
194 "EXPORTS"]
Greg Ward7c6395a2000-06-21 03:33:03 +0000195 for sym in export_symbols:
Greg Wardbf5c7092000-08-02 01:31:56 +0000196 contents.append(sym)
197 self.execute(write_file, (def_file, contents),
198 "writing %s" % def_file)
199
Greg Ward42406482000-09-27 02:08:14 +0000200 # next add options for def-file and to creating import libraries
201
202 # dllwrap uses different options than gcc/ld
203 if self.linker_dll == "dllwrap":
Jeremy Hyltona2f99892002-06-04 20:26:44 +0000204 extra_preargs.extend(["--output-lib", lib_file])
Greg Wardbf5c7092000-08-02 01:31:56 +0000205 # for dllwrap we have to use a special option
Greg Ward42406482000-09-27 02:08:14 +0000206 extra_preargs.extend(["--def", def_file])
207 # we use gcc/ld here and can be sure ld is >= 2.9.10
208 else:
209 # doesn't work: bfd_close build\...\libfoo.a: Invalid operation
210 #extra_preargs.extend(["-Wl,--out-implib,%s" % lib_file])
Jeremy Hyltona2f99892002-06-04 20:26:44 +0000211 # for gcc/ld the def-file is specified as any object files
Greg Ward42406482000-09-27 02:08:14 +0000212 objects.append(def_file)
213
214 #end: if ((export_symbols is not None) and
Fred Drake132dce22000-12-12 23:11:42 +0000215 # (target_desc != self.EXECUTABLE or self.linker_dll == "gcc")):
Fred Drakeb94b8492001-12-06 20:51:35 +0000216
Greg Wardf34506a2000-06-29 22:57:55 +0000217 # who wants symbols and a many times larger output file
Fred Drakeb94b8492001-12-06 20:51:35 +0000218 # should explicitly switch the debug mode on
Greg Wardbf5c7092000-08-02 01:31:56 +0000219 # otherwise we let dllwrap/ld strip the output file
Fred Drakeb94b8492001-12-06 20:51:35 +0000220 # (On my machine: 10KB < stripped_file < ??100KB
Greg Ward42406482000-09-27 02:08:14 +0000221 # unstripped_file = stripped_file + XXX KB
Fred Drakeb94b8492001-12-06 20:51:35 +0000222 # ( XXX=254 for a typical python extension))
223 if not debug:
224 extra_preargs.append("-s")
225
Greg Ward42406482000-09-27 02:08:14 +0000226 UnixCCompiler.link(self,
227 target_desc,
228 objects,
229 output_filename,
230 output_dir,
231 libraries,
232 library_dirs,
233 runtime_library_dirs,
234 None, # export_symbols, we do this in our def-file
235 debug,
236 extra_preargs,
237 extra_postargs,
238 build_temp)
Fred Drakeb94b8492001-12-06 20:51:35 +0000239
Greg Ward42406482000-09-27 02:08:14 +0000240 # link ()
241
242 # -- Miscellaneous methods -----------------------------------------
243
244 # overwrite the one from CCompiler to support rc and res-files
245 def object_filenames (self,
246 source_filenames,
247 strip_dir=0,
248 output_dir=''):
249 if output_dir is None: output_dir = ''
250 obj_names = []
251 for src_name in source_filenames:
252 # use normcase to make sure '.rc' is really '.rc' and not '.RC'
253 (base, ext) = os.path.splitext (os.path.normcase(src_name))
254 if ext not in (self.src_extensions + ['.rc','.res']):
255 raise UnknownFileError, \
256 "unknown file type '%s' (from '%s')" % \
257 (ext, src_name)
258 if strip_dir:
259 base = os.path.basename (base)
260 if ext == '.res' or ext == '.rc':
261 # these need to be compiled to object files
Fred Drakeb94b8492001-12-06 20:51:35 +0000262 obj_names.append (os.path.join (output_dir,
Greg Ward42406482000-09-27 02:08:14 +0000263 base + ext + self.obj_extension))
264 else:
265 obj_names.append (os.path.join (output_dir,
266 base + self.obj_extension))
267 return obj_names
268
269 # object_filenames ()
Greg Ward7c6395a2000-06-21 03:33:03 +0000270
271# class CygwinCCompiler
272
Greg Wardf34506a2000-06-29 22:57:55 +0000273
Greg Ward7c6395a2000-06-21 03:33:03 +0000274# the same as cygwin plus some additional parameters
275class Mingw32CCompiler (CygwinCCompiler):
276
277 compiler_type = 'mingw32'
278
279 def __init__ (self,
280 verbose=0,
281 dry_run=0,
282 force=0):
283
284 CygwinCCompiler.__init__ (self, verbose, dry_run, force)
Fred Drakeb94b8492001-12-06 20:51:35 +0000285
Greg Wardbf5c7092000-08-02 01:31:56 +0000286 # A real mingw32 doesn't need to specify a different entry point,
287 # but cygwin 2.91.57 in no-cygwin-mode needs it.
288 if self.gcc_version <= "2.91.57":
289 entry_point = '--entry _DllMain@12'
290 else:
291 entry_point = ''
Greg Ward7c6395a2000-06-21 03:33:03 +0000292
Greg Wardf34506a2000-06-29 22:57:55 +0000293 self.set_executables(compiler='gcc -mno-cygwin -O -Wall',
Greg Wardbf5c7092000-08-02 01:31:56 +0000294 compiler_so='gcc -mno-cygwin -mdll -O -Wall',
Greg Wardf34506a2000-06-29 22:57:55 +0000295 linker_exe='gcc -mno-cygwin',
Fred Drakeb94b8492001-12-06 20:51:35 +0000296 linker_so='%s -mno-cygwin -mdll -static %s'
Greg Ward42406482000-09-27 02:08:14 +0000297 % (self.linker_dll, entry_point))
Greg Wardbf5c7092000-08-02 01:31:56 +0000298 # Maybe we should also append -mthreads, but then the finished
299 # dlls need another dll (mingwm10.dll see Mingw32 docs)
Fred Drakeb94b8492001-12-06 20:51:35 +0000300 # (-mthreads: Support thread-safe exception handling on `Mingw32')
301
302 # no additional libraries needed
Greg Wardbf5c7092000-08-02 01:31:56 +0000303 self.dll_libraries=[]
Fred Drakeb94b8492001-12-06 20:51:35 +0000304
Greg Ward7c6395a2000-06-21 03:33:03 +0000305 # __init__ ()
Greg Wardbf5c7092000-08-02 01:31:56 +0000306
Greg Ward7c6395a2000-06-21 03:33:03 +0000307# class Mingw32CCompiler
Greg Wardbf5c7092000-08-02 01:31:56 +0000308
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000309# Because these compilers aren't configured in Python's pyconfig.h file by
Greg Wardbf5c7092000-08-02 01:31:56 +0000310# default, we should at least warn the user if he is using a unmodified
311# version.
312
Greg Warde8e9d112000-08-13 01:18:55 +0000313CONFIG_H_OK = "ok"
314CONFIG_H_NOTOK = "not ok"
315CONFIG_H_UNCERTAIN = "uncertain"
316
Greg Wardbf5c7092000-08-02 01:31:56 +0000317def check_config_h():
Greg Warde8e9d112000-08-13 01:18:55 +0000318
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000319 """Check if the current Python installation (specifically, pyconfig.h)
Greg Warde8e9d112000-08-13 01:18:55 +0000320 appears amenable to building extensions with GCC. Returns a tuple
321 (status, details), where 'status' is one of the following constants:
322 CONFIG_H_OK
323 all is well, go ahead and compile
324 CONFIG_H_NOTOK
325 doesn't look good
326 CONFIG_H_UNCERTAIN
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000327 not sure -- unable to read pyconfig.h
Greg Warde8e9d112000-08-13 01:18:55 +0000328 'details' is a human-readable string explaining the situation.
329
330 Note there are two ways to conclude "OK": either 'sys.version' contains
331 the string "GCC" (implying that this Python was built with GCC), or the
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000332 installed "pyconfig.h" contains the string "__GNUC__".
Greg Wardbf5c7092000-08-02 01:31:56 +0000333 """
Greg Warde8e9d112000-08-13 01:18:55 +0000334
335 # XXX since this function also checks sys.version, it's not strictly a
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000336 # "pyconfig.h" check -- should probably be renamed...
Greg Wardbf5c7092000-08-02 01:31:56 +0000337
338 from distutils import sysconfig
Andrew M. Kuchling6e9c0ba2001-03-22 03:50:09 +0000339 import string
Greg Wardbf5c7092000-08-02 01:31:56 +0000340 # if sys.version contains GCC then python was compiled with
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000341 # GCC, and the pyconfig.h file should be OK
Andrew M. Kuchlingac20f772001-03-22 03:48:31 +0000342 if string.find(sys.version,"GCC") >= 0:
Greg Warde8e9d112000-08-13 01:18:55 +0000343 return (CONFIG_H_OK, "sys.version mentions 'GCC'")
Fred Drakeb94b8492001-12-06 20:51:35 +0000344
Greg Warde8e9d112000-08-13 01:18:55 +0000345 fn = sysconfig.get_config_h_filename()
Greg Wardbf5c7092000-08-02 01:31:56 +0000346 try:
347 # It would probably better to read single lines to search.
Fred Drakeb94b8492001-12-06 20:51:35 +0000348 # But we do this only once, and it is fast enough
Greg Warde8e9d112000-08-13 01:18:55 +0000349 f = open(fn)
350 s = f.read()
Greg Wardbf5c7092000-08-02 01:31:56 +0000351 f.close()
Fred Drakeb94b8492001-12-06 20:51:35 +0000352
Greg Warde8e9d112000-08-13 01:18:55 +0000353 except IOError, exc:
Greg Wardbf5c7092000-08-02 01:31:56 +0000354 # if we can't read this file, we cannot say it is wrong
355 # the compiler will complain later about this file as missing
Greg Warde8e9d112000-08-13 01:18:55 +0000356 return (CONFIG_H_UNCERTAIN,
357 "couldn't read '%s': %s" % (fn, exc.strerror))
358
359 else:
Martin v. Löwis4f1cd8b2001-07-26 13:41:06 +0000360 # "pyconfig.h" contains an "#ifdef __GNUC__" or something similar
Andrew M. Kuchlingac20f772001-03-22 03:48:31 +0000361 if string.find(s,"__GNUC__") >= 0:
Greg Warde8e9d112000-08-13 01:18:55 +0000362 return (CONFIG_H_OK, "'%s' mentions '__GNUC__'" % fn)
363 else:
364 return (CONFIG_H_NOTOK, "'%s' does not mention '__GNUC__'" % fn)
365
366
Greg Wardbf5c7092000-08-02 01:31:56 +0000367
368def get_versions():
369 """ Try to find out the versions of gcc, ld and dllwrap.
370 If not possible it returns None for it.
371 """
372 from distutils.version import StrictVersion
373 from distutils.spawn import find_executable
374 import re
Fred Drakeb94b8492001-12-06 20:51:35 +0000375
Greg Wardbf5c7092000-08-02 01:31:56 +0000376 gcc_exe = find_executable('gcc')
377 if gcc_exe:
378 out = os.popen(gcc_exe + ' -dumpversion','r')
379 out_string = out.read()
380 out.close()
381 result = re.search('(\d+\.\d+\.\d+)',out_string)
382 if result:
383 gcc_version = StrictVersion(result.group(1))
384 else:
385 gcc_version = None
386 else:
387 gcc_version = None
388 ld_exe = find_executable('ld')
389 if ld_exe:
390 out = os.popen(ld_exe + ' -v','r')
391 out_string = out.read()
392 out.close()
393 result = re.search('(\d+\.\d+\.\d+)',out_string)
394 if result:
395 ld_version = StrictVersion(result.group(1))
396 else:
397 ld_version = None
398 else:
399 ld_version = None
400 dllwrap_exe = find_executable('dllwrap')
401 if dllwrap_exe:
402 out = os.popen(dllwrap_exe + ' --version','r')
403 out_string = out.read()
404 out.close()
405 result = re.search(' (\d+\.\d+\.\d+)',out_string)
406 if result:
407 dllwrap_version = StrictVersion(result.group(1))
408 else:
409 dllwrap_version = None
410 else:
411 dllwrap_version = None
412 return (gcc_version, ld_version, dllwrap_version)