Files
raddebugger/src/linker/codeview_ext/codeview.c
T
2024-12-06 11:53:16 -08:00

2256 lines
68 KiB
C

// Copyright (c) 2024 Epic Games Tools
// Licensed under the MIT license (https://opensource.org/license/mit/)
////////////////////////////////
internal U64
hash_from_cv_symbol(CV_Symbol *symbol)
{
XXH3_state_t hasher;
XXH3_64bits_reset(&hasher);
XXH3_64bits_update(&hasher, &symbol->kind, sizeof(symbol->kind));
XXH3_64bits_update(&hasher, &symbol->data.size, sizeof(symbol->data.size));
XXH3_64bits_update(&hasher, symbol->data.str, symbol->data.size);
XXH64_hash_t hash = XXH3_64bits_digest(&hasher);
return hash;
}
////////////////////////////////
internal CV_ObjInfo
cv_obj_info_from_symbol(CV_Symbol symbol)
{
CV_ObjInfo result; MemoryZeroStruct(&result);
switch (symbol.kind) {
case CV_SymKind_OBJNAME: {
CV_SymObjName *obj_name = (CV_SymObjName *) symbol.data.str;
result.sig = obj_name->sig;
str8_deserial_read_cstr(symbol.data, sizeof(CV_SymObjName), &result.name);
} break;
case CV_SymKind_OBJNAME_ST: {
NotImplemented;
} break;
default: {
InvalidPath;
} break;
}
return result;
}
internal CV_TypeServerInfo
cv_type_server_info_from_leaf(CV_Leaf leaf)
{
CV_TypeServerInfo result = {0};
switch (leaf.kind) {
case CV_LeafKind_TYPESERVER: {
CV_LeafTypeServer *ts = (CV_LeafTypeServer *) leaf.data.str;
result.name = str8_cstring_capped_reverse(ts + 1, leaf.data.str + leaf.data.size);
result.sig.data1 = ts->sig;
result.age = ts->age;
} break;
case CV_LeafKind_TYPESERVER2: {
CV_LeafTypeServer2 *ts = (CV_LeafTypeServer2 *) leaf.data.str;
Assert(sizeof(result.sig) == sizeof(ts->sig70));
MemoryCopy(&result.sig, &ts->sig70, sizeof(ts->sig70));
result.name = str8_cstring_capped_reverse(ts + 1, leaf.data.str + leaf.data.size);
result.age = ts->age;
} break;
case CV_LeafKind_TYPESERVER_ST: {
Assert("TODO: LF_TYPESERVER_ST");
} break;
default: InvalidPath;
}
return result;
}
internal CV_PrecompInfo
cv_precomp_info_from_leaf(CV_Leaf leaf)
{
CV_PrecompInfo result = {0};
switch (leaf.kind) {
case CV_LeafKind_PRECOMP: {
CV_LeafPreComp *precomp = (CV_LeafPreComp*)leaf.data.str;
result.start_index = precomp->start_index;
result.sig = precomp->sig;
result.leaf_count = precomp->count;
str8_deserial_read_cstr(leaf.data, sizeof(CV_LeafPreComp), &result.obj_name);
} break;
case CV_LeafKind_PRECOMP_16t: {
NotImplemented;
} break;
case CV_LeafKind_PRECOMP_ST: {
NotImplemented;
} break;
default: {
InvalidPath;
} break;
}
return result;
}
////////////////////////////////
//~ Leaf Helpers
internal U64
cv_compute_leaf_record_size(String8 data, U64 align)
{
U64 size = 0;
size += sizeof(CV_LeafSize);
size += sizeof(CV_LeafKind);
size += data.size;
size = AlignPow2(size, align);
return size;
}
internal U64
cv_serialize_leaf_to_buffer(U8 *buffer, U64 buffer_cursor, U64 buffer_size, CV_LeafKind kind, String8 data, U64 align)
{
U64 buffer_cursor_start = buffer_cursor;
// compute record size
U64 record_size = sizeof(kind) + data.size;
Assert(record_size <= CV_LeafSize_Max);
CV_LeafSize record_size16 = (CV_LeafSize)record_size;
// compute pad
static U8 LEAF_PAD_ARR[] = { 0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff };
U64 pad_size = AlignPadPow2(data.size, align);
Assert(pad_size <= ArrayCount(LEAF_PAD_ARR));
// write header
CV_LeafHeader *header_ptr = (CV_LeafHeader *)(buffer + buffer_cursor);
header_ptr->size = record_size16;
header_ptr->kind = kind;
buffer_cursor += sizeof(*header_ptr);
// write body
U8 *leaf_data_ptr = buffer + buffer_cursor;
MemoryCopy(leaf_data_ptr, data.str, data.size);
buffer_cursor += data.size;
// write pad
U8 *pad_data_ptr = buffer + buffer_cursor;
MemoryCopy(pad_data_ptr, &LEAF_PAD_ARR[0], pad_size);
buffer_cursor += pad_size;
U64 write_size = buffer_cursor - buffer_cursor_start;
return write_size;
}
internal String8
cv_serialize_raw_leaf(Arena *arena, CV_LeafKind kind, String8 data, U64 align)
{
U64 buffer_size = cv_compute_leaf_record_size(data, align);
U8 *buffer = push_array_no_zero(arena, U8, buffer_size);
U64 size = cv_serialize_leaf_to_buffer(buffer, 0, buffer_size, kind, data, align);
String8 raw_leaf = str8(buffer, size);
return raw_leaf;
}
internal String8
cv_serialize_leaf(Arena *arena, CV_Leaf *leaf, U64 align)
{
return cv_serialize_raw_leaf(arena, leaf->kind, leaf->data, align);
}
internal CV_Leaf
cv_make_leaf(Arena *arena, CV_LeafKind kind, String8 data)
{
CV_Leaf result = {0};
String8 raw_leaf = cv_serialize_raw_leaf(arena, kind, data, 1);
cv_deserial_leaf(raw_leaf, 0, 1, &result);
return result;
}
internal U64
cv_deserial_leaf(String8 raw_data, U64 off, U64 align, CV_Leaf *leaf_out)
{
// do we have enough bytes to read header?
Assert(raw_data.size >= sizeof(CV_LeafHeader));
CV_LeafHeader *header = (CV_LeafHeader*)(raw_data.str + off);
// leaf size must have enough bytes for the kind enum
Assert(header->size >= sizeof(CV_LeafKind));
// do we have enough bytes to read leaf data?
Assert(sizeof(CV_LeafSize) + header->size <= raw_data.size);
// fill out leaf
leaf_out->kind = header->kind;
leaf_out->data = str8(raw_data.str + sizeof(CV_LeafHeader), header->size - sizeof(CV_LeafKind));
U64 leaf_size = AlignPow2(sizeof(CV_LeafHeader) + leaf_out->data.size, align);
Assert(leaf_size <= raw_data.size);
return leaf_size;
}
internal CV_Leaf
cv_leaf_from_string(String8 raw_data)
{
CV_Leaf result;
cv_deserial_leaf(raw_data, 0, 1, &result);
return result;
}
////////////////////////////////
//~ Symbol Helpers
internal U64
cv_compute_symbol_record_size(CV_Symbol *symbol, U64 align)
{
U64 size = 0;
size += sizeof(CV_SymSize);
size += sizeof(CV_SymKind);
size += AlignPow2(symbol->data.size, align);
return size;
}
internal U64
cv_serialize_symbol_to_buffer(U8 *buffer, U64 buffer_cursor, U64 buffer_size, CV_Symbol *symbol, U64 align)
{
U64 write_size = cv_compute_symbol_record_size(symbol, align);
Assert(buffer_cursor + write_size <= buffer_size);
U64 record_size = 0;
record_size += sizeof(symbol->kind);
record_size += AlignPow2(symbol->data.size, align);
Assert(record_size <= CV_SymSize_Max);
CV_SymSize record_size16 = (CV_SymSize)record_size;
// init header
CV_SymbolHeader *header = (CV_SymbolHeader *)(buffer + buffer_cursor);
header->size = record_size16;
header->kind = symbol->kind;
// copy symbol data
U8 *data_dst = (U8 *)(header + 1);
MemoryCopy(data_dst, symbol->data.str, symbol->data.size);
// set pad bytes
U64 pad_size = AlignPadPow2(symbol->data.size, align);
U8 *pad_dst = data_dst + symbol->data.size;
MemorySet(&pad_dst[0], 0, pad_size);
return write_size;
}
internal String8
cv_serialize_symbol(Arena *arena, CV_Symbol *symbol, U64 align)
{
U64 buffer_size = cv_compute_symbol_record_size(symbol, align);
U8 *buffer = push_array(arena, U8, buffer_size);
cv_serialize_symbol_to_buffer(buffer, 0, buffer_size, symbol, align);
String8 result = str8(buffer, buffer_size);
return result;
}
internal String8
cv_make_symbol(Arena *arena, CV_SymKind kind, String8 data)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
AssertAlways((data.size + sizeof(kind)) <= CV_SymSize_Max);
CV_SymSize symbol_size = (CV_SymSize)data.size + sizeof(kind);
String8List srl = {0};
str8_serial_begin(scratch.arena, &srl);
str8_serial_push_struct(scratch.arena, &srl, &symbol_size);
str8_serial_push_struct(scratch.arena, &srl, &kind);
str8_serial_push_string(scratch.arena, &srl, data);
String8 symbol = str8_serial_end(arena, &srl);
scratch_end(scratch);
ProfEnd();
return symbol;
}
internal String8
cv_make_obj_name(Arena *arena, String8 obj_path, U32 sig)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
CV_SymObjName obj = {0};
obj.sig = sig;
String8List serial = {0};
str8_serial_begin(scratch.arena, &serial);
str8_serial_push_struct(scratch.arena, &serial, &obj);
str8_serial_push_cstr(scratch.arena, &serial, obj_path);
String8 result = str8_serial_end(arena, &serial);
scratch_end(scratch);
ProfEnd();
return result;
}
internal String8
cv_make_comp3(Arena *arena,
CV_Compile3Flags flags, CV_Language lang, CV_Arch arch,
U16 ver_fe_major, U16 ver_fe_minor, U16 ver_fe_build, U16 ver_feqfe,
U16 ver_major, U16 ver_minor, U16 ver_build, U16 ver_qfe,
String8 version_string)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
CV_SymCompile3 comp = {0};
comp.flags = flags | lang;
comp.machine = arch;
comp.ver_fe_major = ver_fe_major;
comp.ver_fe_minor = ver_fe_minor;
comp.ver_fe_build = ver_fe_build;
comp.ver_feqfe = ver_feqfe;
comp.ver_major = ver_major;
comp.ver_minor = ver_minor;
comp.ver_build = ver_build;
comp.ver_qfe = ver_qfe;
String8List serial = {0};
str8_serial_begin(scratch.arena, &serial);
str8_serial_push_struct(scratch.arena, &serial, &comp);
str8_serial_push_cstr(scratch.arena, &serial, version_string);
String8 result = str8_serial_end(arena, &serial);
scratch_end(scratch);
ProfEnd();
return result;
}
internal String8
cv_make_envblock(Arena *arena, String8List string_list)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
String8List serial = {0};
str8_serial_begin(scratch.arena, &serial);
CV_SymEnvBlock envblock = {0};
str8_serial_push_struct(scratch.arena, &serial, &envblock);
for (String8Node *n = string_list.first; n != NULL; n = n->next) {
str8_serial_push_cstr(scratch.arena, &serial, n->string);
}
String8 result = str8_serial_end(arena, &serial);
scratch_end(scratch);
ProfEnd();
return result;
}
internal CV_Symbol
cv_make_proc_ref(Arena *arena, CV_ModIndex imod, U32 stream_offset, String8 name, B32 is_local)
{
U64 buffer_size = sizeof(CV_SymRef2) + name.size + 1;
U8 *buffer = push_array_no_zero(arena, U8, buffer_size);
CV_SymRef2 *ref = (CV_SymRef2*)buffer;
ref->suc_name = 0;
ref->sym_off = stream_offset;
ref->imod = imod + 1; // MSVC adds one
U8 *name_ptr = (U8*)(ref + 1);
MemoryCopy(name_ptr, name.str, name.size);
name_ptr[name.size] = '\0';
CV_Symbol symbol;
symbol.kind = is_local ? CV_SymKind_LPROCREF : CV_SymKind_PROCREF;
symbol.data = str8(buffer, buffer_size);
symbol.offset = max_U64;
return symbol;
}
internal CV_Symbol
cv_make_pub32(Arena *arena, CV_Pub32Flags flags, U32 off, U16 isect, String8 name)
{
U64 buffer_size = sizeof(CV_SymPub32) + name.size + 1;
U8 *buffer = push_array_no_zero(arena, U8, buffer_size);
CV_SymPub32 *pub = (CV_SymPub32 *)buffer;
pub->flags = flags;
pub->off = off;
pub->sec = isect;
U8 *name_ptr = (U8*)(pub + 1);
MemoryCopy(name_ptr, name.str, name.size);
name_ptr[name.size] = '\0';
CV_Symbol symbol;
symbol.kind = CV_SymKind_PUB32;
symbol.data = str8(buffer, buffer_size);
return symbol;
}
internal CV_SymbolList
cv_make_proc_refs(Arena *arena, CV_ModIndex imod, CV_SymbolList symbol_list)
{
CV_SymbolList proc_ref_list = {0};
for (CV_SymbolNode *symbol_node = symbol_list.first; symbol_node != 0; symbol_node = symbol_node->next) {
CV_Symbol *symbol = &symbol_node->data;
if (symbol->kind == CV_SymKind_GPROC32) {
String8 name = cv_name_from_symbol(symbol->kind, symbol->data);
CV_Symbol ref = cv_make_proc_ref(arena, imod, safe_cast_u32(symbol->offset), name, /* is_local: */ 0);
CV_SymbolNode *proc_ref_node = cv_symbol_list_push(arena, &proc_ref_list);
proc_ref_node->data = ref;
} else if (symbol->kind == CV_SymKind_LPROC32) {
String8 name = cv_name_from_symbol(symbol->kind, symbol->data);
CV_Symbol ref = cv_make_proc_ref(arena, imod, safe_cast_u32(symbol->offset), name, /* is_local */ 1);
CV_SymbolNode *proc_ref_node = cv_symbol_list_push(arena, &proc_ref_list);
proc_ref_node->data = ref;
}
}
return proc_ref_list;
}
////////////////////////////////
//~ .debug$S helpers
internal void
cv_parse_debug_s_c13_(Arena *arena, CV_DebugS *debug_s, String8 raw_debug_s)
{
for (U64 cursor = 0; cursor + sizeof(CV_C13SubSectionHeader) <= raw_debug_s.size; ) {
// read header
CV_C13SubSectionHeader header = {0};
cursor += str8_deserial_read_struct(raw_debug_s, cursor, &header);
if (~header.kind & CV_C13SubSectionKind_IgnoreFlag) {
// pick sub-section list
U64 sub_sect_idx = cv_c13_sub_section_idx_from_kind(header.kind);
String8List *sub_sect_list = debug_s->data_list + sub_sect_idx;
// push data to sub-section
Rng1U64 sub_sect_range = r1u64(cursor, cursor + header.size);
String8 sub_sect_data = str8_substr(raw_debug_s, sub_sect_range);
str8_list_push(arena, sub_sect_list, sub_sect_data);
}
// advance
cursor += header.size;
cursor = AlignPow2(cursor, CV_C13SubSectionAlign);
}
}
internal CV_DebugS
cv_parse_debug_s_c13(Arena *arena, String8 raw_debug_s)
{
CV_DebugS debug_s = {0};
cv_parse_debug_s_c13_(arena, &debug_s, raw_debug_s);
return debug_s;
}
internal CV_DebugS
cv_parse_debug_s_c13_list(Arena *arena, String8List raw_debug_s)
{
CV_DebugS debug_s = {0};
for (String8Node *node = raw_debug_s.first; node != 0; node = node->next) {
cv_parse_debug_s_c13_(arena, &debug_s, node->string);
}
return debug_s;
}
internal CV_DebugS
cv_parse_debug_s(Arena *arena, String8 raw_debug_s)
{
CV_DebugS result; MemoryZeroStruct(&result);
if (raw_debug_s.size >= sizeof(CV_Signature)) {
CV_Signature sig = *(CV_Signature *)raw_debug_s.str;
switch (sig) {
case CV_Signature_C13: {
String8 raw_debug_s_past_sig = str8_substr(raw_debug_s, r1u64(sizeof(sig), raw_debug_s.size));
result = cv_parse_debug_s_c13(arena, raw_debug_s_past_sig);
} break;
case CV_Signature_C6: {
Assert(!"TODO: handle C6");
} break;
case CV_Signature_C7: {
Assert(!"TODO: handle C7");
} break;
case CV_Signature_C11: {
Assert(!"TODO: handle C11");
} break;
default: Assert(!"invalid signature"); break;
}
}
return result;
}
internal void
cv_debug_s_concat_in_place(CV_DebugS *dst, CV_DebugS *src)
{
for (U64 sub_sect_idx = 0; sub_sect_idx < ArrayCount(dst->data_list); sub_sect_idx += 1) {
str8_list_concat_in_place(&dst->data_list[sub_sect_idx], &src->data_list[sub_sect_idx]);
}
}
internal String8List
cv_data_c13_from_debug_s(Arena *arena, CV_DebugS *debug_s, B32 write_sig)
{
String8List srl = {0};
str8_serial_begin(arena, &srl);
if (write_sig) {
CV_Signature sig = CV_Signature_C13;
str8_serial_push_struct(arena, &srl, &sig);
}
static CV_C13SubSectionKind layout_arr[] = {
CV_C13SubSectionKind_Symbols,
//CV_C13SubSectionKind_Lines,
CV_C13SubSectionKind_FileChksms,
CV_C13SubSectionKind_FrameData,
CV_C13SubSectionKind_InlineeLines,
CV_C13SubSectionKind_IlLines,
CV_C13SubSectionKind_CrossScopeImports,
CV_C13SubSectionKind_CrossScopeExports,
CV_C13SubSectionKind_FuncMDTokenMap,
CV_C13SubSectionKind_TypeMDTokenMap,
CV_C13SubSectionKind_MergedAssemblyInput,
CV_C13SubSectionKind_CoffSymbolRVA,
CV_C13SubSectionKind_XfgHashType,
CV_C13SubSectionKind_XfgHashVirtual,
};
for (U64 layout_idx = 0; layout_idx < ArrayCount(layout_arr); layout_idx += 1) {
CV_C13SubSectionKind kind = layout_arr[layout_idx];
String8List *data = cv_sub_section_ptr_from_debug_s(debug_s, kind);
if (data->total_size > 0) {
U32 size32 = safe_cast_u32(data->total_size);
str8_serial_push_u32(arena, &srl, kind);
str8_serial_push_u32(arena, &srl, size32);
str8_serial_push_data_list(arena, &srl, data->first);
str8_serial_push_align(arena, &srl, 4);
}
}
String8List *line_data = cv_sub_section_ptr_from_debug_s(debug_s, CV_C13SubSectionKind_Lines);
for (String8Node *line_node = line_data->first; line_node != 0; line_node = line_node->next) {
str8_serial_push_u32(arena, &srl, CV_C13SubSectionKind_Lines);
str8_serial_push_u32(arena, &srl, safe_cast_u32(line_node->string.size));
str8_serial_push_string(arena, &srl, line_node->string);
str8_serial_push_align(arena, &srl, 4);
}
return srl;
}
internal CV_C13SubSectionIdxKind
cv_c13_sub_section_idx_from_kind(CV_C13SubSectionKind kind)
{
switch (kind) {
#define X(n,c) case CV_C13SubSectionKind_##n: return CV_C13SubSectionIdxKind_##n;
CV_C13SubSectionKindXList(X)
#undef X
}
return CV_C13SubSectionIdxKind_NULL;
}
internal String8List *
cv_sub_section_ptr_from_debug_s(CV_DebugS *debug_s, CV_C13SubSectionKind kind)
{
CV_C13SubSectionIdxKind idx = cv_c13_sub_section_idx_from_kind(kind);
return &debug_s->data_list[idx];
}
internal String8List
cv_sub_section_from_debug_s(CV_DebugS debug_s, CV_C13SubSectionKind kind)
{
String8List *list_ptr = cv_sub_section_ptr_from_debug_s(&debug_s, kind);
return *list_ptr;
}
internal String8
cv_string_table_from_debug_s(CV_DebugS debug_s)
{
String8List data_list = cv_sub_section_from_debug_s(debug_s, CV_C13SubSectionKind_StringTable);
String8 string_data = str8_zero();
if (data_list.node_count > 0) {
string_data = data_list.first->string;
}
return string_data;
}
internal String8
cv_file_chksms_from_debug_s(CV_DebugS debug_s)
{
String8List data_list = cv_sub_section_from_debug_s(debug_s, CV_C13SubSectionKind_FileChksms);
String8 file_chksms = str8_zero();
if (data_list.node_count > 0) {
file_chksms = data_list.first->string;
}
return file_chksms;
}
////////////////////////////////
//~ String Table Deduper
internal U64
cv_string_hash_table_hash(String8 string)
{
return hash_from_str8(string);
}
internal int
cv_string_bucket_is_before(void *raw_a, void *raw_b)
{
CV_StringBucket **a = raw_a;
CV_StringBucket **b = raw_b;
int is_before;
if ((*a)->u.idx0 == (*b)->u.idx0) {
is_before = (*a)->u.idx1 < (*b)->u.idx1;
} else {
is_before = (*a)->u.idx0 < (*b)->u.idx0;
}
return is_before;
}
internal CV_StringBucket *
cv_string_hash_table_insert_or_update(CV_StringBucket **buckets, U64 cap, U64 hash, CV_StringBucket *new_bucket)
{
CV_StringBucket *result = 0;
B32 was_bucket_inserted_or_updated = 0;
U64 best_idx = hash % cap;
U64 idx = best_idx;
do {
retry:;
CV_StringBucket *curr_bucket = buckets[idx];
if (curr_bucket == 0) {
CV_StringBucket *compare_bucket = ins_atomic_ptr_eval_cond_assign(&buckets[idx], new_bucket, curr_bucket);
if (compare_bucket == curr_bucket) {
// success, bucket was inserted
was_bucket_inserted_or_updated = 1;
break;
}
// another thread took the bucket...
goto retry;
} else if (str8_match(curr_bucket->string, new_bucket->string, 0)) {
if (cv_string_bucket_is_before(&curr_bucket, &new_bucket)) {
// recycle bucket
result = new_bucket;
// don't need to update, more recent leaf is in the bucket
was_bucket_inserted_or_updated = 1;
break;
}
CV_StringBucket *compare_bucket = ins_atomic_ptr_eval_cond_assign(&buckets[idx], new_bucket, curr_bucket);
if (compare_bucket == curr_bucket) {
// recycle bucket
result = compare_bucket;
// new bucket is in the hash table, exit
was_bucket_inserted_or_updated = 1;
break;
}
// another thread took the bucket...
goto retry;
}
// advance
idx = (idx + 1) % cap;
} while (idx != best_idx);
// are there enough free buckets?
Assert(was_bucket_inserted_or_updated);
return result;
}
internal
THREAD_POOL_TASK_FUNC(cv_count_strings_in_debug_s_arr_task)
{
ProfBeginFunction();
CV_DedupStringTablesTask *task = raw_task;
CV_StringTableRange *range_list = task->range_lists[task_id];
for (CV_StringTableRange *range_n = range_list; range_n != 0; range_n = range_n->next) {
CV_DebugS debug_s = task->arr[range_n->debug_s_idx];
String8 string_buffer = cv_string_table_from_debug_s(debug_s);
Assert(range_n->range.min <= range_n->range.max);
Assert(range_n->range.min <= string_buffer.size);
Assert(range_n->range.max <= string_buffer.size);
U64 count = 0;
for (U64 i = range_n->range.min; i < range_n->range.max; ++i) {
U8 b = string_buffer.str[i];
if (b == '\0') {
count += 1;
}
}
ins_atomic_u64_add_eval(&task->string_counts[range_n->debug_s_idx], count);
}
ProfEnd();
}
internal
THREAD_POOL_TASK_FUNC(cv_dedup_strings_in_debug_s_arr_task)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
U64 debug_s_idx = task_id;
CV_DedupStringTablesTask *task = raw_task;
CV_DebugS debug_s = task->arr[debug_s_idx];
String8 string_table = cv_string_table_from_debug_s(debug_s);
String8List strings_list = str8_split_by_string_chars(scratch.arena, string_table, str8_lit("\0"), 0);
CV_StringBucket *bucket = 0;
U64 total_string_size = 0;
U64 total_insert_count = 0;
U64 string_idx = 0;
for (String8Node *string_n = strings_list.first; string_n != 0; string_n = string_n->next, ++string_idx) {
if (bucket == 0) {
bucket = push_array_no_zero(arena, CV_StringBucket, 1);
}
bucket->u.idx0 = debug_s_idx;
bucket->u.idx1 = string_idx;
bucket->string = string_n->string;
U64 hash = cv_string_hash_table_hash(string_n->string);
CV_StringBucket *insert_or_update = cv_string_hash_table_insert_or_update(task->buckets, task->bucket_cap, hash, bucket);
if (insert_or_update == 0) {
total_string_size += string_n->string.size;
total_insert_count += 1;
}
if (insert_or_update != bucket) {
bucket = 0;
}
}
ins_atomic_u64_add_eval(&task->total_string_size, total_string_size);
ins_atomic_u64_add_eval(&task->total_insert_count, total_insert_count);
scratch_end(scratch);
ProfEnd();
}
internal CV_StringHashTable
cv_dedup_string_tables(TP_Arena *arena, TP_Context *tp, U64 count, CV_DebugS *arr)
{
ProfBeginFunction();
Temp scratch = scratch_begin(arena->v, arena->count);
ProfBegin("Compute Total Weight");
U64 total_weight = 0;
for (U64 i = 0; i < count; ++i) {
String8 string_table = cv_string_table_from_debug_s(arr[i]);
total_weight += string_table.size;
}
ProfEnd();
U64 per_task_weight = CeilIntegerDiv(total_weight, tp->worker_count);
U64 task_weight = 0;
U64 task_id = 0;
CV_StringTableRange **range_lists = push_array(scratch.arena, CV_StringTableRange *, tp->worker_count);
ProfBegin("Divide Work");
for (U64 debug_s_idx = 0; debug_s_idx < count; ++debug_s_idx) {
String8 string_table = cv_string_table_from_debug_s(arr[debug_s_idx]);
for (U64 cursor = 0; cursor < string_table.size; cursor += per_task_weight) {
if (task_weight >= per_task_weight) {
task_id = (task_id + 1) % tp->worker_count;
task_weight = 0;
}
U64 max_range_weight = Min(per_task_weight, string_table.size - cursor);
CV_StringTableRange *node = push_array(scratch.arena, CV_StringTableRange, 1);
node->range = rng_1u64(cursor, cursor + max_range_weight);
node->debug_s_idx = debug_s_idx;
SLLStackPush(range_lists[task_id], node);
task_weight += max_range_weight;
}
}
ProfEnd();
ProfBegin("Count");
CV_DedupStringTablesTask task = {0};
task.arr = arr;
task.range_lists = range_lists;
task.string_counts = push_array(scratch.arena, U64, count);
tp_for_parallel(tp, 0, tp->worker_count, cv_count_strings_in_debug_s_arr_task, &task);
ProfEnd();
ProfBegin("Dedup");
U64 total_string_count = sum_array_u64(count, task.string_counts);
task.bucket_cap = (U64)((F64)total_string_count * 1.3);
task.buckets = push_array(arena->v[0], CV_StringBucket *, task.bucket_cap);
tp_for_parallel(tp, arena, count, cv_dedup_strings_in_debug_s_arr_task, &task);
ProfEnd();
CV_StringHashTable string_ht = {0};
string_ht.total_string_size = task.total_string_size;
string_ht.total_insert_count = task.total_insert_count;
string_ht.bucket_cap = task.bucket_cap;
string_ht.buckets = task.buckets;
scratch_end(scratch);
ProfEnd();
return string_ht;
}
internal void
cv_string_hash_table_assign_buffer_offsets(TP_Context *tp, CV_StringHashTable string_ht)
{
ProfBeginFunction();
Temp scratch = scratch_begin(0,0);
ProfBegin("Count Strings");
U64 string_count = 0;
for (U64 i = 0; i < string_ht.bucket_cap; ++i) {
if (string_ht.buckets[i] != 0) {
string_count += 1;
}
}
ProfEnd();
ProfBegin("Push");
CV_StringBucket **strings = push_array_no_zero(scratch.arena, CV_StringBucket *, string_count);
ProfEnd();
ProfBegin("Copy Present Buckets");
for (U64 i = 0, string_idx = 0; i < string_ht.bucket_cap; ++i) {
if (string_ht.buckets[i] != 0) {
strings[string_idx++] = string_ht.buckets[i];
}
}
ProfEnd();
ProfBegin("Sort");
radsort(strings, string_count, cv_string_bucket_is_before);
ProfEnd();
ProfBegin("Assign Offsets");
for (U64 i = 0, offset_cursor = 0; i < string_count; ++i) {
CV_StringBucket *s = strings[i];
s->u.offset = offset_cursor;
offset_cursor += s->string.size + 1;
}
ProfEnd();
scratch_end(scratch);
ProfEnd();
}
internal CV_StringBucket *
cv_string_hash_table_lookup(CV_StringHashTable ht, String8 string)
{
U64 hash = cv_string_hash_table_hash(string);
U64 best_idx = hash % ht.bucket_cap;
U64 idx = best_idx;
do {
if (ht.buckets[idx] == 0) {
break;
}
if (str8_match(ht.buckets[idx]->string, string, 0)) {
return ht.buckets[idx];
}
idx = (idx + 1) % ht.bucket_cap;
} while (idx != best_idx);
return 0;
}
internal
THREAD_POOL_TASK_FUNC(cv_pack_string_hash_table_task)
{
ProfBeginFunction();
CV_PackStringHashTableTask *task = raw_task;
Rng1U64 range = task->ranges[task_id];
for (U64 bucket_idx = range.min; bucket_idx < range.max; ++bucket_idx) {
CV_StringBucket *bucket = task->buckets[bucket_idx];
if (bucket) {
MemoryCopy(task->buffer + bucket->u.offset, bucket->string.str, bucket->string.size);
task->buffer[bucket->u.offset + bucket->string.size] = '\0';
}
}
ProfEnd();
}
internal String8
cv_pack_string_hash_table(Arena *arena, TP_Context *tp, CV_StringHashTable string_ht)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
U64 buffer_size = string_ht.total_string_size + /* nulls: */ string_ht.total_insert_count;
U8 *buffer = push_array_no_zero(arena, U8, buffer_size);
CV_PackStringHashTableTask task = {0};
task.buckets = string_ht.buckets;
task.buffer = buffer;
task.ranges = tp_divide_work(scratch.arena, string_ht.bucket_cap, tp->worker_count);
tp_for_parallel(tp, 0, tp->worker_count, cv_pack_string_hash_table_task, &task);
String8 result = str8(buffer, buffer_size);
scratch_end(scratch);
ProfEnd();
return result;
}
////////////////////////////////
//~ Symbol Deduper
internal int
cv_symbol_deduper_is_before(void *raw_a, void *raw_b)
{
return raw_a < raw_b;
}
internal CV_SymbolNode **
cv_symbol_deduper_insert_or_update(CV_SymbolNode ***buckets, U64 cap, U64 hash, CV_SymbolNode **new_bucket)
{
CV_SymbolNode **result = 0;
B32 is_inserted_or_updated = 0;
U64 best_idx = hash % cap;
U64 idx = best_idx;
do {
retry:;
CV_SymbolNode **curr_bucket = buckets[idx];
Assert(curr_bucket != new_bucket);
if (curr_bucket == 0) {
CV_SymbolNode **compare_bucket = ins_atomic_ptr_eval_cond_assign(&buckets[idx], new_bucket, curr_bucket);
if (compare_bucket == curr_bucket) {
// success, bucket was inserted
is_inserted_or_updated = 1;
break;
}
// another thread took the bucket...
goto retry;
} else if ((*curr_bucket)->data.kind == (*new_bucket)->data.kind &&
(*curr_bucket)->data.data.size == (*new_bucket)->data.data.size &&
MemoryMatch((*curr_bucket)->data.data.str, (*new_bucket)->data.data.str, (*new_bucket)->data.data.size)) {
if (cv_symbol_deduper_is_before(curr_bucket, new_bucket)) {
result = new_bucket;
is_inserted_or_updated = 1;
// don't need to update, more recent leaf is in the bucket
break;
}
CV_SymbolNode **compare_bucket = ins_atomic_ptr_eval_cond_assign(&buckets[idx], new_bucket, curr_bucket);
if (compare_bucket == curr_bucket) {
result = compare_bucket;
is_inserted_or_updated = 1;
break;
}
// another thread took the bucket...
goto retry;
}
// advance
idx = (idx + 1) % cap;
} while (idx != best_idx);
Assert(is_inserted_or_updated);
return result;
}
internal
THREAD_POOL_TASK_FUNC(cv_symbol_deduper_insert_task)
{
ProfBeginFunction();
CV_SymbolDeduperTask *task = raw_task;
Rng1U64 range = task->ranges[task_id];
for (U64 symbol_idx = range.min; symbol_idx < range.max; ++symbol_idx) {
CV_SymbolNode **symbol_node = &task->symbols[symbol_idx];
U64 hash = hash_from_cv_symbol(&(*symbol_node)->data);
cv_symbol_deduper_insert_or_update(task->u.buckets, task->cap, hash, symbol_node);
}
ProfEnd();
}
internal
THREAD_POOL_TASK_FUNC(cv_symbol_deduper_deref_buckets_task)
{
ProfBeginFunction();
CV_SymbolDeduperTask *task = raw_task;
Rng1U64 range = task->ranges[task_id];
for (U64 bucket_idx = range.min; bucket_idx < range.max; ++bucket_idx) {
CV_SymbolNode **bucket = task->u.buckets[bucket_idx];
if (bucket) {
task->u.deref_buckets[bucket_idx] = *bucket;
}
}
ProfEnd();
}
internal void
cv_dedup_symbol_ptr_array(TP_Context *tp, CV_SymbolPtrArray *symbols)
{
ProfBeginDynamic("Dedup Symbols [Count %llu]", symbols->count);
Temp scratch = scratch_begin(0, 0);
ProfBegin("Setup Task");
CV_SymbolDeduperTask task = {0};
task.symbols = symbols->v;
task.cap = (U64)((F64)symbols->count * 1.3);
task.u.buckets = push_array(scratch.arena, CV_SymbolNode **, task.cap);
ProfEnd();
ProfBegin("Dedup");
task.ranges = tp_divide_work(scratch.arena, symbols->count, tp->worker_count);
tp_for_parallel(tp, 0, tp->worker_count, cv_symbol_deduper_insert_task, &task);
ProfEnd();
ProfBegin("Deref Buckets");
task.ranges = tp_divide_work(scratch.arena, task.cap, tp->worker_count);
tp_for_parallel(tp, 0, tp->worker_count, cv_symbol_deduper_deref_buckets_task, &task);
ProfEnd();
ProfBegin("Copy Extant Buckets");
U64 unique_symbol_count = 0;
for (U64 bucket_idx = 0; bucket_idx < task.cap; ++bucket_idx) {
CV_SymbolNode *bucket = task.u.deref_buckets[bucket_idx];
if (bucket) {
symbols->v[unique_symbol_count++] = bucket;
}
}
ProfEnd();
Assert(unique_symbol_count <= symbols->count);
symbols->count = unique_symbol_count;
ProfBeginDynamic("Sort [Count %llu]", symbols->count);
radsort(symbols->v, symbols->count, cv_symbol_deduper_is_before);
ProfEnd();
scratch_end(scratch);
ProfEnd();
}
////////////////////////////////
//~ .debug$T helpers
internal CV_DebugT
cv_debug_t_from_data_arr(Arena *arena, String8Array data_arr, U64 align)
{
ProfBegin("Upfront parse");
U64 max_leaf_count = 0;
for (U64 data_idx = 0; data_idx < data_arr.count; data_idx += 1) {
String8 data = data_arr.v[data_idx];
for (U64 cursor = 0; cursor < data.size; ) {
CV_Leaf leaf;
cursor += cv_deserial_leaf(data, cursor, align, &leaf);
max_leaf_count += 1;
}
}
ProfEnd();
U8 **leaf_arr = push_array_no_zero(arena, U8 *, max_leaf_count);
U64 leaf_count = 0;
for (U64 data_idx = 0; data_idx < data_arr.count; data_idx += 1) {
String8 data = data_arr.v[data_idx];
U64 cursor = 0;
while (cursor < data.size) {
CV_Leaf leaf;
U64 read_size = cv_deserial_leaf(data, cursor, align, &leaf);
Assert(leaf_count < max_leaf_count);
leaf_arr[leaf_count] = str8_deserial_get_raw_ptr(data, cursor, read_size);
leaf_count += 1;
// advance cursor
cursor += read_size;
}
}
CV_DebugT debug_t = {0};
debug_t.count = leaf_count;
debug_t.v = leaf_arr;
return debug_t;
}
internal CV_DebugT
cv_debug_t_from_data(Arena *arena, String8 data, U64 align)
{
String8Array arr = {0};
arr.count = 1;
arr.v = &data;
return cv_debug_t_from_data_arr(arena, arr, align);
}
internal CV_Leaf
cv_debug_t_get_leaf(CV_DebugT debug_t, U64 leaf_idx)
{
Assert(leaf_idx < debug_t.count);
U8 *ptr = debug_t.v[leaf_idx];
String8 data = str8(ptr, max_U64);
CV_Leaf leaf;
cv_deserial_leaf(data, 0, 1, &leaf);
U64 size = cv_header_struct_size_from_leaf_kind(leaf.kind);
Assert(size <= leaf.data.size);
return leaf;
}
internal String8
cv_debug_t_get_raw_leaf(CV_DebugT debug_t, U64 leaf_idx)
{
Assert(leaf_idx < debug_t.count);
U8 *leaf_ptr = debug_t.v[leaf_idx];
CV_LeafSize *size_ptr = (CV_LeafSize *)leaf_ptr;
CV_LeafSize total_size = sizeof(*size_ptr) + *size_ptr;
String8 raw_leaf = str8(leaf_ptr, total_size);
return raw_leaf;
}
internal CV_LeafHeader *
cv_debug_t_get_leaf_header(CV_DebugT debug_t, U64 leaf_idx)
{
Assert(leaf_idx < debug_t.count);
CV_LeafHeader *leaf_header = (CV_LeafHeader *) debug_t.v[leaf_idx];
return leaf_header;
}
internal B32
cv_debug_t_is_pch(CV_DebugT debug_t)
{
if (debug_t.count > 0) {
CV_Leaf leaf = cv_debug_t_get_leaf(debug_t, 0);
return cv_is_leaf_pch(leaf.kind);
}
return 0;
}
internal B32
cv_debug_t_is_type_server(CV_DebugT debug_t)
{
if (debug_t.count > 0) {
CV_Leaf leaf = cv_debug_t_get_leaf(debug_t, 0);
return cv_is_leaf_type_server(leaf.kind);
}
return 0;
}
internal U64
cv_debug_t_array_count_leaves(U64 count, CV_DebugT *arr)
{
U64 total_leaf_count = 0;
for (U64 i = 0; i < count; i += 1) {
total_leaf_count += arr[i].count;
}
return total_leaf_count;
}
THREAD_POOL_TASK_FUNC(cv_str8_list_from_debug_t_task)
{
CV_Str8ListFromDebugT *task = raw_task;
for (U64 leaf_idx = task->ranges[task_id].min; leaf_idx < task->ranges[task_id].max; ++leaf_idx) {
String8Node *node = &task->nodes[leaf_idx];
node->string = cv_debug_t_get_raw_leaf(task->debug_t, leaf_idx);
str8_list_push_node(&task->lists[task_id], node);
}
}
internal String8List
cv_str8_list_from_debug_t_parallel(TP_Context *tp, Arena *arena, CV_DebugT debug_t)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
// build lists in parallel
CV_Str8ListFromDebugT task = {0};
task.debug_t = debug_t;
task.ranges = tp_divide_work(scratch.arena, debug_t.count, tp->worker_count);
task.lists = push_array(scratch.arena, String8List, tp->worker_count);
task.nodes = push_array_no_zero(arena, String8Node, debug_t.count);
tp_for_parallel(tp, 0, tp->worker_count, cv_str8_list_from_debug_t_task, &task);
// concat output lists
String8List list = {0};
for (U64 task_id = 0; task_id < tp->worker_count; ++task_id) {
str8_list_concat_in_place(&list, &task.lists[task_id]);
}
scratch_end(scratch);
ProfEnd();
return list;
}
// $$Symbols
internal void
cv_parse_symbol_sub_section(Arena *arena, CV_SymbolList *list, U64 offset_base, String8 data, U64 align)
{
for (U64 cursor = 0, opl = data.size; cursor < opl; ) {
// read symbol header
CV_SymbolHeader header;
cursor += str8_deserial_read_struct(data, cursor, &header);
// size from header has to be larger than 2 bytes
if (header.size < sizeof(header.kind)) {
Assert(!"TODO: error handle invalid symbol data");
break;
}
// is there enough bytes in the range?
U64 symbol_opl = cursor + (header.size - sizeof(header.kind));
if (symbol_opl > opl) {
Assert(!"TODO: error handle corrupted symbol data");
break;
}
// get symbol data
Rng1U64 symbol_data_range = r1u64(cursor, symbol_opl);
String8 symbol_data = str8_substr(data, symbol_data_range);
// init symbol
CV_SymbolNode *node = cv_symbol_list_push(arena, list);
node->data.offset = offset_base + cursor;
node->data.kind = header.kind;
node->data.data = symbol_data;
// advance cursor
cursor = symbol_opl;
cursor = AlignPow2(cursor, align);
}
}
internal CV_SymbolList
cv_symbol_list_from_data_list(Arena *arena, String8List data_list, U64 align)
{
CV_SymbolList symbol_list = {0};
U64 cursor = 0;
for (String8Node *sect = data_list.first; sect != 0; cursor += sect->string.size, sect = sect->next) {
cv_parse_symbol_sub_section(arena, &symbol_list, cursor, sect->string, align);
}
return symbol_list;
}
internal void
cv_symbol_list_push_node(CV_SymbolList *list, CV_SymbolNode *node)
{
node->prev = 0;
node->next = 0;
DLLPushBack(list->first, list->last, node);
list->count += 1;
}
internal CV_SymbolNode *
cv_symbol_list_push(Arena *arena, CV_SymbolList *list)
{
CV_SymbolNode *node = push_array(arena, CV_SymbolNode, 1);
cv_symbol_list_push_node(list, node);
return node;
}
internal CV_SymbolNode *
cv_symbol_list_push_data(Arena *arena, CV_SymbolList *list, CV_SymKind kind, String8 data)
{
CV_SymbolNode *node = cv_symbol_list_push(arena, list);
node->data.kind = kind;
node->data.data = data;
return node;
}
internal CV_SymbolNode *
cv_symbol_list_push_many(Arena *arena, CV_SymbolList *list, U64 count)
{
CV_SymbolNode *node_arr = push_array_no_zero(arena, CV_SymbolNode, 1);
for (U64 node_idx = 0; node_idx < count; node_idx += 1) {
cv_symbol_list_push_node(list, &node_arr[node_idx]);
}
return node_arr;
}
internal void
cv_symbol_list_remove_node(CV_SymbolList *list, CV_SymbolNode *node)
{
Assert(list->count > 0);
list->count -= 1;
DLLRemove(list->first, list->last, node);
}
internal void
cv_symbol_list_concat_in_place(CV_SymbolList *list, CV_SymbolList *to_concat)
{
SLLConcatInPlace(list, to_concat);
}
internal void
cv_symbol_list_concat_in_place_arr(CV_SymbolList *list, U64 count, CV_SymbolList *to_concat)
{
SLLConcatInPlaceArray(list, to_concat, count);
}
internal U64
cv_symbol_list_arr_get_count(U64 count, CV_SymbolList *list_arr)
{
U64 result = 0;
for (U64 idx = 0; idx < count; idx += 1) {
result += list_arr[idx].count;
}
return result;
}
internal String8List
cv_data_from_symbol_list(Arena *arena, CV_SymbolList symbol_list, U64 align)
{
String8List data_list = {0};
for (CV_SymbolNode *node = symbol_list.first; node != 0; node = node->next) {
String8 data = cv_serialize_symbol(arena, &node->data, align);
str8_list_push(arena, &data_list, data);
}
return data_list;
}
internal
THREAD_POOL_TASK_FUNC(cv_symbol_list_syncer)
{
ProfBeginFunction();
CV_SymbolListSyncer *task = raw_task;
// context shortcuts
Rng1U64 list_range = task->list_range_arr[task_id];
U64 symbol_base = task->symbol_base_arr[task_id];
for (U64 list_idx = list_range.min, symbol_idx = symbol_base; list_idx < list_range.max; list_idx += 1) {
// pick up assigned list
CV_SymbolList list = task->list_arr[list_idx];
// fill out assigned range in the symbol array
for (CV_SymbolNode *node = list.first; node != 0; node = node->next, symbol_idx += 1) {
task->symbol_arr[symbol_idx] = node;
}
}
ProfEnd();
}
internal CV_SymbolPtrArray
cv_symbol_ptr_array_from_list(Arena *arena, TP_Context *tp, U64 count, CV_SymbolList *list_arr)
{
ProfBeginFunction();
Temp scratch = scratch_begin(&arena, 1);
U64 total_count = cv_symbol_list_arr_get_count(count, list_arr);
CV_SymbolListSyncer task = {0};
task.list_arr = list_arr;
task.symbol_arr = push_array_no_zero(arena, CV_SymbolNode *, total_count);
task.symbol_base_arr = push_array_no_zero(scratch.arena, U64, tp->worker_count);
task.list_range_arr = tp_divide_work(scratch.arena, count, tp->worker_count);
for (U64 thread_idx = 0, symbol_base = 0; thread_idx < tp->worker_count; thread_idx += 1) {
task.symbol_base_arr[thread_idx] = symbol_base;
Rng1U64 range = task.list_range_arr[thread_idx];
for (U64 list_idx = range.min; list_idx < range.max; list_idx += 1) {
symbol_base += list_arr[list_idx].count;
}
}
tp_for_parallel(tp, 0, tp->worker_count, cv_symbol_list_syncer, &task);
CV_SymbolPtrArray result = {0};
result.count = total_count;
result.v = task.symbol_arr;
scratch_end(scratch);
ProfEnd();
return result;
}
internal CV_Scope *
cv_scope_list_push(Arena *arena, CV_ScopeList *list)
{
CV_Scope *node = push_array(arena, CV_Scope, 1);
SLLQueuePush(list->first, list->last, node);
return node;
}
internal CV_SymbolList
cv_global_scope_symbols_from_list(Arena *arena, CV_SymbolList list)
{
CV_SymbolList gsym_list = {0};
S64 scope_depth = 0;
for (CV_SymbolNode *symbol_n = list.first; symbol_n != 0; symbol_n = symbol_n->next) {
CV_Symbol symbol = symbol_n->data;
if (cv_is_global_symbol(symbol.kind) && scope_depth == 0) {
cv_symbol_list_push_data(arena, &gsym_list, symbol.kind, symbol.data);
} else if (cv_is_scope_symbol(symbol.kind)) {
scope_depth += 1;
} else if (cv_is_end_symbol(symbol.kind)) {
scope_depth -= 1;
if (scope_depth < 0) {
break;
}
}
}
return gsym_list;
}
internal CV_ScopeList
cv_symbol_tree_from_symbol_list(Arena *arena, CV_SymbolList list)
{
Temp scratch = scratch_begin(&arena, 1);
CV_ScopeList root = {0};
// setup root frame
CV_ScopeFrame *stack = push_array(scratch.arena, CV_ScopeFrame, 1);
stack->list = &root;
for (CV_SymbolNode *symbol_node = list.first; symbol_node != 0; symbol_node = symbol_node->next) {
// store symbol in current scope
CV_Scope *scope = cv_scope_list_push(arena, stack->list);
scope->symbol = symbol_node->data;
// does this symbol define a new scope?
if (cv_is_scope_symbol(symbol_node->data.kind)) {
CV_ScopeFrame *frame = push_array(scratch.arena, CV_ScopeFrame, 1);
frame->list = push_array(arena, CV_ScopeList, 1);
SLLStackPush(stack, frame);
}
// does this symbol end current scope?
else if (cv_is_end_symbol(symbol_node->data.kind)) {
CV_ScopeFrame *prev_stack_frame = stack->next;
if (prev_stack_frame) {
// set children in parent scope
CV_Scope *parent_scope = prev_stack_frame->list->last;
parent_scope->children = stack->list;
}
// pop frame
SLLStackPop(stack);
}
}
scratch_end(scratch);
return root;
}
internal U64
cv_patch_symbol_tree_offsets(CV_SymbolList list, U64 base_offset, U64 align)
{
Temp scratch = scratch_begin(0, 0);
struct Stack {
struct Stack *next;
CV_Symbol *symbol;
U64 offset;
};
struct Stack *stack = 0;
struct Stack *free_list = 0;
U64 cursor = base_offset;
for (CV_SymbolNode *symbol_n = list.first; symbol_n != 0; symbol_n = symbol_n->next) {
CV_Symbol symbol = symbol_n->data;
if (cv_is_scope_symbol(symbol.kind)) {
// NOTE: We don't patch 'next' offset in PROC symbols because
// it's not used by visual studio and MSVC leaves the offsets
// zeroed. LLD is on the same page.
Assert(symbol.data.size >= sizeof(U32)*2);
// patch symbol parent
if (stack) {
U32 *parent_off_ptr = (U32 *)symbol.data.str;
*parent_off_ptr = stack->offset;
}
// reuse/alloc frame
struct Stack *frame;
if (free_list) {
frame = free_list;
SLLStackPop(free_list);
} else {
frame = push_array_no_zero(scratch.arena, struct Stack, 1);
}
// push frame to the stack
frame->symbol = &symbol_n->data;
frame->offset = cursor;
SLLStackPush(stack, frame);
} else if (cv_is_end_symbol(symbol.kind)) {
// patch symbol end
U32 *end_off_ptr = (U32 *)stack->symbol->data.str + /* skip parent off */ 1;
*end_off_ptr = cursor;
// recycle frame
struct Stack *free_frame = stack;
SLLStackPop(stack);
SLLStackPush(free_list, free_frame);
}
// advance cursor
cursor += cv_compute_symbol_record_size(&symbol, align);
}
scratch_end(scratch);
U64 serial_size = cursor - base_offset;
return serial_size;
}
// $$FileChksms
internal void
cv_parse_checksum_data(Arena *arena, CV_ChecksumList *list, String8 checksum_data)
{
for (U64 cursor = 0, cursor_opl = checksum_data.size; cursor < cursor_opl; ) {
U64 expected_cursor_after_checksum = cursor + sizeof(CV_C13Checksum);
if (expected_cursor_after_checksum > cursor_opl) {
break;
}
CV_C13Checksum *header = (CV_C13Checksum *)str8_deserial_get_raw_ptr(checksum_data, cursor, sizeof(CV_C13Checksum));
cursor += sizeof(CV_C13Checksum);
U64 expected_cursor_after_value = cursor + header->len;
if (expected_cursor_after_value > cursor_opl) {
break;
}
String8 value = str8(0,0);
cursor += str8_deserial_read_block(checksum_data, cursor, header->len, &value);
cursor = AlignPow2(cursor, 4);
CV_ChecksumNode *node = push_array(arena, CV_ChecksumNode, 1);
node->next = 0;
CV_Checksum *data = &node->data;
data->header = header;
data->value = value;
SLLQueuePush(list->first, list->last, node);
list->count += 1;
}
}
internal CV_ChecksumList
cv_c13_parse_checksum_data_list(Arena *arena, String8List checksum_data_list)
{
CV_ChecksumList result = {0};
for (String8Node *node = checksum_data_list.first; node != 0; node = node->next) {
cv_parse_checksum_data(arena, &result, node->string);
}
return result;
}
internal void
cv_c13_patch_string_offsets_in_checksum_list(CV_ChecksumList checksum_list, String8 string_data, U64 string_data_base_offset, CV_StringHashTable string_ht)
{
for (CV_ChecksumNode *node = checksum_list.first; node != 0; node = node->next) {
CV_Checksum *checksum = &node->data;
CV_C13Checksum *header = checksum->header;
String8 name = str8_cstring_capped(string_data.str + header->name_off, string_data.str + string_data.size);
CV_StringBucket *bucket = cv_string_hash_table_lookup(string_ht, name);
U64 name_off64 = string_data_base_offset + bucket->u.offset;
header->name_off = safe_cast_u32(name_off64);
}
}
internal String8List
cv_c13_collect_source_file_names(Arena *arena, CV_ChecksumList checksum_list, String8 string_data)
{
String8List source_file_name_list = {0};
for (CV_ChecksumNode *node = checksum_list.first; node != 0; node = node->next) {
CV_Checksum *checksum = &node->data;
CV_C13Checksum *header = checksum->header;
Assert(header->name_off < string_data.size);
String8 name = str8_cstring_capped(string_data.str + header->name_off, string_data.str + string_data.size);
str8_list_push(arena, &source_file_name_list, name);
}
return source_file_name_list;
}
// $$Lines
internal CV_C13LinesHeaderList
cv_c13_lines_from_sub_sections(Arena *arena, String8 c13_data, Rng1U64 ss_range)
{
ProfBeginFunction();
CV_C13LinesHeaderList parsed_line_list = {0};
String8 sub_sect_data = str8_substr(c13_data, ss_range);
for (U64 cursor = 0; cursor + sizeof(CV_C13SubSecLinesHeader) <= sub_sect_data.size; ) {
CV_C13SubSecLinesHeader *hdr = (CV_C13SubSecLinesHeader *)(sub_sect_data.str + cursor);
cursor += sizeof(*hdr);
// read files
for (; cursor + sizeof(CV_C13File) <= sub_sect_data.size; ) {
// grab next file header
CV_C13File *file = (CV_C13File *)(sub_sect_data.str + cursor);
cursor += sizeof(CV_C13File);
// parse lines and columns
//
// TODO: export columns
U64 max_line_count = (sub_sect_data.size - cursor) / sizeof(CV_C13Line);
U32 line_count = Min(file->num_lines, max_line_count);
// TODO(allen): check order correctness here
U64 line_array_off = cursor;
//U64 col_array_off = line_array_off + line_count * sizeof(CV_C13Line);
// compute line entry size
U64 line_entry_size = sizeof(CV_C13Line);
if (hdr->flags & CV_C13SubSecLinesFlag_HasColumns) {
line_entry_size += sizeof(CV_C13Column);
}
// advance past line and column entries
cursor += line_count * line_entry_size;
// emit parsed lines
CV_C13LinesHeaderNode *lines_parsed_node = push_array_no_zero(arena, CV_C13LinesHeaderNode, 1);
lines_parsed_node->next = 0;
CV_C13LinesHeader *lines_parsed = &lines_parsed_node->v;
lines_parsed->sec_idx = hdr->sec;
lines_parsed->sec_off_lo = hdr->sec_off;
lines_parsed->sec_off_hi = hdr->sec_off + hdr->len;
lines_parsed->file_off = file->file_off;
lines_parsed->line_count = line_count;
lines_parsed->col_count = 0; // TODO: columns
lines_parsed->line_array_off = ss_range.min + line_array_off;
lines_parsed->col_array_off = 0; // TODO: columns
SLLQueuePush(parsed_line_list.first, parsed_line_list.last, lines_parsed_node);
parsed_line_list.count += 1;
}
}
ProfEnd();
return parsed_line_list;
}
internal CV_LineArray
cv_c13_line_array_from_data(Arena *arena, String8 c13_data, U64 sec_base, CV_C13LinesHeader parsed_lines)
{
CV_LineArray result;
result.file_off = parsed_lines.file_off;
result.line_count = parsed_lines.line_count;
result.col_count = parsed_lines.col_count;
result.voffs = push_array_no_zero(arena, U64, parsed_lines.line_count + 1);
result.line_nums = push_array_no_zero(arena, U32, parsed_lines.line_count);
result.col_nums = 0;
CV_C13Line *raw_lines = (CV_C13Line *)str8_deserial_get_raw_ptr(c13_data, parsed_lines.line_array_off, parsed_lines.line_count * sizeof(raw_lines[0]));
for(U64 line_idx = 0; line_idx < parsed_lines.line_count; line_idx += 1)
{
CV_C13Line line = raw_lines[line_idx];
result.voffs[line_idx] = sec_base + parsed_lines.sec_off_lo + line.off;
result.line_nums[line_idx] = CV_C13LineFlags_ExtractLineNumber(line.flags);
}
// emit voff ender
result.voffs[result.line_count] = sec_base + parsed_lines.sec_off_hi;
return result;
}
internal void
cv_c13_patch_checksum_offsets_in_line_data_list(String8List line_data, U64 checksum_rebase)
{
for(String8Node *node = line_data.first; node != 0; node = node->next)
{
String8 raw_data = node->string;
if(raw_data.size < sizeof(CV_C13SubSecLinesHeader))
{
Assert(!"unable to patch checksum in line sub seciton header");
continue;
}
CV_C13File *file_header = (CV_C13File *)(raw_data.str + sizeof(CV_C13SubSecLinesHeader));
U64 rebased_file_off = file_header->file_off + checksum_rebase;
file_header->file_off = safe_cast_u32(rebased_file_off);
}
}
// $$InlineeLines
internal CV_C13InlineeLinesParsedList
cv_c13_inlinee_lines_from_sub_sections(Arena *arena, String8List raw_inlinee_lines)
{
ProfBeginFunction();
CV_C13InlineeLinesParsedList inlinee_lines_list = {0};
for (String8Node *raw_data_node = raw_inlinee_lines.first; raw_data_node != 0; raw_data_node = raw_data_node->next) {
U64 cursor = 0;
CV_C13InlineeLinesSig sig = 0;
cursor += str8_deserial_read_struct(raw_data_node->string, cursor, &sig);
for (; cursor + sizeof(CV_C13InlineeSourceLineHeader) <= raw_data_node->string.size; ) {
CV_C13InlineeSourceLineHeader *hdr = (CV_C13InlineeSourceLineHeader *)(raw_data_node->string.str + cursor);
cursor += sizeof(*hdr);
CV_C13InlineeLinesParsedNode *inlinee_parsed_node = push_array_no_zero(arena, CV_C13InlineeLinesParsedNode, 1);
inlinee_parsed_node->next = 0;
SLLQueuePush(inlinee_lines_list.first, inlinee_lines_list.last, inlinee_parsed_node);
inlinee_lines_list.count += 1;
CV_C13InlineeLinesParsed *inlinee_parsed = &inlinee_parsed_node->v;
inlinee_parsed->inlinee = hdr->inlinee;
inlinee_parsed->file_off = hdr->file_off;
inlinee_parsed->first_source_ln = hdr->first_source_ln;
inlinee_parsed->extra_file_count = 0;
inlinee_parsed->extra_files = 0;
if (sig == CV_C13InlineeLinesSig_EXTRA_FILES) {
if (cursor + sizeof(U32) <= raw_data_node->string.size) {
U32 *extra_file_count_ptr = (U32 *)(raw_data_node->string.str + cursor);
cursor += sizeof(*extra_file_count_ptr);
U32 max_extra_file_count = (raw_data_node->string.size - cursor) / sizeof(U32);
U32 extra_file_count = Min(*extra_file_count_ptr, max_extra_file_count);
U32 *extra_files = (U32 *)(raw_data_node->string.str + cursor);
cursor += sizeof(*extra_files) * extra_file_count;
inlinee_parsed->extra_file_count = extra_file_count;
inlinee_parsed->extra_files = extra_files;
}
}
}
}
ProfEnd();
return inlinee_lines_list;
}
// $$FrameData
internal void
cv_c13_patch_checksum_offsets_in_frame_data_list(String8List frame_data, U32 checksum_rebase)
{
for(String8Node *node = frame_data.first; node != 0; node = node->next)
{
String8 raw_data = node->string;
U64 count = raw_data.size / sizeof(CV_C13FrameData);
CV_C13FrameData *arr = (CV_C13FrameData *)raw_data.str;
CV_C13FrameData *ptr = arr;
CV_C13FrameData *opl = arr + count;
for(; ptr < opl; ptr += 1)
{
U64 rebased_frame_func = ptr->frame_func + checksum_rebase;
ptr->frame_func = safe_cast_u32(rebased_frame_func);
}
}
}
////////////////////////////////
// $$Lines Accel
int
cv_c13_voff_map_compar(const void *raw_a, const void *raw_b)
{
CV_Line *a = (CV_Line*)raw_a;
CV_Line *b = (CV_Line*)raw_b;
int cmp = a->voff < b->voff ? -1 :
a->voff > b->voff ? +1 :
0;
return cmp;
}
internal CV_LinesAccel *
cv_c13_make_lines_accel(Arena *arena, U64 lines_count, CV_LineArray *lines)
{
ProfBeginFunction();
U64 total_voff_count = 0;
for(U64 arr_idx = 0; arr_idx < lines_count; arr_idx += 1) {
total_voff_count += lines[arr_idx].line_count + 1;
}
CV_Line *map = push_array_no_zero(arena, CV_Line, total_voff_count);
U64 map_idx = 0;
for(U64 line_idx = 0; line_idx < lines_count; line_idx += 1) {
CV_LineArray *l = lines + line_idx;
if (l->line_count > 0) {
for(U64 voff_idx = 0; voff_idx < l->line_count; voff_idx += 1) {
map[map_idx].voff = l->voffs[voff_idx];
map[map_idx].file_off = l->file_off;
map[map_idx].line_num = l->line_nums[voff_idx];
map[map_idx].col_num = 0; // TODO: columns
map_idx += 1;
}
map[map_idx].voff = l->voffs[l->line_count];
map[map_idx].file_off = l->file_off;
map[map_idx].line_num = 0;
map[map_idx].col_num = 0;
map_idx += 1;
}
}
Assert(map_idx == total_voff_count);
qsort(map, total_voff_count, sizeof(map[0]), cv_c13_voff_map_compar);
CV_LinesAccel *accel = push_array(arena, CV_LinesAccel, 1);
accel->map_count = total_voff_count;
accel->map = map;
ProfEnd();
return accel;
}
#if 0
internal CV_Line *
cv_line_from_voff(CV_LinesAccel *accel, U64 voff, U64 *out_line_count)
{
ProfBeginFunction();
U64 voff_line_count = 0;
CV_Line *lines = 0;
U64 map_idx = bsearch_nearest_u64(accel->map, accel->map_count, voff, sizeof(accel->map[0]), OffsetOf(CV_Line, voff));
if(map_idx < accel->map_count) {
U64 near_voff = accel->map[map_idx].voff;
for (; map_idx > 0; map_idx -= 1) {
if(accel->map[map_idx - 1].voff != near_voff) {
break;
}
}
lines = accel->map + map_idx;
for(; map_idx < (accel->map_count-1); map_idx += 1) {
if(accel->map[map_idx].voff != near_voff) {
break;
}
voff_line_count += 1;
}
}
*out_line_count = voff_line_count;
ProfEnd();
return lines;
}
#endif
////////////////////////////////
// $$InlineeLines Accel
internal U64
cv_c13_inlinee_lines_accel_hash(void *buffer, U64 size)
{
XXH64_hash_t hash64 = XXH3_64bits(buffer, size);
return hash64;
}
internal B32
cv_c13_inlinee_lines_accel_push(CV_InlineeLinesAccel *accel, CV_C13InlineeLinesParsed *parsed)
{
U64 load_factor = accel->bucket_max * 2/3 + 1;
if(accel->bucket_count > load_factor) {
Assert("TODO: increase max count and rehash buckets");
}
B32 is_pushed = 0;
U64 hash = cv_c13_inlinee_lines_accel_hash(&parsed->inlinee, sizeof(parsed->inlinee));
U64 best_idx = hash % accel->bucket_max;
U64 idx = best_idx;
do {
if(accel->buckets[idx] == 0) {
accel->buckets[idx] = parsed;
accel->bucket_count += 1;
is_pushed = 1;
break;
}
idx = (idx + 1) % accel->bucket_max;
} while(idx != best_idx);
return is_pushed;
}
internal CV_C13InlineeLinesParsed *
cv_c13_inlinee_lines_accel_find(CV_InlineeLinesAccel *accel, CV_ItemId inlinee)
{
CV_C13InlineeLinesParsed *match = 0;
U64 hash = cv_c13_inlinee_lines_accel_hash(&inlinee, sizeof(inlinee));
U64 best_idx = hash % accel->bucket_max;
U64 idx = best_idx;
do {
if(accel->buckets[idx] != 0) {
if(accel->buckets[idx]->inlinee == inlinee) {
match = accel->buckets[idx];
break;
}
}
idx = (idx + 1) % accel->bucket_max;
} while(idx != best_idx);
return match;
}
internal CV_InlineeLinesAccel *
cv_c13_make_inlinee_lines_accel(Arena *arena, CV_C13InlineeLinesParsedList inlinee_lines)
{
ProfBeginFunction();
// alloc hash table
CV_InlineeLinesAccel *accel = push_array(arena, CV_InlineeLinesAccel, 1);
accel->bucket_count = 0;
accel->bucket_max = (U64)((F64)inlinee_lines.count * 2.5);
accel->buckets = push_array(arena, CV_C13InlineeLinesParsed *, accel->bucket_max);
// push parsed inlinees
for(CV_C13InlineeLinesParsedNode *inlinee = inlinee_lines.first; inlinee != 0; inlinee = inlinee->next) {
cv_c13_inlinee_lines_accel_push(accel, &inlinee->v);
}
ProfEnd();
return accel;
}
////////////////////////////////
internal CV_InlineBinaryAnnotsParsed
cv_c13_parse_inline_binary_annots(Arena *arena,
U64 parent_voff,
CV_C13InlineeLinesParsed *inlinee_parsed,
String8 binary_annots)
{
Temp scratch = scratch_begin(&arena, 1);
struct CodeRange {
struct CodeRange *next;
Rng1U64 range;
};
struct SourceLine {
struct SourceLine *next;
U64 voff;
U64 length;
U64 ln;
U64 cn;
CV_InlineRangeKind kind;
};
struct SourceFile {
struct SourceFile *next;
struct SourceLine *line_first;
struct SourceLine *line_last;
U64 line_count;
U64 checksum_off;
Rng1U64 last_code_range;
};
Rng1U64List code_ranges = {0};
struct SourceFile *file_first = 0;
struct SourceFile *file_last = 0;
U64 file_count = 0;
CV_InlineRangeKind range_kind = 0; (void)range_kind;
U32 code_length = 0;
U32 code_offset = 0;
U32 file_off = inlinee_parsed->file_off;
S32 ln = (S32)inlinee_parsed->first_source_ln;
S32 cn = 1;
U64 code_offset_lo = 0;
B32 code_offset_changed = 0;
B32 code_offset_lo_changed = 0;
B32 code_length_changed = 0;
B32 ln_changed = 1;
B32 file_off_changed = 0;
for (U64 cursor = 0, keep_running = 1; cursor < binary_annots.size && keep_running; ) {
U32 op = CV_InlineBinaryAnnotation_Null;
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &op);
switch (op) {
case CV_InlineBinaryAnnotation_Null: {
keep_running = 0;
// this is last run, append range with left over code bytes
code_length = code_offset - code_offset_lo;
code_length_changed = 1;
}break;
case CV_InlineBinaryAnnotation_CodeOffset: {
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &code_offset);
code_offset_changed = 1;
}break;
case CV_InlineBinaryAnnotation_ChangeCodeOffsetBase: {
AssertAlways(!"TODO: test case");
// U32 delta = 0;
// cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &delta);
// code_offset_base = code_offset;
// code_offset_end = code_offset + delta;
// code_offset += delta;
}break;
case CV_InlineBinaryAnnotation_ChangeCodeOffset: {
U32 delta = 0;
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &delta);
code_offset += delta;
if (!code_offset_lo_changed) {
code_offset_lo = code_offset;
code_offset_lo_changed = 1;
}
code_offset_changed = 1;
}break;
case CV_InlineBinaryAnnotation_ChangeCodeLength: {
code_length = 0;
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &code_length);
code_length_changed = 1;
}break;
case CV_InlineBinaryAnnotation_ChangeFile: {
U32 old_file_off = file_off;
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &file_off);
file_off_changed = old_file_off != file_off;
// Compiler isn't obligated to terminate code sequence before chaning files,
// so we have to always force emit code range on file change.
code_length_changed = file_off_changed;
}break;
case CV_InlineBinaryAnnotation_ChangeLineOffset: {
S32 delta = 0;
cursor += cv_decode_inline_annot_s32(binary_annots, cursor, &delta);
ln += delta;
ln_changed = 1;
}break;
case CV_InlineBinaryAnnotation_ChangeLineEndDelta: {
AssertAlways(!"TODO: test case");
// S32 end_delta = 1;
// cursor += cv_decode_inline_annot_s32(binary_annots, cursor, &end_delta);
// ln += end_delta;
}break;
case CV_InlineBinaryAnnotation_ChangeRangeKind: {
AssertAlways(!"TODO: test case");
// cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &range_kind);
}break;
case CV_InlineBinaryAnnotation_ChangeColumnStart: {
AssertAlways(!"TODO: test case");
// S32 delta;
// cursor += cv_decode_inline_annot_s32(binary_annots, cursor, &delta);
// cn += delta;
}break;
case CV_InlineBinaryAnnotation_ChangeColumnEndDelta: {
AssertAlways(!"TODO: test case");
// S32 end_delta;
// cursor += cv_decode_inline_annot_s32(binary_annots, cursor, &end_delta);
// cn += end_delta;
}break;
case CV_InlineBinaryAnnotation_ChangeCodeOffsetAndLineOffset: {
U32 code_offset_and_line_offset = 0;
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &code_offset_and_line_offset);
S32 line_delta = cv_inline_annot_signed_from_unsigned_operand(code_offset_and_line_offset >> 4);
U32 code_delta = (code_offset_and_line_offset & 0xf);
code_offset += code_delta;
ln += line_delta;
if (!code_offset_lo_changed) {
code_offset_lo = code_offset;
code_offset_lo_changed = 1;
}
code_offset_changed = 1;
ln_changed = 1;
}break;
case CV_InlineBinaryAnnotation_ChangeCodeLengthAndCodeOffset: {
U32 offset_delta = 0;
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &code_length);
cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &offset_delta);
code_offset += offset_delta;
if (!code_offset_lo_changed) {
code_offset_lo = code_offset;
code_offset_lo_changed = 1;
}
code_offset_changed = 1;
code_length_changed = 1;
}break;
case CV_InlineBinaryAnnotation_ChangeColumnEnd: {
AssertAlways(!"TODO: test case");
// U32 column_end = 0;
// cursor += cv_decode_inline_annot_u32(binary_annots, cursor, &column_end);
}break;
}
U64 line_code_offset = code_offset;
if (code_length_changed) {
// compute upper bound of the range
U64 code_offset_hi = code_offset + code_length;
// can last code range be extended to cover current sequence too?
if (code_ranges.last != 0 && code_ranges.last->v.max == parent_voff + code_offset_lo) {
code_ranges.last->v.max = parent_voff + code_offset_hi;
} else {
// append range
rng1u64_list_push(arena, &code_ranges, rng_1u64(parent_voff + code_offset_lo, parent_voff + code_offset_hi));
// update last code range in file
if (file_last) {
file_last->last_code_range = code_ranges.last->v;
}
}
// update low offset for next range
code_offset_lo = code_offset_hi;
// advance code offset
code_offset += code_length;
// reset state
code_offset_lo_changed = 0;
code_length_changed = 0;
code_length = 0;
}
if (file_off_changed || (file_first == 0)) {
// append file
struct SourceFile *file = push_array(scratch.arena, struct SourceFile, 1);
file->checksum_off = file_off;
SLLQueuePush(file_first, file_last, file);
++file_count;
// update last code range in file
if (code_ranges.last) {
file->last_code_range = code_ranges.last->v;
}
// reset state
file_off_changed = 0;
}
if (code_offset_changed && ln_changed) {
if (file_last->line_last == 0 || file_last->line_last->ln != (U64)ln) {
// append line
struct SourceLine *line = push_array(scratch.arena, struct SourceLine, 1);
line->voff = parent_voff + line_code_offset;
line->ln = (U64)ln;
line->cn = (U64)cn;
SLLQueuePush(file_last->line_first, file_last->line_last, line);
++file_last->line_count;
}
// reset state
code_offset_changed = 0;
ln_changed = 0;
}
}
CV_LineArray *lines = push_array(arena, CV_LineArray, file_count);
{
U64 lines_idx = 0;
for (struct SourceFile *file = file_first; file != 0; file = file->next, lines_idx += 1) {
CV_LineArray *l = lines + lines_idx;
l->file_off = file->checksum_off;
l->line_count = file->line_count;
l->col_count = 0;
if (file->line_count > 0) {
l->voffs = push_array_no_zero(arena, U64, file->line_count + 1);
l->line_nums = push_array_no_zero(arena, U32, file->line_count);
l->col_nums = 0; // TODO: column info
U64 line_idx = 0;
for (struct SourceLine *line = file->line_first; line != NULL; line = line->next, ++line_idx) {
// emit line voff and line number
l->voffs[line_idx] = line->voff;
l->line_nums[line_idx] = (U32)line->ln;
}
Assert(line_idx == file->line_count);
l->voffs[line_idx] = file->last_code_range.max;
}
}
}
// fill out result
CV_InlineBinaryAnnotsParsed result = {0};
result.lines_count = file_count;
result.lines = lines;
result.code_ranges = code_ranges;
scratch_end(scratch);
return result;
}
////////////////////////////////
internal Rng1U64List
cv_make_defined_range_list_from_gaps(Arena *arena, Rng1U64 defrange, CV_LvarAddrGap *gaps, U64 gap_count)
{
Rng1U64List result = {0};
if (gap_count == 0) {
// no gaps, push whole range
rng1u64_list_push(arena, &result, defrange);
} else {
U64 cursor = defrange.min;
for (U64 gap_idx = 0; gap_idx < gap_count; ++gap_idx) {
// make range
Rng1U64 range = rng_1u64(cursor, cursor + gaps[gap_idx].off);
rng1u64_list_push(arena, &result, range);
// advance
cursor = defrange.min + gaps[gap_idx].off + gaps[gap_idx].len;
}
// emit range past last gap
if (gap_count > 0) {
CV_LvarAddrGap last_gap = gaps[gap_count - 1];
U64 last_range_byte_size = dim_1u64(defrange) - (last_gap.off + last_gap.len);
if (last_range_byte_size) {
Rng1U64 last_range = rng_1u64(defrange.min + last_gap.off + last_gap.len, defrange.max);
rng1u64_list_push(arena, &result, last_range);
}
}
}
return result;
}