Skip to content

Commit 81dca70

Browse files
authored
gh-93939: Build C extensions without setup.py (GH-94474)
Combines GH-93940, GH-94452, and GH-94433
1 parent b03a9e8 commit 81dca70

14 files changed

+89
-1506
lines changed

Doc/whatsnew/3.12.rst

+6
Original file line numberDiff line numberDiff line change
@@ -356,6 +356,12 @@ Changes in the Python API
356356
Build Changes
357357
=============
358358

359+
* Python no longer uses ``setup.py`` to build shared C extension modules.
360+
Build parameters like headers and libraries are detected in ``configure``
361+
script. Extensions are built by ``Makefile``. Most extensions use
362+
``pkg-config`` and fall back to manual detection.
363+
(Contributed by Christian Heimes in :gh:`93939`.)
364+
359365
* ``va_start()`` with two parameters, like ``va_start(args, format),``
360366
is now required to build Python.
361367
``va_start()`` is no longer called with a single parameter.

Lib/_bootsubprocess.py

-97
This file was deleted.

Lib/test/test_descr.py

+9
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,11 @@
2121
except ImportError:
2222
_testcapi = None
2323

24+
try:
25+
import xxsubtype
26+
except ImportError:
27+
xxsubtype = None
28+
2429

2530
class OperatorsTest(unittest.TestCase):
2631

@@ -299,6 +304,7 @@ def test_explicit_reverse_methods(self):
299304
self.assertEqual(float.__rsub__(3.0, 1), -2.0)
300305

301306
@support.impl_detail("the module 'xxsubtype' is internal")
307+
@unittest.skipIf(xxsubtype is None, "requires xxsubtype module")
302308
def test_spam_lists(self):
303309
# Testing spamlist operations...
304310
import copy, xxsubtype as spam
@@ -343,6 +349,7 @@ def foo(self): return 1
343349
self.assertEqual(a.getstate(), 42)
344350

345351
@support.impl_detail("the module 'xxsubtype' is internal")
352+
@unittest.skipIf(xxsubtype is None, "requires xxsubtype module")
346353
def test_spam_dicts(self):
347354
# Testing spamdict operations...
348355
import copy, xxsubtype as spam
@@ -1600,6 +1607,7 @@ def test_refleaks_in_classmethod___init__(self):
16001607
self.assertAlmostEqual(gettotalrefcount() - refs_before, 0, delta=10)
16011608

16021609
@support.impl_detail("the module 'xxsubtype' is internal")
1610+
@unittest.skipIf(xxsubtype is None, "requires xxsubtype module")
16031611
def test_classmethods_in_c(self):
16041612
# Testing C-based class methods...
16051613
import xxsubtype as spam
@@ -1683,6 +1691,7 @@ def test_refleaks_in_staticmethod___init__(self):
16831691
self.assertAlmostEqual(gettotalrefcount() - refs_before, 0, delta=10)
16841692

16851693
@support.impl_detail("the module 'xxsubtype' is internal")
1694+
@unittest.skipIf(xxsubtype is None, "requires xxsubtype module")
16861695
def test_staticmethods_in_c(self):
16871696
# Testing C-based static methods...
16881697
import xxsubtype as spam

Makefile.pre.in

+14-53
Original file line numberDiff line numberDiff line change
@@ -211,12 +211,6 @@ ENSUREPIP= @ENSUREPIP@
211211
LIBMPDEC_A= Modules/_decimal/libmpdec/libmpdec.a
212212
LIBEXPAT_A= Modules/expat/libexpat.a
213213

214-
# OpenSSL options for setup.py so sysconfig can pick up AC_SUBST() vars.
215-
OPENSSL_INCLUDES=@OPENSSL_INCLUDES@
216-
OPENSSL_LIBS=@OPENSSL_LIBS@
217-
OPENSSL_LDFLAGS=@OPENSSL_LDFLAGS@
218-
OPENSSL_RPATH=@OPENSSL_RPATH@
219-
220214
# Module state, compiler flags and linker flags
221215
# Empty CFLAGS and LDFLAGS are omitted.
222216
# states:
@@ -582,9 +576,10 @@ LIBEXPAT_HEADERS= \
582576

583577
# Default target
584578
all: @DEF_MAKE_ALL_RULE@
585-
build_all: check-clean-src $(BUILDPYTHON) platform oldsharedmods sharedmods \
586-
gdbhooks Programs/_testembed scripts
587-
build_wasm: check-clean-src $(BUILDPYTHON) platform oldsharedmods python-config
579+
build_all: check-clean-src $(BUILDPYTHON) platform sharedmods \
580+
gdbhooks Programs/_testembed scripts checksharedmods
581+
build_wasm: check-clean-src $(BUILDPYTHON) platform sharedmods \
582+
python-config checksharedmods
588583

589584
# Check that the source is clean when building out of source.
590585
check-clean-src:
@@ -726,22 +721,6 @@ $(srcdir)/Modules/_blake2/blake2s_impl.c: $(srcdir)/Modules/_blake2/blake2b_impl
726721
$(PYTHON_FOR_REGEN) $(srcdir)/Modules/_blake2/blake2b2s.py
727722
$(PYTHON_FOR_REGEN) $(srcdir)/Tools/clinic/clinic.py -f $@
728723

729-
# Build the shared modules
730-
# Under GNU make, MAKEFLAGS are sorted and normalized; the 's' for
731-
# -s, --silent or --quiet is always the first char.
732-
# Under BSD make, MAKEFLAGS might be " -s -v x=y".
733-
# Ignore macros passed by GNU make, passed after --
734-
sharedmods: $(PYTHON_FOR_BUILD_DEPS) pybuilddir.txt @LIBMPDEC_INTERNAL@ @LIBEXPAT_INTERNAL@
735-
@case "`echo X $$MAKEFLAGS | sed 's/^X //;s/ -- .*//'`" in \
736-
*\ -s*|s*) quiet="-q";; \
737-
*) quiet="";; \
738-
esac; \
739-
echo "$(RUNSHARED) CC='$(CC)' LDSHARED='$(BLDSHARED)' OPT='$(OPT)' \
740-
$(PYTHON_FOR_BUILD) $(srcdir)/setup.py $$quiet build"; \
741-
$(RUNSHARED) CC='$(CC)' LDSHARED='$(BLDSHARED)' OPT='$(OPT)' \
742-
$(PYTHON_FOR_BUILD) $(srcdir)/setup.py $$quiet build
743-
744-
745724
# Build static library
746725
$(LIBRARY): $(LIBRARY_OBJS)
747726
-rm -f $@
@@ -832,10 +811,6 @@ python.worker.js: $(srcdir)/Tools/wasm/python.worker.js
832811
# Build static libmpdec.a
833812
LIBMPDEC_CFLAGS=@LIBMPDEC_CFLAGS@ $(PY_STDMODULE_CFLAGS) $(CCSHARED)
834813

835-
# for setup.py
836-
DECIMAL_CFLAGS=@LIBMPDEC_CFLAGS@
837-
DECIMAL_LDFLAGS=@LIBMPDEC_LDFLAGS@
838-
839814
# "%.o: %c" is not portable
840815
Modules/_decimal/libmpdec/basearith.o: $(srcdir)/Modules/_decimal/libmpdec/basearith.c $(LIBMPDEC_HEADERS) $(PYTHON_HEADERS)
841816
$(CC) -c $(LIBMPDEC_CFLAGS) -o $@ $(srcdir)/Modules/_decimal/libmpdec/basearith.c
@@ -890,10 +865,6 @@ $(LIBMPDEC_A): $(LIBMPDEC_OBJS)
890865
# Build static libexpat.a
891866
LIBEXPAT_CFLAGS=@LIBEXPAT_CFLAGS@ $(PY_STDMODULE_CFLAGS) $(CCSHARED)
892867

893-
# for setup.py
894-
EXPAT_CFLAGS=@LIBEXPAT_CFLAGS@
895-
EXPAT_LDFLAGS=@LIBEXPAT_LDFLAGS@
896-
897868
Modules/expat/xmlparse.o: $(srcdir)/Modules/expat/xmlparse.c $(LIBEXPAT_HEADERS) $(PYTHON_HEADERS)
898869
$(CC) -c $(LIBEXPAT_CFLAGS) -o $@ $(srcdir)/Modules/expat/xmlparse.c
899870

@@ -910,7 +881,7 @@ $(LIBEXPAT_A): $(LIBEXPAT_OBJS)
910881
# create relative links from build/lib.platform/egg.so to Modules/egg.so
911882
# pybuilddir.txt is created too late. We cannot use it in Makefile
912883
# targets. ln --relative is not portable.
913-
oldsharedmods: $(SHAREDMODS) pybuilddir.txt
884+
sharedmods: $(SHAREDMODS) pybuilddir.txt
914885
@target=`cat pybuilddir.txt`; \
915886
$(MKDIR_P) $$target; \
916887
for mod in X $(SHAREDMODS); do \
@@ -919,7 +890,8 @@ oldsharedmods: $(SHAREDMODS) pybuilddir.txt
919890
fi; \
920891
done
921892

922-
checksharedmods: oldsharedmods sharedmods $(PYTHON_FOR_BUILD_DEPS)
893+
# dependency on BUILDPYTHON ensures that the target is run last
894+
checksharedmods: sharedmods $(PYTHON_FOR_BUILD_DEPS) $(BUILDPYTHON)
923895
@$(RUNSHARED) $(PYTHON_FOR_BUILD) $(srcdir)/Tools/scripts/check_extension_modules.py
924896

925897
Modules/Setup.local:
@@ -942,7 +914,7 @@ Makefile Modules/config.c: Makefile.pre \
942914
$(SHELL) $(MAKESETUP) -c $(srcdir)/Modules/config.c.in \
943915
-s Modules \
944916
Modules/Setup.local \
945-
@MODULES_SETUP_STDLIB@ \
917+
Modules/Setup.stdlib \
946918
Modules/Setup.bootstrap \
947919
$(srcdir)/Modules/Setup
948920
@mv config.c Modules
@@ -1762,13 +1734,13 @@ altinstall: commoninstall
17621734

17631735
commoninstall: check-clean-src @FRAMEWORKALTINSTALLFIRST@ \
17641736
altbininstall libinstall inclinstall libainstall \
1765-
sharedinstall oldsharedinstall altmaninstall \
1737+
sharedinstall altmaninstall \
17661738
@FRAMEWORKALTINSTALLLAST@
17671739

17681740
# Install shared libraries enabled by Setup
17691741
DESTDIRS= $(exec_prefix) $(LIBDIR) $(BINLIBDEST) $(DESTSHARED)
17701742

1771-
oldsharedinstall: $(DESTSHARED) all
1743+
sharedinstall: $(DESTSHARED) all
17721744
@for i in X $(SHAREDMODS); do \
17731745
if test $$i != X; then \
17741746
echo $(INSTALL_SHARED) $$i $(DESTSHARED)/`basename $$i`; \
@@ -2252,17 +2224,6 @@ libainstall: all scripts
22522224
else true; \
22532225
fi
22542226

2255-
# Install the dynamically loadable modules
2256-
# This goes into $(exec_prefix)
2257-
sharedinstall: all
2258-
$(RUNSHARED) $(PYTHON_FOR_BUILD) $(srcdir)/setup.py install \
2259-
--prefix=$(prefix) \
2260-
--install-scripts=$(BINDIR) \
2261-
--install-platlib=$(DESTSHARED) \
2262-
--root=$(DESTDIR)/
2263-
-rm $(DESTDIR)$(DESTSHARED)/_sysconfigdata_$(ABIFLAGS)_$(MACHDEP)_$(MULTIARCH).py
2264-
-rm -r $(DESTDIR)$(DESTSHARED)/__pycache__
2265-
22662227
# Here are a couple of targets for MacOSX again, to install a full
22672228
# framework-based Python. frameworkinstall installs everything, the
22682229
# subtargets install specific parts. Much of the actual work is offloaded to
@@ -2536,10 +2497,10 @@ update-config:
25362497
Python/thread.o: @THREADHEADERS@ $(srcdir)/Python/condvar.h
25372498

25382499
# Declare targets that aren't real files
2539-
.PHONY: all build_all build_wasm sharedmods check-clean-src
2540-
.PHONY: oldsharedmods checksharedmods test quicktest
2541-
.PHONY: install altinstall oldsharedinstall bininstall altbininstall
2542-
.PHONY: maninstall libinstall inclinstall libainstall sharedinstall
2500+
.PHONY: all build_all build_wasm check-clean-src
2501+
.PHONY: sharedmods checksharedmods test quicktest
2502+
.PHONY: install altinstall sharedinstall bininstall altbininstall
2503+
.PHONY: maninstall libinstall inclinstall libainstall
25432504
.PHONY: frameworkinstall frameworkinstallframework frameworkinstallstructure
25442505
.PHONY: frameworkinstallmaclib frameworkinstallapps frameworkinstallunixtools
25452506
.PHONY: frameworkaltinstallunixtools recheck clean clobber distclean
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
C extension modules are now built by ``configure`` and ``make``
2+
instead of :mod:`distutils` and ``setup.py``.

Modules/Setup

+1-1
Original file line numberDiff line numberDiff line change
@@ -275,7 +275,7 @@ PYTHONPATH=$(COREPYTHONPATH)
275275
#xx xxmodule.c
276276
#xxlimited xxlimited.c
277277
#xxlimited_35 xxlimited_35.c
278-
xxsubtype xxsubtype.c # Required for the test suite to pass!
278+
#xxsubtype xxsubtype.c
279279

280280
# Testing
281281

Modules/Setup.stdlib.in

+1
Original file line numberDiff line numberDiff line change
@@ -165,6 +165,7 @@
165165
############################################################################
166166
# Test modules
167167

168+
@MODULE_XXSUBTYPE_TRUE@xxsubtype xxsubtype.c
168169
@MODULE__XXTESTFUZZ_TRUE@_xxtestfuzz _xxtestfuzz/_xxtestfuzz.c _xxtestfuzz/fuzzer.c
169170
@MODULE__TESTBUFFER_TRUE@_testbuffer _testbuffer.c
170171
@MODULE__TESTINTERNALCAPI_TRUE@_testinternalcapi _testinternalcapi.c

Python/stdlib_module_names.h

-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Tools/scripts/check_extension_modules.py

+3-8
Original file line numberDiff line numberDiff line change
@@ -130,10 +130,11 @@ class ModuleChecker:
130130
pybuilddir_txt = "pybuilddir.txt"
131131

132132
setup_files = (
133-
SRC_DIR / "Modules/Setup",
133+
# see end of configure.ac
134134
"Modules/Setup.local",
135-
"Modules/Setup.bootstrap",
136135
"Modules/Setup.stdlib",
136+
"Modules/Setup.bootstrap",
137+
SRC_DIR / "Modules/Setup",
137138
)
138139

139140
def __init__(self, cross_compiling: bool = False, strict: bool = False):
@@ -308,12 +309,6 @@ def get_sysconfig_modules(self) -> Iterable[ModuleInfo]:
308309
MODBUILT_NAMES: modules in *static* block
309310
MODSHARED_NAMES: modules in *shared* block
310311
MODDISABLED_NAMES: modules in *disabled* block
311-
312-
Modules built by setup.py addext() have a MODULE_{modname}_STATE entry,
313-
but are not listed in MODSHARED_NAMES.
314-
315-
Modules built by old-style setup.py add() have neither a MODULE_{modname}
316-
entry nor an entry in MODSHARED_NAMES.
317312
"""
318313
moddisabled = set(sysconfig.get_config_var("MODDISABLED_NAMES").split())
319314
if self.cross_compiling:

Tools/scripts/generate_stdlib_module_names.py

-11
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@
1212

1313
SRC_DIR = os.path.dirname(os.path.dirname(os.path.dirname(__file__)))
1414
STDLIB_PATH = os.path.join(SRC_DIR, 'Lib')
15-
SETUP_PY = os.path.join(SRC_DIR, 'setup.py')
1615

1716
IGNORE = {
1817
'__init__',
@@ -64,15 +63,6 @@ def list_packages(names):
6463
names.add(name)
6564

6665

67-
# Extension modules built by setup.py
68-
def list_setup_extensions(names):
69-
cmd = [sys.executable, SETUP_PY, "-q", "build", "--list-module-names"]
70-
output = subprocess.check_output(cmd)
71-
output = output.decode("utf8")
72-
extensions = output.splitlines()
73-
names |= set(extensions)
74-
75-
7666
# Built-in and extension modules built by Modules/Setup*
7767
# includes Windows and macOS extensions.
7868
def list_modules_setup_extensions(names):
@@ -103,7 +93,6 @@ def list_frozen(names):
10393
def list_modules():
10494
names = set(sys.builtin_module_names)
10595
list_modules_setup_extensions(names)
106-
list_setup_extensions(names)
10796
list_packages(names)
10897
list_python_modules(names)
10998
list_frozen(names)

0 commit comments

Comments
 (0)