18c2ecf20Sopenharmony_ci#! /bin/sh 28c2ecf20Sopenharmony_ci# SPDX-License-Identifier: GPL-2.0 38c2ecf20Sopenharmony_ci# Copyright (c) 2020, Google LLC. All rights reserved. 48c2ecf20Sopenharmony_ci# Author: Saravana Kannan <saravanak@google.com> 58c2ecf20Sopenharmony_ci 68c2ecf20Sopenharmony_cifunction help() { 78c2ecf20Sopenharmony_ci cat << EOF 88c2ecf20Sopenharmony_ciUsage: $(basename $0) [-c|-d|-m|-f] [filter options] <list of devices> 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ciThis script needs to be run on the target device once it has booted to a 118c2ecf20Sopenharmony_cishell. 128c2ecf20Sopenharmony_ci 138c2ecf20Sopenharmony_ciThe script takes as input a list of one or more device directories under 148c2ecf20Sopenharmony_ci/sys/devices and then lists the probe dependency chain (suppliers and 158c2ecf20Sopenharmony_ciparents) of these devices. It does a breadth first search of the dependency 168c2ecf20Sopenharmony_cichain, so the last entry in the output is close to the root of the 178c2ecf20Sopenharmony_cidependency chain. 188c2ecf20Sopenharmony_ci 198c2ecf20Sopenharmony_ciBy default it lists the full path to the devices under /sys/devices. 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ciIt also takes an optional modifier flag as the first parameter to change 228c2ecf20Sopenharmony_ciwhat information is listed in the output. If the requested information is 238c2ecf20Sopenharmony_cinot available, the device name is printed. 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_ci -c lists the compatible string of the dependencies 268c2ecf20Sopenharmony_ci -d lists the driver name of the dependencies that have probed 278c2ecf20Sopenharmony_ci -m lists the module name of the dependencies that have a module 288c2ecf20Sopenharmony_ci -f list the firmware node path of the dependencies 298c2ecf20Sopenharmony_ci -g list the dependencies as edges and nodes for graphviz 308c2ecf20Sopenharmony_ci -t list the dependencies as edges for tsort 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_ciThe filter options provide a way to filter out some dependencies: 338c2ecf20Sopenharmony_ci --allow-no-driver By default dependencies that don't have a driver 348c2ecf20Sopenharmony_ci attached are ignored. This is to avoid following 358c2ecf20Sopenharmony_ci device links to "class" devices that are created 368c2ecf20Sopenharmony_ci when the consumer probes (as in, not a probe 378c2ecf20Sopenharmony_ci dependency). If you want to follow these links 388c2ecf20Sopenharmony_ci anyway, use this flag. 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_ci --exclude-devlinks Don't follow device links when tracking probe 418c2ecf20Sopenharmony_ci dependencies. 428c2ecf20Sopenharmony_ci 438c2ecf20Sopenharmony_ci --exclude-parents Don't follow parent devices when tracking probe 448c2ecf20Sopenharmony_ci dependencies. 458c2ecf20Sopenharmony_ci 468c2ecf20Sopenharmony_ciEOF 478c2ecf20Sopenharmony_ci} 488c2ecf20Sopenharmony_ci 498c2ecf20Sopenharmony_cifunction dev_to_detail() { 508c2ecf20Sopenharmony_ci local i=0 518c2ecf20Sopenharmony_ci while [ $i -lt ${#OUT_LIST[@]} ] 528c2ecf20Sopenharmony_ci do 538c2ecf20Sopenharmony_ci local C=${OUT_LIST[i]} 548c2ecf20Sopenharmony_ci local S=${OUT_LIST[i+1]} 558c2ecf20Sopenharmony_ci local D="'$(detail_chosen $C $S)'" 568c2ecf20Sopenharmony_ci if [ ! -z "$D" ] 578c2ecf20Sopenharmony_ci then 588c2ecf20Sopenharmony_ci # This weirdness is needed to work with toybox when 598c2ecf20Sopenharmony_ci # using the -t option. 608c2ecf20Sopenharmony_ci printf '%05u\t%s\n' ${i} "$D" | tr -d \' 618c2ecf20Sopenharmony_ci fi 628c2ecf20Sopenharmony_ci i=$((i+2)) 638c2ecf20Sopenharmony_ci done 648c2ecf20Sopenharmony_ci} 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_cifunction already_seen() { 678c2ecf20Sopenharmony_ci local i=0 688c2ecf20Sopenharmony_ci while [ $i -lt ${#OUT_LIST[@]} ] 698c2ecf20Sopenharmony_ci do 708c2ecf20Sopenharmony_ci if [ "$1" = "${OUT_LIST[$i]}" ] 718c2ecf20Sopenharmony_ci then 728c2ecf20Sopenharmony_ci # if-statement treats 0 (no-error) as true 738c2ecf20Sopenharmony_ci return 0 748c2ecf20Sopenharmony_ci fi 758c2ecf20Sopenharmony_ci i=$(($i+2)) 768c2ecf20Sopenharmony_ci done 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_ci # if-statement treats 1 (error) as false 798c2ecf20Sopenharmony_ci return 1 808c2ecf20Sopenharmony_ci} 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_ci# Return 0 (no-error/true) if parent was added 838c2ecf20Sopenharmony_cifunction add_parent() { 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_ci if [ ${ALLOW_PARENTS} -eq 0 ] 868c2ecf20Sopenharmony_ci then 878c2ecf20Sopenharmony_ci return 1 888c2ecf20Sopenharmony_ci fi 898c2ecf20Sopenharmony_ci 908c2ecf20Sopenharmony_ci local CON=$1 918c2ecf20Sopenharmony_ci # $CON could be a symlink path. So, we need to find the real path and 928c2ecf20Sopenharmony_ci # then go up one level to find the real parent. 938c2ecf20Sopenharmony_ci local PARENT=$(realpath $CON/..) 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_ci while [ ! -e ${PARENT}/driver ] 968c2ecf20Sopenharmony_ci do 978c2ecf20Sopenharmony_ci if [ "$PARENT" = "/sys/devices" ] 988c2ecf20Sopenharmony_ci then 998c2ecf20Sopenharmony_ci return 1 1008c2ecf20Sopenharmony_ci fi 1018c2ecf20Sopenharmony_ci PARENT=$(realpath $PARENT/..) 1028c2ecf20Sopenharmony_ci done 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_ci CONSUMERS+=($PARENT) 1058c2ecf20Sopenharmony_ci OUT_LIST+=(${CON} ${PARENT}) 1068c2ecf20Sopenharmony_ci return 0 1078c2ecf20Sopenharmony_ci} 1088c2ecf20Sopenharmony_ci 1098c2ecf20Sopenharmony_ci# Return 0 (no-error/true) if one or more suppliers were added 1108c2ecf20Sopenharmony_cifunction add_suppliers() { 1118c2ecf20Sopenharmony_ci local CON=$1 1128c2ecf20Sopenharmony_ci local RET=1 1138c2ecf20Sopenharmony_ci 1148c2ecf20Sopenharmony_ci if [ ${ALLOW_DEVLINKS} -eq 0 ] 1158c2ecf20Sopenharmony_ci then 1168c2ecf20Sopenharmony_ci return 1 1178c2ecf20Sopenharmony_ci fi 1188c2ecf20Sopenharmony_ci 1198c2ecf20Sopenharmony_ci SUPPLIER_LINKS=$(ls -1d $CON/supplier:* 2>/dev/null) 1208c2ecf20Sopenharmony_ci for SL in $SUPPLIER_LINKS; 1218c2ecf20Sopenharmony_ci do 1228c2ecf20Sopenharmony_ci SYNC_STATE=$(cat $SL/sync_state_only) 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_ci # sync_state_only links are proxy dependencies. 1258c2ecf20Sopenharmony_ci # They can also have cycles. So, don't follow them. 1268c2ecf20Sopenharmony_ci if [ "$SYNC_STATE" != '0' ] 1278c2ecf20Sopenharmony_ci then 1288c2ecf20Sopenharmony_ci continue 1298c2ecf20Sopenharmony_ci fi 1308c2ecf20Sopenharmony_ci 1318c2ecf20Sopenharmony_ci SUPPLIER=$(realpath $SL/supplier) 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_ci if [ ! -e $SUPPLIER/driver -a ${ALLOW_NO_DRIVER} -eq 0 ] 1348c2ecf20Sopenharmony_ci then 1358c2ecf20Sopenharmony_ci continue 1368c2ecf20Sopenharmony_ci fi 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci CONSUMERS+=($SUPPLIER) 1398c2ecf20Sopenharmony_ci OUT_LIST+=(${CON} ${SUPPLIER}) 1408c2ecf20Sopenharmony_ci RET=0 1418c2ecf20Sopenharmony_ci done 1428c2ecf20Sopenharmony_ci 1438c2ecf20Sopenharmony_ci return $RET 1448c2ecf20Sopenharmony_ci} 1458c2ecf20Sopenharmony_ci 1468c2ecf20Sopenharmony_cifunction detail_compat() { 1478c2ecf20Sopenharmony_ci f=$1/of_node/compatible 1488c2ecf20Sopenharmony_ci if [ -e $f ] 1498c2ecf20Sopenharmony_ci then 1508c2ecf20Sopenharmony_ci echo -n $(cat $f) 1518c2ecf20Sopenharmony_ci else 1528c2ecf20Sopenharmony_ci echo -n $1 1538c2ecf20Sopenharmony_ci fi 1548c2ecf20Sopenharmony_ci} 1558c2ecf20Sopenharmony_ci 1568c2ecf20Sopenharmony_cifunction detail_module() { 1578c2ecf20Sopenharmony_ci f=$1/driver/module 1588c2ecf20Sopenharmony_ci if [ -e $f ] 1598c2ecf20Sopenharmony_ci then 1608c2ecf20Sopenharmony_ci echo -n $(basename $(realpath $f)) 1618c2ecf20Sopenharmony_ci else 1628c2ecf20Sopenharmony_ci echo -n $1 1638c2ecf20Sopenharmony_ci fi 1648c2ecf20Sopenharmony_ci} 1658c2ecf20Sopenharmony_ci 1668c2ecf20Sopenharmony_cifunction detail_driver() { 1678c2ecf20Sopenharmony_ci f=$1/driver 1688c2ecf20Sopenharmony_ci if [ -e $f ] 1698c2ecf20Sopenharmony_ci then 1708c2ecf20Sopenharmony_ci echo -n $(basename $(realpath $f)) 1718c2ecf20Sopenharmony_ci else 1728c2ecf20Sopenharmony_ci echo -n $1 1738c2ecf20Sopenharmony_ci fi 1748c2ecf20Sopenharmony_ci} 1758c2ecf20Sopenharmony_ci 1768c2ecf20Sopenharmony_cifunction detail_fwnode() { 1778c2ecf20Sopenharmony_ci f=$1/firmware_node 1788c2ecf20Sopenharmony_ci if [ ! -e $f ] 1798c2ecf20Sopenharmony_ci then 1808c2ecf20Sopenharmony_ci f=$1/of_node 1818c2ecf20Sopenharmony_ci fi 1828c2ecf20Sopenharmony_ci 1838c2ecf20Sopenharmony_ci if [ -e $f ] 1848c2ecf20Sopenharmony_ci then 1858c2ecf20Sopenharmony_ci echo -n $(realpath $f) 1868c2ecf20Sopenharmony_ci else 1878c2ecf20Sopenharmony_ci echo -n $1 1888c2ecf20Sopenharmony_ci fi 1898c2ecf20Sopenharmony_ci} 1908c2ecf20Sopenharmony_ci 1918c2ecf20Sopenharmony_cifunction detail_graphviz() { 1928c2ecf20Sopenharmony_ci if [ "$2" != "ROOT" ] 1938c2ecf20Sopenharmony_ci then 1948c2ecf20Sopenharmony_ci echo -n "\"$(basename $2)\"->\"$(basename $1)\"" 1958c2ecf20Sopenharmony_ci else 1968c2ecf20Sopenharmony_ci echo -n "\"$(basename $1)\"" 1978c2ecf20Sopenharmony_ci fi 1988c2ecf20Sopenharmony_ci} 1998c2ecf20Sopenharmony_ci 2008c2ecf20Sopenharmony_cifunction detail_tsort() { 2018c2ecf20Sopenharmony_ci echo -n "\"$2\" \"$1\"" 2028c2ecf20Sopenharmony_ci} 2038c2ecf20Sopenharmony_ci 2048c2ecf20Sopenharmony_cifunction detail_device() { echo -n $1; } 2058c2ecf20Sopenharmony_ci 2068c2ecf20Sopenharmony_cialias detail=detail_device 2078c2ecf20Sopenharmony_ciALLOW_NO_DRIVER=0 2088c2ecf20Sopenharmony_ciALLOW_DEVLINKS=1 2098c2ecf20Sopenharmony_ciALLOW_PARENTS=1 2108c2ecf20Sopenharmony_ci 2118c2ecf20Sopenharmony_ciwhile [ $# -gt 0 ] 2128c2ecf20Sopenharmony_cido 2138c2ecf20Sopenharmony_ci ARG=$1 2148c2ecf20Sopenharmony_ci case $ARG in 2158c2ecf20Sopenharmony_ci --help) 2168c2ecf20Sopenharmony_ci help 2178c2ecf20Sopenharmony_ci exit 0 2188c2ecf20Sopenharmony_ci ;; 2198c2ecf20Sopenharmony_ci -c) 2208c2ecf20Sopenharmony_ci alias detail=detail_compat 2218c2ecf20Sopenharmony_ci ;; 2228c2ecf20Sopenharmony_ci -m) 2238c2ecf20Sopenharmony_ci alias detail=detail_module 2248c2ecf20Sopenharmony_ci ;; 2258c2ecf20Sopenharmony_ci -d) 2268c2ecf20Sopenharmony_ci alias detail=detail_driver 2278c2ecf20Sopenharmony_ci ;; 2288c2ecf20Sopenharmony_ci -f) 2298c2ecf20Sopenharmony_ci alias detail=detail_fwnode 2308c2ecf20Sopenharmony_ci ;; 2318c2ecf20Sopenharmony_ci -g) 2328c2ecf20Sopenharmony_ci alias detail=detail_graphviz 2338c2ecf20Sopenharmony_ci ;; 2348c2ecf20Sopenharmony_ci -t) 2358c2ecf20Sopenharmony_ci alias detail=detail_tsort 2368c2ecf20Sopenharmony_ci ;; 2378c2ecf20Sopenharmony_ci --allow-no-driver) 2388c2ecf20Sopenharmony_ci ALLOW_NO_DRIVER=1 2398c2ecf20Sopenharmony_ci ;; 2408c2ecf20Sopenharmony_ci --exclude-devlinks) 2418c2ecf20Sopenharmony_ci ALLOW_DEVLINKS=0 2428c2ecf20Sopenharmony_ci ;; 2438c2ecf20Sopenharmony_ci --exclude-parents) 2448c2ecf20Sopenharmony_ci ALLOW_PARENTS=0 2458c2ecf20Sopenharmony_ci ;; 2468c2ecf20Sopenharmony_ci *) 2478c2ecf20Sopenharmony_ci # Stop at the first argument that's not an option. 2488c2ecf20Sopenharmony_ci break 2498c2ecf20Sopenharmony_ci ;; 2508c2ecf20Sopenharmony_ci esac 2518c2ecf20Sopenharmony_ci shift 2528c2ecf20Sopenharmony_cidone 2538c2ecf20Sopenharmony_ci 2548c2ecf20Sopenharmony_cifunction detail_chosen() { 2558c2ecf20Sopenharmony_ci detail $1 $2 2568c2ecf20Sopenharmony_ci} 2578c2ecf20Sopenharmony_ci 2588c2ecf20Sopenharmony_ciif [ $# -eq 0 ] 2598c2ecf20Sopenharmony_cithen 2608c2ecf20Sopenharmony_ci help 2618c2ecf20Sopenharmony_ci exit 1 2628c2ecf20Sopenharmony_cifi 2638c2ecf20Sopenharmony_ci 2648c2ecf20Sopenharmony_ciCONSUMERS=($@) 2658c2ecf20Sopenharmony_ciOUT_LIST=() 2668c2ecf20Sopenharmony_ci 2678c2ecf20Sopenharmony_ci# Do a breadth first, non-recursive tracking of suppliers. The parent is also 2688c2ecf20Sopenharmony_ci# considered a "supplier" as a device can't probe without its parent. 2698c2ecf20Sopenharmony_cii=0 2708c2ecf20Sopenharmony_ciwhile [ $i -lt ${#CONSUMERS[@]} ] 2718c2ecf20Sopenharmony_cido 2728c2ecf20Sopenharmony_ci CONSUMER=$(realpath ${CONSUMERS[$i]}) 2738c2ecf20Sopenharmony_ci i=$(($i+1)) 2748c2ecf20Sopenharmony_ci 2758c2ecf20Sopenharmony_ci if already_seen ${CONSUMER} 2768c2ecf20Sopenharmony_ci then 2778c2ecf20Sopenharmony_ci continue 2788c2ecf20Sopenharmony_ci fi 2798c2ecf20Sopenharmony_ci 2808c2ecf20Sopenharmony_ci # If this is not a device with a driver, we don't care about its 2818c2ecf20Sopenharmony_ci # suppliers. 2828c2ecf20Sopenharmony_ci if [ ! -e ${CONSUMER}/driver -a ${ALLOW_NO_DRIVER} -eq 0 ] 2838c2ecf20Sopenharmony_ci then 2848c2ecf20Sopenharmony_ci continue 2858c2ecf20Sopenharmony_ci fi 2868c2ecf20Sopenharmony_ci 2878c2ecf20Sopenharmony_ci ROOT=1 2888c2ecf20Sopenharmony_ci 2898c2ecf20Sopenharmony_ci # Add suppliers to CONSUMERS list and output the consumer details. 2908c2ecf20Sopenharmony_ci # 2918c2ecf20Sopenharmony_ci # We don't need to worry about a cycle in the dependency chain causing 2928c2ecf20Sopenharmony_ci # infinite loops. That's because the kernel doesn't allow cycles in 2938c2ecf20Sopenharmony_ci # device links unless it's a sync_state_only device link. And we ignore 2948c2ecf20Sopenharmony_ci # sync_state_only device links inside add_suppliers. 2958c2ecf20Sopenharmony_ci if add_suppliers ${CONSUMER} 2968c2ecf20Sopenharmony_ci then 2978c2ecf20Sopenharmony_ci ROOT=0 2988c2ecf20Sopenharmony_ci fi 2998c2ecf20Sopenharmony_ci 3008c2ecf20Sopenharmony_ci if add_parent ${CONSUMER} 3018c2ecf20Sopenharmony_ci then 3028c2ecf20Sopenharmony_ci ROOT=0 3038c2ecf20Sopenharmony_ci fi 3048c2ecf20Sopenharmony_ci 3058c2ecf20Sopenharmony_ci if [ $ROOT -eq 1 ] 3068c2ecf20Sopenharmony_ci then 3078c2ecf20Sopenharmony_ci OUT_LIST+=(${CONSUMER} "ROOT") 3088c2ecf20Sopenharmony_ci fi 3098c2ecf20Sopenharmony_cidone 3108c2ecf20Sopenharmony_ci 3118c2ecf20Sopenharmony_ci# Can NOT combine sort and uniq using sort -suk2 because stable sort in toybox 3128c2ecf20Sopenharmony_ci# isn't really stable. 3138c2ecf20Sopenharmony_cidev_to_detail | sort -k2 -k1 | uniq -f 1 | sort | cut -f2- 3148c2ecf20Sopenharmony_ci 3158c2ecf20Sopenharmony_ciexit 0 316