Merge remote-tracking branch 'remotes/mdroth/qga-pull-2014-08-08' into staging
* remotes/mdroth/qga-pull-2014-08-08: qga: Disable unsupported commands by default qga: Add guest-get-fsinfo command qga: Add guest-fsfreeze-freeze-list command Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
2d591ce2ae
@ -18,6 +18,7 @@
|
||||
#include <unistd.h>
|
||||
#include <errno.h>
|
||||
#include <fcntl.h>
|
||||
#include <dirent.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <sys/stat.h>
|
||||
@ -575,6 +576,7 @@ static void guest_file_init(void)
|
||||
typedef struct FsMount {
|
||||
char *dirname;
|
||||
char *devtype;
|
||||
unsigned int devmajor, devminor;
|
||||
QTAILQ_ENTRY(FsMount) next;
|
||||
} FsMount;
|
||||
|
||||
@ -596,15 +598,40 @@ static void free_fs_mount_list(FsMountList *mounts)
|
||||
}
|
||||
}
|
||||
|
||||
static int dev_major_minor(const char *devpath,
|
||||
unsigned int *devmajor, unsigned int *devminor)
|
||||
{
|
||||
struct stat st;
|
||||
|
||||
*devmajor = 0;
|
||||
*devminor = 0;
|
||||
|
||||
if (stat(devpath, &st) < 0) {
|
||||
slog("failed to stat device file '%s': %s", devpath, strerror(errno));
|
||||
return -1;
|
||||
}
|
||||
if (S_ISDIR(st.st_mode)) {
|
||||
/* It is bind mount */
|
||||
return -2;
|
||||
}
|
||||
if (S_ISBLK(st.st_mode)) {
|
||||
*devmajor = major(st.st_rdev);
|
||||
*devminor = minor(st.st_rdev);
|
||||
return 0;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
/*
|
||||
* Walk the mount table and build a list of local file systems
|
||||
*/
|
||||
static void build_fs_mount_list(FsMountList *mounts, Error **errp)
|
||||
static void build_fs_mount_list_from_mtab(FsMountList *mounts, Error **errp)
|
||||
{
|
||||
struct mntent *ment;
|
||||
FsMount *mount;
|
||||
char const *mtab = "/proc/self/mounts";
|
||||
FILE *fp;
|
||||
unsigned int devmajor, devminor;
|
||||
|
||||
fp = setmntent(mtab, "r");
|
||||
if (!fp) {
|
||||
@ -624,20 +651,423 @@ static void build_fs_mount_list(FsMountList *mounts, Error **errp)
|
||||
(strcmp(ment->mnt_type, "cifs") == 0)) {
|
||||
continue;
|
||||
}
|
||||
if (dev_major_minor(ment->mnt_fsname, &devmajor, &devminor) == -2) {
|
||||
/* Skip bind mounts */
|
||||
continue;
|
||||
}
|
||||
|
||||
mount = g_malloc0(sizeof(FsMount));
|
||||
mount->dirname = g_strdup(ment->mnt_dir);
|
||||
mount->devtype = g_strdup(ment->mnt_type);
|
||||
mount->devmajor = devmajor;
|
||||
mount->devminor = devminor;
|
||||
|
||||
QTAILQ_INSERT_TAIL(mounts, mount, next);
|
||||
}
|
||||
|
||||
endmntent(fp);
|
||||
}
|
||||
|
||||
static void decode_mntname(char *name, int len)
|
||||
{
|
||||
int i, j = 0;
|
||||
for (i = 0; i <= len; i++) {
|
||||
if (name[i] != '\\') {
|
||||
name[j++] = name[i];
|
||||
} else if (name[i + 1] == '\\') {
|
||||
name[j++] = '\\';
|
||||
i++;
|
||||
} else if (name[i + 1] >= '0' && name[i + 1] <= '3' &&
|
||||
name[i + 2] >= '0' && name[i + 2] <= '7' &&
|
||||
name[i + 3] >= '0' && name[i + 3] <= '7') {
|
||||
name[j++] = (name[i + 1] - '0') * 64 +
|
||||
(name[i + 2] - '0') * 8 +
|
||||
(name[i + 3] - '0');
|
||||
i += 3;
|
||||
} else {
|
||||
name[j++] = name[i];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void build_fs_mount_list(FsMountList *mounts, Error **errp)
|
||||
{
|
||||
FsMount *mount;
|
||||
char const *mountinfo = "/proc/self/mountinfo";
|
||||
FILE *fp;
|
||||
char *line = NULL, *dash;
|
||||
size_t n;
|
||||
char check;
|
||||
unsigned int devmajor, devminor;
|
||||
int ret, dir_s, dir_e, type_s, type_e, dev_s, dev_e;
|
||||
|
||||
fp = fopen(mountinfo, "r");
|
||||
if (!fp) {
|
||||
build_fs_mount_list_from_mtab(mounts, errp);
|
||||
return;
|
||||
}
|
||||
|
||||
while (getline(&line, &n, fp) != -1) {
|
||||
ret = sscanf(line, "%*u %*u %u:%u %*s %n%*s%n%c",
|
||||
&devmajor, &devminor, &dir_s, &dir_e, &check);
|
||||
if (ret < 3) {
|
||||
continue;
|
||||
}
|
||||
dash = strstr(line + dir_e, " - ");
|
||||
if (!dash) {
|
||||
continue;
|
||||
}
|
||||
ret = sscanf(dash, " - %n%*s%n %n%*s%n%c",
|
||||
&type_s, &type_e, &dev_s, &dev_e, &check);
|
||||
if (ret < 1) {
|
||||
continue;
|
||||
}
|
||||
line[dir_e] = 0;
|
||||
dash[type_e] = 0;
|
||||
dash[dev_e] = 0;
|
||||
decode_mntname(line + dir_s, dir_e - dir_s);
|
||||
decode_mntname(dash + dev_s, dev_e - dev_s);
|
||||
if (devmajor == 0) {
|
||||
/* btrfs reports major number = 0 */
|
||||
if (strcmp("btrfs", dash + type_s) != 0 ||
|
||||
dev_major_minor(dash + dev_s, &devmajor, &devminor) < 0) {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
mount = g_malloc0(sizeof(FsMount));
|
||||
mount->dirname = g_strdup(line + dir_s);
|
||||
mount->devtype = g_strdup(dash + type_s);
|
||||
mount->devmajor = devmajor;
|
||||
mount->devminor = devminor;
|
||||
|
||||
QTAILQ_INSERT_TAIL(mounts, mount, next);
|
||||
}
|
||||
free(line);
|
||||
|
||||
fclose(fp);
|
||||
}
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_FSFREEZE)
|
||||
|
||||
static char *get_pci_driver(char const *syspath, int pathlen, Error **errp)
|
||||
{
|
||||
char *path;
|
||||
char *dpath;
|
||||
char *driver = NULL;
|
||||
char buf[PATH_MAX];
|
||||
ssize_t len;
|
||||
|
||||
path = g_strndup(syspath, pathlen);
|
||||
dpath = g_strdup_printf("%s/driver", path);
|
||||
len = readlink(dpath, buf, sizeof(buf) - 1);
|
||||
if (len != -1) {
|
||||
buf[len] = 0;
|
||||
driver = g_strdup(basename(buf));
|
||||
}
|
||||
g_free(dpath);
|
||||
g_free(path);
|
||||
return driver;
|
||||
}
|
||||
|
||||
static int compare_uint(const void *_a, const void *_b)
|
||||
{
|
||||
unsigned int a = *(unsigned int *)_a;
|
||||
unsigned int b = *(unsigned int *)_b;
|
||||
|
||||
return a < b ? -1 : a > b ? 1 : 0;
|
||||
}
|
||||
|
||||
/* Walk the specified sysfs and build a sorted list of host or ata numbers */
|
||||
static int build_hosts(char const *syspath, char const *host, bool ata,
|
||||
unsigned int *hosts, int hosts_max, Error **errp)
|
||||
{
|
||||
char *path;
|
||||
DIR *dir;
|
||||
struct dirent *entry;
|
||||
int i = 0;
|
||||
|
||||
path = g_strndup(syspath, host - syspath);
|
||||
dir = opendir(path);
|
||||
if (!dir) {
|
||||
error_setg_errno(errp, errno, "opendir(\"%s\")", path);
|
||||
g_free(path);
|
||||
return -1;
|
||||
}
|
||||
|
||||
while (i < hosts_max) {
|
||||
entry = readdir(dir);
|
||||
if (!entry) {
|
||||
break;
|
||||
}
|
||||
if (ata && sscanf(entry->d_name, "ata%d", hosts + i) == 1) {
|
||||
++i;
|
||||
} else if (!ata && sscanf(entry->d_name, "host%d", hosts + i) == 1) {
|
||||
++i;
|
||||
}
|
||||
}
|
||||
|
||||
qsort(hosts, i, sizeof(hosts[0]), compare_uint);
|
||||
|
||||
g_free(path);
|
||||
closedir(dir);
|
||||
return i;
|
||||
}
|
||||
|
||||
/* Store disk device info specified by @sysfs into @fs */
|
||||
static void build_guest_fsinfo_for_real_device(char const *syspath,
|
||||
GuestFilesystemInfo *fs,
|
||||
Error **errp)
|
||||
{
|
||||
unsigned int pci[4], host, hosts[8], tgt[3];
|
||||
int i, nhosts = 0, pcilen;
|
||||
GuestDiskAddress *disk;
|
||||
GuestPCIAddress *pciaddr;
|
||||
GuestDiskAddressList *list = NULL;
|
||||
bool has_ata = false, has_host = false, has_tgt = false;
|
||||
char *p, *q, *driver = NULL;
|
||||
|
||||
p = strstr(syspath, "/devices/pci");
|
||||
if (!p || sscanf(p + 12, "%*x:%*x/%x:%x:%x.%x%n",
|
||||
pci, pci + 1, pci + 2, pci + 3, &pcilen) < 4) {
|
||||
g_debug("only pci device is supported: sysfs path \"%s\"", syspath);
|
||||
return;
|
||||
}
|
||||
|
||||
driver = get_pci_driver(syspath, (p + 12 + pcilen) - syspath, errp);
|
||||
if (!driver) {
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
p = strstr(syspath, "/target");
|
||||
if (p && sscanf(p + 7, "%*u:%*u:%*u/%*u:%u:%u:%u",
|
||||
tgt, tgt + 1, tgt + 2) == 3) {
|
||||
has_tgt = true;
|
||||
}
|
||||
|
||||
p = strstr(syspath, "/ata");
|
||||
if (p) {
|
||||
q = p + 4;
|
||||
has_ata = true;
|
||||
} else {
|
||||
p = strstr(syspath, "/host");
|
||||
q = p + 5;
|
||||
}
|
||||
if (p && sscanf(q, "%u", &host) == 1) {
|
||||
has_host = true;
|
||||
nhosts = build_hosts(syspath, p, has_ata, hosts,
|
||||
sizeof(hosts) / sizeof(hosts[0]), errp);
|
||||
if (nhosts < 0) {
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
|
||||
pciaddr = g_malloc0(sizeof(*pciaddr));
|
||||
pciaddr->domain = pci[0];
|
||||
pciaddr->bus = pci[1];
|
||||
pciaddr->slot = pci[2];
|
||||
pciaddr->function = pci[3];
|
||||
|
||||
disk = g_malloc0(sizeof(*disk));
|
||||
disk->pci_controller = pciaddr;
|
||||
|
||||
list = g_malloc0(sizeof(*list));
|
||||
list->value = disk;
|
||||
|
||||
if (strcmp(driver, "ata_piix") == 0) {
|
||||
/* a host per ide bus, target*:0:<unit>:0 */
|
||||
if (!has_host || !has_tgt) {
|
||||
g_debug("invalid sysfs path '%s' (driver '%s')", syspath, driver);
|
||||
goto cleanup;
|
||||
}
|
||||
for (i = 0; i < nhosts; i++) {
|
||||
if (host == hosts[i]) {
|
||||
disk->bus_type = GUEST_DISK_BUS_TYPE_IDE;
|
||||
disk->bus = i;
|
||||
disk->unit = tgt[1];
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (i >= nhosts) {
|
||||
g_debug("no host for '%s' (driver '%s')", syspath, driver);
|
||||
goto cleanup;
|
||||
}
|
||||
} else if (strcmp(driver, "sym53c8xx") == 0) {
|
||||
/* scsi(LSI Logic): target*:0:<unit>:0 */
|
||||
if (!has_tgt) {
|
||||
g_debug("invalid sysfs path '%s' (driver '%s')", syspath, driver);
|
||||
goto cleanup;
|
||||
}
|
||||
disk->bus_type = GUEST_DISK_BUS_TYPE_SCSI;
|
||||
disk->unit = tgt[1];
|
||||
} else if (strcmp(driver, "virtio-pci") == 0) {
|
||||
if (has_tgt) {
|
||||
/* virtio-scsi: target*:0:0:<unit> */
|
||||
disk->bus_type = GUEST_DISK_BUS_TYPE_SCSI;
|
||||
disk->unit = tgt[2];
|
||||
} else {
|
||||
/* virtio-blk: 1 disk per 1 device */
|
||||
disk->bus_type = GUEST_DISK_BUS_TYPE_VIRTIO;
|
||||
}
|
||||
} else if (strcmp(driver, "ahci") == 0) {
|
||||
/* ahci: 1 host per 1 unit */
|
||||
if (!has_host || !has_tgt) {
|
||||
g_debug("invalid sysfs path '%s' (driver '%s')", syspath, driver);
|
||||
goto cleanup;
|
||||
}
|
||||
for (i = 0; i < nhosts; i++) {
|
||||
if (host == hosts[i]) {
|
||||
disk->unit = i;
|
||||
disk->bus_type = GUEST_DISK_BUS_TYPE_SATA;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (i >= nhosts) {
|
||||
g_debug("no host for '%s' (driver '%s')", syspath, driver);
|
||||
goto cleanup;
|
||||
}
|
||||
} else {
|
||||
g_debug("unknown driver '%s' (sysfs path '%s')", driver, syspath);
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
list->next = fs->disk;
|
||||
fs->disk = list;
|
||||
g_free(driver);
|
||||
return;
|
||||
|
||||
cleanup:
|
||||
if (list) {
|
||||
qapi_free_GuestDiskAddressList(list);
|
||||
}
|
||||
g_free(driver);
|
||||
}
|
||||
|
||||
static void build_guest_fsinfo_for_device(char const *devpath,
|
||||
GuestFilesystemInfo *fs,
|
||||
Error **errp);
|
||||
|
||||
/* Store a list of slave devices of virtual volume specified by @syspath into
|
||||
* @fs */
|
||||
static void build_guest_fsinfo_for_virtual_device(char const *syspath,
|
||||
GuestFilesystemInfo *fs,
|
||||
Error **errp)
|
||||
{
|
||||
DIR *dir;
|
||||
char *dirpath;
|
||||
struct dirent entry, *result;
|
||||
|
||||
dirpath = g_strdup_printf("%s/slaves", syspath);
|
||||
dir = opendir(dirpath);
|
||||
if (!dir) {
|
||||
error_setg_errno(errp, errno, "opendir(\"%s\")", dirpath);
|
||||
g_free(dirpath);
|
||||
return;
|
||||
}
|
||||
g_free(dirpath);
|
||||
|
||||
for (;;) {
|
||||
if (readdir_r(dir, &entry, &result) != 0) {
|
||||
error_setg_errno(errp, errno, "readdir_r(\"%s\")", dirpath);
|
||||
break;
|
||||
}
|
||||
if (!result) {
|
||||
break;
|
||||
}
|
||||
|
||||
if (entry.d_type == DT_LNK) {
|
||||
g_debug(" slave device '%s'", entry.d_name);
|
||||
dirpath = g_strdup_printf("%s/slaves/%s", syspath, entry.d_name);
|
||||
build_guest_fsinfo_for_device(dirpath, fs, errp);
|
||||
g_free(dirpath);
|
||||
|
||||
if (*errp) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
closedir(dir);
|
||||
}
|
||||
|
||||
/* Dispatch to functions for virtual/real device */
|
||||
static void build_guest_fsinfo_for_device(char const *devpath,
|
||||
GuestFilesystemInfo *fs,
|
||||
Error **errp)
|
||||
{
|
||||
char *syspath = realpath(devpath, NULL);
|
||||
|
||||
if (!syspath) {
|
||||
error_setg_errno(errp, errno, "realpath(\"%s\")", devpath);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!fs->name) {
|
||||
fs->name = g_strdup(basename(syspath));
|
||||
}
|
||||
|
||||
g_debug(" parse sysfs path '%s'", syspath);
|
||||
|
||||
if (strstr(syspath, "/devices/virtual/block/")) {
|
||||
build_guest_fsinfo_for_virtual_device(syspath, fs, errp);
|
||||
} else {
|
||||
build_guest_fsinfo_for_real_device(syspath, fs, errp);
|
||||
}
|
||||
|
||||
free(syspath);
|
||||
}
|
||||
|
||||
/* Return a list of the disk device(s)' info which @mount lies on */
|
||||
static GuestFilesystemInfo *build_guest_fsinfo(struct FsMount *mount,
|
||||
Error **errp)
|
||||
{
|
||||
GuestFilesystemInfo *fs = g_malloc0(sizeof(*fs));
|
||||
char *devpath = g_strdup_printf("/sys/dev/block/%u:%u",
|
||||
mount->devmajor, mount->devminor);
|
||||
|
||||
fs->mountpoint = g_strdup(mount->dirname);
|
||||
fs->type = g_strdup(mount->devtype);
|
||||
build_guest_fsinfo_for_device(devpath, fs, errp);
|
||||
|
||||
g_free(devpath);
|
||||
return fs;
|
||||
}
|
||||
|
||||
GuestFilesystemInfoList *qmp_guest_get_fsinfo(Error **errp)
|
||||
{
|
||||
FsMountList mounts;
|
||||
struct FsMount *mount;
|
||||
GuestFilesystemInfoList *new, *ret = NULL;
|
||||
Error *local_err = NULL;
|
||||
|
||||
QTAILQ_INIT(&mounts);
|
||||
build_fs_mount_list(&mounts, &local_err);
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
QTAILQ_FOREACH(mount, &mounts, next) {
|
||||
g_debug("Building guest fsinfo for '%s'", mount->dirname);
|
||||
|
||||
new = g_malloc0(sizeof(*ret));
|
||||
new->value = build_guest_fsinfo(mount, &local_err);
|
||||
new->next = ret;
|
||||
ret = new;
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
qapi_free_GuestFilesystemInfoList(ret);
|
||||
ret = NULL;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
free_fs_mount_list(&mounts);
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
||||
typedef enum {
|
||||
FSFREEZE_HOOK_THAW = 0,
|
||||
FSFREEZE_HOOK_FREEZE,
|
||||
@ -710,13 +1140,21 @@ GuestFsfreezeStatus qmp_guest_fsfreeze_status(Error **errp)
|
||||
return GUEST_FSFREEZE_STATUS_THAWED;
|
||||
}
|
||||
|
||||
int64_t qmp_guest_fsfreeze_freeze(Error **errp)
|
||||
{
|
||||
return qmp_guest_fsfreeze_freeze_list(false, NULL, errp);
|
||||
}
|
||||
|
||||
/*
|
||||
* Walk list of mounted file systems in the guest, and freeze the ones which
|
||||
* are real local file systems.
|
||||
*/
|
||||
int64_t qmp_guest_fsfreeze_freeze(Error **errp)
|
||||
int64_t qmp_guest_fsfreeze_freeze_list(bool has_mountpoints,
|
||||
strList *mountpoints,
|
||||
Error **errp)
|
||||
{
|
||||
int ret = 0, i = 0;
|
||||
strList *list;
|
||||
FsMountList mounts;
|
||||
struct FsMount *mount;
|
||||
Error *local_err = NULL;
|
||||
@ -741,6 +1179,19 @@ int64_t qmp_guest_fsfreeze_freeze(Error **errp)
|
||||
ga_set_frozen(ga_state);
|
||||
|
||||
QTAILQ_FOREACH_REVERSE(mount, &mounts, FsMountList, next) {
|
||||
/* To issue fsfreeze in the reverse order of mounts, check if the
|
||||
* mount is listed in the list here */
|
||||
if (has_mountpoints) {
|
||||
for (list = mountpoints; list; list = list->next) {
|
||||
if (strcmp(list->value, mount->dirname) == 0) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!list) {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
fd = qemu_open(mount->dirname, O_RDONLY);
|
||||
if (fd == -1) {
|
||||
error_setg_errno(errp, errno, "failed to open %s", mount->dirname);
|
||||
@ -1460,6 +1911,12 @@ int64_t qmp_guest_set_vcpus(GuestLogicalProcessorList *vcpus, Error **errp)
|
||||
|
||||
#if !defined(CONFIG_FSFREEZE)
|
||||
|
||||
GuestFilesystemInfoList *qmp_guest_get_fsinfo(Error **errp)
|
||||
{
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
GuestFsfreezeStatus qmp_guest_fsfreeze_status(Error **errp)
|
||||
{
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
@ -1474,6 +1931,15 @@ int64_t qmp_guest_fsfreeze_freeze(Error **errp)
|
||||
return 0;
|
||||
}
|
||||
|
||||
int64_t qmp_guest_fsfreeze_freeze_list(bool has_mountpoints,
|
||||
strList *mountpoints,
|
||||
Error **errp)
|
||||
{
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int64_t qmp_guest_fsfreeze_thaw(Error **errp)
|
||||
{
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
@ -1489,6 +1955,44 @@ void qmp_guest_fstrim(bool has_minimum, int64_t minimum, Error **errp)
|
||||
}
|
||||
#endif
|
||||
|
||||
/* add unsupported commands to the blacklist */
|
||||
GList *ga_command_blacklist_init(GList *blacklist)
|
||||
{
|
||||
#if !defined(__linux__)
|
||||
{
|
||||
const char *list[] = {
|
||||
"guest-suspend-disk", "guest-suspend-ram",
|
||||
"guest-suspend-hybrid", "guest-network-get-interfaces",
|
||||
"guest-get-vcpus", "guest-set-vcpus", NULL};
|
||||
char **p = (char **)list;
|
||||
|
||||
while (*p) {
|
||||
blacklist = g_list_append(blacklist, *p++);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#if !defined(CONFIG_FSFREEZE)
|
||||
{
|
||||
const char *list[] = {
|
||||
"guest-get-fsinfo", "guest-fsfreeze-status",
|
||||
"guest-fsfreeze-freeze", "guest-fsfreeze-freeze-list",
|
||||
"guest-fsfreeze-thaw", "guest-get-fsinfo", NULL};
|
||||
char **p = (char **)list;
|
||||
|
||||
while (*p) {
|
||||
blacklist = g_list_append(blacklist, *p++);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#if !defined(CONFIG_FSTRIM)
|
||||
blacklist = g_list_append(blacklist, (char *)"guest-fstrim");
|
||||
#endif
|
||||
|
||||
return blacklist;
|
||||
}
|
||||
|
||||
/* register init/cleanup routines for stateful command groups */
|
||||
void ga_command_state_init(GAState *s, GACommandState *cs)
|
||||
{
|
||||
|
@ -152,6 +152,12 @@ void qmp_guest_file_flush(int64_t handle, Error **errp)
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
}
|
||||
|
||||
GuestFilesystemInfoList *qmp_guest_get_fsinfo(Error **errp)
|
||||
{
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* Return status of freeze/thaw
|
||||
*/
|
||||
@ -206,6 +212,15 @@ error:
|
||||
return 0;
|
||||
}
|
||||
|
||||
int64_t qmp_guest_fsfreeze_freeze_list(bool has_mountpoints,
|
||||
strList *mountpoints,
|
||||
Error **errp)
|
||||
{
|
||||
error_set(errp, QERR_UNSUPPORTED);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Thaw local file systems using Volume Shadow-copy Service.
|
||||
*/
|
||||
@ -431,10 +446,40 @@ int64_t qmp_guest_set_vcpus(GuestLogicalProcessorList *vcpus, Error **errp)
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* add unsupported commands to the blacklist */
|
||||
GList *ga_command_blacklist_init(GList *blacklist)
|
||||
{
|
||||
const char *list_unsupported[] = {
|
||||
"guest-file-open", "guest-file-close", "guest-file-read",
|
||||
"guest-file-write", "guest-file-seek", "guest-file-flush",
|
||||
"guest-suspend-hybrid", "guest-network-get-interfaces",
|
||||
"guest-get-vcpus", "guest-set-vcpus",
|
||||
"guest-fsfreeze-freeze-list", "guest-get-fsinfo",
|
||||
"guest-fstrim", NULL};
|
||||
char **p = (char **)list_unsupported;
|
||||
|
||||
while (*p) {
|
||||
blacklist = g_list_append(blacklist, *p++);
|
||||
}
|
||||
|
||||
if (!vss_init(true)) {
|
||||
const char *list[] = {
|
||||
"guest-get-fsinfo", "guest-fsfreeze-status",
|
||||
"guest-fsfreeze-freeze", "guest-fsfreeze-thaw", NULL};
|
||||
p = (char **)list;
|
||||
|
||||
while (*p) {
|
||||
blacklist = g_list_append(blacklist, *p++);
|
||||
}
|
||||
}
|
||||
|
||||
return blacklist;
|
||||
}
|
||||
|
||||
/* register init/cleanup routines for stateful command groups */
|
||||
void ga_command_state_init(GAState *s, GACommandState *cs)
|
||||
{
|
||||
if (vss_init(true)) {
|
||||
if (!vss_initialized()) {
|
||||
ga_command_state_add(cs, NULL, guest_fsfreeze_cleanup);
|
||||
}
|
||||
}
|
||||
|
@ -19,6 +19,7 @@ typedef struct GAState GAState;
|
||||
typedef struct GACommandState GACommandState;
|
||||
extern GAState *ga_state;
|
||||
|
||||
GList *ga_command_blacklist_init(GList *blacklist);
|
||||
void ga_command_state_init(GAState *s, GACommandState *cs);
|
||||
void ga_command_state_add(GACommandState *cs,
|
||||
void (*init)(void),
|
||||
|
@ -1144,6 +1144,7 @@ int main(int argc, char **argv)
|
||||
goto out_bad;
|
||||
}
|
||||
|
||||
blacklist = ga_command_blacklist_init(blacklist);
|
||||
if (blacklist) {
|
||||
s->blacklist = blacklist;
|
||||
do {
|
||||
|
@ -386,6 +386,23 @@
|
||||
{ 'command': 'guest-fsfreeze-freeze',
|
||||
'returns': 'int' }
|
||||
|
||||
##
|
||||
# @guest-fsfreeze-freeze-list:
|
||||
#
|
||||
# Sync and freeze specified guest filesystems
|
||||
#
|
||||
# @mountpoints: #optional an array of mountpoints of filesystems to be frozen.
|
||||
# If omitted, every mounted filesystem is frozen.
|
||||
#
|
||||
# Returns: Number of file systems currently frozen. On error, all filesystems
|
||||
# will be thawed.
|
||||
#
|
||||
# Since: 2.2
|
||||
##
|
||||
{ 'command': 'guest-fsfreeze-freeze-list',
|
||||
'data': { '*mountpoints': ['str'] },
|
||||
'returns': 'int' }
|
||||
|
||||
##
|
||||
# @guest-fsfreeze-thaw:
|
||||
#
|
||||
@ -642,3 +659,82 @@
|
||||
{ 'command': 'guest-set-vcpus',
|
||||
'data': {'vcpus': ['GuestLogicalProcessor'] },
|
||||
'returns': 'int' }
|
||||
|
||||
##
|
||||
# @GuestDiskBusType
|
||||
#
|
||||
# An enumeration of bus type of disks
|
||||
#
|
||||
# @ide: IDE disks
|
||||
# @fdc: floppy disks
|
||||
# @scsi: SCSI disks
|
||||
# @virtio: virtio disks
|
||||
# @xen: Xen disks
|
||||
# @usb: USB disks
|
||||
# @uml: UML disks
|
||||
# @sata: SATA disks
|
||||
# @sd: SD cards
|
||||
#
|
||||
# Since: 2.2
|
||||
##
|
||||
{ 'enum': 'GuestDiskBusType',
|
||||
'data': [ 'ide', 'fdc', 'scsi', 'virtio', 'xen', 'usb', 'uml', 'sata',
|
||||
'sd' ] }
|
||||
|
||||
##
|
||||
# @GuestPCIAddress:
|
||||
#
|
||||
# @domain: domain id
|
||||
# @bus: bus id
|
||||
# @slot: slot id
|
||||
# @function: function id
|
||||
#
|
||||
# Since: 2.2
|
||||
##
|
||||
{ 'type': 'GuestPCIAddress',
|
||||
'data': {'domain': 'int', 'bus': 'int',
|
||||
'slot': 'int', 'function': 'int'} }
|
||||
|
||||
##
|
||||
# @GuestDiskAddress:
|
||||
#
|
||||
# @pci-controller: controller's PCI address
|
||||
# @type: bus type
|
||||
# @bus: bus id
|
||||
# @target: target id
|
||||
# @unit: unit id
|
||||
#
|
||||
# Since: 2.2
|
||||
##
|
||||
{ 'type': 'GuestDiskAddress',
|
||||
'data': {'pci-controller': 'GuestPCIAddress',
|
||||
'bus-type': 'GuestDiskBusType',
|
||||
'bus': 'int', 'target': 'int', 'unit': 'int'} }
|
||||
|
||||
##
|
||||
# @GuestFilesystemInfo
|
||||
#
|
||||
# @name: disk name
|
||||
# @mountpoint: mount point path
|
||||
# @type: file system type string
|
||||
# @disk: an array of disk hardware information that the volume lies on,
|
||||
# which may be empty if the disk type is not supported
|
||||
#
|
||||
# Since: 2.2
|
||||
##
|
||||
{ 'type': 'GuestFilesystemInfo',
|
||||
'data': {'name': 'str', 'mountpoint': 'str', 'type': 'str',
|
||||
'disk': ['GuestDiskAddress']} }
|
||||
|
||||
##
|
||||
# @guest-get-fsinfo:
|
||||
#
|
||||
# Returns: The list of filesystems information mounted in the guest.
|
||||
# The returned mountpoints may be specified to
|
||||
# @guest-fsfreeze-freeze-list.
|
||||
# Network filesystems (such as CIFS and NFS) are not listed.
|
||||
#
|
||||
# Since: 2.2
|
||||
##
|
||||
{ 'command': 'guest-get-fsinfo',
|
||||
'returns': ['GuestFilesystemInfo'] }
|
||||
|
Loading…
Reference in New Issue
Block a user