xref: /freebsd/contrib/bc/scripts/functions.sh (revision 8cc087a1eee9ec1ca9f7ac1e63ad51bdb5a682eb)
1#! /bin/sh
2#
3# SPDX-License-Identifier: BSD-2-Clause
4#
5# Copyright (c) 2018-2021 Gavin D. Howard and contributors.
6#
7# Redistribution and use in source and binary forms, with or without
8# modification, are permitted provided that the following conditions are met:
9#
10# * Redistributions of source code must retain the above copyright notice, this
11#   list of conditions and the following disclaimer.
12#
13# * Redistributions in binary form must reproduce the above copyright notice,
14#   this list of conditions and the following disclaimer in the documentation
15#   and/or other materials provided with the distribution.
16#
17# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27# POSSIBILITY OF SUCH DAMAGE.
28#
29
30# This script is NOT meant to be run! It is meant to be sourced by other
31# scripts.
32
33# Reads and follows a link until it finds a real file. This is here because the
34# readlink utility is not part of the POSIX standard. Sigh...
35# @param f  The link to find the original file for.
36readlink() {
37
38	_readlink_f="$1"
39	shift
40
41	_readlink_arrow="-> "
42	_readlink_d=$(dirname "$_readlink_f")
43
44	_readlink_lsout=""
45	_readlink_link=""
46
47	_readlink_lsout=$(ls -dl "$_readlink_f")
48	_readlink_link=$(printf '%s' "${_readlink_lsout#*$_readlink_arrow}")
49
50	while [ -z "${_readlink_lsout##*$_readlink_arrow*}" ]; do
51		_readlink_f="$_readlink_d/$_readlink_link"
52		_readlink_d=$(dirname "$_readlink_f")
53		_readlink_lsout=$(ls -dl "$_readlink_f")
54		_readlink_link=$(printf '%s' "${_readlink_lsout#*$_readlink_arrow}")
55	done
56
57	printf '%s' "${_readlink_f##*$_readlink_d/}"
58}
59
60# Quick function for exiting with an error.
61# @param 1  A message to print.
62# @param 2  The exit code to use.
63err_exit() {
64
65	if [ "$#" -ne 2 ]; then
66		printf 'Invalid number of args to err_exit\n'
67		exit 1
68	fi
69
70	printf '%s\n' "$1"
71	exit "$2"
72}
73
74# Check the return code on a test and exit with a fail if it's non-zero.
75# @param d     The calculator under test.
76# @param err   The return code.
77# @param name  The name of the test.
78checktest_retcode() {
79
80	_checktest_retcode_d="$1"
81	shift
82
83	_checktest_retcode_err="$1"
84	shift
85
86	_checktest_retcode_name="$1"
87	shift
88
89	if [ "$_checktest_retcode_err" -ne 0 ]; then
90		printf 'FAIL!!!\n'
91		err_exit "$_checktest_retcode_d failed test '$_checktest_retcode_name' with error code $_checktest_retcode_err" 1
92	fi
93}
94
95# Check the result of a test. First, it checks the error code using
96# checktest_retcode(). Then it checks the output against the expected output
97# and fails if it doesn't match.
98# @param d             The calculator under test.
99# @param err           The error code.
100# @param name          The name of the test.
101# @param test_path     The path to the test.
102# @param results_name  The path to the file with the expected result.
103checktest() {
104
105	_checktest_d="$1"
106	shift
107
108	_checktest_err="$1"
109	shift
110
111	_checktest_name="$1"
112	shift
113
114	_checktest_test_path="$1"
115	shift
116
117	_checktest_results_name="$1"
118	shift
119
120	checktest_retcode "$_checktest_d" "$_checktest_err" "$_checktest_name"
121
122	_checktest_diff=$(diff "$_checktest_test_path" "$_checktest_results_name")
123
124	_checktest_err="$?"
125
126	if [ "$_checktest_err" -ne 0 ]; then
127		printf 'FAIL!!!\n'
128		printf '%s\n' "$_checktest_diff"
129		err_exit "$_checktest_d failed test $_checktest_name" 1
130	fi
131}
132
133# Die. With a message.
134# @param d     The calculator under test.
135# @param msg   The message to print.
136# @param name  The name of the test.
137# @param err   The return code from the test.
138die() {
139
140	_die_d="$1"
141	shift
142
143	_die_msg="$1"
144	shift
145
146	_die_name="$1"
147	shift
148
149	_die_err="$1"
150	shift
151
152	_die_str=$(printf '\n%s %s on test:\n\n    %s\n' "$_die_d" "$_die_msg" "$_die_name")
153
154	err_exit "$_die_str" "$_die_err"
155}
156
157# Check that a test did not crash and die if it did.
158# @param d      The calculator under test.
159# @param error  The error code.
160# @param name   The name of the test.
161checkcrash() {
162
163	_checkcrash_d="$1"
164	shift
165
166	_checkcrash_error="$1"
167	shift
168
169	_checkcrash_name="$1"
170	shift
171
172
173	if [ "$_checkcrash_error" -gt 127 ]; then
174		die "$_checkcrash_d" "crashed ($_checkcrash_error)" \
175			"$_checkcrash_name" "$_checkcrash_error"
176	fi
177}
178
179# Check that a test had an error or crash.
180# @param d        The calculator under test.
181# @param error    The error code.
182# @param name     The name of the test.
183# @param out      The file that the test results were output to.
184# @param exebase  The name of the executable.
185checkerrtest()
186{
187	_checkerrtest_d="$1"
188	shift
189
190	_checkerrtest_error="$1"
191	shift
192
193	_checkerrtest_name="$1"
194	shift
195
196	_checkerrtest_out="$1"
197	shift
198
199	_checkerrtest_exebase="$1"
200	shift
201
202	checkcrash "$_checkerrtest_d" "$_checkerrtest_error" "$_checkerrtest_name"
203
204	if [ "$_checkerrtest_error" -eq 0 ]; then
205		die "$_checkerrtest_d" "returned no error" "$_checkerrtest_name" 127
206	fi
207
208	# This is to check for memory errors with Valgrind, which is told to return
209	# 100 on memory errors.
210	if [ "$_checkerrtest_error" -eq 100 ]; then
211
212		_checkerrtest_output=$(cat "$_checkerrtest_out")
213		_checkerrtest_fatal_error="Fatal error"
214
215		if [ "${_checkerrtest_output##*$_checkerrtest_fatal_error*}" ]; then
216			printf "%s\n" "$_checkerrtest_output"
217			die "$_checkerrtest_d" "had memory errors on a non-fatal error" \
218				"$_checkerrtest_name" "$_checkerrtest_error"
219		fi
220	fi
221
222	if [ ! -s "$_checkerrtest_out" ]; then
223		die "$_checkerrtest_d" "produced no error message" "$_checkerrtest_name" "$_checkerrtest_error"
224	fi
225
226	# To display error messages, uncomment this line. This is useful when
227	# debugging.
228	#cat "$_checkerrtest_out"
229}
230
231# Replace a substring in a string with another. This function is the *real*
232# workhorse behind configure.sh's generation of a Makefile.
233#
234# This function uses a sed call that uses exclamation points `!` as delimiters.
235# As a result, needle can never contain an exclamation point. Oh well.
236#
237# @param str          The string that will have any of the needle replaced by
238#                     replacement.
239# @param needle       The needle to replace in str with replacement.
240# @param replacement  The replacement for needle in str.
241substring_replace() {
242
243	_substring_replace_str="$1"
244	shift
245
246	_substring_replace_needle="$1"
247	shift
248
249	_substring_replace_replacement="$1"
250	shift
251
252	_substring_replace_result=$(printf '%s\n' "$_substring_replace_str" | \
253		sed -e "s!$_substring_replace_needle!$_substring_replace_replacement!g")
254
255	printf '%s' "$_substring_replace_result"
256}
257
258# Generates an NLS path based on the locale and executable name.
259#
260# This is a monstrosity for a reason.
261#
262# @param nlspath   The $NLSPATH
263# @param locale    The locale.
264# @param execname  The name of the executable.
265gen_nlspath() {
266
267	_gen_nlspath_nlspath="$1"
268	shift
269
270	_gen_nlspath_locale="$1"
271	shift
272
273	_gen_nlspath_execname="$1"
274	shift
275
276	# Split the locale into its modifier and other parts.
277	_gen_nlspath_char="@"
278	_gen_nlspath_modifier="${_gen_nlspath_locale#*$_gen_nlspath_char}"
279	_gen_nlspath_tmplocale="${_gen_nlspath_locale%%$_gen_nlspath_char*}"
280
281	# Split the locale into charset and other parts.
282	_gen_nlspath_char="."
283	_gen_nlspath_charset="${_gen_nlspath_tmplocale#*$_gen_nlspath_char}"
284	_gen_nlspath_tmplocale="${_gen_nlspath_tmplocale%%$_gen_nlspath_char*}"
285
286	# Check for an empty charset.
287	if [ "$_gen_nlspath_charset" = "$_gen_nlspath_tmplocale" ]; then
288		_gen_nlspath_charset=""
289	fi
290
291	# Split the locale into territory and language.
292	_gen_nlspath_char="_"
293	_gen_nlspath_territory="${_gen_nlspath_tmplocale#*$_gen_nlspath_char}"
294	_gen_nlspath_language="${_gen_nlspath_tmplocale%%$_gen_nlspath_char*}"
295
296	# Check for empty territory and language.
297	if [ "$_gen_nlspath_territory" = "$_gen_nlspath_tmplocale" ]; then
298		_gen_nlspath_territory=""
299	fi
300
301	if [ "$_gen_nlspath_language" = "$_gen_nlspath_tmplocale" ]; then
302		_gen_nlspath_language=""
303	fi
304
305	# Prepare to replace the format specifiers. This is done by wrapping the in
306	# pipe characters. It just makes it easier to split them later.
307	_gen_nlspath_needles="%%:%L:%N:%l:%t:%c"
308
309	_gen_nlspath_needles=$(printf '%s' "$_gen_nlspath_needles" | tr ':' '\n')
310
311	for _gen_nlspath_i in $_gen_nlspath_needles; do
312		_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "$_gen_nlspath_i" "|$_gen_nlspath_i|")
313	done
314
315	# Replace all the format specifiers.
316	_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "%%" "%")
317	_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "%L" "$_gen_nlspath_locale")
318	_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "%N" "$_gen_nlspath_execname")
319	_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "%l" "$_gen_nlspath_language")
320	_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "%t" "$_gen_nlspath_territory")
321	_gen_nlspath_nlspath=$(substring_replace "$_gen_nlspath_nlspath" "%c" "$_gen_nlspath_charset")
322
323	# Get rid of pipe characters.
324	_gen_nlspath_nlspath=$(printf '%s' "$_gen_nlspath_nlspath" | tr -d '|')
325
326	# Return the result.
327	printf '%s' "$_gen_nlspath_nlspath"
328}
329
330ALL=0
331NOSKIP=1
332SKIP=2
333
334# Filters text out of a file according to the build type.
335# @param in    File to filter.
336# @param out   File to write the filtered output to.
337# @param type  Build type.
338filter_text() {
339
340	_filter_text_in="$1"
341	shift
342
343	_filter_text_out="$1"
344	shift
345
346	_filter_text_buildtype="$1"
347	shift
348
349	# Set up some local variables.
350	_filter_text_status="$ALL"
351	_filter_text_last_line=""
352
353	# We need to set IFS, so we store it here for restoration later.
354	_filter_text_ifs="$IFS"
355
356	# Remove the file- that will be generated.
357	rm -rf "$_filter_text_out"
358
359	# Here is the magic. This loop reads the template line-by-line, and based on
360	# _filter_text_status, either prints it to the markdown manual or not.
361	#
362	# Here is how the template is set up: it is a normal markdown file except
363	# that there are sections surrounded tags that look like this:
364	#
365	# {{ <build_type_list> }}
366	# ...
367	# {{ end }}
368	#
369	# Those tags mean that whatever build types are found in the
370	# <build_type_list> get to keep that section. Otherwise, skip.
371	#
372	# Obviously, the tag itself and its end are not printed to the markdown
373	# manual.
374	while IFS= read -r _filter_text_line; do
375
376		# If we have found an end, reset the status.
377		if [ "$_filter_text_line" = "{{ end }}" ]; then
378
379			# Some error checking. This helps when editing the templates.
380			if [ "$_filter_text_status" -eq "$ALL" ]; then
381				err_exit "{{ end }} tag without corresponding start tag" 2
382			fi
383
384			_filter_text_status="$ALL"
385
386		# We have found a tag that allows our build type to use it.
387		elif [ "${_filter_text_line#\{\{* $_filter_text_buildtype *\}\}}" != "$_filter_text_line" ]; then
388
389			# More error checking. We don't want tags nested.
390			if [ "$_filter_text_status" -ne "$ALL" ]; then
391				err_exit "start tag nested in start tag" 3
392			fi
393
394			_filter_text_status="$NOSKIP"
395
396		# We have found a tag that is *not* allowed for our build type.
397		elif [ "${_filter_text_line#\{\{*\}\}}" != "$_filter_text_line" ]; then
398
399			if [ "$_filter_text_status" -ne "$ALL" ]; then
400				err_exit "start tag nested in start tag" 3
401			fi
402
403			_filter_text_status="$SKIP"
404
405		# This is for normal lines. If we are not skipping, print.
406		else
407			if [ "$_filter_text_status" -ne "$SKIP" ]; then
408				if [ "$_filter_text_line" != "$_filter_text_last_line" ]; then
409					printf '%s\n' "$_filter_text_line" >> "$_filter_text_out"
410				fi
411				_filter_text_last_line="$_filter_text_line"
412			fi
413		fi
414
415	done < "$_filter_text_in"
416
417	# Reset IFS.
418	IFS="$_filter_text_ifs"
419}
420