mirror of
https://github.com/radareorg/radare2.git
synced 2024-11-25 06:09:50 +00:00
8da8ad740f
* For commit comments and compiler errors * Fixes for PR comments * fix some "infer fixes" commits Signed-off-by: Riccardo Schirone <sirmy15@gmail.com>
2421 lines
67 KiB
C
2421 lines
67 KiB
C
#include "types.h"
|
|
#include "tpi.h"
|
|
#include "stream_file.h"
|
|
|
|
static unsigned int base_idx = 0;
|
|
static RList *p_types_list;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void print_base_type(EBASE_TYPES base_type, char **name)
|
|
{
|
|
switch (base_type) {
|
|
case eT_32PINT4:
|
|
*name = "pointer to long";
|
|
break;
|
|
case eT_32PRCHAR:
|
|
*name = "pointer to unsigned char";
|
|
break;
|
|
case eT_32PUCHAR:
|
|
*name = "pointer to unsigned char";
|
|
break;
|
|
case eT_32PULONG:
|
|
*name = "pointer to unsigned long";
|
|
break;
|
|
case eT_32PLONG:
|
|
*name = "pointer to long";
|
|
break;
|
|
case eT_32PUQUAD:
|
|
*name = "pointer to unsigned long long";
|
|
break;
|
|
case eT_32PUSHORT:
|
|
*name = "pointer to unsigned short";
|
|
break;
|
|
case eT_32PVOID:
|
|
*name = "pointer to void";
|
|
break;
|
|
case eT_64PVOID:
|
|
*name = "pointer64 to void";
|
|
break;
|
|
case eT_INT4:
|
|
*name = "long";
|
|
break;
|
|
case eT_INT8:
|
|
*name = "long long";
|
|
break;
|
|
case eT_LONG:
|
|
*name = "long";
|
|
break;
|
|
case eT_QUAD:
|
|
*name = "long long";
|
|
break;
|
|
case eT_RCHAR:
|
|
*name = "unsigned char";
|
|
break;
|
|
case eT_REAL32:
|
|
*name = "float";
|
|
break;
|
|
case eT_REAL64:
|
|
*name = "double";
|
|
break;
|
|
case eT_REAL80:
|
|
*name = "long double";
|
|
break;
|
|
case eT_SHORT:
|
|
*name = "short";
|
|
break;
|
|
case eT_UCHAR:
|
|
*name = "unsigned char";
|
|
break;
|
|
case eT_UINT4:
|
|
*name = "unsigned long";
|
|
break;
|
|
case eT_ULONG:
|
|
*name = "unsigned long";
|
|
break;
|
|
case eT_UQUAD:
|
|
*name = "unsigned long long";
|
|
break;
|
|
case eT_USHORT:
|
|
*name = "unsigned short";
|
|
break;
|
|
case eT_WCHAR:
|
|
*name = "wchar";
|
|
break;
|
|
case eT_VOID:
|
|
*name = "void";
|
|
break;
|
|
case eT_32PWCHAR:
|
|
*name = "pointer to wchar";
|
|
break;
|
|
default:
|
|
*name = "unsupported base type";
|
|
break;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_sval_name_len(SVal *val, int *res_len)
|
|
{
|
|
if (val->value_or_type < eLF_CHAR) {
|
|
SCString *scstr;
|
|
scstr = (SCString *) val->name_or_val;
|
|
*res_len = scstr->size;
|
|
} else {
|
|
switch (val->value_or_type) {
|
|
case eLF_ULONG:
|
|
{
|
|
SVal_LF_ULONG *lf_ulong;
|
|
lf_ulong = (SVal_LF_ULONG *) val->name_or_val;
|
|
*res_len = lf_ulong->name.size;
|
|
break;
|
|
}
|
|
case eLF_USHORT:
|
|
{
|
|
SVal_LF_USHORT *lf_ushort;
|
|
lf_ushort = (SVal_LF_USHORT *) val->name_or_val;
|
|
*res_len = lf_ushort->name.size;
|
|
break;
|
|
}
|
|
default:
|
|
*res_len = 0;
|
|
printf("get_sval_name::oops\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_sval_name(SVal *val, char **name)
|
|
{
|
|
if (val->value_or_type < eLF_CHAR) {
|
|
SCString *scstr;
|
|
scstr = (SCString *) val->name_or_val;
|
|
*name = scstr->name;
|
|
// strcpy(name, scstr->name);
|
|
} else {
|
|
switch (val->value_or_type) {
|
|
case eLF_CHAR:
|
|
{
|
|
SVal_LF_CHAR *lf_char;
|
|
lf_char = (SVal_LF_CHAR *) val->name_or_val;
|
|
*name = lf_char->name.name;
|
|
// strcpy(name, lf_uchar->name.name);
|
|
break;
|
|
}
|
|
case eLF_ULONG:
|
|
{
|
|
SVal_LF_ULONG *lf_ulong;
|
|
lf_ulong = (SVal_LF_ULONG *) val->name_or_val;
|
|
*name = lf_ulong->name.name;
|
|
// strcpy(name, lf_ulong->name.name);
|
|
break;
|
|
}
|
|
case eLF_LONG:
|
|
{
|
|
SVal_LF_LONG *lf_long;
|
|
lf_long = (SVal_LF_LONG *) val->name_or_val;
|
|
*name = lf_long->name.name;
|
|
// strcpy(name, lf_long->name.name);
|
|
break;
|
|
}
|
|
case eLF_USHORT:
|
|
{
|
|
SVal_LF_USHORT *lf_ushort;
|
|
lf_ushort = (SVal_LF_USHORT *) val->name_or_val;
|
|
*name =lf_ushort->name.name;
|
|
// strcpy(name, lf_ushort->name.name);
|
|
break;
|
|
}
|
|
default:
|
|
*name = 0;
|
|
printf("get_sval_name::oops\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//static void get_arglist_type(void *type, void **arglist_type)
|
|
//{
|
|
// STypeInfo *t = (STypeInfo *) type;
|
|
// SLF_ARGLIST *lf_arglist = (SLF_ARGLIST *) t->type_info;
|
|
// RList *l = (RList *) *arglist_type;
|
|
// int i = 0;
|
|
// int tmp = 0;
|
|
|
|
// for (i = 0; i < lf_arglist->count; i++) {
|
|
// tmp = lf_arglist->arg_type[i];
|
|
// if (tmp < base_idx) {
|
|
// // 0 - means NO_TYPE
|
|
// r_list_append(l, 0);
|
|
// } else {
|
|
// r_list_append(l, r_list_get_n(p_types_list, (tmp - base_idx)));
|
|
// }
|
|
// }
|
|
//}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void is_union_fwdref(void *type, int *is_fwdref)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_UNION *lf = (SLF_UNION *) t->type_info;
|
|
|
|
*is_fwdref = lf->prop.bits.fwdref;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void is_struct_class_fwdref(void *type, int *is_fwdref)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *) t->type_info;
|
|
|
|
*is_fwdref = lf->prop.bits.fwdref;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_array_element_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARRAY *lf_array = (SLF_ARRAY *) t->type_info;
|
|
int curr_idx = lf_array->element_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_array_index_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARRAY *lf_array = (SLF_ARRAY *) t->type_info;
|
|
int curr_idx = lf_array->index_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_bitfield_base_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_BITFIELD *lf = (SLF_BITFIELD *) t->type_info;
|
|
int curr_idx = lf->base_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_class_struct_derived(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *) t->type_info;
|
|
int curr_idx = lf->derived;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_class_struct_vshape(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *) t->type_info;
|
|
int curr_idx = lf->vshape;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_mfunction_return_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MFUNCTION *lf = (SLF_MFUNCTION *) t->type_info;
|
|
int curr_idx = lf->return_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_mfunction_class_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MFUNCTION *lf = (SLF_MFUNCTION *) t->type_info;
|
|
int curr_idx = lf->class_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_mfunction_this_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MFUNCTION *lf = (SLF_MFUNCTION *) t->type_info;
|
|
int curr_idx = lf->this_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_mfunction_arglist(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MFUNCTION *lf = (SLF_MFUNCTION *) t->type_info;
|
|
int curr_idx = lf->arglist;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_modifier_modified_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MODIFIER *lf = (SLF_MODIFIER *) t->type_info;
|
|
int curr_idx = lf->modified_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_pointer_utype(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_POINTER *lf = (SLF_POINTER *) t->type_info;
|
|
int curr_idx = lf->utype;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_procedure_return_type(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_PROCEDURE *lf = (SLF_PROCEDURE *) t->type_info;
|
|
int curr_idx = lf->return_type;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_procedure_arglist(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_PROCEDURE *lf = (SLF_PROCEDURE *) t->type_info;
|
|
int curr_idx = lf->arg_list;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_member_index(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MEMBER *lf = (SLF_MEMBER *) t->type_info;
|
|
int curr_idx = lf->inedex;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_nesttype_index(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_NESTTYPE *lf = (SLF_NESTTYPE *) t->type_info;
|
|
int curr_idx = lf->index;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_onemethod_index(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ONEMETHOD *lf = (SLF_ONEMETHOD *) t->type_info;
|
|
int curr_idx = lf->index;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_method_mlist(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_METHOD *lf = (SLF_METHOD *) t->type_info;
|
|
int curr_idx = lf->mlist;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int get_enum_utype(void *type, void **ret_type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUM *lf = (SLF_ENUM *) t->type_info;
|
|
int curr_idx = lf->utype;
|
|
|
|
if (curr_idx < base_idx) {
|
|
*ret_type = 0;
|
|
} else {
|
|
curr_idx -= base_idx;
|
|
*ret_type = r_list_get_n(p_types_list, curr_idx);
|
|
}
|
|
|
|
return curr_idx;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_fieldlist_members(void *type, RList **l)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_FIELDLIST *lf_fieldlist = (SLF_FIELDLIST *) t->type_info;
|
|
|
|
*l = lf_fieldlist->substructs;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_union_members(void *type, RList **l)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_UNION *lf_union = (SLF_UNION *) t->type_info;
|
|
unsigned int indx = 0;
|
|
|
|
if (lf_union->field_list == 0) {
|
|
*l = 0;
|
|
} else {
|
|
SType *tmp = 0;
|
|
indx = lf_union->field_list - base_idx;
|
|
tmp = (SType *)r_list_get_n(p_types_list, indx);
|
|
*l = ((SLF_FIELDLIST *) tmp->type_data.type_info)->substructs;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_struct_class_members(void *type, RList **l)
|
|
{
|
|
SLF_FIELDLIST *lf_fieldlist = 0;
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *) t->type_info;
|
|
unsigned int indx = 0;
|
|
|
|
if (lf->field_list == 0) {
|
|
*l = 0;
|
|
} else {
|
|
SType *tmp = 0;
|
|
indx = lf->field_list - base_idx;
|
|
tmp = (SType *)r_list_get_n(p_types_list, indx);
|
|
lf_fieldlist = (SLF_FIELDLIST *) tmp->type_data.type_info;
|
|
*l = lf_fieldlist->substructs;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enum_members(void *type, RList **l)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUM *lf = (SLF_ENUM *) t->type_info;
|
|
unsigned int indx = 0;
|
|
|
|
if (lf->field_list == 0) {
|
|
*l = 0;
|
|
} else {
|
|
SType *tmp = 0;
|
|
indx = lf->field_list - base_idx;
|
|
tmp = (SType *)r_list_get_n(p_types_list, indx);
|
|
*l = ((SLF_FIELDLIST *) tmp->type_data.type_info)->substructs;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_sval_val(SVal *val, int *res)
|
|
{
|
|
if (val->value_or_type < eLF_CHAR) {
|
|
*res = val->value_or_type;
|
|
} else {
|
|
switch (val->value_or_type) {
|
|
case eLF_ULONG:
|
|
{
|
|
SVal_LF_ULONG *lf_ulong;
|
|
lf_ulong = (SVal_LF_ULONG *) val->name_or_val;
|
|
*res = lf_ulong->value;
|
|
break;
|
|
}
|
|
case eLF_LONG:
|
|
{
|
|
SVal_LF_LONG *lf_long;
|
|
lf_long = (SVal_LF_LONG *) val->name_or_val;
|
|
*res = lf_long->value;
|
|
break;
|
|
}
|
|
case eLF_USHORT:
|
|
{
|
|
SVal_LF_USHORT *lf_ushort;
|
|
lf_ushort = (SVal_LF_USHORT *) val->name_or_val;
|
|
*res = lf_ushort->value;
|
|
break;
|
|
}
|
|
case eLF_CHAR:
|
|
{
|
|
SVal_LF_CHAR *lf_char;
|
|
lf_char = (SVal_LF_CHAR *) val->name_or_val;
|
|
*res = lf_char->value;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
*res = 0;
|
|
printf("get_sval_val::oops\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//static void get_member_indx_val(void *type, int *indx_val)
|
|
//{
|
|
// STypeInfo *t = (STypeInfo *) type;
|
|
// SLF_MEMBER *lf_member = (SLF_MEMBER *)t->type_info;
|
|
|
|
// *indx_val = lf_member->inedex;
|
|
//}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_onemethod_name_len(void *type, int *res_len)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ONEMETHOD *lf_onemethod = (SLF_ONEMETHOD *)t->type_info;
|
|
|
|
*res_len = lf_onemethod->val.str_data.size;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enum_name_len(void *type, int *res_len)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUM *lf_enum = (SLF_ENUM *)t->type_info;
|
|
|
|
*res_len = lf_enum->name.size;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_class_struct_name_len(void *type, int *res_len)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *)t->type_info;
|
|
|
|
get_sval_name_len(&lf->size, res_len);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_array_name_len(void *type, int *res_len)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARRAY *lf_array = (SLF_ARRAY *) t->type_info;
|
|
|
|
get_sval_name_len(&lf_array->size, res_len);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_union_name_len(void *type, int *res_len)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_UNION *lf_union = (SLF_UNION *) t->type_info;
|
|
|
|
get_sval_name_len(&lf_union->size, res_len);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enumerate_name_len(void *type, int *res_len)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUMERATE *lf = (SLF_ENUMERATE *)t->type_info;
|
|
|
|
get_sval_name_len(&lf->enum_value, res_len);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_nesttype_name_len(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_NESTTYPE *lf = (SLF_NESTTYPE *)t->type_info;
|
|
|
|
*res = lf->name.size;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_method_name_len(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_METHOD *lf = (SLF_METHOD *)t->type_info;
|
|
|
|
*res = lf->name.size;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_member_name_len(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MEMBER *lf = (SLF_MEMBER *)t->type_info;
|
|
|
|
get_sval_name_len(&lf->offset, res);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_member_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MEMBER *lf = (SLF_MEMBER *)t->type_info;
|
|
|
|
get_sval_name(&lf->offset, name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_onemethod_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ONEMETHOD *lf = (SLF_ONEMETHOD *)t->type_info;
|
|
|
|
*name = lf->val.str_data.name;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_method_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_METHOD *lf = (SLF_METHOD *)t->type_info;
|
|
|
|
*name = lf->name.name;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_nesttype_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_NESTTYPE *lf = (SLF_NESTTYPE *)t->type_info;
|
|
|
|
*name = lf->name.name;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enumerate_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUMERATE *lf = (SLF_ENUMERATE *)t->type_info;
|
|
|
|
get_sval_name(&lf->enum_value, name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enum_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUM *lf_enum = (SLF_ENUM *)t->type_info;
|
|
|
|
*name = lf_enum->name.name;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_class_struct_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *)t->type_info;
|
|
|
|
get_sval_name(&lf->size, name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_array_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARRAY *lf_array = (SLF_ARRAY *) t->type_info;
|
|
|
|
get_sval_name(&lf_array->size, name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_union_name(void *type, char **name)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_UNION *lf_union = (SLF_UNION *) t->type_info;
|
|
|
|
get_sval_name(&lf_union->size, name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_onemethod_val(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ONEMETHOD *lf = (SLF_ONEMETHOD *) t->type_info;
|
|
|
|
*res = lf->val.val;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_member_val(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_MEMBER *lf = (SLF_MEMBER *)t->type_info;
|
|
|
|
get_sval_val(&lf->offset, res);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enumerate_val(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUMERATE *lf = (SLF_ENUMERATE *)t->type_info;
|
|
|
|
get_sval_val(&lf->enum_value, res);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_class_struct_val(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_STRUCTURE *lf = (SLF_STRUCTURE *) t->type_info;
|
|
|
|
get_sval_val(&lf->size, res);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_array_val(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARRAY *lf_array = (SLF_ARRAY *) t->type_info;
|
|
|
|
get_sval_val(&lf_array->size, res);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_union_val(void *type, int *res)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_UNION *lf_union = (SLF_UNION *) t->type_info;
|
|
|
|
get_sval_val(&lf_union->size, res);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//static void printf_sval_name(SVal *val)
|
|
//{
|
|
// int len = 0;
|
|
// char *name = 0;
|
|
|
|
// get_sval_name_len(val, &len);
|
|
// name = (char *) malloc(len);
|
|
// get_sval_name(val, &name);
|
|
// printf("%s", name);
|
|
|
|
// free(name);
|
|
//}
|
|
|
|
//typedef struct {
|
|
// SParsedPDBStream *parsed_pdb_stream;
|
|
// SPDBInfoStreamD data;
|
|
//} SPDBInfoStream;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_sval(SVal *val)
|
|
{
|
|
if (val->value_or_type < eLF_CHAR) {
|
|
SCString *scstr;
|
|
scstr = (SCString *) val->name_or_val;
|
|
R_FREE(scstr->name);
|
|
R_FREE(val->name_or_val);
|
|
} else {
|
|
switch (val->value_or_type) {
|
|
case eLF_ULONG:
|
|
{
|
|
SVal_LF_ULONG *lf_ulong;
|
|
lf_ulong = (SVal_LF_ULONG *) val->name_or_val;
|
|
R_FREE(lf_ulong->name.name);
|
|
R_FREE(val->name_or_val);
|
|
break;
|
|
}
|
|
case eLF_LONG:
|
|
{
|
|
SVal_LF_LONG *lf_long;
|
|
lf_long = (SVal_LF_LONG *) val->name_or_val;
|
|
R_FREE(lf_long->name.name);
|
|
R_FREE(val->name_or_val);
|
|
break;
|
|
}
|
|
case eLF_SHORT:
|
|
{
|
|
SVal_LF_SHORT *lf_short;
|
|
lf_short = (SVal_LF_SHORT *) val->name_or_val;
|
|
R_FREE(lf_short->name.name);
|
|
R_FREE(val->name_or_val);
|
|
break;
|
|
}
|
|
case eLF_USHORT:
|
|
{
|
|
SVal_LF_USHORT *lf_ushort;
|
|
lf_ushort = (SVal_LF_USHORT *) val->name_or_val;
|
|
R_FREE(lf_ushort->name.name);
|
|
R_FREE(val->name_or_val);
|
|
break;
|
|
}
|
|
case eLF_CHAR:
|
|
{
|
|
SVal_LF_CHAR *lf_char;
|
|
lf_char = (SVal_LF_CHAR *) val->name_or_val;
|
|
R_FREE(lf_char->name.name);
|
|
R_FREE(val->name_or_val);
|
|
break;
|
|
}
|
|
default:
|
|
printf("free_sval()::not supproted type\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_enumerate(void *type_info)
|
|
{
|
|
STypeInfo *typeInfo = (STypeInfo *) type_info;
|
|
SLF_ENUMERATE *lf_en = (SLF_ENUMERATE *) typeInfo->type_info;
|
|
|
|
free_sval(&(lf_en->enum_value));
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_nesttype(void *type_info)
|
|
{
|
|
STypeInfo *typeInfo = (STypeInfo *) type_info;
|
|
SLF_NESTTYPE *lf_nest = (SLF_NESTTYPE *) typeInfo->type_info;
|
|
|
|
free(lf_nest->name.name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_method(void *type_info)
|
|
{
|
|
STypeInfo *typeInfo = (STypeInfo *) type_info;
|
|
SLF_METHOD *lf_meth = (SLF_METHOD *) typeInfo->type_info;
|
|
|
|
free(lf_meth->name.name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_member(void *type_info)
|
|
{
|
|
STypeInfo *typeInfo = (STypeInfo *) type_info;
|
|
SLF_MEMBER *lf_mem = (SLF_MEMBER *) typeInfo->type_info;
|
|
|
|
free_sval(&lf_mem->offset);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_fieldlist(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_FIELDLIST *lf_fieldlist = (SLF_FIELDLIST *) t->type_info;
|
|
RListIter *it;
|
|
STypeInfo *type_info = 0;
|
|
|
|
it = r_list_iterator(lf_fieldlist->substructs);
|
|
while (r_list_iter_next(it)) {
|
|
type_info = (STypeInfo *) r_list_iter_get(it);
|
|
if (type_info->free_)
|
|
type_info->free_(type_info);
|
|
if (type_info->type_info) {
|
|
free(type_info->type_info);
|
|
}
|
|
free(type_info);
|
|
}
|
|
r_list_free(lf_fieldlist->substructs);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_class(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_CLASS *lf_class = (SLF_CLASS *) t->type_info;
|
|
|
|
free_sval(&lf_class->size);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_union(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_UNION *lf_union = (SLF_UNION *) t->type_info;
|
|
|
|
free_sval(&lf_union->size);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_onemethod(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ONEMETHOD *lf_onemethod = (SLF_ONEMETHOD *) t->type_info;
|
|
|
|
free(lf_onemethod->val.str_data.name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_enum(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ENUM *lf_enum = (SLF_ENUM *) t->type_info;
|
|
|
|
free(lf_enum->name.name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_array(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARRAY *lf_array = (SLF_ARRAY *) t->type_info;
|
|
|
|
free_sval(&lf_array->size);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_arglist(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_ARGLIST *lf_arglist = (SLF_ARGLIST *) t->type_info;
|
|
|
|
free(lf_arglist->arg_type);
|
|
lf_arglist->arg_type = 0;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_lf_vtshape(void *type)
|
|
{
|
|
STypeInfo *t = (STypeInfo *) type;
|
|
SLF_VTSHAPE *lf_vtshape = (SLF_VTSHAPE *) t->type_info;
|
|
|
|
free (lf_vtshape->vt_descriptors);
|
|
lf_vtshape->vt_descriptors = 0;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void free_tpi_stream(void *stream)
|
|
{
|
|
STpiStream *tpi_stream = (STpiStream *)stream;
|
|
RListIter *it;
|
|
SType *type = NULL;
|
|
|
|
it = r_list_iterator (tpi_stream->types);
|
|
while (r_list_iter_next (it)) {
|
|
type = (SType *) r_list_iter_get (it);
|
|
if (!type) {
|
|
continue;
|
|
}
|
|
if (type->type_data.free_) {
|
|
type->type_data.free_ (&type->type_data);
|
|
type->type_data.free_ = 0;
|
|
}
|
|
if (type->type_data.type_info) {
|
|
free (type->type_data.type_info);
|
|
type->type_data.free_ = 0;
|
|
type->type_data.type_info = 0;
|
|
}
|
|
free (type);
|
|
type = NULL;
|
|
}
|
|
r_list_free (tpi_stream->types);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_array_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_element_type (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("array: ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) {
|
|
free (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "array: ");
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
if (need_to_free) {
|
|
free (tmp_name);
|
|
tmp_name = 0;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_pointer_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_utype (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("pointer to ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) {
|
|
free (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy(*name, "pointer to ");
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
if (need_to_free) {
|
|
free (tmp_name);
|
|
tmp_name = 0;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_modifier_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_modified_type (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("modifier ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc(name_len + 1);
|
|
if (!(*name)) {
|
|
free (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "modifier ");
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
if (need_to_free) {
|
|
free (tmp_name);
|
|
tmp_name = 0;
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_procedure_print_type(void *type, char **name)
|
|
{
|
|
int name_len = 0;
|
|
|
|
name_len = strlen ("proc ");
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy(*name, "proc ");
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_bitfield_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
SLF_BITFIELD *bitfeild_info = (SLF_BITFIELD *)ti->type_info;
|
|
|
|
base_type = ti->get_base_type (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("bitfield ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
name_len += 4;
|
|
*name = (char *) malloc (name_len + 1 + 1);
|
|
if (!(*name)) {
|
|
if (need_to_free) free (tmp_name);
|
|
return;
|
|
}
|
|
|
|
// name[name_len] = '\0';
|
|
if (tmp_name) {
|
|
sprintf (*name, "%s %s : %d", "bitfield", tmp_name, (int)bitfeild_info->length);
|
|
} else {
|
|
sprintf (*name, "%s : %d", "bitfield", (int)bitfeild_info->length);
|
|
}
|
|
|
|
if (need_to_free)
|
|
free (tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_fieldlist_print_type(void *type, char **name)
|
|
{
|
|
int name_len = 0;
|
|
|
|
name_len = strlen ("fieldlist ");
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "fieldlist ");
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enum_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_utype (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("enum ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) {
|
|
if (need_to_free) free (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "enum ");
|
|
if (tmp_name)
|
|
strcat( *name, tmp_name);
|
|
|
|
if (need_to_free)
|
|
free (tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_class_struct_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
ELeafType lt;
|
|
char *tmp_name = 0, *tmp1 = 0;
|
|
int name_len = 0;
|
|
|
|
lt = ti->leaf_type;
|
|
ti->get_name(ti, &tmp_name);
|
|
|
|
if (lt == eLF_CLASS) {
|
|
tmp1 = "class ";
|
|
} else {
|
|
tmp1 = "struct ";
|
|
}
|
|
name_len = strlen(tmp1);
|
|
if (tmp_name)
|
|
name_len += strlen(tmp_name);
|
|
*name = (char *) malloc(name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy(*name, tmp1);
|
|
if (tmp_name)
|
|
strcat(*name, tmp_name);
|
|
|
|
// if (need_to_free) {
|
|
// free(tmp_name);
|
|
// tmp_name = 0;
|
|
// }
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_arglist_print_type(void *type, char **name)
|
|
{
|
|
(void) type;
|
|
int name_len = 0;
|
|
|
|
name_len = strlen ("arg_list");
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "arg_list");
|
|
// STypeInfo *ti = (STypeInfo *) type;
|
|
// SType *t = 0;
|
|
// char *tmp_name = 0;
|
|
// int name_len = 0;
|
|
// int need_to_free = 1;
|
|
// int base_type = 0;
|
|
|
|
// base_type = ti->get_arg_type(ti, (void **)&t);
|
|
// if (!t) {
|
|
// need_to_free = 0;
|
|
// print_base_type(base_type, &tmp_name);
|
|
// } else {
|
|
// ti = &t->type_data;
|
|
// ti->get_print_type(ti, &tmp_name);
|
|
// }
|
|
|
|
// name_len = strlen("arglist ");
|
|
// name_len += strlen(tmp_name);
|
|
// *name = (char *) malloc(name_len + 1);
|
|
// // name[name_len] = '\0';
|
|
// strcpy(*name, "arglist ");
|
|
// strcat(*name, tmp_name);
|
|
|
|
// if (need_to_free)
|
|
// free(tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_mfunction_print_type(void *type, char **name)
|
|
{
|
|
int name_len = 0;
|
|
|
|
name_len = strlen("mfunction ");
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "mfunction ");
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_union_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
// ELeafType lt;
|
|
char *tmp_name = 0, *tmp1 = 0;
|
|
int name_len = 0;
|
|
|
|
// lt = ti->leaf_type;
|
|
ti->get_name (ti, &tmp_name);
|
|
|
|
tmp1 = "union ";
|
|
name_len = strlen (tmp1);
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, tmp1);
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
// if (need_to_free) {
|
|
// free(tmp_name);
|
|
// tmp_name = 0;
|
|
// }
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_vtshape_print_type(void *type, char **name)
|
|
{
|
|
int name_len = 0;
|
|
|
|
name_len = strlen ("vtshape");
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "vthape");
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_enumerate_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
char *tmp_name = 0, *tmp1 = 0;
|
|
int name_len = 0;
|
|
|
|
ti->get_name (ti, &tmp_name);
|
|
|
|
tmp1 = "enumerate ";
|
|
name_len = strlen (tmp1);
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, tmp1);
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
// if (need_to_free)
|
|
// free(tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_nesttype_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_index (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
if (ti->get_print_type != NULL) {
|
|
ti->get_print_type (ti, &tmp_name);
|
|
} else {
|
|
// TODO: need to investigate why this branch can be...
|
|
// this is possible because there is no support for
|
|
// parsing METHODLIST...
|
|
// need to investigate for this theme
|
|
eprintf ("warning: strange for nesttype\n");
|
|
}
|
|
}
|
|
|
|
name_len = strlen ("nesttype ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) {
|
|
if (need_to_free) free (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "nesttype ");
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
if (need_to_free)
|
|
free (tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_method_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
char *tmp_name = 0, *tmp1 = 0;
|
|
int name_len = 0;
|
|
|
|
ti->get_name (ti, &tmp_name);
|
|
|
|
tmp1 = "method ";
|
|
name_len = strlen (tmp1);
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) return;
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, tmp1);
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
// if (need_to_free)
|
|
// free(tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_member_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_index (ti, (void **) &t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("(member) ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) {
|
|
if (need_to_free) R_FREE (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy(*name, "(member) ");
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
if (need_to_free) {
|
|
R_FREE (tmp_name);
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void get_onemethod_print_type(void *type, char **name)
|
|
{
|
|
STypeInfo *ti = (STypeInfo *) type;
|
|
SType *t = 0;
|
|
char *tmp_name = 0;
|
|
int name_len = 0;
|
|
int need_to_free = 1;
|
|
int base_type = 0;
|
|
|
|
base_type = ti->get_index (ti, (void **)&t);
|
|
if (!t) {
|
|
need_to_free = 0;
|
|
print_base_type (base_type, &tmp_name);
|
|
} else {
|
|
ti = &t->type_data;
|
|
ti->get_print_type (ti, &tmp_name);
|
|
}
|
|
|
|
name_len = strlen ("onemethod ");
|
|
if (tmp_name)
|
|
name_len += strlen (tmp_name);
|
|
*name = (char *) malloc (name_len + 1);
|
|
if (!(*name)) {
|
|
if (need_to_free) free (tmp_name);
|
|
return;
|
|
}
|
|
// name[name_len] = '\0';
|
|
strcpy (*name, "onemethod ");
|
|
if (tmp_name)
|
|
strcat (*name, tmp_name);
|
|
|
|
if (need_to_free)
|
|
free (tmp_name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
void init_scstring(SCString *cstr, unsigned int size, char *name)
|
|
{
|
|
cstr->size = size;
|
|
cstr->name = (char *) malloc (size);
|
|
if (!cstr->name) return;
|
|
strcpy (cstr->name, name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
void deinit_scstring(SCString *cstr)
|
|
{
|
|
free (cstr->name);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
int parse_sctring(SCString *sctr, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int c = 0;
|
|
|
|
sctr->name = 0;
|
|
|
|
while (*leaf_data != 0) {
|
|
CAN_READ((*read_bytes + c), 1, len);
|
|
c++;
|
|
leaf_data++;
|
|
}
|
|
CAN_READ(*read_bytes, 1, len);
|
|
leaf_data += 1;
|
|
(*read_bytes) += (c + 1);
|
|
|
|
init_scstring (sctr, c + 1, (char *)leaf_data - (c + 1));
|
|
return 1;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_sval(SVal *val, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
val->name_or_val = 0;
|
|
|
|
READ(*read_bytes, 2, len, val->value_or_type, leaf_data, unsigned short);
|
|
|
|
if (val->value_or_type < eLF_CHAR) {
|
|
SCString *sctr = (SCString *) malloc (sizeof (SCString));
|
|
if (!sctr) return 0;
|
|
parse_sctring (sctr, leaf_data, read_bytes, len);
|
|
val->name_or_val = sctr;
|
|
} else {
|
|
switch (val->value_or_type) {
|
|
case eLF_CHAR:
|
|
{
|
|
SVal_LF_CHAR lf_char;
|
|
READ (*read_bytes, 2, len, lf_char.value, leaf_data, char);
|
|
parse_sctring (&lf_char.name, leaf_data, read_bytes, len);
|
|
val->name_or_val = malloc (sizeof (SVal_LF_CHAR));
|
|
if (!val->name_or_val) break;
|
|
memcpy (val->name_or_val, &lf_char, sizeof (SVal_LF_CHAR));
|
|
break;
|
|
|
|
}
|
|
case eLF_LONG:
|
|
{
|
|
SVal_LF_LONG lf_long;
|
|
lf_long.value = 0;
|
|
// long = 4 bytes for Windows, but not in Linux x64,
|
|
// so here is using int instead of long when
|
|
// reading long value
|
|
READ (*read_bytes, 4, len, lf_long.value, leaf_data, int);
|
|
parse_sctring (&lf_long.name, leaf_data, read_bytes, len);
|
|
val->name_or_val = malloc (sizeof (SVal_LF_LONG));
|
|
if (!val->name_or_val) break;
|
|
memcpy (val->name_or_val, &lf_long, sizeof (SVal_LF_LONG));
|
|
break;
|
|
}
|
|
case eLF_ULONG:
|
|
{
|
|
SVal_LF_ULONG lf_ulong;
|
|
lf_ulong.value = 0;
|
|
// unsinged long = 4 bytes for Windows, but not in Linux x64,
|
|
// so here is using unsinged int instead of unsigned long when
|
|
// reading ulong value
|
|
READ(*read_bytes, 4, len, lf_ulong.value, leaf_data, unsigned int);
|
|
parse_sctring (&lf_ulong.name, leaf_data, read_bytes, len);
|
|
val->name_or_val = malloc (sizeof (SVal_LF_ULONG));
|
|
if (!val->name_or_val) break;
|
|
memcpy(val->name_or_val, &lf_ulong, sizeof (SVal_LF_ULONG));
|
|
break;
|
|
}
|
|
case eLF_SHORT:
|
|
{
|
|
SVal_LF_SHORT lf_short;
|
|
READ (*read_bytes, 2, len, lf_short.value, leaf_data, short);
|
|
parse_sctring (&lf_short.name, leaf_data, read_bytes, len);
|
|
val->name_or_val = malloc (sizeof (SVal_LF_SHORT));
|
|
if (!val->name_or_val) break;
|
|
memcpy (val->name_or_val, &lf_short, sizeof (SVal_LF_SHORT));
|
|
break;
|
|
}
|
|
case eLF_USHORT:
|
|
{
|
|
SVal_LF_USHORT lf_ushort;
|
|
READ (*read_bytes, 2, len, lf_ushort.value, leaf_data, unsigned short);
|
|
parse_sctring (&lf_ushort.name, leaf_data, read_bytes, len);
|
|
val->name_or_val = malloc (sizeof(SVal_LF_USHORT));
|
|
if (!val->name_or_val) break;
|
|
memcpy(val->name_or_val, &lf_ushort, sizeof (SVal_LF_USHORT));
|
|
break;
|
|
}
|
|
default:
|
|
printf ("parse_sval()::oops\n");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_enumerate(SLF_ENUMERATE *lf_enumerate, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int read_bytes_before = 0, tmp_read_bytes_before = 0;
|
|
|
|
lf_enumerate->enum_value.name_or_val = 0;
|
|
|
|
read_bytes_before = *read_bytes;
|
|
READ(*read_bytes, 2, len, lf_enumerate->fldattr.fldattr, leaf_data, unsigned short);
|
|
|
|
tmp_read_bytes_before = *read_bytes;
|
|
parse_sval(&lf_enumerate->enum_value, leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - tmp_read_bytes_before);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_enumerate->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_enumerate->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("%s:", "parse_lf_enumerate()");
|
|
// printf_sval_name(&lf_enumerate->enum_value);
|
|
// printf("\n");
|
|
|
|
return (*read_bytes - read_bytes_before);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_nesttype(SLF_NESTTYPE *lf_nesttype, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int read_bytes_before = *read_bytes;
|
|
|
|
lf_nesttype->name.name = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_nesttype->pad, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_nesttype->index, leaf_data, unsigned short);
|
|
|
|
parse_sctring(&lf_nesttype->name, leaf_data, read_bytes, len);
|
|
// printf("parse_lf_nesttype(): name = %s\n", lf_nesttype->name.name);
|
|
|
|
return *read_bytes - read_bytes_before;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_method(SLF_METHOD *lf_method, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int read_bytes_before = *read_bytes, tmp_read_bytes_before = 0;
|
|
|
|
lf_method->name.name = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_method->count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_method->mlist, leaf_data, unsigned int);
|
|
|
|
tmp_read_bytes_before = *read_bytes;
|
|
parse_sctring(&lf_method->name, leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - tmp_read_bytes_before);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_method->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_method->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("parse_lf_method(): name = %s\n", lf_method->name.name);
|
|
|
|
return *read_bytes - read_bytes_before;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_member(SLF_MEMBER *lf_member, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
int read_bytes_before = *read_bytes, tmp_read_bytes_before = 0;
|
|
|
|
lf_member->offset.name_or_val = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_member->fldattr.fldattr, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_member->inedex, leaf_data, unsigned int);
|
|
|
|
tmp_read_bytes_before = *read_bytes;
|
|
parse_sval(&lf_member->offset, leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - tmp_read_bytes_before);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_member->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_member->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("parse_lf_member(): name = ");
|
|
// printf_sval_name(&lf_member->offset);
|
|
// printf("\n");
|
|
|
|
return (*read_bytes - read_bytes_before);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_onemethod(SLF_ONEMETHOD *lf_onemethod, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
int read_bytes_before = *read_bytes, tmp_before_read_bytes = 0;
|
|
|
|
lf_onemethod->val.str_data.name = 0;
|
|
lf_onemethod->val.val = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_onemethod->fldattr.fldattr, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_onemethod->index, leaf_data, unsigned int);
|
|
|
|
lf_onemethod->fldattr.fldattr = SWAP_UINT16(lf_onemethod->fldattr.fldattr);
|
|
|
|
if((lf_onemethod->fldattr.bits.mprop == eMTintro) ||
|
|
(lf_onemethod->fldattr.bits.mprop == eMTpureintro)) {
|
|
READ(*read_bytes, 4, len, lf_onemethod->val.val, leaf_data, unsigned int);
|
|
}
|
|
|
|
tmp_before_read_bytes = *read_bytes;
|
|
parse_sctring(&(lf_onemethod->val.str_data), leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - tmp_before_read_bytes);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_onemethod->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_onemethod->pad, *read_bytes, leaf_data, len);
|
|
|
|
return (*read_bytes - read_bytes_before);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static void init_stype_info(STypeInfo *type_info)
|
|
{
|
|
type_info->free_ = 0;
|
|
type_info->get_members = 0;
|
|
type_info->get_name = 0;
|
|
type_info->get_val = 0;
|
|
type_info->get_name_len = 0;
|
|
type_info->get_arg_type = 0;
|
|
type_info->get_element_type = 0;
|
|
type_info->get_index_type = 0;
|
|
type_info->get_base_type = 0;
|
|
type_info->get_derived = 0;
|
|
type_info->get_vshape = 0;
|
|
type_info->get_utype = 0;
|
|
type_info->get_return_type = 0;
|
|
type_info->get_class_type = 0;
|
|
type_info->get_this_type = 0;
|
|
type_info->get_arglist = 0;
|
|
type_info->get_index = 0;
|
|
type_info->get_mlist = 0;
|
|
type_info->get_modified_type = 0;
|
|
type_info->is_fwdref = 0;
|
|
type_info->get_print_type = 0;
|
|
|
|
switch (type_info->leaf_type) {
|
|
case eLF_FIELDLIST:
|
|
type_info->get_members = get_fieldlist_members;
|
|
type_info->free_ = free_lf_fieldlist;
|
|
type_info->get_print_type = get_fieldlist_print_type;
|
|
break;
|
|
case eLF_ENUM:
|
|
type_info->get_name = get_enum_name;
|
|
type_info->get_name_len = get_enum_name_len;
|
|
type_info->get_members = get_enum_members;
|
|
type_info->get_utype = get_enum_utype;
|
|
type_info->free_ = free_lf_enum;
|
|
type_info->get_print_type = get_enum_print_type;
|
|
break;
|
|
case eLF_CLASS:
|
|
case eLF_STRUCTURE:
|
|
type_info->get_name = get_class_struct_name;
|
|
type_info->get_val = get_class_struct_val; // for structure this is size
|
|
type_info->get_name_len = get_class_struct_name_len;
|
|
type_info->get_members = get_struct_class_members;
|
|
type_info->get_derived = get_class_struct_derived;
|
|
type_info->get_vshape = get_class_struct_vshape;
|
|
type_info->is_fwdref = is_struct_class_fwdref;
|
|
type_info->free_ = free_lf_class;
|
|
type_info->get_print_type = get_class_struct_print_type;
|
|
break;
|
|
case eLF_POINTER:
|
|
type_info->get_utype = get_pointer_utype;
|
|
type_info->get_print_type = get_pointer_print_type;
|
|
break;
|
|
case eLF_ARRAY:
|
|
type_info->get_name = get_array_name;
|
|
type_info->get_val = get_array_val;
|
|
type_info->get_name_len = get_array_name_len;
|
|
type_info->get_element_type = get_array_element_type;
|
|
type_info->get_index_type = get_array_index_type;
|
|
type_info->free_ = free_lf_array;
|
|
type_info->get_print_type = get_array_print_type;
|
|
break;
|
|
case eLF_MODIFIER:
|
|
type_info->get_modified_type = get_modifier_modified_type;
|
|
type_info->get_print_type = get_modifier_print_type;
|
|
break;
|
|
case eLF_ARGLIST:
|
|
type_info->free_ = free_lf_arglist;
|
|
type_info->get_print_type = get_arglist_print_type;
|
|
break;
|
|
case eLF_MFUNCTION:
|
|
type_info->get_return_type = get_mfunction_return_type;
|
|
type_info->get_class_type = get_mfunction_class_type;
|
|
type_info->get_this_type = get_mfunction_this_type;
|
|
type_info->get_arglist = get_mfunction_arglist;
|
|
type_info->get_print_type = get_mfunction_print_type;
|
|
break;
|
|
case eLF_METHODLIST:
|
|
break;
|
|
case eLF_PROCEDURE:
|
|
type_info->get_return_type = get_procedure_return_type;
|
|
type_info->get_arglist = get_procedure_arglist;
|
|
type_info->get_print_type = get_procedure_print_type;
|
|
break;
|
|
case eLF_UNION:
|
|
type_info->get_name = get_union_name;
|
|
type_info->get_val = get_union_val;
|
|
type_info->get_name_len = get_union_name_len;
|
|
type_info->get_members = get_union_members;
|
|
type_info->is_fwdref = is_union_fwdref;
|
|
type_info->free_ = free_lf_union;
|
|
type_info->get_print_type = get_union_print_type;
|
|
break;
|
|
case eLF_BITFIELD:
|
|
type_info->get_base_type = get_bitfield_base_type;
|
|
type_info->get_print_type = get_bitfield_print_type;
|
|
break;
|
|
case eLF_VTSHAPE:
|
|
type_info->free_ = free_lf_vtshape;
|
|
type_info->get_print_type = get_vtshape_print_type;
|
|
break;
|
|
case eLF_ENUMERATE:
|
|
type_info->get_name = get_enumerate_name;
|
|
type_info->get_val = get_enumerate_val;
|
|
type_info->get_name_len = get_enumerate_name_len;
|
|
type_info->free_ = free_lf_enumerate;
|
|
type_info->get_print_type = get_enumerate_print_type;
|
|
break;
|
|
case eLF_NESTTYPE:
|
|
type_info->get_name = get_nesttype_name;
|
|
type_info->get_name_len = get_nesttype_name_len;
|
|
type_info->get_index = get_nesttype_index;
|
|
type_info->free_ = free_lf_nesttype;
|
|
type_info->get_print_type = get_nesttype_print_type;
|
|
break;
|
|
case eLF_METHOD:
|
|
type_info->get_name = get_method_name;
|
|
type_info->get_name_len = get_method_name_len;
|
|
type_info->get_mlist = get_method_mlist;
|
|
type_info->free_ = free_lf_method;
|
|
type_info->get_print_type = get_method_print_type;
|
|
break;
|
|
case eLF_MEMBER:
|
|
type_info->get_name = get_member_name;
|
|
type_info->get_val = get_member_val;
|
|
type_info->get_name_len = get_member_name_len;
|
|
type_info->get_index = get_member_index;
|
|
type_info->free_ = free_lf_member;
|
|
type_info->get_print_type = get_member_print_type;
|
|
break;
|
|
case eLF_ONEMETHOD:
|
|
type_info->get_name = get_onemethod_name;
|
|
type_info->get_name_len = get_onemethod_name_len;
|
|
type_info->get_val = get_onemethod_val;
|
|
type_info->get_index = get_onemethod_index;
|
|
type_info->free_ = free_lf_onemethod;
|
|
type_info->get_print_type = get_onemethod_print_type;
|
|
break;
|
|
default:
|
|
// printf("init_stype_info(): unknown type for init\n");
|
|
type_info->get_name = 0;
|
|
type_info->get_val = 0;
|
|
type_info->get_name_len = 0;
|
|
type_info->get_members = 0;
|
|
type_info->get_arg_type = 0;
|
|
type_info->get_element_type = 0;
|
|
type_info->get_index_type = 0;
|
|
type_info->get_base_type = 0;
|
|
type_info->get_derived = 0;
|
|
type_info->get_vshape = 0;
|
|
type_info->get_utype = 0;
|
|
type_info->get_return_type = 0;
|
|
type_info->get_class_type = 0;
|
|
type_info->get_this_type = 0;
|
|
type_info->get_arglist = 0;
|
|
type_info->get_index = 0;
|
|
type_info->get_mlist = 0;
|
|
type_info->get_print_type = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
#define PARSE_LF2(lf_type, lf_func_name, type) { \
|
|
STypeInfo *type_info = (STypeInfo *) malloc (sizeof (STypeInfo)); \
|
|
if (!type_info) return 0; \
|
|
lf_type *lf = (lf_type *) malloc (sizeof (lf_type)); \
|
|
if (!lf) { \
|
|
free (type_info); \
|
|
return 0; \
|
|
} \
|
|
curr_read_bytes = parse_##lf_func_name (lf, p, read_bytes, len); \
|
|
type_info->type_info = (void *) lf; \
|
|
type_info->leaf_type = type; \
|
|
init_stype_info (type_info); \
|
|
r_list_append (lf_fieldlist->substructs, type_info); \
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_fieldlist(SLF_FIELDLIST *lf_fieldlist, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
ELeafType leaf_type;
|
|
int curr_read_bytes = 0;
|
|
unsigned char *p = leaf_data;
|
|
|
|
lf_fieldlist->substructs = r_list_new();
|
|
|
|
while (*read_bytes <= len) {
|
|
READ(*read_bytes, 2, len, leaf_type, p, unsigned short);
|
|
switch (leaf_type) {
|
|
case eLF_ENUMERATE:
|
|
PARSE_LF2(SLF_ENUMERATE, lf_enumerate, eLF_ENUMERATE);
|
|
break;
|
|
case eLF_NESTTYPE:
|
|
PARSE_LF2(SLF_NESTTYPE, lf_nesttype, eLF_NESTTYPE);
|
|
break;
|
|
case eLF_METHOD:
|
|
PARSE_LF2(SLF_METHOD, lf_method, eLF_METHOD);
|
|
break;
|
|
case eLF_MEMBER:
|
|
PARSE_LF2(SLF_MEMBER, lf_member, eLF_MEMBER);
|
|
break;
|
|
case eLF_ONEMETHOD:
|
|
PARSE_LF2(SLF_ONEMETHOD, lf_onemethod, eLF_ONEMETHOD);
|
|
break;
|
|
default:
|
|
// printf("unsupported leaf type in parse_lf_fieldlist()\n");
|
|
return 0;
|
|
}
|
|
|
|
if (curr_read_bytes != 0) {
|
|
p += curr_read_bytes;
|
|
} else return 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_enum(SLF_ENUM *lf_enum, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
unsigned int before_read_bytes = 0;
|
|
|
|
lf_enum->name.name = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_enum->count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 2, len, lf_enum->prop.cv_property, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_enum->utype, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_enum->field_list, leaf_data, unsigned int);
|
|
|
|
lf_enum->prop.cv_property = SWAP_UINT16(lf_enum->prop.cv_property);
|
|
before_read_bytes = *read_bytes;
|
|
parse_sctring(&lf_enum->name, leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - before_read_bytes);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_enum->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_enum->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("parse_lf_enum(): name = %s\n", lf_enum->name.name);
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_class(SLF_CLASS *lf_class, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
// SLF_CLASS lf_class;
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
unsigned int before_read_bytes = 0;
|
|
|
|
lf_class->size.name_or_val = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_class->count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 2, len, lf_class->prop.cv_property, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_class->field_list, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_class->derived, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_class->vshape, leaf_data, unsigned int);
|
|
|
|
before_read_bytes = *read_bytes;
|
|
parse_sval(&lf_class->size, leaf_data, read_bytes, len);
|
|
before_read_bytes = *read_bytes - before_read_bytes;
|
|
leaf_data = (unsigned char *)leaf_data + before_read_bytes;
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_class->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_class->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("%s:", "parse_lf_class()");
|
|
// printf_sval_name(&lf_class->size);
|
|
// printf("\n");
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_structure(SLF_STRUCTURE *lf_structure, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
// SLF_STRUCTURE lf_structure;
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
unsigned int before_read_bytes = 0;
|
|
|
|
lf_structure->size.name_or_val = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_structure->count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 2, len, lf_structure->prop.cv_property, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_structure->field_list, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_structure->derived, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_structure->vshape, leaf_data, unsigned int);
|
|
|
|
lf_structure->prop.cv_property = SWAP_UINT16(lf_structure->prop.cv_property);
|
|
|
|
before_read_bytes = *read_bytes;
|
|
parse_sval(&lf_structure->size, leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - before_read_bytes);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_structure->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_structure->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("parse_lf_structure(): name = ");
|
|
// printf_sval_name(&lf_structure->size);
|
|
// printf("\n");
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_pointer(SLF_POINTER *lf_pointer, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
|
|
READ(*read_bytes, 4, len, lf_pointer->utype, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_pointer->ptr_attr.ptr_attr, leaf_data, unsigned int);
|
|
|
|
lf_pointer->ptr_attr.ptr_attr = SWAP_UINT32(lf_pointer->ptr_attr.ptr_attr);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_pointer->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_pointer->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_array(SLF_ARRAY *lf_array, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
unsigned int before_read_bytes = 0;
|
|
|
|
lf_array->size.name_or_val = 0;
|
|
|
|
READ(*read_bytes, 4, len, lf_array->element_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_array->index_type, leaf_data, unsigned int);
|
|
|
|
before_read_bytes = *read_bytes;
|
|
parse_sval(&lf_array->size, leaf_data, read_bytes, len);
|
|
leaf_data += (*read_bytes - before_read_bytes);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_array->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_array->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("parse_lf_array(): name = ");
|
|
// printf_sval_name(&lf_array->size);
|
|
// printf("\n");
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_modifier(SLF_MODIFIER *lf_modifier, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
|
|
READ(*read_bytes, 4, len, lf_modifier->modified_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 2, len, lf_modifier->umodifier.modifier, leaf_data, unsigned short);
|
|
lf_modifier->umodifier.modifier = SWAP_UINT16(lf_modifier->umodifier.modifier);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_modifier->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_modifier->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_arglist(SLF_ARGLIST *lf_arglist, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
|
|
lf_arglist->arg_type = 0;
|
|
|
|
READ (*read_bytes, 4, len, lf_arglist->count, leaf_data, unsigned int);
|
|
|
|
lf_arglist->arg_type = (unsigned int *) malloc (lf_arglist->count * 4);
|
|
if (!lf_arglist->arg_type) return 0;
|
|
memcpy (lf_arglist->arg_type, leaf_data, lf_arglist->count * 4);
|
|
leaf_data += (lf_arglist->count * 4);
|
|
*read_bytes += (lf_arglist->count * 4);
|
|
|
|
PEEK_READ (*read_bytes, 1, len, lf_arglist->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN (lf_arglist->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_mfunction(SLF_MFUNCTION *lf_mfunction, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
|
|
READ(*read_bytes, 4, len, lf_mfunction->return_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_mfunction->class_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_mfunction->this_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 1, len, lf_mfunction->call_conv, leaf_data, unsigned char);
|
|
READ(*read_bytes, 1, len, lf_mfunction->reserved, leaf_data, unsigned char);
|
|
READ(*read_bytes, 2, len, lf_mfunction->parm_count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_mfunction->arglist, leaf_data, unsigned int);
|
|
READ(*read_bytes, 4, len, lf_mfunction->this_adjust, leaf_data, int);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_mfunction->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_mfunction->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_procedure(SLF_PROCEDURE *lf_procedure, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
|
|
READ(*read_bytes, 4, len, lf_procedure->return_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 1, len, lf_procedure->call_conv, leaf_data, unsigned char);
|
|
READ(*read_bytes, 1, len, lf_procedure->reserved, leaf_data, unsigned char);
|
|
READ(*read_bytes, 2, len, lf_procedure->parm_count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_procedure->arg_list, leaf_data, unsigned int);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_procedure->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_procedure->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_union(SLF_UNION *lf_union, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
unsigned int before_read_bytes = 0;
|
|
|
|
lf_union->size.name_or_val = 0;
|
|
|
|
READ(*read_bytes, 2, len, lf_union->count, leaf_data, unsigned short);
|
|
READ(*read_bytes, 2, len, lf_union->prop.cv_property, leaf_data, unsigned short);
|
|
READ(*read_bytes, 4, len, lf_union->field_list, leaf_data, unsigned int);
|
|
|
|
before_read_bytes = *read_bytes;
|
|
parse_sval(&lf_union->size, leaf_data, read_bytes, len);
|
|
before_read_bytes = *read_bytes - before_read_bytes;
|
|
leaf_data = (unsigned char *)leaf_data + before_read_bytes;
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_union->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_union->pad, *read_bytes, leaf_data, len);
|
|
|
|
// printf("%s:", "parse_lf_union()");
|
|
// printf_sval_name(&lf_union->size);
|
|
// printf("\n");
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_bitfield(SLF_BITFIELD *lf_bitfield, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
|
|
READ(*read_bytes, 4, len, lf_bitfield->base_type, leaf_data, unsigned int);
|
|
READ(*read_bytes, 1, len, lf_bitfield->length, leaf_data, unsigned char);
|
|
READ(*read_bytes, 1, len, lf_bitfield->position, leaf_data, unsigned char);
|
|
|
|
PEEK_READ(*read_bytes, 1, len, lf_bitfield->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN(lf_bitfield->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_lf_vtshape(SLF_VTSHAPE *lf_vtshape, unsigned char *leaf_data, unsigned int *read_bytes, unsigned int len)
|
|
{
|
|
unsigned int tmp_before_read_bytes = *read_bytes;
|
|
unsigned int size; // in bytes;
|
|
|
|
lf_vtshape->vt_descriptors = 0;
|
|
|
|
READ (*read_bytes, 2, len, lf_vtshape->count, leaf_data, unsigned short);
|
|
|
|
size = (4 * lf_vtshape->count + (lf_vtshape->count % 2) * 4) / 8;
|
|
lf_vtshape->vt_descriptors = (char *) malloc (size);
|
|
if (!lf_vtshape->vt_descriptors) return 0;
|
|
memcpy (lf_vtshape->vt_descriptors, leaf_data, size);
|
|
leaf_data += size;
|
|
*read_bytes += size;
|
|
|
|
PEEK_READ (*read_bytes, 1, len, lf_vtshape->pad, leaf_data, unsigned char);
|
|
PAD_ALIGN (lf_vtshape->pad, *read_bytes, leaf_data, len);
|
|
|
|
return *read_bytes - tmp_before_read_bytes;
|
|
}
|
|
|
|
#define PARSE_LF(lf_type, lf_func) { \
|
|
lf_type *lf = (lf_type *) malloc(sizeof(lf_type)); \
|
|
if (!lf) return 0; \
|
|
parse_##lf_func(lf, leaf_data + 2, &read_bytes, type->length); \
|
|
type->type_data.type_info = (void *) lf; \
|
|
init_stype_info(&type->type_data); \
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
static int parse_tpi_stypes(R_STREAM_FILE *stream, SType *type) {
|
|
unsigned char *leaf_data;
|
|
unsigned int read_bytes = 0;
|
|
|
|
stream_file_read(stream, 2, (char *)&type->length);
|
|
if (type->length<1)
|
|
return 0;
|
|
leaf_data = (unsigned char *) malloc(type->length);
|
|
if (!leaf_data) return 0;
|
|
stream_file_read (stream, type->length, (char *)leaf_data);
|
|
type->type_data.leaf_type = *(unsigned short *)leaf_data;
|
|
read_bytes += 2;
|
|
switch (type->type_data.leaf_type) {
|
|
case eLF_FIELDLIST:
|
|
// printf("eLF_FIELDLIST\n");
|
|
PARSE_LF(SLF_FIELDLIST, lf_fieldlist);
|
|
break;
|
|
case eLF_ENUM:
|
|
// printf("eLF_ENUM\n");
|
|
PARSE_LF(SLF_ENUM, lf_enum);
|
|
break;
|
|
// TODO: combine with eLF_STRUCTURE
|
|
case eLF_CLASS:
|
|
// printf("eLF_CLASS\n");
|
|
PARSE_LF(SLF_CLASS, lf_class);
|
|
break;
|
|
case eLF_STRUCTURE:
|
|
// printf("eLF_STRUCTURE\n");
|
|
PARSE_LF(SLF_STRUCTURE, lf_structure);
|
|
break;
|
|
case eLF_POINTER:
|
|
// printf("eLF_POINTER\n");
|
|
{
|
|
SLF_POINTER *lf = (SLF_POINTER *) malloc(sizeof(SLF_POINTER)); \
|
|
if (!lf) { \
|
|
free (leaf_data); \
|
|
return 0; \
|
|
} \
|
|
parse_lf_pointer(lf, leaf_data + 2, &read_bytes, type->length); \
|
|
type->type_data.type_info = (void *) lf; \
|
|
init_stype_info(&type->type_data); \
|
|
}
|
|
// PARSE_LF(SLF_POINTER, lf_pointer);
|
|
break;
|
|
case eLF_ARRAY:
|
|
// printf("eLF_ARRAY\n");
|
|
PARSE_LF(SLF_ARRAY, lf_array);
|
|
break;
|
|
case eLF_MODIFIER:
|
|
// printf("eLF_MODIFIER\n");
|
|
PARSE_LF(SLF_MODIFIER, lf_modifier);
|
|
break;
|
|
case eLF_ARGLIST:
|
|
// printf("eLF_ARGLIST\n");
|
|
PARSE_LF(SLF_ARGLIST, lf_arglist);
|
|
break;
|
|
case eLF_MFUNCTION:
|
|
// printf("eLF_MFUNCTION\n");
|
|
PARSE_LF(SLF_MFUNCTION, lf_mfunction);
|
|
break;
|
|
case eLF_METHODLIST:
|
|
// printf("eLF_METHOD_LIST\n");
|
|
break;
|
|
case eLF_PROCEDURE:
|
|
// printf("eLF_PROCEDURE\n");
|
|
PARSE_LF(SLF_PROCEDURE, lf_procedure);
|
|
break;
|
|
case eLF_UNION:
|
|
// printf("eLF_UNION\n");
|
|
PARSE_LF(SLF_UNION, lf_union);
|
|
break;
|
|
case eLF_BITFIELD:
|
|
// printf("eLF_BITFIELD\n");
|
|
PARSE_LF(SLF_BITFIELD, lf_bitfield);
|
|
break;
|
|
case eLF_VTSHAPE:
|
|
// printf("eLF_VTSHAPE\n");
|
|
PARSE_LF(SLF_VTSHAPE, lf_vtshape);
|
|
break;
|
|
default:
|
|
printf("parse_tpi_streams(): unsupported leaf type\n");
|
|
break;
|
|
}
|
|
|
|
free (leaf_data);
|
|
return read_bytes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
int parse_tpi_stream(void *parsed_pdb_stream, R_STREAM_FILE *stream) {
|
|
int i;
|
|
SType *type = 0;
|
|
STpiStream *tpi_stream = (STpiStream *) parsed_pdb_stream;
|
|
tpi_stream->types = r_list_new();
|
|
p_types_list = tpi_stream->types;
|
|
|
|
stream_file_read(stream, sizeof(STPIHeader), (char *)&tpi_stream->header);
|
|
|
|
base_idx = tpi_stream->header.ti_min;
|
|
|
|
for (i = tpi_stream->header.ti_min; i < tpi_stream->header.ti_max; i++) {
|
|
type = (SType *) malloc(sizeof(SType));
|
|
if (!type) return 0;
|
|
type->tpi_idx = i;
|
|
type->type_data.type_info = 0;
|
|
type->type_data.leaf_type = eLF_MAX;
|
|
init_stype_info(&type->type_data);
|
|
if (!parse_tpi_stypes(stream, type)) {
|
|
free (type);
|
|
return 0;
|
|
}
|
|
r_list_append(tpi_stream->types, type);
|
|
}
|
|
return 1;
|
|
// Postprocessing...
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
void init_tpi_stream(STpiStream *tpi_stream)
|
|
{
|
|
tpi_stream->free_ = free_tpi_stream;
|
|
}
|