diff --git a/README.md b/README.md
index 41cb62037..702b9baad 100644
--- a/README.md
+++ b/README.md
@@ -31,6 +31,7 @@ What does it do?
| CPython 3.10 | ✅ | ✅ | ✅ | ✅ | ✅² | ✅ | ✅ | ✅ | ✅ | ✅ |
| PyPy 3.7 v7.3 | ✅ | N/A | ✅ | N/A | N/A | ✅¹ | ✅¹ | ✅¹ | N/A | N/A |
| PyPy 3.8 v7.3 | ✅ | N/A | ✅ | N/A | N/A | ✅¹ | ✅¹ | ✅¹ | N/A | N/A |
+| PyPy 3.9 v7.3 | ✅ | N/A | ✅ | N/A | N/A | ✅¹ | ✅¹ | ✅¹ | N/A | N/A |
¹ PyPy is only supported for manylinux wheels.
² Windows arm64 support is experimental.
diff --git a/cibuildwheel/resources/build-platforms.toml b/cibuildwheel/resources/build-platforms.toml
index 136254615..6beee5caa 100644
--- a/cibuildwheel/resources/build-platforms.toml
+++ b/cibuildwheel/resources/build-platforms.toml
@@ -12,6 +12,7 @@ python_configurations = [
{ identifier = "cp310-manylinux_i686", version = "3.10", path_str = "/opt/python/cp310-cp310" },
{ identifier = "pp37-manylinux_x86_64", version = "3.7", path_str = "/opt/python/pp37-pypy37_pp73" },
{ identifier = "pp38-manylinux_x86_64", version = "3.8", path_str = "/opt/python/pp38-pypy38_pp73" },
+ { identifier = "pp39-manylinux_x86_64", version = "3.9", path_str = "/opt/python/pp39-pypy39_pp73" },
{ identifier = "cp36-manylinux_aarch64", version = "3.6", path_str = "/opt/python/cp36-cp36m" },
{ identifier = "cp37-manylinux_aarch64", version = "3.7", path_str = "/opt/python/cp37-cp37m" },
{ identifier = "cp38-manylinux_aarch64", version = "3.8", path_str = "/opt/python/cp38-cp38" },
@@ -29,8 +30,10 @@ python_configurations = [
{ identifier = "cp310-manylinux_s390x", version = "3.10", path_str = "/opt/python/cp310-cp310" },
{ identifier = "pp37-manylinux_aarch64", version = "3.7", path_str = "/opt/python/pp37-pypy37_pp73" },
{ identifier = "pp38-manylinux_aarch64", version = "3.8", path_str = "/opt/python/pp38-pypy38_pp73" },
+ { identifier = "pp39-manylinux_aarch64", version = "3.9", path_str = "/opt/python/pp39-pypy39_pp73" },
{ identifier = "pp37-manylinux_i686", version = "3.7", path_str = "/opt/python/pp37-pypy37_pp73" },
{ identifier = "pp38-manylinux_i686", version = "3.8", path_str = "/opt/python/pp38-pypy38_pp73" },
+ { identifier = "pp39-manylinux_i686", version = "3.9", path_str = "/opt/python/pp39-pypy39_pp73" },
{ identifier = "cp36-musllinux_x86_64", version = "3.6", path_str = "/opt/python/cp36-cp36m" },
{ identifier = "cp37-musllinux_x86_64", version = "3.7", path_str = "/opt/python/cp37-cp37m" },
{ identifier = "cp38-musllinux_x86_64", version = "3.8", path_str = "/opt/python/cp38-cp38" },
@@ -73,6 +76,7 @@ python_configurations = [
{ identifier = "cp310-macosx_universal2", version = "3.10", url = "https://www.python.org/ftp/python/3.10.2/python-3.10.2-macos11.pkg" },
{ identifier = "pp37-macosx_x86_64", version = "3.7", url = "https://downloads.python.org/pypy/pypy3.7-v7.3.8-osx64.tar.bz2" },
{ identifier = "pp38-macosx_x86_64", version = "3.8", url = "https://downloads.python.org/pypy/pypy3.8-v7.3.8-osx64.tar.bz2" },
+ { identifier = "pp39-macosx_x86_64", version = "3.9", url = "https://downloads.python.org/pypy/pypy3.9-v7.3.8-osx64.tar.bz2" },
]
[windows]
@@ -91,4 +95,5 @@ python_configurations = [
{ identifier = "cp310-win_arm64", version = "3.10.2", arch = "ARM64" },
{ identifier = "pp37-win_amd64", version = "3.7", arch = "64", url = "https://downloads.python.org/pypy/pypy3.7-v7.3.8-win64.zip" },
{ identifier = "pp38-win_amd64", version = "3.8", arch = "64", url = "https://downloads.python.org/pypy/pypy3.8-v7.3.8-win64.zip" },
+ { identifier = "pp39-win_amd64", version = "3.9", arch = "64", url = "https://downloads.python.org/pypy/pypy3.9-v7.3.8-win64.zip" },
]
diff --git a/cibuildwheel/resources/pinned_docker_images.cfg b/cibuildwheel/resources/pinned_docker_images.cfg
index 3b9468302..623955886 100644
--- a/cibuildwheel/resources/pinned_docker_images.cfg
+++ b/cibuildwheel/resources/pinned_docker_images.cfg
@@ -1,43 +1,43 @@
[x86_64]
manylinux1 = quay.io/pypa/manylinux1_x86_64:2022-02-20-044a1ea
-manylinux2010 = quay.io/pypa/manylinux2010_x86_64:2022-02-20-e7cad68
-manylinux2014 = quay.io/pypa/manylinux2014_x86_64:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_x86_64:2022-02-20-e7cad68
-musllinux_1_1 = quay.io/pypa/musllinux_1_1_x86_64:2022-02-20-e7cad68
+manylinux2010 = quay.io/pypa/manylinux2010_x86_64:2022-02-23-5cd6f37
+manylinux2014 = quay.io/pypa/manylinux2014_x86_64:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_x86_64:2022-02-23-5cd6f37
+musllinux_1_1 = quay.io/pypa/musllinux_1_1_x86_64:2022-02-23-5cd6f37
[i686]
manylinux1 = quay.io/pypa/manylinux1_i686:2022-02-20-044a1ea
-manylinux2010 = quay.io/pypa/manylinux2010_i686:2022-02-20-e7cad68
-manylinux2014 = quay.io/pypa/manylinux2014_i686:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_i686:2022-02-20-e7cad68
-musllinux_1_1 = quay.io/pypa/musllinux_1_1_i686:2022-02-20-e7cad68
+manylinux2010 = quay.io/pypa/manylinux2010_i686:2022-02-23-5cd6f37
+manylinux2014 = quay.io/pypa/manylinux2014_i686:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_i686:2022-02-23-5cd6f37
+musllinux_1_1 = quay.io/pypa/musllinux_1_1_i686:2022-02-23-5cd6f37
[pypy_x86_64]
-manylinux2010 = quay.io/pypa/manylinux2010_x86_64:2022-02-20-e7cad68
-manylinux2014 = quay.io/pypa/manylinux2014_x86_64:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_x86_64:2022-02-20-e7cad68
+manylinux2010 = quay.io/pypa/manylinux2010_x86_64:2022-02-23-5cd6f37
+manylinux2014 = quay.io/pypa/manylinux2014_x86_64:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_x86_64:2022-02-23-5cd6f37
[pypy_i686]
-manylinux2010 = quay.io/pypa/manylinux2010_i686:2022-02-20-e7cad68
-manylinux2014 = quay.io/pypa/manylinux2014_i686:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_i686:2022-02-20-e7cad68
+manylinux2010 = quay.io/pypa/manylinux2010_i686:2022-02-23-5cd6f37
+manylinux2014 = quay.io/pypa/manylinux2014_i686:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_i686:2022-02-23-5cd6f37
[aarch64]
-manylinux2014 = quay.io/pypa/manylinux2014_aarch64:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_aarch64:2022-02-20-e7cad68
-musllinux_1_1 = quay.io/pypa/musllinux_1_1_aarch64:2022-02-20-e7cad68
+manylinux2014 = quay.io/pypa/manylinux2014_aarch64:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_aarch64:2022-02-23-5cd6f37
+musllinux_1_1 = quay.io/pypa/musllinux_1_1_aarch64:2022-02-23-5cd6f37
[ppc64le]
-manylinux2014 = quay.io/pypa/manylinux2014_ppc64le:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_ppc64le:2022-02-20-e7cad68
-musllinux_1_1 = quay.io/pypa/musllinux_1_1_ppc64le:2022-02-20-e7cad68
+manylinux2014 = quay.io/pypa/manylinux2014_ppc64le:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_ppc64le:2022-02-23-5cd6f37
+musllinux_1_1 = quay.io/pypa/musllinux_1_1_ppc64le:2022-02-23-5cd6f37
[s390x]
-manylinux2014 = quay.io/pypa/manylinux2014_s390x:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_s390x:2022-02-20-e7cad68
-musllinux_1_1 = quay.io/pypa/musllinux_1_1_s390x:2022-02-20-e7cad68
+manylinux2014 = quay.io/pypa/manylinux2014_s390x:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_s390x:2022-02-23-5cd6f37
+musllinux_1_1 = quay.io/pypa/musllinux_1_1_s390x:2022-02-23-5cd6f37
[pypy_aarch64]
-manylinux2014 = quay.io/pypa/manylinux2014_aarch64:2022-02-20-e7cad68
-manylinux_2_24 = quay.io/pypa/manylinux_2_24_aarch64:2022-02-20-e7cad68
+manylinux2014 = quay.io/pypa/manylinux2014_aarch64:2022-02-23-5cd6f37
+manylinux_2_24 = quay.io/pypa/manylinux_2_24_aarch64:2022-02-23-5cd6f37
diff --git a/docs/options.md b/docs/options.md
index 46f2d24e1..45f67873a 100644
--- a/docs/options.md
+++ b/docs/options.md
@@ -211,6 +211,7 @@ When setting the options, you can use shell-style globbing syntax, as per [fnmat
| Python 3.10 | cp310-macosx_x86_64
cp310-macosx_universal2
cp310-macosx_arm64 | cp310-win_amd64
cp310-win32
cp310-win_arm64 | cp310-manylinux_x86_64
cp310-manylinux_i686
cp310-musllinux_x86_64
cp310-musllinux_i686 | cp310-manylinux_aarch64
cp310-manylinux_ppc64le
cp310-manylinux_s390x
cp310-musllinux_aarch64
cp310-musllinux_ppc64le
cp310-musllinux_s390x |
| PyPy3.7 v7.3 | pp37-macosx_x86_64 | pp37-win_amd64 | pp37-manylinux_x86_64
pp37-manylinux_i686 | pp37-manylinux_aarch64 |
| PyPy3.8 v7.3 | pp38-macosx_x86_64 | pp38-win_amd64 | pp38-manylinux_x86_64
pp38-manylinux_i686 | pp38-manylinux_aarch64 |
+| PyPy3.9 v7.3 | pp39-macosx_x86_64 | pp39-win_amd64 | pp39-manylinux_x86_64
pp39-manylinux_i686 | pp39-manylinux_aarch64 |
The list of supported and currently selected build identifiers can also be retrieved by passing the `--print-build-identifiers` flag to cibuildwheel.
The format is `python_tag-platform_tag`, with tags similar to those in [PEP 425](https://www.python.org/dev/peps/pep-0425/#details).
diff --git a/test/utils.py b/test/utils.py
index 987159543..0248a7c15 100644
--- a/test/utils.py
+++ b/test/utils.py
@@ -135,7 +135,7 @@ def expected_wheels(
python_abi_tags = ["cp36-cp36m", "cp37-cp37m", "cp38-cp38", "cp39-cp39", "cp310-cp310"]
if machine_arch in ["x86_64", "AMD64", "x86", "aarch64"]:
- python_abi_tags += ["pp37-pypy37_pp73", "pp38-pypy38_pp73"]
+ python_abi_tags += ["pp37-pypy37_pp73", "pp38-pypy38_pp73", "pp39-pypy39_pp73"]
if platform == "macos" and machine_arch == "arm64":
# currently, arm64 macs are only supported by cp39 & cp310
diff --git a/unit_test/option_prepare_test.py b/unit_test/option_prepare_test.py
index d21ba65d5..354cda4a6 100644
--- a/unit_test/option_prepare_test.py
+++ b/unit_test/option_prepare_test.py
@@ -11,7 +11,7 @@
from cibuildwheel import linux, util
from cibuildwheel.__main__ import main
-ALL_IDS = {"cp36", "cp37", "cp38", "cp39", "cp310", "pp37", "pp38"}
+ALL_IDS = {"cp36", "cp37", "cp38", "cp39", "cp310", "pp37", "pp38", "pp39"}
@pytest.fixture
@@ -133,7 +133,7 @@ def test_build_with_override_launches(mock_build_docker, monkeypatch, tmp_path):
identifiers = {x.identifier for x in kwargs["platform_configs"]}
assert identifiers == {
- f"{x}-manylinux_x86_64" for x in ALL_IDS - {"cp36", "cp310", "pp37", "pp38"}
+ f"{x}-manylinux_x86_64" for x in ALL_IDS - {"cp36", "cp310", "pp37", "pp38", "pp39"}
}
assert kwargs["options"].build_options("cp37-manylinux_x86_64").before_all == ""
@@ -146,6 +146,7 @@ def test_build_with_override_launches(mock_build_docker, monkeypatch, tmp_path):
"cp310-manylinux_x86_64",
"pp37-manylinux_x86_64",
"pp38-manylinux_x86_64",
+ "pp39-manylinux_x86_64",
}
kwargs = build_on_docker.call_args_list[3][1]