URL
https://opencores.org/ocsvn/System09/System09/trunk
Subversion Repositories System09
[/] [System09/] [rev_86/] [Tools/] [as09/] [pseudo.c] - Rev 214
Go to most recent revision | Compare with Previous | Blame | View Log
/* * pseudo --- pseudo op processing */ #define RMB 0 /* Reserve Memory Bytes */ #define FCB 1 /* Form Constant Bytes */ #define FDB 2 /* Form Double Bytes (words) */ #define FCC 3 /* Form Constant Characters */ #define ORG 4 /* Origin */ #define EQU 5 /* Equate */ #define ZMB 6 /* Zero memory bytes */ #define FILL 7 /* block fill constant bytes */ #define OPT 8 /* assembler option */ #define NULL_OP 9 /* null pseudo op */ #define PAGE 10 /* new page */ #define INCLUDE 11 /* include <file> or "file" ver TER_2.0 */ #define END 12 /* include <file> terminator ver TER_2.0 */ #define IFD 13 /* if define <symbol> ver TER_2.0 */ #define IFND 14 /* if not define <symbol> ver TER_2.0 */ #define ELSE 15 /* else (for IF statements) ver TER_2.0 */ #define ENDIF 16 /* endif (for IF statements) ver TER_2.0 */ #define BSS 17 /* block storage segment (RAM) ver TER_2.09 */ #define CODE 18 /* code segment ver TER_2.09 25 Jul 89 */ #define DATA 19 /* data segment ver TER_2.09 25 Jul 89 */ #define AUTO 20 /* data segment ver TER_2.09 25 Jul 89 */ struct oper pseudo[] = { "=", PSEUDO, EQU, 0, /* ver TER_2.09 25 Jul 89 */ "auto", PSEUDO, AUTO, 0, /* ver TER_2.09 25 Jul 89 */ "bss", PSEUDO, BSS, 0, /* ver TER_2.09 25 Jul 89 */ "bsz", PSEUDO, ZMB, 0, "code", PSEUDO, CODE, 0, /* ver TER_2.09 25 Jul 89 */ "data", PSEUDO, DATA, 0, /* ver TER_2.09 25 Jul 89 */ "else", PSEUDO, ELSE, 0, /* ver TER_2.0 6/17/89 */ "end", PSEUDO, END, 0, /* ver TER_2.0 6/17/89 */ "endif",PSEUDO, ENDIF, 0, /* ver TER_2.0 6/17/89 */ "equ", PSEUDO, EQU, 0, "fcb", PSEUDO, FCB, 0, "fcc", PSEUDO, FCC, 0, "fdb", PSEUDO, FDB, 0, "fill", PSEUDO, FILL, 0, "ifd", PSEUDO, IFD, 0, /* ver TER_2.0 6/17/89 */ "ifnd", PSEUDO, IFND, 0, /* ver TER_2.0 6/17/89 */ "include", PSEUDO, INCLUDE, 0, /* ver TER_2.0 6/17/89 */ "nam", PSEUDO, NULL_OP,0, "name", PSEUDO, NULL_OP,0, "opt", PSEUDO, OPT, 0, "org", PSEUDO, ORG, 0, "pag", PSEUDO, PAGE, 0, "page", PSEUDO, PAGE, 0, "ram", PSEUDO, BSS, 0, /* ver TER_2.09 25 Jul 89 */ "rmb", PSEUDO, RMB, 0, "spc", PSEUDO, NULL_OP,0, "ttl", PSEUDO, NULL_OP,0, "zmb", PSEUDO, ZMB, 0 }; /* * do_pseudo --- do pseudo op processing */ do_pseudo(op) int op; /* which op */ { char fccdelim, *strsave(); int fill; int c; /*test variable ver TER_2.0 6/18/89 */ char *skip_white(), *savept; /* savept is pointer to string save */ FILE *FdTemp, *fopen(); /* ver TER_2.0 6/17/89 */ void IfMachine(); /* rel TER_2.0 6/18/89 */ void PC_Exchange(); /* ver TER_2.09 25 Jul 89 */ if( op != EQU && *Label ) install(Label,Pc); P_force++; #ifdef DEBUG3 printf("%s, line no. ",Argv[Cfn]); /* current file name */ printf("%d: ",Line_num); /* current line number */ printf(" Pseudo Op=%u\n",op); #endif switch(op){ case RMB: /* reserve memory bytes */ if( eval() ){ Pc += Result; f_record(); /* flush out bytes */ } else error("Undefined Operand during Pass One"); break; case ZMB: /* zero memory bytes */ if( eval() ) while( Result-- ) emit(0); else error("Undefined Operand during Pass One"); break; case FILL: /* fill memory with constant */ eval(); fill = Result; if( *Optr++ != ',' ) error("Bad fill"); else{ Optr = skip_white(Optr); eval(); while( Result-- ) emit(fill); } break; case FCB: /* form constant byte(s) */ do{ Optr = skip_white(Optr); eval(); if( Result > 0xFF ){ if(!Force_byte) warn("Value truncated"); Result = lobyte(Result); } emit(Result); }while( *Optr++ == ',' ); break; case FDB: /* form double byte(s) */ do{ Optr = skip_white(Optr); eval(); eword(Result); }while( *Optr++ == ',' ); break; case FCC: /* form constant characters */ if(*Operand==EOS) break; fccdelim = *Optr++; while( *Optr != EOS && *Optr != fccdelim) emit(*Optr++); if(*Optr == fccdelim) Optr++; else error("Missing Delimiter"); break; case ORG: /* origin */ if( eval() ){ Old_pc = Pc = Result; f_record(); /* flush out any bytes */ } else error("Undefined Operand during Pass One"); break; case EQU: /* equate */ if(*Label==EOS){ error("EQU requires label"); break; } if( eval() ){ install(Label,Result); Old_pc = Result; /* override normal */ } else error("Undefined Operand during Pass One"); break; case OPT: /* assembler option */ P_force=0; if( head(Operand,"l") ) Lflag=1; else if (head(Operand,"nol")) Lflag=0; else if (head(Operand,"c")){ Cflag=1; Ctotal=0; } else if (head(Operand,"noc")) Cflag=0; else if (head(Operand,"contc")){ Cflag=1; } else if (head(Operand,"s")) Sflag = 1; else if (head(Operand,"cre")) CREflag = 1; else if (head(Operand,"p50")){ /* turn on 50 lines/page flag */ Pflag50 = 1; Pflag75 = 0; } /* separately. ver TER_2.0 6/17/89 */ else if (head(Operand,"p75")){ /* turn on 75 lines/page flag */ Pflag50 = 0; Pflag75 = 1; } /* separately. ver TER_2.0 6/17/89 */ else if (head(Operand,"crlf")) /* add <CR> <LF> to */ CRflag = 1; /* S record ver TER_2.08 */ else if (head(Operand,"nnf")) /* no extra line no. */ nfFlag = 0; /* w include files ver TER_2.08 */ else error("Unrecognized OPT"); break; case PAGE: /* go to a new page */ P_force=0; N_page = 1; if (Pass == 2 ) if (Lflag) NewPage(); break; case NULL_OP: /* ignored psuedo ops */ P_force=0; break; case INCLUDE: /* case INCLUDE added ver TER_2.0 6/17/89 */ P_force=0; /* no PC in printed output */ if ((c=FNameGet(InclFName))==0) error("Improper INCLUDE statement"); else { if (FdCount > MAXINCFILES) error("too many INCLUDE files"); else { if ((FdTemp = fopen(InclFName,"r"))==0) { printf("%s, line no. ",Argv[Cfn]); /* current file name */ printf("%d: ",Line_num); /* current line number */ Page_lines++; /* increment lines per page */ printf("warning:can't open INCLUDE file %s\n",InclFName); } else { if ((savept=strsave(InclFName))==0) error("out of memory for INCLUDE file name"); else { InclFiles[FdCount].fp=Fd; /* save current fp */ if (nfFlag) { InclFiles[FdCount].line_num=Line_num; /* save current line count */ Line_num=0; /* reset for new file */ } InclFiles[FdCount].name=Argv[Cfn]; /* save pointer to current name */ Argv[Cfn]=savept; /* now replace pointer to current name with pointer to name of Include file */ Fd=FdTemp; /* now replace current file with INCLUDE fp */ FdCount++; /* and increment "stack" */ #ifdef DEBUG2 printf("pseudo INCLUDE: FdCount=%d\n",FdCount); printf(" new input file pointer=%d\n",Fd); printf(" new file name=%s\n",Argv[Cfn]); #endif } } } } break; case END: P_force=0; if (FdCount>0) /* skip END statements in files received from CLI arguments */ { fclose(Fd); /* close file from this level nest */ FdCount--; /* "pop stack" */ Fd=InclFiles[FdCount].fp; /* restore fp from nested stack */ if(nfFlag) Line_num=InclFiles[FdCount].line_num; Argv[Cfn]=InclFiles[FdCount].name; /* restore file name pointer */ #ifdef DEBUG2 printf("pseudo END: FdCount=%d\n",FdCount); printf(" new input file pointer=%d\n",Fd); printf(" new file name=%s\n",Argv[Cfn]); #endif } break; case IFD: #ifdef DEBUG3 printf("case IFD: in pseudo\n"); #endif P_force=0; c=eval_ifd(); IfMachine(c); break; case IFND: P_force=0; c=eval_ifnd(); IfMachine(c); break; case ELSE: P_force=0; IfMachine(IF_ELSE); break; case ENDIF: P_force=0; IfMachine(IF_ENDIF); break; case CODE: /* CODE,DATA,BSS,AUTO ver TER_2.09 */ PC_Exchange(0); break; case DATA: PC_Exchange(1); break; case BSS: PC_Exchange(2); break; case AUTO: PC_Exchange(3); break; default: fatal("Pseudo error"); break; } } /* * PC_Exchange --- Save current PC and recover requested one added ver TER_2.09 */ void PC_Exchange(PC_ptr_new) int PC_ptr_new; /* request 0=CODE,1=DATA,2=BSS */ { P_force = 0; /* no PC in output cuz wrong first time (?) */ PC_Save[PC_ptr] = Pc; /* save current PC */ PC_ptr = PC_ptr_new; /* remember which one we're using */ Old_pc = Pc = PC_Save[PC_ptr]; /* recover the one requested */ f_record(); /* flush out any bytes, this is really an ORG */ }
Go to most recent revision | Compare with Previous | Blame | View Log