Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 1 | """distutils.unixccompiler |
| 2 | |
| 3 | Contains the UnixCCompiler class, a subclass of CCompiler that handles |
| 4 | the "typical" Unix-style command-line C compiler: |
| 5 | * macros defined with -Dname[=value] |
| 6 | * macros undefined with -Uname |
| 7 | * include search directories specified with -Idir |
| 8 | * libraries specified with -lllib |
| 9 | * library search directories specified with -Ldir |
| 10 | * compile handled by 'cc' (or similar) executable with -c option: |
| 11 | compiles .c to .o |
| 12 | * link static library handled by 'ar' command (possibly with 'ranlib') |
| 13 | * link shared library handled by 'cc -shared' |
| 14 | """ |
| 15 | |
| 16 | # created 1999/07/05, Greg Ward |
| 17 | |
| 18 | __rcsid__ = "$Id$" |
| 19 | |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 20 | import string, re, os |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 21 | from types import * |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 22 | from copy import copy |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 23 | from sysconfig import \ |
| 24 | CC, CCSHARED, CFLAGS, OPT, LDSHARED, LDFLAGS, RANLIB, AR, SO |
Greg Ward | c294113 | 1999-09-08 02:32:19 +0000 | [diff] [blame] | 25 | from ccompiler import CCompiler, gen_preprocess_options, gen_lib_options |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 26 | from util import move_file, newer_pairwise, newer_group |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 27 | |
| 28 | # XXX Things not currently handled: |
| 29 | # * optimization/debug/warning flags; we just use whatever's in Python's |
| 30 | # Makefile and live with it. Is this adequate? If not, we might |
| 31 | # have to have a bunch of subclasses GNUCCompiler, SGICCompiler, |
| 32 | # SunCCompiler, and I suspect down that road lies madness. |
| 33 | # * even if we don't know a warning flag from an optimization flag, |
| 34 | # we need some way for outsiders to feed preprocessor/compiler/linker |
| 35 | # flags in to us -- eg. a sysadmin might want to mandate certain flags |
| 36 | # via a site config file, or a user might want to set something for |
| 37 | # compiling this module distribution only via the setup.py command |
| 38 | # line, whatever. As long as these options come from something on the |
| 39 | # current system, they can be as system-dependent as they like, and we |
| 40 | # should just happily stuff them into the preprocessor/compiler/linker |
| 41 | # options and carry on. |
| 42 | |
| 43 | |
| 44 | class UnixCCompiler (CCompiler): |
| 45 | |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 46 | # XXX perhaps there should really be *three* kinds of include |
| 47 | # directories: those built in to the preprocessor, those from Python's |
| 48 | # Makefiles, and those supplied to {add,set}_include_dirs(). Currently |
| 49 | # we make no distinction between the latter two at this point; it's all |
| 50 | # up to the client class to select the include directories to use above |
| 51 | # and beyond the compiler's defaults. That is, both the Python include |
| 52 | # directories and any module- or package-specific include directories |
| 53 | # are specified via {add,set}_include_dirs(), and there's no way to |
| 54 | # distinguish them. This might be a bug. |
Greg Ward | 65f4a3b | 1999-08-29 18:23:32 +0000 | [diff] [blame] | 55 | |
Greg Ward | 0e3530b | 1999-09-29 12:22:50 +0000 | [diff] [blame] | 56 | compiler_type = 'unix' |
| 57 | |
Greg Ward | 65f4a3b | 1999-08-29 18:23:32 +0000 | [diff] [blame] | 58 | _obj_ext = '.o' |
| 59 | _exe_ext = '' |
| 60 | _shared_lib_ext = SO |
| 61 | _static_lib_ext = '.a' |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 62 | |
| 63 | # Command to create a static library: seems to be pretty consistent |
| 64 | # across the major Unices. Might have to move down into the |
| 65 | # constructor if we need platform-specific guesswork. |
| 66 | archiver = "ar" |
| 67 | archiver_options = "-cr" |
| 68 | |
| 69 | |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 70 | def __init__ (self, |
| 71 | verbose=0, |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 72 | dry_run=0, |
| 73 | force=0): |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 74 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 75 | CCompiler.__init__ (self, verbose, dry_run, force) |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 76 | |
| 77 | self.preprocess_options = None |
| 78 | self.compile_options = None |
| 79 | |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 80 | # Munge CC and OPT together in case there are flags stuck in CC. |
| 81 | # Note that using these variables from sysconfig immediately makes |
| 82 | # this module specific to building Python extensions and |
| 83 | # inappropriate as a general-purpose C compiler front-end. So sue |
| 84 | # me. Note also that we use OPT rather than CFLAGS, because CFLAGS |
| 85 | # is the flags used to compile Python itself -- not only are there |
| 86 | # -I options in there, they are the *wrong* -I options. We'll |
| 87 | # leave selection of include directories up to the class using |
| 88 | # UnixCCompiler! |
| 89 | |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 90 | (self.cc, self.ccflags) = \ |
| 91 | _split_command (CC + ' ' + OPT) |
| 92 | self.ccflags_shared = string.split (CCSHARED) |
| 93 | |
| 94 | (self.ld_shared, self.ldflags_shared) = \ |
| 95 | _split_command (LDSHARED) |
| 96 | |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 97 | self.ld_exec = self.cc |
| 98 | |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 99 | |
| 100 | def compile (self, |
| 101 | sources, |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 102 | output_dir=None, |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 103 | macros=None, |
Greg Ward | 04d7832 | 1999-12-12 16:57:47 +0000 | [diff] [blame] | 104 | include_dirs=None, |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 105 | debug=0, |
Greg Ward | 0e3530b | 1999-09-29 12:22:50 +0000 | [diff] [blame] | 106 | extra_preargs=None, |
| 107 | extra_postargs=None): |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 108 | |
Greg Ward | 10ca82b | 2000-02-10 02:51:32 +0000 | [diff] [blame] | 109 | if type (output_dir) not in (StringType, NoneType): |
| 110 | raise TypeError, "'output_dir' must be a string or None" |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 111 | if output_dir is None: |
| 112 | output_dir = self.output_dir |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 113 | if macros is None: |
| 114 | macros = [] |
Greg Ward | 04d7832 | 1999-12-12 16:57:47 +0000 | [diff] [blame] | 115 | if include_dirs is None: |
| 116 | include_dirs = [] |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 117 | |
| 118 | if type (macros) is not ListType: |
| 119 | raise TypeError, \ |
| 120 | "'macros' (if supplied) must be a list of tuples" |
Greg Ward | 04d7832 | 1999-12-12 16:57:47 +0000 | [diff] [blame] | 121 | if type (include_dirs) not in (ListType, TupleType): |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 122 | raise TypeError, \ |
Greg Ward | 04d7832 | 1999-12-12 16:57:47 +0000 | [diff] [blame] | 123 | "'include_dirs' (if supplied) must be a list of strings" |
| 124 | include_dirs = list (include_dirs) |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 125 | |
Greg Ward | c294113 | 1999-09-08 02:32:19 +0000 | [diff] [blame] | 126 | pp_opts = gen_preprocess_options (self.macros + macros, |
Greg Ward | 04d7832 | 1999-12-12 16:57:47 +0000 | [diff] [blame] | 127 | self.include_dirs + include_dirs) |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 128 | |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 129 | # So we can mangle 'sources' without hurting the caller's data |
| 130 | orig_sources = sources |
| 131 | sources = copy (sources) |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 132 | |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 133 | # Get the list of expected output (object) files and drop files we |
| 134 | # don't have to recompile. (Simplistic check -- we just compare the |
| 135 | # source and object file, no deep dependency checking involving |
| 136 | # header files. Hmmm.) |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 137 | objects = self.object_filenames (sources, output_dir=output_dir) |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 138 | if not self.force: |
| 139 | skipped = newer_pairwise (sources, objects) |
| 140 | for skipped_pair in skipped: |
| 141 | self.announce ("skipping %s (%s up-to-date)" % skipped_pair) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 142 | |
Greg Ward | ef6f515 | 2000-02-03 23:07:19 +0000 | [diff] [blame] | 143 | # Build list of (source,object) tuples for convenience |
| 144 | srcobj = [] |
| 145 | for i in range (len (sources)): |
| 146 | srcobj.append ((sources[i], objects[i])) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 147 | |
Greg Ward | ef6f515 | 2000-02-03 23:07:19 +0000 | [diff] [blame] | 148 | # Compile all source files that weren't eliminated by |
| 149 | # 'newer_pairwise()'. |
| 150 | # XXX use of ccflags_shared means we're blithely assuming |
| 151 | # that we're compiling for inclusion in a shared object! |
| 152 | # (will have to fix this when I add the ability to build a |
| 153 | # new Python) |
| 154 | cc_args = ['-c'] + pp_opts + self.ccflags + self.ccflags_shared |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 155 | if debug: |
| 156 | cc_args[:0] = ['-g'] |
Greg Ward | ef6f515 | 2000-02-03 23:07:19 +0000 | [diff] [blame] | 157 | if extra_preargs: |
| 158 | cc_args[:0] = extra_preargs |
| 159 | if extra_postargs is None: |
| 160 | extra_postargs = [] |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 161 | |
Greg Ward | ef6f515 | 2000-02-03 23:07:19 +0000 | [diff] [blame] | 162 | for (source,object) in srcobj: |
| 163 | self.spawn ([self.cc] + cc_args + |
| 164 | [source, '-o', object] + |
| 165 | extra_postargs) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 166 | |
| 167 | # Have to re-fetch list of object filenames, because we want to |
| 168 | # return *all* of them, including those that weren't recompiled on |
| 169 | # this call! |
| 170 | return self.object_filenames (orig_sources, output_dir) |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 171 | |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 172 | |
| 173 | def _fix_link_args (self, output_dir, libraries, library_dirs): |
| 174 | """Fixes up the arguments supplied to the 'link_*' methods: |
| 175 | if output_dir is None, use self.output_dir; ensure that |
| 176 | libraries and library_dirs are both lists (could be None or |
| 177 | tuples on input -- both are converted to lists). Return |
| 178 | a tuple of the three input arguments.""" |
| 179 | |
| 180 | if output_dir is None: |
| 181 | output_dir = self.output_dir |
| 182 | if libraries is None: |
| 183 | libraries = [] |
| 184 | if library_dirs is None: |
| 185 | library_dirs = [] |
| 186 | |
| 187 | if type (libraries) not in (ListType, TupleType): |
| 188 | raise TypeError, \ |
| 189 | "'libraries' (if supplied) must be a list of strings" |
| 190 | if type (library_dirs) not in (ListType, TupleType): |
| 191 | raise TypeError, \ |
| 192 | "'library_dirs' (if supplied) must be a list of strings" |
| 193 | libraries = list (libraries) |
| 194 | library_dirs = list (library_dirs) |
| 195 | |
| 196 | return (output_dir, libraries, library_dirs) |
| 197 | |
| 198 | |
| 199 | def link_static_lib (self, |
| 200 | objects, |
| 201 | output_libname, |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 202 | output_dir=None, |
| 203 | debug=0): |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 204 | |
| 205 | if type (objects) not in (ListType, TupleType): |
| 206 | raise TypeError, \ |
| 207 | "'objects' must be a list or tuple of strings" |
| 208 | objects = list (objects) |
| 209 | |
Greg Ward | 10ca82b | 2000-02-10 02:51:32 +0000 | [diff] [blame] | 210 | if type (output_dir) not in (StringType, NoneType): |
| 211 | raise TypeError, "'output_dir' must be a string or None" |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 212 | if output_dir is None: |
| 213 | output_dir = self.output_dir |
| 214 | |
| 215 | output_filename = self.library_filename (output_libname) |
| 216 | if output_dir is not None: |
| 217 | output_filename = os.path.join (output_dir, output_filename) |
| 218 | |
| 219 | # Check timestamps: if any of the object files are newer than |
| 220 | # the library file, *or* if "force" is true, then we'll |
| 221 | # recreate the library. |
| 222 | if not self.force: |
| 223 | if self.dry_run: |
| 224 | newer = newer_group (objects, output_filename, missing='newer') |
| 225 | else: |
| 226 | newer = newer_group (objects, output_filename) |
| 227 | |
| 228 | if self.force or newer: |
| 229 | self.spawn ([self.archiver, |
| 230 | self.archiver_options, |
| 231 | output_filename] + |
| 232 | objects) |
| 233 | else: |
| 234 | self.announce ("skipping %s (up-to-date)" % output_filename) |
| 235 | |
| 236 | # link_static_lib () |
| 237 | |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 238 | |
| 239 | def link_shared_lib (self, |
| 240 | objects, |
| 241 | output_libname, |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 242 | output_dir=None, |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 243 | libraries=None, |
Greg Ward | 65f4a3b | 1999-08-29 18:23:32 +0000 | [diff] [blame] | 244 | library_dirs=None, |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 245 | debug=0, |
Greg Ward | 0e3530b | 1999-09-29 12:22:50 +0000 | [diff] [blame] | 246 | extra_preargs=None, |
| 247 | extra_postargs=None): |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 248 | # XXX should we sanity check the library name? (eg. no |
| 249 | # slashes) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 250 | self.link_shared_object ( |
| 251 | objects, |
| 252 | "lib%s%s" % (output_libname, self._shared_lib_ext), |
| 253 | output_dir, |
| 254 | libraries, |
| 255 | library_dirs, |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 256 | debug, |
Greg Ward | 0e3530b | 1999-09-29 12:22:50 +0000 | [diff] [blame] | 257 | extra_preargs, |
| 258 | extra_postargs) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 259 | |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 260 | |
| 261 | def link_shared_object (self, |
| 262 | objects, |
| 263 | output_filename, |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 264 | output_dir=None, |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 265 | libraries=None, |
Greg Ward | 65f4a3b | 1999-08-29 18:23:32 +0000 | [diff] [blame] | 266 | library_dirs=None, |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 267 | debug=0, |
Greg Ward | 0e3530b | 1999-09-29 12:22:50 +0000 | [diff] [blame] | 268 | extra_preargs=None, |
| 269 | extra_postargs=None): |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 270 | |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 271 | (output_dir, libraries, library_dirs) = \ |
| 272 | self._fix_link_args (output_dir, libraries, library_dirs) |
Greg Ward | 04d7832 | 1999-12-12 16:57:47 +0000 | [diff] [blame] | 273 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 274 | lib_opts = gen_lib_options (self, |
| 275 | self.library_dirs + library_dirs, |
| 276 | self.libraries + libraries) |
Greg Ward | 10ca82b | 2000-02-10 02:51:32 +0000 | [diff] [blame] | 277 | if type (output_dir) not in (StringType, NoneType): |
| 278 | raise TypeError, "'output_dir' must be a string or None" |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 279 | if output_dir is not None: |
| 280 | output_filename = os.path.join (output_dir, output_filename) |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 281 | |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 282 | # If any of the input object files are newer than the output shared |
| 283 | # object, relink. Again, this is a simplistic dependency check: |
| 284 | # doesn't look at any of the libraries we might be linking with. |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 285 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 286 | if not self.force: |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 287 | if self.dry_run: |
| 288 | newer = newer_group (objects, output_filename, missing='newer') |
| 289 | else: |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 290 | newer = newer_group (objects, output_filename) |
Greg Ward | b116e45 | 1999-09-21 18:36:15 +0000 | [diff] [blame] | 291 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 292 | if self.force or newer: |
| 293 | ld_args = self.ldflags_shared + objects + \ |
| 294 | lib_opts + ['-o', output_filename] |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 295 | if debug: |
| 296 | ld_args[:0] = ['-g'] |
Greg Ward | 0e3530b | 1999-09-29 12:22:50 +0000 | [diff] [blame] | 297 | if extra_preargs: |
| 298 | ld_args[:0] = extra_preargs |
| 299 | if extra_postargs: |
| 300 | ld_args.extend (extra_postargs) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 301 | self.spawn ([self.ld_shared] + ld_args) |
| 302 | else: |
| 303 | self.announce ("skipping %s (up-to-date)" % output_filename) |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 304 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 305 | # link_shared_object () |
| 306 | |
| 307 | |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 308 | def link_executable (self, |
| 309 | objects, |
| 310 | output_progname, |
| 311 | output_dir=None, |
| 312 | libraries=None, |
| 313 | library_dirs=None, |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 314 | debug=0, |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 315 | extra_preargs=None, |
| 316 | extra_postargs=None): |
| 317 | |
| 318 | (output_dir, libraries, library_dirs) = \ |
| 319 | self._fix_link_args (output_dir, libraries, library_dirs) |
| 320 | |
| 321 | lib_opts = gen_lib_options (self, |
| 322 | self.library_dirs + library_dirs, |
| 323 | self.libraries + libraries) |
| 324 | output_filename = output_progname # Unix-ism! |
| 325 | if output_dir is not None: |
| 326 | output_filename = os.path.join (output_dir, output_filename) |
| 327 | |
| 328 | # Same ol' simplistic-but-still-useful dependency check. |
| 329 | if not self.force: |
| 330 | if self.dry_run: |
| 331 | newer = newer_group (objects, output_filename, missing='newer') |
| 332 | else: |
| 333 | newer = newer_group (objects, output_filename) |
| 334 | |
| 335 | if self.force or newer: |
| 336 | ld_args = objects + lib_opts + ['-o', output_filename] |
Greg Ward | ba233fb | 2000-02-09 02:17:00 +0000 | [diff] [blame] | 337 | if debug: |
| 338 | ld_args[:0] = ['-g'] |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 339 | if extra_preargs: |
| 340 | ld_args[:0] = extra_preargs |
| 341 | if extra_postargs: |
| 342 | ld_args.extend (extra_postargs) |
| 343 | self.spawn ([self.ld_exec] + ld_args) |
| 344 | else: |
| 345 | self.announce ("skipping %s (up-to-date)" % output_filename) |
| 346 | |
| 347 | # link_executable () |
| 348 | |
| 349 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 350 | # -- Filename-mangling (etc.) methods ------------------------------ |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 351 | |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 352 | def object_filenames (self, source_filenames, |
| 353 | keep_dir=0, output_dir=None): |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 354 | outnames = [] |
| 355 | for inname in source_filenames: |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 356 | outname = re.sub (r'\.(c|C|cc|cxx|cpp)$', self._obj_ext, inname) |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 357 | if not keep_dir: |
| 358 | outname = os.path.basename (outname) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 359 | if output_dir is not None: |
| 360 | outname = os.path.join (output_dir, outname) |
| 361 | outnames.append (outname) |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 362 | return outnames |
| 363 | |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 364 | def shared_object_filename (self, source_filename, |
| 365 | keep_dir=0, output_dir=None): |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 366 | outname = re.sub (r'\.(c|C|cc|cxx|cpp)$', self._shared_lib_ext) |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 367 | if not keep_dir: |
| 368 | outname = os.path.basename (outname) |
Greg Ward | 8037cb1 | 1999-09-13 03:12:53 +0000 | [diff] [blame] | 369 | if output_dir is not None: |
| 370 | outname = os.path.join (output_dir, outname) |
| 371 | return outname |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 372 | |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 373 | |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 374 | def library_filename (self, libname): |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 375 | (dirname, basename) = os.path.split (libname) |
| 376 | return os.path.join (dirname, |
| 377 | "lib%s%s" % (basename, self._static_lib_ext)) |
Greg Ward | 5e71744 | 1999-08-14 23:53:53 +0000 | [diff] [blame] | 378 | |
| 379 | def shared_library_filename (self, libname): |
Greg Ward | c9f3187 | 2000-01-09 22:47:53 +0000 | [diff] [blame] | 380 | (dirname, basename) = os.path.split (libname) |
| 381 | return os.path.join (dirname, |
| 382 | "lib%s%s" % (basename, self._shared_lib_ext)) |
Greg Ward | 4fecfce | 1999-10-03 20:45:33 +0000 | [diff] [blame] | 383 | |
| 384 | |
| 385 | def library_dir_option (self, dir): |
| 386 | return "-L" + dir |
| 387 | |
| 388 | def library_option (self, lib): |
| 389 | return "-l" + lib |
| 390 | |
| 391 | |
| 392 | def find_library_file (self, dirs, lib): |
| 393 | |
| 394 | for dir in dirs: |
| 395 | shared = os.path.join (dir, self.shared_library_filename (lib)) |
| 396 | static = os.path.join (dir, self.library_filename (lib)) |
| 397 | |
| 398 | # We're second-guessing the linker here, with not much hard |
| 399 | # data to go on: GCC seems to prefer the shared library, so I'm |
| 400 | # assuming that *all* Unix C compilers do. And of course I'm |
| 401 | # ignoring even GCC's "-static" option. So sue me. |
| 402 | if os.path.exists (shared): |
| 403 | return shared |
| 404 | elif os.path.exists (static): |
| 405 | return static |
| 406 | |
| 407 | else: |
| 408 | # Oops, didn't find it in *any* of 'dirs' |
| 409 | return None |
| 410 | |
| 411 | # find_library_file () |
Greg Ward | 65f4a3b | 1999-08-29 18:23:32 +0000 | [diff] [blame] | 412 | |
Greg Ward | 170bdc0 | 1999-07-10 02:04:22 +0000 | [diff] [blame] | 413 | # class UnixCCompiler |
| 414 | |
| 415 | |
| 416 | def _split_command (cmd): |
| 417 | """Split a command string up into the progam to run (a string) and |
| 418 | the list of arguments; return them as (cmd, arglist).""" |
| 419 | args = string.split (cmd) |
| 420 | return (args[0], args[1:]) |