Michal Marek 411abb
#!/bin/bash
Michal Marek 411abb
# 
Michal Marek edfdc2
# given a Module.base and modules.dep, generate list
Michal Marek 411abb
# of base / supported / unsupported modules
Michal Marek 411abb
Michal Marek 411abb
set -e
Michal Marek 411abb
export LC_COLLATE=C
Michal Marek 411abb
Michal Marek 411abb
usage()
Michal Marek 411abb
{
Michal Marek add86b
	echo "Usage: ${0##*/} -b Module.base [-d dir] [-i] [-e] [-o outdir]"
Michal Marek add86b
	echo "  -i    Ignore supported.conf errors"
Michal Marek add86b
	echo "  -e    Create the -extra filelist (otherwise, treat all modules as supported)"
Michal Marek 411abb
}
Michal Marek 411abb
Michal Marek add86b
options=$(getopt -o b:d:o:ie -- "$@")
Michal Marek 411abb
if test $? -ne 0; then
Michal Marek 411abb
	usage >&2
Michal Marek 411abb
	exit 1
Michal Marek 411abb
fi
Michal Marek 411abb
eval set -- "$options"
Michal Marek 39da39
opt_builddir=
Michal Marek 411abb
opt_out=.
Michal Marek 411abb
opt_dir=.
Michal Marek 411abb
opt_ignore_errors=false
Michal Marek add86b
opt_extra=false
Michal Marek 411abb
while test $# -gt 0; do
Michal Marek 411abb
	opt=$1
Michal Marek 411abb
	shift
Michal Marek 411abb
	case "$opt" in
Michal Marek 373631
	-b | -d | -o)
Michal Marek 411abb
		arg=$1
Michal Marek 411abb
		shift
Michal Marek 411abb
	esac
Michal Marek 411abb
	case "$opt" in
Michal Marek 411abb
	-b)
Michal Marek 39da39
		opt_builddir=$arg ;;
Michal Marek 411abb
	-d)
Michal Marek 411abb
		opt_dir=$arg ;;
Michal Marek 411abb
	-o)
Michal Marek 411abb
		opt_out=$arg ;;
Michal Marek 411abb
	-i)
Michal Marek 411abb
		opt_ignore_errors=true ;;
Michal Marek add86b
	-e)
Michal Marek add86b
		opt_extra=true ;;
Michal Marek 411abb
	--)
Michal Marek 411abb
		break ;;
Michal Marek 411abb
	*)
Michal Marek 411abb
		echo "Unknown option $opt" >&2
Michal Marek 411abb
		exit 1
Michal Marek 411abb
	esac
Michal Marek 411abb
done
Michal Marek 39da39
if test -z "$opt_builddir"; then
Michal Marek 411abb
	usage >&2
Michal Marek 411abb
	exit 1
Michal Marek 411abb
fi
Michal Marek 411abb
Michal Marek 411abb
trap 'rm -rf "$tmp"' EXIT
Michal Marek 411abb
tmp=$(mktemp -d)
Michal Marek 411abb
mkdir "$tmp/empty"
Michal Marek 411abb
3b8c4d
find "$opt_dir" -type f \( -name '*.ko' -o -name '*.ko.xz' -o -name '*.ko.gz' -o -name '*.ko.zst' \) -printf '/%P\n' | \
3b8c4d
	awk -F/ '{ n=$NF; gsub(/-/, "_", n); sub(/\.ko(\.xz|\.gz|\.zst)?$/, "", n); print n " " $0; }' | \
Michal Marek 411abb
	sort >"$tmp/all"
Michal Marek 411abb
Michal Marek edfdc2
err=false
Michal Marek edfdc2
while read mod path; do
Michal Marek add86b
	if $opt_extra; then
Michal Marek add86b
		support=$(/sbin/modinfo -F supported "$opt_dir/$path")
Michal Marek add86b
	else
Michal Marek add86b
		support=yes
Michal Marek add86b
	fi
Michal Marek edfdc2
	case "$support" in
Michal Marek edfdc2
	yes | external)
Michal Marek edfdc2
		echo "$mod"
Michal Marek edfdc2
		;;
Michal Marek edfdc2
	no)
Michal Marek edfdc2
		;;
Michal Marek edfdc2
	"")
6f5ed0
		echo "warning: $mod not listed in supported.conf" >&2
Michal Marek edfdc2
		;;
Michal Marek edfdc2
	*)
Michal Marek edfdc2
		echo "error: invalid support flag for $mod: $support" >&2
Michal Marek edfdc2
		err=true
Michal Marek edfdc2
		;;
Michal Marek edfdc2
	esac
Michal Marek edfdc2
done <"$tmp/all" | sort -u >"$tmp/supp"
Michal Marek edfdc2
if $err; then
Michal Marek edfdc2
	exit 1
Michal Marek edfdc2
fi
Michal Marek edfdc2
Michal Marek 411abb
modules_dep=$(find "$opt_dir" -type f -name modules.dep)
Michal Marek 411abb
if test -z "$modules_dep"; then
Michal Marek 411abb
	echo "Cannot find modules.dep in $opt_dir" >&2
Michal Marek 411abb
	exit 1
Michal Marek 411abb
fi
Michal Marek 411abb
(
Michal Marek 411abb
	echo '%:
Michal Marek 411abb
	@echo $@
Michal Marek 411abb
ifdef EXPLAIN
Michal Marek 411abb
	@for dep in $^; do echo "$$dep needed by $@"; done >> $(EXPLAIN)
Michal Marek 411abb
endif
Michal Marek 411abb
'
3b8c4d
	sed -r 's:[^ ]*/([^/]*)\.ko(\.xz|\.gz|\.zst)?\>:\1:g; y/-/_/' "$modules_dep"
Michal Marek 411abb
) >"$tmp/dep"
Michal Marek 411abb
Michal Marek 411abb
add_dependent_modules()
Michal Marek 411abb
{
Michal Suchanek d8fe17
	xargs -r make $MAKE_ARGS EXPLAIN=$1 -rRs -C "$tmp/empty" -f "$tmp/dep" | sort -u
Michal Marek 411abb
}
Michal Marek 411abb
Michal Marek 411abb
# base
Takashi Iwai 752fbc
if test -f "$opt_builddir/Module.base"; then
Takashi Iwai 752fbc
    sed 'y/-/_/' <"$opt_builddir/Module.base" | add_dependent_modules >"$tmp/base"
Takashi Iwai 752fbc
else
Takashi Iwai 752fbc
    touch "$tmp/base"
Takashi Iwai 752fbc
fi
Michal Marek 411abb
join -j 1 -o 2.2 "$tmp/base" "$tmp/all" >"$opt_out/base-modules"
Michal Marek 411abb
Michal Marek f68583
# base firmware
Michal Suchanek d8fe17
kver=$(make $MAKE_ARGS -s -C "$opt_builddir" kernelrelease)
6f5ed0
fw_dir=/lib/firmware/$kver
6f5ed0
test -d $opt_dir/usr$fw_dir && fw_dir=/usr$fw_dir
6f5ed0
if test -d "$opt_dir$fw_dir"; then
Michal Marek f68583
	join <(/sbin/modinfo -F firmware \
Michal Marek f68583
		$(sed "s:^:$opt_dir:" "$opt_out/base-modules") | sort) \
6f5ed0
	     <(find "$opt_dir$fw_dir" -type f -printf '%P\n' | sort)
6f5ed0
fi | sed "s:^:$fw_dir:" >"$opt_out/base-firmware"
Michal Marek f68583
Michal Marek 39da39
# kmps
Michal Marek 39da39
for f in "$opt_builddir"/Module.*-kmp; do
Takashi Iwai 752fbc
	test -f "$f" || continue
Michal Marek 39da39
	kmp=${f##*/Module.}
Michal Marek 39da39
	sed 'y/-/_/' <"$f" >"$tmp/$kmp"
Michal Marek 39da39
	join -j 1 -o 2.2 "$tmp/$kmp" "$tmp/all" >"$opt_out/$kmp-modules"
Michal Marek 39da39
	cat "$tmp/$kmp"
Michal Marek 39da39
done | sort -u >"$tmp/kmp-all"
Michal Marek 39da39
join -v1 "$tmp/supp" "$tmp/kmp-all" >"$tmp/supp-main"
Michal Marek 39da39
Michal Marek 411abb
# main
Michal Marek 39da39
add_dependent_modules "$tmp/supp-explain" <"$tmp/supp-main" >"$tmp/supp-all"
Michal Marek 39da39
if ! cmp -s "$tmp/supp-main" "$tmp/supp-all"; then
Michal Marek 39da39
	# FIXME: Error message not accurate if a supported KMP module is
Michal Marek 39da39
	# needed by a module in the main package
Michal Marek 411abb
	echo "The following unsupported modules are used by supported modules:" >&2
Michal Marek 411abb
	join -j1 -a2 <(sort "$tmp/supp-explain") \
Michal Marek 39da39
		 <(join -v2 "$tmp/supp-main" "$tmp/supp-all") >&2
Michal Marek 411abb
	echo "Please fix supported.conf." >&2
Michal Marek 411abb
	if ! $opt_ignore_errors; then
Michal Marek 411abb
		exit 1
Michal Marek 411abb
	fi
Michal Marek 411abb
fi
Michal Marek 411abb
join -j 1 -o 2.2 "$tmp/supp-all" "$tmp/all" >"$opt_out/main-modules"
Michal Marek 411abb
Michal Marek 411abb
# unsupported
Takashi Iwai 1fa25f
join -j 1 -v 2 -o 2.2 <(sort -u "$tmp/supp-all" "$tmp/kmp-all") "$tmp/all" | sort -u > "$opt_out/unsupported-modules"
Takashi Iwai 1fa25f
Takashi Iwai 1fa25f
# split again to extra and optional
Takashi Iwai 1fa25f
if $opt_extra && test -f "$opt_builddir/Module.optional"; then
Takashi Iwai 1fa25f
Takashi Iwai 1fa25f
    declare -A modmarks wcmarks
Takashi Iwai 1fa25f
    wcpaths=()
Takashi Iwai 1fa25f
    while read mark path; do
3b8c4d
	case $path in
3b8c4d
	    *.ko.xz|*.ko.gz|*.ko.zst)
3b8c4d
		path=${path%.*};;
3b8c4d
	esac
Takashi Iwai 1fa25f
	path=${path%.ko}
Takashi Iwai 1fa25f
	mod=${path##*/}
Takashi Iwai 1fa25f
	modmarks["$mod"]="$mark"
Takashi Iwai 1fa25f
	# paths with wildcards need to be verified sequentially, so we keep
Takashi Iwai 1fa25f
	# the paths in the array wcpaths and each mark in wcmarks[]
Takashi Iwai 1fa25f
	case "$path" in
Takashi Iwai 1fa25f
	    *[\*\?\[]*)
Takashi Iwai 1fa25f
		wcpaths[${#wcpaths[@]}]="$path"
Takashi Iwai 1fa25f
		wcmarks["$path"]="$mark";;
Takashi Iwai 1fa25f
	esac
Takashi Iwai 1fa25f
    done < "$opt_builddir/Module.optional"
Takashi Iwai 1fa25f
Takashi Iwai 1fa25f
    while read xpath; do
34e68f
	path=$xpath
34e68f
	case $path in
3b8c4d
	    *.ko.xz|*.ko.gz|*.ko.zst)
34e68f
		path=${path%.*};;
3b8c4d
	esac
Takashi Iwai 1fa25f
	path=${path%.ko}
Takashi Iwai 1fa25f
	mod=${path##*/}
Takashi Iwai 1fa25f
	x=${modmarks["$mod"]}
Takashi Iwai 1fa25f
	if [ -n "$x" ]; then
Takashi Iwai 1fa25f
	    test x"$x" = x"-" && echo "$xpath"
Takashi Iwai 1fa25f
	    continue
Takashi Iwai 1fa25f
	fi
Michal Marek 411abb
Takashi Iwai 1fa25f
	# unmatched modules must be handled via wildcard
Takashi Iwai 1fa25f
	path=${path#/lib/modules/*/kernel/}
Takashi Iwai 1fa25f
	for m in "${wcpaths[@]}"; do
Takashi Iwai 1fa25f
	    case "$path" in
Takashi Iwai 1fa25f
		($m)
Takashi Iwai 1fa25f
		    test x${wcmarks["$m"]} = x"-" && echo "$xpath"
Takashi Iwai 1fa25f
		    break;;
Takashi Iwai 1fa25f
	    esac
Takashi Iwai 1fa25f
	done
Takashi Iwai 1fa25f
    done < "$opt_out/unsupported-modules" | sort > "$tmp/unsupp-extra"
Takashi Iwai 1fa25f
Takashi Iwai 1fa25f
    cat "$tmp/supp-all" "$tmp/kmp-all" "$tmp/unsupp-extra" | \
3b8c4d
	sed -r 's:[^ ]*/([^/]*)\.ko(\.xz|\.gz|\.zst)?\>:\1:g; y/-/_/' | sort -u > "$tmp/unsupp-extra-all"
Takashi Iwai 1fa25f
    add_dependent_modules "$tmp/unsupp-explain" <"$tmp/unsupp-extra-all" >"$tmp/unsupp-extra-dep"
Takashi Iwai 1fa25f
    if ! cmp -s "$tmp/unsupp-extra-all" "$tmp/unsupp-extra-dep"; then
Takashi Iwai 1fa25f
	echo "The following optional modules are used by extra modules:" >&2
Takashi Iwai 1fa25f
	join -j1 -a2 <(sort "$tmp/unsupp-explain") \
Takashi Iwai 1fa25f
		 <(join -v2 "$tmp/unsupp-extra-all" "$tmp/unsupp-extra-dep") >&2
Takashi Iwai 1fa25f
	echo "Please fix supported.conf." >&2
Takashi Iwai 1fa25f
	if ! $opt_ignore_errors; then
Takashi Iwai 1fa25f
		exit 1
Takashi Iwai 1fa25f
	fi
Takashi Iwai 1fa25f
    fi
Michal Marek 411abb
Takashi Iwai 1fa25f
    join -j 1 -v 2 "$tmp/unsupp-extra" "$opt_out/unsupported-modules" > "$opt_out/optional-modules"
Takashi Iwai 1fa25f
    mv "$tmp/unsupp-extra" "$opt_out/unsupported-modules"
Takashi Iwai 1fa25f
fi
Michal Marek 411abb
Takashi Iwai 1fa25f
exit 0