| /* |
| * Copyright 2005 Oliver Stieber |
| * Copyright 2007-2008 Stefan Dösinger for CodeWeavers |
| * Copyright 2009-2010 Henri Verbeet for CodeWeavers. |
| * |
| * 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 "wine/port.h" |
| #include "wined3d_private.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(d3d); |
| |
| static void wined3d_query_init(struct wined3d_query *query, struct wined3d_device *device, |
| enum wined3d_query_type type, const void *data, DWORD data_size, |
| const struct wined3d_query_ops *query_ops, void *parent) |
| { |
| query->ref = 1; |
| query->parent = parent; |
| query->device = device; |
| query->state = QUERY_CREATED; |
| query->type = type; |
| query->data = data; |
| query->data_size = data_size; |
| query->query_ops = query_ops; |
| } |
| |
| static struct wined3d_event_query *wined3d_event_query_from_query(struct wined3d_query *query) |
| { |
| return CONTAINING_RECORD(query, struct wined3d_event_query, query); |
| } |
| |
| static struct wined3d_occlusion_query *wined3d_occlusion_query_from_query(struct wined3d_query *query) |
| { |
| return CONTAINING_RECORD(query, struct wined3d_occlusion_query, query); |
| } |
| |
| static struct wined3d_timestamp_query *wined3d_timestamp_query_from_query(struct wined3d_query *query) |
| { |
| return CONTAINING_RECORD(query, struct wined3d_timestamp_query, query); |
| } |
| |
| BOOL wined3d_event_query_supported(const struct wined3d_gl_info *gl_info) |
| { |
| return gl_info->supported[ARB_SYNC] || gl_info->supported[NV_FENCE] || gl_info->supported[APPLE_FENCE]; |
| } |
| |
| void wined3d_event_query_destroy(struct wined3d_event_query *query) |
| { |
| if (query->context) context_free_event_query(query); |
| HeapFree(GetProcessHeap(), 0, query); |
| } |
| |
| static enum wined3d_event_query_result wined3d_event_query_test(const struct wined3d_event_query *query, |
| const struct wined3d_device *device, DWORD flags) |
| { |
| struct wined3d_context *context; |
| const struct wined3d_gl_info *gl_info; |
| enum wined3d_event_query_result ret; |
| BOOL fence_result; |
| |
| TRACE("query %p, device %p, flags %#x.\n", query, device, flags); |
| |
| if (!query->context) |
| { |
| TRACE("Query not started\n"); |
| return WINED3D_EVENT_QUERY_NOT_STARTED; |
| } |
| |
| if (!query->context->gl_info->supported[ARB_SYNC] && query->context->tid != GetCurrentThreadId()) |
| { |
| WARN("Event query tested from wrong thread\n"); |
| return WINED3D_EVENT_QUERY_WRONG_THREAD; |
| } |
| |
| context = context_acquire(device, context_get_rt_surface(query->context)); |
| gl_info = context->gl_info; |
| |
| if (gl_info->supported[ARB_SYNC]) |
| { |
| GLenum gl_ret = GL_EXTCALL(glClientWaitSync(query->object.sync, |
| (flags & WINED3DGETDATA_FLUSH) ? GL_SYNC_FLUSH_COMMANDS_BIT : 0, 0)); |
| checkGLcall("glClientWaitSync"); |
| |
| switch (gl_ret) |
| { |
| case GL_ALREADY_SIGNALED: |
| case GL_CONDITION_SATISFIED: |
| ret = WINED3D_EVENT_QUERY_OK; |
| break; |
| |
| case GL_TIMEOUT_EXPIRED: |
| ret = WINED3D_EVENT_QUERY_WAITING; |
| break; |
| |
| case GL_WAIT_FAILED: |
| default: |
| ERR("glClientWaitSync returned %#x.\n", gl_ret); |
| ret = WINED3D_EVENT_QUERY_ERROR; |
| } |
| } |
| else if (gl_info->supported[APPLE_FENCE]) |
| { |
| fence_result = GL_EXTCALL(glTestFenceAPPLE(query->object.id)); |
| checkGLcall("glTestFenceAPPLE"); |
| if (fence_result) ret = WINED3D_EVENT_QUERY_OK; |
| else ret = WINED3D_EVENT_QUERY_WAITING; |
| } |
| else if (gl_info->supported[NV_FENCE]) |
| { |
| fence_result = GL_EXTCALL(glTestFenceNV(query->object.id)); |
| checkGLcall("glTestFenceNV"); |
| if (fence_result) ret = WINED3D_EVENT_QUERY_OK; |
| else ret = WINED3D_EVENT_QUERY_WAITING; |
| } |
| else |
| { |
| ERR("Event query created despite lack of GL support\n"); |
| ret = WINED3D_EVENT_QUERY_ERROR; |
| } |
| |
| context_release(context); |
| return ret; |
| } |
| |
| enum wined3d_event_query_result wined3d_event_query_finish(const struct wined3d_event_query *query, |
| const struct wined3d_device *device) |
| { |
| struct wined3d_context *context; |
| const struct wined3d_gl_info *gl_info; |
| enum wined3d_event_query_result ret; |
| |
| TRACE("query %p, device %p.\n", query, device); |
| |
| if (!query->context) |
| { |
| TRACE("Query not started\n"); |
| return WINED3D_EVENT_QUERY_NOT_STARTED; |
| } |
| gl_info = query->context->gl_info; |
| |
| if (query->context->tid != GetCurrentThreadId() && !gl_info->supported[ARB_SYNC]) |
| { |
| /* A glFinish does not reliably wait for draws in other contexts. The caller has |
| * to find its own way to cope with the thread switch |
| */ |
| WARN("Event query finished from wrong thread\n"); |
| return WINED3D_EVENT_QUERY_WRONG_THREAD; |
| } |
| |
| context = context_acquire(device, context_get_rt_surface(query->context)); |
| |
| if (gl_info->supported[ARB_SYNC]) |
| { |
| /* Apple seems to be into arbitrary limits, and timeouts larger than |
| * 0xfffffffffffffbff immediately return GL_TIMEOUT_EXPIRED. We don't |
| * really care and can live with waiting a few μs less. (OS X 10.7.4). */ |
| GLenum gl_ret = GL_EXTCALL(glClientWaitSync(query->object.sync, GL_SYNC_FLUSH_COMMANDS_BIT, ~(GLuint64)0xffff)); |
| checkGLcall("glClientWaitSync"); |
| |
| switch (gl_ret) |
| { |
| case GL_ALREADY_SIGNALED: |
| case GL_CONDITION_SATISFIED: |
| ret = WINED3D_EVENT_QUERY_OK; |
| break; |
| |
| /* We don't expect a timeout for a ~584 year wait */ |
| default: |
| ERR("glClientWaitSync returned %#x.\n", gl_ret); |
| ret = WINED3D_EVENT_QUERY_ERROR; |
| } |
| } |
| else if (context->gl_info->supported[APPLE_FENCE]) |
| { |
| GL_EXTCALL(glFinishFenceAPPLE(query->object.id)); |
| checkGLcall("glFinishFenceAPPLE"); |
| ret = WINED3D_EVENT_QUERY_OK; |
| } |
| else if (context->gl_info->supported[NV_FENCE]) |
| { |
| GL_EXTCALL(glFinishFenceNV(query->object.id)); |
| checkGLcall("glFinishFenceNV"); |
| ret = WINED3D_EVENT_QUERY_OK; |
| } |
| else |
| { |
| ERR("Event query created without GL support\n"); |
| ret = WINED3D_EVENT_QUERY_ERROR; |
| } |
| |
| context_release(context); |
| return ret; |
| } |
| |
| void wined3d_event_query_issue(struct wined3d_event_query *query, const struct wined3d_device *device) |
| { |
| const struct wined3d_gl_info *gl_info; |
| struct wined3d_context *context; |
| |
| if (query->context) |
| { |
| if (!query->context->gl_info->supported[ARB_SYNC] && query->context->tid != GetCurrentThreadId()) |
| { |
| context_free_event_query(query); |
| context = context_acquire(device, NULL); |
| context_alloc_event_query(context, query); |
| } |
| else |
| { |
| context = context_acquire(device, context_get_rt_surface(query->context)); |
| } |
| } |
| else |
| { |
| context = context_acquire(device, NULL); |
| context_alloc_event_query(context, query); |
| } |
| |
| gl_info = context->gl_info; |
| |
| if (gl_info->supported[ARB_SYNC]) |
| { |
| if (query->object.sync) GL_EXTCALL(glDeleteSync(query->object.sync)); |
| checkGLcall("glDeleteSync"); |
| query->object.sync = GL_EXTCALL(glFenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0)); |
| checkGLcall("glFenceSync"); |
| } |
| else if (gl_info->supported[APPLE_FENCE]) |
| { |
| GL_EXTCALL(glSetFenceAPPLE(query->object.id)); |
| checkGLcall("glSetFenceAPPLE"); |
| } |
| else if (gl_info->supported[NV_FENCE]) |
| { |
| GL_EXTCALL(glSetFenceNV(query->object.id, GL_ALL_COMPLETED_NV)); |
| checkGLcall("glSetFenceNV"); |
| } |
| |
| context_release(context); |
| } |
| |
| ULONG CDECL wined3d_query_incref(struct wined3d_query *query) |
| { |
| ULONG refcount = InterlockedIncrement(&query->ref); |
| |
| TRACE("%p increasing refcount to %u.\n", query, refcount); |
| |
| return refcount; |
| } |
| |
| static void wined3d_query_destroy_object(void *object) |
| { |
| struct wined3d_query *query = object; |
| |
| /* Queries are specific to the GL context that created them. Not |
| * deleting the query will obviously leak it, but that's still better |
| * than potentially deleting a different query with the same id in this |
| * context, and (still) leaking the actual query. */ |
| if (query->type == WINED3D_QUERY_TYPE_EVENT) |
| { |
| wined3d_event_query_destroy(wined3d_event_query_from_query(query)); |
| } |
| else if (query->type == WINED3D_QUERY_TYPE_OCCLUSION) |
| { |
| struct wined3d_occlusion_query *oq = wined3d_occlusion_query_from_query(query); |
| |
| if (oq->context) |
| context_free_occlusion_query(oq); |
| HeapFree(GetProcessHeap(), 0, oq); |
| } |
| else if (query->type == WINED3D_QUERY_TYPE_TIMESTAMP) |
| { |
| struct wined3d_timestamp_query *tq = wined3d_timestamp_query_from_query(query); |
| |
| if (tq->context) |
| context_free_timestamp_query(tq); |
| HeapFree(GetProcessHeap(), 0, tq); |
| } |
| else if (query->type == WINED3D_QUERY_TYPE_TIMESTAMP_DISJOINT |
| || query->type == WINED3D_QUERY_TYPE_TIMESTAMP_FREQ) |
| { |
| HeapFree(GetProcessHeap(), 0, query); |
| } |
| else |
| { |
| ERR("Query %p has invalid type %#x.\n", query, query->type); |
| } |
| } |
| |
| ULONG CDECL wined3d_query_decref(struct wined3d_query *query) |
| { |
| ULONG refcount = InterlockedDecrement(&query->ref); |
| |
| TRACE("%p decreasing refcount to %u.\n", query, refcount); |
| |
| if (!refcount) |
| wined3d_cs_emit_destroy_object(query->device->cs, wined3d_query_destroy_object, query); |
| |
| return refcount; |
| } |
| |
| HRESULT CDECL wined3d_query_get_data(struct wined3d_query *query, |
| void *data, UINT data_size, DWORD flags) |
| { |
| TRACE("query %p, data %p, data_size %u, flags %#x.\n", |
| query, data, data_size, flags); |
| |
| if (flags) |
| WARN("Ignoring flags %#x.\n", flags); |
| |
| if (query->state == QUERY_BUILDING) |
| { |
| WARN("Query is building, returning S_FALSE.\n"); |
| return S_FALSE; |
| } |
| |
| if (query->state == QUERY_CREATED) |
| { |
| WARN("Query wasn't started yet.\n"); |
| return WINED3DERR_INVALIDCALL; |
| } |
| |
| if (!query->query_ops->query_poll(query, flags)) |
| return S_FALSE; |
| |
| if (data) |
| memcpy(data, query->data, min(data_size, query->data_size)); |
| |
| return S_OK; |
| } |
| |
| UINT CDECL wined3d_query_get_data_size(const struct wined3d_query *query) |
| { |
| TRACE("query %p.\n", query); |
| |
| return query->data_size; |
| } |
| |
| HRESULT CDECL wined3d_query_issue(struct wined3d_query *query, DWORD flags) |
| { |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| wined3d_cs_emit_query_issue(query->device->cs, query, flags); |
| |
| if (flags & WINED3DISSUE_BEGIN) |
| query->state = QUERY_BUILDING; |
| else |
| query->state = QUERY_SIGNALLED; |
| |
| return WINED3D_OK; |
| } |
| |
| static BOOL wined3d_occlusion_query_ops_poll(struct wined3d_query *query, DWORD flags) |
| { |
| struct wined3d_occlusion_query *oq = wined3d_occlusion_query_from_query(query); |
| struct wined3d_device *device = query->device; |
| const struct wined3d_gl_info *gl_info; |
| struct wined3d_context *context; |
| GLuint available; |
| |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| if (oq->context->tid != GetCurrentThreadId()) |
| { |
| FIXME("%p Wrong thread, returning 1.\n", query); |
| oq->samples = 1; |
| return TRUE; |
| } |
| |
| context = context_acquire(device, context_get_rt_surface(oq->context)); |
| gl_info = context->gl_info; |
| |
| GL_EXTCALL(glGetQueryObjectuiv(oq->id, GL_QUERY_RESULT_AVAILABLE, &available)); |
| checkGLcall("glGetQueryObjectuiv(GL_QUERY_RESULT_AVAILABLE)"); |
| TRACE("available %#x.\n", available); |
| |
| if (available) |
| { |
| if (gl_info->supported[ARB_TIMER_QUERY]) |
| { |
| GLuint64 result; |
| GL_EXTCALL(glGetQueryObjectui64v(oq->id, GL_QUERY_RESULT, &result)); |
| checkGLcall("glGetQueryObjectui64v(GL_QUERY_RESULT)"); |
| oq->samples = result; |
| } |
| else |
| { |
| GLuint result; |
| GL_EXTCALL(glGetQueryObjectuiv(oq->id, GL_QUERY_RESULT, &result)); |
| checkGLcall("glGetQueryObjectuiv(GL_QUERY_RESULT)"); |
| oq->samples = result; |
| } |
| TRACE("Returning 0x%s samples.\n", wine_dbgstr_longlong(oq->samples)); |
| } |
| |
| context_release(context); |
| |
| return available; |
| } |
| |
| static BOOL wined3d_event_query_ops_poll(struct wined3d_query *query, DWORD flags) |
| { |
| struct wined3d_event_query *event_query = wined3d_event_query_from_query(query); |
| enum wined3d_event_query_result ret; |
| |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| ret = wined3d_event_query_test(event_query, query->device, flags); |
| switch (ret) |
| { |
| case WINED3D_EVENT_QUERY_OK: |
| case WINED3D_EVENT_QUERY_NOT_STARTED: |
| return event_query->signalled = TRUE; |
| |
| case WINED3D_EVENT_QUERY_WAITING: |
| return event_query->signalled = FALSE; |
| |
| case WINED3D_EVENT_QUERY_WRONG_THREAD: |
| FIXME("(%p) Wrong thread, reporting GPU idle.\n", query); |
| return event_query->signalled = TRUE; |
| |
| case WINED3D_EVENT_QUERY_ERROR: |
| ERR("The GL event query failed.\n"); |
| return event_query->signalled = TRUE; |
| |
| default: |
| ERR("Unexpected wined3d_event_query_test result %#x.\n", ret); |
| return event_query->signalled = TRUE; |
| } |
| } |
| |
| void * CDECL wined3d_query_get_parent(const struct wined3d_query *query) |
| { |
| TRACE("query %p.\n", query); |
| |
| return query->parent; |
| } |
| |
| enum wined3d_query_type CDECL wined3d_query_get_type(const struct wined3d_query *query) |
| { |
| TRACE("query %p.\n", query); |
| |
| return query->type; |
| } |
| |
| static void wined3d_event_query_ops_issue(struct wined3d_query *query, DWORD flags) |
| { |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| if (flags & WINED3DISSUE_END) |
| { |
| struct wined3d_event_query *event_query = wined3d_event_query_from_query(query); |
| |
| wined3d_event_query_issue(event_query, query->device); |
| } |
| else if (flags & WINED3DISSUE_BEGIN) |
| { |
| /* Started implicitly at query creation. */ |
| ERR("Event query issued with START flag - what to do?\n"); |
| } |
| } |
| |
| static void wined3d_occlusion_query_ops_issue(struct wined3d_query *query, DWORD flags) |
| { |
| struct wined3d_occlusion_query *oq = wined3d_occlusion_query_from_query(query); |
| struct wined3d_device *device = query->device; |
| const struct wined3d_gl_info *gl_info = &device->adapter->gl_info; |
| struct wined3d_context *context; |
| |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| /* This is allowed according to MSDN and our tests. Reset the query and |
| * restart. */ |
| if (flags & WINED3DISSUE_BEGIN) |
| { |
| if (query->state == QUERY_BUILDING) |
| { |
| if (oq->context->tid != GetCurrentThreadId()) |
| { |
| FIXME("Wrong thread, can't restart query.\n"); |
| |
| context_free_occlusion_query(oq); |
| context = context_acquire(query->device, NULL); |
| context_alloc_occlusion_query(context, oq); |
| } |
| else |
| { |
| context = context_acquire(device, context_get_rt_surface(oq->context)); |
| |
| GL_EXTCALL(glEndQuery(GL_SAMPLES_PASSED)); |
| checkGLcall("glEndQuery()"); |
| } |
| } |
| else |
| { |
| if (oq->context) |
| context_free_occlusion_query(oq); |
| context = context_acquire(query->device, NULL); |
| context_alloc_occlusion_query(context, oq); |
| } |
| |
| GL_EXTCALL(glBeginQuery(GL_SAMPLES_PASSED, oq->id)); |
| checkGLcall("glBeginQuery()"); |
| |
| context_release(context); |
| } |
| if (flags & WINED3DISSUE_END) |
| { |
| /* MSDN says END on a non-building occlusion query returns an error, |
| * but our tests show that it returns OK. But OpenGL doesn't like it, |
| * so avoid generating an error. */ |
| if (query->state == QUERY_BUILDING) |
| { |
| if (oq->context->tid != GetCurrentThreadId()) |
| { |
| FIXME("Wrong thread, can't end query.\n"); |
| } |
| else |
| { |
| context = context_acquire(device, context_get_rt_surface(oq->context)); |
| |
| GL_EXTCALL(glEndQuery(GL_SAMPLES_PASSED)); |
| checkGLcall("glEndQuery()"); |
| |
| context_release(context); |
| } |
| } |
| } |
| } |
| |
| static BOOL wined3d_timestamp_query_ops_poll(struct wined3d_query *query, DWORD flags) |
| { |
| struct wined3d_timestamp_query *tq = wined3d_timestamp_query_from_query(query); |
| struct wined3d_device *device = query->device; |
| const struct wined3d_gl_info *gl_info; |
| struct wined3d_context *context; |
| GLuint64 timestamp; |
| GLuint available; |
| |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| if (tq->context->tid != GetCurrentThreadId()) |
| { |
| FIXME("%p Wrong thread, returning 1.\n", query); |
| tq->timestamp = 1; |
| return TRUE; |
| } |
| |
| context = context_acquire(device, context_get_rt_surface(tq->context)); |
| gl_info = context->gl_info; |
| |
| GL_EXTCALL(glGetQueryObjectuiv(tq->id, GL_QUERY_RESULT_AVAILABLE, &available)); |
| checkGLcall("glGetQueryObjectuiv(GL_QUERY_RESULT_AVAILABLE)"); |
| TRACE("available %#x.\n", available); |
| |
| if (available) |
| { |
| GL_EXTCALL(glGetQueryObjectui64v(tq->id, GL_QUERY_RESULT, ×tamp)); |
| checkGLcall("glGetQueryObjectui64v(GL_QUERY_RESULT)"); |
| TRACE("Returning timestamp %s.\n", wine_dbgstr_longlong(timestamp)); |
| tq->timestamp = timestamp; |
| } |
| |
| context_release(context); |
| |
| return available; |
| } |
| |
| static void wined3d_timestamp_query_ops_issue(struct wined3d_query *query, DWORD flags) |
| { |
| struct wined3d_timestamp_query *tq = wined3d_timestamp_query_from_query(query); |
| const struct wined3d_gl_info *gl_info; |
| struct wined3d_context *context; |
| |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| if (flags & WINED3DISSUE_BEGIN) |
| { |
| WARN("Ignoring WINED3DISSUE_BEGIN with a TIMESTAMP query.\n"); |
| } |
| if (flags & WINED3DISSUE_END) |
| { |
| if (tq->context) |
| context_free_timestamp_query(tq); |
| context = context_acquire(query->device, NULL); |
| gl_info = context->gl_info; |
| context_alloc_timestamp_query(context, tq); |
| GL_EXTCALL(glQueryCounter(tq->id, GL_TIMESTAMP)); |
| checkGLcall("glQueryCounter()"); |
| context_release(context); |
| } |
| } |
| |
| static BOOL wined3d_timestamp_disjoint_query_ops_poll(struct wined3d_query *query, DWORD flags) |
| { |
| TRACE("query %p, flags %#x.\n", query, flags); |
| |
| return TRUE; |
| } |
| |
| static void wined3d_timestamp_disjoint_query_ops_issue(struct wined3d_query *query, DWORD flags) |
| { |
| TRACE("query %p, flags %#x.\n", query, flags); |
| } |
| |
| static const struct wined3d_query_ops event_query_ops = |
| { |
| wined3d_event_query_ops_poll, |
| wined3d_event_query_ops_issue, |
| }; |
| |
| static HRESULT wined3d_event_query_create(struct wined3d_device *device, |
| enum wined3d_query_type type, void *parent, struct wined3d_query **query) |
| { |
| const struct wined3d_gl_info *gl_info = &device->adapter->gl_info; |
| struct wined3d_event_query *object; |
| |
| TRACE("device %p, type %#x, parent %p, query %p.\n", device, type, parent, query); |
| |
| if (!wined3d_event_query_supported(gl_info)) |
| { |
| WARN("Event queries not supported.\n"); |
| return WINED3DERR_NOTAVAILABLE; |
| } |
| |
| if (!(object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*object)))) |
| return E_OUTOFMEMORY; |
| |
| wined3d_query_init(&object->query, device, type, &object->signalled, |
| sizeof(object->signalled), &event_query_ops, parent); |
| |
| TRACE("Created query %p.\n", object); |
| *query = &object->query; |
| |
| return WINED3D_OK; |
| } |
| |
| static const struct wined3d_query_ops occlusion_query_ops = |
| { |
| wined3d_occlusion_query_ops_poll, |
| wined3d_occlusion_query_ops_issue, |
| }; |
| |
| static HRESULT wined3d_occlusion_query_create(struct wined3d_device *device, |
| enum wined3d_query_type type, void *parent, struct wined3d_query **query) |
| { |
| const struct wined3d_gl_info *gl_info = &device->adapter->gl_info; |
| struct wined3d_occlusion_query *object; |
| |
| TRACE("device %p, type %#x, parent %p, query %p.\n", device, type, parent, query); |
| |
| if (!gl_info->supported[ARB_OCCLUSION_QUERY]) |
| { |
| WARN("Unsupported in local OpenGL implementation: ARB_OCCLUSION_QUERY.\n"); |
| return WINED3DERR_NOTAVAILABLE; |
| } |
| |
| if (!(object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*object)))) |
| return E_OUTOFMEMORY; |
| |
| wined3d_query_init(&object->query, device, type, &object->samples, |
| sizeof(object->samples), &occlusion_query_ops, parent); |
| |
| TRACE("Created query %p.\n", object); |
| *query = &object->query; |
| |
| return WINED3D_OK; |
| } |
| |
| static const struct wined3d_query_ops timestamp_query_ops = |
| { |
| wined3d_timestamp_query_ops_poll, |
| wined3d_timestamp_query_ops_issue, |
| }; |
| |
| static HRESULT wined3d_timestamp_query_create(struct wined3d_device *device, |
| enum wined3d_query_type type, void *parent, struct wined3d_query **query) |
| { |
| const struct wined3d_gl_info *gl_info = &device->adapter->gl_info; |
| struct wined3d_timestamp_query *object; |
| |
| TRACE("device %p, type %#x, parent %p, query %p.\n", device, type, parent, query); |
| |
| if (!gl_info->supported[ARB_TIMER_QUERY]) |
| { |
| WARN("Unsupported in local OpenGL implementation: ARB_TIMER_QUERY.\n"); |
| return WINED3DERR_NOTAVAILABLE; |
| } |
| |
| if (!(object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*object)))) |
| return E_OUTOFMEMORY; |
| |
| wined3d_query_init(&object->query, device, type, &object->timestamp, |
| sizeof(object->timestamp), ×tamp_query_ops, parent); |
| |
| TRACE("Created query %p.\n", object); |
| *query = &object->query; |
| |
| return WINED3D_OK; |
| } |
| |
| static const struct wined3d_query_ops timestamp_disjoint_query_ops = |
| { |
| wined3d_timestamp_disjoint_query_ops_poll, |
| wined3d_timestamp_disjoint_query_ops_issue, |
| }; |
| |
| static HRESULT wined3d_timestamp_disjoint_query_create(struct wined3d_device *device, |
| enum wined3d_query_type type, void *parent, struct wined3d_query **query) |
| { |
| const struct wined3d_gl_info *gl_info = &device->adapter->gl_info; |
| struct wined3d_query *object; |
| |
| TRACE("device %p, type %#x, parent %p, query %p.\n", device, type, parent, query); |
| |
| if (!gl_info->supported[ARB_TIMER_QUERY]) |
| { |
| WARN("Unsupported in local OpenGL implementation: ARB_TIMER_QUERY.\n"); |
| return WINED3DERR_NOTAVAILABLE; |
| } |
| |
| if (!(object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*object)))) |
| return E_OUTOFMEMORY; |
| |
| if (type == WINED3D_QUERY_TYPE_TIMESTAMP_DISJOINT) |
| { |
| static const struct wined3d_query_data_timestamp_disjoint disjoint_data = {1000 * 1000 * 1000, FALSE}; |
| |
| wined3d_query_init(object, device, type, &disjoint_data, |
| sizeof(disjoint_data), ×tamp_disjoint_query_ops, parent); |
| } |
| else |
| { |
| static const UINT64 freq = 1000 * 1000 * 1000; |
| |
| wined3d_query_init(object, device, type, &freq, |
| sizeof(freq), ×tamp_disjoint_query_ops, parent); |
| } |
| |
| TRACE("Created query %p.\n", object); |
| *query = object; |
| |
| return WINED3D_OK; |
| } |
| |
| HRESULT CDECL wined3d_query_create(struct wined3d_device *device, |
| enum wined3d_query_type type, void *parent, struct wined3d_query **query) |
| { |
| TRACE("device %p, type %#x, parent %p, query %p.\n", device, type, parent, query); |
| |
| switch (type) |
| { |
| case WINED3D_QUERY_TYPE_EVENT: |
| return wined3d_event_query_create(device, type, parent, query); |
| |
| case WINED3D_QUERY_TYPE_OCCLUSION: |
| return wined3d_occlusion_query_create(device, type, parent, query); |
| |
| case WINED3D_QUERY_TYPE_TIMESTAMP: |
| return wined3d_timestamp_query_create(device, type, parent, query); |
| |
| case WINED3D_QUERY_TYPE_TIMESTAMP_DISJOINT: |
| case WINED3D_QUERY_TYPE_TIMESTAMP_FREQ: |
| return wined3d_timestamp_disjoint_query_create(device, type, parent, query); |
| |
| default: |
| FIXME("Unhandled query type %#x.\n", type); |
| return WINED3DERR_NOTAVAILABLE; |
| } |
| } |