/* * virobject.c: libvirt reference counted object * * Copyright (C) 2012-2014 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 * . * */ #include #define VIR_PARENT_REQUIRED /* empty, to allow virObject to have no parent */ #include "virobject.h" #include "virthread.h" #include "virerror.h" #include "virlog.h" #include "virprobe.h" #define VIR_FROM_THIS VIR_FROM_NONE VIR_LOG_INIT("util.object"); static unsigned int magicCounter = 0xCAFE0000; struct _virClass { virClass *parent; GType type; unsigned int magic; char *name; size_t objectSize; virObjectDisposeCallback dispose; }; typedef struct _virObjectPrivate virObjectPrivate; struct _virObjectPrivate { virClass *klass; }; G_DEFINE_TYPE_WITH_PRIVATE(virObject, vir_object, G_TYPE_OBJECT) #define VIR_OBJECT_NOTVALID(obj) (!obj || !VIR_IS_OBJECT(obj)) #define VIR_OBJECT_USAGE_PRINT_WARNING(anyobj, objclass) \ do { \ virObject *obj = anyobj; \ if (!obj) \ VIR_WARN("Object cannot be NULL"); \ if (VIR_OBJECT_NOTVALID(obj)) \ VIR_WARN("Object %p (%s) is not a %s instance", \ anyobj, g_type_name_from_instance((void*)anyobj), #objclass); \ } while (0) static virClass *virObjectClassImpl; static virClass *virObjectLockableClass; static virClass *virObjectRWLockableClass; static void virObjectLockableDispose(void *anyobj); static void virObjectRWLockableDispose(void *anyobj); static int virObjectOnceInit(void) { if (!(virObjectClassImpl = virClassNew(NULL, "virObject", sizeof(virObject), 0, NULL))) return -1; if (!VIR_CLASS_NEW(virObjectLockable, virObjectClassImpl)) return -1; if (!VIR_CLASS_NEW(virObjectRWLockable, virObjectClassImpl)) return -1; return 0; } VIR_ONCE_GLOBAL_INIT(virObject); /** * virClassForObject: * * Returns the class instance for the base virObject type */ virClass * virClassForObject(void) { if (virObjectInitialize() < 0) return NULL; return virObjectClassImpl; } /** * virClassForObjectLockable: * * Returns the class instance for the virObjectLockable type */ virClass * virClassForObjectLockable(void) { if (virObjectInitialize() < 0) return NULL; return virObjectLockableClass; } /** * virClassForObjectRWLockable: * * Returns the class instance for the virObjectRWLockable type */ virClass * virClassForObjectRWLockable(void) { if (virObjectInitialize() < 0) return NULL; return virObjectRWLockableClass; } static void virClassDummyInit(void *klass G_GNUC_UNUSED) { } static void virObjectDummyInit(void *obj G_GNUC_UNUSED) { } /** * virClassNew: * @parent: the parent class * @name: the class name * @objectSize: total size of the object struct * @dispose: callback to run to free object fields * * Register a new object class with @name. The @objectSize * should give the total size of the object struct, which * is expected to have a 'virObject parent;' field as (or * contained in) its first member. When the last reference * on the object is released, the @dispose callback will be * invoked to free memory of the local object fields, as * well as the dispose callbacks of the parent classes. * * Returns a new class instance */ virClass * virClassNew(virClass *parent, const char *name, size_t objectSize, size_t parentSize, virObjectDisposeCallback dispose) { virClass *klass; if (parent == NULL && STRNEQ(name, "virObject")) { virReportInvalidNonNullArg(parent); return NULL; } else if (objectSize <= parentSize || parentSize != (parent ? parent->objectSize : 0)) { virReportInvalidArg(objectSize, _("object size %1$zu of %2$s is not larger than parent class %3$zu"), objectSize, name, parent->objectSize); return NULL; } klass = g_new0(virClass, 1); klass->parent = parent; klass->magic = g_atomic_int_add(&magicCounter, 1); klass->name = g_strdup(name); klass->objectSize = objectSize; if (parent == NULL) { klass->type = vir_object_get_type(); } else { klass->type = g_type_register_static_simple(parent->type, name, sizeof(virObjectClass), (GClassInitFunc)virClassDummyInit, objectSize, (GInstanceInitFunc)virObjectDummyInit, 0); } klass->dispose = dispose; return klass; } /** * virClassIsDerivedFrom: * @klass: the klass to check * @parent: the possible parent class * * Determine if @klass is derived from @parent * * Return true if @klass is derived from @parent, false otherwise */ bool virClassIsDerivedFrom(virClass *klass, virClass *parent) { while (klass) { if (klass->magic == parent->magic) return true; klass = klass->parent; } return false; } /** * virObjectNew: * @klass: the klass of object to create * * Allocates a new object of type @klass. The returned * object will be an instance of "virObject *", which * can be cast to the struct associated with @klass. * * The initial reference count of the object will be 1. * * Returns the new object */ void * virObjectNew(virClass *klass) { virObject *obj = NULL; virObjectPrivate *priv; obj = g_object_new(klass->type, NULL); priv = vir_object_get_instance_private(obj); priv->klass = klass; PROBE(OBJECT_NEW, "obj=%p classname=%s", obj, priv->klass->name); return obj; } void * virObjectLockableNew(virClass *klass) { virObjectLockable *obj; if (!virClassIsDerivedFrom(klass, virClassForObjectLockable())) { virReportInvalidArg(klass, _("Class %1$s must derive from virObjectLockable"), virClassName(klass)); return NULL; } if (!(obj = virObjectNew(klass))) return NULL; if (virMutexInit(&obj->lock) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Unable to initialize mutex")); virObjectUnref(obj); return NULL; } return obj; } void * virObjectRWLockableNew(virClass *klass) { virObjectRWLockable *obj; if (!virClassIsDerivedFrom(klass, virClassForObjectRWLockable())) { virReportInvalidArg(klass, _("Class %1$s must derive from virObjectRWLockable"), virClassName(klass)); return NULL; } if (!(obj = virObjectNew(klass))) return NULL; if (virRWLockInit(&obj->lock) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Unable to initialize RW lock")); virObjectUnref(obj); return NULL; } return obj; } static void vir_object_finalize(GObject *gobj) { virObject *obj = VIR_OBJECT(gobj); virObjectPrivate *priv = vir_object_get_instance_private(obj); virClass *klass = priv->klass; PROBE(OBJECT_DISPOSE, "obj=%p", gobj); while (klass) { if (klass->dispose) klass->dispose(obj); klass = klass->parent; } G_OBJECT_CLASS(vir_object_parent_class)->finalize(gobj); } static void vir_object_init(virObject *obj G_GNUC_UNUSED) { } static void vir_object_class_init(virObjectClass *klass) { GObjectClass *obj = G_OBJECT_CLASS(klass); obj->finalize = vir_object_finalize; } static void virObjectLockableDispose(void *anyobj) { virObjectLockable *obj = anyobj; virMutexDestroy(&obj->lock); } static void virObjectRWLockableDispose(void *anyobj) { virObjectRWLockable *obj = anyobj; virRWLockDestroy(&obj->lock); } /** * virObjectUnref: * @anyobj: any instance of virObject * * * Decrement the reference count on @anyobj and if * it hits zero, runs the "dispose" callbacks associated * with the object class and its parents before freeing * @anyobj. */ void virObjectUnref(void *anyobj) { virObject *obj = anyobj; if (VIR_OBJECT_NOTVALID(obj)) return; g_object_unref(anyobj); PROBE(OBJECT_UNREF, "obj=%p", obj); } /** * virObjectRef: * @anyobj: any instance of virObject * * * Increment the reference count on @anyobj and return * the same pointer * * Returns @anyobj */ void * virObjectRef(void *anyobj) { virObject *obj = anyobj; if (VIR_OBJECT_NOTVALID(obj)) return NULL; g_object_ref(obj); PROBE(OBJECT_REF, "obj=%p", obj); return anyobj; } static virObjectLockable * virObjectGetLockableObj(void *anyobj) { if (virObjectIsClass(anyobj, virObjectLockableClass)) return anyobj; VIR_OBJECT_USAGE_PRINT_WARNING(anyobj, virObjectLockable); return NULL; } static virObjectRWLockable * virObjectGetRWLockableObj(void *anyobj) { if (virObjectIsClass(anyobj, virObjectRWLockableClass)) return anyobj; VIR_OBJECT_USAGE_PRINT_WARNING(anyobj, virObjectRWLockable); return NULL; } /** * virObjectLockGuard: * @anyobj: any instance of virObjectLockable * * Acquire a lock on @anyobj that will be managed by the virLockGuard object * returned to the caller. */ virLockGuard virObjectLockGuard(void *anyobj) { virObjectLockable *obj = virObjectGetLockableObj(anyobj); return virLockGuardLock(&obj->lock); } /** * virObjectLock: * @anyobj: any instance of virObjectLockable * * Acquire a lock on @anyobj. The lock must be released by * virObjectUnlock. * * The caller is expected to have acquired a reference * on the object before locking it (eg virObjectRef). * The object must be unlocked before releasing this * reference. */ void virObjectLock(void *anyobj) { virObjectLockable *obj = virObjectGetLockableObj(anyobj); if (!obj) return; virMutexLock(&obj->lock); } /** * virObjectRWLockRead: * @anyobj: any instance of virObjectRWLockable * * Acquire a read lock on @anyobj. The lock must be * released by virObjectRWUnlock. * * The caller is expected to have acquired a reference * on the object before locking it (eg virObjectRef). * The object must be unlocked before releasing this * reference. * * NB: It's possible to return without the lock if * @anyobj was invalid - this has been considered * a programming error rather than something that * should be checked. */ void virObjectRWLockRead(void *anyobj) { virObjectRWLockable *obj = virObjectGetRWLockableObj(anyobj); if (!obj) return; virRWLockRead(&obj->lock); } /** * virObjectRWLockWrite: * @anyobj: any instance of virObjectRWLockable * * Acquire a write lock on @anyobj. The lock must be * released by virObjectRWUnlock. * * The caller is expected to have acquired a reference * on the object before locking it (eg virObjectRef). * The object must be unlocked before releasing this * reference. * * NB: It's possible to return without the lock if * @anyobj was invalid - this has been considered * a programming error rather than something that * should be checked. */ void virObjectRWLockWrite(void *anyobj) { virObjectRWLockable *obj = virObjectGetRWLockableObj(anyobj); if (!obj) return; virRWLockWrite(&obj->lock); } /** * virObjectUnlock: * @anyobj: any instance of virObjectLockable * * Release a lock on @anyobj. The lock must have been acquired by * virObjectLock. */ void virObjectUnlock(void *anyobj) { virObjectLockable *obj = virObjectGetLockableObj(anyobj); if (!obj) return; virMutexUnlock(&obj->lock); } /** * virObjectRWUnlock: * @anyobj: any instance of virObjectRWLockable * * Release a lock on @anyobj. The lock must have been acquired by * virObjectRWLockRead or virObjectRWLockWrite. */ void virObjectRWUnlock(void *anyobj) { virObjectRWLockable *obj = virObjectGetRWLockableObj(anyobj); if (!obj) return; virRWLockUnlock(&obj->lock); } /** * virObjectIsClass: * @anyobj: any instance of virObject * * @klass: the class to check * * Checks whether @anyobj is an instance of * @klass * * Returns true if @anyobj is an instance of @klass */ bool virObjectIsClass(void *anyobj, virClass *klass) { virObject *obj = anyobj; virObjectPrivate *priv; if (VIR_OBJECT_NOTVALID(obj)) return false; priv = vir_object_get_instance_private(obj); return virClassIsDerivedFrom(priv->klass, klass); } /** * virClassName: * @klass: the object class * * Returns the name of @klass */ const char * virClassName(virClass *klass) { return klass->name; } /** * virObjectListFree: * @list: A pointer to a NULL-terminated list of object pointers to free * * Unrefs all members of @list and frees the list itself. */ void virObjectListFree(void *list) { void **next; if (!list) return; for (next = (void **) list; *next; next++) virObjectUnref(*next); g_free(list); } /** * virObjectListFreeCount: * @list: A pointer to a list of object pointers to freea * @count: Number of elements in the list. * * Unrefs all members of @list and frees the list itself. */ void virObjectListFreeCount(void *list, size_t count) { size_t i; if (!list) return; for (i = 0; i < count; i++) virObjectUnref(((void **)list)[i]); g_free(list); }