]> git.ipfire.org Git - thirdparty/elfutils.git/commitdiff
ar: Correct -N COUNT off-by-one
authorMark Wielaard <mark@klomp.org>
Sun, 28 Aug 2022 15:46:08 +0000 (17:46 +0200)
committerMark Wielaard <mark@klomp.org>
Wed, 14 Sep 2022 18:48:29 +0000 (20:48 +0200)
When using instance [COUNT], the instance check is wrong.
instance-- == 0 should be --instance == 0.

Add a testcase run-ar-N.sh that uses -N COUNT with extract and delete
operations checking the right instance was extracted and deleted.

https://sourceware.org/bugzilla/show_bug.cgi?id=28725

Reported-by: panxiaohe <panxh_ran@163.com>
Signed-off-by: Mark Wielaard <mark@klomp.org>
src/ChangeLog
src/ar.c
tests/ChangeLog
tests/Makefile.am
tests/run-ar-N.sh [new file with mode: 0755]

index 88db40511b26b1318be05372f16515a7c702fdaa..9348c562245da6b4bb862fea6e149af58e982b80 100644 (file)
@@ -1,3 +1,9 @@
+2022-08-28  Mark Wielaard  <mark@klomp.org>
+
+       * ar.c (do_oper_extract): Predecrement instance before compare
+       to zero.
+       (do_oper_delete): Likewise.
+
 2022-08-10  Andreas Schwab  <schwab@suse.de>
 
        * readelf.c (print_attributes): Also handle SHT_RISCV_ATTRIBUTES.
index 9e8df12015b0feaa9672891f75e28e261746b75a..04456c18a03a04d2153f79bb6353e8c01aa613e6 100644 (file)
--- a/src/ar.c
+++ b/src/ar.c
@@ -518,7 +518,7 @@ do_oper_extract (int oper, const char *arfname, char **argv, int argc,
          ENTRY entry;
          entry.key = arhdr->ar_name;
          ENTRY *res = hsearch (entry, FIND);
-         if (res != NULL && (instance < 0 || instance-- == 0)
+         if (res != NULL && (instance < 0 || --instance == 0)
              && !found[(char **) res->data - argv])
            found[(char **) res->data - argv] = do_extract = true;
        }
@@ -952,7 +952,7 @@ do_oper_delete (const char *arfname, char **argv, int argc,
          ENTRY entry;
          entry.key = arhdr->ar_name;
          ENTRY *res = hsearch (entry, FIND);
-         if (res != NULL && (instance < 0 || instance-- == 0)
+         if (res != NULL && (instance < 0 || --instance == 0)
              && !found[(char **) res->data - argv])
            found[(char **) res->data - argv] = do_delete = true;
        }
index 8d87f05d8945fc34c56e81eeec52460ed7fb2f22..f2cc787518c4c9a0adc16f92b1b1ee5ce89779f5 100644 (file)
@@ -1,3 +1,9 @@
+2022-08-26  Mark Wielaard  <mark@klomp.org>
+
+       * run-ar-N.sh: New test.
+       * Makefile.am (TESTS): Add run-ar-N.sh.
+       (EXTRA_DIST): Likewise.
+
 2022-09-02  Frank Ch. Eigler  <fche@redhat.com>
 
        * run-debuginfod-response-headers.sh: Use case-insensitive
index 87988fb9921d872e7717ebf0c833c4f6511e8796..855148983facaa290061cf811efdddba532624dd 100644 (file)
@@ -100,6 +100,7 @@ test-nlist$(EXEEXT): test-nlist.c
          $(test_nlist_CFLAGS) $(GCOV_FLAGS) -o $@ $< $(test_nlist_LDADD)
 
 TESTS = run-arextract.sh run-arsymtest.sh run-ar.sh newfile test-nlist \
+       run-ar-N.sh \
        update1 update2 update3 update4 \
        run-show-die-info.sh run-get-files.sh run-get-lines.sh \
        run-next-files.sh run-next-lines.sh \
@@ -254,6 +255,7 @@ endif
 endif
 
 EXTRA_DIST = run-arextract.sh run-arsymtest.sh run-ar.sh \
+            run-ar-N.sh \
             run-show-die-info.sh run-get-files.sh run-get-lines.sh \
             run-next-files.sh run-next-lines.sh testfile-only-debug-line.bz2 \
             run-get-pubnames.sh run-get-aranges.sh \
diff --git a/tests/run-ar-N.sh b/tests/run-ar-N.sh
new file mode 100755 (executable)
index 0000000..de8f62b
--- /dev/null
@@ -0,0 +1,65 @@
+#! /usr/bin/env bash
+# Copyright (C) 2022 Mark J. Wielaard <mark@klomp.org>
+# This file is part of elfutils.
+#
+# This file 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.
+#
+# elfutils 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, see <http://www.gnu.org/licenses/>.
+
+. $srcdir/test-subr.sh
+
+tempfiles testfile test.ar
+
+echo create test.ar with 3 testfile
+echo 1 > testfile
+testrun ${abs_top_builddir}/src/ar -vr test.ar testfile
+echo 2 > testfile
+testrun ${abs_top_builddir}/src/ar -vq test.ar testfile
+testrun ${abs_top_builddir}/src/ar -t test.ar
+echo 3 > testfile
+testrun ${abs_top_builddir}/src/ar -vq test.ar testfile
+testrun_compare ${abs_top_builddir}/src/ar -t test.ar << EOF
+testfile
+testfile
+testfile
+EOF
+
+echo list content of testfile 1 2 3
+testrun ${abs_top_builddir}/src/ar -vx -N 1 test.ar testfile
+diff -u testfile - << EOF
+1
+EOF
+testrun ${abs_top_builddir}/src/ar -vx -N 2 test.ar testfile
+diff -u testfile - << EOF
+2
+EOF
+testrun ${abs_top_builddir}/src/ar -vx -N 3 test.ar testfile
+diff -u testfile - << EOF
+3
+EOF
+
+echo delete testfile 2
+testrun ${abs_top_builddir}/src/ar -vd -N 2 test.ar testfile
+testrun_compare ${abs_top_builddir}/src/ar -t test.ar << EOF
+testfile
+testfile
+EOF
+testrun ${abs_top_builddir}/src/ar -vx -N 1 test.ar testfile
+diff -u testfile - << EOF
+1
+EOF
+testrun ${abs_top_builddir}/src/ar -vx -N 2 test.ar testfile
+diff -u testfile - << EOF
+3
+EOF
+
+exit 0