#	$NetBSD: sets.subr,v 1.209 2024/07/11 17:27:25 riastradh Exp $
#

#
# The following variables contain defaults for sets.subr functions and callers:
#	setsdir			path to src/distrib/sets
#	nlists			list of base sets
#	xlists			list of x11 sets
#	obsolete		controls if obsolete files are selected instead
#	module			if != "no", enable MODULE sets
#	shlib			shared library format (a.out, elf, or "")
#	stlib			static library format (a.out, elf)
#
# The following <bsd.own.mk> variables are exported to the environment:
#	MACHINE
#	MACHINE_ARCH
#	MACHINE_CPU
#	HAVE_ACPI
#	HAVE_BINUTILS
#	HAVE_GCC
#	HAVE_GDB
#	HAVE_NVMM
#	HAVE_OPENSSL
#	HAVE_SSP
#	HAVE_UEFI
#	TOOLCHAIN_MISSING
#	OBJECT_FMT
# as well as:
#

#
# The following variables refer to tools that are used when building sets:
#
: ${AWK:=awk}
: ${CKSUM:=cksum}
: ${COMM:=comm}
: ${DATE:=date}
: ${DB:=db}
: ${EGREP:=egrep}
: ${ENV_CMD:=env}       # ${ENV} is special to sh(1), ksh(1), etc.
: ${FGREP:=fgrep}
: ${FIND:=find}
: ${GREP:=grep}
: ${GZIP_CMD:=gzip}     # ${GZIP} is special to gzip(1)
: ${HOSTNAME_CMD:=hostname}	# ${HOSTNAME} is special to bash(1)
: ${HOST_SH:=sh}
: ${IDENT:=ident}
: ${JOIN:=join}
: ${LS:=ls}
: ${MAKE:=make}
: ${MKTEMP:=mktemp}
: ${MTREE:=mtree}
: ${PASTE:=paste}
: ${PAX:=pax}
: ${PRINTF:=printf}
: ${SED:=sed}
: ${SORT:=sort}
: ${STAT:=stat}
: ${TSORT:=tsort}
: ${UNAME:=uname}
: ${WC:=wc}
: ${XARGS:=xargs}

#
# If printf is a shell builtin command, then we can
# implement cheaper versions of basename and dirname
# that do not involve any fork/exec overhead.
# If printf is not builtin, approximate it using echo,
# and hope there are no weird file names that cause
# some versions of echo to do the wrong thing.
# (Converting to this version of dirname speeded up the
# syspkgdeps script by an order of magnitude, from 68
# seconds to 6.3 seconds on one particular host.)
#
# Note that naive approximations for dirname
# using ${foo%/*} do not do the right thing in cases
# where the result should be "/" or ".".
#
case "$(type printf)" in
*builtin*)
	basename ()
	{
		local bn
		bn="${1##*/}"
		bn="${bn%$2}"
		printf "%s\n" "$bn"
	}
	dirname ()
	{
		local dn
		case "$1" in
		?*/*)	dn="${1%/*}" ;;
		/*)	dn=/ ;;
		*)	dn=. ;;
		esac
		printf "%s\n" "$dn"
	}
	;;
*)
	basename ()
	{
		local bn
		bn="${1##*/}"
		bn="${bn%$2}"
		echo "$bn"
	}
	dirname ()
	{
		local dn
		case "$1" in
		?*/*)	dn="${1%/*}" ;;
		/*)	dn=/ ;;
		*)	dn=. ;;
		esac
		echo "$dn"
	}
	;;
esac

#####

oIFS=$IFS
IFS="
"

for x in $( MAKEVERBOSE= ${MAKE} -B -f ${rundir}/mkvars.mk mkvars ); do
	eval export $x
done

IFS=$oIFS

MKVARS="$( MAKEVERBOSE= ${MAKE} -B -f ${rundir}/mkvars.mk mkvars | ${SED} -e 's,=.*,,' | ${XARGS} )"

#####

setsdir=${rundir}
obsolete=0
if [ "${MKKMOD}" = "no" ]; then
	module=no			# MODULEs are off.
	modset=""
else
	module=yes
	modset="modules"
fi
if [ "${MKATF}" = "no" ]; then
	testset=""
else
	testset="tests"
fi
if [ "${MKDEBUG}" = "no" -a "${MKDEBUGLIB}" = "no" ]; then
	debugset=""
	xdebugset=""
else
	debugset="debug"
	xdebugset="xdebug"
fi
if [ -z "${debugset}" -o "${MKCOMPAT}" = "no" ]; then
	debug32set=""
else
	debug32set="debug32"
fi
if [ -z "${debug32set}" ]; then
	debug64set=""
else
	if [ "${MACHINE_ARCH}" = "mips64eb" -o "${MACHINE_ARCH}" = "mips64el" ]; then
		debug64set="debug64"
	else
		debug64set=""
	fi
fi
if [ "${MKDTB}" = "no" ]; then
	dtbset=""
else
	dtbset="dtb"
fi
if [ "${MKHTML}" = "no" ]; then
	manhtmlset=""
else
	manhtmlset="manhtml"
fi
if [ "${MKCOMPAT}" = "no" ]; then
	base32set=""
else
	base32set="base32"
fi
if [ "${MKCOMPAT}" != "no" ]; then
	if [ "${MACHINE_ARCH}" = "mips64eb" -o "${MACHINE_ARCH}" = "mips64el" ]; then
		base64set="base64"
	else
		base64set=""
	fi
else
	base64set=""
fi

# XXX This should not be encoded here -- this mostly duplicates
# information in compat/archdirs.mk, except that it also identifies
# which compat architectures are `32-bit' and which ones are `64-bit'.
case $MACHINE_ARCH in
aarch64)
	compat32arches='eabi eabihf'
	;;
aarch64eb)
	compat32arches=eabi
	;;
mips64eb|mips64el)
	compat32arches=o32
	compat64arches=64
	;;
mipsn64eb|mipsn64el)
	compat32arches='n32 o32'
	;;
powerpc64)
	compat32arches=powerpc
	;;
riscv64)
	compat32arches=rv32
	;;
sparc64)
	compat32arches=sparc
	;;
x86_64)	compat32arches=i386
	;;
esac
: ${compat32arches:=}
: ${compat64arches:=}


# Determine lib type. Do this first so stlib also gets set.
if [ "${OBJECT_FMT}" = "ELF" ]; then
	shlib=elf
else
	shlib=aout
fi
stlib=$shlib
# Now check for MKPIC or specials and turn off shlib if need be.
if [ "${MKPIC}" = "no" ]; then
	shlib=no
fi
nlists="base $base32set $base64set comp $debugset $debug32set $debug64set $dtbset etc games gpufw man $manhtmlset misc $modset rescue $testset text"
xlists="xbase xcomp $xdebugset xetc xfont xserver"

OSRELEASE=$(${HOST_SH} ${NETBSDSRCDIR}/sys/conf/osrelease.sh -k)
if [ "${KERNEL_DIR}" = "yes" ]; then
	MODULEDIR="netbsd/modules"
else
	MODULEDIR="stand/${MACHINE}/${OSRELEASE}/modules"
fi
SUBST="s#@MODULEDIR@#${MODULEDIR}#g"
SUBST="${SUBST};s#@OSRELEASE@#${OSRELEASE}#g"
SUBST="${SUBST};s#@MACHINE@#${MACHINE}#g"

#
# list_set_files setfile [...]
#
# Produce a packing list for setfile(s).
# In each file, a record consists of a path and a System Package name,
# separated by whitespace. E.g.,
#
# 	# $NetBSD: sets.subr,v 1.209 2024/07/11 17:27:25 riastradh Exp $
# 	.			base-sys-root	[keyword[,...]]
# 	./altroot		base-sys-root
# 	./bin			base-sys-root
# 	./bin/[			base-util-root
# 	./bin/cat		base-util-root
#		[...]
#
# A # in the first column marks a comment.
#
# If ${obsolete} != 0, only entries with an "obsolete" keyword will
# be printed.  All other keywords must be present.
#
# The third field is an optional comma separated list of keywords to
# control if a record is printed; every keyword listed must be enabled
# for the record to be printed. The list of all available make variables
# that can be turned on or off can be found by running in this directory:
#
#	make -f mkvars.mk mkvarsyesno
#
# These MK<NAME> variables can be used as selectors in the sets as <name>.
#
# The following extra keywords are also available, listed by:
#
#	make -f mkvars.mk mkextravars
#
# These are:
#    1. The HAVE_<name>:
#	ssp			${HAVE_SSP} != no
#	libgcc_eh		${HAVE_LIBGCC_EH} != no
#	acpi			${HAVE_ACPI} != no
#	binutils=<n>		<n> = value of ${HAVE_BINUTILS}
#	gcc=<n>			<n> = value of ${HAVE_GCC}
#	gdb=<n>			<n> = value of ${HAVE_GDB}
#	mesa_ver=<n>		<n> = value of ${HAVE_MESA_VER}
#	nvmm			${HAVE_NVMM} != no
#	openssl=<n>		<n> = value of ${HAVE_OPENSSL}
#	uefi			${HAVE_UEFI} != no
#	xorg_server_ver=<n>	<n> = value of ${HAVE_XORG_SERVER_VER}
#	xorg_glamor		${HAVE_XORG_GLAMOR} != no
#
#    2. The USE_<name>:
#	use_inet6		${USE_INET6} != no
#	use_kerberos		${USE_KERBEROS} != no
#	use_ldap		${USE_LDAP} != no
#	use_yp			${USE_YP} != no
#
#    3. Finally:
#	dummy			dummy entry (ignored)
#	obsolete		file is obsolete, and only printed if
#				${obsolete} != 0
#
#	solaris			${MKDTRACE} != no or ${MKZFS} != no or ${MKCTF} != no
#
#
#	endian=<n>		<n> = value of ${TARGET_ENDIANNESS}
#
#
#	.cat			if ${MKMANZ} != "no" && ${MKCATPAGES} != "no"
#				  automatically append ".gz" to the filename
#
#	.man			if ${MKMANZ} != "no" && ${MKMAN} != "no"
#				  automatically append ".gz" to the filename
#
list_set_files()
{
	if [ ${MAKEVERBOSE:-2} -lt 3 ]; then
		verbose=false
	else
		verbose=true
	fi
	local CONFIGS="$( list_kernel_configs )"
	print_set_lists "$@" | \
	${AWK} -v obsolete=${obsolete} '
		function addkmod(line, fname, prefix, pat, patlen) {
			if (substr(line, 1, patlen) != pat) {
				return
			}
			for (d in kmodarchdirs) {
				xd = prefix kmodarchdirs[d]
				xline = xd substr(line, patlen + 1)
				xfname = xd substr(fname, patlen + 1)
				list[xline] = xfname
				emit(xline)
			}
		}
		function adddebugkernel(line, fname, pat, patlen) {
			if (pat == "" || substr(line, 1, patlen) != pat) {
				return 0
			}
			split("'"${CONFIGS}"'", configs)
			for (d in configs) {
				xfname = fname
				sub("@CONFIG@", configs[d], xfname)
				xline = line;
				sub("@CONFIG@", configs[d], xline)
				list[xline] = xfname
				emit(xline)
			}
			return 1
		}
		function emit(fname) {
			emitf[fname] = 1
		}
		BEGIN {
			if (obsolete)
				wanted["obsolete"] = 1

			ncpaths = 0
			split("'"${MKVARS}"'", needvars)
			doingcompat = 0
			doingcompattests = 0
			ignoredkeywords["compatdir"] = 1
			ignoredkeywords["compatfile"] = 1
			ignoredkeywords["compattestdir"] = 1
			ignoredkeywords["compattestfile"] = 1
			ignoredkeywords["compatx11dir"] = 1
			ignoredkeywords["compatx11file"] = 1
			for (vi in needvars) {
				nv = needvars[vi]
				kw = tolower(nv)
				sub(/^mk/, "", kw)
				sub(/^have_/, "", kw)
				sub(/^target_endianness/, "endian", kw)
				if (nv != "HAVE_GCC" && nv != "HAVE_GDB" && ENVIRON[nv] != "no" && nv != "COMPATARCHDIRS" && nv != "KMODARCHDIRS") {
					wanted[kw] = 1
				}
			}

			if ("compat" in wanted) {
				doingcompat = 1;
				split("'"${COMPATARCHDIRS}"'", compatarchdirs, ",");
				compatdirkeywords["compatdir"] = 1
				compatfilekeywords["compatfile"] = 1

				if (wanted["compattests"]) {
					doingcompattests = 1;
					compatdirkeywords["compattestdir"] = 1
					compatfilekeywords["compattestfile"] = 1
				}
				if (wanted["compatx11"]) {
					doingcompatx11 = 1;
					compatdirkeywords["compatx11dir"] = 1
					compatfilekeywords["compatx11file"] = 1
				}
			}

			if (("kmod" in wanted) && ("compatmodules" in wanted)) {
				split("'"${KMODARCHDIRS}"'", kmodarchdirs, ",");
				kmodprefix = "./stand/"
				kmodpat = kmodprefix ENVIRON["MACHINE"]
				l_kmodpat = length(kmodpat)
				kmoddbprefix = "./usr/libdata/debug/stand/"
				kmoddbpat = kmoddbprefix ENVIRON["MACHINE"]
				l_kmoddbpat = length(kmoddbpat)
			}
			if ("debug" in wanted) {
				debugkernelname = "./usr/libdata/debug/netbsd-@CONFIG@.debug"
				l_debugkernelname = length(debugkernelname);
			}

			if ("'"${TOOLCHAIN_MISSING}"'" != "yes") {
				if ("binutils" in wanted)
					wanted["binutils=" "'"${HAVE_BINUTILS}"'"] = 1
				if ("gcc" in wanted)
					wanted["gcc=" "'"${HAVE_GCC}"'"] = 1
				if ("gdb" in wanted)
					wanted["gdb=" "'"${HAVE_GDB}"'"] = 1
			}
			if ("acpi" in wanted) {
				wanted["acpi=" "'"${HAVE_ACPI}"'"] = 1
			}
			if ("mesa_ver" in wanted) {
				wanted["mesa_ver=" "'"${HAVE_MESA_VER}"'"] = 1
			}
			if ("nvmm" in wanted) {
				wanted["nvmm=" "'"${HAVE_NVMM}"'"] = 1
			}
			if ("openssl" in wanted) {
				wanted["openssl=" "'"${HAVE_OPENSSL}"'"] = 1
			}
			if ("xorg_server_ver" in wanted) {
				wanted["xorg_server_ver=" "'"${HAVE_XORG_SERVER_VER}"'"] = 1
			}
			if ("uefi" in wanted) {
				wanted["uefi=" "'"${HAVE_UEFI}"'"] = 1
			}
			if (("man" in wanted) && ("catpages" in wanted))
				wanted[".cat"] = 1
			if (("man" in wanted) && ("manpages" in wanted))
				wanted[".man"] = 1
			if ("endian" in wanted)
				wanted["endian=" "'"${TARGET_ENDIANNESS}"'"] = 1
			if ("machine" in wanted)
				wanted["machine=" "'"${MACHINE}"'"] = 1
			if ("machine_arch" in wanted)
				wanted["machine_arch=" "'"${MACHINE_ARCH}"'"] = 1
			if ("machine_cpu" in wanted)
				wanted["machine_cpu=" "'"${MACHINE_CPU}"'"] = 1
		}

		/^#/ {
			next;
		}

		/^-/ {
			notwanted[substr($1, 2)] = 1;
			delete list [substr($1, 2)];
			next;
		}

		NF > 2 && $3 != "-" {
			if (notwanted[$1] != "")
				next;
			split($3, keywords, ",")
			show = 1
			haveobs = 0
			iscompatfile = 0
			havekmod = 0
			iscompatdir = 0
			omitcompat = 0
			takecompat[$1] = 0
			for (ki in keywords) {
				kw = keywords[ki]
				if (("manz" in wanted) &&
				    (kw == ".cat" || kw == ".man"))
					$1 = $1 ".gz"
				if (substr(kw, 1, 1) == "!") {
					kw = substr(kw, 2)
					if (kw in wanted)
						show = 0
				} else if (kw in compatdirkeywords) {
					iscompatdir = 1
				} else if (kw in compatfilekeywords) {
					iscompatfile = 1
				} else if (kw == "nocompatmodules") {
					havekmod = -1
				} else if (kw == "omitcompat") {
					omitcompat = 1
				} else if (kw ~ /^takecompat=/) {
					takecompat[$1] = 1
					takecompatarch[substr(kw,
					    1 + length("takecompat=")), $1] = 1
				} else if (kw in ignoredkeywords) {
					# ignore
				} else if (! (kw in wanted)) {
					show = 0
				} else if (kw == "kmod" && havekmod == 0) {
					havekmod = 1
				}
				if (kw == "obsolete")
					haveobs = 1
			}

			if (takecompat[$1] && !(iscompatdir || iscompatfile)) {
				next
			}
			if (iscompatdir && !omitcompat) {
				for (d in cpaths) {
					if (cpaths[d] == $1 "/") {
						break
					}
				}
				cpaths[ncpaths++] = $1 "/"
			}
			if (obsolete && ! haveobs)
				next
			if (!show)
				next
			if (adddebugkernel($0, $1, debugkernelname, l_debugkernelname))
				next

			list[$1] = $0
			if (havekmod > 0) {
				addkmod($0, $1, kmodprefix, kmodpat, l_kmodpat)
				addkmod($0, $1, kmoddbprefix, kmoddbpat, l_kmoddbpat)
				emit($1)
				next
			}

			if (!doingcompat || !(iscompatfile || iscompatdir)) {
				emit($1)
				next
			}

			if (iscompatfile) {
				if (omitcompat) {
					emit($1)
				} else if (takecompat[$1]) {
					emitcompat[$1] = 1
				} else {
					emit($1)
					emitcompat[$1] = 1
				}
				next
			}
			if (iscompatdir) {
				if (omitcompat) {
					# /lib in base
					emit($1)
				} else if (takecompat[$1]) {
					# /lib in base32
					# nothing to do
				} else {
					# /usr/include in comp
					emit($1)
				}
			} else {
				emit($1)
			}
			if (omitcompat)
				next
			for (d in compatarchdirs) {
				if (takecompat[$1] &&
				    !takecompatarch[compatarchdirs[d], $1])
					continue
				tmp = $0
				xfile = $1 "/" compatarchdirs[d]
				tmp = xfile substr(tmp, length($1) + 1)
				if (xfile in notwanted)
					continue;
				sub("compatdir","compat",tmp);
				sub("compattestdir","compat",tmp);
				list[xfile] = tmp
				emit(xfile)
			}
			next
		}

		{
			if ($1 in notwanted)
				next;
			if (! obsolete) {
				list[$1] = $0
				emit($1)
			}
		}

		END {
			for (i in list) {
				if (i in emitf)
					print list[i]
				if (! (i in emitcompat))
					continue;
				l_i = length(i)
				l = 0
				for (j in cpaths) {
					lx = length(cpaths[j])
					if (lx >= l_i || cpaths[j] != substr(i, 1, lx)) {
						continue;
					}
					if (lx > l) {
						l = lx;
						cpath = cpaths[j];
					}
				}
				for (d in compatarchdirs) {
					if (takecompat[$1] &&
					    !takecompatarch[compatarchdirs[d],
						i]) {
						continue
					}
					tmp = list[i]
					extrapath = compatarchdirs[d] "/"
					xfile = cpath extrapath substr(i, l + 1)
					if (xfile in notwanted)
						continue;
					sub("compatfile","compat",tmp);
					sub("compattestfile","compat",tmp);
					tmp = xfile substr(tmp, l_i + 1)
					print tmp;
				}
			}
		}'

}

#
# list_set_lists setname
#
# Print to stdout a list of files, one filename per line, which
# concatenate to create the packing list for setname. E.g.,
#
# 	.../lists/base/mi
# 	.../lists/base/rescue.mi
# 	.../lists/base/md.i386
#		[...]
#
# For a given setname $set, the following files may be selected from
# .../list/$set:
#	mi
#	mi.ext.*
#	ad.${MACHINE_ARCH}
# (or)	ad.${MACHINE_CPU}
#	ad.${MACHINE_CPU}.shl
#	md.${MACHINE}.${MACHINE_ARCH}
# (or)	md.${MACHINE}
#	stl.mi
#	stl.${stlib}
#	shl.mi
#	shl.mi.ext.*
#	shl.${shlib}
#	shl.${shlib}.ext.*
#	module.mi			if ${module} != no
#	module.${MACHINE}		if ${module} != no
#	module.ad.${MACHINE_ARCH}	if ${module} != no
# (or)	module.ad.${MACHINE_CPU}	if ${module} != no
#	rescue.shl
#	rescue.${MACHINE}
#	rescue.ad.${MACHINE_ARCH}
# (or)	rescue.ad.${MACHINE_CPU}
# 	rescue.ad.${MACHINE_CPU}.shl
#
# Environment:
# 	shlib
# 	stlib
#
list_set_lists()
{
	setname=$1

	list_set_lists_mi $setname
	list_set_lists_ad $setname
	list_set_lists_md $setname
	list_set_lists_stl $setname
	list_set_lists_shl $setname
	list_set_lists_module $setname
	list_set_lists_rescue $setname
	return 0
}

list_set_lists_mi()
{
	setdir=$setsdir/lists/$1
	# always exist!
	echo $setdir/mi
}

list_set_lists_ad()
{
	setdir=$setsdir/lists/$1
	[ "${MACHINE}" != "${MACHINE_ARCH}" ] && \
	list_set_lists_common_ad $1
}

list_set_lists_md()
{
	setdir=$setsdir/lists/$1
	echo_if_exist $setdir/md.${MACHINE}.${MACHINE_ARCH} || \
	echo_if_exist $setdir/md.${MACHINE}
}

list_set_lists_stl()
{
	setdir=$setsdir/lists/$1
	echo_if_exist $setdir/stl.mi
	echo_if_exist $setdir/stl.${stlib}
}

list_set_lists_shl()
{
	setdir=$setsdir/lists/$1
	[ "$shlib" != "no" ] || return
	echo_if_exist $setdir/shl.mi
	echo_if_exist $setdir/shl.${shlib}
}

list_set_lists_module()
{
	setdir=$setsdir/lists/$1
	[ "$module" != "no" ] || return
	echo_if_exist $setdir/module.mi
	echo_if_exist $setdir/module.${MACHINE}
	echo_if_exist $setdir/module.ad.${MACHINE}
	echo_if_exist $setdir/module.md.${MACHINE}
	# XXX module never has .shl
	[ "${MACHINE}" != "${MACHINE_ARCH}" ] && \
	list_set_lists_common_ad $1 module
}

list_set_lists_rescue()
{
	setdir=$setsdir/lists/$1
	echo_if_exist $setdir/rescue.mi
	echo_if_exist $setdir/rescue.${MACHINE}
	[ "${MACHINE}" != "${MACHINE_ARCH}" ] && \
	list_set_lists_common_ad $1 rescue
}

list_set_lists_common_ad()
{
	setdir=$setsdir/lists/$1; _prefix=$2

	[ -n "$_prefix" ] && prefix="$_prefix".

	# Prefer a <prefix>.ad.${MACHINE_ARCH} over a
	# <prefix>.ad.${MACHINE_CPU}, since the arch-
	# specific one will be more specific than the
	# cpu-specific one.
	echo_if_exist $setdir/${prefix}ad.${MACHINE_ARCH} || \
	echo_if_exist $setdir/${prefix}ad.${MACHINE_CPU}
	[ "$shlib" != "no" ] && \
	echo_if_exist $setdir/${prefix}ad.${MACHINE_CPU}.shl
}

echo_if_exist()
{
	[ -f $1 ] && echo $1
	return $?
}

echo_if_exist_foreach()
{
	local _list=$1; shift
	for _suffix in $@; do
		echo_if_exist ${_list}.${_suffix}
	done
}

print_set_lists()
{
	for setname; do
		list=$(list_set_lists $setname)
		for l in $list; do
			echo $l
			if $verbose; then
				echo >&2 "DEBUG: list_set_files: $l"
			fi
		done \
		| ${XARGS} ${SED} ${SUBST} \
		| case $setname in
		base|debug)
			awk '
				!/^#/ && !/^$/ {
					print $1, $2, \
					    ($3 ? $3"," : "")"omitcompat"
				}
			'
			;;
		*)	cat
			;;
		esac

		case $setname in
		base32|base64)
			ursetname=base
			;;
		debug32|debug64)
			ursetname=debug
			;;
		*)	continue
			;;
		esac
		case $setname in
		*32)	compatarches=$compat32arches
			;;
		*64)	compatarches=$compat64arches
			;;
		esac
		list=$(list_set_lists $ursetname)
		for l in $list; do
			echo $l
			if $verbose; then
				echo >&2 "DEBUG: list_set_files: $l"
			fi
		done \
		| ${XARGS} ${SED} ${SUBST} \
		| awk -v compatarches="$compatarches" '
			BEGIN {
				split(compatarches, compatarch, " ")
				flags = ""
				for (i in compatarch)
					flags = (flags ? flags"," : "") \
					    "takecompat="compatarch[i]
			}
			!/^#/ && !/^$/ {
				print $1, $2, ($3 ? $3"," : "")flags
			}
		'
	done
}


list_kernel_configs()
{
	(cd ${NETBSDSRCDIR}/etc
	MAKEFLAGS= \
	${MAKE} -m ${NETBSDSRCDIR}/share/mk -V '${ALL_KERNELS}')
}

# arch_to_cpu mach
#
# Print the ${MACHINE_CPU} for ${MACHINE_ARCH}=mach,
# as determined by <bsd.own.mk>.
#
arch_to_cpu()
{
	MACHINE_ARCH=${1} MAKEFLAGS= \
	${MAKE} -m ${NETBSDSRCDIR}/share/mk \
		-f ${NETBSDSRCDIR}/share/mk/bsd.own.mk \
		-V '${MACHINE_CPU}'
}

# arch_to_endian mach
#
# Print the ${TARGET_ENDIANNESS} for ${MACHINE_ARCH}=mach,
# as determined by <bsd.endian.mk>.
#
arch_to_endian()
{
	MACHINE_ARCH=${1} MAKEFLAGS= \
	${MAKE} -m ${NETBSDSRCDIR}/share/mk \
		-f ${NETBSDSRCDIR}/share/mk/bsd.endian.mk \
		-V '${TARGET_ENDIANNESS}'
}

#####

# print_mkvars
print_mkvars()
{
	for v in $MKVARS; do
		eval echo $v=\$$v
	done
}

# print_set_lists_{base,x,ext}
# list_set_lists_{base,x,ext}
# list_set_files_{base,x,ext}
for func in print_set_lists list_set_lists list_set_files; do
	for x in base x ext; do
		if [ $x = base ]; then
			list=nlists
		else
			list=${x}lists
		fi
		eval ${func}_${x} \(\) \{ $func \$$list \; \}
	done
done