2018-01-06 12:35:33 +11:00
|
|
|
#!/usr/bin/env bash
|
2023-03-09 21:52:19 -08:00
|
|
|
# -*- mode: bash; tab-width: 4 -*-
|
2022-06-26 18:01:27 -07:00
|
|
|
# vi: ts=4:sw=4:et
|
2023-03-09 21:52:19 -08:00
|
|
|
# vim: filetype=bash
|
2011-11-07 22:28:40 +01:00
|
|
|
|
2017-08-18 23:55:24 -07:00
|
|
|
if [ "${BASH_VERSINFO[0]}" -lt 4 ]; then
|
|
|
|
echo "Your BASH shell version (${BASH_VERSION}) is too old." >&2
|
|
|
|
echo "Run bootstrap on a machine with BASH 4.x" >&2
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
|
2020-07-10 19:30:16 -04:00
|
|
|
# see #849
|
|
|
|
unset CDPATH
|
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
########################################
|
|
|
|
# Common meta-language implementation. Syntax:
|
|
|
|
#
|
|
|
|
# The template file is processed line by line, with @@VAR@@ placeholders
|
|
|
|
# being replaced with a value of the VAR variable.
|
|
|
|
# Special lines start with '#!' and a keyword:
|
|
|
|
#
|
|
|
|
# #!//
|
|
|
|
# Comment, the rest of the line is ignored
|
|
|
|
# #!if COND
|
|
|
|
# Conditional: the lines until the matching #!end-if are processed
|
|
|
|
# only if the conditional COND evaluates to true.
|
|
|
|
# #!foreach NAME
|
|
|
|
# Iterate over NAME entities (the iterator must be set up first
|
|
|
|
# using the set_iter function), processing the lines until the matching
|
|
|
|
# #!end-foreach line.
|
2018-11-23 22:14:07 -08:00
|
|
|
#
|
|
|
|
# Also, several forms of @@VAR@@ expansions are possible:
|
|
|
|
#
|
|
|
|
# @@VAR@@
|
|
|
|
# Just the value of the variable VAR
|
|
|
|
# @@VAR|@@
|
|
|
|
# The value of VAR made into Kconfig-name: all non-alphanumeric character
|
|
|
|
# replaced with underscores; upper-cased.
|
|
|
|
# @@VAR?val@@
|
|
|
|
# If VAR is non-empty, insert value "val". Otherwise, insert nothing.
|
|
|
|
# @@*ITER@@
|
|
|
|
# Expands to a space separated list of values for the iterator ITER.
|
|
|
|
# Postprocess operations, if any, are applied to each value.
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
declare -A info
|
|
|
|
|
|
|
|
debug()
|
|
|
|
{
|
|
|
|
if [ -n "${DEBUG}" ]; then
|
|
|
|
echo "DEBUG :: $@" >&2
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
msg()
|
|
|
|
{
|
|
|
|
if [ -z "${QUIET}" ]; then
|
|
|
|
echo "INFO :: $@" >&2
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
warn()
|
|
|
|
{
|
|
|
|
echo "WARN :: $@" >&2
|
|
|
|
}
|
|
|
|
|
|
|
|
error()
|
|
|
|
{
|
|
|
|
echo "ERROR :: $@" >&2
|
|
|
|
exit 1
|
|
|
|
}
|
|
|
|
|
|
|
|
find_end()
|
|
|
|
{
|
|
|
|
local token="${1}"
|
|
|
|
local count=1
|
|
|
|
|
|
|
|
# Skip first line, we know it has the proper '#!' command on it
|
|
|
|
endline=$[l + 1]
|
|
|
|
while [ "${endline}" -le "${end}" ]; do
|
|
|
|
case "${tlines[${endline}]}" in
|
|
|
|
"#!${token} "*)
|
|
|
|
count=$[count + 1]
|
|
|
|
;;
|
|
|
|
"#!end-${token}")
|
|
|
|
count=$[count - 1]
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
if [ "${count}" = 0 ]; then
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
endline=$[endline + 1]
|
|
|
|
done
|
2017-11-29 00:14:04 -08:00
|
|
|
error "${template}:${l}: '${token}' token is unpaired"
|
2017-07-04 19:19:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
set_iter()
|
|
|
|
{
|
|
|
|
local name="${1}"
|
2017-08-24 14:58:14 -07:00
|
|
|
local -a temp
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
if [ "${info[iter_${name}]+set}" = "set" ]; then
|
|
|
|
error "Iterator over '${name}' is already set up"
|
|
|
|
fi
|
|
|
|
shift
|
|
|
|
debug "Setting iterator over '${name}' to '$*'"
|
2017-08-24 14:58:14 -07:00
|
|
|
temp=("$@")
|
2017-07-04 19:19:42 -07:00
|
|
|
info[iter_${name}]="$*"
|
2017-08-24 14:58:14 -07:00
|
|
|
info[#${name}]=${#temp[@]}
|
2017-07-04 19:19:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
run_if()
|
|
|
|
{
|
2017-08-24 15:29:55 -07:00
|
|
|
local cond="$*"
|
2017-07-04 19:19:42 -07:00
|
|
|
local endline
|
|
|
|
|
|
|
|
find_end "if"
|
|
|
|
if eval "${cond}"; then
|
|
|
|
debug "True conditional '${cond}' at lines ${l}..${endline}"
|
|
|
|
run_lines $[l + 1] $[endline - 1]
|
|
|
|
else
|
|
|
|
debug "False conditional '${cond}' at lines ${l}..${endline}"
|
|
|
|
fi
|
|
|
|
lnext=$[endline + 1]
|
|
|
|
debug "Continue at line ${lnext}"
|
|
|
|
}
|
|
|
|
|
|
|
|
do_foreach()
|
|
|
|
{
|
|
|
|
local var="${1}"
|
2017-07-06 18:20:44 -07:00
|
|
|
local -A saveinfo
|
|
|
|
local v k
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
shift
|
|
|
|
if [ "`type -t enter_${var}`" != "function" ]; then
|
|
|
|
error "No parameter setup routine for iterator over '${var}'"
|
|
|
|
fi
|
2019-02-13 15:51:00 -08:00
|
|
|
if [ "x${info[iter_${var}]+set}" != "xset" ]; then
|
|
|
|
error "Iterator over '${var}' not configured"
|
|
|
|
fi
|
2017-07-04 19:19:42 -07:00
|
|
|
for v in ${info[iter_${var}]}; do
|
2017-07-06 18:20:44 -07:00
|
|
|
# This works in bash 4.4, but not in bash 4.3:
|
|
|
|
# local saveinfo=`declare -p info`
|
|
|
|
# ...
|
|
|
|
# eval "${saveinfo}"
|
|
|
|
# Therefore, need to save key-by-key
|
|
|
|
saveinfo=()
|
|
|
|
for k in "${!info[@]}"; do
|
|
|
|
saveinfo["${k}"]=${info["${k}"]}
|
|
|
|
done
|
2017-08-24 15:29:55 -07:00
|
|
|
if eval "enter_${var} ${v}"; then
|
|
|
|
eval "$@"
|
|
|
|
fi
|
2017-07-06 18:20:44 -07:00
|
|
|
info=()
|
|
|
|
for k in "${!saveinfo[@]}"; do
|
|
|
|
info["${k}"]=${saveinfo["${k}"]}
|
|
|
|
done
|
2017-07-04 19:19:42 -07:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
run_foreach()
|
|
|
|
{
|
|
|
|
local endline
|
2017-08-24 15:29:55 -07:00
|
|
|
local var="${1}"
|
|
|
|
shift
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
if [ "${info[iter_${var}]+set}" != "set" ]; then
|
2017-11-29 00:14:04 -08:00
|
|
|
error "${template}:${l}: iterator over '${var}' is not defined"
|
2017-07-04 19:19:42 -07:00
|
|
|
fi
|
|
|
|
find_end "foreach"
|
|
|
|
debug "Loop over '${var}', lines ${l}..${endline}"
|
2017-08-24 15:29:55 -07:00
|
|
|
do_foreach ${var} run_lines_if $[l + 1] $[endline - 1] "$*"
|
2017-07-04 19:19:42 -07:00
|
|
|
lnext=$[endline + 1]
|
|
|
|
debug "Continue at line ${lnext}"
|
|
|
|
}
|
|
|
|
|
2017-08-24 15:29:55 -07:00
|
|
|
run_lines_if()
|
|
|
|
{
|
|
|
|
local start="${1}"
|
|
|
|
local end="${2}"
|
|
|
|
shift 2
|
|
|
|
local cond="$*"
|
|
|
|
local a prev
|
|
|
|
|
|
|
|
for a in ${cond}; do
|
|
|
|
if [ -n "${prev}" ]; then
|
|
|
|
case "${prev}" in
|
|
|
|
if-differs)
|
|
|
|
if [ "${info[${a}]}" = "${saveinfo[${a}]}" ]; then
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
;;
|
|
|
|
*)
|
2017-11-29 00:14:04 -08:00
|
|
|
error "${template}:${l}: unknown condition '${prev}' for loop"
|
2017-08-24 15:29:55 -07:00
|
|
|
;;
|
|
|
|
esac
|
|
|
|
prev=
|
|
|
|
else
|
|
|
|
prev=${a}
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
run_lines "${start}" "${end}"
|
|
|
|
}
|
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
run_lines()
|
|
|
|
{
|
|
|
|
local start="${1}"
|
|
|
|
local end="${2}"
|
2018-11-23 22:14:07 -08:00
|
|
|
local l lnext s s1 v vn vp pp p val val0 is_iter pp_saved
|
|
|
|
local -a vpl
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
debug "Running lines ${start}..${end}"
|
|
|
|
l=${start}
|
|
|
|
while [ "${l}" -le "${end}" ]; do
|
|
|
|
lnext=$[l+1]
|
|
|
|
s="${tlines[${l}]}"
|
|
|
|
# Expand @@foo@@ to ${info[foo]}. First escape variables/backslashes for evals below.
|
|
|
|
s="${s//\\/\\\\}"
|
|
|
|
s="${s//\$/\\\$}"
|
|
|
|
s1=
|
|
|
|
while [ -n "${s}" ]; do
|
|
|
|
case "${s}" in
|
|
|
|
*@@*@@*)
|
|
|
|
v="${s#*@@}"
|
|
|
|
v="${v%%@@*}"
|
2018-11-23 22:14:07 -08:00
|
|
|
# $v now has the complete reference. First check if it is cached already.
|
2017-07-04 19:19:42 -07:00
|
|
|
if [ "${info[${v}]+set}" != "set" ]; then
|
2018-11-23 22:14:07 -08:00
|
|
|
case "${v}" in
|
|
|
|
"*"*) is_iter=y; vn="${v#\*}";;
|
|
|
|
*) is_iter=n; vn="${v}";;
|
|
|
|
esac
|
|
|
|
# $vn is now the reference without the preceding iterator
|
|
|
|
vp="${vn%%[|?]*}"
|
|
|
|
pp="${vn#${vp}}"
|
|
|
|
# $vp is name of the variable proper, $pp is any postprocessing
|
|
|
|
if [ "${is_iter}" = "n" ]; then
|
|
|
|
if [ "${info[${vp}]+set}" != "set" ]; then
|
|
|
|
error "${template}:${l}: reference to undefined variable '${vp}'"
|
|
|
|
fi
|
|
|
|
vpl=( "${info[${vp}]}" )
|
|
|
|
else
|
|
|
|
if [ "${info[iter_${vp}]+set}" != "set" ]; then
|
|
|
|
error "${template}:${l}: iterator over '${vp} is not defined"
|
|
|
|
fi
|
|
|
|
vpl=( ${info[iter_${vp}]} )
|
|
|
|
fi
|
|
|
|
# ${vpl[@]} now is an array of values to be transformed.
|
|
|
|
val=
|
|
|
|
pp_saved="${pp}"
|
|
|
|
for val0 in "${vpl[@]}"; do
|
|
|
|
debug "val0 [${val0}]"
|
|
|
|
pp="${pp_saved}"
|
|
|
|
# Apply postprocessing(s)
|
|
|
|
while [ -n "${pp}" ]; do
|
|
|
|
case "${pp}" in
|
|
|
|
"|"*)
|
|
|
|
# Kconfigize
|
|
|
|
pp="${pp#|}"
|
|
|
|
val0=${val0//[^0-9A-Za-z_]/_}
|
|
|
|
val0=${val0^^}
|
|
|
|
;;
|
|
|
|
"?"*)
|
|
|
|
pp="${pp#?}"
|
|
|
|
p="${pp%%[|?]*}"
|
|
|
|
pp="${pp#${p}}"
|
|
|
|
val0="${val0:+${p}}"
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
val="${val:+${val} }${val0}"
|
2017-08-24 14:58:14 -07:00
|
|
|
done
|
2018-11-23 22:14:07 -08:00
|
|
|
# Cache for future references.
|
2017-08-24 14:58:14 -07:00
|
|
|
info[${v}]="${val}"
|
2017-07-04 19:19:42 -07:00
|
|
|
fi
|
|
|
|
s1="${s1}${s%%@@*}\${info[${v}]}"
|
|
|
|
s="${s#*@@*@@}"
|
|
|
|
;;
|
|
|
|
*@@*)
|
2017-11-29 00:14:04 -08:00
|
|
|
error "${template}:${l}: non-paired @@ markers"
|
2017-07-04 19:19:42 -07:00
|
|
|
;;
|
|
|
|
*)
|
|
|
|
s1="${s1}${s}"
|
|
|
|
break
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
s=${s1}
|
|
|
|
|
|
|
|
debug "Evaluate: ${s}"
|
|
|
|
case "${s}" in
|
|
|
|
"#!if "*)
|
2017-08-24 15:29:55 -07:00
|
|
|
run_if ${s#* }
|
2017-07-04 19:19:42 -07:00
|
|
|
;;
|
|
|
|
"#!foreach "*)
|
2017-08-24 15:29:55 -07:00
|
|
|
run_foreach ${s#* }
|
2017-07-04 19:19:42 -07:00
|
|
|
;;
|
|
|
|
"#!//"*)
|
|
|
|
# Comment, do nothing
|
|
|
|
;;
|
|
|
|
"#!"*)
|
2017-11-29 00:14:04 -08:00
|
|
|
error "${template}:${l}: unrecognized command"
|
2017-07-04 19:19:42 -07:00
|
|
|
;;
|
|
|
|
*)
|
|
|
|
# Not a special command
|
|
|
|
eval "echo \"${s//\"/\\\"}\""
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
l=${lnext}
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
run_template()
|
|
|
|
{
|
|
|
|
local -a tlines
|
|
|
|
local src="${1}"
|
|
|
|
|
|
|
|
if [ ! -r "${src}" ]; then
|
|
|
|
error "Template '${src}' not found"
|
|
|
|
fi
|
2017-11-29 00:14:04 -08:00
|
|
|
template="${src}"
|
2017-07-04 19:19:42 -07:00
|
|
|
debug "Running template ${src}"
|
|
|
|
mapfile -O 1 -t tlines < "${src}"
|
|
|
|
run_lines 1 ${#tlines[@]}
|
|
|
|
}
|
|
|
|
|
|
|
|
########################################
|
|
|
|
|
2017-08-24 14:58:14 -07:00
|
|
|
# Leave only relevant portion of the string
|
|
|
|
relevantize()
|
2017-07-04 19:19:42 -07:00
|
|
|
{
|
2017-07-06 22:18:43 -07:00
|
|
|
local p pb pa vx
|
2017-08-24 14:58:14 -07:00
|
|
|
local v="${1}"
|
2017-07-06 22:18:43 -07:00
|
|
|
shift
|
|
|
|
|
2017-08-24 14:58:14 -07:00
|
|
|
# Find the first match and contract to the matching portion.
|
2017-07-06 22:18:43 -07:00
|
|
|
for p in "$@"; do
|
|
|
|
pb=${p%|*}
|
|
|
|
pa=${p#*|}
|
|
|
|
eval "vx=\${v#${pb}${pa}}"
|
|
|
|
if [ "${v%${pa}${vx}}" != "${v}" ]; then
|
|
|
|
v=${v%${pa}${vx}}
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
2017-08-24 14:58:14 -07:00
|
|
|
echo "${v}"
|
2017-07-04 19:19:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
# Helper for cmp_versions: compare an upstream/debian portion of
|
|
|
|
# a version. Returns 0 if equal, otherwise echoes "-1" or "1" and
|
|
|
|
# returns 1.
|
|
|
|
equal_versions()
|
|
|
|
{
|
|
|
|
local v1="${1}"
|
|
|
|
local v2="${2}"
|
|
|
|
local p1 p2
|
|
|
|
|
|
|
|
# Compare alternating non-numerical/numerical portions, until
|
|
|
|
# non-equal portion is found or either string is exhausted.
|
|
|
|
while [ -n "${v1}" -a -n "${v2}" ]; do
|
|
|
|
# Find non-numerical portions and compare lexicographically
|
|
|
|
p1="${v1%%[0-9]*}"
|
|
|
|
p2="${v2%%[0-9]*}"
|
|
|
|
v1="${v1#${p1}}"
|
|
|
|
v2="${v2#${p2}}"
|
|
|
|
#debug "lex [${p1}] v [${p2}]"
|
|
|
|
if [ "${p1}" \< "${p2}" ]; then
|
|
|
|
echo "-1"
|
|
|
|
return 1
|
|
|
|
elif [ "${p1}" \> "${p2}" ]; then
|
|
|
|
echo "1"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
#debug "rem [${v1}] v [${v2}]"
|
|
|
|
# Find numerical portions and compare numerically
|
|
|
|
p1="${v1%%[^0-9]*}"
|
|
|
|
p2="${v2%%[^0-9]*}"
|
|
|
|
v1="${v1#${p1}}"
|
|
|
|
v2="${v2#${p2}}"
|
|
|
|
#debug "num [${p1}] v [${p2}]"
|
|
|
|
if [ "${p1:-0}" -lt "${p2:-0}" ]; then
|
|
|
|
echo "-1"
|
|
|
|
return 1
|
|
|
|
elif [ "${p1:-0}" -gt "${p2:-0}" ]; then
|
|
|
|
echo "1"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
#debug "rem [${v1}] v [${v2}]"
|
|
|
|
done
|
|
|
|
if [ -n "${v1}" ]; then
|
|
|
|
echo "1"
|
|
|
|
return 1
|
|
|
|
elif [ -n "${v2}" ]; then
|
|
|
|
echo "-1"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
# Compare two version strings, similar to sort -V. But we don't
|
|
|
|
# want to depend on GNU sort availability on the host.
|
|
|
|
# See http://www.debian.org/doc/debian-policy/ch-controlfields.html
|
|
|
|
# for description of what the version is expected to be.
|
|
|
|
# Returns "-1", "0" or "1" if first version is earlier, same or
|
|
|
|
# later than the second.
|
|
|
|
cmp_versions()
|
|
|
|
{
|
|
|
|
local v1="${1}"
|
|
|
|
local v2="${2}"
|
2020-08-21 09:26:19 +02:00
|
|
|
local e1=0 e2=0 d1=0 d2=0
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
# Case-insensitive comparison
|
|
|
|
v1="${v1^^}"
|
|
|
|
v2="${v2^^}"
|
|
|
|
|
|
|
|
# Find if the versions contain epoch part
|
|
|
|
case "${v1}" in
|
|
|
|
*:*)
|
|
|
|
e1="${v1%%:*}"
|
|
|
|
v1="${v1#*:}"
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
case "${v2}" in
|
|
|
|
*:*)
|
|
|
|
e2="${v2%%:*}"
|
|
|
|
v2="${v2#*:}"
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
# Compare epochs numerically
|
|
|
|
if [ "${e1}" -lt "${e2}" ]; then
|
|
|
|
echo "-1"
|
|
|
|
return
|
|
|
|
elif [ "${e1}" -gt "${e2}" ]; then
|
|
|
|
echo "1"
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Find if the version contains a "debian" part.
|
|
|
|
# v1/v2 will now contain "upstream" part.
|
|
|
|
case "${v1}" in
|
|
|
|
*-*)
|
|
|
|
d1=${v1##*-}
|
|
|
|
v1=${v1%-*}
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
case "${v2}" in
|
|
|
|
*-*)
|
|
|
|
d2=${v2##*-}
|
|
|
|
v2=${v2%-*}
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
# Compare upstream
|
|
|
|
if equal_versions "${v1}" "${v2}" && equal_versions "${d1}" "${d2}"; then
|
|
|
|
echo "0"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
# Sort versions, descending
|
|
|
|
sort_versions()
|
|
|
|
{
|
|
|
|
local sorted
|
|
|
|
local remains="$*"
|
|
|
|
local next_remains
|
|
|
|
local v vx found
|
|
|
|
|
|
|
|
while [ -n "${remains}" ]; do
|
|
|
|
#debug "Sorting [${remains}]"
|
|
|
|
for v in ${remains}; do
|
|
|
|
found=yes
|
|
|
|
next_remains=
|
|
|
|
#debug "Candidate ${v}"
|
|
|
|
for vx in ${remains}; do
|
|
|
|
#debug "${v} vs ${vx} :: `cmp_versions ${v} ${vx}`"
|
|
|
|
case `cmp_versions ${v} ${vx}` in
|
|
|
|
1)
|
2022-06-26 18:01:27 -07:00
|
|
|
next_remains+=" ${vx}"
|
|
|
|
;;
|
2017-07-04 19:19:42 -07:00
|
|
|
0)
|
2022-06-26 18:01:27 -07:00
|
|
|
;;
|
2017-07-04 19:19:42 -07:00
|
|
|
-1)
|
2022-06-26 18:01:27 -07:00
|
|
|
found=no
|
|
|
|
#debug "Bad: earlier than ${vx}"
|
|
|
|
break
|
|
|
|
;;
|
2017-07-04 19:19:42 -07:00
|
|
|
esac
|
|
|
|
done
|
|
|
|
if [ "${found}" = "yes" ]; then
|
|
|
|
# $v is less than all other members in next_remains
|
|
|
|
sorted+=" ${v}"
|
|
|
|
remains="${next_remains}"
|
|
|
|
#debug "Good candidate ${v} sorted [${sorted}] remains [${remains}]"
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
done
|
|
|
|
echo "${sorted}"
|
|
|
|
}
|
|
|
|
|
|
|
|
read_file()
|
|
|
|
{
|
|
|
|
local l p
|
|
|
|
|
|
|
|
while read l; do
|
|
|
|
l="${p}${l}"
|
|
|
|
p=
|
|
|
|
case "${l}" in
|
|
|
|
"")
|
|
|
|
continue
|
|
|
|
;;
|
|
|
|
*\\)
|
|
|
|
p="${l%\\}"
|
|
|
|
continue
|
|
|
|
;;
|
|
|
|
"#"*)
|
|
|
|
continue
|
|
|
|
;;
|
|
|
|
*=*)
|
|
|
|
echo "info[${l%%=*}]=${l#*=}"
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
error "syntax error in '${1}': '${l}'"
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done < "${1}"
|
|
|
|
}
|
2011-11-07 22:28:40 +01:00
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
read_package_desc()
|
|
|
|
{
|
|
|
|
read_file "packages/${1}/package.desc"
|
|
|
|
}
|
2017-04-22 18:41:50 -07:00
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
read_version_desc()
|
|
|
|
{
|
|
|
|
read_file "packages/${1}/${2}/version.desc"
|
|
|
|
}
|
|
|
|
|
|
|
|
find_forks()
|
|
|
|
{
|
|
|
|
local -A info
|
|
|
|
|
|
|
|
info[preferred]=${1}
|
|
|
|
eval `read_package_desc ${1}`
|
|
|
|
|
|
|
|
if [ -n "${info[master]}" ]; then
|
|
|
|
pkg_nforks[${info[master]}]=$[pkg_nforks[${info[master]}]+1]
|
|
|
|
pkg_forks[${info[master]}]+=" ${1} "
|
|
|
|
else
|
|
|
|
pkg_preferred[${1}]=${info[preferred]}
|
|
|
|
pkg_nforks[${1}]=$[pkg_nforks[${1}]+1]
|
|
|
|
pkg_forks[${1}]+=" ${1} "
|
|
|
|
pkg_milestones[${1}]=`sort_versions ${info[milestones]}`
|
2017-07-06 22:18:43 -07:00
|
|
|
pkg_relevantpattern[${1}]=${info[relevantpattern]}
|
2017-07-04 19:19:42 -07:00
|
|
|
pkg_masters+=( "${1}" )
|
|
|
|
fi
|
|
|
|
# Keep sorting so that preferred fork is first
|
|
|
|
if [ -n "${pkg_preferred[${1}]}" ]; then
|
|
|
|
pkg_forks[${1}]="${pkg_preferred[${1}]} ${pkg_forks[${1}]##* ${pkg_preferred[${1}]} } ${pkg_forks[${1}]%% ${pkg_preferred[${1}]} *}"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
enter_fork()
|
|
|
|
{
|
|
|
|
local fork="${1}"
|
|
|
|
local versions
|
|
|
|
local only_obsolete only_experimental
|
2019-02-13 15:51:00 -08:00
|
|
|
local -A seen_selectors
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
# Set defaults
|
|
|
|
info[obsolete]=
|
|
|
|
info[experimental]=
|
|
|
|
info[repository]=
|
|
|
|
info[repository_branch]=
|
|
|
|
info[repository_cset]=
|
|
|
|
info[repository_subdir]=
|
|
|
|
info[bootstrap]=
|
|
|
|
info[fork]=${fork}
|
2017-07-08 10:54:53 -07:00
|
|
|
info[pkg_name]=${fork}
|
|
|
|
info[pkg_label]=${fork}
|
2017-07-04 19:19:42 -07:00
|
|
|
info[mirrors]=
|
2019-05-27 13:17:33 +00:00
|
|
|
info[src_release]=
|
|
|
|
info[src_devel]=
|
|
|
|
info[src_custom]=
|
2017-07-04 19:19:42 -07:00
|
|
|
info[archive_filename]='@{pkg_name}-@{version}'
|
|
|
|
info[archive_dirname]='@{pkg_name}-@{version}'
|
2017-08-24 14:58:14 -07:00
|
|
|
info[versionlocked]=
|
|
|
|
info[origin]=
|
2017-09-27 22:29:35 -07:00
|
|
|
info[signature_format]=
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
eval `read_package_desc ${fork}`
|
|
|
|
|
|
|
|
if [ -r "packages/${info[origin]}.help" ]; then
|
|
|
|
info[originhelp]=`sed 's/^/ /' "packages/${info[origin]}.help"`
|
|
|
|
else
|
|
|
|
info[originhelp]=" ${info[master]} from ${info[origin]}."
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ -n "${info[repository]}" ]; then
|
|
|
|
info[vcs]=${info[repository]%% *}
|
|
|
|
info[repository_url]=${info[repository]#* }
|
|
|
|
fi
|
|
|
|
|
2020-12-14 21:43:37 +13:00
|
|
|
info[mirrors]=${info[mirrors]//$\(/\\$\(}
|
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
versions=`cd packages/${fork} && \
|
|
|
|
for f in */version.desc; do [ -r "${f}" ] && echo "${f%/version.desc}"; done`
|
|
|
|
versions=`sort_versions ${versions}`
|
|
|
|
|
|
|
|
set_iter version ${versions}
|
|
|
|
info[all_versions]=${versions}
|
|
|
|
|
2019-02-13 15:51:00 -08:00
|
|
|
check_relevant_pattern()
|
|
|
|
{
|
|
|
|
if [ "x${seen_selectors[${info[ver_sel]}]+set}" = "xset" ]; then
|
|
|
|
error "${info[pkg_name]}: version ${info[ver]} conflicts with version ${seen_selectors[${info[ver_sel]}]} (${info[ver_sel]} selector)"
|
|
|
|
else
|
|
|
|
seen_selectors[${info[ver_sel]}]=${info[ver]}
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
do_foreach version check_relevant_pattern
|
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
# If a fork does not define any versions at all ("rolling release"), do not
|
|
|
|
# consider it obsolete/experimental unless it is so marked in the fork's
|
|
|
|
# description.
|
|
|
|
if [ -n "${versions}" ]; then
|
|
|
|
only_obsolete=yes
|
|
|
|
only_experimental=yes
|
2019-02-13 15:51:00 -08:00
|
|
|
|
|
|
|
check_obsolete_experimental()
|
|
|
|
{
|
|
|
|
[ -z "${info[obsolete]}" ] && only_obsolete=
|
|
|
|
[ -z "${info[experimental]}" ] && only_experimental=
|
|
|
|
}
|
2017-07-04 19:19:42 -07:00
|
|
|
do_foreach version check_obsolete_experimental
|
|
|
|
info[only_obsolete]=${only_obsolete}
|
|
|
|
info[only_experimental]=${only_experimental}
|
|
|
|
else
|
|
|
|
info[only_obsolete]=${info[obsolete]}
|
|
|
|
info[only_experimental]=${info[experimental]}
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
enter_version()
|
|
|
|
{
|
|
|
|
local version="${1}"
|
|
|
|
|
|
|
|
eval `read_version_desc ${info[fork]} ${version}`
|
|
|
|
info[ver]=${version}
|
2017-08-24 14:58:14 -07:00
|
|
|
info[ver_sel]=`relevantize ${version} ${info[relevantpattern]}`
|
2017-07-04 19:19:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
enter_milestone()
|
|
|
|
{
|
|
|
|
local ms="${1}"
|
|
|
|
|
|
|
|
info[ms]=${ms}
|
|
|
|
if [ -n "${info[ver]}" ]; then
|
2019-05-27 13:17:33 +00:00
|
|
|
if [ -n "${info[version_number]}" ]; then
|
|
|
|
info[version_cmp_milestone]=`cmp_versions ${info[version_number]} ${info[ms]}`
|
|
|
|
else
|
|
|
|
info[version_cmp_milestone]=`cmp_versions ${info[ver]} ${info[ms]}`
|
|
|
|
fi
|
2017-07-04 19:19:42 -07:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
gen_packages()
|
|
|
|
{
|
2017-07-06 22:18:43 -07:00
|
|
|
local -A pkg_forks pkg_milestones pkg_nforks pkg_relevantpattern
|
2017-07-04 19:19:42 -07:00
|
|
|
local -a pkg_masters pkg_all pkg_preferred
|
|
|
|
|
|
|
|
pkg_all=( `cd packages && \
|
|
|
|
ls */package.desc 2>/dev/null | \
|
|
|
|
while read f; do [ -r "${f}" ] && echo "${f%/package.desc}"; done | \
|
|
|
|
xargs echo` )
|
|
|
|
|
|
|
|
debug "Packages: ${pkg_all[@]}"
|
|
|
|
|
|
|
|
# We need to group forks of the same package into the same
|
|
|
|
# config file. Discover such relationships and only iterate
|
|
|
|
# over "master" packages at the top.
|
|
|
|
for p in "${pkg_all[@]}"; do
|
|
|
|
find_forks "${p}"
|
|
|
|
done
|
|
|
|
msg "Master packages: ${pkg_masters[@]}"
|
|
|
|
|
|
|
|
# Now for each master, create its kconfig file with version
|
2017-08-23 16:05:40 -07:00
|
|
|
# definitions. As a byproduct, generate a list of all package
|
|
|
|
# versions for maintenance purposes.
|
|
|
|
exec 3>"maintainer/package-versions"
|
2017-07-04 19:19:42 -07:00
|
|
|
for p in "${pkg_masters[@]}"; do
|
|
|
|
msg "Generating '${config_versions_dir}/${p}.in'"
|
|
|
|
exec >"${config_versions_dir}/${p}.in"
|
|
|
|
# Base definitions for the whole config file
|
|
|
|
info=( \
|
|
|
|
[master]=${p} \
|
|
|
|
[nforks]=${pkg_nforks[${p}]} \
|
2017-07-06 22:18:43 -07:00
|
|
|
[relevantpattern]=${pkg_relevantpattern[${p}]} \
|
2017-07-04 19:19:42 -07:00
|
|
|
)
|
|
|
|
set_iter fork ${pkg_forks[${p}]}
|
|
|
|
set_iter milestone ${pkg_milestones[${p}]}
|
|
|
|
|
|
|
|
run_template "maintainer/kconfig-versions.template"
|
2017-08-23 16:05:40 -07:00
|
|
|
run_template "maintainer/package-versions.template" >&3
|
2017-07-04 19:19:42 -07:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
msg "*** Generating package version descriptions"
|
|
|
|
config_versions_dir=config/versions
|
|
|
|
rm -rf "${config_versions_dir}"
|
|
|
|
mkdir -p "${config_versions_dir}"
|
|
|
|
gen_packages
|
|
|
|
|
|
|
|
get_components()
|
|
|
|
{
|
|
|
|
local dir="${1}"
|
|
|
|
local f b
|
|
|
|
|
|
|
|
for f in ${dir}/*.in; do
|
|
|
|
b=${f#${dir}/}
|
|
|
|
echo ${b%.in}
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
enter_choice()
|
|
|
|
{
|
|
|
|
local choice="${1}"
|
2017-11-29 00:14:04 -08:00
|
|
|
local input="config/${info[dir]}/${choice}.in"
|
|
|
|
local l ln
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
info[choice]="${choice}"
|
2017-11-29 00:14:04 -08:00
|
|
|
info[pkg]="${choice}"
|
2017-07-04 19:19:42 -07:00
|
|
|
|
|
|
|
# Not local, we need these arrays be set in enter_dependency/enter_help
|
|
|
|
deplines=( )
|
|
|
|
helplines=( )
|
2017-11-29 00:14:04 -08:00
|
|
|
ln=0
|
2017-07-04 19:19:42 -07:00
|
|
|
while read l; do
|
2017-11-29 00:14:04 -08:00
|
|
|
ln=$[ln+1]
|
2017-07-04 19:19:42 -07:00
|
|
|
case "${l}" in
|
|
|
|
"## help "*)
|
2017-11-29 00:14:04 -08:00
|
|
|
helplines+=( "${l#\#\# help }" )
|
2017-07-04 19:19:42 -07:00
|
|
|
;;
|
2017-11-29 00:14:04 -08:00
|
|
|
"## depends "*|"## select "*|"## default "*)
|
|
|
|
deplines+=( "${l#\#\# }" )
|
|
|
|
;;
|
|
|
|
"## no-package")
|
|
|
|
info[pkg]=
|
|
|
|
;;
|
|
|
|
"## package "*)
|
|
|
|
info[pkg]=${l#\#\# package }
|
|
|
|
;;
|
|
|
|
"##"|"## help")
|
|
|
|
# accept empty, for formatting
|
|
|
|
;;
|
|
|
|
"##"*)
|
|
|
|
error "${input}:${ln}: unrecognized command"
|
2017-07-04 19:19:42 -07:00
|
|
|
;;
|
|
|
|
esac
|
2017-11-29 00:14:04 -08:00
|
|
|
done < "${input}"
|
2017-07-04 19:19:42 -07:00
|
|
|
set_iter dependency "${!deplines[@]}"
|
|
|
|
set_iter help "${!helplines[@]}"
|
|
|
|
}
|
|
|
|
|
|
|
|
enter_dependency()
|
|
|
|
{
|
|
|
|
info[depline]="${deplines[${1}]}"
|
|
|
|
}
|
|
|
|
|
|
|
|
enter_help()
|
|
|
|
{
|
|
|
|
info[helpline]="${helplines[${1}]}"
|
|
|
|
}
|
|
|
|
|
|
|
|
gen_selection()
|
|
|
|
{
|
|
|
|
local type="${1}"
|
|
|
|
local dir="${2}"
|
|
|
|
local label="${3}"
|
|
|
|
|
2017-11-19 00:10:53 -08:00
|
|
|
msg "Generating ${dir}.in (${type})"
|
2017-07-04 19:19:42 -07:00
|
|
|
exec >"${config_gen_dir}/${dir}.in"
|
|
|
|
info=( \
|
|
|
|
[dir]=${dir} \
|
|
|
|
[label]="${label}" \
|
|
|
|
)
|
|
|
|
set_iter choice `get_components config/${dir}`
|
|
|
|
run_template "maintainer/kconfig-${type}.template"
|
|
|
|
}
|
|
|
|
|
|
|
|
msg "*** Generating menu/choice selections"
|
|
|
|
config_gen_dir=config/gen
|
|
|
|
rm -rf "${config_gen_dir}"
|
|
|
|
mkdir -p "${config_gen_dir}"
|
|
|
|
|
|
|
|
gen_selection choice arch "Target Architecture"
|
|
|
|
gen_selection choice kernel "Target OS"
|
|
|
|
gen_selection choice cc "Compiler"
|
|
|
|
gen_selection choice binutils "Binutils"
|
|
|
|
gen_selection choice libc "C library"
|
2023-09-19 12:42:19 +02:00
|
|
|
gen_selection menu linker "Linkers"
|
2017-07-04 19:19:42 -07:00
|
|
|
gen_selection menu debug "Debug facilities"
|
|
|
|
gen_selection menu comp_tools "Companion tools"
|
2017-11-29 00:14:04 -08:00
|
|
|
gen_selection menu comp_libs "Companion libraries"
|
2017-07-04 19:19:42 -07:00
|
|
|
|
2018-02-23 19:05:41 -08:00
|
|
|
msg "*** Gathering the list of data files to install"
|
|
|
|
{
|
2018-03-17 17:41:36 -07:00
|
|
|
declare -A seen_files
|
2018-02-23 19:05:41 -08:00
|
|
|
echo -n "verbatim_data ="
|
2023-07-29 14:58:39 +03:00
|
|
|
find -L COPYING config contrib licenses.d packages samples scripts -type f | LANG=C sort | while read f; do
|
2018-03-17 22:19:45 -07:00
|
|
|
# Implement some kind of .installignore for these files?
|
2018-03-17 14:44:05 -07:00
|
|
|
case "${f}" in
|
2019-01-27 22:55:41 -08:00
|
|
|
# Avoid temp files
|
2019-02-09 15:55:16 -08:00
|
|
|
*.sw[po])
|
2019-01-27 22:55:41 -08:00
|
|
|
continue
|
|
|
|
;;
|
2018-11-26 23:50:47 -08:00
|
|
|
# And, some files automake insists we must have
|
2018-03-17 22:19:45 -07:00
|
|
|
scripts/compile | scripts/missing | scripts/depcomp | scripts/ltmain.sh | scripts/install-sh)
|
2018-03-17 14:44:05 -07:00
|
|
|
continue
|
|
|
|
;;
|
2019-10-10 18:32:19 +02:00
|
|
|
#
|
2018-03-17 14:44:05 -07:00
|
|
|
# will produce. FIXME: create this file at the time of 'ct-ng build'.
|
2019-04-04 16:43:51 -07:00
|
|
|
config/configure.in.in | config/configure.in)
|
|
|
|
continue
|
2018-03-17 14:44:05 -07:00
|
|
|
;;
|
|
|
|
esac
|
2018-03-17 17:41:36 -07:00
|
|
|
# Checks & substitutions above may result in duplicate files
|
|
|
|
if [ -n "${seen_files[${f}]}" ]; then
|
|
|
|
continue
|
|
|
|
fi
|
2018-02-23 19:05:41 -08:00
|
|
|
echo " \\"
|
2022-06-26 18:01:27 -07:00
|
|
|
echo -en "\t${f}"
|
2018-03-17 17:41:36 -07:00
|
|
|
seen_files[${f}]=y
|
2018-02-23 19:05:41 -08:00
|
|
|
done
|
|
|
|
} > verbatim-data.mk
|
|
|
|
|
2018-01-06 18:31:15 -08:00
|
|
|
msg "*** Running autoreconf"
|
2019-10-10 18:32:19 +02:00
|
|
|
autoreconf -Wall --force -I m4
|
2017-05-07 15:12:08 -07:00
|
|
|
|
2017-07-04 19:19:42 -07:00
|
|
|
msg "*** Done!"
|