| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "common/map.h"
- #include "toolchain/check/context.h"
- #include "toolchain/check/convert.h"
- #include "toolchain/check/handle.h"
- #include "toolchain/diagnostics/format_providers.h"
- namespace Carbon::Check {
- auto HandleParseNode(Context& context, Parse::StructTypeLiteralStartId node_id)
- -> bool {
- context.scope_stack().Push();
- context.node_stack().Push(node_id);
- context.param_and_arg_refs_stack().Push();
- return true;
- }
- auto HandleParseNode(Context& context, Parse::StructLiteralStartId node_id)
- -> bool {
- context.scope_stack().Push();
- context.node_stack().Push(node_id);
- context.args_type_info_stack().Push();
- context.param_and_arg_refs_stack().Push();
- return true;
- }
- auto HandleParseNode(Context& context,
- Parse::StructFieldDesignatorId /*node_id*/) -> bool {
- // This leaves the designated name on top because the `.` isn't interesting.
- CARBON_CHECK(context.node_stack().PeekIs<SemIR::NameId>());
- return true;
- }
- auto HandleParseNode(Context& context, Parse::StructCommaId /*node_id*/)
- -> bool {
- context.param_and_arg_refs_stack().ApplyComma();
- return true;
- }
- auto HandleParseNode(Context& context, Parse::StructFieldId node_id) -> bool {
- auto value_inst_id = context.node_stack().PopExpr();
- auto [name_node, name_id] = context.node_stack().PopNameWithNodeId();
- // Store the name for the type.
- context.args_type_info_stack().AddInstId(
- context.AddInstInNoBlock<SemIR::StructTypeField>(
- name_node,
- {.name_id = name_id,
- .field_type_id = context.insts().Get(value_inst_id).type_id()}));
- // Push the value back on the stack as an argument.
- context.node_stack().Push(node_id, value_inst_id);
- return true;
- }
- auto HandleParseNode(Context& context, Parse::StructTypeFieldId node_id)
- -> bool {
- auto [type_node, type_id] = context.node_stack().PopExprWithNodeId();
- SemIR::TypeId cast_type_id = ExprAsType(context, type_node, type_id).type_id;
- auto [name_node, name_id] = context.node_stack().PopNameWithNodeId();
- auto inst_id = context.AddInst<SemIR::StructTypeField>(
- name_node, {.name_id = name_id, .field_type_id = cast_type_id});
- context.node_stack().Push(node_id, inst_id);
- return true;
- }
- static auto DiagnoseDuplicateNames(Context& context,
- SemIR::InstBlockId type_block_id,
- bool is_struct_type_literal) -> bool {
- auto& sem_ir = context.sem_ir();
- auto fields = sem_ir.inst_blocks().Get(type_block_id);
- Map<SemIR::NameId, SemIR::InstId> names;
- auto& insts = sem_ir.insts();
- for (SemIR::InstId field_inst_id : fields) {
- auto field_inst = insts.GetAs<SemIR::StructTypeField>(field_inst_id);
- auto result = names.Insert(field_inst.name_id, field_inst_id);
- if (!result.is_inserted()) {
- CARBON_DIAGNOSTIC(StructNameDuplicate, Error,
- "duplicated field name `{1}` in "
- "{0:struct type literal|struct literal}",
- BoolAsSelect, SemIR::NameId);
- CARBON_DIAGNOSTIC(StructNamePrevious, Note,
- "field with the same name here");
- context.emitter()
- .Build(field_inst_id, StructNameDuplicate, is_struct_type_literal,
- field_inst.name_id)
- .Note(result.value(), StructNamePrevious)
- .Emit();
- return true;
- }
- }
- return false;
- }
- auto HandleParseNode(Context& context, Parse::StructLiteralId node_id) -> bool {
- auto refs_id = context.param_and_arg_refs_stack().EndAndPop(
- Parse::NodeKind::StructLiteralStart);
- context.scope_stack().Pop();
- context.node_stack()
- .PopAndDiscardSoloNodeId<Parse::NodeKind::StructLiteralStart>();
- auto type_block_id = context.args_type_info_stack().Pop();
- if (DiagnoseDuplicateNames(context, type_block_id,
- /*is_struct_type_literal=*/false)) {
- context.node_stack().Push(node_id, SemIR::InstId::BuiltinError);
- return true;
- }
- auto type_id = context.GetStructType(type_block_id);
- auto value_id = context.AddInst<SemIR::StructLiteral>(
- node_id, {.type_id = type_id, .elements_id = refs_id});
- context.node_stack().Push(node_id, value_id);
- return true;
- }
- auto HandleParseNode(Context& context, Parse::StructTypeLiteralId node_id)
- -> bool {
- auto refs_id = context.param_and_arg_refs_stack().EndAndPop(
- Parse::NodeKind::StructTypeLiteralStart);
- context.scope_stack().Pop();
- context.node_stack()
- .PopAndDiscardSoloNodeId<Parse::NodeKind::StructTypeLiteralStart>();
- CARBON_CHECK(refs_id != SemIR::InstBlockId::Empty,
- "{{}} is handled by StructLiteral.");
- if (DiagnoseDuplicateNames(context, refs_id,
- /*is_struct_type_literal=*/true)) {
- context.node_stack().Push(node_id, SemIR::InstId::BuiltinError);
- return true;
- }
- context.AddInstAndPush<SemIR::StructType>(
- node_id, {.type_id = SemIR::TypeId::TypeType, .fields_id = refs_id});
- return true;
- }
- } // namespace Carbon::Check
|