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