/* * Copyright (c) 2021, Stephan Unverwerth * Copyright (c) 2021-2022, Jesse Buhagiar * Copyright (c) 2022-2023, Jelle Raaijmakers * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace GL { #define APPEND_TO_CALL_LIST_AND_RETURN_IF_NEEDED(name, ...) \ if (should_append_to_listing()) { \ append_to_listing<&GLContext::name>(__VA_ARGS__); \ if (!should_execute_after_appending_to_listing()) \ return; \ } #define APPEND_TO_CALL_LIST_WITH_ARG_AND_RETURN_IF_NEEDED(name, arg) \ if (should_append_to_listing()) { \ auto ptr = store_in_listing(arg); \ append_to_listing<&GLContext::name>(*ptr); \ if (!should_execute_after_appending_to_listing()) \ return; \ } #define RETURN_WITH_ERROR_IF(condition, error) \ if (condition) { \ dbgln_if(GL_DEBUG, "{}(): error {:#x}", __func__, error); \ if (m_error == GL_NO_ERROR) \ m_error = error; \ return; \ } #define RETURN_VALUE_WITH_ERROR_IF(condition, error, return_value) \ if (condition) { \ dbgln_if(GL_DEBUG, "{}(): error {:#x}", __func__, error); \ if (m_error == GL_NO_ERROR) \ m_error = error; \ return return_value; \ } constexpr size_t MODELVIEW_MATRIX_STACK_LIMIT = 64; constexpr size_t PROJECTION_MATRIX_STACK_LIMIT = 8; constexpr size_t TEXTURE_MATRIX_STACK_LIMIT = 8; struct ContextParameter { GLenum type; bool is_capability { false }; u8 count { 1 }; union { bool boolean_value; GLint integer_value; GLint integer_list[4]; GLdouble double_value; GLdouble double_list[4]; } value; }; struct VertexAttribPointer { GLint size { 4 }; GLenum type { GL_FLOAT }; bool normalize; GLsizei stride { 0 }; void const* pointer { 0 }; }; enum Face { Front = 0, Back = 1, }; enum class PackingType { Pack, Unpack, }; class GLContext final { public: GLContext(RefPtr driver, NonnullOwnPtr, Gfx::Bitmap&); ~GLContext(); NonnullRefPtr frontbuffer() const { return m_frontbuffer; }; void present(); void gl_begin(GLenum mode); void gl_clear(GLbitfield mask); void gl_clear_color(GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha); void gl_clear_depth(GLfloat depth); void gl_clear_stencil(GLint s); void gl_color(GLfloat r, GLfloat g, GLfloat b, GLfloat a); void gl_delete_textures(GLsizei n, GLuint const* textures); void gl_end(); void gl_frustum(GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble near_val, GLdouble far_val); void gl_gen_textures(GLsizei n, GLuint* textures); GLenum gl_get_error(); GLubyte const* gl_get_string(GLenum name); void gl_load_identity(); void gl_load_matrix(FloatMatrix4x4 const& matrix); void gl_matrix_mode(GLenum mode); void gl_ortho(GLdouble left, GLdouble right, GLdouble bottom, GLdouble top, GLdouble near_val, GLdouble far_val); void gl_push_matrix(); void gl_pop_matrix(); void gl_mult_matrix(FloatMatrix4x4 const& matrix); void gl_rotate(GLfloat angle, GLfloat x, GLfloat y, GLfloat z); void gl_scale(GLfloat x, GLfloat y, GLfloat z); void gl_translate(GLfloat x, GLfloat y, GLfloat z); void gl_vertex(GLfloat x, GLfloat y, GLfloat z, GLfloat w); void gl_viewport(GLint x, GLint y, GLsizei width, GLsizei height); void gl_enable(GLenum); void gl_disable(GLenum); GLboolean gl_is_enabled(GLenum); void gl_front_face(GLenum); void gl_cull_face(GLenum); GLuint gl_gen_lists(GLsizei range); void gl_call_list(GLuint list); void gl_call_lists(GLsizei n, GLenum type, void const* lists); void gl_delete_lists(GLuint list, GLsizei range); void gl_list_base(GLuint base); void gl_end_list(void); void gl_new_list(GLuint list, GLenum mode); GLboolean gl_is_list(GLuint list); void gl_flush(); void gl_finish(); void gl_blend_func(GLenum src_factor, GLenum dst_factor); void gl_shade_model(GLenum mode); void gl_alpha_func(GLenum func, GLclampf ref); void gl_hint(GLenum target, GLenum mode); void gl_read_buffer(GLenum mode); void gl_draw_buffer(GLenum buffer); void gl_read_pixels(GLint x, GLint y, GLsizei width, GLsizei height, GLenum format, GLenum type, GLvoid* pixels); void gl_tex_image_2d(GLenum target, GLint level, GLint internal_format, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, GLvoid const* data); void gl_tex_sub_image_2d(GLenum target, GLint level, GLint xoffset, GLint yoffset, GLsizei width, GLsizei height, GLenum format, GLenum type, GLvoid const* data); void gl_tex_parameter(GLenum target, GLenum pname, GLfloat param); void gl_tex_parameterfv(GLenum target, GLenum pname, GLfloat const* params); void gl_tex_coord(GLfloat s, GLfloat t, GLfloat r, GLfloat q); void gl_multi_tex_coord(GLenum target, GLfloat s, GLfloat t, GLfloat r, GLfloat q); void gl_tex_env(GLenum target, GLenum pname, FloatVector4 params); void gl_tex_envv(GLenum target, GLenum pname, void const* params, GLenum type); void gl_bind_texture(GLenum target, GLuint texture); GLboolean gl_is_texture(GLuint texture); void gl_active_texture(GLenum texture); void gl_depth_mask(GLboolean flag); void gl_enable_client_state(GLenum cap); void gl_disable_client_state(GLenum cap); void gl_client_active_texture(GLenum target); void gl_vertex_pointer(GLint size, GLenum type, GLsizei stride, void const* pointer); void gl_color_pointer(GLint size, GLenum type, GLsizei stride, void const* pointer); void gl_tex_coord_pointer(GLint size, GLenum type, GLsizei stride, void const* pointer); void gl_draw_arrays(GLenum mode, GLint first, GLsizei count); void gl_draw_elements(GLenum mode, GLsizei count, GLenum type, void const* indices); void gl_draw_pixels(GLsizei width, GLsizei height, GLenum format, GLenum type, void const* data); void gl_color_mask(GLboolean red, GLboolean green, GLboolean blue, GLboolean alpha); void gl_get_booleanv(GLenum pname, GLboolean* data); void gl_get_doublev(GLenum pname, GLdouble* params); void gl_get_floatv(GLenum pname, GLfloat* params); void gl_get_integerv(GLenum pname, GLint* data); void gl_depth_range(GLdouble min, GLdouble max); void gl_depth_func(GLenum func); void gl_polygon_mode(GLenum face, GLenum mode); void gl_polygon_offset(GLfloat factor, GLfloat units); void gl_fogfv(GLenum pname, GLfloat const* params); void gl_fogf(GLenum pname, GLfloat param); void gl_fogi(GLenum pname, GLint param); void gl_pixel_storei(GLenum pname, GLint param); void gl_scissor(GLint x, GLint y, GLsizei width, GLsizei height); void gl_stencil_func_separate(GLenum face, GLenum func, GLint ref, GLuint mask); void gl_stencil_mask_separate(GLenum face, GLuint mask); void gl_stencil_op_separate(GLenum face, GLenum sfail, GLenum dpfail, GLenum dppass); void gl_normal(GLfloat nx, GLfloat ny, GLfloat nz); void gl_normal_pointer(GLenum type, GLsizei stride, void const* pointer); void gl_raster_pos(GLfloat x, GLfloat y, GLfloat z, GLfloat w); void gl_line_width(GLfloat width); void gl_push_attrib(GLbitfield mask); void gl_pop_attrib(); void gl_light_model(GLenum pname, GLfloat x, GLfloat y, GLfloat z, GLfloat w); void gl_light_modelv(GLenum pname, void const* params, GLenum type); void gl_bitmap(GLsizei width, GLsizei height, GLfloat xorig, GLfloat yorig, GLfloat xmove, GLfloat ymove, GLubyte const* bitmap); void gl_copy_tex_image_2d(GLenum target, GLint level, GLenum internalformat, GLint x, GLint y, GLsizei width, GLsizei height, GLint border); void gl_get_tex_image(GLenum target, GLint level, GLenum format, GLenum type, void* pixels); void gl_get_tex_parameter_integerv(GLenum target, GLint level, GLenum pname, GLint* params); void gl_rect(GLdouble x1, GLdouble y1, GLdouble x2, GLdouble y2); void gl_tex_gen(GLenum coord, GLenum pname, GLint param); void gl_tex_gen_floatv(GLenum coord, GLenum pname, GLfloat const* params); void gl_lightf(GLenum light, GLenum pname, GLfloat param); void gl_lightfv(GLenum light, GLenum pname, GLfloat const* params); void gl_lightiv(GLenum light, GLenum pname, GLint const* params); void gl_materialf(GLenum face, GLenum pname, GLfloat param); void gl_materialfv(GLenum face, GLenum pname, GLfloat const* params); void gl_materialiv(GLenum face, GLenum pname, GLint const* params); void gl_color_material(GLenum face, GLenum mode); void gl_get_light(GLenum light, GLenum pname, void* params, GLenum type); void gl_get_material(GLenum face, GLenum pname, void* params, GLenum type); void gl_clip_plane(GLenum plane, GLdouble const* equation); void gl_get_clip_plane(GLenum plane, GLdouble* equation); void gl_array_element(GLint i); void gl_copy_tex_sub_image_2d(GLenum target, GLint level, GLint xoffset, GLint yoffset, GLint x, GLint y, GLsizei width, GLsizei height); void gl_point_size(GLfloat size); void gl_bind_buffer(GLenum target, GLuint buffer); void gl_buffer_data(GLenum target, GLsizeiptr size, void const* data, GLenum usage); void gl_buffer_sub_data(GLenum target, GLintptr offset, GLsizeiptr size, void const* data); void gl_delete_buffers(GLsizei n, GLuint const* buffers); void gl_gen_buffers(GLsizei n, GLuint* buffers); GLuint gl_create_shader(GLenum shader_type); void gl_delete_shader(GLuint shader); void gl_shader_source(GLuint shader, GLsizei count, GLchar const** string, GLint const* length); void gl_compile_shader(GLuint shader); void gl_get_shader(GLuint shader, GLenum pname, GLint* params); GLuint gl_create_program(); void gl_delete_program(GLuint program); void gl_attach_shader(GLuint program, GLuint shader); void gl_link_program(GLuint program); void gl_use_program(GLuint program); void gl_get_program(GLuint program, GLenum pname, GLint* params); private: void sync_device_config(); void sync_device_sampler_config(); void sync_device_texture_units(); void sync_light_state(); void sync_stencil_configuration(); void sync_clip_planes(); ErrorOr build_extension_string(); template T* store_in_listing(T value) { VERIFY(m_current_listing_index.has_value()); auto& listing = m_current_listing_index->listing; listing.saved_arguments.empend(make(move(value))); return listing.saved_arguments.last()->template get_pointer(); } template void append_to_listing(Args&&... args) { VERIFY(m_current_listing_index.has_value()); m_current_listing_index->listing.entries.empend(member, Listing::ArgumentsFor { forward(args)... }); } Optional get_context_parameter(GLenum pname); GPU::PackingSpecification get_packing_specification(PackingType); template void get_floating_point(GLenum pname, T* params); template void get_light_param(GLenum light, GLenum pname, T* params); template void get_material_param(Face face, GLenum pname, T* params); void invoke_list(size_t list_index); [[nodiscard]] bool should_append_to_listing() const { return m_current_listing_index.has_value(); } [[nodiscard]] bool should_execute_after_appending_to_listing() const { return m_current_listing_index.has_value() && m_current_listing_index->mode == GL_COMPILE_AND_EXECUTE; } // FIXME: we store GPU::Texture objects that do not point back to either the driver or device, so we need // to destruct the latter two at the very end. Fix this by making all GPU objects point back to // the device that created them, and the device back to the driver. RefPtr m_driver; NonnullOwnPtr m_rasterizer; GPU::DeviceInfo const m_device_info; GLenum m_current_draw_mode; GLenum m_current_matrix_mode { GL_MODELVIEW }; FloatMatrix4x4& projection_matrix() { return m_projection_matrix_stack.last(); } FloatMatrix4x4& model_view_matrix() { return m_model_view_matrix_stack.last(); } Vector m_projection_matrix_stack { FloatMatrix4x4::identity() }; Vector m_model_view_matrix_stack { FloatMatrix4x4::identity() }; Vector* m_current_matrix_stack { &m_model_view_matrix_stack }; FloatMatrix4x4* m_current_matrix { &m_current_matrix_stack->last() }; ALWAYS_INLINE void update_current_matrix(FloatMatrix4x4 const& new_matrix) { *m_current_matrix = new_matrix; if (m_current_matrix_mode == GL_TEXTURE) m_texture_units_dirty = true; } Gfx::IntRect m_viewport; FloatVector4 m_clear_color { 0.0f, 0.0f, 0.0f, 0.0f }; float m_clear_depth { 1.f }; u8 m_clear_stencil { 0 }; FloatVector4 m_current_vertex_color { 1.0f, 1.0f, 1.0f, 1.0f }; Vector m_current_vertex_tex_coord; FloatVector3 m_current_vertex_normal { 0.0f, 0.0f, 1.0f }; Vector m_vertex_list; GLenum m_error = GL_NO_ERROR; bool m_in_draw_state = false; bool m_depth_test_enabled { false }; bool m_depth_offset_enabled { false }; bool m_cull_faces = false; GLenum m_front_face = GL_CCW; GLenum m_culled_sides = GL_BACK; bool m_blend_enabled = false; GLenum m_blend_source_factor = GL_ONE; GLenum m_blend_destination_factor = GL_ZERO; bool m_alpha_test_enabled = false; GLenum m_alpha_test_func = GL_ALWAYS; GLclampf m_alpha_test_ref_value = 0; bool m_dither_enabled { true }; bool m_normalize { false }; // Stencil configuration bool m_stencil_test_enabled { false }; bool m_stencil_configuration_dirty { true }; struct StencilFunctionOptions { GLenum func { GL_ALWAYS }; GLint reference_value { 0 }; GLuint mask { NumericLimits::max() }; }; Array m_stencil_function; struct StencilOperationOptions { GLenum op_fail { GL_KEEP }; GLenum op_depth_fail { GL_KEEP }; GLenum op_pass { GL_KEEP }; GLuint write_mask { NumericLimits::max() }; }; Array m_stencil_operation; GLenum m_current_read_buffer = GL_BACK; GLenum m_current_draw_buffer = GL_BACK; // User-defined clip planes struct ClipPlaneAttributes { Array eye_clip_plane; // TODO: Change to use device-defined constant GLuint enabled { 0 }; } m_clip_plane_attributes; bool m_clip_planes_dirty { true }; // Client side arrays bool m_client_side_vertex_array_enabled { false }; bool m_client_side_color_array_enabled { false }; Vector m_client_side_texture_coord_array_enabled; size_t m_client_active_texture { 0 }; bool m_client_side_normal_array_enabled { false }; NonnullRefPtr m_frontbuffer; // Texture objects template RefPtr get_default_texture(GLenum target) { auto default_texture = m_default_textures.get(target); VERIFY(default_texture.has_value()); return static_cast(default_texture.value()); } NameAllocator m_texture_name_allocator; HashMap> m_allocated_textures; HashMap> m_default_textures; Vector m_texture_units; TextureUnit* m_active_texture_unit; size_t m_active_texture_unit_index { 0 }; bool m_texture_units_dirty { true }; // Texture coordinate generation state struct TextureCoordinateGeneration { bool enabled { false }; GLenum generation_mode { GL_EYE_LINEAR }; FloatVector4 object_plane_coefficients { 0.0f, 0.0f, 0.0f, 0.0f }; FloatVector4 eye_plane_coefficients { 0.0f, 0.0f, 0.0f, 0.0f }; }; Vector> m_texture_coordinate_generation; ALWAYS_INLINE TextureCoordinateGeneration& texture_coordinate_generation(size_t texture_unit, GLenum capability) { return m_texture_coordinate_generation[texture_unit][capability - GL_TEXTURE_GEN_S]; } bool m_sampler_config_is_dirty { true }; bool m_light_state_is_dirty { true }; NameAllocator m_shader_name_allocator; NameAllocator m_program_name_allocator; HashMap> m_allocated_shaders; HashMap> m_allocated_programs; RefPtr m_current_program; struct Listing { template struct TupleTypeForArgumentListOf_; template struct TupleTypeForArgumentListOf_ { using Type = Tuple; }; template using TupleTypeForArgumentListOf = typename TupleTypeForArgumentListOf_::Type; template using ArgumentsFor = TupleTypeForArgumentListOf; template struct FunctionAndArgs { Variant function; Variant...> arguments; }; using FunctionsAndArgs = FunctionAndArgs< decltype(&GLContext::gl_begin), decltype(&GLContext::gl_clear), decltype(&GLContext::gl_clear_color), decltype(&GLContext::gl_clear_depth), decltype(&GLContext::gl_clear_stencil), decltype(&GLContext::gl_color), decltype(&GLContext::gl_end), decltype(&GLContext::gl_frustum), decltype(&GLContext::gl_load_identity), decltype(&GLContext::gl_load_matrix), decltype(&GLContext::gl_matrix_mode), decltype(&GLContext::gl_ortho), decltype(&GLContext::gl_push_matrix), decltype(&GLContext::gl_pop_matrix), decltype(&GLContext::gl_mult_matrix), decltype(&GLContext::gl_rotate), decltype(&GLContext::gl_scale), decltype(&GLContext::gl_translate), decltype(&GLContext::gl_vertex), decltype(&GLContext::gl_viewport), decltype(&GLContext::gl_enable), decltype(&GLContext::gl_disable), decltype(&GLContext::gl_front_face), decltype(&GLContext::gl_cull_face), decltype(&GLContext::gl_call_list), decltype(&GLContext::gl_call_lists), decltype(&GLContext::gl_blend_func), decltype(&GLContext::gl_shade_model), decltype(&GLContext::gl_alpha_func), decltype(&GLContext::gl_hint), decltype(&GLContext::gl_read_buffer), decltype(&GLContext::gl_tex_parameter), decltype(&GLContext::gl_tex_parameterfv), decltype(&GLContext::gl_depth_mask), decltype(&GLContext::gl_draw_pixels), decltype(&GLContext::gl_depth_range), decltype(&GLContext::gl_polygon_offset), decltype(&GLContext::gl_scissor), decltype(&GLContext::gl_stencil_func_separate), decltype(&GLContext::gl_stencil_mask_separate), decltype(&GLContext::gl_stencil_op_separate), decltype(&GLContext::gl_normal), decltype(&GLContext::gl_raster_pos), decltype(&GLContext::gl_line_width), decltype(&GLContext::gl_push_attrib), decltype(&GLContext::gl_pop_attrib), decltype(&GLContext::gl_light_model), decltype(&GLContext::gl_bitmap), decltype(&GLContext::gl_copy_tex_image_2d), decltype(&GLContext::gl_rect), decltype(&GLContext::gl_tex_env), decltype(&GLContext::gl_tex_gen), decltype(&GLContext::gl_tex_gen_floatv), decltype(&GLContext::gl_fogf), decltype(&GLContext::gl_fogfv), decltype(&GLContext::gl_fogi), decltype(&GLContext::gl_lightf), decltype(&GLContext::gl_lightfv), decltype(&GLContext::gl_lightiv), decltype(&GLContext::gl_materialf), decltype(&GLContext::gl_materialfv), decltype(&GLContext::gl_materialiv), decltype(&GLContext::gl_color_material), decltype(&GLContext::gl_get_light), decltype(&GLContext::gl_clip_plane), decltype(&GLContext::gl_copy_tex_sub_image_2d), decltype(&GLContext::gl_point_size)>; using ExtraSavedArguments = Variant< FloatMatrix4x4>; Vector> saved_arguments; Vector entries; }; static constexpr size_t max_allowed_gl_call_depth { 128 }; size_t m_gl_call_depth { 0 }; Vector m_listings; size_t m_list_base { 0 }; struct CurrentListing { Listing listing; size_t index { 0 }; GLenum mode { GL_COMPILE }; }; Optional m_current_listing_index; VertexAttribPointer m_client_vertex_pointer; VertexAttribPointer m_client_color_pointer; Vector m_client_tex_coord_pointer; VertexAttribPointer m_client_normal_pointer; struct PixelParameters { i32 image_height { 0 }; bool least_significant_bit_first { false }; u8 pack_alignment { 4 }; i32 row_length { 0 }; i32 skip_images { 0 }; i32 skip_pixels { 0 }; i32 skip_rows { 0 }; bool swap_bytes { false }; }; PixelParameters m_packing_parameters; PixelParameters m_unpacking_parameters; // Point drawing configuration bool m_point_smooth { false }; float m_point_size { 1.f }; // Line drawing configuration bool m_line_smooth { false }; float m_line_width { 1.f }; // Lighting configuration bool m_lighting_enabled { false }; Vector m_light_states; Array m_material_states; // Color material bool m_color_material_enabled { false }; GLenum m_color_material_face { GL_FRONT_AND_BACK }; GLenum m_color_material_mode { GL_AMBIENT_AND_DIFFUSE }; // GL Extension string ByteBuffer m_extensions; // Buffer objects NameAllocator m_buffer_name_allocator; HashMap> m_allocated_buffers; RefPtr m_array_buffer; RefPtr m_element_array_buffer; }; // Transposes input matrices (column-major) to our Matrix (row-major). template constexpr FloatMatrix4x4 transpose_input_matrix(I const* matrix) { Array elements; for (size_t i = 0; i < 16; ++i) elements[i] = static_cast(matrix[i]); // clang-format off return { elements[0], elements[4], elements[8], elements[12], elements[1], elements[5], elements[9], elements[13], elements[2], elements[6], elements[10], elements[14], elements[3], elements[7], elements[11], elements[15], }; // clang-format on } template<> constexpr FloatMatrix4x4 transpose_input_matrix(float const* matrix) { // clang-format off return { matrix[0], matrix[4], matrix[8], matrix[12], matrix[1], matrix[5], matrix[9], matrix[13], matrix[2], matrix[6], matrix[10], matrix[14], matrix[3], matrix[7], matrix[11], matrix[15], }; // clang-format on } ErrorOr> create_context(Gfx::Bitmap&); void make_context_current(GLContext*); }