| /* |
| * 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; |
| } |