| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | """distutils.command.build_ext
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Implements the Distutils 'build_ext' command, for building extension | 
					
						
							| 
									
										
										
										
											2000-06-28 14:48:01 +00:00
										 |  |  | modules (currently limited to C extensions, should accommodate C++ | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | extensions ASAP)."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-19 13:12:28 +00:00
										 |  |  | # This module should be kept compatible with Python 1.5.2. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-02 01:49:45 +00:00
										 |  |  | __revision__ = "$Id$" | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import sys, os, string, re | 
					
						
							|  |  |  | from types import * | 
					
						
							|  |  |  | from distutils.core import Command | 
					
						
							|  |  |  | from distutils.errors import * | 
					
						
							| 
									
										
										
										
											2002-11-14 01:44:35 +00:00
										 |  |  | from distutils.sysconfig import customize_compiler, get_python_version | 
					
						
							| 
									
										
										
										
											2000-05-26 01:31:53 +00:00
										 |  |  | from distutils.dep_util import newer_group | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  | from distutils.extension import Extension | 
					
						
							| 
									
										
										
										
											2002-06-04 20:14:43 +00:00
										 |  |  | from distutils import log | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-21 18:27:12 +00:00
										 |  |  | # An extension name is just a dot-separated list of Python NAMEs (ie. | 
					
						
							|  |  |  | # the same as a fully-qualified module name). | 
					
						
							|  |  |  | extension_name_re = re.compile \ | 
					
						
							|  |  |  |     (r'^[a-zA-Z_][a-zA-Z_0-9]*(\.[a-zA-Z_][a-zA-Z_0-9]*)*$') | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-24 01:23:37 +00:00
										 |  |  | def show_compilers (): | 
					
						
							|  |  |  |     from distutils.ccompiler import show_compilers | 
					
						
							|  |  |  |     show_compilers() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-18 00:13:53 +00:00
										 |  |  | class build_ext (Command): | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-30 18:34:15 +00:00
										 |  |  |     description = "build C/C++ extensions (compile/link to build directory)" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |     # XXX thoughts on how to deal with complex command-line options like | 
					
						
							|  |  |  |     # these, i.e. how to make it so fancy_getopt can suck them off the | 
					
						
							|  |  |  |     # command line and make it look like setup.py defined the appropriate | 
					
						
							|  |  |  |     # lists of tuples of what-have-you. | 
					
						
							|  |  |  |     #   - each command needs a callback to process its command-line options | 
					
						
							|  |  |  |     #   - Command.__init__() needs access to its share of the whole | 
					
						
							|  |  |  |     #     command line (must ultimately come from | 
					
						
							|  |  |  |     #     Distribution.parse_command_line()) | 
					
						
							|  |  |  |     #   - it then calls the current command class' option-parsing | 
					
						
							|  |  |  |     #     callback to deal with weird options like -D, which have to | 
					
						
							|  |  |  |     #     parse the option text and churn out some custom data | 
					
						
							|  |  |  |     #     structure | 
					
						
							|  |  |  |     #   - that data structure (in this case, a list of 2-tuples) | 
					
						
							|  |  |  |     #     will then be present in the command object by the time | 
					
						
							| 
									
										
										
										
											2000-02-18 00:35:22 +00:00
										 |  |  |     #     we get to finalize_options() (i.e. the constructor | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |     #     takes care of both command-line and client options | 
					
						
							| 
									
										
										
										
											2000-02-18 00:35:22 +00:00
										 |  |  |     #     in between initialize_options() and finalize_options()) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-16 01:44:45 +00:00
										 |  |  |     sep_by = " (separated by '%s')" % os.pathsep | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |     user_options = [ | 
					
						
							| 
									
										
										
										
											2000-03-01 01:43:28 +00:00
										 |  |  |         ('build-lib=', 'b', | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |          "directory for compiled extension modules"), | 
					
						
							| 
									
										
										
										
											2000-03-01 01:43:28 +00:00
										 |  |  |         ('build-temp=', 't', | 
					
						
							|  |  |  |          "directory for temporary files (build by-products)"), | 
					
						
							|  |  |  |         ('inplace', 'i', | 
					
						
							| 
									
										
										
										
											2000-09-06 02:06:27 +00:00
										 |  |  |          "ignore build-lib and put compiled extensions into the source " + | 
					
						
							| 
									
										
										
										
											2000-03-01 01:43:28 +00:00
										 |  |  |          "directory alongside your pure Python modules"), | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |         ('include-dirs=', 'I', | 
					
						
							| 
									
										
										
										
											2000-09-16 01:44:45 +00:00
										 |  |  |          "list of directories to search for header files" + sep_by), | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |         ('define=', 'D', | 
					
						
							|  |  |  |          "C preprocessor macros to define"), | 
					
						
							|  |  |  |         ('undef=', 'U', | 
					
						
							|  |  |  |          "C preprocessor macros to undefine"), | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  |         ('libraries=', 'l', | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |          "external C libraries to link with"), | 
					
						
							|  |  |  |         ('library-dirs=', 'L', | 
					
						
							| 
									
										
										
										
											2000-09-16 01:44:45 +00:00
										 |  |  |          "directories to search for external C libraries" + sep_by), | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |         ('rpath=', 'R', | 
					
						
							|  |  |  |          "directories to search for shared C libraries at runtime"), | 
					
						
							|  |  |  |         ('link-objects=', 'O', | 
					
						
							|  |  |  |          "extra explicit link objects to include in the link"), | 
					
						
							|  |  |  |         ('debug', 'g', | 
					
						
							|  |  |  |          "compile/link with debugging information"), | 
					
						
							| 
									
										
										
										
											2000-04-10 00:19:42 +00:00
										 |  |  |         ('force', 'f', | 
					
						
							| 
									
										
										
										
											2000-05-12 00:34:12 +00:00
										 |  |  |          "forcibly build everything (ignore file timestamps)"), | 
					
						
							|  |  |  |         ('compiler=', 'c', | 
					
						
							|  |  |  |          "specify the compiler type"), | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |         ('swig-cpp', None, | 
					
						
							|  |  |  |          "make SWIG create C++ files (default is C)"), | 
					
						
							| 
									
										
										
										
											2000-02-18 00:25:39 +00:00
										 |  |  |         ] | 
					
						
							| 
									
										
										
										
											2000-06-24 01:23:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-25 01:41:15 +00:00
										 |  |  |     boolean_options = ['inplace', 'debug', 'force', 'swig-cpp'] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-07 03:00:06 +00:00
										 |  |  |     help_options = [ | 
					
						
							|  |  |  |         ('help-compiler', None, | 
					
						
							| 
									
										
										
										
											2000-06-24 01:23:37 +00:00
										 |  |  |          "list available compilers", show_compilers), | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  |         ] | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-18 00:35:22 +00:00
										 |  |  |     def initialize_options (self): | 
					
						
							| 
									
										
										
										
											1999-09-08 02:42:30 +00:00
										 |  |  |         self.extensions = None | 
					
						
							| 
									
										
										
										
											2000-03-01 01:43:28 +00:00
										 |  |  |         self.build_lib = None | 
					
						
							|  |  |  |         self.build_temp = None | 
					
						
							|  |  |  |         self.inplace = 0 | 
					
						
							| 
									
										
										
										
											1999-09-13 13:55:34 +00:00
										 |  |  |         self.package = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         self.include_dirs = None | 
					
						
							|  |  |  |         self.define = None | 
					
						
							|  |  |  |         self.undef = None | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  |         self.libraries = None | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         self.library_dirs = None | 
					
						
							|  |  |  |         self.rpath = None | 
					
						
							|  |  |  |         self.link_objects = None | 
					
						
							| 
									
										
										
										
											2000-02-09 02:20:14 +00:00
										 |  |  |         self.debug = None | 
					
						
							| 
									
										
										
										
											2000-04-10 00:19:42 +00:00
										 |  |  |         self.force = None | 
					
						
							| 
									
										
										
										
											2000-05-12 00:34:12 +00:00
										 |  |  |         self.compiler = None | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |         self.swig_cpp = None | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-29 12:49:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-18 00:35:22 +00:00
										 |  |  |     def finalize_options (self): | 
					
						
							| 
									
										
										
										
											2000-01-30 18:34:15 +00:00
										 |  |  |         from distutils import sysconfig | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         self.set_undefined_options('build', | 
					
						
							|  |  |  |                                    ('build_lib', 'build_lib'), | 
					
						
							|  |  |  |                                    ('build_temp', 'build_temp'), | 
					
						
							|  |  |  |                                    ('compiler', 'compiler'), | 
					
						
							|  |  |  |                                    ('debug', 'debug'), | 
					
						
							|  |  |  |                                    ('force', 'force')) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-13 13:55:34 +00:00
										 |  |  |         if self.package is None: | 
					
						
							| 
									
										
										
										
											1999-09-21 18:27:12 +00:00
										 |  |  |             self.package = self.distribution.ext_package | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.extensions = self.distribution.ext_modules | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-13 13:55:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-07-26 13:41:06 +00:00
										 |  |  |         # Make sure Python's include directories (for Python.h, pyconfig.h, | 
					
						
							| 
									
										
										
										
											2000-04-14 00:50:49 +00:00
										 |  |  |         # etc.) are in the include search path. | 
					
						
							|  |  |  |         py_include = sysconfig.get_python_inc() | 
					
						
							|  |  |  |         plat_py_include = sysconfig.get_python_inc(plat_specific=1) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.include_dirs is None: | 
					
						
							| 
									
										
										
										
											1999-09-21 18:27:12 +00:00
										 |  |  |             self.include_dirs = self.distribution.include_dirs or [] | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         if type(self.include_dirs) is StringType: | 
					
						
							|  |  |  |             self.include_dirs = string.split(self.include_dirs, os.pathsep) | 
					
						
							| 
									
										
										
										
											1999-09-29 12:49:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-29 04:13:49 +00:00
										 |  |  |         # Put the Python "system" include dir at the end, so that | 
					
						
							|  |  |  |         # any local include dirs take precedence. | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         self.include_dirs.append(py_include) | 
					
						
							| 
									
										
										
										
											2000-04-14 00:50:49 +00:00
										 |  |  |         if plat_py_include != py_include: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.include_dirs.append(plat_py_include) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         if type(self.libraries) is StringType: | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  |             self.libraries = [self.libraries] | 
					
						
							| 
									
										
										
										
											2000-03-18 15:21:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |         # Life is easier if we're not forever checking for None, so | 
					
						
							|  |  |  |         # simplify these options to empty lists if unset | 
					
						
							|  |  |  |         if self.libraries is None: | 
					
						
							|  |  |  |             self.libraries = [] | 
					
						
							|  |  |  |         if self.library_dirs is None: | 
					
						
							|  |  |  |             self.library_dirs = [] | 
					
						
							| 
									
										
										
										
											2001-01-09 03:15:47 +00:00
										 |  |  |         elif type(self.library_dirs) is StringType: | 
					
						
							|  |  |  |             self.library_dirs = string.split(self.library_dirs, os.pathsep) | 
					
						
							| 
									
										
										
										
											2001-02-17 04:48:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |         if self.rpath is None: | 
					
						
							|  |  |  |             self.rpath = [] | 
					
						
							| 
									
										
										
										
											2001-02-17 04:48:41 +00:00
										 |  |  |         elif type(self.rpath) is StringType: | 
					
						
							|  |  |  |             self.rpath = string.split(self.rpath, os.pathsep) | 
					
						
							| 
									
										
										
										
											2000-02-05 02:23:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |         # for extensions under windows use different directories | 
					
						
							|  |  |  |         # for Release and Debug builds. | 
					
						
							|  |  |  |         # also Python's library directory must be appended to library_dirs | 
					
						
							|  |  |  |         if os.name == 'nt': | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.library_dirs.append(os.path.join(sys.exec_prefix, 'libs')) | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |             if self.debug: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |                 self.build_temp = os.path.join(self.build_temp, "Debug") | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |                 self.build_temp = os.path.join(self.build_temp, "Release") | 
					
						
							| 
									
										
										
										
											2001-01-24 15:43:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-04-25 17:26:37 +00:00
										 |  |  |             # Append the source distribution include and library directories, | 
					
						
							|  |  |  |             # this allows distutils on windows to work in the source tree | 
					
						
							|  |  |  |             self.include_dirs.append(os.path.join(sys.exec_prefix, 'PC')) | 
					
						
							|  |  |  |             self.library_dirs.append(os.path.join(sys.exec_prefix, 'PCBuild')) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-31 18:56:00 +00:00
										 |  |  |         # OS/2 (EMX) doesn't support Debug vs Release builds, but has the  | 
					
						
							|  |  |  |         # import libraries in its "Config" subdirectory | 
					
						
							|  |  |  |         if os.name == 'os2': | 
					
						
							|  |  |  |             self.library_dirs.append(os.path.join(sys.exec_prefix, 'Config')) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-11 06:22:31 +00:00
										 |  |  |         # for extensions under Cygwin and AtheOS Python's library directory must be | 
					
						
							| 
									
										
										
										
											2001-01-24 15:43:09 +00:00
										 |  |  |         # appended to library_dirs | 
					
						
							| 
									
										
										
										
											2002-06-11 06:22:31 +00:00
										 |  |  |         if sys.platform[:6] == 'cygwin' or sys.platform[:6] == 'atheos': | 
					
						
							| 
									
										
										
										
											2001-01-24 15:43:09 +00:00
										 |  |  |             if string.find(sys.executable, sys.exec_prefix) != -1: | 
					
						
							|  |  |  |                 # building third party extensions | 
					
						
							| 
									
										
										
										
											2002-04-25 17:26:37 +00:00
										 |  |  |                 self.library_dirs.append(os.path.join(sys.prefix, "lib", | 
					
						
							| 
									
										
										
										
											2002-11-14 01:44:35 +00:00
										 |  |  |                                                       "python" + get_python_version(), | 
					
						
							| 
									
										
										
										
											2002-04-25 17:26:37 +00:00
										 |  |  |                                                       "config")) | 
					
						
							| 
									
										
										
										
											2001-01-24 15:43:09 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 # building python standard extensions | 
					
						
							|  |  |  |                 self.library_dirs.append('.') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-17 20:15:41 +00:00
										 |  |  |         # The argument parsing will result in self.define being a string, but | 
					
						
							|  |  |  |         # it has to be a list of 2-tuples.  All the preprocessor symbols | 
					
						
							|  |  |  |         # specified by the 'define' option will be set to '1'.  Multiple | 
					
						
							|  |  |  |         # symbols can be separated with commas. | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-17 20:15:41 +00:00
										 |  |  |         if self.define: | 
					
						
							|  |  |  |             defines = string.split(self.define, ',') | 
					
						
							|  |  |  |             self.define = map(lambda symbol: (symbol, '1'), defines) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The option for macros to undefine is also a string from the | 
					
						
							|  |  |  |         # option parsing, but has to be a list.  Multiple symbols can also | 
					
						
							|  |  |  |         # be separated with commas here. | 
					
						
							|  |  |  |         if self.undef: | 
					
						
							|  |  |  |             self.undef = string.split(self.undef, ',') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-18 00:35:22 +00:00
										 |  |  |     # finalize_options () | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def run (self): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-30 18:34:15 +00:00
										 |  |  |         from distutils.ccompiler import new_compiler | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         # 'self.extensions', as supplied by setup.py, is a list of | 
					
						
							|  |  |  |         # Extension instances.  See the documentation for Extension (in | 
					
						
							| 
									
										
										
										
											2000-06-17 23:04:31 +00:00
										 |  |  |         # distutils.extension) for details. | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  |         # | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         # For backwards compatibility with Distutils 0.8.2 and earlier, we | 
					
						
							|  |  |  |         # also allow the 'extensions' list to be a list of tuples: | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         #    (ext_name, build_info) | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         # where build_info is a dictionary containing everything that | 
					
						
							|  |  |  |         # Extension instances do except the name, with a few things being | 
					
						
							|  |  |  |         # differently named.  We convert these 2-tuples to Extension | 
					
						
							|  |  |  |         # instances as needed. | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-08 02:42:30 +00:00
										 |  |  |         if not self.extensions: | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  |         # If we were asked to build any C/C++ libraries, make sure that the | 
					
						
							|  |  |  |         # directory where we put them is in the library search path for | 
					
						
							|  |  |  |         # linking extensions. | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  |         if self.distribution.has_c_libraries(): | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             build_clib = self.get_finalized_command('build_clib') | 
					
						
							|  |  |  |             self.libraries.extend(build_clib.get_library_names() or []) | 
					
						
							|  |  |  |             self.library_dirs.append(build_clib.build_clib) | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         # Setup the CCompiler object that we'll use to do all the | 
					
						
							|  |  |  |         # compiling and linking | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         self.compiler = new_compiler(compiler=self.compiler, | 
					
						
							|  |  |  |                                      verbose=self.verbose, | 
					
						
							|  |  |  |                                      dry_run=self.dry_run, | 
					
						
							|  |  |  |                                      force=self.force) | 
					
						
							| 
									
										
										
										
											2000-06-25 02:10:46 +00:00
										 |  |  |         customize_compiler(self.compiler) | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # And make sure that any compile/link-related options (which might | 
					
						
							|  |  |  |         # come from the command-line or from the setup script) are set in | 
					
						
							|  |  |  |         # that CCompiler object -- that way, they automatically apply to | 
					
						
							|  |  |  |         # all compiling and linking done here. | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.include_dirs is not None: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.compiler.set_include_dirs(self.include_dirs) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.define is not None: | 
					
						
							|  |  |  |             # 'define' option is a list of (name,value) tuples | 
					
						
							|  |  |  |             for (name,value) in self.define: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |                 self.compiler.define_macro(name, value) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.undef is not None: | 
					
						
							|  |  |  |             for macro in self.undef: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |                 self.compiler.undefine_macro(macro) | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  |         if self.libraries is not None: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.compiler.set_libraries(self.libraries) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.library_dirs is not None: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.compiler.set_library_dirs(self.library_dirs) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.rpath is not None: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.compiler.set_runtime_library_dirs(self.rpath) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  |         if self.link_objects is not None: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             self.compiler.set_link_objects(self.link_objects) | 
					
						
							| 
									
										
										
										
											2000-03-02 01:32:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-26 21:45:14 +00:00
										 |  |  |         # Now actually compile and link everything. | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         self.build_extensions() | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # run () | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def check_extensions_list (self, extensions): | 
					
						
							| 
									
										
										
										
											2000-02-05 02:23:16 +00:00
										 |  |  |         """Ensure that the list of extensions (presumably provided as a
 | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         command option 'extensions') is valid, i.e. it is a list of | 
					
						
							|  |  |  |         Extension objects.  We also support the old-style list of 2-tuples, | 
					
						
							|  |  |  |         where the tuples are (ext_name, build_info), which are converted to | 
					
						
							|  |  |  |         Extension instances here. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Raise DistutilsSetupError if the structure is invalid anywhere; | 
					
						
							|  |  |  |         just returns otherwise. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if type(extensions) is not ListType: | 
					
						
							| 
									
										
										
										
											2000-04-15 22:15:07 +00:00
										 |  |  |             raise DistutilsSetupError, \ | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |                   "'ext_modules' option must be a list of Extension instances" | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         for i in range(len(extensions)): | 
					
						
							|  |  |  |             ext = extensions[i] | 
					
						
							|  |  |  |             if isinstance(ext, Extension): | 
					
						
							|  |  |  |                 continue                # OK! (assume type-checking done | 
					
						
							|  |  |  |                                         # by Extension constructor) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             (ext_name, build_info) = ext | 
					
						
							| 
									
										
										
										
											2002-06-04 20:14:43 +00:00
										 |  |  |             log.warn(("old-style (ext_name, build_info) tuple found in " | 
					
						
							|  |  |  |                       "ext_modules for extension '%s'" | 
					
						
							|  |  |  |                       "-- please convert to Extension instance" % ext_name)) | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |             if type(ext) is not TupleType and len(ext) != 2: | 
					
						
							| 
									
										
										
										
											2000-04-15 22:15:07 +00:00
										 |  |  |                 raise DistutilsSetupError, \ | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |                       ("each element of 'ext_modules' option must be an " | 
					
						
							|  |  |  |                        "Extension instance or 2-tuple") | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |             if not (type(ext_name) is StringType and | 
					
						
							|  |  |  |                     extension_name_re.match(ext_name)): | 
					
						
							| 
									
										
										
										
											2000-04-15 22:15:07 +00:00
										 |  |  |                 raise DistutilsSetupError, \ | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |                       ("first element of each tuple in 'ext_modules' " | 
					
						
							|  |  |  |                        "must be the extension name (a string)") | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |             if type(build_info) is not DictionaryType: | 
					
						
							| 
									
										
										
										
											2000-04-15 22:15:07 +00:00
										 |  |  |                 raise DistutilsSetupError, \ | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |                       ("second element of each tuple in 'ext_modules' " | 
					
						
							|  |  |  |                        "must be a dictionary (build info)") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # OK, the (ext_name, build_info) dict is type-safe: convert it | 
					
						
							|  |  |  |             # to an Extension instance. | 
					
						
							|  |  |  |             ext = Extension(ext_name, build_info['sources']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Easy stuff: one-to-one mapping from dict elements to | 
					
						
							|  |  |  |             # instance attributes. | 
					
						
							|  |  |  |             for key in ('include_dirs', | 
					
						
							|  |  |  |                         'library_dirs', | 
					
						
							|  |  |  |                         'libraries', | 
					
						
							|  |  |  |                         'extra_objects', | 
					
						
							|  |  |  |                         'extra_compile_args', | 
					
						
							|  |  |  |                         'extra_link_args'): | 
					
						
							| 
									
										
										
										
											2000-08-26 02:21:55 +00:00
										 |  |  |                 val = build_info.get(key) | 
					
						
							|  |  |  |                 if val is not None: | 
					
						
							|  |  |  |                     setattr(ext, key, val) | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Medium-easy stuff: same syntax/semantics, different names. | 
					
						
							|  |  |  |             ext.runtime_library_dirs = build_info.get('rpath') | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  |             if build_info.has_key('def_file'): | 
					
						
							| 
									
										
										
										
											2002-06-04 20:14:43 +00:00
										 |  |  |                 log.warn("'def_file' element of build info dict " | 
					
						
							|  |  |  |                          "no longer supported") | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Non-trivial stuff: 'macros' split into 'define_macros' | 
					
						
							|  |  |  |             # and 'undef_macros'. | 
					
						
							|  |  |  |             macros = build_info.get('macros') | 
					
						
							|  |  |  |             if macros: | 
					
						
							|  |  |  |                 ext.define_macros = [] | 
					
						
							|  |  |  |                 ext.undef_macros = [] | 
					
						
							|  |  |  |                 for macro in macros: | 
					
						
							|  |  |  |                     if not (type(macro) is TupleType and | 
					
						
							| 
									
										
										
										
											2000-07-27 01:21:54 +00:00
										 |  |  |                             1 <= len(macro) <= 2): | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |                         raise DistutilsSetupError, \ | 
					
						
							|  |  |  |                               ("'macros' element of build info dict " | 
					
						
							|  |  |  |                                "must be 1- or 2-tuple") | 
					
						
							|  |  |  |                     if len(macro) == 1: | 
					
						
							|  |  |  |                         ext.undef_macros.append(macro[0]) | 
					
						
							|  |  |  |                     elif len(macro) == 2: | 
					
						
							|  |  |  |                         ext.define_macros.append(macro) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             extensions[i] = ext | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # for extensions | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # check_extensions_list () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-29 12:49:35 +00:00
										 |  |  |     def get_source_files (self): | 
					
						
							| 
									
										
										
										
											2000-07-14 13:35:07 +00:00
										 |  |  |         self.check_extensions_list(self.extensions) | 
					
						
							| 
									
										
										
										
											1999-09-29 12:49:35 +00:00
										 |  |  |         filenames = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Wouldn't it be neat if we knew the names of header files too... | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         for ext in self.extensions: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             filenames.extend(ext.sources) | 
					
						
							| 
									
										
										
										
											1999-09-29 12:49:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return filenames | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  |     def get_outputs (self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Sanity check the 'extensions' list -- can't assume this is being | 
					
						
							|  |  |  |         # done in the same run as a 'build_extensions()' call (in fact, we | 
					
						
							|  |  |  |         # can probably assume that it *isn't*!). | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         self.check_extensions_list(self.extensions) | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # And build the list of output (built) filenames.  Note that this | 
					
						
							|  |  |  |         # ignores the 'inplace' flag, and assumes everything goes in the | 
					
						
							|  |  |  |         # "build" tree. | 
					
						
							|  |  |  |         outputs = [] | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         for ext in self.extensions: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             fullname = self.get_ext_fullname(ext.name) | 
					
						
							|  |  |  |             outputs.append(os.path.join(self.build_lib, | 
					
						
							|  |  |  |                                         self.get_ext_filename(fullname))) | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  |         return outputs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # get_outputs () | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |     def build_extensions(self): | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  |         # First, sanity-check the 'extensions' list | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         self.check_extensions_list(self.extensions) | 
					
						
							| 
									
										
										
										
											2000-03-29 02:13:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-31 01:09:52 +00:00
										 |  |  |         for ext in self.extensions: | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |             self.build_extension(ext) | 
					
						
							| 
									
										
										
										
											1999-09-29 12:49:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |     def build_extension(self, ext): | 
					
						
							|  |  |  |         sources = ext.sources | 
					
						
							|  |  |  |         if sources is None or type(sources) not in (ListType, TupleType): | 
					
						
							|  |  |  |             raise DistutilsSetupError, \ | 
					
						
							|  |  |  |                   ("in 'ext_modules' option (extension '%s'), " + | 
					
						
							|  |  |  |                    "'sources' must be present and must be " + | 
					
						
							|  |  |  |                    "a list of source filenames") % ext.name | 
					
						
							|  |  |  |         sources = list(sources) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fullname = self.get_ext_fullname(ext.name) | 
					
						
							|  |  |  |         if self.inplace: | 
					
						
							|  |  |  |             # ignore build-lib -- put the compiled extension into | 
					
						
							|  |  |  |             # the source tree along with pure Python modules | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             modpath = string.split(fullname, '.') | 
					
						
							|  |  |  |             package = string.join(modpath[0:-1], '.') | 
					
						
							|  |  |  |             base = modpath[-1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             build_py = self.get_finalized_command('build_py') | 
					
						
							|  |  |  |             package_dir = build_py.get_package_dir(package) | 
					
						
							|  |  |  |             ext_filename = os.path.join(package_dir, | 
					
						
							|  |  |  |                                         self.get_ext_filename(base)) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             ext_filename = os.path.join(self.build_lib, | 
					
						
							|  |  |  |                                         self.get_ext_filename(fullname)) | 
					
						
							| 
									
										
										
										
											2002-06-12 20:08:56 +00:00
										 |  |  |         depends = sources + ext.depends | 
					
						
							|  |  |  |         if not (self.force or newer_group(depends, ext_filename, 'newer')): | 
					
						
							| 
									
										
										
										
											2002-06-04 20:14:43 +00:00
										 |  |  |             log.debug("skipping '%s' extension (up-to-date)", ext.name) | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |             return | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2002-06-04 20:14:43 +00:00
										 |  |  |             log.info("building '%s' extension", ext.name) | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # First, scan the sources for SWIG definition files (.i), run | 
					
						
							|  |  |  |         # SWIG on 'em to create .c files, and modify the sources list | 
					
						
							|  |  |  |         # accordingly. | 
					
						
							|  |  |  |         sources = self.swig_sources(sources) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Next, compile the source code to object files. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # XXX not honouring 'define_macros' or 'undef_macros' -- the | 
					
						
							|  |  |  |         # CCompiler API needs to change to accommodate this, and I | 
					
						
							|  |  |  |         # want to do one thing at a time! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Two possible sources for extra compiler arguments: | 
					
						
							|  |  |  |         #   - 'extra_compile_args' in Extension object | 
					
						
							|  |  |  |         #   - CFLAGS environment variable (not particularly | 
					
						
							|  |  |  |         #     elegant, but people seem to expect it and I | 
					
						
							|  |  |  |         #     guess it's useful) | 
					
						
							|  |  |  |         # The environment variable should take precedence, and | 
					
						
							|  |  |  |         # any sensible compiler will give precedence to later | 
					
						
							|  |  |  |         # command line args.  Hence we combine them in order: | 
					
						
							|  |  |  |         extra_args = ext.extra_compile_args or [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         macros = ext.define_macros[:] | 
					
						
							|  |  |  |         for undef in ext.undef_macros: | 
					
						
							|  |  |  |             macros.append((undef,)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         objects = self.compiler.compile(sources, | 
					
						
							|  |  |  |                                         output_dir=self.build_temp, | 
					
						
							|  |  |  |                                         macros=macros, | 
					
						
							|  |  |  |                                         include_dirs=ext.include_dirs, | 
					
						
							|  |  |  |                                         debug=self.debug, | 
					
						
							| 
									
										
										
										
											2002-06-13 17:32:20 +00:00
										 |  |  |                                         extra_postargs=extra_args, | 
					
						
							|  |  |  |                                         depends=ext.depends) | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-12-06 22:59:54 +00:00
										 |  |  |         # XXX -- this is a Vile HACK! | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         # The setup.py script for Python on Unix needs to be able to | 
					
						
							|  |  |  |         # get this list so it can perform all the clean up needed to | 
					
						
							|  |  |  |         # avoid keeping object files around when cleaning out a failed | 
					
						
							|  |  |  |         # build of an extension module.  Since Distutils does not | 
					
						
							|  |  |  |         # track dependencies, we have to get rid of intermediates to | 
					
						
							|  |  |  |         # ensure all the intermediates will be properly re-built. | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         self._built_objects = objects[:] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |         # Now link the object files together into a "shared object" -- | 
					
						
							|  |  |  |         # of course, first we have to figure out all the other things | 
					
						
							|  |  |  |         # that go into the mix. | 
					
						
							|  |  |  |         if ext.extra_objects: | 
					
						
							|  |  |  |             objects.extend(ext.extra_objects) | 
					
						
							|  |  |  |         extra_args = ext.extra_link_args or [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												This patch fixes the following bugs:
[#413582] g++ must be called for c++ extensions
[#454030] distutils cannot link C++ code with GCC
topdir = "Lib/distutils"
* bcppcompiler.py
  (BCPPCompiler.create_static_lib): Fixed prototype, removing extra_preargs
  and extra_postargs parameters. Included target_lang parameter.
  (BCPPCompiler.link): Included target_lang parameter.
* msvccompiler.py
  (MSVCCompiler.create_static_lib): Fixed prototype, removing extra_preargs
  and extra_postargs parameters. Included target_lang parameter.
  (MSVCCompiler.link): Included target_lang parameter.
* ccompiler.py
  (CCompiler): New language_map and language_order attributes, used by
  CCompiler.detect_language().
  (CCompiler.detect_language): New method, will return the language of
  a given source, or list of sources. Individual source language is
  detected using the language_map dict. When mixed sources are used,
  language_order will stablish the language precedence.
  (CCompiler.create_static_lib, CCompiler.link, CCompiler.link_executable,
   CCompiler.link_shared_object, CCompiler.link_shared_lib):
  Inlcuded target_lang parameter.
* cygwinccompiler.py
  (CygwinCCompiler.link): Included target_lang parameter.
* emxccompiler.py
  (EMXCCompiler.link): Included target_lang parameter.
* mwerkscompiler.py
  (MWerksCompiler.link): Included target_lang parameter.
* extension.py
  (Extension.__init__): New 'language' parameter/attribute, initialized
  to None by default. If provided will overlap the automatic detection
  made by CCompiler.detect_language(), in build_ext command.
* sysconfig.py
  (customize_compiler): Check Makefile for CXX option, and also the
  environment variable CXX. Use the resulting value in the 'compiler_cxx'
  parameter of compiler.set_executables().
* unixccompiler.py
  (UnixCCompiler): Included 'compiler_cxx' in executables dict, defaulting
  to 'cc'.
  (UnixCCompiler.create_static_lib): Included target_lang parameter.
  (UnixCCompiler.link): Included target_lang parameter, and made
  linker command use compiler_cxx, if target_lang is 'c++'.
* command/build_ext.py
  (build_ext.build_extension): Pass new ext.language attribute
  to compiler.link_shared_object()'s target_lang parameter. If
  ext.language is not provided, detect language using
  compiler.detect_language(sources) instead.
* command/config.py
  (config._link): Pass already available lang parameter as target_lang
  parameter of compiler.link_executable().
											
										 
											2002-11-05 16:12:02 +00:00
										 |  |  |         # Detect target language, if not provided | 
					
						
							|  |  |  |         language = ext.language or self.compiler.detect_language(sources) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |         self.compiler.link_shared_object( | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  |             objects, ext_filename, | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |             libraries=self.get_libraries(ext), | 
					
						
							|  |  |  |             library_dirs=ext.library_dirs, | 
					
						
							|  |  |  |             runtime_library_dirs=ext.runtime_library_dirs, | 
					
						
							|  |  |  |             extra_postargs=extra_args, | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  |             export_symbols=self.get_export_symbols(ext), | 
					
						
							| 
									
										
										
										
											2001-01-26 18:00:48 +00:00
										 |  |  |             debug=self.debug, | 
					
						
							| 
									
										
											  
											
												This patch fixes the following bugs:
[#413582] g++ must be called for c++ extensions
[#454030] distutils cannot link C++ code with GCC
topdir = "Lib/distutils"
* bcppcompiler.py
  (BCPPCompiler.create_static_lib): Fixed prototype, removing extra_preargs
  and extra_postargs parameters. Included target_lang parameter.
  (BCPPCompiler.link): Included target_lang parameter.
* msvccompiler.py
  (MSVCCompiler.create_static_lib): Fixed prototype, removing extra_preargs
  and extra_postargs parameters. Included target_lang parameter.
  (MSVCCompiler.link): Included target_lang parameter.
* ccompiler.py
  (CCompiler): New language_map and language_order attributes, used by
  CCompiler.detect_language().
  (CCompiler.detect_language): New method, will return the language of
  a given source, or list of sources. Individual source language is
  detected using the language_map dict. When mixed sources are used,
  language_order will stablish the language precedence.
  (CCompiler.create_static_lib, CCompiler.link, CCompiler.link_executable,
   CCompiler.link_shared_object, CCompiler.link_shared_lib):
  Inlcuded target_lang parameter.
* cygwinccompiler.py
  (CygwinCCompiler.link): Included target_lang parameter.
* emxccompiler.py
  (EMXCCompiler.link): Included target_lang parameter.
* mwerkscompiler.py
  (MWerksCompiler.link): Included target_lang parameter.
* extension.py
  (Extension.__init__): New 'language' parameter/attribute, initialized
  to None by default. If provided will overlap the automatic detection
  made by CCompiler.detect_language(), in build_ext command.
* sysconfig.py
  (customize_compiler): Check Makefile for CXX option, and also the
  environment variable CXX. Use the resulting value in the 'compiler_cxx'
  parameter of compiler.set_executables().
* unixccompiler.py
  (UnixCCompiler): Included 'compiler_cxx' in executables dict, defaulting
  to 'cc'.
  (UnixCCompiler.create_static_lib): Included target_lang parameter.
  (UnixCCompiler.link): Included target_lang parameter, and made
  linker command use compiler_cxx, if target_lang is 'c++'.
* command/build_ext.py
  (build_ext.build_extension): Pass new ext.language attribute
  to compiler.link_shared_object()'s target_lang parameter. If
  ext.language is not provided, detect language using
  compiler.detect_language(sources) instead.
* command/config.py
  (config._link): Pass already available lang parameter as target_lang
  parameter of compiler.link_executable().
											
										 
											2002-11-05 16:12:02 +00:00
										 |  |  |             build_temp=self.build_temp, | 
					
						
							|  |  |  |             target_lang=language) | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |     def swig_sources (self, sources): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """Walk the list of source files in 'sources', looking for SWIG
 | 
					
						
							|  |  |  |         interface (.i) files.  Run SWIG on all that are found, and | 
					
						
							|  |  |  |         return a modified 'sources' list with SWIG source files replaced | 
					
						
							|  |  |  |         by the generated C (or C++) files. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         new_sources = [] | 
					
						
							|  |  |  |         swig_sources = [] | 
					
						
							|  |  |  |         swig_targets = {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |         # XXX this drops generated C/C++ files into the source tree, which | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |         # is fine for developers who want to distribute the generated | 
					
						
							|  |  |  |         # source -- but there should be an option to put SWIG output in | 
					
						
							|  |  |  |         # the temp dir. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |         if self.swig_cpp: | 
					
						
							|  |  |  |             target_ext = '.cpp' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             target_ext = '.c' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |         for source in sources: | 
					
						
							|  |  |  |             (base, ext) = os.path.splitext(source) | 
					
						
							| 
									
										
										
										
											2000-06-25 02:10:46 +00:00
										 |  |  |             if ext == ".i":             # SWIG interface file | 
					
						
							| 
									
										
										
										
											2002-01-18 20:30:53 +00:00
										 |  |  |                 new_sources.append(base + '_wrap' + target_ext) | 
					
						
							| 
									
										
										
										
											2000-06-25 02:10:46 +00:00
										 |  |  |                 swig_sources.append(source) | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |                 swig_targets[source] = new_sources[-1] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 new_sources.append(source) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-25 02:10:46 +00:00
										 |  |  |         if not swig_sources: | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |             return new_sources | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         swig = self.find_swig() | 
					
						
							| 
									
										
										
										
											2002-01-18 20:30:53 +00:00
										 |  |  |         swig_cmd = [swig, "-python"] | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |         if self.swig_cpp: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             swig_cmd.append("-c++") | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for source in swig_sources: | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |             target = swig_targets[source] | 
					
						
							| 
									
										
										
										
											2002-06-04 20:14:43 +00:00
										 |  |  |             log.info("swigging %s to %s", source, target) | 
					
						
							| 
									
										
										
										
											2000-06-27 01:37:10 +00:00
										 |  |  |             self.spawn(swig_cmd + ["-o", target, source]) | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return new_sources | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # swig_sources () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def find_swig (self): | 
					
						
							|  |  |  |         """Return the name of the SWIG executable.  On Unix, this is
 | 
					
						
							|  |  |  |         just "swig" -- it should be in the PATH.  Tries a bit harder on | 
					
						
							|  |  |  |         Windows. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if os.name == "posix": | 
					
						
							|  |  |  |             return "swig" | 
					
						
							|  |  |  |         elif os.name == "nt": | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Look for SWIG in its standard installation directory on | 
					
						
							|  |  |  |             # Windows (or so I presume!).  If we find it there, great; | 
					
						
							|  |  |  |             # if not, act like Unix and assume it's in the PATH. | 
					
						
							|  |  |  |             for vers in ("1.3", "1.2", "1.1"): | 
					
						
							|  |  |  |                 fn = os.path.join("c:\\swig%s" % vers, "swig.exe") | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |                 if os.path.isfile(fn): | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |                     return fn | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return "swig.exe" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-31 18:56:00 +00:00
										 |  |  |         elif os.name == "os2": | 
					
						
							|  |  |  |             # assume swig available in the PATH. | 
					
						
							|  |  |  |             return "swig.exe" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-24 00:19:35 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             raise DistutilsPlatformError, \ | 
					
						
							|  |  |  |                   ("I don't know how to find (much less run) SWIG " | 
					
						
							|  |  |  |                    "on platform '%s'") % os.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # find_swig () | 
					
						
							| 
									
										
										
										
											2001-12-06 21:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-17 23:04:31 +00:00
										 |  |  |     # -- Name generators ----------------------------------------------- | 
					
						
							|  |  |  |     # (extension names, filenames, whatever) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-01 01:43:28 +00:00
										 |  |  |     def get_ext_fullname (self, ext_name): | 
					
						
							|  |  |  |         if self.package is None: | 
					
						
							|  |  |  |             return ext_name | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return self.package + '.' + ext_name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_ext_filename (self, ext_name): | 
					
						
							| 
									
										
										
										
											2001-03-10 09:33:14 +00:00
										 |  |  |         r"""Convert the name of an extension (eg. "foo.bar") into the name
 | 
					
						
							| 
									
										
										
										
											2000-06-17 23:04:31 +00:00
										 |  |  |         of the file from which it will be loaded (eg. "foo/bar.so", or | 
					
						
							|  |  |  |         "foo\bar.pyd"). | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-15 01:19:03 +00:00
										 |  |  |         from distutils.sysconfig import get_config_var | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |         ext_path = string.split(ext_name, '.') | 
					
						
							| 
									
										
										
										
											2002-01-31 18:56:00 +00:00
										 |  |  |         # OS/2 has an 8 character module (extension) limit :-( | 
					
						
							|  |  |  |         if os.name == "os2": | 
					
						
							|  |  |  |             ext_path[len(ext_path) - 1] = ext_path[len(ext_path) - 1][:8] | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |         # extensions in debug_mode are named 'module_d.pyd' under windows | 
					
						
							| 
									
										
										
										
											2000-09-15 01:19:03 +00:00
										 |  |  |         so_ext = get_config_var('SO') | 
					
						
							| 
									
										
										
										
											2000-03-31 03:50:23 +00:00
										 |  |  |         if os.name == 'nt' and self.debug: | 
					
						
							| 
									
										
										
										
											2000-09-30 18:27:54 +00:00
										 |  |  |             return apply(os.path.join, ext_path) + '_d' + so_ext | 
					
						
							|  |  |  |         return apply(os.path.join, ext_path) + so_ext | 
					
						
							| 
									
										
										
										
											1999-08-14 23:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  |     def get_export_symbols (self, ext): | 
					
						
							|  |  |  |         """Return the list of symbols that a shared extension has to
 | 
					
						
							|  |  |  |         export.  This either uses 'ext.export_symbols' or, if it's not | 
					
						
							|  |  |  |         provided, "init" + module_name.  Only relevant on Windows, where | 
					
						
							|  |  |  |         the .pyd file (DLL) must export the module "init" function. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-13 00:42:35 +00:00
										 |  |  |         initfunc_name = "init" + string.split(ext.name,'.')[-1] | 
					
						
							|  |  |  |         if initfunc_name not in ext.export_symbols: | 
					
						
							|  |  |  |             ext.export_symbols.append(initfunc_name) | 
					
						
							|  |  |  |         return ext.export_symbols | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_libraries (self, ext): | 
					
						
							|  |  |  |         """Return the list of libraries to link against when building a
 | 
					
						
							|  |  |  |         shared extension.  On most platforms, this is just 'ext.libraries'; | 
					
						
							| 
									
										
										
										
											2002-01-31 18:56:00 +00:00
										 |  |  |         on Windows and OS/2, we add the Python library (eg. python20.dll). | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         # The python library is always needed on Windows.  For MSVC, this | 
					
						
							|  |  |  |         # is redundant, since the library is mentioned in a pragma in | 
					
						
							| 
									
										
										
										
											2001-07-26 13:41:06 +00:00
										 |  |  |         # pyconfig.h that MSVC groks.  The other Windows compilers all seem | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  |         # to need it mentioned explicitly, though, so that's what we do. | 
					
						
							| 
									
										
										
										
											2000-09-28 19:28:35 +00:00
										 |  |  |         # Append '_d' to the python import library on debug builds. | 
					
						
							| 
									
										
										
										
											2002-06-18 18:40:54 +00:00
										 |  |  |         if sys.platform == "win32": | 
					
						
							|  |  |  |             from distutils.msvccompiler import MSVCCompiler | 
					
						
							|  |  |  |             if not isinstance(self.compiler, MSVCCompiler): | 
					
						
							|  |  |  |                 template = "python%d%d" | 
					
						
							|  |  |  |                 if self.debug: | 
					
						
							|  |  |  |                     template = template + '_d' | 
					
						
							|  |  |  |                 pythonlib = (template % | 
					
						
							|  |  |  |                        (sys.hexversion >> 24, (sys.hexversion >> 16) & 0xff)) | 
					
						
							|  |  |  |                 # don't extend ext.libraries, it may be shared with other | 
					
						
							|  |  |  |                 # extensions, it is a reference to the original list | 
					
						
							|  |  |  |                 return ext.libraries + [pythonlib] | 
					
						
							| 
									
										
										
										
											2002-10-31 14:26:37 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 return ext.libraries | 
					
						
							| 
									
										
										
										
											2002-01-31 18:56:00 +00:00
										 |  |  |         elif sys.platform == "os2emx": | 
					
						
							|  |  |  |             # EMX/GCC requires the python library explicitly, and I | 
					
						
							|  |  |  |             # believe VACPP does as well (though not confirmed) - AIM Apr01 | 
					
						
							|  |  |  |             template = "python%d%d" | 
					
						
							|  |  |  |             # debug versions of the main DLL aren't supported, at least  | 
					
						
							|  |  |  |             # not at this time - AIM Apr01 | 
					
						
							|  |  |  |             #if self.debug: | 
					
						
							|  |  |  |             #    template = template + '_d' | 
					
						
							|  |  |  |             pythonlib = (template % | 
					
						
							|  |  |  |                    (sys.hexversion >> 24, (sys.hexversion >> 16) & 0xff)) | 
					
						
							|  |  |  |             # don't extend ext.libraries, it may be shared with other | 
					
						
							|  |  |  |             # extensions, it is a reference to the original list | 
					
						
							|  |  |  |             return ext.libraries + [pythonlib] | 
					
						
							| 
									
										
										
										
											2001-01-24 15:43:09 +00:00
										 |  |  |         elif sys.platform[:6] == "cygwin": | 
					
						
							|  |  |  |             template = "python%d.%d" | 
					
						
							|  |  |  |             pythonlib = (template % | 
					
						
							|  |  |  |                    (sys.hexversion >> 24, (sys.hexversion >> 16) & 0xff)) | 
					
						
							|  |  |  |             # don't extend ext.libraries, it may be shared with other | 
					
						
							|  |  |  |             # extensions, it is a reference to the original list | 
					
						
							|  |  |  |             return ext.libraries + [pythonlib] | 
					
						
							| 
									
										
										
										
											2002-06-11 06:22:31 +00:00
										 |  |  |         elif sys.platform[:6] == "atheos": | 
					
						
							|  |  |  |             from distutils import sysconfig | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             template = "python%d.%d" | 
					
						
							|  |  |  |             pythonlib = (template % | 
					
						
							|  |  |  |                    (sys.hexversion >> 24, (sys.hexversion >> 16) & 0xff)) | 
					
						
							|  |  |  |             # Get SHLIBS from Makefile | 
					
						
							|  |  |  |             extra = [] | 
					
						
							|  |  |  |             for lib in sysconfig.get_config_var('SHLIBS').split(): | 
					
						
							|  |  |  |                 if lib.startswith('-l'): | 
					
						
							|  |  |  |                     extra.append(lib[2:]) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     extra.append(lib) | 
					
						
							|  |  |  |             # don't extend ext.libraries, it may be shared with other | 
					
						
							|  |  |  |             # extensions, it is a reference to the original list | 
					
						
							|  |  |  |             return ext.libraries + [pythonlib, "m"] + extra | 
					
						
							| 
									
										
										
										
											2000-08-01 23:54:29 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             return ext.libraries | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-25 01:10:04 +00:00
										 |  |  | # class build_ext |