* (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc.
@ 2007-06-23 15:13 Jim Meyering
2007-06-24 9:01 ` Junio C Hamano
0 siblings, 1 reply; 13+ messages in thread
From: Jim Meyering @ 2007-06-23 15:13 UTC (permalink / raw)
To: git
Hi Jun,
Here's a copy of my patch, from here:
http://thread.gmane.org/gmane.comp.version-control.git/48469/focus=48636
Jim
-----------------------------
From 42e3a6f676e9ae4e9640bc2ff36b7ab0b061a60e Mon Sep 17 00:00:00 2001
From: Jim Meyering <jim@meyering.net>
Date: Sat, 26 May 2007 13:43:07 +0200
Subject: [PATCH] Don't ignore write failure from git-diff, git-log, etc.
Currently, when git-diff writes to a full device or gets an I/O error,
it fails to detect the write error:
$ git-diff |wc -c
3984
$ git-diff > /dev/full && echo ignored write failure
ignored write failure
git-log does the same thing:
$ git-log -n1 > /dev/full && echo ignored write failure
ignored write failure
Each and every git command should report such a failure.
Some already do, but with the patch below, they all do, and we
won't have to rely on code in each command's implementation to
perform the right incantation.
$ ./git-log -n1 > /dev/full
fatal: write failure on standard output: No space left on device
[Exit 128]
$ ./git-diff > /dev/full
fatal: write failure on standard output: No space left on device
[Exit 128]
You can demonstrate this with git's own --version output, too:
(but git --help detects the failure without this patch)
$ ./git --version > /dev/full
fatal: write failure on standard output: No space left on device
[Exit 128]
Note that the fcntl test (for whether the fileno may be closed) is
required in order to avoid EBADF upon closing an already-closed stdout,
as would happen for each git command that already closes stdout; I think
update-index was the one I noticed in the failure of t5400, before I
added that test.
Also, to be consistent with e.g., write_or_die, do not
diagnose EPIPE write failures.
Signed-off-by: Jim Meyering <jim@meyering.net>
---
git.c | 19 ++++++++++++++++++-
1 files changed, 18 insertions(+), 1 deletions(-)
diff --git a/git.c b/git.c
index 29b55a1..8258885 100644
--- a/git.c
+++ b/git.c
@@ -308,6 +308,7 @@ static void handle_internal_command(int argc, const char **argv, char **envp)
for (i = 0; i < ARRAY_SIZE(commands); i++) {
struct cmd_struct *p = commands+i;
const char *prefix;
+ int status;
if (strcmp(p->cmd, cmd))
continue;
@@ -321,7 +322,23 @@ static void handle_internal_command(int argc, const char **argv, char **envp)
die("%s must be run in a work tree", cmd);
trace_argv_printf(argv, argc, "trace: built-in: git");
- exit(p->fn(argc, argv, prefix));
+ status = p->fn(argc, argv, prefix);
+
+ /* Close stdout if necessary, and diagnose any failure
+ other than EPIPE. */
+ if (fcntl(fileno (stdout), F_GETFD) >= 0) {
+ errno = 0;
+ if ((ferror(stdout) || fclose(stdout))
+ && errno != EPIPE) {
+ if (errno == 0)
+ die("write failure on standard output");
+ else
+ die("write failure on standard output"
+ ": %s", strerror(errno));
+ }
+ }
+
+ exit(status);
}
}
--
1.5.2.73.g18bece
^ permalink raw reply related [flat|nested] 13+ messages in thread
* Re: (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc.
2007-06-23 15:13 (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc Jim Meyering
@ 2007-06-24 9:01 ` Junio C Hamano
2007-06-24 17:08 ` Linus Torvalds
0 siblings, 1 reply; 13+ messages in thread
From: Junio C Hamano @ 2007-06-24 9:01 UTC (permalink / raw)
To: Jim Meyering; +Cc: git, torvalds
Jim Meyering <jim@meyering.net> writes:
> From: Jim Meyering <jim@meyering.net>
> Date: Sat, 26 May 2007 13:43:07 +0200
> Subject: [PATCH] Don't ignore write failure from git-diff, git-log, etc.
>
> Currently, when git-diff writes to a full device or gets an I/O error,
> it fails to detect the write error:
> ...
> Also, to be consistent with e.g., write_or_die, do not
> diagnose EPIPE write failures.
I still do not like the fact that this patch makes an error from
the final stdout flushing override the return value from p->fn()
even when the function already diagnosed an error, but otherwise
I think it is a good change, as it allows us to catch one error
case that we currently don't, without introducing an annoying
EPIPE diagnosis.
Naks, or vetoes?
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc.
2007-06-24 9:01 ` Junio C Hamano
@ 2007-06-24 17:08 ` Linus Torvalds
2007-06-24 17:10 ` [PATCH 1/2] Clean up internal command handling Linus Torvalds
` (2 more replies)
0 siblings, 3 replies; 13+ messages in thread
From: Linus Torvalds @ 2007-06-24 17:08 UTC (permalink / raw)
To: Junio C Hamano; +Cc: Jim Meyering, git
On Sun, 24 Jun 2007, Junio C Hamano wrote:
>
> I still do not like the fact that this patch makes an error from
> the final stdout flushing override the return value from p->fn()
> even when the function already diagnosed an error
Yeah.
I also don't think it's very _pretty_ code, and it violates my personal
coding standards by adding way too deep indentation for the new error
cases. It was already three indents deep (reasonably fine, but that
NOT_BARE test wass already pretty ugly), but now it becomes five
indentation levels deep at its deepest, which is just a sign that things
should be split up.
I'd also like to know why it does that fcntl() is done, and I also wonder
about that "ferror()" call: it is entirely possible that ferror() is set
due to EPIPE, and in that case, it will *not* set errno to EPIPE at all,
so it will *still* complain about what I consider an invalid situation.
I dunno. I think the ENOSPC worry is a very real and valid one, but I
would really tend prefer something different.
How about this following series of two patches instead, which I'll send as
replies to this email..
Linus
^ permalink raw reply [flat|nested] 13+ messages in thread
* [PATCH 1/2] Clean up internal command handling
2007-06-24 17:08 ` Linus Torvalds
@ 2007-06-24 17:10 ` Linus Torvalds
2007-06-24 17:29 ` [PATCH 2/2] Check for IO errors after running a command Linus Torvalds
2007-06-24 19:13 ` (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc Jim Meyering
2 siblings, 0 replies; 13+ messages in thread
From: Linus Torvalds @ 2007-06-24 17:10 UTC (permalink / raw)
To: Junio C Hamano; +Cc: Jim Meyering, git
This should change no code at all, it just moves the definition of "struct
cmd_struct" out, and then splits out the running of the right command into
the "run_command()" function.
It also removes the long-unused 'envp' pointer passing.
This is just preparation for adding some more error checking.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
---
git.c | 52 ++++++++++++++++++++++++++++++----------------------
1 files changed, 30 insertions(+), 22 deletions(-)
diff --git a/git.c b/git.c
index 29b55a1..6c728e4 100644
--- a/git.c
+++ b/git.c
@@ -216,14 +216,34 @@ const char git_version_string[] = GIT_VERSION;
*/
#define NOT_BARE (1<<2)
-static void handle_internal_command(int argc, const char **argv, char **envp)
+struct cmd_struct {
+ const char *cmd;
+ int (*fn)(int, const char **, const char *);
+ int option;
+};
+
+static NORETURN void run_command(struct cmd_struct *p, int argc, const char **argv)
+{
+ const char *prefix;
+
+ prefix = NULL;
+ if (p->option & RUN_SETUP)
+ prefix = setup_git_directory();
+ if (p->option & USE_PAGER)
+ setup_pager();
+ if (p->option & NOT_BARE) {
+ if (is_bare_repository() || is_inside_git_dir())
+ die("%s must be run in a work tree", p->cmd);
+ }
+ trace_argv_printf(argv, argc, "trace: built-in: git");
+
+ exit(p->fn(argc, argv, prefix));
+}
+
+static void handle_internal_command(int argc, const char **argv)
{
const char *cmd = argv[0];
- static struct cmd_struct {
- const char *cmd;
- int (*fn)(int, const char **, const char *);
- int option;
- } commands[] = {
+ static struct cmd_struct commands[] = {
{ "add", cmd_add, RUN_SETUP | NOT_BARE },
{ "annotate", cmd_annotate, RUN_SETUP | USE_PAGER },
{ "apply", cmd_apply },
@@ -307,25 +327,13 @@ static void handle_internal_command(int argc, const char **argv, char **envp)
for (i = 0; i < ARRAY_SIZE(commands); i++) {
struct cmd_struct *p = commands+i;
- const char *prefix;
if (strcmp(p->cmd, cmd))
continue;
-
- prefix = NULL;
- if (p->option & RUN_SETUP)
- prefix = setup_git_directory();
- if (p->option & USE_PAGER)
- setup_pager();
- if ((p->option & NOT_BARE) &&
- (is_bare_repository() || is_inside_git_dir()))
- die("%s must be run in a work tree", cmd);
- trace_argv_printf(argv, argc, "trace: built-in: git");
-
- exit(p->fn(argc, argv, prefix));
+ run_command(p, argc, argv);
}
}
-int main(int argc, const char **argv, char **envp)
+int main(int argc, const char **argv)
{
const char *cmd = argv[0] ? argv[0] : "git-help";
char *slash = strrchr(cmd, '/');
@@ -358,7 +366,7 @@ int main(int argc, const char **argv, char **envp)
if (!prefixcmp(cmd, "git-")) {
cmd += 4;
argv[0] = cmd;
- handle_internal_command(argc, argv, envp);
+ handle_internal_command(argc, argv);
die("cannot handle %s internally", cmd);
}
@@ -390,7 +398,7 @@ int main(int argc, const char **argv, char **envp)
while (1) {
/* See if it's an internal command */
- handle_internal_command(argc, argv, envp);
+ handle_internal_command(argc, argv);
/* .. then try the external ones */
execv_git_cmd(argv);
^ permalink raw reply related [flat|nested] 13+ messages in thread
* [PATCH 2/2] Check for IO errors after running a command
2007-06-24 17:08 ` Linus Torvalds
2007-06-24 17:10 ` [PATCH 1/2] Clean up internal command handling Linus Torvalds
@ 2007-06-24 17:29 ` Linus Torvalds
2007-06-25 9:44 ` Junio C Hamano
2007-06-25 19:54 ` Jim Meyering
2007-06-24 19:13 ` (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc Jim Meyering
2 siblings, 2 replies; 13+ messages in thread
From: Linus Torvalds @ 2007-06-24 17:29 UTC (permalink / raw)
To: Junio C Hamano; +Cc: Jim Meyering, git
This is trying to implement the strict IO error checks that Jim Meyering
suggested, but explicitly limits it to just regular files. If a pipe gets
closed on us, we shouldn't complain about it.
[ Side note: feel free to change the S_ISREG() to a !S_ISFIFO(). That
would allow easier testing with /dev/full, which tends to be a character
special device. That said, I think sockets and pipes are generally
interchangeable, which is why I limited it to just regular files. ]
If the subcommand already returned an error, that takes precedence (and we
assume that the subcommand already printed out any relevant messages
relating to it)
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
---
Hmm? I'm not saying this is the only way to do this, but I think this is
at least likely to be obviously just an improvement, and it leaves room
for further tweaking of the logic if Jim or others find other cases that
should be handled.
Side note: I think I made a mistake in making the run_command() a NORETURN
function and putting the exit() into it. It's probably better to instead
just make it return "int", and make the caller do
exit(run_command(...));
and that makes it much prettier to have "run_command()" just return early
if an error happens (or doesn't happen).
For example, then we could just do
status = p->fn(...);
if (status)
return status;
/* Somebody closed stdout? */
if (fstat(fileno(stdout), &st))
return 0;
/* Ignore write errors for pipes and sockets.. */
if (S_ISFIFO(st.st_mode) || S_ISSOCK(st.st_mode))
return 0;
which makes it easy to explain what's going on, and avoids having any deep
indentation at all.
I dunno. This passes all the tests, but it's not like we currently test
for ENOSPC/EIO anyway, or even can do that. If we _just_ disable the thing
for pipes/sockets, we could add a test using /dev/full.
I did check that changing it to !S_ISFIFO() gets the right behaviour, and
"git log | head" doesn't complain, while "git log > /dev/full" does. So
this has gotten some very rudimentary testing, but not in the exact form
I'm actually sending it out.
git.c | 15 ++++++++++++++-
1 files changed, 14 insertions(+), 1 deletions(-)
diff --git a/git.c b/git.c
index 6c728e4..db0118a 100644
--- a/git.c
+++ b/git.c
@@ -224,6 +224,8 @@ struct cmd_struct {
static NORETURN void run_command(struct cmd_struct *p, int argc, const char **argv)
{
+ int status;
+ struct stat st;
const char *prefix;
prefix = NULL;
@@ -237,7 +239,18 @@ static NORETURN void run_command(struct cmd_struct *p, int argc, const char **ar
}
trace_argv_printf(argv, argc, "trace: built-in: git");
- exit(p->fn(argc, argv, prefix));
+ status = p->fn(argc, argv, prefix);
+ if (status)
+ exit(status);
+
+ /* Check for ENOSPC and EIO errors.. */
+ if (!fstat(fileno(stdout), &st) && S_ISREG(st.st_mode)) {
+ if (ferror(stdout))
+ die("write failure on standard output");
+ if (fflush(stdout) || fclose(stdout))
+ die("write failure on standard output: %s", strerror(errno));
+ }
+ exit(0);
}
static void handle_internal_command(int argc, const char **argv)
^ permalink raw reply related [flat|nested] 13+ messages in thread
* Re: (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc.
2007-06-24 17:08 ` Linus Torvalds
2007-06-24 17:10 ` [PATCH 1/2] Clean up internal command handling Linus Torvalds
2007-06-24 17:29 ` [PATCH 2/2] Check for IO errors after running a command Linus Torvalds
@ 2007-06-24 19:13 ` Jim Meyering
2 siblings, 0 replies; 13+ messages in thread
From: Jim Meyering @ 2007-06-24 19:13 UTC (permalink / raw)
To: Linus Torvalds; +Cc: Junio C Hamano, git
Linus Torvalds <torvalds@linux-foundation.org> wrote:
...
> I also don't think it's very _pretty_ code, and it violates my personal
> coding standards by adding way too deep indentation for the new error
> cases. It was already three indents deep (reasonably fine, but that
> NOT_BARE test wass already pretty ugly), but now it becomes five
> indentation levels deep at its deepest, which is just a sign that things
> should be split up.
I too disliked the form of my patch, and said so.
> I'd also like to know why it does that fcntl() is done,
In the quoted message, I explained that stdout was already
closed in some cases. The fcntl test avoids the EINVAL you'd
get for closing an already-closed file descriptor.
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-24 17:29 ` [PATCH 2/2] Check for IO errors after running a command Linus Torvalds
@ 2007-06-25 9:44 ` Junio C Hamano
2007-06-25 13:20 ` Johannes Schindelin
` (2 more replies)
2007-06-25 19:54 ` Jim Meyering
1 sibling, 3 replies; 13+ messages in thread
From: Junio C Hamano @ 2007-06-25 9:44 UTC (permalink / raw)
To: Linus Torvalds; +Cc: Jim Meyering, git, Matthias Lederhofer
Linus Torvalds <torvalds@linux-foundation.org> writes:
> Side note: I think I made a mistake in making the run_command() a NORETURN
> function and putting the exit() into it. It's probably better to instead
> just make it return "int", and make the caller do
>
> exit(run_command(...));
>
> and that makes it much prettier to have "run_command()" just return early
> if an error happens (or doesn't happen).
>
> For example, then we could just do
>
> status = p->fn(...);
> if (status)
> return status;
> /* Somebody closed stdout? */
> if (fstat(fileno(stdout), &st))
> return 0;
> /* Ignore write errors for pipes and sockets.. */
> if (S_ISFIFO(st.st_mode) || S_ISSOCK(st.st_mode))
> return 0;
>
> which makes it easy to explain what's going on, and avoids having any deep
> indentation at all.
I took the liberty of munging your two patches to follow your
comments above (it was a perfect guinea-pig opportunity for
Johannes's "rebase -i").
The changes to git.c (run_command) conflicted with GIT_WORK_TREE
changes in a minor way. Matthias, could you sanity check the
result once I push it out to 'next', please?
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-25 9:44 ` Junio C Hamano
@ 2007-06-25 13:20 ` Johannes Schindelin
2007-06-25 13:42 ` Johannes Sixt
2007-06-25 14:01 ` Jim Meyering
2007-06-26 13:33 ` Matthias Lederhofer
2 siblings, 1 reply; 13+ messages in thread
From: Johannes Schindelin @ 2007-06-25 13:20 UTC (permalink / raw)
To: Junio C Hamano; +Cc: git
Hi Junio,
On Mon, 25 Jun 2007, Junio C Hamano wrote:
> I took the liberty of munging your two patches to follow your comments
> above (it was a perfect guinea-pig opportunity for Johannes's "rebase
> -i").
And? How was that experience?
I am actually quite pleased how it worked out in the end; I never
understood the syntax of rebase, and stayed away from it for that reason.
By reworking patch-series into rebase -i, I learnt it on the way, and find
it actually quite useful.
One thing we might consider, however: when rebasing, the current branch
gets updated at each step. Some might consider this a bug, and prefer
rebase to work on a detached HEAD, and only update the branch at the end,
so that <branchname>@{1} refers to the state _before_ rebase.
Thoughts?
Ciao,
Dscho
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-25 13:20 ` Johannes Schindelin
@ 2007-06-25 13:42 ` Johannes Sixt
0 siblings, 0 replies; 13+ messages in thread
From: Johannes Sixt @ 2007-06-25 13:42 UTC (permalink / raw)
To: git
Johannes Schindelin wrote:
> One thing we might consider, however: when rebasing, the current branch
> gets updated at each step. Some might consider this a bug, and prefer
> rebase to work on a detached HEAD, and only update the branch at the end,
> so that <branchname>@{1} refers to the state _before_ rebase.
YESSSS!
Finding the commit before the rebase in the reflog is a nightmare.
-- Hannes
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-25 9:44 ` Junio C Hamano
2007-06-25 13:20 ` Johannes Schindelin
@ 2007-06-25 14:01 ` Jim Meyering
2007-06-25 15:57 ` Linus Torvalds
2007-06-26 13:33 ` Matthias Lederhofer
2 siblings, 1 reply; 13+ messages in thread
From: Jim Meyering @ 2007-06-25 14:01 UTC (permalink / raw)
To: Junio C Hamano; +Cc: Linus Torvalds, git, Matthias Lederhofer
Junio C Hamano <gitster@pobox.com> wrote:
> Linus Torvalds <torvalds@linux-foundation.org> writes:
...
>> For example, then we could just do
>>
>> status = p->fn(...);
>> if (status)
>> return status;
>> /* Somebody closed stdout? */
>> if (fstat(fileno(stdout), &st))
>> return 0;
>> /* Ignore write errors for pipes and sockets.. */
>> if (S_ISFIFO(st.st_mode) || S_ISSOCK(st.st_mode))
>> return 0;
>>
>> which makes it easy to explain what's going on, and avoids having any deep
>> indentation at all.
>
> I took the liberty of munging your two patches to follow your
> comments above
That has the disadvantage of ignoring *all* pipe and socket write errors.
IMHO, git would be better served if it didn't do that, since writing to
those can fail with EIO and even a new one: EACCES (though this latter
is only for sockets). Also possible, according to POSIX: ENOBUFS.
Of course, one can probably argue that those are all unlikely.
They may be even less likely than an actual EPIPE, but the point is
that people and tools using git plumbing should be able to rely on
it to report such write failures, no matter how unusual they are.
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-25 14:01 ` Jim Meyering
@ 2007-06-25 15:57 ` Linus Torvalds
0 siblings, 0 replies; 13+ messages in thread
From: Linus Torvalds @ 2007-06-25 15:57 UTC (permalink / raw)
To: Jim Meyering; +Cc: Junio C Hamano, git, Matthias Lederhofer
On Mon, 25 Jun 2007, Jim Meyering wrote:
>
> That has the disadvantage of ignoring *all* pipe and socket write errors.
.. which is the only wayt to do it.
There's *no*way* to tell what the error was for the case of
if (ferror(stdout))
..
because the original errno has long long since been thrown away.
> Of course, one can probably argue that those are all unlikely.
> They may be even less likely than an actual EPIPE, but the point is
> that people and tools using git plumbing should be able to rely on
> it to report such write failures, no matter how unusual they are.
..but since what you suggest is physically impossible in a half-way
portable manner, and since all relevant such errors are likely to have
happened long before, I would suggest:
- Use my patch
OR
- Stop using stdio.
You *cannot* make stdio error handling sane. It's simply not possible.
The whole point of stdio is to simplify things, but it does so to the
point where portable and reliable error handling is not an option any
more.
Replace all command output that you care about with some stdio
replacement. We do that for all the paths we *really* care about, where
we use raw unistd IO and do our own buffering (ie things like the
"write_in_full()" stuff and "write_sha1_file()" etc.
Anybody who thinks he can handle errors with stdio is simply barking up
the wrogn tree. It can be done, but it can be done only by essentially
making stdio be a really awkward way to do IO, and you're better off with
the raw unistd.h interfaces.
In particular, you need to make sure you check *each* return value of
every single stdio operation. Even then, you don't actually know if
"errno" is set correctly if an error happens in the middle (ie most stdio
routines are just defined to return EOF or nonzero on error, and it's not
at all clear that errno is reliable).
If you don't, a buffer flush error may have happened at any time, and
whatever error code it had has been thrown away, and turned into one
single bit (the "ferror()" thing).
And yes, some libc's might have extensions that actually guarantee more.
But even then I would be *very* surprised if they actually work and have
been tested to any real degree (glibc does not. The stream error is a
single bit. I checked)
So really: if you care about a particular read or write, use the
"write_in_full()" and "read_in_full()" functions. NOTHING ELSE!
Linus
^ permalink raw reply [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-24 17:29 ` [PATCH 2/2] Check for IO errors after running a command Linus Torvalds
2007-06-25 9:44 ` Junio C Hamano
@ 2007-06-25 19:54 ` Jim Meyering
1 sibling, 0 replies; 13+ messages in thread
From: Jim Meyering @ 2007-06-25 19:54 UTC (permalink / raw)
To: Linus Torvalds; +Cc: Junio C Hamano, git
Linus Torvalds <torvalds@linux-foundation.org> wrote:
> This is trying to implement the strict IO error checks that Jim Meyering
> suggested, but explicitly limits it to just regular files. If a pipe gets
> closed on us, we shouldn't complain about it.
...
> Hmm? I'm not saying this is the only way to do this, but I think this is
> at least likely to be obviously just an improvement, and it leaves room
> for further tweaking of the logic if Jim or others find other cases that
> should be handled.
...
> + /* Check for ENOSPC and EIO errors.. */
> + if (!fstat(fileno(stdout), &st) && S_ISREG(st.st_mode)) {
> + if (ferror(stdout))
> + die("write failure on standard output");
> + if (fflush(stdout) || fclose(stdout))
> + die("write failure on standard output: %s", strerror(errno));
> + }
> + exit(0);
Here is a patch relative to "next", to restore some of the
functionality that was provided by my initially-proposed change.
>From 379aee16596d29b83c95068964c349399b9b9f47 Mon Sep 17 00:00:00 2001
From: Jim Meyering <jim@meyering.net>
Date: Mon, 25 Jun 2007 18:54:12 +0200
Subject: [PATCH] When detecting write failure, print strerror when possible.
Do not call "die" solely on the basis of ferror.
Instead, call both ferror and fclose, and *then* decide whether/how to die.
Without this patch, some commands unnecessarily omit strerror(errno)
when they fail:
$ ./git-ls-tree HEAD > /dev/full
fatal: write failure on standard output
With the patch, git reports the desired ENOSPC diagnostic:
fatal: write failure on standard output: No space left on device
FWIW, this version of close_stream is similar to the one
I included in another recent patch, but, is slightly cleaner.
Also, rather than returning zero or EOF, this one simply returns
zero or nonzero.
* git.c (close_stream): New function.
(run_command): Don't die solely because of ferror. Use close_stream.
Signed-off-by: Jim Meyering <jim@meyering.net>
---
git.c | 26 ++++++++++++++++++++++----
1 files changed, 22 insertions(+), 4 deletions(-)
diff --git a/git.c b/git.c
index b949cbb..b00bb1c 100644
--- a/git.c
+++ b/git.c
@@ -246,6 +246,21 @@ struct cmd_struct {
int option;
};
+static int
+close_stream(FILE *stream)
+{
+ int prev_fail = ferror(stream);
+ int fclose_fail = fclose(stream);
+
+ /* If there was a previous failure, but fclose succeeded,
+ clear errno, since ferror does not set it, and its value
+ may be unrelated to the ferror-reported failure. */
+ if (prev_fail && !fclose_fail)
+ errno = 0;
+
+ return prev_fail || fclose_fail;
+}
+
static int run_command(struct cmd_struct *p, int argc, const char **argv)
{
int status;
@@ -274,10 +289,13 @@ static int run_command(struct cmd_struct *p, int argc, const char **argv)
return 0;
/* Check for ENOSPC and EIO errors.. */
- if (ferror(stdout))
- die("write failure on standard output");
- if (fflush(stdout) || fclose(stdout))
- die("write failure on standard output: %s", strerror(errno));
+ if (close_stream(stdout)) {
+ if (errno == 0)
+ die("write failure on standard output");
+ else
+ die("write failure on standard output: %s",
+ strerror(errno));
+ }
return 0;
}
^ permalink raw reply related [flat|nested] 13+ messages in thread
* Re: [PATCH 2/2] Check for IO errors after running a command
2007-06-25 9:44 ` Junio C Hamano
2007-06-25 13:20 ` Johannes Schindelin
2007-06-25 14:01 ` Jim Meyering
@ 2007-06-26 13:33 ` Matthias Lederhofer
2 siblings, 0 replies; 13+ messages in thread
From: Matthias Lederhofer @ 2007-06-26 13:33 UTC (permalink / raw)
To: Junio C Hamano; +Cc: git
Junio C Hamano <gitster@pobox.com> wrote:
> The changes to git.c (run_command) conflicted with GIT_WORK_TREE
> changes in a minor way. Matthias, could you sanity check the
> result once I push it out to 'next', please?
The changes look fine, the tests pass and my own short manual test
passed.
^ permalink raw reply [flat|nested] 13+ messages in thread
end of thread, other threads:[~2007-06-26 13:33 UTC | newest]
Thread overview: 13+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2007-06-23 15:13 (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc Jim Meyering
2007-06-24 9:01 ` Junio C Hamano
2007-06-24 17:08 ` Linus Torvalds
2007-06-24 17:10 ` [PATCH 1/2] Clean up internal command handling Linus Torvalds
2007-06-24 17:29 ` [PATCH 2/2] Check for IO errors after running a command Linus Torvalds
2007-06-25 9:44 ` Junio C Hamano
2007-06-25 13:20 ` Johannes Schindelin
2007-06-25 13:42 ` Johannes Sixt
2007-06-25 14:01 ` Jim Meyering
2007-06-25 15:57 ` Linus Torvalds
2007-06-26 13:33 ` Matthias Lederhofer
2007-06-25 19:54 ` Jim Meyering
2007-06-24 19:13 ` (resend) [PATCH] Don't ignore write failure from git-diff, git-log, etc Jim Meyering
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).