| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "toolchain/check/context.h"
- #include "toolchain/check/convert.h"
- #include "toolchain/sem_ir/inst.h"
- namespace Carbon::Check {
- // TODO: Find a better home for this. We'll likely need it for more than just
- // expression statements.
- static auto HandleDiscardedExpr(Context& context, SemIR::InstId expr_id)
- -> void {
- // If we discard an initializing expression, convert it to a value or
- // reference so that it has something to initialize.
- auto expr = context.insts().Get(expr_id);
- Convert(context, expr.parse_node(), expr_id,
- {.kind = ConversionTarget::Discarded, .type_id = expr.type_id()});
- // TODO: This will eventually need to do some "do not discard" analysis.
- }
- auto HandleExprStatement(Context& context, Parse::Node /*parse_node*/) -> bool {
- HandleDiscardedExpr(context, context.node_stack().PopExpr());
- return true;
- }
- auto HandleReturnStatement(Context& context, Parse::Node parse_node) -> bool {
- CARBON_CHECK(!context.return_scope_stack().empty());
- auto fn_inst = context.insts().GetAs<SemIR::FunctionDecl>(
- context.return_scope_stack().back());
- const auto& callable = context.functions().Get(fn_inst.function_id);
- if (context.parse_tree().node_kind(context.node_stack().PeekParseNode()) ==
- Parse::NodeKind::ReturnStatementStart) {
- context.node_stack()
- .PopAndDiscardSoloParseNode<Parse::NodeKind::ReturnStatementStart>();
- if (callable.return_type_id.is_valid()) {
- // TODO: Add a note pointing at the return type's parse node.
- CARBON_DIAGNOSTIC(ReturnStatementMissingExpr, Error, "Must return a {0}.",
- std::string);
- context.emitter()
- .Build(parse_node, ReturnStatementMissingExpr,
- context.sem_ir().StringifyType(callable.return_type_id))
- .Emit();
- }
- context.AddInst(SemIR::Return{parse_node});
- } else {
- auto arg = context.node_stack().PopExpr();
- context.node_stack()
- .PopAndDiscardSoloParseNode<Parse::NodeKind::ReturnStatementStart>();
- if (!callable.return_type_id.is_valid()) {
- CARBON_DIAGNOSTIC(
- ReturnStatementDisallowExpr, Error,
- "No return expression should be provided in this context.");
- CARBON_DIAGNOSTIC(ReturnStatementImplicitNote, Note,
- "There was no return type provided.");
- context.emitter()
- .Build(parse_node, ReturnStatementDisallowExpr)
- .Note(fn_inst.parse_node, ReturnStatementImplicitNote)
- .Emit();
- } else if (callable.return_slot_id.is_valid()) {
- arg = Initialize(context, parse_node, callable.return_slot_id, arg);
- } else {
- arg = ConvertToValueOfType(context, parse_node, arg,
- callable.return_type_id);
- }
- context.AddInst(SemIR::ReturnExpr{parse_node, arg});
- }
- // Switch to a new, unreachable, empty instruction block. This typically won't
- // contain any semantics IR, but it can do if there are statements following
- // the `return` statement.
- context.inst_block_stack().Pop();
- context.inst_block_stack().PushUnreachable();
- return true;
- }
- auto HandleReturnStatementStart(Context& context, Parse::Node parse_node)
- -> bool {
- // No action, just a bracketing node.
- context.node_stack().Push(parse_node);
- return true;
- }
- } // namespace Carbon::Check
|