name_component.cpp 3.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  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/check/name_component.h"
  5. #include "toolchain/check/context.h"
  6. #include "toolchain/check/pattern_match.h"
  7. namespace Carbon::Check {
  8. auto PopNameComponent(Context& context) -> NameComponent {
  9. Parse::NodeId first_param_node_id = Parse::InvalidNodeId();
  10. Parse::NodeId last_param_node_id = Parse::InvalidNodeId();
  11. // Explicit params.
  12. auto [params_loc_id, param_patterns_id] =
  13. context.node_stack().PopWithNodeIdIf<Parse::NodeKind::TuplePattern>();
  14. if (param_patterns_id) {
  15. first_param_node_id =
  16. context.node_stack()
  17. .PopForSoloNodeId<Parse::NodeKind::TuplePatternStart>();
  18. last_param_node_id = params_loc_id;
  19. } else {
  20. param_patterns_id = SemIR::InstBlockId::Invalid;
  21. }
  22. // Implicit params.
  23. auto [implicit_params_loc_id, implicit_param_patterns_id] =
  24. context.node_stack()
  25. .PopWithNodeIdIf<Parse::NodeKind::ImplicitParamList>();
  26. if (implicit_param_patterns_id) {
  27. // Implicit params always come before explicit params.
  28. first_param_node_id =
  29. context.node_stack()
  30. .PopForSoloNodeId<Parse::NodeKind::ImplicitParamListStart>();
  31. // Only use the end of implicit params if there weren't explicit params.
  32. if (last_param_node_id.is_valid()) {
  33. last_param_node_id = params_loc_id;
  34. }
  35. } else {
  36. implicit_param_patterns_id = SemIR::InstBlockId::Invalid;
  37. }
  38. auto [implicit_params_id, params_id] = CalleePatternMatch(
  39. context, *implicit_param_patterns_id, *param_patterns_id);
  40. auto [name_loc_id, name_id] = context.node_stack().PopNameWithNodeId();
  41. return {
  42. .name_loc_id = name_loc_id,
  43. .name_id = name_id,
  44. .first_param_node_id = first_param_node_id,
  45. .last_param_node_id = last_param_node_id,
  46. .implicit_params_loc_id = implicit_params_loc_id,
  47. .implicit_params_id = implicit_params_id,
  48. .implicit_param_patterns_id = *implicit_param_patterns_id,
  49. .params_loc_id = params_loc_id,
  50. .params_id = params_id,
  51. .param_patterns_id = *param_patterns_id,
  52. .pattern_block_id = context.pattern_block_stack().Pop(),
  53. };
  54. }
  55. // Pop the name of a declaration from the node stack, and diagnose if it has
  56. // parameters.
  57. auto PopNameComponentWithoutParams(Context& context, Lex::TokenKind introducer)
  58. -> NameComponent {
  59. NameComponent name = PopNameComponent(context);
  60. if (name.implicit_params_id.is_valid() || name.params_id.is_valid()) {
  61. CARBON_DIAGNOSTIC(UnexpectedDeclNameParams, Error,
  62. "`{0}` declaration cannot have parameters",
  63. Lex::TokenKind);
  64. // Point to the lexically first parameter list in the diagnostic.
  65. context.emitter().Emit(name.implicit_params_id.is_valid()
  66. ? name.implicit_params_loc_id
  67. : name.params_loc_id,
  68. UnexpectedDeclNameParams, introducer);
  69. name.implicit_params_id = SemIR::InstBlockId::Invalid;
  70. name.params_id = SemIR::InstBlockId::Invalid;
  71. }
  72. return name;
  73. }
  74. } // namespace Carbon::Check