Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Python-style list comprehensions to GDScript #81788

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
118 changes: 89 additions & 29 deletions modules/gdscript/gdscript_analyzer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1486,6 +1486,7 @@ void GDScriptAnalyzer::resolve_node(GDScriptParser::Node *p_node, bool p_is_root
case GDScriptParser::Node::TERNARY_OPERATOR:
case GDScriptParser::Node::TYPE_TEST:
case GDScriptParser::Node::UNARY_OPERATOR:
case GDScriptParser::Node::COMPREHENSION:
reduce_expression(static_cast<GDScriptParser::ExpressionNode *>(p_node), p_is_root);
break;
case GDScriptParser::Node::BREAK:
Expand Down Expand Up @@ -1990,13 +1991,16 @@ void GDScriptAnalyzer::resolve_if(GDScriptParser::IfNode *p_if) {
}
}

void GDScriptAnalyzer::resolve_for(GDScriptParser::ForNode *p_for) {
void GDScriptAnalyzer::resolve_iteration(GDScriptParser::ExpressionNode *list,
GDScriptParser::IdentifierNode *variable,
bool *use_conversion_assign,
GDScriptParser::TypeNode *datatype_specifier) {
Comment on lines +1994 to +1997
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void GDScriptAnalyzer::resolve_iteration(GDScriptParser::ExpressionNode *list,
GDScriptParser::IdentifierNode *variable,
bool *use_conversion_assign,
GDScriptParser::TypeNode *datatype_specifier) {
void GDScriptAnalyzer::resolve_iteration(GDScriptParser::ExpressionNode *p_list, GDScriptParser::IdentifierNode *p_variable,
bool *r_use_conversion_assign, GDScriptParser::TypeNode *r_datatype_specifier) {

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);
if (list && list->type == GDScriptParser::Node::CALL) {
GDScriptParser::CallNode *call = static_cast<GDScriptParser::CallNode *>(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);
Expand Down Expand Up @@ -2053,19 +2057,19 @@ void GDScriptAnalyzer::resolve_for(GDScriptParser::ForNode *p_for) {
reduced = Vector3i(args[0], args[1], args[2]);
break;
}
p_for->list->is_constant = true;
p_for->list->reduced_value = reduced;
list->is_constant = true;
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));
if (list->is_constant) {
list->set_datatype(type_from_variant(list->reduced_value, 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);
list->set_datatype(list_type);
}
}
}
Expand All @@ -2078,16 +2082,16 @@ void GDScriptAnalyzer::resolve_for(GDScriptParser::ForNode *p_for) {
variable_type.kind = GDScriptParser::DataType::BUILTIN;
variable_type.builtin_type = Variant::INT;
list_visible_type = "Array[int]"; // NOTE: `range()` has `Array` return type.
} else if (p_for->list) {
resolve_node(p_for->list, false);
GDScriptParser::DataType list_type = p_for->list->get_datatype();
} else if (list) {
resolve_node(list, false);
GDScriptParser::DataType list_type = list->get_datatype();
list_visible_type = list_type.to_string();
if (!list_type.is_hard_type()) {
mark_node_unsafe(p_for->list);
mark_node_unsafe(list);
}
if (list_type.is_variant()) {
variable_type.kind = GDScriptParser::DataType::VARIANT;
mark_node_unsafe(p_for->list);
mark_node_unsafe(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;
Expand All @@ -2111,49 +2115,53 @@ void GDScriptAnalyzer::resolve_for(GDScriptParser::ForNode *p_for) {
List<GDScriptParser::DataType> par_types;
int default_arg_count = 0;
BitField<MethodFlags> method_flags;
if (get_function_signature(p_for->list, false, list_type, CoreStringNames::get_singleton()->_iter_get, return_type, par_types, default_arg_count, method_flags)) {
if (get_function_signature(list, false, list_type, CoreStringNames::get_singleton()->_iter_get, return_type, par_types, default_arg_count, method_flags)) {
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);
push_error(vformat(R"(Unable to iterate on object of type "%s".)", list_type.to_string()), 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);
push_error(vformat(R"(Unable to iterate on value of type "%s".)", list_type.to_string()), list);
}
}

if (p_for->variable) {
if (p_for->datatype_specifier) {
GDScriptParser::DataType specified_type = type_from_metatype(resolve_datatype(p_for->datatype_specifier));
if (variable) {
if (datatype_specifier) {
GDScriptParser::DataType specified_type = type_from_metatype(resolve_datatype(datatype_specifier));
if (!specified_type.is_variant()) {
if (variable_type.is_variant() || !variable_type.is_hard_type()) {
mark_node_unsafe(p_for->variable);
p_for->use_conversion_assign = true;
} else if (!is_type_compatible(specified_type, variable_type, true, p_for->variable)) {
mark_node_unsafe(variable);
*use_conversion_assign = true;
} else if (!is_type_compatible(specified_type, variable_type, true, variable)) {
if (is_type_compatible(variable_type, specified_type)) {
mark_node_unsafe(p_for->variable);
p_for->use_conversion_assign = true;
mark_node_unsafe(variable);
*use_conversion_assign = true;
} else {
push_error(vformat(R"(Unable to iterate on value of type "%s" with variable of type "%s".)", list_visible_type, specified_type.to_string()), p_for->datatype_specifier);
push_error(vformat(R"(Unable to iterate on value of type "%s" with variable of type "%s".)", list_visible_type, specified_type.to_string()), datatype_specifier);
}
} else if (!is_type_compatible(specified_type, variable_type)) {
p_for->use_conversion_assign = true;
*use_conversion_assign = true;
}
}
p_for->variable->set_datatype(specified_type);
variable->set_datatype(specified_type);
} else {
p_for->variable->set_datatype(variable_type);
variable->set_datatype(variable_type);
#ifdef DEBUG_ENABLED
if (!variable_type.is_hard_type()) {
parser->push_warning(p_for->variable, GDScriptWarning::UNTYPED_DECLARATION, R"("for" iterator variable)", p_for->variable->name);
parser->push_warning(variable, GDScriptWarning::UNTYPED_DECLARATION, R"("for" iterator variable)", variable->name);
}
#endif
}
}
}

void GDScriptAnalyzer::resolve_for(GDScriptParser::ForNode *p_for) {
resolve_iteration(p_for->list, p_for->variable, &p_for->use_conversion_assign, p_for->datatype_specifier);

resolve_suite(p_for->loop);
p_for->set_datatype(p_for->loop->get_datatype());
Expand Down Expand Up @@ -2399,6 +2407,9 @@ void GDScriptAnalyzer::reduce_expression(GDScriptParser::ExpressionNode *p_expre
case GDScriptParser::Node::CAST:
reduce_cast(static_cast<GDScriptParser::CastNode *>(p_expression));
break;
case GDScriptParser::Node::COMPREHENSION:
reduce_comprehension(static_cast<GDScriptParser::ComprehensionNode *>(p_expression));
break;
case GDScriptParser::Node::DICTIONARY:
reduce_dictionary(static_cast<GDScriptParser::DictionaryNode *>(p_expression));
break;
Expand Down Expand Up @@ -2476,6 +2487,41 @@ void GDScriptAnalyzer::reduce_array(GDScriptParser::ArrayNode *p_array) {
p_array->set_datatype(arr_type);
}

void GDScriptAnalyzer::reduce_comprehension(GDScriptParser::ComprehensionNode *p_comprehension) {
for (int i = 0; i < p_comprehension->for_ifs.size(); i++) {
comprehension_identifier_stack.push_back(p_comprehension->for_ifs[i].variable);
}

for (int i = p_comprehension->for_ifs.size() - 1; i >= 0; i--) {
GDScriptParser::ComprehensionNode::ForIf for_if = p_comprehension->for_ifs[i];
comprehension_identifier_stack.pop_back();

reduce_expression(for_if.list);
resolve_iteration(for_if.list, for_if.variable, &for_if.use_conversion_assign);
}

for (int i = p_comprehension->for_ifs.size() - 1; i >= 0; i--) {
comprehension_identifier_stack.push_back(p_comprehension->for_ifs[i].variable);
}

reduce_expression(p_comprehension->expression);

for (int i = p_comprehension->for_ifs.size() - 1; i >= 0; i--) {
GDScriptParser::ComprehensionNode::ForIf for_if = p_comprehension->for_ifs[i];

reduce_expression(for_if.condition);

comprehension_identifier_stack.pop_back();
}

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_comprehension->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) {
Expand Down Expand Up @@ -3745,6 +3791,20 @@ void GDScriptAnalyzer::reduce_identifier(GDScriptParser::IdentifierNode *p_ident
}
}

// Check if we are inside a comprehension. This allows the comprehension to access the variable it is iterating over.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure that identifier is resolved correctly. We recently fixed a bug with scopes (see #79880). It is worth checking that the resolution here is consistent: a local identifier should not shadow class or global identifier in the list expression. Also, the implementation is probably a little over-complicated (but I'm not sure).

if (comprehension_identifier_stack.size() > 0) {
for (int i = comprehension_identifier_stack.size() - 1; i >= 0; i--) {
GDScriptParser::IdentifierNode *current_identifier = comprehension_identifier_stack[i];

if (current_identifier->name == p_identifier->name) {
GDScriptParser::DataType variable_data_type = current_identifier->get_datatype();
p_identifier->set_datatype(variable_data_type);
p_identifier->source = GDScriptParser::IdentifierNode::LOCAL_ITERATOR;
return;
}
}
}

bool found_source = false;
// Check if identifier is local.
// If that's the case, the declaration already was solved before.
Expand Down
4 changes: 4 additions & 0 deletions modules/gdscript/gdscript_analyzer.h
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ class GDScriptAnalyzer {
const GDScriptParser::EnumNode *current_enum = nullptr;
GDScriptParser::LambdaNode *current_lambda = nullptr;
List<GDScriptParser::LambdaNode *> pending_body_resolution_lambdas;
List<GDScriptParser::IdentifierNode *> comprehension_identifier_stack; //?: could this be a Vector instead?
bool static_context = false;

// Tests for detecting invalid overloading of script members
Expand All @@ -61,6 +62,8 @@ class GDScriptAnalyzer {

void decide_suite_type(GDScriptParser::Node *p_suite, GDScriptParser::Node *p_statement);

void resolve_iteration(GDScriptParser::ExpressionNode *list, GDScriptParser::IdentifierNode *variable, bool *use_conversion_assign, GDScriptParser::TypeNode *datatype_specifier = nullptr);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void resolve_iteration(GDScriptParser::ExpressionNode *list, GDScriptParser::IdentifierNode *variable, bool *use_conversion_assign, GDScriptParser::TypeNode *datatype_specifier = nullptr);
void resolve_iteration(GDScriptParser::ExpressionNode *p_list, GDScriptParser::IdentifierNode *p_variable, bool *r_use_conversion_assign, GDScriptParser::TypeNode *r_datatype_specifier = nullptr);


void resolve_annotation(GDScriptParser::AnnotationNode *p_annotation);
void resolve_class_member(GDScriptParser::ClassNode *p_class, StringName p_name, const GDScriptParser::Node *p_source = nullptr);
void resolve_class_member(GDScriptParser::ClassNode *p_class, int p_index, const GDScriptParser::Node *p_source = nullptr);
Expand Down Expand Up @@ -93,6 +96,7 @@ class GDScriptAnalyzer {
void reduce_binary_op(GDScriptParser::BinaryOpNode *p_binary_op);
void reduce_call(GDScriptParser::CallNode *p_call, bool p_is_await = false, bool p_is_root = false);
void reduce_cast(GDScriptParser::CastNode *p_cast);
void reduce_comprehension(GDScriptParser::ComprehensionNode *p_comprehension);
void reduce_dictionary(GDScriptParser::DictionaryNode *p_dictionary);
void reduce_get_node(GDScriptParser::GetNodeNode *p_get_node);
void reduce_identifier(GDScriptParser::IdentifierNode *p_identifier, bool can_be_builtin = false);
Expand Down
4 changes: 2 additions & 2 deletions modules/gdscript/gdscript_byte_codegen.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1783,8 +1783,8 @@ void GDScriptByteCodeGenerator::start_block() {
push_stack_identifiers();
}

void GDScriptByteCodeGenerator::end_block() {
pop_stack_identifiers();
void GDScriptByteCodeGenerator::end_block(bool expect_no_locals) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void GDScriptByteCodeGenerator::end_block(bool expect_no_locals) {
void GDScriptByteCodeGenerator::end_block(bool p_expect_no_locals) {

pop_stack_identifiers(expect_no_locals);
}

void GDScriptByteCodeGenerator::clean_temporaries() {
Expand Down
8 changes: 5 additions & 3 deletions modules/gdscript/gdscript_byte_codegen.h
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,9 @@ class GDScriptByteCodeGenerator : public GDScriptCodeGenerator {

Vector<StackSlot> locals;
Vector<StackSlot> temporaries;

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change

List<int> used_temporaries;

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change

List<int> temporaries_pending_clear;
RBMap<Variant::Type, List<int>> temporaries_pool;

Expand Down Expand Up @@ -184,13 +186,13 @@ class GDScriptByteCodeGenerator : public GDScriptCodeGenerator {
}
}

void pop_stack_identifiers() {
void pop_stack_identifiers(bool expect_no_locals = true) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void pop_stack_identifiers(bool expect_no_locals = true) {
void pop_stack_identifiers(bool p_expect_no_locals = true) {

int current_locals = stack_identifiers_counts.back()->get();
stack_identifiers_counts.pop_back();
stack_identifiers = stack_id_stack.back()->get();
stack_id_stack.pop_back();
#ifdef DEBUG_ENABLED
if (!used_temporaries.is_empty()) {
if (!used_temporaries.is_empty() && expect_no_locals) {
ERR_PRINT("Leaving block with non-zero temporary variables: " + itos(used_temporaries.size()));
}
#endif
Expand Down Expand Up @@ -468,7 +470,7 @@ class GDScriptByteCodeGenerator : public GDScriptCodeGenerator {
virtual void end_parameters() override;

virtual void start_block() override;
virtual void end_block() override;
virtual void end_block(bool expect_no_locals) override;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
virtual void end_block(bool expect_no_locals) override;
virtual void end_block(bool p_expect_no_locals) override;


virtual void write_start(GDScript *p_script, const StringName &p_function_name, bool p_static, Variant p_rpc_config, const GDScriptDataType &p_return_type) override;
virtual GDScriptFunction *write_end() override;
Expand Down
2 changes: 1 addition & 1 deletion modules/gdscript/gdscript_codegen.h
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ class GDScriptCodeGenerator {
virtual void end_parameters() = 0;

virtual void start_block() = 0;
virtual void end_block() = 0;
virtual void end_block(bool expect_no_locals) = 0;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
virtual void end_block(bool expect_no_locals) = 0;
virtual void end_block(bool p_expect_no_locals) = 0;


virtual void write_start(GDScript *p_script, const StringName &p_function_name, bool p_static, Variant p_rpc_config, const GDScriptDataType &p_return_type) = 0;
virtual GDScriptFunction *write_end() = 0;
Expand Down
63 changes: 63 additions & 0 deletions modules/gdscript/gdscript_compiler.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1373,6 +1373,69 @@ GDScriptCodeGenerator::Address GDScriptCompiler::_parse_expression(CodeGen &code

return result;
} break;
case GDScriptParser::Node::COMPREHENSION: {
const GDScriptParser::ComprehensionNode *comprehension = static_cast<const GDScriptParser::ComprehensionNode *>(p_expression);
GDScriptCodeGenerator::Address result = codegen.add_temporary(_gdtype_from_datatype(comprehension->get_datatype(), codegen.script));

Vector<GDScriptCodeGenerator::Address> values;

gen->write_construct_array(result, values);

for (const GDScriptParser::ComprehensionNode::ForIf &for_if : comprehension->for_ifs) {
codegen.start_block();

GDScriptCodeGenerator::Address list = _parse_expression(codegen, r_error, for_if.list);
if (r_error) {
return GDScriptCodeGenerator::Address();
}

GDScriptCodeGenerator::Address iterator = codegen.add_local(for_if.variable->name, _gdtype_from_datatype(for_if.variable->get_datatype(), codegen.script));

gen->start_for(iterator.type, _gdtype_from_datatype(for_if.list->get_datatype(), codegen.script));

gen->write_for_assignment(list);
if (list.mode == GDScriptCodeGenerator::Address::TEMPORARY) {
codegen.generator->pop_temporary();
}

gen->write_for(iterator, for_if.use_conversion_assign);

if (for_if.condition != nullptr) {
GDScriptCodeGenerator::Address condition = _parse_expression(codegen, r_error, for_if.condition);
if (r_error) {
return GDScriptCodeGenerator::Address();
}

gen->write_if(condition);

if (condition.mode == GDScriptCodeGenerator::Address::TEMPORARY) {
codegen.generator->pop_temporary();
}
}
}

GDScriptCodeGenerator::Address expression = _parse_expression(codegen, r_error, comprehension->expression);
if (r_error) {
return GDScriptCodeGenerator::Address();
}
values.push_back(expression);
if (expression.mode == GDScriptCodeGenerator::Address::TEMPORARY) {
codegen.generator->pop_temporary();
}

gen->write_call_builtin_type(GDScriptCodeGenerator::Address(), result, Variant::Type::ARRAY, "push_back", values);

for (int i = comprehension->for_ifs.size() - 1; i >= 0; i--) {
if (comprehension->for_ifs[i].condition != nullptr) {
gen->write_endif();
}
gen->write_endfor();

codegen.end_block(false);
}

return result;
} break;
default: {
ERR_FAIL_V_MSG(GDScriptCodeGenerator::Address(), "Bug in bytecode compiler, unexpected node in parse tree while parsing expression."); // Unreachable code.
} break;
Expand Down
4 changes: 2 additions & 2 deletions modules/gdscript/gdscript_compiler.h
Original file line number Diff line number Diff line change
Expand Up @@ -108,10 +108,10 @@ class GDScriptCompiler {
generator->start_block();
}

void end_block() {
void end_block(bool expect_no_locals = true) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void end_block(bool expect_no_locals = true) {
void end_block(bool p_expect_no_locals = true) {

locals = locals_stack.back()->get();
locals_stack.pop_back();
generator->end_block();
generator->end_block(expect_no_locals);
}
};

Expand Down
Loading