picrin/lib/value.h

273 lines
5.4 KiB
C

/**
* See Copyright Notice in picrin.h
*/
#ifndef PICRIN_VALUE_H
#define PICRIN_VALUE_H
#if defined(__cplusplus)
extern "C" {
#endif
enum {
PIC_TYPE_INVALID = 1,
PIC_TYPE_FLOAT = 2,
PIC_TYPE_INT = 3,
PIC_TYPE_CHAR = 4,
PIC_TYPE_EOF = 5,
PIC_TYPE_UNDEF = 6,
PIC_TYPE_TRUE = 8,
PIC_TYPE_NIL = 7,
PIC_TYPE_FALSE = 9,
PIC_IVAL_END = 10,
/* -------------------- */
PIC_TYPE_SYMBOL = 16,
PIC_TYPE_STRING = 17,
PIC_TYPE_BLOB = 18,
PIC_TYPE_DATA = 19,
PIC_TYPE_PAIR = 20,
PIC_TYPE_VECTOR = 21,
PIC_TYPE_DICT = 22,
PIC_TYPE_RECORD = 23,
PIC_TYPE_ATTR = 24,
PIC_TYPE_PORT = 25,
PIC_TYPE_IREP = 27,
PIC_TYPE_FRAME = 28,
PIC_TYPE_PROC_FUNC = 29,
PIC_TYPE_PROC_IREP = 30,
PIC_TYPE_MAX = 63
};
#if !PIC_NAN_BOXING
PIC_STATIC_INLINE void
make_value(struct value *v, int type)
{
static const struct value zero = { 0 };
*v = zero;
v->type = type;
}
PIC_STATIC_INLINE void
make_int_value(struct value *v, int i)
{
make_value(v, PIC_TYPE_INT);
v->u.i = i;
}
PIC_STATIC_INLINE void
make_float_value(struct value *v, double f)
{
make_value(v, PIC_TYPE_FLOAT);
v->u.f = f;
}
PIC_STATIC_INLINE struct value
make_char_value(struct value *v, char c)
{
make_value(v, PIC_TYPE_CHAR);
v->u.c = c;
}
PIC_STATIC_INLINE void
make_obj_value(struct value *v, void *p, int type)
{
make_value(v, type);
v->u.p = p;
}
PIC_STATIC_INLINE int
value_type(struct value *v)
{
return (int)(v->type);
}
PIC_STATIC_INLINE int
value_int(struct value *v)
{
return v->u.i;
}
PIC_STATIC_INLINE double
value_float(struct value *v)
{
return v->u.f;
}
PIC_STATIC_INLINE char
value_char(struct value *v)
{
return v->u.c;
}
PIC_STATIC_INLINE void *
value_ptr(struct value *v)
{
return v->u.p;
}
PIC_STATIC_INLINE bool
value_eq_p(struct value *x, struct value *y)
{
return memcmp(x, y, sizeof(struct value)) == 0;
}
PIC_STATIC_INLINE bool
value_eqv_p(struct value *x, struct value *y)
{
return memcmp(x, y, sizeof(struct value)) == 0;
}
#else /* NAN_BOXING */
/**
* value representation by nan-boxing:
* float : FFFFFFFFFFFFFFFF FFFFFFFFFFFFFFFF FFFFFFFFFFFFFFFF FFFFFFFFFFFFFFFF
* ptr : 111111111111TTTT TTPPPPPPPPPPPPPP PPPPPPPPPPPPPPPP PPPPPPPPPPPPPPPP
* int : 111111111111TTTT TT00000000000000 IIIIIIIIIIIIIIII IIIIIIIIIIIIIIII
* char : 111111111111TTTT TT00000000000000 CCCCCCCCCCCCCCCC CCCCCCCCCCCCCCCC
*/
PIC_STATIC_INLINE void
make_value(struct value *v, int type)
{
v->v = 0xfff0000000000000ul | ((uint64_t)(type) << 46);
}
PIC_STATIC_INLINE void
make_int_value(struct value *v, int i)
{
make_value(v, PIC_TYPE_INT);
v->v |= (unsigned)i;
}
PIC_STATIC_INLINE void
make_float_value(struct value *v, double f)
{
if (f != f) {
v->v = 0x7ff8000000000000ul;
} else {
union { double f; uint64_t i; } u;
u.f = f;
v->v = u.i;
}
}
PIC_STATIC_INLINE void
make_char_value(struct value *v, char c)
{
make_value(v, PIC_TYPE_CHAR);
v->v |= (unsigned char)c;
}
PIC_STATIC_INLINE void
make_obj_value(struct value *v, void *ptr, int type)
{
make_value(v, type);
v->v |= 0x3ffffffffffful & ((uint64_t)ptr >> 2);
}
PIC_STATIC_INLINE int
value_type(struct value *v)
{
return 0xfff0000000000000ul >= v->v ? PIC_TYPE_FLOAT : ((v->v >> 46) & 0x3f);
}
PIC_STATIC_INLINE int
value_int(struct value *v)
{
union { int i; unsigned u; } u;
u.u = v->v & 0xfffffffful;
return u.i;
}
PIC_STATIC_INLINE double
value_float(struct value *v)
{
union { double f; uint64_t i; } u;
u.i = v->v;
return u.f;
}
PIC_STATIC_INLINE char
value_char(struct value *v)
{
return v->v & 0xfffffffful;
}
PIC_STATIC_INLINE void *
value_ptr(struct value *v)
{
return (void *)((0x3ffffffffffful & v->v) << 2);
}
PIC_STATIC_INLINE bool
value_eq_p(struct value *x, struct value *y)
{
return x->v == y->v;
}
PIC_STATIC_INLINE bool
value_eqv_p(struct value *x, struct value *y)
{
return x->v == y->v;
}
#endif /* NAN_BOXING end */
#define DEFPRED(name, type) \
PIC_STATIC_INLINE bool \
value_##name##_p(struct value *v) { \
return value_type(v) == type; \
}
DEFPRED(invalid, PIC_TYPE_INVALID)
DEFPRED(float, PIC_TYPE_FLOAT)
DEFPRED(int, PIC_TYPE_INT)
DEFPRED(char, PIC_TYPE_CHAR)
DEFPRED(eof, PIC_TYPE_EOF)
DEFPRED(undef, PIC_TYPE_UNDEF)
DEFPRED(true, PIC_TYPE_TRUE)
DEFPRED(nil, PIC_TYPE_NIL)
DEFPRED(false, PIC_TYPE_FALSE)
DEFPRED(str, PIC_TYPE_STRING)
DEFPRED(vec, PIC_TYPE_VECTOR)
DEFPRED(blob, PIC_TYPE_BLOB)
DEFPRED(dict, PIC_TYPE_DICT)
DEFPRED(attr, PIC_TYPE_ATTR)
DEFPRED(rec, PIC_TYPE_RECORD)
DEFPRED(sym, PIC_TYPE_SYMBOL)
DEFPRED(pair, PIC_TYPE_PAIR)
DEFPRED(proc_func, PIC_TYPE_PROC_FUNC)
DEFPRED(proc_irep, PIC_TYPE_PROC_IREP)
DEFPRED(irep, PIC_TYPE_IREP)
DEFPRED(data, PIC_TYPE_DATA)
DEFPRED(port, PIC_TYPE_PORT)
#undef DEFPRED
PIC_STATIC_INLINE bool
value_obj_p(struct value *v)
{
return value_type(v) > PIC_IVAL_END;
}
void *pic_ptr(pic_state *, pic_value);
int pic_type(pic_state *, pic_value);
pic_value pic_invalid_value(pic_state *);
pic_value pic_obj_value(pic_state *, void *, int);
bool pic_invalid_p(pic_state *, pic_value);
bool pic_attr_p(pic_state *, pic_value);
bool pic_rec_p(pic_state *, pic_value);
bool pic_irep_p(pic_state *, pic_value);
bool pic_proc_func_p(pic_state *, pic_value);
bool pic_proc_irep_p(pic_state *, pic_value);
bool pic_obj_p(pic_state *, pic_value);
#if defined(__cplusplus)
}
#endif
#endif