mirror of https://github.com/pyodide/pyodide.git
463 lines
14 KiB
Python
Executable File
463 lines
14 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
"""Helper for cross-compiling distutils-based Python extensions.
|
|
|
|
distutils has never had a proper cross-compilation story. This is a hack, which
|
|
miraculously works, to get around that.
|
|
|
|
The gist is:
|
|
|
|
- Compile the package natively, replacing calls to the compiler and linker with
|
|
wrappers that store the arguments in a log, and then delegate along to the
|
|
real native compiler and linker.
|
|
|
|
- Remove all of the native build products.
|
|
|
|
- Play back the log, replacing the native compiler with emscripten and
|
|
adjusting include paths and flags as necessary for cross-compiling to
|
|
emscripten. This overwrites the results from the original native compilation.
|
|
|
|
While this results in more work than strictly necessary (it builds a native
|
|
version of the package, even though we then throw it away), it seems to be the
|
|
only reliable way to automatically build a package that interleaves
|
|
configuration with build.
|
|
"""
|
|
|
|
|
|
import argparse
|
|
import importlib.machinery
|
|
import json
|
|
import os
|
|
from pathlib import Path
|
|
import re
|
|
import subprocess
|
|
import shutil
|
|
import sys
|
|
|
|
|
|
# absolute import is necessary as this file will be symlinked
|
|
# under tools
|
|
from pyodide_build import common
|
|
|
|
|
|
ROOTDIR = common.ROOTDIR
|
|
symlinks = set(["cc", "c++", "ld", "ar", "gcc", "gfortran"])
|
|
|
|
|
|
def collect_args(basename):
|
|
"""
|
|
This is called when this script is called through a symlink that looks like
|
|
a compiler or linker.
|
|
|
|
It writes the arguments to the build.log, and then delegates to the real
|
|
native compiler or linker.
|
|
"""
|
|
# Remove the symlink compiler from the PATH, so we can delegate to the
|
|
# native compiler
|
|
env = dict(os.environ)
|
|
path = env["PATH"]
|
|
while str(ROOTDIR) + ":" in path:
|
|
path = path.replace(str(ROOTDIR) + ":", "")
|
|
env["PATH"] = path
|
|
|
|
skip_host = "SKIP_HOST" in os.environ
|
|
|
|
# Skip compilations of C/Fortran extensions for the target environment.
|
|
# We still need to generate the output files for distutils to continue
|
|
# the build.
|
|
# TODO: This may need slight tuning for new projects. In particular,
|
|
# currently ar is not skipped, so a known failure would happen when
|
|
# we create some object files (that are empty as gcc is skipped), on
|
|
# which we run the actual ar command.
|
|
skip = False
|
|
if (
|
|
basename in ["gcc", "cc", "c++", "gfortran", "ld"]
|
|
and "-o" in sys.argv[1:]
|
|
# do not skip numpy as it is needed as build time
|
|
# dependency by other packages (e.g. matplotlib)
|
|
and skip_host
|
|
):
|
|
out_idx = sys.argv.index("-o")
|
|
if (out_idx + 1) < len(sys.argv):
|
|
# get the index of the output file path
|
|
out_idx += 1
|
|
with open(sys.argv[out_idx], "wb") as fh:
|
|
fh.write(b"")
|
|
skip = True
|
|
|
|
with open("build.log", "a") as fd:
|
|
# TODO: store skip status in the build.log
|
|
json.dump([basename] + sys.argv[1:], fd)
|
|
fd.write("\n")
|
|
|
|
if skip:
|
|
sys.exit(0)
|
|
compiler_command = [basename]
|
|
if shutil.which("ccache") is not None:
|
|
# Enable ccache if it's installed
|
|
compiler_command.insert(0, "ccache")
|
|
|
|
sys.exit(subprocess.run(compiler_command + sys.argv[1:], env=env).returncode)
|
|
|
|
|
|
def make_symlinks(env):
|
|
"""
|
|
Makes sure all of the symlinks that make this script look like a compiler
|
|
exist.
|
|
"""
|
|
exec_path = Path(__file__).resolve()
|
|
for symlink in symlinks:
|
|
symlink_path = ROOTDIR / symlink
|
|
if os.path.lexists(symlink_path) and not symlink_path.exists():
|
|
# remove broken symlink so it can be re-created
|
|
symlink_path.unlink()
|
|
if not symlink_path.exists():
|
|
symlink_path.symlink_to(exec_path)
|
|
if symlink == "c++":
|
|
var = "CXX"
|
|
else:
|
|
var = symlink.upper()
|
|
env[var] = symlink
|
|
|
|
|
|
def capture_compile(args):
|
|
env = dict(os.environ)
|
|
make_symlinks(env)
|
|
env["PATH"] = str(ROOTDIR) + ":" + os.environ["PATH"]
|
|
|
|
cmd = [sys.executable, "setup.py", "install"]
|
|
if args.install_dir == "skip":
|
|
cmd[-1] = "build"
|
|
elif args.install_dir != "":
|
|
cmd.extend(["--home", args.install_dir])
|
|
|
|
result = subprocess.run(cmd, env=env)
|
|
if result.returncode != 0:
|
|
build_log_path = Path("build.log")
|
|
if build_log_path.exists():
|
|
build_log_path.unlink()
|
|
sys.exit(result.returncode)
|
|
|
|
|
|
def f2c(args, dryrun=False):
|
|
"""Apply f2c to compilation arguments
|
|
|
|
Parameters
|
|
----------
|
|
args : iterable
|
|
input compiler arguments
|
|
dryrun : bool, default=True
|
|
if False run f2c on detected fortran files
|
|
|
|
Returns
|
|
-------
|
|
new_args : list
|
|
output compiler arguments
|
|
|
|
|
|
Examples
|
|
--------
|
|
|
|
>>> f2c(['gfortran', 'test.f'], dryrun=True)
|
|
['gfortran', 'test.c']
|
|
"""
|
|
new_args = []
|
|
found_source = False
|
|
for arg in args:
|
|
if arg.endswith(".f"):
|
|
filename = os.path.abspath(arg)
|
|
if not dryrun:
|
|
subprocess.check_call(
|
|
["f2c", os.path.basename(filename)], cwd=os.path.dirname(filename)
|
|
)
|
|
new_args.append(arg[:-2] + ".c")
|
|
found_source = True
|
|
else:
|
|
new_args.append(arg)
|
|
|
|
new_args_str = " ".join(args)
|
|
if ".so" in new_args_str and "libgfortran.so" not in new_args_str:
|
|
found_source = True
|
|
|
|
if not found_source:
|
|
print(f"f2c: source not found, skipping: {new_args_str}")
|
|
return None
|
|
return new_args
|
|
|
|
|
|
def handle_command(line, args, dryrun=False):
|
|
"""Handle a compilation command
|
|
|
|
Parameters
|
|
----------
|
|
line : iterable
|
|
an iterable with the compilation arguments
|
|
args : {object, namedtuple}
|
|
an container with additional compilation options,
|
|
in particular containing ``args.cflags`` and ``args.ldflags``
|
|
dryrun : bool, default=False
|
|
if True do not run the resulting command, only return it
|
|
|
|
Examples
|
|
--------
|
|
|
|
>>> from collections import namedtuple
|
|
>>> Args = namedtuple('args', ['cflags', 'ldflags', 'host'])
|
|
>>> args = Args(cflags='', ldflags='', host='')
|
|
>>> handle_command(['gcc', 'test.c'], args, dryrun=True)
|
|
emcc test.c
|
|
['emcc', 'test.c']
|
|
"""
|
|
# This is a special case to skip the compilation tests in numpy that aren't
|
|
# actually part of the build
|
|
for arg in line:
|
|
if r"/file.c" in arg or "_configtest" in arg:
|
|
return
|
|
if re.match(r"/tmp/.*/source\.[bco]+", arg):
|
|
return
|
|
if arg == "-print-multiarch":
|
|
return
|
|
if arg.startswith("/tmp"):
|
|
return
|
|
|
|
if line[0] == "gfortran":
|
|
result = f2c(line)
|
|
if result is None:
|
|
return
|
|
line = result
|
|
new_args = ["emcc"]
|
|
elif line[0] == "ar":
|
|
new_args = ["emar"]
|
|
elif line[0] == "c++":
|
|
new_args = ["em++"]
|
|
else:
|
|
new_args = ["emcc"]
|
|
# distutils doesn't use the c++ compiler when compiling c++ <sigh>
|
|
if any(arg.endswith(".cpp") for arg in line):
|
|
new_args = ["em++"]
|
|
library_output = line[-1].endswith(".so")
|
|
|
|
if library_output:
|
|
new_args.extend(args.ldflags.split())
|
|
elif new_args[0] in ("emcc", "em++"):
|
|
new_args.extend(args.cflags.split())
|
|
|
|
lapack_dir = None
|
|
|
|
# Go through and adjust arguments
|
|
for arg in line[1:]:
|
|
if arg.startswith("-I"):
|
|
if (
|
|
str(Path(arg[2:]).resolve()).startswith(sys.prefix + "/include/python")
|
|
and "site-packages" not in arg
|
|
):
|
|
arg = arg.replace("-I" + sys.prefix, "-I" + args.target)
|
|
# Don't include any system directories
|
|
elif arg[2:].startswith("/usr"):
|
|
continue
|
|
# Don't include any system directories
|
|
if arg.startswith("-L/usr"):
|
|
continue
|
|
# threading is disabled for now
|
|
if arg == "-pthread":
|
|
continue
|
|
# On Mac, we need to omit some darwin-specific arguments
|
|
if arg in ["-bundle", "-undefined", "dynamic_lookup"]:
|
|
continue
|
|
# The native build is possibly multithreaded, but the emscripten one
|
|
# definitely isn't
|
|
arg = re.sub(r"/python([0-9]\.[0-9]+)m", r"/python\1", arg)
|
|
if arg.endswith(".o"):
|
|
arg = arg[:-2] + ".bc"
|
|
output = arg
|
|
elif arg.endswith(".so"):
|
|
arg = arg[:-3] + ".wasm"
|
|
output = arg
|
|
|
|
# Fix for scipy to link to the correct BLAS/LAPACK files
|
|
if arg.startswith("-L") and "CLAPACK-WA" in arg:
|
|
out_idx = line.index("-o")
|
|
out_idx += 1
|
|
module_name = line[out_idx]
|
|
module_name = Path(module_name).name.split(".")[0]
|
|
|
|
lapack_dir = arg.replace("-L", "")
|
|
# For convinience we determine needed scipy link libraries
|
|
# here, instead of in patch files
|
|
link_libs = ["F2CLIBS/libf2c.a", "blas_WA.a"]
|
|
if module_name in [
|
|
"_flapack",
|
|
"_flinalg",
|
|
"_calc_lwork",
|
|
"cython_lapack",
|
|
"_iterative",
|
|
"_arpack",
|
|
]:
|
|
link_libs.append("lapack_WA.a")
|
|
|
|
for lib_name in link_libs:
|
|
arg = os.path.join(lapack_dir, f"{lib_name}")
|
|
new_args.append(arg)
|
|
|
|
new_args.extend(["-s", "INLINING_LIMIT=5"])
|
|
continue
|
|
|
|
# Use -Os for files that are statically linked to CLAPACK
|
|
if (
|
|
arg.startswith("-O")
|
|
and "CLAPACK" in " ".join(line)
|
|
and "-L" in " ".join(line)
|
|
):
|
|
new_args.append("-Os")
|
|
continue
|
|
|
|
if new_args[-1].startswith("-B") and "compiler_compat" in arg:
|
|
# conda uses custom compiler search paths with the compiler_compat folder.
|
|
# Ignore it.
|
|
del new_args[-1]
|
|
continue
|
|
|
|
# See https://github.com/emscripten-core/emscripten/issues/8650
|
|
if arg in ["-lfreetype", "-lz", "-lpng16", "-lgfortran"]:
|
|
continue
|
|
|
|
new_args.append(arg)
|
|
|
|
# This can only be used for incremental rebuilds -- it generates
|
|
# an error during clean build of numpy
|
|
# if os.path.isfile(output):
|
|
# print('SKIPPING: ' + ' '.join(new_args))
|
|
# return
|
|
|
|
print(" ".join(new_args))
|
|
|
|
if not dryrun:
|
|
result = subprocess.run(new_args)
|
|
if result.returncode != 0:
|
|
sys.exit(result.returncode)
|
|
|
|
# Emscripten .so files shouldn't have the native platform slug
|
|
if library_output:
|
|
renamed = output[:-5] + ".so"
|
|
for ext in importlib.machinery.EXTENSION_SUFFIXES:
|
|
if ext == ".so":
|
|
continue
|
|
if renamed.endswith(ext):
|
|
renamed = renamed[: -len(ext)] + ".so"
|
|
break
|
|
if not dryrun:
|
|
os.rename(output, renamed)
|
|
return new_args
|
|
|
|
|
|
def replay_compile(args):
|
|
# If pure Python, there will be no build.log file, which is fine -- just do
|
|
# nothing
|
|
build_log_path = Path("build.log")
|
|
if build_log_path.is_file():
|
|
with open(build_log_path, "r") as fd:
|
|
for line in fd:
|
|
line = json.loads(line)
|
|
handle_command(line, args)
|
|
|
|
|
|
def clean_out_native_artifacts():
|
|
for root, dirs, files in os.walk("."):
|
|
for file in files:
|
|
path = Path(root) / file
|
|
if path.suffix in (".o", ".so", ".a"):
|
|
path.unlink()
|
|
|
|
|
|
def install_for_distribution(args):
|
|
commands = [
|
|
sys.executable,
|
|
"setup.py",
|
|
"install",
|
|
"--skip-build",
|
|
"--prefix=install",
|
|
"--old-and-unmanageable",
|
|
]
|
|
try:
|
|
subprocess.check_call(commands)
|
|
except Exception:
|
|
print(
|
|
f'Warning: {" ".join(str(arg) for arg in commands)} failed '
|
|
f"with distutils, possibly due to the use of distutils "
|
|
f"that does not support the --old-and-unmanageable "
|
|
"argument. Re-trying the install without this argument."
|
|
)
|
|
subprocess.check_call(commands[:-1])
|
|
|
|
|
|
def build_wrap(args):
|
|
build_log_path = Path("build.log")
|
|
if not build_log_path.is_file():
|
|
capture_compile(args)
|
|
clean_out_native_artifacts()
|
|
replay_compile(args)
|
|
install_for_distribution(args)
|
|
|
|
|
|
def make_parser(parser):
|
|
basename = Path(sys.argv[0]).name
|
|
if basename in symlinks:
|
|
# skip parsing of all arguments
|
|
parser._actions = []
|
|
else:
|
|
parser.description = (
|
|
"Cross compile a Python distutils package. "
|
|
"Run from the root directory of the package's source"
|
|
)
|
|
parser.add_argument(
|
|
"--cflags",
|
|
type=str,
|
|
nargs="?",
|
|
default=common.DEFAULTCFLAGS,
|
|
help="Extra compiling flags",
|
|
)
|
|
parser.add_argument(
|
|
"--ldflags",
|
|
type=str,
|
|
nargs="?",
|
|
default=common.DEFAULTLDFLAGS,
|
|
help="Extra linking flags",
|
|
)
|
|
parser.add_argument(
|
|
"--target",
|
|
type=str,
|
|
nargs="?",
|
|
default=common.TARGETPYTHON,
|
|
help="The path to the target Python installation",
|
|
)
|
|
parser.add_argument(
|
|
"--install-dir",
|
|
type=str,
|
|
nargs="?",
|
|
default="",
|
|
help=(
|
|
"Directory for installing built host packages. Defaults to setup.py "
|
|
"default. Set to 'skip' to skip installation. Installation is "
|
|
"needed if you want to build other packages that depend on this one."
|
|
),
|
|
)
|
|
return parser
|
|
|
|
|
|
def main(args):
|
|
basename = Path(sys.argv[0]).name
|
|
if basename in symlinks:
|
|
collect_args(basename)
|
|
else:
|
|
build_wrap(args)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
basename = Path(sys.argv[0]).name
|
|
if basename in symlinks:
|
|
main(None)
|
|
else:
|
|
parser = make_parser(argparse.ArgumentParser())
|
|
args = parser.parse_args()
|
|
main(args)
|