mirror of https://gitee.com/openkylin/qemu.git
qemu-nbd: print error messages from the daemon through a pipe
In order to get nice error messages, keep the qemu-nbd process running until before issuing NBD_DO_IT and connected to the daemon with a pipe. This lets the qemu-nbd process relay error messages from the daemon and exit with a nonzero status if appropriate. Suggested-by: Kevin Wolf <kwolf@redhat.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com> Signed-off-by: Kevin Wolf <kwolf@redhat.com>
This commit is contained in:
parent
a517e88baa
commit
c1f8fdc362
64
qemu-nbd.c
64
qemu-nbd.c
|
@ -226,8 +226,13 @@ static void *nbd_client_thread(void *arg)
|
||||||
/* update partition table */
|
/* update partition table */
|
||||||
pthread_create(&show_parts_thread, NULL, show_parts, NULL);
|
pthread_create(&show_parts_thread, NULL, show_parts, NULL);
|
||||||
|
|
||||||
|
if (verbose) {
|
||||||
fprintf(stderr, "NBD device %s is now connected to %s\n",
|
fprintf(stderr, "NBD device %s is now connected to %s\n",
|
||||||
device, srcpath);
|
device, srcpath);
|
||||||
|
} else {
|
||||||
|
/* Close stderr so that the qemu-nbd process exits. */
|
||||||
|
dup2(STDOUT_FILENO, STDERR_FILENO);
|
||||||
|
}
|
||||||
|
|
||||||
ret = nbd_client(fd);
|
ret = nbd_client(fd);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -406,6 +411,58 @@ int main(int argc, char **argv)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (device && !verbose) {
|
||||||
|
int stderr_fd[2];
|
||||||
|
pid_t pid;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (qemu_pipe(stderr_fd) == -1) {
|
||||||
|
err(EXIT_FAILURE, "Error setting up communication pipe");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now daemonize, but keep a communication channel open to
|
||||||
|
* print errors and exit with the proper status code.
|
||||||
|
*/
|
||||||
|
pid = fork();
|
||||||
|
if (pid == 0) {
|
||||||
|
close(stderr_fd[0]);
|
||||||
|
ret = qemu_daemon(0, 0);
|
||||||
|
|
||||||
|
/* Temporarily redirect stderr to the parent's pipe... */
|
||||||
|
dup2(stderr_fd[1], STDERR_FILENO);
|
||||||
|
if (ret == -1) {
|
||||||
|
err(EXIT_FAILURE, "Failed to daemonize");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ... close the descriptor we inherited and go on. */
|
||||||
|
close(stderr_fd[1]);
|
||||||
|
} else {
|
||||||
|
bool errors = false;
|
||||||
|
char *buf;
|
||||||
|
|
||||||
|
/* In the parent. Print error messages from the child until
|
||||||
|
* it closes the pipe.
|
||||||
|
*/
|
||||||
|
close(stderr_fd[1]);
|
||||||
|
buf = g_malloc(1024);
|
||||||
|
while ((ret = read(stderr_fd[0], buf, 1024)) > 0) {
|
||||||
|
errors = true;
|
||||||
|
ret = qemu_write_full(STDERR_FILENO, buf, ret);
|
||||||
|
if (ret == -1) {
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (ret == -1) {
|
||||||
|
err(EXIT_FAILURE, "Cannot read from daemon");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Usually the daemon should not print any message.
|
||||||
|
* Exit with zero status in that case.
|
||||||
|
*/
|
||||||
|
exit(errors);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bdrv_init();
|
bdrv_init();
|
||||||
|
|
||||||
bs = bdrv_new("hda");
|
bs = bdrv_new("hda");
|
||||||
|
@ -433,13 +490,6 @@ int main(int argc, char **argv)
|
||||||
err(EXIT_FAILURE, "Failed to open %s", device);
|
err(EXIT_FAILURE, "Failed to open %s", device);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!verbose) {
|
|
||||||
/* detach client and server */
|
|
||||||
if (qemu_daemon(0, 0) == -1) {
|
|
||||||
err(EXIT_FAILURE, "Failed to daemonize");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sockpath == NULL) {
|
if (sockpath == NULL) {
|
||||||
sockpath = g_malloc(128);
|
sockpath = g_malloc(128);
|
||||||
snprintf(sockpath, 128, SOCKET_PATH, basename(device));
|
snprintf(sockpath, 128, SOCKET_PATH, basename(device));
|
||||||
|
|
Loading…
Reference in New Issue