提交 4296cea2 编写于 作者: D Daniel P. Berrange

Don't let parent of daemon exit until basic initialization is done

The daemonizing code lets the parent exit almost immediately. This
means that it may think it has successfully started even when
important failures occur like not being able to acquire the PID
file. It also means network sockets are not yet open.

To address this when daemonizing the parent passes an open pipe
file descriptor to the child. The child does its basic initialization
and then writes a status code to the pipe indicating either success,
or failure. This ensures that when daemonizing, the parent does not
exit until the pidfile is acquired & basic network sockets are open.

Initialization of the libvirt drivers is still done asynchronously
since this may take a very long time.

* daemon/libvirtd.c: Force parent to stay around until basic config
  file, pidfile & network socket init is completed
上级 a71f79c3
...@@ -185,6 +185,30 @@ static int max_client_requests = 5; ...@@ -185,6 +185,30 @@ static int max_client_requests = 5;
static sig_atomic_t sig_errors = 0; static sig_atomic_t sig_errors = 0;
static int sig_lasterrno = 0; static int sig_lasterrno = 0;
enum {
VIR_DAEMON_ERR_NONE = 0,
VIR_DAEMON_ERR_PIDFILE,
VIR_DAEMON_ERR_RUNDIR,
VIR_DAEMON_ERR_INIT,
VIR_DAEMON_ERR_SIGNAL,
VIR_DAEMON_ERR_PRIVS,
VIR_DAEMON_ERR_NETWORK,
VIR_DAEMON_ERR_CONFIG,
VIR_DAEMON_ERR_LAST
};
VIR_ENUM_DECL(virDaemonErr)
VIR_ENUM_IMPL(virDaemonErr, VIR_DAEMON_ERR_LAST,
"Initialization successful",
"Unable to obtain pidfile",
"Unable to create rundir",
"Unable to initialize libvirt",
"Unable to setup signal handlers",
"Unable to drop privileges",
"Unable to initialize network sockets",
"Unable to load configuration file")
static void sig_handler(int sig, siginfo_t * siginfo, static void sig_handler(int sig, siginfo_t * siginfo,
void* context ATTRIBUTE_UNUSED) { void* context ATTRIBUTE_UNUSED) {
int origerrno; int origerrno;
...@@ -375,7 +399,11 @@ qemudDispatchSignalEvent(int watch ATTRIBUTE_UNUSED, ...@@ -375,7 +399,11 @@ qemudDispatchSignalEvent(int watch ATTRIBUTE_UNUSED,
} }
static int qemudGoDaemon(void) { static int daemonForkIntoBackground(void) {
int statuspipe[2];
if (pipe(statuspipe) < 0)
return -1;
int pid = fork(); int pid = fork();
switch (pid) { switch (pid) {
case 0: case 0:
...@@ -384,6 +412,8 @@ static int qemudGoDaemon(void) { ...@@ -384,6 +412,8 @@ static int qemudGoDaemon(void) {
int stdoutfd = -1; int stdoutfd = -1;
int nextpid; int nextpid;
close(statuspipe[0]);
if ((stdinfd = open("/dev/null", O_RDONLY)) < 0) if ((stdinfd = open("/dev/null", O_RDONLY)) < 0)
goto cleanup; goto cleanup;
if ((stdoutfd = open("/dev/null", O_WRONLY)) < 0) if ((stdoutfd = open("/dev/null", O_WRONLY)) < 0)
...@@ -407,7 +437,7 @@ static int qemudGoDaemon(void) { ...@@ -407,7 +437,7 @@ static int qemudGoDaemon(void) {
nextpid = fork(); nextpid = fork();
switch (nextpid) { switch (nextpid) {
case 0: case 0:
return 0; return statuspipe[1];
case -1: case -1:
return -1; return -1;
default: default:
...@@ -428,15 +458,29 @@ static int qemudGoDaemon(void) { ...@@ -428,15 +458,29 @@ static int qemudGoDaemon(void) {
default: default:
{ {
int got, status = 0; int got, exitstatus = 0;
/* We wait to make sure the next child forked int ret;
successfully */ char status;
if ((got = waitpid(pid, &status, 0)) < 0 ||
close(statuspipe[1]);
/* We wait to make sure the first child forked successfully */
if ((got = waitpid(pid, &exitstatus, 0)) < 0 ||
got != pid || got != pid ||
status != 0) { exitstatus != 0) {
return -1; return -1;
} }
_exit(0);
/* Now block until the second child initializes successfully */
again:
ret = read(statuspipe[0], &status, 1);
if (ret == -1 && errno == EINTR)
goto again;
if (ret == 1 && status != 0) {
fprintf(stderr, "error: %s\n", virDaemonErrTypeToString(status));
}
_exit(ret == 1 && status == 0 ? 0 : 1);
} }
} }
} }
...@@ -871,8 +915,6 @@ static struct qemud_server *qemudInitialize(void) { ...@@ -871,8 +915,6 @@ static struct qemud_server *qemudInitialize(void) {
virEventUpdateTimeoutImpl, virEventUpdateTimeoutImpl,
virEventRemoveTimeoutImpl); virEventRemoveTimeoutImpl);
virStateInitialize(server->privileged);
return server; return server;
} }
...@@ -2910,6 +2952,7 @@ int main(int argc, char **argv) { ...@@ -2910,6 +2952,7 @@ int main(int argc, char **argv) {
struct qemud_server *server = NULL; struct qemud_server *server = NULL;
const char *pid_file = NULL; const char *pid_file = NULL;
const char *remote_config_file = NULL; const char *remote_config_file = NULL;
int statuswrite = -1;
int ret = 1; int ret = 1;
struct option opts[] = { struct option opts[] = {
...@@ -2993,7 +3036,7 @@ int main(int argc, char **argv) { ...@@ -2993,7 +3036,7 @@ int main(int argc, char **argv) {
if (godaemon) { if (godaemon) {
char ebuf[1024]; char ebuf[1024];
if (qemudGoDaemon() < 0) { if ((statuswrite = daemonForkIntoBackground()) < 0) {
VIR_ERROR(_("Failed to fork as daemon: %s"), VIR_ERROR(_("Failed to fork as daemon: %s"),
virStrerror(errno, ebuf, sizeof ebuf)); virStrerror(errno, ebuf, sizeof ebuf));
goto error; goto error;
...@@ -3008,8 +3051,11 @@ int main(int argc, char **argv) { ...@@ -3008,8 +3051,11 @@ int main(int argc, char **argv) {
/* If we have a pidfile set, claim it now, exiting if already taken */ /* If we have a pidfile set, claim it now, exiting if already taken */
if (pid_file != NULL && if (pid_file != NULL &&
qemudWritePidFile (pid_file) < 0) qemudWritePidFile (pid_file) < 0) {
pid_file = NULL; /* Prevent unlinking of someone else's pid ! */
ret = VIR_DAEMON_ERR_PIDFILE;
goto error; goto error;
}
/* Ensure the rundir exists (on tmpfs on some systems) */ /* Ensure the rundir exists (on tmpfs on some systems) */
if (geteuid() == 0) { if (geteuid() == 0) {
...@@ -3020,7 +3066,8 @@ int main(int argc, char **argv) { ...@@ -3020,7 +3066,8 @@ int main(int argc, char **argv) {
char ebuf[1024]; char ebuf[1024];
VIR_ERROR(_("unable to create rundir %s: %s"), rundir, VIR_ERROR(_("unable to create rundir %s: %s"), rundir,
virStrerror(errno, ebuf, sizeof(ebuf))); virStrerror(errno, ebuf, sizeof(ebuf)));
return -1; ret = VIR_DAEMON_ERR_RUNDIR;
goto error;
} }
} }
} }
...@@ -3031,34 +3078,71 @@ int main(int argc, char **argv) { ...@@ -3031,34 +3078,71 @@ int main(int argc, char **argv) {
* which is also passed into all libvirt stateful * which is also passed into all libvirt stateful
* drivers * drivers
*/ */
if (qemudSetupPrivs() < 0) if (qemudSetupPrivs() < 0) {
ret = VIR_DAEMON_ERR_PRIVS;
goto error; goto error;
}
if (!(server = qemudInitialize())) { if (!(server = qemudInitialize())) {
ret = 2; ret = VIR_DAEMON_ERR_INIT;
goto error; goto error;
} }
if ((daemonSetupSignals(server)) < 0) if ((daemonSetupSignals(server)) < 0) {
ret = VIR_DAEMON_ERR_SIGNAL;
goto error; goto error;
}
/* Read the config file (if it exists). */ /* Read the config file (if it exists). */
if (remoteReadConfigFile (server, remote_config_file) < 0) if (remoteReadConfigFile (server, remote_config_file) < 0) {
ret = VIR_DAEMON_ERR_CONFIG;
goto error; goto error;
}
/* Disable error func, now logging is setup */ /* Disable error func, now logging is setup */
virSetErrorFunc(NULL, virshErrorHandler); virSetErrorFunc(NULL, virshErrorHandler);
if (qemudNetworkInit(server) < 0) { if (qemudNetworkInit(server) < 0) {
ret = 2; ret = VIR_DAEMON_ERR_NETWORK;
goto error; goto error;
} }
qemudRunLoop(server); /* Tell parent of daemon that basic initialization is complete
* In particular we're ready to accept net connections & have
* written the pidfile
*/
if (statuswrite != -1) {
char status = 0;
while (write(statuswrite, &status, 1) == -1 &&
errno == EINTR)
;
close(statuswrite);
statuswrite = -1;
}
/* Start the stateful HV drivers
* This is delibrately done after telling the parent process
* we're ready, since it can take a long time and this will
* seriously delay OS bootup process */
if (virStateInitialize(server->privileged) < 0) {
VIR_ERROR0("Driver state initialization failed");
goto error;
}
qemudRunLoop(server);
ret = 0; ret = 0;
error: error:
if (statuswrite != -1) {
if (ret != 0) {
/* Tell parent of daemon what failed */
char status = ret;
while (write(statuswrite, &status, 1) == -1 &&
errno == EINTR)
;
}
close(statuswrite);
}
if (server) if (server)
qemudCleanup(server); qemudCleanup(server);
if (pid_file) if (pid_file)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册