|  | /* | 
|  | * Virtio Shared dma-buf | 
|  | * | 
|  | * Copyright Red Hat, Inc. 2023 | 
|  | * | 
|  | * Authors: | 
|  | *     Albert Esteve <aesteve@redhat.com> | 
|  | * | 
|  | * This work is licensed under the terms of the GNU GPL, version 2 or later. | 
|  | * See the COPYING file in the top-level directory. | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  |  | 
|  | #include "hw/virtio/virtio-dmabuf.h" | 
|  |  | 
|  |  | 
|  | static GMutex lock; | 
|  | static GHashTable *resource_uuids; | 
|  |  | 
|  | /* | 
|  | * uuid_equal_func: wrapper for UUID is_equal function to | 
|  | * satisfy g_hash_table_new expected parameters signatures. | 
|  | */ | 
|  | static int uuid_equal_func(const void *lhv, const void *rhv) | 
|  | { | 
|  | return qemu_uuid_is_equal(lhv, rhv); | 
|  | } | 
|  |  | 
|  | static bool virtio_add_resource(QemuUUID *uuid, VirtioSharedObject *value) | 
|  | { | 
|  | bool result = true; | 
|  |  | 
|  | g_mutex_lock(&lock); | 
|  | if (resource_uuids == NULL) { | 
|  | resource_uuids = g_hash_table_new_full(qemu_uuid_hash, | 
|  | uuid_equal_func, | 
|  | NULL, | 
|  | g_free); | 
|  | } | 
|  | if (g_hash_table_lookup(resource_uuids, uuid) == NULL) { | 
|  | g_hash_table_insert(resource_uuids, uuid, value); | 
|  | } else { | 
|  | result = false; | 
|  | } | 
|  | g_mutex_unlock(&lock); | 
|  |  | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool virtio_add_dmabuf(QemuUUID *uuid, int udmabuf_fd) | 
|  | { | 
|  | bool result; | 
|  | VirtioSharedObject *vso; | 
|  | if (udmabuf_fd < 0) { | 
|  | return false; | 
|  | } | 
|  | vso = g_new(VirtioSharedObject, 1); | 
|  | vso->type = TYPE_DMABUF; | 
|  | vso->value = GINT_TO_POINTER(udmabuf_fd); | 
|  | result = virtio_add_resource(uuid, vso); | 
|  | if (!result) { | 
|  | g_free(vso); | 
|  | } | 
|  |  | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool virtio_add_vhost_device(QemuUUID *uuid, struct vhost_dev *dev) | 
|  | { | 
|  | bool result; | 
|  | VirtioSharedObject *vso; | 
|  | if (dev == NULL) { | 
|  | return false; | 
|  | } | 
|  | vso = g_new(VirtioSharedObject, 1); | 
|  | vso->type = TYPE_VHOST_DEV; | 
|  | vso->value = dev; | 
|  | result = virtio_add_resource(uuid, vso); | 
|  | if (!result) { | 
|  | g_free(vso); | 
|  | } | 
|  |  | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool virtio_remove_resource(const QemuUUID *uuid) | 
|  | { | 
|  | bool result; | 
|  | g_mutex_lock(&lock); | 
|  | result = g_hash_table_remove(resource_uuids, uuid); | 
|  | g_mutex_unlock(&lock); | 
|  |  | 
|  | return result; | 
|  | } | 
|  |  | 
|  | static VirtioSharedObject *get_shared_object(const QemuUUID *uuid) | 
|  | { | 
|  | gpointer lookup_res = NULL; | 
|  |  | 
|  | g_mutex_lock(&lock); | 
|  | if (resource_uuids != NULL) { | 
|  | lookup_res = g_hash_table_lookup(resource_uuids, uuid); | 
|  | } | 
|  | g_mutex_unlock(&lock); | 
|  |  | 
|  | return (VirtioSharedObject *) lookup_res; | 
|  | } | 
|  |  | 
|  | int virtio_lookup_dmabuf(const QemuUUID *uuid) | 
|  | { | 
|  | VirtioSharedObject *vso = get_shared_object(uuid); | 
|  | if (vso == NULL) { | 
|  | return -1; | 
|  | } | 
|  | assert(vso->type == TYPE_DMABUF); | 
|  | return GPOINTER_TO_INT(vso->value); | 
|  | } | 
|  |  | 
|  | struct vhost_dev *virtio_lookup_vhost_device(const QemuUUID *uuid) | 
|  | { | 
|  | VirtioSharedObject *vso = get_shared_object(uuid); | 
|  | if (vso == NULL) { | 
|  | return NULL; | 
|  | } | 
|  | assert(vso->type == TYPE_VHOST_DEV); | 
|  | return (struct vhost_dev *) vso->value; | 
|  | } | 
|  |  | 
|  | SharedObjectType virtio_object_type(const QemuUUID *uuid) | 
|  | { | 
|  | VirtioSharedObject *vso = get_shared_object(uuid); | 
|  | if (vso == NULL) { | 
|  | return TYPE_INVALID; | 
|  | } | 
|  | return vso->type; | 
|  | } | 
|  |  | 
|  | void virtio_free_resources(void) | 
|  | { | 
|  | g_mutex_lock(&lock); | 
|  | g_hash_table_destroy(resource_uuids); | 
|  | /* Reference count shall be 0 after the implicit unref on destroy */ | 
|  | resource_uuids = NULL; | 
|  | g_mutex_unlock(&lock); | 
|  | } |