=================================================================== RCS file: /home/cvs/OpenXM_contrib2/asir2000/parse/lex.c,v retrieving revision 1.1 retrieving revision 1.50 diff -u -p -r1.1 -r1.50 --- OpenXM_contrib2/asir2000/parse/lex.c 1999/12/03 07:39:12 1.1 +++ OpenXM_contrib2/asir2000/parse/lex.c 2015/08/14 13:51:56 1.50 @@ -1,46 +1,146 @@ -/* $OpenXM: OpenXM/src/asir99/parse/lex.c,v 1.1.1.1 1999/11/10 08:12:34 noro Exp $ */ +/* + * Copyright (c) 1994-2000 FUJITSU LABORATORIES LIMITED + * All rights reserved. + * + * FUJITSU LABORATORIES LIMITED ("FLL") hereby grants you a limited, + * non-exclusive and royalty-free license to use, copy, modify and + * redistribute, solely for non-commercial and non-profit purposes, the + * computer program, "Risa/Asir" ("SOFTWARE"), subject to the terms and + * conditions of this Agreement. For the avoidance of doubt, you acquire + * only a limited right to use the SOFTWARE hereunder, and FLL or any + * third party developer retains all rights, including but not limited to + * copyrights, in and to the SOFTWARE. + * + * (1) FLL does not grant you a license in any way for commercial + * purposes. You may use the SOFTWARE only for non-commercial and + * non-profit purposes only, such as academic, research and internal + * business use. + * (2) The SOFTWARE is protected by the Copyright Law of Japan and + * international copyright treaties. If you make copies of the SOFTWARE, + * with or without modification, as permitted hereunder, you shall affix + * to all such copies of the SOFTWARE the above copyright notice. + * (3) An explicit reference to this SOFTWARE and its copyright owner + * shall be made on your publication or presentation in any form of the + * results obtained by use of the SOFTWARE. + * (4) In the event that you modify the SOFTWARE, you shall notify FLL by + * e-mail at risa-admin@sec.flab.fujitsu.co.jp of the detailed specification + * for such modification or the source code of the modified part of the + * SOFTWARE. + * + * THE SOFTWARE IS PROVIDED AS IS WITHOUT ANY WARRANTY OF ANY KIND. FLL + * MAKES ABSOLUTELY NO WARRANTIES, EXPRESSED, IMPLIED OR STATUTORY, AND + * EXPRESSLY DISCLAIMS ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE OR NONINFRINGEMENT OF THIRD PARTIES' + * RIGHTS. NO FLL DEALER, AGENT, EMPLOYEES IS AUTHORIZED TO MAKE ANY + * MODIFICATIONS, EXTENSIONS, OR ADDITIONS TO THIS WARRANTY. + * UNDER NO CIRCUMSTANCES AND UNDER NO LEGAL THEORY, TORT, CONTRACT, + * OR OTHERWISE, SHALL FLL BE LIABLE TO YOU OR ANY OTHER PERSON FOR ANY + * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, PUNITIVE OR CONSEQUENTIAL + * DAMAGES OF ANY CHARACTER, INCLUDING, WITHOUT LIMITATION, DAMAGES + * ARISING OUT OF OR RELATING TO THE SOFTWARE OR THIS AGREEMENT, DAMAGES + * FOR LOSS OF GOODWILL, WORK STOPPAGE, OR LOSS OF DATA, OR FOR ANY + * DAMAGES, EVEN IF FLL SHALL HAVE BEEN INFORMED OF THE POSSIBILITY OF + * SUCH DAMAGES, OR FOR ANY CLAIM BY ANY OTHER PARTY. EVEN IF A PART + * OF THE SOFTWARE HAS BEEN DEVELOPED BY A THIRD PARTY, THE THIRD PARTY + * DEVELOPER SHALL HAVE NO LIABILITY IN CONNECTION WITH THE USE, + * PERFORMANCE OR NON-PERFORMANCE OF THE SOFTWARE. + * + * $OpenXM: OpenXM_contrib2/asir2000/parse/lex.c,v 1.49 2015/08/08 14:19:42 fujimoto Exp $ +*/ #include #include "ca.h" #include "al.h" #include "base.h" #include "parse.h" -#if !defined(THINK_C) #include #include -#endif +#if defined(VISUAL) || defined(__MINGW32__) +#include "ytab.h" +#else #include "y.tab.h" +#endif +#if FEP +#include +#endif -extern IN asir_infile; +static int Getc(); +static void Ungetc(int c); +static void Gets(char *s); +static int skipspace(); + +extern INFILE asir_infile; extern struct oTKWD kwd[]; +extern Obj VOIDobj; -int afternl(); -int myatoi(); -int aftercomment(); - extern int main_parser; extern char *parse_strp; +extern int recv_intr; -static int skipspace(); -static int Getc(); -static void Ungetc(); -static void Gets(); +#define NBUFSIZ (BUFSIZ*10) +#define TBUFSIZ (BUFSIZ) -yylex() +#define REALLOC_NBUF \ +if ( i >= nbufsize ) {\ + nbufsize += NBUFSIZ;\ + if ( nbuf == nbuf0 ) {\ + nbuf = (char *)MALLOC_ATOMIC(nbufsize);\ + bcopy(nbuf0,nbuf,nbufsize-NBUFSIZ);\ + } else\ + nbuf = REALLOC(nbuf,nbufsize);\ +} + +#define REALLOC_TBUF \ +if ( i >= tbufsize ) {\ + tbufsize += TBUFSIZ;\ + if ( tbuf == tbuf0 ) {\ + tbuf = (char *)MALLOC_ATOMIC(tbufsize);\ + bcopy(tbuf0,tbuf,tbufsize-TBUFSIZ);\ + } else\ + tbuf = REALLOC(tbuf,tbufsize);\ +} + +#define READ_ALNUM_NBUF \ +while ( 1 ) {\ + c = Getc();\ + if ( isalnum(c) ) {\ + REALLOC_NBUF nbuf[i++] = c;\ + } else\ + break;\ +} + +#define READ_DIGIT_NBUF \ +while ( 1 ) {\ + c = Getc();\ + if ( isdigit(c) ) {\ + REALLOC_NBUF nbuf[i++] = c;\ + } else\ + break;\ +} + +int yylex() { #define yylvalp (&yylval) register int c,c1; register int *ptr; char *cptr; int d,i,j; -#if defined(__MWERKS__) - char nbuf[BUFSIZ],tbuf[BUFSIZ]; -#else - char nbuf[BUFSIZ*10],tbuf[BUFSIZ]; -#endif + char nbuf0[NBUFSIZ],tbuf0[TBUFSIZ]; + char *nbuf, *tbuf; + int nbufsize, tbufsize; N n,n1; Q q; Obj r; + int floatingpoint = 0; + double dbl; + Real real; + double atof(); + extern int bigfloat; + + /* initialize buffer pointers */ + nbuf = nbuf0; tbuf = tbuf0; + nbufsize = NBUFSIZ; tbufsize = TBUFSIZ; + switch ( c = skipspace() ) { case EOF : asir_terminate(2); break; @@ -48,20 +148,20 @@ yylex() while ( ( c = Getc() ) == '0' ); if ( c == '.' ) { Ungetc(c); c = '0'; - } else if ( c == 'x' ) { + } else if ( c == 'x' || c == 'X' ) { for ( i = 0; i < 8; i++ ) nbuf[i] = '0'; - for ( ; isalnum(c = Getc()); nbuf[i++] = c ); - Ungetc(c); nbuf[i] = 0; + READ_ALNUM_NBUF + Ungetc(c); REALLOC_NBUF nbuf[i] = 0; hexton(nbuf,&n1); NTOQ(n1,1,q); r = (Obj)q; yylvalp->p = (pointer)r; return ( FORMULA ); - } else if ( c == 'b' ) { + } else if ( c == 'b' || c == 'B' ) { for ( i = 0; i < 32; i++ ) nbuf[i] = '0'; - for ( ; isalnum(c = Getc()); nbuf[i++] = c ); - Ungetc(c); nbuf[i] = 0; + READ_ALNUM_NBUF + Ungetc(c); REALLOC_NBUF nbuf[i] = 0; binaryton(nbuf,&n1); NTOQ(n1,1,q); r = (Obj)q; yylvalp->p = (pointer)r; @@ -78,28 +178,46 @@ yylex() break; if ( c == '\\' ) c = Getc(); - tbuf[i] = c; + REALLOC_TBUF tbuf[i] = c; } - tbuf[i] = 0; + REALLOC_TBUF tbuf[i] = 0; cptr = (char *)MALLOC(strlen(tbuf)+1); strcpy(cptr,tbuf); yylvalp->p = (pointer)cptr; return LCASE; break; case '"' : -#if 0 - for ( i = 0; (nbuf[i] = Getc()) != '"'; i++ ); -#endif i = 0; do { c = Getc(); if ( c == '\\' ) { c1 = Getc(); - if ( c1 == 'n' ) + if ( c1 == 'n' ) { c1 = '\n'; - nbuf[i++] = c1; - } else - nbuf[i++] = c; + }else if ( c1 == 'r' ) { + c1 = '\r'; + }else if ( c1 == 't' ) { + c1 = '\t'; + }else if ( isdigit(c1) ){ + d = c1 - '0'; + c1 = Getc(); + if ( isdigit(c1) ) { + d = 8*d + (c1 - '0'); + c1 = Getc(); + if ( isdigit(c1) ) { + d = 8*d + (c1 - '0'); + }else { + Ungetc(c1); + } + }else { + Ungetc(c1); + } + c1 = d; + } + REALLOC_NBUF nbuf[i++] = c1; + } else { + REALLOC_NBUF nbuf[i++] = c; + } } while ( c != '"' ); - nbuf[i-1] = 0; + nbuf[i-1] = 0; /* REALLOC_NBUF is not necessary */ cptr = (char *)MALLOC(strlen(nbuf)+1); strcpy(cptr,nbuf); yylvalp->p = (pointer) cptr; return ( STR ); break; @@ -155,32 +273,39 @@ yylex() if ( isdigit(c) ) { for ( i = 0; i < DLENGTH; i++ ) nbuf[i] = '0'; - for ( nbuf[i++] = c; isdigit(c = Getc()); nbuf[i++] = c); + REALLOC_NBUF nbuf[i++] = c; + READ_DIGIT_NBUF if ( c == '.' ) { - double dbl; - Real real; - double atof(); - extern int bigfloat; + floatingpoint = 1; - for ( nbuf[i++] = c; isdigit(c = Getc()); nbuf[i++] = c); - if ( c == 'e' ) { - nbuf[i++] = c; - if ( ((c = Getc()) == '+') || (c == '-') ) { - nbuf[i++] = c; c = Getc(); - } - for ( ; isdigit(c); nbuf[i++] = c, c = Getc()); + REALLOC_NBUF nbuf[i++] = c; + READ_DIGIT_NBUF + if ( c == 'e' || c == 'E' ) { + REALLOC_NBUF nbuf[i++] = c; + c = Getc(); + if ( (c == '+') || (c == '-') ) { + REALLOC_NBUF nbuf[i++] = c; + } else + Ungetc(c); + READ_DIGIT_NBUF } - Ungetc(c); nbuf[i] = 0; -#if PARI + } else if ( c == 'e' || c == 'E' ) { + floatingpoint = 1; + REALLOC_NBUF nbuf[i++] = c; + c = Getc(); + if ( (c == '+') || (c == '-') ) { + REALLOC_NBUF nbuf[i++] = c; + } else + Ungetc(c); + READ_DIGIT_NBUF + } + if ( floatingpoint ) { + Ungetc(c); REALLOC_NBUF nbuf[i] = 0; if ( !bigfloat ) { dbl = (double)atof(nbuf+DLENGTH); MKReal(dbl,real); r = (Obj)real; } else strtobf(nbuf,(BF *)&r); -#else - dbl = (double)atof(nbuf+DLENGTH); - MKReal(dbl,real); r = (Obj)real; -#endif } else { Ungetc(c); i -= DLENGTH; d = (i%DLENGTH?i/DLENGTH+1:i/DLENGTH); @@ -193,11 +318,35 @@ yylex() } yylvalp->p = (pointer)r; return ( FORMULA ); - } else if ( isalpha(c) ) { - for ( i = 1, tbuf[0] = c; isalpha(c = Getc())||isdigit(c)||(c=='_'); i++ ) - tbuf[i] = c; - tbuf[i] = 0; Ungetc(c); - if ( isupper(tbuf[0]) ) { + } else if ( isalpha(c) || c == ':' || c == '_' ) { + if ( c == ':' ) { + c1 = Getc(); + if ( c1 != ':' ) { + Ungetc(c1); + return c; + } + c1 = Getc(); + if ( !isalpha(c1) ) { + Ungetc(c1); + return COLONCOLON; + } + i = 0; + tbuf[i++] = ':'; + tbuf[i++] = ':'; + tbuf[i++] = c1; + } else { + i = 0; + tbuf[i++] = c; + } + while ( 1 ) { + c = Getc(); + if ( isalpha(c)||isdigit(c)||(c=='_')||(c=='.') ) { + REALLOC_TBUF tbuf[i++] = c; + } else + break; + } + REALLOC_TBUF tbuf[i] = 0; Ungetc(c); + if ( isupper(tbuf[0]) || (tbuf[0] == '_' && isupper(tbuf[1])) ) { cptr = (char *)MALLOC(strlen(tbuf)+1); strcpy(cptr,tbuf); yylvalp->p = (pointer)cptr; return UCASE; @@ -214,8 +363,11 @@ yylex() } } else if ( c == '@' ) { if ( isdigit(c = Getc()) ) { - for ( i = 1, nbuf[0] = c; isdigit(c = Getc()); nbuf[i++] = c); - Ungetc(c); nbuf[i] = 0; yylvalp->i = atoi(nbuf); + i = 0; + nbuf[i++] = c; + READ_DIGIT_NBUF + Ungetc(c); REALLOC_NBUF nbuf[i] = 0; + yylvalp->i = atoi(nbuf); return ANS; } else if ( c == '@' ) { yylvalp->i = MAX(0,APVS->n-1); @@ -250,12 +402,25 @@ yylex() return FOP_AND; break; } } else if ( isalpha(c) ) { - for ( i = 2, tbuf[0] = '@', tbuf[1] = c; - isalpha(c = Getc()); tbuf[i++] = c); - Ungetc(c); tbuf[i] = 0; + i = 0; + tbuf[i++] = '@'; + tbuf[i++] = c; + while ( 1 ) { + c = Getc(); + if ( isalpha(c) ) { + REALLOC_TBUF tbuf[i++] = c; + } else + break; + } + Ungetc(c); REALLOC_TBUF tbuf[i] = 0; if ( !strcmp(tbuf,"@p") ) return GFPNGEN; - else if ( !strcmp(tbuf,"@i") ) { + else if ( !strcmp(tbuf,"@s") ) + return GFSNGEN; + else if ( !strcmp(tbuf,"@void") ) { + yylvalp->p = VOIDobj; + return FORMULA; + } else if ( !strcmp(tbuf,"@i") ) { extern pointer IU; yylvalp->p = IU; @@ -275,6 +440,15 @@ yylex() } else if ( !strcmp(tbuf,"@equiv") ) { yylvalp->i = (int)L_EQUIV; return FOP_EQUIV; + } else if ( !strcmp(tbuf,"@grlex") ) { + yylvalp->p = Symbol_grlex; + return FORMULA; + } else if ( !strcmp(tbuf,"@glex") ) { + yylvalp->p = Symbol_glex; + return FORMULA; + } else if ( !strcmp(tbuf,"@lex") ) { + yylvalp->p = Symbol_lex; + return FORMULA; } else { cptr = (char *)MALLOC(strlen(tbuf)+1); strcpy(cptr,tbuf); yylvalp->p = (pointer)cptr; @@ -288,15 +462,35 @@ yylex() return ( c ); } +void purge_stdin() +{ +#if defined(__FreeBSD__) || defined(__DARWIN__) + fpurge(stdin); +#elif defined(linux) + stdin->_IO_read_end = stdin->_IO_read_base; + stdin->_IO_read_ptr = stdin->_IO_read_base; +#elif defined(VISUAL_LIB) + void w_purge_stdin(); + + w_purge_stdin(); +#elif defined(sparc) || defined(__alpha) || defined(__SVR4) || defined(mips) || defined(VISUAL) || defined(__MINGW32__) || defined(_IBMR2) + stdin->_ptr = stdin->_base; stdin->_cnt = 0; +#elif (defined(__MACH__) && defined(__ppc__)) || defined(__CYGWIN__) || defined(__FreeBSD__) || defined(__INTERIX) + stdin->_r = 0; stdin->_p = stdin->_bf._base; +#else +--->FIXIT +#endif +} + static int skipspace() { int c,c1; for ( c = Getc(); ; ) switch ( c ) { - case ' ': case '\t': + case ' ': case '\t': case '\r': c = Getc(); break; case '\n': - c = afternl(); break; + c = afternl(); break; case '/': if ( (c1 = Getc()) == '*' ) c = aftercomment(); @@ -311,7 +505,7 @@ static int skipspace() { int afternl() { int c,ac,i,quote; - char *ptr; + char *ptr,*buf0; char *av[BUFSIZ]; static int ilevel = 0; char buf[BUFSIZ]; @@ -320,12 +514,15 @@ int afternl() { asir_infile->ln++; while ( (c = Getc()) == '#' ) { Gets(buf); - for ( quote = 0, ptr = buf; *ptr; ptr++ ) +#define LINE "line" + if ( !strncmp(buf,LINE,strlen(LINE)) ) buf0 = buf+strlen(LINE); + else buf0 = buf; + for ( quote = 0, ptr = buf0; *ptr; ptr++ ) if ( *ptr == '"' ) quote = quote ? 0 : 1; else if ( quote && (*ptr == ' ') ) *ptr = '_'; - stoarg(buf,&ac,av); + stoarg(buf0,&ac,av); if ( ac == 3 ) if ( (i = atoi(av[2])) == 1 ) ilevel++; @@ -341,6 +538,7 @@ int aftercomment() { int c,c1; for ( c = Getc(); ; ) { + if ( c == '\n' ) asir_infile->ln++; c1 = Getc(); if ( (c == '*') && (c1 == '/') ) return Getc(); @@ -349,8 +547,7 @@ int aftercomment() { } } -int myatoi(s) -char *s; +int myatoi(char *s) { int i,r; for ( i = 0, r = 0; i < DLENGTH; i++ ) r = r * 10 + ( s[i] - '0' ); @@ -358,11 +555,20 @@ char *s; } extern int ox_do_copy; +extern int I_am_server; +extern JMP_BUF main_env; +extern int at_root; +extern LIST LastStackTrace; +extern char *CUR_FUNC; -void yyerror(s) -char *s; +void yyerror(char *s) { - if ( main_parser ) + STRING fname,name,kwd; + USINT u; + NODE t; + LIST l,l2; + + if ( main_parser ) { if ( ox_do_copy ) { /* push errors to DebugStack */ } else { @@ -371,29 +577,65 @@ char *s; else fprintf(stderr,"\"%s\", near line %d: %s\n",asir_infile->name,asir_infile->ln,s); } - else + if ( I_am_server ) { + if ( NEXT(asir_infile) ) { + /* error in a file; record the position */ + MKSTR(fname,asir_infile->name); + if ( CPVS == GPVS ) + MKSTR(name,""); + else + MKSTR(name,CUR_FUNC); + MKUSINT(u,asir_infile->ln); + t = mknode(3,fname,name,u); MKLIST(l,t); + /* line number at the toplevel */ + MKSTR(fname,"toplevel"); MKUSINT(u,at_root); + t = mknode(2,fname,u); MKLIST(l2,t); + t = mknode(2,l2,l); + } else { + MKSTR(fname,"toplevel"); MKUSINT(u,asir_infile->ln); + t = mknode(2,fname,u); MKLIST(l,t); + t = mknode(1,l); + } + MKLIST(l,t); + MKSTR(kwd,"asir_where"); t = mknode(2,kwd,l); + MKLIST(LastStackTrace,t); + set_lasterror(s); + LONGJMP(main_env,1); + } + } else fprintf(stderr,"exprparse : %s\n",s); } int echoback; -extern int read_exec_file, do_fep, do_file; +extern int do_fep, do_file; -int readline_getc(); -void readline_ungetc(); -int Egetc(); -void Eungetc(); - unsigned char encrypt_char(unsigned char); unsigned char decrypt_char(unsigned char); -int Egetc(fp) -FILE *fp; +int Egetc(FILE *fp) { int c; if ( fp ) { - c = getc(fp); +#if FEP + if ( do_fep && isatty(fileno(fp)) ) + c = readline_getc(); + else +#endif + c = getc(fp); +#if defined(VISUAL) || defined(__MINGW32__) + if ( recv_intr ) { +#include + if ( recv_intr == 1 ) { + recv_intr = 0; + int_handler(SIGINT); + } else { + recv_intr = 0; + ox_usr1_handler(0); + } + } +#endif if ( c == EOF ) return c; if ( asir_infile->encoded ) @@ -408,14 +650,17 @@ FILE *fp; } } -void Eungetc(c,fp) -int c; -FILE *fp; +void Eungetc(int c,FILE *fp) { if ( fp ) { if ( asir_infile->encoded ) c = (int)encrypt_char((unsigned char)c); - ungetc(c,fp); +#if FEP + if ( do_fep && isatty(fileno(fp)) ) + readline_ungetc(); + else +#endif + ungetc(c,fp); } else *--parse_strp = c; } @@ -428,13 +673,18 @@ static int Getc() { if ((c = Egetc(asir_infile->fp)) == EOF) if ( NEXT(asir_infile) ) { closecurrentinput(); - c = Getc(); + /* if the input is the top level, generate error */ + if ( !NEXT(asir_infile) ) + error("end-of-file detected during parsing"); + else + c = Getc(); break; - } else if ( read_exec_file || do_file ) - asir_terminate(2); - else { + } else if ( asir_infile->fp || do_file ) { if ( asir_infile->fp ) clearerr(asir_infile->fp); + asir_terminate(2); + } else { + error("end-of-line detected during parsing"); } else break; @@ -446,7 +696,7 @@ static int Getc() { return ( c ); } -static void Ungetc(c) { +static void Ungetc(int c) { if ( main_parser ) { Eungetc(c,asir_infile->fp); if ( echoback ) @@ -455,8 +705,7 @@ static void Ungetc(c) { *--parse_strp = c; } -static void Gets(s) -char *s; +static void Gets(char *s) { int c; @@ -464,3 +713,57 @@ char *s; *s++ = c; *s = 0; } + +#if FEP + +static char *readline_line; +static int readline_nc,readline_index; +char *readline_console(); + +int readline_getc() +{ + char buf[BUFSIZ]; + + if ( !readline_nc ) { + if ( readline_line ) + free(readline_line); + sprompt(buf); + readline_line = readline_console(buf); + readline_nc = strlen(readline_line); + readline_index = 0; + } + readline_nc--; + return readline_line[readline_index++]; +} + +void readline_ungetc() +{ + readline_nc++; readline_index--; +} + +char *readline_console(char *prompt) +{ + char *line; + int exp_result; + char *expansion; + + while ( 1 ) { + line = (char *)readline(prompt); + if ( line && *line ) { + using_history(); + exp_result = history_expand(line,&expansion); + if ( !exp_result ) { + free(expansion); + for ( ; isspace((unsigned char)*line); line++ ); + add_history(line); + break; + } else if ( exp_result > 0 ) { + free(line); + line = expansion; + break; + } + } + } + return line; +} +#endif