Reset unassigned local variables to null in the loops
This commit is contained in:
parent
ef090ee6ea
commit
8d2506e811
@ -1875,6 +1875,7 @@ Error GDScriptCompiler::_parse_block(CodeGen &codegen, const GDScriptParser::Sui
|
|||||||
GDScriptCodeGenerator::Address local = codegen.locals[lv->identifier->name];
|
GDScriptCodeGenerator::Address local = codegen.locals[lv->identifier->name];
|
||||||
GDScriptDataType local_type = _gdtype_from_datatype(lv->get_datatype(), codegen.script);
|
GDScriptDataType local_type = _gdtype_from_datatype(lv->get_datatype(), codegen.script);
|
||||||
|
|
||||||
|
bool initialized = false;
|
||||||
if (lv->initializer != nullptr) {
|
if (lv->initializer != nullptr) {
|
||||||
// For typed arrays we need to make sure this is already initialized correctly so typed assignment work.
|
// For typed arrays we need to make sure this is already initialized correctly so typed assignment work.
|
||||||
if (local_type.has_type && local_type.builtin_type == Variant::ARRAY) {
|
if (local_type.has_type && local_type.builtin_type == Variant::ARRAY) {
|
||||||
@ -1896,15 +1897,23 @@ Error GDScriptCompiler::_parse_block(CodeGen &codegen, const GDScriptParser::Sui
|
|||||||
if (src_address.mode == GDScriptCodeGenerator::Address::TEMPORARY) {
|
if (src_address.mode == GDScriptCodeGenerator::Address::TEMPORARY) {
|
||||||
codegen.generator->pop_temporary();
|
codegen.generator->pop_temporary();
|
||||||
}
|
}
|
||||||
|
initialized = true;
|
||||||
} else if (local_type.has_type) {
|
} else if (local_type.has_type) {
|
||||||
// Initialize with default for type.
|
// Initialize with default for type.
|
||||||
if (local_type.has_container_element_type()) {
|
if (local_type.has_container_element_type()) {
|
||||||
codegen.generator->write_construct_typed_array(local, local_type.get_container_element_type(), Vector<GDScriptCodeGenerator::Address>());
|
codegen.generator->write_construct_typed_array(local, local_type.get_container_element_type(), Vector<GDScriptCodeGenerator::Address>());
|
||||||
|
initialized = true;
|
||||||
} else if (local_type.kind == GDScriptDataType::BUILTIN) {
|
} else if (local_type.kind == GDScriptDataType::BUILTIN) {
|
||||||
codegen.generator->write_construct(local, local_type.builtin_type, Vector<GDScriptCodeGenerator::Address>());
|
codegen.generator->write_construct(local, local_type.builtin_type, Vector<GDScriptCodeGenerator::Address>());
|
||||||
|
initialized = true;
|
||||||
}
|
}
|
||||||
// The `else` branch is for objects, in such case we leave it as `null`.
|
// The `else` branch is for objects, in such case we leave it as `null`.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Assigns a null for the unassigned variables in loops.
|
||||||
|
if (!initialized && p_block->is_loop) {
|
||||||
|
codegen.generator->write_construct(local, Variant::NIL, Vector<GDScriptCodeGenerator::Address>());
|
||||||
|
}
|
||||||
} break;
|
} break;
|
||||||
case GDScriptParser::Node::CONSTANT: {
|
case GDScriptParser::Node::CONSTANT: {
|
||||||
// Local constants.
|
// Local constants.
|
||||||
|
@ -1835,9 +1835,9 @@ GDScriptParser::ForNode *GDScriptParser::parse_for() {
|
|||||||
}
|
}
|
||||||
suite->add_local(SuiteNode::Local(n_for->variable, current_function));
|
suite->add_local(SuiteNode::Local(n_for->variable, current_function));
|
||||||
}
|
}
|
||||||
suite->parent_for = n_for;
|
|
||||||
|
|
||||||
n_for->loop = parse_suite(R"("for" block)", suite);
|
n_for->loop = parse_suite(R"("for" block)", suite);
|
||||||
|
n_for->loop->is_loop = true;
|
||||||
complete_extents(n_for);
|
complete_extents(n_for);
|
||||||
|
|
||||||
// Reset break/continue state.
|
// Reset break/continue state.
|
||||||
@ -2169,6 +2169,7 @@ GDScriptParser::WhileNode *GDScriptParser::parse_while() {
|
|||||||
is_continue_match = false;
|
is_continue_match = false;
|
||||||
|
|
||||||
n_while->loop = parse_suite(R"("while" block)");
|
n_while->loop = parse_suite(R"("while" block)");
|
||||||
|
n_while->loop->is_loop = true;
|
||||||
complete_extents(n_while);
|
complete_extents(n_while);
|
||||||
|
|
||||||
// Reset break/continue state.
|
// Reset break/continue state.
|
||||||
|
@ -1055,12 +1055,12 @@ public:
|
|||||||
HashMap<StringName, int> locals_indices;
|
HashMap<StringName, int> locals_indices;
|
||||||
|
|
||||||
FunctionNode *parent_function = nullptr;
|
FunctionNode *parent_function = nullptr;
|
||||||
ForNode *parent_for = nullptr;
|
|
||||||
IfNode *parent_if = nullptr;
|
IfNode *parent_if = nullptr;
|
||||||
|
|
||||||
bool has_return = false;
|
bool has_return = false;
|
||||||
bool has_continue = false;
|
bool has_continue = false;
|
||||||
bool has_unreachable_code = false; // Just so warnings aren't given more than once per block.
|
bool has_unreachable_code = false; // Just so warnings aren't given more than once per block.
|
||||||
|
bool is_loop = false;
|
||||||
|
|
||||||
bool has_local(const StringName &p_name) const;
|
bool has_local(const StringName &p_name) const;
|
||||||
const Local &get_local(const StringName &p_name) const;
|
const Local &get_local(const StringName &p_name) const;
|
||||||
|
Loading…
Reference in New Issue
Block a user