1
/*
2
 * Copyright 2009 Benjamin Otte
3
 *
4
 * Permission is hereby granted, free of charge, to any person
5
 * obtaining a copy of this software and associated documentation
6
 * files (the "Software"), to deal in the Software without
7
 * restriction, including without limitation the rights to use, copy,
8
 * modify, merge, publish, distribute, sublicense, and/or sell copies
9
 * of the Software, and to permit persons to whom the Software is
10
 * furnished to do so, subject to the following conditions:
11
 *
12
 * The above copyright notice and this permission notice shall be
13
 * included in all copies or substantial portions of the Software.
14
 *
15
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
19
 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
20
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
21
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22
 * SOFTWARE.
23
 *
24
 * Author: Benjamin Otte <otte@gnome.org>
25
 */
26

            
27
#include "cairo-test.h"
28
#include <pthread.h>
29

            
30
#define N_THREADS 8
31

            
32
#define WIDTH 64
33
#define HEIGHT 8
34

            
35
static void *
36
24
draw_thread (void *arg)
37
{
38
24
    cairo_surface_t *surface = arg;
39
    cairo_t *cr;
40
    int x, y;
41

            
42
24
    cr = cairo_create (surface);
43
24
    cairo_surface_destroy (surface);
44

            
45
216
    for (y = 0; y < HEIGHT; y++) {
46
12480
        for (x = 0; x < WIDTH; x++) {
47
12288
            cairo_rectangle (cr, x, y, 1, 1);
48
12288
            cairo_set_source_rgba (cr, 0, 0.75, 0.75, (double) x / WIDTH);
49
12288
            cairo_fill (cr);
50
        }
51
    }
52

            
53
24
    surface = cairo_surface_reference (cairo_get_target (cr));
54
24
    cairo_destroy (cr);
55

            
56
24
    return surface;
57
}
58

            
59
static cairo_test_status_t
60
3
draw (cairo_t *cr, int width, int height)
61
{
62
    pthread_t threads[N_THREADS];
63
3
    cairo_test_status_t test_status = CAIRO_TEST_SUCCESS;
64
    int i;
65

            
66
27
    for (i = 0; i < N_THREADS; i++) {
67
	cairo_surface_t *surface;
68

            
69
24
        surface = cairo_surface_create_similar (cairo_get_target (cr),
70
						CAIRO_CONTENT_COLOR,
71
						WIDTH, HEIGHT);
72
24
        if (pthread_create (&threads[i], NULL, draw_thread, surface) != 0) {
73
	    threads[i] = pthread_self ();
74
            test_status = cairo_test_status_from_status (cairo_test_get_context (cr),
75
							 cairo_surface_status (surface));
76
            cairo_surface_destroy (surface);
77
	    break;
78
        }
79
    }
80

            
81
27
    for (i = 0; i < N_THREADS; i++) {
82
	void *surface;
83

            
84
24
        if (pthread_equal (threads[i], pthread_self ()))
85
            break;
86

            
87
24
        if (pthread_join (threads[i], &surface) == 0) {
88
24
	    cairo_set_source_surface (cr, surface, 0, 0);
89
24
	    cairo_surface_destroy (surface);
90
24
	    cairo_paint (cr);
91

            
92
24
	    cairo_translate (cr, 0, HEIGHT);
93
	} else {
94
            test_status = CAIRO_TEST_FAILURE;
95
	}
96
    }
97

            
98
3
    return test_status;
99
}
100

            
101
1
CAIRO_TEST (pthread_similar,
102
	    "Draw lots of 1x1 rectangles on similar surfaces in lots of threads",
103
	    "threads", /* keywords */
104
	    NULL, /* requirements */
105
	    WIDTH, HEIGHT * N_THREADS,
106
	    NULL, draw)