| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283 |
- // 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/semantics/semantics_context.h"
- #include "toolchain/semantics/semantics_node.h"
- namespace Carbon {
- auto SemanticsHandleExpressionStatement(SemanticsContext& context,
- ParseTree::Node /*parse_node*/)
- -> bool {
- // Pop the expression without investigating its contents.
- // TODO: This will probably eventually need to do some "do not discard"
- // analysis.
- context.node_stack().PopAndDiscardId();
- return true;
- }
- auto SemanticsHandleReturnStatement(SemanticsContext& context,
- ParseTree::Node parse_node) -> bool {
- CARBON_CHECK(!context.return_scope_stack().empty());
- const auto& fn_node =
- context.semantics_ir().GetNode(context.return_scope_stack().back());
- const auto& callable =
- context.semantics_ir().GetFunction(fn_node.GetAsFunctionDeclaration());
- if (context.parse_tree().node_kind(context.node_stack().PeekParseNode()) ==
- ParseNodeKind::ReturnStatementStart) {
- context.node_stack().PopAndDiscardSoloParseNode(
- ParseNodeKind::ReturnStatementStart);
- if (callable.return_type_id.is_valid()) {
- // TODO: Add a note pointing at the return type's parse node.
- CARBON_DIAGNOSTIC(ReturnStatementMissingExpression, Error,
- "Must return a {0}.", std::string);
- context.emitter()
- .Build(parse_node, ReturnStatementMissingExpression,
- context.semantics_ir().StringifyType(callable.return_type_id))
- .Emit();
- }
- context.AddNode(SemanticsNode::Return::Make(parse_node));
- } else {
- auto arg = context.node_stack().Pop<SemanticsNodeId>();
- context.node_stack().PopAndDiscardSoloParseNode(
- ParseNodeKind::ReturnStatementStart);
- if (!callable.return_type_id.is_valid()) {
- CARBON_DIAGNOSTIC(
- ReturnStatementDisallowExpression, 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, ReturnStatementDisallowExpression)
- .Note(fn_node.parse_node(), ReturnStatementImplicitNote)
- .Emit();
- } else {
- arg =
- context.ImplicitAsRequired(parse_node, arg, callable.return_type_id);
- }
- context.AddNode(SemanticsNode::ReturnExpression::Make(
- parse_node, context.semantics_ir().GetNode(arg).type_id(), arg));
- }
- // Switch to a new, unreachable, empty node block. This typically won't
- // contain any semantics IR, but it can do if there are statements following
- // the `return` statement.
- context.node_block_stack().Pop();
- context.node_block_stack().PushUnreachable();
- return true;
- }
- auto SemanticsHandleReturnStatementStart(SemanticsContext& context,
- ParseTree::Node parse_node) -> bool {
- // No action, just a bracketing node.
- context.node_stack().Push(parse_node);
- return true;
- }
- } // namespace Carbon
|