/* * qemu_agent.c: interaction with QEMU guest agent * * Copyright (C) 2006-2014 Red Hat, Inc. * 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 * License along with this library. If not, see * . */ #include #include #include #include #include #include #include "qemu_agent.h" #include "qemu_domain.h" #include "viralloc.h" #include "virlog.h" #include "virerror.h" #include "virjson.h" #include "virfile.h" #include "virprocess.h" #include "virtime.h" #include "virobject.h" #include "virstring.h" #include "virenum.h" #include "virsocket.h" #include "virutil.h" #define VIR_FROM_THIS VIR_FROM_QEMU VIR_LOG_INIT("qemu.qemu_agent"); #define LINE_ENDING "\n" #define DEBUG_IO 0 #define DEBUG_RAW_IO 0 /* We read from QEMU until seeing a \r\n pair to indicate a * completed reply or event. To avoid memory denial-of-service * though, we must have a size limit on amount of data we * buffer. 10 MB is large enough that it ought to cope with * normal QEMU replies, and small enough that we're not * consuming unreasonable mem. */ #define QEMU_AGENT_MAX_RESPONSE (10 * 1024 * 1024) /* When you are the first to uncomment this, * don't forget to uncomment the corresponding * part in qemuAgentIOProcessEvent as well. * static struct { const char *type; void (*handler)(qemuAgentPtr agent, virJSONValuePtr data); } eventHandlers[] = { }; */ typedef struct _qemuAgentMessage qemuAgentMessage; typedef qemuAgentMessage *qemuAgentMessagePtr; struct _qemuAgentMessage { char *txBuffer; int txOffset; int txLength; /* Used by the JSON agent to hold reply / error */ char *rxBuffer; int rxLength; void *rxObject; /* True if rxBuffer / rxObject are ready, or a * fatal error occurred on the agent channel */ bool finished; /* true for sync command */ bool sync; /* id of the issued sync command */ unsigned long long id; bool first; }; struct _qemuAgent { virObjectLockable parent; virCond notify; int fd; GMainContext *context; GSocket *socket; GSource *watch; bool running; bool singleSync; bool inSync; virDomainObjPtr vm; qemuAgentCallbacksPtr cb; /* If there's a command being processed this will be * non-NULL */ qemuAgentMessagePtr msg; /* Buffer incoming data ready for agent * code to process & find message boundaries */ size_t bufferOffset; size_t bufferLength; char *buffer; /* If anything went wrong, this will be fed back * the next agent msg */ virError lastError; /* Some guest agent commands don't return anything * but fire up an event on qemu agent instead. * Take that as indication of successful completion */ qemuAgentEvent await_event; int timeout; }; static virClassPtr qemuAgentClass; static void qemuAgentDispose(void *obj); static int qemuAgentOnceInit(void) { if (!VIR_CLASS_NEW(qemuAgent, virClassForObjectLockable())) return -1; return 0; } VIR_ONCE_GLOBAL_INIT(qemuAgent); #if DEBUG_RAW_IO static char * qemuAgentEscapeNonPrintable(const char *text) { size_t i; g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER; for (i = 0; text[i] != '\0'; i++) { if (text[i] == '\\') virBufferAddLit(&buf, "\\\\"); else if (g_ascii_isprint(text[i]) || text[i] == '\n' || (text[i] == '\r' && text[i+1] == '\n')) virBufferAddChar(&buf, text[i]); else virBufferAsprintf(&buf, "\\x%02x", text[i]); } return virBufferContentAndReset(&buf); } #endif static void qemuAgentDispose(void *obj) { qemuAgentPtr agent = obj; VIR_DEBUG("agent=%p", agent); if (agent->cb && agent->cb->destroy) (agent->cb->destroy)(agent, agent->vm); virCondDestroy(&agent->notify); VIR_FREE(agent->buffer); g_main_context_unref(agent->context); virResetError(&agent->lastError); } static int qemuAgentOpenUnix(const char *socketpath) { struct sockaddr_un addr; int agentfd; int ret = -1; if ((agentfd = socket(AF_UNIX, SOCK_STREAM, 0)) < 0) { virReportSystemError(errno, "%s", _("failed to create socket")); return -1; } if (virSetCloseExec(agentfd) < 0) { virReportSystemError(errno, "%s", _("Unable to set agent " "close-on-exec flag")); goto error; } memset(&addr, 0, sizeof(addr)); addr.sun_family = AF_UNIX; if (virStrcpyStatic(addr.sun_path, socketpath) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Socket path %s too big for destination"), socketpath); goto error; } ret = connect(agentfd, (struct sockaddr *)&addr, sizeof(addr)); if (ret < 0) { virReportSystemError(errno, "%s", _("failed to connect to agent socket")); goto error; } return agentfd; error: VIR_FORCE_CLOSE(agentfd); return -1; } static int qemuAgentIOProcessEvent(qemuAgentPtr agent, virJSONValuePtr obj) { const char *type; VIR_DEBUG("agent=%p obj=%p", agent, obj); type = virJSONValueObjectGetString(obj, "event"); if (!type) { VIR_WARN("missing event type in message"); errno = EINVAL; return -1; } /* for (i = 0; i < G_N_ELEMENTS(eventHandlers); i++) { if (STREQ(eventHandlers[i].type, type)) { virJSONValuePtr data = virJSONValueObjectGet(obj, "data"); VIR_DEBUG("handle %s handler=%p data=%p", type, eventHandlers[i].handler, data); (eventHandlers[i].handler)(agent, data); break; } } */ return 0; } static int qemuAgentIOProcessLine(qemuAgentPtr agent, const char *line, qemuAgentMessagePtr msg) { virJSONValuePtr obj = NULL; int ret = -1; VIR_DEBUG("Line [%s]", line); if (!(obj = virJSONValueFromString(line))) { /* receiving garbage on first sync is regular situation */ if (msg && msg->sync && msg->first) { VIR_DEBUG("Received garbage on sync"); msg->finished = true; return 0; } goto cleanup; } if (virJSONValueGetType(obj) != VIR_JSON_TYPE_OBJECT) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Parsed JSON reply '%s' isn't an object"), line); goto cleanup; } if (virJSONValueObjectHasKey(obj, "QMP") == 1) { ret = 0; } else if (virJSONValueObjectHasKey(obj, "event") == 1) { ret = qemuAgentIOProcessEvent(agent, obj); } else if (virJSONValueObjectHasKey(obj, "error") == 1 || virJSONValueObjectHasKey(obj, "return") == 1) { if (msg) { if (msg->sync) { unsigned long long id; if (virJSONValueObjectGetNumberUlong(obj, "return", &id) < 0) { VIR_DEBUG("Ignoring delayed reply on sync"); ret = 0; goto cleanup; } VIR_DEBUG("Guest returned ID: %llu", id); if (msg->id != id) { VIR_DEBUG("Guest agent returned ID: %llu instead of %llu", id, msg->id); ret = 0; goto cleanup; } } msg->rxObject = obj; msg->finished = true; obj = NULL; } else { /* we are out of sync */ VIR_DEBUG("Ignoring delayed reply"); } ret = 0; } else { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unknown JSON reply '%s'"), line); } cleanup: virJSONValueFree(obj); return ret; } static int qemuAgentIOProcessData(qemuAgentPtr agent, char *data, size_t len, qemuAgentMessagePtr msg) { int used = 0; size_t i = 0; #if DEBUG_IO # if DEBUG_RAW_IO g_autofree char *str1 = qemuAgentEscapeNonPrintable(data); VIR_ERROR(_("[%s]"), str1); # else VIR_DEBUG("Data %zu bytes [%s]", len, data); # endif #endif while (used < len) { char *nl = strstr(data + used, LINE_ENDING); if (nl) { int got = nl - (data + used); for (i = 0; i < strlen(LINE_ENDING); i++) data[used + got + i] = '\0'; if (qemuAgentIOProcessLine(agent, data + used, msg) < 0) return -1; used += got + strlen(LINE_ENDING); } else { break; } } VIR_DEBUG("Total used %d bytes out of %zd available in buffer", used, len); return used; } /* This method processes data that has been received * from the agent. Looking for async events and * replies/errors. */ static int qemuAgentIOProcess(qemuAgentPtr agent) { int len; qemuAgentMessagePtr msg = NULL; /* See if there's a message ready for reply; that is, * one that has completed writing all its data. */ if (agent->msg && agent->msg->txOffset == agent->msg->txLength) msg = agent->msg; #if DEBUG_IO # if DEBUG_RAW_IO g_autofree char *str1 = qemuAgentEscapeNonPrintable(msg ? msg->txBuffer : ""); g_autofree char *str2 = qemuAgentEscapeNonPrintable(agent->buffer); VIR_ERROR(_("Process %zu %p %p [[[%s]]][[[%s]]]"), agent->bufferOffset, agent->msg, msg, str1, str2); # else VIR_DEBUG("Process %zu", agent->bufferOffset); # endif #endif len = qemuAgentIOProcessData(agent, agent->buffer, agent->bufferOffset, msg); if (len < 0) return -1; if (len < agent->bufferOffset) { memmove(agent->buffer, agent->buffer + len, agent->bufferOffset - len); agent->bufferOffset -= len; } else { VIR_FREE(agent->buffer); agent->bufferOffset = agent->bufferLength = 0; } #if DEBUG_IO VIR_DEBUG("Process done %zu used %d", agent->bufferOffset, len); #endif if (msg && msg->finished) virCondBroadcast(&agent->notify); return len; } /* * Called when the agent is able to write data * Call this function while holding the agent lock. */ static int qemuAgentIOWrite(qemuAgentPtr agent) { int done; /* If no active message, or fully transmitted, then no-op */ if (!agent->msg || agent->msg->txOffset == agent->msg->txLength) return 0; done = safewrite(agent->fd, agent->msg->txBuffer + agent->msg->txOffset, agent->msg->txLength - agent->msg->txOffset); if (done < 0) { if (errno == EAGAIN) return 0; virReportSystemError(errno, "%s", _("Unable to write to agent")); return -1; } agent->msg->txOffset += done; return done; } /* * Called when the agent has incoming data to read * Call this function while holding the agent lock. * * Returns -1 on error, or number of bytes read */ static int qemuAgentIORead(qemuAgentPtr agent) { size_t avail = agent->bufferLength - agent->bufferOffset; int ret = 0; if (avail < 1024) { if (agent->bufferLength >= QEMU_AGENT_MAX_RESPONSE) { virReportSystemError(ERANGE, _("No complete agent response found in %d bytes"), QEMU_AGENT_MAX_RESPONSE); return -1; } if (VIR_REALLOC_N(agent->buffer, agent->bufferLength + 1024) < 0) return -1; agent->bufferLength += 1024; avail += 1024; } /* Read as much as we can get into our buffer, until we block on EAGAIN, or hit EOF */ while (avail > 1) { int got; got = read(agent->fd, agent->buffer + agent->bufferOffset, avail - 1); if (got < 0) { if (errno == EAGAIN) break; virReportSystemError(errno, "%s", _("Unable to read from agent")); ret = -1; break; } if (got == 0) break; ret += got; avail -= got; agent->bufferOffset += got; agent->buffer[agent->bufferOffset] = '\0'; } #if DEBUG_IO VIR_DEBUG("Now read %zu bytes of data", agent->bufferOffset); #endif return ret; } static gboolean qemuAgentIO(GSocket *socket, GIOCondition cond, gpointer opaque); static void qemuAgentRegister(qemuAgentPtr agent) { GIOCondition cond = 0; if (agent->lastError.code == VIR_ERR_OK) { cond |= G_IO_IN; if (agent->msg && agent->msg->txOffset < agent->msg->txLength) cond |= G_IO_OUT; } agent->watch = g_socket_create_source(agent->socket, cond, NULL); virObjectRef(agent); g_source_set_callback(agent->watch, (GSourceFunc)qemuAgentIO, agent, (GDestroyNotify)virObjectUnref); g_source_attach(agent->watch, agent->context); } static void qemuAgentUnregister(qemuAgentPtr agent) { if (agent->watch) { g_source_destroy(agent->watch); g_source_unref(agent->watch); agent->watch = NULL; } } static void qemuAgentUpdateWatch(qemuAgentPtr agent) { qemuAgentUnregister(agent); if (agent->socket) qemuAgentRegister(agent); } static gboolean qemuAgentIO(GSocket *socket G_GNUC_UNUSED, GIOCondition cond, gpointer opaque) { qemuAgentPtr agent = opaque; bool error = false; bool eof = false; virObjectRef(agent); /* lock access to the agent and protect fd */ virObjectLock(agent); #if DEBUG_IO VIR_DEBUG("Agent %p I/O on watch %d socket %p cond %d", agent, agent->socket, cond); #endif if (agent->fd == -1 || !agent->watch) { virObjectUnlock(agent); virObjectUnref(agent); return G_SOURCE_REMOVE; } if (agent->lastError.code != VIR_ERR_OK) { if (cond & (G_IO_HUP | G_IO_ERR)) eof = true; error = true; } else { if (cond & G_IO_OUT) { if (qemuAgentIOWrite(agent) < 0) error = true; } if (!error && cond & G_IO_IN) { int got = qemuAgentIORead(agent); if (got < 0) { error = true; } else if (got == 0) { eof = true; } else { /* Ignore hangup/error cond if we read some data, to * give time for that data to be consumed */ cond = 0; if (qemuAgentIOProcess(agent) < 0) error = true; } } if (!error && cond & G_IO_HUP) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("End of file from agent socket")); eof = true; } if (!error && !eof && cond & G_IO_ERR) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid file descriptor while waiting for agent")); eof = true; } } if (error || eof) { if (agent->lastError.code != VIR_ERR_OK) { /* Already have an error, so clear any new error */ virResetLastError(); } else { if (virGetLastErrorCode() == VIR_ERR_OK) virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Error while processing agent IO")); virCopyLastError(&agent->lastError); virResetLastError(); } VIR_DEBUG("Error on agent %s", NULLSTR(agent->lastError.message)); /* If IO process resulted in an error & we have a message, * then wakeup that waiter */ if (agent->msg && !agent->msg->finished) { agent->msg->finished = true; virCondSignal(&agent->notify); } } qemuAgentUpdateWatch(agent); /* We have to unlock to avoid deadlock against command thread, * but is this safe ? I think it is, because the callback * will try to acquire the virDomainObjPtr mutex next */ if (eof) { void (*eofNotify)(qemuAgentPtr, virDomainObjPtr) = agent->cb->eofNotify; virDomainObjPtr vm = agent->vm; /* Make sure anyone waiting wakes up now */ virCondSignal(&agent->notify); virObjectUnlock(agent); virObjectUnref(agent); VIR_DEBUG("Triggering EOF callback"); (eofNotify)(agent, vm); } else if (error) { void (*errorNotify)(qemuAgentPtr, virDomainObjPtr) = agent->cb->errorNotify; virDomainObjPtr vm = agent->vm; /* Make sure anyone waiting wakes up now */ virCondSignal(&agent->notify); virObjectUnlock(agent); virObjectUnref(agent); VIR_DEBUG("Triggering error callback"); (errorNotify)(agent, vm); } else { virObjectUnlock(agent); virObjectUnref(agent); } return G_SOURCE_REMOVE; } qemuAgentPtr qemuAgentOpen(virDomainObjPtr vm, const virDomainChrSourceDef *config, GMainContext *context, qemuAgentCallbacksPtr cb, bool singleSync) { qemuAgentPtr agent; g_autoptr(GError) gerr = NULL; if (!cb || !cb->eofNotify) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("EOF notify callback must be supplied")); return NULL; } if (qemuAgentInitialize() < 0) return NULL; if (!(agent = virObjectLockableNew(qemuAgentClass))) return NULL; agent->timeout = QEMU_DOMAIN_PRIVATE(vm)->agentTimeout; agent->fd = -1; if (virCondInit(&agent->notify) < 0) { virReportSystemError(errno, "%s", _("cannot initialize agent condition")); virObjectUnref(agent); return NULL; } agent->vm = vm; agent->cb = cb; agent->singleSync = singleSync; if (config->type != VIR_DOMAIN_CHR_TYPE_UNIX) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unable to handle agent type: %s"), virDomainChrTypeToString(config->type)); goto cleanup; } agent->fd = qemuAgentOpenUnix(config->data.nix.path); if (agent->fd == -1) goto cleanup; agent->context = g_main_context_ref(context); agent->socket = g_socket_new_from_fd(agent->fd, &gerr); if (!agent->socket) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Unable to create socket object: %s"), gerr->message); goto cleanup; } qemuAgentRegister(agent); agent->running = true; VIR_DEBUG("New agent %p fd=%d", agent, agent->fd); return agent; cleanup: /* We don't want the 'destroy' callback invoked during * cleanup from construction failure, because that can * give a double-unref on virDomainObjPtr in the caller, * so kill the callbacks now. */ agent->cb = NULL; qemuAgentClose(agent); return NULL; } static void qemuAgentNotifyCloseLocked(qemuAgentPtr agent) { if (agent) { agent->running = false; /* If there is somebody waiting for a message * wake him up. No message will arrive anyway. */ if (agent->msg && !agent->msg->finished) { agent->msg->finished = true; virCondSignal(&agent->notify); } } } void qemuAgentNotifyClose(qemuAgentPtr agent) { if (!agent) return; VIR_DEBUG("agent=%p", agent); virObjectLock(agent); qemuAgentNotifyCloseLocked(agent); virObjectUnlock(agent); } void qemuAgentClose(qemuAgentPtr agent) { if (!agent) return; VIR_DEBUG("agent=%p", agent); virObjectLock(agent); if (agent->socket) { qemuAgentUnregister(agent); g_object_unref(agent->socket); agent->socket = NULL; agent->fd = -1; } qemuAgentNotifyCloseLocked(agent); virObjectUnlock(agent); virObjectUnref(agent); } #define QEMU_AGENT_WAIT_TIME 5 /** * qemuAgentSend: * @agent: agent object * @msg: Message * @seconds: number of seconds to wait for the result, it can be either * -2, -1, 0 or positive. * * Send @msg to agent @agent. If @seconds is equal to * VIR_DOMAIN_QEMU_AGENT_COMMAND_BLOCK(-2), this function will block forever * waiting for the result. The value of * VIR_DOMAIN_QEMU_AGENT_COMMAND_DEFAULT(-1) means use default timeout value * and VIR_DOMAIN_QEMU_AGENT_COMMAND_NOWAIT(0) makes this function return * immediately without waiting. Any positive value means the number of seconds * to wait for the result. * * Returns: 0 on success, * -2 on timeout, * -1 otherwise */ static int qemuAgentSend(qemuAgentPtr agent, qemuAgentMessagePtr msg, int seconds) { int ret = -1; unsigned long long then = 0; /* Check whether qemu quit unexpectedly */ if (agent->lastError.code != VIR_ERR_OK) { VIR_DEBUG("Attempt to send command while error is set %s", NULLSTR(agent->lastError.message)); virSetError(&agent->lastError); return -1; } if (seconds > VIR_DOMAIN_QEMU_AGENT_COMMAND_BLOCK) { unsigned long long now; if (virTimeMillisNow(&now) < 0) return -1; if (seconds == VIR_DOMAIN_QEMU_AGENT_COMMAND_DEFAULT) seconds = QEMU_AGENT_WAIT_TIME; then = now + seconds * 1000ull; } agent->msg = msg; qemuAgentUpdateWatch(agent); while (!agent->msg->finished) { if ((then && virCondWaitUntil(&agent->notify, &agent->parent.lock, then) < 0) || (!then && virCondWait(&agent->notify, &agent->parent.lock) < 0)) { if (errno == ETIMEDOUT) { virReportError(VIR_ERR_AGENT_UNRESPONSIVE, "%s", _("Guest agent not available for now")); ret = -2; } else { virReportSystemError(errno, "%s", _("Unable to wait on agent socket " "condition")); } agent->inSync = false; goto cleanup; } } if (agent->lastError.code != VIR_ERR_OK) { VIR_DEBUG("Send command resulted in error %s", NULLSTR(agent->lastError.message)); virSetError(&agent->lastError); goto cleanup; } ret = 0; cleanup: agent->msg = NULL; qemuAgentUpdateWatch(agent); return ret; } /** * qemuAgentGuestSync: * @agent: agent object * * Send guest-sync with unique ID * and wait for reply. If we get one, check if * received ID is equal to given. * * Returns: 0 on success, * -1 otherwise */ static int qemuAgentGuestSync(qemuAgentPtr agent) { int ret = -1; int send_ret; unsigned long long id; qemuAgentMessage sync_msg; int timeout = VIR_DOMAIN_QEMU_AGENT_COMMAND_DEFAULT; if (agent->singleSync && agent->inSync) return 0; /* if user specified a custom agent timeout that is lower than the * default timeout, use the shorter timeout instead */ if ((agent->timeout >= 0) && (agent->timeout < QEMU_AGENT_WAIT_TIME)) timeout = agent->timeout; memset(&sync_msg, 0, sizeof(sync_msg)); /* set only on first sync */ sync_msg.first = true; retry: if (virTimeMillisNow(&id) < 0) return -1; sync_msg.txBuffer = g_strdup_printf("{\"execute\":\"guest-sync\", " "\"arguments\":{\"id\":%llu}}\n", id); sync_msg.txLength = strlen(sync_msg.txBuffer); sync_msg.sync = true; sync_msg.id = id; VIR_DEBUG("Sending guest-sync command with ID: %llu", id); send_ret = qemuAgentSend(agent, &sync_msg, timeout); VIR_DEBUG("qemuAgentSend returned: %d", send_ret); if (send_ret < 0) goto cleanup; if (!sync_msg.rxObject) { if (sync_msg.first) { VIR_FREE(sync_msg.txBuffer); memset(&sync_msg, 0, sizeof(sync_msg)); goto retry; } else { if (agent->running) virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Missing agent reply object")); else virReportError(VIR_ERR_AGENT_UNRESPONSIVE, "%s", _("Guest agent disappeared while executing command")); goto cleanup; } } if (agent->singleSync) agent->inSync = true; ret = 0; cleanup: virJSONValueFree(sync_msg.rxObject); VIR_FREE(sync_msg.txBuffer); return ret; } static const char * qemuAgentStringifyErrorClass(const char *klass) { if (STREQ_NULLABLE(klass, "BufferOverrun")) return "Buffer overrun"; else if (STREQ_NULLABLE(klass, "CommandDisabled")) return "The command has been disabled for this instance"; else if (STREQ_NULLABLE(klass, "CommandNotFound")) return "The command has not been found"; else if (STREQ_NULLABLE(klass, "FdNotFound")) return "File descriptor not found"; else if (STREQ_NULLABLE(klass, "InvalidParameter")) return "Invalid parameter"; else if (STREQ_NULLABLE(klass, "InvalidParameterType")) return "Invalid parameter type"; else if (STREQ_NULLABLE(klass, "InvalidParameterValue")) return "Invalid parameter value"; else if (STREQ_NULLABLE(klass, "OpenFileFailed")) return "Cannot open file"; else if (STREQ_NULLABLE(klass, "QgaCommandFailed")) return "Guest agent command failed"; else if (STREQ_NULLABLE(klass, "QMPBadInputObjectMember")) return "Bad QMP input object member"; else if (STREQ_NULLABLE(klass, "QMPExtraInputObjectMember")) return "Unexpected extra object member"; else if (STREQ_NULLABLE(klass, "UndefinedError")) return "An undefined error has occurred"; else if (STREQ_NULLABLE(klass, "Unsupported")) return "this feature or command is not currently supported"; else if (klass) return klass; else return "unknown QEMU command error"; } /* Ignoring OOM in this method, since we're already reporting * a more important error * * XXX see qerror.h for different klasses & fill out useful params */ static const char * qemuAgentStringifyError(virJSONValuePtr error) { const char *klass = virJSONValueObjectGetString(error, "class"); const char *detail = virJSONValueObjectGetString(error, "desc"); /* The QMP 'desc' field is usually sufficient for our generic * error reporting needs. However, if not present, translate * the class into something readable. */ if (!detail) detail = qemuAgentStringifyErrorClass(klass); return detail; } static const char * qemuAgentCommandName(virJSONValuePtr cmd) { const char *name = virJSONValueObjectGetString(cmd, "execute"); if (name) return name; else return ""; } static int qemuAgentCheckError(virJSONValuePtr cmd, virJSONValuePtr reply, bool report_unsupported) { if (virJSONValueObjectHasKey(reply, "error")) { virJSONValuePtr error = virJSONValueObjectGet(reply, "error"); g_autofree char *cmdstr = virJSONValueToString(cmd, false); g_autofree char *replystr = virJSONValueToString(reply, false); /* Log the full JSON formatted command & error */ VIR_DEBUG("unable to execute QEMU agent command %s: %s", NULLSTR(cmdstr), NULLSTR(replystr)); /* Only send the user the command name + friendly error */ if (!error) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unable to execute QEMU agent command '%s'"), qemuAgentCommandName(cmd)); return -1; } if (!report_unsupported) { const char *klass = virJSONValueObjectGetString(error, "class"); if (STREQ_NULLABLE(klass, "CommandNotFound") || STREQ_NULLABLE(klass, "CommandDisabled")) return -2; } virReportError(VIR_ERR_INTERNAL_ERROR, _("unable to execute QEMU agent command '%s': %s"), qemuAgentCommandName(cmd), qemuAgentStringifyError(error)); return -1; } else if (!virJSONValueObjectHasKey(reply, "return")) { g_autofree char *cmdstr = virJSONValueToString(cmd, false); g_autofree char *replystr = virJSONValueToString(reply, false); VIR_DEBUG("Neither 'return' nor 'error' is set in the JSON reply %s: %s", NULLSTR(cmdstr), NULLSTR(replystr)); virReportError(VIR_ERR_INTERNAL_ERROR, _("unable to execute QEMU agent command '%s'"), qemuAgentCommandName(cmd)); return -1; } return 0; } static int qemuAgentCommandFull(qemuAgentPtr agent, virJSONValuePtr cmd, virJSONValuePtr *reply, int seconds, bool report_unsupported) { int ret = -1; qemuAgentMessage msg; g_autofree char *cmdstr = NULL; int await_event = agent->await_event; *reply = NULL; memset(&msg, 0, sizeof(msg)); if (!agent->running) { virReportError(VIR_ERR_AGENT_UNRESPONSIVE, "%s", _("Guest agent disappeared while executing command")); goto cleanup; } if (qemuAgentGuestSync(agent) < 0) goto cleanup; if (!(cmdstr = virJSONValueToString(cmd, false))) goto cleanup; msg.txBuffer = g_strdup_printf("%s" LINE_ENDING, cmdstr); msg.txLength = strlen(msg.txBuffer); VIR_DEBUG("Send command '%s' for write, seconds = %d", cmdstr, seconds); ret = qemuAgentSend(agent, &msg, seconds); VIR_DEBUG("Receive command reply ret=%d rxObject=%p", ret, msg.rxObject); if (ret < 0) goto cleanup; /* If we haven't obtained any reply but we wait for an * event, then don't report this as error */ if (!msg.rxObject) { if (!await_event) { if (agent->running) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Missing agent reply object")); } else { virReportError(VIR_ERR_AGENT_UNRESPONSIVE, "%s", _("Guest agent disappeared while executing command")); } ret = -1; } goto cleanup; } *reply = msg.rxObject; ret = qemuAgentCheckError(cmd, *reply, report_unsupported); cleanup: VIR_FREE(msg.txBuffer); agent->await_event = QEMU_AGENT_EVENT_NONE; return ret; } static int qemuAgentCommand(qemuAgentPtr agent, virJSONValuePtr cmd, virJSONValuePtr *reply, int seconds) { return qemuAgentCommandFull(agent, cmd, reply, seconds, true); } static virJSONValuePtr G_GNUC_NULL_TERMINATED qemuAgentMakeCommand(const char *cmdname, ...) { virJSONValuePtr obj = virJSONValueNewObject(); virJSONValuePtr jargs = NULL; va_list args; va_start(args, cmdname); if (virJSONValueObjectAppendString(obj, "execute", cmdname) < 0) goto error; if (virJSONValueObjectCreateVArgs(&jargs, args) < 0) goto error; if (jargs && virJSONValueObjectAppend(obj, "arguments", jargs) < 0) goto error; va_end(args); return obj; error: virJSONValueFree(obj); virJSONValueFree(jargs); va_end(args); return NULL; } static virJSONValuePtr qemuAgentMakeStringsArray(const char **strings, unsigned int len) { size_t i; virJSONValuePtr ret = virJSONValueNewArray(), str; for (i = 0; i < len; i++) { str = virJSONValueNewString(strings[i]); if (!str) goto error; if (virJSONValueArrayAppend(ret, str) < 0) { virJSONValueFree(str); goto error; } } return ret; error: virJSONValueFree(ret); return NULL; } void qemuAgentNotifyEvent(qemuAgentPtr agent, qemuAgentEvent event) { virObjectLock(agent); VIR_DEBUG("agent=%p event=%d await_event=%d", agent, event, agent->await_event); if (agent->await_event == event) { agent->await_event = QEMU_AGENT_EVENT_NONE; /* somebody waiting for this event, wake him up. */ if (agent->msg && !agent->msg->finished) { agent->msg->finished = true; virCondSignal(&agent->notify); } } virObjectUnlock(agent); } VIR_ENUM_DECL(qemuAgentShutdownMode); VIR_ENUM_IMPL(qemuAgentShutdownMode, QEMU_AGENT_SHUTDOWN_LAST, "powerdown", "reboot", "halt", ); int qemuAgentShutdown(qemuAgentPtr agent, qemuAgentShutdownMode mode) { int ret = -1; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; cmd = qemuAgentMakeCommand("guest-shutdown", "s:mode", qemuAgentShutdownModeTypeToString(mode), NULL); if (!cmd) return -1; if (mode == QEMU_AGENT_SHUTDOWN_REBOOT) agent->await_event = QEMU_AGENT_EVENT_RESET; else agent->await_event = QEMU_AGENT_EVENT_SHUTDOWN; ret = qemuAgentCommand(agent, cmd, &reply, VIR_DOMAIN_QEMU_AGENT_COMMAND_SHUTDOWN); virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } /* * qemuAgentFSFreeze: * @agent: agent object * @mountpoints: Array of mountpoint paths to be frozen, or NULL for all * @nmountpoints: Number of mountpoints to be frozen, or 0 for all * * Issue guest-fsfreeze-freeze command to guest agent, * which freezes file systems mounted on specified mountpoints * (or all file systems when @mountpoints is NULL), and returns * number of frozen file systems on success. * * Returns: number of file system frozen on success, * -1 on error. */ int qemuAgentFSFreeze(qemuAgentPtr agent, const char **mountpoints, unsigned int nmountpoints) { int ret = -1; virJSONValuePtr cmd, arg = NULL; virJSONValuePtr reply = NULL; if (mountpoints && nmountpoints) { arg = qemuAgentMakeStringsArray(mountpoints, nmountpoints); if (!arg) return -1; cmd = qemuAgentMakeCommand("guest-fsfreeze-freeze-list", "a:mountpoints", &arg, NULL); } else { cmd = qemuAgentMakeCommand("guest-fsfreeze-freeze", NULL); } if (!cmd) goto cleanup; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) goto cleanup; if (virJSONValueObjectGetNumberInt(reply, "return", &ret) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("malformed return value")); } cleanup: virJSONValueFree(arg); virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } /* * qemuAgentFSThaw: * @agent: agent object * * Issue guest-fsfreeze-thaw command to guest agent, * which unfreezes all mounted file systems and returns * number of thawed file systems on success. * * Returns: number of file system thawed on success, * -1 on error. */ int qemuAgentFSThaw(qemuAgentPtr agent) { int ret = -1; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; cmd = qemuAgentMakeCommand("guest-fsfreeze-thaw", NULL); if (!cmd) return -1; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) goto cleanup; if (virJSONValueObjectGetNumberInt(reply, "return", &ret) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("malformed return value")); } cleanup: virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } VIR_ENUM_DECL(qemuAgentSuspendMode); VIR_ENUM_IMPL(qemuAgentSuspendMode, VIR_NODE_SUSPEND_TARGET_LAST, "guest-suspend-ram", "guest-suspend-disk", "guest-suspend-hybrid", ); int qemuAgentSuspend(qemuAgentPtr agent, unsigned int target) { int ret = -1; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; cmd = qemuAgentMakeCommand(qemuAgentSuspendModeTypeToString(target), NULL); if (!cmd) return -1; agent->await_event = QEMU_AGENT_EVENT_SUSPEND; ret = qemuAgentCommand(agent, cmd, &reply, agent->timeout); virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } int qemuAgentArbitraryCommand(qemuAgentPtr agent, const char *cmd_str, char **result, int timeout) { int ret = -1; virJSONValuePtr cmd = NULL; virJSONValuePtr reply = NULL; *result = NULL; if (timeout < VIR_DOMAIN_QEMU_AGENT_COMMAND_MIN) { virReportError(VIR_ERR_INVALID_ARG, _("guest agent timeout '%d' is " "less than the minimum '%d'"), timeout, VIR_DOMAIN_QEMU_AGENT_COMMAND_MIN); goto cleanup; } if (!(cmd = virJSONValueFromString(cmd_str))) goto cleanup; if ((ret = qemuAgentCommand(agent, cmd, &reply, timeout)) < 0) goto cleanup; if (!(*result = virJSONValueToString(reply, false))) ret = -1; cleanup: virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } int qemuAgentFSTrim(qemuAgentPtr agent, unsigned long long minimum) { int ret = -1; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; cmd = qemuAgentMakeCommand("guest-fstrim", "U:minimum", minimum, NULL); if (!cmd) return ret; ret = qemuAgentCommand(agent, cmd, &reply, agent->timeout); virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } int qemuAgentGetVCPUs(qemuAgentPtr agent, qemuAgentCPUInfoPtr *info) { int ret = -1; size_t i; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; virJSONValuePtr data = NULL; size_t ndata; if (!(cmd = qemuAgentMakeCommand("guest-get-vcpus", NULL))) return -1; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) goto cleanup; if (!(data = virJSONValueObjectGetArray(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest-get-vcpus reply was missing return data")); goto cleanup; } ndata = virJSONValueArraySize(data); *info = g_new0(qemuAgentCPUInfo, ndata); for (i = 0; i < ndata; i++) { virJSONValuePtr entry = virJSONValueArrayGet(data, i); qemuAgentCPUInfoPtr in = *info + i; if (!entry) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("array element missing in guest-get-vcpus return " "value")); goto cleanup; } if (virJSONValueObjectGetNumberUint(entry, "logical-id", &in->id) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'logical-id' missing in reply of guest-get-vcpus")); goto cleanup; } if (virJSONValueObjectGetBoolean(entry, "online", &in->online) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'online' missing in reply of guest-get-vcpus")); goto cleanup; } if (virJSONValueObjectGetBoolean(entry, "can-offline", &in->offlinable) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'can-offline' missing in reply of guest-get-vcpus")); goto cleanup; } } ret = ndata; cleanup: virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } /* returns the value provided by the guest agent or -1 on internal error */ static int qemuAgentSetVCPUsCommand(qemuAgentPtr agent, qemuAgentCPUInfoPtr info, size_t ninfo, int *nmodified) { int ret = -1; virJSONValuePtr cmd = NULL; virJSONValuePtr reply = NULL; virJSONValuePtr cpus = NULL; virJSONValuePtr cpu = NULL; size_t i; *nmodified = 0; /* create the key data array */ cpus = virJSONValueNewArray(); for (i = 0; i < ninfo; i++) { qemuAgentCPUInfoPtr in = &info[i]; /* don't set state for cpus that were not touched */ if (!in->modified) continue; (*nmodified)++; /* create single cpu object */ cpu = virJSONValueNewObject(); if (virJSONValueObjectAppendNumberInt(cpu, "logical-id", in->id) < 0) goto cleanup; if (virJSONValueObjectAppendBoolean(cpu, "online", in->online) < 0) goto cleanup; if (virJSONValueArrayAppend(cpus, cpu) < 0) goto cleanup; cpu = NULL; } if (*nmodified == 0) { ret = 0; goto cleanup; } if (!(cmd = qemuAgentMakeCommand("guest-set-vcpus", "a:vcpus", &cpus, NULL))) goto cleanup; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) goto cleanup; /* All negative values are invalid. Return of 0 is bogus since we wouldn't * call the guest agent so that 0 cpus would be set successfully. Reporting * more successfully set vcpus that we've asked for is invalid. */ if (virJSONValueObjectGetNumberInt(reply, "return", &ret) < 0 || ret <= 0 || ret > *nmodified) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest agent returned malformed or invalid return value")); ret = -1; } cleanup: virJSONValueFree(cmd); virJSONValueFree(reply); virJSONValueFree(cpu); virJSONValueFree(cpus); return ret; } /** * Set the VCPU state using guest agent. * * Attempts to set the guest agent state for all cpus or until a proper error is * reported by the guest agent. This may require multiple calls. * * Returns -1 on error, 0 on success. */ int qemuAgentSetVCPUs(qemuAgentPtr agent, qemuAgentCPUInfoPtr info, size_t ninfo) { int rv; int nmodified; size_t i; do { if ((rv = qemuAgentSetVCPUsCommand(agent, info, ninfo, &nmodified)) < 0) return -1; /* all vcpus were set successfully */ if (rv == nmodified) return 0; /* un-mark vcpus that were already set */ for (i = 0; i < ninfo && rv > 0; i++) { if (!info[i].modified) continue; info[i].modified = false; rv--; } } while (1); return 0; } /* modify the cpu info structure to set the correct amount of cpus */ int qemuAgentUpdateCPUInfo(unsigned int nvcpus, qemuAgentCPUInfoPtr cpuinfo, int ncpuinfo) { size_t i; int nonline = 0; int nofflinable = 0; ssize_t cpu0 = -1; /* count the active and offlinable cpus */ for (i = 0; i < ncpuinfo; i++) { if (cpuinfo[i].id == 0) cpu0 = i; if (cpuinfo[i].online) nonline++; if (cpuinfo[i].offlinable && cpuinfo[i].online) nofflinable++; /* This shouldn't happen, but we can't trust the guest agent */ if (!cpuinfo[i].online && !cpuinfo[i].offlinable) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Invalid data provided by guest agent")); return -1; } } /* CPU0 was made offlinable in linux a while ago, but certain parts (suspend * to ram) of the kernel still don't cope well with that. Make sure that if * all remaining vCPUs are offlinable, vCPU0 will not be selected to be * offlined automatically */ if (nofflinable == nonline && cpu0 >= 0 && cpuinfo[cpu0].online) { cpuinfo[cpu0].offlinable = false; nofflinable--; } /* the guest agent reported less cpus than requested */ if (nvcpus > ncpuinfo) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest agent reports less cpu than requested")); return -1; } /* not enough offlinable CPUs to support the request */ if (nvcpus < nonline - nofflinable) { virReportError(VIR_ERR_INVALID_ARG, "%s", _("Cannot offline enough CPUs")); return -1; } for (i = 0; i < ncpuinfo; i++) { if (nvcpus < nonline) { /* unplug */ if (cpuinfo[i].offlinable && cpuinfo[i].online) { cpuinfo[i].online = false; cpuinfo[i].modified = true; nonline--; } } else if (nvcpus > nonline) { /* plug */ if (!cpuinfo[i].online) { cpuinfo[i].online = true; cpuinfo[i].modified = true; nonline++; } } else { /* done */ break; } } return 0; } /** * qemuAgentGetHostname: * * Gets the guest hostname using the guest agent. * * Returns 0 on success and fills @hostname. On error -1 is returned with an * error reported and if '@report_unsupported' is false -2 is returned if the * guest agent does not support the command without reporting an error */ int qemuAgentGetHostname(qemuAgentPtr agent, char **hostname, bool report_unsupported) { g_autoptr(virJSONValue) cmd = qemuAgentMakeCommand("guest-get-host-name", NULL); g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr data = NULL; const char *result = NULL; int rc; if (!cmd) return -1; if ((rc = qemuAgentCommandFull(agent, cmd, &reply, agent->timeout, report_unsupported)) < 0) return rc; if (!(data = virJSONValueObjectGet(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("malformed return value")); return -1; } if (!(result = virJSONValueObjectGetString(data, "host-name"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'host-name' missing in guest-get-host-name reply")); return -1; } *hostname = g_strdup(result); return 0; } int qemuAgentGetTime(qemuAgentPtr agent, long long *seconds, unsigned int *nseconds) { int ret = -1; unsigned long long json_time; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; cmd = qemuAgentMakeCommand("guest-get-time", NULL); if (!cmd) return ret; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) goto cleanup; if (virJSONValueObjectGetNumberUlong(reply, "return", &json_time) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("malformed return value")); goto cleanup; } /* guest agent returns time in nanoseconds, * we need it in seconds here */ *seconds = json_time / 1000000000LL; *nseconds = json_time % 1000000000LL; ret = 0; cleanup: virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } /** * qemuAgentSetTime: * @setTime: time to set * @sync: let guest agent to read domain's RTC (@setTime is ignored) */ int qemuAgentSetTime(qemuAgentPtr agent, long long seconds, unsigned int nseconds, bool rtcSync) { int ret = -1; virJSONValuePtr cmd; virJSONValuePtr reply = NULL; if (rtcSync) { cmd = qemuAgentMakeCommand("guest-set-time", NULL); } else { /* guest agent expect time with nanosecond granularity. * Impressing. */ long long json_time; /* Check if we overflow. For some reason qemu doesn't handle unsigned * long long on the agent well as it silently truncates numbers to * signed long long. Therefore we must check overflow against LLONG_MAX * not ULLONG_MAX. */ if (seconds > LLONG_MAX / 1000000000LL) { virReportError(VIR_ERR_INVALID_ARG, _("Time '%lld' is too big for guest agent"), seconds); return ret; } json_time = seconds * 1000000000LL; json_time += nseconds; cmd = qemuAgentMakeCommand("guest-set-time", "I:time", json_time, NULL); } if (!cmd) return ret; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) goto cleanup; ret = 0; cleanup: virJSONValueFree(cmd); virJSONValueFree(reply); return ret; } static void qemuAgentDiskAddressFree(qemuAgentDiskAddressPtr info) { if (!info) return; g_free(info->serial); g_free(info->bus_type); g_free(info->devnode); g_free(info); } void qemuAgentFSInfoFree(qemuAgentFSInfoPtr info) { size_t i; if (!info) return; g_free(info->mountpoint); g_free(info->name); g_free(info->fstype); for (i = 0; i < info->ndisks; i++) qemuAgentDiskAddressFree(info->disks[i]); g_free(info->disks); g_free(info); } static int qemuAgentGetFSInfoFillDisks(virJSONValuePtr jsondisks, qemuAgentFSInfoPtr fsinfo) { size_t ndisks; size_t i; if (!virJSONValueIsArray(jsondisks)) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed guest-get-fsinfo 'disk' data array")); return -1; } ndisks = virJSONValueArraySize(jsondisks); if (ndisks) fsinfo->disks = g_new0(qemuAgentDiskAddressPtr, ndisks); fsinfo->ndisks = ndisks; for (i = 0; i < fsinfo->ndisks; i++) { virJSONValuePtr jsondisk = virJSONValueArrayGet(jsondisks, i); virJSONValuePtr pci; qemuAgentDiskAddressPtr disk; const char *val; if (!jsondisk) { virReportError(VIR_ERR_INTERNAL_ERROR, _("array element '%zd' of '%zd' missing in " "guest-get-fsinfo 'disk' data"), i, fsinfo->ndisks); return -1; } fsinfo->disks[i] = g_new0(qemuAgentDiskAddress, 1); disk = fsinfo->disks[i]; if ((val = virJSONValueObjectGetString(jsondisk, "bus-type"))) disk->bus_type = g_strdup(val); if ((val = virJSONValueObjectGetString(jsondisk, "serial"))) disk->serial = g_strdup(val); if ((val = virJSONValueObjectGetString(jsondisk, "dev"))) disk->devnode = g_strdup(val); #define GET_DISK_ADDR(jsonObject, var, name) \ do { \ if (virJSONValueObjectGetNumberUint(jsonObject, name, var) < 0) { \ virReportError(VIR_ERR_INTERNAL_ERROR, \ _("'%s' missing in guest-get-fsinfo " \ "'disk' data"), name); \ return -1; \ } \ } while (0) GET_DISK_ADDR(jsondisk, &disk->bus, "bus"); GET_DISK_ADDR(jsondisk, &disk->target, "target"); GET_DISK_ADDR(jsondisk, &disk->unit, "unit"); if (!(pci = virJSONValueObjectGet(jsondisk, "pci-controller"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'pci-controller' missing in guest-get-fsinfo " "'disk' data")); return -1; } GET_DISK_ADDR(pci, &disk->pci_controller.domain, "domain"); GET_DISK_ADDR(pci, &disk->pci_controller.bus, "bus"); GET_DISK_ADDR(pci, &disk->pci_controller.slot, "slot"); GET_DISK_ADDR(pci, &disk->pci_controller.function, "function"); #undef GET_DISK_ADDR } return 0; } /* Returns: number of entries in '@info' on success * -2 when agent command is not supported by the agent and * 'report_unsupported' is false (libvirt error is not reported) * -1 otherwise (libvirt error is reported) */ int qemuAgentGetFSInfo(qemuAgentPtr agent, qemuAgentFSInfoPtr **info, bool report_unsupported) { size_t i; int ret = -1; g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr data; size_t ndata = 0; qemuAgentFSInfoPtr *info_ret = NULL; int rc; cmd = qemuAgentMakeCommand("guest-get-fsinfo", NULL); if (!cmd) return ret; if ((rc = qemuAgentCommandFull(agent, cmd, &reply, agent->timeout, report_unsupported)) < 0) return rc; if (!(data = virJSONValueObjectGet(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest-get-fsinfo reply was missing return data")); goto cleanup; } if (!virJSONValueIsArray(data)) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed guest-get-fsinfo data array")); goto cleanup; } ndata = virJSONValueArraySize(data); if (ndata == 0) { ret = 0; *info = NULL; goto cleanup; } info_ret = g_new0(qemuAgentFSInfoPtr, ndata); for (i = 0; i < ndata; i++) { /* Reverse the order to arrange in mount order */ virJSONValuePtr entry = virJSONValueArrayGet(data, ndata - 1 - i); virJSONValuePtr disk; unsigned long long bytes_val; const char *result = NULL; if (!entry) { virReportError(VIR_ERR_INTERNAL_ERROR, _("array element '%zd' of '%zd' missing in " "guest-get-fsinfo return data"), i, ndata); goto cleanup; } info_ret[i] = g_new0(qemuAgentFSInfo, 1); if (!(result = virJSONValueObjectGetString(entry, "mountpoint"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'mountpoint' missing in reply of " "guest-get-fsinfo")); goto cleanup; } info_ret[i]->mountpoint = g_strdup(result); if (!(result = virJSONValueObjectGetString(entry, "name"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'name' missing in reply of guest-get-fsinfo")); goto cleanup; } info_ret[i]->name = g_strdup(result); if (!(result = virJSONValueObjectGetString(entry, "type"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'type' missing in reply of guest-get-fsinfo")); goto cleanup; } info_ret[i]->fstype = g_strdup(result); /* 'used-bytes' and 'total-bytes' were added in qemu-ga 3.0 */ if (virJSONValueObjectHasKey(entry, "used-bytes")) { if (virJSONValueObjectGetNumberUlong(entry, "used-bytes", &bytes_val) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Error getting 'used-bytes' in reply of guest-get-fsinfo")); goto cleanup; } info_ret[i]->used_bytes = bytes_val; } else { info_ret[i]->used_bytes = -1; } if (virJSONValueObjectHasKey(entry, "total-bytes")) { if (virJSONValueObjectGetNumberUlong(entry, "total-bytes", &bytes_val) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Error getting 'total-bytes' in reply of guest-get-fsinfo")); goto cleanup; } info_ret[i]->total_bytes = bytes_val; } else { info_ret[i]->total_bytes = -1; } if (!(disk = virJSONValueObjectGet(entry, "disk"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'disk' missing in reply of guest-get-fsinfo")); goto cleanup; } if (qemuAgentGetFSInfoFillDisks(disk, info_ret[i]) < 0) goto cleanup; } *info = g_steal_pointer(&info_ret); ret = ndata; cleanup: if (info_ret) { for (i = 0; i < ndata; i++) qemuAgentFSInfoFree(info_ret[i]); g_free(info_ret); } return ret; } static int qemuAgentGetInterfaceOneAddress(virDomainIPAddressPtr ip_addr, virJSONValuePtr ip_addr_obj, const char *name) { const char *type, *addr; type = virJSONValueObjectGetString(ip_addr_obj, "ip-address-type"); if (!type) { virReportError(VIR_ERR_INTERNAL_ERROR, _("qemu agent didn't provide 'ip-address-type'" " field for interface '%s'"), name); return -1; } if (STRNEQ(type, "ipv4") && STRNEQ(type, "ipv6")) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unknown ip address type '%s'"), type); return -1; } addr = virJSONValueObjectGetString(ip_addr_obj, "ip-address"); if (!addr) { virReportError(VIR_ERR_INTERNAL_ERROR, _("qemu agent didn't provide 'ip-address'" " field for interface '%s'"), name); return -1; } if (virJSONValueObjectGetNumberUint(ip_addr_obj, "prefix", &ip_addr->prefix) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("malformed 'prefix' field")); return -1; } if (STREQ(type, "ipv4")) ip_addr->type = VIR_IP_ADDR_TYPE_IPV4; else ip_addr->type = VIR_IP_ADDR_TYPE_IPV6; ip_addr->addr = g_strdup(addr); return 0; } /** * qemuAgentGetInterfaceAddresses: * @ifaces_ret: the array to put/update the interface in * @ifaces_count: the number of interfaces in that array * @ifaces_store: hash table into @ifaces_ret by interface name * @iface_obj: one item from the JSON array of interfaces * * This function processes @iface_obj (which represents * information about a single interface) and adds the information * into the ifaces_ret array. * * If we're processing an interface alias, the suffix is stripped * and information is appended to the entry found via the @ifaces_store * hash table. * * Otherwise, the next free position in @ifaces_ret is used, * its address added to @ifaces_store, and @ifaces_count incremented. */ static int qemuAgentGetInterfaceAddresses(virDomainInterfacePtr **ifaces_ret, size_t *ifaces_count, GHashTable *ifaces_store, virJSONValuePtr iface_obj) { virJSONValuePtr ip_addr_arr = NULL; const char *hwaddr, *name = NULL; virDomainInterfacePtr iface = NULL; g_autofree char *ifname = NULL; size_t addrs_count = 0; size_t j; /* interface name is required to be presented */ name = virJSONValueObjectGetString(iface_obj, "name"); if (!name) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("qemu agent didn't provide 'name' field")); return -1; } /* Handle interface alias (:) */ ifname = g_strdelimit(g_strdup(name), ":", '\0'); iface = virHashLookup(ifaces_store, ifname); /* If the hash table doesn't contain this iface, add it */ if (!iface) { if (VIR_EXPAND_N(*ifaces_ret, *ifaces_count, 1) < 0) return -1; iface = g_new0(virDomainInterface, 1); (*ifaces_ret)[*ifaces_count - 1] = iface; if (virHashAddEntry(ifaces_store, ifname, iface) < 0) return -1; iface->naddrs = 0; iface->name = g_strdup(ifname); hwaddr = virJSONValueObjectGetString(iface_obj, "hardware-address"); iface->hwaddr = g_strdup(hwaddr); } /* as well as IP address which - moreover - * can be presented multiple times */ ip_addr_arr = virJSONValueObjectGet(iface_obj, "ip-addresses"); if (!ip_addr_arr) return 0; if (!virJSONValueIsArray(ip_addr_arr)) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Malformed ip-addresses array")); return -1; } /* If current iface already exists, continue with the count */ addrs_count = iface->naddrs; if (VIR_EXPAND_N(iface->addrs, addrs_count, virJSONValueArraySize(ip_addr_arr)) < 0) return -1; for (j = 0; j < virJSONValueArraySize(ip_addr_arr); j++) { virJSONValuePtr ip_addr_obj = virJSONValueArrayGet(ip_addr_arr, j); virDomainIPAddressPtr ip_addr = iface->addrs + iface->naddrs; iface->naddrs++; if (qemuAgentGetInterfaceOneAddress(ip_addr, ip_addr_obj, name) < 0) return -1; } return 0; } static int qemuAgentGetAllInterfaceAddresses(virDomainInterfacePtr **ifaces_ret, virJSONValuePtr ret_array) { g_autoptr(GHashTable) ifaces_store = NULL; size_t ifaces_count = 0; size_t i; *ifaces_ret = NULL; /* Hash table to handle the interface alias */ ifaces_store = virHashNew(NULL); for (i = 0; i < virJSONValueArraySize(ret_array); i++) { virJSONValuePtr iface_obj = virJSONValueArrayGet(ret_array, i); if (qemuAgentGetInterfaceAddresses(ifaces_ret, &ifaces_count, ifaces_store, iface_obj) < 0) goto error; } return ifaces_count; error: if (*ifaces_ret) { for (i = 0; i < ifaces_count; i++) virDomainInterfaceFree((*ifaces_ret)[i]); } VIR_FREE(*ifaces_ret); return -1; } /* * qemuAgentGetInterfaces: * @agent: agent object * @ifaces: pointer to an array of pointers pointing to interface objects * * Issue guest-network-get-interfaces to guest agent, which returns a * list of interfaces of a running domain along with their IP and MAC * addresses. * * Returns: number of interfaces on success, -1 on error. */ int qemuAgentGetInterfaces(qemuAgentPtr agent, virDomainInterfacePtr **ifaces) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr ret_array = NULL; if (!(cmd = qemuAgentMakeCommand("guest-network-get-interfaces", NULL))) return -1; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) return -1; if (!(ret_array = virJSONValueObjectGetArray(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("qemu agent didn't return an array of interfaces")); return -1; } return qemuAgentGetAllInterfaceAddresses(ifaces, ret_array); } int qemuAgentSetUserPassword(qemuAgentPtr agent, const char *user, const char *password, bool crypted) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; g_autofree char *password64 = NULL; password64 = g_base64_encode((unsigned char *)password, strlen(password)); if (!(cmd = qemuAgentMakeCommand("guest-set-user-password", "b:crypted", crypted, "s:username", user, "s:password", password64, NULL))) return -1; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) return -1; return 0; } /* Returns: 0 on success * -2 when agent command is not supported by the agent and * 'report_unsupported' is false (libvirt error is not reported) * -1 otherwise (libvirt error is reported) */ int qemuAgentGetUsers(qemuAgentPtr agent, virTypedParameterPtr *params, int *nparams, int *maxparams, bool report_unsupported) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr data = NULL; size_t ndata; size_t i; int rc; if (!(cmd = qemuAgentMakeCommand("guest-get-users", NULL))) return -1; if ((rc = qemuAgentCommandFull(agent, cmd, &reply, agent->timeout, report_unsupported)) < 0) return rc; if (!(data = virJSONValueObjectGetArray(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest-get-users reply was missing return data")); return -1; } ndata = virJSONValueArraySize(data); if (virTypedParamsAddUInt(params, nparams, maxparams, "user.count", ndata) < 0) return -1; for (i = 0; i < ndata; i++) { virJSONValuePtr entry = virJSONValueArrayGet(data, i); char param_name[VIR_TYPED_PARAM_FIELD_LENGTH]; const char *strvalue; double logintime; if (!entry) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("array element missing in guest-get-users return " "value")); return -1; } if (!(strvalue = virJSONValueObjectGetString(entry, "user"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'user' missing in reply of guest-get-users")); return -1; } g_snprintf(param_name, VIR_TYPED_PARAM_FIELD_LENGTH, "user.%zu.name", i); if (virTypedParamsAddString(params, nparams, maxparams, param_name, strvalue) < 0) return -1; /* 'domain' is only present for windows guests */ if ((strvalue = virJSONValueObjectGetString(entry, "domain"))) { g_snprintf(param_name, VIR_TYPED_PARAM_FIELD_LENGTH, "user.%zu.domain", i); if (virTypedParamsAddString(params, nparams, maxparams, param_name, strvalue) < 0) return -1; } if (virJSONValueObjectGetNumberDouble(entry, "login-time", &logintime) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'login-time' missing in reply of guest-get-users")); return -1; } g_snprintf(param_name, VIR_TYPED_PARAM_FIELD_LENGTH, "user.%zu.login-time", i); if (virTypedParamsAddULLong(params, nparams, maxparams, param_name, logintime * 1000) < 0) return -1; } return 0; } /* Returns: 0 on success * -2 when agent command is not supported by the agent and * 'report_unsupported' is false (libvirt error is not reported) * -1 otherwise (libvirt error is reported) */ int qemuAgentGetOSInfo(qemuAgentPtr agent, virTypedParameterPtr *params, int *nparams, int *maxparams, bool report_unsupported) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr data = NULL; int rc; if (!(cmd = qemuAgentMakeCommand("guest-get-osinfo", NULL))) return -1; if ((rc = qemuAgentCommandFull(agent, cmd, &reply, agent->timeout, report_unsupported)) < 0) return rc; if (!(data = virJSONValueObjectGetObject(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest-get-osinfo reply was missing return data")); return -1; } #define OSINFO_ADD_PARAM(agent_string_, param_string_) \ do { \ const char *result; \ if ((result = virJSONValueObjectGetString(data, agent_string_))) { \ if (virTypedParamsAddString(params, nparams, maxparams, \ param_string_, result) < 0) { \ return -1; \ } \ } \ } while (0) OSINFO_ADD_PARAM("id", "os.id"); OSINFO_ADD_PARAM("name", "os.name"); OSINFO_ADD_PARAM("pretty-name", "os.pretty-name"); OSINFO_ADD_PARAM("version", "os.version"); OSINFO_ADD_PARAM("version-id", "os.version-id"); OSINFO_ADD_PARAM("machine", "os.machine"); OSINFO_ADD_PARAM("variant", "os.variant"); OSINFO_ADD_PARAM("variant-id", "os.variant-id"); OSINFO_ADD_PARAM("kernel-release", "os.kernel-release"); OSINFO_ADD_PARAM("kernel-version", "os.kernel-version"); return 0; } /* Returns: 0 on success * -2 when agent command is not supported by the agent and * 'report_unsupported' is false (libvirt error is not reported) * -1 otherwise (libvirt error is reported) */ int qemuAgentGetTimezone(qemuAgentPtr agent, virTypedParameterPtr *params, int *nparams, int *maxparams, bool report_unsupported) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr data = NULL; const char *name; int offset; int rc; if (!(cmd = qemuAgentMakeCommand("guest-get-timezone", NULL))) return -1; if ((rc = qemuAgentCommandFull(agent, cmd, &reply, agent->timeout, report_unsupported)) < 0) return rc; if (!(data = virJSONValueObjectGetObject(reply, "return"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("guest-get-timezone reply was missing return data")); return -1; } if ((name = virJSONValueObjectGetString(data, "zone")) && virTypedParamsAddString(params, nparams, maxparams, "timezone.name", name) < 0) return -1; if ((virJSONValueObjectGetNumberInt(data, "offset", &offset)) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("'offset' missing in reply of guest-get-timezone")); return -1; } if (virTypedParamsAddInt(params, nparams, maxparams, "timezone.offset", offset) < 0) return -1; return 0; } /* qemuAgentSetResponseTimeout: * @agent: agent object * @timeout: number of seconds to wait for agent response * * The agent object must be locked prior to calling this function. */ void qemuAgentSetResponseTimeout(qemuAgentPtr agent, int timeout) { agent->timeout = timeout; } /** * qemuAgentSSHGetAuthorizedKeys: * @agent: agent object * @user: user to get authorized keys for * @keys: Array of authorized keys * * Fetch the public keys from @user's $HOME/.ssh/authorized_keys. * * Returns: number of keys returned on success, * -1 otherwise (error is reported) */ int qemuAgentSSHGetAuthorizedKeys(qemuAgentPtr agent, const char *user, char ***keys) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; virJSONValuePtr data = NULL; size_t ndata; size_t i; char **keys_ret = NULL; if (!(cmd = qemuAgentMakeCommand("guest-ssh-get-authorized-keys", "s:username", user, NULL))) return -1; if (qemuAgentCommand(agent, cmd, &reply, agent->timeout) < 0) return -1; if (!(data = virJSONValueObjectGetObject(reply, "return")) || !(data = virJSONValueObjectGetArray(data, "keys"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("qemu agent didn't return an array of keys")); return -1; } ndata = virJSONValueArraySize(data); keys_ret = g_new0(char *, ndata + 1); for (i = 0; i < ndata; i++) { virJSONValuePtr entry = virJSONValueArrayGet(data, i); if (!entry) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("array element missing in guest-ssh-get-authorized-keys return value")); goto error; } keys_ret[i] = g_strdup(virJSONValueGetString(entry)); } *keys = g_steal_pointer(&keys_ret); return ndata; error: virStringListFreeCount(keys_ret, ndata); return -1; } /** * qemuAgentSSHAddAuthorizedKeys: * @agent: agent object * @user: user to add authorized keys for * @keys: Array of authorized keys * @nkeys: number of items in @keys array * @reset: whether to truncate authorized keys file before writing * * Append SSH @keys into the @user's authorized keys file. If * @reset is true then the file is truncated before write and * thus contains only newly added @keys. * * Returns: 0 on success, * -1 otherwise (error is reported) */ int qemuAgentSSHAddAuthorizedKeys(qemuAgentPtr agent, const char *user, const char **keys, size_t nkeys, bool reset) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; g_autoptr(virJSONValue) jkeys = NULL; jkeys = qemuAgentMakeStringsArray(keys, nkeys); if (jkeys == NULL) return -1; if (!(cmd = qemuAgentMakeCommand("guest-ssh-add-authorized-keys", "s:username", user, "a:keys", &jkeys, "b:reset", reset, NULL))) return -1; return qemuAgentCommand(agent, cmd, &reply, agent->timeout); } /** * qemuAgentSSHRemoveAuthorizedKeys: * @agent: agent object * @user: user to remove authorized keys for * @keys: Array of authorized keys * @nkeys: number of items in @keys array * * Remove SSH @keys from the @user's authorized keys file. It's * not considered an error when trying to remove a non-existent * key. * * Returns: 0 on success, * -1 otherwise (error is reported) */ int qemuAgentSSHRemoveAuthorizedKeys(qemuAgentPtr agent, const char *user, const char **keys, size_t nkeys) { g_autoptr(virJSONValue) cmd = NULL; g_autoptr(virJSONValue) reply = NULL; g_autoptr(virJSONValue) jkeys = NULL; jkeys = qemuAgentMakeStringsArray(keys, nkeys); if (jkeys == NULL) return -1; if (!(cmd = qemuAgentMakeCommand("guest-ssh-remove-authorized-keys", "s:username", user, "a:keys", &jkeys, NULL))) return -1; return qemuAgentCommand(agent, cmd, &reply, agent->timeout); }