提交 674ed722 编写于 作者: M Markus Armbruster

qmp qemu-ga: Fix qemu-ga not to accept "control"

Commit cf869d53 "qmp: support out-of-band (oob) execution"
accidentally made qemu-ga accept and ignore "control".  Fix that.

Out-of-band execution in a monitor that doesn't support it now fails
with

    {"error": {"class": "GenericError", "desc": "QMP input member 'control' is unexpected"}}

instead of

    {"error": {"class": "GenericError", "desc": "Please enable out-of-band first for the session during capabilities negotiation"}}

The old description is suboptimal when out-of-band cannot not be
enabled, or the command doesn't support out-of-band execution.

The new description is a bit unspecific, but it'll do.
Signed-off-by: NMarkus Armbruster <armbru@redhat.com>
Reviewed-by: NEric Blake <eblake@redhat.com>
Message-Id: <20180703085358.13941-12-armbru@redhat.com>
上级 d4d7ed73
...@@ -41,7 +41,6 @@ void qmp_register_command(QmpCommandList *cmds, const char *name, ...@@ -41,7 +41,6 @@ void qmp_register_command(QmpCommandList *cmds, const char *name,
QmpCommandFunc *fn, QmpCommandOptions options); QmpCommandFunc *fn, QmpCommandOptions options);
void qmp_unregister_command(QmpCommandList *cmds, const char *name); void qmp_unregister_command(QmpCommandList *cmds, const char *name);
QmpCommand *qmp_find_command(QmpCommandList *cmds, const char *name); QmpCommand *qmp_find_command(QmpCommandList *cmds, const char *name);
QObject *qmp_dispatch(QmpCommandList *cmds, QObject *request);
void qmp_disable_command(QmpCommandList *cmds, const char *name); void qmp_disable_command(QmpCommandList *cmds, const char *name);
void qmp_enable_command(QmpCommandList *cmds, const char *name); void qmp_enable_command(QmpCommandList *cmds, const char *name);
...@@ -49,7 +48,10 @@ bool qmp_command_is_enabled(const QmpCommand *cmd); ...@@ -49,7 +48,10 @@ bool qmp_command_is_enabled(const QmpCommand *cmd);
const char *qmp_command_name(const QmpCommand *cmd); const char *qmp_command_name(const QmpCommand *cmd);
bool qmp_has_success_response(const QmpCommand *cmd); bool qmp_has_success_response(const QmpCommand *cmd);
QObject *qmp_build_error_object(Error *err); QObject *qmp_build_error_object(Error *err);
QDict *qmp_dispatch_check_obj(const QObject *request, Error **errp); QDict *qmp_dispatch_check_obj(const QObject *request, bool allow_oob,
Error **errp);
QObject *qmp_dispatch(QmpCommandList *cmds, QObject *request,
bool allow_oob);
bool qmp_is_oob(QDict *dict); bool qmp_is_oob(QDict *dict);
typedef void (*qmp_cmd_callback_fn)(QmpCommand *cmd, void *opaque); typedef void (*qmp_cmd_callback_fn)(QmpCommand *cmd, void *opaque);
......
...@@ -1319,11 +1319,6 @@ static bool qmp_cmd_oob_check(Monitor *mon, QDict *req, Error **errp) ...@@ -1319,11 +1319,6 @@ static bool qmp_cmd_oob_check(Monitor *mon, QDict *req, Error **errp)
} }
if (qmp_is_oob(req)) { if (qmp_is_oob(req)) {
if (!qmp_oob_enabled(mon)) {
error_setg(errp, "Please enable out-of-band first "
"for the session during capabilities negotiation");
return false;
}
if (!(cmd->options & QCO_ALLOW_OOB)) { if (!(cmd->options & QCO_ALLOW_OOB)) {
error_setg(errp, "The command %s does not support OOB", error_setg(errp, "The command %s does not support OOB",
command); command);
...@@ -4195,7 +4190,7 @@ static void monitor_qmp_dispatch_one(QMPRequest *req_obj) ...@@ -4195,7 +4190,7 @@ static void monitor_qmp_dispatch_one(QMPRequest *req_obj)
old_mon = cur_mon; old_mon = cur_mon;
cur_mon = mon; cur_mon = mon;
rsp = qmp_dispatch(mon->qmp.commands, req); rsp = qmp_dispatch(mon->qmp.commands, req, qmp_oob_enabled(mon));
cur_mon = old_mon; cur_mon = old_mon;
...@@ -4286,7 +4281,7 @@ static void handle_qmp_command(JSONMessageParser *parser, GQueue *tokens) ...@@ -4286,7 +4281,7 @@ static void handle_qmp_command(JSONMessageParser *parser, GQueue *tokens)
} /* else will fail qmp_dispatch() */ } /* else will fail qmp_dispatch() */
/* Check against the request in general layout */ /* Check against the request in general layout */
qdict = qmp_dispatch_check_obj(req, &err); qdict = qmp_dispatch_check_obj(req, qmp_oob_enabled(mon), &err);
if (!qdict) { if (!qdict) {
goto err; goto err;
} }
......
...@@ -20,7 +20,8 @@ ...@@ -20,7 +20,8 @@
#include "qapi/qmp/qbool.h" #include "qapi/qmp/qbool.h"
#include "sysemu/sysemu.h" #include "sysemu/sysemu.h"
QDict *qmp_dispatch_check_obj(const QObject *request, Error **errp) QDict *qmp_dispatch_check_obj(const QObject *request, bool allow_oob,
Error **errp)
{ {
const QDictEntry *ent; const QDictEntry *ent;
const char *arg_name; const char *arg_name;
...@@ -52,7 +53,7 @@ QDict *qmp_dispatch_check_obj(const QObject *request, Error **errp) ...@@ -52,7 +53,7 @@ QDict *qmp_dispatch_check_obj(const QObject *request, Error **errp)
"QMP input member 'arguments' must be an object"); "QMP input member 'arguments' must be an object");
return NULL; return NULL;
} }
} else if (!strcmp(arg_name, "control")) { } else if (!strcmp(arg_name, "control") && allow_oob) {
if (qobject_type(arg_obj) != QTYPE_QDICT) { if (qobject_type(arg_obj) != QTYPE_QDICT) {
error_setg(errp, error_setg(errp,
"QMP input member 'control' must be a dict"); "QMP input member 'control' must be a dict");
...@@ -74,7 +75,7 @@ QDict *qmp_dispatch_check_obj(const QObject *request, Error **errp) ...@@ -74,7 +75,7 @@ QDict *qmp_dispatch_check_obj(const QObject *request, Error **errp)
} }
static QObject *do_qmp_dispatch(QmpCommandList *cmds, QObject *request, static QObject *do_qmp_dispatch(QmpCommandList *cmds, QObject *request,
Error **errp) bool allow_oob, Error **errp)
{ {
Error *local_err = NULL; Error *local_err = NULL;
const char *command; const char *command;
...@@ -82,7 +83,7 @@ static QObject *do_qmp_dispatch(QmpCommandList *cmds, QObject *request, ...@@ -82,7 +83,7 @@ static QObject *do_qmp_dispatch(QmpCommandList *cmds, QObject *request,
QmpCommand *cmd; QmpCommand *cmd;
QObject *ret = NULL; QObject *ret = NULL;
dict = qmp_dispatch_check_obj(request, errp); dict = qmp_dispatch_check_obj(request, allow_oob, errp);
if (!dict) { if (!dict) {
return NULL; return NULL;
} }
...@@ -157,13 +158,14 @@ bool qmp_is_oob(QDict *dict) ...@@ -157,13 +158,14 @@ bool qmp_is_oob(QDict *dict)
return qbool_get_bool(bool_obj); return qbool_get_bool(bool_obj);
} }
QObject *qmp_dispatch(QmpCommandList *cmds, QObject *request) QObject *qmp_dispatch(QmpCommandList *cmds, QObject *request,
bool allow_oob)
{ {
Error *err = NULL; Error *err = NULL;
QObject *ret; QObject *ret;
QDict *rsp; QDict *rsp;
ret = do_qmp_dispatch(cmds, request, &err); ret = do_qmp_dispatch(cmds, request, allow_oob, &err);
rsp = qdict_new(); rsp = qdict_new();
if (err) { if (err) {
......
...@@ -586,7 +586,7 @@ static void process_command(GAState *s, QDict *req) ...@@ -586,7 +586,7 @@ static void process_command(GAState *s, QDict *req)
g_assert(req); g_assert(req);
g_debug("processing command"); g_debug("processing command");
rsp = qmp_dispatch(&ga_commands, QOBJECT(req)); rsp = qmp_dispatch(&ga_commands, QOBJECT(req), false);
if (rsp) { if (rsp) {
ret = send_response(s, rsp); ret = send_response(s, rsp);
if (ret < 0) { if (ret < 0) {
......
...@@ -245,16 +245,17 @@ static void test_qga_invalid_id(gconstpointer fix) ...@@ -245,16 +245,17 @@ static void test_qga_invalid_id(gconstpointer fix)
static void test_qga_invalid_oob(gconstpointer fix) static void test_qga_invalid_oob(gconstpointer fix)
{ {
/* FIXME "control" is ignored; it should be rejected */
const TestFixture *fixture = fix; const TestFixture *fixture = fix;
QDict *ret; QDict *ret, *error;
const char *class;
ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping'," ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping',"
" 'control': {'run-oob': true}}"); " 'control': {'run-oob': true}}");
g_assert_nonnull(ret); g_assert_nonnull(ret);
qmp_assert_no_error(ret);
qdict_get_qdict(ret, "return"); error = qdict_get_qdict(ret, "error");
class = qdict_get_try_str(error, "class");
g_assert_cmpstr(class, ==, "GenericError");
qobject_unref(ret); qobject_unref(ret);
} }
......
...@@ -102,7 +102,7 @@ static void test_dispatch_cmd(void) ...@@ -102,7 +102,7 @@ static void test_dispatch_cmd(void)
qdict_put_str(req, "execute", "user_def_cmd"); qdict_put_str(req, "execute", "user_def_cmd");
resp = qmp_dispatch(&qmp_commands, QOBJECT(req)); resp = qmp_dispatch(&qmp_commands, QOBJECT(req), false);
assert(resp != NULL); assert(resp != NULL);
assert(!qdict_haskey(qobject_to(QDict, resp), "error")); assert(!qdict_haskey(qobject_to(QDict, resp), "error"));
...@@ -119,7 +119,7 @@ static void test_dispatch_cmd_failure(void) ...@@ -119,7 +119,7 @@ static void test_dispatch_cmd_failure(void)
qdict_put_str(req, "execute", "user_def_cmd2"); qdict_put_str(req, "execute", "user_def_cmd2");
resp = qmp_dispatch(&qmp_commands, QOBJECT(req)); resp = qmp_dispatch(&qmp_commands, QOBJECT(req), false);
assert(resp != NULL); assert(resp != NULL);
assert(qdict_haskey(qobject_to(QDict, resp), "error")); assert(qdict_haskey(qobject_to(QDict, resp), "error"));
...@@ -133,7 +133,7 @@ static void test_dispatch_cmd_failure(void) ...@@ -133,7 +133,7 @@ static void test_dispatch_cmd_failure(void)
qdict_put_str(req, "execute", "user_def_cmd"); qdict_put_str(req, "execute", "user_def_cmd");
resp = qmp_dispatch(&qmp_commands, QOBJECT(req)); resp = qmp_dispatch(&qmp_commands, QOBJECT(req), false);
assert(resp != NULL); assert(resp != NULL);
assert(qdict_haskey(qobject_to(QDict, resp), "error")); assert(qdict_haskey(qobject_to(QDict, resp), "error"));
...@@ -147,7 +147,7 @@ static QObject *test_qmp_dispatch(QDict *req) ...@@ -147,7 +147,7 @@ static QObject *test_qmp_dispatch(QDict *req)
QDict *resp; QDict *resp;
QObject *ret; QObject *ret;
resp_obj = qmp_dispatch(&qmp_commands, QOBJECT(req)); resp_obj = qmp_dispatch(&qmp_commands, QOBJECT(req), false);
assert(resp_obj); assert(resp_obj);
resp = qobject_to(QDict, resp_obj); resp = qobject_to(QDict, resp_obj);
assert(resp && !qdict_haskey(resp, "error")); assert(resp && !qdict_haskey(resp, "error"));
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册