| #!/bin/bash | 
 |  | 
 | set -eu | 
 |  | 
 | # Copyright 2020 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. | 
 |  | 
 | # Tool to evaluate the transitive closure of the ninja dependency graph of the | 
 | # files and targets a given target depends on. | 
 | # | 
 | # i.e. the list of things that, if changed, could cause a change to a target. | 
 |  | 
 | readonly me=$(basename "${0}") | 
 |  | 
 | readonly usage="usage: ${me} {options} target [target...] | 
 |  | 
 | Evaluate the transitive closure of files and ninja targets that one or more | 
 | targets depend on. | 
 |  | 
 | Dependency Options: | 
 |  | 
 |   -(no)order_deps   Whether to include order-only dependencies. (Default false) | 
 |   -(no)implicit     Whether to include implicit dependencies. (Default true) | 
 |   -(no)explicit     Whether to include regular / explicit deps. (Default true) | 
 |  | 
 |   -nofollow         Unanchored regular expression. Matching paths and targets | 
 |                     always get reported. Their dependencies do not get reported | 
 |                     unless first encountered in a 'container' file type. | 
 |                     Multiple allowed and combined using '|'. | 
 |                     e.g. -nofollow='*.so' not -nofollow='.so$' | 
 |                     -nofollow='*.so|*.dex' or -nofollow='*.so' -nofollow='.dex' | 
 |                     (Defaults to no matches) | 
 |   -container        Unanchored regular expression. Matching file extensions get | 
 |                     treated as 'container' files for -nofollow option. | 
 |                     Multiple allowed and combines using '|' | 
 |                     (Default 'apex|apk|zip|jar|tar|tgz') | 
 |  | 
 | Output Options: | 
 |  | 
 |   -(no)quiet        Suppresses progress output to stderr and interactive | 
 |     alias -(no)q    prompts. By default, when stderr is a tty, progress gets | 
 |                     reported to stderr; when both stderr and stdin are tty, | 
 |                     the script asks user whether to delete intermediate files. | 
 |                     When suppressed or not prompted, script always deletes the | 
 |                     temporary / intermediate files. | 
 |   -sep=<delim>      Use 'delim' as output field separator between notice | 
 |                     checksum and notice filename in notice output. | 
 |                     e.g. sep='\\t' | 
 |                     (Default space) | 
 |   -csv              Shorthand for -sep=',' | 
 |   -directories=<f>  Output directory names of dependencies to 'f'. | 
 |     alias -d        User '/dev/stdout' to send directories to stdout. Defaults | 
 |                     to no directory output. | 
 |   -notices=<file>   Output license and notice file paths to 'file'. | 
 |     alias -n        Use '/dev/stdout' to send notices to stdout. Defaults to no | 
 |                     license/notice output. | 
 |   -projects=<file>  Output git project names to 'file'. Use '/dev/stdout' to | 
 |     alias -p        send projects to stdout. Defaults to no project output. | 
 |   -targets=<fils>   Output target dependencies to 'file'. Use '/dev/stdout' to | 
 |     alias -t        send targets to stdout. | 
 |                     When no directory, notice, project or target output options | 
 |                     given, defaults to stdout. Otherwise, defaults to no target | 
 |                     output. | 
 |  | 
 | At minimum, before running this script, you must first run: | 
 | $ source build/envsetup.sh | 
 | $ lunch | 
 | $ m nothing | 
 | to setup the build environment, choose a target platform, and build the ninja | 
 | dependency graph. | 
 | " | 
 |  | 
 | function die() { echo -e "${*}" >&2; exit 2; } | 
 |  | 
 | # Reads one input target per line from stdin; outputs (isnotice target) tuples. | 
 | # | 
 | # output target is a ninja target that the input target depends on | 
 | # isnotice in {0,1} with 1 for output targets believed to be license or notice | 
 | function getDeps() { | 
 |     (tr '\n' '\0' | xargs -0 -r "${ninja_bin}" -f "${ninja_file}" -t query) \ | 
 |     | awk -v include_order="${include_order_deps}" \ | 
 |         -v include_implicit="${include_implicit_deps}" \ | 
 |         -v include_explicit="${include_deps}" \ | 
 |         -v containers="${container_types}" \ | 
 |     ' | 
 |       BEGIN { | 
 |         ininput = 0 | 
 |         isnotice = 0 | 
 |         currFileName = "" | 
 |         currExt = "" | 
 |       } | 
 |       $1 == "outputs:" || $1 == "validations:" { | 
 |         ininput = 0 | 
 |       } | 
 |       ininput == 0 && $0 ~ /^\S\S*:$/ { | 
 |         isnotice = ($0 ~ /.*NOTICE.*[.]txt:$/) | 
 |         currFileName = gensub(/^.*[/]([^/]*)[:]$/, "\\1", "g") | 
 |         currExt = gensub(/^.*[.]([^./]*)[:]$/, "\\1", "g") | 
 |       } | 
 |       ininput != 0 && $1 !~ /^[|][|]?/ { | 
 |         if (include_explicit == "true") { | 
 |           fileName = gensub(/^.*[/]([^/]*)$/, "\\1", "g") | 
 |           print ( \ | 
 |               (isnotice && $0 !~ /^\s*build[/]soong[/]scripts[/]/) \ | 
 |               || $0 ~ /NOTICE|LICEN[CS]E/ \ | 
 |               || $0 ~ /(notice|licen[cs]e)[.]txt/ \ | 
 |           )" "(fileName == currFileName||currExt ~ "^(" containers ")$")" "gensub(/^\s*/, "", "g") | 
 |         } | 
 |       } | 
 |       ininput != 0 && $1 == "|" { | 
 |         if (include_implicit == "true") { | 
 |           fileName = gensub(/^.*[/]([^/]*)$/, "\\1", "g") | 
 |           $1 = "" | 
 |           print ( \ | 
 |               (isnotice && $0 !~ /^\s*build[/]soong[/]scripts[/]/) \ | 
 |               || $0 ~ /NOTICE|LICEN[CS]E/ \ | 
 |               || $0 ~ /(notice|licen[cs]e)[.]txt/ \ | 
 |           )" "(fileName == currFileName||currExt ~ "^(" containers ")$")" "gensub(/^\s*/, "", "g") | 
 |         } | 
 |       } | 
 |       ininput != 0 && $1 == "||" { | 
 |         if (include_order == "true") { | 
 |           fileName = gensub(/^.*[/]([^/]*)$/, "\\1", "g") | 
 |           $1 = "" | 
 |           print ( \ | 
 |               (isnotice && $0 !~ /^\s*build[/]soong[/]scripts[/]/) \ | 
 |               || $0 ~ /NOTICE|LICEN[CS]E/ \ | 
 |               || $0 ~ /(notice|licen[cs]e)[.]txt/ \ | 
 |           )" "(fileName == currFileName||currExt ~ "^(" containers ")$")" "gensub(/^\s*/, "", "g") | 
 |         } | 
 |       } | 
 |       $1 == "input:" { | 
 |         ininput = 1 | 
 |       } | 
 |     ' | 
 | } | 
 |  | 
 | # Reads one input directory per line from stdin; outputs unique git projects. | 
 | function getProjects() { | 
 |     while read d; do | 
 |         while [ "${d}" != '.' ] && [ "${d}" != '/' ]; do | 
 |             if [ -d "${d}/.git/" ]; then | 
 |                 echo "${d}" | 
 |                 break | 
 |             fi | 
 |             d=$(dirname "${d}") | 
 |         done | 
 |     done | sort -u | 
 | } | 
 |  | 
 |  | 
 | if [ -z "${ANDROID_BUILD_TOP}" ]; then | 
 |     die "${me}: Run 'lunch' to configure the build environment" | 
 | fi | 
 |  | 
 | if [ -z "${TARGET_PRODUCT}" ]; then | 
 |     die "${me}: Run 'lunch' to configure the build environment" | 
 | fi | 
 |  | 
 | readonly ninja_file="${ANDROID_BUILD_TOP}/out/combined-${TARGET_PRODUCT}.ninja" | 
 | if [ ! -f "${ninja_file}" ]; then | 
 |     die "${me}: Run 'm nothing' to build the dependency graph" | 
 | fi | 
 |  | 
 | readonly ninja_bin="${ANDROID_BUILD_TOP}/prebuilts/build-tools/linux-x86/bin/ninja" | 
 | if [ ! -x "${ninja_bin}" ]; then | 
 |     die "${me}: Cannot find ninja executable expected at ${ninja_bin}" | 
 | fi | 
 |  | 
 |  | 
 | # parse the command-line | 
 |  | 
 | declare -a targets # one or more targets to evaluate | 
 |  | 
 | include_order_deps=false    # whether to trace through || "order dependencies" | 
 | include_implicit_deps=true  # whether to trace through | "implicit deps" | 
 | include_deps=true           # whether to trace through regular explicit deps | 
 | quiet=false                 # whether to suppress progress | 
 |  | 
 | projects_out=''             # where to output the list of projects | 
 | directories_out=''          # where to output the list of directories | 
 | targets_out=''              # where to output the list of targets/source files | 
 | notices_out=''              # where to output the list of license/notice files | 
 |  | 
 | sep=" "                     # separator between md5sum and notice filename | 
 |  | 
 | nofollow=''                 # regularexp must fully match targets to skip | 
 |  | 
 | container_types=''          # regularexp must full match file extension | 
 |                             # defaults to 'apex|apk|zip|jar|tar|tgz' below. | 
 |  | 
 | use_stdin=false             # whether to read targets from stdin i.e. target - | 
 |  | 
 | while [ $# -gt 0 ]; do | 
 |     case "${1:-}" in | 
 |       -) | 
 |         use_stdin=true | 
 |       ;; | 
 |       -*) | 
 |         flag=$(expr "${1}" : '^-*\(.*\)$') | 
 |         case "${flag:-}" in | 
 |           order_deps) | 
 |             include_order_deps=true;; | 
 |           noorder_deps) | 
 |             include_order_deps=false;; | 
 |           implicit) | 
 |             include_implicit_deps=true;; | 
 |           noimplicit) | 
 |             include_implicit_deps=false;; | 
 |           explicit) | 
 |             include_deps=true;; | 
 |           noexplicit) | 
 |             include_deps=false;; | 
 |           csv) | 
 |             sep=",";; | 
 |           sep) | 
 |             sep="${2?"${usage}"}"; shift;; | 
 |           sep=) | 
 |             sep=$(expr "${flag}" : '^sep=\(.*\)$');; | 
 |           q) ;& | 
 |           quiet) | 
 |             quiet=true;; | 
 |           noq) ;& | 
 |           noquiet) | 
 |             quiet=false;; | 
 |           nofollow) | 
 |             case "${nofollow}" in | 
 |               '') | 
 |                 nofollow="${2?"${usage}"}";; | 
 |               *) | 
 |                 nofollow="${nofollow}|${2?"${usage}"}";; | 
 |             esac | 
 |             shift | 
 |           ;; | 
 |           nofollow=*) | 
 |             case "${nofollow}" in | 
 |               '') | 
 |                 nofollow=$(expr "${flag}" : '^nofollow=\(.*\)$');; | 
 |               *) | 
 |                 nofollow="${nofollow}|"$(expr "${flag}" : '^nofollow=\(.*\)$');; | 
 |             esac | 
 |           ;; | 
 |           container) | 
 |             container_types="${container_types}|${2?"${usage}"}";; | 
 |           container=*) | 
 |             container_types="${container_types}|"$(expr "${flag}" : '^container=\(.*\)$');; | 
 |           p) ;& | 
 |           projects) | 
 |             projects_out="${2?"${usage}"}"; shift;; | 
 |           p=*) ;& | 
 |           projects=*) | 
 |             projects_out=$(expr "${flag}" : '^.*=\(.*\)$');; | 
 |           d) ;& | 
 |           directores) | 
 |             directories_out="${2?"${usage}"}"; shift;; | 
 |           d=*) ;& | 
 |           directories=*) | 
 |             directories_out=$(expr "${flag}" : '^.*=\(.*\)$');; | 
 |           t) ;& | 
 |           targets) | 
 |             targets_out="${2?"${usage}"}"; shift;; | 
 |           t=*) ;& | 
 |           targets=) | 
 |             targets_out=$(expr "${flag}" : '^.*=\(.*\)$');; | 
 |           n) ;& | 
 |           notices) | 
 |             notices_out="${2?"${usage}"}"; shift;; | 
 |           n=*) ;& | 
 |           notices=) | 
 |             notices_out=$(expr "${flag}" : '^.*=\(.*\)$');; | 
 |           *) | 
 |             die "${usage}\n\nUnknown flag ${1}";; | 
 |         esac | 
 |       ;; | 
 |       *) | 
 |         targets+=("${1:-}") | 
 |       ;; | 
 |     esac | 
 |     shift | 
 | done | 
 |  | 
 |  | 
 | # fail fast if command-line arguments are invalid | 
 |  | 
 | if [ ! -v targets[0] ] && ! ${use_stdin}; then | 
 |     die "${usage}\n\nNo target specified." | 
 | fi | 
 |  | 
 | if [ -z "${projects_out}" ] \ | 
 |   && [ -z "${directories_out}" ] \ | 
 |   && [ -z "${targets_out}" ] \ | 
 |   && [ -z "${notices_out}" ] | 
 | then | 
 |     targets_out='/dev/stdout' | 
 | fi | 
 |  | 
 | if [ -z "${container_types}" ]; then | 
 |   container_types='apex|apk|zip|jar|tar|tgz' | 
 | fi | 
 |  | 
 | # showProgress when stderr is a tty | 
 | if [ -t 2 ] && ! ${quiet}; then | 
 |     showProgress=true | 
 | else | 
 |     showProgress=false | 
 | fi | 
 |  | 
 | # interactive when both stderr and stdin are tty | 
 | if ${showProgress} && [ -t 0 ]; then | 
 |     interactive=true | 
 | else | 
 |     interactive=false | 
 | fi | 
 |  | 
 |  | 
 | readonly tmpFiles=$(mktemp -d "${TMPDIR}.tdeps.XXXXXXXXX") | 
 | if [ -z "${tmpFiles}" ]; then | 
 |     die "${me}: unable to create temporary directory" | 
 | fi | 
 |  | 
 | # The deps files contain unique (isnotice target) tuples where | 
 | # isnotice in {0,1} with 1 when ninja target 'target' is a license or notice. | 
 | readonly oldDeps="${tmpFiles}/old" | 
 | readonly newDeps="${tmpFiles}/new" | 
 | readonly allDeps="${tmpFiles}/all" | 
 |  | 
 | if ${use_stdin}; then # start deps by reading 1 target per line from stdin | 
 |   awk ' | 
 |     NF > 0 { | 
 |       print ( \ | 
 |           $0 ~ /NOTICE|LICEN[CS]E/ \ | 
 |           || $0 ~ /(notice|licen[cs]e)[.]txt/ \ | 
 |       )" "gensub(/\s*$/, "", "g", gensub(/^\s*/, "", "g")) | 
 |     } | 
 |   ' > "${newDeps}" | 
 | else # start with no deps by clearing file | 
 |   : > "${newDeps}" | 
 | fi | 
 |  | 
 | # extend deps by appending targets from command-line | 
 | for idx in "${!targets[*]}"; do | 
 |     isnotice='0' | 
 |     case "${targets[${idx}]}" in | 
 |       *NOTICE*) ;& | 
 |       *LICEN[CS]E*) ;& | 
 |       *notice.txt) ;& | 
 |       *licen[cs]e.txt) | 
 |         isnotice='1';; | 
 |     esac | 
 |     echo "${isnotice} 1 ${targets[${idx}]}" >> "${newDeps}" | 
 | done | 
 |  | 
 | # remove duplicates and start with new, old and all the same | 
 | sort -u < "${newDeps}" > "${allDeps}" | 
 | cp "${allDeps}" "${newDeps}" | 
 | cp "${allDeps}" "${oldDeps}" | 
 |  | 
 | # report depth of dependenciens when showProgress | 
 | depth=0 | 
 |  | 
 | # 1st iteration always unfiltered | 
 | filter='cat' | 
 | while [ $(wc -l < "${newDeps}") -gt 0 ]; do | 
 |     if ${showProgress}; then | 
 |         echo "depth ${depth} has "$(wc -l < "${newDeps}")" targets" >&2 | 
 |         depth=$(expr ${depth} + 1) | 
 |     fi | 
 |     ( # recalculate dependencies by combining unique inputs of new deps w. old | 
 |         set +e | 
 |         sh -c "${filter}" < "${newDeps}" | cut -d\  -f3- | getDeps | 
 |         set -e | 
 |         cat "${oldDeps}" | 
 |     ) | sort -u > "${allDeps}" | 
 |     # recalculate new dependencies as net additions to old dependencies | 
 |     set +e | 
 |     diff "${oldDeps}" "${allDeps}" --old-line-format='' --new-line-format='%L' \ | 
 |       --unchanged-line-format='' > "${newDeps}" | 
 |     set -e | 
 |     # apply filters on subsequent iterations | 
 |     case "${nofollow}" in | 
 |       '') | 
 |         filter='cat';; | 
 |       *) | 
 |         filter="egrep -v '^[01] 0 (${nofollow})$'" | 
 |       ;; | 
 |     esac | 
 |     # recalculate old dependencies for next iteration | 
 |     cp "${allDeps}" "${oldDeps}" | 
 | done | 
 |  | 
 | # found all deps -- clean up last iteration of old and new | 
 | rm -f "${oldDeps}" | 
 | rm -f "${newDeps}" | 
 |  | 
 | if ${showProgress}; then | 
 |     echo $(wc -l < "${allDeps}")" targets" >&2 | 
 | fi | 
 |  | 
 | if [ -n "${targets_out}" ]; then | 
 |     cut -d\  -f3- "${allDeps}" | sort -u > "${targets_out}" | 
 | fi | 
 |  | 
 | if [ -n "${directories_out}" ] \ | 
 |   || [ -n "${projects_out}" ] \ | 
 |   || [ -n "${notices_out}" ] | 
 | then | 
 |     readonly allDirs="${tmpFiles}/dirs" | 
 |     ( | 
 |         cut -d\  -f3- "${allDeps}" | tr '\n' '\0' | xargs -0 dirname | 
 |     ) | sort -u > "${allDirs}" | 
 |     if ${showProgress}; then | 
 |         echo $(wc -l < "${allDirs}")" directories" >&2 | 
 |     fi | 
 |  | 
 |     case "${directories_out}" in | 
 |       '')        : do nothing;; | 
 |       *) | 
 |         cat "${allDirs}" > "${directories_out}" | 
 |       ;; | 
 |     esac | 
 | fi | 
 |  | 
 | if [ -n "${projects_out}" ] \ | 
 |   || [ -n "${notices_out}" ] | 
 | then | 
 |     readonly allProj="${tmpFiles}/projects" | 
 |     set +e | 
 |     egrep -v '^out[/]' "${allDirs}" | getProjects > "${allProj}" | 
 |     set -e | 
 |     if ${showProgress}; then | 
 |         echo $(wc -l < "${allProj}")" projects" >&2 | 
 |     fi | 
 |  | 
 |     case "${projects_out}" in | 
 |       '')        : do nothing;; | 
 |       *) | 
 |         cat "${allProj}" > "${projects_out}" | 
 |       ;; | 
 |     esac | 
 | fi | 
 |  | 
 | case "${notices_out}" in | 
 |   '')        : do nothing;; | 
 |   *) | 
 |     readonly allNotice="${tmpFiles}/notices" | 
 |     set +e | 
 |     egrep '^1' "${allDeps}" | cut -d\  -f3- | egrep -v '^out/' > "${allNotice}" | 
 |     set -e | 
 |     cat "${allProj}" | while read proj; do | 
 |         for f in LICENSE LICENCE NOTICE license.txt notice.txt; do | 
 |             if [ -f "${proj}/${f}" ]; then | 
 |                 echo "${proj}/${f}" | 
 |             fi | 
 |         done | 
 |     done >> "${allNotice}" | 
 |     if ${showProgress}; then | 
 |       echo $(cat "${allNotice}" | sort -u | wc -l)" notice targets" >&2 | 
 |     fi | 
 |     readonly hashedNotice="${tmpFiles}/hashednotices" | 
 |     ( # md5sum outputs checksum space indicator(space or *) filename newline | 
 |         set +e | 
 |         sort -u "${allNotice}" | tr '\n' '\0' | xargs -0 -r md5sum 2>/dev/null | 
 |         set -e | 
 |       # use sed to replace space and indicator with separator | 
 |     ) > "${hashedNotice}" | 
 |     if ${showProgress}; then | 
 |         echo $(cut -d\  -f2- "${hashedNotice}" | sort -u | wc -l)" notice files" >&2 | 
 |         echo $(cut -d\  -f1 "${hashedNotice}" | sort -u | wc -l)" distinct notices" >&2 | 
 |     fi | 
 |     sed 's/^\([^ ]*\) [* ]/\1'"${sep}"'/g' "${hashedNotice}" | sort > "${notices_out}" | 
 |   ;; | 
 | esac | 
 |  | 
 | if ${interactive}; then | 
 |     echo -n "$(date '+%F %-k:%M:%S') Delete ${tmpFiles} ? [n] " >&2 | 
 |     read answer | 
 |     case "${answer}" in [yY]*) rm -fr "${tmpFiles}";; esac | 
 | else | 
 |     rm -fr "${tmpFiles}" | 
 | fi |