|
|
@ -285,7 +285,7 @@ Error GDScriptAnalyzer::check_class_member_name_conflict(const GDScriptParser::C
|
|
|
|
return OK;
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GDScriptAnalyzer::get_class_node_current_scope_classes(GDScriptParser::ClassNode *p_node, List<GDScriptParser::ClassNode *> *p_list) {
|
|
|
|
void GDScriptAnalyzer::get_class_node_current_scope_classes(GDScriptParser::ClassNode *p_node, List<GDScriptParser::ClassNode *> *p_list, GDScriptParser::Node *p_source) {
|
|
|
|
ERR_FAIL_NULL(p_node);
|
|
|
|
ERR_FAIL_NULL(p_node);
|
|
|
|
ERR_FAIL_NULL(p_list);
|
|
|
|
ERR_FAIL_NULL(p_list);
|
|
|
|
|
|
|
|
|
|
|
@ -299,11 +299,13 @@ void GDScriptAnalyzer::get_class_node_current_scope_classes(GDScriptParser::Clas
|
|
|
|
|
|
|
|
|
|
|
|
// Prioritize node base type over its outer class
|
|
|
|
// Prioritize node base type over its outer class
|
|
|
|
if (p_node->base_type.class_type != nullptr) {
|
|
|
|
if (p_node->base_type.class_type != nullptr) {
|
|
|
|
get_class_node_current_scope_classes(p_node->base_type.class_type, p_list);
|
|
|
|
ensure_cached_parser_for_class(p_node->base_type.class_type, p_node, vformat(R"(Trying to fetch classes in the current scope: base class of "%s")", p_node->fqcn), p_source);
|
|
|
|
|
|
|
|
get_class_node_current_scope_classes(p_node->base_type.class_type, p_list, p_source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (p_node->outer != nullptr) {
|
|
|
|
if (p_node->outer != nullptr) {
|
|
|
|
get_class_node_current_scope_classes(p_node->outer, p_list);
|
|
|
|
ensure_cached_parser_for_class(p_node->outer, p_node, vformat(R"(Trying to fetch classes in the current scope: outer class of "%s")", p_node->fqcn), p_source);
|
|
|
|
|
|
|
|
get_class_node_current_scope_classes(p_node->outer, p_list, p_source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -312,6 +314,13 @@ Error GDScriptAnalyzer::resolve_class_inheritance(GDScriptParser::ClassNode *p_c
|
|
|
|
p_source = p_class;
|
|
|
|
p_source = p_class;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = ensure_cached_parser_for_class(p_class, nullptr, vformat(R"(Trying to resolve class inheritance of "%s")", p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
Finally finally([&]() {
|
|
|
|
|
|
|
|
for (GDScriptParser::ClassNode *look_class = p_class; look_class != nullptr; look_class = look_class->base_type.class_type) {
|
|
|
|
|
|
|
|
ensure_cached_parser_for_class(look_class->base_type.class_type, look_class, vformat(R"(Trying to resolve class inheritance of "%s")", p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
if (p_class->base_type.is_resolving()) {
|
|
|
|
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);
|
|
|
|
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;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
@ -323,21 +332,17 @@ Error GDScriptAnalyzer::resolve_class_inheritance(GDScriptParser::ClassNode *p_c
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
String script_path = p_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = parser->get_depended_parser_for(script_path);
|
|
|
|
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
push_error(vformat(R"(Could not find script "%s".)", script_path), p_source);
|
|
|
|
// Error already pushed.
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
push_error(vformat(R"(Could not parse script "%s": %s.)", script_path, error_names[err]), p_source);
|
|
|
|
push_error(vformat(R"(Could not parse script "%s": %s.)", p_class->get_datatype().script_path, error_names[err]), p_source);
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
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();
|
|
|
|
GDScriptAnalyzer *other_analyzer = parser_ref->get_analyzer();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
|
|
|
|
|
|
|
@ -471,7 +476,7 @@ Error GDScriptAnalyzer::resolve_class_inheritance(GDScriptParser::ClassNode *p_c
|
|
|
|
// Look for other classes in script.
|
|
|
|
// Look for other classes in script.
|
|
|
|
bool found = false;
|
|
|
|
bool found = false;
|
|
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
|
|
get_class_node_current_scope_classes(p_class, &script_classes);
|
|
|
|
get_class_node_current_scope_classes(p_class, &script_classes, id);
|
|
|
|
for (GDScriptParser::ClassNode *look_class : script_classes) {
|
|
|
|
for (GDScriptParser::ClassNode *look_class : script_classes) {
|
|
|
|
if (look_class->identifier && look_class->identifier->name == name) {
|
|
|
|
if (look_class->identifier && look_class->identifier->name == name) {
|
|
|
|
if (!look_class->get_datatype().is_set()) {
|
|
|
|
if (!look_class->get_datatype().is_set()) {
|
|
|
@ -763,7 +768,7 @@ GDScriptParser::DataType GDScriptAnalyzer::resolve_datatype(GDScriptParser::Type
|
|
|
|
// Classes in current scope.
|
|
|
|
// Classes in current scope.
|
|
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
|
|
List<GDScriptParser::ClassNode *> script_classes;
|
|
|
|
bool found = false;
|
|
|
|
bool found = false;
|
|
|
|
get_class_node_current_scope_classes(parser->current_class, &script_classes);
|
|
|
|
get_class_node_current_scope_classes(parser->current_class, &script_classes, p_type);
|
|
|
|
for (GDScriptParser::ClassNode *script_class : script_classes) {
|
|
|
|
for (GDScriptParser::ClassNode *script_class : script_classes) {
|
|
|
|
if (found) {
|
|
|
|
if (found) {
|
|
|
|
break;
|
|
|
|
break;
|
|
|
@ -883,6 +888,15 @@ void GDScriptAnalyzer::resolve_class_member(GDScriptParser::ClassNode *p_class,
|
|
|
|
p_source = member.get_source_node();
|
|
|
|
p_source = member.get_source_node();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = ensure_cached_parser_for_class(p_class, nullptr, vformat(R"(Trying to resolve class member "%s" of "%s")", member.get_name(), p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
Finally finally([&]() {
|
|
|
|
|
|
|
|
ensure_cached_parser_for_class(member.get_datatype().class_type, p_class, vformat(R"(Trying to resolve datatype of class member "%s" of "%s")", member.get_name(), p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
GDScriptParser::DataType member_type = member.get_datatype();
|
|
|
|
|
|
|
|
if (member_type.has_container_element_type(0)) {
|
|
|
|
|
|
|
|
ensure_cached_parser_for_class(member_type.get_container_element_type(0).class_type, p_class, vformat(R"(Trying to resolve datatype of class member "%s" of "%s")", member.get_name(), p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
if (member.get_datatype().is_resolving()) {
|
|
|
|
if (member.get_datatype().is_resolving()) {
|
|
|
|
push_error(vformat(R"(Could not resolve member "%s": Cyclic reference.)", member.get_name()), p_source);
|
|
|
|
push_error(vformat(R"(Could not resolve member "%s": Cyclic reference.)", member.get_name()), p_source);
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -892,42 +906,39 @@ void GDScriptAnalyzer::resolve_class_member(GDScriptParser::ClassNode *p_class,
|
|
|
|
return;
|
|
|
|
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;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
String script_path = p_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = parser->get_depended_parser_for(script_path);
|
|
|
|
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
push_error(vformat(R"(Could not find script "%s" (While resolving "%s").)", script_path, member.get_name()), p_source);
|
|
|
|
// Error already pushed.
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
if (err) {
|
|
|
|
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);
|
|
|
|
push_error(vformat(R"(Could not parse script "%s": %s (While resolving external class member "%s").)", p_class->get_datatype().script_path, error_names[err], member.get_name()), p_source);
|
|
|
|
return;
|
|
|
|
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();
|
|
|
|
GDScriptAnalyzer *other_analyzer = parser_ref->get_analyzer();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
|
|
|
|
|
|
|
|
int error_count = other_parser->errors.size();
|
|
|
|
int error_count = other_parser->errors.size();
|
|
|
|
other_analyzer->resolve_class_member(p_class, p_index);
|
|
|
|
other_analyzer->resolve_class_member(p_class, p_index);
|
|
|
|
if (other_parser->errors.size() > error_count) {
|
|
|
|
if (other_parser->errors.size() > error_count) {
|
|
|
|
push_error(vformat(R"(Could not resolve member "%s".)", member.get_name()), p_source);
|
|
|
|
push_error(vformat(R"(Could not resolve external class member "%s".)", member.get_name()), p_source);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
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;
|
|
|
|
GDScriptParser::ClassNode *previous_class = parser->current_class;
|
|
|
|
parser->current_class = p_class;
|
|
|
|
parser->current_class = p_class;
|
|
|
|
|
|
|
|
|
|
|
@ -1170,27 +1181,25 @@ void GDScriptAnalyzer::resolve_class_interface(GDScriptParser::ClassNode *p_clas
|
|
|
|
p_source = p_class;
|
|
|
|
p_source = p_class;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = ensure_cached_parser_for_class(p_class, nullptr, vformat(R"(Trying to resolve class interface of "%s")", p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
|
|
|
|
if (!p_class->resolved_interface) {
|
|
|
|
if (!p_class->resolved_interface) {
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
|
|
#ifdef DEBUG_ENABLED
|
|
|
|
bool has_static_data = p_class->has_static_data;
|
|
|
|
bool has_static_data = p_class->has_static_data;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
String script_path = p_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = parser->get_depended_parser_for(script_path);
|
|
|
|
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
push_error(vformat(R"(Could not find script "%s".)", script_path), p_source);
|
|
|
|
// Error already pushed.
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
push_error(vformat(R"(Could not resolve script "%s": %s.)", script_path, error_names[err]), p_source);
|
|
|
|
push_error(vformat(R"(Could not parse script "%s": %s.)", p_class->get_datatype().script_path, error_names[err]), p_source);
|
|
|
|
return;
|
|
|
|
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();
|
|
|
|
GDScriptAnalyzer *other_analyzer = parser_ref->get_analyzer();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
|
|
|
|
|
|
|
@ -1198,6 +1207,7 @@ void GDScriptAnalyzer::resolve_class_interface(GDScriptParser::ClassNode *p_clas
|
|
|
|
other_analyzer->resolve_class_interface(p_class);
|
|
|
|
other_analyzer->resolve_class_interface(p_class);
|
|
|
|
if (other_parser->errors.size() > error_count) {
|
|
|
|
if (other_parser->errors.size() > error_count) {
|
|
|
|
push_error(vformat(R"(Could not resolve class "%s".)", p_class->fqcn), p_source);
|
|
|
|
push_error(vformat(R"(Could not resolve class "%s".)", p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -1261,26 +1271,24 @@ void GDScriptAnalyzer::resolve_class_body(GDScriptParser::ClassNode *p_class, co
|
|
|
|
p_source = p_class;
|
|
|
|
p_source = p_class;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = ensure_cached_parser_for_class(p_class, nullptr, vformat(R"(Trying to resolve class body of "%s")", p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
|
|
|
|
if (p_class->resolved_body) {
|
|
|
|
if (p_class->resolved_body) {
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
if (!parser->has_class(p_class)) {
|
|
|
|
String script_path = p_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref = parser->get_depended_parser_for(script_path);
|
|
|
|
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
push_error(vformat(R"(Could not find script "%s".)", script_path), p_source);
|
|
|
|
// Error already pushed.
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
Error err = parser_ref->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
push_error(vformat(R"(Could not resolve script "%s": %s.)", script_path, error_names[err]), p_source);
|
|
|
|
push_error(vformat(R"(Could not parse script "%s": %s.)", p_class->get_datatype().script_path, error_names[err]), p_source);
|
|
|
|
return;
|
|
|
|
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();
|
|
|
|
GDScriptAnalyzer *other_analyzer = parser_ref->get_analyzer();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
GDScriptParser *other_parser = parser_ref->get_parser();
|
|
|
|
|
|
|
|
|
|
|
@ -1288,6 +1296,7 @@ void GDScriptAnalyzer::resolve_class_body(GDScriptParser::ClassNode *p_class, co
|
|
|
|
other_analyzer->resolve_class_body(p_class);
|
|
|
|
other_analyzer->resolve_class_body(p_class);
|
|
|
|
if (other_parser->errors.size() > error_count) {
|
|
|
|
if (other_parser->errors.size() > error_count) {
|
|
|
|
push_error(vformat(R"(Could not resolve class "%s".)", p_class->fqcn), p_source);
|
|
|
|
push_error(vformat(R"(Could not resolve class "%s".)", p_class->fqcn), p_source);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -3645,12 +3654,116 @@ GDScriptParser::DataType GDScriptAnalyzer::make_global_class_meta_type(const Str
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> GDScriptAnalyzer::ensure_cached_parser_for_class(const GDScriptParser::ClassNode *p_class, const GDScriptParser::ClassNode *p_from_class, const String &p_context, const GDScriptParser::Node *p_source) {
|
|
|
|
|
|
|
|
if (p_class == nullptr) {
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (parser->has_class(p_class)) {
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (HashMap<const GDScriptParser::ClassNode *, Ref<GDScriptParserRef>>::Iterator E = external_class_parser_cache.find(p_class)) {
|
|
|
|
|
|
|
|
return E->value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (p_from_class == nullptr) {
|
|
|
|
|
|
|
|
p_from_class = parser->head;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
String script_path = p_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> parser_ref;
|
|
|
|
|
|
|
|
for (const GDScriptParser::ClassNode *look_class = p_from_class; look_class != nullptr; look_class = look_class->base_type.class_type) {
|
|
|
|
|
|
|
|
if (parser->has_class(look_class)) {
|
|
|
|
|
|
|
|
parser_ref = find_cached_parser_for_class(p_class, parser);
|
|
|
|
|
|
|
|
if (parser_ref.is_valid()) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (HashMap<const GDScriptParser::ClassNode *, Ref<GDScriptParserRef>>::Iterator E = external_class_parser_cache.find(look_class)) {
|
|
|
|
|
|
|
|
parser_ref = find_cached_parser_for_class(p_class, E->value);
|
|
|
|
|
|
|
|
if (parser_ref.is_valid()) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
String look_class_script_path = look_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
if (HashMap<String, Ref<GDScriptParserRef>>::Iterator E = parser->depended_parsers.find(look_class_script_path)) {
|
|
|
|
|
|
|
|
parser_ref = find_cached_parser_for_class(p_class, E->value);
|
|
|
|
|
|
|
|
if (parser_ref.is_valid()) {
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (parser_ref.is_null()) {
|
|
|
|
|
|
|
|
push_error(vformat(R"(Parser bug: Could not find external parser. (%s))", p_context), p_source);
|
|
|
|
|
|
|
|
// A null parser will be inserted into the cache and this error won't spam for the same class.
|
|
|
|
|
|
|
|
// This is ok, the values of external_class_parser_cache are not assumed to be valid references.
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
external_class_parser_cache.insert(p_class, parser_ref);
|
|
|
|
|
|
|
|
return parser_ref;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> GDScriptAnalyzer::find_cached_parser_for_class(const GDScriptParser::ClassNode *p_class, const Ref<GDScriptParserRef> &p_dependant_parser) {
|
|
|
|
|
|
|
|
if (p_dependant_parser.is_null()) {
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (HashMap<const GDScriptParser::ClassNode *, Ref<GDScriptParserRef>>::Iterator E = p_dependant_parser->get_analyzer()->external_class_parser_cache.find(p_class)) {
|
|
|
|
|
|
|
|
if (E->value.is_valid()) {
|
|
|
|
|
|
|
|
// Silently ensure it's parsed.
|
|
|
|
|
|
|
|
E->value->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
|
|
|
|
if (E->value->get_parser()->has_class(p_class)) {
|
|
|
|
|
|
|
|
return E->value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (p_dependant_parser->get_parser()->has_class(p_class)) {
|
|
|
|
|
|
|
|
return p_dependant_parser;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Silently ensure it's parsed.
|
|
|
|
|
|
|
|
p_dependant_parser->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
|
|
|
|
return find_cached_parser_for_class(p_class, p_dependant_parser->get_parser());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScriptParserRef> GDScriptAnalyzer::find_cached_parser_for_class(const GDScriptParser::ClassNode *p_class, GDScriptParser *p_dependant_parser) {
|
|
|
|
|
|
|
|
if (p_dependant_parser == nullptr) {
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
String script_path = p_class->get_datatype().script_path;
|
|
|
|
|
|
|
|
if (HashMap<String, Ref<GDScriptParserRef>>::Iterator E = p_dependant_parser->depended_parsers.find(script_path)) {
|
|
|
|
|
|
|
|
if (E->value.is_valid()) {
|
|
|
|
|
|
|
|
// Silently ensure it's parsed.
|
|
|
|
|
|
|
|
E->value->raise_status(GDScriptParserRef::PARSED);
|
|
|
|
|
|
|
|
if (E->value->get_parser()->has_class(p_class)) {
|
|
|
|
|
|
|
|
return E->value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ref<GDScript> GDScriptAnalyzer::get_depended_shallow_script(const String &p_path, Error &r_error) {
|
|
|
|
|
|
|
|
// To keep a local cache of the parser for resolving external nodes later.
|
|
|
|
|
|
|
|
parser->get_depended_parser_for(p_path);
|
|
|
|
|
|
|
|
Ref<GDScript> scr = GDScriptCache::get_shallow_script(p_path, r_error, parser->script_path);
|
|
|
|
|
|
|
|
return scr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GDScriptAnalyzer::reduce_identifier_from_base_set_class(GDScriptParser::IdentifierNode *p_identifier, GDScriptParser::DataType p_identifier_datatype) {
|
|
|
|
void GDScriptAnalyzer::reduce_identifier_from_base_set_class(GDScriptParser::IdentifierNode *p_identifier, GDScriptParser::DataType p_identifier_datatype) {
|
|
|
|
ERR_FAIL_NULL(p_identifier);
|
|
|
|
ERR_FAIL_NULL(p_identifier);
|
|
|
|
|
|
|
|
|
|
|
|
p_identifier->set_datatype(p_identifier_datatype);
|
|
|
|
p_identifier->set_datatype(p_identifier_datatype);
|
|
|
|
Error err = OK;
|
|
|
|
Error err = OK;
|
|
|
|
Ref<GDScript> scr = GDScriptCache::get_shallow_script(p_identifier_datatype.script_path, err, parser->script_path);
|
|
|
|
Ref<GDScript> scr = get_depended_shallow_script(p_identifier_datatype.script_path, err);
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
push_error(vformat(R"(Error while getting cache for script "%s".)", p_identifier_datatype.script_path), p_identifier);
|
|
|
|
push_error(vformat(R"(Error while getting cache for script "%s".)", p_identifier_datatype.script_path), p_identifier);
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -3767,7 +3880,7 @@ void GDScriptAnalyzer::reduce_identifier_from_base(GDScriptParser::IdentifierNod
|
|
|
|
bool is_base = true;
|
|
|
|
bool is_base = true;
|
|
|
|
|
|
|
|
|
|
|
|
if (base_class != nullptr) {
|
|
|
|
if (base_class != nullptr) {
|
|
|
|
get_class_node_current_scope_classes(base_class, &script_classes);
|
|
|
|
get_class_node_current_scope_classes(base_class, &script_classes, p_identifier);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool is_constructor = base.is_meta_type && p_identifier->name == SNAME("new");
|
|
|
|
bool is_constructor = base.is_meta_type && p_identifier->name == SNAME("new");
|
|
|
@ -4340,7 +4453,7 @@ void GDScriptAnalyzer::reduce_preload(GDScriptParser::PreloadNode *p_preload) {
|
|
|
|
const String &res_type = ResourceLoader::get_resource_type(p_preload->resolved_path);
|
|
|
|
const String &res_type = ResourceLoader::get_resource_type(p_preload->resolved_path);
|
|
|
|
if (res_type == "GDScript") {
|
|
|
|
if (res_type == "GDScript") {
|
|
|
|
Error err = OK;
|
|
|
|
Error err = OK;
|
|
|
|
Ref<GDScript> res = GDScriptCache::get_shallow_script(p_preload->resolved_path, err, parser->script_path);
|
|
|
|
Ref<GDScript> res = get_depended_shallow_script(p_preload->resolved_path, err);
|
|
|
|
p_preload->resource = res;
|
|
|
|
p_preload->resource = res;
|
|
|
|
if (err != OK) {
|
|
|
|
if (err != OK) {
|
|
|
|
push_error(vformat(R"(Could not preload resource script "%s".)", p_preload->resolved_path), p_preload->path);
|
|
|
|
push_error(vformat(R"(Could not preload resource script "%s".)", p_preload->resolved_path), p_preload->path);
|
|
|
@ -4361,6 +4474,7 @@ void GDScriptAnalyzer::reduce_preload(GDScriptParser::PreloadNode *p_preload) {
|
|
|
|
p_preload->is_constant = true;
|
|
|
|
p_preload->is_constant = true;
|
|
|
|
p_preload->reduced_value = p_preload->resource;
|
|
|
|
p_preload->reduced_value = p_preload->resource;
|
|
|
|
p_preload->set_datatype(type_from_variant(p_preload->reduced_value, p_preload));
|
|
|
|
p_preload->set_datatype(type_from_variant(p_preload->reduced_value, p_preload));
|
|
|
|
|
|
|
|
ensure_cached_parser_for_class(p_preload->get_datatype().class_type, nullptr, vformat(R"(Trying to resolve preload of "%s")", p_preload->resolved_path), p_preload);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GDScriptAnalyzer::reduce_self(GDScriptParser::SelfNode *p_self) {
|
|
|
|
void GDScriptAnalyzer::reduce_self(GDScriptParser::SelfNode *p_self) {
|
|
|
@ -4916,7 +5030,7 @@ Array GDScriptAnalyzer::make_array_from_element_datatype(const GDScriptParser::D
|
|
|
|
Ref<Script> script_type = p_element_datatype.script_type;
|
|
|
|
Ref<Script> script_type = p_element_datatype.script_type;
|
|
|
|
if (p_element_datatype.kind == GDScriptParser::DataType::CLASS && script_type.is_null()) {
|
|
|
|
if (p_element_datatype.kind == GDScriptParser::DataType::CLASS && script_type.is_null()) {
|
|
|
|
Error err = OK;
|
|
|
|
Error err = OK;
|
|
|
|
Ref<GDScript> scr = GDScriptCache::get_shallow_script(p_element_datatype.script_path, err, parser->script_path);
|
|
|
|
Ref<GDScript> scr = get_depended_shallow_script(p_element_datatype.script_path, err);
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
push_error(vformat(R"(Error while getting cache for script "%s".)", p_element_datatype.script_path), p_source_node);
|
|
|
|
push_error(vformat(R"(Error while getting cache for script "%s".)", p_element_datatype.script_path), p_source_node);
|
|
|
|
return array;
|
|
|
|
return array;
|
|
|
|