name_component.cpp 3.4 KB

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