/* * $Id: val.c,v 1.35 2006-07-01 16:07:06 bacon Exp $ */ #include #ifndef XP_AWK_STAND_ALONE #include #include #include #endif static xp_awk_val_nil_t __awk_nil = { XP_AWK_VAL_NIL, 0 }; xp_awk_val_t* xp_awk_val_nil = (xp_awk_val_t*)&__awk_nil; static xp_awk_val_int_t __awk_int[] = { { XP_AWK_VAL_INT, 0, -1 }, { XP_AWK_VAL_INT, 0, 0 }, { XP_AWK_VAL_INT, 0, 1 }, { XP_AWK_VAL_INT, 0, 2 }, { XP_AWK_VAL_INT, 0, 3 }, { XP_AWK_VAL_INT, 0, 4 }, { XP_AWK_VAL_INT, 0, 5 }, { XP_AWK_VAL_INT, 0, 6 }, { XP_AWK_VAL_INT, 0, 7 }, { XP_AWK_VAL_INT, 0, 8 }, { XP_AWK_VAL_INT, 0, 9 }, }; xp_awk_val_t* xp_awk_makeintval (xp_awk_run_t* run, xp_long_t v) { xp_awk_val_int_t* val; if (v >= __awk_int[0].val && v <= __awk_int[xp_countof(__awk_int)-1].val) { return (xp_awk_val_t*)&__awk_int[v-__awk_int[0].val]; } if (run->icache_count > 0) { val = run->icache[--run->icache_count]; } else { val = (xp_awk_val_int_t*) xp_malloc (xp_sizeof(xp_awk_val_int_t)); if (val == XP_NULL) return XP_NULL; } val->type = XP_AWK_VAL_INT; val->ref = 0; val->val = v; return (xp_awk_val_t*)val; } xp_awk_val_t* xp_awk_makerealval (xp_awk_run_t* run, xp_real_t v) { xp_awk_val_real_t* val; if (run->rcache_count > 0) { val = run->rcache[--run->rcache_count]; } else { val = (xp_awk_val_real_t*) xp_malloc (xp_sizeof(xp_awk_val_real_t)); if (val == XP_NULL) return XP_NULL; } val->type = XP_AWK_VAL_REAL; val->ref = 0; val->val = v; return (xp_awk_val_t*)val; } xp_awk_val_t* xp_awk_makestrval0 (const xp_char_t* str) { return xp_awk_makestrval (str, xp_strlen(str)); } xp_awk_val_t* xp_awk_makestrval (const xp_char_t* str, xp_size_t len) { xp_awk_val_str_t* val; val = (xp_awk_val_str_t*) xp_malloc (xp_sizeof(xp_awk_val_str_t)); if (val == XP_NULL) return XP_NULL; val->type = XP_AWK_VAL_STR; val->ref = 0; val->len = len; val->buf = xp_strxdup (str, len); if (val->buf == XP_NULL) { xp_free (val); return XP_NULL; } return (xp_awk_val_t*)val; } xp_awk_val_t* xp_awk_makestrval2 ( const xp_char_t* str1, xp_size_t len1, const xp_char_t* str2, xp_size_t len2) { xp_awk_val_str_t* val; val = (xp_awk_val_str_t*) xp_malloc (xp_sizeof(xp_awk_val_str_t)); if (val == XP_NULL) return XP_NULL; val->type = XP_AWK_VAL_STR; val->ref = 0; val->len = len1 + len2; val->buf = xp_strxdup2 (str1, len1, str2, len2); if (val->buf == XP_NULL) { xp_free (val); return XP_NULL; } return (xp_awk_val_t*)val; } xp_awk_val_t* xp_awk_makerexval (const xp_char_t* str, xp_size_t len) { xp_awk_val_rex_t* val; /* TDOO: implement the regular expression value */ val = (xp_awk_val_rex_t*) xp_malloc (xp_sizeof(xp_awk_val_rex_t)); if (val == XP_NULL) return XP_NULL; val->type = XP_AWK_VAL_STR; val->ref = 0; val->len = len; val->buf = xp_strxdup (str, len); if (val->buf == XP_NULL) { xp_free (val); return XP_NULL; } return (xp_awk_val_t*)val; } static void __free_map_val (void* run, void* v) { /* xp_printf (XP_T("refdown in map free...")); xp_awk_printval (v); xp_printf (XP_T("\n")); */ xp_awk_refdownval (run, v); } xp_awk_val_t* xp_awk_makemapval (xp_awk_run_t* run) { xp_awk_val_map_t* val; val = (xp_awk_val_map_t*) xp_malloc (xp_sizeof(xp_awk_val_map_t)); if (val == XP_NULL) return XP_NULL; val->type = XP_AWK_VAL_MAP; val->ref = 0; val->map = xp_awk_map_open (XP_NULL, run, 256, __free_map_val); if (val->map == XP_NULL) { xp_free (val); return XP_NULL; } return (xp_awk_val_t*)val; } xp_bool_t xp_awk_isbuiltinval (xp_awk_val_t* val) { return val == XP_NULL || val == xp_awk_val_nil || (val >= (xp_awk_val_t*)&__awk_int[0] && val <= (xp_awk_val_t*)&__awk_int[xp_countof(__awk_int)-1]); } void xp_awk_freeval (xp_awk_run_t* run, xp_awk_val_t* val, xp_bool_t cache) { if (xp_awk_isbuiltinval(val)) return; /*xp_printf (XP_T("freeing [cache=%d] ... "), cache); xp_awk_printval (val); xp_printf (XP_T("\n"));*/ switch (val->type) { case XP_AWK_VAL_NIL: xp_free (val); return; case XP_AWK_VAL_INT: if (cache == xp_true && run->icache_count < xp_countof(run->icache)) { run->icache[run->icache_count++] = (xp_awk_val_int_t*)val; } else xp_free (val); return; case XP_AWK_VAL_REAL: if (cache == xp_true && run->rcache_count < xp_countof(run->rcache)) { run->rcache[run->rcache_count++] = (xp_awk_val_real_t*)val; } else xp_free (val); return; case XP_AWK_VAL_STR: xp_free (((xp_awk_val_str_t*)val)->buf); xp_free (val); return; case XP_AWK_VAL_REX: /* TODO: implement the regular expression value properly */ xp_free (((xp_awk_val_rex_t*)val)->buf); xp_free (val); return; case XP_AWK_VAL_MAP: xp_awk_map_close (((xp_awk_val_map_t*)val)->map); xp_free (val); return; } xp_assert (!"should never happen - invalid value type"); } void xp_awk_refupval (xp_awk_val_t* val) { if (xp_awk_isbuiltinval(val)) return; /* xp_printf (XP_T("ref up ")); xp_awk_printval (val); xp_printf (XP_T("\n")); */ val->ref++; } void xp_awk_refdownval (xp_awk_run_t* run, xp_awk_val_t* val) { if (xp_awk_isbuiltinval(val)) return; /* xp_printf (XP_T("%p, %p, %p\n"), xp_awk_val_nil, &__awk_nil, val); xp_printf (XP_T("ref down [count=>%d]\n"), (int)val->ref); xp_awk_printval (val); xp_printf (XP_T("\n")); */ xp_assert (val->ref > 0); val->ref--; if (val->ref <= 0) { /* xp_printf (XP_T("**FREEING ")); xp_awk_printval (val); xp_printf (XP_T("\n")); */ xp_awk_freeval(run, val, xp_true); } } void xp_awk_refdownval_nofree (xp_awk_run_t* run, xp_awk_val_t* val) { if (xp_awk_isbuiltinval(val)) return; xp_assert (val->ref > 0); val->ref--; } xp_bool_t xp_awk_valtobool (xp_awk_val_t* val) { if (val == XP_NULL) return xp_false; switch (val->type) { case XP_AWK_VAL_NIL: return xp_false; case XP_AWK_VAL_INT: return ((xp_awk_val_int_t*)val)->val != 0; case XP_AWK_VAL_REAL: return ((xp_awk_val_real_t*)val)->val != 0.0; case XP_AWK_VAL_STR: return ((xp_awk_val_str_t*)val)->len > 0; case XP_AWK_VAL_MAP: return xp_false; /* TODO: is this correct? */ } xp_assert (!"should never happen - invalid value type"); return xp_false; } xp_char_t* xp_awk_valtostr (xp_awk_val_t* v, int* errnum, xp_str_t* buf) { if (v->type == XP_AWK_VAL_NIL) { if (buf == XP_NULL) { xp_char_t* tmp; tmp = xp_strdup (XP_T("")); if (tmp == XP_NULL) *errnum = XP_AWK_ENOMEM; return tmp; } else { xp_str_clear (buf); return XP_STR_BUF(buf); } } if (v->type == XP_AWK_VAL_INT) { xp_char_t* tmp; xp_long_t t; xp_size_t len = 0; t = ((xp_awk_val_int_t*)v)->val; if (t == 0) { /* handle zero */ if (buf == XP_NULL) { tmp = xp_malloc (2 * xp_sizeof(xp_char_t)); if (tmp == XP_NULL) { *errnum = XP_AWK_ENOMEM; return XP_NULL; } tmp[0] = XP_T('0'); tmp[1] = XP_T('\0'); return tmp; } else { if (xp_str_cat (buf, XP_T("0")) == (xp_size_t)-1) { *errnum = XP_AWK_ENOMEM; return XP_NULL; } return XP_STR_BUF(buf); } } /* non-zero values */ if (t < 0) { t = -t; len++; } while (t > 0) { len++; t /= 10; } if (buf == XP_NULL) { tmp = xp_malloc ((len + 1) * xp_sizeof(xp_char_t)); if (tmp == XP_NULL) { *errnum = XP_AWK_ENOMEM; return XP_NULL; } tmp[len] = XP_T('\0'); } else { /* get the current end of the buffer */ tmp = XP_STR_BUF(buf) + XP_STR_LEN(buf); /* extend the buffer */ if (xp_str_nccat ( buf, XP_T(' '), len) == (xp_size_t)-1) { *errnum = XP_AWK_ENOMEM; return XP_NULL; } } t = ((xp_awk_val_int_t*)v)->val; if (t < 0) t = -t; while (t > 0) { tmp[--len] = (xp_char_t)(t % 10) + XP_T('0'); t /= 10; } if (((xp_awk_val_int_t*)v)->val < 0) tmp[--len] = XP_T('-'); /*return (buf == XP_NULL) tmp: XP_STR_BUF(buf);*/ return tmp; } if (v->type == XP_AWK_VAL_STR) { xp_char_t* tmp; if (buf == XP_NULL) { tmp = xp_strxdup ( ((xp_awk_val_str_t*)v)->buf, ((xp_awk_val_str_t*)v)->len); if (tmp == XP_NULL) *errnum = XP_AWK_ENOMEM; } else { tmp = XP_STR_BUF(buf) + XP_STR_LEN(buf); if (xp_str_ncat (buf, ((xp_awk_val_str_t*)v)->buf, ((xp_awk_val_str_t*)v)->len) == (xp_size_t)-1) { *errnum = XP_AWK_ENOMEM; tmp = XP_NULL; } } return tmp; } /* TODO: process more value types */ *errnum = XP_AWK_EVALTYPE; return XP_NULL; } static int __print_pair (xp_awk_pair_t* pair, void* arg) { xp_printf (XP_T(" %s=>"), pair->key); xp_awk_printval (pair->val); xp_printf (XP_T(" ")); return 0; } void xp_awk_printval (xp_awk_val_t* val) { /* TODO: better value printing...................... */ switch (val->type) { case XP_AWK_VAL_NIL: xp_printf (XP_T("nil")); break; case XP_AWK_VAL_INT: #if defined(__LCC__) xp_printf (XP_T("%lld"), (long long)((xp_awk_val_int_t*)val)->val); #elif defined(__BORLANDC__) || defined(_MSC_VER) xp_printf (XP_T("%I64d"), (__int64)((xp_awk_nde_int_t*)val)->val); #elif defined(vax) || defined(__vax) || defined(_SCO_DS) xp_printf (XP_T("%ld"), (long)((xp_awk_val_int_t*)val)->val); #else xp_printf (XP_T("%lld"), (long long)((xp_awk_val_int_t*)val)->val); #endif break; case XP_AWK_VAL_REAL: xp_printf (XP_T("%Lf"), (long double)((xp_awk_val_real_t*)val)->val); break; case XP_AWK_VAL_STR: xp_printf (XP_T("%s"), ((xp_awk_val_str_t*)val)->buf); break; case XP_AWK_VAL_REX: xp_printf (XP_T("REX[%s]"), ((xp_awk_val_rex_t*)val)->buf); break; case XP_AWK_VAL_MAP: xp_printf (XP_T("MAP[")); xp_awk_map_walk (((xp_awk_val_map_t*)val)->map, __print_pair, XP_NULL); xp_printf (XP_T("]")); break; default: xp_assert (!"should never happen - invalid value type"); xp_printf (XP_T("**** INTERNAL ERROR - INVALID VALUE TYPE ****\n")); } }