4
All of this script relies heavily on Eo internals and will break if they
5
change. Need to make sure this is always in sync.
8
ptr_size = int(gdb.parse_and_eval('sizeof(void *)'))
15
BITS_GENERATION_COUNTER = 6
24
BITS_MID_TABLE_ID = 11
27
BITS_GENERATION_COUNTER = 26
36
SHIFT_DOMAIN = (BITS_MID_TABLE_ID + BITS_TABLE_ID +
37
BITS_ENTRY_ID + BITS_GENERATION_COUNTER)
38
SHIFT_MID_TABLE_ID = (BITS_TABLE_ID +
39
BITS_ENTRY_ID + BITS_GENERATION_COUNTER)
40
SHIFT_TABLE_ID = (BITS_ENTRY_ID + BITS_GENERATION_COUNTER)
41
SHIFT_ENTRY_ID = (BITS_GENERATION_COUNTER)
44
MAX_DOMAIN = (1 << BITS_DOMAIN)
45
MAX_MID_TABLE_ID = (1 << BITS_MID_TABLE_ID)
46
MAX_TABLE_ID = ((1 << BITS_TABLE_ID) - DROPPED_TABLES)
47
MAX_ENTRY_ID = ((1 << BITS_ENTRY_ID) - DROPPED_ENTRIES)
48
MAX_GENERATIONS = (1 << BITS_GENERATION_COUNTER)
51
MASK_DOMAIN = (MAX_DOMAIN - 1)
52
MASK_MID_TABLE_ID = (MAX_MID_TABLE_ID - 1)
53
MASK_TABLE_ID = ((1 << BITS_TABLE_ID) - 1)
54
MASK_ENTRY_ID = ((1 << BITS_ENTRY_ID) - 1)
55
MASK_GENERATIONS = (MAX_GENERATIONS - 1)
56
MASK_OBJ_TAG = (1 << (REF_TAG_SHIFT))
59
null_void_ptr = gdb.parse_and_eval('(_Eo_Object *) 0')
60
null_eo_object_ptr = gdb.parse_and_eval('(_Eo_Object *) 0')
61
zero_uintptr_t = gdb.parse_and_eval('(uintptr_t) 0')
64
class Eo_resolve(gdb.Function):
66
gdb.Function.__init__(self, 'eo_resolve')
68
def invoke(self, arg):
69
obj_id = int(str(arg.cast(zero_uintptr_t.type)), 0)
71
mid_table_id = (obj_id >> SHIFT_MID_TABLE_ID) & MASK_MID_TABLE_ID
72
table_id = (obj_id >> SHIFT_TABLE_ID) & MASK_TABLE_ID
73
entry_id = (obj_id >> SHIFT_ENTRY_ID) & MASK_ENTRY_ID
74
tag_bit = (obj_id) & MASK_OBJ_TAG
75
generation = obj_id & MASK_GENERATIONS
77
if (obj_id == 0) or (tag_bit == 0):
78
gdb.write('Pointer is NULL or not a valid object.\n')
79
return null_eo_object_ptr
81
entries = gdb.parse_and_eval('_eo_gdb_main_domain->tables[0]->' +
82
'eo_ids_tables[{0}]'.format(mid_table_id))
85
gdb.write('Pointer is not a valid object.\n')
86
return null_eo_object_ptr
88
entry = entries[table_id]['entries'][entry_id]
90
if (not entry['active']) or (int(entry['generation']) != generation):
91
gdb.write('Pointer is no longer active.\n')
92
return null_eo_object_ptr
100
class Eo_data_get(gdb.Function):
102
gdb.Function.__init__(self, 'eo_data_get')
104
def invoke(self, ptr, kls_name):
105
ptr = ptr.cast(null_eo_object_ptr.type)
108
gdb.write('Object is not a valid pointer (NULL).\n')
111
kls_name = kls_name.string()
112
extns = ptr['klass']['mro']
116
while int(extns[i]) != 0:
117
if extns[i]['desc']['name'].string() == kls_name:
122
gdb.write('Class "{}" not found in the object mro.\n'
127
if int(kls['desc']['type'].cast(zero_uintptr_t.type)) != 3:
128
return gdb.parse_and_eval('(void *) (((char *) {}) + {})'
129
.format(ptr, kls['data_offset']))
131
extn_off = ptr['klass']['extn_data_off']
132
if int(extn_off) == 0:
136
while int(extn_off[i]['klass']) != 0:
137
kls = extn_off[i]['klass']
138
if kls['desc']['name'].string() == kls_name:
139
return gdb.parse_and_eval('(void *) (((char *) {}) + {})'
140
.format(ptr, kls['data_offset']))