07f2f0a5ae
A recent kernel with kernel modesetting support is required. Thanks to jsg@ and kettenis@ for their work.
210 lines
5.0 KiB
C
210 lines
5.0 KiB
C
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <X11/Xutil.h> /* for XDestroyImage */
|
|
#include <pixman.h>
|
|
|
|
#include "test.h"
|
|
|
|
static void _render_copy(struct test_target *tt,
|
|
int x, int y, int w, int h,
|
|
uint8_t red, uint8_t green, uint8_t blue, uint8_t alpha)
|
|
{
|
|
XRenderColor color;
|
|
Picture src;
|
|
Pixmap tmp;
|
|
|
|
tmp = XCreatePixmap(tt->dpy->dpy, DefaultRootWindow(tt->dpy->dpy),
|
|
10+w, 20+h, tt->format->depth);
|
|
src = XRenderCreatePicture(tt->dpy->dpy, tmp, tt->format, 0, NULL);
|
|
|
|
/* magenta border */
|
|
color.red = 0xffff;
|
|
color.green = 0;
|
|
color.blue = 0xffff;
|
|
color.alpha = 0xffff;
|
|
XRenderFillRectangle(tt->dpy->dpy, PictOpSrc, src, &color, 0, 0, w+10, h+20);
|
|
|
|
color.red = red * alpha;
|
|
color.green = green * alpha;
|
|
color.blue = blue * alpha;
|
|
color.alpha = alpha << 8 | alpha;
|
|
XRenderFillRectangle(tt->dpy->dpy, PictOpSrc, src, &color, 5, 10, w, h);
|
|
|
|
XRenderComposite(tt->dpy->dpy, PictOpSrc,
|
|
src, 0, tt->picture,
|
|
5, 10,
|
|
0, 0,
|
|
x, y,
|
|
w, h);
|
|
|
|
XRenderFreePicture(tt->dpy->dpy, src);
|
|
XFreePixmap(tt->dpy->dpy, tmp);
|
|
}
|
|
|
|
static void render_copy(struct test_target *real,
|
|
struct test_target *ref)
|
|
{
|
|
int x = rand() % (2*real->width) - real->width;
|
|
int y = rand() % (2*real->height) - real->height;
|
|
int w = rand() % (2*real->width);
|
|
int h = rand() % (2*real->height);
|
|
int red = rand() & 0xff;
|
|
int green = rand() & 0xff;
|
|
int blue = rand() & 0xff;
|
|
int alpha = rand() & 0xff;
|
|
|
|
_render_copy(real, x, y, w, h, red, green, blue, alpha);
|
|
_render_copy(ref, x, y, w, h, red, green, blue, alpha);
|
|
}
|
|
|
|
static void fill_rect(struct test_target *tt,
|
|
int alu, int color,
|
|
int x, int y, int w, int h)
|
|
{
|
|
XGCValues val;
|
|
|
|
val.function = alu;
|
|
val.foreground = color;
|
|
XChangeGC(tt->dpy->dpy, tt->gc, GCFunction | GCForeground, &val);
|
|
|
|
XFillRectangle(tt->dpy->dpy, tt->draw, tt->gc, x, y, w, h);
|
|
}
|
|
|
|
static void clear(struct test_target *tt)
|
|
{
|
|
fill_rect(tt,
|
|
GXcopy, 0,
|
|
0, 0, tt->width, tt->height);
|
|
}
|
|
|
|
static void basic_fill(struct test_target *real,
|
|
struct test_target *ref)
|
|
{
|
|
int x = rand() % (2*real->width) - real->width;
|
|
int y = rand() % (2*real->height) - real->height;
|
|
int w = rand() % (2*real->width);
|
|
int h = rand() % (2*real->height);
|
|
int color = rand();
|
|
int alu = rand() % 16;
|
|
|
|
fill_rect(real, alu, color, x, y, w, h);
|
|
fill_rect(ref, alu, color, x, y, w, h);
|
|
}
|
|
|
|
static void basic_copy(struct test_target *real,
|
|
struct test_target *ref)
|
|
{
|
|
int sx = rand() % (2*real->width) - ref->width;
|
|
int sy = rand() % (2*real->height) - ref->height;
|
|
int dx = rand() % (2*real->width) - ref->width;
|
|
int dy = rand() % (2*real->height) - ref->height;
|
|
int w = rand() % (2*real->width);
|
|
int h = rand() % (2*real->height);
|
|
XGCValues val;
|
|
|
|
val.function = rand() % 16;
|
|
|
|
XChangeGC(real->dpy->dpy, real->gc, GCFunction, &val);
|
|
XCopyArea(real->dpy->dpy,
|
|
real->draw, real->draw, real->gc,
|
|
sx, sy, w, h, dx, dy);
|
|
|
|
XChangeGC(ref->dpy->dpy, ref->gc, GCFunction, &val);
|
|
XCopyArea(ref->dpy->dpy,
|
|
ref->draw, ref->draw, ref->gc,
|
|
sx, sy, w, h, dx, dy);
|
|
}
|
|
|
|
static void _put(struct test_target *tt,
|
|
int x, int y, int w,int h, int color, int alu)
|
|
{
|
|
XImage image;
|
|
XGCValues val;
|
|
|
|
val.function = alu;
|
|
|
|
test_init_image(&image, &tt->dpy->shm, tt->format, w, h);
|
|
pixman_fill((uint32_t*)image.data,
|
|
image.bytes_per_line/sizeof(uint32_t),
|
|
image.bits_per_pixel,
|
|
0, 0, w, h, color);
|
|
|
|
XChangeGC(tt->dpy->dpy, tt->gc, GCFunction, &val);
|
|
if (rand() & 1) {
|
|
XShmPutImage(tt->dpy->dpy, tt->draw, tt->gc, &image,
|
|
0, 0, x, y, w, h, 0);
|
|
XSync(tt->dpy->dpy, 1);
|
|
} else {
|
|
XPutImage(tt->dpy->dpy, tt->draw, tt->gc, &image,
|
|
0, 0, x, y, w, h);
|
|
}
|
|
}
|
|
|
|
static void basic_put(struct test_target *real,
|
|
struct test_target *ref)
|
|
{
|
|
int x = rand() % (2*real->width) - real->width;
|
|
int y = rand() % (2*real->height) - real->height;
|
|
int w = rand() % real->width;
|
|
int h = rand() % real->height;
|
|
int color = rand();
|
|
int alu = rand() % 16;
|
|
|
|
_put(real, x, y, w, h, color, alu);
|
|
_put(ref, x, y, w, h, color, alu);
|
|
}
|
|
|
|
static void rect_tests(struct test *test, int iterations, enum target target)
|
|
{
|
|
struct test_target real, ref;
|
|
void (* const ops[])(struct test_target *, struct test_target *) = {
|
|
basic_copy,
|
|
basic_fill,
|
|
basic_put,
|
|
render_copy,
|
|
};
|
|
int n;
|
|
|
|
printf("Running mixed ops stress against %s: ",
|
|
test_target_name(target));
|
|
fflush(stdout);
|
|
|
|
test_target_create_render(&test->real, target, &real);
|
|
test_target_create_render(&test->ref, target, &ref);
|
|
|
|
clear(&real);
|
|
clear(&ref);
|
|
|
|
for (n = 0; n < iterations; n++)
|
|
ops[rand() % ARRAY_SIZE(ops)](&real, &ref);
|
|
|
|
test_compare(test,
|
|
real.draw, real.format,
|
|
ref.draw, ref.format,
|
|
0, 0, real.width, real.height,
|
|
"");
|
|
|
|
printf("passed [%d iterations]\n", n);
|
|
|
|
test_target_destroy_render(&test->real, &real);
|
|
test_target_destroy_render(&test->ref, &ref);
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
struct test test;
|
|
int i;
|
|
|
|
test_init(&test, argc, argv);
|
|
|
|
for (i = 0; i <= DEFAULT_ITERATIONS; i++) {
|
|
int iterations = 1 << i;
|
|
rect_tests(&test, iterations, 0);
|
|
rect_tests(&test, iterations, 1);
|
|
}
|
|
|
|
return 0;
|
|
}
|