/*
* Copyright © 2013 Intel Corporation
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice (including the next
* paragraph) shall be included in all copies or substantial portions of the
* Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*/
#include <gtest/gtest.h>
#include <signal.h>
#include <setjmp.h>
#include "glxclient.h"
#include "glx_error.h"
extern bool GetGLXScreenConfigs_called;
extern struct glx_screen *psc;
struct attribute_test_vector {
const char *string;
int value;
};
#define E(x) { # x, x }
static bool got_sigsegv;
static jmp_buf jmp;
static void
sigsegv_handler(int sig)
{
(void) sig;
got_sigsegv = true;
longjmp(jmp, 1);
}
static bool query_renderer_string_called = false;
static bool query_renderer_integer_called = false;
static int
fake_query_renderer_integer(struct glx_screen *psc, int attribute,
unsigned int *value)
{
(void) psc;
(void) attribute;
(void) value;
query_renderer_integer_called = true;
return -1;
}
static int
fake_query_renderer_string(struct glx_screen *psc, int attribute,
const char **value)
{
(void) psc;
(void) attribute;
(void) value;
query_renderer_string_called = true;
return -1;
}
struct glx_screen_vtable fake_vtable = {
NULL,
NULL,
fake_query_renderer_integer,
fake_query_renderer_string
};
class query_renderer_string_test : public ::testing::Test {
public:
virtual void SetUp();
virtual void TearDown();
struct glx_screen scr;
struct sigaction sa;
struct sigaction old_sa;
Display dpy;
};
class query_renderer_integer_test : public query_renderer_string_test {
};
void query_renderer_string_test::SetUp()
{
memset(&scr, 0, sizeof(scr));
scr.vtable = &fake_vtable;
psc = &scr;
got_sigsegv = false;
sa.sa_handler = sigsegv_handler;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
sigaction(SIGSEGV, &sa, &old_sa);
}
void query_renderer_string_test::TearDown()
{
sigaction(SIGSEGV, &old_sa, NULL);
}
/**
* glXQueryRendererStringMESA will return \c NULL if the query_render_string
* vtable entry is \c NULL. It will also not segfault.
*/
TEST_F(query_renderer_string_test, null_query_render_string)
{
struct glx_screen_vtable vtable = {
NULL,
NULL,
NULL,
NULL
};
scr.vtable = &vtable;
if (setjmp(jmp) == 0) {
const char *str =
glXQueryRendererStringMESA(&dpy, 0, 0, GLX_RENDERER_VENDOR_ID_MESA);
EXPECT_EQ((char *)0, str);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryRendererStringMESA will not call the screen query_render_string
* function with an invalid GLX enum value, and it will return NULL.
*/
TEST_F(query_renderer_string_test, invalid_attribute)
{
static const attribute_test_vector invalid_attributes[] = {
/* These values are just plain invalid for use with this extension.
*/
E(0),
E(GLX_VENDOR),
E(GLX_VERSION),
E(GLX_EXTENSIONS),
E(GLX_RENDERER_VENDOR_ID_MESA + 0x10000),
E(GLX_RENDERER_DEVICE_ID_MESA + 0x10000),
/* These enums are part of the extension, but they are not allowed for
* the string query.
*/
E(GLX_RENDERER_VERSION_MESA),
E(GLX_RENDERER_ACCELERATED_MESA),
E(GLX_RENDERER_VIDEO_MEMORY_MESA),
E(GLX_RENDERER_UNIFIED_MEMORY_ARCHITECTURE_MESA),
E(GLX_RENDERER_PREFERRED_PROFILE_MESA),
E(GLX_RENDERER_OPENGL_CORE_PROFILE_VERSION_MESA),
E(GLX_RENDERER_OPENGL_COMPATIBILITY_PROFILE_VERSION_MESA),
E(GLX_RENDERER_OPENGL_ES_PROFILE_VERSION_MESA),
E(GLX_RENDERER_OPENGL_ES2_PROFILE_VERSION_MESA),
E(GLX_RENDERER_ID_MESA),
};
for (unsigned i = 0; i < ARRAY_SIZE(invalid_attributes); i++) {
query_renderer_integer_called = false;
query_renderer_string_called = false;
const char *str =
glXQueryRendererStringMESA(&dpy, 0, 0, invalid_attributes[i].value);
EXPECT_EQ((char *)0, str) << invalid_attributes[i].string;
EXPECT_FALSE(query_renderer_integer_called)
<< invalid_attributes[i].string;
EXPECT_FALSE(query_renderer_string_called)
<< invalid_attributes[i].string;
}
}
/**
* glXQueryRendererStringMESA will not call GetGLXScreenConfigs if the display
* pointer is \c NULL. It will also not segfault.
*/
TEST_F(query_renderer_string_test, null_display_pointer)
{
if (setjmp(jmp) == 0) {
GetGLXScreenConfigs_called = false;
const char *str =
glXQueryRendererStringMESA(NULL, 0, 0, GLX_RENDERER_VENDOR_ID_MESA);
EXPECT_EQ((char *)0, str);
EXPECT_FALSE(GetGLXScreenConfigs_called);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryRendererStringMESA will return error if GetGLXScreenConfigs returns
* NULL. It will also not segfault.
*/
TEST_F(query_renderer_string_test, null_screen_pointer)
{
psc = NULL;
if (setjmp(jmp) == 0) {
GetGLXScreenConfigs_called = false;
const char *str =
glXQueryRendererStringMESA(&dpy, 0, 0, GLX_RENDERER_VENDOR_ID_MESA);
EXPECT_EQ((char *)0, str);
EXPECT_TRUE(GetGLXScreenConfigs_called);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryRendererStringMESA will not call the screen query_render_string
* function if the renderer is invalid, and it will return NULL.
*/
TEST_F(query_renderer_string_test, invalid_renderer_index)
{
static const int invalid_renderer_indices[] = {
-1,
1,
999,
};
if (setjmp(jmp) == 0) {
for (unsigned i = 0; i < ARRAY_SIZE(invalid_renderer_indices); i++) {
const char *str =
glXQueryRendererStringMESA(&dpy, 0,
invalid_renderer_indices[i],
GLX_RENDERER_VENDOR_ID_MESA);
EXPECT_EQ((char *)0, str) << invalid_renderer_indices[i];
EXPECT_FALSE(query_renderer_integer_called)
<< invalid_renderer_indices[i];
EXPECT_FALSE(query_renderer_string_called)
<< invalid_renderer_indices[i];
}
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryCurrentRendererStringMESA will return error if there is no context
* current. It will also not segfault.
*/
TEST_F(query_renderer_string_test, no_current_context)
{
if (setjmp(jmp) == 0) {
const char *str =
glXQueryCurrentRendererStringMESA(GLX_RENDERER_VENDOR_ID_MESA);
EXPECT_EQ((char *)0, str);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryCurrentRendererIntegerMESA will return \c NULL if the
* query_render_string vtable entry is \c NULL. It will also not segfault.
*/
TEST_F(query_renderer_integer_test, null_query_render_string)
{
struct glx_screen_vtable vtable = {
NULL,
NULL,
NULL,
NULL
};
scr.vtable = &vtable;
if (setjmp(jmp) == 0) {
unsigned value = 0xDEADBEEF;
Bool success = glXQueryRendererIntegerMESA(&dpy, 0, 0,
GLX_RENDERER_VENDOR_ID_MESA,
&value);
EXPECT_FALSE(success);
EXPECT_EQ(0xDEADBEEF, value);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryCurrentRendererIntegerMESA will not call the screen
* query_render_string function with an invalid GLX enum value, and it will
* return NULL.
*/
TEST_F(query_renderer_integer_test, invalid_attribute)
{
static const attribute_test_vector invalid_attributes[] = {
/* These values are just plain invalid for use with this extension.
*/
E(0),
E(GLX_VENDOR),
E(GLX_VERSION),
E(GLX_EXTENSIONS),
E(GLX_RENDERER_VENDOR_ID_MESA + 0x10000),
E(GLX_RENDERER_DEVICE_ID_MESA + 0x10000),
E(GLX_RENDERER_VERSION_MESA + 0x10000),
E(GLX_RENDERER_ACCELERATED_MESA + 0x10000),
E(GLX_RENDERER_VIDEO_MEMORY_MESA + 0x10000),
E(GLX_RENDERER_UNIFIED_MEMORY_ARCHITECTURE_MESA + 0x10000),
E(GLX_RENDERER_PREFERRED_PROFILE_MESA + 0x10000),
E(GLX_RENDERER_OPENGL_CORE_PROFILE_VERSION_MESA + 0x10000),
E(GLX_RENDERER_OPENGL_COMPATIBILITY_PROFILE_VERSION_MESA + 0x10000),
E(GLX_RENDERER_OPENGL_ES_PROFILE_VERSION_MESA + 0x10000),
E(GLX_RENDERER_OPENGL_ES2_PROFILE_VERSION_MESA + 0x10000),
E(GLX_RENDERER_ID_MESA + 0x10000),
};
for (unsigned i = 0; i < ARRAY_SIZE(invalid_attributes); i++) {
query_renderer_integer_called = false;
query_renderer_string_called = false;
unsigned value = 0xDEADBEEF;
Bool success =
glXQueryRendererIntegerMESA(&dpy, 0, 0,
invalid_attributes[i].value,
&value);
EXPECT_FALSE(success) << invalid_attributes[i].string;
EXPECT_EQ(0xDEADBEEF, value) << invalid_attributes[i].string;
EXPECT_FALSE(query_renderer_integer_called)
<< invalid_attributes[i].string;
EXPECT_FALSE(query_renderer_string_called)
<< invalid_attributes[i].string;
}
}
/**
* glXQueryCurrentRendererIntegerMESA will not call GetGLXScreenConfigs if the
* display pointer is \c NULL. It will also not segfault.
*/
TEST_F(query_renderer_integer_test, null_display_pointer)
{
if (setjmp(jmp) == 0) {
GetGLXScreenConfigs_called = false;
unsigned value = 0xDEADBEEF;
Bool success =
glXQueryRendererIntegerMESA(NULL, 0, 0, GLX_RENDERER_VENDOR_ID_MESA,
&value);
EXPECT_FALSE(success);
EXPECT_EQ(0xDEADBEEF, value);
EXPECT_FALSE(GetGLXScreenConfigs_called);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryCurrentRendererIntegerMESA will return error if GetGLXScreenConfigs
* returns NULL. It will also not segfault.
*/
TEST_F(query_renderer_integer_test, null_screen_pointer)
{
psc = NULL;
if (setjmp(jmp) == 0) {
GetGLXScreenConfigs_called = false;
unsigned value = 0xDEADBEEF;
Bool success =
glXQueryRendererIntegerMESA(&dpy, 0, 0, GLX_RENDERER_VENDOR_ID_MESA,
&value);
EXPECT_FALSE(success);
EXPECT_EQ(0xDEADBEEF, value);
EXPECT_TRUE(GetGLXScreenConfigs_called);
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryRendererIntegerMESA will not call the screen query_render_integer
* function if the renderer is invalid, and it will return NULL.
*/
TEST_F(query_renderer_integer_test, invalid_renderer_index)
{
static const int invalid_renderer_indices[] = {
-1,
1,
999,
};
if (setjmp(jmp) == 0) {
for (unsigned i = 0; i < ARRAY_SIZE(invalid_renderer_indices); i++) {
unsigned value = 0xDEADBEEF;
Bool success =
glXQueryRendererIntegerMESA(&dpy, 0,
invalid_renderer_indices[i],
GLX_RENDERER_VENDOR_ID_MESA,
&value);
EXPECT_FALSE(success) << invalid_renderer_indices[i];
EXPECT_EQ(0xDEADBEEF, value) << invalid_renderer_indices[i];
EXPECT_FALSE(query_renderer_integer_called)
<< invalid_renderer_indices[i];
EXPECT_FALSE(query_renderer_string_called)
<< invalid_renderer_indices[i];
}
} else {
EXPECT_FALSE(got_sigsegv);
}
}
/**
* glXQueryCurrentRendererIntegerMESA will return error if there is no context
* current. It will also not segfault.
*/
TEST_F(query_renderer_integer_test, no_current_context)
{
if (setjmp(jmp) == 0) {
unsigned value = 0xDEADBEEF;
Bool success =
glXQueryCurrentRendererIntegerMESA(GLX_RENDERER_VENDOR_ID_MESA,
&value);
EXPECT_FALSE(success);
EXPECT_EQ(0xDEADBEEF, value);
} else {
EXPECT_FALSE(got_sigsegv);
}
}