lockd: add support for admin protocol in virtlockd

Add a virtlockd-admin-sock can serves the admin protocol for the virtlockd
daemon and define a virtlockd:///{system,session}  URI scheme for
connecting to it.

Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
This commit is contained in:
Daniel P. Berrange 2018-01-19 14:54:00 +00:00 committed by Daniel P. Berrangé
parent 85d45ff05d
commit ce7ae55ea1
11 changed files with 142 additions and 34 deletions

1
cfg.mk
View File

@ -775,7 +775,6 @@ sc_prohibit_cross_inclusion:
case $$dir in \ case $$dir in \
util/) safe="util";; \ util/) safe="util";; \
access/ | conf/) safe="($$dir|conf|util)";; \ access/ | conf/) safe="($$dir|conf|util)";; \
locking/) safe="($$dir|util|conf|rpc)";; \
cpu/| network/| node_device/| rpc/| security/| storage/) \ cpu/| network/| node_device/| rpc/| security/| storage/) \
safe="($$dir|util|conf|storage)";; \ safe="($$dir|util|conf|storage)";; \
xenapi/ | xenconfig/ ) safe="($$dir|util|conf|xen|cpu)";; \ xenapi/ | xenconfig/ ) safe="($$dir|util|conf|xen|cpu)";; \

View File

@ -2644,6 +2644,7 @@ virtlockd_LDFLAGS = \
$(PIE_LDFLAGS) \ $(PIE_LDFLAGS) \
$(NULL) $(NULL)
virtlockd_LDADD = \ virtlockd_LDADD = \
libvirt_driver_admin.la \
libvirt-net-rpc-server.la \ libvirt-net-rpc-server.la \
libvirt-net-rpc.la \ libvirt-net-rpc.la \
libvirt_util.la \ libvirt_util.la \

View File

@ -40,6 +40,7 @@
#define LIBVIRTD_ADMIN_SOCK_NAME "libvirt-admin-sock" #define LIBVIRTD_ADMIN_SOCK_NAME "libvirt-admin-sock"
#define VIRTLOGD_ADMIN_SOCK_NAME "virtlogd-admin-sock" #define VIRTLOGD_ADMIN_SOCK_NAME "virtlogd-admin-sock"
#define VIRTLOCKD_ADMIN_SOCK_NAME "virtlockd-admin-sock"
VIR_LOG_INIT("libvirt-admin"); VIR_LOG_INIT("libvirt-admin");
@ -134,6 +135,8 @@ getSocketPath(virURIPtr uri)
sockbase = LIBVIRTD_ADMIN_SOCK_NAME; sockbase = LIBVIRTD_ADMIN_SOCK_NAME;
} else if (STREQ_NULLABLE(uri->scheme, "virtlogd")) { } else if (STREQ_NULLABLE(uri->scheme, "virtlogd")) {
sockbase = VIRTLOGD_ADMIN_SOCK_NAME; sockbase = VIRTLOGD_ADMIN_SOCK_NAME;
} else if (STREQ_NULLABLE(uri->scheme, "virtlockd")) {
sockbase = VIRTLOCKD_ADMIN_SOCK_NAME;
} else { } else {
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
_("Unsupported URI scheme '%s'"), _("Unsupported URI scheme '%s'"),

View File

@ -32,6 +32,7 @@
#include "lock_daemon.h" #include "lock_daemon.h"
#include "lock_daemon_config.h" #include "lock_daemon_config.h"
#include "admin/admin_server_dispatch.h"
#include "virutil.h" #include "virutil.h"
#include "virfile.h" #include "virfile.h"
#include "virpidfile.h" #include "virpidfile.h"
@ -148,7 +149,7 @@ static virLockDaemonPtr
virLockDaemonNew(virLockDaemonConfigPtr config, bool privileged) virLockDaemonNew(virLockDaemonConfigPtr config, bool privileged)
{ {
virLockDaemonPtr lockd; virLockDaemonPtr lockd;
virNetServerPtr srv; virNetServerPtr srv = NULL;
if (VIR_ALLOC(lockd) < 0) if (VIR_ALLOC(lockd) < 0)
return NULL; return NULL;
@ -160,6 +161,9 @@ virLockDaemonNew(virLockDaemonConfigPtr config, bool privileged)
return NULL; return NULL;
} }
if (!(lockd->dmn = virNetDaemonNew()))
goto error;
if (!(srv = virNetServerNew("virtlockd", 1, if (!(srv = virNetServerNew("virtlockd", 1,
1, 1, 0, config->max_clients, 1, 1, 0, config->max_clients,
config->max_clients, -1, 0, config->max_clients, -1, 0,
@ -170,8 +174,22 @@ virLockDaemonNew(virLockDaemonConfigPtr config, bool privileged)
(void*)(intptr_t)(privileged ? 0x1 : 0x0)))) (void*)(intptr_t)(privileged ? 0x1 : 0x0))))
goto error; goto error;
if (!(lockd->dmn = virNetDaemonNew()) || if (virNetDaemonAddServer(lockd->dmn, srv) < 0)
virNetDaemonAddServer(lockd->dmn, srv) < 0) goto error;
virObjectUnref(srv);
srv = NULL;
if (!(srv = virNetServerNew("admin", 1,
1, 1, 0, config->admin_max_clients,
config->admin_max_clients, -1, 0,
NULL,
remoteAdmClientNew,
remoteAdmClientPreExecRestart,
remoteAdmClientFree,
lockd->dmn)))
goto error;
if (virNetDaemonAddServer(lockd->dmn, srv) < 0)
goto error; goto error;
virObjectUnref(srv); virObjectUnref(srv);
srv = NULL; srv = NULL;
@ -206,6 +224,14 @@ virLockDaemonNewServerPostExecRestart(virNetDaemonPtr dmn ATTRIBUTE_UNUSED,
virLockDaemonClientPreExecRestart, virLockDaemonClientPreExecRestart,
virLockDaemonClientFree, virLockDaemonClientFree,
opaque); opaque);
} else if (STREQ(name, "admin")) {
return virNetServerNewPostExecRestart(object,
name,
remoteAdmClientNew,
remoteAdmClientNewPostExecRestart,
remoteAdmClientPreExecRestart,
remoteAdmClientFree,
dmn);
} else { } else {
virReportError(VIR_ERR_INTERNAL_ERROR, virReportError(VIR_ERR_INTERNAL_ERROR,
_("Unexpected server name '%s' during restart"), _("Unexpected server name '%s' during restart"),
@ -420,10 +446,12 @@ virLockDaemonForkIntoBackground(const char *argv0)
static int static int
virLockDaemonUnixSocketPaths(bool privileged, virLockDaemonUnixSocketPaths(bool privileged,
char **sockfile) char **sockfile,
char **adminSockfile)
{ {
if (privileged) { if (privileged) {
if (VIR_STRDUP(*sockfile, LOCALSTATEDIR "/run/libvirt/virtlockd-sock") < 0) if (VIR_STRDUP(*sockfile, LOCALSTATEDIR "/run/libvirt/virtlockd-sock") < 0 ||
VIR_STRDUP(*adminSockfile, LOCALSTATEDIR "/run/libvirt/virtlockd-admin-sock") < 0)
goto error; goto error;
} else { } else {
char *rundir = NULL; char *rundir = NULL;
@ -440,7 +468,8 @@ virLockDaemonUnixSocketPaths(bool privileged,
} }
umask(old_umask); umask(old_umask);
if (virAsprintf(sockfile, "%s/virtlockd-sock", rundir) < 0) { if (virAsprintf(sockfile, "%s/virtlockd-sock", rundir) < 0 ||
virAsprintf(adminSockfile, "%s/virtlockd-admin-sock", rundir) < 0) {
VIR_FREE(rundir); VIR_FREE(rundir);
goto error; goto error;
} }
@ -557,20 +586,40 @@ virLockDaemonSetupSignals(virNetDaemonPtr dmn)
static int static int
virLockDaemonSetupNetworkingSystemD(virNetServerPtr srv) virLockDaemonSetupNetworkingSystemD(virNetServerPtr lockSrv, virNetServerPtr adminSrv)
{ {
virNetServerServicePtr svc;
unsigned int nfds; unsigned int nfds;
size_t i;
if ((nfds = virGetListenFDs()) == 0) if ((nfds = virGetListenFDs()) == 0)
return 0; return 0;
if (nfds > 1) if (nfds > 2)
VIR_DEBUG("Too many (%d) file descriptors from systemd", nfds); VIR_DEBUG("Too many (%d) file descriptors from systemd", nfds);
nfds = 1;
for (i = 0; i < nfds && i < 2; i++) {
virNetServerServicePtr svc;
char *path = virGetUNIXSocketPath(3 + i);
virNetServerPtr srv;
if (!path)
return -1;
if (strstr(path, "virtlockd-admin-sock")) {
srv = adminSrv;
} else if (strstr(path, "virtlockd-sock")) {
srv = lockSrv;
} else {
virReportError(VIR_ERR_INTERNAL_ERROR,
_("Unknown UNIX socket %s passed in"),
path);
VIR_FREE(path);
return -1;
}
VIR_FREE(path);
/* Systemd passes FDs, starting immediately after stderr, /* Systemd passes FDs, starting immediately after stderr,
* so the first FD we'll get is '3'. */ * so the first FD we'll get is '3'. */
if (!(svc = virNetServerServiceNewFD(3, 0, if (!(svc = virNetServerServiceNewFD(3 + i, 0,
#if WITH_GNUTLS #if WITH_GNUTLS
NULL, NULL,
#endif #endif
@ -581,6 +630,7 @@ virLockDaemonSetupNetworkingSystemD(virNetServerPtr srv)
virObjectUnref(svc); virObjectUnref(svc);
return -1; return -1;
} }
}
return 1; return 1;
} }
@ -1112,8 +1162,10 @@ virLockDaemonUsage(const char *argv0, bool privileged)
} }
int main(int argc, char **argv) { int main(int argc, char **argv) {
virNetServerPtr srv = NULL; virNetServerPtr lockSrv = NULL;
virNetServerPtr adminSrv = NULL;
virNetServerProgramPtr lockProgram = NULL; virNetServerProgramPtr lockProgram = NULL;
virNetServerProgramPtr adminProgram = NULL;
char *remote_config_file = NULL; char *remote_config_file = NULL;
int statuswrite = -1; int statuswrite = -1;
int ret = 1; int ret = 1;
@ -1123,6 +1175,7 @@ int main(int argc, char **argv) {
char *pid_file = NULL; char *pid_file = NULL;
int pid_file_fd = -1; int pid_file_fd = -1;
char *sock_file = NULL; char *sock_file = NULL;
char *admin_sock_file = NULL;
int timeout = -1; /* -t: Shutdown timeout */ int timeout = -1; /* -t: Shutdown timeout */
char *state_file = NULL; char *state_file = NULL;
bool implicit_conf = false; bool implicit_conf = false;
@ -1250,12 +1303,13 @@ int main(int argc, char **argv) {
VIR_DEBUG("Decided on pid file path '%s'", NULLSTR(pid_file)); VIR_DEBUG("Decided on pid file path '%s'", NULLSTR(pid_file));
if (virLockDaemonUnixSocketPaths(privileged, if (virLockDaemonUnixSocketPaths(privileged,
&sock_file) < 0) { &sock_file,
&admin_sock_file) < 0) {
VIR_ERROR(_("Can't determine socket paths")); VIR_ERROR(_("Can't determine socket paths"));
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
VIR_DEBUG("Decided on socket paths '%s'", VIR_DEBUG("Decided on socket paths '%s' and '%s'",
sock_file); sock_file, admin_sock_file);
if (virLockDaemonExecRestartStatePath(privileged, if (virLockDaemonExecRestartStatePath(privileged,
&state_file) < 0) { &state_file) < 0) {
@ -1330,22 +1384,30 @@ int main(int argc, char **argv) {
goto cleanup; goto cleanup;
} }
srv = virNetDaemonGetServer(lockDaemon->dmn, "virtlockd"); lockSrv = virNetDaemonGetServer(lockDaemon->dmn, "virtlockd");
if ((rv = virLockDaemonSetupNetworkingSystemD(srv)) < 0) { adminSrv = virNetDaemonGetServer(lockDaemon->dmn, "admin");
if ((rv = virLockDaemonSetupNetworkingSystemD(lockSrv, adminSrv)) < 0) {
ret = VIR_LOCK_DAEMON_ERR_NETWORK; ret = VIR_LOCK_DAEMON_ERR_NETWORK;
goto cleanup; goto cleanup;
} }
/* Only do this, if systemd did not pass a FD */ /* Only do this, if systemd did not pass a FD */
if (rv == 0 && if (rv == 0) {
virLockDaemonSetupNetworkingNative(srv, sock_file) < 0) { if (virLockDaemonSetupNetworkingNative(lockSrv, sock_file) < 0 ||
virLockDaemonSetupNetworkingNative(adminSrv, admin_sock_file) < 0) {
ret = VIR_LOCK_DAEMON_ERR_NETWORK; ret = VIR_LOCK_DAEMON_ERR_NETWORK;
goto cleanup; goto cleanup;
} }
virObjectUnref(srv); }
virObjectUnref(lockSrv);
virObjectUnref(adminSrv);
} }
srv = virNetDaemonGetServer(lockDaemon->dmn, "virtlockd"); lockSrv = virNetDaemonGetServer(lockDaemon->dmn, "virtlockd");
/* If exec-restarting from old virtlockd, we won't have an
* admin server present */
if (virNetDaemonHasServer(lockDaemon->dmn, "admin"))
adminSrv = virNetDaemonGetServer(lockDaemon->dmn, "admin");
if (timeout != -1) { if (timeout != -1) {
VIR_DEBUG("Registering shutdown timeout %d", timeout); VIR_DEBUG("Registering shutdown timeout %d", timeout);
@ -1366,11 +1428,25 @@ int main(int argc, char **argv) {
goto cleanup; goto cleanup;
} }
if (virNetServerAddProgram(srv, lockProgram) < 0) { if (virNetServerAddProgram(lockSrv, lockProgram) < 0) {
ret = VIR_LOCK_DAEMON_ERR_INIT; ret = VIR_LOCK_DAEMON_ERR_INIT;
goto cleanup; goto cleanup;
} }
if (adminSrv != NULL) {
if (!(adminProgram = virNetServerProgramNew(ADMIN_PROGRAM,
ADMIN_PROTOCOL_VERSION,
adminProcs,
adminNProcs))) {
ret = VIR_LOCK_DAEMON_ERR_INIT;
goto cleanup;
}
if (virNetServerAddProgram(adminSrv, adminProgram) < 0) {
ret = VIR_LOCK_DAEMON_ERR_INIT;
goto cleanup;
}
}
/* Disable error func, now logging is setup */ /* Disable error func, now logging is setup */
virSetErrorFunc(NULL, virLockDaemonErrorHandler); virSetErrorFunc(NULL, virLockDaemonErrorHandler);
@ -1401,8 +1477,10 @@ int main(int argc, char **argv) {
ret = 0; ret = 0;
cleanup: cleanup:
virObjectUnref(srv);
virObjectUnref(lockProgram); virObjectUnref(lockProgram);
virObjectUnref(adminProgram);
virObjectUnref(lockSrv);
virObjectUnref(adminSrv);
virLockDaemonFree(lockDaemon); virLockDaemonFree(lockDaemon);
if (statuswrite != -1) { if (statuswrite != -1) {
if (ret != 0) { if (ret != 0) {
@ -1418,6 +1496,7 @@ int main(int argc, char **argv) {
virPidFileReleasePath(pid_file, pid_file_fd); virPidFileReleasePath(pid_file, pid_file_fd);
VIR_FREE(pid_file); VIR_FREE(pid_file);
VIR_FREE(sock_file); VIR_FREE(sock_file);
VIR_FREE(admin_sock_file);
VIR_FREE(state_file); VIR_FREE(state_file);
VIR_FREE(run_dir); VIR_FREE(run_dir);
return ret; return ret;

View File

@ -72,6 +72,7 @@ virLockDaemonConfigNew(bool privileged ATTRIBUTE_UNUSED)
return NULL; return NULL;
data->max_clients = 1024; data->max_clients = 1024;
data->admin_max_clients = 5000;
return data; return data;
} }
@ -100,6 +101,8 @@ virLockDaemonConfigLoadOptions(virLockDaemonConfigPtr data,
return -1; return -1;
if (virConfGetValueUInt(conf, "max_clients", &data->max_clients) < 0) if (virConfGetValueUInt(conf, "max_clients", &data->max_clients) < 0)
return -1; return -1;
if (virConfGetValueUInt(conf, "admin_max_clients", &data->admin_max_clients) < 0)
return -1;
return 0; return 0;
} }

View File

@ -34,6 +34,7 @@ struct _virLockDaemonConfig {
char *log_filters; char *log_filters;
char *log_outputs; char *log_outputs;
unsigned int max_clients; unsigned int max_clients;
unsigned int admin_max_clients;
}; };

View File

@ -3,6 +3,8 @@ module Test_virtlockd =
log_filters=\"3:remote 4:event\" log_filters=\"3:remote 4:event\"
log_outputs=\"3:syslog:libvirtd\" log_outputs=\"3:syslog:libvirtd\"
log_buffer_size = 64 log_buffer_size = 64
max_clients = 10
admin_max_clients = 10
" "
test Virtlockd.lns get conf = test Virtlockd.lns get conf =
@ -10,3 +12,5 @@ log_buffer_size = 64
{ "log_filters" = "3:remote 4:event" } { "log_filters" = "3:remote 4:event" }
{ "log_outputs" = "3:syslog:libvirtd" } { "log_outputs" = "3:syslog:libvirtd" }
{ "log_buffer_size" = "64" } { "log_buffer_size" = "64" }
{ "max_clients" = "10" }
{ "admin_max_clients" = "10" }

View File

@ -0,0 +1,10 @@
[Unit]
Description=Virtual machine lock manager admin socket
Before=libvirtd.service
[Socket]
ListenStream=@localstatedir@/run/libvirt/virtlockd-admin-sock
Server=virtlockd.service
[Install]
WantedBy=sockets.target

View File

@ -29,6 +29,7 @@ module Virtlockd =
| str_entry "log_outputs" | str_entry "log_outputs"
| int_entry "log_buffer_size" | int_entry "log_buffer_size"
| int_entry "max_clients" | int_entry "max_clients"
| int_entry "admin_max_clients"
(* Each enty in the config is one of the following three ... *) (* Each enty in the config is one of the following three ... *)
let entry = logging_entry let entry = logging_entry

View File

@ -65,3 +65,9 @@
# to virtlockd. So 'max_clients' will affect how many VMs can # to virtlockd. So 'max_clients' will affect how many VMs can
# be run on a host # be run on a host
#max_clients = 1024 #max_clients = 1024
# Same processing controls, but this time for the admin interface.
# For description of each option, be so kind to scroll few lines
# upwards.
#admin_max_clients = 5

View File

@ -1,6 +1,7 @@
[Unit] [Unit]
Description=Virtual machine lock manager Description=Virtual machine lock manager
Requires=virtlockd.socket Requires=virtlockd.socket
Requires=virtlockd-admin.socket
Before=libvirtd.service Before=libvirtd.service
Documentation=man:virtlockd(8) Documentation=man:virtlockd(8)
Documentation=https://libvirt.org Documentation=https://libvirt.org