Magisk/build.py

603 lines
18 KiB
Python
Raw Normal View History

2017-06-04 01:39:52 +08:00
#!/usr/bin/env python3
2020-12-26 16:04:41 -08:00
import argparse
import errno
import lzma
2021-05-12 13:40:53 +08:00
import multiprocessing
import os
2020-12-26 16:04:41 -08:00
import os.path as op
2021-05-12 13:40:53 +08:00
import platform
import shutil
import stat
2021-05-12 13:40:53 +08:00
import subprocess
import sys
import textwrap
2021-05-12 13:40:53 +08:00
import urllib.request
2022-04-15 10:04:04 -07:00
import tarfile
2017-11-12 04:17:56 +08:00
2017-06-03 20:19:01 +08:00
def error(str):
2022-08-17 01:59:23 -07:00
if no_color:
print(f'\n! {str}\n')
2020-12-25 15:54:47 -08:00
else:
print(f'\n\033[41m{str}\033[0m\n')
sys.exit(1)
2017-06-03 20:19:01 +08:00
def header(str):
2022-08-17 01:59:23 -07:00
if no_color:
2020-12-25 15:54:47 -08:00
print(f'\n{str}\n')
else:
print(f'\n\033[44m{str}\033[0m\n')
2017-06-03 20:19:01 +08:00
2018-08-06 02:01:04 +08:00
def vprint(str):
if args.verbose:
print(str)
2018-08-06 02:01:04 +08:00
2020-12-26 16:04:41 -08:00
is_windows = os.name == 'nt'
2022-06-30 14:50:21 -07:00
EXE_EXT = '.exe' if is_windows else ''
2020-12-26 16:04:41 -08:00
2022-08-17 01:59:23 -07:00
no_color = False
if is_windows:
try:
import colorama
colorama.init()
except ImportError:
# We can't do ANSI color codes in terminal on Windows without colorama
no_color = True
2020-12-26 16:04:41 -08:00
2017-06-03 20:19:01 +08:00
# Environment checks
if not sys.version_info >= (3, 6):
error('Requires Python 3.6+')
2017-06-03 20:19:01 +08:00
2020-10-17 06:42:34 -07:00
if 'ANDROID_SDK_ROOT' not in os.environ:
error('Please add Android SDK path to ANDROID_SDK_ROOT environment variable!')
2017-06-03 20:19:01 +08:00
try:
2020-04-03 03:33:39 -07:00
subprocess.run(['javac', '-version'],
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
2017-06-03 20:19:01 +08:00
except FileNotFoundError:
2020-04-03 03:33:39 -07:00
error('Please install JDK and make sure \'javac\' is available in PATH')
2017-06-03 20:19:01 +08:00
2018-05-13 03:04:40 +08:00
cpu_count = multiprocessing.cpu_count()
2021-05-12 13:40:53 +08:00
archs = ['armeabi-v7a', 'x86', 'arm64-v8a', 'x86_64']
2022-06-30 14:50:21 -07:00
triples = ['armv7a-linux-androideabi', 'i686-linux-android', 'aarch64-linux-android', 'x86_64-linux-android']
2022-03-17 03:15:39 -07:00
default_targets = ['magisk', 'magiskinit', 'magiskboot', 'magiskpolicy', 'busybox']
support_targets = default_targets + ['resetprop', 'test']
2022-06-30 14:50:21 -07:00
rust_targets = ['magisk', 'magiskinit', 'magiskboot', 'magiskpolicy']
2020-04-03 03:33:39 -07:00
sdk_path = os.environ['ANDROID_SDK_ROOT']
ndk_root = op.join(sdk_path, 'ndk')
2020-04-03 03:33:39 -07:00
ndk_path = op.join(ndk_root, 'magisk')
ndk_build = op.join(ndk_path, 'ndk-build')
2022-06-30 14:50:21 -07:00
rust_bin = op.join(ndk_path, 'toolchains', 'rust', 'bin')
cargo = op.join(rust_bin, 'cargo' + EXE_EXT)
2020-04-03 03:33:39 -07:00
gradlew = op.join('.', 'gradlew' + ('.bat' if is_windows else ''))
2022-06-30 14:50:21 -07:00
adb_path = op.join(sdk_path, 'platform-tools', 'adb' + EXE_EXT)
native_gen_path = op.realpath(op.join('native', 'out', 'generated'))
2018-05-13 03:04:40 +08:00
2019-10-17 18:02:31 -04:00
# Global vars
config = {}
STDOUT = None
2020-12-26 16:04:41 -08:00
build_tools = None
def mv(source, target):
try:
shutil.move(source, target)
2020-04-03 03:33:39 -07:00
vprint(f'mv {source} -> {target}')
2019-02-11 17:14:29 -05:00
except:
pass
def cp(source, target):
try:
shutil.copyfile(source, target)
2020-04-03 03:33:39 -07:00
vprint(f'cp {source} -> {target}')
2019-02-11 17:14:29 -05:00
except:
pass
def rm(file):
try:
os.remove(file)
2020-04-03 03:33:39 -07:00
vprint(f'rm {file}')
except OSError as e:
if e.errno != errno.ENOENT:
raise
2017-06-03 20:19:01 +08:00
def rm_on_error(func, path, _):
# Remove a read-only file on Windows will get "WindowsError: [Error 5] Access is denied"
# Clear the "read-only" and retry
os.chmod(path, stat.S_IWRITE)
os.unlink(path)
2020-04-03 03:33:39 -07:00
def rm_rf(path):
vprint(f'rm -rf {path}')
shutil.rmtree(path, ignore_errors=True, onerror=rm_on_error)
2020-04-03 03:33:39 -07:00
def mkdir(path, mode=0o755):
try:
os.mkdir(path, mode)
2019-02-11 17:14:29 -05:00
except:
pass
2020-04-03 03:33:39 -07:00
def mkdir_p(path, mode=0o755):
os.makedirs(path, mode, exist_ok=True)
2022-06-30 14:50:21 -07:00
def execv(cmd, env=None):
return subprocess.run(cmd, stdout=STDOUT, env=env)
def system(cmd):
return subprocess.run(cmd, shell=True, stdout=STDOUT)
2022-06-30 14:50:21 -07:00
def cmd_out(cmd, env=None):
return subprocess.run(cmd, stdout=subprocess.PIPE, stderr=subprocess.DEVNULL, env=env) \
.stdout.strip().decode('utf-8')
2020-12-24 04:46:31 -08:00
def xz(data):
return lzma.compress(data, preset=9, check=lzma.CHECK_NONE)
2020-04-03 03:33:39 -07:00
def parse_props(file):
props = {}
with open(file, 'r') as f:
for line in [l.strip(' \t\r\n') for l in f]:
if line.startswith('#') or len(line) == 0:
continue
prop = line.split('=')
if len(prop) != 2:
continue
2020-12-24 04:46:31 -08:00
value = prop[1].strip(' \t\r\n')
if len(value) == 0:
continue
props[prop[0].strip(' \t\r\n')] = value
2020-04-03 03:33:39 -07:00
return props
def load_config(args):
2020-12-25 05:34:15 -08:00
commit_hash = cmd_out(['git', 'rev-parse', '--short=8', 'HEAD'])
2020-04-03 03:33:39 -07:00
2020-12-25 05:34:15 -08:00
# Default values
2020-12-24 04:46:31 -08:00
config['version'] = commit_hash
config['outdir'] = 'out'
2020-12-25 05:34:15 -08:00
# Load prop files
2020-12-24 04:46:31 -08:00
if op.exists(args.config):
config.update(parse_props(args.config))
2019-10-23 19:57:47 +09:00
2020-12-25 05:34:15 -08:00
for key, value in parse_props('gradle.properties').items():
if key.startswith('magisk.'):
config[key[7:]] = value
try:
config['versionCode'] = int(config['versionCode'])
except ValueError:
error('Config error: "versionCode" is required to be an integer')
mkdir_p(config['outdir'])
global STDOUT
STDOUT = None if args.verbose else subprocess.DEVNULL
def clean_elf():
2019-10-17 18:02:31 -04:00
if is_windows:
2020-04-03 03:33:39 -07:00
elf_cleaner = op.join('tools', 'elf-cleaner.exe')
else:
2020-04-03 03:33:39 -07:00
elf_cleaner = op.join('native', 'out', 'elf-cleaner')
if not op.exists(elf_cleaner):
execv(['g++', '-std=c++11', 'tools/termux-elf-cleaner/termux-elf-cleaner.cpp',
2020-12-26 16:04:41 -08:00
'-o', elf_cleaner])
args = [elf_cleaner]
2022-03-17 03:15:39 -07:00
args.extend(op.join('native', 'out', arch, bin)
for arch in archs for bin in ['magisk', 'magiskpolicy'])
execv(args)
def binary_dump(src, var_name):
out_str = f'constexpr unsigned char {var_name}[] = {{'
for i, c in enumerate(xz(src.read())):
if i % 16 == 0:
out_str += '\n'
out_str += f'0x{c:02X},'
out_str += '\n};\n'
return out_str
2018-08-10 04:41:21 +08:00
def run_ndk_build(flags):
2020-04-06 22:45:08 -07:00
os.chdir('native')
flags = 'NDK_PROJECT_PATH=. NDK_APPLICATION_MK=src/Application.mk ' + flags
proc = system(f'{ndk_build} {flags} -j{cpu_count}')
if proc.returncode != 0:
error('Build binary failed!')
2020-04-06 22:45:08 -07:00
os.chdir('..')
2022-06-30 14:50:21 -07:00
for arch in archs:
for tgt in support_targets + ['libpreload.so']:
source = op.join('native', 'libs', arch, tgt)
target = op.join('native', 'out', arch, tgt)
mv(source, target)
def run_cargo_build(args):
os.chdir(op.join('native', 'src'))
2022-06-30 14:50:21 -07:00
targets = set(args.target) & set(rust_targets)
env = os.environ.copy()
2022-06-30 18:12:07 -07:00
env['CARGO_BUILD_RUSTC'] = op.join(rust_bin, 'rustc' + EXE_EXT)
2022-06-30 14:50:21 -07:00
# Install cxxbridge and generate C++ bindings
2022-07-24 05:39:14 -07:00
native_out = op.join('..', 'out')
2022-08-17 01:59:23 -07:00
local_cargo_root = op.join(native_out, '.cargo')
cfg = op.join('.cargo', 'config.toml')
cfg_bak = op.join('.cargo', 'config.toml.bak')
2022-08-17 01:59:23 -07:00
try:
# Hide the config file for cargo install
mv(cfg, cfg_bak)
cxx_src = op.join('external', 'cxx-rs', 'gen', 'cmd')
mkdir_p(local_cargo_root)
cmds = [cargo, 'install', '--root', local_cargo_root, '--path', cxx_src]
if not args.verbose:
cmds.append('-q')
proc = execv(cmds, env)
if proc.returncode != 0:
error('cxxbridge-cmd installation failed!')
finally:
# Make sure the config file rename is always reverted
mv(cfg_bak, cfg)
2022-06-30 14:50:21 -07:00
cxxbridge = op.join(local_cargo_root, 'bin', 'cxxbridge' + EXE_EXT)
mkdir(native_gen_path)
for p in ['base', 'boot', 'core', 'init', 'sepolicy']:
text = cmd_out([cxxbridge, op.join(p, 'lib.rs')])
2022-06-30 14:50:21 -07:00
write_if_diff(op.join(native_gen_path, f'{p}-rs.cpp'), text)
text = cmd_out([cxxbridge, '--header', op.join(p, 'lib.rs')])
2022-06-30 14:50:21 -07:00
write_if_diff(op.join(native_gen_path, f'{p}-rs.hpp'), text)
# Start building the actual build commands
cmds = [cargo, 'build', '-Z', 'build-std=std,panic_abort',
'-Z', 'build-std-features=panic_immediate_abort']
for target in targets:
cmds.append('-p')
cmds.append(target)
rust_out = 'debug'
if args.release:
cmds.append('-r')
rust_out = 'release'
if not args.verbose:
cmds.append('-q')
os_name = platform.system().lower()
llvm_bin = op.join(ndk_path, 'toolchains', 'llvm', 'prebuilt', f'{os_name}-x86_64', 'bin')
env['TARGET_CC'] = op.join(llvm_bin, 'clang' + EXE_EXT)
env['RUSTFLAGS'] = '-Clinker-plugin-lto'
for (arch, triple) in zip(archs, triples):
env['TARGET_CFLAGS'] = f'--target={triple}21'
rust_triple = 'thumbv7neon-linux-androideabi' if triple.startswith('armv7') else triple
proc = execv([*cmds, '--target', rust_triple], env)
if proc.returncode != 0:
error('Build binary failed!')
arch_out = op.join(native_out, arch)
mkdir(arch_out)
for tgt in targets:
source = op.join('target', rust_triple, rust_out, f'lib{tgt}.a')
target = op.join(arch_out, f'lib{tgt}-rs.a')
mv(source, target)
os.chdir(op.join('..', '..'))
def write_if_diff(file_name, text):
do_write = True
if op.exists(file_name):
with open(file_name, 'r') as f:
orig = f.read()
do_write = orig != text
if do_write:
with open(file_name, 'w') as f:
f.write(text)
2022-05-15 01:01:54 -07:00
def dump_bin_header(args):
stub = op.join(config['outdir'], f'stub-{"release" if args.release else "debug"}.apk')
2020-04-03 03:33:39 -07:00
if not op.exists(stub):
2020-12-06 06:30:45 +08:00
error('Build stub APK before building "magiskinit"')
mkdir_p(native_gen_path)
with open(stub, 'rb') as src:
text = binary_dump(src, 'manager_xz')
write_if_diff(op.join(native_gen_path, 'binaries.h'), text)
for arch in archs:
preload = op.join('native', 'out', arch, 'libpreload.so')
with open(preload, 'rb') as src:
text = binary_dump(src, 'preload_xz')
write_if_diff(op.join(native_gen_path, f'{arch}_binaries.h'), text)
def dump_flag_header():
flag_txt = textwrap.dedent('''\
#pragma once
#define quote(s) #s
#define str(s) quote(s)
#define MAGISK_FULL_VER MAGISK_VERSION "(" str(MAGISK_VER_CODE) ")"
#define NAME_WITH_VER(name) str(name) " " MAGISK_FULL_VER
''')
flag_txt += f'#define MAGISK_VERSION "{config["version"]}"\n'
flag_txt += f'#define MAGISK_VER_CODE {config["versionCode"]}\n'
flag_txt += f'#define MAGISK_DEBUG {0 if args.release else 1}\n'
mkdir_p(native_gen_path)
write_if_diff(op.join(native_gen_path, 'flags.h'), flag_txt)
2020-03-27 23:23:26 -07:00
2018-05-13 03:04:40 +08:00
def build_binary(args):
2020-04-03 03:33:39 -07:00
# Verify NDK install
2022-04-15 10:04:04 -07:00
try:
with open(op.join(ndk_path, 'ONDK_VERSION'), 'r') as ondk_ver:
assert ondk_ver.read().strip(' \t\r\n') == config['ondkVersion']
except:
error('Unmatched NDK. Please install/upgrade NDK with "build.py ndk"')
2020-04-03 03:33:39 -07:00
if 'target' not in vars(args):
vars(args)['target'] = []
if args.target:
2019-05-30 21:17:58 -07:00
args.target = set(args.target) & set(support_targets)
if not args.target:
return
else:
2019-05-30 21:17:58 -07:00
args.target = default_targets
header('* Building binaries: ' + ' '.join(args.target))
2022-06-30 14:50:21 -07:00
run_cargo_build(args)
dump_flag_header()
2021-05-12 13:40:53 +08:00
flag = ''
if 'magisk' in args.target:
2021-05-12 13:40:53 +08:00
flag += ' B_MAGISK=1'
2022-03-17 03:15:39 -07:00
if 'magiskpolicy' in args.target:
flag += ' B_POLICY=1'
2021-01-18 04:25:26 -08:00
if 'test' in args.target:
2021-05-12 13:40:53 +08:00
flag += ' B_TEST=1'
2021-01-18 04:25:26 -08:00
if 'magiskinit' in args.target:
flag += ' B_PRELOAD=1'
2020-01-21 00:48:52 +08:00
if 'resetprop' in args.target:
2021-01-18 04:25:26 -08:00
flag += ' B_PROP=1'
2020-01-21 00:48:52 +08:00
if 'magiskboot' in args.target:
2021-01-18 04:25:26 -08:00
flag += ' B_BOOT=1'
2021-01-18 04:25:26 -08:00
if flag:
run_ndk_build(flag)
2022-05-30 03:47:31 -07:00
clean_elf()
# magiskinit and busybox has to be built separately
if 'magiskinit' in args.target:
dump_bin_header(args)
run_ndk_build('B_INIT=1')
2021-05-12 13:40:53 +08:00
if 'busybox' in args.target:
run_ndk_build('B_BB=1')
def build_apk(args, module):
2022-05-15 01:01:54 -07:00
build_type = 'Release' if args.release else 'Debug'
2019-03-08 10:19:22 -05:00
proc = execv([gradlew, f'{module}:assemble{build_type}',
2020-12-26 16:04:41 -08:00
'-PconfigPath=' + op.abspath(args.config)])
if proc.returncode != 0:
2019-10-17 18:02:31 -04:00
error(f'Build {module} failed!')
2017-06-03 22:04:22 +08:00
build_type = build_type.lower()
2021-05-13 00:21:04 -07:00
apk = f'{module}-{build_type}.apk'
source = op.join(module, 'build', 'outputs', 'apk', build_type, apk)
target = op.join(config['outdir'], apk)
mv(source, target)
header('Output: ' + target)
2018-08-10 05:54:26 +08:00
2018-08-20 12:02:38 +08:00
def build_app(args):
2021-02-23 23:56:58 -08:00
header('* Building the Magisk app')
build_apk(args, 'app')
2018-05-27 14:55:24 +08:00
2018-08-20 12:02:38 +08:00
def build_stub(args):
header('* Building the stub app')
2020-03-27 23:23:26 -07:00
build_apk(args, 'stub')
2020-07-04 04:09:19 -07:00
2017-06-03 20:19:01 +08:00
def cleanup(args):
support_targets = {'native', 'java'}
if args.target:
args.target = set(args.target) & support_targets
else:
args.target = support_targets
if 'native' in args.target:
header('* Cleaning native')
2020-04-03 03:33:39 -07:00
rm_rf(op.join('native', 'libs'))
rm_rf(op.join('native', 'obj'))
2022-08-17 01:59:23 -07:00
rm_rf(op.join('native', 'out'))
rm_rf(op.join('native', 'src', 'target'))
rm_rf(op.join('native', 'src', 'external', 'cxx-rs', 'target'))
if 'java' in args.target:
header('* Cleaning java')
execv([gradlew, 'app:clean', 'app:shared:clean', 'stub:clean'])
2018-05-13 03:04:40 +08:00
2020-04-03 03:33:39 -07:00
def setup_ndk(args):
os_name = platform.system().lower()
2022-04-15 10:04:04 -07:00
ndk_ver = config['ondkVersion']
url = f'https://github.com/topjohnwu/ondk/releases/download/{ndk_ver}/ondk-{ndk_ver}-{os_name}.tar.gz'
ndk_archive = url.split('/')[-1]
2020-04-03 03:33:39 -07:00
2022-04-15 10:04:04 -07:00
header(f'* Downloading and extracting {ndk_archive}')
with urllib.request.urlopen(url) as response:
with tarfile.open(mode='r|gz', fileobj=response) as tar:
tar.extractall(ndk_root)
2020-04-03 03:33:39 -07:00
rm_rf(ndk_path)
2022-04-15 10:04:04 -07:00
mv(op.join(ndk_root, f'ondk-{ndk_ver}'), ndk_path)
2020-04-03 03:33:39 -07:00
2021-03-02 23:18:44 -08:00
header('* Patching static libs')
2021-10-20 03:17:42 -07:00
for target in ['aarch64-linux-android', 'arm-linux-androideabi',
'i686-linux-android', 'x86_64-linux-android']:
arch = target.split('-')[0]
lib_dir = op.join(
ndk_path, 'toolchains', 'llvm', 'prebuilt', f'{os_name}-x86_64',
'sysroot', 'usr', 'lib', f'{target}', '21')
if not op.exists(lib_dir):
continue
src_dir = op.join('tools', 'ndk-bins', '21', arch)
rm(op.join(src_dir, '.DS_Store'))
2022-04-15 10:04:04 -07:00
shutil.copytree(src_dir, lib_dir, copy_function=cp, dirs_exist_ok=True)
2020-04-03 03:33:39 -07:00
def setup_avd(args):
if not args.skip:
2022-05-19 22:54:49 -07:00
build_all(args)
header('* Setting up emulator')
abi = cmd_out([adb_path, 'shell', 'getprop', 'ro.product.cpu.abi'])
proc = execv([adb_path, 'push', f'native/out/{abi}/busybox', 'scripts/avd_magisk.sh', '/data/local/tmp'])
if proc.returncode != 0:
error('adb push failed!')
apk = 'out/app-release.apk' if args.release else 'out/app-debug.apk'
proc = execv([adb_path, 'push', apk, '/data/local/tmp/magisk.apk'])
if proc.returncode != 0:
error('adb push failed!')
2022-01-19 05:12:11 -08:00
proc = execv([adb_path, 'shell', 'sh', '/data/local/tmp/avd_magisk.sh'])
if proc.returncode != 0:
error('avd_magisk.sh failed!')
def patch_avd_ramdisk(args):
if not args.skip:
2022-05-19 22:54:49 -07:00
args.release = False
build_all(args)
2022-01-19 05:12:11 -08:00
header('* Patching emulator ramdisk.img')
# Create a backup to prevent accidental overwrites
backup = args.ramdisk + '.bak'
if not op.exists(backup):
cp(args.ramdisk, backup)
ini = op.join(op.dirname(args.ramdisk), 'advancedFeatures.ini')
with open(ini, 'r') as f:
adv_ft = f.read()
# Need to turn off system as root
if 'SystemAsRoot = on' in adv_ft:
# Create a backup
cp(ini, ini + '.bak')
adv_ft = adv_ft.replace('SystemAsRoot = on', 'SystemAsRoot = off')
with open(ini, 'w') as f:
f.write(adv_ft)
abi = cmd_out([adb_path, 'shell', 'getprop', 'ro.product.cpu.abi'])
proc = execv([adb_path, 'push', f'native/out/{abi}/busybox', 'scripts/avd_patch.sh', '/data/local/tmp'])
2022-01-19 05:12:11 -08:00
if proc.returncode != 0:
error('adb push failed!')
apk = 'out/app-release.apk' if args.release else 'out/app-debug.apk'
proc = execv([adb_path, 'push', apk, '/data/local/tmp/magisk.apk'])
if proc.returncode != 0:
error('adb push failed!')
2022-02-11 04:34:26 +08:00
proc = execv([adb_path, 'push', backup, '/data/local/tmp/ramdisk.cpio.tmp'])
2022-01-19 05:12:11 -08:00
if proc.returncode != 0:
error('adb push failed!')
proc = execv([adb_path, 'shell', 'sh', '/data/local/tmp/avd_patch.sh'])
if proc.returncode != 0:
error('avd_patch.sh failed!')
proc = execv([adb_path, 'pull', '/data/local/tmp/ramdisk.cpio.gz', args.ramdisk])
if proc.returncode != 0:
2022-01-19 05:12:11 -08:00
error('adb pull failed!')
2018-08-10 05:54:26 +08:00
def build_all(args):
build_stub(args)
build_binary(args)
build_app(args)
2018-08-10 05:54:26 +08:00
2017-06-03 20:19:01 +08:00
parser = argparse.ArgumentParser(description='Magisk build script')
2020-12-26 16:04:41 -08:00
parser.set_defaults(func=lambda x: None)
parser.add_argument('-r', '--release', action='store_true',
help='compile in release mode')
parser.add_argument('-v', '--verbose', action='store_true',
help='verbose output')
parser.add_argument('-c', '--config', default='config.prop',
2020-12-25 05:34:15 -08:00
help='custom config file (default: config.prop)')
2017-06-03 20:19:01 +08:00
subparsers = parser.add_subparsers(title='actions')
all_parser = subparsers.add_parser(
'all', help='build everything')
2017-06-03 20:19:01 +08:00
all_parser.set_defaults(func=build_all)
2018-08-10 05:54:26 +08:00
binary_parser = subparsers.add_parser('binary', help='build binaries')
binary_parser.add_argument(
'target', nargs='*', help=f"{', '.join(support_targets)}, \
2019-05-30 21:17:58 -07:00
or empty for defaults ({', '.join(default_targets)})")
2017-06-03 20:19:01 +08:00
binary_parser.set_defaults(func=build_binary)
2021-02-23 23:56:58 -08:00
app_parser = subparsers.add_parser('app', help='build the Magisk app')
app_parser.set_defaults(func=build_app)
2017-06-03 20:19:01 +08:00
stub_parser = subparsers.add_parser('stub', help='build the stub app')
2018-08-10 05:15:39 +08:00
stub_parser.set_defaults(func=build_stub)
avd_parser = subparsers.add_parser(
2022-01-19 05:12:11 -08:00
'emulator', help='setup AVD for development')
avd_parser.add_argument('-s', '--skip', action='store_true',
help='skip building binaries and the app')
avd_parser.set_defaults(func=setup_avd)
2022-01-19 05:12:11 -08:00
avd_patch_parser = subparsers.add_parser(
'avd_patch', help='patch AVD ramdisk.img')
avd_patch_parser.add_argument('ramdisk', help='path to ramdisk.img')
avd_patch_parser.add_argument('-s', '--skip', action='store_true',
help='skip building binaries and the app')
2022-01-19 05:12:11 -08:00
avd_patch_parser.set_defaults(func=patch_avd_ramdisk)
clean_parser = subparsers.add_parser('clean', help='cleanup')
clean_parser.add_argument(
'target', nargs='*', help='native, java, or empty to clean both')
2017-06-03 20:19:01 +08:00
clean_parser.set_defaults(func=cleanup)
2020-04-03 03:33:39 -07:00
ndk_parser = subparsers.add_parser('ndk', help='setup Magisk NDK')
ndk_parser.set_defaults(func=setup_ndk)
2017-06-03 20:19:01 +08:00
if len(sys.argv) == 1:
parser.print_help()
sys.exit(1)
2017-06-03 20:19:01 +08:00
args = parser.parse_args()
load_config(args)
# Call corresponding functions
2017-06-03 20:19:01 +08:00
args.func(args)