mirror of https://github.com/xqemu/xqemu.git
qom: Add object_new_with_props() / object_new_withpropv() helpers
It is reasonably common to want to create an object, set a number of properties, register it in the hierarchy and then mark it as complete (if a user creatable type). This requires quite a lot of error prone, verbose, boilerplate code to achieve. First a pair of functions object_set_props() / object_set_propv() are added which allow for a list of objects to be set in one single API call. Then object_new_with_props() / object_new_with_propv() constructors are added which simplify the sequence of calls to create an object, populate properties, register in the object composition tree and mark the object complete, into a single method call. Usage would be: Error *err = NULL; Object *obj; obj = object_new_with_propv(TYPE_MEMORY_BACKEND_FILE, object_get_objects_root(), "hostmem0", &err, "share", "yes", "mem-path", "/dev/shm/somefile", "prealloc", "yes", "size", "1048576", NULL); Note all property values are passed in string form and will be parsed into their required data types, using normal QOM semantics for parsing from string format. Signed-off-by: Daniel P. Berrange <berrange@redhat.com> Reviewed-by: Eric Blake <eblake@redhat.com> Signed-off-by: Andreas Färber <afaerber@suse.de>
This commit is contained in:
parent
bc2256c4ae
commit
a31bdae5a7
|
@ -24,6 +24,12 @@
|
||||||
#define QEMU_WARN_UNUSED_RESULT
|
#define QEMU_WARN_UNUSED_RESULT
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if QEMU_GNUC_PREREQ(4, 0)
|
||||||
|
#define QEMU_SENTINEL __attribute__((sentinel))
|
||||||
|
#else
|
||||||
|
#define QEMU_SENTINEL
|
||||||
|
#endif
|
||||||
|
|
||||||
#if QEMU_GNUC_PREREQ(4, 3)
|
#if QEMU_GNUC_PREREQ(4, 3)
|
||||||
#define QEMU_ARTIFICIAL __attribute__((always_inline, artificial))
|
#define QEMU_ARTIFICIAL __attribute__((always_inline, artificial))
|
||||||
#else
|
#else
|
||||||
|
|
|
@ -606,6 +606,134 @@ Object *object_new(const char *typename);
|
||||||
*/
|
*/
|
||||||
Object *object_new_with_type(Type type);
|
Object *object_new_with_type(Type type);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* object_new_with_props:
|
||||||
|
* @typename: The name of the type of the object to instantiate.
|
||||||
|
* @parent: the parent object
|
||||||
|
* @id: The unique ID of the object
|
||||||
|
* @errp: pointer to error object
|
||||||
|
* @...: list of property names and values
|
||||||
|
*
|
||||||
|
* This function will initialize a new object using heap allocated memory.
|
||||||
|
* The returned object has a reference count of 1, and will be freed when
|
||||||
|
* the last reference is dropped.
|
||||||
|
*
|
||||||
|
* The @id parameter will be used when registering the object as a
|
||||||
|
* child of @parent in the composition tree.
|
||||||
|
*
|
||||||
|
* The variadic parameters are a list of pairs of (propname, propvalue)
|
||||||
|
* strings. The propname of %NULL indicates the end of the property
|
||||||
|
* list. If the object implements the user creatable interface, the
|
||||||
|
* object will be marked complete once all the properties have been
|
||||||
|
* processed.
|
||||||
|
*
|
||||||
|
* <example>
|
||||||
|
* <title>Creating an object with properties</title>
|
||||||
|
* <programlisting>
|
||||||
|
* Error *err = NULL;
|
||||||
|
* Object *obj;
|
||||||
|
*
|
||||||
|
* obj = object_new_with_props(TYPE_MEMORY_BACKEND_FILE,
|
||||||
|
* object_get_objects_root(),
|
||||||
|
* "hostmem0",
|
||||||
|
* &err,
|
||||||
|
* "share", "yes",
|
||||||
|
* "mem-path", "/dev/shm/somefile",
|
||||||
|
* "prealloc", "yes",
|
||||||
|
* "size", "1048576",
|
||||||
|
* NULL);
|
||||||
|
*
|
||||||
|
* if (!obj) {
|
||||||
|
* g_printerr("Cannot create memory backend: %s\n",
|
||||||
|
* error_get_pretty(err));
|
||||||
|
* }
|
||||||
|
* </programlisting>
|
||||||
|
* </example>
|
||||||
|
*
|
||||||
|
* The returned object will have one stable reference maintained
|
||||||
|
* for as long as it is present in the object hierarchy.
|
||||||
|
*
|
||||||
|
* Returns: The newly allocated, instantiated & initialized object.
|
||||||
|
*/
|
||||||
|
Object *object_new_with_props(const char *typename,
|
||||||
|
Object *parent,
|
||||||
|
const char *id,
|
||||||
|
Error **errp,
|
||||||
|
...) QEMU_SENTINEL;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* object_new_with_propv:
|
||||||
|
* @typename: The name of the type of the object to instantiate.
|
||||||
|
* @parent: the parent object
|
||||||
|
* @id: The unique ID of the object
|
||||||
|
* @errp: pointer to error object
|
||||||
|
* @vargs: list of property names and values
|
||||||
|
*
|
||||||
|
* See object_new_with_props() for documentation.
|
||||||
|
*/
|
||||||
|
Object *object_new_with_propv(const char *typename,
|
||||||
|
Object *parent,
|
||||||
|
const char *id,
|
||||||
|
Error **errp,
|
||||||
|
va_list vargs);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* object_set_props:
|
||||||
|
* @obj: the object instance to set properties on
|
||||||
|
* @errp: pointer to error object
|
||||||
|
* @...: list of property names and values
|
||||||
|
*
|
||||||
|
* This function will set a list of properties on an existing object
|
||||||
|
* instance.
|
||||||
|
*
|
||||||
|
* The variadic parameters are a list of pairs of (propname, propvalue)
|
||||||
|
* strings. The propname of %NULL indicates the end of the property
|
||||||
|
* list.
|
||||||
|
*
|
||||||
|
* <example>
|
||||||
|
* <title>Update an object's properties</title>
|
||||||
|
* <programlisting>
|
||||||
|
* Error *err = NULL;
|
||||||
|
* Object *obj = ...get / create object...;
|
||||||
|
*
|
||||||
|
* obj = object_set_props(obj,
|
||||||
|
* &err,
|
||||||
|
* "share", "yes",
|
||||||
|
* "mem-path", "/dev/shm/somefile",
|
||||||
|
* "prealloc", "yes",
|
||||||
|
* "size", "1048576",
|
||||||
|
* NULL);
|
||||||
|
*
|
||||||
|
* if (!obj) {
|
||||||
|
* g_printerr("Cannot set properties: %s\n",
|
||||||
|
* error_get_pretty(err));
|
||||||
|
* }
|
||||||
|
* </programlisting>
|
||||||
|
* </example>
|
||||||
|
*
|
||||||
|
* The returned object will have one stable reference maintained
|
||||||
|
* for as long as it is present in the object hierarchy.
|
||||||
|
*
|
||||||
|
* Returns: -1 on error, 0 on success
|
||||||
|
*/
|
||||||
|
int object_set_props(Object *obj,
|
||||||
|
Error **errp,
|
||||||
|
...) QEMU_SENTINEL;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* object_set_propv:
|
||||||
|
* @obj: the object instance to set properties on
|
||||||
|
* @errp: pointer to error object
|
||||||
|
* @vargs: list of property names and values
|
||||||
|
*
|
||||||
|
* See object_set_props() for documentation.
|
||||||
|
*
|
||||||
|
* Returns: -1 on error, 0 on success
|
||||||
|
*/
|
||||||
|
int object_set_propv(Object *obj,
|
||||||
|
Error **errp,
|
||||||
|
va_list vargs);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* object_initialize_with_type:
|
* object_initialize_with_type:
|
||||||
* @data: A pointer to the memory to be used for the object.
|
* @data: A pointer to the memory to be used for the object.
|
||||||
|
|
109
qom/object.c
109
qom/object.c
|
@ -11,6 +11,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "qom/object.h"
|
#include "qom/object.h"
|
||||||
|
#include "qom/object_interfaces.h"
|
||||||
#include "qemu-common.h"
|
#include "qemu-common.h"
|
||||||
#include "qapi/visitor.h"
|
#include "qapi/visitor.h"
|
||||||
#include "qapi-visit.h"
|
#include "qapi-visit.h"
|
||||||
|
@ -439,6 +440,114 @@ Object *object_new(const char *typename)
|
||||||
return object_new_with_type(ti);
|
return object_new_with_type(ti);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Object *object_new_with_props(const char *typename,
|
||||||
|
Object *parent,
|
||||||
|
const char *id,
|
||||||
|
Error **errp,
|
||||||
|
...)
|
||||||
|
{
|
||||||
|
va_list vargs;
|
||||||
|
Object *obj;
|
||||||
|
|
||||||
|
va_start(vargs, errp);
|
||||||
|
obj = object_new_with_propv(typename, parent, id, errp, vargs);
|
||||||
|
va_end(vargs);
|
||||||
|
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Object *object_new_with_propv(const char *typename,
|
||||||
|
Object *parent,
|
||||||
|
const char *id,
|
||||||
|
Error **errp,
|
||||||
|
va_list vargs)
|
||||||
|
{
|
||||||
|
Object *obj;
|
||||||
|
ObjectClass *klass;
|
||||||
|
Error *local_err = NULL;
|
||||||
|
|
||||||
|
klass = object_class_by_name(typename);
|
||||||
|
if (!klass) {
|
||||||
|
error_setg(errp, "invalid object type: %s", typename);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (object_class_is_abstract(klass)) {
|
||||||
|
error_setg(errp, "object type '%s' is abstract", typename);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
obj = object_new(typename);
|
||||||
|
|
||||||
|
if (object_set_propv(obj, &local_err, vargs) < 0) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
object_property_add_child(parent, id, obj, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (object_dynamic_cast(obj, TYPE_USER_CREATABLE)) {
|
||||||
|
user_creatable_complete(obj, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
object_unparent(obj);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
object_unref(OBJECT(obj));
|
||||||
|
return obj;
|
||||||
|
|
||||||
|
error:
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
}
|
||||||
|
object_unref(obj);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int object_set_props(Object *obj,
|
||||||
|
Error **errp,
|
||||||
|
...)
|
||||||
|
{
|
||||||
|
va_list vargs;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
va_start(vargs, errp);
|
||||||
|
ret = object_set_propv(obj, errp, vargs);
|
||||||
|
va_end(vargs);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int object_set_propv(Object *obj,
|
||||||
|
Error **errp,
|
||||||
|
va_list vargs)
|
||||||
|
{
|
||||||
|
const char *propname;
|
||||||
|
Error *local_err = NULL;
|
||||||
|
|
||||||
|
propname = va_arg(vargs, char *);
|
||||||
|
while (propname != NULL) {
|
||||||
|
const char *value = va_arg(vargs, char *);
|
||||||
|
|
||||||
|
g_assert(value != NULL);
|
||||||
|
object_property_parse(obj, value, propname, &local_err);
|
||||||
|
if (local_err) {
|
||||||
|
error_propagate(errp, local_err);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
propname = va_arg(vargs, char *);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Object *object_dynamic_cast(Object *obj, const char *typename)
|
Object *object_dynamic_cast(Object *obj, const char *typename)
|
||||||
{
|
{
|
||||||
if (obj && object_class_dynamic_cast(object_get_class(obj), typename)) {
|
if (obj && object_class_dynamic_cast(object_get_class(obj), typename)) {
|
||||||
|
|
|
@ -5,6 +5,7 @@ check-qjson
|
||||||
check-qlist
|
check-qlist
|
||||||
check-qstring
|
check-qstring
|
||||||
check-qom-interface
|
check-qom-interface
|
||||||
|
check-qom-proplist
|
||||||
rcutorture
|
rcutorture
|
||||||
test-aio
|
test-aio
|
||||||
test-bitops
|
test-bitops
|
||||||
|
|
|
@ -68,6 +68,8 @@ check-unit-y += tests/test-bitops$(EXESUF)
|
||||||
check-unit-$(CONFIG_HAS_GLIB_SUBPROCESS_TESTS) += tests/test-qdev-global-props$(EXESUF)
|
check-unit-$(CONFIG_HAS_GLIB_SUBPROCESS_TESTS) += tests/test-qdev-global-props$(EXESUF)
|
||||||
check-unit-y += tests/check-qom-interface$(EXESUF)
|
check-unit-y += tests/check-qom-interface$(EXESUF)
|
||||||
gcov-files-check-qom-interface-y = qom/object.c
|
gcov-files-check-qom-interface-y = qom/object.c
|
||||||
|
check-unit-y += tests/check-qom-proplist$(EXESUF)
|
||||||
|
gcov-files-check-qom-proplist-y = qom/object.c
|
||||||
check-unit-y += tests/test-qemu-opts$(EXESUF)
|
check-unit-y += tests/test-qemu-opts$(EXESUF)
|
||||||
gcov-files-test-qemu-opts-y = qom/test-qemu-opts.c
|
gcov-files-test-qemu-opts-y = qom/test-qemu-opts.c
|
||||||
check-unit-y += tests/test-write-threshold$(EXESUF)
|
check-unit-y += tests/test-write-threshold$(EXESUF)
|
||||||
|
@ -267,7 +269,7 @@ test-qapi-obj-y = tests/test-qapi-visit.o tests/test-qapi-types.o \
|
||||||
|
|
||||||
$(test-obj-y): QEMU_INCLUDES += -Itests
|
$(test-obj-y): QEMU_INCLUDES += -Itests
|
||||||
QEMU_CFLAGS += -I$(SRC_PATH)/tests
|
QEMU_CFLAGS += -I$(SRC_PATH)/tests
|
||||||
qom-core-obj = qom/object.o qom/qom-qobject.o qom/container.o
|
qom-core-obj = qom/object.o qom/qom-qobject.o qom/container.o qom/object_interfaces.o
|
||||||
|
|
||||||
tests/check-qint$(EXESUF): tests/check-qint.o libqemuutil.a
|
tests/check-qint$(EXESUF): tests/check-qint.o libqemuutil.a
|
||||||
tests/check-qstring$(EXESUF): tests/check-qstring.o libqemuutil.a
|
tests/check-qstring$(EXESUF): tests/check-qstring.o libqemuutil.a
|
||||||
|
@ -276,6 +278,7 @@ tests/check-qlist$(EXESUF): tests/check-qlist.o libqemuutil.a
|
||||||
tests/check-qfloat$(EXESUF): tests/check-qfloat.o libqemuutil.a
|
tests/check-qfloat$(EXESUF): tests/check-qfloat.o libqemuutil.a
|
||||||
tests/check-qjson$(EXESUF): tests/check-qjson.o libqemuutil.a libqemustub.a
|
tests/check-qjson$(EXESUF): tests/check-qjson.o libqemuutil.a libqemustub.a
|
||||||
tests/check-qom-interface$(EXESUF): tests/check-qom-interface.o $(qom-core-obj) libqemuutil.a libqemustub.a
|
tests/check-qom-interface$(EXESUF): tests/check-qom-interface.o $(qom-core-obj) libqemuutil.a libqemustub.a
|
||||||
|
tests/check-qom-proplist$(EXESUF): tests/check-qom-proplist.o $(qom-core-obj) libqemuutil.a libqemustub.a
|
||||||
tests/test-coroutine$(EXESUF): tests/test-coroutine.o $(block-obj-y) libqemuutil.a libqemustub.a
|
tests/test-coroutine$(EXESUF): tests/test-coroutine.o $(block-obj-y) libqemuutil.a libqemustub.a
|
||||||
tests/test-aio$(EXESUF): tests/test-aio.o $(block-obj-y) libqemuutil.a libqemustub.a
|
tests/test-aio$(EXESUF): tests/test-aio.o $(block-obj-y) libqemuutil.a libqemustub.a
|
||||||
tests/test-rfifolock$(EXESUF): tests/test-rfifolock.o libqemuutil.a libqemustub.a
|
tests/test-rfifolock$(EXESUF): tests/test-rfifolock.o libqemuutil.a libqemustub.a
|
||||||
|
|
|
@ -0,0 +1,186 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2015 Red Hat, Inc.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library. If not, see
|
||||||
|
* <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
#include "qom/object.h"
|
||||||
|
#include "qemu/module.h"
|
||||||
|
|
||||||
|
|
||||||
|
#define TYPE_DUMMY "qemu-dummy"
|
||||||
|
|
||||||
|
typedef struct DummyObject DummyObject;
|
||||||
|
typedef struct DummyObjectClass DummyObjectClass;
|
||||||
|
|
||||||
|
#define DUMMY_OBJECT(obj) \
|
||||||
|
OBJECT_CHECK(DummyObject, (obj), TYPE_DUMMY)
|
||||||
|
|
||||||
|
struct DummyObject {
|
||||||
|
Object parent_obj;
|
||||||
|
|
||||||
|
bool bv;
|
||||||
|
char *sv;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct DummyObjectClass {
|
||||||
|
ObjectClass parent_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static void dummy_set_bv(Object *obj,
|
||||||
|
bool value,
|
||||||
|
Error **errp)
|
||||||
|
{
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(obj);
|
||||||
|
|
||||||
|
dobj->bv = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool dummy_get_bv(Object *obj,
|
||||||
|
Error **errp)
|
||||||
|
{
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(obj);
|
||||||
|
|
||||||
|
return dobj->bv;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void dummy_set_sv(Object *obj,
|
||||||
|
const char *value,
|
||||||
|
Error **errp)
|
||||||
|
{
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(obj);
|
||||||
|
|
||||||
|
g_free(dobj->sv);
|
||||||
|
dobj->sv = g_strdup(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *dummy_get_sv(Object *obj,
|
||||||
|
Error **errp)
|
||||||
|
{
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(obj);
|
||||||
|
|
||||||
|
return g_strdup(dobj->sv);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void dummy_init(Object *obj)
|
||||||
|
{
|
||||||
|
object_property_add_bool(obj, "bv",
|
||||||
|
dummy_get_bv,
|
||||||
|
dummy_set_bv,
|
||||||
|
NULL);
|
||||||
|
object_property_add_str(obj, "sv",
|
||||||
|
dummy_get_sv,
|
||||||
|
dummy_set_sv,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void dummy_finalize(Object *obj)
|
||||||
|
{
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(obj);
|
||||||
|
|
||||||
|
g_free(dobj->sv);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static const TypeInfo dummy_info = {
|
||||||
|
.name = TYPE_DUMMY,
|
||||||
|
.parent = TYPE_OBJECT,
|
||||||
|
.instance_size = sizeof(DummyObject),
|
||||||
|
.instance_init = dummy_init,
|
||||||
|
.instance_finalize = dummy_finalize,
|
||||||
|
.class_size = sizeof(DummyObjectClass),
|
||||||
|
};
|
||||||
|
|
||||||
|
static void test_dummy_createv(void)
|
||||||
|
{
|
||||||
|
Error *err = NULL;
|
||||||
|
Object *parent = object_get_objects_root();
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(
|
||||||
|
object_new_with_props(TYPE_DUMMY,
|
||||||
|
parent,
|
||||||
|
"dummy0",
|
||||||
|
&err,
|
||||||
|
"bv", "yes",
|
||||||
|
"sv", "Hiss hiss hiss",
|
||||||
|
NULL));
|
||||||
|
|
||||||
|
g_assert(err == NULL);
|
||||||
|
g_assert_cmpstr(dobj->sv, ==, "Hiss hiss hiss");
|
||||||
|
g_assert(dobj->bv == true);
|
||||||
|
|
||||||
|
g_assert(object_resolve_path_component(parent, "dummy0")
|
||||||
|
== OBJECT(dobj));
|
||||||
|
|
||||||
|
object_unparent(OBJECT(dobj));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static Object *new_helper(Error **errp,
|
||||||
|
Object *parent,
|
||||||
|
...)
|
||||||
|
{
|
||||||
|
va_list vargs;
|
||||||
|
Object *obj;
|
||||||
|
|
||||||
|
va_start(vargs, parent);
|
||||||
|
obj = object_new_with_propv(TYPE_DUMMY,
|
||||||
|
parent,
|
||||||
|
"dummy0",
|
||||||
|
errp,
|
||||||
|
vargs);
|
||||||
|
va_end(vargs);
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void test_dummy_createlist(void)
|
||||||
|
{
|
||||||
|
Error *err = NULL;
|
||||||
|
Object *parent = object_get_objects_root();
|
||||||
|
DummyObject *dobj = DUMMY_OBJECT(
|
||||||
|
new_helper(&err,
|
||||||
|
parent,
|
||||||
|
"bv", "yes",
|
||||||
|
"sv", "Hiss hiss hiss",
|
||||||
|
NULL));
|
||||||
|
|
||||||
|
g_assert(err == NULL);
|
||||||
|
g_assert_cmpstr(dobj->sv, ==, "Hiss hiss hiss");
|
||||||
|
g_assert(dobj->bv == true);
|
||||||
|
|
||||||
|
g_assert(object_resolve_path_component(parent, "dummy0")
|
||||||
|
== OBJECT(dobj));
|
||||||
|
|
||||||
|
object_unparent(OBJECT(dobj));
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
g_test_init(&argc, &argv, NULL);
|
||||||
|
|
||||||
|
module_call_init(MODULE_INIT_QOM);
|
||||||
|
type_register_static(&dummy_info);
|
||||||
|
|
||||||
|
g_test_add_func("/qom/proplist/createlist", test_dummy_createlist);
|
||||||
|
g_test_add_func("/qom/proplist/createv", test_dummy_createv);
|
||||||
|
|
||||||
|
return g_test_run();
|
||||||
|
}
|
Loading…
Reference in New Issue