| Jeff Gaston | ef633e0 | 2017-10-10 13:20:31 -0700 | [diff] [blame] | 1 | #!/bin/bash -eu | 
 | 2 | # | 
 | 3 | # Copyright 2017 Google Inc. All rights reserved. | 
 | 4 | # | 
 | 5 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
 | 6 | # you may not use this file except in compliance with the License. | 
 | 7 | # You may obtain a copy of the License at | 
 | 8 | # | 
 | 9 | #     http://www.apache.org/licenses/LICENSE-2.0 | 
 | 10 | # | 
 | 11 | # Unless required by applicable law or agreed to in writing, software | 
 | 12 | # distributed under the License is distributed on an "AS IS" BASIS, | 
 | 13 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | 14 | # See the License for the specific language governing permissions and | 
 | 15 | # limitations under the License. | 
 | 16 |  | 
 | 17 | set -e | 
 | 18 |  | 
 | 19 | # This file makes it easy to confirm that a set of changes in source code don't result in any | 
 | 20 | # changes to the generated ninja files. This is to reduce the effort required to be confident | 
 | 21 | # in the correctness of refactorings | 
 | 22 |  | 
 | 23 | function die() { | 
 | 24 |   echo "$@" >&2 | 
 | 25 |   exit 1 | 
 | 26 | } | 
 | 27 |  | 
 | 28 | function usage() { | 
 | 29 |   violation="$1" | 
 | 30 |   die "$violation | 
 | 31 |  | 
 | 32 |   Usage: diff_build_graphs.sh [--products=product1,product2...] <OLD_VERSIONS> <NEW_VERSIONS> | 
 | 33 |  | 
 | 34 |   This file builds and parses the build files (Android.mk, Android.bp, etc) for each requested | 
 | 35 |   product and for both sets of versions, and checks whether the ninja files (which implement | 
 | 36 |   the build graph) changed between the two versions. | 
 | 37 |  | 
 | 38 |   Example: diff_build_graphs.sh 'build/soong:work^ build/blueprint:work^' 'build/soong:work build/blueprint:work' | 
 | 39 |  | 
 | 40 |   Options: | 
 | 41 |     --products=PRODUCTS  comma-separated list of products to check" | 
 | 42 | } | 
 | 43 |  | 
 | 44 | PRODUCTS_ARG="" | 
 | 45 | OLD_VERSIONS="" | 
 | 46 | NEW_VERSIONS="" | 
 | 47 | function parse_args() { | 
 | 48 |   # parse optional arguments | 
 | 49 |   while true; do | 
 | 50 |     arg="${1-}" | 
 | 51 |     case "$arg" in | 
 | 52 |       --products=*) PRODUCTS_ARG="$arg";; | 
 | 53 |       *) break;; | 
 | 54 |     esac | 
 | 55 |     shift | 
 | 56 |   done | 
 | 57 |   # parse required arguments | 
 | 58 |   if [ "$#" != "2" ]; then | 
 | 59 |     usage "" | 
 | 60 |   fi | 
 | 61 |   #argument validation | 
 | 62 |   OLD_VERSIONS="$1" | 
 | 63 |   NEW_VERSIONS="$2" | 
 | 64 |  | 
 | 65 | } | 
 | 66 | parse_args "$@" | 
 | 67 |  | 
 | 68 |  | 
 | 69 | # find some file paths | 
 | 70 | cd "$(dirname $0)" | 
 | 71 | SCRIPT_DIR="$PWD" | 
 | 72 | cd ../../.. | 
 | 73 | CHECKOUT_ROOT="$PWD" | 
 | 74 | OUT_DIR="${OUT_DIR-}" | 
 | 75 | if [ -z "$OUT_DIR" ]; then | 
 | 76 |   OUT_DIR=out | 
 | 77 | fi | 
 | 78 | WORK_DIR="$OUT_DIR/diff" | 
 | 79 | OUT_DIR_OLD="$WORK_DIR/out_old" | 
 | 80 | OUT_DIR_NEW="$WORK_DIR/out_new" | 
 | 81 | OUT_DIR_TEMP="$WORK_DIR/out_temp" | 
 | 82 |  | 
 | 83 |  | 
 | 84 | function checkout() { | 
 | 85 |   versionSpecs="$1" | 
 | 86 |   for versionSpec in $versionSpecs; do | 
 | 87 |     project="$(echo $versionSpec | sed 's|\([^:]*\):\([^:]*\)|\1|')" | 
 | 88 |     ref="$(echo     $versionSpec | sed 's|\([^:]*\):\([^:]*\)|\2|')" | 
 | 89 |     echo "checking out ref $ref in project $project" | 
 | 90 |     git -C "$project" checkout "$ref" | 
 | 91 |   done | 
 | 92 | } | 
 | 93 |  | 
 | 94 | function run_build() { | 
 | 95 |   echo | 
 | 96 |   echo "Starting build" | 
 | 97 |   # rebuild multiproduct_kati, in case it was missing before, | 
 | 98 |   # or in case it is affected by some of the changes we're testing | 
 | 99 |   make blueprint_tools | 
 | 100 |   # find multiproduct_kati and have it build the ninja files for each product | 
 | 101 |   builder="$(echo $OUT_DIR/soong/host/*/bin/multiproduct_kati)" | 
 | 102 |   BUILD_NUMBER=sample "$builder" $PRODUCTS_ARG --keep --out "$OUT_DIR_TEMP" || true | 
 | 103 |   echo | 
 | 104 | } | 
 | 105 |  | 
 | 106 | function diffProduct() { | 
 | 107 |   product="$1" | 
 | 108 |  | 
 | 109 |   zip1="$OUT_DIR_OLD/${product}.zip" | 
 | 110 |   unzipped1="$OUT_DIR_OLD/$product" | 
 | 111 |  | 
 | 112 |   zip2="$OUT_DIR_NEW/${product}.zip" | 
 | 113 |   unzipped2="$OUT_DIR_NEW/$product" | 
 | 114 |  | 
 | 115 |   unzip -qq "$zip1" -d "$unzipped1" | 
 | 116 |   unzip -qq "$zip2" -d "$unzipped2" | 
 | 117 |  | 
 | 118 |   #do a diff of the ninja files | 
 | 119 |   diffFile="$WORK_DIR/diff.txt" | 
 | 120 |   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 | 
 | 121 |   if [[ -s "$diffFile" ]]; then | 
 | 122 |     # outputs are different, so remove the unzipped versions but keep the zipped versions | 
| Jeff Gaston | 0c573d3 | 2017-11-28 16:58:37 -0800 | [diff] [blame] | 123 |     echo "First few differences (total diff linecount=$(wc -l $diffFile)) for product $product:" | 
| Jeff Gaston | ef633e0 | 2017-10-10 13:20:31 -0700 | [diff] [blame] | 124 |     cat "$diffFile" | head -n 10 | 
 | 125 |     echo "End of differences for product $product" | 
 | 126 |     rm -rf "$unzipped1" "$unzipped2" | 
 | 127 |   else | 
 | 128 |     # outputs are the same, so remove all of the outputs | 
 | 129 |     rm -rf "$zip1" "$unzipped1" "$zip2" "$unzipped2" | 
 | 130 |   fi | 
 | 131 | } | 
 | 132 |  | 
 | 133 | function do_builds() { | 
 | 134 |   #reset work dir | 
 | 135 |   rm -rf "$WORK_DIR" | 
 | 136 |   mkdir "$WORK_DIR" | 
 | 137 |  | 
 | 138 |   #build new code | 
 | 139 |   checkout "$NEW_VERSIONS" | 
 | 140 |   run_build | 
 | 141 |   mv "$OUT_DIR_TEMP" "$OUT_DIR_NEW" | 
 | 142 |  | 
 | 143 |   #build old code | 
 | 144 |   #TODO do we want to cache old results? Maybe by the time we care to cache old results this will | 
 | 145 |   #be running on a remote server somewhere and be completely different | 
 | 146 |   checkout "$OLD_VERSIONS" | 
 | 147 |   run_build | 
 | 148 |   mv "$OUT_DIR_TEMP" "$OUT_DIR_OLD" | 
 | 149 |  | 
 | 150 |   #cleanup | 
 | 151 |   echo created "$OUT_DIR_OLD" and "$OUT_DIR_NEW" | 
 | 152 | } | 
 | 153 |  | 
 | 154 | function main() { | 
 | 155 |   do_builds | 
 | 156 |   checkout "$NEW_VERSIONS" | 
 | 157 |  | 
 | 158 |   #find all products | 
 | 159 |   productsFile="$WORK_DIR/all_products.txt" | 
 | 160 |   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" | 
 | 161 |   echo Diffing products | 
 | 162 |   for product in $(cat $productsFile); do | 
 | 163 |     diffProduct "$product" | 
 | 164 |   done | 
 | 165 |   echo Done diffing products | 
 | 166 |   echo "Any differing outputs can be seen at $OUT_DIR_OLD/*.zip and $OUT_DIR_NEW/*.zip" | 
 | 167 |   echo "See $WORK_DIR/diff.txt for the full list of differences for the latest product checked" | 
 | 168 | } | 
 | 169 |  | 
 | 170 | main |