/*
* Copyright (c) 2011 Intel Corporation. All Rights Reserved.
*
* 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, sub license, 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 (including the
* next paragraph) 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 NON-INFRINGEMENT.
* IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS 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 <stddef.h>
#include <errno.h>
#include <sys/select.h>
#ifdef IN_LIBVA
# include "va/wayland/va_wayland.h"
#else
# include <va/va_wayland.h>
#endif
#include <wayland-server.h>
static void *open_display(void);
static void close_display(void *win_display);
static int create_window(void *win_display,
int x, int y, int width, int height);
static int check_window_event(void *win_display, void *drawable,
int *width, int *height, int *quit);
struct display;
struct drawable;
static VAStatus
va_put_surface(
VADisplay dpy,
struct drawable *wl_drawable,
VASurfaceID va_surface,
const VARectangle *src_rect,
const VARectangle *dst_rect,
const VARectangle *cliprects,
unsigned int num_cliprects,
unsigned int flags
);
/* Glue code for the current PutSurface test design */
#define CAST_DRAWABLE(a) (struct drawable *)(a)
static inline VADisplay
vaGetDisplay(VANativeDisplay native_dpy)
{
return vaGetDisplayWl(native_dpy);
}
static VAStatus
vaPutSurface(
VADisplay dpy,
VASurfaceID surface,
struct drawable *wl_drawable,
short src_x,
short src_y,
unsigned short src_w,
unsigned short src_h,
short dst_x,
short dst_y,
unsigned short dst_w,
unsigned short dst_h,
const VARectangle *cliprects,
unsigned int num_cliprects,
unsigned int flags
)
{
VARectangle src_rect, dst_rect;
src_rect.x = src_x;
src_rect.y = src_y;
src_rect.width = src_w;
src_rect.height = src_h;
dst_rect.x = src_x;
dst_rect.y = src_y;
dst_rect.width = src_w;
dst_rect.height = src_h;
return va_put_surface(dpy, wl_drawable, surface, &src_rect, &dst_rect,
cliprects, num_cliprects, flags);
}
#include "putsurface_common.c"
struct display {
struct wl_display *display;
struct wl_compositor *compositor;
struct wl_shell *shell;
struct wl_registry *registry;
int event_fd;
};
struct drawable {
struct wl_display *display;
struct wl_surface *surface;
unsigned int redraw_pending : 1;
};
static void
frame_redraw_callback(void *data, struct wl_callback *callback, uint32_t time)
{
struct drawable * const drawable = data;
drawable->redraw_pending = 0;
wl_callback_destroy(callback);
}
static const struct wl_callback_listener frame_callback_listener = {
frame_redraw_callback
};
static VAStatus
va_put_surface(
VADisplay dpy,
struct drawable *wl_drawable,
VASurfaceID va_surface,
const VARectangle *src_rect,
const VARectangle *dst_rect,
const VARectangle *cliprects,
unsigned int num_cliprects,
unsigned int flags
)
{
struct display *d;
struct wl_callback *callback;
VAStatus va_status;
struct wl_buffer *buffer;
if (!wl_drawable)
return VA_STATUS_ERROR_INVALID_SURFACE;
d = wl_display_get_user_data(wl_drawable->display);
if (!d)
return VA_STATUS_ERROR_INVALID_DISPLAY;
/* Wait for the previous frame to complete redraw */
if (wl_drawable->redraw_pending) {
wl_display_flush(d->display);
while (wl_drawable->redraw_pending)
wl_display_dispatch(wl_drawable->display);
}
va_status = vaGetSurfaceBufferWl(va_dpy, va_surface, VA_FRAME_PICTURE, &buffer);
if (va_status != VA_STATUS_SUCCESS)
return va_status;
wl_surface_attach(wl_drawable->surface, buffer, 0, 0);
wl_surface_damage(
wl_drawable->surface,
dst_rect->x, dst_rect->y, dst_rect->width, dst_rect->height
);
wl_display_flush(d->display);
wl_drawable->redraw_pending = 1;
callback = wl_surface_frame(wl_drawable->surface);
wl_surface_commit(wl_drawable->surface);
wl_callback_add_listener(callback, &frame_callback_listener, wl_drawable);
return VA_STATUS_SUCCESS;
}
static void
registry_handle_global(
void *data,
struct wl_registry *registry,
uint32_t id,
const char *interface,
uint32_t version
)
{
struct display * const d = data;
if (strcmp(interface, "wl_compositor") == 0)
d->compositor =
wl_registry_bind(registry, id, &wl_compositor_interface, 1);
else if (strcmp(interface, "wl_shell") == 0)
d->shell = wl_registry_bind(registry, id, &wl_shell_interface, 1);
}
static const struct wl_registry_listener registry_listener = {
registry_handle_global,
NULL,
};
static void *
open_display(void)
{
struct display *d;
d = calloc(1, sizeof *d);
if (!d)
return NULL;
d->display = wl_display_connect(NULL);
if (!d->display)
return NULL;
wl_display_set_user_data(d->display, d);
d->registry = wl_display_get_registry(d->display);
wl_registry_add_listener(d->registry, ®istry_listener, d);
d->event_fd = wl_display_get_fd(d->display);
wl_display_dispatch(d->display);
return d->display;
}
static void
close_display(void *win_display)
{
struct display * const d = wl_display_get_user_data(win_display);
if (d->shell) {
wl_shell_destroy(d->shell);
d->shell = NULL;
}
if (d->compositor) {
wl_compositor_destroy(d->compositor);
d->compositor = NULL;
}
if (d->display) {
wl_display_disconnect(d->display);
d->display = NULL;
}
free(d);
}
static int
create_window(void *win_display, int x, int y, int width, int height)
{
struct wl_display * const display = win_display;
struct display * const d = wl_display_get_user_data(display);
struct wl_surface *surface1, *surface2;
struct wl_shell_surface *shell_surface;
struct wl_shell_surface *shell_surface_2;
struct drawable *drawable1, *drawable2;
surface1 = wl_compositor_create_surface(d->compositor);
shell_surface = wl_shell_get_shell_surface(d->shell, surface1);
wl_shell_surface_set_toplevel(shell_surface);
drawable1 = malloc(sizeof(*drawable1));
drawable1->display = display;
drawable1->surface = surface1;
drawable1->redraw_pending = 0;
/* global out */
drawable_thread0 = drawable1;
if (multi_thread == 0)
return 0;
surface2 = wl_compositor_create_surface(d->compositor);
shell_surface_2 = wl_shell_get_shell_surface(d->shell, surface2);
wl_shell_surface_set_toplevel(shell_surface_2);
drawable2 = malloc(sizeof(*drawable2));
drawable2->display = display;
drawable1->surface = surface2;
drawable2->redraw_pending = 0;
/* global out */
drawable_thread1 = drawable2;
return 0;
}
static int
check_window_event(
void *win_display,
void *drawable,
int *width,
int *height,
int *quit
)
{
struct wl_display * const display = win_display;
struct display * const d = wl_display_get_user_data(display);
struct timeval tv;
fd_set rfds;
int retval;
if (check_event == 0)
return 0;
tv.tv_sec = 0;
tv.tv_usec = 0;
do {
FD_ZERO(&rfds);
FD_SET(d->event_fd, &rfds);
retval = select(d->event_fd + 1, &rfds, NULL, NULL, &tv);
if (retval < 0) {
perror("select");
break;
}
if (retval == 1)
wl_display_dispatch(d->display);
} while (retval > 0);
#if 0
/* bail on any focused key press */
if(event.type == KeyPress) {
*quit = 1;
return 0;
}
#endif
#if 0
/* rescale the video to fit the window */
if(event.type == ConfigureNotify) {
*width = event.xconfigure.width;
*height = event.xconfigure.height;
printf("Scale window to %dx%d\n", width, height);
}
#endif
return 0;
}