%{ #include #include #include "picrin.h" #include "picrin/pair.h" #define YYERROR_VERBOSE 1 struct parser_control { pic_state *pic; void *yyscanner; pic_value value; bool incomp; int yynerrs; }; void yyerror(struct parser_control *, const char *); /* just for supressing warnings. a little bit evil */ int yylex(); int yylex_(); void yylex_init(); void yyset_in(); void yy_scan_string(); void yylex_destroy(); %} %pure_parser %parse-param {struct parser_control *p} %lex-param {struct parser_control *p} %union { int i; double f; char *cstr; char c; pic_value datum; } %token tLPAREN tRPAREN tDOT tVPAREN %token tQUOTE tQUASIQUOTE tUNQUOTE tUNQUOTE_SPLICING %token tINT tBOOLEAN %token tFLOAT %token tSYMBOL tSTRING %token tCHAR %type program_data %type datum simple_datum compound_datum abbrev %type list list_data vector vector_data %% program : program_data { p->value = $1; } | incomplete_program_data { p->incomp = true; p->value = pic_undef_value(); } ; program_data : datum { $$ = pic_cons(p->pic, $1, pic_nil_value()); } | datum program_data { $$ = pic_cons(p->pic, $1, $2); } ; incomplete_program_data : incomplete_datum | datum incomplete_program_data ; datum : simple_datum | compound_datum ; simple_datum : tSYMBOL { $$ = pic_symbol_value(pic_intern_cstr(p->pic, $1)); free($1); } | tSTRING { $$ = pic_str_new_cstr(p->pic, $1); free($1); } | tINT { $$ = pic_int_value($1); } | tFLOAT { $$ = pic_float_value($1); } | tBOOLEAN { $$ = pic_bool_value($1); } | tCHAR { $$ = pic_char_value($1); } ; compound_datum : list | vector | abbrev ; list : tLPAREN list_data tRPAREN { $$ = $2; } ; list_data : /* none */ { $$ = pic_nil_value(); } | datum tDOT datum { $$ = pic_cons(p->pic, $1, $3); } | datum list_data { $$ = pic_cons(p->pic, $1, $2); } ; vector : tVPAREN vector_data tRPAREN { $$ = pic_obj_value(pic_vec_new_from_list(p->pic, $2)); } ; vector_data : /* none */ { $$ = pic_nil_value(); } | datum vector_data { $$ = pic_cons(p->pic, $1, $2); } ; abbrev : tQUOTE datum { $$ = pic_list(p->pic, 2, pic_symbol_value(p->pic->sQUOTE), $2); } | tQUASIQUOTE datum { $$ = pic_list(p->pic, 2, pic_symbol_value(p->pic->sQUASIQUOTE), $2); } | tUNQUOTE datum { $$ = pic_list(p->pic, 2, pic_symbol_value(p->pic->sUNQUOTE), $2); } | tUNQUOTE_SPLICING datum { $$ = pic_list(p->pic, 2, pic_symbol_value(p->pic->sUNQUOTE_SPLICING), $2); } ; incomplete_datum : tLPAREN incomplete_data | tVPAREN incomplete_data | incomplete_abbrev ; incomplete_tail : /* none */ | incomplete_datum ; incomplete_data : incomplete_tail | datum tDOT incomplete_tail | datum incomplete_data ; incomplete_abbrev : tQUOTE incomplete_tail | tQUASIQUOTE incomplete_tail | tUNQUOTE incomplete_tail | tUNQUOTE_SPLICING incomplete_tail ; %% void yyerror(struct parser_control *p, const char *msg) { puts(msg); p->yynerrs++; } int yylex(YYSTYPE *yylvalp, struct parser_control *p) { return yylex_(yylvalp, p->yyscanner); } int pic_parse_file(pic_state *pic, FILE *file, pic_value *v) { struct parser_control p; p.pic = pic; p.incomp = false; p.yynerrs = 0; yylex_init(&p.yyscanner); yyset_in(file, p.yyscanner); yyparse(&p); yylex_destroy(p.yyscanner); if (p.yynerrs > 0) { p.value = pic_undef_value(); return -1; } if (p.incomp) { return 0; } *v = p.value; return pic_length(pic, p.value); } int pic_parse_cstr(pic_state *pic, const char *str, pic_value *v) { struct parser_control p; p.pic = pic; p.incomp = false; p.yynerrs = 0; yylex_init(&p.yyscanner); yy_scan_string(str, p.yyscanner); yyparse(&p); yylex_destroy(p.yyscanner); if (p.yynerrs > 0) { return -1; } if (p.incomp) { return 0; } *v = p.value; return pic_length(pic, p.value); }