| /* |
| * IDirect3DQuery9 implementation |
| * |
| * Copyright 2002-2003 Raphael Junqueira |
| * Copyright 2002-2003 Jason Edmeades |
| * Copyright 2005 Oliver Stieber |
| * |
| * 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, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include "config.h" |
| #include "d3d9_private.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(d3d9); |
| |
| static inline struct d3d9_query *impl_from_IDirect3DQuery9(IDirect3DQuery9 *iface) |
| { |
| return CONTAINING_RECORD(iface, struct d3d9_query, IDirect3DQuery9_iface); |
| } |
| |
| static HRESULT WINAPI d3d9_query_QueryInterface(IDirect3DQuery9 *iface, REFIID riid, void **out) |
| { |
| TRACE("iface %p, riid %s, out %p.\n", iface, debugstr_guid(riid), out); |
| |
| if (IsEqualGUID(riid, &IID_IDirect3DQuery9) |
| || IsEqualGUID(riid, &IID_IUnknown)) |
| { |
| IDirect3DQuery9_AddRef(iface); |
| *out = iface; |
| return S_OK; |
| } |
| |
| WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(riid)); |
| |
| *out = NULL; |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI d3d9_query_AddRef(IDirect3DQuery9 *iface) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| ULONG refcount = InterlockedIncrement(&query->refcount); |
| |
| TRACE("%p increasing refcount to %u.\n", iface, refcount); |
| |
| return refcount; |
| } |
| |
| static ULONG WINAPI d3d9_query_Release(IDirect3DQuery9 *iface) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| ULONG refcount = InterlockedDecrement(&query->refcount); |
| |
| TRACE("%p decreasing refcount to %u.\n", iface, refcount); |
| |
| if (!refcount) |
| { |
| wined3d_mutex_lock(); |
| wined3d_query_decref(query->wined3d_query); |
| wined3d_mutex_unlock(); |
| |
| IDirect3DDevice9Ex_Release(query->parent_device); |
| HeapFree(GetProcessHeap(), 0, query); |
| } |
| return refcount; |
| } |
| |
| static HRESULT WINAPI d3d9_query_GetDevice(IDirect3DQuery9 *iface, IDirect3DDevice9 **device) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| |
| TRACE("iface %p, device %p.\n", iface, device); |
| |
| *device = (IDirect3DDevice9 *)query->parent_device; |
| IDirect3DDevice9_AddRef(*device); |
| |
| TRACE("Returning device %p.\n", *device); |
| |
| return D3D_OK; |
| } |
| |
| static D3DQUERYTYPE WINAPI d3d9_query_GetType(IDirect3DQuery9 *iface) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| D3DQUERYTYPE type; |
| |
| TRACE("iface %p.\n", iface); |
| |
| wined3d_mutex_lock(); |
| type = wined3d_query_get_type(query->wined3d_query); |
| wined3d_mutex_unlock(); |
| |
| return type; |
| } |
| |
| static DWORD WINAPI d3d9_query_GetDataSize(IDirect3DQuery9 *iface) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| |
| TRACE("iface %p.\n", iface); |
| |
| return query->data_size; |
| } |
| |
| static HRESULT WINAPI d3d9_query_Issue(IDirect3DQuery9 *iface, DWORD flags) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| HRESULT hr; |
| |
| TRACE("iface %p, flags %#x.\n", iface, flags); |
| |
| wined3d_mutex_lock(); |
| hr = wined3d_query_issue(query->wined3d_query, flags); |
| wined3d_mutex_unlock(); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI d3d9_query_GetData(IDirect3DQuery9 *iface, void *data, DWORD size, DWORD flags) |
| { |
| struct d3d9_query *query = impl_from_IDirect3DQuery9(iface); |
| enum wined3d_query_type type; |
| HRESULT hr; |
| |
| TRACE("iface %p, data %p, size %u, flags %#x.\n", |
| iface, data, size, flags); |
| |
| wined3d_mutex_lock(); |
| type = wined3d_query_get_type(query->wined3d_query); |
| if (type == WINED3D_QUERY_TYPE_TIMESTAMP_DISJOINT && data) |
| { |
| struct wined3d_query_data_timestamp_disjoint data_disjoint; |
| |
| if (size > sizeof(data_disjoint.disjoint)) |
| size = sizeof(data_disjoint.disjoint); |
| |
| hr = wined3d_query_get_data(query->wined3d_query, &data_disjoint, sizeof(data_disjoint), flags); |
| if (SUCCEEDED(hr)) |
| memcpy(data, &data_disjoint.disjoint, size); |
| } |
| else |
| { |
| hr = wined3d_query_get_data(query->wined3d_query, data, size, flags); |
| } |
| wined3d_mutex_unlock(); |
| |
| if (hr == D3DERR_INVALIDCALL) |
| { |
| if (data) |
| { |
| memset(data, 0, size); |
| memset(data, 0xdd, min(size, query->data_size)); |
| } |
| return S_OK; |
| } |
| return hr; |
| } |
| |
| |
| static const struct IDirect3DQuery9Vtbl d3d9_query_vtbl = |
| { |
| d3d9_query_QueryInterface, |
| d3d9_query_AddRef, |
| d3d9_query_Release, |
| d3d9_query_GetDevice, |
| d3d9_query_GetType, |
| d3d9_query_GetDataSize, |
| d3d9_query_Issue, |
| d3d9_query_GetData, |
| }; |
| |
| HRESULT query_init(struct d3d9_query *query, struct d3d9_device *device, D3DQUERYTYPE type) |
| { |
| HRESULT hr; |
| |
| if (type > D3DQUERYTYPE_MEMORYPRESSURE) |
| { |
| WARN("Invalid query type %#x.\n", type); |
| return D3DERR_NOTAVAILABLE; |
| } |
| |
| query->IDirect3DQuery9_iface.lpVtbl = &d3d9_query_vtbl; |
| query->refcount = 1; |
| |
| wined3d_mutex_lock(); |
| if (FAILED(hr = wined3d_query_create(device->wined3d_device, type, |
| query, &d3d9_null_wined3d_parent_ops, &query->wined3d_query))) |
| { |
| wined3d_mutex_unlock(); |
| WARN("Failed to create wined3d query, hr %#x.\n", hr); |
| return hr; |
| } |
| |
| if (type == D3DQUERYTYPE_OCCLUSION) |
| query->data_size = sizeof(DWORD); |
| else if (type == D3DQUERYTYPE_TIMESTAMPDISJOINT) |
| query->data_size = sizeof(BOOL); |
| else |
| query->data_size = wined3d_query_get_data_size(query->wined3d_query); |
| wined3d_mutex_unlock(); |
| |
| query->parent_device = &device->IDirect3DDevice9Ex_iface; |
| IDirect3DDevice9Ex_AddRef(query->parent_device); |
| |
| return D3D_OK; |
| } |