| #   Copyright (C) 1995-2024 Free Software Foundation, Inc. | 
 |  | 
 | # 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. | 
 |  | 
 | # Please email any bugs, comments, and/or additions to this file to: | 
 | # bug-dejagnu@prep.ai.mit.edu | 
 |  | 
 | # Written by Ian Lance Taylor <ian@cygnus.com> | 
 |  | 
 | if ![is_remote host] { | 
 |     if {[which $AR] == 0} then { | 
 |         perror "$AR does not exist" | 
 |         return | 
 |     } | 
 | } | 
 |  | 
 | set obj o | 
 | if { [istarget "*-*-vms"] } then { | 
 |     set obj obj | 
 | } | 
 |  | 
 | # send_user "Version [binutil_version $AR]" | 
 |  | 
 | # Test long file name support | 
 |  | 
 | proc long_filenames { bfdtests } { | 
 |     global AR | 
 |     global host_triplet | 
 |     global base_dir | 
 |  | 
 |     set testname "ar long file names" | 
 |  | 
 |     set n1 "abcdefghijklmnopqrstuvwxyz1" | 
 |     set n2 "abcdefghijklmnopqrstuvwxyz2" | 
 |     set file1 tmpdir/$n1 | 
 |     set file2 tmpdir/$n2 | 
 |  | 
 |     remote_file build delete $file1 | 
 |     remote_file host delete $n1 | 
 |  | 
 |     # Some file systems truncate file names at 14 characters, which | 
 |     # makes it impossible to run this test.  Check for that now. | 
 |     set status [catch "set f [open tmpdir/$n1 w]" errs] | 
 |     if { $status != 0 } { | 
 | 	verbose -log "open tmpdir/$n1 returned $errs" | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |     puts $f "first" | 
 |     close $f | 
 |  | 
 |     remote_file build delete $file2 | 
 |     remote_file host delete $n2 | 
 |  | 
 |     set status [catch "set f [open tmpdir/$n2 w]" errs] | 
 |     if { $status != 0 } { | 
 | 	verbose -log "open tmpdir/$n2 returned $errs" | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |     puts $f "second" | 
 |     close $f | 
 |  | 
 |     if [is_remote host] { | 
 | 	set file1 [remote_download host $file1] | 
 | 	set file2 [remote_download host $file2] | 
 | 	set dest artest.a | 
 |     } else { | 
 | 	set dest tmpdir/artest.a | 
 |     } | 
 |  | 
 |     remote_file host delete $dest | 
 |  | 
 |     set got [binutils_run $AR "rc $dest $file1 $file2"] | 
 |     if [is_remote host] { | 
 | 	remote_upload host $file1 tmpdir/$n1 | 
 |     } | 
 |  | 
 |     set f [open tmpdir/$n1 r] | 
 |     gets $f string | 
 |     close $f | 
 |     if ![string match "first" $string] { | 
 | 	verbose -log "reading tmpdir/$n1 returned $string" | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     remote_file host delete $dest | 
 |     set got [binutils_run $AR "rc $dest $file1 $file2"] | 
 |  | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/$n1 | 
 |     remote_file build delete tmpdir/$n2 | 
 |  | 
 |     set got [binutils_run $AR "t $dest"] | 
 |     regsub "\[\r\n \t\]*$" "$got" "" got | 
 |     if ![string match "$n1*$n2" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	remote_file host delete $file1 | 
 | 	remote_file host delete $file2 | 
 |     } | 
 |  | 
 |     set exec_output [binutils_run $AR "x $dest"] | 
 |     set exec_output [prune_warnings $exec_output] | 
 |     if ![string match "" $exec_output] { | 
 | 	verbose -log $exec_output | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     foreach bfdtest $bfdtests { | 
 | 	set exec_output [binutils_run "$base_dir/$bfdtest" "$dest"] | 
 | 	if ![string match "" $exec_output] { | 
 | 	    verbose -log $exec_output | 
 | 	    fail "$testname ($bfdtest)" | 
 | 	    return | 
 | 	} | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	remote_upload host $n1 tmpdir/$n1 | 
 | 	remote_upload host $n2 tmpdir/$n2 | 
 | 	set file1 tmpdir/$n1 | 
 | 	set file2 tmpdir/$n2 | 
 |     } else { | 
 | 	set file1 $n1 | 
 | 	set file2 $n2 | 
 |     } | 
 |  | 
 |     if ![file exists $file1] { | 
 | 	verbose -log "$file1 does not exist" | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |     if ![file exists $file2] { | 
 | 	verbose -log "$file2 does not exist" | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set f [open $file1 r] | 
 |     if { [gets $f line] == -1 || $line != "first" } { | 
 | 	verbose -log "$file1 contents:" | 
 | 	verbose -log "$line" | 
 | 	close $f | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |     close $f | 
 |  | 
 |     set f [open $file2 r] | 
 |     if { [gets $f line] == -1 || $line != "second" } { | 
 | 	verbose -log "$file2 contents:" | 
 | 	verbose -log "$line" | 
 | 	close $f | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |     close $f | 
 |  | 
 |     file delete $file1 $file2 | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test building the symbol table. | 
 |  | 
 | proc symbol_table { } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar symbol table" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "rc $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $NM "--print-armap $archive"] | 
 |     if { ![string match "*text_symbol in bintest.${obj}*" $got] \ | 
 | 	 || ![string match "*data_symbol in bintest.${obj}*" $got] \ | 
 | 	 || ![string match "*common_symbol in bintest.${obj}*" $got] \ | 
 | 	 || [string match "*static_text_symbol in bintest.${obj}*" $got] \ | 
 | 	 || [string match "*static_data_symbol in bintest.${obj}*" $got] \ | 
 | 	 || [string match "*external_symbol in bintest.${obj}*" $got] } { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test building a thin archive. | 
 |  | 
 | proc thin_archive { bfdtests } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global base_dir | 
 |     global obj | 
 |  | 
 |     set testname "ar thin archive" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "rcT $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     foreach bfdtest $bfdtests { | 
 | 	set exec_output [binutils_run "$base_dir/$bfdtest" "$archive"] | 
 | 	if ![string match "" $exec_output] { | 
 | 	    verbose -log $exec_output | 
 | 	    fail "$testname ($bfdtest)" | 
 | 	    return | 
 | 	} | 
 |     } | 
 |  | 
 |     set got [binutils_run $NM "--print-armap $archive"] | 
 |     if { ![string match "*text_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || ![string match "*data_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || ![string match "*common_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || [string match "*static_text_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || [string match "*static_data_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || [string match "*external_symbol in *bintest.${obj}*" $got] } { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test building a thin archive with a nested archive. | 
 |  | 
 | proc thin_archive_with_nested { bfdtests } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global base_dir | 
 |     global obj | 
 |  | 
 |     set testname "ar thin archive with nested archive" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set archive2 artest2.a | 
 | 	set archive3 artest3.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set archive2 tmpdir/artest2.a | 
 | 	set archive3 tmpdir/artest3.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "rc $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest2.a | 
 |  | 
 |     set got [binutils_run $AR "rcT $archive2 ${archive}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest3.a | 
 |  | 
 |     set got [binutils_run $AR "rc --thin $archive3 ${archive}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     foreach bfdtest $bfdtests { | 
 | 	set exec_output [binutils_run "$base_dir/$bfdtest" "$archive"] | 
 | 	if ![string match "" $exec_output] { | 
 | 	    verbose -log $exec_output | 
 | 	    fail "$testname ($bfdtest)" | 
 | 	    return | 
 | 	} | 
 |  | 
 | 	set exec_output [binutils_run "$base_dir/$bfdtest" "$archive2"] | 
 | 	if ![string match "" $exec_output] { | 
 | 	    verbose -log $exec_output | 
 | 	    fail "$testname ($bfdtest)" | 
 | 	    return | 
 | 	} | 
 |  | 
 | 	set exec_output [binutils_run "$base_dir/$bfdtest" "$archive3"] | 
 | 	if ![string match "" $exec_output] { | 
 | 	    verbose -log $exec_output | 
 | 	    fail "$testname ($bfdtest)" | 
 | 	    return | 
 | 	} | 
 |     } | 
 |  | 
 |     set got [binutils_run $NM "--print-armap $archive"] | 
 |     if { ![string match "*text_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || ![string match "*data_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || ![string match "*common_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || [string match "*static_text_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || [string match "*static_data_symbol in *bintest.${obj}*" $got] \ | 
 | 	 || [string match "*external_symbol in *bintest.${obj}*" $got] } { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test POSIX-compatible argument parsing. | 
 |  | 
 | proc argument_parsing { } { | 
 |     global AR | 
 |     global AS | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar argument parsing" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "-r -c $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test building a deterministic archive. | 
 |  | 
 | proc deterministic_archive { } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar deterministic archive" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "rcD $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "tv $archive"] | 
 |     # This only checks the file mode and uid/gid.  We can't easily match | 
 |     # date because it's printed with the user's timezone. | 
 |     if ![string match "rw-r--r-- 0/0 *bintest.${obj}*" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "tvO $archive"] | 
 |     if ![string match "rw-r--r-- 0/0 *bintest.${obj} 0x*" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test replacing a member of a deterministic archive. | 
 |  | 
 | proc replacing_deterministic_member { } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "replacing deterministic member" | 
 |  | 
 |     if [is_remote host] { | 
 | 	# The kind of filename trickery that we are about to | 
 | 	#  play is hard to do if we have to operate remotely. | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     file mkdir tmpdir/ar | 
 |      | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     # Wait a second and then build a second object file - with the same name | 
 |     # as the first, but in a different directory. | 
 |     sleep 1 | 
 |     if ![binutils_assemble $srcdir/$subdir/copytest.s tmpdir/ar/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set archive tmpdir/artest.a | 
 |     set older_objfile tmpdir/bintest.${obj} | 
 |     set newer_objfile tmpdir/ar/bintest.${obj} | 
 |     set older_length [file size $older_objfile] | 
 |     # set newer_length [file size $newer_objfile] | 
 |      | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     # Build the archive with the *newer* object file. | 
 |      | 
 |     set got [binutils_run $AR "rcD $archive ${newer_objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail "$testname: (could not build archive)" | 
 | 	return | 
 |     } | 
 |  | 
 |     # Now replace the newer file with the older one.  On a normal | 
 |     # archive this will not work, but one created to be deterministic | 
 |     # should always replace its members. | 
 |      | 
 |     set got [binutils_run $AR "ruD $archive $older_objfile"] | 
 |     # The archiver will warn that 'u' and 'D' do not work together | 
 |     if ![string match "*not meaningful*" $got] { | 
 | 	fail "$testname: (failed to replace file)" | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "tvO $archive"] | 
 |     if ![string match "rw-r--r-- 0/0 *${older_length} *bintest.${obj} 0x*" $got] { | 
 | 	fail "$testname (wrong size, expected: $older_length)" | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test replacing a member of a non-deterministic archive. | 
 | # This test expects SOURCE_DATE_EPOCH to not be set in the environment. | 
 |  | 
 | proc replacing_non_deterministic_member { } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "replacing non-deterministic member" | 
 |  | 
 |     if [is_remote host] { | 
 | 	# The kind of filename trickery that we are about to | 
 | 	#  play is hard to do if we have to operate remotely. | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     file mkdir tmpdir/ar | 
 |      | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     # Wait a second and then build a second object file - with the same name | 
 |     # as the first, but in a different directory. | 
 |     sleep 1 | 
 |     if ![binutils_assemble $srcdir/$subdir/copytest.s tmpdir/ar/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set archive tmpdir/artest.a | 
 |     set older_objfile tmpdir/bintest.${obj} | 
 |     set newer_objfile tmpdir/ar/bintest.${obj} | 
 |     # set older_length [file size $older_objfile] | 
 |     set newer_length [file size $newer_objfile] | 
 |      | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     # Build the archive with the *newer* object file. | 
 |      | 
 |     set got [binutils_run $AR "rcU $archive ${newer_objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail "$testname: (could not build archive)" | 
 | 	return | 
 |     } | 
 |  | 
 |     # Now try to replace the newer file with the older one.  This should not work. | 
 |      | 
 |     set got [binutils_run $AR "ruU $archive $older_objfile"] | 
 |     if ![string match "" $got] { | 
 | 	fail "$testname: (failed to replace file)" | 
 | 	return | 
 |     } | 
 |  | 
 |     # Since this archive is non-deterministic, we do not know what the | 
 |     # user or group ids will be, so we have to use */* to match them. | 
 |     set got [binutils_run $AR "tvO $archive"] | 
 |     if ![string match "\[rw-\]* */* *${newer_length} *bintest.${obj} 0x*" $got] { | 
 | 	fail "$testname (wrong size, expected: $newer_length)" | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test replacing a member of deterministic archive created by using SOURCE_DATE_EPOCH. | 
 |  | 
 | proc replacing_sde_deterministic_member { } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "replacing SOURCE_DATE_EPOCH deterministic member" | 
 |  | 
 |     if [is_remote host] { | 
 | 	# The kind of filename trickery that we are about to | 
 | 	#  play is hard to do if we have to operate remotely. | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     file mkdir tmpdir/ar | 
 |      | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     # Wait a second and then build a second object file - with the same name | 
 |     # as the first, but in a different directory. | 
 |     sleep 1 | 
 |     if ![binutils_assemble $srcdir/$subdir/copytest.s tmpdir/ar/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set archive tmpdir/artest.a | 
 |     set older_objfile tmpdir/bintest.${obj} | 
 |     set newer_objfile tmpdir/ar/bintest.${obj} | 
 |     set older_length [file size $older_objfile] | 
 |     # set newer_length [file size $newer_objfile] | 
 |      | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     # Build the archive with the *newer* object file. | 
 |     setenv SOURCE_DATE_EPOCH "1000" | 
 |      | 
 |     set got [binutils_run $AR "rcU $archive ${newer_objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail "$testname: (could not build archive)" | 
 | 	unsetenv SOURCE_DATE_EPOCH | 
 | 	return | 
 |     } | 
 |  | 
 |     # Now replace the newer file with the older one.  On a normal | 
 |     # archive this will not work, but one created to be deterministic | 
 |     # should always replace its members. | 
 |      | 
 |     set got [binutils_run $AR "ruU $archive $older_objfile"] | 
 |     if ![string match "" $got] { | 
 | 	fail "$testname: (failed to replace file)" | 
 | 	unsetenv SOURCE_DATE_EPOCH | 
 | 	return | 
 |     } | 
 |  | 
 |     # Since this archive has fixed source dates, but non-deterministic | 
 |     # uid and gid values we have to use */* to match them. | 
 |     set got [binutils_run $AR "tvO $archive"] | 
 |     if ![string match "\[rw-\]* */* *${older_length} *bintest.${obj} 0x*" $got] { | 
 | 	fail "$testname (wrong size, expected: $older_length)" | 
 | 	unsetenv SOURCE_DATE_EPOCH | 
 | 	return | 
 |     } | 
 |  | 
 |     # FIXME - it would be nice if we could check to see that the time & date | 
 |     # in the archive listing matches SOURCE_DATE_EPOCH. | 
 |  | 
 |     unsetenv SOURCE_DATE_EPOCH | 
 |     pass $testname | 
 | } | 
 |  | 
 |  | 
 | proc unique_symbol { } { | 
 |     global AR | 
 |     global AS | 
 |     global NM | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar unique symbol in archive" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/unique.s tmpdir/unique.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/unique.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/unique.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "-s -r -c $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $NM "--print-armap $archive"] | 
 |     if ![string match "*foo in *unique.${obj}*" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test deleting an element. | 
 |  | 
 | proc delete_an_element { } { | 
 |     global AR | 
 |     global AS | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar deleting an element" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "-r -c $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "-d $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test moving an element. | 
 |  | 
 | proc move_an_element { } { | 
 |     global AR | 
 |     global AS | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar moving an element" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "-r -c $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "-m $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # PR 19775: Test creating and listing archives with an empty element. | 
 |  | 
 | proc empty_archive { } { | 
 |     global AR | 
 |     global srcdir | 
 |     global subdir | 
 |  | 
 |     set testname "archive with empty element" | 
 |  | 
 |     # FIXME: There ought to be a way to dynamically create an empty file. | 
 |     set empty $srcdir/$subdir/empty | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host $empty] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile $empty | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "-r -c $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     # This commmand used to fail with: "Malformed archive". | 
 |     set got [binutils_run $AR "-t $archive"] | 
 |     if ![string match "empty | 
 | " $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Test extracting an element. | 
 |  | 
 | proc extract_an_element { } { | 
 |     global AR | 
 |     global AS | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar extracting an element" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set archive artest.a | 
 |  | 
 |     if [is_remote host] { | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete $archive | 
 |  | 
 |     set got [binutils_run $AR "-r -c $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "--output=tmpdir -x $archive ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     remote_file build delete $archive | 
 |     remote_file build delete tmpdir/$archive | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | proc many_files { } { | 
 |     global AR | 
 |     global AS | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar many files" | 
 |  | 
 |     set ofiles {} | 
 |     set max_file 150 | 
 |     for { set i 0 } { $i < $max_file } { incr i } { | 
 | 	set sfile "tmpdir/d-$i.s" | 
 | 	if [catch { set ofd [open $sfile w] } x] { | 
 | 	    perror "$x" | 
 | 	    unresolved $testname | 
 | 	    return | 
 | 	} | 
 |  | 
 | 	puts $ofd " .globl data_sym$i" | 
 | 	puts $ofd " .data" | 
 | 	puts $ofd "data_sym$i:" | 
 | 	puts $ofd " .long $i" | 
 | 	close $ofd | 
 |  | 
 | 	set ofile "tmpdir/d-$i.${obj}" | 
 | 	if ![binutils_assemble $sfile $ofile] { | 
 | 	    unsupported $testname | 
 | 	    return | 
 | 	} | 
 |  | 
 | 	set objfile $ofile | 
 | 	if [is_remote host] { | 
 | 	    remote_file host delete $sfile | 
 | 	    set objfile [remote_download host $ofile] | 
 | 	    remote_file build delete $ofile | 
 | 	} | 
 | 	remote_file build delete $sfile | 
 | 	lappend ofiles $objfile | 
 |     } | 
 |  | 
 |     set archive tmpdir/many.a | 
 |     remote_file host delete $archive | 
 |  | 
 |     set got [binutils_run $AR "cr $archive $ofiles"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     remote_file host delete $archive | 
 |     eval remote_file host delete $ofiles | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | proc test_add_dependencies { } { | 
 |     global AR | 
 |     global AS | 
 |     global srcdir | 
 |     global subdir | 
 |     global obj | 
 |  | 
 |     set testname "ar adding library dependencies" | 
 |  | 
 |     if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.${obj}] { | 
 | 	unsupported $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     if [is_remote host] { | 
 | 	set archive artest.a | 
 | 	set objfile [remote_download host tmpdir/bintest.${obj}] | 
 | 	remote_file host delete $archive | 
 |     } else { | 
 | 	set archive tmpdir/artest.a | 
 | 	set objfile tmpdir/bintest.${obj} | 
 |     } | 
 |  | 
 |     remote_file build delete tmpdir/artest.a | 
 |  | 
 |     set got [binutils_run $AR "-r -c $archive --record-libdeps /foo/bar ${objfile}"] | 
 |     if ![string match "" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     set got [binutils_run $AR "-t $archive"] | 
 |     if ![string match "*bintest.${obj} | 
 | __.LIBDEP*" $got] { | 
 | 	fail $testname | 
 | 	return | 
 |     } | 
 |  | 
 |     pass $testname | 
 | } | 
 |  | 
 | # Run the tests. | 
 |  | 
 | # Only run the bfdtest checks if the programs exist.  Since these | 
 | # programs are built but not installed, running the testsuite on an | 
 | # installed toolchain will produce ERRORs about missing bfdtest1 and | 
 | # bfdtest2 executables. | 
 | if { [file exists $base_dir/bfdtest1] && [file exists $base_dir/bfdtest2] } { | 
 |     set bfdtests [list bfdtest1 bfdtest2] | 
 |  | 
 |     long_filenames $bfdtests | 
 |  | 
 |     # xcoff, ecoff, and vms archive support doesn't handle thin archives | 
 |     if { ![is_xcoff_format] | 
 | 	 && ![istarget "*-*-*ecoff"] | 
 | 	 && ![istarget "*-*-vms"] } { | 
 | 	thin_archive $bfdtests | 
 | 	thin_archive_with_nested $bfdtests | 
 |     } | 
 | } | 
 |  | 
 | symbol_table | 
 | argument_parsing | 
 | deterministic_archive | 
 | replacing_deterministic_member | 
 | replacing_non_deterministic_member | 
 | replacing_sde_deterministic_member | 
 | delete_an_element | 
 | move_an_element | 
 | empty_archive | 
 | extract_an_element | 
 | many_files | 
 | test_add_dependencies | 
 |  | 
 | if { [is_elf_format] && [supports_gnu_unique] } { | 
 |     unique_symbol | 
 | } |