Pillow/winbuild/build_prepare.py

495 lines
16 KiB
Python
Raw Normal View History

import os
import shutil
import subprocess
import winreg
from itertools import count
from commands import *
SF_MIRROR = "http://iweb.dl.sourceforge.net"
# use PYTHON to select architecture
architectures = {
"x86": {"vcvars_arch": "x86", "msbuild_arch": "Win32"},
"x64": {"vcvars_arch": "x86_amd64", "msbuild_arch": "x64"},
}
# use PYTHON + architecture to select config
pythons = {
"pypy3.6": {"config-x86": "3.5"},
"3.5": {"config-x86": "3.5", "config-x64": "3.5"},
"3.6": {"config-x86": "3.6", "config-x64": "3.6"},
"3.7": {"config-x86": "3.6", "config-x64": "3.6"},
}
# select deps and libs
configs = {
"3.5": {
"deps": [
"libjpeg-turbo-2.0.3",
"zlib-1.2.11",
"tiff-4.0.10",
"libwebp-1.0.3",
"freetype-2.10.1",
"lcms2-2.9",
"openjpeg-2.3.1",
"ghostscript-9.27",
# "tcl-8.6",
# "tk-8.6",
],
"vcvars_ver": "14.0",
"vs_ver": "2015",
},
"3.6": {
"deps": [
"libjpeg-turbo-2.0.3",
"zlib-1.2.11",
"tiff-4.0.10",
"libwebp-1.0.3",
"freetype-2.10.1",
"lcms2-2.9",
"openjpeg-2.3.1",
"ghostscript-9.27",
# "tcl-8.6",
# "tk-8.6",
],
"vs_ver": "2017",
},
}
header = [
cmd_set("BUILD", "{build_dir}"),
cmd_set("INCLUDE", "{inc_dir}"),
cmd_set("INCLIB", "{lib_dir}"),
cmd_set("LIB", "{lib_dir}"),
cmd_append("PATH", "{bin_dir}"),
"@echo on",
]
# dependencies
deps = {
"jpeg-9c": {
"name": "libjpeg",
# FIXME HTTP 403
"url": "http://www.ijg.org/files/jpegsr9c.zip",
"filename": "jpegsr9c.zip",
"build": [
# FIXME builds with -MT, not -MD
cmd_nmake("makefile.vc", "setup-vc6"),
cmd_nmake("makefile.vc", "clean"),
cmd_nmake("makefile.vc", "libjpeg.lib", "nodebug=1"),
],
"headers": [r"j*.h"],
"libs": [r"*.lib"],
},
"libjpeg-turbo-2.0.3": {
"name": "libjpeg",
"url": SF_MIRROR + "/project/libjpeg-turbo/2.0.3/libjpeg-turbo-2.0.3.tar.gz",
"filename": "libjpeg-turbo-2.0.3.tar.gz",
"build": [
cmd_cmake([
"-DENABLE_SHARED:BOOL=FALSE",
"-DWITH_JPEG8:BOOL=TRUE",
"-DWITH_CRT_DLL:BOOL=TRUE",
]),
cmd_nmake(target="clean"),
cmd_nmake(target="jpeg-static"),
cmd_copy("jpeg-static.lib", "libjpeg.lib"),
cmd_nmake(target="cjpeg-static"),
cmd_copy("cjpeg-static.exe", "cjpeg.exe"),
cmd_nmake(target="djpeg-static"),
cmd_copy("djpeg-static.exe", "djpeg.exe"),
],
"headers": ["j*.h"],
"libs": ["libjpeg.lib"],
"bins": ["cjpeg.exe", "djpeg.exe"],
},
"zlib-1.2.11": {
"name": "zlib",
"url": "http://zlib.net/zlib1211.zip",
"filename": "zlib1211.zip",
"build": [
cmd_nmake(r"win32\Makefile.msc", "clean"),
cmd_nmake(r"win32\Makefile.msc", "zlib.lib"),
cmd_copy("zlib.lib", "z.lib"),
],
"headers": [r"z*.h"],
"libs": [r"*.lib"],
},
"tiff-4.0.10": {
"name": "libtiff",
# FIXME FTP timeout
"url": "ftp://download.osgeo.org/libtiff/tiff-4.0.10.tar.gz",
"filename": "tiff-4.0.10.tar.gz",
"build": [
cmd_copy(r"{script_dir}\tiff.opt", "nmake.opt"),
cmd_nmake("makefile.vc", "clean"),
cmd_nmake("makefile.vc", "lib", "RuntimeLibrary=-MT"),
],
"headers": [r"libtiff\tiff*.h"],
"libs": [r"libtiff\*.lib"],
# "bins": [r"libtiff\*.dll"],
},
"libwebp-1.0.3": {
"name": "libwebp",
"url": "http://downloads.webmproject.org/releases/webp/libwebp-1.0.3.tar.gz", # noqa: E501
"filename": "libwebp-1.0.3.tar.gz",
"build": [
cmd_rmdir(r"output\release-static"), # clean
cmd_nmake(
"Makefile.vc",
"all",
["CFG=release-static", "OBJDIR=output", "ARCH={architecture}"]
),
cmd_mkdir(r"{inc_dir}\webp"),
cmd_copy(r"src\webp\*.h", r"{inc_dir}\webp"),
],
"libs": [r"output\release-static\{architecture}\lib\*.lib"],
},
"freetype-2.10.1": {
"name": "freetype",
"url": "https://download.savannah.gnu.org/releases/freetype/freetype-2.10.1.tar.gz", # noqa: E501
"filename": "freetype-2.10.1.tar.gz",
"build": [
cmd_rmdir("objs"),
# freetype setting is /MD for .dll and /MT for .lib, we need /MD
cmd_patch_replace(
r"builds\windows\vc2010\freetype.vcxproj",
"MultiThreaded<",
"MultiThreadedDLL<"
),
cmd_msbuild(r"builds\windows\vc2010\freetype.sln", "Release Static", "Clean"), # TODO failing on GHA # noqa: E501
cmd_msbuild(r"builds\windows\vc2010\freetype.sln", "Release Static", "Build"),
cmd_xcopy("include", "{inc_dir}"),
],
"libs": [r"objs\{msbuild_arch}\Release Static\freetype.lib"],
# "bins": [r"objs\{msbuild_arch}\Release\freetype.dll"],
},
"lcms2-2.9": {
"name": "lcms2",
"url": SF_MIRROR + "/project/lcms/lcms/2.8/lcms2-2.9.tar.gz",
"filename": "lcms2-2.9.tar.gz",
"build": [
cmd_rmdir("Lib"),
cmd_rmdir(r"Projects\VC{vs_ver}\Release"),
# lcms2-2.8\VC2015 setting is /MD for x86 and /MT for x64, we need /MD always
cmd_patch_replace(
r"Projects\VC2017\lcms2.sln", "MultiThreaded<", "MultiThreadedDLL<"
),
cmd_msbuild(r"Projects\VC{vs_ver}\lcms2.sln", "Release", "Clean"),
cmd_msbuild(r"Projects\VC{vs_ver}\lcms2.sln", "Release", "lcms2_static"),
cmd_xcopy("include", "{inc_dir}"),
],
"libs": [r"Lib\MS\*.lib"],
},
"openjpeg-2.3.1": {
"name": "openjpeg",
"url": "https://github.com/uclouvain/openjpeg/archive/v2.3.1.tar.gz",
"filename": "openjpeg-2.3.1.tar.gz",
"build": [
cmd_cmake(("-DBUILD_THIRDPARTY:BOOL=OFF", "-DBUILD_SHARED_LIBS:BOOL=OFF")),
cmd_nmake(target="clean"),
cmd_nmake(),
cmd_mkdir(r"{inc_dir}\openjpeg-2.3.1"),
cmd_copy(r"src\lib\openjp2\*.h", r"{inc_dir}\openjpeg-2.3.1"),
],
"libs": [r"bin\*.lib"],
},
"ghostscript-9.27": {
"name": "ghostscript",
"url": "https://github.com/ArtifexSoftware/ghostpdl-downloads/releases/download/gs927/ghostscript-9.27.tar.gz", # noqa: E501
"filename": "ghostscript-9.27.tar.gz",
"build": [
cmd_set("MSVC_VERSION", 14),
cmd_if_eq("{architecture}", "x64", cmd_set("WIN64", '""')),
cmd_nmake(r"psi\msvc.mak"),
],
"bins": [r"bin\*"],
},
"tcl-8.5": {
"name": "tcl",
"url": SF_MIRROR + "/project/tcl/Tcl/8.5.19/tcl8519-src.zip",
"filename": "tcl8519-src.zip",
},
"tk-8.5": {
"name": "tk",
"url": SF_MIRROR + "/project/tcl/Tcl/8.5.19/tk8519-src.zip",
"filename": "tk8519-src.zip",
},
"tcl-8.6": {
"name": "tcl",
"url": SF_MIRROR + "/project/tcl/Tcl/8.6.9/tcl869-src.zip",
"filename": "tcl869-src.zip",
"headers": [r"generic\*.h"],
},
"tk-8.6": {
"name": "tk",
"url": SF_MIRROR + "/project/tcl/Tcl/8.6.9/tk869-src.zip",
"filename": "tk869-src.zip",
"build": [
r"""mkdir {inc_dir}\X11""",
r"""copy /Y /B xlib\X11\* "{inc_dir}\X11\" """,
],
"headers": [r"generic\*.h"],
},
}
# based on distutils._msvccompiler from CPython 3.7.4
def find_vs2017(config):
root = os.environ.get("ProgramFiles(x86)") or os.environ.get("ProgramFiles")
if not root:
print("Program Files not found")
return None
try:
vspath = subprocess.check_output([
os.path.join(root, "Microsoft Visual Studio", "Installer", "vswhere.exe"),
"-latest",
"-prerelease",
"-requires", "Microsoft.VisualStudio.Component.VC.Tools.x86.x64",
"-property", "installationPath",
"-products", "*",
]).decode(encoding="mbcs").strip()
except (subprocess.CalledProcessError, OSError, UnicodeDecodeError):
print("vswhere not found")
return None
if not os.path.isdir(os.path.join(vspath, "VC", "Auxiliary", "Build")):
print("Visual Studio seems to be missing C compiler")
return None
vs = {
"header": [],
# nmake selected by vcvarsall
"nmake": "nmake.exe",
}
msbuild = os.path.join(vspath, "MSBuild", "15.0", "Bin", "MSBuild.exe")
if os.path.isfile(msbuild):
# default_platform_toolset = "v140"
vs["msbuild"] = '"{}"'.format(msbuild)
# vs["header"].append(cmd_set("DefaultPlatformToolset", default_platform_toolset))
else:
print("Visual Studio MSBuild not found")
return None
vcvarsall = os.path.join(vspath, "VC", "Auxiliary", "Build", "vcvarsall.bat")
if not os.path.isfile(vcvarsall):
print("Visual Studio vcvarsall not found")
return None
vcvars_ver = "-vcvars_ver={}".format(config["vcvars_ver"]) if "vcvars_ver" in config else ""
vs["header"].append('call "{}" {{vcvars_arch}} {}'.format(vcvarsall, vcvars_ver))
return vs
def find_sdk71a():
try:
key = winreg.OpenKeyEx(
winreg.HKEY_LOCAL_MACHINE,
r"SOFTWARE\Microsoft\Microsoft SDKs\Windows\v7.1A",
access=winreg.KEY_READ | winreg.KEY_WOW64_32KEY,
)
except OSError:
return None
with key:
for i in count():
try:
v_name, v_value, v_type = winreg.EnumValue(key, i)
except OSError:
return None
if v_name == "InstallationFolder" and v_type == winreg.REG_SZ:
sdk_dir = v_value
break
else:
return None
if not os.path.isdir(sdk_dir):
return None
sdk = {
"header": [
# for win32.mak
cmd_append("INCLUDE", os.path.join(sdk_dir, "Include")),
# for ghostscript
cmd_set("RCOMP", '"{}"'.format(os.path.join(sdk_dir, "Bin", "RC.EXE"))),
]
}
return sdk
def match(values, target):
for key, value in values.items():
if key in target:
return {"name": key, **value}
def extract_dep(url, filename):
import urllib.request
import tarfile
import zipfile
file = os.path.join(depends_dir, filename)
if not os.path.exists(file):
ex = None
for i in range(3):
try:
print("Fetching %s (attempt %d)..." % (url, i + 1))
content = urllib.request.urlopen(url).read()
with open(file, "wb") as f:
f.write(content)
break
except urllib.error.URLError as e:
ex = e
else:
raise RuntimeError(ex)
if filename.endswith(".zip"):
with zipfile.ZipFile(file) as zf:
zf.extractall(build_dir)
elif filename.endswith(".tar.gz") or filename.endswith(".tgz"):
with tarfile.open(file, "r:gz") as tgz:
tgz.extractall(build_dir)
else:
raise RuntimeError("Unknown archive type: " + filename)
def write_script(name, lines):
name = os.path.join(script_dir, name)
lines = [line.format(**prefs) for line in lines]
print("Writing " + name)
with open(name, "w") as f:
f.write("\n\r".join(lines))
for line in lines:
print(" " + line)
def get_footer(dep):
lines = []
for out in dep.get("headers", []):
lines.append(cmd_copy(out, "{inc_dir}"))
for out in dep.get("libs", []):
lines.append(cmd_copy(out, "{lib_dir}"))
for out in dep.get("bins", []):
lines.append(cmd_copy(out, "{bin_dir}"))
return lines
def build_dep(name):
dep = deps[name]
file = "build_dep_{name}.cmd".format(name=dep["name"])
extract_dep(dep["url"], dep["filename"])
lines = ["cd /D %s" % os.path.join(build_dir, name)]
lines.extend(prefs["header"])
lines.extend(dep.get("build", []))
lines.extend(get_footer(dep))
write_script(file, lines)
return file
def build_dep_all():
lines = ["cd {script_dir}"]
for dep_name in prefs["deps"]:
lines.append('cmd.exe /c "%s"' % build_dep(dep_name))
write_script("build_dep_all.ps1", lines)
def build_pillow(wheel=False):
if not wheel:
op, filename = "install", "build_pillow.cmd"
else:
op, filename = "bdist_wheel", "build_pillow_wheel.cmd"
lines = []
if path_dir is not None and not wheel:
lines.append(cmd_xcopy("{bin_dir}", path_dir))
lines.extend(prefs["header"])
lines.extend(
[
cmd_cd("{pillow_dir}"),
cmd_append("LIB", r"{python_dir}\tcl"),
r'"{{python_dir}}\python.exe" setup.py build_ext {}'.format(op),
# r"""%PYTHON%\python.exe selftest.py --installed""",
]
)
write_script(filename, lines)
if __name__ == "__main__":
script_dir = os.path.dirname(os.path.realpath(__file__))
depends_dir = os.path.join(script_dir, "depends")
python_dir = os.environ["PYTHON"]
# copy binaries to this directory
path_dir = os.environ.get("PILLOW_BIN")
# use PYTHON to select architecture
arch_prefs = match(architectures, python_dir)
if arch_prefs is None:
architecture = "x86"
print("WARN: Could not determine architecture, guessing " + architecture)
arch_prefs = architectures[architecture]
else:
architecture = arch_prefs["name"]
# use PYTHON to select python version
python_prefs = match(pythons, python_dir)
if python_prefs is None:
raise KeyError("Failed to determine Python version from PYTHON: " + python_dir)
# use python version + architecture to select build config
config_name = python_prefs["config-" + architecture]
config = configs[config_name]
vs2017 = find_vs2017(config)
if vs2017 is None:
raise RuntimeError("Visual Studio 2017 not found")
sdk71a = find_sdk71a()
if sdk71a is None:
raise RuntimeError("Windows SDK v7.1A not found")
build_dir = os.path.join(script_dir, "build", config_name, architecture)
lib_dir = os.path.join(build_dir, "lib")
inc_dir = os.path.join(build_dir, "inc")
bin_dir = os.path.join(build_dir, "bin")
# for path in [lib_dir, inc_dir, bin_dir]:
# shutil.rmtree(path)
for path in [depends_dir, build_dir, lib_dir, inc_dir, bin_dir]:
os.makedirs(path, exist_ok=True)
prefs = {
"python_version": python_prefs["name"],
"architecture": architecture,
"script_dir": script_dir,
"depends_dir": depends_dir,
"python_dir": python_dir,
"build_dir": build_dir,
"lib_dir": lib_dir,
"inc_dir": inc_dir,
"bin_dir": bin_dir,
"pillow_dir": os.path.realpath(os.path.join(script_dir, "..")),
# TODO auto find:
"cmake": "cmake.exe",
}
dicts = [vs2017, sdk71a, arch_prefs, python_prefs, config]
for x in dicts:
prefs.update(x)
prefs["header"] = sum((x.get("header", []) for x in dicts), header)
del prefs["name"]
print("Target: Python {python_version} {architecture}".format(**prefs))
build_dep_all()
build_pillow()
build_pillow(wheel=True)