/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
/*
Copyright (C) 2009-2015 Red Hat, Inc.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, see .
*/
/* Do repeated updates to the same rectangle to trigger stream creation.
*
* TODO: check that stream actually starts programatically (maybe stap?)
* TODO: stop updating same rect, check (prog) that stream stops
*/
#include
#include
#include
#include
#include
#include
#include
#include
#include "test_display_base.h"
static int sized;
static int render_last_frame;
static void create_overlay(Command *command , int width, int height)
{
CommandDrawBitmap *cmd = &command->bitmap;
uint32_t *dst;
cmd->surface_id = 0;
cmd->bbox.left = 0;
cmd->bbox.top = 0;
cmd->bbox.right = width;
cmd->bbox.bottom = height;
cmd->num_clip_rects = 0;
cmd->bitmap = g_malloc(width * height * 4 );
dst = (uint32_t *)cmd->bitmap;
for (int i = 0; i < width * height; i++, dst++) {
*dst = 0x8B008B;
}
}
#define NUM_COMMANDS 2000
#define SIZED_INTERVAL 100
#define OVERLAY_FRAME 500
#define OVERLAY_WIDTH 200
#define OVERLAY_HEIGHT 200
/*
* Create a frame in a stream that displays a row that moves
* from the top to the bottom repeatedly.
* Upon the OVERLAY_FRAME-th, a drawable is created on top of a part of the stream,
* and from then on, all the stream frames has a clipping that keeps this drawable
* visible, and in addition a clipping_factor is subtracted from the right limit of their clipping.
* If sized=TRUE, a higher and wider frame than the original stream is created every SIZED_INTERVAL.
* The sized frames can be distinguished by a change in the color of the top and bottom limits of the
* surface.
*/
static void create_clipped_frame(Test *test, Command *command, int clipping_factor)
{
static int count = 0;
CommandDrawBitmap *cmd = &command->bitmap;
int max_height = test->height;
int max_width = test->width;
int width;
int height;
int cur_line, end_line;
uint32_t *dst;
count++;
if (count == NUM_COMMANDS) {
count = 0;
}
if (count == OVERLAY_FRAME) {
create_overlay(command, OVERLAY_WIDTH, OVERLAY_HEIGHT);
return;
}
cmd->surface_id = 0;
cmd->bbox.left = 0;
cmd->bbox.right = max_width - 50;
assert(max_height > 600);
cmd->bbox.top = 50;
cmd->bbox.bottom = max_height - 50;
height = cmd->bbox.bottom - cmd->bbox.top;
width = cmd->bbox.right - cmd->bbox.left;
cur_line = (height/30)*(count % 30);
end_line = cur_line + (height/30);
if (end_line >= height || height - end_line < 8) {
end_line = height;
}
if (sized && count % SIZED_INTERVAL == 0) {
cmd->bbox.top = 0;
cmd->bbox.bottom = max_height;
cmd->bbox.left = 0;
cmd->bbox.right = max_width;
height = max_height;
width = max_width;
cur_line += 50;
end_line += 50;
}
cmd->bitmap = g_malloc(width*height*4);
memset(cmd->bitmap, 0xff, width*height*4);
dst = (uint32_t *)(cmd->bitmap + cur_line*width*4);
for (; cur_line < end_line; cur_line++) {
int col;
for (col = 0; col < width; col++, dst++) {
*dst = 0x00FF00;
}
}
if (sized && count % SIZED_INTERVAL == 0) {
int i;
uint32_t color = 0xffffff & rand();
dst = (uint32_t *)cmd->bitmap;
for (i = 0; i < 50*width; i++, dst++) {
*dst = color;
}
dst = ((uint32_t *)(cmd->bitmap + (height - 50)*4*width));
for (i = 0; i < 50*width; i++, dst++) {
*dst = color;
}
}
if (count < OVERLAY_FRAME) {
cmd->num_clip_rects = 0;
} else {
cmd->num_clip_rects = 2;
cmd->clip_rects = calloc(sizeof(QXLRect), 2);
cmd->clip_rects[0].left = OVERLAY_WIDTH;
cmd->clip_rects[0].top = cmd->bbox.top;
cmd->clip_rects[0].right = cmd->bbox.right - clipping_factor;
cmd->clip_rects[0].bottom = OVERLAY_HEIGHT;
cmd->clip_rects[1].left = cmd->bbox.left;
cmd->clip_rects[1].top = OVERLAY_HEIGHT;
cmd->clip_rects[1].right = cmd->bbox.right - clipping_factor;
cmd->clip_rects[1].bottom = cmd->bbox.bottom;
}
}
static void create_frame1(Test *test, Command *command)
{
create_clipped_frame(test, command, 0);
}
void create_frame2(Test *test, Command *command)
{
create_clipped_frame(test, command, 200);
}
typedef void (*create_frame_cb)(Test *test, Command *command);
/*
* The test contains two types of streams. The first stream doesn't
* have a clipping besides the on that the display the overlay drawable.
* Expected result: If render_last_frame=false, the last frame should
* be sent losslessly. Otherwise, red_update_area should be called, and the
* stream is upgraded by a screenshot.
*
* In the second test, the stream clip changes in the middle (becomes smaller).
* Expected result: red_update_area should is, and the
* stream is upgraded by a screenshot (including lossy areas that belong to old frames
* and were never covered by a lossless drawable).
*
*/
static void get_stream_commands(Command *commands, int num_commands,
create_frame_cb cb)
{
int i;
commands[0].command = DESTROY_PRIMARY;
commands[1].command = CREATE_PRIMARY;
commands[1].create_primary.width = 1280;
commands[1].create_primary.height = 1024;
commands[num_commands - 1].command = SLEEP;
commands[num_commands - 1].sleep.secs = 20;
for (i = 2; i < num_commands - 1; i++) {
commands[i].command = SIMPLE_DRAW_BITMAP;
commands[i].cb = cb;
}
if (render_last_frame) {
commands[num_commands - 2].command = SIMPLE_UPDATE;
}
}
static void get_commands(Command **commands, int *num_commands)
{
*num_commands = NUM_COMMANDS * 2;
*commands = calloc(sizeof(Command), *num_commands);
get_stream_commands(*commands, NUM_COMMANDS, create_frame1);
get_stream_commands((*commands) + NUM_COMMANDS, NUM_COMMANDS, create_frame2);
}
int main(int argc, char **argv)
{
SpiceCoreInterface *core;
Command *commands;
int num_commands;
int i;
Test *test;
spice_test_config_parse_args(argc, argv);
sized = 0;
for (i = 1 ; i < argc; ++i) {
if (strcmp(argv[i], "sized") == 0) {
sized = 1;
}
/* render last frame */
if (strcmp(argv[i], "render") == 0) {
render_last_frame = 1;
}
}
srand(time(NULL));
// todo: add args list of test numbers with explenations
core = basic_event_loop_init();
test = test_new(core);
spice_server_set_streaming_video(test->server, SPICE_STREAM_VIDEO_ALL);
test_add_display_interface(test);
get_commands(&commands, &num_commands);
test_set_command_list(test, commands, num_commands);
basic_event_loop_mainloop();
free(commands);
return 0;
}