| 
									
										
										
										
											2001-07-18 18:39:56 +00:00
										 |  |  | """Provide access to Python's configuration information.  The specific
 | 
					
						
							|  |  |  | configuration variables available depend heavily on the platform and | 
					
						
							|  |  |  | configuration.  The values may be retrieved using | 
					
						
							|  |  |  | get_config_var(name), and the list of variables is available via | 
					
						
							|  |  |  | get_config_vars().keys().  Additional convenience functions are also | 
					
						
							|  |  |  | available. | 
					
						
							| 
									
										
										
										
											1998-12-18 23:46:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Written by:   Fred L. Drake, Jr. | 
					
						
							|  |  |  | Email:        <fdrake@acm.org> | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-01-06 14:46:06 +00:00
										 |  |  | import os | 
					
						
							|  |  |  | import re | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  | import sys | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from .errors import DistutilsPlatformError | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # These are needed in a couple of spots, so just compute them once. | 
					
						
							|  |  |  | PREFIX = os.path.normpath(sys.prefix) | 
					
						
							|  |  |  | EXEC_PREFIX = os.path.normpath(sys.exec_prefix) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Path to the base directory of the project. On Windows the binary may | 
					
						
							|  |  |  | # live in project/PCBuild9.  If we're dealing with an x64 Windows build, | 
					
						
							|  |  |  | # it'll live in project/PCbuild/amd64. | 
					
						
							|  |  |  | project_base = os.path.dirname(os.path.abspath(sys.executable)) | 
					
						
							|  |  |  | if os.name == "nt" and "pcbuild" in project_base[-8:].lower(): | 
					
						
							|  |  |  |     project_base = os.path.abspath(os.path.join(project_base, os.path.pardir)) | 
					
						
							|  |  |  | # PC/VS7.1 | 
					
						
							|  |  |  | if os.name == "nt" and "\\pc\\v" in project_base[-10:].lower(): | 
					
						
							|  |  |  |     project_base = os.path.abspath(os.path.join(project_base, os.path.pardir, | 
					
						
							|  |  |  |                                                 os.path.pardir)) | 
					
						
							|  |  |  | # PC/AMD64 | 
					
						
							|  |  |  | if os.name == "nt" and "\\pcbuild\\amd64" in project_base[-14:].lower(): | 
					
						
							|  |  |  |     project_base = os.path.abspath(os.path.join(project_base, os.path.pardir, | 
					
						
							|  |  |  |                                                 os.path.pardir)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # python_build: (Boolean) if true, we're either building Python or | 
					
						
							|  |  |  | # building an extension with an un-installed Python, so we use | 
					
						
							|  |  |  | # different (hard-wired) directories. | 
					
						
							|  |  |  | # Setup.local is available for Makefile builds including VPATH builds, | 
					
						
							|  |  |  | # Setup.dist is available on Windows | 
					
						
							| 
									
										
										
										
											2010-01-29 11:41:03 +00:00
										 |  |  | def _python_build(): | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     for fn in ("Setup.dist", "Setup.local"): | 
					
						
							|  |  |  |         if os.path.isfile(os.path.join(project_base, "Modules", fn)): | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |     return False | 
					
						
							| 
									
										
										
										
											2010-01-29 11:41:03 +00:00
										 |  |  | python_build = _python_build() | 
					
						
							| 
									
										
										
										
											2002-11-14 01:43:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-24 19:43:47 +00:00
										 |  |  | # Calculate the build qualifier flags if they are defined.  Adding the flags | 
					
						
							|  |  |  | # to the include and lib directories only makes sense for an installation, not | 
					
						
							|  |  |  | # an in-source build. | 
					
						
							|  |  |  | build_flags = '' | 
					
						
							|  |  |  | try: | 
					
						
							|  |  |  |     if not python_build: | 
					
						
							|  |  |  |         build_flags = sys.abiflags | 
					
						
							|  |  |  | except AttributeError: | 
					
						
							|  |  |  |     # It's not a configure-based build, so the sys module doesn't have | 
					
						
							|  |  |  |     # this attribute, which is fine. | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  | def get_python_version(): | 
					
						
							|  |  |  |     """Return a string containing the major and minor Python version,
 | 
					
						
							|  |  |  |     leaving off the patchlevel.  Sample return values could be '1.5' | 
					
						
							|  |  |  |     or '2.2'. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     return sys.version[:3] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-29 11:41:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  | def get_python_inc(plat_specific=0, prefix=None): | 
					
						
							|  |  |  |     """Return the directory containing installed Python header files.
 | 
					
						
							| 
									
										
										
										
											2000-03-09 15:54:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     If 'plat_specific' is false (the default), this is the path to the | 
					
						
							|  |  |  |     non-platform-specific header files, i.e. Python.h and so on; | 
					
						
							|  |  |  |     otherwise, this is the path to platform-specific header files | 
					
						
							| 
									
										
										
										
											2001-07-26 13:41:06 +00:00
										 |  |  |     (namely pyconfig.h). | 
					
						
							| 
									
										
										
										
											2000-03-09 15:54:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-04-10 01:17:49 +00:00
										 |  |  |     If 'prefix' is supplied, use it instead of sys.prefix or | 
					
						
							|  |  |  |     sys.exec_prefix -- i.e., ignore 'plat_specific'. | 
					
						
							| 
									
										
										
										
											2001-12-06 20:51:35 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     if prefix is None: | 
					
						
							|  |  |  |         prefix = plat_specific and EXEC_PREFIX or PREFIX | 
					
						
							|  |  |  |     if os.name == "posix": | 
					
						
							|  |  |  |         if python_build: | 
					
						
							| 
									
										
										
										
											2010-09-20 10:29:54 +00:00
										 |  |  |             # Assume the executable is in the build directory.  The | 
					
						
							|  |  |  |             # pyconfig.h file should be in the same directory.  Since | 
					
						
							|  |  |  |             # the build directory may not be the source directory, we | 
					
						
							|  |  |  |             # must use "srcdir" from the makefile to find the "Include" | 
					
						
							|  |  |  |             # directory. | 
					
						
							|  |  |  |             base = os.path.dirname(os.path.abspath(sys.executable)) | 
					
						
							|  |  |  |             if plat_specific: | 
					
						
							|  |  |  |                 return base | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 incdir = os.path.join(get_config_var('srcdir'), 'Include') | 
					
						
							|  |  |  |                 return os.path.normpath(incdir) | 
					
						
							| 
									
										
										
										
											2010-11-24 19:43:47 +00:00
										 |  |  |         python_dir = 'python' + get_python_version() + build_flags | 
					
						
							|  |  |  |         return os.path.join(prefix, "include", python_dir) | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     elif os.name == "nt": | 
					
						
							|  |  |  |         return os.path.join(prefix, "include") | 
					
						
							|  |  |  |     elif os.name == "os2": | 
					
						
							|  |  |  |         return os.path.join(prefix, "Include") | 
					
						
							| 
									
										
										
										
											2000-03-09 03:16:05 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |         raise DistutilsPlatformError( | 
					
						
							|  |  |  |             "I don't know where Python installs its C header files " | 
					
						
							|  |  |  |             "on platform '%s'" % os.name) | 
					
						
							| 
									
										
										
										
											2000-03-09 03:16:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  | def get_python_lib(plat_specific=0, standard_lib=0, prefix=None): | 
					
						
							|  |  |  |     """Return the directory containing the Python library (standard or
 | 
					
						
							| 
									
										
										
										
											2000-03-09 15:54:52 +00:00
										 |  |  |     site additions). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If 'plat_specific' is true, return the directory containing | 
					
						
							|  |  |  |     platform-specific modules, i.e. any module from a non-pure-Python | 
					
						
							|  |  |  |     module distribution; otherwise, return the platform-shared library | 
					
						
							|  |  |  |     directory.  If 'standard_lib' is true, return the directory | 
					
						
							|  |  |  |     containing standard Python library modules; otherwise, return the | 
					
						
							|  |  |  |     directory for site-specific modules. | 
					
						
							| 
									
										
										
										
											2000-03-09 03:16:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-04-10 01:17:49 +00:00
										 |  |  |     If 'prefix' is supplied, use it instead of sys.prefix or | 
					
						
							|  |  |  |     sys.exec_prefix -- i.e., ignore 'plat_specific'. | 
					
						
							| 
									
										
										
										
											2000-03-09 15:54:52 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     if prefix is None: | 
					
						
							|  |  |  |         prefix = plat_specific and EXEC_PREFIX or PREFIX | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if os.name == "posix": | 
					
						
							|  |  |  |         libpython = os.path.join(prefix, | 
					
						
							|  |  |  |                                  "lib", "python" + get_python_version()) | 
					
						
							|  |  |  |         if standard_lib: | 
					
						
							|  |  |  |             return libpython | 
					
						
							| 
									
										
										
										
											2000-03-09 03:16:05 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |             return os.path.join(libpython, "site-packages") | 
					
						
							|  |  |  |     elif os.name == "nt": | 
					
						
							|  |  |  |         if standard_lib: | 
					
						
							|  |  |  |             return os.path.join(prefix, "Lib") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             if get_python_version() < "2.2": | 
					
						
							|  |  |  |                 return prefix | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return os.path.join(prefix, "Lib", "site-packages") | 
					
						
							|  |  |  |     elif os.name == "os2": | 
					
						
							|  |  |  |         if standard_lib: | 
					
						
							|  |  |  |             return os.path.join(prefix, "Lib") | 
					
						
							| 
									
										
										
										
											2002-01-31 18:56:00 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |             return os.path.join(prefix, "Lib", "site-packages") | 
					
						
							| 
									
										
										
										
											2000-03-09 03:16:05 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |         raise DistutilsPlatformError( | 
					
						
							|  |  |  |             "I don't know where Python installs its library " | 
					
						
							|  |  |  |             "on platform '%s'" % os.name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def customize_compiler(compiler): | 
					
						
							|  |  |  |     """Do any platform-specific customization of a CCompiler instance.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Mainly needed on Unix, so we can plug in the information that | 
					
						
							|  |  |  |     varies across Unices and is stored in Python's Makefile. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     if compiler.compiler_type == "unix": | 
					
						
							|  |  |  |         (cc, cxx, opt, cflags, ccshared, ldshared, so_ext, ar, ar_flags) = \ | 
					
						
							|  |  |  |             get_config_vars('CC', 'CXX', 'OPT', 'CFLAGS', | 
					
						
							|  |  |  |                             'CCSHARED', 'LDSHARED', 'SO', 'AR', 'ARFLAGS') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if 'CC' in os.environ: | 
					
						
							|  |  |  |             cc = os.environ['CC'] | 
					
						
							|  |  |  |         if 'CXX' in os.environ: | 
					
						
							|  |  |  |             cxx = os.environ['CXX'] | 
					
						
							|  |  |  |         if 'LDSHARED' in os.environ: | 
					
						
							|  |  |  |             ldshared = os.environ['LDSHARED'] | 
					
						
							|  |  |  |         if 'CPP' in os.environ: | 
					
						
							|  |  |  |             cpp = os.environ['CPP'] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cpp = cc + " -E"           # not always | 
					
						
							|  |  |  |         if 'LDFLAGS' in os.environ: | 
					
						
							|  |  |  |             ldshared = ldshared + ' ' + os.environ['LDFLAGS'] | 
					
						
							|  |  |  |         if 'CFLAGS' in os.environ: | 
					
						
							|  |  |  |             cflags = opt + ' ' + os.environ['CFLAGS'] | 
					
						
							|  |  |  |             ldshared = ldshared + ' ' + os.environ['CFLAGS'] | 
					
						
							|  |  |  |         if 'CPPFLAGS' in os.environ: | 
					
						
							|  |  |  |             cpp = cpp + ' ' + os.environ['CPPFLAGS'] | 
					
						
							|  |  |  |             cflags = cflags + ' ' + os.environ['CPPFLAGS'] | 
					
						
							|  |  |  |             ldshared = ldshared + ' ' + os.environ['CPPFLAGS'] | 
					
						
							|  |  |  |         if 'AR' in os.environ: | 
					
						
							|  |  |  |             ar = os.environ['AR'] | 
					
						
							|  |  |  |         if 'ARFLAGS' in os.environ: | 
					
						
							|  |  |  |             archiver = ar + ' ' + os.environ['ARFLAGS'] | 
					
						
							| 
									
										
											  
											
												Merged revisions 59376-59406 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r59377 | georg.brandl | 2007-12-06 01:24:23 +0100 (Thu, 06 Dec 2007) | 2 lines
  Add another GHOP student to ACKS.
........
  r59378 | raymond.hettinger | 2007-12-06 01:56:53 +0100 (Thu, 06 Dec 2007) | 5 lines
  Fix Issue 1045.
  Factor-out common calling code by simplifying the length_hint API.
  Speed-up the function by caching the PyObject_String for the attribute lookup.
........
  r59380 | georg.brandl | 2007-12-06 02:52:24 +0100 (Thu, 06 Dec 2007) | 2 lines
  Diverse markup fixes.
........
  r59383 | georg.brandl | 2007-12-06 10:45:39 +0100 (Thu, 06 Dec 2007) | 2 lines
  Better re.split examples.
........
  r59386 | christian.heimes | 2007-12-06 14:15:13 +0100 (Thu, 06 Dec 2007) | 2 lines
  Fixed get_config_h_filename for Windows. Without the patch it can't find the pyconfig.h file inside a build tree.
  Added several small unit tests for sysconfig.
........
  r59387 | christian.heimes | 2007-12-06 14:30:11 +0100 (Thu, 06 Dec 2007) | 1 line
  Silence more warnings, _CRT_NONSTDC_NO_DEPRECATE is already defined in pyconfig.h but several projects don't include it.
........
  r59389 | christian.heimes | 2007-12-06 14:55:01 +0100 (Thu, 06 Dec 2007) | 1 line
  Disabled one test that is failing on Unix
........
  r59399 | christian.heimes | 2007-12-06 22:13:06 +0100 (Thu, 06 Dec 2007) | 8 lines
  Several Windows related cleanups:
  * Removed a #define from pyconfig.h. The macro was already defined a few lines higher.
  * Fixed path to tix in the build_tkinter.py script
  * Changed make_buildinfo.c to use versions of unlink and strcat which are considered safe by Windows (as suggested by MvL).
  * Removed two defines from pyproject.vsprops that are no longer required. Both are defined in pyconfig.h and make_buildinfo.c doesn't use the unsafe versions any more (as suggested by MvL).
  * Added some more information about PGO and the property files to PCbuild9/readme.txt.
  Are you fine with the changes, Martin?
........
  r59400 | raymond.hettinger | 2007-12-07 02:53:01 +0100 (Fri, 07 Dec 2007) | 4 lines
  Don't have the docs berate themselves.  Keep a professional tone.
  If a todo is needed, put it in the tracker.
........
  r59402 | georg.brandl | 2007-12-07 10:07:10 +0100 (Fri, 07 Dec 2007) | 3 lines
  Increase unit test coverage of SimpleXMLRPCServer.
  Written for GHOP by Turkay Eren.
........
  r59406 | georg.brandl | 2007-12-07 16:16:57 +0100 (Fri, 07 Dec 2007) | 2 lines
  Update to windows doc from Robert.
........
											
										 
											2007-12-08 15:33:56 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |             archiver = ar + ' ' + ar_flags | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         cc_cmd = cc + ' ' + cflags | 
					
						
							|  |  |  |         compiler.set_executables( | 
					
						
							|  |  |  |             preprocessor=cpp, | 
					
						
							|  |  |  |             compiler=cc_cmd, | 
					
						
							|  |  |  |             compiler_so=cc_cmd + ' ' + ccshared, | 
					
						
							|  |  |  |             compiler_cxx=cxx, | 
					
						
							|  |  |  |             linker_so=ldshared, | 
					
						
							|  |  |  |             linker_exe=cc, | 
					
						
							|  |  |  |             archiver=archiver) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         compiler.shared_lib_extension = so_ext | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_config_h_filename(): | 
					
						
							|  |  |  |     """Return full pathname of installed pyconfig.h file.""" | 
					
						
							|  |  |  |     if python_build: | 
					
						
							|  |  |  |         if os.name == "nt": | 
					
						
							|  |  |  |             inc_dir = os.path.join(project_base, "PC") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             inc_dir = project_base | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         inc_dir = get_python_inc(plat_specific=1) | 
					
						
							|  |  |  |     if get_python_version() < '2.2': | 
					
						
							|  |  |  |         config_h = 'config.h' | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         # The name of the config.h file changed in 2.2 | 
					
						
							|  |  |  |         config_h = 'pyconfig.h' | 
					
						
							|  |  |  |     return os.path.join(inc_dir, config_h) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-18 23:46:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-01-06 14:46:06 +00:00
										 |  |  | def get_makefile_filename(): | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     """Return full pathname of installed Makefile from the Python build.""" | 
					
						
							|  |  |  |     if python_build: | 
					
						
							|  |  |  |         return os.path.join(os.path.dirname(sys.executable), "Makefile") | 
					
						
							|  |  |  |     lib_dir = get_python_lib(plat_specific=1, standard_lib=1) | 
					
						
							| 
									
										
										
										
											2010-11-24 19:43:47 +00:00
										 |  |  |     config_file = 'config-{}{}'.format(get_python_version(), build_flags) | 
					
						
							|  |  |  |     return os.path.join(lib_dir, config_file, 'Makefile') | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def parse_config_h(fp, g=None): | 
					
						
							|  |  |  |     """Parse a config.h-style file.
 | 
					
						
							| 
									
										
										
										
											1998-12-18 23:46:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     A dictionary containing name/value pairs is returned.  If an | 
					
						
							|  |  |  |     optional dictionary is passed in as the second argument, it is | 
					
						
							|  |  |  |     used instead of a new dictionary. | 
					
						
							| 
									
										
										
										
											1999-01-06 16:28:34 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     if g is None: | 
					
						
							|  |  |  |         g = {} | 
					
						
							|  |  |  |     define_rx = re.compile("#define ([A-Z][A-Za-z0-9_]+) (.*)\n") | 
					
						
							|  |  |  |     undef_rx = re.compile("/[*] #undef ([A-Z][A-Za-z0-9_]+) [*]/\n") | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     while True: | 
					
						
							|  |  |  |         line = fp.readline() | 
					
						
							|  |  |  |         if not line: | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  |         m = define_rx.match(line) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             n, v = m.group(1, 2) | 
					
						
							|  |  |  |             try: v = int(v) | 
					
						
							|  |  |  |             except ValueError: pass | 
					
						
							|  |  |  |             g[n] = v | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             m = undef_rx.match(line) | 
					
						
							|  |  |  |             if m: | 
					
						
							|  |  |  |                 g[m.group(1)] = 0 | 
					
						
							|  |  |  |     return g | 
					
						
							| 
									
										
										
										
											1998-12-18 23:46:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-17 00:53:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Regexes needed for parsing Makefile (and similar syntaxes, | 
					
						
							|  |  |  | # like old-style Setup files). | 
					
						
							|  |  |  | _variable_rx = re.compile("([a-zA-Z][a-zA-Z0-9_]+)\s*=\s*(.*)") | 
					
						
							|  |  |  | _findvar1_rx = re.compile(r"\$\(([A-Za-z][A-Za-z0-9_]*)\)") | 
					
						
							|  |  |  | _findvar2_rx = re.compile(r"\${([A-Za-z][A-Za-z0-9_]*)}") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-15 00:03:13 +00:00
										 |  |  | def parse_makefile(fn, g=None): | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     """Parse a Makefile-style file.
 | 
					
						
							| 
									
										
										
										
											2000-03-09 15:54:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     A dictionary containing name/value pairs is returned.  If an | 
					
						
							|  |  |  |     optional dictionary is passed in as the second argument, it is | 
					
						
							|  |  |  |     used instead of a new dictionary. | 
					
						
							| 
									
										
										
										
											1999-01-06 16:28:34 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     from distutils.text_file import TextFile | 
					
						
							| 
									
										
										
										
											2010-10-23 17:02:31 +00:00
										 |  |  |     fp = TextFile(fn, strip_comments=1, skip_blanks=1, join_lines=1, errors="surrogateescape") | 
					
						
							| 
									
										
										
										
											1998-12-18 23:46:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     if g is None: | 
					
						
							|  |  |  |         g = {} | 
					
						
							|  |  |  |     done = {} | 
					
						
							|  |  |  |     notdone = {} | 
					
						
							| 
									
										
										
										
											2010-01-29 11:41:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     while True: | 
					
						
							|  |  |  |         line = fp.readline() | 
					
						
							|  |  |  |         if line is None: # eof | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  |         m = _variable_rx.match(line) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             n, v = m.group(1, 2) | 
					
						
							|  |  |  |             v = v.strip() | 
					
						
							|  |  |  |             # `$$' is a literal `$' in make | 
					
						
							|  |  |  |             tmpv = v.replace('$$', '') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if "$" in tmpv: | 
					
						
							|  |  |  |                 notdone[n] = v | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     v = int(v) | 
					
						
							|  |  |  |                 except ValueError: | 
					
						
							|  |  |  |                     # insert literal `$' | 
					
						
							|  |  |  |                     done[n] = v.replace('$$', '$') | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     done[n] = v | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 09:43:17 +00:00
										 |  |  |     # Variables with a 'PY_' prefix in the makefile. These need to | 
					
						
							|  |  |  |     # be made available without that prefix through sysconfig. | 
					
						
							|  |  |  |     # Special care is needed to ensure that variable expansion works, even | 
					
						
							|  |  |  |     # if the expansion uses the name without a prefix. | 
					
						
							|  |  |  |     renamed_variables = ('CFLAGS', 'LDFLAGS', 'CPPFLAGS') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     # do variable interpolation here | 
					
						
							|  |  |  |     while notdone: | 
					
						
							|  |  |  |         for name in list(notdone): | 
					
						
							|  |  |  |             value = notdone[name] | 
					
						
							|  |  |  |             m = _findvar1_rx.search(value) or _findvar2_rx.search(value) | 
					
						
							|  |  |  |             if m: | 
					
						
							|  |  |  |                 n = m.group(1) | 
					
						
							|  |  |  |                 found = True | 
					
						
							|  |  |  |                 if n in done: | 
					
						
							|  |  |  |                     item = str(done[n]) | 
					
						
							|  |  |  |                 elif n in notdone: | 
					
						
							|  |  |  |                     # get it on a subsequent round | 
					
						
							|  |  |  |                     found = False | 
					
						
							|  |  |  |                 elif n in os.environ: | 
					
						
							|  |  |  |                     # do it like make: fall back to environment | 
					
						
							|  |  |  |                     item = os.environ[n] | 
					
						
							| 
									
										
										
										
											2010-07-23 09:43:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 elif n in renamed_variables: | 
					
						
							|  |  |  |                     if name.startswith('PY_') and name[3:] in renamed_variables: | 
					
						
							|  |  |  |                         item = "" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     elif 'PY_' + n in notdone: | 
					
						
							|  |  |  |                         found = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         item = str(done['PY_' + n]) | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     done[n] = item = "" | 
					
						
							|  |  |  |                 if found: | 
					
						
							|  |  |  |                     after = value[m.end():] | 
					
						
							|  |  |  |                     value = value[:m.start()] + item + after | 
					
						
							|  |  |  |                     if "$" in after: | 
					
						
							|  |  |  |                         notdone[name] = value | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         try: value = int(value) | 
					
						
							|  |  |  |                         except ValueError: | 
					
						
							|  |  |  |                             done[name] = value.strip() | 
					
						
							|  |  |  |                         else: | 
					
						
							|  |  |  |                             done[name] = value | 
					
						
							|  |  |  |                         del notdone[name] | 
					
						
							| 
									
										
										
										
											2010-07-23 09:43:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         if name.startswith('PY_') \ | 
					
						
							|  |  |  |                             and name[3:] in renamed_variables: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             name = name[3:] | 
					
						
							|  |  |  |                             if name not in done: | 
					
						
							|  |  |  |                                 done[name] = value | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 # bogus variable reference; just drop it since we can't deal | 
					
						
							|  |  |  |                 del notdone[name] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fp.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-10 09:37:12 +00:00
										 |  |  |     # strip spurious spaces | 
					
						
							|  |  |  |     for k, v in done.items(): | 
					
						
							|  |  |  |         if isinstance(v, str): | 
					
						
							|  |  |  |             done[k] = v.strip() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     # save the results in the global dictionary | 
					
						
							|  |  |  |     g.update(done) | 
					
						
							|  |  |  |     return g | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def expand_makefile_vars(s, vars): | 
					
						
							|  |  |  |     """Expand Makefile-style variables -- "${foo}" or "$(foo)" -- in
 | 
					
						
							| 
									
										
										
										
											2000-09-17 00:53:02 +00:00
										 |  |  |     'string' according to 'vars' (a dictionary mapping variable names to | 
					
						
							|  |  |  |     values).  Variables not present in 'vars' are silently expanded to the | 
					
						
							|  |  |  |     empty string.  The variable values in 'vars' should not contain further | 
					
						
							|  |  |  |     variable expansions; if 'vars' is the output of 'parse_makefile()', | 
					
						
							|  |  |  |     you're fine.  Returns a variable-expanded version of 's'. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # This algorithm does multiple expansion, so if vars['foo'] contains | 
					
						
							|  |  |  |     # "${bar}", it will expand ${foo} to ${bar}, and then expand | 
					
						
							|  |  |  |     # ${bar}... and so forth.  This is fine as long as 'vars' comes from | 
					
						
							|  |  |  |     # 'parse_makefile()', which takes care of such expansions eagerly, | 
					
						
							|  |  |  |     # according to make's variable expansion semantics. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-30 03:52:21 +00:00
										 |  |  |     while True: | 
					
						
							| 
									
										
										
										
											2000-09-17 00:53:02 +00:00
										 |  |  |         m = _findvar1_rx.search(s) or _findvar2_rx.search(s) | 
					
						
							|  |  |  |         if m: | 
					
						
							|  |  |  |             (beg, end) = m.span() | 
					
						
							|  |  |  |             s = s[0:beg] + vars.get(m.group(1)) + s[end:] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  |     return s | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _config_vars = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _init_posix(): | 
					
						
							|  |  |  |     """Initialize the module as appropriate for POSIX systems.""" | 
					
						
							|  |  |  |     g = {} | 
					
						
							|  |  |  |     # load the installed Makefile: | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         filename = get_makefile_filename() | 
					
						
							|  |  |  |         parse_makefile(filename, g) | 
					
						
							|  |  |  |     except IOError as msg: | 
					
						
							|  |  |  |         my_msg = "invalid Python installation: unable to open %s" % filename | 
					
						
							|  |  |  |         if hasattr(msg, "strerror"): | 
					
						
							|  |  |  |             my_msg = my_msg + " (%s)" % msg.strerror | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         raise DistutilsPlatformError(my_msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # load the installed pyconfig.h: | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         filename = get_config_h_filename() | 
					
						
							| 
									
										
										
										
											2010-10-29 22:36:08 +00:00
										 |  |  |         with open(filename) as file: | 
					
						
							|  |  |  |             parse_config_h(file, g) | 
					
						
							| 
									
										
										
										
											2010-07-22 12:50:05 +00:00
										 |  |  |     except IOError as msg: | 
					
						
							|  |  |  |         my_msg = "invalid Python installation: unable to open %s" % filename | 
					
						
							|  |  |  |         if hasattr(msg, "strerror"): | 
					
						
							|  |  |  |             my_msg = my_msg + " (%s)" % msg.strerror | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         raise DistutilsPlatformError(my_msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # On AIX, there are wrong paths to the linker scripts in the Makefile | 
					
						
							|  |  |  |     # -- these paths are relative to the Python source, but when installed | 
					
						
							|  |  |  |     # the scripts are in another directory. | 
					
						
							|  |  |  |     if python_build: | 
					
						
							|  |  |  |         g['LDSHARED'] = g['BLDSHARED'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif get_python_version() < '2.1': | 
					
						
							|  |  |  |         # The following two branches are for 1.5.2 compatibility. | 
					
						
							|  |  |  |         if sys.platform == 'aix4':          # what about AIX 3.x ? | 
					
						
							|  |  |  |             # Linker script is in the config directory, not in Modules as the | 
					
						
							|  |  |  |             # Makefile says. | 
					
						
							|  |  |  |             python_lib = get_python_lib(standard_lib=1) | 
					
						
							|  |  |  |             ld_so_aix = os.path.join(python_lib, 'config', 'ld_so_aix') | 
					
						
							|  |  |  |             python_exp = os.path.join(python_lib, 'config', 'python.exp') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             g['LDSHARED'] = "%s %s -bI:%s" % (ld_so_aix, g['CC'], python_exp) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     global _config_vars | 
					
						
							|  |  |  |     _config_vars = g | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _init_nt(): | 
					
						
							|  |  |  |     """Initialize the module as appropriate for NT""" | 
					
						
							|  |  |  |     g = {} | 
					
						
							|  |  |  |     # set basic install directories | 
					
						
							|  |  |  |     g['LIBDEST'] = get_python_lib(plat_specific=0, standard_lib=1) | 
					
						
							|  |  |  |     g['BINLIBDEST'] = get_python_lib(plat_specific=1, standard_lib=1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # XXX hmmm.. a normal install puts include files here | 
					
						
							|  |  |  |     g['INCLUDEPY'] = get_python_inc(plat_specific=0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     g['SO'] = '.pyd' | 
					
						
							|  |  |  |     g['EXE'] = ".exe" | 
					
						
							|  |  |  |     g['VERSION'] = get_python_version().replace(".", "") | 
					
						
							|  |  |  |     g['BINDIR'] = os.path.dirname(os.path.abspath(sys.executable)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     global _config_vars | 
					
						
							|  |  |  |     _config_vars = g | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _init_os2(): | 
					
						
							|  |  |  |     """Initialize the module as appropriate for OS/2""" | 
					
						
							|  |  |  |     g = {} | 
					
						
							|  |  |  |     # set basic install directories | 
					
						
							|  |  |  |     g['LIBDEST'] = get_python_lib(plat_specific=0, standard_lib=1) | 
					
						
							|  |  |  |     g['BINLIBDEST'] = get_python_lib(plat_specific=1, standard_lib=1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # XXX hmmm.. a normal install puts include files here | 
					
						
							|  |  |  |     g['INCLUDEPY'] = get_python_inc(plat_specific=0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     g['SO'] = '.pyd' | 
					
						
							|  |  |  |     g['EXE'] = ".exe" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     global _config_vars | 
					
						
							|  |  |  |     _config_vars = g | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_config_vars(*args): | 
					
						
							|  |  |  |     """With no arguments, return a dictionary of all configuration
 | 
					
						
							|  |  |  |     variables relevant for the current platform.  Generally this includes | 
					
						
							|  |  |  |     everything needed to build extensions and install both pure modules and | 
					
						
							|  |  |  |     extensions.  On Unix, this means every variable defined in Python's | 
					
						
							|  |  |  |     installed Makefile; on Windows and Mac OS it's a much smaller set. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     With arguments, return a list of values that result from looking up | 
					
						
							|  |  |  |     each argument in the configuration variable dictionary. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     global _config_vars | 
					
						
							|  |  |  |     if _config_vars is None: | 
					
						
							|  |  |  |         func = globals().get("_init_" + os.name) | 
					
						
							|  |  |  |         if func: | 
					
						
							|  |  |  |             func() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             _config_vars = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Normalized versions of prefix and exec_prefix are handy to have; | 
					
						
							|  |  |  |         # in fact, these are the standard versions used most places in the | 
					
						
							|  |  |  |         # Distutils. | 
					
						
							|  |  |  |         _config_vars['prefix'] = PREFIX | 
					
						
							|  |  |  |         _config_vars['exec_prefix'] = EXEC_PREFIX | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Convert srcdir into an absolute path if it appears necessary. | 
					
						
							|  |  |  |         # Normally it is relative to the build directory.  However, during | 
					
						
							|  |  |  |         # testing, for example, we might be running a non-installed python | 
					
						
							|  |  |  |         # from a different directory. | 
					
						
							|  |  |  |         if python_build and os.name == "posix": | 
					
						
							|  |  |  |             base = os.path.dirname(os.path.abspath(sys.executable)) | 
					
						
							|  |  |  |             if (not os.path.isabs(_config_vars['srcdir']) and | 
					
						
							|  |  |  |                 base != os.getcwd()): | 
					
						
							|  |  |  |                 # srcdir is relative and we are not in the same directory | 
					
						
							|  |  |  |                 # as the executable. Assume executable is in the build | 
					
						
							|  |  |  |                 # directory and make srcdir absolute. | 
					
						
							|  |  |  |                 srcdir = os.path.join(base, _config_vars['srcdir']) | 
					
						
							|  |  |  |                 _config_vars['srcdir'] = os.path.normpath(srcdir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if sys.platform == 'darwin': | 
					
						
							|  |  |  |             kernel_version = os.uname()[2] # Kernel version (8.4.3) | 
					
						
							|  |  |  |             major_version = int(kernel_version.split('.')[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if major_version < 8: | 
					
						
							|  |  |  |                 # On Mac OS X before 10.4, check if -arch and -isysroot | 
					
						
							|  |  |  |                 # are in CFLAGS or LDFLAGS and remove them if they are. | 
					
						
							|  |  |  |                 # This is needed when building extensions on a 10.3 system | 
					
						
							|  |  |  |                 # using a universal build of python. | 
					
						
							|  |  |  |                 for key in ('LDFLAGS', 'BASECFLAGS', | 
					
						
							|  |  |  |                         # a number of derived variables. These need to be | 
					
						
							|  |  |  |                         # patched up as well. | 
					
						
							|  |  |  |                         'CFLAGS', 'PY_CFLAGS', 'BLDSHARED'): | 
					
						
							|  |  |  |                     flags = _config_vars[key] | 
					
						
							|  |  |  |                     flags = re.sub('-arch\s+\w+\s', ' ', flags, re.ASCII) | 
					
						
							|  |  |  |                     flags = re.sub('-isysroot [^ \t]*', ' ', flags) | 
					
						
							|  |  |  |                     _config_vars[key] = flags | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Allow the user to override the architecture flags using | 
					
						
							|  |  |  |                 # an environment variable. | 
					
						
							|  |  |  |                 # NOTE: This name was introduced by Apple in OSX 10.5 and | 
					
						
							|  |  |  |                 # is used by several scripting languages distributed with | 
					
						
							|  |  |  |                 # that OS release. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if 'ARCHFLAGS' in os.environ: | 
					
						
							|  |  |  |                     arch = os.environ['ARCHFLAGS'] | 
					
						
							|  |  |  |                     for key in ('LDFLAGS', 'BASECFLAGS', | 
					
						
							|  |  |  |                         # a number of derived variables. These need to be | 
					
						
							|  |  |  |                         # patched up as well. | 
					
						
							|  |  |  |                         'CFLAGS', 'PY_CFLAGS', 'BLDSHARED'): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         flags = _config_vars[key] | 
					
						
							|  |  |  |                         flags = re.sub('-arch\s+\w+\s', ' ', flags) | 
					
						
							|  |  |  |                         flags = flags + ' ' + arch | 
					
						
							|  |  |  |                         _config_vars[key] = flags | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if args: | 
					
						
							|  |  |  |         vals = [] | 
					
						
							|  |  |  |         for name in args: | 
					
						
							|  |  |  |             vals.append(_config_vars.get(name)) | 
					
						
							|  |  |  |         return vals | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return _config_vars | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_config_var(name): | 
					
						
							|  |  |  |     """Return the value of a single variable using the dictionary
 | 
					
						
							|  |  |  |     returned by 'get_config_vars()'.  Equivalent to | 
					
						
							|  |  |  |     get_config_vars().get(name) | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     return get_config_vars().get(name) |