readelf: print 0x0 as 0, and remove trailing spaces
[binutils-gdb.git] / binutils / testsuite / binutils-all / ar.exp
index 6efc1598472586233223023a94b326e58b38d1df..3b841825f6f0b93d7efd64859960a839cb43506e 100644 (file)
@@ -1,16 +1,15 @@
-#   Copyright 1995, 1997, 2002, 2004, 2007, 2008, 2009, 2010, 2012
-#   Free Software Foundation, Inc.
+#   Copyright (C) 1995-2022 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.
@@ -27,6 +26,11 @@ if ![is_remote host] {
     }
 }
 
+set obj o
+if { [istarget "*-*-vms"] } then {
+    set obj obj
+}
+
 # send_user "Version [binutil_version $AR]"
 
 # Test long file name support
@@ -110,7 +114,7 @@ proc long_filenames { bfdtests } {
        fail $testname
        return
     }
-    
+
     if [is_remote host] {
        remote_file host delete $file1
        remote_file host delete $file2
@@ -186,21 +190,22 @@ proc symbol_table { } {
     global NM
     global srcdir
     global subdir
+    global obj
 
     set testname "ar symbol table"
 
-    if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/bintest.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/bintest.o
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -212,12 +217,12 @@ proc symbol_table { } {
     }
 
     set got [binutils_run $NM "--print-armap $archive"]
-    if { ![string match "*text_symbol in bintest.o*" $got] \
-        || ![string match "*data_symbol in bintest.o*" $got] \
-        || ![string match "*common_symbol in bintest.o*" $got] \
-        || [string match "*static_text_symbol in bintest.o*" $got] \
-        || [string match "*static_data_symbol in bintest.o*" $got] \
-        || [string match "*external_symbol in bintest.o*" $got] } {
+    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
     }
@@ -234,21 +239,22 @@ proc thin_archive { bfdtests } {
     global srcdir
     global subdir
     global base_dir
+    global obj
 
     set testname "ar thin archive"
 
-    if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/bintest.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/bintest.o
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -269,12 +275,12 @@ proc thin_archive { bfdtests } {
     }
 
     set got [binutils_run $NM "--print-armap $archive"]
-    if { ![string match "*text_symbol in *bintest.o*" $got] \
-        || ![string match "*data_symbol in *bintest.o*" $got] \
-        || ![string match "*common_symbol in *bintest.o*" $got] \
-        || [string match "*static_text_symbol in *bintest.o*" $got] \
-        || [string match "*static_data_symbol in *bintest.o*" $got] \
-        || [string match "*external_symbol in *bintest.o*" $got] } {
+    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
     }
@@ -291,23 +297,26 @@ proc thin_archive_with_nested { bfdtests } {
     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.o] {
-       unresolved $testname
+    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 objfile [remote_download host tmpdir/bintest.o]
+       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 objfile tmpdir/bintest.o
+       set archive3 tmpdir/artest3.a
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -326,6 +335,14 @@ proc thin_archive_with_nested { bfdtests } {
        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] {
@@ -340,15 +357,22 @@ proc thin_archive_with_nested { bfdtests } {
            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.o*" $got] \
-        || ![string match "*data_symbol in *bintest.o*" $got] \
-        || ![string match "*common_symbol in *bintest.o*" $got] \
-        || [string match "*static_text_symbol in *bintest.o*" $got] \
-        || [string match "*static_data_symbol in *bintest.o*" $got] \
-        || [string match "*external_symbol in *bintest.o*" $got] } {
+    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
     }
@@ -363,21 +387,22 @@ proc argument_parsing { } {
     global AS
     global srcdir
     global subdir
+    global obj
 
     set testname "ar argument parsing"
 
-    if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/bintest.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/bintest.o
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -399,21 +424,22 @@ proc deterministic_archive { } {
     global NM
     global srcdir
     global subdir
+    global obj
 
     set testname "ar deterministic archive"
 
-    if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/bintest.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/bintest.o
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -427,7 +453,13 @@ proc deterministic_archive { } {
     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.o*" $got] {
+    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
     }
@@ -441,20 +473,22 @@ proc unique_symbol { } {
     global NM
     global srcdir
     global subdir
+    global obj
 
     set testname "ar unique symbol in archive"
 
-    if ![binutils_assemble $srcdir/$subdir/unique.s tmpdir/unique.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/unique.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/unique.o
+       set objfile tmpdir/unique.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -466,7 +500,7 @@ proc unique_symbol { } {
     }
 
     set got [binutils_run $NM "--print-armap $archive"]
-    if ![string match "*foo in *unique.o*" $got] {
+    if ![string match "*foo in *unique.${obj}*" $got] {
        fail $testname
        return
     }
@@ -481,21 +515,22 @@ proc delete_an_element { } {
     global AS
     global srcdir
     global subdir
+    global obj
 
     set testname "ar deleting an element"
 
-    if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/bintest.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/bintest.o
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -522,21 +557,22 @@ proc move_an_element { } {
     global AS
     global srcdir
     global subdir
+    global obj
 
     set testname "ar moving an element"
 
-    if ![binutils_assemble $srcdir/$subdir/bintest.s tmpdir/bintest.o] {
-       unresolved $testname
+    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.o]
+       set objfile [remote_download host tmpdir/bintest.${obj}]
        remote_file host delete $archive
     } else {
        set archive tmpdir/artest.a
-       set objfile tmpdir/bintest.o
+       set objfile tmpdir/bintest.${obj}
     }
 
     remote_file build delete tmpdir/artest.a
@@ -556,20 +592,216 @@ proc move_an_element { } {
     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\r" $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}\r__.LIBDEP*" $got] {
+       fail $testname
+       return
+    }
+
+    pass $testname
+}
+
 # Run the tests.
 
-set bfdtests [list bfdtest1 bfdtest2]
+# 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
+    }
+}
 
-long_filenames $bfdtests
 symbol_table
-thin_archive $bfdtests
-thin_archive_with_nested $bfdtests
 argument_parsing
 deterministic_archive
 delete_an_element
 move_an_element
-if { [is_elf_format]
-     && ![istarget "*-*-hpux*"]
-     && ![istarget "msp*-*-*"] } {
+empty_archive
+extract_an_element
+many_files
+test_add_dependencies
+
+if { [is_elf_format] && [supports_gnu_unique] } {
     unique_symbol
 }