1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright © 2013 Keith Packard
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Permission to use, copy, modify, distribute, and sell this software and its
5*4882a593Smuzhiyun * documentation for any purpose is hereby granted without fee, provided that
6*4882a593Smuzhiyun * the above copyright notice appear in all copies and that both that copyright
7*4882a593Smuzhiyun * notice and this permission notice appear in supporting documentation, and
8*4882a593Smuzhiyun * that the name of the copyright holders not be used in advertising or
9*4882a593Smuzhiyun * publicity pertaining to distribution of the software without specific,
10*4882a593Smuzhiyun * written prior permission. The copyright holders make no representations
11*4882a593Smuzhiyun * about the suitability of this software for any purpose. It is provided "as
12*4882a593Smuzhiyun * is" without express or implied warranty.
13*4882a593Smuzhiyun *
14*4882a593Smuzhiyun * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
15*4882a593Smuzhiyun * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
16*4882a593Smuzhiyun * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
17*4882a593Smuzhiyun * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
18*4882a593Smuzhiyun * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
19*4882a593Smuzhiyun * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
20*4882a593Smuzhiyun * OF THIS SOFTWARE.
21*4882a593Smuzhiyun */
22*4882a593Smuzhiyun
23*4882a593Smuzhiyun #ifdef HAVE_XORG_CONFIG_H
24*4882a593Smuzhiyun #include <xorg-config.h>
25*4882a593Smuzhiyun #endif
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #include "present_priv.h"
28*4882a593Smuzhiyun #include <gcstruct.h>
29*4882a593Smuzhiyun #include <misync.h>
30*4882a593Smuzhiyun #include <misyncstr.h>
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun /*
33*4882a593Smuzhiyun * Wraps SyncFence objects so we can add a SyncTrigger to find out
34*4882a593Smuzhiyun * when the SyncFence gets destroyed and clean up appropriately
35*4882a593Smuzhiyun */
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun struct present_fence {
38*4882a593Smuzhiyun SyncTrigger trigger;
39*4882a593Smuzhiyun SyncFence *fence;
40*4882a593Smuzhiyun void (*callback)(void *param);
41*4882a593Smuzhiyun void *param;
42*4882a593Smuzhiyun };
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun /*
45*4882a593Smuzhiyun * SyncTrigger callbacks
46*4882a593Smuzhiyun */
47*4882a593Smuzhiyun static Bool
present_fence_sync_check_trigger(SyncTrigger * trigger,int64_t oldval)48*4882a593Smuzhiyun present_fence_sync_check_trigger(SyncTrigger *trigger, int64_t oldval)
49*4882a593Smuzhiyun {
50*4882a593Smuzhiyun struct present_fence *present_fence = container_of(trigger, struct present_fence, trigger);
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun return present_fence->callback != NULL;
53*4882a593Smuzhiyun }
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun static void
present_fence_sync_trigger_fired(SyncTrigger * trigger)56*4882a593Smuzhiyun present_fence_sync_trigger_fired(SyncTrigger *trigger)
57*4882a593Smuzhiyun {
58*4882a593Smuzhiyun struct present_fence *present_fence = container_of(trigger, struct present_fence, trigger);
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun if (present_fence->callback)
61*4882a593Smuzhiyun (*present_fence->callback)(present_fence->param);
62*4882a593Smuzhiyun }
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun static void
present_fence_sync_counter_destroyed(SyncTrigger * trigger)65*4882a593Smuzhiyun present_fence_sync_counter_destroyed(SyncTrigger *trigger)
66*4882a593Smuzhiyun {
67*4882a593Smuzhiyun struct present_fence *present_fence = container_of(trigger, struct present_fence, trigger);
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun present_fence->fence = NULL;
70*4882a593Smuzhiyun }
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun struct present_fence *
present_fence_create(SyncFence * fence)73*4882a593Smuzhiyun present_fence_create(SyncFence *fence)
74*4882a593Smuzhiyun {
75*4882a593Smuzhiyun struct present_fence *present_fence;
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun present_fence = calloc (1, sizeof (struct present_fence));
78*4882a593Smuzhiyun if (!present_fence)
79*4882a593Smuzhiyun return NULL;
80*4882a593Smuzhiyun
81*4882a593Smuzhiyun present_fence->fence = fence;
82*4882a593Smuzhiyun present_fence->trigger.pSync = (SyncObject *) fence;
83*4882a593Smuzhiyun present_fence->trigger.CheckTrigger = present_fence_sync_check_trigger;
84*4882a593Smuzhiyun present_fence->trigger.TriggerFired = present_fence_sync_trigger_fired;
85*4882a593Smuzhiyun present_fence->trigger.CounterDestroyed = present_fence_sync_counter_destroyed;
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun if (SyncAddTriggerToSyncObject(&present_fence->trigger) != Success) {
88*4882a593Smuzhiyun free (present_fence);
89*4882a593Smuzhiyun return NULL;
90*4882a593Smuzhiyun }
91*4882a593Smuzhiyun return present_fence;
92*4882a593Smuzhiyun }
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun void
present_fence_destroy(struct present_fence * present_fence)95*4882a593Smuzhiyun present_fence_destroy(struct present_fence *present_fence)
96*4882a593Smuzhiyun {
97*4882a593Smuzhiyun if (present_fence) {
98*4882a593Smuzhiyun if (present_fence->fence)
99*4882a593Smuzhiyun SyncDeleteTriggerFromSyncObject(&present_fence->trigger);
100*4882a593Smuzhiyun free(present_fence);
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun }
103*4882a593Smuzhiyun
104*4882a593Smuzhiyun void
present_fence_set_triggered(struct present_fence * present_fence)105*4882a593Smuzhiyun present_fence_set_triggered(struct present_fence *present_fence)
106*4882a593Smuzhiyun {
107*4882a593Smuzhiyun if (present_fence)
108*4882a593Smuzhiyun if (present_fence->fence)
109*4882a593Smuzhiyun (*present_fence->fence->funcs.SetTriggered) (present_fence->fence);
110*4882a593Smuzhiyun }
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun Bool
present_fence_check_triggered(struct present_fence * present_fence)113*4882a593Smuzhiyun present_fence_check_triggered(struct present_fence *present_fence)
114*4882a593Smuzhiyun {
115*4882a593Smuzhiyun if (!present_fence)
116*4882a593Smuzhiyun return TRUE;
117*4882a593Smuzhiyun if (!present_fence->fence)
118*4882a593Smuzhiyun return TRUE;
119*4882a593Smuzhiyun return (*present_fence->fence->funcs.CheckTriggered)(present_fence->fence);
120*4882a593Smuzhiyun }
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun void
present_fence_set_callback(struct present_fence * present_fence,void (* callback)(void * param),void * param)123*4882a593Smuzhiyun present_fence_set_callback(struct present_fence *present_fence,
124*4882a593Smuzhiyun void (*callback) (void *param),
125*4882a593Smuzhiyun void *param)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun present_fence->callback = callback;
128*4882a593Smuzhiyun present_fence->param = param;
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun XID
present_fence_id(struct present_fence * present_fence)132*4882a593Smuzhiyun present_fence_id(struct present_fence *present_fence)
133*4882a593Smuzhiyun {
134*4882a593Smuzhiyun if (!present_fence)
135*4882a593Smuzhiyun return None;
136*4882a593Smuzhiyun if (!present_fence->fence)
137*4882a593Smuzhiyun return None;
138*4882a593Smuzhiyun return present_fence->fence->sync.id;
139*4882a593Smuzhiyun }
140