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 /core | |
| parent | 5796c413571140798d9ca597cda4d34c6762eabe (diff) | |
Map type info and fmt printing
Diffstat (limited to 'core')
| -rw-r--r-- | core/_preload.odin | 74 | ||||
| -rw-r--r-- | core/fmt.odin | 49 |
2 files changed, 95 insertions, 28 deletions
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); |