2009-11-19 02:05:30 +01:00
|
|
|
/*
|
2010-05-12 21:34:42 +02:00
|
|
|
* QError Module
|
2009-11-19 02:05:30 +01:00
|
|
|
*
|
|
|
|
* Copyright (C) 2009 Red Hat Inc.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Luiz Capitulino <lcapitulino@redhat.com>
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
|
|
|
|
* See the COPYING.LIB file in the top-level directory.
|
|
|
|
*/
|
2010-03-22 10:29:05 +01:00
|
|
|
|
|
|
|
#include "monitor.h"
|
2009-11-19 02:05:30 +01:00
|
|
|
#include "qjson.h"
|
|
|
|
#include "qerror.h"
|
|
|
|
#include "qemu-common.h"
|
|
|
|
|
|
|
|
static void qerror_destroy_obj(QObject *obj);
|
|
|
|
|
|
|
|
static const QType qerror_type = {
|
|
|
|
.code = QTYPE_QERROR,
|
|
|
|
.destroy = qerror_destroy_obj,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The 'desc' parameter is a printf-like string, the format of the format
|
|
|
|
* string is:
|
|
|
|
*
|
|
|
|
* %(KEY)
|
|
|
|
*
|
|
|
|
* Where KEY is a QDict key, which has to be passed to qerror_from_info().
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
*
|
|
|
|
* "foo error on device: %(device) slot: %(slot_nr)"
|
|
|
|
*
|
|
|
|
* A single percent sign can be printed if followed by a second one,
|
|
|
|
* for example:
|
|
|
|
*
|
|
|
|
* "running out of foo: %(foo)%%"
|
2010-03-25 17:22:30 +01:00
|
|
|
*
|
|
|
|
* Please keep the entries in alphabetical order.
|
2012-01-04 23:23:32 +01:00
|
|
|
* Use scripts/check-qerror.sh to check.
|
2009-11-19 02:05:30 +01:00
|
|
|
*/
|
2009-12-04 21:44:44 +01:00
|
|
|
static const QErrorStringTable qerror_table[] = {
|
2010-02-19 19:52:45 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_BAD_BUS_FOR_DEVICE,
|
|
|
|
.desc = "Device '%(device)' can't go on a %(bad_bus_type) bus",
|
|
|
|
},
|
2011-11-14 22:09:47 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_BLOCK_FORMAT_FEATURE_NOT_SUPPORTED,
|
|
|
|
.desc = "Block format '%(format)' used by device '%(name)' does not support feature '%(feature)'",
|
|
|
|
},
|
2010-02-19 18:05:39 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_BUS_NOT_FOUND,
|
|
|
|
.desc = "Bus '%(bus)' not found",
|
|
|
|
},
|
2010-02-19 19:53:36 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_BUS_NO_HOTPLUG,
|
|
|
|
.desc = "Bus '%(bus)' does not support hotplugging",
|
|
|
|
},
|
2009-11-27 01:58:57 +01:00
|
|
|
{
|
2009-12-07 21:37:02 +01:00
|
|
|
.error_fmt = QERR_COMMAND_NOT_FOUND,
|
|
|
|
.desc = "The command %(name) has not been found",
|
2009-11-27 01:58:57 +01:00
|
|
|
},
|
2011-12-07 05:03:42 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_COMMAND_DISABLED,
|
|
|
|
.desc = "The command %(name) has been disabled for this instance",
|
|
|
|
},
|
2009-12-07 21:36:59 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_ENCRYPTED,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Device '%(device)' is encrypted",
|
2009-12-07 21:36:59 +01:00
|
|
|
},
|
2010-02-19 19:53:54 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_INIT_FAILED,
|
|
|
|
.desc = "Device '%(device)' could not be initialized",
|
|
|
|
},
|
2010-03-25 17:22:39 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_IN_USE,
|
|
|
|
.desc = "Device '%(device)' is in use",
|
|
|
|
},
|
2011-11-14 22:09:44 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_FEATURE_BLOCKS_MIGRATION,
|
|
|
|
.desc = "Migration is disabled when using feature '%(feature)' in device '%(device)'",
|
|
|
|
},
|
2009-12-07 21:37:03 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_LOCKED,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Device '%(device)' is locked",
|
2009-12-07 21:37:03 +01:00
|
|
|
},
|
2010-02-19 18:05:59 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_MULTIPLE_BUSSES,
|
|
|
|
.desc = "Device '%(device)' has multiple child busses",
|
|
|
|
},
|
2009-12-07 21:37:02 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_NOT_ACTIVE,
|
2010-06-23 17:37:47 +02:00
|
|
|
.desc = "Device '%(device)' has not been activated",
|
2009-12-07 21:37:02 +01:00
|
|
|
},
|
2010-03-25 17:22:30 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_NOT_ENCRYPTED,
|
|
|
|
.desc = "Device '%(device)' is not encrypted",
|
|
|
|
},
|
2009-11-19 02:05:32 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_NOT_FOUND,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Device '%(device)' not found",
|
2009-11-19 02:05:32 +01:00
|
|
|
},
|
2009-12-07 21:37:04 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_NOT_REMOVABLE,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Device '%(device)' is not removable",
|
2009-12-07 21:37:04 +01:00
|
|
|
},
|
2010-02-19 18:06:18 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_NO_BUS,
|
|
|
|
.desc = "Device '%(device)' has no child bus",
|
|
|
|
},
|
2011-01-06 15:14:37 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DEVICE_NO_HOTPLUG,
|
|
|
|
.desc = "Device '%(device)' does not support hotplugging",
|
|
|
|
},
|
2010-03-25 17:22:31 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_DUPLICATE_ID,
|
|
|
|
.desc = "Duplicate ID '%(id)' for %(object)",
|
|
|
|
},
|
2009-12-07 21:37:10 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_FD_NOT_FOUND,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "File descriptor named '%(name)' not found",
|
2009-12-07 21:37:10 +01:00
|
|
|
},
|
2009-12-07 21:37:12 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_FD_NOT_SUPPLIED,
|
|
|
|
.desc = "No file descriptor supplied via SCM_RIGHTS",
|
|
|
|
},
|
2011-10-19 19:15:57 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_FEATURE_DISABLED,
|
|
|
|
.desc = "The feature '%(name)' is not enabled",
|
|
|
|
},
|
2009-12-07 21:37:06 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_INVALID_BLOCK_FORMAT,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Invalid block format '%(name)'",
|
2009-12-07 21:37:06 +01:00
|
|
|
},
|
2009-12-07 21:37:13 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_INVALID_PARAMETER,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Invalid parameter '%(name)'",
|
2009-12-07 21:37:13 +01:00
|
|
|
},
|
2009-11-19 02:05:34 +01:00
|
|
|
{
|
2009-12-07 21:37:02 +01:00
|
|
|
.error_fmt = QERR_INVALID_PARAMETER_TYPE,
|
|
|
|
.desc = "Invalid parameter type, expected: %(expected)",
|
2009-11-19 02:05:34 +01:00
|
|
|
},
|
2010-03-25 17:22:33 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_INVALID_PARAMETER_VALUE,
|
|
|
|
.desc = "Parameter '%(name)' expects %(expected)",
|
|
|
|
},
|
2009-11-27 01:58:57 +01:00
|
|
|
{
|
2009-12-07 21:37:02 +01:00
|
|
|
.error_fmt = QERR_INVALID_PASSWORD,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Password incorrect",
|
2009-11-27 01:58:57 +01:00
|
|
|
},
|
2011-11-22 19:29:52 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_IO_ERROR,
|
|
|
|
.desc = "An IO error has occurred",
|
|
|
|
},
|
2009-12-04 18:24:08 +01:00
|
|
|
{
|
2009-12-07 21:37:02 +01:00
|
|
|
.error_fmt = QERR_JSON_PARSING,
|
|
|
|
.desc = "Invalid JSON syntax",
|
2009-12-04 18:24:08 +01:00
|
|
|
},
|
2011-07-19 22:41:52 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_JSON_PARSE_ERROR,
|
|
|
|
.desc = "JSON parse error, %(message)",
|
|
|
|
|
|
|
|
},
|
2009-11-19 02:05:35 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_KVM_MISSING_CAP,
|
|
|
|
.desc = "Using KVM without %(capability), %(feature) unavailable",
|
|
|
|
},
|
2010-07-27 12:19:19 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_MIGRATION_EXPECTED,
|
|
|
|
.desc = "An incoming migration is expected before this command can be executed",
|
|
|
|
},
|
2009-11-27 01:58:57 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_MISSING_PARAMETER,
|
2010-02-19 17:42:46 +01:00
|
|
|
.desc = "Parameter '%(name)' is missing",
|
|
|
|
},
|
2010-02-19 19:54:06 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_NO_BUS_FOR_DEVICE,
|
|
|
|
.desc = "No '%(bus)' bus found for device '%(device)'",
|
|
|
|
},
|
2010-02-19 17:42:46 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_OPEN_FILE_FAILED,
|
|
|
|
.desc = "Could not open '%(filename)'",
|
2009-11-27 01:58:57 +01:00
|
|
|
},
|
2011-12-12 21:29:26 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_PERMISSION_DENIED,
|
|
|
|
.desc = "Insufficient permission to perform this operation",
|
|
|
|
},
|
2010-02-19 13:11:41 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_PROPERTY_NOT_FOUND,
|
|
|
|
.desc = "Property '%(device).%(property)' not found",
|
|
|
|
},
|
2010-02-19 13:17:58 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_PROPERTY_VALUE_BAD,
|
|
|
|
.desc = "Property '%(device).%(property)' doesn't take value '%(value)'",
|
|
|
|
},
|
2010-03-16 17:40:48 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_PROPERTY_VALUE_IN_USE,
|
|
|
|
.desc = "Property '%(device).%(property)' can't take value '%(value)', it's in use",
|
|
|
|
},
|
2010-03-16 17:44:38 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_PROPERTY_VALUE_NOT_FOUND,
|
|
|
|
.desc = "Property '%(device).%(property)' can't find value '%(value)'",
|
|
|
|
},
|
2011-12-18 17:05:08 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_PROPERTY_VALUE_OUT_OF_RANGE,
|
|
|
|
.desc = "Property '%(device).%(property)' doesn't take "
|
|
|
|
"value %(value) (minimum: %(min), maximum: %(max)'",
|
|
|
|
},
|
2009-11-27 01:58:57 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_QMP_BAD_INPUT_OBJECT,
|
2010-04-07 19:53:49 +02:00
|
|
|
.desc = "Expected '%(expected)' in QMP input",
|
2009-11-27 01:58:57 +01:00
|
|
|
},
|
2010-04-07 19:46:33 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_QMP_BAD_INPUT_OBJECT_MEMBER,
|
|
|
|
.desc = "QMP input object member '%(member)' expects '%(expected)'",
|
|
|
|
},
|
2010-06-01 21:15:23 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_QMP_EXTRA_MEMBER,
|
|
|
|
.desc = "QMP input object member '%(member)' is unexpected",
|
|
|
|
},
|
2011-07-29 20:57:54 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_RESET_REQUIRED,
|
|
|
|
.desc = "Resetting the Virtual Machine is required",
|
|
|
|
},
|
2009-12-07 21:37:07 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_SET_PASSWD_FAILED,
|
|
|
|
.desc = "Could not set password",
|
|
|
|
},
|
Introduce a 'client_add' monitor command accepting an open FD
Allow client connections for VNC and socket based character
devices to be passed in over the monitor using SCM_RIGHTS.
One intended usage scenario is to start QEMU with VNC on a
UNIX domain socket. An unprivileged user which cannot access
the UNIX domain socket, can then connect to QEMU's VNC server
by passing an open FD to libvirt, which passes it onto QEMU.
{ "execute": "get_fd", "arguments": { "fdname": "myclient" } }
{ "return": {} }
{ "execute": "add_client", "arguments": { "protocol": "vnc",
"fdname": "myclient",
"skipauth": true } }
{ "return": {} }
In this case 'protocol' can be 'vnc' or 'spice', or the name
of a character device (eg from -chardev id=XXXX)
The 'skipauth' parameter can be used to skip any configured
VNC authentication scheme, which is useful if the mgmt layer
talking to the monitor has already authenticated the client
in another way.
* console.h: Define 'vnc_display_add_client' method
* monitor.c: Implement 'client_add' command
* qemu-char.c, qemu-char.h: Add 'qemu_char_add_client' method
* qerror.c, qerror.h: Add QERR_ADD_CLIENT_FAILED
* qmp-commands.hx: Declare 'client_add' command
* ui/vnc.c: Implement 'vnc_display_add_client' method
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2011-06-23 14:31:42 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_ADD_CLIENT_FAILED,
|
|
|
|
.desc = "Could not add client",
|
|
|
|
},
|
2009-12-07 21:37:14 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_TOO_MANY_FILES,
|
|
|
|
.desc = "Too many open files",
|
|
|
|
},
|
2009-11-27 01:58:57 +01:00
|
|
|
{
|
2009-12-07 21:37:02 +01:00
|
|
|
.error_fmt = QERR_UNDEFINED_ERROR,
|
2011-11-22 11:06:26 +01:00
|
|
|
.desc = "An undefined error has occurred",
|
2009-11-27 01:58:57 +01:00
|
|
|
},
|
2011-03-07 10:05:04 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_UNSUPPORTED,
|
|
|
|
.desc = "this feature or command is not currently supported",
|
|
|
|
},
|
2011-02-09 11:09:38 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_UNKNOWN_BLOCK_FORMAT_FEATURE,
|
|
|
|
.desc = "'%(device)' uses a %(format) feature which is not "
|
|
|
|
"supported by this qemu version: %(feature)",
|
|
|
|
},
|
2011-12-04 18:05:28 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_VIRTFS_FEATURE_BLOCKS_MIGRATION,
|
|
|
|
.desc = "Migration is disabled when VirtFS export path '%(path)' "
|
|
|
|
"is mounted in the guest using mount_tag '%(tag)'",
|
|
|
|
},
|
2009-12-07 21:37:08 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_VNC_SERVER_FAILED,
|
|
|
|
.desc = "Could not start VNC server on %(target)",
|
|
|
|
},
|
2011-07-19 22:41:55 +02:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_QGA_LOGGING_FAILED,
|
|
|
|
.desc = "Guest agent failed to log non-optional log statement",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.error_fmt = QERR_QGA_COMMAND_FAILED,
|
|
|
|
.desc = "Guest agent command failed, error was '%(message)'",
|
|
|
|
},
|
2011-11-08 06:00:31 +01:00
|
|
|
{
|
|
|
|
.error_fmt = QERR_INVALID_PARAMETER_COMBINATION,
|
2011-12-10 00:19:46 +01:00
|
|
|
.desc = "Invalid parameter combination",
|
2011-11-08 06:00:31 +01:00
|
|
|
},
|
2009-11-19 02:05:30 +01:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qerror_new(): Create a new QError
|
|
|
|
*
|
|
|
|
* Return strong reference.
|
|
|
|
*/
|
|
|
|
QError *qerror_new(void)
|
|
|
|
{
|
|
|
|
QError *qerr;
|
|
|
|
|
2011-08-21 05:09:37 +02:00
|
|
|
qerr = g_malloc0(sizeof(*qerr));
|
2009-11-19 02:05:30 +01:00
|
|
|
QOBJECT_INIT(qerr, &qerror_type);
|
|
|
|
|
|
|
|
return qerr;
|
|
|
|
}
|
|
|
|
|
2010-09-23 21:28:05 +02:00
|
|
|
static void GCC_FMT_ATTR(2, 3) qerror_abort(const QError *qerr,
|
|
|
|
const char *fmt, ...)
|
2009-11-19 02:05:30 +01:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
fprintf(stderr, "qerror: bad call in function '%s':\n", qerr->func);
|
|
|
|
fprintf(stderr, "qerror: -> ");
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vfprintf(stderr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
fprintf(stderr, "\nqerror: call at %s:%d\n", qerr->file, qerr->linenr);
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
2010-09-23 21:28:05 +02:00
|
|
|
static void GCC_FMT_ATTR(2, 0) qerror_set_data(QError *qerr,
|
|
|
|
const char *fmt, va_list *va)
|
2009-11-19 02:05:30 +01:00
|
|
|
{
|
|
|
|
QObject *obj;
|
|
|
|
|
|
|
|
obj = qobject_from_jsonv(fmt, va);
|
|
|
|
if (!obj) {
|
|
|
|
qerror_abort(qerr, "invalid format '%s'", fmt);
|
|
|
|
}
|
|
|
|
if (qobject_type(obj) != QTYPE_QDICT) {
|
|
|
|
qerror_abort(qerr, "error format is not a QDict '%s'", fmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
qerr->error = qobject_to_qdict(obj);
|
|
|
|
|
|
|
|
obj = qdict_get(qerr->error, "class");
|
|
|
|
if (!obj) {
|
|
|
|
qerror_abort(qerr, "missing 'class' key in '%s'", fmt);
|
|
|
|
}
|
|
|
|
if (qobject_type(obj) != QTYPE_QSTRING) {
|
|
|
|
qerror_abort(qerr, "'class' key value should be a QString");
|
|
|
|
}
|
|
|
|
|
|
|
|
obj = qdict_get(qerr->error, "data");
|
|
|
|
if (!obj) {
|
|
|
|
qerror_abort(qerr, "missing 'data' key in '%s'", fmt);
|
|
|
|
}
|
|
|
|
if (qobject_type(obj) != QTYPE_QDICT) {
|
|
|
|
qerror_abort(qerr, "'data' key value should be a QDICT");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qerror_set_desc(QError *qerr, const char *fmt)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
// FIXME: inefficient loop
|
|
|
|
|
|
|
|
for (i = 0; qerror_table[i].error_fmt; i++) {
|
|
|
|
if (strcmp(qerror_table[i].error_fmt, fmt) == 0) {
|
|
|
|
qerr->entry = &qerror_table[i];
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qerror_abort(qerr, "error format '%s' not found", fmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qerror_from_info(): Create a new QError from error information
|
|
|
|
*
|
|
|
|
* The information consists of:
|
|
|
|
*
|
|
|
|
* - file the file name of where the error occurred
|
|
|
|
* - linenr the line number of where the error occurred
|
|
|
|
* - func the function name of where the error occurred
|
|
|
|
* - fmt JSON printf-like dictionary, there must exist keys 'class' and
|
|
|
|
* 'data'
|
|
|
|
* - va va_list of all arguments specified by fmt
|
|
|
|
*
|
|
|
|
* Return strong reference.
|
|
|
|
*/
|
|
|
|
QError *qerror_from_info(const char *file, int linenr, const char *func,
|
|
|
|
const char *fmt, va_list *va)
|
|
|
|
{
|
|
|
|
QError *qerr;
|
|
|
|
|
|
|
|
qerr = qerror_new();
|
2010-02-18 19:46:49 +01:00
|
|
|
loc_save(&qerr->loc);
|
2009-11-19 02:05:30 +01:00
|
|
|
qerr->linenr = linenr;
|
|
|
|
qerr->file = file;
|
|
|
|
qerr->func = func;
|
|
|
|
|
|
|
|
if (!fmt) {
|
|
|
|
qerror_abort(qerr, "QDict not specified");
|
|
|
|
}
|
|
|
|
|
|
|
|
qerror_set_data(qerr, fmt, va);
|
|
|
|
qerror_set_desc(qerr, fmt);
|
|
|
|
|
|
|
|
return qerr;
|
|
|
|
}
|
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
static void parse_error(const QErrorStringTable *entry, int c)
|
2009-11-19 02:05:30 +01:00
|
|
|
{
|
2011-06-01 19:14:47 +02:00
|
|
|
fprintf(stderr, "expected '%c' in '%s'", c, entry->desc);
|
|
|
|
abort();
|
2009-11-19 02:05:30 +01:00
|
|
|
}
|
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
static const char *append_field(QDict *error, QString *outstr,
|
|
|
|
const QErrorStringTable *entry,
|
2009-11-19 02:05:30 +01:00
|
|
|
const char *start)
|
|
|
|
{
|
|
|
|
QObject *obj;
|
|
|
|
QDict *qdict;
|
|
|
|
QString *key_qs;
|
|
|
|
const char *end, *key;
|
|
|
|
|
|
|
|
if (*start != '%')
|
2011-06-01 19:14:47 +02:00
|
|
|
parse_error(entry, '%');
|
2009-11-19 02:05:30 +01:00
|
|
|
start++;
|
|
|
|
if (*start != '(')
|
2011-06-01 19:14:47 +02:00
|
|
|
parse_error(entry, '(');
|
2009-11-19 02:05:30 +01:00
|
|
|
start++;
|
|
|
|
|
|
|
|
end = strchr(start, ')');
|
|
|
|
if (!end)
|
2011-06-01 19:14:47 +02:00
|
|
|
parse_error(entry, ')');
|
2009-11-19 02:05:30 +01:00
|
|
|
|
|
|
|
key_qs = qstring_from_substr(start, 0, end - start - 1);
|
|
|
|
key = qstring_get_str(key_qs);
|
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
qdict = qobject_to_qdict(qdict_get(error, "data"));
|
2009-11-19 02:05:30 +01:00
|
|
|
obj = qdict_get(qdict, key);
|
|
|
|
if (!obj) {
|
2011-06-01 19:14:47 +02:00
|
|
|
abort();
|
2009-11-19 02:05:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (qobject_type(obj)) {
|
|
|
|
case QTYPE_QSTRING:
|
|
|
|
qstring_append(outstr, qdict_get_str(qdict, key));
|
|
|
|
break;
|
|
|
|
case QTYPE_QINT:
|
|
|
|
qstring_append_int(outstr, qdict_get_int(qdict, key));
|
|
|
|
break;
|
|
|
|
default:
|
2011-06-01 19:14:47 +02:00
|
|
|
abort();
|
2009-11-19 02:05:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QDECREF(key_qs);
|
|
|
|
return ++end;
|
|
|
|
}
|
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
static QString *qerror_format_desc(QDict *error,
|
|
|
|
const QErrorStringTable *entry)
|
2009-11-19 02:05:30 +01:00
|
|
|
{
|
|
|
|
QString *qstring;
|
2011-06-01 19:14:47 +02:00
|
|
|
const char *p;
|
2009-11-19 02:05:30 +01:00
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
assert(entry != NULL);
|
2009-11-19 02:05:30 +01:00
|
|
|
|
|
|
|
qstring = qstring_new();
|
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
for (p = entry->desc; *p != '\0';) {
|
2009-11-19 02:05:30 +01:00
|
|
|
if (*p != '%') {
|
|
|
|
qstring_append_chr(qstring, *p++);
|
|
|
|
} else if (*(p + 1) == '%') {
|
|
|
|
qstring_append_chr(qstring, '%');
|
|
|
|
p += 2;
|
|
|
|
} else {
|
2011-06-01 19:14:47 +02:00
|
|
|
p = append_field(error, qstring, entry, p);
|
2009-11-19 02:05:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-07 21:37:16 +01:00
|
|
|
return qstring;
|
|
|
|
}
|
|
|
|
|
2011-06-01 19:14:48 +02:00
|
|
|
QString *qerror_format(const char *fmt, QDict *error)
|
|
|
|
{
|
|
|
|
const QErrorStringTable *entry = NULL;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; qerror_table[i].error_fmt; i++) {
|
|
|
|
if (strcmp(qerror_table[i].error_fmt, fmt) == 0) {
|
|
|
|
entry = &qerror_table[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return qerror_format_desc(error, entry);
|
|
|
|
}
|
|
|
|
|
2011-06-01 19:14:47 +02:00
|
|
|
/**
|
|
|
|
* qerror_human(): Format QError data into human-readable string.
|
|
|
|
*/
|
|
|
|
QString *qerror_human(const QError *qerror)
|
|
|
|
{
|
|
|
|
return qerror_format_desc(qerror->error, qerror->entry);
|
|
|
|
}
|
|
|
|
|
2009-12-07 21:37:16 +01:00
|
|
|
/**
|
|
|
|
* qerror_print(): Print QError data
|
|
|
|
*
|
|
|
|
* This function will print the member 'desc' of the specified QError object,
|
2010-02-18 17:25:24 +01:00
|
|
|
* it uses error_report() for this, so that the output is routed to the right
|
2009-12-07 21:37:16 +01:00
|
|
|
* place (ie. stderr or Monitor's device).
|
|
|
|
*/
|
2010-02-18 19:46:49 +01:00
|
|
|
void qerror_print(QError *qerror)
|
2009-12-07 21:37:16 +01:00
|
|
|
{
|
|
|
|
QString *qstring = qerror_human(qerror);
|
2010-02-18 19:46:49 +01:00
|
|
|
loc_push_restore(&qerror->loc);
|
2010-02-18 17:25:24 +01:00
|
|
|
error_report("%s", qstring_get_str(qstring));
|
2010-02-18 19:46:49 +01:00
|
|
|
loc_pop(&qerror->loc);
|
2009-11-19 02:05:30 +01:00
|
|
|
QDECREF(qstring);
|
|
|
|
}
|
|
|
|
|
2010-03-22 10:29:05 +01:00
|
|
|
void qerror_report_internal(const char *file, int linenr, const char *func,
|
|
|
|
const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list va;
|
|
|
|
QError *qerror;
|
|
|
|
|
|
|
|
va_start(va, fmt);
|
|
|
|
qerror = qerror_from_info(file, linenr, func, fmt, &va);
|
|
|
|
va_end(va);
|
|
|
|
|
|
|
|
if (monitor_cur_is_qmp()) {
|
|
|
|
monitor_set_error(cur_mon, qerror);
|
|
|
|
} else {
|
|
|
|
qerror_print(qerror);
|
|
|
|
QDECREF(qerror);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-02 19:34:45 +02:00
|
|
|
/* Evil... */
|
|
|
|
struct Error
|
|
|
|
{
|
|
|
|
QDict *obj;
|
|
|
|
const char *fmt;
|
|
|
|
char *msg;
|
|
|
|
};
|
|
|
|
|
|
|
|
void qerror_report_err(Error *err)
|
|
|
|
{
|
|
|
|
QError *qerr;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
qerr = qerror_new();
|
|
|
|
loc_save(&qerr->loc);
|
|
|
|
QINCREF(err->obj);
|
|
|
|
qerr->error = err->obj;
|
|
|
|
|
|
|
|
for (i = 0; qerror_table[i].error_fmt; i++) {
|
|
|
|
if (strcmp(qerror_table[i].error_fmt, err->fmt) == 0) {
|
|
|
|
qerr->entry = &qerror_table[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (monitor_cur_is_qmp()) {
|
|
|
|
monitor_set_error(cur_mon, qerr);
|
|
|
|
} else {
|
|
|
|
qerror_print(qerr);
|
|
|
|
QDECREF(qerr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-19 02:05:30 +01:00
|
|
|
/**
|
|
|
|
* qobject_to_qerror(): Convert a QObject into a QError
|
|
|
|
*/
|
|
|
|
QError *qobject_to_qerror(const QObject *obj)
|
|
|
|
{
|
|
|
|
if (qobject_type(obj) != QTYPE_QERROR) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return container_of(obj, QError, base);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qerror_destroy_obj(): Free all memory allocated by a QError
|
|
|
|
*/
|
|
|
|
static void qerror_destroy_obj(QObject *obj)
|
|
|
|
{
|
|
|
|
QError *qerr;
|
|
|
|
|
|
|
|
assert(obj != NULL);
|
|
|
|
qerr = qobject_to_qerror(obj);
|
|
|
|
|
|
|
|
QDECREF(qerr->error);
|
2011-08-21 05:09:37 +02:00
|
|
|
g_free(qerr);
|
2009-11-19 02:05:30 +01:00
|
|
|
}
|