|
|
|
/*
|
|
|
|
* Copyright © 2015 Samsung Electronics Co., Ltd
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and
|
|
|
|
* its documentation for any purpose is hereby granted without fee, provided
|
|
|
|
* that the above copyright notice appear in all copies and that both that
|
|
|
|
* copyright notice and this permission notice appear in supporting
|
|
|
|
* documentation, and that the name of the copyright holders not be used in
|
|
|
|
* advertising or publicity pertaining to distribution of the software
|
|
|
|
* without specific, written prior permission. The copyright holders make
|
|
|
|
* no representations about the suitability of this software for any
|
|
|
|
* purpose. It is provided "as is" without express or implied warranty.
|
|
|
|
*
|
|
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
|
|
|
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
|
|
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
|
|
|
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
|
|
|
|
* CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h> /* memcpy */
|
|
|
|
#include <cairo.h>
|
|
|
|
|
|
|
|
#include "weston-test-client-helper.h"
|
|
|
|
|
|
|
|
char *server_parameters="--use-pixman --width=320 --height=240";
|
|
|
|
|
|
|
|
/** write_surface_as_png()
|
|
|
|
*
|
|
|
|
* Writes out a given weston test surface to disk as a PNG image
|
|
|
|
* using the provided filename (with path).
|
|
|
|
*
|
|
|
|
* @returns true if successfully saved file; false otherwise.
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
write_surface_as_png(const struct surface* weston_surface, const char *fname) {
|
|
|
|
cairo_surface_t *cairo_surface;
|
|
|
|
cairo_status_t status;
|
|
|
|
int bpp = 4; /* Assume ARGB */
|
|
|
|
int stride = bpp * weston_surface->width;
|
|
|
|
|
|
|
|
cairo_surface = cairo_image_surface_create_for_data(weston_surface->data,
|
|
|
|
CAIRO_FORMAT_ARGB32,
|
|
|
|
weston_surface->width,
|
|
|
|
weston_surface->height,
|
|
|
|
stride);
|
|
|
|
printf("Writing PNG to disk\n");
|
|
|
|
status = cairo_surface_write_to_png(cairo_surface, fname);
|
|
|
|
if (status != CAIRO_STATUS_SUCCESS) {
|
|
|
|
printf("Failed to save screenshot: %s\n",
|
|
|
|
cairo_status_to_string(status));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
cairo_surface_destroy(cairo_surface);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** load_surface_from_png()
|
|
|
|
*
|
|
|
|
* Reads a PNG image from disk using the given filename (and path)
|
|
|
|
* and returns as a freshly allocated weston test surface.
|
|
|
|
*
|
|
|
|
* @returns weston test surface with image, which should be free'd
|
|
|
|
* when no longer used; or, NULL in case of error.
|
|
|
|
*/
|
|
|
|
static struct surface*
|
|
|
|
load_surface_from_png(const char *fname) {
|
|
|
|
struct surface *reference;
|
|
|
|
cairo_surface_t *reference_cairo_surface;
|
|
|
|
cairo_status_t status;
|
|
|
|
size_t source_data_size;
|
|
|
|
int bpp;
|
|
|
|
int stride;
|
|
|
|
|
|
|
|
printf("Loading reference image %s\n", fname);
|
|
|
|
reference_cairo_surface = cairo_image_surface_create_from_png(fname);
|
|
|
|
status = cairo_surface_status(reference_cairo_surface);
|
|
|
|
if (status != CAIRO_STATUS_SUCCESS) {
|
|
|
|
printf("Could not open %s: %s\n", fname, cairo_status_to_string(status));
|
|
|
|
cairo_surface_destroy(reference_cairo_surface);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disguise the cairo surface in a weston test surface */
|
|
|
|
reference = xzalloc(sizeof *reference);
|
|
|
|
if (reference == NULL) {
|
|
|
|
perror("xzalloc reference");
|
|
|
|
cairo_surface_destroy(reference_cairo_surface);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
reference->width = cairo_image_surface_get_width(reference_cairo_surface);
|
|
|
|
reference->height = cairo_image_surface_get_height(reference_cairo_surface);
|
|
|
|
stride = cairo_image_surface_get_stride(reference_cairo_surface);
|
|
|
|
source_data_size = stride * reference->height;
|
|
|
|
|
|
|
|
/* Check that the file's stride matches our assumption */
|
|
|
|
bpp = 4;
|
|
|
|
if (stride != bpp * reference->width) {
|
|
|
|
printf("Mismatched stride for screenshot reference image %s\n", fname);
|
|
|
|
cairo_surface_destroy(reference_cairo_surface);
|
|
|
|
free(reference);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate new buffer for our weston reference, and copy the data from
|
|
|
|
the cairo surface so we can destroy it */
|
|
|
|
reference->data = xzalloc(source_data_size);
|
|
|
|
if (reference->data == NULL) {
|
|
|
|
perror("xzalloc reference data");
|
|
|
|
cairo_surface_destroy(reference_cairo_surface);
|
|
|
|
free(reference);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
memcpy(reference->data,
|
|
|
|
cairo_image_surface_get_data(reference_cairo_surface),
|
|
|
|
source_data_size);
|
|
|
|
|
|
|
|
cairo_surface_destroy(reference_cairo_surface);
|
|
|
|
return reference;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** create_screenshot_surface()
|
|
|
|
*
|
|
|
|
* Allocates and initializes a weston test surface for use in
|
|
|
|
* storing a screenshot of the client's output. Establishes a
|
|
|
|
* shm backed wl_buffer for retrieving screenshot image data
|
|
|
|
* from the server, sized to match the client's output display.
|
|
|
|
*
|
|
|
|
* @returns stack allocated surface image, which should be
|
|
|
|
* free'd when done using it.
|
|
|
|
*/
|
|
|
|
static struct surface*
|
|
|
|
create_screenshot_surface(struct client *client) {
|
|
|
|
struct surface* screenshot;
|
|
|
|
screenshot = xzalloc(sizeof *screenshot);
|
|
|
|
if (screenshot == NULL)
|
|
|
|
return NULL;
|
|
|
|
screenshot->wl_buffer = create_shm_buffer(client,
|
|
|
|
client->output->width,
|
|
|
|
client->output->height,
|
|
|
|
&screenshot->data);
|
|
|
|
screenshot->height = client->output->height;
|
|
|
|
screenshot->width = client->output->width;
|
|
|
|
|
|
|
|
return screenshot;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** capture_screenshot_of_output()
|
|
|
|
*
|
|
|
|
* Requests a screenshot from the server of the output that the
|
|
|
|
* client appears on. The image data returned from the server
|
|
|
|
* can be accessed from the screenshot surface's data member.
|
|
|
|
*
|
|
|
|
* @returns a new surface object, which should be free'd when no
|
|
|
|
* longer needed.
|
|
|
|
*/
|
|
|
|
static struct surface *
|
|
|
|
capture_screenshot_of_output(struct client *client) {
|
|
|
|
struct surface *screenshot;
|
|
|
|
|
|
|
|
/* Create a surface to hold the screenshot */
|
|
|
|
screenshot = create_screenshot_surface(client);
|
|
|
|
|
|
|
|
client->test->buffer_copy_done = 0;
|
|
|
|
weston_test_capture_screenshot(client->test->weston_test,
|
|
|
|
client->output->wl_output,
|
|
|
|
screenshot->wl_buffer);
|
|
|
|
while (client->test->buffer_copy_done == 0)
|
|
|
|
if (wl_display_dispatch(client->wl_display) < 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* FIXME: Document somewhere the orientation the screenshot is taken
|
|
|
|
* and how the clip coords are interpreted, in case of scaling/transform.
|
|
|
|
* If we're using read_pixels() just make sure it is documented somewhere.
|
|
|
|
* Protocol docs in the XML, comparison function docs in Doxygen style.
|
|
|
|
*/
|
|
|
|
|
|
|
|
return screenshot;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
draw_stuff(char *pixels, int w, int h)
|
|
|
|
{
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
for (x = 0; x < w; x++)
|
|
|
|
for (y = 0; y < h; y++) {
|
|
|
|
pixels[y * w * 4 + x * 4] = x;
|
|
|
|
pixels[y * w * 4 + x * 4 + 1] = x + y;
|
|
|
|
pixels[y * w * 4 + x * 4 + 2] = y;
|
|
|
|
pixels[y * w * 4 + x * 4 + 3] = 255;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(internal_screenshot)
|
|
|
|
{
|
|
|
|
struct wl_buffer *buf;
|
|
|
|
struct client *client;
|
|
|
|
struct wl_surface *surface;
|
|
|
|
struct surface *screenshot = NULL;
|
|
|
|
struct surface *reference_good = NULL;
|
|
|
|
struct surface *reference_bad = NULL;
|
|
|
|
struct rectangle clip;
|
|
|
|
const char *fname;
|
|
|
|
bool match = false;
|
|
|
|
bool dump_all_images = true;
|
|
|
|
void *pixels;
|
|
|
|
|
|
|
|
/* Create the client */
|
|
|
|
printf("Creating client for test\n");
|
|
|
|
client = create_client_and_test_surface(100, 100, 100, 100);
|
|
|
|
assert(client);
|
|
|
|
surface = client->surface->wl_surface;
|
|
|
|
|
|
|
|
buf = create_shm_buffer(client, 100, 100, &pixels);
|
|
|
|
draw_stuff(pixels, 100, 100);
|
|
|
|
wl_surface_attach(surface, buf, 0, 0);
|
|
|
|
wl_surface_damage(surface, 0, 0, 100, 100);
|
|
|
|
wl_surface_commit(surface);
|
|
|
|
|
|
|
|
/* Take a snapshot. Result will be in screenshot->wl_buffer. */
|
|
|
|
printf("Taking a screenshot\n");
|
|
|
|
screenshot = capture_screenshot_of_output(client);
|
|
|
|
assert(screenshot);
|
|
|
|
|
|
|
|
/* Load good reference image */
|
|
|
|
fname = screenshot_reference_filename("internal-screenshot-good", 0);
|
|
|
|
printf("Loading good reference image %s\n", fname);
|
|
|
|
reference_good = load_surface_from_png(fname);
|
|
|
|
assert(reference_good);
|
|
|
|
|
|
|
|
/* Load bad reference image */
|
|
|
|
fname = screenshot_reference_filename("internal-screenshot-bad", 0);
|
|
|
|
printf("Loading bad reference image %s\n", fname);
|
|
|
|
reference_bad = load_surface_from_png(fname);
|
|
|
|
assert(reference_bad);
|
|
|
|
|
|
|
|
/* Test check_surfaces_equal()
|
|
|
|
* We expect this to fail since we use a bad reference image
|
|
|
|
*/
|
|
|
|
match = check_surfaces_equal(screenshot, reference_bad);
|
|
|
|
printf("Screenshot %s reference image\n", match? "equal to" : "different from");
|
|
|
|
assert(!match);
|
|
|
|
free(reference_bad->data);
|
|
|
|
free(reference_bad);
|
|
|
|
|
|
|
|
/* Test check_surfaces_match_in_clip()
|
|
|
|
* Alpha-blending and other effects can cause irrelevant discrepancies, so look only
|
|
|
|
* at a small portion of the solid-colored background
|
|
|
|
*/
|
|
|
|
clip.x = 100;
|
|
|
|
clip.y = 100;
|
|
|
|
clip.width = 100;
|
|
|
|
clip.height = 100;
|
|
|
|
printf("Clip: %d,%d %d x %d\n", clip.x, clip.y, clip.width, clip.height);
|
|
|
|
match = check_surfaces_match_in_clip(screenshot, reference_good,
|
|
|
|
&clip);
|
|
|
|
printf("Screenshot %s reference image in clipped area\n", match? "matches" : "doesn't match");
|
|
|
|
free(reference_good->data);
|
|
|
|
free(reference_good);
|
|
|
|
|
|
|
|
/* Test dumping of non-matching images */
|
|
|
|
if (!match || dump_all_images) {
|
|
|
|
fname = screenshot_output_filename("internal-screenshot", 0);
|
|
|
|
write_surface_as_png(screenshot, fname);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(screenshot);
|
|
|
|
|
|
|
|
printf("Test complete\n");
|
|
|
|
assert(match);
|
|
|
|
}
|