Magisk/build.py

776 lines
21 KiB
Python
Raw Permalink 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
2023-05-28 17:30:20 -07:00
import glob
2020-12-26 16:04:41 -08:00
import lzma
2021-05-12 13:40:53 +08:00
import multiprocessing
import os
import platform
import re
2021-05-12 13:40:53 +08:00
import shutil
import stat
2021-05-12 13:40:53 +08:00
import subprocess
import sys
2023-05-28 17:30:20 -07:00
import tarfile
import textwrap
2021-05-12 13:40:53 +08:00
import urllib.request
2024-05-08 21:50:59 -07:00
from pathlib import Path
2023-05-02 16:28:02 -07:00
from zipfile import ZipFile
2017-11-12 04:17:56 +08:00
2023-05-11 21:01:04 -07:00
def color_print(code, str):
2022-08-17 01:59:23 -07:00
if no_color:
2023-05-16 02:07:49 -07:00
print(str)
2020-12-25 15:54:47 -08:00
else:
2023-05-16 02:07:49 -07:00
str = str.replace("\n", f"\033[0m\n{code}")
2023-05-11 21:01:04 -07:00
print(f"{code}{str}\033[0m")
def error(str):
color_print("\033[41;39m", f"\n! {str}\n")
sys.exit(1)
2017-06-03 20:19:01 +08:00
def header(str):
2023-05-11 21:01:04 -07:00
color_print("\033[44;39m", f"\n{str}\n")
2017-06-03 20:19:01 +08:00
2018-08-06 02:01:04 +08:00
def vprint(str):
if args.verbose > 0:
print(str)
2018-08-06 02:01:04 +08:00
2024-08-03 00:05:49 -07:00
# Environment checks and detection
2023-05-11 02:39:57 -07:00
is_windows = os.name == "nt"
EXE_EXT = ".exe" if is_windows else ""
2020-12-26 16:04:41 -08:00
2023-04-25 04:46:25 +08:00
no_color = False
if is_windows:
2022-08-17 01:59:23 -07:00
try:
import colorama
2023-05-11 02:39:57 -07:00
2022-08-17 01:59:23 -07:00
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
2022-12-26 00:09:27 -08:00
if not sys.version_info >= (3, 8):
2023-05-11 02:39:57 -07:00
error("Requires Python 3.8+")
2017-06-03 20:19:01 +08:00
try:
sdk_path = Path(os.environ["ANDROID_HOME"])
except KeyError:
try:
sdk_path = Path(os.environ["ANDROID_SDK_ROOT"])
except KeyError:
error("Please set Android SDK path to environment variable ANDROID_HOME")
2017-06-03 20:19:01 +08:00
2018-05-13 03:04:40 +08:00
cpu_count = multiprocessing.cpu_count()
os_name = platform.system().lower()
# Common constants
support_abis = {
"armeabi-v7a": "thumbv7neon-linux-androideabi",
"x86": "i686-linux-android",
"arm64-v8a": "aarch64-linux-android",
"x86_64": "x86_64-linux-android",
"riscv64": "riscv64-linux-android",
}
2024-07-24 16:37:22 -07:00
default_targets = {"magisk", "magiskinit", "magiskboot", "magiskpolicy"}
support_targets = default_targets | {"resetprop"}
rust_targets = {"magisk", "magiskinit", "magiskboot", "magiskpolicy"}
2023-05-11 02:39:57 -07:00
# Common paths
2024-05-08 21:50:59 -07:00
ndk_root = sdk_path / "ndk"
ndk_path = ndk_root / "magisk"
ndk_build = ndk_path / "ndk-build"
rust_bin = ndk_path / "toolchains" / "rust" / "bin"
llvm_bin = ndk_path / "toolchains" / "llvm" / "prebuilt" / f"{os_name}-x86_64" / "bin"
2024-07-31 17:21:18 -07:00
cargo = rust_bin / "cargo"
gradlew = Path.cwd() / "gradlew"
adb_path = sdk_path / "platform-tools" / "adb"
2024-05-08 21:50:59 -07:00
native_gen_path = Path("native", "out", "generated").resolve()
2018-05-13 03:04:40 +08:00
2019-10-17 18:02:31 -04:00
# Global vars
config = {}
args = {}
build_abis = {}
2020-12-26 16:04:41 -08:00
2024-08-03 00:05:49 -07:00
###################
# Helper functions
###################
2024-05-08 21:50:59 -07:00
def mv(source: Path, target: Path):
try:
shutil.move(source, target)
2023-05-11 02:39:57 -07:00
vprint(f"mv {source} -> {target}")
2019-02-11 17:14:29 -05:00
except:
pass
2024-05-08 21:50:59 -07:00
def cp(source: Path, target: Path):
try:
shutil.copyfile(source, target)
2023-05-11 02:39:57 -07:00
vprint(f"cp {source} -> {target}")
2019-02-11 17:14:29 -05:00
except:
pass
2024-05-08 21:50:59 -07:00
def rm(file: Path):
try:
os.remove(file)
2023-05-11 02:39:57 -07:00
vprint(f"rm {file}")
2023-06-15 05:57:19 -07:00
except FileNotFoundError as e:
pass
2017-06-03 20:19:01 +08:00
def rm_on_error(func, path, _):
2023-06-15 05:57:19 -07:00
# Removing a read-only file on Windows will get "WindowsError: [Error 5] Access is denied"
# Clear the "read-only" bit and retry
try:
os.chmod(path, stat.S_IWRITE)
os.unlink(path)
except FileNotFoundError as e:
pass
2024-05-08 21:50:59 -07:00
def rm_rf(path: Path):
2023-05-11 02:39:57 -07:00
vprint(f"rm -rf {path}")
if sys.version_info >= (3, 12):
shutil.rmtree(path, ignore_errors=False, onexc=rm_on_error)
else:
shutil.rmtree(path, ignore_errors=False, onerror=rm_on_error)
2020-04-03 03:33:39 -07:00
2024-07-31 16:57:30 -07:00
def execv(cmds: list, env=None):
out = None if args.force_out or args.verbose > 0 else subprocess.DEVNULL
2024-07-31 17:21:18 -07:00
# Use shell on Windows to support PATHEXT
return subprocess.run(cmds, stdout=out, env=env, shell=is_windows)
2024-08-03 01:28:53 -07:00
def cmd_out(cmds: list):
2023-05-11 02:39:57 -07:00
return (
2024-07-31 17:21:18 -07:00
subprocess.run(
cmds,
stdout=subprocess.PIPE,
stderr=subprocess.DEVNULL,
shell=is_windows,
)
2023-05-11 02:39:57 -07:00
.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)
2024-08-03 00:05:49 -07:00
###############
# Build Native
###############
def clean_elf():
2019-10-17 18:02:31 -04:00
if is_windows:
2024-05-08 21:50:59 -07:00
elf_cleaner = Path("tools", "elf-cleaner.exe")
else:
2024-05-08 21:50:59 -07:00
elf_cleaner = Path("native", "out", "elf-cleaner")
if not elf_cleaner.exists():
2023-05-11 02:39:57 -07:00
execv(
[
"gcc",
'-DPACKAGE_NAME="termux-elf-cleaner"',
'-DPACKAGE_VERSION="2.1.1"',
'-DCOPYRIGHT="Copyright (C) 2022 Termux."',
"tools/termux-elf-cleaner/elf-cleaner.cpp",
"tools/termux-elf-cleaner/arghandling.c",
"-o",
elf_cleaner,
]
)
2024-07-31 16:57:30 -07:00
cmds = [elf_cleaner, "--api-level", "23"]
cmds.extend(glob.glob("native/out/*/magisk"))
cmds.extend(glob.glob("native/out/*/magiskpolicy"))
execv(cmds)
2024-08-03 00:05:49 -07:00
def run_ndk_build(cmds: list):
2023-05-11 02:39:57 -07:00
os.chdir("native")
2024-07-31 16:57:30 -07:00
cmds.append("NDK_PROJECT_PATH=.")
cmds.append("NDK_APPLICATION_MK=src/Application.mk")
cmds.append(f"APP_ABI={' '.join(build_abis.keys())}")
2024-07-31 16:57:30 -07:00
cmds.append(f"-j{cpu_count}")
if args.verbose > 1:
2024-07-31 16:57:30 -07:00
cmds.append("V=1")
if not args.release:
cmds.append("MAGISK_DEBUG=1")
proc = execv([ndk_build, *cmds])
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("Build binary failed!")
os.chdir("..")
2024-07-24 16:37:22 -07:00
2024-08-03 01:28:53 -07:00
for arch in build_abis.keys():
2024-07-24 16:37:22 -07:00
arch_dir = Path("native", "libs", arch)
2024-08-03 01:28:53 -07:00
out_dir = Path("native", "out", arch)
for source in arch_dir.iterdir():
target = out_dir / source.name
mv(source, target)
2022-06-30 14:50:21 -07:00
2024-08-03 00:05:49 -07:00
def build_cpp_src(targets: set):
2024-07-24 16:37:22 -07:00
dump_flag_header()
2024-07-31 16:57:30 -07:00
cmds = []
2024-07-24 16:37:22 -07:00
clean = False
if "magisk" in targets:
2024-07-31 16:57:30 -07:00
cmds.append("B_MAGISK=1")
2024-07-24 16:37:22 -07:00
clean = True
if "magiskpolicy" in targets:
2024-07-31 16:57:30 -07:00
cmds.append("B_POLICY=1")
2024-07-24 16:37:22 -07:00
clean = True
if "magiskinit" in targets:
2024-07-31 16:57:30 -07:00
cmds.append("B_PRELOAD=1")
2024-07-24 16:37:22 -07:00
if "resetprop" in targets:
2024-07-31 16:57:30 -07:00
cmds.append("B_PROP=1")
2024-07-24 16:37:22 -07:00
2024-07-31 16:57:30 -07:00
if cmds:
2024-08-03 00:05:49 -07:00
run_ndk_build(cmds)
2024-07-24 16:37:22 -07:00
2024-07-31 16:57:30 -07:00
cmds.clear()
2024-07-24 16:37:22 -07:00
if "magiskinit" in targets:
2024-07-31 16:57:30 -07:00
cmds.append("B_INIT=1")
2024-07-24 16:37:22 -07:00
if "magiskboot" in targets:
2024-07-31 16:57:30 -07:00
cmds.append("B_BOOT=1")
2024-07-24 16:37:22 -07:00
2024-07-31 16:57:30 -07:00
if cmds:
cmds.append("B_CRT0=1")
2024-08-03 00:05:49 -07:00
run_ndk_build(cmds)
2024-07-24 16:37:22 -07:00
if clean:
clean_elf()
def run_cargo(cmds):
env = os.environ.copy()
2023-06-15 05:57:19 -07:00
env["PATH"] = f'{rust_bin}{os.pathsep}{env["PATH"]}'
2024-05-08 21:50:59 -07:00
env["CARGO_BUILD_RUSTC"] = str(rust_bin / f"rustc{EXE_EXT}")
2024-08-03 01:28:53 -07:00
env["CARGO_BUILD_RUSTFLAGS"] = f"-Z threads={min(8, cpu_count)}"
return execv([cargo, *cmds], env)
2024-08-03 00:05:49 -07:00
def build_rust_src(targets: set):
2024-07-24 16:37:22 -07:00
targets = targets.copy()
if "resetprop" in targets:
2023-05-11 02:39:57 -07:00
targets.add("magisk")
2024-07-24 16:37:22 -07:00
targets = targets & rust_targets
if not targets:
2023-08-30 16:23:27 -07:00
return
2024-07-24 16:37:22 -07:00
os.chdir(Path("native", "src"))
2024-08-03 01:28:53 -07:00
# Start building the build commands
2023-10-17 13:29:15 -07:00
cmds = ["build", "-p", ""]
2022-06-30 14:50:21 -07:00
if args.release:
2023-05-11 02:39:57 -07:00
cmds.append("-r")
2024-08-03 01:28:53 -07:00
profile = "release"
else:
profile = "debug"
if args.verbose == 0:
2023-05-11 02:39:57 -07:00
cmds.append("-q")
elif args.verbose > 1:
cmds.append("--verbose")
2022-06-30 14:50:21 -07:00
2024-08-03 01:28:53 -07:00
for triple in build_abis.values():
cmds.append("--target")
cmds.append(triple)
2024-08-03 01:28:53 -07:00
for tgt in targets:
cmds[2] = tgt
proc = run_cargo(cmds)
if proc.returncode != 0:
error("Build binary failed!")
2023-10-17 13:29:15 -07:00
2024-08-03 01:28:53 -07:00
os.chdir(Path("..", ".."))
2022-06-30 14:50:21 -07:00
2024-08-03 01:28:53 -07:00
native_out = Path("native", "out")
rust_out = native_out / "rust"
for arch, triple in build_abis.items():
2024-05-08 21:50:59 -07:00
arch_out = native_out / arch
arch_out.mkdir(mode=0o755, exist_ok=True)
2022-06-30 14:50:21 -07:00
for tgt in targets:
2024-08-03 01:28:53 -07:00
source = rust_out / triple / profile / f"lib{tgt}.a"
2024-05-08 21:50:59 -07:00
target = arch_out / f"lib{tgt}-rs.a"
2022-06-30 14:50:21 -07:00
mv(source, target)
2024-05-08 21:50:59 -07:00
def write_if_diff(file_name: Path, text: str):
do_write = True
2024-05-08 21:50:59 -07:00
if file_name.exists():
2023-05-11 02:39:57 -07:00
with open(file_name, "r") as f:
orig = f.read()
do_write = orig != text
if do_write:
2023-05-11 02:39:57 -07:00
with open(file_name, "w") as f:
f.write(text)
def dump_flag_header():
2023-05-11 02:39:57 -07:00
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
2023-05-11 02:39:57 -07:00
"""
)
flag_txt += f'#define MAGISK_VERSION "{config["version"]}"\n'
flag_txt += f'#define MAGISK_VER_CODE {config["versionCode"]}\n'
2023-05-11 02:39:57 -07:00
flag_txt += f"#define MAGISK_DEBUG {0 if args.release else 1}\n"
2024-05-08 21:50:59 -07:00
native_gen_path.mkdir(mode=0o755, parents=True, exist_ok=True)
write_if_diff(Path(native_gen_path, "flags.h"), flag_txt)
2020-03-27 23:23:26 -07:00
2024-08-03 00:05:49 -07:00
def build_native():
2020-04-03 03:33:39 -07:00
# Verify NDK install
2022-04-15 10:04:04 -07:00
try:
2024-05-08 21:50:59 -07:00
with open(Path(ndk_path, "ONDK_VERSION"), "r") as ondk_ver:
2023-05-11 02:39:57 -07:00
assert ondk_ver.read().strip(" \t\r\n") == config["ondkVersion"]
2022-04-15 10:04:04 -07:00
except:
error('Unmatched NDK. Please install/upgrade NDK with "build.py ndk"')
2020-04-03 03:33:39 -07:00
2024-07-24 16:37:22 -07:00
if "targets" not in vars(args) or not args.targets:
targets = default_targets
else:
2024-07-24 16:37:22 -07:00
targets = set(args.targets) & support_targets
if not targets:
return
2024-08-03 00:05:49 -07:00
header("* Building: " + " ".join(targets))
if sccache := shutil.which("sccache"):
os.environ["RUSTC_WRAPPER"] = sccache
os.environ["NDK_CCACHE"] = sccache
os.environ["CARGO_INCREMENTAL"] = "0"
if ccache := shutil.which("ccache"):
os.environ["NDK_CCACHE"] = ccache
2024-08-03 00:05:49 -07:00
build_rust_src(targets)
build_cpp_src(targets)
############
# Build App
############
2021-05-12 13:40:53 +08:00
2023-05-11 21:01:04 -07:00
def find_jdk():
env = os.environ.copy()
if "ANDROID_STUDIO" in env:
studio = env["ANDROID_STUDIO"]
2024-05-08 21:50:59 -07:00
jbr = Path(studio, "jbr", "bin")
if not jbr.exists():
jbr = Path(studio, "Contents", "jbr", "Contents", "Home", "bin")
if jbr.exists():
2023-06-05 02:27:02 -07:00
env["PATH"] = f'{jbr}{os.pathsep}{env["PATH"]}'
2023-05-11 21:01:04 -07:00
no_jdk = False
try:
proc = subprocess.run(
2023-06-05 02:27:02 -07:00
"javac -version",
2023-05-11 21:01:04 -07:00
stdout=subprocess.DEVNULL,
stderr=subprocess.DEVNULL,
env=env,
2023-06-15 01:26:54 -07:00
shell=True,
2023-05-11 21:01:04 -07:00
)
no_jdk = proc.returncode != 0
except FileNotFoundError:
no_jdk = True
if no_jdk:
error(
"Please set Android Studio's path to environment variable ANDROID_STUDIO,\n"
+ "or install JDK 17 and make sure 'javac' is available in PATH"
)
return env
2024-08-03 00:05:49 -07:00
def build_apk(module: str):
2023-05-11 21:01:04 -07:00
env = find_jdk()
2023-05-11 02:39:57 -07:00
2023-05-11 21:01:04 -07:00
build_type = "Release" if args.release else "Debug"
2023-05-11 02:39:57 -07:00
proc = execv(
[
gradlew,
f"{module}:assemble{build_type}",
2024-05-08 21:50:59 -07:00
f"-PconfigPath={args.config.resolve()}",
2023-05-11 21:01:04 -07:00
],
env=env,
2023-05-11 02:39:57 -07:00
)
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error(f"Build {module} failed!")
2017-06-03 22:04:22 +08:00
build_type = build_type.lower()
2024-07-04 00:21:34 -07:00
paths = module.split(":")
apk = f"{paths[-1]}-{build_type}.apk"
source = Path(*paths, "build", "outputs", "apk", build_type, apk)
2024-05-08 21:50:59 -07:00
target = config["outdir"] / apk
2021-05-13 00:21:04 -07:00
mv(source, target)
2024-05-08 21:50:59 -07:00
header(f"Output: {target}")
2018-08-10 05:54:26 +08:00
2024-08-03 00:05:49 -07:00
def build_app():
2023-05-11 02:39:57 -07:00
header("* Building the Magisk app")
2024-08-03 00:05:49 -07:00
build_apk(":app:apk")
2024-07-04 02:27:20 -07:00
build_type = "release" if args.release else "debug"
# Rename apk-variant.apk to app-variant.apk
source = config["outdir"] / f"apk-{build_type}.apk"
target = config["outdir"] / f"app-{build_type}.apk"
mv(source, target)
2022-12-26 15:23:06 -08:00
# Stub building is directly integrated into the main app
# build process. Copy the stub APK into output directory.
source = Path("app", "core", "src", build_type, "assets", "stub.apk")
2024-07-04 02:27:20 -07:00
target = config["outdir"] / f"stub-{build_type}.apk"
2022-12-26 15:23:06 -08:00
cp(source, target)
2024-08-03 00:05:49 -07:00
def build_stub():
2023-05-11 02:39:57 -07:00
header("* Building the stub app")
2024-08-03 00:05:49 -07:00
build_apk(":app:stub")
2022-12-26 15:23:06 -08:00
2024-08-03 00:05:49 -07:00
################
# Build General
################
2018-05-27 14:55:24 +08:00
2024-08-03 00:05:49 -07:00
def cleanup():
support_targets = {"native", "cpp", "rust", "app"}
2024-07-24 16:37:22 -07:00
if args.targets:
targets = set(args.targets) & support_targets
if "native" in targets:
targets.add("cpp")
targets.add("rust")
else:
2024-07-24 16:37:22 -07:00
targets = support_targets
2024-07-24 16:37:22 -07:00
if "cpp" in targets:
2023-06-20 14:50:02 -07:00
header("* Cleaning C++")
2024-05-08 21:50:59 -07:00
rm_rf(Path("native", "libs"))
rm_rf(Path("native", "obj"))
2023-06-20 14:50:02 -07:00
2024-07-24 16:37:22 -07:00
if "rust" in targets:
2023-06-20 14:50:02 -07:00
header("* Cleaning Rust")
2024-05-08 21:50:59 -07:00
rm_rf(Path("native", "src", "target"))
rm(Path("native", "src", "boot", "proto", "mod.rs"))
rm(Path("native", "src", "boot", "proto", "update_metadata.rs"))
2023-05-28 17:30:20 -07:00
for rs_gen in glob.glob("native/**/*-rs.*pp", recursive=True):
rm(rs_gen)
2024-07-31 16:57:30 -07:00
if "native" in targets:
rm_rf(Path("native", "out"))
2024-08-03 00:05:49 -07:00
if "app" in targets:
header("* Cleaning app")
2024-07-24 16:46:47 -07:00
execv([gradlew, ":app:clean"], env=find_jdk())
2018-05-13 03:04:40 +08:00
2024-08-03 00:05:49 -07:00
def build_all():
build_native()
build_app()
############
# Utilities
############
def cargo_cli():
args.force_out = True
if len(args.commands) >= 1 and args.commands[0] == "--":
args.commands = args.commands[1:]
os.chdir(Path("native", "src"))
run_cargo(args.commands)
os.chdir(Path("..", ".."))
def setup_ndk():
2023-05-11 02:39:57 -07:00
ndk_ver = config["ondkVersion"]
2023-06-15 01:26:54 -07:00
url = f"https://github.com/topjohnwu/ondk/releases/download/{ndk_ver}/ondk-{ndk_ver}-{os_name}.tar.xz"
2023-05-11 02:39:57 -07:00
ndk_archive = url.split("/")[-1]
2024-05-08 21:50:59 -07:00
ondk_path = Path(ndk_root, f"ondk-{ndk_ver}")
2020-04-03 03:33:39 -07:00
2023-05-11 02:39:57 -07:00
header(f"* Downloading and extracting {ndk_archive}")
2023-06-15 05:57:19 -07:00
rm_rf(ondk_path)
2022-04-15 10:04:04 -07:00
with urllib.request.urlopen(url) as response:
2023-06-15 01:26:54 -07:00
with tarfile.open(mode="r|xz", fileobj=response) as tar:
2024-06-18 03:36:16 -07:00
if hasattr(tarfile, "data_filter"):
tar.extractall(ndk_root, filter="tar")
else:
tar.extractall(ndk_root)
2020-04-03 03:33:39 -07:00
rm_rf(ndk_path)
2023-06-15 05:57:19 -07:00
mv(ondk_path, ndk_path)
2020-04-03 03:33:39 -07:00
2024-08-03 00:05:49 -07:00
def push_files(script):
2023-05-11 02:39:57 -07:00
abi = cmd_out([adb_path, "shell", "getprop", "ro.product.cpu.abi"])
2024-07-20 03:10:18 -07:00
if not abi:
error("Cannot detect emulator ABI")
2024-05-08 21:50:59 -07:00
apk = Path(
config["outdir"], ("app-release.apk" if args.release else "app-debug.apk")
)
2023-05-02 16:28:02 -07:00
# Extract busybox from APK
2024-05-08 21:50:59 -07:00
busybox = Path(config["outdir"], "busybox")
2023-05-02 16:28:02 -07:00
with ZipFile(apk) as zf:
2023-05-11 02:39:57 -07:00
with zf.open(f"lib/{abi}/libbusybox.so") as libbb:
with open(busybox, "wb") as bb:
2023-05-02 16:28:02 -07:00
bb.write(libbb.read())
2023-05-02 16:28:02 -07:00
try:
2023-05-11 02:39:57 -07:00
proc = execv([adb_path, "push", busybox, script, "/data/local/tmp"])
2023-05-02 16:28:02 -07:00
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("adb push failed!")
2023-05-02 16:28:02 -07:00
finally:
rm_rf(busybox)
2023-05-11 02:39:57 -07:00
proc = execv([adb_path, "push", apk, "/data/local/tmp/magisk.apk"])
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("adb push failed!")
2023-05-02 16:28:02 -07:00
2024-08-03 00:05:49 -07:00
def setup_avd():
2023-05-02 16:28:02 -07:00
if not args.skip:
2024-08-03 00:05:49 -07:00
build_all()
2023-05-02 16:28:02 -07:00
2023-05-11 02:39:57 -07:00
header("* Setting up emulator")
2023-05-02 16:28:02 -07:00
2024-08-03 00:05:49 -07:00
push_files(Path("scripts", "avd_magisk.sh"))
2023-05-02 16:28:02 -07:00
2023-05-11 02:39:57 -07:00
proc = execv([adb_path, "shell", "sh", "/data/local/tmp/avd_magisk.sh"])
2022-01-19 05:12:11 -08:00
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("avd_magisk.sh failed!")
2022-01-19 05:12:11 -08:00
2024-08-03 00:05:49 -07:00
def patch_avd_file():
if not args.skip:
2024-08-03 00:05:49 -07:00
build_all()
2022-01-19 05:12:11 -08:00
2024-07-24 16:37:22 -07:00
input = Path(args.image)
if args.output:
2024-07-24 16:37:22 -07:00
output = Path(args.output)
else:
2024-07-24 16:37:22 -07:00
output = input.parent / f"{input.name}.magisk"
src_file = f"/data/local/tmp/{input.name}"
out_file = f"{src_file}.magisk"
2022-01-19 05:12:11 -08:00
2024-07-24 16:37:22 -07:00
header(f"* Patching {input.name}")
2022-01-19 05:12:11 -08:00
2024-08-03 00:05:49 -07:00
push_files(Path("scripts", "avd_patch.sh"))
2024-07-24 16:37:22 -07:00
proc = execv([adb_path, "push", input, "/data/local/tmp"])
2022-01-19 05:12:11 -08:00
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("adb push failed!")
2022-01-19 05:12:11 -08:00
proc = execv([adb_path, "shell", "sh", "/data/local/tmp/avd_patch.sh", src_file])
2022-01-19 05:12:11 -08:00
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("avd_patch.sh failed!")
2022-01-19 05:12:11 -08:00
2024-07-24 16:37:22 -07:00
proc = execv([adb_path, "pull", out_file, output])
if proc.returncode != 0:
2023-05-11 02:39:57 -07:00
error("adb pull failed!")
2024-07-24 16:37:22 -07:00
header(f"Output: {output}")
2024-08-03 00:05:49 -07:00
def setup_rustup():
2024-05-09 00:34:12 -07:00
wrapper_dir = Path(args.wrapper_dir)
rm_rf(wrapper_dir)
wrapper_dir.mkdir(mode=0o755, parents=True, exist_ok=True)
if "CARGO_HOME" in os.environ:
cargo_home = Path(os.environ["CARGO_HOME"])
else:
cargo_home = Path.home() / ".cargo"
cargo_bin = cargo_home / "bin"
for src in cargo_bin.iterdir():
tgt = wrapper_dir / src.name
tgt.symlink_to(src)
# Build rustup_wrapper
wrapper_src = Path("tools", "rustup_wrapper")
cargo_toml = wrapper_src / "Cargo.toml"
2024-08-03 01:28:53 -07:00
cmds = ["build", "--release", f"--manifest-path={cargo_toml}"]
if args.verbose > 1:
cmds.append("--verbose")
run_cargo(cmds)
# Replace rustup with wrapper
wrapper = wrapper_dir / (f"rustup{EXE_EXT}")
wrapper.unlink(missing_ok=True)
cp(wrapper_src / "target" / "release" / (f"rustup_wrapper{EXE_EXT}"), wrapper)
2024-05-09 00:34:12 -07:00
wrapper.chmod(0o755)
2024-08-03 00:05:49 -07:00
##################
# Config and args
##################
2017-06-03 20:19:01 +08:00
2024-08-03 00:05:49 -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
key = prop[0].strip(" \t\r\n")
value = prop[1].strip(" \t\r\n")
if not key or not value:
continue
props[key] = value
return props
def load_config():
commit_hash = cmd_out(["git", "rev-parse", "--short=8", "HEAD"])
# Default values
config["version"] = commit_hash
config["versionCode"] = 1000000
config["outdir"] = "out"
args.config = Path(args.config)
# Load prop files
if args.config.exists():
config.update(parse_props(args.config))
if Path("gradle.properties").exists():
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')
config["outdir"] = Path(config["outdir"])
config["outdir"].mkdir(mode=0o755, parents=True, exist_ok=True)
if "abiList" in config:
abiList = re.split("\\s*,\\s*", config["abiList"])
archs = set(abiList) & support_abis.keys()
else:
archs = {"armeabi-v7a", "x86", "arm64-v8a", "x86_64"}
triples = map(support_abis.get, archs)
global build_abis
build_abis = dict(zip(archs, triples))
def parse_args():
parser = argparse.ArgumentParser(description="Magisk build script")
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="count", default=0, help="verbose output"
)
parser.add_argument(
"-c",
"--config",
default="config.prop",
help="custom config file (default: config.prop)",
)
subparsers = parser.add_subparsers(title="actions")
all_parser = subparsers.add_parser("all", help="build everything")
native_parser = subparsers.add_parser("native", help="build native binaries")
native_parser.add_argument(
"targets",
nargs="*",
help=f"{', '.join(support_targets)}, \
or empty for defaults ({', '.join(default_targets)})",
)
app_parser = subparsers.add_parser("app", help="build the Magisk app")
stub_parser = subparsers.add_parser("stub", help="build the stub app")
clean_parser = subparsers.add_parser("clean", help="cleanup")
clean_parser.add_argument(
"targets", nargs="*", help="native, cpp, rust, java, or empty to clean all"
)
ndk_parser = subparsers.add_parser("ndk", help="setup Magisk NDK")
emu_parser = subparsers.add_parser("emulator", help="setup AVD for development")
emu_parser.add_argument(
"-s", "--skip", action="store_true", help="skip building binaries and the app"
)
avd_patch_parser = subparsers.add_parser(
"avd_patch", help="patch AVD ramdisk.img or init_boot.img"
)
avd_patch_parser.add_argument("image", help="path to ramdisk.img or init_boot.img")
avd_patch_parser.add_argument("output", help="optional output file name", nargs="?")
avd_patch_parser.add_argument(
"-s", "--skip", action="store_true", help="skip building binaries and the app"
)
cargo_parser = subparsers.add_parser(
"cargo", help="call 'cargo' commands against the project"
)
cargo_parser.add_argument("commands", nargs=argparse.REMAINDER)
rustup_parser = subparsers.add_parser("rustup", help="setup rustup wrapper")
rustup_parser.add_argument(
"wrapper_dir", help="path to setup rustup wrapper binaries"
)
# Set callbacks
all_parser.set_defaults(func=build_all)
native_parser.set_defaults(func=build_native)
cargo_parser.set_defaults(func=cargo_cli)
rustup_parser.set_defaults(func=setup_rustup)
app_parser.set_defaults(func=build_app)
stub_parser.set_defaults(func=build_stub)
emu_parser.set_defaults(func=setup_avd)
avd_patch_parser.set_defaults(func=patch_avd_file)
clean_parser.set_defaults(func=cleanup)
ndk_parser.set_defaults(func=setup_ndk)
if len(sys.argv) == 1:
parser.print_help()
sys.exit(1)
return parser.parse_args()
args = parse_args()
load_config()
vars(args)["force_out"] = False
args.func()