2010-12-06 17:03:22 +00:00
|
|
|
/*
|
|
|
|
* virnetmessage.c: basic RPC message encoding/decoding
|
|
|
|
*
|
2012-02-02 22:47:04 +00:00
|
|
|
* Copyright (C) 2010-2012 Red Hat, Inc.
|
2010-12-06 17:03:22 +00:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2011-10-21 10:30:12 +00:00
|
|
|
#include <unistd.h>
|
2010-12-06 17:03:22 +00:00
|
|
|
|
|
|
|
#include "virnetmessage.h"
|
|
|
|
#include "memory.h"
|
|
|
|
#include "virterror_internal.h"
|
|
|
|
#include "logging.h"
|
2011-10-21 10:30:12 +00:00
|
|
|
#include "virfile.h"
|
|
|
|
#include "util.h"
|
2010-12-06 17:03:22 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_RPC
|
|
|
|
#define virNetError(code, ...) \
|
|
|
|
virReportErrorHelper(VIR_FROM_THIS, code, __FILE__, \
|
|
|
|
__FUNCTION__, __LINE__, __VA_ARGS__)
|
|
|
|
|
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 16:42:58 +00:00
|
|
|
virNetMessagePtr virNetMessageNew(bool tracked)
|
2010-12-06 17:03:22 +00:00
|
|
|
{
|
|
|
|
virNetMessagePtr msg;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(msg) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
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 16:42:58 +00:00
|
|
|
msg->tracked = tracked;
|
|
|
|
VIR_DEBUG("msg=%p tracked=%d", msg, tracked);
|
2010-12-06 17:03:22 +00:00
|
|
|
|
|
|
|
return msg;
|
|
|
|
}
|
|
|
|
|
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 16:42:58 +00:00
|
|
|
|
|
|
|
void virNetMessageClear(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
bool tracked = msg->tracked;
|
2011-10-21 10:30:12 +00:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
VIR_DEBUG("msg=%p nfds=%zu", msg, msg->nfds);
|
|
|
|
|
|
|
|
for (i = 0 ; i < msg->nfds ; i++)
|
|
|
|
VIR_FORCE_CLOSE(msg->fds[i]);
|
|
|
|
VIR_FREE(msg->fds);
|
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 16:42:58 +00:00
|
|
|
memset(msg, 0, sizeof(*msg));
|
|
|
|
msg->tracked = tracked;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-06 17:03:22 +00:00
|
|
|
void virNetMessageFree(virNetMessagePtr msg)
|
|
|
|
{
|
2011-10-21 10:30:12 +00:00
|
|
|
size_t i;
|
2010-12-06 17:03:22 +00:00
|
|
|
if (!msg)
|
|
|
|
return;
|
|
|
|
|
2011-10-21 10:30:12 +00:00
|
|
|
VIR_DEBUG("msg=%p nfds=%zu cb=%p", msg, msg->nfds, msg->cb);
|
|
|
|
|
2011-05-16 17:13:11 +00:00
|
|
|
if (msg->cb)
|
|
|
|
msg->cb(msg, msg->opaque);
|
|
|
|
|
2011-10-21 10:30:12 +00:00
|
|
|
for (i = 0 ; i < msg->nfds ; i++)
|
|
|
|
VIR_FORCE_CLOSE(msg->fds[i]);
|
|
|
|
VIR_FREE(msg->fds);
|
2010-12-06 17:03:22 +00:00
|
|
|
VIR_FREE(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void virNetMessageQueuePush(virNetMessagePtr *queue, virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
virNetMessagePtr tmp = *queue;
|
|
|
|
|
|
|
|
if (tmp) {
|
|
|
|
while (tmp->next)
|
|
|
|
tmp = tmp->next;
|
|
|
|
tmp->next = msg;
|
|
|
|
} else {
|
|
|
|
*queue = msg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virNetMessagePtr virNetMessageQueueServe(virNetMessagePtr *queue)
|
|
|
|
{
|
|
|
|
virNetMessagePtr tmp = *queue;
|
|
|
|
|
|
|
|
if (tmp) {
|
|
|
|
*queue = tmp->next;
|
|
|
|
tmp->next = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int virNetMessageDecodeLength(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
unsigned int len;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
xdrmem_create(&xdr, msg->buffer,
|
|
|
|
msg->bufferLength, XDR_DECODE);
|
|
|
|
if (!xdr_u_int(&xdr, &len)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to decode message length"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
msg->bufferOffset = xdr_getpos(&xdr);
|
|
|
|
|
|
|
|
if (len < VIR_NET_MESSAGE_LEN_MAX) {
|
2011-06-28 16:45:32 +00:00
|
|
|
virNetError(VIR_ERR_RPC,
|
|
|
|
_("packet %d bytes received from server too small, want %d"),
|
|
|
|
len, VIR_NET_MESSAGE_LEN_MAX);
|
2010-12-06 17:03:22 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Length includes length word - adjust to real length to read. */
|
|
|
|
len -= VIR_NET_MESSAGE_LEN_MAX;
|
|
|
|
|
|
|
|
if (len > VIR_NET_MESSAGE_MAX) {
|
2011-06-28 16:45:32 +00:00
|
|
|
virNetError(VIR_ERR_RPC,
|
|
|
|
_("packet %d bytes received from server too large, want %d"),
|
|
|
|
len, VIR_NET_MESSAGE_MAX);
|
2010-12-06 17:03:22 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extend our declared buffer length and carry
|
|
|
|
on reading the header + payload */
|
|
|
|
msg->bufferLength += len;
|
|
|
|
|
|
|
|
VIR_DEBUG("Got length, now need %zu total (%u more)",
|
|
|
|
msg->bufferLength, len);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @msg: the complete incoming message, whose header to decode
|
|
|
|
*
|
|
|
|
* Decodes the header part of the message, but does not
|
|
|
|
* validate the decoded fields in the header. It expects
|
|
|
|
* bufferLength to refer to length of the data packet. Upon
|
|
|
|
* return bufferOffset will refer to the amount of the packet
|
|
|
|
* consumed by decoding of the header.
|
|
|
|
*
|
|
|
|
* returns 0 if successfully decoded, -1 upon fatal error
|
|
|
|
*/
|
|
|
|
int virNetMessageDecodeHeader(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
msg->bufferOffset = VIR_NET_MESSAGE_LEN_MAX;
|
|
|
|
|
|
|
|
/* Parse the header. */
|
|
|
|
xdrmem_create(&xdr,
|
|
|
|
msg->buffer + msg->bufferOffset,
|
|
|
|
msg->bufferLength - msg->bufferOffset,
|
|
|
|
XDR_DECODE);
|
|
|
|
|
|
|
|
if (!xdr_virNetMessageHeader(&xdr, &msg->header)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to decode message header"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg->bufferOffset += xdr_getpos(&xdr);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @msg: the outgoing message, whose header to encode
|
|
|
|
*
|
|
|
|
* Encodes the length word and header of the message, setting the
|
|
|
|
* message offset ready to encode the payload. Leaves space
|
|
|
|
* for the length field later. Upon return bufferLength will
|
|
|
|
* refer to the total available space for message, while
|
|
|
|
* bufferOffset will refer to current space used by header
|
|
|
|
*
|
|
|
|
* returns 0 if successfully encoded, -1 upon fatal error
|
|
|
|
*/
|
|
|
|
int virNetMessageEncodeHeader(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
int ret = -1;
|
|
|
|
unsigned int len = 0;
|
|
|
|
|
|
|
|
msg->bufferLength = sizeof(msg->buffer);
|
|
|
|
msg->bufferOffset = 0;
|
|
|
|
|
|
|
|
/* Format the header. */
|
|
|
|
xdrmem_create(&xdr,
|
|
|
|
msg->buffer,
|
|
|
|
msg->bufferLength,
|
|
|
|
XDR_ENCODE);
|
|
|
|
|
|
|
|
/* The real value is filled in shortly */
|
|
|
|
if (!xdr_u_int(&xdr, &len)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode message length"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!xdr_virNetMessageHeader(&xdr, &msg->header)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode message header"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = xdr_getpos(&xdr);
|
|
|
|
xdr_setpos(&xdr, 0);
|
|
|
|
|
|
|
|
/* Fill in current length - may be re-written later
|
|
|
|
* if a payload is added
|
|
|
|
*/
|
|
|
|
if (!xdr_u_int(&xdr, &len)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to re-encode message length"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg->bufferOffset += len;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-21 10:30:12 +00:00
|
|
|
int virNetMessageEncodeNumFDs(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
unsigned int numFDs = msg->nfds;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
xdrmem_create(&xdr, msg->buffer + msg->bufferOffset,
|
|
|
|
msg->bufferLength - msg->bufferOffset, XDR_ENCODE);
|
|
|
|
|
|
|
|
if (numFDs > VIR_NET_MESSAGE_NUM_FDS_MAX) {
|
|
|
|
virNetError(VIR_ERR_RPC,
|
|
|
|
_("Too many FDs to send %d, expected %d maximum"),
|
|
|
|
numFDs, VIR_NET_MESSAGE_NUM_FDS_MAX);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!xdr_u_int(&xdr, &numFDs)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode number of FDs"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
msg->bufferOffset += xdr_getpos(&xdr);
|
|
|
|
|
|
|
|
VIR_DEBUG("Send %zu FDs to peer", msg->nfds);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int virNetMessageDecodeNumFDs(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
unsigned int numFDs;
|
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
xdrmem_create(&xdr, msg->buffer + msg->bufferOffset,
|
|
|
|
msg->bufferLength - msg->bufferOffset, XDR_DECODE);
|
|
|
|
if (!xdr_u_int(&xdr, &numFDs)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to decode number of FDs"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
msg->bufferOffset += xdr_getpos(&xdr);
|
|
|
|
|
|
|
|
if (numFDs > VIR_NET_MESSAGE_NUM_FDS_MAX) {
|
|
|
|
virNetError(VIR_ERR_RPC,
|
|
|
|
_("Received too many FDs %d, expected %d maximum"),
|
|
|
|
numFDs, VIR_NET_MESSAGE_NUM_FDS_MAX);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg->nfds = numFDs;
|
|
|
|
if (VIR_ALLOC_N(msg->fds, msg->nfds) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
for (i = 0 ; i < msg->nfds ; i++)
|
|
|
|
msg->fds[i] = -1;
|
|
|
|
|
|
|
|
VIR_DEBUG("Got %zu FDs from peer", msg->nfds);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-06 17:03:22 +00:00
|
|
|
int virNetMessageEncodePayload(virNetMessagePtr msg,
|
|
|
|
xdrproc_t filter,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
unsigned int msglen;
|
|
|
|
|
|
|
|
/* Serialise payload of the message. This assumes that
|
|
|
|
* virNetMessageEncodeHeader has already been run, so
|
|
|
|
* just appends to that data */
|
|
|
|
xdrmem_create(&xdr, msg->buffer + msg->bufferOffset,
|
|
|
|
msg->bufferLength - msg->bufferOffset, XDR_ENCODE);
|
|
|
|
|
|
|
|
if (!(*filter)(&xdr, data)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode message payload"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the length stored in buffer. */
|
|
|
|
msg->bufferOffset += xdr_getpos(&xdr);
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
|
|
|
|
/* Re-encode the length word. */
|
|
|
|
VIR_DEBUG("Encode length as %zu", msg->bufferOffset);
|
|
|
|
xdrmem_create(&xdr, msg->buffer, VIR_NET_MESSAGE_HEADER_XDR_LEN, XDR_ENCODE);
|
|
|
|
msglen = msg->bufferOffset;
|
|
|
|
if (!xdr_u_int(&xdr, &msglen)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode message length"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
|
|
|
|
msg->bufferLength = msg->bufferOffset;
|
|
|
|
msg->bufferOffset = 0;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int virNetMessageDecodePayload(virNetMessagePtr msg,
|
|
|
|
xdrproc_t filter,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
|
|
|
|
/* Deserialise payload of the message. This assumes that
|
|
|
|
* virNetMessageDecodeHeader has already been run, so
|
|
|
|
* just start from after that data */
|
|
|
|
xdrmem_create(&xdr, msg->buffer + msg->bufferOffset,
|
|
|
|
msg->bufferLength - msg->bufferOffset, XDR_DECODE);
|
|
|
|
|
|
|
|
if (!(*filter)(&xdr, data)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to decode message payload"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the length stored in buffer. */
|
|
|
|
msg->bufferLength += xdr_getpos(&xdr);
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int virNetMessageEncodePayloadRaw(virNetMessagePtr msg,
|
|
|
|
const char *data,
|
|
|
|
size_t len)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
unsigned int msglen;
|
|
|
|
|
|
|
|
if ((msg->bufferLength - msg->bufferOffset) < len) {
|
|
|
|
virNetError(VIR_ERR_RPC,
|
|
|
|
_("Stream data too long to send (%zu bytes needed, %zu bytes available)"),
|
|
|
|
len, (msg->bufferLength - msg->bufferOffset));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(msg->buffer + msg->bufferOffset, data, len);
|
|
|
|
msg->bufferOffset += len;
|
|
|
|
|
|
|
|
/* Re-encode the length word. */
|
|
|
|
VIR_DEBUG("Encode length as %zu", msg->bufferOffset);
|
|
|
|
xdrmem_create(&xdr, msg->buffer, VIR_NET_MESSAGE_HEADER_XDR_LEN, XDR_ENCODE);
|
|
|
|
msglen = msg->bufferOffset;
|
|
|
|
if (!xdr_u_int(&xdr, &msglen)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode message length"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
|
|
|
|
msg->bufferLength = msg->bufferOffset;
|
|
|
|
msg->bufferOffset = 0;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-28 16:42:06 +00:00
|
|
|
int virNetMessageEncodePayloadEmpty(virNetMessagePtr msg)
|
|
|
|
{
|
|
|
|
XDR xdr;
|
|
|
|
unsigned int msglen;
|
|
|
|
|
|
|
|
/* Re-encode the length word. */
|
|
|
|
VIR_DEBUG("Encode length as %zu", msg->bufferOffset);
|
|
|
|
xdrmem_create(&xdr, msg->buffer, VIR_NET_MESSAGE_HEADER_XDR_LEN, XDR_ENCODE);
|
|
|
|
msglen = msg->bufferOffset;
|
|
|
|
if (!xdr_u_int(&xdr, &msglen)) {
|
|
|
|
virNetError(VIR_ERR_RPC, "%s", _("Unable to encode message length"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
|
|
|
|
msg->bufferLength = msg->bufferOffset;
|
|
|
|
msg->bufferOffset = 0;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
xdr_destroy(&xdr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-06 17:03:22 +00:00
|
|
|
void virNetMessageSaveError(virNetMessageErrorPtr rerr)
|
|
|
|
{
|
|
|
|
/* This func may be called several times & the first
|
|
|
|
* error is the one we want because we don't want
|
|
|
|
* cleanup code overwriting the first one.
|
|
|
|
*/
|
|
|
|
if (rerr->code != VIR_ERR_OK)
|
|
|
|
return;
|
|
|
|
|
2012-02-02 22:47:04 +00:00
|
|
|
memset(rerr, 0, sizeof(*rerr));
|
2010-12-06 17:03:22 +00:00
|
|
|
virErrorPtr verr = virGetLastError();
|
|
|
|
if (verr) {
|
|
|
|
rerr->code = verr->code;
|
|
|
|
rerr->domain = verr->domain;
|
2012-02-02 22:47:04 +00:00
|
|
|
if (verr->message && VIR_ALLOC(rerr->message) == 0)
|
|
|
|
*rerr->message = strdup(verr->message);
|
2010-12-06 17:03:22 +00:00
|
|
|
rerr->level = verr->level;
|
2012-02-02 22:47:04 +00:00
|
|
|
if (verr->str1 && VIR_ALLOC(rerr->str1) == 0)
|
|
|
|
*rerr->str1 = strdup(verr->str1);
|
|
|
|
if (verr->str2 && VIR_ALLOC(rerr->str2) == 0)
|
|
|
|
*rerr->str2 = strdup(verr->str2);
|
|
|
|
if (verr->str3 && VIR_ALLOC(rerr->str3) == 0)
|
|
|
|
*rerr->str3 = strdup(verr->str3);
|
2010-12-06 17:03:22 +00:00
|
|
|
rerr->int1 = verr->int1;
|
|
|
|
rerr->int2 = verr->int2;
|
|
|
|
} else {
|
|
|
|
rerr->code = VIR_ERR_INTERNAL_ERROR;
|
|
|
|
rerr->domain = VIR_FROM_RPC;
|
2012-02-02 22:47:04 +00:00
|
|
|
if (VIR_ALLOC(rerr->message) == 0)
|
|
|
|
*rerr->message = strdup(_("Library function returned error but did not set virError"));
|
2010-12-06 17:03:22 +00:00
|
|
|
rerr->level = VIR_ERR_ERROR;
|
|
|
|
}
|
|
|
|
}
|
2011-10-21 10:30:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
int virNetMessageDupFD(virNetMessagePtr msg,
|
|
|
|
size_t slot)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
if (slot >= msg->nfds) {
|
|
|
|
virNetError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("No FD available at slot %zu"), slot);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((fd = dup(msg->fds[slot])) < 0) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("Unable to duplicate FD %d"),
|
|
|
|
msg->fds[slot]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (virSetInherit(fd, false) < 0) {
|
|
|
|
VIR_FORCE_CLOSE(fd);
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("Cannot set close-on-exec %d"),
|
|
|
|
fd);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return fd;
|
|
|
|
}
|