handle_brace_expression.cpp 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  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 HandleBraceExpression(Context& context) -> void {
  7. auto state = context.PopState();
  8. state.state = State::BraceExpressionFinishAsUnknown;
  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::BraceExpressionParameterAsUnknown);
  15. }
  16. }
  17. // Prints a diagnostic for brace expression syntax errors.
  18. static auto HandleBraceExpressionParameterError(Context& context,
  19. Context::StateStackEntry state,
  20. State param_finish_state)
  21. -> void {
  22. bool is_type =
  23. param_finish_state == State::BraceExpressionParameterFinishAsType;
  24. bool is_value =
  25. param_finish_state == State::BraceExpressionParameterFinishAsValue;
  26. bool is_unknown =
  27. param_finish_state == State::BraceExpressionParameterFinishAsUnknown;
  28. CARBON_CHECK(is_type || is_value || is_unknown);
  29. CARBON_DIAGNOSTIC(ExpectedStructLiteralField, Error, "Expected {0}{1}{2}.",
  30. llvm::StringRef, llvm::StringRef, llvm::StringRef);
  31. context.emitter().Emit(*context.position(), ExpectedStructLiteralField,
  32. (is_type || is_unknown) ? "`.field: field_type`" : "",
  33. is_unknown ? " or " : "",
  34. (is_value || is_unknown) ? "`.field = value`" : "");
  35. state.state = param_finish_state;
  36. state.has_error = true;
  37. context.PushState(state);
  38. }
  39. // Handles BraceExpressionParameterAs(Type|Value|Unknown).
  40. static auto HandleBraceExpressionParameter(Context& context,
  41. State after_designator_state,
  42. State param_finish_state) -> void {
  43. auto state = context.PopState();
  44. if (!context.PositionIs(Lex::TokenKind::Period)) {
  45. HandleBraceExpressionParameterError(context, state, param_finish_state);
  46. return;
  47. }
  48. state.state = after_designator_state;
  49. context.PushState(state);
  50. context.PushState(State::PeriodAsStruct);
  51. }
  52. auto HandleBraceExpressionParameterAsType(Context& context) -> void {
  53. HandleBraceExpressionParameter(
  54. context, State::BraceExpressionParameterAfterDesignatorAsType,
  55. State::BraceExpressionParameterFinishAsType);
  56. }
  57. auto HandleBraceExpressionParameterAsValue(Context& context) -> void {
  58. HandleBraceExpressionParameter(
  59. context, State::BraceExpressionParameterAfterDesignatorAsValue,
  60. State::BraceExpressionParameterFinishAsValue);
  61. }
  62. auto HandleBraceExpressionParameterAsUnknown(Context& context) -> void {
  63. HandleBraceExpressionParameter(
  64. context, State::BraceExpressionParameterAfterDesignatorAsUnknown,
  65. State::BraceExpressionParameterFinishAsUnknown);
  66. }
  67. // Handles BraceExpressionParameterAfterDesignatorAs(Type|Value|Unknown).
  68. static auto HandleBraceExpressionParameterAfterDesignator(
  69. Context& context, State param_finish_state) -> void {
  70. auto state = context.PopState();
  71. if (state.has_error) {
  72. auto recovery_pos = context.FindNextOf(
  73. {Lex::TokenKind::Equal, Lex::TokenKind::Colon, Lex::TokenKind::Comma});
  74. if (!recovery_pos ||
  75. context.tokens().GetKind(*recovery_pos) == Lex::TokenKind::Comma) {
  76. state.state = param_finish_state;
  77. context.PushState(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. HandleBraceExpressionParameterError(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::BraceExpressionParameterFinishAsUnknown) {
  94. auto finish_state = context.PopState();
  95. CARBON_CHECK(finish_state.state == State::BraceExpressionFinishAsUnknown);
  96. if (is_type) {
  97. finish_state.state = State::BraceExpressionFinishAsType;
  98. param_finish_state = State::BraceExpressionParameterFinishAsType;
  99. } else {
  100. finish_state.state = State::BraceExpressionFinishAsValue;
  101. param_finish_state = State::BraceExpressionParameterFinishAsValue;
  102. }
  103. context.PushState(finish_state);
  104. }
  105. auto want_param_finish_state =
  106. is_type ? State::BraceExpressionParameterFinishAsType
  107. : State::BraceExpressionParameterFinishAsValue;
  108. if (param_finish_state != want_param_finish_state) {
  109. HandleBraceExpressionParameterError(context, state, param_finish_state);
  110. return;
  111. }
  112. // Struct type fields and value fields use the same grammar except
  113. // that one has a `:` separator and the other has an `=` separator.
  114. state.state = param_finish_state;
  115. state.token = context.Consume();
  116. context.PushState(state);
  117. context.PushState(State::Expression);
  118. }
  119. auto HandleBraceExpressionParameterAfterDesignatorAsType(Context& context)
  120. -> void {
  121. HandleBraceExpressionParameterAfterDesignator(
  122. context, State::BraceExpressionParameterFinishAsType);
  123. }
  124. auto HandleBraceExpressionParameterAfterDesignatorAsValue(Context& context)
  125. -> void {
  126. HandleBraceExpressionParameterAfterDesignator(
  127. context, State::BraceExpressionParameterFinishAsValue);
  128. }
  129. auto HandleBraceExpressionParameterAfterDesignatorAsUnknown(Context& context)
  130. -> void {
  131. HandleBraceExpressionParameterAfterDesignator(
  132. context, State::BraceExpressionParameterFinishAsUnknown);
  133. }
  134. // Handles BraceExpressionParameterFinishAs(Type|Value|Unknown).
  135. static auto HandleBraceExpressionParameterFinish(Context& context,
  136. NodeKind node_kind,
  137. State param_state) -> void {
  138. auto state = context.PopState();
  139. if (state.has_error) {
  140. context.AddLeafNode(NodeKind::StructFieldUnknown, state.token,
  141. /*has_error=*/true);
  142. } else {
  143. context.AddNode(node_kind, state.token, state.subtree_start,
  144. /*has_error=*/false);
  145. }
  146. if (context.ConsumeListToken(
  147. NodeKind::StructComma, Lex::TokenKind::CloseCurlyBrace,
  148. state.has_error) == Context::ListTokenKind::Comma) {
  149. context.PushState(param_state);
  150. }
  151. }
  152. auto HandleBraceExpressionParameterFinishAsType(Context& context) -> void {
  153. HandleBraceExpressionParameterFinish(context, NodeKind::StructFieldType,
  154. State::BraceExpressionParameterAsType);
  155. }
  156. auto HandleBraceExpressionParameterFinishAsValue(Context& context) -> void {
  157. HandleBraceExpressionParameterFinish(context, NodeKind::StructFieldValue,
  158. State::BraceExpressionParameterAsValue);
  159. }
  160. auto HandleBraceExpressionParameterFinishAsUnknown(Context& context) -> void {
  161. HandleBraceExpressionParameterFinish(
  162. context, NodeKind::StructFieldUnknown,
  163. State::BraceExpressionParameterAsUnknown);
  164. }
  165. // Handles BraceExpressionFinishAs(Type|Value|Unknown).
  166. static auto HandleBraceExpressionFinish(Context& context, NodeKind node_kind)
  167. -> void {
  168. auto state = context.PopState();
  169. context.AddNode(node_kind, context.Consume(), state.subtree_start,
  170. state.has_error);
  171. }
  172. auto HandleBraceExpressionFinishAsType(Context& context) -> void {
  173. HandleBraceExpressionFinish(context, NodeKind::StructTypeLiteral);
  174. }
  175. auto HandleBraceExpressionFinishAsValue(Context& context) -> void {
  176. HandleBraceExpressionFinish(context, NodeKind::StructLiteral);
  177. }
  178. auto HandleBraceExpressionFinishAsUnknown(Context& context) -> void {
  179. HandleBraceExpressionFinish(context, NodeKind::StructLiteral);
  180. }
  181. } // namespace Carbon::Parse