URL
https://opencores.org/ocsvn/rf6809/rf6809/trunk
Subversion Repositories rf6809
[/] [rf6809/] [trunk/] [software/] [a09/] [a09.c] - Rev 16
Go to most recent revision | Compare with Previous | Blame | View Log
/* A09, 6809 Assembler. (C) Copyright 1993,1994 L.C. Benschop. Parts (C) Copyright 2001-2010 H. Seib. Mods for 12-bit FPGA core 6809 by R. Finch This version of the program is distributed under the terms and conditions of the GNU General Public License version 2. See file the COPYING for details. THERE IS NO WARRANTY ON THIS PROGRAM. Generates binary image file from the lowest to the highest address with actually assembled data. Machine dependencies: char is 8 bits. short is 16 bits. integer arithmetic is twos complement. syntax: a09 [-{b|r|s|x|f}filename]|[-c] [-lfilename] [-ooption] [-dsym=value]* sourcefile. Options -Vfilename Verilog rommem declaration output -c suppresses code output -bfilename binary output file name (default name minus a09 suffix plus .bin) -rfilename Flex9 RELASMB-compatible output file name (relocatable) -sfilename s-record output file name (default its a binary file) -xfilename intel hex output file name (-"-) -ffilename Flex9 output file name (-"-) -lfilename list file name (default no listing) -dsym[=value] define a symbol -oopt defines an option recognized pseudoops: ext defines an external symbol (relocating mode only) extern -"- public defines a public symbol (effective in relocating mode only) macro defines a macro exitm terminates a macro expansion endm ends a macro definition if ifn ifc ifnc else endif dup endd org equ set setdp sets direct page in 6809 / 6309 mode text fcb fcw fdb fcc rmb reg end include title nam ttl sttl setpg pag spc rep rpt repeat opt define an option (see below for possible options) err abs force absolute mode common endcom def define enddef name symlen bin, binary includes a binary file at the current position recognized options: On | Off Meaning ------------------------------------------------------------ PAG | NOP* Page Formatting CON | NOC* Print skipped conditional code MAC*| NOM Print macro calling line EXP | NOE* Print macro expansion lines SYM*| NOS Print symbol table MUL*| NMU Print multiple object code lines LIS*| NOL Print assembler listing LP1 | NO1* Print Pass 1 Listing DAT*| NOD Print date in listing NUM | NON* Print line numbers INV | NOI* Print invisible lines TSC | NOT* Strict TSC Compatibility WAR*| NOW Print warnings CLL*| NCL Check line length LFN | NLF* Print long file names LLL*| NLL List library lines GAS | NOG* Gnu AS source style compatibility REL*| NOR Print Relocation table in Relocating mode M68*| H63 | M00 MC6809 / HD6309 / MC6800 mode TXT | NTX* Print text table LPA | LNP* Listing in f9dasm patch format * denotes default value v0.1 93/11/03 Initial version. v0.2 94/03/21 Fixed PC relative addressing bug Added SET, SETDP, INCLUDE. IF/ELSE/ENDIF No macros yet, and no separate linkable modules. v0.1.X 99/12/20 added Intel hex option (SLB) -- H.Seib Additions: -- v0.2.S 01/10/13 converted to readable format v0.3 01/10/15 added transfer address processing increased max. #symbols and other constants (hey, this isn't CP/M anymore :-) added a bit of intelligence to command line processing loads file(s) into internal storage prior to processing (this aids a lot in preventing recursive invocations) added LIB,LIBRARY pseudoop (same as INCLUDE) added OPT,OPTION pseudoop v0.4 01/10/22 added -t switch to increase compatibility with the TSC FLEX Assembler line, which uses a slightly more rigid source file format v0.5 01/10/23 added SETPG, SETLI, PAG, SPC, NAM(TTL,TITLE), STTL, REP(RPT,REPEAT) v0.6 01/10/23 added EXITM, IFN, IFC, IFNC, DUP, ENDD v0.7 01/10/29 added REG, ERR, TEXT v0.8 01/10/30 converted command line handling to -O switch for options and added a bunch of options RZB pseudoop added v0.9 01/11/05 clean-ups, increased TSC FLEX Assembler compatibility v0.10 01/11/15 ASM and VERSION texts predefined v0.11 01/11/27 increased TSC FLEX Assembler compatibility (labels made case sensitive) v0.12 01/11/28 added some convenience mnemonics and simulated 6800 instructions to increase TSC FLEX Assembler compatibility v1.02 01/04/11 check for address range overlaps implemented v1.03 02/09/24 macro parser had an error; corrected v1.04 02/12/10 error message output adjusted for Visual C++ 6.0 IDE compatibility added OPT LFN/NLF for that purpose v1.05 02/12/23 crude form of backwards ORG added to binary output v1.06 03/06/10 added OPT LLL/NLL to include/suppress listing of LIB lines v1.07 03/06/23 outsymtable() only starts with a page feed if current line is is NOT the first line on a page v1.08 03/07/01 SKIP count for the IF(N) and IF(N)C statements added to normal statements (i.e., outside macro definitions) v1.09 05/06/02 some cleanups added -r format for FLEX9 RELASMB compatible output added ABS, GLOBAL, EXT, DEFINE, ENDDEF, COMMON, ENDCOM REL|NOR option added GAS|NOG option added (rudimentary) M68|H63 option added v1.10 05/06/03 made options available as numeric strings added local label support added SYMLEN TXT|NTX option added v1.11 05/06/06 IFD/IFND directives added added special checks for "0,-[-]indexreg" and "0,indexreg+[+]" when determining the postbyte for indexed addressing modes v1.12 05/06/20 OPT CON|NOC correctly implemented v1.13 skipped for users that might have an uneasy feeling about the number "13" >:-) v1.14 05/08/01 text symbol replacement enhanced; it was a bit TOO general. Now, &xxx is only treated as a text variable when xxx REALLY is a text symbol; in all other cases, it's not preprocessed. Let the assembler engine handle eventual errors. Thanks to Peter from Australia for pointing this out! v1.15 05/08/02 only 1 local label was allowed per address; now, as many as possible can be at the same location. Thanks to Peter from Australia for pointing this out! v1.16 05/08/05 SETDP is illegal in relocating mode! SETDP without any parameter DISABLES Direct mode entirely. Cured an interesting bug... the following sequence: ORG 1 LDA TEST SHIT NOP ORG 1 TEST RMB 1 caused a phase error; this was caused by the invalid assumption in scanlabel() that forward references have a higher memory location than the current address. Thanks to Peter from Australia for pointing this out! v1.17 08/08/05 made tests for above problem more rigorous; now ANY forward reference to a label that's not yet defined is treated as uncertain. Removed a nasty bug that caused the following code to produce incorrect values: org $8000 (anything >= $8000 will do) lbl rmb 8 SizeBad equ (*-lbl)/2 OPT NOL changed to OPT NO1 to allow implementing OPT LIS|NOL. OPT LIS*|NOL added Thanks to Peter from Australia for pointing these out! v1.18 09/08/05 backward search for local labels now searches for "<= local address" instead of "< local address" Thanks to Peter from Australia for pointing this out! Added INCD / DECD convenience mnemonics (realized as ADDD/SUBD 1) v1.19 10/08/05 Added a bunch of 6800-style mnemonics; if they conflict with 6809 mnemonics ("INC A" for example), they are only active in TSC mode v1.20 16/08/05 changed special checks for "0,-[-]indexreg" and "0,indexreg+[+]" to include all known labels and changed the scope to ",[-[-]]indexreg[+[+]]" v1.21 21/02/06 Bug in "xxx >0,reg" found by Margus Kliimask v1.22 03/09/08 Addded BIN(ARY) pseudo-op to include binary files v1.23 13/02/09 6800 code generation added accept multiple input files (first one defines default output and listing file names) v1.24 13/02/09 made compilable with gcc v1.25 05/03/09 6800 alternate mnemonics work better now v1.26 14/03/09 assembling DOS format files in Loonix works better now v1.27 20/01/10 LPA/NLP options added v1.28 21/04/10 INCD/DECD produced invalid code */ #include <stdio.h> #include <string.h> #include <ctype.h> #include <malloc.h> #include <stdarg.h> #include <stdlib.h> #include <time.h> /*****************************************************************************/ /* Definitions */ /*****************************************************************************/ #define VERSION "1.30" #define VERSNUM "$011D" /* can be queried as &VERSION */ #define UNIX 0 /* set to != 0 for UNIX specials */ #define MAX_PASSNO 9 #define MAXLABELS 8192 #define MAXMACROS 1024 #define MAXTEXTS 1024 #define MAXRELOCS 32768 #define MAXIDLEN 32 #define MAXLISTBYTES 7 #define FNLEN 256 #define LINELEN 1024 /*****************************************************************************/ /* Line buffer definitions */ /*****************************************************************************/ struct linebuf { struct linebuf * next; /* pointer to next line */ struct linebuf * prev; /* pointer to previous line */ char *fn; /* pointer to original file name */ long ln; /* line number therein */ unsigned char lvl; /* line level */ unsigned char rel; /* relocation mode */ char txt[1]; /* text buffer */ }; char *fnms[128] = {0}; /* we process up to 128 different fls*/ short nfnms; /* # loaded files */ /* special bits in lvl : */ #define LINCAT_MACDEF 0x20 /* macro definition */ #define LINCAT_MACEXP 0x40 /* macro expansion */ #define LINCAT_INVISIBLE 0x80 /* does not appear in listing */ #define LINCAT_LVLMASK 0x1F /* mask for line levels (0..31) */ struct linebuf *rootline = NULL; /* pointer to 1st line of the file */ struct linebuf *curline = NULL; /* pointer to currently processed ln */ /*****************************************************************************/ /* Opcode definitions */ /*****************************************************************************/ struct oprecord { char * name; /* opcode mnemonic */ unsigned char cat; /* opcode category */ unsigned __int64 code; /* category-dependent additional code*/ unsigned char tgtNdx; }; /* Instruction categories : */ #define OPCAT_ONEBYTE 0x00 /* one byte opcodes NOP */ #define OPCAT_TWOBYTE 0x01 /* two byte opcodes SWI2 */ #define OPCAT_THREEBYTE 0x02 /* three byte opcodes TAB */ #define OPCAT_FOURBYTE 0x03 /* four byte opcodes ABA */ #define OPCAT_IMMBYTE 0x04 /* opcodes w. imm byte ANDCC */ #define OPCAT_LEA 0x05 /* load effective address LEAX */ #define OPCAT_SBRANCH 0x06 /* short branches BGE */ #define OPCAT_LBR2BYTE 0x07 /* long branches 2 byte opc LBGE */ #define OPCAT_LBR1BYTE 0x08 /* long branches 2 byte opc LBRA */ #define OPCAT_ARITH 0x09 /* accumulator instr. ADDA */ #define OPCAT_DBLREG1BYTE 0x0a /* double reg instr 1 byte opc LDX */ #define OPCAT_DBLREG2BYTE 0x0b /* double reg instr 2 byte opc LDY */ #define OPCAT_SINGLEADDR 0x0c /* single address instrs NEG */ #define OPCAT_2REG 0x0d /* 2 register instr TFR,EXG */ #define OPCAT_STACK 0x0e /* stack instr PSHx,PULx */ #define OPCAT_BITDIRECT 0x0f /* direct bitmanipulation AIM */ #define OPCAT_BITTRANS 0x10 /* direct bit transfer BAND */ #define OPCAT_BLOCKTRANS 0x11 /* block transfer TFM */ #define OPCAT_IREG 0x12 /* inter-register operations ADCR */ #define OPCAT_QUADREG1BYTE 0x13 /* quad reg instr 1 byte opc LDQ */ #define OPCAT_2IMMBYTE 0x14 /* 2byte opcode w. imm byte BITMD */ #define OPCAT_2ARITH 0x15 /* 2byte opcode accum. instr. SUBE */ #define OPCAT_ACCARITH 0x16 /* acc. instr. w.explicit acc ADD */ #define OPCAT_IDXEXT 0x17 /* indexed/extended, 6800-style JMP */ #define OPCAT_ACCADDR 0x18 /* single address instrs, 6800 NEG */ #define OPCAT_PSEUDO 0x3f /* pseudo-ops */ #define OPCAT_6309 0x40 /* valid for 6309 only! */ #define OPCAT_NOIMM 0x80 /* immediate not allowed! STD */ /* the various Pseudo-Ops */ #define PSEUDO_RMB 0 #define PSEUDO_ELSE 1 #define PSEUDO_END 2 #define PSEUDO_ENDIF 3 #define PSEUDO_ENDM 4 #define PSEUDO_EQU 5 #define PSEUDO_EXT 6 #define PSEUDO_FCB 7 #define PSEUDO_FCC 8 #define PSEUDO_FCW 9 #define PSEUDO_IF 10 #define PSEUDO_MACRO 11 #define PSEUDO_ORG 12 #define PSEUDO_PUB 13 #define PSEUDO_SETDP 14 #define PSEUDO_SET 15 #define PSEUDO_INCLUDE 16 #define PSEUDO_OPT 17 #define PSEUDO_NAM 18 #define PSEUDO_STTL 19 #define PSEUDO_PAG 20 #define PSEUDO_SPC 21 #define PSEUDO_REP 22 #define PSEUDO_SETPG 23 #define PSEUDO_SETLI 24 #define PSEUDO_EXITM 25 #define PSEUDO_IFN 26 #define PSEUDO_IFC 27 #define PSEUDO_IFNC 28 #define PSEUDO_DUP 29 #define PSEUDO_ENDD 30 #define PSEUDO_REG 31 #define PSEUDO_ERR 32 #define PSEUDO_TEXT 33 #define PSEUDO_RZB 34 #define PSEUDO_ABS 35 #define PSEUDO_DEF 36 #define PSEUDO_ENDDEF 37 #define PSEUDO_COMMON 38 #define PSEUDO_ENDCOM 39 #define PSEUDO_NAME 40 #define PSEUDO_SYMLEN 41 #define PSEUDO_IFD 42 #define PSEUDO_IFND 43 #define PSEUDO_BINARY 44 #define PSEUDO_FCDW 45 struct oprecord optable09[]= { { "ABA", OPCAT_FOURBYTE, 0x0340040ab0e0LL, 0 }, { "ABS", OPCAT_PSEUDO, PSEUDO_ABS, 0 }, { "ABX", OPCAT_ONEBYTE, 0x3a, 0 }, { "ABY", OPCAT_TWOBYTE, 0x0310a5, 0 }, { "ADC", OPCAT_ACCARITH, 0x89, 0 }, { "ADCA", OPCAT_ARITH, 0x89, 0 }, { "ADCB", OPCAT_ARITH, 0xc9, 0 }, { "ADCD", OPCAT_6309 | OPCAT_DBLREG1BYTE, 0x189, 0 }, { "ADCR", OPCAT_6309 | OPCAT_IREG, 0x1031, 0 }, { "ADD", OPCAT_ACCARITH, 0x8b, 0 }, { "ADDA", OPCAT_ARITH, 0x8b, 0 }, { "ADDB", OPCAT_ARITH, 0xcb, 0 }, { "ADDD", OPCAT_DBLREG1BYTE, 0xc3, 0 }, { "ADDE", OPCAT_6309 | OPCAT_2ARITH, 0x118b, 0 }, { "ADDF", OPCAT_6309 | OPCAT_2ARITH, 0x11cb, 0 }, { "ADDR", OPCAT_6309 | OPCAT_IREG, 0x1030, 0 }, { "ADDW", OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x108b, 0 }, { "AIM", OPCAT_6309 | OPCAT_BITDIRECT, 0x02, 0 }, { "AND", OPCAT_ACCARITH, 0x84, 0 }, { "ANDA", OPCAT_ARITH, 0x84, 0 }, { "ANDB", OPCAT_ARITH, 0xc4, 0 }, { "ANDCC", OPCAT_IMMBYTE, 0x1c, 0 }, { "ANDD", OPCAT_6309 | OPCAT_DBLREG1BYTE, 0x184, 0 }, { "ANDR", OPCAT_6309 | OPCAT_IREG, 0x1034, 0 }, { "ASL", OPCAT_SINGLEADDR, 0x08, 0 }, { "ASLA", OPCAT_ONEBYTE, 0x48, 0 }, { "ASLB", OPCAT_ONEBYTE, 0x58, 0 }, { "ASLD", OPCAT_TWOBYTE, 0x058049, 0 }, { "ASLD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x1048, 0 }, { "ASR", OPCAT_SINGLEADDR, 0x07, 0 }, { "ASRA", OPCAT_ONEBYTE, 0x47, 0 }, { "ASRB", OPCAT_ONEBYTE, 0x57, 0 }, { "ASRD", OPCAT_TWOBYTE, 0x047056, 0 }, { "ASRD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x1047, 0 }, { "BAND", OPCAT_6309 | OPCAT_BITTRANS, 0x1130, 0 }, { "BCC", OPCAT_SBRANCH, 0x24, 0 }, { "BCS", OPCAT_SBRANCH, 0x25, 0 }, { "BEC", OPCAT_SBRANCH, 0x24, 0 }, { "BEOR", OPCAT_6309 | OPCAT_BITTRANS, 0x1134, 0 }, { "BEQ", OPCAT_SBRANCH, 0x27, 0 }, { "BES", OPCAT_SBRANCH, 0x25, 0 }, { "BGE", OPCAT_SBRANCH, 0x2c, 0 }, { "BGT", OPCAT_SBRANCH, 0x2e, 0 }, { "BHI", OPCAT_SBRANCH, 0x22, 0 }, { "BHS", OPCAT_SBRANCH, 0x24, 0 }, { "BIAND", OPCAT_6309 | OPCAT_BITTRANS, 0x1131, 0 }, { "BIEOR", OPCAT_6309 | OPCAT_BITTRANS, 0x1135, 0 }, { "BIN", OPCAT_PSEUDO, PSEUDO_BINARY, 0 }, { "BINARY", OPCAT_PSEUDO, PSEUDO_BINARY, 0 }, { "BIOR", OPCAT_6309 | OPCAT_BITTRANS, 0x1133, 0 }, { "BIT", OPCAT_ACCARITH, 0x85, 0 }, { "BITA", OPCAT_ARITH, 0x85, 0 }, { "BITB", OPCAT_ARITH, 0xc5, 0 }, { "BITD", OPCAT_6309 | OPCAT_DBLREG1BYTE, 0x185, 0 }, { "BITMD", OPCAT_6309 | OPCAT_2IMMBYTE, 0x113c, 0 }, { "BLE", OPCAT_SBRANCH, 0x2f, 0 }, { "BLO", OPCAT_SBRANCH, 0x25, 0 }, { "BLS", OPCAT_SBRANCH, 0x23, 0 }, { "BLT", OPCAT_SBRANCH, 0x2d, 0 }, { "BMI", OPCAT_SBRANCH, 0x2b, 0 }, { "BNE", OPCAT_SBRANCH, 0x26, 0 }, { "BOR", OPCAT_6309 | OPCAT_BITTRANS, 0x1132, 0 }, { "BPL", OPCAT_SBRANCH, 0x2a, 0 }, { "BRA", OPCAT_SBRANCH, 0x20, 0 }, { "BRN", OPCAT_SBRANCH, 0x21, 0 }, { "BSR", OPCAT_SBRANCH, 0x8d, 0 }, { "BVC", OPCAT_SBRANCH, 0x28, 0 }, { "BVS", OPCAT_SBRANCH, 0x29, 0 }, { "CBA", OPCAT_FOURBYTE, 0x0340040a10e0LL, 0 }, { "CLC", OPCAT_TWOBYTE, 0x1c0fe, 0 }, { "CLF", OPCAT_TWOBYTE, 0x1c0bf, 0 }, { "CLI", OPCAT_TWOBYTE, 0x1c0ef, 0 }, { "CLIF", OPCAT_TWOBYTE, 0x1c0af, 0 }, { "CLR", OPCAT_SINGLEADDR, 0x0f, 0 }, { "CLRA", OPCAT_ONEBYTE, 0x4f, 0 }, { "CLRB", OPCAT_ONEBYTE, 0x5f, 0 }, { "CLRD", OPCAT_TWOBYTE, 0x4f05f, 0 }, { "CLRD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x104f, 0 }, { "CLRE", OPCAT_6309 | OPCAT_TWOBYTE, 0x114f, 0 }, { "CLRF", OPCAT_6309 | OPCAT_TWOBYTE, 0x115f, 0 }, { "CLRW", OPCAT_6309 | OPCAT_TWOBYTE, 0x105f, 0 }, { "CLV", OPCAT_TWOBYTE, 0x1c0fd, 0 }, { "CLZ", OPCAT_TWOBYTE, 0x1c0fb, 0 }, { "CMP", OPCAT_ACCARITH, 0x81, 0 }, { "CMPA", OPCAT_ARITH, 0x81, 0 }, { "CMPB", OPCAT_ARITH, 0xc1, 0 }, { "CMPD", OPCAT_DBLREG1BYTE, 0x183, 0 }, { "CMPE", OPCAT_6309 | OPCAT_2ARITH, 0x1181, 0 }, { "CMPF", OPCAT_6309 | OPCAT_2ARITH, 0x11c1, 0 }, { "CMPR", OPCAT_6309 | OPCAT_IREG, 0x1037, 0 }, { "CMPS", OPCAT_DBLREG1BYTE, 0x28c, 4 }, { "CMPU", OPCAT_DBLREG1BYTE, 0x283, 3 }, { "CMPW", OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x1081, 0 }, { "CMPX", OPCAT_DBLREG1BYTE, 0x8c, 1 }, { "CMPY", OPCAT_DBLREG1BYTE, 0x18c, 2 }, { "COM", OPCAT_SINGLEADDR, 0x03, 0 }, { "COMA", OPCAT_ONEBYTE, 0x43, 0 }, { "COMB", OPCAT_ONEBYTE, 0x53, 0 }, { "COMD", OPCAT_6309 | OPCAT_ONEBYTE, 0x143, 0 }, { "COME", OPCAT_6309 | OPCAT_TWOBYTE, 0x1143, 0 }, { "COMF", OPCAT_6309 | OPCAT_TWOBYTE, 0x1153, 0 }, { "COMW", OPCAT_6309 | OPCAT_TWOBYTE, 0x1053, 0 }, { "COMMON", OPCAT_PSEUDO, PSEUDO_COMMON, 0 }, { "CPD", OPCAT_DBLREG1BYTE, 0x183, 0 }, { "CPX", OPCAT_DBLREG1BYTE, 0x8c, 1 }, { "CPY", OPCAT_DBLREG1BYTE, 0x18c, 2 }, { "CWAI", OPCAT_IMMBYTE, 0x3c, 0 }, { "DAA", OPCAT_ONEBYTE, 0x19, 0 }, { "DEC", OPCAT_SINGLEADDR, 0x0a, 0 }, { "DECA", OPCAT_ONEBYTE, 0x4a, 0 }, { "DECB", OPCAT_ONEBYTE, 0x5a, 0 }, { "DECD", OPCAT_THREEBYTE, 0x083000001LL, 0 }, { "DECD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x104a, 0 }, { "DECE", OPCAT_6309 | OPCAT_TWOBYTE, 0x114a, 0 }, { "DECF", OPCAT_6309 | OPCAT_TWOBYTE, 0x115a, 0 }, { "DECW", OPCAT_6309 | OPCAT_TWOBYTE, 0x105a, 0 }, { "DEF", OPCAT_PSEUDO, PSEUDO_DEF, 0 }, { "DEFINE", OPCAT_PSEUDO, PSEUDO_DEF, 0 }, { "DES", OPCAT_TWOBYTE, 0x327ff, 4 }, { "DEU", OPCAT_TWOBYTE, 0x335ff, 3 }, { "DEX", OPCAT_TWOBYTE, 0x301ff, 1 }, { "DEY", OPCAT_TWOBYTE, 0x313ff, 2 }, { "DIVD", OPCAT_6309 | OPCAT_2ARITH, 0x118d, 0 }, { "DIVQ", OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x118e, 0 }, { "DUP", OPCAT_PSEUDO, PSEUDO_DUP, 0 }, { "EIM", OPCAT_6309 | OPCAT_BITDIRECT, 0x05, 0 }, { "ELSE", OPCAT_PSEUDO, PSEUDO_ELSE, 0 }, { "END", OPCAT_PSEUDO, PSEUDO_END, 0 }, { "ENDCOM", OPCAT_PSEUDO, PSEUDO_ENDCOM, 0 }, { "ENDD", OPCAT_PSEUDO, PSEUDO_ENDD, 0 }, { "ENDDEF", OPCAT_PSEUDO, PSEUDO_ENDDEF, 0 }, { "ENDIF", OPCAT_PSEUDO, PSEUDO_ENDIF, 0 }, { "ENDM", OPCAT_PSEUDO, PSEUDO_ENDM, 0 }, { "EOR", OPCAT_ACCARITH, 0x88, 0 }, { "EORA", OPCAT_ARITH, 0x88, 0 }, { "EORB", OPCAT_ARITH, 0xc8, 0 }, { "EORD", OPCAT_6309 | OPCAT_DBLREG1BYTE, 0x188, 0 }, { "EORR", OPCAT_6309 | OPCAT_IREG, 0x1036, 0 }, { "EQU", OPCAT_PSEUDO, PSEUDO_EQU, 0 }, { "ERR", OPCAT_PSEUDO, PSEUDO_ERR, 0 }, { "EXG", OPCAT_2REG, 0x1e, 0 }, { "EXITM", OPCAT_PSEUDO, PSEUDO_EXITM, 0 }, { "EXT", OPCAT_PSEUDO, PSEUDO_EXT, 0 }, { "EXTERN", OPCAT_PSEUDO, PSEUDO_EXT, 0 }, { "FCB", OPCAT_PSEUDO, PSEUDO_FCB, 0 }, { "FCC", OPCAT_PSEUDO, PSEUDO_FCC, 0 }, { "FCDW", OPCAT_PSEUDO, PSEUDO_FCDW, 0 }, { "FCW", OPCAT_PSEUDO, PSEUDO_FCW, 0 }, { "FDB", OPCAT_PSEUDO, PSEUDO_FCW, 0 }, { "GLOBAL", OPCAT_PSEUDO, PSEUDO_PUB, 0 }, { "IF", OPCAT_PSEUDO, PSEUDO_IF, 0 }, { "IFC", OPCAT_PSEUDO, PSEUDO_IFC, 0 }, { "IFD", OPCAT_PSEUDO, PSEUDO_IFD, 0 }, { "IFN", OPCAT_PSEUDO, PSEUDO_IFN, 0 }, { "IFNC", OPCAT_PSEUDO, PSEUDO_IFNC, 0 }, { "IFND", OPCAT_PSEUDO, PSEUDO_IFND, 0 }, { "INC", OPCAT_SINGLEADDR, 0x0c, 0 }, { "INCA", OPCAT_ONEBYTE, 0x4c, 0 }, { "INCB", OPCAT_ONEBYTE, 0x5c, 0 }, { "INCD", OPCAT_THREEBYTE, 0x0c3000001LL, 0 }, { "INCD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x104c, 0 }, { "INCE", OPCAT_6309 | OPCAT_TWOBYTE, 0x114c, 0 }, { "INCF", OPCAT_6309 | OPCAT_TWOBYTE, 0x115c, 0 }, { "INCLUDE", OPCAT_PSEUDO, PSEUDO_INCLUDE, 0 }, { "INCW", OPCAT_6309 | OPCAT_TWOBYTE, 0x105c, 0 }, { "INS", OPCAT_TWOBYTE, 0x32601, 4 }, { "INU", OPCAT_TWOBYTE, 0x33401, 3 }, { "INX", OPCAT_TWOBYTE, 0x30001, 1 }, { "INY", OPCAT_TWOBYTE, 0x31201, 2 }, { "JMP", OPCAT_SINGLEADDR, 0x0e, 0 }, { "JSR", OPCAT_DBLREG1BYTE, 0x8d, 0 }, { "LBCC", OPCAT_LBR1BYTE, 0x124, 0 }, { "LBCS", OPCAT_LBR1BYTE, 0x125, 0 }, { "LBEC", OPCAT_LBR1BYTE, 0x124, 0 }, { "LBEQ", OPCAT_LBR1BYTE, 0x127, 0 }, { "LBES", OPCAT_LBR1BYTE, 0x125, 0 }, { "LBGE", OPCAT_LBR1BYTE, 0x12c, 0 }, { "LBGT", OPCAT_LBR1BYTE, 0x12e, 0 }, { "LBHI", OPCAT_LBR1BYTE, 0x122, 0 }, { "LBHS", OPCAT_LBR1BYTE, 0x124, 0 }, { "LBLE", OPCAT_LBR1BYTE, 0x12f, 0 }, { "LBLO", OPCAT_LBR1BYTE, 0x125, 0 }, { "LBLS", OPCAT_LBR1BYTE, 0x123, 0 }, { "LBLT", OPCAT_LBR1BYTE, 0x12d, 0 }, { "LBMI", OPCAT_LBR1BYTE, 0x12b, 0 }, { "LBNE", OPCAT_LBR1BYTE, 0x126, 0 }, { "LBPL", OPCAT_LBR1BYTE, 0x12a, 0 }, { "LBRA", OPCAT_LBR1BYTE, 0x16, 0 }, { "LBRN", OPCAT_LBR1BYTE, 0x121, 0 }, { "LBSR", OPCAT_LBR1BYTE, 0x17, 0 }, { "LBVC", OPCAT_LBR1BYTE, 0x128, 0 }, { "LBVS", OPCAT_LBR1BYTE, 0x129, 0 }, { "LD", OPCAT_ACCARITH, 0x86, 0 }, { "LDA", OPCAT_ARITH, 0x86, 0 }, { "LDAA", OPCAT_ARITH, 0x86, 0 }, { "LDAB", OPCAT_ARITH, 0xc6, 0 }, { "LDAD", OPCAT_DBLREG1BYTE, 0xcc, 0 }, { "LDB", OPCAT_ARITH, 0xc6, 0 }, { "LDBT", OPCAT_6309 | OPCAT_BITTRANS, 0x1136, 0 }, { "LDD", OPCAT_DBLREG1BYTE, 0xcc, 0 }, { "LDE", OPCAT_6309 | OPCAT_2ARITH, 0x1186, 0 }, { "LDF", OPCAT_6309 | OPCAT_2ARITH, 0x11c6, 0 }, { "LDMD", OPCAT_6309 | OPCAT_2IMMBYTE, 0x113d, 0 }, { "LDQ", OPCAT_6309 | OPCAT_QUADREG1BYTE,0x10cc, 0 }, { "LDS", OPCAT_DBLREG1BYTE, 0x1ce, 4 }, { "LDU", OPCAT_DBLREG1BYTE, 0xce, 3 }, { "LDW", OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x1086, 0 }, { "LDX", OPCAT_DBLREG1BYTE, 0x8e, 1 }, { "LDY", OPCAT_DBLREG1BYTE, 0x18e, 2 }, { "LEAS", OPCAT_LEA, 0x32, 4 }, { "LEAU", OPCAT_LEA, 0x33, 3 }, { "LEAX", OPCAT_LEA, 0x30, 1 }, { "LEAY", OPCAT_LEA, 0x31, 2 }, { "LIB", OPCAT_PSEUDO, PSEUDO_INCLUDE, 0 }, { "LIBRARY", OPCAT_PSEUDO, PSEUDO_INCLUDE, 0 }, { "LSL", OPCAT_SINGLEADDR, 0x08, 0 }, { "LSLA", OPCAT_ONEBYTE, 0x48, 0 }, { "LSLB", OPCAT_ONEBYTE, 0x58, 0 }, { "LSLD", OPCAT_TWOBYTE, 0x58049, 0 }, { "LSLD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x1048, 0 }, { "LSR", OPCAT_SINGLEADDR, 0x04, 0 }, { "LSRA", OPCAT_ONEBYTE, 0x44, 0 }, { "LSRB", OPCAT_ONEBYTE, 0x54, 0 }, { "LSRD", OPCAT_TWOBYTE, 0x44056, 0 }, { "LSRD63", OPCAT_6309 | OPCAT_TWOBYTE, 0x1044, 0 }, { "LSRW", OPCAT_6309 | OPCAT_TWOBYTE, 0x1054, 0 }, { "MACRO", OPCAT_PSEUDO, PSEUDO_MACRO, 0 }, { "MUL", OPCAT_ONEBYTE, 0x3d, 0 }, { "MULD", OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x118f, 0 }, { "NAM", OPCAT_PSEUDO, PSEUDO_NAM, 0 }, { "NAME", OPCAT_PSEUDO, PSEUDO_NAME, 0 }, { "NEG", OPCAT_SINGLEADDR, 0x00, 0 }, { "NEGA", OPCAT_ONEBYTE, 0x40, 0 }, { "NEGB", OPCAT_ONEBYTE, 0x50, 0 }, { "NEGD", OPCAT_6309 | OPCAT_ONEBYTE, 0x140, 0 }, { "NOP", OPCAT_ONEBYTE, 0x12, 0 }, { "OIM", OPCAT_6309 | OPCAT_BITDIRECT, 0x01, 0 }, { "OPT", OPCAT_PSEUDO, PSEUDO_OPT, 0 }, { "OPTION", OPCAT_PSEUDO, PSEUDO_OPT, 0 }, { "ORA", OPCAT_ARITH, 0x8a, 0 }, { "ORAA", OPCAT_ARITH, 0x8a, 0 }, { "ORAB", OPCAT_ARITH, 0xca, 0 }, { "ORB", OPCAT_ARITH, 0xca, 0 }, { "ORCC", OPCAT_IMMBYTE, 0x1a, 0 }, { "ORD", OPCAT_6309 | OPCAT_DBLREG1BYTE, 0x18a, 0 }, { "ORG", OPCAT_PSEUDO, PSEUDO_ORG, 0 }, { "ORR", OPCAT_6309 | OPCAT_IREG, 0x1035, 0 }, { "PAG", OPCAT_PSEUDO, PSEUDO_PAG, 0 }, { "PAGE", OPCAT_PSEUDO, PSEUDO_PAG, 0 }, { "PSH", OPCAT_STACK, 0x34, 0 }, { "PSHA", OPCAT_TWOBYTE, 0x34002, 0 }, { "PSHB", OPCAT_TWOBYTE, 0x34004, 0 }, { "PSHD", OPCAT_TWOBYTE, 0x34006, 0 }, { "PSHS", OPCAT_STACK, 0x34, 4 }, { "PSHSW", OPCAT_6309 | OPCAT_TWOBYTE, 0x1038, 0 }, { "PSHU", OPCAT_STACK, 0x36, 3 }, { "PSHUW", OPCAT_6309 | OPCAT_TWOBYTE, 0x103a, 0 }, { "PSHX", OPCAT_TWOBYTE, 0x34010, 1 }, { "PSHY", OPCAT_TWOBYTE, 0x34020, 2 }, { "PUB", OPCAT_PSEUDO, PSEUDO_PUB, 0 }, { "PUBLIC", OPCAT_PSEUDO, PSEUDO_PUB, 0 }, { "PUL", OPCAT_STACK, 0x35, 0 }, { "PULA", OPCAT_TWOBYTE, 0x35002, 0 }, { "PULB", OPCAT_TWOBYTE, 0x35004, 0 }, { "PULD", OPCAT_TWOBYTE, 0x35006, 0 }, { "PULS", OPCAT_STACK, 0x35, 4 }, { "PULSW", OPCAT_6309 | OPCAT_TWOBYTE, 0x1039, 0 }, { "PULU", OPCAT_STACK, 0x37, 3 }, { "PULUW", OPCAT_6309 | OPCAT_TWOBYTE, 0x103b, 0 }, { "PULX", OPCAT_TWOBYTE, 0x35010, 1 }, { "PULY", OPCAT_TWOBYTE, 0x35020, 2 }, { "REG", OPCAT_PSEUDO, PSEUDO_REG, 0 }, { "REP", OPCAT_PSEUDO, PSEUDO_REP, 0 }, { "REPEAT", OPCAT_PSEUDO, PSEUDO_REP, 0 }, { "RESET", OPCAT_ONEBYTE, 0x3e, 0 }, { "RMB", OPCAT_PSEUDO, PSEUDO_RMB, 0 }, { "ROL", OPCAT_SINGLEADDR, 0x09, 0 }, { "ROLA", OPCAT_ONEBYTE, 0x49, 0 }, { "ROLB", OPCAT_ONEBYTE, 0x59, 0 }, { "ROLD", OPCAT_6309 | OPCAT_ONEBYTE, 0x149, 0 }, { "ROLW", OPCAT_6309 | OPCAT_TWOBYTE, 0x1059, 0 }, { "ROR", OPCAT_SINGLEADDR, 0x06, 0 }, { "RORA", OPCAT_ONEBYTE, 0x46, 0 }, { "RORB", OPCAT_ONEBYTE, 0x56, 0 }, { "RORD", OPCAT_6309 | OPCAT_ONEBYTE, 0x146, 0 }, { "RORW", OPCAT_6309 | OPCAT_TWOBYTE, 0x1056, 0 }, { "RPT", OPCAT_PSEUDO, PSEUDO_REP, 0 }, { "RTF", OPCAT_ONEBYTE, 0x38, 0 }, { "RTI", OPCAT_ONEBYTE, 0x3b, 0 }, { "RTS", OPCAT_ONEBYTE, 0x39, 0 }, { "RZB", OPCAT_PSEUDO, PSEUDO_RZB, 0 }, { "SBA", OPCAT_FOURBYTE, 0x0340040a00e0LL, 0 }, { "SBC", OPCAT_ACCARITH, 0x82, 0 }, { "SBCA", OPCAT_ARITH, 0x82, 0 }, { "SBCB", OPCAT_ARITH, 0xc2, 0 }, { "SBCD", OPCAT_6309 | OPCAT_DBLREG1BYTE, 0x182, 0 }, { "SBCR", OPCAT_6309 | OPCAT_IREG, 0x1033, 0 }, { "SEC", OPCAT_TWOBYTE, 0x1a001, 0 }, { "SEF", OPCAT_TWOBYTE, 0x1a040, 0 }, { "SEI", OPCAT_TWOBYTE, 0x1a010, 0 }, { "SEIF", OPCAT_TWOBYTE, 0x1a050, 0 }, { "SET", OPCAT_PSEUDO, PSEUDO_SET, 0 }, { "SETDP", OPCAT_PSEUDO, PSEUDO_SETDP, 0 }, { "SETLI", OPCAT_PSEUDO, PSEUDO_SETLI, 0 }, { "SETPG", OPCAT_PSEUDO, PSEUDO_SETPG, 0 }, { "SEV", OPCAT_TWOBYTE, 0x1a002, 0 }, { "SEX", OPCAT_ONEBYTE, 0x1d, 0 }, { "SEXW", OPCAT_6309 | OPCAT_ONEBYTE, 0x14, 0 }, { "SEZ", OPCAT_TWOBYTE, 0x1a004, 0 }, { "SPC", OPCAT_PSEUDO, PSEUDO_SPC, 0 }, { "STA", OPCAT_NOIMM | OPCAT_ARITH, 0x87, 0 }, { "STAA", OPCAT_NOIMM | OPCAT_ARITH, 0x87, 0 }, { "STAB", OPCAT_NOIMM | OPCAT_ARITH, 0xc7, 0 }, { "STAD", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0xcd, 0 }, { "STB", OPCAT_NOIMM | OPCAT_ARITH, 0xc7, 0 }, { "STBT", OPCAT_6309 | OPCAT_BITTRANS, 0x1137, 0 }, { "STD", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0xcd, 0 }, { "STE", OPCAT_NOIMM | OPCAT_6309 | OPCAT_2ARITH, 0x1187, 0 }, { "STF", OPCAT_NOIMM | OPCAT_6309 | OPCAT_2ARITH, 0x11c7, 0 }, { "STQ", OPCAT_NOIMM | OPCAT_6309 | OPCAT_QUADREG1BYTE,0x10cd, 0 }, { "STS", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0x1cf, 4 }, { "STTL", OPCAT_PSEUDO, PSEUDO_STTL, 0 }, { "STU", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0xcf, 3 }, { "STW", OPCAT_NOIMM | OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x1087, 0 }, { "STX", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0x8f, 1 }, { "STY", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0x18f, 2 }, { "SUB", OPCAT_ACCARITH, 0x80, 0 }, { "SUBA", OPCAT_ARITH, 0x80, 0 }, { "SUBB", OPCAT_ARITH, 0xc0, 0 }, { "SUBD", OPCAT_DBLREG1BYTE, 0x83, 0 }, { "SUBE", OPCAT_6309 | OPCAT_2ARITH, 0x1180, 0 }, { "SUBF", OPCAT_6309 | OPCAT_2ARITH, 0x11c0, 0 }, { "SUBW", OPCAT_6309 | OPCAT_DBLREG2BYTE, 0x1080, 0 }, { "SUBR", OPCAT_6309 | OPCAT_IREG, 0x1032, 0 }, { "SWI", OPCAT_ONEBYTE, 0x3f, 0 }, { "SWI2", OPCAT_ONEBYTE, 0x13f, 0 }, { "SWI3", OPCAT_ONEBYTE, 0x23f, 0 }, { "SYMLEN", OPCAT_PSEUDO, PSEUDO_SYMLEN, 0 }, { "SYNC", OPCAT_ONEBYTE, 0x13, 0 }, { "TAB", OPCAT_THREEBYTE, 0x1f08904d, 0 }, { "TAP", OPCAT_TWOBYTE, 0x1f08a, 0 }, { "TBA", OPCAT_THREEBYTE, 0x1f09804dLL, 0 }, { "TEXT", OPCAT_PSEUDO, PSEUDO_TEXT, 0 }, { "TFM", OPCAT_6309 | OPCAT_BLOCKTRANS, 0x1138, 0 }, { "TFR", OPCAT_2REG, 0x1f, 0 }, { "TIM", OPCAT_6309 | OPCAT_BITDIRECT, 0x0b, 0 }, { "TITLE", OPCAT_PSEUDO, PSEUDO_NAM, 0 }, { "TPA", OPCAT_TWOBYTE, 0x1f0a8, 0 }, { "TST", OPCAT_SINGLEADDR, 0x0d, 0 }, { "TSTA", OPCAT_ONEBYTE, 0x4d, 0 }, { "TSTB", OPCAT_ONEBYTE, 0x5d, 0 }, { "TSTD", OPCAT_6309 | OPCAT_ONEBYTE, 0x14d, 0 }, { "TSTE", OPCAT_6309 | OPCAT_TWOBYTE, 0x114d, 0 }, { "TSTF", OPCAT_6309 | OPCAT_TWOBYTE, 0x115d, 0 }, { "TSTW", OPCAT_6309 | OPCAT_TWOBYTE, 0x105d, 0 }, { "TSX", OPCAT_TWOBYTE, 0x1f041, 1 }, { "TSY", OPCAT_FOURBYTE, 0x034040035020LL, 2 }, /* PSHS S/PULS Y */ { "TTL", OPCAT_PSEUDO, PSEUDO_NAM, 0 }, { "TXS", OPCAT_TWOBYTE, 0x1f014, 4 }, { "TYS", OPCAT_FOURBYTE, 0x034020035040LL, 4 }, /* PSHS Y/PULS S */ { "WAI", OPCAT_TWOBYTE, 0x3c0ff, 0 }, { "WORD", OPCAT_PSEUDO, PSEUDO_FCW, 0 }, }; struct oprecord optable00[]= { { "ABA", OPCAT_ONEBYTE, 0x1b }, { "ABS", OPCAT_PSEUDO, PSEUDO_ABS }, { "ADC", OPCAT_ACCARITH, 0x89 }, { "ADCA", OPCAT_ARITH, 0x89 }, { "ADCB", OPCAT_ARITH, 0xc9 }, { "ADD", OPCAT_ACCARITH, 0x8b }, { "ADDA", OPCAT_ARITH, 0x8b }, { "ADDB", OPCAT_ARITH, 0xcb }, { "AND", OPCAT_ACCARITH, 0x84 }, { "ANDA", OPCAT_ARITH, 0x84 }, { "ANDB", OPCAT_ARITH, 0xc4 }, { "ASL", OPCAT_ACCADDR, 0x08 }, { "ASLA", OPCAT_ONEBYTE, 0x48 }, { "ASLB", OPCAT_ONEBYTE, 0x58 }, { "ASR", OPCAT_ACCADDR, 0x07 }, { "ASRA", OPCAT_ONEBYTE, 0x47 }, { "ASRB", OPCAT_ONEBYTE, 0x57 }, { "BCC", OPCAT_SBRANCH, 0x24 }, { "BCS", OPCAT_SBRANCH, 0x25 }, { "BEC", OPCAT_SBRANCH, 0x24 }, { "BEQ", OPCAT_SBRANCH, 0x27 }, { "BES", OPCAT_SBRANCH, 0x25 }, { "BGE", OPCAT_SBRANCH, 0x2c }, { "BGT", OPCAT_SBRANCH, 0x2e }, { "BHI", OPCAT_SBRANCH, 0x22 }, { "BHS", OPCAT_SBRANCH, 0x24 }, { "BIN", OPCAT_PSEUDO, PSEUDO_BINARY }, { "BINARY", OPCAT_PSEUDO, PSEUDO_BINARY }, { "BIT", OPCAT_ACCARITH, 0x85 }, { "BITA", OPCAT_ARITH, 0x85 }, { "BITB", OPCAT_ARITH, 0xc5 }, { "BLE", OPCAT_SBRANCH, 0x2f }, { "BLO", OPCAT_SBRANCH, 0x25 }, { "BLS", OPCAT_SBRANCH, 0x23 }, { "BLT", OPCAT_SBRANCH, 0x2d }, { "BMI", OPCAT_SBRANCH, 0x2b }, { "BNE", OPCAT_SBRANCH, 0x26 }, { "BPL", OPCAT_SBRANCH, 0x2a }, { "BRA", OPCAT_SBRANCH, 0x20 }, { "BSR", OPCAT_SBRANCH, 0x8d }, { "BVC", OPCAT_SBRANCH, 0x28 }, { "BVS", OPCAT_SBRANCH, 0x29 }, { "CBA", OPCAT_ONEBYTE, 0x11 }, { "CLC", OPCAT_ONEBYTE, 0x0c }, { "CLI", OPCAT_ONEBYTE, 0x0e }, { "CLR", OPCAT_ACCADDR, 0x0f }, { "CLRA", OPCAT_ONEBYTE, 0x4f }, { "CLRB", OPCAT_ONEBYTE, 0x5f }, { "CLV", OPCAT_ONEBYTE, 0x0a }, { "CMP", OPCAT_ACCARITH, 0x81 }, { "CMPA", OPCAT_ARITH, 0x81 }, { "CMPB", OPCAT_ARITH, 0xc1 }, { "COM", OPCAT_ACCADDR, 0x03 }, { "COMA", OPCAT_ONEBYTE, 0x43 }, { "COMB", OPCAT_ONEBYTE, 0x53 }, { "CPX", OPCAT_DBLREG1BYTE, 0x8c }, { "DAA", OPCAT_ONEBYTE, 0x19 }, { "DEC", OPCAT_ACCADDR, 0x0a }, { "DECA", OPCAT_ONEBYTE, 0x4a }, { "DECB", OPCAT_ONEBYTE, 0x5a }, { "DEF", OPCAT_PSEUDO, PSEUDO_DEF }, { "DEFINE", OPCAT_PSEUDO, PSEUDO_DEF }, { "DES", OPCAT_ONEBYTE, 0x34 }, { "DEX", OPCAT_ONEBYTE, 0x09 }, { "DUP", OPCAT_PSEUDO, PSEUDO_DUP }, { "ELSE", OPCAT_PSEUDO, PSEUDO_ELSE }, { "END", OPCAT_PSEUDO, PSEUDO_END }, { "ENDCOM", OPCAT_PSEUDO, PSEUDO_ENDCOM }, { "ENDD", OPCAT_PSEUDO, PSEUDO_ENDD }, { "ENDDEF", OPCAT_PSEUDO, PSEUDO_ENDDEF }, { "ENDIF", OPCAT_PSEUDO, PSEUDO_ENDIF }, { "ENDM", OPCAT_PSEUDO, PSEUDO_ENDM }, { "EOR", OPCAT_ACCARITH, 0x88 }, { "EORA", OPCAT_ARITH, 0x88 }, { "EORB", OPCAT_ARITH, 0xc8 }, { "EQU", OPCAT_PSEUDO, PSEUDO_EQU }, { "ERR", OPCAT_PSEUDO, PSEUDO_ERR }, { "EXITM", OPCAT_PSEUDO, PSEUDO_EXITM }, { "EXT", OPCAT_PSEUDO, PSEUDO_EXT }, { "EXTERN", OPCAT_PSEUDO, PSEUDO_EXT }, { "FCB", OPCAT_PSEUDO, PSEUDO_FCB }, { "FCC", OPCAT_PSEUDO, PSEUDO_FCC }, { "FCW", OPCAT_PSEUDO, PSEUDO_FCW }, { "FDB", OPCAT_PSEUDO, PSEUDO_FCW }, { "GLOBAL", OPCAT_PSEUDO, PSEUDO_PUB }, { "IF", OPCAT_PSEUDO, PSEUDO_IF }, { "IFC", OPCAT_PSEUDO, PSEUDO_IFC }, { "IFD", OPCAT_PSEUDO, PSEUDO_IFD }, { "IFN", OPCAT_PSEUDO, PSEUDO_IFN }, { "IFNC", OPCAT_PSEUDO, PSEUDO_IFNC }, { "IFND", OPCAT_PSEUDO, PSEUDO_IFND }, { "INC", OPCAT_ACCADDR, 0x0c }, { "INCA", OPCAT_ONEBYTE, 0x4c }, { "INCB", OPCAT_ONEBYTE, 0x5c }, { "INCLUDE", OPCAT_PSEUDO, PSEUDO_INCLUDE }, { "INS", OPCAT_ONEBYTE, 0x31 }, { "INX", OPCAT_ONEBYTE, 0x08 }, { "JMP", OPCAT_IDXEXT, 0x4e }, { "JSR", OPCAT_IDXEXT, 0x8d }, { "LDA", OPCAT_ACCARITH, 0x86 }, { "LDAA", OPCAT_ARITH, 0x86 }, { "LDAB", OPCAT_ARITH, 0xc6 }, { "LDB", OPCAT_ARITH, 0xc6 }, { "LDS", OPCAT_DBLREG1BYTE, 0x8e }, { "LDX", OPCAT_DBLREG1BYTE, 0xce }, { "LIB", OPCAT_PSEUDO, PSEUDO_INCLUDE }, { "LIBRARY", OPCAT_PSEUDO, PSEUDO_INCLUDE }, { "LSL", OPCAT_ACCADDR, 0x08 }, { "LSLA", OPCAT_ONEBYTE, 0x48 }, { "LSLB", OPCAT_ONEBYTE, 0x58 }, { "LSR", OPCAT_ACCADDR, 0x04 }, { "LSRA", OPCAT_ONEBYTE, 0x44 }, { "LSRB", OPCAT_ONEBYTE, 0x54 }, { "MACRO", OPCAT_PSEUDO, PSEUDO_MACRO }, { "NAM", OPCAT_PSEUDO, PSEUDO_NAM }, { "NAME", OPCAT_PSEUDO, PSEUDO_NAME }, { "NEG", OPCAT_ACCADDR, 0x00 }, { "NEGA", OPCAT_ONEBYTE, 0x40 }, { "NEGB", OPCAT_ONEBYTE, 0x50 }, { "NOP", OPCAT_ONEBYTE, 0x01 }, { "OPT", OPCAT_PSEUDO, PSEUDO_OPT }, { "OPTION", OPCAT_PSEUDO, PSEUDO_OPT }, { "ORA", OPCAT_ARITH, 0x8a }, { "ORAA", OPCAT_ARITH, 0x8a }, { "ORAB", OPCAT_ARITH, 0xca }, { "ORB", OPCAT_ARITH, 0xca }, { "ORG", OPCAT_PSEUDO, PSEUDO_ORG }, { "PAG", OPCAT_PSEUDO, PSEUDO_PAG }, { "PAGE", OPCAT_PSEUDO, PSEUDO_PAG }, { "PSHA", OPCAT_ONEBYTE, 0x36 }, { "PSHB", OPCAT_ONEBYTE, 0x37 }, { "PUB", OPCAT_PSEUDO, PSEUDO_PUB }, { "PUBLIC", OPCAT_PSEUDO, PSEUDO_PUB }, { "PULA", OPCAT_ONEBYTE, 0x32 }, { "PULB", OPCAT_ONEBYTE, 0x33 }, { "REG", OPCAT_PSEUDO, PSEUDO_REG }, { "REP", OPCAT_PSEUDO, PSEUDO_REP }, { "REPEAT", OPCAT_PSEUDO, PSEUDO_REP }, { "RMB", OPCAT_PSEUDO, PSEUDO_RMB }, { "ROL", OPCAT_ACCADDR, 0x09 }, { "ROLA", OPCAT_ONEBYTE, 0x49 }, { "ROLB", OPCAT_ONEBYTE, 0x59 }, { "ROR", OPCAT_ACCADDR, 0x06 }, { "RORA", OPCAT_ONEBYTE, 0x46 }, { "RORB", OPCAT_ONEBYTE, 0x56 }, { "RPT", OPCAT_PSEUDO, PSEUDO_REP }, { "RTI", OPCAT_ONEBYTE, 0x3b }, { "RTS", OPCAT_ONEBYTE, 0x39 }, { "RZB", OPCAT_PSEUDO, PSEUDO_RZB }, { "SBA", OPCAT_ONEBYTE, 0x10 }, { "SBC", OPCAT_ACCARITH, 0x82 }, { "SBCA", OPCAT_ARITH, 0x82 }, { "SBCB", OPCAT_ARITH, 0xc2 }, { "SEC", OPCAT_ONEBYTE, 0x0d }, { "SEI", OPCAT_ONEBYTE, 0x0f }, { "SET", OPCAT_PSEUDO, PSEUDO_SET }, //{ "SETDP", OPCAT_PSEUDO, PSEUDO_SETDP }, { "SETLI", OPCAT_PSEUDO, PSEUDO_SETLI }, { "SETPG", OPCAT_PSEUDO, PSEUDO_SETPG }, { "SEV", OPCAT_ONEBYTE, 0x0b }, { "SPC", OPCAT_PSEUDO, PSEUDO_SPC }, { "STA", OPCAT_NOIMM | OPCAT_ACCARITH, 0x87 }, { "STAA", OPCAT_NOIMM | OPCAT_ARITH, 0x87 }, { "STAB", OPCAT_NOIMM | OPCAT_ARITH, 0xc7 }, { "STS", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0x8f }, { "STTL", OPCAT_PSEUDO, PSEUDO_STTL }, { "STX", OPCAT_NOIMM | OPCAT_DBLREG1BYTE, 0xcf }, { "SUB", OPCAT_ACCARITH, 0x80 }, { "SUBA", OPCAT_ARITH, 0x80 }, { "SUBB", OPCAT_ARITH, 0xc0 }, { "SWI", OPCAT_ONEBYTE, 0x3f }, { "SYMLEN", OPCAT_PSEUDO, PSEUDO_SYMLEN }, { "TAB", OPCAT_ONEBYTE, 0x16 }, { "TAP", OPCAT_ONEBYTE, 0x06 }, { "TBA", OPCAT_ONEBYTE, 0x17 }, { "TEXT", OPCAT_PSEUDO, PSEUDO_TEXT }, { "TITLE", OPCAT_PSEUDO, PSEUDO_NAM }, { "TPA", OPCAT_ONEBYTE, 0x07 }, { "TST", OPCAT_ACCADDR, 0x0d }, { "TSTA", OPCAT_ONEBYTE, 0x4d }, { "TSTB", OPCAT_ONEBYTE, 0x5d }, { "TSX", OPCAT_ONEBYTE, 0x30 }, { "TTL", OPCAT_PSEUDO, PSEUDO_NAM }, { "TXS", OPCAT_ONEBYTE, 0x35 }, { "WAI", OPCAT_ONEBYTE, 0x3e }, }; /* expression categories... all zeros is ordinary constant. bit 1 indicates address within module. bit 2 indicates external address. bit 4 indicates this can't be relocated if it's an address. bit 5 indicates address (if any) is negative. */ #define EXPRCAT_INTADDR 0x02 #define EXPRCAT_EXTADDR 0x04 #define EXPRCAT_PUBLIC 0x08 #define EXPRCAT_FIXED 0x10 #define EXPRCAT_NEGATIVE 0x20 /*****************************************************************************/ /* Symbol definitions */ /*****************************************************************************/ struct symrecord { char name[MAXIDLEN + 1]; /* symbol name */ char cat; /* symbol category */ char isFar; /* far assumed */ char isFarkw; /* FAR keyword used in symbol definiton */ unsigned value; /* symbol value */ union { struct symrecord *parent; /* parent symbol (for COMMON) */ long flags; /* forward reference flag (otherwise)*/ } u; }; /* symbol categories : */ #define SYMCAT_CONSTANT 0x00 /* constant value (from equ) */ #define SYMCAT_VARIABLE 0x01 /* variable value (from set) */ #define SYMCAT_LABEL 0x02 /* address within module (label) */ #define SYMCAT_VARADDR 0x03 /* variable containing address */ #define SYMCAT_EXTERN 0x04 /* address in other module (extern) */ #define SYMCAT_VAREXTERN 0x05 /* variable containing extern addr. */ #define SYMCAT_UNRESOLVED 0x06 /* unresolved address */ #define SYMCAT_VARUNRESOLVED 0x07 /* var. containing unresolved addr */ #define SYMCAT_PUBLIC 0x08 /* public label */ #define SYMCAT_MACRO 0x09 /* macro definition */ #define SYMCAT_PUBLICUNDEF 0x0A /* public label (yet undefined) */ #define SYMCAT_PARMNAME 0x0B /* parameter name */ #define SYMCAT_EMPTY 0x0D /* empty */ #define SYMCAT_REG 0x0E /* REG directive */ #define SYMCAT_TEXT 0x0F /* /Dsymbol or TEXT label */ #define SYMCAT_COMMONDATA 0x12 /* COMMON Data */ #define SYMCAT_COMMON 0x14 /* COMMON block */ #define SYMCAT_LOCALLABEL 0x22 /* local label */ #define SYMCAT_EMPTYLOCAL 0x26 /* empty local label */ /* symbol flags: */ #define SYMFLAG_FORWARD 0x01 /* forward reference */ #define SYMFLAG_PASSED 0x02 /* passed forward reference */ long symcounter = 0; /* # currently loaded symbols */ struct symrecord symtable[MAXLABELS]; /* symbol table (fixed size) */ long lclcounter = 0; /* # currently loaded local symbols */ struct symrecord lcltable[MAXLABELS]; /* local symbol table (fixed size) */ /*****************************************************************************/ /* regrecord structure definition */ /*****************************************************************************/ struct regrecord { char *name; /* name of the register */ unsigned char tfr, psh; /* bit value for tfr and psh/pul */ }; /*****************************************************************************/ /* regtable : table of all registers */ /*****************************************************************************/ struct regrecord regtable09[]= { { "D", 0x00, 0x06 }, { "X", 0x01, 0x10 }, { "Y", 0x02, 0x20 }, { "U", 0x03, 0x40 }, { "S", 0x04, 0x40 }, { "PC", 0x05, 0x80 }, { "A", 0x08, 0x02 }, { "B", 0x09, 0x04 }, { "CC", 0x0a, 0x01 }, { "CCR", 0x0a, 0x01 }, { "DP", 0x0b, 0x08 }, { "DPR", 0x0b, 0x08 }, { 0, 0, 0 } }; struct regrecord regtable63[]= /* same for HD6309 */ { { "D", 0x00, 0x06 }, { "X", 0x01, 0x10 }, { "Y", 0x02, 0x20 }, { "U", 0x03, 0x40 }, { "S", 0x04, 0x40 }, { "PC", 0x05, 0x80 }, { "W", 0x06, 0x00 }, { "V", 0x07, 0x00 }, { "A", 0x08, 0x02 }, { "B", 0x09, 0x04 }, { "CC", 0x0a, 0x01 }, { "CCR", 0x0a, 0x01 }, { "DP", 0x0b, 0x08 }, { "DPR", 0x0b, 0x08 }, { "0", 0x0c, 0x00 }, { "E", 0x0e, 0x00 }, { "F", 0x0f, 0x00 }, { 0, 0, 0 } }; struct regrecord regtable00[]= { { "X", 0x01, 0x10 }, { "S", 0x04, 0x40 }, { "PC", 0x05, 0x80 }, { "A", 0x08, 0x02 }, { "B", 0x09, 0x04 }, { "CC", 0x0a, 0x01 }, { "CCR", 0x0a, 0x01 }, { 0, 0, 0 } }; /*****************************************************************************/ /* bitregtable : table of all bit transfer registers */ /*****************************************************************************/ struct regrecord bitregtable09[] = { { "CC", 0x00, 0x00 }, { "A", 0x01, 0x01 }, { "B", 0x02, 0x02 }, }; struct regrecord bitregtable00[] = { { "CC", 0x00, 0x00 }, { "A", 0x01, 0x01 }, { "B", 0x02, 0x02 }, }; /*****************************************************************************/ /* relocrecord structure definition */ /*****************************************************************************/ struct relocrecord { unsigned addr; /* address that needs relocation */ char exprcat; /* expression category */ struct symrecord *sym; /* symbol for relocation */ }; long relcounter = 0; /* # currently defined relocations */ struct relocrecord reltable[MAXRELOCS]; /* relocation table (fixed size) */ long relhdrfoff; /* FLEX Relocatable Global Hdr Offset*/ /*****************************************************************************/ /* Options definitions */ /*****************************************************************************/ #define OPTION_M09 0x00000001L /* MC6809 mode */ #define OPTION_H63 0x00000002L /* HD6309 mode */ #define OPTION_M00 0x00000004L /* MC6800 mode */ #define OPTION_PAG 0x00000008L /* page formatting ON */ #define OPTION_CON 0x00000010L /* print cond. skipped code */ #define OPTION_MAC 0x00000020L /* print macro calling line (default)*/ #define OPTION_EXP 0x00000040L /* print macro expansion lines */ #define OPTION_SYM 0x00000080L /* print symbol table (default) */ #define OPTION_MUL 0x00000100L /* print multiple oc lines (default) */ #define OPTION_LP1 0x00000200L /* print pass 1 listing */ #define OPTION_DAT 0x00000400L /* print date in listing (default) */ #define OPTION_NUM 0x00000800L /* print line numbers */ #define OPTION_INV 0x00001000L /* print invisible lines */ #define OPTION_TSC 0x00002000L /* TSC-compatible source format */ #define OPTION_WAR 0x00004000L /* print warnings */ #define OPTION_CLL 0x00008000L /* check line length (default) */ #define OPTION_LFN 0x00010000L /* print long file names */ #define OPTION_LLL 0x00020000L /* list library lines */ #define OPTION_GAS 0x00040000L /* Gnu AS compatibility */ #define OPTION_REL 0x00080000L /* print relocation table */ #define OPTION_TXT 0x00100000L /* print text table */ #define OPTION_LIS 0x00200000L /* print assembler output listing */ #define OPTION_LPA 0x00400000L /* listing in f9dasm patch format */ #define OPTION_RTF 0x00800000L #define OPTION_X32 0x01000000L /* 32 bit index registers */ #define OPTION_X16 0x02000000L /* 32 bit index registers */ struct { char *Name; unsigned long dwAdd; unsigned long dwRem; } Options[] = {/*Name Add Remove */ { "PAG", OPTION_PAG, 0 }, { "NOP", 0, OPTION_PAG }, { "CON", OPTION_CON, 0 }, { "NOC", 0, OPTION_CON }, { "MAC", OPTION_MAC, 0 }, { "NOM", 0, OPTION_MAC }, { "EXP", OPTION_EXP, 0 }, { "NOE", 0, OPTION_EXP }, { "SYM", OPTION_SYM, 0 }, { "NOS", 0, OPTION_SYM }, { "MUL", OPTION_MUL, 0 }, { "NMU", 0, OPTION_MUL }, { "LP1", OPTION_LP1, 0 }, { "NO1", 0, OPTION_LP1 }, { "DAT", OPTION_DAT, 0 }, { "NOD", 0, OPTION_DAT }, { "NUM", OPTION_NUM, 0 }, { "NON", 0, OPTION_NUM }, { "INV", OPTION_INV, 0 }, { "NOI", 0, OPTION_INV }, { "TSC", OPTION_TSC, 0 }, { "NOT", 0, OPTION_TSC }, { "WAR", OPTION_WAR, 0 }, { "NOW", 0, OPTION_WAR }, { "CLL", OPTION_CLL, 0 }, { "NCL", 0, OPTION_CLL }, { "LFN", OPTION_LFN, 0 }, { "NLF", 0, OPTION_LFN }, { "LLL", OPTION_LLL, 0 }, { "NLL", 0, OPTION_LLL }, { "GAS", OPTION_GAS, 0 }, { "NOG", 0, OPTION_GAS }, { "REL", OPTION_REL, 0 }, { "NOR", 0, OPTION_REL }, { "H63", OPTION_H63, OPTION_M09 | OPTION_M00 }, { "M68", OPTION_M09, OPTION_H63 | OPTION_M00 }, { "M09", OPTION_M09, OPTION_H63 | OPTION_M00 }, { "M00", OPTION_M00, OPTION_H63 | OPTION_M09 }, { "TXT", OPTION_TXT, 0 }, { "NTX", 0, OPTION_TXT }, { "LIS", OPTION_LIS, 0 }, { "NOL", 0, OPTION_LIS }, { "LPA", OPTION_LPA, OPTION_NUM | OPTION_CLL }, // LPA inhibits NUM / CLL! { "NLP", 0, OPTION_LPA }, { "X32", OPTION_X32, OPTION_X16 }, { "X16", OPTION_X16, OPTION_X32 } }; unsigned long dwOptions = /* options flags, init to default */ OPTION_M09 | OPTION_MAC | OPTION_SYM | OPTION_MUL | OPTION_DAT | OPTION_WAR | OPTION_CLL | OPTION_LLL | OPTION_REL | OPTION_LIS; /*****************************************************************************/ /* arrays of error/warning messages */ /*****************************************************************************/ /* defined error flags */ #define ERR_OK 0x0000 /* all is well */ #define ERR_EXPR 0x0001 /* Error in expression */ #define ERR_ILLEGAL_ADDR 0x0002 /* Illegal adressing mode */ #define ERR_LABEL_UNDEF 0x0004 /* Undefined label */ #define ERR_LABEL_MULT 0x0008 /* Label multiply defined */ #define ERR_RANGE 0x0010 /* Relative branch out of range */ #define ERR_LABEL_MISSING 0x0020 /* Missing label */ #define ERR_OPTION_UNK 0x0040 /* Option unknown */ #define ERR_MALLOC 0x0080 /* Out of memory */ #define ERR_NESTING 0x0100 /* Nesting not allowed */ #define ERR_RELOCATING 0x0200 /* Statement not valid for reloc mode*/ #define ERR_ERRTXT 0x4000 /* ERR text output */ #define ERR_ILLEGAL_MNEM 0x8000 /* Illegal mnemonic */ char *errormsg[]= { "Error in expression", /* 1 ERR_EXPR */ "Illegal addressing mode", /* 2 ERR_ILLEGAL_ADDR */ "Undefined label", /* 4 ERR_LABEL_UNDEF */ "Multiple definitions of label", /* 8 ERR_LABEL_MULT */ "Relative branch out of range", /* 16 ERR_RANGE */ "Missing label", /* 32 ERR_LABEL_MISSING */ "Unknown option specified", /* 64 ERR_OPTION_UNK */ "Out of memory", /* 128 ERR_MALLOC */ "Nesting not allowed", /* 256 ERR_NESTING */ "Illegal for current relocation mode",/* 512 ERR_RELOCATING */ "", /* 1024 */ "", /* 2048 */ "", /* 4096 */ "", /* 8192 */ NULL, /* 16384 ERR_ERRTXT (ERR output) */ "Illegal mnemonic" /* 32768 ERR_ILLEGAL_MNEM */ }; /* defined warning flags */ #define WRN_OK 0x0000 /* All OK */ #define WRN_OPT 0x0001 /* Long branch in short range */ #define WRN_SYM 0x0002 /* Symbolic text undefined */ #define WRN_AREA 0x0004 /* Area already in use */ #define WRN_AMBIG 0x0008 /* Ambiguous 6800 opcode */ char *warningmsg[] = { "Long branch within short branch " /* 1 WRN_OPT */ "range could be optimized", "Symbolic text undefined", /* 2 WRN_SYM */ "Area already in use", /* 4 WRN_AREA */ "Ambiguous 6800 alternate notation", /* 8 WRN_AMBIG */ "", /* 16 */ "", /* 32 */ "", /* 64 */ "", /* 128 */ "", /* 256 */ "", /* 512 */ "", /* 1024 */ "", /* 2048 */ "", /* 4096 */ "", /* 8192 */ "", /* 18384 */ "" /* 32768 */ }; /*****************************************************************************/ /* Listing Definitions */ /*****************************************************************************/ #define LIST_OFF 0x00 /* listing is generally off */ #define LIST_ON 0x01 /* listing is generally on */ char listing = LIST_OFF; /* listing flag */ /*****************************************************************************/ /* Global variables */ /*****************************************************************************/ FILE *listfile = NULL; /* list file */ FILE *objfile = NULL; /* object file */ char listname[FNLEN + 1]; /* list file name */ char objname[FNLEN + 1]; /* object file name */ char srcname[FNLEN + 1]; /* source file name */ /* assembler mode specifics: */ struct oprecord *optable = optable09; /* used op table */ /* size of this table */ int optablesize = sizeof(optable09) / sizeof(optable09[0]); struct regrecord *regtable = regtable09;/* used register table */ /* used bit register table */ struct regrecord *bitregtable = bitregtable09; void scanoperands09(struct relocrecord *pp); void (*scanoperands)(struct relocrecord *) = scanoperands09; char pass; /* Assembler pass = 1 or 2 */ char relocatable = 0; /* relocatable object flag */ char absmode = 1; /* absolute mode */ long global = 0; /* all labels global flag */ long common = 0; /* common definition flag */ struct symrecord * commonsym = NULL; /* current common main symbol */ char terminate; /* termination flag */ char generating; /* code generation flag */ unsigned loccounter,oldlc; /* Location counter */ char inpline[LINELEN]; /* Current input line (not expanded) */ char srcline[LINELEN]; /* Current source line */ char * srcptr; /* Pointer to line being parsed */ char unknown; /* flag to indicate value unknown */ char certain; /* flag to indicate value is certain at pass 1*/ long error; /* flags indicating errors in current line. */ long errors; /* number of errors in current pass */ long warning; /* flags indicating warnings in current line */ long warnings; /* number of warnings in current pass */ long nTotErrors; /* total # of errors */ long nTotWarnings; /* total # warnings */ char exprcat; /* category of expression being parsed, eg. label or constant, this is important when generating relocatable object code. */ int maxidlen = MAXIDLEN; /* maximum ID length */ char modulename[MAXIDLEN + 1] = ""; /* module name buffer */ char namebuf[MAXIDLEN + 1]; /* name buffer for parsing */ char unamebuf[MAXIDLEN + 1]; /* name buffer in uppercase */ char isFar,isFarkw; int vercount; char mode; char isX32; char isPostIndexed; /* adressing mode: */ #define ADRMODE_IMM 0 /* 0 = immediate */ #define ADRMODE_DIR 1 /* 1 = direct */ #define ADRMODE_EXT 2 /* 2 = extended */ #define ADRMODE_IDX 3 /* 3 = indexed */ #define ADRMODE_POST 4 /* 4 = postbyte */ #define ADRMODE_PCR 5 /* 5 = PC relative (with postbyte) */ #define ADRMODE_IND 6 /* 6 = indirect */ #define ADRMODE_PIN 7 /* 7 = PC relative & indirect */ #define ADRMODE_DBL_IND 8 char opsize; /* desired operand size : */ /* 0=dunno,1=5, 2=8, 3=16, 4=32 */ long operand; unsigned short postbyte; long dpsetting = 0; /* Direct Page Default = 0 */ unsigned short codebuf[256]; int codeptr; /* byte offset within instruction */ int suppress; /* 0=no suppress */ /* 1=until ENDIF */ /* 2=until ELSE */ /* 3=until ENDM */ char condline = 0; /* flag whether on conditional line */ int ifcount; /* count of nested IFs within */ /* suppressed text */ #define OUT_NONE -1 /* no output */ #define OUT_BIN 0 /* binary output */ #define OUT_SREC 1 /* Motorola S-Records */ #define OUT_IHEX 2 /* Intel Hex Records */ #define OUT_FLEX 3 /* Flex9 ASMB-compatible output */ #define OUT_GAS 4 /* GNU relocation output */ #define OUT_REL 5 /* Flex9 RELASMB output */ #define OUT_VER 6 int outmode = OUT_BIN; /* default to binary output */ unsigned hexaddr; int hexcount; unsigned short hexbuffer[256]; unsigned int chksum; unsigned vslide; int nRepNext = 0; /* # repetitions for REP pseudo-op */ int nSkipCount = 0; /* # lines to skip */ unsigned tfradr = 0; int tfradrset = 0; int nCurLine = 0; /* current output line on page */ int nCurCol = 0; /* current output column on line */ int nCurPage = 0; /* current page # */ int nLinesPerPage = 66; /* # lines on a page */ int nColsPerLine = 200; /* # columns per line */ char szTitle[128] = ""; /* title for listings */ char szSubtitle[128] = ""; /* subtitle for listings */ char szBuf1[LINELEN]; /* general-purpose buffers for parse */ char szBuf2[LINELEN]; struct linebuf *macros[MAXMACROS]; /* pointers to the macros */ int nMacros = 0; /* # parsed macros */ int inMacro = 0; /* flag whether in macro definition */ char *texts[MAXTEXTS]; /* pointers to the texts */ int nPredefinedTexts = 0; /* # predefined texts */ int nTexts = 0; /* # currently defined texts */ //unsigned char bUsedBytes[8192] = {0}; /* 1 bit per byte of the address spc */ /*****************************************************************************/ /* Necessary forward declarations */ /*****************************************************************************/ void processline(); struct linebuf *readfile(char *name, unsigned char lvl, struct linebuf *after); struct linebuf *readbinary(char *name, unsigned char lvl, struct linebuf *after, struct symrecord *lp); /*****************************************************************************/ /* allocline : allocates a line of text */ /*****************************************************************************/ struct linebuf * allocline ( struct linebuf *prev, char *fn, int line, unsigned char lvl, char *text ) { struct linebuf *pNew = malloc(sizeof(struct linebuf) + strlen(text)); if (!pNew) return NULL; pNew->next = (prev) ? prev->next : NULL; pNew->prev = prev; if (prev) prev->next = pNew; if (pNew->next) pNew->next->prev = pNew; pNew->lvl = lvl; pNew->fn = fn; pNew->ln = line; pNew->rel = ' '; strcpy(pNew->txt, text); return pNew; } /*****************************************************************************/ /* expandfn : eventually expands a file name to full-blown path */ /*****************************************************************************/ char const *expandfn(char const *fn) { #ifdef WIN32 static char szBuf[_MAX_PATH]; /* allocate big buffer */ if (dwOptions & OPTION_LFN) /* if long file names wanted */ return _fullpath(szBuf, fn, sizeof(szBuf)); #endif return fn; /* can't do that yet */ } /*****************************************************************************/ /* PageFeed : advances the list file */ /*****************************************************************************/ void PageFeed() { time_t tim; struct tm *ltm; time(&tim); ltm = localtime(&tim); fputc('\x0c', listfile); /* print header */ nCurPage++; /* advance to next page */ fprintf(listfile, "\n\n%-32.32s ", szTitle); if (dwOptions & OPTION_DAT) fprintf(listfile, "%04d-%02d-%02d ", ltm->tm_year + 1900, ltm->tm_mon + 1, ltm->tm_mday); fprintf(listfile, "A09 %d Assembler V" VERSION " Page %d\n", (dwOptions & OPTION_H63) ? 6309 : (dwOptions & OPTION_M00) ? 6800 : 6809, nCurPage); fprintf(listfile, "%-.79s\n\n", szSubtitle); nCurLine = 5; /* remember current line */ nCurCol = 0; /* and reset current column */ } /*****************************************************************************/ /* putlist : puts something to the list file */ /*****************************************************************************/ void putlist(char *szFmt, ...) { char szList[1024]; /* buffer for 1k list output */ char *p; va_list al; va_start(al, szFmt); vsprintf(szList, szFmt, al); /* generate formatted output buffer */ va_end(al); for (p = szList; *p; p++) /* then walk through the buffer */ { fputc(*p, listfile); /* print out each character */ if (*p == '\n') /* if newline sent */ { nCurLine++; nCurCol = 0; if ((nCurLine >= nLinesPerPage) && /* if beyond # lines per page */ (dwOptions & OPTION_PAG)) /* if pagination activated */ PageFeed(); /* do a page feed */ } else /* if another character */ { nCurCol++; /* advance to next column */ if (dwOptions & OPTION_CLL) /* if line length checked */ { /* check if word would go too far */ if ((nCurCol >= nColsPerLine * 3 / 4) && (*p == ' ')) { int i; /* look whether more delimiters */ char c; for (i = nCurCol + 1; i < nColsPerLine; i++) { c = p[i - nCurCol]; if ((c == '\t') || (c == ' ') || (!c)) break; } if (i >= nColsPerLine) /* if no more delimiters, */ nCurCol = nColsPerLine; /* make sure to advance to new line */ } if (nCurCol >= nColsPerLine) /* if it IS too far */ { putlist("\n"); /* recurse with a newline */ nCurCol = 0; } } } } } /*****************************************************************************/ /* findop : finds a mnemonic in table using binary search */ /*****************************************************************************/ struct oprecord * findop(char * nm) { int lo,hi,i,s; lo = 0; hi = optablesize - 1; do { i = (lo + hi) / 2; s = strcmp(optable[i].name, nm); if (s < 0) lo = i + 1; else if (s > 0) hi = i - 1; else break; } while (hi >= lo); if (s) return NULL; return optable + i; } /*****************************************************************************/ /* findlocal : finds a local symbol table record */ /*****************************************************************************/ struct symrecord * findlocal(struct symrecord *sym, char forward, int insert) { static struct symrecord empty = {"", SYMCAT_EMPTYLOCAL, 0, {0}}; int lo,hi,i,j,s; if ((!sym) || /* if no main symbol for that */ ((!insert) && /* or not inserting, but */ (sym->cat == SYMCAT_EMPTYLOCAL))) /* yet undefined label */ return sym; /* pass back main symbol */ lo = 0; /* do binary search for the thing */ hi = lclcounter - 1; s = 1; i = 0; while (hi >= lo) { i = (lo + hi) / 2; s = lcltable[i].value - loccounter; /* binary search for current address */ if (s < 0) lo = i + 1; else if (s > 0) hi = i - 1; else /* if found, */ { /* go to 1st of this value */ while ((i) && (lcltable[i - 1].value == loccounter)) i--; while ((i < lclcounter) && /* search for the NAME now */ (lcltable[i].value == loccounter)) { s = strcmp(sym->name, lcltable[i].name); if (s <= 0) { if (s) i--; break; } i++; } if (s) /* if not found, re-set */ { if (i >= lclcounter) s = 1; else s = lcltable[i].value - loccounter; } break; } } if (insert) /* if inserting, */ { if (!s) /* if address is already in use */ { /* but not the correct label */ if (strcmp(sym->name, lcltable[i].name)) error |= ERR_LABEL_MULT; /* set error */ return lcltable + i; /* return the local symbol */ } i = (s < 0 ? i + 1 : i); if (lclcounter == MAXLABELS) { printf("%s(%ld): error 25: out of local symbol storage\n", expandfn(curline->fn), curline->ln); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 25: out of local symbol storage\n"); exit(4); } sym->cat = SYMCAT_LOCALLABEL; for (j = lclcounter; j > i; j--) lcltable[j] = lcltable[j - 1]; lclcounter++; strcpy(lcltable[i].name, sym->name); lcltable[i].cat = SYMCAT_LOCALLABEL; lcltable[i].value = loccounter; lcltable[i].u.parent = NULL; return lcltable + i; /* pass back this symbol */ } if (forward) /* if forward search */ { i = (s < 0 ? i - 1 : i); for (; i < lclcounter; i++) { if ((!strcmp(lcltable[i].name, sym->name)) && (lcltable[i].value > loccounter)) return lcltable + i; } } else /* if backward search */ { i = (s > 0 ? i + 1 : i); for (; i >= 0; i--) { if ((!strcmp(lcltable[i].name, sym->name)) && (lcltable[i].value <= loccounter)) return lcltable + i; } } return ∅ /* if not found, return empty label */ } /*****************************************************************************/ /* findsym : finds symbol table record; inserts if not found */ /* uses binary search, maintains sorted table */ /*****************************************************************************/ struct symrecord * findsym (char * nm, int insert) { int lo,hi,i,j,s; char islocal = 0, forward = 0; char name[MAXIDLEN + 1] = ""; /* copy name to internal buffer */ strncpy(name, (nm) ? nm : "", sizeof(name) - 1); for (i = 0; name[i]; i++) /* determine whether local label */ if ((name[i] < '0') || (name[i] > '9')) break; if (i) /* if starting with number */ { if (!name[i]) /* if ONLY numeric */ islocal = 1; /* this is a local label */ else /* otherwise check direction */ { switch(toupper(name[i])) { case 'B' : /* backward reference ? */ islocal = 2; /* this is a local label reference */ forward = 0; break; case 'F' : /* forward reference ? */ islocal = 2; /* this is a local label reference */ forward = 1; break; } if (islocal && name[i + 1]) /* if followed by anything else */ islocal = 0; /* reset flag for local label */ else /* otherwise */ name[i] = 0; /* remove the direction */ } } lo = 0; /* do binary search for the thing */ hi = symcounter - 1; s = 1; i = 0; while (hi >= lo) { i = (lo + hi) / 2; s = strcmp(symtable[i].name, name); if (s < 0) lo = i + 1; else if (s > 0) hi = i - 1; else break; } if (s) /* if symbol not found */ { if (!insert) /* if inserting prohibited, */ return NULL; /* return without pointer */ i = (s < 0 ? i + 1 : i); if (symcounter == MAXLABELS) { printf("%s(%ld): error 23: out of symbol storage\n", expandfn(curline->fn), curline->ln); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 23: out of symbol storage\n"); exit(4); } if (commonsym >= symtable + i) commonsym++; for (j = 0; j < symcounter; j++) if (symtable[j].u.parent >= symtable + i) symtable[j].u.parent++; for (j = symcounter; j > i; j--) symtable[j] = symtable[j-1]; symcounter++; strcpy(symtable[i].name, name); symtable[i].cat = (islocal) ? SYMCAT_EMPTYLOCAL : SYMCAT_EMPTY; symtable[i].value = 0; symtable[i].u.flags = 0; } if (islocal) /* if searching for a local label */ return findlocal(symtable + i, /* search for the local label */ forward, (islocal < 2)); return symtable + i; /* return the found or inserted sym */ } /*****************************************************************************/ /* findsymat : finds 1st symbol fo a given address */ /*****************************************************************************/ char *findsymat(unsigned addr) { /* since the symbol table is sorted by name, this needs a sequential search */ int i; for (i = 0; i < symcounter; i++) if (symtable[i].cat != SYMCAT_EMPTY) { if (symtable[i].cat == SYMCAT_TEXT) continue; if (symtable[i].value == addr) return symtable[i].name; } return NULL; } /*****************************************************************************/ /* settext : sets a text symbol */ /*****************************************************************************/ int settext(char *namebuf, char *text) { struct symrecord *lp = findsym(namebuf, 1); int len; int special = 0; if (!lp) { error |= ERR_LABEL_UNDEF; return -1; } if (lp->cat != SYMCAT_EMPTY && lp->cat != SYMCAT_TEXT) { error |= ERR_LABEL_MULT; return -1; } if (lp->cat != SYMCAT_EMPTY) free(texts[lp->value]); else if (nTexts >= MAXTEXTS) { error |= ERR_MALLOC; return -1; } else { lp->cat = SYMCAT_TEXT; lp->value = nTexts++; } if (!text) /* if NULL input */ text = ""; /* assume empty string */ for (len = 0; text[len]; len++) if ((isspace(text[len])) || (text[len] == ',')) special = 1; if (!len) /* empty string? */ special = 1; /* that's special anyway. */ if (special && len && /* if delimited special string */ (text[0] == '\'' || text[0] == '\"') && (text[len - 1] == text[0])) special = 0; /* forget that "special" flag */ len += (special) ? 3 : 1; texts[lp->value] = malloc(len); if (texts[lp->value]) { if (special) texts[lp->value][0] = '\"'; strcpy(texts[lp->value] + special, text); if (special) strcat(texts[lp->value], "\""); } return lp->value; } /*****************************************************************************/ /* outsymtable : prints the symbol table */ /*****************************************************************************/ void outsymtable() { int i,j = 0; if (dwOptions & OPTION_PAG) /* if pagination active, */ { if (nCurLine > 5) /* if not on 1st line, */ PageFeed(); /* shift to next page */ } else putlist("\n"); putlist("SYMBOL TABLE"); for (i = 0; i < symcounter; i++) if (symtable[i].cat != SYMCAT_EMPTY) { /* suppress listing of predef texts */ if ((symtable[i].cat == SYMCAT_TEXT) && (symtable[i].value < nPredefinedTexts)) continue; /* if local label */ if (symtable[i].cat == SYMCAT_LOCALLABEL) { int k; /* walk local label list */ for (k = 0; k < lclcounter; k++) if (!strcmp(lcltable[k].name, symtable[i].name)) { if (j % 4 == 0) putlist("\n"); putlist( " %9s %02d %08X", lcltable[k].name, lcltable[k].cat, lcltable[k].value); j++; } } else /* if normal label */ { if (j % 4 == 0) putlist("\n"); putlist( " %9s %02d %08X", symtable[i].name,symtable[i].cat, symtable[i].value); j++; } } putlist("\n%d SYMBOLS\n", j); } /*****************************************************************************/ /* outreltable : prints the relocation table */ /*****************************************************************************/ void outreltable() { int i,j = 0; if (dwOptions & OPTION_PAG) /* if pagination active, */ { if (nCurLine > 5) /* if not on 1st line, */ PageFeed(); /* shift to next page */ } else putlist("\n"); putlist("RELOCATION TABLE"); for (i = 0; i < relcounter; i++) { char name[10]; sprintf(name, "%c%-.8s", (reltable[i].exprcat & EXPRCAT_NEGATIVE) ? '-' : ' ', reltable[i].sym->name); if (j % 4 == 0) putlist("\n"); putlist( " %9s %02d %04X", name, reltable[i].sym->cat, reltable[i].addr); j++; } putlist("\n%d RELOCATIONS\n", j); } /*****************************************************************************/ /* outtexttable : prints the text table */ /*****************************************************************************/ void outtexttable() { int i,j = 0; if (dwOptions & OPTION_PAG) /* if pagination active, */ { if (nCurLine > 5) /* if not on 1st line, */ PageFeed(); /* shift to next page */ } else putlist("\n"); putlist("TEXT TABLE"); for (i = 0; i < symcounter; i++) if (symtable[i].cat == SYMCAT_TEXT) { /* suppress listing of predef texts */ if (symtable[i].value < nPredefinedTexts) continue; putlist("\n %9s %s", symtable[i].name, texts[symtable[i].value]); j++; } putlist("\n%d TEXTS\n", j); } /*****************************************************************************/ /* findreg : finds a register per name */ /*****************************************************************************/ struct regrecord * findreg(char *nm) { int i; for (i = 0; regtable[i].name != NULL; i++) { if (strcmp(regtable[i].name,nm) == 0) return regtable + i; } return 0; } /*****************************************************************************/ /* findreg63 : finds a register per name for HD63 operations */ /*****************************************************************************/ struct regrecord * findreg63(char *nm) { int i; for (i = 0; i < (sizeof(regtable63) / sizeof(regtable63[0])); i++) { if (strcmp(regtable63[i].name,nm) == 0) return regtable63 + i; } return 0; } /*****************************************************************************/ /* findbitreg : finds a bit transfer register per name (6309 only) */ /*****************************************************************************/ struct regrecord * findbitreg(char *nm) { int i; for (i = 0; i < (sizeof(bitregtable) / sizeof(bitregtable[0])); i++) { if (strcmp(bitregtable[i].name,nm) == 0) return bitregtable + i; } return 0; } /*****************************************************************************/ /* strupr : converts a string to uppercase (crude) */ /*****************************************************************************/ char *strupr(char *name) { int i; if (!name) return name; for (i = 0; name[i]; i++) if ((name[i] >= 'a') && (name[i] <= 'z')) name[i] -= ('a' - 'A'); return name; } /*****************************************************************************/ /* addreloc : adds a relocation record to the list */ /*****************************************************************************/ void addreloc(struct relocrecord *p) { struct relocrecord rel = {0}; /* internal copy */ if (p) /* if there's a record, */ rel = *p; /* copy to internal */ if ((!relocatable) || /* if generating unrelocatable binary*/ (!rel.sym) || /* or no symbol */ (pass == 1)) /* or in pass 1 */ return; /* do nothing here */ switch (rel.sym->cat) /* check symbol category */ { case SYMCAT_PUBLIC : /* public label ? */ case SYMCAT_LABEL : /* normal label ? */ case SYMCAT_EXTERN : /* external symbol ? */ break; /* these are allowed */ case SYMCAT_COMMONDATA : /* Common data ? */ rel.sym = rel.sym->u.parent; /* switch to COMMON block */ break; /* and allow it. */ default : /* anything else */ return; /* isn't */ } if (relcounter >= MAXRELOCS) /* if no more space */ { /* should NEVER happen... but then...*/ error |= ERR_MALLOC; /* set mem alloc err */ return; /* and get out of here */ } reltable[relcounter++] = rel; /* add relocation record */ switch (p->sym->cat) /* do specials... */ { case SYMCAT_PUBLIC : /* public label ? */ case SYMCAT_LABEL : /* normal label ? */ case SYMCAT_COMMONDATA : /* common data ? */ curline->rel = /* remember there's a relocation */ (p->exprcat & EXPRCAT_NEGATIVE) ? '-' : '+'; break; case SYMCAT_EXTERN : /* external symbol ? */ curline->rel = (p->exprcat & EXPRCAT_NEGATIVE) ? 'x' : 'X'; break; } } /*****************************************************************************/ /* scanname : scans a name from the input buffer */ /*****************************************************************************/ void scanname() { int i = 0; char c; char cValid; while (1) { c = *srcptr++; if ((!(dwOptions & OPTION_TSC)) && /* TSC Assembler is case-sensitive */ (!(dwOptions & OPTION_GAS)) && /* GNU Assembler is case-sensitive */ (c >= 'a' && c <= 'z')) c -= 32; cValid = 0; /* check for validity */ if ((c >= '0' && c <= '9') || /* normally, labels may consist of */ (c >= 'A' && c <= 'Z') || /* the characters A..Z,a..z,_ */ (c >= 'a' && c <= 'z') || (c == '_')) cValid = 1; if (dwOptions & OPTION_GAS) /* for GNU AS compatibility, */ { /* the following rules apply: */ if ((c == '.') || /* .$ are valid symbol characters */ (c == '$')) cValid = 1; } if (!cValid) /* if invalid character encountered */ break; /* stop here */ if (i < maxidlen) { namebuf[i] = c; unamebuf[i] = toupper(c); i++; } } namebuf[i] = '\0'; unamebuf[i] = '\0'; srcptr--; } /*****************************************************************************/ /* skipspace : skips whitespace characters */ /*****************************************************************************/ void skipspace() { char c; do { c = *srcptr++; } while (c == ' ' || c == '\t'); srcptr--; } long scanexpr(int, struct relocrecord *); /*****************************************************************************/ /* scandecimal : scans a decimal number */ /*****************************************************************************/ long scandecimal() { char c; long t = 0; c = *srcptr++; while (isdigit(c)) { t = t * 10 + c - '0'; c = *srcptr++; } srcptr--; return t; } /*****************************************************************************/ /* scanhex : scans hex number */ /*****************************************************************************/ long scanhex() { long t = 0, i = 0; srcptr++; scanname(); while (unamebuf[i] >= '0' && unamebuf[i] <= 'F') { t = t * 16 + unamebuf[i] - '0'; if (unamebuf[i] > '9') t -= 7; i++; } if (i==0) error |= ERR_EXPR; return t; } /*****************************************************************************/ /* scanchar : scan a character */ /*****************************************************************************/ long scanchar() { long t; srcptr++; t = *srcptr; if (t) srcptr++; if (*srcptr == '\'') srcptr++; return t; } /*****************************************************************************/ /* scanbin : scans a binary value */ /*****************************************************************************/ long scanbin() { char c; short t = 0; srcptr++; c = *srcptr++; while (c == '0' || c == '1') { t = t * 2 + c - '0'; c = *srcptr++; } srcptr--; return t; } /*****************************************************************************/ /* scanoct : scans an octal value */ /*****************************************************************************/ long scanoct() { char c; long t = 0; srcptr++; c = *srcptr++; while (c >= '0' && c <= '7') { t = t * 8 + c - '0'; c = *srcptr++; } srcptr--; return t; } /*****************************************************************************/ /* scanstring : scans a string into a buffer */ /*****************************************************************************/ char * scanstring(char *dest, int nlen) { char *s = srcptr; char *d = dest; int nInString = 0; char c; if (*srcptr == '\'' || *srcptr == '\"') { nInString = 1; srcptr++; } while (*srcptr) { if (!nInString && (*srcptr == ' ' || *srcptr == ',')) break; else if (nInString && *s == *srcptr) { srcptr++; break; } c = *srcptr++; if (!nInString && c >= 'a' && c <= 'z') c -= 32; *d++ = c; } *d = '\0'; return dest; } /*****************************************************************************/ /* scanlabel : scans a label */ /*****************************************************************************/ unsigned scanlabel(struct relocrecord *pp) { struct symrecord * p; scanname(); p = findsym(namebuf, 1); if (p->cat == SYMCAT_EMPTY) { p->cat = SYMCAT_UNRESOLVED; p->value = 0; p->u.flags |= SYMFLAG_FORWARD; } if (p->cat == SYMCAT_MACRO || p->cat == SYMCAT_PARMNAME || p->cat == SYMCAT_TEXT) error |= ERR_EXPR; exprcat = p->cat & (EXPRCAT_PUBLIC | EXPRCAT_EXTADDR | EXPRCAT_INTADDR); if (exprcat == (EXPRCAT_EXTADDR | EXPRCAT_INTADDR) || exprcat == (EXPRCAT_PUBLIC | EXPRCAT_INTADDR)) unknown = 1; #if 1 /* anything that's not yet defined is uncertain in pass 2! */ if ((p->u.flags & (SYMFLAG_FORWARD | SYMFLAG_PASSED)) == SYMFLAG_FORWARD) certain = 0; #else if (((exprcat == EXPRCAT_INTADDR || exprcat == EXPRCAT_PUBLIC) && // (unsigned short)(p->value) > (unsigned short)loccounter) || // (p->u.flags & SYMFLAG_FORWARD)) || ((p->u.flags & (SYMFLAG_FORWARD | SYMFLAG_PASSED)) == SYMFLAG_FORWARD)) || exprcat == EXPRCAT_EXTADDR) certain = 0; #endif if ((!absmode) && /* if in relocating mode */ (exprcat)) /* and this is not a constant */ certain = 0; /* this is NOT certain */ if (exprcat == EXPRCAT_PUBLIC || exprcat == (EXPRCAT_EXTADDR | EXPRCAT_INTADDR) || exprcat == (EXPRCAT_PUBLIC | EXPRCAT_INTADDR)) exprcat = EXPRCAT_INTADDR; if (pp) { pp->exprcat = exprcat; pp->sym = p; } if (p->isFar) isFar = 1; if (p->isFarkw) isFarkw = 1; return p->value; } /*****************************************************************************/ /* isfactorstart : returns whether passed character possibly starts a factor */ /*****************************************************************************/ int isfactorstart(char c) { if (isalpha(c)) return 1; else if (isdigit(c)) return 1; else switch (c) { case '*' : case '$' : case '%' : case '@' : case '\'' : case '(' : case '-' : case '+' : case '!' : case '~' : return 1; } return 0; } /*****************************************************************************/ /* scanfactor : scans an expression factor */ /*****************************************************************************/ long scanfactor(struct relocrecord *p) { char c; long t; if (!(dwOptions & OPTION_TSC)) skipspace(); c = *srcptr; if (isalpha(c)) return (unsigned)scanlabel(p); else if (isdigit(c)) { char *locptr = srcptr; /* watch out for local labels */ char caft; while ((*locptr >= '0') && /* advance to next nonnumeric */ (*locptr <= '9')) locptr++; caft = toupper(*locptr); /* get next character in uppercase */ if ((caft == 'B') || (caft == 'F')) /* if this might be a local reference*/ return scanlabel(p); /* look it up. */ switch (c) { case '0' : /* GNU AS bin/oct/hex? */ if (dwOptions & OPTION_GAS) /* if GNU AS extensions, */ { if (srcptr[1] == 'b') /* if binary value, */ { srcptr++; /* advance behind 0 */ return scanbin(); /* and treat rest as binary value */ } else if (srcptr[1] == 'x') /* if hex value, */ { srcptr++; /* advance behind 0 */ return scanhex(); /* and treat rest as hex value */ } return scanoct(); /* otherwise treat as octal */ } /* else fall thru on purpose */ default : /* decimal in any case ? */ return scandecimal(); } } else switch (c) { case '*' : srcptr++; exprcat |= EXPRCAT_INTADDR; return loccounter; case '$' : return scanhex(); case '%' : return scanbin(); case '@' : return scanoct(); case '\'' : return scanchar(); case '(' : srcptr++; t = scanexpr(0, p); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ')') srcptr++; else error |= ERR_EXPR; return t; case '-' : srcptr++; t = scanfactor(p); exprcat ^= EXPRCAT_NEGATIVE; if (p) p->exprcat ^= EXPRCAT_NEGATIVE; return -t; case '+' : srcptr++; return scanfactor(p); case '!' : srcptr++; exprcat |= EXPRCAT_FIXED; return !scanfactor(p); case '~' : srcptr++; exprcat |= EXPRCAT_FIXED; return ~scanfactor(p); } error |= ERR_EXPR; return 0; } /*****************************************************************************/ /* some nice macros */ /*****************************************************************************/ /*define EXITEVAL { srcptr--; return t; } */ #define EXITEVAL { srcptr--; parsing = 0; break; } #define RESOLVECAT if((oldcat & 15) == 0) oldcat = 0; \ if ((exprcat & 15) == 0)exprcat = 0; \ if ((exprcat == EXPRCAT_INTADDR && \ oldcat == (EXPRCAT_INTADDR | EXPRCAT_NEGATIVE)) || \ (exprcat == (EXPRCAT_INTADDR | EXPRCAT_NEGATIVE)&& \ oldcat == EXPRCAT_INTADDR)) \ { \ exprcat = 0; \ oldcat = 0; \ } \ exprcat |= oldcat; /* resolve such cases as constant added to address or difference between two addresses in same module */ /*****************************************************************************/ /* scanexpr : scan expression */ /*****************************************************************************/ long scanexpr(int level, struct relocrecord *pp) /* This is what you call _recursive_ descent!!!*/ { long t, u; char oldcat,c,parsing=1; struct relocrecord ip = {0}, p = {0}; exprcat = 0; if (level == 10) return scanfactor(pp); t = scanexpr(level + 1, &ip); /* ip.exprcat = exprcat; */ while (parsing) { p.sym = NULL; if (!(dwOptions & OPTION_TSC)) skipspace(); c = *srcptr++; switch(c) { case '*': oldcat = exprcat; t *= scanexpr(10, &p); exprcat |= oldcat | EXPRCAT_FIXED; break; case '/': oldcat = exprcat; u = scanexpr(10, &p); if (u) t /= u; else error |= ERR_EXPR; exprcat |= oldcat | EXPRCAT_FIXED; break; case '%': oldcat = exprcat; u = scanexpr(10, &p); if (u) t %= u; else error |= ERR_EXPR; exprcat |= oldcat | EXPRCAT_FIXED; break; case '+': if (level == 9) EXITEVAL oldcat = exprcat; t += scanexpr(9, &p); RESOLVECAT break; case '-': if (level == 9) EXITEVAL oldcat = exprcat; t -= scanexpr(9, &p); exprcat ^= EXPRCAT_NEGATIVE; RESOLVECAT break; case '<': if (*(srcptr) == '<') { if (level >= 8) EXITEVAL srcptr++; oldcat = exprcat; t <<= scanexpr(8, &p); exprcat |= oldcat | EXPRCAT_FIXED; } else if (*(srcptr) == '=') { if (level >= 7) EXITEVAL srcptr++; oldcat = exprcat; t = t <= scanexpr(7, &p); exprcat |= oldcat | EXPRCAT_FIXED; } else { if (level >= 7) EXITEVAL oldcat = exprcat; t = t < scanexpr(7, &p); exprcat |= oldcat | EXPRCAT_FIXED; } break; case '>': if (*(srcptr) == '>') { if (level >= 8) EXITEVAL srcptr++; oldcat = exprcat; t >>= scanexpr(8, &p); exprcat |= oldcat | EXPRCAT_FIXED; } else if (*(srcptr) == '=') { if (level>=7) EXITEVAL srcptr++; oldcat = exprcat; t = t >= scanexpr(7, &p); exprcat |= oldcat | EXPRCAT_FIXED; } else { if (level >= 7) EXITEVAL oldcat = exprcat; t = t > scanexpr(7, &p); exprcat |= oldcat | EXPRCAT_FIXED; } break; case '!': if (level >= 6 || *srcptr != '=') EXITEVAL srcptr++; oldcat = exprcat; t = t != scanexpr(6, &p); exprcat |= oldcat | EXPRCAT_FIXED; break; case '=': if (level >= 6) EXITEVAL if (*srcptr == '=') srcptr++; oldcat = exprcat; t = (t == scanexpr(6, &p)); exprcat |= oldcat | EXPRCAT_FIXED; break; case '&': if (level >= 5) EXITEVAL oldcat = exprcat; t &= scanexpr(5, &p); exprcat |= oldcat | EXPRCAT_FIXED; break; case '^': if (level >= 4) EXITEVAL oldcat = exprcat; t ^= scanexpr(4, &p); exprcat |= oldcat | EXPRCAT_FIXED; break; case '|': if (level >= 3) EXITEVAL oldcat = exprcat; t |= scanexpr(3, &p); exprcat |= oldcat | EXPRCAT_FIXED; break; default: EXITEVAL } p.exprcat = exprcat; if (p.sym) { if (ip.sym) /* if 2 symbols, cancel 'em */ { /* a simple safeguard against cancelling local vs. external symbols or operations between 2 external symbols. This means that an external symbol has to be the last one in an expression, or the others have to be paired so that they cancel each other's effect AND their subexpression has to be parenthesized.*/ if ((ip.sym->cat == SYMCAT_EXTERN) || (p.sym->cat == SYMCAT_EXTERN)) error |= ERR_EXPR; else ip.sym = NULL; /* this might be TOO crude... */ } else /* if new symbol */ ip = p; /* use this one */ } } *pp = ip; return t; } /*****************************************************************************/ /* scanindexreg : scans an index register */ /*****************************************************************************/ int scanindexreg() { switch (toupper(*srcptr)) { case 'X': return 1; case 'Y': postbyte |= 0x200; return 1; case 'U': postbyte |= 0x400; return 1; case 'S': postbyte |= 0x600; return 1; } return 0; } /*****************************************************************************/ /* set3 : sets mode to at least ADRMODE_POST */ /*****************************************************************************/ void set3() { if (mode < ADRMODE_POST) mode = ADRMODE_POST; } /*****************************************************************************/ /* scanspecial : scans for increments */ /*****************************************************************************/ void scanspecial() { set3(); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == '-') { srcptr++; if (*srcptr == '-') { srcptr++; postbyte = 0x803; } else postbyte = 0x802; if ((dwOptions & OPTION_H63) && /* special for ,--W and [,--W] */ (postbyte == 0x803) && (toupper(*srcptr) == 'W')) { postbyte = (mode == ADRMODE_IND) ? 0x900 : 0x80f; srcptr++; } else if (!scanindexreg()) error |= ERR_ILLEGAL_ADDR; else srcptr++; } else { postbyte = 0x800; if ((dwOptions & OPTION_H63) && /* special for ,W, [,W], ,W++, [,W++]*/ (toupper(*srcptr) == 'W')) { srcptr++; /* advance behind W */ if (*srcptr == '+') { srcptr++; if (*srcptr == '+') /* ,W++ and [,W++] */ { postbyte = (mode == ADRMODE_IND) ? 0xB00 : 0xA0F; srcptr++; } else error |= ERR_ILLEGAL_ADDR; } else /* ,W and [,W] */ postbyte = (mode == ADRMODE_IND) ? 0x900 : 0x80F; } else /* normal index register addressing */ { if (!scanindexreg()) error |= ERR_ILLEGAL_ADDR; else srcptr++; if (*srcptr == '+') { srcptr++; if (*srcptr == '+') { srcptr++; postbyte += 1; } } else postbyte += 4; } } } /*****************************************************************************/ /* scanindexed : */ /*****************************************************************************/ void scanindexed() { set3(); postbyte = 0; if ((dwOptions & OPTION_H63) && /* special for W as index register */ (toupper(*srcptr) == 'W')) { srcptr++; postbyte = (mode == ADRMODE_IND) ? 0xb00 : 0xa0f; opsize = 3; } else if (scanindexreg()) { srcptr++; if (opsize == 0) { if (unknown || !certain) opsize = 3; else if (operand >= -256 && operand < 256 && mode == ADRMODE_POST) opsize = 1; else if (operand >= -2048 && operand < 2048) opsize = 2; else if (operand >=-8388608 && operand < 8388608) opsize = 3; else opsize = 4; } switch (opsize) { case 1: postbyte |= (operand & 511); opsize = 0; break; case 2: postbyte |= 0x808; break; case 3: postbyte |= 0x809; break; case 4: postbyte |= 0x80A; break; } } else { /*pc relative*/ if (toupper(*srcptr) != 'P') error |= ERR_ILLEGAL_ADDR; else { srcptr++; if (toupper(*srcptr) != 'C') error |= ERR_ILLEGAL_ADDR; else { srcptr++; if (toupper(*srcptr) == 'R') srcptr++; } } mode++; postbyte |= 0x80c; if (opsize == 1) opsize = 2; } } /*****************************************************************************/ /* scanoperands : */ /*****************************************************************************/ #define RESTORE { srcptr = oldsrcptr; c = *srcptr; goto dodefault; } void scanoperands09(struct relocrecord *pp) { char c, *oldsrcptr, *ptr; unsigned short accpost, h63 = 0; unsigned short isIndexed = 0; unsigned short indirect_byte = 0; isFar = 0; isPostIndexed = 0; unknown = 0; opsize = 0; certain = 1; scano1: skipspace(); c = *srcptr; mode = ADRMODE_IMM; if (c == '[') { c = *++srcptr; if (c=='[') { c = *++srcptr; mode = ADRMODE_DBL_IND; postbyte = 0x80F; } else mode = ADRMODE_IND; indirect_byte = 0x100; } switch (toupper(c)) { case 'D': accpost = 0x80b; accoffset: oldsrcptr = srcptr; srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr != ',') RESTORE else { isIndexed = 1; if ((h63) && (!(dwOptions & OPTION_H63))) error |= ERR_ILLEGAL_ADDR; postbyte |= accpost; srcptr++; if (!scanindexreg()) RESTORE else { srcptr++; set3(); } } break; case 'A': accpost = 0x806; goto accoffset; case 'B': accpost = 0x805; goto accoffset; case 'E': accpost = 0x807; h63 = 1; goto accoffset; case 'F': if (toupper(srcptr[1])=='A' && toupper(srcptr[2])=='R' && (srcptr[3]==' '||srcptr[3]=='\t')) { isFar = 1; isFarkw = 1; srcptr += 3; goto scano1; } accpost = 0x80a; h63 = 1; goto accoffset; case 'W' : accpost = 0x80e; h63 = 1; goto accoffset; case ',': srcptr++; scanspecial(); break; case '#': if (mode == ADRMODE_IND) error |= ERR_ILLEGAL_ADDR; else mode = ADRMODE_IMM; srcptr++; if (*srcptr=='>') { srcptr++; operand = (scanexpr(0, pp) >> 24LL) & 0xffffffLL; isFar = 0; opsize = 3; } else if (*srcptr=='<') { srcptr++; operand = scanexpr(0, pp) & 0xffffffLL; isFar = 0; opsize = 3; } else operand = scanexpr(0, pp); break; case '<': srcptr++; if (*srcptr == '<') { srcptr++; opsize = 1; } else opsize = 2; goto dodefault; case '>': srcptr++; opsize = 3; /* fall thru on purpose */ default: dodefault: operand = scanexpr(0, pp); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') { isIndexed = 1; srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); if ((operand == 0) && /* special for "0,-[-]indexreg */ (!unknown) && (certain) && /* and "0,indexreg[+[+]] */ (opsize < 3) && /* but NOT for ">0,indexreg"! */ ((*srcptr == '-') || (scanindexreg() /* && (srcptr[1] == '+') */ ))) scanspecial(); else scanindexed(); } else { if (opsize == 0) { if ((unsigned)operand >= 16777216) opsize = 4; else if (unknown || !certain || dpsetting == -1 || (signed)(operand - dpsetting * 4096) >= 2048 || (signed)(operand - dpsetting * 4096) < -2048) opsize = 3; else opsize = 2; } /* if (opsize == 4) { if ((operand & 0xff800000) == 0xff800000) opsize = 3; else if ((operand & 0xff800000) == 0x0) opsize = 3; } */ if (opsize == 1) opsize = 2; if (mode == ADRMODE_IND) { postbyte = 0x80f; // Why is the following set ? // opsize = 3; } else if (mode == ADRMODE_DBL_IND) ; else { switch(opsize) { case 2: mode = ADRMODE_DIR; break; case 3: mode = ADRMODE_EXT; break; // case 4: case 4: mode = ADRMODE_EXT; isFar = 1; break; default: mode = opsize - 1; } } } } if (mode >= ADRMODE_IND) { if (!(dwOptions & OPTION_TSC)) skipspace(); if (mode != ADRMODE_DBL_IND) postbyte |= 0x100; if (*srcptr != ']') error |= ERR_ILLEGAL_ADDR; srcptr++; if (mode==ADRMODE_DBL_IND) { if (*srcptr != ']') error |= ERR_ILLEGAL_ADDR; srcptr++; } } skipspace(); if (*srcptr==',') { srcptr++; isPostIndexed = 1; scanindexed(); // scanindexed will reset the postbyte postbyte |= indirect_byte | 0x80; } else { if (postbyte==0x90F || postbyte==0x80F) opsize = 3; if (mode == ADRMODE_IND || mode==ADRMODE_DBL_IND) { /* if ((postbyte & 0xF)==4) // 0 offset opsize = 0; // else if ((postbyte & 0xf)==8 || (postbyte & 0xF)== else opsize = 3; */ } } if (pass > 1 && unknown) error |= ERR_LABEL_UNDEF; } void scanoperands00(struct relocrecord *pp) { char c, *s = srcptr; unknown = 0; opsize = 0; certain = 1; operand = 0; skipspace(); c = *srcptr; mode = ADRMODE_IMM; switch (toupper(c)) { case 'X' : /* "X"? */ scanname(); if (!strcmp(unamebuf, "X")) /* if it's "X" alone, */ goto XWithout; /* assume it means "0,X" */ srcptr = s; /* else restore current offset */ goto dodefault; /* and treat as label starting with X*/ case ',': /* ","? */ Indexed : srcptr++; /* must be followed by "X" */ if (!(dwOptions & OPTION_TSC)) skipspace(); scanname(); if (strcmp(unamebuf, "X")) /* if it's NOT "X" alone, */ { error |= ERR_ILLEGAL_ADDR; break; } XWithout : if ((unsigned)operand < 256) mode = ADRMODE_IDX; else error |= ERR_ILLEGAL_ADDR; break; case '#': srcptr++; operand = scanexpr(0, pp); break; case '<': srcptr++; if (*srcptr == '<') { srcptr++; opsize = 1; } else opsize = 2; goto dodefault; case '>': srcptr++; opsize = 3; /* fall thru on purpose */ default: dodefault: operand = scanexpr(0, pp); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') goto Indexed; else { if (opsize == 0) { if (unknown || !certain || (unsigned short)(operand) >= 256) opsize = 3; else opsize = 2; } if (opsize == 1) opsize = 2; mode = opsize - 1; } } if (pass > 1 && unknown) error |= ERR_LABEL_UNDEF; } /*****************************************************************************/ /* writerelhdr : writes a FLEX Relocatable Format header */ /*****************************************************************************/ void writerelhdr(char wcommon) { int i; #if 0 struct /* Header Layout */ { unsigned char Signature; /* Signature (always $03) */ unsigned char flags1; /* Flags 1 */ unsigned datasize; /* size of binary data */ unsigned char unknown1[4]; /* unknown data */ unsigned short extsize; /* size of External table */ unsigned startaddr; /* program start address */ unsigned char unknown2[2]; /* unknown data */ unsigned short globalsize; /* size of global table */ unsigned char unknown3[2]; /* unknown data */ unsigned short namesize; /* length of module name */ unsigned char flags2; /* Flags 2 */ unsigned char unknown4[3]; /* unknown data; filler? */ } hdr; memset(&hdr, 0, sizeof(hdr)); #endif fputc(0x04, objfile); /* write signature */ if (wcommon) /* if writing common data, */ fputc(0x18, objfile); /* Flags1 = $18 */ else if (!absmode) /* if writing Relative data, */ fputc(0x10, objfile); /* Flags1 = $10 */ else /* if writing Absolute data, */ fputc(0x12, objfile); /* Flags1 = $12 */ if (wcommon) /* if writing common data */ { /* write size of data */ // fputc((unsigned char)((commonsym->value >> 24) & 0xff), objfile); fputc((unsigned char)((commonsym->value >> 16) & 0xff), objfile); fputc((unsigned char)((commonsym->value >> 8) & 0xff), objfile); fputc((unsigned char)(commonsym->value & 0xFF), objfile); } else /* otherwise */ { /* write size of binary data */ // fputc((unsigned char)((loccounter >> 24) & 0xff), objfile); fputc((unsigned char)((loccounter >> 16) & 0xff), objfile); fputc((unsigned char)((loccounter >> 8) & 0xff), objfile); fputc((unsigned char)(loccounter & 0xFF), objfile); } //fputc(0, objfile); /* unknown data */ fputc(0, objfile); fputc(0, objfile); fputc(0, objfile); if (wcommon) /* if writing common data */ { /* external table is empty */ fputc(0, objfile); fputc(0, objfile); } else { int extrel = 0; /* # external relocation records */ for (i = 0; i < relcounter; i++) /* calc # external symbols */ if ((reltable[i].sym->cat == SYMCAT_EXTERN) || (reltable[i].sym->cat == SYMCAT_COMMON)) extrel++; /* then calculate table size */ extrel = (extrel * 8) + (relcounter * 3); /* and write it out */ fputc((unsigned char)(extrel >> 8), objfile); fputc((unsigned char)(extrel & 0xFF), objfile); } if (wcommon || /* if writing common data */ (!tfradrset)) /* or no transfer address given */ { /* start address is empty */ // fputc(0, objfile); fputc(0, objfile); fputc(0, objfile); fputc(0, objfile); } else /* write transfer address */ { // fputc((unsigned char)((tfradr >> 24) & 0xff), objfile); fputc((unsigned char)((tfradr >> 16) & 0xff), objfile); fputc((unsigned char)((tfradr >> 8) & 0xff), objfile); fputc((unsigned char)(tfradr & 0xFF), objfile); } fputc(0, objfile); /* unknown data */ fputc(0, objfile); if (wcommon) { /* always 1 Global - the LABEL */ fputc(0, objfile); fputc(12, objfile); } else /* calculate & write out global size */ { int globals = 0; for (i = 0; i < symcounter; i++) if (symtable[i].cat == SYMCAT_PUBLIC) globals++; globals *= 12; fputc((unsigned char)(globals >> 8), objfile); fputc((unsigned char)(globals & 0xFF), objfile); } fputc(0, objfile); /* unknown data */ fputc(0, objfile); if (wcommon) { /* no name size yet ... */ size_t len; char name[9] = ""; sprintf(name, "%-.8s", commonsym->name); strupr(name); len = strlen(name); if (len) /* if there, */ len++; /* append a $04 */ fputc((unsigned char)(len >> 8), objfile); fputc((unsigned char)(len & 0xFF), objfile); } else /* write out module name size */ { size_t len = strlen(modulename); if (len) /* if there, */ len++; /* append a $04 */ fputc((unsigned char)(len >> 8), objfile); fputc((unsigned char)(len & 0xFF), objfile); } if ((!wcommon) && (tfradrset)) /* if transfer address set */ fputc(0x80, objfile); /* write $80 flag */ else fputc(0, objfile); fputc(0, objfile); /* unknown data */ fputc(0, objfile); fputc(0, objfile); } /*****************************************************************************/ /* writerelcommon : writes out all common blocks */ /*****************************************************************************/ void writerelcommon() { int i, j; char name[9]; /* work through symbol list */ for (i = 0; i < symcounter; i++) { if (symtable[i].cat == SYMCAT_COMMON) /* if that is a common block */ { commonsym = symtable + i; /* write it out */ writerelhdr(1); /* then write the global definition */ sprintf(name, "%-8.8s", symtable[i].name); strupr(name); fwrite(name, 1, 8, objfile); if (symtable[i].value >= 65536) fputc(2, objfile); /* unknown data */ else fputc(1, objfile); /* unknown data */ // fputc((unsigned char)(symtable[i].value >> 24), objfile); fputc((unsigned char)(symtable[i].value >> 16), objfile); fputc((unsigned char)(symtable[i].value >> 8), objfile); fputc((unsigned char)(symtable[i].value & 0xFF), objfile); fputc(0x13, objfile); /* unknown flag */ /* then write the Common name */ sprintf(name, "%-.8s", symtable[i].name); strupr(name); for (j = 0; name[j]; j++) fputc(name[j], objfile); fputc(0x04, objfile); j = (int)ftell(objfile); /* fill last sector with zeroes */ while (j % 252) { fputc(0, objfile); j++; } } } } /*****************************************************************************/ /* writerelext : writes out a FLEX Relocatable External table */ /*****************************************************************************/ void writerelext() { int i; char name[9]; unsigned char flags; for (i = 0; i < relcounter; i++) /* write out the external data */ { // fputc((unsigned char)(reltable[i].addr >> 24), objfile); fputc((unsigned char)(reltable[i].addr >> 16), objfile); fputc((unsigned char)(reltable[i].addr >> 8), objfile); fputc((unsigned char)(reltable[i].addr & 0xFF), objfile); flags = 0x00; /* reset flags */ if (reltable[i].exprcat & EXPRCAT_NEGATIVE) flags |= 0x20; /* eventually add subtraction flag */ if ((reltable[i].sym->cat == SYMCAT_EXTERN) || (reltable[i].sym->cat == SYMCAT_COMMON)) flags |= 0x80; /* eventually add External flag */ fputc(flags, objfile); /* write the flag bytes */ if (flags & 0x80) /* eventually write external symbol */ { sprintf(name, "%-8.8s", reltable[i].sym->name); strupr(name); fwrite(name, 1, 8, objfile); } } } /*****************************************************************************/ /* writerelglobal : writes out FLEX Relocatable Global table */ /*****************************************************************************/ void writerelglobal() { int i; char name[9]; for (i = 0; i < symcounter; i++) /* write out the global data */ { if (symtable[i].cat == SYMCAT_PUBLIC) { sprintf(name, "%-8.8s", symtable[i].name); strupr(name); fwrite(name, 1, 8, objfile); fputc(0, objfile); /* unknown data */ // fputc((unsigned char)(symtable[i].value >> 24), objfile); fputc((unsigned char)(symtable[i].value >> 16), objfile); fputc((unsigned char)(symtable[i].value >> 8), objfile); fputc((unsigned char)(symtable[i].value & 0xFF), objfile); fputc(0x02, objfile); /* unknown flag */ } } } /*****************************************************************************/ /* writerelmodname : writes out FLEX Relocatable Module Name */ /*****************************************************************************/ void writerelmodname() { int i; if (!modulename[0]) return; strupr(modulename); for (i = 0; modulename[i]; i++) fputc(modulename[i], objfile); fputc(0x04, objfile); } /*****************************************************************************/ /* flushver : write verilog */ /*****************************************************************************/ int calcParity(unsigned wd) { int nn; int bit; int par; par = 0; for (nn = 0; nn < 32; nn++) { bit = (wd >> nn) & 1; par = par ^ bit; } return par; } void flushver() { int i; int chk; if (hexcount) { if (objfile) { for (i = 0; i < hexcount; i++) fprintf(objfile, "rommem[%5d] <= 12'h%03X;\r\n", (hexaddr+i) & 0x3fff, hexbuffer[i] & 0xfff); hexaddr += hexcount; hexcount = 0; chksum = 0; vercount++; } } } /*****************************************************************************/ /* flushhex : write Motorola s-records */ /* Flushes using 12-bit bytes */ /*****************************************************************************/ void flushhex() { int i; if (hexcount) { if (objfile) { fprintf(objfile, "S1%03X%06X", (hexcount + 3) & 0xfff, hexaddr & 0xffffff); for (i = 0; i < hexcount; i++) fprintf(objfile, "%03X", hexbuffer[i] & 0xfff); chksum += (hexaddr & 0xfff) + ((hexaddr >> 12) & 0xfff) + hexcount + 3; fprintf(objfile, "%03X\n", 0xfff - (chksum & 0xfff)); } hexaddr += hexcount; hexcount = 0; chksum = 0; } } /*****************************************************************************/ /* flushihex : write Intel hex record */ /*****************************************************************************/ void flushihex() { int i; unsigned char *j; if (hexcount) { if (objfile) { j = &hexbuffer[0]; fprintf(objfile, ":%03X%06X00", hexcount, hexaddr & 0xffffff); chksum = hexcount + ((hexaddr >> 12) & 0xfff) + (hexaddr & 0xfff); for (i = 0; i < hexcount; i++, j++) { chksum += (*j) & 0xfff; fprintf(objfile, "%03X", (*j) & 0xfff); } fprintf(objfile, "%03X\n", (-(signed)chksum) & 0xfff); } hexaddr += hexcount; hexcount = 0; chksum = 0; } } /*****************************************************************************/ /* flushflex : write FLEX binary record */ /*****************************************************************************/ void flushflex() { int i; unsigned short *j; if (hexcount) { j = &hexbuffer[0]; if (objfile) { fputc(0x02, objfile); /* start of record indicator */ fputc((hexaddr >> 32) & 0xf, /* load address high part */ objfile); fputc((hexaddr >> 24) & 0xff, /* load address high part */ objfile); fputc((hexaddr >> 20) & 0xf, /* load address high part */ objfile); fputc((hexaddr >> 12) & 0xff, /* load address high part */ objfile); fputc((hexaddr >> 8) & 0xf, objfile); /* load address low part */ fputc(hexaddr & 0xff, objfile); /* load address low part */ fputc((hexcount >> 8) & 0xf, objfile); /* # following data bytes */ fputc(hexcount & 0xff, objfile); /* # following data bytes */ for (i = 0; i < hexcount; i++, j++) { /* then put all data bytes */ fputc(((*j)>>8)&0xff, objfile); fputc(*j, objfile); } } hexaddr += hexcount; /* set new address */ hexcount = 0; /* reset counter */ } } /*****************************************************************************/ /* outver : add a byte to verilog output */ /*****************************************************************************/ void outver (unsigned short x) { if (hexcount==4) flushver(); hexbuffer[hexcount] = x; hexcount++; chksum += x; } /*****************************************************************************/ /* outhex : add a byte to motorola s-record output */ /*****************************************************************************/ void outhex (unsigned short x) { if (hexcount == 16) flushhex(); hexbuffer[hexcount++] = x; chksum += x; } /*****************************************************************************/ /* outihex : add a byte to intel hex output */ /*****************************************************************************/ void outihex (unsigned short x) { if (hexcount == 32) flushihex(); hexbuffer[hexcount++] = x; chksum += x; } /*****************************************************************************/ /* outflex : adds a byte to FLEX output */ /*****************************************************************************/ void outflex(unsigned short x) { if (hexcount == 255) /* if buffer full */ flushflex(); /* flush it */ hexbuffer[hexcount++] = x; /* then put byte into buffer */ } /*****************************************************************************/ /* outbyte : writes one byte to the output in the selected format */ /*****************************************************************************/ void outbyte(unsigned short uc, int off) { int nByte = (loccounter + off) / 8; unsigned short nBitMask = (unsigned short) (1 << ((loccounter + off) % 12)); //if (bUsedBytes[nByte] & nBitMask) /* if address already used */ // warning |= WRN_AREA; /* set warning code */ //else /* otherwise */ // bUsedBytes[nByte] |= nBitMask; /* mark it as used */ switch (outmode) { case OUT_BIN : /* binary file */ fputc((uc >> 8) & 0xf, objfile); fputc(uc, objfile); break; case OUT_REL : /* FLEX Relocatable */ fputc((uc >> 8) & 0xf, objfile); fputc(uc, objfile); break; case OUT_SREC : /* Motorola S-records */ outhex(uc); break; case OUT_IHEX : /* Intel Hex */ outihex(uc); break; case OUT_FLEX : /* FLEX */ outflex(uc); break; case OUT_VER: outver(uc); break; } } /*****************************************************************************/ /* outbuffer : writes the output to a file in the selected format */ /*****************************************************************************/ void outbuffer() { int i; for (i = 0; i < codeptr; i++) outbyte(codebuf[i], i); } /*****************************************************************************/ /* report : reports an error */ /*****************************************************************************/ void report() { int i; for (i = 0; i < 16; i++) { if (error & 1) { printf("%s(%ld) : error %d: %s in \"%s\"\n", expandfn(curline->fn), curline->ln, i + 1, errormsg[i], curline->txt); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error %d: %s\n", i + 1, errormsg[i]); errors++; } error >>= 1; } if ((dwOptions & OPTION_TSC)) /* suppress warning 1 in TSC mode */ warning &= ~WRN_OPT; if (!(dwOptions & OPTION_WAR)) /* reset warnings if not wanted */ warning = WRN_OK; for (i = 0; i < 16; i++) { if (warning & 1) { printf("%s(%ld) : warning %d: %s in \"%s\"\n", expandfn(curline->fn), curline->ln, i + 1, warningmsg[i], curline->txt); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** warning %d: %s\n", i + 1, warningmsg[i]); warnings++; } warning >>= 1; } } /*****************************************************************************/ /* outlist : lists the code bytes for an instruction */ /*****************************************************************************/ void outlist(struct oprecord *op) { int i; if ((curline->lvl & LINCAT_INVISIBLE) &&/* don't list invisible lines */ !(dwOptions & OPTION_INV)) return; if ((curline->lvl & LINCAT_MACEXP) && /* don't list macro expansions if */ !(dwOptions & OPTION_EXP)) /* not explicitly requested */ return; if ((curline->lvl & LINCAT_LVLMASK) && /* if level 1..31 */ !(dwOptions & OPTION_LLL)) /* and this is not to be listed */ return; if ((suppress || nSkipCount) && /* if this is conditionally skipped */ !(dwOptions & OPTION_CON)) /* and this is not to be listed */ return; if ((condline) && /* if this is a condition line */ !(dwOptions & OPTION_CON)) /* and this is not to be listed */ return; if (dwOptions & OPTION_NUM) /* if number output */ putlist("%4d ", curline->ln); /* print out the line number */ if (!absmode) /* if in relocating assembler mode */ putlist("%c", curline->rel); /* output relocation information */ if (dwOptions & OPTION_LPA) /* if in patch mode */ { if ((op) && (op->cat == OPCAT_PSEUDO)) { switch (op->code) { case PSEUDO_SETDP : if (dpsetting >= 0) putlist("setdp %02X\n", dpsetting); break; case PSEUDO_ORG : putlist("insert %08X \\ ORG $%08X\n", loccounter, loccounter); break; case PSEUDO_FCB : case PSEUDO_FCC : putlist("data %04X", oldlc); if (codeptr > 1) putlist("-%04X", oldlc + codeptr - 1); putlist("\n"); break; case PSEUDO_FCW : putlist("word %04x", oldlc); if (codeptr > 2) putlist("-%04X", oldlc + codeptr - 1); putlist("\n"); break; case PSEUDO_FCDW : putlist("dword %08x", oldlc); if (codeptr > 2) putlist("-%08X", oldlc + codeptr - 1); putlist("\n"); break; } } if (codeptr > 0) /* if there are code bytes */ { char *name = findsymat(oldlc); if (name) putlist("label %04X %s\n", oldlc, name); putlist("patch "); /* write "patch" */ } else if (*curline->txt) putlist(" "); else putlist("comment %04X", oldlc); } else if ((warning & WRN_OPT) && /* excessive branch, TSC style */ (dwOptions & OPTION_TSC) && (dwOptions & OPTION_WAR)) putlist(">"); else if (curline->lvl & LINCAT_MACDEF) /* if in macro definition */ putlist("#"); /* prefix line with # */ else if (curline->lvl & LINCAT_MACEXP) /* if in macro expansion */ putlist("+"); /* prefix line with + */ else if (curline->lvl & LINCAT_INVISIBLE) putlist("-"); else if (curline->txt) /* otherwise */ putlist(" "); /* prefix line with blank */ if (codeptr > 0) putlist("%08X ", oldlc); else if (*curline->txt) putlist(" "); else { putlist( "\n"); return; } for (i = 0; i < codeptr && i < MAXLISTBYTES; i++) { if (dwOptions & OPTION_LPA) putlist("%03X ", codebuf[i]&0xfff); else putlist("%03X", codebuf[i]&0xfff); } for (; i <= MAXLISTBYTES; i++) { if (dwOptions & OPTION_LPA) putlist(" "); else putlist(" "); } if ((dwOptions & OPTION_LPA) && (*curline->txt)) putlist("* "); if (strcmp(curline->txt, srcline) && /* if text inserted */ (dwOptions & OPTION_EXP)) /* and expansion activated */ putlist("%s", curline->txt); /* just print out the source line */ else /* otherwise */ putlist("%s", srcline); /* print possibly expanded line */ putlist("\n"); /* send newline */ if (codeptr > MAXLISTBYTES && /* if there are additional bytes, */ (dwOptions & OPTION_MUL)) { /* print them. */ if (dwOptions & OPTION_LPA) /* if in patch mode */ putlist("patch"); /* write "patch" */ for (i = MAXLISTBYTES; i < codeptr; i++) { if (!(i % MAXLISTBYTES)) { if (i != MAXLISTBYTES) putlist("\n"); if (dwOptions & OPTION_NUM) /* if number output */ putlist(" "); if (!absmode) putlist(" "); putlist(" %08X ", oldlc + i); } if (dwOptions & OPTION_LPA) /* if in patch mode */ putlist("%03X ", codebuf[i]&0xfff); else putlist("%03X", codebuf[i]&0xfff); } putlist("\n"); } if (strcmp(curline->txt, srcline) && /* if text inserted */ (dwOptions & OPTION_EXP)) /* and expansion activated */ { if (dwOptions & OPTION_NUM) putlist("%4d ", curline->ln); if (!absmode) putlist(" "); putlist("+ ( %s )\n", srcline); } } /*****************************************************************************/ /* setlabel : sets a label */ /*****************************************************************************/ void setlabel(struct symrecord * lp) { if (lp) { lp->isFar = isFar; lp->isFarkw = isFarkw; if (lp->cat == SYMCAT_PUBLICUNDEF) { lp->cat = SYMCAT_PUBLIC; lp->value = loccounter; } else if (lp->cat == SYMCAT_EMPTYLOCAL) { lp->cat = SYMCAT_LOCALLABEL; } else if (lp->cat == SYMCAT_LOCALLABEL) ; else if (lp->cat != SYMCAT_EMPTY && lp->cat != SYMCAT_UNRESOLVED) { if ((lp->cat != SYMCAT_LABEL && lp->cat != SYMCAT_PUBLIC) || lp->value != loccounter) { lp->value = loccounter; if (pass==MAX_PASSNO) error |= ERR_LABEL_MULT; } } else { lp->cat = (global) ? SYMCAT_PUBLIC : SYMCAT_LABEL; lp->value = loccounter; } } } /*****************************************************************************/ /* putbyte : adds a byte to the instruction code buffer */ /*****************************************************************************/ void putbyte(unsigned short b) { codebuf[codeptr++] = b; /* and finally put the byte there. */ } /*****************************************************************************/ /* putword : adds a word to the instruction code buffer */ /*****************************************************************************/ void putword(unsigned w) { putbyte((unsigned short)((w >> 12L) & 0xfffL)); putbyte((unsigned short)(w & 0xfff)); } /*****************************************************************************/ /* putdword : adds a doubleword to the instruction code buffer */ /*****************************************************************************/ void putdword(unsigned __int64 d) { putbyte((unsigned short)((d >> 36LL) & 0xfffLL)); putbyte((unsigned short)((d >> 24LL) & 0xfffLL)); putbyte((unsigned short)((d >> 12LL) & 0xfffLL)); putbyte((unsigned short)(d & 0xfffLL)); } /*****************************************************************************/ /* doaddress : assemble the right addressing bytes for an instruction */ /*****************************************************************************/ void doaddress(struct relocrecord *p) { int offs; int addrelocation = 0; int isNear = 0; if (p) /* create relocation record */ p->addr = (unsigned)(loccounter + codeptr); switch (mode) { case ADRMODE_IMM : if (opsize == 2) putbyte((unsigned short)operand); else if (opsize == 5) /* LDQ special */ putdword(operand); else { putword((unsigned)operand); addrelocation = 1; } break; case ADRMODE_DIR : putbyte((unsigned short)operand); break; case ADRMODE_EXT : if (((codebuf[0] == 0x07e) || /* special for JMP */ (codebuf[0] == 0x0bd) || (codebuf[0] == 0x015 && (codebuf[1]==0x07E || codebuf[1]==0x0bd)) ) && /* and JSR */ pass > 1) { int nDiff = (int)operand - (int)loccounter - 3; isNear = (operand & 0xFF0000L) == (loccounter & 0xff0000L); if (((nDiff & 0xff80) == 0x0000) || ((nDiff & 0xff80) == 0xff80)) warning |= (certain) ? WRN_OPT : 0; } if (codebuf[0]==0x015 && codebuf[1]==0x0bd && isFar) { if (isNear && !isFarkw) { codebuf[0]=0x0bd; isFar = 0; } else codebuf[0]=0x0cf; codeptr--; } if (codebuf[0]==0x015 && codebuf[1]==0x07E && isFar) { if (isNear && !isFarkw) { codebuf[0]=0x07E; isFar = 0; } else codebuf[0]=0x08f; codeptr--; } if (isFar) { putbyte((unsigned short)(operand >> 24)); } putword((unsigned)operand); addrelocation = 1; break; case ADRMODE_IDX : putbyte((unsigned short)operand); break; case ADRMODE_POST : case ADRMODE_IND : case ADRMODE_DBL_IND: putbyte(postbyte); switch (opsize) { case 2: putbyte((unsigned short)operand); break; case 3: putword((unsigned)operand); addrelocation = 1; break; case 4: putbyte((unsigned short)(operand>>24)); putword((unsigned)operand); addrelocation = 1; break; case 5: putword((unsigned short)(operand>>24)); putword((unsigned)operand); addrelocation = 1; break; } break; case ADRMODE_PCR : case ADRMODE_PIN : offs = (unsigned)operand - loccounter - codeptr - 2; if (offs < -2048 || offs >= 2048 || opsize == 3 || unknown || !certain) { if ((!unknown) && opsize == 2) error |= ERR_RANGE; offs--; opsize = 3; postbyte++; } putbyte(postbyte); if (opsize == 3) { putword((unsigned)offs); addrelocation = 1; } else putbyte((unsigned short)offs); } if (addrelocation) addreloc(p); } /*****************************************************************************/ /* onebyte : saves integer as one instruction byte */ /*****************************************************************************/ void onebyte(int co) { putbyte((unsigned short)co); } /*****************************************************************************/ /* twobyte : saves integer as two instruction bytes */ /*****************************************************************************/ void twobyte(int co) { putword((unsigned)co); } /*****************************************************************************/ /* threebyte : saves long integer as three instruction bytes */ /*****************************************************************************/ void threebyte(unsigned __int64 co) { putbyte((unsigned short)((co >> 24LL) & 0xfffLL)); putbyte((unsigned short)((co >> 12LL) & 0xfffLL)); putbyte((unsigned short)(co & 0xfffL)); } /*****************************************************************************/ /* fourbyte : saves long integer as four instruction bytes */ /*****************************************************************************/ void fourbyte(unsigned __int64 co) { putbyte((unsigned short)((co >> 36LL) & 0xfffLL)); putbyte((unsigned short)((co >> 24LL) & 0xfffLL)); putbyte((unsigned short)((co >> 12LL) & 0xfffLL)); putbyte((unsigned short)(co & 0xfffL)); } /*****************************************************************************/ /* oneimm : saves one immediate value */ /*****************************************************************************/ void oneimm(int co) { struct relocrecord p = {0}; scanoperands(&p); if (mode >= ADRMODE_POST) error |= ERR_ILLEGAL_ADDR; putbyte((unsigned short)co); /* addreloc(0, 2, p); */ /* no relocation for immediate op's */ putbyte((unsigned short)operand); } /*****************************************************************************/ /* lea : */ /*****************************************************************************/ void lea(int co) { struct relocrecord p = {0}; scanoperands(&p); if (isFar) onebyte(0x015); onebyte((unsigned short)co); if (mode == ADRMODE_IMM) error |= ERR_ILLEGAL_ADDR; if (mode < ADRMODE_POST) { if (isFar) opsize = 4; else opsize = 3; postbyte = 0x80f; mode = ADRMODE_POST; } doaddress(&p); } /*****************************************************************************/ /* sbranch : processes a short branch */ /*****************************************************************************/ void sbranch(int co) { struct relocrecord p = {0}; int offs; scanoperands(&p); if (mode != ADRMODE_DIR && mode != ADRMODE_EXT) error |= ERR_ILLEGAL_ADDR; offs = operand - loccounter - 2; if (!unknown && (offs < -2048 || offs >= 2048)) error |= ERR_RANGE; if (pass > 1 && unknown) error |= ERR_LABEL_UNDEF; putbyte((unsigned short)co); putbyte((unsigned short)offs); } /*****************************************************************************/ /* lbra : does a long branch */ /*****************************************************************************/ void lbra(int co) { struct relocrecord p = {0}; int nDiff; scanoperands(&p); if (mode != ADRMODE_DIR && mode != ADRMODE_EXT) error |= ERR_ILLEGAL_ADDR; putbyte((unsigned short)co); nDiff = operand - loccounter - 3; putword((unsigned)nDiff); if (((nDiff & 0xff8000) == 0x000000) || ((nDiff & 0xff8000) == 0xff8000)) warning |= (certain) ? WRN_OPT : 0; } /*****************************************************************************/ /* lbranch : does a long branch */ /*****************************************************************************/ void lbranch(int co) { struct relocrecord p = {0}; int nDiff; scanoperands(&p); if (mode != ADRMODE_DIR && mode != ADRMODE_EXT) error |= ERR_ILLEGAL_ADDR; putword((unsigned)co); nDiff = operand - loccounter - 4; putword((unsigned)nDiff); if (((nDiff & 0xff8000) == 0x000000) || ((nDiff & 0xff8000) == 0xff8000)) warning |= (certain) ? WRN_OPT : 0; } /*****************************************************************************/ /* arith : process arithmetic operation */ /*****************************************************************************/ void arith(int co, char noimm) { struct relocrecord p = {0}; scanoperands(&p); switch (mode) { case ADRMODE_IMM : if (noimm) error |= ERR_ILLEGAL_ADDR; opsize = 2; putbyte((unsigned short)co); break; case ADRMODE_DIR : putbyte((unsigned short)(co + 0x010)); break; case ADRMODE_EXT : if (isFar) putbyte(0x015); putbyte((unsigned short)(co + 0x030)); break; default: if (isFar) putbyte(0x015); if (isPostIndexed) { //putbyte(0x01B); postbyte |= 0x80; } putbyte((unsigned short)(co + 0x020)); } doaddress(&p); } /*****************************************************************************/ /* accarith : process arithmetic operation with explicit accumulator */ /*****************************************************************************/ void accarith(int co, char noimm, char ignore) { char *s = srcptr; /* remember current offset */ char correct = 1; /* flag whether correct */ skipspace(); /* skip space */ scanname(); /* get following name */ if (strcmp(unamebuf, "A") && /* has to be followed by A or B */ strcmp(unamebuf, "B")) correct = 0; #if 1 if (*srcptr == ',') /* if directly followed by a comma */ srcptr++; /* skip it */ #else if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr++ != ',') /* and a comma */ correct = 0; #endif if (!correct) /* if NOT followed by "A," or "B," */ { if (ignore) /* if ignoring that, */ srcptr = s; /* go back to parm start */ else /* otherwise */ error |= ERR_EXPR; /* flag as an error */ } else if (unamebuf[0] == 'B') /* eventually transform to acc.B */ co |= 0x40; arith(co, noimm); /* then process as arithmetic */ } /*****************************************************************************/ /* idxextarith : indexed / extended arithmetic, 6800 style */ /*****************************************************************************/ void idxextarith(int co) { struct relocrecord p = {0}; scanoperands(&p); switch (mode) { case ADRMODE_IMM : error |= ERR_ILLEGAL_ADDR; opsize = 3; putbyte((unsigned short)co); break; case ADRMODE_DIR : mode = ADRMODE_EXT; /* implicitly convert to extended */ putbyte((unsigned short)(co + 0x030)); break; case ADRMODE_EXT : if (isFar) putbyte(0x015); putbyte((unsigned short)(co + 0x030)); break; default: if (isFar) putbyte(0x015); if (isPostIndexed) //putbyte(0x01B); putbyte((unsigned short)(co + 0x020)); break; } doaddress(&p); } /*****************************************************************************/ /* darith : process direct arithmetic */ /*****************************************************************************/ void darith(int co, char noimm, char tgt) { struct relocrecord p = {0}; scanoperands(&p); switch (mode) { case ADRMODE_IMM : if (noimm) error |= ERR_ILLEGAL_ADDR; if (isX32 && tgt) opsize = 5; else opsize = 3; putbyte((unsigned short)co); break; case ADRMODE_DIR : putbyte((unsigned short)(co + 0x010)); break; case ADRMODE_EXT : if (isFar) putbyte(0x015); putbyte((unsigned short)(co + 0x030)); break; default: if (isFar) putbyte(0x015); if (isPostIndexed) { //putbyte(0x01B); postbyte |= 0x80; } putbyte((unsigned short)(co + 0x020)); break; } doaddress(&p); } /*****************************************************************************/ /* d2arith : process direct word arithmetic */ /*****************************************************************************/ void d2arith(int co, char noimm, char tgt) { struct relocrecord p = {0}; scanoperands(&p); switch (mode) { case ADRMODE_IMM : if (noimm) error |= ERR_ILLEGAL_ADDR; if (isX32 && tgt) opsize = 5; else opsize = 3; putword((unsigned)co); break; case ADRMODE_DIR : putword((unsigned)(co + 0x010)); break; case ADRMODE_EXT : if (isFar) { putbyte(0x015); } putword((unsigned)(co + 0x030)); break; default: if (isFar) putbyte(0x015); if (isPostIndexed) { //putbyte(0x01B); postbyte |= 0x80; } putword((unsigned)(co + 0x020)); } doaddress(&p); } /*****************************************************************************/ /* qarith : process direct doubleword arithmetic */ /*****************************************************************************/ void qarith(int co, char noimm) { struct relocrecord p = {0}; scanoperands(&p); switch (mode) { case ADRMODE_IMM : if (noimm) error |= ERR_ILLEGAL_ADDR; opsize = 5; putbyte((unsigned short)0xcd); /* this can ONLY be LDQ! */ break; case ADRMODE_DIR : putword((unsigned)(co + 0x010)); break; case ADRMODE_EXT : putword((unsigned)(co + 0x030)); break; default: putword((unsigned)(co + 0x020)); break; } doaddress(&p); } /*****************************************************************************/ /* oneaddr : */ /*****************************************************************************/ void oneaddr(int co) { struct relocrecord p = {0}; char *s; skipspace(); if ((dwOptions & OPTION_TSC) && /* if TSC mode, check for 6800 */ (co != 0x0e)) /* convenience things */ { s = srcptr; scanname(); /* look whether followed by A or B */ if (((!strcmp(unamebuf, "A")) || (!strcmp(unamebuf, "B"))) && (*srcptr != ',')) /* and NO comma */ { /* if so, replace by 6809 mnemonic */ warning |= WRN_AMBIG; /* ... but not without a warning */ onebyte(co | ((unamebuf[0] == 'A') ? 0x40 : 0x50)); return; } srcptr = s; } scanoperands(&p); switch (mode) { case ADRMODE_IMM : error |= ERR_ILLEGAL_ADDR; break; case ADRMODE_DIR : if ((dwOptions & OPTION_M00) && /* on MC6800, a DIRect JMP is not OK */ (co == 0x0e)) error |= ERR_ILLEGAL_ADDR; else putbyte((unsigned short)co); break; case ADRMODE_EXT : if (isFar) putbyte(0x015); putbyte((unsigned short)(co + 0x70)); break; default: if (isFar) putbyte(0x015); if (isPostIndexed) { //putbyte(0x01B); postbyte |= 0x80; } putbyte((unsigned short)(co + 0x60)); break; } doaddress(&p); } /*****************************************************************************/ /* accaddr : */ /*****************************************************************************/ void accaddr(int co) { struct relocrecord p = {0}; char *s; skipspace(); if (dwOptions & OPTION_TSC) /* if TSC mode, check for 6800 */ { s = srcptr; scanname(); /* look whether followed by A or B */ if (((!strcmp(unamebuf, "A")) || (!strcmp(unamebuf, "B"))) && (*srcptr != ',')) /* and NO comma */ { /* if so, replace */ onebyte(co | ((unamebuf[0] == 'A') ? 0x40 : 0x50)); return; } srcptr = s; } scanoperands(&p); switch (mode) { case ADRMODE_IMM : error |= ERR_ILLEGAL_ADDR; break; case ADRMODE_DIR : mode = ADRMODE_EXT; /* silently convert to extended */ putbyte((unsigned short)(co + 0x70)); break; case ADRMODE_EXT : if (isFar) putbyte(0x15); putbyte((unsigned short)(co + 0x70)); break; default: if (isFar) putbyte(0x15); if (isPostIndexed) { //putbyte(0x1B); postbyte |= 0x80; } putbyte((unsigned short)(co + 0x60)); break; } doaddress(&p); } /*****************************************************************************/ /* tfrexg : */ /*****************************************************************************/ void tfrexg(int co) { struct regrecord * p; putbyte((unsigned char)co); skipspace(); scanname(); if (dwOptions & OPTION_H63) p = findreg63(unamebuf); else p = findreg(unamebuf); if (!p) error |= ERR_ILLEGAL_ADDR; else postbyte = (p->tfr) << 4; skipspace(); if (*srcptr == ',') srcptr++; else error |= ERR_ILLEGAL_ADDR; skipspace(); scanname(); if ((p = findreg(unamebuf)) == 0) error |= ERR_ILLEGAL_ADDR; else postbyte |= p->tfr; putbyte(postbyte); } /*****************************************************************************/ /* pshpul : operates on PSH / PUL mnemonics */ /*****************************************************************************/ void pshpul(int co) { struct regrecord *p; struct relocrecord sp = {0}; postbyte = 0; skipspace(); if (*srcptr == '#') { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); postbyte = (unsigned short)scanexpr(0, &sp); } else do { if (*srcptr=='f' || *srcptr=='F') { if (srcptr[1]=='a' || srcptr[1]=='A') { if (srcptr[2]=='r' || srcptr[2]=='R') { if (srcptr[3]==' ' || srcptr[3]=='\t') { isFar = 1; srcptr += 4; } } } } if (*srcptr == ',') srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); scanname(); if ((p = findreg(unamebuf)) == 0) error |= ERR_ILLEGAL_ADDR; else postbyte |= p->psh; if (!(dwOptions & OPTION_TSC)) skipspace(); } while (*srcptr == ','); if (isFar) putbyte((unsigned short)0x15); putbyte((unsigned short)co); putbyte(postbyte); } /*****************************************************************************/ /* bitdirect : */ /*****************************************************************************/ void bitdirect(int co) { struct relocrecord p = {0}; unsigned short dir; skipspace(); if (*srcptr++ != '#') error |= ERR_EXPR; dir = (unsigned)scanexpr(0, &p); if (dir & 0xfff000) error |= ERR_EXPR; if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr++ != ',') error |= ERR_EXPR; scanoperands(&p); switch (mode) { case ADRMODE_IMM : error |= ERR_ILLEGAL_ADDR; break; case ADRMODE_DIR : putbyte((unsigned short)co); break; case ADRMODE_EXT : if (isFar) putbyte(0x15); putbyte((unsigned short)(co + 0x70)); break; default: if (isFar) putbyte(0x15); putbyte((unsigned short)(co + 0x60)); break; } putbyte((unsigned short)dir); doaddress(&p); } /*****************************************************************************/ /* bittrans : */ /*****************************************************************************/ void bittrans(int co) { struct regrecord *p; struct relocrecord rp = {0}; long t; putword((unsigned)co); skipspace(); scanname(); if ((p = findbitreg(unamebuf)) == 0) error |= ERR_ILLEGAL_ADDR; else postbyte = (p->tfr) << 6; if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') srcptr++; else error |= ERR_ILLEGAL_ADDR; if (!(dwOptions & OPTION_TSC)) skipspace(); t = scanfactor(&rp); if (t & 0xfffff8) error |= ERR_ILLEGAL_ADDR; else postbyte |= (t << 3); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') srcptr++; else error |= ERR_ILLEGAL_ADDR; t = scanfactor(&rp); if (t & 0xfffff8) error |= ERR_ILLEGAL_ADDR; else postbyte |= t; putbyte((unsigned short)postbyte); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') srcptr++; else error |= ERR_ILLEGAL_ADDR; scanoperands(&rp); switch (mode) { case ADRMODE_DIR : putbyte((unsigned short)operand); break; default: error |= ERR_ILLEGAL_ADDR; } } /*****************************************************************************/ /* blocktrans : */ /*****************************************************************************/ void blocktrans(int co) { char reg1,reg2; char mode[3] = ""; static char regnames[] = "DXYUS"; static char *modes[] = { "++", "--", "+", ",+" }; int i; skipspace(); reg1 = toupper(*srcptr); for (i = 0; regnames[i]; i++) if (reg1 == regnames[i]) break; if (!regnames[i]) error |= ERR_ILLEGAL_ADDR; else reg1 = i; mode[0] = *++srcptr; if ((mode[0] != '+') && (mode[0] != '-')) { if (!(dwOptions & OPTION_TSC)) skipspace(); mode[0] = *srcptr; } else srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr != ',') error |= ERR_ILLEGAL_ADDR; srcptr++; reg2 = toupper(*srcptr); for (i = 0; regnames[i]; i++) if (reg2 == regnames[i]) break; if (!regnames[i]) error |= ERR_ILLEGAL_ADDR; else reg2 = i; mode[1] = *++srcptr; if ((mode[1] != '+') && (mode[1] != '-')) { if (!(dwOptions & OPTION_TSC)) skipspace(); mode[1] = *srcptr; } else srcptr++; if ((mode[1] == ';') || (mode[1] == '*') || (mode[1] == ' ') || (mode[1] == '\t')) mode[1] = '\0'; for (i = 0; i < (sizeof(modes) / sizeof(modes[0])); i++) if (!strcmp(mode, modes[i])) break; if (i >= (sizeof(modes) / sizeof(modes[0]))) error |= ERR_ILLEGAL_ADDR; else co |= i; putword((unsigned)co); putbyte((unsigned short)((reg1 << 4) | reg2)); } /*****************************************************************************/ /* expandline : un-tabify current line */ /*****************************************************************************/ void expandline() { int i, j = 0, k, j1; for (i = 0; i < LINELEN && j < LINELEN; i++) { if (inpline[i] == '\n') { srcline[j] = 0; break; } else if (inpline[i] == '\t') { j1 = j; for (k = 0; k < 8 - j1 % 8 && j < LINELEN; k++) srcline[j++] = ' '; } else if (inpline[i] == '\r') { continue; } else srcline[j++] = inpline[i]; } srcline[LINELEN - 1] = 0; } /*****************************************************************************/ /* expandtext : expands all texts in a line */ /*****************************************************************************/ void expandtext() { char *p; int i, j = 0; int doit = 1; for (p = curline->txt; (*p) && (j < LINELEN); ) { if (*p == '\"') doit = !doit; if (*p == '\\' && p[1] == '&') srcline[j++] = *(++p); else if (*p == '&' && (p[1] < '0' || p[1] > '9') && doit) { struct symrecord *lp; srcptr = p + 1; scanname(); lp = findsym(namebuf, 0); if ((lp) && (*namebuf) && /* if symbol IS a text constant, */ (lp->cat == SYMCAT_TEXT)) { /* insert its content */ p = srcptr; for (i = 0; j < LINELEN && texts[lp->value][i]; i++) srcline[j++] = texts[lp->value][i]; } else /* otherwise */ srcline[j++] = *p++; /* simply use the '&' and go on. */ } else srcline[j++] = *p++; } srcline[j >= LINELEN ? LINELEN - 1 : j] = '\0'; } /*****************************************************************************/ /* readfile : reads in a file and recurses through includes */ /*****************************************************************************/ struct linebuf *readfile(char *name, unsigned char lvl, struct linebuf *after) { FILE *srcfile; struct linebuf *pNew; int lineno = 0; int i; int nfnidx = -1; for (i = 0; i < nfnms; i++) /* prohibit recursion */ if (!strcmp(name, fnms[i])) { nfnidx = i; break; } if (nfnidx < 0) { if (nfnms >= (sizeof(fnms) / sizeof(fnms[0]))) { printf("%s(0) : error 21: nesting level too deep\n", name); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 21: nesting level too deep\n"); exit(4); } nfnidx = nfnms++; } if ((srcfile = fopen(name, "r")) == 0) { printf("%s(0) : error 17: cannot open source file\n", name); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 17: cannot open source file\n"); exit(4); } if (!fnms[nfnidx]) /* if not yet done, */ fnms[nfnidx] = strdup(name); /* remember the file name */ while (fgets(inpline, LINELEN, srcfile)) { expandline(); pNew = allocline(after, fnms[nfnidx], ++lineno, lvl, srcline); if (!pNew) { printf("%s(%d) : error 22: memory allocation error\n", name, lineno); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 22: memory allocation error\n"); exit(4); } if (!after) /* if 1st line */ rootline = pNew; /* remember it as root */ after = pNew; /* insert behind the new line */ } fclose(srcfile); /* then close the file */ return after; /* pass back last line inserted */ } /*****************************************************************************/ /* readbinary: reads in a binary file and converts to fcw / fcb lines */ /*****************************************************************************/ struct linebuf *readbinary ( char *name, unsigned char lvl, struct linebuf *after, struct symrecord *lp ) { FILE *srcfile; struct linebuf *pNew; int lineno = 0; int i; int nfnidx = -1; unsigned char binlin[16]; int binlen; int fcbstart; for (i = 0; i < nfnms; i++) /* prohibit recursion */ if (!strcmp(name, fnms[i])) { nfnidx = i; break; } if (nfnidx < 0) { if (nfnms >= (sizeof(fnms) / sizeof(fnms[0]))) { printf("%s(0) : error 21: nesting level too deep\n", name); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 21: nesting level too deep\n"); exit(4); } nfnidx = nfnms++; } if ((srcfile = fopen(name, "rb")) == 0) { printf("%s(0) : error 17: cannot open source file\n", name); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 17: cannot open source file\n"); exit(4); } if (!fnms[nfnidx]) /* if not yet done, */ fnms[nfnidx] = strdup(name); /* remember the file name */ while ((binlen = (int)fread(binlin, 1, sizeof(binlin), srcfile)) > 0) { sprintf(inpline, "%s", (lineno || (!lp)) ? "" : lp->name); i = 0; if ((binlen & (~1)) && (binlen > 7)) { sprintf(inpline + strlen(inpline), "\tFCW\t" /* , (binlen & 1) ? 'B' : 'W' */); for (; i < (binlen & ~1); i += 2) sprintf(inpline + strlen(inpline), "%s$%02X%02X", (i) ? "," : "", binlin[i], binlin[i + 1]); expandline(); pNew = allocline(after, fnms[nfnidx], ++lineno, lvl, srcline); if (!pNew) { printf("%s(%d) : error 22: memory allocation error\n", name, lineno); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 22: memory allocation error\n"); exit(4); } if (!after) /* if 1st line */ rootline = pNew; /* remember it as root */ after = pNew; /* insert behind the new line */ inpline[0] = '\0'; /* reset input line */ } if ((binlen & 1) || (binlen <= 7)) { fcbstart = i; sprintf(inpline + strlen(inpline), "\tFCB\t"); for (; i < binlen; i++) sprintf(inpline + strlen(inpline), "%s$%02X", (i > fcbstart) ? "," : "", binlin[i]); expandline(); pNew = allocline(after, fnms[nfnidx], ++lineno, lvl, srcline); if (!pNew) { printf("%s(%d) : error 22: memory allocation error\n", name, lineno); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 22: memory allocation error\n"); exit(4); } if (!after) /* if 1st line */ rootline = pNew; /* remember it as root */ after = pNew; /* insert behind the new line */ } } fclose(srcfile); /* then close the file */ return after; /* pass back last line inserted */ } /*****************************************************************************/ /* setoptiontexts : sets up the option text variables */ /*****************************************************************************/ void setoptiontexts() { int i; /* walk option list */ for (i = 0; i < (sizeof(Options) / sizeof(Options[0])); i++) { if ((dwOptions & Options[i].dwAdd) || (!(dwOptions & ~Options[i].dwRem))) settext(Options[i].Name, "1"); else settext(Options[i].Name, "0"); } } /*****************************************************************************/ /* setoption : processes an option string */ /*****************************************************************************/ int setoption ( char *szOpt ) { char iopt[4]; int i; for (i = 0; szOpt[i] && i < sizeof(iopt); i++) iopt[i] = toupper(szOpt[i]); if (i >= sizeof(iopt)) i--; iopt[i] = '\0'; /* search option list */ for (i = 0; i < (sizeof(Options) / sizeof(Options[0])); i++) { if (!strcmp(iopt, Options[i].Name)) /* if option found */ { dwOptions |= Options[i].dwAdd; /* add flags */ dwOptions &= ~Options[i].dwRem; /* and remove flags */ switch (Options[i].dwAdd) /* afterprocessing for specials: */ { case OPTION_M09 : /* switch to MC6809 processor */ optable = optable09; optablesize = sizeof(optable09) / sizeof(optable09[0]); regtable = regtable09; bitregtable = bitregtable09; scanoperands = scanoperands09; break; case OPTION_H63 : /* switch to HD6309 processor */ optable = optable09; optablesize = sizeof(optable09) / sizeof(optable09[0]); regtable = regtable63; bitregtable = bitregtable09; scanoperands = scanoperands09; break; case OPTION_M00 : /* switch to MC6800 processor */ optable = optable00; optablesize = sizeof(optable00) / sizeof(optable00[0]); regtable = regtable00; bitregtable = bitregtable00; scanoperands = scanoperands00; break; } setoptiontexts(); return 0; /* then return OK */ } } return 1; /* unknown option */ } /*****************************************************************************/ /* pseudoop : processes all known pseudo-ops */ /*****************************************************************************/ void pseudoop(int co, struct symrecord * lp) { int i, j; char c; struct relocrecord p = {0}; if (common && /* if in COMMON definition mode */ (co != PSEUDO_ENDCOM) && /* and this is neither ENDCOM */ (co != PSEUDO_RMB)) /* nor RMB */ error |= ERR_EXPR; /* this is an error. */ switch (co) { case PSEUDO_ABS : /* ABS */ absmode = 1; /* reset mode to absolute */ break; case PSEUDO_DEF : /* DEFINE */ global++; /* all labels from now on are global */ break; case PSEUDO_ENDDEF : /* ENDDEF */ if (global) /* all labels from now on are local */ global--; else error |= ERR_EXPR; /* must be paired */ break; case PSEUDO_COMMON : /* label COMMON */ if (absmode) /* if in absolute assembler mode */ error |= ERR_RELOCATING; if (!lp) error |= ERR_LABEL_MISSING; else lp->cat = SYMCAT_COMMON; common++; /* go into common mode */ commonsym = lp; /* remember common symbol */ break; case PSEUDO_ENDCOM : /* ENDCOM */ if (absmode) /* if in absolute assembler mode */ error |= ERR_RELOCATING; if (common) /* terminate common mode */ common--; else error |= ERR_EXPR; /* must be paired */ break; case PSEUDO_RMB : /* [label] RMB <absolute expression> */ case PSEUDO_RZB : /* [label] RZB <absolute expression> */ operand = scanexpr(0, &p); if (unknown) error |= ERR_LABEL_UNDEF; if (common) /* if in common mode */ { if ((lp->cat != SYMCAT_EMPTY) && (lp->cat != SYMCAT_UNRESOLVED) && (lp->cat != SYMCAT_COMMONDATA)) error |= ERR_LABEL_MULT; if (lp->cat != SYMCAT_COMMONDATA) /* if not yet done, */ { lp->cat = SYMCAT_COMMONDATA; /* set tymbol type */ lp->u.parent = commonsym; /* remember COMMON symbol */ lp->value = commonsym->value; /* remember offset from COMMON symbol*/ commonsym->value += /* append # bytes to reserve to -"- */ (unsigned short)operand; } break; } setlabel(lp); if (generating && pass == MAX_PASSNO) { if (co != 0 || outmode == OUT_BIN) for (i = 0; i < operand; i++) outbyte(0, i); else switch (outmode) { case OUT_SREC : /* Motorola S51-09 ? */ flushhex(); break; case OUT_IHEX : /* Intel Hex ? */ flushihex(); break; case OUT_FLEX : /* FLEX binary ? */ flushflex(); break; } } loccounter += operand; hexaddr = loccounter; break; case PSEUDO_EQU : /* label EQU x */ nRepNext = 0; /* reset eventual repeat */ operand = scanexpr(0, &p); if (!lp) error |= ERR_LABEL_MISSING; else { if (lp->cat == SYMCAT_EMPTY || lp->cat == SYMCAT_UNRESOLVED || (lp->value == (unsigned)operand && pass > 1)) { if (exprcat == EXPRCAT_INTADDR) lp->cat = SYMCAT_LABEL; else lp->cat = SYMCAT_CONSTANT; lp->value = (unsigned)operand; } else error |= ERR_LABEL_MULT; } break; case PSEUDO_PUB : /* PUBLIC a[,b[,c...]] */ case PSEUDO_EXT : /* EXTERN a[,b[,c...]] */ nRepNext = 0; /* reset eventual repeat */ skipspace(); while (isalnum(*srcptr)) { scanname(); /* parse option */ lp = findsym(namebuf, 1); /* look up the symbol */ switch (co) { case PSEUDO_PUB : /* PUBLIC a[,b[,c...]] */ if (lp->cat == SYMCAT_EMPTY || lp->cat == SYMCAT_UNRESOLVED) lp->cat = SYMCAT_PUBLICUNDEF; else if (lp->cat == SYMCAT_LABEL) lp->cat = SYMCAT_PUBLIC; else if (lp->cat == SYMCAT_PUBLICUNDEF) error |= ERR_LABEL_UNDEF; else if (lp->cat != SYMCAT_PUBLIC) error |= ERR_LABEL_MULT; break; case PSEUDO_EXT : /* EXTERN a[,b[,c...]] */ if (absmode) /* if not in relocating asm mode, */ error |= ERR_RELOCATING; /* set error */ if (lp->cat == SYMCAT_EMPTY || lp->cat == SYMCAT_UNRESOLVED) lp->cat = SYMCAT_EXTERN; else if (lp->cat != SYMCAT_EXTERN) error |= ERR_LABEL_MULT; break; } if (*srcptr == ',') { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); } } break; case PSEUDO_FCB : /* [label] FCB expr[,expr...] */ setlabel(lp); generating = 1; do { if (*srcptr == ',') srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == '\"') { srcptr++; while (*srcptr != '\"' && *srcptr) putbyte(*srcptr++); if (*srcptr == '\"') srcptr++; } else { putbyte((unsigned short)scanexpr(0, &p)); if (unknown && pass == MAX_PASSNO) error |= ERR_LABEL_UNDEF; } if (!(dwOptions & OPTION_TSC)) skipspace(); } while (*srcptr == ','); break; case PSEUDO_FCC : /* [label] FCC expr[,expr...] */ setlabel(lp); if (!(dwOptions & OPTION_TSC)) skipspace(); if (!(dwOptions & OPTION_TSC)) /* if standard */ { /* accept ONE sequence with an */ c = *srcptr++; /* arbitrary delimiter character */ while (*srcptr != c && *srcptr) putbyte(*srcptr++); if (*srcptr == c) srcptr++; } else /* if TSC extended format */ { /* accept MORE sequences */ do { if (*srcptr == ',') srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); c = *srcptr; if ((c == '$') || isalnum(c)) { putbyte((unsigned short)scanexpr(0, &p)); if (unknown && pass == MAX_PASSNO) error |= ERR_LABEL_UNDEF; } else { srcptr++; while (*srcptr != c && *srcptr) putbyte(*srcptr++); if (*srcptr == c) srcptr++; } if (!(dwOptions & OPTION_TSC)) skipspace(); } while (*srcptr == ','); } break; case PSEUDO_FCW : /* [label] FCW,FDB expr[,expr...] */ setlabel(lp); generating = 1; do { if (*srcptr == ',') srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); putword((unsigned)scanexpr(0, &p)); if (unknown && pass == MAX_PASSNO) error |= ERR_LABEL_UNDEF; if (!(dwOptions & OPTION_TSC)) skipspace(); } while (*srcptr == ','); break; case PSEUDO_FCDW: /* [label] FCDW,FDB expr[,expr...] */ setlabel(lp); generating = 1; do { if (*srcptr == ',') srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); putdword((unsigned __int64)scanexpr(0, &p)); if (unknown && pass == MAX_PASSNO) error |= ERR_LABEL_UNDEF; if (!(dwOptions & OPTION_TSC)) skipspace(); } while (*srcptr == ','); break; case PSEUDO_ELSE : /* ELSE */ if (inMacro) /* don't process if in MACRO def. */ break; nRepNext = 0; /* reset eventual repeat */ suppress = 1; condline = 1; /* this is a conditional line */ break; case PSEUDO_ENDIF : /* ENDIF */ if (inMacro) /* don't process if in MACRO def. */ break; nRepNext = 0; /* reset eventual repeat */ condline = 1; /* this is a conditional line */ break; case PSEUDO_IF : /* IF <expression>[,<skip>] */ case PSEUDO_IFN : /* IFN <expression>[,<skip>] */ if (inMacro) /* don't process if in MACRO def. */ break; condline = 1; /* this is a conditional line */ nRepNext = 0; /* reset eventual repeat */ operand = scanexpr(0, &p); if (unknown) error |= ERR_LABEL_UNDEF; if (co == PSEUDO_IFN) /* if IFN */ operand = !operand; /* reverse operand */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') /* skip count? */ { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); nSkipCount = scanexpr(0, &p); if (!operand) nSkipCount = 0; } else if (!operand) suppress = 2; break; case PSEUDO_IFC : /* IFC <string1>,<string2>[,<skip>] */ case PSEUDO_IFNC : /* IFNC <string1>,<string2>[,<skip>] */ if (inMacro) /* don't process if in MACRO def. */ break; condline = 1; /* this is a conditional line */ if (!(dwOptions & OPTION_TSC)) skipspace(); scanstring(szBuf1, sizeof(szBuf1)); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr != ',') /* if not on comma */ { *szBuf2 = '\0'; /* reset 2nd string */ error |= ERR_EXPR; /* set error */ } else { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); scanstring(szBuf2, sizeof(szBuf2)); if (!(dwOptions & OPTION_TSC)) skipspace(); } operand = !strcmp(szBuf1, szBuf2); if (co == PSEUDO_IFNC) operand = !operand; if (*srcptr == ',') /* if skip count */ { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); nSkipCount = scanexpr(0, &p); if (!operand) nSkipCount = 0; } else if (!operand) suppress = 2; break; case PSEUDO_IFD : /* IFD <symbol>[,skipcount] */ case PSEUDO_IFND : /* IFND <symbol>[,skipcount] */ /* ATTENTION: it is easy to produce phasing errors with these 2, */ /* since symbols are NOT reset when starting pass 2! */ if (inMacro) /* don't process if in MACRO def. */ break; skipspace(); scanname(); /* parse symbol name */ lp = findsym(namebuf, 0); /* look up the symbol */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') /* if skip count */ { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); nSkipCount = scanexpr(0, &p); if (!lp != (co == PSEUDO_IFND)) nSkipCount = 0; } /* if no skip count and NOT matched */ else if (!lp != (co == PSEUDO_IFND)) suppress = 2; /* suppress until ELSE or ENDIF */ condline = 1; /* this is a conditional line */ break; case PSEUDO_ORG : /* ORG <expression> */ nRepNext = 0; /* reset eventual repeat */ operand = scanexpr(0, &p); if (unknown) error |= ERR_LABEL_UNDEF; if (relocatable && /* if in relocating assembler mode */ (!absmode)) { error |= ERR_RELOCATING; /* set error */ break; /* and ignore */ } if (generating && pass == MAX_PASSNO) { switch (outmode) { case OUT_BIN : /* binary output file */ j = (int)(unsigned)operand - (int)loccounter; if (j > 0) /* if forward gap */ { for (i = 0; i < j; i++) /* seek forward that many bytes */ fseek(objfile, 1, SEEK_CUR); } else /* if backward gap */ { j = -j; for (i = 0; i < j; i++) /* seek back that many bytes */ fseek(objfile, -1, SEEK_CUR); } break; case OUT_VER: flushver(); break; case OUT_SREC : /* motorola s51-09 */ flushhex(); break; case OUT_IHEX : /* intel hex format */ flushihex(); break; case OUT_FLEX : /* FLEX binary */ flushflex(); break; } } loccounter = operand; hexaddr = loccounter; break; case PSEUDO_SETDP : /* SETDP [<abs page value>] */ nRepNext = 0; /* reset eventual repeat */ skipspace(); if ((!*srcptr) || (*srcptr == '*') || (*srcptr == ';')) operand = -1; else operand = scanexpr(0, &p); if (unknown) error |= ERR_LABEL_UNDEF; if (!(operand & 4095)) operand = (unsigned)operand >> 12; if ((unsigned)operand > 4095) operand = -1; if (absmode) dpsetting = operand; else error |= ERR_RELOCATING; break; case PSEUDO_SET : /* label SET <non external expr> */ nRepNext = 0; /* reset eventual repeat */ operand = scanexpr(0, &p); if (!lp) error |= ERR_LABEL_MISSING; else { if (lp->cat & SYMCAT_VARIABLE || lp->cat == SYMCAT_UNRESOLVED) { if (exprcat == EXPRCAT_INTADDR) lp->cat = SYMCAT_VARADDR; else lp->cat = SYMCAT_VARIABLE; lp->value = (unsigned)operand; } else error |= ERR_LABEL_MULT; } break; case PSEUDO_END : /* END [loadaddr] */ nRepNext = 0; /* reset eventual repeat */ if ((curline->lvl & 0x0f) == 0) /* only in outermost level! */ { skipspace(); /* skip blanks */ if (isfactorstart(*srcptr)) /* if possible transfer address */ { tfradr = (unsigned) /* get transfer address */ scanexpr(0, &p); if (error) /* if error in here */ tfradr = 0; /* reset to zero */ else /* otherwise */ tfradrset = 1; /* remember transfer addr. is set */ } terminate = 1; } break; case PSEUDO_INCLUDE : /* INCLUDE <filename> */ nRepNext = 0; /* reset eventual repeat */ if (inMacro || /* if in macro definition */ (curline->lvl & LINCAT_MACEXP)) /* or macro expansion */ error |= ERR_EXPR; /* this is an error. */ else if (pass == 1) /* otherwise expand if in pass 1 */ { char fname[FNLEN + 1]; int instring = 0; char *osrc = srcptr; if ((curline->lvl & 0x0f) == 0x0f)/* if impossible */ { #if 0 allocline(curline, "NULL", -1, 0x0f, "Error including file - nesting level too deep"); #endif error |= ERR_MALLOC; /* set OUT OF MEMORY error */ break; } if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == '\"') { srcptr++; instring = 1; } for (i = 0; i < FNLEN; i++) { if (*srcptr == 0 || (!instring && *srcptr == ' ') || *srcptr == '"') break; fname[i] = *srcptr++; } fname[i] = 0; curline->lvl |= LINCAT_INVISIBLE; /* preclude listing of INCLUDE line */ readfile(fname, /* append include after current line */ (unsigned char)((curline->lvl & 0x0f) + 1), curline); expandtext(); /* re-expand current line */ srcptr = osrc; } break; case PSEUDO_OPT : /* OPT,OPTION option[,option...] */ nRepNext = 0; /* reset eventual repeat */ skipspace(); while (isalnum(*srcptr)) { scanname(); /* parse option */ if (setoption(unamebuf)) error |= ERR_OPTION_UNK; if (*srcptr == ',') { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); } } break; case PSEUDO_NAM : /* NAM,TTL <text> */ case PSEUDO_STTL : /* STTL <text> */ nRepNext = 0; /* reset eventual repeat */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (isalnum(*srcptr)) { char *tgt = (co == PSEUDO_NAM) ? szTitle : szSubtitle; char *lnblnk = tgt; int nBytes = 0; while (*srcptr && nBytes < (sizeof(szTitle) - 1)) { if (*srcptr != ' ') lnblnk = tgt; *tgt++ = *srcptr++; nBytes++; } lnblnk[1] = '\0'; /* terminate after last nonblank */ while (*srcptr) /* skip rest if too long */ srcptr++; } break; case PSEUDO_PAG : /* PAG [<abs expression>] */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (isfactorstart(*srcptr)) /* if possible new page number */ { int nPage = scanexpr(0, &p); /* get new page # */ if (!error && listfile) /* if valid and writing listfile */ nCurPage = nPage - 1; else break; } curline->lvl |= LINCAT_INVISIBLE; if ((listing & LIST_ON) && listfile && (dwOptions & OPTION_PAG) && (!(curline->lvl & LINCAT_LVLMASK) || (dwOptions & OPTION_LLL)) && (pass > 1 || (dwOptions & OPTION_LP1))) PageFeed(); break; case PSEUDO_SPC : /* SPC <n[,keep]> */ { int nSpc = 1, nKeep = 0; if (!(dwOptions & OPTION_TSC)) skipspace(); if (isfactorstart(*srcptr)) /* if possible new page number */ { nSpc = scanexpr(0, &p); /* get # space lines */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') /* if followed by , */ { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); nKeep = scanexpr(0, &p); /* get # keep lines */ } if (error) break; } curline->lvl |= LINCAT_INVISIBLE; if (listing & LIST_ON) { if (nSpc > 0) /* if spaces needed */ { if ((dwOptions & OPTION_PAG) && (nCurLine + nSpc + nKeep >= nLinesPerPage)) PageFeed(); else for (; nSpc; nSpc--) putlist("\n"); } else if (nKeep && (dwOptions & OPTION_PAG) && (nCurLine + nKeep >= nLinesPerPage)) PageFeed(); } } break; case PSEUDO_REP : /* REP n */ if (!(dwOptions & OPTION_TSC)) skipspace(); nRepNext = scanexpr(0, &p); /* get # repetitions */ curline->lvl |= LINCAT_INVISIBLE; break; case PSEUDO_SETPG : /* SETPG pagelen */ if (!(dwOptions & OPTION_TSC)) skipspace(); nLinesPerPage = scanexpr(0, &p); /* get # lines per page */ if (nLinesPerPage < 10) /* adjust to boundary values */ nLinesPerPage = 10; else if (nLinesPerPage > 1000) nLinesPerPage = 1000; curline->lvl |= LINCAT_INVISIBLE; break; case PSEUDO_SETLI : /* SETLI linelen */ if (!(dwOptions & OPTION_TSC)) skipspace(); nColsPerLine = scanexpr(0, &p); /* get # columns per line */ if (nColsPerLine < 40) /* adjust to boundary values */ nColsPerLine = 40; else if (nColsPerLine > 2000) nColsPerLine = 2000; curline->lvl |= LINCAT_INVISIBLE; break; case PSEUDO_SYMLEN : /* SYMLEN symbollength */ if (!(dwOptions & OPTION_TSC)) skipspace(); maxidlen = scanexpr(0, &p); /* get # significant ID places */ if (maxidlen < 6) /* adjust to boundary values */ maxidlen = 6; else if ((outmode == OUT_REL) && (maxidlen > 8)) maxidlen = 8; else if (maxidlen > MAXIDLEN) maxidlen = MAXIDLEN; curline->lvl |= LINCAT_INVISIBLE; break; case PSEUDO_MACRO : /* label MACRO */ if (!lp) /* a macro NEEDS a label! */ error |= ERR_LABEL_MISSING; if (lp->cat == SYMCAT_EMPTY) { if (nMacros < MAXMACROS) /* if space for another macro defin. */ { lp->cat = SYMCAT_MACRO; /* remember it's a macro */ macros[nMacros] = curline; /* remember pointer to start line */ lp->value = nMacros++; /* and remember the macro */ } else error |= ERR_MALLOC; } else if (lp->cat != SYMCAT_MACRO || macros[lp->value] != curline) error |= ERR_LABEL_MULT; inMacro++; curline->lvl |= LINCAT_MACDEF; if (inMacro > 1) error |= ERR_NESTING; break; case PSEUDO_ENDM : /* ENDM */ if (!inMacro) error |= ERR_EXPR; else inMacro--; break; case PSEUDO_EXITM : /* EXITM */ if (!inMacro && /* only allowed inside macros */ !(curline->lvl & LINCAT_MACEXP)) error |= ERR_EXPR; break; case PSEUDO_REG : /* label REG <register list> */ if (!lp) /* label is mandatory! */ error |= ERR_LABEL_MISSING; { struct regrecord *p; postbyte = 0; do { if (*srcptr == ',') srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); scanname(); if ((p = findreg(unamebuf)) == 0) error |= ERR_ILLEGAL_ADDR; else postbyte |= p->psh; if (!(dwOptions & OPTION_TSC)) skipspace(); } while (*srcptr == ','); if (lp->cat == SYMCAT_EMPTY) { lp->cat = SYMCAT_REG; lp->value = postbyte; } else if (lp->cat != SYMCAT_REG || lp->value != postbyte) error |= ERR_LABEL_MULT; } break; case PSEUDO_ERR : /* ERR text? */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (pass != 1) /* ignore in pass 1 */ { errormsg[14] = srcptr; error |= ERR_ERRTXT; } break; case PSEUDO_TEXT : /* TEXT text ? */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (!lp) /* label is mandatory! */ error |= ERR_LABEL_MISSING; else if (lp->cat != SYMCAT_EMPTY && /* and must be text, if there */ lp->cat != SYMCAT_TEXT) error |= ERR_LABEL_MULT; else /* if all OK, (re)define text */ settext(lp->name, srcptr); break; case PSEUDO_NAME : /* NAME <modulename> ? */ if (!(dwOptions & OPTION_TSC)) skipspace(); scanname(); if (!namebuf[0]) /* name must be given */ error |= ERR_EXPR; strncpy(modulename, namebuf, 8); break; case PSEUDO_BINARY : /* BIN[ARY] <filename> ? */ nRepNext = 0; /* reset eventual repeat */ if (pass == 1) /* expand if in pass 1 */ { char fname[FNLEN + 1]; int instring = 0; char *osrc = srcptr; if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == '\"') { srcptr++; instring = 1; } for (i = 0; i < FNLEN; i++) { if (*srcptr == 0 || (!instring && *srcptr == ' ') || *srcptr == '"') break; fname[i] = *srcptr++; } fname[i] = 0; curline->lvl |= LINCAT_INVISIBLE; /* preclude listing of BINARY line */ readbinary(fname, /* append binary after current line */ (unsigned char)((curline->lvl & 0x0f) + 1), curline, lp); expandtext(); /* re-expand current line */ srcptr = osrc; } break; } } /*****************************************************************************/ /* macskip : skips a range of macro lines */ /*****************************************************************************/ struct linebuf *macskip(struct linebuf *pmac, int nSkips) { if (nSkips < 0) /* we need to go to the line BEFORE */ nSkips--; /* the one we need! */ while (nSkips) { if (nSkips < 0) { if (!pmac->prev) break; pmac = pmac->prev; nSkips++; } else { if (!pmac->next) break; pmac = pmac->next; nSkips--; } } return pmac; } /*****************************************************************************/ /* expandmacro : expands a macro definition below the current line */ /*****************************************************************************/ void expandmacro(struct symrecord *lp, struct symrecord *lpmac) { struct oprecord *op; char szMacInv[LINELEN]; /* macro invocation line */ char szLine[LINELEN]; /* current macro line */ char *szMacParm[10]; char *s, *d; /* source / destination work pointers*/ char *srcsave = srcptr; struct linebuf *cursave = curline; int nMacParms = 1; int nInString = 0; int nMacLine = 1; /* current macro line */ /* current macro line */ struct linebuf *pmac = macros[lpmac->value]->next; struct linebuf *pcur = curline; /* current expanded macro line */ struct linebuf *pdup = NULL; /* DUP start line */ int nDup = 0; /* # repetitions for DUP */ int terminate = 0; /* terminate macro expansion if set */ int skipit = 0; /* skip this line */ int suppress[64] = {0}; /* internal suppression (max.64 lvl) */ int ifcount = 0; /* internal if counter */ struct relocrecord p = {0}; if ((listing & LIST_ON) && /* if listing pass 1 */ (dwOptions & OPTION_LIS) && (dwOptions & OPTION_LP1)) outlist(NULL); /* show macro invocation BEFORE */ /* processing the expansions */ skipspace(); /* skip spaces before macro args */ if (!lp) /* if no macro label */ szMacParm[0] = ""; /* set &0 to blank */ else /* otherwise */ szMacParm[0] = lp->name; /* set &0 to line label */ /* initialize all parameters to "" */ for (nMacParms = 1; nMacParms < 10; nMacParms++) szMacParm[nMacParms] = ""; nMacParms = 1; /* reset # parsed parms to 1 */ strcpy(szMacInv, srcptr); /* copy the current line for mangling*/ srcptr = szMacInv; /* set pointer to internal buffer */ do { while (*srcptr == ',') /* skip parameter delimiter(s) */ { *srcptr++ = '\0'; /* delimit & advance behind it */ if (!(dwOptions & OPTION_TSC)) skipspace(); } /* OK, next string... */ if ((*srcptr == '\'') || /* if delimited string */ (*srcptr == '\"')) nInString = 1; /* remember we're in delimited strg */ szMacParm[nMacParms] = srcptr++; /* store parameter start pointer */ while (*srcptr) /* walk to end of parameter */ { if (!nInString && ((*srcptr == ' ') || (*srcptr == ','))) break; else if (nInString && *srcptr == *szMacParm[nMacParms]) { srcptr++; break; } srcptr++; } if (*srcptr && *srcptr != ',') { *srcptr++ = '\0'; if (!(dwOptions & OPTION_TSC)) skipspace(); } nMacParms++; if (nMacParms >= 10) break; } while (*srcptr == ','); /*---------------------------------------------------------------------------*/ /* OK, got macro arguments &0...&9 now */ /*---------------------------------------------------------------------------*/ while (pmac) /* walk through the macro lines */ { srcptr = s = pmac->txt; d = szLine; op = NULL; skipit = 0; while (*s) /* first, expand the line */ { if (*s == '\\' && s[1] == '&') { s++; *d++ = *s++; } else if (*s == '&' && s[1] >= '0' && s[1] <= '9') { strcpy(d, szMacParm[s[1] - '0']); s += 2; d += strlen(d); } else *d++ = *s++; } *d = '\0'; srcptr = szLine; /* then, look whether code or macro */ if (isalnum(*srcptr)) { scanname(); lp = findsym(namebuf, 1); if (*srcptr == ':') srcptr++; } skipspace(); if (isalnum(*srcptr)) /* then parse opcode */ { scanname(); op = findop(unamebuf); } skipspace(); /* and skip to eventual parameter */ if (op && op->cat == OPCAT_PSEUDO) /* if pseudo-op */ { switch (op->code) /* examine for macro expansion */ { case PSEUDO_ENDM : /* ENDM ? */ terminate = 1; /* terminate macro expansion */ break; case PSEUDO_EXITM : /* EXITM ? */ if (!suppress[ifcount]) /* if not suppressed */ terminate = 1; /* terminate macro expansion */ break; case PSEUDO_DUP : /* DUP ? */ if (pdup != NULL) /* nesting not allowed */ error |= ERR_NESTING; else { nDup = scanexpr(0, &p); /* scan the expression */ pdup = pmac; /* remember DUP start line */ if (nDup < 1 || nDup > 255) /* if invalid # repetitions */ { error |= ERR_EXPR; /* set error here */ nDup = 1; } else skipit = 1; /* skip this line */ } break; case PSEUDO_ENDD : /* ENDD ? */ if (!pdup) /* if no DUP defined */ error |= ERR_EXPR; /* can't do that here. */ else { nDup--; /* decrement # duplications */ if (!nDup) /* if done, */ pdup = NULL; /* reset dup start */ else /* otherwise */ pmac = pdup; /* reset current line to DUP op */ skipit = 1; /* this line isn't really there */ } break; case PSEUDO_IF : /* IF */ case PSEUDO_IFN : /* IFN */ ifcount++; /* increment # ifs */ /* take suppression from higher level*/ suppress[ifcount] = suppress[ifcount - 1]; if (suppress[ifcount]) /* if already suppressed */ suppress[ifcount]++; /* inrease # suppressions */ else /* otherwise evaluate expression */ { operand = scanexpr(0, &p); if (unknown) error |= ERR_LABEL_UNDEF; if (op->code == PSEUDO_IFN) /* if IFN */ operand = !operand; /* invert operand */ if (!operand) /* if evaulation to zer0 */ suppress[ifcount]++; /* suppress until ELSE or ENDIF */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') /* if skip count passed */ { int nSkips; srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); nSkips = scanexpr(0, &p); /* scan skip count */ ifcount--; /* this needs no ENDIF */ if (nSkips < -255 || nSkips > 255 || nSkips == 0) { error |= ERR_EXPR; break; } if (operand) /* if evaluation is true */ pmac = macskip(pmac, nSkips); /* then skip the amount of lines */ } } skipit = 1; /* don't add this line! */ break; case PSEUDO_IFC : /* IFC */ case PSEUDO_IFNC : /* IFNC */ ifcount++; /* increment # ifs */ /* take suppression from higher level*/ suppress[ifcount] = suppress[ifcount - 1]; if (suppress[ifcount]) /* if already suppressed */ suppress[ifcount]++; /* inrease # suppressions */ else /* otherwise evaluate expression */ { if (!(dwOptions & OPTION_TSC)) skipspace(); scanstring(szBuf1, sizeof(szBuf1)); if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr != ',') /* if not on comma */ { *szBuf2 = '\0'; /* reset 2nd string */ error |= ERR_EXPR; /* set error */ } else { srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); scanstring(szBuf2, sizeof(szBuf2)); } operand = !strcmp(szBuf1, szBuf2); if (op->code == PSEUDO_IFNC) operand = !operand; if (!operand) /* if evaulation to zer0 */ suppress[ifcount]++; /* suppress until ELSE or ENDIF */ if (!(dwOptions & OPTION_TSC)) skipspace(); if (*srcptr == ',') /* if skip count passed */ { int nSkips; srcptr++; if (!(dwOptions & OPTION_TSC)) skipspace(); nSkips = scanexpr(0, &p); /* scan skip count */ ifcount--; /* this needs no ENDIF */ if (nSkips < -255 || nSkips > 255 || nSkips == 0) { error |= ERR_EXPR; break; } if (operand) /* if evaluation is true */ pmac = macskip(pmac, nSkips); /* then skip the amount of lines */ } } if (!error) skipit = 1; /* don't add this line! */ break; case PSEUDO_ELSE : /* ELSE */ if (!suppress[ifcount]) /* if IF not suppressed */ suppress[ifcount]++; /* suppress ELSE clause */ else /* otherwise */ suppress[ifcount]--; /* decrement suppression */ skipit = 1; /* don't add this line! */ break; case PSEUDO_ENDIF : /* ENDIF */ if (ifcount) ifcount--; if (suppress[ifcount]) suppress[ifcount]--; skipit = 1; /* don't add this line! */ break; } } if (terminate) /* if macro termination needed */ break; /* terminate here */ if (!skipit && !suppress[ifcount]) /* if not skipping this one */ { /* add line to source */ pcur = allocline(pcur, curline->fn, curline->ln, LINCAT_MACEXP, szLine); if (!pcur) { error |= ERR_MALLOC; break; } else { curline = pcur; error = ERR_OK; warning = WRN_OK; expandtext(); processline(); } } pmac = pmac->next; } curline = cursave; expandtext(); srcptr = srcsave; /* restore source pointer */ codeptr = 0; } /*****************************************************************************/ /* processline : processes a source line */ /*****************************************************************************/ void processline() { struct symrecord *lp, *lpmac; struct oprecord *op = NULL; __int64 co; short cat; short c; char noimm; #if 0 srcptr = curline->txt; #else srcptr = srcline; #endif oldlc = loccounter; unknown = 0; certain = 1; lp = 0; codeptr = 0; condline = 0; isFar = 0; isFarkw = 0; if (inMacro) curline->lvl |= LINCAT_MACDEF; if (isalnum(*srcptr)) /* look for label on line start */ { scanname(); if (stricmp(namebuf, "far")==0) { isFar = 1; isFarkw = 1; while(*srcptr==' ' || *srcptr=='\t') srcptr++; scanname(); } lp = findsym(namebuf, 1); if (*srcptr == ':') srcptr++; if ((lp) && (lp->cat != SYMCAT_COMMONDATA) && (lp->u.flags & SYMFLAG_FORWARD)) lp->u.flags |= SYMFLAG_PASSED; } skipspace(); if ((isalnum(*srcptr)) || ((dwOptions & OPTION_GAS) && (*srcptr == '.'))) { scanname(); if ((dwOptions & OPTION_H63) && /* eventually adjust some mnemonics */ ((!strcmp(unamebuf, "ASLD")) || /* that are available in the 6309 */ (!strcmp(unamebuf, "ASRD")) || /* but are implemented as (slower) */ (!strcmp(unamebuf, "LSLD")) || /* convenience instructions on the */ (!strcmp(unamebuf, "LSRD")) || /* 6809 */ (!strcmp(unamebuf, "DECD")) || (!strcmp(unamebuf, "INCD")) || (!strcmp(unamebuf, "CLRD")))) strcat(unamebuf, "63"); op = findop(unamebuf); if (op) { if ((dwOptions & OPTION_TSC)) /* if TSC compatible, skip space NOW */ skipspace(); /* since it's not allowed inside arg */ if (op->cat != OPCAT_PSEUDO) { setlabel(lp); generating = 1; } co = op->code; cat = op->cat; /* only pseudo-ops in common mode! */ if (common && (cat != OPCAT_PSEUDO)) error |= ERR_EXPR; noimm = cat & OPCAT_NOIMM; /* isolate "no immediate possible" */ cat &= ~OPCAT_NOIMM; if (dwOptions & OPTION_X32) isX32 = 1; else isX32 = 0; if (dwOptions & OPTION_H63) /* if in HD6309 mode, */ cat &= ~OPCAT_6309; /* mask out the 6309 flag (=allow) */ switch (cat) { case OPCAT_ONEBYTE : onebyte(co); break; case OPCAT_TWOBYTE : twobyte(co); break; case OPCAT_THREEBYTE : threebyte(co); break; case OPCAT_FOURBYTE : fourbyte(co); break; case OPCAT_2IMMBYTE : /* 6309 only */ putbyte((unsigned short)(co >> 12LL)); /* fall thru on purpose! */ case OPCAT_IMMBYTE : oneimm(co); break; case OPCAT_LEA : lea(co); break; case OPCAT_SBRANCH : sbranch(co); break; case OPCAT_LBR2BYTE : lbranch(co); break; case OPCAT_LBR1BYTE : lbra(co); break; case OPCAT_2ARITH : /* 6309 only */ putbyte((unsigned short)(co >> 12)); /* fall thru on purpose! */ case OPCAT_ARITH : arith(co, noimm); break; case OPCAT_ACCARITH : /* 6800-style arith */ accarith(co, noimm, 0); break; case OPCAT_DBLREG1BYTE : darith(co, noimm, op->tgtNdx); break; case OPCAT_DBLREG2BYTE : d2arith(co, noimm, op->tgtNdx); break; case OPCAT_SINGLEADDR : oneaddr(co); break; case OPCAT_IREG : /* 6309 only */ putbyte((unsigned short)(co >> 12)); /* fall thru on purpose! */ case OPCAT_2REG : tfrexg(co); break; case OPCAT_STACK : pshpul(co); break; case OPCAT_BITDIRECT : /* 6309 only */ bitdirect(co); break; case OPCAT_BITTRANS : /* 6309 only */ bittrans(co); break; case OPCAT_BLOCKTRANS : /* 6309 only */ blocktrans(co); break; case OPCAT_QUADREG1BYTE : qarith(co, noimm); break; case OPCAT_IDXEXT : idxextarith(co); break; case OPCAT_ACCADDR : accaddr(co); break; case OPCAT_PSEUDO : pseudoop(co, lp); break; default : error |= ERR_ILLEGAL_MNEM; break; } c = *srcptr; /* get current character */ if (((dwOptions & OPTION_TSC) && (c == '*')) || ((dwOptions & OPTION_GAS) && (c == '|')) || (c == ';')) c = '\0'; if (c != ' ' && *(srcptr - 1) != ' ' && c != '\0') error |= ERR_ILLEGAL_ADDR; } else { lpmac = findsym(namebuf, 0); /* look whether opcode is a macro */ if (lpmac && lpmac->cat == SYMCAT_MACRO) { if (pass == 1) /* if in pass 1 */ expandmacro(lp, lpmac); /* expand macro below current line */ } else error |= ERR_ILLEGAL_MNEM; } } else { nRepNext = 0; /* reset eventual repeat if no code */ setlabel(lp); } if (inMacro) /* if in macro definition */ { codeptr = 0; /* ignore the code */ error &= (ERR_MALLOC | ERR_NESTING); /* ignore most errors */ warning &= WRN_SYM; /* ignore most warnings */ } if (pass == MAX_PASSNO) { outbuffer(); if ((listing & LIST_ON) && (dwOptions & OPTION_LIS)) outlist(op); } else if ((listing & LIST_ON) && (dwOptions & OPTION_LIS) && (dwOptions & OPTION_LP1)) { if (curline->lvl & LINCAT_MACEXP || /* prevent 2nd listing of macro */ !curline->next || /* since this is done in expansion */ !(curline->next->lvl & LINCAT_MACEXP)) outlist(op); } if (error || warning) report(); loccounter += codeptr; } /*****************************************************************************/ /* suppressline : suppresses a line */ /*****************************************************************************/ void suppressline() { struct oprecord * op = NULL; srcptr = srcline; oldlc = loccounter; codeptr = 0; condline = 0; if (nSkipCount > 0) { nSkipCount--; condline = 1; /* this is STILL conditional */ } else { if (isalnum(*srcptr)) { scanname(); if (*srcptr == ':') srcptr++; } skipspace(); scanname(); op = findop(unamebuf); if (op && op->cat == OPCAT_PSEUDO) /* examine pseudo-ops in detail */ { if ((op->code == PSEUDO_IF) || /* IF variants */ (op->code == PSEUDO_IFN) || (op->code == PSEUDO_IFC) || (op->code == PSEUDO_IFNC) || (op->code == PSEUDO_IFD) || (op->code == PSEUDO_IFND)) { ifcount++; condline = 1; /* this is a conditional line */ } else if (op->code == PSEUDO_ENDIF) /* ENDIF */ { if (ifcount > 0) ifcount--; else if (suppress == 1 || suppress == 2) suppress = 0; condline = 1; /* this is a conditional line */ } else if (op->code == PSEUDO_ELSE) /* ELSE */ { if (ifcount == 0 && suppress == 2) suppress = 0; condline = 1; /* this is a conditional line */ } } } if (((pass == MAX_PASSNO) || (dwOptions & OPTION_LP1)) && (listing & LIST_ON) && (dwOptions & OPTION_LIS)) outlist(op); } /*****************************************************************************/ /* usage : prints out correct usage */ /*****************************************************************************/ void usage(char *nm) { printf("Usage: %s [-option*] srcname*\n", nm ? nm : "a09"); printf("Available options are:\n"); printf("-B[objname] ........ output to binary file (default)\n"); printf("-F[objname] ........ output to FLEX binary file\n"); /* printf("-G[objname] ........ output Gnu .o format file\n"); */ printf("-R[objname] ........ output to FLEX relocatable object file\n"); printf("-S[objname] ........ output to Motorola S51-09 file\n"); printf("-X[objname] ........ output to Intel Hex file\n"); printf("-L[listname] ....... create listing file \n"); printf("-V[objname]......... output Verilog rommem declarations \n"); printf("-C ................. suppress code output\n"); printf("-Dsymbol[=value] ... predefines a symbol\n"); printf(" for TSC 6809 Assembler compatibility,\n"); printf(" you should only use symbols A through C\n"); printf("-Ooption ........... sets an option (as in OPT pseudoop)\n"); printf("-W ................. suppress warnings\n"); printf("srcname ............ source file name(s)\n"); exit(2); } /*****************************************************************************/ /* getoptions : retrieves the options from the passed argument array */ /*****************************************************************************/ void getoptions (int argc, char* argv[]) { int i, j; char *ld; for (i = 1; i < argc; i++) { #if !UNIX /* code for DOS / Windows / OS2 */ if ((argv[i][0] == '-') || (argv[i][0] == '/')) #else /* code for UNIX derivates */ if (argv[i][0] == '-') #endif { for (j = 1; j < (int)strlen(argv[i]); j++) { switch (tolower(argv[i][j])) { case 'c' : /* suppress code output */ outmode = OUT_NONE; break; case 'b' : /* define binary output file */ case 's' : /* define Motorola output file */ case 'x' : /* define Intel Hex output file */ case 'f' : /* define FLEX output file */ case 'r' : /* define FLEX relocatable output f. */ case 'v' : /* case 'g' : */ /* define GNU output file */ strcpy(objname, /* copy in the name */ argv[i] + j + 1); switch (tolower(argv[i][j])) /* then set output mode */ { case 'b' : outmode = OUT_BIN; break; case 's' : outmode = OUT_SREC; break; case 'x' : outmode = OUT_IHEX; break; case 'f' : outmode = OUT_FLEX; break; case 'v': outmode = OUT_VER; break; /* case 'g' : outmode = OUT_GAS; relocatable = 1; dwOptions |= OPTION_REL; break; */ case 'r' : outmode = OUT_REL; maxidlen = 8; /* only 8 significant ID chars! */ relocatable = 1; absmode = 0; break; } j = strlen(argv[i]); /* advance behind copied name */ break; case 'l' : /* define listing file */ strcpy(listname, /* copy in the name */ argv[i] + j + 1); j = strlen(argv[i]); /* advance behind copied name */ listing = LIST_ON; /* remember we're listing */ break; case 'd' : /* define a symbol ? */ srcptr = argv[i] + j + 1; /* parse in the name */ scanname(); if (!namebuf[0]) /* if no name there */ usage(argv[0]); /* show usage and get out */ if (*srcptr == '=') /* if followed by value */ srcptr++; /* advance behind it */ else /* otherwise go to "" */ srcptr = argv[i] + strlen(argv[i]); settext(namebuf, srcptr); /* define the symbol */ j = strlen(argv[i]); /* advance behind copied name */ break; case 'o' : /* option */ if (setoption(argv[i] + j + 1)) usage(argv[0]); j = strlen(argv[i]); /* advance behind option */ break; } } for (j = i; j < argc; j++) /* remove all consumed arguments */ argv[j] = argv[j + 1]; argc--; /* reduce # arguments */ i--; /* and restart with next one */ } } if (argc < 2) /* if not at least one filename left */ usage(argv[0]); /* complain & terminate */ strcpy(srcname, argv[1]); /* copy it in. */ if (!objname[0]) /* if no object name defined */ { strcpy(objname, srcname); /* copy in the source name */ ld = strrchr(objname, '.'); /* look whether there's a dot in it */ if (!ld) /* if not */ ld = objname + strlen(objname); /* append extension */ switch (outmode) /* which output mode? */ { case OUT_BIN : /* binary ? */ #if !UNIX strcpy(ld, ".bin"); /* DOS / Windows / OS2 */ #else strcpy(ld, ".b"); /* UNIX */ #endif break; case OUT_SREC : /* Motorola S51-09 ? */ strcpy(ld, ".s09"); break; case OUT_IHEX : /* Intel Hex ? */ strcpy(ld, ".hex"); break; case OUT_FLEX : /* FLEX binary ? */ strcpy(ld, ".bin"); break; case OUT_GAS : /* GNU relocatable object ? */ strcpy(ld, ".o"); break; case OUT_REL : /* Flex9 relocatable object ? */ strcpy(ld, ".rel"); break; case OUT_VER : /* Verilog texr */ strcpy(ld, ".ver"); break; } } if ((listing & LIST_ON) && !listname[0])/* if no list file specified */ { strcpy(listname, srcname); /* copy in the source name */ ld = strrchr(listname, '.'); /* look whether there's a dot in it */ if (!ld) /* if not */ ld = listname + strlen(listname); /* append extension */ strcpy(ld, ".lst"); /* .lst */ } } /*****************************************************************************/ /* processfile : processes the input file */ /*****************************************************************************/ void processfile(struct linebuf *pline) { struct linebuf *plast = pline; vercount = 0; while (!terminate && pline) { curline = pline; error = ERR_OK; warning = WRN_OK; expandtext(); /* expand text symbols */ srcptr = srcline; if (suppress || nSkipCount) suppressline(); else { if (nRepNext) { for (; nRepNext > 0; nRepNext--) { processline(); error = ERR_OK; warning = WRN_OK; } nRepNext = 0; } else { if (pass != 1 || !(curline->lvl & LINCAT_MACEXP)) processline(); error = ERR_OK; warning = WRN_OK; } } plast = pline; pline = pline->next; } if (suppress) { printf("%s(%ld) : error 18: improperly nested IF statements\n", expandfn(plast->fn), plast->ln); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 18: improperly nested IF statements\n"); errors++; suppress = 0; } if (common) { printf("%s(%ld) : error 24: improperly nested COMMON statements\n", expandfn(plast->fn), plast->ln); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 24: improperly nested COMMON statements\n"); errors++; common = 0; } } void dopass(int passno, int lastpass) { int i; char buf[4]; sprintf(buf, "%d", passno); pass = passno; settext("PASS", buf); loccounter = 0; errors = 0; warnings = 0; generating = 0; terminate = 0; for (i = 0; i < symcounter; i++) /* reset all PASSED flags */ if (symtable[i].cat != SYMCAT_COMMONDATA) symtable[i].u.flags &= ~SYMFLAG_PASSED; if (lastpass) { if (listing & LIST_ON) { if ((dwOptions & OPTION_LP1)) { if (dwOptions & OPTION_PAG) PageFeed(); else putlist("\n"); putlist( "*** Pass 2 ***\n\n"); } else if (nTotErrors || nTotWarnings) putlist( "%ld error(s), %ld warning(s) unlisted in pass 1\n", nTotErrors, nTotWarnings); } if ((outmode >= OUT_BIN) && ((objfile = fopen(objname, ((outmode != OUT_SREC) && (outmode != OUT_IHEX)) ? "wb" : "w")) == 0)) { printf("%s(0) : error 20: cannot write object file %s\n", srcname, objname); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 20: cannot write object file %s\n", objname); exit(4); } if (outmode == OUT_REL) /* if writing FLEX Relocatable */ { writerelcommon(); /* write out common blocks */ relhdrfoff = ftell(objfile); writerelhdr(0); /* write out initial header */ } } processfile(rootline); if (errors) { printf("%ld error(s) in pass %d\n", errors, passno); nTotErrors += errors; } if (warnings) { printf("%ld warning(s) in pass %d\n", warnings, passno); nTotWarnings += warnings; } } /*****************************************************************************/ /* main : the main function */ /*****************************************************************************/ int main (int argc, char *argv[]) { int i; struct linebuf *pLastLine = NULL; settext("ASM", "A09"); /* initialize predefined texts */ settext("VERSION", VERSNUM); settext("PASS", "1"); setoptiontexts(); nPredefinedTexts = nTexts; getoptions(argc, argv); pass = 1; error = 0; warning = 0; nTotErrors = errors = 0; nTotWarnings = warnings = 0; generating = 0; common = 0; terminate = 0; if (!absmode) /* in relocating mode */ dpsetting = -1; /* there IS no Direct Page */ printf("A09 Assembler V" VERSION "\n"); if ((listing & LIST_ON) && ((listfile = fopen(listname, "w")) == 0)) { printf("%s(0) : error 19: Cannot open list file %s\n", srcname, listname); exit(4); } for (i = 1; argv[i]; i++) /* read in all source files */ pLastLine = readfile(argv[i], 0, pLastLine); if (!rootline) /* if no lines in there */ { printf("%s(0) : error 23: no source lines in file\n", srcname); if (((dwOptions & OPTION_LP1) || pass == MAX_PASSNO) && (listing & LIST_ON)) putlist( "*** Error 23: no source lines in file\n"); exit(4); } /* Pass 1 - parse all symbols */ if ((listing & LIST_ON) && (dwOptions & OPTION_LP1)) putlist( "*** Pass 1 ***\n\n"); processfile(rootline); if (errors) { printf("%ld error(s) in pass 1\n",errors); if ((listing & LIST_ON) && (dwOptions & OPTION_LP1)) putlist( "%ld error(s) in pass 1.\n", errors); nTotErrors = errors; } if (warnings) { printf("%ld warning(s) in pass 1\n", warnings); if ((listing & LIST_ON) && (dwOptions & OPTION_LP1)) putlist( "%ld warning(s) in pass 1.\n", warnings); nTotWarnings = warnings; } // Resolve potential phasing errors by performing multiple passes. dopass(2,0); dopass(3,0); dopass(4,0); dopass(5,0); dopass(6,0); dopass(7,0); dopass(8,0); dopass(9,1); // /* Pass 2 - generate output */ //settext("PASS", "3"); //loccounter = 0; //errors = 0; //warnings = 0; //generating = 0; //terminate = 0; ////memset(bUsedBytes, 0, sizeof(bUsedBytes)); //for (i = 0; i < symcounter; i++) /* reset all PASSED flags */ // if (symtable[i].cat != SYMCAT_COMMONDATA) // symtable[i].u.flags &= ~SYMFLAG_PASSED; // //if (listing & LIST_ON) // { // if ((dwOptions & OPTION_LP1)) // { // if (dwOptions & OPTION_PAG) // PageFeed(); // else // putlist("\n"); // putlist( "*** Pass 2 ***\n\n"); // } // else if (nTotErrors || nTotWarnings) // putlist( "%ld error(s), %ld warning(s) unlisted in pass 1\n", // nTotErrors, nTotWarnings); // } // //if ((outmode >= OUT_BIN) && // ((objfile = fopen(objname, // ((outmode != OUT_SREC) && (outmode != OUT_IHEX)) ? "wb" : "w")) // == 0)) // { // printf("%s(0) : error 20: cannot write object file %s\n", srcname, objname); // if (((dwOptions & OPTION_LP1) || pass == 2) && (listing & LIST_ON)) // putlist( "*** Error 20: cannot write object file %s\n", objname); // exit(4); // } // //if (outmode == OUT_REL) /* if writing FLEX Relocatable */ // { // writerelcommon(); /* write out common blocks */ // relhdrfoff = ftell(objfile); // writerelhdr(0); /* write out initial header */ // } // //processfile(rootline); // //if (errors) // { // printf("%ld error(s) in pass 2\n", errors); // nTotErrors += errors; // } //if (warnings) // { // printf("%ld warning(s) in pass 2\n", warnings); // nTotWarnings += warnings; // } if (listing & LIST_ON) { if (errors || warnings) putlist("\n"); if (errors) putlist( "%ld error(s) in pass 2.\n", errors); if (warnings) putlist( "%ld warning(s) in pass 2.\n", warnings); if (dwOptions & OPTION_SYM) outsymtable(); if ((relocatable) && (dwOptions & OPTION_REL)) outreltable(); if ((dwOptions & OPTION_TXT) && (nPredefinedTexts < nTexts)) outtexttable(); putlist( "\n%ld error(s), %ld warning(s)\n", nTotErrors, nTotWarnings); fclose(listfile); } else printf("Last assembled address: %04X\n", loccounter - 1); switch (outmode) /* look whether object cleanup needed*/ { case OUT_VER: flushver(); break; case OUT_SREC : /* Motorola S51-09 */ flushhex(); chksum = (tfradr & 0xfff) + ((tfradr >> 12) & 0xfff) + 3; fprintf(objfile, "S9003%06X%03X\n", tfradr, 0xfff - (chksum & 0xfff)); break; case OUT_IHEX : /* Intel Hex */ flushihex(); #if 0 /* put transfer address in here... unfortunately, the official Intel documentation doesn't allow this mechanism */ chksum = ((tfradr >> 8) & 0xff) + (tfradr & 0xff) + 1; fprintf(objfile, ":00%04X01%02X\n", tfradr, (-(signed)chksum) & 0xff); #else fprintf(objfile, ":00000001FF\n"); #endif break; case OUT_FLEX : /* FLEX binary */ flushflex(); if (tfradrset) /* if transfer address set */ { /* write out transfer address block */ fputc(0x16, objfile); fputc((tfradr >> 8) & 0xff, objfile); fputc(tfradr & 0xff, objfile); } break; case OUT_REL : /* FLEX Relocatable */ writerelext(); /* write out External table */ writerelglobal(); /* write out Global table */ writerelmodname(); /* write out Module Name */ i = (int)ftell(objfile); /* fill last sector with zeroes */ while (i % 252) { fputc(0, objfile); i++; } /* reposition on header */ fseek(objfile, relhdrfoff, SEEK_SET); writerelhdr(0); /* rewrite completed header */ break; } if (objfile) fclose(objfile); if (errors) unlink(objname); return (errors) ? 1 : 0; }
Go to most recent revision | Compare with Previous | Blame | View Log