# $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