Lucio Andrés Illanes Albornoz aeeaa0
#
8d7a8a
# Copyright (c) 2016, 2017, 2018, 2019, 2020, 2021, 2022, 2023 LucĂ­a Andrea Illanes Albornoz <lucia@luciaillanes.de>
Lucio Andrés Illanes Albornoz aeeaa0
# set +o errexit -o noglob -o nounset is assumed.
Lucio Andrés Illanes Albornoz aeeaa0
#
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_check_prereqsV() - check for existence of list of commands
4243a1
# @_rstatus:	out reference to status string
4243a1
# @...:		commands list as positional parameters
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure
4243a1
#
4243a1
rtl_check_prereqsV() {
e9fa07
	local	_rcp_rstatus="${1#\$}"	\
e9fa07
		_rcp_cmd="" _rcp_cmds_missing="" _rcp_rc=0;
e9fa07
	shift;
e9fa07
e9fa07
	for _rcp_cmd in "${@}"; do
e9fa07
		if ! which "${_rcp_cmd}" >/dev/null 2>&1; then
e9fa07
			_rcp_cmds_missing="${_rcp_cmds_missing:+${_rcp_cmds_missing} }${_rcp_cmd}";
Lucio Andrés Illanes Albornoz aeeaa0
		fi;
Lucio Andrés Illanes Albornoz aeeaa0
	done;
e9fa07
	if [ "${_rcp_cmds_missing:+1}" = 1 ]; then
e9fa07
		_rcp_rc=1;
e9fa07
		rtl_setrstatus "${_rcp_rstatus}" 'Error: missing prerequisite package(s): '"${_rcp_cmds_missing}";
Lucio Andrés Illanes Albornoz aeeaa0
	fi;
e9fa07
	return "${_rcp_rc}";
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_clean_env() - unset environment variables w/ exceptions
4243a1
# @_env_vars_except:	list of environment variables to keep
4243a1
#
4243a1
# Returns:		zero (0) on success, non-zero (>0) on failure
4243a1
#
Lucio Andrés Illanes Albornoz aeeaa0
rtl_clean_env() {
e9fa07
	local	_rce_env_vars_except="${1}"	\
e9fa07
		_rce_env_var="" _rce_env_vars="" _rce_env_vars_unset="";
e9fa07
e9fa07
	_rce_env_vars="$(export | sed -ne '/^export/{s/^export //;s/=.*$//p}')";
e9fa07
e9fa07
	for _rce_env_var in ${_rce_env_vars}; do
e9fa07
		if [ "${_rce_env_var#DEFAULT_}" != "${_rce_env_var}" ]\
e9fa07
		|| [ "${_rce_env_var#PKG_}" != "${_rce_env_var}" ]; then
e9fa07
			rtl_lconcat \$_rce_env_vars_except "${_rce_env_var}";
Lucio Andrés Illanes Albornoz aeeaa0
		fi;
Lucio Andrés Illanes Albornoz aeeaa0
	done;
e9fa07
	rtl_lfilter2 \$_rce_env_vars \$_rce_env_vars_unset "${_rce_env_vars_except}";
e9fa07
	rtl_unset_vars ${_rce_env_vars_unset};
e9fa07
e9fa07
	return 0;
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_get_cpu_count() - obtain CPU count
4243a1
# @_rstatus:	out reference to status string
4243a1
# @_rcount:	out reference to CPU count
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure
4243a1
#
Lucio Andrés Illanes Albornoz aeeaa0
rtl_get_cpu_count() {
e9fa07
	local	_rgcc_rstatus="${1#\$}" _rgcc_rcount="${2#\$}"		\
e9fa07
		_rgcc_line="" _rgcc_ncpus=0 _rgcc_rc=0 _rgcc_sname="";
e9fa07
e9fa07
	_rgcc_sname="$(uname -s 2>/dev/null)" || return 1;
e9fa07
	case "${_rgcc_sname}" in
e9fa07
Lucio Andrés Illanes Albornoz aeeaa0
	Linux)	if [ ! -e "/proc/cpuinfo" ]; then
e9fa07
			_rgcc_rc=1;
e9fa07
			rtl_setrstatus "${_rgcc_rstatus}" 'Error: /proc/cpuinfo non-existent.';
e9fa07
		else	while read -r _rgcc_line; do
e9fa07
				if rtl_match "${_rgcc_line}" "processor*:"; then
e9fa07
					: $((_rgcc_ncpus+=1));
Lucio Andrés Illanes Albornoz aeeaa0
				fi;
e9fa07
			done < /proc/cpuinfo;
e9fa07
			_rgcc_rc=0;
e9fa07
			rtl_setrstatus "${_rgcc_rstatus}" "";
Lucio Andrés Illanes Albornoz aeeaa0
		fi; ;;
e9fa07
e9fa07
	*)	_rgcc_rc=1;
e9fa07
		rtl_setrstatus "${_rgcc_rstatus}" 'Error: unknown platform \`'"${_rgcc_sname}"''\''.';
e9fa07
		;;
e9fa07
e9fa07
	esac;
e9fa07
	eval ${_rgcc_rcount}='${_rgcc_ncpus}';
e9fa07
	return "${_rgcc_rc}";
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_get_var_unsafe() - get value of variable
4243a1
# @_rvval:	out reference to variable value
4243a1
# @[-u]:	optionally convert variable name to upper case
4243a1
# @_vname:	variable name
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure
4243a1
# N.B.:		This function is *unsafe* and impossible to implement otherwise w/o filtering @_vname
4243a1
#		and implicitly allows for code execution and other undefined behaviour via @_vname.
4243a1
#		Do *not* pass untrusted input through @_vname.
4243a1
#
Lucio Andrés Illanes Albornoz aeeaa0
rtl_get_var_unsafe() {
4243a1
	local	_rgvu_rvval="${1#\$}"	\
e9fa07
		_rgvu_vname="";
e9fa07
	shift;
e9fa07
Lucio Andrés Illanes Albornoz aeeaa0
	if [ "x${1}" = "x-u" ]; then
e9fa07
		shift; _rgvu_vname="${1}"; rtl_toupper \$_rgvu_vname;
Lucio Andrés Illanes Albornoz aeeaa0
	else
e9fa07
		_rgvu_vname="${1}";
Lucio Andrés Illanes Albornoz aeeaa0
	fi;
4243a1
	eval ${_rgvu_rvval}="\${${_rgvu_vname}:-}";
e9fa07
	return 0;
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_get_vars_unsafe_fast() - get values of multiple variables w/ pattern
4243a1
# @_pattern:	pattern to match against set output
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure, matching variable values on stdout
4243a1
# N.B.:		This function is *unsafe* and impossible to implement otherwise w/o parsing set
4243a1
#		output properly and may produce spurious data.
4243a1
#
4243a1
rtl_get_vars_unsafe_fast() {
4243a1
	local _rgvuf_pattern="${1}";
e9fa07
4243a1
	set | awk -F= '/'"${_rgvuf_pattern}"'/{print $1}' | sort;
e9fa07
	return 0;
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_kill_tree() - kill tree of processes
4243a1
# @_rpids:	inout reference to list of PIDs
4243a1
# @_pid:	top-level PID
4243a1
# @_signal:	signal(7) to kill with
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure
4243a1
#
Lucio Andrés Illanes Albornoz aeeaa0
rtl_kill_tree() {
e9fa07
	local	_rkt_rpids="${1#\$}" _rkt_pid="${2}" _rkt_signal="${3:-TERM}"	\
e9fa07
		_rkt_pid_child="" _rkt_pids="";
e9fa07
e9fa07
	if _rkt_pids="$(pgrep -P "${_rkt_pid}")"\
e9fa07
	&& [ "${_rkt_pids:+1}" = 1 ]; then
e9fa07
		for _rkt_pid_child in ${_rkt_pids}; do
e9fa07
			rtl_kill_tree "${_rkt_rpids}" "${_rkt_pid_child}" "${_rkt_signal}";
Lucio Andrés Illanes Albornoz aeeaa0
		done;
Lucio Andrés Illanes Albornoz aeeaa0
	fi;
e9fa07
	if [ "${_rkt_pid:-0}" -ne "${$}" ]\
e9fa07
	&& kill "-${_rkt_signal}" "${_rkt_pid}" 2>/dev/null; then
e9fa07
		rtl_lconcat "${_rkt_rpids}" "${_rkt_pid}";
Lucio Andrés Illanes Albornoz aeeaa0
	fi;
e9fa07
	return 0;
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_run_cmdlineV() - run command line w/ field splitting applied
4243a1
# @_sep:	single non-zero, possibly multi-character, separator
4243a1
# @_cmd:	command name
4243a1
# @...:		command arguments as positional parameters
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure
4243a1
# N.B.:		This is required in situations where any of the command arguments 1) are passed from
4243a1
#		and as a single, expanded parameter (e.g. "${MAKEFLAGS:-}") 2) the value of the parameter
4243a1
#		contains multiple parameters separated by @_sep that must be passed as separate arguments
4243a1
#		to @_cmd.
4243a1
#
4243a1
rtl_run_cmdlineV() {
4243a1
	local	_rrc_sep="${1}" _rrc_cmd="${2}"	\
4243a1
		_rrc_cmdline="" _rrc_rc="" IFS;
e9fa07
	shift 2;
e9fa07
Lucio Andrés Illanes Albornoz aeeaa0
	while [ ${#} -gt 0 ]; do
e9fa07
		[ "${1:+1}" = 1 ] &&\
4243a1
			_rrc_cmdline="${_rrc_cmdline:+${_rrc_cmdline}${_rrc_sep}}${1}";
Lucio Andrés Illanes Albornoz aeeaa0
		shift;
Lucio Andrés Illanes Albornoz aeeaa0
	done;
4243a1
	IFS="${_rrc_sep}"; ${_rrc_cmd} ${_rrc_cmdline}; _rrc_rc=$?;
4243a1
	return ${_rrc_rc};
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
4243a1
#
4243a1
# rtl_set_var_from_spec() - set variable from name-value specification
4243a1
# @_rstatus:	out reference to status string
4243a1
# @_arg:	variable name-value specification separated by single "=" (e.g. name=value)
4243a1
#
4243a1
# Returns:	zero (0) on success, non-zero (>0) on failure
4243a1
#
4243a1
rtl_set_var_from_spec() {
4243a1
	local	_rsvfs_rstatus="${1#\$}" _rsvfs_arg="${2}"	\
4243a1
		_rsvfs_rc=0 _rsvfs_vname="" _rsvfs_vval="";
f6f7f2
4243a1
	_rsvfs_vname="${_rsvfs_arg%%=*}";
4243a1
	_rsvfs_vval="${_rsvfs_arg#*=}";
f6f7f2
4243a1
	if [ "${_rsvfs_vval:+1}" != 1 ]; then
4243a1
		_rsvfs_rc=1;
4243a1
		rtl_setrstatus "${_rsvfs_rstatus}" 'empty value specified for \${'"${_rsvfs_vname}"'}.';
f6f7f2
	else
4243a1
		rtl_set_var_unsafe "${_rsvfs_vname}" "${_rsvfs_vval}";
f6f7f2
	fi;
4243a1
	return "${_rsvfs_rc}";
f6f7f2
};
f6f7f2
4243a1
#
4243a1
# rtl_set_var() - set variable from variables w/ template
4243a1
# @_vars_set_vname:	list of variable names
4243a1
# @_vname_dst:		variable name to set
4243a1
# @_vname_src_tmpls:	variable name template (e.g. "DEFAULT PKG_")
4243a1
#
4243a1
# Returns:		zero (0) on success, non-zero (>0) on failure
4243a1
#
4243a1
rtl_set_var() {
e9fa07
	local	_rsv_vars_set_vname="${1}" _rsv_vname_dst="${2}" _rsv_vname_src_tmpls="${3}"	\
e9fa07
		_rsv_vars_set_old="" _rsv_vars_set_tmp="" _rsv_vname_src="" _rsv_vnames_src="";
e9fa07
e9fa07
	rtl_toupper2 \$_rsv_vname_src_tmpls \$_rsv_vnames_src;
e9fa07
	for _rsv_vname_src in ${_rsv_vnames_src}; do
e9fa07
		_rsv_vname_src="${_rsv_vname_src}_${_rsv_vname_dst}";
e9fa07
		eval _rsv_vval_src="\${${_rsv_vname_src}:-}";
e9fa07
		if [ "${_rsv_vval_src:+1}" = 1 ]; then
e9fa07
			eval PKG_${_rsv_vname_dst}='${_rsv_vval_src}';
e9fa07
			_rsv_vars_set_tmp="${_rsv_vars_set_tmp:+${_rsv_vars_set_tmp} }PKG_${_rsv_vname_dst}";
Lucio Andrés Illanes Albornoz aeeaa0
		fi;
Lucio Andrés Illanes Albornoz aeeaa0
	done;
e9fa07
	eval _rsv_vars_set_old="\${${_rsv_vars_set_vname}}";
e9fa07
	rtl_set_var_unsafe "${_rsv_vars_set_vname}" "${_rsv_vars_set_old:+${_rsv_vars_set_old} }${_rsv_vars_set_tmp}";
e9fa07
e9fa07
	return 0;
Lucio Andrés Illanes Albornoz aeeaa0
};
Lucio Andrés Illanes Albornoz aeeaa0
6a630d
#
6a630d
# rtl_set_var_unsafe() - set value of variable
6a630d
# @[-u]:	optionally convert variable name to upper case
6a630d
# @_vname:	variable name
6a630d
# @_vval:	variable value
6a630d
#
6a630d
# Returns:	zero (0) on success, non-zero (>0) on failure
6a630d
# N.B.:		This function is *unsafe* and impossible to implement otherwise w/o filtering @_vname
6a630d
#		and implicitly allows for code execution and other undefined behaviour via @_vname.
6a630d
#		Do *not* pass untrusted input through @_vname.
938c5c
#
6a630d
rtl_set_var_unsafe() {
6a630d
	local _rsvu_vname="" _rsvu_vval="";
6a630d
	if [ "x${1}" = "x-u" ]; then
6a630d
		shift; _rsvu_vname="${1}"; _rsvu_vval="${2}"; rtl_toupper \$_rsvu_vname;
6a630d
	else
6a630d
		_rsvu_vname="${1}"; _rsvu_vval="${2}";
6a630d
	fi;
6a630d
	eval ${_rsvu_vname}='${_rsvu_vval}';
6a630d
	return 0;
6a630d
};
6a630d
64844b
# vim:filetype=sh textwidth=0