2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* renderer_canvas_render_rd.cpp */
/**************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/**************************************************************************/
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
/* */
/* 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 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. */
/**************************************************************************/
2019-06-22 18:34:26 +02:00
2020-12-04 19:26:24 +01:00
# include "renderer_canvas_render_rd.h"
2021-10-28 15:43:36 +02:00
2020-11-07 23:33:38 +01:00
# include "core/config/project_settings.h"
2020-11-26 13:50:21 +01:00
# include "core/math/geometry_2d.h"
2021-05-19 14:12:55 +02:00
# include "core/math/math_defs.h"
2019-06-24 21:13:06 +02:00
# include "core/math/math_funcs.h"
2020-12-04 19:26:24 +01:00
# include "renderer_compositor_rd.h"
2022-03-21 12:25:25 +01:00
# include "servers/rendering/renderer_rd/storage_rd/material_storage.h"
2022-04-12 13:41:50 +02:00
# include "servers/rendering/renderer_rd/storage_rd/particles_storage.h"
2022-03-21 12:25:25 +01:00
# include "servers/rendering/renderer_rd/storage_rd/texture_storage.h"
2021-05-20 16:25:06 +02:00
# include "servers/rendering/rendering_server_default.h"
2019-07-21 16:31:30 +02:00
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : _update_transform_2d_to_mat4 ( const Transform2D & p_transform , float * p_mat4 ) {
2022-04-24 23:59:24 +02:00
p_mat4 [ 0 ] = p_transform . columns [ 0 ] [ 0 ] ;
p_mat4 [ 1 ] = p_transform . columns [ 0 ] [ 1 ] ;
2019-06-16 04:45:24 +02:00
p_mat4 [ 2 ] = 0 ;
p_mat4 [ 3 ] = 0 ;
2022-04-24 23:59:24 +02:00
p_mat4 [ 4 ] = p_transform . columns [ 1 ] [ 0 ] ;
p_mat4 [ 5 ] = p_transform . columns [ 1 ] [ 1 ] ;
2019-06-16 04:45:24 +02:00
p_mat4 [ 6 ] = 0 ;
p_mat4 [ 7 ] = 0 ;
p_mat4 [ 8 ] = 0 ;
p_mat4 [ 9 ] = 0 ;
p_mat4 [ 10 ] = 1 ;
p_mat4 [ 11 ] = 0 ;
2022-04-24 23:59:24 +02:00
p_mat4 [ 12 ] = p_transform . columns [ 2 ] [ 0 ] ;
p_mat4 [ 13 ] = p_transform . columns [ 2 ] [ 1 ] ;
2019-06-16 04:45:24 +02:00
p_mat4 [ 14 ] = 0 ;
p_mat4 [ 15 ] = 1 ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : _update_transform_2d_to_mat2x4 ( const Transform2D & p_transform , float * p_mat2x4 ) {
2022-04-24 23:59:24 +02:00
p_mat2x4 [ 0 ] = p_transform . columns [ 0 ] [ 0 ] ;
p_mat2x4 [ 1 ] = p_transform . columns [ 1 ] [ 0 ] ;
2019-06-16 04:45:24 +02:00
p_mat2x4 [ 2 ] = 0 ;
2022-04-24 23:59:24 +02:00
p_mat2x4 [ 3 ] = p_transform . columns [ 2 ] [ 0 ] ;
2019-06-16 04:45:24 +02:00
2022-04-24 23:59:24 +02:00
p_mat2x4 [ 4 ] = p_transform . columns [ 0 ] [ 1 ] ;
p_mat2x4 [ 5 ] = p_transform . columns [ 1 ] [ 1 ] ;
2019-06-16 04:45:24 +02:00
p_mat2x4 [ 6 ] = 0 ;
2022-04-24 23:59:24 +02:00
p_mat2x4 [ 7 ] = p_transform . columns [ 2 ] [ 1 ] ;
2019-06-16 04:45:24 +02:00
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : _update_transform_2d_to_mat2x3 ( const Transform2D & p_transform , float * p_mat2x3 ) {
2022-04-24 23:59:24 +02:00
p_mat2x3 [ 0 ] = p_transform . columns [ 0 ] [ 0 ] ;
p_mat2x3 [ 1 ] = p_transform . columns [ 0 ] [ 1 ] ;
p_mat2x3 [ 2 ] = p_transform . columns [ 1 ] [ 0 ] ;
p_mat2x3 [ 3 ] = p_transform . columns [ 1 ] [ 1 ] ;
p_mat2x3 [ 4 ] = p_transform . columns [ 2 ] [ 0 ] ;
p_mat2x3 [ 5 ] = p_transform . columns [ 2 ] [ 1 ] ;
2019-06-24 21:13:06 +02:00
}
2020-10-17 07:08:21 +02:00
void RendererCanvasRenderRD : : _update_transform_to_mat4 ( const Transform3D & p_transform , float * p_mat4 ) {
2022-04-25 00:07:35 +02:00
p_mat4 [ 0 ] = p_transform . basis . rows [ 0 ] [ 0 ] ;
p_mat4 [ 1 ] = p_transform . basis . rows [ 1 ] [ 0 ] ;
p_mat4 [ 2 ] = p_transform . basis . rows [ 2 ] [ 0 ] ;
2019-06-16 04:45:24 +02:00
p_mat4 [ 3 ] = 0 ;
2022-04-25 00:07:35 +02:00
p_mat4 [ 4 ] = p_transform . basis . rows [ 0 ] [ 1 ] ;
p_mat4 [ 5 ] = p_transform . basis . rows [ 1 ] [ 1 ] ;
p_mat4 [ 6 ] = p_transform . basis . rows [ 2 ] [ 1 ] ;
2019-06-16 04:45:24 +02:00
p_mat4 [ 7 ] = 0 ;
2022-04-25 00:07:35 +02:00
p_mat4 [ 8 ] = p_transform . basis . rows [ 0 ] [ 2 ] ;
p_mat4 [ 9 ] = p_transform . basis . rows [ 1 ] [ 2 ] ;
p_mat4 [ 10 ] = p_transform . basis . rows [ 2 ] [ 2 ] ;
2019-06-16 04:45:24 +02:00
p_mat4 [ 11 ] = 0 ;
p_mat4 [ 12 ] = p_transform . origin . x ;
p_mat4 [ 13 ] = p_transform . origin . y ;
p_mat4 [ 14 ] = p_transform . origin . z ;
p_mat4 [ 15 ] = 1 ;
}
2020-12-04 19:26:24 +01:00
RendererCanvasRender : : PolygonID RendererCanvasRenderRD : : request_polygon ( const Vector < int > & p_indices , const Vector < Point2 > & p_points , const Vector < Color > & p_colors , const Vector < Point2 > & p_uvs , const Vector < int > & p_bones , const Vector < float > & p_weights ) {
2019-06-24 21:13:06 +02:00
// Care must be taken to generate array formats
// in ways where they could be reused, so we will
// put single-occuring elements first, and repeated
// elements later. This way the generated formats are
// the same no matter the length of the arrays.
// This dramatically reduces the amount of pipeline objects
// that need to be created for these formats.
2022-04-02 07:29:04 +02:00
RendererRD : : MeshStorage * mesh_storage = RendererRD : : MeshStorage : : get_singleton ( ) ;
2019-06-24 21:13:06 +02:00
uint32_t vertex_count = p_points . size ( ) ;
uint32_t stride = 2 ; //vertices always repeat
2019-07-10 22:44:55 +02:00
if ( ( uint32_t ) p_colors . size ( ) = = vertex_count | | p_colors . size ( ) = = 1 ) {
2019-06-24 21:13:06 +02:00
stride + = 4 ;
}
if ( ( uint32_t ) p_uvs . size ( ) = = vertex_count ) {
stride + = 2 ;
}
2019-08-26 22:43:58 +02:00
if ( ( uint32_t ) p_bones . size ( ) = = vertex_count * 4 & & ( uint32_t ) p_weights . size ( ) = = vertex_count * 4 ) {
2019-06-24 21:13:06 +02:00
stride + = 4 ;
}
2019-07-10 22:44:55 +02:00
uint32_t buffer_size = stride * p_points . size ( ) ;
2019-06-24 21:13:06 +02:00
2020-02-17 22:06:54 +01:00
Vector < uint8_t > polygon_buffer ;
2019-06-24 21:13:06 +02:00
polygon_buffer . resize ( buffer_size * sizeof ( float ) ) ;
2020-04-21 17:16:45 +02:00
Vector < RD : : VertexAttribute > descriptions ;
2020-12-02 02:40:47 +01:00
descriptions . resize ( 5 ) ;
2019-07-10 22:44:55 +02:00
Vector < RID > buffers ;
2020-12-02 02:40:47 +01:00
buffers . resize ( 5 ) ;
2019-06-24 21:13:06 +02:00
{
2022-04-07 12:23:40 +02:00
uint8_t * r = polygon_buffer . ptrw ( ) ;
float * fptr = reinterpret_cast < float * > ( r ) ;
uint32_t * uptr = reinterpret_cast < uint32_t * > ( r ) ;
2019-07-10 22:44:55 +02:00
uint32_t base_offset = 0 ;
2019-06-24 21:13:06 +02:00
{ //vertices
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2019-06-24 21:13:06 +02:00
vd . format = RD : : DATA_FORMAT_R32G32_SFLOAT ;
vd . offset = base_offset * sizeof ( float ) ;
2020-03-27 19:21:27 +01:00
vd . location = RS : : ARRAY_VERTEX ;
2019-06-24 21:13:06 +02:00
vd . stride = stride * sizeof ( float ) ;
descriptions . write [ 0 ] = vd ;
const Vector2 * points_ptr = p_points . ptr ( ) ;
for ( uint32_t i = 0 ; i < vertex_count ; i + + ) {
fptr [ base_offset + i * stride + 0 ] = points_ptr [ i ] . x ;
fptr [ base_offset + i * stride + 1 ] = points_ptr [ i ] . y ;
}
base_offset + = 2 ;
}
//colors
2019-07-10 22:44:55 +02:00
if ( ( uint32_t ) p_colors . size ( ) = = vertex_count | | p_colors . size ( ) = = 1 ) {
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2019-06-24 21:13:06 +02:00
vd . format = RD : : DATA_FORMAT_R32G32B32A32_SFLOAT ;
vd . offset = base_offset * sizeof ( float ) ;
2020-03-27 19:21:27 +01:00
vd . location = RS : : ARRAY_COLOR ;
2019-06-24 21:13:06 +02:00
vd . stride = stride * sizeof ( float ) ;
descriptions . write [ 1 ] = vd ;
2019-07-10 22:44:55 +02:00
if ( p_colors . size ( ) = = 1 ) {
Color color = p_colors [ 0 ] ;
for ( uint32_t i = 0 ; i < vertex_count ; i + + ) {
fptr [ base_offset + i * stride + 0 ] = color . r ;
fptr [ base_offset + i * stride + 1 ] = color . g ;
fptr [ base_offset + i * stride + 2 ] = color . b ;
fptr [ base_offset + i * stride + 3 ] = color . a ;
}
} else {
const Color * color_ptr = p_colors . ptr ( ) ;
2019-06-24 21:13:06 +02:00
2019-07-10 22:44:55 +02:00
for ( uint32_t i = 0 ; i < vertex_count ; i + + ) {
fptr [ base_offset + i * stride + 0 ] = color_ptr [ i ] . r ;
fptr [ base_offset + i * stride + 1 ] = color_ptr [ i ] . g ;
fptr [ base_offset + i * stride + 2 ] = color_ptr [ i ] . b ;
fptr [ base_offset + i * stride + 3 ] = color_ptr [ i ] . a ;
}
2019-06-24 21:13:06 +02:00
}
base_offset + = 4 ;
} else {
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2019-06-24 21:13:06 +02:00
vd . format = RD : : DATA_FORMAT_R32G32B32A32_SFLOAT ;
2019-07-10 22:44:55 +02:00
vd . offset = 0 ;
2020-03-27 19:21:27 +01:00
vd . location = RS : : ARRAY_COLOR ;
2019-06-24 21:13:06 +02:00
vd . stride = 0 ;
descriptions . write [ 1 ] = vd ;
2022-08-01 03:59:14 +02:00
buffers . write [ 1 ] = mesh_storage - > mesh_get_default_rd_buffer ( RendererRD : : MeshStorage : : DEFAULT_RD_BUFFER_COLOR ) ;
2019-06-24 21:13:06 +02:00
}
//uvs
if ( ( uint32_t ) p_uvs . size ( ) = = vertex_count ) {
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2019-06-24 21:13:06 +02:00
vd . format = RD : : DATA_FORMAT_R32G32_SFLOAT ;
vd . offset = base_offset * sizeof ( float ) ;
2020-03-27 19:21:27 +01:00
vd . location = RS : : ARRAY_TEX_UV ;
2019-06-24 21:13:06 +02:00
vd . stride = stride * sizeof ( float ) ;
descriptions . write [ 2 ] = vd ;
const Vector2 * uv_ptr = p_uvs . ptr ( ) ;
for ( uint32_t i = 0 ; i < vertex_count ; i + + ) {
fptr [ base_offset + i * stride + 0 ] = uv_ptr [ i ] . x ;
fptr [ base_offset + i * stride + 1 ] = uv_ptr [ i ] . y ;
}
base_offset + = 2 ;
} else {
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2019-06-24 21:13:06 +02:00
vd . format = RD : : DATA_FORMAT_R32G32_SFLOAT ;
2019-07-10 22:44:55 +02:00
vd . offset = 0 ;
2020-03-27 19:21:27 +01:00
vd . location = RS : : ARRAY_TEX_UV ;
2019-06-24 21:13:06 +02:00
vd . stride = 0 ;
descriptions . write [ 2 ] = vd ;
2022-08-01 03:59:14 +02:00
buffers . write [ 2 ] = mesh_storage - > mesh_get_default_rd_buffer ( RendererRD : : MeshStorage : : DEFAULT_RD_BUFFER_TEX_UV ) ;
2019-06-24 21:13:06 +02:00
}
//bones
2019-08-26 22:43:58 +02:00
if ( ( uint32_t ) p_indices . size ( ) = = vertex_count * 4 & & ( uint32_t ) p_weights . size ( ) = = vertex_count * 4 ) {
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2020-12-02 02:40:47 +01:00
vd . format = RD : : DATA_FORMAT_R16G16B16A16_UINT ;
2019-06-24 21:13:06 +02:00
vd . offset = base_offset * sizeof ( float ) ;
2020-03-27 19:21:27 +01:00
vd . location = RS : : ARRAY_BONES ;
2019-06-24 21:13:06 +02:00
vd . stride = stride * sizeof ( float ) ;
descriptions . write [ 3 ] = vd ;
const int * bone_ptr = p_bones . ptr ( ) ;
for ( uint32_t i = 0 ; i < vertex_count ; i + + ) {
2019-08-26 22:43:58 +02:00
uint16_t * bone16w = ( uint16_t * ) & uptr [ base_offset + i * stride ] ;
bone16w [ 0 ] = bone_ptr [ i * 4 + 0 ] ;
bone16w [ 1 ] = bone_ptr [ i * 4 + 1 ] ;
bone16w [ 2 ] = bone_ptr [ i * 4 + 2 ] ;
bone16w [ 3 ] = bone_ptr [ i * 4 + 3 ] ;
2020-12-02 02:40:47 +01:00
}
base_offset + = 2 ;
} else {
RD : : VertexAttribute vd ;
vd . format = RD : : DATA_FORMAT_R32G32B32A32_UINT ;
vd . offset = 0 ;
vd . location = RS : : ARRAY_BONES ;
vd . stride = 0 ;
descriptions . write [ 3 ] = vd ;
2022-08-01 03:59:14 +02:00
buffers . write [ 3 ] = mesh_storage - > mesh_get_default_rd_buffer ( RendererRD : : MeshStorage : : DEFAULT_RD_BUFFER_BONES ) ;
2020-12-02 02:40:47 +01:00
}
//weights
if ( ( uint32_t ) p_weights . size ( ) = = vertex_count * 4 ) {
RD : : VertexAttribute vd ;
vd . format = RD : : DATA_FORMAT_R16G16B16A16_UNORM ;
vd . offset = base_offset * sizeof ( float ) ;
vd . location = RS : : ARRAY_WEIGHTS ;
vd . stride = stride * sizeof ( float ) ;
descriptions . write [ 4 ] = vd ;
const float * weight_ptr = p_weights . ptr ( ) ;
for ( uint32_t i = 0 ; i < vertex_count ; i + + ) {
uint16_t * weight16w = ( uint16_t * ) & uptr [ base_offset + i * stride ] ;
2019-08-26 22:43:58 +02:00
weight16w [ 0 ] = CLAMP ( weight_ptr [ i * 4 + 0 ] * 65535 , 0 , 65535 ) ;
weight16w [ 1 ] = CLAMP ( weight_ptr [ i * 4 + 1 ] * 65535 , 0 , 65535 ) ;
weight16w [ 2 ] = CLAMP ( weight_ptr [ i * 4 + 2 ] * 65535 , 0 , 65535 ) ;
weight16w [ 3 ] = CLAMP ( weight_ptr [ i * 4 + 3 ] * 65535 , 0 , 65535 ) ;
2019-06-24 21:13:06 +02:00
}
2020-12-02 02:40:47 +01:00
base_offset + = 2 ;
2019-06-24 21:13:06 +02:00
} else {
2020-04-21 17:16:45 +02:00
RD : : VertexAttribute vd ;
2020-12-02 02:40:47 +01:00
vd . format = RD : : DATA_FORMAT_R32G32B32A32_SFLOAT ;
2019-07-10 22:44:55 +02:00
vd . offset = 0 ;
2020-12-02 02:40:47 +01:00
vd . location = RS : : ARRAY_WEIGHTS ;
2019-06-24 21:13:06 +02:00
vd . stride = 0 ;
2020-12-02 02:40:47 +01:00
descriptions . write [ 4 ] = vd ;
2022-08-01 03:59:14 +02:00
buffers . write [ 4 ] = mesh_storage - > mesh_get_default_rd_buffer ( RendererRD : : MeshStorage : : DEFAULT_RD_BUFFER_WEIGHTS ) ;
2019-06-24 21:13:06 +02:00
}
//check that everything is as it should be
2019-07-10 22:44:55 +02:00
ERR_FAIL_COND_V ( base_offset ! = stride , 0 ) ; //bug
2019-06-24 21:13:06 +02:00
}
RD : : VertexFormatID vertex_id = RD : : get_singleton ( ) - > vertex_format_create ( descriptions ) ;
ERR_FAIL_COND_V ( vertex_id = = RD : : INVALID_ID , 0 ) ;
PolygonBuffers pb ;
pb . vertex_buffer = RD : : get_singleton ( ) - > vertex_buffer_create ( polygon_buffer . size ( ) , polygon_buffer ) ;
for ( int i = 0 ; i < descriptions . size ( ) ; i + + ) {
2019-07-10 22:44:55 +02:00
if ( buffers [ i ] = = RID ( ) ) { //if put in vertex, use as vertex
buffers . write [ i ] = pb . vertex_buffer ;
}
2019-06-24 21:13:06 +02:00
}
2019-07-10 22:44:55 +02:00
2019-06-24 21:13:06 +02:00
pb . vertex_array = RD : : get_singleton ( ) - > vertex_array_create ( p_points . size ( ) , vertex_id , buffers ) ;
if ( p_indices . size ( ) ) {
//create indices, as indices were requested
2020-02-17 22:06:54 +01:00
Vector < uint8_t > index_buffer ;
2019-06-24 21:13:06 +02:00
index_buffer . resize ( p_indices . size ( ) * sizeof ( int32_t ) ) ;
{
2020-02-17 22:06:54 +01:00
uint8_t * w = index_buffer . ptrw ( ) ;
2021-04-27 16:19:21 +02:00
memcpy ( w , p_indices . ptr ( ) , sizeof ( int32_t ) * p_indices . size ( ) ) ;
2019-06-24 21:13:06 +02:00
}
pb . index_buffer = RD : : get_singleton ( ) - > index_buffer_create ( p_indices . size ( ) , RD : : INDEX_BUFFER_FORMAT_UINT32 , index_buffer ) ;
pb . indices = RD : : get_singleton ( ) - > index_array_create ( pb . index_buffer , 0 , p_indices . size ( ) ) ;
}
pb . vertex_format_id = vertex_id ;
PolygonID id = polygon_buffers . last_id + + ;
polygon_buffers . polygons [ id ] = pb ;
return id ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : free_polygon ( PolygonID p_polygon ) {
2019-06-24 21:13:06 +02:00
PolygonBuffers * pb_ptr = polygon_buffers . polygons . getptr ( p_polygon ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL ( pb_ptr ) ;
2019-06-24 21:13:06 +02:00
PolygonBuffers & pb = * pb_ptr ;
if ( pb . indices . is_valid ( ) ) {
RD : : get_singleton ( ) - > free ( pb . indices ) ;
}
if ( pb . index_buffer . is_valid ( ) ) {
RD : : get_singleton ( ) - > free ( pb . index_buffer ) ;
}
RD : : get_singleton ( ) - > free ( pb . vertex_array ) ;
RD : : get_singleton ( ) - > free ( pb . vertex_buffer ) ;
polygon_buffers . polygons . erase ( p_polygon ) ;
}
2020-10-24 17:15:43 +02:00
////////////////////
2023-08-15 11:08:29 +02:00
void RendererCanvasRenderRD : : _bind_canvas_texture ( RD : : DrawListID p_draw_list , RID p_texture , RS : : CanvasItemTextureFilter p_base_filter , RS : : CanvasItemTextureRepeat p_base_repeat , RID & r_last_texture , PushConstant & push_constant , Size2 & r_texpixel_size , bool p_texture_is_data ) {
2020-10-24 17:15:43 +02:00
if ( p_texture = = RID ( ) ) {
p_texture = default_canvas_texture ;
}
2019-06-24 21:13:06 +02:00
2020-10-24 17:15:43 +02:00
if ( r_last_texture = = p_texture ) {
return ; //nothing to do, its the same
2019-07-05 03:54:32 +02:00
}
2020-10-24 17:15:43 +02:00
RID uniform_set ;
Color specular_shininess ;
Size2i size ;
bool use_normal ;
bool use_specular ;
2023-08-15 11:08:29 +02:00
bool success = RendererRD : : TextureStorage : : get_singleton ( ) - > canvas_texture_get_uniform_set ( p_texture , p_base_filter , p_base_repeat , shader . default_version_rd_shader , CANVAS_TEXTURE_UNIFORM_SET , bool ( push_constant . flags & FLAGS_CONVERT_ATTRIBUTES_TO_LINEAR ) , uniform_set , size , specular_shininess , use_normal , use_specular , p_texture_is_data ) ;
2020-10-24 17:15:43 +02:00
//something odd happened
if ( ! success ) {
_bind_canvas_texture ( p_draw_list , default_canvas_texture , p_base_filter , p_base_repeat , r_last_texture , push_constant , r_texpixel_size ) ;
return ;
2019-07-05 03:54:32 +02:00
}
2020-10-24 17:15:43 +02:00
RD : : get_singleton ( ) - > draw_list_bind_uniform_set ( p_draw_list , uniform_set , CANVAS_TEXTURE_UNIFORM_SET ) ;
if ( specular_shininess . a < 0.999 ) {
push_constant . flags | = FLAGS_DEFAULT_SPECULAR_MAP_USED ;
} else {
push_constant . flags & = ~ FLAGS_DEFAULT_SPECULAR_MAP_USED ;
2019-06-24 21:13:06 +02:00
}
2020-10-24 17:15:43 +02:00
if ( use_normal ) {
push_constant . flags | = FLAGS_DEFAULT_NORMAL_MAP_USED ;
2019-06-24 21:13:06 +02:00
} else {
2020-10-24 17:15:43 +02:00
push_constant . flags & = ~ FLAGS_DEFAULT_NORMAL_MAP_USED ;
2019-06-24 21:13:06 +02:00
}
2020-10-24 17:15:43 +02:00
push_constant . specular_shininess = uint32_t ( CLAMP ( specular_shininess . a * 255.0 , 0 , 255 ) ) < < 24 ;
push_constant . specular_shininess | = uint32_t ( CLAMP ( specular_shininess . b * 255.0 , 0 , 255 ) ) < < 16 ;
push_constant . specular_shininess | = uint32_t ( CLAMP ( specular_shininess . g * 255.0 , 0 , 255 ) ) < < 8 ;
push_constant . specular_shininess | = uint32_t ( CLAMP ( specular_shininess . r * 255.0 , 0 , 255 ) ) ;
r_texpixel_size . x = 1.0 / float ( size . x ) ;
r_texpixel_size . y = 1.0 / float ( size . y ) ;
push_constant . color_texture_pixel_size [ 0 ] = r_texpixel_size . x ;
push_constant . color_texture_pixel_size [ 1 ] = r_texpixel_size . y ;
r_last_texture = p_texture ;
2019-06-24 21:13:06 +02:00
}
2022-12-07 19:03:08 +01:00
void RendererCanvasRenderRD : : _render_item ( RD : : DrawListID p_draw_list , RID p_render_target , const Item * p_item , RD : : FramebufferFormatID p_framebuffer_format , const Transform2D & p_canvas_transform_inverse , Item * & current_clip , Light * p_lights , PipelineVariants * p_pipeline_variants , bool & r_sdf_used ) {
2019-06-16 04:45:24 +02:00
//create an empty push constant
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
2022-04-02 07:29:04 +02:00
RendererRD : : MeshStorage * mesh_storage = RendererRD : : MeshStorage : : get_singleton ( ) ;
2022-04-12 13:41:50 +02:00
RendererRD : : ParticlesStorage * particles_storage = RendererRD : : ParticlesStorage : : get_singleton ( ) ;
2019-07-10 22:44:55 +02:00
2020-10-24 17:15:43 +02:00
RS : : CanvasItemTextureFilter current_filter = default_filter ;
RS : : CanvasItemTextureRepeat current_repeat = default_repeat ;
if ( p_item - > texture_filter ! = RS : : CANVAS_ITEM_TEXTURE_FILTER_DEFAULT ) {
current_filter = p_item - > texture_filter ;
}
if ( p_item - > texture_repeat ! = RS : : CANVAS_ITEM_TEXTURE_REPEAT_DEFAULT ) {
current_repeat = p_item - > texture_repeat ;
}
2019-06-16 04:45:24 +02:00
PushConstant push_constant ;
2019-06-24 21:13:06 +02:00
Transform2D base_transform = p_canvas_transform_inverse * p_item - > final_transform ;
2021-06-22 10:30:11 +02:00
Transform2D draw_transform ;
2019-06-24 21:13:06 +02:00
_update_transform_2d_to_mat2x3 ( base_transform , push_constant . world ) ;
2019-07-05 03:54:32 +02:00
Color base_color = p_item - > final_modulate ;
2023-08-03 14:10:01 +02:00
bool use_linear_colors = texture_storage - > render_target_is_using_hdr ( p_render_target ) ;
2019-07-05 03:54:32 +02:00
2019-06-16 04:45:24 +02:00
for ( int i = 0 ; i < 4 ; i + + ) {
push_constant . modulation [ i ] = 0 ;
push_constant . ninepatch_margins [ i ] = 0 ;
push_constant . src_rect [ i ] = 0 ;
push_constant . dst_rect [ i ] = 0 ;
}
push_constant . flags = 0 ;
push_constant . color_texture_pixel_size [ 0 ] = 0 ;
push_constant . color_texture_pixel_size [ 1 ] = 0 ;
2019-06-26 20:11:52 +02:00
2019-10-05 09:30:13 +02:00
push_constant . pad [ 0 ] = 0 ;
2019-06-16 04:45:24 +02:00
push_constant . pad [ 1 ] = 0 ;
2019-07-05 03:54:32 +02:00
push_constant . lights [ 0 ] = 0 ;
push_constant . lights [ 1 ] = 0 ;
push_constant . lights [ 2 ] = 0 ;
push_constant . lights [ 3 ] = 0 ;
uint32_t base_flags = 0 ;
2023-08-03 14:10:01 +02:00
base_flags | = use_linear_colors ? FLAGS_CONVERT_ATTRIBUTES_TO_LINEAR : 0 ;
2019-07-05 03:54:32 +02:00
2019-07-10 22:44:55 +02:00
uint16_t light_count = 0 ;
PipelineLightMode light_mode ;
2019-07-05 03:54:32 +02:00
{
Light * light = p_lights ;
while ( light ) {
if ( light - > render_index_cache > = 0 & & p_item - > light_mask & light - > item_mask & & p_item - > z_final > = light - > z_min & & p_item - > z_final < = light - > z_max & & p_item - > global_rect_cache . intersects_transformed ( light - > xform_cache , light - > rect_cache ) ) {
uint32_t light_index = light - > render_index_cache ;
push_constant . lights [ light_count > > 2 ] | = light_index < < ( ( light_count & 3 ) * 8 ) ;
2019-07-10 22:44:55 +02:00
2019-07-05 03:54:32 +02:00
light_count + + ;
2020-10-24 17:15:43 +02:00
2023-01-21 02:04:00 +01:00
if ( light_count = = MAX_LIGHTS_PER_ITEM - 1 ) {
2019-07-05 03:54:32 +02:00
break ;
}
}
light = light - > next_ptr ;
}
base_flags | = light_count < < FLAGS_LIGHT_COUNT_SHIFT ;
}
2020-11-03 20:51:53 +01:00
light_mode = ( light_count > 0 | | using_directional_lights ) ? PIPELINE_LIGHT_MODE_ENABLED : PIPELINE_LIGHT_MODE_DISABLED ;
2019-07-21 16:31:30 +02:00
PipelineVariants * pipeline_variants = p_pipeline_variants ;
2019-06-16 04:45:24 +02:00
2019-06-24 21:13:06 +02:00
bool reclip = false ;
2020-10-24 17:15:43 +02:00
RID last_texture ;
Size2 texpixel_size ;
2021-06-17 16:30:20 +02:00
bool skipping = false ;
2019-06-27 00:48:05 +02:00
const Item : : Command * c = p_item - > commands ;
while ( c ) {
2021-06-17 16:30:20 +02:00
if ( skipping & & c - > type ! = Item : : Command : : TYPE_ANIMATION_SLICE ) {
c = c - > next ;
continue ;
}
2020-10-24 17:15:43 +02:00
push_constant . flags = base_flags | ( push_constant . flags & ( FLAGS_DEFAULT_NORMAL_MAP_USED | FLAGS_DEFAULT_SPECULAR_MAP_USED ) ) ; //reset on each command for sanity, keep canvastexture binding config
2019-06-16 04:45:24 +02:00
switch ( c - > type ) {
case Item : : Command : : TYPE_RECT : {
const Item : : CommandRect * rect = static_cast < const Item : : CommandRect * > ( c ) ;
2021-04-14 06:30:23 +02:00
if ( rect - > flags & CANVAS_RECT_TILE ) {
current_repeat = RenderingServer : : CanvasItemTextureRepeat : : CANVAS_ITEM_TEXTURE_REPEAT_ENABLED ;
}
2019-06-16 04:45:24 +02:00
//bind pipeline
2022-08-12 13:03:28 +02:00
if ( rect - > flags & CANVAS_RECT_LCD ) {
RID pipeline = pipeline_variants - > variants [ light_mode ] [ PIPELINE_VARIANT_QUAD_LCD_BLEND ] . get_render_pipeline ( RD : : INVALID_ID , p_framebuffer_format ) ;
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( p_draw_list , pipeline ) ;
RD : : get_singleton ( ) - > draw_list_set_blend_constants ( p_draw_list , rect - > modulate ) ;
} else {
2019-07-10 22:44:55 +02:00
RID pipeline = pipeline_variants - > variants [ light_mode ] [ PIPELINE_VARIANT_QUAD ] . get_render_pipeline ( RD : : INVALID_ID , p_framebuffer_format ) ;
2019-06-16 04:45:24 +02:00
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( p_draw_list , pipeline ) ;
}
//bind textures
2023-08-15 11:08:29 +02:00
_bind_canvas_texture ( p_draw_list , rect - > texture , current_filter , current_repeat , last_texture , push_constant , texpixel_size , bool ( rect - > flags & CANVAS_RECT_MSDF ) ) ;
2019-07-05 03:54:32 +02:00
2019-06-16 04:45:24 +02:00
Rect2 src_rect ;
Rect2 dst_rect ;
2020-10-24 17:15:43 +02:00
if ( rect - > texture ! = RID ( ) ) {
2019-06-16 04:45:24 +02:00
src_rect = ( rect - > flags & CANVAS_RECT_REGION ) ? Rect2 ( rect - > source . position * texpixel_size , rect - > source . size * texpixel_size ) : Rect2 ( 0 , 0 , 1 , 1 ) ;
dst_rect = Rect2 ( rect - > rect . position , rect - > rect . size ) ;
if ( dst_rect . size . width < 0 ) {
dst_rect . position . x + = dst_rect . size . width ;
dst_rect . size . width * = - 1 ;
}
if ( dst_rect . size . height < 0 ) {
dst_rect . position . y + = dst_rect . size . height ;
dst_rect . size . height * = - 1 ;
}
if ( rect - > flags & CANVAS_RECT_FLIP_H ) {
src_rect . size . x * = - 1 ;
2023-01-14 07:20:42 +01:00
push_constant . flags | = FLAGS_FLIP_H ;
2019-06-16 04:45:24 +02:00
}
if ( rect - > flags & CANVAS_RECT_FLIP_V ) {
src_rect . size . y * = - 1 ;
2023-01-14 07:20:42 +01:00
push_constant . flags | = FLAGS_FLIP_V ;
2019-06-16 04:45:24 +02:00
}
if ( rect - > flags & CANVAS_RECT_TRANSPOSE ) {
2022-07-20 12:54:52 +02:00
push_constant . flags | = FLAGS_TRANSPOSE_RECT ;
2019-06-16 04:45:24 +02:00
}
if ( rect - > flags & CANVAS_RECT_CLIP_UV ) {
push_constant . flags | = FLAGS_CLIP_RECT_UV ;
}
} else {
dst_rect = Rect2 ( rect - > rect . position , rect - > rect . size ) ;
if ( dst_rect . size . width < 0 ) {
dst_rect . position . x + = dst_rect . size . width ;
dst_rect . size . width * = - 1 ;
}
if ( dst_rect . size . height < 0 ) {
dst_rect . position . y + = dst_rect . size . height ;
dst_rect . size . height * = - 1 ;
}
src_rect = Rect2 ( 0 , 0 , 1 , 1 ) ;
}
2020-12-27 14:30:33 +01:00
if ( rect - > flags & CANVAS_RECT_MSDF ) {
push_constant . flags | = FLAGS_USE_MSDF ;
push_constant . msdf [ 0 ] = rect - > px_range ; // Pixel range.
push_constant . msdf [ 1 ] = rect - > outline ; // Outline size.
push_constant . msdf [ 2 ] = 0.f ; // Reserved.
push_constant . msdf [ 3 ] = 0.f ; // Reserved.
2022-08-12 13:03:28 +02:00
} else if ( rect - > flags & CANVAS_RECT_LCD ) {
push_constant . flags | = FLAGS_USE_LCD ;
2020-12-27 14:30:33 +01:00
}
2023-08-03 14:10:01 +02:00
Color modulated = rect - > modulate * base_color ;
if ( use_linear_colors ) {
modulated = modulated . srgb_to_linear ( ) ;
}
push_constant . modulation [ 0 ] = modulated . r ;
push_constant . modulation [ 1 ] = modulated . g ;
push_constant . modulation [ 2 ] = modulated . b ;
push_constant . modulation [ 3 ] = modulated . a ;
2019-06-16 04:45:24 +02:00
push_constant . src_rect [ 0 ] = src_rect . position . x ;
push_constant . src_rect [ 1 ] = src_rect . position . y ;
push_constant . src_rect [ 2 ] = src_rect . size . width ;
push_constant . src_rect [ 3 ] = src_rect . size . height ;
push_constant . dst_rect [ 0 ] = dst_rect . position . x ;
push_constant . dst_rect [ 1 ] = dst_rect . position . y ;
push_constant . dst_rect [ 2 ] = dst_rect . size . width ;
push_constant . dst_rect [ 3 ] = dst_rect . size . height ;
RD : : get_singleton ( ) - > draw_list_set_push_constant ( p_draw_list , & push_constant , sizeof ( PushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_bind_index_array ( p_draw_list , shader . quad_index_array ) ;
RD : : get_singleton ( ) - > draw_list_draw ( p_draw_list , true ) ;
} break ;
case Item : : Command : : TYPE_NINEPATCH : {
const Item : : CommandNinePatch * np = static_cast < const Item : : CommandNinePatch * > ( c ) ;
//bind pipeline
{
2019-07-10 22:44:55 +02:00
RID pipeline = pipeline_variants - > variants [ light_mode ] [ PIPELINE_VARIANT_NINEPATCH ] . get_render_pipeline ( RD : : INVALID_ID , p_framebuffer_format ) ;
2019-06-16 04:45:24 +02:00
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( p_draw_list , pipeline ) ;
}
//bind textures
2020-10-24 17:15:43 +02:00
_bind_canvas_texture ( p_draw_list , np - > texture , current_filter , current_repeat , last_texture , push_constant , texpixel_size ) ;
2019-07-05 03:54:32 +02:00
2019-06-16 04:45:24 +02:00
Rect2 src_rect ;
Rect2 dst_rect ( np - > rect . position . x , np - > rect . position . y , np - > rect . size . x , np - > rect . size . y ) ;
2020-10-24 17:15:43 +02:00
if ( np - > texture = = RID ( ) ) {
2019-06-16 04:45:24 +02:00
texpixel_size = Size2 ( 1 , 1 ) ;
src_rect = Rect2 ( 0 , 0 , 1 , 1 ) ;
} else {
if ( np - > source ! = Rect2 ( ) ) {
src_rect = Rect2 ( np - > source . position . x * texpixel_size . width , np - > source . position . y * texpixel_size . height , np - > source . size . x * texpixel_size . width , np - > source . size . y * texpixel_size . height ) ;
2020-10-24 17:15:43 +02:00
push_constant . color_texture_pixel_size [ 0 ] = 1.0 / np - > source . size . width ;
push_constant . color_texture_pixel_size [ 1 ] = 1.0 / np - > source . size . height ;
2019-06-16 04:45:24 +02:00
} else {
src_rect = Rect2 ( 0 , 0 , 1 , 1 ) ;
}
}
2023-08-03 14:10:01 +02:00
Color modulated = np - > color * base_color ;
if ( use_linear_colors ) {
modulated = modulated . srgb_to_linear ( ) ;
}
push_constant . modulation [ 0 ] = modulated . r ;
push_constant . modulation [ 1 ] = modulated . g ;
push_constant . modulation [ 2 ] = modulated . b ;
push_constant . modulation [ 3 ] = modulated . a ;
2019-06-16 04:45:24 +02:00
push_constant . src_rect [ 0 ] = src_rect . position . x ;
push_constant . src_rect [ 1 ] = src_rect . position . y ;
push_constant . src_rect [ 2 ] = src_rect . size . width ;
push_constant . src_rect [ 3 ] = src_rect . size . height ;
push_constant . dst_rect [ 0 ] = dst_rect . position . x ;
push_constant . dst_rect [ 1 ] = dst_rect . position . y ;
push_constant . dst_rect [ 2 ] = dst_rect . size . width ;
push_constant . dst_rect [ 3 ] = dst_rect . size . height ;
2019-06-19 22:03:19 +02:00
push_constant . flags | = int ( np - > axis_x ) < < FLAGS_NINEPATCH_H_MODE_SHIFT ;
push_constant . flags | = int ( np - > axis_y ) < < FLAGS_NINEPATCH_V_MODE_SHIFT ;
2019-06-16 04:45:24 +02:00
if ( np - > draw_center ) {
push_constant . flags | = FLAGS_NINEPACH_DRAW_CENTER ;
}
2020-12-22 17:24:29 +01:00
push_constant . ninepatch_margins [ 0 ] = np - > margin [ SIDE_LEFT ] ;
push_constant . ninepatch_margins [ 1 ] = np - > margin [ SIDE_TOP ] ;
push_constant . ninepatch_margins [ 2 ] = np - > margin [ SIDE_RIGHT ] ;
push_constant . ninepatch_margins [ 3 ] = np - > margin [ SIDE_BOTTOM ] ;
2019-06-16 04:45:24 +02:00
RD : : get_singleton ( ) - > draw_list_set_push_constant ( p_draw_list , & push_constant , sizeof ( PushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_bind_index_array ( p_draw_list , shader . quad_index_array ) ;
RD : : get_singleton ( ) - > draw_list_draw ( p_draw_list , true ) ;
Fix various typos with codespell
Found via `codespell -q 3 -S ./thirdparty,*.po,./DONORS.md -L ackward,ang,ans,ba,beng,cas,childs,childrens,dof,doubleclick,fave,findn,hist,inout,leapyear,lod,nd,numer,ois,ony,paket,seeked,sinc,switchs,te,uint`
2021-07-07 17:17:32 +02:00
// Restore if overridden.
2020-10-24 17:15:43 +02:00
push_constant . color_texture_pixel_size [ 0 ] = texpixel_size . x ;
push_constant . color_texture_pixel_size [ 1 ] = texpixel_size . y ;
2019-06-16 04:45:24 +02:00
} break ;
2019-06-24 21:13:06 +02:00
case Item : : Command : : TYPE_POLYGON : {
const Item : : CommandPolygon * polygon = static_cast < const Item : : CommandPolygon * > ( c ) ;
PolygonBuffers * pb = polygon_buffers . polygons . getptr ( polygon - > polygon . polygon_id ) ;
ERR_CONTINUE ( ! pb ) ;
//bind pipeline
{
2020-03-27 19:21:27 +01:00
static const PipelineVariant variant [ RS : : PRIMITIVE_MAX ] = { PIPELINE_VARIANT_ATTRIBUTE_POINTS , PIPELINE_VARIANT_ATTRIBUTE_LINES , PIPELINE_VARIANT_ATTRIBUTE_LINES_STRIP , PIPELINE_VARIANT_ATTRIBUTE_TRIANGLES , PIPELINE_VARIANT_ATTRIBUTE_TRIANGLE_STRIP } ;
ERR_CONTINUE ( polygon - > primitive < 0 | | polygon - > primitive > = RS : : PRIMITIVE_MAX ) ;
2019-07-10 22:44:55 +02:00
RID pipeline = pipeline_variants - > variants [ light_mode ] [ variant [ polygon - > primitive ] ] . get_render_pipeline ( pb - > vertex_format_id , p_framebuffer_format ) ;
2019-06-24 21:13:06 +02:00
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( p_draw_list , pipeline ) ;
}
2019-06-16 04:45:24 +02:00
2020-03-27 19:21:27 +01:00
if ( polygon - > primitive = = RS : : PRIMITIVE_LINES ) {
2019-06-24 21:13:06 +02:00
//not supported in most hardware, so pointless
//RD::get_singleton()->draw_list_set_line_width(p_draw_list, polygon->line_width);
}
2019-06-16 04:45:24 +02:00
2019-06-24 21:13:06 +02:00
//bind textures
2019-06-16 04:45:24 +02:00
2020-10-24 17:15:43 +02:00
_bind_canvas_texture ( p_draw_list , polygon - > texture , current_filter , current_repeat , last_texture , push_constant , texpixel_size ) ;
2019-07-05 03:54:32 +02:00
2023-08-03 14:10:01 +02:00
Color color = base_color ;
if ( use_linear_colors ) {
color = color . srgb_to_linear ( ) ;
}
push_constant . modulation [ 0 ] = color . r ;
push_constant . modulation [ 1 ] = color . g ;
push_constant . modulation [ 2 ] = color . b ;
push_constant . modulation [ 3 ] = color . a ;
2019-06-16 04:45:24 +02:00
2019-06-24 21:13:06 +02:00
for ( int j = 0 ; j < 4 ; j + + ) {
push_constant . src_rect [ j ] = 0 ;
push_constant . dst_rect [ j ] = 0 ;
push_constant . ninepatch_margins [ j ] = 0 ;
2019-06-16 04:45:24 +02:00
}
2019-06-24 21:13:06 +02:00
RD : : get_singleton ( ) - > draw_list_set_push_constant ( p_draw_list , & push_constant , sizeof ( PushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_bind_vertex_array ( p_draw_list , pb - > vertex_array ) ;
if ( pb - > indices . is_valid ( ) ) {
RD : : get_singleton ( ) - > draw_list_bind_index_array ( p_draw_list , pb - > indices ) ;
}
RD : : get_singleton ( ) - > draw_list_draw ( p_draw_list , pb - > indices . is_valid ( ) ) ;
2019-06-16 04:45:24 +02:00
} break ;
2019-06-24 21:13:06 +02:00
case Item : : Command : : TYPE_PRIMITIVE : {
const Item : : CommandPrimitive * primitive = static_cast < const Item : : CommandPrimitive * > ( c ) ;
2019-06-16 04:45:24 +02:00
2019-06-24 21:13:06 +02:00
//bind pipeline
{
static const PipelineVariant variant [ 4 ] = { PIPELINE_VARIANT_PRIMITIVE_POINTS , PIPELINE_VARIANT_PRIMITIVE_LINES , PIPELINE_VARIANT_PRIMITIVE_TRIANGLES , PIPELINE_VARIANT_PRIMITIVE_TRIANGLES } ;
ERR_CONTINUE ( primitive - > point_count = = 0 | | primitive - > point_count > 4 ) ;
2019-07-10 22:44:55 +02:00
RID pipeline = pipeline_variants - > variants [ light_mode ] [ variant [ primitive - > point_count - 1 ] ] . get_render_pipeline ( RD : : INVALID_ID , p_framebuffer_format ) ;
2019-06-24 21:13:06 +02:00
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( p_draw_list , pipeline ) ;
}
2019-06-16 04:45:24 +02:00
2019-06-24 21:13:06 +02:00
//bind textures
2022-10-14 01:27:58 +02:00
_bind_canvas_texture ( p_draw_list , primitive - > texture , current_filter , current_repeat , last_texture , push_constant , texpixel_size ) ;
2019-07-05 03:54:32 +02:00
2022-03-08 15:10:48 +01:00
RD : : get_singleton ( ) - > draw_list_bind_index_array ( p_draw_list , primitive_arrays . index_array [ MIN ( 3u , primitive - > point_count ) - 1 ] ) ;
2019-06-26 20:11:52 +02:00
2022-03-08 15:10:48 +01:00
for ( uint32_t j = 0 ; j < MIN ( 3u , primitive - > point_count ) ; j + + ) {
2019-06-24 21:13:06 +02:00
push_constant . points [ j * 2 + 0 ] = primitive - > points [ j ] . x ;
push_constant . points [ j * 2 + 1 ] = primitive - > points [ j ] . y ;
push_constant . uvs [ j * 2 + 0 ] = primitive - > uvs [ j ] . x ;
push_constant . uvs [ j * 2 + 1 ] = primitive - > uvs [ j ] . y ;
2019-07-05 03:54:32 +02:00
Color col = primitive - > colors [ j ] * base_color ;
2023-08-03 14:10:01 +02:00
if ( use_linear_colors ) {
col = col . srgb_to_linear ( ) ;
}
2019-06-24 21:13:06 +02:00
push_constant . colors [ j * 2 + 0 ] = ( uint32_t ( Math : : make_half_float ( col . g ) ) < < 16 ) | Math : : make_half_float ( col . r ) ;
push_constant . colors [ j * 2 + 1 ] = ( uint32_t ( Math : : make_half_float ( col . a ) ) < < 16 ) | Math : : make_half_float ( col . b ) ;
2019-06-16 04:45:24 +02:00
}
2019-06-24 21:13:06 +02:00
RD : : get_singleton ( ) - > draw_list_set_push_constant ( p_draw_list , & push_constant , sizeof ( PushConstant ) ) ;
2019-06-26 20:11:52 +02:00
RD : : get_singleton ( ) - > draw_list_draw ( p_draw_list , true ) ;
2019-06-24 21:13:06 +02:00
2019-06-26 20:11:52 +02:00
if ( primitive - > point_count = = 4 ) {
for ( uint32_t j = 1 ; j < 3 ; j + + ) {
//second half of triangle
push_constant . points [ j * 2 + 0 ] = primitive - > points [ j + 1 ] . x ;
push_constant . points [ j * 2 + 1 ] = primitive - > points [ j + 1 ] . y ;
push_constant . uvs [ j * 2 + 0 ] = primitive - > uvs [ j + 1 ] . x ;
push_constant . uvs [ j * 2 + 1 ] = primitive - > uvs [ j + 1 ] . y ;
2019-07-05 03:54:32 +02:00
Color col = primitive - > colors [ j + 1 ] * base_color ;
2023-08-03 14:10:01 +02:00
if ( use_linear_colors ) {
col = col . srgb_to_linear ( ) ;
}
2019-06-26 20:11:52 +02:00
push_constant . colors [ j * 2 + 0 ] = ( uint32_t ( Math : : make_half_float ( col . g ) ) < < 16 ) | Math : : make_half_float ( col . r ) ;
push_constant . colors [ j * 2 + 1 ] = ( uint32_t ( Math : : make_half_float ( col . a ) ) < < 16 ) | Math : : make_half_float ( col . b ) ;
}
2019-06-24 21:13:06 +02:00
2019-06-26 20:11:52 +02:00
RD : : get_singleton ( ) - > draw_list_set_push_constant ( p_draw_list , & push_constant , sizeof ( PushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_draw ( p_draw_list , true ) ;
}
2019-06-16 04:45:24 +02:00
} break ;
2020-02-18 14:57:11 +01:00
case Item : : Command : : TYPE_MESH :
case Item : : Command : : TYPE_MULTIMESH :
case Item : : Command : : TYPE_PARTICLES : {
2021-05-10 18:12:44 +02:00
RID mesh ;
RID mesh_instance ;
RID texture ;
Color modulate ( 1 , 1 , 1 , 1 ) ;
float world_backup [ 6 ] ;
int instance_count = 1 ;
for ( int j = 0 ; j < 6 ; j + + ) {
world_backup [ j ] = push_constant . world [ j ] ;
2019-06-16 04:45:24 +02:00
}
2021-05-10 18:12:44 +02:00
if ( c - > type = = Item : : Command : : TYPE_MESH ) {
const Item : : CommandMesh * m = static_cast < const Item : : CommandMesh * > ( c ) ;
mesh = m - > mesh ;
mesh_instance = m - > mesh_instance ;
texture = m - > texture ;
modulate = m - > modulate ;
2021-06-22 10:30:11 +02:00
_update_transform_2d_to_mat2x3 ( base_transform * draw_transform * m - > transform , push_constant . world ) ;
2021-05-10 18:12:44 +02:00
} else if ( c - > type = = Item : : Command : : TYPE_MULTIMESH ) {
const Item : : CommandMultiMesh * mm = static_cast < const Item : : CommandMultiMesh * > ( c ) ;
RID multimesh = mm - > multimesh ;
2022-04-02 07:29:04 +02:00
mesh = mesh_storage - > multimesh_get_mesh ( multimesh ) ;
2021-05-10 18:12:44 +02:00
texture = mm - > texture ;
2022-04-02 07:29:04 +02:00
if ( mesh_storage - > multimesh_get_transform_format ( multimesh ) ! = RS : : MULTIMESH_TRANSFORM_2D ) {
2021-05-10 18:12:44 +02:00
break ;
}
2019-06-16 04:45:24 +02:00
2022-04-02 07:29:04 +02:00
instance_count = mesh_storage - > multimesh_get_instances_to_draw ( multimesh ) ;
2019-06-16 04:45:24 +02:00
2021-11-13 03:38:19 +01:00
if ( instance_count = = 0 ) {
break ;
}
2022-04-02 07:29:04 +02:00
RID uniform_set = mesh_storage - > multimesh_get_2d_uniform_set ( multimesh , shader . default_version_rd_shader , TRANSFORMS_UNIFORM_SET ) ;
2021-05-10 18:12:44 +02:00
RD : : get_singleton ( ) - > draw_list_bind_uniform_set ( p_draw_list , uniform_set , TRANSFORMS_UNIFORM_SET ) ;
push_constant . flags | = 1 ; //multimesh, trails disabled
2022-04-02 07:29:04 +02:00
if ( mesh_storage - > multimesh_uses_colors ( multimesh ) ) {
2021-05-10 18:12:44 +02:00
push_constant . flags | = FLAGS_INSTANCING_HAS_COLORS ;
2019-06-16 04:45:24 +02:00
}
2022-04-02 07:29:04 +02:00
if ( mesh_storage - > multimesh_uses_custom_data ( multimesh ) ) {
2021-05-10 18:12:44 +02:00
push_constant . flags | = FLAGS_INSTANCING_HAS_CUSTOM_DATA ;
}
} else if ( c - > type = = Item : : Command : : TYPE_PARTICLES ) {
const Item : : CommandParticles * pt = static_cast < const Item : : CommandParticles * > ( c ) ;
2022-04-12 13:41:50 +02:00
ERR_BREAK ( particles_storage - > particles_get_mode ( pt - > particles ) ! = RS : : PARTICLES_MODE_2D ) ;
particles_storage - > particles_request_process ( pt - > particles ) ;
2021-05-20 16:25:06 +02:00
2022-12-22 20:04:34 +01:00
if ( particles_storage - > particles_is_inactive ( pt - > particles ) | | particles_storage - > particles_get_frame_counter ( pt - > particles ) = = 0 ) {
2021-05-10 18:12:44 +02:00
break ;
}
2021-05-20 16:25:06 +02:00
RenderingServerDefault : : redraw_request ( ) ; // active particles means redraw request
2022-04-12 13:41:50 +02:00
int dpc = particles_storage - > particles_get_draw_passes ( pt - > particles ) ;
2021-05-10 18:12:44 +02:00
if ( dpc = = 0 ) {
break ; //nothing to draw
}
uint32_t divisor = 1 ;
2022-04-12 13:41:50 +02:00
instance_count = particles_storage - > particles_get_amount ( pt - > particles , divisor ) ;
2019-06-16 04:45:24 +02:00
2022-04-12 13:41:50 +02:00
RID uniform_set = particles_storage - > particles_get_instance_buffer_uniform_set ( pt - > particles , shader . default_version_rd_shader , TRANSFORMS_UNIFORM_SET ) ;
2021-05-10 18:12:44 +02:00
RD : : get_singleton ( ) - > draw_list_bind_uniform_set ( p_draw_list , uniform_set , TRANSFORMS_UNIFORM_SET ) ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
push_constant . flags | = divisor ;
instance_count / = divisor ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
push_constant . flags | = FLAGS_INSTANCING_HAS_COLORS ;
push_constant . flags | = FLAGS_INSTANCING_HAS_CUSTOM_DATA ;
2019-06-16 04:45:24 +02:00
2022-04-12 13:41:50 +02:00
mesh = particles_storage - > particles_get_draw_pass_mesh ( pt - > particles , 0 ) ; //higher ones are ignored
2021-05-10 18:12:44 +02:00
texture = pt - > texture ;
2021-05-20 16:25:06 +02:00
2022-04-12 13:41:50 +02:00
if ( particles_storage - > particles_has_collision ( pt - > particles ) & & texture_storage - > render_target_is_sdf_enabled ( p_render_target ) ) {
2021-05-20 16:25:06 +02:00
//pass collision information
2022-10-28 01:39:28 +02:00
Transform2D xform = p_item - > final_transform ;
2021-05-20 16:25:06 +02:00
2022-04-07 16:00:51 +02:00
RID sdf_texture = texture_storage - > render_target_get_sdf_texture ( p_render_target ) ;
2021-05-20 16:25:06 +02:00
Rect2 to_screen ;
{
2022-04-07 16:00:51 +02:00
Rect2 sdf_rect = texture_storage - > render_target_get_sdf_rect ( p_render_target ) ;
2021-05-20 16:25:06 +02:00
to_screen . size = Vector2 ( 1.0 / sdf_rect . size . width , 1.0 / sdf_rect . size . height ) ;
to_screen . position = - sdf_rect . position * to_screen . size ;
}
2022-04-12 13:41:50 +02:00
particles_storage - > particles_set_canvas_sdf_collision ( pt - > particles , true , xform , to_screen , sdf_texture ) ;
2021-05-20 16:25:06 +02:00
} else {
2022-04-12 13:41:50 +02:00
particles_storage - > particles_set_canvas_sdf_collision ( pt - > particles , false , Transform2D ( ) , Rect2 ( ) , RID ( ) ) ;
2021-05-20 16:25:06 +02:00
}
2022-12-07 19:03:08 +01:00
// Signal that SDF texture needs to be updated.
r_sdf_used | = particles_storage - > particles_has_collision ( pt - > particles ) ;
2021-05-10 18:12:44 +02:00
}
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
if ( mesh . is_null ( ) ) {
2019-06-16 04:45:24 +02:00
break ;
2021-05-10 18:12:44 +02:00
}
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
_bind_canvas_texture ( p_draw_list , texture , current_filter , current_repeat , last_texture , push_constant , texpixel_size ) ;
2019-06-16 04:45:24 +02:00
2022-04-02 07:29:04 +02:00
uint32_t surf_count = mesh_storage - > mesh_get_surface_count ( mesh ) ;
2021-05-10 18:12:44 +02:00
static const PipelineVariant variant [ RS : : PRIMITIVE_MAX ] = { PIPELINE_VARIANT_ATTRIBUTE_POINTS , PIPELINE_VARIANT_ATTRIBUTE_LINES , PIPELINE_VARIANT_ATTRIBUTE_LINES_STRIP , PIPELINE_VARIANT_ATTRIBUTE_TRIANGLES , PIPELINE_VARIANT_ATTRIBUTE_TRIANGLE_STRIP } ;
2019-06-16 04:45:24 +02:00
2023-08-03 14:10:01 +02:00
Color modulated = modulate * base_color ;
if ( use_linear_colors ) {
modulated = modulated . srgb_to_linear ( ) ;
}
push_constant . modulation [ 0 ] = modulated . r ;
push_constant . modulation [ 1 ] = modulated . g ;
push_constant . modulation [ 2 ] = modulated . b ;
push_constant . modulation [ 3 ] = modulated . a ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
for ( int j = 0 ; j < 4 ; j + + ) {
push_constant . src_rect [ j ] = 0 ;
push_constant . dst_rect [ j ] = 0 ;
push_constant . ninepatch_margins [ j ] = 0 ;
2019-06-16 04:45:24 +02:00
}
2021-05-10 18:12:44 +02:00
for ( uint32_t j = 0 ; j < surf_count ; j + + ) {
2022-04-02 07:29:04 +02:00
void * surface = mesh_storage - > mesh_get_surface ( mesh , j ) ;
2019-06-16 04:45:24 +02:00
2022-04-02 07:29:04 +02:00
RS : : PrimitiveType primitive = mesh_storage - > mesh_surface_get_primitive ( surface ) ;
2021-05-10 18:12:44 +02:00
ERR_CONTINUE ( primitive < 0 | | primitive > = RS : : PRIMITIVE_MAX ) ;
2019-06-16 04:45:24 +02:00
2023-08-29 21:04:32 +02:00
uint64_t input_mask = pipeline_variants - > variants [ light_mode ] [ variant [ primitive ] ] . get_vertex_input_mask ( ) ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
RID vertex_array ;
RD : : VertexFormatID vertex_format = RD : : INVALID_FORMAT_ID ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
if ( mesh_instance . is_valid ( ) ) {
2023-08-09 19:27:28 +02:00
mesh_storage - > mesh_instance_surface_get_vertex_arrays_and_format ( mesh_instance , j , input_mask , false , vertex_array , vertex_format ) ;
2019-06-16 04:45:24 +02:00
} else {
2023-08-09 19:27:28 +02:00
mesh_storage - > mesh_surface_get_vertex_arrays_and_format ( surface , input_mask , false , vertex_array , vertex_format ) ;
2019-06-16 04:45:24 +02:00
}
2021-05-10 18:12:44 +02:00
RID pipeline = pipeline_variants - > variants [ light_mode ] [ variant [ primitive ] ] . get_render_pipeline ( vertex_format , p_framebuffer_format ) ;
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( p_draw_list , pipeline ) ;
2019-06-16 04:45:24 +02:00
2022-04-02 07:29:04 +02:00
RID index_array = mesh_storage - > mesh_surface_get_index_array ( surface , 0 ) ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
if ( index_array . is_valid ( ) ) {
RD : : get_singleton ( ) - > draw_list_bind_index_array ( p_draw_list , index_array ) ;
2019-06-16 04:45:24 +02:00
}
2021-05-10 18:12:44 +02:00
RD : : get_singleton ( ) - > draw_list_bind_vertex_array ( p_draw_list , vertex_array ) ;
RD : : get_singleton ( ) - > draw_list_set_push_constant ( p_draw_list , & push_constant , sizeof ( PushConstant ) ) ;
2019-06-16 04:45:24 +02:00
2021-05-10 18:12:44 +02:00
RD : : get_singleton ( ) - > draw_list_draw ( p_draw_list , index_array . is_valid ( ) , instance_count ) ;
2019-06-16 04:45:24 +02:00
}
2021-05-10 18:12:44 +02:00
for ( int j = 0 ; j < 6 ; j + + ) {
push_constant . world [ j ] = world_backup [ j ] ;
2019-06-16 04:45:24 +02:00
}
} break ;
case Item : : Command : : TYPE_TRANSFORM : {
2019-06-24 21:13:06 +02:00
const Item : : CommandTransform * transform = static_cast < const Item : : CommandTransform * > ( c ) ;
2021-06-22 10:30:11 +02:00
draw_transform = transform - > xform ;
2019-06-25 03:24:07 +02:00
_update_transform_2d_to_mat2x3 ( base_transform * transform - > xform , push_constant . world ) ;
2019-06-16 04:45:24 +02:00
} break ;
case Item : : Command : : TYPE_CLIP_IGNORE : {
2019-06-24 21:13:06 +02:00
const Item : : CommandClipIgnore * ci = static_cast < const Item : : CommandClipIgnore * > ( c ) ;
2019-06-16 04:45:24 +02:00
if ( current_clip ) {
if ( ci - > ignore ! = reclip ) {
2019-06-24 21:13:06 +02:00
if ( ci - > ignore ) {
RD : : get_singleton ( ) - > draw_list_disable_scissor ( p_draw_list ) ;
2019-06-16 04:45:24 +02:00
reclip = true ;
} else {
2019-06-24 21:13:06 +02:00
RD : : get_singleton ( ) - > draw_list_enable_scissor ( p_draw_list , current_clip - > final_clip_rect ) ;
2019-06-16 04:45:24 +02:00
reclip = false ;
}
}
}
} break ;
2021-06-17 16:30:20 +02:00
case Item : : Command : : TYPE_ANIMATION_SLICE : {
const Item : : CommandAnimationSlice * as = static_cast < const Item : : CommandAnimationSlice * > ( c ) ;
2023-01-29 10:14:22 +01:00
double current_time = RendererCompositorRD : : get_singleton ( ) - > get_total_time ( ) ;
2021-06-17 16:30:20 +02:00
double local_time = Math : : fposmod ( current_time - as - > offset , as - > animation_length ) ;
skipping = ! ( local_time > = as - > slice_begin & & local_time < as - > slice_end ) ;
RenderingServerDefault : : redraw_request ( ) ; // animation visible means redraw request
} break ;
2019-06-16 04:45:24 +02:00
}
2019-06-27 00:48:05 +02:00
c = c - > next ;
2019-06-16 04:45:24 +02:00
}
2019-06-24 21:13:06 +02:00
if ( current_clip & & reclip ) {
//will make it re-enable clipping if needed afterwards
2020-04-02 01:20:12 +02:00
current_clip = nullptr ;
2019-06-24 21:13:06 +02:00
}
2019-06-16 04:45:24 +02:00
}
2020-12-04 19:26:24 +01:00
RID RendererCanvasRenderRD : : _create_base_uniform_set ( RID p_to_render_target , bool p_backbuffer ) {
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
2022-04-12 13:41:50 +02:00
RendererRD : : MaterialStorage * material_storage = RendererRD : : MaterialStorage : : get_singleton ( ) ;
2022-04-07 16:00:51 +02:00
2020-10-24 17:15:43 +02:00
//re create canvas state
Vector < RD : : Uniform > uniforms ;
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_UNIFORM_BUFFER ;
2020-10-24 17:15:43 +02:00
u . binding = 1 ;
2022-03-06 12:57:09 +01:00
u . append_id ( state . canvas_state_buffer ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_UNIFORM_BUFFER ;
2020-10-24 17:15:43 +02:00
u . binding = 2 ;
2022-03-06 12:57:09 +01:00
u . append_id ( state . lights_uniform_buffer ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_TEXTURE ;
2020-10-24 17:15:43 +02:00
u . binding = 3 ;
2022-04-07 16:00:51 +02:00
u . append_id ( RendererRD : : TextureStorage : : get_singleton ( ) - > decal_atlas_get_texture ( ) ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_TEXTURE ;
2020-10-24 17:15:43 +02:00
u . binding = 4 ;
2022-03-06 12:57:09 +01:00
u . append_id ( state . shadow_texture ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_SAMPLER ;
2020-10-24 17:15:43 +02:00
u . binding = 5 ;
2022-03-06 12:57:09 +01:00
u . append_id ( state . shadow_sampler ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_TEXTURE ;
2020-10-24 17:15:43 +02:00
u . binding = 6 ;
RID screen ;
if ( p_backbuffer ) {
2022-04-07 16:00:51 +02:00
screen = texture_storage - > render_target_get_rd_texture ( p_to_render_target ) ;
2020-10-24 17:15:43 +02:00
} else {
2022-04-07 16:00:51 +02:00
screen = texture_storage - > render_target_get_rd_backbuffer ( p_to_render_target ) ;
2020-10-24 17:15:43 +02:00
if ( screen . is_null ( ) ) { //unallocated backbuffer
2022-08-01 03:59:14 +02:00
screen = RendererRD : : TextureStorage : : get_singleton ( ) - > texture_rd_get_default ( RendererRD : : TextureStorage : : DEFAULT_RD_TEXTURE_WHITE ) ;
2020-10-24 17:15:43 +02:00
}
}
2022-03-06 12:57:09 +01:00
u . append_id ( screen ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
2020-11-26 13:50:21 +01:00
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_TEXTURE ;
2020-11-26 13:50:21 +01:00
u . binding = 7 ;
2022-04-07 16:00:51 +02:00
RID sdf = texture_storage - > render_target_get_sdf_texture ( p_to_render_target ) ;
2022-03-06 12:57:09 +01:00
u . append_id ( sdf ) ;
2020-11-26 13:50:21 +01:00
uniforms . push_back ( u ) ;
}
2020-10-24 17:15:43 +02:00
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_STORAGE_BUFFER ;
2020-11-26 13:50:21 +01:00
u . binding = 9 ;
2022-04-03 18:56:43 +02:00
u . append_id ( RendererRD : : MaterialStorage : : get_singleton ( ) - > global_shader_uniforms_get_storage_buffer ( ) ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
2023-09-05 16:41:57 +02:00
uniforms . append_array ( material_storage - > samplers_rd_get_default ( ) . get_uniforms ( SAMPLERS_BINDING_FIRST_INDEX ) ) ;
2023-05-26 23:35:16 +02:00
2020-10-24 17:15:43 +02:00
RID uniform_set = RD : : get_singleton ( ) - > uniform_set_create ( uniforms , shader . default_version_rd_shader , BASE_UNIFORM_SET ) ;
if ( p_backbuffer ) {
2022-04-07 16:00:51 +02:00
texture_storage - > render_target_set_backbuffer_uniform_set ( p_to_render_target , uniform_set ) ;
2020-10-24 17:15:43 +02:00
} else {
2022-04-07 16:00:51 +02:00
texture_storage - > render_target_set_framebuffer_uniform_set ( p_to_render_target , uniform_set ) ;
2020-10-24 17:15:43 +02:00
}
return uniform_set ;
}
2022-12-07 19:03:08 +01:00
void RendererCanvasRenderRD : : _render_items ( RID p_to_render_target , int p_item_count , const Transform2D & p_canvas_transform_inverse , Light * p_lights , bool & r_sdf_used , bool p_to_backbuffer ) {
2022-03-21 12:25:25 +01:00
RendererRD : : MaterialStorage * material_storage = RendererRD : : MaterialStorage : : get_singleton ( ) ;
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
2020-04-02 01:20:12 +02:00
Item * current_clip = nullptr ;
2019-06-16 04:45:24 +02:00
2019-07-05 03:54:32 +02:00
Transform2D canvas_transform_inverse = p_canvas_transform_inverse ;
2019-06-16 04:45:24 +02:00
2020-10-28 19:34:27 +01:00
RID framebuffer ;
RID fb_uniform_set ;
2019-06-24 21:13:06 +02:00
bool clear = false ;
2020-10-28 19:34:27 +01:00
Vector < Color > clear_colors ;
if ( p_to_backbuffer ) {
2022-04-07 16:00:51 +02:00
framebuffer = texture_storage - > render_target_get_rd_backbuffer_framebuffer ( p_to_render_target ) ;
fb_uniform_set = texture_storage - > render_target_get_backbuffer_uniform_set ( p_to_render_target ) ;
2020-10-28 19:34:27 +01:00
} else {
2022-04-07 16:00:51 +02:00
framebuffer = texture_storage - > render_target_get_rd_framebuffer ( p_to_render_target ) ;
2020-10-28 19:34:27 +01:00
2022-04-07 16:00:51 +02:00
if ( texture_storage - > render_target_is_clear_requested ( p_to_render_target ) ) {
2020-10-28 19:34:27 +01:00
clear = true ;
2022-04-07 16:00:51 +02:00
clear_colors . push_back ( texture_storage - > render_target_get_clear_request_color ( p_to_render_target ) ) ;
texture_storage - > render_target_disable_clear_request ( p_to_render_target ) ;
2020-10-28 19:34:27 +01:00
}
2022-10-10 13:04:01 +02:00
// TODO: Obtain from framebuffer format eventually when this is implemented.
2022-04-07 16:00:51 +02:00
fb_uniform_set = texture_storage - > render_target_get_framebuffer_uniform_set ( p_to_render_target ) ;
2020-10-28 19:34:27 +01:00
}
2020-10-24 17:15:43 +02:00
if ( fb_uniform_set . is_null ( ) | | ! RD : : get_singleton ( ) - > uniform_set_is_valid ( fb_uniform_set ) ) {
2020-10-28 19:34:27 +01:00
fb_uniform_set = _create_base_uniform_set ( p_to_render_target , p_to_backbuffer ) ;
2020-10-24 17:15:43 +02:00
}
2019-07-10 22:44:55 +02:00
RD : : FramebufferFormatID fb_format = RD : : get_singleton ( ) - > framebuffer_get_format ( framebuffer ) ;
2019-06-16 04:45:24 +02:00
2019-10-03 22:39:08 +02:00
RD : : DrawListID draw_list = RD : : get_singleton ( ) - > draw_list_begin ( framebuffer , clear ? RD : : INITIAL_ACTION_CLEAR : RD : : INITIAL_ACTION_KEEP , RD : : FINAL_ACTION_READ , RD : : INITIAL_ACTION_KEEP , RD : : FINAL_ACTION_DISCARD , clear_colors ) ;
2019-06-16 04:45:24 +02:00
2020-10-24 17:15:43 +02:00
RD : : get_singleton ( ) - > draw_list_bind_uniform_set ( draw_list , fb_uniform_set , BASE_UNIFORM_SET ) ;
RD : : get_singleton ( ) - > draw_list_bind_uniform_set ( draw_list , state . default_transforms_uniform_set , TRANSFORMS_UNIFORM_SET ) ;
2019-07-21 16:31:30 +02:00
RID prev_material ;
PipelineVariants * pipeline_variants = & shader . pipeline_variants ;
2019-06-16 04:45:24 +02:00
for ( int i = 0 ; i < p_item_count ; i + + ) {
Item * ci = items [ i ] ;
if ( current_clip ! = ci - > final_clip_owner ) {
current_clip = ci - > final_clip_owner ;
//setup clip
if ( current_clip ) {
RD : : get_singleton ( ) - > draw_list_enable_scissor ( draw_list , current_clip - > final_clip_rect ) ;
} else {
RD : : get_singleton ( ) - > draw_list_disable_scissor ( draw_list ) ;
}
}
2021-08-13 04:45:44 +02:00
RID material = ci - > material_owner = = nullptr ? ci - > material : ci - > material_owner - > material ;
2020-10-28 19:34:27 +01:00
2023-03-07 07:09:39 +01:00
if ( ci - > use_canvas_group ) {
2022-10-12 21:45:47 +02:00
if ( ci - > canvas_group - > mode = = RS : : CANVAS_GROUP_MODE_CLIP_AND_DRAW ) {
2023-03-07 07:09:39 +01:00
material = default_clip_children_material ;
2022-10-07 00:27:11 +02:00
} else {
if ( material . is_null ( ) ) {
2022-10-12 21:45:47 +02:00
if ( ci - > canvas_group - > mode = = RS : : CANVAS_GROUP_MODE_CLIP_ONLY ) {
material = default_clip_children_material ;
} else {
material = default_canvas_group_material ;
}
2022-10-07 00:27:11 +02:00
}
}
2020-10-28 19:34:27 +01:00
}
if ( material ! = prev_material ) {
2022-03-21 12:25:25 +01:00
CanvasMaterialData * material_data = nullptr ;
2020-10-28 19:34:27 +01:00
if ( material . is_valid ( ) ) {
2022-08-01 03:59:14 +02:00
material_data = static_cast < CanvasMaterialData * > ( material_storage - > material_get_data ( material , RendererRD : : MaterialStorage : : SHADER_TYPE_2D ) ) ;
2019-07-21 16:31:30 +02:00
}
if ( material_data ) {
2019-07-27 15:23:24 +02:00
if ( material_data - > shader_data - > version . is_valid ( ) & & material_data - > shader_data - > valid ) {
2019-07-21 16:31:30 +02:00
pipeline_variants = & material_data - > shader_data - > pipeline_variants ;
2021-08-06 10:17:09 +02:00
// Update uniform set.
2023-08-03 14:10:01 +02:00
RID uniform_set = texture_storage - > render_target_is_using_hdr ( p_to_render_target ) ? material_data - > uniform_set : material_data - > uniform_set_srgb ;
if ( uniform_set . is_valid ( ) & & RD : : get_singleton ( ) - > uniform_set_is_valid ( uniform_set ) ) { // Material may not have a uniform set.
RD : : get_singleton ( ) - > draw_list_bind_uniform_set ( draw_list , uniform_set , MATERIAL_UNIFORM_SET ) ;
2022-12-16 01:07:00 +01:00
material_data - > set_as_used ( ) ;
2019-07-21 16:31:30 +02:00
}
} else {
pipeline_variants = & shader . pipeline_variants ;
}
} else {
pipeline_variants = & shader . pipeline_variants ;
}
}
2022-12-07 19:03:08 +01:00
_render_item ( draw_list , p_to_render_target , ci , fb_format , canvas_transform_inverse , current_clip , p_lights , pipeline_variants , r_sdf_used ) ;
2019-07-21 16:31:30 +02:00
2020-10-28 19:34:27 +01:00
prev_material = material ;
2019-06-16 04:45:24 +02:00
}
RD : : get_singleton ( ) - > draw_list_end ( ) ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : canvas_render_items ( RID p_to_render_target , Item * p_item_list , const Color & p_modulate , Light * p_light_list , Light * p_directional_light_list , const Transform2D & p_canvas_transform , RenderingServer : : CanvasItemTextureFilter p_default_filter , RenderingServer : : CanvasItemTextureRepeat p_default_repeat , bool p_snap_2d_vertices_to_pixel , bool & r_sdf_used ) {
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
2022-03-21 12:25:25 +01:00
RendererRD : : MaterialStorage * material_storage = RendererRD : : MaterialStorage : : get_singleton ( ) ;
2022-04-02 07:29:04 +02:00
RendererRD : : MeshStorage * mesh_storage = RendererRD : : MeshStorage : : get_singleton ( ) ;
2022-03-21 12:25:25 +01:00
2020-11-26 13:50:21 +01:00
r_sdf_used = false ;
2019-06-16 04:45:24 +02:00
int item_count = 0 ;
//setup canvas state uniforms if needed
2019-07-10 22:44:55 +02:00
2019-07-05 03:54:32 +02:00
Transform2D canvas_transform_inverse = p_canvas_transform . affine_inverse ( ) ;
2019-06-16 04:45:24 +02:00
2020-11-03 20:51:53 +01:00
//setup directional lights if exist
uint32_t light_count = 0 ;
uint32_t directional_light_count = 0 ;
2019-06-16 04:45:24 +02:00
{
2020-11-03 20:51:53 +01:00
Light * l = p_directional_light_list ;
uint32_t index = 0 ;
2019-06-16 04:45:24 +02:00
2020-11-03 20:51:53 +01:00
while ( l ) {
if ( index = = state . max_lights_per_render ) {
l - > render_index_cache = - 1 ;
l = l - > next_ptr ;
continue ;
}
2019-06-16 04:45:24 +02:00
2021-09-29 19:08:41 +02:00
CanvasLight * clight = canvas_light_owner . get_or_null ( l - > light_internal ) ;
2020-11-03 20:51:53 +01:00
if ( ! clight ) { //unused or invalid texture
l - > render_index_cache = - 1 ;
l = l - > next_ptr ;
ERR_CONTINUE ( ! clight ) ;
}
2019-07-05 03:54:32 +02:00
2022-04-24 23:59:24 +02:00
Vector2 canvas_light_dir = l - > xform_cache . columns [ 1 ] . normalized ( ) ;
2019-07-05 03:54:32 +02:00
2020-11-03 20:51:53 +01:00
state . light_uniforms [ index ] . position [ 0 ] = - canvas_light_dir . x ;
state . light_uniforms [ index ] . position [ 1 ] = - canvas_light_dir . y ;
2019-07-05 03:54:32 +02:00
2020-11-03 20:51:53 +01:00
_update_transform_2d_to_mat2x4 ( clight - > shadow . directional_xform , state . light_uniforms [ index ] . shadow_matrix ) ;
2019-07-27 15:23:24 +02:00
2020-11-03 20:51:53 +01:00
state . light_uniforms [ index ] . height = l - > height ; //0..1 here
2020-10-29 22:09:16 +01:00
2020-11-03 20:51:53 +01:00
for ( int i = 0 ; i < 4 ; i + + ) {
state . light_uniforms [ index ] . shadow_color [ i ] = uint8_t ( CLAMP ( int32_t ( l - > shadow_color [ i ] * 255.0 ) , 0 , 255 ) ) ;
state . light_uniforms [ index ] . color [ i ] = l - > color [ i ] ;
}
2023-03-21 10:18:15 +01:00
state . light_uniforms [ index ] . color [ 3 ] * = l - > energy ; //use alpha for energy, so base color can go separate
2020-11-03 20:51:53 +01:00
if ( state . shadow_fb . is_valid ( ) ) {
state . light_uniforms [ index ] . shadow_pixel_size = ( 1.0 / state . shadow_texture_size ) * ( 1.0 + l - > shadow_smooth ) ;
state . light_uniforms [ index ] . shadow_z_far_inv = 1.0 / clight - > shadow . z_far ;
state . light_uniforms [ index ] . shadow_y_ofs = clight - > shadow . y_offset ;
} else {
state . light_uniforms [ index ] . shadow_pixel_size = 1.0 ;
state . light_uniforms [ index ] . shadow_z_far_inv = 1.0 ;
state . light_uniforms [ index ] . shadow_y_ofs = 0 ;
}
state . light_uniforms [ index ] . flags = l - > blend_mode < < LIGHT_FLAGS_BLEND_SHIFT ;
state . light_uniforms [ index ] . flags | = l - > shadow_filter < < LIGHT_FLAGS_FILTER_SHIFT ;
if ( clight - > shadow . enabled ) {
state . light_uniforms [ index ] . flags | = LIGHT_FLAGS_HAS_SHADOW ;
}
l - > render_index_cache = index ;
index + + ;
l = l - > next_ptr ;
}
light_count = index ;
directional_light_count = light_count ;
using_directional_lights = directional_light_count > 0 ;
2019-06-16 04:45:24 +02:00
}
2019-07-05 03:54:32 +02:00
//setup lights if exist
{
Light * l = p_light_list ;
2020-11-03 20:51:53 +01:00
uint32_t index = light_count ;
2019-07-05 03:54:32 +02:00
while ( l ) {
2019-07-10 22:44:55 +02:00
if ( index = = state . max_lights_per_render ) {
2019-07-05 03:54:32 +02:00
l - > render_index_cache = - 1 ;
l = l - > next_ptr ;
continue ;
}
2021-09-29 19:08:41 +02:00
CanvasLight * clight = canvas_light_owner . get_or_null ( l - > light_internal ) ;
2019-07-10 22:44:55 +02:00
if ( ! clight ) { //unused or invalid texture
2019-07-05 03:54:32 +02:00
l - > render_index_cache = - 1 ;
l = l - > next_ptr ;
ERR_CONTINUE ( ! clight ) ;
}
Vector2 canvas_light_pos = p_canvas_transform . xform ( l - > xform . get_origin ( ) ) ; //convert light position to canvas coordinates, as all computation is done in canvas coords to avoid precision loss
state . light_uniforms [ index ] . position [ 0 ] = canvas_light_pos . x ;
state . light_uniforms [ index ] . position [ 1 ] = canvas_light_pos . y ;
2023-02-17 04:11:24 +01:00
_update_transform_2d_to_mat2x4 ( l - > light_shader_xform . affine_inverse ( ) , state . light_uniforms [ index ] . matrix ) ;
2019-07-07 06:49:40 +02:00
_update_transform_2d_to_mat2x4 ( l - > xform_cache . affine_inverse ( ) , state . light_uniforms [ index ] . shadow_matrix ) ;
2019-07-05 03:54:32 +02:00
2022-04-24 23:59:24 +02:00
state . light_uniforms [ index ] . height = l - > height * ( p_canvas_transform . columns [ 0 ] . length ( ) + p_canvas_transform . columns [ 1 ] . length ( ) ) * 0.5 ; //approximate height conversion to the canvas size, since all calculations are done in canvas coords to avoid precision loss
2019-07-05 03:54:32 +02:00
for ( int i = 0 ; i < 4 ; i + + ) {
2020-10-24 17:15:43 +02:00
state . light_uniforms [ index ] . shadow_color [ i ] = uint8_t ( CLAMP ( int32_t ( l - > shadow_color [ i ] * 255.0 ) , 0 , 255 ) ) ;
2019-07-05 03:54:32 +02:00
state . light_uniforms [ index ] . color [ i ] = l - > color [ i ] ;
}
2023-03-21 10:18:15 +01:00
state . light_uniforms [ index ] . color [ 3 ] * = l - > energy ; //use alpha for energy, so base color can go separate
2019-07-05 03:54:32 +02:00
2020-10-24 17:15:43 +02:00
if ( state . shadow_fb . is_valid ( ) ) {
state . light_uniforms [ index ] . shadow_pixel_size = ( 1.0 / state . shadow_texture_size ) * ( 1.0 + l - > shadow_smooth ) ;
state . light_uniforms [ index ] . shadow_z_far_inv = 1.0 / clight - > shadow . z_far ;
state . light_uniforms [ index ] . shadow_y_ofs = clight - > shadow . y_offset ;
2019-07-05 03:54:32 +02:00
} else {
state . light_uniforms [ index ] . shadow_pixel_size = 1.0 ;
2020-10-24 17:15:43 +02:00
state . light_uniforms [ index ] . shadow_z_far_inv = 1.0 ;
state . light_uniforms [ index ] . shadow_y_ofs = 0 ;
2019-07-05 03:54:32 +02:00
}
2019-07-10 22:44:55 +02:00
2020-11-03 20:51:53 +01:00
state . light_uniforms [ index ] . flags = l - > blend_mode < < LIGHT_FLAGS_BLEND_SHIFT ;
2019-07-07 06:49:40 +02:00
state . light_uniforms [ index ] . flags | = l - > shadow_filter < < LIGHT_FLAGS_FILTER_SHIFT ;
2020-10-24 17:15:43 +02:00
if ( clight - > shadow . enabled ) {
2019-07-07 06:49:40 +02:00
state . light_uniforms [ index ] . flags | = LIGHT_FLAGS_HAS_SHADOW ;
}
2019-07-05 03:54:32 +02:00
2020-10-24 17:15:43 +02:00
if ( clight - > texture . is_valid ( ) ) {
2022-04-07 16:00:51 +02:00
Rect2 atlas_rect = RendererRD : : TextureStorage : : get_singleton ( ) - > decal_atlas_get_texture_rect ( clight - > texture ) ;
2020-10-24 17:15:43 +02:00
state . light_uniforms [ index ] . atlas_rect [ 0 ] = atlas_rect . position . x ;
state . light_uniforms [ index ] . atlas_rect [ 1 ] = atlas_rect . position . y ;
state . light_uniforms [ index ] . atlas_rect [ 2 ] = atlas_rect . size . width ;
state . light_uniforms [ index ] . atlas_rect [ 3 ] = atlas_rect . size . height ;
} else {
state . light_uniforms [ index ] . atlas_rect [ 0 ] = 0 ;
state . light_uniforms [ index ] . atlas_rect [ 1 ] = 0 ;
state . light_uniforms [ index ] . atlas_rect [ 2 ] = 0 ;
state . light_uniforms [ index ] . atlas_rect [ 3 ] = 0 ;
}
2019-07-05 03:54:32 +02:00
l - > render_index_cache = index ;
index + + ;
l = l - > next_ptr ;
}
2020-11-03 20:51:53 +01:00
light_count = index ;
}
if ( light_count > 0 ) {
2021-01-26 01:52:58 +01:00
RD : : get_singleton ( ) - > buffer_update ( state . lights_uniform_buffer , 0 , sizeof ( LightUniform ) * light_count , & state . light_uniforms [ 0 ] ) ;
2020-11-03 20:51:53 +01:00
}
{
//update canvas state uniform buffer
State : : Buffer state_buffer ;
2022-04-07 16:00:51 +02:00
Size2i ssize = texture_storage - > render_target_get_size ( p_to_render_target ) ;
2020-11-03 20:51:53 +01:00
2020-10-17 07:08:21 +02:00
Transform3D screen_transform ;
2022-07-16 11:47:54 +02:00
screen_transform . translate_local ( - ( ssize . width / 2.0f ) , - ( ssize . height / 2.0f ) , 0.0f ) ;
2020-11-03 20:51:53 +01:00
screen_transform . scale ( Vector3 ( 2.0f / ssize . width , 2.0f / ssize . height , 1.0f ) ) ;
_update_transform_to_mat4 ( screen_transform , state_buffer . screen_transform ) ;
_update_transform_2d_to_mat4 ( p_canvas_transform , state_buffer . canvas_transform ) ;
Transform2D normal_transform = p_canvas_transform ;
2022-04-24 23:59:24 +02:00
normal_transform . columns [ 0 ] . normalize ( ) ;
normal_transform . columns [ 1 ] . normalize ( ) ;
normal_transform . columns [ 2 ] = Vector2 ( ) ;
2020-11-03 20:51:53 +01:00
_update_transform_2d_to_mat4 ( normal_transform , state_buffer . canvas_normal_transform ) ;
state_buffer . canvas_modulate [ 0 ] = p_modulate . r ;
state_buffer . canvas_modulate [ 1 ] = p_modulate . g ;
state_buffer . canvas_modulate [ 2 ] = p_modulate . b ;
state_buffer . canvas_modulate [ 3 ] = p_modulate . a ;
2022-04-07 16:00:51 +02:00
Size2 render_target_size = texture_storage - > render_target_get_size ( p_to_render_target ) ;
2020-11-03 20:51:53 +01:00
state_buffer . screen_pixel_size [ 0 ] = 1.0 / render_target_size . x ;
state_buffer . screen_pixel_size [ 1 ] = 1.0 / render_target_size . y ;
state_buffer . time = state . time ;
state_buffer . use_pixel_snap = p_snap_2d_vertices_to_pixel ;
state_buffer . directional_light_count = directional_light_count ;
2020-11-26 13:50:21 +01:00
Vector2 canvas_scale = p_canvas_transform . get_scale ( ) ;
state_buffer . sdf_to_screen [ 0 ] = render_target_size . width / canvas_scale . x ;
state_buffer . sdf_to_screen [ 1 ] = render_target_size . height / canvas_scale . y ;
state_buffer . screen_to_sdf [ 0 ] = 1.0 / state_buffer . sdf_to_screen [ 0 ] ;
state_buffer . screen_to_sdf [ 1 ] = 1.0 / state_buffer . sdf_to_screen [ 1 ] ;
2022-04-07 16:00:51 +02:00
Rect2 sdf_rect = texture_storage - > render_target_get_sdf_rect ( p_to_render_target ) ;
2020-11-26 13:50:21 +01:00
Rect2 sdf_tex_rect ( sdf_rect . position / canvas_scale , sdf_rect . size / canvas_scale ) ;
state_buffer . sdf_to_tex [ 0 ] = 1.0 / sdf_tex_rect . size . width ;
state_buffer . sdf_to_tex [ 1 ] = 1.0 / sdf_tex_rect . size . height ;
state_buffer . sdf_to_tex [ 2 ] = - sdf_tex_rect . position . x / sdf_tex_rect . size . width ;
state_buffer . sdf_to_tex [ 3 ] = - sdf_tex_rect . position . y / sdf_tex_rect . size . height ;
//print_line("w: " + itos(ssize.width) + " s: " + rtos(canvas_scale));
state_buffer . tex_to_sdf = 1.0 / ( ( canvas_scale . x + canvas_scale . y ) * 0.5 ) ;
2021-01-26 01:52:58 +01:00
RD : : get_singleton ( ) - > buffer_update ( state . canvas_state_buffer , 0 , sizeof ( State : : Buffer ) , & state_buffer ) ;
2019-07-05 03:54:32 +02:00
}
2020-10-24 17:15:43 +02:00
{ //default filter/repeat
default_filter = p_default_filter ;
default_repeat = p_default_repeat ;
}
2019-06-16 04:45:24 +02:00
Item * ci = p_item_list ;
2022-09-15 02:36:39 +02:00
//fill the list until rendering is possible.
bool material_screen_texture_cached = false ;
bool material_screen_texture_mipmaps_cached = false ;
2019-07-21 16:31:30 +02:00
Rect2 back_buffer_rect ;
bool backbuffer_copy = false ;
2022-08-09 18:29:49 +02:00
bool backbuffer_gen_mipmaps = false ;
2019-07-21 16:31:30 +02:00
2020-10-28 19:34:27 +01:00
Item * canvas_group_owner = nullptr ;
2023-03-07 07:09:39 +01:00
bool skip_item = false ;
2020-10-28 19:34:27 +01:00
2021-05-10 18:12:44 +02:00
bool update_skeletons = false ;
2021-05-20 16:25:06 +02:00
bool time_used = false ;
2021-05-10 18:12:44 +02:00
2022-04-07 16:31:45 +02:00
bool backbuffer_cleared = false ;
2019-06-16 04:45:24 +02:00
while ( ci ) {
2020-10-28 19:34:27 +01:00
if ( ci - > copy_back_buffer & & canvas_group_owner = = nullptr ) {
2019-07-21 16:31:30 +02:00
backbuffer_copy = true ;
2019-06-16 04:45:24 +02:00
2019-07-21 16:31:30 +02:00
if ( ci - > copy_back_buffer - > full ) {
back_buffer_rect = Rect2 ( ) ;
} else {
back_buffer_rect = ci - > copy_back_buffer - > rect ;
}
}
2021-08-13 04:45:44 +02:00
RID material = ci - > material_owner = = nullptr ? ci - > material : ci - > material_owner - > material ;
if ( material . is_valid ( ) ) {
2022-08-01 03:59:14 +02:00
CanvasMaterialData * md = static_cast < CanvasMaterialData * > ( material_storage - > material_get_data ( material , RendererRD : : MaterialStorage : : SHADER_TYPE_2D ) ) ;
2019-07-29 20:29:43 +02:00
if ( md & & md - > shader_data - > valid ) {
2020-10-28 19:34:27 +01:00
if ( md - > shader_data - > uses_screen_texture & & canvas_group_owner = = nullptr ) {
2022-09-15 02:36:39 +02:00
if ( ! material_screen_texture_cached ) {
2019-07-29 20:29:43 +02:00
backbuffer_copy = true ;
back_buffer_rect = Rect2 ( ) ;
2022-08-09 18:29:49 +02:00
backbuffer_gen_mipmaps = md - > shader_data - > uses_screen_texture_mipmaps ;
2022-09-15 02:36:39 +02:00
} else if ( ! material_screen_texture_mipmaps_cached ) {
backbuffer_gen_mipmaps = md - > shader_data - > uses_screen_texture_mipmaps ;
2019-07-29 20:29:43 +02:00
}
2019-07-27 15:23:24 +02:00
}
2019-07-29 20:29:43 +02:00
2020-11-26 13:50:21 +01:00
if ( md - > shader_data - > uses_sdf ) {
r_sdf_used = true ;
}
2021-05-20 16:25:06 +02:00
if ( md - > shader_data - > uses_time ) {
time_used = true ;
}
2019-07-21 16:31:30 +02:00
}
}
2021-05-10 18:12:44 +02:00
if ( ci - > skeleton . is_valid ( ) ) {
const Item : : Command * c = ci - > commands ;
while ( c ) {
if ( c - > type = = Item : : Command : : TYPE_MESH ) {
const Item : : CommandMesh * cm = static_cast < const Item : : CommandMesh * > ( c ) ;
if ( cm - > mesh_instance . is_valid ( ) ) {
2022-04-02 07:29:04 +02:00
mesh_storage - > mesh_instance_check_for_update ( cm - > mesh_instance ) ;
2023-01-27 23:55:22 +01:00
mesh_storage - > mesh_instance_set_canvas_item_transform ( cm - > mesh_instance , canvas_transform_inverse * ci - > final_transform ) ;
2021-05-10 18:12:44 +02:00
update_skeletons = true ;
}
}
2021-09-29 01:50:01 +02:00
c = c - > next ;
2021-05-10 18:12:44 +02:00
}
}
2020-10-28 19:34:27 +01:00
if ( ci - > canvas_group_owner ! = nullptr ) {
if ( canvas_group_owner = = nullptr ) {
2022-04-07 16:31:45 +02:00
// Canvas group begins here, render until before this item
2021-05-10 18:12:44 +02:00
if ( update_skeletons ) {
2022-04-02 07:29:04 +02:00
mesh_storage - > update_mesh_instances ( ) ;
2021-05-10 18:12:44 +02:00
update_skeletons = false ;
}
2022-04-07 16:31:45 +02:00
2022-12-07 19:03:08 +01:00
_render_items ( p_to_render_target , item_count , canvas_transform_inverse , p_light_list , r_sdf_used ) ;
2020-10-28 19:34:27 +01:00
item_count = 0 ;
2022-10-12 21:45:47 +02:00
if ( ci - > canvas_group_owner - > canvas_group - > mode ! = RS : : CANVAS_GROUP_MODE_TRANSPARENT ) {
2022-10-07 00:27:11 +02:00
Rect2i group_rect = ci - > canvas_group_owner - > global_rect_cache ;
2022-04-07 16:00:51 +02:00
texture_storage - > render_target_copy_to_back_buffer ( p_to_render_target , group_rect , false ) ;
2022-10-12 21:45:47 +02:00
if ( ci - > canvas_group_owner - > canvas_group - > mode = = RS : : CANVAS_GROUP_MODE_CLIP_AND_DRAW ) {
2023-03-07 07:09:39 +01:00
ci - > canvas_group_owner - > use_canvas_group = false ;
2022-10-12 21:45:47 +02:00
items [ item_count + + ] = ci - > canvas_group_owner ;
}
2022-04-07 16:31:45 +02:00
} else if ( ! backbuffer_cleared ) {
texture_storage - > render_target_clear_back_buffer ( p_to_render_target , Rect2i ( ) , Color ( 0 , 0 , 0 , 0 ) ) ;
backbuffer_cleared = true ;
2020-10-28 19:34:27 +01:00
}
backbuffer_copy = false ;
canvas_group_owner = ci - > canvas_group_owner ; //continue until owner found
}
ci - > canvas_group_owner = nullptr ; //must be cleared
}
2023-03-07 07:09:39 +01:00
if ( canvas_group_owner = = nullptr & & ci - > canvas_group ! = nullptr & & ci - > canvas_group - > mode ! = RS : : CANVAS_GROUP_MODE_CLIP_AND_DRAW ) {
skip_item = true ;
2022-04-07 16:31:45 +02:00
}
2020-10-28 19:34:27 +01:00
if ( ci = = canvas_group_owner ) {
2021-05-10 18:12:44 +02:00
if ( update_skeletons ) {
2022-04-02 07:29:04 +02:00
mesh_storage - > update_mesh_instances ( ) ;
2021-05-10 18:12:44 +02:00
update_skeletons = false ;
}
2022-12-07 19:03:08 +01:00
_render_items ( p_to_render_target , item_count , canvas_transform_inverse , p_light_list , r_sdf_used , true ) ;
2020-10-28 19:34:27 +01:00
item_count = 0 ;
if ( ci - > canvas_group - > blur_mipmaps ) {
2022-04-07 16:00:51 +02:00
texture_storage - > render_target_gen_back_buffer_mipmaps ( p_to_render_target , ci - > global_rect_cache ) ;
2020-10-28 19:34:27 +01:00
}
canvas_group_owner = nullptr ;
2022-10-07 23:13:40 +02:00
// Backbuffer is dirty now and needs to be re-cleared if another CanvasGroup needs it.
backbuffer_cleared = false ;
2023-03-07 07:09:39 +01:00
// Tell the renderer to paint this as a canvas group
ci - > use_canvas_group = true ;
} else {
ci - > use_canvas_group = false ;
2020-10-28 19:34:27 +01:00
}
2019-07-21 16:31:30 +02:00
if ( backbuffer_copy ) {
//render anything pending, including clearing if no items
2021-05-10 18:12:44 +02:00
if ( update_skeletons ) {
2022-04-02 07:29:04 +02:00
mesh_storage - > update_mesh_instances ( ) ;
2021-05-10 18:12:44 +02:00
update_skeletons = false ;
}
2022-04-07 16:31:45 +02:00
2022-12-07 19:03:08 +01:00
_render_items ( p_to_render_target , item_count , canvas_transform_inverse , p_light_list , r_sdf_used ) ;
2019-06-16 04:45:24 +02:00
item_count = 0 ;
2022-08-09 18:29:49 +02:00
texture_storage - > render_target_copy_to_back_buffer ( p_to_render_target , back_buffer_rect , backbuffer_gen_mipmaps ) ;
2019-07-27 15:23:24 +02:00
2019-07-21 16:31:30 +02:00
backbuffer_copy = false ;
2022-09-15 02:36:39 +02:00
material_screen_texture_cached = true ; // After a backbuffer copy, screen texture makes no further copies.
material_screen_texture_mipmaps_cached = backbuffer_gen_mipmaps ;
2023-03-07 07:09:39 +01:00
backbuffer_gen_mipmaps = false ;
2022-09-15 02:36:39 +02:00
}
if ( backbuffer_gen_mipmaps ) {
texture_storage - > render_target_gen_back_buffer_mipmaps ( p_to_render_target , back_buffer_rect ) ;
backbuffer_gen_mipmaps = false ;
material_screen_texture_mipmaps_cached = true ;
2019-07-21 16:31:30 +02:00
}
2019-06-16 04:45:24 +02:00
2023-03-07 07:09:39 +01:00
if ( skip_item ) {
skip_item = false ;
} else {
items [ item_count + + ] = ci ;
}
2019-06-16 04:45:24 +02:00
2019-07-21 16:31:30 +02:00
if ( ! ci - > next | | item_count = = MAX_RENDER_ITEMS - 1 ) {
2021-05-10 18:12:44 +02:00
if ( update_skeletons ) {
2022-04-02 07:29:04 +02:00
mesh_storage - > update_mesh_instances ( ) ;
2021-05-10 18:12:44 +02:00
update_skeletons = false ;
}
2023-03-07 07:09:39 +01:00
_render_items ( p_to_render_target , item_count , canvas_transform_inverse , p_light_list , r_sdf_used , canvas_group_owner ! = nullptr ) ;
2019-07-21 16:31:30 +02:00
//then reset
item_count = 0 ;
2019-06-16 04:45:24 +02:00
}
ci = ci - > next ;
}
2021-05-20 16:25:06 +02:00
if ( time_used ) {
RenderingServerDefault : : redraw_request ( ) ;
}
2019-06-16 04:45:24 +02:00
}
2020-12-04 19:26:24 +01:00
RID RendererCanvasRenderRD : : light_create ( ) {
2019-07-05 03:54:32 +02:00
CanvasLight canvas_light ;
return canvas_light_owner . make_rid ( canvas_light ) ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : light_set_texture ( RID p_rid , RID p_texture ) {
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
2022-03-20 12:28:24 +01:00
2021-09-29 19:08:41 +02:00
CanvasLight * cl = canvas_light_owner . get_or_null ( p_rid ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL ( cl ) ;
2019-07-05 03:54:32 +02:00
if ( cl - > texture = = p_texture ) {
return ;
}
2023-01-31 05:59:53 +01:00
ERR_FAIL_COND ( p_texture . is_valid ( ) & & ! texture_storage - > owns_texture ( p_texture ) ) ;
2020-10-24 17:15:43 +02:00
if ( cl - > texture . is_valid ( ) ) {
2022-04-07 16:00:51 +02:00
texture_storage - > texture_remove_from_decal_atlas ( cl - > texture ) ;
2020-10-24 17:15:43 +02:00
}
2019-07-05 03:54:32 +02:00
cl - > texture = p_texture ;
2020-10-24 17:15:43 +02:00
if ( cl - > texture . is_valid ( ) ) {
2022-04-07 16:00:51 +02:00
texture_storage - > texture_add_to_decal_atlas ( cl - > texture ) ;
2020-10-24 17:15:43 +02:00
}
2019-07-05 03:54:32 +02:00
}
2020-05-14 14:29:06 +02:00
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : light_set_use_shadow ( RID p_rid , bool p_enable ) {
2021-09-29 19:08:41 +02:00
CanvasLight * cl = canvas_light_owner . get_or_null ( p_rid ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL ( cl ) ;
2019-07-05 03:54:32 +02:00
2020-10-24 17:15:43 +02:00
cl - > shadow . enabled = p_enable ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : _update_shadow_atlas ( ) {
2020-10-24 17:15:43 +02:00
if ( state . shadow_fb = = RID ( ) ) {
//ah, we lack the shadow texture..
RD : : get_singleton ( ) - > free ( state . shadow_texture ) ; //erase placeholder
2019-07-05 03:54:32 +02:00
2019-07-07 06:49:40 +02:00
Vector < RID > fb_textures ;
2019-07-05 03:54:32 +02:00
{ //texture
RD : : TextureFormat tf ;
2020-11-27 04:50:05 +01:00
tf . texture_type = RD : : TEXTURE_TYPE_2D ;
2020-10-24 17:15:43 +02:00
tf . width = state . shadow_texture_size ;
tf . height = state . max_lights_per_render * 2 ;
2019-07-05 03:54:32 +02:00
tf . usage_bits = RD : : TEXTURE_USAGE_COLOR_ATTACHMENT_BIT | RD : : TEXTURE_USAGE_SAMPLING_BIT ;
tf . format = RD : : DATA_FORMAT_R32_SFLOAT ;
2020-10-24 17:15:43 +02:00
state . shadow_texture = RD : : get_singleton ( ) - > texture_create ( tf , RD : : TextureView ( ) ) ;
fb_textures . push_back ( state . shadow_texture ) ;
2019-07-05 03:54:32 +02:00
}
{
RD : : TextureFormat tf ;
2020-11-27 04:50:05 +01:00
tf . texture_type = RD : : TEXTURE_TYPE_2D ;
2020-10-24 17:15:43 +02:00
tf . width = state . shadow_texture_size ;
tf . height = state . max_lights_per_render * 2 ;
2019-07-05 03:54:32 +02:00
tf . usage_bits = RD : : TEXTURE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT ;
2020-10-24 17:15:43 +02:00
tf . format = RD : : DATA_FORMAT_D32_SFLOAT ;
2019-07-05 03:54:32 +02:00
//chunks to write
2020-10-24 17:15:43 +02:00
state . shadow_depth_texture = RD : : get_singleton ( ) - > texture_create ( tf , RD : : TextureView ( ) ) ;
fb_textures . push_back ( state . shadow_depth_texture ) ;
2019-07-05 03:54:32 +02:00
}
2019-07-07 06:49:40 +02:00
2020-10-24 17:15:43 +02:00
state . shadow_fb = RD : : get_singleton ( ) - > framebuffer_create ( fb_textures ) ;
2019-07-05 03:54:32 +02:00
}
2020-11-03 20:51:53 +01:00
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : light_update_shadow ( RID p_rid , int p_shadow_index , const Transform2D & p_light_xform , int p_light_mask , float p_near , float p_far , LightOccluderInstance * p_occluders ) {
2021-09-29 19:08:41 +02:00
CanvasLight * cl = canvas_light_owner . get_or_null ( p_rid ) ;
2020-11-03 20:51:53 +01:00
ERR_FAIL_COND ( ! cl - > shadow . enabled ) ;
_update_shadow_atlas ( ) ;
2019-07-05 03:54:32 +02:00
2020-10-24 17:15:43 +02:00
cl - > shadow . z_far = p_far ;
cl - > shadow . y_offset = float ( p_shadow_index * 2 + 1 ) / float ( state . max_lights_per_render * 2 ) ;
Vector < Color > cc ;
cc . push_back ( Color ( p_far , p_far , p_far , 1.0 ) ) ;
2019-07-05 03:54:32 +02:00
for ( int i = 0 ; i < 4 ; i + + ) {
//make sure it remains orthogonal, makes easy to read angle later
//light.basis.scale(Vector3(to_light.elements[0].length(),to_light.elements[1].length(),1));
2020-10-24 17:15:43 +02:00
Rect2i rect ( ( state . shadow_texture_size / 4 ) * i , p_shadow_index * 2 , ( state . shadow_texture_size / 4 ) , 2 ) ;
2022-07-05 16:21:03 +02:00
RD : : InitialAction initial_action = i = = 0 ? RD : : INITIAL_ACTION_CLEAR_REGION : RD : : INITIAL_ACTION_CLEAR_REGION_CONTINUE ;
RD : : DrawListID draw_list = RD : : get_singleton ( ) - > draw_list_begin ( state . shadow_fb , initial_action , i ! = 3 ? RD : : FINAL_ACTION_CONTINUE : RD : : FINAL_ACTION_READ , initial_action , RD : : FINAL_ACTION_DISCARD , cc , 1.0 , 0 , rect ) ;
2019-07-05 03:54:32 +02:00
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-20 01:11:13 +02:00
Projection projection ;
2019-07-05 03:54:32 +02:00
{
real_t fov = 90 ;
real_t nearp = p_near ;
real_t farp = p_far ;
real_t aspect = 1.0 ;
2022-08-13 17:45:42 +02:00
real_t ymax = nearp * Math : : tan ( Math : : deg_to_rad ( fov * 0.5 ) ) ;
2019-07-05 03:54:32 +02:00
real_t ymin = - ymax ;
real_t xmin = ymin * aspect ;
real_t xmax = ymax * aspect ;
projection . set_frustum ( xmin , xmax , ymin , ymax , nearp , farp ) ;
}
2022-09-25 01:19:55 +02:00
Vector3 cam_target = Basis : : from_euler ( Vector3 ( 0 , 0 , Math_TAU * ( ( i + 3 ) / 4.0 ) ) ) . xform ( Vector3 ( 0 , 1 , 0 ) ) ;
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-20 01:11:13 +02:00
projection = projection * Projection ( Transform3D ( ) . looking_at ( cam_target , Vector3 ( 0 , 0 , - 1 ) ) . affine_inverse ( ) ) ;
2019-07-05 03:54:32 +02:00
ShadowRenderPushConstant push_constant ;
for ( int y = 0 ; y < 4 ; y + + ) {
for ( int x = 0 ; x < 4 ; x + + ) {
2022-10-04 18:44:48 +02:00
push_constant . projection [ y * 4 + x ] = projection . columns [ y ] [ x ] ;
2019-07-05 03:54:32 +02:00
}
}
2019-07-07 06:49:40 +02:00
static const Vector2 directions [ 4 ] = { Vector2 ( 1 , 0 ) , Vector2 ( 0 , 1 ) , Vector2 ( - 1 , 0 ) , Vector2 ( 0 , - 1 ) } ;
push_constant . direction [ 0 ] = directions [ i ] . x ;
push_constant . direction [ 1 ] = directions [ i ] . y ;
2020-10-24 17:15:43 +02:00
push_constant . z_far = p_far ;
push_constant . pad = 0 ;
2019-07-05 03:54:32 +02:00
LightOccluderInstance * instance = p_occluders ;
while ( instance ) {
2021-09-29 19:08:41 +02:00
OccluderPolygon * co = occluder_polygon_owner . get_or_null ( instance - > occluder ) ;
2019-07-05 03:54:32 +02:00
if ( ! co | | co - > index_array . is_null ( ) | | ! ( p_light_mask & instance - > light_mask ) ) {
instance = instance - > next ;
continue ;
}
2019-07-07 06:49:40 +02:00
_update_transform_2d_to_mat2x4 ( p_light_xform * instance - > xform_cache , push_constant . modelview ) ;
2019-07-05 03:54:32 +02:00
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( draw_list , shadow_render . render_pipelines [ co - > cull_mode ] ) ;
RD : : get_singleton ( ) - > draw_list_bind_vertex_array ( draw_list , co - > vertex_array ) ;
RD : : get_singleton ( ) - > draw_list_bind_index_array ( draw_list , co - > index_array ) ;
RD : : get_singleton ( ) - > draw_list_set_push_constant ( draw_list , & push_constant , sizeof ( ShadowRenderPushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_draw ( draw_list , true ) ;
instance = instance - > next ;
}
RD : : get_singleton ( ) - > draw_list_end ( ) ;
}
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : light_update_directional_shadow ( RID p_rid , int p_shadow_index , const Transform2D & p_light_xform , int p_light_mask , float p_cull_distance , const Rect2 & p_clip_rect , LightOccluderInstance * p_occluders ) {
2021-09-29 19:08:41 +02:00
CanvasLight * cl = canvas_light_owner . get_or_null ( p_rid ) ;
2020-11-03 20:51:53 +01:00
ERR_FAIL_COND ( ! cl - > shadow . enabled ) ;
_update_shadow_atlas ( ) ;
2022-04-24 23:59:24 +02:00
Vector2 light_dir = p_light_xform . columns [ 1 ] . normalized ( ) ;
2020-11-03 20:51:53 +01:00
2021-09-20 20:48:52 +02:00
Vector2 center = p_clip_rect . get_center ( ) ;
2020-11-03 20:51:53 +01:00
float to_edge_distance = ABS ( light_dir . dot ( p_clip_rect . get_support ( light_dir ) ) - light_dir . dot ( center ) ) ;
Vector2 from_pos = center - light_dir * ( to_edge_distance + p_cull_distance ) ;
float distance = to_edge_distance * 2.0 + p_cull_distance ;
float half_size = p_clip_rect . size . length ( ) * 0.5 ; //shadow length, must keep this no matter the angle
cl - > shadow . z_far = distance ;
cl - > shadow . y_offset = float ( p_shadow_index * 2 + 1 ) / float ( state . max_lights_per_render * 2 ) ;
Transform2D to_light_xform ;
to_light_xform [ 2 ] = from_pos ;
to_light_xform [ 1 ] = light_dir ;
2020-12-06 19:16:06 +01:00
to_light_xform [ 0 ] = - light_dir . orthogonal ( ) ;
2020-11-03 20:51:53 +01:00
to_light_xform . invert ( ) ;
Vector < Color > cc ;
cc . push_back ( Color ( 1 , 1 , 1 , 1 ) ) ;
Rect2i rect ( 0 , p_shadow_index * 2 , state . shadow_texture_size , 2 ) ;
2022-07-05 16:21:03 +02:00
RD : : DrawListID draw_list = RD : : get_singleton ( ) - > draw_list_begin ( state . shadow_fb , RD : : INITIAL_ACTION_CLEAR_REGION , RD : : FINAL_ACTION_READ , RD : : INITIAL_ACTION_CLEAR_REGION , RD : : FINAL_ACTION_DISCARD , cc , 1.0 , 0 , rect ) ;
2020-11-03 20:51:53 +01:00
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-20 01:11:13 +02:00
Projection projection ;
2020-11-03 20:51:53 +01:00
projection . set_orthogonal ( - half_size , half_size , - 0.5 , 0.5 , 0.0 , distance ) ;
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-20 01:11:13 +02:00
projection = projection * Projection ( Transform3D ( ) . looking_at ( Vector3 ( 0 , 1 , 0 ) , Vector3 ( 0 , 0 , - 1 ) ) . affine_inverse ( ) ) ;
2020-11-03 20:51:53 +01:00
ShadowRenderPushConstant push_constant ;
for ( int y = 0 ; y < 4 ; y + + ) {
for ( int x = 0 ; x < 4 ; x + + ) {
2022-10-04 18:44:48 +02:00
push_constant . projection [ y * 4 + x ] = projection . columns [ y ] [ x ] ;
2020-11-03 20:51:53 +01:00
}
}
push_constant . direction [ 0 ] = 0.0 ;
push_constant . direction [ 1 ] = 1.0 ;
push_constant . z_far = distance ;
push_constant . pad = 0 ;
LightOccluderInstance * instance = p_occluders ;
while ( instance ) {
2021-09-29 19:08:41 +02:00
OccluderPolygon * co = occluder_polygon_owner . get_or_null ( instance - > occluder ) ;
2020-11-03 20:51:53 +01:00
if ( ! co | | co - > index_array . is_null ( ) | | ! ( p_light_mask & instance - > light_mask ) ) {
instance = instance - > next ;
continue ;
}
_update_transform_2d_to_mat2x4 ( to_light_xform * instance - > xform_cache , push_constant . modelview ) ;
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( draw_list , shadow_render . render_pipelines [ co - > cull_mode ] ) ;
RD : : get_singleton ( ) - > draw_list_bind_vertex_array ( draw_list , co - > vertex_array ) ;
RD : : get_singleton ( ) - > draw_list_bind_index_array ( draw_list , co - > index_array ) ;
RD : : get_singleton ( ) - > draw_list_set_push_constant ( draw_list , & push_constant , sizeof ( ShadowRenderPushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_draw ( draw_list , true ) ;
instance = instance - > next ;
}
RD : : get_singleton ( ) - > draw_list_end ( ) ;
Transform2D to_shadow ;
2022-04-24 23:59:24 +02:00
to_shadow . columns [ 0 ] . x = 1.0 / - ( half_size * 2.0 ) ;
to_shadow . columns [ 2 ] . x = 0.5 ;
2020-11-03 20:51:53 +01:00
cl - > shadow . directional_xform = to_shadow * to_light_xform ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : render_sdf ( RID p_render_target , LightOccluderInstance * p_occluders ) {
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
RID fb = texture_storage - > render_target_get_sdf_framebuffer ( p_render_target ) ;
Rect2i rect = texture_storage - > render_target_get_sdf_rect ( p_render_target ) ;
2020-11-26 13:50:21 +01:00
Transform2D to_sdf ;
2022-04-24 23:59:24 +02:00
to_sdf . columns [ 0 ] * = rect . size . width ;
to_sdf . columns [ 1 ] * = rect . size . height ;
to_sdf . columns [ 2 ] = rect . position ;
2020-11-26 13:50:21 +01:00
Transform2D to_clip ;
2022-04-24 23:59:24 +02:00
to_clip . columns [ 0 ] * = 2.0 ;
to_clip . columns [ 1 ] * = 2.0 ;
to_clip . columns [ 2 ] = - Vector2 ( 1.0 , 1.0 ) ;
2020-11-26 13:50:21 +01:00
to_clip = to_clip * to_sdf . affine_inverse ( ) ;
Vector < Color > cc ;
cc . push_back ( Color ( 0 , 0 , 0 , 0 ) ) ;
RD : : DrawListID draw_list = RD : : get_singleton ( ) - > draw_list_begin ( fb , RD : : INITIAL_ACTION_CLEAR , RD : : FINAL_ACTION_READ , RD : : INITIAL_ACTION_CLEAR , RD : : FINAL_ACTION_DISCARD , cc ) ;
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-20 01:11:13 +02:00
Projection projection ;
2020-11-26 13:50:21 +01:00
ShadowRenderPushConstant push_constant ;
for ( int y = 0 ; y < 4 ; y + + ) {
for ( int x = 0 ; x < 4 ; x + + ) {
2022-10-04 18:44:48 +02:00
push_constant . projection [ y * 4 + x ] = projection . columns [ y ] [ x ] ;
2020-11-26 13:50:21 +01:00
}
}
push_constant . direction [ 0 ] = 0.0 ;
push_constant . direction [ 1 ] = 0.0 ;
push_constant . z_far = 0 ;
push_constant . pad = 0 ;
LightOccluderInstance * instance = p_occluders ;
while ( instance ) {
2021-09-29 19:08:41 +02:00
OccluderPolygon * co = occluder_polygon_owner . get_or_null ( instance - > occluder ) ;
2020-11-26 13:50:21 +01:00
if ( ! co | | co - > sdf_index_array . is_null ( ) ) {
instance = instance - > next ;
continue ;
}
_update_transform_2d_to_mat2x4 ( to_clip * instance - > xform_cache , push_constant . modelview ) ;
RD : : get_singleton ( ) - > draw_list_bind_render_pipeline ( draw_list , shadow_render . sdf_render_pipelines [ co - > sdf_is_lines ? SHADOW_RENDER_SDF_LINES : SHADOW_RENDER_SDF_TRIANGLES ] ) ;
RD : : get_singleton ( ) - > draw_list_bind_vertex_array ( draw_list , co - > sdf_vertex_array ) ;
RD : : get_singleton ( ) - > draw_list_bind_index_array ( draw_list , co - > sdf_index_array ) ;
RD : : get_singleton ( ) - > draw_list_set_push_constant ( draw_list , & push_constant , sizeof ( ShadowRenderPushConstant ) ) ;
RD : : get_singleton ( ) - > draw_list_draw ( draw_list , true ) ;
instance = instance - > next ;
}
RD : : get_singleton ( ) - > draw_list_end ( ) ;
2022-04-07 16:00:51 +02:00
texture_storage - > render_target_sdf_process ( p_render_target ) ; //done rendering, process it
2020-11-26 13:50:21 +01:00
}
2020-12-04 19:26:24 +01:00
RID RendererCanvasRenderRD : : occluder_polygon_create ( ) {
2019-07-05 03:54:32 +02:00
OccluderPolygon occluder ;
2020-11-26 13:50:21 +01:00
occluder . line_point_count = 0 ;
occluder . sdf_point_count = 0 ;
occluder . sdf_index_count = 0 ;
2020-03-27 19:21:27 +01:00
occluder . cull_mode = RS : : CANVAS_OCCLUDER_POLYGON_CULL_DISABLED ;
2019-07-05 03:54:32 +02:00
return occluder_polygon_owner . make_rid ( occluder ) ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : occluder_polygon_set_shape ( RID p_occluder , const Vector < Vector2 > & p_points , bool p_closed ) {
2021-09-29 19:08:41 +02:00
OccluderPolygon * oc = occluder_polygon_owner . get_or_null ( p_occluder ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL ( oc ) ;
2019-07-05 03:54:32 +02:00
2020-11-26 13:50:21 +01:00
Vector < Vector2 > lines ;
2020-12-02 02:40:47 +01:00
if ( p_points . size ( ) ) {
int lc = p_points . size ( ) * 2 ;
2020-11-26 13:50:21 +01:00
2020-12-02 02:40:47 +01:00
lines . resize ( lc - ( p_closed ? 0 : 2 ) ) ;
{
Vector2 * w = lines . ptrw ( ) ;
const Vector2 * r = p_points . ptr ( ) ;
int max = lc / 2 ;
if ( ! p_closed ) {
max - - ;
}
for ( int i = 0 ; i < max ; i + + ) {
Vector2 a = r [ i ] ;
Vector2 b = r [ ( i + 1 ) % ( lc / 2 ) ] ;
w [ i * 2 + 0 ] = a ;
w [ i * 2 + 1 ] = b ;
}
2020-11-26 13:50:21 +01:00
}
}
2023-03-14 03:42:54 +01:00
if ( ( oc - > line_point_count ! = lines . size ( ) | | lines . size ( ) = = 0 ) & & oc - > vertex_array . is_valid ( ) ) {
2019-07-05 03:54:32 +02:00
RD : : get_singleton ( ) - > free ( oc - > vertex_array ) ;
RD : : get_singleton ( ) - > free ( oc - > vertex_buffer ) ;
RD : : get_singleton ( ) - > free ( oc - > index_array ) ;
RD : : get_singleton ( ) - > free ( oc - > index_buffer ) ;
oc - > vertex_array = RID ( ) ;
oc - > vertex_buffer = RID ( ) ;
oc - > index_array = RID ( ) ;
oc - > index_buffer = RID ( ) ;
2020-11-26 13:50:21 +01:00
oc - > line_point_count = lines . size ( ) ;
2019-07-05 03:54:32 +02:00
}
2020-11-26 13:50:21 +01:00
if ( lines . size ( ) ) {
2023-03-14 03:42:54 +01:00
oc - > line_point_count = lines . size ( ) ;
2020-02-17 22:06:54 +01:00
Vector < uint8_t > geometry ;
Vector < uint8_t > indices ;
2020-12-02 02:40:47 +01:00
int lc = lines . size ( ) ;
2019-07-05 03:54:32 +02:00
geometry . resize ( lc * 6 * sizeof ( float ) ) ;
indices . resize ( lc * 3 * sizeof ( uint16_t ) ) ;
{
2020-02-17 22:06:54 +01:00
uint8_t * vw = geometry . ptrw ( ) ;
2022-04-07 12:23:40 +02:00
float * vwptr = reinterpret_cast < float * > ( vw ) ;
2020-02-17 22:06:54 +01:00
uint8_t * iw = indices . ptrw ( ) ;
uint16_t * iwptr = ( uint16_t * ) iw ;
2019-07-05 03:54:32 +02:00
2020-11-26 13:50:21 +01:00
const Vector2 * lr = lines . ptr ( ) ;
2019-07-05 03:54:32 +02:00
const int POLY_HEIGHT = 16384 ;
for ( int i = 0 ; i < lc / 2 ; i + + ) {
vwptr [ i * 12 + 0 ] = lr [ i * 2 + 0 ] . x ;
vwptr [ i * 12 + 1 ] = lr [ i * 2 + 0 ] . y ;
vwptr [ i * 12 + 2 ] = POLY_HEIGHT ;
vwptr [ i * 12 + 3 ] = lr [ i * 2 + 1 ] . x ;
vwptr [ i * 12 + 4 ] = lr [ i * 2 + 1 ] . y ;
vwptr [ i * 12 + 5 ] = POLY_HEIGHT ;
vwptr [ i * 12 + 6 ] = lr [ i * 2 + 1 ] . x ;
vwptr [ i * 12 + 7 ] = lr [ i * 2 + 1 ] . y ;
vwptr [ i * 12 + 8 ] = - POLY_HEIGHT ;
vwptr [ i * 12 + 9 ] = lr [ i * 2 + 0 ] . x ;
vwptr [ i * 12 + 10 ] = lr [ i * 2 + 0 ] . y ;
vwptr [ i * 12 + 11 ] = - POLY_HEIGHT ;
iwptr [ i * 6 + 0 ] = i * 4 + 0 ;
iwptr [ i * 6 + 1 ] = i * 4 + 1 ;
iwptr [ i * 6 + 2 ] = i * 4 + 2 ;
iwptr [ i * 6 + 3 ] = i * 4 + 2 ;
iwptr [ i * 6 + 4 ] = i * 4 + 3 ;
iwptr [ i * 6 + 5 ] = i * 4 + 0 ;
}
}
2022-10-19 02:59:31 +02:00
//if same buffer len is being set, just use buffer_update to avoid a pipeline flush
2019-07-05 03:54:32 +02:00
if ( oc - > vertex_array . is_null ( ) ) {
//create from scratch
//vertices
2022-10-19 02:59:31 +02:00
// TODO: geometry is always of length lc * 6 * sizeof(float), so in doubles builds this will receive half the data it needs
2019-07-05 03:54:32 +02:00
oc - > vertex_buffer = RD : : get_singleton ( ) - > vertex_buffer_create ( lc * 6 * sizeof ( real_t ) , geometry ) ;
Vector < RID > buffer ;
buffer . push_back ( oc - > vertex_buffer ) ;
oc - > vertex_array = RD : : get_singleton ( ) - > vertex_array_create ( 4 * lc / 2 , shadow_render . vertex_format , buffer ) ;
//indices
oc - > index_buffer = RD : : get_singleton ( ) - > index_buffer_create ( 3 * lc , RD : : INDEX_BUFFER_FORMAT_UINT16 , indices ) ;
oc - > index_array = RD : : get_singleton ( ) - > index_array_create ( oc - > index_buffer , 0 , 3 * lc ) ;
} else {
//update existing
2020-02-17 22:06:54 +01:00
const uint8_t * vr = geometry . ptr ( ) ;
RD : : get_singleton ( ) - > buffer_update ( oc - > vertex_buffer , 0 , geometry . size ( ) , vr ) ;
const uint8_t * ir = indices . ptr ( ) ;
RD : : get_singleton ( ) - > buffer_update ( oc - > index_buffer , 0 , indices . size ( ) , ir ) ;
2019-07-05 03:54:32 +02:00
}
}
2020-11-26 13:50:21 +01:00
// sdf
Vector < int > sdf_indices ;
2020-12-02 02:40:47 +01:00
if ( p_points . size ( ) ) {
if ( p_closed ) {
sdf_indices = Geometry2D : : triangulate_polygon ( p_points ) ;
oc - > sdf_is_lines = false ;
} else {
int max = p_points . size ( ) ;
sdf_indices . resize ( max * 2 ) ;
2020-11-26 13:50:21 +01:00
2020-12-02 02:40:47 +01:00
int * iw = sdf_indices . ptrw ( ) ;
for ( int i = 0 ; i < max ; i + + ) {
iw [ i * 2 + 0 ] = i ;
iw [ i * 2 + 1 ] = ( i + 1 ) % max ;
}
oc - > sdf_is_lines = true ;
2020-11-26 13:50:21 +01:00
}
}
2023-03-14 03:42:54 +01:00
if ( ( ( oc - > sdf_index_count ! = sdf_indices . size ( ) & & oc - > sdf_point_count ! = p_points . size ( ) ) | | p_points . size ( ) = = 0 ) & & oc - > sdf_vertex_array . is_valid ( ) ) {
2020-11-26 13:50:21 +01:00
RD : : get_singleton ( ) - > free ( oc - > sdf_vertex_array ) ;
RD : : get_singleton ( ) - > free ( oc - > sdf_vertex_buffer ) ;
RD : : get_singleton ( ) - > free ( oc - > sdf_index_array ) ;
RD : : get_singleton ( ) - > free ( oc - > sdf_index_buffer ) ;
oc - > sdf_vertex_array = RID ( ) ;
oc - > sdf_vertex_buffer = RID ( ) ;
oc - > sdf_index_array = RID ( ) ;
oc - > sdf_index_buffer = RID ( ) ;
oc - > sdf_index_count = sdf_indices . size ( ) ;
oc - > sdf_point_count = p_points . size ( ) ;
oc - > sdf_is_lines = false ;
}
if ( sdf_indices . size ( ) ) {
if ( oc - > sdf_vertex_array . is_null ( ) ) {
//create from scratch
//vertices
oc - > sdf_vertex_buffer = RD : : get_singleton ( ) - > vertex_buffer_create ( p_points . size ( ) * 2 * sizeof ( real_t ) , p_points . to_byte_array ( ) ) ;
oc - > sdf_index_buffer = RD : : get_singleton ( ) - > index_buffer_create ( sdf_indices . size ( ) , RD : : INDEX_BUFFER_FORMAT_UINT32 , sdf_indices . to_byte_array ( ) ) ;
oc - > sdf_index_array = RD : : get_singleton ( ) - > index_array_create ( oc - > sdf_index_buffer , 0 , sdf_indices . size ( ) ) ;
Vector < RID > buffer ;
buffer . push_back ( oc - > sdf_vertex_buffer ) ;
oc - > sdf_vertex_array = RD : : get_singleton ( ) - > vertex_array_create ( p_points . size ( ) , shadow_render . sdf_vertex_format , buffer ) ;
//indices
} else {
//update existing
2022-08-29 03:51:53 +02:00
RD : : get_singleton ( ) - > buffer_update ( oc - > sdf_vertex_buffer , 0 , sizeof ( real_t ) * 2 * p_points . size ( ) , p_points . ptr ( ) ) ;
RD : : get_singleton ( ) - > buffer_update ( oc - > sdf_index_buffer , 0 , sdf_indices . size ( ) * sizeof ( int32_t ) , sdf_indices . ptr ( ) ) ;
2020-11-26 13:50:21 +01:00
}
}
2019-07-05 03:54:32 +02:00
}
2020-05-14 14:29:06 +02:00
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : occluder_polygon_set_cull_mode ( RID p_occluder , RS : : CanvasOccluderPolygonCullMode p_mode ) {
2021-09-29 19:08:41 +02:00
OccluderPolygon * oc = occluder_polygon_owner . get_or_null ( p_occluder ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL ( oc ) ;
2019-07-05 03:54:32 +02:00
oc - > cull_mode = p_mode ;
}
2022-03-21 12:25:25 +01:00
void RendererCanvasRenderRD : : CanvasShaderData : : set_code ( const String & p_code ) {
2019-07-21 16:31:30 +02:00
//compile
2019-06-16 04:45:24 +02:00
2019-07-21 16:31:30 +02:00
code = p_code ;
valid = false ;
ubo_size = 0 ;
uniforms . clear ( ) ;
uses_screen_texture = false ;
2022-08-09 18:29:49 +02:00
uses_screen_texture_mipmaps = false ;
2020-11-26 13:50:21 +01:00
uses_sdf = false ;
2021-05-20 16:25:06 +02:00
uses_time = false ;
2019-06-16 04:45:24 +02:00
2021-12-09 10:42:46 +01:00
if ( code . is_empty ( ) ) {
2019-07-21 16:31:30 +02:00
return ; //just invalid, but no error
}
2019-06-16 04:45:24 +02:00
2021-11-16 16:25:42 +01:00
ShaderCompiler : : GeneratedCode gen_code ;
2019-06-16 04:45:24 +02:00
2019-07-21 16:31:30 +02:00
int blend_mode = BLEND_MODE_MIX ;
2021-11-16 16:25:42 +01:00
ShaderCompiler : : IdentifierActions actions ;
actions . entry_point_stages [ " vertex " ] = ShaderCompiler : : STAGE_VERTEX ;
actions . entry_point_stages [ " fragment " ] = ShaderCompiler : : STAGE_FRAGMENT ;
actions . entry_point_stages [ " light " ] = ShaderCompiler : : STAGE_FRAGMENT ;
2019-07-21 16:31:30 +02:00
actions . render_mode_values [ " blend_add " ] = Pair < int * , int > ( & blend_mode , BLEND_MODE_ADD ) ;
actions . render_mode_values [ " blend_mix " ] = Pair < int * , int > ( & blend_mode , BLEND_MODE_MIX ) ;
actions . render_mode_values [ " blend_sub " ] = Pair < int * , int > ( & blend_mode , BLEND_MODE_SUB ) ;
actions . render_mode_values [ " blend_mul " ] = Pair < int * , int > ( & blend_mode , BLEND_MODE_MUL ) ;
actions . render_mode_values [ " blend_premul_alpha " ] = Pair < int * , int > ( & blend_mode , BLEND_MODE_PMALPHA ) ;
actions . render_mode_values [ " blend_disabled " ] = Pair < int * , int > ( & blend_mode , BLEND_MODE_DISABLED ) ;
2020-11-26 13:50:21 +01:00
actions . usage_flag_pointers [ " texture_sdf " ] = & uses_sdf ;
2021-05-20 16:25:06 +02:00
actions . usage_flag_pointers [ " TIME " ] = & uses_time ;
2019-07-21 16:31:30 +02:00
actions . uniforms = & uniforms ;
2022-04-05 12:40:26 +02:00
RendererCanvasRenderRD * canvas_singleton = static_cast < RendererCanvasRenderRD * > ( RendererCanvasRender : : singleton ) ;
2019-07-21 16:31:30 +02:00
2020-03-27 19:21:27 +01:00
Error err = canvas_singleton - > shader . compiler . compile ( RS : : SHADER_CANVAS_ITEM , code , & actions , path , gen_code ) ;
2021-08-16 10:25:20 +02:00
ERR_FAIL_COND_MSG ( err ! = OK , " Shader compilation failed. " ) ;
2019-07-21 16:31:30 +02:00
2022-08-09 18:29:49 +02:00
uses_screen_texture_mipmaps = gen_code . uses_screen_texture_mipmaps ;
2023-01-05 20:55:59 +01:00
uses_screen_texture = gen_code . uses_screen_texture ;
2022-08-09 18:29:49 +02:00
2019-07-21 16:31:30 +02:00
if ( version . is_null ( ) ) {
version = canvas_singleton - > shader . canvas_shader . version_create ( ) ;
}
#if 0
print_line ( " **compiling shader: " ) ;
print_line ( " **defines: \n " ) ;
for ( int i = 0 ; i < gen_code . defines . size ( ) ; i + + ) {
print_line ( gen_code . defines [ i ] ) ;
}
2022-08-09 18:29:49 +02:00
HashMap < String , String > : : Iterator el = gen_code . code . begin ( ) ;
while ( el ) {
print_line ( " \n **code " + el - > key + " : \n " + el - > value ) ;
+ + el ;
}
2019-07-21 16:31:30 +02:00
print_line ( " \n **uniforms: \n " + gen_code . uniforms ) ;
2022-08-09 18:29:49 +02:00
print_line ( " \n **vertex_globals: \n " + gen_code . stage_globals [ ShaderCompiler : : STAGE_VERTEX ] ) ;
print_line ( " \n **fragment_globals: \n " + gen_code . stage_globals [ ShaderCompiler : : STAGE_FRAGMENT ] ) ;
2019-07-21 16:31:30 +02:00
# endif
2021-11-16 16:25:42 +01:00
canvas_singleton - > shader . canvas_shader . version_set_code ( version , gen_code . code , gen_code . uniforms , gen_code . stage_globals [ ShaderCompiler : : STAGE_VERTEX ] , gen_code . stage_globals [ ShaderCompiler : : STAGE_FRAGMENT ] , gen_code . defines ) ;
2019-07-27 15:23:24 +02:00
ERR_FAIL_COND ( ! canvas_singleton - > shader . canvas_shader . version_is_valid ( version ) ) ;
2019-07-21 16:31:30 +02:00
ubo_size = gen_code . uniform_total_size ;
ubo_offsets = gen_code . uniform_offsets ;
texture_uniforms = gen_code . texture_uniforms ;
//update them pipelines
RD : : PipelineColorBlendState : : Attachment attachment ;
switch ( blend_mode ) {
case BLEND_MODE_DISABLED : {
// nothing to do here, disabled by default
} break ;
case BLEND_MODE_MIX : {
attachment . enable_blend = true ;
attachment . color_blend_op = RD : : BLEND_OP_ADD ;
attachment . src_color_blend_factor = RD : : BLEND_FACTOR_SRC_ALPHA ;
attachment . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
2020-10-28 19:34:27 +01:00
attachment . alpha_blend_op = RD : : BLEND_OP_ADD ;
2019-07-21 16:31:30 +02:00
attachment . src_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
} break ;
case BLEND_MODE_ADD : {
attachment . enable_blend = true ;
attachment . alpha_blend_op = RD : : BLEND_OP_ADD ;
attachment . color_blend_op = RD : : BLEND_OP_ADD ;
attachment . src_color_blend_factor = RD : : BLEND_FACTOR_SRC_ALPHA ;
attachment . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment . src_alpha_blend_factor = RD : : BLEND_FACTOR_SRC_ALPHA ;
attachment . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
} break ;
case BLEND_MODE_SUB : {
attachment . enable_blend = true ;
2023-05-26 16:50:35 +02:00
attachment . alpha_blend_op = RD : : BLEND_OP_REVERSE_SUBTRACT ;
attachment . color_blend_op = RD : : BLEND_OP_REVERSE_SUBTRACT ;
2019-07-21 16:31:30 +02:00
attachment . src_color_blend_factor = RD : : BLEND_FACTOR_SRC_ALPHA ;
attachment . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment . src_alpha_blend_factor = RD : : BLEND_FACTOR_SRC_ALPHA ;
attachment . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
} break ;
case BLEND_MODE_MUL : {
attachment . enable_blend = true ;
attachment . alpha_blend_op = RD : : BLEND_OP_ADD ;
attachment . color_blend_op = RD : : BLEND_OP_ADD ;
attachment . src_color_blend_factor = RD : : BLEND_FACTOR_DST_COLOR ;
attachment . dst_color_blend_factor = RD : : BLEND_FACTOR_ZERO ;
attachment . src_alpha_blend_factor = RD : : BLEND_FACTOR_DST_ALPHA ;
attachment . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ZERO ;
} break ;
case BLEND_MODE_PMALPHA : {
attachment . enable_blend = true ;
attachment . alpha_blend_op = RD : : BLEND_OP_ADD ;
attachment . color_blend_op = RD : : BLEND_OP_ADD ;
attachment . src_color_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
attachment . src_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
} break ;
}
RD : : PipelineColorBlendState blend_state ;
blend_state . attachments . push_back ( attachment ) ;
2022-08-12 13:03:28 +02:00
RD : : PipelineColorBlendState : : Attachment attachment_lcd ;
attachment_lcd . enable_blend = true ;
attachment_lcd . alpha_blend_op = RD : : BLEND_OP_ADD ;
attachment_lcd . color_blend_op = RD : : BLEND_OP_ADD ;
attachment_lcd . src_color_blend_factor = RD : : BLEND_FACTOR_CONSTANT_COLOR ;
attachment_lcd . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_COLOR ;
attachment_lcd . src_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment_lcd . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
RD : : PipelineColorBlendState blend_state_lcd ;
blend_state_lcd . attachments . push_back ( attachment_lcd ) ;
2019-07-21 16:31:30 +02:00
//update pipelines
for ( int i = 0 ; i < PIPELINE_LIGHT_MODE_MAX ; i + + ) {
for ( int j = 0 ; j < PIPELINE_VARIANT_MAX ; j + + ) {
RD : : RenderPrimitive primitive [ PIPELINE_VARIANT_MAX ] = {
RD : : RENDER_PRIMITIVE_TRIANGLES ,
RD : : RENDER_PRIMITIVE_TRIANGLES ,
RD : : RENDER_PRIMITIVE_TRIANGLES ,
RD : : RENDER_PRIMITIVE_LINES ,
RD : : RENDER_PRIMITIVE_POINTS ,
RD : : RENDER_PRIMITIVE_TRIANGLES ,
2019-08-26 22:43:58 +02:00
RD : : RENDER_PRIMITIVE_TRIANGLE_STRIPS ,
2019-07-21 16:31:30 +02:00
RD : : RENDER_PRIMITIVE_LINES ,
2019-08-26 22:43:58 +02:00
RD : : RENDER_PRIMITIVE_LINESTRIPS ,
2019-07-21 16:31:30 +02:00
RD : : RENDER_PRIMITIVE_POINTS ,
2022-08-12 13:03:28 +02:00
RD : : RENDER_PRIMITIVE_TRIANGLES ,
2019-07-21 16:31:30 +02:00
} ;
2019-08-26 22:43:58 +02:00
2019-07-21 16:31:30 +02:00
ShaderVariant shader_variants [ PIPELINE_LIGHT_MODE_MAX ] [ PIPELINE_VARIANT_MAX ] = {
2022-08-12 13:03:28 +02:00
{
//non lit
2019-07-21 16:31:30 +02:00
SHADER_VARIANT_QUAD ,
SHADER_VARIANT_NINEPATCH ,
SHADER_VARIANT_PRIMITIVE ,
SHADER_VARIANT_PRIMITIVE ,
SHADER_VARIANT_PRIMITIVE_POINTS ,
SHADER_VARIANT_ATTRIBUTES ,
SHADER_VARIANT_ATTRIBUTES ,
2019-08-26 22:43:58 +02:00
SHADER_VARIANT_ATTRIBUTES ,
SHADER_VARIANT_ATTRIBUTES ,
2022-08-12 13:03:28 +02:00
SHADER_VARIANT_ATTRIBUTES_POINTS ,
SHADER_VARIANT_QUAD ,
} ,
{
//lit
2019-07-21 16:31:30 +02:00
SHADER_VARIANT_QUAD_LIGHT ,
SHADER_VARIANT_NINEPATCH_LIGHT ,
SHADER_VARIANT_PRIMITIVE_LIGHT ,
SHADER_VARIANT_PRIMITIVE_LIGHT ,
SHADER_VARIANT_PRIMITIVE_POINTS_LIGHT ,
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
2019-08-26 22:43:58 +02:00
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
2022-08-12 13:03:28 +02:00
SHADER_VARIANT_ATTRIBUTES_POINTS_LIGHT ,
SHADER_VARIANT_QUAD_LIGHT ,
} ,
2019-07-21 16:31:30 +02:00
} ;
RID shader_variant = canvas_singleton - > shader . canvas_shader . version_get_shader ( version , shader_variants [ i ] [ j ] ) ;
2022-08-12 13:03:28 +02:00
if ( j = = PIPELINE_VARIANT_QUAD_LCD_BLEND ) {
pipeline_variants . variants [ i ] [ j ] . setup ( shader_variant , primitive [ j ] , RD : : PipelineRasterizationState ( ) , RD : : PipelineMultisampleState ( ) , RD : : PipelineDepthStencilState ( ) , blend_state_lcd , RD : : DYNAMIC_STATE_BLEND_CONSTANTS ) ;
} else {
pipeline_variants . variants [ i ] [ j ] . setup ( shader_variant , primitive [ j ] , RD : : PipelineRasterizationState ( ) , RD : : PipelineMultisampleState ( ) , RD : : PipelineDepthStencilState ( ) , blend_state , 0 ) ;
}
2019-06-16 04:45:24 +02:00
}
2019-07-21 16:31:30 +02:00
}
2019-06-16 04:45:24 +02:00
2019-07-21 16:31:30 +02:00
valid = true ;
}
2022-03-21 12:25:25 +01:00
bool RendererCanvasRenderRD : : CanvasShaderData : : is_animated ( ) const {
2019-07-21 16:31:30 +02:00
return false ;
}
2020-05-14 14:29:06 +02:00
2022-03-21 12:25:25 +01:00
bool RendererCanvasRenderRD : : CanvasShaderData : : casts_shadows ( ) const {
2019-07-21 16:31:30 +02:00
return false ;
}
2020-05-14 14:29:06 +02:00
2022-03-21 12:25:25 +01:00
RS : : ShaderNativeSourceCode RendererCanvasRenderRD : : CanvasShaderData : : get_native_source_code ( ) const {
2022-04-05 12:40:26 +02:00
RendererCanvasRenderRD * canvas_singleton = static_cast < RendererCanvasRenderRD * > ( RendererCanvasRender : : singleton ) ;
2021-01-06 00:01:50 +01:00
return canvas_singleton - > shader . canvas_shader . version_get_native_source_code ( version ) ;
}
2022-03-21 12:25:25 +01:00
RendererCanvasRenderRD : : CanvasShaderData : : ~ CanvasShaderData ( ) {
2022-04-05 12:40:26 +02:00
RendererCanvasRenderRD * canvas_singleton = static_cast < RendererCanvasRenderRD * > ( RendererCanvasRender : : singleton ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL ( canvas_singleton ) ;
2019-07-21 16:31:30 +02:00
//pipeline variants will clear themselves if shader is gone
if ( version . is_valid ( ) ) {
canvas_singleton - > shader . canvas_shader . version_free ( version ) ;
}
}
2019-06-16 04:45:24 +02:00
2022-08-01 03:59:14 +02:00
RendererRD : : MaterialStorage : : ShaderData * RendererCanvasRenderRD : : _create_shader_func ( ) {
2022-03-21 12:25:25 +01:00
CanvasShaderData * shader_data = memnew ( CanvasShaderData ) ;
2019-07-21 16:31:30 +02:00
return shader_data ;
}
2020-05-14 14:29:06 +02:00
2022-05-13 15:04:37 +02:00
bool RendererCanvasRenderRD : : CanvasMaterialData : : update_parameters ( const HashMap < StringName , Variant > & p_parameters , bool p_uniform_dirty , bool p_textures_dirty ) {
2022-04-05 12:40:26 +02:00
RendererCanvasRenderRD * canvas_singleton = static_cast < RendererCanvasRenderRD * > ( RendererCanvasRender : : singleton ) ;
2023-08-03 14:10:01 +02:00
bool uniform_set_changed = update_parameters_uniform_set ( p_parameters , p_uniform_dirty , p_textures_dirty , shader_data - > uniforms , shader_data - > ubo_offsets . ptr ( ) , shader_data - > texture_uniforms , shader_data - > default_texture_params , shader_data - > ubo_size , uniform_set , canvas_singleton - > shader . canvas_shader . version_get_shader ( shader_data - > version , 0 ) , MATERIAL_UNIFORM_SET , true , false , RD : : BARRIER_MASK_ALL_BARRIERS ) ;
bool uniform_set_srgb_changed = update_parameters_uniform_set ( p_parameters , p_uniform_dirty , p_textures_dirty , shader_data - > uniforms , shader_data - > ubo_offsets . ptr ( ) , shader_data - > texture_uniforms , shader_data - > default_texture_params , shader_data - > ubo_size , uniform_set_srgb , canvas_singleton - > shader . canvas_shader . version_get_shader ( shader_data - > version , 0 ) , MATERIAL_UNIFORM_SET , false , false , RD : : BARRIER_MASK_ALL_BARRIERS ) ;
return uniform_set_changed | | uniform_set_srgb_changed ;
2019-07-21 16:31:30 +02:00
}
2020-05-14 14:29:06 +02:00
2022-03-21 12:25:25 +01:00
RendererCanvasRenderRD : : CanvasMaterialData : : ~ CanvasMaterialData ( ) {
2021-07-08 00:55:20 +02:00
free_parameters_uniform_set ( uniform_set ) ;
2023-08-03 14:10:01 +02:00
free_parameters_uniform_set ( uniform_set_srgb ) ;
2019-07-21 16:31:30 +02:00
}
2019-06-16 04:45:24 +02:00
2022-08-01 03:59:14 +02:00
RendererRD : : MaterialStorage : : MaterialData * RendererCanvasRenderRD : : _create_material_func ( CanvasShaderData * p_shader ) {
2022-03-21 12:25:25 +01:00
CanvasMaterialData * material_data = memnew ( CanvasMaterialData ) ;
2019-07-21 16:31:30 +02:00
material_data - > shader_data = p_shader ;
//update will happen later anyway so do nothing.
return material_data ;
}
2019-06-16 04:45:24 +02:00
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : set_time ( double p_time ) {
2019-07-21 16:31:30 +02:00
state . time = p_time ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : update ( ) {
2019-07-21 16:31:30 +02:00
}
2022-06-21 02:08:33 +02:00
RendererCanvasRenderRD : : RendererCanvasRenderRD ( ) {
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage * texture_storage = RendererRD : : TextureStorage : : get_singleton ( ) ;
2022-03-21 12:25:25 +01:00
RendererRD : : MaterialStorage * material_storage = RendererRD : : MaterialStorage : : get_singleton ( ) ;
2019-07-21 16:31:30 +02:00
{ //create default samplers
2019-06-16 04:45:24 +02:00
2020-03-27 19:21:27 +01:00
default_samplers . default_filter = RS : : CANVAS_ITEM_TEXTURE_FILTER_LINEAR ;
default_samplers . default_repeat = RS : : CANVAS_ITEM_TEXTURE_REPEAT_DISABLED ;
2019-06-16 04:45:24 +02:00
}
{ //shader variants
2019-07-10 22:44:55 +02:00
String global_defines ;
2022-03-10 18:43:27 +01:00
uint64_t uniform_max_size = RD : : get_singleton ( ) - > limit_get ( RD : : LIMIT_MAX_UNIFORM_BUFFER_SIZE ) ;
2019-07-10 22:44:55 +02:00
if ( uniform_max_size < 65536 ) {
//Yes, you guessed right, ARM again
state . max_lights_per_render = 64 ;
global_defines + = " #define MAX_LIGHTS 64 \n " ;
} else {
state . max_lights_per_render = DEFAULT_MAX_LIGHTS_PER_RENDER ;
global_defines + = " #define MAX_LIGHTS " + itos ( DEFAULT_MAX_LIGHTS_PER_RENDER ) + " \n " ;
}
2023-05-26 23:35:16 +02:00
global_defines + = " \n #define SAMPLERS_BINDING_FIRST_INDEX " + itos ( SAMPLERS_BINDING_FIRST_INDEX ) + " \n " ;
2019-07-10 22:44:55 +02:00
state . light_uniforms = memnew_arr ( LightUniform , state . max_lights_per_render ) ;
2019-06-16 04:45:24 +02:00
Vector < String > variants ;
2019-07-10 22:44:55 +02:00
//non light variants
2019-06-16 04:45:24 +02:00
variants . push_back ( " " ) ; //none by default is first variant
variants . push_back ( " #define USE_NINEPATCH \n " ) ; //ninepatch is the second variant
2020-03-11 18:59:18 +01:00
variants . push_back ( " #define USE_PRIMITIVE \n " ) ; //primitive is the third
2019-06-24 21:13:06 +02:00
variants . push_back ( " #define USE_PRIMITIVE \n #define USE_POINT_SIZE \n " ) ; //points need point size
variants . push_back ( " #define USE_ATTRIBUTES \n " ) ; // attributes for vertex arrays
variants . push_back ( " #define USE_ATTRIBUTES \n #define USE_POINT_SIZE \n " ) ; //attributes with point size
2019-07-10 22:44:55 +02:00
//light variants
variants . push_back ( " #define USE_LIGHTING \n " ) ; //none by default is first variant
variants . push_back ( " #define USE_LIGHTING \n #define USE_NINEPATCH \n " ) ; //ninepatch is the second variant
2020-03-11 18:59:18 +01:00
variants . push_back ( " #define USE_LIGHTING \n #define USE_PRIMITIVE \n " ) ; //primitive is the third
2019-07-10 22:44:55 +02:00
variants . push_back ( " #define USE_LIGHTING \n #define USE_PRIMITIVE \n #define USE_POINT_SIZE \n " ) ; //points need point size
variants . push_back ( " #define USE_LIGHTING \n #define USE_ATTRIBUTES \n " ) ; // attributes for vertex arrays
variants . push_back ( " #define USE_LIGHTING \n #define USE_ATTRIBUTES \n #define USE_POINT_SIZE \n " ) ; //attributes with point size
2019-06-16 04:45:24 +02:00
2019-07-10 22:44:55 +02:00
shader . canvas_shader . initialize ( variants , global_defines ) ;
2019-06-16 04:45:24 +02:00
2019-07-10 22:44:55 +02:00
shader . default_version = shader . canvas_shader . version_create ( ) ;
shader . default_version_rd_shader = shader . canvas_shader . version_get_shader ( shader . default_version , SHADER_VARIANT_QUAD ) ;
2019-06-16 04:45:24 +02:00
2020-10-28 19:34:27 +01:00
RD : : PipelineColorBlendState blend_state ;
RD : : PipelineColorBlendState : : Attachment blend_attachment ;
blend_attachment . enable_blend = true ;
blend_attachment . color_blend_op = RD : : BLEND_OP_ADD ;
blend_attachment . src_color_blend_factor = RD : : BLEND_FACTOR_SRC_ALPHA ;
blend_attachment . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
blend_attachment . alpha_blend_op = RD : : BLEND_OP_ADD ;
blend_attachment . src_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
blend_attachment . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
blend_state . attachments . push_back ( blend_attachment ) ;
2022-08-12 13:03:28 +02:00
RD : : PipelineColorBlendState : : Attachment attachment_lcd ;
attachment_lcd . enable_blend = true ;
attachment_lcd . alpha_blend_op = RD : : BLEND_OP_ADD ;
attachment_lcd . color_blend_op = RD : : BLEND_OP_ADD ;
attachment_lcd . src_color_blend_factor = RD : : BLEND_FACTOR_CONSTANT_COLOR ;
attachment_lcd . dst_color_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_COLOR ;
attachment_lcd . src_alpha_blend_factor = RD : : BLEND_FACTOR_ONE ;
attachment_lcd . dst_alpha_blend_factor = RD : : BLEND_FACTOR_ONE_MINUS_SRC_ALPHA ;
RD : : PipelineColorBlendState blend_state_lcd ;
blend_state_lcd . attachments . push_back ( attachment_lcd ) ;
2019-07-10 22:44:55 +02:00
for ( int i = 0 ; i < PIPELINE_LIGHT_MODE_MAX ; i + + ) {
2019-06-16 04:45:24 +02:00
for ( int j = 0 ; j < PIPELINE_VARIANT_MAX ; j + + ) {
2019-06-24 21:13:06 +02:00
RD : : RenderPrimitive primitive [ PIPELINE_VARIANT_MAX ] = {
RD : : RENDER_PRIMITIVE_TRIANGLES ,
RD : : RENDER_PRIMITIVE_TRIANGLES ,
RD : : RENDER_PRIMITIVE_TRIANGLES ,
RD : : RENDER_PRIMITIVE_LINES ,
RD : : RENDER_PRIMITIVE_POINTS ,
RD : : RENDER_PRIMITIVE_TRIANGLES ,
2019-08-19 00:40:52 +02:00
RD : : RENDER_PRIMITIVE_TRIANGLE_STRIPS ,
2019-06-24 21:13:06 +02:00
RD : : RENDER_PRIMITIVE_LINES ,
2019-08-19 00:40:52 +02:00
RD : : RENDER_PRIMITIVE_LINESTRIPS ,
2019-06-24 21:13:06 +02:00
RD : : RENDER_PRIMITIVE_POINTS ,
2022-08-12 13:03:28 +02:00
RD : : RENDER_PRIMITIVE_TRIANGLES ,
2019-06-24 21:13:06 +02:00
} ;
2019-08-26 22:43:58 +02:00
2019-07-10 22:44:55 +02:00
ShaderVariant shader_variants [ PIPELINE_LIGHT_MODE_MAX ] [ PIPELINE_VARIANT_MAX ] = {
2022-08-12 13:03:28 +02:00
{
//non lit
2019-07-10 22:44:55 +02:00
SHADER_VARIANT_QUAD ,
SHADER_VARIANT_NINEPATCH ,
SHADER_VARIANT_PRIMITIVE ,
SHADER_VARIANT_PRIMITIVE ,
SHADER_VARIANT_PRIMITIVE_POINTS ,
SHADER_VARIANT_ATTRIBUTES ,
SHADER_VARIANT_ATTRIBUTES ,
2019-08-19 00:40:52 +02:00
SHADER_VARIANT_ATTRIBUTES ,
SHADER_VARIANT_ATTRIBUTES ,
2022-08-12 13:03:28 +02:00
SHADER_VARIANT_ATTRIBUTES_POINTS ,
SHADER_VARIANT_QUAD ,
} ,
{
//lit
2019-07-10 22:44:55 +02:00
SHADER_VARIANT_QUAD_LIGHT ,
SHADER_VARIANT_NINEPATCH_LIGHT ,
SHADER_VARIANT_PRIMITIVE_LIGHT ,
SHADER_VARIANT_PRIMITIVE_LIGHT ,
SHADER_VARIANT_PRIMITIVE_POINTS_LIGHT ,
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
2019-08-19 00:40:52 +02:00
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
SHADER_VARIANT_ATTRIBUTES_LIGHT ,
2022-08-12 13:03:28 +02:00
SHADER_VARIANT_ATTRIBUTES_POINTS_LIGHT ,
SHADER_VARIANT_QUAD_LIGHT ,
} ,
2019-06-24 21:13:06 +02:00
} ;
2019-06-16 04:45:24 +02:00
2019-07-10 22:44:55 +02:00
RID shader_variant = shader . canvas_shader . version_get_shader ( shader . default_version , shader_variants [ i ] [ j ] ) ;
2022-08-12 13:03:28 +02:00
if ( j = = PIPELINE_VARIANT_QUAD_LCD_BLEND ) {
shader . pipeline_variants . variants [ i ] [ j ] . setup ( shader_variant , primitive [ j ] , RD : : PipelineRasterizationState ( ) , RD : : PipelineMultisampleState ( ) , RD : : PipelineDepthStencilState ( ) , blend_state_lcd , RD : : DYNAMIC_STATE_BLEND_CONSTANTS ) ;
} else {
shader . pipeline_variants . variants [ i ] [ j ] . setup ( shader_variant , primitive [ j ] , RD : : PipelineRasterizationState ( ) , RD : : PipelineMultisampleState ( ) , RD : : PipelineDepthStencilState ( ) , blend_state , 0 ) ;
}
2019-06-16 04:45:24 +02:00
}
}
}
2019-07-21 16:31:30 +02:00
{
//shader compiler
2021-11-16 16:25:42 +01:00
ShaderCompiler : : DefaultIdentifierActions actions ;
2019-07-21 16:31:30 +02:00
actions . renames [ " VERTEX " ] = " vertex " ;
actions . renames [ " LIGHT_VERTEX " ] = " light_vertex " ;
actions . renames [ " SHADOW_VERTEX " ] = " shadow_vertex " ;
actions . renames [ " UV " ] = " uv " ;
2023-08-06 10:44:16 +02:00
actions . renames [ " POINT_SIZE " ] = " point_size " ;
2019-07-21 16:31:30 +02:00
2022-03-18 10:10:55 +01:00
actions . renames [ " MODEL_MATRIX " ] = " model_matrix " ;
2019-07-21 16:31:30 +02:00
actions . renames [ " CANVAS_MATRIX " ] = " canvas_data.canvas_transform " ;
actions . renames [ " SCREEN_MATRIX " ] = " canvas_data.screen_transform " ;
actions . renames [ " TIME " ] = " canvas_data.time " ;
2021-05-19 14:12:55 +02:00
actions . renames [ " PI " ] = _MKSTR ( Math_PI ) ;
actions . renames [ " TAU " ] = _MKSTR ( Math_TAU ) ;
actions . renames [ " E " ] = _MKSTR ( Math_E ) ;
2019-07-21 16:31:30 +02:00
actions . renames [ " AT_LIGHT_PASS " ] = " false " ;
actions . renames [ " INSTANCE_CUSTOM " ] = " instance_custom " ;
actions . renames [ " COLOR " ] = " color " ;
actions . renames [ " NORMAL " ] = " normal " ;
2020-12-23 10:34:26 +01:00
actions . renames [ " NORMAL_MAP " ] = " normal_map " ;
2021-01-04 14:08:01 +01:00
actions . renames [ " NORMAL_MAP_DEPTH " ] = " normal_map_depth " ;
2019-07-21 16:31:30 +02:00
actions . renames [ " TEXTURE " ] = " color_texture " ;
actions . renames [ " TEXTURE_PIXEL_SIZE " ] = " draw_data.color_texture_pixel_size " ;
actions . renames [ " NORMAL_TEXTURE " ] = " normal_texture " ;
actions . renames [ " SPECULAR_SHININESS_TEXTURE " ] = " specular_texture " ;
actions . renames [ " SPECULAR_SHININESS " ] = " specular_shininess " ;
actions . renames [ " SCREEN_UV " ] = " screen_uv " ;
actions . renames [ " SCREEN_PIXEL_SIZE " ] = " canvas_data.screen_pixel_size " ;
actions . renames [ " FRAGCOORD " ] = " gl_FragCoord " ;
actions . renames [ " POINT_COORD " ] = " gl_PointCoord " ;
2021-11-09 02:12:11 +01:00
actions . renames [ " INSTANCE_ID " ] = " gl_InstanceIndex " ;
actions . renames [ " VERTEX_ID " ] = " gl_VertexIndex " ;
2019-07-21 16:31:30 +02:00
2021-01-09 20:45:03 +01:00
actions . renames [ " LIGHT_POSITION " ] = " light_position " ;
2022-09-28 20:45:08 +02:00
actions . renames [ " LIGHT_DIRECTION " ] = " light_direction " ;
actions . renames [ " LIGHT_IS_DIRECTIONAL " ] = " is_directional " ;
2019-07-21 16:31:30 +02:00
actions . renames [ " LIGHT_COLOR " ] = " light_color " ;
actions . renames [ " LIGHT_ENERGY " ] = " light_energy " ;
actions . renames [ " LIGHT " ] = " light " ;
actions . renames [ " SHADOW_MODULATE " ] = " shadow_modulate " ;
2020-11-26 13:50:21 +01:00
actions . renames [ " texture_sdf " ] = " texture_sdf " ;
actions . renames [ " texture_sdf_normal " ] = " texture_sdf_normal " ;
actions . renames [ " sdf_to_screen_uv " ] = " sdf_to_screen_uv " ;
actions . renames [ " screen_uv_to_sdf " ] = " screen_uv_to_sdf " ;
2019-07-21 16:31:30 +02:00
actions . usage_defines [ " COLOR " ] = " #define COLOR_USED \n " ;
actions . usage_defines [ " SCREEN_UV " ] = " #define SCREEN_UV_USED \n " ;
actions . usage_defines [ " SCREEN_PIXEL_SIZE " ] = " @SCREEN_UV " ;
actions . usage_defines [ " NORMAL " ] = " #define NORMAL_USED \n " ;
2020-12-23 10:34:26 +01:00
actions . usage_defines [ " NORMAL_MAP " ] = " #define NORMAL_MAP_USED \n " ;
2019-07-21 16:31:30 +02:00
actions . usage_defines [ " LIGHT " ] = " #define LIGHT_SHADER_CODE_USED \n " ;
2022-06-21 06:28:50 +02:00
actions . usage_defines [ " SPECULAR_SHININESS " ] = " #define SPECULAR_SHININESS_USED \n " ;
2023-08-06 10:44:16 +02:00
actions . usage_defines [ " POINT_SIZE " ] = " #define USE_POINT_SIZE \n " ;
2019-07-21 16:31:30 +02:00
actions . render_mode_defines [ " skip_vertex_transform " ] = " #define SKIP_TRANSFORM_USED \n " ;
2020-11-26 13:50:21 +01:00
actions . render_mode_defines [ " unshaded " ] = " #define MODE_UNSHADED \n " ;
actions . render_mode_defines [ " light_only " ] = " #define MODE_LIGHT_ONLY \n " ;
2023-08-30 14:59:06 +02:00
actions . render_mode_defines [ " world_vertex_coords " ] = " #define USE_WORLD_VERTEX_COORDS \n " ;
2019-07-21 16:31:30 +02:00
actions . custom_samplers [ " TEXTURE " ] = " texture_sampler " ;
actions . custom_samplers [ " NORMAL_TEXTURE " ] = " texture_sampler " ;
actions . custom_samplers [ " SPECULAR_SHININESS_TEXTURE " ] = " texture_sampler " ;
2020-10-24 17:15:43 +02:00
actions . base_texture_binding_index = 1 ;
actions . texture_layout_set = MATERIAL_UNIFORM_SET ;
2019-07-21 16:31:30 +02:00
actions . base_uniform_string = " material. " ;
2019-07-27 15:23:24 +02:00
actions . default_filter = ShaderLanguage : : FILTER_LINEAR ;
actions . default_repeat = ShaderLanguage : : REPEAT_DISABLE ;
2019-09-15 06:01:52 +02:00
actions . base_varying_index = 4 ;
2019-07-21 16:31:30 +02:00
2022-04-03 18:56:43 +02:00
actions . global_buffer_array_variable = " global_shader_uniforms.data " ;
2020-04-17 04:52:00 +02:00
2019-07-21 16:31:30 +02:00
shader . compiler . initialize ( actions ) ;
}
2019-07-05 03:54:32 +02:00
{ //shadow rendering
Vector < String > versions ;
2020-11-26 13:50:21 +01:00
versions . push_back ( " \n #define MODE_SHADOW \n " ) ; //shadow
versions . push_back ( " \n #define MODE_SDF \n " ) ; //sdf
2019-07-05 03:54:32 +02:00
shadow_render . shader . initialize ( versions ) ;
{
Vector < RD : : AttachmentFormat > attachments ;
RD : : AttachmentFormat af_color ;
af_color . format = RD : : DATA_FORMAT_R32_SFLOAT ;
af_color . usage_flags = RD : : TEXTURE_USAGE_SAMPLING_BIT | RD : : TEXTURE_USAGE_COLOR_ATTACHMENT_BIT ;
attachments . push_back ( af_color ) ;
RD : : AttachmentFormat af_depth ;
2020-10-24 17:15:43 +02:00
af_depth . format = RD : : DATA_FORMAT_D32_SFLOAT ;
2019-07-05 03:54:32 +02:00
af_depth . usage_flags = RD : : TEXTURE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT ;
attachments . push_back ( af_depth ) ;
shadow_render . framebuffer_format = RD : : get_singleton ( ) - > framebuffer_format_create ( attachments ) ;
}
2020-11-26 13:50:21 +01:00
{
Vector < RD : : AttachmentFormat > attachments ;
RD : : AttachmentFormat af_color ;
af_color . format = RD : : DATA_FORMAT_R8_UNORM ;
af_color . usage_flags = RD : : TEXTURE_USAGE_SAMPLING_BIT | RD : : TEXTURE_USAGE_STORAGE_BIT | RD : : TEXTURE_USAGE_COLOR_ATTACHMENT_BIT ;
attachments . push_back ( af_color ) ;
shadow_render . sdf_framebuffer_format = RD : : get_singleton ( ) - > framebuffer_format_create ( attachments ) ;
}
2019-07-05 03:54:32 +02:00
//pipelines
2020-04-21 17:16:45 +02:00
Vector < RD : : VertexAttribute > vf ;
RD : : VertexAttribute vd ;
2020-11-26 13:50:21 +01:00
vd . format = sizeof ( real_t ) = = sizeof ( float ) ? RD : : DATA_FORMAT_R32G32B32_SFLOAT : RD : : DATA_FORMAT_R64G64B64_SFLOAT ;
2019-07-05 03:54:32 +02:00
vd . location = 0 ;
vd . offset = 0 ;
2020-11-26 13:50:21 +01:00
vd . stride = sizeof ( real_t ) * 3 ;
2019-07-05 03:54:32 +02:00
vf . push_back ( vd ) ;
shadow_render . vertex_format = RD : : get_singleton ( ) - > vertex_format_create ( vf ) ;
2020-11-26 13:50:21 +01:00
vd . format = sizeof ( real_t ) = = sizeof ( float ) ? RD : : DATA_FORMAT_R32G32_SFLOAT : RD : : DATA_FORMAT_R64G64_SFLOAT ;
vd . stride = sizeof ( real_t ) * 2 ;
vf . write [ 0 ] = vd ;
shadow_render . sdf_vertex_format = RD : : get_singleton ( ) - > vertex_format_create ( vf ) ;
2019-07-05 03:54:32 +02:00
shadow_render . shader_version = shadow_render . shader . version_create ( ) ;
for ( int i = 0 ; i < 3 ; i + + ) {
RD : : PipelineRasterizationState rs ;
rs . cull_mode = i = = 0 ? RD : : POLYGON_CULL_DISABLED : ( i = = 1 ? RD : : POLYGON_CULL_FRONT : RD : : POLYGON_CULL_BACK ) ;
RD : : PipelineDepthStencilState ds ;
ds . enable_depth_write = true ;
ds . enable_depth_test = true ;
ds . depth_compare_operator = RD : : COMPARE_OP_LESS ;
2020-11-26 13:50:21 +01:00
shadow_render . render_pipelines [ i ] = RD : : get_singleton ( ) - > render_pipeline_create ( shadow_render . shader . version_get_shader ( shadow_render . shader_version , SHADOW_RENDER_MODE_SHADOW ) , shadow_render . framebuffer_format , shadow_render . vertex_format , RD : : RENDER_PRIMITIVE_TRIANGLES , rs , RD : : PipelineMultisampleState ( ) , ds , RD : : PipelineColorBlendState : : create_disabled ( ) , 0 ) ;
}
for ( int i = 0 ; i < 2 ; i + + ) {
shadow_render . sdf_render_pipelines [ i ] = RD : : get_singleton ( ) - > render_pipeline_create ( shadow_render . shader . version_get_shader ( shadow_render . shader_version , SHADOW_RENDER_MODE_SDF ) , shadow_render . sdf_framebuffer_format , shadow_render . sdf_vertex_format , i = = 0 ? RD : : RENDER_PRIMITIVE_TRIANGLES : RD : : RENDER_PRIMITIVE_LINES , RD : : PipelineRasterizationState ( ) , RD : : PipelineMultisampleState ( ) , RD : : PipelineDepthStencilState ( ) , RD : : PipelineColorBlendState : : create_disabled ( ) , 0 ) ;
2019-07-05 03:54:32 +02:00
}
}
2019-06-16 04:45:24 +02:00
{ //bindings
2020-10-24 17:15:43 +02:00
state . canvas_state_buffer = RD : : get_singleton ( ) - > uniform_buffer_create ( sizeof ( State : : Buffer ) ) ;
state . lights_uniform_buffer = RD : : get_singleton ( ) - > uniform_buffer_create ( sizeof ( LightUniform ) * state . max_lights_per_render ) ;
RD : : SamplerState shadow_sampler_state ;
shadow_sampler_state . mag_filter = RD : : SAMPLER_FILTER_LINEAR ;
shadow_sampler_state . min_filter = RD : : SAMPLER_FILTER_LINEAR ;
shadow_sampler_state . repeat_u = RD : : SAMPLER_REPEAT_MODE_REPEAT ; //shadow wrap around
shadow_sampler_state . compare_op = RD : : COMPARE_OP_GREATER ;
shadow_sampler_state . enable_compare = true ;
state . shadow_sampler = RD : : get_singleton ( ) - > sampler_create ( shadow_sampler_state ) ;
2019-06-16 04:45:24 +02:00
}
2019-06-24 21:13:06 +02:00
{
//polygon buffers
polygon_buffers . last_id = 1 ;
}
2019-06-16 04:45:24 +02:00
{ // default index buffer
2020-02-17 22:06:54 +01:00
Vector < uint8_t > pv ;
2023-09-04 00:59:03 +02:00
pv . resize ( 6 * 2 ) ;
2019-06-16 04:45:24 +02:00
{
2020-02-17 22:06:54 +01:00
uint8_t * w = pv . ptrw ( ) ;
2023-09-04 00:59:03 +02:00
uint16_t * p16 = ( uint16_t * ) w ;
p16 [ 0 ] = 0 ;
p16 [ 1 ] = 1 ;
p16 [ 2 ] = 2 ;
p16 [ 3 ] = 0 ;
p16 [ 4 ] = 2 ;
p16 [ 5 ] = 3 ;
2019-06-16 04:45:24 +02:00
}
2023-09-04 00:59:03 +02:00
shader . quad_index_buffer = RD : : get_singleton ( ) - > index_buffer_create ( 6 , RenderingDevice : : INDEX_BUFFER_FORMAT_UINT16 , pv ) ;
2019-06-24 21:13:06 +02:00
shader . quad_index_array = RD : : get_singleton ( ) - > index_array_create ( shader . quad_index_buffer , 0 , 6 ) ;
}
{ //primitive
primitive_arrays . index_array [ 0 ] = shader . quad_index_array = RD : : get_singleton ( ) - > index_array_create ( shader . quad_index_buffer , 0 , 1 ) ;
primitive_arrays . index_array [ 1 ] = shader . quad_index_array = RD : : get_singleton ( ) - > index_array_create ( shader . quad_index_buffer , 0 , 2 ) ;
primitive_arrays . index_array [ 2 ] = shader . quad_index_array = RD : : get_singleton ( ) - > index_array_create ( shader . quad_index_buffer , 0 , 3 ) ;
primitive_arrays . index_array [ 3 ] = shader . quad_index_array = RD : : get_singleton ( ) - > index_array_create ( shader . quad_index_buffer , 0 , 6 ) ;
2019-06-16 04:45:24 +02:00
}
2020-10-24 17:15:43 +02:00
{
//default shadow texture to keep uniform set happy
RD : : TextureFormat tf ;
2020-11-27 04:50:05 +01:00
tf . texture_type = RD : : TEXTURE_TYPE_2D ;
2020-10-24 17:15:43 +02:00
tf . width = 4 ;
tf . height = 4 ;
tf . usage_bits = RD : : TEXTURE_USAGE_SAMPLING_BIT ;
tf . format = RD : : DATA_FORMAT_R32_SFLOAT ;
state . shadow_texture = RD : : get_singleton ( ) - > texture_create ( tf , RD : : TextureView ( ) ) ;
}
{
Vector < RD : : Uniform > uniforms ;
{
RD : : Uniform u ;
2020-10-17 03:19:21 +02:00
u . uniform_type = RD : : UNIFORM_TYPE_STORAGE_BUFFER ;
2020-10-24 17:15:43 +02:00
u . binding = 0 ;
2022-04-02 07:29:04 +02:00
u . append_id ( RendererRD : : MeshStorage : : get_singleton ( ) - > get_default_rd_storage_buffer ( ) ) ;
2020-10-24 17:15:43 +02:00
uniforms . push_back ( u ) ;
}
state . default_transforms_uniform_set = RD : : get_singleton ( ) - > uniform_set_create ( uniforms , shader . default_version_rd_shader , TRANSFORMS_UNIFORM_SET ) ;
}
2022-04-07 16:00:51 +02:00
default_canvas_texture = texture_storage - > canvas_texture_allocate ( ) ;
texture_storage - > canvas_texture_initialize ( default_canvas_texture ) ;
2020-10-24 17:15:43 +02:00
2021-02-17 17:44:49 +01:00
state . shadow_texture_size = GLOBAL_GET ( " rendering/2d/shadow_atlas/size " ) ;
2019-06-16 04:45:24 +02:00
2019-07-21 16:31:30 +02:00
//create functions for shader and material
2022-08-01 03:59:14 +02:00
material_storage - > shader_set_data_request_function ( RendererRD : : MaterialStorage : : SHADER_TYPE_2D , _create_shader_funcs ) ;
material_storage - > material_set_data_request_function ( RendererRD : : MaterialStorage : : SHADER_TYPE_2D , _create_material_funcs ) ;
2019-07-21 16:31:30 +02:00
state . time = 0 ;
2020-10-28 19:34:27 +01:00
{
2022-03-21 12:25:25 +01:00
default_canvas_group_shader = material_storage - > shader_allocate ( ) ;
material_storage - > shader_initialize ( default_canvas_group_shader ) ;
2021-02-09 17:19:03 +01:00
2022-03-21 12:25:25 +01:00
material_storage - > shader_set_code ( default_canvas_group_shader , R " (
2021-08-18 03:09:22 +02:00
// Default CanvasGroup shader.
2021-07-19 08:06:51 +02:00
shader_type canvas_item ;
2023-02-04 00:40:19 +01:00
render_mode unshaded ;
2021-02-09 17:19:03 +01:00
2023-01-05 20:55:59 +01:00
uniform sampler2D screen_texture : hint_screen_texture , repeat_disable , filter_nearest ;
2021-07-19 08:06:51 +02:00
void fragment ( ) {
2023-01-05 20:55:59 +01:00
vec4 c = textureLod ( screen_texture , SCREEN_UV , 0.0 ) ;
2021-07-19 08:06:51 +02:00
if ( c . a > 0.0001 ) {
c . rgb / = c . a ;
}
COLOR * = c ;
}
) " );
2022-03-21 12:25:25 +01:00
default_canvas_group_material = material_storage - > material_allocate ( ) ;
material_storage - > material_initialize ( default_canvas_group_material ) ;
2021-02-09 17:19:03 +01:00
2022-03-21 12:25:25 +01:00
material_storage - > material_set_shader ( default_canvas_group_material , default_canvas_group_shader ) ;
2020-10-28 19:34:27 +01:00
}
2022-10-07 00:27:11 +02:00
{
default_clip_children_shader = material_storage - > shader_allocate ( ) ;
material_storage - > shader_initialize ( default_clip_children_shader ) ;
material_storage - > shader_set_code ( default_clip_children_shader , R " (
// Default clip children shader.
2022-10-12 21:45:47 +02:00
2022-10-07 00:27:11 +02:00
shader_type canvas_item ;
2023-02-04 00:40:19 +01:00
render_mode unshaded ;
2022-10-12 21:45:47 +02:00
2023-01-05 20:55:59 +01:00
uniform sampler2D screen_texture : hint_screen_texture , repeat_disable , filter_nearest ;
2022-10-07 00:27:11 +02:00
void fragment ( ) {
2023-01-05 20:55:59 +01:00
vec4 c = textureLod ( screen_texture , SCREEN_UV , 0.0 ) ;
2022-10-07 00:27:11 +02:00
COLOR . rgb = c . rgb ;
}
) " );
default_clip_children_material = material_storage - > material_allocate ( ) ;
material_storage - > material_initialize ( default_clip_children_material ) ;
material_storage - > material_set_shader ( default_clip_children_material , default_clip_children_shader ) ;
}
2020-02-20 15:13:08 +01:00
static_assert ( sizeof ( PushConstant ) = = 128 ) ;
2019-06-16 04:45:24 +02:00
}
2019-06-24 21:13:06 +02:00
2020-12-04 19:26:24 +01:00
bool RendererCanvasRenderRD : : free ( RID p_rid ) {
2019-07-05 03:54:32 +02:00
if ( canvas_light_owner . owns ( p_rid ) ) {
2021-09-29 19:08:41 +02:00
CanvasLight * cl = canvas_light_owner . get_or_null ( p_rid ) ;
2023-09-09 17:04:18 +02:00
ERR_FAIL_NULL_V ( cl , false ) ;
2020-10-24 17:15:43 +02:00
light_set_use_shadow ( p_rid , false ) ;
2019-07-05 03:54:32 +02:00
canvas_light_owner . free ( p_rid ) ;
} else if ( occluder_polygon_owner . owns ( p_rid ) ) {
2020-11-26 13:50:21 +01:00
occluder_polygon_set_shape ( p_rid , Vector < Vector2 > ( ) , false ) ;
2019-07-05 03:54:32 +02:00
occluder_polygon_owner . free ( p_rid ) ;
} else {
return false ;
}
return true ;
}
2020-12-04 19:26:24 +01:00
void RendererCanvasRenderRD : : set_shadow_texture_size ( int p_size ) {
2020-10-24 17:15:43 +02:00
p_size = nearest_power_of_2_templated ( p_size ) ;
if ( p_size = = state . shadow_texture_size ) {
return ;
}
state . shadow_texture_size = p_size ;
if ( state . shadow_fb . is_valid ( ) ) {
RD : : get_singleton ( ) - > free ( state . shadow_texture ) ;
RD : : get_singleton ( ) - > free ( state . shadow_depth_texture ) ;
state . shadow_fb = RID ( ) ;
{
//create a default shadow texture to keep uniform set happy (and that it gets erased when a new one is created)
RD : : TextureFormat tf ;
2020-11-27 04:50:05 +01:00
tf . texture_type = RD : : TEXTURE_TYPE_2D ;
2020-10-24 17:15:43 +02:00
tf . width = 4 ;
tf . height = 4 ;
tf . usage_bits = RD : : TEXTURE_USAGE_SAMPLING_BIT ;
tf . format = RD : : DATA_FORMAT_R32_SFLOAT ;
state . shadow_texture = RD : : get_singleton ( ) - > texture_create ( tf , RD : : TextureView ( ) ) ;
}
}
}
2020-12-04 19:26:24 +01:00
RendererCanvasRenderRD : : ~ RendererCanvasRenderRD ( ) {
2022-03-21 12:25:25 +01:00
RendererRD : : MaterialStorage * material_storage = RendererRD : : MaterialStorage : : get_singleton ( ) ;
2019-06-24 21:13:06 +02:00
//canvas state
2022-03-21 12:25:25 +01:00
material_storage - > material_free ( default_canvas_group_material ) ;
material_storage - > shader_free ( default_canvas_group_shader ) ;
2020-10-28 19:34:27 +01:00
2022-10-07 00:27:11 +02:00
material_storage - > material_free ( default_clip_children_material ) ;
material_storage - > shader_free ( default_clip_children_shader ) ;
2019-07-29 23:19:31 +02:00
{
if ( state . canvas_state_buffer . is_valid ( ) ) {
RD : : get_singleton ( ) - > free ( state . canvas_state_buffer ) ;
}
memdelete_arr ( state . light_uniforms ) ;
RD : : get_singleton ( ) - > free ( state . lights_uniform_buffer ) ;
2019-06-24 21:13:06 +02:00
}
2019-07-29 23:19:31 +02:00
//shadow rendering
{
shadow_render . shader . version_free ( shadow_render . shader_version ) ;
//this will also automatically clear all pipelines
RD : : get_singleton ( ) - > free ( state . shadow_sampler ) ;
}
2019-06-24 21:13:06 +02:00
//bindings
//shaders
shader . canvas_shader . version_free ( shader . default_version ) ;
//buffers
2019-07-29 23:19:31 +02:00
{
RD : : get_singleton ( ) - > free ( shader . quad_index_array ) ;
RD : : get_singleton ( ) - > free ( shader . quad_index_buffer ) ;
//primitives are erase by dependency
}
2019-06-24 21:13:06 +02:00
2020-10-24 17:15:43 +02:00
if ( state . shadow_fb . is_valid ( ) ) {
RD : : get_singleton ( ) - > free ( state . shadow_depth_texture ) ;
}
RD : : get_singleton ( ) - > free ( state . shadow_texture ) ;
2022-04-07 16:00:51 +02:00
RendererRD : : TextureStorage : : get_singleton ( ) - > canvas_texture_free ( default_canvas_texture ) ;
2019-06-24 21:13:06 +02:00
//pipelines don't need freeing, they are all gone after shaders are gone
}