diff options
author | Assaf Gordon <assafgordon@gmail.com> | 2015-11-12 05:20:29 -0500 |
---|---|---|
committer | Assaf Gordon <assafgordon@gmail.com> | 2015-11-12 15:17:55 -0500 |
commit | 4c5c6cad85ad67d96df261ec9abdab66590d9ceb (patch) | |
tree | c73bb9ec4a84db74db3c511b19fb6353c5915406 | |
parent | 0daa359de8edd09fe2df76cb26c7cef16da52ce3 (diff) | |
download | coreutils-4c5c6cad85ad67d96df261ec9abdab66590d9ceb.tar.xz |
csplit: check and report fwrite errors with errno
discussed in:
http://lists.gnu.org/archive/html/coreutils/2015-10/msg00091.html
* src/csplit.c: (save_line_to_file): check fwrite failures, report
and exit immediately instead of deferring to 'close_output'.
* tests/misc/csplit-io-err.sh: test fwrite failure using LD_PRELOAD.
* tests/local.mk: add new test.
-rw-r--r-- | src/csplit.c | 8 | ||||
-rw-r--r-- | tests/local.mk | 1 | ||||
-rwxr-xr-x | tests/misc/csplit-io-err.sh | 78 |
3 files changed, 86 insertions, 1 deletions
diff --git a/src/csplit.c b/src/csplit.c index c97acb862..a2034bf77 100644 --- a/src/csplit.c +++ b/src/csplit.c @@ -1051,7 +1051,13 @@ close_output_file (void) static void save_line_to_file (const struct cstring *line) { - fwrite (line->str, sizeof (char), line->len, output_stream); + size_t l = fwrite (line->str, sizeof (char), line->len, output_stream); + if (l != line->len) + { + error (0, errno, _("write error for %s"), quoteaf (output_filename)); + output_stream = NULL; + cleanup_fatal (); + } bytes_written += line->len; } diff --git a/tests/local.mk b/tests/local.mk index 675607e7a..adf96f015 100644 --- a/tests/local.mk +++ b/tests/local.mk @@ -274,6 +274,7 @@ all_tests = \ tests/misc/csplit.sh \ tests/misc/csplit-1000.sh \ tests/misc/csplit-heap.sh \ + tests/misc/csplit-io-err.sh \ tests/misc/csplit-suppress-matched.pl \ tests/misc/date-sec.sh \ tests/misc/dircolors.pl \ diff --git a/tests/misc/csplit-io-err.sh b/tests/misc/csplit-io-err.sh new file mode 100755 index 000000000..6bdd5f143 --- /dev/null +++ b/tests/misc/csplit-io-err.sh @@ -0,0 +1,78 @@ +#!/bin/sh +# Ensure we handle i/o errors correctly in csplit + +# Copyright (C) 2015 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, see <http://www.gnu.org/licenses/>. + +. "${srcdir=.}/tests/init.sh"; path_prepend_ ./src +print_ver_ csplit +require_gcc_shared_ + +if ! test -w /dev/full || ! test -c /dev/full; then + skip_ '/dev/full is required' +fi + +# Ensure error messages are in English +LC_ALL=C +export LC_ALL + +# Replace fwrite and ferror, always returning an error +cat > k.c <<'EOF' || framework_failure_ +#include <stdio.h> +#include <errno.h> + +#undef fwrite +#undef fwrite_unlocked + +size_t +fwrite (const void *ptr, size_t size, size_t nitems, FILE *stream) +{ + fclose (fopen ("preloaded","w")); /* marker for preloaded interception */ + errno = ENOSPC; + return 0; +} + +size_t +fwrite_unlocked (const void *ptr, size_t size, size_t nitems, FILE *stream) +{ + return fwrite (ptr, size, nitems, stream); +} +EOF + +# Get the wording of the OS-dependent ENOSPC message +returns_ 1 seq 1 >/dev/full 2>msgt || framework_failure_ +sed 's/seq: write error: //' msgt > msg || framework_failure_ + +# Create the expected error message +{ printf "%s" "csplit: write error for 'xx01': " ; cat msg ; } > exp \ + || framework_failure_ + +# compile/link the interception shared library: +gcc_shared_ k.c k.so \ + || skip_ 'failed to build forced-fwrite-failure shared library' + +# Split the input, and force fwrite() failure - +# the 'csplit' command should fail with exit code 1 +# (checked with 'returns_ 1 ... || fail=1') +seq 10 \ + | LD_PRELOAD=$LD_PRELOAD:./k.so returns_ 1 csplit - 1 4 2>out \ + || fail=1 + +test -e preloaded || skip_ 'LD_PRELOAD interception failed' + +# Ensure we got the expected error message +compare exp out || fail=1 + +Exit $fail |