diff options
| author | Andrew Kelley <superjoe30@gmail.com> | 2017-11-16 22:14:50 -0500 |
|---|---|---|
| committer | Andrew Kelley <superjoe30@gmail.com> | 2017-11-16 22:14:50 -0500 |
| commit | d1086893829d91d0e1335e14b0434a23ff0e8999 (patch) | |
| tree | 17e55b50e624ae06a93522a38aba3c0e83edf37e /src/analyze.cpp | |
| parent | 7a74dbadd79d2b26a449027dd83753ae4d2a8032 (diff) | |
| parent | 1473eb9ae0196db729716aa0d29f5ce263412307 (diff) | |
| download | zig-d1086893829d91d0e1335e14b0434a23ff0e8999.tar.gz zig-d1086893829d91d0e1335e14b0434a23ff0e8999.zip | |
Merge branch 'unions'
closes #144
Diffstat (limited to 'src/analyze.cpp')
| -rw-r--r-- | src/analyze.cpp | 337 |
1 files changed, 331 insertions, 6 deletions
diff --git a/src/analyze.cpp b/src/analyze.cpp index 343b1ecdb0..ebad9fe0cb 100644 --- a/src/analyze.cpp +++ b/src/analyze.cpp @@ -1008,11 +1008,12 @@ TypeTableEntry *get_partial_container_type(CodeGen *g, Scope *scope, ContainerKi } size_t line = decl_node ? decl_node->line : 0; + unsigned dwarf_kind = ZigLLVMTag_DW_structure_type(); ImportTableEntry *import = get_scope_import(scope); entry->type_ref = LLVMStructCreateNamed(LLVMGetGlobalContext(), name); entry->di_type = ZigLLVMCreateReplaceableCompositeType(g->dbuilder, - ZigLLVMTag_DW_structure_type(), name, + dwarf_kind, name, ZigLLVMFileToScope(import->di_file), import->di_file, (unsigned)(line + 1)); buf_init_from_str(&entry->name, name); @@ -1285,7 +1286,7 @@ static void resolve_enum_type(CodeGen *g, TypeTableEntry *enum_type) { return; resolve_enum_zero_bits(g, enum_type); - if (enum_type->data.enumeration.is_invalid) + if (type_is_invalid(enum_type)) return; AstNode *decl_node = enum_type->data.enumeration.decl_node; @@ -1834,7 +1835,246 @@ static void resolve_struct_type(CodeGen *g, TypeTableEntry *struct_type) { } static void resolve_union_type(CodeGen *g, TypeTableEntry *union_type) { - zig_panic("TODO"); + assert(union_type->id == TypeTableEntryIdUnion); + + if (union_type->data.unionation.complete) + return; + + resolve_union_zero_bits(g, union_type); + if (type_is_invalid(union_type)) + return; + + AstNode *decl_node = union_type->data.unionation.decl_node; + + if (union_type->data.unionation.embedded_in_current) { + if (!union_type->data.unionation.reported_infinite_err) { + union_type->data.unionation.reported_infinite_err = true; + add_node_error(g, decl_node, buf_sprintf("union '%s' contains itself", buf_ptr(&union_type->name))); + } + return; + } + + assert(!union_type->data.unionation.zero_bits_loop_flag); + assert(decl_node->type == NodeTypeContainerDecl); + assert(union_type->di_type); + + uint32_t field_count = union_type->data.unionation.src_field_count; + + assert(union_type->data.unionation.fields); + + uint32_t gen_field_count = union_type->data.unionation.gen_field_count; + ZigLLVMDIType **union_inner_di_types = allocate<ZigLLVMDIType*>(gen_field_count); + + TypeTableEntry *most_aligned_union_member = nullptr; + uint64_t size_of_most_aligned_member_in_bits = 0; + uint64_t biggest_align_in_bits = 0; + uint64_t biggest_size_in_bits = 0; + + bool auto_layout = (union_type->data.unionation.layout == ContainerLayoutAuto); + ZigLLVMDIEnumerator **di_enumerators = allocate<ZigLLVMDIEnumerator*>(field_count); + + Scope *scope = &union_type->data.unionation.decls_scope->base; + ImportTableEntry *import = get_scope_import(scope); + + // set temporary flag + union_type->data.unionation.embedded_in_current = true; + + for (uint32_t i = 0; i < field_count; i += 1) { + AstNode *field_node = decl_node->data.container_decl.fields.at(i); + TypeUnionField *type_union_field = &union_type->data.unionation.fields[i]; + TypeTableEntry *field_type = type_union_field->type_entry; + + ensure_complete_type(g, field_type); + if (type_is_invalid(field_type)) { + union_type->data.unionation.is_invalid = true; + continue; + } + + if (!type_has_bits(field_type)) + continue; + + di_enumerators[i] = ZigLLVMCreateDebugEnumerator(g->dbuilder, buf_ptr(type_union_field->name), i); + + uint64_t store_size_in_bits = 8*LLVMStoreSizeOfType(g->target_data_ref, field_type->type_ref); + uint64_t abi_align_in_bits = 8*LLVMABIAlignmentOfType(g->target_data_ref, field_type->type_ref); + + assert(store_size_in_bits > 0); + assert(abi_align_in_bits > 0); + + union_inner_di_types[type_union_field->gen_index] = ZigLLVMCreateDebugMemberType(g->dbuilder, + ZigLLVMTypeToScope(union_type->di_type), buf_ptr(type_union_field->name), + import->di_file, (unsigned)(field_node->line + 1), + store_size_in_bits, + abi_align_in_bits, + 0, + 0, field_type->di_type); + + biggest_size_in_bits = max(biggest_size_in_bits, store_size_in_bits); + + if (!most_aligned_union_member || abi_align_in_bits > biggest_align_in_bits) { + most_aligned_union_member = field_type; + biggest_align_in_bits = abi_align_in_bits; + size_of_most_aligned_member_in_bits = store_size_in_bits; + } + } + + // unset temporary flag + union_type->data.unionation.embedded_in_current = false; + union_type->data.unionation.complete = true; + union_type->data.unionation.union_size_bytes = biggest_size_in_bits / 8; + union_type->data.unionation.most_aligned_union_member = most_aligned_union_member; + + if (union_type->data.unionation.is_invalid) + return; + + if (union_type->zero_bits) { + union_type->type_ref = LLVMVoidType(); + + uint64_t debug_size_in_bits = 0; + uint64_t debug_align_in_bits = 0; + ZigLLVMDIType **di_root_members = nullptr; + size_t debug_member_count = 0; + ZigLLVMDIType *replacement_di_type = ZigLLVMCreateDebugUnionType(g->dbuilder, + ZigLLVMFileToScope(import->di_file), + buf_ptr(&union_type->name), + import->di_file, (unsigned)(decl_node->line + 1), + debug_size_in_bits, + debug_align_in_bits, + 0, di_root_members, (int)debug_member_count, 0, ""); + + ZigLLVMReplaceTemporary(g->dbuilder, union_type->di_type, replacement_di_type); + union_type->di_type = replacement_di_type; + return; + } + + assert(most_aligned_union_member != nullptr); + + bool want_safety = auto_layout && (field_count >= 2); + uint64_t padding_in_bits = biggest_size_in_bits - size_of_most_aligned_member_in_bits; + + + if (!want_safety) { + if (padding_in_bits > 0) { + TypeTableEntry *u8_type = get_int_type(g, false, 8); + TypeTableEntry *padding_array = get_array_type(g, u8_type, padding_in_bits / 8); + LLVMTypeRef union_element_types[] = { + most_aligned_union_member->type_ref, + padding_array->type_ref, + }; + LLVMStructSetBody(union_type->type_ref, union_element_types, 2, false); + } else { + LLVMStructSetBody(union_type->type_ref, &most_aligned_union_member->type_ref, 1, false); + } + union_type->data.unionation.union_type_ref = union_type->type_ref; + union_type->data.unionation.gen_tag_index = SIZE_MAX; + union_type->data.unionation.gen_union_index = SIZE_MAX; + + assert(8*LLVMABIAlignmentOfType(g->target_data_ref, union_type->type_ref) >= biggest_align_in_bits); + assert(8*LLVMStoreSizeOfType(g->target_data_ref, union_type->type_ref) >= biggest_size_in_bits); + + // create debug type for union + ZigLLVMDIType *replacement_di_type = ZigLLVMCreateDebugUnionType(g->dbuilder, + ZigLLVMFileToScope(import->di_file), buf_ptr(&union_type->name), + import->di_file, (unsigned)(decl_node->line + 1), + biggest_size_in_bits, biggest_align_in_bits, 0, union_inner_di_types, + gen_field_count, 0, ""); + + ZigLLVMReplaceTemporary(g->dbuilder, union_type->di_type, replacement_di_type); + union_type->di_type = replacement_di_type; + return; + } + + LLVMTypeRef union_type_ref; + if (padding_in_bits > 0) { + TypeTableEntry *u8_type = get_int_type(g, false, 8); + TypeTableEntry *padding_array = get_array_type(g, u8_type, padding_in_bits / 8); + LLVMTypeRef union_element_types[] = { + most_aligned_union_member->type_ref, + padding_array->type_ref, + }; + union_type_ref = LLVMStructType(union_element_types, 2, false); + } else { + union_type_ref = most_aligned_union_member->type_ref; + } + union_type->data.unionation.union_type_ref = union_type_ref; + + assert(8*LLVMABIAlignmentOfType(g->target_data_ref, union_type_ref) >= biggest_align_in_bits); + assert(8*LLVMStoreSizeOfType(g->target_data_ref, union_type_ref) >= biggest_size_in_bits); + + // create llvm type for root struct + TypeTableEntry *tag_int_type = get_smallest_unsigned_int_type(g, field_count - 1); + TypeTableEntry *tag_type_entry = tag_int_type; + union_type->data.unionation.tag_type = tag_type_entry; + uint64_t align_of_tag_in_bits = 8*LLVMABIAlignmentOfType(g->target_data_ref, tag_int_type->type_ref); + + if (align_of_tag_in_bits >= biggest_align_in_bits) { + union_type->data.unionation.gen_tag_index = 0; + union_type->data.unionation.gen_union_index = 1; + } else { + union_type->data.unionation.gen_union_index = 0; + union_type->data.unionation.gen_tag_index = 1; + } + + LLVMTypeRef root_struct_element_types[2]; + root_struct_element_types[union_type->data.unionation.gen_tag_index] = tag_type_entry->type_ref; + root_struct_element_types[union_type->data.unionation.gen_union_index] = union_type_ref; + LLVMStructSetBody(union_type->type_ref, root_struct_element_types, 2, false); + + + // create debug type for root struct + + // create debug type for tag + uint64_t tag_debug_size_in_bits = 8*LLVMStoreSizeOfType(g->target_data_ref, tag_type_entry->type_ref); + uint64_t tag_debug_align_in_bits = 8*LLVMABIAlignmentOfType(g->target_data_ref, tag_type_entry->type_ref); + ZigLLVMDIType *tag_di_type = ZigLLVMCreateDebugEnumerationType(g->dbuilder, + ZigLLVMTypeToScope(union_type->di_type), "AnonEnum", + import->di_file, (unsigned)(decl_node->line + 1), + tag_debug_size_in_bits, tag_debug_align_in_bits, di_enumerators, field_count, + tag_type_entry->di_type, ""); + + // create debug type for union + ZigLLVMDIType *union_di_type = ZigLLVMCreateDebugUnionType(g->dbuilder, + ZigLLVMTypeToScope(union_type->di_type), "AnonUnion", + import->di_file, (unsigned)(decl_node->line + 1), + biggest_size_in_bits, biggest_align_in_bits, 0, union_inner_di_types, + gen_field_count, 0, ""); + + uint64_t union_offset_in_bits = 8*LLVMOffsetOfElement(g->target_data_ref, union_type->type_ref, + union_type->data.unionation.gen_union_index); + uint64_t tag_offset_in_bits = 8*LLVMOffsetOfElement(g->target_data_ref, union_type->type_ref, + union_type->data.unionation.gen_tag_index); + + ZigLLVMDIType *union_member_di_type = ZigLLVMCreateDebugMemberType(g->dbuilder, + ZigLLVMTypeToScope(union_type->di_type), "union_field", + import->di_file, (unsigned)(decl_node->line + 1), + biggest_size_in_bits, + biggest_align_in_bits, + union_offset_in_bits, + 0, union_di_type); + ZigLLVMDIType *tag_member_di_type = ZigLLVMCreateDebugMemberType(g->dbuilder, + ZigLLVMTypeToScope(union_type->di_type), "tag_field", + import->di_file, (unsigned)(decl_node->line + 1), + tag_debug_size_in_bits, + tag_debug_align_in_bits, + tag_offset_in_bits, + 0, tag_di_type); + + ZigLLVMDIType *di_root_members[2]; + di_root_members[union_type->data.unionation.gen_tag_index] = tag_member_di_type; + di_root_members[union_type->data.unionation.gen_union_index] = union_member_di_type; + + uint64_t debug_size_in_bits = 8*LLVMStoreSizeOfType(g->target_data_ref, union_type->type_ref); + uint64_t debug_align_in_bits = 8*LLVMABISizeOfType(g->target_data_ref, union_type->type_ref); + ZigLLVMDIType *replacement_di_type = ZigLLVMCreateDebugStructType(g->dbuilder, + ZigLLVMFileToScope(import->di_file), + buf_ptr(&union_type->name), + import->di_file, (unsigned)(decl_node->line + 1), + debug_size_in_bits, + debug_align_in_bits, + 0, nullptr, di_root_members, 2, 0, nullptr, ""); + + ZigLLVMReplaceTemporary(g->dbuilder, union_type->di_type, replacement_di_type); + union_type->di_type = replacement_di_type; } static void resolve_enum_zero_bits(CodeGen *g, TypeTableEntry *enum_type) { @@ -1873,7 +2113,7 @@ static void resolve_enum_zero_bits(CodeGen *g, TypeTableEntry *enum_type) { type_enum_field->value = i; type_ensure_zero_bits_known(g, field_type); - if (field_type->id == TypeTableEntryIdInvalid) { + if (type_is_invalid(field_type)) { enum_type->data.enumeration.is_invalid = true; continue; } @@ -1980,7 +2220,69 @@ static void resolve_struct_zero_bits(CodeGen *g, TypeTableEntry *struct_type) { } static void resolve_union_zero_bits(CodeGen *g, TypeTableEntry *union_type) { - zig_panic("TODO resolve_union_zero_bits"); + assert(union_type->id == TypeTableEntryIdUnion); + + if (union_type->data.unionation.zero_bits_known) + return; + + if (union_type->data.unionation.zero_bits_loop_flag) { + union_type->data.unionation.zero_bits_known = true; + return; + } + + union_type->data.unionation.zero_bits_loop_flag = true; + + AstNode *decl_node = union_type->data.unionation.decl_node; + assert(decl_node->type == NodeTypeContainerDecl); + assert(union_type->di_type); + + assert(!union_type->data.unionation.fields); + uint32_t field_count = (uint32_t)decl_node->data.container_decl.fields.length; + union_type->data.unionation.src_field_count = field_count; + union_type->data.unionation.fields = allocate<TypeUnionField>(field_count); + + uint32_t biggest_align_bytes = 0; + + Scope *scope = &union_type->data.unionation.decls_scope->base; + + uint32_t gen_field_index = 0; + for (uint32_t i = 0; i < field_count; i += 1) { + AstNode *field_node = decl_node->data.container_decl.fields.at(i); + TypeUnionField *type_union_field = &union_type->data.unionation.fields[i]; + type_union_field->name = field_node->data.struct_field.name; + TypeTableEntry *field_type = analyze_type_expr(g, scope, field_node->data.struct_field.type); + type_union_field->type_entry = field_type; + type_union_field->value = i; + + type_ensure_zero_bits_known(g, field_type); + if (type_is_invalid(field_type)) { + union_type->data.unionation.is_invalid = true; + continue; + } + + if (!type_has_bits(field_type)) + continue; + + type_union_field->gen_index = gen_field_index; + gen_field_index += 1; + + uint32_t field_align_bytes = get_abi_alignment(g, field_type); + if (field_align_bytes > biggest_align_bytes) { + biggest_align_bytes = field_align_bytes; + } + } + + bool auto_layout = (union_type->data.unionation.layout == ContainerLayoutAuto); + + union_type->data.unionation.zero_bits_loop_flag = false; + union_type->data.unionation.gen_field_count = gen_field_index; + union_type->zero_bits = (gen_field_index == 0 && (field_count < 2 || !auto_layout)); + union_type->data.unionation.zero_bits_known = true; + + // also compute abi_alignment + if (!union_type->zero_bits) { + union_type->data.unionation.abi_alignment = biggest_align_bytes; + } } static void get_fully_qualified_decl_name_internal(Buf *buf, Scope *scope, uint8_t sep) { @@ -2851,6 +3153,18 @@ TypeStructField *find_struct_type_field(TypeTableEntry *type_entry, Buf *name) { return nullptr; } +TypeUnionField *find_union_type_field(TypeTableEntry *type_entry, Buf *name) { + assert(type_entry->id == TypeTableEntryIdUnion); + assert(type_entry->data.unionation.complete); + for (uint32_t i = 0; i < type_entry->data.unionation.src_field_count; i += 1) { + TypeUnionField *field = &type_entry->data.unionation.fields[i]; + if (buf_eql_buf(field->name, name)) { + return field; + } + } + return nullptr; +} + static bool is_container(TypeTableEntry *type_entry) { switch (type_entry->id) { case TypeTableEntryIdInvalid: @@ -4703,6 +5017,8 @@ ConstParent *get_const_val_parent(CodeGen *g, ConstExprValue *value) { return &value->data.x_array.s_none.parent; } else if (type_entry->id == TypeTableEntryIdStruct) { return &value->data.x_struct.parent; + } else if (type_entry->id == TypeTableEntryIdUnion) { + return &value->data.x_union.parent; } return nullptr; } @@ -4914,7 +5230,8 @@ uint32_t get_abi_alignment(CodeGen *g, TypeTableEntry *type_entry) { assert(type_entry->data.enumeration.abi_alignment != 0); return type_entry->data.enumeration.abi_alignment; } else if (type_entry->id == TypeTableEntryIdUnion) { - zig_panic("TODO"); + assert(type_entry->data.unionation.abi_alignment != 0); + return type_entry->data.unionation.abi_alignment; } else if (type_entry->id == TypeTableEntryIdOpaque) { return 1; } else { @@ -4929,3 +5246,11 @@ TypeTableEntry *get_align_amt_type(CodeGen *g) { } return g->align_amt_type; } + +uint32_t type_ptr_hash(const TypeTableEntry *ptr) { + return hash_ptr((void*)ptr); +} + +bool type_ptr_eql(const TypeTableEntry *a, const TypeTableEntry *b) { + return a == b; +} |
