mirror of
https://git.proxmox.com/git/qemu
synced 2025-08-08 09:32:33 +00:00
guest agent: add RPC blacklist command-line option
This adds a command-line option, -b/--blacklist, that accepts a comma-seperated list of RPCs to disable, or prints a list of available RPCs if passed "?". In consequence this also adds general blacklisting and RPC listing facilities to the new QMP dispatch/registry facilities, should the QMP monitor ever have a need for such a thing. Ideally, to avoid support/compatability issues in the future, blacklisting guest agent functionality will be the exceptional case, but we add the functionality here to handle guest administrators with specific requirements. Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com> Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com> Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
This commit is contained in:
parent
4cb016587a
commit
abd6cf6d8e
@ -31,11 +31,14 @@ typedef struct QmpCommand
|
|||||||
QmpCommandType type;
|
QmpCommandType type;
|
||||||
QmpCommandFunc *fn;
|
QmpCommandFunc *fn;
|
||||||
QTAILQ_ENTRY(QmpCommand) node;
|
QTAILQ_ENTRY(QmpCommand) node;
|
||||||
|
bool enabled;
|
||||||
} QmpCommand;
|
} QmpCommand;
|
||||||
|
|
||||||
void qmp_register_command(const char *name, QmpCommandFunc *fn);
|
void qmp_register_command(const char *name, QmpCommandFunc *fn);
|
||||||
QmpCommand *qmp_find_command(const char *name);
|
QmpCommand *qmp_find_command(const char *name);
|
||||||
QObject *qmp_dispatch(QObject *request);
|
QObject *qmp_dispatch(QObject *request);
|
||||||
|
void qmp_disable_command(const char *name);
|
||||||
|
char **qmp_get_command_list(void);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -79,6 +79,10 @@ static QObject *do_qmp_dispatch(QObject *request, Error **errp)
|
|||||||
error_set(errp, QERR_COMMAND_NOT_FOUND, command);
|
error_set(errp, QERR_COMMAND_NOT_FOUND, command);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
if (!cmd->enabled) {
|
||||||
|
error_set(errp, QERR_COMMAND_DISABLED, command);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (!qdict_haskey(dict, "arguments")) {
|
if (!qdict_haskey(dict, "arguments")) {
|
||||||
args = qdict_new();
|
args = qdict_new();
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
#include "qapi/qmp-core.h"
|
#include "qapi/qmp-core.h"
|
||||||
|
|
||||||
static QTAILQ_HEAD(, QmpCommand) qmp_commands =
|
static QTAILQ_HEAD(QmpCommandList, QmpCommand) qmp_commands =
|
||||||
QTAILQ_HEAD_INITIALIZER(qmp_commands);
|
QTAILQ_HEAD_INITIALIZER(qmp_commands);
|
||||||
|
|
||||||
void qmp_register_command(const char *name, QmpCommandFunc *fn)
|
void qmp_register_command(const char *name, QmpCommandFunc *fn)
|
||||||
@ -24,17 +24,50 @@ void qmp_register_command(const char *name, QmpCommandFunc *fn)
|
|||||||
cmd->name = name;
|
cmd->name = name;
|
||||||
cmd->type = QCT_NORMAL;
|
cmd->type = QCT_NORMAL;
|
||||||
cmd->fn = fn;
|
cmd->fn = fn;
|
||||||
|
cmd->enabled = true;
|
||||||
QTAILQ_INSERT_TAIL(&qmp_commands, cmd, node);
|
QTAILQ_INSERT_TAIL(&qmp_commands, cmd, node);
|
||||||
}
|
}
|
||||||
|
|
||||||
QmpCommand *qmp_find_command(const char *name)
|
QmpCommand *qmp_find_command(const char *name)
|
||||||
{
|
{
|
||||||
QmpCommand *i;
|
QmpCommand *cmd;
|
||||||
|
|
||||||
QTAILQ_FOREACH(i, &qmp_commands, node) {
|
QTAILQ_FOREACH(cmd, &qmp_commands, node) {
|
||||||
if (strcmp(i->name, name) == 0) {
|
if (strcmp(cmd->name, name) == 0) {
|
||||||
return i;
|
return cmd;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void qmp_disable_command(const char *name)
|
||||||
|
{
|
||||||
|
QmpCommand *cmd;
|
||||||
|
|
||||||
|
QTAILQ_FOREACH(cmd, &qmp_commands, node) {
|
||||||
|
if (strcmp(cmd->name, name) == 0) {
|
||||||
|
cmd->enabled = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char **qmp_get_command_list(void)
|
||||||
|
{
|
||||||
|
QmpCommand *cmd;
|
||||||
|
int count = 1;
|
||||||
|
char **list_head, **list;
|
||||||
|
|
||||||
|
QTAILQ_FOREACH(cmd, &qmp_commands, node) {
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
list_head = list = g_malloc0(count * sizeof(char *));
|
||||||
|
|
||||||
|
QTAILQ_FOREACH(cmd, &qmp_commands, node) {
|
||||||
|
*list = strdup(cmd->name);
|
||||||
|
list++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return list_head;
|
||||||
|
}
|
||||||
|
37
qemu-ga.c
37
qemu-ga.c
@ -27,6 +27,7 @@
|
|||||||
#include "signal.h"
|
#include "signal.h"
|
||||||
#include "qerror.h"
|
#include "qerror.h"
|
||||||
#include "error_int.h"
|
#include "error_int.h"
|
||||||
|
#include "qapi/qmp-core.h"
|
||||||
|
|
||||||
#define QGA_VIRTIO_PATH_DEFAULT "/dev/virtio-ports/org.qemu.guest_agent.0"
|
#define QGA_VIRTIO_PATH_DEFAULT "/dev/virtio-ports/org.qemu.guest_agent.0"
|
||||||
#define QGA_PIDFILE_DEFAULT "/var/run/qemu-ga.pid"
|
#define QGA_PIDFILE_DEFAULT "/var/run/qemu-ga.pid"
|
||||||
@ -91,6 +92,8 @@ static void usage(const char *cmd)
|
|||||||
" -v, --verbose log extra debugging information\n"
|
" -v, --verbose log extra debugging information\n"
|
||||||
" -V, --version print version information and exit\n"
|
" -V, --version print version information and exit\n"
|
||||||
" -d, --daemonize become a daemon\n"
|
" -d, --daemonize become a daemon\n"
|
||||||
|
" -b, --blacklist comma-seperated list of RPCs to disable (no spaces, \"?\""
|
||||||
|
" to list available RPCs)\n"
|
||||||
" -h, --help display this help and exit\n"
|
" -h, --help display this help and exit\n"
|
||||||
"\n"
|
"\n"
|
||||||
"Report bugs to <mdroth@linux.vnet.ibm.com>\n"
|
"Report bugs to <mdroth@linux.vnet.ibm.com>\n"
|
||||||
@ -548,7 +551,7 @@ static void init_guest_agent(GAState *s)
|
|||||||
|
|
||||||
int main(int argc, char **argv)
|
int main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
const char *sopt = "hVvdm:p:l:f:";
|
const char *sopt = "hVvdm:p:l:f:b:";
|
||||||
const char *method = NULL, *path = NULL, *pidfile = QGA_PIDFILE_DEFAULT;
|
const char *method = NULL, *path = NULL, *pidfile = QGA_PIDFILE_DEFAULT;
|
||||||
const struct option lopt[] = {
|
const struct option lopt[] = {
|
||||||
{ "help", 0, NULL, 'h' },
|
{ "help", 0, NULL, 'h' },
|
||||||
@ -559,13 +562,16 @@ int main(int argc, char **argv)
|
|||||||
{ "method", 0, NULL, 'm' },
|
{ "method", 0, NULL, 'm' },
|
||||||
{ "path", 0, NULL, 'p' },
|
{ "path", 0, NULL, 'p' },
|
||||||
{ "daemonize", 0, NULL, 'd' },
|
{ "daemonize", 0, NULL, 'd' },
|
||||||
|
{ "blacklist", 0, NULL, 'b' },
|
||||||
{ NULL, 0, NULL, 0 }
|
{ NULL, 0, NULL, 0 }
|
||||||
};
|
};
|
||||||
int opt_ind = 0, ch, daemonize = 0;
|
int opt_ind = 0, ch, daemonize = 0, i, j, len;
|
||||||
GLogLevelFlags log_level = G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL;
|
GLogLevelFlags log_level = G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL;
|
||||||
FILE *log_file = stderr;
|
FILE *log_file = stderr;
|
||||||
GAState *s;
|
GAState *s;
|
||||||
|
|
||||||
|
module_call_init(MODULE_INIT_QAPI);
|
||||||
|
|
||||||
while ((ch = getopt_long(argc, argv, sopt, lopt, &opt_ind)) != -1) {
|
while ((ch = getopt_long(argc, argv, sopt, lopt, &opt_ind)) != -1) {
|
||||||
switch (ch) {
|
switch (ch) {
|
||||||
case 'm':
|
case 'm':
|
||||||
@ -595,6 +601,32 @@ int main(int argc, char **argv)
|
|||||||
case 'd':
|
case 'd':
|
||||||
daemonize = 1;
|
daemonize = 1;
|
||||||
break;
|
break;
|
||||||
|
case 'b': {
|
||||||
|
char **list_head, **list;
|
||||||
|
if (*optarg == '?') {
|
||||||
|
list_head = list = qmp_get_command_list();
|
||||||
|
while (*list != NULL) {
|
||||||
|
printf("%s\n", *list);
|
||||||
|
g_free(*list);
|
||||||
|
list++;
|
||||||
|
}
|
||||||
|
g_free(list_head);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
for (j = 0, i = 0, len = strlen(optarg); i < len; i++) {
|
||||||
|
if (optarg[i] == ',') {
|
||||||
|
optarg[i] = 0;
|
||||||
|
qmp_disable_command(&optarg[j]);
|
||||||
|
g_debug("disabling command: %s", &optarg[j]);
|
||||||
|
j = i + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (j < i) {
|
||||||
|
qmp_disable_command(&optarg[j]);
|
||||||
|
g_debug("disabling command: %s", &optarg[j]);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
case 'h':
|
case 'h':
|
||||||
usage(argv[0]);
|
usage(argv[0]);
|
||||||
return 0;
|
return 0;
|
||||||
@ -624,7 +656,6 @@ int main(int argc, char **argv)
|
|||||||
ga_command_state_init_all(s->command_state);
|
ga_command_state_init_all(s->command_state);
|
||||||
ga_state = s;
|
ga_state = s;
|
||||||
|
|
||||||
module_call_init(MODULE_INIT_QAPI);
|
|
||||||
init_guest_agent(ga_state);
|
init_guest_agent(ga_state);
|
||||||
register_signal_handlers();
|
register_signal_handlers();
|
||||||
|
|
||||||
|
4
qerror.c
4
qerror.c
@ -64,6 +64,10 @@ static const QErrorStringTable qerror_table[] = {
|
|||||||
.error_fmt = QERR_COMMAND_NOT_FOUND,
|
.error_fmt = QERR_COMMAND_NOT_FOUND,
|
||||||
.desc = "The command %(name) has not been found",
|
.desc = "The command %(name) has not been found",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
.error_fmt = QERR_COMMAND_DISABLED,
|
||||||
|
.desc = "The command %(name) has been disabled for this instance",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
.error_fmt = QERR_DEVICE_ENCRYPTED,
|
.error_fmt = QERR_DEVICE_ENCRYPTED,
|
||||||
.desc = "Device '%(device)' is encrypted",
|
.desc = "Device '%(device)' is encrypted",
|
||||||
|
3
qerror.h
3
qerror.h
@ -66,6 +66,9 @@ QError *qobject_to_qerror(const QObject *obj);
|
|||||||
#define QERR_COMMAND_NOT_FOUND \
|
#define QERR_COMMAND_NOT_FOUND \
|
||||||
"{ 'class': 'CommandNotFound', 'data': { 'name': %s } }"
|
"{ 'class': 'CommandNotFound', 'data': { 'name': %s } }"
|
||||||
|
|
||||||
|
#define QERR_COMMAND_DISABLED \
|
||||||
|
"{ 'class': 'CommandDisabled', 'data': { 'name': %s } }"
|
||||||
|
|
||||||
#define QERR_DEVICE_ENCRYPTED \
|
#define QERR_DEVICE_ENCRYPTED \
|
||||||
"{ 'class': 'DeviceEncrypted', 'data': { 'device': %s } }"
|
"{ 'class': 'DeviceEncrypted', 'data': { 'device': %s } }"
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user