(root)/
binutils-2.41/
ld/
testsuite/
ld-elf/
no-section-header.exp
# Expect script for -z nosectionheader and --strip-section-headers tests
# Copyright (C) 2023 Free Software Foundation, Inc.
#
# This file is part of the GNU Binutils.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
# MA 02110-1301, USA.
#
# Written by H.J. Lu (hongjiu.lu@intel.com)
#

if { ![is_elf_format] } {
    return
}

proc binutils_test { prog_name ld_options test readelf_expected nm_expected} {
    global as
    global ld
    global READELF
    global NM
    global objcopy
    global strip
    global srcdir
    global subdir
    global link_output

    eval set prog \$$prog_name

    set test_name "$prog_name --strip-section-headers $ld_options ($test)"

    if { ![ld_assemble $as $srcdir/$subdir/$test.s tmpdir/$test.o ] } {
	unresolved "$test_name"
	return
    }

    append ld_options " -z separate-code -z stack-size=0"
    if { ![ld_link $ld tmpdir/$test "$ld_options tmpdir/$test.o"] } {
	if { [string match "*not supported*" $link_output]
	     || [string match "*unrecognized option*" $link_output]
	     || [string match "*-z .* ignored*" $link_output] } {
	    unsupported "$ld_options is not supported by this target"
	} else {
	    unresolved "$test_name"
	}
	return
    }

    send_log "$prog --strip-section-headers tmpdir/$test\n"
    set got [remote_exec host "$prog --strip-section-headers tmpdir/$test"]
    if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	send_log "$got\n"
	fail "$test_name"
	return
    }

    send_log "$READELF -lSDs --wide tmpdir/$test > tmpdir/$test.out\n"
    set got [remote_exec host "$READELF -lSDs --wide tmpdir/$test" "" "/dev/null" "tmpdir/$test.out"]
    if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	send_log "$got\n"
	unresolved "$test_name"
	return
    }

    if { [regexp_diff "tmpdir/$test.out" "$srcdir/$subdir/$readelf_expected"] } then {
	fail "$test_name"
	return
    }

    if { [string match "*-shared *" $ld_options] } {
	send_log "$NM -D tmpdir/$test > tmpdir/$test.out\n"
	set got [remote_exec host "$NM -D tmpdir/$test" "" "/dev/null" "tmpdir/$test.out"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    unresolved "$test_name"
	    return
	}

	if { [regexp_diff "tmpdir/$test.out" "$srcdir/$subdir/$nm_expected"] } then {
	    fail "$test_name"
	    return
	}
    }

    pass "$test_name"
}

if { [istarget "mips*-*-*"] } {
    set gnu_hash_style "sysv"
} else {
    set gnu_hash_style "gnu"
}

binutils_test objcopy "--hash-style=both" start start-noheader.rd \
	start-noheader.nd
binutils_test objcopy "--hash-style=gnu" start start-noheader.rd \
	start-noheader.nd
binutils_test objcopy "--hash-style=sysv" start start-noheader.rd \
	start-noheader.nd
binutils_test objcopy "--hash-style=both -shared" start \
	start-shared-noheader-$gnu_hash_style.rd start-shared-noheader.nd
binutils_test objcopy "--hash-style=gnu -shared" start \
	start-shared-noheader-$gnu_hash_style.rd start-shared-noheader.nd
binutils_test objcopy "--hash-style=sysv -shared" start \
	start-shared-noheader-sysv.rd start-shared-noheader.nd
binutils_test strip "--hash-style=both" start start-noheader.rd \
	start-noheader.nd
binutils_test strip "--hash-style=gnu" start start-noheader.rd \
	start-noheader.nd
binutils_test strip "--hash-style=sysv" start start-noheader.rd \
	start-noheader.nd
binutils_test strip "--hash-style=both -shared" start \
	start-shared-noheader-$gnu_hash_style.rd start-shared-noheader.nd
binutils_test strip "--hash-style=gnu -shared" start \
	start-shared-noheader-$gnu_hash_style.rd start-shared-noheader.nd
binutils_test strip "--hash-style=sysv -shared" start \
	start-shared-noheader-sysv.rd start-shared-noheader.nd


if { ![check_compiler_available] || ![check_shared_lib_support] } {
    return
}

proc binutils_run_test { prog } {
    global CC_FOR_TARGET
    global READELF
    global NM
    global objcopy
    global strip
    global srcdir
    global subdir
    # Add $NOPIE_CFLAGS and $NOPIE_LDFLAGS if non-PIE is required.
    global NOPIE_CFLAGS NOPIE_LDFLAGS

    set sec_hdr "sec-hdr"
    if { "$prog" == "" } {
	set prog_name none
    } else {
	set prog_name $prog
	set ld_options ""
	switch -- $prog {
	    objcopy
		{ set prog $objcopy }
	    strip
		{ set prog $strip }
	    default
		{
		    fail "Build pr25617-1a-no-sec-hdr.so ($prog_name)"
		    break
		}
	}
    }

    run_cc_link_tests [list \
	[list \
	    "Build pr25617-1a-no-sec-hdr.so ($prog_name)" \
	    "-shared -Wl,-z,separate-code,--hash-style=sysv" \
	    "-fPIC" \
	    {pr25617-1a.c} \
	    [list \
		[list "readelf" "-lWSDs" "pr25617-1a-$sec_hdr.rd"] \
		[list "nm" "-D" "pr25617-1a-no-sec-hdr.nd"] \
	    ]\
	    "pr25617-1a-no-sec-hdr.so" \
	] \
	[list \
	    "Build pr25617-1a-now-no-sec-hdr.so ($prog_name)" \
	    "-shared -Wl,-z,separate-code,-z,now,--hash-style=gnu" \
	    "-fPIC" \
	    {pr25617-1a.c} \
	    [list \
		[list "readelf" "-lWSDs" "pr25617-1a-$sec_hdr.rd"] \
		[list "nm" "-D" "pr25617-1a-no-sec-hdr.nd"] \
	    ]\
	    "pr25617-1a-now-no-sec-hdr.so" \
	] \
	[list \
	    "Build pr25617-1 (-z nosectionheader, $prog_name)" \
	    "$NOPIE_LDFLAGS -Wl,-z,separate-code,--no-as-needed \
	     -Wl,--hash-style=sysv -Wl,-z,nosectionheader \
	     tmpdir/pr25617-1a-no-sec-hdr.so" \
	    "$NOPIE_CFLAGS" \
	    {pr25617-1b.c} \
	    {{readelf -lWSDs pr25617-1-no-sec-hdr.rd} \
	     {nm -D pr25617-1-no-sec-hdr.nd}} \
	    "pr25617-1-no-sec-hdr" \
	] \
	[list \
	    "Build pr25617-1 (PIE, -z nosectionheader, $prog_name)" \
	    "-pie -Wl,-z,separate-code,--no-as-needed,--hash-style=gnu \
	     -Wl,-z,nosectionheader tmpdir/pr25617-1a-now-no-sec-hdr.so" \
	    "-fPIE" \
	    {pr25617-1b.c} \
	    {{readelf -lWSDs pr25617-1-no-sec-hdr.rd} \
	     {nm -D pr25617-1-no-sec-hdr.nd}} \
	    "pr25617-1-pie-no-sec-hdr" \
	] \
	[list \
	    "Build pr25617-1 (static, -z nosectionheader, $prog_name)" \
	    "-static -Wl,-z,separate-code -Wl,-z,nosectionheader" \
	    "" \
	    {pr25617-1a.c pr25617-1b.c} \
	    {{readelf -lSWDs pr25617-1-static-no-sec-hdr.rd}} \
	    "pr25617-1-static-no-sec-hdr" \
	] \
    ]

    run_ld_link_exec_tests [list \
	[list \
	    "Run pr25617-1 (-z nosectionheader, $prog_name)" \
	    "$NOPIE_LDFLAGS -Wl,-z,separate-code,--no-as-needed \
	     -Wl,--hash-style=sysv -Wl,-z,nosectionheader \
	     tmpdir/pr25617-1a-no-sec-hdr.so" \
	    "" \
	    {pr25617-1b.c} \
	    "pr25617-1-no-sec-hdr" \
	    "pass.out" \
	    "$NOPIE_CFLAGS" \
	] \
	[list \
	    "Run pr25617-1 (PIE, -z nosectionheader, $prog_name)" \
	    "-pie -Wl,-z,separate-code,--no-as-needed,--hash-style=gnu \
	     -Wl,-z,nosectionheader tmpdir/pr25617-1a-now-no-sec-hdr.so" \
	    "" \
	    {pr25617-1b.c} \
	    "pr25617-1-pie-no-sec-hdr" \
	    "pass.out" \
	    "-fPIE" \
	] \
	[list \
	    "Run pr25617-1 (static, -z nosectionheader, $prog_name)" \
	    "-static -Wl,-z,separate-code -Wl,-z,nosectionheader" \
	    "" \
	    {pr25617-1a.c pr25617-1b.c} \
	    "pr25617-1-static-no-sec-hdr" \
	    "pass.out" \
	] \
    ]

    if { "$prog_name" != "none" } {
	send_log "$prog --strip-section-headers tmpdir/pr25617-1a-no-sec-hdr.so\n"
	set got [remote_exec host "$prog --strip-section-headers tmpdir/pr25617-1a-no-sec-hdr.so"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    fail "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
	    return
	}

	send_log "$READELF -lWSDs tmpdir/pr25617-1a-no-sec-hdr.so > tmpdir/dump.out\n"
	set got [remote_exec host "$READELF -lWSDs tmpdir/pr25617-1a-no-sec-hdr.so" "" "/dev/null" "tmpdir/dump.out"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    unresolved "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
	    return
	}

	if { [regexp_diff "tmpdir/dump.out" "$srcdir/$subdir/pr25617-1a-no-sec-hdr.rd"] } then {
	    unresolved "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
	    return
	}

	send_log "$NM -D tmpdir/pr25617-1a-no-sec-hdr.so > tmpdir/dump.out\n"
	set got [remote_exec host "$NM -D tmpdir/pr25617-1a-no-sec-hdr.so" "" "/dev/null" "tmpdir/dump.out"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    unresolved "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
	    return
	}

	if { [regexp_diff "tmpdir/dump.out" "$srcdir/$subdir/pr25617-1a-no-sec-hdr.nd"] } then {
	    unresolved "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
	    return
	}

	if { [isnative] } {
	    send_log "tmpdir/pr25617-1-no-sec-hdr > tmpdir/pr25617-1.out\n"
	    catch "exec tmpdir/pr25617-1-no-sec-hdr > tmpdir/pr25617-1.out" got
	    if ![string match "" $got] then {
		send_log "$got\n"
		unresolved "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
		return
	    }

	    send_log "diff tmpdir/pr25617-1.out $srcdir/$subdir/pass.out\n"
	    catch "exec diff tmpdir/pr25617-1.out $srcdir/$subdir/pass.out" got
	    if ![string match "" $got] then {
		send_log "$got\n"
		fail "Update pr25617-1a-no-sec-hdr.so ($prog_name)"
		return
	    }
	}

	pass "Update pr25617-1a-no-sec-hdr.so ($prog_name)"

	send_log "$prog --strip-section-headers tmpdir/pr25617-1a-now-no-sec-hdr.so\n"
	set got [remote_exec host "$prog --strip-section-headers tmpdir/pr25617-1a-now-no-sec-hdr.so"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    fail "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
	    return
	}

	send_log "$READELF -lWSDs tmpdir/pr25617-1a-now-no-sec-hdr.so > tmpdir/dump.out\n"
	set got [remote_exec host "$READELF -lWSDs tmpdir/pr25617-1a-now-no-sec-hdr.so" "" "/dev/null" "tmpdir/dump.out"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    unresolved "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
	    return
	}

	if { [regexp_diff "tmpdir/dump.out" "$srcdir/$subdir/pr25617-1a-no-sec-hdr.rd"] } then {
	    unresolved "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
	    return
	}

	send_log "$NM -D tmpdir/pr25617-1a-now-no-sec-hdr.so > tmpdir/dump.out\n"
	set got [remote_exec host "$NM -D tmpdir/pr25617-1a-now-no-sec-hdr.so" "" "/dev/null" "tmpdir/dump.out"]
	if { [lindex $got 0] != 0 || ![string match "" [lindex $got 1]] } then {
	    send_log "$got\n"
	    unresolved "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
	    return
	}

	if { [regexp_diff "tmpdir/dump.out" "$srcdir/$subdir/pr25617-1a-no-sec-hdr.nd"] } then {
	    unresolved "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
	    return
	}

	if { [isnative] } {
	    send_log "tmpdir/pr25617-1-pie-no-sec-hdr > tmpdir/pr25617-1-pie.out\n"
	    catch "exec tmpdir/pr25617-1-pie-no-sec-hdr > tmpdir/pr25617-1-pie.out" got
	    if ![string match "" $got] then {
		send_log "$got\n"
		unresolved "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
		return
	    }

	    send_log "diff tmpdir/pr25617-1-pie.out $srcdir/$subdir/pass.out\n"
	    catch "exec diff tmpdir/pr25617-1-pie.out $srcdir/$subdir/pass.out" got
	    if ![string match "" $got] then {
		send_log "$got\n"
		fail "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
		return
	    }
	}

	set got [run_host_cmd $CC_FOR_TARGET "-o tmpdir/pr25617-1 tmpdir/pr25617-1b.o tmpdir/pr25617-1a-now-no-sec-hdr.so"]
	if ![string match "*pr25617-1a-now-no-sec-hdr.so*file in wrong format*" $got] then {
	    send_log "$got\n"
	    fail "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
	    return
	}

	pass "Update pr25617-1a-now-no-sec-hdr.so ($prog_name)"
     }
}

binutils_run_test ""
binutils_run_test objcopy
binutils_run_test strip