| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  | # gh-91321: Build a basic C++ test extension to check that the Python C API is | 
					
						
							|  |  |  | # compatible with C++ and does not emit C++ compiler warnings. | 
					
						
							| 
									
										
										
										
											2023-04-12 23:17:36 -05:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  | import platform | 
					
						
							| 
									
										
										
										
											2023-08-23 04:52:56 +02:00
										 |  |  | import shlex | 
					
						
							| 
									
										
										
										
											2025-04-25 10:19:26 +02:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2023-08-23 04:52:56 +02:00
										 |  |  | import sysconfig | 
					
						
							| 
									
										
										
										
											2023-10-06 02:37:28 +02:00
										 |  |  | from test import support | 
					
						
							| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | from setuptools import setup, Extension | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-22 20:30:18 +02:00
										 |  |  | SOURCE = 'extension.cpp' | 
					
						
							| 
									
										
										
										
											2024-06-28 14:41:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-06 02:37:28 +02:00
										 |  |  | if not support.MS_WINDOWS: | 
					
						
							| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  |     # C++ compiler flags for GCC and clang | 
					
						
							|  |  |  |     CPPFLAGS = [ | 
					
						
							|  |  |  |         # gh-91321: The purpose of _testcppext extension is to check that building | 
					
						
							|  |  |  |         # a C++ extension using the Python C API does not emit C++ compiler | 
					
						
							|  |  |  |         # warnings | 
					
						
							|  |  |  |         '-Werror', | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2025-03-04 14:10:09 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     CPPFLAGS_PEDANTIC = [ | 
					
						
							|  |  |  |         # Ask for strict(er) compliance with the standard. | 
					
						
							|  |  |  |         # We cannot do this for c++03 unlimited API, since several headers in | 
					
						
							|  |  |  |         # Include/cpython/ use commas at end of `enum` declarations, a C++11 | 
					
						
							|  |  |  |         # feature for which GCC has no narrower option than -Wpedantic itself. | 
					
						
							|  |  |  |         '-pedantic-errors', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # We also use `long long`, a C++11 feature we can enable individually. | 
					
						
							|  |  |  |         '-Wno-long-long', | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  | else: | 
					
						
							| 
									
										
										
										
											2024-06-28 14:41:37 +02:00
										 |  |  |     # MSVC compiler flags | 
					
						
							|  |  |  |     CPPFLAGS = [ | 
					
						
							| 
									
										
										
										
											2024-09-19 22:32:01 +02:00
										 |  |  |         # Display warnings level 1 to 4 | 
					
						
							|  |  |  |         '/W4', | 
					
						
							| 
									
										
										
										
											2024-06-28 14:41:37 +02:00
										 |  |  |         # Treat all compiler warnings as compiler errors | 
					
						
							|  |  |  |         '/WX', | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2025-03-04 14:10:09 +01:00
										 |  |  |     CPPFLAGS_PEDANTIC = [] | 
					
						
							| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def main(): | 
					
						
							| 
									
										
										
										
											2022-06-14 11:43:08 +02:00
										 |  |  |     cppflags = list(CPPFLAGS) | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |     std = os.environ.get("CPYTHON_TEST_CPP_STD", "") | 
					
						
							|  |  |  |     module_name = os.environ["CPYTHON_TEST_EXT_NAME"] | 
					
						
							| 
									
										
										
										
											2024-12-13 14:23:20 +01:00
										 |  |  |     limited = bool(os.environ.get("CPYTHON_TEST_LIMITED", "")) | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     cppflags = list(CPPFLAGS) | 
					
						
							|  |  |  |     cppflags.append(f'-DMODULE_NAME={module_name}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Add -std=STD or /std:STD (MSVC) compiler flag | 
					
						
							|  |  |  |     if std: | 
					
						
							|  |  |  |         if support.MS_WINDOWS: | 
					
						
							|  |  |  |             cppflags.append(f'/std:{std}') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cppflags.append(f'-std={std}') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-04 14:10:09 +01:00
										 |  |  |         if limited or (std != 'c++03'): | 
					
						
							|  |  |  |             # See CPPFLAGS_PEDANTIC docstring | 
					
						
							|  |  |  |             cppflags.extend(CPPFLAGS_PEDANTIC) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-19 22:58:13 +01:00
										 |  |  |     # gh-105776: When "gcc -std=11" is used as the C++ compiler, -std=c11 | 
					
						
							|  |  |  |     # option emits a C++ compiler warning. Remove "-std11" option from the | 
					
						
							|  |  |  |     # CC command. | 
					
						
							|  |  |  |     cmd = (sysconfig.get_config_var('CC') or '') | 
					
						
							|  |  |  |     if cmd is not None: | 
					
						
							|  |  |  |         if support.MS_WINDOWS: | 
					
						
							|  |  |  |             std_prefix = '/std' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             std_prefix = '-std' | 
					
						
							|  |  |  |         cmd = shlex.split(cmd) | 
					
						
							|  |  |  |         cmd = [arg for arg in cmd if not arg.startswith(std_prefix)] | 
					
						
							|  |  |  |         cmd = shlex.join(cmd) | 
					
						
							|  |  |  |         # CC env var overrides sysconfig CC variable in setuptools | 
					
						
							|  |  |  |         os.environ['CC'] = cmd | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-13 14:23:20 +01:00
										 |  |  |     # Define Py_LIMITED_API macro | 
					
						
							|  |  |  |     if limited: | 
					
						
							|  |  |  |         version = sys.hexversion | 
					
						
							|  |  |  |         cppflags.append(f'-DPy_LIMITED_API={version:#x}') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |     # On Windows, add PCbuild\amd64\ to include and library directories | 
					
						
							|  |  |  |     include_dirs = [] | 
					
						
							|  |  |  |     library_dirs = [] | 
					
						
							|  |  |  |     if support.MS_WINDOWS: | 
					
						
							|  |  |  |         srcdir = sysconfig.get_config_var('srcdir') | 
					
						
							|  |  |  |         machine = platform.uname().machine | 
					
						
							|  |  |  |         pcbuild = os.path.join(srcdir, 'PCbuild', machine) | 
					
						
							|  |  |  |         if os.path.exists(pcbuild): | 
					
						
							|  |  |  |             # pyconfig.h is generated in PCbuild\amd64\ | 
					
						
							|  |  |  |             include_dirs.append(pcbuild) | 
					
						
							|  |  |  |             # python313.lib is generated in PCbuild\amd64\ | 
					
						
							|  |  |  |             library_dirs.append(pcbuild) | 
					
						
							|  |  |  |             print(f"Add PCbuild directory: {pcbuild}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Display information to help debugging | 
					
						
							|  |  |  |     for env_name in ('CC', 'CFLAGS', 'CPPFLAGS'): | 
					
						
							|  |  |  |         if env_name in os.environ: | 
					
						
							|  |  |  |             print(f"{env_name} env var: {os.environ[env_name]!r}") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             print(f"{env_name} env var: <missing>") | 
					
						
							|  |  |  |     print(f"extra_compile_args: {cppflags!r}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ext = Extension( | 
					
						
							|  |  |  |         module_name, | 
					
						
							| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  |         sources=[SOURCE], | 
					
						
							|  |  |  |         language='c++', | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |         extra_compile_args=cppflags, | 
					
						
							|  |  |  |         include_dirs=include_dirs, | 
					
						
							|  |  |  |         library_dirs=library_dirs) | 
					
						
							|  |  |  |     setup(name=f'internal_{module_name}', | 
					
						
							|  |  |  |           version='0.0', | 
					
						
							|  |  |  |           ext_modules=[ext]) | 
					
						
							| 
									
										
										
										
											2022-05-13 00:20:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     main() |