| version 1.11, 2002/11/04 11:08:59 | version 1.28, 2004/09/17 02:42:57 | 
|  |  | 
| /* $OpenXM: OpenXM/src/kan96xx/Kan/stackmachine.c,v 1.10 2002/11/04 10:53:56 takayama Exp $ */ | /* $OpenXM: OpenXM/src/kan96xx/Kan/stackmachine.c,v 1.27 2004/09/16 23:53:44 takayama Exp $ */ | 
| /*   stackmachin.c */ | /*   stackmachin.c */ | 
|  |  | 
| #include <stdio.h> | #include <stdio.h> | 
| 
| Line 54  struct context *PrimitiveContextp = &StandardContext; |  | 
| Line 54  struct context *PrimitiveContextp = &StandardContext; |  | 
|  |  | 
| static struct object ObjTmp; /* for poor compiler */ | static struct object ObjTmp; /* for poor compiler */ | 
|  |  | 
|  | int Calling_ctrlC_hook = 0; | 
|  |  | 
| int StandardMacros = 1; | int StandardMacros = 1; | 
| int StartAFile = 0; | int StartAFile = 0; | 
| char *StartFile; | char *StartFile; | 
| 
| Line 74  static void pstack(void); |  | 
| Line 76  static void pstack(void); |  | 
| static struct object executableStringToExecutableArray(char *str); | static struct object executableStringToExecutableArray(char *str); | 
|  |  | 
| extern int SerialCurrent; | extern int SerialCurrent; | 
|  | extern int QuoteMode; | 
|  |  | 
| int SGClock = 0; | int SGClock = 0; | 
| int UserCtrlC = 0; | int UserCtrlC = 0; | 
| int OXlock = 0; | int OXlock = 0; | 
| int OXlockSaved = 0; | int OXlockSaved = 0; | 
|  |  | 
|  | char *UD_str; | 
|  | int  UD_attr; | 
|  |  | 
| struct object * newObject() | struct object * newObject() | 
| { | { | 
| struct object *r; | struct object *r; | 
| 
| Line 215  int putUserDictionary(str,h0,h1,ob,dic) |  | 
| Line 221  int putUserDictionary(str,h0,h1,ob,dic) |  | 
| } | } | 
| r = x; | r = x; | 
| if (Strict2) { | if (Strict2) { | 
| switch((dic[x]).attr) { | switch(((dic[x]).attr) & (PROTECT | ABSOLUTE_PROTECT)) { | 
| case PROTECT: | case PROTECT: | 
| r = -PROTECT;   /* Protected, but we rewrite it. */ | r = -PROTECT;   /* Protected, but we rewrite it. */ | 
| break; | break; | 
| 
| Line 223  int putUserDictionary(str,h0,h1,ob,dic) |  | 
| Line 229  int putUserDictionary(str,h0,h1,ob,dic) |  | 
| r = -ABSOLUTE_PROTECT;  /* Protected and we do not rewrite it. */ | r = -ABSOLUTE_PROTECT;  /* Protected and we do not rewrite it. */ | 
| return(r); | return(r); | 
| default: | default: | 
| (dic[x]).attr = 0; | /* (dic[x]).attr = 0; */ /* It is not necesarry, I think. */ | 
| break; | break; | 
| } | } | 
| } | } | 
| 
| Line 246  struct object findUserDictionary(str,h0,h1,cp) |  | 
| Line 252  struct object findUserDictionary(str,h0,h1,cp) |  | 
| char *str;    /* key */ | char *str;    /* key */ | 
| int h0,h1;    /* The hashing values of the key. */ | int h0,h1;    /* The hashing values of the key. */ | 
| struct context *cp; | struct context *cp; | 
|  | /* Set char *UD_str, int UD_attr (attributes) */ | 
| { | { | 
| int x; | int x; | 
| struct dictionary *dic; | struct dictionary *dic; | 
|  | extern char *UD_str; | 
|  | extern int UD_attr; | 
|  | UD_str = NULL; UD_attr = -1; | 
| dic = cp->userDictionary; | dic = cp->userDictionary; | 
| x = h0; | x = h0; | 
| while (1) { | while (1) { | 
| if ((dic[x]).key == EMPTY) { break; } | if ((dic[x]).key == EMPTY) { break; } | 
| if (strcmp((dic[x]).key,str) == 0) { | if (strcmp((dic[x]).key,str) == 0) { | 
|  | UD_str = (dic[x]).key; UD_attr = (dic[x]).attr; | 
| return( (dic[x]).obj ); | return( (dic[x]).obj ); | 
| } | } | 
| x = (x+h1) % USER_DICTIONARY_SIZE; | x = (x+h1) % USER_DICTIONARY_SIZE; | 
| 
| Line 284  int putUserDictionary2(str,h0,h1,attr,dic) |  | 
| Line 295  int putUserDictionary2(str,h0,h1,attr,dic) |  | 
| } | } | 
| return(0); | return(0); | 
| } | } | 
|  | if (OR_ATTR_FOR_ALL_WORDS & attr) { | 
|  | for (i=0; i<USER_DICTIONARY_SIZE; i++) { | 
|  | if ((dic[i]).key !=EMPTY) (dic[i]).attr |= attr&(~OR_ATTR_FOR_ALL_WORDS); | 
|  | } | 
|  | return(0); | 
|  | } | 
| x = h0; | x = h0; | 
| if (str[0] == '\0') { | if (str[0] == '\0') { | 
| errorKan1("%s\n","putUserDictionary2(): You are defining a value with the null key."); | errorKan1("%s\n","putUserDictionary2(): You are defining a value with the null key."); | 
|  |  | 
| { | { | 
| int h=0; | int h=0; | 
| while (*str != '\0') { | while (*str != '\0') { | 
| h = ((h*128)+(*str)) % USER_DICTIONARY_SIZE; | h = ((h*128)+((unsigned char)(*str))) % USER_DICTIONARY_SIZE; | 
| str++; | str++; | 
| } | } | 
| return(h); | return(h); | 
|  |  | 
| int hash1(str) | int hash1(str) | 
| char *str; | char *str; | 
| { | { | 
| return(8-(str[0]%8)); | return(8-((unsigned char)(str[0])%8)); | 
| } | } | 
|  |  | 
| void hashInitialize(struct dictionary *dic) | void hashInitialize(struct dictionary *dic) | 
| 
| Line 729  void scanner() { |  | 
| Line 746  void scanner() { |  | 
| struct object ob; | struct object ob; | 
| extern int Quiet; | extern int Quiet; | 
| extern void ctrlC(); | extern void ctrlC(); | 
| int tmp; | int tmp, status; | 
| char *tmp2; | char *tmp2; | 
| extern int ErrorMessageMode; | extern int ErrorMessageMode; | 
| int jval; | int jval; | 
|  | extern int InSendmsg2; | 
|  | int infixOn = 0; | 
|  | struct tokens infixToken; | 
|  | extern int RestrictedMode, RestrictedMode_saved; | 
| getokenSM(INIT); | getokenSM(INIT); | 
| initSystemDictionary(); | initSystemDictionary(); | 
|  |  | 
| 
| Line 750  void scanner() { |  | 
| Line 771  void scanner() { |  | 
| } | } | 
|  |  | 
| /* setup quiet mode or not */ | /* setup quiet mode or not */ | 
| token.kind = EXECUTABLE_STRING; | token.kind = EXECUTABLE_STRING; token.tflag = 0; | 
| if (Quiet) { | if (Quiet) { | 
| token.token = " /@@@.quiet 1 def "; | token.token = " /@@@.quiet 1 def "; | 
| }else { | }else { | 
| token.token = " /@@@.quiet 0 def "; | token.token = " /@@@.quiet 0 def "; | 
| } | } | 
| executeToken(token); /* execute startup commands */ | executeToken(token); /* execute startup commands */ | 
| token.kind = ID; | token.kind = ID; token.tflag = 0; | 
| token.token = "exec"; | token.token = "exec"; | 
| token = lookupTokens(token); /* set hashing values */ | token = lookupTokens(token); /* set hashing values */ | 
| tmp = findSystemDictionary(token.token); | tmp = findSystemDictionary(token.token); | 
| 
| Line 773  void scanner() { |  | 
| Line 794  void scanner() { |  | 
| StartFile = (char *)sGC_malloc(sizeof(char)*(strlen(StartFile)+ | StartFile = (char *)sGC_malloc(sizeof(char)*(strlen(StartFile)+ | 
| 40)); | 40)); | 
| sprintf(StartFile,"$%s$ run\n",tmp2); | sprintf(StartFile,"$%s$ run\n",tmp2); | 
| token.kind = EXECUTABLE_STRING; | token.kind = EXECUTABLE_STRING; token.tflag = 0; | 
| token.token = StartFile; | token.token = StartFile; | 
| executeToken(token);    /* execute startup commands */ | executeToken(token);    /* execute startup commands */ | 
| token.kind = ID; | token.kind = ID; token.tflag = 0; | 
| token.token = "exec"; | token.token = "exec"; | 
| token = lookupTokens(token); /* set hashing values */ | token = lookupTokens(token); /* set hashing values */ | 
| tmp = findSystemDictionary(token.token); | tmp = findSystemDictionary(token.token); | 
| 
| Line 786  void scanner() { |  | 
| Line 807  void scanner() { |  | 
| } | } | 
|  |  | 
| if (StartAString) { | if (StartAString) { | 
| token.kind = EXECUTABLE_STRING; | token.kind = EXECUTABLE_STRING;  token.tflag = 0; | 
| token.token = StartString; | token.token = StartString; | 
| executeToken(token);    /* execute startup commands */ | executeToken(token);    /* execute startup commands */ | 
| token.kind = ID; | token.kind = ID; token.tflag = 0; | 
| token.token = "exec"; | token.token = "exec"; | 
| token = lookupTokens(token); /* set hashing values */ | token = lookupTokens(token); /* set hashing values */ | 
| tmp = findSystemDictionary(token.token); | tmp = findSystemDictionary(token.token); | 
| 
| Line 815  void scanner() { |  | 
| Line 836  void scanner() { |  | 
| if (DebugStack >= 1) { | if (DebugStack >= 1) { | 
| fprintf(Fstack,"\nscanner> "); | fprintf(Fstack,"\nscanner> "); | 
| } | } | 
| KSexecuteString(" ctrlC-hook "); /* Execute User Defined functions. */ | if (!Calling_ctrlC_hook) { /* to avoid recursive call of ctrlC-hook. */ | 
| KSexecuteString(" (Computation is interrupted.) "); | Calling_ctrlC_hook = 1; RestrictedMode = 0; | 
|  | KSexecuteString(" ctrlC-hook "); /* Execute User Defined functions. */ | 
|  | RestrictedMode = RestrictedMode_saved; | 
|  | } | 
|  | Calling_ctrlC_hook = 0; | 
|  | KSexecuteString(" (Computation is interrupted.) "); /* move to ctrlC-hook? */ | 
|  | InSendmsg2 = 0; | 
|  | infixOn = 0; | 
| continue ; | continue ; | 
| } else {  } | } else {  } | 
| if (DebugStack >= 1) { printOperandStack(); } | if (DebugStack >= 1) { printOperandStack(); } | 
| token = getokenSM(GET); | token = getokenSM(GET); | 
| if ((tmp=executeToken(token)) < 0) break; | if ((status=executeToken(token)) < 0) break; | 
| /***if (tmp == 1) fprintf(stderr," --- exit --- \n");*/ | /***if (status == 1) fprintf(stderr," --- exit --- \n");*/ | 
|  | /* fprintf(stderr,"token.token=%s, status=%d, infixOn=%d\n",token.token,status,infixOn); */ | 
|  | if (status & STATUS_INFIX) { | 
|  | infixOn = 1;  infixToken = token; infixToken.tflag |= NO_DELAY; | 
|  | }else if (infixOn) { | 
|  | infixOn = 0; | 
|  | if ((status=executeToken(infixToken)) < 0) break; | 
|  | } | 
| } | } | 
| } | } | 
|  |  | 
|  |  | 
| extern int SGClock; | extern int SGClock; | 
| extern int UserCtrlC; | extern int UserCtrlC; | 
| extern int OXlock; | extern int OXlock; | 
|  | extern int RestrictedMode, RestrictedMode_saved; | 
|  |  | 
| signal(sig,SIG_IGN); | signal(sig,SIG_IGN); | 
| /* see 133p */ | /* see 133p */ | 
|  | RestrictedMode = RestrictedMode_saved; | 
| cancelAlarm(); | cancelAlarm(); | 
| if (sig == SIGALRM) { | if (sig == SIGALRM) { | 
| fprintf(stderr,"ctrlC by SIGALRM\n"); | fprintf(stderr,"ctrlC by SIGALRM\n"); | 
|  |  | 
| return; | return; | 
| } | } | 
| if (ErrorMessageMode != 1) { | if (ErrorMessageMode != 1) { | 
|  | (void *) traceShowStack(); | 
| fprintf(Fstack,"User interruption by ctrl-C. We are in the top-level.\n"); | fprintf(Fstack,"User interruption by ctrl-C. We are in the top-level.\n"); | 
| fprintf(Fstack,"Type in quit in order to exit sm1.\n"); | fprintf(Fstack,"Type in quit in order to exit sm1.\n"); | 
| } | } | 
|  | traceClearStack(); | 
| if (GotoP) { | if (GotoP) { | 
| fprintf(Fstack,"The interpreter was looking for the label <<%s>>. It is also aborted.\n",GotoLabel); | fprintf(Fstack,"The interpreter was looking for the label <<%s>>. It is also aborted.\n",GotoLabel); | 
| GotoP = 0; | GotoP = 0; | 
| 
| Line 894  int executeToken(token) |  | 
| Line 933  int executeToken(token) |  | 
| int primitive; | int primitive; | 
| int size; | int size; | 
| int status; | int status; | 
| struct tokens *tokenArray; |  | 
| int i,h0,h1; | int i,h0,h1; | 
| extern int WarningMessageMode; | extern int WarningMessageMode; | 
| extern int Strict; | extern int Strict; | 
|  | extern int InSendmsg2; | 
|  | extern int RestrictedMode, RestrictedMode_saved; | 
|  | int localRestrictedMode_saved; | 
|  |  | 
|  | localRestrictedMode_saved = 0; | 
| if (GotoP) { /* for goto */ | if (GotoP) { /* for goto */ | 
| if (token.kind == ID && isLiteral(token.token)) { | if (token.kind == ID && isLiteral(token.token)) { | 
| if (strcmp(&((token.token)[1]),GotoLabel) == 0) { | if (strcmp(&((token.token)[1]),GotoLabel) == 0) { | 
| 
| Line 944  int executeToken(token) |  | 
| Line 986  int executeToken(token) |  | 
| h1 = ((token.object.lc.op)->rc).ival; | h1 = ((token.object.lc.op)->rc).ival; | 
| ob=findUserDictionary(token.token,h0,h1,CurrentContextp); | ob=findUserDictionary(token.token,h0,h1,CurrentContextp); | 
| primitive = ((token.object.rc.op)->lc).ival; | primitive = ((token.object.rc.op)->lc).ival; | 
|  | if (!(token.tflag & NO_DELAY)) { | 
|  | if ((ob.tag >= 0) && (UD_attr & ATTR_INFIX)) { | 
|  | return STATUS_INFIX; | 
|  | } | 
|  | } | 
| if (ob.tag >= 0) { | if (ob.tag >= 0) { | 
| /* there is a definition in the user dictionary */ | /* there is a definition in the user dictionary */ | 
| if (ob.tag == SexecutableArray) { | if (ob.tag == SexecutableArray) { | 
| tokenArray = ob.lc.tokenArray; | if (RestrictedMode) { | 
| size = ob.rc.ival; | if (UD_attr & ATTR_EXPORT) { | 
| for (i=0; i<size; i++) { | localRestrictedMode_saved = RestrictedMode; RestrictedMode = 0; | 
| status = executeToken(tokenArray[i]); | }else{ | 
| if (status != 0) return(status); | tracePushName(token.token); | 
|  | errorStackmachine("You cannot execute this function in restricted mode.\n"); | 
|  | } | 
| } | } | 
|  |  | 
|  | status = executeExecutableArray(ob,token.token,0); | 
|  |  | 
|  | if (localRestrictedMode_saved) RestrictedMode = localRestrictedMode_saved; | 
|  | if ((status & STATUS_BREAK) || (status < 0)) return status; | 
| }else { | }else { | 
| Kpush(ob); | Kpush(ob); | 
| } | } | 
| } else if (primitive) { | } else if (primitive) { | 
|  | tracePushName(token.token); | 
| /* system operator */ | /* system operator */ | 
| ob.tag = Soperator; | ob.tag = Soperator; | 
| ob.lc.ival = primitive; | ob.lc.ival = primitive; | 
| return(executePrimitive(ob)); | status = executePrimitive(ob); | 
|  | tracePopName(); | 
|  | return(status); | 
| } else { | } else { | 
|  | if (QuoteMode) { | 
|  | if (InSendmsg2) return(DO_QUOTE); | 
|  | else { | 
|  | Kpush(KpoString(token.token)); | 
|  | return(0); /* normal exit.*/ | 
|  | } | 
|  | } | 
| if (WarningMessageMode == 1 || WarningMessageMode == 2) { | if (WarningMessageMode == 1 || WarningMessageMode == 2) { | 
| char tmpc[1024]; | char tmpc[1024]; | 
| if (strlen(token.token) < 900) { | if (strlen(token.token) < 900) { | 
| 
| Line 1009  errorStackmachine(str) |  | 
| Line 1073  errorStackmachine(str) |  | 
| char message0[1024]; | char message0[1024]; | 
| char *message; | char *message; | 
| extern int ErrorMessageMode; | extern int ErrorMessageMode; | 
|  | extern int RestrictedMode, RestrictedMode_saved; | 
|  | RestrictedMode = RestrictedMode_saved; | 
| cancelAlarm(); | cancelAlarm(); | 
| if (ErrorMessageMode == 1 || ErrorMessageMode == 2) { | if (ErrorMessageMode == 1 || ErrorMessageMode == 2) { | 
| pushErrorStack(KnewErrorPacket(SerialCurrent,-1,str)); | pushErrorStack(KnewErrorPacket(SerialCurrent,-1,str)); | 
| 
| Line 1043  errorStackmachine(str) |  | 
| Line 1109  errorStackmachine(str) |  | 
| fprintf(stderr,str); | fprintf(stderr,str); | 
| } | } | 
| fprintf(stderr,"\n"); | fprintf(stderr,"\n"); | 
|  | (void) traceShowStack(); | 
| } | } | 
|  | traceClearStack(); | 
| if (GotoP) { | if (GotoP) { | 
| fprintf(Fstack,"The interpreter was looking for the label <<%s>>. It is also aborted.\n",GotoLabel); | fprintf(Fstack,"The interpreter was looking for the label <<%s>>. It is also aborted.\n",GotoLabel); | 
| GotoP = 0; | GotoP = 0; | 
| 
| Line 1089  KSexecuteString(s) |  | 
| Line 1157  KSexecuteString(s) |  | 
| jmp_buf saved_EnvOfStackMachine; | jmp_buf saved_EnvOfStackMachine; | 
| void (*sigfunc)(); | void (*sigfunc)(); | 
| int localCatchCtrlC ; | int localCatchCtrlC ; | 
|  | extern int RestrictedMode, RestrictedMode_saved; | 
|  |  | 
| localCatchCtrlC = CatchCtrlC; | localCatchCtrlC = CatchCtrlC; | 
| /* If CatchCtrlC is rewrited in this program, | /* If CatchCtrlC is rewrited in this program, | 
| 
| Line 1114  KSexecuteString(s) |  | 
| Line 1183  KSexecuteString(s) |  | 
| } | } | 
| recursive--; | recursive--; | 
| if (localCatchCtrlC) { signal(SIGINT, sigfunc); } | if (localCatchCtrlC) { signal(SIGINT, sigfunc); } | 
| KSexecuteString(" ctrlC-hook "); /* Execute User Defined functions. */ | if (!Calling_ctrlC_hook) { | 
| KSexecuteString(" (Computation is interrupted.) "); | Calling_ctrlC_hook = 1; RestrictedMode = 0; | 
|  | KSexecuteString(" ctrlC-hook "); /* Execute User Defined functions. */ | 
|  | RestrictedMode_saved; | 
|  | } | 
|  | Calling_ctrlC_hook = 0; | 
|  | KSexecuteString(" (Computation is interrupted.) "); /* move to ctrlC-hook?*/ | 
| return(-1); | return(-1); | 
| }else{ } | }else{ } | 
| }else{ | }else{ | 
| 
| Line 1132  KSexecuteString(s) |  | 
| Line 1206  KSexecuteString(s) |  | 
| } | } | 
| recursive = 0; | recursive = 0; | 
| if (localCatchCtrlC) { signal(SIGINT, sigfunc); } | if (localCatchCtrlC) { signal(SIGINT, sigfunc); } | 
| KSexecuteString(" ctrlC-hook "); /* Execute User Defined functions. */ | if (!Calling_ctrlC_hook) { | 
|  | Calling_ctrlC_hook = 1; RestrictedMode = 0; | 
|  | KSexecuteString(" ctrlC-hook "); /* Execute User Defined functions. */ | 
|  | RestrictedMode = RestrictedMode_saved; | 
|  | } | 
|  | Calling_ctrlC_hook = 0; | 
|  | Calling_ctrlC_hook = 0; | 
| KSexecuteString(" (Computation is interrupted.) "); | KSexecuteString(" (Computation is interrupted.) "); | 
| return(-1); | return(-1); | 
| }else { } | }else { } | 
| 
| Line 1141  KSexecuteString(s) |  | 
| Line 1221  KSexecuteString(s) |  | 
|  |  | 
| recursive++; | recursive++; | 
| token.token = s; | token.token = s; | 
| token.kind = EXECUTABLE_STRING; | token.kind = EXECUTABLE_STRING; token.tflag = 0; | 
| executeToken(token); | executeToken(token); | 
| token.kind = ID; | token.kind = ID; token.tflag = 0; | 
| token.token = "exec"; | token.token = "exec"; | 
| token = lookupTokens(token); /* no use */ | token = lookupTokens(token); /* no use */ | 
| tmp = findSystemDictionary(token.token); | tmp = findSystemDictionary(token.token); | 
| 
| Line 1162  KSdefineMacros() { |  | 
| Line 1242  KSdefineMacros() { |  | 
| struct object ob; | struct object ob; | 
|  |  | 
| if (StandardMacros && (strlen(SMacros))) { | if (StandardMacros && (strlen(SMacros))) { | 
| token.kind = EXECUTABLE_STRING; | token.kind = EXECUTABLE_STRING; token.tflag = 0; | 
| token.token = SMacros; | token.token = SMacros; | 
| executeToken(token);    /* execute startup commands */ | executeToken(token);    /* execute startup commands */ | 
| token.kind = ID; | token.kind = ID; token.tflag = 0; | 
| token.token = "exec"; | token.token = "exec"; | 
| token = lookupTokens(token); /* no use */ | token = lookupTokens(token); /* no use */ | 
| tmp = findSystemDictionary(token.token); | tmp = findSystemDictionary(token.token); | 
| 
| Line 1195  void KSstart() { |  | 
| Line 1275  void KSstart() { |  | 
| } else {  }  */ | } else {  }  */ | 
|  |  | 
| /* setup quiet mode or not */ | /* setup quiet mode or not */ | 
| token.kind = EXECUTABLE_STRING; | token.kind = EXECUTABLE_STRING; token.tflag = 0; | 
| if (Quiet) { | if (Quiet) { | 
| token.token = " /@@@.quiet 1 def "; | token.token = " /@@@.quiet 1 def "; | 
| }else { | }else { | 
| token.token = " /@@@.quiet 0 def "; | token.token = " /@@@.quiet 0 def "; | 
| } | } | 
| executeToken(token); /* execute startup commands */ | executeToken(token); /* execute startup commands */ | 
| token.kind = ID; | token.kind = ID; token.tflag = 0; | 
| token.token = "exec"; | token.token = "exec"; | 
| token = lookupTokens(token); /* set hashing values */ | token = lookupTokens(token); /* set hashing values */ | 
| tmp = findSystemDictionary(token.token); | tmp = findSystemDictionary(token.token); | 
| 
| Line 1484  struct object KSdupErrors() { |  | 
| Line 1564  struct object KSdupErrors() { |  | 
| void cancelAlarm() { | void cancelAlarm() { | 
| alarm((unsigned int) 0); | alarm((unsigned int) 0); | 
| signal(SIGALRM,SIG_DFL); | signal(SIGALRM,SIG_DFL); | 
|  | } | 
|  |  | 
|  | /* back-trace */ | 
|  | #define TraceNameStackSize 3000 | 
|  | char *TraceNameStack[TraceNameStackSize]; | 
|  | int TraceNameStackp = 0; | 
|  | void tracePushName(char *s) { | 
|  | char *t; | 
|  | /* | 
|  | t = (char *)sGC_malloc(strlen(s)+1); | 
|  | if (t == NULL) { | 
|  | fprintf(stderr,"No more memory.\n"); return; | 
|  | } | 
|  | strcpy(t,s); | 
|  | */ | 
|  | t = s; | 
|  | TraceNameStack[TraceNameStackp++] = t; | 
|  | if (TraceNameStackp >= TraceNameStackSize) { | 
|  | fprintf(stderr,"Warning: TraceNameStack overflow. Clearing the stack.\n"); | 
|  | TraceNameStackp = 0; | 
|  | } | 
|  | } | 
|  | void traceClearStack(void) { | 
|  | TraceNameStackp = 0; | 
|  | } | 
|  | char *tracePopName(void) { | 
|  | if (TraceNameStackp <= 0) return (char *) NULL; | 
|  | return TraceNameStack[--TraceNameStackp]; | 
|  | } | 
|  | #define TRACE_MSG_SIZE 320 | 
|  | char *traceShowStack(void) { | 
|  | char *s; | 
|  | char *t; | 
|  | int p; | 
|  | s = (char *) sGC_malloc(TRACE_MSG_SIZE); | 
|  | if (s == NULL) { | 
|  | fprintf(stderr,"No more memory.\n"); return NULL; | 
|  | } | 
|  | sprintf(s,"Trace: "); | 
|  | p = strlen(s); | 
|  | do { | 
|  | t = tracePopName(); | 
|  | if (t == NULL) { | 
|  | s[p] = ';'; s[p+1] = 0; | 
|  | break; | 
|  | }else if ((strlen(t) + p) > (TRACE_MSG_SIZE-10)) { | 
|  | /* fprintf(stderr,"p=%d, TraceNameStackp=%d, strlen(t)=%d, t=%s\n",p,TraceNameStackp,strlen(t),t); */ | 
|  | strcpy(&(s[p])," ..."); | 
|  | break; | 
|  | } | 
|  | strcpy(&(s[p]),t); p += strlen(t); | 
|  | strcpy(&(s[p]),"<-"); p += 2; | 
|  | } while (t != (char *)NULL); | 
|  | fprintf(stderr,"%s\n",s); | 
|  | return s; | 
|  | } | 
|  |  | 
|  | /* | 
|  | if (fname != NULL) fname is pushed to the trace stack. | 
|  | */ | 
|  | int executeExecutableArray(struct object ob,char *fname,int withGotoP) { | 
|  | struct tokens *tokenArray; | 
|  | int size,i; | 
|  | int status; | 
|  | int infixOn; | 
|  | struct tokens infixToken; | 
|  | extern int GotoP; | 
|  |  | 
|  | infixOn = 0; | 
|  | if (ob.tag != SexecutableArray) errorStackmachine("Error (executeTokenArray): the argument is not a token array."); | 
|  |  | 
|  | if (fname != NULL) tracePushName(fname); | 
|  | tokenArray = ob.lc.tokenArray; | 
|  | size = ob.rc.ival; | 
|  | for (i=0; i<size; i++) { | 
|  | status = executeToken(tokenArray[i]); | 
|  | if ((status & STATUS_BREAK) || (status < 0) || (withGotoP && GotoP)) { | 
|  | if (fname != NULL) tracePopName(); | 
|  | return(status); | 
|  | } | 
|  |  | 
|  | if (status & STATUS_INFIX) { | 
|  | if (i == size-1) errorStackmachine("Infix operator at the end of an executable array."); | 
|  | infixOn = 1; infixToken = tokenArray[i]; | 
|  | infixToken.tflag |= NO_DELAY; | 
|  | continue; | 
|  | }else if (infixOn) { | 
|  | infixOn = 0; | 
|  | status = executeToken(infixToken); | 
|  | if ((status & STATUS_BREAK) || (status < 0) || (withGotoP && GotoP)) { | 
|  | if (fname != NULL) tracePopName(); | 
|  | return(status); | 
|  | } | 
|  | } | 
|  | } | 
|  | if (fname != NULL) tracePopName(); | 
|  | return(0); /* normal exit */ | 
| } | } |