.TITLE SMAC Structured Programming Macros Package .IDENT /2.00/ ;++ ; Title: ; SMAC Structured Programming Macros Package ; ; Facility: ; Package of support macros to allow MACRO programmers to write ; more structured code. ; ; Abstract: ; The SMAC package implements the common constructs of structured ; programming to allow MACRO programmers to write more structured ; programs. ; The following structures are implemented: ; ; IF condition THEN ; ELSE ; ENDIF ; ; REPEAT ; UNTIL |condition | ; |FOREVER | ; |ONCE | ; ; WHILE condition DO ; ENDWHILE ; ; The REPEAT and WHILE structures can include the BREAK statement, which ; conditionally exits from the structure, and the NEXT statement, which ; conditionally branches to the bottom of the structure. ; ; The form of these two statements are: ; ; BREAK [label] IF condition ; ; NEXT [label] IF condition ; ; The optional label must be defined at the appropriate place in the ; structure. If omitted, these statements apply to the innermost structure ; which contains them. ; ; A CONDITION has the form of: ; |test ; |test AND test AND test .... | ; | OR OR | ; ; Up to six tests can be included in one condition. The tests are performed ; in left to right order, with no precedence. ; ; A TEST has the form: ; ; ; RELATION is the test to be performed and is specified as the appropriate ; suffix to the branch opcode; i.e. BBC is BC, BEQL is EQL, BLBS is LBS, etc. ; ARG1 and ARG2 are the operands for the test. If ARG2 is omitted, a TEST ; instruction is generated. If both arguments are omitted, the condition ; codes are assumed to be set and only the branch is generated. If both ; arguments are specified, a CMP instruction is generated. TYPE controls ; the data type of the comparison (default is L). ; ; Some examples: ; IF THEN ; ENDIF ; ; IF AND THEN ; ENDIF ; ; WHILE OR DO ; ENDWHILE ; ; REPEAT ; BREAK IF ; UNTIL FOREVER ; ; WHILE DO ; BREAK ALPHA IF ; ENDWHILE ;ALPHA: ; ; REPEAT ; NEXT BETA IF ;BETA: ; UNTIL ; ; ; This package also includes some macros for generating some conceptual ; data types: COUNT's, VECTOR's, and STRING's. A COUNT is an integer ; scalar variable (implemented as a longword). A VECTOR is a linear array ; of storage locations of a specified length and size. A string is a ; descriptor pointing to a vector of bytes. The vector may optionally be ; initialized with a character string. The forms of these declarations are: ; ; COUNT [initial-value] ; VECTOR length,[SIZE=| L | ; | W | ; | B | ; STRING length,[] ; ; Other miscellaneous macros: ; ; CALL routine arg1,arg2,arg3,... ; ; will stack the arguments in reverse order and perform a CALLS to the ; address ROUTINE. If the argument is an address, a PUSHAL is generated, ; and if the argument is a literal, a PUSHL is generated. ; ; ; Author: ; Gary L. Grebus, Creation Date: 08-Oct-1980 ; Battelle Columbus Labs ; ; Modified by: ; ;-- .PAGE .MACRO IF T1, C1, T2, C2, T3, C3, T4, C4, T5, C5, T6, C6 SMAC.INI ;; Init if needed .NARG ..NARG .IF BLANK, ;; Must have one connective .ERROR ; Missing condition or THEN in IF statement .MEXIT .ENDC .IF NE,<<..NARG/2>*2-..NARG> ;; Must be a pairs of arguments .ERROR ; Missing condition or THEN in IF statement .MEXIT .ENDC ..NOOR = 1 SMAC.IF ,,,,,,- ,,,,C1,T1 .IF EQ,..NOOR ;; An "OR" construct was used SMAC.TAG \SMAC..LBLN,2 .ENDC SMAC.PUSH \SMAC..LBLN,\SMAC..IF SMAC..LBLN = SMAC..LBLN + 1 .ENDM IF .PAGE .MACRO ELSE .IF NDF, SMAC..INI .ERROR ; ELSE occurs before any IF's .MEXIT .ENDC SMAC.PEEK ..LBL, ..TYP ;;Check top entry on stack .IF NE,<..TYP - SMAC..IF> .ERROR ; Incorrect nesting of ELSE block .MEXIT .IF_FALSE SMAC.POP ..LBL, ..TYP ;; Get top stack entry .IRP ..N, \..LBL SMAC.EMIT .ENDR SMAC.TAG \..LBL, 3 SMAC.PUSH \..LBL, \SMAC..ELSE .ENDC .ENDM ELSE .PAGE .MACRO ENDIF .IF NDF, SMAC..INI .ERROR ; ENDIF occurs before any IF's .MEXIT .ENDC SMAC.PEEK ..LBL, ..TYP ;; Get top items off stack .IF EQ,<..TYP - SMAC..IF> ;; If correctly nested IF SMAC.TAG \..LBL, 3 ;; Generate terminal tag .IF_FALSE .IF EQ,<..TYP-SMAC..ELSE> ;; or correctly nested ELSE SMAC.TAG \..LBL,4 ;; Generate terminal tag .IF_FALSE .ERROR ; ENDIF does not terminate an IF block .MEXIT .ENDC .ENDC SMAC.POP ..LBL, ..TYP ;; Since we know nesting ok, now ;; remove stuff from stack .ENDM ENDIF .PAGE .MACRO REPEAT SMAC.INI ;; Init if needed ;; Stack label, type, and terminator suffix SMAC.PUSH \SMAC..LBLN, \SMAC..REPEAT SMAC.TAG \SMAC..LBLN, 1 SMAC..LBLN = SMAC..LBLN + 1 .ENDM REPEAT .PAGE .MACRO UNTIL T1,C1,T2,C2,T3,C3,T4,C4,T5,C5,T6,C6 .IF NDF,SMAC..INI .ERROR ; UNTIL occurs before any REPEAT's .MEXIT .ENDC SMAC.PEEK ..LBL, ..TYP .IF NE,<..TYP-SMAC..REPEAT> .ERROR ; UNTIL does not terminate a REPEAT .MEXIT .ENDC SMAC.POP ..LBL, ..TYP ;; Since nesting is OK, ok to pop .IRP CC,\..LBL SMAC.TAG 'CC',2 ;; Generate NEXT tag .ENDR .NARG ..NARG .IF EQ,..NARG .ERROR ; UNTIL requires a condition, FOREVER, or ONCE .MEXIT .ENDC SMAC.UNTIL ,,,,,,,,,,- C1, T1 SMAC.TAG \..LBL,3 .ENDM UNTIL .PAGE .MACRO WHILE T1, C1, T2, C2, T3, C3, T4, C4, T5, C5, T6, C6 SMAC.INI ;; Init if needed .NARG ..NARG .IF BLANK, ;; Must have one connective .ERROR ; Missing condition or DO in WHILE statement .MEXIT .ENDC .IF NE,<<..NARG/2>*2-..NARG> ;; Must be a pairs of arguments .ERROR ; Missing condition or DO in WHILE statement .MEXIT .ENDC ..NOOR = 1 SMAC.TAG \SMAC..LBLN,1 ;; Generate top-of-loop tag SMAC.WHILE ,,,,,,- ,,,,C1,T1 .IF EQ,..NOOR ;; An "OR" construct was used SMAC.TAG \SMAC..LBLN,4 .ENDC SMAC.PUSH \SMAC..LBLN,\SMAC..WHILE SMAC..LBLN = SMAC..LBLN + 1 .ENDM WHILE .PAGE .MACRO ENDWHILE .IF NDF, SMAC..INI .ERROR ; ENDWHILE occurs before any WHILE's .MEXIT .ENDC SMAC.PEEK ..LBL, ..TYP ;; Get top items off stack .IF NE,<..TYP - SMAC..WHILE>;; If not correctly nested .ERROR ; ENDWHILE does not terminate a WHILE .MEXIT .ENDC SMAC.POP ..LBL, ..TYP ;; Since we know nesting ok, now ;; remove stuff from stack SMAC.TAG \..LBL,2 ;; Generate NEXT tag .IRP CC,\..LBL ;; Generate loop branch SMAC.EMIT .ENDR SMAC.TAG \..LBL,3 ;; Generate BREAK and exit tag .ENDM ENDWHILE .PAGE .MACRO BREAK BRKID, IFDUM,T1,C1,T2,C2,T3,C3,T4,C4,T5,C5,T6,C6 ;; Macro to implement break from a structure .IF NDF, SMAC..INI .ERROR ; BREAK occurs before any structures .MEXIT .ENDC .IF IDN,BRKID, ;; No ID. Break innermost struct. SMAC.PEEK ..LBL, ..TYP, LEV=0 ;; Get info on structure to break .IF LE,..TYP-4 ;; Is it a breakable structure .ERROR ; Can't BREAK from an IF structure .MEXIT .IF_FALSE ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,- ,, .IRP CC,\..LBL SMAC.EMIT .ENDR ENDIF .ENDC .IF_FALSE ;; Break to a label ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,,- , SMAC.EMIT ENDIF .ENDC .ENDM BREAK .PAGE .MACRO NEXT NXTID, IFDUM,T1,C1,T2,C2,T3,C3,T4,C4,T5,C5,T6,C6 ;; Macro to implement next interation in a structure .IF NDF, SMAC..INI .ERROR ; NEXT occurs before any structures .MEXIT .ENDC .IF IDN,NXTID, ;; No ID. Next innermost struct. SMAC.PEEK ..LBL, ..TYP, LEV=0 ;; Get info on structure to next .IF LE,..TYP-4 ;; Is it a nextable structure .ERROR ; Can't NEXT in an IF structure .MEXIT .IF_FALSE ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,- ,, .IRP CC,\..LBL SMAC.EMIT .ENDR ENDIF .ENDC .IF_FALSE ;; Break to a label ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,,- , SMAC.EMIT ENDIF .ENDC .ENDM NEXT .PAGE .MACRO CALL NAME,A0,A1,A2,A3,A4,A5,A6,A7,A8,A9 ;; macro to to a CALL_S type call. The argument stacking is handled. ;; Argument is pushed as a value if literal, as an address if an address ..CNT = 0 .IRP ARG, .IF NB,ARG .NTYPE ..TYP,ARG ;; Get addressing type ..TYP = ..TYP@-4&^XF .IF IDN,0, ;; If exactly zero PUSHL #0 ;; Stack zero .MEXIT .ENDC ..FLG = 0 .IIF LE,..TYP-1, ..FLG=1 .IIF EQ,..TYP-5, ..FLG=1 .IF EQ,..FLG ;; If mode is an address PUSHAL ARG .IF_FALSE PUSHL ARG ;; Else push value .ENDC ..CNT = ..CNT + 1 .ENDC .ENDR .IRP CC,\..CNT CALLS #'CC',NAME .ENDR .ENDM CALL .PAGE .MACRO COUNT INIT ;; Macro to define a data type called a "count". .IF NB,INIT .LONG INIT .IF_FALSE .LONG 0 .ENDC .ENDM COUNT .MACRO STRING LEN, INIT, ?SYM ;; macro to generate a descriptor followed by the string it describes. ;; the initialization is optional. .NCHR ..CNT, .IF GE, .WORD LEN .IF_FALSE .WORD ..CNT .IF_TRUE_FALSE .BYTE DSC$K_DTYPE_T .BYTE DSC$K_CLASS_S .ADDRESS SYM SYM: .ASCII ~INIT~ .IF_TRUE . = . + .ENDC .ENDM STRING .MACRO VECTOR NRELS,SIZE=L ;; Macro to define a vector of storage locations .BLK'SIZE NRELS .ENDM VECTOR .PAGE .MACRO ENB_LONG ;; Macro to enable use of all word displacement branching in the structures. ;; This is not the default. SMAC..LONGM = 1 .ENDM ENB_LONG .MACRO DSB_LONG ;; Macro to disable use of word displacement branching in the structures. ;; This is the default action SMAC..LONGM = 0 .ENDM DSB_LONG .PAGE .MACRO SMAC.IF C6,T6,C5,T5,C4,T4,C3,T3,C2,T2,C1,REL,ARG1,ARG2,TYPE=L ;; Process last pair of args and recursively call if any args left. .IRP CC,\SMAC..LBLN .IF IDN,, ..NOOR = 0 SMAC.GENTST , , , , <_.'CC'.2>, SMAC..NORM .IF_FALSE ;;Else simple or 'AND' connected SMAC.GENTST , , , , <_.'CC'.3>, SMAC..BRREV .ENDC .ENDR .IF NB, ;; If more tests SMAC.IF <>, <>, , , , , - , , , , C2, T2 .ENDC .ENDM SMAC.IF .PAGE .MACRO SMAC.UNTIL C6,T6,C5,T5,C4,T4,C3,T3,C2,T2,C1,REL,ARG1,ARG2,TYPE=L ;; Process last pair of args and recursively call if any args left. .IRP CC,\..LBL .IF IDN,, ;;If UNTIL FOREVER SMAC.EMIT .IF_FALSE .IF DIF,, ;; And not REPEAT ONCE .IF DIF,, ;;If not OR, assume simple or AND SMAC.GENTST , , , , - <_.'CC'.1>, SMAC..BRREV .IF_FALSE ;;Else OR connected SMAC.GENTST , , , ,- <_.'CC'.3>, SMAC..NORM .ENDC .ENDC .ENDC .ENDR .IF NB, ;; If more connectives SMAC.UNTIL <>, <>, , , , , - , , , , C2, T2 .ENDC .ENDM SMAC.UNTIL .PAGE .MACRO SMAC.WHILE C6,T6,C5,T5,C4,T4,C3,T3,C2,T2,C1,REL,ARG1,ARG2,TYPE=L ;; Process last pair of args and recursively call if any args left. .IRP CC,\SMAC..LBLN .IF IDN,, ..NOOR = 0 SMAC.GENTST , , , , <_.'CC'.4>, SMAC..NORM .IF_FALSE ;;Else simple or 'AND' connected SMAC.GENTST , , , , <_.'CC'.3>, SMAC..BRREV .ENDC .ENDR .IF NB, ;; If more connectives SMAC.WHILE <>, <>, , , , , - , , , , C2, T2 .ENDC .ENDM SMAC.WHILE .PAGE .MACRO SMAC.BRKNXT T1, C1, T2, C2, T3, C3, T4, C4, T5, C5, T6, C6 ;; Internal macro used by BREAK and NEXT to do an internal IF-THEN-ENDIF ..NOOR = 1 SMAC.IF ,,,,,,,,,,C1,T1 .IF EQ,..NOOR ;; An 'OR' construct was used SMAC.TAG \SMAC..LBLN,2 .ENDC SMAC.PUSH \SMAC..LBLN,\SMAC..IF SMAC..LBLN = SMAC..LBLN + 1 .ENDM SMAC.BRKNXT .PAGE .MACRO SMAC.INI ;; Define some constants and variables SMAC..INI = 1 ;; Init flag SMAC..IF = 1 ;; Structure type codes SMAC..REPEAT = 5 SMAC..WHILE = 6 SMAC..ELSE = 2 SMAC..BRREV = 1 ;; Branch sense codes SMAC..NORM = 0 SMAC..SP = 0 ;; Stack pointer SMAC..LBLN = 0 ;; Label generation counter SMAC..LONGM = 0 ;; Use short branches .IIF NDF,SMAC..LISTON, SMAC..LISTON = 0 $DSCDEF .MACRO SMAC.INI ;; Redefine this macro as a dummy .ENDM SMAC.INI .ENDM SMAC.INI .PAGE .MACRO SMAC.GENBR REL, DEST, SENSE ..LCP = %LENGTH(REL) - 1 ;;Possible postion of "U" if unsigned .IF IDN,%EXTRACT(..LCP,1,REL), SMAC.'REL , U, .IF_FALSE SMAC.'REL , <>, .ENDC .ENDM SMAC.GENBR .PAGE .MACRO SMAC.GENTST REL, ARG1, ARG2, TYPE, DEST, SENSE .IF BLANK,REL .ERROR ; Relation not specified .MEXIT .ENDC .IF BLANK,ARG2 ;; If no second arg .IF BLANK,ARG1 ;; And no first argument SMAC.GENBR , , ;; Just need a branch .IF_FALSE .IF DIF,%EXTRACT(0,2,REL),;;If not Low Bit test .IF DIF,REL,OKAY ;; or equivalent .IF DIF,REL,ERROR ;; Generate normal one arg form SMAC.EMIT SMAC.GENBR , , ;; Test and branch .MEXIT .ENDC .ENDC .ENDC ;; Drop through to here for low bit form SMAC.GENBR , , .ENDC .IF_FALSE .IF DIF,%EXTRACT(0,1,REL),;; If not a Branch Bit test SMAC.EMIT SMAC.GENBR , , .IF_FALSE SMAC.GENBR , , .ENDC .ENDC .ENDM SMAC.GENTST .PAGE .MACRO SMAC.GSLB REL, DEST, SIGNED, SENSE .IF EQ,SMAC..LONGM ;; If not in long mode, its easy SMAC.SHORTB , , , .MEXIT .ENDC .IF NDF,DEST ;; If branch target unknown SMAC.LONGB , , , ;; Use long branch .MEXIT .IF_FALSE .IF LT,<128-<.-DEST>> ;; Defined but too far away SMAC.LONGB , , , .MEXIT .IF_FALSE SMAC.SHORTB , , , .MEXIT .ENDC .ENDC .ENDM SMAC.GSLB .PAGE .MACRO SMAC.GSLB.B REL, SENSE, ARG1, DEST, DUMMY ;; Macro normally called with parameters: ;; , , DEST ;; where DEST can have one or two arguments preceeding the branch dest. .IF NB,DUMMY ;; If DUMMY has a value, then we have a two argument form. Combine ;; ARG1 and ARG2 into ARG1 SMAC.GSLB.B , , , .MEXIT .IF_FALSE ;; Normal expansion .IF EQ,SMAC..LONGM ;; If not in long mode, its easy SMAC.SHORTB.B , , , .MEXIT .ENDC .IF NDF,DEST ;; If branch target unknown SMAC.LONGB.B , , , ;; Use long branch .MEXIT .IF_FALSE .IF LT,<128-<.-DEST>> ;; Defined but too far away SMAC.LONGB.B , , , .MEXIT .IF_FALSE SMAC.SHORTB.B , , , .MEXIT .ENDC .ENDC .ENDC .ENDM SMAC.GSLB.B .PAGE .MACRO SMAC.SHORTB REL, DEST, SIGNED, SENSE .IF DIF,SENSE,SMAC..BRREV ;; Normal case branch SMAC.EMIT .MEXIT .IF_FALSE SMAC.R'REL , .MEXIT .ENDC .ENDM SMAC.SHORTB .MACRO SMAC.LONGB REL, DEST, SIGNED, SENSE, ?LBL1 ;; Macro to simulate a word displacement, conditional branch. ;; Generate a reverse sense branch to skip over a BRW instruction .IF IDN,SENSE,SMAC..BRREV ;; Reverse sense implies normal branch SMAC.EMIT .IF_FALSE SMAC.R'REL , .ENDC SMAC.EMIT SMAC.EMITL .ENDM SMAC.LONGB .PAGE .MACRO SMAC.SHORTB.B REL, ARG1, DEST, SENSE ;; Special case for branch bit instructions .IF DIF,SENSE,SMAC..BRREV ;; Normal case branch SMAC.EMIT .MEXIT .IF_FALSE SMAC.R'REL , .MEXIT .ENDC .ENDM SMAC.SHORTB.B .MACRO SMAC.LONGB.B REL, ARG1, DEST, SENSE, ?LBL1 ;; Macro to simulate a word displacement, conditional branch. ;; Generate a reverse sense branch to skip over a BRW instruction ;; Special case for branch bit instructions .IF IDN,SENSE,SMAC..BRREV ;; Reverse sense implies normal branch SMAC.EMIT .IF_FALSE SMAC.R'REL , .ENDC SMAC.EMIT SMAC.EMITL .ENDM SMAC.LONGB.B .PAGE .MACRO SMAC.TAG LBL, SUFFIX SMAC.EMITL <_.'LBL'.'SUFFIX> .ENDM SMAC.TAG .MACRO SMAC.EMIT VAL .IIF NE,SMAC..LISTON, .SHOW ME VAL .IIF NE,SMAC..LISTON, .NOSHOW ME .ENDM SMAC.EMIT .MACRO SMAC.EMITL VAL .IIF NE,SMAC..LISTON, .SHOW ME VAL: .IIF NE,SMAC..LISTON, .NOSHOW ME .ENDM SMAC.EMITL .PAGE .MACRO SMAC.PUSH LBL,TYP SMAC..SP = SMAC..SP + 1 .IRP CC, \SMAC..SP SMAC..STKT'CC = TYP SMAC..STKL'CC = LBL .ENDR .ENDM SMAC.PUSH .MACRO SMAC.PEEK LBL,TYP,LEV=0 .IF LE,SMAC..SP-LEV .ERROR ; Incorrect nesting or missing statemnt. .MEXIT .IF_FALSE ..TMP = SMAC..SP - LEV .IRP CC,\..TMP LBL = SMAC..STKL'CC TYP = SMAC..STKT'CC .ENDR .ENDC .ENDM SMAC.PEEK .MACRO SMAC.POP LBL,TYP SMAC.PEEK , SMAC..SP = SMAC..SP - 1 .ENDM SMAC.POP .PAGE ;; Special case branch instruction macros .MACRO SMAC.BC DEST, SIGNED, SENSE ;; Note: No brackets around DEST in following call SMAC.GSLB.B , , DEST .ENDM SMAC.BC .MACRO SMAC.BS DEST, SIGNED, SENSE ;; Note: No brackets around DEST in following call SMAC.GSLB.B , , DEST .ENDM SMAC.BS .MACRO SMAC.LBS DEST, SIGNED, SENSE ;; Note: No brackets around DEST in following call SMAC.GSLB.B , , DEST .ENDM SMAC.LBS .MACRO SMAC.LBC DEST, SIGNED, SENSE ;; Note: No brackets around DEST in following call SMAC.GSLB.B , , DEST .ENDM SMAC.LBC .MACRO SMAC.ERROR DEST, SIGNED, SENSE ;; Equiv to LBC ;; Note: No brackets around DEST in following call SMAC.GSLB.B , , DEST .ENDM SMAC.ERROR .MACRO SMAC.OKAY DEST, SIGNED, SENSE ;; Equiv to LBS ;; Note: No brackets around DEST in following call SMAC.GSLB.B , , DEST .ENDM SMAC.OKAY .MACRO SMAC.B DEST, SIGNED SMAC.EMIT .ENDM SMAC.B .PAGE ; Macros to handle each possible type of branch .MACRO SMAC.EQL DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.EQL .MACRO SMAC.NEQ DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.NEQ .MACRO SMAC.LEQ DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.LEQ .MACRO SMAC.GEQ DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.GEQ .MACRO SMAC.LSS DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.LSS .MACRO SMAC.GTR DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.GTR .MACRO SMAC.VS DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.VS .MACRO SMAC.VC DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.VC .MACRO SMAC.SC DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.SC .MACRO SMAC.CC DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.CC .MACRO SMAC.W DEST, SIGNED, SENSE SMAC.GSLB , , , .ENDM SMAC.W .MACRO SMAC.REQL DEST, SIGNED SMAC.EMIT .ENDM SMAC.REQL .MACRO SMAC.RNEQ DEST, SIGNED SMAC.EMIT .ENDM SMAC.RNEQ .MACRO SMAC.RGTR DEST, SIGNED SMAC.EMIT .ENDM SMAC.RGTR .MACRO SMAC.RLSS DEST, SIGNED SMAC.EMIT .ENDM SMAC.RLSS .MACRO SMAC.RGEQ DEST, SIGNED SMAC.EMIT .ENDM SMAC.RGEQ .MACRO SMAC.RLEQ DEST, SIGNED SMAC.EMIT .ENDM SMAC.RLEQ .MACRO SMAC.RVC DEST, SIGNED SMAC.EMIT .ENDM SMAC.RVC .MACRO SMAC.RVS DEST, SIGNED SMAC.EMIT .ENDM SMAC.RVS .MACRO SMAC.RCC DEST, SIGNED SMAC.EMIT .ENDM SMAC.RCC .MACRO SMAC.RSC DEST, SIGNED SMAC.EMIT .ENDM SMAC.RSC .MACRO SMAC.RLBC ARG1, DEST SMAC.EMIT .ENDM SMAC.RLBC .MACRO SMAC.RLBS ARG1, DEST SMAC.EMIT .ENDM SMAC.RLBS .MACRO SMAC.RBS ARG1, DEST SMAC.EMIT .ENDM SMAC.RBS .MACRO SMAC.RBC ARG1, DEST SMAC.EMIT .ENDM SMAC.RBC .MACRO SMAC.RW DEST, SIGNED SMAC.EMIT .ENDM SMAC.RW .END .PAGE .MACRO BREAK BRKID, IFDUM,T1,C1,T2,C2,T3,C3,T4,C4,T5,C5,T6,C6 ;; Macro to implement break from a structure .IF NDF, SMAC..INI .ERROR ; BREAK occurs before any structures .MEXIT .ENDC .IF IDN,BRKID, ;; No ID. Break innermost struct. SMAC.PEEK ..LBL, ..TYP, LEV=0 ;; Get info on structure to break .IF LE,..TYP-4 ;; Is it a breakable structure .ERROR ; Can't BREAK from an IF structure .MEXIT .IF_FALSE ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,- ,, .IRP CC,\..LBL SMAC.EMIT .ENDR ENDIF .ENDC .IF_FALSE ;; Break to a label ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,,- , SMAC.EMIT ENDIF .ENDC .ENDM BREAK .PAGE .MACRO NEXT NXTID, IFDUM,T1,C1,T2,C2,T3,C3,T4,C4,T5,C5,T6,C6 ;; Macro to implement next interation in a structure .IF NDF, SMAC..INI .ERROR ; NEXT occurs before any structures .MEXIT .ENDC .IF IDN,NXTID, ;; No ID. Next innermost struct. SMAC.PEEK ..LBL, ..TYP, LEV=0 ;; Get info on structure to next .IF LE,..TYP-4 ;; Is it a nextable structure .ERROR ; Can't NEXT in an IF structure .MEXIT .IF_FALSE ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,- ,, .IRP CC,\..LBL SMAC.EMIT .ENDR ENDIF .ENDC .IF_FALSE ;; Break to a label ;; Generates a pseudo IF-THEN-ENDIF SMAC.BRKNXT ,,,,,,,,,,- , SMAC.EMIT ENDIF .ENDC .ENDM NEXT .PAGE .MACRO CALL NAME,A0,A1,A2,A3,A4,A5,A6,A7,A8,A9 ;; macro to to a CALL_S type call. The argument stacking is handled. ;; Argument is pushed as a value if literal, as an address if an address ..CNT = 0 .IRP ARG, .IF NB,ARG .NTYPE ..TYP,ARG ;; Get addressing type ..TYP = ..TYP@-4&^XF .IF IDN,0, ;; If exactly zero PUSHL #0 ;; Stack zero .MEXIT .ENDC ..FLG = 0 .IIF LE,..TYP-1, ..FLG=1 .IIF EQ,..TYP-5, ..FLG=1 .IF EQ,..FLG ;; If mode is an address PUSHAL ARG .IF_FALSE PUSHL ARG ;; Else push value .ENDC ..CNT = ..CNT + 1 .ENDC .ENDR .IRP CC,\..CNT CALLS #'CC',NAME .ENDR .ENDM CALL .PAGE .MACRO COUNT INIT ;; Macro to define a data type called a "count". .IF NB,INIT .LONG INIT .IF_FALSE .LONG 0 .ENDC .ENDM COUNT .MACRO STRING LEN, INIT, ?SYM ;; macro to generate a descriptor followed by the string it describes. ;; the initialization is optional. .NCHR ..CNT, .IF GE, .WORD LEN .IF_FALSE .WORD ..CNT .IF_TRUE_FALSE .BYTE DSC$K_DTYPE_T .BYTE DSC$K_CLASS_S .ADDRESS SYM SYM: .ASCII ~INIT~ .IF_TRUE . = . + .ENDC .ENDM STRING .MACRO VECTOR NRELS,SIZE=L ;; Macro to define a vector of storage locations .BLK'SIZE NRELS .ENDM VECTOR .PAGE .MACRO SMAC.BRKNXT T1, C1, T2, C2, T3, C3, T4, C4, T5, C5, T6, C6 ;; Internal macro used by BREAK and NEXT to do an internal IF-THEN-ENDIF ..NOOR = 1 SMAC.IF ,,,,,,,,,,C1,T1 .IF EQ,..NOOR ;; An 'OR' construct was used SMAC.TAG \SMAC..LBLN,2 .ENDC SMAC.PUSH \SMAC..LBLN,\SMAC..IF SMAC..LBLN = SMAC..LBLN + 1 .ENDM SMAC.BRKNXT