picrin/lib/proc.c

598 lines
15 KiB
C
Raw Normal View History

2014-08-25 00:38:09 -04:00
/**
* See Copyright Notice in picrin.h
*/
#include "picrin.h"
2017-03-28 10:09:40 -04:00
#include "object.h"
#include "state.h"
2017-04-14 10:40:07 -04:00
struct frame *
pic_make_frame_unsafe(pic_state *pic, int n)
{
struct frame *fp;
int i;
fp = (struct frame *)pic_obj_alloc_unsafe(pic, PIC_TYPE_FRAME);
fp->regs = n ? pic_malloc(pic, sizeof(pic_value) * n) : NULL;
fp->regc = n;
fp->up = NULL;
for (i = 0; i < n; ++i) {
fp->regs[i] = pic_invalid_value(pic);
}
return fp;
}
2016-02-06 10:09:40 -05:00
2017-03-28 11:03:23 -04:00
pic_value
pic_lambda(pic_state *pic, pic_func_t f, int n, ...)
{
pic_value proc;
va_list ap;
va_start(ap, n);
proc = pic_vlambda(pic, f, n, ap);
va_end(ap);
return proc;
}
pic_value
pic_vlambda(pic_state *pic, pic_func_t f, int n, va_list ap)
{
2017-04-14 10:40:07 -04:00
struct proc *proc;
2017-03-28 11:03:23 -04:00
int i;
2017-04-14 10:40:07 -04:00
assert(n >= 0);
proc = (struct proc *)pic_obj_alloc(pic, PIC_TYPE_PROC_FUNC);
proc->u.func = f;
proc->env = NULL;
if (n != 0) {
proc->env = pic_make_frame_unsafe(pic, n);
}
2017-03-28 11:03:23 -04:00
for (i = 0; i < n; ++i) {
2017-04-14 10:40:07 -04:00
proc->env->regs[i] = va_arg(ap, pic_value);
2017-03-28 11:03:23 -04:00
}
2017-04-14 10:40:07 -04:00
return obj_value(pic, proc);
2017-04-09 02:05:59 -04:00
}
pic_value
2017-04-14 10:40:07 -04:00
pic_make_proc_func(pic_state *pic, pic_func_t func)
2017-04-09 02:05:59 -04:00
{
struct proc *proc;
2017-04-12 00:18:06 -04:00
proc = (struct proc *)pic_obj_alloc(pic, PIC_TYPE_PROC_FUNC);
proc->u.func = func;
2017-04-14 10:40:07 -04:00
proc->env = NULL;
2017-04-09 02:05:59 -04:00
return obj_value(pic, proc);
}
pic_value
2017-04-14 10:40:07 -04:00
pic_make_proc_irep_unsafe(pic_state *pic, struct irep *irep, struct frame *fp)
2017-04-09 02:05:59 -04:00
{
struct proc *proc;
2017-04-14 10:40:07 -04:00
proc = (struct proc *)pic_obj_alloc_unsafe(pic, PIC_TYPE_PROC_IREP);
proc->u.irep = irep;
2017-04-14 10:40:07 -04:00
proc->env = fp;
2017-04-09 02:05:59 -04:00
return obj_value(pic, proc);
2017-03-28 11:03:23 -04:00
}
2016-02-06 10:09:40 -05:00
2016-02-22 14:03:42 -05:00
PIC_NORETURN static void
arg_error(pic_state *pic, int actual, bool varg, int expected)
{
const char *msg;
2017-04-14 15:12:26 -04:00
msg = pic_str(pic, pic_strf_value(pic, "wrong number of arguments (%d for %s%d)", actual, (varg ? "at least " : ""), expected), NULL);
2016-02-22 14:03:42 -05:00
pic_error(pic, msg, 0);
}
2017-04-14 10:40:07 -04:00
#define GET_ARGC(pic) (pic->cxt->pc[1])
#define GET_PROC(pic) (pic->cxt->fp->regs[0])
#define GET_CONT(pic) (pic->cxt->fp->regs[1])
#define GET_ARG(pic,n) (pic->cxt->fp->regs[(n)+2])
2016-02-06 10:09:40 -05:00
/**
* char type desc.
* ---- ---- ----
* o pic_value * object
* i int * int
* I int *, bool * int with exactness
* f double * float
* F double *, bool * float with exactness
* c char * char
* z char ** c string
2016-02-20 07:16:10 -05:00
* b unsigned char *, int * bytevector
* u void **, const pic_data_type * user data type
2016-02-20 01:31:14 -05:00
* m pic_value * symbol
2016-02-20 05:47:46 -05:00
* v pic_value * vector
* s pic_value * string
* l pic_value * lambda
* p pic_value * port
* d pic_value * dictionary
* r pic_value * record
2016-02-06 10:09:40 -05:00
*
2016-02-20 07:16:10 -05:00
* + aliasing operator
2016-02-06 10:09:40 -05:00
* | optional operator
2016-02-20 03:27:13 -05:00
* * int *, pic_value ** variable length operator
* ---- ---- ----
2016-02-06 10:09:40 -05:00
*/
int
pic_get_args(pic_state *pic, const char *format, ...)
{
char c;
2016-02-20 07:16:10 -05:00
const char *p = format;
int paramc = 0, optc = 0;
2017-03-28 11:03:23 -04:00
bool proc = 0, rest = 0, opt = 0;
2017-04-14 10:40:07 -04:00
int i, argc = GET_ARGC(pic) - 1; /* one for continuation */
2016-02-06 10:09:40 -05:00
va_list ap;
/* parse format */
2016-02-20 07:16:10 -05:00
if ((c = *p) != '\0') {
if (c == '&') {
2016-02-20 07:16:10 -05:00
proc = 1;
p++;
2016-02-06 10:09:40 -05:00
}
2016-02-20 07:16:10 -05:00
while ((c = *p++) != '\0') {
if (c == '+')
continue;
if (c == '|') {
2016-02-20 07:16:10 -05:00
opt = 1; break;
} else if (c == '*') {
2016-02-20 07:16:10 -05:00
rest = 1; break;
}
2016-02-20 07:16:10 -05:00
paramc++;
2016-02-06 10:09:40 -05:00
}
2016-02-20 07:16:10 -05:00
if (opt) {
while ((c = *p++) != '\0') {
if (c == '+')
continue;
if (c == '*') {
rest = 1; break;
}
optc++;
}
2016-02-06 10:09:40 -05:00
}
2016-02-20 07:16:10 -05:00
if (rest) c = *p++;
assert(opt <= optc); /* at least 1 char after '|'? */
assert(c == '\0'); /* no extra chars? */
2016-02-06 10:09:40 -05:00
}
if (argc < paramc || (paramc + optc < argc && ! rest)) {
2016-02-22 14:03:42 -05:00
arg_error(pic, argc, rest, paramc);
2016-02-06 10:09:40 -05:00
}
va_start(ap, format);
/* dispatch */
if (proc) {
2016-02-19 10:03:16 -05:00
pic_value *proc;
2016-02-19 10:03:16 -05:00
proc = va_arg(ap, pic_value *);
2017-03-28 11:03:23 -04:00
*proc = GET_PROC(pic);
2016-02-20 07:16:10 -05:00
format++; /* skip '&' */
}
2017-03-28 18:11:27 -04:00
for (i = 0; i < argc && i < paramc + optc; ++i) {
2016-02-06 10:09:40 -05:00
c = *format++;
if (c == '|') {
c = *format++;
}
2016-02-06 10:09:40 -05:00
switch (c) {
case 'o': {
pic_value *p;
p = va_arg(ap, pic_value*);
2017-03-28 11:03:23 -04:00
*p = GET_ARG(pic, i);
2016-02-06 10:09:40 -05:00
break;
}
2016-02-20 07:16:10 -05:00
case 'u': {
void **data;
const pic_data_type *type;
pic_value v;
data = va_arg(ap, void **);
type = va_arg(ap, const pic_data_type *);
2017-03-28 11:03:23 -04:00
v = GET_ARG(pic, i);
2016-02-20 07:16:10 -05:00
if (pic_data_p(pic, v, type)) {
*data = pic_data(pic, v);
}
else {
2016-02-22 14:03:42 -05:00
const char *msg;
2017-03-28 10:31:15 -04:00
msg = pic_str(pic, pic_strf_value(pic, "pic_get_args: data type \"%s\" required", type->type_name), NULL);
2016-02-22 14:03:42 -05:00
pic_error(pic, msg, 1, v);
2016-02-20 07:16:10 -05:00
}
break;
}
case 'b': {
unsigned char **buf;
int *len;
pic_value v;
buf = va_arg(ap, unsigned char **);
len = va_arg(ap, int *);
2017-03-28 11:03:23 -04:00
v = GET_ARG(pic, i);
2016-02-20 07:16:10 -05:00
if (pic_blob_p(pic, v)) {
unsigned char *tmp = pic_blob(pic, v, len);
if (buf) *buf = tmp;
}
else {
2016-02-22 14:03:42 -05:00
pic_error(pic, "pic_get_args: bytevector required", 1, v);
2016-02-20 07:16:10 -05:00
}
break;
}
2016-02-06 10:09:40 -05:00
#define NUM_CASE(c1, c2, ctype) \
case c1: case c2: { \
ctype *n; \
bool *e, dummy; \
pic_value v; \
\
n = va_arg(ap, ctype *); \
e = (c == c2 ? va_arg(ap, bool *) : &dummy); \
\
2017-03-28 11:03:23 -04:00
v = GET_ARG(pic, i); \
switch (pic_type(pic, v)) { \
2016-02-22 14:03:42 -05:00
case PIC_TYPE_FLOAT: \
*n = pic_float(pic, v); \
2016-02-06 10:09:40 -05:00
*e = false; \
break; \
2016-02-22 14:03:42 -05:00
case PIC_TYPE_INT: \
*n = pic_int(pic, v); \
2016-02-06 10:09:40 -05:00
*e = true; \
break; \
default: \
2016-02-22 14:03:42 -05:00
pic_error(pic, "pic_get_args: float or int required", 1, v); \
2016-02-06 10:09:40 -05:00
} \
break; \
}
NUM_CASE('i', 'I', int)
NUM_CASE('f', 'F', double)
#define VAL_CASE(c, type, ctype, conv) \
case c: { \
ctype *ptr; \
pic_value v; \
\
ptr = va_arg(ap, ctype *); \
2017-03-28 11:03:23 -04:00
v = GET_ARG(pic, i); \
if (pic_## type ##_p(pic, v)) { \
2016-02-06 10:09:40 -05:00
*ptr = conv; \
} \
else { \
2016-02-22 14:03:42 -05:00
pic_error(pic, "pic_get_args: " #type " required", 1, v); \
2016-02-06 10:09:40 -05:00
} \
break; \
}
VAL_CASE('c', char, char, pic_char(pic, v))
2017-03-28 10:31:15 -04:00
VAL_CASE('z', str, const char *, pic_str(pic, v, NULL))
2016-02-06 10:09:40 -05:00
2016-02-19 05:08:45 -05:00
#define OBJ_CASE(c, type) VAL_CASE(c, type, pic_value, v)
2016-02-20 01:31:14 -05:00
OBJ_CASE('m', sym)
2016-02-19 13:26:52 -05:00
OBJ_CASE('s', str)
2016-02-19 10:03:16 -05:00
OBJ_CASE('l', proc)
2016-02-19 07:56:45 -05:00
OBJ_CASE('v', vec)
2016-02-19 05:08:45 -05:00
OBJ_CASE('d', dict)
2017-03-28 18:11:27 -04:00
#define pic_port_p(pic,v) pic_port_p(pic,v,NULL)
2016-02-20 02:51:24 -05:00
OBJ_CASE('p', port)
2017-03-28 18:11:27 -04:00
#undef pic_port_p
2016-02-20 03:27:13 -05:00
OBJ_CASE('r', rec)
2016-02-19 05:08:45 -05:00
2016-02-06 10:09:40 -05:00
default:
2016-02-22 14:03:42 -05:00
pic_error(pic, "pic_get_args: invalid argument specifier given", 1, pic_char_value(pic, c));
2016-02-06 10:09:40 -05:00
}
2016-02-20 07:16:10 -05:00
2016-03-03 05:55:27 -05:00
if (*format == '+') {
2016-02-20 07:16:10 -05:00
pic_value *p;
2016-03-03 05:55:27 -05:00
format++;
2017-03-28 11:03:23 -04:00
p = va_arg(ap, pic_value *);
*p = GET_ARG(pic, i);
2016-02-20 07:16:10 -05:00
}
2016-02-06 10:09:40 -05:00
}
if (rest) {
int *n;
pic_value **argv;
2016-02-06 10:09:40 -05:00
n = va_arg(ap, int *);
argv = va_arg(ap, pic_value **);
2017-03-28 11:03:23 -04:00
*n = argc - i;
*argv = &GET_ARG(pic, i);
2016-02-06 10:09:40 -05:00
}
2016-02-06 10:09:40 -05:00
va_end(ap);
2016-02-06 10:09:40 -05:00
return argc;
}
2017-03-28 11:03:23 -04:00
pic_value
pic_closure_ref(pic_state *pic, int n)
2016-02-06 10:09:40 -05:00
{
2017-04-14 10:40:07 -04:00
struct frame *fp = pic->cxt->fp->up;
assert(n >= 0);
2017-04-14 10:40:07 -04:00
if (fp == NULL || fp->regc <= n) {
2017-03-28 11:03:23 -04:00
pic_error(pic, "pic_closure_ref: index out of range", 1, pic_int_value(pic, n));
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
return fp->regs[n];
2016-02-06 10:09:40 -05:00
}
2017-03-28 11:03:23 -04:00
void
pic_closure_set(pic_state *pic, int n, pic_value v)
2016-02-06 10:09:40 -05:00
{
2017-04-14 10:40:07 -04:00
struct frame *fp = pic->cxt->fp->up;
assert(n >= 0);
2017-04-14 10:40:07 -04:00
if (fp == NULL || fp->regc <= n) {
2017-03-28 11:03:23 -04:00
pic_error(pic, "pic_closure_ref: index out of range", 1, pic_int_value(pic, n));
2016-02-20 03:27:13 -05:00
}
2017-04-14 10:40:07 -04:00
fp->regs[n] = v;
2016-02-06 10:09:40 -05:00
}
2017-04-09 02:05:59 -04:00
pic_value
pic_call(pic_state *pic, pic_value proc, int n, ...)
{
pic_value r;
va_list ap;
va_start(ap, n);
r = pic_vcall(pic, proc, n, ap);
va_end(ap);
return r;
}
pic_value
pic_vcall(pic_state *pic, pic_value proc, int n, va_list ap)
{
2017-04-19 01:00:02 -04:00
size_t ai = pic_enter(pic);
2017-04-09 02:05:59 -04:00
pic_value *args = pic_alloca(pic, sizeof(pic_value) * n);
2017-04-19 01:00:02 -04:00
pic_value r;
2017-04-09 02:05:59 -04:00
int i;
for (i = 0; i < n; ++i) {
args[i] = va_arg(ap, pic_value);
}
2017-04-19 01:00:02 -04:00
r = pic_apply(pic, proc, n, args);
pic_leave(pic, ai);
return pic_protect(pic, r);
2017-04-09 02:05:59 -04:00
}
2016-02-06 10:09:40 -05:00
pic_value
2016-02-19 10:03:16 -05:00
pic_apply(pic_state *pic, pic_value proc, int argc, pic_value *argv)
2016-02-06 10:09:40 -05:00
{
2017-04-14 10:40:07 -04:00
struct context cxt;
2017-04-09 02:05:59 -04:00
2017-04-14 10:40:07 -04:00
#define MKCALL(argc) (cxt.tmpcode[0] = OP_CALL, cxt.tmpcode[1] = (argc), cxt.tmpcode)
2017-04-09 02:05:59 -04:00
2017-04-14 10:40:07 -04:00
cxt.pc = MKCALL(argc + 1);
cxt.sp = pic_make_frame_unsafe(pic, argc + 3);
cxt.sp->regs[0] = proc;
cxt.sp->regs[1] = pic->halt;
if (argc != 0) {
int i;
for (i = 0; i < argc; ++i) {
cxt.sp->regs[i + 2] = argv[i];
}
}
cxt.fp = NULL;
cxt.irep = NULL;
cxt.prev = pic->cxt;
pic->cxt = &cxt;
2017-04-20 16:22:28 -04:00
#define SAVE (pic->ai = cxt.ai)
2017-04-15 16:20:55 -04:00
2017-04-20 16:22:28 -04:00
if (PIC_SETJMP(cxt.jmp) == 0) {
cxt.ai = pic->ai;
2017-04-09 02:05:59 -04:00
}
2017-04-14 10:40:07 -04:00
#define A (cxt.pc[1])
#define B (cxt.pc[2])
#define C (cxt.pc[3])
#define Bx ((C << 8) + B)
#define REG(i) (cxt.sp->regs[i])
2017-04-09 02:05:59 -04:00
#if PIC_DIRECT_THREADED_VM
# define VM_LOOP JUMP;
# define CASE(x) L_##x:
2017-04-14 10:40:07 -04:00
# define NEXT(n) (cxt.pc += n); JUMP;
# define JUMP goto *oplabels[*cxt.pc];
2017-04-09 02:05:59 -04:00
# define VM_LOOP_END
#else
2017-04-14 10:40:07 -04:00
# define VM_LOOP for (;;) { switch (*cxt.pc) {
2017-04-09 02:05:59 -04:00
# define CASE(x) case x:
2017-04-14 10:40:07 -04:00
# define NEXT(n) (cxt.pc += n); break
2017-04-09 02:05:59 -04:00
# define JUMP break
# define VM_LOOP_END } }
#endif
2016-02-06 10:09:40 -05:00
#if PIC_DIRECT_THREADED_VM
static const void *oplabels[] = {
2017-04-14 10:40:07 -04:00
[OP_HALT] = &&L_OP_HALT, [OP_CALL] = &&L_OP_CALL, [OP_PROC] = &&L_OP_PROC,
[OP_LOAD] = &&L_OP_LOAD, [OP_LREF] = &&L_OP_LREF, [OP_LSET] = &&L_OP_LSET,
[OP_GREF] = &&L_OP_GREF, [OP_GSET] = &&L_OP_GSET, [OP_COND] = &&L_OP_COND,
[OP_LOADT] = &&L_OP_LOADT, [OP_LOADF] = &&L_OP_LOADF, [OP_LOADN] = &&L_OP_LOADN,
[OP_LOADU] = &&L_OP_LOADU, [OP_LOADI] = &&L_OP_LOADI
2016-02-06 10:09:40 -05:00
};
#endif
VM_LOOP {
2017-04-14 10:40:07 -04:00
CASE(OP_HALT) {
pic_value ret = cxt.fp->regs[1];
pic->cxt = pic->cxt->prev;
pic_protect(pic, ret);
return ret;
2016-02-06 10:09:40 -05:00
}
CASE(OP_CALL) {
2016-02-21 06:32:00 -05:00
struct proc *proc;
2017-04-14 10:40:07 -04:00
if (! pic_proc_p(pic, REG(0))) {
pic_error(pic, "invalid application", 1, REG(0));
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
proc = proc_ptr(pic, REG(0));
2017-03-30 10:29:08 -04:00
if (proc->tt == PIC_TYPE_PROC_FUNC) {
2017-04-14 10:40:07 -04:00
pic_value v;
cxt.sp->up = proc->env; /* push static link */
cxt.fp = cxt.sp;
cxt.sp = NULL;
cxt.irep = NULL;
v = proc->u.func(pic);
2017-04-14 10:40:07 -04:00
if (cxt.sp != NULL) { /* tail call */
SAVE;
JUMP;
} else {
cxt.sp = pic_make_frame_unsafe(pic, 3);
cxt.sp->regs[0] = cxt.fp->regs[1]; /* cont. */
cxt.sp->regs[1] = v;
cxt.pc = MKCALL(1);
SAVE;
JUMP;
}
} else {
struct irep *irep = proc->u.irep;
2016-02-06 10:09:40 -05:00
2017-04-14 10:40:07 -04:00
if (A != irep->argc) {
if (! ((irep->flags & IREP_VARG) != 0 && A >= irep->argc)) {
arg_error(pic, A, (irep->flags & IREP_VARG), irep->argc);
}
}
if (irep->flags & IREP_VARG) {
REG(irep->argc + 1) = pic_make_list(pic, A - irep->argc, &REG(irep->argc + 1));
SAVE; /* TODO: get rid of this */
}
2016-02-06 10:09:40 -05:00
2017-04-14 10:40:07 -04:00
cxt.sp->up = proc->env; /* push static link */
cxt.fp = cxt.sp;
cxt.sp = pic_make_frame_unsafe(pic, irep->frame_size);
cxt.pc = irep->code;
cxt.irep = irep;
JUMP;
2016-02-06 10:09:40 -05:00
}
}
2017-04-14 10:40:07 -04:00
CASE(OP_LREF) {
struct frame *f;
int depth = B;
for (f = cxt.fp; depth--; f = f->up);
REG(A) = f->regs[C];
NEXT(4);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LSET) {
struct frame *f;
int depth = B;
for (f = cxt.fp; depth--; f = f->up);
f->regs[C] = REG(A);
NEXT(4);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_GREF) {
REG(A) = pic_global_ref(pic, cxt.irep->obj[B]);
NEXT(3);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_GSET) {
pic_global_set(pic, cxt.irep->obj[B], REG(A));
NEXT(3);
}
CASE(OP_COND) {
if (pic_false_p(pic, REG(A))) {
NEXT(Bx);
} else {
NEXT(4);
}
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_PROC) {
REG(A) = pic_make_proc_irep_unsafe(pic, cxt.irep->irep[B], cxt.fp);
NEXT(3);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LOAD) {
REG(A) = cxt.irep->obj[B];
NEXT(3);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LOADU) {
REG(A) = pic_undef_value(pic);
NEXT(2);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LOADT) {
REG(A) = pic_true_value(pic);
NEXT(2);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LOADF) {
REG(A) = pic_false_value(pic);
NEXT(2);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LOADN) {
REG(A) = pic_nil_value(pic);
NEXT(2);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
CASE(OP_LOADI) {
REG(A) = pic_int_value(pic, (signed char) B);
NEXT(3);
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
} VM_LOOP_END
2016-02-06 10:09:40 -05:00
}
2016-02-13 23:33:15 -05:00
pic_value
2016-02-19 10:03:16 -05:00
pic_applyk(pic_state *pic, pic_value proc, int argc, pic_value *args)
2016-02-06 10:09:40 -05:00
{
2017-04-14 10:40:07 -04:00
const code_t *pc;
struct frame *sp;
2016-02-06 10:09:40 -05:00
2017-04-14 10:40:07 -04:00
#define MKCALLK(argc) \
(pic->cxt->tmpcode[0] = OP_CALL, pic->cxt->tmpcode[1] = (argc), pic->cxt->tmpcode)
2016-02-06 10:09:40 -05:00
2017-04-14 10:40:07 -04:00
pc = MKCALLK(argc + 1);
sp = pic_make_frame_unsafe(pic, argc + 3);
sp->regs[0] = proc;
sp->regs[1] = GET_CONT(pic);
if (argc != 0) {
int i;
for (i = 0; i < argc; ++i) {
sp->regs[i + 2] = args[i];
}
2016-02-06 10:09:40 -05:00
}
2017-04-14 10:40:07 -04:00
pic->cxt->pc = pc;
pic->cxt->sp = sp;
return pic_invalid_value(pic);
2016-02-06 10:09:40 -05:00
}
2014-08-25 00:38:09 -04:00
static pic_value
pic_proc_proc_p(pic_state *pic)
{
pic_value v;
pic_get_args(pic, "o", &v);
return pic_bool_value(pic, pic_proc_p(pic, v));
2014-08-25 00:38:09 -04:00
}
static pic_value
pic_proc_apply(pic_state *pic)
{
2016-02-19 10:03:16 -05:00
pic_value proc, *args, *arg_list;
2016-02-14 03:14:33 -05:00
int argc, n, i;
2014-08-25 00:38:09 -04:00
pic_get_args(pic, "l*", &proc, &argc, &args);
if (argc == 0) {
2016-02-22 14:03:42 -05:00
pic_error(pic, "apply: wrong number of arguments", 0);
2014-08-25 00:38:09 -04:00
}
2016-02-14 03:14:33 -05:00
n = argc - 1 + pic_length(pic, args[argc - 1]);
2014-08-25 00:38:09 -04:00
2016-02-14 03:14:33 -05:00
arg_list = pic_alloca(pic, sizeof(pic_value) * n);
for (i = 0; i < argc - 1; ++i) {
arg_list[i] = args[i];
}
while (i < n) {
arg_list[i] = pic_list_ref(pic, args[argc - 1], i - argc + 1);
i++;
}
return pic_applyk(pic, proc, n, arg_list);
2014-08-25 00:38:09 -04:00
}
void
pic_init_proc(pic_state *pic)
{
pic_defun(pic, "procedure?", pic_proc_proc_p);
pic_defun(pic, "apply", pic_proc_apply);
}