2011-10-05 18:31:54 +01:00
|
|
|
/*
|
|
|
|
* qemu_agent.h: interaction with QEMU guest agent
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006-2012 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
|
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/>.
|
2011-10-05 18:31:54 +01:00
|
|
|
*/
|
|
|
|
|
2019-06-18 11:12:37 -05:00
|
|
|
#pragma once
|
2011-10-05 18:31:54 +01:00
|
|
|
|
2019-06-18 11:12:37 -05:00
|
|
|
#include "internal.h"
|
|
|
|
#include "domain_conf.h"
|
2011-10-05 18:31:54 +01:00
|
|
|
|
|
|
|
typedef struct _qemuAgent qemuAgent;
|
|
|
|
|
|
|
|
typedef struct _qemuAgentCallbacks qemuAgentCallbacks;
|
|
|
|
struct _qemuAgentCallbacks {
|
2021-03-11 08:16:13 +01:00
|
|
|
void (*destroy)(qemuAgent *mon,
|
|
|
|
virDomainObj *vm);
|
|
|
|
void (*eofNotify)(qemuAgent *mon,
|
|
|
|
virDomainObj *vm);
|
|
|
|
void (*errorNotify)(qemuAgent *mon,
|
|
|
|
virDomainObj *vm);
|
2011-10-05 18:31:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAgent *qemuAgentOpen(virDomainObj *vm,
|
2016-01-08 16:21:30 +01:00
|
|
|
const virDomainChrSourceDef *config,
|
2020-02-12 14:54:19 +00:00
|
|
|
GMainContext *context,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAgentCallbacks *cb,
|
2020-03-05 17:47:01 +03:00
|
|
|
bool singleSync);
|
2011-10-05 18:31:54 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentClose(qemuAgent *mon);
|
2011-10-05 18:31:54 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentNotifyClose(qemuAgent *mon);
|
processSerialChangedEvent: Close agent monitor early
https://bugzilla.redhat.com/show_bug.cgi?id=890648
So, imagine you've issued an API that involves guest agent. For
instance, you want to query guest's IP addresses. So the API acquires
QUERY_JOB, locks the guest agent and issues the agent command.
However, for some reason, guest agent replies to initial ping
correctly, but then crashes tragically while executing real command
(in this case guest-network-get-interfaces). Since initial ping went
well, libvirt thinks guest agent is accessible and awaits reply to the
real command. But it will never come. What will is a monitor event.
Our handler (processSerialChangedEvent) will try to acquire
MODIFY_JOB, which will fail obviously because the other thread that's
executing the API already holds a job. So the event handler exits
early, and the QUERY_JOB is never released nor ended.
The way how to solve this is to put flag somewhere in the monitor
internals. The flag is called @running and agent commands are issued
iff the flag is set. The flag itself is set when we connect to the
agent socket. And unset whenever we see DISCONNECT event from the
agent. Moreover, we must wake up all the threads waiting for the
agent. This is done by signalizing the condition they're waiting on.
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
2015-05-07 11:19:38 +02:00
|
|
|
|
2012-06-15 18:00:13 +02:00
|
|
|
typedef enum {
|
|
|
|
QEMU_AGENT_EVENT_NONE = 0,
|
|
|
|
QEMU_AGENT_EVENT_SHUTDOWN,
|
2012-09-04 12:01:43 +02:00
|
|
|
QEMU_AGENT_EVENT_SUSPEND,
|
|
|
|
QEMU_AGENT_EVENT_RESET,
|
2012-06-15 18:00:13 +02:00
|
|
|
} qemuAgentEvent;
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentNotifyEvent(qemuAgent *mon,
|
2012-06-15 18:00:13 +02:00
|
|
|
qemuAgentEvent event);
|
|
|
|
|
2011-10-05 18:31:54 +01:00
|
|
|
typedef enum {
|
|
|
|
QEMU_AGENT_SHUTDOWN_POWERDOWN,
|
|
|
|
QEMU_AGENT_SHUTDOWN_REBOOT,
|
|
|
|
QEMU_AGENT_SHUTDOWN_HALT,
|
|
|
|
|
|
|
|
QEMU_AGENT_SHUTDOWN_LAST,
|
|
|
|
} qemuAgentShutdownMode;
|
|
|
|
|
2020-11-20 22:09:40 +04:00
|
|
|
typedef struct _qemuAgentDiskAddress qemuAgentDiskAddress;
|
|
|
|
struct _qemuAgentDiskAddress {
|
2020-01-10 17:32:13 -06:00
|
|
|
char *serial;
|
|
|
|
virPCIDeviceAddress pci_controller;
|
|
|
|
char *bus_type;
|
|
|
|
unsigned int bus;
|
|
|
|
unsigned int target;
|
|
|
|
unsigned int unit;
|
|
|
|
char *devnode;
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDeviceCCWAddress *ccw_addr;
|
2020-01-10 17:32:13 -06:00
|
|
|
};
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentDiskAddressFree(qemuAgentDiskAddress *addr);
|
2020-11-20 22:09:41 +04:00
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(qemuAgentDiskAddress, qemuAgentDiskAddressFree);
|
2020-01-10 17:32:13 -06:00
|
|
|
|
2020-11-20 22:09:45 +04:00
|
|
|
typedef struct _qemuAgentDiskInfo qemuAgentDiskInfo;
|
|
|
|
struct _qemuAgentDiskInfo {
|
|
|
|
char *name;
|
|
|
|
bool partition;
|
|
|
|
char **dependencies;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAgentDiskAddress *address;
|
2020-11-20 22:09:45 +04:00
|
|
|
char *alias;
|
|
|
|
};
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentDiskInfoFree(qemuAgentDiskInfo *info);
|
2020-11-20 22:09:45 +04:00
|
|
|
|
2020-01-10 17:32:13 -06:00
|
|
|
typedef struct _qemuAgentFSInfo qemuAgentFSInfo;
|
|
|
|
struct _qemuAgentFSInfo {
|
|
|
|
char *mountpoint; /* path to mount point */
|
|
|
|
char *name; /* device name in the guest (e.g. "sda1") */
|
|
|
|
char *fstype; /* filesystem type */
|
|
|
|
long long total_bytes;
|
|
|
|
long long used_bytes;
|
|
|
|
size_t ndisks;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAgentDiskAddress **disks;
|
2020-01-10 17:32:13 -06:00
|
|
|
};
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentFSInfoFree(qemuAgentFSInfo *info);
|
2020-01-10 17:32:13 -06:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentShutdown(qemuAgent *mon,
|
2011-10-05 18:31:54 +01:00
|
|
|
qemuAgentShutdownMode mode);
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentFSFreeze(qemuAgent *mon,
|
2014-05-01 20:06:19 -04:00
|
|
|
const char **mountpoints, unsigned int nmountpoints);
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentFSThaw(qemuAgent *mon);
|
|
|
|
int qemuAgentGetFSInfo(qemuAgent *mon,
|
|
|
|
qemuAgentFSInfo ***info,
|
2020-03-16 08:37:13 +01:00
|
|
|
bool report_unsupported);
|
2019-08-23 11:31:21 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentSuspend(qemuAgent *mon,
|
2012-02-13 12:27:25 +01:00
|
|
|
unsigned int target);
|
2012-08-23 12:29:22 +09:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentArbitraryCommand(qemuAgent *mon,
|
2012-08-23 12:29:22 +09:00
|
|
|
const char *cmd,
|
|
|
|
char **result,
|
|
|
|
int timeout);
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentFSTrim(qemuAgent *mon,
|
2012-11-20 17:10:29 +01:00
|
|
|
unsigned long long minimum);
|
2013-04-12 12:14:02 +02:00
|
|
|
|
|
|
|
|
|
|
|
typedef struct _qemuAgentCPUInfo qemuAgentCPUInfo;
|
|
|
|
struct _qemuAgentCPUInfo {
|
|
|
|
unsigned int id; /* logical cpu ID */
|
|
|
|
bool online; /* true if the CPU is activated */
|
|
|
|
bool offlinable; /* true if the CPU can be offlined */
|
2016-06-20 14:15:50 +02:00
|
|
|
|
|
|
|
bool modified; /* set to true if the vcpu state needs to be changed */
|
2013-04-12 12:14:02 +02:00
|
|
|
};
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetVCPUs(qemuAgent *mon, qemuAgentCPUInfo **info);
|
|
|
|
int qemuAgentSetVCPUs(qemuAgent *mon, qemuAgentCPUInfo *cpus, size_t ncpus);
|
2013-07-30 12:04:21 +02:00
|
|
|
int qemuAgentUpdateCPUInfo(unsigned int nvcpus,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAgentCPUInfo *cpuinfo,
|
2013-07-30 12:04:21 +02:00
|
|
|
int ncpuinfo);
|
2014-04-02 19:05:42 +02:00
|
|
|
|
2018-09-05 01:20:53 -03:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAgentGetHostname(qemuAgent *mon,
|
2020-03-16 08:26:34 +01:00
|
|
|
char **hostname,
|
|
|
|
bool report_unsupported);
|
2018-09-05 01:20:53 -03:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetTime(qemuAgent *mon,
|
2014-04-02 19:05:42 +02:00
|
|
|
long long *seconds,
|
|
|
|
unsigned int *nseconds);
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentSetTime(qemuAgent *mon,
|
2014-04-02 19:05:42 +02:00
|
|
|
long long seconds,
|
|
|
|
unsigned int nseconds,
|
|
|
|
bool sync);
|
2015-01-26 00:08:48 +05:30
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetInterfaces(qemuAgent *mon,
|
2015-01-26 00:08:48 +05:30
|
|
|
virDomainInterfacePtr **ifaces);
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentSetUserPassword(qemuAgent *mon,
|
2015-05-18 12:42:07 +02:00
|
|
|
const char *user,
|
|
|
|
const char *password,
|
|
|
|
bool crypted);
|
2019-08-23 11:31:17 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetUsers(qemuAgent *mon,
|
2019-08-23 11:31:17 -05:00
|
|
|
virTypedParameterPtr *params,
|
|
|
|
int *nparams,
|
2020-03-16 08:37:13 +01:00
|
|
|
int *maxparams,
|
|
|
|
bool report_unsupported);
|
2019-08-23 11:31:18 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetOSInfo(qemuAgent *mon,
|
2019-08-23 11:31:18 -05:00
|
|
|
virTypedParameterPtr *params,
|
|
|
|
int *nparams,
|
2020-03-16 08:37:13 +01:00
|
|
|
int *maxparams,
|
|
|
|
bool report_unsupported);
|
2019-08-23 11:31:19 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetTimezone(qemuAgent *mon,
|
2019-08-23 11:31:19 -05:00
|
|
|
virTypedParameterPtr *params,
|
|
|
|
int *nparams,
|
2020-03-16 08:37:13 +01:00
|
|
|
int *maxparams,
|
|
|
|
bool report_unsupported);
|
Add API to change qemu agent response timeout
Some layered products such as oVirt have requested a way to avoid being
blocked by guest agent commands when querying a loaded vm. For example,
many guest agent commands are polled periodically to monitor changes,
and rather than blocking the calling process, they'd prefer to simply
time out when an agent query is taking too long.
This patch adds a way for the user to specify a custom agent timeout
that is applied to all agent commands.
One special case to note here is the 'guest-sync' command. 'guest-sync'
is issued internally prior to calling any other command. (For example,
when libvirt wants to call 'guest-get-fsinfo', we first call
'guest-sync' and then call 'guest-get-fsinfo').
Previously, the 'guest-sync' command used a 5-second timeout
(VIR_DOMAIN_QEMU_AGENT_COMMAND_DEFAULT), whereas the actual command that
followed always blocked indefinitely
(VIR_DOMAIN_QEMU_AGENT_COMMAND_BLOCK). As part of this patch, if a
custom timeout is specified that is shorter than
5 seconds, this new timeout is also used for 'guest-sync'. If there is
no custom timeout or if the custom timeout is longer than 5 seconds, we
will continue to use the 5-second timeout.
Signed-off-by: Jonathon Jongsma <jjongsma@redhat.com>
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
Reviewed-by: Michal Privoznik <mprivozn@redhat.com>
2019-11-13 16:06:09 -06:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuAgentSetResponseTimeout(qemuAgent *mon,
|
Add API to change qemu agent response timeout
Some layered products such as oVirt have requested a way to avoid being
blocked by guest agent commands when querying a loaded vm. For example,
many guest agent commands are polled periodically to monitor changes,
and rather than blocking the calling process, they'd prefer to simply
time out when an agent query is taking too long.
This patch adds a way for the user to specify a custom agent timeout
that is applied to all agent commands.
One special case to note here is the 'guest-sync' command. 'guest-sync'
is issued internally prior to calling any other command. (For example,
when libvirt wants to call 'guest-get-fsinfo', we first call
'guest-sync' and then call 'guest-get-fsinfo').
Previously, the 'guest-sync' command used a 5-second timeout
(VIR_DOMAIN_QEMU_AGENT_COMMAND_DEFAULT), whereas the actual command that
followed always blocked indefinitely
(VIR_DOMAIN_QEMU_AGENT_COMMAND_BLOCK). As part of this patch, if a
custom timeout is specified that is shorter than
5 seconds, this new timeout is also used for 'guest-sync'. If there is
no custom timeout or if the custom timeout is longer than 5 seconds, we
will continue to use the 5-second timeout.
Signed-off-by: Jonathon Jongsma <jjongsma@redhat.com>
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
Reviewed-by: Michal Privoznik <mprivozn@redhat.com>
2019-11-13 16:06:09 -06:00
|
|
|
int timeout);
|
2020-11-07 13:12:53 +04:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentSSHGetAuthorizedKeys(qemuAgent *agent,
|
2020-11-07 13:12:53 +04:00
|
|
|
const char *user,
|
|
|
|
char ***keys);
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentSSHAddAuthorizedKeys(qemuAgent *agent,
|
2020-11-07 13:12:53 +04:00
|
|
|
const char *user,
|
|
|
|
const char **keys,
|
|
|
|
size_t nkeys,
|
|
|
|
bool reset);
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentSSHRemoveAuthorizedKeys(qemuAgent *agent,
|
2020-11-07 13:12:53 +04:00
|
|
|
const char *user,
|
|
|
|
const char **keys,
|
|
|
|
size_t nkeys);
|
2020-11-20 22:09:45 +04:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuAgentGetDisks(qemuAgent *mon,
|
|
|
|
qemuAgentDiskInfo ***disks,
|
2020-11-20 22:09:45 +04:00
|
|
|
bool report_unsupported);
|