# cfgtest.sh: sofort's config test framework,
# for use from within a project's custom cfgdefs.sh.
# in the common scenario, host-specific tests are preceded
# by a single invocation of cfgtest_host_section, whereas
# native (build) system tests are preceded by the invocation
# of cfgtest_native_section.
# cfgdefs fraework variables:
# mb_cfgtest_cc: the compiler used for the current test
# mb_cfgtest_cflags: the compiler flags used for the current test
# mb_cfgtest_cfgtype: the type of the current test (host/native)
# mb_cfgtest_makevar: the make variable affected by the current test
# mb_cfgtest_headers: headers for ad-hoc inclusion with the current test
cfgtest_newline()
{
printf '\n' >> $mb_pwd/cfgdefs.mk
}
cfgtest_comment()
{
mb_internal_str='#'
for mb_internal_arg ; do
mb_internal_str="$mb_internal_str $mb_internal_arg"
done
printf '%s\n' "$mb_internal_str" >> $mb_pwd/cfgdefs.mk
}
cfgtest_host_section()
{
mb_cfgtest_cc="$ccenv_host_cc"
mb_cfgtest_cfgtype='host'
mb_cfgtest_cflags=$(make -s -f "$mb_pwd/Makefile.tmp" .cflags-host)
cfgtest_comment 'host-specific tests'
}
cfgtest_native_section()
{
mb_cfgtest_cc="$mb_native_cc"
mb_cfgtest_cfgtype='native'
mb_cfgtest_cflags=$(make -s -f "$mb_pwd/Makefile.tmp" .cflags-native)
cfgtest_comment 'native system tests'
}
cfgtest_prolog()
{
cfgtest_line_dots='.......................'
cfgtest_line_dots="${cfgtest_line_dots}${cfgtest_line_dots}"
cfgtest_tool_desc=" == trying ${mb_cfgtest_cfgtype} ${1}: ${2}"
cfgtest_tool_dlen="${#cfgtest_line_dots}"
printf '\n%s\n' '________________________' >&3
printf "cfgtest: probing for ${mb_cfgtest_cfgtype} ${1}: ${2}\n\n" >&3
printf "%${cfgtest_tool_dlen}.${cfgtest_tool_dlen}s" \
"${cfgtest_tool_desc} ${mb_line_dots}"
}
cfgtest_epilog()
{
cfgtest_line_dots='.......................'
cfgtest_tool_dlen="$((${#cfgtest_line_dots} - ${#1}))"
printf "%${cfgtest_tool_dlen}.${cfgtest_tool_dlen}s %s.\n" \
"${cfgtest_line_dots}" "${1}"
if [ "${1}" = '-----' ]; then
printf '\n\ncfgtest: header is missing or cannot be found.\n' >&3
printf '%s\n' '------------------------' >&3
return 1
elif [ "${1}" = '(error)' ]; then
printf '\n\ncfgtest: interface is missing or cannot be used.\n' >&3
printf '%s\n' '------------------------' >&3
return 1
fi
}
cfgtest_makevar_append()
{
mb_internal_str='+='
for mb_internal_arg ; do
if ! [ -z "$mb_internal_arg" ]; then
mb_internal_str="$mb_internal_str $mb_internal_arg"
fi
done
printf '%-24s%s\n' "$mb_cfgtest_makevar" "$mb_internal_str" \
>> $mb_pwd/cfgdefs.mk
unset cfgtest_internal_unit_test
}
cfgtest_cflags_append()
{
if [ $mb_cfgtest_cfgtype = 'host' ]; then
mb_internal_makevar='CFLAGS_CONFIG'
else
mb_internal_makevar='NATIVE_CFLAGS'
fi
mb_cfgtest_makevar_saved=$mb_cfgtest_makevar
mb_cfgtest_makevar=$mb_internal_makevar
cfgtest_makevar_append "$@"
mb_cfgtest_makevar=$mb_cfgtest_makevar_saved
}
cfgtest_ldflags_append()
{
if [ $mb_cfgtest_cfgtype = 'host' ]; then
mb_internal_makevar='LDFLAGS_CONFIG'
else
mb_internal_makevar='NATIVE_LDFLAGS'
fi
mb_cfgtest_makevar_saved=$mb_cfgtest_makevar
mb_cfgtest_makevar=$mb_internal_makevar
cfgtest_makevar_append "$@"
mb_cfgtest_makevar=$mb_cfgtest_makevar_saved
}
cfgtest_header_presence()
{
cfgtest_prolog 'header' "${1}"
printf '%s -E -xc - \\\n' "$mb_cfgtest_cc" >&3
for cfgtest_cflag in $mb_cfgtest_cflags; do
printf '\t%s \\\n' "$cfgtest_cflag" >&3
done
printf '\t%s\n\n' '--include='"${1}" >&3
cfgtest_cmd=$(printf '%s -E -xc - %s %s' \
"$mb_cfgtest_cc" "$mb_cfgtest_cflags" \
'--include='"${1}")
$(printf '%s' "$cfgtest_cmd") \
< /dev/null \
> /dev/null 2>&3 \
|| cfgtest_epilog '-----' \
|| return
mb_internal_str=$(printf '%s%s' '-DHAVE_' "${1}" \
| sed -e 's/\./_/g' -e 's@/@_@g' \
| tr "[:lower:]" "[:upper:]")
if [ -z ${cfgtest_internal_unit_test:-} ]; then
cfgtest_cflags_append "$mb_internal_str"
else
cfgtest_makevar_append "$mb_internal_str"
fi
printf 'cfgtest: %s header <%s> was found and may be included.\n' \
"$mb_cfgtest_cfgtype" "${1}" >&3
printf '%s\n' '------------------------' >&3
cfgtest_epilog "${1}"
}
cfgtest_header_absence()
{
cfgtest_prolog 'header absence' "${1}"
printf '%s -E -xc - \\\n' "$mb_cfgtest_cc" >&3
for cfgtest_cflag in $mb_cfgtest_cflags; do
printf '\t%s \\\n' "$cfgtest_cflag" >&3
done
printf '\t%s\n\n' '--include='"${1}" >&3
cfgtest_cmd=$(printf '%s -E -xc - %s %s' \
"$mb_cfgtest_cc" "$mb_cfgtest_cflags" \
'--include='"${1}")
$(printf '%s' "$cfgtest_cmd") \
< /dev/null \
> /dev/null 2>&3 \
&& cfgtest_epilog "${1}" \
&& return
mb_internal_str=$(printf '%s%s' '-DHAVE_NO_' "$@" \
| sed -e 's/\./_/g' -e 's@/@_@g' \
| tr "[:lower:]" "[:upper:]")
if [ -z ${cfgtest_internal_unit_test:-} ]; then
cfgtest_cflags_append "$mb_internal_str"
else
cfgtest_makevar_append "$mb_internal_str"
fi
printf 'cfgtest: %s header <%s> may not be included.\n' \
"$mb_cfgtest_cfgtype" "${1}" >&3
printf '%s\n' '------------------------' >&3
cfgtest_epilog '-----'
}
cfgtest_interface_presence()
{
cfgtest_prolog 'interface' "${1}"
mb_internal_cflags=
for mb_header in $mb_cfgtest_headers; do
mb_internal_cflags="$mb_internal_cflags --include=$mb_header"
done
cfgtest_code_snippet=$(printf 'void * addr = &%s;' "${1}")
printf 'printf %s "%s" \\\n' "'%s'" "$cfgtest_code_snippet" >&3
printf '| %s -S -xc - -o -' "$mb_cfgtest_cc" >&3
for cfgtest_cflag in $mb_cfgtest_cflags; do
printf ' \\\n\t%s' "$cfgtest_cflag" >&3
done
for cfgtest_cflag in $mb_internal_cflags; do
printf ' \\\n\t%s' "$cfgtest_cflag" >&3
done
printf '\n\n' >&3
cfgtest_cmd=$(printf '%s -S -xc - -o - %s %s' \
"$mb_cfgtest_cc" "$mb_cfgtest_cflags" \
"$mb_internal_cflags")
printf '%s' "$cfgtest_code_snippet" \
| $(printf '%s' "$cfgtest_cmd") \
> /dev/null 2>&3 \
|| cfgtest_epilog '(error)' \
|| return
mb_internal_str=$(printf '%s%s' '-DHAVE_' "$@" \
| sed -e 's/\./_/g' \
| tr "[:lower:]" "[:upper:]")
if [ -z ${cfgtest_internal_unit_test:-} ]; then
cfgtest_cflags_append "$mb_internal_str"
else
cfgtest_makevar_append "$mb_internal_str"
fi
printf 'cfgtest: %s interface `%s'"'"' is available.\n' \
"$mb_cfgtest_cfgtype" "${1}" >&3
printf '%s\n' '------------------------' >&3
cfgtest_epilog "${1}"
return 0
}
cfgtest_decl_presence()
{
mb_internal_cflags=''
for mb_header in $mb_cfgtest_headers; do
mb_internal_cflags="$mb_internal_cflags --include=$mb_header"
done
printf 'void * any = (void *)%s;' "$@" \
| $mb_cfgtest_cc -S -xc - -o - \
$mb_cfgtest_cflags \
$mb_internal_cflags \
> /dev/null 2>/dev/null \
|| return 1
# does the argument solely consist of the macro or enum member name?
mb_internal_str=$(printf '%s' "$@" | tr -d '[a-z][A-Z][0-9][_]')
if [ -n "$mb_internal_str" ]; then
return 0
fi
mb_internal_str=$(printf '%s%s' '-DHAVE_DECL_' "$@" \
| sed -e 's/\./_/g' \
| tr "[:lower:]" "[:upper:]")
if [ -z ${cfgtest_internal_unit_test:-} ]; then
cfgtest_cflags_append "$mb_internal_str"
else
cfgtest_makevar_append "$mb_internal_str"
fi
return 0
}
cfgtest_type_size()
{
mb_internal_cflags=''
mb_internal_size=''
mb_internal_test='char x[(sizeof(%s) == %s) ? 1 : -1];'
for mb_header in $mb_cfgtest_headers; do
mb_internal_cflags="$mb_internal_cflags --include=$mb_header"
done
for mb_internal_guess in 8 4 2 1 16 32 64 128; do
if [ -z $mb_internal_size ]; then
mb_internal_type="$@"
mb_internal_str=$(printf "$mb_internal_test" \
"$mb_internal_type" \
"$mb_internal_guess")
printf '%s' "$mb_internal_str" \
| $mb_cfgtest_cc -S -xc - -o - \
$mb_cfgtest_cflags \
$mb_internal_cflags \
> /dev/null 2>/dev/null \
&& mb_internal_size=$mb_internal_guess
fi
done
# unrecognized type, or type size not within range
if [ -z $mb_internal_size ]; then
return 1
fi
# -DSIZEOF_TYPE=SIZE
mb_internal_str=$(printf '%s%s=%s' '-DSIZEOF_' \
"$mb_internal_type" \
"$mb_internal_size" \
| sed -e 's/\ /_/g' -e 's/*/P/g' \
| tr "[:lower:]" "[:upper:]")
if [ -z ${cfgtest_internal_unit_test:-} ]; then
cfgtest_cflags_append "$mb_internal_str"
else
cfgtest_makevar_append "$mb_internal_str"
fi
return 0
}
cfgtest_code_snippet()
{
mb_internal_cflags=''
mb_internal_test="$@"
for mb_header in $mb_cfgtest_headers; do
mb_internal_cflags="$mb_internal_cflags --include=$mb_header"
done
printf '%s' "$mb_internal_test" \
| $mb_cfgtest_cc -S -xc - -o - \
$mb_cfgtest_cflags \
$mb_internal_cflags \
> /dev/null 2>/dev/null \
|| return 1
return 0
}
cfgtest_code_snippet_asm()
{
mb_internal_cflags=''
mb_internal_test="$@"
cfgtest_tmp=$(mktemp ./tmp_XXXXXXXXXXXXXXXX)
cfgtest_ret=1
for mb_header in $mb_cfgtest_headers; do
mb_internal_cflags="$mb_internal_cflags --include=$mb_header"
done
printf '%s' "$mb_internal_test" \
| $mb_cfgtest_cc -c -xc - \
-o $cfgtest_tmp \
$mb_cfgtest_cflags \
$mb_internal_cflags \
> /dev/null 2>/dev/null \
&& cfgtest_ret=0
rm -f "$cfgtest_tmp"
unset cfgtest_tmp
return $cfgtest_ret
}
cfgtest_library_presence()
{
printf 'int main(void){return 0;}' \
| $mb_cfgtest_cc -o a.out -xc - \
$mb_cfgtest_cflags \
$@ \
> /dev/null 2>/dev/null \
|| return 1
rm -f a.out
return 0
}
cfgtest_unit_header_presence()
{
cfgtest_internal_unit_test='unit_test'
cfgtest_header_presence "$@" || return 1
return 0
}
cfgtest_unit_header_absence()
{
cfgtest_internal_unit_test='unit_test'
cfgtest_header_absence "$@" || return 1
return 0
}
cfgtest_unit_interface_presence()
{
cfgtest_internal_unit_test='unit_test'
cfgtest_interface_presence "$@" || return 1
return 0
}
cfgtest_unit_decl_presence()
{
cfgtest_internal_unit_test='unit_test'
cfgtest_decl_presence "$@" || return 1
return 0
}
cfgtest_unit_type_size()
{
cfgtest_internal_unit_test='unit_test'
cfgtest_type_size "$@" || return 1
return 0
}