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 = ¤t->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