|  | #!/bin/bash -eu | 
|  | # | 
|  | # Copyright 2017 Google Inc. All rights reserved. | 
|  | # | 
|  | # 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 | 
|  |  | 
|  | # This file makes it easy to confirm that a set of changes in source code don't result in any | 
|  | # changes to the generated ninja files. This is to reduce the effort required to be confident | 
|  | # in the correctness of refactorings | 
|  |  | 
|  | function die() { | 
|  | echo "$@" >&2 | 
|  | exit 1 | 
|  | } | 
|  |  | 
|  | function usage() { | 
|  | violation="$1" | 
|  | die "$violation | 
|  |  | 
|  | Usage: diff_build_graphs.sh [--products=product1,product2...] <OLD_VERSIONS> <NEW_VERSIONS> | 
|  |  | 
|  | This file builds and parses the build files (Android.mk, Android.bp, etc) for each requested | 
|  | product and for both sets of versions, and checks whether the ninja files (which implement | 
|  | the build graph) changed between the two versions. | 
|  |  | 
|  | Example: diff_build_graphs.sh 'build/soong:work^ build/blueprint:work^' 'build/soong:work build/blueprint:work' | 
|  |  | 
|  | Options: | 
|  | --products=PRODUCTS  comma-separated list of products to check" | 
|  | } | 
|  |  | 
|  | PRODUCTS_ARG="" | 
|  | OLD_VERSIONS="" | 
|  | NEW_VERSIONS="" | 
|  | function parse_args() { | 
|  | # parse optional arguments | 
|  | while true; do | 
|  | arg="${1-}" | 
|  | case "$arg" in | 
|  | --products=*) PRODUCTS_ARG="$arg";; | 
|  | *) break;; | 
|  | esac | 
|  | shift | 
|  | done | 
|  | # parse required arguments | 
|  | if [ "$#" != "2" ]; then | 
|  | usage "" | 
|  | fi | 
|  | #argument validation | 
|  | OLD_VERSIONS="$1" | 
|  | NEW_VERSIONS="$2" | 
|  |  | 
|  | } | 
|  | parse_args "$@" | 
|  |  | 
|  |  | 
|  | # find some file paths | 
|  | cd "$(dirname $0)" | 
|  | SCRIPT_DIR="$PWD" | 
|  | cd ../../.. | 
|  | CHECKOUT_ROOT="$PWD" | 
|  | OUT_DIR="${OUT_DIR-}" | 
|  | if [ -z "$OUT_DIR" ]; then | 
|  | OUT_DIR=out | 
|  | fi | 
|  | WORK_DIR="$OUT_DIR/diff" | 
|  | OUT_DIR_OLD="$WORK_DIR/out_old" | 
|  | OUT_DIR_NEW="$WORK_DIR/out_new" | 
|  | OUT_DIR_TEMP="$WORK_DIR/out_temp" | 
|  |  | 
|  |  | 
|  | function checkout() { | 
|  | versionSpecs="$1" | 
|  | for versionSpec in $versionSpecs; do | 
|  | project="$(echo $versionSpec | sed 's|\([^:]*\):\([^:]*\)|\1|')" | 
|  | ref="$(echo     $versionSpec | sed 's|\([^:]*\):\([^:]*\)|\2|')" | 
|  | echo "checking out ref $ref in project $project" | 
|  | git -C "$project" checkout "$ref" | 
|  | done | 
|  | } | 
|  |  | 
|  | function run_build() { | 
|  | echo | 
|  | echo "Starting build" | 
|  | # rebuild multiproduct_kati, in case it was missing before, | 
|  | # or in case it is affected by some of the changes we're testing | 
|  | make blueprint_tools | 
|  | # find multiproduct_kati and have it build the ninja files for each product | 
|  | builder="$(echo $OUT_DIR/soong/host/*/bin/multiproduct_kati)" | 
|  | BUILD_NUMBER=sample "$builder" $PRODUCTS_ARG --keep --out "$OUT_DIR_TEMP" || true | 
|  | echo | 
|  | } | 
|  |  | 
|  | function diffProduct() { | 
|  | product="$1" | 
|  |  | 
|  | zip1="$OUT_DIR_OLD/${product}.zip" | 
|  | unzipped1="$OUT_DIR_OLD/$product" | 
|  |  | 
|  | zip2="$OUT_DIR_NEW/${product}.zip" | 
|  | unzipped2="$OUT_DIR_NEW/$product" | 
|  |  | 
|  | unzip -qq "$zip1" -d "$unzipped1" | 
|  | unzip -qq "$zip2" -d "$unzipped2" | 
|  |  | 
|  | #do a diff of the ninja files | 
|  | diffFile="$WORK_DIR/diff.txt" | 
|  | diff -r "$unzipped1" "$unzipped2" -x build_date.txt -x build_number.txt -x '\.*' -x '*.log' -x build_fingerprint.txt -x build.ninja.d -x '*.zip' > $diffFile || true | 
|  | if [[ -s "$diffFile" ]]; then | 
|  | # outputs are different, so remove the unzipped versions but keep the zipped versions | 
|  | echo "First few differences (total diff linecount=$(wc -l $diffFile)) for product $product:" | 
|  | cat "$diffFile" | head -n 10 | 
|  | echo "End of differences for product $product" | 
|  | rm -rf "$unzipped1" "$unzipped2" | 
|  | else | 
|  | # outputs are the same, so remove all of the outputs | 
|  | rm -rf "$zip1" "$unzipped1" "$zip2" "$unzipped2" | 
|  | fi | 
|  | } | 
|  |  | 
|  | function do_builds() { | 
|  | #reset work dir | 
|  | rm -rf "$WORK_DIR" | 
|  | mkdir "$WORK_DIR" | 
|  |  | 
|  | #build new code | 
|  | checkout "$NEW_VERSIONS" | 
|  | run_build | 
|  | mv "$OUT_DIR_TEMP" "$OUT_DIR_NEW" | 
|  |  | 
|  | #build old code | 
|  | #TODO do we want to cache old results? Maybe by the time we care to cache old results this will | 
|  | #be running on a remote server somewhere and be completely different | 
|  | checkout "$OLD_VERSIONS" | 
|  | run_build | 
|  | mv "$OUT_DIR_TEMP" "$OUT_DIR_OLD" | 
|  |  | 
|  | #cleanup | 
|  | echo created "$OUT_DIR_OLD" and "$OUT_DIR_NEW" | 
|  | } | 
|  |  | 
|  | function main() { | 
|  | do_builds | 
|  | checkout "$NEW_VERSIONS" | 
|  |  | 
|  | #find all products | 
|  | productsFile="$WORK_DIR/all_products.txt" | 
|  | find $OUT_DIR_OLD $OUT_DIR_NEW -mindepth 1 -maxdepth 1 -name "*.zip" | sed "s|^$OUT_DIR_OLD/||" | sed "s|^$OUT_DIR_NEW/||" | sed "s|\.zip$||" | sort | uniq > "$productsFile" | 
|  | echo Diffing products | 
|  | for product in $(cat $productsFile); do | 
|  | diffProduct "$product" | 
|  | done | 
|  | echo Done diffing products | 
|  | echo "Any differing outputs can be seen at $OUT_DIR_OLD/*.zip and $OUT_DIR_NEW/*.zip" | 
|  | echo "See $WORK_DIR/diff.txt for the full list of differences for the latest product checked" | 
|  | } | 
|  |  | 
|  | main |