/* valareturnstatement.vala * * Copyright (C) 2006-2010 Jürg Billeter * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * Author: * Jürg Billeter */ /** * Represents a return statement in the source code. */ public class Vala.ReturnStatement : CodeNode, Statement { /** * The optional expression to return. */ public Expression? return_expression { get { return _return_expression; } private set { _return_expression = value; if (_return_expression != null) { _return_expression.parent_node = this; } } } private Expression _return_expression; /** * Creates a new return statement. * * @param return_expression the return expression * @param source_reference reference to source code * @return newly created return statement */ public ReturnStatement (Expression? return_expression = null, SourceReference? source_reference = null) { this.source_reference = source_reference; this.return_expression = return_expression; } public override void accept (CodeVisitor visitor) { visitor.visit_return_statement (this); } public override void accept_children (CodeVisitor visitor) { if (return_expression != null) { return_expression.accept (visitor); visitor.visit_end_full_expression (return_expression); } } public override void replace_expression (Expression old_node, Expression new_node) { if (return_expression == old_node) { return_expression = new_node; } } public override void get_error_types (Collection collection, SourceReference? source_reference = null) { if (return_expression != null) { return_expression.get_error_types (collection, source_reference); } } public override bool check (CodeContext context) { if (checked) { return !error; } checked = true; if (context.analyzer.current_return_type == null) { error = true; Report.error (source_reference, "Return not allowed in this context"); return false; } if (return_expression == null) { if (!(context.analyzer.current_return_type is VoidType)) { error = true; Report.error (source_reference, "Return without value in non-void function"); } return !error; } if (context.analyzer.current_return_type is VoidType) { error = true; Report.error (source_reference, "Return with value in void function"); return false; } return_expression.target_type = context.analyzer.current_return_type.copy (); if (!return_expression.check (context)) { // ignore inner error error = true; return false; } if (return_expression.value_type == null) { error = true; Report.error (source_reference, "Invalid expression in return value"); return false; } if (!return_expression.value_type.compatible (context.analyzer.current_return_type)) { error = true; Report.error (source_reference, "Return: Cannot convert from `%s' to `%s'", return_expression.value_type.to_string (), context.analyzer.current_return_type.to_string ()); return false; } if (return_expression.value_type.is_disposable () && !context.analyzer.current_return_type.value_owned) { error = true; Report.error (source_reference, "Return value transfers ownership but method return type hasn't been declared to transfer ownership"); return false; } unowned LocalVariable? local = return_expression.symbol_reference as LocalVariable; if (local != null && local.variable_type.is_disposable () && !context.analyzer.current_return_type.value_owned) { error = true; Report.error (source_reference, "Local variable with strong reference used as return value and method return type has not been declared to transfer ownership"); return false; } if (return_expression is NullLiteral && !context.analyzer.current_return_type.nullable) { Report.warning (source_reference, "`null' incompatible with return type `%s'", context.analyzer.current_return_type.to_string ()); } return !error; } public override void emit (CodeGenerator codegen) { if (return_expression != null) { return_expression.emit (codegen); codegen.visit_end_full_expression (return_expression); } codegen.visit_return_statement (this); } public override void get_defined_variables (Collection collection) { if (return_expression != null) { return_expression.get_defined_variables (collection); } } public override void get_used_variables (Collection collection) { if (return_expression != null) { return_expression.get_used_variables (collection); } } }