Make genrule sandbox script a python script

Unfortunately, genrules are not always available with `m`, instead we
need to know their output paths in order to build them and diff them.
Rewriting in Python lets us store module:output path maps more easily.

Test: ./genrule_sandbox_test.py gen_fstab.gs201 \
      libbt_topshim_bridge_header \
      android-support-multidex-instrumentation-version
Change-Id: If74130e5a4381cc0e1fab396ebb90dfd5a595a1c
This commit is contained in:
Liz Kammer 2023-06-09 11:23:15 -04:00
parent eb7f45ca9c
commit 767fad4b05
2 changed files with 174 additions and 111 deletions

174
tests/genrule_sandbox_test.py Executable file
View file

@ -0,0 +1,174 @@
#!/usr/bin/env python3
# Copyright (C) 2023 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.
import argparse
import collections
import json
import os.path
import subprocess
import tempfile
SRC_ROOT_DIR = os.path.abspath(__file__ + "/../../../..")
def _module_graph_path(out_dir):
return os.path.join(SRC_ROOT_DIR, out_dir, "soong", "module-actions.json")
def _build_with_soong(targets, target_product, out_dir, extra_env={}):
env = {
"TARGET_PRODUCT": target_product,
"TARGET_BUILD_VARIANT": "userdebug",
}
env.update(os.environ)
env.update(extra_env)
args = [
"build/soong/soong_ui.bash",
"--make-mode",
"--skip-soong-tests",
]
args.extend(targets)
try:
out = subprocess.check_output(
args,
cwd=SRC_ROOT_DIR,
env=env,
)
except subprocess.CalledProcessError as e:
print(e)
print(e.stdout)
print(e.stderr)
exit(1)
def _find_outputs_for_modules(modules, out_dir, target_product):
module_path = os.path.join(
SRC_ROOT_DIR, out_dir, "soong", "module-actions.json"
)
if not os.path.exists(module_path):
_build_with_soong(["json-module-graph"], target_product, out_dir)
action_graph = json.load(open(_module_graph_path(out_dir)))
module_to_outs = collections.defaultdict(set)
for mod in action_graph:
name = mod["Name"]
if name in modules:
for act in mod["Module"]["Actions"]:
if "}generate " in act["Desc"]:
module_to_outs[name].update(act["Outputs"])
return module_to_outs
def _store_outputs_to_tmp(output_files):
try:
tempdir = tempfile.TemporaryDirectory()
for f in output_files:
out = subprocess.check_output(
["cp", "--parents", f, tempdir.name],
cwd=SRC_ROOT_DIR,
)
return tempdir
except subprocess.CalledProcessError as e:
print(e)
print(e.stdout)
print(e.stderr)
def _diff_outs(file1, file2, show_diff):
base_args = ["diff"]
if not show_diff:
base_args.append("--brief")
try:
args = base_args + [file1, file2]
output = subprocess.check_output(
args,
cwd=SRC_ROOT_DIR,
)
except subprocess.CalledProcessError as e:
if e.returncode == 1:
if show_diff:
return output
return True
return None
def _compare_outputs(module_to_outs, tempdir, show_diff):
different_modules = collections.defaultdict(list)
for module, outs in module_to_outs.items():
for out in outs:
output = None
diff = _diff_outs(os.path.join(tempdir.name, out), out, show_diff)
if diff:
different_modules[module].append(diff)
tempdir.cleanup()
return different_modules
def main():
parser = argparse.ArgumentParser()
parser.add_argument(
"--target_product",
"-t",
default="aosp_cf_arm64_phone",
help="optional, target product, always runs as eng",
)
parser.add_argument(
"modules",
nargs="+",
help="modules to compare builds with genrule sandboxing enabled/not",
)
parser.add_argument(
"--show-diff",
"-d",
action="store_true",
required=False,
help="whether to display differing files",
)
args = parser.parse_args()
out_dir = os.environ.get("OUT_DIR", "out")
target_product = args.target_product
modules = set(args.modules)
module_to_outs = _find_outputs_for_modules(modules, out_dir, target_product)
all_outs = set()
for outs in module_to_outs.values():
all_outs.update(outs)
print("build without sandboxing")
_build_with_soong(list(all_outs), target_product, out_dir)
tempdir = _store_outputs_to_tmp(all_outs)
print("build with sandboxing")
_build_with_soong(
list(all_outs),
target_product,
out_dir,
extra_env={"GENRULE_SANDBOXING": "true"},
)
diffs = _compare_outputs(module_to_outs, tempdir, args.show_diff)
if len(diffs) == 0:
print("All modules are correct")
elif args.show_diff:
for m, d in diffs.items():
print(f"Module {m} has diffs {d}")
else:
print(f"Modules {list(diffs.keys())} have diffs")
if __name__ == "__main__":
main()

View file

@ -1,111 +0,0 @@
#!/bin/bash
# Copyright (C) 2023 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.
set -e
# Build the given genrule modules with GENRULE_SANDBOXING enabled and disabled,
# then compare the output of the modules and report result.
function die() { format=$1; shift; printf >&2 "$format\n" $@; exit 1; }
function usage() {
die "usage: ${0##*/} <-t lunch_target> [module]..."
}
if [ ! -e "build/make/core/Makefile" ]; then
die "$0 must be run from the top of the Android source tree."
fi
declare TARGET=
while getopts "t:" opt; do
case $opt in
t)
TARGET=$OPTARG ;;
*) usage ;;
esac
done
shift $((OPTIND-1))
MODULES="$@"
source build/envsetup.sh
if [[ -n $TARGET ]]; then
lunch $TARGET
fi
if [[ -z ${OUT_DIR+x} ]]; then
OUT_DIR="out"
fi
OUTPUT_DIR="$(mktemp -d tmp.XXXXXX)"
PASS=true
function cleanup {
if [ $PASS = true ]; then
rm -rf "${OUTPUT_DIR}"
fi
}
trap cleanup EXIT
declare -A GEN_PATH_MAP
function find_gen_paths() {
for module in $MODULES; do
module_path=$(pathmod "$module")
package_path=${module_path#$ANDROID_BUILD_TOP}
gen_path=$OUT_DIR/soong/.intermediates$package_path/$module
GEN_PATH_MAP[$module]=$gen_path
done
}
function store_outputs() {
local dir=$1; shift
for module in $MODULES; do
dest_dir=$dir/${module}
mkdir -p $dest_dir
gen_path=${GEN_PATH_MAP[$module]}
cp -r $gen_path $dest_dir
done
}
function cmp_outputs() {
local dir1=$1; shift
local dir2=$1; shift
for module in $MODULES; do
if ! diff -rq --exclude=genrule.sbox.textproto $dir1/$module $dir2/$module; then
PASS=false
echo "$module differ"
fi
done
if [ $PASS = true ]; then
echo "Test passed"
fi
}
if [ ! -f "$ANDROID_PRODUCT_OUT/module-info.json" ]; then
refreshmod
fi
find_gen_paths
m --skip-soong-tests GENRULE_SANDBOXING=true "${MODULES[@]}"
store_outputs "$OUTPUT_DIR/sandbox"
m --skip-soong-tests GENRULE_SANDBOXING=false "${MODULES[@]}"
store_outputs "$OUTPUT_DIR/non_sandbox"
cmp_outputs "$OUTPUT_DIR/non_sandbox" "$OUTPUT_DIR/sandbox"