/*
* 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: */