xref: /OK3568_Linux_fs/external/xserver/present/present_fence.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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