2011-12-16 17:49:28 +00:00
|
|
|
/*
|
|
|
|
* Cogl
|
|
|
|
*
|
2014-02-22 01:28:54 +00:00
|
|
|
* A Low Level GPU Graphics and Utilities API
|
2011-12-16 17:49:28 +00:00
|
|
|
*
|
|
|
|
* Copyright (C) 2012 Intel Corporation.
|
|
|
|
*
|
2014-02-22 01:28:54 +00:00
|
|
|
* 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:
|
2011-12-16 17:49:28 +00:00
|
|
|
*
|
2014-02-22 01:28:54 +00:00
|
|
|
* The above copyright notice and this permission notice shall be
|
|
|
|
* included in all copies or substantial portions of the Software.
|
2011-12-16 17:49:28 +00:00
|
|
|
*
|
2014-02-22 01:28:54 +00:00
|
|
|
* 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.
|
2011-12-16 17:49:28 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Neil Roberts <neil@linux.intel.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "cogl-poll.h"
|
2013-04-16 23:46:03 +01:00
|
|
|
#include "cogl-poll-private.h"
|
2011-12-16 17:49:28 +00:00
|
|
|
#include "cogl-winsys-private.h"
|
2013-04-16 23:46:03 +01:00
|
|
|
#include "cogl-renderer-private.h"
|
2011-12-16 17:49:28 +00:00
|
|
|
|
2013-05-14 02:51:55 +01:00
|
|
|
struct _CoglPollSource
|
2013-04-18 16:26:21 +01:00
|
|
|
{
|
|
|
|
int fd;
|
2013-05-14 02:51:55 +01:00
|
|
|
CoglPollPrepareCallback prepare;
|
2013-04-18 16:26:21 +01:00
|
|
|
CoglPollDispatchCallback dispatch;
|
|
|
|
void *user_data;
|
2013-05-14 02:51:55 +01:00
|
|
|
};
|
2013-04-18 16:26:21 +01:00
|
|
|
|
2013-04-16 23:46:03 +01:00
|
|
|
int
|
|
|
|
cogl_poll_renderer_get_info (CoglRenderer *renderer,
|
|
|
|
CoglPollFD **poll_fds,
|
|
|
|
int *n_poll_fds,
|
|
|
|
int64_t *timeout)
|
2011-12-16 17:49:28 +00:00
|
|
|
{
|
2013-07-09 13:39:11 +01:00
|
|
|
GList *l, *next;
|
2011-12-16 17:49:28 +00:00
|
|
|
|
2013-04-16 23:46:03 +01:00
|
|
|
_COGL_RETURN_VAL_IF_FAIL (cogl_is_renderer (renderer), 0);
|
|
|
|
_COGL_RETURN_VAL_IF_FAIL (poll_fds != NULL, 0);
|
|
|
|
_COGL_RETURN_VAL_IF_FAIL (n_poll_fds != NULL, 0);
|
|
|
|
_COGL_RETURN_VAL_IF_FAIL (timeout != NULL, 0);
|
2011-12-16 17:49:28 +00:00
|
|
|
|
2013-05-14 02:51:55 +01:00
|
|
|
*timeout = -1;
|
2011-12-16 17:49:28 +00:00
|
|
|
|
2013-06-08 23:03:25 +01:00
|
|
|
if (!_cogl_list_empty (&renderer->idle_closures))
|
2013-06-28 12:11:20 +01:00
|
|
|
*timeout = 0;
|
2013-04-16 23:46:03 +01:00
|
|
|
|
2013-07-09 13:39:11 +01:00
|
|
|
/* This loop needs to cope with the prepare callback removing its
|
|
|
|
* own fd */
|
|
|
|
for (l = renderer->poll_sources; l; l = next)
|
2013-04-18 16:26:21 +01:00
|
|
|
{
|
|
|
|
CoglPollSource *source = l->data;
|
2013-07-09 13:39:11 +01:00
|
|
|
|
|
|
|
next = l->next;
|
|
|
|
|
2013-05-14 02:51:55 +01:00
|
|
|
if (source->prepare)
|
2013-04-18 16:26:21 +01:00
|
|
|
{
|
2013-05-14 02:51:55 +01:00
|
|
|
int64_t source_timeout = source->prepare (source->user_data);
|
2013-06-28 12:11:20 +01:00
|
|
|
if (source_timeout >= 0 &&
|
|
|
|
(*timeout == -1 || *timeout > source_timeout))
|
2013-05-14 02:51:55 +01:00
|
|
|
*timeout = source_timeout;
|
2013-04-18 16:26:21 +01:00
|
|
|
}
|
|
|
|
}
|
2011-12-16 17:49:28 +00:00
|
|
|
|
2013-07-09 13:39:11 +01:00
|
|
|
/* This is deliberately set after calling the prepare callbacks in
|
|
|
|
* case one of them removes its fd */
|
|
|
|
*poll_fds = (void *)renderer->poll_fds->data;
|
|
|
|
*n_poll_fds = renderer->poll_fds->len;
|
|
|
|
|
2013-04-16 23:46:03 +01:00
|
|
|
return renderer->poll_fds_age;
|
2011-12-16 17:49:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-04-16 23:46:03 +01:00
|
|
|
cogl_poll_renderer_dispatch (CoglRenderer *renderer,
|
|
|
|
const CoglPollFD *poll_fds,
|
|
|
|
int n_poll_fds)
|
2011-12-16 17:49:28 +00:00
|
|
|
{
|
2013-07-09 13:39:11 +01:00
|
|
|
GList *l, *next;
|
2011-12-16 17:49:28 +00:00
|
|
|
|
2013-04-16 23:46:03 +01:00
|
|
|
_COGL_RETURN_IF_FAIL (cogl_is_renderer (renderer));
|
|
|
|
|
2013-04-18 14:19:43 +01:00
|
|
|
_cogl_closure_list_invoke_no_args (&renderer->idle_closures);
|
2012-11-12 11:58:10 -05:00
|
|
|
|
2013-07-09 13:39:11 +01:00
|
|
|
/* This loop needs to cope with the dispatch callback removing its
|
|
|
|
* own fd */
|
|
|
|
for (l = renderer->poll_sources; l; l = next)
|
2013-04-18 16:26:21 +01:00
|
|
|
{
|
|
|
|
CoglPollSource *source = l->data;
|
|
|
|
int i;
|
|
|
|
|
2013-07-09 13:39:11 +01:00
|
|
|
next = l->next;
|
|
|
|
|
2013-05-14 02:51:55 +01:00
|
|
|
if (source->fd == -1)
|
|
|
|
{
|
|
|
|
source->dispatch (source->user_data, 0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-04-18 16:26:21 +01:00
|
|
|
for (i = 0; i < n_poll_fds; i++)
|
|
|
|
{
|
|
|
|
const CoglPollFD *pollfd = &poll_fds[i];
|
|
|
|
|
|
|
|
if (pollfd->fd == source->fd)
|
|
|
|
{
|
2013-05-14 02:51:55 +01:00
|
|
|
source->dispatch (source->user_data, pollfd->revents);
|
2013-04-18 16:26:21 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-04-16 23:46:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
find_pollfd (CoglRenderer *renderer, int fd)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < renderer->poll_fds->len; i++)
|
|
|
|
{
|
|
|
|
CoglPollFD *pollfd = &g_array_index (renderer->poll_fds, CoglPollFD, i);
|
|
|
|
|
|
|
|
if (pollfd->fd == fd)
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_cogl_poll_renderer_remove_fd (CoglRenderer *renderer, int fd)
|
|
|
|
{
|
|
|
|
int i = find_pollfd (renderer, fd);
|
2013-04-18 16:26:21 +01:00
|
|
|
GList *l;
|
2013-04-16 23:46:03 +01:00
|
|
|
|
|
|
|
if (i < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_array_remove_index_fast (renderer->poll_fds, i);
|
|
|
|
renderer->poll_fds_age++;
|
2013-04-18 16:26:21 +01:00
|
|
|
|
|
|
|
for (l = renderer->poll_sources; l; l = l->next)
|
|
|
|
{
|
|
|
|
CoglPollSource *source = l->data;
|
|
|
|
if (source->fd == fd)
|
|
|
|
{
|
|
|
|
renderer->poll_sources =
|
|
|
|
g_list_delete_link (renderer->poll_sources, l);
|
|
|
|
g_slice_free (CoglPollSource, source);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-04-16 23:46:03 +01:00
|
|
|
}
|
|
|
|
|
2013-06-28 12:15:26 +01:00
|
|
|
void
|
|
|
|
_cogl_poll_renderer_modify_fd (CoglRenderer *renderer,
|
|
|
|
int fd,
|
|
|
|
CoglPollFDEvent events)
|
|
|
|
{
|
|
|
|
int fd_index = find_pollfd (renderer, fd);
|
|
|
|
|
|
|
|
if (fd_index == -1)
|
|
|
|
g_warn_if_reached ();
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CoglPollFD *pollfd =
|
|
|
|
&g_array_index (renderer->poll_sources, CoglPollFD, fd_index);
|
|
|
|
|
|
|
|
pollfd->events = events;
|
|
|
|
renderer->poll_fds_age++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-16 23:46:03 +01:00
|
|
|
void
|
|
|
|
_cogl_poll_renderer_add_fd (CoglRenderer *renderer,
|
|
|
|
int fd,
|
2013-04-18 16:26:21 +01:00
|
|
|
CoglPollFDEvent events,
|
2013-05-14 02:51:55 +01:00
|
|
|
CoglPollPrepareCallback prepare,
|
2013-04-18 16:26:21 +01:00
|
|
|
CoglPollDispatchCallback dispatch,
|
|
|
|
void *user_data)
|
2013-04-16 23:46:03 +01:00
|
|
|
{
|
|
|
|
CoglPollFD pollfd = {
|
|
|
|
fd,
|
|
|
|
events
|
|
|
|
};
|
2013-04-18 16:26:21 +01:00
|
|
|
CoglPollSource *source;
|
2013-04-16 23:46:03 +01:00
|
|
|
|
|
|
|
_cogl_poll_renderer_remove_fd (renderer, fd);
|
|
|
|
|
2013-04-18 16:26:21 +01:00
|
|
|
source = g_slice_new0 (CoglPollSource);
|
|
|
|
source->fd = fd;
|
2013-05-14 02:51:55 +01:00
|
|
|
source->prepare = prepare;
|
2013-04-18 16:26:21 +01:00
|
|
|
source->dispatch = dispatch;
|
|
|
|
source->user_data = user_data;
|
|
|
|
|
|
|
|
renderer->poll_sources = g_list_prepend (renderer->poll_sources, source);
|
|
|
|
|
2013-04-16 23:46:03 +01:00
|
|
|
g_array_append_val (renderer->poll_fds, pollfd);
|
|
|
|
renderer->poll_fds_age++;
|
2011-12-16 17:49:28 +00:00
|
|
|
}
|
2013-04-18 14:19:43 +01:00
|
|
|
|
2013-05-14 02:51:55 +01:00
|
|
|
CoglPollSource *
|
|
|
|
_cogl_poll_renderer_add_source (CoglRenderer *renderer,
|
|
|
|
CoglPollPrepareCallback prepare,
|
|
|
|
CoglPollDispatchCallback dispatch,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
CoglPollSource *source;
|
|
|
|
|
|
|
|
source = g_slice_new0 (CoglPollSource);
|
|
|
|
source->fd = -1;
|
|
|
|
source->prepare = prepare;
|
|
|
|
source->dispatch = dispatch;
|
|
|
|
source->user_data = user_data;
|
|
|
|
|
|
|
|
renderer->poll_sources = g_list_prepend (renderer->poll_sources, source);
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_cogl_poll_renderer_remove_source (CoglRenderer *renderer,
|
|
|
|
CoglPollSource *source)
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
for (l = renderer->poll_sources; l; l = l->next)
|
|
|
|
{
|
|
|
|
if (l->data == source)
|
|
|
|
{
|
|
|
|
renderer->poll_sources =
|
|
|
|
g_list_delete_link (renderer->poll_sources, l);
|
|
|
|
g_slice_free (CoglPollSource, source);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-18 14:19:43 +01:00
|
|
|
CoglClosure *
|
|
|
|
_cogl_poll_renderer_add_idle (CoglRenderer *renderer,
|
|
|
|
CoglIdleCallback idle_cb,
|
|
|
|
void *user_data,
|
|
|
|
CoglUserDataDestroyCallback destroy_cb)
|
|
|
|
{
|
|
|
|
return _cogl_closure_list_add (&renderer->idle_closures,
|
|
|
|
idle_cb,
|
|
|
|
user_data,
|
|
|
|
destroy_cb);
|
|
|
|
}
|