Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 1 | """distutils.ccompiler |
| 2 | |
| 3 | Contains MSVCCompiler, an implementation of the abstract CCompiler class |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 4 | for the Microsoft Visual Studio.""" |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 5 | |
| 6 | |
| 7 | # created 1999/08/19, Perry Stoll |
| 8 | # |
Greg Ward | 3ce77fd | 2000-03-02 01:49:45 +0000 | [diff] [blame] | 9 | __revision__ = "$Id$" |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 10 | |
| 11 | import os |
| 12 | import sys |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 13 | import string |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 14 | from distutils.errors import * |
Greg Ward | 3d50b90 | 1999-09-08 02:36:01 +0000 | [diff] [blame] | 15 | from distutils.ccompiler import \ |
| 16 | CCompiler, gen_preprocess_options, gen_lib_options |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 17 | |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 18 | |
| 19 | def get_devstudio_versions (): |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 20 | """Get list of devstudio versions from the Windows registry. Return a |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 21 | list of strings containing version numbers; the list will be |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 22 | empty if we were unable to access the registry (eg. couldn't import |
| 23 | a registry-access module) or the appropriate registry keys weren't |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 24 | found.""" |
| 25 | |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 26 | try: |
| 27 | import win32api |
| 28 | import win32con |
| 29 | except ImportError: |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 30 | return [] |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 31 | |
| 32 | K = 'Software\\Microsoft\\Devstudio' |
| 33 | L = [] |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 34 | for base in (win32con.HKEY_CLASSES_ROOT, |
| 35 | win32con.HKEY_LOCAL_MACHINE, |
| 36 | win32con.HKEY_CURRENT_USER, |
| 37 | win32con.HKEY_USERS): |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 38 | try: |
| 39 | k = win32api.RegOpenKeyEx(base,K) |
| 40 | i = 0 |
| 41 | while 1: |
| 42 | try: |
| 43 | p = win32api.RegEnumKey(k,i) |
| 44 | if p[0] in '123456789' and p not in L: |
| 45 | L.append(p) |
| 46 | except win32api.error: |
| 47 | break |
| 48 | i = i + 1 |
| 49 | except win32api.error: |
| 50 | pass |
| 51 | L.sort() |
| 52 | L.reverse() |
| 53 | return L |
| 54 | |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 55 | # get_devstudio_versions () |
| 56 | |
| 57 | |
| 58 | def get_msvc_paths (path, version='6.0', platform='x86'): |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 59 | """Get a list of devstudio directories (include, lib or path). Return |
| 60 | a list of strings; will be empty list if unable to access the |
| 61 | registry or appropriate registry keys not found.""" |
| 62 | |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 63 | try: |
| 64 | import win32api |
| 65 | import win32con |
| 66 | except ImportError: |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 67 | return [] |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 68 | |
| 69 | L = [] |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 70 | if path=='lib': |
| 71 | path= 'Library' |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 72 | path = string.upper(path + ' Dirs') |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 73 | K = ('Software\\Microsoft\\Devstudio\\%s\\' + |
| 74 | 'Build System\\Components\\Platforms\\Win32 (%s)\\Directories') % \ |
| 75 | (version,platform) |
| 76 | for base in (win32con.HKEY_CLASSES_ROOT, |
| 77 | win32con.HKEY_LOCAL_MACHINE, |
| 78 | win32con.HKEY_CURRENT_USER, |
| 79 | win32con.HKEY_USERS): |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 80 | try: |
| 81 | k = win32api.RegOpenKeyEx(base,K) |
| 82 | i = 0 |
| 83 | while 1: |
| 84 | try: |
| 85 | (p,v,t) = win32api.RegEnumValue(k,i) |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 86 | if string.upper(p) == path: |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 87 | V = string.split(v,';') |
| 88 | for v in V: |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 89 | if v == '' or v in L: continue |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 90 | L.append(v) |
| 91 | break |
| 92 | i = i + 1 |
| 93 | except win32api.error: |
| 94 | break |
| 95 | except win32api.error: |
| 96 | pass |
| 97 | return L |
| 98 | |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 99 | # get_msvc_paths() |
| 100 | |
| 101 | |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 102 | def find_exe (exe, version_number): |
| 103 | """Try to find an MSVC executable program 'exe' (from version |
| 104 | 'version_number' of MSVC) in several places: first, one of the MSVC |
| 105 | program search paths from the registry; next, the directories in the |
| 106 | PATH environment variable. If any of those work, return an absolute |
| 107 | path that is known to exist. If none of them work, just return the |
| 108 | original program name, 'exe'.""" |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 109 | |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 110 | for p in get_msvc_paths ('path', version_number): |
| 111 | fn = os.path.join (os.path.abspath(p), exe) |
| 112 | if os.path.isfile(fn): |
| 113 | return fn |
| 114 | |
| 115 | # didn't find it; try existing path |
| 116 | for p in string.split (os.environ['Path'],';'): |
| 117 | fn = os.path.join(os.path.abspath(p),exe) |
| 118 | if os.path.isfile(fn): |
| 119 | return fn |
| 120 | |
| 121 | return exe # last desperate hope |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 122 | |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 123 | |
Greg Ward | 5de8cee | 2000-02-11 02:52:39 +0000 | [diff] [blame] | 124 | def set_path_env_var (name, version_number): |
| 125 | """Set environment variable 'name' to an MSVC path type value obtained |
| 126 | from 'get_msvc_paths()'. This is equivalent to a SET command prior |
| 127 | to execution of spawned commands.""" |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 128 | |
Greg Ward | 5de8cee | 2000-02-11 02:52:39 +0000 | [diff] [blame] | 129 | p = get_msvc_paths (name, version_number) |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 130 | if p: |
Greg Ward | 5de8cee | 2000-02-11 02:52:39 +0000 | [diff] [blame] | 131 | os.environ[name] = string.join (p,';') |
Greg Ward | 62e3393 | 2000-02-10 02:52:42 +0000 | [diff] [blame] | 132 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 133 | |
Greg Ward | 3d50b90 | 1999-09-08 02:36:01 +0000 | [diff] [blame] | 134 | class MSVCCompiler (CCompiler) : |
| 135 | """Concrete class that implements an interface to Microsoft Visual C++, |
| 136 | as defined by the CCompiler abstract class.""" |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 137 | |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 138 | compiler_type = 'msvc' |
| 139 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 140 | def __init__ (self, |
| 141 | verbose=0, |
Greg Ward | c74138d | 1999-10-03 20:47:52 +0000 | [diff] [blame] | 142 | dry_run=0, |
| 143 | force=0): |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 144 | |
Greg Ward | c74138d | 1999-10-03 20:47:52 +0000 | [diff] [blame] | 145 | CCompiler.__init__ (self, verbose, dry_run, force) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 146 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 147 | self.add_library_dir( os.path.join( sys.exec_prefix, 'libs' ) ) |
| 148 | |
Greg Ward | 5de8cee | 2000-02-11 02:52:39 +0000 | [diff] [blame] | 149 | versions = get_devstudio_versions () |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 150 | |
Greg Ward | 5de8cee | 2000-02-11 02:52:39 +0000 | [diff] [blame] | 151 | if versions: |
| 152 | version = versions[0] # highest version |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 153 | |
Greg Ward | 5de8cee | 2000-02-11 02:52:39 +0000 | [diff] [blame] | 154 | self.cc = _find_exe("cl.exe", version) |
| 155 | self.link = _find_exe("link.exe", version) |
| 156 | set_path_env_var ('lib', version) |
| 157 | set_path_env_var ('include', version) |
| 158 | path=get_msvc_paths('path', version) |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 159 | try: |
| 160 | for p in string.split(os.environ['path'],';'): |
| 161 | path.append(p) |
| 162 | except KeyError: |
| 163 | pass |
| 164 | os.environ['path'] = string.join(path,';') |
| 165 | else: |
| 166 | # devstudio not found in the registry |
| 167 | self.cc = "cl.exe" |
| 168 | self.link = "link.exe" |
| 169 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 170 | self.preprocess_options = None |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 171 | self.compile_options = [ '/nologo', '/Ox', '/MD', '/W3' ] |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 172 | self.compile_options_debug = [ |
Greg Ward | 6998809 | 2000-02-11 02:47:15 +0000 | [diff] [blame] | 173 | '/nologo', '/Od', '/MDd', '/W3', '/Z7', '/D_DEBUG' |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 174 | ] |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 175 | |
Greg Ward | 1b9c6f7 | 2000-02-08 02:39:44 +0000 | [diff] [blame] | 176 | self.ldflags_shared = ['/DLL', '/nologo', '/INCREMENTAL:NO'] |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 177 | self.ldflags_shared_debug = [ |
| 178 | '/DLL', '/nologo', '/INCREMENTAL:no', '/pdb:None', '/DEBUG' |
| 179 | ] |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 180 | self.ldflags_static = [ '/nologo'] |
| 181 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 182 | |
| 183 | # -- Worker methods ------------------------------------------------ |
| 184 | # (must be implemented by subclasses) |
| 185 | |
| 186 | _c_extensions = [ '.c' ] |
Greg Ward | 3d50b90 | 1999-09-08 02:36:01 +0000 | [diff] [blame] | 187 | _cpp_extensions = [ '.cc', '.cpp' ] |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 188 | |
| 189 | _obj_ext = '.obj' |
Greg Ward | 3d50b90 | 1999-09-08 02:36:01 +0000 | [diff] [blame] | 190 | _exe_ext = '.exe' |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 191 | _shared_lib_ext = '.dll' |
| 192 | _static_lib_ext = '.lib' |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 193 | |
| 194 | # XXX the 'output_dir' parameter is ignored by the methods in this |
| 195 | # class! I just put it in to be consistent with CCompiler and |
| 196 | # UnixCCompiler, but someone who actually knows Visual C++ will |
| 197 | # have to make it work... |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 198 | |
| 199 | def compile (self, |
| 200 | sources, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 201 | output_dir=None, |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 202 | macros=None, |
Greg Ward | 0bdd90a | 1999-12-12 17:19:58 +0000 | [diff] [blame] | 203 | include_dirs=None, |
Greg Ward | 386b844 | 2000-02-09 02:18:39 +0000 | [diff] [blame] | 204 | debug=0, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 205 | extra_preargs=None, |
| 206 | extra_postargs=None): |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 207 | |
| 208 | if macros is None: |
| 209 | macros = [] |
Greg Ward | 0bdd90a | 1999-12-12 17:19:58 +0000 | [diff] [blame] | 210 | if include_dirs is None: |
| 211 | include_dirs = [] |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 212 | |
| 213 | objectFiles = [] |
| 214 | |
Greg Ward | 0bdd90a | 1999-12-12 17:19:58 +0000 | [diff] [blame] | 215 | base_pp_opts = \ |
| 216 | gen_preprocess_options (self.macros + macros, |
| 217 | self.include_dirs + include_dirs) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 218 | |
| 219 | base_pp_opts.append('/c') |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 220 | |
| 221 | if debug: |
| 222 | compile_options = self.compile_options_debug |
| 223 | else: |
| 224 | compile_options = self.compile_options |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 225 | |
| 226 | for srcFile in sources: |
| 227 | base,ext = os.path.splitext(srcFile) |
| 228 | objFile = base + ".obj" |
| 229 | |
| 230 | if ext in self._c_extensions: |
| 231 | fileOpt = "/Tc" |
| 232 | elif ext in self._cpp_extensions: |
| 233 | fileOpt = "/Tp" |
| 234 | |
| 235 | inputOpt = fileOpt + srcFile |
| 236 | outputOpt = "/Fo" + objFile |
| 237 | |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 238 | cc_args = compile_options + \ |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 239 | base_pp_opts + \ |
| 240 | [outputOpt, inputOpt] |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 241 | |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 242 | if extra_preargs: |
| 243 | cc_args[:0] = extra_preargs |
| 244 | if extra_postargs: |
| 245 | cc_args.extend (extra_postargs) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 246 | |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 247 | self.spawn ([self.cc] + cc_args) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 248 | objectFiles.append( objFile ) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 249 | return objectFiles |
Greg Ward | 3d50b90 | 1999-09-08 02:36:01 +0000 | [diff] [blame] | 250 | |
| 251 | |
Greg Ward | 386b844 | 2000-02-09 02:18:39 +0000 | [diff] [blame] | 252 | # XXX the signature of this method is different from CCompiler and |
| 253 | # UnixCCompiler -- but those extra parameters (libraries, library_dirs) |
| 254 | # are actually used. So: are they really *needed*, or can they be |
| 255 | # ditched? If needed, the CCompiler API will have to change... |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 256 | def link_static_lib (self, |
| 257 | objects, |
| 258 | output_libname, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 259 | output_dir=None, |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 260 | libraries=None, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 261 | library_dirs=None, |
Greg Ward | 386b844 | 2000-02-09 02:18:39 +0000 | [diff] [blame] | 262 | debug=0, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 263 | extra_preargs=None, |
| 264 | extra_postargs=None): |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 265 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 266 | if libraries is None: |
| 267 | libraries = [] |
| 268 | if library_dirs is None: |
| 269 | library_dirs = [] |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 270 | |
Greg Ward | 3d50b90 | 1999-09-08 02:36:01 +0000 | [diff] [blame] | 271 | lib_opts = gen_lib_options (self.libraries + libraries, |
| 272 | self.library_dirs + library_dirs, |
| 273 | "%s.lib", "/LIBPATH:%s") |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 274 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 275 | ld_args = self.ldflags_static + lib_opts + \ |
| 276 | objects + ['/OUT:' + output_filename] |
Greg Ward | 386b844 | 2000-02-09 02:18:39 +0000 | [diff] [blame] | 277 | if debug: |
| 278 | pass # XXX what goes here? |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 279 | if extra_preargs: |
| 280 | ld_args[:0] = extra_preargs |
| 281 | if extra_postargs: |
| 282 | ld_args.extend (extra_postargs) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 283 | |
| 284 | self.spawn ( [ self.link ] + ld_args ) |
| 285 | |
| 286 | |
| 287 | def link_shared_lib (self, |
| 288 | objects, |
| 289 | output_libname, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 290 | output_dir=None, |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 291 | libraries=None, |
| 292 | library_dirs=None, |
Greg Ward | 386b844 | 2000-02-09 02:18:39 +0000 | [diff] [blame] | 293 | debug=0, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 294 | extra_preargs=None, |
| 295 | extra_postargs=None): |
| 296 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 297 | # XXX should we sanity check the library name? (eg. no |
| 298 | # slashes) |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 299 | self.link_shared_object (objects, |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 300 | self.shared_library_name(output_libname), |
| 301 | output_dir=output_dir, |
| 302 | libraries=libraries, |
| 303 | library_dirs=library_dirs, |
| 304 | debug=debug, |
| 305 | extra_preargs=extra_preargs, |
| 306 | extra_postargs=extra_postargs) |
| 307 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 308 | |
| 309 | def link_shared_object (self, |
| 310 | objects, |
| 311 | output_filename, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 312 | output_dir=None, |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 313 | libraries=None, |
| 314 | library_dirs=None, |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 315 | debug=0, |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 316 | extra_preargs=None, |
| 317 | extra_postargs=None): |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 318 | """Link a bunch of stuff together to create a shared object |
| 319 | file. Much like 'link_shared_lib()', except the output |
| 320 | filename is explicitly supplied as 'output_filename'.""" |
| 321 | if libraries is None: |
| 322 | libraries = [] |
| 323 | if library_dirs is None: |
| 324 | library_dirs = [] |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 325 | |
Greg Ward | c74138d | 1999-10-03 20:47:52 +0000 | [diff] [blame] | 326 | lib_opts = gen_lib_options (self, |
| 327 | self.library_dirs + library_dirs, |
| 328 | self.libraries + libraries) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 329 | |
Greg Ward | 386b844 | 2000-02-09 02:18:39 +0000 | [diff] [blame] | 330 | if debug: |
Greg Ward | 4ba9b2e | 2000-02-10 02:15:52 +0000 | [diff] [blame] | 331 | ldflags = self.ldflags_shared_debug |
| 332 | basename, ext = os.path.splitext (output_filename) |
| 333 | #XXX not sure this belongs here |
| 334 | # extensions in debug_mode are named 'module_d.pyd' |
| 335 | output_filename = basename + '_d' + ext |
| 336 | else: |
| 337 | ldflags = self.ldflags_shared |
| 338 | |
| 339 | ld_args = ldflags + lib_opts + \ |
| 340 | objects + ['/OUT:' + output_filename] |
| 341 | |
Greg Ward | df178f9 | 1999-09-29 12:29:10 +0000 | [diff] [blame] | 342 | if extra_preargs: |
| 343 | ld_args[:0] = extra_preargs |
| 344 | if extra_postargs: |
| 345 | ld_args.extend (extra_postargs) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 346 | |
| 347 | self.spawn ( [ self.link ] + ld_args ) |
| 348 | |
| 349 | |
| 350 | # -- Filename mangling methods ------------------------------------- |
| 351 | |
| 352 | def _change_extensions( self, filenames, newExtension ): |
| 353 | object_filenames = [] |
| 354 | |
| 355 | for srcFile in filenames: |
| 356 | base,ext = os.path.splitext( srcFile ) |
| 357 | # XXX should we strip off any existing path? |
| 358 | object_filenames.append( base + newExtension ) |
| 359 | |
| 360 | return object_filenames |
| 361 | |
| 362 | def object_filenames (self, source_filenames): |
| 363 | """Return the list of object filenames corresponding to each |
| 364 | specified source filename.""" |
| 365 | return self._change_extensions( source_filenames, self._obj_ext ) |
| 366 | |
| 367 | def shared_object_filename (self, source_filename): |
| 368 | """Return the shared object filename corresponding to a |
| 369 | specified source filename.""" |
| 370 | return self._change_extensions( source_filenames, self._shared_lib_ext ) |
| 371 | |
| 372 | def library_filename (self, libname): |
| 373 | """Return the static library filename corresponding to the |
| 374 | specified library name.""" |
Greg Ward | c8a95c8 | 2000-01-17 18:00:04 +0000 | [diff] [blame] | 375 | return "%s%s" %( libname, self._static_lib_ext ) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 376 | |
| 377 | def shared_library_filename (self, libname): |
| 378 | """Return the shared library filename corresponding to the |
| 379 | specified library name.""" |
Greg Ward | c8a95c8 | 2000-01-17 18:00:04 +0000 | [diff] [blame] | 380 | return "%s%s" %( libname, self._shared_lib_ext ) |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 381 | |
Greg Ward | c74138d | 1999-10-03 20:47:52 +0000 | [diff] [blame] | 382 | |
| 383 | def library_dir_option (self, dir): |
| 384 | return "/LIBPATH:" + dir |
| 385 | |
| 386 | def library_option (self, lib): |
| 387 | return self.library_filename (lib) |
| 388 | |
| 389 | |
| 390 | def find_library_file (self, dirs, lib): |
| 391 | |
| 392 | for dir in dirs: |
| 393 | libfile = os.path.join (dir, self.library_filename (lib)) |
| 394 | if os.path.exists (libfile): |
| 395 | return libfile |
| 396 | |
| 397 | else: |
| 398 | # Oops, didn't find it in *any* of 'dirs' |
| 399 | return None |
| 400 | |
| 401 | # find_library_file () |
| 402 | |
Greg Ward | dbd1276 | 1999-08-29 18:15:07 +0000 | [diff] [blame] | 403 | # class MSVCCompiler |