blob: 92aefdac1af4bb39933ec8022359c4f0d5a361e9 [file] [log] [blame]
Philip Withnall4f499172013-02-02 12:02:32 +00001/*
2 * Copyright © 2008-2011 Kristian Høgsberg
3 * Copyright © 2011 Intel Corporation
4 * Copyright © 2012 Raspberry Pi Foundation
5 * Copyright © 2013 Philip Withnall
6 *
7 * Permission to use, copy, modify, distribute, and sell this software and
8 * its documentation for any purpose is hereby granted without fee, provided
9 * that the above copyright notice appear in all copies and that both that
10 * copyright notice and this permission notice appear in supporting
11 * documentation, and that the name of the copyright holders not be used in
12 * advertising or publicity pertaining to distribution of the software
13 * without specific, written prior permission. The copyright holders make
14 * no representations about the suitability of this software for any
15 * purpose. It is provided "as is" without express or implied warranty.
16 *
17 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
18 * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
19 * FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
20 * SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
21 * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
22 * CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
23 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
24 */
25
Daniel Stonec228e232013-05-22 18:03:19 +030026#include "config.h"
Philip Withnall4f499172013-02-02 12:02:32 +000027
28#include <errno.h>
29#include <stdlib.h>
30#include <stdio.h>
31#include <string.h>
32#include <math.h>
33#include <sys/mman.h>
34#include <sys/types.h>
35#include <fcntl.h>
36#include <unistd.h>
37#include <linux/fb.h>
Kristian Høgsberg7e597f22013-02-18 16:35:26 -050038#include <linux/input.h>
Philip Withnall4f499172013-02-02 12:02:32 +000039
40#include <libudev.h>
41
Philip Withnall4f499172013-02-02 12:02:32 +000042#include "compositor.h"
43#include "launcher-util.h"
44#include "pixman-renderer.h"
Kristian Høgsberg7e597f22013-02-18 16:35:26 -050045#include "udev-seat.h"
Adrian Negreanu4aa756d2013-09-06 15:16:09 +030046#include "gl-renderer.h"
Philip Withnall4f499172013-02-02 12:02:32 +000047
48struct fbdev_compositor {
49 struct weston_compositor base;
50 uint32_t prev_state;
51
52 struct udev *udev;
Rob Bradfordd355b802013-05-31 18:09:55 +010053 struct udev_input input;
Adrian Negreanu4aa756d2013-09-06 15:16:09 +030054 int use_pixman;
Kristian Høgsberg61741a22013-09-17 16:02:57 -070055 struct wl_listener session_listener;
Philip Withnall4f499172013-02-02 12:02:32 +000056};
57
58struct fbdev_screeninfo {
59 unsigned int x_resolution; /* pixels, visible area */
60 unsigned int y_resolution; /* pixels, visible area */
61 unsigned int width_mm; /* visible screen width in mm */
62 unsigned int height_mm; /* visible screen height in mm */
63 unsigned int bits_per_pixel;
64
65 size_t buffer_length; /* length of frame buffer memory in bytes */
66 size_t line_length; /* length of a line in bytes */
67 char id[16]; /* screen identifier */
68
69 pixman_format_code_t pixel_format; /* frame buffer pixel format */
70 unsigned int refresh_rate; /* Hertz */
71};
72
73struct fbdev_output {
74 struct fbdev_compositor *compositor;
75 struct weston_output base;
76
77 struct weston_mode mode;
78 struct wl_event_source *finish_frame_timer;
79
80 /* Frame buffer details. */
81 const char *device; /* ownership shared with fbdev_parameters */
82 struct fbdev_screeninfo fb_info;
83 void *fb; /* length is fb_info.buffer_length */
84
85 /* pixman details. */
86 pixman_image_t *hw_surface;
87 pixman_image_t *shadow_surface;
88 void *shadow_buf;
89 uint8_t depth;
90};
91
Philip Withnall4f499172013-02-02 12:02:32 +000092struct fbdev_parameters {
93 int tty;
94 char *device;
Adrian Negreanu4aa756d2013-09-06 15:16:09 +030095 int use_gl;
Philip Withnall4f499172013-02-02 12:02:32 +000096};
97
Kristian Høgsberg7e597f22013-02-18 16:35:26 -050098static const char default_seat[] = "seat0";
99
Philip Withnall4f499172013-02-02 12:02:32 +0000100static inline struct fbdev_output *
101to_fbdev_output(struct weston_output *base)
102{
103 return container_of(base, struct fbdev_output, base);
104}
105
Philip Withnall4f499172013-02-02 12:02:32 +0000106static inline struct fbdev_compositor *
107to_fbdev_compositor(struct weston_compositor *base)
108{
109 return container_of(base, struct fbdev_compositor, base);
110}
111
112static void
Jonas Ådahle5a12252013-04-05 23:07:11 +0200113fbdev_output_start_repaint_loop(struct weston_output *output)
114{
115 uint32_t msec;
116 struct timeval tv;
117
118 gettimeofday(&tv, NULL);
119 msec = tv.tv_sec * 1000 + tv.tv_usec / 1000;
120 weston_output_finish_frame(output, msec);
121}
122
123static void
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300124fbdev_output_repaint_pixman(struct weston_output *base, pixman_region32_t *damage)
Philip Withnall4f499172013-02-02 12:02:32 +0000125{
126 struct fbdev_output *output = to_fbdev_output(base);
127 struct weston_compositor *ec = output->base.compositor;
128 pixman_box32_t *rects;
129 int nrects, i, src_x, src_y, x1, y1, x2, y2, width, height;
130
131 /* Repaint the damaged region onto the back buffer. */
132 pixman_renderer_output_set_buffer(base, output->shadow_surface);
133 ec->renderer->repaint_output(base, damage);
134
135 /* Transform and composite onto the frame buffer. */
136 width = pixman_image_get_width(output->shadow_surface);
137 height = pixman_image_get_height(output->shadow_surface);
138 rects = pixman_region32_rectangles(damage, &nrects);
139
140 for (i = 0; i < nrects; i++) {
141 switch (base->transform) {
142 default:
143 case WL_OUTPUT_TRANSFORM_NORMAL:
144 x1 = rects[i].x1;
145 x2 = rects[i].x2;
146 y1 = rects[i].y1;
147 y2 = rects[i].y2;
148 break;
149 case WL_OUTPUT_TRANSFORM_180:
150 x1 = width - rects[i].x2;
151 x2 = width - rects[i].x1;
152 y1 = height - rects[i].y2;
153 y2 = height - rects[i].y1;
154 break;
155 case WL_OUTPUT_TRANSFORM_90:
156 x1 = height - rects[i].y2;
157 x2 = height - rects[i].y1;
158 y1 = rects[i].x1;
159 y2 = rects[i].x2;
160 break;
161 case WL_OUTPUT_TRANSFORM_270:
162 x1 = rects[i].y1;
163 x2 = rects[i].y2;
164 y1 = width - rects[i].x2;
165 y2 = width - rects[i].x1;
166 break;
167 }
168 src_x = x1;
169 src_y = y1;
170
171 pixman_image_composite32(PIXMAN_OP_SRC,
172 output->shadow_surface, /* src */
173 NULL /* mask */,
174 output->hw_surface, /* dest */
175 src_x, src_y, /* src_x, src_y */
176 0, 0, /* mask_x, mask_y */
177 x1, y1, /* dest_x, dest_y */
178 x2 - x1, /* width */
179 y2 - y1 /* height */);
180 }
181
182 /* Update the damage region. */
183 pixman_region32_subtract(&ec->primary_plane.damage,
184 &ec->primary_plane.damage, damage);
185
186 /* Schedule the end of the frame. We do not sync this to the frame
187 * buffer clock because users who want that should be using the DRM
188 * compositor. FBIO_WAITFORVSYNC blocks and FB_ACTIVATE_VBL requires
189 * panning, which is broken in most kernel drivers.
190 *
191 * Finish the frame synchronised to the specified refresh rate. The
192 * refresh rate is given in mHz and the interval in ms. */
193 wl_event_source_timer_update(output->finish_frame_timer,
194 1000000 / output->mode.refresh);
195}
196
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300197static void
198fbdev_output_repaint(struct weston_output *base, pixman_region32_t *damage)
199{
200 struct fbdev_output *output = to_fbdev_output(base);
201 struct fbdev_compositor *fbc = output->compositor;
202 struct weston_compositor *ec = & fbc->base;
203
204 if (fbc->use_pixman) {
205 fbdev_output_repaint_pixman(base,damage);
206 } else {
207 ec->renderer->repaint_output(base, damage);
208 /* Update the damage region. */
209 pixman_region32_subtract(&ec->primary_plane.damage,
210 &ec->primary_plane.damage, damage);
211
212 wl_event_source_timer_update(output->finish_frame_timer,
213 1000000 / output->mode.refresh);
214 }
215}
216
Philip Withnall4f499172013-02-02 12:02:32 +0000217static int
218finish_frame_handler(void *data)
219{
220 struct fbdev_output *output = data;
Philip Withnall4f499172013-02-02 12:02:32 +0000221
Jonas Ådahle5a12252013-04-05 23:07:11 +0200222 fbdev_output_start_repaint_loop(&output->base);
Philip Withnall4f499172013-02-02 12:02:32 +0000223
224 return 1;
225}
226
227static pixman_format_code_t
228calculate_pixman_format(struct fb_var_screeninfo *vinfo,
229 struct fb_fix_screeninfo *finfo)
230{
231 /* Calculate the pixman format supported by the frame buffer from the
232 * buffer's metadata. Return 0 if no known pixman format is supported
233 * (since this has depth 0 it's guaranteed to not conflict with any
234 * actual pixman format).
235 *
236 * Documentation on the vinfo and finfo structures:
237 * http://www.mjmwired.net/kernel/Documentation/fb/api.txt
238 *
239 * TODO: Try a bit harder to support other formats, including setting
240 * the preferred format in the hardware. */
241 int type;
242
243 weston_log("Calculating pixman format from:\n"
244 STAMP_SPACE " - type: %i (aux: %i)\n"
245 STAMP_SPACE " - visual: %i\n"
246 STAMP_SPACE " - bpp: %i (grayscale: %i)\n"
247 STAMP_SPACE " - red: offset: %i, length: %i, MSB: %i\n"
248 STAMP_SPACE " - green: offset: %i, length: %i, MSB: %i\n"
249 STAMP_SPACE " - blue: offset: %i, length: %i, MSB: %i\n"
250 STAMP_SPACE " - transp: offset: %i, length: %i, MSB: %i\n",
251 finfo->type, finfo->type_aux, finfo->visual,
252 vinfo->bits_per_pixel, vinfo->grayscale,
253 vinfo->red.offset, vinfo->red.length, vinfo->red.msb_right,
254 vinfo->green.offset, vinfo->green.length,
255 vinfo->green.msb_right,
256 vinfo->blue.offset, vinfo->blue.length,
257 vinfo->blue.msb_right,
258 vinfo->transp.offset, vinfo->transp.length,
259 vinfo->transp.msb_right);
260
261 /* We only handle packed formats at the moment. */
262 if (finfo->type != FB_TYPE_PACKED_PIXELS)
263 return 0;
264
265 /* We only handle true-colour frame buffers at the moment. */
Marc Chalainffbddff2013-09-03 16:47:43 +0200266 switch(finfo->visual) {
267 case FB_VISUAL_TRUECOLOR:
268 case FB_VISUAL_DIRECTCOLOR:
269 if (vinfo->grayscale != 0)
270 return 0;
271 break;
272 default:
273 return 0;
274 }
Philip Withnall4f499172013-02-02 12:02:32 +0000275
276 /* We only support formats with MSBs on the left. */
277 if (vinfo->red.msb_right != 0 || vinfo->green.msb_right != 0 ||
278 vinfo->blue.msb_right != 0)
279 return 0;
280
281 /* Work out the format type from the offsets. We only support RGBA and
282 * ARGB at the moment. */
283 type = PIXMAN_TYPE_OTHER;
284
285 if ((vinfo->transp.offset >= vinfo->red.offset ||
286 vinfo->transp.length == 0) &&
287 vinfo->red.offset >= vinfo->green.offset &&
288 vinfo->green.offset >= vinfo->blue.offset)
289 type = PIXMAN_TYPE_ARGB;
290 else if (vinfo->red.offset >= vinfo->green.offset &&
291 vinfo->green.offset >= vinfo->blue.offset &&
292 vinfo->blue.offset >= vinfo->transp.offset)
293 type = PIXMAN_TYPE_RGBA;
294
295 if (type == PIXMAN_TYPE_OTHER)
296 return 0;
297
298 /* Build the format. */
299 return PIXMAN_FORMAT(vinfo->bits_per_pixel, type,
300 vinfo->transp.length,
301 vinfo->red.length,
302 vinfo->green.length,
303 vinfo->blue.length);
304}
305
306static int
307calculate_refresh_rate(struct fb_var_screeninfo *vinfo)
308{
309 uint64_t quot;
310
311 /* Calculate monitor refresh rate. Default is 60 Hz. Units are mHz. */
312 quot = (vinfo->upper_margin + vinfo->lower_margin + vinfo->yres);
313 quot *= (vinfo->left_margin + vinfo->right_margin + vinfo->xres);
314 quot *= vinfo->pixclock;
315
316 if (quot > 0) {
317 uint64_t refresh_rate;
318
319 refresh_rate = 1000000000000000LLU / quot;
320 if (refresh_rate > 200000)
321 refresh_rate = 200000; /* cap at 200 Hz */
322
323 return refresh_rate;
324 }
325
326 return 60 * 1000; /* default to 60 Hz */
327}
328
329static int
330fbdev_query_screen_info(struct fbdev_output *output, int fd,
331 struct fbdev_screeninfo *info)
332{
333 struct fb_var_screeninfo varinfo;
334 struct fb_fix_screeninfo fixinfo;
335
336 /* Probe the device for screen information. */
337 if (ioctl(fd, FBIOGET_FSCREENINFO, &fixinfo) < 0 ||
338 ioctl(fd, FBIOGET_VSCREENINFO, &varinfo) < 0) {
339 return -1;
340 }
341
342 /* Store the pertinent data. */
343 info->x_resolution = varinfo.xres;
344 info->y_resolution = varinfo.yres;
345 info->width_mm = varinfo.width;
346 info->height_mm = varinfo.height;
347 info->bits_per_pixel = varinfo.bits_per_pixel;
348
349 info->buffer_length = fixinfo.smem_len;
350 info->line_length = fixinfo.line_length;
351 strncpy(info->id, fixinfo.id, sizeof(info->id) / sizeof(*info->id));
352
353 info->pixel_format = calculate_pixman_format(&varinfo, &fixinfo);
354 info->refresh_rate = calculate_refresh_rate(&varinfo);
355
356 if (info->pixel_format == 0) {
357 weston_log("Frame buffer uses an unsupported format.\n");
358 return -1;
359 }
360
361 return 1;
362}
363
364static int
365fbdev_set_screen_info(struct fbdev_output *output, int fd,
366 struct fbdev_screeninfo *info)
367{
368 struct fb_var_screeninfo varinfo;
369
370 /* Grab the current screen information. */
371 if (ioctl(fd, FBIOGET_VSCREENINFO, &varinfo) < 0) {
372 return -1;
373 }
374
375 /* Update the information. */
376 varinfo.xres = info->x_resolution;
377 varinfo.yres = info->y_resolution;
378 varinfo.width = info->width_mm;
379 varinfo.height = info->height_mm;
380 varinfo.bits_per_pixel = info->bits_per_pixel;
381
382 /* Try to set up an ARGB (x8r8g8b8) pixel format. */
383 varinfo.grayscale = 0;
384 varinfo.transp.offset = 24;
385 varinfo.transp.length = 0;
386 varinfo.transp.msb_right = 0;
387 varinfo.red.offset = 16;
388 varinfo.red.length = 8;
389 varinfo.red.msb_right = 0;
390 varinfo.green.offset = 8;
391 varinfo.green.length = 8;
392 varinfo.green.msb_right = 0;
393 varinfo.blue.offset = 0;
394 varinfo.blue.length = 8;
395 varinfo.blue.msb_right = 0;
396
397 /* Set the device's screen information. */
398 if (ioctl(fd, FBIOPUT_VSCREENINFO, &varinfo) < 0) {
399 return -1;
400 }
401
402 return 1;
403}
404
405static void fbdev_frame_buffer_destroy(struct fbdev_output *output);
406
407/* Returns an FD for the frame buffer device. */
408static int
409fbdev_frame_buffer_open(struct fbdev_output *output, const char *fb_dev,
410 struct fbdev_screeninfo *screen_info)
411{
412 int fd = -1;
413
414 weston_log("Opening fbdev frame buffer.\n");
415
416 /* Open the frame buffer device. */
417 fd = open(fb_dev, O_RDWR | O_CLOEXEC);
418 if (fd < 0) {
419 weston_log("Failed to open frame buffer device ‘%s’: %s\n",
420 fb_dev, strerror(errno));
421 return -1;
422 }
423
424 /* Grab the screen info. */
425 if (fbdev_query_screen_info(output, fd, screen_info) < 0) {
426 weston_log("Failed to get frame buffer info: %s\n",
427 strerror(errno));
428
429 close(fd);
430 return -1;
431 }
432
433 return fd;
434}
435
436/* Closes the FD on success or failure. */
437static int
438fbdev_frame_buffer_map(struct fbdev_output *output, int fd)
439{
440 int retval = -1;
441
442 weston_log("Mapping fbdev frame buffer.\n");
443
444 /* Map the frame buffer. Write-only mode, since we don't want to read
445 * anything back (because it's slow). */
446 output->fb = mmap(NULL, output->fb_info.buffer_length,
447 PROT_WRITE, MAP_SHARED, fd, 0);
448 if (output->fb == MAP_FAILED) {
449 weston_log("Failed to mmap frame buffer: %s\n",
450 strerror(errno));
451 goto out_close;
452 }
453
454 /* Create a pixman image to wrap the memory mapped frame buffer. */
455 output->hw_surface =
456 pixman_image_create_bits(output->fb_info.pixel_format,
457 output->fb_info.x_resolution,
458 output->fb_info.y_resolution,
459 output->fb,
460 output->fb_info.line_length);
461 if (output->hw_surface == NULL) {
462 weston_log("Failed to create surface for frame buffer.\n");
463 goto out_unmap;
464 }
465
466 /* Success! */
467 retval = 0;
468
469out_unmap:
470 if (retval != 0 && output->fb != NULL)
471 fbdev_frame_buffer_destroy(output);
472
473out_close:
474 if (fd >= 0)
475 close(fd);
476
477 return retval;
478}
479
480static void
481fbdev_frame_buffer_destroy(struct fbdev_output *output)
482{
483 weston_log("Destroying fbdev frame buffer.\n");
484
485 if (munmap(output->fb, output->fb_info.buffer_length) < 0)
486 weston_log("Failed to munmap frame buffer: %s\n",
487 strerror(errno));
488
489 output->fb = NULL;
490}
491
492static void fbdev_output_destroy(struct weston_output *base);
493static void fbdev_output_disable(struct weston_output *base);
494
495static int
496fbdev_output_create(struct fbdev_compositor *compositor,
497 const char *device)
498{
499 struct fbdev_output *output;
500 pixman_transform_t transform;
501 int fb_fd;
502 int shadow_width, shadow_height;
503 int width, height;
504 unsigned int bytes_per_pixel;
505 struct wl_event_loop *loop;
506
507 weston_log("Creating fbdev output.\n");
508
509 output = calloc(1, sizeof *output);
510 if (!output)
511 return -1;
512
513 output->compositor = compositor;
514 output->device = device;
515
516 /* Create the frame buffer. */
517 fb_fd = fbdev_frame_buffer_open(output, device, &output->fb_info);
518 if (fb_fd < 0) {
519 weston_log("Creating frame buffer failed.\n");
520 goto out_free;
521 }
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300522 if (compositor->use_pixman) {
523 if (fbdev_frame_buffer_map(output, fb_fd) < 0) {
524 weston_log("Mapping frame buffer failed.\n");
525 goto out_free;
526 }
Philip Withnall4f499172013-02-02 12:02:32 +0000527 }
528
Jonas Ådahle5a12252013-04-05 23:07:11 +0200529 output->base.start_repaint_loop = fbdev_output_start_repaint_loop;
Philip Withnall4f499172013-02-02 12:02:32 +0000530 output->base.repaint = fbdev_output_repaint;
531 output->base.destroy = fbdev_output_destroy;
532 output->base.assign_planes = NULL;
533 output->base.set_backlight = NULL;
534 output->base.set_dpms = NULL;
535 output->base.switch_mode = NULL;
536
537 /* only one static mode in list */
538 output->mode.flags =
539 WL_OUTPUT_MODE_CURRENT | WL_OUTPUT_MODE_PREFERRED;
540 output->mode.width = output->fb_info.x_resolution;
541 output->mode.height = output->fb_info.y_resolution;
542 output->mode.refresh = output->fb_info.refresh_rate;
543 wl_list_init(&output->base.mode_list);
544 wl_list_insert(&output->base.mode_list, &output->mode.link);
545
Hardeningff39efa2013-09-18 23:56:35 +0200546 output->base.current_mode = &output->mode;
Philip Withnall4f499172013-02-02 12:02:32 +0000547 output->base.subpixel = WL_OUTPUT_SUBPIXEL_UNKNOWN;
548 output->base.make = "unknown";
549 output->base.model = output->fb_info.id;
550
551 weston_output_init(&output->base, &compositor->base,
552 0, 0, output->fb_info.width_mm,
553 output->fb_info.height_mm,
Alexander Larsson4ea95522013-05-22 14:41:37 +0200554 WL_OUTPUT_TRANSFORM_NORMAL,
555 1);
Philip Withnall4f499172013-02-02 12:02:32 +0000556
557 width = output->fb_info.x_resolution;
558 height = output->fb_info.y_resolution;
559
560 pixman_transform_init_identity(&transform);
561 switch (output->base.transform) {
562 default:
563 case WL_OUTPUT_TRANSFORM_NORMAL:
564 shadow_width = width;
565 shadow_height = height;
566 pixman_transform_rotate(&transform,
567 NULL, 0, 0);
568 pixman_transform_translate(&transform, NULL,
569 0, 0);
570 break;
571 case WL_OUTPUT_TRANSFORM_180:
572 shadow_width = width;
573 shadow_height = height;
574 pixman_transform_rotate(&transform,
575 NULL, -pixman_fixed_1, 0);
576 pixman_transform_translate(NULL, &transform,
577 pixman_int_to_fixed(shadow_width),
578 pixman_int_to_fixed(shadow_height));
579 break;
580 case WL_OUTPUT_TRANSFORM_270:
581 shadow_width = height;
582 shadow_height = width;
583 pixman_transform_rotate(&transform,
584 NULL, 0, pixman_fixed_1);
585 pixman_transform_translate(&transform,
586 NULL,
587 pixman_int_to_fixed(shadow_width),
588 0);
589 break;
590 case WL_OUTPUT_TRANSFORM_90:
591 shadow_width = height;
592 shadow_height = width;
593 pixman_transform_rotate(&transform,
594 NULL, 0, -pixman_fixed_1);
595 pixman_transform_translate(&transform,
596 NULL,
597 0,
598 pixman_int_to_fixed(shadow_height));
599 break;
600 }
601
602 bytes_per_pixel = output->fb_info.bits_per_pixel / 8;
603
604 output->shadow_buf = malloc(width * height * bytes_per_pixel);
605 output->shadow_surface =
606 pixman_image_create_bits(output->fb_info.pixel_format,
607 shadow_width, shadow_height,
608 output->shadow_buf,
609 shadow_width * bytes_per_pixel);
610 if (output->shadow_buf == NULL || output->shadow_surface == NULL) {
611 weston_log("Failed to create surface for frame buffer.\n");
612 goto out_hw_surface;
613 }
614
615 /* No need in transform for normal output */
616 if (output->base.transform != WL_OUTPUT_TRANSFORM_NORMAL)
617 pixman_image_set_transform(output->shadow_surface, &transform);
618
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300619 if (compositor->use_pixman) {
620 if (pixman_renderer_output_create(&output->base) < 0)
621 goto out_shadow_surface;
622 } else {
623 setenv("HYBRIS_EGLPLATFORM", "wayland", 1);
624 if (gl_renderer_output_create(&output->base,
625 (EGLNativeWindowType)NULL) < 0) {
626 weston_log("gl_renderer_output_create failed.\n");
627 goto out_shadow_surface;
628 }
629 }
630
Philip Withnall4f499172013-02-02 12:02:32 +0000631
632 loop = wl_display_get_event_loop(compositor->base.wl_display);
633 output->finish_frame_timer =
634 wl_event_loop_add_timer(loop, finish_frame_handler, output);
635
636 wl_list_insert(compositor->base.output_list.prev, &output->base.link);
637
638 weston_log("fbdev output %d×%d px\n",
639 output->mode.width, output->mode.height);
640 weston_log_continue(STAMP_SPACE "guessing %d Hz and 96 dpi\n",
641 output->mode.refresh / 1000);
642
643 return 0;
644
645out_shadow_surface:
646 pixman_image_unref(output->shadow_surface);
647 output->shadow_surface = NULL;
648out_hw_surface:
649 free(output->shadow_buf);
650 pixman_image_unref(output->hw_surface);
651 output->hw_surface = NULL;
652 weston_output_destroy(&output->base);
653 fbdev_frame_buffer_destroy(output);
654out_free:
655 free(output);
656
657 return -1;
658}
659
660static void
661fbdev_output_destroy(struct weston_output *base)
662{
663 struct fbdev_output *output = to_fbdev_output(base);
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300664 struct fbdev_compositor *compositor = output->compositor;
Philip Withnall4f499172013-02-02 12:02:32 +0000665
666 weston_log("Destroying fbdev output.\n");
667
668 /* Close the frame buffer. */
669 fbdev_output_disable(base);
670
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300671 if (compositor->use_pixman) {
672 if (base->renderer_state != NULL)
673 pixman_renderer_output_destroy(base);
Philip Withnall4f499172013-02-02 12:02:32 +0000674
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300675 if (output->shadow_surface != NULL) {
676 pixman_image_unref(output->shadow_surface);
677 output->shadow_surface = NULL;
678 }
Philip Withnall4f499172013-02-02 12:02:32 +0000679
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300680 if (output->shadow_buf != NULL) {
681 free(output->shadow_buf);
682 output->shadow_buf = NULL;
683 }
684 } else {
685 gl_renderer_output_destroy(base);
Philip Withnall4f499172013-02-02 12:02:32 +0000686 }
687
688 /* Remove the output. */
689 wl_list_remove(&output->base.link);
690 weston_output_destroy(&output->base);
691
692 free(output);
693}
694
695/* strcmp()-style return values. */
696static int
697compare_screen_info (const struct fbdev_screeninfo *a,
698 const struct fbdev_screeninfo *b)
699{
700 if (a->x_resolution == b->x_resolution &&
701 a->y_resolution == b->y_resolution &&
702 a->width_mm == b->width_mm &&
703 a->height_mm == b->height_mm &&
704 a->bits_per_pixel == b->bits_per_pixel &&
705 a->pixel_format == b->pixel_format &&
706 a->refresh_rate == b->refresh_rate)
707 return 0;
708
709 return 1;
710}
711
712static int
713fbdev_output_reenable(struct fbdev_compositor *compositor,
714 struct weston_output *base)
715{
716 struct fbdev_output *output = to_fbdev_output(base);
717 struct fbdev_screeninfo new_screen_info;
718 int fb_fd;
Rob Bradfordf8ef42f2013-07-26 16:29:38 +0100719 const char *device;
Philip Withnall4f499172013-02-02 12:02:32 +0000720
721 weston_log("Re-enabling fbdev output.\n");
722
723 /* Create the frame buffer. */
724 fb_fd = fbdev_frame_buffer_open(output, output->device,
725 &new_screen_info);
726 if (fb_fd < 0) {
727 weston_log("Creating frame buffer failed.\n");
728 goto err;
729 }
730
731 /* Check whether the frame buffer details have changed since we were
732 * disabled. */
733 if (compare_screen_info (&output->fb_info, &new_screen_info) != 0) {
734 /* Perform a mode-set to restore the old mode. */
735 if (fbdev_set_screen_info(output, fb_fd,
736 &output->fb_info) < 0) {
737 weston_log("Failed to restore mode settings. "
738 "Attempting to re-open output anyway.\n");
739 }
740
Rob Bradford581b3fd2013-07-26 16:29:39 +0100741 close(fb_fd);
742
Philip Withnall4f499172013-02-02 12:02:32 +0000743 /* Remove and re-add the output so that resources depending on
744 * the frame buffer X/Y resolution (such as the shadow buffer)
745 * are re-initialised. */
Rob Bradfordf8ef42f2013-07-26 16:29:38 +0100746 device = output->device;
Philip Withnall4f499172013-02-02 12:02:32 +0000747 fbdev_output_destroy(base);
Rob Bradfordf8ef42f2013-07-26 16:29:38 +0100748 fbdev_output_create(compositor, device);
Philip Withnall4f499172013-02-02 12:02:32 +0000749
750 return 0;
751 }
752
753 /* Map the device if it has the same details as before. */
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300754 if (compositor->use_pixman) {
755 if (fbdev_frame_buffer_map(output, fb_fd) < 0) {
756 weston_log("Mapping frame buffer failed.\n");
757 goto err;
758 }
Philip Withnall4f499172013-02-02 12:02:32 +0000759 }
760
761 return 0;
762
763err:
764 return -1;
765}
766
767/* NOTE: This leaves output->fb_info populated, caching data so that if
768 * fbdev_output_reenable() is called again, it can determine whether a mode-set
769 * is needed. */
770static void
771fbdev_output_disable(struct weston_output *base)
772{
773 struct fbdev_output *output = to_fbdev_output(base);
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300774 struct fbdev_compositor *compositor = output->compositor;
Philip Withnall4f499172013-02-02 12:02:32 +0000775
776 weston_log("Disabling fbdev output.\n");
777
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300778 if ( ! compositor->use_pixman) return;
779
Philip Withnall4f499172013-02-02 12:02:32 +0000780 if (output->hw_surface != NULL) {
781 pixman_image_unref(output->hw_surface);
782 output->hw_surface = NULL;
783 }
784
785 fbdev_frame_buffer_destroy(output);
786}
787
788static void
Philip Withnall4f499172013-02-02 12:02:32 +0000789fbdev_compositor_destroy(struct weston_compositor *base)
790{
791 struct fbdev_compositor *compositor = to_fbdev_compositor(base);
Philip Withnall4f499172013-02-02 12:02:32 +0000792
Rob Bradfordd355b802013-05-31 18:09:55 +0100793 udev_input_destroy(&compositor->input);
Philip Withnall4f499172013-02-02 12:02:32 +0000794
795 /* Destroy the output. */
796 weston_compositor_shutdown(&compositor->base);
797
798 /* Chain up. */
799 compositor->base.renderer->destroy(&compositor->base);
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700800 weston_launcher_destroy(compositor->base.launcher);
Philip Withnall4f499172013-02-02 12:02:32 +0000801
802 free(compositor);
803}
804
805static void
Kristian Høgsberg61741a22013-09-17 16:02:57 -0700806session_notify(struct wl_listener *listener, void *data)
Philip Withnall4f499172013-02-02 12:02:32 +0000807{
Kristian Høgsberg61741a22013-09-17 16:02:57 -0700808 struct fbdev_compositor *compositor = data;
Philip Withnall4f499172013-02-02 12:02:32 +0000809 struct weston_output *output;
810
Kristian Høgsberg61741a22013-09-17 16:02:57 -0700811 if (compositor->base.session_active) {
Philip Withnall4f499172013-02-02 12:02:32 +0000812 weston_log("entering VT\n");
813 compositor->base.focus = 1;
814 compositor->base.state = compositor->prev_state;
815
816 wl_list_for_each(output, &compositor->base.output_list, link) {
817 fbdev_output_reenable(compositor, output);
818 }
819
820 weston_compositor_damage_all(&compositor->base);
821
Rob Bradfordd355b802013-05-31 18:09:55 +0100822 udev_input_enable(&compositor->input, compositor->udev);
Kristian Høgsberg61741a22013-09-17 16:02:57 -0700823 } else {
Philip Withnall4f499172013-02-02 12:02:32 +0000824 weston_log("leaving VT\n");
Rob Bradfordd355b802013-05-31 18:09:55 +0100825 udev_input_disable(&compositor->input);
Philip Withnall4f499172013-02-02 12:02:32 +0000826
827 wl_list_for_each(output, &compositor->base.output_list, link) {
828 fbdev_output_disable(output);
829 }
830
831 compositor->base.focus = 0;
832 compositor->prev_state = compositor->base.state;
Philipp Brüschweiler57edf7f2013-03-29 13:01:56 +0100833 weston_compositor_offscreen(&compositor->base);
Philip Withnall4f499172013-02-02 12:02:32 +0000834
835 /* If we have a repaint scheduled (from the idle handler), make
836 * sure we cancel that so we don't try to pageflip when we're
Philipp Brüschweiler57edf7f2013-03-29 13:01:56 +0100837 * vt switched away. The OFFSCREEN state will prevent
Philip Withnall4f499172013-02-02 12:02:32 +0000838 * further attemps at repainting. When we switch
839 * back, we schedule a repaint, which will process
840 * pending frame callbacks. */
841
842 wl_list_for_each(output,
843 &compositor->base.output_list, link) {
844 output->repaint_needed = 0;
845 }
Philip Withnall4f499172013-02-02 12:02:32 +0000846 };
847}
848
849static void
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700850fbdev_restore(struct weston_compositor *compositor)
Philip Withnall4f499172013-02-02 12:02:32 +0000851{
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700852 weston_launcher_restore(compositor->launcher);
Philip Withnall4f499172013-02-02 12:02:32 +0000853}
854
855static void
Kristian Høgsberge3148752013-05-06 23:19:49 -0400856switch_vt_binding(struct weston_seat *seat, uint32_t time, uint32_t key, void *data)
Philip Withnall4f499172013-02-02 12:02:32 +0000857{
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700858 struct weston_compositor *compositor = data;
Philip Withnall4f499172013-02-02 12:02:32 +0000859
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700860 weston_launcher_activate_vt(compositor->launcher, key - KEY_F1 + 1);
Philip Withnall4f499172013-02-02 12:02:32 +0000861}
862
863static struct weston_compositor *
Kristian Høgsberg4172f662013-02-20 15:27:49 -0500864fbdev_compositor_create(struct wl_display *display, int *argc, char *argv[],
Kristian Høgsberg14e438c2013-05-26 21:48:14 -0400865 struct weston_config *config,
866 struct fbdev_parameters *param)
Philip Withnall4f499172013-02-02 12:02:32 +0000867{
868 struct fbdev_compositor *compositor;
Rob Bradford2387fde2013-05-31 18:09:57 +0100869 const char *seat_id = default_seat;
Philip Withnall4f499172013-02-02 12:02:32 +0000870 uint32_t key;
871
872 weston_log("initializing fbdev backend\n");
873
874 compositor = calloc(1, sizeof *compositor);
875 if (compositor == NULL)
876 return NULL;
877
878 if (weston_compositor_init(&compositor->base, display, argc, argv,
Kristian Høgsberg14e438c2013-05-26 21:48:14 -0400879 config) < 0)
Philip Withnall4f499172013-02-02 12:02:32 +0000880 goto out_free;
881
Adrian Negreanuc8384232013-07-28 18:27:23 +0300882 /* Check if we run fbdev-backend using weston-launch */
Kristian Høgsberg6ff3ff52013-10-02 10:53:33 -0700883 compositor->base.launcher =
884 weston_launcher_connect(&compositor->base, param->tty);
Kristian Høgsberg05ad1e42013-09-17 14:41:03 -0700885 if (compositor->base.launcher == NULL && geteuid() != 0) {
Adrian Negreanuc8384232013-07-28 18:27:23 +0300886 weston_log("fatal: fbdev backend should be run "
887 "using weston-launch binary or as root\n");
888 goto out_compositor;
889 }
890
Philip Withnall4f499172013-02-02 12:02:32 +0000891 compositor->udev = udev_new();
892 if (compositor->udev == NULL) {
893 weston_log("Failed to initialize udev context.\n");
894 goto out_compositor;
895 }
896
897 /* Set up the TTY. */
Kristian Høgsberg61741a22013-09-17 16:02:57 -0700898 compositor->session_listener.notify = session_notify;
899 wl_signal_add(&compositor->base.session_signal,
900 &compositor->session_listener);
Kristian Høgsberg6ff3ff52013-10-02 10:53:33 -0700901 compositor->base.launcher =
902 weston_launcher_connect(&compositor->base, param->tty);
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700903 if (!compositor->base.launcher) {
904 weston_log("Failed to set up launcher.\n");
Philip Withnall4f499172013-02-02 12:02:32 +0000905 goto out_udev;
906 }
907
908 compositor->base.destroy = fbdev_compositor_destroy;
909 compositor->base.restore = fbdev_restore;
910
911 compositor->base.focus = 1;
912 compositor->prev_state = WESTON_COMPOSITOR_ACTIVE;
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300913 compositor->use_pixman = !param->use_gl;
Philip Withnall4f499172013-02-02 12:02:32 +0000914
915 for (key = KEY_F1; key < KEY_F9; key++)
916 weston_compositor_add_key_binding(&compositor->base, key,
917 MODIFIER_CTRL | MODIFIER_ALT,
918 switch_vt_binding,
919 compositor);
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300920 if (compositor->use_pixman) {
921 if (pixman_renderer_init(&compositor->base) < 0)
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700922 goto out_launcher;
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300923 } else {
924 if (gl_renderer_create(&compositor->base, EGL_DEFAULT_DISPLAY,
925 gl_renderer_opaque_attribs, NULL) < 0) {
926 weston_log("gl_renderer_create failed.\n");
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700927 goto out_launcher;
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300928 }
929 }
Philip Withnall4f499172013-02-02 12:02:32 +0000930
931 if (fbdev_output_create(compositor, param->device) < 0)
932 goto out_pixman;
933
Rob Bradford2387fde2013-05-31 18:09:57 +0100934 udev_input_init(&compositor->input, &compositor->base, compositor->udev, seat_id);
Philip Withnall4f499172013-02-02 12:02:32 +0000935
936 return &compositor->base;
937
938out_pixman:
939 compositor->base.renderer->destroy(&compositor->base);
940
Kristian Høgsberg3f495872013-09-18 23:00:17 -0700941out_launcher:
942 weston_launcher_destroy(compositor->base.launcher);
Philip Withnall4f499172013-02-02 12:02:32 +0000943
944out_udev:
945 udev_unref(compositor->udev);
946
947out_compositor:
948 weston_compositor_shutdown(&compositor->base);
949
950out_free:
951 free(compositor);
952
953 return NULL;
954}
955
956WL_EXPORT struct weston_compositor *
Kristian Høgsberg4172f662013-02-20 15:27:49 -0500957backend_init(struct wl_display *display, int *argc, char *argv[],
Kristian Høgsberg14e438c2013-05-26 21:48:14 -0400958 struct weston_config *config)
Philip Withnall4f499172013-02-02 12:02:32 +0000959{
960 /* TODO: Ideally, available frame buffers should be enumerated using
961 * udev, rather than passing a device node in as a parameter. */
962 struct fbdev_parameters param = {
963 .tty = 0, /* default to current tty */
964 .device = "/dev/fb0", /* default frame buffer */
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300965 .use_gl = 0,
Philip Withnall4f499172013-02-02 12:02:32 +0000966 };
967
968 const struct weston_option fbdev_options[] = {
969 { WESTON_OPTION_INTEGER, "tty", 0, &param.tty },
970 { WESTON_OPTION_STRING, "device", 0, &param.device },
Adrian Negreanu4aa756d2013-09-06 15:16:09 +0300971 { WESTON_OPTION_BOOLEAN, "use-gl", 0, &param.use_gl },
Philip Withnall4f499172013-02-02 12:02:32 +0000972 };
973
974 parse_options(fbdev_options, ARRAY_LENGTH(fbdev_options), argc, argv);
975
Kristian Høgsberg14e438c2013-05-26 21:48:14 -0400976 return fbdev_compositor_create(display, argc, argv, config, &param);
Philip Withnall4f499172013-02-02 12:02:32 +0000977}