| #! /bin/bash | 
 |  | 
 | # Recursively list Android image directory. | 
 | set -eu | 
 | set -o pipefail | 
 |  | 
 | function die() { format=$1; shift; printf "$format\n" "$@"; exit 1; } | 
 |  | 
 | # Figure out the filer utility. | 
 | declare filer= | 
 | [[ -z "${ANDROID_HOST_OUT:-}" ]] || filer=${ANDROID_HOST_OUT}/bin/debugfs_static | 
 | if [[ "${1:-}" =~ --debugfs_path=(.*) ]]; then | 
 |   filer=${BASH_REMATCH[1]} | 
 |   shift | 
 | fi | 
 | if [[ -z "${filer:-}" ]]; then | 
 |   maybefiler="$(dirname $0)/debugfs_static" | 
 |   [[ ! -x "$maybefiler" ]] || filer="$maybefiler" | 
 | fi | 
 |  | 
 | (( $# >0 )) || die "%s [--debugfs_path=<path>] IMAGE" "$0" | 
 |  | 
 | [[ -n "${filer:-}" ]] || die "cannot locate 'debugfs' executable: \ | 
 | --debugfs_path= is missing, ANDROID_HOST_OUT is not set, \ | 
 | and 'debugfs_static' is not colocated with this script" | 
 | declare -r image="$1" | 
 |  | 
 | function dolevel() { | 
 |   printf "%s/\n" "$1" | 
 |   # Each line of the file output consists of 6 fields separated with '/'. | 
 |   # The second one contains the file's attributes, and the fifth its name. | 
 |   $filer -R "ls -l -p $1" "$image" 2>/dev/null |\ | 
 |     sed -nr 's|^/.*/(.*)/.*/.*/(.+)/.*/$|\2 \1|p' | LANG=C sort | \ | 
 |   while read name attr; do | 
 |     [[ "$name" != '.' && "$name" != '..' ]] || continue | 
 |     path="$1/$name" | 
 |     # If the second char of the attributes is '4', it is a directory. | 
 |     if [[ $attr =~ ^.4 ]]; then | 
 |       dolevel "$path" | 
 |     else | 
 |       printf "%s\n" "$path" | 
 |     fi | 
 |   done | 
 | } | 
 |  | 
 | # The filer always prints its version on stderr, so we are going | 
 | # to redirect it to the bit bucket. On the other hand, the filer's | 
 | # return code on error is still 0. Let's run it once to without | 
 | # redirecting stderr to see that there is at least one entry. | 
 | $filer -R "ls -l -p" "$image" | grep -q -m1 -P '^/.*/.*/.*/.*/.+/.*/$' | 
 | dolevel . |