Training courses

Kernel and Embedded Linux

Bootlin training courses

Embedded Linux, kernel,
Yocto Project, Buildroot, real-time,
graphics, boot time, debugging...

Bootlin logo

Elixir Cross Referencer

#	$NetBSD: t_script.sh,v 1.7 2014/11/16 04:47:18 uebayasi Exp $
#
# Copyright (c) 2014 The NetBSD Foundation, Inc.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# 1. Redistributions of source code must retain the above copyright
#    notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
#    notice, this list of conditions and the following disclaimer in the
#    documentation and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
# ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
# TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
# PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
# BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
#

################################################################################

atf_test_case order_default
order_default_head() {
	atf_set "descr" "check if default object ordering works"
	atf_set "require.progs" "cc" "ld" "readelf" "nm" "sed" "grep"
}

order_default_body() {
	cat > test.x << EOF
SECTIONS {
	/* do nothing; but ld has implicit scripts internally */
	/* which usually do: *(.data) *(.data.*) */
}
EOF
	order_assert_descending
}

################################################################################

atf_test_case order_merge
order_merge_head() {
	atf_set "descr" "check if glob merge keeps object ordering"
	atf_set "require.progs" ${order_require_progs}
}

order_merge_body() {
	cat > test.x << EOF
SECTIONS {
	.data : {
		*(.data .data.*)
	}
}
EOF
	order_assert_descending
}

################################################################################

atf_test_case order_reorder
order_reorder_head() {
	atf_set "descr" "check if object reordering works"
	atf_set "require.progs" ${order_require_progs}
}

order_reorder_body() {
	cat > test.x << EOF
SECTIONS {
	.data : {
		*(.data)
		*(.data.a)
		*(.data.b)
		*(.data.c)
	}
}
EOF
	order_assert_ascending
}

################################################################################

atf_test_case order_sort
order_sort_head() {
	atf_set "descr" "check if object sort works"
	atf_set "require.progs" ${order_require_progs}
}

order_sort_body() {
	cat > test.x << EOF
SECTIONS {
	.data : {
		*(.data)
		/* SORT_BY_NAME */
		SORT(*)(.data.*)
	}
}
EOF
	order_assert_ascending
}

################################################################################

atf_test_case multisec
multisec_head() {
	atf_set "descr" "check if multiple SECTIONS commands work"
	atf_set "require.progs" ${order_require_progs}
}

multisec_body() {
	cat > test.c << EOF
#include <sys/cdefs.h>
char a __section(".data.a") = 'a';
char b __section(".data.b") = 'b';
char c __section(".data.c") = 'c';
EOF
	atf_check -s exit:0 -o ignore -e ignore cc -c test.c

	cat > test.x << EOF
SECTIONS {
	.data : {
		*(.data)
		*(.data.a)
	}
}
SECTIONS {
	.data : {
		*(.data)
		*(.data.b)
	}
}
EOF
	atf_check -s exit:0 -o ignore -e ignore \
	    ld -r -T test.x -Map test.map -o test.ro test.o
	extract_section_names test.ro >test.secs
	extract_symbol_names test.ro >test.syms
	assert_nosec '\.data\.a'
	assert_nosec '\.data\.b'
	assert_sec '\.data\.c'
}

################################################################################

order_require_progs="cc ld readelf nm sed grep"

order_assert_ascending() {
	order_assert_order a b c
}

order_assert_descending() {
	order_assert_order c b a
}

order_assert_order() {
	order_compile
	order_link
	{
		match $1 && match $2 && match $3
	} <test.syms
	atf_check test "$?" -eq 0
}

order_compile() {
	for i in a b c; do
		cat > $i.c << EOF
#include <sys/cdefs.h>
char $i __section(".data.$i") = '$i';
EOF
		atf_check -s exit:0 -o ignore -e ignore cc -c $i.c
	done
	cat > test.c << EOF
int main(void) { return 0; }
EOF
	atf_check -s exit:0 -o ignore -e ignore cc -c test.c
}

order_link() {
	# c -> b -> a
	atf_check -s exit:0 -o ignore -e ignore \
	    ld -r -T test.x -Map test.map -o x.o c.o b.o a.o
	atf_check -s exit:0 -o ignore -e ignore \
	    cc -o test test.o x.o
	extract_symbol_names test |
	grep '^[abc]$' >test.syms
}

extract_section_names() {
	readelf -S "$1" |
	sed -ne '/\] \./ { s/^.*\] //; s/ .*$//; p }'
}

extract_symbol_names() {
	nm -n "$1" |
	sed -e 's/^.* //'
}

match() {
	read line
	case "$line" in
	*"$1"*) return 0;
	esac
	return 1
}

assert_sec() {
	atf_check -s exit:0 -o ignore -e ignore \
	    grep "^$1\$" test.secs
}

assert_nosec() {
	atf_check -s exit:1 -o ignore -e ignore \
	    grep "^$1\$" test.secs
}

################################################################################

atf_init_test_cases()
{
	atf_add_test_case order_default
	atf_add_test_case order_merge
	atf_add_test_case order_reorder
	atf_add_test_case order_sort
	atf_add_test_case multisec
}