2007-02-14 01:40:09 +00:00
|
|
|
/*
|
2009-09-16 11:37:26 +00:00
|
|
|
* libvirtd.c: daemon start of day, guest process & i/o management
|
2007-02-14 01:40:09 +00:00
|
|
|
*
|
build: use correct type for pid and similar types
No thanks to 64-bit windows, with 64-bit pid_t, we have to avoid
constructs like 'int pid'. Our API in libvirt-qemu cannot be
changed without breaking ABI; but then again, libvirt-qemu can
only be used on systems that support UNIX sockets, which rules
out Windows (even if qemu could be compiled there) - so for all
points on the call chain that interact with this API decision,
we require a different variable name to make it clear that we
audited the use for safety.
Adding a syntax-check rule only solves half the battle; anywhere
that uses printf on a pid_t still needs to be converted, but that
will be a separate patch.
* cfg.mk (sc_correct_id_types): New syntax check.
* src/libvirt-qemu.c (virDomainQemuAttach): Document why we didn't
use pid_t for pid, and validate for overflow.
* include/libvirt/libvirt-qemu.h (virDomainQemuAttach): Tweak name
for syntax check.
* src/vmware/vmware_conf.c (vmwareExtractPid): Likewise.
* src/driver.h (virDrvDomainQemuAttach): Likewise.
* tools/virsh.c (cmdQemuAttach): Likewise.
* src/remote/qemu_protocol.x (qemu_domain_attach_args): Likewise.
* src/qemu_protocol-structs (qemu_domain_attach_args): Likewise.
* src/util/cgroup.c (virCgroupPidCode, virCgroupKillInternal):
Likewise.
* src/qemu/qemu_command.c(qemuParseProcFileStrings): Likewise.
(qemuParseCommandLinePid): Use pid_t for pid.
* daemon/libvirtd.c (daemonForkIntoBackground): Likewise.
* src/conf/domain_conf.h (_virDomainObj): Likewise.
* src/probes.d (rpc_socket_new): Likewise.
* src/qemu/qemu_command.h (qemuParseCommandLinePid): Likewise.
* src/qemu/qemu_driver.c (qemudGetProcessInfo, qemuDomainAttach):
Likewise.
* src/qemu/qemu_process.c (qemuProcessAttach): Likewise.
* src/qemu/qemu_process.h (qemuProcessAttach): Likewise.
* src/uml/uml_driver.c (umlGetProcessInfo): Likewise.
* src/util/virnetdev.h (virNetDevSetNamespace): Likewise.
* src/util/virnetdev.c (virNetDevSetNamespace): Likewise.
* tests/testutils.c (virtTestCaptureProgramOutput): Likewise.
* src/conf/storage_conf.h (_virStoragePerms): Use mode_t, uid_t,
and gid_t rather than int.
* src/security/security_dac.c (virSecurityDACSetOwnership): Likewise.
* src/conf/storage_conf.c (virStorageDefParsePerms): Avoid
compiler warning.
2012-02-10 23:08:11 +00:00
|
|
|
* Copyright (C) 2006-2012 Red Hat, Inc.
|
2007-02-14 01:40:09 +00:00
|
|
|
* Copyright (C) 2006 Daniel P. Berrange
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2012-09-20 22:30:55 +00:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 10:06:23 +00:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2007-02-14 01:40:09 +00:00
|
|
|
*
|
|
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
|
|
*/
|
|
|
|
|
2008-01-29 18:15:54 +00:00
|
|
|
#include <config.h>
|
2007-02-14 15:42:55 +00:00
|
|
|
|
2007-02-14 01:40:09 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2011-05-16 17:13:11 +00:00
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <sys/stat.h>
|
2007-02-14 01:40:09 +00:00
|
|
|
#include <getopt.h>
|
2011-05-16 17:13:11 +00:00
|
|
|
#include <stdlib.h>
|
2007-09-19 02:28:01 +00:00
|
|
|
#include <grp.h>
|
2010-11-16 19:01:37 +00:00
|
|
|
#include <locale.h>
|
2007-04-10 23:17:46 +00:00
|
|
|
|
2008-11-04 23:22:06 +00:00
|
|
|
#include "libvirt_internal.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2011-07-19 18:32:58 +00:00
|
|
|
#include "virfile.h"
|
2011-08-05 14:11:11 +00:00
|
|
|
#include "virpidfile.h"
|
2012-09-24 17:10:37 +00:00
|
|
|
#include "virprocess.h"
|
Standardize use of header files, making internal.h primary.
* qemud/internal.h, qemud/qemud.h: Rename this file so it
doesn't conflict with src/internal.h.
* HACKING: Document how header files should be used.
* qemud/Makefile.am: Add src/ directory to includes.
* qemud/event.c, qemud/mdns.c, qemud/qemud.c, qemud/remote.c,
qemud/remote_protocol.c, qemud/remote_protocol.h,
qemud/remote_protocol.x, src/buf.c, src/libvirt.c,
src/nodeinfo.c, src/qemu_conf.c, src/qemu_driver.c,
src/stats_linux.c, src/storage_backend.c, src/storage_backend_fs.c,
src/storage_backend_iscsi.c, src/storage_backend_logical.c,
src/storage_conf.c, src/storage_driver.c, src/util.c,
src/util.h, src/virsh.c, src/virterror.c, src/xend_internal.c,
src/xml.c, tests/reconnect.c, tests/xmlrpctest.c,
tests/qparamtest.c: Standardize use of header files.
* docs/*, po/*: Rebuild docs.
2008-05-23 08:24:41 +00:00
|
|
|
|
2009-02-09 17:52:38 +00:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_QEMU
|
|
|
|
|
2009-09-16 11:37:26 +00:00
|
|
|
#include "libvirtd.h"
|
2012-04-04 12:09:09 +00:00
|
|
|
#include "libvirtd-config.h"
|
2009-07-10 11:20:03 +00:00
|
|
|
|
2012-12-13 18:01:25 +00:00
|
|
|
#include "viruuid.h"
|
2009-09-16 15:55:16 +00:00
|
|
|
#include "remote_driver.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-12 16:35:35 +00:00
|
|
|
#include "virconf.h"
|
2012-02-22 13:17:13 +00:00
|
|
|
#include "virnetlink.h"
|
2011-05-16 17:13:11 +00:00
|
|
|
#include "virnetserver.h"
|
|
|
|
#include "remote.h"
|
2012-12-12 17:00:34 +00:00
|
|
|
#include "virhook.h"
|
2011-07-11 19:42:15 +00:00
|
|
|
#include "viraudit.h"
|
2013-04-03 10:36:23 +00:00
|
|
|
#include "virstring.h"
|
2013-04-17 11:01:24 +00:00
|
|
|
#include "locking/lock_manager.h"
|
|
|
|
#include "viraccessmanager.h"
|
2007-02-14 01:40:09 +00:00
|
|
|
|
2008-11-21 12:16:08 +00:00
|
|
|
#ifdef WITH_DRIVER_MODULES
|
2010-03-09 18:22:22 +00:00
|
|
|
# include "driver.h"
|
2008-11-21 12:16:08 +00:00
|
|
|
#else
|
2010-03-09 18:22:22 +00:00
|
|
|
# ifdef WITH_QEMU
|
|
|
|
# include "qemu/qemu_driver.h"
|
|
|
|
# endif
|
|
|
|
# ifdef WITH_LXC
|
|
|
|
# include "lxc/lxc_driver.h"
|
|
|
|
# endif
|
2012-08-30 18:53:02 +00:00
|
|
|
# ifdef WITH_XEN
|
|
|
|
# include "xen/xen_driver.h"
|
|
|
|
# endif
|
2011-02-10 22:42:34 +00:00
|
|
|
# ifdef WITH_LIBXL
|
|
|
|
# include "libxl/libxl_driver.h"
|
|
|
|
# endif
|
2010-03-09 18:22:22 +00:00
|
|
|
# ifdef WITH_UML
|
|
|
|
# include "uml/uml_driver.h"
|
|
|
|
# endif
|
2013-05-15 10:31:36 +00:00
|
|
|
# ifdef WITH_VBOX
|
|
|
|
# include "vbox/vbox_driver.h"
|
|
|
|
# endif
|
2010-03-09 18:22:22 +00:00
|
|
|
# ifdef WITH_NETWORK
|
|
|
|
# include "network/bridge_driver.h"
|
|
|
|
# endif
|
2012-09-18 01:27:06 +00:00
|
|
|
# ifdef WITH_INTERFACE
|
|
|
|
# include "interface/interface_driver.h"
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
2012-06-05 16:28:52 +00:00
|
|
|
# ifdef WITH_STORAGE
|
2010-03-09 18:22:22 +00:00
|
|
|
# include "storage/storage_driver.h"
|
|
|
|
# endif
|
|
|
|
# ifdef WITH_NODE_DEVICES
|
|
|
|
# include "node_device/node_device_driver.h"
|
|
|
|
# endif
|
|
|
|
# ifdef WITH_SECRETS
|
|
|
|
# include "secret/secret_driver.h"
|
|
|
|
# endif
|
2010-03-25 17:46:09 +00:00
|
|
|
# ifdef WITH_NWFILTER
|
|
|
|
# include "nwfilter/nwfilter_driver.h"
|
|
|
|
# endif
|
2009-12-22 13:50:50 +00:00
|
|
|
#endif
|
2008-11-17 12:18:18 +00:00
|
|
|
|
2010-11-16 14:54:17 +00:00
|
|
|
#include "configmake.h"
|
|
|
|
|
2012-10-31 19:03:57 +00:00
|
|
|
#include "virdbus.h"
|
|
|
|
|
2012-09-20 11:58:29 +00:00
|
|
|
#if WITH_SASL
|
2011-05-16 17:13:11 +00:00
|
|
|
virNetSASLContextPtr saslCtxt = NULL;
|
2011-06-30 17:18:08 +00:00
|
|
|
#endif
|
2011-05-16 17:13:11 +00:00
|
|
|
virNetServerProgramPtr remoteProgram = NULL;
|
|
|
|
virNetServerProgramPtr qemuProgram = NULL;
|
2012-12-21 14:20:04 +00:00
|
|
|
virNetServerProgramPtr lxcProgram = NULL;
|
2007-06-11 12:04:54 +00:00
|
|
|
|
2009-10-16 11:14:54 +00:00
|
|
|
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,
|
2010-03-26 14:53:32 +00:00
|
|
|
VIR_DAEMON_ERR_HOOKS,
|
2010-09-15 13:44:11 +00:00
|
|
|
VIR_DAEMON_ERR_AUDIT,
|
2009-10-16 11:14:54 +00:00
|
|
|
|
|
|
|
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",
|
2010-03-26 14:53:32 +00:00
|
|
|
"Unable to load configuration file",
|
2010-09-15 13:44:11 +00:00
|
|
|
"Unable to look for hook scripts",
|
|
|
|
"Unable to initialize audit system")
|
2009-10-16 11:14:54 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static int daemonForkIntoBackground(const char *argv0)
|
2007-06-11 12:04:54 +00:00
|
|
|
{
|
2009-10-16 11:14:54 +00:00
|
|
|
int statuspipe[2];
|
|
|
|
if (pipe(statuspipe) < 0)
|
|
|
|
return -1;
|
|
|
|
|
build: use correct type for pid and similar types
No thanks to 64-bit windows, with 64-bit pid_t, we have to avoid
constructs like 'int pid'. Our API in libvirt-qemu cannot be
changed without breaking ABI; but then again, libvirt-qemu can
only be used on systems that support UNIX sockets, which rules
out Windows (even if qemu could be compiled there) - so for all
points on the call chain that interact with this API decision,
we require a different variable name to make it clear that we
audited the use for safety.
Adding a syntax-check rule only solves half the battle; anywhere
that uses printf on a pid_t still needs to be converted, but that
will be a separate patch.
* cfg.mk (sc_correct_id_types): New syntax check.
* src/libvirt-qemu.c (virDomainQemuAttach): Document why we didn't
use pid_t for pid, and validate for overflow.
* include/libvirt/libvirt-qemu.h (virDomainQemuAttach): Tweak name
for syntax check.
* src/vmware/vmware_conf.c (vmwareExtractPid): Likewise.
* src/driver.h (virDrvDomainQemuAttach): Likewise.
* tools/virsh.c (cmdQemuAttach): Likewise.
* src/remote/qemu_protocol.x (qemu_domain_attach_args): Likewise.
* src/qemu_protocol-structs (qemu_domain_attach_args): Likewise.
* src/util/cgroup.c (virCgroupPidCode, virCgroupKillInternal):
Likewise.
* src/qemu/qemu_command.c(qemuParseProcFileStrings): Likewise.
(qemuParseCommandLinePid): Use pid_t for pid.
* daemon/libvirtd.c (daemonForkIntoBackground): Likewise.
* src/conf/domain_conf.h (_virDomainObj): Likewise.
* src/probes.d (rpc_socket_new): Likewise.
* src/qemu/qemu_command.h (qemuParseCommandLinePid): Likewise.
* src/qemu/qemu_driver.c (qemudGetProcessInfo, qemuDomainAttach):
Likewise.
* src/qemu/qemu_process.c (qemuProcessAttach): Likewise.
* src/qemu/qemu_process.h (qemuProcessAttach): Likewise.
* src/uml/uml_driver.c (umlGetProcessInfo): Likewise.
* src/util/virnetdev.h (virNetDevSetNamespace): Likewise.
* src/util/virnetdev.c (virNetDevSetNamespace): Likewise.
* tests/testutils.c (virtTestCaptureProgramOutput): Likewise.
* src/conf/storage_conf.h (_virStoragePerms): Use mode_t, uid_t,
and gid_t rather than int.
* src/security/security_dac.c (virSecurityDACSetOwnership): Likewise.
* src/conf/storage_conf.c (virStorageDefParsePerms): Avoid
compiler warning.
2012-02-10 23:08:11 +00:00
|
|
|
pid_t pid = fork();
|
2007-02-14 01:40:09 +00:00
|
|
|
switch (pid) {
|
|
|
|
case 0:
|
|
|
|
{
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
/* intermediate child */
|
2007-02-14 01:40:09 +00:00
|
|
|
int stdinfd = -1;
|
|
|
|
int stdoutfd = -1;
|
2007-02-16 18:26:18 +00:00
|
|
|
int nextpid;
|
2007-02-14 01:40:09 +00:00
|
|
|
|
2010-11-09 20:48:48 +00:00
|
|
|
VIR_FORCE_CLOSE(statuspipe[0]);
|
2009-10-16 11:14:54 +00:00
|
|
|
|
2008-12-17 18:04:55 +00:00
|
|
|
if ((stdinfd = open("/dev/null", O_RDONLY)) < 0)
|
2007-02-14 01:40:09 +00:00
|
|
|
goto cleanup;
|
2008-12-17 18:04:55 +00:00
|
|
|
if ((stdoutfd = open("/dev/null", O_WRONLY)) < 0)
|
2007-02-14 01:40:09 +00:00
|
|
|
goto cleanup;
|
|
|
|
if (dup2(stdinfd, STDIN_FILENO) != STDIN_FILENO)
|
|
|
|
goto cleanup;
|
|
|
|
if (dup2(stdoutfd, STDOUT_FILENO) != STDOUT_FILENO)
|
|
|
|
goto cleanup;
|
|
|
|
if (dup2(stdoutfd, STDERR_FILENO) != STDERR_FILENO)
|
|
|
|
goto cleanup;
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
if (stdinfd > STDERR_FILENO && VIR_CLOSE(stdinfd) < 0)
|
2007-02-14 01:40:09 +00:00
|
|
|
goto cleanup;
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
if (stdoutfd > STDERR_FILENO && VIR_CLOSE(stdoutfd) < 0)
|
2007-02-14 01:40:09 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (setsid() < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
nextpid = fork();
|
|
|
|
switch (nextpid) {
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
case 0: /* grandchild */
|
2009-10-16 11:14:54 +00:00
|
|
|
return statuspipe[1];
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
case -1: /* error */
|
|
|
|
goto cleanup;
|
|
|
|
default: /* intermediate child succeeded */
|
|
|
|
_exit(EXIT_SUCCESS);
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-11-09 20:48:48 +00:00
|
|
|
VIR_FORCE_CLOSE(stdoutfd);
|
|
|
|
VIR_FORCE_CLOSE(stdinfd);
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
VIR_FORCE_CLOSE(statuspipe[1]);
|
|
|
|
_exit(EXIT_FAILURE);
|
2007-02-14 01:40:09 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
case -1: /* error in parent */
|
|
|
|
goto error;
|
2007-02-14 01:40:09 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
/* parent */
|
2009-10-16 11:14:54 +00:00
|
|
|
int ret;
|
|
|
|
char status;
|
|
|
|
|
2010-11-09 20:48:48 +00:00
|
|
|
VIR_FORCE_CLOSE(statuspipe[1]);
|
2009-10-16 11:14:54 +00:00
|
|
|
|
|
|
|
/* We wait to make sure the first child forked successfully */
|
2012-09-24 16:59:31 +00:00
|
|
|
if (virProcessWait(pid, NULL) < 0)
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
goto error;
|
2009-10-16 11:14:54 +00:00
|
|
|
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
/* If we get here, then the grandchild was spawned, so we
|
|
|
|
* must exit. Block until the second child initializes
|
|
|
|
* successfully */
|
2009-10-16 11:14:54 +00:00
|
|
|
again:
|
|
|
|
ret = read(statuspipe[0], &status, 1);
|
|
|
|
if (ret == -1 && errno == EINTR)
|
|
|
|
goto again;
|
|
|
|
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
VIR_FORCE_CLOSE(statuspipe[0]);
|
|
|
|
|
|
|
|
if (ret != 1) {
|
|
|
|
char ebuf[1024];
|
|
|
|
|
2010-02-24 16:29:35 +00:00
|
|
|
fprintf(stderr,
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
_("%s: error: unable to determine if daemon is "
|
|
|
|
"running: %s\n"), argv0,
|
|
|
|
virStrerror(errno, ebuf, sizeof(ebuf)));
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
} else if (status != 0) {
|
|
|
|
fprintf(stderr,
|
|
|
|
_("%s: error: %s. Check /var/log/messages or run "
|
|
|
|
"without --daemon for more info.\n"), argv0,
|
2010-02-24 16:29:35 +00:00
|
|
|
virDaemonErrTypeToString(status));
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
_exit(EXIT_SUCCESS);
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
|
|
|
}
|
daemon: clean up daemonization
Valgrind detected a pipe fd leak before the parent exits on success,
introduced in commit 4296cea; by itself, the leak is not bad, since
we immediately called _exit(), but we might as well be clean to make
valgrind analysis easier. Meanwhile, if the daemon grandchild detects
an error, the parent failed to flush the error message before exiting.
Also, we had the possibility of both parent and child returning to the
caller, such that the user could see duplicated reports of failure
from the two return paths. And we might as well be robust to the
(unlikely) situation of being started with stdin closed.
* daemon/libvirtd.c (daemonForkIntoBackground): Use exit if an
error message was generated, avoid fd leaks for valgrind's sake,
avoid returning to caller in both parent and child, and don't
close a just-dup'd stdin.
Based on a report by Alex Jia.
* How to reproduce?
% service libvirtd stop
% valgrind -v --track-fds=yes /usr/sbin/libvirtd --daemon
* Actual valgrind result:
==16804== FILE DESCRIPTORS: 7 open at exit.
==16804== Open file descriptor 7:
==16804== at 0x321FAD8B87: pipe (in /lib64/libc-2.12.so)
==16804== by 0x41F34D: daemonForkIntoBackground (libvirtd.c:186)
==16804== by 0x4207A0: main (libvirtd.c:1420)
Signed-off-by: Alex Jia <ajia@redhat.com>
Signed-off-by: Eric Blake <eblake@redhat.com>
2011-12-22 08:28:04 +00:00
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_FORCE_CLOSE(statuspipe[0]);
|
|
|
|
VIR_FORCE_CLOSE(statuspipe[1]);
|
|
|
|
return -1;
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
|
|
|
|
2007-06-11 12:04:54 +00:00
|
|
|
|
|
|
|
static int
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonPidFilePath(bool privileged,
|
|
|
|
char **pidfile)
|
2007-06-11 12:04:54 +00:00
|
|
|
{
|
2011-05-16 17:13:11 +00:00
|
|
|
if (privileged) {
|
2013-05-03 12:39:39 +00:00
|
|
|
if (VIR_STRDUP(*pidfile, LOCALSTATEDIR "/run/libvirtd.pid") < 0)
|
|
|
|
goto error;
|
2011-05-16 17:13:11 +00:00
|
|
|
} else {
|
2012-05-03 16:36:27 +00:00
|
|
|
char *rundir = NULL;
|
|
|
|
mode_t old_umask;
|
2007-09-19 01:56:55 +00:00
|
|
|
|
2012-05-24 12:29:42 +00:00
|
|
|
if (!(rundir = virGetUserRuntimeDirectory()))
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2007-09-19 01:56:55 +00:00
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
old_umask = umask(077);
|
|
|
|
if (virFileMakePath(rundir) < 0) {
|
|
|
|
umask(old_umask);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
umask(old_umask);
|
|
|
|
|
|
|
|
if (virAsprintf(pidfile, "%s/libvirtd.pid", rundir) < 0) {
|
|
|
|
VIR_FREE(rundir);
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2011-06-30 23:08:41 +00:00
|
|
|
}
|
2007-06-11 12:04:54 +00:00
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
VIR_FREE(rundir);
|
2007-06-11 12:04:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2008-06-06 10:52:01 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
error:
|
2008-06-06 10:52:01 +00:00
|
|
|
return -1;
|
2007-06-11 12:04:54 +00:00
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static int
|
|
|
|
daemonUnixSocketPaths(struct daemonConfig *config,
|
|
|
|
bool privileged,
|
|
|
|
char **sockfile,
|
|
|
|
char **rosockfile)
|
2009-01-22 17:49:41 +00:00
|
|
|
{
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->unix_sock_dir) {
|
|
|
|
if (virAsprintf(sockfile, "%s/libvirt-sock", config->unix_sock_dir) < 0)
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2011-05-16 17:13:11 +00:00
|
|
|
if (privileged &&
|
|
|
|
virAsprintf(rosockfile, "%s/libvirt-sock-ro", config->unix_sock_dir) < 0)
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2009-10-16 10:29:01 +00:00
|
|
|
} else {
|
2011-05-16 17:13:11 +00:00
|
|
|
if (privileged) {
|
2013-05-03 12:39:39 +00:00
|
|
|
if (VIR_STRDUP(*sockfile, LOCALSTATEDIR "/run/libvirt/libvirt-sock") < 0 ||
|
|
|
|
VIR_STRDUP(*rosockfile, LOCALSTATEDIR "/run/libvirt/libvirt-sock-ro") < 0)
|
|
|
|
goto error;
|
2011-05-16 17:13:11 +00:00
|
|
|
} else {
|
2012-05-03 16:36:27 +00:00
|
|
|
char *rundir = NULL;
|
|
|
|
mode_t old_umask;
|
2007-02-16 18:30:55 +00:00
|
|
|
|
2012-05-24 12:29:42 +00:00
|
|
|
if (!(rundir = virGetUserRuntimeDirectory()))
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2007-06-11 12:04:54 +00:00
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
old_umask = umask(077);
|
|
|
|
if (virFileMakePath(rundir) < 0) {
|
|
|
|
umask(old_umask);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
umask(old_umask);
|
|
|
|
|
2012-05-03 12:40:32 +00:00
|
|
|
if (virAsprintf(sockfile, "%s/libvirt-sock", rundir) < 0) {
|
2012-05-03 16:36:27 +00:00
|
|
|
VIR_FREE(rundir);
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2011-05-16 17:13:11 +00:00
|
|
|
}
|
2007-06-11 12:04:54 +00:00
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
VIR_FREE(rundir);
|
2011-05-16 17:13:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2007-02-16 18:30:55 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
error:
|
|
|
|
return -1;
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
static void daemonErrorHandler(void *opaque ATTRIBUTE_UNUSED,
|
|
|
|
virErrorPtr err ATTRIBUTE_UNUSED)
|
2009-10-19 17:28:28 +00:00
|
|
|
{
|
|
|
|
/* Don't do anything, since logging infrastructure already
|
|
|
|
* took care of reporting the error */
|
|
|
|
}
|
|
|
|
|
2011-01-21 17:25:01 +00:00
|
|
|
static int daemonErrorLogFilter(virErrorPtr err, int priority)
|
|
|
|
{
|
|
|
|
/* These error codes don't really reflect real errors. They
|
|
|
|
* are expected events that occur when an app tries to check
|
|
|
|
* whether a particular guest already exists. This filters
|
|
|
|
* them to a lower log level to prevent pollution of syslog
|
|
|
|
*/
|
|
|
|
switch (err->code) {
|
|
|
|
case VIR_ERR_NO_DOMAIN:
|
|
|
|
case VIR_ERR_NO_NETWORK:
|
|
|
|
case VIR_ERR_NO_STORAGE_POOL:
|
|
|
|
case VIR_ERR_NO_STORAGE_VOL:
|
|
|
|
case VIR_ERR_NO_NODE_DEVICE:
|
|
|
|
case VIR_ERR_NO_INTERFACE:
|
|
|
|
case VIR_ERR_NO_NWFILTER:
|
|
|
|
case VIR_ERR_NO_SECRET:
|
|
|
|
case VIR_ERR_NO_DOMAIN_SNAPSHOT:
|
2011-09-13 10:24:13 +00:00
|
|
|
case VIR_ERR_OPERATION_INVALID:
|
2011-01-21 17:25:01 +00:00
|
|
|
return VIR_LOG_DEBUG;
|
|
|
|
}
|
|
|
|
|
|
|
|
return priority;
|
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static void daemonInitialize(void)
|
|
|
|
{
|
2008-12-02 11:37:55 +00:00
|
|
|
/*
|
|
|
|
* Note that the order is important: the first ones have a higher
|
|
|
|
* priority when calling virStateInitialize. We must register
|
|
|
|
* the network, storage and nodedev drivers before any domain
|
|
|
|
* drivers, since their resources must be auto-started before
|
|
|
|
* any domains can be auto-started.
|
|
|
|
*/
|
2008-11-21 12:16:08 +00:00
|
|
|
#ifdef WITH_DRIVER_MODULES
|
|
|
|
/* We don't care if any of these fail, because the whole point
|
|
|
|
* is to allow users to only install modules they want to use.
|
2009-10-26 23:02:46 +00:00
|
|
|
* If they try to open a connection for a module that
|
2008-11-21 12:16:08 +00:00
|
|
|
* is not loaded they'll get a suitable error at that point
|
|
|
|
*/
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# ifdef WITH_NETWORK
|
2008-11-21 12:16:08 +00:00
|
|
|
virDriverLoadModule("network");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_STORAGE
|
2008-11-21 12:16:08 +00:00
|
|
|
virDriverLoadModule("storage");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_NODE_DEVICES
|
2008-11-21 12:27:11 +00:00
|
|
|
virDriverLoadModule("nodedev");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_SECRETS
|
2009-08-14 19:48:55 +00:00
|
|
|
virDriverLoadModule("secret");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_NWFILTER
|
|
|
|
virDriverLoadModule("nwfilter");
|
|
|
|
# endif
|
2012-09-18 01:27:06 +00:00
|
|
|
# ifdef WITH_INTERFACE
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
virDriverLoadModule("interface");
|
|
|
|
# endif
|
2012-08-30 18:53:02 +00:00
|
|
|
# ifdef WITH_XEN
|
|
|
|
virDriverLoadModule("xen");
|
|
|
|
# endif
|
|
|
|
# ifdef WITH_LIBXL
|
|
|
|
virDriverLoadModule("libxl");
|
|
|
|
# endif
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# ifdef WITH_QEMU
|
2008-12-02 11:37:55 +00:00
|
|
|
virDriverLoadModule("qemu");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_LXC
|
2008-12-02 11:37:55 +00:00
|
|
|
virDriverLoadModule("lxc");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_UML
|
2008-12-02 11:37:55 +00:00
|
|
|
virDriverLoadModule("uml");
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# endif
|
2013-05-15 10:31:36 +00:00
|
|
|
# ifdef WITH_VBOX
|
|
|
|
virDriverLoadModule("vbox");
|
|
|
|
# endif
|
2008-11-21 12:16:08 +00:00
|
|
|
#else
|
2010-03-09 18:22:22 +00:00
|
|
|
# ifdef WITH_NETWORK
|
2008-11-17 12:18:18 +00:00
|
|
|
networkRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
2012-09-18 01:27:06 +00:00
|
|
|
# ifdef WITH_INTERFACE
|
2009-07-21 14:02:16 +00:00
|
|
|
interfaceRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
2012-06-05 16:28:52 +00:00
|
|
|
# ifdef WITH_STORAGE
|
2008-11-17 12:18:18 +00:00
|
|
|
storageRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
Only build server side drivers as modules
The driver modules all use symbols which are defined in libvirt.so.
Thus for loading of modules to work, the binary that libvirt.so
is linked to must export its symbols back to modules. If the
libvirt.so itself is dlopen()d then the RTLD_GLOBAL flag must
be set. Unfortunately few, if any, programming languages use
the RTLD_GLOBAL flag when loading modules :-( This means is it
not practical to use driver modules for any libvirt client side
drivers (OpenVZ, VMWare, Hyper-V, Remote client, test).
This patch changes the build process so only server side drivers
are built as modules (Xen, QEMU, LXC, UML)
* daemon/libvirtd.c: Add missing load of 'interface' driver
* src/Makefile.am: Only build server side drivers as modules
* src/libvirt.c: Don't load any driver modules
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2011-06-01 15:36:21 +00:00
|
|
|
# ifdef WITH_NODE_DEVICES
|
2008-11-21 12:27:11 +00:00
|
|
|
nodedevRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_SECRETS
|
2009-08-14 19:48:55 +00:00
|
|
|
secretRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
2010-03-25 17:46:09 +00:00
|
|
|
# ifdef WITH_NWFILTER
|
|
|
|
nwfilterRegister();
|
|
|
|
# endif
|
2012-08-30 18:53:02 +00:00
|
|
|
# ifdef WITH_XEN
|
|
|
|
xenRegister();
|
|
|
|
# endif
|
2011-02-10 22:42:34 +00:00
|
|
|
# ifdef WITH_LIBXL
|
|
|
|
libxlRegister();
|
|
|
|
# endif
|
2010-03-09 18:22:22 +00:00
|
|
|
# ifdef WITH_QEMU
|
2008-12-02 11:37:55 +00:00
|
|
|
qemuRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_LXC
|
2008-12-02 11:37:55 +00:00
|
|
|
lxcRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
|
|
|
# ifdef WITH_UML
|
2008-12-02 11:37:55 +00:00
|
|
|
umlRegister();
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
2013-05-15 10:31:36 +00:00
|
|
|
# ifdef WITH_VBOX
|
|
|
|
vboxRegister();
|
|
|
|
# endif
|
2008-11-17 12:18:18 +00:00
|
|
|
#endif
|
2007-12-05 15:34:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static int daemonSetupNetworking(virNetServerPtr srv,
|
|
|
|
struct daemonConfig *config,
|
|
|
|
const char *sock_path,
|
|
|
|
const char *sock_path_ro,
|
|
|
|
bool ipsock,
|
|
|
|
bool privileged)
|
|
|
|
{
|
|
|
|
virNetServerServicePtr svc = NULL;
|
|
|
|
virNetServerServicePtr svcRO = NULL;
|
|
|
|
virNetServerServicePtr svcTCP = NULL;
|
2013-01-08 21:02:05 +00:00
|
|
|
#if WITH_GNUTLS
|
2011-05-16 17:13:11 +00:00
|
|
|
virNetServerServicePtr svcTLS = NULL;
|
2013-01-07 14:54:18 +00:00
|
|
|
#endif
|
2011-05-16 17:13:11 +00:00
|
|
|
gid_t unix_sock_gid = 0;
|
|
|
|
int unix_sock_ro_mask = 0;
|
|
|
|
int unix_sock_rw_mask = 0;
|
|
|
|
|
|
|
|
if (config->unix_sock_group) {
|
|
|
|
if (virGetGroupID(config->unix_sock_group, &unix_sock_gid) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
2007-12-05 15:34:05 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virStrToLong_i(config->unix_sock_ro_perms, NULL, 8, &unix_sock_ro_mask) != 0) {
|
|
|
|
VIR_ERROR(_("Failed to parse mode '%s'"), config->unix_sock_ro_perms);
|
|
|
|
goto error;
|
|
|
|
}
|
2007-12-05 15:34:05 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virStrToLong_i(config->unix_sock_rw_perms, NULL, 8, &unix_sock_rw_mask) != 0) {
|
|
|
|
VIR_ERROR(_("Failed to parse mode '%s'"), config->unix_sock_rw_perms);
|
|
|
|
goto error;
|
|
|
|
}
|
2007-12-05 15:34:05 +00:00
|
|
|
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Registering unix socket %s", sock_path);
|
2011-05-16 17:13:11 +00:00
|
|
|
if (!(svc = virNetServerServiceNewUNIX(sock_path,
|
|
|
|
unix_sock_rw_mask,
|
|
|
|
unix_sock_gid,
|
|
|
|
config->auth_unix_rw,
|
2013-01-08 21:02:05 +00:00
|
|
|
#if WITH_GNUTLS
|
2013-01-07 14:54:18 +00:00
|
|
|
NULL,
|
|
|
|
#endif
|
2011-05-16 17:13:11 +00:00
|
|
|
false,
|
2013-07-25 12:45:38 +00:00
|
|
|
config->max_queued_clients,
|
2013-01-07 14:54:18 +00:00
|
|
|
config->max_client_requests)))
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2012-06-15 12:48:34 +00:00
|
|
|
if (sock_path_ro) {
|
|
|
|
VIR_DEBUG("Registering unix socket %s", sock_path_ro);
|
|
|
|
if (!(svcRO = virNetServerServiceNewUNIX(sock_path_ro,
|
|
|
|
unix_sock_ro_mask,
|
|
|
|
unix_sock_gid,
|
|
|
|
config->auth_unix_ro,
|
2013-01-08 21:02:05 +00:00
|
|
|
#if WITH_GNUTLS
|
2013-01-07 14:54:18 +00:00
|
|
|
NULL,
|
|
|
|
#endif
|
2012-06-15 12:48:34 +00:00
|
|
|
true,
|
2013-07-25 12:45:38 +00:00
|
|
|
config->max_queued_clients,
|
2013-01-07 14:54:18 +00:00
|
|
|
config->max_client_requests)))
|
2012-06-15 12:48:34 +00:00
|
|
|
goto error;
|
|
|
|
}
|
2009-03-02 11:13:37 +00:00
|
|
|
|
2011-10-11 12:48:42 +00:00
|
|
|
if (virNetServerAddService(srv, svc,
|
|
|
|
config->mdns_adv && !ipsock ?
|
|
|
|
"_libvirt._tcp" :
|
|
|
|
NULL) < 0)
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2011-10-11 12:48:42 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (svcRO &&
|
|
|
|
virNetServerAddService(srv, svcRO, NULL) < 0)
|
|
|
|
goto error;
|
2007-12-05 18:21:27 +00:00
|
|
|
|
2007-06-26 23:48:46 +00:00
|
|
|
if (ipsock) {
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->listen_tcp) {
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Registering TCP socket %s:%s",
|
|
|
|
config->listen_addr, config->tcp_port);
|
2011-05-16 17:13:11 +00:00
|
|
|
if (!(svcTCP = virNetServerServiceNewTCP(config->listen_addr,
|
|
|
|
config->tcp_port,
|
|
|
|
config->auth_tcp,
|
2013-01-08 21:02:05 +00:00
|
|
|
#if WITH_GNUTLS
|
2013-01-07 14:54:18 +00:00
|
|
|
NULL,
|
|
|
|
#endif
|
2011-05-16 17:13:11 +00:00
|
|
|
false,
|
2013-07-25 12:45:38 +00:00
|
|
|
config->max_queued_clients,
|
2013-01-07 14:54:18 +00:00
|
|
|
config->max_client_requests)))
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2007-06-11 12:04:54 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virNetServerAddService(srv, svcTCP,
|
|
|
|
config->mdns_adv ? "_libvirt._tcp" : NULL) < 0)
|
|
|
|
goto error;
|
2007-06-11 12:04:54 +00:00
|
|
|
}
|
|
|
|
|
2013-01-08 21:02:05 +00:00
|
|
|
#if WITH_GNUTLS
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->listen_tls) {
|
|
|
|
virNetTLSContextPtr ctxt = NULL;
|
2010-09-14 16:50:25 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->ca_file ||
|
|
|
|
config->cert_file ||
|
|
|
|
config->key_file) {
|
|
|
|
if (!(ctxt = virNetTLSContextNewServer(config->ca_file,
|
|
|
|
config->crl_file,
|
|
|
|
config->cert_file,
|
|
|
|
config->key_file,
|
|
|
|
(const char *const*)config->tls_allowed_dn_list,
|
2011-07-21 10:13:11 +00:00
|
|
|
config->tls_no_sanity_certificate ? false : true,
|
2011-05-16 17:13:11 +00:00
|
|
|
config->tls_no_verify_certificate ? false : true)))
|
|
|
|
goto error;
|
|
|
|
} else {
|
|
|
|
if (!(ctxt = virNetTLSContextNewServerPath(NULL,
|
|
|
|
!privileged,
|
|
|
|
(const char *const*)config->tls_allowed_dn_list,
|
2011-07-21 10:13:11 +00:00
|
|
|
config->tls_no_sanity_certificate ? false : true,
|
2011-05-16 17:13:11 +00:00
|
|
|
config->tls_no_verify_certificate ? false : true)))
|
|
|
|
goto error;
|
2010-09-14 16:50:25 +00:00
|
|
|
}
|
|
|
|
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Registering TLS socket %s:%s",
|
|
|
|
config->listen_addr, config->tls_port);
|
2011-05-16 17:13:11 +00:00
|
|
|
if (!(svcTLS =
|
|
|
|
virNetServerServiceNewTCP(config->listen_addr,
|
|
|
|
config->tls_port,
|
|
|
|
config->auth_tls,
|
2013-01-07 14:54:18 +00:00
|
|
|
ctxt,
|
2011-05-16 17:13:11 +00:00
|
|
|
false,
|
2013-07-25 12:45:38 +00:00
|
|
|
config->max_queued_clients,
|
2013-01-07 14:54:18 +00:00
|
|
|
config->max_client_requests))) {
|
2012-07-11 13:35:48 +00:00
|
|
|
virObjectUnref(ctxt);
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2010-09-14 16:50:25 +00:00
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virNetServerAddService(srv, svcTLS,
|
|
|
|
config->mdns_adv &&
|
|
|
|
!config->listen_tcp ? "_libvirt._tcp" : NULL) < 0)
|
|
|
|
goto error;
|
Detect heap allocation failure; factor out some duplication.
* qemud/qemud.c (tls_port, tcp_port, mdns_name, tls_allowed_ip_list):
(tls_allowed_dn_list): Remove "const", now that we free these.
(unix_sock_rw_mask): Rename from unix_sock_rw_perms, so that
the latter name can be used as a local string variable, so that the
variable name matches the config attribute name.
(unix_sock_ro_mask): Rename from unix_sock_ro_perms, likewise.
(remoteCheckDN, remoteCheckAccess): Adapt to const removal.
(qemudDispatchServer): Check for heap allocation failure.
(remoteConfigGetStringList): New function, based on code from Dan Berrangé.
(CHECK_TYPE): Remove macro.
(checkType): New function.
(GET_CONF_INT, GET_CONF_STR): New macros.
(remoteReadConfigFile): Use new macros to avoid duplication and to
check for allocation failure.
* src/conf.h (virConfTypeName): New static inline function.
2007-11-30 15:43:42 +00:00
|
|
|
|
2012-07-11 13:35:48 +00:00
|
|
|
virObjectUnref(ctxt);
|
2007-06-11 12:04:54 +00:00
|
|
|
}
|
2013-01-07 14:54:18 +00:00
|
|
|
#else
|
|
|
|
(void)privileged;
|
|
|
|
if (config->listen_tls) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("This libvirtd build does not support TLS"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
#endif
|
2007-06-11 12:04:54 +00:00
|
|
|
}
|
|
|
|
|
2012-09-20 11:58:29 +00:00
|
|
|
#if WITH_SASL
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->auth_unix_rw == REMOTE_AUTH_SASL ||
|
|
|
|
config->auth_unix_ro == REMOTE_AUTH_SASL ||
|
2013-01-08 21:02:05 +00:00
|
|
|
# if WITH_GNUTLS
|
2013-01-07 14:54:18 +00:00
|
|
|
config->auth_tls == REMOTE_AUTH_SASL ||
|
|
|
|
# endif
|
|
|
|
config->auth_tcp == REMOTE_AUTH_SASL) {
|
2011-05-16 17:13:11 +00:00
|
|
|
saslCtxt = virNetSASLContextNewServer(
|
|
|
|
(const char *const*)config->sasl_allowed_username_list);
|
|
|
|
if (!saslCtxt)
|
2010-09-14 16:50:25 +00:00
|
|
|
goto error;
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
2011-06-30 17:18:08 +00:00
|
|
|
#endif
|
2007-02-14 01:40:09 +00:00
|
|
|
|
|
|
|
return 0;
|
2007-06-11 12:04:54 +00:00
|
|
|
|
2010-09-14 16:50:25 +00:00
|
|
|
error:
|
2013-01-08 21:02:05 +00:00
|
|
|
#if WITH_GNUTLS
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(svcTLS);
|
2013-01-07 14:54:18 +00:00
|
|
|
#endif
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(svcTCP);
|
|
|
|
virObjectUnref(svc);
|
|
|
|
virObjectUnref(svcRO);
|
2007-06-11 12:04:54 +00:00
|
|
|
return -1;
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-22 12:53:26 +00:00
|
|
|
/*
|
|
|
|
* Set up the logging environment
|
2011-03-03 06:48:06 +00:00
|
|
|
* By default if daemonized all errors go to the logfile libvirtd.log,
|
|
|
|
* but if verbose or error debugging is asked for then also output
|
2011-03-08 10:31:20 +00:00
|
|
|
* informational and debug messages. Default size if 64 kB.
|
2008-12-22 12:53:26 +00:00
|
|
|
*/
|
2008-12-22 16:16:10 +00:00
|
|
|
static int
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonSetupLogging(struct daemonConfig *config,
|
|
|
|
bool privileged,
|
|
|
|
bool verbose,
|
|
|
|
bool godaemon)
|
2009-10-23 06:56:21 +00:00
|
|
|
{
|
2008-12-22 12:53:26 +00:00
|
|
|
virLogReset();
|
|
|
|
|
2008-12-22 16:16:10 +00:00
|
|
|
/*
|
2009-08-06 13:55:07 +00:00
|
|
|
* Libvirtd's order of precedence is:
|
|
|
|
* cmdline > environment > config
|
|
|
|
*
|
|
|
|
* In order to achieve this, we must process configuration in
|
|
|
|
* different order for the log level versus the filters and
|
|
|
|
* outputs. Because filters and outputs append, we have to look at
|
|
|
|
* the environment first and then only check the config file if
|
|
|
|
* there was no result from the environment. The default output is
|
|
|
|
* then applied only if there was no setting from either of the
|
|
|
|
* first two. Because we don't have a way to determine if the log
|
|
|
|
* level has been set, we must process variables in the opposite
|
|
|
|
* order, each one overriding the previous.
|
2008-12-22 16:16:10 +00:00
|
|
|
*/
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->log_level != 0)
|
|
|
|
virLogSetDefaultPriority(config->log_level);
|
2009-08-06 13:45:50 +00:00
|
|
|
|
2009-08-06 13:55:07 +00:00
|
|
|
virLogSetFromEnv();
|
2009-08-06 13:45:50 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
virLogSetBufferSize(config->log_buffer_size);
|
2008-12-22 12:53:26 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virLogGetNbFilters() == 0)
|
|
|
|
virLogParseFilters(config->log_filters);
|
|
|
|
|
|
|
|
if (virLogGetNbOutputs() == 0)
|
|
|
|
virLogParseOutputs(config->log_outputs);
|
2008-12-22 12:53:26 +00:00
|
|
|
|
|
|
|
/*
|
2012-10-24 20:51:44 +00:00
|
|
|
* If no defined outputs, and either running
|
|
|
|
* as daemon or not on a tty, then first try
|
|
|
|
* to direct it to the systemd journal
|
|
|
|
* (if it exists)....
|
2012-09-27 12:13:09 +00:00
|
|
|
*/
|
2012-10-24 20:51:44 +00:00
|
|
|
if (virLogGetNbOutputs() == 0 &&
|
|
|
|
(godaemon || !isatty(STDIN_FILENO))) {
|
2012-09-27 12:13:09 +00:00
|
|
|
char *tmp;
|
|
|
|
if (access("/run/systemd/journal/socket", W_OK) >= 0) {
|
|
|
|
if (virAsprintf(&tmp, "%d:journald", virLogGetDefaultPriority()) < 0)
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2012-09-27 12:13:09 +00:00
|
|
|
virLogParseOutputs(tmp);
|
|
|
|
VIR_FREE(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* otherwise direct to libvirtd.log when running
|
2009-08-06 13:45:50 +00:00
|
|
|
* as daemon. Otherwise the default output is stderr.
|
2008-12-22 12:53:26 +00:00
|
|
|
*/
|
2009-08-06 13:45:50 +00:00
|
|
|
if (virLogGetNbOutputs() == 0) {
|
2009-07-01 11:21:15 +00:00
|
|
|
char *tmp = NULL;
|
2011-03-03 06:48:06 +00:00
|
|
|
|
2009-01-20 21:50:31 +00:00
|
|
|
if (godaemon) {
|
2011-05-16 17:13:11 +00:00
|
|
|
if (privileged) {
|
2011-03-03 06:48:06 +00:00
|
|
|
if (virAsprintf(&tmp, "%d:file:%s/log/libvirt/libvirtd.log",
|
|
|
|
virLogGetDefaultPriority(),
|
|
|
|
LOCALSTATEDIR) == -1)
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2011-03-03 06:48:06 +00:00
|
|
|
} else {
|
2012-05-24 12:29:42 +00:00
|
|
|
char *logdir = virGetUserCacheDirectory();
|
2012-05-03 16:36:27 +00:00
|
|
|
mode_t old_umask;
|
|
|
|
|
|
|
|
if (!logdir)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
old_umask = umask(077);
|
|
|
|
if (virFileMakePath(logdir) < 0) {
|
|
|
|
umask(old_umask);
|
2011-05-16 17:13:11 +00:00
|
|
|
goto error;
|
2012-05-03 16:36:27 +00:00
|
|
|
}
|
|
|
|
umask(old_umask);
|
2011-03-03 06:48:06 +00:00
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
if (virAsprintf(&tmp, "%d:file:%s/libvirtd.log",
|
|
|
|
virLogGetDefaultPriority(), logdir) == -1) {
|
|
|
|
VIR_FREE(logdir);
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2011-06-02 21:56:32 +00:00
|
|
|
}
|
2012-05-03 16:36:27 +00:00
|
|
|
VIR_FREE(logdir);
|
2011-03-03 06:48:06 +00:00
|
|
|
}
|
2009-01-20 21:50:31 +00:00
|
|
|
} else {
|
2011-03-03 06:48:06 +00:00
|
|
|
if (virAsprintf(&tmp, "%d:stderr", virLogGetDefaultPriority()) < 0)
|
2013-07-04 09:58:18 +00:00
|
|
|
goto error;
|
2009-01-20 21:50:31 +00:00
|
|
|
}
|
2009-07-01 11:21:15 +00:00
|
|
|
virLogParseOutputs(tmp);
|
|
|
|
VIR_FREE(tmp);
|
2009-06-03 10:36:17 +00:00
|
|
|
}
|
2009-08-06 13:55:07 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Command line override for --verbose
|
|
|
|
*/
|
|
|
|
if ((verbose) && (virLogGetDefaultPriority() > VIR_LOG_INFO))
|
|
|
|
virLogSetDefaultPriority(VIR_LOG_INFO);
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-12-22 16:16:10 +00:00
|
|
|
|
2013-04-17 11:01:24 +00:00
|
|
|
static int
|
|
|
|
daemonSetupAccessManager(struct daemonConfig *config)
|
|
|
|
{
|
|
|
|
virAccessManagerPtr mgr;
|
|
|
|
const char *none[] = { "none", NULL };
|
|
|
|
const char **driver = (const char **)config->access_drivers;
|
|
|
|
|
|
|
|
if (!driver ||
|
|
|
|
!driver[0])
|
|
|
|
driver = none;
|
|
|
|
|
|
|
|
if (!(mgr = virAccessManagerNewStack(driver)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
virAccessManagerSetDefault(mgr);
|
|
|
|
virObjectUnref(mgr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-12 07:56:50 +00:00
|
|
|
/* Display version information. */
|
|
|
|
static void
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonVersion(const char *argv0)
|
2008-12-12 07:56:50 +00:00
|
|
|
{
|
2012-10-17 09:23:12 +00:00
|
|
|
printf("%s (%s) %s\n", argv0, PACKAGE_NAME, PACKAGE_VERSION);
|
2008-12-12 07:56:50 +00:00
|
|
|
}
|
|
|
|
|
2009-01-22 17:49:41 +00:00
|
|
|
#ifdef __sun
|
|
|
|
static int
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonSetupPrivs(void)
|
2009-01-22 17:49:41 +00:00
|
|
|
{
|
2012-10-17 09:23:12 +00:00
|
|
|
chown("/var/run/libvirt", SYSTEM_UID, SYSTEM_UID);
|
2009-01-22 17:49:41 +00:00
|
|
|
|
2012-10-17 09:23:12 +00:00
|
|
|
if (__init_daemon_priv(PU_RESETGROUPS | PU_CLEARLIMITSET,
|
|
|
|
SYSTEM_UID, SYSTEM_UID, PRIV_XVM_CONTROL, NULL)) {
|
2011-05-09 09:24:09 +00:00
|
|
|
VIR_ERROR(_("additional privileges are required"));
|
2009-01-22 17:49:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-10-17 09:23:12 +00:00
|
|
|
if (priv_set(PRIV_OFF, PRIV_ALLSETS, PRIV_FILE_LINK_ANY, PRIV_PROC_INFO,
|
|
|
|
PRIV_PROC_SESSION, PRIV_PROC_EXEC, PRIV_PROC_FORK, NULL)) {
|
2011-05-09 09:24:09 +00:00
|
|
|
VIR_ERROR(_("failed to set reduced privileges"));
|
2009-01-22 17:49:41 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else
|
2011-05-16 17:13:11 +00:00
|
|
|
# define daemonSetupPrivs() 0
|
2009-01-22 17:49:41 +00:00
|
|
|
#endif
|
|
|
|
|
2009-10-16 10:48:50 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static void daemonShutdownHandler(virNetServerPtr srv,
|
|
|
|
siginfo_t *sig ATTRIBUTE_UNUSED,
|
|
|
|
void *opaque ATTRIBUTE_UNUSED)
|
2009-10-16 10:48:50 +00:00
|
|
|
{
|
2011-05-16 17:13:11 +00:00
|
|
|
virNetServerQuit(srv);
|
|
|
|
}
|
2009-10-16 10:48:50 +00:00
|
|
|
|
2011-08-15 07:40:46 +00:00
|
|
|
static void daemonReloadHandler(virNetServerPtr srv ATTRIBUTE_UNUSED,
|
|
|
|
siginfo_t *sig ATTRIBUTE_UNUSED,
|
|
|
|
void *opaque ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
VIR_INFO("Reloading configuration on SIGHUP");
|
|
|
|
virHookCall(VIR_HOOK_DRIVER_DAEMON, "-",
|
2012-02-27 16:06:22 +00:00
|
|
|
VIR_HOOK_DAEMON_OP_RELOAD, SIGHUP, "SIGHUP", NULL, NULL);
|
2011-08-15 07:40:46 +00:00
|
|
|
if (virStateReload() < 0)
|
|
|
|
VIR_WARN("Error while reloading drivers");
|
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static int daemonSetupSignals(virNetServerPtr srv)
|
|
|
|
{
|
|
|
|
if (virNetServerAddSignalHandler(srv, SIGINT, daemonShutdownHandler, NULL) < 0)
|
2009-10-16 10:48:50 +00:00
|
|
|
return -1;
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virNetServerAddSignalHandler(srv, SIGQUIT, daemonShutdownHandler, NULL) < 0)
|
|
|
|
return -1;
|
|
|
|
if (virNetServerAddSignalHandler(srv, SIGTERM, daemonShutdownHandler, NULL) < 0)
|
|
|
|
return -1;
|
2011-08-15 07:40:46 +00:00
|
|
|
if (virNetServerAddSignalHandler(srv, SIGHUP, daemonReloadHandler, NULL) < 0)
|
|
|
|
return -1;
|
2011-05-16 17:13:11 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2009-10-16 10:48:50 +00:00
|
|
|
|
2012-10-31 19:03:55 +00:00
|
|
|
|
|
|
|
static void daemonInhibitCallback(bool inhibit, void *opaque)
|
|
|
|
{
|
|
|
|
virNetServerPtr srv = opaque;
|
|
|
|
|
|
|
|
if (inhibit)
|
|
|
|
virNetServerAddShutdownInhibition(srv);
|
|
|
|
else
|
|
|
|
virNetServerRemoveShutdownInhibition(srv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-31 19:03:57 +00:00
|
|
|
#ifdef HAVE_DBUS
|
|
|
|
static DBusConnection *sessionBus;
|
|
|
|
static DBusConnection *systemBus;
|
|
|
|
|
|
|
|
static void daemonStopWorker(void *opaque)
|
|
|
|
{
|
|
|
|
virNetServerPtr srv = opaque;
|
|
|
|
|
|
|
|
VIR_DEBUG("Begin stop srv=%p", srv);
|
|
|
|
|
|
|
|
ignore_value(virStateStop());
|
|
|
|
|
|
|
|
VIR_DEBUG("Completed stop srv=%p", srv);
|
|
|
|
|
|
|
|
/* Exit libvirtd cleanly */
|
|
|
|
virNetServerQuit(srv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* We do this in a thread to not block the main loop */
|
|
|
|
static void daemonStop(virNetServerPtr srv)
|
|
|
|
{
|
|
|
|
virThread thr;
|
|
|
|
virObjectRef(srv);
|
|
|
|
if (virThreadCreate(&thr, false, daemonStopWorker, srv) < 0)
|
|
|
|
virObjectUnref(srv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static DBusHandlerResult
|
|
|
|
handleSessionMessageFunc(DBusConnection *connection ATTRIBUTE_UNUSED,
|
|
|
|
DBusMessage *message,
|
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
virNetServerPtr srv = opaque;
|
|
|
|
|
|
|
|
VIR_DEBUG("srv=%p", srv);
|
|
|
|
|
|
|
|
if (dbus_message_is_signal(message,
|
|
|
|
DBUS_INTERFACE_LOCAL,
|
|
|
|
"Disconnected"))
|
|
|
|
daemonStop(srv);
|
|
|
|
|
|
|
|
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static DBusHandlerResult
|
|
|
|
handleSystemMessageFunc(DBusConnection *connection ATTRIBUTE_UNUSED,
|
|
|
|
DBusMessage *message,
|
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
virNetServerPtr srv = opaque;
|
|
|
|
|
|
|
|
VIR_DEBUG("srv=%p", srv);
|
|
|
|
|
|
|
|
if (dbus_message_is_signal(message,
|
|
|
|
"org.freedesktop.login1.Manager",
|
|
|
|
"PrepareForShutdown"))
|
|
|
|
daemonStop(srv);
|
|
|
|
|
|
|
|
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static void daemonRunStateInit(void *opaque)
|
|
|
|
{
|
|
|
|
virNetServerPtr srv = opaque;
|
2013-04-17 11:01:24 +00:00
|
|
|
virIdentityPtr sysident = virIdentityGetSystem();
|
|
|
|
|
|
|
|
virIdentitySetCurrent(sysident);
|
2009-10-16 10:48:50 +00:00
|
|
|
|
2012-12-05 19:21:14 +00:00
|
|
|
/* Since driver initialization can take time inhibit daemon shutdown until
|
|
|
|
we're done so clients get a chance to connect */
|
|
|
|
daemonInhibitCallback(true, srv);
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
/* Start the stateful HV drivers
|
2011-08-23 17:02:02 +00:00
|
|
|
* This is deliberately done after telling the parent process
|
2011-05-16 17:13:11 +00:00
|
|
|
* we're ready, since it can take a long time and this will
|
|
|
|
* seriously delay OS bootup process */
|
2012-10-31 19:03:55 +00:00
|
|
|
if (virStateInitialize(virNetServerIsPrivileged(srv),
|
|
|
|
daemonInhibitCallback,
|
|
|
|
srv) < 0) {
|
2011-05-16 17:13:11 +00:00
|
|
|
VIR_ERROR(_("Driver state initialization failed"));
|
2011-07-21 17:23:21 +00:00
|
|
|
/* Ensure the main event loop quits */
|
|
|
|
kill(getpid(), SIGTERM);
|
2012-12-05 19:21:14 +00:00
|
|
|
goto cleanup;
|
2009-10-16 10:48:50 +00:00
|
|
|
}
|
|
|
|
|
2012-10-31 19:03:57 +00:00
|
|
|
#ifdef HAVE_DBUS
|
|
|
|
/* Tie the non-priviledged libvirtd to the session/shutdown lifecycle */
|
|
|
|
if (!virNetServerIsPrivileged(srv)) {
|
|
|
|
|
|
|
|
sessionBus = virDBusGetSessionBus();
|
|
|
|
if (sessionBus != NULL)
|
|
|
|
dbus_connection_add_filter(sessionBus,
|
|
|
|
handleSessionMessageFunc, srv, NULL);
|
|
|
|
|
|
|
|
systemBus = virDBusGetSystemBus();
|
|
|
|
if (systemBus != NULL) {
|
|
|
|
dbus_connection_add_filter(systemBus,
|
|
|
|
handleSystemMessageFunc, srv, NULL);
|
|
|
|
dbus_bus_add_match(systemBus,
|
|
|
|
"type='signal',sender='org.freedesktop.login1', interface='org.freedesktop.login1.Manager'",
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2011-05-16 17:13:11 +00:00
|
|
|
/* Only now accept clients from network */
|
|
|
|
virNetServerUpdateServices(srv, true);
|
2012-12-05 19:21:14 +00:00
|
|
|
cleanup:
|
|
|
|
daemonInhibitCallback(false, srv);
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(srv);
|
2013-04-17 11:01:24 +00:00
|
|
|
virObjectUnref(sysident);
|
|
|
|
virIdentitySetCurrent(NULL);
|
2011-05-16 17:13:11 +00:00
|
|
|
}
|
2009-10-16 10:48:50 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
static int daemonStateInit(virNetServerPtr srv)
|
|
|
|
{
|
|
|
|
virThread thr;
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectRef(srv);
|
2011-05-16 17:13:11 +00:00
|
|
|
if (virThreadCreate(&thr, false, daemonRunStateInit, srv) < 0) {
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(srv);
|
2011-05-16 17:13:11 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2009-10-16 10:48:50 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
static int migrateProfile(void)
|
|
|
|
{
|
|
|
|
char *old_base = NULL;
|
|
|
|
char *updated = NULL;
|
|
|
|
char *home = NULL;
|
|
|
|
char *xdg_dir = NULL;
|
|
|
|
char *config_dir = NULL;
|
|
|
|
const char *config_home;
|
|
|
|
int ret = -1;
|
|
|
|
mode_t old_umask;
|
|
|
|
|
2012-06-11 11:29:09 +00:00
|
|
|
VIR_DEBUG("Checking if user profile needs migrating");
|
|
|
|
|
2012-05-24 12:29:42 +00:00
|
|
|
if (!(home = virGetUserDirectory()))
|
2012-05-03 16:36:27 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virAsprintf(&old_base, "%s/.libvirt", home) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if the new directory is there or the old one is not: do nothing */
|
2012-05-24 12:29:42 +00:00
|
|
|
if (!(config_dir = virGetUserConfigDirectory()))
|
2012-05-03 16:36:27 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!virFileIsDir(old_base) || virFileExists(config_dir)) {
|
2012-06-11 11:29:09 +00:00
|
|
|
VIR_DEBUG("No old profile in '%s' / "
|
|
|
|
"new profile directory already present '%s'",
|
|
|
|
old_base, config_dir);
|
2012-05-03 16:36:27 +00:00
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* test if we already attempted to migrate first */
|
|
|
|
if (virAsprintf(&updated, "%s/DEPRECATED-DIRECTORY", old_base) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virFileExists(updated)) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-10-09 10:18:15 +00:00
|
|
|
config_home = virGetEnvBlockSUID("XDG_CONFIG_HOME");
|
2012-05-03 16:36:27 +00:00
|
|
|
if (config_home && config_home[0] != '\0') {
|
2013-05-03 12:39:39 +00:00
|
|
|
if (VIR_STRDUP(xdg_dir, config_home) < 0)
|
|
|
|
goto cleanup;
|
2012-05-03 16:36:27 +00:00
|
|
|
} else {
|
|
|
|
if (virAsprintf(&xdg_dir, "%s/.config", home) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
old_umask = umask(077);
|
|
|
|
if (virFileMakePath(xdg_dir) < 0) {
|
|
|
|
umask(old_umask);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
umask(old_umask);
|
|
|
|
|
|
|
|
if (rename(old_base, config_dir) < 0) {
|
|
|
|
int fd = creat(updated, 0600);
|
|
|
|
VIR_FORCE_CLOSE(fd);
|
|
|
|
VIR_ERROR(_("Unable to migrate %s to %s"), old_base, config_dir);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2012-06-11 11:29:09 +00:00
|
|
|
VIR_DEBUG("Profile migrated from %s to %s", old_base, config_dir);
|
2012-05-03 16:36:27 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(home);
|
|
|
|
VIR_FREE(old_base);
|
|
|
|
VIR_FREE(xdg_dir);
|
|
|
|
VIR_FREE(config_dir);
|
|
|
|
VIR_FREE(updated);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-04-04 09:32:00 +00:00
|
|
|
/* Print command-line usage. */
|
|
|
|
static void
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonUsage(const char *argv0, bool privileged)
|
2007-04-04 09:32:00 +00:00
|
|
|
{
|
2012-10-17 09:23:12 +00:00
|
|
|
fprintf(stderr,
|
|
|
|
_("\n"
|
|
|
|
"Usage:\n"
|
|
|
|
" %s [options]\n"
|
|
|
|
"\n"
|
|
|
|
"Options:\n"
|
2013-08-13 11:14:15 +00:00
|
|
|
" -h | --help Display program help:\n"
|
2012-10-17 09:23:12 +00:00
|
|
|
" -v | --verbose Verbose messages.\n"
|
|
|
|
" -d | --daemon Run as a daemon & write PID file.\n"
|
|
|
|
" -l | --listen Listen for TCP/IP connections.\n"
|
|
|
|
" -t | --timeout <secs> Exit after timeout period.\n"
|
|
|
|
" -f | --config <file> Configuration file.\n"
|
2013-08-13 11:14:15 +00:00
|
|
|
" -V | --version Display version information.\n"
|
2012-10-17 09:23:12 +00:00
|
|
|
" -p | --pid-file <file> Change name of PID file.\n"
|
|
|
|
"\n"
|
|
|
|
"libvirt management daemon:\n"),
|
|
|
|
argv0);
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
if (privileged) {
|
|
|
|
fprintf(stderr,
|
2012-10-17 09:23:12 +00:00
|
|
|
_("\n"
|
|
|
|
" Default paths:\n"
|
|
|
|
"\n"
|
|
|
|
" Configuration file (unless overridden by -f):\n"
|
2013-09-18 14:45:14 +00:00
|
|
|
" %s\n"
|
2012-10-17 09:23:12 +00:00
|
|
|
"\n"
|
|
|
|
" Sockets:\n"
|
2013-09-18 14:45:14 +00:00
|
|
|
" %s\n"
|
|
|
|
" %s\n"
|
2012-10-17 09:23:12 +00:00
|
|
|
"\n"
|
|
|
|
" TLS:\n"
|
2013-09-18 12:01:27 +00:00
|
|
|
" CA certificate: %s\n"
|
|
|
|
" Server certificate: %s\n"
|
|
|
|
" Server private key: %s\n"
|
2012-10-17 09:23:12 +00:00
|
|
|
"\n"
|
|
|
|
" PID file (unless overridden by -p):\n"
|
|
|
|
" %s/run/libvirtd.pid\n"
|
|
|
|
"\n"),
|
2013-09-18 14:45:14 +00:00
|
|
|
LIBVIRTD_CONFIGURATION_FILE,
|
|
|
|
LIBVIRTD_PRIV_UNIX_SOCKET,
|
|
|
|
LIBVIRTD_PRIV_UNIX_SOCKET_RO,
|
2013-09-18 12:01:27 +00:00
|
|
|
LIBVIRT_CACERT,
|
|
|
|
LIBVIRT_SERVERCERT,
|
|
|
|
LIBVIRT_SERVERKEY,
|
2011-05-16 17:13:11 +00:00
|
|
|
LOCALSTATEDIR);
|
|
|
|
} else {
|
2012-10-17 09:23:12 +00:00
|
|
|
fprintf(stderr, "%s",
|
|
|
|
_("\n"
|
|
|
|
" Default paths:\n"
|
|
|
|
"\n"
|
|
|
|
" Configuration file (unless overridden by -f):\n"
|
|
|
|
" $XDG_CONFIG_HOME/libvirt/libvirtd.conf\n"
|
|
|
|
"\n"
|
|
|
|
" Sockets:\n"
|
|
|
|
" $XDG_RUNTIME_DIR/libvirt/libvirt-sock\n"
|
|
|
|
"\n"
|
|
|
|
" TLS:\n"
|
|
|
|
" CA certificate: $HOME/.pki/libvirt/cacert.pem\n"
|
|
|
|
" Server certificate: $HOME/.pki/libvirt/servercert.pem\n"
|
|
|
|
" Server private key: $HOME/.pki/libvirt/serverkey.pem\n"
|
|
|
|
"\n"
|
|
|
|
" PID file:\n"
|
|
|
|
" $XDG_RUNTIME_DIR/libvirt/libvirtd.pid\n"
|
|
|
|
"\n"));
|
2011-05-16 17:13:11 +00:00
|
|
|
}
|
2007-04-04 09:32:00 +00:00
|
|
|
}
|
|
|
|
|
2007-02-14 01:40:09 +00:00
|
|
|
#define MAX_LISTEN 5
|
|
|
|
int main(int argc, char **argv) {
|
2011-05-16 17:13:11 +00:00
|
|
|
virNetServerPtr srv = NULL;
|
|
|
|
char *remote_config_file = NULL;
|
2009-10-16 11:14:54 +00:00
|
|
|
int statuswrite = -1;
|
2007-02-23 12:48:36 +00:00
|
|
|
int ret = 1;
|
2011-08-05 14:11:11 +00:00
|
|
|
int pid_file_fd = -1;
|
2011-05-16 17:13:11 +00:00
|
|
|
char *pid_file = NULL;
|
|
|
|
char *sock_file = NULL;
|
|
|
|
char *sock_file_ro = NULL;
|
|
|
|
int timeout = -1; /* -t: Shutdown timeout */
|
|
|
|
int verbose = 0;
|
|
|
|
int godaemon = 0;
|
|
|
|
int ipsock = 0;
|
|
|
|
struct daemonConfig *config;
|
|
|
|
bool privileged = geteuid() == 0 ? true : false;
|
2011-07-08 17:44:25 +00:00
|
|
|
bool implicit_conf = false;
|
2011-08-29 02:52:23 +00:00
|
|
|
char *run_dir = NULL;
|
|
|
|
mode_t old_umask;
|
2007-02-14 01:40:09 +00:00
|
|
|
|
|
|
|
struct option opts[] = {
|
2013-08-13 11:14:15 +00:00
|
|
|
{ "verbose", no_argument, &verbose, 'v'},
|
|
|
|
{ "daemon", no_argument, &godaemon, 'd'},
|
|
|
|
{ "listen", no_argument, &ipsock, 'l'},
|
2007-08-07 13:24:22 +00:00
|
|
|
{ "config", required_argument, NULL, 'f'},
|
2007-04-04 09:32:00 +00:00
|
|
|
{ "timeout", required_argument, NULL, 't'},
|
|
|
|
{ "pid-file", required_argument, NULL, 'p'},
|
2013-08-13 11:14:15 +00:00
|
|
|
{ "version", no_argument, NULL, 'V' },
|
|
|
|
{ "help", no_argument, NULL, 'h' },
|
2007-02-14 01:40:09 +00:00
|
|
|
{0, 0, 0, 0}
|
|
|
|
};
|
|
|
|
|
2012-10-17 09:23:12 +00:00
|
|
|
if (setlocale(LC_ALL, "") == NULL ||
|
|
|
|
bindtextdomain(PACKAGE, LOCALEDIR) == NULL ||
|
2010-11-16 19:01:37 +00:00
|
|
|
textdomain(PACKAGE) == NULL ||
|
|
|
|
virInitialize() < 0) {
|
2011-05-16 17:13:11 +00:00
|
|
|
fprintf(stderr, _("%s: initialization failed\n"), argv[0]);
|
2010-11-16 19:01:37 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2010-05-20 07:52:20 +00:00
|
|
|
}
|
2009-10-16 10:29:01 +00:00
|
|
|
|
2012-08-02 13:41:23 +00:00
|
|
|
if (strstr(argv[0], "lt-libvirtd") ||
|
|
|
|
strstr(argv[0], "/daemon/.libs/libvirtd")) {
|
2012-07-23 15:21:13 +00:00
|
|
|
char *tmp = strrchr(argv[0], '/');
|
|
|
|
if (!tmp) {
|
|
|
|
fprintf(stderr, _("%s: cannot identify driver directory\n"), argv[0]);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
*tmp = '\0';
|
|
|
|
char *driverdir;
|
2013-07-04 09:58:18 +00:00
|
|
|
if (virAsprintfQuiet(&driverdir, "%s/../../src/.libs", argv[0]) < 0) {
|
2012-07-23 15:21:13 +00:00
|
|
|
fprintf(stderr, _("%s: initialization failed\n"), argv[0]);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (access(driverdir, R_OK) < 0) {
|
|
|
|
fprintf(stderr, _("%s: expected driver directory '%s' is missing\n"),
|
|
|
|
argv[0], driverdir);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2012-08-21 13:21:47 +00:00
|
|
|
virLockManagerSetPluginDir(driverdir);
|
|
|
|
#ifdef WITH_DRIVER_MODULES
|
2012-07-23 15:21:13 +00:00
|
|
|
virDriverModuleInitialize(driverdir);
|
2012-08-21 13:21:47 +00:00
|
|
|
#endif
|
2012-07-23 15:21:13 +00:00
|
|
|
*tmp = '/';
|
|
|
|
/* Must not free 'driverdir' - it is still used */
|
|
|
|
}
|
2012-04-02 17:51:11 +00:00
|
|
|
|
2007-02-14 01:40:09 +00:00
|
|
|
while (1) {
|
|
|
|
int optidx = 0;
|
|
|
|
int c;
|
|
|
|
char *tmp;
|
|
|
|
|
2013-08-13 11:14:15 +00:00
|
|
|
c = getopt_long(argc, argv, "ldf:p:t:vVh", opts, &optidx);
|
2007-02-14 01:40:09 +00:00
|
|
|
|
|
|
|
if (c == -1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (c) {
|
|
|
|
case 0:
|
|
|
|
/* Got one of the flags */
|
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
verbose = 1;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
godaemon = 1;
|
|
|
|
break;
|
2007-06-26 23:48:46 +00:00
|
|
|
case 'l':
|
|
|
|
ipsock = 1;
|
2007-02-14 01:40:09 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 't':
|
2008-02-08 09:15:16 +00:00
|
|
|
if (virStrToLong_i(optarg, &tmp, 10, &timeout) != 0
|
2007-11-14 10:53:05 +00:00
|
|
|
|| timeout <= 0
|
|
|
|
/* Ensure that we can multiply by 1000 without overflowing. */
|
2011-09-19 04:16:45 +00:00
|
|
|
|| timeout > INT_MAX / 1000) {
|
|
|
|
VIR_ERROR(_("Invalid value for timeout"));
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2007-02-14 01:40:09 +00:00
|
|
|
break;
|
2007-02-23 12:48:36 +00:00
|
|
|
|
|
|
|
case 'p':
|
2011-05-16 17:13:11 +00:00
|
|
|
VIR_FREE(pid_file);
|
2013-05-03 12:39:39 +00:00
|
|
|
if (VIR_STRDUP_QUIET(pid_file, optarg) < 0) {
|
2011-08-12 11:41:29 +00:00
|
|
|
VIR_ERROR(_("Can't allocate memory"));
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2007-06-11 12:04:54 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'f':
|
2011-05-16 17:13:11 +00:00
|
|
|
VIR_FREE(remote_config_file);
|
2013-05-03 12:39:39 +00:00
|
|
|
if (VIR_STRDUP_QUIET(remote_config_file, optarg) < 0) {
|
2011-08-12 11:41:29 +00:00
|
|
|
VIR_ERROR(_("Can't allocate memory"));
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2007-02-23 12:48:36 +00:00
|
|
|
break;
|
|
|
|
|
2013-08-13 11:14:15 +00:00
|
|
|
case 'V':
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonVersion(argv[0]);
|
2013-08-13 11:14:15 +00:00
|
|
|
exit(EXIT_SUCCESS);
|
2008-12-12 07:56:50 +00:00
|
|
|
|
2013-08-13 11:14:15 +00:00
|
|
|
case 'h':
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonUsage(argv[0], privileged);
|
2013-08-13 11:14:15 +00:00
|
|
|
exit(EXIT_SUCCESS);
|
2007-02-14 01:40:09 +00:00
|
|
|
|
2013-08-13 11:14:15 +00:00
|
|
|
case '?':
|
2007-02-14 01:40:09 +00:00
|
|
|
default:
|
2013-08-13 11:14:15 +00:00
|
|
|
daemonUsage(argv[0], privileged);
|
2012-10-17 09:23:12 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-16 11:03:02 +00:00
|
|
|
if (optind != argc) {
|
|
|
|
fprintf(stderr, "%s: unexpected, non-option, command line arguments\n",
|
|
|
|
argv[0]);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2011-08-12 11:41:29 +00:00
|
|
|
if (!(config = daemonConfigNew(privileged))) {
|
|
|
|
VIR_ERROR(_("Can't create initial configuration"));
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
/* No explicit config, so try and find a default one */
|
2011-07-08 17:44:25 +00:00
|
|
|
if (remote_config_file == NULL) {
|
|
|
|
implicit_conf = true;
|
|
|
|
if (daemonConfigFilePath(privileged,
|
2011-08-12 11:41:29 +00:00
|
|
|
&remote_config_file) < 0) {
|
|
|
|
VIR_ERROR(_("Can't determine config path"));
|
2011-07-08 17:44:25 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2011-07-08 17:44:25 +00:00
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
/* Read the config file if it exists*/
|
|
|
|
if (remote_config_file &&
|
2012-04-04 12:14:19 +00:00
|
|
|
daemonConfigLoadFile(config, remote_config_file, implicit_conf) < 0) {
|
2012-04-04 14:00:17 +00:00
|
|
|
virErrorPtr err = virGetLastError();
|
|
|
|
if (err && err->message)
|
|
|
|
VIR_ERROR(_("Can't load config file: %s: %s"),
|
|
|
|
err->message, remote_config_file);
|
|
|
|
else
|
|
|
|
VIR_ERROR(_("Can't load config file: %s"), remote_config_file);
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
|
2012-06-11 11:29:09 +00:00
|
|
|
if (!privileged &&
|
2012-06-15 12:48:34 +00:00
|
|
|
migrateProfile() < 0) {
|
|
|
|
VIR_ERROR(_("Exiting due to failure to migrate profile"));
|
2012-05-03 16:36:27 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2012-06-15 12:48:34 +00:00
|
|
|
}
|
2012-05-03 16:36:27 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->host_uuid &&
|
|
|
|
virSetHostUUIDStr(config->host_uuid) < 0) {
|
|
|
|
VIR_ERROR(_("invalid host UUID: %s"), config->host_uuid);
|
|
|
|
exit(EXIT_FAILURE);
|
2009-01-12 18:22:32 +00:00
|
|
|
}
|
|
|
|
|
2011-08-12 11:41:29 +00:00
|
|
|
if (daemonSetupLogging(config, privileged, verbose, godaemon) < 0) {
|
|
|
|
VIR_ERROR(_("Can't initialize logging"));
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
|
2013-04-17 11:01:24 +00:00
|
|
|
if (daemonSetupAccessManager(config) < 0) {
|
|
|
|
VIR_ERROR(_("Can't initialize access manager"));
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2011-08-05 14:11:11 +00:00
|
|
|
if (!pid_file &&
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonPidFilePath(privileged,
|
2011-08-12 11:41:29 +00:00
|
|
|
&pid_file) < 0) {
|
|
|
|
VIR_ERROR(_("Can't determine pid file path."));
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Decided on pid file path '%s'", NULLSTR(pid_file));
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
if (daemonUnixSocketPaths(config,
|
|
|
|
privileged,
|
|
|
|
&sock_file,
|
2011-08-12 11:41:29 +00:00
|
|
|
&sock_file_ro) < 0) {
|
|
|
|
VIR_ERROR(_("Can't determine socket paths"));
|
2011-05-16 17:13:11 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2011-08-12 11:41:29 +00:00
|
|
|
}
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Decided on socket paths '%s' and '%s'",
|
|
|
|
sock_file, NULLSTR(sock_file_ro));
|
2011-05-16 17:13:11 +00:00
|
|
|
|
2008-05-20 16:17:36 +00:00
|
|
|
if (godaemon) {
|
2009-02-05 16:28:03 +00:00
|
|
|
char ebuf[1024];
|
2010-12-13 10:18:45 +00:00
|
|
|
|
|
|
|
if (chdir("/") < 0) {
|
|
|
|
VIR_ERROR(_("cannot change to root directory: %s"),
|
|
|
|
virStrerror(errno, ebuf, sizeof(ebuf)));
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
2010-12-13 10:18:45 +00:00
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if ((statuswrite = daemonForkIntoBackground(argv[0])) < 0) {
|
2009-02-05 16:28:03 +00:00
|
|
|
VIR_ERROR(_("Failed to fork as daemon: %s"),
|
2012-03-29 09:52:04 +00:00
|
|
|
virStrerror(errno, ebuf, sizeof(ebuf)));
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
2008-05-20 16:17:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-22 17:49:41 +00:00
|
|
|
/* Ensure the rundir exists (on tmpfs on some systems) */
|
2011-05-16 17:13:11 +00:00
|
|
|
if (privileged) {
|
2013-05-03 12:39:39 +00:00
|
|
|
if (VIR_STRDUP_QUIET(run_dir, LOCALSTATEDIR "/run/libvirt") < 0) {
|
|
|
|
VIR_ERROR(_("Can't allocate memory"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2011-08-29 02:52:23 +00:00
|
|
|
} else {
|
2012-05-24 12:29:42 +00:00
|
|
|
run_dir = virGetUserRuntimeDirectory();
|
2011-08-29 02:52:23 +00:00
|
|
|
|
2012-05-03 16:36:27 +00:00
|
|
|
if (!run_dir) {
|
2011-08-29 02:52:23 +00:00
|
|
|
VIR_ERROR(_("Can't determine user directory"));
|
|
|
|
goto cleanup;
|
2009-01-22 17:49:41 +00:00
|
|
|
}
|
|
|
|
}
|
2012-06-11 11:31:16 +00:00
|
|
|
if (privileged)
|
|
|
|
old_umask = umask(022);
|
|
|
|
else
|
|
|
|
old_umask = umask(077);
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Ensuring run dir '%s' exists", run_dir);
|
2011-08-29 02:52:23 +00:00
|
|
|
if (virFileMakePath(run_dir) < 0) {
|
|
|
|
char ebuf[1024];
|
|
|
|
VIR_ERROR(_("unable to create rundir %s: %s"), run_dir,
|
|
|
|
virStrerror(errno, ebuf, sizeof(ebuf)));
|
|
|
|
ret = VIR_DAEMON_ERR_RUNDIR;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
umask(old_umask);
|
2009-01-22 17:49:41 +00:00
|
|
|
|
2011-08-05 14:11:11 +00:00
|
|
|
/* Try to claim the pidfile, exiting if we can't */
|
|
|
|
if ((pid_file_fd = virPidFileAcquirePath(pid_file, getpid())) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_PIDFILE;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2012-05-03 14:39:04 +00:00
|
|
|
if (virNetlinkStartup() < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (!(srv = virNetServerNew(config->min_workers,
|
|
|
|
config->max_workers,
|
2011-08-12 12:04:31 +00:00
|
|
|
config->prio_workers,
|
2011-05-16 17:13:11 +00:00
|
|
|
config->max_clients,
|
2011-08-24 13:33:34 +00:00
|
|
|
config->keepalive_interval,
|
|
|
|
config->keepalive_count,
|
|
|
|
!!config->keepalive_required,
|
2011-05-16 17:13:11 +00:00
|
|
|
config->mdns_adv ? config->mdns_name : NULL,
|
2012-03-15 18:14:51 +00:00
|
|
|
remoteClientInitHook,
|
2012-08-09 11:54:54 +00:00
|
|
|
NULL,
|
2012-08-09 11:31:34 +00:00
|
|
|
remoteClientFreeFunc,
|
2012-03-15 18:14:51 +00:00
|
|
|
NULL))) {
|
2011-05-16 17:13:11 +00:00
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2009-06-12 13:20:13 +00:00
|
|
|
/* Beyond this point, nothing should rely on using
|
|
|
|
* getuid/geteuid() == 0, for privilege level checks.
|
|
|
|
*/
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Dropping privileges (if required)");
|
2011-05-16 17:13:11 +00:00
|
|
|
if (daemonSetupPrivs() < 0) {
|
2009-10-16 11:14:54 +00:00
|
|
|
ret = VIR_DAEMON_ERR_PRIVS;
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
2009-01-22 17:49:41 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonInitialize();
|
2011-02-17 13:17:59 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
remoteProcs[REMOTE_PROC_AUTH_LIST].needAuth = false;
|
|
|
|
remoteProcs[REMOTE_PROC_AUTH_SASL_INIT].needAuth = false;
|
|
|
|
remoteProcs[REMOTE_PROC_AUTH_SASL_STEP].needAuth = false;
|
|
|
|
remoteProcs[REMOTE_PROC_AUTH_SASL_START].needAuth = false;
|
|
|
|
remoteProcs[REMOTE_PROC_AUTH_POLKIT].needAuth = false;
|
|
|
|
if (!(remoteProgram = virNetServerProgramNew(REMOTE_PROGRAM,
|
|
|
|
REMOTE_PROTOCOL_VERSION,
|
|
|
|
remoteProcs,
|
|
|
|
remoteNProcs))) {
|
2009-10-16 11:14:54 +00:00
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virNetServerAddProgram(srv, remoteProgram) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
2007-12-05 15:34:05 +00:00
|
|
|
}
|
2007-02-16 18:28:17 +00:00
|
|
|
|
2012-12-21 14:20:04 +00:00
|
|
|
if (!(lxcProgram = virNetServerProgramNew(LXC_PROGRAM,
|
|
|
|
LXC_PROTOCOL_VERSION,
|
|
|
|
lxcProcs,
|
|
|
|
lxcNProcs))) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virNetServerAddProgram(srv, lxcProgram) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (!(qemuProgram = virNetServerProgramNew(QEMU_PROGRAM,
|
|
|
|
QEMU_PROTOCOL_VERSION,
|
|
|
|
qemuProcs,
|
|
|
|
qemuNProcs))) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virNetServerAddProgram(srv, qemuProgram) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
2009-10-16 10:48:50 +00:00
|
|
|
|
2012-06-15 12:48:34 +00:00
|
|
|
if (timeout != -1) {
|
|
|
|
VIR_DEBUG("Registering shutdown timeout %d", timeout);
|
2011-05-16 17:13:11 +00:00
|
|
|
virNetServerAutoShutdown(srv,
|
2012-10-31 19:03:55 +00:00
|
|
|
timeout);
|
2012-06-15 12:48:34 +00:00
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
if ((daemonSetupSignals(srv)) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_SIGNAL;
|
|
|
|
goto cleanup;
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
2007-02-14 01:40:09 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->audit_level) {
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Attempting to configure auditing subsystem");
|
2010-09-15 13:44:11 +00:00
|
|
|
if (virAuditOpen() < 0) {
|
2011-05-16 17:13:11 +00:00
|
|
|
if (config->audit_level > 1) {
|
2010-09-15 13:44:11 +00:00
|
|
|
ret = VIR_DAEMON_ERR_AUDIT;
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
2010-09-15 13:44:11 +00:00
|
|
|
}
|
2012-06-15 12:48:34 +00:00
|
|
|
VIR_DEBUG("Proceeding without auditing");
|
2010-09-15 13:44:11 +00:00
|
|
|
}
|
|
|
|
}
|
2011-05-16 17:13:11 +00:00
|
|
|
virAuditLog(config->audit_logging);
|
2010-09-15 13:44:11 +00:00
|
|
|
|
2010-03-26 14:53:32 +00:00
|
|
|
/* setup the hooks if any */
|
2010-03-30 13:06:13 +00:00
|
|
|
if (virHookInitialize() < 0) {
|
2010-03-26 14:53:32 +00:00
|
|
|
ret = VIR_DAEMON_ERR_HOOKS;
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
2010-03-26 14:53:32 +00:00
|
|
|
}
|
|
|
|
|
2009-10-19 17:28:28 +00:00
|
|
|
/* Disable error func, now logging is setup */
|
2011-05-16 17:13:11 +00:00
|
|
|
virSetErrorFunc(NULL, daemonErrorHandler);
|
2011-01-21 17:25:01 +00:00
|
|
|
virSetErrorLogPriorityFunc(daemonErrorLogFilter);
|
2009-10-19 17:28:28 +00:00
|
|
|
|
2010-03-26 14:53:32 +00:00
|
|
|
/*
|
|
|
|
* Call the daemon startup hook
|
|
|
|
* TODO: should we abort the daemon startup if the script returned
|
|
|
|
* an error ?
|
|
|
|
*/
|
|
|
|
virHookCall(VIR_HOOK_DRIVER_DAEMON, "-", VIR_HOOK_DAEMON_OP_START,
|
2012-02-27 16:06:22 +00:00
|
|
|
0, "start", NULL, NULL);
|
2010-03-26 14:53:32 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (daemonSetupNetworking(srv, config,
|
|
|
|
sock_file, sock_file_ro,
|
|
|
|
ipsock, privileged) < 0) {
|
2009-10-16 11:14:54 +00:00
|
|
|
ret = VIR_DAEMON_ERR_NETWORK;
|
2011-05-16 17:13:11 +00:00
|
|
|
goto cleanup;
|
2007-12-05 15:34:05 +00:00
|
|
|
}
|
|
|
|
|
2009-10-16 11:14:54 +00:00
|
|
|
/* 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)
|
|
|
|
;
|
2010-11-09 20:48:48 +00:00
|
|
|
VIR_FORCE_CLOSE(statuswrite);
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
/* Initialize drivers & then start accepting new clients from network */
|
|
|
|
if (daemonStateInit(srv) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_INIT;
|
|
|
|
goto cleanup;
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
2007-02-14 01:40:09 +00:00
|
|
|
|
2012-08-29 01:21:38 +00:00
|
|
|
#if defined(__linux__) && defined(NETLINK_ROUTE)
|
2012-08-22 04:10:23 +00:00
|
|
|
/* Register the netlink event service for NETLINK_ROUTE */
|
|
|
|
if (virNetlinkEventServiceStart(NETLINK_ROUTE, 0) < 0) {
|
2012-02-22 13:17:13 +00:00
|
|
|
ret = VIR_DAEMON_ERR_NETWORK;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2012-08-29 01:21:38 +00:00
|
|
|
#endif
|
2012-02-22 13:17:13 +00:00
|
|
|
|
2012-08-29 01:21:38 +00:00
|
|
|
#if defined(__linux__) && defined(NETLINK_KOBJECT_UEVENT)
|
2012-08-22 04:10:25 +00:00
|
|
|
/* Register the netlink event service for NETLINK_KOBJECT_UEVENT */
|
|
|
|
if (virNetlinkEventServiceStart(NETLINK_KOBJECT_UEVENT, 1) < 0) {
|
|
|
|
ret = VIR_DAEMON_ERR_NETWORK;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2012-08-29 01:21:38 +00:00
|
|
|
#endif
|
2012-08-22 04:10:25 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
/* Run event loop. */
|
|
|
|
virNetServerRun(srv);
|
2009-10-16 15:34:37 +00:00
|
|
|
|
2007-02-23 12:48:36 +00:00
|
|
|
ret = 0;
|
|
|
|
|
2010-03-26 14:53:32 +00:00
|
|
|
virHookCall(VIR_HOOK_DRIVER_DAEMON, "-", VIR_HOOK_DAEMON_OP_SHUTDOWN,
|
2012-02-27 16:06:22 +00:00
|
|
|
0, "shutdown", NULL, NULL);
|
2010-03-26 14:53:32 +00:00
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
cleanup:
|
2012-08-22 04:10:24 +00:00
|
|
|
virNetlinkEventServiceStopAll();
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(remoteProgram);
|
2012-12-21 14:20:04 +00:00
|
|
|
virObjectUnref(lxcProgram);
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(qemuProgram);
|
2011-08-04 08:54:58 +00:00
|
|
|
virNetServerClose(srv);
|
2012-07-11 13:35:52 +00:00
|
|
|
virObjectUnref(srv);
|
2012-05-03 14:39:04 +00:00
|
|
|
virNetlinkShutdown();
|
2009-10-16 11:14:54 +00:00
|
|
|
if (statuswrite != -1) {
|
|
|
|
if (ret != 0) {
|
|
|
|
/* Tell parent of daemon what failed */
|
|
|
|
char status = ret;
|
|
|
|
while (write(statuswrite, &status, 1) == -1 &&
|
|
|
|
errno == EINTR)
|
|
|
|
;
|
|
|
|
}
|
2010-11-09 20:48:48 +00:00
|
|
|
VIR_FORCE_CLOSE(statuswrite);
|
2009-10-16 11:14:54 +00:00
|
|
|
}
|
2011-08-05 14:11:11 +00:00
|
|
|
if (pid_file_fd != -1)
|
|
|
|
virPidFileReleasePath(pid_file, pid_file_fd);
|
2011-05-16 17:13:11 +00:00
|
|
|
|
|
|
|
VIR_FREE(sock_file);
|
|
|
|
VIR_FREE(sock_file_ro);
|
|
|
|
VIR_FREE(pid_file);
|
|
|
|
VIR_FREE(remote_config_file);
|
2011-08-29 02:52:23 +00:00
|
|
|
VIR_FREE(run_dir);
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
daemonConfigFree(config);
|
|
|
|
|
2013-01-16 02:53:03 +00:00
|
|
|
virStateCleanup();
|
|
|
|
|
2007-02-23 12:48:36 +00:00
|
|
|
return ret;
|
2007-02-14 01:40:09 +00:00
|
|
|
}
|