handle_brace_expr.cpp 7.7 KB

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