handle_brace_expr.cpp 7.6 KB

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