# # Helper module to build macOS version of various source ports # Copyright (C) 2020-2022 Alexey Lysiuk # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # from .base import * class Bzip2Target(MakeTarget): def __init__(self, name='bzip2'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://sourceware.org/pub/bzip2/bzip2-1.0.8.tar.gz', 'ab5a03176ee106d3f0fa90e381da478ddae405918153cca248e682cd0c4a2269') def detect(self, state: BuildState) -> bool: return state.has_source_file('bzlib.h') def configure(self, state: BuildState): super().configure(state) opts = state.options # Add explicit targets in order to skip testing step that is incompatible with cross-compilation opts['bzip2'] = None opts['bzip2recover'] = None # Copy compiler flags from environment to command line argument, they would be overridden by Makefile otherwise cflags = 'CFLAGS' opts[cflags] = state.environment[cflags] + ' -D_FILE_OFFSET_BITS=64 -O2' def post_build(self, state: BuildState): state.options['PREFIX'] = state.install_path self.install(state, state.options) self.write_pc_file(state, description='bzip2 compression library', version='1.0.8', libs='-lbz2') class FfiTarget(ConfigureMakeStaticDependencyTarget): def __init__(self, name='ffi'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/libffi/libffi/releases/download/v3.4.2/libffi-3.4.2.tar.gz', '540fb721619a6aba3bdeef7d940d8e9e0e6d2c193595bc243241b77ff9e93620') def detect(self, state: BuildState) -> bool: return state.has_source_file('libffi.pc.in') def post_build(self, state: BuildState): super().post_build(state) for header in ('ffi.h', 'ffitarget.h'): self.make_platform_header(state, header) class FlacTarget(CMakeStaticDependencyTarget): def __init__(self, name='flac'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://ftp.osuosl.org/pub/xiph/releases/flac/flac-1.3.4.tar.xz', '8ff0607e75a322dd7cd6ec48f4f225471404ae2730d0ea945127b1355155e737') def configure(self, state: BuildState): opts = state.options opts['CMAKE_EXE_LINKER_FLAGS'] = f'-framework CoreFoundation -L{state.lib_path}' opts['BUILD_CXXLIBS'] = 'NO' opts['BUILD_EXAMPLES'] = 'NO' opts['OGG_PACKAGE'] = 'ogg' super().configure(state) def post_build(self, state: BuildState): super().post_build(state) shutil.copytree(state.install_path / 'share/FLAC/cmake', state.install_path / 'lib/cmake/FLAC') self.keep_module_target(state, 'FLAC::FLAC') class FluidSynthTarget(CMakeStaticDependencyTarget): def __init__(self, name='fluidsynth'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/FluidSynth/fluidsynth/archive/refs/tags/v2.2.7.tar.gz', '460d86d8d687f567dc4780890b72538c7ff6b2082080ef2f9359d41670a309cf') def configure(self, state: BuildState): opts = state.options opts['CMAKE_EXE_LINKER_FLAGS'] = f'-framework Foundation -L{state.lib_path}' opts['DEFAULT_SOUNDFONT'] = 'default.sf2' opts['LIB_SUFFIX'] = None opts['enable-framework'] = 'NO' opts['enable-readline'] = 'NO' opts['enable-sdl2'] = 'NO' super().configure(state) class GettextTarget(ConfigureMakeStaticDependencyTarget): def __init__(self, name='gettext'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://ftp.gnu.org/gnu/gettext/gettext-0.21.tar.xz', 'd20fcbb537e02dcf1383197ba05bd0734ef7bf5db06bdb241eb69b7d16b73192') def detect(self, state: BuildState) -> bool: return state.has_source_file('gettext-runtime') def configure(self, state: BuildState): opts = state.options opts['--enable-csharp'] = 'no' opts['--enable-java'] = 'no' opts['--enable-libasprintf'] = 'no' super().configure(state) class GlibTarget(BuildTarget): def __init__(self, name='glib'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://download.gnome.org/sources/glib/2.68/glib-2.68.3.tar.xz', 'e7e1a3c20c026109c45c9ec4a31d8dcebc22e86c69486993e565817d64be3138') def detect(self, state: BuildState) -> bool: return state.has_source_file('glib.doap') def configure(self, state: BuildState): super().configure(state) environment = state.environment environment['LDFLAGS'] += ' -framework CoreFoundation -framework Foundation' c_compiler = state.c_compiler() assert c_compiler cxx_compiler = state.cxx_compiler() assert cxx_compiler cpu = state.architecture() cpu_family = 'arm' if 'arm64' == cpu else cpu cross_file = state.build_path / (state.architecture() + '.txt') with open(cross_file, 'w') as f: f.write(f''' [binaries] c = '{c_compiler}' cpp = '{cxx_compiler}' objc = '{c_compiler}' objcpp = '{cxx_compiler}' pkgconfig = '{state.prefix_path}/bin/pkg-config' strip = '/usr/bin/strip' [host_machine] system = 'darwin' cpu_family = '{cpu_family}' cpu = '{cpu}' endian = 'little' ''') args = ( state.bin_path / 'meson', f'--prefix={state.install_path}', '--buildtype=release', '--default-library=static', f'--cross-file={cross_file}', state.source ) subprocess.check_call(args, cwd=state.build_path, env=environment) def build(self, state: BuildState): args = ('ninja',) subprocess.check_call(args, cwd=state.build_path, env=state.environment) def post_build(self, state: BuildState): self.install(state, tool='ninja') self.make_platform_header(state, '../lib/glib-2.0/include/glibconfig.h') @staticmethod def _process_pkg_config(pcfile: Path, line: str) -> str: return 'exec_prefix=${prefix}\n' + line if line.startswith('libdir=') else line class IconvTarget(ConfigureMakeStaticDependencyTarget): def __init__(self, name='iconv'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://ftp.gnu.org/gnu/libiconv/libiconv-1.16.tar.gz', 'e6a1b1b589654277ee790cce3734f07876ac4ccfaecbee8afa0b649cf529cc04') def detect(self, state: BuildState) -> bool: return state.has_source_file('include/iconv.h.in') def configure(self, state: BuildState): state.options['--enable-extra-encodings'] = 'yes' super().configure(state) class InstPatchTarget(CMakeStaticDependencyTarget): def __init__(self, name='instpatch'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/swami/libinstpatch/archive/v1.1.6.tar.gz', '8e9861b04ede275d712242664dab6ffa9166c7940fea3b017638681d25e10299') def configure(self, state: BuildState): state.options['LIB_SUFFIX'] = None # Workaround for missing frameworks in dependencies, no clue what's wrong at the moment state.environment['LDFLAGS'] = '-framework CoreFoundation -framework Foundation' super().configure(state) class IntlTarget(GettextTarget): def __init__(self, name='intl'): super().__init__(name) def configure(self, state: BuildState): # There is no way to configure intl only, do this for the runtime self.src_root = 'gettext-runtime' super().configure(state) def build(self, state: BuildState): # Build intl only, avoid complete gettext runtime self.src_root += '/intl' super().build(state) def post_build(self, state: BuildState): # Install intl only, avoid complete gettext runtime state.build_path /= self.src_root self.install(state) class JpegTurboTarget(CMakeStaticDependencyTarget): def __init__(self, name='jpeg-turbo'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://sourceforge.net/projects/libjpeg-turbo/files/2.1.3/libjpeg-turbo-2.1.3.tar.gz', '467b310903832b033fe56cd37720d1b73a6a3bd0171dbf6ff0b620385f4f76d0') def configure(self, state: BuildState): opts = state.options opts['ENABLE_SHARED'] = 'NO' opts['WITH_TURBOJPEG'] = 'NO' super().configure(state) class MoltenVKTarget(MakeTarget): def __init__(self, name='moltenvk'): super().__init__(name) # Building for multiple architectures is handled internally self.multi_platform = False def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/KhronosGroup/MoltenVK/archive/refs/tags/v1.1.10.tar.gz', 'fac11c2501195c9ce042103685c7778e35484562e6c084963a22072dd0a602e0') def detect(self, state: BuildState) -> bool: return state.has_source_file('MoltenVKPackaging.xcodeproj') def configure(self, state: BuildState): state.options['macos'] = None # Unset platform to avoid using specified macOS deployment target and SDK # MoltenVK defines minimal OS version itself, and usually, it requires the very recent SDK state.platform = None super().configure(state) def build(self, state: BuildState): args = ['./fetchDependencies', '--macos'] if state.verbose: args.append('-v') subprocess.check_call(args, cwd=state.build_path) super().build(state) def post_build(self, state: BuildState): if state.xcode: return if state.install_path.exists(): shutil.rmtree(state.install_path) include_path = state.install_path / 'include' os.makedirs(include_path) lib_path = state.install_path / 'lib' os.makedirs(lib_path) src_path = state.build_path / 'Package/Latest/MoltenVK' shutil.copytree(src_path / 'include/MoltenVK', include_path / 'MoltenVK') shutil.copy(state.build_path / 'LICENSE', state.install_path / 'apache2.txt') shutil.copy(src_path / 'dylib/macOS/libMoltenVK.dylib', lib_path) class Mpg123Target(CMakeStaticDependencyTarget): def __init__(self, name='mpg123'): super().__init__(name) self.src_root = 'ports/cmake' def prepare_source(self, state: BuildState): state.download_source( 'https://www.mpg123.de/download/mpg123-1.29.3.tar.bz2', '963885d8cc77262f28b77187c7d189e32195e64244de2530b798ddf32183e847', patches='mpg123-fix-cmake') def configure(self, state: BuildState): state.options['CMAKE_EXE_LINKER_FLAGS'] = '-framework AudioUnit' super().configure(state) def post_build(self, state: BuildState): super().post_build(state) self.keep_module_target(state, 'MPG123::libmpg123') class OggTarget(CMakeStaticDependencyTarget): def __init__(self, name='ogg'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/xiph/ogg/releases/download/v1.3.5/libogg-1.3.5.tar.xz', 'c4d91be36fc8e54deae7575241e03f4211eb102afb3fc0775fbbc1b740016705') class OpenALTarget(CMakeStaticDependencyTarget): def __init__(self, name='openal'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://openal-soft.org/openal-releases/openal-soft-1.22.1.tar.bz2', 'f7f98538d882b513812ae315770a5c88f6ee593e1cccd08648dd1bebbc5c98af') def configure(self, state: BuildState): opts = state.options opts['ALSOFT_EXAMPLES'] = 'NO' opts['ALSOFT_UTILS'] = 'NO' opts['LIBTYPE'] = 'STATIC' super().configure(state) FRAMEWORKS = '-framework ApplicationServices -framework AudioToolbox -framework AudioUnit -framework CoreAudio' def post_build(self, state: BuildState): super().post_build(state) def update_cmake_libs(line: str): link_libs = ' INTERFACE_LINK_LIBRARIES ' return f'{link_libs}"{OpenALTarget.FRAMEWORKS}"\n' if line.startswith(link_libs) else line config_path = state.install_path / 'lib/cmake/OpenAL/OpenALTargets.cmake' self.update_text_file(config_path, update_cmake_libs) @staticmethod def _process_pkg_config(pcfile: Path, line: str) -> str: libs_private = 'Libs.private:' return f'{libs_private} {OpenALTarget.FRAMEWORKS}\n' if line.startswith(libs_private) else line class OpusTarget(CMakeStaticDependencyTarget): def __init__(self, name='opus'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://ftp.osuosl.org/pub/xiph/releases/opus/opus-1.3.1.tar.gz', '65b58e1e25b2a114157014736a3d9dfeaad8d41be1c8179866f144a2fb44ff9d', patches='opus-fix-cmake') def configure(self, state: BuildState): state.options['PC_BUILD'] = 'floating-point' super().configure(state) @staticmethod def _process_pkg_config(pcfile: Path, line: str) -> str: version = 'Version:' cflags = 'Cflags:' libs = 'Libs:' if line.startswith(version): return version + ' 1.3.1\n' elif line.startswith(cflags): return cflags + ' -I${includedir}/opus\n' elif line.startswith(libs): return libs + ' -L${libdir} -lopus\n' return line class PcreTarget(ConfigureMakeStaticDependencyTarget): def __init__(self, name='pcre'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://ftp.pcre.org/pub/pcre/pcre-8.45.tar.bz2', '4dae6fdcd2bb0bb6c37b5f97c33c2be954da743985369cddac3546e3218bffb8') def detect(self, state: BuildState) -> bool: return state.has_source_file('pcre.h.in') def configure(self, state: BuildState): opts = state.options opts['--enable-unicode-properties'] = 'yes' opts['--enable-cpp'] = 'no' super().configure(state) def post_build(self, state: BuildState): super().post_build(state) self.update_config_script(state.install_path / 'bin/pcre-config') class SndFileTarget(CMakeStaticDependencyTarget): def __init__(self, name='sndfile'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/libsndfile/libsndfile/releases/download/1.1.0/libsndfile-1.1.0.tar.xz', '0f98e101c0f7c850a71225fb5feaf33b106227b3d331333ddc9bacee190bcf41') def configure(self, state: BuildState): opts = state.options opts['BUILD_REGTEST'] = 'NO' opts['BUILD_TESTING'] = 'NO' opts['ENABLE_MPEG'] = 'NO' super().configure(state) class VorbisTarget(CMakeStaticDependencyTarget): def __init__(self, name='vorbis'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://ftp.osuosl.org/pub/xiph/releases/vorbis/libvorbis-1.3.7.tar.xz', 'b33cc4934322bcbf6efcbacf49e3ca01aadbea4114ec9589d1b1e9d20f72954b') class VpxTarget(ConfigureMakeDependencyTarget): def __init__(self, name='vpx'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/webmproject/libvpx/archive/refs/tags/v1.11.0.tar.gz', '965e51c91ad9851e2337aebcc0f517440c637c506f3a03948062e3d5ea129a83') def configure(self, state: BuildState): hosts = { 'x86_64': 'x86_64-darwin16-gcc', 'arm64': 'arm64-darwin20-gcc', } opts = state.options opts['--disable-examples'] = None opts['--disable-unit-tests'] = None opts['--target'] = hosts[state.architecture()] super().configure(state) def detect(self, state: BuildState) -> bool: return state.has_source_file('vpxstats.h') class ZlibNgTarget(CMakeStaticDependencyTarget): def __init__(self, name='zlib-ng'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/zlib-ng/zlib-ng/archive/refs/tags/2.0.6.tar.gz', '8258b75a72303b661a238047cb348203d88d9dddf85d480ed885f375916fcab6') def detect(self, state: BuildState) -> bool: return state.has_source_file('zlib-ng.h') def configure(self, state: BuildState): opts = state.options opts['ZLIB_COMPAT'] = 'YES' opts['ZLIB_ENABLE_TESTS'] = 'NO' opts['ZLIB_FULL_VERSION'] = '1.2.11' super().configure(state) class ZMusicTarget(CMakeStaticDependencyTarget): def __init__(self, name='zmusic'): super().__init__(name) def prepare_source(self, state: BuildState): state.download_source( 'https://github.com/coelckers/ZMusic/archive/refs/tags/1.1.9.tar.gz', '4bf39917028bbe2f382748b4400a1cbaeb8975157fe404cff53411297d6ac68c') def detect(self, state: BuildState) -> bool: return state.has_source_file('include/zmusic.h') def configure(self, state: BuildState): opts = state.options opts['DYN_FLUIDSYNTH'] = 'OFF' opts['DYN_MPG123'] = 'OFF' opts['DYN_SNDFILE'] = 'OFF' super().configure(state)