handle_brace_expr.cpp 7.4 KB

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