| /* |
| * Mesa 3-D graphics library |
| * Version: 7.11 |
| * |
| * Copyright (C) 2011 Benjamin Franzke <benjaminfranzke@googlemail.com> |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining a |
| * copy of this software and associated documentation files (the "Software"), |
| * to deal in the Software without restriction, including without limitation |
| * the rights to use, copy, modify, merge, publish, distribute, sublicense, |
| * and/or sell copies of the Software, and to permit persons to whom the |
| * Software is furnished to do so, subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be included |
| * in all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
| * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
| * DEALINGS IN THE SOFTWARE. |
| */ |
| |
| #include <stdlib.h> |
| #include <sys/types.h> |
| #include <sys/mman.h> |
| #include <unistd.h> |
| |
| #include "pipe/p_compiler.h" |
| #include "pipe/p_defines.h" |
| #include "pipe/p_state.h" |
| #include "util/u_format.h" |
| #include "util/u_math.h" |
| #include "util/u_memory.h" |
| #include "state_tracker/sw_winsys.h" |
| |
| #include <wayland-client.h> |
| #include "wayland_sw_winsys.h" |
| |
| struct wayland_sw_displaytarget |
| { |
| int fd; |
| unsigned size; |
| |
| unsigned width; |
| unsigned height; |
| unsigned stride; |
| |
| enum pipe_format format; |
| |
| void *map; |
| unsigned map_count; |
| }; |
| |
| struct wayland_sw_winsys |
| { |
| struct sw_winsys base; |
| |
| struct wl_display *display; |
| }; |
| |
| static INLINE struct wayland_sw_displaytarget * |
| wayland_sw_displaytarget(struct sw_displaytarget *dt) |
| { |
| return (struct wayland_sw_displaytarget *) dt; |
| } |
| |
| static INLINE struct wayland_sw_winsys * |
| wayland_sw_winsys(struct sw_winsys *ws) |
| { |
| return (struct wayland_sw_winsys *) ws; |
| } |
| |
| static void |
| wayland_displaytarget_display(struct sw_winsys *ws, |
| struct sw_displaytarget *dt, |
| void *context_private) |
| { |
| } |
| |
| static void |
| wayland_displaytarget_unmap(struct sw_winsys *ws, |
| struct sw_displaytarget *dt) |
| { |
| struct wayland_sw_displaytarget *wldt = wayland_sw_displaytarget(dt); |
| |
| wldt->map_count--; |
| if (wldt->map_count > 0) |
| return; |
| |
| munmap(wldt->map, wldt->size); |
| wldt->map = NULL; |
| } |
| |
| static void * |
| wayland_displaytarget_map(struct sw_winsys *ws, |
| struct sw_displaytarget *dt, |
| unsigned flags) |
| { |
| struct wayland_sw_displaytarget *wldt = wayland_sw_displaytarget(dt); |
| uint mmap_flags = 0; |
| |
| if (wldt->map) { |
| wldt->map_count++; |
| return wldt->map; |
| } |
| |
| if (flags & PIPE_TRANSFER_READ) |
| mmap_flags |= PROT_READ; |
| if (flags & PIPE_TRANSFER_WRITE) |
| mmap_flags |= PROT_WRITE; |
| |
| wldt->map = mmap(NULL, wldt->size, mmap_flags, |
| MAP_SHARED, wldt->fd, 0); |
| |
| if (wldt->map == MAP_FAILED) |
| return NULL; |
| |
| wldt->map_count = 1; |
| |
| return wldt->map; |
| } |
| |
| static void |
| wayland_displaytarget_destroy(struct sw_winsys *ws, |
| struct sw_displaytarget *dt) |
| { |
| struct wayland_sw_displaytarget *wldt = wayland_sw_displaytarget(dt); |
| |
| if (wldt->map) |
| wayland_displaytarget_unmap(ws, dt); |
| |
| FREE(wldt); |
| } |
| |
| static boolean |
| wayland_is_displaytarget_format_supported(struct sw_winsys *ws, |
| unsigned tex_usage, |
| enum pipe_format format) |
| { |
| switch (format) { |
| case PIPE_FORMAT_B8G8R8X8_UNORM: |
| case PIPE_FORMAT_B8G8R8A8_UNORM: |
| return TRUE; |
| default: |
| return FALSE; |
| } |
| } |
| |
| static struct sw_displaytarget * |
| wayland_displaytarget_create(struct sw_winsys *ws, |
| unsigned tex_usage, |
| enum pipe_format format, |
| unsigned width, unsigned height, |
| unsigned alignment, |
| unsigned *stride) |
| { |
| struct wayland_sw_displaytarget *wldt; |
| unsigned nblocksy, format_stride; |
| char filename[] = "/tmp/wayland-shm-XXXXXX"; |
| |
| if (!wayland_is_displaytarget_format_supported(ws, tex_usage, format)) |
| return NULL; |
| |
| wldt = CALLOC_STRUCT(wayland_sw_displaytarget); |
| if (!wldt) |
| return NULL; |
| |
| wldt->map = NULL; |
| |
| wldt->format = format; |
| wldt->width = width; |
| wldt->height = height; |
| |
| format_stride = util_format_get_stride(format, width); |
| wldt->stride = align(format_stride, alignment); |
| |
| nblocksy = util_format_get_nblocksy(format, height); |
| wldt->size = wldt->stride * nblocksy; |
| |
| wldt->fd = mkstemp(filename); |
| if (wldt->fd < 0) { |
| FREE(wldt); |
| return NULL; |
| } |
| |
| if (ftruncate(wldt->fd, wldt->size) < 0) { |
| unlink(filename); |
| close(wldt->fd); |
| FREE(wldt); |
| return NULL; |
| } |
| |
| unlink(filename); |
| |
| *stride = wldt->stride; |
| |
| return (struct sw_displaytarget *) wldt; |
| } |
| |
| static struct sw_displaytarget * |
| wayland_displaytarget_from_handle(struct sw_winsys *ws, |
| const struct pipe_resource *templet, |
| struct winsys_handle *whandle, |
| unsigned *stride) |
| { |
| struct wayland_sw_displaytarget *wldt; |
| unsigned nblocksy; |
| |
| if (!wayland_is_displaytarget_format_supported(ws, 0, templet->format)) |
| return NULL; |
| |
| wldt = CALLOC_STRUCT(wayland_sw_displaytarget); |
| if (!wldt) |
| return NULL; |
| |
| wldt->fd = whandle->fd; |
| wldt->stride = whandle->stride; |
| wldt->width = templet->width0; |
| wldt->height = templet->height0; |
| wldt->format = templet->format; |
| |
| nblocksy = util_format_get_nblocksy(wldt->format, wldt->height); |
| |
| wldt->size = wldt->stride * nblocksy; |
| |
| wldt->map = NULL; |
| |
| *stride = wldt->stride; |
| |
| return (struct sw_displaytarget *) wldt; |
| } |
| |
| |
| static boolean |
| wayland_displaytarget_get_handle(struct sw_winsys *ws, |
| struct sw_displaytarget *dt, |
| struct winsys_handle *whandle) |
| { |
| struct wayland_sw_displaytarget *wldt = wayland_sw_displaytarget(dt); |
| |
| whandle->fd = wldt->fd; |
| whandle->stride = wldt->stride; |
| whandle->size = wldt->size; |
| |
| return TRUE; |
| } |
| |
| static void |
| wayland_destroy(struct sw_winsys *ws) |
| { |
| struct wayland_sw_winsys *wayland = wayland_sw_winsys(ws); |
| |
| FREE(wayland); |
| } |
| |
| struct sw_winsys * |
| wayland_create_sw_winsys(struct wl_display *display) |
| { |
| struct wayland_sw_winsys *wlws; |
| |
| wlws = CALLOC_STRUCT(wayland_sw_winsys); |
| if (!wlws) |
| return NULL; |
| |
| wlws->display = display; |
| |
| wlws->base.destroy = wayland_destroy; |
| wlws->base.is_displaytarget_format_supported = |
| wayland_is_displaytarget_format_supported; |
| |
| wlws->base.displaytarget_create = wayland_displaytarget_create; |
| wlws->base.displaytarget_from_handle = wayland_displaytarget_from_handle; |
| wlws->base.displaytarget_get_handle = wayland_displaytarget_get_handle; |
| wlws->base.displaytarget_destroy = wayland_displaytarget_destroy; |
| wlws->base.displaytarget_map = wayland_displaytarget_map; |
| wlws->base.displaytarget_unmap = wayland_displaytarget_unmap; |
| |
| wlws->base.displaytarget_display = wayland_displaytarget_display; |
| |
| return &wlws->base; |
| } |
| |
| /* vim: set sw=3 ts=8 sts=3 expandtab: */ |