zdoom-macos-deps/aedi/target/library_tier1.py

518 lines
18 KiB
Python
Raw Normal View History

#
# Helper module to build macOS version of various source ports
# Copyright (C) 2020-2021 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 <http://www.gnu.org/licenses/>.
#
2021-05-01 08:54:52 +00:00
from .base import *
from ..state import BuildState
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 = self.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] = self.environment[cflags] + ' -D_FILE_OFFSET_BITS=64 -O2'
def post_build(self, state: BuildState):
self.options['PREFIX'] = state.install_path
self.install(state, self.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(
2021-07-03 12:53:59 +00:00
'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)
2021-06-02 07:32:25 +00:00
class FlacTarget(CMakeStaticDependencyTarget):
def __init__(self, name='flac'):
super().__init__(name)
2021-06-02 07:32:25 +00:00
opts = self.options
opts['BUILD_CXXLIBS'] = 'NO'
opts['BUILD_EXAMPLES'] = 'NO'
opts['OGG_PACKAGE'] = 'ogg'
opts['VERSION'] = '1.3.3'
def prepare_source(self, state: BuildState):
state.download_source(
'https://ftp.osuosl.org/pub/xiph/releases/flac/flac-1.3.3.tar.xz',
2021-06-02 07:32:25 +00:00
'213e82bd716c9de6db2f98bcadbc4c24c7e2efe8c75939a1a84e28539c4e1748',
2021-06-19 08:21:06 +00:00
patches='flac-add-cmake')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('FLAC/flac.pc.in')
2021-06-02 07:32:25 +00:00
def configure(self, state: BuildState):
self.options['CMAKE_EXE_LINKER_FLAGS'] = f'-framework CoreFoundation -L{state.lib_path}'
2021-06-02 07:32:25 +00:00
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')
shutil.copytree(state.install_path / 'share/pkgconfig', state.install_path / 'lib/pkgconfig')
2021-06-02 07:32:25 +00:00
self.keep_module_target(state, 'FLAC::FLAC')
2021-06-02 07:32:25 +00:00
class FluidSynthTarget(CMakeStaticDependencyTarget):
def __init__(self, name='fluidsynth'):
super().__init__(name)
opts = self.options
opts['LIB_SUFFIX'] = None
opts['enable-framework'] = 'NO'
opts['enable-readline'] = 'NO'
opts['enable-sdl2'] = 'NO'
def prepare_source(self, state: BuildState):
state.download_source(
2021-07-15 06:39:00 +00:00
'https://github.com/FluidSynth/fluidsynth/archive/refs/tags/v2.2.2.tar.gz',
'695aedbfd53160fef7a9a1f66cd6d5cc8a5da0fd472eee458d82b848b6065f9a')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('fluidsynth.pc.in')
def configure(self, state: BuildState):
# TODO: Figure out why private dependencies aren't pulled
self.options['CMAKE_EXE_LINKER_FLAGS'] = state.run_pkg_config('--libs', 'glib-2.0')
super().configure(state)
class GettextTarget(ConfigureMakeStaticDependencyTarget):
def __init__(self, name='gettext'):
super().__init__(name)
opts = self.options
opts['--enable-csharp'] = 'no'
opts['--enable-java'] = 'no'
opts['--enable-libasprintf'] = 'no'
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')
class GlibTarget(BuildTarget):
def __init__(self, name='glib'):
super().__init__(name)
def prepare_source(self, state: BuildState):
state.download_source(
2021-07-04 09:38:35 +00:00
'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 = self.environment
2021-07-04 09:38:35 +00:00
environment['LDFLAGS'] += ' -framework CoreFoundation -framework Foundation'
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 = '{state.c_compiler()}'
cpp = '{state.cxx_compiler()}'
objc = '{state.c_compiler()}'
objcpp = '{state.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=self.environment)
def post_build(self, state: BuildState):
self.install(state, tool='ninja')
self.make_platform_header(state, '../lib/glib-2.0/include/glibconfig.h')
2021-07-04 09:38:35 +00:00
@staticmethod
def _process_pkg_config(pcfile: str, 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)
self.options['--enable-extra-encodings'] = 'yes'
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')
class InstPatchTarget(CMakeStaticDependencyTarget):
def __init__(self, name='instpatch'):
super().__init__(name)
self.options['LIB_SUFFIX'] = None
# Workaround for missing frameworks in dependencies, no clue what's wrong at the moment
self.environment['LDFLAGS'] = '-framework CoreFoundation -framework Foundation'
def prepare_source(self, state: BuildState):
state.download_source(
2021-01-24 09:46:20 +00:00
'https://github.com/swami/libinstpatch/archive/v1.1.6.tar.gz',
'8e9861b04ede275d712242664dab6ffa9166c7940fea3b017638681d25e10299')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('libinstpatch-1.0.pc.in')
class IntlTarget(GettextTarget):
def __init__(self, name='intl'):
super().__init__(name)
self.src_root = 'gettext-runtime'
self.make.src_root += self.src_root + os.sep + 'intl'
def post_build(self, state: BuildState):
# Do install of intl only, avoid complete gettext runtime
self.src_root = self.make.src_root
self.install(state)
class JpegTurboTarget(CMakeStaticDependencyTarget):
def __init__(self, name='jpeg-turbo'):
super().__init__(name)
self.options['WITH_TURBOJPEG'] = 'NO'
def prepare_source(self, state: BuildState):
state.download_source(
2021-04-24 09:13:40 +00:00
'https://downloads.sourceforge.net/project/libjpeg-turbo/2.1.0/libjpeg-turbo-2.1.0.tar.gz',
'bef89803e506f27715c5627b1e3219c95b80fc31465d4452de2a909d382e4444')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('turbojpeg.h')
class MoltenVKTarget(MakeTarget):
def __init__(self, name='moltenvk'):
super().__init__(name)
self.options['macos'] = None
# Building for multiple architectures is handled internally
self.multi_platform = False
def prepare_source(self, state: BuildState):
state.download_source(
2021-06-29 08:48:16 +00:00
'https://github.com/KhronosGroup/MoltenVK/archive/refs/tags/v1.1.4.tar.gz',
'f9bba6d3bf3648e7685c247cb6d126d62508af614bc549cedd5859a7da64967e')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('MoltenVKPackaging.xcodeproj')
def configure(self, state: BuildState):
# 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', '-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)
lib_path = state.install_path / 'lib'
os.makedirs(lib_path)
src_path = state.build_path / 'Package/Latest/MoltenVK'
shutil.copytree(src_path / 'include', state.install_path / 'include')
shutil.copy(state.build_path + 'LICENSE', state.install_path / 'apache2.txt')
shutil.copy(src_path / 'dylib/macOS/libMoltenVK.dylib', lib_path)
2021-06-01 09:51:30 +00:00
class Mpg123Target(CMakeStaticDependencyTarget):
def __init__(self, name='mpg123'):
super().__init__(name)
2021-06-01 09:51:30 +00:00
self.src_root = 'ports/cmake'
self.options['CMAKE_EXE_LINKER_FLAGS'] = '-framework AudioUnit'
def prepare_source(self, state: BuildState):
state.download_source(
2021-07-25 09:06:02 +00:00
'https://downloads.sourceforge.net/project/mpg123/mpg123/1.28.2/mpg123-1.28.2.tar.bz2',
'7eefd4b68fdac7e138d04c37efe12155a8ebf25a5bccf0fb7e775af22d21db00',
2021-06-12 09:57:03 +00:00
patches='mpg123-fix-cmake')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('libmpg123.pc.in')
def post_build(self, state: BuildState):
super().post_build(state)
self.keep_module_target(state, 'MPG123::libmpg123')
2021-05-31 08:42:51 +00:00
class OggTarget(CMakeStaticDependencyTarget):
def __init__(self, name='ogg'):
super().__init__(name)
def prepare_source(self, state: BuildState):
state.download_source(
2021-06-04 06:43:22 +00:00
'https://github.com/xiph/ogg/releases/download/v1.3.5/libogg-1.3.5.tar.xz',
'c4d91be36fc8e54deae7575241e03f4211eb102afb3fc0775fbbc1b740016705')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('ogg.pc.in')
class OpenALTarget(CMakeStaticDependencyTarget):
def __init__(self, name='openal'):
super().__init__(name)
opts = self.options
opts['ALSOFT_EXAMPLES'] = 'NO'
opts['ALSOFT_UTILS'] = 'NO'
def prepare_source(self, state: BuildState):
state.download_source(
2021-02-05 08:02:31 +00:00
'https://openal-soft.org/openal-releases/openal-soft-1.21.1.tar.bz2',
'c8ad767e9a3230df66756a21cc8ebf218a9d47288f2514014832204e666af5d8')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('openal.pc.in')
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/OpenALConfig.cmake'
self.update_text_file(config_path, update_cmake_libs)
@staticmethod
def _process_pkg_config(pcfile: str, line: str) -> str:
libs_private = 'Libs.private:'
return f'{libs_private} {OpenALTarget.FRAMEWORKS}\n' if line.startswith(libs_private) else line
2021-06-02 09:12:03 +00:00
class OpusTarget(CMakeStaticDependencyTarget):
def __init__(self, name='opus'):
super().__init__(name)
2021-06-02 09:12:03 +00:00
self.options['PC_BUILD'] = 'floating-point'
def prepare_source(self, state: BuildState):
state.download_source(
'https://ftp.osuosl.org/pub/xiph/releases/opus/opus-1.3.1.tar.gz',
2021-06-02 09:12:03 +00:00
'65b58e1e25b2a114157014736a3d9dfeaad8d41be1c8179866f144a2fb44ff9d',
patches='opus-fix-cmake')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('opus.pc.in')
2021-06-02 09:12:03 +00:00
@staticmethod
def _process_pkg_config(pcfile: str, 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)
opts = self.options
opts['--enable-unicode-properties'] = 'yes'
opts['--enable-cpp'] = 'no'
def prepare_source(self, state: BuildState):
state.download_source(
2021-07-03 12:43:43 +00:00
'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')
2021-07-03 12:49:16 +00:00
def post_build(self, state: BuildState):
super().post_build(state)
self.update_config_script(state.install_path / 'bin/pcre-config')
2021-07-03 12:49:16 +00:00
class SndFileTarget(CMakeStaticDependencyTarget):
def __init__(self, name='sndfile'):
super().__init__(name)
opts = self.options
opts['BUILD_REGTEST'] = 'NO'
opts['BUILD_TESTING'] = 'NO'
def prepare_source(self, state: BuildState):
state.download_source(
2021-01-25 09:31:03 +00:00
'https://github.com/libsndfile/libsndfile/releases/download/1.0.31/libsndfile-1.0.31.tar.bz2',
'a8cfb1c09ea6e90eff4ca87322d4168cdbe5035cb48717b40bf77e751cc02163')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('sndfile.pc.in')
2021-05-31 08:54:51 +00:00
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')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('vorbis.pc.in')
class VpxTarget(ConfigureMakeDependencyTarget):
def __init__(self, name='vpx'):
super().__init__(name)
opts = self.options
opts['--disable-examples'] = None
opts['--disable-unit-tests'] = None
def prepare_source(self, state: BuildState):
state.download_source(
2021-03-30 12:13:48 +00:00
'https://github.com/webmproject/libvpx/archive/v1.10.0.tar.gz',
'85803ccbdbdd7a3b03d930187cb055f1353596969c1f92ebec2db839fa4f834a')
def configure(self, state: BuildState):
hosts = {
'x86_64': 'x86_64-darwin13-gcc',
'arm64': 'arm64-darwin20-gcc',
}
self.options['--target'] = hosts[state.architecture()]
super().configure(state)
def detect(self, state: BuildState) -> bool:
return state.has_source_file('vpxstats.h')
2021-03-20 13:26:17 +00:00
class ZlibNgTarget(CMakeStaticDependencyTarget):
def __init__(self, name='zlib-ng'):
super().__init__(name)
opts = self.options
opts['ZLIB_COMPAT'] = 'YES'
opts['ZLIB_ENABLE_TESTS'] = 'NO'
opts['ZLIB_FULL_VERSION'] = '1.2.11'
def prepare_source(self, state: BuildState):
state.download_source(
2021-06-27 07:08:38 +00:00
'https://github.com/zlib-ng/zlib-ng/archive/refs/tags/2.0.5.tar.gz',
'eca3fe72aea7036c31d00ca120493923c4d5b99fe02e6d3322f7c88dbdcd0085')
2021-03-20 13:26:17 +00:00
def detect(self, state: BuildState) -> bool:
return state.has_source_file('zlib-ng.h')
class ZMusicTarget(CMakeStaticDependencyTarget):
def __init__(self, name='zmusic'):
super().__init__(name)
opts = self.options
opts['DYN_FLUIDSYNTH'] = 'OFF'
opts['DYN_MPG123'] = 'OFF'
opts['DYN_SNDFILE'] = 'OFF'
def prepare_source(self, state: BuildState):
state.download_source(
2021-05-22 15:43:04 +00:00
'https://github.com/coelckers/ZMusic/archive/refs/tags/1.1.8.tar.gz',
'73082f661b7b0bb33348d1d186c132deec9132a1613480348a00172b49c9fd68')
def detect(self, state: BuildState) -> bool:
return state.has_source_file('include/zmusic.h')