diff options
author | Michel Dänzer <michel.daenzer@amd.com> | 2015-03-02 18:59:54 +0900 |
---|---|---|
committer | Michel Dänzer <michel@daenzer.net> | 2015-03-17 10:59:43 +0900 |
commit | b4af8a327ed8420f0ff4ea0f113f4a59406ed4d3 (patch) | |
tree | b183ef9efc1f4c04dae1aa5614c6c58bfa52c421 /src/radeon_drm_queue.c | |
parent | 7c3470f4b659206ed23f761948936ede3a2dba3d (diff) |
Add DRM event queue helpers v2
v2: Rename struct radeon_drm_queue to struct radeon_drm_queue_event,
thanks to Richard Wilbur <richard.wilbur@gmail.com> for the suggestion.
Also changed the corresponding parameter and local variable names from
'q' to 'e'.
Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
Diffstat (limited to 'src/radeon_drm_queue.c')
-rw-r--r-- | src/radeon_drm_queue.c | 181 |
1 files changed, 181 insertions, 0 deletions
diff --git a/src/radeon_drm_queue.c b/src/radeon_drm_queue.c new file mode 100644 index 00000000..5e54ef8b --- /dev/null +++ b/src/radeon_drm_queue.c @@ -0,0 +1,181 @@ +/* + * Copyright © 2007 Red Hat, Inc. + * Copyright © 2015 Advanced Micro Devices, Inc. + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice (including the next + * paragraph) shall be included in all copies or substantial portions of the + * Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + * SOFTWARE. + * + * Authors: + * Dave Airlie <airlied@redhat.com> + * + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <xorg-server.h> + +#include "radeon.h" +#include "radeon_drm_queue.h" +#include "radeon_list.h" + + +struct radeon_drm_queue_entry { + struct xorg_list list; + uint64_t id; + void *data; + ClientPtr client; + ScrnInfoPtr scrn; + radeon_drm_handler_proc handler; + radeon_drm_abort_proc abort; +}; + +static int radeon_drm_queue_refcnt; +static struct xorg_list radeon_drm_queue; + + +/* + * Handle a DRM event + */ +void +radeon_drm_queue_handler(int fd, unsigned int frame, unsigned int sec, + unsigned int usec, void *user_ptr) +{ + struct radeon_drm_queue_entry *user_data = user_ptr; + struct radeon_drm_queue_entry *e, *tmp; + + xorg_list_for_each_entry_safe(e, tmp, &radeon_drm_queue, list) { + if (e == user_data) { + xorg_list_del(&e->list); + e->handler(e->scrn, frame, + (uint64_t)sec * 1000000 + usec, e->data); + free(e); + break; + } + } +} + +/* + * Enqueue a potential drm response; when the associated response + * appears, we've got data to pass to the handler from here + */ +struct radeon_drm_queue_entry * +radeon_drm_queue_alloc(ScrnInfoPtr scrn, ClientPtr client, + uint64_t id, void *data, + radeon_drm_handler_proc handler, + radeon_drm_abort_proc abort) +{ + struct radeon_drm_queue_entry *e; + + e = calloc(1, sizeof(struct radeon_drm_queue_entry)); + if (!e) + return NULL; + + e->client = client; + e->scrn = scrn; + e->id = id; + e->data = data; + e->handler = handler; + e->abort = abort; + + xorg_list_add(&e->list, &radeon_drm_queue); + + return e; +} + +/* + * Abort one queued DRM entry, removing it + * from the list, calling the abort function and + * freeing the memory + */ +static void +radeon_drm_abort_one(struct radeon_drm_queue_entry *e) +{ + xorg_list_del(&e->list); + e->abort(e->scrn, e->data); + free(e); +} + +/* + * Abort drm queue entries for a client + */ +void +radeon_drm_abort_client(ClientPtr client) +{ + struct radeon_drm_queue_entry *e, *tmp; + + xorg_list_for_each_entry_safe(e, tmp, &radeon_drm_queue, list) { + if (e->client == client) + radeon_drm_abort_one(e); + } +} + +/* + * Abort specific drm queue entry + */ +void +radeon_drm_abort_entry(struct radeon_drm_queue_entry *entry) +{ + radeon_drm_abort_one(entry); +} + +/* + * Abort specific drm queue entry by ID + */ +void +radeon_drm_abort_id(uint64_t id) +{ + struct radeon_drm_queue_entry *e, *tmp; + + xorg_list_for_each_entry_safe(e, tmp, &radeon_drm_queue, list) { + if (e->id == id) { + radeon_drm_abort_one(e); + break; + } + } +} + +/* + * Initialize the DRM event queue + */ +void +radeon_drm_queue_init() +{ + if (radeon_drm_queue_refcnt++) + return; + + xorg_list_init(&radeon_drm_queue); +} + +/* + * Deinitialize the DRM event queue + */ +void +radeon_drm_queue_close(ScrnInfoPtr scrn) +{ + struct radeon_drm_queue_entry *e, *tmp; + + xorg_list_for_each_entry_safe(e, tmp, &radeon_drm_queue, list) { + if (e->scrn == scrn) + radeon_drm_abort_one(e); + } + + radeon_drm_queue_refcnt--; +} |