2010-12-01 16:36:40 +00:00
|
|
|
/*
|
|
|
|
* virnetserverprogram.c: generic network RPC server program
|
|
|
|
*
|
2012-09-12 23:43:26 +02:00
|
|
|
* Copyright (C) 2006-2012 Red Hat, Inc.
|
2010-12-01 16:36:40 +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 16:30:55 -06:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 18:06:23 +08:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2010-12-01 16:36:40 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "virnetserverprogram.h"
|
|
|
|
#include "virnetserverclient.h"
|
|
|
|
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2011-10-21 12:12:28 +01:00
|
|
|
#include "virfile.h"
|
2012-12-13 15:49:48 +00:00
|
|
|
#include "virthread.h"
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_RPC
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("rpc.netserverprogram");
|
|
|
|
|
2010-12-01 16:36:40 +00:00
|
|
|
struct _virNetServerProgram {
|
2018-04-13 13:51:23 +02:00
|
|
|
virObject parent;
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
unsigned program;
|
|
|
|
unsigned version;
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramProc *procs;
|
2010-12-01 16:36:40 +00:00
|
|
|
size_t nprocs;
|
|
|
|
};
|
|
|
|
|
2012-07-11 14:35:52 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virClass *virNetServerProgramClass;
|
2012-07-11 14:35:52 +01:00
|
|
|
static void virNetServerProgramDispose(void *obj);
|
|
|
|
|
|
|
|
static int virNetServerProgramOnceInit(void)
|
|
|
|
{
|
2018-04-17 17:42:33 +02:00
|
|
|
if (!VIR_CLASS_NEW(virNetServerProgram, virClassForObject()))
|
2012-07-11 14:35:52 +01:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-20 12:23:29 -05:00
|
|
|
VIR_ONCE_GLOBAL_INIT(virNetServerProgram);
|
2012-07-11 14:35:52 +01:00
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgram *virNetServerProgramNew(unsigned program,
|
2010-12-01 16:36:40 +00:00
|
|
|
unsigned version,
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramProc *procs,
|
2010-12-01 16:36:40 +00:00
|
|
|
size_t nprocs)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgram *prog;
|
2010-12-01 16:36:40 +00:00
|
|
|
|
2012-07-11 14:35:52 +01:00
|
|
|
if (virNetServerProgramInitialize() < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!(prog = virObjectNew(virNetServerProgramClass)))
|
2010-12-01 16:36:40 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
prog->program = program;
|
|
|
|
prog->version = version;
|
|
|
|
prog->procs = procs;
|
|
|
|
prog->nprocs = nprocs;
|
|
|
|
|
2012-07-11 14:35:52 +01:00
|
|
|
VIR_DEBUG("prog=%p", prog);
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
return prog;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramGetID(virNetServerProgram *prog)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
|
|
|
return prog->program;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramGetVersion(virNetServerProgram *prog)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
|
|
|
return prog->version;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramMatches(virNetServerProgram *prog,
|
|
|
|
virNetMessage *msg)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
|
|
|
if (prog->program == msg->header.prog &&
|
|
|
|
prog->version == msg->header.vers)
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virNetServerProgramProc *virNetServerProgramGetProc(virNetServerProgram *prog,
|
2010-12-01 16:36:40 +00:00
|
|
|
int procedure)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramProc *proc;
|
2012-09-12 23:43:26 +02:00
|
|
|
|
2010-12-01 16:36:40 +00:00
|
|
|
if (procedure < 0)
|
|
|
|
return NULL;
|
|
|
|
if (procedure >= prog->nprocs)
|
|
|
|
return NULL;
|
|
|
|
|
2012-09-12 23:43:26 +02:00
|
|
|
proc = &prog->procs[procedure];
|
|
|
|
|
|
|
|
if (!proc->func)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return proc;
|
2010-12-01 16:36:40 +00:00
|
|
|
}
|
|
|
|
|
2011-08-12 14:04:31 +02:00
|
|
|
unsigned int
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramGetPriority(virNetServerProgram *prog,
|
2011-08-12 14:04:31 +02:00
|
|
|
int procedure)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramProc *proc = virNetServerProgramGetProc(prog, procedure);
|
2011-08-12 14:04:31 +02:00
|
|
|
|
|
|
|
if (!proc)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return proc->priority;
|
|
|
|
}
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
static int
|
2011-08-26 17:11:03 +02:00
|
|
|
virNetServerProgramSendError(unsigned program,
|
|
|
|
unsigned version,
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg,
|
|
|
|
struct virNetMessageError *rerr,
|
2010-12-01 16:36:40 +00:00
|
|
|
int procedure,
|
|
|
|
int type,
|
2016-03-29 17:33:14 +02:00
|
|
|
unsigned int serial)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
2016-03-29 17:33:14 +02:00
|
|
|
VIR_DEBUG("prog=%d ver=%d proc=%d type=%d serial=%u msg=%p rerr=%p",
|
2011-08-26 17:11:03 +02:00
|
|
|
program, version, procedure, type, serial, msg, rerr);
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
virNetMessageSaveError(rerr);
|
|
|
|
|
|
|
|
/* Return header. */
|
2011-08-26 17:11:03 +02:00
|
|
|
msg->header.prog = program;
|
|
|
|
msg->header.vers = version;
|
2010-12-01 16:36:40 +00:00
|
|
|
msg->header.proc = procedure;
|
|
|
|
msg->header.type = type;
|
|
|
|
msg->header.serial = serial;
|
|
|
|
msg->header.status = VIR_NET_ERROR;
|
|
|
|
|
|
|
|
if (virNetMessageEncodeHeader(msg) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virNetMessageEncodePayload(msg, (xdrproc_t)xdr_virNetMessageError, rerr) < 0)
|
|
|
|
goto error;
|
|
|
|
xdr_free((xdrproc_t)xdr_virNetMessageError, (void*)rerr);
|
|
|
|
|
|
|
|
/* Put reply on end of tx queue to send out */
|
|
|
|
if (virNetServerClientSendMessage(client, msg) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 07:52:31 +01:00
|
|
|
error:
|
2010-12-01 16:36:40 +00:00
|
|
|
VIR_WARN("Failed to serialize remote error '%p'", rerr);
|
|
|
|
xdr_free((xdrproc_t)xdr_virNetMessageError, (void*)rerr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @client: the client to send the error to
|
|
|
|
* @req: the message this error is in reply to
|
|
|
|
*
|
|
|
|
* Send an error message to the client
|
|
|
|
*
|
|
|
|
* Returns 0 if the error was sent, -1 upon fatal error
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramSendReplyError(virNetServerProgram *prog,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg,
|
|
|
|
struct virNetMessageError *rerr,
|
|
|
|
struct virNetMessageHeader *req)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* For data streams, errors are sent back as data streams
|
|
|
|
* For method calls, errors are sent back as method replies
|
|
|
|
*/
|
2011-08-26 17:11:03 +02:00
|
|
|
return virNetServerProgramSendError(prog->program,
|
|
|
|
prog->version,
|
2010-12-01 16:36:40 +00:00
|
|
|
client,
|
|
|
|
msg,
|
|
|
|
rerr,
|
|
|
|
req->proc,
|
|
|
|
req->type == VIR_NET_STREAM ? VIR_NET_STREAM : VIR_NET_REPLY,
|
|
|
|
req->serial);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramSendStreamError(virNetServerProgram *prog,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg,
|
|
|
|
struct virNetMessageError *rerr,
|
2010-12-01 16:36:40 +00:00
|
|
|
int procedure,
|
2016-03-29 17:33:14 +02:00
|
|
|
unsigned int serial)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
2011-08-26 17:11:03 +02:00
|
|
|
return virNetServerProgramSendError(prog->program,
|
|
|
|
prog->version,
|
2010-12-01 16:36:40 +00:00
|
|
|
client,
|
|
|
|
msg,
|
|
|
|
rerr,
|
|
|
|
procedure,
|
|
|
|
VIR_NET_STREAM,
|
|
|
|
serial);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramUnknownError(virNetServerClient *client,
|
|
|
|
virNetMessage *msg,
|
|
|
|
struct virNetMessageHeader *req)
|
2011-08-26 17:11:03 +02:00
|
|
|
{
|
|
|
|
virNetMessageError rerr;
|
|
|
|
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
_("Cannot find program %d version %d"), req->prog, req->vers);
|
2011-08-26 17:11:03 +02:00
|
|
|
|
|
|
|
memset(&rerr, 0, sizeof(rerr));
|
|
|
|
return virNetServerProgramSendError(req->prog,
|
|
|
|
req->vers,
|
|
|
|
client,
|
|
|
|
msg,
|
|
|
|
&rerr,
|
|
|
|
req->proc,
|
|
|
|
VIR_NET_REPLY,
|
|
|
|
req->serial);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-01 16:36:40 +00:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramDispatchCall(virNetServerProgram *prog,
|
|
|
|
virNetServer *server,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg);
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* @server: the unlocked server object
|
|
|
|
* @client: the unlocked client object
|
|
|
|
* @msg: the complete incoming message packet, with header already decoded
|
|
|
|
*
|
|
|
|
* This function is intended to be called from worker threads
|
|
|
|
* when an incoming message is ready to be dispatched for
|
|
|
|
* execution.
|
|
|
|
*
|
|
|
|
* Upon successful return the '@msg' instance will be released
|
|
|
|
* by this function (or more often, reused to send a reply).
|
|
|
|
* Upon failure, the '@msg' must be freed by the caller.
|
|
|
|
*
|
|
|
|
* Returns 0 if the message was dispatched, -1 upon fatal error
|
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramDispatch(virNetServerProgram *prog,
|
|
|
|
virNetServer *server,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
virNetMessageError rerr;
|
|
|
|
|
|
|
|
memset(&rerr, 0, sizeof(rerr));
|
|
|
|
|
2016-03-29 17:33:14 +02:00
|
|
|
VIR_DEBUG("prog=%d ver=%d type=%d status=%d serial=%u proc=%d",
|
2010-12-01 16:36:40 +00:00
|
|
|
msg->header.prog, msg->header.vers, msg->header.type,
|
|
|
|
msg->header.status, msg->header.serial, msg->header.proc);
|
|
|
|
|
|
|
|
/* Check version, etc. */
|
|
|
|
if (msg->header.prog != prog->program) {
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
_("program mismatch (actual %x, expected %x)"),
|
|
|
|
msg->header.prog, prog->program);
|
2010-12-01 16:36:40 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg->header.vers != prog->version) {
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
_("version mismatch (actual %x, expected %x)"),
|
|
|
|
msg->header.vers, prog->version);
|
2010-12-01 16:36:40 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (msg->header.type) {
|
|
|
|
case VIR_NET_CALL:
|
2011-10-21 12:12:28 +01:00
|
|
|
case VIR_NET_CALL_WITH_FDS:
|
2010-12-01 16:36:40 +00:00
|
|
|
ret = virNetServerProgramDispatchCall(prog, server, client, msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_NET_STREAM:
|
|
|
|
/* Since stream data is non-acked, async, we may continue to receive
|
|
|
|
* stream packets after we closed down a stream. Just drop & ignore
|
|
|
|
* these.
|
|
|
|
*/
|
2016-03-29 17:33:14 +02:00
|
|
|
VIR_INFO("Ignoring unexpected stream data serial=%u proc=%d status=%d",
|
2011-08-16 09:44:28 -07:00
|
|
|
msg->header.serial, msg->header.proc, msg->header.status);
|
|
|
|
/* Send a dummy reply to free up 'msg' & unblock client rx */
|
Fix tracking of RPC messages wrt streams
Commit 2c85644b0b51fbe5b6244e6773531af29933a727 attempted to
fix a problem with tracking RPC messages from streams by doing
- if (msg->header.type == VIR_NET_REPLY) {
+ if (msg->header.type == VIR_NET_REPLY ||
+ (msg->header.type == VIR_NET_STREAM &&
+ msg->header.status != VIR_NET_CONTINUE)) {
client->nrequests--;
In other words any stream packet, with status NET_OK or NET_ERROR
would cause nrequests to be decremented. This is great if the
packet from from a synchronous virStreamFinish or virStreamAbort
API call, but wildly wrong if from a server initiated abort.
The latter resulted in 'nrequests' being decremented below zero.
This then causes all I/O for that client to be stopped.
Instead of trying to infer whether we need to decrement the
nrequests field, from the message type/status, introduce an
explicit 'bool tracked' field to mark whether the virNetMessagePtr
object is subject to tracking.
Also add a virNetMessageClear function to allow a message
contents to be cleared out, without adversely impacting the
'tracked' field as a naive memset() would do
* src/rpc/virnetmessage.c, src/rpc/virnetmessage.h: Add
a 'bool tracked' field and virNetMessageClear() API
* daemon/remote.c, daemon/stream.c, src/rpc/virnetclientprogram.c,
src/rpc/virnetclientstream.c, src/rpc/virnetserverclient.c,
src/rpc/virnetserverprogram.c: Switch over to use
virNetMessageClear() and pass in the 'bool tracked' value
when creating messages.
2011-08-31 17:42:58 +01:00
|
|
|
virNetMessageClear(msg);
|
2011-08-16 09:44:28 -07:00
|
|
|
msg->header.type = VIR_NET_REPLY;
|
2019-10-21 15:19:01 -03:00
|
|
|
if (virNetServerClientSendMessage(client, msg) < 0)
|
|
|
|
return -1;
|
2010-12-01 16:36:40 +00:00
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
|
2018-02-14 09:43:59 +00:00
|
|
|
case VIR_NET_REPLY:
|
|
|
|
case VIR_NET_REPLY_WITH_FDS:
|
|
|
|
case VIR_NET_MESSAGE:
|
|
|
|
case VIR_NET_STREAM_HOLE:
|
2010-12-01 16:36:40 +00:00
|
|
|
default:
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
_("Unexpected message type %u"),
|
|
|
|
msg->header.type);
|
2010-12-01 16:36:40 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
2014-03-25 07:52:31 +01:00
|
|
|
error:
|
2011-10-21 12:12:28 +01:00
|
|
|
if (msg->header.type == VIR_NET_CALL ||
|
|
|
|
msg->header.type == VIR_NET_CALL_WITH_FDS) {
|
2011-10-06 10:58:05 +01:00
|
|
|
ret = virNetServerProgramSendReplyError(prog, client, msg, &rerr, &msg->header);
|
|
|
|
} else {
|
|
|
|
/* Send a dummy reply to free up 'msg' & unblock client rx */
|
|
|
|
virNetMessageClear(msg);
|
|
|
|
msg->header.type = VIR_NET_REPLY;
|
2019-10-21 15:19:01 -03:00
|
|
|
if (virNetServerClientSendMessage(client, msg) < 0)
|
|
|
|
return -1;
|
2011-10-06 10:58:05 +01:00
|
|
|
ret = 0;
|
|
|
|
}
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @server: the unlocked server object
|
|
|
|
* @client: the unlocked client object
|
|
|
|
* @msg: the complete incoming method call, with header already decoded
|
|
|
|
*
|
2011-12-03 17:06:07 -07:00
|
|
|
* This method is used to dispatch a message representing an
|
2010-12-01 16:36:40 +00:00
|
|
|
* incoming method call from a client. It decodes the payload
|
2020-07-09 12:42:21 +08:00
|
|
|
* to obtain method call arguments, invokes the method and
|
2010-12-01 16:36:40 +00:00
|
|
|
* then sends a reply packet with the return values
|
|
|
|
*
|
|
|
|
* Returns 0 if the reply was sent, or -1 upon fatal error
|
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramDispatchCall(virNetServerProgram *prog,
|
|
|
|
virNetServer *server,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
2019-10-01 15:37:09 +01:00
|
|
|
g_autofree char *arg = NULL;
|
|
|
|
g_autofree char *ret = NULL;
|
2010-12-01 16:36:40 +00:00
|
|
|
int rv = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetServerProgramProc *dispatcher;
|
2010-12-01 16:36:40 +00:00
|
|
|
virNetMessageError rerr;
|
2011-10-21 12:12:28 +01:00
|
|
|
size_t i;
|
2019-10-01 15:37:09 +01:00
|
|
|
g_autoptr(virIdentity) identity = NULL;
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
memset(&rerr, 0, sizeof(rerr));
|
|
|
|
|
|
|
|
if (msg->header.status != VIR_NET_OK) {
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
_("Unexpected message status %u"),
|
|
|
|
msg->header.status);
|
2010-12-01 16:36:40 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispatcher = virNetServerProgramGetProc(prog, msg->header.proc);
|
|
|
|
|
|
|
|
if (!dispatcher) {
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
_("unknown procedure: %d"),
|
|
|
|
msg->header.proc);
|
2010-12-01 16:36:40 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2017-12-21 15:29:08 +01:00
|
|
|
/* If the client is not authenticated, don't allow any RPC ops
|
|
|
|
* which are except for authentication ones */
|
2017-12-21 15:29:01 +01:00
|
|
|
if (dispatcher->needAuth &&
|
2017-12-21 15:29:08 +01:00
|
|
|
!virNetServerClientIsAuthenticated(client)) {
|
2010-12-01 16:36:40 +00:00
|
|
|
/* Explicitly *NOT* calling remoteDispatchAuthError() because
|
2012-02-03 19:20:22 +01:00
|
|
|
we want back-compatibility with libvirt clients which don't
|
2010-12-01 16:36:40 +00:00
|
|
|
support the VIR_ERR_AUTH_FAILED error code */
|
2012-07-18 11:41:47 +01:00
|
|
|
virReportError(VIR_ERR_RPC,
|
|
|
|
"%s", _("authentication required"));
|
2010-12-01 16:36:40 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2020-09-24 20:58:46 +02:00
|
|
|
arg = g_new0(char, dispatcher->arg_len);
|
|
|
|
ret = g_new0(char, dispatcher->ret_len);
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
if (virNetMessageDecodePayload(msg, dispatcher->arg_filter, arg) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2012-01-23 15:07:15 +00:00
|
|
|
if (!(identity = virNetServerClientGetIdentity(client)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virIdentitySetCurrent(identity) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2010-12-01 16:36:40 +00:00
|
|
|
/*
|
|
|
|
* When the RPC handler is called:
|
|
|
|
*
|
|
|
|
* - Server object is unlocked
|
|
|
|
* - Client object is unlocked
|
|
|
|
*
|
|
|
|
* Without locking, it is safe to use:
|
|
|
|
*
|
|
|
|
* 'args and 'ret'
|
|
|
|
*/
|
2011-10-21 12:12:28 +01:00
|
|
|
rv = (dispatcher->func)(server, client, msg, &rerr, arg, ret);
|
|
|
|
|
2012-01-23 15:07:15 +00:00
|
|
|
if (virIdentitySetCurrent(NULL) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2011-10-21 12:12:28 +01:00
|
|
|
/*
|
2012-12-21 14:18:28 +00:00
|
|
|
* If rv == 1, this indicates the dispatch func has
|
|
|
|
* populated 'msg' with a list of FDs to return to
|
|
|
|
* the caller.
|
|
|
|
*
|
|
|
|
* Otherwise we must clear out the FDs we got from
|
|
|
|
* the client originally.
|
2011-10-21 12:12:28 +01:00
|
|
|
*
|
|
|
|
*/
|
2012-12-21 14:18:28 +00:00
|
|
|
if (rv != 1) {
|
2013-05-21 15:59:54 +08:00
|
|
|
for (i = 0; i < msg->nfds; i++)
|
2012-12-21 14:18:28 +00:00
|
|
|
VIR_FORCE_CLOSE(msg->fds[i]);
|
|
|
|
VIR_FREE(msg->fds);
|
|
|
|
msg->nfds = 0;
|
|
|
|
}
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
xdr_free(dispatcher->arg_filter, arg);
|
|
|
|
|
|
|
|
if (rv < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
/* Return header. We're re-using same message object, so
|
|
|
|
* only need to tweak type/status fields */
|
|
|
|
/*msg->header.prog = msg->header.prog;*/
|
|
|
|
/*msg->header.vers = msg->header.vers;*/
|
|
|
|
/*msg->header.proc = msg->header.proc;*/
|
2011-10-21 12:12:28 +01:00
|
|
|
msg->header.type = msg->nfds ? VIR_NET_REPLY_WITH_FDS : VIR_NET_REPLY;
|
2010-12-01 16:36:40 +00:00
|
|
|
/*msg->header.serial = msg->header.serial;*/
|
|
|
|
msg->header.status = VIR_NET_OK;
|
|
|
|
|
|
|
|
if (virNetMessageEncodeHeader(msg) < 0) {
|
|
|
|
xdr_free(dispatcher->ret_filter, ret);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2011-10-21 12:12:28 +01:00
|
|
|
if (msg->nfds &&
|
|
|
|
virNetMessageEncodeNumFDs(msg) < 0) {
|
|
|
|
xdr_free(dispatcher->ret_filter, ret);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2010-12-01 16:36:40 +00:00
|
|
|
if (virNetMessageEncodePayload(msg, dispatcher->ret_filter, ret) < 0) {
|
|
|
|
xdr_free(dispatcher->ret_filter, ret);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
xdr_free(dispatcher->ret_filter, ret);
|
|
|
|
|
|
|
|
/* Put reply on end of tx queue to send out */
|
|
|
|
return virNetServerClientSendMessage(client, msg);
|
|
|
|
|
2014-03-25 07:52:31 +01:00
|
|
|
error:
|
2010-12-01 16:36:40 +00:00
|
|
|
/* Bad stuff (de-)serializing message, but we have an
|
|
|
|
* RPC error message we can send back to the client */
|
|
|
|
rv = virNetServerProgramSendReplyError(prog, client, msg, &rerr, &msg->header);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramSendStreamData(virNetServerProgram *prog,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg,
|
2010-12-01 16:36:40 +00:00
|
|
|
int procedure,
|
2016-03-29 17:33:14 +02:00
|
|
|
unsigned int serial,
|
2010-12-01 16:36:40 +00:00
|
|
|
const char *data,
|
|
|
|
size_t len)
|
|
|
|
{
|
|
|
|
VIR_DEBUG("client=%p msg=%p data=%p len=%zu", client, msg, data, len);
|
|
|
|
|
|
|
|
/* Return header. We're reusing same message object, so
|
|
|
|
* only need to tweak type/status fields */
|
|
|
|
msg->header.prog = prog->program;
|
|
|
|
msg->header.vers = prog->version;
|
|
|
|
msg->header.proc = procedure;
|
|
|
|
msg->header.type = VIR_NET_STREAM;
|
|
|
|
msg->header.serial = serial;
|
|
|
|
/*
|
|
|
|
* NB
|
2016-04-04 16:44:27 +02:00
|
|
|
* data != NULL + len > 0 => VIR_NET_CONTINUE (Sending back data)
|
|
|
|
* data != NULL + len == 0 => VIR_NET_CONTINUE (Sending read EOF)
|
|
|
|
* data == NULL => VIR_NET_OK (Sending finish handshake confirmation)
|
2010-12-01 16:36:40 +00:00
|
|
|
*/
|
|
|
|
msg->header.status = data ? VIR_NET_CONTINUE : VIR_NET_OK;
|
|
|
|
|
|
|
|
if (virNetMessageEncodeHeader(msg) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (data && len) {
|
|
|
|
if (virNetMessageEncodePayloadRaw(msg, data, len) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2011-06-28 17:42:06 +01:00
|
|
|
} else {
|
|
|
|
if (virNetMessageEncodePayloadEmpty(msg) < 0)
|
|
|
|
return -1;
|
2010-12-01 16:36:40 +00:00
|
|
|
}
|
2011-08-28 13:46:52 +08:00
|
|
|
VIR_DEBUG("Total %zu", msg->bufferLength);
|
2010-12-01 16:36:40 +00:00
|
|
|
|
|
|
|
return virNetServerClientSendMessage(client, msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int virNetServerProgramSendStreamHole(virNetServerProgram *prog,
|
|
|
|
virNetServerClient *client,
|
|
|
|
virNetMessage *msg,
|
2016-04-11 11:44:42 +02:00
|
|
|
int procedure,
|
|
|
|
unsigned int serial,
|
|
|
|
long long length,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
virNetStreamHole data;
|
|
|
|
|
|
|
|
VIR_DEBUG("client=%p msg=%p length=%lld", client, msg, length);
|
|
|
|
|
|
|
|
memset(&data, 0, sizeof(data));
|
|
|
|
data.length = length;
|
|
|
|
data.flags = flags;
|
|
|
|
|
|
|
|
msg->header.prog = prog->program;
|
|
|
|
msg->header.vers = prog->version;
|
|
|
|
msg->header.proc = procedure;
|
|
|
|
msg->header.type = VIR_NET_STREAM_HOLE;
|
|
|
|
msg->header.serial = serial;
|
|
|
|
msg->header.status = VIR_NET_CONTINUE;
|
|
|
|
|
|
|
|
if (virNetMessageEncodeHeader(msg) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virNetMessageEncodePayload(msg,
|
2018-04-25 14:42:34 +02:00
|
|
|
(xdrproc_t)xdr_virNetStreamHole,
|
2016-04-11 11:44:42 +02:00
|
|
|
&data) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return virNetServerClientSendMessage(client, msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-14 14:45:33 +02:00
|
|
|
void virNetServerProgramDispose(void *obj G_GNUC_UNUSED)
|
2010-12-01 16:36:40 +00:00
|
|
|
{
|
|
|
|
}
|