From 19ad6a9db3ab699cdb7cee3bd0232ccb6d0e1e0e Mon Sep 17 00:00:00 2001 From: Ray Strode Date: Fri, 19 Dec 2008 01:45:41 -0500 Subject: [PATCH] Restore framebuffer when switching back to VT The kernel currently automatically redirects output to the kernel framebuffer when switching VTs away from wayland. It doesn't restore output back to wayland's fb when coming back to the VT. This patch works around that issue. --- egl-compositor.c | 103 ++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 98 insertions(+), 5 deletions(-) diff --git a/egl-compositor.c b/egl-compositor.c index a0194541..33081a33 100644 --- a/egl-compositor.c +++ b/egl-compositor.c @@ -38,6 +38,7 @@ #include #include #include +#include #include #include #include @@ -76,8 +77,10 @@ struct egl_compositor { EGLSurface surface; EGLContext context; EGLConfig config; + uint32_t fb_id; struct wl_display *wl_display; int gem_fd; + int tty_fd; int width, height; struct egl_surface *background; struct egl_surface *overlay; @@ -86,6 +89,9 @@ struct egl_compositor { struct wl_list input_device_list; struct wl_list surface_list; + struct wl_event_source *enter_vt_source; + struct wl_event_source *leave_vt_source; + /* Repaint state. */ struct wl_event_source *timer_source; int repaint_needed; @@ -860,7 +866,7 @@ egl_device_get_position(struct egl_input_device *device, int32_t *x, int32_t *y) } static uint32_t -create_frontbuffer(int fd, int *width, int *height, int *stride) +create_frontbuffer(int fd, int *width, int *height, int *stride, uint32_t *fb_id) { drmModeConnector *connector; drmModeRes *resources; @@ -868,7 +874,6 @@ create_frontbuffer(int fd, int *width, int *height, int *stride) struct drm_mode_modeinfo *mode; struct drm_i915_gem_create create; struct drm_gem_flink flink; - unsigned int fb_id; int i, ret; resources = drmModeGetResources(fd); @@ -916,13 +921,13 @@ create_frontbuffer(int fd, int *width, int *height, int *stride) } ret = drmModeAddFB(fd, mode->hdisplay, mode->vdisplay, - 32, 32, mode->hdisplay * 4, create.handle, &fb_id); + 32, 32, mode->hdisplay * 4, create.handle, fb_id); if (ret) { fprintf(stderr, "failed to add fb: %m\n"); return 0; } - ret = drmModeSetCrtc(fd, encoder->crtc_id, fb_id, 0, 0, + ret = drmModeSetCrtc(fd, encoder->crtc_id, *fb_id, 0, 0, &connector->connector_id, 1, mode); if (ret) { fprintf(stderr, "failed to set mode: %m\n"); @@ -1038,6 +1043,93 @@ static const GOptionEntry option_entries[] = { { NULL } }; +static void on_enter_vt(int signal_number, void *data) +{ + struct egl_compositor *ec = data; + + drmModeConnector *connector; + drmModeRes *resources; + drmModeEncoder *encoder; + struct drm_mode_modeinfo *mode; + int i, ret; + int fd; + + fd = ec->gem_fd; + resources = drmModeGetResources(fd); + if (!resources) { + fprintf(stderr, "drmModeGetResources failed\n"); + return; + } + + for (i = 0; i < resources->count_connectors; i++) { + connector = drmModeGetConnector(fd, resources->connectors[i]); + if (connector == NULL) + continue; + + if (connector->connection == DRM_MODE_CONNECTED && + connector->count_modes > 0) + break; + + drmModeFreeConnector(connector); + } + + if (i == resources->count_connectors) { + fprintf(stderr, "No currently active connector found.\n"); + return; + } + + mode = &connector->modes[0]; + + for (i = 0; i < resources->count_encoders; i++) { + encoder = drmModeGetEncoder(fd, resources->encoders[i]); + + if (encoder == NULL) + continue; + + if (encoder->encoder_id == connector->encoder_id) + break; + + drmModeFreeEncoder(encoder); + } + + ret = drmModeSetCrtc(fd, encoder->crtc_id, ec->fb_id, 0, 0, + &connector->connector_id, 1, mode); + if (ret) { + fprintf(stderr, "failed to set mode: %m\n"); + return; + } + +} + +static void on_leave_vt(int signal_number, void *data) +{ + struct egl_compositor *ec = data; + + ioctl (ec->tty_fd, VT_RELDISP, 1); +} + +static void watch_for_vt_changes(struct egl_compositor *ec, struct wl_event_loop *loop) +{ + int fd; + struct vt_mode mode = { 0 }; + + ec->tty_fd = open("/dev/tty0", O_RDWR | O_NOCTTY); + mode.mode = VT_PROCESS; + mode.relsig = SIGUSR1; + mode.acqsig = SIGUSR2; + + if (!ioctl (fd, VT_SETMODE, &mode) < 0) { + fprintf(stderr, "failed to take control of vt handling\n"); + } + + ec->leave_vt_source = wl_event_loop_add_signal (loop, SIGUSR1, + on_leave_vt, + ec); + ec->enter_vt_source = wl_event_loop_add_signal (loop, SIGUSR2, + on_enter_vt, + ec); +} + static struct egl_compositor * egl_compositor_create(struct wl_display *display) { @@ -1071,7 +1163,7 @@ egl_compositor_create(struct wl_display *display) return NULL; fb_name = create_frontbuffer(eglGetDisplayFD(ec->display), - &ec->width, &ec->height, &stride); + &ec->width, &ec->height, &stride, &ec->fb_id); ec->surface = eglCreateSurfaceForName(ec->display, ec->config, fb_name, ec->width, ec->height, stride, attribs); if (ec->surface == NULL) { @@ -1123,6 +1215,7 @@ egl_compositor_create(struct wl_display *display) wl_display_add_global(display, &shooter->base); loop = wl_display_get_event_loop(ec->wl_display); + watch_for_vt_changes (ec, loop); ec->timer_source = wl_event_loop_add_timer(loop, repaint, ec); ec->repaint_needed = 0; ec->repaint_on_timeout = 0;