| 
									
										
										
										
											2024-03-18 20:15:20 +01: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. | 
					
						
							|  |  |  | import os | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  | import platform | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  | import shlex | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import sysconfig | 
					
						
							|  |  |  | from test import support | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from setuptools import setup, Extension | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SOURCE = 'extension.c' | 
					
						
							| 
									
										
										
										
											2024-06-28 14:41:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 00:06:24 +01:00
										 |  |  | if not support.MS_WINDOWS: | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  |     # C compiler flags for GCC and clang | 
					
						
							|  |  |  |     CFLAGS = [ | 
					
						
							|  |  |  |         # The purpose of test_cext extension is to check that building a C | 
					
						
							|  |  |  |         # extension using the Python C API does not emit C compiler warnings. | 
					
						
							|  |  |  |         '-Werror', | 
					
						
							| 
									
										
										
										
											2024-06-27 12:22:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # gh-120593: Check the 'const' qualifier | 
					
						
							|  |  |  |         '-Wcast-qual', | 
					
						
							| 
									
										
										
										
											2025-03-04 14:10:09 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Ask for strict(er) compliance with the standard | 
					
						
							|  |  |  |         '-pedantic-errors', | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  |     ] | 
					
						
							| 
									
										
										
										
											2024-03-21 00:06:24 +01:00
										 |  |  |     if not support.Py_GIL_DISABLED: | 
					
						
							|  |  |  |         CFLAGS.append( | 
					
						
							|  |  |  |             # gh-116869: The Python C API must be compatible with building | 
					
						
							|  |  |  |             # with the -Werror=declaration-after-statement compiler flag. | 
					
						
							|  |  |  |             '-Werror=declaration-after-statement', | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  | else: | 
					
						
							| 
									
										
										
										
											2024-06-28 14:41:37 +02:00
										 |  |  |     # MSVC compiler flags | 
					
						
							|  |  |  |     CFLAGS = [ | 
					
						
							| 
									
										
										
										
											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', | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def main(): | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  |     std = os.environ.get("CPYTHON_TEST_STD", "") | 
					
						
							|  |  |  |     module_name = os.environ["CPYTHON_TEST_EXT_NAME"] | 
					
						
							|  |  |  |     limited = bool(os.environ.get("CPYTHON_TEST_LIMITED", "")) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cflags = list(CFLAGS) | 
					
						
							|  |  |  |     cflags.append(f'-DMODULE_NAME={module_name}') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |     # Add -std=STD or /std:STD (MSVC) compiler flag | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  |     if std: | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |         if support.MS_WINDOWS: | 
					
						
							|  |  |  |             cflags.append(f'/std:{std}') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cflags.append(f'-std={std}') | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-19 22:58:13 +01:00
										 |  |  |     # Remove existing -std or /std options from CC command line. | 
					
						
							|  |  |  |     # Python adds -std=c11 option. | 
					
						
							|  |  |  |     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 13:19:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |     # Define Py_LIMITED_API macro | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  |     if limited: | 
					
						
							|  |  |  |         version = sys.hexversion | 
					
						
							|  |  |  |         cflags.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 | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  |     for env_name in ('CC', 'CFLAGS'): | 
					
						
							|  |  |  |         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: {cflags!r}") | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ext = Extension( | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  |         module_name, | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  |         sources=[SOURCE], | 
					
						
							| 
									
										
										
										
											2024-03-19 15:03:27 +01:00
										 |  |  |         extra_compile_args=cflags, | 
					
						
							|  |  |  |         include_dirs=include_dirs, | 
					
						
							|  |  |  |         library_dirs=library_dirs) | 
					
						
							| 
									
										
										
										
											2024-03-19 13:19:45 +01:00
										 |  |  |     setup(name=f'internal_{module_name}', | 
					
						
							|  |  |  |           version='0.0', | 
					
						
							|  |  |  |           ext_modules=[ext]) | 
					
						
							| 
									
										
										
										
											2024-03-18 20:15:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     main() |