6d4e4bb612
[skip changelog]
890 lines
30 KiB
RPMSpec
890 lines
30 KiB
RPMSpec
# Note: When this is updated to 7.4,
|
|
# the installation layout will change in a backwards-incompatible way.
|
|
# That'll be a good time to rename this to pypy2.7 and adapt %%pypyprefix to be
|
|
# %%{_libdir}/pypy%%{pyversion} (see e.g. pypy3.7 or pypy3.8 for inspiration).
|
|
%global basever 7.3
|
|
Name: pypy
|
|
Version: %{basever}.13
|
|
%global pyversion 2.7
|
|
Release: %autorelease
|
|
Summary: Python implementation with a Just-In-Time compiler
|
|
|
|
# PyPy is MIT
|
|
# Python standard library is Python
|
|
# pypy/module/unicodedata is UCD
|
|
# Bundled pycparser is is BSD
|
|
# Bundled pycparser.ply is BSD
|
|
# Bundled bits from cryptography are ASL 2.0 or BSD
|
|
# LGPL and another free license we'd need to ask spot about are present in some
|
|
# java jars that we're not building with atm (in fact, we're deleting them
|
|
# before building). If we restore those we'll have to work out the new
|
|
# licensing terms
|
|
License: MIT and Python and UCD and BSD and (ASL 2.0 or BSD)
|
|
URL: https://www.pypy.org/
|
|
|
|
# https://fedoraproject.org/wiki/Changes/EncourageI686LeafRemoval
|
|
ExcludeArch: %{ix86}
|
|
|
|
# High-level configuration of the build:
|
|
|
|
# Whether to use RPM build wheels from the python-{pip,setuptools}-wheel package
|
|
# Uses upstream bundled prebuilt wheels otherwise
|
|
# setuptools >= 45.0 no longer support Python 2.7, hence disabled
|
|
%bcond_with rpmwheels
|
|
|
|
# PyPy consists of an implementation of an interpreter (with JIT compilation)
|
|
# for the full Python language written in a high-level language, leaving many
|
|
# of the implementation details as "pluggable" policies.
|
|
#
|
|
# The implementation language is then compiled down to .c code, from which we
|
|
# obtain a binary.
|
|
#
|
|
# This allows us to build a near-arbitrary collection of different
|
|
# implementations of Python with differing tradeoffs
|
|
#
|
|
# (As it happens, the implementation language is itself Python, albeit a
|
|
# restricted subset "RPython", chosen to making it amenable to being compiled.
|
|
# The result implements the full Python language though)
|
|
|
|
# We could build many different implementations of Python.
|
|
# For now, let's focus on the implementation that appears to be receiving the
|
|
# most attention upstream: the JIT-enabled build, with all standard
|
|
# optimizations
|
|
|
|
# Building a configuration can take significant time:
|
|
|
|
# A build of pypy (with jit) on i686 took 77 mins:
|
|
# [Timer] Timings:
|
|
# [Timer] annotate --- 583.3 s
|
|
# [Timer] rtype_lltype --- 760.9 s
|
|
# [Timer] pyjitpl_lltype --- 567.3 s
|
|
# [Timer] backendopt_lltype --- 375.6 s
|
|
# [Timer] stackcheckinsertion_lltype --- 54.1 s
|
|
# [Timer] database_c --- 852.2 s
|
|
# [Timer] source_c --- 1007.3 s
|
|
# [Timer] compile_c --- 419.9 s
|
|
# [Timer] ===========================================
|
|
# [Timer] Total: --- 4620.5 s
|
|
#
|
|
# A build of pypy (nojit) on x86_64 took about an hour:
|
|
# [Timer] Timings:
|
|
# [Timer] annotate --- 537.5 s
|
|
# [Timer] rtype_lltype --- 667.3 s
|
|
# [Timer] backendopt_lltype --- 385.4 s
|
|
# [Timer] stackcheckinsertion_lltype --- 42.5 s
|
|
# [Timer] database_c --- 625.3 s
|
|
# [Timer] source_c --- 1040.2 s
|
|
# [Timer] compile_c --- 273.9 s
|
|
# [Timer] ===========================================
|
|
# [Timer] Total: --- 3572.0 s
|
|
#
|
|
#
|
|
# A build of pypy-stackless on i686 took about 87 mins:
|
|
# [Timer] Timings:
|
|
# [Timer] annotate --- 584.2 s
|
|
# [Timer] rtype_lltype --- 777.3 s
|
|
# [Timer] backendopt_lltype --- 365.9 s
|
|
# [Timer] stackcheckinsertion_lltype --- 39.3 s
|
|
# [Timer] database_c --- 1089.6 s
|
|
# [Timer] source_c --- 1868.6 s
|
|
# [Timer] compile_c --- 490.4 s
|
|
# [Timer] ===========================================
|
|
# [Timer] Total: --- 5215.3 s
|
|
|
|
|
|
# We will build a "pypy" binary.
|
|
#
|
|
# Unfortunately, the JIT support is only available on some architectures.
|
|
#
|
|
# rpython/jit/backend/detect_cpu.py:getcpuclassname currently supports the
|
|
# following options:
|
|
# 'i386', 'x86'
|
|
# 'x86-without-sse2':
|
|
# 'x86_64'
|
|
# 'armv6', 'armv7' (versions 6 and 7, hard- and soft-float ABI)
|
|
# 'cli'
|
|
# 'llvm'
|
|
#
|
|
# We will only build with JIT support on those architectures, and build without
|
|
# it on the other archs. The resulting binary will typically be slower than
|
|
# CPython for the latter case.
|
|
|
|
%global src_name %{ver_name}-v%{version}-src
|
|
|
|
%ifarch %{ix86} x86_64 %{arm} s390x %{power64} aarch64
|
|
%global with_jit 1
|
|
%else
|
|
%global with_jit 0
|
|
%endif
|
|
|
|
# Should we build a "pypy-stackless" binary?
|
|
%global with_stackless 0
|
|
|
|
# Should we build the emacs JIT-viewing mode?
|
|
%if 0%{?rhel} == 6
|
|
%global with_emacs 0
|
|
%else
|
|
%global with_emacs 1
|
|
%endif
|
|
|
|
# Easy way to enable/disable verbose logging:
|
|
%global verbose_logs 0
|
|
|
|
# Easy way to turn off the selftests:
|
|
%global run_selftests 1
|
|
|
|
%global pypy_include_dir %{pypyprefix}/include
|
|
%global pypyprefix %{_libdir}/%{name}-%{basever}
|
|
%global pylibver 2.7
|
|
%global pymajorlibver 2
|
|
%global ver_name %{name}%{pymajorlibver}
|
|
|
|
# We refer to this subdir of the source tree in a few places during the build:
|
|
%global goal_dir pypy/goal
|
|
|
|
%ifarch %{ix86} x86_64 %{arm}
|
|
%global _package_note_linker gold
|
|
%endif
|
|
|
|
# Source and patches:
|
|
Source0: https://downloads.python.org/pypy/pypy%{pyversion}-v%{version}-src.tar.bz2
|
|
|
|
# Supply various useful RPM macros for building python modules against pypy:
|
|
# __pypy, pypy_sitelib, pypy_sitearch
|
|
Source1: macros.%{name}
|
|
# __pypy2, pypy2_sitelib, pypy2_sitearch
|
|
Source2: macros.%{name}%{pymajorlibver}
|
|
|
|
# Patch pypy.translator.platform so that stdout from "make" etc gets logged,
|
|
# rather than just stderr, so that the command-line invocations of the compiler
|
|
# and linker are captured:
|
|
Patch0: 006-always-log-stdout.patch
|
|
|
|
# Disable the printing of a quote from IRC on startup (these are stored in
|
|
# ROT13 form in lib_pypy/_pypy_irc_topic.py). Some are cute, but some could
|
|
# cause confusion for end-users (and many are in-jokes within the PyPy
|
|
# community that won't make sense outside of it). [Sorry to be a killjoy]
|
|
Patch1: 007-remove-startup-message.patch
|
|
|
|
# Glibc's libcrypt was replaced with libxcrypt in f28, crypt.h header has
|
|
# to be added to privent compilation error.
|
|
# https://fedoraproject.org/wiki/Changes/Replace_glibc_libcrypt_with_libxcrypt
|
|
Patch2: 009-add-libxcrypt-support.patch
|
|
|
|
# Instead of bundled wheels, use our RPM packaged wheels from
|
|
# /usr/share/python-wheels
|
|
# We conditionally apply this, but we use autosetup, so we use Source here
|
|
Source189: 189-use-rpm-wheels.patch
|
|
|
|
# 00382 #
|
|
# Make mailcap refuse to match unsafe filenames/types/params (GH-91993)
|
|
#
|
|
# Upstream: https://github.com/python/cpython/issues/68966
|
|
#
|
|
# Tracker bug: https://bugzilla.redhat.com/show_bug.cgi?id=2075390
|
|
#
|
|
# Backported from python3.
|
|
Patch382: 382-cve-2015-20107.patch
|
|
|
|
# 00394 #
|
|
# gh-98433: Fix quadratic time idna decoding.
|
|
#
|
|
# There was an unnecessary quadratic loop in idna decoding. This restores
|
|
# the behavior to linear.
|
|
#
|
|
# Backported from python3.
|
|
Patch394: 394-cve-2022-45061-cpu-denial-of-service-via-inefficient-idna-decoder.patch
|
|
|
|
# 00399 #
|
|
# CVE-2023-24329
|
|
#
|
|
# gh-102153: Start stripping C0 control and space chars in `urlsplit` (GH-102508)
|
|
#
|
|
# `urllib.parse.urlsplit` has already been respecting the WHATWG spec a bit GH-25595.
|
|
#
|
|
# This adds more sanitizing to respect the "Remove any leading C0 control or space from input" [rule](https://url.spec.whatwg.org/GH-url-parsing:~:text=Remove%%20any%%20leading%%20and%%20trailing%%20C0%%20control%%20or%%20space%%20from%%20input.) in response to [CVE-2023-24329](https://nvd.nist.gov/vuln/detail/CVE-2023-24329).
|
|
#
|
|
# Backported from Python 3.12
|
|
Patch399: 399-cve-2023-24329.patch
|
|
|
|
# Build-time requirements:
|
|
|
|
# pypy's can be rebuilt using itself, rather than with CPython; doing so
|
|
# halves the build time.
|
|
#
|
|
# Turn it off with this boolean, to revert back to rebuilding using CPython
|
|
# and avoid a cycle in the build-time dependency graph:
|
|
|
|
%global use_self_when_building 1
|
|
%if 0%{use_self_when_building}
|
|
BuildRequires: pypy2
|
|
%global bootstrap_python_interp pypy2
|
|
%else
|
|
# exception to use Python 2: https://pagure.io/fesco/issue/2130
|
|
BuildRequires: python27
|
|
%global bootstrap_python_interp python2
|
|
%endif
|
|
|
|
BuildRequires: gcc
|
|
|
|
BuildRequires: libffi-devel
|
|
BuildRequires: tcl-devel
|
|
BuildRequires: tk-devel
|
|
|
|
BuildRequires: sqlite-devel
|
|
|
|
BuildRequires: zlib-devel
|
|
BuildRequires: bzip2-devel
|
|
BuildRequires: ncurses-devel
|
|
BuildRequires: expat-devel
|
|
BuildRequires: openssl-devel
|
|
BuildRequires: gdbm-devel
|
|
BuildRequires: chrpath
|
|
|
|
BuildRequires: python-rpm-macros
|
|
|
|
%ifnarch s390
|
|
BuildRequires: valgrind-devel
|
|
%endif
|
|
|
|
%if %{run_selftests}
|
|
# Used by the selftests, though not by the build:
|
|
BuildRequires: gc-devel
|
|
|
|
# For use in the selftests, for recording stats:
|
|
BuildRequires: time
|
|
|
|
# For use in the selftests, for imposing a per-test timeout:
|
|
BuildRequires: perl-interpreter
|
|
%endif
|
|
|
|
# All arches have execstack
|
|
BuildRequires: execstack
|
|
|
|
# For byte-compiling the JIT-viewing mode:
|
|
%if %{with_emacs}
|
|
BuildRequires: emacs
|
|
%endif
|
|
|
|
# For %%autosetup -S git
|
|
BuildRequires: %{_bindir}/git
|
|
|
|
%if %{with rpmwheels}
|
|
BuildRequires: python-setuptools-wheel < 45
|
|
BuildRequires: python-pip-wheel
|
|
%endif
|
|
|
|
# Metadata for the core package (the JIT build):
|
|
Requires: %{name}-libs%{?_isa} = %{version}-%{release}
|
|
Provides: %{ver_name} = %{version}-%{release}
|
|
Provides: %{ver_name}%{_isa} = %{version}-%{release}
|
|
Provides: pypy%{pyversion} = %{version}-%{release}
|
|
Provides: pypy%{pyversion}%{_isa} = %{version}-%{release}
|
|
Provides: %{ver_name}(abi) = %{basever}
|
|
|
|
%description
|
|
PyPy's implementation of Python, featuring a Just-In-Time compiler on some CPU
|
|
architectures, and various optimized implementations of the standard types
|
|
(strings, dictionaries, etc)
|
|
|
|
%if 0%{with_jit}
|
|
This build of PyPy has JIT-compilation enabled.
|
|
%else
|
|
This build of PyPy has JIT-compilation disabled, as it is not supported on this
|
|
CPU architecture.
|
|
%endif
|
|
|
|
|
|
%package libs
|
|
Summary: Run-time libraries used by PyPy implementations of Python
|
|
|
|
%if %{without rpmwheels}
|
|
# PyPy is MIT and Python and UCD and BSD and (ASL 2.0 or BSD) (see the main package license)
|
|
# setuptools is MIT and bundles:
|
|
# packaging: BSD or ASL 2.0
|
|
# pyparsing: MIT
|
|
# six: MIT
|
|
# pip is MIT and bundles:
|
|
# appdirs: MIT
|
|
# distlib: Python
|
|
# distro: ASL 2.0
|
|
# html5lib: MIT
|
|
# six: MIT
|
|
# colorama: BSD
|
|
# CacheControl: ASL 2.0
|
|
# msgpack-python: ASL 2.0
|
|
# lockfile: MIT
|
|
# progress: ISC
|
|
# ipaddress: Python
|
|
# packaging: ASL 2.0 or BSD
|
|
# pep517: MIT
|
|
# pyparsing: MIT
|
|
# pytoml: MIT
|
|
# retrying: ASL 2.0
|
|
# requests: ASL 2.0
|
|
# chardet: LGPLv2
|
|
# idna: BSD
|
|
# urllib3: MIT
|
|
# certifi: MPLv2.0
|
|
# setuptools: MIT
|
|
# webencodings: BSD
|
|
License: MIT and Python and UCD and BSD and (ASL 2.0 or BSD) and BSD and ASL 2.0 and ISC and LGPLv2 and MPLv2.0 and (ASL 2.0 or BSD)
|
|
%endif
|
|
|
|
# We supply an emacs mode for the JIT viewer.
|
|
# (This doesn't bring in all of emacs, just the directory structure)
|
|
%if %{with_emacs}
|
|
Requires: emacs-filesystem >= %{_emacs_version}
|
|
%endif
|
|
|
|
%if %{with rpmwheels}
|
|
Requires: python-setuptools-wheel < 45
|
|
Requires: python-pip-wheel
|
|
%else
|
|
Provides: bundled(python2dist(setuptools)) = 44.0.0
|
|
Provides: bundled(python2dist(packaging)) = 16.8
|
|
Provides: bundled(python2dist(pyparsing)) = 2.2.1
|
|
Provides: bundled(python2dist(six)) = 1.10.0
|
|
|
|
Provides: bundled(python2dist(pip)) = 20.0.2
|
|
Provides: bundled(python2dist(appdirs)) = 1.4.3
|
|
Provides: bundled(python2dist(CacheControl)) = 0.12.6
|
|
Provides: bundled(python2dist(contextlib2)) = 0.6.0
|
|
Provides: bundled(python2dist(certifi)) = 2019.11.28
|
|
Provides: bundled(python2dist(chardet)) = 3.0.4
|
|
Provides: bundled(python2dist(colorama)) = 0.4.3
|
|
Provides: bundled(python2dist(distlib)) = 0.3.0
|
|
Provides: bundled(python2dist(distro)) = 1.4.0
|
|
Provides: bundled(python2dist(html5lib)) = 1.0.1
|
|
Provides: bundled(python2dist(idna)) = 2.8
|
|
Provides: bundled(python2dist(ipaddress)) = 1.0.23
|
|
Provides: bundled(python2dist(lockfile)) = 0.12.2
|
|
Provides: bundled(python2dist(msgpack)) = 0.6.2
|
|
Provides: bundled(python2dist(packaging)) = 20.1
|
|
Provides: bundled(python2dist(pep517)) = 0.7.0
|
|
Provides: bundled(python2dist(progress)) = 1.5
|
|
Provides: bundled(python2dist(pyparsing)) = 2.4.6
|
|
Provides: bundled(python2dist(pytoml)) = 0.1.21
|
|
Provides: bundled(python2dist(requests)) = 2.22.0
|
|
Provides: bundled(python2dist(retrying)) = 1.3.3
|
|
Provides: bundled(python2dist(setuptools)) = 44.0.0
|
|
Provides: bundled(python2dist(six)) = 1.14.0
|
|
Provides: bundled(python2dist(urllib3)) = 1.25.7
|
|
Provides: bundled(python2dist(webencodings)) = 0.5.1
|
|
%endif
|
|
|
|
# Find the version in lib_pypy/cffi/_pycparser/__init__.py
|
|
Provides: bundled(python2dist(pycparser)) = 2.21
|
|
|
|
# Find the version in lib_pypy/cffi/_pycparser/ply/__init__.py
|
|
Provides: bundled(python2dist(ply)) = 3.9
|
|
|
|
# Find the version in lib_pypy/_cffi_ssl/cryptography/__about__.py
|
|
Provides: bundled(python2dist(cryptography)) = 2.7
|
|
|
|
Provides: %{ver_name}-libs = %{version}-%{release}
|
|
Provides: %{ver_name}-libs%{_isa} = %{version}-%{release}
|
|
Provides: pypy%{pyversion}-libs = %{version}-%{release}
|
|
Provides: pypy%{pyversion}-libs%{_isa} = %{version}-%{release}
|
|
|
|
%description libs
|
|
Libraries required by the various PyPy implementations of Python.
|
|
|
|
|
|
%package devel
|
|
Summary: Development tools for working with PyPy
|
|
Requires: %{name}%{?_isa} = %{version}-%{release}
|
|
Provides: %{ver_name}-devel = %{version}-%{release}
|
|
Provides: %{ver_name}-devel%{_isa} = %{version}-%{release}
|
|
Provides: pypy%{pyversion}-devel = %{version}-%{release}
|
|
Provides: pypy%{pyversion}-devel%{_isa} = %{version}-%{release}
|
|
|
|
%description devel
|
|
Header files for building C extension modules against PyPy
|
|
|
|
|
|
%if 0%{with_stackless}
|
|
%package stackless
|
|
Summary: Stackless Python interpreter built using PyPy
|
|
Requires: %{name}-libs%{?_isa} = %{version}-%{release}
|
|
Provides: %{ver_name}-stackless = %{version}-%{release}
|
|
Provides: %{ver_name}-stackless%{_isa} = %{version}-%{release}
|
|
Provides: pypy%{pyversion}-stackless = %{version}-%{release}
|
|
Provides: pypy%{pyversion}-stackless%{_isa} = %{version}-%{release}
|
|
%description stackless
|
|
Build of PyPy with support for micro-threads for massive concurrency
|
|
%endif
|
|
|
|
|
|
%prep
|
|
%autosetup -n pypy%{pyversion}-v%{version}-src -p1 -S git
|
|
|
|
# Temporary workaround for https://bugzilla.redhat.com/show_bug.cgi?id=1954999
|
|
%{?!apply_patch:%define apply_patch(qp:m:) {%__apply_patch %**}}
|
|
|
|
%if %{with rpmwheels}
|
|
%apply_patch -m %(basename %{SOURCE189}) %{SOURCE189}
|
|
rm lib-python/2.7/ensurepip/_bundled/*.whl
|
|
rmdir lib-python/2.7/ensurepip/_bundled
|
|
%endif
|
|
|
|
# Replace /usr/local/bin/python or /usr/bin/env python shebangs with /usr/bin/python2 or pypy2:
|
|
find \( -name "*.py" -o -name "py.cleanup" \) -exec \
|
|
sed \
|
|
-i -r -e "s@/usr/(local/)?bin/(env )?python(2|3)?@/usr/bin/%{bootstrap_python_interp}@" \
|
|
"{}" \
|
|
\;
|
|
|
|
for f in rpython/translator/goal/bpnn.py ; do
|
|
# Detect shebang lines && remove them:
|
|
sed -e '/^#!/Q 0' -e 'Q 1' $f \
|
|
&& sed -i '1d' $f
|
|
chmod a-x $f
|
|
done
|
|
|
|
rm -rf lib-python/3
|
|
|
|
# Replace all lib-python python shebangs with pypy
|
|
find lib-python/%{pylibver} -name "*.py" -exec \
|
|
sed -r -i '1s|^#!\s*/usr/bin.*python.*|#!/usr/bin/%{name}|' \
|
|
"{}" \
|
|
\;
|
|
|
|
%if ! 0%{use_self_when_building}
|
|
# use the pycparser from PyPy even on CPython
|
|
ln -s lib_pypy/cffi/_pycparser pycparser
|
|
%endif
|
|
|
|
# Remove windows executable binaries
|
|
rm lib-python/2.7/distutils/command/*.exe
|
|
|
|
%build
|
|
%ifarch s390x
|
|
# pypy3 requires z10 at least
|
|
%global optflags %(echo %{optflags} | sed 's/-march=z9-109 /-march=z10 /')
|
|
%endif
|
|
|
|
BuildPyPy() {
|
|
ExeName=$1
|
|
Options=$2
|
|
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "STARTING BUILD OF: $ExeName"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
|
|
pushd %{goal_dir}
|
|
|
|
# The build involves invoking a python script, passing in particular
|
|
# arguments, environment variables, etc.
|
|
# Some notes on those follow:
|
|
|
|
# The generated binary embeds copies of the values of all environment
|
|
# variables. We need to unset "RPM_BUILD_ROOT" to avoid a fatal error from
|
|
# /usr/lib/rpm/check-buildroot
|
|
# during the postprocessing of the rpmbuild, complaining about this
|
|
# reference to the buildroot
|
|
|
|
|
|
# By default, pypy's autogenerated C code is placed in
|
|
# /tmp/usession-N
|
|
#
|
|
# and it appears that this stops rpm from extracting the source code to the
|
|
# debuginfo package
|
|
#
|
|
# The logic in pypy-1.4/pypy/tool/udir.py indicates that it is generated in:
|
|
# $PYPY_USESSION_DIR/usession-$PYPY_USESSION_BASENAME-N
|
|
# and so we set PYPY_USESSION_DIR so that this tempdir is within the build
|
|
# location, and set $PYPY_USESSION_BASENAME so that the tempdir is unique
|
|
# for each invocation of BuildPyPy
|
|
|
|
# Compilation flags for C code:
|
|
# pypy-1.4/pypy/translator/c/genc.py:gen_makefile
|
|
# assembles a Makefile within
|
|
# THE_UDIR/testing_1/Makefile
|
|
# calling out to platform.gen_makefile
|
|
# For us, that's
|
|
# pypy-1.4/pypy/translator/platform/linux.py: class BaseLinux(BasePosix):
|
|
# which by default has:
|
|
# CFLAGS = ['-O3', '-pthread', '-fomit-frame-pointer',
|
|
# '-Wall', '-Wno-unused']
|
|
# plus all substrings from CFLAGS in the environment.
|
|
# This is used to generate a value for CFLAGS that's written into the Makefile
|
|
|
|
# How will we track garbage-collection roots in the generated code?
|
|
# http://pypy.readthedocs.org/en/latest/config/translation.gcrootfinder.html
|
|
|
|
# This is the most portable option, and avoids a reliance on non-guaranteed
|
|
# behaviors within GCC's code generator: use an explicitly-maintained stack
|
|
# of root pointers:
|
|
%global gcrootfinder_options --gcrootfinder=shadowstack
|
|
|
|
export CFLAGS=$(echo "$RPM_OPT_FLAGS" | sed -e 's/-g//')
|
|
|
|
# The generated C code leads to many thousands of warnings of the form:
|
|
# warning: variable 'l_v26003' set but not used [-Wunused-but-set-variable]
|
|
# Suppress them:
|
|
export CFLAGS=$(echo "$CFLAGS" -Wno-unused -fPIC)
|
|
|
|
# If we're already built the JIT-enabled "pypy", then use it for subsequent
|
|
# builds (of other configurations):
|
|
if test -x './pypy' ; then
|
|
INTERP='./pypy'
|
|
%ifarch %{arm}
|
|
# Reduce memory usage on arm during installation
|
|
PYPY_GC_MAX_DELTA=200MB $INTERP --jit loop_longevity=300 ../../rpython/bin/rpython -Ojit targetpypystandalone
|
|
%endif
|
|
else
|
|
# First pypy build within this rpm build?
|
|
# Fall back to using the bootstrap python interpreter, which might be a
|
|
# system copy of pypy from an earlier rpm, or be cpython's /usr/bin/python:
|
|
INTERP='%{bootstrap_python_interp}'
|
|
fi
|
|
|
|
# Here's where we actually invoke the build:
|
|
RPM_BUILD_ROOT= \
|
|
PYPY_USESSION_DIR=$(pwd) \
|
|
PYPY_USESSION_BASENAME=$ExeName \
|
|
$INTERP ../../rpython/bin/rpython \
|
|
%{gcrootfinder_options} \
|
|
$Options \
|
|
targetpypystandalone
|
|
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "FINISHED BUILDING: $ExeName"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
|
|
popd
|
|
}
|
|
|
|
BuildPyPy \
|
|
%{name} \
|
|
%if 0%{with_jit}
|
|
"-Ojit" \
|
|
%else
|
|
"-O2" \
|
|
%endif
|
|
%{nil}
|
|
|
|
%if 0%{with_stackless}
|
|
BuildPyPy \
|
|
%{name}-stackless \
|
|
"--stackless"
|
|
%endif
|
|
|
|
%if %{with_emacs}
|
|
%{_emacs_bytecompile} rpython/jit/tool/pypytrace-mode.el
|
|
%endif
|
|
|
|
|
|
%install
|
|
|
|
mkdir -p %{buildroot}/%{_bindir}
|
|
mkdir -p %{buildroot}/%{pypyprefix}
|
|
|
|
#%if 0%{with_stackless}
|
|
#InstallPyPy %{name}-stackless
|
|
#%endif
|
|
|
|
|
|
# Run installing script, archive-name %{name}-%{basever} in %{buildroot}/%{_libdir} == %{pypyprefix}
|
|
%{bootstrap_python_interp} pypy/tool/release/package.py --archive-name %{name}-%{basever} --builddir %{buildroot}/%{_libdir} --no-embedded-dependencies
|
|
|
|
# Remove shebang lines from .py files that aren't executable, and
|
|
# remove executability from .py files that don't have a shebang line:
|
|
find \
|
|
%{buildroot} \
|
|
-name "*.py" \
|
|
\( \
|
|
\( \! -perm /u+x,g+x,o+x -exec sed -e '/^#!/Q 0' -e 'Q 1' {} \; \
|
|
-print -exec sed -i '1d' {} \; \
|
|
\) \
|
|
-o \
|
|
\( \
|
|
-perm /u+x,g+x,o+x ! -exec grep -m 1 -q '^#!' {} \; \
|
|
-exec chmod a-x {} \; \
|
|
\) \
|
|
\)
|
|
|
|
|
|
execstack --clear-execstack %{buildroot}/%{pypyprefix}/bin/pypy
|
|
|
|
# Bytecompile all of the .py files we ship, using our pypy binary, giving us
|
|
# .pyc files for pypy.
|
|
#
|
|
# Note that some of the test files deliberately contain syntax errors, so
|
|
# we are running it in subshell, to be able to ignore the failures and not to terminate the build.
|
|
(%{py_byte_compile %{buildroot}%{pypyprefix}/bin/pypy %{buildroot}%{pypyprefix}}) || :
|
|
|
|
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'import _tkinter'
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'import Tkinter'
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'import _sqlite3'
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'import _curses'
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'import curses'
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'import syslog'
|
|
%{buildroot}%{pypyprefix}/bin/%{name} -c 'from _sqlite3 import *'
|
|
|
|
|
|
# Header files for C extension modules.
|
|
# Upstream's packaging process (pypy/tool/release/package.py)
|
|
# creates an "include" subdir and copies all *.h/*.inl from "include" there
|
|
# (it also has an apparently out-of-date comment about copying them from
|
|
# pypy/_interfaces, but this directory doesn't seem to exist, and it doesn't
|
|
# seem to do this as of 2011-01-13)
|
|
|
|
# Capture the RPython source code files from the build within the debuginfo
|
|
# package (rhbz#666975)
|
|
%global pypy_debuginfo_dir /usr/src/debug/pypy-%{version}-src
|
|
mkdir -p %{buildroot}%{pypy_debuginfo_dir}
|
|
|
|
# copy over everything:
|
|
cp -a pypy %{buildroot}%{pypy_debuginfo_dir}
|
|
|
|
# ...then delete files that aren't:
|
|
# - *.py files
|
|
# - the Makefile
|
|
# - typeids.txt
|
|
# - dynamic-symbols-*
|
|
#find \
|
|
# %{buildroot}%{pypy_debuginfo_dir} \
|
|
# \( -type f \
|
|
# -a \
|
|
# \! \( -name "*.py" \
|
|
# -o \
|
|
# -name "Makefile" \
|
|
# -o \
|
|
# -name "typeids.txt" \
|
|
# -o \
|
|
# -name "dynamic-symbols-*" \
|
|
# \) \
|
|
# \) \
|
|
# -delete
|
|
|
|
# Alternatively, we could simply keep everything. This leads to a ~350MB
|
|
# debuginfo package, but it makes it easy to hack on the Makefile and C build
|
|
# flags by rebuilding/linking the sources.
|
|
# To do so, remove the above "find" command.
|
|
|
|
# We don't need bytecode for these files; they are being included for reference
|
|
# purposes.
|
|
# There are some rpmlint warnings from these files:
|
|
# non-executable-script
|
|
# wrong-script-interpreter
|
|
# zero-length
|
|
# script-without-shebang
|
|
# dangling-symlink
|
|
# but given that the objective is to preserve a copy of the source code, those
|
|
# are acceptable.
|
|
|
|
# Install the JIT trace mode for Emacs:
|
|
%if %{with_emacs}
|
|
mkdir -p %{buildroot}/%{_emacs_sitelispdir}
|
|
cp -a rpython/jit/tool/pypytrace-mode.el %{buildroot}/%{_emacs_sitelispdir}/%{name}trace-mode.el
|
|
cp -a rpython/jit/tool/pypytrace-mode.elc %{buildroot}/%{_emacs_sitelispdir}/%{name}trace-mode.elc
|
|
%endif
|
|
|
|
# Create executables pypy, pypy2 and pypy2.7
|
|
ln -sf %{pypyprefix}/bin/%{name} %{buildroot}%{_bindir}/%{name}%{pylibver}
|
|
ln -sf %{_bindir}/%{name}%{pylibver} %{buildroot}%{_bindir}/%{name}%{pymajorlibver}
|
|
ln -sf %{_bindir}/%{name}%{pymajorlibver} %{buildroot}%{_bindir}/%{name}
|
|
|
|
# Move files to the right places and remove unnecessary files
|
|
mv %{buildroot}/%{pypyprefix}/bin/libpypy-c.so %{buildroot}/%{_libdir}
|
|
rm -rf %{buildroot}/%{_libdir}/%{name}-%{basever}.tar.bz2
|
|
rm -rf %{buildroot}/%{pypyprefix}/README.rst
|
|
rm -rf %{buildroot}/%{pypyprefix}/README.rst
|
|
rm -rf %{buildroot}/%{pypy_include_dir}/README
|
|
chrpath --delete %{buildroot}/%{pypyprefix}/bin/%{name}
|
|
|
|
# Install macros for rpm:
|
|
mkdir -p %{buildroot}/%{_rpmconfigdir}/macros.d
|
|
install -m 644 %{SOURCE1} %{buildroot}/%{_rpmconfigdir}/macros.d
|
|
install -m 644 %{SOURCE2} %{buildroot}/%{_rpmconfigdir}/macros.d
|
|
|
|
# Remove build script from the package
|
|
#rm %{buildroot}/%{pypyprefix}/lib_pypy/ctypes_config_cache/rebuild.py
|
|
|
|
# since 5.10.0, the debug binaries are built and shipped, making the
|
|
# pypy package ~350 MiB. let's remove them here for now and TODO figure out why
|
|
rm -f %{buildroot}%{pypyprefix}/bin/pypy.debug
|
|
rm -f %{buildroot}%{pypyprefix}/bin/libpypy-c.so.debug
|
|
|
|
%check
|
|
topdir=$(pwd)
|
|
|
|
SkipTest() {
|
|
TEST_NAME=$1
|
|
sed -i -e"s|^$TEST_NAME$||g" testnames.txt
|
|
}
|
|
|
|
CheckPyPy() {
|
|
# We'll be exercising one of the freshly-built binaries using the
|
|
# test suite from the standard library (overridden in places by pypy's
|
|
# modified version)
|
|
ExeName=$1
|
|
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "STARTING TEST OF: $ExeName"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
|
|
pushd %{goal_dir}
|
|
|
|
# I'm seeing numerous cases where tests seem to hang, or fail unpredictably
|
|
# So we'll run each test in its own process, with a timeout
|
|
|
|
# Use regrtest to explicitly list all tests:
|
|
( ./$ExeName -c \
|
|
"from test.regrtest import findtests; print('\n'.join(findtests()))"
|
|
) > testnames.txt
|
|
|
|
# Skip some tests:
|
|
# "audioop" doesn't exist for pypy yet:
|
|
SkipTest test_audioop
|
|
|
|
# The gdb CPython hooks haven't been ported to cpyext:
|
|
SkipTest test_gdb
|
|
|
|
# hotshot relies heavily on _hotshot, which doesn't exist:
|
|
SkipTest test_hotshot
|
|
|
|
# "strop" module doesn't exist for pypy yet:
|
|
SkipTest test_strop
|
|
|
|
# I'm seeing Koji builds hanging e.g.:
|
|
# http://koji.fedoraproject.org/koji/getfile?taskID=3386821&name=build.log
|
|
# The only test that seems to have timed out in that log is
|
|
# test_multiprocessing, so skip it for now:
|
|
SkipTest test_multiprocessing
|
|
|
|
echo "== Test names =="
|
|
cat testnames.txt
|
|
echo "================="
|
|
|
|
echo "" > failed-tests.txt
|
|
|
|
for TestName in $(cat testnames.txt) ; do
|
|
|
|
echo "===================" $TestName "===================="
|
|
|
|
# Use /usr/bin/time (rather than the shell "time" builtin) to gather
|
|
# info on the process (time/CPU/memory). This passes on the exit
|
|
# status of the underlying command
|
|
#
|
|
# Use perl's alarm command to impose a timeout
|
|
# 900 seconds is 15 minutes per test.
|
|
# If a test hangs, that test should get terminated, allowing the build
|
|
# to continue.
|
|
#
|
|
# Invoke pypy on test.regrtest to run the specific test suite
|
|
# verbosely
|
|
#
|
|
# For now, || true, so that any failures don't halt the build:
|
|
( /usr/bin/time \
|
|
perl -e 'alarm shift @ARGV; exec @ARGV' 900 \
|
|
./$ExeName -m test.regrtest -v $TestName ) \
|
|
|| (echo $TestName >> failed-tests.txt) \
|
|
|| true
|
|
done
|
|
|
|
echo "== Failed tests =="
|
|
cat failed-tests.txt
|
|
echo "================="
|
|
|
|
popd
|
|
|
|
# Doublecheck pypy's own test suite, using the built pypy binary:
|
|
|
|
# Disabled for now:
|
|
# x86_64 shows various failures inside:
|
|
# jit/backend/x86/test
|
|
# followed by a segfault inside
|
|
# jit/backend/x86/test/test_runner.py
|
|
#
|
|
# i686 shows various failures inside:
|
|
# jit/backend/x86/test
|
|
# with the x86_64 failure leading to cancellation of the i686 build
|
|
|
|
# Here's the disabled code:
|
|
# pushd pypy
|
|
# time translator/goal/$ExeName test_all.py
|
|
# popd
|
|
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "FINISHED TESTING: $ExeName"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
echo "--------------------------------------------------------------"
|
|
}
|
|
|
|
#python testrunner/runner.py --logfile=pytest-A.log --config=pypy/pytest-A.cfg --config=pypy/pytest-A.py --root=pypy --timeout=3600
|
|
#python pypy/test_all.py --pypy=pypy/goal/pypy --timeout=3600 --resultlog=cpython.log lib-python
|
|
#python pypy/test_all.py --pypy=pypy/goal/pypy --resultlog=pypyjit.log pypy/module/pypyjit/test
|
|
#pypy/goal/pypy pypy/test_all.py --resultlog=pypyjit_new.log
|
|
|
|
%if %{run_selftests}
|
|
CheckPyPy %{name}-c
|
|
|
|
%if 0%{with_stackless}
|
|
CheckPyPy %{name}-c-stackless
|
|
%endif
|
|
|
|
%endif # run_selftests
|
|
|
|
# Because there's a bunch of binary subpackages and creating
|
|
# /usr/share/doc/pypy3-this and /usr/share/doc/pypy3-that
|
|
# is just confusing for the user.
|
|
%global _docdir_fmt %{name}
|
|
|
|
%files libs
|
|
%doc README.rst
|
|
|
|
%dir %{pypyprefix}
|
|
%dir %{pypyprefix}/lib-python
|
|
%license %{pypyprefix}/LICENSE
|
|
%{_libdir}/libpypy-c.so
|
|
%{pypyprefix}/lib-python/%{pylibver}/
|
|
%{pypyprefix}/lib_pypy/
|
|
%{pypyprefix}/site-packages/
|
|
%if %{with_emacs}
|
|
%{_emacs_sitelispdir}/%{name}trace-mode.el
|
|
%{_emacs_sitelispdir}/%{name}trace-mode.elc
|
|
%endif
|
|
|
|
%files
|
|
%doc README.rst
|
|
%{_bindir}/%{name}
|
|
%{_bindir}/%{name}%{pylibver}
|
|
%{_bindir}/%{name}%{pymajorlibver}
|
|
%{pypyprefix}/bin/
|
|
|
|
%files devel
|
|
%dir %{pypy_include_dir}
|
|
%{pypy_include_dir}/*.h
|
|
%{pypy_include_dir}/_numpypy
|
|
%{_rpmconfigdir}/macros.d/macros.%{name}
|
|
%{_rpmconfigdir}/macros.d/macros.%{name}%{pymajorlibver}
|
|
|
|
%if 0%{with_stackless}
|
|
%files stackless
|
|
%doc README.rst
|
|
%{_bindir}/%{name}-stackless
|
|
%endif
|
|
|
|
|
|
%changelog
|
|
%autochangelog
|