handle_brace_expr.cpp 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  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/parse/context.h"
  5. namespace Carbon::Parse {
  6. auto HandleBraceExpr(Context& context) -> void {
  7. auto state = context.PopState();
  8. context.PushState(state, State::BraceExprFinishAsUnknown);
  9. CARBON_CHECK(context.ConsumeAndAddLeafNodeIf(Lex::TokenKind::OpenCurlyBrace,
  10. NodeKind::Placeholder));
  11. if (!context.PositionIs(Lex::TokenKind::CloseCurlyBrace)) {
  12. context.PushState(State::BraceExprParamAsUnknown);
  13. }
  14. }
  15. // Prints a diagnostic for brace expression syntax errors.
  16. static auto HandleBraceExprParamError(Context& context,
  17. Context::StateStackEntry state,
  18. State param_finish_state) -> void {
  19. bool is_type = param_finish_state == State::BraceExprParamFinishAsType;
  20. bool is_value = param_finish_state == State::BraceExprParamFinishAsValue;
  21. bool is_unknown = param_finish_state == State::BraceExprParamFinishAsUnknown;
  22. CARBON_CHECK(is_type || is_value || is_unknown);
  23. CARBON_DIAGNOSTIC(ExpectedStructLiteralField, Error, "Expected {0}{1}{2}.",
  24. llvm::StringLiteral, llvm::StringLiteral,
  25. llvm::StringLiteral);
  26. context.emitter().Emit(
  27. *context.position(), ExpectedStructLiteralField,
  28. (is_type || is_unknown) ? llvm::StringLiteral("`.field: field_type`")
  29. : llvm::StringLiteral(""),
  30. is_unknown ? llvm::StringLiteral(" or ") : llvm::StringLiteral(""),
  31. (is_value || is_unknown) ? llvm::StringLiteral("`.field = value`")
  32. : llvm::StringLiteral(""));
  33. state.has_error = true;
  34. context.PushState(state, param_finish_state);
  35. }
  36. // Handles BraceExprParamAs(Type|Value|Unknown).
  37. static auto HandleBraceExprParam(Context& context, State after_designator_state,
  38. State param_finish_state) -> void {
  39. auto state = context.PopState();
  40. if (!context.PositionIs(Lex::TokenKind::Period)) {
  41. HandleBraceExprParamError(context, state, param_finish_state);
  42. return;
  43. }
  44. context.PushState(state, after_designator_state);
  45. context.PushState(State::PeriodAsStruct);
  46. }
  47. auto HandleBraceExprParamAsType(Context& context) -> void {
  48. HandleBraceExprParam(context, State::BraceExprParamAfterDesignatorAsType,
  49. State::BraceExprParamFinishAsType);
  50. }
  51. auto HandleBraceExprParamAsValue(Context& context) -> void {
  52. HandleBraceExprParam(context, State::BraceExprParamAfterDesignatorAsValue,
  53. State::BraceExprParamFinishAsValue);
  54. }
  55. auto HandleBraceExprParamAsUnknown(Context& context) -> void {
  56. HandleBraceExprParam(context, State::BraceExprParamAfterDesignatorAsUnknown,
  57. State::BraceExprParamFinishAsUnknown);
  58. }
  59. // Handles BraceExprParamAfterDesignatorAs(Type|Value|Unknown).
  60. static auto HandleBraceExprParamAfterDesignator(Context& context,
  61. State param_finish_state)
  62. -> void {
  63. auto state = context.PopState();
  64. if (state.has_error) {
  65. auto recovery_pos = context.FindNextOf(
  66. {Lex::TokenKind::Equal, Lex::TokenKind::Colon, Lex::TokenKind::Comma});
  67. if (!recovery_pos ||
  68. context.tokens().GetKind(*recovery_pos) == Lex::TokenKind::Comma) {
  69. context.PushState(state, param_finish_state);
  70. return;
  71. }
  72. context.SkipTo(*recovery_pos);
  73. }
  74. // Work out the kind of this element.
  75. bool is_type;
  76. if (context.PositionIs(Lex::TokenKind::Colon)) {
  77. is_type = true;
  78. } else if (context.PositionIs(Lex::TokenKind::Equal)) {
  79. is_type = false;
  80. } else {
  81. HandleBraceExprParamError(context, state, param_finish_state);
  82. return;
  83. }
  84. // If we're changing from unknown, update the related finish states.
  85. if (param_finish_state == State::BraceExprParamFinishAsUnknown) {
  86. auto finish_state = context.PopState();
  87. CARBON_CHECK(finish_state.state == State::BraceExprFinishAsUnknown);
  88. if (is_type) {
  89. finish_state.state = State::BraceExprFinishAsType;
  90. param_finish_state = State::BraceExprParamFinishAsType;
  91. } else {
  92. finish_state.state = State::BraceExprFinishAsValue;
  93. param_finish_state = State::BraceExprParamFinishAsValue;
  94. }
  95. context.PushState(finish_state);
  96. }
  97. auto want_param_finish_state = is_type ? State::BraceExprParamFinishAsType
  98. : State::BraceExprParamFinishAsValue;
  99. if (param_finish_state != want_param_finish_state) {
  100. HandleBraceExprParamError(context, state, param_finish_state);
  101. return;
  102. }
  103. // Struct type fields and value fields use the same grammar except
  104. // that one has a `:` separator and the other has an `=` separator.
  105. state.token = context.Consume();
  106. context.PushState(state, param_finish_state);
  107. context.PushState(State::Expr);
  108. }
  109. auto HandleBraceExprParamAfterDesignatorAsType(Context& context) -> void {
  110. HandleBraceExprParamAfterDesignator(context,
  111. State::BraceExprParamFinishAsType);
  112. }
  113. auto HandleBraceExprParamAfterDesignatorAsValue(Context& context) -> void {
  114. HandleBraceExprParamAfterDesignator(context,
  115. State::BraceExprParamFinishAsValue);
  116. }
  117. auto HandleBraceExprParamAfterDesignatorAsUnknown(Context& context) -> void {
  118. HandleBraceExprParamAfterDesignator(context,
  119. State::BraceExprParamFinishAsUnknown);
  120. }
  121. // Handles BraceExprParamFinishAs(Type|Value|Unknown).
  122. static auto HandleBraceExprParamFinish(Context& context, NodeKind node_kind,
  123. State param_state) -> void {
  124. auto state = context.PopState();
  125. if (state.has_error) {
  126. context.AddLeafNode(NodeKind::InvalidParse, state.token,
  127. /*has_error=*/true);
  128. context.ReturnErrorOnState();
  129. } else {
  130. context.AddNode(node_kind, state.token, state.subtree_start,
  131. /*has_error=*/false);
  132. }
  133. if (context.ConsumeListToken(
  134. NodeKind::StructComma, Lex::TokenKind::CloseCurlyBrace,
  135. state.has_error) == Context::ListTokenKind::Comma) {
  136. context.PushState(param_state);
  137. }
  138. }
  139. auto HandleBraceExprParamFinishAsType(Context& context) -> void {
  140. HandleBraceExprParamFinish(context, NodeKind::StructTypeField,
  141. State::BraceExprParamAsType);
  142. }
  143. auto HandleBraceExprParamFinishAsValue(Context& context) -> void {
  144. HandleBraceExprParamFinish(context, NodeKind::StructField,
  145. State::BraceExprParamAsValue);
  146. }
  147. auto HandleBraceExprParamFinishAsUnknown(Context& context) -> void {
  148. HandleBraceExprParamFinish(context, NodeKind::InvalidParse,
  149. State::BraceExprParamAsUnknown);
  150. }
  151. // Handles BraceExprFinishAs(Type|Value|Unknown).
  152. static auto HandleBraceExprFinish(Context& context, NodeKind start_kind,
  153. NodeKind end_kind) -> void {
  154. auto state = context.PopState();
  155. context.ReplacePlaceholderNode(state.subtree_start, start_kind, state.token);
  156. context.AddNode(end_kind, context.Consume(), state.subtree_start,
  157. state.has_error);
  158. }
  159. auto HandleBraceExprFinishAsType(Context& context) -> void {
  160. HandleBraceExprFinish(context, NodeKind::StructTypeLiteralStart,
  161. NodeKind::StructTypeLiteral);
  162. }
  163. auto HandleBraceExprFinishAsValue(Context& context) -> void {
  164. HandleBraceExprFinish(context, NodeKind::StructLiteralStart,
  165. NodeKind::StructLiteral);
  166. }
  167. auto HandleBraceExprFinishAsUnknown(Context& context) -> void {
  168. HandleBraceExprFinishAsValue(context);
  169. }
  170. } // namespace Carbon::Parse