tests: Drop redundant debug output
[platform/upstream/weston.git] / tests / internal-screenshot-test.c
1 /*
2  * Copyright © 2015 Samsung Electronics Co., Ltd
3  *
4  * Permission to use, copy, modify, distribute, and sell this software and
5  * its documentation for any purpose is hereby granted without fee, provided
6  * that the above copyright notice appear in all copies and that both that
7  * copyright notice and this permission notice appear in supporting
8  * documentation, and that the name of the copyright holders not be used in
9  * advertising or publicity pertaining to distribution of the software
10  * without specific, written prior permission.  The copyright holders make
11  * no representations about the suitability of this software for any
12  * purpose.  It is provided "as is" without express or implied warranty.
13  *
14  * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
15  * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
16  * FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
17  * SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
18  * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
19  * CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
20  * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
21  */
22
23 #include "config.h"
24
25 #include <unistd.h>
26 #include <stdio.h>
27 #include <string.h> /* memcpy */
28 #include <cairo.h>
29
30 #include "weston-test-client-helper.h"
31
32 char *server_parameters="--use-pixman --width=320 --height=240";
33
34 /** write_surface_as_png()
35  *
36  * Writes out a given weston test surface to disk as a PNG image
37  * using the provided filename (with path).
38  *
39  * @returns true if successfully saved file; false otherwise.
40  */
41 static bool
42 write_surface_as_png(const struct surface* weston_surface, const char *fname) {
43         cairo_surface_t *cairo_surface;
44         cairo_status_t status;
45         int bpp = 4; /* Assume ARGB */
46         int stride = bpp * weston_surface->width;
47
48         cairo_surface = cairo_image_surface_create_for_data(weston_surface->data,
49                                                             CAIRO_FORMAT_ARGB32,
50                                                             weston_surface->width,
51                                                             weston_surface->height,
52                                                             stride);
53         printf("Writing PNG to disk\n");
54         status = cairo_surface_write_to_png(cairo_surface, fname);
55         if (status != CAIRO_STATUS_SUCCESS) {
56                 printf("Failed to save screenshot: %s\n",
57                        cairo_status_to_string(status));
58                 return false;
59         }
60         cairo_surface_destroy(cairo_surface);
61         return true;
62 }
63
64 /** load_surface_from_png()
65  *
66  * Reads a PNG image from disk using the given filename (and path)
67  * and returns as a freshly allocated weston test surface.
68  *
69  * @returns weston test surface with image, which should be free'd
70  * when no longer used; or, NULL in case of error.
71  */
72 static struct surface*
73 load_surface_from_png(const char *fname) {
74         struct surface *reference;
75         cairo_surface_t *reference_cairo_surface;
76         cairo_status_t status;
77         size_t source_data_size;
78         int bpp;
79         int stride;
80
81         reference_cairo_surface = cairo_image_surface_create_from_png(fname);
82         status = cairo_surface_status(reference_cairo_surface);
83         if (status != CAIRO_STATUS_SUCCESS) {
84                 printf("Could not open %s: %s\n", fname, cairo_status_to_string(status));
85                 cairo_surface_destroy(reference_cairo_surface);
86                 return NULL;
87         }
88
89         /* Disguise the cairo surface in a weston test surface */
90         reference = xzalloc(sizeof *reference);
91         if (reference == NULL) {
92                 perror("xzalloc reference");
93                 cairo_surface_destroy(reference_cairo_surface);
94                 return NULL;
95         }
96         reference->width = cairo_image_surface_get_width(reference_cairo_surface);
97         reference->height = cairo_image_surface_get_height(reference_cairo_surface);
98         stride = cairo_image_surface_get_stride(reference_cairo_surface);
99         source_data_size = stride * reference->height;
100
101         /* Check that the file's stride matches our assumption */
102         bpp = 4;
103         if (stride != bpp * reference->width) {
104                 printf("Mismatched stride for screenshot reference image %s\n", fname);
105                 cairo_surface_destroy(reference_cairo_surface);
106                 free(reference);
107                 return NULL;
108         }
109
110         /* Allocate new buffer for our weston reference, and copy the data from
111            the cairo surface so we can destroy it */
112         reference->data = xzalloc(source_data_size);
113         if (reference->data == NULL) {
114                 perror("xzalloc reference data");
115                 cairo_surface_destroy(reference_cairo_surface);
116                 free(reference);
117                 return NULL;
118         }
119         memcpy(reference->data,
120                cairo_image_surface_get_data(reference_cairo_surface),
121                source_data_size);
122
123         cairo_surface_destroy(reference_cairo_surface);
124         return reference;
125 }
126
127 /** create_screenshot_surface()
128  *
129  *  Allocates and initializes a weston test surface for use in
130  *  storing a screenshot of the client's output.  Establishes a
131  *  shm backed wl_buffer for retrieving screenshot image data
132  *  from the server, sized to match the client's output display.
133  *
134  *  @returns stack allocated surface image, which should be
135  *  free'd when done using it.
136  */
137 static struct surface*
138 create_screenshot_surface(struct client *client) {
139         struct surface* screenshot;
140         screenshot = xzalloc(sizeof *screenshot);
141         if (screenshot == NULL)
142                 return NULL;
143         screenshot->wl_buffer = create_shm_buffer(client,
144                                                   client->output->width,
145                                                   client->output->height,
146                                                   &screenshot->data);
147         screenshot->height = client->output->height;
148         screenshot->width = client->output->width;
149
150         return screenshot;
151 }
152
153 /** capture_screenshot_of_output()
154  *
155  * Requests a screenshot from the server of the output that the
156  * client appears on.  The image data returned from the server
157  * can be accessed from the screenshot surface's data member.
158  *
159  * @returns a new surface object, which should be free'd when no
160  * longer needed.
161  */
162 static struct surface *
163 capture_screenshot_of_output(struct client *client) {
164         struct surface *screenshot;
165
166         /* Create a surface to hold the screenshot */
167         screenshot = create_screenshot_surface(client);
168
169         client->test->buffer_copy_done = 0;
170         weston_test_capture_screenshot(client->test->weston_test,
171                                        client->output->wl_output,
172                                        screenshot->wl_buffer);
173         while (client->test->buffer_copy_done == 0)
174                 if (wl_display_dispatch(client->wl_display) < 0)
175                         break;
176
177         /* FIXME: Document somewhere the orientation the screenshot is taken
178          * and how the clip coords are interpreted, in case of scaling/transform.
179          * If we're using read_pixels() just make sure it is documented somewhere.
180          * Protocol docs in the XML, comparison function docs in Doxygen style.
181          */
182
183         return screenshot;
184 }
185
186 static void
187 draw_stuff(char *pixels, int w, int h)
188 {
189         int x, y;
190
191         for (x = 0; x < w; x++)
192                 for (y = 0; y < h; y++) {
193                         pixels[y * w * 4 + x * 4] = x;
194                         pixels[y * w * 4 + x * 4 + 1] = x + y;
195                         pixels[y * w * 4 + x * 4 + 2] = y;
196                         pixels[y * w * 4 + x * 4 + 3] = 255;
197                 }
198 }
199
200 TEST(internal_screenshot)
201 {
202         struct wl_buffer *buf;
203         struct client *client;
204         struct wl_surface *surface;
205         struct surface *screenshot = NULL;
206         struct surface *reference_good = NULL;
207         struct surface *reference_bad = NULL;
208         struct rectangle clip;
209         const char *fname;
210         bool match = false;
211         bool dump_all_images = true;
212         void *pixels;
213
214         /* Create the client */
215         printf("Creating client for test\n");
216         client = create_client_and_test_surface(100, 100, 100, 100);
217         assert(client);
218         surface = client->surface->wl_surface;
219
220         buf = create_shm_buffer(client, 100, 100, &pixels);
221         draw_stuff(pixels, 100, 100);
222         wl_surface_attach(surface, buf, 0, 0);
223         wl_surface_damage(surface, 0, 0, 100, 100);
224         wl_surface_commit(surface);
225
226         /* Take a snapshot.  Result will be in screenshot->wl_buffer. */
227         printf("Taking a screenshot\n");
228         screenshot = capture_screenshot_of_output(client);
229         assert(screenshot);
230
231         /* Load good reference image */
232         fname = screenshot_reference_filename("internal-screenshot-good", 0);
233         printf("Loading good reference image %s\n", fname);
234         reference_good = load_surface_from_png(fname);
235         assert(reference_good);
236
237         /* Load bad reference image */
238         fname = screenshot_reference_filename("internal-screenshot-bad", 0);
239         printf("Loading bad reference image %s\n", fname);
240         reference_bad = load_surface_from_png(fname);
241         assert(reference_bad);
242
243         /* Test check_surfaces_equal()
244          * We expect this to fail since we use a bad reference image
245          */
246         match = check_surfaces_equal(screenshot, reference_bad);
247         printf("Screenshot %s reference image\n", match? "equal to" : "different from");
248         assert(!match);
249         free(reference_bad->data);
250         free(reference_bad);
251
252         /* Test check_surfaces_match_in_clip()
253          * Alpha-blending and other effects can cause irrelevant discrepancies, so look only
254          * at a small portion of the solid-colored background
255          */
256         clip.x = 100;
257         clip.y = 100;
258         clip.width = 100;
259         clip.height = 100;
260         printf("Clip: %d,%d %d x %d\n", clip.x, clip.y, clip.width, clip.height);
261         match = check_surfaces_match_in_clip(screenshot, reference_good,
262                                              &clip);
263         printf("Screenshot %s reference image in clipped area\n", match? "matches" : "doesn't match");
264         free(reference_good->data);
265         free(reference_good);
266
267         /* Test dumping of non-matching images */
268         if (!match || dump_all_images) {
269                 fname = screenshot_output_filename("internal-screenshot", 0);
270                 write_surface_as_png(screenshot, fname);
271         }
272
273         free(screenshot);
274
275         printf("Test complete\n");
276         assert(match);
277 }