Skip to content

Commit 716c1f6

Browse files
peffgitster
authored andcommitted
pipe_command(): mark stdin descriptor as non-blocking
Our pipe_command() helper lets you both write to and read from a child process on its stdin/stdout. It's supposed to work without deadlocks because we use poll() to check when descriptors are ready for reading or writing. But there's a bug: if both the data to be written and the data to be read back exceed the pipe buffer, we'll deadlock. The issue is that the code assumes that if you have, say, a 2MB buffer to write and poll() tells you that the pipe descriptor is ready for writing, that calling: write(cmd->in, buf, 2*1024*1024); will do a partial write, filling the pipe buffer and then returning what it did write. And that is what it would do on a socket, but not for a pipe. When writing to a pipe, at least on Linux, it will block waiting for the child process to read() more. And now we have a potential deadlock, because the child may be writing back to us, waiting for us to read() ourselves. An easy way to trigger this is: git -c add.interactive.useBuiltin=true \ -c interactive.diffFilter=cat \ checkout -p HEAD~200 The diff against HEAD~200 will be big, and the filter wants to write all of it back to us (obviously this is a dummy filter, but in the real world something like diff-highlight would similarly stream back a big output). If you set add.interactive.useBuiltin to false, the problem goes away, because now we're not using pipe_command() anymore (instead, that part happens in perl). But this isn't a bug in the interactive code at all. It's the underlying pipe_command() code which is broken, and has been all along. We presumably didn't notice because most calls only do input _or_ output, not both. And the few that do both, like gpg calls, may have large inputs or outputs, but never both at the same time (e.g., consider signing, which has a large payload but a small signature comes back). The obvious fix is to put the descriptor into non-blocking mode, and indeed, that makes the problem go away. Callers shouldn't need to care, because they never see the descriptor (they hand us a buffer to feed into it). The included test fails reliably on Linux without this patch. Curiously, it doesn't fail in our Windows CI environment, but has been reported to do so for individual developers. It should pass in any environment after this patch (courtesy of the compat/ layers added in the last few commits). Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent c6d3cce commit 716c1f6

File tree

2 files changed

+23
-0
lines changed

2 files changed

+23
-0
lines changed

run-command.c

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
#include "config.h"
1111
#include "packfile.h"
1212
#include "hook.h"
13+
#include "compat/nonblock.h"
1314

1415
void child_process_init(struct child_process *child)
1516
{
@@ -1451,6 +1452,15 @@ int pipe_command(struct child_process *cmd,
14511452
return -1;
14521453

14531454
if (in) {
1455+
if (enable_pipe_nonblock(cmd->in) < 0) {
1456+
error_errno("unable to make pipe non-blocking");
1457+
close(cmd->in);
1458+
if (out)
1459+
close(cmd->out);
1460+
if (err)
1461+
close(cmd->err);
1462+
return -1;
1463+
}
14541464
io[nr].fd = cmd->in;
14551465
io[nr].type = POLLOUT;
14561466
io[nr].u.out.buf = in;

t/t3701-add-interactive.sh

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -766,6 +766,19 @@ test_expect_success 'detect bogus diffFilter output' '
766766
force_color test_must_fail git add -p <y
767767
'
768768

769+
test_expect_success 'handle very large filtered diff' '
770+
git reset --hard &&
771+
# The specific number here is not important, but it must
772+
# be large enough that the output of "git diff --color"
773+
# fills up the pipe buffer. 10,000 results in ~200k of
774+
# colored output.
775+
test_seq 10000 >test &&
776+
test_config interactive.diffFilter cat &&
777+
printf y >y &&
778+
force_color git add -p >output 2>&1 <y &&
779+
git diff-files --exit-code -- test
780+
'
781+
769782
test_expect_success 'diff.algorithm is passed to `git diff-files`' '
770783
git reset --hard &&
771784

0 commit comments

Comments
 (0)