diff options
| author | Ginger Bill <bill@gingerbill.org> | 2017-02-07 00:10:58 +0000 |
|---|---|---|
| committer | Ginger Bill <bill@gingerbill.org> | 2017-02-07 00:10:58 +0000 |
| commit | 219ca0ac4677235d595d9bd6e1be08eedfdf7d66 (patch) | |
| tree | 453fc6cb9bd0e8ecc0952cde48994b67b779c08d | |
| parent | 5796c413571140798d9ca597cda4d34c6762eabe (diff) | |
Map type info and fmt printing
| -rw-r--r-- | code/demo.odin | 6 | ||||
| -rw-r--r-- | core/_preload.odin | 74 | ||||
| -rw-r--r-- | core/fmt.odin | 49 | ||||
| -rw-r--r-- | src/check_expr.c | 16 | ||||
| -rw-r--r-- | src/checker.c | 14 | ||||
| -rw-r--r-- | src/ir.c | 24 | ||||
| -rw-r--r-- | src/types.c | 8 |
7 files changed, 139 insertions, 52 deletions
diff --git a/code/demo.odin b/code/demo.odin index d91d1d5c7..45ae781a2 100644 --- a/code/demo.odin +++ b/code/demo.odin @@ -12,7 +12,7 @@ main :: proc() { { - m := map[f32]int{}; + m: map[f32]int; reserve(^m, 16); defer free(m); @@ -39,9 +39,7 @@ main :: proc() { _, ok := m["c"]; assert(ok && c == 7654); - for val, key in m { - fmt.printf("m[\"%s\"] == %v\n", key, val); - } + fmt.println(m); } diff --git a/core/_preload.odin b/core/_preload.odin index ec473348e..1eb8a2d06 100644 --- a/core/_preload.odin +++ b/core/_preload.odin @@ -20,11 +20,12 @@ Type_Info_Member :: struct #ordered { offset: int, // offsets are not used in tuples } Type_Info_Record :: struct #ordered { - fields: []Type_Info_Member, - size: int, // in bytes - align: int, // in bytes - packed: bool, - ordered: bool, + fields: []Type_Info_Member, + size: int, // in bytes + align: int, // in bytes + packed: bool, + ordered: bool, + custom_align: bool, } Type_Info_Enum_Value :: raw_union { f: f64, @@ -90,10 +91,16 @@ Type_Info :: union { Union: Type_Info_Record, Raw_Union: Type_Info_Record, Enum: struct #ordered { - base: ^Type_Info, - names: []string, + base: ^Type_Info, + names: []string, values: []Type_Info_Enum_Value, }, + Map: struct #ordered { + key: ^Type_Info, + value: ^Type_Info, + generated_struct: ^Type_Info, + count: int, // == 0 if dynamic + }, } // // NOTE(bill): only the ones that are needed (not all types) @@ -113,6 +120,21 @@ type_info_base :: proc(info: ^Type_Info) -> ^Type_Info { } +type_info_base_without_enum :: proc(info: ^Type_Info) -> ^Type_Info { + if info == nil { + return nil; + } + base := info; + match type i in base { + case Type_Info.Named: + base = i.base; + case Type_Info.Enum: + base = i.base; + } + return base; +} + + assume :: proc(cond: bool) #foreign __llvm_core "llvm.assume"; @@ -444,26 +466,26 @@ __default_hash_string :: proc(s: string) -> u64 { return __default_hash(cast([]byte)s); } -Map_Key :: struct #ordered { +__Map_Key :: struct #ordered { hash: u64, str: string, } -Map_Find_Result :: struct #ordered { +__Map_Find_Result :: struct #ordered { hash_index: int, entry_prev: int, entry_index: int, } -Map_Entry_Header :: struct #ordered { - key: Map_Key, +__Map_Entry_Header :: struct #ordered { + key: __Map_Key, next: int, /* value: Value_Type, */ } -Map_Header :: struct #ordered { +__Map_Header :: struct #ordered { m: ^Raw_Dynamic_Map, is_key_string: bool, entry_size: int, @@ -471,13 +493,13 @@ Map_Header :: struct #ordered { value_offset: int, } -__dynamic_map_reserve :: proc(using header: Map_Header, capacity: int) -> bool { +__dynamic_map_reserve :: proc(using header: __Map_Header, capacity: int) -> bool { h := __dynamic_array_reserve(^m.hashes, size_of(int), align_of(int), capacity); e := __dynamic_array_reserve(^m.entries, entry_size, entry_align, capacity); return h && e; } -__dynamic_map_rehash :: proc(using header: Map_Header, new_count: int) { +__dynamic_map_rehash :: proc(using header: __Map_Header, new_count: int) { new_header := header; nm: Raw_Dynamic_Map; new_header.m = ^nm; @@ -519,7 +541,7 @@ __dynamic_map_rehash :: proc(using header: Map_Header, new_count: int) { header.m^ = nm; } -__dynamic_map_get :: proc(h: Map_Header, key: Map_Key) -> rawptr { +__dynamic_map_get :: proc(h: __Map_Header, key: __Map_Key) -> rawptr { index := __dynamic_map_find(h, key).entry_index; if index >= 0 { data := cast(^byte)__dynamic_map_get_entry(h, index); @@ -529,7 +551,7 @@ __dynamic_map_get :: proc(h: Map_Header, key: Map_Key) -> rawptr { return nil; } -__dynamic_map_set :: proc(using h: Map_Header, key: Map_Key, value: rawptr) { +__dynamic_map_set :: proc(using h: __Map_Header, key: __Map_Key, value: rawptr) { index: int; if m.hashes.count == 0 { @@ -559,17 +581,17 @@ __dynamic_map_set :: proc(using h: Map_Header, key: Map_Key, value: rawptr) { } -__dynamic_map_grow :: proc(using h: Map_Header) { +__dynamic_map_grow :: proc(using h: __Map_Header) { new_count := 2*m.entries.count + 8; __dynamic_map_rehash(h, new_count); } -__dynamic_map_full :: proc(using h: Map_Header) -> bool { +__dynamic_map_full :: proc(using h: __Map_Header) -> bool { return cast(int)(0.75 * cast(f64)m.hashes.count) <= m.entries.count; } -__dynamic_map_hash_equal :: proc(h: Map_Header, a, b: Map_Key) -> bool { +__dynamic_map_hash_equal :: proc(h: __Map_Header, a, b: __Map_Key) -> bool { if a.hash == b.hash { if h.is_key_string { return a.str == b.str; @@ -579,8 +601,8 @@ __dynamic_map_hash_equal :: proc(h: Map_Header, a, b: Map_Key) -> bool { return false; } -__dynamic_map_find :: proc(using h: Map_Header, key: Map_Key) -> Map_Find_Result { - fr := Map_Find_Result{-1, -1, -1}; +__dynamic_map_find :: proc(using h: __Map_Header, key: __Map_Key) -> __Map_Find_Result { + fr := __Map_Find_Result{-1, -1, -1}; if m.hashes.count > 0 { fr.hash_index = cast(int)(key.hash % cast(u64)m.hashes.count); fr.entry_index = m.hashes[fr.hash_index]; @@ -596,7 +618,7 @@ __dynamic_map_find :: proc(using h: Map_Header, key: Map_Key) -> Map_Find_Result return fr; } -__dynamic_map_add_entry :: proc(using h: Map_Header, key: Map_Key) -> int { +__dynamic_map_add_entry :: proc(using h: __Map_Header, key: __Map_Key) -> int { prev := m.entries.count; c := __dynamic_array_append_nothing(^m.entries, entry_size, entry_align); if c != prev { @@ -608,19 +630,19 @@ __dynamic_map_add_entry :: proc(using h: Map_Header, key: Map_Key) -> int { } -__dynamic_map_remove :: proc(using h: Map_Header, key: Map_Key) { +__dynamic_map_remove :: proc(using h: __Map_Header, key: __Map_Key) { fr := __dynamic_map_find(h, key); if fr.entry_index >= 0 { __dynamic_map_erase(h, fr); } } -__dynamic_map_get_entry :: proc(using h: Map_Header, index: int) -> ^Map_Entry_Header { +__dynamic_map_get_entry :: proc(using h: __Map_Header, index: int) -> ^__Map_Entry_Header { data := cast(^byte)m.entries.data + index*entry_size; - return cast(^Map_Entry_Header)data; + return cast(^__Map_Entry_Header)data; } -__dynamic_map_erase :: proc(using h: Map_Header, fr: Map_Find_Result) { +__dynamic_map_erase :: proc(using h: __Map_Header, fr: __Map_Find_Result) { if fr.entry_prev < 0 { m.hashes[fr.hash_index] = __dynamic_map_get_entry(h, fr.entry_index).next; } else { diff --git a/core/fmt.odin b/core/fmt.odin index 105888caf..91519167b 100644 --- a/core/fmt.odin +++ b/core/fmt.odin @@ -191,18 +191,30 @@ buffer_write_type :: proc(buf: ^Buffer, ti: ^Type_Info) { buffer_write_string(buf, "]"); buffer_write_type(buf, info.elem); + case Map: + buffer_write_string(buf, "map["); + buffer_write_type(buf, info.key); + buffer_write_byte(buf, ']'); + buffer_write_type(buf, info.value); + case Struct: buffer_write_string(buf, "struct "); if info.packed { buffer_write_string(buf, "#packed "); } if info.ordered { buffer_write_string(buf, "#ordered "); } - buffer_write_string(buf, "{"); + if info.custom_align { + buffer_write_string(buf, "#align "); + fi := Fmt_Info{buf = buf}; + fmt_int(^fi, cast(u64)info.align, false, 'd'); + buffer_write_byte(buf, ' '); + } + buffer_write_byte(buf, '{'); for field, i in info.fields { buffer_write_string(buf, field.name); buffer_write_string(buf, ": "); buffer_write_type(buf, field.type_info); buffer_write_byte(buf, ','); } - buffer_write_string(buf, "}"); + buffer_write_byte(buf, '}'); case Union: buffer_write_string(buf, "union {"); @@ -778,6 +790,39 @@ fmt_value :: proc(fi: ^Fmt_Info, v: any, verb: rune) { fmt_arg(fi, any{info.elem, cast(rawptr)data}, 'v'); } + case Map: + if verb != 'v' { + fmt_bad_verb(fi, verb); + return; + } + + buffer_write_string(fi.buf, "map["); + defer buffer_write_byte(fi.buf, ']'); + entries := ^(cast(^Raw_Dynamic_Map)v.data).entries; + gs, _ := union_cast(^Struct)info.generated_struct; + ed, _ := union_cast(^Dynamic_Array)gs.fields[1].type_info; + entry_type, _ := union_cast(^Struct)ed.elem; + entry_size := ed.elem_size; + for i in 0..<entries.count { + if i > 0 { + buffer_write_string(fi.buf, ", "); + } + data := cast(^byte)entries.data + i*entry_size; + + header := cast(^__Map_Entry_Header)data; + if types.is_string(info.key) { + buffer_write_string(fi.buf, header.key.str); + } else { + fi := Fmt_Info{buf = fi.buf}; + fmt_arg(^fi, any{info.key, cast(rawptr)^header.key.hash}, 'v'); + } + + buffer_write_string(fi.buf, "="); + + value := data + entry_type.fields[2].offset; + fmt_arg(fi, any{info.value, cast(rawptr)value}, 'v'); + } + case Slice: if verb != 'v' { fmt_bad_verb(fi, verb); diff --git a/src/check_expr.c b/src/check_expr.c index 13f66d060..a8f5fe55b 100644 --- a/src/check_expr.c +++ b/src/check_expr.c @@ -1097,8 +1097,8 @@ Type *make_map_tuple_type(gbAllocator a, Type *value) { Type *t = make_type_tuple(a); t->Tuple.variables = gb_alloc_array(a, Entity *, 2); t->Tuple.variable_count = 2; - t->Tuple.variables[0] = make_entity_param(a, NULL, blank_token, value, false, false); - t->Tuple.variables[1] = make_entity_param(a, NULL, blank_token, t_bool, false, false); + t->Tuple.variables[0] = make_entity_field(a, NULL, blank_token, value, false, 0); + t->Tuple.variables[1] = make_entity_field(a, NULL, blank_token, t_bool, false, 1); return t; } @@ -1148,9 +1148,9 @@ void check_map_type(Checker *c, Type *type, AstNode *node) { isize field_count = 3; Entity **fields = gb_alloc_array(a, Entity *, field_count); - fields[0] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("key")), t_map_key, false, false); - fields[1] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("next")), t_int, false, false); - fields[2] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("value")), value, false, false); + fields[0] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("key")), t_map_key, false, 0); + fields[1] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("next")), t_int, false, 1); + fields[2] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("value")), value, false, 2); check_close_scope(c); @@ -1159,7 +1159,6 @@ void check_map_type(Checker *c, Type *type, AstNode *node) { entry_type->Record.field_count = field_count; type_set_offsets(c->sizes, a, entry_type); - type->Map.entry_type = entry_type; } @@ -1181,8 +1180,8 @@ void check_map_type(Checker *c, Type *type, AstNode *node) { isize field_count = 2; Entity **fields = gb_alloc_array(a, Entity *, field_count); - fields[0] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("hashes")), hashes_type, false, false); - fields[1] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("entries")), entries_type, false, false); + fields[0] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("hashes")), hashes_type, false, 0); + fields[1] = make_entity_field(a, c->context.scope, make_token_ident(str_lit("entries")), entries_type, false, 1); check_close_scope(c); @@ -1191,7 +1190,6 @@ void check_map_type(Checker *c, Type *type, AstNode *node) { generated_struct_type->Record.field_count = field_count; type_set_offsets(c->sizes, a, generated_struct_type); - type->Map.generated_struct_type = generated_struct_type; } diff --git a/src/checker.c b/src/checker.c index 12915197a..78643e03c 100644 --- a/src/checker.c +++ b/src/checker.c @@ -958,6 +958,12 @@ void add_type_info_type(Checker *c, Type *t) { } } break; + case Type_Map: { + add_type_info_type(c, bt->Map.key); + add_type_info_type(c, bt->Map.value); + add_type_info_type(c, bt->Map.generated_struct_type); + } break; + case Type_Tuple: for (isize i = 0; i < bt->Tuple.variable_count; i++) { Entity *var = bt->Tuple.variables[i]; @@ -1093,7 +1099,7 @@ void init_preload(Checker *c) { - if (record->field_count != 19) { + if (record->field_count != 20) { compiler_error("Invalid `Type_Info` layout"); } t_type_info_named = record->fields[ 1]->type; @@ -1114,6 +1120,7 @@ void init_preload(Checker *c) { t_type_info_union = record->fields[16]->type; t_type_info_raw_union = record->fields[17]->type; t_type_info_enum = record->fields[18]->type; + t_type_info_map = record->fields[19]->type; t_type_info_named_ptr = make_type_pointer(heap_allocator(), t_type_info_named); t_type_info_integer_ptr = make_type_pointer(heap_allocator(), t_type_info_integer); @@ -1133,6 +1140,7 @@ void init_preload(Checker *c) { t_type_info_union_ptr = make_type_pointer(heap_allocator(), t_type_info_union); t_type_info_raw_union_ptr = make_type_pointer(heap_allocator(), t_type_info_raw_union); t_type_info_enum_ptr = make_type_pointer(heap_allocator(), t_type_info_enum); + t_type_info_map_ptr = make_type_pointer(heap_allocator(), t_type_info_map); } if (t_allocator == NULL) { @@ -1155,12 +1163,12 @@ void init_preload(Checker *c) { } if (t_map_key == NULL) { - Entity *e = find_core_entity(c, str_lit("Map_Key")); + Entity *e = find_core_entity(c, str_lit("__Map_Key")); t_map_key = e->type; } if (t_map_header == NULL) { - Entity *e = find_core_entity(c, str_lit("Map_Header")); + Entity *e = find_core_entity(c, str_lit("__Map_Header")); t_map_header = e->type; } @@ -6286,14 +6286,16 @@ void ir_gen_tree(irGen *s) { tag = ir_emit_conv(proc, ti_ptr, t_type_info_struct_ptr); { - irValue *packed = ir_make_const_bool(a, t->Record.struct_is_packed); - irValue *ordered = ir_make_const_bool(a, t->Record.struct_is_ordered); - irValue *size = ir_make_const_int(a, type_size_of(m->sizes, a, t)); - irValue *align = ir_make_const_int(a, type_align_of(m->sizes, a, t)); + irValue *size = ir_make_const_int(a, type_size_of(m->sizes, a, t)); + irValue *align = ir_make_const_int(a, type_align_of(m->sizes, a, t)); + irValue *packed = ir_make_const_bool(a, t->Record.struct_is_packed); + irValue *ordered = ir_make_const_bool(a, t->Record.struct_is_ordered); + irValue *custom_align = ir_make_const_bool(a, t->Record.custom_align); ir_emit_store(proc, ir_emit_struct_ep(proc, tag, 1), size); ir_emit_store(proc, ir_emit_struct_ep(proc, tag, 2), align); ir_emit_store(proc, ir_emit_struct_ep(proc, tag, 3), packed); ir_emit_store(proc, ir_emit_struct_ep(proc, tag, 4), ordered); + ir_emit_store(proc, ir_emit_struct_ep(proc, tag, 5), custom_align); } irValue *memory = ir_type_info_member_offset(proc, type_info_member_data, t->Record.field_count, &type_info_member_index); @@ -6516,6 +6518,20 @@ void ir_gen_tree(irGen *s) { // TODO(bill): Type_Info for procedures } break; + + case Type_Map: { + tag = ir_emit_conv(proc, ti_ptr, t_type_info_map_ptr); + + irValue *key = ir_emit_struct_ep(proc, tag, 0); + irValue *value = ir_emit_struct_ep(proc, tag, 1); + irValue *generated_struct = ir_emit_struct_ep(proc, tag, 2); + irValue *count = ir_emit_struct_ep(proc, tag, 3); + + ir_emit_store(proc, key, ir_get_type_info_ptr(proc, type_info_data, t->Map.key)); + ir_emit_store(proc, value, ir_get_type_info_ptr(proc, type_info_data, t->Map.value)); + ir_emit_store(proc, generated_struct, ir_get_type_info_ptr(proc, type_info_data, t->Map.generated_struct_type)); + ir_emit_store(proc, count, ir_make_const_int(a, t->Map.count)); + } break; } if (tag != NULL) { diff --git a/src/types.c b/src/types.c index be28c82f0..eae9ec556 100644 --- a/src/types.c +++ b/src/types.c @@ -300,6 +300,7 @@ gb_global Type *t_type_info_struct = NULL; gb_global Type *t_type_info_union = NULL; gb_global Type *t_type_info_raw_union = NULL; gb_global Type *t_type_info_enum = NULL; +gb_global Type *t_type_info_map = NULL; gb_global Type *t_type_info_named_ptr = NULL; @@ -320,6 +321,7 @@ gb_global Type *t_type_info_struct_ptr = NULL; gb_global Type *t_type_info_union_ptr = NULL; gb_global Type *t_type_info_raw_union_ptr = NULL; gb_global Type *t_type_info_enum_ptr = NULL; +gb_global Type *t_type_info_map_ptr = NULL; @@ -1561,8 +1563,7 @@ i64 type_align_of_internal(BaseTypeSizes s, gbAllocator allocator, Type *t, Type case Type_Map: { if (t->Map.count == 0) { // Dynamic - // NOTE(bill): same as a dynamic array - return s.word_size; + return type_align_of_internal(s, allocator, t->Map.generated_struct_type, path); } GB_PANIC("TODO(bill): Fixed map alignment"); } break; @@ -1778,8 +1779,7 @@ i64 type_size_of_internal(BaseTypeSizes s, gbAllocator allocator, Type *t, TypeP case Type_Map: { if (t->Map.count == 0) { // Dynamic - // NOTE(bill): same as a two dynamic arrays - return 2 * type_size_of_internal(s, allocator, t_raw_dynamic_array, path); + return type_size_of_internal(s, allocator, t->Map.generated_struct_type, path); } GB_PANIC("TODO(bill): Fixed map size"); } |