/* * Copyright (c) 2021, Leon Albrecht * Copyright (c) 2022, Jelle Raaijmakers * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #ifdef __serenity__ # define REFERENCE_IMAGE_DIR "/usr/Tests/LibGL/reference-images" #else # define REFERENCE_IMAGE_DIR "reference-images" #endif #define SAVE_OUTPUT false static NonnullOwnPtr create_testing_context(int width, int height) { auto bitmap = MUST(Gfx::Bitmap::try_create(Gfx::BitmapFormat::BGRx8888, { width, height })); auto context = GL::create_context(*bitmap); GL::make_context_current(context); // Assume some defaults for our testing contexts glFrontFace(GL_CCW); glCullFace(GL_BACK); glEnable(GL_CULL_FACE); glClearColor(0.0f, 0.0f, 0.0f, 1.0f); glClear(GL_COLOR_BUFFER_BIT); return context; } static void expect_bitmap_equals_reference(Gfx::Bitmap const& bitmap, StringView test_name) { auto reference_filename = String::formatted("{}.qoi", test_name); if constexpr (SAVE_OUTPUT) { auto target_path = LexicalPath("/home/anon").append(reference_filename); auto qoi_buffer = Gfx::QOIWriter::encode(bitmap); auto qoi_output_stream = MUST(Core::OutputFileStream::open(target_path.string())); auto number_of_bytes_written = qoi_output_stream.write(qoi_buffer); qoi_output_stream.close(); EXPECT_EQ(number_of_bytes_written, qoi_buffer.size()); } auto reference_image_path = String::formatted(REFERENCE_IMAGE_DIR "/{}", reference_filename); auto reference_bitmap = MUST(Gfx::Bitmap::try_load_from_file(reference_image_path)); EXPECT_EQ(reference_bitmap->visually_equals(bitmap), true); } TEST_CASE(0001_simple_triangle) { auto context = create_testing_context(64, 64); glBegin(GL_TRIANGLES); glColor3f(1, 1, 1); glVertex2f(0, 1); glVertex2f(-1, -1); glVertex2f(1, -1); glEnd(); EXPECT_EQ(glGetError(), 0u); context->present(); expect_bitmap_equals_reference(context->frontbuffer(), "0001_simple_triangle"sv); } TEST_CASE(0002_quad_color_interpolation) { auto context = create_testing_context(64, 64); glBegin(GL_QUADS); glColor3f(1, 0, 0); glVertex2i(-1, -1); glColor3f(0, 1, 0); glVertex2i(1, -1); glColor3f(0, 0, 1); glVertex2i(1, 1); glColor3f(1, 0, 1); glVertex2i(-1, 1); glEnd(); EXPECT_EQ(glGetError(), 0u); context->present(); expect_bitmap_equals_reference(context->frontbuffer(), "0002_quad_color_interpolation"sv); } TEST_CASE(0003_rect_w_coordinate_regression) { auto context = create_testing_context(64, 64); glEnable(GL_DEPTH_TEST); glClear(GL_DEPTH_BUFFER_BIT); glColor3f(0, 1, 0); glRectf(-0.5f, -0.5f, 0.5f, 0.5f); glBegin(GL_TRIANGLES); glColor3f(1, 0, 0); glVertex2i(-1, -1); glVertex2i(1, -1); glVertex2i(-1, 1); glEnd(); EXPECT_EQ(glGetError(), 0u); context->present(); expect_bitmap_equals_reference(context->frontbuffer(), "0003_rect_w_coordinate_regression"sv); } TEST_CASE(0004_points) { auto context = create_testing_context(64, 64); // Aliased points for (size_t i = 0; i < 3; ++i) { glPointSize(1.f + i); glBegin(GL_POINTS); glVertex2f(-.5f + i * .5f, .5f); glEnd(); } // Anti-aliased points glEnable(GL_POINT_SMOOTH); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); for (size_t i = 0; i < 3; ++i) { glPointSize(3.f - i); glBegin(GL_POINTS); glVertex2f(-.5f + i * .5f, -.5f); glEnd(); } EXPECT_EQ(glGetError(), 0u); context->present(); expect_bitmap_equals_reference(context->frontbuffer(), "0004_points"sv); } TEST_CASE(0005_lines_antialiased) { auto context = create_testing_context(64, 64); // Draw anti-aliased lines glEnable(GL_LINE_SMOOTH); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glBegin(GL_LINES); for (size_t i = 0; i < 6; ++i) { glVertex2f(-.9f, .25f - i * .1f); glVertex2f(.9f, .9f - i * .36f); } glEnd(); EXPECT_EQ(glGetError(), 0u); context->present(); expect_bitmap_equals_reference(context->frontbuffer(), "0005_lines"sv); }