[PATCH v4 09/17] nodedev: Use virObjectLookup{Keys|Hash}

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Use the virObjectLookupKeys in _virNodeDeviceObj and use the
virObjectLookupHash in _virNodeDeviceObjList.

Convert the code to use the LookupHash object and APIs rather
than code within this module that uses virHash* calls.

Since the _virNodeDeviceObjList only now contains the @parent
object, the virClassNew must be removed from OnceInit because
instantiation would fail since the object size would be the same
as the parent object size.

Usage of HashLookup{Find|Search} APIs returns a locked/reffed
object so remove the virObjectLock after FindBy*Locked calls.

The only function requiring taking a lock is the Add function
since it needs to be synchronized in such a way to avoid
multiple threads attempting to add the same named node device
at the same time.

Also change the virNodeDeviceObjListFilter to follow convention
of other drivers of virNodeDeviceObjListACLFilter.

The NumOfDevicesCallback and GetNamesCallback can use the same
callback function with the only difference being the filling in
of the @names array from the passed @data structure if it exists.

Signed-off-by: John Ferlan <jferlan@xxxxxxxxxx>
---
 src/conf/virnodedeviceobj.c | 286 +++++++++++++-------------------------------
 1 file changed, 80 insertions(+), 206 deletions(-)

diff --git a/src/conf/virnodedeviceobj.c b/src/conf/virnodedeviceobj.c
index b0dcee1..7f025b6 100644
--- a/src/conf/virnodedeviceobj.c
+++ b/src/conf/virnodedeviceobj.c
@@ -34,41 +34,28 @@
 VIR_LOG_INIT("conf.virnodedeviceobj");
 
 struct _virNodeDeviceObj {
-    virObjectLockable parent;
+    virObjectLookupKeys parent;
 
     virNodeDeviceDefPtr def;		/* device definition */
 };
 
 struct _virNodeDeviceObjList {
-    virObjectLockable parent;
-
-    /* name string -> virNodeDeviceObj mapping
-     * for O(1), lockless lookup-by-name */
-    virHashTable *objs;
-
+    virObjectLookupHash parent;
 };
 
 
 static virClassPtr virNodeDeviceObjClass;
-static virClassPtr virNodeDeviceObjListClass;
 static void virNodeDeviceObjDispose(void *opaque);
-static void virNodeDeviceObjListDispose(void *opaque);
 
 static int
 virNodeDeviceObjOnceInit(void)
 {
-    if (!(virNodeDeviceObjClass = virClassNew(virClassForObjectLockable(),
+    if (!(virNodeDeviceObjClass = virClassNew(virClassForObjectLookupKeys(),
                                               "virNodeDeviceObj",
                                               sizeof(virNodeDeviceObj),
                                               virNodeDeviceObjDispose)))
         return -1;
 
-    if (!(virNodeDeviceObjListClass = virClassNew(virClassForObjectLockable(),
-                                                  "virNodeDeviceObjList",
-                                                  sizeof(virNodeDeviceObjList),
-                                                  virNodeDeviceObjListDispose)))
-        return -1;
-
     return 0;
 }
 
@@ -85,14 +72,14 @@ virNodeDeviceObjDispose(void *opaque)
 
 
 static virNodeDeviceObjPtr
-virNodeDeviceObjNew(void)
+virNodeDeviceObjNew(const char *name)
 {
     virNodeDeviceObjPtr obj;
 
     if (virNodeDeviceObjInitialize() < 0)
         return NULL;
 
-    if (!(obj = virObjectLockableNew(virNodeDeviceObjClass)))
+    if (!(obj = virObjectLookupKeysNew(virNodeDeviceObjClass, name, NULL)))
         return NULL;
 
     virObjectLock(obj);
@@ -229,17 +216,9 @@ virNodeDeviceObjListSearch(virNodeDeviceObjListPtr devs,
                            virHashSearcher callback,
                            const void *data)
 {
-    virNodeDeviceObjPtr obj;
-
-    virObjectLock(devs);
-    obj = virHashSearch(devs->objs, callback, data, NULL);
-    virObjectRef(obj);
-    virObjectUnlock(devs);
-
-    if (obj)
-        virObjectLock(obj);
-
-    return obj;
+    virObjectLookupKeysPtr obj = virObjectLookupHashSearch(devs, callback,
+                                                           (void *)data);
+    return (virNodeDeviceObjPtr) obj;
 }
 
 
@@ -274,7 +253,8 @@ static virNodeDeviceObjPtr
 virNodeDeviceObjListFindByNameLocked(virNodeDeviceObjListPtr devs,
                                      const char *name)
 {
-    return virObjectRef(virHashLookup(devs->objs, name));
+    virObjectLookupKeysPtr obj = virObjectLookupHashFindLocked(devs, name);
+    return (virNodeDeviceObjPtr) obj;
 }
 
 
@@ -282,15 +262,8 @@ virNodeDeviceObjPtr
 virNodeDeviceObjListFindByName(virNodeDeviceObjListPtr devs,
                                const char *name)
 {
-    virNodeDeviceObjPtr obj;
-
-    virObjectLock(devs);
-    obj = virNodeDeviceObjListFindByNameLocked(devs, name);
-    virObjectUnlock(devs);
-    if (obj)
-        virObjectLock(obj);
-
-    return obj;
+    virObjectLookupKeysPtr obj = virObjectLookupHashFind(devs, name);
+    return (virNodeDeviceObjPtr) obj;
 }
 
 
@@ -445,32 +418,10 @@ virNodeDeviceObjListFindSCSIHostByWWNs(virNodeDeviceObjListPtr devs,
 }
 
 
-static void
-virNodeDeviceObjListDispose(void *obj)
-{
-    virNodeDeviceObjListPtr devs = obj;
-
-    virHashFree(devs->objs);
-}
-
-
 virNodeDeviceObjListPtr
 virNodeDeviceObjListNew(void)
 {
-    virNodeDeviceObjListPtr devs;
-
-    if (virNodeDeviceObjInitialize() < 0)
-        return NULL;
-
-    if (!(devs = virObjectLockableNew(virNodeDeviceObjListClass)))
-        return NULL;
-
-    if (!(devs->objs = virHashCreate(50, virObjectFreeHashData))) {
-        virObjectUnref(devs);
-        return NULL;
-    }
-
-    return devs;
+    return virObjectLookupHashNew(virClassForObjectLookupHash(), 50, false);
 }
 
 
@@ -486,29 +437,31 @@ virNodeDeviceObjListAssignDef(virNodeDeviceObjListPtr devs,
                               virNodeDeviceDefPtr def)
 {
     virNodeDeviceObjPtr obj;
+    virNodeDeviceObjPtr ret = NULL;
 
-    virObjectLock(devs);
+    virObjectRWLockWrite(devs);
 
     if ((obj = virNodeDeviceObjListFindByNameLocked(devs, def->name))) {
-        virObjectLock(obj);
         virNodeDeviceDefFree(obj->def);
         obj->def = def;
     } else {
-        if (!(obj = virNodeDeviceObjNew()))
+        if (!(obj = virNodeDeviceObjNew(def->name)))
             goto cleanup;
 
-        if (virHashAddEntry(devs->objs, def->name, obj) < 0) {
-            virNodeDeviceObjEndAPI(&obj);
+        if (virObjectLookupHashAdd(devs, (virObjectLookupKeysPtr)obj) < 0)
             goto cleanup;
-        }
 
         obj->def = def;
         virObjectRef(obj);
     }
 
+    ret = obj;
+    obj = NULL;
+
  cleanup:
-    virObjectUnlock(devs);
-    return obj;
+    virNodeDeviceObjEndAPI(&obj);
+    virObjectRWUnlock(devs);
+    return ret;
 }
 
 
@@ -516,20 +469,7 @@ void
 virNodeDeviceObjListRemove(virNodeDeviceObjListPtr devs,
                            virNodeDeviceObjPtr obj)
 {
-    virNodeDeviceDefPtr def;
-
-    if (!obj)
-        return;
-    def = obj->def;
-
-    virObjectRef(obj);
-    virObjectUnlock(obj);
-    virObjectLock(devs);
-    virObjectLock(obj);
-    virHashRemoveEntry(devs->objs, def->name);
-    virObjectUnlock(obj);
-    virObjectUnref(obj);
-    virObjectUnlock(devs);
+    virObjectLookupHashRemove(devs, (virObjectLookupKeysPtr)obj);
 }
 
 
@@ -730,29 +670,33 @@ virNodeDeviceCapMatch(virNodeDeviceObjPtr obj,
 }
 
 
-struct virNodeDeviceCountData {
-    virConnectPtr conn;
-    virNodeDeviceObjListFilter filter;
-    const char *matchstr;
-    int count;
-};
-
 static int
-virNodeDeviceObjListNumOfDevicesCallback(void *payload,
-                                         const void *name ATTRIBUTE_UNUSED,
-                                         void *opaque)
+virNodeDeviceObjListGetHelper(void *payload,
+                              const void *name ATTRIBUTE_UNUSED,
+                              void *opaque)
 {
     virNodeDeviceObjPtr obj = payload;
     virNodeDeviceDefPtr def;
-    struct virNodeDeviceCountData *data = opaque;
+    virObjectLookupHashForEachDataPtr data = opaque;
     virNodeDeviceObjListFilter filter = data->filter;
+    char **names = (char **)data->elems;
+
+    if (data->error)
+        return 0;
 
     virObjectLock(obj);
     def = obj->def;
+
     if ((!filter || filter(data->conn, def)) &&
-        (!data->matchstr || virNodeDeviceObjHasCap(obj, data->matchstr)))
-        data->count++;
+        (!data->matchStr || virNodeDeviceObjHasCap(obj, data->matchStr))) {
+        if (names && VIR_STRDUP(names[data->nElems], def->name) < 0) {
+            data->error = true;
+            goto cleanup;
+        }
+        data->nElems++;
+     }
 
+ cleanup:
     virObjectUnlock(obj);
     return 0;
 }
@@ -764,55 +708,11 @@ virNodeDeviceObjListNumOfDevices(virNodeDeviceObjListPtr devs,
                                  const char *cap,
                                  virNodeDeviceObjListFilter filter)
 {
-    struct virNodeDeviceCountData data = {
-        .conn = conn, .filter = filter, .matchstr = cap, .count = 0 };
+    virObjectLookupHashForEachData data = {
+        .conn = conn, .filter = filter, .error = false, .matchStr = cap,
+        .nElems = 0, .elems = NULL, .maxElems = -2 };
 
-    virObjectLock(devs);
-    virHashForEach(devs->objs, virNodeDeviceObjListNumOfDevicesCallback, &data);
-    virObjectUnlock(devs);
-
-    return data.count;
-}
-
-
-struct virNodeDeviceGetNamesData {
-    virConnectPtr conn;
-    virNodeDeviceObjListFilter filter;
-    const char *matchstr;
-    int nnames;
-    char **names;
-    int maxnames;
-    bool error;
-};
-
-static int
-virNodeDeviceObjListGetNamesCallback(void *payload,
-                                     const void *name ATTRIBUTE_UNUSED,
-                                     void *opaque)
-{
-    virNodeDeviceObjPtr obj = payload;
-    virNodeDeviceDefPtr def;
-    struct virNodeDeviceGetNamesData *data = opaque;
-    virNodeDeviceObjListFilter filter = data->filter;
-
-    if (data->error)
-        return 0;
-
-    virObjectLock(obj);
-    def = obj->def;
-
-    if ((!filter || filter(data->conn, def)) &&
-        (!data->matchstr || virNodeDeviceObjHasCap(obj, data->matchstr))) {
-        if (VIR_STRDUP(data->names[data->nnames], def->name) < 0) {
-            data->error = true;
-            goto cleanup;
-        }
-        data->nnames++;
-     }
-
- cleanup:
-    virObjectUnlock(obj);
-    return 0;
+    return virObjectLookupHashForEach(devs, virNodeDeviceObjListGetHelper, &data);
 }
 
 
@@ -824,23 +724,11 @@ virNodeDeviceObjListGetNames(virNodeDeviceObjListPtr devs,
                              char **const names,
                              int maxnames)
 {
-    struct virNodeDeviceGetNamesData data = {
-        .conn = conn, .filter = filter, .matchstr = cap, .names = names,
-        .nnames = 0, .maxnames = maxnames, .error = false };
-
-    virObjectLock(devs);
-    virHashForEach(devs->objs, virNodeDeviceObjListGetNamesCallback, &data);
-    virObjectUnlock(devs);
-
-    if (data.error)
-        goto error;
-
-    return data.nnames;
+    virObjectLookupHashForEachData data = {
+        .conn = conn, .filter = filter, .error = false, .matchStr = cap,
+        .nElems = 0, .elems = (void **)names, .maxElems = maxnames };
 
- error:
-    while (--data.nnames)
-        VIR_FREE(data.names[data.nnames]);
-    return -1;
+    return virObjectLookupHashForEach(devs, virNodeDeviceObjListGetHelper, &data);
 }
 
 
@@ -876,15 +764,6 @@ virNodeDeviceMatch(virNodeDeviceObjPtr obj,
 #undef MATCH
 
 
-struct virNodeDeviceObjListExportData {
-    virConnectPtr conn;
-    virNodeDeviceObjListFilter filter;
-    unsigned int flags;
-    virNodeDevicePtr *devices;
-    int ndevices;
-    bool error;
-};
-
 static int
 virNodeDeviceObjListExportCallback(void *payload,
                                    const void *name ATTRIBUTE_UNUSED,
@@ -892,8 +771,10 @@ virNodeDeviceObjListExportCallback(void *payload,
 {
     virNodeDeviceObjPtr obj = payload;
     virNodeDeviceDefPtr def;
-    struct virNodeDeviceObjListExportData *data = opaque;
+    virObjectLookupHashForEachDataPtr data = opaque;
+    virNodeDeviceObjListFilter filter = data->filter;
     virNodeDevicePtr device = NULL;
+    virNodeDevicePtr *devices = (virNodeDevicePtr *)data->elems;
 
     if (data->error)
         return 0;
@@ -901,19 +782,24 @@ virNodeDeviceObjListExportCallback(void *payload,
     virObjectLock(obj);
     def = obj->def;
 
-    if ((!data->filter || data->filter(data->conn, def)) &&
-        virNodeDeviceMatch(obj, data->flags)) {
-        if (data->devices) {
-            if (!(device = virGetNodeDevice(data->conn, def->name)) ||
-                VIR_STRDUP(device->parent, def->parent) < 0) {
-                virObjectUnref(device);
-                data->error = true;
-                goto cleanup;
-            }
-            data->devices[data->ndevices] = device;
-        }
-        data->ndevices++;
+    if (filter && !filter(data->conn, def))
+        goto cleanup;
+
+    if (!virNodeDeviceMatch(obj, data->flags))
+        goto cleanup;
+
+    if (!devices) {
+        data->nElems++;
+        goto cleanup;
+    }
+
+    if (!(device = virGetNodeDevice(data->conn, def->name)) ||
+        VIR_STRDUP(device->parent, def->parent) < 0) {
+        virObjectUnref(device);
+        data->error = true;
+        goto cleanup;
     }
+    devices[data->nElems++] = device;
 
  cleanup:
     virObjectUnlock(obj);
@@ -928,31 +814,19 @@ virNodeDeviceObjListExport(virConnectPtr conn,
                            virNodeDeviceObjListFilter filter,
                            unsigned int flags)
 {
-    struct virNodeDeviceObjListExportData data = {
-        .conn = conn, .filter = filter, .flags = flags,
-        .devices = NULL, .ndevices = 0, .error = false };
-
-    virObjectLock(devs);
-    if (devices &&
-        VIR_ALLOC_N(data.devices, virHashSize(devs->objs) + 1) < 0) {
-        virObjectUnlock(devs);
-        return -1;
-    }
-
-    virHashForEach(devs->objs, virNodeDeviceObjListExportCallback, &data);
-    virObjectUnlock(devs);
+    int ret;
+    virObjectLookupHashForEachData data = {
+        .conn = conn, .filter = filter, .error = false, .nElems = 0,
+        .elems = NULL, .maxElems = 0, .flags = flags };
 
-    if (data.error)
-        goto cleanup;
+    if (devices)
+        data.maxElems = -1;
 
-    if (data.devices) {
-        ignore_value(VIR_REALLOC_N(data.devices, data.ndevices + 1));
-        *devices = data.devices;
-     }
+    ret = virObjectLookupHashForEach(devs, virNodeDeviceObjListExportCallback,
+                                     &data);
 
-    return data.ndevices;
+    if (devices)
+        *devices = (virNodeDevicePtr *)data.elems;
 
- cleanup:
-    virObjectListFree(data.devices);
-    return -1;
+    return ret;
 }
-- 
2.9.4

--
libvir-list mailing list
libvir-list@xxxxxxxxxx
https://www.redhat.com/mailman/listinfo/libvir-list



[Index of Archives]     [Virt Tools]     [Libvirt Users]     [Lib OS Info]     [Fedora Users]     [Fedora Desktop]     [Fedora SELinux]     [Big List of Linux Books]     [Yosemite News]     [KDE Users]     [Fedora Tools]
  Powered by Linux