791 lines
18 KiB
C
791 lines
18 KiB
C
/**
|
|
* See Copyright Notice in picrin.h
|
|
*/
|
|
|
|
#include "picrin.h"
|
|
#include "picrin/extra.h"
|
|
#include "picrin/private/object.h"
|
|
|
|
#undef EOF
|
|
#define EOF (-1)
|
|
|
|
KHASH_DECLARE(read, int, pic_value)
|
|
KHASH_DEFINE(read, int, pic_value, kh_int_hash_func, kh_int_hash_equal)
|
|
|
|
struct reader_control {
|
|
int typecase;
|
|
khash_t(read) labels;
|
|
};
|
|
|
|
#define CASE_DEFAULT 0
|
|
#define CASE_FOLD 1
|
|
|
|
typedef pic_value (*pic_reader_t)(pic_state *, xFILE *file, int c, struct reader_control *);
|
|
|
|
static pic_reader_t reader_table[256];
|
|
static pic_reader_t reader_dispatch[256];
|
|
|
|
static pic_value read_value(pic_state *pic, xFILE *file, int c, struct reader_control *p);
|
|
static pic_value read_nullable(pic_state *pic, xFILE *file, int c, struct reader_control *p);
|
|
|
|
PIC_NORETURN static void
|
|
read_error(pic_state *pic, const char *msg, pic_value irritants)
|
|
{
|
|
pic_raise(pic, pic_make_error(pic, "read", msg, irritants));
|
|
}
|
|
|
|
static int
|
|
skip(pic_state *pic, xFILE *file, int c)
|
|
{
|
|
while (isspace(c)) {
|
|
c = xfgetc(pic, file);
|
|
}
|
|
return c;
|
|
}
|
|
|
|
static int
|
|
next(pic_state *pic, xFILE *file)
|
|
{
|
|
return xfgetc(pic, file);
|
|
}
|
|
|
|
static int
|
|
peek(pic_state *pic, xFILE *file)
|
|
{
|
|
int c;
|
|
|
|
xungetc(pic, (c = xfgetc(pic, file)), file);
|
|
|
|
return c;
|
|
}
|
|
|
|
static bool
|
|
expect(pic_state *pic, xFILE *file, const char *str)
|
|
{
|
|
int c;
|
|
|
|
while ((c = (int)*str++) != 0) {
|
|
if (c != peek(pic, file))
|
|
return false;
|
|
next(pic, file);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool
|
|
isdelim(int c)
|
|
{
|
|
return c == EOF || strchr("();,|\" \t\n\r", c) != NULL; /* ignores "#", "'" */
|
|
}
|
|
|
|
static int
|
|
case_fold(int c, struct reader_control *p)
|
|
{
|
|
if (p->typecase == CASE_FOLD) {
|
|
c = tolower(c);
|
|
}
|
|
return c;
|
|
}
|
|
|
|
static pic_value
|
|
read_comment(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
do {
|
|
c = next(pic, file);
|
|
} while (! (c == EOF || c == '\n'));
|
|
|
|
return pic_invalid_value(pic);
|
|
}
|
|
|
|
static pic_value
|
|
read_block_comment(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
int x, y;
|
|
int i = 1;
|
|
|
|
y = next(pic, file);
|
|
|
|
while (y != EOF && i > 0) {
|
|
x = y;
|
|
y = next(pic, file);
|
|
if (x == '|' && y == '#') {
|
|
i--;
|
|
}
|
|
if (x == '#' && y == '|') {
|
|
i++;
|
|
}
|
|
}
|
|
|
|
return pic_invalid_value(pic);
|
|
}
|
|
|
|
static pic_value
|
|
read_datum_comment(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
read_value(pic, file, next(pic, file), p);
|
|
|
|
return pic_invalid_value(pic);
|
|
}
|
|
|
|
static pic_value
|
|
read_directive(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
switch (peek(pic, file)) {
|
|
case 'n':
|
|
if (expect(pic, file, "no-fold-case")) {
|
|
p->typecase = CASE_DEFAULT;
|
|
return pic_invalid_value(pic);
|
|
}
|
|
break;
|
|
case 'f':
|
|
if (expect(pic, file, "fold-case")) {
|
|
p->typecase = CASE_FOLD;
|
|
return pic_invalid_value(pic);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return read_comment(pic, file, c, p);
|
|
}
|
|
|
|
static pic_value
|
|
read_quote(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
return pic_list(pic, 2, pic_intern_lit(pic, "quote"), read_value(pic, file, next(pic, file), p));
|
|
}
|
|
|
|
static pic_value
|
|
read_quasiquote(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
return pic_list(pic, 2, pic_intern_lit(pic, "quasiquote"), read_value(pic, file, next(pic, file), p));
|
|
}
|
|
|
|
static pic_value
|
|
read_unquote(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
pic_value tag;
|
|
|
|
if (peek(pic, file) == '@') {
|
|
tag = pic_intern_lit(pic, "unquote-splicing");
|
|
next(pic, file);
|
|
} else {
|
|
tag = pic_intern_lit(pic, "unquote");
|
|
}
|
|
return pic_list(pic, 2, tag, read_value(pic, file, next(pic, file), p));
|
|
}
|
|
|
|
static pic_value
|
|
read_syntax_quote(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
return pic_list(pic, 2, pic_intern_lit(pic, "syntax-quote"), read_value(pic, file, next(pic, file), p));
|
|
}
|
|
|
|
static pic_value
|
|
read_syntax_quasiquote(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
return pic_list(pic, 2, pic_intern_lit(pic, "syntax-quasiquote"), read_value(pic, file, next(pic, file), p));
|
|
}
|
|
|
|
static pic_value
|
|
read_syntax_unquote(pic_state *pic, xFILE *file, int PIC_UNUSED(c), struct reader_control *p)
|
|
{
|
|
pic_value tag;
|
|
|
|
if (peek(pic, file) == '@') {
|
|
tag = pic_intern_lit(pic, "syntax-unquote-splicing");
|
|
next(pic, file);
|
|
} else {
|
|
tag = pic_intern_lit(pic, "syntax-unquote");
|
|
}
|
|
return pic_list(pic, 2, tag, read_value(pic, file, next(pic, file), p));
|
|
}
|
|
|
|
static pic_value
|
|
read_atom(pic_state *pic, xFILE *file, int c, struct reader_control *p) {
|
|
int len;
|
|
char *buf;
|
|
pic_value str;
|
|
|
|
len = 1;
|
|
buf = pic_malloc(pic, len + 1);
|
|
buf[0] = case_fold(c, p);
|
|
buf[1] = 0;
|
|
|
|
while (! isdelim(peek(pic, file))) {
|
|
c = next(pic, file);
|
|
len += 1;
|
|
buf = pic_realloc(pic, buf, len + 1);
|
|
buf[len - 1] = case_fold(c, p);
|
|
buf[len] = 0;
|
|
}
|
|
|
|
str = pic_str_value(pic, buf, len);
|
|
pic_free(pic, buf);
|
|
|
|
return str;
|
|
}
|
|
|
|
static pic_value
|
|
read_symbol(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
return pic_intern(pic, read_atom(pic, file, c, p));
|
|
}
|
|
|
|
static pic_value
|
|
read_number(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
pic_value str = read_atom(pic, file, c, p), num;
|
|
|
|
num = pic_funcall(pic, "picrin.base", "string->number", 1, str);
|
|
if (! pic_false_p(pic, num)) {
|
|
return num;
|
|
}
|
|
return pic_intern(pic, str);
|
|
}
|
|
|
|
static unsigned
|
|
read_uinteger(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
unsigned u = 0;
|
|
|
|
if (! isdigit(c)) {
|
|
read_error(pic, "expected one or more digits", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
u = c - '0';
|
|
while (isdigit(c = peek(pic, file))) {
|
|
u = u * 10 + next(pic, file) - '0';
|
|
}
|
|
|
|
return u;
|
|
}
|
|
|
|
static pic_value
|
|
read_true(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
if ((c = peek(pic, file)) == 'r') {
|
|
if (! expect(pic, file, "rue")) {
|
|
read_error(pic, "unexpected character while reading #true", pic_nil_value(pic));
|
|
}
|
|
} else if (! isdelim(c)) {
|
|
read_error(pic, "non-delimiter character given after #t", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
return pic_true_value(pic);
|
|
}
|
|
|
|
static pic_value
|
|
read_false(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
if ((c = peek(pic, file)) == 'a') {
|
|
if (! expect(pic, file, "alse")) {
|
|
read_error(pic, "unexpected character while reading #false", pic_nil_value(pic));
|
|
}
|
|
} else if (! isdelim(c)) {
|
|
read_error(pic, "non-delimiter character given after #f", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
return pic_false_value(pic);
|
|
}
|
|
|
|
static pic_value
|
|
read_char(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
c = next(pic, file);
|
|
|
|
if (! isdelim(peek(pic, file))) {
|
|
switch (c) {
|
|
default: read_error(pic, "unexpected character after char literal", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
case 'a': c = '\a'; if (! expect(pic, file, "larm")) goto fail; break;
|
|
case 'b': c = '\b'; if (! expect(pic, file, "ackspace")) goto fail; break;
|
|
case 'd': c = 0x7F; if (! expect(pic, file, "elete")) goto fail; break;
|
|
case 'e': c = 0x1B; if (! expect(pic, file, "scape")) goto fail; break;
|
|
case 'n':
|
|
if ((c = peek(pic, file)) == 'e') {
|
|
c = '\n';
|
|
if (! expect(pic, file, "ewline"))
|
|
goto fail;
|
|
} else {
|
|
c = '\0';
|
|
if (! expect(pic, file, "ull"))
|
|
goto fail;
|
|
}
|
|
break;
|
|
case 'r': c = '\r'; if (! expect(pic, file, "eturn")) goto fail; break;
|
|
case 's': c = ' '; if (! expect(pic, file, "pace")) goto fail; break;
|
|
case 't': c = '\t'; if (! expect(pic, file, "ab")) goto fail; break;
|
|
}
|
|
}
|
|
|
|
return pic_char_value(pic, (char)c);
|
|
|
|
fail:
|
|
read_error(pic, "unexpected character while reading character literal", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
static pic_value
|
|
read_string(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
char *buf;
|
|
int size, cnt;
|
|
pic_value str;
|
|
|
|
size = 256;
|
|
buf = pic_malloc(pic, size);
|
|
cnt = 0;
|
|
|
|
/* TODO: intraline whitespaces */
|
|
|
|
while ((c = next(pic, file)) != '"') {
|
|
if (c == '\\') {
|
|
switch (c = next(pic, file)) {
|
|
case 'a': c = '\a'; break;
|
|
case 'b': c = '\b'; break;
|
|
case 't': c = '\t'; break;
|
|
case 'n': c = '\n'; break;
|
|
case 'r': c = '\r'; break;
|
|
}
|
|
}
|
|
buf[cnt++] = (char)c;
|
|
if (cnt >= size) {
|
|
buf = pic_realloc(pic, buf, size *= 2);
|
|
}
|
|
}
|
|
buf[cnt] = '\0';
|
|
|
|
str = pic_str_value(pic, buf, cnt);
|
|
pic_free(pic, buf);
|
|
return str;
|
|
}
|
|
|
|
static pic_value
|
|
read_pipe(pic_state *pic, xFILE *file, int c, struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
char *buf;
|
|
int size, cnt;
|
|
pic_value sym;
|
|
/* Currently supports only ascii chars */
|
|
char HEX_BUF[3];
|
|
size_t i = 0;
|
|
|
|
size = 256;
|
|
buf = pic_malloc(pic, size);
|
|
cnt = 0;
|
|
while ((c = next(pic, file)) != '|') {
|
|
if (c == '\\') {
|
|
switch ((c = next(pic, file))) {
|
|
case 'a': c = '\a'; break;
|
|
case 'b': c = '\b'; break;
|
|
case 't': c = '\t'; break;
|
|
case 'n': c = '\n'; break;
|
|
case 'r': c = '\r'; break;
|
|
case 'x':
|
|
i = 0;
|
|
while ((HEX_BUF[i++] = (char)next(pic, file)) != ';') {
|
|
if (i >= sizeof HEX_BUF)
|
|
read_error(pic, "expected ';'", pic_list(pic, 1, pic_char_value(pic, HEX_BUF[sizeof(HEX_BUF) - 1])));
|
|
}
|
|
c = (char)strtol(HEX_BUF, NULL, 16);
|
|
break;
|
|
}
|
|
}
|
|
buf[cnt++] = (char)c;
|
|
if (cnt >= size) {
|
|
buf = pic_realloc(pic, buf, size *= 2);
|
|
}
|
|
}
|
|
buf[cnt] = '\0';
|
|
|
|
sym = pic_intern_cstr(pic, buf);
|
|
pic_free(pic, buf);
|
|
|
|
return sym;
|
|
}
|
|
|
|
static pic_value
|
|
read_blob(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
int nbits, n;
|
|
int len;
|
|
unsigned char *dat;
|
|
pic_value blob;
|
|
|
|
nbits = 0;
|
|
|
|
while (isdigit(c = next(pic, file))) {
|
|
nbits = 10 * nbits + c - '0';
|
|
}
|
|
|
|
if (nbits != 8) {
|
|
read_error(pic, "unsupported bytevector bit width", pic_list(pic, 1, pic_int_value(pic, nbits)));
|
|
}
|
|
|
|
if (c != '(') {
|
|
read_error(pic, "expected '(' character", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
len = 0;
|
|
dat = NULL;
|
|
c = next(pic, file);
|
|
while ((c = skip(pic, file, c)) != ')') {
|
|
n = read_uinteger(pic, file, c, p);
|
|
if (n < 0 || (1 << nbits) <= n) {
|
|
read_error(pic, "invalid element in bytevector literal", pic_list(pic, 1, pic_int_value(pic, n)));
|
|
}
|
|
len += 1;
|
|
dat = pic_realloc(pic, dat, len);
|
|
dat[len - 1] = (unsigned char)n;
|
|
c = next(pic, file);
|
|
}
|
|
|
|
blob = pic_blob_value(pic, dat, len);
|
|
|
|
pic_free(pic, dat);
|
|
return blob;
|
|
}
|
|
|
|
static pic_value
|
|
read_undef_or_blob(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
if ((c = peek(pic, file)) == 'n') {
|
|
if (! expect(pic, file, "ndefined")) {
|
|
read_error(pic, "unexpected character while reading #undefined", pic_nil_value(pic));
|
|
}
|
|
return pic_undef_value(pic);
|
|
}
|
|
if (! isdigit(c)) {
|
|
read_error(pic, "expect #undefined or #u8(...), but illegal character given", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
return read_blob(pic, file, 'u', p);
|
|
}
|
|
|
|
static pic_value
|
|
read_pair(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
static const int tCLOSE = ')';
|
|
pic_value car, cdr;
|
|
|
|
retry:
|
|
|
|
c = skip(pic, file, ' ');
|
|
|
|
if (c == tCLOSE) {
|
|
return pic_nil_value(pic);
|
|
}
|
|
if (c == '.' && isdelim(peek(pic, file))) {
|
|
cdr = read_value(pic, file, next(pic, file), p);
|
|
|
|
closing:
|
|
if ((c = skip(pic, file, ' ')) != tCLOSE) {
|
|
if (pic_invalid_p(pic, read_nullable(pic, file, c, p))) {
|
|
goto closing;
|
|
}
|
|
read_error(pic, "unmatched parenthesis", pic_nil_value(pic));
|
|
}
|
|
return cdr;
|
|
}
|
|
else {
|
|
car = read_nullable(pic, file, c, p);
|
|
|
|
if (pic_invalid_p(pic, car)) {
|
|
goto retry;
|
|
}
|
|
|
|
cdr = read_pair(pic, file, '(', p);
|
|
return pic_cons(pic, car, cdr);
|
|
}
|
|
}
|
|
|
|
static pic_value
|
|
read_vector(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
pic_value list, it, elem, vec;
|
|
int i = 0;
|
|
|
|
list = read_value(pic, file, c, p);
|
|
|
|
vec = pic_make_vec(pic, pic_length(pic, list), NULL);
|
|
|
|
pic_for_each (elem, list, it) {
|
|
pic_vec_set(pic, vec, i++, elem);
|
|
}
|
|
|
|
return vec;
|
|
}
|
|
|
|
static pic_value
|
|
read_label_set(pic_state *pic, xFILE *file, int i, struct reader_control *p)
|
|
{
|
|
khash_t(read) *h = &p->labels;
|
|
pic_value val;
|
|
int c, ret, it;
|
|
|
|
it = kh_put(read, h, i, &ret);
|
|
|
|
switch ((c = skip(pic, file, ' '))) {
|
|
case '(':
|
|
{
|
|
pic_value tmp;
|
|
|
|
kh_val(h, it) = val = pic_cons(pic, pic_undef_value(pic), pic_undef_value(pic));
|
|
|
|
tmp = read_value(pic, file, c, p);
|
|
pic_pair_ptr(pic, val)->car = pic_car(pic, tmp);
|
|
pic_pair_ptr(pic, val)->cdr = pic_cdr(pic, tmp);
|
|
|
|
return val;
|
|
}
|
|
case '#':
|
|
{
|
|
bool vect;
|
|
|
|
if (peek(pic, file) == '(') {
|
|
vect = true;
|
|
} else {
|
|
vect = false;
|
|
}
|
|
|
|
if (vect) {
|
|
pic_value tmp;
|
|
|
|
kh_val(h, it) = val = pic_make_vec(pic, 0, NULL);
|
|
|
|
tmp = read_value(pic, file, c, p);
|
|
PIC_SWAP(pic_value *, pic_vec_ptr(pic, tmp)->data, pic_vec_ptr(pic, val)->data);
|
|
PIC_SWAP(int, pic_vec_ptr(pic, tmp)->len, pic_vec_ptr(pic, val)->len);
|
|
|
|
return val;
|
|
}
|
|
|
|
PIC_FALLTHROUGH;
|
|
}
|
|
default:
|
|
{
|
|
kh_val(h, it) = val = read_value(pic, file, c, p);
|
|
|
|
return val;
|
|
}
|
|
}
|
|
}
|
|
|
|
static pic_value
|
|
read_label_ref(pic_state *pic, xFILE *PIC_UNUSED(file), int i, struct reader_control *p)
|
|
{
|
|
khash_t(read) *h = &p->labels;
|
|
int it;
|
|
|
|
it = kh_get(read, h, i);
|
|
if (it == kh_end(h)) {
|
|
read_error(pic, "label of given index not defined", pic_list(pic, 1, pic_int_value(pic, i)));
|
|
}
|
|
return kh_val(h, it);
|
|
}
|
|
|
|
static pic_value
|
|
read_label(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
int i;
|
|
|
|
i = 0;
|
|
do {
|
|
i = i * 10 + c - '0';
|
|
} while (isdigit(c = next(pic, file)));
|
|
|
|
if (c == '=') {
|
|
return read_label_set(pic, file, i, p);
|
|
}
|
|
if (c == '#') {
|
|
return read_label_ref(pic, file, i, p);
|
|
}
|
|
read_error(pic, "broken label expression", pic_nil_value(pic));
|
|
}
|
|
|
|
static pic_value
|
|
read_unmatch(pic_state *pic, xFILE *PIC_UNUSED(file), int PIC_UNUSED(c), struct reader_control *PIC_UNUSED(p))
|
|
{
|
|
read_error(pic, "unmatched parenthesis", pic_nil_value(pic));
|
|
}
|
|
|
|
static pic_value
|
|
read_dispatch(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
c = next(pic, file);
|
|
|
|
if (c == EOF) {
|
|
read_error(pic, "unexpected EOF", pic_nil_value(pic));
|
|
}
|
|
|
|
if (reader_dispatch[c] == NULL) {
|
|
read_error(pic, "invalid character at the seeker head", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
return reader_dispatch[c](pic, file, c, p);
|
|
}
|
|
|
|
static pic_value
|
|
read_nullable(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
c = skip(pic, file, c);
|
|
|
|
if (c == EOF) {
|
|
read_error(pic, "unexpected EOF", pic_nil_value(pic));
|
|
}
|
|
|
|
if (reader_table[c] == NULL) {
|
|
read_error(pic, "invalid character at the seeker head", pic_list(pic, 1, pic_char_value(pic, c)));
|
|
}
|
|
|
|
return reader_table[c](pic, file, c, p);
|
|
}
|
|
|
|
static pic_value
|
|
read_value(pic_state *pic, xFILE *file, int c, struct reader_control *p)
|
|
{
|
|
pic_value val;
|
|
|
|
retry:
|
|
val = read_nullable(pic, file, c, p);
|
|
|
|
if (pic_invalid_p(pic, val)) {
|
|
c = next(pic, file);
|
|
goto retry;
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
static void
|
|
reader_table_init(void)
|
|
{
|
|
int c;
|
|
|
|
for (c = 0; c < 256; ++c) {
|
|
reader_table[c] = NULL;
|
|
}
|
|
for (c = 0; c < 256; ++c) {
|
|
reader_dispatch[c] = NULL;
|
|
}
|
|
|
|
/* default reader */
|
|
for (c = 1; c < 256; ++c) {
|
|
reader_table[c] = read_symbol;
|
|
}
|
|
|
|
reader_table[')'] = read_unmatch;
|
|
reader_table[';'] = read_comment;
|
|
reader_table['\''] = read_quote;
|
|
reader_table['`'] = read_quasiquote;
|
|
reader_table[','] = read_unquote;
|
|
reader_table['"'] = read_string;
|
|
reader_table['|'] = read_pipe;
|
|
reader_table['('] = read_pair;
|
|
reader_table['#'] = read_dispatch;
|
|
reader_table['+'] = read_number;
|
|
reader_table['-'] = read_number;
|
|
for (c = '0'; c <= '9'; ++c) {
|
|
reader_table[c] = read_number;
|
|
}
|
|
|
|
reader_dispatch['!'] = read_directive;
|
|
reader_dispatch['|'] = read_block_comment;
|
|
reader_dispatch[';'] = read_datum_comment;
|
|
reader_dispatch['t'] = read_true;
|
|
reader_dispatch['f'] = read_false;
|
|
reader_dispatch['\''] = read_syntax_quote;
|
|
reader_dispatch['`'] = read_syntax_quasiquote;
|
|
reader_dispatch[','] = read_syntax_unquote;
|
|
reader_dispatch['\\'] = read_char;
|
|
reader_dispatch['('] = read_vector;
|
|
reader_dispatch['u'] = read_undef_or_blob;
|
|
|
|
/* read labels */
|
|
for (c = '0'; c <= '9'; ++c) {
|
|
reader_dispatch[c] = read_label;
|
|
}
|
|
}
|
|
|
|
static void
|
|
reader_init(pic_state *PIC_UNUSED(pic), struct reader_control *p)
|
|
{
|
|
p->typecase = CASE_DEFAULT;
|
|
kh_init(read, &p->labels);
|
|
}
|
|
|
|
static void
|
|
reader_destroy(pic_state *pic, struct reader_control *p)
|
|
{
|
|
kh_destroy(read, &p->labels);
|
|
}
|
|
|
|
pic_value
|
|
pic_read(pic_state *pic, pic_value port)
|
|
{
|
|
struct reader_control p;
|
|
size_t ai = pic_enter(pic);
|
|
pic_value val;
|
|
xFILE *file = pic_fileno(pic, port);
|
|
int c;
|
|
pic_value e;
|
|
|
|
reader_init(pic, &p);
|
|
|
|
pic_try {
|
|
while ((c = skip(pic, file, next(pic, file))) != EOF) {
|
|
val = read_nullable(pic, file, c, &p);
|
|
|
|
if (! pic_invalid_p(pic, val)) {
|
|
break;
|
|
}
|
|
pic_leave(pic, ai);
|
|
}
|
|
if (c == EOF) {
|
|
val = pic_eof_object(pic);
|
|
}
|
|
}
|
|
pic_catch(e) {
|
|
reader_destroy(pic, &p);
|
|
pic_raise(pic, e);
|
|
}
|
|
|
|
pic_leave(pic, ai);
|
|
return pic_protect(pic, val);
|
|
}
|
|
|
|
pic_value
|
|
pic_read_cstr(pic_state *pic, const char *str)
|
|
{
|
|
pic_value port = pic_open_port(pic, xfopen_buf(pic, str, strlen(str), "r"));
|
|
pic_value form, e;
|
|
|
|
pic_try {
|
|
form = pic_read(pic, port);
|
|
}
|
|
pic_catch(e) {
|
|
pic_close_port(pic, port);
|
|
pic_raise(pic, e);
|
|
}
|
|
|
|
pic_close_port(pic, port);
|
|
|
|
return form;
|
|
}
|
|
|
|
static pic_value
|
|
pic_read_read(pic_state *pic)
|
|
{
|
|
pic_value port = pic_stdin(pic);
|
|
|
|
pic_get_args(pic, "|p", &port);
|
|
|
|
return pic_read(pic, port);
|
|
}
|
|
|
|
void
|
|
pic_init_read(pic_state *pic)
|
|
{
|
|
reader_table_init();
|
|
|
|
pic_defun(pic, "read", pic_read_read);
|
|
}
|