mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-11-04 09:57:43 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			1034 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1034 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# > pyroma .
 | 
						|
# ------------------------------
 | 
						|
# Checking .
 | 
						|
# Found Pillow
 | 
						|
# ------------------------------
 | 
						|
# Final rating: 10/10
 | 
						|
# Your cheese is so fresh most people think it's a cream: Mascarpone
 | 
						|
# ------------------------------
 | 
						|
from __future__ import annotations
 | 
						|
 | 
						|
import os
 | 
						|
import re
 | 
						|
import shutil
 | 
						|
import struct
 | 
						|
import subprocess
 | 
						|
import sys
 | 
						|
import warnings
 | 
						|
 | 
						|
from setuptools import Extension, setup
 | 
						|
from setuptools.command.build_ext import build_ext
 | 
						|
 | 
						|
 | 
						|
def get_version():
 | 
						|
    version_file = "src/PIL/_version.py"
 | 
						|
    with open(version_file, encoding="utf-8") as f:
 | 
						|
        exec(compile(f.read(), version_file, "exec"))
 | 
						|
    return locals()["__version__"]
 | 
						|
 | 
						|
 | 
						|
PILLOW_VERSION = get_version()
 | 
						|
FREETYPE_ROOT = None
 | 
						|
HARFBUZZ_ROOT = None
 | 
						|
FRIBIDI_ROOT = None
 | 
						|
IMAGEQUANT_ROOT = None
 | 
						|
JPEG2K_ROOT = None
 | 
						|
JPEG_ROOT = None
 | 
						|
LCMS_ROOT = None
 | 
						|
TIFF_ROOT = None
 | 
						|
ZLIB_ROOT = None
 | 
						|
FUZZING_BUILD = "LIB_FUZZING_ENGINE" in os.environ
 | 
						|
 | 
						|
if sys.platform == "win32" and sys.version_info >= (3, 13):
 | 
						|
    import atexit
 | 
						|
 | 
						|
    atexit.register(
 | 
						|
        lambda: warnings.warn(
 | 
						|
            f"Pillow {PILLOW_VERSION} does not support Python "
 | 
						|
            f"{sys.version_info.major}.{sys.version_info.minor} and does not provide "
 | 
						|
            "prebuilt Windows binaries. We do not recommend building from source on "
 | 
						|
            "Windows.",
 | 
						|
            RuntimeWarning,
 | 
						|
        )
 | 
						|
    )
 | 
						|
 | 
						|
 | 
						|
_IMAGING = ("decode", "encode", "map", "display", "outline", "path")
 | 
						|
 | 
						|
_LIB_IMAGING = (
 | 
						|
    "Access",
 | 
						|
    "AlphaComposite",
 | 
						|
    "Resample",
 | 
						|
    "Reduce",
 | 
						|
    "Bands",
 | 
						|
    "BcnDecode",
 | 
						|
    "BitDecode",
 | 
						|
    "Blend",
 | 
						|
    "Chops",
 | 
						|
    "ColorLUT",
 | 
						|
    "Convert",
 | 
						|
    "ConvertYCbCr",
 | 
						|
    "Copy",
 | 
						|
    "Crop",
 | 
						|
    "Dib",
 | 
						|
    "Draw",
 | 
						|
    "Effects",
 | 
						|
    "EpsEncode",
 | 
						|
    "File",
 | 
						|
    "Fill",
 | 
						|
    "Filter",
 | 
						|
    "FliDecode",
 | 
						|
    "Geometry",
 | 
						|
    "GetBBox",
 | 
						|
    "GifDecode",
 | 
						|
    "GifEncode",
 | 
						|
    "HexDecode",
 | 
						|
    "Histo",
 | 
						|
    "JpegDecode",
 | 
						|
    "JpegEncode",
 | 
						|
    "Matrix",
 | 
						|
    "ModeFilter",
 | 
						|
    "Negative",
 | 
						|
    "Offset",
 | 
						|
    "Pack",
 | 
						|
    "PackDecode",
 | 
						|
    "Palette",
 | 
						|
    "Paste",
 | 
						|
    "Quant",
 | 
						|
    "QuantOctree",
 | 
						|
    "QuantHash",
 | 
						|
    "QuantHeap",
 | 
						|
    "PcdDecode",
 | 
						|
    "PcxDecode",
 | 
						|
    "PcxEncode",
 | 
						|
    "Point",
 | 
						|
    "RankFilter",
 | 
						|
    "RawDecode",
 | 
						|
    "RawEncode",
 | 
						|
    "Storage",
 | 
						|
    "SgiRleDecode",
 | 
						|
    "SunRleDecode",
 | 
						|
    "TgaRleDecode",
 | 
						|
    "TgaRleEncode",
 | 
						|
    "Unpack",
 | 
						|
    "UnpackYCC",
 | 
						|
    "UnsharpMask",
 | 
						|
    "XbmDecode",
 | 
						|
    "XbmEncode",
 | 
						|
    "ZipDecode",
 | 
						|
    "ZipEncode",
 | 
						|
    "TiffDecode",
 | 
						|
    "Jpeg2KDecode",
 | 
						|
    "Jpeg2KEncode",
 | 
						|
    "BoxBlur",
 | 
						|
    "QuantPngQuant",
 | 
						|
    "codec_fd",
 | 
						|
)
 | 
						|
 | 
						|
DEBUG = False
 | 
						|
 | 
						|
 | 
						|
class DependencyException(Exception):
 | 
						|
    pass
 | 
						|
 | 
						|
 | 
						|
class RequiredDependencyException(Exception):
 | 
						|
    pass
 | 
						|
 | 
						|
 | 
						|
PLATFORM_MINGW = os.name == "nt" and "GCC" in sys.version
 | 
						|
 | 
						|
 | 
						|
def _dbg(s, tp=None):
 | 
						|
    if DEBUG:
 | 
						|
        if tp:
 | 
						|
            print(s % tp)
 | 
						|
            return
 | 
						|
        print(s)
 | 
						|
 | 
						|
 | 
						|
def _find_library_dirs_ldconfig():
 | 
						|
    # Based on ctypes.util from Python 2
 | 
						|
 | 
						|
    ldconfig = "ldconfig" if shutil.which("ldconfig") else "/sbin/ldconfig"
 | 
						|
    if sys.platform.startswith("linux") or sys.platform.startswith("gnu"):
 | 
						|
        if struct.calcsize("l") == 4:
 | 
						|
            machine = os.uname()[4] + "-32"
 | 
						|
        else:
 | 
						|
            machine = os.uname()[4] + "-64"
 | 
						|
        mach_map = {
 | 
						|
            "x86_64-64": "libc6,x86-64",
 | 
						|
            "ppc64-64": "libc6,64bit",
 | 
						|
            "sparc64-64": "libc6,64bit",
 | 
						|
            "s390x-64": "libc6,64bit",
 | 
						|
            "ia64-64": "libc6,IA-64",
 | 
						|
        }
 | 
						|
        abi_type = mach_map.get(machine, "libc6")
 | 
						|
 | 
						|
        # Assuming GLIBC's ldconfig (with option -p)
 | 
						|
        # Alpine Linux uses musl that can't print cache
 | 
						|
        args = [ldconfig, "-p"]
 | 
						|
        expr = rf".*\({abi_type}.*\) => (.*)"
 | 
						|
        env = dict(os.environ)
 | 
						|
        env["LC_ALL"] = "C"
 | 
						|
        env["LANG"] = "C"
 | 
						|
 | 
						|
    elif sys.platform.startswith("freebsd"):
 | 
						|
        args = [ldconfig, "-r"]
 | 
						|
        expr = r".* => (.*)"
 | 
						|
        env = {}
 | 
						|
 | 
						|
    try:
 | 
						|
        p = subprocess.Popen(
 | 
						|
            args, stderr=subprocess.DEVNULL, stdout=subprocess.PIPE, env=env
 | 
						|
        )
 | 
						|
    except OSError:  # E.g. command not found
 | 
						|
        return []
 | 
						|
    [data, _] = p.communicate()
 | 
						|
    if isinstance(data, bytes):
 | 
						|
        data = data.decode("latin1")
 | 
						|
 | 
						|
    dirs = []
 | 
						|
    for dll in re.findall(expr, data):
 | 
						|
        dir = os.path.dirname(dll)
 | 
						|
        if dir not in dirs:
 | 
						|
            dirs.append(dir)
 | 
						|
    return dirs
 | 
						|
 | 
						|
 | 
						|
def _add_directory(path, subdir, where=None):
 | 
						|
    if subdir is None:
 | 
						|
        return
 | 
						|
    subdir = os.path.realpath(subdir)
 | 
						|
    if os.path.isdir(subdir) and subdir not in path:
 | 
						|
        if where is None:
 | 
						|
            _dbg("Appending path %s", subdir)
 | 
						|
            path.append(subdir)
 | 
						|
        else:
 | 
						|
            _dbg("Inserting path %s", subdir)
 | 
						|
            path.insert(where, subdir)
 | 
						|
    elif subdir in path and where is not None:
 | 
						|
        path.remove(subdir)
 | 
						|
        path.insert(where, subdir)
 | 
						|
 | 
						|
 | 
						|
def _find_include_file(self, include):
 | 
						|
    for directory in self.compiler.include_dirs:
 | 
						|
        _dbg("Checking for include file %s in %s", (include, directory))
 | 
						|
        if os.path.isfile(os.path.join(directory, include)):
 | 
						|
            _dbg("Found %s", include)
 | 
						|
            return 1
 | 
						|
    return 0
 | 
						|
 | 
						|
 | 
						|
def _find_library_file(self, library):
 | 
						|
    ret = self.compiler.find_library_file(self.compiler.library_dirs, library)
 | 
						|
    if ret:
 | 
						|
        _dbg("Found library %s at %s", (library, ret))
 | 
						|
    else:
 | 
						|
        _dbg("Couldn't find library %s in %s", (library, self.compiler.library_dirs))
 | 
						|
    return ret
 | 
						|
 | 
						|
 | 
						|
def _find_include_dir(self, dirname, include):
 | 
						|
    for directory in self.compiler.include_dirs:
 | 
						|
        _dbg("Checking for include file %s in %s", (include, directory))
 | 
						|
        if os.path.isfile(os.path.join(directory, include)):
 | 
						|
            _dbg("Found %s in %s", (include, directory))
 | 
						|
            return True
 | 
						|
        subdir = os.path.join(directory, dirname)
 | 
						|
        _dbg("Checking for include file %s in %s", (include, subdir))
 | 
						|
        if os.path.isfile(os.path.join(subdir, include)):
 | 
						|
            _dbg("Found %s in %s", (include, subdir))
 | 
						|
            return subdir
 | 
						|
 | 
						|
 | 
						|
def _cmd_exists(cmd: str) -> bool:
 | 
						|
    if "PATH" not in os.environ:
 | 
						|
        return False
 | 
						|
    return any(
 | 
						|
        os.access(os.path.join(path, cmd), os.X_OK)
 | 
						|
        for path in os.environ["PATH"].split(os.pathsep)
 | 
						|
    )
 | 
						|
 | 
						|
 | 
						|
def _pkg_config(name):
 | 
						|
    command = os.environ.get("PKG_CONFIG", "pkg-config")
 | 
						|
    for keep_system in (True, False):
 | 
						|
        try:
 | 
						|
            command_libs = [command, "--libs-only-L", name]
 | 
						|
            command_cflags = [command, "--cflags-only-I", name]
 | 
						|
            stderr = None
 | 
						|
            if keep_system:
 | 
						|
                command_libs.append("--keep-system-libs")
 | 
						|
                command_cflags.append("--keep-system-cflags")
 | 
						|
                stderr = subprocess.DEVNULL
 | 
						|
            if not DEBUG:
 | 
						|
                command_libs.append("--silence-errors")
 | 
						|
                command_cflags.append("--silence-errors")
 | 
						|
            libs = re.split(
 | 
						|
                r"(^|\s+)-L",
 | 
						|
                subprocess.check_output(command_libs, stderr=stderr)
 | 
						|
                .decode("utf8")
 | 
						|
                .strip(),
 | 
						|
            )[::2][1:]
 | 
						|
            cflags = re.split(
 | 
						|
                r"(^|\s+)-I",
 | 
						|
                subprocess.check_output(command_cflags).decode("utf8").strip(),
 | 
						|
            )[::2][1:]
 | 
						|
            return libs, cflags
 | 
						|
        except Exception:
 | 
						|
            pass
 | 
						|
 | 
						|
 | 
						|
class pil_build_ext(build_ext):
 | 
						|
    class feature:
 | 
						|
        features = [
 | 
						|
            "zlib",
 | 
						|
            "jpeg",
 | 
						|
            "jxl",
 | 
						|
            "tiff",
 | 
						|
            "freetype",
 | 
						|
            "raqm",
 | 
						|
            "lcms",
 | 
						|
            "webp",
 | 
						|
            "webpmux",
 | 
						|
            "jpeg2000",
 | 
						|
            "imagequant",
 | 
						|
            "xcb",
 | 
						|
        ]
 | 
						|
 | 
						|
        required = {"jpeg", "zlib"}
 | 
						|
        vendor = set()
 | 
						|
 | 
						|
        def __init__(self):
 | 
						|
            for f in self.features:
 | 
						|
                setattr(self, f, None)
 | 
						|
 | 
						|
        def require(self, feat):
 | 
						|
            return feat in self.required
 | 
						|
 | 
						|
        def want(self, feat):
 | 
						|
            return getattr(self, feat) is None
 | 
						|
 | 
						|
        def want_vendor(self, feat):
 | 
						|
            return feat in self.vendor
 | 
						|
 | 
						|
        def __iter__(self):
 | 
						|
            yield from self.features
 | 
						|
 | 
						|
    feature = feature()
 | 
						|
 | 
						|
    user_options = (
 | 
						|
        build_ext.user_options
 | 
						|
        + [(f"disable-{x}", None, f"Disable support for {x}") for x in feature]
 | 
						|
        + [(f"enable-{x}", None, f"Enable support for {x}") for x in feature]
 | 
						|
        + [
 | 
						|
            (f"vendor-{x}", None, f"Use vendored version of {x}")
 | 
						|
            for x in ("raqm", "fribidi")
 | 
						|
        ]
 | 
						|
        + [
 | 
						|
            ("disable-platform-guessing", None, "Disable platform guessing on Linux"),
 | 
						|
            ("debug", None, "Debug logging"),
 | 
						|
        ]
 | 
						|
        + [("add-imaging-libs=", None, "Add libs to _imaging build")]
 | 
						|
    )
 | 
						|
 | 
						|
    def initialize_options(self):
 | 
						|
        self.disable_platform_guessing = None
 | 
						|
        self.add_imaging_libs = ""
 | 
						|
        build_ext.initialize_options(self)
 | 
						|
        for x in self.feature:
 | 
						|
            setattr(self, f"disable_{x}", None)
 | 
						|
            setattr(self, f"enable_{x}", None)
 | 
						|
        for x in ("raqm", "fribidi"):
 | 
						|
            setattr(self, f"vendor_{x}", None)
 | 
						|
 | 
						|
    def finalize_options(self):
 | 
						|
        build_ext.finalize_options(self)
 | 
						|
        if self.debug:
 | 
						|
            global DEBUG
 | 
						|
            DEBUG = True
 | 
						|
        if not self.parallel:
 | 
						|
            # If --parallel (or -j) wasn't specified, we want to reproduce the same
 | 
						|
            # behavior as before, that is, auto-detect the number of jobs.
 | 
						|
            try:
 | 
						|
                self.parallel = int(
 | 
						|
                    os.environ.get("MAX_CONCURRENCY", min(4, os.cpu_count()))
 | 
						|
                )
 | 
						|
            except TypeError:
 | 
						|
                self.parallel = None
 | 
						|
        for x in self.feature:
 | 
						|
            if getattr(self, f"disable_{x}"):
 | 
						|
                setattr(self.feature, x, False)
 | 
						|
                self.feature.required.discard(x)
 | 
						|
                _dbg("Disabling %s", x)
 | 
						|
                if getattr(self, f"enable_{x}"):
 | 
						|
                    msg = f"Conflicting options: --enable-{x} and --disable-{x}"
 | 
						|
                    raise ValueError(msg)
 | 
						|
                if x == "freetype":
 | 
						|
                    _dbg("--disable-freetype implies --disable-raqm")
 | 
						|
                    if getattr(self, "enable_raqm"):
 | 
						|
                        msg = (
 | 
						|
                            "Conflicting options: --enable-raqm and --disable-freetype"
 | 
						|
                        )
 | 
						|
                        raise ValueError(msg)
 | 
						|
                    setattr(self, "disable_raqm", True)
 | 
						|
            if getattr(self, f"enable_{x}"):
 | 
						|
                _dbg("Requiring %s", x)
 | 
						|
                self.feature.required.add(x)
 | 
						|
                if x == "raqm":
 | 
						|
                    _dbg("--enable-raqm implies --enable-freetype")
 | 
						|
                    self.feature.required.add("freetype")
 | 
						|
        for x in ("raqm", "fribidi"):
 | 
						|
            if getattr(self, f"vendor_{x}"):
 | 
						|
                if getattr(self, "disable_raqm"):
 | 
						|
                    msg = f"Conflicting options: --vendor-{x} and --disable-raqm"
 | 
						|
                    raise ValueError(msg)
 | 
						|
                if x == "fribidi" and not getattr(self, "vendor_raqm"):
 | 
						|
                    msg = f"Conflicting options: --vendor-{x} and not --vendor-raqm"
 | 
						|
                    raise ValueError(msg)
 | 
						|
                _dbg("Using vendored version of %s", x)
 | 
						|
                self.feature.vendor.add(x)
 | 
						|
 | 
						|
    def _update_extension(self, name, libraries, define_macros=None, sources=None):
 | 
						|
        for extension in self.extensions:
 | 
						|
            if extension.name == name:
 | 
						|
                extension.libraries += libraries
 | 
						|
                if define_macros is not None:
 | 
						|
                    extension.define_macros += define_macros
 | 
						|
                if sources is not None:
 | 
						|
                    extension.sources += sources
 | 
						|
                if FUZZING_BUILD:
 | 
						|
                    extension.language = "c++"
 | 
						|
                    extension.extra_link_args = ["--stdlib=libc++"]
 | 
						|
                break
 | 
						|
 | 
						|
    def _remove_extension(self, name):
 | 
						|
        for extension in self.extensions:
 | 
						|
            if extension.name == name:
 | 
						|
                self.extensions.remove(extension)
 | 
						|
                break
 | 
						|
 | 
						|
    def get_macos_sdk_path(self):
 | 
						|
        try:
 | 
						|
            sdk_path = (
 | 
						|
                subprocess.check_output(["xcrun", "--show-sdk-path"])
 | 
						|
                .strip()
 | 
						|
                .decode("latin1")
 | 
						|
            )
 | 
						|
        except Exception:
 | 
						|
            sdk_path = None
 | 
						|
        if (
 | 
						|
            not sdk_path
 | 
						|
            or sdk_path == "/Applications/Xcode.app/Contents/Developer"
 | 
						|
            "/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk"
 | 
						|
        ):
 | 
						|
            commandlinetools_sdk_path = (
 | 
						|
                "/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk"
 | 
						|
            )
 | 
						|
            if os.path.exists(commandlinetools_sdk_path):
 | 
						|
                sdk_path = commandlinetools_sdk_path
 | 
						|
        return sdk_path
 | 
						|
 | 
						|
    def build_extensions(self):
 | 
						|
        library_dirs = []
 | 
						|
        include_dirs = []
 | 
						|
 | 
						|
        pkg_config = None
 | 
						|
        if _cmd_exists(os.environ.get("PKG_CONFIG", "pkg-config")):
 | 
						|
            pkg_config = _pkg_config
 | 
						|
 | 
						|
        #
 | 
						|
        # add configured kits
 | 
						|
        for root_name, lib_name in {
 | 
						|
            "JPEG_ROOT": "libjpeg",
 | 
						|
            "JPEG2K_ROOT": "libopenjp2",
 | 
						|
            "TIFF_ROOT": ("libtiff-5", "libtiff-4"),
 | 
						|
            "ZLIB_ROOT": "zlib",
 | 
						|
            "FREETYPE_ROOT": "freetype2",
 | 
						|
            "HARFBUZZ_ROOT": "harfbuzz",
 | 
						|
            "FRIBIDI_ROOT": "fribidi",
 | 
						|
            "LCMS_ROOT": "lcms2",
 | 
						|
            "IMAGEQUANT_ROOT": "libimagequant",
 | 
						|
        }.items():
 | 
						|
            root = globals()[root_name]
 | 
						|
 | 
						|
            if root is None and root_name in os.environ:
 | 
						|
                prefix = os.environ[root_name]
 | 
						|
                root = (os.path.join(prefix, "lib"), os.path.join(prefix, "include"))
 | 
						|
 | 
						|
            if root is None and pkg_config:
 | 
						|
                if isinstance(lib_name, tuple):
 | 
						|
                    for lib_name2 in lib_name:
 | 
						|
                        _dbg(f"Looking for `{lib_name2}` using pkg-config.")
 | 
						|
                        root = pkg_config(lib_name2)
 | 
						|
                        if root:
 | 
						|
                            break
 | 
						|
                else:
 | 
						|
                    _dbg(f"Looking for `{lib_name}` using pkg-config.")
 | 
						|
                    root = pkg_config(lib_name)
 | 
						|
 | 
						|
            if isinstance(root, tuple):
 | 
						|
                lib_root, include_root = root
 | 
						|
            else:
 | 
						|
                lib_root = include_root = root
 | 
						|
 | 
						|
            if lib_root is not None:
 | 
						|
                if not isinstance(lib_root, (tuple, list)):
 | 
						|
                    lib_root = (lib_root,)
 | 
						|
                for lib_dir in lib_root:
 | 
						|
                    _add_directory(library_dirs, lib_dir)
 | 
						|
            if include_root is not None:
 | 
						|
                if not isinstance(include_root, (tuple, list)):
 | 
						|
                    include_root = (include_root,)
 | 
						|
                for include_dir in include_root:
 | 
						|
                    _add_directory(include_dirs, include_dir)
 | 
						|
 | 
						|
        # respect CFLAGS/CPPFLAGS/LDFLAGS
 | 
						|
        for k in ("CFLAGS", "CPPFLAGS", "LDFLAGS"):
 | 
						|
            if k in os.environ:
 | 
						|
                for match in re.finditer(r"-I([^\s]+)", os.environ[k]):
 | 
						|
                    _add_directory(include_dirs, match.group(1))
 | 
						|
                for match in re.finditer(r"-L([^\s]+)", os.environ[k]):
 | 
						|
                    _add_directory(library_dirs, match.group(1))
 | 
						|
 | 
						|
        # include, rpath, if set as environment variables:
 | 
						|
        for k in ("C_INCLUDE_PATH", "CPATH", "INCLUDE"):
 | 
						|
            if k in os.environ:
 | 
						|
                for d in os.environ[k].split(os.path.pathsep):
 | 
						|
                    _add_directory(include_dirs, d)
 | 
						|
 | 
						|
        for k in ("LD_RUN_PATH", "LIBRARY_PATH", "LIB"):
 | 
						|
            if k in os.environ:
 | 
						|
                for d in os.environ[k].split(os.path.pathsep):
 | 
						|
                    _add_directory(library_dirs, d)
 | 
						|
 | 
						|
        _add_directory(library_dirs, os.path.join(sys.prefix, "lib"))
 | 
						|
        _add_directory(include_dirs, os.path.join(sys.prefix, "include"))
 | 
						|
 | 
						|
        #
 | 
						|
        # add platform directories
 | 
						|
 | 
						|
        if self.disable_platform_guessing:
 | 
						|
            pass
 | 
						|
 | 
						|
        elif sys.platform == "cygwin":
 | 
						|
            # pythonX.Y.dll.a is in the /usr/lib/pythonX.Y/config directory
 | 
						|
            self.compiler.shared_lib_extension = ".dll.a"
 | 
						|
            _add_directory(
 | 
						|
                library_dirs,
 | 
						|
                os.path.join(
 | 
						|
                    "/usr/lib", "python{}.{}".format(*sys.version_info), "config"
 | 
						|
                ),
 | 
						|
            )
 | 
						|
 | 
						|
        elif sys.platform == "darwin":
 | 
						|
            # attempt to make sure we pick freetype2 over other versions
 | 
						|
            _add_directory(include_dirs, "/sw/include/freetype2")
 | 
						|
            _add_directory(include_dirs, "/sw/lib/freetype2/include")
 | 
						|
            # fink installation directories
 | 
						|
            _add_directory(library_dirs, "/sw/lib")
 | 
						|
            _add_directory(include_dirs, "/sw/include")
 | 
						|
            # darwin ports installation directories
 | 
						|
            _add_directory(library_dirs, "/opt/local/lib")
 | 
						|
            _add_directory(include_dirs, "/opt/local/include")
 | 
						|
 | 
						|
            # if Homebrew is installed, use its lib and include directories
 | 
						|
            try:
 | 
						|
                prefix = (
 | 
						|
                    subprocess.check_output(["brew", "--prefix"])
 | 
						|
                    .strip()
 | 
						|
                    .decode("latin1")
 | 
						|
                )
 | 
						|
            except Exception:
 | 
						|
                # Homebrew not installed
 | 
						|
                prefix = None
 | 
						|
 | 
						|
            ft_prefix = None
 | 
						|
 | 
						|
            if prefix:
 | 
						|
                # add Homebrew's include and lib directories
 | 
						|
                _add_directory(library_dirs, os.path.join(prefix, "lib"))
 | 
						|
                _add_directory(include_dirs, os.path.join(prefix, "include"))
 | 
						|
                _add_directory(
 | 
						|
                    include_dirs, os.path.join(prefix, "opt", "zlib", "include")
 | 
						|
                )
 | 
						|
                ft_prefix = os.path.join(prefix, "opt", "freetype")
 | 
						|
 | 
						|
            if ft_prefix and os.path.isdir(ft_prefix):
 | 
						|
                # freetype might not be linked into Homebrew's prefix
 | 
						|
                _add_directory(library_dirs, os.path.join(ft_prefix, "lib"))
 | 
						|
                _add_directory(include_dirs, os.path.join(ft_prefix, "include"))
 | 
						|
            else:
 | 
						|
                # fall back to freetype from XQuartz if
 | 
						|
                # Homebrew's freetype is missing
 | 
						|
                _add_directory(library_dirs, "/usr/X11/lib")
 | 
						|
                _add_directory(include_dirs, "/usr/X11/include")
 | 
						|
 | 
						|
            sdk_path = self.get_macos_sdk_path()
 | 
						|
            if sdk_path:
 | 
						|
                _add_directory(library_dirs, os.path.join(sdk_path, "usr", "lib"))
 | 
						|
                _add_directory(include_dirs, os.path.join(sdk_path, "usr", "include"))
 | 
						|
        elif (
 | 
						|
            sys.platform.startswith("linux")
 | 
						|
            or sys.platform.startswith("gnu")
 | 
						|
            or sys.platform.startswith("freebsd")
 | 
						|
        ):
 | 
						|
            for dirname in _find_library_dirs_ldconfig():
 | 
						|
                _add_directory(library_dirs, dirname)
 | 
						|
            if sys.platform.startswith("linux") and os.environ.get("ANDROID_ROOT"):
 | 
						|
                # termux support for android.
 | 
						|
                # system libraries (zlib) are installed in /system/lib
 | 
						|
                # headers are at $PREFIX/include
 | 
						|
                # user libs are at $PREFIX/lib
 | 
						|
                _add_directory(
 | 
						|
                    library_dirs,
 | 
						|
                    os.path.join(
 | 
						|
                        os.environ["ANDROID_ROOT"],
 | 
						|
                        "lib" if struct.calcsize("l") == 4 else "lib64",
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
 | 
						|
        elif sys.platform.startswith("netbsd"):
 | 
						|
            _add_directory(library_dirs, "/usr/pkg/lib")
 | 
						|
            _add_directory(include_dirs, "/usr/pkg/include")
 | 
						|
 | 
						|
        elif sys.platform.startswith("sunos5"):
 | 
						|
            _add_directory(library_dirs, "/opt/local/lib")
 | 
						|
            _add_directory(include_dirs, "/opt/local/include")
 | 
						|
 | 
						|
        # FIXME: check /opt/stuff directories here?
 | 
						|
 | 
						|
        # standard locations
 | 
						|
        if not self.disable_platform_guessing:
 | 
						|
            _add_directory(library_dirs, "/usr/local/lib")
 | 
						|
            _add_directory(include_dirs, "/usr/local/include")
 | 
						|
 | 
						|
            _add_directory(library_dirs, "/usr/lib")
 | 
						|
            _add_directory(include_dirs, "/usr/include")
 | 
						|
            # alpine, at least
 | 
						|
            _add_directory(library_dirs, "/lib")
 | 
						|
 | 
						|
        if sys.platform == "win32":
 | 
						|
            # on Windows, look for the OpenJPEG libraries in the location that
 | 
						|
            # the official installer puts them
 | 
						|
            program_files = os.environ.get("ProgramFiles", "")
 | 
						|
            best_version = (0, 0)
 | 
						|
            best_path = None
 | 
						|
            for name in os.listdir(program_files):
 | 
						|
                if name.startswith("OpenJPEG "):
 | 
						|
                    version = tuple(int(x) for x in name[9:].strip().split("."))
 | 
						|
                    if version > best_version:
 | 
						|
                        best_version = version
 | 
						|
                        best_path = os.path.join(program_files, name)
 | 
						|
 | 
						|
            if best_path:
 | 
						|
                _dbg("Adding %s to search list", best_path)
 | 
						|
                _add_directory(library_dirs, os.path.join(best_path, "lib"))
 | 
						|
                _add_directory(include_dirs, os.path.join(best_path, "include"))
 | 
						|
 | 
						|
        #
 | 
						|
        # insert new dirs *before* default libs, to avoid conflicts
 | 
						|
        # between Python PYD stub libs and real libraries
 | 
						|
 | 
						|
        self.compiler.library_dirs = library_dirs + self.compiler.library_dirs
 | 
						|
        self.compiler.include_dirs = include_dirs + self.compiler.include_dirs
 | 
						|
 | 
						|
        #
 | 
						|
        # look for available libraries
 | 
						|
 | 
						|
        feature = self.feature
 | 
						|
 | 
						|
        if feature.want("zlib"):
 | 
						|
            _dbg("Looking for zlib")
 | 
						|
            if _find_include_file(self, "zlib.h"):
 | 
						|
                if _find_library_file(self, "z"):
 | 
						|
                    feature.zlib = "z"
 | 
						|
                elif sys.platform == "win32" and _find_library_file(self, "zlib"):
 | 
						|
                    feature.zlib = "zlib"  # alternative name
 | 
						|
 | 
						|
        if feature.want("jpeg"):
 | 
						|
            _dbg("Looking for jpeg")
 | 
						|
            if _find_include_file(self, "jpeglib.h"):
 | 
						|
                if _find_library_file(self, "jpeg"):
 | 
						|
                    feature.jpeg = "jpeg"
 | 
						|
                elif sys.platform == "win32" and _find_library_file(self, "libjpeg"):
 | 
						|
                    feature.jpeg = "libjpeg"  # alternative name
 | 
						|
 | 
						|
        feature.openjpeg_version = None
 | 
						|
        if feature.want("jpeg2000"):
 | 
						|
            _dbg("Looking for jpeg2000")
 | 
						|
            best_version = None
 | 
						|
            best_path = None
 | 
						|
 | 
						|
            # Find the best version
 | 
						|
            for directory in self.compiler.include_dirs:
 | 
						|
                _dbg("Checking for openjpeg-#.# in %s", directory)
 | 
						|
                try:
 | 
						|
                    listdir = os.listdir(directory)
 | 
						|
                except Exception:
 | 
						|
                    # OSError, FileNotFoundError
 | 
						|
                    continue
 | 
						|
                for name in listdir:
 | 
						|
                    if name.startswith("openjpeg-") and os.path.isfile(
 | 
						|
                        os.path.join(directory, name, "openjpeg.h")
 | 
						|
                    ):
 | 
						|
                        _dbg("Found openjpeg.h in %s/%s", (directory, name))
 | 
						|
                        version = tuple(int(x) for x in name[9:].split("."))
 | 
						|
                        if best_version is None or version > best_version:
 | 
						|
                            best_version = version
 | 
						|
                            best_path = os.path.join(directory, name)
 | 
						|
                            _dbg(
 | 
						|
                                "Best openjpeg version %s so far in %s",
 | 
						|
                                (best_version, best_path),
 | 
						|
                            )
 | 
						|
 | 
						|
            if best_version and _find_library_file(self, "openjp2"):
 | 
						|
                # Add the directory to the include path so we can include
 | 
						|
                # <openjpeg.h> rather than having to cope with the versioned
 | 
						|
                # include path
 | 
						|
                _add_directory(self.compiler.include_dirs, best_path, 0)
 | 
						|
                feature.jpeg2000 = "openjp2"
 | 
						|
                feature.openjpeg_version = ".".join(str(x) for x in best_version)
 | 
						|
 | 
						|
        if feature.want("jxl"):
 | 
						|
            _dbg("Looking for jxl")
 | 
						|
            if _find_include_file(self, "jxl/encode.h") and _find_include_file(
 | 
						|
                self, "jxl/decode.h"
 | 
						|
            ):
 | 
						|
                if _find_library_file(self, "jxl"):
 | 
						|
                    feature.jxl = "jxl jxl_threads"
 | 
						|
 | 
						|
        if feature.want("imagequant"):
 | 
						|
            _dbg("Looking for imagequant")
 | 
						|
            if _find_include_file(self, "libimagequant.h"):
 | 
						|
                if _find_library_file(self, "imagequant"):
 | 
						|
                    feature.imagequant = "imagequant"
 | 
						|
                elif _find_library_file(self, "libimagequant"):
 | 
						|
                    feature.imagequant = "libimagequant"
 | 
						|
 | 
						|
        if feature.want("tiff"):
 | 
						|
            _dbg("Looking for tiff")
 | 
						|
            if _find_include_file(self, "tiff.h"):
 | 
						|
                if _find_library_file(self, "tiff"):
 | 
						|
                    feature.tiff = "tiff"
 | 
						|
                if sys.platform in ["win32", "darwin"] and _find_library_file(
 | 
						|
                    self, "libtiff"
 | 
						|
                ):
 | 
						|
                    feature.tiff = "libtiff"
 | 
						|
 | 
						|
        if feature.want("freetype"):
 | 
						|
            _dbg("Looking for freetype")
 | 
						|
            if _find_library_file(self, "freetype"):
 | 
						|
                # look for freetype2 include files
 | 
						|
                freetype_version = 0
 | 
						|
                for subdir in self.compiler.include_dirs:
 | 
						|
                    _dbg("Checking for include file %s in %s", ("ft2build.h", subdir))
 | 
						|
                    if os.path.isfile(os.path.join(subdir, "ft2build.h")):
 | 
						|
                        _dbg("Found %s in %s", ("ft2build.h", subdir))
 | 
						|
                        freetype_version = 21
 | 
						|
                        subdir = os.path.join(subdir, "freetype2")
 | 
						|
                        break
 | 
						|
                    subdir = os.path.join(subdir, "freetype2")
 | 
						|
                    _dbg("Checking for include file %s in %s", ("ft2build.h", subdir))
 | 
						|
                    if os.path.isfile(os.path.join(subdir, "ft2build.h")):
 | 
						|
                        _dbg("Found %s in %s", ("ft2build.h", subdir))
 | 
						|
                        freetype_version = 21
 | 
						|
                        break
 | 
						|
                if freetype_version:
 | 
						|
                    feature.freetype = "freetype"
 | 
						|
                    if subdir:
 | 
						|
                        _add_directory(self.compiler.include_dirs, subdir, 0)
 | 
						|
 | 
						|
        if feature.freetype and feature.want("raqm"):
 | 
						|
            if not feature.want_vendor("raqm"):  # want system Raqm
 | 
						|
                _dbg("Looking for Raqm")
 | 
						|
                if _find_include_file(self, "raqm.h"):
 | 
						|
                    if _find_library_file(self, "raqm"):
 | 
						|
                        feature.raqm = "raqm"
 | 
						|
                    elif _find_library_file(self, "libraqm"):
 | 
						|
                        feature.raqm = "libraqm"
 | 
						|
            else:  # want to build Raqm from src/thirdparty
 | 
						|
                _dbg("Looking for HarfBuzz")
 | 
						|
                feature.harfbuzz = None
 | 
						|
                hb_dir = _find_include_dir(self, "harfbuzz", "hb.h")
 | 
						|
                if hb_dir:
 | 
						|
                    if isinstance(hb_dir, str):
 | 
						|
                        _add_directory(self.compiler.include_dirs, hb_dir, 0)
 | 
						|
                    if _find_library_file(self, "harfbuzz"):
 | 
						|
                        feature.harfbuzz = "harfbuzz"
 | 
						|
                if feature.harfbuzz:
 | 
						|
                    if not feature.want_vendor("fribidi"):  # want system FriBiDi
 | 
						|
                        _dbg("Looking for FriBiDi")
 | 
						|
                        feature.fribidi = None
 | 
						|
                        fribidi_dir = _find_include_dir(self, "fribidi", "fribidi.h")
 | 
						|
                        if fribidi_dir:
 | 
						|
                            if isinstance(fribidi_dir, str):
 | 
						|
                                _add_directory(
 | 
						|
                                    self.compiler.include_dirs, fribidi_dir, 0
 | 
						|
                                )
 | 
						|
                            if _find_library_file(self, "fribidi"):
 | 
						|
                                feature.fribidi = "fribidi"
 | 
						|
                                feature.raqm = True
 | 
						|
                    else:  # want to build FriBiDi shim from src/thirdparty
 | 
						|
                        feature.raqm = True
 | 
						|
 | 
						|
        if feature.want("lcms"):
 | 
						|
            _dbg("Looking for lcms")
 | 
						|
            if _find_include_file(self, "lcms2.h"):
 | 
						|
                if _find_library_file(self, "lcms2"):
 | 
						|
                    feature.lcms = "lcms2"
 | 
						|
                elif _find_library_file(self, "lcms2_static"):
 | 
						|
                    # alternate Windows name.
 | 
						|
                    feature.lcms = "lcms2_static"
 | 
						|
 | 
						|
        if feature.jxl:
 | 
						|
            # jxl and jxl_threads are required
 | 
						|
            libs = feature.jxl.split()
 | 
						|
            defs = []
 | 
						|
 | 
						|
            self._update_extension("PIL._jxl", libs, defs)
 | 
						|
        else:
 | 
						|
            self._remove_extension("PIL._jxl")
 | 
						|
 | 
						|
        if feature.want("webp"):
 | 
						|
            _dbg("Looking for webp")
 | 
						|
            if _find_include_file(self, "webp/encode.h") and _find_include_file(
 | 
						|
                self, "webp/decode.h"
 | 
						|
            ):
 | 
						|
                # In Google's precompiled zip it is call "libwebp":
 | 
						|
                if _find_library_file(self, "webp"):
 | 
						|
                    feature.webp = "webp"
 | 
						|
                elif _find_library_file(self, "libwebp"):
 | 
						|
                    feature.webp = "libwebp"
 | 
						|
 | 
						|
        if feature.want("webpmux"):
 | 
						|
            _dbg("Looking for webpmux")
 | 
						|
            if _find_include_file(self, "webp/mux.h") and _find_include_file(
 | 
						|
                self, "webp/demux.h"
 | 
						|
            ):
 | 
						|
                if _find_library_file(self, "webpmux") and _find_library_file(
 | 
						|
                    self, "webpdemux"
 | 
						|
                ):
 | 
						|
                    feature.webpmux = "webpmux"
 | 
						|
                if _find_library_file(self, "libwebpmux") and _find_library_file(
 | 
						|
                    self, "libwebpdemux"
 | 
						|
                ):
 | 
						|
                    feature.webpmux = "libwebpmux"
 | 
						|
 | 
						|
        if feature.want("xcb"):
 | 
						|
            _dbg("Looking for xcb")
 | 
						|
            if _find_include_file(self, "xcb/xcb.h"):
 | 
						|
                if _find_library_file(self, "xcb"):
 | 
						|
                    feature.xcb = "xcb"
 | 
						|
 | 
						|
        for f in feature:
 | 
						|
            if not getattr(feature, f) and feature.require(f):
 | 
						|
                if f in ("jpeg", "zlib"):
 | 
						|
                    raise RequiredDependencyException(f)
 | 
						|
                raise DependencyException(f)
 | 
						|
 | 
						|
        #
 | 
						|
        # core library
 | 
						|
 | 
						|
        libs = self.add_imaging_libs.split()
 | 
						|
        defs = []
 | 
						|
        if feature.tiff:
 | 
						|
            libs.append(feature.tiff)
 | 
						|
            defs.append(("HAVE_LIBTIFF", None))
 | 
						|
            if sys.platform == "win32":
 | 
						|
                # This define needs to be defined if-and-only-if it was defined
 | 
						|
                # when compiling LibTIFF. LibTIFF doesn't expose it in `tiffconf.h`,
 | 
						|
                # so we have to guess; by default it is defined in all Windows builds.
 | 
						|
                # See #4237, #5243, #5359 for more information.
 | 
						|
                defs.append(("USE_WIN32_FILEIO", None))
 | 
						|
        if feature.jpeg:
 | 
						|
            libs.append(feature.jpeg)
 | 
						|
            defs.append(("HAVE_LIBJPEG", None))
 | 
						|
        if feature.jpeg2000:
 | 
						|
            libs.append(feature.jpeg2000)
 | 
						|
            defs.append(("HAVE_OPENJPEG", None))
 | 
						|
            if sys.platform == "win32" and not PLATFORM_MINGW:
 | 
						|
                defs.append(("OPJ_STATIC", None))
 | 
						|
        if feature.zlib:
 | 
						|
            libs.append(feature.zlib)
 | 
						|
            defs.append(("HAVE_LIBZ", None))
 | 
						|
        if feature.imagequant:
 | 
						|
            libs.append(feature.imagequant)
 | 
						|
            defs.append(("HAVE_LIBIMAGEQUANT", None))
 | 
						|
        if feature.xcb:
 | 
						|
            libs.append(feature.xcb)
 | 
						|
            defs.append(("HAVE_XCB", None))
 | 
						|
        if sys.platform == "win32":
 | 
						|
            libs.extend(["kernel32", "user32", "gdi32"])
 | 
						|
        if struct.unpack("h", b"\0\1")[0] == 1:
 | 
						|
            defs.append(("WORDS_BIGENDIAN", None))
 | 
						|
 | 
						|
        defs.append(("PILLOW_VERSION", f'"{PILLOW_VERSION}"'))
 | 
						|
 | 
						|
        self._update_extension("PIL._imaging", libs, defs)
 | 
						|
 | 
						|
        #
 | 
						|
        # additional libraries
 | 
						|
 | 
						|
        if feature.freetype:
 | 
						|
            srcs = []
 | 
						|
            libs = ["freetype"]
 | 
						|
            defs = []
 | 
						|
            if feature.raqm:
 | 
						|
                if not feature.want_vendor("raqm"):  # using system Raqm
 | 
						|
                    defs.append(("HAVE_RAQM", None))
 | 
						|
                    defs.append(("HAVE_RAQM_SYSTEM", None))
 | 
						|
                    libs.append(feature.raqm)
 | 
						|
                else:  # building Raqm from src/thirdparty
 | 
						|
                    defs.append(("HAVE_RAQM", None))
 | 
						|
                    srcs.append("src/thirdparty/raqm/raqm.c")
 | 
						|
                    libs.append(feature.harfbuzz)
 | 
						|
                    if not feature.want_vendor("fribidi"):  # using system FriBiDi
 | 
						|
                        defs.append(("HAVE_FRIBIDI_SYSTEM", None))
 | 
						|
                        libs.append(feature.fribidi)
 | 
						|
                    else:  # building FriBiDi shim from src/thirdparty
 | 
						|
                        srcs.append("src/thirdparty/fribidi-shim/fribidi.c")
 | 
						|
            self._update_extension("PIL._imagingft", libs, defs, srcs)
 | 
						|
 | 
						|
        else:
 | 
						|
            self._remove_extension("PIL._imagingft")
 | 
						|
 | 
						|
        if feature.lcms:
 | 
						|
            extra = []
 | 
						|
            if sys.platform == "win32":
 | 
						|
                extra.extend(["user32", "gdi32"])
 | 
						|
            self._update_extension("PIL._imagingcms", [feature.lcms] + extra)
 | 
						|
        else:
 | 
						|
            self._remove_extension("PIL._imagingcms")
 | 
						|
 | 
						|
        if feature.webp:
 | 
						|
            libs = [feature.webp]
 | 
						|
            defs = []
 | 
						|
 | 
						|
            if feature.webpmux:
 | 
						|
                defs.append(("HAVE_WEBPMUX", None))
 | 
						|
                libs.append(feature.webpmux)
 | 
						|
                libs.append(feature.webpmux.replace("pmux", "pdemux"))
 | 
						|
 | 
						|
            self._update_extension("PIL._webp", libs, defs)
 | 
						|
        else:
 | 
						|
            self._remove_extension("PIL._webp")
 | 
						|
 | 
						|
        tk_libs = ["psapi"] if sys.platform in ("win32", "cygwin") else []
 | 
						|
        self._update_extension("PIL._imagingtk", tk_libs)
 | 
						|
 | 
						|
        build_ext.build_extensions(self)
 | 
						|
 | 
						|
        #
 | 
						|
        # sanity checks
 | 
						|
 | 
						|
        self.summary_report(feature)
 | 
						|
 | 
						|
    def summary_report(self, feature):
 | 
						|
        print("-" * 68)
 | 
						|
        print("PIL SETUP SUMMARY")
 | 
						|
        print("-" * 68)
 | 
						|
        print(f"version      Pillow {PILLOW_VERSION}")
 | 
						|
        v = sys.version.split("[")
 | 
						|
        print(f"platform     {sys.platform} {v[0].strip()}")
 | 
						|
        for v in v[1:]:
 | 
						|
            print(f"             [{v.strip()}")
 | 
						|
        print("-" * 68)
 | 
						|
 | 
						|
        raqm_extra_info = ""
 | 
						|
        if feature.want_vendor("raqm"):
 | 
						|
            raqm_extra_info += "bundled"
 | 
						|
            if feature.want_vendor("fribidi"):
 | 
						|
                raqm_extra_info += ", FriBiDi shim"
 | 
						|
 | 
						|
        options = [
 | 
						|
            (feature.jpeg, "JPEG"),
 | 
						|
            (feature.jpeg2000, "OPENJPEG (JPEG2000)", feature.openjpeg_version),
 | 
						|
            (feature.zlib, "ZLIB (PNG/ZIP)"),
 | 
						|
            (feature.imagequant, "LIBIMAGEQUANT"),
 | 
						|
            (feature.tiff, "LIBTIFF"),
 | 
						|
            (feature.freetype, "FREETYPE2"),
 | 
						|
            (feature.raqm, "RAQM (Text shaping)", raqm_extra_info),
 | 
						|
            (feature.lcms, "LITTLECMS2"),
 | 
						|
            (feature.jxl, "JXL"),
 | 
						|
            (feature.webp, "WEBP"),
 | 
						|
            (feature.webpmux, "WEBPMUX"),
 | 
						|
            (feature.xcb, "XCB (X protocol)"),
 | 
						|
        ]
 | 
						|
 | 
						|
        all = 1
 | 
						|
        for option in options:
 | 
						|
            if option[0]:
 | 
						|
                extra_info = ""
 | 
						|
                if len(option) >= 3 and option[2]:
 | 
						|
                    extra_info = f" ({option[2]})"
 | 
						|
                print(f"--- {option[1]} support available{extra_info}")
 | 
						|
            else:
 | 
						|
                print(f"*** {option[1]} support not available")
 | 
						|
                all = 0
 | 
						|
 | 
						|
        print("-" * 68)
 | 
						|
 | 
						|
        if not all:
 | 
						|
            print("To add a missing option, make sure you have the required")
 | 
						|
            print("library and headers.")
 | 
						|
            print(
 | 
						|
                "See https://pillow.readthedocs.io/en/latest/installation."
 | 
						|
                "html#building-from-source"
 | 
						|
            )
 | 
						|
            print("")
 | 
						|
 | 
						|
        print("To check the build, run the selftest.py script.")
 | 
						|
        print("")
 | 
						|
 | 
						|
 | 
						|
def debug_build():
 | 
						|
    return hasattr(sys, "gettotalrefcount") or FUZZING_BUILD
 | 
						|
 | 
						|
 | 
						|
files = ["src/_imaging.c"]
 | 
						|
for src_file in _IMAGING:
 | 
						|
    files.append("src/" + src_file + ".c")
 | 
						|
for src_file in _LIB_IMAGING:
 | 
						|
    files.append(os.path.join("src/libImaging", src_file + ".c"))
 | 
						|
ext_modules = [
 | 
						|
    Extension("PIL._imaging", files),
 | 
						|
    Extension("PIL._imagingft", ["src/_imagingft.c"]),
 | 
						|
    Extension("PIL._imagingcms", ["src/_imagingcms.c"]),
 | 
						|
    Extension("PIL._jxl", ["src/_jxl.c"]),
 | 
						|
    Extension("PIL._webp", ["src/_webp.c"]),
 | 
						|
    Extension("PIL._imagingtk", ["src/_imagingtk.c", "src/Tk/tkImaging.c"]),
 | 
						|
    Extension("PIL._imagingmath", ["src/_imagingmath.c"]),
 | 
						|
    Extension("PIL._imagingmorph", ["src/_imagingmorph.c"]),
 | 
						|
]
 | 
						|
 | 
						|
try:
 | 
						|
    setup(
 | 
						|
        cmdclass={"build_ext": pil_build_ext},
 | 
						|
        ext_modules=ext_modules,
 | 
						|
        zip_safe=not (debug_build() or PLATFORM_MINGW),
 | 
						|
    )
 | 
						|
except RequiredDependencyException as err:
 | 
						|
    msg = f"""
 | 
						|
 | 
						|
The headers or library files could not be found for {str(err)},
 | 
						|
a required dependency when compiling Pillow from source.
 | 
						|
 | 
						|
Please see the install instructions at:
 | 
						|
   https://pillow.readthedocs.io/en/latest/installation.html
 | 
						|
 | 
						|
"""
 | 
						|
    sys.stderr.write(msg)
 | 
						|
    raise RequiredDependencyException(msg)
 | 
						|
except DependencyException as err:
 | 
						|
    msg = f"""
 | 
						|
 | 
						|
The headers or library files could not be found for {str(err)},
 | 
						|
which was requested by the option flag --enable-{str(err)}
 | 
						|
 | 
						|
"""
 | 
						|
    sys.stderr.write(msg)
 | 
						|
    raise DependencyException(msg)
 |