| 
									
										
										
										
											2009-11-18 23:05:30 -02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * QError: QEMU Error data-type. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include "qjson.h"
 | 
					
						
							|  |  |  | #include "qerror.h"
 | 
					
						
							|  |  |  | #include "qstring.h"
 | 
					
						
							|  |  |  | #include "qemu-common.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-18 16:24:31 +01:00
										 |  |  | #include "qemu-error.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:30 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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)%%" | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-12-04 20:44:44 +00:00
										 |  |  | static const QErrorStringTable qerror_table[] = { | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02: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-26 22:58:57 -02:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:02 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_DEVICE_NOT_ACTIVE, | 
					
						
							| 
									
										
										
										
											2010-02-19 17:42:46 +01:00
										 |  |  |         .desc      = "Device '%(device)' has not been activated by the guest", | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:02 +01:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:32 -02:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_DEVICE_NOT_FOUND, | 
					
						
							| 
									
										
										
										
											2010-02-19 17:42:46 +01:00
										 |  |  |         .desc      = "Device '%(device)' not found", | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:32 -02: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
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											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", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											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-18 23:05:34 -02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:02 +01:00
										 |  |  |         .error_fmt = QERR_INVALID_PARAMETER_TYPE, | 
					
						
							|  |  |  |         .desc      = "Invalid parameter type, expected: %(expected)", | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:34 -02:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02: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-26 22:58:57 -02:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-12-04 15:24:08 -02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:02 +01:00
										 |  |  |         .error_fmt = QERR_JSON_PARSING, | 
					
						
							|  |  |  |         .desc      = "Invalid JSON syntax", | 
					
						
							| 
									
										
										
										
											2009-12-04 15:24:08 -02:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:35 -02:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_KVM_MISSING_CAP, | 
					
						
							|  |  |  |         .desc      = "Using KVM without %(capability), %(feature) unavailable", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_MISSING_PARAMETER, | 
					
						
							| 
									
										
										
										
											2010-02-19 17:42:46 +01:00
										 |  |  |         .desc      = "Parameter '%(name)' is missing", | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_OPEN_FILE_FAILED, | 
					
						
							|  |  |  |         .desc      = "Could not open '%(filename)'", | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											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)'", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_QMP_BAD_INPUT_OBJECT, | 
					
						
							|  |  |  |         .desc      = "Bad QMP input object", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:07 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_SET_PASSWD_FAILED, | 
					
						
							|  |  |  |         .desc      = "Could not set password", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:14 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_TOO_MANY_FILES, | 
					
						
							|  |  |  |         .desc      = "Too many open files", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:02 +01:00
										 |  |  |         .error_fmt = QERR_UNDEFINED_ERROR, | 
					
						
							|  |  |  |         .desc      = "An undefined error has ocurred", | 
					
						
							| 
									
										
										
										
											2009-11-26 22:58:57 -02:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:08 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         .error_fmt = QERR_VNC_SERVER_FAILED, | 
					
						
							|  |  |  |         .desc      = "Could not start VNC server on %(target)", | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:30 -02:00
										 |  |  |     {} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * qerror_new(): Create a new QError | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Return strong reference. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | QError *qerror_new(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QError *qerr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qerr = qemu_mallocz(sizeof(*qerr)); | 
					
						
							|  |  |  |     QOBJECT_INIT(qerr, &qerror_type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return qerr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qerror_abort(const QError *qerr, const char *fmt, ...) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     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(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void qerror_set_data(QError *qerr, const char *fmt, va_list *va) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     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-18 23:05:30 -02: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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void parse_error(const QError *qerror, int c) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     qerror_abort(qerror, "expected '%c' in '%s'", c, qerror->entry->desc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *append_field(QString *outstr, const QError *qerror, | 
					
						
							|  |  |  |                                 const char *start) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QObject *obj; | 
					
						
							|  |  |  |     QDict *qdict; | 
					
						
							|  |  |  |     QString *key_qs; | 
					
						
							|  |  |  |     const char *end, *key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (*start != '%') | 
					
						
							|  |  |  |         parse_error(qerror, '%'); | 
					
						
							|  |  |  |     start++; | 
					
						
							|  |  |  |     if (*start != '(') | 
					
						
							|  |  |  |         parse_error(qerror, '('); | 
					
						
							|  |  |  |     start++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     end = strchr(start, ')'); | 
					
						
							|  |  |  |     if (!end) | 
					
						
							|  |  |  |         parse_error(qerror, ')'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key_qs = qstring_from_substr(start, 0, end - start - 1); | 
					
						
							|  |  |  |     key = qstring_get_str(key_qs); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qdict = qobject_to_qdict(qdict_get(qerror->error, "data")); | 
					
						
							|  |  |  |     obj = qdict_get(qdict, key); | 
					
						
							|  |  |  |     if (!obj) { | 
					
						
							|  |  |  |         qerror_abort(qerror, "key '%s' not found in QDict", key); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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: | 
					
						
							|  |  |  |             qerror_abort(qerror, "invalid type '%c'", qobject_type(obj)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QDECREF(key_qs); | 
					
						
							|  |  |  |     return ++end; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:16 +01:00
										 |  |  |  * qerror_human(): Format QError data into human-readable string. | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:30 -02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:16 +01:00
										 |  |  |  * Formats according to member 'desc' of the specified QError object. | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:30 -02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:16 +01:00
										 |  |  | QString *qerror_human(const QError *qerror) | 
					
						
							| 
									
										
										
										
											2009-11-18 23:05:30 -02:00
										 |  |  | { | 
					
						
							|  |  |  |     const char *p; | 
					
						
							|  |  |  |     QString *qstring; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert(qerror->entry != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qstring = qstring_new(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (p = qerror->entry->desc; *p != '\0';) { | 
					
						
							|  |  |  |         if (*p != '%') { | 
					
						
							|  |  |  |             qstring_append_chr(qstring, *p++); | 
					
						
							|  |  |  |         } else if (*(p + 1) == '%') { | 
					
						
							|  |  |  |             qstring_append_chr(qstring, '%'); | 
					
						
							|  |  |  |             p += 2; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             p = append_field(qstring, qerror, p); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 21:37:16 +01:00
										 |  |  |     return qstring; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * 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-18 23:05:30 -02:00
										 |  |  |     QDECREF(qstring); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * 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); | 
					
						
							|  |  |  |     qemu_free(qerr); | 
					
						
							|  |  |  | } |