a166833bfa
- Adds a list of default levels for all warning so they can be set individually. - Add warnings set by default to error for: - Using `get_node()` without `@onready`. - Using `@onready` together with `@export`. - Inferring a static type with a Variant value. - Overriding a native engine method. - Adjust how annotations to ignore warnings are treated so they also apply to method parameters. - Clean up a bit how ignored warnings are set. There were two sets but only one was actually being used. - Set all warnings to the `WARN` level for tests, so they they can be properly tested. - Fix enum types in native methods signatures being set to `int`. - Fix native enums being treated as Dictionary by mistake. - Make name of native enum types use the class they are defined in, not the direct super class of the script. This ensures they are always equal even when coming from different sources. - Fix error for signature mismatch that was only showing the first default argument as having a default. Now it shows for all.
5013 lines
194 KiB
C++
5013 lines
194 KiB
C++
/**************************************************************************/
|
|
/* 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. */
|
|
/**************************************************************************/
|
|
|
|
#include "gdscript_analyzer.h"
|
|
|
|
#include "core/config/engine.h"
|
|
#include "core/config/project_settings.h"
|
|
#include "core/core_string_names.h"
|
|
#include "core/io/file_access.h"
|
|
#include "core/io/resource_loader.h"
|
|
#include "core/object/class_db.h"
|
|
#include "core/object/script_language.h"
|
|
#include "core/templates/hash_map.h"
|
|
#include "gdscript.h"
|
|
#include "gdscript_utility_functions.h"
|
|
#include "scene/resources/packed_scene.h"
|
|
|
|
#define UNNAMED_ENUM "<anonymous enum>"
|
|
#define ENUM_SEPARATOR "::"
|
|
|
|
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);
|
|
info.arguments.push_back(pi);
|
|
}
|
|
}
|
|
|
|
return info;
|
|
}
|
|
|
|
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;
|
|
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;
|
|
type.is_meta_type = true;
|
|
return type;
|
|
}
|
|
|
|
// 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) {
|
|
GDScriptParser::DataType type;
|
|
type.type_source = GDScriptParser::DataType::ANNOTATED_EXPLICIT;
|
|
type.kind = GDScriptParser::DataType::ENUM;
|
|
type.builtin_type = p_meta ? Variant::DICTIONARY : Variant::INT;
|
|
type.enum_type = p_enum_name;
|
|
type.is_constant = true;
|
|
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) {
|
|
// Find out which base class declared the enum, so the name is always the same even when coming from other contexts.
|
|
StringName native_base = p_native_class;
|
|
while (true && native_base != StringName()) {
|
|
if (ClassDB::has_enum(native_base, p_enum_name, true)) {
|
|
break;
|
|
}
|
|
native_base = ClassDB::get_parent_class_nocheck(native_base);
|
|
}
|
|
|
|
GDScriptParser::DataType type = make_enum_type(p_enum_name, native_base, p_meta);
|
|
if (p_meta) {
|
|
type.builtin_type = Variant::NIL; // Native enum types are not Dictionaries
|
|
}
|
|
|
|
List<StringName> enum_values;
|
|
ClassDB::get_enum_constants(native_base, p_enum_name, &enum_values, true);
|
|
|
|
for (const StringName &E : enum_values) {
|
|
type.enum_values[E] = ClassDB::get_integer_constant(native_base, E);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
bool GDScriptAnalyzer::has_member_name_conflict_in_script_class(const StringName &p_member_name, const GDScriptParser::ClassNode *p_class, const GDScriptParser::Node *p_member) {
|
|
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;
|
|
}
|
|
if (p_member->type != GDScriptParser::Node::FUNCTION && member->type == GDScriptParser::ClassNode::Member::FUNCTION) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
if (p_member_name == CoreStringNames::get_singleton()->_script) {
|
|
return true;
|
|
}
|
|
|
|
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)) {
|
|
push_error(vformat(R"(The member "%s" shadows a native class.)", p_member_name), p_member_node);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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) {
|
|
// TODO check outer classes for static members only
|
|
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;
|
|
if (has_member_name_conflict_in_script_class(p_member_name, current_class_node, p_member_node)) {
|
|
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);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
current_data_type = ¤t_class_node->base_type;
|
|
}
|
|
|
|
// No need for native class recursion because Node exposes all Object's properties.
|
|
if (current_data_type && current_data_type->kind == GDScriptParser::DataType::Kind::NATIVE) {
|
|
if (current_data_type->native_type != StringName()) {
|
|
return check_native_member_name_conflict(
|
|
p_member_name,
|
|
p_member_node,
|
|
current_data_type->native_type);
|
|
}
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
void GDScriptAnalyzer::get_class_node_current_scope_classes(GDScriptParser::ClassNode *p_node, List<GDScriptParser::ClassNode *> *p_list) {
|
|
ERR_FAIL_NULL(p_node);
|
|
ERR_FAIL_NULL(p_list);
|
|
|
|
if (p_list->find(p_node) != nullptr) {
|
|
return;
|
|
}
|
|
|
|
p_list->push_back(p_node);
|
|
|
|
// TODO: Try to solve class inheritance if not yet resolving.
|
|
|
|
// 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);
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
GDScriptParser::ClassNode *previous_class = parser->current_class;
|
|
parser->current_class = p_class;
|
|
|
|
if (p_class->identifier) {
|
|
StringName class_name = p_class->identifier->name;
|
|
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)) {
|
|
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);
|
|
}
|
|
}
|
|
|
|
GDScriptParser::DataType resolving_datatype;
|
|
resolving_datatype.kind = GDScriptParser::DataType::RESOLVING;
|
|
p_class->base_type = resolving_datatype;
|
|
|
|
// 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;
|
|
class_type.script_path = parser->script_path;
|
|
class_type.builtin_type = Variant::OBJECT;
|
|
p_class->set_datatype(class_type);
|
|
|
|
GDScriptParser::DataType result;
|
|
if (!p_class->extends_used) {
|
|
result.type_source = GDScriptParser::DataType::ANNOTATED_INFERRED;
|
|
result.kind = GDScriptParser::DataType::NATIVE;
|
|
result.native_type = SNAME("RefCounted");
|
|
} else {
|
|
result.type_source = GDScriptParser::DataType::ANNOTATED_EXPLICIT;
|
|
|
|
GDScriptParser::DataType base;
|
|
|
|
int extends_index = 0;
|
|
|
|
if (!p_class->extends_path.is_empty()) {
|
|
if (p_class->extends_path.is_relative_path()) {
|
|
p_class->extends_path = class_type.script_path.get_base_dir().path_join(p_class->extends_path).simplify_path();
|
|
}
|
|
Ref<GDScriptParserRef> ext_parser = get_parser_for(p_class->extends_path);
|
|
if (ext_parser.is_null()) {
|
|
push_error(vformat(R"(Could not resolve super class path "%s".)", p_class->extends_path), p_class);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Error err = ext_parser->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
if (err != OK) {
|
|
push_error(vformat(R"(Could not resolve super class inheritance from "%s".)", p_class->extends_path), p_class);
|
|
return err;
|
|
}
|
|
|
|
base = ext_parser->get_parser()->head->get_datatype();
|
|
} else {
|
|
if (p_class->extends.is_empty()) {
|
|
push_error("Could not resolve an empty super class path.", p_class);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
const StringName &name = p_class->extends[extends_index++];
|
|
base.type_source = GDScriptParser::DataType::ANNOTATED_EXPLICIT;
|
|
|
|
if (ScriptServer::is_global_class(name)) {
|
|
String base_path = ScriptServer::get_global_class_path(name);
|
|
|
|
if (base_path == parser->script_path) {
|
|
base = parser->head->get_datatype();
|
|
} else {
|
|
Ref<GDScriptParserRef> base_parser = get_parser_for(base_path);
|
|
if (base_parser.is_null()) {
|
|
push_error(vformat(R"(Could not resolve super class "%s".)", name), p_class);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Error err = base_parser->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
if (err != OK) {
|
|
push_error(vformat(R"(Could not resolve super class inheritance from "%s".)", name), p_class);
|
|
return err;
|
|
}
|
|
base = base_parser->get_parser()->head->get_datatype();
|
|
}
|
|
} 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);
|
|
if (info.path.get_extension().to_lower() != GDScriptLanguage::get_singleton()->get_extension()) {
|
|
push_error(vformat(R"(Singleton %s is not a GDScript.)", info.name), p_class);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Ref<GDScriptParserRef> info_parser = get_parser_for(info.path);
|
|
if (info_parser.is_null()) {
|
|
push_error(vformat(R"(Could not parse singleton from "%s".)", info.path), p_class);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Error err = info_parser->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
if (err != OK) {
|
|
push_error(vformat(R"(Could not resolve super class inheritance from "%s".)", name), p_class);
|
|
return err;
|
|
}
|
|
base = info_parser->get_parser()->head->get_datatype();
|
|
} else if (class_exists(name)) {
|
|
base.kind = GDScriptParser::DataType::NATIVE;
|
|
base.native_type = name;
|
|
} else {
|
|
// Look for other classes in script.
|
|
bool found = false;
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
get_class_node_current_scope_classes(p_class, &script_classes);
|
|
for (GDScriptParser::ClassNode *look_class : script_classes) {
|
|
if (look_class->identifier && look_class->identifier->name == name) {
|
|
if (!look_class->get_datatype().is_set()) {
|
|
Error err = resolve_class_inheritance(look_class, p_class);
|
|
if (err) {
|
|
return err;
|
|
}
|
|
}
|
|
base = look_class->get_datatype();
|
|
found = true;
|
|
break;
|
|
}
|
|
if (look_class->has_member(name)) {
|
|
resolve_class_member(look_class, name, p_class);
|
|
base = look_class->get_member(name).get_datatype();
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
base = id_type;
|
|
}
|
|
|
|
result = base;
|
|
}
|
|
|
|
if (!result.is_set() || result.has_no_type()) {
|
|
// TODO: More specific error messages.
|
|
push_error(vformat(R"(Could not resolve inheritance for class "%s".)", p_class->identifier == nullptr ? "<main>" : p_class->identifier->name), p_class);
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
// 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;
|
|
}
|
|
|
|
p_class->base_type = result;
|
|
class_type.native_type = result.native_type;
|
|
p_class->set_datatype(class_type);
|
|
|
|
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;
|
|
}
|
|
|
|
if (p_recursive) {
|
|
for (int i = 0; i < p_class->members.size(); i++) {
|
|
if (p_class->members[i].type == GDScriptParser::ClassNode::Member::CLASS) {
|
|
err = resolve_class_inheritance(p_class->members[i].m_class, true);
|
|
if (err) {
|
|
return err;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
GDScriptParser::DataType GDScriptAnalyzer::resolve_datatype(GDScriptParser::TypeNode *p_type) {
|
|
GDScriptParser::DataType bad_type;
|
|
bad_type.kind = GDScriptParser::DataType::VARIANT;
|
|
bad_type.type_source = GDScriptParser::DataType::INFERRED;
|
|
|
|
if (p_type == nullptr) {
|
|
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();
|
|
}
|
|
|
|
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;
|
|
result.builtin_type = Variant::OBJECT;
|
|
|
|
if (p_type->type_chain.is_empty()) {
|
|
// void.
|
|
result.kind = GDScriptParser::DataType::BUILTIN;
|
|
result.builtin_type = Variant::NIL;
|
|
p_type->set_datatype(result);
|
|
return result;
|
|
}
|
|
|
|
StringName first = p_type->type_chain[0]->name;
|
|
|
|
if (first == SNAME("Variant")) {
|
|
if (p_type->type_chain.size() > 1) {
|
|
// 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;
|
|
}
|
|
result.kind = GDScriptParser::DataType::VARIANT;
|
|
} else if (first == SNAME("Object")) {
|
|
// Object is treated like a native type, not a built-in.
|
|
result.kind = GDScriptParser::DataType::NATIVE;
|
|
result.native_type = SNAME("Object");
|
|
} else if (GDScriptParser::get_builtin_type(first) < Variant::VARIANT_MAX) {
|
|
// Built-in types.
|
|
if (p_type->type_chain.size() > 1) {
|
|
push_error(R"(Built-in types don't contain nested types.)", p_type->type_chain[1]);
|
|
return bad_type;
|
|
}
|
|
result.kind = GDScriptParser::DataType::BUILTIN;
|
|
result.builtin_type = GDScriptParser::get_builtin_type(first);
|
|
|
|
if (result.builtin_type == Variant::ARRAY) {
|
|
GDScriptParser::DataType container_type = type_from_metatype(resolve_datatype(p_type->container_type));
|
|
if (container_type.kind != GDScriptParser::DataType::VARIANT) {
|
|
container_type.is_constant = false;
|
|
result.set_container_element_type(container_type);
|
|
}
|
|
}
|
|
} else if (class_exists(first)) {
|
|
// Native engine classes.
|
|
result.kind = GDScriptParser::DataType::NATIVE;
|
|
result.native_type = first;
|
|
} else if (ScriptServer::is_global_class(first)) {
|
|
if (parser->script_path == ScriptServer::get_global_class_path(first)) {
|
|
result = parser->head->get_datatype();
|
|
} else {
|
|
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);
|
|
if (!ref.is_valid() || ref->raise_status(GDScriptParserRef::INHERITANCE_SOLVED) != OK) {
|
|
push_error(vformat(R"(Could not parse global class "%s" from "%s".)", first, ScriptServer::get_global_class_path(first)), p_type);
|
|
return bad_type;
|
|
}
|
|
result = ref->get_parser()->head->get_datatype();
|
|
} else {
|
|
result = make_script_meta_type(ResourceLoader::load(path, "Script"));
|
|
}
|
|
}
|
|
} 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);
|
|
if (ref->raise_status(GDScriptParserRef::INHERITANCE_SOLVED) != OK) {
|
|
push_error(vformat(R"(Could not parse singleton "%s" from "%s".)", first, autoload.path), p_type);
|
|
return bad_type;
|
|
}
|
|
result = ref->get_parser()->head->get_datatype();
|
|
} else if (ClassDB::has_enum(parser->current_class->base_type.native_type, first)) {
|
|
// Native enum in current class.
|
|
result = make_native_enum_type(first, parser->current_class->base_type.native_type);
|
|
} else {
|
|
// Classes in current scope.
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
bool found = false;
|
|
get_class_node_current_scope_classes(parser->current_class, &script_classes);
|
|
for (GDScriptParser::ClassNode *script_class : script_classes) {
|
|
if (found) {
|
|
break;
|
|
}
|
|
|
|
if (script_class->identifier && script_class->identifier->name == first) {
|
|
result = script_class->get_datatype();
|
|
break;
|
|
}
|
|
if (script_class->members_indices.has(first)) {
|
|
resolve_class_member(script_class, first, p_type);
|
|
|
|
GDScriptParser::ClassNode::Member member = script_class->get_member(first);
|
|
switch (member.type) {
|
|
case GDScriptParser::ClassNode::Member::CLASS:
|
|
result = member.get_datatype();
|
|
found = true;
|
|
break;
|
|
case GDScriptParser::ClassNode::Member::ENUM:
|
|
result = member.get_datatype();
|
|
found = true;
|
|
break;
|
|
case GDScriptParser::ClassNode::Member::CONSTANT:
|
|
if (member.get_datatype().is_meta_type) {
|
|
result = member.get_datatype();
|
|
found = true;
|
|
break;
|
|
} else if (Ref<Script>(member.constant->initializer->reduced_value).is_valid()) {
|
|
Ref<GDScript> gdscript = member.constant->initializer->reduced_value;
|
|
if (gdscript.is_valid()) {
|
|
Ref<GDScriptParserRef> ref = get_parser_for(gdscript->get_script_path());
|
|
if (ref->raise_status(GDScriptParserRef::INHERITANCE_SOLVED) != OK) {
|
|
push_error(vformat(R"(Could not parse script from "%s".)", gdscript->get_script_path()), p_type);
|
|
return bad_type;
|
|
}
|
|
result = ref->get_parser()->head->get_datatype();
|
|
} else {
|
|
result = make_script_meta_type(member.constant->initializer->reduced_value);
|
|
}
|
|
found = true;
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
default:
|
|
push_error(vformat(R"("%s" is a %s but does not contain a type.)", first, member.get_type_name()), p_type);
|
|
return bad_type;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (!result.is_set()) {
|
|
push_error(vformat(R"("%s" was not found in the current scope.)", first), p_type);
|
|
return bad_type;
|
|
}
|
|
|
|
if (p_type->type_chain.size() > 1) {
|
|
if (result.kind == GDScriptParser::DataType::CLASS) {
|
|
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]);
|
|
return bad_type;
|
|
} 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]);
|
|
return bad_type;
|
|
}
|
|
}
|
|
} else if (result.kind == GDScriptParser::DataType::NATIVE) {
|
|
// Only enums allowed for native.
|
|
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]);
|
|
return bad_type;
|
|
}
|
|
} 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]);
|
|
return bad_type;
|
|
}
|
|
}
|
|
|
|
if (result.builtin_type != Variant::ARRAY && p_type->container_type != nullptr) {
|
|
push_error("Only arrays can specify the collection element type.", p_type);
|
|
}
|
|
|
|
p_type->set_datatype(result);
|
|
return result;
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// 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;
|
|
}
|
|
}
|
|
|
|
GDScriptParser::ClassNode *previous_class = parser->current_class;
|
|
parser->current_class = p_class;
|
|
|
|
GDScriptParser::DataType resolving_datatype;
|
|
resolving_datatype.kind = GDScriptParser::DataType::RESOLVING;
|
|
|
|
{
|
|
#ifdef DEBUG_ENABLED
|
|
HashSet<GDScriptWarning::Code> previously_ignored_warnings = parser->ignored_warnings;
|
|
GDScriptParser::Node *member_node = member.get_source_node();
|
|
if (member_node && member_node->type != GDScriptParser::Node::ANNOTATION) {
|
|
// Apply @warning_ignore annotations before resolving member.
|
|
for (GDScriptParser::AnnotationNode *&E : member_node->annotations) {
|
|
if (E->name == SNAME("@warning_ignore")) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.variable);
|
|
}
|
|
}
|
|
for (GDScriptWarning::Code ignored_warning : member_node->ignored_warnings) {
|
|
parser->ignored_warnings.insert(ignored_warning);
|
|
}
|
|
}
|
|
#endif
|
|
switch (member.type) {
|
|
case GDScriptParser::ClassNode::Member::VARIABLE: {
|
|
check_class_member_name_conflict(p_class, member.variable->identifier->name, member.variable);
|
|
member.variable->set_datatype(resolving_datatype);
|
|
resolve_variable(member.variable, false);
|
|
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : member.variable->annotations) {
|
|
if (E->name != SNAME("@warning_ignore")) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.variable);
|
|
}
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
if (member.variable->exported && member.variable->onready) {
|
|
parser->push_warning(member.variable, GDScriptWarning::ONREADY_WITH_EXPORT);
|
|
}
|
|
#endif
|
|
} break;
|
|
case GDScriptParser::ClassNode::Member::CONSTANT: {
|
|
check_class_member_name_conflict(p_class, member.constant->identifier->name, member.constant);
|
|
member.constant->set_datatype(resolving_datatype);
|
|
resolve_constant(member.constant, false);
|
|
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : member.constant->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.constant);
|
|
}
|
|
} break;
|
|
case GDScriptParser::ClassNode::Member::SIGNAL: {
|
|
check_class_member_name_conflict(p_class, member.signal->identifier->name, member.signal);
|
|
|
|
member.signal->set_datatype(resolving_datatype);
|
|
|
|
// 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);
|
|
|
|
for (int j = 0; j < member.signal->parameters.size(); j++) {
|
|
GDScriptParser::ParameterNode *param = member.signal->parameters[j];
|
|
GDScriptParser::DataType param_type = type_from_metatype(resolve_datatype(param->datatype_specifier));
|
|
param->set_datatype(param_type);
|
|
mi.arguments.push_back(PropertyInfo(param_type.builtin_type, param->identifier->name));
|
|
// TODO: add signal parameter default values
|
|
}
|
|
member.signal->set_datatype(make_signal_type(mi));
|
|
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : member.signal->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.signal);
|
|
}
|
|
} break;
|
|
case GDScriptParser::ClassNode::Member::ENUM: {
|
|
check_class_member_name_conflict(p_class, member.m_enum->identifier->name, member.m_enum);
|
|
|
|
member.m_enum->set_datatype(resolving_datatype);
|
|
GDScriptParser::DataType enum_type = make_enum_type(member.m_enum->identifier->name, p_class->fqcn, true);
|
|
|
|
const GDScriptParser::EnumNode *prev_enum = current_enum;
|
|
current_enum = member.m_enum;
|
|
|
|
Dictionary dictionary;
|
|
for (int j = 0; j < member.m_enum->values.size(); j++) {
|
|
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;
|
|
dictionary[String(element.identifier->name)] = element.value;
|
|
}
|
|
|
|
current_enum = prev_enum;
|
|
|
|
dictionary.make_read_only();
|
|
member.m_enum->set_datatype(enum_type);
|
|
member.m_enum->dictionary = dictionary;
|
|
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : member.m_enum->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.m_enum);
|
|
}
|
|
} break;
|
|
case GDScriptParser::ClassNode::Member::FUNCTION:
|
|
for (GDScriptParser::AnnotationNode *&E : member.function->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.function);
|
|
}
|
|
resolve_function_signature(member.function, p_source);
|
|
break;
|
|
case GDScriptParser::ClassNode::Member::ENUM_VALUE: {
|
|
member.enum_value.identifier->set_datatype(resolving_datatype);
|
|
|
|
if (member.enum_value.custom_value) {
|
|
check_class_member_name_conflict(p_class, member.enum_value.identifier->name, member.enum_value.custom_value);
|
|
|
|
const GDScriptParser::EnumNode *prev_enum = current_enum;
|
|
current_enum = member.enum_value.parent_enum;
|
|
reduce_expression(member.enum_value.custom_value);
|
|
current_enum = prev_enum;
|
|
|
|
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 {
|
|
check_class_member_name_conflict(p_class, member.enum_value.identifier->name, member.enum_value.parent_enum);
|
|
|
|
if (member.enum_value.index > 0) {
|
|
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;
|
|
} else {
|
|
member.enum_value.value = 0;
|
|
}
|
|
member.enum_value.resolved = true;
|
|
}
|
|
|
|
// Also update the original references.
|
|
member.enum_value.parent_enum->values.set(member.enum_value.index, member.enum_value);
|
|
|
|
member.enum_value.identifier->set_datatype(make_enum_type(UNNAMED_ENUM, p_class->fqcn, false));
|
|
} break;
|
|
case GDScriptParser::ClassNode::Member::CLASS:
|
|
check_class_member_name_conflict(p_class, member.m_class->identifier->name, member.m_class);
|
|
// If it's already resolving, that's ok.
|
|
if (!member.m_class->base_type.is_resolving()) {
|
|
resolve_class_inheritance(member.m_class, p_source);
|
|
}
|
|
break;
|
|
case GDScriptParser::ClassNode::Member::GROUP:
|
|
// No-op, but needed to silence warnings.
|
|
break;
|
|
case GDScriptParser::ClassNode::Member::UNDEFINED:
|
|
ERR_PRINT("Trying to resolve undefined member.");
|
|
break;
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
parser->ignored_warnings = previously_ignored_warnings;
|
|
#endif
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
for (int i = 0; i < p_class->members.size(); i++) {
|
|
resolve_class_member(p_class, i);
|
|
}
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
if (p_class->resolved_body) {
|
|
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".)", 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;
|
|
}
|
|
|
|
p_class->resolved_body = true;
|
|
|
|
GDScriptParser::ClassNode *previous_class = parser->current_class;
|
|
parser->current_class = p_class;
|
|
|
|
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);
|
|
}
|
|
|
|
// Do functions, properties, and groups now.
|
|
for (int i = 0; i < p_class->members.size(); i++) {
|
|
GDScriptParser::ClassNode::Member member = p_class->members[i];
|
|
if (member.type == GDScriptParser::ClassNode::Member::FUNCTION) {
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : member.function->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, member.function);
|
|
}
|
|
resolve_function_body(member.function);
|
|
} 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);
|
|
|
|
resolve_function_body(member.variable->getter);
|
|
}
|
|
if (member.variable->setter != nullptr) {
|
|
resolve_function_signature(member.variable->setter);
|
|
|
|
if (member.variable->setter->parameters.size() > 0) {
|
|
member.variable->setter->parameters[0]->datatype_specifier = member.variable->datatype_specifier;
|
|
member.variable->setter->parameters[0]->set_datatype(member.get_datatype());
|
|
}
|
|
|
|
resolve_function_body(member.variable->setter);
|
|
}
|
|
}
|
|
} else if (member.type == GDScriptParser::ClassNode::Member::GROUP) {
|
|
// Apply annotation (`@export_{category,group,subgroup}`).
|
|
resolve_annotation(member.annotation);
|
|
member.annotation->apply(parser, nullptr);
|
|
}
|
|
}
|
|
|
|
// Check unused variables and datatypes of property getters and setters.
|
|
for (int i = 0; i < p_class->members.size(); i++) {
|
|
GDScriptParser::ClassNode::Member member = p_class->members[i];
|
|
if (member.type == GDScriptParser::ClassNode::Member::VARIABLE) {
|
|
#ifdef DEBUG_ENABLED
|
|
HashSet<GDScriptWarning::Code> previously_ignored_warnings = parser->ignored_warnings;
|
|
for (GDScriptWarning::Code ignored_warning : member.variable->ignored_warnings) {
|
|
parser->ignored_warnings.insert(ignored_warning);
|
|
}
|
|
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 {
|
|
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);
|
|
|
|
} else {
|
|
has_valid_getter = true;
|
|
#ifdef DEBUG_ENABLED
|
|
if (member.variable->datatype.builtin_type == Variant::INT && return_datatype.builtin_type == Variant::FLOAT) {
|
|
parser->push_warning(member.variable, GDScriptWarning::NARROWING_CONVERSION);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
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
|
|
if (member.variable->datatype.builtin_type == Variant::FLOAT && setter_function->parameters[0]->datatype.builtin_type == Variant::INT) {
|
|
parser->push_warning(member.variable, GDScriptWarning::NARROWING_CONVERSION);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
parser->ignored_warnings = previously_ignored_warnings;
|
|
#endif // DEBUG_ENABLED
|
|
}
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::resolve_node(GDScriptParser::Node *p_node, bool p_is_root) {
|
|
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:
|
|
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);
|
|
}
|
|
break;
|
|
case GDScriptParser::Node::CONSTANT:
|
|
resolve_constant(static_cast<GDScriptParser::ConstantNode *>(p_node), true);
|
|
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:
|
|
resolve_variable(static_cast<GDScriptParser::VariableNode *>(p_node), true);
|
|
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:
|
|
resolve_parameter(static_cast<GDScriptParser::ParameterNode *>(p_node));
|
|
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:
|
|
case GDScriptParser::Node::LAMBDA:
|
|
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:
|
|
reduce_expression(static_cast<GDScriptParser::ExpressionNode *>(p_node), p_is_root);
|
|
break;
|
|
case GDScriptParser::Node::BREAK:
|
|
case GDScriptParser::Node::BREAKPOINT:
|
|
case GDScriptParser::Node::CONTINUE:
|
|
case GDScriptParser::Node::ENUM:
|
|
case GDScriptParser::Node::FUNCTION:
|
|
case GDScriptParser::Node::PASS:
|
|
case GDScriptParser::Node::SIGNAL:
|
|
// Nothing to do.
|
|
break;
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::resolve_annotation(GDScriptParser::AnnotationNode *p_annotation) {
|
|
ERR_FAIL_COND_MSG(!parser->valid_annotations.has(p_annotation->name), vformat(R"(Annotation "%s" not found to validate.)", p_annotation->name));
|
|
|
|
if (p_annotation->is_resolved) {
|
|
return;
|
|
}
|
|
p_annotation->is_resolved = true;
|
|
|
|
const MethodInfo &annotation_info = parser->valid_annotations[p_annotation->name].info;
|
|
|
|
const List<PropertyInfo>::Element *E = annotation_info.arguments.front();
|
|
for (int i = 0; i < p_annotation->arguments.size(); i++) {
|
|
GDScriptParser::ExpressionNode *argument = p_annotation->arguments[i];
|
|
const PropertyInfo &argument_info = E->get();
|
|
|
|
if (E->next() != nullptr) {
|
|
E = E->next();
|
|
}
|
|
|
|
reduce_expression(argument);
|
|
|
|
if (!argument->is_constant) {
|
|
push_error(vformat(R"(Argument %d of annotation "%s" isn't a constant expression.)", i + 1, p_annotation->name), argument);
|
|
return;
|
|
}
|
|
|
|
Variant value = argument->reduced_value;
|
|
|
|
if (value.get_type() != argument_info.type) {
|
|
#ifdef DEBUG_ENABLED
|
|
if (argument_info.type == Variant::INT && value.get_type() == Variant::FLOAT) {
|
|
parser->push_warning(argument, GDScriptWarning::NARROWING_CONVERSION);
|
|
}
|
|
#endif
|
|
|
|
if (!Variant::can_convert_strict(value.get_type(), argument_info.type)) {
|
|
push_error(vformat(R"(Invalid argument for annotation "%s": argument %d should be "%s" but is "%s".)", p_annotation->name, i + 1, Variant::get_type_name(argument_info.type), argument->get_datatype().to_string()), argument);
|
|
return;
|
|
}
|
|
|
|
Variant converted_to;
|
|
const Variant *converted_from = &value;
|
|
Callable::CallError call_error;
|
|
Variant::construct(argument_info.type, converted_to, &converted_from, 1, call_error);
|
|
|
|
if (call_error.error != Callable::CallError::CALL_OK) {
|
|
push_error(vformat(R"(Cannot convert argument %d of annotation "%s" from "%s" to "%s".)", i + 1, p_annotation->name, Variant::get_type_name(value.get_type()), Variant::get_type_name(argument_info.type)), argument);
|
|
return;
|
|
}
|
|
|
|
value = converted_to;
|
|
}
|
|
|
|
p_annotation->resolved_arguments.push_back(value);
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::resolve_function_signature(GDScriptParser::FunctionNode *p_function, const GDScriptParser::Node *p_source, bool p_is_lambda) {
|
|
if (p_source == nullptr) {
|
|
p_source = p_function;
|
|
}
|
|
|
|
StringName function_name = p_function->identifier != nullptr ? p_function->identifier->name : StringName();
|
|
|
|
if (p_function->get_datatype().is_resolving()) {
|
|
push_error(vformat(R"(Could not resolve function "%s": Cyclic reference.)", function_name), p_source);
|
|
return;
|
|
}
|
|
|
|
if (p_function->resolved_signature) {
|
|
return;
|
|
}
|
|
p_function->resolved_signature = true;
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
HashSet<GDScriptWarning::Code> previously_ignored_warnings = parser->ignored_warnings;
|
|
for (GDScriptWarning::Code ignored_warning : p_function->ignored_warnings) {
|
|
parser->ignored_warnings.insert(ignored_warning);
|
|
}
|
|
#endif
|
|
|
|
GDScriptParser::FunctionNode *previous_function = parser->current_function;
|
|
parser->current_function = p_function;
|
|
|
|
GDScriptParser::DataType prev_datatype = p_function->get_datatype();
|
|
|
|
GDScriptParser::DataType resolving_datatype;
|
|
resolving_datatype.kind = GDScriptParser::DataType::RESOLVING;
|
|
p_function->set_datatype(resolving_datatype);
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
int default_value_count = 0;
|
|
#endif // TOOLS_ENABLED
|
|
|
|
for (int i = 0; i < p_function->parameters.size(); i++) {
|
|
resolve_parameter(p_function->parameters[i]);
|
|
#ifdef DEBUG_ENABLED
|
|
if (p_function->parameters[i]->usages == 0 && !String(p_function->parameters[i]->identifier->name).begins_with("_")) {
|
|
parser->push_warning(p_function->parameters[i]->identifier, GDScriptWarning::UNUSED_PARAMETER, function_name, p_function->parameters[i]->identifier->name);
|
|
}
|
|
is_shadowing(p_function->parameters[i]->identifier, "function parameter");
|
|
#endif // DEBUG_ENABLED
|
|
#ifdef TOOLS_ENABLED
|
|
if (p_function->parameters[i]->initializer) {
|
|
default_value_count++;
|
|
|
|
if (p_function->parameters[i]->initializer->is_constant) {
|
|
p_function->default_arg_values.push_back(p_function->parameters[i]->initializer->reduced_value);
|
|
} else {
|
|
p_function->default_arg_values.push_back(Variant()); // Prevent shift.
|
|
}
|
|
}
|
|
#endif // TOOLS_ENABLED
|
|
}
|
|
|
|
if (!p_is_lambda && function_name == GDScriptLanguage::get_singleton()->strings._init) {
|
|
// 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) {
|
|
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);
|
|
}
|
|
}
|
|
} else {
|
|
if (p_function->return_type != nullptr) {
|
|
p_function->set_datatype(type_from_metatype(resolve_datatype(p_function->return_type)));
|
|
} 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);
|
|
}
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
// 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;
|
|
StringName native_base;
|
|
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, &native_base)) {
|
|
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;
|
|
valid = valid && default_value_count >= default_par_count + par_count_diff;
|
|
|
|
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.
|
|
String parent_signature = String(function_name) + "(";
|
|
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++;
|
|
}
|
|
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;
|
|
}
|
|
|
|
push_error(vformat(R"(The function signature doesn't match the parent. Parent signature is "%s".)", parent_signature), p_function);
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
if (native_base != StringName()) {
|
|
parser->push_warning(p_function, GDScriptWarning::NATIVE_METHOD_OVERRIDE, function_name, native_base);
|
|
}
|
|
#endif
|
|
}
|
|
#endif // TOOLS_ENABLED
|
|
}
|
|
|
|
if (p_function->get_datatype().is_resolving()) {
|
|
p_function->set_datatype(prev_datatype);
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
parser->ignored_warnings = previously_ignored_warnings;
|
|
#endif
|
|
parser->current_function = previous_function;
|
|
}
|
|
|
|
void GDScriptAnalyzer::resolve_function_body(GDScriptParser::FunctionNode *p_function, bool p_is_lambda) {
|
|
if (p_function->resolved_body) {
|
|
return;
|
|
}
|
|
p_function->resolved_body = true;
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
HashSet<GDScriptWarning::Code> previously_ignored_warnings = parser->ignored_warnings;
|
|
for (GDScriptWarning::Code ignored_warning : p_function->ignored_warnings) {
|
|
parser->ignored_warnings.insert(ignored_warning);
|
|
}
|
|
#endif
|
|
|
|
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();
|
|
|
|
if (!p_function->get_datatype().is_hard_type() && return_type.is_set()) {
|
|
// 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());
|
|
} 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)) {
|
|
if (!p_function->body->has_return && (p_is_lambda || p_function->identifier->name != GDScriptLanguage::get_singleton()->strings._init)) {
|
|
push_error(R"(Not all code paths return a value.)", p_function);
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
parser->ignored_warnings = previously_ignored_warnings;
|
|
#endif
|
|
parser->current_function = previous_function;
|
|
}
|
|
|
|
void GDScriptAnalyzer::decide_suite_type(GDScriptParser::Node *p_suite, GDScriptParser::Node *p_statement) {
|
|
if (p_statement == nullptr) {
|
|
return;
|
|
}
|
|
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());
|
|
p_suite->datatype.type_source = GDScriptParser::DataType::INFERRED;
|
|
}
|
|
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];
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : stmt->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, stmt);
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
HashSet<GDScriptWarning::Code> previously_ignored_warnings = parser->ignored_warnings;
|
|
for (GDScriptWarning::Code ignored_warning : stmt->ignored_warnings) {
|
|
parser->ignored_warnings.insert(ignored_warning);
|
|
}
|
|
#endif // DEBUG_ENABLED
|
|
|
|
resolve_node(stmt);
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
parser->ignored_warnings = previously_ignored_warnings;
|
|
#endif // DEBUG_ENABLED
|
|
|
|
decide_suite_type(p_suite, stmt);
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::resolve_assignable(GDScriptParser::AssignableNode *p_assignable, const char *p_kind) {
|
|
GDScriptParser::DataType type;
|
|
type.kind = GDScriptParser::DataType::VARIANT;
|
|
|
|
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) {
|
|
specified_type = type_from_metatype(resolve_datatype(p_assignable->datatype_specifier));
|
|
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 (has_specified_type && specified_type.has_container_element_type()) {
|
|
update_array_literal_element_type(array, specified_type.get_container_element_type());
|
|
}
|
|
}
|
|
|
|
if (is_constant && !p_assignable->initializer->is_constant) {
|
|
bool is_initializer_value_reduced = false;
|
|
Variant initializer_value = make_expression_reduced_value(p_assignable->initializer, is_initializer_value_reduced);
|
|
if (is_initializer_value_reduced) {
|
|
p_assignable->initializer->is_constant = true;
|
|
p_assignable->initializer->reduced_value = initializer_value;
|
|
} else {
|
|
push_error(vformat(R"(Assigned value for %s "%s" isn't a constant expression.)", p_kind, p_assignable->identifier->name), p_assignable->initializer);
|
|
}
|
|
}
|
|
|
|
if (has_specified_type && p_assignable->initializer->is_constant) {
|
|
update_const_expression_builtin_type(p_assignable->initializer, specified_type, "assign");
|
|
}
|
|
GDScriptParser::DataType initializer_type = p_assignable->initializer->get_datatype();
|
|
|
|
if (p_assignable->infer_datatype) {
|
|
if (!initializer_type.is_set() || initializer_type.has_no_type() || !initializer_type.is_hard_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.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);
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
if (initializer_type.is_hard_type() && initializer_type.is_variant()) {
|
|
parser->push_warning(p_assignable, GDScriptWarning::INFERENCE_ON_VARIANT, p_kind);
|
|
}
|
|
#endif
|
|
} 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);
|
|
p_assignable->use_conversion_assign = true;
|
|
if (!initializer_type.is_variant() && !is_type_compatible(specified_type, initializer_type, true, p_assignable->initializer)) {
|
|
downgrade_node_type_source(p_assignable->initializer);
|
|
}
|
|
} else if (!is_type_compatible(specified_type, initializer_type, true, p_assignable->initializer)) {
|
|
if (!is_constant && is_type_compatible(initializer_type, specified_type)) {
|
|
mark_node_unsafe(p_assignable->initializer);
|
|
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);
|
|
}
|
|
} else if (specified_type.has_container_element_type() && !initializer_type.has_container_element_type()) {
|
|
mark_node_unsafe(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);
|
|
} else {
|
|
// Check if it is call to get_node() on self (using shorthand $ or not), so we can check if @onready is needed.
|
|
if (p_variable->initializer && (p_variable->initializer->type == GDScriptParser::Node::GET_NODE || p_variable->initializer->type == GDScriptParser::Node::CALL)) {
|
|
bool is_get_node = p_variable->initializer->type == GDScriptParser::Node::GET_NODE;
|
|
bool is_using_shorthand = is_get_node;
|
|
if (!is_get_node) {
|
|
is_using_shorthand = false;
|
|
GDScriptParser::CallNode *call = static_cast<GDScriptParser::CallNode *>(p_variable->initializer);
|
|
if (call->function_name == SNAME("get_node")) {
|
|
switch (call->get_callee_type()) {
|
|
case GDScriptParser::Node::IDENTIFIER: {
|
|
is_get_node = true;
|
|
} break;
|
|
case GDScriptParser::Node::SUBSCRIPT: {
|
|
GDScriptParser::SubscriptNode *subscript = static_cast<GDScriptParser::SubscriptNode *>(call->callee);
|
|
is_get_node = subscript->is_attribute && subscript->base->type == GDScriptParser::Node::SELF;
|
|
} break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (is_get_node) {
|
|
parser->push_warning(p_variable, GDScriptWarning::GET_NODE_DEFAULT_WITHOUT_ONREADY, is_using_shorthand ? "$" : "get_node()");
|
|
}
|
|
}
|
|
}
|
|
#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);
|
|
}
|
|
|
|
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.
|
|
// Use int, Vector2i, Vector3i instead, which also can be used as range iterators.
|
|
if (p_for->list && p_for->list->type == GDScriptParser::Node::CALL) {
|
|
GDScriptParser::CallNode *call = static_cast<GDScriptParser::CallNode *>(p_for->list);
|
|
GDScriptParser::Node::Type callee_type = call->get_callee_type();
|
|
if (callee_type == GDScriptParser::Node::IDENTIFIER) {
|
|
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;
|
|
} else if (all_is_constant) {
|
|
args.write[i] = call->arguments[i]->reduced_value;
|
|
}
|
|
|
|
GDScriptParser::DataType arg_type = call->arguments[i]->get_datatype();
|
|
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]);
|
|
}
|
|
}
|
|
}
|
|
|
|
Variant reduced;
|
|
|
|
if (all_is_constant) {
|
|
switch (args.size()) {
|
|
case 1:
|
|
reduced = (int32_t)args[0];
|
|
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;
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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;
|
|
} else if (p_for->list) {
|
|
resolve_node(p_for->list, false);
|
|
GDScriptParser::DataType list_type = p_for->list->get_datatype();
|
|
if (!list_type.is_hard_type()) {
|
|
mark_node_unsafe(p_for->list);
|
|
}
|
|
if (list_type.is_variant()) {
|
|
variable_type.kind = GDScriptParser::DataType::VARIANT;
|
|
mark_node_unsafe(p_for->list);
|
|
} else if (list_type.has_container_element_type()) {
|
|
variable_type = list_type.get_container_element_type();
|
|
variable_type.type_source = list_type.type_source;
|
|
} else if (list_type.is_typed_container_type()) {
|
|
variable_type = list_type.get_typed_container_type();
|
|
variable_type.type_source = list_type.type_source;
|
|
} else if (list_type.builtin_type == Variant::INT || list_type.builtin_type == Variant::FLOAT || list_type.builtin_type == Variant::STRING) {
|
|
variable_type.type_source = list_type.type_source;
|
|
variable_type.kind = GDScriptParser::DataType::BUILTIN;
|
|
variable_type.builtin_type = list_type.builtin_type;
|
|
} else if (list_type.builtin_type == Variant::VECTOR2I || list_type.builtin_type == Variant::VECTOR3I) {
|
|
variable_type.type_source = list_type.type_source;
|
|
variable_type.kind = GDScriptParser::DataType::BUILTIN;
|
|
variable_type.builtin_type = Variant::INT;
|
|
} else if (list_type.builtin_type == Variant::VECTOR2 || list_type.builtin_type == Variant::VECTOR3) {
|
|
variable_type.type_source = list_type.type_source;
|
|
variable_type.kind = GDScriptParser::DataType::BUILTIN;
|
|
variable_type.builtin_type = Variant::FLOAT;
|
|
} else if (list_type.builtin_type == Variant::OBJECT) {
|
|
GDScriptParser::DataType return_type;
|
|
List<GDScriptParser::DataType> par_types;
|
|
int default_arg_count = 0;
|
|
bool is_static = false;
|
|
bool is_vararg = false;
|
|
if (get_function_signature(p_for->list, false, list_type, CoreStringNames::get_singleton()->_iter_get, return_type, par_types, default_arg_count, is_static, is_vararg)) {
|
|
variable_type = return_type;
|
|
variable_type.type_source = list_type.type_source;
|
|
} else if (!list_type.is_hard_type()) {
|
|
variable_type.kind = GDScriptParser::DataType::VARIANT;
|
|
} else {
|
|
push_error(vformat(R"(Unable to iterate on object of type "%s".)", list_type.to_string()), p_for->list);
|
|
}
|
|
} else if (list_type.builtin_type == Variant::ARRAY || list_type.builtin_type == Variant::DICTIONARY || !list_type.is_hard_type()) {
|
|
variable_type.kind = GDScriptParser::DataType::VARIANT;
|
|
} else {
|
|
push_error(vformat(R"(Unable to iterate on value of type "%s".)", list_type.to_string()), p_for->list);
|
|
}
|
|
}
|
|
if (p_for->variable) {
|
|
p_for->variable->set_datatype(variable_type);
|
|
}
|
|
|
|
resolve_suite(p_for->loop);
|
|
p_for->set_datatype(p_for->loop->get_datatype());
|
|
#ifdef DEBUG_ENABLED
|
|
if (p_for->variable) {
|
|
is_shadowing(p_for->variable, R"("for" iterator variable)");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void GDScriptAnalyzer::resolve_while(GDScriptParser::WhileNode *p_while) {
|
|
resolve_node(p_while->condition, false);
|
|
|
|
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);
|
|
if (p_assert->message != nullptr) {
|
|
reduce_expression(p_assert->message);
|
|
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);
|
|
}
|
|
}
|
|
|
|
p_assert->set_datatype(p_assert->condition->get_datatype());
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
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);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
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) {
|
|
if (p_match_pattern == nullptr) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType result;
|
|
|
|
switch (p_match_pattern->pattern_type) {
|
|
case GDScriptParser::PatternNode::PT_LITERAL:
|
|
if (p_match_pattern->literal) {
|
|
reduce_literal(p_match_pattern->literal);
|
|
result = p_match_pattern->literal->get_datatype();
|
|
}
|
|
break;
|
|
case GDScriptParser::PatternNode::PT_EXPRESSION:
|
|
if (p_match_pattern->expression) {
|
|
GDScriptParser::ExpressionNode *expr = p_match_pattern->expression;
|
|
reduce_expression(expr);
|
|
result = expr->get_datatype();
|
|
if (!expr->is_constant) {
|
|
while (expr && expr->type == GDScriptParser::Node::SUBSCRIPT) {
|
|
GDScriptParser::SubscriptNode *sub = static_cast<GDScriptParser::SubscriptNode *>(expr);
|
|
if (!sub->is_attribute) {
|
|
expr = nullptr;
|
|
} else {
|
|
expr = sub->base;
|
|
}
|
|
}
|
|
if (!expr || expr->type != GDScriptParser::Node::IDENTIFIER) {
|
|
push_error(R"(Expression in match pattern must be a constant expression, an identifier, or an attribute access ("A.B").)", expr);
|
|
}
|
|
}
|
|
}
|
|
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);
|
|
#ifdef DEBUG_ENABLED
|
|
is_shadowing(p_match_pattern->bind, "pattern bind");
|
|
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);
|
|
}
|
|
#endif
|
|
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++) {
|
|
if (p_match_pattern->dictionary[i].key) {
|
|
reduce_expression(p_match_pattern->dictionary[i].key);
|
|
if (!p_match_pattern->dictionary[i].key->is_constant) {
|
|
push_error(R"(Expression in dictionary pattern key must be a constant.)", p_match_pattern->dictionary[i].key);
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
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;
|
|
|
|
GDScriptParser::DataType expected_type;
|
|
bool has_expected_type = parser->current_function != nullptr;
|
|
if (has_expected_type) {
|
|
expected_type = parser->current_function->get_datatype();
|
|
}
|
|
|
|
if (p_return->return_value != nullptr) {
|
|
bool is_void_function = has_expected_type && expected_type.is_hard_type() && expected_type.kind == GDScriptParser::DataType::BUILTIN && expected_type.builtin_type == Variant::NIL;
|
|
bool is_call = p_return->return_value->type == GDScriptParser::Node::CALL;
|
|
if (is_void_function && is_call) {
|
|
// Pretend the call is a root expression to allow those that are "void".
|
|
reduce_call(static_cast<GDScriptParser::CallNode *>(p_return->return_value), false, true);
|
|
} else {
|
|
reduce_expression(p_return->return_value);
|
|
}
|
|
if (is_void_function) {
|
|
p_return->void_return = true;
|
|
const GDScriptParser::DataType &return_type = p_return->return_value->datatype;
|
|
if (is_call && !return_type.is_hard_type()) {
|
|
String function_name = parser->current_function->identifier ? parser->current_function->identifier->name.operator String() : String("<anonymous function>");
|
|
String called_function_name = static_cast<GDScriptParser::CallNode *>(p_return->return_value)->function_name.operator String();
|
|
#ifdef DEBUG_ENABLED
|
|
parser->push_warning(p_return, GDScriptWarning::UNSAFE_VOID_RETURN, function_name, called_function_name);
|
|
#endif
|
|
mark_node_unsafe(p_return);
|
|
} else if (!is_call) {
|
|
push_error("A void function cannot return a value.", p_return);
|
|
}
|
|
result.type_source = GDScriptParser::DataType::ANNOTATED_EXPLICIT;
|
|
result.kind = GDScriptParser::DataType::BUILTIN;
|
|
result.builtin_type = Variant::NIL;
|
|
result.is_constant = true;
|
|
} else {
|
|
if (p_return->return_value->type == GDScriptParser::Node::ARRAY && has_expected_type && expected_type.has_container_element_type()) {
|
|
update_array_literal_element_type(static_cast<GDScriptParser::ArrayNode *>(p_return->return_value), expected_type.get_container_element_type());
|
|
}
|
|
if (has_expected_type && expected_type.is_hard_type() && p_return->return_value->is_constant) {
|
|
update_const_expression_builtin_type(p_return->return_value, expected_type, "return");
|
|
}
|
|
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;
|
|
}
|
|
|
|
if (has_expected_type && !expected_type.is_variant()) {
|
|
if (result.is_variant() || !result.is_hard_type()) {
|
|
mark_node_unsafe(p_return);
|
|
if (!is_type_compatible(expected_type, result, true, p_return)) {
|
|
downgrade_node_type_source(p_return);
|
|
}
|
|
} else if (!is_type_compatible(expected_type, result, true, p_return)) {
|
|
mark_node_unsafe(p_return);
|
|
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);
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
} else if (expected_type.builtin_type == Variant::INT && result.builtin_type == Variant::FLOAT) {
|
|
parser->push_warning(p_return, GDScriptWarning::NARROWING_CONVERSION);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
p_return->set_datatype(result);
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_expression(GDScriptParser::ExpressionNode *p_expression, bool p_is_root) {
|
|
// This one makes some magic happen.
|
|
|
|
if (p_expression == nullptr) {
|
|
return;
|
|
}
|
|
|
|
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:
|
|
reduce_call(static_cast<GDScriptParser::CallNode *>(p_expression), false, p_is_root);
|
|
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;
|
|
case GDScriptParser::Node::LAMBDA:
|
|
reduce_lambda(static_cast<GDScriptParser::LambdaNode *>(p_expression));
|
|
break;
|
|
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), p_is_root);
|
|
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);
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
static bool enum_has_value(const GDScriptParser::DataType p_type, int64_t p_value) {
|
|
for (const KeyValue<StringName, int64_t> &E : p_type.enum_values) {
|
|
if (E.value == p_value) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
void GDScriptAnalyzer::update_const_expression_builtin_type(GDScriptParser::ExpressionNode *p_expression, const GDScriptParser::DataType &p_type, const char *p_usage, bool p_is_cast) {
|
|
if (p_expression->get_datatype() == p_type) {
|
|
return;
|
|
}
|
|
if (p_type.kind != GDScriptParser::DataType::BUILTIN && p_type.kind != GDScriptParser::DataType::ENUM) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType expression_type = p_expression->get_datatype();
|
|
bool is_enum_cast = p_is_cast && p_type.kind == GDScriptParser::DataType::ENUM && p_type.is_meta_type == false && expression_type.builtin_type == Variant::INT;
|
|
if (!is_enum_cast && !is_type_compatible(p_type, expression_type, true, p_expression)) {
|
|
push_error(vformat(R"(Cannot %s a value of type "%s" as "%s".)", p_usage, expression_type.to_string(), p_type.to_string()), p_expression);
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType value_type = type_from_variant(p_expression->reduced_value, p_expression);
|
|
if (expression_type.is_variant() && !is_enum_cast && !is_type_compatible(p_type, value_type, true, p_expression)) {
|
|
push_error(vformat(R"(Cannot %s a value of type "%s" as "%s".)", p_usage, value_type.to_string(), p_type.to_string()), p_expression);
|
|
return;
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
if (p_type.kind == GDScriptParser::DataType::ENUM && value_type.builtin_type == Variant::INT && !enum_has_value(p_type, p_expression->reduced_value)) {
|
|
parser->push_warning(p_expression, GDScriptWarning::INT_AS_ENUM_WITHOUT_MATCH, p_usage, p_expression->reduced_value.stringify(), p_type.to_string());
|
|
}
|
|
#endif
|
|
|
|
if (value_type.builtin_type == p_type.builtin_type) {
|
|
p_expression->set_datatype(p_type);
|
|
return;
|
|
}
|
|
|
|
Variant converted_to;
|
|
const Variant *converted_from = &p_expression->reduced_value;
|
|
Callable::CallError call_error;
|
|
Variant::construct(p_type.builtin_type, converted_to, &converted_from, 1, call_error);
|
|
if (call_error.error) {
|
|
push_error(vformat(R"(Failed to convert a value of type "%s" to "%s".)", value_type.to_string(), p_type.to_string()), p_expression);
|
|
return;
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
if (p_type.builtin_type == Variant::INT && value_type.builtin_type == Variant::FLOAT) {
|
|
parser->push_warning(p_expression, GDScriptWarning::NARROWING_CONVERSION);
|
|
}
|
|
#endif
|
|
|
|
p_expression->reduced_value = converted_to;
|
|
p_expression->set_datatype(p_type);
|
|
}
|
|
|
|
// 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(GDScriptParser::ArrayNode *p_array, const GDScriptParser::DataType &p_element_type) {
|
|
for (int i = 0; i < p_array->elements.size(); i++) {
|
|
GDScriptParser::ExpressionNode *element_node = p_array->elements[i];
|
|
if (element_node->is_constant) {
|
|
update_const_expression_builtin_type(element_node, p_element_type, "include");
|
|
}
|
|
const GDScriptParser::DataType &element_type = element_node->get_datatype();
|
|
if (element_type.has_no_type() || element_type.is_variant() || !element_type.is_hard_type()) {
|
|
mark_node_unsafe(element_node);
|
|
continue;
|
|
}
|
|
if (!is_type_compatible(p_element_type, element_type, true, p_array)) {
|
|
push_error(vformat(R"(Cannot have an element of type "%s" in an array of type "Array[%s]".)", element_type.to_string(), p_element_type.to_string()), element_node);
|
|
return;
|
|
}
|
|
}
|
|
|
|
GDScriptParser::DataType array_type = p_array->get_datatype();
|
|
array_type.set_container_element_type(p_element_type);
|
|
p_array->set_datatype(array_type);
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_assignment(GDScriptParser::AssignmentNode *p_assignment) {
|
|
reduce_expression(p_assignment->assignee);
|
|
reduce_expression(p_assignment->assigned_value);
|
|
|
|
if (p_assignment->assigned_value == nullptr || p_assignment->assignee == nullptr) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType assignee_type = p_assignment->assignee->get_datatype();
|
|
|
|
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);
|
|
}
|
|
|
|
// Check if assigned value is an array literal, so we can make it a typed array too if appropriate.
|
|
if (p_assignment->assigned_value->type == GDScriptParser::Node::ARRAY && assignee_type.has_container_element_type()) {
|
|
update_array_literal_element_type(static_cast<GDScriptParser::ArrayNode *>(p_assignment->assigned_value), assignee_type.get_container_element_type());
|
|
}
|
|
|
|
if (p_assignment->operation == GDScriptParser::AssignmentNode::OP_NONE && assignee_type.is_hard_type() && p_assignment->assigned_value->is_constant) {
|
|
update_const_expression_builtin_type(p_assignment->assigned_value, assignee_type, "assign");
|
|
}
|
|
|
|
GDScriptParser::DataType assigned_value_type = p_assignment->assigned_value->get_datatype();
|
|
|
|
bool assignee_is_variant = assignee_type.is_variant();
|
|
bool assignee_is_hard = assignee_type.is_hard_type();
|
|
bool assigned_is_variant = assigned_value_type.is_variant();
|
|
bool assigned_is_hard = assigned_value_type.is_hard_type();
|
|
bool compatible = true;
|
|
bool downgrades_assignee = false;
|
|
bool downgrades_assigned = false;
|
|
GDScriptParser::DataType op_type = assigned_value_type;
|
|
if (p_assignment->operation != GDScriptParser::AssignmentNode::OP_NONE && !op_type.is_variant()) {
|
|
op_type = get_operation_type(p_assignment->variant_op, assignee_type, assigned_value_type, compatible, p_assignment->assigned_value);
|
|
|
|
if (assignee_is_variant) {
|
|
// variant assignee
|
|
mark_node_unsafe(p_assignment);
|
|
} else if (!compatible) {
|
|
// incompatible hard types and non-variant assignee
|
|
mark_node_unsafe(p_assignment);
|
|
if (assigned_is_variant) {
|
|
// incompatible hard non-variant assignee and hard variant assigned
|
|
p_assignment->use_conversion_assign = true;
|
|
} else {
|
|
// incompatible hard non-variant types
|
|
push_error(vformat(R"(Invalid operands "%s" and "%s" for assignment operator.)", assignee_type.to_string(), assigned_value_type.to_string()), p_assignment);
|
|
}
|
|
} else if (op_type.type_source == GDScriptParser::DataType::UNDETECTED && !assigned_is_variant) {
|
|
// incompatible non-variant types (at least one weak)
|
|
downgrades_assignee = !assignee_is_hard;
|
|
downgrades_assigned = !assigned_is_hard;
|
|
}
|
|
}
|
|
p_assignment->set_datatype(op_type);
|
|
|
|
if (assignee_is_variant) {
|
|
if (!assignee_is_hard) {
|
|
// weak variant assignee
|
|
mark_node_unsafe(p_assignment);
|
|
}
|
|
} else {
|
|
if (assignee_is_hard && !assigned_is_hard) {
|
|
// hard non-variant assignee and weak assigned
|
|
mark_node_unsafe(p_assignment);
|
|
p_assignment->use_conversion_assign = true;
|
|
downgrades_assigned = downgrades_assigned || (!assigned_is_variant && !is_type_compatible(assignee_type, op_type, true, p_assignment->assigned_value));
|
|
} else if (compatible) {
|
|
if (op_type.is_variant()) {
|
|
// non-variant assignee and variant result
|
|
mark_node_unsafe(p_assignment);
|
|
if (assignee_is_hard) {
|
|
// hard non-variant assignee and variant result
|
|
p_assignment->use_conversion_assign = true;
|
|
} else {
|
|
// weak non-variant assignee and variant result
|
|
downgrades_assignee = true;
|
|
}
|
|
} else if (!is_type_compatible(assignee_type, op_type, assignee_is_hard, p_assignment->assigned_value)) {
|
|
// non-variant assignee and incompatible result
|
|
mark_node_unsafe(p_assignment);
|
|
if (assignee_is_hard) {
|
|
if (is_type_compatible(op_type, assignee_type)) {
|
|
// hard non-variant assignee and maybe compatible result
|
|
p_assignment->use_conversion_assign = true;
|
|
} else {
|
|
// hard non-variant assignee and incompatible result
|
|
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);
|
|
}
|
|
} else {
|
|
// weak non-variant assignee and incompatible result
|
|
downgrades_assignee = true;
|
|
}
|
|
} else if (assignee_type.has_container_element_type() && !op_type.has_container_element_type()) {
|
|
// typed array assignee and untyped array result
|
|
mark_node_unsafe(p_assignment);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (downgrades_assignee) {
|
|
downgrade_node_type_source(p_assignment->assignee);
|
|
}
|
|
if (downgrades_assigned) {
|
|
downgrade_node_type_source(p_assignment->assigned_value);
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
if (assignee_type.is_hard_type() && assignee_type.builtin_type == Variant::INT && assigned_value_type.builtin_type == Variant::FLOAT) {
|
|
parser->push_warning(p_assignment->assigned_value, GDScriptWarning::NARROWING_CONVERSION);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_await(GDScriptParser::AwaitNode *p_await) {
|
|
if (p_await->to_await == nullptr) {
|
|
GDScriptParser::DataType await_type;
|
|
await_type.kind = GDScriptParser::DataType::VARIANT;
|
|
p_await->set_datatype(await_type);
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType awaiting_type;
|
|
|
|
if (p_await->to_await->type == GDScriptParser::Node::CALL) {
|
|
reduce_call(static_cast<GDScriptParser::CallNode *>(p_await->to_await), true);
|
|
awaiting_type = p_await->to_await->get_datatype();
|
|
} else {
|
|
reduce_expression(p_await->to_await);
|
|
}
|
|
|
|
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;
|
|
|
|
awaiting_type = p_await->to_await->get_datatype();
|
|
} else {
|
|
awaiting_type.kind = GDScriptParser::DataType::VARIANT;
|
|
awaiting_type.type_source = GDScriptParser::DataType::UNDETECTED;
|
|
}
|
|
|
|
p_await->set_datatype(awaiting_type);
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
awaiting_type = p_await->to_await->get_datatype();
|
|
if (!(awaiting_type.has_no_type() || awaiting_type.is_coroutine || awaiting_type.builtin_type == Variant::SIGNAL)) {
|
|
parser->push_warning(p_await, GDScriptWarning::REDUNDANT_AWAIT);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_binary_op(GDScriptParser::BinaryOpNode *p_binary_op) {
|
|
reduce_expression(p_binary_op->left_operand);
|
|
|
|
if (p_binary_op->operation == GDScriptParser::BinaryOpNode::OP_TYPE_TEST && p_binary_op->right_operand && p_binary_op->right_operand->type == GDScriptParser::Node::IDENTIFIER) {
|
|
reduce_identifier(static_cast<GDScriptParser::IdentifierNode *>(p_binary_op->right_operand), true);
|
|
} else {
|
|
reduce_expression(p_binary_op->right_operand);
|
|
}
|
|
// TODO: Right operand must be a valid type with the `is` operator. Need to check here.
|
|
|
|
GDScriptParser::DataType left_type;
|
|
if (p_binary_op->left_operand) {
|
|
left_type = p_binary_op->left_operand->get_datatype();
|
|
}
|
|
GDScriptParser::DataType right_type;
|
|
if (p_binary_op->right_operand) {
|
|
right_type = p_binary_op->right_operand->get_datatype();
|
|
}
|
|
|
|
if (!left_type.is_set() || !right_type.is_set()) {
|
|
return;
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
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);
|
|
}
|
|
#endif
|
|
|
|
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) {
|
|
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 {
|
|
push_error(vformat(R"(Invalid operands to operator %s, %s and %s.)",
|
|
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);
|
|
}
|
|
}
|
|
} else {
|
|
if (p_binary_op->operation == GDScriptParser::BinaryOpNode::OP_TYPE_TEST) {
|
|
GDScriptParser::DataType test_type = right_type;
|
|
test_type.is_meta_type = false;
|
|
|
|
if (!is_type_compatible(test_type, left_type)) {
|
|
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.");
|
|
}
|
|
}
|
|
p_binary_op->set_datatype(type_from_variant(p_binary_op->reduced_value, p_binary_op));
|
|
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType result;
|
|
|
|
if (p_binary_op->operation == GDScriptParser::BinaryOpNode::OP_TYPE_TEST) {
|
|
GDScriptParser::DataType test_type = right_type;
|
|
test_type.is_meta_type = false;
|
|
|
|
if (!is_type_compatible(test_type, left_type) && !is_type_compatible(left_type, test_type)) {
|
|
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);
|
|
} else {
|
|
// TODO: Warning.
|
|
mark_node_unsafe(p_binary_op);
|
|
}
|
|
}
|
|
|
|
// "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 if ((p_binary_op->variant_op == Variant::OP_EQUAL || p_binary_op->variant_op == Variant::OP_NOT_EQUAL) &&
|
|
((left_type.kind == GDScriptParser::DataType::BUILTIN && left_type.builtin_type == Variant::NIL) || (right_type.kind == GDScriptParser::DataType::BUILTIN && right_type.builtin_type == Variant::NIL))) {
|
|
// "==" and "!=" operators always return a boolean when comparing to null.
|
|
result.type_source = GDScriptParser::DataType::ANNOTATED_EXPLICIT;
|
|
result.kind = GDScriptParser::DataType::BUILTIN;
|
|
result.builtin_type = Variant::BOOL;
|
|
} else if (left_type.is_variant() || right_type.is_variant()) {
|
|
// 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;
|
|
result = get_operation_type(p_binary_op->variant_op, left_type, right_type, valid, p_binary_op);
|
|
if (!valid) {
|
|
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);
|
|
}
|
|
} else {
|
|
ERR_PRINT("Parser bug: unknown binary operation.");
|
|
}
|
|
|
|
p_binary_op->set_datatype(result);
|
|
}
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
#ifndef DISABLE_DEPRECATED
|
|
const char *GDScriptAnalyzer::get_rename_from_map(const char *map[][2], String key) {
|
|
for (int index = 0; map[index][0]; index++) {
|
|
if (map[index][0] == key) {
|
|
return map[index][1];
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// Checks if an identifier/function name has been renamed in Godot 4, uses ProjectConverter3To4 for rename map.
|
|
// Returns the new name if found, nullptr otherwise.
|
|
const char *GDScriptAnalyzer::check_for_renamed_identifier(String identifier, GDScriptParser::Node::Type type) {
|
|
switch (type) {
|
|
case GDScriptParser::Node::IDENTIFIER: {
|
|
// Check properties
|
|
const char *result = get_rename_from_map(ProjectConverter3To4::gdscript_properties_renames, identifier);
|
|
if (result) {
|
|
return result;
|
|
}
|
|
// Check enum values
|
|
result = get_rename_from_map(ProjectConverter3To4::enum_renames, identifier);
|
|
if (result) {
|
|
return result;
|
|
}
|
|
// Check color constants
|
|
result = get_rename_from_map(ProjectConverter3To4::color_renames, identifier);
|
|
if (result) {
|
|
return result;
|
|
}
|
|
// Check type names
|
|
result = get_rename_from_map(ProjectConverter3To4::class_renames, identifier);
|
|
if (result) {
|
|
return result;
|
|
}
|
|
return get_rename_from_map(ProjectConverter3To4::builtin_types_renames, identifier);
|
|
}
|
|
case GDScriptParser::Node::CALL: {
|
|
const char *result = get_rename_from_map(ProjectConverter3To4::gdscript_function_renames, identifier);
|
|
if (result) {
|
|
return result;
|
|
}
|
|
// Built-in Types are mistaken for function calls when the built-in type is not found.
|
|
// Check built-in types if function rename not found
|
|
return get_rename_from_map(ProjectConverter3To4::builtin_types_renames, identifier);
|
|
}
|
|
// Signal references don't get parsed through the GDScriptAnalyzer. No support for signal rename hints.
|
|
default:
|
|
// No rename found, return null
|
|
return nullptr;
|
|
}
|
|
}
|
|
#endif // DISABLE_DEPRECATED
|
|
#endif // TOOLS_ENABLED
|
|
|
|
void GDScriptAnalyzer::reduce_call(GDScriptParser::CallNode *p_call, bool p_is_await, bool p_is_root) {
|
|
bool all_is_constant = true;
|
|
HashMap<int, GDScriptParser::ArrayNode *> arrays; // For array literal to potentially type when passing.
|
|
for (int i = 0; i < p_call->arguments.size(); i++) {
|
|
reduce_expression(p_call->arguments[i]);
|
|
if (p_call->arguments[i]->type == GDScriptParser::Node::ARRAY) {
|
|
arrays[i] = static_cast<GDScriptParser::ArrayNode *>(p_call->arguments[i]);
|
|
}
|
|
all_is_constant = all_is_constant && p_call->arguments[i]->is_constant;
|
|
}
|
|
|
|
GDScriptParser::Node::Type callee_type = p_call->get_callee_type();
|
|
GDScriptParser::DataType call_type;
|
|
|
|
if (!p_call->is_super && callee_type == GDScriptParser::Node::IDENTIFIER) {
|
|
// 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".
|
|
}
|
|
|
|
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:
|
|
case Variant::DICTIONARY:
|
|
case Variant::ARRAY:
|
|
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) {
|
|
// 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;
|
|
Variant value;
|
|
Variant::construct(builtin_type, value, (const Variant **)args.ptr(), args.size(), err);
|
|
|
|
switch (err.error) {
|
|
case Callable::CallError::CALL_ERROR_INVALID_ARGUMENT:
|
|
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,
|
|
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();
|
|
}
|
|
signature += ")";
|
|
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:
|
|
case Callable::CallError::CALL_ERROR_METHOD_NOT_CONST:
|
|
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;
|
|
|
|
for (const MethodInfo &info : constructors) {
|
|
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], true);
|
|
|
|
if (!is_type_compatible(par_type, p_call->arguments[i]->get_datatype(), true)) {
|
|
types_match = false;
|
|
break;
|
|
#ifdef DEBUG_ENABLED
|
|
} else {
|
|
if (par_type.builtin_type == Variant::INT && p_call->arguments[i]->get_datatype().builtin_type == Variant::FLOAT && builtin_type != Variant::INT) {
|
|
parser->push_warning(p_call, GDScriptWarning::NARROWING_CONVERSION, p_call->function_name);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
if (types_match) {
|
|
for (int i = 0; i < p_call->arguments.size(); i++) {
|
|
if (p_call->arguments[i]->is_constant) {
|
|
update_const_expression_builtin_type(p_call->arguments[i], type_from_property(info.arguments[i], true), "pass");
|
|
}
|
|
}
|
|
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();
|
|
}
|
|
signature += ")";
|
|
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;
|
|
} else if (GDScriptUtilityFunctions::function_exists(function_name)) {
|
|
MethodInfo function_info = GDScriptUtilityFunctions::get_function_info(function_name);
|
|
|
|
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)) {
|
|
push_error(vformat(R"*(Cannot get return value of call to "%s()" because it returns "void".)*", function_name), p_call);
|
|
}
|
|
|
|
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];
|
|
push_error(vformat(R"*(Invalid argument for "%s()" function: argument %d should be "%s" but is "%s".)*", function_name, err.argument + 1,
|
|
type_from_property(wrong_arg, true).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;
|
|
case Callable::CallError::CALL_ERROR_METHOD_NOT_CONST:
|
|
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);
|
|
|
|
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)) {
|
|
push_error(vformat(R"*(Cannot get return value of call to "%s()" because it returns "void".)*", function_name), p_call);
|
|
}
|
|
|
|
if (all_is_constant && Variant::get_utility_function_type(function_name) == Variant::UTILITY_FUNC_TYPE_MATH) {
|
|
// 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;
|
|
Variant::call_utility_function(function_name, &value, (const Variant **)args.ptr(), args.size(), err);
|
|
|
|
switch (err.error) {
|
|
case Callable::CallError::CALL_ERROR_INVALID_ARGUMENT: {
|
|
String expected_type_name;
|
|
if (err.argument < function_info.arguments.size()) {
|
|
expected_type_name = type_from_property(function_info.arguments[err.argument], true).to_string();
|
|
} else {
|
|
expected_type_name = Variant::get_type_name((Variant::Type)err.expected);
|
|
}
|
|
|
|
push_error(vformat(R"*(Invalid argument for "%s()" function: argument %d should be "%s" but is "%s".)*", function_name, err.argument + 1,
|
|
expected_type_name, 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;
|
|
case Callable::CallError::CALL_ERROR_METHOD_NOT_CONST:
|
|
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;
|
|
}
|
|
}
|
|
|
|
GDScriptParser::DataType base_type;
|
|
call_type.kind = GDScriptParser::DataType::VARIANT;
|
|
bool is_self = false;
|
|
|
|
if (p_call->is_super) {
|
|
base_type = parser->current_class->base_type;
|
|
base_type.is_meta_type = false;
|
|
is_self = true;
|
|
|
|
if (p_call->callee == nullptr && !lambda_stack.is_empty()) {
|
|
push_error("Cannot use `super()` inside a lambda.", p_call);
|
|
}
|
|
} else if (callee_type == GDScriptParser::Node::IDENTIFIER) {
|
|
base_type = parser->current_class->get_datatype();
|
|
base_type.is_meta_type = false;
|
|
is_self = true;
|
|
} else if (callee_type == GDScriptParser::Node::SUBSCRIPT) {
|
|
GDScriptParser::SubscriptNode *subscript = static_cast<GDScriptParser::SubscriptNode *>(p_call->callee);
|
|
if (subscript->base == nullptr) {
|
|
// Invalid syntax, error already set on parser.
|
|
p_call->set_datatype(call_type);
|
|
mark_node_unsafe(p_call);
|
|
return;
|
|
}
|
|
if (!subscript->is_attribute) {
|
|
// Invalid call. Error already sent in parser.
|
|
// TODO: Could check if Callable here.
|
|
p_call->set_datatype(call_type);
|
|
mark_node_unsafe(p_call);
|
|
return;
|
|
}
|
|
if (subscript->attribute == nullptr) {
|
|
// Invalid call. Error already sent in parser.
|
|
p_call->set_datatype(call_type);
|
|
mark_node_unsafe(p_call);
|
|
return;
|
|
}
|
|
|
|
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();
|
|
is_self = subscript->base->type == GDScriptParser::Node::SELF;
|
|
}
|
|
} else {
|
|
// Invalid call. Error already sent in parser.
|
|
// TODO: Could check if Callable here too.
|
|
p_call->set_datatype(call_type);
|
|
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;
|
|
|
|
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)) {
|
|
// If the function require typed arrays we must make literals be typed.
|
|
for (const KeyValue<int, GDScriptParser::ArrayNode *> &E : arrays) {
|
|
int index = E.key;
|
|
if (index < par_types.size() && par_types[index].has_container_element_type()) {
|
|
update_array_literal_element_type(E.value, par_types[index].get_container_element_type());
|
|
}
|
|
}
|
|
validate_call_arg(par_types, default_arg_count, is_vararg, p_call);
|
|
|
|
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;
|
|
}
|
|
|
|
if (is_self && parser->current_function != nullptr && parser->current_function->is_static && !is_static) {
|
|
// 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);
|
|
} else if (!is_self && base_type.is_meta_type && !is_static) {
|
|
base_type.is_meta_type = false; // For `to_string()`.
|
|
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);
|
|
} else if (is_self && !is_static) {
|
|
mark_lambda_use_self();
|
|
}
|
|
|
|
if (!p_is_root && !p_is_await && return_type.is_hard_type() && return_type.kind == GDScriptParser::DataType::BUILTIN && return_type.builtin_type == Variant::NIL) {
|
|
push_error(vformat(R"*(Cannot get return value of call to "%s()" because it returns "void".)*", p_call->function_name), p_call);
|
|
}
|
|
|
|
#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);
|
|
}
|
|
|
|
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);
|
|
}
|
|
#endif // DEBUG_ENABLED
|
|
|
|
call_type = return_type;
|
|
} else {
|
|
bool found = false;
|
|
|
|
// Enums do not have functions other than the built-in dictionary ones.
|
|
if (base_type.kind == GDScriptParser::DataType::ENUM && base_type.is_meta_type) {
|
|
if (base_type.builtin_type == Variant::DICTIONARY) {
|
|
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 {
|
|
push_error(vformat(R"*(The native enum "%s" does not behave like Dictionary and does not have methods of its own.)*", 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.
|
|
GDScriptParser::IdentifierNode *callee_id;
|
|
if (callee_type == GDScriptParser::Node::IDENTIFIER) {
|
|
callee_id = static_cast<GDScriptParser::IdentifierNode *>(p_call->callee);
|
|
} else {
|
|
// Can only be attribute.
|
|
callee_id = static_cast<GDScriptParser::SubscriptNode *>(p_call->callee)->attribute;
|
|
}
|
|
if (callee_id) {
|
|
reduce_identifier_from_base(callee_id, &base_type);
|
|
GDScriptParser::DataType callee_datatype = callee_id->get_datatype();
|
|
if (callee_datatype.is_set() && !callee_datatype.is_variant()) {
|
|
found = true;
|
|
if (callee_datatype.builtin_type == Variant::CALLABLE) {
|
|
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 {
|
|
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);
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
} else if (!is_self && !(base_type.is_hard_type() && base_type.kind == GDScriptParser::DataType::BUILTIN)) {
|
|
parser->push_warning(p_call, GDScriptWarning::UNSAFE_METHOD_ACCESS, p_call->function_name, base_type.to_string());
|
|
mark_node_unsafe(p_call);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
if (!found && (is_self || (base_type.is_hard_type() && base_type.kind == GDScriptParser::DataType::BUILTIN))) {
|
|
String base_name = is_self && !p_call->is_super ? "self" : base_type.to_string();
|
|
#ifdef TOOLS_ENABLED
|
|
#ifndef DISABLE_DEPRECATED
|
|
String rename_hint = String();
|
|
if (GLOBAL_GET(GDScriptWarning::get_settings_path_from_code(GDScriptWarning::Code::RENAMED_IN_GD4_HINT)).booleanize()) {
|
|
const char *renamed_function_name = check_for_renamed_identifier(p_call->function_name, p_call->type);
|
|
if (renamed_function_name) {
|
|
rename_hint = " " + vformat(R"(Did you mean to use "%s"?)", String(renamed_function_name) + "()");
|
|
}
|
|
}
|
|
push_error(vformat(R"*(Function "%s()" not found in base %s.%s)*", p_call->function_name, base_name, rename_hint), p_call->is_super ? p_call : p_call->callee);
|
|
#else // !DISABLE_DEPRECATED
|
|
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);
|
|
#endif // DISABLE_DEPRECATED
|
|
#else
|
|
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);
|
|
#endif
|
|
} else if (!found && (!p_call->is_super && base_type.is_hard_type() && base_type.kind == GDScriptParser::DataType::NATIVE && base_type.is_meta_type)) {
|
|
push_error(vformat(R"*(Static function "%s()" not found in base "%s".)*", p_call->function_name, base_type.native_type), p_call);
|
|
}
|
|
}
|
|
|
|
if (call_type.is_coroutine && !p_is_await && !p_is_root) {
|
|
push_error(vformat(R"*(Function "%s()" is a coroutine, so it must be called with "await".)*", p_call->function_name), p_call);
|
|
}
|
|
|
|
p_call->set_datatype(call_type);
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_cast(GDScriptParser::CastNode *p_cast) {
|
|
reduce_expression(p_cast->operand);
|
|
|
|
GDScriptParser::DataType cast_type = type_from_metatype(resolve_datatype(p_cast->cast_type));
|
|
|
|
if (!cast_type.is_set()) {
|
|
mark_node_unsafe(p_cast);
|
|
return;
|
|
}
|
|
|
|
p_cast->set_datatype(cast_type);
|
|
if (p_cast->operand->is_constant) {
|
|
update_const_expression_builtin_type(p_cast->operand, cast_type, "cast", true);
|
|
if (cast_type.is_variant() || p_cast->operand->get_datatype() == cast_type) {
|
|
p_cast->is_constant = true;
|
|
p_cast->reduced_value = p_cast->operand->reduced_value;
|
|
}
|
|
}
|
|
|
|
if (p_cast->operand->type == GDScriptParser::Node::ARRAY && cast_type.has_container_element_type()) {
|
|
update_array_literal_element_type(static_cast<GDScriptParser::ArrayNode *>(p_cast->operand), cast_type.get_container_element_type());
|
|
}
|
|
|
|
if (!cast_type.is_variant()) {
|
|
GDScriptParser::DataType op_type = p_cast->operand->get_datatype();
|
|
if (op_type.is_variant() || !op_type.is_hard_type()) {
|
|
mark_node_unsafe(p_cast);
|
|
#ifdef DEBUG_ENABLED
|
|
if (op_type.is_variant() && !op_type.is_hard_type()) {
|
|
parser->push_warning(p_cast, GDScriptWarning::UNSAFE_CAST, cast_type.to_string());
|
|
}
|
|
#endif
|
|
} else {
|
|
bool valid = false;
|
|
if (op_type.builtin_type == Variant::INT && cast_type.kind == GDScriptParser::DataType::ENUM) {
|
|
mark_node_unsafe(p_cast);
|
|
valid = true;
|
|
} else if (op_type.kind == GDScriptParser::DataType::BUILTIN && cast_type.kind == GDScriptParser::DataType::BUILTIN) {
|
|
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);
|
|
}
|
|
|
|
if (!valid) {
|
|
push_error(vformat(R"(Invalid cast. Cannot convert from "%s" to "%s".)", op_type.to_string(), cast_type.to_string()), p_cast->cast_type);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_dictionary(GDScriptParser::DictionaryNode *p_dictionary) {
|
|
HashMap<Variant, GDScriptParser::ExpressionNode *, VariantHasher, StringLikeVariantComparator> elements;
|
|
|
|
for (int i = 0; i < p_dictionary->elements.size(); i++) {
|
|
const GDScriptParser::DictionaryNode::Pair &element = p_dictionary->elements[i];
|
|
if (p_dictionary->style == GDScriptParser::DictionaryNode::PYTHON_DICT) {
|
|
reduce_expression(element.key);
|
|
}
|
|
reduce_expression(element.value);
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
// 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;
|
|
result.native_type = SNAME("Node");
|
|
result.builtin_type = Variant::OBJECT;
|
|
|
|
if (!ClassDB::is_parent_class(parser->current_class->base_type.native_type, result.native_type)) {
|
|
push_error(R"*(Cannot use shorthand "get_node()" notation ("$") on a class that isn't a node.)*", p_get_node);
|
|
}
|
|
|
|
mark_lambda_use_self();
|
|
|
|
p_get_node->set_datatype(result);
|
|
}
|
|
|
|
GDScriptParser::DataType GDScriptAnalyzer::make_global_class_meta_type(const StringName &p_class_name, const GDScriptParser::Node *p_source) {
|
|
GDScriptParser::DataType type;
|
|
|
|
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;
|
|
}
|
|
|
|
Error err = ref->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
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;
|
|
}
|
|
|
|
return ref->get_parser()->head->get_datatype();
|
|
} else {
|
|
return make_script_meta_type(ResourceLoader::load(path, "Script"));
|
|
}
|
|
}
|
|
|
|
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;
|
|
Ref<GDScript> scr = GDScriptCache::get_shallow_script(p_identifier_datatype.script_path, err);
|
|
if (err) {
|
|
push_error(vformat(R"(Error while getting cache for script "%s".)", p_identifier_datatype.script_path), p_identifier);
|
|
return;
|
|
}
|
|
p_identifier->reduced_value = scr->find_class(p_identifier_datatype.class_type->fqcn);
|
|
p_identifier->is_constant = true;
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_identifier_from_base(GDScriptParser::IdentifierNode *p_identifier, GDScriptParser::DataType *p_base) {
|
|
if (!p_identifier->get_datatype().has_no_type()) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType base;
|
|
if (p_base == nullptr) {
|
|
base = type_from_metatype(parser->current_class->get_datatype());
|
|
} else {
|
|
base = *p_base;
|
|
}
|
|
|
|
const StringName &name = p_identifier->name;
|
|
|
|
if (base.kind == GDScriptParser::DataType::ENUM) {
|
|
if (base.is_meta_type) {
|
|
if (base.enum_values.has(name)) {
|
|
p_identifier->set_datatype(type_from_metatype(base));
|
|
p_identifier->is_constant = true;
|
|
p_identifier->reduced_value = base.enum_values[name];
|
|
return;
|
|
}
|
|
|
|
// Enum does not have this value, return.
|
|
return;
|
|
} else {
|
|
push_error(R"(Cannot get property from enum value.)", p_identifier);
|
|
return;
|
|
}
|
|
}
|
|
|
|
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;
|
|
p_identifier->set_datatype(type_from_variant(result, p_identifier));
|
|
} else if (base.is_hard_type()) {
|
|
#ifdef TOOLS_ENABLED
|
|
#ifndef DISABLE_DEPRECATED
|
|
String rename_hint = String();
|
|
if (GLOBAL_GET(GDScriptWarning::get_settings_path_from_code(GDScriptWarning::Code::RENAMED_IN_GD4_HINT)).booleanize()) {
|
|
const char *renamed_identifier_name = check_for_renamed_identifier(name, p_identifier->type);
|
|
if (renamed_identifier_name) {
|
|
rename_hint = " " + vformat(R"(Did you mean to use "%s"?)", renamed_identifier_name);
|
|
}
|
|
}
|
|
push_error(vformat(R"(Cannot find constant "%s" on base "%s".%s)", name, base.to_string(), rename_hint), p_identifier);
|
|
#else // !DISABLE_DEPRECATED
|
|
push_error(vformat(R"(Cannot find constant "%s" on base "%s".)", name, base.to_string()), p_identifier);
|
|
#endif // DISABLE_DEPRECATED
|
|
#else
|
|
push_error(vformat(R"(Cannot find constant "%s" on base "%s".)", name, base.to_string()), p_identifier);
|
|
#endif
|
|
}
|
|
} else {
|
|
switch (base.builtin_type) {
|
|
case Variant::NIL: {
|
|
if (base.is_hard_type()) {
|
|
push_error(vformat(R"(Invalid get index "%s" on base Nil)", name), p_identifier);
|
|
}
|
|
return;
|
|
}
|
|
case Variant::DICTIONARY: {
|
|
GDScriptParser::DataType dummy;
|
|
dummy.kind = GDScriptParser::DataType::VARIANT;
|
|
p_identifier->set_datatype(dummy);
|
|
return;
|
|
}
|
|
default: {
|
|
Callable::CallError temp;
|
|
Variant dummy;
|
|
Variant::construct(base.builtin_type, dummy, nullptr, 0, temp);
|
|
List<PropertyInfo> properties;
|
|
dummy.get_property_list(&properties);
|
|
for (const PropertyInfo &prop : properties) {
|
|
if (prop.name == name) {
|
|
p_identifier->set_datatype(type_from_property(prop));
|
|
return;
|
|
}
|
|
}
|
|
if (base.is_hard_type()) {
|
|
#ifdef TOOLS_ENABLED
|
|
#ifndef DISABLE_DEPRECATED
|
|
String rename_hint = String();
|
|
if (GLOBAL_GET(GDScriptWarning::get_settings_path_from_code(GDScriptWarning::Code::RENAMED_IN_GD4_HINT)).booleanize()) {
|
|
const char *renamed_identifier_name = check_for_renamed_identifier(name, p_identifier->type);
|
|
if (renamed_identifier_name) {
|
|
rename_hint = " " + vformat(R"(Did you mean to use "%s"?)", renamed_identifier_name);
|
|
}
|
|
}
|
|
push_error(vformat(R"(Cannot find property "%s" on base "%s".%s)", name, base.to_string(), rename_hint), p_identifier);
|
|
#else // !DISABLE_DEPRECATED
|
|
push_error(vformat(R"(Cannot find property "%s" on base "%s".)", name, base.to_string()), p_identifier);
|
|
#endif // DISABLE_DEPRECATED
|
|
#else
|
|
push_error(vformat(R"(Cannot find property "%s" on base "%s".)", name, base.to_string()), p_identifier);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::ClassNode *base_class = base.class_type;
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
bool is_base = true;
|
|
|
|
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());
|
|
return;
|
|
}
|
|
|
|
if (script_class->has_member(name)) {
|
|
resolve_class_member(script_class, name, p_identifier);
|
|
|
|
GDScriptParser::ClassNode::Member member = script_class->get_member(name);
|
|
switch (member.type) {
|
|
case GDScriptParser::ClassNode::Member::CONSTANT: {
|
|
p_identifier->set_datatype(member.get_datatype());
|
|
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;
|
|
return;
|
|
}
|
|
|
|
case GDScriptParser::ClassNode::Member::ENUM_VALUE: {
|
|
p_identifier->set_datatype(member.get_datatype());
|
|
p_identifier->is_constant = true;
|
|
p_identifier->reduced_value = member.enum_value.value;
|
|
p_identifier->source = GDScriptParser::IdentifierNode::MEMBER_CONSTANT;
|
|
return;
|
|
}
|
|
|
|
case GDScriptParser::ClassNode::Member::ENUM: {
|
|
p_identifier->set_datatype(member.get_datatype());
|
|
p_identifier->is_constant = true;
|
|
p_identifier->reduced_value = member.m_enum->dictionary;
|
|
p_identifier->source = GDScriptParser::IdentifierNode::MEMBER_CONSTANT;
|
|
return;
|
|
}
|
|
|
|
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;
|
|
|
|
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;
|
|
}
|
|
} 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
|
|
}
|
|
}
|
|
}
|
|
|
|
if (is_base) {
|
|
is_base = script_class->base_type.class_type != nullptr;
|
|
if (!is_base && p_base != nullptr) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check native members. No need for native class recursion because Node exposes all Object's properties.
|
|
const StringName &native = base.native_type;
|
|
|
|
if (class_exists(native)) {
|
|
MethodInfo method_info;
|
|
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()));
|
|
p_identifier->source = GDScriptParser::IdentifierNode::INHERITED_VARIABLE;
|
|
}
|
|
return;
|
|
}
|
|
if (ClassDB::get_method_info(native, name, &method_info)) {
|
|
// Method is callable.
|
|
p_identifier->set_datatype(make_callable_type(method_info));
|
|
p_identifier->source = GDScriptParser::IdentifierNode::INHERITED_VARIABLE;
|
|
return;
|
|
}
|
|
if (ClassDB::get_signal(native, name, &method_info)) {
|
|
// Signal is a type too.
|
|
p_identifier->set_datatype(make_signal_type(method_info));
|
|
p_identifier->source = GDScriptParser::IdentifierNode::INHERITED_VARIABLE;
|
|
return;
|
|
}
|
|
if (ClassDB::has_enum(native, name)) {
|
|
p_identifier->set_datatype(make_native_enum_type(name, native));
|
|
p_identifier->source = GDScriptParser::IdentifierNode::MEMBER_CONSTANT;
|
|
return;
|
|
}
|
|
bool valid = false;
|
|
|
|
int64_t int_constant = ClassDB::get_integer_constant(native, name, &valid);
|
|
if (valid) {
|
|
p_identifier->is_constant = true;
|
|
p_identifier->reduced_value = int_constant;
|
|
p_identifier->source = GDScriptParser::IdentifierNode::MEMBER_CONSTANT;
|
|
|
|
// 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));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_identifier(GDScriptParser::IdentifierNode *p_identifier, bool can_be_builtin) {
|
|
// TODO: This is an opportunity to further infer types.
|
|
|
|
// Check if we are inside an enum. This allows enum values to access other elements of the same enum.
|
|
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) {
|
|
StringName enum_name = current_enum->identifier ? current_enum->identifier->name : UNNAMED_ENUM;
|
|
GDScriptParser::DataType type = make_enum_type(enum_name, parser->current_class->fqcn, false);
|
|
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.
|
|
}
|
|
}
|
|
}
|
|
|
|
bool found_source = false;
|
|
// 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());
|
|
found_source = true;
|
|
break;
|
|
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;
|
|
found_source = true;
|
|
break;
|
|
case GDScriptParser::IdentifierNode::MEMBER_SIGNAL:
|
|
case GDScriptParser::IdentifierNode::INHERITED_VARIABLE:
|
|
mark_lambda_use_self();
|
|
break;
|
|
case GDScriptParser::IdentifierNode::MEMBER_VARIABLE:
|
|
mark_lambda_use_self();
|
|
p_identifier->variable_source->usages++;
|
|
[[fallthrough]];
|
|
case GDScriptParser::IdentifierNode::LOCAL_VARIABLE:
|
|
p_identifier->set_datatype(p_identifier->variable_source->get_datatype());
|
|
found_source = true;
|
|
break;
|
|
case GDScriptParser::IdentifierNode::LOCAL_ITERATOR:
|
|
p_identifier->set_datatype(p_identifier->bind_source->get_datatype());
|
|
found_source = true;
|
|
break;
|
|
case GDScriptParser::IdentifierNode::LOCAL_BIND: {
|
|
GDScriptParser::DataType result = p_identifier->bind_source->get_datatype();
|
|
result.is_constant = true;
|
|
p_identifier->set_datatype(result);
|
|
found_source = true;
|
|
} break;
|
|
case GDScriptParser::IdentifierNode::UNDEFINED_SOURCE:
|
|
break;
|
|
}
|
|
|
|
// Not a local, so check members.
|
|
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) {
|
|
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) {
|
|
// 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 access %s "%s" from the static function "%s()".)*", source_is_signal ? "signal" : "instance variable", p_identifier->name, parent_function->identifier->name), p_identifier);
|
|
}
|
|
|
|
if (!lambda_stack.is_empty()) {
|
|
// 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) {
|
|
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;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
StringName name = p_identifier->name;
|
|
p_identifier->source = GDScriptParser::IdentifierNode::UNDEFINED_SOURCE;
|
|
|
|
// 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) {
|
|
if (can_be_builtin) {
|
|
p_identifier->set_datatype(make_builtin_meta_type(builtin_type));
|
|
return;
|
|
} else {
|
|
push_error(R"(Builtin type cannot be used as a name on its own.)", p_identifier);
|
|
}
|
|
}
|
|
|
|
if (class_exists(name)) {
|
|
p_identifier->set_datatype(make_native_meta_type(name));
|
|
return;
|
|
}
|
|
|
|
if (ScriptServer::is_global_class(name)) {
|
|
p_identifier->set_datatype(make_global_class_meta_type(name, p_identifier));
|
|
return;
|
|
}
|
|
|
|
// 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;
|
|
if (ResourceLoader::get_resource_type(autoload.path) == "GDScript") {
|
|
Ref<GDScriptParserRef> singl_parser = get_parser_for(autoload.path);
|
|
if (singl_parser.is_valid()) {
|
|
Error err = singl_parser->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
if (err == OK) {
|
|
result = type_from_metatype(singl_parser->get_parser()->head->get_datatype());
|
|
}
|
|
}
|
|
} else if (ResourceLoader::get_resource_type(autoload.path) == "PackedScene") {
|
|
if (GDScriptLanguage::get_singleton()->has_any_global_constant(name)) {
|
|
Variant constant = GDScriptLanguage::get_singleton()->get_any_global_constant(name);
|
|
Node *node = Object::cast_to<Node>(constant);
|
|
if (node != nullptr) {
|
|
Ref<GDScript> scr = node->get_script();
|
|
if (scr.is_valid()) {
|
|
Ref<GDScriptParserRef> singl_parser = get_parser_for(scr->get_script_path());
|
|
if (singl_parser.is_valid()) {
|
|
Error err = singl_parser->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
if (err == OK) {
|
|
result = type_from_metatype(singl_parser->get_parser()->head->get_datatype());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
result.is_constant = true;
|
|
p_identifier->set_datatype(result);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (GDScriptLanguage::get_singleton()->has_any_global_constant(name)) {
|
|
Variant constant = GDScriptLanguage::get_singleton()->get_any_global_constant(name);
|
|
p_identifier->set_datatype(type_from_variant(constant, p_identifier));
|
|
p_identifier->is_constant = true;
|
|
p_identifier->reduced_value = constant;
|
|
return;
|
|
}
|
|
|
|
// Not found.
|
|
// Check if it's a builtin function.
|
|
if (GDScriptUtilityFunctions::function_exists(name)) {
|
|
push_error(vformat(R"(Built-in function "%s" cannot be used as an identifier.)", name), p_identifier);
|
|
} else {
|
|
#ifdef TOOLS_ENABLED
|
|
#ifndef DISABLE_DEPRECATED
|
|
String rename_hint = String();
|
|
if (GLOBAL_GET(GDScriptWarning::get_settings_path_from_code(GDScriptWarning::Code::RENAMED_IN_GD4_HINT)).booleanize()) {
|
|
const char *renamed_identifier_name = check_for_renamed_identifier(name, p_identifier->type);
|
|
if (renamed_identifier_name) {
|
|
rename_hint = " " + vformat(R"(Did you mean to use "%s"?)", renamed_identifier_name);
|
|
}
|
|
}
|
|
push_error(vformat(R"(Identifier "%s" not declared in the current scope.%s)", name, rename_hint), p_identifier);
|
|
#else // !DISABLE_DEPRECATED
|
|
push_error(vformat(R"(Identifier "%s" not declared in the current scope.)", name), p_identifier);
|
|
#endif // DISABLE_DEPRECATED
|
|
#else
|
|
push_error(vformat(R"(Identifier "%s" not declared in the current scope.)", name), p_identifier);
|
|
#endif
|
|
}
|
|
GDScriptParser::DataType dummy;
|
|
dummy.kind = GDScriptParser::DataType::VARIANT;
|
|
p_identifier->set_datatype(dummy); // Just so type is set to something.
|
|
}
|
|
|
|
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);
|
|
resolve_function_signature(p_lambda->function, p_lambda, true);
|
|
resolve_function_body(p_lambda->function, true);
|
|
lambda_stack.pop_back();
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_literal(GDScriptParser::LiteralNode *p_literal) {
|
|
p_literal->reduced_value = p_literal->value;
|
|
p_literal->is_constant = true;
|
|
|
|
p_literal->set_datatype(type_from_variant(p_literal->reduced_value, p_literal));
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_preload(GDScriptParser::PreloadNode *p_preload) {
|
|
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.
|
|
if (p_preload->resolved_path.is_relative_path()) {
|
|
p_preload->resolved_path = parser->script_path.get_base_dir().path_join(p_preload->resolved_path);
|
|
}
|
|
p_preload->resolved_path = p_preload->resolved_path.simplify_path();
|
|
if (!ResourceLoader::exists(p_preload->resolved_path)) {
|
|
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);
|
|
}
|
|
} else {
|
|
// TODO: Don't load if validating: use completion cache.
|
|
|
|
// 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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
p_preload->is_constant = true;
|
|
p_preload->reduced_value = p_preload->resource;
|
|
p_preload->set_datatype(type_from_variant(p_preload->reduced_value, p_preload));
|
|
}
|
|
|
|
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()));
|
|
mark_lambda_use_self();
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_subscript(GDScriptParser::SubscriptNode *p_subscript) {
|
|
if (p_subscript->base == nullptr) {
|
|
return;
|
|
}
|
|
if (p_subscript->base->type == GDScriptParser::Node::IDENTIFIER) {
|
|
reduce_identifier(static_cast<GDScriptParser::IdentifierNode *>(p_subscript->base), true);
|
|
} else {
|
|
reduce_expression(p_subscript->base);
|
|
}
|
|
|
|
GDScriptParser::DataType result_type;
|
|
|
|
if (p_subscript->is_attribute) {
|
|
if (p_subscript->attribute == nullptr) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType base_type = p_subscript->base->get_datatype();
|
|
bool valid = false;
|
|
// If the base is a metatype, use the analyzer instead.
|
|
if (p_subscript->base->is_constant && !base_type.is_meta_type) {
|
|
// Just try to get it.
|
|
Variant value = p_subscript->base->reduced_value.get_named(p_subscript->attribute->name, valid);
|
|
if (valid) {
|
|
p_subscript->is_constant = true;
|
|
p_subscript->reduced_value = value;
|
|
result_type = type_from_variant(value, p_subscript);
|
|
}
|
|
} else if (base_type.is_variant() || !base_type.is_hard_type()) {
|
|
valid = true;
|
|
result_type.kind = GDScriptParser::DataType::VARIANT;
|
|
mark_node_unsafe(p_subscript);
|
|
} else {
|
|
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;
|
|
} else if (!base_type.is_meta_type || !base_type.is_constant) {
|
|
valid = base_type.kind != GDScriptParser::DataType::BUILTIN;
|
|
#ifdef DEBUG_ENABLED
|
|
if (valid) {
|
|
parser->push_warning(p_subscript, GDScriptWarning::UNSAFE_PROPERTY_ACCESS, p_subscript->attribute->name, base_type.to_string());
|
|
}
|
|
#endif
|
|
result_type.kind = GDScriptParser::DataType::VARIANT;
|
|
mark_node_unsafe(p_subscript);
|
|
}
|
|
}
|
|
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;
|
|
}
|
|
} else {
|
|
if (p_subscript->index == nullptr) {
|
|
return;
|
|
}
|
|
reduce_expression(p_subscript->index);
|
|
|
|
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);
|
|
result_type.kind = GDScriptParser::DataType::VARIANT;
|
|
} else {
|
|
p_subscript->is_constant = true;
|
|
p_subscript->reduced_value = value;
|
|
result_type = type_from_variant(value, p_subscript);
|
|
}
|
|
} 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:
|
|
case Variant::QUATERNION:
|
|
case Variant::AABB:
|
|
case Variant::OBJECT:
|
|
error = index_type.builtin_type != Variant::STRING && index_type.builtin_type != Variant::STRING_NAME;
|
|
break;
|
|
// Expect String or number.
|
|
case Variant::BASIS:
|
|
case Variant::VECTOR2:
|
|
case Variant::VECTOR2I:
|
|
case Variant::VECTOR3:
|
|
case Variant::VECTOR3I:
|
|
case Variant::VECTOR4:
|
|
case Variant::VECTOR4I:
|
|
case Variant::TRANSFORM2D:
|
|
case Variant::TRANSFORM3D:
|
|
case Variant::PROJECTION:
|
|
error = index_type.builtin_type != Variant::INT && index_type.builtin_type != Variant::FLOAT &&
|
|
index_type.builtin_type != Variant::STRING && index_type.builtin_type != Variant::STRING_NAME;
|
|
break;
|
|
// Expect String or int.
|
|
case Variant::COLOR:
|
|
error = index_type.builtin_type != Variant::INT && index_type.builtin_type != Variant::STRING && index_type.builtin_type != Variant::STRING_NAME;
|
|
break;
|
|
// Don't support indexing, but we will check it later.
|
|
case Variant::RID:
|
|
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;
|
|
result_type.type_source = base_type.is_hard_type() ? GDScriptParser::DataType::ANNOTATED_INFERRED : GDScriptParser::DataType::INFERRED;
|
|
|
|
if (base_type.kind != GDScriptParser::DataType::BUILTIN) {
|
|
base_type.builtin_type = Variant::OBJECT;
|
|
}
|
|
switch (base_type.builtin_type) {
|
|
// Can't index at all.
|
|
case Variant::RID:
|
|
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:
|
|
case Variant::VECTOR4I:
|
|
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:
|
|
case Variant::VECTOR4:
|
|
case Variant::QUATERNION:
|
|
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.
|
|
case Variant::TRANSFORM3D:
|
|
case Variant::PROJECTION:
|
|
case Variant::PLANE:
|
|
case Variant::COLOR:
|
|
case Variant::DICTIONARY:
|
|
case Variant::OBJECT:
|
|
result_type.kind = GDScriptParser::DataType::VARIANT;
|
|
result_type.type_source = GDScriptParser::DataType::UNDETECTED;
|
|
break;
|
|
// 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;
|
|
// Here for completeness.
|
|
case Variant::VARIANT_MAX:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
p_subscript->set_datatype(result_type);
|
|
}
|
|
|
|
void GDScriptAnalyzer::reduce_ternary_op(GDScriptParser::TernaryOpNode *p_ternary_op, bool p_is_root) {
|
|
reduce_expression(p_ternary_op->condition);
|
|
reduce_expression(p_ternary_op->true_expr, p_is_root);
|
|
reduce_expression(p_ternary_op->false_expr, p_is_root);
|
|
|
|
GDScriptParser::DataType result;
|
|
|
|
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) {
|
|
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;
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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;
|
|
#ifdef DEBUG_ENABLED
|
|
parser->push_warning(p_ternary_op, GDScriptWarning::INCOMPATIBLE_TERNARY);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
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;
|
|
|
|
if (p_unary_op->operand == nullptr) {
|
|
result.kind = GDScriptParser::DataType::VARIANT;
|
|
p_unary_op->set_datatype(result);
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType operand_type = p_unary_op->operand->get_datatype();
|
|
|
|
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());
|
|
result = type_from_variant(p_unary_op->reduced_value, p_unary_op);
|
|
}
|
|
|
|
if (operand_type.is_variant()) {
|
|
result.kind = GDScriptParser::DataType::VARIANT;
|
|
mark_node_unsafe(p_unary_op);
|
|
} else {
|
|
bool valid = false;
|
|
result = get_operation_type(p_unary_op->variant_op, operand_type, valid, p_unary_op);
|
|
|
|
if (!valid) {
|
|
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);
|
|
}
|
|
}
|
|
|
|
p_unary_op->set_datatype(result);
|
|
}
|
|
|
|
Variant GDScriptAnalyzer::make_expression_reduced_value(GDScriptParser::ExpressionNode *p_expression, bool &is_reduced) {
|
|
Variant value;
|
|
|
|
if (p_expression->is_constant) {
|
|
is_reduced = true;
|
|
value = p_expression->reduced_value;
|
|
} else if (p_expression->type == GDScriptParser::Node::ARRAY) {
|
|
value = make_array_reduced_value(static_cast<GDScriptParser::ArrayNode *>(p_expression), is_reduced);
|
|
} else if (p_expression->type == GDScriptParser::Node::DICTIONARY) {
|
|
value = make_dictionary_reduced_value(static_cast<GDScriptParser::DictionaryNode *>(p_expression), is_reduced);
|
|
} else if (p_expression->type == GDScriptParser::Node::SUBSCRIPT) {
|
|
value = make_subscript_reduced_value(static_cast<GDScriptParser::SubscriptNode *>(p_expression), is_reduced);
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
Variant GDScriptAnalyzer::make_array_reduced_value(GDScriptParser::ArrayNode *p_array, bool &is_reduced) {
|
|
Array array = p_array->get_datatype().has_container_element_type() ? make_array_from_element_datatype(p_array->get_datatype().get_container_element_type()) : Array();
|
|
|
|
array.resize(p_array->elements.size());
|
|
for (int i = 0; i < p_array->elements.size(); i++) {
|
|
GDScriptParser::ExpressionNode *element = p_array->elements[i];
|
|
|
|
bool is_element_value_reduced = false;
|
|
Variant element_value = make_expression_reduced_value(element, is_element_value_reduced);
|
|
if (!is_element_value_reduced) {
|
|
return Variant();
|
|
}
|
|
|
|
array[i] = element_value;
|
|
}
|
|
|
|
array.make_read_only();
|
|
|
|
is_reduced = true;
|
|
return array;
|
|
}
|
|
|
|
Variant GDScriptAnalyzer::make_dictionary_reduced_value(GDScriptParser::DictionaryNode *p_dictionary, bool &is_reduced) {
|
|
Dictionary dictionary;
|
|
|
|
for (int i = 0; i < p_dictionary->elements.size(); i++) {
|
|
const GDScriptParser::DictionaryNode::Pair &element = p_dictionary->elements[i];
|
|
|
|
bool is_element_key_reduced = false;
|
|
Variant element_key = make_expression_reduced_value(element.key, is_element_key_reduced);
|
|
if (!is_element_key_reduced) {
|
|
return Variant();
|
|
}
|
|
|
|
bool is_element_value_reduced = false;
|
|
Variant element_value = make_expression_reduced_value(element.value, is_element_value_reduced);
|
|
if (!is_element_value_reduced) {
|
|
return Variant();
|
|
}
|
|
|
|
dictionary[element_key] = element_value;
|
|
}
|
|
|
|
dictionary.make_read_only();
|
|
|
|
is_reduced = true;
|
|
return dictionary;
|
|
}
|
|
|
|
Variant GDScriptAnalyzer::make_subscript_reduced_value(GDScriptParser::SubscriptNode *p_subscript, bool &is_reduced) {
|
|
bool is_base_value_reduced = false;
|
|
Variant base_value = make_expression_reduced_value(p_subscript->base, is_base_value_reduced);
|
|
if (!is_base_value_reduced) {
|
|
return Variant();
|
|
}
|
|
|
|
if (p_subscript->is_attribute) {
|
|
bool is_valid = false;
|
|
Variant value = base_value.get_named(p_subscript->attribute->name, is_valid);
|
|
if (is_valid) {
|
|
is_reduced = true;
|
|
return value;
|
|
} else {
|
|
return Variant();
|
|
}
|
|
} else {
|
|
bool is_index_value_reduced = false;
|
|
Variant index_value = make_expression_reduced_value(p_subscript->index, is_index_value_reduced);
|
|
if (!is_index_value_reduced) {
|
|
return Variant();
|
|
}
|
|
|
|
bool is_valid = false;
|
|
Variant value = base_value.get(index_value, &is_valid);
|
|
if (is_valid) {
|
|
is_reduced = true;
|
|
return value;
|
|
} else {
|
|
return Variant();
|
|
}
|
|
}
|
|
}
|
|
|
|
Array GDScriptAnalyzer::make_array_from_element_datatype(const GDScriptParser::DataType &p_element_datatype, const GDScriptParser::Node *p_source_node) {
|
|
Array array;
|
|
|
|
Ref<Script> script_type = p_element_datatype.script_type;
|
|
if (p_element_datatype.kind == GDScriptParser::DataType::CLASS && script_type.is_null()) {
|
|
Error err = OK;
|
|
Ref<GDScript> scr = GDScriptCache::get_shallow_script(p_element_datatype.script_path, err);
|
|
if (err) {
|
|
push_error(vformat(R"(Error while getting cache for script "%s".)", p_element_datatype.script_path), p_source_node);
|
|
return array;
|
|
}
|
|
script_type.reference_ptr(scr->find_class(p_element_datatype.class_type->fqcn));
|
|
}
|
|
|
|
array.set_typed(p_element_datatype.builtin_type, p_element_datatype.native_type, script_type);
|
|
|
|
return array;
|
|
}
|
|
|
|
Variant GDScriptAnalyzer::make_variable_default_value(GDScriptParser::VariableNode *p_variable) {
|
|
Variant result = Variant();
|
|
|
|
if (p_variable->initializer) {
|
|
bool is_initializer_value_reduced = false;
|
|
Variant initializer_value = make_expression_reduced_value(p_variable->initializer, is_initializer_value_reduced);
|
|
if (is_initializer_value_reduced) {
|
|
result = initializer_value;
|
|
}
|
|
} else {
|
|
GDScriptParser::DataType datatype = p_variable->get_datatype();
|
|
if (datatype.is_hard_type() && datatype.kind == GDScriptParser::DataType::BUILTIN && datatype.builtin_type != Variant::OBJECT) {
|
|
if (datatype.builtin_type == Variant::ARRAY && datatype.has_container_element_type()) {
|
|
result = make_array_from_element_datatype(datatype.get_container_element_type());
|
|
} else {
|
|
VariantInternal::initialize(&result, datatype.builtin_type);
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
GDScriptParser::DataType GDScriptAnalyzer::type_from_variant(const Variant &p_value, const GDScriptParser::Node *p_source) {
|
|
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.
|
|
|
|
if (p_value.get_type() == Variant::OBJECT) {
|
|
// Object is treated as a native type, not a builtin type.
|
|
result.kind = GDScriptParser::DataType::NATIVE;
|
|
|
|
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()) {
|
|
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.
|
|
String script_path = gds->get_script_path();
|
|
Ref<GDScriptParserRef> ref = get_parser_for(script_path);
|
|
if (ref.is_null()) {
|
|
push_error(vformat(R"(Could not find script "%s".)", script_path), p_source);
|
|
GDScriptParser::DataType error_type;
|
|
error_type.kind = GDScriptParser::DataType::VARIANT;
|
|
return error_type;
|
|
}
|
|
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);
|
|
}
|
|
}
|
|
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;
|
|
}
|
|
|
|
result.kind = GDScriptParser::DataType::CLASS;
|
|
result.native_type = found->get_datatype().native_type;
|
|
result.class_type = found;
|
|
result.script_path = ref->get_parser()->script_path;
|
|
} else {
|
|
result.kind = GDScriptParser::DataType::SCRIPT;
|
|
result.native_type = scr->get_instance_base_type();
|
|
result.script_path = scr->get_path();
|
|
}
|
|
result.script_type = scr;
|
|
} else {
|
|
result.kind = GDScriptParser::DataType::NATIVE;
|
|
if (result.native_type == GDScriptNativeClass::get_class_static()) {
|
|
result.is_meta_type = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
GDScriptParser::DataType GDScriptAnalyzer::type_from_metatype(const GDScriptParser::DataType &p_meta_type) {
|
|
GDScriptParser::DataType result = p_meta_type;
|
|
result.is_meta_type = false;
|
|
if (p_meta_type.kind == GDScriptParser::DataType::ENUM) {
|
|
result.builtin_type = Variant::INT;
|
|
} else {
|
|
result.is_constant = false;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
GDScriptParser::DataType GDScriptAnalyzer::type_from_property(const PropertyInfo &p_property, bool p_is_arg) const {
|
|
GDScriptParser::DataType result;
|
|
result.type_source = GDScriptParser::DataType::ANNOTATED_EXPLICIT;
|
|
if (p_property.type == Variant::NIL && (p_is_arg || (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;
|
|
result.native_type = p_property.class_name == StringName() ? SNAME("Object") : p_property.class_name;
|
|
} else {
|
|
result.kind = GDScriptParser::DataType::BUILTIN;
|
|
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.");
|
|
}
|
|
elem_type.is_constant = false;
|
|
result.set_container_element_type(elem_type);
|
|
} else if (p_property.type == Variant::INT) {
|
|
// Check if it's enum.
|
|
if (p_property.class_name != StringName()) {
|
|
Vector<String> names = String(p_property.class_name).split(".");
|
|
if (names.size() == 2) {
|
|
result = make_native_enum_type(names[1], names[0]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
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, StringName *r_native_class) {
|
|
r_static = false;
|
|
r_vararg = false;
|
|
r_default_arg_count = 0;
|
|
if (r_native_class) {
|
|
*r_native_class = StringName();
|
|
}
|
|
StringName function_name = p_function;
|
|
|
|
bool was_enum = false;
|
|
if (p_base_type.kind == GDScriptParser::DataType::ENUM) {
|
|
was_enum = true;
|
|
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;
|
|
}
|
|
}
|
|
|
|
if (p_base_type.kind == GDScriptParser::DataType::BUILTIN) {
|
|
// Construct a base type to get methods.
|
|
Callable::CallError err;
|
|
Variant dummy;
|
|
Variant::construct(p_base_type.builtin_type, dummy, nullptr, 0, err);
|
|
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);
|
|
|
|
for (const MethodInfo &E : methods) {
|
|
if (E.name == p_function) {
|
|
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);
|
|
// 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);
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
StringName base_native = p_base_type.native_type;
|
|
if (base_native != StringName()) {
|
|
// Empty native class might happen in some Script implementations.
|
|
// Just ignore it.
|
|
if (!class_exists(base_native)) {
|
|
push_error(vformat("Native class %s used in script doesn't exist or isn't exposed.", base_native), p_source);
|
|
return false;
|
|
} else if (p_is_constructor && !ClassDB::can_instantiate(base_native)) {
|
|
if (p_base_type.kind == GDScriptParser::DataType::CLASS) {
|
|
push_error(vformat(R"(Class "%s" cannot be constructed as it is based on abstract native class "%s".)", p_base_type.class_type->fqcn.get_file(), base_native), p_source);
|
|
} else if (p_base_type.kind == GDScriptParser::DataType::SCRIPT) {
|
|
push_error(vformat(R"(Script "%s" cannot be constructed as it is based on abstract native class "%s".)", p_base_type.script_path.get_file(), base_native), p_source);
|
|
} else {
|
|
push_error(vformat(R"(Native class "%s" cannot be constructed as it is abstract.)", base_native), p_source);
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (p_is_constructor) {
|
|
function_name = "_init";
|
|
r_static = true;
|
|
}
|
|
|
|
GDScriptParser::ClassNode *base_class = p_base_type.class_type;
|
|
GDScriptParser::FunctionNode *found_function = nullptr;
|
|
|
|
while (found_function == nullptr && base_class != nullptr) {
|
|
if (base_class->has_member(function_name)) {
|
|
if (base_class->get_member(function_name).type != GDScriptParser::ClassNode::Member::FUNCTION) {
|
|
// TODO: If this is Callable it can have a better error message.
|
|
push_error(vformat(R"(Member "%s" is not a function.)", function_name), p_source);
|
|
return false;
|
|
}
|
|
|
|
resolve_class_member(base_class, function_name, p_source);
|
|
found_function = base_class->get_member(function_name).function;
|
|
}
|
|
|
|
resolve_class_inheritance(base_class, p_source);
|
|
base_class = base_class->base_type.class_type;
|
|
}
|
|
|
|
if (found_function != nullptr) {
|
|
r_static = p_is_constructor || found_function->is_static;
|
|
for (int i = 0; i < found_function->parameters.size(); i++) {
|
|
r_par_types.push_back(found_function->parameters[i]->get_datatype());
|
|
if (found_function->parameters[i]->initializer != nullptr) {
|
|
r_default_arg_count++;
|
|
}
|
|
}
|
|
r_return_type = p_is_constructor ? p_base_type : found_function->get_datatype();
|
|
r_return_type.is_meta_type = false;
|
|
r_return_type.is_coroutine = found_function->is_coroutine;
|
|
|
|
return true;
|
|
}
|
|
|
|
Ref<Script> base_script = p_base_type.script_type;
|
|
|
|
while (base_script.is_valid() && base_script->has_method(function_name)) {
|
|
MethodInfo info = base_script->get_method_info(function_name);
|
|
|
|
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.
|
|
if (p_base_type.is_meta_type && !p_is_constructor && (p_base_type.kind == GDScriptParser::DataType::SCRIPT || p_base_type.kind == GDScriptParser::DataType::CLASS)) {
|
|
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());
|
|
|
|
if (ClassDB::get_method_info(script_class, function_name, &info)) {
|
|
return function_signature_from_info(info, r_return_type, r_par_types, r_default_arg_count, r_static, r_vararg);
|
|
}
|
|
}
|
|
|
|
if (p_is_constructor) {
|
|
// 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;
|
|
}
|
|
|
|
MethodInfo info;
|
|
if (ClassDB::get_method_info(base_native, function_name, &info)) {
|
|
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;
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
MethodBind *native_method = ClassDB::get_method(base_native, function_name);
|
|
if (native_method && r_native_class) {
|
|
*r_native_class = native_method->get_instance_class();
|
|
}
|
|
#endif
|
|
return valid;
|
|
}
|
|
|
|
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;
|
|
r_static = (p_info.flags & METHOD_FLAG_STATIC) != 0;
|
|
|
|
for (const PropertyInfo &E : p_info.arguments) {
|
|
r_par_types.push_back(type_from_property(E, true));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void GDScriptAnalyzer::validate_call_arg(const MethodInfo &p_method, const GDScriptParser::CallNode *p_call) {
|
|
List<GDScriptParser::DataType> arg_types;
|
|
|
|
for (const PropertyInfo &E : p_method.arguments) {
|
|
arg_types.push_back(type_from_property(E, true));
|
|
}
|
|
|
|
validate_call_arg(arg_types, p_method.default_arguments.size(), (p_method.flags & METHOD_FLAG_VARARG) != 0, p_call);
|
|
}
|
|
|
|
void 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) {
|
|
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);
|
|
}
|
|
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()]);
|
|
}
|
|
|
|
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];
|
|
|
|
if (par_type.is_hard_type() && p_call->arguments[i]->is_constant) {
|
|
update_const_expression_builtin_type(p_call->arguments[i], par_type, "pass");
|
|
}
|
|
GDScriptParser::DataType arg_type = p_call->arguments[i]->get_datatype();
|
|
|
|
if ((arg_type.is_variant() || !arg_type.is_hard_type()) && !(par_type.is_hard_type() && par_type.is_variant())) {
|
|
// Argument can be anything, so this is unsafe.
|
|
mark_node_unsafe(p_call->arguments[i]);
|
|
} else if (par_type.is_hard_type() && !is_type_compatible(par_type, arg_type, true)) {
|
|
// Supertypes are acceptable for dynamic compliance, but it's unsafe.
|
|
mark_node_unsafe(p_call);
|
|
if (!is_type_compatible(arg_type, par_type)) {
|
|
push_error(vformat(R"*(Invalid argument for "%s()" function: argument %d should be "%s" but is "%s".)*",
|
|
p_call->function_name, i + 1, par_type.to_string(), arg_type.to_string()),
|
|
p_call->arguments[i]);
|
|
}
|
|
#ifdef DEBUG_ENABLED
|
|
} 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) {
|
|
parser->push_warning(p_call, GDScriptWarning::NARROWING_CONVERSION, p_call->function_name);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
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;
|
|
|
|
{
|
|
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;
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
StringName parent = base.native_type;
|
|
while (parent != StringName()) {
|
|
ERR_FAIL_COND_V_MSG(!class_exists(parent), false, "Non-existent native base class.");
|
|
|
|
if (ClassDB::has_method(parent, name, true)) {
|
|
parser->push_warning(p_local, GDScriptWarning::SHADOWED_VARIABLE_BASE_CLASS, p_context, p_local->name, "method", parent);
|
|
return true;
|
|
} else if (ClassDB::has_signal(parent, name, true)) {
|
|
parser->push_warning(p_local, GDScriptWarning::SHADOWED_VARIABLE_BASE_CLASS, p_context, p_local->name, "signal", parent);
|
|
return true;
|
|
} else if (ClassDB::has_property(parent, name, true)) {
|
|
parser->push_warning(p_local, GDScriptWarning::SHADOWED_VARIABLE_BASE_CLASS, p_context, p_local->name, "property", parent);
|
|
return true;
|
|
} else if (ClassDB::has_integer_constant(parent, name, true)) {
|
|
parser->push_warning(p_local, GDScriptWarning::SHADOWED_VARIABLE_BASE_CLASS, p_context, p_local->name, "constant", parent);
|
|
return true;
|
|
} else if (ClassDB::has_enum(parent, name, true)) {
|
|
parser->push_warning(p_local, GDScriptWarning::SHADOWED_VARIABLE_BASE_CLASS, p_context, p_local->name, "enum", parent);
|
|
return true;
|
|
}
|
|
parent = ClassDB::get_parent_class(parent);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
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;
|
|
nil_type.type_source = GDScriptParser::DataType::ANNOTATED_INFERRED;
|
|
return get_operation_type(p_operation, p_a, nil_type, r_valid, p_source);
|
|
}
|
|
|
|
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) {
|
|
Variant::Type a_type = p_a.builtin_type;
|
|
Variant::Type b_type = p_b.builtin_type;
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
Variant::ValidatedOperatorEvaluator op_eval = Variant::get_validated_operator_evaluator(p_operation, a_type, b_type);
|
|
bool hard_operation = p_a.is_hard_type() && p_b.is_hard_type();
|
|
bool validated = op_eval != nullptr;
|
|
|
|
GDScriptParser::DataType result;
|
|
if (validated) {
|
|
r_valid = true;
|
|
result.type_source = hard_operation ? GDScriptParser::DataType::ANNOTATED_INFERRED : GDScriptParser::DataType::INFERRED;
|
|
result.kind = GDScriptParser::DataType::BUILTIN;
|
|
result.builtin_type = Variant::get_operator_return_type(p_operation, a_type, b_type);
|
|
} else {
|
|
r_valid = !hard_operation;
|
|
result.kind = GDScriptParser::DataType::VARIANT;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
// TODO: Add safe/unsafe return variable (for variant cases)
|
|
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) {
|
|
// 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);
|
|
}
|
|
if (!valid && p_target.builtin_type == Variant::INT && p_source.kind == GDScriptParser::DataType::ENUM && !p_source.is_meta_type) {
|
|
// Enum value is also integer.
|
|
valid = true;
|
|
}
|
|
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() && p_source.has_container_element_type()) {
|
|
valid = p_target.get_container_element_type() == p_source.get_container_element_type();
|
|
}
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
if (p_target.kind == GDScriptParser::DataType::ENUM) {
|
|
if (p_source.kind == GDScriptParser::DataType::BUILTIN && p_source.builtin_type == Variant::INT) {
|
|
#ifdef DEBUG_ENABLED
|
|
if (p_source_node) {
|
|
parser->push_warning(p_source_node, GDScriptWarning::INT_AS_ENUM_WITHOUT_CAST);
|
|
}
|
|
#endif
|
|
return true;
|
|
}
|
|
if (p_source.kind == GDScriptParser::DataType::ENUM) {
|
|
if (p_source.native_type == p_target.native_type) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// 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:
|
|
case GDScriptParser::DataType::ENUM:
|
|
case GDScriptParser::DataType::RESOLVING:
|
|
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());
|
|
}
|
|
return ClassDB::is_parent_class(src_native, p_target.native_type);
|
|
}
|
|
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 == p_target.class_type || 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:
|
|
case GDScriptParser::DataType::ENUM:
|
|
case GDScriptParser::DataType::RESOLVING:
|
|
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) {
|
|
#ifdef DEBUG_ENABLED
|
|
if (p_node == nullptr) {
|
|
return;
|
|
}
|
|
|
|
for (int i = p_node->start_line; i <= p_node->end_line; i++) {
|
|
parser->unsafe_lines.insert(i);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void GDScriptAnalyzer::downgrade_node_type_source(GDScriptParser::Node *p_node) {
|
|
GDScriptParser::IdentifierNode *identifier = nullptr;
|
|
if (p_node->type == GDScriptParser::Node::IDENTIFIER) {
|
|
identifier = static_cast<GDScriptParser::IdentifierNode *>(p_node);
|
|
} else if (p_node->type == GDScriptParser::Node::SUBSCRIPT) {
|
|
GDScriptParser::SubscriptNode *subscript = static_cast<GDScriptParser::SubscriptNode *>(p_node);
|
|
if (subscript->is_attribute) {
|
|
identifier = subscript->attribute;
|
|
}
|
|
}
|
|
if (identifier == nullptr) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::Node *source = nullptr;
|
|
switch (identifier->source) {
|
|
case GDScriptParser::IdentifierNode::MEMBER_VARIABLE: {
|
|
source = identifier->variable_source;
|
|
} break;
|
|
case GDScriptParser::IdentifierNode::FUNCTION_PARAMETER: {
|
|
source = identifier->parameter_source;
|
|
} break;
|
|
case GDScriptParser::IdentifierNode::LOCAL_VARIABLE: {
|
|
source = identifier->variable_source;
|
|
} break;
|
|
case GDScriptParser::IdentifierNode::LOCAL_ITERATOR: {
|
|
source = identifier->bind_source;
|
|
} break;
|
|
default:
|
|
break;
|
|
}
|
|
if (source == nullptr) {
|
|
return;
|
|
}
|
|
|
|
GDScriptParser::DataType datatype;
|
|
datatype.kind = GDScriptParser::DataType::VARIANT;
|
|
source->set_datatype(datatype);
|
|
}
|
|
|
|
void GDScriptAnalyzer::mark_lambda_use_self() {
|
|
for (GDScriptParser::LambdaNode *lambda : lambda_stack) {
|
|
lambda->use_self = true;
|
|
}
|
|
}
|
|
|
|
bool GDScriptAnalyzer::class_exists(const StringName &p_class) const {
|
|
return ClassDB::class_exists(p_class) && ClassDB::is_class_exposed(p_class);
|
|
}
|
|
|
|
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;
|
|
ref = GDScriptCache::get_parser(p_path, GDScriptParserRef::EMPTY, err, parser->script_path);
|
|
if (ref.is_valid()) {
|
|
depended_parsers[p_path] = ref;
|
|
}
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
Error GDScriptAnalyzer::resolve_inheritance() {
|
|
return resolve_class_inheritance(parser->head, true);
|
|
}
|
|
|
|
Error GDScriptAnalyzer::resolve_interface() {
|
|
resolve_class_interface(parser->head, true);
|
|
return parser->errors.is_empty() ? OK : ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Error GDScriptAnalyzer::resolve_body() {
|
|
resolve_class_body(parser->head, true);
|
|
return parser->errors.is_empty() ? OK : ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Error GDScriptAnalyzer::resolve_dependencies() {
|
|
for (KeyValue<String, Ref<GDScriptParserRef>> &K : depended_parsers) {
|
|
if (K.value.is_null()) {
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
K.value->raise_status(GDScriptParserRef::INHERITANCE_SOLVED);
|
|
}
|
|
|
|
return parser->errors.is_empty() ? OK : ERR_PARSE_ERROR;
|
|
}
|
|
|
|
Error GDScriptAnalyzer::analyze() {
|
|
parser->errors.clear();
|
|
Error err = OK;
|
|
|
|
err = resolve_inheritance();
|
|
if (err) {
|
|
return err;
|
|
}
|
|
|
|
// Apply annotations.
|
|
for (GDScriptParser::AnnotationNode *&E : parser->head->annotations) {
|
|
resolve_annotation(E);
|
|
E->apply(parser, parser->head);
|
|
}
|
|
|
|
resolve_interface();
|
|
resolve_body();
|
|
if (!parser->errors.is_empty()) {
|
|
return ERR_PARSE_ERROR;
|
|
}
|
|
|
|
return resolve_dependencies();
|
|
}
|
|
|
|
GDScriptAnalyzer::GDScriptAnalyzer(GDScriptParser *p_parser) {
|
|
parser = p_parser;
|
|
}
|