2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* gdscript_analyzer.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. */
/**************************************************************************/
2020-05-02 00:14:56 +02:00
# include "gdscript_analyzer.h"
2021-08-20 15:52:58 +02:00
# include "core/config/engine.h"
2020-11-07 23:33:38 +01:00
# include "core/config/project_settings.h"
2022-11-08 23:41:50 +01:00
# include "core/core_string_names.h"
2021-06-11 14:51:48 +02:00
# include "core/io/file_access.h"
2020-05-02 00:14:56 +02:00
# include "core/io/resource_loader.h"
2020-11-07 23:33:38 +01:00
# include "core/object/class_db.h"
# include "core/object/script_language.h"
# include "core/templates/hash_map.h"
2020-06-11 00:53:25 +02:00
# include "gdscript.h"
2020-11-26 15:56:32 +01:00
# include "gdscript_utility_functions.h"
2022-10-09 18:41:28 +02:00
# include "scene/resources/packed_scene.h"
2020-05-02 00:14:56 +02:00
2022-12-04 04:02:03 +01:00
# define UNNAMED_ENUM "<anonymous enum>"
# define ENUM_SEPARATOR "::"
2020-11-26 15:56:32 +01:00
static MethodInfo info_from_utility_func ( const StringName & p_function ) {
ERR_FAIL_COND_V ( ! Variant : : has_utility_function ( p_function ) , MethodInfo ( ) ) ;
MethodInfo info ( p_function ) ;
if ( Variant : : has_utility_function_return_value ( p_function ) ) {
info . return_val . type = Variant : : get_utility_function_return_type ( p_function ) ;
if ( info . return_val . type = = Variant : : NIL ) {
info . return_val . usage | = PROPERTY_USAGE_NIL_IS_VARIANT ;
}
}
if ( Variant : : is_utility_function_vararg ( p_function ) ) {
info . flags | = METHOD_FLAG_VARARG ;
} else {
for ( int i = 0 ; i < Variant : : get_utility_function_argument_count ( p_function ) ; i + + ) {
PropertyInfo pi ;
# ifdef DEBUG_METHODS_ENABLED
pi . name = Variant : : get_utility_function_argument_name ( p_function , i ) ;
# else
pi . name = " arg " + itos ( i + 1 ) ;
# endif
pi . type = Variant : : get_utility_function_argument_type ( p_function , i ) ;
if ( pi . type = = Variant : : NIL ) {
pi . usage | = PROPERTY_USAGE_NIL_IS_VARIANT ;
}
info . arguments . push_back ( pi ) ;
}
}
return info ;
}
2020-06-11 00:53:25 +02:00
static GDScriptParser : : DataType make_callable_type ( const MethodInfo & p_info ) {
GDScriptParser : : DataType type ;
type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
type . kind = GDScriptParser : : DataType : : BUILTIN ;
type . builtin_type = Variant : : CALLABLE ;
type . is_constant = true ;
type . method_info = p_info ;
return type ;
}
static GDScriptParser : : DataType make_signal_type ( const MethodInfo & p_info ) {
GDScriptParser : : DataType type ;
type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
type . kind = GDScriptParser : : DataType : : BUILTIN ;
type . builtin_type = Variant : : SIGNAL ;
type . is_constant = true ;
type . method_info = p_info ;
return type ;
}
static GDScriptParser : : DataType make_native_meta_type ( const StringName & p_class_name ) {
GDScriptParser : : DataType type ;
type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
type . kind = GDScriptParser : : DataType : : NATIVE ;
type . builtin_type = Variant : : OBJECT ;
type . native_type = p_class_name ;
2023-01-12 00:03:53 +01:00
type . is_constant = true ;
type . is_meta_type = true ;
return type ;
}
static GDScriptParser : : DataType make_script_meta_type ( const Ref < Script > & p_script ) {
GDScriptParser : : DataType type ;
type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
type . kind = GDScriptParser : : DataType : : SCRIPT ;
type . builtin_type = Variant : : OBJECT ;
type . native_type = p_script - > get_instance_base_type ( ) ;
type . script_type = p_script ;
type . script_path = p_script - > get_path ( ) ;
type . is_constant = true ;
2020-06-11 00:53:25 +02:00
type . is_meta_type = true ;
return type ;
}
2022-12-04 04:02:03 +01:00
// In enum types, native_type is used to store the class (native or otherwise) that the enum belongs to.
// This disambiguates between similarly named enums in base classes or outer classes
static GDScriptParser : : DataType make_enum_type ( const StringName & p_enum_name , const String & p_base_name , const bool p_meta = false ) {
2020-06-12 02:49:58 +02:00
GDScriptParser : : DataType type ;
type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
type . kind = GDScriptParser : : DataType : : ENUM ;
2022-12-04 04:02:03 +01:00
type . builtin_type = p_meta ? Variant : : DICTIONARY : Variant : : INT ;
type . enum_type = p_enum_name ;
2020-06-12 02:49:58 +02:00
type . is_constant = true ;
2022-12-04 04:02:03 +01:00
type . is_meta_type = p_meta ;
// For enums, native_type is only used to check compatibility in is_type_compatible()
// We can set anything readable here for error messages, as long as it uniquely identifies the type of the enum
type . native_type = p_base_name + ENUM_SEPARATOR + p_enum_name ;
return type ;
}
static GDScriptParser : : DataType make_native_enum_type ( const StringName & p_enum_name , const StringName & p_native_class , const bool p_meta = true ) {
GDScriptParser : : DataType type = make_enum_type ( p_enum_name , p_native_class , p_meta ) ;
2020-06-12 02:49:58 +02:00
List < StringName > enum_values ;
2021-08-17 15:06:54 +02:00
ClassDB : : get_enum_constants ( p_native_class , p_enum_name , & enum_values ) ;
2020-06-12 02:49:58 +02:00
2021-07-16 05:45:57 +02:00
for ( const StringName & E : enum_values ) {
2021-08-17 15:06:54 +02:00
type . enum_values [ E ] = ClassDB : : get_integer_constant ( p_native_class , E ) ;
2020-06-12 02:49:58 +02:00
}
return type ;
}
2020-06-11 00:53:25 +02:00
static GDScriptParser : : DataType make_builtin_meta_type ( Variant : : Type p_type ) {
GDScriptParser : : DataType type ;
type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
type . kind = GDScriptParser : : DataType : : BUILTIN ;
type . builtin_type = p_type ;
type . is_constant = true ;
type . is_meta_type = true ;
return type ;
}
2022-12-04 04:02:03 +01:00
static StringName enum_get_value_name ( const GDScriptParser : : DataType p_type , int64_t p_val ) {
// Check that an enum has a given value, not key.
// Make sure that implicit conversion to int64_t is sensible before calling!
HashMap < StringName , int64_t > : : ConstIterator i = p_type . enum_values . begin ( ) ;
while ( i ) {
if ( i - > value = = p_val ) {
return i - > key ;
}
+ + i ;
}
return StringName ( ) ;
}
2022-11-08 23:41:50 +01:00
bool GDScriptAnalyzer : : has_member_name_conflict_in_script_class ( const StringName & p_member_name , const GDScriptParser : : ClassNode * p_class , const GDScriptParser : : Node * p_member ) {
2021-08-24 18:49:03 +02:00
if ( p_class - > members_indices . has ( p_member_name ) ) {
int index = p_class - > members_indices [ p_member_name ] ;
const GDScriptParser : : ClassNode : : Member * member = & p_class - > members [ index ] ;
if ( member - > type = = GDScriptParser : : ClassNode : : Member : : VARIABLE | |
member - > type = = GDScriptParser : : ClassNode : : Member : : CONSTANT | |
member - > type = = GDScriptParser : : ClassNode : : Member : : ENUM | |
member - > type = = GDScriptParser : : ClassNode : : Member : : ENUM_VALUE | |
member - > type = = GDScriptParser : : ClassNode : : Member : : CLASS | |
member - > type = = GDScriptParser : : ClassNode : : Member : : SIGNAL ) {
return true ;
}
2022-11-08 23:41:50 +01:00
if ( p_member - > type ! = GDScriptParser : : Node : : FUNCTION & & member - > type = = GDScriptParser : : ClassNode : : Member : : FUNCTION ) {
return true ;
}
2021-08-24 18:49:03 +02:00
}
return false ;
}
bool GDScriptAnalyzer : : has_member_name_conflict_in_native_type ( const StringName & p_member_name , const StringName & p_native_type_string ) {
if ( ClassDB : : has_signal ( p_native_type_string , p_member_name ) ) {
return true ;
}
if ( ClassDB : : has_property ( p_native_type_string , p_member_name ) ) {
return true ;
}
if ( ClassDB : : has_integer_constant ( p_native_type_string , p_member_name ) ) {
return true ;
}
2022-11-08 23:41:50 +01:00
if ( p_member_name = = CoreStringNames : : get_singleton ( ) - > _script ) {
return true ;
}
2021-08-24 18:49:03 +02:00
return false ;
}
Error GDScriptAnalyzer : : check_native_member_name_conflict ( const StringName & p_member_name , const GDScriptParser : : Node * p_member_node , const StringName & p_native_type_string ) {
if ( has_member_name_conflict_in_native_type ( p_member_name , p_native_type_string ) ) {
push_error ( vformat ( R " (Member " % s " redefined (original in native class '%s')) " , p_member_name , p_native_type_string ) , p_member_node ) ;
return ERR_PARSE_ERROR ;
}
if ( class_exists ( p_member_name ) ) {
2021-09-13 15:51:29 +02:00
push_error ( vformat ( R " (The member " % s " shadows a native class.) " , p_member_name ) , p_member_node ) ;
2021-08-24 18:49:03 +02:00
return ERR_PARSE_ERROR ;
}
2021-09-29 16:23:16 +02:00
if ( GDScriptParser : : get_builtin_type ( p_member_name ) ! = Variant : : VARIANT_MAX ) {
push_error ( vformat ( R " (The member " % s " cannot have the same name as a builtin type.) " , p_member_name ) , p_member_node ) ;
return ERR_PARSE_ERROR ;
}
2021-08-24 18:49:03 +02:00
return OK ;
}
Error GDScriptAnalyzer : : check_class_member_name_conflict ( const GDScriptParser : : ClassNode * p_class_node , const StringName & p_member_name , const GDScriptParser : : Node * p_member_node ) {
2022-12-04 04:02:03 +01:00
// TODO check outer classes for static members only
2021-08-24 18:49:03 +02:00
const GDScriptParser : : DataType * current_data_type = & p_class_node - > base_type ;
while ( current_data_type & & current_data_type - > kind = = GDScriptParser : : DataType : : Kind : : CLASS ) {
GDScriptParser : : ClassNode * current_class_node = current_data_type - > class_type ;
2022-11-08 23:41:50 +01:00
if ( has_member_name_conflict_in_script_class ( p_member_name , current_class_node , p_member_node ) ) {
2022-12-03 03:30:19 +01:00
String parent_class_name = current_class_node - > fqcn ;
if ( current_class_node - > identifier ! = nullptr ) {
parent_class_name = current_class_node - > identifier - > name ;
}
push_error ( vformat ( R " (The member " % s " already exists in parent class %s.) " , p_member_name , parent_class_name ) , p_member_node ) ;
2021-08-24 18:49:03 +02:00
return ERR_PARSE_ERROR ;
}
current_data_type = & current_class_node - > base_type ;
}
2022-11-08 23:41:50 +01:00
// No need for native class recursion because Node exposes all Object's properties.
2021-08-24 18:49:03 +02:00
if ( current_data_type & & current_data_type - > kind = = GDScriptParser : : DataType : : Kind : : NATIVE ) {
2022-02-06 14:12:19 +01:00
if ( current_data_type - > native_type ! = StringName ( ) ) {
2021-08-24 18:49:03 +02:00
return check_native_member_name_conflict (
p_member_name ,
p_member_node ,
current_data_type - > native_type ) ;
}
}
return OK ;
}
2022-12-04 22:55:40 +01:00
void GDScriptAnalyzer : : get_class_node_current_scope_classes ( GDScriptParser : : ClassNode * p_node , List < GDScriptParser : : ClassNode * > * p_list ) {
2022-12-18 06:38:53 +01:00
ERR_FAIL_NULL ( p_node ) ;
ERR_FAIL_NULL ( p_list ) ;
2022-12-04 22:55:40 +01:00
if ( p_list - > find ( p_node ) ! = nullptr ) {
return ;
}
2022-12-18 06:38:53 +01:00
2022-12-04 22:55:40 +01:00
p_list - > push_back ( p_node ) ;
2022-12-11 03:57:35 +01:00
// TODO: Try to solve class inheritance if not yet resolving.
2022-12-04 22:55:40 +01:00
// Prioritize node base type over its outer class
if ( p_node - > base_type . class_type ! = nullptr ) {
get_class_node_current_scope_classes ( p_node - > base_type . class_type , p_list ) ;
}
if ( p_node - > outer ! = nullptr ) {
get_class_node_current_scope_classes ( p_node - > outer , p_list ) ;
}
}
2022-12-11 03:57:35 +01:00
Error GDScriptAnalyzer : : resolve_class_inheritance ( GDScriptParser : : ClassNode * p_class , const GDScriptParser : : Node * p_source ) {
if ( p_source = = nullptr & & parser - > has_class ( p_class ) ) {
p_source = p_class ;
}
if ( p_class - > base_type . is_resolving ( ) ) {
push_error ( vformat ( R " (Could not resolve class " % s " : Cyclic reference.) " , type_from_metatype ( p_class - > get_datatype ( ) ) . to_string ( ) ) , p_source ) ;
return ERR_PARSE_ERROR ;
}
if ( ! p_class - > base_type . has_no_type ( ) ) {
// Already resolved.
return OK ;
}
if ( ! parser - > has_class ( p_class ) ) {
String script_path = p_class - > get_datatype ( ) . script_path ;
Ref < GDScriptParserRef > parser_ref = get_parser_for ( script_path ) ;
if ( parser_ref . is_null ( ) ) {
push_error ( vformat ( R " (Could not find script " % s " .) " , script_path ) , p_source ) ;
return ERR_PARSE_ERROR ;
}
Error err = parser_ref - > raise_status ( GDScriptParserRef : : PARSED ) ;
if ( err ) {
push_error ( vformat ( R " (Could not parse script " % s " : %s.) " , script_path , error_names [ err ] ) , p_source ) ;
return ERR_PARSE_ERROR ;
}
ERR_FAIL_COND_V_MSG ( ! parser_ref - > get_parser ( ) - > has_class ( p_class ) , ERR_PARSE_ERROR , R " (Parser bug: Mismatched external parser.) " ) ;
GDScriptAnalyzer * other_analyzer = parser_ref - > get_analyzer ( ) ;
GDScriptParser * other_parser = parser_ref - > get_parser ( ) ;
int error_count = other_parser - > errors . size ( ) ;
other_analyzer - > resolve_class_inheritance ( p_class ) ;
if ( other_parser - > errors . size ( ) > error_count ) {
push_error ( vformat ( R " (Could not resolve inheritance for class " % s " .) " , p_class - > fqcn ) , p_source ) ;
return ERR_PARSE_ERROR ;
}
2020-05-02 00:14:56 +02:00
return OK ;
}
2022-12-11 03:57:35 +01:00
GDScriptParser : : ClassNode * previous_class = parser - > current_class ;
parser - > current_class = p_class ;
2021-09-13 15:51:29 +02:00
if ( p_class - > identifier ) {
StringName class_name = p_class - > identifier - > name ;
2022-12-11 03:57:35 +01:00
if ( GDScriptParser : : get_builtin_type ( class_name ) < Variant : : VARIANT_MAX ) {
push_error ( vformat ( R " (Class " % s " hides a built-in type.) " , class_name ) , p_class - > identifier ) ;
} else if ( class_exists ( class_name ) ) {
2021-09-13 15:51:29 +02:00
push_error ( vformat ( R " (Class " % s " hides a native class.) " , class_name ) , p_class - > identifier ) ;
} else if ( ScriptServer : : is_global_class ( class_name ) & & ( ScriptServer : : get_global_class_path ( class_name ) ! = parser - > script_path | | p_class ! = parser - > head ) ) {
push_error ( vformat ( R " (Class " % s " hides a global script class.) " , class_name ) , p_class - > identifier ) ;
} else if ( ProjectSettings : : get_singleton ( ) - > has_autoload ( class_name ) & & ProjectSettings : : get_singleton ( ) - > get_autoload ( class_name ) . is_singleton ) {
push_error ( vformat ( R " (Class " % s " hides an autoload singleton.) " , class_name ) , p_class - > identifier ) ;
}
}
2022-12-11 03:57:35 +01:00
GDScriptParser : : DataType resolving_datatype ;
resolving_datatype . kind = GDScriptParser : : DataType : : RESOLVING ;
p_class - > base_type = resolving_datatype ;
2020-06-11 00:53:25 +02:00
// Set datatype for class.
GDScriptParser : : DataType class_type ;
class_type . is_constant = true ;
class_type . is_meta_type = true ;
class_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
class_type . kind = GDScriptParser : : DataType : : CLASS ;
class_type . class_type = p_class ;
2020-07-16 03:02:44 +02:00
class_type . script_path = parser - > script_path ;
2021-10-06 17:01:34 +02:00
class_type . builtin_type = Variant : : OBJECT ;
2020-06-11 00:53:25 +02:00
p_class - > set_datatype ( class_type ) ;
2022-12-11 03:57:35 +01:00
GDScriptParser : : DataType result ;
2020-05-02 00:14:56 +02:00
if ( ! p_class - > extends_used ) {
result . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
result . kind = GDScriptParser : : DataType : : NATIVE ;
2022-02-06 14:12:19 +01:00
result . native_type = SNAME ( " RefCounted " ) ;
2020-05-02 00:14:56 +02:00
} else {
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
GDScriptParser : : DataType base ;
2020-06-11 00:53:25 +02:00
int extends_index = 0 ;
2020-12-15 13:04:21 +01:00
if ( ! p_class - > extends_path . is_empty ( ) ) {
2021-07-14 17:23:54 +02:00
if ( p_class - > extends_path . is_relative_path ( ) ) {
2022-08-30 02:34:01 +02:00
p_class - > extends_path = class_type . script_path . get_base_dir ( ) . path_join ( p_class - > extends_path ) . simplify_path ( ) ;
2021-07-14 17:23:54 +02:00
}
2022-09-29 11:53:28 +02:00
Ref < GDScriptParserRef > ext_parser = get_parser_for ( p_class - > extends_path ) ;
if ( ext_parser . is_null ( ) ) {
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (Could not resolve super class path " % s " .) " , p_class - > extends_path ) , p_class ) ;
return ERR_PARSE_ERROR ;
}
2022-12-11 03:57:35 +01:00
Error err = ext_parser - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2020-06-11 00:53:25 +02:00
if ( err ! = OK ) {
push_error ( vformat ( R " (Could not resolve super class inheritance from " % s " .) " , p_class - > extends_path ) , p_class ) ;
return err ;
}
2022-09-29 11:53:28 +02:00
base = ext_parser - > get_parser ( ) - > head - > get_datatype ( ) ;
2020-05-02 00:14:56 +02:00
} else {
2020-12-15 13:04:21 +01:00
if ( p_class - > extends . is_empty ( ) ) {
2022-03-12 07:33:11 +01:00
push_error ( " Could not resolve an empty super class path. " , p_class ) ;
2020-07-16 03:02:44 +02:00
return ERR_PARSE_ERROR ;
}
2020-06-11 00:53:25 +02:00
const StringName & name = p_class - > extends [ extends_index + + ] ;
2020-05-02 00:14:56 +02:00
base . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
if ( ScriptServer : : is_global_class ( name ) ) {
2020-06-11 00:53:25 +02:00
String base_path = ScriptServer : : get_global_class_path ( name ) ;
2020-07-16 03:02:44 +02:00
if ( base_path = = parser - > script_path ) {
base = parser - > head - > get_datatype ( ) ;
} else {
2022-09-29 11:53:28 +02:00
Ref < GDScriptParserRef > base_parser = get_parser_for ( base_path ) ;
if ( base_parser . is_null ( ) ) {
2020-07-16 03:02:44 +02:00
push_error ( vformat ( R " (Could not resolve super class " % s " .) " , name ) , p_class ) ;
return ERR_PARSE_ERROR ;
}
2020-06-11 00:53:25 +02:00
2022-12-11 03:57:35 +01:00
Error err = base_parser - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2020-07-16 03:02:44 +02:00
if ( err ! = OK ) {
push_error ( vformat ( R " (Could not resolve super class inheritance from " % s " .) " , name ) , p_class ) ;
return err ;
}
2022-09-29 11:53:28 +02:00
base = base_parser - > get_parser ( ) - > head - > get_datatype ( ) ;
2020-05-02 00:14:56 +02:00
}
2020-06-11 00:53:25 +02:00
} else if ( ProjectSettings : : get_singleton ( ) - > has_autoload ( name ) & & ProjectSettings : : get_singleton ( ) - > get_autoload ( name ) . is_singleton ) {
const ProjectSettings : : AutoloadInfo & info = ProjectSettings : : get_singleton ( ) - > get_autoload ( name ) ;
2022-12-11 03:57:35 +01:00
if ( info . path . get_extension ( ) . to_lower ( ) ! = GDScriptLanguage : : get_singleton ( ) - > get_extension ( ) ) {
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (Singleton %s is not a GDScript.) " , info . name ) , p_class ) ;
return ERR_PARSE_ERROR ;
}
2022-09-29 11:53:28 +02:00
Ref < GDScriptParserRef > info_parser = get_parser_for ( info . path ) ;
if ( info_parser . is_null ( ) ) {
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (Could not parse singleton from " % s " .) " , info . path ) , p_class ) ;
return ERR_PARSE_ERROR ;
}
2022-12-11 03:57:35 +01:00
Error err = info_parser - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2020-06-11 00:53:25 +02:00
if ( err ! = OK ) {
push_error ( vformat ( R " (Could not resolve super class inheritance from " % s " .) " , name ) , p_class ) ;
return err ;
}
2022-12-11 03:57:35 +01:00
base = info_parser - > get_parser ( ) - > head - > get_datatype ( ) ;
2021-08-17 15:06:54 +02:00
} else if ( class_exists ( name ) & & ClassDB : : can_instantiate ( name ) ) {
2020-06-11 00:53:25 +02:00
base . kind = GDScriptParser : : DataType : : NATIVE ;
base . native_type = name ;
2020-05-02 00:14:56 +02:00
} else {
2020-06-11 00:53:25 +02:00
// Look for other classes in script.
bool found = false ;
2022-12-04 22:55:40 +01:00
List < GDScriptParser : : ClassNode * > script_classes ;
get_class_node_current_scope_classes ( p_class , & script_classes ) ;
for ( GDScriptParser : : ClassNode * look_class : script_classes ) {
2020-07-16 03:02:44 +02:00
if ( look_class - > identifier & & look_class - > identifier - > name = = name ) {
if ( ! look_class - > get_datatype ( ) . is_set ( ) ) {
2022-12-11 03:57:35 +01:00
Error err = resolve_class_inheritance ( look_class , p_class ) ;
2020-07-16 03:02:44 +02:00
if ( err ) {
return err ;
}
}
base = look_class - > get_datatype ( ) ;
found = true ;
break ;
}
2022-12-11 03:57:35 +01:00
if ( look_class - > has_member ( name ) ) {
resolve_class_member ( look_class , name , p_class ) ;
base = look_class - > get_member ( name ) . get_datatype ( ) ;
2020-06-11 00:53:25 +02:00
found = true ;
break ;
}
2020-05-02 00:14:56 +02:00
}
2020-06-11 00:53:25 +02:00
if ( ! found ) {
push_error ( vformat ( R " (Could not find base class " % s " .) " , name ) , p_class ) ;
return ERR_PARSE_ERROR ;
}
}
}
for ( int index = extends_index ; index < p_class - > extends . size ( ) ; index + + ) {
if ( base . kind ! = GDScriptParser : : DataType : : CLASS ) {
push_error ( R " (Super type " % s " is not a GDScript. Cannot get nested types.) " , p_class ) ;
return ERR_PARSE_ERROR ;
}
// TODO: Extends could use identifier nodes. That way errors can be pointed out properly and it can be used here.
GDScriptParser : : IdentifierNode * id = parser - > alloc_node < GDScriptParser : : IdentifierNode > ( ) ;
id - > name = p_class - > extends [ index ] ;
reduce_identifier_from_base ( id , & base ) ;
GDScriptParser : : DataType id_type = id - > get_datatype ( ) ;
if ( ! id_type . is_set ( ) ) {
push_error ( vformat ( R " (Could not find type " % s " under base " % s " .) " , id - > name , base . to_string ( ) ) , p_class ) ;
2020-05-02 00:14:56 +02:00
}
2020-06-11 00:53:25 +02:00
base = id_type ;
2020-05-02 00:14:56 +02:00
}
result = base ;
}
2022-12-11 03:57:35 +01:00
if ( ! result . is_set ( ) | | result . has_no_type ( ) ) {
2020-05-02 00:14:56 +02:00
// TODO: More specific error messages.
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (Could not resolve inheritance for class " % s " .) " , p_class - > identifier = = nullptr ? " <main> " : p_class - > identifier - > name ) , p_class ) ;
2020-05-02 00:14:56 +02:00
return ERR_PARSE_ERROR ;
}
2020-08-26 21:01:08 +02:00
// Check for cyclic inheritance.
const GDScriptParser : : ClassNode * base_class = result . class_type ;
while ( base_class ) {
if ( base_class - > fqcn = = p_class - > fqcn ) {
push_error ( " Cyclic inheritance. " , p_class ) ;
return ERR_PARSE_ERROR ;
}
base_class = base_class - > base_type . class_type ;
}
2020-06-11 00:53:25 +02:00
p_class - > base_type = result ;
class_type . native_type = result . native_type ;
p_class - > set_datatype ( class_type ) ;
2020-05-02 00:14:56 +02:00
2022-12-11 03:57:35 +01:00
parser - > current_class = previous_class ;
return OK ;
}
Error GDScriptAnalyzer : : resolve_class_inheritance ( GDScriptParser : : ClassNode * p_class , bool p_recursive ) {
Error err = resolve_class_inheritance ( p_class ) ;
if ( err ) {
return err ;
}
2020-05-02 00:14:56 +02:00
if ( p_recursive ) {
for ( int i = 0 ; i < p_class - > members . size ( ) ; i + + ) {
if ( p_class - > members [ i ] . type = = GDScriptParser : : ClassNode : : Member : : CLASS ) {
2022-12-11 03:57:35 +01:00
err = resolve_class_inheritance ( p_class - > members [ i ] . m_class , true ) ;
2020-08-26 21:01:08 +02:00
if ( err ) {
return err ;
}
2020-05-02 00:14:56 +02:00
}
}
}
return OK ;
}
2020-06-12 00:31:28 +02:00
GDScriptParser : : DataType GDScriptAnalyzer : : resolve_datatype ( GDScriptParser : : TypeNode * p_type ) {
2022-12-11 03:57:35 +01:00
GDScriptParser : : DataType bad_type ;
bad_type . kind = GDScriptParser : : DataType : : VARIANT ;
bad_type . type_source = GDScriptParser : : DataType : : INFERRED ;
2020-05-02 00:14:56 +02:00
if ( p_type = = nullptr ) {
2022-12-11 03:57:35 +01:00
return bad_type ;
}
if ( p_type - > get_datatype ( ) . is_resolving ( ) ) {
push_error ( R " (Could not resolve datatype: Cyclic reference.) " , p_type ) ;
return bad_type ;
}
if ( ! p_type - > get_datatype ( ) . has_no_type ( ) ) {
return p_type - > get_datatype ( ) ;
2020-05-02 00:14:56 +02:00
}
2022-12-11 03:57:35 +01:00
GDScriptParser : : DataType resolving_datatype ;
resolving_datatype . kind = GDScriptParser : : DataType : : RESOLVING ;
p_type - > set_datatype ( resolving_datatype ) ;
GDScriptParser : : DataType result ;
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
2020-06-11 00:53:25 +02:00
result . builtin_type = Variant : : OBJECT ;
2020-12-15 13:04:21 +01:00
if ( p_type - > type_chain . is_empty ( ) ) {
2020-05-02 00:14:56 +02:00
// void.
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = Variant : : NIL ;
2020-06-12 00:31:28 +02:00
p_type - > set_datatype ( result ) ;
2020-05-02 00:14:56 +02:00
return result ;
}
2020-06-11 00:53:25 +02:00
StringName first = p_type - > type_chain [ 0 ] - > name ;
2020-05-02 00:14:56 +02:00
2022-02-06 14:12:19 +01:00
if ( first = = SNAME ( " Variant " ) ) {
2020-06-12 00:31:28 +02:00
if ( p_type - > type_chain . size ( ) > 1 ) {
2022-12-11 03:57:35 +01:00
// TODO: Variant does actually have a nested Type though.
push_error ( R " (Variant doesn't contain nested types.) " , p_type - > type_chain [ 1 ] ) ;
return bad_type ;
2020-06-12 00:31:28 +02:00
}
2022-12-11 03:57:35 +01:00
result . kind = GDScriptParser : : DataType : : VARIANT ;
} else if ( first = = SNAME ( " Object " ) ) {
// Object is treated like a native type, not a built-in.
2020-07-26 04:30:55 +02:00
result . kind = GDScriptParser : : DataType : : NATIVE ;
2022-02-06 14:12:19 +01:00
result . native_type = SNAME ( " Object " ) ;
2022-12-11 03:57:35 +01:00
} else if ( GDScriptParser : : get_builtin_type ( first ) < Variant : : VARIANT_MAX ) {
2020-05-02 00:14:56 +02:00
// Built-in types.
2020-06-11 00:53:25 +02:00
if ( p_type - > type_chain . size ( ) > 1 ) {
push_error ( R " (Built-in types don't contain nested types.) " , p_type - > type_chain [ 1 ] ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-05-02 00:14:56 +02:00
}
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = GDScriptParser : : get_builtin_type ( first ) ;
2021-03-09 16:32:35 +01:00
if ( result . builtin_type = = Variant : : ARRAY ) {
2023-01-12 00:03:53 +01:00
GDScriptParser : : DataType container_type = type_from_metatype ( resolve_datatype ( p_type - > container_type ) ) ;
2021-03-09 16:32:35 +01:00
if ( container_type . kind ! = GDScriptParser : : DataType : : VARIANT ) {
result . set_container_element_type ( container_type ) ;
}
}
2020-06-11 00:53:25 +02:00
} else if ( class_exists ( first ) ) {
2020-05-02 00:14:56 +02:00
// Native engine classes.
result . kind = GDScriptParser : : DataType : : NATIVE ;
result . native_type = first ;
} else if ( ScriptServer : : is_global_class ( first ) ) {
2020-07-16 03:02:44 +02:00
if ( parser - > script_path = = ScriptServer : : get_global_class_path ( first ) ) {
result = parser - > head - > get_datatype ( ) ;
} else {
2022-04-25 06:33:18 +02:00
String path = ScriptServer : : get_global_class_path ( first ) ;
String ext = path . get_extension ( ) ;
if ( ext = = GDScriptLanguage : : get_singleton ( ) - > get_extension ( ) ) {
Ref < GDScriptParserRef > ref = get_parser_for ( path ) ;
2022-12-11 03:57:35 +01:00
if ( ! ref . is_valid ( ) | | ref - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ! = OK ) {
2022-04-25 06:33:18 +02:00
push_error ( vformat ( R " (Could not parse global class " % s " from " % s " .) " , first , ScriptServer : : get_global_class_path ( first ) ) , p_type ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2022-04-25 06:33:18 +02:00
}
result = ref - > get_parser ( ) - > head - > get_datatype ( ) ;
} else {
2023-01-12 00:03:53 +01:00
result = make_script_meta_type ( ResourceLoader : : load ( path , " Script " ) ) ;
2020-07-16 03:02:44 +02:00
}
2020-06-11 00:53:25 +02:00
}
} else if ( ProjectSettings : : get_singleton ( ) - > has_autoload ( first ) & & ProjectSettings : : get_singleton ( ) - > get_autoload ( first ) . is_singleton ) {
const ProjectSettings : : AutoloadInfo & autoload = ProjectSettings : : get_singleton ( ) - > get_autoload ( first ) ;
Ref < GDScriptParserRef > ref = get_parser_for ( autoload . path ) ;
2022-12-11 03:57:35 +01:00
if ( ref - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ! = OK ) {
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (Could not parse singleton " % s " from " % s " .) " , first , autoload . path ) , p_type ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-06-11 00:53:25 +02:00
}
result = ref - > get_parser ( ) - > head - > get_datatype ( ) ;
2021-08-17 15:06:54 +02:00
} else if ( ClassDB : : has_enum ( parser - > current_class - > base_type . native_type , first ) ) {
2020-06-12 02:49:58 +02:00
// Native enum in current class.
2022-12-04 04:02:03 +01:00
result = make_native_enum_type ( first , parser - > current_class - > base_type . native_type ) ;
2020-05-02 00:14:56 +02:00
} else {
// Classes in current scope.
2022-12-04 22:55:40 +01:00
List < GDScriptParser : : ClassNode * > script_classes ;
2022-12-04 04:02:03 +01:00
bool found = false ;
2022-12-04 22:55:40 +01:00
get_class_node_current_scope_classes ( parser - > current_class , & script_classes ) ;
for ( GDScriptParser : : ClassNode * script_class : script_classes ) {
2022-12-04 04:02:03 +01:00
if ( found ) {
break ;
}
2020-07-16 03:02:44 +02:00
if ( script_class - > identifier & & script_class - > identifier - > name = = first ) {
result = script_class - > get_datatype ( ) ;
break ;
}
2020-05-02 00:14:56 +02:00
if ( script_class - > members_indices . has ( first ) ) {
2022-12-11 03:57:35 +01:00
resolve_class_member ( script_class , first , p_type ) ;
GDScriptParser : : ClassNode : : Member member = script_class - > get_member ( first ) ;
2020-05-02 00:14:56 +02:00
switch ( member . type ) {
case GDScriptParser : : ClassNode : : Member : : CLASS :
2022-12-11 03:57:35 +01:00
result = member . get_datatype ( ) ;
2022-12-04 04:02:03 +01:00
found = true ;
2020-05-02 00:14:56 +02:00
break ;
2020-06-12 02:49:58 +02:00
case GDScriptParser : : ClassNode : : Member : : ENUM :
2022-12-11 03:57:35 +01:00
result = member . get_datatype ( ) ;
2022-12-04 04:02:03 +01:00
found = true ;
2020-06-12 02:49:58 +02:00
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : ClassNode : : Member : : CONSTANT :
2022-12-11 03:57:35 +01:00
if ( member . get_datatype ( ) . is_meta_type ) {
result = member . get_datatype ( ) ;
2022-12-04 04:02:03 +01:00
found = true ;
2020-06-11 00:53:25 +02:00
break ;
2021-05-26 14:23:17 +02:00
} else if ( Ref < Script > ( member . constant - > initializer - > reduced_value ) . is_valid ( ) ) {
Ref < GDScript > gdscript = member . constant - > initializer - > reduced_value ;
if ( gdscript . is_valid ( ) ) {
2022-12-01 11:20:42 +01:00
Ref < GDScriptParserRef > ref = get_parser_for ( gdscript - > get_script_path ( ) ) ;
2022-12-11 03:57:35 +01:00
if ( ref - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ! = OK ) {
2022-12-01 11:20:42 +01:00
push_error ( vformat ( R " (Could not parse script from " % s " .) " , gdscript - > get_script_path ( ) ) , p_type ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2021-05-26 14:23:17 +02:00
}
result = ref - > get_parser ( ) - > head - > get_datatype ( ) ;
} else {
2023-01-12 00:03:53 +01:00
result = make_script_meta_type ( member . constant - > initializer - > reduced_value ) ;
2021-05-26 14:23:17 +02:00
}
2022-12-04 04:02:03 +01:00
found = true ;
2021-05-26 14:23:17 +02:00
break ;
2020-06-11 00:53:25 +02:00
}
[[fallthrough]] ;
2020-05-02 00:14:56 +02:00
default :
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " ( " % s " is a %s but does not contain a type.) " , first , member . get_type_name ( ) ) , p_type ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-05-02 00:14:56 +02:00
}
}
}
2020-06-12 02:49:58 +02:00
}
if ( ! result . is_set ( ) ) {
push_error ( vformat ( R " ( " % s " was not found in the current scope.) " , first ) , p_type ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-05-02 00:14:56 +02:00
}
2020-06-11 00:53:25 +02:00
if ( p_type - > type_chain . size ( ) > 1 ) {
2020-06-12 02:49:58 +02:00
if ( result . kind = = GDScriptParser : : DataType : : CLASS ) {
2020-06-11 00:53:25 +02:00
for ( int i = 1 ; i < p_type - > type_chain . size ( ) ; i + + ) {
GDScriptParser : : DataType base = result ;
reduce_identifier_from_base ( p_type - > type_chain [ i ] , & base ) ;
result = p_type - > type_chain [ i ] - > get_datatype ( ) ;
if ( ! result . is_set ( ) ) {
push_error ( vformat ( R " (Could not find type " % s " under base " % s " .) " , p_type - > type_chain [ i ] - > name , base . to_string ( ) ) , p_type - > type_chain [ 1 ] ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-06-11 00:53:25 +02:00
} else if ( ! result . is_meta_type ) {
push_error ( vformat ( R " (Member " % s " under base " % s " is not a valid type.) " , p_type - > type_chain [ i ] - > name , base . to_string ( ) ) , p_type - > type_chain [ 1 ] ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-06-11 00:53:25 +02:00
}
}
2020-06-12 02:49:58 +02:00
} else if ( result . kind = = GDScriptParser : : DataType : : NATIVE ) {
// Only enums allowed for native.
2022-12-04 04:02:03 +01:00
if ( ClassDB : : has_enum ( result . native_type , p_type - > type_chain [ 1 ] - > name ) ) {
if ( p_type - > type_chain . size ( ) > 2 ) {
push_error ( R " (Enums cannot contain nested types.) " , p_type - > type_chain [ 2 ] ) ;
return bad_type ;
} else {
result = make_native_enum_type ( p_type - > type_chain [ 1 ] - > name , result . native_type ) ;
}
} else {
push_error ( vformat ( R " (Could not find type " % s " in " % s " .) " , p_type - > type_chain [ 1 ] - > name , first ) , p_type - > type_chain [ 1 ] ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-06-12 02:49:58 +02:00
}
} else {
push_error ( vformat ( R " (Could not find nested type " % s " under base " % s " .) " , p_type - > type_chain [ 1 ] - > name , result . to_string ( ) ) , p_type - > type_chain [ 1 ] ) ;
2022-12-11 03:57:35 +01:00
return bad_type ;
2020-06-11 00:53:25 +02:00
}
2020-05-02 00:14:56 +02:00
}
2021-03-09 16:32:35 +01:00
if ( result . builtin_type ! = Variant : : ARRAY & & p_type - > container_type ! = nullptr ) {
push_error ( " Only arrays can specify the collection element type. " , p_type ) ;
}
2020-06-12 00:31:28 +02:00
p_type - > set_datatype ( result ) ;
2020-05-02 00:14:56 +02:00
return result ;
}
2022-12-11 03:57:35 +01:00
void GDScriptAnalyzer : : resolve_class_member ( GDScriptParser : : ClassNode * p_class , StringName p_name , const GDScriptParser : : Node * p_source ) {
ERR_FAIL_COND ( ! p_class - > has_member ( p_name ) ) ;
resolve_class_member ( p_class , p_class - > members_indices [ p_name ] , p_source ) ;
}
void GDScriptAnalyzer : : resolve_class_member ( GDScriptParser : : ClassNode * p_class , int p_index , const GDScriptParser : : Node * p_source ) {
ERR_FAIL_INDEX ( p_index , p_class - > members . size ( ) ) ;
GDScriptParser : : ClassNode : : Member & member = p_class - > members . write [ p_index ] ;
if ( p_source = = nullptr & & parser - > has_class ( p_class ) ) {
p_source = member . get_source_node ( ) ;
}
if ( member . get_datatype ( ) . is_resolving ( ) ) {
push_error ( vformat ( R " (Could not resolve member " % s " : Cyclic reference.) " , member . get_name ( ) ) , p_source ) ;
return ;
}
if ( member . get_datatype ( ) . is_set ( ) ) {
return ;
}
if ( ! parser - > has_class ( p_class ) ) {
String script_path = p_class - > get_datatype ( ) . script_path ;
Ref < GDScriptParserRef > parser_ref = get_parser_for ( script_path ) ;
if ( parser_ref . is_null ( ) ) {
push_error ( vformat ( R " (Could not find script " % s " (While resolving " % s " ).) " , script_path , member . get_name ( ) ) , p_source ) ;
return ;
}
Error err = parser_ref - > raise_status ( GDScriptParserRef : : PARSED ) ;
if ( err ) {
push_error ( vformat ( R " (Could not resolve script " % s " : %s (While resolving " % s " ).) " , script_path , error_names [ err ] , member . get_name ( ) ) , p_source ) ;
return ;
}
ERR_FAIL_COND_MSG ( ! parser_ref - > get_parser ( ) - > has_class ( p_class ) , R " (Parser bug: Mismatched external parser.) " ) ;
GDScriptAnalyzer * other_analyzer = parser_ref - > get_analyzer ( ) ;
GDScriptParser * other_parser = parser_ref - > get_parser ( ) ;
int error_count = other_parser - > errors . size ( ) ;
other_analyzer - > resolve_class_member ( p_class , p_index ) ;
if ( other_parser - > errors . size ( ) > error_count ) {
push_error ( vformat ( R " (Could not resolve member " % s " .) " , member . get_name ( ) ) , p_source ) ;
}
2020-06-11 00:53:25 +02:00
return ;
}
2022-12-11 03:57:35 +01:00
// If it's already resolving, that's ok.
if ( ! p_class - > base_type . is_resolving ( ) ) {
Error err = resolve_class_inheritance ( p_class ) ;
if ( err ) {
return ;
}
}
2020-06-11 00:53:25 +02:00
2020-05-02 00:14:56 +02:00
GDScriptParser : : ClassNode * previous_class = parser - > current_class ;
parser - > current_class = p_class ;
2022-12-11 03:57:35 +01:00
GDScriptParser : : DataType resolving_datatype ;
resolving_datatype . kind = GDScriptParser : : DataType : : RESOLVING ;
2020-05-02 00:14:56 +02:00
2022-12-11 03:57:35 +01:00
{
2020-05-02 00:14:56 +02:00
switch ( member . type ) {
case GDScriptParser : : ClassNode : : Member : : VARIABLE : {
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . variable - > identifier - > name , member . variable ) ;
2022-12-11 03:57:35 +01:00
member . variable - > set_datatype ( resolving_datatype ) ;
2022-12-22 21:43:36 +01:00
resolve_variable ( member . variable , false ) ;
2021-03-17 14:57:30 +01:00
// Apply annotations.
2021-07-16 05:45:57 +02:00
for ( GDScriptParser : : AnnotationNode * & E : member . variable - > annotations ) {
E - > apply ( parser , member . variable ) ;
2020-05-02 00:14:56 +02:00
}
2020-06-11 00:53:25 +02:00
} break ;
case GDScriptParser : : ClassNode : : Member : : CONSTANT : {
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . constant - > identifier - > name , member . constant ) ;
2022-12-11 03:57:35 +01:00
member . constant - > set_datatype ( resolving_datatype ) ;
2022-12-22 21:43:36 +01:00
resolve_constant ( member . constant , false ) ;
2021-03-17 14:57:30 +01:00
// Apply annotations.
2021-07-16 05:45:57 +02:00
for ( GDScriptParser : : AnnotationNode * & E : member . constant - > annotations ) {
E - > apply ( parser , member . constant ) ;
2021-03-17 14:57:30 +01:00
}
2020-06-11 00:53:25 +02:00
} break ;
case GDScriptParser : : ClassNode : : Member : : SIGNAL : {
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . signal - > identifier - > name , member . signal ) ;
2022-12-11 03:57:35 +01:00
member . signal - > set_datatype ( resolving_datatype ) ;
2022-12-16 04:45:46 +01:00
// This is the _only_ way to declare a signal. Therefore, we can generate its
// MethodInfo inline so it's a tiny bit more efficient.
MethodInfo mi = MethodInfo ( member . signal - > identifier - > name ) ;
2020-06-11 00:53:25 +02:00
for ( int j = 0 ; j < member . signal - > parameters . size ( ) ; j + + ) {
2022-12-16 04:45:46 +01:00
GDScriptParser : : ParameterNode * param = member . signal - > parameters [ j ] ;
2023-01-12 00:03:53 +01:00
GDScriptParser : : DataType param_type = type_from_metatype ( resolve_datatype ( param - > datatype_specifier ) ) ;
2022-12-16 04:45:46 +01:00
param - > set_datatype ( param_type ) ;
mi . arguments . push_back ( PropertyInfo ( param_type . builtin_type , param - > identifier - > name ) ) ;
// TODO: add signal parameter default values
2020-06-11 00:53:25 +02:00
}
2022-12-16 04:45:46 +01:00
member . signal - > set_datatype ( make_signal_type ( mi ) ) ;
2021-03-17 14:57:30 +01:00
// Apply annotations.
2021-07-16 05:45:57 +02:00
for ( GDScriptParser : : AnnotationNode * & E : member . signal - > annotations ) {
E - > apply ( parser , member . signal ) ;
2021-03-17 14:57:30 +01:00
}
2020-06-11 00:53:25 +02:00
} break ;
case GDScriptParser : : ClassNode : : Member : : ENUM : {
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . m_enum - > identifier - > name , member . m_enum ) ;
2022-12-11 03:57:35 +01:00
member . m_enum - > set_datatype ( resolving_datatype ) ;
2022-12-04 04:02:03 +01:00
GDScriptParser : : DataType enum_type = make_enum_type ( member . m_enum - > identifier - > name , p_class - > fqcn , true ) ;
2020-06-12 02:49:58 +02:00
2022-12-11 03:57:35 +01:00
const GDScriptParser : : EnumNode * prev_enum = current_enum ;
2020-08-18 22:44:20 +02:00
current_enum = member . m_enum ;
2022-12-27 04:06:11 +01:00
Dictionary dictionary ;
2020-07-16 03:02:44 +02:00
for ( int j = 0 ; j < member . m_enum - > values . size ( ) ; j + + ) {
2020-08-18 22:44:20 +02:00
GDScriptParser : : EnumNode : : Value & element = member . m_enum - > values . write [ j ] ;
if ( element . custom_value ) {
reduce_expression ( element . custom_value ) ;
if ( ! element . custom_value - > is_constant ) {
push_error ( R " (Enum values must be constant.) " , element . custom_value ) ;
} else if ( element . custom_value - > reduced_value . get_type ( ) ! = Variant : : INT ) {
push_error ( R " (Enum values must be integers.) " , element . custom_value ) ;
} else {
element . value = element . custom_value - > reduced_value ;
element . resolved = true ;
}
} else {
if ( element . index > 0 ) {
element . value = element . parent_enum - > values [ element . index - 1 ] . value + 1 ;
} else {
element . value = 0 ;
}
element . resolved = true ;
}
enum_type . enum_values [ element . identifier - > name ] = element . value ;
2022-12-27 04:06:11 +01:00
dictionary [ String ( element . identifier - > name ) ] = element . value ;
2020-06-12 02:49:58 +02:00
}
2020-06-11 00:53:25 +02:00
2022-12-11 03:57:35 +01:00
current_enum = prev_enum ;
2020-08-18 22:44:20 +02:00
2022-12-04 04:02:03 +01:00
dictionary . set_read_only ( true ) ;
2020-06-11 00:53:25 +02:00
member . m_enum - > set_datatype ( enum_type ) ;
2022-12-27 04:06:11 +01:00
member . m_enum - > dictionary = dictionary ;
2021-03-17 14:57:30 +01:00
// Apply annotations.
2021-07-16 05:45:57 +02:00
for ( GDScriptParser : : AnnotationNode * & E : member . m_enum - > annotations ) {
E - > apply ( parser , member . m_enum ) ;
2021-03-17 14:57:30 +01:00
}
2020-06-11 00:53:25 +02:00
} break ;
case GDScriptParser : : ClassNode : : Member : : FUNCTION :
2022-12-11 03:57:35 +01:00
resolve_function_signature ( member . function , p_source ) ;
2020-05-02 00:14:56 +02:00
break ;
2020-08-18 22:44:20 +02:00
case GDScriptParser : : ClassNode : : Member : : ENUM_VALUE : {
if ( member . enum_value . custom_value ) {
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . enum_value . identifier - > name , member . enum_value . custom_value ) ;
2022-12-11 03:57:35 +01:00
member . enum_value . identifier - > set_datatype ( resolving_datatype ) ;
const GDScriptParser : : EnumNode * prev_enum = current_enum ;
2020-08-18 22:44:20 +02:00
current_enum = member . enum_value . parent_enum ;
reduce_expression ( member . enum_value . custom_value ) ;
2022-12-11 03:57:35 +01:00
current_enum = prev_enum ;
2020-08-18 22:44:20 +02:00
if ( ! member . enum_value . custom_value - > is_constant ) {
push_error ( R " (Enum values must be constant.) " , member . enum_value . custom_value ) ;
} else if ( member . enum_value . custom_value - > reduced_value . get_type ( ) ! = Variant : : INT ) {
push_error ( R " (Enum values must be integers.) " , member . enum_value . custom_value ) ;
} else {
member . enum_value . value = member . enum_value . custom_value - > reduced_value ;
member . enum_value . resolved = true ;
}
} else {
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . enum_value . identifier - > name , member . enum_value . parent_enum ) ;
2020-08-18 22:44:20 +02:00
if ( member . enum_value . index > 0 ) {
2022-12-17 04:46:00 +01:00
const GDScriptParser : : EnumNode : : Value & prev_value = member . enum_value . parent_enum - > values [ member . enum_value . index - 1 ] ;
resolve_class_member ( p_class , prev_value . identifier - > name , member . enum_value . identifier ) ;
member . enum_value . value = prev_value . value + 1 ;
2020-08-18 22:44:20 +02:00
} else {
member . enum_value . value = 0 ;
}
member . enum_value . resolved = true ;
}
2022-12-11 03:57:35 +01:00
2020-08-18 22:44:20 +02:00
// Also update the original references.
2022-12-11 03:57:35 +01:00
member . enum_value . parent_enum - > values . set ( member . enum_value . index , member . enum_value ) ;
2022-12-04 04:02:03 +01:00
member . enum_value . identifier - > set_datatype ( make_enum_type ( UNNAMED_ENUM , p_class - > fqcn , false ) ) ;
2020-08-18 22:44:20 +02:00
} break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : ClassNode : : Member : : CLASS :
2021-08-24 18:49:03 +02:00
check_class_member_name_conflict ( p_class , member . m_class - > identifier - > name , member . m_class ) ;
2022-12-11 03:57:35 +01:00
// If it's already resolving, that's ok.
if ( ! member . m_class - > base_type . is_resolving ( ) ) {
resolve_class_inheritance ( member . m_class , p_source ) ;
}
2021-08-24 18:49:03 +02:00
break ;
2022-07-03 21:30:08 +02:00
case GDScriptParser : : ClassNode : : Member : : GROUP :
// No-op, but needed to silence warnings.
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : ClassNode : : Member : : UNDEFINED :
ERR_PRINT ( " Trying to resolve undefined member. " ) ;
2020-05-02 00:14:56 +02:00
break ;
}
}
2020-06-11 00:53:25 +02:00
2022-12-11 03:57:35 +01:00
parser - > current_class = previous_class ;
}
void GDScriptAnalyzer : : resolve_class_interface ( GDScriptParser : : ClassNode * p_class , const GDScriptParser : : Node * p_source ) {
if ( p_source = = nullptr & & parser - > has_class ( p_class ) ) {
p_source = p_class ;
}
if ( ! p_class - > resolved_interface ) {
if ( ! parser - > has_class ( p_class ) ) {
String script_path = p_class - > get_datatype ( ) . script_path ;
Ref < GDScriptParserRef > parser_ref = get_parser_for ( script_path ) ;
if ( parser_ref . is_null ( ) ) {
push_error ( vformat ( R " (Could not find script " % s " .) " , script_path ) , p_source ) ;
return ;
}
Error err = parser_ref - > raise_status ( GDScriptParserRef : : PARSED ) ;
if ( err ) {
push_error ( vformat ( R " (Could not resolve script " % s " : %s.) " , script_path , error_names [ err ] ) , p_source ) ;
return ;
}
ERR_FAIL_COND_MSG ( ! parser_ref - > get_parser ( ) - > has_class ( p_class ) , R " (Parser bug: Mismatched external parser.) " ) ;
GDScriptAnalyzer * other_analyzer = parser_ref - > get_analyzer ( ) ;
GDScriptParser * other_parser = parser_ref - > get_parser ( ) ;
int error_count = other_parser - > errors . size ( ) ;
other_analyzer - > resolve_class_interface ( p_class ) ;
if ( other_parser - > errors . size ( ) > error_count ) {
push_error ( vformat ( R " (Could not resolve class " % s " .) " , p_class - > fqcn ) , p_source ) ;
}
return ;
}
p_class - > resolved_interface = true ;
if ( resolve_class_inheritance ( p_class ) ! = OK ) {
return ;
}
GDScriptParser : : DataType base_type = p_class - > base_type ;
if ( base_type . kind = = GDScriptParser : : DataType : : CLASS ) {
GDScriptParser : : ClassNode * base_class = base_type . class_type ;
resolve_class_interface ( base_class , p_class ) ;
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
for ( int i = 0 ; i < p_class - > members . size ( ) ; i + + ) {
resolve_class_member ( p_class , i ) ;
}
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
}
2020-06-11 00:53:25 +02:00
2022-12-11 03:57:35 +01:00
void GDScriptAnalyzer : : resolve_class_interface ( GDScriptParser : : ClassNode * p_class , bool p_recursive ) {
resolve_class_interface ( p_class ) ;
if ( p_recursive ) {
for ( int i = 0 ; i < p_class - > members . size ( ) ; i + + ) {
GDScriptParser : : ClassNode : : Member member = p_class - > members [ i ] ;
if ( member . type = = GDScriptParser : : ClassNode : : Member : : CLASS ) {
resolve_class_interface ( member . m_class , true ) ;
}
}
}
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
void GDScriptAnalyzer : : resolve_class_body ( GDScriptParser : : ClassNode * p_class , const GDScriptParser : : Node * p_source ) {
if ( p_source = = nullptr & & parser - > has_class ( p_class ) ) {
p_source = p_class ;
}
2020-06-11 00:53:25 +02:00
if ( p_class - > resolved_body ) {
return ;
}
2022-12-11 03:57:35 +01:00
if ( ! parser - > has_class ( p_class ) ) {
String script_path = p_class - > get_datatype ( ) . script_path ;
Ref < GDScriptParserRef > parser_ref = get_parser_for ( script_path ) ;
if ( parser_ref . is_null ( ) ) {
push_error ( vformat ( R " (Could not find script " % s " .) " , script_path ) , p_source ) ;
return ;
}
Error err = parser_ref - > raise_status ( GDScriptParserRef : : PARSED ) ;
if ( err ) {
push_error ( vformat ( R " (Could not resolve script " % s " : %s.) " , script_path , error_names [ err ] ) , p_source ) ;
return ;
}
ERR_FAIL_COND_MSG ( ! parser_ref - > get_parser ( ) - > has_class ( p_class ) , R " (Parser bug: Mismatched external parser.) " ) ;
GDScriptAnalyzer * other_analyzer = parser_ref - > get_analyzer ( ) ;
GDScriptParser * other_parser = parser_ref - > get_parser ( ) ;
int error_count = other_parser - > errors . size ( ) ;
other_analyzer - > resolve_class_body ( p_class ) ;
if ( other_parser - > errors . size ( ) > error_count ) {
push_error ( vformat ( R " (Could not resolve class " % s " .) " , p_class - > fqcn ) , p_source ) ;
}
return ;
}
2020-06-11 00:53:25 +02:00
p_class - > resolved_body = true ;
GDScriptParser : : ClassNode * previous_class = parser - > current_class ;
parser - > current_class = p_class ;
2022-12-11 03:57:35 +01:00
resolve_class_interface ( p_class , p_source ) ;
GDScriptParser : : DataType base_type = p_class - > base_type ;
if ( base_type . kind = = GDScriptParser : : DataType : : CLASS ) {
GDScriptParser : : ClassNode * base_class = base_type . class_type ;
resolve_class_body ( base_class , p_class ) ;
}
2021-09-06 07:04:43 +02:00
// Do functions and properties now.
2020-06-11 00:53:25 +02:00
for ( int i = 0 ; i < p_class - > members . size ( ) ; i + + ) {
GDScriptParser : : ClassNode : : Member member = p_class - > members [ i ] ;
2021-09-06 07:04:43 +02:00
if ( member . type = = GDScriptParser : : ClassNode : : Member : : FUNCTION ) {
// Apply annotations.
for ( GDScriptParser : : AnnotationNode * & E : member . function - > annotations ) {
E - > apply ( parser , member . function ) ;
}
2022-01-04 13:32:43 +01:00
# ifdef DEBUG_ENABLED
2022-05-19 17:00:06 +02:00
HashSet < uint32_t > previously_ignored = parser - > ignored_warning_codes ;
2022-01-04 13:32:43 +01:00
for ( uint32_t ignored_warning : member . function - > ignored_warnings ) {
parser - > ignored_warning_codes . insert ( ignored_warning ) ;
}
# endif // DEBUG_ENABLED
resolve_function_body ( member . function ) ;
# ifdef DEBUG_ENABLED
parser - > ignored_warning_codes = previously_ignored ;
# endif // DEBUG_ENABLED
2021-09-06 07:04:43 +02:00
} else if ( member . type = = GDScriptParser : : ClassNode : : Member : : VARIABLE & & member . variable - > property ! = GDScriptParser : : VariableNode : : PROP_NONE ) {
if ( member . variable - > property = = GDScriptParser : : VariableNode : : PROP_INLINE ) {
if ( member . variable - > getter ! = nullptr ) {
member . variable - > getter - > set_datatype ( member . variable - > datatype ) ;
2021-03-17 14:57:30 +01:00
2021-09-06 07:04:43 +02:00
resolve_function_body ( member . variable - > getter ) ;
}
if ( member . variable - > setter ! = nullptr ) {
2021-10-12 14:39:23 +02:00
resolve_function_signature ( member . variable - > setter ) ;
2021-09-06 07:04:43 +02:00
if ( member . variable - > setter - > parameters . size ( ) > 0 ) {
member . variable - > setter - > parameters [ 0 ] - > datatype_specifier = member . variable - > datatype_specifier ;
2021-10-12 14:39:23 +02:00
member . variable - > setter - > parameters [ 0 ] - > set_datatype ( member . get_datatype ( ) ) ;
2021-09-06 07:04:43 +02:00
}
resolve_function_body ( member . variable - > setter ) ;
}
}
2021-03-17 14:57:30 +01:00
}
2020-06-11 00:53:25 +02:00
}
2021-09-06 07:04:43 +02:00
// Check unused variables and datatypes of property getters and setters.
2020-06-12 00:31:28 +02:00
for ( int i = 0 ; i < p_class - > members . size ( ) ; i + + ) {
GDScriptParser : : ClassNode : : Member member = p_class - > members [ i ] ;
2021-09-06 07:04:43 +02:00
if ( member . type = = GDScriptParser : : ClassNode : : Member : : VARIABLE ) {
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2022-05-19 17:00:06 +02:00
HashSet < uint32_t > previously_ignored = parser - > ignored_warning_codes ;
2022-01-04 13:32:43 +01:00
for ( uint32_t ignored_warning : member . function - > ignored_warnings ) {
parser - > ignored_warning_codes . insert ( ignored_warning ) ;
}
2021-09-06 07:04:43 +02:00
if ( member . variable - > usages = = 0 & & String ( member . variable - > identifier - > name ) . begins_with ( " _ " ) ) {
parser - > push_warning ( member . variable - > identifier , GDScriptWarning : : UNUSED_PRIVATE_CLASS_VARIABLE , member . variable - > identifier - > name ) ;
}
# endif
if ( member . variable - > property = = GDScriptParser : : VariableNode : : PROP_SETGET ) {
GDScriptParser : : FunctionNode * getter_function = nullptr ;
GDScriptParser : : FunctionNode * setter_function = nullptr ;
bool has_valid_getter = false ;
bool has_valid_setter = false ;
if ( member . variable - > getter_pointer ! = nullptr ) {
if ( p_class - > has_function ( member . variable - > getter_pointer - > name ) ) {
getter_function = p_class - > get_member ( member . variable - > getter_pointer - > name ) . function ;
}
if ( getter_function = = nullptr ) {
push_error ( vformat ( R " (Getter " % s " not found.) " , member . variable - > getter_pointer - > name ) , member . variable ) ;
} else {
2022-10-30 02:38:26 +02:00
GDScriptParser : : DataType return_datatype = getter_function - > datatype ;
if ( getter_function - > return_type ! = nullptr ) {
return_datatype = getter_function - > return_type - > datatype ;
return_datatype . is_meta_type = false ;
}
if ( getter_function - > parameters . size ( ) ! = 0 | | return_datatype . has_no_type ( ) ) {
push_error ( vformat ( R " (Function " % s " cannot be used as getter because of its signature.) " , getter_function - > identifier - > name ) , member . variable ) ;
} else if ( ! is_type_compatible ( member . variable - > datatype , return_datatype , true ) ) {
push_error ( vformat ( R " (Function with return type " % s " cannot be used as getter for a property of type " % s " .) " , return_datatype . to_string ( ) , member . variable - > datatype . to_string ( ) ) , member . variable ) ;
2021-09-06 07:04:43 +02:00
2022-10-30 02:38:26 +02:00
} else {
has_valid_getter = true ;
2021-09-06 07:04:43 +02:00
# ifdef DEBUG_ENABLED
2022-10-30 02:38:26 +02:00
if ( member . variable - > datatype . builtin_type = = Variant : : INT & & return_datatype . builtin_type = = Variant : : FLOAT ) {
parser - > push_warning ( member . variable , GDScriptWarning : : NARROWING_CONVERSION ) ;
}
2021-09-06 07:04:43 +02:00
# endif
2022-10-30 02:38:26 +02:00
}
2021-09-06 07:04:43 +02:00
}
}
if ( member . variable - > setter_pointer ! = nullptr ) {
if ( p_class - > has_function ( member . variable - > setter_pointer - > name ) ) {
setter_function = p_class - > get_member ( member . variable - > setter_pointer - > name ) . function ;
}
if ( setter_function = = nullptr ) {
push_error ( vformat ( R " (Setter " % s " not found.) " , member . variable - > setter_pointer - > name ) , member . variable ) ;
} else if ( setter_function - > parameters . size ( ) ! = 1 ) {
push_error ( vformat ( R " (Function " % s " cannot be used as setter because of its signature.) " , setter_function - > identifier - > name ) , member . variable ) ;
} else if ( ! is_type_compatible ( member . variable - > datatype , setter_function - > parameters [ 0 ] - > datatype , true ) ) {
push_error ( vformat ( R " (Function with argument type " % s " cannot be used as setter for a property of type " % s " .) " , setter_function - > parameters [ 0 ] - > datatype . to_string ( ) , member . variable - > datatype . to_string ( ) ) , member . variable ) ;
} else {
has_valid_setter = true ;
# ifdef DEBUG_ENABLED
2021-10-18 09:01:16 +02:00
if ( member . variable - > datatype . builtin_type = = Variant : : FLOAT & & setter_function - > parameters [ 0 ] - > datatype . builtin_type = = Variant : : INT ) {
2021-09-06 07:04:43 +02:00
parser - > push_warning ( member . variable , GDScriptWarning : : NARROWING_CONVERSION ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2021-09-06 07:04:43 +02:00
}
}
if ( member . variable - > datatype . is_variant ( ) & & has_valid_getter & & has_valid_setter ) {
if ( ! is_type_compatible ( getter_function - > datatype , setter_function - > parameters [ 0 ] - > datatype , true ) ) {
push_error ( vformat ( R " (Getter with type " % s " cannot be used along with setter of type " % s " .) " , getter_function - > datatype . to_string ( ) , setter_function - > parameters [ 0 ] - > datatype . to_string ( ) ) , member . variable ) ;
}
}
2022-01-04 13:32:43 +01:00
# ifdef DEBUG_ENABLED
parser - > ignored_warning_codes = previously_ignored ;
# endif // DEBUG_ENABLED
2021-09-06 07:04:43 +02:00
}
}
2020-06-12 00:31:28 +02:00
}
2022-12-11 03:57:35 +01:00
parser - > current_class = previous_class ;
}
void GDScriptAnalyzer : : resolve_class_body ( GDScriptParser : : ClassNode * p_class , bool p_recursive ) {
resolve_class_body ( p_class ) ;
if ( p_recursive ) {
for ( int i = 0 ; i < p_class - > members . size ( ) ; i + + ) {
GDScriptParser : : ClassNode : : Member member = p_class - > members [ i ] ;
if ( member . type = = GDScriptParser : : ClassNode : : Member : : CLASS ) {
resolve_class_body ( member . m_class , true ) ;
}
}
}
2020-06-11 00:53:25 +02:00
}
2022-10-13 19:31:12 +02:00
void GDScriptAnalyzer : : resolve_node ( GDScriptParser : : Node * p_node , bool p_is_root ) {
2020-06-11 00:53:25 +02:00
ERR_FAIL_COND_MSG ( p_node = = nullptr , " Trying to resolve type of a null node. " ) ;
switch ( p_node - > type ) {
case GDScriptParser : : Node : : NONE :
break ; // Unreachable.
case GDScriptParser : : Node : : CLASS :
2022-12-11 03:57:35 +01:00
if ( OK = = resolve_class_inheritance ( static_cast < GDScriptParser : : ClassNode * > ( p_node ) , true ) ) {
resolve_class_interface ( static_cast < GDScriptParser : : ClassNode * > ( p_node ) , true ) ;
resolve_class_body ( static_cast < GDScriptParser : : ClassNode * > ( p_node ) , true ) ;
}
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : Node : : CONSTANT :
2022-12-22 21:43:36 +01:00
resolve_constant ( static_cast < GDScriptParser : : ConstantNode * > ( p_node ) , true ) ;
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : Node : : FOR :
resolve_for ( static_cast < GDScriptParser : : ForNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : IF :
resolve_if ( static_cast < GDScriptParser : : IfNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : SUITE :
resolve_suite ( static_cast < GDScriptParser : : SuiteNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : VARIABLE :
2022-12-22 21:43:36 +01:00
resolve_variable ( static_cast < GDScriptParser : : VariableNode * > ( p_node ) , true ) ;
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : Node : : WHILE :
resolve_while ( static_cast < GDScriptParser : : WhileNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : ANNOTATION :
resolve_annotation ( static_cast < GDScriptParser : : AnnotationNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : ASSERT :
resolve_assert ( static_cast < GDScriptParser : : AssertNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : MATCH :
resolve_match ( static_cast < GDScriptParser : : MatchNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : MATCH_BRANCH :
resolve_match_branch ( static_cast < GDScriptParser : : MatchBranchNode * > ( p_node ) , nullptr ) ;
break ;
case GDScriptParser : : Node : : PARAMETER :
2020-09-06 09:56:52 +02:00
resolve_parameter ( static_cast < GDScriptParser : : ParameterNode * > ( p_node ) ) ;
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : Node : : PATTERN :
resolve_match_pattern ( static_cast < GDScriptParser : : PatternNode * > ( p_node ) , nullptr ) ;
break ;
case GDScriptParser : : Node : : RETURN :
resolve_return ( static_cast < GDScriptParser : : ReturnNode * > ( p_node ) ) ;
break ;
case GDScriptParser : : Node : : TYPE :
resolve_datatype ( static_cast < GDScriptParser : : TypeNode * > ( p_node ) ) ;
break ;
// Resolving expression is the same as reducing them.
case GDScriptParser : : Node : : ARRAY :
case GDScriptParser : : Node : : ASSIGNMENT :
case GDScriptParser : : Node : : AWAIT :
case GDScriptParser : : Node : : BINARY_OPERATOR :
case GDScriptParser : : Node : : CALL :
case GDScriptParser : : Node : : CAST :
case GDScriptParser : : Node : : DICTIONARY :
case GDScriptParser : : Node : : GET_NODE :
case GDScriptParser : : Node : : IDENTIFIER :
2021-03-26 13:03:16 +01:00
case GDScriptParser : : Node : : LAMBDA :
2020-06-11 00:53:25 +02:00
case GDScriptParser : : Node : : LITERAL :
case GDScriptParser : : Node : : PRELOAD :
case GDScriptParser : : Node : : SELF :
case GDScriptParser : : Node : : SUBSCRIPT :
case GDScriptParser : : Node : : TERNARY_OPERATOR :
case GDScriptParser : : Node : : UNARY_OPERATOR :
2022-10-13 19:31:12 +02:00
reduce_expression ( static_cast < GDScriptParser : : ExpressionNode * > ( p_node ) , p_is_root ) ;
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : Node : : BREAK :
case GDScriptParser : : Node : : BREAKPOINT :
case GDScriptParser : : Node : : CONTINUE :
case GDScriptParser : : Node : : ENUM :
2022-12-28 06:41:03 +01:00
case GDScriptParser : : Node : : FUNCTION :
2020-06-11 00:53:25 +02:00
case GDScriptParser : : Node : : PASS :
case GDScriptParser : : Node : : SIGNAL :
// Nothing to do.
break ;
}
}
void GDScriptAnalyzer : : resolve_annotation ( GDScriptParser : : AnnotationNode * p_annotation ) {
// TODO: Add second validation function for annotations, so they can use checked types.
}
2022-12-28 06:41:03 +01:00
void GDScriptAnalyzer : : resolve_function_signature ( GDScriptParser : : FunctionNode * p_function , const GDScriptParser : : Node * p_source , bool p_is_lambda ) {
2022-12-11 03:57:35 +01:00
if ( p_source = = nullptr ) {
p_source = p_function ;
}
2022-12-28 06:41:03 +01:00
StringName function_name = p_function - > identifier ! = nullptr ? p_function - > identifier - > name : StringName ( ) ;
2022-12-11 03:57:35 +01:00
if ( p_function - > get_datatype ( ) . is_resolving ( ) ) {
2022-12-28 06:41:03 +01:00
push_error ( vformat ( R " (Could not resolve function " % s " : Cyclic reference.) " , function_name ) , p_source ) ;
2022-12-11 03:57:35 +01:00
return ;
}
2020-06-11 00:53:25 +02:00
if ( p_function - > resolved_signature ) {
return ;
}
p_function - > resolved_signature = true ;
GDScriptParser : : FunctionNode * previous_function = parser - > current_function ;
parser - > current_function = p_function ;
2022-12-11 03:57:35 +01:00
GDScriptParser : : DataType prev_datatype = p_function - > get_datatype ( ) ;
GDScriptParser : : DataType resolving_datatype ;
resolving_datatype . kind = GDScriptParser : : DataType : : RESOLVING ;
p_function - > set_datatype ( resolving_datatype ) ;
2022-03-14 15:55:37 +01:00
# ifdef TOOLS_ENABLED
int default_value_count = 0 ;
# endif // TOOLS_ENABLED
2020-06-11 00:53:25 +02:00
for ( int i = 0 ; i < p_function - > parameters . size ( ) ; i + + ) {
2020-09-06 09:56:52 +02:00
resolve_parameter ( p_function - > parameters [ i ] ) ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
if ( p_function - > parameters [ i ] - > usages = = 0 & & ! String ( p_function - > parameters [ i ] - > identifier - > name ) . begins_with ( " _ " ) ) {
2022-12-28 06:41:03 +01:00
parser - > push_warning ( p_function - > parameters [ i ] - > identifier , GDScriptWarning : : UNUSED_PARAMETER , function_name , p_function - > parameters [ i ] - > identifier - > name ) ;
2020-06-12 00:31:28 +02:00
}
is_shadowing ( p_function - > parameters [ i ] - > identifier , " function parameter " ) ;
2020-11-29 03:37:57 +01:00
# endif // DEBUG_ENABLED
# ifdef TOOLS_ENABLED
2022-12-22 21:43:36 +01:00
if ( p_function - > parameters [ i ] - > initializer ) {
2022-03-14 15:55:37 +01:00
default_value_count + + ;
2022-12-22 21:43:36 +01:00
if ( p_function - > parameters [ i ] - > initializer - > is_constant ) {
p_function - > default_arg_values . push_back ( p_function - > parameters [ i ] - > initializer - > reduced_value ) ;
2022-11-10 17:22:52 +01:00
} else {
p_function - > default_arg_values . push_back ( Variant ( ) ) ; // Prevent shift.
2022-03-14 15:55:37 +01:00
}
2020-11-29 03:37:57 +01:00
}
# endif // TOOLS_ENABLED
2020-06-11 00:53:25 +02:00
}
2022-12-28 06:41:03 +01:00
if ( ! p_is_lambda & & function_name = = GDScriptLanguage : : get_singleton ( ) - > strings . _init ) {
2020-07-16 03:02:44 +02:00
// Constructor.
GDScriptParser : : DataType return_type = parser - > current_class - > get_datatype ( ) ;
return_type . is_meta_type = false ;
p_function - > set_datatype ( return_type ) ;
if ( p_function - > return_type ) {
2021-10-03 20:55:54 +02:00
GDScriptParser : : DataType declared_return = resolve_datatype ( p_function - > return_type ) ;
if ( declared_return . kind ! = GDScriptParser : : DataType : : BUILTIN | | declared_return . builtin_type ! = Variant : : NIL ) {
push_error ( " Constructor cannot have an explicit return type. " , p_function - > return_type ) ;
}
2020-07-16 03:02:44 +02:00
}
} else {
2022-05-24 18:40:36 +02:00
if ( p_function - > return_type ! = nullptr ) {
2022-12-04 04:02:03 +01:00
p_function - > set_datatype ( type_from_metatype ( resolve_datatype ( p_function - > return_type ) ) ) ;
2022-05-24 18:40:36 +02:00
} else {
// In case the function is not typed, we can safely assume it's a Variant, so it's okay to mark as "inferred" here.
// It's not "undetected" to not mix up with unknown functions.
GDScriptParser : : DataType return_type ;
return_type . type_source = GDScriptParser : : DataType : : INFERRED ;
return_type . kind = GDScriptParser : : DataType : : VARIANT ;
p_function - > set_datatype ( return_type ) ;
}
2022-03-06 15:09:12 +01:00
2022-03-07 08:01:52 +01:00
# ifdef TOOLS_ENABLED
2022-03-06 15:09:12 +01:00
// Check if the function signature matches the parent. If not it's an error since it breaks polymorphism.
// Not for the constructor which can vary in signature.
GDScriptParser : : DataType base_type = parser - > current_class - > base_type ;
GDScriptParser : : DataType parent_return_type ;
List < GDScriptParser : : DataType > parameters_types ;
int default_par_count = 0 ;
bool is_static = false ;
bool is_vararg = false ;
2022-12-28 06:41:03 +01:00
if ( ! p_is_lambda & & get_function_signature ( p_function , false , base_type , function_name , parent_return_type , parameters_types , default_par_count , is_static , is_vararg ) ) {
2022-03-06 15:09:12 +01:00
bool valid = p_function - > is_static = = is_static ;
valid = valid & & parent_return_type = = p_function - > get_datatype ( ) ;
int par_count_diff = p_function - > parameters . size ( ) - parameters_types . size ( ) ;
valid = valid & & par_count_diff > = 0 ;
2022-03-14 15:55:37 +01:00
valid = valid & & default_value_count > = default_par_count + par_count_diff ;
2022-03-06 15:09:12 +01:00
int i = 0 ;
for ( const GDScriptParser : : DataType & par_type : parameters_types ) {
valid = valid & & par_type = = p_function - > parameters [ i + + ] - > get_datatype ( ) ;
}
if ( ! valid ) {
// Compute parent signature as a string to show in the error message.
2022-12-04 04:02:03 +01:00
String parent_signature = String ( function_name ) + " ( " ;
2022-03-06 15:09:12 +01:00
int j = 0 ;
for ( const GDScriptParser : : DataType & par_type : parameters_types ) {
if ( j > 0 ) {
parent_signature + = " , " ;
}
String parameter = par_type . to_string ( ) ;
if ( parameter = = " null " ) {
parameter = " Variant " ;
}
parent_signature + = parameter ;
if ( j = = parameters_types . size ( ) - default_par_count ) {
parent_signature + = " = default " ;
}
j + + ;
}
2022-12-01 01:46:01 +01:00
parent_signature + = " ) -> " ;
const String return_type = parent_return_type . is_hard_type ( ) ? parent_return_type . to_string ( ) : " Variant " ;
if ( return_type = = " null " ) {
parent_signature + = " void " ;
} else {
parent_signature + = return_type ;
}
2022-03-06 15:09:12 +01:00
push_error ( vformat ( R " (The function signature doesn't match the parent. Parent signature is " % s " .) " , parent_signature ) , p_function ) ;
}
}
2022-03-14 15:55:37 +01:00
# endif // TOOLS_ENABLED
2020-07-16 03:02:44 +02:00
}
2020-06-11 00:53:25 +02:00
2022-12-11 03:57:35 +01:00
if ( p_function - > get_datatype ( ) . is_resolving ( ) ) {
p_function - > set_datatype ( prev_datatype ) ;
}
2020-06-11 00:53:25 +02:00
parser - > current_function = previous_function ;
}
2022-12-28 06:41:03 +01:00
void GDScriptAnalyzer : : resolve_function_body ( GDScriptParser : : FunctionNode * p_function , bool p_is_lambda ) {
2020-06-11 00:53:25 +02:00
if ( p_function - > resolved_body ) {
return ;
}
p_function - > resolved_body = true ;
GDScriptParser : : FunctionNode * previous_function = parser - > current_function ;
parser - > current_function = p_function ;
resolve_suite ( p_function - > body ) ;
GDScriptParser : : DataType return_type = p_function - > body - > get_datatype ( ) ;
2022-05-24 18:40:36 +02:00
if ( ! p_function - > get_datatype ( ) . is_hard_type ( ) & & return_type . is_set ( ) ) {
2020-06-11 00:53:25 +02:00
// Use the suite inferred type if return isn't explicitly set.
return_type . type_source = GDScriptParser : : DataType : : INFERRED ;
p_function - > set_datatype ( p_function - > body - > get_datatype ( ) ) ;
2020-06-12 00:31:28 +02:00
} else if ( p_function - > get_datatype ( ) . is_hard_type ( ) & & ( p_function - > get_datatype ( ) . kind ! = GDScriptParser : : DataType : : BUILTIN | | p_function - > get_datatype ( ) . builtin_type ! = Variant : : NIL ) ) {
2022-12-28 06:41:03 +01:00
if ( ! p_function - > body - > has_return & & ( p_is_lambda | | p_function - > identifier - > name ! = GDScriptLanguage : : get_singleton ( ) - > strings . _init ) ) {
2020-06-12 00:31:28 +02:00
push_error ( R " (Not all code paths return a value.) " , p_function ) ;
}
2020-06-11 00:53:25 +02:00
}
parser - > current_function = previous_function ;
}
void GDScriptAnalyzer : : decide_suite_type ( GDScriptParser : : Node * p_suite , GDScriptParser : : Node * p_statement ) {
2020-07-16 03:02:44 +02:00
if ( p_statement = = nullptr ) {
return ;
}
2020-06-11 00:53:25 +02:00
switch ( p_statement - > type ) {
case GDScriptParser : : Node : : IF :
case GDScriptParser : : Node : : FOR :
case GDScriptParser : : Node : : MATCH :
case GDScriptParser : : Node : : PATTERN :
case GDScriptParser : : Node : : RETURN :
case GDScriptParser : : Node : : WHILE :
// Use return or nested suite type as this suite type.
if ( p_suite - > get_datatype ( ) . is_set ( ) & & ( p_suite - > get_datatype ( ) ! = p_statement - > get_datatype ( ) ) ) {
// Mixed types.
// TODO: This could use the common supertype instead.
p_suite - > datatype . kind = GDScriptParser : : DataType : : VARIANT ;
p_suite - > datatype . type_source = GDScriptParser : : DataType : : UNDETECTED ;
} else {
p_suite - > set_datatype ( p_statement - > get_datatype ( ) ) ;
2020-08-07 19:51:56 +02:00
p_suite - > datatype . type_source = GDScriptParser : : DataType : : INFERRED ;
2020-06-11 00:53:25 +02:00
}
break ;
default :
break ;
}
}
void GDScriptAnalyzer : : resolve_suite ( GDScriptParser : : SuiteNode * p_suite ) {
for ( int i = 0 ; i < p_suite - > statements . size ( ) ; i + + ) {
GDScriptParser : : Node * stmt = p_suite - > statements [ i ] ;
2022-01-04 13:32:43 +01:00
for ( GDScriptParser : : AnnotationNode * & annotation : stmt - > annotations ) {
annotation - > apply ( parser , stmt ) ;
}
# ifdef DEBUG_ENABLED
2022-05-19 17:00:06 +02:00
HashSet < uint32_t > previously_ignored = parser - > ignored_warning_codes ;
2022-01-04 13:32:43 +01:00
for ( uint32_t ignored_warning : stmt - > ignored_warnings ) {
parser - > ignored_warning_codes . insert ( ignored_warning ) ;
}
# endif // DEBUG_ENABLED
2020-06-11 00:53:25 +02:00
resolve_node ( stmt ) ;
2022-01-04 13:32:43 +01:00
# ifdef DEBUG_ENABLED
parser - > ignored_warning_codes = previously_ignored ;
# endif // DEBUG_ENABLED
2020-06-11 00:53:25 +02:00
decide_suite_type ( p_suite , stmt ) ;
}
}
2022-12-22 21:43:36 +01:00
void GDScriptAnalyzer : : resolve_assignable ( GDScriptParser : : AssignableNode * p_assignable , const char * p_kind ) {
GDScriptParser : : DataType type ;
type . kind = GDScriptParser : : DataType : : VARIANT ;
bool is_variable = p_assignable - > type = = GDScriptParser : : Node : : VARIABLE ;
bool is_constant = p_assignable - > type = = GDScriptParser : : Node : : CONSTANT ;
GDScriptParser : : DataType specified_type ;
bool has_specified_type = p_assignable - > datatype_specifier ! = nullptr ;
if ( has_specified_type ) {
2023-01-12 00:03:53 +01:00
specified_type = type_from_metatype ( resolve_datatype ( p_assignable - > datatype_specifier ) ) ;
2022-12-22 21:43:36 +01:00
type = specified_type ;
}
if ( p_assignable - > initializer ! = nullptr ) {
reduce_expression ( p_assignable - > initializer ) ;
if ( p_assignable - > initializer - > type = = GDScriptParser : : Node : : ARRAY ) {
GDScriptParser : : ArrayNode * array = static_cast < GDScriptParser : : ArrayNode * > ( p_assignable - > initializer ) ;
if ( ( p_assignable - > infer_datatype & & array - > elements . size ( ) > 0 ) | | ( has_specified_type & & specified_type . has_container_element_type ( ) ) ) {
update_array_literal_element_type ( specified_type , array ) ;
}
}
if ( is_constant ) {
if ( p_assignable - > initializer - > type = = GDScriptParser : : Node : : ARRAY ) {
2023-01-08 04:41:06 +01:00
const_fold_array ( static_cast < GDScriptParser : : ArrayNode * > ( p_assignable - > initializer ) , true ) ;
2022-12-22 21:43:36 +01:00
} else if ( p_assignable - > initializer - > type = = GDScriptParser : : Node : : DICTIONARY ) {
2023-01-08 04:41:06 +01:00
const_fold_dictionary ( static_cast < GDScriptParser : : DictionaryNode * > ( p_assignable - > initializer ) , true ) ;
2022-12-22 21:43:36 +01:00
}
if ( ! p_assignable - > initializer - > is_constant ) {
push_error ( vformat ( R " (Assigned value for %s " % s " isn't a constant expression.) " , p_kind , p_assignable - > identifier - > name ) , p_assignable - > initializer ) ;
}
}
GDScriptParser : : DataType initializer_type = p_assignable - > initializer - > get_datatype ( ) ;
if ( p_assignable - > infer_datatype ) {
if ( ! initializer_type . is_set ( ) | | initializer_type . has_no_type ( ) ) {
push_error ( vformat ( R " (Cannot infer the type of " % s " %s because the value doesn't have a set type.) " , p_assignable - > identifier - > name , p_kind ) , p_assignable - > initializer ) ;
} else if ( initializer_type . is_variant ( ) & & ! initializer_type . is_hard_type ( ) ) {
push_error ( vformat ( R " (Cannot infer the type of " % s " %s because the value is Variant. Use explicit " Variant " type if this is intended.) " , p_assignable - > identifier - > name , p_kind ) , p_assignable - > initializer ) ;
} else if ( initializer_type . kind = = GDScriptParser : : DataType : : BUILTIN & & initializer_type . builtin_type = = Variant : : NIL & & ! is_constant ) {
push_error ( vformat ( R " (Cannot infer the type of " % s " %s because the value is " null " .) " , p_assignable - > identifier - > name , p_kind ) , p_assignable - > initializer ) ;
}
} else {
if ( ! initializer_type . is_set ( ) ) {
push_error ( vformat ( R " (Could not resolve type for %s " % s " .) " , p_kind , p_assignable - > identifier - > name ) , p_assignable - > initializer ) ;
}
}
if ( ! has_specified_type ) {
type = initializer_type ;
if ( ! type . is_set ( ) | | ( type . is_hard_type ( ) & & type . kind = = GDScriptParser : : DataType : : BUILTIN & & type . builtin_type = = Variant : : NIL & & ! is_constant ) ) {
type . kind = GDScriptParser : : DataType : : VARIANT ;
}
if ( p_assignable - > infer_datatype | | is_constant ) {
type . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
} else {
type . type_source = GDScriptParser : : DataType : : INFERRED ;
}
} else if ( ! specified_type . is_variant ( ) ) {
if ( initializer_type . is_variant ( ) | | ! initializer_type . is_hard_type ( ) ) {
mark_node_unsafe ( p_assignable - > initializer ) ;
if ( is_variable ) {
static_cast < GDScriptParser : : VariableNode * > ( p_assignable ) - > use_conversion_assign = true ;
}
} else if ( ! is_type_compatible ( specified_type , initializer_type , true , p_assignable - > initializer ) ) {
if ( is_variable & & is_type_compatible ( initializer_type , specified_type , true , p_assignable - > initializer ) ) {
mark_node_unsafe ( p_assignable - > initializer ) ;
static_cast < GDScriptParser : : VariableNode * > ( p_assignable ) - > use_conversion_assign = true ;
} else {
push_error ( vformat ( R " (Cannot assign a value of type %s to %s " % s " with specified type %s.) " , initializer_type . to_string ( ) , p_kind , p_assignable - > identifier - > name , specified_type . to_string ( ) ) , p_assignable - > initializer ) ;
}
# ifdef DEBUG_ENABLED
} else if ( specified_type . builtin_type = = Variant : : INT & & initializer_type . builtin_type = = Variant : : FLOAT ) {
parser - > push_warning ( p_assignable - > initializer , GDScriptWarning : : NARROWING_CONVERSION ) ;
# endif
}
}
}
type . is_constant = is_constant ;
p_assignable - > set_datatype ( type ) ;
}
void GDScriptAnalyzer : : resolve_variable ( GDScriptParser : : VariableNode * p_variable , bool p_is_local ) {
static constexpr const char * kind = " variable " ;
resolve_assignable ( p_variable , kind ) ;
# ifdef DEBUG_ENABLED
if ( p_is_local ) {
if ( p_variable - > usages = = 0 & & ! String ( p_variable - > identifier - > name ) . begins_with ( " _ " ) ) {
parser - > push_warning ( p_variable , GDScriptWarning : : UNUSED_VARIABLE , p_variable - > identifier - > name ) ;
} else if ( p_variable - > assignments = = 0 ) {
parser - > push_warning ( p_variable , GDScriptWarning : : UNASSIGNED_VARIABLE , p_variable - > identifier - > name ) ;
}
is_shadowing ( p_variable - > identifier , kind ) ;
}
# endif
}
void GDScriptAnalyzer : : resolve_constant ( GDScriptParser : : ConstantNode * p_constant , bool p_is_local ) {
static constexpr const char * kind = " constant " ;
resolve_assignable ( p_constant , kind ) ;
# ifdef DEBUG_ENABLED
if ( p_is_local ) {
if ( p_constant - > usages = = 0 ) {
parser - > push_warning ( p_constant , GDScriptWarning : : UNUSED_LOCAL_CONSTANT , p_constant - > identifier - > name ) ;
}
is_shadowing ( p_constant - > identifier , kind ) ;
}
# endif
}
void GDScriptAnalyzer : : resolve_parameter ( GDScriptParser : : ParameterNode * p_parameter ) {
static constexpr const char * kind = " parameter " ;
resolve_assignable ( p_parameter , kind ) ;
}
2020-06-11 00:53:25 +02:00
void GDScriptAnalyzer : : resolve_if ( GDScriptParser : : IfNode * p_if ) {
reduce_expression ( p_if - > condition ) ;
resolve_suite ( p_if - > true_block ) ;
p_if - > set_datatype ( p_if - > true_block - > get_datatype ( ) ) ;
if ( p_if - > false_block ! = nullptr ) {
resolve_suite ( p_if - > false_block ) ;
decide_suite_type ( p_if , p_if - > false_block ) ;
}
}
void GDScriptAnalyzer : : resolve_for ( GDScriptParser : : ForNode * p_for ) {
bool list_resolved = false ;
// Optimize constant range() call to not allocate an array.
2022-02-18 00:48:33 +01:00
// Use int, Vector2i, Vector3i instead, which also can be used as range iterators.
2020-07-16 03:02:44 +02:00
if ( p_for - > list & & p_for - > list - > type = = GDScriptParser : : Node : : CALL ) {
2020-06-11 00:53:25 +02:00
GDScriptParser : : CallNode * call = static_cast < GDScriptParser : : CallNode * > ( p_for - > list ) ;
2020-08-05 21:41:46 +02:00
GDScriptParser : : Node : : Type callee_type = call - > get_callee_type ( ) ;
if ( callee_type = = GDScriptParser : : Node : : IDENTIFIER ) {
2020-06-11 00:53:25 +02:00
GDScriptParser : : IdentifierNode * callee = static_cast < GDScriptParser : : IdentifierNode * > ( call - > callee ) ;
if ( callee - > name = = " range " ) {
list_resolved = true ;
if ( call - > arguments . size ( ) < 1 ) {
push_error ( R " *(Invalid call for " range ( ) " function. Expected at least 1 argument, none given.)* " , call - > callee ) ;
} else if ( call - > arguments . size ( ) > 3 ) {
push_error ( vformat ( R " *(Invalid call for " range ( ) " function. Expected at most 3 arguments, %d given.)* " , call - > arguments . size ( ) ) , call - > callee ) ;
} else {
// Now we can optimize it.
bool all_is_constant = true ;
Vector < Variant > args ;
args . resize ( call - > arguments . size ( ) ) ;
for ( int i = 0 ; i < call - > arguments . size ( ) ; i + + ) {
reduce_expression ( call - > arguments [ i ] ) ;
if ( ! call - > arguments [ i ] - > is_constant ) {
all_is_constant = false ;
2020-11-25 15:35:07 +01:00
} else if ( all_is_constant ) {
2020-06-11 00:53:25 +02:00
args . write [ i ] = call - > arguments [ i ] - > reduced_value ;
}
GDScriptParser : : DataType arg_type = call - > arguments [ i ] - > get_datatype ( ) ;
2020-11-27 13:53:35 +01:00
if ( ! arg_type . is_variant ( ) ) {
if ( arg_type . kind ! = GDScriptParser : : DataType : : BUILTIN ) {
all_is_constant = false ;
push_error ( vformat ( R " *(Invalid argument for " range ( ) " call. Argument %d should be int or float but " % s " was given.)* " , i + 1 , arg_type . to_string ( ) ) , call - > arguments [ i ] ) ;
} else if ( arg_type . builtin_type ! = Variant : : INT & & arg_type . builtin_type ! = Variant : : FLOAT ) {
all_is_constant = false ;
push_error ( vformat ( R " *(Invalid argument for " range ( ) " call. Argument %d should be int or float but " % s " was given.)* " , i + 1 , arg_type . to_string ( ) ) , call - > arguments [ i ] ) ;
}
2020-06-11 00:53:25 +02:00
}
}
Variant reduced ;
if ( all_is_constant ) {
switch ( args . size ( ) ) {
case 1 :
2022-11-01 17:07:40 +01:00
reduced = ( int32_t ) args [ 0 ] ;
2020-06-11 00:53:25 +02:00
break ;
case 2 :
reduced = Vector2i ( args [ 0 ] , args [ 1 ] ) ;
break ;
case 3 :
reduced = Vector3i ( args [ 0 ] , args [ 1 ] , args [ 2 ] ) ;
break ;
}
p_for - > list - > is_constant = true ;
p_for - > list - > reduced_value = reduced ;
}
}
2020-11-25 15:35:07 +01:00
if ( p_for - > list - > is_constant ) {
p_for - > list - > set_datatype ( type_from_variant ( p_for - > list - > reduced_value , p_for - > list ) ) ;
} else {
GDScriptParser : : DataType list_type ;
list_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
list_type . kind = GDScriptParser : : DataType : : BUILTIN ;
list_type . builtin_type = Variant : : ARRAY ;
p_for - > list - > set_datatype ( list_type ) ;
}
2020-06-11 00:53:25 +02:00
}
}
}
2021-08-14 05:44:22 +02:00
GDScriptParser : : DataType variable_type ;
if ( list_resolved ) {
variable_type . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
variable_type . kind = GDScriptParser : : DataType : : BUILTIN ;
variable_type . builtin_type = Variant : : INT ; // Can this ever be a float or something else?
p_for - > variable - > set_datatype ( variable_type ) ;
2021-10-02 19:53:56 +02:00
} else if ( p_for - > list ) {
2022-10-13 19:31:12 +02:00
resolve_node ( p_for - > list , false ) ;
2021-08-14 05:44:22 +02:00
if ( p_for - > list - > datatype . has_container_element_type ( ) ) {
variable_type = p_for - > list - > datatype . get_container_element_type ( ) ;
variable_type . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
} else if ( p_for - > list - > datatype . is_typed_container_type ( ) ) {
variable_type = p_for - > list - > datatype . get_typed_container_type ( ) ;
variable_type . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
} else {
// Last resort
// TODO: Must other cases be handled? Must we mark as unsafe?
variable_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
variable_type . kind = GDScriptParser : : DataType : : VARIANT ;
}
2020-06-11 00:53:25 +02:00
}
2021-10-02 19:53:56 +02:00
if ( p_for - > variable ) {
p_for - > variable - > set_datatype ( variable_type ) ;
}
2020-06-11 00:53:25 +02:00
resolve_suite ( p_for - > loop ) ;
p_for - > set_datatype ( p_for - > loop - > get_datatype ( ) ) ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
if ( p_for - > variable ) {
is_shadowing ( p_for - > variable , R " ( " for " iterator variable) " ) ;
}
# endif
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : resolve_while ( GDScriptParser : : WhileNode * p_while ) {
2022-10-13 19:31:12 +02:00
resolve_node ( p_while - > condition , false ) ;
2020-06-11 00:53:25 +02:00
resolve_suite ( p_while - > loop ) ;
p_while - > set_datatype ( p_while - > loop - > get_datatype ( ) ) ;
}
void GDScriptAnalyzer : : resolve_assert ( GDScriptParser : : AssertNode * p_assert ) {
reduce_expression ( p_assert - > condition ) ;
2020-06-12 00:31:28 +02:00
if ( p_assert - > message ! = nullptr ) {
2020-12-27 06:57:50 +01:00
reduce_expression ( p_assert - > message ) ;
2022-07-04 05:53:31 +02:00
if ( ! p_assert - > message - > get_datatype ( ) . has_no_type ( ) & & ( p_assert - > message - > get_datatype ( ) . kind ! = GDScriptParser : : DataType : : BUILTIN | | p_assert - > message - > get_datatype ( ) . builtin_type ! = Variant : : STRING ) ) {
push_error ( R " (Expected string for assert error message.) " , p_assert - > message ) ;
2020-12-27 06:57:50 +01:00
}
2020-06-12 00:31:28 +02:00
}
2020-06-11 00:53:25 +02:00
p_assert - > set_datatype ( p_assert - > condition - > get_datatype ( ) ) ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
if ( p_assert - > condition - > is_constant ) {
if ( p_assert - > condition - > reduced_value . booleanize ( ) ) {
parser - > push_warning ( p_assert - > condition , GDScriptWarning : : ASSERT_ALWAYS_TRUE ) ;
} else {
parser - > push_warning ( p_assert - > condition , GDScriptWarning : : ASSERT_ALWAYS_FALSE ) ;
}
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : resolve_match ( GDScriptParser : : MatchNode * p_match ) {
reduce_expression ( p_match - > test ) ;
for ( int i = 0 ; i < p_match - > branches . size ( ) ; i + + ) {
resolve_match_branch ( p_match - > branches [ i ] , p_match - > test ) ;
decide_suite_type ( p_match , p_match - > branches [ i ] ) ;
}
}
void GDScriptAnalyzer : : resolve_match_branch ( GDScriptParser : : MatchBranchNode * p_match_branch , GDScriptParser : : ExpressionNode * p_match_test ) {
for ( int i = 0 ; i < p_match_branch - > patterns . size ( ) ; i + + ) {
resolve_match_pattern ( p_match_branch - > patterns [ i ] , p_match_test ) ;
}
resolve_suite ( p_match_branch - > block ) ;
decide_suite_type ( p_match_branch , p_match_branch - > block ) ;
}
void GDScriptAnalyzer : : resolve_match_pattern ( GDScriptParser : : PatternNode * p_match_pattern , GDScriptParser : : ExpressionNode * p_match_test ) {
2020-07-16 03:02:44 +02:00
if ( p_match_pattern = = nullptr ) {
return ;
}
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType result ;
switch ( p_match_pattern - > pattern_type ) {
case GDScriptParser : : PatternNode : : PT_LITERAL :
2020-07-16 03:02:44 +02:00
if ( p_match_pattern - > literal ) {
reduce_literal ( p_match_pattern - > literal ) ;
result = p_match_pattern - > literal - > get_datatype ( ) ;
}
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : PatternNode : : PT_EXPRESSION :
2020-07-16 03:02:44 +02:00
if ( p_match_pattern - > expression ) {
reduce_expression ( p_match_pattern - > expression ) ;
if ( ! p_match_pattern - > expression - > is_constant ) {
push_error ( R " (Expression in match pattern must be a constant.) " , p_match_pattern - > expression ) ;
}
result = p_match_pattern - > expression - > get_datatype ( ) ;
2020-06-11 00:53:25 +02:00
}
break ;
case GDScriptParser : : PatternNode : : PT_BIND :
if ( p_match_test ! = nullptr ) {
result = p_match_test - > get_datatype ( ) ;
} else {
result . kind = GDScriptParser : : DataType : : VARIANT ;
}
p_match_pattern - > bind - > set_datatype ( result ) ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
is_shadowing ( p_match_pattern - > bind , " pattern bind " ) ;
2022-06-03 19:42:03 +02:00
if ( p_match_pattern - > bind - > usages = = 0 & & ! String ( p_match_pattern - > bind - > name ) . begins_with ( " _ " ) ) {
parser - > push_warning ( p_match_pattern - > bind , GDScriptWarning : : UNUSED_VARIABLE , p_match_pattern - > bind - > name ) ;
2020-06-12 00:31:28 +02:00
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : PatternNode : : PT_ARRAY :
for ( int i = 0 ; i < p_match_pattern - > array . size ( ) ; i + + ) {
resolve_match_pattern ( p_match_pattern - > array [ i ] , nullptr ) ;
decide_suite_type ( p_match_pattern , p_match_pattern - > array [ i ] ) ;
}
result = p_match_pattern - > get_datatype ( ) ;
break ;
case GDScriptParser : : PatternNode : : PT_DICTIONARY :
for ( int i = 0 ; i < p_match_pattern - > dictionary . size ( ) ; i + + ) {
2020-07-16 03:02:44 +02:00
if ( p_match_pattern - > dictionary [ i ] . key ) {
reduce_expression ( p_match_pattern - > dictionary [ i ] . key ) ;
if ( ! p_match_pattern - > dictionary [ i ] . key - > is_constant ) {
2022-01-25 21:10:07 +01:00
push_error ( R " (Expression in dictionary pattern key must be a constant.) " , p_match_pattern - > dictionary [ i ] . key ) ;
2020-07-16 03:02:44 +02:00
}
2020-06-11 00:53:25 +02:00
}
2020-07-16 03:02:44 +02:00
if ( p_match_pattern - > dictionary [ i ] . value_pattern ) {
resolve_match_pattern ( p_match_pattern - > dictionary [ i ] . value_pattern , nullptr ) ;
decide_suite_type ( p_match_pattern , p_match_pattern - > dictionary [ i ] . value_pattern ) ;
}
2020-06-11 00:53:25 +02:00
}
result = p_match_pattern - > get_datatype ( ) ;
break ;
case GDScriptParser : : PatternNode : : PT_WILDCARD :
case GDScriptParser : : PatternNode : : PT_REST :
result . kind = GDScriptParser : : DataType : : VARIANT ;
break ;
}
p_match_pattern - > set_datatype ( result ) ;
}
void GDScriptAnalyzer : : resolve_return ( GDScriptParser : : ReturnNode * p_return ) {
GDScriptParser : : DataType result ;
2021-09-06 07:04:43 +02:00
GDScriptParser : : DataType expected_type ;
bool has_expected_type = false ;
if ( parser - > current_function ! = nullptr ) {
expected_type = parser - > current_function - > get_datatype ( ) ;
has_expected_type = true ;
}
2020-06-11 00:53:25 +02:00
if ( p_return - > return_value ! = nullptr ) {
reduce_expression ( p_return - > return_value ) ;
2021-03-09 16:32:35 +01:00
if ( p_return - > return_value - > type = = GDScriptParser : : Node : : ARRAY ) {
// Check if assigned value is an array literal, so we can make it a typed array too if appropriate.
2021-09-06 07:04:43 +02:00
if ( has_expected_type & & expected_type . has_container_element_type ( ) & & p_return - > return_value - > type = = GDScriptParser : : Node : : ARRAY ) {
update_array_literal_element_type ( expected_type , static_cast < GDScriptParser : : ArrayNode * > ( p_return - > return_value ) ) ;
2021-03-09 16:32:35 +01:00
}
}
2022-12-28 06:12:16 +01:00
if ( has_expected_type & & expected_type . is_hard_type ( ) & & expected_type . kind = = GDScriptParser : : DataType : : BUILTIN & & expected_type . builtin_type = = Variant : : NIL ) {
push_error ( " A void function cannot return a value. " , p_return ) ;
}
2020-06-11 00:53:25 +02:00
result = p_return - > return_value - > get_datatype ( ) ;
} else {
// Return type is null by default.
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = Variant : : NIL ;
result . is_constant = true ;
}
2021-09-06 07:04:43 +02:00
if ( has_expected_type ) {
expected_type . is_meta_type = false ;
if ( expected_type . is_hard_type ( ) ) {
if ( ! is_type_compatible ( expected_type , result ) ) {
// Try other way. Okay but not safe.
if ( ! is_type_compatible ( result , expected_type ) ) {
push_error ( vformat ( R " (Cannot return value of type " % s " because the function return type is " % s " .) " , result . to_string ( ) , expected_type . to_string ( ) ) , p_return ) ;
} else {
// TODO: Add warning.
mark_node_unsafe ( p_return ) ;
}
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2021-09-06 07:04:43 +02:00
} else if ( expected_type . builtin_type = = Variant : : INT & & result . builtin_type = = Variant : : FLOAT ) {
parser - > push_warning ( p_return , GDScriptWarning : : NARROWING_CONVERSION ) ;
} else if ( result . is_variant ( ) ) {
mark_node_unsafe ( p_return ) ;
2020-07-16 03:02:44 +02:00
# endif
2021-09-06 07:04:43 +02:00
}
2020-06-12 00:31:28 +02:00
}
}
2020-06-11 00:53:25 +02:00
p_return - > set_datatype ( result ) ;
}
2021-10-15 01:12:01 +02:00
void GDScriptAnalyzer : : reduce_expression ( GDScriptParser : : ExpressionNode * p_expression , bool p_is_root ) {
2020-06-11 00:53:25 +02:00
// This one makes some magic happen.
2020-07-06 17:24:24 +02:00
if ( p_expression = = nullptr ) {
return ;
}
2020-06-11 00:53:25 +02:00
if ( p_expression - > reduced ) {
// Don't do this more than once.
return ;
}
p_expression - > reduced = true ;
switch ( p_expression - > type ) {
case GDScriptParser : : Node : : ARRAY :
reduce_array ( static_cast < GDScriptParser : : ArrayNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : ASSIGNMENT :
reduce_assignment ( static_cast < GDScriptParser : : AssignmentNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : AWAIT :
reduce_await ( static_cast < GDScriptParser : : AwaitNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : BINARY_OPERATOR :
reduce_binary_op ( static_cast < GDScriptParser : : BinaryOpNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : CALL :
2022-10-13 19:31:12 +02:00
reduce_call ( static_cast < GDScriptParser : : CallNode * > ( p_expression ) , false , p_is_root ) ;
2020-06-11 00:53:25 +02:00
break ;
case GDScriptParser : : Node : : CAST :
reduce_cast ( static_cast < GDScriptParser : : CastNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : DICTIONARY :
reduce_dictionary ( static_cast < GDScriptParser : : DictionaryNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : GET_NODE :
reduce_get_node ( static_cast < GDScriptParser : : GetNodeNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : IDENTIFIER :
reduce_identifier ( static_cast < GDScriptParser : : IdentifierNode * > ( p_expression ) ) ;
break ;
2021-03-26 13:03:16 +01:00
case GDScriptParser : : Node : : LAMBDA :
reduce_lambda ( static_cast < GDScriptParser : : LambdaNode * > ( p_expression ) ) ;
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : Node : : LITERAL :
reduce_literal ( static_cast < GDScriptParser : : LiteralNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : PRELOAD :
reduce_preload ( static_cast < GDScriptParser : : PreloadNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : SELF :
reduce_self ( static_cast < GDScriptParser : : SelfNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : SUBSCRIPT :
reduce_subscript ( static_cast < GDScriptParser : : SubscriptNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : TERNARY_OPERATOR :
reduce_ternary_op ( static_cast < GDScriptParser : : TernaryOpNode * > ( p_expression ) ) ;
break ;
case GDScriptParser : : Node : : UNARY_OPERATOR :
reduce_unary_op ( static_cast < GDScriptParser : : UnaryOpNode * > ( p_expression ) ) ;
break ;
// Non-expressions. Here only to make sure new nodes aren't forgotten.
case GDScriptParser : : Node : : NONE :
case GDScriptParser : : Node : : ANNOTATION :
case GDScriptParser : : Node : : ASSERT :
case GDScriptParser : : Node : : BREAK :
case GDScriptParser : : Node : : BREAKPOINT :
case GDScriptParser : : Node : : CLASS :
case GDScriptParser : : Node : : CONSTANT :
case GDScriptParser : : Node : : CONTINUE :
case GDScriptParser : : Node : : ENUM :
case GDScriptParser : : Node : : FOR :
case GDScriptParser : : Node : : FUNCTION :
case GDScriptParser : : Node : : IF :
case GDScriptParser : : Node : : MATCH :
case GDScriptParser : : Node : : MATCH_BRANCH :
case GDScriptParser : : Node : : PARAMETER :
case GDScriptParser : : Node : : PASS :
case GDScriptParser : : Node : : PATTERN :
case GDScriptParser : : Node : : RETURN :
case GDScriptParser : : Node : : SIGNAL :
case GDScriptParser : : Node : : SUITE :
case GDScriptParser : : Node : : TYPE :
case GDScriptParser : : Node : : VARIABLE :
case GDScriptParser : : Node : : WHILE :
ERR_FAIL_MSG ( " Reaching unreachable case " ) ;
}
}
void GDScriptAnalyzer : : reduce_array ( GDScriptParser : : ArrayNode * p_array ) {
for ( int i = 0 ; i < p_array - > elements . size ( ) ; i + + ) {
GDScriptParser : : ExpressionNode * element = p_array - > elements [ i ] ;
reduce_expression ( element ) ;
}
// It's array in any case.
GDScriptParser : : DataType arr_type ;
arr_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
arr_type . kind = GDScriptParser : : DataType : : BUILTIN ;
arr_type . builtin_type = Variant : : ARRAY ;
arr_type . is_constant = true ;
p_array - > set_datatype ( arr_type ) ;
}
2021-03-09 16:32:35 +01:00
// When an array literal is stored (or passed as function argument) to a typed context, we then assume the array is typed.
// This function determines which type is that (if any).
void GDScriptAnalyzer : : update_array_literal_element_type ( const GDScriptParser : : DataType & p_base_type , GDScriptParser : : ArrayNode * p_array_literal ) {
GDScriptParser : : DataType array_type = p_array_literal - > get_datatype ( ) ;
if ( p_array_literal - > elements . size ( ) = = 0 ) {
// Empty array literal, just make the same type as the storage.
array_type . set_container_element_type ( p_base_type . get_container_element_type ( ) ) ;
} else {
// Check if elements match.
bool all_same_type = true ;
bool all_have_type = true ;
GDScriptParser : : DataType element_type ;
for ( int i = 0 ; i < p_array_literal - > elements . size ( ) ; i + + ) {
if ( i = = 0 ) {
element_type = p_array_literal - > elements [ 0 ] - > get_datatype ( ) ;
} else {
GDScriptParser : : DataType this_element_type = p_array_literal - > elements [ i ] - > get_datatype ( ) ;
if ( this_element_type . has_no_type ( ) ) {
all_same_type = false ;
all_have_type = false ;
break ;
} else if ( element_type ! = this_element_type ) {
if ( ! is_type_compatible ( element_type , this_element_type , false ) ) {
if ( is_type_compatible ( this_element_type , element_type , false ) ) {
// This element is a super-type to the previous type, so we use the super-type.
element_type = this_element_type ;
} else {
// It's incompatible.
all_same_type = false ;
break ;
}
}
}
}
}
if ( all_same_type ) {
2021-10-15 15:40:50 +02:00
element_type . is_constant = false ;
2021-03-09 16:32:35 +01:00
array_type . set_container_element_type ( element_type ) ;
} else if ( all_have_type ) {
push_error ( vformat ( R " (Variant array is not compatible with an array of type " % s " .) " , p_base_type . get_container_element_type ( ) . to_string ( ) ) , p_array_literal ) ;
}
}
// Update the type on the value itself.
p_array_literal - > set_datatype ( array_type ) ;
}
2020-06-11 00:53:25 +02:00
void GDScriptAnalyzer : : reduce_assignment ( GDScriptParser : : AssignmentNode * p_assignment ) {
reduce_expression ( p_assignment - > assignee ) ;
reduce_expression ( p_assignment - > assigned_value ) ;
2020-07-06 17:24:24 +02:00
if ( p_assignment - > assigned_value = = nullptr | | p_assignment - > assignee = = nullptr ) {
return ;
}
2021-03-09 16:32:35 +01:00
GDScriptParser : : DataType assignee_type = p_assignment - > assignee - > get_datatype ( ) ;
2023-01-08 04:41:06 +01:00
if ( assignee_type . is_constant | | ( p_assignment - > assignee - > type = = GDScriptParser : : Node : : SUBSCRIPT & & static_cast < GDScriptParser : : SubscriptNode * > ( p_assignment - > assignee ) - > base - > is_constant ) ) {
push_error ( " Cannot assign a new value to a constant. " , p_assignment - > assignee ) ;
}
2021-03-09 16:32:35 +01:00
// Check if assigned value is an array literal, so we can make it a typed array too if appropriate.
if ( assignee_type . has_container_element_type ( ) & & p_assignment - > assigned_value - > type = = GDScriptParser : : Node : : ARRAY ) {
update_array_literal_element_type ( assignee_type , static_cast < GDScriptParser : : ArrayNode * > ( p_assignment - > assigned_value ) ) ;
}
GDScriptParser : : DataType assigned_value_type = p_assignment - > assigned_value - > get_datatype ( ) ;
2021-12-26 20:32:22 +01:00
bool compatible = true ;
GDScriptParser : : DataType op_type = assigned_value_type ;
2023-01-11 00:01:11 +01:00
if ( p_assignment - > operation ! = GDScriptParser : : AssignmentNode : : OP_NONE & & ! op_type . is_variant ( ) ) {
2021-12-26 20:32:22 +01:00
op_type = get_operation_type ( p_assignment - > variant_op , assignee_type , assigned_value_type , compatible , p_assignment - > assigned_value ) ;
}
p_assignment - > set_datatype ( op_type ) ;
2020-07-25 04:27:58 +02:00
2022-12-04 04:02:03 +01:00
// If Assignee is a variant, then you can assign anything
// When the assigned value has a known type, further checks are possible.
2023-01-11 00:01:11 +01:00
if ( assignee_type . is_hard_type ( ) & & ! assignee_type . is_variant ( ) & & op_type . is_hard_type ( ) & & ! op_type . is_variant ( ) ) {
2020-08-06 07:42:26 +02:00
if ( compatible ) {
2022-01-27 15:34:33 +01:00
compatible = is_type_compatible ( assignee_type , op_type , true , p_assignment - > assigned_value ) ;
2020-08-06 07:42:26 +02:00
if ( ! compatible ) {
2022-07-04 00:10:16 +02:00
// Try reverse test since it can be a masked subtype.
if ( ! is_type_compatible ( op_type , assignee_type , true ) ) {
2022-12-04 04:02:03 +01:00
push_error ( vformat ( R " (Value of type " % s " cannot be assigned to a variable of type " % s " .) " , assigned_value_type . to_string ( ) , assignee_type . to_string ( ) ) , p_assignment - > assigned_value ) ;
2020-08-06 07:42:26 +02:00
} else {
2022-07-04 00:10:16 +02:00
// TODO: Add warning.
2020-08-18 20:12:51 +02:00
mark_node_unsafe ( p_assignment ) ;
2022-07-04 00:10:16 +02:00
p_assignment - > use_conversion_assign = true ;
2020-08-06 07:42:26 +02:00
}
2020-07-25 04:27:58 +02:00
}
2020-08-06 07:42:26 +02:00
} else {
2021-03-09 16:32:35 +01:00
push_error ( vformat ( R " (Invalid operands " % s " and " % s " for assignment operator.) " , assignee_type . to_string ( ) , assigned_value_type . to_string ( ) ) , p_assignment ) ;
2020-06-11 00:53:25 +02:00
}
2022-07-04 00:10:16 +02:00
} else if ( assignee_type . is_hard_type ( ) & & ! assignee_type . is_variant ( ) ) {
mark_node_unsafe ( p_assignment ) ;
p_assignment - > use_conversion_assign = true ;
} else {
2020-06-11 00:53:25 +02:00
mark_node_unsafe ( p_assignment ) ;
2021-12-25 01:53:57 +01:00
if ( assignee_type . is_hard_type ( ) & & ! assignee_type . is_variant ( ) ) {
2021-05-26 19:05:31 +02:00
p_assignment - > use_conversion_assign = true ;
}
2020-06-11 00:53:25 +02:00
}
if ( p_assignment - > assignee - > type = = GDScriptParser : : Node : : IDENTIFIER ) {
// Change source type so it's not wrongly detected later.
GDScriptParser : : IdentifierNode * identifier = static_cast < GDScriptParser : : IdentifierNode * > ( p_assignment - > assignee ) ;
switch ( identifier - > source ) {
case GDScriptParser : : IdentifierNode : : MEMBER_VARIABLE : {
GDScriptParser : : DataType id_type = identifier - > variable_source - > get_datatype ( ) ;
if ( ! id_type . is_hard_type ( ) ) {
id_type . kind = GDScriptParser : : DataType : : VARIANT ;
id_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
identifier - > variable_source - > set_datatype ( id_type ) ;
}
} break ;
2021-09-17 16:52:30 +02:00
case GDScriptParser : : IdentifierNode : : FUNCTION_PARAMETER : {
GDScriptParser : : DataType id_type = identifier - > parameter_source - > get_datatype ( ) ;
if ( ! id_type . is_hard_type ( ) ) {
2021-10-06 16:39:19 +02:00
id_type . kind = GDScriptParser : : DataType : : VARIANT ;
id_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
2021-09-17 16:52:30 +02:00
identifier - > parameter_source - > set_datatype ( id_type ) ;
}
} break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : IdentifierNode : : LOCAL_VARIABLE : {
GDScriptParser : : DataType id_type = identifier - > variable_source - > get_datatype ( ) ;
if ( ! id_type . is_hard_type ( ) ) {
2021-10-06 16:39:19 +02:00
id_type . kind = GDScriptParser : : DataType : : VARIANT ;
id_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
2020-06-11 00:53:25 +02:00
identifier - > variable_source - > set_datatype ( id_type ) ;
}
} break ;
case GDScriptParser : : IdentifierNode : : LOCAL_ITERATOR : {
GDScriptParser : : DataType id_type = identifier - > bind_source - > get_datatype ( ) ;
if ( ! id_type . is_hard_type ( ) ) {
2021-10-06 16:39:19 +02:00
id_type . kind = GDScriptParser : : DataType : : VARIANT ;
id_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
2022-12-04 04:02:03 +01:00
identifier - > bind_source - > set_datatype ( id_type ) ;
2020-06-11 00:53:25 +02:00
}
} break ;
default :
// Nothing to do.
break ;
}
}
2020-06-12 00:31:28 +02:00
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2022-12-30 15:57:25 +01:00
if ( assignee_type . is_hard_type ( ) & & assignee_type . builtin_type = = Variant : : INT & & assigned_value_type . builtin_type = = Variant : : FLOAT ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_assignment - > assigned_value , GDScriptWarning : : NARROWING_CONVERSION ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : reduce_await ( GDScriptParser : : AwaitNode * p_await ) {
2020-07-27 14:50:36 +02:00
if ( p_await - > to_await = = nullptr ) {
GDScriptParser : : DataType await_type ;
await_type . kind = GDScriptParser : : DataType : : VARIANT ;
p_await - > set_datatype ( await_type ) ;
return ;
}
2021-10-15 01:30:06 +02:00
GDScriptParser : : DataType awaiting_type ;
2020-06-12 00:31:28 +02:00
if ( p_await - > to_await - > type = = GDScriptParser : : Node : : CALL ) {
reduce_call ( static_cast < GDScriptParser : : CallNode * > ( p_await - > to_await ) , true ) ;
2021-10-15 01:30:06 +02:00
awaiting_type = p_await - > to_await - > get_datatype ( ) ;
2020-06-12 00:31:28 +02:00
} else {
reduce_expression ( p_await - > to_await ) ;
}
2020-06-11 00:53:25 +02:00
2021-10-15 01:30:06 +02:00
if ( p_await - > to_await - > is_constant ) {
p_await - > is_constant = p_await - > to_await - > is_constant ;
p_await - > reduced_value = p_await - > to_await - > reduced_value ;
2020-06-11 00:53:25 +02:00
2021-10-15 01:30:06 +02:00
awaiting_type = p_await - > to_await - > get_datatype ( ) ;
} else {
awaiting_type . kind = GDScriptParser : : DataType : : VARIANT ;
awaiting_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
}
2020-06-11 00:53:25 +02:00
p_await - > set_datatype ( awaiting_type ) ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2022-05-24 18:40:36 +02:00
awaiting_type = p_await - > to_await - > get_datatype ( ) ;
if ( ! ( awaiting_type . has_no_type ( ) | | awaiting_type . is_coroutine | | awaiting_type . builtin_type = = Variant : : SIGNAL ) ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_await , GDScriptWarning : : REDUNDANT_AWAIT ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : reduce_binary_op ( GDScriptParser : : BinaryOpNode * p_binary_op ) {
reduce_expression ( p_binary_op - > left_operand ) ;
2020-07-16 03:02:44 +02:00
if ( p_binary_op - > operation = = GDScriptParser : : BinaryOpNode : : OP_TYPE_TEST & & p_binary_op - > right_operand & & p_binary_op - > right_operand - > type = = GDScriptParser : : Node : : IDENTIFIER ) {
2020-06-12 02:49:58 +02:00
reduce_identifier ( static_cast < GDScriptParser : : IdentifierNode * > ( p_binary_op - > right_operand ) , true ) ;
} else {
reduce_expression ( p_binary_op - > right_operand ) ;
}
2020-06-11 00:53:25 +02:00
// TODO: Right operand must be a valid type with the `is` operator. Need to check here.
2020-07-16 03:02:44 +02:00
GDScriptParser : : DataType left_type ;
if ( p_binary_op - > left_operand ) {
2021-09-30 18:56:33 +02:00
left_type = p_binary_op - > left_operand - > get_datatype ( ) ;
2020-07-16 03:02:44 +02:00
}
GDScriptParser : : DataType right_type ;
if ( p_binary_op - > right_operand ) {
2021-09-30 18:56:33 +02:00
right_type = p_binary_op - > right_operand - > get_datatype ( ) ;
2020-07-16 03:02:44 +02:00
}
if ( ! left_type . is_set ( ) | | ! right_type . is_set ( ) ) {
return ;
}
2020-06-12 00:31:28 +02:00
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
if ( p_binary_op - > variant_op = = Variant : : OP_DIVIDE & & left_type . builtin_type = = Variant : : INT & & right_type . builtin_type = = Variant : : INT ) {
parser - > push_warning ( p_binary_op , GDScriptWarning : : INTEGER_DIVISION ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-12 00:31:28 +02:00
2020-06-11 00:53:25 +02:00
if ( p_binary_op - > left_operand - > is_constant & & p_binary_op - > right_operand - > is_constant ) {
p_binary_op - > is_constant = true ;
if ( p_binary_op - > variant_op < Variant : : OP_MAX ) {
2020-07-24 20:02:29 +02:00
bool valid = false ;
Variant : : evaluate ( p_binary_op - > variant_op , p_binary_op - > left_operand - > reduced_value , p_binary_op - > right_operand - > reduced_value , p_binary_op - > reduced_value , valid ) ;
if ( ! valid ) {
if ( p_binary_op - > reduced_value . get_type ( ) = = Variant : : STRING ) {
push_error ( vformat ( R " (%s in operator %s.) " , p_binary_op - > reduced_value , Variant : : get_operator_name ( p_binary_op - > variant_op ) ) , p_binary_op ) ;
} else {
2020-12-01 15:16:31 +01:00
push_error ( vformat ( R " (Invalid operands to operator %s, %s and %s.) " ,
2020-07-24 20:02:29 +02:00
Variant : : get_operator_name ( p_binary_op - > variant_op ) ,
Variant : : get_type_name ( p_binary_op - > left_operand - > reduced_value . get_type ( ) ) ,
Variant : : get_type_name ( p_binary_op - > right_operand - > reduced_value . get_type ( ) ) ) ,
p_binary_op ) ;
}
}
2020-06-11 00:53:25 +02:00
} else {
if ( p_binary_op - > operation = = GDScriptParser : : BinaryOpNode : : OP_TYPE_TEST ) {
2020-06-12 00:31:28 +02:00
GDScriptParser : : DataType test_type = right_type ;
2020-06-11 00:53:25 +02:00
test_type . is_meta_type = false ;
2022-10-02 02:13:06 +02:00
if ( ! is_type_compatible ( test_type , left_type , false ) ) {
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (Expression is of type " % s " so it can't be of type " % s " .) " ) , p_binary_op - > left_operand ) ;
p_binary_op - > reduced_value = false ;
} else {
p_binary_op - > reduced_value = true ;
}
} else {
ERR_PRINT ( " Parser bug: unknown binary operation. " ) ;
}
}
2020-08-31 14:53:02 +02:00
p_binary_op - > set_datatype ( type_from_variant ( p_binary_op - > reduced_value , p_binary_op ) ) ;
2020-06-12 00:31:28 +02:00
2020-06-11 00:53:25 +02:00
return ;
}
GDScriptParser : : DataType result ;
2020-06-12 00:31:28 +02:00
if ( left_type . is_variant ( ) | | right_type . is_variant ( ) ) {
2020-06-11 00:53:25 +02:00
// Cannot infer type because one operand can be anything.
result . kind = GDScriptParser : : DataType : : VARIANT ;
mark_node_unsafe ( p_binary_op ) ;
} else {
if ( p_binary_op - > variant_op < Variant : : OP_MAX ) {
bool valid = false ;
2021-06-09 01:52:21 +02:00
result = get_operation_type ( p_binary_op - > variant_op , left_type , right_type , valid , p_binary_op ) ;
2020-06-11 00:53:25 +02:00
if ( ! valid ) {
2021-06-09 01:52:21 +02:00
push_error ( vformat ( R " (Invalid operands " % s " and " % s " for " % s " operator.) " , left_type . to_string ( ) , right_type . to_string ( ) , Variant : : get_operator_name ( p_binary_op - > variant_op ) ) , p_binary_op ) ;
2020-06-11 00:53:25 +02:00
}
} else {
if ( p_binary_op - > operation = = GDScriptParser : : BinaryOpNode : : OP_TYPE_TEST ) {
2020-06-12 00:31:28 +02:00
GDScriptParser : : DataType test_type = right_type ;
2020-06-11 00:53:25 +02:00
test_type . is_meta_type = false ;
2022-10-02 02:13:06 +02:00
if ( ! is_type_compatible ( test_type , left_type , false ) ) {
2020-07-16 03:02:44 +02:00
// Test reverse as well to consider for subtypes.
2022-10-02 02:13:06 +02:00
if ( ! is_type_compatible ( left_type , test_type , false ) ) {
if ( left_type . is_hard_type ( ) ) {
push_error ( vformat ( R " (Expression is of type " % s " so it can't be of type " % s " .) " , left_type . to_string ( ) , test_type . to_string ( ) ) , p_binary_op - > left_operand ) ;
2020-07-16 03:02:44 +02:00
} else {
// TODO: Warning.
mark_node_unsafe ( p_binary_op ) ;
}
2020-06-11 00:53:25 +02:00
}
}
// "is" operator is always a boolean anyway.
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = Variant : : BOOL ;
} else {
ERR_PRINT ( " Parser bug: unknown binary operation. " ) ;
}
}
}
p_binary_op - > set_datatype ( result ) ;
}
2021-10-15 01:12:01 +02:00
void GDScriptAnalyzer : : reduce_call ( GDScriptParser : : CallNode * p_call , bool p_is_await , bool p_is_root ) {
2020-06-11 00:53:25 +02:00
bool all_is_constant = true ;
2022-05-13 15:04:37 +02:00
HashMap < int , GDScriptParser : : ArrayNode * > arrays ; // For array literal to potentially type when passing.
2020-06-11 00:53:25 +02:00
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
reduce_expression ( p_call - > arguments [ i ] ) ;
2021-03-09 16:32:35 +01:00
if ( p_call - > arguments [ i ] - > type = = GDScriptParser : : Node : : ARRAY ) {
arrays [ i ] = static_cast < GDScriptParser : : ArrayNode * > ( p_call - > arguments [ i ] ) ;
}
2020-06-11 00:53:25 +02:00
all_is_constant = all_is_constant & & p_call - > arguments [ i ] - > is_constant ;
}
2020-08-05 21:41:46 +02:00
GDScriptParser : : Node : : Type callee_type = p_call - > get_callee_type ( ) ;
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType call_type ;
2020-08-05 21:41:46 +02:00
if ( ! p_call - > is_super & & callee_type = = GDScriptParser : : Node : : IDENTIFIER ) {
2020-06-11 00:53:25 +02:00
// Call to name directly.
StringName function_name = p_call - > function_name ;
Variant : : Type builtin_type = GDScriptParser : : get_builtin_type ( function_name ) ;
if ( builtin_type < Variant : : VARIANT_MAX ) {
// Is a builtin constructor.
call_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
call_type . kind = GDScriptParser : : DataType : : BUILTIN ;
call_type . builtin_type = builtin_type ;
if ( builtin_type = = Variant : : OBJECT ) {
call_type . kind = GDScriptParser : : DataType : : NATIVE ;
call_type . native_type = function_name ; // "Object".
}
2020-11-30 13:42:22 +01:00
bool safe_to_fold = true ;
switch ( builtin_type ) {
// Those are stored by reference so not suited for compile-time construction.
// Because in this case they would be the same reference in all constructed values.
case Variant : : OBJECT :
2020-12-25 08:03:55 +01:00
case Variant : : DICTIONARY :
case Variant : : ARRAY :
2020-11-30 13:42:22 +01:00
case Variant : : PACKED_BYTE_ARRAY :
case Variant : : PACKED_INT32_ARRAY :
case Variant : : PACKED_INT64_ARRAY :
case Variant : : PACKED_FLOAT32_ARRAY :
case Variant : : PACKED_FLOAT64_ARRAY :
case Variant : : PACKED_STRING_ARRAY :
case Variant : : PACKED_VECTOR2_ARRAY :
case Variant : : PACKED_VECTOR3_ARRAY :
case Variant : : PACKED_COLOR_ARRAY :
safe_to_fold = false ;
break ;
default :
break ;
}
if ( all_is_constant & & safe_to_fold ) {
2020-06-11 00:53:25 +02:00
// Construct here.
Vector < const Variant * > args ;
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
args . push_back ( & ( p_call - > arguments [ i ] - > reduced_value ) ) ;
}
Callable : : CallError err ;
2020-11-09 04:19:09 +01:00
Variant value ;
Variant : : construct ( builtin_type , value , ( const Variant * * ) args . ptr ( ) , args . size ( ) , err ) ;
2020-06-11 00:53:25 +02:00
switch ( err . error ) {
case Callable : : CallError : : CALL_ERROR_INVALID_ARGUMENT :
2022-12-04 04:02:03 +01:00
push_error ( vformat ( R " (Invalid argument for %s constructor: argument %d should be " % s " but is " % s " .) " , Variant : : get_type_name ( builtin_type ) , err . argument + 1 ,
2020-06-11 00:53:25 +02:00
Variant : : get_type_name ( Variant : : Type ( err . expected ) ) , p_call - > arguments [ err . argument ] - > get_datatype ( ) . to_string ( ) ) ,
p_call - > arguments [ err . argument ] ) ;
break ;
case Callable : : CallError : : CALL_ERROR_INVALID_METHOD : {
String signature = Variant : : get_type_name ( builtin_type ) + " ( " ;
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
if ( i > 0 ) {
signature + = " , " ;
}
signature + = p_call - > arguments [ i ] - > get_datatype ( ) . to_string ( ) ;
}
2020-08-18 18:12:18 +02:00
signature + = " ) " ;
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (No constructor of " % s " matches the signature " % s " .) " , Variant : : get_type_name ( builtin_type ) , signature ) , p_call - > callee ) ;
} break ;
case Callable : : CallError : : CALL_ERROR_TOO_MANY_ARGUMENTS :
push_error ( vformat ( R " (Too many arguments for %s constructor. Received %d but expected %d.) " , Variant : : get_type_name ( builtin_type ) , p_call - > arguments . size ( ) , err . expected ) , p_call ) ;
break ;
case Callable : : CallError : : CALL_ERROR_TOO_FEW_ARGUMENTS :
push_error ( vformat ( R " (Too few arguments for %s constructor. Received %d but expected %d.) " , Variant : : get_type_name ( builtin_type ) , p_call - > arguments . size ( ) , err . expected ) , p_call ) ;
break ;
case Callable : : CallError : : CALL_ERROR_INSTANCE_IS_NULL :
2022-06-27 22:10:04 +02:00
case Callable : : CallError : : CALL_ERROR_METHOD_NOT_CONST :
2020-06-11 00:53:25 +02:00
break ; // Can't happen in a builtin constructor.
case Callable : : CallError : : CALL_OK :
p_call - > is_constant = true ;
p_call - > reduced_value = value ;
break ;
}
} else {
// TODO: Check constructors without constants.
// If there's one argument, try to use copy constructor (those aren't explicitly defined).
if ( p_call - > arguments . size ( ) = = 1 ) {
GDScriptParser : : DataType arg_type = p_call - > arguments [ 0 ] - > get_datatype ( ) ;
if ( arg_type . is_variant ( ) ) {
mark_node_unsafe ( p_call - > arguments [ 0 ] ) ;
} else {
if ( arg_type . kind = = GDScriptParser : : DataType : : BUILTIN & & arg_type . builtin_type = = builtin_type ) {
// Okay.
p_call - > set_datatype ( call_type ) ;
return ;
}
}
}
List < MethodInfo > constructors ;
Variant : : get_constructor_list ( builtin_type , & constructors ) ;
bool match = false ;
2021-07-16 05:45:57 +02:00
for ( const MethodInfo & info : constructors ) {
2020-06-11 00:53:25 +02:00
if ( p_call - > arguments . size ( ) < info . arguments . size ( ) - info . default_arguments . size ( ) ) {
continue ;
}
if ( p_call - > arguments . size ( ) > info . arguments . size ( ) ) {
continue ;
}
bool types_match = true ;
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
GDScriptParser : : DataType par_type = type_from_property ( info . arguments [ i ] ) ;
2020-08-18 18:12:18 +02:00
if ( ! is_type_compatible ( par_type , p_call - > arguments [ i ] - > get_datatype ( ) , true ) ) {
2020-06-11 00:53:25 +02:00
types_match = false ;
break ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-11 00:53:25 +02:00
} else {
2022-03-12 13:40:07 +01:00
if ( par_type . builtin_type = = Variant : : INT & & p_call - > arguments [ i ] - > get_datatype ( ) . builtin_type = = Variant : : FLOAT & & builtin_type ! = Variant : : INT ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_call , GDScriptWarning : : NARROWING_CONVERSION , p_call - > function_name ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
}
if ( types_match ) {
match = true ;
call_type = type_from_property ( info . return_val ) ;
break ;
}
}
if ( ! match ) {
String signature = Variant : : get_type_name ( builtin_type ) + " ( " ;
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
if ( i > 0 ) {
signature + = " , " ;
}
signature + = p_call - > arguments [ i ] - > get_datatype ( ) . to_string ( ) ;
}
2020-08-18 18:12:18 +02:00
signature + = " ) " ;
2020-06-11 00:53:25 +02:00
push_error ( vformat ( R " (No constructor of " % s " matches the signature " % s " .) " , Variant : : get_type_name ( builtin_type ) , signature ) , p_call ) ;
}
}
p_call - > set_datatype ( call_type ) ;
return ;
2020-11-26 15:56:32 +01:00
} else if ( GDScriptUtilityFunctions : : function_exists ( function_name ) ) {
MethodInfo function_info = GDScriptUtilityFunctions : : get_function_info ( function_name ) ;
2023-01-09 13:55:05 +01:00
if ( ! p_is_root & & ! p_is_await & & function_info . return_val . type = = Variant : : NIL & & ( ( function_info . return_val . usage & PROPERTY_USAGE_NIL_IS_VARIANT ) = = 0 ) ) {
2022-12-30 15:57:25 +01:00
push_error ( vformat ( R " *(Cannot get return value of call to " % s ( ) " because it returns " void " .)* " , function_name ) , p_call ) ;
}
2020-11-26 15:56:32 +01:00
if ( all_is_constant & & GDScriptUtilityFunctions : : is_function_constant ( function_name ) ) {
// Can call on compilation.
Vector < const Variant * > args ;
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
args . push_back ( & ( p_call - > arguments [ i ] - > reduced_value ) ) ;
}
Variant value ;
Callable : : CallError err ;
GDScriptUtilityFunctions : : get_function ( function_name ) ( & value , ( const Variant * * ) args . ptr ( ) , args . size ( ) , err ) ;
switch ( err . error ) {
case Callable : : CallError : : CALL_ERROR_INVALID_ARGUMENT : {
PropertyInfo wrong_arg = function_info . arguments [ err . argument ] ;
2022-12-04 04:02:03 +01:00
push_error ( vformat ( R " *(Invalid argument for " % s ( ) " function: argument %d should be " % s " but is " % s " .)* " , function_name , err . argument + 1 ,
2020-11-26 15:56:32 +01:00
type_from_property ( wrong_arg ) . to_string ( ) , p_call - > arguments [ err . argument ] - > get_datatype ( ) . to_string ( ) ) ,
p_call - > arguments [ err . argument ] ) ;
} break ;
case Callable : : CallError : : CALL_ERROR_INVALID_METHOD :
push_error ( vformat ( R " (Invalid call for function " % s " .) " , function_name ) , p_call ) ;
break ;
case Callable : : CallError : : CALL_ERROR_TOO_MANY_ARGUMENTS :
push_error ( vformat ( R " *(Too many arguments for " % s ( ) " call. Expected at most %d but received %d.)* " , function_name , err . expected , p_call - > arguments . size ( ) ) , p_call ) ;
break ;
case Callable : : CallError : : CALL_ERROR_TOO_FEW_ARGUMENTS :
push_error ( vformat ( R " *(Too few arguments for " % s ( ) " call. Expected at least %d but received %d.)* " , function_name , err . expected , p_call - > arguments . size ( ) ) , p_call ) ;
break ;
2022-06-27 22:10:04 +02:00
case Callable : : CallError : : CALL_ERROR_METHOD_NOT_CONST :
2020-11-26 15:56:32 +01:00
case Callable : : CallError : : CALL_ERROR_INSTANCE_IS_NULL :
break ; // Can't happen in a builtin constructor.
case Callable : : CallError : : CALL_OK :
p_call - > is_constant = true ;
p_call - > reduced_value = value ;
break ;
}
} else {
validate_call_arg ( function_info , p_call ) ;
}
p_call - > set_datatype ( type_from_property ( function_info . return_val ) ) ;
return ;
} else if ( Variant : : has_utility_function ( function_name ) ) {
MethodInfo function_info = info_from_utility_func ( function_name ) ;
2020-06-11 00:53:25 +02:00
2023-01-09 13:55:05 +01:00
if ( ! p_is_root & & ! p_is_await & & function_info . return_val . type = = Variant : : NIL & & ( ( function_info . return_val . usage & PROPERTY_USAGE_NIL_IS_VARIANT ) = = 0 ) ) {
2022-12-30 15:57:25 +01:00
push_error ( vformat ( R " *(Cannot get return value of call to " % s ( ) " because it returns " void " .)* " , function_name ) , p_call ) ;
}
2020-11-26 15:56:32 +01:00
if ( all_is_constant & & Variant : : get_utility_function_type ( function_name ) = = Variant : : UTILITY_FUNC_TYPE_MATH ) {
2020-06-11 00:53:25 +02:00
// Can call on compilation.
Vector < const Variant * > args ;
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
args . push_back ( & ( p_call - > arguments [ i ] - > reduced_value ) ) ;
}
Variant value ;
Callable : : CallError err ;
2020-11-26 15:56:32 +01:00
Variant : : call_utility_function ( function_name , & value , ( const Variant * * ) args . ptr ( ) , args . size ( ) , err ) ;
2020-06-11 00:53:25 +02:00
switch ( err . error ) {
case Callable : : CallError : : CALL_ERROR_INVALID_ARGUMENT : {
2022-10-30 22:24:39 +01:00
String expected_type_name ;
if ( err . argument < function_info . arguments . size ( ) ) {
expected_type_name = type_from_property ( function_info . arguments [ err . argument ] ) . to_string ( ) ;
} else {
expected_type_name = Variant : : get_type_name ( ( Variant : : Type ) err . expected ) ;
}
2022-12-04 04:02:03 +01:00
push_error ( vformat ( R " *(Invalid argument for " % s ( ) " function: argument %d should be " % s " but is " % s " .)* " , function_name , err . argument + 1 ,
2022-10-30 22:24:39 +01:00
expected_type_name , p_call - > arguments [ err . argument ] - > get_datatype ( ) . to_string ( ) ) ,
2020-06-11 00:53:25 +02:00
p_call - > arguments [ err . argument ] ) ;
} break ;
case Callable : : CallError : : CALL_ERROR_INVALID_METHOD :
2020-11-26 15:56:32 +01:00
push_error ( vformat ( R " (Invalid call for function " % s " .) " , function_name ) , p_call ) ;
2020-06-11 00:53:25 +02:00
break ;
case Callable : : CallError : : CALL_ERROR_TOO_MANY_ARGUMENTS :
2020-11-26 15:56:32 +01:00
push_error ( vformat ( R " *(Too many arguments for " % s ( ) " call. Expected at most %d but received %d.)* " , function_name , err . expected , p_call - > arguments . size ( ) ) , p_call ) ;
2020-06-11 00:53:25 +02:00
break ;
case Callable : : CallError : : CALL_ERROR_TOO_FEW_ARGUMENTS :
2020-11-26 15:56:32 +01:00
push_error ( vformat ( R " *(Too few arguments for " % s ( ) " call. Expected at least %d but received %d.)* " , function_name , err . expected , p_call - > arguments . size ( ) ) , p_call ) ;
2020-06-11 00:53:25 +02:00
break ;
2022-06-27 22:10:04 +02:00
case Callable : : CallError : : CALL_ERROR_METHOD_NOT_CONST :
2020-06-11 00:53:25 +02:00
case Callable : : CallError : : CALL_ERROR_INSTANCE_IS_NULL :
break ; // Can't happen in a builtin constructor.
case Callable : : CallError : : CALL_OK :
p_call - > is_constant = true ;
p_call - > reduced_value = value ;
break ;
}
} else {
validate_call_arg ( function_info , p_call ) ;
}
2020-06-12 00:31:28 +02:00
p_call - > set_datatype ( type_from_property ( function_info . return_val ) ) ;
2020-06-11 00:53:25 +02:00
return ;
}
}
GDScriptParser : : DataType base_type ;
2020-06-12 00:31:28 +02:00
call_type . kind = GDScriptParser : : DataType : : VARIANT ;
2020-06-11 00:53:25 +02:00
bool is_self = false ;
if ( p_call - > is_super ) {
base_type = parser - > current_class - > base_type ;
2021-05-26 20:33:18 +02:00
base_type . is_meta_type = false ;
2020-06-11 00:53:25 +02:00
is_self = true ;
2022-03-30 16:53:49 +02:00
if ( p_call - > callee = = nullptr & & ! lambda_stack . is_empty ( ) ) {
push_error ( " Cannot use `super()` inside a lambda. " , p_call ) ;
}
2020-08-05 21:41:46 +02:00
} else if ( callee_type = = GDScriptParser : : Node : : IDENTIFIER ) {
2020-06-11 00:53:25 +02:00
base_type = parser - > current_class - > get_datatype ( ) ;
2021-05-26 20:33:18 +02:00
base_type . is_meta_type = false ;
2020-06-11 00:53:25 +02:00
is_self = true ;
2020-08-05 21:41:46 +02:00
} else if ( callee_type = = GDScriptParser : : Node : : SUBSCRIPT ) {
2020-06-11 00:53:25 +02:00
GDScriptParser : : SubscriptNode * subscript = static_cast < GDScriptParser : : SubscriptNode * > ( p_call - > callee ) ;
2021-04-05 16:17:59 +02:00
if ( subscript - > base = = nullptr ) {
// Invalid syntax, error already set on parser.
p_call - > set_datatype ( call_type ) ;
mark_node_unsafe ( p_call ) ;
return ;
}
2020-06-11 00:53:25 +02:00
if ( ! subscript - > is_attribute ) {
2020-06-12 00:31:28 +02:00
// Invalid call. Error already sent in parser.
2020-06-11 00:53:25 +02:00
// TODO: Could check if Callable here.
2020-06-12 00:31:28 +02:00
p_call - > set_datatype ( call_type ) ;
2020-06-11 00:53:25 +02:00
mark_node_unsafe ( p_call ) ;
return ;
}
2021-05-16 16:48:53 +02:00
if ( subscript - > attribute = = nullptr ) {
// Invalid call. Error already sent in parser.
p_call - > set_datatype ( call_type ) ;
mark_node_unsafe ( p_call ) ;
return ;
}
2020-06-11 00:53:25 +02:00
2021-05-16 16:48:53 +02:00
GDScriptParser : : IdentifierNode * base_id = nullptr ;
if ( subscript - > base - > type = = GDScriptParser : : Node : : IDENTIFIER ) {
base_id = static_cast < GDScriptParser : : IdentifierNode * > ( subscript - > base ) ;
}
if ( base_id & & GDScriptParser : : get_builtin_type ( base_id - > name ) < Variant : : VARIANT_MAX ) {
base_type = make_builtin_meta_type ( GDScriptParser : : get_builtin_type ( base_id - > name ) ) ;
} else {
reduce_expression ( subscript - > base ) ;
base_type = subscript - > base - > get_datatype ( ) ;
}
2020-06-11 00:53:25 +02:00
} else {
2020-06-12 00:31:28 +02:00
// Invalid call. Error already sent in parser.
2020-06-11 00:53:25 +02:00
// TODO: Could check if Callable here too.
2020-06-12 00:31:28 +02:00
p_call - > set_datatype ( call_type ) ;
2020-06-11 00:53:25 +02:00
mark_node_unsafe ( p_call ) ;
return ;
}
bool is_static = false ;
bool is_vararg = false ;
int default_arg_count = 0 ;
GDScriptParser : : DataType return_type ;
List < GDScriptParser : : DataType > par_types ;
2022-03-06 15:09:12 +01:00
bool is_constructor = ( base_type . is_meta_type | | ( p_call - > callee & & p_call - > callee - > type = = GDScriptParser : : Node : : IDENTIFIER ) ) & & p_call - > function_name = = SNAME ( " new " ) ;
if ( get_function_signature ( p_call , is_constructor , base_type , p_call - > function_name , return_type , par_types , default_arg_count , is_static , is_vararg ) ) {
2021-03-09 16:32:35 +01:00
// If the function require typed arrays we must make literals be typed.
2021-08-09 22:13:42 +02:00
for ( const KeyValue < int , GDScriptParser : : ArrayNode * > & E : arrays ) {
int index = E . key ;
2021-03-09 16:32:35 +01:00
if ( index < par_types . size ( ) & & par_types [ index ] . has_container_element_type ( ) ) {
2021-08-09 22:13:42 +02:00
update_array_literal_element_type ( par_types [ index ] , E . value ) ;
2021-03-09 16:32:35 +01:00
}
}
2020-06-11 00:53:25 +02:00
validate_call_arg ( par_types , default_arg_count , is_vararg , p_call ) ;
2022-02-02 17:57:24 +01:00
if ( base_type . kind = = GDScriptParser : : DataType : : ENUM & & base_type . is_meta_type ) {
// Enum type is treated as a dictionary value for function calls.
base_type . is_meta_type = false ;
}
2020-06-11 00:53:25 +02:00
if ( is_self & & parser - > current_function ! = nullptr & & parser - > current_function - > is_static & & ! is_static ) {
2022-04-20 19:22:22 +02:00
// Get the parent function above any lambda.
GDScriptParser : : FunctionNode * parent_function = parser - > current_function ;
while ( parent_function - > source_lambda ) {
parent_function = parent_function - > source_lambda - > parent_function ;
}
push_error ( vformat ( R " *(Cannot call non-static function " % s ( ) " from static function " % s ( ) " .)* " , p_call - > function_name , parent_function - > identifier - > name ) , p_call ) ;
2021-05-26 14:23:17 +02:00
} else if ( ! is_self & & base_type . is_meta_type & & ! is_static ) {
base_type . is_meta_type = false ; // For `to_string()`.
2022-03-30 16:53:49 +02:00
push_error ( vformat ( R " *(Cannot call non-static function " % s ( ) " on the class " % s " directly. Make an instance instead.)* " , p_call - > function_name , base_type . to_string ( ) ) , p_call ) ;
2022-04-20 19:22:22 +02:00
} else if ( is_self & & ! is_static ) {
mark_lambda_use_self ( ) ;
2020-06-11 00:53:25 +02:00
}
2023-01-09 13:55:05 +01:00
if ( ! p_is_root & & ! p_is_await & & return_type . is_hard_type ( ) & & return_type . kind = = GDScriptParser : : DataType : : BUILTIN & & return_type . builtin_type = = Variant : : NIL ) {
2022-12-30 15:57:25 +01:00
push_error ( vformat ( R " *(Cannot get return value of call to " % s ( ) " because it returns " void " .)* " , p_call - > function_name ) , p_call ) ;
}
2022-10-13 19:31:12 +02:00
# ifdef DEBUG_ENABLED
if ( p_is_root & & return_type . kind ! = GDScriptParser : : DataType : : UNRESOLVED & & return_type . builtin_type ! = Variant : : NIL ) {
parser - > push_warning ( p_call , GDScriptWarning : : RETURN_VALUE_DISCARDED , p_call - > function_name ) ;
}
2022-10-13 21:42:11 +02:00
2022-11-24 21:06:11 +01:00
if ( is_static & & ! base_type . is_meta_type & & ! ( is_self & & parser - > current_function ! = nullptr & & parser - > current_function - > is_static ) ) {
String caller_type = String ( base_type . native_type ) ;
if ( caller_type . is_empty ( ) ) {
caller_type = base_type . to_string ( ) ;
}
parser - > push_warning ( p_call , GDScriptWarning : : STATIC_CALLED_ON_INSTANCE , p_call - > function_name , caller_type ) ;
2022-10-13 21:42:11 +02:00
}
2022-10-13 19:31:12 +02:00
# endif // DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
call_type = return_type ;
2020-06-11 00:53:25 +02:00
} else {
2020-06-12 00:31:28 +02:00
bool found = false ;
2022-04-06 19:14:38 +02:00
2022-12-04 04:02:03 +01:00
// Enums do not have functions other than the built-in dictionary ones.
if ( base_type . kind = = GDScriptParser : : DataType : : ENUM & & base_type . is_meta_type ) {
push_error ( vformat ( R " *(Enums only have Dictionary built-in methods. Function " % s ( ) " does not exist for enum " % s " .)* " , p_call - > function_name , base_type . enum_type ) , p_call - > callee ) ;
} else if ( ! p_call - > is_super & & callee_type ! = GDScriptParser : : Node : : NONE ) { // Check if the name exists as something else.
2020-06-12 00:31:28 +02:00
GDScriptParser : : IdentifierNode * callee_id ;
2020-08-05 21:41:46 +02:00
if ( callee_type = = GDScriptParser : : Node : : IDENTIFIER ) {
2020-06-12 00:31:28 +02:00
callee_id = static_cast < GDScriptParser : : IdentifierNode * > ( p_call - > callee ) ;
} else {
// Can only be attribute.
callee_id = static_cast < GDScriptParser : : SubscriptNode * > ( p_call - > callee ) - > attribute ;
}
2020-07-16 03:02:44 +02:00
if ( callee_id ) {
reduce_identifier_from_base ( callee_id , & base_type ) ;
2020-08-05 21:41:46 +02:00
GDScriptParser : : DataType callee_datatype = callee_id - > get_datatype ( ) ;
if ( callee_datatype . is_set ( ) & & ! callee_datatype . is_variant ( ) ) {
2020-07-16 03:02:44 +02:00
found = true ;
2020-08-05 21:41:46 +02:00
if ( callee_datatype . builtin_type = = Variant : : CALLABLE ) {
2020-07-16 03:02:44 +02:00
push_error ( vformat ( R " *(Name " % s " is a Callable. You can call it with " % s . call ( ) " instead.)* " , p_call - > function_name , p_call - > function_name ) , p_call - > callee ) ;
} else {
2020-08-05 21:41:46 +02:00
push_error ( vformat ( R " *(Name " % s " called as a function but is a " % s " .)* " , p_call - > function_name , callee_datatype . to_string ( ) ) , p_call - > callee ) ;
2020-07-16 03:02:44 +02:00
}
# ifdef DEBUG_ENABLED
2021-01-03 14:14:01 +01:00
} else if ( ! is_self & & ! ( base_type . is_hard_type ( ) & & base_type . kind = = GDScriptParser : : DataType : : BUILTIN ) ) {
2020-07-16 03:02:44 +02:00
parser - > push_warning ( p_call , GDScriptWarning : : UNSAFE_METHOD_ACCESS , p_call - > function_name , base_type . to_string ( ) ) ;
mark_node_unsafe ( p_call ) ;
# endif
2020-06-12 00:31:28 +02:00
}
}
}
2021-01-03 14:14:01 +01:00
if ( ! found & & ( is_self | | ( base_type . is_hard_type ( ) & & base_type . kind = = GDScriptParser : : DataType : : BUILTIN ) ) ) {
2020-06-12 00:31:28 +02:00
String base_name = is_self & & ! p_call - > is_super ? " self " : base_type . to_string ( ) ;
push_error ( vformat ( R " *(Function " % s ( ) " not found in base %s.)* " , p_call - > function_name , base_name ) , p_call - > is_super ? p_call : p_call - > callee ) ;
2022-04-06 19:14:38 +02:00
} else if ( ! found & & ( ! p_call - > is_super & & base_type . is_hard_type ( ) & & base_type . kind = = GDScriptParser : : DataType : : NATIVE & & base_type . is_meta_type ) ) {
2022-12-04 04:02:03 +01:00
push_error ( vformat ( R " *(Static function " % s ( ) " not found in base " % s " .)* " , p_call - > function_name , base_type . native_type ) , p_call ) ;
2020-06-12 00:31:28 +02:00
}
}
2021-10-15 01:12:01 +02:00
if ( call_type . is_coroutine & & ! p_is_await & & ! p_is_root ) {
2022-03-30 16:53:49 +02:00
push_error ( vformat ( R " *(Function " % s ( ) " is a coroutine, so it must be called with " await " .)* " , p_call - > function_name ) , p_call ) ;
2020-06-11 00:53:25 +02:00
}
2020-06-12 00:31:28 +02:00
p_call - > set_datatype ( call_type ) ;
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : reduce_cast ( GDScriptParser : : CastNode * p_cast ) {
reduce_expression ( p_cast - > operand ) ;
2023-01-12 00:03:53 +01:00
GDScriptParser : : DataType cast_type = type_from_metatype ( resolve_datatype ( p_cast - > cast_type ) ) ;
2020-06-11 00:53:25 +02:00
if ( ! cast_type . is_set ( ) ) {
2022-01-27 15:34:33 +01:00
mark_node_unsafe ( p_cast ) ;
2020-06-11 00:53:25 +02:00
return ;
}
p_cast - > set_datatype ( cast_type ) ;
if ( ! cast_type . is_variant ( ) ) {
2020-07-16 03:02:44 +02:00
GDScriptParser : : DataType op_type = p_cast - > operand - > get_datatype ( ) ;
if ( ! op_type . is_variant ( ) ) {
bool valid = false ;
2022-12-04 04:02:03 +01:00
bool more_informative_error = false ;
2022-01-27 15:34:33 +01:00
if ( op_type . kind = = GDScriptParser : : DataType : : ENUM & & cast_type . kind = = GDScriptParser : : DataType : : ENUM ) {
2022-12-04 04:02:03 +01:00
// Enum casts are compatible when value from operand exists in target enum
if ( p_cast - > operand - > is_constant & & p_cast - > operand - > reduced ) {
if ( enum_get_value_name ( cast_type , p_cast - > operand - > reduced_value ) ! = StringName ( ) ) {
valid = true ;
} else {
valid = false ;
more_informative_error = true ;
push_error ( vformat ( R " (Invalid cast. Enum " % s " does not have value corresponding to " % s . % s " (%d).) " ,
cast_type . to_string ( ) , op_type . enum_type ,
enum_get_value_name ( op_type , p_cast - > operand - > reduced_value ) , // Can never be null
p_cast - > operand - > reduced_value . operator uint64_t ( ) ) ,
p_cast - > cast_type ) ;
}
} else {
// Can't statically tell whether int has a corresponding enum value. Valid but dangerous!
mark_node_unsafe ( p_cast ) ;
valid = true ;
}
2022-01-27 15:34:33 +01:00
} else if ( op_type . kind = = GDScriptParser : : DataType : : BUILTIN & & op_type . builtin_type = = Variant : : INT & & cast_type . kind = = GDScriptParser : : DataType : : ENUM ) {
2022-12-04 04:02:03 +01:00
// Int assignment to enum not valid when exact int assigned is known but is not an enum value
if ( p_cast - > operand - > is_constant & & p_cast - > operand - > reduced ) {
if ( enum_get_value_name ( cast_type , p_cast - > operand - > reduced_value ) ! = StringName ( ) ) {
valid = true ;
} else {
valid = false ;
more_informative_error = true ;
push_error ( vformat ( R " (Invalid cast. Enum " % s " does not have enum value %d.) " , cast_type . to_string ( ) , p_cast - > operand - > reduced_value . operator uint64_t ( ) ) , p_cast - > cast_type ) ;
}
} else {
// Can't statically tell whether int has a corresponding enum value. Valid but dangerous!
mark_node_unsafe ( p_cast ) ;
valid = true ;
}
2022-01-27 15:34:33 +01:00
} else if ( op_type . kind = = GDScriptParser : : DataType : : BUILTIN & & cast_type . kind = = GDScriptParser : : DataType : : BUILTIN ) {
2020-07-16 03:02:44 +02:00
valid = Variant : : can_convert ( op_type . builtin_type , cast_type . builtin_type ) ;
} else if ( op_type . kind ! = GDScriptParser : : DataType : : BUILTIN & & cast_type . kind ! = GDScriptParser : : DataType : : BUILTIN ) {
valid = is_type_compatible ( cast_type , op_type ) | | is_type_compatible ( op_type , cast_type ) ;
}
2020-06-11 00:53:25 +02:00
2022-12-04 04:02:03 +01:00
if ( ! valid & & ! more_informative_error ) {
2020-07-16 03:02:44 +02:00
push_error ( vformat ( R " (Invalid cast. Cannot convert from " % s " to " % s " .) " , op_type . to_string ( ) , cast_type . to_string ( ) ) , p_cast - > cast_type ) ;
2020-06-11 00:53:25 +02:00
}
}
2020-07-16 03:02:44 +02:00
} else {
mark_node_unsafe ( p_cast ) ;
2020-06-11 00:53:25 +02:00
}
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
if ( p_cast - > operand - > get_datatype ( ) . is_variant ( ) ) {
parser - > push_warning ( p_cast , GDScriptWarning : : UNSAFE_CAST , cast_type . to_string ( ) ) ;
mark_node_unsafe ( p_cast ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
// TODO: Perform cast on constants.
}
void GDScriptAnalyzer : : reduce_dictionary ( GDScriptParser : : DictionaryNode * p_dictionary ) {
2022-12-06 03:46:47 +01:00
HashMap < Variant , GDScriptParser : : ExpressionNode * , VariantHasher , StringLikeVariantComparator > elements ;
2020-08-19 16:14:16 +02:00
2020-06-11 00:53:25 +02:00
for ( int i = 0 ; i < p_dictionary - > elements . size ( ) ; i + + ) {
const GDScriptParser : : DictionaryNode : : Pair & element = p_dictionary - > elements [ i ] ;
2020-07-16 03:02:44 +02:00
if ( p_dictionary - > style = = GDScriptParser : : DictionaryNode : : PYTHON_DICT ) {
reduce_expression ( element . key ) ;
}
2020-06-11 00:53:25 +02:00
reduce_expression ( element . value ) ;
2020-08-19 16:14:16 +02:00
if ( element . key - > is_constant ) {
if ( elements . has ( element . key - > reduced_value ) ) {
push_error ( vformat ( R " (Key " % s " was already used in this dictionary (at line %d).) " , element . key - > reduced_value , elements [ element . key - > reduced_value ] - > start_line ) , element . key ) ;
} else {
elements [ element . key - > reduced_value ] = element . value ;
}
}
2020-06-11 00:53:25 +02:00
}
// It's dictionary in any case.
GDScriptParser : : DataType dict_type ;
dict_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
dict_type . kind = GDScriptParser : : DataType : : BUILTIN ;
dict_type . builtin_type = Variant : : DICTIONARY ;
dict_type . is_constant = true ;
p_dictionary - > set_datatype ( dict_type ) ;
}
void GDScriptAnalyzer : : reduce_get_node ( GDScriptParser : : GetNodeNode * p_get_node ) {
GDScriptParser : : DataType result ;
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
result . kind = GDScriptParser : : DataType : : NATIVE ;
2022-02-06 14:12:19 +01:00
result . native_type = SNAME ( " Node " ) ;
2020-06-11 00:53:25 +02:00
result . builtin_type = Variant : : OBJECT ;
2021-08-17 15:06:54 +02:00
if ( ! ClassDB : : is_parent_class ( parser - > current_class - > base_type . native_type , result . native_type ) ) {
2020-06-11 00:53:25 +02:00
push_error ( R " *(Cannot use shorthand " get_node ( ) " notation ( " $ " ) on a class that isn't a node.)* " , p_get_node ) ;
}
2022-04-20 19:22:22 +02:00
mark_lambda_use_self ( ) ;
2020-06-11 00:53:25 +02:00
p_get_node - > set_datatype ( result ) ;
}
2020-11-26 16:27:48 +01:00
GDScriptParser : : DataType GDScriptAnalyzer : : make_global_class_meta_type ( const StringName & p_class_name , const GDScriptParser : : Node * p_source ) {
2022-02-28 12:20:00 +01:00
GDScriptParser : : DataType type ;
2022-04-25 06:33:18 +02:00
String path = ScriptServer : : get_global_class_path ( p_class_name ) ;
String ext = path . get_extension ( ) ;
if ( ext = = GDScriptLanguage : : get_singleton ( ) - > get_extension ( ) ) {
Ref < GDScriptParserRef > ref = get_parser_for ( path ) ;
if ( ref . is_null ( ) ) {
push_error ( vformat ( R " (Could not find script for class " % s " .) " , p_class_name ) , p_source ) ;
type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
type . kind = GDScriptParser : : DataType : : VARIANT ;
return type ;
}
2022-12-11 03:57:35 +01:00
Error err = ref - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2022-04-25 06:33:18 +02:00
if ( err ) {
push_error ( vformat ( R " (Could not resolve class " % s " , because of a parser error.) " , p_class_name ) , p_source ) ;
type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
type . kind = GDScriptParser : : DataType : : VARIANT ;
return type ;
}
2022-11-25 11:21:15 +01:00
return ref - > get_parser ( ) - > head - > get_datatype ( ) ;
2022-04-25 06:33:18 +02:00
} else {
2023-01-12 00:03:53 +01:00
return make_script_meta_type ( ResourceLoader : : load ( path , " Script " ) ) ;
2020-11-26 16:27:48 +01:00
}
2020-06-11 00:53:25 +02:00
}
2022-12-18 06:38:53 +01:00
void GDScriptAnalyzer : : reduce_identifier_from_base_set_class ( GDScriptParser : : IdentifierNode * p_identifier , GDScriptParser : : DataType p_identifier_datatype ) {
ERR_FAIL_NULL ( p_identifier ) ;
p_identifier - > set_datatype ( p_identifier_datatype ) ;
Error err = OK ;
GDScript * scr = GDScriptCache : : get_full_script ( p_identifier_datatype . script_path , err ) . ptr ( ) ;
ERR_FAIL_COND_MSG ( err ! = OK , " Error while getting full script. " ) ;
scr = scr - > find_class ( p_identifier_datatype . class_type - > fqcn ) ;
p_identifier - > reduced_value = scr ;
p_identifier - > is_constant = true ;
}
2020-06-11 00:53:25 +02:00
void GDScriptAnalyzer : : reduce_identifier_from_base ( GDScriptParser : : IdentifierNode * p_identifier , GDScriptParser : : DataType * p_base ) {
2022-12-11 03:57:35 +01:00
if ( ! p_identifier - > get_datatype ( ) . has_no_type ( ) ) {
return ;
}
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType base ;
if ( p_base = = nullptr ) {
base = type_from_metatype ( parser - > current_class - > get_datatype ( ) ) ;
} else {
base = * p_base ;
}
2020-06-12 02:49:58 +02:00
const StringName & name = p_identifier - > name ;
2022-02-02 17:57:24 +01:00
if ( base . kind = = GDScriptParser : : DataType : : ENUM ) {
if ( base . is_meta_type ) {
if ( base . enum_values . has ( name ) ) {
2022-12-04 04:02:03 +01:00
p_identifier - > set_datatype ( type_from_metatype ( base ) ) ;
2022-02-02 17:57:24 +01:00
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = base . enum_values [ name ] ;
return ;
}
2022-12-04 04:02:03 +01:00
// Enum does not have this value, return.
return ;
2022-02-02 17:57:24 +01:00
} else {
push_error ( R " (Cannot get property from enum value.) " , p_identifier ) ;
return ;
}
}
2020-06-12 02:49:58 +02:00
if ( base . kind = = GDScriptParser : : DataType : : BUILTIN ) {
if ( base . is_meta_type ) {
bool valid = true ;
Variant result = Variant : : get_constant_value ( base . builtin_type , name , & valid ) ;
if ( valid ) {
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = result ;
2020-08-31 14:53:02 +02:00
p_identifier - > set_datatype ( type_from_variant ( result , p_identifier ) ) ;
2021-09-17 16:52:30 +02:00
} else if ( base . is_hard_type ( ) ) {
2020-06-12 02:49:58 +02:00
push_error ( vformat ( R " (Cannot find constant " % s " on type " % s " .) " , name , base . to_string ( ) ) , p_identifier ) ;
}
} else {
2020-09-02 17:08:49 +02:00
switch ( base . builtin_type ) {
case Variant : : NIL : {
2021-09-17 16:52:30 +02:00
if ( base . is_hard_type ( ) ) {
push_error ( vformat ( R " (Invalid get index " % s " on base Nil) " , name ) , p_identifier ) ;
}
2020-06-12 02:49:58 +02:00
return ;
}
2020-09-02 17:08:49 +02:00
case Variant : : DICTIONARY : {
GDScriptParser : : DataType dummy ;
dummy . kind = GDScriptParser : : DataType : : VARIANT ;
p_identifier - > set_datatype ( dummy ) ;
return ;
}
default : {
Callable : : CallError temp ;
2020-11-09 04:19:09 +01:00
Variant dummy ;
Variant : : construct ( base . builtin_type , dummy , nullptr , 0 , temp ) ;
2020-09-02 17:08:49 +02:00
List < PropertyInfo > properties ;
dummy . get_property_list ( & properties ) ;
2021-07-16 05:45:57 +02:00
for ( const PropertyInfo & prop : properties ) {
2020-09-02 17:08:49 +02:00
if ( prop . name = = name ) {
p_identifier - > set_datatype ( type_from_property ( prop ) ) ;
return ;
}
}
2021-09-17 16:52:30 +02:00
if ( base . is_hard_type ( ) ) {
push_error ( vformat ( R " (Cannot find property " % s " on base " % s " .) " , name , base . to_string ( ) ) , p_identifier ) ;
}
2020-09-02 17:08:49 +02:00
}
2020-06-12 02:49:58 +02:00
}
}
return ;
}
2020-06-11 00:53:25 +02:00
GDScriptParser : : ClassNode * base_class = base . class_type ;
2022-12-18 06:38:53 +01:00
List < GDScriptParser : : ClassNode * > script_classes ;
bool is_base = true ;
2020-06-11 00:53:25 +02:00
2022-12-18 06:38:53 +01:00
if ( base_class ! = nullptr ) {
get_class_node_current_scope_classes ( base_class , & script_classes ) ;
}
for ( GDScriptParser : : ClassNode * script_class : script_classes ) {
if ( p_base = = nullptr & & script_class - > identifier & & script_class - > identifier - > name = = name ) {
reduce_identifier_from_base_set_class ( p_identifier , script_class - > get_datatype ( ) ) ;
2020-07-16 03:02:44 +02:00
return ;
}
2022-12-11 03:57:35 +01:00
2022-12-18 06:38:53 +01:00
if ( script_class - > has_member ( name ) ) {
resolve_class_member ( script_class , name , p_identifier ) ;
2022-12-11 03:57:35 +01:00
2022-12-18 06:38:53 +01:00
GDScriptParser : : ClassNode : : Member member = script_class - > get_member ( name ) ;
2020-06-12 02:49:58 +02:00
switch ( member . type ) {
2022-12-18 06:38:53 +01:00
case GDScriptParser : : ClassNode : : Member : : CONSTANT : {
p_identifier - > set_datatype ( member . get_datatype ( ) ) ;
2020-06-12 02:49:58 +02:00
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = member . constant - > initializer - > reduced_value ;
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT ;
p_identifier - > constant_source = member . constant ;
2022-12-18 06:38:53 +01:00
return ;
}
case GDScriptParser : : ClassNode : : Member : : ENUM_VALUE : {
p_identifier - > set_datatype ( member . get_datatype ( ) ) ;
2020-06-12 02:49:58 +02:00
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = member . enum_value . value ;
2021-03-26 13:03:16 +01:00
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT ;
2022-12-18 06:38:53 +01:00
return ;
}
case GDScriptParser : : ClassNode : : Member : : ENUM : {
p_identifier - > set_datatype ( member . get_datatype ( ) ) ;
2022-12-27 04:06:11 +01:00
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = member . m_enum - > dictionary ;
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT ;
2022-12-04 04:02:03 +01:00
return ;
}
2022-12-18 06:38:53 +01:00
case GDScriptParser : : ClassNode : : Member : : VARIABLE : {
if ( is_base & & ! base . is_meta_type ) {
p_identifier - > set_datatype ( member . get_datatype ( ) ) ;
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_VARIABLE ;
p_identifier - > variable_source = member . variable ;
member . variable - > usages + = 1 ;
return ;
}
} break ;
2022-12-11 03:57:35 +01:00
2022-12-18 06:38:53 +01:00
case GDScriptParser : : ClassNode : : Member : : SIGNAL : {
if ( is_base & & ! base . is_meta_type ) {
p_identifier - > set_datatype ( member . get_datatype ( ) ) ;
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_SIGNAL ;
return ;
2022-12-04 22:55:40 +01:00
}
2022-12-18 06:38:53 +01:00
} break ;
case GDScriptParser : : ClassNode : : Member : : FUNCTION : {
if ( is_base & & ! base . is_meta_type ) {
p_identifier - > set_datatype ( make_callable_type ( member . function - > info ) ) ;
return ;
}
} break ;
case GDScriptParser : : ClassNode : : Member : : CLASS : {
reduce_identifier_from_base_set_class ( p_identifier , member . get_datatype ( ) ) ;
return ;
}
default : {
// Do nothing
2020-06-11 00:53:25 +02:00
}
}
}
2022-12-18 06:38:53 +01:00
if ( is_base ) {
is_base = script_class - > base_type . class_type ! = nullptr ;
if ( ! is_base & & p_base ! = nullptr ) {
break ;
}
}
2020-06-11 00:53:25 +02:00
}
2022-11-08 23:41:50 +01:00
// Check native members. No need for native class recursion because Node exposes all Object's properties.
2021-08-17 15:06:54 +02:00
const StringName & native = base . native_type ;
2020-06-11 00:53:25 +02:00
if ( class_exists ( native ) ) {
MethodInfo method_info ;
2021-09-21 20:29:06 +02:00
if ( ClassDB : : has_property ( native , name ) ) {
StringName getter_name = ClassDB : : get_property_getter ( native , name ) ;
MethodBind * getter = ClassDB : : get_method ( native , getter_name ) ;
if ( getter ! = nullptr ) {
p_identifier - > set_datatype ( type_from_property ( getter - > get_return_info ( ) ) ) ;
2022-04-20 19:22:22 +02:00
p_identifier - > source = GDScriptParser : : IdentifierNode : : INHERITED_VARIABLE ;
2021-09-21 20:29:06 +02:00
}
2020-06-11 00:53:25 +02:00
return ;
}
if ( ClassDB : : get_method_info ( native , name , & method_info ) ) {
// Method is callable.
p_identifier - > set_datatype ( make_callable_type ( method_info ) ) ;
2022-04-20 19:22:22 +02:00
p_identifier - > source = GDScriptParser : : IdentifierNode : : INHERITED_VARIABLE ;
2020-06-11 00:53:25 +02:00
return ;
}
if ( ClassDB : : get_signal ( native , name , & method_info ) ) {
// Signal is a type too.
p_identifier - > set_datatype ( make_signal_type ( method_info ) ) ;
2022-04-20 19:22:22 +02:00
p_identifier - > source = GDScriptParser : : IdentifierNode : : INHERITED_VARIABLE ;
2020-06-11 00:53:25 +02:00
return ;
}
2020-06-12 02:49:58 +02:00
if ( ClassDB : : has_enum ( native , name ) ) {
2022-12-04 04:02:03 +01:00
p_identifier - > set_datatype ( make_native_enum_type ( name , native ) ) ;
2022-04-20 19:22:22 +02:00
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT ;
2020-06-12 02:49:58 +02:00
return ;
}
2020-06-11 00:53:25 +02:00
bool valid = false ;
2022-12-04 04:02:03 +01:00
2022-05-09 11:47:10 +02:00
int64_t int_constant = ClassDB : : get_integer_constant ( native , name , & valid ) ;
2020-06-11 00:53:25 +02:00
if ( valid ) {
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = int_constant ;
2022-04-20 19:22:22 +02:00
p_identifier - > source = GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT ;
2022-12-04 04:02:03 +01:00
// Check whether this constant, which exists, belongs to an enum
StringName enum_name = ClassDB : : get_integer_constant_enum ( native , name ) ;
if ( enum_name ! = StringName ( ) ) {
p_identifier - > set_datatype ( make_native_enum_type ( enum_name , native , false ) ) ;
} else {
p_identifier - > set_datatype ( type_from_variant ( int_constant , p_identifier ) ) ;
}
2020-06-11 00:53:25 +02:00
}
}
}
2020-06-12 02:49:58 +02:00
void GDScriptAnalyzer : : reduce_identifier ( GDScriptParser : : IdentifierNode * p_identifier , bool can_be_builtin ) {
2022-12-04 04:02:03 +01:00
// TODO: This is an opportunity to further infer types.
2020-08-18 22:44:20 +02:00
2022-12-04 04:02:03 +01:00
// Check if we are inside an enum. This allows enum values to access other elements of the same enum.
2020-08-18 22:44:20 +02:00
if ( current_enum ) {
for ( int i = 0 ; i < current_enum - > values . size ( ) ; i + + ) {
const GDScriptParser : : EnumNode : : Value & element = current_enum - > values [ i ] ;
if ( element . identifier - > name = = p_identifier - > name ) {
2022-12-04 04:02:03 +01:00
StringName enum_name = current_enum - > identifier - > name ? current_enum - > identifier - > name : UNNAMED_ENUM ;
GDScriptParser : : DataType type = make_enum_type ( enum_name , parser - > current_class - > fqcn , false ) ;
2020-08-18 22:44:20 +02:00
if ( element . parent_enum - > identifier ) {
type . enum_type = element . parent_enum - > identifier - > name ;
}
p_identifier - > set_datatype ( type ) ;
if ( element . resolved ) {
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = element . value ;
} else {
push_error ( R " (Cannot use another enum element before it was declared.) " , p_identifier ) ;
}
return ; // Found anyway.
}
}
}
2021-03-26 13:03:16 +01:00
bool found_source = false ;
2020-06-11 00:53:25 +02:00
// Check if identifier is local.
// If that's the case, the declaration already was solved before.
switch ( p_identifier - > source ) {
case GDScriptParser : : IdentifierNode : : FUNCTION_PARAMETER :
p_identifier - > set_datatype ( p_identifier - > parameter_source - > get_datatype ( ) ) ;
2021-03-26 13:03:16 +01:00
found_source = true ;
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : IdentifierNode : : LOCAL_CONSTANT :
case GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT :
p_identifier - > set_datatype ( p_identifier - > constant_source - > get_datatype ( ) ) ;
p_identifier - > is_constant = true ;
// TODO: Constant should have a value on the node itself.
p_identifier - > reduced_value = p_identifier - > constant_source - > initializer - > reduced_value ;
2021-03-26 13:03:16 +01:00
found_source = true ;
break ;
2022-11-21 20:38:02 +01:00
case GDScriptParser : : IdentifierNode : : MEMBER_SIGNAL :
2022-04-20 19:22:22 +02:00
case GDScriptParser : : IdentifierNode : : INHERITED_VARIABLE :
mark_lambda_use_self ( ) ;
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : IdentifierNode : : MEMBER_VARIABLE :
2022-04-20 19:22:22 +02:00
mark_lambda_use_self ( ) ;
2020-06-12 00:31:28 +02:00
p_identifier - > variable_source - > usages + + ;
[[fallthrough]] ;
case GDScriptParser : : IdentifierNode : : LOCAL_VARIABLE :
2020-06-11 00:53:25 +02:00
p_identifier - > set_datatype ( p_identifier - > variable_source - > get_datatype ( ) ) ;
2021-03-26 13:03:16 +01:00
found_source = true ;
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : IdentifierNode : : LOCAL_ITERATOR :
p_identifier - > set_datatype ( p_identifier - > bind_source - > get_datatype ( ) ) ;
2021-03-26 13:03:16 +01:00
found_source = true ;
break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : IdentifierNode : : LOCAL_BIND : {
GDScriptParser : : DataType result = p_identifier - > bind_source - > get_datatype ( ) ;
result . is_constant = true ;
p_identifier - > set_datatype ( result ) ;
2021-03-26 13:03:16 +01:00
found_source = true ;
} break ;
2020-06-11 00:53:25 +02:00
case GDScriptParser : : IdentifierNode : : UNDEFINED_SOURCE :
break ;
}
// Not a local, so check members.
2021-03-26 13:03:16 +01:00
if ( ! found_source ) {
reduce_identifier_from_base ( p_identifier ) ;
if ( p_identifier - > source ! = GDScriptParser : : IdentifierNode : : UNDEFINED_SOURCE | | p_identifier - > get_datatype ( ) . is_set ( ) ) {
// Found.
found_source = true ;
}
}
if ( found_source ) {
2022-12-18 06:38:53 +01:00
bool source_is_variable = p_identifier - > source = = GDScriptParser : : IdentifierNode : : MEMBER_VARIABLE | | p_identifier - > source = = GDScriptParser : : IdentifierNode : : INHERITED_VARIABLE ;
bool source_is_signal = p_identifier - > source = = GDScriptParser : : IdentifierNode : : MEMBER_SIGNAL ;
if ( ( source_is_variable | | source_is_signal ) & & parser - > current_function & & parser - > current_function - > is_static ) {
2022-04-20 19:22:22 +02:00
// Get the parent function above any lambda.
GDScriptParser : : FunctionNode * parent_function = parser - > current_function ;
while ( parent_function - > source_lambda ) {
parent_function = parent_function - > source_lambda - > parent_function ;
}
2022-12-18 06:38:53 +01:00
push_error ( vformat ( R " *(Cannot access %s " % s " from the static function " % s ( ) " .)* " , source_is_signal ? " signal " : " instance variable " , p_identifier - > name , parent_function - > identifier - > name ) , p_identifier ) ;
2021-03-26 13:03:16 +01:00
}
2022-04-20 19:22:22 +02:00
if ( ! lambda_stack . is_empty ( ) ) {
2022-12-18 06:38:53 +01:00
// If the identifier is a member variable (including the native class properties) or a signal, we consider the lambda to be using `self`, so we keep a reference to the current instance.
if ( source_is_variable | | source_is_signal ) {
2022-04-20 19:22:22 +02:00
mark_lambda_use_self ( ) ;
return ; // No need to capture.
}
// If the identifier is local, check if it's any kind of capture by comparing their source function.
// Only capture locals and enum values. Constants are still accessible from the lambda using the script reference. If not, this method is done.
if ( p_identifier - > source = = GDScriptParser : : IdentifierNode : : UNDEFINED_SOURCE | | p_identifier - > source = = GDScriptParser : : IdentifierNode : : MEMBER_CONSTANT ) {
return ;
}
GDScriptParser : : FunctionNode * function_test = lambda_stack . back ( ) - > get ( ) - > function ;
// Make sure we aren't capturing variable in the same lambda.
// This also add captures for nested lambdas.
while ( function_test ! = nullptr & & function_test ! = p_identifier - > source_function & & function_test - > source_lambda ! = nullptr & & ! function_test - > source_lambda - > captures_indices . has ( p_identifier - > name ) ) {
function_test - > source_lambda - > captures_indices [ p_identifier - > name ] = function_test - > source_lambda - > captures . size ( ) ;
function_test - > source_lambda - > captures . push_back ( p_identifier ) ;
function_test = function_test - > source_lambda - > parent_function ;
}
2021-03-26 13:03:16 +01:00
}
2022-04-20 19:22:22 +02:00
2020-06-11 00:53:25 +02:00
return ;
}
StringName name = p_identifier - > name ;
p_identifier - > source = GDScriptParser : : IdentifierNode : : UNDEFINED_SOURCE ;
2020-08-25 21:00:56 +02:00
// Check globals. We make an exception for Variant::OBJECT because it's the base class for
// non-builtin types so we allow doing e.g. Object.new()
Variant : : Type builtin_type = GDScriptParser : : get_builtin_type ( name ) ;
if ( builtin_type ! = Variant : : OBJECT & & builtin_type < Variant : : VARIANT_MAX ) {
2020-06-12 02:49:58 +02:00
if ( can_be_builtin ) {
2020-08-25 21:00:56 +02:00
p_identifier - > set_datatype ( make_builtin_meta_type ( builtin_type ) ) ;
2020-06-12 02:49:58 +02:00
return ;
} else {
push_error ( R " (Builtin type cannot be used as a name on its own.) " , p_identifier ) ;
}
2020-06-11 00:53:25 +02:00
}
if ( class_exists ( name ) ) {
p_identifier - > set_datatype ( make_native_meta_type ( name ) ) ;
return ;
}
if ( ScriptServer : : is_global_class ( name ) ) {
2020-11-26 16:27:48 +01:00
p_identifier - > set_datatype ( make_global_class_meta_type ( name , p_identifier ) ) ;
2020-06-11 00:53:25 +02:00
return ;
}
2020-09-01 01:08:46 +02:00
// Try singletons.
// Do this before globals because this might be a singleton loading another one before it's compiled.
if ( ProjectSettings : : get_singleton ( ) - > has_autoload ( name ) ) {
const ProjectSettings : : AutoloadInfo & autoload = ProjectSettings : : get_singleton ( ) - > get_autoload ( name ) ;
if ( autoload . is_singleton ) {
// Singleton exists, so it's at least a Node.
GDScriptParser : : DataType result ;
result . kind = GDScriptParser : : DataType : : NATIVE ;
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
2022-10-09 18:41:28 +02:00
if ( ResourceLoader : : get_resource_type ( autoload . path ) = = " GDScript " ) {
2022-09-29 11:53:28 +02:00
Ref < GDScriptParserRef > singl_parser = get_parser_for ( autoload . path ) ;
if ( singl_parser . is_valid ( ) ) {
2022-12-11 03:57:35 +01:00
Error err = singl_parser - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2020-09-01 01:08:46 +02:00
if ( err = = OK ) {
2022-09-29 11:53:28 +02:00
result = type_from_metatype ( singl_parser - > get_parser ( ) - > head - > get_datatype ( ) ) ;
2020-09-01 01:08:46 +02:00
}
}
2022-11-22 05:24:10 +01:00
} else if ( ResourceLoader : : get_resource_type ( autoload . path ) = = " PackedScene " ) {
2022-12-20 05:09:32 +01:00
if ( GDScriptLanguage : : get_singleton ( ) - > has_any_global_constant ( name ) ) {
Variant constant = GDScriptLanguage : : get_singleton ( ) - > get_any_global_constant ( name ) ;
2022-11-24 00:13:13 +01:00
Node * node = Object : : cast_to < Node > ( constant ) ;
if ( node ! = nullptr ) {
2022-12-01 11:20:42 +01:00
Ref < GDScript > scr = node - > get_script ( ) ;
2022-11-24 00:13:13 +01:00
if ( scr . is_valid ( ) ) {
2022-12-01 11:20:42 +01:00
Ref < GDScriptParserRef > singl_parser = get_parser_for ( scr - > get_script_path ( ) ) ;
2022-11-24 00:13:13 +01:00
if ( singl_parser . is_valid ( ) ) {
2022-12-11 03:57:35 +01:00
Error err = singl_parser - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2022-11-24 00:13:13 +01:00
if ( err = = OK ) {
result = type_from_metatype ( singl_parser - > get_parser ( ) - > head - > get_datatype ( ) ) ;
}
}
2022-11-22 05:24:10 +01:00
}
}
}
2020-09-01 01:08:46 +02:00
}
result . is_constant = true ;
p_identifier - > set_datatype ( result ) ;
return ;
}
}
2022-12-20 05:09:32 +01:00
if ( GDScriptLanguage : : get_singleton ( ) - > has_any_global_constant ( name ) ) {
Variant constant = GDScriptLanguage : : get_singleton ( ) - > get_any_global_constant ( name ) ;
2020-08-31 14:53:02 +02:00
p_identifier - > set_datatype ( type_from_variant ( constant , p_identifier ) ) ;
2020-06-11 00:53:25 +02:00
p_identifier - > is_constant = true ;
p_identifier - > reduced_value = constant ;
return ;
}
// Not found.
2020-06-12 00:31:28 +02:00
// Check if it's a builtin function.
2020-11-26 15:56:32 +01:00
if ( GDScriptUtilityFunctions : : function_exists ( name ) ) {
2020-06-12 00:31:28 +02:00
push_error ( vformat ( R " (Built-in function " % s " cannot be used as an identifier.) " , name ) , p_identifier ) ;
} else {
push_error ( vformat ( R " (Identifier " % s " not declared in the current scope.) " , name ) , p_identifier ) ;
}
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType dummy ;
dummy . kind = GDScriptParser : : DataType : : VARIANT ;
p_identifier - > set_datatype ( dummy ) ; // Just so type is set to something.
}
2021-03-26 13:03:16 +01:00
void GDScriptAnalyzer : : reduce_lambda ( GDScriptParser : : LambdaNode * p_lambda ) {
// Lambda is always a Callable.
GDScriptParser : : DataType lambda_type ;
lambda_type . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
lambda_type . kind = GDScriptParser : : DataType : : BUILTIN ;
lambda_type . builtin_type = Variant : : CALLABLE ;
p_lambda - > set_datatype ( lambda_type ) ;
if ( p_lambda - > function = = nullptr ) {
return ;
}
lambda_stack . push_back ( p_lambda ) ;
2022-12-28 06:41:03 +01:00
resolve_function_signature ( p_lambda - > function , p_lambda , true ) ;
resolve_function_body ( p_lambda - > function , true ) ;
lambda_stack . pop_back ( ) ;
2021-03-26 13:03:16 +01:00
2021-03-28 16:03:13 +02:00
int captures_amount = p_lambda - > captures . size ( ) ;
if ( captures_amount > 0 ) {
// Create space for lambda parameters.
// At the beginning to not mess with optional parameters.
int param_count = p_lambda - > function - > parameters . size ( ) ;
p_lambda - > function - > parameters . resize ( param_count + captures_amount ) ;
for ( int i = param_count - 1 ; i > = 0 ; i - - ) {
p_lambda - > function - > parameters . write [ i + captures_amount ] = p_lambda - > function - > parameters [ i ] ;
p_lambda - > function - > parameters_indices [ p_lambda - > function - > parameters [ i ] - > identifier - > name ] = i + captures_amount ;
}
// Add captures as extra parameters at the beginning.
for ( int i = 0 ; i < p_lambda - > captures . size ( ) ; i + + ) {
GDScriptParser : : IdentifierNode * capture = p_lambda - > captures [ i ] ;
GDScriptParser : : ParameterNode * capture_param = parser - > alloc_node < GDScriptParser : : ParameterNode > ( ) ;
capture_param - > identifier = capture ;
capture_param - > usages = capture - > usages ;
capture_param - > set_datatype ( capture - > get_datatype ( ) ) ;
p_lambda - > function - > parameters . write [ i ] = capture_param ;
p_lambda - > function - > parameters_indices [ capture - > name ] = i ;
}
}
2021-03-26 13:03:16 +01:00
}
2020-06-11 00:53:25 +02:00
void GDScriptAnalyzer : : reduce_literal ( GDScriptParser : : LiteralNode * p_literal ) {
p_literal - > reduced_value = p_literal - > value ;
p_literal - > is_constant = true ;
2020-08-31 14:53:02 +02:00
p_literal - > set_datatype ( type_from_variant ( p_literal - > reduced_value , p_literal ) ) ;
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : reduce_preload ( GDScriptParser : : PreloadNode * p_preload ) {
2020-08-19 15:45:00 +02:00
if ( ! p_preload - > path ) {
return ;
}
reduce_expression ( p_preload - > path ) ;
if ( ! p_preload - > path - > is_constant ) {
push_error ( " Preloaded path must be a constant string. " , p_preload - > path ) ;
return ;
}
if ( p_preload - > path - > reduced_value . get_type ( ) ! = Variant : : STRING ) {
push_error ( " Preloaded path must be a constant string. " , p_preload - > path ) ;
} else {
p_preload - > resolved_path = p_preload - > path - > reduced_value ;
// TODO: Save this as script dependency.
2021-08-30 01:43:47 +02:00
if ( p_preload - > resolved_path . is_relative_path ( ) ) {
2022-08-30 02:34:01 +02:00
p_preload - > resolved_path = parser - > script_path . get_base_dir ( ) . path_join ( p_preload - > resolved_path ) ;
2020-08-19 15:45:00 +02:00
}
p_preload - > resolved_path = p_preload - > resolved_path . simplify_path ( ) ;
2022-08-31 17:52:37 +02:00
if ( ! ResourceLoader : : exists ( p_preload - > resolved_path ) ) {
2022-10-16 05:20:57 +02:00
Ref < FileAccess > file_check = FileAccess : : create ( FileAccess : : ACCESS_RESOURCES ) ;
if ( file_check - > file_exists ( p_preload - > resolved_path ) ) {
push_error ( vformat ( R " (Preload file " % s " has no resource loaders (unrecognized file extension).) " , p_preload - > resolved_path ) , p_preload - > path ) ;
} else {
push_error ( vformat ( R " (Preload file " % s " does not exist.) " , p_preload - > resolved_path ) , p_preload - > path ) ;
}
2020-08-19 15:45:00 +02:00
} else {
// TODO: Don't load if validating: use completion cache.
2022-10-09 18:41:28 +02:00
// Must load GDScript and PackedScenes separately to permit cyclic references
// as ResourceLoader::load() detect and reject those.
if ( ResourceLoader : : get_resource_type ( p_preload - > resolved_path ) = = " GDScript " ) {
Error err = OK ;
Ref < GDScript > res = GDScriptCache : : get_shallow_script ( p_preload - > resolved_path , err , parser - > script_path ) ;
p_preload - > resource = res ;
if ( err ! = OK ) {
push_error ( vformat ( R " (Could not preload resource script " % s " .) " , p_preload - > resolved_path ) , p_preload - > path ) ;
}
} else if ( ResourceLoader : : get_resource_type ( p_preload - > resolved_path ) = = " PackedScene " ) {
Error err = OK ;
Ref < PackedScene > res = GDScriptCache : : get_packed_scene ( p_preload - > resolved_path , err , parser - > script_path ) ;
p_preload - > resource = res ;
if ( err ! = OK ) {
push_error ( vformat ( R " (Could not preload resource scene " % s " .) " , p_preload - > resolved_path ) , p_preload - > path ) ;
}
} else {
p_preload - > resource = ResourceLoader : : load ( p_preload - > resolved_path ) ;
if ( p_preload - > resource . is_null ( ) ) {
push_error ( vformat ( R " (Could not preload resource file " % s " .) " , p_preload - > resolved_path ) , p_preload - > path ) ;
}
2020-08-19 15:45:00 +02:00
}
}
}
2020-06-11 00:53:25 +02:00
p_preload - > is_constant = true ;
p_preload - > reduced_value = p_preload - > resource ;
2020-08-31 14:53:02 +02:00
p_preload - > set_datatype ( type_from_variant ( p_preload - > reduced_value , p_preload ) ) ;
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : reduce_self ( GDScriptParser : : SelfNode * p_self ) {
p_self - > is_constant = false ;
p_self - > set_datatype ( type_from_metatype ( parser - > current_class - > get_datatype ( ) ) ) ;
2022-04-20 19:22:22 +02:00
mark_lambda_use_self ( ) ;
2020-06-11 00:53:25 +02:00
}
void GDScriptAnalyzer : : reduce_subscript ( GDScriptParser : : SubscriptNode * p_subscript ) {
2021-06-24 22:03:40 +02:00
if ( p_subscript - > base = = nullptr ) {
return ;
}
2020-06-12 02:49:58 +02:00
if ( p_subscript - > base - > type = = GDScriptParser : : Node : : IDENTIFIER ) {
reduce_identifier ( static_cast < GDScriptParser : : IdentifierNode * > ( p_subscript - > base ) , true ) ;
} else {
reduce_expression ( p_subscript - > base ) ;
2022-04-13 20:03:13 +02:00
if ( p_subscript - > base - > type = = GDScriptParser : : Node : : ARRAY ) {
2023-01-08 04:41:06 +01:00
const_fold_array ( static_cast < GDScriptParser : : ArrayNode * > ( p_subscript - > base ) , false ) ;
2022-04-13 20:03:13 +02:00
} else if ( p_subscript - > base - > type = = GDScriptParser : : Node : : DICTIONARY ) {
2023-01-08 04:41:06 +01:00
const_fold_dictionary ( static_cast < GDScriptParser : : DictionaryNode * > ( p_subscript - > base ) , false ) ;
2022-04-13 20:03:13 +02:00
}
2020-06-12 02:49:58 +02:00
}
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType result_type ;
if ( p_subscript - > is_attribute ) {
2020-07-06 17:24:24 +02:00
if ( p_subscript - > attribute = = nullptr ) {
return ;
}
2022-12-11 13:59:43 +01:00
GDScriptParser : : DataType base_type = p_subscript - > base - > get_datatype ( ) ;
2022-12-11 03:57:35 +01:00
bool valid = false ;
// If the base is a metatype, use the analyzer instead.
if ( p_subscript - > base - > is_constant & & ! base_type . is_meta_type ) {
2020-06-11 00:53:25 +02:00
// Just try to get it.
2020-11-07 02:29:22 +01:00
Variant value = p_subscript - > base - > reduced_value . get_named ( p_subscript - > attribute - > name , valid ) ;
2022-12-11 03:57:35 +01:00
if ( valid ) {
2020-06-11 00:53:25 +02:00
p_subscript - > is_constant = true ;
p_subscript - > reduced_value = value ;
2020-08-31 14:53:02 +02:00
result_type = type_from_variant ( value , p_subscript ) ;
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
} else if ( base_type . is_variant ( ) | | ! base_type . is_hard_type ( ) ) {
valid = true ;
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
mark_node_unsafe ( p_subscript ) ;
2020-06-11 00:53:25 +02:00
} else {
2022-12-11 03:57:35 +01:00
reduce_identifier_from_base ( p_subscript - > attribute , & base_type ) ;
GDScriptParser : : DataType attr_type = p_subscript - > attribute - > get_datatype ( ) ;
if ( attr_type . is_set ( ) ) {
valid = true ;
result_type = attr_type ;
p_subscript - > is_constant = p_subscript - > attribute - > is_constant ;
p_subscript - > reduced_value = p_subscript - > attribute - > reduced_value ;
2022-12-19 00:36:41 +01:00
} else if ( ! base_type . is_meta_type | | ! base_type . is_constant ) {
2022-12-11 03:57:35 +01:00
valid = base_type . kind ! = GDScriptParser : : DataType : : BUILTIN ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2022-12-11 03:57:35 +01:00
if ( valid ) {
parser - > push_warning ( p_subscript , GDScriptWarning : : UNSAFE_PROPERTY_ACCESS , p_subscript - > attribute - > name , base_type . to_string ( ) ) ;
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
# endif
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
2020-06-11 00:53:25 +02:00
}
}
2022-12-11 03:57:35 +01:00
if ( ! valid ) {
push_error ( vformat ( R " (Cannot find member " % s " in base " % s " .) " , p_subscript - > attribute - > name , type_from_metatype ( base_type ) . to_string ( ) ) , p_subscript - > attribute ) ;
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
}
2020-06-11 00:53:25 +02:00
} else {
2021-04-23 20:42:33 +02:00
if ( p_subscript - > index = = nullptr ) {
return ;
}
reduce_expression ( p_subscript - > index ) ;
2020-06-11 00:53:25 +02:00
if ( p_subscript - > base - > is_constant & & p_subscript - > index - > is_constant ) {
// Just try to get it.
bool valid = false ;
Variant value = p_subscript - > base - > reduced_value . get ( p_subscript - > index - > reduced_value , & valid ) ;
if ( ! valid ) {
push_error ( vformat ( R " (Cannot get index " % s " from " % s " .) " , p_subscript - > index - > reduced_value , p_subscript - > base - > reduced_value ) , p_subscript - > index ) ;
2023-01-08 04:41:06 +01:00
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
2020-06-11 00:53:25 +02:00
} else {
p_subscript - > is_constant = true ;
p_subscript - > reduced_value = value ;
2020-08-31 14:53:02 +02:00
result_type = type_from_variant ( value , p_subscript ) ;
2020-06-11 00:53:25 +02:00
}
} else {
GDScriptParser : : DataType base_type = p_subscript - > base - > get_datatype ( ) ;
GDScriptParser : : DataType index_type = p_subscript - > index - > get_datatype ( ) ;
if ( base_type . is_variant ( ) ) {
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
mark_node_unsafe ( p_subscript ) ;
} else {
if ( base_type . kind = = GDScriptParser : : DataType : : BUILTIN & & ! index_type . is_variant ( ) ) {
// Check if indexing is valid.
bool error = index_type . kind ! = GDScriptParser : : DataType : : BUILTIN & & base_type . builtin_type ! = Variant : : DICTIONARY ;
if ( ! error ) {
switch ( base_type . builtin_type ) {
// Expect int or real as index.
case Variant : : PACKED_BYTE_ARRAY :
case Variant : : PACKED_COLOR_ARRAY :
case Variant : : PACKED_FLOAT32_ARRAY :
case Variant : : PACKED_FLOAT64_ARRAY :
case Variant : : PACKED_INT32_ARRAY :
case Variant : : PACKED_INT64_ARRAY :
case Variant : : PACKED_STRING_ARRAY :
case Variant : : PACKED_VECTOR2_ARRAY :
case Variant : : PACKED_VECTOR3_ARRAY :
case Variant : : ARRAY :
case Variant : : STRING :
error = index_type . builtin_type ! = Variant : : INT & & index_type . builtin_type ! = Variant : : FLOAT ;
break ;
// Expect String only.
case Variant : : RECT2 :
case Variant : : RECT2I :
case Variant : : PLANE :
2021-01-20 08:02:02 +01:00
case Variant : : QUATERNION :
2020-06-11 00:53:25 +02:00
case Variant : : AABB :
case Variant : : OBJECT :
2022-12-06 03:46:47 +01:00
error = index_type . builtin_type ! = Variant : : STRING & & index_type . builtin_type ! = Variant : : STRING_NAME ;
2020-06-11 00:53:25 +02:00
break ;
// Expect String or number.
case Variant : : BASIS :
case Variant : : VECTOR2 :
case Variant : : VECTOR2I :
case Variant : : VECTOR3 :
case Variant : : VECTOR3I :
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
case Variant : : VECTOR4 :
case Variant : : VECTOR4I :
2020-06-11 00:53:25 +02:00
case Variant : : TRANSFORM2D :
2021-04-28 09:36:08 +02:00
case Variant : : TRANSFORM3D :
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
case Variant : : PROJECTION :
2020-06-11 00:53:25 +02:00
error = index_type . builtin_type ! = Variant : : INT & & index_type . builtin_type ! = Variant : : FLOAT & &
2022-12-06 03:46:47 +01:00
index_type . builtin_type ! = Variant : : STRING & & index_type . builtin_type ! = Variant : : STRING_NAME ;
2020-06-11 00:53:25 +02:00
break ;
// Expect String or int.
case Variant : : COLOR :
2022-12-06 03:46:47 +01:00
error = index_type . builtin_type ! = Variant : : INT & & index_type . builtin_type ! = Variant : : STRING & & index_type . builtin_type ! = Variant : : STRING_NAME ;
2020-06-11 00:53:25 +02:00
break ;
// Don't support indexing, but we will check it later.
2020-11-09 14:53:05 +01:00
case Variant : : RID :
2020-06-11 00:53:25 +02:00
case Variant : : BOOL :
case Variant : : CALLABLE :
case Variant : : FLOAT :
case Variant : : INT :
case Variant : : NIL :
case Variant : : NODE_PATH :
case Variant : : SIGNAL :
case Variant : : STRING_NAME :
break ;
// Here for completeness.
case Variant : : DICTIONARY :
case Variant : : VARIANT_MAX :
break ;
}
if ( error ) {
push_error ( vformat ( R " (Invalid index type " % s " for a base of type " % s " .) " , index_type . to_string ( ) , base_type . to_string ( ) ) , p_subscript - > index ) ;
}
}
} else if ( base_type . kind ! = GDScriptParser : : DataType : : BUILTIN & & ! index_type . is_variant ( ) ) {
if ( index_type . builtin_type ! = Variant : : STRING & & index_type . builtin_type ! = Variant : : STRING_NAME ) {
push_error ( vformat ( R " (Only String or StringName can be used as index for type " % s " , but received a " % s " .) " , base_type . to_string ( ) , index_type . to_string ( ) ) , p_subscript - > index ) ;
}
}
// Check resulting type if possible.
result_type . builtin_type = Variant : : NIL ;
result_type . kind = GDScriptParser : : DataType : : BUILTIN ;
2020-08-19 19:35:26 +02:00
result_type . type_source = base_type . is_hard_type ( ) ? GDScriptParser : : DataType : : ANNOTATED_INFERRED : GDScriptParser : : DataType : : INFERRED ;
2020-06-11 00:53:25 +02:00
2021-10-06 17:01:34 +02:00
if ( base_type . kind ! = GDScriptParser : : DataType : : BUILTIN ) {
base_type . builtin_type = Variant : : OBJECT ;
}
2020-06-11 00:53:25 +02:00
switch ( base_type . builtin_type ) {
// Can't index at all.
2020-11-09 14:53:05 +01:00
case Variant : : RID :
2020-06-11 00:53:25 +02:00
case Variant : : BOOL :
case Variant : : CALLABLE :
case Variant : : FLOAT :
case Variant : : INT :
case Variant : : NIL :
case Variant : : NODE_PATH :
case Variant : : SIGNAL :
case Variant : : STRING_NAME :
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
push_error ( vformat ( R " (Cannot use subscript operator on a base of type " % s " .) " , base_type . to_string ( ) ) , p_subscript - > base ) ;
break ;
// Return int.
case Variant : : PACKED_BYTE_ARRAY :
case Variant : : PACKED_INT32_ARRAY :
case Variant : : PACKED_INT64_ARRAY :
case Variant : : VECTOR2I :
case Variant : : VECTOR3I :
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
case Variant : : VECTOR4I :
2020-06-11 00:53:25 +02:00
result_type . builtin_type = Variant : : INT ;
break ;
// Return float.
case Variant : : PACKED_FLOAT32_ARRAY :
case Variant : : PACKED_FLOAT64_ARRAY :
case Variant : : VECTOR2 :
case Variant : : VECTOR3 :
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
case Variant : : VECTOR4 :
2021-01-20 08:02:02 +01:00
case Variant : : QUATERNION :
2020-06-11 00:53:25 +02:00
result_type . builtin_type = Variant : : FLOAT ;
break ;
// Return Color.
case Variant : : PACKED_COLOR_ARRAY :
result_type . builtin_type = Variant : : COLOR ;
break ;
// Return String.
case Variant : : PACKED_STRING_ARRAY :
case Variant : : STRING :
result_type . builtin_type = Variant : : STRING ;
break ;
// Return Vector2.
case Variant : : PACKED_VECTOR2_ARRAY :
case Variant : : TRANSFORM2D :
case Variant : : RECT2 :
result_type . builtin_type = Variant : : VECTOR2 ;
break ;
// Return Vector2I.
case Variant : : RECT2I :
result_type . builtin_type = Variant : : VECTOR2I ;
break ;
// Return Vector3.
case Variant : : PACKED_VECTOR3_ARRAY :
case Variant : : AABB :
case Variant : : BASIS :
result_type . builtin_type = Variant : : VECTOR3 ;
break ;
// Depends on the index.
2021-04-28 09:36:08 +02:00
case Variant : : TRANSFORM3D :
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
case Variant : : PROJECTION :
2020-06-11 00:53:25 +02:00
case Variant : : PLANE :
case Variant : : COLOR :
case Variant : : DICTIONARY :
2021-10-06 17:01:34 +02:00
case Variant : : OBJECT :
2020-06-11 00:53:25 +02:00
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
result_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
break ;
2021-03-09 16:32:35 +01:00
// Can have an element type.
case Variant : : ARRAY :
if ( base_type . has_container_element_type ( ) ) {
result_type = base_type . get_container_element_type ( ) ;
result_type . type_source = base_type . type_source ;
} else {
result_type . kind = GDScriptParser : : DataType : : VARIANT ;
result_type . type_source = GDScriptParser : : DataType : : UNDETECTED ;
}
break ;
2020-06-11 00:53:25 +02:00
// Here for completeness.
case Variant : : VARIANT_MAX :
break ;
}
}
}
}
p_subscript - > set_datatype ( result_type ) ;
}
void GDScriptAnalyzer : : reduce_ternary_op ( GDScriptParser : : TernaryOpNode * p_ternary_op ) {
reduce_expression ( p_ternary_op - > condition ) ;
reduce_expression ( p_ternary_op - > true_expr ) ;
reduce_expression ( p_ternary_op - > false_expr ) ;
GDScriptParser : : DataType result ;
2020-07-16 03:02:44 +02:00
if ( p_ternary_op - > condition & & p_ternary_op - > condition - > is_constant & & p_ternary_op - > true_expr - > is_constant & & p_ternary_op - > false_expr & & p_ternary_op - > false_expr - > is_constant ) {
2020-06-11 00:53:25 +02:00
p_ternary_op - > is_constant = true ;
if ( p_ternary_op - > condition - > reduced_value . booleanize ( ) ) {
p_ternary_op - > reduced_value = p_ternary_op - > true_expr - > reduced_value ;
} else {
p_ternary_op - > reduced_value = p_ternary_op - > false_expr - > reduced_value ;
}
}
2020-07-16 03:02:44 +02:00
GDScriptParser : : DataType true_type ;
if ( p_ternary_op - > true_expr ) {
true_type = p_ternary_op - > true_expr - > get_datatype ( ) ;
} else {
true_type . kind = GDScriptParser : : DataType : : VARIANT ;
}
GDScriptParser : : DataType false_type ;
if ( p_ternary_op - > false_expr ) {
false_type = p_ternary_op - > false_expr - > get_datatype ( ) ;
} else {
false_type . kind = GDScriptParser : : DataType : : VARIANT ;
}
2020-06-11 00:53:25 +02:00
if ( true_type . is_variant ( ) | | false_type . is_variant ( ) ) {
result . kind = GDScriptParser : : DataType : : VARIANT ;
} else {
result = true_type ;
if ( ! is_type_compatible ( true_type , false_type ) ) {
result = false_type ;
if ( ! is_type_compatible ( false_type , true_type ) ) {
result . type_source = GDScriptParser : : DataType : : UNDETECTED ;
result . kind = GDScriptParser : : DataType : : VARIANT ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_ternary_op , GDScriptWarning : : INCOMPATIBLE_TERNARY ) ;
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
}
}
p_ternary_op - > set_datatype ( result ) ;
}
void GDScriptAnalyzer : : reduce_unary_op ( GDScriptParser : : UnaryOpNode * p_unary_op ) {
reduce_expression ( p_unary_op - > operand ) ;
GDScriptParser : : DataType result ;
2020-08-18 01:49:04 +02:00
if ( p_unary_op - > operand = = nullptr ) {
result . kind = GDScriptParser : : DataType : : VARIANT ;
p_unary_op - > set_datatype ( result ) ;
return ;
}
2022-12-15 23:15:53 +01:00
GDScriptParser : : DataType operand_type = p_unary_op - > operand - > get_datatype ( ) ;
2020-06-11 00:53:25 +02:00
if ( p_unary_op - > operand - > is_constant ) {
p_unary_op - > is_constant = true ;
p_unary_op - > reduced_value = Variant : : evaluate ( p_unary_op - > variant_op , p_unary_op - > operand - > reduced_value , Variant ( ) ) ;
2020-08-31 14:53:02 +02:00
result = type_from_variant ( p_unary_op - > reduced_value , p_unary_op ) ;
2022-10-02 02:13:06 +02:00
}
if ( operand_type . is_variant ( ) ) {
2020-06-11 00:53:25 +02:00
result . kind = GDScriptParser : : DataType : : VARIANT ;
mark_node_unsafe ( p_unary_op ) ;
} else {
bool valid = false ;
2022-10-02 02:13:06 +02:00
result = get_operation_type ( p_unary_op - > variant_op , operand_type , valid , p_unary_op ) ;
2020-06-11 00:53:25 +02:00
if ( ! valid ) {
2022-10-02 02:13:06 +02:00
push_error ( vformat ( R " (Invalid operand of type " % s " for unary operator " % s " .) " , operand_type . to_string ( ) , Variant : : get_operator_name ( p_unary_op - > variant_op ) ) , p_unary_op ) ;
2020-06-11 00:53:25 +02:00
}
}
p_unary_op - > set_datatype ( result ) ;
}
2023-01-08 04:41:06 +01:00
void GDScriptAnalyzer : : const_fold_array ( GDScriptParser : : ArrayNode * p_array , bool p_is_const ) {
2020-09-11 16:16:20 +02:00
for ( int i = 0 ; i < p_array - > elements . size ( ) ; i + + ) {
GDScriptParser : : ExpressionNode * element = p_array - > elements [ i ] ;
2022-04-13 20:03:13 +02:00
if ( element - > type = = GDScriptParser : : Node : : ARRAY ) {
2023-01-08 04:41:06 +01:00
const_fold_array ( static_cast < GDScriptParser : : ArrayNode * > ( element ) , p_is_const ) ;
2022-04-13 20:03:13 +02:00
} else if ( element - > type = = GDScriptParser : : Node : : DICTIONARY ) {
2023-01-08 04:41:06 +01:00
const_fold_dictionary ( static_cast < GDScriptParser : : DictionaryNode * > ( element ) , p_is_const ) ;
2022-04-13 20:03:13 +02:00
}
2023-01-08 04:41:06 +01:00
if ( ! element - > is_constant ) {
2020-09-11 16:16:20 +02:00
return ;
}
}
Array array ;
array . resize ( p_array - > elements . size ( ) ) ;
for ( int i = 0 ; i < p_array - > elements . size ( ) ; i + + ) {
array [ i ] = p_array - > elements [ i ] - > reduced_value ;
}
2023-01-08 04:41:06 +01:00
if ( p_is_const ) {
array . set_read_only ( true ) ;
}
2020-09-11 16:16:20 +02:00
p_array - > is_constant = true ;
p_array - > reduced_value = array ;
}
2023-01-08 04:41:06 +01:00
void GDScriptAnalyzer : : const_fold_dictionary ( GDScriptParser : : DictionaryNode * p_dictionary , bool p_is_const ) {
2020-09-11 16:16:20 +02:00
for ( int i = 0 ; i < p_dictionary - > elements . size ( ) ; i + + ) {
const GDScriptParser : : DictionaryNode : : Pair & element = p_dictionary - > elements [ i ] ;
2022-04-13 20:03:13 +02:00
if ( element . value - > type = = GDScriptParser : : Node : : ARRAY ) {
2023-01-08 04:41:06 +01:00
const_fold_array ( static_cast < GDScriptParser : : ArrayNode * > ( element . value ) , p_is_const ) ;
2022-04-13 20:03:13 +02:00
} else if ( element . value - > type = = GDScriptParser : : Node : : DICTIONARY ) {
2023-01-08 04:41:06 +01:00
const_fold_dictionary ( static_cast < GDScriptParser : : DictionaryNode * > ( element . value ) , p_is_const ) ;
2022-04-13 20:03:13 +02:00
}
2023-01-08 04:41:06 +01:00
if ( ! element . key - > is_constant | | ! element . value - > is_constant ) {
2020-09-11 16:16:20 +02:00
return ;
}
}
Dictionary dict ;
for ( int i = 0 ; i < p_dictionary - > elements . size ( ) ; i + + ) {
const GDScriptParser : : DictionaryNode : : Pair & element = p_dictionary - > elements [ i ] ;
dict [ element . key - > reduced_value ] = element . value - > reduced_value ;
}
2023-01-08 04:41:06 +01:00
if ( p_is_const ) {
dict . set_read_only ( true ) ;
}
2020-09-11 16:16:20 +02:00
p_dictionary - > is_constant = true ;
p_dictionary - > reduced_value = dict ;
}
2020-08-31 14:53:02 +02:00
GDScriptParser : : DataType GDScriptAnalyzer : : type_from_variant ( const Variant & p_value , const GDScriptParser : : Node * p_source ) {
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType result ;
result . is_constant = true ;
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = p_value . get_type ( ) ;
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ; // Constant has explicit type.
2022-12-22 12:02:34 +01:00
if ( p_value . get_type ( ) = = Variant : : OBJECT ) {
2022-12-04 11:35:41 +01:00
// Object is treated as a native type, not a builtin type.
result . kind = GDScriptParser : : DataType : : NATIVE ;
2020-06-11 00:53:25 +02:00
Object * obj = p_value ;
if ( ! obj ) {
return GDScriptParser : : DataType ( ) ;
}
result . native_type = obj - > get_class_name ( ) ;
Ref < Script > scr = p_value ; // Check if value is a script itself.
if ( scr . is_valid ( ) ) {
result . is_meta_type = true ;
} else {
result . is_meta_type = false ;
scr = obj - > get_script ( ) ;
}
if ( scr . is_valid ( ) ) {
2022-10-09 18:41:28 +02:00
Ref < GDScript > gds = scr ;
if ( gds . is_valid ( ) ) {
// This might be an inner class, so we want to get the parser for the root.
// But still get the inner class from that tree.
2022-12-12 13:35:55 +01:00
String script_path = gds - > get_script_path ( ) ;
Ref < GDScriptParserRef > ref = get_parser_for ( script_path ) ;
2022-10-09 18:41:28 +02:00
if ( ref . is_null ( ) ) {
2022-12-12 13:35:55 +01:00
push_error ( vformat ( R " (Could not find script " % s " .) " , script_path ) , p_source ) ;
2022-10-09 18:41:28 +02:00
GDScriptParser : : DataType error_type ;
error_type . kind = GDScriptParser : : DataType : : VARIANT ;
return error_type ;
}
2022-12-12 13:35:55 +01:00
Error err = ref - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
GDScriptParser : : ClassNode * found = nullptr ;
if ( err = = OK ) {
found = ref - > get_parser ( ) - > find_class ( gds - > fully_qualified_name ) ;
if ( found ! = nullptr ) {
err = resolve_class_inheritance ( found , p_source ) ;
2022-12-11 03:57:35 +01:00
}
2022-12-12 13:35:55 +01:00
}
if ( err | | found = = nullptr ) {
push_error ( vformat ( R " (Could not resolve script " % s " .) " , script_path ) , p_source ) ;
GDScriptParser : : DataType error_type ;
error_type . kind = GDScriptParser : : DataType : : VARIANT ;
return error_type ;
2022-10-09 18:41:28 +02:00
}
2023-01-03 03:54:51 +01:00
result . kind = GDScriptParser : : DataType : : CLASS ;
result . native_type = found - > get_datatype ( ) . native_type ;
2022-12-15 22:46:15 +01:00
result . class_type = found ;
result . script_path = ref - > get_parser ( ) - > script_path ;
2020-06-11 00:53:25 +02:00
} else {
2022-10-09 18:41:28 +02:00
result . kind = GDScriptParser : : DataType : : SCRIPT ;
2022-12-11 03:57:35 +01:00
result . native_type = scr - > get_instance_base_type ( ) ;
2023-01-03 03:54:51 +01:00
result . script_path = scr - > get_path ( ) ;
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
result . script_type = scr ;
2020-06-11 00:53:25 +02:00
} else {
result . kind = GDScriptParser : : DataType : : NATIVE ;
if ( result . native_type = = GDScriptNativeClass : : get_class_static ( ) ) {
result . is_meta_type = true ;
}
}
}
return result ;
}
2022-12-04 04:02:03 +01:00
GDScriptParser : : DataType GDScriptAnalyzer : : type_from_metatype ( const GDScriptParser : : DataType & p_meta_type ) {
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType result = p_meta_type ;
result . is_meta_type = false ;
2022-02-02 17:57:24 +01:00
if ( p_meta_type . kind = = GDScriptParser : : DataType : : ENUM ) {
result . builtin_type = Variant : : INT ;
2022-12-04 04:02:03 +01:00
} else {
result . is_constant = false ;
2022-02-02 17:57:24 +01:00
}
2020-06-11 00:53:25 +02:00
return result ;
}
GDScriptParser : : DataType GDScriptAnalyzer : : type_from_property ( const PropertyInfo & p_property ) const {
GDScriptParser : : DataType result ;
result . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
if ( p_property . type = = Variant : : NIL & & ( p_property . usage & PROPERTY_USAGE_NIL_IS_VARIANT ) ) {
// Variant
result . kind = GDScriptParser : : DataType : : VARIANT ;
return result ;
}
result . builtin_type = p_property . type ;
if ( p_property . type = = Variant : : OBJECT ) {
result . kind = GDScriptParser : : DataType : : NATIVE ;
2022-02-06 14:12:19 +01:00
result . native_type = p_property . class_name = = StringName ( ) ? SNAME ( " Object " ) : p_property . class_name ;
2020-06-11 00:53:25 +02:00
} else {
result . kind = GDScriptParser : : DataType : : BUILTIN ;
2021-03-09 16:32:35 +01:00
result . builtin_type = p_property . type ;
if ( p_property . type = = Variant : : ARRAY & & p_property . hint = = PROPERTY_HINT_ARRAY_TYPE ) {
// Check element type.
StringName elem_type_name = p_property . hint_string ;
GDScriptParser : : DataType elem_type ;
elem_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
Variant : : Type elem_builtin_type = GDScriptParser : : get_builtin_type ( elem_type_name ) ;
if ( elem_builtin_type < Variant : : VARIANT_MAX ) {
// Builtin type.
elem_type . kind = GDScriptParser : : DataType : : BUILTIN ;
elem_type . builtin_type = elem_builtin_type ;
} else if ( class_exists ( elem_type_name ) ) {
elem_type . kind = GDScriptParser : : DataType : : NATIVE ;
elem_type . builtin_type = Variant : : OBJECT ;
elem_type . native_type = p_property . hint_string ;
} else if ( ScriptServer : : is_global_class ( elem_type_name ) ) {
// Just load this as it shouldn't be a GDScript.
Ref < Script > script = ResourceLoader : : load ( ScriptServer : : get_global_class_path ( elem_type_name ) ) ;
elem_type . kind = GDScriptParser : : DataType : : SCRIPT ;
elem_type . builtin_type = Variant : : OBJECT ;
elem_type . native_type = script - > get_instance_base_type ( ) ;
elem_type . script_type = script ;
} else {
ERR_FAIL_V_MSG ( result , " Could not find element type from property hint of a typed array. " ) ;
}
2021-10-15 15:40:50 +02:00
elem_type . is_constant = false ;
2021-03-09 16:32:35 +01:00
result . set_container_element_type ( elem_type ) ;
}
2020-06-11 00:53:25 +02:00
}
return result ;
}
2022-03-06 15:09:12 +01:00
bool GDScriptAnalyzer : : get_function_signature ( GDScriptParser : : Node * p_source , bool p_is_constructor , GDScriptParser : : DataType p_base_type , const StringName & p_function , GDScriptParser : : DataType & r_return_type , List < GDScriptParser : : DataType > & r_par_types , int & r_default_arg_count , bool & r_static , bool & r_vararg ) {
2020-06-11 00:53:25 +02:00
r_static = false ;
r_vararg = false ;
r_default_arg_count = 0 ;
2020-07-16 03:02:44 +02:00
StringName function_name = p_function ;
2020-06-11 00:53:25 +02:00
2022-12-04 04:02:03 +01:00
bool was_enum = false ;
2022-02-02 17:57:24 +01:00
if ( p_base_type . kind = = GDScriptParser : : DataType : : ENUM ) {
2022-12-04 04:02:03 +01:00
was_enum = true ;
2022-02-02 17:57:24 +01:00
if ( p_base_type . is_meta_type ) {
// Enum type can be treated as a dictionary value.
p_base_type . kind = GDScriptParser : : DataType : : BUILTIN ;
p_base_type . is_meta_type = false ;
} else {
push_error ( " Cannot call function on enum value. " , p_source ) ;
return false ;
}
}
2020-06-11 00:53:25 +02:00
if ( p_base_type . kind = = GDScriptParser : : DataType : : BUILTIN ) {
// Construct a base type to get methods.
Callable : : CallError err ;
2020-11-09 04:19:09 +01:00
Variant dummy ;
Variant : : construct ( p_base_type . builtin_type , dummy , nullptr , 0 , err ) ;
2020-06-11 00:53:25 +02:00
if ( err . error ! = Callable : : CallError : : CALL_OK ) {
ERR_FAIL_V_MSG ( false , " Could not construct base Variant type. " ) ;
}
List < MethodInfo > methods ;
dummy . get_method_list ( & methods ) ;
2021-07-16 05:45:57 +02:00
for ( const MethodInfo & E : methods ) {
if ( E . name = = p_function ) {
2021-06-03 14:51:51 +02:00
function_signature_from_info ( E , r_return_type , r_par_types , r_default_arg_count , r_static , r_vararg ) ;
r_static = Variant : : is_builtin_method_static ( p_base_type . builtin_type , function_name ) ;
2022-12-04 04:02:03 +01:00
// Cannot use non-const methods on enums.
if ( ! r_static & & was_enum & & ! ( E . flags & METHOD_FLAG_CONST ) ) {
push_error ( vformat ( R " *(Cannot call non-const Dictionary function " % s ( ) " on enum " % s " .)* " , p_function , p_base_type . enum_type ) , p_source ) ;
}
2021-06-03 14:51:51 +02:00
return true ;
2020-06-11 00:53:25 +02:00
}
}
return false ;
}
2022-03-06 15:09:12 +01:00
if ( p_is_constructor ) {
2020-07-16 03:02:44 +02:00
function_name = " _init " ;
r_static = true ;
}
2020-06-11 00:53:25 +02:00
GDScriptParser : : ClassNode * base_class = p_base_type . class_type ;
GDScriptParser : : FunctionNode * found_function = nullptr ;
while ( found_function = = nullptr & & base_class ! = nullptr ) {
2020-07-16 03:02:44 +02:00
if ( base_class - > has_member ( function_name ) ) {
if ( base_class - > get_member ( function_name ) . type ! = GDScriptParser : : ClassNode : : Member : : FUNCTION ) {
2020-06-11 00:53:25 +02:00
// TODO: If this is Callable it can have a better error message.
2020-07-16 03:02:44 +02:00
push_error ( vformat ( R " (Member " % s " is not a function.) " , function_name ) , p_source ) ;
2020-06-11 00:53:25 +02:00
return false ;
}
2022-12-11 03:57:35 +01:00
resolve_class_member ( base_class , function_name , p_source ) ;
2020-07-16 03:02:44 +02:00
found_function = base_class - > get_member ( function_name ) . function ;
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
resolve_class_inheritance ( base_class , p_source ) ;
2020-06-11 00:53:25 +02:00
base_class = base_class - > base_type . class_type ;
}
if ( found_function ! = nullptr ) {
2022-03-06 15:09:12 +01:00
r_static = p_is_constructor | | found_function - > is_static ;
2020-06-11 00:53:25 +02:00
for ( int i = 0 ; i < found_function - > parameters . size ( ) ; i + + ) {
r_par_types . push_back ( found_function - > parameters [ i ] - > get_datatype ( ) ) ;
2022-12-22 21:43:36 +01:00
if ( found_function - > parameters [ i ] - > initializer ! = nullptr ) {
2020-06-11 00:53:25 +02:00
r_default_arg_count + + ;
}
}
2022-12-27 21:49:26 +01:00
r_return_type = p_is_constructor ? p_base_type : found_function - > get_datatype ( ) ;
2021-05-26 14:23:17 +02:00
r_return_type . is_meta_type = false ;
2020-06-11 00:53:25 +02:00
r_return_type . is_coroutine = found_function - > is_coroutine ;
return true ;
}
Ref < Script > base_script = p_base_type . script_type ;
2022-04-25 06:33:18 +02:00
while ( base_script . is_valid ( ) & & base_script - > has_method ( function_name ) ) {
2020-07-16 03:02:44 +02:00
MethodInfo info = base_script - > get_method_info ( function_name ) ;
2020-06-11 00:53:25 +02:00
if ( ! ( info = = MethodInfo ( ) ) ) {
return function_signature_from_info ( info , r_return_type , r_par_types , r_default_arg_count , r_static , r_vararg ) ;
}
base_script = base_script - > get_base_script ( ) ;
}
// If the base is a script, it might be trying to access members of the Script class itself.
2022-03-06 15:09:12 +01:00
if ( p_base_type . is_meta_type & & ! p_is_constructor & & ( p_base_type . kind = = GDScriptParser : : DataType : : SCRIPT | | p_base_type . kind = = GDScriptParser : : DataType : : CLASS ) ) {
2020-06-11 00:53:25 +02:00
MethodInfo info ;
StringName script_class = p_base_type . kind = = GDScriptParser : : DataType : : SCRIPT ? p_base_type . script_type - > get_class_name ( ) : StringName ( GDScript : : get_class_static ( ) ) ;
2020-07-16 03:02:44 +02:00
if ( ClassDB : : get_method_info ( script_class , function_name , & info ) ) {
2020-06-11 00:53:25 +02:00
return function_signature_from_info ( info , r_return_type , r_par_types , r_default_arg_count , r_static , r_vararg ) ;
}
}
StringName base_native = p_base_type . native_type ;
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
if ( base_native ! = StringName ( ) ) {
// Empty native class might happen in some Script implementations.
2020-06-11 00:53:25 +02:00
// Just ignore it.
2020-07-16 03:02:44 +02:00
if ( ! class_exists ( base_native ) ) {
ERR_FAIL_V_MSG ( false , vformat ( " Native class %s used in script doesn't exist or isn't exposed. " , base_native ) ) ;
}
2020-06-11 00:53:25 +02:00
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
2022-03-06 15:09:12 +01:00
if ( p_is_constructor ) {
2020-07-16 03:02:44 +02:00
// Native types always have a default constructor.
r_return_type = p_base_type ;
r_return_type . type_source = GDScriptParser : : DataType : : ANNOTATED_EXPLICIT ;
r_return_type . is_meta_type = false ;
return true ;
2020-06-11 00:53:25 +02:00
}
MethodInfo info ;
2021-08-17 15:06:54 +02:00
if ( ClassDB : : get_method_info ( base_native , function_name , & info ) ) {
2021-08-20 15:52:58 +02:00
bool valid = function_signature_from_info ( info , r_return_type , r_par_types , r_default_arg_count , r_static , r_vararg ) ;
if ( valid & & Engine : : get_singleton ( ) - > has_singleton ( base_native ) ) {
r_static = true ;
}
return valid ;
2020-06-11 00:53:25 +02:00
}
return false ;
}
bool GDScriptAnalyzer : : function_signature_from_info ( const MethodInfo & p_info , GDScriptParser : : DataType & r_return_type , List < GDScriptParser : : DataType > & r_par_types , int & r_default_arg_count , bool & r_static , bool & r_vararg ) {
r_return_type = type_from_property ( p_info . return_val ) ;
r_default_arg_count = p_info . default_arguments . size ( ) ;
r_vararg = ( p_info . flags & METHOD_FLAG_VARARG ) ! = 0 ;
2022-04-06 19:14:38 +02:00
r_static = ( p_info . flags & METHOD_FLAG_STATIC ) ! = 0 ;
2020-06-11 00:53:25 +02:00
2021-07-16 05:45:57 +02:00
for ( const PropertyInfo & E : p_info . arguments ) {
r_par_types . push_back ( type_from_property ( E ) ) ;
2020-06-11 00:53:25 +02:00
}
return true ;
}
bool GDScriptAnalyzer : : validate_call_arg ( const MethodInfo & p_method , const GDScriptParser : : CallNode * p_call ) {
List < GDScriptParser : : DataType > arg_types ;
2021-07-16 05:45:57 +02:00
for ( const PropertyInfo & E : p_method . arguments ) {
arg_types . push_back ( type_from_property ( E ) ) ;
2020-06-11 00:53:25 +02:00
}
return validate_call_arg ( arg_types , p_method . default_arguments . size ( ) , ( p_method . flags & METHOD_FLAG_VARARG ) ! = 0 , p_call ) ;
}
bool GDScriptAnalyzer : : validate_call_arg ( const List < GDScriptParser : : DataType > & p_par_types , int p_default_args_count , bool p_is_vararg , const GDScriptParser : : CallNode * p_call ) {
bool valid = true ;
if ( p_call - > arguments . size ( ) < p_par_types . size ( ) - p_default_args_count ) {
push_error ( vformat ( R " *(Too few arguments for " % s ( ) " call. Expected at least %d but received %d.)* " , p_call - > function_name , p_par_types . size ( ) - p_default_args_count , p_call - > arguments . size ( ) ) , p_call ) ;
valid = false ;
}
if ( ! p_is_vararg & & p_call - > arguments . size ( ) > p_par_types . size ( ) ) {
push_error ( vformat ( R " *(Too many arguments for " % s ( ) " call. Expected at most %d but received %d.)* " , p_call - > function_name , p_par_types . size ( ) , p_call - > arguments . size ( ) ) , p_call - > arguments [ p_par_types . size ( ) ] ) ;
valid = false ;
}
for ( int i = 0 ; i < p_call - > arguments . size ( ) ; i + + ) {
if ( i > = p_par_types . size ( ) ) {
// Already on vararg place.
break ;
}
GDScriptParser : : DataType par_type = p_par_types [ i ] ;
GDScriptParser : : DataType arg_type = p_call - > arguments [ i ] - > get_datatype ( ) ;
2020-06-12 00:31:28 +02:00
if ( arg_type . is_variant ( ) ) {
2020-06-11 00:53:25 +02:00
// Argument can be anything, so this is unsafe.
mark_node_unsafe ( p_call - > arguments [ i ] ) ;
2020-09-06 09:56:52 +02:00
} else if ( par_type . is_hard_type ( ) & & ! is_type_compatible ( par_type , arg_type , true ) ) {
2020-06-11 00:53:25 +02:00
// Supertypes are acceptable for dynamic compliance, but it's unsafe.
mark_node_unsafe ( p_call ) ;
if ( ! is_type_compatible ( arg_type , par_type ) ) {
2022-12-04 04:02:03 +01:00
push_error ( vformat ( R " *(Invalid argument for " % s ( ) " function: argument %d should be " % s " but is " % s " .)* " ,
2020-06-11 00:53:25 +02:00
p_call - > function_name , i + 1 , par_type . to_string ( ) , arg_type . to_string ( ) ) ,
p_call - > arguments [ i ] ) ;
valid = false ;
}
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-11 00:53:25 +02:00
} else {
if ( par_type . kind = = GDScriptParser : : DataType : : BUILTIN & & par_type . builtin_type = = Variant : : INT & & arg_type . kind = = GDScriptParser : : DataType : : BUILTIN & & arg_type . builtin_type = = Variant : : FLOAT ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_call , GDScriptWarning : : NARROWING_CONVERSION , p_call - > function_name ) ;
2020-06-11 00:53:25 +02:00
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
}
return valid ;
}
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2020-06-12 00:31:28 +02:00
bool GDScriptAnalyzer : : is_shadowing ( GDScriptParser : : IdentifierNode * p_local , const String & p_context ) {
const StringName & name = p_local - > name ;
GDScriptParser : : DataType base = parser - > current_class - > get_datatype ( ) ;
GDScriptParser : : ClassNode * base_class = base . class_type ;
2021-12-13 09:39:16 +01:00
{
List < MethodInfo > gdscript_funcs ;
GDScriptLanguage : : get_singleton ( ) - > get_public_functions ( & gdscript_funcs ) ;
for ( MethodInfo & info : gdscript_funcs ) {
if ( info . name = = name ) {
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_GLOBAL_IDENTIFIER , p_context , name , " built-in function " ) ;
return true ;
}
}
if ( Variant : : has_utility_function ( name ) ) {
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_GLOBAL_IDENTIFIER , p_context , name , " built-in function " ) ;
return true ;
} else if ( ClassDB : : class_exists ( name ) ) {
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_GLOBAL_IDENTIFIER , p_context , name , " global class " ) ;
return true ;
}
}
2020-06-12 00:31:28 +02:00
while ( base_class ! = nullptr ) {
if ( base_class - > has_member ( name ) ) {
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_VARIABLE , p_context , p_local - > name , base_class - > get_member ( name ) . get_type_name ( ) , itos ( base_class - > get_member ( name ) . get_line ( ) ) ) ;
return true ;
}
base_class = base_class - > base_type . class_type ;
}
2022-12-11 03:57:35 +01:00
StringName parent = base . native_type ;
2020-06-12 00:31:28 +02:00
while ( parent ! = StringName ( ) ) {
2022-12-11 03:57:35 +01:00
ERR_FAIL_COND_V_MSG ( ! class_exists ( parent ) , false , " Non-existent native base class. " ) ;
2021-08-17 15:06:54 +02:00
if ( ClassDB : : has_method ( parent , name , true ) ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_VARIABLE_BASE_CLASS , p_context , p_local - > name , " method " , parent ) ;
return true ;
2021-08-17 15:06:54 +02:00
} else if ( ClassDB : : has_signal ( parent , name , true ) ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_VARIABLE_BASE_CLASS , p_context , p_local - > name , " signal " , parent ) ;
return true ;
2021-08-17 15:06:54 +02:00
} else if ( ClassDB : : has_property ( parent , name , true ) ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_VARIABLE_BASE_CLASS , p_context , p_local - > name , " property " , parent ) ;
return true ;
2021-08-17 15:06:54 +02:00
} else if ( ClassDB : : has_integer_constant ( parent , name , true ) ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_VARIABLE_BASE_CLASS , p_context , p_local - > name , " constant " , parent ) ;
return true ;
2021-08-17 15:06:54 +02:00
} else if ( ClassDB : : has_enum ( parent , name , true ) ) {
2020-06-12 00:31:28 +02:00
parser - > push_warning ( p_local , GDScriptWarning : : SHADOWED_VARIABLE_BASE_CLASS , p_context , p_local - > name , " enum " , parent ) ;
return true ;
}
2021-08-17 15:06:54 +02:00
parent = ClassDB : : get_parent_class ( parent ) ;
2020-06-12 00:31:28 +02:00
}
return false ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-12 00:31:28 +02:00
2020-11-26 18:41:55 +01:00
GDScriptParser : : DataType GDScriptAnalyzer : : get_operation_type ( Variant : : Operator p_operation , const GDScriptParser : : DataType & p_a , bool & r_valid , const GDScriptParser : : Node * p_source ) {
// Unary version.
GDScriptParser : : DataType nil_type ;
nil_type . builtin_type = Variant : : NIL ;
2021-12-23 13:45:24 +01:00
nil_type . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
2020-11-26 18:41:55 +01:00
return get_operation_type ( p_operation , p_a , nil_type , r_valid , p_source ) ;
}
2020-06-11 00:53:25 +02:00
2020-11-26 18:41:55 +01:00
GDScriptParser : : DataType GDScriptAnalyzer : : get_operation_type ( Variant : : Operator p_operation , const GDScriptParser : : DataType & p_a , const GDScriptParser : : DataType & p_b , bool & r_valid , const GDScriptParser : : Node * p_source ) {
2020-06-11 00:53:25 +02:00
GDScriptParser : : DataType result ;
result . kind = GDScriptParser : : DataType : : VARIANT ;
Variant : : Type a_type = p_a . builtin_type ;
Variant : : Type b_type = p_b . builtin_type ;
2022-02-02 17:57:24 +01:00
if ( p_a . kind = = GDScriptParser : : DataType : : ENUM ) {
if ( p_a . is_meta_type ) {
a_type = Variant : : DICTIONARY ;
} else {
a_type = Variant : : INT ;
}
}
if ( p_b . kind = = GDScriptParser : : DataType : : ENUM ) {
if ( p_b . is_meta_type ) {
b_type = Variant : : DICTIONARY ;
} else {
b_type = Variant : : INT ;
}
}
2020-11-26 18:41:55 +01:00
Variant : : ValidatedOperatorEvaluator op_eval = Variant : : get_validated_operator_evaluator ( p_operation , a_type , b_type ) ;
2020-06-11 00:53:25 +02:00
2021-12-23 13:45:24 +01:00
bool hard_operation = p_a . is_hard_type ( ) & & p_b . is_hard_type ( ) ;
bool validated = op_eval ! = nullptr ;
if ( hard_operation & & ! validated ) {
2020-11-26 18:41:55 +01:00
r_valid = false ;
return result ;
2021-12-23 13:45:24 +01:00
} else if ( hard_operation & & validated ) {
r_valid = true ;
result . type_source = GDScriptParser : : DataType : : ANNOTATED_INFERRED ;
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = Variant : : get_operator_return_type ( p_operation , a_type , b_type ) ;
} else if ( ! hard_operation & & ! validated ) {
r_valid = true ;
result . type_source = GDScriptParser : : DataType : : UNDETECTED ;
result . kind = GDScriptParser : : DataType : : VARIANT ;
result . builtin_type = Variant : : NIL ;
} else if ( ! hard_operation & & validated ) {
r_valid = true ;
result . type_source = GDScriptParser : : DataType : : INFERRED ;
result . kind = GDScriptParser : : DataType : : BUILTIN ;
result . builtin_type = Variant : : get_operator_return_type ( p_operation , a_type , b_type ) ;
2020-06-11 00:53:25 +02:00
}
return result ;
}
// TODO: Add safe/unsafe return variable (for variant cases)
2022-01-27 15:34:33 +01:00
bool GDScriptAnalyzer : : is_type_compatible ( const GDScriptParser : : DataType & p_target , const GDScriptParser : : DataType & p_source , bool p_allow_implicit_conversion , const GDScriptParser : : Node * p_source_node ) {
2020-06-11 00:53:25 +02:00
// These return "true" so it doesn't affect users negatively.
ERR_FAIL_COND_V_MSG ( ! p_target . is_set ( ) , true , " Parser bug (please report): Trying to check compatibility of unset target type " ) ;
ERR_FAIL_COND_V_MSG ( ! p_source . is_set ( ) , true , " Parser bug (please report): Trying to check compatibility of unset value type " ) ;
if ( p_target . kind = = GDScriptParser : : DataType : : VARIANT ) {
// Variant can receive anything.
return true ;
}
if ( p_source . kind = = GDScriptParser : : DataType : : VARIANT ) {
// TODO: This is acceptable but unsafe. Make sure unsafe line is set.
return true ;
}
if ( p_target . kind = = GDScriptParser : : DataType : : BUILTIN ) {
bool valid = p_source . kind = = GDScriptParser : : DataType : : BUILTIN & & p_target . builtin_type = = p_source . builtin_type ;
if ( ! valid & & p_allow_implicit_conversion ) {
valid = Variant : : can_convert_strict ( p_source . builtin_type , p_target . builtin_type ) ;
}
2022-01-27 15:34:33 +01:00
if ( ! valid & & p_target . builtin_type = = Variant : : INT & & p_source . kind = = GDScriptParser : : DataType : : ENUM & & ! p_source . is_meta_type ) {
2020-06-12 02:49:58 +02:00
// Enum value is also integer.
valid = true ;
}
2021-03-09 16:32:35 +01:00
if ( valid & & p_target . builtin_type = = Variant : : ARRAY & & p_source . builtin_type = = Variant : : ARRAY ) {
// Check the element type.
if ( p_target . has_container_element_type ( ) ) {
if ( ! p_source . has_container_element_type ( ) ) {
// TODO: Maybe this is valid but unsafe?
// Variant array can't be appended to typed array.
valid = false ;
} else {
2022-12-12 13:27:41 +01:00
valid = is_type_compatible ( p_target . get_container_element_type ( ) , p_source . get_container_element_type ( ) , p_allow_implicit_conversion ) ;
2021-03-09 16:32:35 +01:00
}
}
}
2020-06-11 00:53:25 +02:00
return valid ;
}
2020-06-12 02:49:58 +02:00
if ( p_target . kind = = GDScriptParser : : DataType : : ENUM ) {
if ( p_source . kind = = GDScriptParser : : DataType : : BUILTIN & & p_source . builtin_type = = Variant : : INT ) {
2022-01-27 15:34:33 +01:00
# ifdef DEBUG_ENABLED
if ( p_source_node ) {
parser - > push_warning ( p_source_node , GDScriptWarning : : INT_ASSIGNED_TO_ENUM ) ;
}
# endif
2020-06-12 02:49:58 +02:00
return true ;
}
2021-08-15 21:25:47 +02:00
if ( p_source . kind = = GDScriptParser : : DataType : : ENUM ) {
if ( p_source . native_type = = p_target . native_type ) {
return true ;
}
}
2020-06-12 02:49:58 +02:00
return false ;
}
2020-06-11 00:53:25 +02:00
// From here on the target type is an object, so we have to test polymorphism.
if ( p_source . kind = = GDScriptParser : : DataType : : BUILTIN & & p_source . builtin_type = = Variant : : NIL ) {
// null is acceptable in object.
return true ;
}
StringName src_native ;
Ref < Script > src_script ;
const GDScriptParser : : ClassNode * src_class = nullptr ;
switch ( p_source . kind ) {
case GDScriptParser : : DataType : : NATIVE :
if ( p_target . kind ! = GDScriptParser : : DataType : : NATIVE ) {
// Non-native class cannot be supertype of native.
return false ;
}
if ( p_source . is_meta_type ) {
src_native = GDScriptNativeClass : : get_class_static ( ) ;
} else {
src_native = p_source . native_type ;
}
break ;
case GDScriptParser : : DataType : : SCRIPT :
if ( p_target . kind = = GDScriptParser : : DataType : : CLASS ) {
// A script type cannot be a subtype of a GDScript class.
return false ;
}
if ( p_source . is_meta_type ) {
src_native = p_source . script_type - > get_class_name ( ) ;
} else {
src_script = p_source . script_type ;
src_native = src_script - > get_instance_base_type ( ) ;
}
break ;
case GDScriptParser : : DataType : : CLASS :
if ( p_source . is_meta_type ) {
src_native = GDScript : : get_class_static ( ) ;
} else {
src_class = p_source . class_type ;
const GDScriptParser : : ClassNode * base = src_class ;
while ( base - > base_type . kind = = GDScriptParser : : DataType : : CLASS ) {
base = base - > base_type . class_type ;
}
src_native = base - > base_type . native_type ;
src_script = base - > base_type . script_type ;
}
break ;
case GDScriptParser : : DataType : : VARIANT :
case GDScriptParser : : DataType : : BUILTIN :
2020-06-12 02:49:58 +02:00
case GDScriptParser : : DataType : : ENUM :
2022-12-11 03:57:35 +01:00
case GDScriptParser : : DataType : : RESOLVING :
2020-06-11 00:53:25 +02:00
case GDScriptParser : : DataType : : UNRESOLVED :
break ; // Already solved before.
}
switch ( p_target . kind ) {
case GDScriptParser : : DataType : : NATIVE : {
if ( p_target . is_meta_type ) {
return ClassDB : : is_parent_class ( src_native , GDScriptNativeClass : : get_class_static ( ) ) ;
}
2021-08-17 15:06:54 +02:00
return ClassDB : : is_parent_class ( src_native , p_target . native_type ) ;
2020-06-11 00:53:25 +02:00
}
case GDScriptParser : : DataType : : SCRIPT :
if ( p_target . is_meta_type ) {
return ClassDB : : is_parent_class ( src_native , p_target . script_type - > get_class_name ( ) ) ;
}
while ( src_script . is_valid ( ) ) {
if ( src_script = = p_target . script_type ) {
return true ;
}
src_script = src_script - > get_base_script ( ) ;
}
return false ;
case GDScriptParser : : DataType : : CLASS :
if ( p_target . is_meta_type ) {
return ClassDB : : is_parent_class ( src_native , GDScript : : get_class_static ( ) ) ;
}
while ( src_class ! = nullptr ) {
if ( src_class - > fqcn = = p_target . class_type - > fqcn ) {
return true ;
}
src_class = src_class - > base_type . class_type ;
}
return false ;
case GDScriptParser : : DataType : : VARIANT :
case GDScriptParser : : DataType : : BUILTIN :
2020-06-12 02:49:58 +02:00
case GDScriptParser : : DataType : : ENUM :
2022-12-11 03:57:35 +01:00
case GDScriptParser : : DataType : : RESOLVING :
2020-06-11 00:53:25 +02:00
case GDScriptParser : : DataType : : UNRESOLVED :
break ; // Already solved before.
}
return false ;
}
void GDScriptAnalyzer : : push_error ( const String & p_message , const GDScriptParser : : Node * p_origin ) {
mark_node_unsafe ( p_origin ) ;
parser - > push_error ( p_message , p_origin ) ;
}
void GDScriptAnalyzer : : mark_node_unsafe ( const GDScriptParser : : Node * p_node ) {
2020-07-16 03:02:44 +02:00
# ifdef DEBUG_ENABLED
2022-12-11 03:57:35 +01:00
if ( p_node = = nullptr ) {
return ;
}
2020-06-11 00:53:25 +02:00
for ( int i = p_node - > start_line ; i < = p_node - > end_line ; i + + ) {
parser - > unsafe_lines . insert ( i ) ;
}
2020-07-16 03:02:44 +02:00
# endif
2020-06-11 00:53:25 +02:00
}
2022-04-20 19:22:22 +02:00
void GDScriptAnalyzer : : mark_lambda_use_self ( ) {
for ( GDScriptParser : : LambdaNode * lambda : lambda_stack ) {
lambda - > use_self = true ;
}
}
2021-03-09 16:32:35 +01:00
bool GDScriptAnalyzer : : class_exists ( const StringName & p_class ) const {
2021-08-17 15:06:54 +02:00
return ClassDB : : class_exists ( p_class ) & & ClassDB : : is_class_exposed ( p_class ) ;
2020-06-11 00:53:25 +02:00
}
Ref < GDScriptParserRef > GDScriptAnalyzer : : get_parser_for ( const String & p_path ) {
Ref < GDScriptParserRef > ref ;
if ( depended_parsers . has ( p_path ) ) {
ref = depended_parsers [ p_path ] ;
} else {
Error err = OK ;
2020-07-16 03:02:44 +02:00
ref = GDScriptCache : : get_parser ( p_path , GDScriptParserRef : : EMPTY , err , parser - > script_path ) ;
2021-12-14 15:54:25 +01:00
if ( ref . is_valid ( ) ) {
depended_parsers [ p_path ] = ref ;
}
2020-06-11 00:53:25 +02:00
}
return ref ;
}
Error GDScriptAnalyzer : : resolve_inheritance ( ) {
2022-12-11 03:57:35 +01:00
return resolve_class_inheritance ( parser - > head , true ) ;
2020-06-11 00:53:25 +02:00
}
Error GDScriptAnalyzer : : resolve_interface ( ) {
2022-12-11 03:57:35 +01:00
resolve_class_interface ( parser - > head , true ) ;
2020-12-15 13:04:21 +01:00
return parser - > errors . is_empty ( ) ? OK : ERR_PARSE_ERROR ;
2020-06-11 00:53:25 +02:00
}
Error GDScriptAnalyzer : : resolve_body ( ) {
2022-12-11 03:57:35 +01:00
resolve_class_body ( parser - > head , true ) ;
2020-12-15 13:04:21 +01:00
return parser - > errors . is_empty ( ) ? OK : ERR_PARSE_ERROR ;
2020-06-11 00:53:25 +02:00
}
2022-12-11 03:57:35 +01:00
Error GDScriptAnalyzer : : resolve_dependencies ( ) {
2022-05-08 10:09:19 +02:00
for ( KeyValue < String , Ref < GDScriptParserRef > > & K : depended_parsers ) {
if ( K . value . is_null ( ) ) {
2020-08-26 19:50:27 +02:00
return ERR_PARSE_ERROR ;
}
2022-12-11 03:57:35 +01:00
K . value - > raise_status ( GDScriptParserRef : : INHERITANCE_SOLVED ) ;
2020-06-12 00:31:28 +02:00
}
2022-12-11 03:57:35 +01:00
2020-12-15 13:04:21 +01:00
return parser - > errors . is_empty ( ) ? OK : ERR_PARSE_ERROR ;
2020-05-02 00:14:56 +02:00
}
Error GDScriptAnalyzer : : analyze ( ) {
parser - > errors . clear ( ) ;
2022-12-11 03:57:35 +01:00
Error err = OK ;
err = resolve_inheritance ( ) ;
2020-05-02 00:14:56 +02:00
if ( err ) {
return err ;
}
2022-12-11 03:57:35 +01:00
resolve_interface ( ) ;
resolve_body ( ) ;
if ( ! parser - > errors . is_empty ( ) ) {
return ERR_PARSE_ERROR ;
}
return resolve_dependencies ( ) ;
2020-05-02 00:14:56 +02:00
}
GDScriptAnalyzer : : GDScriptAnalyzer ( GDScriptParser * p_parser ) {
parser = p_parser ;
}