| #!/usr/bin/env python |
| # |
| # Copyright (C) 2016 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| # |
| from __future__ import print_function |
| |
| import argparse |
| import glob |
| import multiprocessing |
| import os |
| import shutil |
| import subprocess |
| import sys |
| import re |
| |
| |
| THIS_DIR = os.path.realpath(os.path.dirname(__file__)) |
| ORIG_ENV = dict(os.environ) |
| |
| |
| def android_path(*args): |
| out_dir = os.path.realpath(os.path.join(THIS_DIR, '../..', *args)) |
| return out_dir |
| |
| |
| def build_path(*args): |
| # Our multistage build directories will be placed under OUT_DIR if it is in |
| # the environment. By default they will be placed under |
| # $ANDROID_BUILD_TOP/out. |
| top_out = ORIG_ENV.get('OUT_DIR', android_path('out')) |
| if not os.path.isabs(top_out): |
| top_out = os.path.realpath(top_out) |
| out_dir = os.path.join(top_out, *args) |
| return out_dir |
| |
| |
| def install_file(src, dst): |
| print('Copying ' + src) |
| shutil.copy2(src, dst) |
| |
| |
| def install_directory(src, dst): |
| print('Copying ' + src) |
| shutil.copytree(src, dst) |
| |
| |
| def build(out_dir): |
| products = ( |
| 'aosp_arm', |
| 'aosp_arm64', |
| 'aosp_mips', |
| # 'aosp_mips64', |
| 'aosp_x86', |
| 'aosp_x86_64', |
| ) |
| for product in products: |
| build_product(out_dir, product) |
| |
| |
| def build_product(out_dir, product): |
| env = dict(ORIG_ENV) |
| env['ANDROID_USE_BUILDCACHE'] = 'false' |
| env['FORCE_BUILD_LLVM_COMPONENTS'] = 'true' |
| env['FORCE_BUILD_RS_COMPAT'] = 'true' |
| env['OUT_DIR'] = out_dir |
| env['SKIP_LLVM_TESTS'] = 'true' |
| env['SOONG_ALLOW_MISSING_DEPENDENCIES'] = 'true' |
| env['TARGET_BUILD_VARIANT'] = 'userdebug' |
| env['TARGET_PRODUCT'] = product |
| |
| jobs_arg = '-j{}'.format(multiprocessing.cpu_count()) |
| targets = [ |
| # PHONY target specified in frameworks/rs/Android.mk. |
| 'rs-prebuilts-full', |
| # We have to explicitly specify the jar for JACK to build. |
| android_path('out/target/common/obj/JAVA_LIBRARIES/' + |
| 'android-support-v8-renderscript_intermediates/classes.jar') |
| ] |
| subprocess.check_call( |
| ['make', jobs_arg] + targets, cwd=android_path(), env=env) |
| |
| |
| def package_toolchain(build_dir, build_name, host, dist_dir): |
| package_name = 'renderscript-' + build_name |
| install_host_dir = build_path('install', host) |
| install_dir = os.path.join(install_host_dir, package_name) |
| |
| # Remove any previously installed toolchain so it doesn't pollute the |
| # build. |
| if os.path.exists(install_host_dir): |
| shutil.rmtree(install_host_dir) |
| |
| install_toolchain(build_dir, install_dir, host) |
| |
| tarball_name = package_name + '-' + host |
| package_path = os.path.join(dist_dir, tarball_name) + '.tar.bz2' |
| print('Packaging ' + package_path) |
| args = [ |
| 'tar', '-cjC', install_host_dir, '-f', package_path, package_name |
| ] |
| subprocess.check_call(args) |
| |
| |
| def install_toolchain(build_dir, install_dir, host): |
| install_built_host_files(build_dir, install_dir, host) |
| install_clang_headers(build_dir, install_dir, host) |
| install_built_device_files(build_dir, install_dir, host) |
| install_license_files(install_dir) |
| # We need to package libwinpthread-1.dll for Windows. This is explicitly |
| # linked whenever pthreads is used, and the build system doesn't allow |
| # us to link just that library statically (ldflags are stripped out |
| # of ldlibs and vice-versa). |
| # Bug: http://b/34273721 |
| if host.startswith('windows'): |
| install_winpthreads(install_dir) |
| |
| |
| def install_winpthreads(install_dir): |
| """Installs the winpthreads runtime to the Windows bin directory.""" |
| lib_name = 'libwinpthread-1.dll' |
| mingw_dir = android_path( |
| 'prebuilts/gcc/linux-x86/host/x86_64-w64-mingw32-4.8') |
| # RenderScript NDK toolchains for Windows only contains 32-bit binaries. |
| lib_path = os.path.join(mingw_dir, 'x86_64-w64-mingw32/lib32', lib_name) |
| |
| lib_install = os.path.join(install_dir, 'bin', lib_name) |
| install_file(lib_path, lib_install) |
| |
| |
| def install_built_host_files(build_dir, install_dir, host): |
| is_windows = host.startswith('windows') |
| is_darwin = host.startswith('darwin-x86') |
| bin_ext = '.exe' if is_windows else '' |
| |
| if is_windows: |
| lib_ext = '.dll' |
| elif is_darwin: |
| lib_ext = '.dylib' |
| else: |
| lib_ext = '.so' |
| |
| built_files = [ |
| 'bin/llvm-rs-cc' + bin_ext, |
| 'bin/bcc_compat' + bin_ext, |
| ] |
| |
| if is_windows: |
| built_files.extend([ |
| 'lib/libbcc' + lib_ext, |
| 'lib/libbcinfo' + lib_ext, |
| 'lib/libclang' + lib_ext, |
| 'lib/libLLVM' + lib_ext, |
| ]) |
| else: |
| built_files.extend([ |
| 'lib64/libbcc' + lib_ext, |
| 'lib64/libbcinfo' + lib_ext, |
| 'lib64/libclang' + lib_ext, |
| 'lib64/libLLVM' + lib_ext, |
| 'lib64/libc++' + lib_ext, |
| ]) |
| |
| for built_file in built_files: |
| dirname = os.path.dirname(built_file) |
| # Put dlls and exes into bin/ for windows. |
| # Bug: http://b/34273721 |
| if is_windows: |
| dirname = 'bin' |
| install_path = os.path.join(install_dir, dirname) |
| if not os.path.exists(install_path): |
| os.makedirs(install_path) |
| |
| built_path = os.path.join(build_dir, 'host', host, built_file) |
| install_file(built_path, install_path) |
| |
| file_name = os.path.basename(built_file) |
| |
| # Only strip bin files (not libs) on darwin. |
| if not is_darwin or built_file.startswith('bin/'): |
| subprocess.check_call( |
| ['strip', os.path.join(install_path, file_name)]) |
| |
| |
| def install_clang_headers(build_dir, install_dir, host): |
| def should_copy(path): |
| if os.path.basename(path) in ('Makefile', 'CMakeLists.txt'): |
| return False |
| _, ext = os.path.splitext(path) |
| if ext == '.mk': |
| return False |
| return True |
| |
| headers_src = android_path('external/clang/lib/Headers') |
| headers_dst = os.path.join( |
| install_dir, 'clang-include') |
| os.makedirs(headers_dst) |
| for header in os.listdir(headers_src): |
| if not should_copy(header): |
| continue |
| install_file(os.path.join(headers_src, header), headers_dst) |
| |
| install_file(android_path('bionic/libc/include/stdatomic.h'), headers_dst) |
| |
| |
| def install_built_device_files(build_dir, install_dir, host): |
| product_to_arch = { |
| 'generic': 'arm', |
| 'generic_arm64': 'arm64', |
| 'generic_mips': 'mips', |
| # 'generic_mips64': 'mips64el', |
| 'generic_x86': 'x86', |
| 'generic_x86_64': 'x86_64', |
| } |
| |
| bc_lib = 'librsrt' |
| |
| static_libs = { |
| 'libRScpp_static', |
| 'libcompiler_rt' |
| } |
| |
| shared_libs = { |
| 'libRSSupport.so', |
| 'libRSSupportIO.so', |
| 'libblasV8.so', |
| } |
| |
| for product, arch in product_to_arch.items(): |
| lib_dir = os.path.join(install_dir, 'platform', arch) |
| os.makedirs(lib_dir) |
| |
| # Copy librsrt_ARCH.bc. |
| lib_name = bc_lib + '_' + arch + '.bc' |
| if not host.startswith('windows'): |
| built_lib = os.path.join(build_dir, 'host', host, 'lib64', lib_name) |
| else: |
| built_lib = os.path.join(build_dir, 'host', 'linux-x86', 'lib64', lib_name) |
| install_file(built_lib, os.path.join(lib_dir, bc_lib + '.bc')) |
| |
| # Copy static libs and share libs. |
| product_dir = os.path.join(build_dir, 'target/product', product) |
| static_lib_dir = os.path.join(product_dir, 'obj/STATIC_LIBRARIES') |
| shared_lib_dir = os.path.join(product_dir, 'obj/lib') |
| for static_lib in static_libs: |
| built_lib = os.path.join( |
| static_lib_dir, static_lib + '_intermediates/' + static_lib + '.a') |
| lib_name = static_lib + '.a' |
| install_file(built_lib, os.path.join(lib_dir, lib_name)) |
| for shared_lib in shared_libs: |
| built_lib = os.path.join(shared_lib_dir, shared_lib) |
| lib_name = shared_lib |
| install_file(built_lib, os.path.join(lib_dir, lib_name)) |
| |
| # Copy renderscript-v8.jar. |
| lib_dir = os.path.join(install_dir, 'platform') |
| jar_dir = os.path.join(build_dir, 'target/common/obj/JAVA_LIBRARIES/' |
| 'android-support-v8-renderscript_intermediates/classes.jar') |
| install_file(jar_dir, os.path.join(lib_dir, 'renderscript-v8.jar')) |
| |
| # Copy RS runtime headers. |
| headers_dst_base = os.path.join(install_dir, 'platform', 'rs') |
| |
| headers_src = android_path('frameworks/rs/script_api/include') |
| headers_dst = os.path.join(headers_dst_base, 'scriptc') |
| install_directory(headers_src, headers_dst) |
| |
| # Copy RS C++ API headers. |
| headers_src = android_path('frameworks/rs/cpp/util') |
| headers_dst = os.path.join(headers_dst_base, 'cpp/util') |
| install_directory(headers_src, headers_dst) |
| install_file(android_path('frameworks/rs/rsDefines.h'), headers_dst_base) |
| install_file(android_path('frameworks/rs/cpp/RenderScript.h'), os.path.join(headers_dst_base, 'cpp')) |
| install_file(android_path('frameworks/rs/cpp/rsCppStructs.h'), os.path.join(headers_dst_base, 'cpp')) |
| |
| |
| def install_license_files(install_dir): |
| projects = ( |
| 'external/clang', |
| 'external/compiler-rt', |
| 'external/llvm', |
| 'frameworks/compile/slang', |
| 'frameworks/compile/libbcc', |
| # 'frameworks/rs', # No notice license file found. |
| ) |
| |
| notices = [] |
| for project in projects: |
| project_path = android_path(project) |
| license_pattern = os.path.join(project_path, 'MODULE_LICENSE_*') |
| for license_file in glob.glob(license_pattern): |
| install_file(license_file, install_dir) |
| with open(os.path.join(project_path, 'NOTICE')) as notice_file: |
| notices.append(notice_file.read()) |
| with open(os.path.join(install_dir, 'NOTICE'), 'w') as notice_file: |
| notice_file.write('\n'.join(notices)) |
| |
| |
| def parse_args(): |
| parser = argparse.ArgumentParser() |
| |
| parser.add_argument( |
| '--build-name', default='dev', help='Release name for the package.') |
| |
| return parser.parse_args() |
| |
| |
| def main(): |
| args = parse_args() |
| |
| if sys.platform.startswith('linux'): |
| hosts = ['linux-x86', 'windows-x86'] |
| elif sys.platform == 'darwin': |
| hosts = ['darwin-x86'] |
| else: |
| raise RuntimeError('Unsupported host: {}'.format(sys.platform)) |
| |
| out_dir = build_path() |
| build(out_dir=out_dir) |
| |
| dist_dir = ORIG_ENV.get('DIST_DIR', out_dir) |
| for host in hosts: |
| package_toolchain(out_dir, args.build_name, host, dist_dir) |
| |
| |
| if __name__ == '__main__': |
| main() |