summary | shortlog | log | commit | commitdiff | tree
raw | patch | inline | side by side (parent: 23d6d11)
raw | patch | inline | side by side (parent: 23d6d11)
author | Junio C Hamano <junkio@cox.net> | |
Tue, 12 Sep 2006 07:26:57 +0000 (00:26 -0700) | ||
committer | Junio C Hamano <junkio@cox.net> | |
Wed, 13 Sep 2006 05:39:45 +0000 (22:39 -0700) |
Franck noticed that the code around polling and relaying messages
from the child process was quite bogus. Here is an attempt to
clean it up a bit, based on his patch:
- When POLLHUP is set, it goes ahead and reads the file
descriptor. Worse yet, it does not check the return value of
read() for errors when it does.
- When we processed one POLLIN, we should just go back and see
if any more data is available. We can check if the child is
still there when poll gave control back at us but without any
actual input.
[jc: with simplification suggested by Franck. ]
Signed-off-by: Junio C Hamano <junkio@cox.net>
from the child process was quite bogus. Here is an attempt to
clean it up a bit, based on his patch:
- When POLLHUP is set, it goes ahead and reads the file
descriptor. Worse yet, it does not check the return value of
read() for errors when it does.
- When we processed one POLLIN, we should just go back and see
if any more data is available. We can check if the child is
still there when poll gave control back at us but without any
actual input.
[jc: with simplification suggested by Franck. ]
Signed-off-by: Junio C Hamano <junkio@cox.net>
builtin-upload-archive.c | patch | blob | history |
index 42cb9f8876b5c1ad870b9438e5525d52c6929cf0..115a12dc13a09e9bfd4c965ca632709e0f1cd89b 100644 (file)
--- a/builtin-upload-archive.c
+++ b/builtin-upload-archive.c
static const char deadchild[] =
"git-upload-archive: archiver died with error";
+static const char lostchild[] =
+"git-upload-archive: archiver process was lost";
+
static int run_upload_archive(int argc, const char **argv, const char *prefix)
{
return ar.write_archive(&ar.args);
}
+static void error_clnt(const char *fmt, ...)
+{
+ char buf[1024];
+ va_list params;
+ int len;
+
+ va_start(params, fmt);
+ len = vsprintf(buf, fmt, params);
+ va_end(params);
+ send_sideband(1, 3, buf, len, LARGE_PACKET_MAX);
+ die("sent error to the client: %s", buf);
+}
+
+static void process_input(int child_fd, int band)
+{
+ char buf[16384];
+ ssize_t sz = read(child_fd, buf, sizeof(buf));
+ if (sz < 0) {
+ if (errno != EINTR)
+ error_clnt("read error: %s\n", strerror(errno));
+ return;
+ }
+ send_sideband(1, band, buf, sz, LARGE_PACKET_MAX);
+}
+
int cmd_upload_archive(int argc, const char **argv, const char *prefix)
{
pid_t writer;
while (1) {
struct pollfd pfd[2];
- char buf[16384];
- ssize_t sz;
- pid_t pid;
int status;
pfd[0].fd = fd1[0];
}
continue;
}
- if (pfd[0].revents & (POLLIN|POLLHUP)) {
+ if (pfd[0].revents & POLLIN)
/* Data stream ready */
- sz = read(pfd[0].fd, buf, sizeof(buf));
- send_sideband(1, 1, buf, sz, LARGE_PACKET_MAX);
- }
- if (pfd[1].revents & (POLLIN|POLLHUP)) {
+ process_input(pfd[0].fd, 1);
+ if (pfd[1].revents & POLLIN)
/* Status stream ready */
- sz = read(pfd[1].fd, buf, sizeof(buf));
- send_sideband(1, 2, buf, sz, LARGE_PACKET_MAX);
- }
-
- if (((pfd[0].revents | pfd[1].revents) & POLLHUP) == 0)
- continue;
- /* did it die? */
- pid = waitpid(writer, &status, WNOHANG);
- if (!pid) {
- fprintf(stderr, "Hmph, HUP?\n");
+ process_input(pfd[1].fd, 2);
+ if ((pfd[0].revents | pfd[1].revents) == POLLIN)
continue;
- }
- if (!WIFEXITED(status) || WEXITSTATUS(status) > 0)
- send_sideband(1, 3, deadchild, strlen(deadchild),
- LARGE_PACKET_MAX);
+
+ if (waitpid(writer, &status, 0) < 0)
+ error_clnt("%s", lostchild);
+ else if (!WIFEXITED(status) || WEXITSTATUS(status) > 0)
+ error_clnt("%s", deadchild);
packet_flush(1);
break;
}