1 /**************************************************************************
2 *
3 * Copyright (C) 2014 Red Hat Inc.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the "Software"),
7 * to deal in the Software without restriction, including without limitation
8 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 * and/or sell copies of the Software, and to permit persons to whom the
10 * Software is furnished to do so, subject to the following conditions:
11 *
12 * The above copyright notice and this permission notice shall be included
13 * in all copies or substantial portions of the Software.
14 *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
16 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
19 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
20 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
21 * OTHER DEALINGS IN THE SOFTWARE.
22 *
23 **************************************************************************/
24
25 #include "util/u_pointer.h"
26 #include "util/u_memory.h"
27 #include "util/u_hash_table.h"
28
29 #include "vrend_object.h"
30
31 struct vrend_object_types {
32 void (*unref)(void *);
33 } obj_types[VIRGL_MAX_OBJECTS];
34
35 static void (*resource_unref)(void *);
36
vrend_object_set_destroy_callback(int type,void (* cb)(void *))37 void vrend_object_set_destroy_callback(int type, void (*cb)(void *))
38 {
39 obj_types[type].unref = cb;
40 }
41
vrend_resource_set_destroy_callback(void (* cb)(void *))42 void vrend_resource_set_destroy_callback(void (*cb)(void *))
43 {
44 resource_unref = cb;
45 }
46
47 static unsigned
hash_func(void * key)48 hash_func(void *key)
49 {
50 intptr_t ip = pointer_to_intptr(key);
51 return (unsigned)(ip & 0xffffffff);
52 }
53
54 static int
compare(void * key1,void * key2)55 compare(void *key1, void *key2)
56 {
57 if (key1 < key2)
58 return -1;
59 if (key1 > key2)
60 return 1;
61 else
62 return 0;
63 }
64
65 static struct util_hash_table *res_hash;
66
67 struct vrend_object {
68 enum virgl_object_type type;
69 uint32_t handle;
70 void *data;
71 bool free_data;
72 };
73
free_object(void * value)74 static void free_object(void *value)
75 {
76 struct vrend_object *obj = value;
77
78 if (obj->free_data) {
79 if (obj_types[obj->type].unref)
80 obj_types[obj->type].unref(obj->data);
81 else {
82 /* for objects with no callback just free them */
83 free(obj->data);
84 }
85 }
86 free(obj);
87 }
88
vrend_object_init_ctx_table(void)89 struct util_hash_table *vrend_object_init_ctx_table(void)
90 {
91 struct util_hash_table *ctx_hash;
92 ctx_hash = util_hash_table_create(hash_func, compare, free_object);
93 return ctx_hash;
94 }
95
vrend_object_fini_ctx_table(struct util_hash_table * ctx_hash)96 void vrend_object_fini_ctx_table(struct util_hash_table *ctx_hash)
97 {
98 if (!ctx_hash)
99 return;
100
101 util_hash_table_destroy(ctx_hash);
102 }
103
free_res(void * value)104 static void free_res(void *value)
105 {
106 struct vrend_object *obj = value;
107 (*resource_unref)(obj->data);
108 free(obj);
109 }
110
111 void
vrend_object_init_resource_table(void)112 vrend_object_init_resource_table(void)
113 {
114 if (!res_hash)
115 res_hash = util_hash_table_create(hash_func, compare, free_res);
116 }
117
vrend_object_fini_resource_table(void)118 void vrend_object_fini_resource_table(void)
119 {
120 if (res_hash) {
121 util_hash_table_destroy(res_hash);
122 }
123 res_hash = NULL;
124 }
125
126 uint32_t
vrend_object_insert_nofree(struct util_hash_table * handle_hash,void * data,UNUSED uint32_t length,uint32_t handle,enum virgl_object_type type,bool free_data)127 vrend_object_insert_nofree(struct util_hash_table *handle_hash,
128 void *data, UNUSED uint32_t length, uint32_t handle,
129 enum virgl_object_type type, bool free_data)
130 {
131 struct vrend_object *obj = CALLOC_STRUCT(vrend_object);
132
133 if (!obj)
134 return 0;
135 obj->handle = handle;
136 obj->data = data;
137 obj->type = type;
138 obj->free_data = free_data;
139 util_hash_table_set(handle_hash, intptr_to_pointer(obj->handle), obj);
140 return obj->handle;
141 }
142
143 uint32_t
vrend_object_insert(struct util_hash_table * handle_hash,void * data,uint32_t length,uint32_t handle,enum virgl_object_type type)144 vrend_object_insert(struct util_hash_table *handle_hash,
145 void *data, uint32_t length, uint32_t handle, enum virgl_object_type type)
146 {
147 return vrend_object_insert_nofree(handle_hash, data, length,
148 handle, type, true);
149 }
150
151 void
vrend_object_remove(struct util_hash_table * handle_hash,uint32_t handle,UNUSED enum virgl_object_type type)152 vrend_object_remove(struct util_hash_table *handle_hash,
153 uint32_t handle, UNUSED enum virgl_object_type type)
154 {
155 util_hash_table_remove(handle_hash, intptr_to_pointer(handle));
156 }
157
vrend_object_lookup(struct util_hash_table * handle_hash,uint32_t handle,enum virgl_object_type type)158 void *vrend_object_lookup(struct util_hash_table *handle_hash,
159 uint32_t handle, enum virgl_object_type type)
160 {
161 struct vrend_object *obj;
162
163 obj = util_hash_table_get(handle_hash, intptr_to_pointer(handle));
164 if (!obj) {
165 return NULL;
166 }
167
168 if (obj->type != type)
169 return NULL;
170 return obj->data;
171 }
172
vrend_resource_insert(void * data,uint32_t handle)173 int vrend_resource_insert(void *data, uint32_t handle)
174 {
175 struct vrend_object *obj;
176
177 if (!handle)
178 return 0;
179
180 obj = CALLOC_STRUCT(vrend_object);
181 if (!obj)
182 return 0;
183
184 obj->handle = handle;
185 obj->data = data;
186 util_hash_table_set(res_hash, intptr_to_pointer(obj->handle), obj);
187 return obj->handle;
188 }
189
vrend_resource_remove(uint32_t handle)190 void vrend_resource_remove(uint32_t handle)
191 {
192 util_hash_table_remove(res_hash, intptr_to_pointer(handle));
193 }
194
vrend_resource_lookup(uint32_t handle,UNUSED uint32_t ctx_id)195 void *vrend_resource_lookup(uint32_t handle, UNUSED uint32_t ctx_id)
196 {
197 struct vrend_object *obj;
198 obj = util_hash_table_get(res_hash, intptr_to_pointer(handle));
199 if (!obj)
200 return NULL;
201 return obj->data;
202 }
203