handle_brace_expr.cpp 7.5 KB

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