xref: /OK3568_Linux_fs/external/xserver/present/present_event.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 
29*4882a593Smuzhiyun static RESTYPE present_event_type;
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun static int
present_free_event(void * data,XID id)32*4882a593Smuzhiyun present_free_event(void *data, XID id)
33*4882a593Smuzhiyun {
34*4882a593Smuzhiyun     present_event_ptr present_event = (present_event_ptr) data;
35*4882a593Smuzhiyun     present_window_priv_ptr window_priv = present_window_priv(present_event->window);
36*4882a593Smuzhiyun     present_event_ptr *previous, current;
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun     for (previous = &window_priv->events; (current = *previous); previous = &current->next) {
39*4882a593Smuzhiyun         if (current == present_event) {
40*4882a593Smuzhiyun             *previous = present_event->next;
41*4882a593Smuzhiyun             break;
42*4882a593Smuzhiyun         }
43*4882a593Smuzhiyun     }
44*4882a593Smuzhiyun     free((void *) present_event);
45*4882a593Smuzhiyun     return 1;
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun void
present_free_events(WindowPtr window)50*4882a593Smuzhiyun present_free_events(WindowPtr window)
51*4882a593Smuzhiyun {
52*4882a593Smuzhiyun     present_window_priv_ptr window_priv = present_window_priv(window);
53*4882a593Smuzhiyun     present_event_ptr event;
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun     if (!window_priv)
56*4882a593Smuzhiyun         return;
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun     while ((event = window_priv->events))
59*4882a593Smuzhiyun         FreeResource(event->id, RT_NONE);
60*4882a593Smuzhiyun }
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun static void
present_event_swap(xGenericEvent * from,xGenericEvent * to)63*4882a593Smuzhiyun present_event_swap(xGenericEvent *from, xGenericEvent *to)
64*4882a593Smuzhiyun {
65*4882a593Smuzhiyun     *to = *from;
66*4882a593Smuzhiyun     swaps(&to->sequenceNumber);
67*4882a593Smuzhiyun     swapl(&to->length);
68*4882a593Smuzhiyun     swaps(&to->evtype);
69*4882a593Smuzhiyun     switch (from->evtype) {
70*4882a593Smuzhiyun     case PresentConfigureNotify: {
71*4882a593Smuzhiyun         xPresentConfigureNotify *c = (xPresentConfigureNotify *) to;
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun         swapl(&c->eid);
74*4882a593Smuzhiyun         swapl(&c->window);
75*4882a593Smuzhiyun         swaps(&c->x);
76*4882a593Smuzhiyun         swaps(&c->y);
77*4882a593Smuzhiyun         swaps(&c->width);
78*4882a593Smuzhiyun         swaps(&c->height);
79*4882a593Smuzhiyun         swaps(&c->off_x);
80*4882a593Smuzhiyun         swaps(&c->off_y);
81*4882a593Smuzhiyun         swaps(&c->pixmap_width);
82*4882a593Smuzhiyun         swaps(&c->pixmap_height);
83*4882a593Smuzhiyun         swapl(&c->pixmap_flags);
84*4882a593Smuzhiyun         break;
85*4882a593Smuzhiyun     }
86*4882a593Smuzhiyun     case PresentCompleteNotify:
87*4882a593Smuzhiyun     {
88*4882a593Smuzhiyun         xPresentCompleteNotify *c = (xPresentCompleteNotify *) to;
89*4882a593Smuzhiyun         swapl(&c->eid);
90*4882a593Smuzhiyun         swapl(&c->window);
91*4882a593Smuzhiyun         swapl(&c->serial);
92*4882a593Smuzhiyun         swapll(&c->ust);
93*4882a593Smuzhiyun         swapll(&c->msc);
94*4882a593Smuzhiyun         break;
95*4882a593Smuzhiyun     }
96*4882a593Smuzhiyun     case PresentIdleNotify:
97*4882a593Smuzhiyun     {
98*4882a593Smuzhiyun         xPresentIdleNotify *c = (xPresentIdleNotify *) to;
99*4882a593Smuzhiyun         swapl(&c->eid);
100*4882a593Smuzhiyun         swapl(&c->window);
101*4882a593Smuzhiyun         swapl(&c->serial);
102*4882a593Smuzhiyun         swapl(&c->idle_fence);
103*4882a593Smuzhiyun         break;
104*4882a593Smuzhiyun     }
105*4882a593Smuzhiyun     }
106*4882a593Smuzhiyun }
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun void
present_send_config_notify(WindowPtr window,int x,int y,int w,int h,int bw,WindowPtr sibling)109*4882a593Smuzhiyun present_send_config_notify(WindowPtr window, int x, int y, int w, int h, int bw, WindowPtr sibling)
110*4882a593Smuzhiyun {
111*4882a593Smuzhiyun     present_window_priv_ptr window_priv = present_window_priv(window);
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun     if (window_priv) {
114*4882a593Smuzhiyun         xPresentConfigureNotify cn = {
115*4882a593Smuzhiyun             .type = GenericEvent,
116*4882a593Smuzhiyun             .extension = present_request,
117*4882a593Smuzhiyun             .length = (sizeof(xPresentConfigureNotify) - 32) >> 2,
118*4882a593Smuzhiyun             .evtype = PresentConfigureNotify,
119*4882a593Smuzhiyun             .eid = 0,
120*4882a593Smuzhiyun             .window = window->drawable.id,
121*4882a593Smuzhiyun             .x = x,
122*4882a593Smuzhiyun             .y = y,
123*4882a593Smuzhiyun             .width = w,
124*4882a593Smuzhiyun             .height = h,
125*4882a593Smuzhiyun             .off_x = 0,
126*4882a593Smuzhiyun             .off_y = 0,
127*4882a593Smuzhiyun             .pixmap_width = w,
128*4882a593Smuzhiyun             .pixmap_height = h,
129*4882a593Smuzhiyun             .pixmap_flags = 0
130*4882a593Smuzhiyun         };
131*4882a593Smuzhiyun         present_event_ptr event;
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun         for (event = window_priv->events; event; event = event->next) {
134*4882a593Smuzhiyun             if (event->mask & (1 << PresentConfigureNotify)) {
135*4882a593Smuzhiyun                 cn.eid = event->id;
136*4882a593Smuzhiyun                 WriteEventsToClient(event->client, 1, (xEvent *) &cn);
137*4882a593Smuzhiyun             }
138*4882a593Smuzhiyun         }
139*4882a593Smuzhiyun     }
140*4882a593Smuzhiyun }
141*4882a593Smuzhiyun 
142*4882a593Smuzhiyun static present_complete_notify_proc complete_notify;
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun void
present_register_complete_notify(present_complete_notify_proc proc)145*4882a593Smuzhiyun present_register_complete_notify(present_complete_notify_proc proc)
146*4882a593Smuzhiyun {
147*4882a593Smuzhiyun     complete_notify = proc;
148*4882a593Smuzhiyun }
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun void
present_send_complete_notify(WindowPtr window,CARD8 kind,CARD8 mode,CARD32 serial,uint64_t ust,uint64_t msc)151*4882a593Smuzhiyun present_send_complete_notify(WindowPtr window, CARD8 kind, CARD8 mode, CARD32 serial, uint64_t ust, uint64_t msc)
152*4882a593Smuzhiyun {
153*4882a593Smuzhiyun     present_window_priv_ptr window_priv = present_window_priv(window);
154*4882a593Smuzhiyun 
155*4882a593Smuzhiyun     if (window_priv) {
156*4882a593Smuzhiyun         xPresentCompleteNotify cn = {
157*4882a593Smuzhiyun             .type = GenericEvent,
158*4882a593Smuzhiyun             .extension = present_request,
159*4882a593Smuzhiyun             .length = (sizeof(xPresentCompleteNotify) - 32) >> 2,
160*4882a593Smuzhiyun             .evtype = PresentCompleteNotify,
161*4882a593Smuzhiyun             .kind = kind,
162*4882a593Smuzhiyun             .mode = mode,
163*4882a593Smuzhiyun             .eid = 0,
164*4882a593Smuzhiyun             .window = window->drawable.id,
165*4882a593Smuzhiyun             .serial = serial,
166*4882a593Smuzhiyun             .ust = ust,
167*4882a593Smuzhiyun             .msc = msc,
168*4882a593Smuzhiyun         };
169*4882a593Smuzhiyun         present_event_ptr event;
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun         for (event = window_priv->events; event; event = event->next) {
172*4882a593Smuzhiyun             if (event->mask & PresentCompleteNotifyMask) {
173*4882a593Smuzhiyun                 cn.eid = event->id;
174*4882a593Smuzhiyun                 WriteEventsToClient(event->client, 1, (xEvent *) &cn);
175*4882a593Smuzhiyun             }
176*4882a593Smuzhiyun         }
177*4882a593Smuzhiyun     }
178*4882a593Smuzhiyun     if (complete_notify)
179*4882a593Smuzhiyun         (*complete_notify)(window, kind, mode, serial, ust, msc);
180*4882a593Smuzhiyun }
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun void
present_send_idle_notify(WindowPtr window,CARD32 serial,PixmapPtr pixmap,struct present_fence * idle_fence)183*4882a593Smuzhiyun present_send_idle_notify(WindowPtr window, CARD32 serial, PixmapPtr pixmap, struct present_fence *idle_fence)
184*4882a593Smuzhiyun {
185*4882a593Smuzhiyun     present_window_priv_ptr window_priv = present_window_priv(window);
186*4882a593Smuzhiyun 
187*4882a593Smuzhiyun     if (window_priv) {
188*4882a593Smuzhiyun         xPresentIdleNotify in = {
189*4882a593Smuzhiyun             .type = GenericEvent,
190*4882a593Smuzhiyun             .extension = present_request,
191*4882a593Smuzhiyun             .length = (sizeof(xPresentIdleNotify) - 32) >> 2,
192*4882a593Smuzhiyun             .evtype = PresentIdleNotify,
193*4882a593Smuzhiyun             .eid = 0,
194*4882a593Smuzhiyun             .window = window->drawable.id,
195*4882a593Smuzhiyun             .serial = serial,
196*4882a593Smuzhiyun             .pixmap = pixmap->drawable.id,
197*4882a593Smuzhiyun             .idle_fence = present_fence_id(idle_fence)
198*4882a593Smuzhiyun         };
199*4882a593Smuzhiyun         present_event_ptr event;
200*4882a593Smuzhiyun 
201*4882a593Smuzhiyun         for (event = window_priv->events; event; event = event->next) {
202*4882a593Smuzhiyun             if (event->mask & PresentIdleNotifyMask) {
203*4882a593Smuzhiyun                 in.eid = event->id;
204*4882a593Smuzhiyun                 WriteEventsToClient(event->client, 1, (xEvent *) &in);
205*4882a593Smuzhiyun             }
206*4882a593Smuzhiyun         }
207*4882a593Smuzhiyun     }
208*4882a593Smuzhiyun }
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun int
present_select_input(ClientPtr client,XID eid,WindowPtr window,CARD32 mask)211*4882a593Smuzhiyun present_select_input(ClientPtr client, XID eid, WindowPtr window, CARD32 mask)
212*4882a593Smuzhiyun {
213*4882a593Smuzhiyun     present_window_priv_ptr window_priv;
214*4882a593Smuzhiyun     present_event_ptr event;
215*4882a593Smuzhiyun     int ret;
216*4882a593Smuzhiyun 
217*4882a593Smuzhiyun     /* Check to see if we're modifying an existing event selection */
218*4882a593Smuzhiyun     ret = dixLookupResourceByType((void **) &event, eid, present_event_type,
219*4882a593Smuzhiyun                                  client, DixWriteAccess);
220*4882a593Smuzhiyun     if (ret == Success) {
221*4882a593Smuzhiyun         /* Match error for the wrong window; also don't modify some other
222*4882a593Smuzhiyun          * client's event selection
223*4882a593Smuzhiyun          */
224*4882a593Smuzhiyun         if (event->window != window || event->client != client)
225*4882a593Smuzhiyun             return BadMatch;
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun         if (mask)
228*4882a593Smuzhiyun             event->mask = mask;
229*4882a593Smuzhiyun         else
230*4882a593Smuzhiyun             FreeResource(eid, RT_NONE);
231*4882a593Smuzhiyun         return Success;
232*4882a593Smuzhiyun     }
233*4882a593Smuzhiyun     if (ret != BadValue)
234*4882a593Smuzhiyun         return ret;
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun     if (mask == 0)
237*4882a593Smuzhiyun         return Success;
238*4882a593Smuzhiyun 
239*4882a593Smuzhiyun     LEGAL_NEW_RESOURCE(eid, client);
240*4882a593Smuzhiyun 
241*4882a593Smuzhiyun     window_priv = present_get_window_priv(window, TRUE);
242*4882a593Smuzhiyun     if (!window_priv)
243*4882a593Smuzhiyun         return BadAlloc;
244*4882a593Smuzhiyun 
245*4882a593Smuzhiyun     event = calloc (1, sizeof (present_event_rec));
246*4882a593Smuzhiyun     if (!event)
247*4882a593Smuzhiyun         return BadAlloc;
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun     event->client = client;
250*4882a593Smuzhiyun     event->window = window;
251*4882a593Smuzhiyun     event->id = eid;
252*4882a593Smuzhiyun     event->mask = mask;
253*4882a593Smuzhiyun 
254*4882a593Smuzhiyun     event->next = window_priv->events;
255*4882a593Smuzhiyun     window_priv->events = event;
256*4882a593Smuzhiyun 
257*4882a593Smuzhiyun     if (!AddResource(event->id, present_event_type, (void *) event))
258*4882a593Smuzhiyun         return BadAlloc;
259*4882a593Smuzhiyun 
260*4882a593Smuzhiyun     return Success;
261*4882a593Smuzhiyun }
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun Bool
present_event_init(void)264*4882a593Smuzhiyun present_event_init(void)
265*4882a593Smuzhiyun {
266*4882a593Smuzhiyun     present_event_type = CreateNewResourceType(present_free_event, "PresentEvent");
267*4882a593Smuzhiyun     if (!present_event_type)
268*4882a593Smuzhiyun         return FALSE;
269*4882a593Smuzhiyun 
270*4882a593Smuzhiyun     GERegisterExtension(present_request, present_event_swap);
271*4882a593Smuzhiyun     return TRUE;
272*4882a593Smuzhiyun }
273