handle_loop_statement.cpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "toolchain/check/context.h"
  5. #include "toolchain/check/convert.h"
  6. namespace Carbon::Check {
  7. auto HandleBreakStatement(Context& /*context*/,
  8. Parse::BreakStatementId /*parse_node*/) -> bool {
  9. return true;
  10. }
  11. auto HandleBreakStatementStart(Context& context,
  12. Parse::BreakStatementStartId parse_node)
  13. -> bool {
  14. auto& stack = context.break_continue_stack();
  15. if (stack.empty()) {
  16. CARBON_DIAGNOSTIC(BreakOutsideLoop, Error,
  17. "`break` can only be used in a loop.");
  18. context.emitter().Emit(parse_node, BreakOutsideLoop);
  19. } else {
  20. context.AddInst(SemIR::Branch{parse_node, stack.back().break_target});
  21. }
  22. context.inst_block_stack().Pop();
  23. context.inst_block_stack().PushUnreachable();
  24. return true;
  25. }
  26. auto HandleContinueStatement(Context& /*context*/,
  27. Parse::ContinueStatementId /*parse_node*/)
  28. -> bool {
  29. return true;
  30. }
  31. auto HandleContinueStatementStart(Context& context,
  32. Parse::ContinueStatementStartId parse_node)
  33. -> bool {
  34. auto& stack = context.break_continue_stack();
  35. if (stack.empty()) {
  36. CARBON_DIAGNOSTIC(ContinueOutsideLoop, Error,
  37. "`continue` can only be used in a loop.");
  38. context.emitter().Emit(parse_node, ContinueOutsideLoop);
  39. } else {
  40. context.AddInst(SemIR::Branch{parse_node, stack.back().continue_target});
  41. }
  42. context.inst_block_stack().Pop();
  43. context.inst_block_stack().PushUnreachable();
  44. return true;
  45. }
  46. auto HandleForHeader(Context& context, Parse::ForHeaderId parse_node) -> bool {
  47. return context.TODO(parse_node, "HandleForHeader");
  48. }
  49. auto HandleForHeaderStart(Context& context, Parse::ForHeaderStartId parse_node)
  50. -> bool {
  51. return context.TODO(parse_node, "HandleForHeaderStart");
  52. }
  53. auto HandleForIn(Context& context, Parse::ForInId parse_node) -> bool {
  54. context.decl_state_stack().Pop(DeclState::Var);
  55. return context.TODO(parse_node, "HandleForIn");
  56. }
  57. auto HandleForStatement(Context& context, Parse::ForStatementId parse_node)
  58. -> bool {
  59. return context.TODO(parse_node, "HandleForStatement");
  60. }
  61. auto HandleWhileConditionStart(Context& context,
  62. Parse::WhileConditionStartId parse_node)
  63. -> bool {
  64. // Branch to the loop header block. Note that we create a new block here even
  65. // if the current block is empty; this ensures that the loop always has a
  66. // preheader block.
  67. auto loop_header_id = context.AddDominatedBlockAndBranch(parse_node);
  68. context.inst_block_stack().Pop();
  69. // Start emitting the loop header block.
  70. context.inst_block_stack().Push(loop_header_id);
  71. context.AddCurrentCodeBlockToFunction();
  72. context.node_stack().Push(parse_node, loop_header_id);
  73. return true;
  74. }
  75. auto HandleWhileCondition(Context& context, Parse::WhileConditionId parse_node)
  76. -> bool {
  77. auto cond_value_id = context.node_stack().PopExpr();
  78. auto loop_header_id =
  79. context.node_stack().Peek<Parse::NodeKind::WhileConditionStart>();
  80. cond_value_id = ConvertToBoolValue(context, parse_node, cond_value_id);
  81. // Branch to either the loop body or the loop exit block.
  82. auto loop_body_id =
  83. context.AddDominatedBlockAndBranchIf(parse_node, cond_value_id);
  84. auto loop_exit_id = context.AddDominatedBlockAndBranch(parse_node);
  85. context.inst_block_stack().Pop();
  86. // Start emitting the loop body.
  87. context.inst_block_stack().Push(loop_body_id);
  88. context.AddCurrentCodeBlockToFunction();
  89. context.break_continue_stack().push_back(
  90. {.break_target = loop_exit_id, .continue_target = loop_header_id});
  91. context.node_stack().Push(parse_node, loop_exit_id);
  92. return true;
  93. }
  94. auto HandleWhileStatement(Context& context, Parse::WhileStatementId parse_node)
  95. -> bool {
  96. auto loop_exit_id =
  97. context.node_stack().Pop<Parse::NodeKind::WhileCondition>();
  98. auto loop_header_id =
  99. context.node_stack().Pop<Parse::NodeKind::WhileConditionStart>();
  100. context.break_continue_stack().pop_back();
  101. // Add the loop backedge.
  102. context.AddInst(SemIR::Branch{parse_node, loop_header_id});
  103. context.inst_block_stack().Pop();
  104. // Start emitting the loop exit block.
  105. context.inst_block_stack().Push(loop_exit_id);
  106. context.AddCurrentCodeBlockToFunction();
  107. return true;
  108. }
  109. } // namespace Carbon::Check