X-Git-Url: https://git.m6w6.name/?p=m6w6%2Fext-psi;a=blobdiff_plain;f=src%2Fparser.h;h=29bf9d326ce17727b66bd18969c9d774110de184;hp=aadb31c3ce546f619b2bf232b533628b4ce3ef9d;hb=b0a8b273be0744ed445fbe6c935a04ee7a749f14;hpb=19f0eb5807fb8d0f3aa305f2476f094e66a02709 diff --git a/src/parser.h b/src/parser.h index aadb31c..29bf9d3 100644 --- a/src/parser.h +++ b/src/parser.h @@ -13,6 +13,8 @@ #define BSIZE 256 #define PSI_T_POINTER PSI_T_ASTERISK +#define PSI_T_LONG_DOUBLE (PSI_T_DOUBLE << 16) + typedef int token_t; /* in php_psi.h */ @@ -65,6 +67,40 @@ static inline decl_type *init_decl_type(token_t type, const char *name) { return t; } +static inline decl_type *init_decl_type_ex(token_t type, int argc, ...) { + va_list argv; + char *ptr, *arg; + unsigned i; + size_t len, pos = 0, all = 0; + decl_type *t = calloc(1, sizeof(*t)); + + va_start(argv, argc); + for (i = 0; i < argc; ++i) { + arg = va_arg(argv, char *); + len = va_arg(argv, size_t); + + if (len) { + if (all) { + pos = all; + ptr = realloc(ptr, 1 + (all += len)); + } else { + ptr = malloc(1 + (all = len)); + } + memcpy(ptr + pos, arg, len); + } + } + va_end(argv); + + if (!all) { + ptr = calloc(1, 1); + } else { + ptr[all] = 0; + } + t->type = type; + t->name = ptr; + return t; +} + static inline decl_type *real_decl_type(decl_type *type) { while (type->real) { type = type->real; @@ -341,6 +377,10 @@ typedef struct decl_struct { char *name; decl_args *args; size_t size; + struct { + void *type; + void (*dtor)(void *type); + } engine; } decl_struct; static inline decl_struct *init_decl_struct(const char *name, decl_args *args) { @@ -822,13 +862,18 @@ static inline set_value *init_set_value(set_func *func, decl_vars *vars) { static inline set_value *add_inner_set_value(set_value *val, set_value *inner) { val->inner = realloc(val->inner, ++val->count * sizeof(*val->inner)); val->inner[val->count-1] = inner; + inner->outer.set = val; return val; } static inline void free_set_value(set_value *val) { - free_set_func(val->func); - free_decl_vars(val->vars); - if (val->inner) { + if (val->func) { + free_set_func(val->func); + } + if (val->vars) { + free_decl_vars(val->vars); + } + if (val->inner && (!val->outer.set || val->outer.set->inner != val->inner)) { size_t i; for (i = 0; i < val->count; ++i) { free_set_value(val->inner[i]); @@ -1140,7 +1185,16 @@ static inline void add_decl_lib(decl_libs *libs, void *dlopened) { static inline impl_val *deref_impl_val(impl_val *ret_val, decl_var *var) { unsigned i; - if (var->arg->var != var) for (i = 1; i < var->pointer_level; ++i) { + ZEND_ASSERT(var->arg->var != var); +#if 0 + fprintf(stderr, "deref: %s pl=%u:%u as=%u:%u %p\n", + var->name, var->pointer_level, var->arg->var->pointer_level, + var->array_size, var->arg->var->array_size, ret_val); +#endif + for (i = 0; i < var->pointer_level; ++i) { +#if 0 + fprintf(stderr, "-- %p %p %p\n", ret_val, *(void**)ret_val, ret_val->ptr); +#endif ret_val = *(void **) ret_val; } return ret_val; @@ -1150,11 +1204,21 @@ static inline impl_val *enref_impl_val(void *ptr, decl_var *var) { impl_val *val, *val_ptr; unsigned i; + ZEND_ASSERT(var->arg->var == var); +#if 0 + fprintf(stderr, "enref: %s pl=%u:%u as=%u:%u\n", + var->name, var->pointer_level, var->arg->var->pointer_level, + var->array_size, var->arg->var->array_size); +#endif if (!var->pointer_level && real_decl_type(var->arg->type)->type != PSI_T_STRUCT) { return ptr; } + val = val_ptr = calloc(var->pointer_level + 1, sizeof(void *)); - for (i = 1; i < var->pointer_level; ++i) { + for (i = !var->arg->var->array_size; i < var->pointer_level; ++i) { +#if 0 + fprintf(stderr, "++\n"); +#endif val_ptr->ptr = (void **) val_ptr + 1; val_ptr = val_ptr->ptr; } @@ -1163,16 +1227,14 @@ static inline impl_val *enref_impl_val(void *ptr, decl_var *var) { } static inline impl_val *struct_member_ref(decl_arg *set_arg, impl_val *struct_ptr, impl_val **to_free) { - void *ptr = (char *) struct_ptr->ptr + set_arg->layout->pos; - impl_val *val = enref_impl_val(ptr, set_arg->var); - - if (val != ptr) { - *to_free = val; - } - - return val; + void *ptr = (char *) struct_ptr + set_arg->layout->pos; +#if 0 + fprintf(stderr, "struct member %s: %p\n", set_arg->var->name, ptr); +#endif + return ptr; } + #define PSI_ERROR 16 #define PSI_WARNING 32 typedef void (*psi_error_cb)(PSI_Token *token, int type, const char *msg, ...); @@ -1230,6 +1292,10 @@ typedef struct PSI_Parser { char *cur, *tok, *lim, *eof, *ctx, *mrk, buf[BSIZE]; } PSI_Parser; +static inline size_t PSI_TokenAllocSize(size_t token_len, size_t fname_len) { + return sizeof(PSI_Token) + token_len + fname_len + sizeof(unsigned) + 2; +} + static inline PSI_Token *PSI_TokenAlloc(PSI_Parser *P) { PSI_Token *T; size_t token_len, fname_len; @@ -1243,7 +1309,7 @@ static inline PSI_Token *PSI_TokenAlloc(PSI_Parser *P) { token_len = P->cur - P->tok; fname_len = strlen(P->psi.file.fn); - T = calloc(1, sizeof(*T) + token_len + fname_len + sizeof(unsigned) + 2); + T = calloc(1, PSI_TokenAllocSize(token_len, fname_len)); T->type = token_typ; T->size = token_len; T->text = &T->buf[0]; @@ -1258,8 +1324,7 @@ static inline PSI_Token *PSI_TokenAlloc(PSI_Parser *P) { } static inline PSI_Token *PSI_TokenCopy(PSI_Token *src) { - size_t fname_len = strlen(src->file); - size_t strct_len = sizeof(*src) + src->size + fname_len + sizeof(unsigned) + 2; + size_t strct_len = PSI_TokenAllocSize(src->size, strlen(src->file)); PSI_Token *ptr = malloc(strct_len); memcpy(ptr, src, strct_len); @@ -1270,6 +1335,32 @@ static inline PSI_Token *PSI_TokenCopy(PSI_Token *src) { return ptr; } +static inline PSI_Token *PSI_TokenCat(unsigned argc, ...) { + va_list argv; + unsigned i; + PSI_Token *T = NULL; + + va_start(argv, argc); + for (i = 0; i < argc; ++i) { + PSI_Token *arg = va_arg(argv, PSI_Token *); + + if (T) { + size_t fname_len = strlen(T->file); + + T = realloc(T, PSI_TokenAllocSize(T->size + arg->size, fname_len)); + memmove(&T->buf[T->size + 1], T->file, fname_len + 1); + memcpy(T->file - 1, arg->text, arg->size + 1); + T->file = &T->buf[T->size + 1]; + } else { + T = PSI_TokenCopy(arg); + T->type = PSI_T_NAME; + } + } + va_end(argv); + + return T; +} + static inline const char *PSI_TokenLocation(PSI_Token *t) { return t ? t->file : ":0:0"; }