/* plugin.c - drm backend renderer plugin * * Copyright (C) 2006-2009 Red Hat, Inc. * 2008 Charlie Brej * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA * 02111-1307, USA. * * Written by: Charlie Brej * Kristian Høgsberg * Peter Jones * Ray Strode */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ply-buffer.h" #include "ply-event-loop.h" #include "ply-list.h" #include "ply-logger.h" #include "ply-rectangle.h" #include "ply-region.h" #include "ply-terminal.h" #include "ply-renderer.h" #include "ply-renderer-plugin.h" #include "ply-renderer-driver.h" #ifdef PLY_ENABLE_LIBDRM_INTEL #include "ply-renderer-i915-driver.h" #endif #ifdef PLY_ENABLE_LIBDRM_RADEON #include "ply-renderer-radeon-driver.h" #endif #ifdef PLY_ENABLE_LIBDRM_NOUVEAU #include "ply-renderer-nouveau-driver.h" #endif #define BYTES_PER_PIXEL (4) #ifndef PLY_MAX_COMMAND_LINE_SIZE #define PLY_MAX_COMMAND_LINE_SIZE 512 #endif struct _ply_renderer_head { ply_renderer_backend_t *backend; ply_pixel_buffer_t *pixel_buffer; ply_rectangle_t area; unsigned long row_stride; drmModeConnector *connector; drmModeModeInfo *mode; uint32_t controller_id; uint32_t encoder_id; uint32_t console_buffer_id; uint32_t scan_out_buffer_id; }; struct _ply_renderer_input_source { ply_renderer_backend_t *backend; ply_fd_watch_t *terminal_input_watch; ply_buffer_t *key_buffer; ply_renderer_input_source_handler_t handler; void *user_data; }; struct _ply_renderer_backend { ply_event_loop_t *loop; ply_terminal_t *terminal; ply_renderer_driver_interface_t *driver_interface; ply_renderer_driver_t *driver; uint32_t driver_supports_mapping_console; int device_fd; char *device_name; drmModeRes *resources; ply_renderer_input_source_t input_source; ply_list_t *heads; int32_t dither_red; int32_t dither_green; int32_t dither_blue; uint32_t is_active : 1; }; ply_renderer_plugin_interface_t *ply_renderer_backend_get_interface (void); static void ply_renderer_head_redraw (ply_renderer_backend_t *backend, ply_renderer_head_t *head); static bool open_input_source (ply_renderer_backend_t *backend, ply_renderer_input_source_t *input_source); static ply_renderer_head_t * ply_renderer_head_new (ply_renderer_backend_t *backend, drmModeConnector *connector, uint32_t encoder_id, uint32_t controller_id, uint32_t console_buffer_id, drmModeModeInfo *mode) { ply_renderer_head_t *head; head = calloc (1, sizeof (ply_renderer_head_t)); head->backend = backend; head->connector = connector; head->encoder_id = encoder_id; head->controller_id = controller_id; head->console_buffer_id = console_buffer_id; head->mode = mode; head->area.x = 0; head->area.y = 0; head->area.width = mode->hdisplay; head->area.height = mode->vdisplay; head->pixel_buffer = ply_pixel_buffer_new (head->area.width, head->area.height); ply_trace ("Creating %ldx%ld renderer head", head->area.width, head->area.height); ply_pixel_buffer_fill_with_color (head->pixel_buffer, NULL, 0.0, 0.0, 0.0, 1.0); return head; } static void ply_renderer_head_free (ply_renderer_head_t *head) { ply_trace ("freeing %ldx%ld renderer head", head->area.width, head->area.height); ply_pixel_buffer_free (head->pixel_buffer); drmModeFreeConnector (head->connector); free (head); } static bool ply_renderer_head_set_scan_out_buffer (ply_renderer_backend_t *backend, ply_renderer_head_t *head, uint32_t buffer_id) { /* Tell the controller to use the allocated scan out buffer */ if (drmModeSetCrtc (backend->device_fd, head->controller_id, buffer_id, 0, 0, &head->connector->connector_id, 1, head->mode) < 0) return false; return true; } static bool ply_renderer_head_map (ply_renderer_backend_t *backend, ply_renderer_head_t *head) { assert (backend != NULL); assert (backend->device_fd >= 0); assert (backend->driver_interface != NULL); assert (backend->driver != NULL); assert (head != NULL); ply_trace ("Creating buffer for %ldx%ld renderer head", head->area.width, head->area.height); head->scan_out_buffer_id = backend->driver_interface->create_buffer (backend->driver, head->area.width, head->area.height, &head->row_stride); if (head->scan_out_buffer_id == 0) return false; ply_trace ("Mapping buffer for %ldx%ld renderer head", head->area.width, head->area.height); if (!backend->driver_interface->map_buffer (backend->driver, head->scan_out_buffer_id)) { backend->driver_interface->destroy_buffer (backend->driver, head->scan_out_buffer_id); head->scan_out_buffer_id = 0; return false; } /* FIXME: Maybe we should blit the fbcon contents instead of the (blank) * shadow buffer? */ ply_renderer_head_redraw (backend, head); ply_trace ("Setting scan out buffer of %ldx%ld head to our buffer", head->area.width, head->area.height); if (!ply_renderer_head_set_scan_out_buffer (backend, head, head->scan_out_buffer_id)) { backend->driver_interface->destroy_buffer (backend->driver, head->scan_out_buffer_id); head->scan_out_buffer_id = 0; return false; } return true; } static void ply_renderer_head_unmap (ply_renderer_backend_t *backend, ply_renderer_head_t *head) { ply_trace ("unmapping %ldx%ld renderer head", head->area.width, head->area.height); backend->driver_interface->unmap_buffer (backend->driver, head->scan_out_buffer_id); backend->driver_interface->destroy_buffer (backend->driver, head->scan_out_buffer_id); head->scan_out_buffer_id = 0; } static void flush_area (const char *src, unsigned long src_row_stride, char *dst, unsigned long dst_row_stride, ply_rectangle_t *area_to_flush) { unsigned long x1, y1, x2, y2, y; x1 = area_to_flush->x; y1 = area_to_flush->y; x2 = x1 + area_to_flush->width; y2 = y1 + area_to_flush->height; if (area_to_flush->width * 4 == src_row_stride && area_to_flush->width * 4 == dst_row_stride) { memcpy (dst, src, area_to_flush->width * area_to_flush->height * 4); return; } for (y = y1; y < y2; y++) { memcpy (dst, src, area_to_flush->width * 4); dst += dst_row_stride; src += src_row_stride; } } static void ply_renderer_head_flush_area (ply_renderer_head_t *head, ply_rectangle_t *area_to_flush, char *map_address) { uint32_t *shadow_buffer; char *dst, *src; shadow_buffer = ply_pixel_buffer_get_argb32_data (head->pixel_buffer); dst = &map_address[area_to_flush->y * head->row_stride + area_to_flush->x * BYTES_PER_PIXEL]; src = (char *) &shadow_buffer[area_to_flush->y * head->area.width + area_to_flush->x]; flush_area (src, head->area.width * 4, dst, head->row_stride, area_to_flush); } static void free_heads (ply_renderer_backend_t *backend) { ply_list_node_t *node; node = ply_list_get_first_node (backend->heads); while (node != NULL) { ply_list_node_t *next_node; ply_renderer_head_t *head; head = (ply_renderer_head_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (backend->heads, node); ply_renderer_head_free (head); ply_list_remove_node (backend->heads, node); node = next_node; } } static ply_renderer_backend_t * create_backend (const char *device_name, ply_terminal_t *terminal) { ply_renderer_backend_t *backend; backend = calloc (1, sizeof (ply_renderer_backend_t)); if (device_name != NULL) backend->device_name = strdup (device_name); else backend->device_name = strdup ("/dev/dri/card0"); ply_trace ("creating renderer backend for device %s", backend->device_name); backend->device_fd = -1; backend->loop = ply_event_loop_get_default (); backend->heads = ply_list_new (); backend->input_source.key_buffer = ply_buffer_new (); backend->terminal = terminal; return backend; } static void destroy_backend (ply_renderer_backend_t *backend) { ply_trace ("destroying renderer backend for device %s", backend->device_name); free_heads (backend); ply_list_free (backend->heads); free (backend->device_name); free (backend); } static char * find_driver_for_device (const char *device_name) { char *driver; int major_number, minor_number; struct stat file_attributes; char *device_path; char device_link_path[PATH_MAX + 1] = ""; if (stat (device_name, &file_attributes) < 0) return NULL; if (!S_ISCHR (file_attributes.st_mode)) return NULL; major_number = major (file_attributes.st_rdev); minor_number = minor (file_attributes.st_rdev); asprintf (&device_path, "/sys/dev/char/%d:%d/device/driver", major_number, minor_number); if (readlink (device_path, device_link_path, sizeof (device_link_path) - 1) < 0) { free (device_path); return NULL; } free (device_path); driver = strrchr (device_link_path, '/'); if (driver == NULL) return NULL; return strdup (driver + strlen ("/")); } static void activate (ply_renderer_backend_t *backend) { ply_list_node_t *node; ply_trace ("taking master and scanning out"); backend->is_active = true; drmSetMaster (backend->device_fd); node = ply_list_get_first_node (backend->heads); while (node != NULL) { ply_list_node_t *next_node; ply_renderer_head_t *head; head = (ply_renderer_head_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (backend->heads, node); if (head->scan_out_buffer_id != 0) ply_renderer_head_set_scan_out_buffer (backend, head, head->scan_out_buffer_id); node = next_node; } } static void deactivate (ply_renderer_backend_t *backend) { ply_trace ("dropping master"); drmDropMaster (backend->device_fd); backend->is_active = false; } static void on_active_vt_changed (ply_renderer_backend_t *backend) { if (ply_terminal_is_active (backend->terminal)) { ply_trace ("activating on vt change"); activate (backend); } else { ply_trace ("deactivating on vt change"); deactivate (backend); } } static bool nouveau_force_drm (void) { int fd; char kernel_command_line[PLY_MAX_COMMAND_LINE_SIZE]; ply_trace ("opening /proc/cmdline"); fd = open ("/proc/cmdline", O_RDONLY); if (fd < 0) { ply_trace ("couldn't open it: %m"); return false; } ply_trace ("reading kernel command line"); if (read (fd, kernel_command_line, sizeof (kernel_command_line)) < 0) { ply_trace ("couldn't read it: %m"); return false; } ply_trace ("Kernel command line is: '%s'", kernel_command_line); close (fd); if (ply_string_has_prefix (kernel_command_line, "plymouth:force-drm") || strstr (kernel_command_line, " plymouth:force-drm") != NULL) return true; else return false; } static bool load_driver (ply_renderer_backend_t *backend) { char *driver_name; int device_fd; driver_name = find_driver_for_device (backend->device_name); ply_trace ("Attempting to load driver '%s'", driver_name); device_fd = drmOpen (driver_name, NULL); if (device_fd < 0) { ply_trace ("drmOpen failed"); free (driver_name); return false; } backend->driver_interface = NULL; #ifdef PLY_ENABLE_LIBDRM_INTEL if (backend->driver_interface == NULL && strcmp (driver_name, "i915") == 0) { backend->driver_interface = ply_renderer_i915_driver_get_interface (); backend->driver_supports_mapping_console = true; } #endif #ifdef PLY_ENABLE_LIBDRM_RADEON if (backend->driver_interface == NULL && strcmp (driver_name, "radeon") == 0) { backend->driver_interface = ply_renderer_radeon_driver_get_interface (); backend->driver_supports_mapping_console = false; } #endif #ifdef PLY_ENABLE_LIBDRM_NOUVEAU if (backend->driver_interface == NULL && ( strcmp (driver_name, "nouveau") == 0 || strcmp (driver_name, "lbm-nouveau") == 0)) { if (nouveau_force_drm ()) { backend->driver_interface = ply_renderer_nouveau_driver_get_interface (); backend->driver_supports_mapping_console = false; } else ply_trace("falling back to framebuffer for nouveau to avoid DRM hang"); } #endif free (driver_name); if (backend->driver_interface == NULL) { close (device_fd); return false; } backend->driver = backend->driver_interface->create_driver (device_fd); if (backend->driver == NULL) { close (device_fd); return false; } backend->device_fd = device_fd; return true; } static void unload_driver (ply_renderer_backend_t *backend) { if (backend->driver == NULL) return; ply_trace ("unloading driver"); assert (backend->driver_interface != NULL); backend->driver_interface->destroy_driver (backend->driver); backend->driver = NULL; backend->driver_interface = NULL; if (backend->device_fd >= 0) { drmClose (backend->device_fd); backend->device_fd = -1; } } static bool open_device (ply_renderer_backend_t *backend) { assert (backend != NULL); assert (backend->device_name != NULL); if (!load_driver (backend)) return false; if (!ply_terminal_open (backend->terminal)) { ply_trace ("could not open terminal: %m"); return false; } if (!ply_terminal_is_vt (backend->terminal)) { ply_trace ("terminal is not a VT"); ply_terminal_close (backend->terminal); return false; } ply_terminal_watch_for_active_vt_change (backend->terminal, (ply_terminal_active_vt_changed_handler_t) on_active_vt_changed, backend); return true; } static void close_device (ply_renderer_backend_t *backend) { ply_trace ("closing device"); free_heads (backend); ply_terminal_stop_watching_for_active_vt_change (backend->terminal, (ply_terminal_active_vt_changed_handler_t) on_active_vt_changed, backend); unload_driver (backend); } static drmModeModeInfo * get_active_mode_for_connector (ply_renderer_backend_t *backend, drmModeConnector *connector) { return &connector->modes[0]; } static bool controller_is_available (ply_renderer_backend_t *backend, uint32_t controller_id) { ply_list_node_t *node; node = ply_list_get_first_node (backend->heads); while (node != NULL) { ply_list_node_t *next_node; ply_renderer_head_t *head; head = (ply_renderer_head_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (backend->heads, node); if (head->controller_id == controller_id) return false; node = next_node; } return true; } static uint32_t find_controller_for_encoder (ply_renderer_backend_t *backend, drmModeEncoder *encoder) { int i; uint32_t possible_crtcs; /* Monitor is already lit. We'll use the same controller. */ if (encoder->crtc_id != 0) return encoder->crtc_id; /* Monitor cable is plugged in, but the monitor isn't lit * yet. Let's pick an available controller and light it up * ourselves. */ for (i = 0, possible_crtcs = encoder->possible_crtcs; possible_crtcs != 0x0; i++, possible_crtcs >>= 1) { /* controller isn't compatible with encoder */ if ((possible_crtcs & 0x1) == 0) continue; /* controller is already being used */ if (!controller_is_available (backend, backend->resources->crtcs[i])) continue; assert (i < backend->resources->count_crtcs); return backend->resources->crtcs[i]; } return 0; } static bool encoder_is_available (ply_renderer_backend_t *backend, uint32_t encoder_id) { ply_list_node_t *node; node = ply_list_get_first_node (backend->heads); while (node != NULL) { ply_list_node_t *next_node; ply_renderer_head_t *head; head = (ply_renderer_head_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (backend->heads, node); if (head->encoder_id == encoder_id) return false; node = next_node; } return true; } static drmModeEncoder * find_unused_encoder_for_connector (ply_renderer_backend_t *backend, drmModeConnector *connector) { int i; drmModeEncoder *encoder; for (i = 0; i < connector->count_encoders; i++) { encoder = drmModeGetEncoder (backend->device_fd, connector->encoders[i]); if (encoder == NULL) continue; if (encoder_is_available (backend, encoder->encoder_id)) return encoder; drmModeFreeEncoder (encoder); } return NULL; } static drmModeEncoder * find_encoder_for_connector (ply_renderer_backend_t *backend, drmModeConnector *connector) { int i; drmModeEncoder *encoder; assert (backend != NULL); for (i = 0; i < connector->count_encoders; i++) { encoder = drmModeGetEncoder (backend->device_fd, connector->encoders[i]); if (encoder == NULL) continue; if (encoder->encoder_id == connector->encoder_id) return encoder; drmModeFreeEncoder (encoder); } /* No encoder yet, pick one */ return find_unused_encoder_for_connector (backend, connector); } static uint32_t get_console_buffer_id (ply_renderer_backend_t *backend, uint32_t controller_id) { drmModeCrtc *controller; uint32_t console_buffer_id; console_buffer_id = 0; controller = drmModeGetCrtc (backend->device_fd, controller_id); if (controller == NULL) return 0; console_buffer_id = controller->buffer_id; drmModeFreeCrtc (controller); return console_buffer_id; } static bool create_heads_for_active_connectors (ply_renderer_backend_t *backend) { int i; drmModeConnector *connector; for (i = 0; i < backend->resources->count_connectors; i++) { ply_renderer_head_t *head; drmModeEncoder *encoder; uint32_t controller_id; uint32_t encoder_id; uint32_t console_buffer_id; drmModeModeInfo *mode; connector = drmModeGetConnector (backend->device_fd, backend->resources->connectors[i]); if (connector == NULL) continue; if (connector->connection != DRM_MODE_CONNECTED) { drmModeFreeConnector (connector); continue; } if (connector->count_modes <= 0) { drmModeFreeConnector (connector); continue; } encoder = find_encoder_for_connector (backend, connector); if (encoder == NULL) { drmModeFreeConnector (connector); continue; } encoder_id = encoder->encoder_id; controller_id = find_controller_for_encoder (backend, encoder); drmModeFreeEncoder (encoder); if (controller_id == 0) { drmModeFreeConnector (connector); continue; } mode = get_active_mode_for_connector (backend, connector); console_buffer_id = get_console_buffer_id (backend, controller_id); head = ply_renderer_head_new (backend, connector, encoder_id, controller_id, console_buffer_id, mode); ply_list_append_data (backend->heads, head); } #ifdef PLY_ENABLE_GDM_TRANSITION /* If the driver doesn't support mapping the fb console * then we can't get a smooth crossfade transition to * the display manager unless we use the /dev/fb interface * or the plymouth deactivate interface. * * In multihead configurations, we'd rather have working * multihead, but otherwise bail now. */ if (!backend->driver_supports_mapping_console && ply_list_get_length (backend->heads) == 1) { ply_trace ("Only one monitor configured, and driver doesn't " "support mapping console, so letting frame-buffer " "take over"); free_heads (backend); return false; } #endif return ply_list_get_length (backend->heads) > 0; } static bool query_device (ply_renderer_backend_t *backend) { assert (backend != NULL); assert (backend->device_fd >= 0); backend->resources = drmModeGetResources (backend->device_fd); if (backend->resources == NULL) { ply_trace ("Could not get card resources"); return false; } if (!create_heads_for_active_connectors (backend)) { ply_trace ("Could not initialize heads"); return false; } return true; } static bool map_to_device (ply_renderer_backend_t *backend) { ply_list_node_t *node; bool head_mapped; head_mapped = false; node = ply_list_get_first_node (backend->heads); while (node != NULL) { ply_list_node_t *next_node; ply_renderer_head_t *head; head = (ply_renderer_head_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (backend->heads, node); if (ply_renderer_head_map (backend, head)) head_mapped = true; node = next_node; } if (ply_terminal_is_active (backend->terminal)) activate (backend); else ply_terminal_activate_vt (backend->terminal); return head_mapped; } static bool ply_renderer_head_set_scan_out_buffer_to_console (ply_renderer_backend_t *backend, ply_renderer_head_t *head, bool should_set_to_black) { unsigned long width; unsigned long height; unsigned long row_stride; uint32_t *shadow_buffer; ply_pixel_buffer_t *pixel_buffer; char *map_address; ply_rectangle_t area; if (!backend->driver_interface->fetch_buffer (backend->driver, head->console_buffer_id, &width, &height, &row_stride)) return false; if (!backend->driver_interface->map_buffer (backend->driver, head->console_buffer_id)) { backend->driver_interface->destroy_buffer (backend->driver, head->console_buffer_id); return false; } if (head->area.width != width || head->area.height != height) { /* Force black if the fb console resolution doesn't match our resolution */ area.x = 0; area.y = 0; area.width = width; area.height = height; should_set_to_black = true; ply_trace ("Console fb is %ldx%ld and screen contents are %ldx%ld. " "They aren't the same dimensions; forcing black", width, height, head->area.width, head->area.height); } else area = head->area; if (should_set_to_black) { pixel_buffer = ply_pixel_buffer_new (width, height); shadow_buffer = ply_pixel_buffer_get_argb32_data (pixel_buffer); } else { pixel_buffer = NULL; shadow_buffer = ply_pixel_buffer_get_argb32_data (head->pixel_buffer); } ply_trace ("Drawing %s to console fb", should_set_to_black? "black" : "screen contents"); map_address = backend->driver_interface->begin_flush (backend->driver, head->console_buffer_id); flush_area ((char *) shadow_buffer, area.width * 4, map_address, row_stride, &area); backend->driver_interface->end_flush (backend->driver, head->console_buffer_id); backend->driver_interface->unmap_buffer (backend->driver, head->console_buffer_id); ply_trace ("Setting scan out hardware to console fb"); ply_renderer_head_set_scan_out_buffer (backend, head, head->console_buffer_id); backend->driver_interface->destroy_buffer (backend->driver, head->console_buffer_id); if (pixel_buffer != NULL) ply_pixel_buffer_free (pixel_buffer); return true; } static void unmap_from_device (ply_renderer_backend_t *backend) { ply_list_node_t *node; bool should_set_to_black; /* We only copy what's on screen back to the fb console * if there's one head (since in multihead set ups the fb console * is cloned). */ should_set_to_black = ply_list_get_length (backend->heads) > 1; node = ply_list_get_first_node (backend->heads); while (node != NULL) { ply_list_node_t *next_node; ply_renderer_head_t *head; head = (ply_renderer_head_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (backend->heads, node); if (backend->is_active) { ply_trace ("scanning out %s directly to console", should_set_to_black? "black" : "splash"); ply_renderer_head_set_scan_out_buffer_to_console (backend, head, should_set_to_black); } ply_renderer_head_unmap (backend, head); node = next_node; } } static void reset_scan_out_buffer_if_needed (ply_renderer_backend_t *backend, ply_renderer_head_t *head) { drmModeCrtc *controller; if (!ply_terminal_is_active (backend->terminal)) return; controller = drmModeGetCrtc (backend->device_fd, head->controller_id); if (controller == NULL) return; if (controller->buffer_id != head->scan_out_buffer_id) { ply_trace ("Something stole the monitor"); ply_renderer_head_set_scan_out_buffer (backend, head, head->scan_out_buffer_id); } drmModeFreeCrtc (controller); } static void flush_head (ply_renderer_backend_t *backend, ply_renderer_head_t *head) { ply_region_t *updated_region; ply_list_t *areas_to_flush; ply_list_node_t *node; ply_pixel_buffer_t *pixel_buffer; char *map_address; assert (backend != NULL); if (!backend->is_active) return; ply_terminal_set_mode (backend->terminal, PLY_TERMINAL_MODE_GRAPHICS); ply_terminal_set_unbuffered_input (backend->terminal); pixel_buffer = head->pixel_buffer; updated_region = ply_pixel_buffer_get_updated_areas (pixel_buffer); areas_to_flush = ply_region_get_sorted_rectangle_list (updated_region); map_address = backend->driver_interface->begin_flush (backend->driver, head->scan_out_buffer_id); node = ply_list_get_first_node (areas_to_flush); while (node != NULL) { ply_list_node_t *next_node; ply_rectangle_t *area_to_flush; area_to_flush = (ply_rectangle_t *) ply_list_node_get_data (node); next_node = ply_list_get_next_node (areas_to_flush, node); reset_scan_out_buffer_if_needed (backend, head); ply_renderer_head_flush_area (head, area_to_flush, map_address); node = next_node; } backend->driver_interface->end_flush (backend->driver, head->scan_out_buffer_id); ply_region_clear (updated_region); } static void ply_renderer_head_redraw (ply_renderer_backend_t *backend, ply_renderer_head_t *head) { ply_region_t *region; ply_trace ("Redrawing %ldx%ld renderer head", head->area.width, head->area.height); region = ply_pixel_buffer_get_updated_areas (head->pixel_buffer); ply_region_add_rectangle (region, &head->area); flush_head (backend, head); } static ply_list_t * get_heads (ply_renderer_backend_t *backend) { return backend->heads; } static ply_pixel_buffer_t * get_buffer_for_head (ply_renderer_backend_t *backend, ply_renderer_head_t *head) { if (head->backend != backend) return NULL; return head->pixel_buffer; } static bool has_input_source (ply_renderer_backend_t *backend, ply_renderer_input_source_t *input_source) { return input_source == &backend->input_source; } static ply_renderer_input_source_t * get_input_source (ply_renderer_backend_t *backend) { return &backend->input_source; } static void on_key_event (ply_renderer_input_source_t *input_source, int terminal_fd) { ply_buffer_append_from_fd (input_source->key_buffer, terminal_fd); if (input_source->handler != NULL) input_source->handler (input_source->user_data, input_source->key_buffer, input_source); } static void on_input_source_disconnected (ply_renderer_input_source_t *input_source) { ply_trace ("input source disconnected, reopening"); open_input_source (input_source->backend, input_source); } static bool open_input_source (ply_renderer_backend_t *backend, ply_renderer_input_source_t *input_source) { int terminal_fd; assert (backend != NULL); assert (has_input_source (backend, input_source)); terminal_fd = ply_terminal_get_fd (backend->terminal); input_source->backend = backend; input_source->terminal_input_watch = ply_event_loop_watch_fd (backend->loop, terminal_fd, PLY_EVENT_LOOP_FD_STATUS_HAS_DATA, (ply_event_handler_t) on_key_event, (ply_event_handler_t) on_input_source_disconnected, input_source); return true; } static void set_handler_for_input_source (ply_renderer_backend_t *backend, ply_renderer_input_source_t *input_source, ply_renderer_input_source_handler_t handler, void *user_data) { assert (backend != NULL); assert (has_input_source (backend, input_source)); input_source->handler = handler; input_source->user_data = user_data; } static void close_input_source (ply_renderer_backend_t *backend, ply_renderer_input_source_t *input_source) { assert (backend != NULL); assert (has_input_source (backend, input_source)); ply_event_loop_stop_watching_fd (backend->loop, input_source->terminal_input_watch); input_source->terminal_input_watch = NULL; input_source->backend = NULL; } ply_renderer_plugin_interface_t * ply_renderer_backend_get_interface (void) { static ply_renderer_plugin_interface_t plugin_interface = { .create_backend = create_backend, .destroy_backend = destroy_backend, .open_device = open_device, .close_device = close_device, .query_device = query_device, .map_to_device = map_to_device, .unmap_from_device = unmap_from_device, .activate = activate, .deactivate = deactivate, .flush_head = flush_head, .get_heads = get_heads, .get_buffer_for_head = get_buffer_for_head, .get_input_source = get_input_source, .open_input_source = open_input_source, .set_handler_for_input_source = set_handler_for_input_source, .close_input_source = close_input_source }; return &plugin_interface; } /* vim: set ts=4 sw=4 et ai ci cino={.5s,^-2,+.5s,t0,g0,e-2,n-2,p2s,(0,=.5s,:.5s */