1*7688df22SAndroid Build Coastguard Worker /*
2*7688df22SAndroid Build Coastguard Worker * DRM based mode setting test program
3*7688df22SAndroid Build Coastguard Worker * Copyright (C) 2013 Red Hat
4*7688df22SAndroid Build Coastguard Worker * Author: Rob Clark <[email protected]>
5*7688df22SAndroid Build Coastguard Worker *
6*7688df22SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
7*7688df22SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
8*7688df22SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
9*7688df22SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10*7688df22SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
11*7688df22SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
12*7688df22SAndroid Build Coastguard Worker *
13*7688df22SAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included in
14*7688df22SAndroid Build Coastguard Worker * all copies or substantial portions of the Software.
15*7688df22SAndroid Build Coastguard Worker *
16*7688df22SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*7688df22SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*7688df22SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*7688df22SAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*7688df22SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*7688df22SAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*7688df22SAndroid Build Coastguard Worker * IN THE SOFTWARE.
23*7688df22SAndroid Build Coastguard Worker */
24*7688df22SAndroid Build Coastguard Worker
25*7688df22SAndroid Build Coastguard Worker #include <assert.h>
26*7688df22SAndroid Build Coastguard Worker #include <errno.h>
27*7688df22SAndroid Build Coastguard Worker #include <stdio.h>
28*7688df22SAndroid Build Coastguard Worker #include <stdlib.h>
29*7688df22SAndroid Build Coastguard Worker #include <stdint.h>
30*7688df22SAndroid Build Coastguard Worker #include <string.h>
31*7688df22SAndroid Build Coastguard Worker #include <signal.h>
32*7688df22SAndroid Build Coastguard Worker #include <sys/time.h>
33*7688df22SAndroid Build Coastguard Worker #include <pthread.h>
34*7688df22SAndroid Build Coastguard Worker #include <unistd.h>
35*7688df22SAndroid Build Coastguard Worker
36*7688df22SAndroid Build Coastguard Worker #include "xf86drm.h"
37*7688df22SAndroid Build Coastguard Worker #include "xf86drmMode.h"
38*7688df22SAndroid Build Coastguard Worker
39*7688df22SAndroid Build Coastguard Worker #include "util/common.h"
40*7688df22SAndroid Build Coastguard Worker
41*7688df22SAndroid Build Coastguard Worker #include "buffers.h"
42*7688df22SAndroid Build Coastguard Worker #include "cursor.h"
43*7688df22SAndroid Build Coastguard Worker
44*7688df22SAndroid Build Coastguard Worker struct cursor {
45*7688df22SAndroid Build Coastguard Worker int fd;
46*7688df22SAndroid Build Coastguard Worker uint32_t bo_handle;
47*7688df22SAndroid Build Coastguard Worker uint32_t crtc_id;
48*7688df22SAndroid Build Coastguard Worker uint32_t crtc_w, crtc_h;
49*7688df22SAndroid Build Coastguard Worker uint32_t w, h;
50*7688df22SAndroid Build Coastguard Worker
51*7688df22SAndroid Build Coastguard Worker /* current state */
52*7688df22SAndroid Build Coastguard Worker uint32_t enabled, x, y;
53*7688df22SAndroid Build Coastguard Worker int32_t dx, dy;
54*7688df22SAndroid Build Coastguard Worker };
55*7688df22SAndroid Build Coastguard Worker
56*7688df22SAndroid Build Coastguard Worker #define MAX_CURSORS 8
57*7688df22SAndroid Build Coastguard Worker static struct cursor cursors[MAX_CURSORS];
58*7688df22SAndroid Build Coastguard Worker static int ncursors;
59*7688df22SAndroid Build Coastguard Worker
60*7688df22SAndroid Build Coastguard Worker static pthread_t cursor_thread;
61*7688df22SAndroid Build Coastguard Worker static int cursor_running;
62*7688df22SAndroid Build Coastguard Worker
63*7688df22SAndroid Build Coastguard Worker /*
64*7688df22SAndroid Build Coastguard Worker * Timer driven program loops through these steps to move/enable/disable
65*7688df22SAndroid Build Coastguard Worker * the cursor
66*7688df22SAndroid Build Coastguard Worker */
67*7688df22SAndroid Build Coastguard Worker
68*7688df22SAndroid Build Coastguard Worker struct cursor_step {
69*7688df22SAndroid Build Coastguard Worker void (*run)(struct cursor *cursor, const struct cursor_step *step);
70*7688df22SAndroid Build Coastguard Worker uint32_t msec;
71*7688df22SAndroid Build Coastguard Worker uint32_t repeat;
72*7688df22SAndroid Build Coastguard Worker int arg;
73*7688df22SAndroid Build Coastguard Worker };
74*7688df22SAndroid Build Coastguard Worker
75*7688df22SAndroid Build Coastguard Worker static uint32_t indx, count;
76*7688df22SAndroid Build Coastguard Worker
set_cursor(struct cursor * cursor,const struct cursor_step * step)77*7688df22SAndroid Build Coastguard Worker static void set_cursor(struct cursor *cursor, const struct cursor_step *step)
78*7688df22SAndroid Build Coastguard Worker {
79*7688df22SAndroid Build Coastguard Worker int enabled = (step->arg ^ count) & 0x1;
80*7688df22SAndroid Build Coastguard Worker uint32_t handle = 0;
81*7688df22SAndroid Build Coastguard Worker
82*7688df22SAndroid Build Coastguard Worker if (enabled)
83*7688df22SAndroid Build Coastguard Worker handle = cursor->bo_handle;
84*7688df22SAndroid Build Coastguard Worker
85*7688df22SAndroid Build Coastguard Worker cursor->enabled = enabled;
86*7688df22SAndroid Build Coastguard Worker
87*7688df22SAndroid Build Coastguard Worker drmModeSetCursor(cursor->fd, cursor->crtc_id, handle, cursor->w, cursor->h);
88*7688df22SAndroid Build Coastguard Worker }
89*7688df22SAndroid Build Coastguard Worker
move_cursor(struct cursor * cursor,const struct cursor_step * step)90*7688df22SAndroid Build Coastguard Worker static void move_cursor(struct cursor *cursor, const struct cursor_step *step)
91*7688df22SAndroid Build Coastguard Worker {
92*7688df22SAndroid Build Coastguard Worker int x = cursor->x;
93*7688df22SAndroid Build Coastguard Worker int y = cursor->y;
94*7688df22SAndroid Build Coastguard Worker
95*7688df22SAndroid Build Coastguard Worker if (!cursor->enabled)
96*7688df22SAndroid Build Coastguard Worker drmModeSetCursor(cursor->fd, cursor->crtc_id,
97*7688df22SAndroid Build Coastguard Worker cursor->bo_handle, cursor->w, cursor->h);
98*7688df22SAndroid Build Coastguard Worker
99*7688df22SAndroid Build Coastguard Worker /* calculate new cursor position: */
100*7688df22SAndroid Build Coastguard Worker x += cursor->dx * step->arg;
101*7688df22SAndroid Build Coastguard Worker y += cursor->dy * step->arg;
102*7688df22SAndroid Build Coastguard Worker
103*7688df22SAndroid Build Coastguard Worker if (x < 0) {
104*7688df22SAndroid Build Coastguard Worker x = 0;
105*7688df22SAndroid Build Coastguard Worker cursor->dx = 1;
106*7688df22SAndroid Build Coastguard Worker } else if (x > (int)cursor->crtc_w) {
107*7688df22SAndroid Build Coastguard Worker x = cursor->crtc_w - 1;
108*7688df22SAndroid Build Coastguard Worker cursor->dx = -1;
109*7688df22SAndroid Build Coastguard Worker }
110*7688df22SAndroid Build Coastguard Worker
111*7688df22SAndroid Build Coastguard Worker if (y < 0) {
112*7688df22SAndroid Build Coastguard Worker y = 0;
113*7688df22SAndroid Build Coastguard Worker cursor->dy = 1;
114*7688df22SAndroid Build Coastguard Worker } else if (y > (int)cursor->crtc_h) {
115*7688df22SAndroid Build Coastguard Worker y = cursor->crtc_h - 1;
116*7688df22SAndroid Build Coastguard Worker cursor->dy = -1;
117*7688df22SAndroid Build Coastguard Worker }
118*7688df22SAndroid Build Coastguard Worker
119*7688df22SAndroid Build Coastguard Worker cursor->x = x;
120*7688df22SAndroid Build Coastguard Worker cursor->y = y;
121*7688df22SAndroid Build Coastguard Worker
122*7688df22SAndroid Build Coastguard Worker drmModeMoveCursor(cursor->fd, cursor->crtc_id, x, y);
123*7688df22SAndroid Build Coastguard Worker }
124*7688df22SAndroid Build Coastguard Worker
125*7688df22SAndroid Build Coastguard Worker static const struct cursor_step steps[] = {
126*7688df22SAndroid Build Coastguard Worker { set_cursor, 10, 0, 1 }, /* enable */
127*7688df22SAndroid Build Coastguard Worker { move_cursor, 1, 100, 1 },
128*7688df22SAndroid Build Coastguard Worker { move_cursor, 1, 10, 10 },
129*7688df22SAndroid Build Coastguard Worker { set_cursor, 1, 100, 0 }, /* disable/enable loop */
130*7688df22SAndroid Build Coastguard Worker { move_cursor, 1, 10, 10 },
131*7688df22SAndroid Build Coastguard Worker { move_cursor, 9, 100, 1 },
132*7688df22SAndroid Build Coastguard Worker { move_cursor, 11, 100, 5 },
133*7688df22SAndroid Build Coastguard Worker { set_cursor, 17, 10, 0 }, /* disable/enable loop */
134*7688df22SAndroid Build Coastguard Worker { move_cursor, 9, 100, 1 },
135*7688df22SAndroid Build Coastguard Worker { set_cursor, 13, 10, 0 }, /* disable/enable loop */
136*7688df22SAndroid Build Coastguard Worker { move_cursor, 9, 100, 1 },
137*7688df22SAndroid Build Coastguard Worker { set_cursor, 13, 10, 0 }, /* disable/enable loop */
138*7688df22SAndroid Build Coastguard Worker { set_cursor, 10, 0, 0 }, /* disable */
139*7688df22SAndroid Build Coastguard Worker };
140*7688df22SAndroid Build Coastguard Worker
cursor_thread_func(void * data)141*7688df22SAndroid Build Coastguard Worker static void *cursor_thread_func(void *data)
142*7688df22SAndroid Build Coastguard Worker {
143*7688df22SAndroid Build Coastguard Worker while (cursor_running) {
144*7688df22SAndroid Build Coastguard Worker const struct cursor_step *step = &steps[indx % ARRAY_SIZE(steps)];
145*7688df22SAndroid Build Coastguard Worker int i;
146*7688df22SAndroid Build Coastguard Worker
147*7688df22SAndroid Build Coastguard Worker for (i = 0; i < ncursors; i++) {
148*7688df22SAndroid Build Coastguard Worker struct cursor *cursor = &cursors[i];
149*7688df22SAndroid Build Coastguard Worker step->run(cursor, step);
150*7688df22SAndroid Build Coastguard Worker }
151*7688df22SAndroid Build Coastguard Worker
152*7688df22SAndroid Build Coastguard Worker /* iterate to next count/step: */
153*7688df22SAndroid Build Coastguard Worker if (count < step->repeat) {
154*7688df22SAndroid Build Coastguard Worker count++;
155*7688df22SAndroid Build Coastguard Worker } else {
156*7688df22SAndroid Build Coastguard Worker count = 0;
157*7688df22SAndroid Build Coastguard Worker indx++;
158*7688df22SAndroid Build Coastguard Worker }
159*7688df22SAndroid Build Coastguard Worker
160*7688df22SAndroid Build Coastguard Worker usleep(1000 * step->msec);
161*7688df22SAndroid Build Coastguard Worker }
162*7688df22SAndroid Build Coastguard Worker
163*7688df22SAndroid Build Coastguard Worker return NULL;
164*7688df22SAndroid Build Coastguard Worker }
165*7688df22SAndroid Build Coastguard Worker
cursor_init(int fd,uint32_t bo_handle,uint32_t crtc_id,uint32_t crtc_w,uint32_t crtc_h,uint32_t w,uint32_t h)166*7688df22SAndroid Build Coastguard Worker int cursor_init(int fd, uint32_t bo_handle, uint32_t crtc_id,
167*7688df22SAndroid Build Coastguard Worker uint32_t crtc_w, uint32_t crtc_h, uint32_t w, uint32_t h)
168*7688df22SAndroid Build Coastguard Worker {
169*7688df22SAndroid Build Coastguard Worker struct cursor *cursor = &cursors[ncursors];
170*7688df22SAndroid Build Coastguard Worker
171*7688df22SAndroid Build Coastguard Worker assert(ncursors < MAX_CURSORS);
172*7688df22SAndroid Build Coastguard Worker
173*7688df22SAndroid Build Coastguard Worker cursor->fd = fd;
174*7688df22SAndroid Build Coastguard Worker cursor->bo_handle = bo_handle;
175*7688df22SAndroid Build Coastguard Worker cursor->crtc_id = crtc_id;
176*7688df22SAndroid Build Coastguard Worker cursor->crtc_w = crtc_w;
177*7688df22SAndroid Build Coastguard Worker cursor->crtc_h = crtc_h;
178*7688df22SAndroid Build Coastguard Worker cursor->w = w;
179*7688df22SAndroid Build Coastguard Worker cursor->h = h;
180*7688df22SAndroid Build Coastguard Worker
181*7688df22SAndroid Build Coastguard Worker cursor->enabled = 0;
182*7688df22SAndroid Build Coastguard Worker cursor->x = w/2;
183*7688df22SAndroid Build Coastguard Worker cursor->y = h/2;
184*7688df22SAndroid Build Coastguard Worker cursor->dx = 1;
185*7688df22SAndroid Build Coastguard Worker cursor->dy = 1;
186*7688df22SAndroid Build Coastguard Worker
187*7688df22SAndroid Build Coastguard Worker ncursors++;
188*7688df22SAndroid Build Coastguard Worker
189*7688df22SAndroid Build Coastguard Worker return 0;
190*7688df22SAndroid Build Coastguard Worker }
191*7688df22SAndroid Build Coastguard Worker
cursor_start(void)192*7688df22SAndroid Build Coastguard Worker int cursor_start(void)
193*7688df22SAndroid Build Coastguard Worker {
194*7688df22SAndroid Build Coastguard Worker cursor_running = 1;
195*7688df22SAndroid Build Coastguard Worker pthread_create(&cursor_thread, NULL, cursor_thread_func, NULL);
196*7688df22SAndroid Build Coastguard Worker printf("starting cursor\n");
197*7688df22SAndroid Build Coastguard Worker return 0;
198*7688df22SAndroid Build Coastguard Worker }
199*7688df22SAndroid Build Coastguard Worker
cursor_stop(void)200*7688df22SAndroid Build Coastguard Worker int cursor_stop(void)
201*7688df22SAndroid Build Coastguard Worker {
202*7688df22SAndroid Build Coastguard Worker cursor_running = 0;
203*7688df22SAndroid Build Coastguard Worker pthread_join(cursor_thread, NULL);
204*7688df22SAndroid Build Coastguard Worker printf("cursor stopped\n");
205*7688df22SAndroid Build Coastguard Worker return 0;
206*7688df22SAndroid Build Coastguard Worker }
207