2 * Copyright 2011 Joakim Sindholt <opensource@zhasha.com>
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * on the rights to use, copy, modify, merge, publish, distribute, sub
8 * license, and/or sell copies of the Software, and to permit persons to whom
9 * the Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHOR(S) AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,
19 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
20 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
21 * USE OR OTHER DEALINGS IN THE SOFTWARE. */
24 #include "util/u_atomic.h"
25 #include "util/u_hash_table.h"
27 #include "nine_helpers.h"
28 #include "nine_pdata.h"
29 #include "nine_lock.h"
31 #define DBG_CHANNEL DBG_UNKNOWN
34 NineUnknown_ctor( struct NineUnknown
*This
,
35 struct NineUnknownParams
*pParams
)
37 This
->refs
= pParams
->container
? 0 : 1;
39 This
->forward
= !This
->refs
;
40 This
->container
= pParams
->container
;
41 This
->device
= pParams
->device
;
42 if (This
->refs
&& This
->device
)
43 NineUnknown_AddRef(NineUnknown(This
->device
));
45 This
->vtable
= pParams
->vtable
;
46 This
->vtable_internal
= pParams
->vtable
;
47 This
->guids
= pParams
->guids
;
48 This
->dtor
= pParams
->dtor
;
50 This
->pdata
= util_hash_table_create(ht_guid_hash
, ht_guid_compare
);
58 NineUnknown_dtor( struct NineUnknown
*This
)
60 if (This
->refs
&& This
->device
) /* Possible only if early exit after a ctor failed */
61 (void) NineUnknown_Release(NineUnknown(This
->device
));
64 util_hash_table_foreach(This
->pdata
, ht_guid_delete
, NULL
);
65 util_hash_table_destroy(This
->pdata
);
72 NineUnknown_QueryInterface( struct NineUnknown
*This
,
79 DBG("This=%p riid=%p id=%s ppvObject=%p\n",
80 This
, riid
, riid
? GUID_sprintf(guid_str
, riid
) : "", ppvObject
);
84 if (!ppvObject
) return E_POINTER
;
87 if (GUID_equal(This
->guids
[i
], riid
)) {
89 /* Tests showed that this call succeeds even on objects with
90 * zero refcount. This can happen if the app released all references
91 * but the resource is still bound.
93 NineUnknown_AddRef(This
);
96 } while (This
->guids
[++i
]);
103 NineUnknown_AddRef( struct NineUnknown
*This
)
107 return NineUnknown_AddRef(This
->container
);
109 r
= p_atomic_inc_return(&This
->refs
);
113 NineUnknown_AddRef(NineUnknown(This
->device
));
119 NineUnknown_Release( struct NineUnknown
*This
)
122 return NineUnknown_Release(This
->container
);
124 ULONG r
= p_atomic_dec_return(&This
->refs
);
128 if (NineUnknown_Release(NineUnknown(This
->device
)) == 0)
129 return r
; /* everything's gone */
131 /* Containers (here with !forward) take care of item destruction */
132 if (!This
->container
&& This
->bind
== 0) {
139 /* No need to lock the mutex protecting nine (when D3DCREATE_MULTITHREADED)
140 * for AddRef and Release, except for dtor as some of the dtors require it. */
142 NineUnknown_ReleaseWithDtorLock( struct NineUnknown
*This
)
145 return NineUnknown_ReleaseWithDtorLock(This
->container
);
147 ULONG r
= p_atomic_dec_return(&This
->refs
);
151 if (NineUnknown_ReleaseWithDtorLock(NineUnknown(This
->device
)) == 0)
152 return r
; /* everything's gone */
154 /* Containers (here with !forward) take care of item destruction */
155 if (!This
->container
&& This
->bind
== 0) {
156 NineLockGlobalMutex();
158 NineUnlockGlobalMutex();
165 NineUnknown_GetDevice( struct NineUnknown
*This
,
166 IDirect3DDevice9
**ppDevice
)
168 user_assert(ppDevice
, E_POINTER
);
169 NineUnknown_AddRef(NineUnknown(This
->device
));
170 *ppDevice
= (IDirect3DDevice9
*)This
->device
;
175 NineUnknown_SetPrivateData( struct NineUnknown
*This
,
182 struct pheader
*header
;
183 const void *user_data
= pData
;
187 DBG("This=%p GUID=%s pData=%p SizeOfData=%u Flags=%x\n",
188 This
, GUID_sprintf(guid_str
, refguid
), pData
, SizeOfData
, Flags
);
192 if (Flags
& D3DSPD_IUNKNOWN
)
193 user_assert(SizeOfData
== sizeof(IUnknown
*), D3DERR_INVALIDCALL
);
195 /* data consists of a header and the actual data. avoiding 2 mallocs */
196 header
= CALLOC_VARIANT_LENGTH_STRUCT(pheader
, SizeOfData
);
197 if (!header
) { return E_OUTOFMEMORY
; }
198 header
->unknown
= (Flags
& D3DSPD_IUNKNOWN
) ? TRUE
: FALSE
;
200 /* if the refguid already exists, delete it */
201 NineUnknown_FreePrivateData(This
, refguid
);
203 /* IUnknown special case */
204 if (header
->unknown
) {
205 /* here the pointer doesn't point to the data we want, so point at the
206 * pointer making what we eventually copy is the pointer itself */
210 header
->size
= SizeOfData
;
211 header_data
= (void *)header
+ sizeof(*header
);
212 memcpy(header_data
, user_data
, header
->size
);
213 memcpy(&header
->guid
, refguid
, sizeof(header
->guid
));
215 err
= util_hash_table_set(This
->pdata
, &header
->guid
, header
);
216 if (err
== PIPE_OK
) {
217 if (header
->unknown
) { IUnknown_AddRef(*(IUnknown
**)header_data
); }
222 if (err
== PIPE_ERROR_OUT_OF_MEMORY
) { return E_OUTOFMEMORY
; }
224 return D3DERR_DRIVERINTERNALERROR
;
228 NineUnknown_GetPrivateData( struct NineUnknown
*This
,
233 struct pheader
*header
;
238 DBG("This=%p GUID=%s pData=%p pSizeOfData=%p\n",
239 This
, GUID_sprintf(guid_str
, refguid
), pData
, pSizeOfData
);
243 header
= util_hash_table_get(This
->pdata
, refguid
);
244 if (!header
) { return D3DERR_NOTFOUND
; }
246 user_assert(pSizeOfData
, E_POINTER
);
247 sizeofdata
= *pSizeOfData
;
248 *pSizeOfData
= header
->size
;
253 if (sizeofdata
< header
->size
) {
254 return D3DERR_MOREDATA
;
257 header_data
= (void *)header
+ sizeof(*header
);
258 if (header
->unknown
) { IUnknown_AddRef(*(IUnknown
**)header_data
); }
259 memcpy(pData
, header_data
, header
->size
);
265 NineUnknown_FreePrivateData( struct NineUnknown
*This
,
268 struct pheader
*header
;
271 DBG("This=%p GUID=%s\n", This
, GUID_sprintf(guid_str
, refguid
));
275 header
= util_hash_table_get(This
->pdata
, refguid
);
277 return D3DERR_NOTFOUND
;
279 ht_guid_delete(NULL
, header
, NULL
);
280 util_hash_table_remove(This
->pdata
, refguid
);