mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-05 13:17:51 +03:00
Implement keepalive protocol in libvirt daemon
This commit is contained in:
parent
fd7e85ac6a
commit
f4324e3292
@ -66,6 +66,10 @@ module Libvirtd =
|
|||||||
let auditing_entry = int_entry "audit_level"
|
let auditing_entry = int_entry "audit_level"
|
||||||
| bool_entry "audit_logging"
|
| bool_entry "audit_logging"
|
||||||
|
|
||||||
|
let keepalive_entry = int_entry "keepalive_interval"
|
||||||
|
| int_entry "keepalive_count"
|
||||||
|
| bool_entry "keepalive_required"
|
||||||
|
|
||||||
(* Each enty in the config is one of the following three ... *)
|
(* Each enty in the config is one of the following three ... *)
|
||||||
let entry = network_entry
|
let entry = network_entry
|
||||||
| sock_acl_entry
|
| sock_acl_entry
|
||||||
@ -75,6 +79,7 @@ module Libvirtd =
|
|||||||
| processing_entry
|
| processing_entry
|
||||||
| logging_entry
|
| logging_entry
|
||||||
| auditing_entry
|
| auditing_entry
|
||||||
|
| keepalive_entry
|
||||||
let comment = [ label "#comment" . del /#[ \t]*/ "# " . store /([^ \t\n][^\n]*)?/ . del /\n/ "\n" ]
|
let comment = [ label "#comment" . del /#[ \t]*/ "# " . store /([^ \t\n][^\n]*)?/ . del /\n/ "\n" ]
|
||||||
let empty = [ label "#empty" . eol ]
|
let empty = [ label "#empty" . eol ]
|
||||||
|
|
||||||
|
@ -146,6 +146,10 @@ struct daemonConfig {
|
|||||||
|
|
||||||
int audit_level;
|
int audit_level;
|
||||||
int audit_logging;
|
int audit_logging;
|
||||||
|
|
||||||
|
int keepalive_interval;
|
||||||
|
unsigned int keepalive_count;
|
||||||
|
int keepalive_required;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@ -899,6 +903,10 @@ daemonConfigNew(bool privileged ATTRIBUTE_UNUSED)
|
|||||||
data->audit_level = 1;
|
data->audit_level = 1;
|
||||||
data->audit_logging = 0;
|
data->audit_logging = 0;
|
||||||
|
|
||||||
|
data->keepalive_interval = 5;
|
||||||
|
data->keepalive_count = 5;
|
||||||
|
data->keepalive_required = 0;
|
||||||
|
|
||||||
localhost = virGetHostname(NULL);
|
localhost = virGetHostname(NULL);
|
||||||
if (localhost == NULL) {
|
if (localhost == NULL) {
|
||||||
/* we couldn't resolve the hostname; assume that we are
|
/* we couldn't resolve the hostname; assume that we are
|
||||||
@ -1062,6 +1070,10 @@ daemonConfigLoad(struct daemonConfig *data,
|
|||||||
GET_CONF_STR (conf, filename, log_outputs);
|
GET_CONF_STR (conf, filename, log_outputs);
|
||||||
GET_CONF_INT (conf, filename, log_buffer_size);
|
GET_CONF_INT (conf, filename, log_buffer_size);
|
||||||
|
|
||||||
|
GET_CONF_INT (conf, filename, keepalive_interval);
|
||||||
|
GET_CONF_INT (conf, filename, keepalive_count);
|
||||||
|
GET_CONF_INT (conf, filename, keepalive_required);
|
||||||
|
|
||||||
virConfFree (conf);
|
virConfFree (conf);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -1452,6 +1464,9 @@ int main(int argc, char **argv) {
|
|||||||
config->max_workers,
|
config->max_workers,
|
||||||
config->prio_workers,
|
config->prio_workers,
|
||||||
config->max_clients,
|
config->max_clients,
|
||||||
|
config->keepalive_interval,
|
||||||
|
config->keepalive_count,
|
||||||
|
!!config->keepalive_required,
|
||||||
config->mdns_adv ? config->mdns_name : NULL,
|
config->mdns_adv ? config->mdns_name : NULL,
|
||||||
use_polkit_dbus,
|
use_polkit_dbus,
|
||||||
remoteClientInitHook))) {
|
remoteClientInitHook))) {
|
||||||
|
@ -366,3 +366,28 @@
|
|||||||
# it with the output of the 'uuidgen' command and then
|
# it with the output of the 'uuidgen' command and then
|
||||||
# uncomment this entry
|
# uncomment this entry
|
||||||
#host_uuid = "00000000-0000-0000-0000-000000000000"
|
#host_uuid = "00000000-0000-0000-0000-000000000000"
|
||||||
|
|
||||||
|
###################################################################
|
||||||
|
# Keepalive protocol:
|
||||||
|
# This allows libvirtd to detect broken client connections or even
|
||||||
|
# dead client. A keepalive message is sent to a client after
|
||||||
|
# keepalive_interval seconds of inactivity to check if the client is
|
||||||
|
# still responding; keepalive_count is a maximum number of keepalive
|
||||||
|
# messages that are allowed to be sent to the client without getting
|
||||||
|
# any response before the connection is considered broken. In other
|
||||||
|
# words, the connection is automatically closed approximately after
|
||||||
|
# keepalive_interval * (keepalive_count + 1) seconds since the last
|
||||||
|
# message received from the client. If keepalive_interval is set to
|
||||||
|
# -1, libvirtd will never send keepalive requests; however clients
|
||||||
|
# can still send them and the deamon will send responses. When
|
||||||
|
# keepalive_count is set to 0, connections will be automatically
|
||||||
|
# closed after keepalive_interval seconds of inactivity without
|
||||||
|
# sending any keepalive messages.
|
||||||
|
#
|
||||||
|
#keepalive_interval = 5
|
||||||
|
#keepalive_count = 5
|
||||||
|
#
|
||||||
|
# If set to 1, libvirtd will refuse to talk to clients that do not
|
||||||
|
# support keepalive protocol. Defaults to 0.
|
||||||
|
#
|
||||||
|
#keepalive_required = 1
|
||||||
|
@ -61,6 +61,7 @@ struct daemonClientPrivate {
|
|||||||
virConnectPtr conn;
|
virConnectPtr conn;
|
||||||
|
|
||||||
daemonClientStreamPtr streams;
|
daemonClientStreamPtr streams;
|
||||||
|
bool keepalive_supported;
|
||||||
};
|
};
|
||||||
|
|
||||||
# if HAVE_SASL
|
# if HAVE_SASL
|
||||||
|
@ -581,7 +581,7 @@ int remoteClientInitHook(virNetServerPtr srv ATTRIBUTE_UNUSED,
|
|||||||
/*----- Functions. -----*/
|
/*----- Functions. -----*/
|
||||||
|
|
||||||
static int
|
static int
|
||||||
remoteDispatchOpen(virNetServerPtr server ATTRIBUTE_UNUSED,
|
remoteDispatchOpen(virNetServerPtr server,
|
||||||
virNetServerClientPtr client,
|
virNetServerClientPtr client,
|
||||||
virNetMessagePtr msg ATTRIBUTE_UNUSED,
|
virNetMessagePtr msg ATTRIBUTE_UNUSED,
|
||||||
virNetMessageErrorPtr rerr,
|
virNetMessageErrorPtr rerr,
|
||||||
@ -600,6 +600,12 @@ remoteDispatchOpen(virNetServerPtr server ATTRIBUTE_UNUSED,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (virNetServerKeepAliveRequired(server) && !priv->keepalive_supported) {
|
||||||
|
virNetError(VIR_ERR_OPERATION_FAILED, "%s",
|
||||||
|
_("keepalive support is required to connect"));
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
name = args->name ? *args->name : NULL;
|
name = args->name ? *args->name : NULL;
|
||||||
|
|
||||||
/* If this connection arrived on a readonly socket, force
|
/* If this connection arrived on a readonly socket, force
|
||||||
@ -3226,6 +3232,16 @@ static int remoteDispatchSupportsFeature(
|
|||||||
struct daemonClientPrivate *priv =
|
struct daemonClientPrivate *priv =
|
||||||
virNetServerClientGetPrivateData(client);
|
virNetServerClientGetPrivateData(client);
|
||||||
|
|
||||||
|
/* This feature is checked before opening the connection, thus we must
|
||||||
|
* check it first.
|
||||||
|
*/
|
||||||
|
if (args->feature == VIR_DRV_FEATURE_PROGRAM_KEEPALIVE) {
|
||||||
|
if (virNetServerClientStartKeepAlive(client) < 0)
|
||||||
|
goto cleanup;
|
||||||
|
supported = 1;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
if (!priv->conn) {
|
if (!priv->conn) {
|
||||||
virNetError(VIR_ERR_INTERNAL_ERROR, "%s", _("connection not open"));
|
virNetError(VIR_ERR_INTERNAL_ERROR, "%s", _("connection not open"));
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
@ -3242,6 +3258,7 @@ static int remoteDispatchSupportsFeature(
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
done:
|
||||||
ret->supported = supported;
|
ret->supported = supported;
|
||||||
rv = 0;
|
rv = 0;
|
||||||
|
|
||||||
|
@ -1262,6 +1262,7 @@ virNetServerAutoShutdown;
|
|||||||
virNetServerClose;
|
virNetServerClose;
|
||||||
virNetServerFree;
|
virNetServerFree;
|
||||||
virNetServerIsPrivileged;
|
virNetServerIsPrivileged;
|
||||||
|
virNetServerKeepAliveRequired;
|
||||||
virNetServerNew;
|
virNetServerNew;
|
||||||
virNetServerQuit;
|
virNetServerQuit;
|
||||||
virNetServerRef;
|
virNetServerRef;
|
||||||
@ -1294,6 +1295,7 @@ virNetServerClientSendMessage;
|
|||||||
virNetServerClientSetCloseHook;
|
virNetServerClientSetCloseHook;
|
||||||
virNetServerClientSetIdentity;
|
virNetServerClientSetIdentity;
|
||||||
virNetServerClientSetPrivateData;
|
virNetServerClientSetPrivateData;
|
||||||
|
virNetServerClientStartKeepAlive;
|
||||||
|
|
||||||
|
|
||||||
# virnetserverprogram.h
|
# virnetserverprogram.h
|
||||||
|
@ -102,6 +102,10 @@ struct _virNetServer {
|
|||||||
size_t nclients_max;
|
size_t nclients_max;
|
||||||
virNetServerClientPtr *clients;
|
virNetServerClientPtr *clients;
|
||||||
|
|
||||||
|
int keepaliveInterval;
|
||||||
|
unsigned int keepaliveCount;
|
||||||
|
bool keepaliveRequired;
|
||||||
|
|
||||||
unsigned int quit :1;
|
unsigned int quit :1;
|
||||||
|
|
||||||
virNetTLSContextPtr tls;
|
virNetTLSContextPtr tls;
|
||||||
@ -261,6 +265,9 @@ static int virNetServerDispatchNewClient(virNetServerServicePtr svc ATTRIBUTE_UN
|
|||||||
virNetServerDispatchNewMessage,
|
virNetServerDispatchNewMessage,
|
||||||
srv);
|
srv);
|
||||||
|
|
||||||
|
virNetServerClientInitKeepAlive(client, srv->keepaliveInterval,
|
||||||
|
srv->keepaliveCount);
|
||||||
|
|
||||||
virNetServerUnlock(srv);
|
virNetServerUnlock(srv);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -300,6 +307,9 @@ virNetServerPtr virNetServerNew(size_t min_workers,
|
|||||||
size_t max_workers,
|
size_t max_workers,
|
||||||
size_t priority_workers,
|
size_t priority_workers,
|
||||||
size_t max_clients,
|
size_t max_clients,
|
||||||
|
int keepaliveInterval,
|
||||||
|
unsigned int keepaliveCount,
|
||||||
|
bool keepaliveRequired,
|
||||||
const char *mdnsGroupName,
|
const char *mdnsGroupName,
|
||||||
bool connectDBus ATTRIBUTE_UNUSED,
|
bool connectDBus ATTRIBUTE_UNUSED,
|
||||||
virNetServerClientInitHook clientInitHook)
|
virNetServerClientInitHook clientInitHook)
|
||||||
@ -321,6 +331,9 @@ virNetServerPtr virNetServerNew(size_t min_workers,
|
|||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
srv->nclients_max = max_clients;
|
srv->nclients_max = max_clients;
|
||||||
|
srv->keepaliveInterval = keepaliveInterval;
|
||||||
|
srv->keepaliveCount = keepaliveCount;
|
||||||
|
srv->keepaliveRequired = keepaliveRequired;
|
||||||
srv->sigwrite = srv->sigread = -1;
|
srv->sigwrite = srv->sigread = -1;
|
||||||
srv->clientInitHook = clientInitHook;
|
srv->clientInitHook = clientInitHook;
|
||||||
srv->privileged = geteuid() == 0 ? true : false;
|
srv->privileged = geteuid() == 0 ? true : false;
|
||||||
@ -840,3 +853,12 @@ void virNetServerClose(virNetServerPtr srv)
|
|||||||
|
|
||||||
virNetServerUnlock(srv);
|
virNetServerUnlock(srv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool virNetServerKeepAliveRequired(virNetServerPtr srv)
|
||||||
|
{
|
||||||
|
bool required;
|
||||||
|
virNetServerLock(srv);
|
||||||
|
required = srv->keepaliveRequired;
|
||||||
|
virNetServerUnlock(srv);
|
||||||
|
return required;
|
||||||
|
}
|
||||||
|
@ -41,6 +41,9 @@ virNetServerPtr virNetServerNew(size_t min_workers,
|
|||||||
size_t max_workers,
|
size_t max_workers,
|
||||||
size_t priority_workers,
|
size_t priority_workers,
|
||||||
size_t max_clients,
|
size_t max_clients,
|
||||||
|
int keepaliveInterval,
|
||||||
|
unsigned int keepaliveCount,
|
||||||
|
bool keepaliveRequired,
|
||||||
const char *mdnsGroupName,
|
const char *mdnsGroupName,
|
||||||
bool connectDBus,
|
bool connectDBus,
|
||||||
virNetServerClientInitHook clientInitHook);
|
virNetServerClientInitHook clientInitHook);
|
||||||
@ -88,4 +91,6 @@ void virNetServerFree(virNetServerPtr srv);
|
|||||||
|
|
||||||
void virNetServerClose(virNetServerPtr srv);
|
void virNetServerClose(virNetServerPtr srv);
|
||||||
|
|
||||||
|
bool virNetServerKeepAliveRequired(virNetServerPtr srv);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -33,6 +33,7 @@
|
|||||||
#include "virterror_internal.h"
|
#include "virterror_internal.h"
|
||||||
#include "memory.h"
|
#include "memory.h"
|
||||||
#include "threads.h"
|
#include "threads.h"
|
||||||
|
#include "virkeepalive.h"
|
||||||
|
|
||||||
#define VIR_FROM_THIS VIR_FROM_RPC
|
#define VIR_FROM_THIS VIR_FROM_RPC
|
||||||
#define virNetError(code, ...) \
|
#define virNetError(code, ...) \
|
||||||
@ -100,6 +101,9 @@ struct _virNetServerClient
|
|||||||
void *privateData;
|
void *privateData;
|
||||||
virNetServerClientFreeFunc privateDataFreeFunc;
|
virNetServerClientFreeFunc privateDataFreeFunc;
|
||||||
virNetServerClientCloseFunc privateDataCloseFunc;
|
virNetServerClientCloseFunc privateDataCloseFunc;
|
||||||
|
|
||||||
|
virKeepAlivePtr keepalive;
|
||||||
|
int keepaliveFilter;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -213,15 +217,15 @@ static void virNetServerClientUpdateEvent(virNetServerClientPtr client)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int virNetServerClientAddFilter(virNetServerClientPtr client,
|
static int
|
||||||
virNetServerClientFilterFunc func,
|
virNetServerClientAddFilterLocked(virNetServerClientPtr client,
|
||||||
void *opaque)
|
virNetServerClientFilterFunc func,
|
||||||
|
void *opaque)
|
||||||
{
|
{
|
||||||
virNetServerClientFilterPtr filter;
|
virNetServerClientFilterPtr filter;
|
||||||
|
virNetServerClientFilterPtr *place;
|
||||||
int ret = -1;
|
int ret = -1;
|
||||||
|
|
||||||
virNetServerClientLock(client);
|
|
||||||
|
|
||||||
if (VIR_ALLOC(filter) < 0) {
|
if (VIR_ALLOC(filter) < 0) {
|
||||||
virReportOOMError();
|
virReportOOMError();
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
@ -231,22 +235,34 @@ int virNetServerClientAddFilter(virNetServerClientPtr client,
|
|||||||
filter->func = func;
|
filter->func = func;
|
||||||
filter->opaque = opaque;
|
filter->opaque = opaque;
|
||||||
|
|
||||||
filter->next = client->filters;
|
place = &client->filters;
|
||||||
client->filters = filter;
|
while (*place)
|
||||||
|
place = &(*place)->next;
|
||||||
|
*place = filter;
|
||||||
|
|
||||||
ret = filter->id;
|
ret = filter->id;
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int virNetServerClientAddFilter(virNetServerClientPtr client,
|
||||||
|
virNetServerClientFilterFunc func,
|
||||||
|
void *opaque)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
virNetServerClientLock(client);
|
||||||
|
ret = virNetServerClientAddFilterLocked(client, func, opaque);
|
||||||
virNetServerClientUnlock(client);
|
virNetServerClientUnlock(client);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
void virNetServerClientRemoveFilter(virNetServerClientPtr client,
|
virNetServerClientRemoveFilterLocked(virNetServerClientPtr client,
|
||||||
int filterID)
|
int filterID)
|
||||||
{
|
{
|
||||||
virNetServerClientFilterPtr tmp, prev;
|
virNetServerClientFilterPtr tmp, prev;
|
||||||
virNetServerClientLock(client);
|
|
||||||
|
|
||||||
prev = NULL;
|
prev = NULL;
|
||||||
tmp = client->filters;
|
tmp = client->filters;
|
||||||
@ -263,7 +279,13 @@ void virNetServerClientRemoveFilter(virNetServerClientPtr client,
|
|||||||
prev = tmp;
|
prev = tmp;
|
||||||
tmp = tmp->next;
|
tmp = tmp->next;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void virNetServerClientRemoveFilter(virNetServerClientPtr client,
|
||||||
|
int filterID)
|
||||||
|
{
|
||||||
|
virNetServerClientLock(client);
|
||||||
|
virNetServerClientRemoveFilterLocked(client, filterID);
|
||||||
virNetServerClientUnlock(client);
|
virNetServerClientUnlock(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -337,6 +359,7 @@ virNetServerClientPtr virNetServerClientNew(virNetSocketPtr sock,
|
|||||||
client->readonly = readonly;
|
client->readonly = readonly;
|
||||||
client->tlsCtxt = tls;
|
client->tlsCtxt = tls;
|
||||||
client->nrequests_max = nrequests_max;
|
client->nrequests_max = nrequests_max;
|
||||||
|
client->keepaliveFilter = -1;
|
||||||
|
|
||||||
client->sockTimer = virEventAddTimeout(-1, virNetServerClientSockTimerFunc,
|
client->sockTimer = virEventAddTimeout(-1, virNetServerClientSockTimerFunc,
|
||||||
client, NULL);
|
client, NULL);
|
||||||
@ -603,6 +626,7 @@ void virNetServerClientFree(virNetServerClientPtr client)
|
|||||||
void virNetServerClientClose(virNetServerClientPtr client)
|
void virNetServerClientClose(virNetServerClientPtr client)
|
||||||
{
|
{
|
||||||
virNetServerClientCloseFunc cf;
|
virNetServerClientCloseFunc cf;
|
||||||
|
virKeepAlivePtr ka;
|
||||||
|
|
||||||
virNetServerClientLock(client);
|
virNetServerClientLock(client);
|
||||||
VIR_DEBUG("client=%p refs=%d", client, client->refs);
|
VIR_DEBUG("client=%p refs=%d", client, client->refs);
|
||||||
@ -611,6 +635,20 @@ void virNetServerClientClose(virNetServerClientPtr client)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (client->keepaliveFilter >= 0)
|
||||||
|
virNetServerClientRemoveFilterLocked(client, client->keepaliveFilter);
|
||||||
|
|
||||||
|
if (client->keepalive) {
|
||||||
|
virKeepAliveStop(client->keepalive);
|
||||||
|
ka = client->keepalive;
|
||||||
|
client->keepalive = NULL;
|
||||||
|
client->refs++;
|
||||||
|
virNetServerClientUnlock(client);
|
||||||
|
virKeepAliveFree(ka);
|
||||||
|
virNetServerClientLock(client);
|
||||||
|
client->refs--;
|
||||||
|
}
|
||||||
|
|
||||||
if (client->privateDataCloseFunc) {
|
if (client->privateDataCloseFunc) {
|
||||||
cf = client->privateDataCloseFunc;
|
cf = client->privateDataCloseFunc;
|
||||||
client->refs++;
|
client->refs++;
|
||||||
@ -1066,6 +1104,7 @@ int virNetServerClientSendMessage(virNetServerClientPtr client,
|
|||||||
VIR_DEBUG("msg=%p proc=%d len=%zu offset=%zu",
|
VIR_DEBUG("msg=%p proc=%d len=%zu offset=%zu",
|
||||||
msg, msg->header.proc,
|
msg, msg->header.proc,
|
||||||
msg->bufferLength, msg->bufferOffset);
|
msg->bufferLength, msg->bufferOffset);
|
||||||
|
|
||||||
virNetServerClientLock(client);
|
virNetServerClientLock(client);
|
||||||
|
|
||||||
msg->donefds = 0;
|
msg->donefds = 0;
|
||||||
@ -1082,6 +1121,7 @@ int virNetServerClientSendMessage(virNetServerClientPtr client,
|
|||||||
}
|
}
|
||||||
|
|
||||||
virNetServerClientUnlock(client);
|
virNetServerClientUnlock(client);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1095,3 +1135,84 @@ bool virNetServerClientNeedAuth(virNetServerClientPtr client)
|
|||||||
virNetServerClientUnlock(client);
|
virNetServerClientUnlock(client);
|
||||||
return need;
|
return need;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
virNetServerClientKeepAliveDeadCB(void *opaque)
|
||||||
|
{
|
||||||
|
virNetServerClientImmediateClose(opaque);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
virNetServerClientKeepAliveSendCB(void *opaque,
|
||||||
|
virNetMessagePtr msg)
|
||||||
|
{
|
||||||
|
return virNetServerClientSendMessage(opaque, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
virNetServerClientFreeCB(void *opaque)
|
||||||
|
{
|
||||||
|
virNetServerClientFree(opaque);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
virNetServerClientKeepAliveFilter(virNetServerClientPtr client,
|
||||||
|
virNetMessagePtr msg,
|
||||||
|
void *opaque ATTRIBUTE_UNUSED)
|
||||||
|
{
|
||||||
|
if (virKeepAliveCheckMessage(client->keepalive, msg)) {
|
||||||
|
virNetMessageFree(msg);
|
||||||
|
client->nrequests--;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
virNetServerClientInitKeepAlive(virNetServerClientPtr client,
|
||||||
|
int interval,
|
||||||
|
unsigned int count)
|
||||||
|
{
|
||||||
|
virKeepAlivePtr ka;
|
||||||
|
int ret = -1;
|
||||||
|
|
||||||
|
virNetServerClientLock(client);
|
||||||
|
|
||||||
|
if (!(ka = virKeepAliveNew(interval, count, client,
|
||||||
|
virNetServerClientKeepAliveSendCB,
|
||||||
|
virNetServerClientKeepAliveDeadCB,
|
||||||
|
virNetServerClientFreeCB)))
|
||||||
|
goto cleanup;
|
||||||
|
/* keepalive object has a reference to client */
|
||||||
|
client->refs++;
|
||||||
|
|
||||||
|
client->keepaliveFilter =
|
||||||
|
virNetServerClientAddFilterLocked(client,
|
||||||
|
virNetServerClientKeepAliveFilter,
|
||||||
|
NULL);
|
||||||
|
if (client->keepaliveFilter < 0)
|
||||||
|
goto cleanup;
|
||||||
|
|
||||||
|
client->keepalive = ka;
|
||||||
|
ka = NULL;
|
||||||
|
|
||||||
|
cleanup:
|
||||||
|
virNetServerClientUnlock(client);
|
||||||
|
if (ka)
|
||||||
|
virKeepAliveStop(ka);
|
||||||
|
virKeepAliveFree(ka);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
virNetServerClientStartKeepAlive(virNetServerClientPtr client)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
virNetServerClientLock(client);
|
||||||
|
ret = virKeepAliveStart(client->keepalive, 0, 0);
|
||||||
|
virNetServerClientUnlock(client);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
@ -99,6 +99,13 @@ bool virNetServerClientWantClose(virNetServerClientPtr client);
|
|||||||
|
|
||||||
int virNetServerClientInit(virNetServerClientPtr client);
|
int virNetServerClientInit(virNetServerClientPtr client);
|
||||||
|
|
||||||
|
int virNetServerClientInitKeepAlive(virNetServerClientPtr client,
|
||||||
|
int interval,
|
||||||
|
unsigned int count);
|
||||||
|
bool virNetServerClientCheckKeepAlive(virNetServerClientPtr client,
|
||||||
|
virNetMessagePtr msg);
|
||||||
|
int virNetServerClientStartKeepAlive(virNetServerClientPtr client);
|
||||||
|
|
||||||
const char *virNetServerClientLocalAddrString(virNetServerClientPtr client);
|
const char *virNetServerClientLocalAddrString(virNetServerClientPtr client);
|
||||||
const char *virNetServerClientRemoteAddrString(virNetServerClientPtr client);
|
const char *virNetServerClientRemoteAddrString(virNetServerClientPtr client);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user