2006-03-04 15:54:37 +00:00
|
|
|
/*
|
2006-08-03 05:05:48 +00:00
|
|
|
* $Id: val.c,v 1.49 2006-08-03 05:05:48 bacon Exp $
|
2006-03-04 15:54:37 +00:00
|
|
|
*/
|
|
|
|
|
2006-03-31 16:35:37 +00:00
|
|
|
#include <xp/awk/awk_i.h>
|
2006-03-04 15:54:37 +00:00
|
|
|
|
2006-04-16 04:31:38 +00:00
|
|
|
#ifndef XP_AWK_STAND_ALONE
|
2006-03-07 16:09:18 +00:00
|
|
|
#include <xp/bas/string.h>
|
|
|
|
#include <xp/bas/memory.h>
|
2006-03-27 11:43:17 +00:00
|
|
|
#include <xp/bas/assert.h>
|
2006-03-07 16:09:18 +00:00
|
|
|
#endif
|
|
|
|
|
2006-03-22 16:05:50 +00:00
|
|
|
static xp_awk_val_nil_t __awk_nil = { XP_AWK_VAL_NIL, 0 };
|
2006-03-07 15:55:14 +00:00
|
|
|
xp_awk_val_t* xp_awk_val_nil = (xp_awk_val_t*)&__awk_nil;
|
|
|
|
|
2006-07-10 04:51:38 +00:00
|
|
|
static xp_awk_val_str_t __awk_zls = { XP_AWK_VAL_STR, 0, XP_T(""), 0 };
|
|
|
|
xp_awk_val_t* xp_awk_val_zls = (xp_awk_val_t*)&__awk_zls;
|
|
|
|
|
2006-03-22 16:05:50 +00:00
|
|
|
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 },
|
|
|
|
};
|
|
|
|
|
2006-07-12 07:25:15 +00:00
|
|
|
xp_awk_val_t* xp_awk_val_zero = (xp_awk_val_t*)&__awk_int[1];
|
|
|
|
xp_awk_val_t* xp_awk_val_one = (xp_awk_val_t*)&__awk_int[2];
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
xp_awk_val_t* xp_awk_makeintval (xp_awk_run_t* run, xp_long_t v)
|
2006-03-04 15:54:37 +00:00
|
|
|
{
|
|
|
|
xp_awk_val_int_t* val;
|
|
|
|
|
2006-03-22 16:05:50 +00:00
|
|
|
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];
|
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
if (run->icache_count > 0)
|
2006-03-27 11:43:17 +00:00
|
|
|
{
|
2006-04-21 17:24:31 +00:00
|
|
|
val = run->icache[--run->icache_count];
|
2006-03-27 11:43:17 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-03-28 16:33:09 +00:00
|
|
|
val = (xp_awk_val_int_t*)
|
2006-04-24 14:38:46 +00:00
|
|
|
xp_malloc (xp_sizeof(xp_awk_val_int_t));
|
2006-03-27 11:43:17 +00:00
|
|
|
if (val == XP_NULL) return XP_NULL;
|
|
|
|
}
|
2006-03-04 15:54:37 +00:00
|
|
|
|
2006-03-05 17:07:33 +00:00
|
|
|
val->type = XP_AWK_VAL_INT;
|
2006-03-26 16:36:30 +00:00
|
|
|
val->ref = 0;
|
2006-03-04 15:54:37 +00:00
|
|
|
val->val = v;
|
2006-03-05 17:07:33 +00:00
|
|
|
|
2006-07-06 15:54:41 +00:00
|
|
|
/*xp_printf (XP_T("makeintval => %p\n"), val);*/
|
2006-03-05 17:07:33 +00:00
|
|
|
return (xp_awk_val_t*)val;
|
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
xp_awk_val_t* xp_awk_makerealval (xp_awk_run_t* run, xp_real_t v)
|
2006-03-05 17:07:33 +00:00
|
|
|
{
|
|
|
|
xp_awk_val_real_t* val;
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
if (run->rcache_count > 0)
|
2006-03-28 16:33:09 +00:00
|
|
|
{
|
2006-04-21 17:24:31 +00:00
|
|
|
val = run->rcache[--run->rcache_count];
|
2006-03-28 16:33:09 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
val = (xp_awk_val_real_t*)
|
|
|
|
xp_malloc (xp_sizeof(xp_awk_val_real_t));
|
|
|
|
if (val == XP_NULL) return XP_NULL;
|
|
|
|
}
|
2006-03-05 17:07:33 +00:00
|
|
|
|
|
|
|
val->type = XP_AWK_VAL_REAL;
|
2006-03-26 16:36:30 +00:00
|
|
|
val->ref = 0;
|
2006-03-05 17:07:33 +00:00
|
|
|
val->val = v;
|
|
|
|
|
2006-07-06 15:54:41 +00:00
|
|
|
/*xp_printf (XP_T("makerealval => %p\n"), val);*/
|
2006-03-05 17:07:33 +00:00
|
|
|
return (xp_awk_val_t*)val;
|
|
|
|
}
|
|
|
|
|
2006-06-26 15:09:28 +00:00
|
|
|
xp_awk_val_t* xp_awk_makestrval0 (const xp_char_t* str)
|
|
|
|
{
|
|
|
|
return xp_awk_makestrval (str, xp_strlen(str));
|
|
|
|
}
|
|
|
|
|
2006-03-05 17:07:33 +00:00
|
|
|
xp_awk_val_t* xp_awk_makestrval (const xp_char_t* str, xp_size_t len)
|
|
|
|
{
|
|
|
|
xp_awk_val_str_t* val;
|
|
|
|
|
2006-04-24 14:38:46 +00:00
|
|
|
val = (xp_awk_val_str_t*) xp_malloc (xp_sizeof(xp_awk_val_str_t));
|
2006-03-05 17:07:33 +00:00
|
|
|
if (val == XP_NULL) return XP_NULL;
|
|
|
|
|
|
|
|
val->type = XP_AWK_VAL_STR;
|
2006-03-26 16:36:30 +00:00
|
|
|
val->ref = 0;
|
2006-03-05 17:07:33 +00:00
|
|
|
val->len = len;
|
|
|
|
val->buf = xp_strxdup (str, len);
|
2006-04-17 16:12:02 +00:00
|
|
|
if (val->buf == XP_NULL)
|
|
|
|
{
|
2006-03-05 17:07:33 +00:00
|
|
|
xp_free (val);
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-06 15:54:41 +00:00
|
|
|
/*xp_printf (XP_T("makestrval => %p\n"), val);*/
|
2006-03-04 15:54:37 +00:00
|
|
|
return (xp_awk_val_t*)val;
|
|
|
|
}
|
2006-03-05 17:07:33 +00:00
|
|
|
|
2006-04-06 16:25:37 +00:00
|
|
|
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;
|
|
|
|
|
2006-04-24 14:38:46 +00:00
|
|
|
val = (xp_awk_val_str_t*) xp_malloc (xp_sizeof(xp_awk_val_str_t));
|
2006-04-06 16:25:37 +00:00
|
|
|
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);
|
2006-04-17 16:12:02 +00:00
|
|
|
if (val->buf == XP_NULL)
|
|
|
|
{
|
|
|
|
xp_free (val);
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-06 15:54:41 +00:00
|
|
|
/*xp_printf (XP_T("makestrval2 => %p\n"), val);*/
|
2006-04-17 16:12:02 +00:00
|
|
|
return (xp_awk_val_t*)val;
|
|
|
|
}
|
|
|
|
|
2006-07-26 15:00:01 +00:00
|
|
|
xp_awk_val_t* xp_awk_makerexval (
|
|
|
|
const xp_char_t* buf, xp_size_t len, void* code)
|
2006-04-24 11:26:00 +00:00
|
|
|
{
|
|
|
|
xp_awk_val_rex_t* val;
|
|
|
|
|
2006-04-24 14:38:46 +00:00
|
|
|
val = (xp_awk_val_rex_t*) xp_malloc (xp_sizeof(xp_awk_val_rex_t));
|
2006-04-24 11:26:00 +00:00
|
|
|
if (val == XP_NULL) return XP_NULL;
|
|
|
|
|
2006-07-25 16:41:40 +00:00
|
|
|
val->type = XP_AWK_VAL_REX;
|
2006-04-24 11:26:00 +00:00
|
|
|
val->ref = 0;
|
|
|
|
val->len = len;
|
2006-07-26 15:00:01 +00:00
|
|
|
val->buf = xp_strxdup (buf, len);
|
2006-04-24 11:26:00 +00:00
|
|
|
if (val->buf == XP_NULL)
|
|
|
|
{
|
|
|
|
xp_free (val);
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-26 16:43:35 +00:00
|
|
|
val->code = xp_malloc (XP_AWK_REX_LEN(code));
|
2006-07-26 15:00:01 +00:00
|
|
|
if (val->code == XP_NULL)
|
|
|
|
{
|
|
|
|
xp_free (val->buf);
|
|
|
|
xp_free (val);
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-26 16:43:35 +00:00
|
|
|
xp_memcpy (val->code, code, XP_AWK_REX_LEN(code));
|
2006-04-24 11:26:00 +00:00
|
|
|
return (xp_awk_val_t*)val;
|
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
static void __free_map_val (void* run, void* v)
|
2006-04-17 16:12:02 +00:00
|
|
|
{
|
2006-04-18 10:28:03 +00:00
|
|
|
/*
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("refdown in map free..."));
|
2006-04-18 10:28:03 +00:00
|
|
|
xp_awk_printval (v);
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("\n"));
|
2006-04-18 10:28:03 +00:00
|
|
|
*/
|
2006-04-21 17:24:31 +00:00
|
|
|
xp_awk_refdownval (run, v);
|
2006-04-17 16:12:02 +00:00
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
xp_awk_val_t* xp_awk_makemapval (xp_awk_run_t* run)
|
2006-04-17 16:12:02 +00:00
|
|
|
{
|
|
|
|
xp_awk_val_map_t* val;
|
|
|
|
|
2006-04-18 10:28:03 +00:00
|
|
|
val = (xp_awk_val_map_t*) xp_malloc (xp_sizeof(xp_awk_val_map_t));
|
2006-04-17 16:12:02 +00:00
|
|
|
if (val == XP_NULL) return XP_NULL;
|
|
|
|
|
|
|
|
val->type = XP_AWK_VAL_MAP;
|
|
|
|
val->ref = 0;
|
2006-04-21 17:24:31 +00:00
|
|
|
val->map = xp_awk_map_open (XP_NULL, run, 256, __free_map_val);
|
2006-04-17 16:12:02 +00:00
|
|
|
if (val->map == XP_NULL)
|
|
|
|
{
|
2006-04-06 16:25:37 +00:00
|
|
|
xp_free (val);
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (xp_awk_val_t*)val;
|
|
|
|
}
|
|
|
|
|
2006-04-24 11:26:00 +00:00
|
|
|
|
2006-03-26 16:36:30 +00:00
|
|
|
xp_bool_t xp_awk_isbuiltinval (xp_awk_val_t* val)
|
|
|
|
{
|
2006-07-10 04:51:38 +00:00
|
|
|
return val == XP_NULL ||
|
2006-07-12 17:04:02 +00:00
|
|
|
val == xp_awk_val_nil || val == xp_awk_val_zls ||
|
|
|
|
val == xp_awk_val_zero || val == xp_awk_val_one ||
|
2006-03-26 16:36:30 +00:00
|
|
|
(val >= (xp_awk_val_t*)&__awk_int[0] &&
|
|
|
|
val <= (xp_awk_val_t*)&__awk_int[xp_countof(__awk_int)-1]);
|
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
void xp_awk_freeval (xp_awk_run_t* run, xp_awk_val_t* val, xp_bool_t cache)
|
2006-03-04 15:54:37 +00:00
|
|
|
{
|
2006-03-26 16:36:30 +00:00
|
|
|
if (xp_awk_isbuiltinval(val)) return;
|
2006-03-22 16:05:50 +00:00
|
|
|
|
2006-05-13 16:33:07 +00:00
|
|
|
/*xp_printf (XP_T("freeing [cache=%d] ... "), cache);
|
2006-04-18 10:28:03 +00:00
|
|
|
xp_awk_printval (val);
|
2006-05-13 16:33:07 +00:00
|
|
|
xp_printf (XP_T("\n"));*/
|
2006-03-04 15:54:37 +00:00
|
|
|
switch (val->type)
|
|
|
|
{
|
2006-03-27 11:43:17 +00:00
|
|
|
case XP_AWK_VAL_NIL:
|
2006-07-10 04:51:38 +00:00
|
|
|
xp_free (val);
|
2006-04-17 16:12:02 +00:00
|
|
|
return;
|
2006-03-27 11:43:17 +00:00
|
|
|
|
|
|
|
case XP_AWK_VAL_INT:
|
2006-04-20 16:17:01 +00:00
|
|
|
if (cache == xp_true &&
|
2006-04-21 17:24:31 +00:00
|
|
|
run->icache_count < xp_countof(run->icache))
|
2006-03-27 11:43:17 +00:00
|
|
|
{
|
2006-04-21 17:24:31 +00:00
|
|
|
run->icache[run->icache_count++] =
|
2006-03-27 11:43:17 +00:00
|
|
|
(xp_awk_val_int_t*)val;
|
|
|
|
}
|
2006-04-17 16:12:02 +00:00
|
|
|
else xp_free (val);
|
|
|
|
return;
|
2006-03-27 11:43:17 +00:00
|
|
|
|
|
|
|
case XP_AWK_VAL_REAL:
|
2006-04-20 16:17:01 +00:00
|
|
|
if (cache == xp_true &&
|
2006-04-21 17:24:31 +00:00
|
|
|
run->rcache_count < xp_countof(run->rcache))
|
2006-03-28 16:33:09 +00:00
|
|
|
{
|
2006-04-21 17:24:31 +00:00
|
|
|
run->rcache[run->rcache_count++] =
|
2006-03-28 16:33:09 +00:00
|
|
|
(xp_awk_val_real_t*)val;
|
|
|
|
}
|
2006-04-17 16:12:02 +00:00
|
|
|
else xp_free (val);
|
|
|
|
return;
|
2006-03-27 11:43:17 +00:00
|
|
|
|
2006-03-04 15:54:37 +00:00
|
|
|
case XP_AWK_VAL_STR:
|
|
|
|
xp_free (((xp_awk_val_str_t*)val)->buf);
|
|
|
|
xp_free (val);
|
2006-04-17 16:12:02 +00:00
|
|
|
return;
|
|
|
|
|
2006-04-24 11:26:00 +00:00
|
|
|
case XP_AWK_VAL_REX:
|
|
|
|
xp_free (((xp_awk_val_rex_t*)val)->buf);
|
2006-07-26 15:00:01 +00:00
|
|
|
xp_free (((xp_awk_val_rex_t*)val)->code);
|
2006-04-24 11:26:00 +00:00
|
|
|
xp_free (val);
|
|
|
|
return;
|
|
|
|
|
2006-04-17 16:12:02 +00:00
|
|
|
case XP_AWK_VAL_MAP:
|
|
|
|
xp_awk_map_close (((xp_awk_val_map_t*)val)->map);
|
|
|
|
xp_free (val);
|
|
|
|
return;
|
2006-03-04 15:54:37 +00:00
|
|
|
}
|
2006-03-27 11:43:17 +00:00
|
|
|
|
2006-04-17 16:12:02 +00:00
|
|
|
xp_assert (!"should never happen - invalid value type");
|
2006-03-04 15:54:37 +00:00
|
|
|
}
|
|
|
|
|
2006-03-23 13:26:04 +00:00
|
|
|
void xp_awk_refupval (xp_awk_val_t* val)
|
|
|
|
{
|
2006-03-26 16:36:30 +00:00
|
|
|
if (xp_awk_isbuiltinval(val)) return;
|
2006-03-23 15:36:20 +00:00
|
|
|
/*
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("ref up "));
|
2006-03-23 15:36:20 +00:00
|
|
|
xp_awk_printval (val);
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("\n"));
|
2006-03-23 15:36:20 +00:00
|
|
|
*/
|
2006-03-23 13:26:04 +00:00
|
|
|
val->ref++;
|
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
void xp_awk_refdownval (xp_awk_run_t* run, xp_awk_val_t* val)
|
2006-03-23 13:26:04 +00:00
|
|
|
{
|
2006-03-26 16:36:30 +00:00
|
|
|
if (xp_awk_isbuiltinval(val)) return;
|
2006-03-23 15:36:20 +00:00
|
|
|
|
|
|
|
/*
|
2006-05-06 12:52:36 +00:00
|
|
|
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);
|
2006-03-23 15:36:20 +00:00
|
|
|
xp_awk_printval (val);
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("\n"));
|
2006-03-23 15:36:20 +00:00
|
|
|
*/
|
2006-03-23 13:26:04 +00:00
|
|
|
|
|
|
|
xp_assert (val->ref > 0);
|
|
|
|
val->ref--;
|
2006-03-26 16:36:30 +00:00
|
|
|
if (val->ref <= 0)
|
|
|
|
{
|
2006-07-06 15:54:41 +00:00
|
|
|
/*
|
2006-07-06 13:57:32 +00:00
|
|
|
xp_printf (XP_T("**FREEING ["));
|
2006-03-26 16:36:30 +00:00
|
|
|
xp_awk_printval (val);
|
2006-07-06 13:57:32 +00:00
|
|
|
xp_printf (XP_T("]\n"));
|
2006-07-06 15:54:41 +00:00
|
|
|
*/
|
2006-04-21 17:24:31 +00:00
|
|
|
xp_awk_freeval(run, val, xp_true);
|
2006-03-26 16:36:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-21 17:24:31 +00:00
|
|
|
void xp_awk_refdownval_nofree (xp_awk_run_t* run, xp_awk_val_t* val)
|
2006-03-26 16:36:30 +00:00
|
|
|
{
|
|
|
|
if (xp_awk_isbuiltinval(val)) return;
|
|
|
|
|
|
|
|
xp_assert (val->ref > 0);
|
|
|
|
val->ref--;
|
2006-03-23 13:26:04 +00:00
|
|
|
}
|
|
|
|
|
2006-06-21 11:45:26 +00:00
|
|
|
xp_bool_t xp_awk_valtobool (xp_awk_val_t* val)
|
2006-03-15 15:34:59 +00:00
|
|
|
{
|
|
|
|
if (val == XP_NULL) return xp_false;
|
|
|
|
|
|
|
|
switch (val->type)
|
|
|
|
{
|
|
|
|
case XP_AWK_VAL_NIL:
|
|
|
|
return xp_false;
|
|
|
|
case XP_AWK_VAL_INT:
|
2006-04-07 04:23:11 +00:00
|
|
|
return ((xp_awk_val_int_t*)val)->val != 0;
|
2006-03-15 15:34:59 +00:00
|
|
|
case XP_AWK_VAL_REAL:
|
2006-04-07 04:23:11 +00:00
|
|
|
return ((xp_awk_val_real_t*)val)->val != 0.0;
|
2006-03-15 15:34:59 +00:00
|
|
|
case XP_AWK_VAL_STR:
|
2006-04-07 04:23:11 +00:00
|
|
|
return ((xp_awk_val_str_t*)val)->len > 0;
|
2006-07-25 16:41:40 +00:00
|
|
|
case XP_AWK_VAL_REX: /* TODO: is this correct? */
|
|
|
|
return ((xp_awk_val_rex_t*)val)->len > 0;
|
2006-04-17 16:12:02 +00:00
|
|
|
case XP_AWK_VAL_MAP:
|
|
|
|
return xp_false; /* TODO: is this correct? */
|
2006-03-15 15:34:59 +00:00
|
|
|
}
|
|
|
|
|
2006-04-17 16:12:02 +00:00
|
|
|
xp_assert (!"should never happen - invalid value type");
|
2006-03-22 16:05:50 +00:00
|
|
|
return xp_false;
|
2006-03-15 15:34:59 +00:00
|
|
|
}
|
|
|
|
|
2006-07-13 03:10:35 +00:00
|
|
|
xp_char_t* xp_awk_valtostr (
|
2006-08-03 05:05:48 +00:00
|
|
|
xp_awk_val_t* v, int* errnum,
|
|
|
|
xp_bool_t clear_buf, xp_str_t* buf, xp_size_t* len)
|
2006-06-21 11:45:26 +00:00
|
|
|
{
|
2006-06-28 08:56:59 +00:00
|
|
|
if (v->type == XP_AWK_VAL_NIL)
|
|
|
|
{
|
|
|
|
if (buf == XP_NULL)
|
|
|
|
{
|
|
|
|
xp_char_t* tmp;
|
|
|
|
tmp = xp_strdup (XP_T(""));
|
2006-07-13 03:10:35 +00:00
|
|
|
if (tmp == XP_NULL)
|
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len != XP_NULL) *len = 0;
|
2006-06-28 08:56:59 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-08-03 05:05:48 +00:00
|
|
|
if (clear_buf) xp_str_clear (buf);
|
|
|
|
|
2006-07-13 03:10:35 +00:00
|
|
|
if (len != XP_NULL) *len = XP_STR_LEN(buf);
|
2006-06-28 08:56:59 +00:00
|
|
|
return XP_STR_BUF(buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-06-21 11:45:26 +00:00
|
|
|
if (v->type == XP_AWK_VAL_INT)
|
|
|
|
{
|
|
|
|
xp_char_t* tmp;
|
|
|
|
xp_long_t t;
|
2006-07-13 03:10:35 +00:00
|
|
|
xp_size_t l = 0;
|
2006-06-21 11:45:26 +00:00
|
|
|
|
|
|
|
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');
|
2006-07-13 03:10:35 +00:00
|
|
|
if (len != XP_NULL) *len = 1;
|
2006-06-21 11:45:26 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-08-03 05:05:48 +00:00
|
|
|
if (clear_buf) xp_str_clear (buf);
|
2006-06-21 11:45:26 +00:00
|
|
|
if (xp_str_cat (buf, XP_T("0")) == (xp_size_t)-1)
|
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-13 03:10:35 +00:00
|
|
|
if (len != XP_NULL) *len = XP_STR_LEN(buf);
|
2006-06-21 11:45:26 +00:00
|
|
|
return XP_STR_BUF(buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* non-zero values */
|
2006-07-13 03:10:35 +00:00
|
|
|
if (t < 0) { t = -t; l++; }
|
|
|
|
while (t > 0) { l++; t /= 10; }
|
2006-06-21 11:45:26 +00:00
|
|
|
|
|
|
|
if (buf == XP_NULL)
|
|
|
|
{
|
2006-07-13 03:10:35 +00:00
|
|
|
tmp = xp_malloc ((l + 1) * xp_sizeof(xp_char_t));
|
2006-06-21 11:45:26 +00:00
|
|
|
if (tmp == XP_NULL)
|
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-13 03:10:35 +00:00
|
|
|
tmp[l] = XP_T('\0');
|
2006-08-03 05:05:48 +00:00
|
|
|
if (len != XP_NULL) *len = l;
|
2006-06-21 11:45:26 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-07-13 03:10:35 +00:00
|
|
|
/* clear the buffer */
|
2006-08-03 05:05:48 +00:00
|
|
|
if (clear_buf) xp_str_clear (buf);
|
2006-07-06 13:57:32 +00:00
|
|
|
|
2006-08-03 05:05:48 +00:00
|
|
|
tmp = XP_STR_BUF(buf) + XP_STR_LEN(buf);
|
2006-06-21 11:45:26 +00:00
|
|
|
|
|
|
|
/* extend the buffer */
|
|
|
|
if (xp_str_nccat (
|
2006-07-13 03:10:35 +00:00
|
|
|
buf, XP_T(' '), l) == (xp_size_t)-1)
|
2006-06-21 11:45:26 +00:00
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
t = ((xp_awk_val_int_t*)v)->val;
|
|
|
|
if (t < 0) t = -t;
|
|
|
|
|
|
|
|
while (t > 0)
|
|
|
|
{
|
2006-07-13 03:10:35 +00:00
|
|
|
tmp[--l] = (xp_char_t)(t % 10) + XP_T('0');
|
2006-06-21 11:45:26 +00:00
|
|
|
t /= 10;
|
|
|
|
}
|
|
|
|
|
2006-07-13 03:10:35 +00:00
|
|
|
if (((xp_awk_val_int_t*)v)->val < 0) tmp[--l] = XP_T('-');
|
2006-06-21 11:45:26 +00:00
|
|
|
|
2006-08-03 05:05:48 +00:00
|
|
|
if (buf != XP_NULL)
|
|
|
|
{
|
|
|
|
tmp = XP_STR_BUF(buf);
|
|
|
|
if (len != XP_NULL) *len = XP_STR_LEN(buf);
|
|
|
|
}
|
|
|
|
|
2006-06-21 11:45:26 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2006-07-17 04:17:40 +00:00
|
|
|
if (v->type == XP_AWK_VAL_REAL)
|
|
|
|
{
|
2006-08-03 05:05:48 +00:00
|
|
|
/* TODO: change the code */
|
2006-07-17 06:21:39 +00:00
|
|
|
xp_char_t tbuf[256], * tmp;
|
|
|
|
|
|
|
|
#if (XP_SIZEOF_LONG_DOUBLE != 0)
|
|
|
|
xp_sprintf (
|
|
|
|
tbuf, xp_countof(tbuf), XP_T("%Lf"),
|
|
|
|
(long double)((xp_awk_val_real_t*)v)->val);
|
|
|
|
#elif (XP_SIZEOF_DOUBLE != 0)
|
|
|
|
xp_sprintf (
|
|
|
|
tbuf, xp_countof(tbuf), XP_T("%f"),
|
|
|
|
(double)((xp_awk_val_real_t*)v)->val);
|
|
|
|
#else
|
2006-07-17 14:27:09 +00:00
|
|
|
#error unsupported floating-point data type
|
2006-07-17 06:21:39 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
if (buf == XP_NULL)
|
|
|
|
{
|
|
|
|
tmp = xp_strdup (tbuf);
|
|
|
|
if (tmp == XP_NULL)
|
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len != XP_NULL) *len = xp_strlen(tmp);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-08-03 05:05:48 +00:00
|
|
|
if (clear_buf) xp_str_clear (buf);
|
|
|
|
|
2006-07-17 06:21:39 +00:00
|
|
|
if (xp_str_cat (buf, tbuf) == (xp_size_t)-1)
|
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = XP_STR_BUF(buf);
|
|
|
|
if (len != XP_NULL) *len = XP_STR_LEN(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
2006-07-17 04:17:40 +00:00
|
|
|
|
2006-06-21 11:45:26 +00:00
|
|
|
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);
|
2006-08-03 05:05:48 +00:00
|
|
|
if (tmp == XP_NULL)
|
|
|
|
{
|
|
|
|
*errnum = XP_AWK_ENOMEM;
|
|
|
|
return XP_NULL;
|
|
|
|
}
|
2006-06-21 11:45:26 +00:00
|
|
|
|
2006-08-03 05:05:48 +00:00
|
|
|
if (len != XP_NULL) *len = ((xp_awk_val_str_t*)v)->len;
|
2006-06-21 11:45:26 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2006-08-03 05:05:48 +00:00
|
|
|
if (clear_buf) xp_str_clear (buf);
|
2006-06-21 11:45:26 +00:00
|
|
|
|
|
|
|
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;
|
2006-08-03 05:05:48 +00:00
|
|
|
return XP_NULL;
|
2006-06-21 11:45:26 +00:00
|
|
|
}
|
2006-08-03 05:05:48 +00:00
|
|
|
|
|
|
|
tmp = XP_STR_BUF(buf);
|
|
|
|
if (len != XP_NULL) *len = XP_STR_LEN(buf);
|
2006-06-21 11:45:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: process more value types */
|
|
|
|
|
2006-07-17 04:17:40 +00:00
|
|
|
xp_printf (XP_T("*** ERROR: WRONG VALUE TYPE [%d] in xp_awk_valtostr v=> %p***\n"), v->type, v);
|
2006-06-26 15:09:28 +00:00
|
|
|
*errnum = XP_AWK_EVALTYPE;
|
2006-06-21 11:45:26 +00:00
|
|
|
return XP_NULL;
|
|
|
|
}
|
|
|
|
|
2006-07-17 04:17:40 +00:00
|
|
|
int xp_awk_valtonum (xp_awk_val_t* v, xp_long_t* l, xp_real_t* r)
|
|
|
|
{
|
|
|
|
if (v->type == XP_AWK_VAL_NIL)
|
|
|
|
{
|
|
|
|
*l = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (v->type == XP_AWK_VAL_INT)
|
|
|
|
{
|
|
|
|
*l = ((xp_awk_val_int_t*)v)->val;
|
|
|
|
return 0; /* long */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (v->type == XP_AWK_VAL_REAL)
|
|
|
|
{
|
|
|
|
*r = ((xp_awk_val_real_t*)v)->val;
|
|
|
|
return 1; /* real */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (v->type == XP_AWK_VAL_STR)
|
|
|
|
{
|
|
|
|
const xp_char_t* endptr;
|
|
|
|
|
|
|
|
/* don't care about val->len */
|
|
|
|
*l = xp_awk_strtolong (((xp_awk_val_str_t*)v)->buf, 0, &endptr);
|
|
|
|
|
|
|
|
if (*endptr == XP_T('.') ||
|
|
|
|
*endptr == XP_T('E') ||
|
|
|
|
*endptr == XP_T('e'))
|
|
|
|
{
|
|
|
|
*r = xp_awk_strtoreal (((xp_awk_val_str_t*)v)->buf);
|
|
|
|
return 1; /* real */
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0; /* long */
|
|
|
|
}
|
|
|
|
|
2006-07-17 06:21:39 +00:00
|
|
|
xp_printf (XP_T("*** ERROR: WRONG VALUE TYPE [%d] in xp_awk_valtonum v=> %p***\n"), v->type, v);
|
2006-07-17 04:17:40 +00:00
|
|
|
return -1; /* error */
|
|
|
|
}
|
|
|
|
|
2006-04-30 17:12:51 +00:00
|
|
|
static int __print_pair (xp_awk_pair_t* pair, void* arg)
|
2006-04-20 05:44:29 +00:00
|
|
|
{
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T(" %s=>"), pair->key);
|
2006-04-20 05:44:29 +00:00
|
|
|
xp_awk_printval (pair->val);
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T(" "));
|
2006-04-20 05:44:29 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-03-04 15:54:37 +00:00
|
|
|
void xp_awk_printval (xp_awk_val_t* val)
|
|
|
|
{
|
2006-04-14 10:56:42 +00:00
|
|
|
/* TODO: better value printing...................... */
|
2006-03-04 15:54:37 +00:00
|
|
|
switch (val->type)
|
|
|
|
{
|
2006-03-07 15:55:14 +00:00
|
|
|
case XP_AWK_VAL_NIL:
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("nil"));
|
2006-03-07 15:55:14 +00:00
|
|
|
break;
|
|
|
|
|
2006-03-04 15:54:37 +00:00
|
|
|
case XP_AWK_VAL_INT:
|
2006-03-15 15:34:59 +00:00
|
|
|
#if defined(__LCC__)
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("%lld"),
|
2006-03-15 15:34:59 +00:00
|
|
|
(long long)((xp_awk_val_int_t*)val)->val);
|
|
|
|
#elif defined(__BORLANDC__) || defined(_MSC_VER)
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("%I64d"),
|
2006-03-14 16:40:00 +00:00
|
|
|
(__int64)((xp_awk_nde_int_t*)val)->val);
|
2006-04-14 10:56:42 +00:00
|
|
|
#elif defined(vax) || defined(__vax) || defined(_SCO_DS)
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("%ld"),
|
2006-03-15 15:34:59 +00:00
|
|
|
(long)((xp_awk_val_int_t*)val)->val);
|
2006-03-14 16:40:00 +00:00
|
|
|
#else
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("%lld"),
|
2006-03-04 15:54:37 +00:00
|
|
|
(long long)((xp_awk_val_int_t*)val)->val);
|
2006-03-14 16:40:00 +00:00
|
|
|
#endif
|
2006-03-04 15:54:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case XP_AWK_VAL_REAL:
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("%Lf"),
|
2006-03-04 15:54:37 +00:00
|
|
|
(long double)((xp_awk_val_real_t*)val)->val);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case XP_AWK_VAL_STR:
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("%s"), ((xp_awk_val_str_t*)val)->buf);
|
2006-03-04 15:54:37 +00:00
|
|
|
break;
|
|
|
|
|
2006-04-24 11:26:00 +00:00
|
|
|
case XP_AWK_VAL_REX:
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("REX[%s]"), ((xp_awk_val_rex_t*)val)->buf);
|
2006-04-24 11:26:00 +00:00
|
|
|
break;
|
|
|
|
|
2006-04-17 16:12:02 +00:00
|
|
|
case XP_AWK_VAL_MAP:
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("MAP["));
|
2006-04-30 17:12:51 +00:00
|
|
|
xp_awk_map_walk (((xp_awk_val_map_t*)val)->map, __print_pair, XP_NULL);
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("]"));
|
2006-04-17 16:12:02 +00:00
|
|
|
break;
|
|
|
|
|
2006-03-04 15:54:37 +00:00
|
|
|
default:
|
2006-04-16 04:31:38 +00:00
|
|
|
xp_assert (!"should never happen - invalid value type");
|
2006-05-06 12:52:36 +00:00
|
|
|
xp_printf (XP_T("**** INTERNAL ERROR - INVALID VALUE TYPE ****\n"));
|
2006-03-04 15:54:37 +00:00
|
|
|
}
|
|
|
|
}
|