Browse Source

waifulib: allow compiling using host clang and NDK sysroots, thus enabling building engine from any environment that has clang(termux as example)

half-secret
Alibek Omarov 5 years ago
parent
commit
6143c74dc9
  1. 106
      scripts/waifulib/xcompile.py

106
scripts/waifulib/xcompile.py

@ -41,13 +41,15 @@ class Android:
def __init__(self, ctx, arch, toolchain, api): def __init__(self, ctx, arch, toolchain, api):
self.ctx = ctx self.ctx = ctx
self.api = api
for i in ['ANDROID_NDK_HOME', 'ANDROID_NDK']: for i in ['ANDROID_NDK_HOME', 'ANDROID_NDK']:
self.ndk_home = os.getenv(i) self.ndk_home = os.getenv(i)
if self.ndk_home != None: if self.ndk_home != None:
break break
if not self.ndk_home: if not self.ndk_home:
conf.fatal('Set ANDROID_NDK_HOME environment variable pointing to the root of Android NDK!') ctx.fatal('Set ANDROID_NDK_HOME environment variable pointing to the root of Android NDK!')
# TODO: this were added at some point of NDK development # TODO: this were added at some point of NDK development
# but I don't know at which version # but I don't know at which version
@ -86,13 +88,18 @@ class Android:
elif self.ndk_rev >= 19 and self.api < 16: elif self.ndk_rev >= 19 and self.api < 16:
Logs.warn('API level automatically was set to 16 due to NDK support') Logs.warn('API level automatically was set to 16 due to NDK support')
self.api = 16 self.api = 16
else: self.api = api
self.toolchain_path = self.gen_toolchain_path() self.toolchain_path = self.gen_toolchain_path()
# TODO: Crystax support? # TODO: Crystax support?
# TODO: Support for everything else than linux-x86_64? # TODO: Support for everything else than linux-x86_64?
# TODO: Determine if I actually need to implement listed above # TODO: Determine if I actually need to implement listed above
def is_host(self):
'''
Checks if we using host compiler(implies clang)
'''
return self.toolchain == 'host'
def is_arm(self): def is_arm(self):
''' '''
Checks if selected architecture is **32-bit** ARM Checks if selected architecture is **32-bit** ARM
@ -126,7 +133,16 @@ class Android:
def is_hardfp(self): def is_hardfp(self):
return self.is_hardfloat return self.is_hardfloat
def gen_toolchain_path(self): def ndk_triplet(self):
if self.is_x86():
triplet = 'i686-linux-android'
elif self.is_arm():
triplet = 'arm-linux-androideabi'
else:
triplet = self.arch + '-linux-android'
return triplet
def gen_gcc_toolchain_path(self):
path = 'toolchains' path = 'toolchains'
if sys.platform.startswith('linux'): if sys.platform.startswith('linux'):
@ -149,13 +165,6 @@ class Android:
raise Exception('Clang is not supported for this NDK') raise Exception('Clang is not supported for this NDK')
toolchain_folder = 'llvm' toolchain_folder = 'llvm'
if self.is_x86():
triplet = 'i686-linux-android{}-'.format(self.api)
elif self.is_arm():
triplet = 'armv7a-linux-androideabi{}-'.format(self.api)
else:
triplet = self.arch + '-linux-android{}-'.format(self.api)
else: else:
if self.is_x86() or self.is_amd64(): if self.is_x86() or self.is_amd64():
toolchain_folder = self.arch + '-' + self.toolchain toolchain_folder = self.arch + '-' + self.toolchain
@ -164,42 +173,70 @@ class Android:
else: else:
toolchain_folder = self.arch + '-linux-android-' + self.toolchain toolchain_folder = self.arch + '-linux-android-' + self.toolchain
return os.path.abspath(os.path.join(self.ndk_home, path, toolchain_folder, 'prebuilt', toolchain_host))
def gen_toolchain_path(self):
if self.is_clang():
if self.is_x86(): if self.is_x86():
triplet = 'i686-linux-android-' triplet = 'i686-linux-android{}-'.format(self.api)
elif self.is_arm(): elif self.is_arm():
triplet = 'arm-linux-androideabi-' triplet = 'armv7a-linux-androideabi{}-'.format(self.api)
else: else:
triplet = self.arch + '-linux-android-' triplet = self.arch + '-linux-android{}-'.format(self.api)
else:
return os.path.join(path, toolchain_folder, 'prebuilt', toolchain_host, 'bin', triplet) triplet = self.ndk_triplet() + '-'
return os.path.join(self.gen_gcc_toolchain_path(), 'bin', triplet)
def cc(self): def cc(self):
return os.path.abspath(os.path.join(self.ndk_home, self.toolchain_path + ('clang' if self.is_clang() else 'gcc'))) if self.is_host():
return 'clang'
return self.toolchain_path + ('clang' if self.is_clang() else 'gcc')
def cxx(self): def cxx(self):
return os.path.abspath(os.path.join(self.ndk_home, self.toolchain_path + ('clang++' if self.is_clang() else 'g++'))) if self.is_host():
return 'clang++'
return self.toolchain_path + ('clang++' if self.is_clang() else 'g++')
def system_stl(self): def system_stl(self):
# TODO: proper STL support # TODO: proper STL support
return os.path.abspath(os.path.join(self.ndk_home, 'sources', 'cxx-stl', 'system', 'include')) return os.path.abspath(os.path.join(self.ndk_home, 'sources', 'cxx-stl', 'system', 'include'))
def libsysroot(self):
arch = self.arch
if self.is_arm():
arch = 'arm'
elif self.is_arm64():
arch = 'arm64'
path = 'platforms/android-{}/arch-{}'.format(self.api, arch)
return os.path.abspath(os.path.join(self.ndk_home, path))
def sysroot(self): def sysroot(self):
if self.ndk_rev >= 19: if self.ndk_rev >= 19 or self.is_host():
return os.path.abspath(os.path.join(self.ndk_home, 'sysroot')) return os.path.abspath(os.path.join(self.ndk_home, 'sysroot'))
else: else:
arch = self.arch return self.libsysroot()
if self.is_arm():
arch = 'arm'
elif self.is_arm64():
arch = 'arm64'
path = 'platforms/android-{}/arch-{}'.format(self.api, arch)
return os.path.abspath(os.path.join(self.ndk_home, path)) def clang_host_triplet(self):
triplet = ''
if self.is_arm():
triplet += 'arm'
elif self.is_x86():
triplet += 'i686'
else:
triplet += self.arch
triplet += '-linux-android'
return triplet
def cflags(self): def cflags(self):
cflags = [] cflags = []
if self.is_host():
cflags += ['-nostdlib', '--target=%s' % self.clang_host_triplet()]
if self.ndk_rev < 20: if self.ndk_rev < 20:
cflags = ['--sysroot={0}'.format(self.sysroot())] cflags += ['--sysroot={0}'.format(self.sysroot())]
elif self.is_host():
cflags += ['-isysroot={0}'.format(self.sysroot())]
cflags += ['-DANDROID', '-D__ANDROID__'] cflags += ['-DANDROID', '-D__ANDROID__']
cflags += ['-I{0}'.format(self.system_stl())] cflags += ['-I{0}'.format(self.system_stl())]
if self.is_arm(): if self.is_arm():
@ -222,15 +259,26 @@ class Android:
# they go before object list # they go before object list
def linkflags(self): def linkflags(self):
linkflags = [] linkflags = []
if self.ndk_rev < 20: if self.is_host():
linkflags = ['--sysroot={0}'.format(self.sysroot())] linkflags += ['-fuse-ld=lld', '-nostdlib', '--target=%s' % self.clang_host_triplet(),
'--gcc-toolchain=%s' % self.gen_gcc_toolchain_path()]
if self.ndk_rev < 20 or self.is_host():
linkflags += ['--sysroot={0}'.format(self.libsysroot())]
if self.is_host():
linkflags += ['-L{0}/usr/lib/{1}'.format(self.sysroot(), self.ndk_triplet()),
'-L{0}/sysroot/usr/lib/{1}/'.format(self.gen_gcc_toolchain_path(), self.ndk_triplet())]
return linkflags return linkflags
def ldflags(self): def ldflags(self):
ldflags = ['-lgcc', '-no-canonical-prefixes'] ldflags = ['-lgcc', '-no-canonical-prefixes']
if self.is_arm(): if self.is_arm():
if self.arch == 'armeabi-v7a': if self.arch == 'armeabi-v7a':
ldflags += ['-march=armv7-a', '-Wl,--fix-cortex-a8', '-mthumb'] ldflags += ['-march=armv7-a']
if not self.is_host(): # lld only
ldflags += ['-Wl,--fix-cortex-a8']
ldflags += ['-mthumb']
if self.is_hardfloat: if self.is_hardfloat:
ldflags += ['-Wl,--no-warn-mismatch', '-lm_hard'] ldflags += ['-Wl,--no-warn-mismatch', '-lm_hard']
else: else:

Loading…
Cancel
Save