aboutsummaryrefslogtreecommitdiff
path: root/scripts/run-ckati.sh
blob: b670c8af8883cfdcacfae7a8931b9a2b6dff0e26 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
#! /bin/bash -eu

# Run CKati step separately, tracing given Makefile variables.
# It is expected that the regular Android null build (`m nothing`)
# has been run so that $OUT_DIR/soong/Android-${TARGET_PRODUCT}.mk,
# $OUT_DIR/soong/make_vars-${TARGET_PRODUCT}.mk, etc. files exist.
#
# The output file is in JSON format and can be processed with, say,
# `jq`. For instance, the following invocation outputs all assignment
# traces concisely:
#  jq -c  '.assignments[] | (select (.operation == "assign")) | {("n"): .name, ("l"): .value_stack[0]?, ("v"): .value }' out/ckati.trace
# generates
#  {"n":"<var1>","l":"<file>:<line>","v":"<value1>"}
#  ...

function die() { format=$1; shift; printf "$format\n" $@; exit 1; }
function usage() { die "Usage: %s [-o FILE] VAR ...\n(without -o the output goes to ${outfile})"  ${0##*/}; }

[[ -d build/soong ]] || die "run this script from the top of the Android source tree"
declare -r out=${OUT_DIR:-out}
[[ -x ${out}/soong_ui ]] || die "run Android build first"
: ${TARGET_PRODUCT:?not set, run lunch?}
: ${TARGET_BUILD_VARIANT:?not set, run lunch?}
declare -r androidmk=${out}/soong/Android-${TARGET_PRODUCT}.mk
declare -r makevarsmk=${out}/soong/make_vars-${TARGET_PRODUCT}.mk
declare -r target_device_dir=$(${out}/soong_ui --dumpvar-mode TARGET_DEVICE_DIR)
: ${target_device_dir:?cannot find device directory for ${TARGET_PRODUCT}}
declare -r target_device=$(${out}/soong_ui --dumpvar-mode TARGET_DEVICE)
: ${target_device:?cannot find target device for ${TARGET_PRODUCT}}
declare -r timestamp_file=${out}/build_date.txt
# Files should exist, so ls should succeed:
ls -1d "$androidmk" "$makevarsmk" "$target_device_dir" "$timestamp_file" >/dev/null

outfile=${out}/ckati.trace
while getopts "ho:" opt; do
  case $opt in
    h) usage ;;
    o) outfile=$OPTARG ;;
    ?) usage ;;
  esac
done

if (($#>0)); then
  declare -a tracing=(--variable_assignment_trace_filter="$*" --dump_variable_assignment_trace "$outfile")
else
  printf "running ckati without tracing variables\n"
fi

# Touch one input for ckati, otherwise it will just print
# 'No need to regenerate ninja file' and exit.
touch "$androidmk"
prebuilts/build-tools/linux-x86/bin/ckati \
  --gen_all_targets \
  -i \
  --ignore_optional_include=out/%.P \
  --ninja \
  --ninja_dir=out \
  --ninja_suffix=-${TARGET_PRODUCT} \
  --no_builtin_rules \
  --no_ninja_prelude \
  --regen \
  --top_level_phony \
  --use_find_emulator \
  --use_ninja_phony_output \
  --use_ninja_symlink_outputs \
  --werror_find_emulator \
  --werror_implicit_rules \
  --werror_overriding_commands \
  --werror_phony_looks_real \
  --werror_real_to_phony \
  --werror_suffix_rules \
  --werror_writable \
  --writable out/ \
  -f build/make/core/main.mk \
  "${tracing[@]}" \
  ANDROID_JAVA_HOME=prebuilts/jdk/jdk17/linux-x86 \
  ASAN_SYMBOLIZER_PATH=$PWD/prebuilts/clang/host/linux-x86/llvm-binutils-stable/llvm-symbolizer \
  BUILD_DATETIME_FILE="$timestamp_file" \
  BUILD_HOSTNAME=$(hostname) \
  BUILD_USERNAME="$USER" \
  JAVA_HOME=$PWD/prebuilts/jdk/jdk17/linux-x86 \
  KATI_PACKAGE_MK_DIR="{$out}/target/product/${target_device}/CONFIG/kati_packaging" \
  OUT_DIR="$out" \
  PATH="$PWD/prebuilts/build-tools/path/linux-x86:$PWD/${out}/.path" \
  PYTHONDONTWRITEBYTECODE=1 \
  SOONG_ANDROID_MK="$androidmk" \
  SOONG_MAKEVARS_MK="$makevarsmk" \
  TARGET_BUILD_VARIANT="$TARGET_BUILD_VARIANT" \
  TARGET_DEVICE_DIR="$target_device_dir" \
  TARGET_PRODUCT=${TARGET_PRODUCT} \
  TMPDIR="$PWD/$out/soong/.temp"