nuttx-apps/interpreters/bas/program.c

778 lines
19 KiB
C

/* Program storage. */
/* #includes */ /*{{{C}}}*//*{{{*/
#undef _POSIX_SOURCE
#define _POSIX_SOURCE 1
#undef _POSIX_C_SOURCE
#define _POSIX_C_SOURCE 2
#include "config.h"
#include <assert.h>
#include <errno.h>
#ifdef HAVE_GETTEXT
#include <libintl.h>
#define _(String) gettext(String)
#else
#define _(String) String
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "auto.h"
#include "error.h"
#include "fs.h"
#include "program.h"
/*}}}*/
struct Program *Program_new(struct Program *this) /*{{{*/
{
this->trace=0;
this->size=0;
this->numbered=1;
this->capacity=0;
this->runnable=0;
this->unsaved=0;
this->code=(struct Token**)0;
this->scope=(struct Scope*)0;
String_new(&this->name);
return this;
}
/*}}}*/
void Program_destroy(struct Program *this) /*{{{*/
{
while (this->size) Token_destroy(this->code[--this->size]);
if (this->capacity) free(this->code);
this->code=(struct Token**)0;
this->scope=(struct Scope*)0;
String_destroy(&this->name);
}
/*}}}*/
void Program_norun(struct Program *this) /*{{{*/
{
this->runnable=0;
this->scope=(struct Scope*)0;
}
/*}}}*/
void Program_store(struct Program *this, struct Token *line, long int where) /*{{{*/
{
int i;
assert(line->type==T_INTEGER || line->type==T_UNNUMBERED);
this->runnable=0;
this->unsaved=1;
if (line->type==T_UNNUMBERED) this->numbered=0;
if (where)
{
int last=-1;
for (i=0; i<this->size; ++i)
{
assert(this->code[i]->type==T_INTEGER || this->code[i]->type==T_UNNUMBERED);
if (where>last && where<this->code[i]->u.integer)
{
if ((this->size+1)>=this->capacity)
{
this->code=realloc(this->code,sizeof(struct Token*)*(this->capacity?(this->capacity*=2):(this->capacity=256)));
}
memmove(&this->code[i+1],&this->code[i],(this->size-i)*sizeof(struct Token*));
this->code[i]=line;
++this->size;
return;
}
else if (where==this->code[i]->u.integer)
{
Token_destroy(this->code[i]);
this->code[i]=line;
return;
}
last=this->code[i]->u.integer;
}
}
else i=this->size;
if ((this->size+1)>=this->capacity)
{
this->code=realloc(this->code,sizeof(struct Token*)*(this->capacity?(this->capacity*=2):(this->capacity=256)));
}
this->code[i]=line;
++this->size;
}
/*}}}*/
void Program_delete(struct Program *this, const struct Pc *from, const struct Pc *to) /*{{{*/
{
int i, first, last;
this->runnable=0;
this->unsaved=1;
first=from ? from->line : 0;
last=to ? to->line : this->size-1;
for (i=first; i<=last; ++i) Token_destroy(this->code[i]);
if ((last+1)!=this->size) memmove(&this->code[first],&this->code[last+1],(this->size-last+1)*sizeof(struct Token*));
this->size-=(last-first+1);
}
/*}}}*/
void Program_addScope(struct Program *this, struct Scope *scope) /*{{{*/
{
struct Scope *s;
s=this->scope;
this->scope=scope;
scope->next=s;
}
/*}}}*/
struct Pc *Program_goLine(struct Program *this, long int line, struct Pc *pc) /*{{{*/
{
int i;
for (i=0; i<this->size; ++i)
{
if (this->code[i]->type==T_INTEGER && line==this->code[i]->u.integer)
{
pc->line=i;
pc->token=this->code[i]+1;
return pc;
}
}
return (struct Pc*)0;
}
/*}}}*/
struct Pc *Program_fromLine(struct Program *this, long int line, struct Pc *pc) /*{{{*/
{
int i;
for (i=0; i<this->size; ++i)
{
if (this->code[i]->type==T_INTEGER && this->code[i]->u.integer>=line)
{
pc->line=i;
pc->token=this->code[i]+1;
return pc;
}
}
return (struct Pc*)0;
}
/*}}}*/
struct Pc *Program_toLine(struct Program *this, long int line, struct Pc *pc) /*{{{*/
{
int i;
for (i=this->size-1; i>=0; --i)
{
if (this->code[i]->type==T_INTEGER && this->code[i]->u.integer<=line)
{
pc->line=i;
pc->token=this->code[i]+1;
return pc;
}
}
return (struct Pc*)0;
}
/*}}}*/
int Program_scopeCheck(struct Program *this, struct Pc *pc, struct Pc *fn) /*{{{*/
{
struct Scope *scope;
if (fn==(struct Pc*)0) /* jump from global block must go to global pc */
{
for (scope=this->scope; scope; scope=scope->next)
{
if (pc->line<scope->begin.line) continue;
if (pc->line==scope->begin.line && pc->token<=scope->begin.token) continue;
if (pc->line>scope->end.line) continue;
if (pc->line==scope->end.line && pc->token>scope->end.token) continue;
return -1;
}
}
else /* jump from local block must go to local block */
{
scope=&(fn->token+1)->u.identifier->sym->u.sub.u.def.scope;
if (pc->line<scope->begin.line) return -1;
if (pc->line==scope->begin.line && pc->token<=scope->begin.token) return -1;
if (pc->line>scope->end.line) return -1;
if (pc->line==scope->end.line && pc->token>scope->end.token) return -1;
}
return 0;
}
/*}}}*/
struct Pc *Program_dataLine(struct Program *this, long int line, struct Pc *pc) /*{{{*/
{
if ((pc=Program_goLine(this,line,pc))==(struct Pc*)0) return (struct Pc*)0;
while (pc->token->type!=T_DATA)
{
if (pc->token->type==T_EOL) return (struct Pc*)0;
else ++pc->token;
}
return pc;
}
/*}}}*/
struct Pc *Program_imageLine(struct Program *this, long int line, struct Pc *pc) /*{{{*/
{
if ((pc=Program_goLine(this,line,pc))==(struct Pc*)0) return (struct Pc*)0;
while (pc->token->type!=T_IMAGE)
{
if (pc->token->type==T_EOL) return (struct Pc*)0;
else ++pc->token;
}
++pc->token;
if (pc->token->type!=T_STRING) return (struct Pc*)0;
return pc;
}
/*}}}*/
long int Program_lineNumber(const struct Program *this, const struct Pc *pc) /*{{{*/
{
if (pc->line==-1) return 0;
if (this->numbered) return (this->code[pc->line]->u.integer);
else return (pc->line+1);
}
/*}}}*/
struct Pc *Program_beginning(struct Program *this, struct Pc *pc) /*{{{*/
{
if (this->size==0) return (struct Pc*)0;
else
{
pc->line=0;
pc->token=this->code[0]+1;
return pc;
}
}
/*}}}*/
struct Pc *Program_end(struct Program *this, struct Pc *pc) /*{{{*/
{
if (this->size==0) return (struct Pc*)0;
else
{
pc->line=this->size-1;
pc->token=this->code[this->size-1];
while (pc->token->type!=T_EOL) ++pc->token;
return pc;
}
}
/*}}}*/
struct Pc *Program_nextLine(struct Program *this, struct Pc *pc) /*{{{*/
{
if (pc->line+1==this->size) return (struct Pc*)0;
else
{
pc->token=this->code[++pc->line]+1;
return pc;
}
}
/*}}}*/
int Program_skipEOL(struct Program *this, struct Pc *pc, int dev, int tr) /*{{{*/
{
if (pc->token->type==T_EOL)
{
if (pc->line==-1 || pc->line+1==this->size) return 0;
{
pc->token=this->code[++pc->line]+1;
Program_trace(this,pc,dev,tr);
return 1;
}
}
else return 1;
}
/*}}}*/
void Program_trace(struct Program *this, struct Pc *pc, int dev, int tr) /*{{{*/
{
if (tr && this->trace && pc->line!=-1)
{
char buf[40];
sprintf(buf,"<%ld>\n",this->code[pc->line]->u.integer);
FS_putChars(dev,buf);
}
}
/*}}}*/
void Program_PCtoError(struct Program *this, struct Pc *pc, struct Value *v) /*{{{*/
{
struct String s;
String_new(&s);
if (pc->line>=0)
{
if (pc->line<(this->size-1) || pc->token->type!=T_EOL)
{
String_appendPrintf(&s,_(" in line %ld at:\n"),Program_lineNumber(this,pc));
Token_toString(this->code[pc->line],(struct Token*)0,&s,(int*)0,-1);
Token_toString(this->code[pc->line],pc->token,&s,(int*)0,-1);
String_appendPrintf(&s,"^\n");
}
else
{
String_appendPrintf(&s,_(" at: end of program\n"));
}
}
else
{
String_appendPrintf(&s,_(" at: "));
if (pc->token->type!=T_EOL) Token_toString(pc->token,(struct Token*)0,&s,(int*)0,-1);
else String_appendPrintf(&s,_("end of line\n"));
}
Value_errorSuffix(v,s.character);
String_destroy(&s);
}
/*}}}*/
struct Value *Program_merge(struct Program *this, int dev, struct Value *value) /*{{{*/
{
struct String s;
int l,err=0;
l=0;
while (String_new(&s),(err=FS_appendToString(dev,&s,1))!=-1 && s.length)
{
struct Token *line;
++l;
if (l!=1 || s.character[0]!='#')
{
line=Token_newCode(s.character);
if (line->type==T_INTEGER && line->u.integer>0) Program_store(this,line,this->numbered?line->u.integer:0);
else if (line->type==T_UNNUMBERED) Program_store(this,line,0);
else
{
Token_destroy(line);
return Value_new_ERROR(value,INVALIDLINE,l);
}
}
String_destroy(&s);
}
String_destroy(&s);
if (err) return Value_new_ERROR(value,IOERROR,FS_errmsg);
return (struct Value*)0;
}
/*}}}*/
int Program_lineNumberWidth(struct Program *this) /*{{{*/
{
int i,w=0;
for (i=0; i<this->size; ++i) if (this->code[i]->type==T_INTEGER)
{
int nw,ln;
for (ln=this->code[i]->u.integer,nw=1; ln/=10; ++nw);
if (nw>w) w=nw;
}
return w;
}
/*}}}*/
struct Value *Program_list(struct Program *this, int dev, int watchIntr, struct Pc *from, struct Pc *to, struct Value *value) /*{{{*/
{
int i,w;
int indent=0;
struct String s;
w=Program_lineNumberWidth(this);
for (i=0; i<this->size; ++i)
{
String_new(&s);
Token_toString(this->code[i],(struct Token*)0,&s,&indent,w);
if ((from==(struct Pc *)0 || from->line<=i) && (to==(struct Pc*)0 || to->line>=i))
{
if (FS_putString(dev,&s)==-1) return Value_new_ERROR(value,IOERROR,FS_errmsg);
if (watchIntr && FS_intr) return Value_new_ERROR(value,BREAK);
}
String_destroy(&s);
}
return (struct Value*)0;
}
/*}}}*/
struct Value *Program_analyse(struct Program *this, struct Pc *pc, struct Value *value) /*{{{*/
{
int i;
for (i=0; i<this->size; ++i)
{
pc->token=this->code[i];
pc->line=i;
if (pc->token->type==T_INTEGER || pc->token->type==T_UNNUMBERED) ++pc->token;
for (;;)
{
if (pc->token->type==T_GOTO || pc->token->type==T_RESUME || pc->token->type==T_RETURN || pc->token->type==T_END || pc->token->type==T_STOP)
{
++pc->token;
while (pc->token->type==T_INTEGER)
{
++pc->token;
if (pc->token->type==T_COMMA) ++pc->token;
else break;
}
if (pc->token->type==T_COLON)
{
++pc->token;
switch (pc->token->type)
{
case T_EOL:
case T_DEFPROC:
case T_SUB:
case T_DEFFN:
case T_FUNCTION:
case T_COLON:
case T_REM:
case T_QUOTE: break; /* those are fine to be unreachable */
default: return Value_new_ERROR(value,UNREACHABLE);
}
}
}
if (pc->token->type==T_EOL) break;
else ++pc->token;
}
}
return (struct Value*)0;
}
/*}}}*/
void Program_renum(struct Program *this, int first, int inc) /*{{{*/
{
int i;
struct Token *token;
for (i=0; i<this->size; ++i)
{
for (token=this->code[i]; token->type!=T_EOL; )
{
if (token->type==T_GOTO || token->type==T_GOSUB || token->type==T_RESTORE || token->type==T_RESUME || token->type==T_USING)
{
++token;
while (token->type==T_INTEGER)
{
struct Pc dst;
if (Program_goLine(this,token->u.integer,&dst)) token->u.integer=first+dst.line*inc;
++token;
if (token->type==T_COMMA) ++token;
else break;
}
}
else ++token;
}
}
for (i=0; i<this->size; ++i)
{
assert(this->code[i]->type==T_INTEGER || this->code[i]->type==T_UNNUMBERED);
this->code[i]->type=T_INTEGER;
this->code[i]->u.integer=first+i*inc;
}
this->numbered=1;
this->runnable=0;
this->unsaved=1;
}
/*}}}*/
void Program_unnum(struct Program *this) /*{{{*/
{
char *ref;
int i;
struct Token *token;
ref=malloc(this->size);
memset(ref,0,this->size);
for (i=0; i<this->size; ++i)
{
for (token=this->code[i]; token->type!=T_EOL; ++token)
{
if (token->type==T_GOTO || token->type==T_GOSUB || token->type==T_RESTORE || token->type==T_RESUME)
{
++token;
while (token->type==T_INTEGER)
{
struct Pc dst;
if (Program_goLine(this,token->u.integer,&dst)) ref[dst.line]=1;
++token;
if (token->type==T_COMMA) ++token;
else break;
}
}
}
}
for (i=0; i<this->size; ++i)
{
assert(this->code[i]->type==T_INTEGER || this->code[i]->type==T_UNNUMBERED);
if (!ref[i])
{
this->code[i]->type=T_UNNUMBERED;
this->numbered=0;
}
}
free(ref);
this->runnable=0;
this->unsaved=1;
}
/*}}}*/
int Program_setname(struct Program *this, const char *filename) /*{{{*/
{
if (this->name.length) String_delete(&this->name,0,this->name.length);
if (filename) return String_appendChars(&this->name,filename);
else return 0;
}
/*}}}*/
/*
The list of line numbers is circular, which avoids the need to have one
extra pointer for the head (for ordered output). Instead only a pointer
to the tail is needed. The tail's next element is the head of the list.
tail --> last element <-- ... <-- first element <--,
\ /
\_________________________________/
*/
struct Xref
{
const void *key;
struct LineNumber
{
struct Pc line;
struct LineNumber *next;
} *lines;
struct Xref *l,*r;
};
static void Xref_add(struct Xref **root, int (*cmp)(const void*,const void*), const void *key, struct Pc *line) /*{{{*/
{
int res;
struct LineNumber **tail;
struct LineNumber *new;
while (*root && (res=cmp(key,(*root)->key))) root=(res<0)?&(*root)->l:&(*root)->r;
if (*root==(struct Xref*)0)
{
*root=malloc(sizeof(struct Xref));
(*root)->key=key;
(*root)->l=(*root)->r=(struct Xref*)0;
/* create new circular list */
(*root)->lines=new=malloc(sizeof(struct LineNumber));
new->line=*line;
new->next=new;
}
else
{
/* add to existing circular list */
tail=&(*root)->lines;
if ((*tail)->line.line!=line->line)
{
new=malloc(sizeof(struct LineNumber));
new->line=*line;
new->next=(*tail)->next;
(*tail)->next=new;
*tail=new;
}
}
}
/*}}}*/
static void Xref_destroy(struct Xref *root) /*{{{*/
{
if (root)
{
struct LineNumber *cur,*next,*tail;
Xref_destroy(root->l);
Xref_destroy(root->r);
cur=tail=root->lines;
do
{
next=cur->next;
free(cur);
cur=next;
} while (cur!=tail);
free(root);
}
}
/*}}}*/
static void Xref_print(struct Xref *root, void (*print)(const void *key, struct Program *p, int chn), struct Program *p, int chn) /*{{{*/
{
if (root)
{
const struct LineNumber *cur,*tail;
Xref_print(root->l,print,p,chn);
print(root->key,p,chn);
cur=tail=root->lines;
do
{
char buf[128];
cur=cur->next;
if (FS_charpos(chn)>72) FS_putChars(chn,"\n ");
sprintf(buf," %ld",Program_lineNumber(p,&cur->line));
FS_putChars(chn,buf);
} while (cur!=tail);
FS_putChar(chn,'\n');
Xref_print(root->r,print,p,chn);
}
}
/*}}}*/
static int cmpLine(const void *a, const void *b) /*{{{*/
{
const register struct Pc *pcA=(const struct Pc*)a,*pcB=(const struct Pc*)b;
return pcA->line-pcB->line;
}
/*}}}*/
static void printLine(const void *k, struct Program *p, int chn) /*{{{*/
{
char buf[80];
sprintf(buf,"%8ld",Program_lineNumber(p,(const struct Pc*)k));
FS_putChars(chn,buf);
}
/*}}}*/
static int cmpName(const void *a, const void *b) /*{{{*/
{
const register char *funcA=(const char*)a,*funcB=(const char*)b;
return strcmp(funcA,funcB);
}
/*}}}*/
static void printName(const void *k, struct Program *p, int chn) /*{{{*/
{
size_t len=strlen((const char*)k);
FS_putChars(chn,(const char*)k);
if (len<8) FS_putChars(chn," "+len);
}
/*}}}*/
void Program_xref(struct Program *this, int chn) /*{{{*/
{
struct Pc pc;
struct Xref *func,*var,*gosub,*goto_;
int nl=0;
assert(this->runnable);
func=(struct Xref*)0;
var=(struct Xref*)0;
gosub=(struct Xref*)0;
goto_=(struct Xref*)0;
for (pc.line=0; pc.line<this->size; ++pc.line)
{
struct On *on;
for (on=(struct On*)0,pc.token=this->code[pc.line]; pc.token->type!=T_EOL; ++pc.token)
{
switch (pc.token->type)
{
case T_ON: /*{{{*/
{
on=&pc.token->u.on;
break;
}
/*}}}*/
case T_GOTO: /*{{{*/
{
if (on)
{
int key;
for (key=0; key<on->pcLength; ++key) Xref_add(&goto_,cmpLine,&on->pc[key],&pc);
on=(struct On*)0;
}
else Xref_add(&goto_,cmpLine,&pc.token->u.gotopc,&pc);
break;
}
/*}}}*/
case T_GOSUB: /*{{{*/
{
if (on)
{
int key;
for (key=0; key<on->pcLength; ++key) Xref_add(&gosub,cmpLine,&on->pc[key],&pc);
on=(struct On*)0;
}
else Xref_add(&gosub,cmpLine,&pc.token->u.gosubpc,&pc);
break;
}
/*}}}*/
case T_DEFFN:
case T_DEFPROC:
case T_FUNCTION:
case T_SUB: /*{{{*/
{
++pc.token;
Xref_add(&func,cmpName,&pc.token->u.identifier->name,&pc);
break;
}
/*}}}*/
default: break;
}
}
}
for (pc.line=0; pc.line<this->size; ++pc.line)
{
for (pc.token=this->code[pc.line]; pc.token->type!=T_EOL; ++pc.token)
{
switch (pc.token->type)
{
case T_DEFFN:
case T_DEFPROC:
case T_FUNCTION:
case T_SUB: /* skip identifier already added above */ /*{{{*/
{
++pc.token;
break;
}
/*}}}*/
case T_IDENTIFIER: /*{{{*/
{
/* formal parameters have no assigned symbol */
if (pc.token->u.identifier->sym) switch (pc.token->u.identifier->sym->type)
{
case GLOBALVAR:
{
Xref_add(&var,cmpName,&pc.token->u.identifier->name,&pc);
break;
}
case USERFUNCTION:
{
Xref_add(&func,cmpName,&pc.token->u.identifier->name,&pc);
break;
}
default: break;
}
break;
}
/*}}}*/
default: break;
}
}
}
if (func)
{
FS_putChars(chn,_("Function Referenced in line\n"));
Xref_print(func,printName,this,chn);
Xref_destroy(func);
nl=1;
}
if (var)
{
if (nl) FS_putChar(chn,'\n');
FS_putChars(chn,_("Variable Referenced in line\n"));
Xref_print(var,printName,this,chn);
Xref_destroy(func);
nl=1;
}
if (gosub)
{
if (nl) FS_putChar(chn,'\n');
FS_putChars(chn,_("Gosub Referenced in line\n"));
Xref_print(gosub,printLine,this,chn);
Xref_destroy(gosub);
nl=1;
}
if (goto_)
{
if (nl) FS_putChar(chn,'\n');
FS_putChars(chn,_("Goto Referenced in line\n"));
Xref_print(goto_,printLine,this,chn);
Xref_destroy(goto_);
nl=1;
}
}
/*}}}*/