picrin/src/pair.c

163 lines
2.6 KiB
C
Raw Normal View History

2013-10-22 23:39:48 -04:00
#include <stdarg.h>
2013-10-10 04:06:26 -04:00
#include "picrin.h"
2013-10-19 23:34:57 -04:00
#include "picrin/pair.h"
2013-10-10 04:06:26 -04:00
pic_value
pic_cons(pic_state *pic, pic_value car, pic_value cdr)
{
struct pic_pair *pair;
2013-10-13 03:55:07 -04:00
pair = (struct pic_pair *)pic_obj_alloc(pic, sizeof(struct pic_pair), PIC_TT_PAIR);
2013-10-10 04:06:26 -04:00
pair->car = car;
pair->cdr = cdr;
return pic_obj_value(pair);
}
pic_value
pic_car(pic_state *pic, pic_value obj)
{
struct pic_pair *pair;
2013-10-20 10:30:01 -04:00
if (! pic_pair_p(obj)) {
pic_error(pic, "pair required");
}
2013-10-10 04:54:35 -04:00
pair = (struct pic_pair *)obj.u.data;
2013-10-10 04:06:26 -04:00
return pair->car;
}
pic_value
pic_cdr(pic_state *pic, pic_value obj)
{
struct pic_pair *pair;
2013-10-20 10:30:01 -04:00
if (! pic_pair_p(obj)) {
pic_error(pic, "pair required");
}
2013-10-10 04:54:35 -04:00
pair = (struct pic_pair *)obj.u.data;
2013-10-10 04:06:26 -04:00
return pair->cdr;
}
2013-10-19 23:34:57 -04:00
2013-10-20 23:03:35 -04:00
bool
pic_list_p(pic_state *pic, pic_value obj)
{
while (pic_pair_p(obj))
obj = pic_pair_ptr(obj)->cdr;
return pic_nil_p(obj);
}
2013-10-22 23:39:48 -04:00
pic_value
pic_list(pic_state *pic, size_t c, ...)
{
va_list ap;
pic_value v;
va_start(ap, c);
v = pic_nil_value();
while (c--) {
v = pic_cons(pic, va_arg(ap, pic_value), v);
}
va_end(ap);
return pic_reverse(pic, v);
}
int
pic_length(pic_state *pic, pic_value obj)
{
int c = 0;
while (! pic_nil_p(obj)) {
obj = pic_cdr(pic, obj);
++c;
}
return c;
}
2013-10-21 11:19:43 -04:00
pic_value
pic_reverse(pic_state *pic, pic_value list)
{
pic_value v, acc = pic_nil_value();
for (v = list; ! pic_nil_p(v); v = pic_cdr(pic ,v)) {
acc = pic_cons(pic, pic_car(pic, v), acc);
}
return acc;
}
2013-10-19 23:34:57 -04:00
pic_value
pic_assq(pic_state *pic, pic_value key, pic_value assoc)
{
pic_value cell;
enter:
if (pic_nil_p(assoc))
return assoc;
cell = pic_car(pic, assoc);
if (pic_eq_p(pic, key, pic_car(pic, cell)))
return cell;
assoc = pic_cdr(pic, assoc);
goto enter;
}
2013-10-20 01:04:39 -04:00
pic_value
pic_acons(pic_state *pic, pic_value key, pic_value val, pic_value assoc)
{
return pic_cons(pic, pic_cons(pic, key, val), assoc);
}
2013-10-22 23:01:06 -04:00
2013-10-24 11:37:08 -04:00
static pic_value
pic_pair_pair_p(pic_state *pic)
{
pic_value v;
pic_get_args(pic, "o", &v);
return pic_bool_value(pic_pair_p(v));
}
2013-10-22 23:01:06 -04:00
static pic_value
pic_pair_set_car(pic_state *pic)
{
pic_value v,w;
pic_get_args(pic, "oo", &v, &w);
if (! pic_pair_p(v))
pic_error(pic, "pair expected");
pic_pair_ptr(v)->car = w;
return pic_true_value();
}
static pic_value
pic_pair_set_cdr(pic_state *pic)
{
pic_value v,w;
pic_get_args(pic, "oo", &v, &w);
if (! pic_pair_p(v))
pic_error(pic, "pair expected");
pic_pair_ptr(v)->cdr = w;
return pic_true_value();
}
void
pic_init_pair(pic_state *pic)
{
2013-10-24 11:37:08 -04:00
pic_defun(pic, "pair?", pic_pair_pair_p);
2013-10-22 23:01:06 -04:00
pic_defun(pic, "set-car!", pic_pair_set_car);
pic_defun(pic, "set-cdr!", pic_pair_set_cdr);
}