/* * Copyright (c) 2021, Stephan Unverwerth * Copyright (c) 2022, 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 namespace SoftGPU { struct TexCoordGenerationConfig { TexCoordGenerationMode mode { TexCoordGenerationMode::EyeLinear }; FloatVector4 coefficients {}; }; struct RasterizerOptions { bool shade_smooth { true }; bool enable_stencil_test { false }; bool enable_depth_test { false }; bool enable_depth_write { true }; bool enable_alpha_test { false }; AlphaTestFunction alpha_test_func { AlphaTestFunction::Always }; float alpha_test_ref_value { 0 }; bool enable_blending { false }; BlendFactor blend_source_factor { BlendFactor::One }; BlendFactor blend_destination_factor { BlendFactor::One }; u32 color_mask { 0xffffffff }; float depth_min { 0.f }; float depth_max { 1.f }; DepthTestFunction depth_func { DepthTestFunction::Less }; PolygonMode polygon_mode { PolygonMode::Fill }; FloatVector4 fog_color { 0.0f, 0.0f, 0.0f, 0.0f }; float fog_density { 1.0f }; FogMode fog_mode { FogMode::Exp }; bool fog_enabled { false }; float fog_start { 0.0f }; float fog_end { 1.0f }; bool scissor_enabled { false }; bool normalization_enabled { false }; Gfx::IntRect scissor_box; bool enable_color_write { true }; float depth_offset_factor { 0 }; float depth_offset_constant { 0 }; bool depth_offset_enabled { false }; bool enable_culling { false }; WindingOrder front_face { WindingOrder::CounterClockwise }; bool cull_back { true }; bool cull_front { false }; Array texcoord_generation_enabled_coordinates {}; Array, NUM_SAMPLERS> texcoord_generation_config {}; Gfx::IntRect viewport; bool lighting_enabled { false }; bool color_material_enabled { false }; ColorMaterialFace color_material_face { ColorMaterialFace::FrontAndBack }; ColorMaterialMode color_material_mode { ColorMaterialMode::AmbientAndDiffuse }; }; struct LightModelParameters { FloatVector4 scene_ambient_color { 0.2f, 0.2f, 0.2f, 1.0f }; bool viewer_at_infinity { false }; ColorControl color_control { ColorControl::SingleColor }; bool two_sided_lighting { false }; }; struct PixelQuad; struct RasterPosition { FloatVector4 window_coordinates { 0.0f, 0.0f, 0.0f, 1.0f }; float eye_coordinate_distance { 0.0f }; bool valid { true }; FloatVector4 color_rgba { 1.0f, 1.0f, 1.0f, 1.0f }; float color_index { 1.0f }; FloatVector4 texture_coordinates { 0.0f, 0.0f, 0.0f, 1.0f }; }; struct StencilConfiguration { StencilTestFunction test_function; StencilType reference_value; StencilType test_mask; StencilOperation on_stencil_test_fail; StencilOperation on_depth_test_fail; StencilOperation on_pass; StencilType write_mask; }; class Device final { public: Device(const Gfx::IntSize& min_size); DeviceInfo info() const; void draw_primitives(PrimitiveType, FloatMatrix4x4 const& model_view_transform, FloatMatrix4x4 const& projection_transform, FloatMatrix4x4 const& texture_transform, Vector const& vertices, Vector const& enabled_texture_units); void resize(Gfx::IntSize const& min_size); void clear_color(FloatVector4 const&); void clear_depth(DepthType); void clear_stencil(StencilType); void blit_color_buffer_to(Gfx::Bitmap& target); void blit_to_color_buffer_at_raster_position(Gfx::Bitmap const&); void blit_to_depth_buffer_at_raster_position(Vector const&, int, int); void set_options(const RasterizerOptions&); void set_light_model_params(const LightModelParameters&); RasterizerOptions options() const { return m_options; } LightModelParameters light_model() const { return m_lighting_model; } ColorType get_color_buffer_pixel(int x, int y); DepthType get_depthbuffer_value(int x, int y); NonnullRefPtr create_image(ImageFormat format, unsigned width, unsigned height, unsigned depth, unsigned levels, unsigned layers); void set_sampler_config(unsigned, SamplerConfig const&); void set_light_state(unsigned, Light const&); void set_material_state(Face, Material const&); void set_stencil_configuration(Face, StencilConfiguration const&); RasterPosition raster_position() const { return m_raster_position; } void set_raster_position(RasterPosition const& raster_position); void set_raster_position(FloatVector4 const& position, FloatMatrix4x4 const& model_view_transform, FloatMatrix4x4 const& projection_transform); private: void draw_statistics_overlay(Gfx::Bitmap&); Gfx::IntRect get_rasterization_rect_of_size(Gfx::IntSize size); void rasterize_triangle(const Triangle& triangle); void setup_blend_factors(); void shade_fragments(PixelQuad&); bool test_alpha(PixelQuad&); RefPtr> m_frame_buffer {}; RasterizerOptions m_options; LightModelParameters m_lighting_model; Clipper m_clipper; Vector m_triangle_list; Vector m_processed_triangles; Vector m_clipped_vertices; Array m_samplers; Vector m_enabled_texture_units; AlphaBlendFactors m_alpha_blend_factors; Array m_lights; Array m_materials; RasterPosition m_raster_position; Array m_stencil_configuration; }; }