Trailing-Edge
-
PDP-10 Archives
-
decuslib10-02
-
43,50263/sddutl.mac
There are 2 other files named sddutl.mac in the archive. Click here to see a list.
TITLE S$$UTL SYSTEM UTILITY ROUTINES
SUBTTL S$$EFI 'ENTRY.FUNCTION' INITIALIZATION ROUTINE
ENTRY S$$EFI
EXTERN S$$MFB,S$$KWD,S$$PBP,S$$FLP,S$$IPR,S$$CPF,S$$FLR
RADIX 10
SEARCH S$$NDF
COMMENT/
XWD STNO,PARBLK ; WHERE STNO IS THE STATEMENT # OF THE
CALL: JSP R11,S$$EFI ; 'ENTRY.FUNCTION' DECLARATION, PARBLK
PROT DESCR ; IS THE PARAMETER BLOCK, AND PROT DESCR AND
LABL DESCR ; LABL DESCR THE PROTOTYPE AND LABEL DESCRIP-
TORS. THE PROGRAM IS INITIALIZED IF NECESSARY, THE FUNCTION DEFINED AND
CALLING SEQUENCE MODIFIED TO APPEAR LIKE A NORMAL FUNCTION DEFINITION,
AND THE FUNCTION IS CALLED USING THE NEW DEFINITION/
S$$EFI: PUSH ES,1(R11) ; SAVE LABEL DESCR ON ES
PUSH ES,(R11) ; SAVE PROTOTYPE DESCR ON ES
PUSH SS,R3 ; SAVE # OF ARGS
PUSH SS,S$$PBP ; SAVE PARBLK+1
PUSH SS,S$$KWD+2 ; SAVE &STNO
PUSH SS,S$$FLP ; SAVE FAILPOINT
PUSH SS,R11 ; SAVE LINK
MOVE R10,-2(R11) ; GET NEW &STNO,PARBLK
HLRZM R10,S$$KWD+2 ; SET &STNO
MOVEI R9,1(R10) ; GET PARBLK+1
MOVEM R9,S$$PBP ; SET
HRLI R9,1B18+4 ; FORM NEW WORD FOR CALLING SEQUENCE
MOVEM R9,-2(R11) ; AND STORE IT THERE
SKIPL (R10) ; HAS PROGRAM BEEN INITIALIZED?
JSP R11,S$$IPR ; NO, INITIALIZE IT
HRROI R10,EFIFAL ; SET UP DUMMY FAILPOINT
MOVEM R10,S$$FLP
SETZ R0, ; MAKE FUNCTION BLOCK, LOCAL VARS POSSIBLE,
JSP R11,S$$MFB ; BUT NO FUNCTION WORD
POP SS,R11 ; RESTORE LINK
MOVEM R10,1(R11) ; SAVE FUNCTION DEFINITION
MOVEM R9,(R11) ; PARAMETERS IN CALLING SEQUENCE
MOVEI R10,S$$CPF ; GET 'CALL PROGRAMMER-DEFINED FUN'
POP SS,S$$FLP ; RESTORE FAILPOINT
POP SS,S$$KWD+2 ; AND &STNO
POP SS,S$$PBP ; AND PARBLK+1
POP SS,R3 ; AND # OF ARGS
HRRM R10,-1(R11) ; CHANGE CALLING SEQUENCE
JRST (R10) ; AND GO THERE
; FAILURE DURING FUNCTION DEFINITION
EFIFAL: SUB SS,[XWD 1,1] ; POP SS
POP SS,S$$FLP ; RESTORE FAILPOINT
POP SS,S$$KWD+2 ; RESTORE &STNO
POP SS,S$$PBP ; RESTORE PARBLK+1
JRST S$$FLR ; AND FAIL
PRGEND
SUBTTL S$$MFB MAKE FUNCTION BLOCK ROUTINE
ENTRY S$$MFB
EXTERN S$$GFP,S$$PGL,S$$LKV,S$$LKL,S$$LKF,S$$GNS,S$$SRT,S$$NRT
EXTERN S$$FRT,S$$MKS
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R11,S$$MFB ; WITH LABEL, THEN FUNCTION PROTOTYPE
PUSHED ONTO ES, AND FLAG IN R0 INDICATING FUNCTION TYPE (-1='ENTRY.FOR-
TRAN.FUNCTION',0='ENTRY.FUNCTION',1='DEFINE' FUNCTION). RETURNS XWD
-(NARG+NLV+1),ARGBLK IN R10, XWD STARTL,NLV+1 IN R9, AND, IF REQUIRED,
A FUNCTION WORD POINTER IN R8/
S$$MFB: MOVEM R0,FTPFLG ; SAVE FLAG
JUMPLE R0,.+2 ; IS IT 'DEFINE'?
SETZ R0, ; YES, WANT LOCAL VARS ALSO
HRRM R11,MFBRET ; SAVE LINK
MOVE R1,(ES) ; GET PROTOTYPE
TLNN R1,^O770000 ; IS IT STRING?
JSP R11,S$$GFP ; YES, GET FUNCTION PARAMETERS
CFERR 6,S$$PGL ; NO, BAD PROTOTYPE
MOVEM R9,SAVNLV ; SAVE # OF LOCAL VARS
MOVEI R0,1(R10) ; GET NARG+NLV+2
JSP R6,S$$GNS ; GET ARGUMENT BLOCK
MOVNI R2,(R10) ; GET -(NARG+NLV+1)
HRLI R1,(R2) ; INTO LH OF R1
MOVEM R1,SAVABP ; SAVE XWD -(NARG+NLV+1),ARGBLK
HRLZI R11,-1(R10) ; NARG+NLV,0
ADDI R10,(R1) ; ARGBLK+NARG+NLV+1
HRRM R10,(R1) ; SAVE IN ARGBLK
HRRI R11,(R10) ; AND AS POINTER TO LAST ARG BLOCK ENTRY
VARLOP: POP ES,R1 ; GET NEXT PREVIOUS SYMBOL
JSP R10,S$$LKV ; DO VARIABLE LOOKUP
HRLZI R3,3B23 ; DEDICATED TYPE MASK
AND R3,(R2) ; GET DEDICATED TYPE
LSH R3,-12 ; FORM XWD NAMPTR,DEDTYP
LSHC R2,-18
MOVEM R3,(R11) ; SAVE IN ARG BLOCK
SUB R11,[XWD 1,1] ; DECREMENT POINTER
JUMPGE R11,VARLOP ; LOOP IF ANY SYMBOLS LEFT
ADD ES,[XWD 1,1] ; PRESERVE FUNCTION SYMBOL
MOVE R1,(ES) ; GET IT
SKIPLE FTPFLG ; IS FUNCTION WORD WANTED?
JSP R10,S$$LKF ; YES, DO FUNCTION LOOKUP
HRRM R2,RSTFPT ; SAVE PTR TO FUNCTION WORD
MOVE R1,-2(ES) ; GET LABEL DESCR
SETO R0, ; MUST BE STRING
JSP R7,S$$MKS
CFERR 6,S$$PGL
SETZ R0, ; GET CHAR COUNT OF LABEL
HRRZ R0,(R1)
JUMPN R0,.+2 ; SKIP IF NON-NULL, OR
MOVE R1,(ES) ; USE FUNCTION SYMBOL
JSP R10,S$$LKL ; DO LABEL LOOKUP
MOVE R10,SAVABP ; GET XWD -(NARG+NLV+1),ARGBLK
AOS R9,SAVNLV ; GET XWD STARTL,NLV+1
HRLI R9,(R2)
SUB ES,[XWD 3,3] ; POP FUNC SYM, PROTOTYPE, LABEL OFF ES
RSTFPT: MOVEI R8,.-. ; GET FUNCTION WORD POINTER
MFBRET: JRST .-. ; RETURN
; STORAGE
FTPFLG: S$$SRT ; TO FORCE LOADING OF NON-DUMMY 'RETURN', 'FRETURN',
SAVNLV: S$$FRT ; AND 'NRETURN' IN CASE ONLY INDIRECT REFERENCES TO
SAVABP: S$$NRT ; THEM ARE MADE
PRGEND
SUBTTL S$$CPF CALL PROGRAMMER-DEFINED FUNCTION ROUTINE
ENTRY S$$CPF
EXTERN S$$PGL,S$$STP,S$$STB,S$$TMS,S$$KWD,S$$PBP,S$$FLP,S$$CPS
EXTERN S$$MKS,S$$MKI,S$$MKR,S$$TAC,S$$FLR
RADIX 10
SEARCH S$$NDF
COMMENT/
XWD ,PARBLK+1 ; WHERE PARBLK IS THE PARAMETER BLOCK
CALL: JSP R11,S$$CPF ; NLV IS THE # OF LOCAL VARIABLES, STARTL
XWD STARTL,NLV+1 ; IS THE LOCATION OF THE STARTING LABEL
XWD -(NARG+NLV+1),ARGBLK ; WORD, NARG IS THE # OF FORMAL
ARGUMENTS, AND ARGBLK IS THE ARGUMENT BLOCK POINTER. EXPECTS RETURN
LINK IN S$$PGL, CALL MODE LINK IN R12, AND # OF ARGS IN R3, WITH ARGS
PUSHED ONTO ES/
S$$CPF: HRL R12,S$$PGL ; GET PROGRAM LINK
PUSH SS,R12 ; SAVE , WITH R12, ON SS
MOVN R12,S$$STB ; SAVE CURRENT ES - BASE
ADD R12,ES ; IN CASE STACK OVERFLOWS AND CHANGES BASE
MOVE R10,(R11) ; GET STARTL,NLV+1
HLRM R10,STLABL ; SAVE PTR IN XCT INSTR
MOVEI R10,(R10) ; GET NLV+1
SETZ R0, ; NULL VALUES
PUSH ES,R0 ; PUSH EXTRA VALUES ONTO ES
SOJG R10,.-1 ; LOOP
ADD R12,S$$STB ; ADD BASE AND POINT TO FIRST
MOVEI R12,1(R12) ; VALUE ON ES BY SUTRACTING
SUBI R12,(R3) ; # OF ARGUMENTS
MOVE R8,-2(R11) ; SAVE NEW PARBLK+1
MOVE R11,1(R11) ; GET -(NARG+NLV+1),ARGBLK
PUSH SS,S$$FLP ; SAVE FAILPOINT
PUSH SS,S$$TMS ; SAVE STATEMENT START TIME ON SS
MOVE R1,S$$PBP ; GET OLD PARBLK+1
HRL R1,S$$KWD+2 ; GET OLD &STNO
PUSH SS,R1 ; SAVE THEM ON SS
PUSH SS,S$$STP-1 ; SAVE SS PREVIOUS
PUSH SS,S$$STP ; SAVE ES PREVIOUS
MOVEI R1,CPFRET ; GET FUNCTION CALL RETURN LOC
HRLI R1,(R11) ; AND ARG BLOCK POINTER
PUSH SS,R1 ; SAVE ON SS
SETZ R1, ; NULL VALUE FOR FUNCTION VAR
SAVLOP: MOVE R10,1(R11) ; GET NAMLOC,VARTYP OF NEXT VAR
JRA R10,.+1(R10) ; GET NAME DESCR, GO TO:
JRST SAVUND ; UNDEDICATED VAR
JRST SAVSTR ; DEDICATED STRING
JRST SAVINT ; DEDICATED INTEGER
JRST SAVREL ; OR DEDICATED REAL
SAVUND: SETM R2,(R10) ; GET OLD VALUE, WITHOUT INPUT
MOVEM R1,(R10) ; REPLACE WITH NEW VALUE, POSSIBLE OUTPUT
MOVE R1,R2
SAVCOM: EXCH R1,(R12) ; SAVE OLD VALUE ON ES AND GET NEXT VALUE
AOBJP R11,SAVFIN ; JUMP OUT IF NO MORE VARS
AOJA R12,SAVLOP ; OR BUMP ES POINTER AND LOOP
SAVSTR: MOVE R9,R1 ; SAVE NEW VALUE
MOVE R1,(R10) ; GET OLD VALUE
JSP R7,S$$CPS ; MAKE A COPY
EXCH R1,R9 ; EXCHANGE WITH NEW VALUE
MOVE R2,(R10) ; SET UP FOR STORING NEW VALUE
HLRZ R0,(R2) ; IN DEDICATED STRING, COMPUTE MAX CHARS AVAILA-
SUBI R0,1 ; BLE
IMULI R0,5
JSP R7,S$$MKS ; STORE NEW VALUE
CFERR 1,S$$PGL
HRRM R3,@(R10) ; SAVE CHAR COUNT
MOVE R1,R9 ; GET COPY OF OLD VALUE
JRST SAVCOM ; GO SAVE ON ES
SAVINT: JSP R7,S$$MKI ; MAKE INTEGER FROM NEW VALUE
CFERR 1,S$$PGL
EXCH R1,(R10) ; EXCHANGE WITH OLD VALUE
TLO R1,1B18 ; MAKE DESCRIPTOR FROM OLD VALUE
TLZ R1,1B19
JRST SAVCOM ; GO SAVE ON ES
SAVREL: JSP R7,S$$MKR ; DITTO FOR REALS
CFERR 1,S$$PGL
EXCH R1,(R10)
LSH R1,-2
TLO R1,3B19
JRST SAVCOM
SAVFIN: MOVN R1,S$$STB ; COMPUTE NEW ES PREVIOUS
ADD R1,ES ; FROM CURRENT ES
MOVEM R1,S$$STP ; SAVE
MOVN R1,S$$STB-1 ; DITTO FOR SS
ADD R1,SS
MOVEM R1,S$$STP-1
AOS S$$KWD+3 ; INCREMENT &FNCLEVEL
HRRZM R8,S$$PBP ; SAVE NEW PARBLK+1
SETZM S$$KWD+2 ; ZERO &STNO
STLABL: XCT .-. ; EXECUTE LABEL WORD (JUMP TO STARTING LABEL)
; RETURN, FRETURN, AND NRETURN OF A PROGRAMMER-DEFINED FUNCTION
; EXPECTS 0, -1, OR 1 IN RH(R12) , RESPECTIVELY
CPFRET: HLRZ R11,1(SS) ; GET ARGBLK POINTER
POP SS,S$$STP ; RESTORE ES PREVIOUS
POP SS,S$$STP-1 ; RESTORE SS PREVIOUS
POP SS,R1
HLRZM R1,S$$KWD+2 ; RESTORE OLD &STNO
HRRZM R1,S$$PBP ; RESTORE OLD PARBLK+1
POP SS,S$$TMS ; RESTORE START TIME FOR OLD STNO
POP SS,S$$FLP ; RESTORE FAILPOINT POINTER
POP SS,R1 ; GET OLD LINK,R12
MOVEI R12,(R12) ; CLEAR LH OF RETURN INDEX
ADDI R12,(R1) ; ADD OLD R12
HLRZM R1,S$$PGL ; RESTORE OLD PROGRAM LINK
MOVE R10,1(R11) ; GET NAMPTR,TYPE FOR FUNCTION VARIABLE
JRA R10,.+1(R10) ; GET NAME DESCR, AND SAVE VALUE FOR:
JRST FVLUND ; UNDEDICATED VARIABLE
JRST FVLSTR ; DEDICATED STRING
JRST FVLINT ; DEDICATED INTEGER
JRST FVLREL ; OR DEDICATED REAL
FVLSTR: MOVE R1,(R10) ; GET DESCRIPTOR
JSP R7,S$$CPS ; MAKE A COPY
JRST FVLCOM ; GO SAVE
FVLINT: MOVE R1,(R10) ; GET INTEGER
TLO R1,1B18 ; MAKE DESCRIPTOR
TLZ R1,1B19
JRST FVLCOM ; GO SAVE
FVLREL: MOVE R1,(R10) ; DITTO FOR REAL
LSH R1,-2
TLO R1,3B19
JRST FVLCOM
FVLUND: SETM R1,(R10) ; GET DESCR (WITHOUT INPUT)
FVLCOM: MOVEM R1,S$$TAC ; SAVE IN TEMP AC
HRRZ R9,(R11) ; COMPUTE NARG+NLV+1
SUBI R9,(R11)
MOVE R11,(R11) ; POINTER TO BOTTOM OF ARGBLOCK
RSTLOP: MOVE R10,(R11) ; GET NEXT NAMPTR,TYPE
POP ES,R1 ; POP CORRESPONDING SAVED VALUE OFF ES
JRA R10,.+1(R10) ; GET NAME DESCR AND RESTORE:
JRST RSTUND ; UNDEDICATED VARIABLE
JRST RSTSTR ; DEDICATED STRING
JRST RSTINT ; DEDICATED INTEGER
JRST RSTREL ; OR DEDICATED REAL
RSTSTR: MOVE R2,(R10) ; MOVE OLD STRING COPY INTO DEDICATED
HRRZ R0,(R1) ; STRING LOC
JSP R7,S$$MKS
JFCL
HRRM R3,@(R10) ; SAVE CHAR COUNT
JRST RSTCOM ; GO LOOP
RSTINT: LSH R1,2 ; GET INTEGER FROM DESCR
ASH R1,-2
MOVEM R1,(R10) ; SAVE IN DED LOC
JRST RSTCOM ; GO LOOP
RSTREL: LSH R1,2 ; DITTO FOR REAL
MOVEM R1,(R10)
JRST RSTCOM
RSTUND: SETAM R1,(R10) ; SAVE DESCR IN LOC (BUT NO OUTPUT)
RSTCOM: SOJLE R9,RSTFIN ; JUMP OUT IF NO MORE VARS
SOJA R11,RSTLOP ; OR DECREMENT ARGBLK POINTER AND LOOP
RSTFIN: SETZ R1,
EXCH R1,S$$TAC ; GET FUNCTION VALUE
TLNN R12,-1 ; SKIP IF INDEX WAS -1 (FRETURN)
JRST (R12) ; 'RETURN' OR 'NRETURN'
JRST S$$FLR ; 'FRETURN', FAIL
PRGEND
SUBTTL S$$GFP GET FUNCTION PARAMETERS ROUTINE
ENTRY S$$GFP
EXTERN S$$BKT,S$$GRS
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R11,S$$GFP ; WITH PROTOTYPE ON ES, AND LOCAL VAR-
RIABLE FLAG (-1=NONE, 0=POSSIBLY) IN R0. RETURNS TO 0(R11) IF PROTOTYPE
IS BAD, OR TO 1(R11) WITH SYMBOLS PUSHED ONTO ES, SYMBOL COUNT IN R10,
AND, IF REQUIRED, LOCAL VARIABLE COUNT IN R9/
S$$GFP: JRST INIBKT ; INITIALIZE BREAK TABLES, THEN MODIFY THIS LOC
SETZM PROTCT ; INITIALIZE SYMBOL COUNT
MOVE R10,(ES) ; INITIALIZE CURSOR
SETZ R0, ; GET TOTAL CHAR COUNT
HRRZ R9,(R10)
JSP R8,GETSYM ; GET FIRST SYMBOL
JRST (R11) ; ERROR IF NO MORE CHARS
CAIE R7,"(" ; IS NEXT CHAR OPEN PAREN?
JRST (R11) ; NO, ERROR
JSP R8,SPNBLN ; SPAN BLANKS
JRST (R11) ; ERROR IF NO MORE CHARS
CAIN R7,")" ; IS NEXT CHAR CLOSE PAREN?
JRST CLSPRN ; YES
VARLOP: JSP R8,GETSYM ; GET NEXT SYMBOL
JRST (R11) ; ERROR IF NO MORE CHARS
MOVE R8,PROTCT ; GET # OF SYMBOLS
CAILE R8,16 ; < OR = 15 ARGS + FUNCTION?
JRST (R11) ; NO, BAD PROTOTYPE
CAIE R7," " ; IS NEXT CHAR BLANK OR TAB?
CAIN R7,^O11
JSP R8,SPNBLN ; YES, SPAN BLANKS
JUMPL R9,(R11) ; NO, OR IF OUT OF CHARS, ERROR
CAIE R7,"," ; IS NEXT CHAR A COMMA?
JRST CLSCHK ; NO, CHECK FOR CLOSE PAREN
JSP R8,SPNBLN ; YES, SPAN BLANKS
JRST (R11) ; ERROR IF NO MORE CHARS
JRST VARLOP ; LOOP
CLSCHK: CAIE R7,")" ; IS NEXT CHAR CLOSE PAREN?
JRST (R11) ; NO, ERROR
CLSPRN: JSP R8,SPNBLN ; SPAN BLANKS AFTER ")"
JRST GFPEND ; FINISH UP IF NO MORE CHARS
SKIPE PROTFL ; WERE LOCAL VARS EXPECTED?
JRST (R11) ; NO, ERROR
LVRLOP: AOS PROTFL ; ADD 1 TO LOCAL VAR COUNT
JSP R8,GETSYM ; GET NEXT SYMBOL
JRST GFPEND ; FINISH UP IF NO MORE CHARS
CAIE R7," " ; IS NEXT CHAR BLANK OR TAB?
CAIN R7,^O11
JSP R8,SPNBLN ; YES, SPAN BLANKS
JUMPL R9,GFPEND ; NO, OR IF OUT OF CHARS, FINISH UP
CAIE R7,"," ; IS NEXT CHAR A COMMA?
JRST (R11) ; NO, ERROR
JSP R8,SPNBLN ; YES, SPAN BLANKS
JRST (R11) ; ERROR IF NO MORE CHARS
JRST LVRLOP ; LOOP
GFPEND: MOVE R9,PROTFL ; GET LOCAL VAR COUNT
MOVE R10,PROTCT ; GET SYMBOL COUNT
JRST 1(R11) ; RETURN
; INITIALIZE BREAK TABLES
INIBKT: MOVE R1,.+2 ; GET 'MOVEM R0,PROTFL'
PROTFL: MOVEM R1,S$$GFP ; PLUG CALLING SEQUENCE
PROTCT: MOVEM R0,PROTFL ; SAVE R0
FRSTBL: MOVEI R0,1 ; BIT MARK
MOVEI R1,R3 ; PTR TO DUMMY FRSTBL IN R3-R6
MOVE R2,[XWD -26,"A"] ; UPPER CASE LETTERS
SETZB R3,R4 ; CLEAR TABLE
RESTBL: SETZB R5,R6
DPB R0,S$$BKT(R2) ; MARK UPPER CASE BITS
AOBJN R2,.-1
MOVE R2,[XWD -26,^O141] ; LOWER CASE LETTERS
DPB R0,S$$BKT(R2) ; MARK LOWER CASE BITS
AOBJN R2,.-1
MOVEI R1,R7 ; COPY INTO DUMMY RESTBL
MOVE R10,[XWD R3,R7] ; IN R7-R10
BLT R10,R10
MOVE R2,[XWD -10,"0"] ; DIGITS
DPB R0,S$$BKT(R2) ; MARK DIGIT BITS
AOBJN R2,.-1
DPB R0,S$$BKT+"." ; MARK DOT
DPB R0,S$$BKT+"-" ; MARK DASH
MOVE R0,[XWD R3,FRSTBL] ; MOVE TABLES INTO CORE
BLT R0,RESTBL+3
JRST S$$GFP+1 ; CONTINUE
; SPAN BLANKS ROUTINE: JSP R8,SPNBLN ; AUTOMATICALLY SKIPS THE FIRST
; CHARACTER, AND THEN SKIPS OVER SUCCEEDING BLANKS AND TABS. RETURNS
; TO 0(R8) IF IT RUNS OUT OF CHARACTERS, OR TO 1(R8) WITH THE NEXT CHAR
; IN R7, AND THE CURSOR BACKED UP TO JUST IN FRONT OF IT
SPNBLN: IBP R10 ; MOVE CURSOR 1 CHAR
SUBI R9,1 ; FORWARD AUTOMATICALLY
SPNLOP: MOVE R6,R10 ; SAVE CURSOR IN CASE OF BACKUP
SOJL R9,(R8) ; DECREMENT CHAR COUNT, LEAVE IF <0
ILDB R7,R10 ; GET NEXT CHAR
CAIE R7," " ; IS IT BALNK OR TAB?
CAIN R7,^O11
JRST SPNLOP ; YES, LOOP
MOVE R10,R6 ; NO, BACKUP CURSOR
AOJA R9,1(R8) ; AND RETURN
; GET SYMBOL ROUTINE: JSP R8,GETSYM ; PARSES SYMBOL, CREATES NEW STRING
; , PUSHES IT ONTO ES AND INCREMENTS PROTCT, AND RETURNS TO 0(R8) IF
; NO MORE CHARS, OR TO 1(R8) WITH NEXT CHAR IN R7, WITH CURSOR BACKED
; UP TO JUST IN FRONT OF IT. FAILS TO (R11) IF CAN'T FIND SYMBOL
GETSYM: MOVEI R0,1 ; AT LEAST 1 CHAR
MOVEI R1,FRSTBL ; GET BREAK TABLE FOR LETTERS
SOJL R9,(R11) ; FAIL IF NO MORE CHARS
ILDB R7,R10 ; GET CHAR
LDB R2,S$$BKT(R7) ; GET BREAK BIT
JUMPE R2,(R11) ; FAIL IF NOT LETTER
PUSH SS,R7 ; PUSH CHAR ONTO SS
MOVEI R1,RESTBL ; GET BREAK TABLE FOR LETTERS, DIGITS,.,-
SYMLOP: MOVE R6,R10 ; SAVE CURSOR IN CASE OF BACKUP
SOJL R9,SYMEND+2 ; DECREMENT CHAR COUNT, QUIT IF <0
ILDB R7,R10 ; GET CHAR
LDB R2,S$$BKT(R7) ; GET BREAK BIT
JUMPE R2,SYMEND ; QUIT IF NOT ON
PUSH SS,R7 ; PUSH CHAR ONTO SS
AOJA R0,SYMLOP ; INCREMENT SYMBOL CHAR COUNT AND LOOP
SYMEND: ADDI R8,1 ; RETURN TO 1(R8)
MOVE R10,R6 ; BACKUP CURSOR
ADDI R9,1
HRRM R0,SAVCNT ; SAVE SYMBOL CHAR COUNT
MUL R0,[^F0.2B0] ; COMPUTE # WORDS NEEDED
ADDI R0,2
JSP R6,S$$GRS ; GET BLOCK FOR SYMBOL
HRLI R1,^O700 ; FORM STRING DESCR
PUSH ES,R1 ; SAVE ON ES
AOS PROTCT ; INCREMENT SYMBOL COUNT
SAVCNT: MOVEI R2,.-. ; GET SYMBOL CHAR COUNT
HRRM R2,(R1) ; SAVE IN STRING BLOCK
HRLI R2,(R2) ; FORM XWD NCHAR,NCHAR
SUB SS,R2 ; RESTORE SS TO INITIAL VALUE
MOVN R2,R2 ; FORM XWD -NCHAR,FIRST CHAR PTR
HRRI R2,(SS)
AOBJN R2,.+1
SYMCHR: MOVE R0,(R2) ; GET NEXT CHAR OFF SS
IDPB R0,R1 ; PUT IN STRING
AOBJN R2,SYMCHR ; LOOP FOR EACH CHAR
JRST (R8) ; RETURN
PRGEND
SUBTTL S$$NGS,S$$NGF STRING NEGATION ROUTINES
ENTRY S$$NGS,S$$NGF
EXTERN S$$STB,S$$STP,S$$FLR,S$$FLP
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$NGS ; START NEGATION, FAILPT IS LOC TO GO TO
FAILPT ; IF NEGATION ARG FAILS (AND NEGATION THUS SUCCEEDS)
CALL: JRST S$$NGF ; NEGATION FAILS (BECAUSE ARG SUCCEEDED)/
S$$NGS: PUSH SS,S$$FLP ; SAVE OLD FAILPOINT
PUSH SS,S$$STP-1 ; SAVE SS PREVIOUS
MOVN R1,S$$STB ; COMPUTE NEW ES PREVIOUS
ADD R1,ES
EXCH R1,S$$STP ; EXCHANGE WITH OLD ES PREVIOUS
PUSH SS,R1 ; SAVE OLD ES PREVIOUS
PUSH SS,(R12) ; SAVE FAILPOINT
MOVN R1,S$$STB-1 ; COMPUTE NEW SS PREVIOUS
ADD R1,SS
MOVEM R1,S$$STP-1 ; AND SAVE
HRROI R1,NEGFAL ; COMPUTE NEW FAILPOINT
MOVEM R1,S$$FLP ; SAVE
JRST 1(R12) ; CONTINUE WITH ARGUMENT
NEGFAL: MOVE SS,S$$STB-1 ; RESTORE SS
ADD SS,S$$STP-1
POP SS,R12 ; GET FAILPOINT
POP SS,ES ; GET OLD ES PREVIOUS
EXCH ES,S$$STP ; EXCHANGE WITH CURRENT ES PREVIOUS
ADD ES,S$$STB ; UPDATE ES
POP SS,S$$STP-1 ; RESTORE OLD SS PREVIOUS
POP SS,S$$FLP ; GET OLD FAILPOINT
SETZ R1, ; RESULT IS NULL
JRST (R12) ; SUCCEED TO FAILPOINT
S$$NGF: SUB SS,[XWD 1,1] ; THROW AWAY FAILPOINT
POP SS,S$$STP ; RESTORE ES PREVIOUS
POP SS,S$$STP-1 ; RESTORE SS PREVIOUS
POP SS,S$$FLP ; RESTORE OLD FAILPOINT
JRST S$$FLR ; FAIL
PRGEND
SUBTTL S$$ADD,S$$SUB,S$$MUL,S$$DIV DESCRIPTOR ARITHMETIC ROUTINES
ENTRY S$$ADD,S$$SUB,S$$MUL,S$$DIV
EXTERN S$$PGL,S$$STN,S$$ITR
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$XXX ; WITH SECOND (RIGHT) ARG IN R1 AND
FIRST (LEFT) ARG ON ES, RETURNS RESULT IN R1 WITH ES POPED/
S$$ADD: JSP R11,ARITH ; ADD, INDEX=0
S$$SUB: JSP R11,ARITH ; SUBTRACT, INDEX=1
S$$MUL: JSP R11,ARITH ; MULTIPLY, INDEX=2
S$$DIV: JSP R11,ARITH ; DIVIDE, INDEX=3
ARITH: SUBI R11,S$$ADD+1
MOVEM R12,S$$PGL ; SAVE LINK
SETZ R2, ; GET DESCR TYPE
ROTC R1,2
JRST .+1(R2) ; CONVERT TO VALUE
JSP R7,S$$STN-1 ; STRING, CONVERT TO INTEGER OR REAL
CFERR 1,S$$PGL ; OTHER, OR FAILED STRING CONVERSION
ASH R1,-2 ; INTEGER, RESTORE VALUE
MOVEI R10,(R2) ; REAL, OR SUCEEDED STRING CONV, SAVE TYPE
MOVE R9,R1 ; SAVE VALUE
POP ES,R1 ; GET LEFT HAND SIDE
SETZ R2, ; SIMILAR CONVERSION
ROTC R1,2
JRST .+1(R2)
JSP R7,S$$STN-1
CFERR 1,S$$PGL
ASH R1,-2
CAIE R2,(R10) ; TYPES THE SAME?
JRST MIXMOD ; NO, MIXED MODE
XCT OPERAT(R11) ; YES, PERFORM OPERATION
MAKDSC: JRST .-1(R2) ; FORM DESCRIPTOR
LSH R1,2 ; FOR INTEGER
ROTC R1,-2 ; AND REAL
JRST (R12) ; RETURN
; OPERATION TABLE
OPERAT: XCT ADDOP-2(R2)
XCT SUBOP-2(R2)
XCT MULOP-2(R2)
XCT DIVOP-2(R2)
; MODE TABLES
ADDOP: ADD R1,R9
FAD R1,R9
SUBOP: SUB R1,R9
FSB R1,R9
MULOP: IMUL R1,R9
FMP R1,R9
DIVOP: JSP R4,[MOVEI R3,(R2)
IDIV R1,R9
MOVEI R2,(R3)
JRST (R4)]
FDV R1,R9
MIXMOD: CAIE R10,3 ; IS RIGHT HAND SIDE REAL?
EXCH R1,R9 ; NO, IS INTEGER, EXCHANGE WITH LHS
JSP R3,S$$ITR ; CONVERT ARG THAT IS INTEGER TO REAL
CAIE R10,3 ; WERE SIDES EXCHANGED?
EXCH R1,R9 ; YES, RE-EXCHANGE
MOVEI R2,3 ; NOW BITH SIDES ARE REAL
XCT OPERAT(R11) ; PERFORM OPERATION
ROTC R1,-2 ; FORM REAL DESCR
JRST (R12) ; RETURN
PRGEND
SUBTTL S$$EXP DESCRIPTOR MODE EXPONENTIATION ROUTINE
ENTRY S$$EXP
EXTERN S$$PGL,S$$STN,S$$ITR,EXP1.0,EXP2.0,EXP3.0
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$EXP ; WITH SECOND (RIGHT) ARG IN R1 AND
FIRST (LEFT) ARG ON ES, RETURNS RESULT IN R1 WITH ES POPPED/
S$$EXP: MOVEM R12,S$$PGL ; SAVE LINK
SETZ R2, ; GET DESCR TYPE
ROTC R1,2
JRST .+1(R2) ; CONVERT TO INTEGER OR REAL VALUE
JSP R7,S$$STN-1
CFERR 1,S$$PGL
ASH R1,-2
MOVEI R10,(R2) ; SAVE TYPE AND VALUE OF RHS
MOVE R9,R1
POP ES,R1 ; GET LHS
SETZ R2, ; LIKEWISE CONVERT
ROTC R1,2
JRST .+1(R2)
JSP R7,S$$STN-1
CFERR 1,S$$PGL
ASH R1,-2
CAIL R2,(R10) ; IS IT INTEGER ** REAL?
JRST .+3 ; NO
JSP R3,S$$ITR ; YES, MAKE IT REAL ** REAL
MOVEI R2,3
MOVE R0,R1 ; GET ARGS INTO POSITION FOR FORTRAN LIBRARY
MOVE R1,R9 ; CALL
PUSHJ SS,@EXPTBL-2(R2) ; EXECUTE PROPER EXPONENTIATION
IORI R2,(R10) ; FORM DOMINANT TYPE
MOVE R1,R0 ; GET VAL INTO POSITION
JRST .-1(R2) ; AND MAKE DESCR
LSH R1,2 ; INTEGER
ROTC R1,-2 ; REAL
JRST (R12) ; RETURN
EXPTBL: JRST EXP1.0 ; I ** I
JRST @EXPTB1-2(R10) ; R ** ?
EXPTB1: JRST EXP2.0 ; R ** I
JRST EXP3.0 ; R ** R
PRGEND
SUBTTL S$$NEG DESCRIPTOR UNARY - ROUTINE
ENTRY S$$NEG
EXTERN S$$PGL,S$$STN
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$NEG ; WITH DESCRIPTOR IN R1/
S$$NEG: MOVEM R12,S$$PGL ; SAVE LINK
SETZ R2, ; GET TYPE
ROTC R1,2
JRST .+1(R2) ; CONVERT TO INTEGER OR REAL VALUE
JSP R7,S$$STN-1 ; STRING
CFERR 1,S$$PGL ; OTHER, OR STRING CONV FAILED
ASH R1,-2 ; INTEGER
MOVN R1,R1 ; REAL, AND NEGATE
JRST .-1(R2) ; MAKE DESCR AGAIN
LSH R1,2 ; INTEGER
ROTC R1,-2 ; REAL
JRST (R12) ; RETURN
PRGEND
SUBTTL S$$CNC CONCATENATION ROUTINE
ENTRY S$$CNC
EXTERN S$$PGL,S$$GRS,S$$MKS,S$$TAC,S$$MST,S$$PTS,S$$PTX
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$CNC ; WHERE NCONC IS THE NUMBER OF ELEMENTS
NCONC ; IN THE CONCATENATION, ALL BUT THE LAST PUSHED ONTO ES
AND THE LAST IN R1. IF ALL BUT ONE ELEMENT IS NULL-VALUED, THE RESULT
IS THAT ELEMENT. IF ALL NON-NULL ELEMENTS ARE EITHER STRINGS, INTEGERS,
OR REALS, THE RESULT IS THEIR CONCATENATED STRING. IF ALL NON-NULL
ELEMENTS ARE EITHER STRINGS, INTEGERS, REALS, OR PATTERNS, THE RESULT IS
THEIR CONCATENATED PATTERN. IF ANY ELEMENT IS NONE OF THESE, AND ANOTHER
NON-NULL ELEMENT EXISTS, IT IS A TYPE ERROR. THE RESULT DESCRIPTOR IS RE-
TURNED IN R1, WITH ES RESTORED TO ITS INITIAL STATE (NCONC-1 ELEMENTS
POPPED OFF)/
S$$CNC: MOVEM R12,S$$PGL ; SAVE PROG LINK
PUSH ES,R1 ; SAVE LAST ELT
MOVN R11,(R12) ; FORM -(NCONC,NCONC)
HRLI R11,-1(R11)
MOVE R10,ES ; FORM RESET ES
ADD R10,R11
MOVEM R10,SAVNES ; AND SAVE
HRRI R11,(R10) ; FORM XWD -NCONC,PTR TO FIRST ELEMENT
AOBJN R11,.+1
MOVEM R11,SAVELP ; AND SAVE
SETZ R10, ; INITIALIZE ELEMENT COUNT=0
SETZB R9,R8 ; INITIALIZE CHAR COUNT, SAVED ELT =0
; SEARCH LOOP, CHECK EACH ELEMENT ON ES
SRCHLP: MOVE R1,(R11) ; GET NEXT ELEMENT
JUMPE R1,GTNXTS ; SKIP OUT IF NULL
JUMPL R1,SRCNUM ; JUMP IF INTEGER OR REAL
TLNE R1,^O770000 ; IS IT STRING?
JRST SRCPAT ; NO, TRY PATTERN
HRRZ R3,(R1) ; GET CHAR COUNT
JUMPN R3,SRCSTR ; JUMP IF NONZERO
SETZM (R11) ; OR MAKE ELEMENT NULL
JRST GTNXTS ; AND SKIP OUT
CNVNUM: SETO R0, ; MAKE STRING FROM INTEGER OR REAL
JSP R7,S$$MKS ; WILL ALLWAYS SKIP OVER NEXT INSTR
SRCSTR: SKIPN R8,R1 ; SAVE DESCR IN R8 AND SKIP
MOVEM R1,(R11) ; SAVE NEW STRING AS ELT
ADDI R9,(R3) ; ADD CHARS TO TOTAL
AOJA R10,GTNXTS ; INCREMENT ELT COUNT AND LOOP
SRCNUM: JUMPN R10,CNVNUM ; IF ELT COUNT > 0, GO CONVERT TO STRING
HRRM R11,SAVNPT ; OR SAVE PTR TO ELT
NUMLOP: AOBJP R11,RETSAV+1 ; POINT TO NEXT ELT OR FINISH
MOVE R8,(R11) ; GET NEXT ELT
JUMPE R8,NUMLOP ; LOOP IF NULL
TLNE R8,^O770000 ; IS IT STRING?
JRST NUMCNV ; NO
HRRZ R3,(R8) ; YES, GET CHAR COUNT
JUMPN R3,NUMCNV ; DON'T JUMP IF 0
SETZM (R11) ; SET ELT TO NULL
JRST NUMLOP ; AND KEEP LOOPING
NUMCNV: SETO R0, ; CONVERT SAVED NUMBER TO STRING
JSP R7,S$$MKS
SAVNES: BLOCK 1 ; NEVER EXECUTED, USE FOR STORAGE
SAVNPT: MOVEM R1,.-. ; SAVE NEW STRING DESCR IN ELT LOC
MOVEI R9,(R3) ; INITIALIZE CHAR COUNT
MOVE R1,R8 ; GET CURRENT DESCR
AOJA R10,SRCHLP+2 ; INCREMENT ELT COUNT AND PROCEED
SRCPAT: TLC R1,1B20 ; IS IT PATTERN?
TLNN R1,3B21
AOBJP R10,SAVELT ; YES, INCREMENT ELT COUNT, MARK AS PAT
JUMPN R10,SPCERR ; NO, ERROR IF ANOTHER NON-NULL ELEMENT
TLC R1,1B20 ; RESTORE DESCR
SPCLOP: AOBJP R11,RETSAV+1 ; POINT TO NEXT ELT OR FINISH
MOVE R8,(R11) ; GET NEXT ELT
JUMPE R8,SPCLOP ; LOOP IF NULL
TLNE R8,^O770000 ; IS IT STRING?
SPCERR: CFERR 1,S$$PGL ; NO, ERROR
HRRZ R3,(R8) ; GET # OF CHARS
JUMPE R3,SPCLOP ; LOOP IF 0
CFERR 1,S$$PGL ; OR ERROR
SAVELT: TLC R1,1B20 ; RESTORE DESCR
MOVE R8,R1 ; SAVE LATEST ELEMENT
GTNXTS: AOBJN R11,SRCHLP ; POINT TO NEXT ELEMENT AND LOOP
; ELEMENT SEARCH IS OVER, FORM NEW STRING OR PATTERN
CAIG R10,1 ; IS # ELTS >1 ?
JRST RETSAV ; NO, FINISH
MOVE R11,SAVELP ; YES, GET FIRST ELT POINTER
CAIL R10,^O777777 ; IS PATTERN FLAG ON?
JRST MAKPAT ; YES, GO MAKE PATTERN
MOVEI R0,(R9) ; NO, STRING, COMPUTE # OF WORDS NEEDED
MUL R0,[^F0.2B0]
ADDI R0,2
JSP R6,S$$GRS ; GET BLOCK
HRLI R1,^O700 ; FORM DESCR
MOVE R8,R1 ; SAVE
HRRM R9,(R1) ; AND SAVE # OF CHARS IN STRING BLOCK
MOVE R7,[XWD STRCHR,CHRLOP] ; MOVE CHAR LOOP INTO R4-R7
BLT R7,CHRBOT
STRLOP: MOVE R2,(R11) ; GET NEXT ELT
JUMPE R2, STRBOT ; SKIP OUT IF NULL
HRRZ R3,(R2) ; GET CHAR COUNT
JRST CHRLOP ; START LOOP
STRCHR: PHASE 4
CHRLOP: ILDB R0,R2 ; R4: GET CHAR FROM ELT
IDPB R0,R1 ; R5: PUT CHAR IN NEW STRING
SOJG R3,CHRLOP ; R6: LOOP
CHRBOT: JRST STRBOT ; R7: OR EXIT
DEPHASE
STRBOT: AOBJN R11,STRLOP ; LOOP FOR EACH ELEMENT
RETSAV: MOVE R1,R8 ; RESTORE RESULT DESCR
MOVE ES,SAVNES ; RESTORE POPPED ES
JRST 1(R12) ; RETURN
; AT LEAST ONE ELEMENT IS A PATTERN, CREATE PATTERN ROUTINE AND DATA BLOCK
MAKPAT: MOVEI R0,1(R10) ; GET # ELTS + 1
CAIG R0,2 ; IS # ELTS > 1?
JRST RETSAV ; NO, FINISH
JSP R6,S$$GRS ; GET DATA BLOCK
HRLI R1,3B20 ; MAKE PATTERN DESCR
MOVE R8,R1 ; SAVE
HRLI R1,^O700 ; FAKE STRING DESCR AND SAVE
MOVEM R1,S$$TAC ; IN CASE OF GARBAGE COLLECTION
LSH R0,1 ; GET 2*(#ELTS+1)
JSP R6,S$$GRS ; GET BLOCK FOR PATTERN ROUTINE
ADDI R1,1 ; PTR TO FIRST INST
HRRM R1,(R8) ; SAVE POINTER TO ROUTINE IN DATA BLOCK
MOVE R9,[MOVE R1,1(DT)] ; INSTR. TO FETCH ELT FROM DATBLK
MOVE R7,[JSP R9,S$$MST] ; INSTR. IF STRING
MOVE R6,[JSP R9,S$$PTX] ; ISTR. IF PATTERN
MOVEI R10,1(R8) ; FIRST ELT PTR IN DATA BLOCK
PATLOP: MOVE R2,(R11) ; GET NEXT ELT
JUMPE R2,PATBOT ; SKIP OUT IF NULL
TLNE R2,^O770000 ; IS IT STRING?
JRST PATELT ; NO, PATTERN
MOVEM R7,1(R1) ; YES, SAVE STRING MATCH INSTR
PATRET: MOVEM R2,(R10) ; SAVE ELT IN DATA BLOCK
MOVEM R9,(R1) ; SAVE ELT FETCH INSTR
ADDI R1,2 ; NEXT INSTR LOC IN PATTERN ROUTINE
ADDI R9,1 ; NEXT ELT FETCH INSTR
ADDI R10,1 ; NEXT DATA BLOCK LOC
PATBOT: AOBJN R11,PATLOP ; LOOP FOR EACH ELT
MOVE R6,[JRST S$$PTS] ; LAST INSTR OF PATTERN ROUTINE
MOVEM R6,(R1)
SETZM S$$TAC ; CLEAR DUMMY STRING DESCR
JRST RETSAV ; FINISH
PATELT: TLNE R2,1B22 ; IS SUBPAT RESTARTEABLE?
TLO R8,1B22 ; YES, SET RESTARTEABLE BIT OF WHOLE PAT
MOVEM R6,1(R1) ; SAVE PAT MATCH INST
JRST PATRET ; REJOIN LOOP
; STORAGE
SAVELP: BLOCK 1
PRGEND
SUBTTL S$$IVN,S$$IVV INDIRECT VARIABLE NAME AND VALUE ROUTINES
ENTRY S$$IVN,S$$IVV
EXTERN S$$PGL,S$$LKV,S$$CPS
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$IVN ; WITH KEY DESCRIPTOR IN R1, RETURNS
NAME DESCRIPTOR IN R1 (SAME AS KEY IF KEY IS NAME)
CALL: JSP R12,S$$IVV ; WITH KEY DESCRIPTOR IN R1, RETURNS
VALUE DESCRIPTOR IN R1 (DOES NO LOOKUP IF KEY IS NAME)/
S$$IVN: JSP R11,S$$IVV+1 ; NAME, INDEX=0
S$$IVV: JSP R11,S$$IVV+1 ; VALUE, INDEX=1
SUBI R11,S$$IVN+1
SETZ R2, ; GET DESCR TYPE
ROTC R1,4
CAIE R2,4 ; IS IT NAME?
JRST INDLKP ; NO, DO LOOKUP
ROTC R1,-4 ; RESTORE DESCR
INDCOM: XCT [JRST (R12)
MOVE R2,R1](R11) ; RETURN FOR NAME CALL
MOVE R1,(R2) ; GET VALUE
TLNE R2,1B22 ; IS IT DEDICATED INTEGER OR REAL?
JRST DEDVAR ; YES
TLNE R2,1B23 ; IS IT DEDICATED STRING?
JSP R7,S$$CPS ; YES, COPY
JRST (R12) ; RETURN VALUE
DEDVAR: TLNN R2,1B23 ; IS IT DEDICATED REAL?
JRST MKIDSC ; NO, MAKE INT DESCR
LSH R1,-2 ; MAKE REAL DESCR
TLO R1,3B19
JRST (R12) ; RETURN
MKIDSC: TLZ R1,1B19 ; MAKE INTEGER DESCR
TLO R1,1B18
JRST (R12) ; RETURN
INDLKP: MOVEM R12,S$$PGL ; SAVE LINK
ROTC R1,-4 ; RESTORE DESCR
JSP R10,S$$LKV ; DO LOOKUP
MOVE R1,(R2) ; GET NAME
JRST INDCOM ; RETURN OR GET VALUE
PRGEND
SUBTTL S$$ILB INDIRECT LABEL FUNCTION
ENTRY S$$ILB
EXTERN S$$PGL,S$$LKL
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$ILB ; WITH KEY DESCRIPTOR IN R1, EXECUTES
VALUE LOCATION/
S$$ILB: MOVEM R12,S$$PGL ; SAVE PROG LINK
JSP R10,S$$LKL ; DO LOOKUP
XCT (R2) ; PERFORM GOTO
PRGEND
SUBTTL S$$LKV,S$$LKL,S$$LKF VARIABLE, LABEL, AND FUNCTION LOOKUP
ENTRY S$$LKV,S$$LKL,S$$LKF
EXTERN S$$LKS,S$$PGL,S$$GLP,S$$GNS,S$$UDF
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R10,S$$LKV[S$$LKL,S$$LKF] ; WITH KEY DESCRIPTOR IN
R1, RETURNS POINTER TO VALUE LOCATION IN R2/
S$$LKV: JSP R9,S$$LKF+1 ; VARIABLE, INDEX = 0
S$$LKL: JSP R9,S$$LKF+1 ; LABEL, INDEX = 1
S$$LKF: JSP R9,S$$LKF+1 ; FUNCTION, INDEX = 2
SUBI R9,S$$LKL
MOVEI R0,(R9) ; GET TYPE
TLO R0,1B18 ; FORM TYPE*2+1
ROT R0,-4 ; IN BITS 0-4
JSP R8,S$$LKS ; DO LOOKUP
JRST (R10) ; FOUND
XCT NEWLKV(R9) ; NEW ENTRY, GET APPROPRIATE VALUE
NEWLKC: MOVEM R5,(R2) ; SAVE
JRST (R10) ; RETURN
NEWLKV: JRST .+3 ; VARIABLE
MOVE R5,[UFERR 8,S$$PGL] ; LABEL
MOVE R5,[XWD 1B19,S$$UDF] ; FUNCTION
HRRZ R5,S$$GLP+1 ; GET GLOBAL VARIABLE BLOCK
HLRZ R6,(R5) ; GET SIZE
ANDI R6,^O177777
CAMG R6,S$$GLP+2 ; ROOM LEFT?
JRST NEWVBL ; NO, MAKE NEW BLOCK
ADD R5,S$$GLP+2 ; YES, POINT TO NEXT AVAILABLE LOC
AOS S$$GLP+2 ; INCREMENT LOC INDEX
NEWVBC: TLO R5,1B19 ; FORM NAME DESCR
JRST NEWLKC ; GO BACK TO SEQUENCE
NEWVBL: HRRM R2,NEWVR2 ; SAVE VALUE POINTER
MOVEI R0,P$GVXT ; GET GLOBAL VARIABLE BLOCK EXTENSION SIZE
JSP R6,S$$GNS ; GET NONRETURNABLE BLOCK
MOVE R2,S$$GLP+1 ; GET VAR BLOCK LIST
HRRM R1,(R2) ; APPEND NEW BLOCK
HRRI R2,(R1)
MOVEM R2,S$$GLP+1
MOVEI R2,2 ; NEW AVAIL INDEX
MOVEM R2,S$$GLP+2
NEWVR2: MOVEI R2,.-. ; RESTORE VALUE POINTER
MOVEI R5,1(R1) ; FORM POINTER TO VARIABLE LOC
HRLI R1,1(R1) ; SET INITIAL VALUES
MOVEI R3,P$GVXT-1(R1) ; OF VARIABLES IN NEW VAR BLOCK
SETZM 1(R1) ; TO NULL
HRRI R1,2(R1)
BLT R1,(R3)
JRST NEWVBC ; FORM NAME AND GO BACK TO SEQUENCE
PRGEND
SUBTTL S$$LKS INDIRECTION SYMBOL LOOKUP ROUTINE
ENTRY S$$LKS
EXTERN S$$MKS,S$$PGL,S$$PBP,S$$SY1,S$$SY2,S$$GLP,S$$TBM,S$$MNS
RADIX 10
SEARCH S$$NDF
COMMENT"
CALL: JSP R8,S$$LKS ; WITH TYPE/0,0 IN R0, KEY DESCRIPTOR
IN R1. RETURNS TO 0(R8) IF FOUND, WITH POINTER TO VALUE LOC
IN R2. RETURNS TO 1(R8) IF NEW ENTRY, WITH POINTER TO VALUE LOC
IN R2, AND STRING VALUE OF KEY MADE NONRETURNABLE"
S$$LKS: TLNN R1,^O770000 ; IS IT A STRING?
JRST .+5 ; YES
MOVN R0,R0 ; NO, TRY TO CREATE ONE
JSP R7,S$$MKS
CFERR 1,S$$PGL ; NO GO
MOVN R0,R0 ; RESTORE TYPE
CAML R0,[7B4] ; IS IT < TYPE 7?
JRST SPCLKS ; NO, SPECIAL
ADD R0,@S$$PBP ; ADD TABLE NUMBER TO LH
HLRI R0, ; AND ZERO RH
JSP R7,S$$SY1 ; LOOKUP SYMBOL
JRST (R8) ; FOUND
ADD R0,[1B4] ; MAKE TYPE GLOBAL
TLZ R0,^O17777 ; WITH TABLE # = 0
JSP R7,S$$SY2 ; AND RETRY LOOKUP
JRST (R8) ; FOUND
MNELKS: MOVE R4,S$$GLP ; NOT FOUND, GET GLOBAL TABLE DESCR
JSP R7,S$$TBM ; MAKE NEW ENTRY
HRRM R2,RR2LKS ; SAVE VALUE POINTER
JSP R6,S$$MNS ; MAKE STRING BLOCK NONRETURNABLE
RR2LKS: MOVEI R2,.-. ; RESTORE VALUE POINTER
JRST 1(R8) ; RETURN NEW ENTRY
SPCLKS: JSP R7,S$$SY1 ; LOOKUP SYMBOL
JRST (R8) ; FOUND
JRST MNELKS ; NOT FOUND, MAKE NEW ENTRY
PRGEND
SUBTTL S$$TBM NEW TABLE ENTRY FUNCTION
ENTRY S$$TBM
EXTERN S$$GNS,S$$GRS,S$$TA1,S$$GLP
RADIX 10
SEARCH S$$NDF
COMMENT"
CALL: JSP R7,S$$TBM ; WITH TYPE/NO.,MAJORKEY IN R0, KEY
DESCRIPTOR IN R1, NEXT ENTRY POINTER IN R2, AND TABLE DESCRIPTOR
IN R4. RETURNS POINTER TO VALUE LOC OF NEW ENTRY IN R2, WITH
R0 AND R1 UNCHANGED"
S$$TBM: MOVE R3,1(R4) ; GET CURRENT SIZE AND POINTER
CAML R3,-1(R3) ; WITHIN CURRENT BLOCK?
JRST NEWEXT ; NO, GET NEW EXTENSION BLOCK
NEWEXR: HLRZ R5,R3 ; GET CURRENT SIZE
HRLI R3,4(R5) ; ADD 4 LOCS
MOVEM R3,1(R4) ; UPDATE CURRENT SIZE, POINTER
ADDI R3,(R5) ; PTR TO NEW ENTRY
MOVEM R0,1(R3) ; SAVE TYPE/NO.,MAJORKEY
MOVEM R1,2(R3) ; SAVE KEY DESCR
HLL R2,(R2) ; FORM CHAIN WORD
MOVEM R2,(R3) ; AND SAVE
HRLM R3,(R2) ; SPLICE ENTRY INTO CHAIN
MOVS R2,R2
HRRM R3,(R2)
MOVEI R2,3(R3) ; FORM POINTER TO VALUE LOC
JRST (R7) ; AND RETURN
NEWEXT: MOVEM R0,SAVTMP ; SAVE R0,R2,R4, AND KEY DESCR
MOVEM R2,SAVTMP+1
MOVEM R4,SAVTMP+2
MOVEM R1,S$$TA1
HRRZ R0,(R4) ; GET EXTENSION BLOCK SIZE
CAMN R4,S$$GLP ; IS TABLE GLOBAL SYMBOL TABLE?
JRST TBMGNS ; YES, GET NONRETURNABLE BLOCK
JSP R6,S$$GRS ; NO, GET RETURNABLE BLOCK
TBMGNR: MOVE R4,SAVTMP+2 ; RESTORE R4
MOVE R3,1(R4) ; GET LAST EXT POINTER
HRRM R1,-1(R3) ; SAVE EXTENSION POINTER TO NEW ONE
SUBI R0,2 ; EXTENSION SIZE MAX
HRLZM R0,1(R1) ; SAVE IN NEW EXTENSION BLOCK
MOVEI R3,2(R1) ; FORM NEW EXTENSION POINTER
SETZ R1,
EXCH R1,S$$TA1 ; RESTORE KEY DESCR
MOVE R2,SAVTMP+1 ; RESTORE R2 AND R0
MOVE R0,SAVTMP
JRST NEWEXR ; RETURN TO SEQUENCE
TBMGNS: JSP R6,S$$GNS ; GET NONRETURNABLE BLOCK
JRST TBMGNR
; STORAGE
SAVTMP: BLOCK 3
PRGEND
SUBTTL S$$TMR 'TIMER' OPTION ROUTINES
ENTRY S$$STT,S$$TMF,S$$TMX,S$$TMO
EXTERN S$$STE,S$$KWD,S$$PBP,S$$TMS,S$$OUC,S$$OUT,S$$ITS
RADIX 10
SEARCH S$$NDF
COMMENT/
STATEMENT TIMING
CALL: JSP R12,S$$STT ; FINISH TIMING ON LAST STATEMENT, START
XWD STNO,FAILPT ; TIMING ON NEW ONE, AND THEN GO TO
S$$STE. DOES NOTHING IF TIMING IS NOT ACTIVE FOR CURRENT ROUTINE,
OR DOES NOT FINISH TIMING ON LAST STATEMENT IF &STNO IS 0.
FUNCTION RETURN TIMING
CALL: JSP R5,S$$TMF ; SIMILAR TO S$$STT, BUT CALLED FROM
FUNCTION RETURN OR SYSTEM EXIT TO CLOSE OUT TIMING ON LAST &STNO.
TIMER EXIT STATISTICS
CALL: JSP R7,S$$TMX ; WITH PROGRAM LIST IN R6, OUTPUTS TIMING
STATISTICS FOR EACH PROGRAM BEING TIMED.
PROGRAM TIMER STATISTICS
CALL: JSP R7,S$$TMO ; WITH PARBLK POINTER IN R6, TIMING
BLOCK POINTER IN R5, OUTPUTS TIMING STATISTICS FOR PROGRAM
SPECIFIED BY PARBLK POINTER, LEAVES R6 UNCHANGED/
; STATEMENT AND FUNCTION RETURN TIMING
S$$STT: JSP R4,TMRCOM ; STATEMENT TIMING, INDEX = 0
S$$TMF: JSP R4,TMRCOM ; RETURN TIMING, INDEX = 1
TMRCOM: SUBI R4,S$$STT+1
HRRZ R3,@S$$PBP ; GET TIMING BLOCK POINTER
JUMPE R3,TMRFIN(R4) ; SKIP OUT IF NO TIMING
SETZ R0,
RUNTIM R0, ; GET CURRENT RUNTIME
MOVE R1,S$$TMS ; GET PREVIOUS RUNTIME
MOVEM R0,S$$TMS ; SAVE CURRENT RUNTIME
MOVE R2,S$$KWD+2 ; GET &STNO
JUMPE R2,TMRFIN(R4) ; SKIP OUT IF 0 (NO TIMING YET)
SUB R0,R1 ; COMPUTE ELAPSED TIME FOR STATEMENT
ADDM R0,(R3) ; ADD TO TOTAL TIME
HRLI R0,1 ; 1 MORE STATEMENT FOR COUNT
ADDI R3,(R2) ; TIMING BLOCK ENTRY FOR STATEMENT
ADDM R0,(R3) ; ADD 1,TIME TO TOTAL EXECUTION FOR STATEMENT
JRST TMRFIN(R4) ; EXIT
TMRFIN: JRST S$$STE ; STATEMENT TIMING EXIT
JRST (R5) ; RETURN TIMING EXIT
; TIMER EXIT STATISTICS
S$$TMX: HRRM R7,TMXFIN ; SAVE RETURN LINK
HLRZ R6,R6 ; GET FIRST PROGRAM PARBLK
TMXLOP: HRRZ R5,1(R6) ; GET TIMING BLOCK POINTER
JUMPE R5,.+2 ; SKIP IF 0
JSP R7,S$$TMO ; OTHERWISE OUTPUT STATISTICS
HRRZ R6,(R6) ; GET NEXT PARBLK POINTER
JUMPN R6,TMXLOP ; AND LOOP IF NONZERO
TMXFIN: JRST .-. ; OR RETURN
; PROGRAM TIMER STATISTICS
S$$TMO: MOVE R1,MSG1 ; "////TIMING STATISTICS FOR "
MOVEM R1,@S$$OUC ; OUTPUT
MOVE R1,-1(R6) ; GET PROGRAM NAME STRING DESCR
MOVEM R1,@S$$OUT ; OUTPUT
MOVE R1,[POINT 7,MSG2S+5,27] ; INSERT IN TOTAL TIME MESSAGE
HRRZ R2,(R5) ; GET TOTAL TIME
HRRM R2,PERCNT ; SAVE IN PERCENT CALCULATION
SETZM MSG2S+6 ; CLEAR CONVERSION AREA
JSP R4,S$$ITS ; CONVERT TO STRING
MOVE R1,MSG2 ; "// TOTAL TIME FOR PROGRAM = XXX MS."
MOVEM R1,@S$$OUT ; OUTPUT
MOVE R1,MSG3 ; TIMING STATISTICS COLLUMN HEADER
MOVEM R1,@S$$OUT ; OUTPUT
SETZM STSTAT+3 ; INITIALIZE STATEMENT #
HLRZ R1,(R5) ; SET UP LOOP POINTER FOR STATEMENT BLOCK
ANDI R1,^O177777
MOVNI R1,(R1)
HRLI R5,(R1)
AOBJP R5,(R7) ; START AT FIRST STATEMENT ENTRY
HRLI R7,(R6) ; SAVE PARBLK POINTER
TMOLP1: AOS STSTAT+3 ; INCREMENT STATEMENT NUMBER
MOVE R1,(R5) ; GET COUNT,TIME
HLRZM R1,STSTAT+2 ; SAVE COUNT
HRRZM R1,STSTAT+1 ; SAVE TIME
MOVEI R2,100 ; COMPUTE % OF TOTAL
IMULI R2,(R1)
PERCNT: IDIVI R2,.-.
HRRZM R2,STSTAT ; SAVE
MOVE R3,MSG4S ; BLANK OUT NUMERIC FIELDS
MOVEM R3,MSG4S+1
MOVEM R3,MSG4S+4
MOVEM R3,MSG4S+7
MOVEM R3,MSG4S+10
MOVEI R6,3 ; 4 STATISTICS LOOP
TMOLP2: MOVE R1,STPOIN(R6) ; GET BYTE POINTER FOR STATISTIC
MOVE R2,STSTAT(R6) ; GET STATISTIC
JSP R4,S$$ITS ; CONVERT TO STRING IN MESSAGE
SOJGE R6,TMOLP2 ; LOOP FOR EACH STATISTIC
MOVE R1,MSG4 ; GET STATISTICS LINE
MOVEM R1,@S$$OUT ; OUTPUT
AOBJN R5,TMOLP1 ; LOOP FOR EACH STATEMENT
HLRZ R6,R7 ; RESTORE PARBLK POINTER
JRST (R7) ; RETURN
; STORAGE
MSG1: POINT 7,.+1,35
BYTE (2)2(16)7(18)27
BYTE (7)^O12,^O12,^O12,^O12,"*"
ASCII/TIMING STATISTICS FOR /
MSG2: POINT 7,.+1,35
BYTE (2)2(16)9(18)39
MSG2S: BYTE (7)^O12,^O12," "," ","T"
ASCII/OTAL TIME FOR PROGRAM = MS./
MSG3: POINT 7,.+1,35
BYTE (2)2(16)13(18)62
BYTE (7)^O12,^O12," ","S","T"
ASCII/ATEMENT # OF EXECUTIONS TIME IN MS. /
ASCII/ % OF TOTAL TIME/
MSG4: POINT 7,.+1,35
BYTE (2)2(16)12(18)55
MSG4S: REPEAT 11,< ASCII/ />
STPOIN: POINT 7,MSG4S+10
POINT 7,MSG4S+7
POINT 7,MSG4S+4
POINT 7,MSG4S+1
STSTAT: BLOCK 4
PRGEND
SUBTTL S$$STE STATEMENT ENTRY ROUTINE
ENTRY S$$STE
EXTERN S$$KWD,S$$FLP,S$$ITS,S$$SST,S$$PBP,S$$OUT
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$STE ; WHERE STNO IS THE STATEMENT NUMBER
XWD STNO,FAILPT ; AND FAILPT IS THE STATEMENT FAILPOINT
&LASTNO IS SET TO &STNO,&STNO IS SET TO STNO, &STCOUNT IS INCRE-
MENTED AND TESTED AGAINST &STLIMIT, AND THE FAILPOINT POINTER IS SET.
IF &STNTRACE IS NOT 0, A STATEMENT TRACE MESSAGE IS OUTPUT/
S$$STE: MOVE R1,(R12) ; GET STNO, FAILPT
HRRZM R1,S$$FLP ; SET FAILPOINT
HLRZ R1,R1 ; GET STNO
EXCH R1,S$$KWD+2 ; UPDATE &STNO, GET OLD &STNO
MOVEM R1,S$$KWD+1 ; SAVE AS NEW &LASTNO
AOS R1,S$$KWD+4 ; INCREMENT &STCOUNT
CAML R1,S$$KWD+13 ; IS IT < &STLIMIT
UFERR 6,R12 ; NO, ERROR
SKIPN S$$KWD+11 ; IS &STNTRACE ON?
JRST 1(R12) ; NO, RETURN
SETZM TRCMSG+5 ; INITIALIZE TRACE MESSAGE
SETZM TRCMSG+7
SETZM TRCMSG+8
SETZM TRCMSG+11
SETZM TRCMSG+12
MOVE R1,[POINT 7,TRCMSG+5] ; EDIT IN &STNO
MOVE R2,S$$KWD+2
JSP R4,S$$ITS
MOVE R1,S$$PBP ; EDIT IN PROGRAM NAME
MOVE R1,-2(R1)
HRRZ R2,(R1)
CAILE R2,10
MOVEI R2,10
MOVE R3,[POINT 7,TRCMSG+7]
ILDB R0,R1
IDPB R0,R3
SOJG R2,.-2
MOVE R1,[POINT 7,TRCMSG+11] ; EDIT IN TIME
SETZ R2,
RUNTIM R2,
SUB R2,S$$SST
JSP R4,S$$ITS
MOVE R1,MSGDSC ; OUTPUT TRACE MESSAGE
MOVEM R1,@S$$OUT
JRST 1(R12) ; RETURN
; STORAGE
MSGDSC: POINT 7,.+1,35
BYTE (2)2(16)14(18)65
TRCMSG: ASCII/*STNTRACE* OF STATEMENT /
BLOCK 1
ASCII/ IN /
BLOCK 2
ASCII/ AT TIME= /
BLOCK 2
PRGEND
SUBTTL S$$CPS COPY STRING ROUTINE
ENTRY S$$CPS
EXTERN S$$GRS,S$$TA1
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R7,S$$CPS ; WITH STRING DESCRIPTOR IN R1, RETURNS
NEW STRING DESCRIPTOR IN R1/
S$$CPS: JUMPE R1,(R7) ; RETURN IF NULL
HRRZ R2,(R1) ; GET CHAR COUNT
JUMPN R2,.+3 ; IS IT 0?
SETZ R1, ; YES, SET TO NULL VALUE
JRST (R7) ; AND RETURN
MUL R2,[^F0.2B0] ; COMPUTE NUMBER OF WORDS NEEDED
MOVEI R0,2(R2)
MOVEM R1,S$$TA1 ; SAVE OLD DESCR
JSP R6,S$$GRS ; GET BLOCK FOR NEW STRING
HRLI R1,^O700 ; FORM STRING DESCR
MOVE R2,R1 ; FORM BLT WORD
HRL R2,S$$TA1
HRRZ R3,@S$$TA1 ; TRANSFER CHAR COUNT
HRRM R3,(R2)
MOVE R3,R0 ; FORM END ADDR FOR BLT
ADDI R3,-1(R2)
AOBJP R2,.+1 ; START BLT ON SECOND WORD OF BLOCKS
BLT R2,(R3)
SETZM S$$TA1
JRST (R7) ; RETURN
PRGEND
SUBTTL S$$GNP GET NEXT NUMERICAL PARAMETER ROUTINE
ENTRY S$$GNP
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R4,S$$GNP ; WITH BYTE POINTER IN R1, BYTE COUNT
IN R3. PROCESSES SIGNED INTEGER INCLUDING DELIMITER AND RETURNS
TO 1(R4) WITH INTEGER VALUE IN R2 AND DELIMITER IN R0, OR RE-
TURNS TO 0(R4) IF DELIMITER IS NOT FOUND, WITH INTEGER IN R2/
S$$GNP: SETZ R2, ; INITIAL INTEGER VALUE
SOJL R3,(R4) ; RETURN IF NO MORE CHARS
HLLI R4, ; INITIAL SIGN IS +
ILDB R0,R1 ; GET FIRST CHAR
CAIE R0,"-" ; IS IT A - SIGN?
JRST TRYPLS ; NO, TRY PLUS
HRLI R4,-1 ; SET SIGN TO -
JRST NXTDIG ; GO INTO LOOP
TRYPLS: CAIN R0,"+" ; IS IT A + SIGN
JRST NXTDIG ; YES, GO INTO LOOP
DIGLOP: CAIL R0,"0" ; IS IT A DIGIT?
CAILE R0,"9"
AOJA R4,GNPFIN ; NO, DELIMITER FOUND
SUBI R0,"0" ; GET INTEGER DIGIT
IMULI R2,10 ; TOT = TOT*10+DIGIT
ADD R2,R0
NXTDIG: SOJL R3,GNPFIN ; QUIT IF NO MORE CHARS
ILDB R0,R1 ; GET NEXT CHAR
JRST DIGLOP ; AND LOOP
GNPFIN: JUMPGE R4,(R4) ; RETURN IF + VALUE
MOVN R2,R2 ; OR NEGATE
JRST (R4) ; AND RETURN
PRGEND
SUBTTL S$$ASG ASSIGNMENT ROUTINE
ENTRY S$$ASG
EXTERN S$$DSG,S$$PGL
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R9,S$$ASG ; WITH VALUE IN R1 AND NAME DESCRIPTOR
ON ES, DOES NORMAL OR DEDICATED ASSIGNMENT/
S$$ASG: MOVEM R9,S$$PGL ; SAVE PROGRAM LINK
POP ES,R8 ; GET NAME OFF ES
TLNE R8,3B23 ; ARE DEDICATED BITS ON?
JRST S$$DSG ; YES, DEDICATED ASSIGNMENT
MOVEM R1,(R8) ; NO, NORMAL ASSIGNMENT (POSSIBLE OUTPUT)
JRST (R9) ; RETURN
PRGEND
SUBTTL S$$MVS MOVE STRING ROUTINE
ENTRY S$$MVS
RADIX 10
SEARCH S$$NDF
P$BRKE=8 ; BREAK EVEN POINT OF REGISTER LOOP
COMMENT/
CALL: JSP R7,S$$MVS ; WITH OBJECT BYTE POINTER IN R1, SOURCE
BYTE POINTER IN R2, AND CHARACTER COUNT (>0) IN R3/
S$$MVS: CAIL R3,P$BRKE ; FEWER CHARS THAN BREAK EVEN POINT?
JRST MOVLOP ; NO, MOVE LOOP INTO FAST REGISTERS
CHRLP1: ILDB R0,R2 ; GET CHAR FROM SOURCE
IDPB R0,R1 ; PUT CHAR IN OBJECT
SOJG R3,CHRLP1 ; LOOP
JRST (R7) ; OR RETURN
MOVLOP: HLL R7,S$$MVS+1 ; INSERT JRST IN LH OF R7
MOVE R6,[XWD CHRLOP,CHRLP2] ; MOVE LOOP INTO R4-R6
BLT R6,CHRLPE
JRST CHRLP2 ; START LOOP
CHRLOP: PHASE 4
CHRLP2: ILDB R0,R2 ; R4: GET CHAR
IDPB R0,R1 ; R5: PUT CHAR
CHRLPE: SOJG R3,CHRLP2 ; R6: LOOP
DEPHASE
PRGEND
SUBTTL S$$EQS STRING EQUALITY TEST ROUTINE
ENTRY S$$EQS
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R5,S$$EQS ; WITH STRING DESCRIPTORS IN R1 AND R2,
RETURNS TO 0(R5) IF EQUAL AND 1(R5) IF NOT EQUAL/
S$$EQS: CAMN R1,R2 ; ARE DESCRIPTORS EQUAL?
JRST (R5) ; YES, STRINGS MUST BE EQUAL
SETZ R0, ; ZERO R0 INCASE DESCR IS 0 (POINTS TO R0)
HRRZ R3,(R2) ; GET FIRST CHAR COUNT, INCLUDING NULL OR ZERO
HRRZ R0,(R1) ; GET SECOND CHAR COUNT, INCLUDING NULL OR ZERO
CAIE R0,(R3) ; ARE COUNTS EQUAL?
JRST 1(R5) ; NO, STRINGS UNEQUAL
JUMPE R0,(R5) ; STRINGS EQUAL IF 0 CHAR
CAIG R0,5 ; <6 CHARS?
JRST CHRLOP ; YES, DO CHAR LOOP
MUL R3,POINT2 ; NO, COMPUTE # WORDS
ROT R4,4 ; AND # OF REM CHARS
MOVE R4,REMTBL(R4)
HRRM R4,GETREM ; SAVE REM CHARS
TLC R1,^B1001B27 ; SET BYTE PTRS
TLC R2,^B1001B27 ; FOR 35-BIT BYTES
WRDLOP: ILDB R0,R1 ; GET WORD FROM FIRST
ILDB R4,R2 ; GET WORD FROM SECOND
CAME R0,R4 ; EQUAL?
JRST 1(R5) ; NO, STRINGS UNEQUAL
SOJG R3,WRDLOP ; LOOP FOR EACH WORD
TLC R1,^B1001B27 ; SET BYTE PTRS
TLC R2,^B1001B27 ; BACK TO 7-BIT BYTES
GETREM: MOVEI R3,.-. ; GET REM CHARS
CHRLOP: ILDB R0,R1 ; GET CHAR FROM FIRST
ILDB R4,R2 ; GET CHAR FROM SECOND
CAIE R0,(R4) ; EQUAL?
JRST 1(R5) ; NO, STRINGS UNEQUAL
SOJG R3,CHRLOP ; LOOP FOR EACH CHAR
JRST (R5) ; STRINGS EQUAL IF ALL CHARS MATCH
; STORAGE
REMTBL=.-1 ; REM=0, IMPOSSIBLE
1 ; REM=1, 1 CHAR REM
POINT2: ^O63146300000 ; REM=2, IMPOSSIBLE, USE SPACE
2 ; REM=3, 2 CHAR REM
3 ; REM=4, 3 CHAR REM
0 ; REM=5, IMPOSSIBLE
4 ; REM=6, 4 CHAR REM
5 ; REM=7, 5 CHAR REM
PRGEND
SUBTTL S$$SRT,S$$NRT,S$$FRT 'RETURN','NRETURN','FRETURN' LABELS
ENTRY S$$SRT,S$$NRT,S$$FRT
EXTERN S$$KWD,S$$RTP,S$$TMF,S$$PGL
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JRST S$$SRT[S$$NRT,S$$FRT] ; RESULT OF JUMP TO 'RETURN',
'NRETURN', OR 'FRETURN' LABELS/
S$$FRT: JSP R12,S$$NRT+1 ; 'FRETURN', INDEX=-1
S$$SRT: JSP R12,S$$NRT+1 ; 'RETURN', INDEX=0
S$$NRT: JSP R12,S$$NRT+1 ; 'NRETURN', INDEX=1
SUBI R12,S$$NRT
SOSGE S$$KWD+3 ; DECREMENT &FNCLEVEL
UFERR 2,S$$PGL ; RETURN FROM 0 LEVEL
MOVE R1,@RTNTYP(R12) ; GET RETURN TYPE
MOVEM R1,S$$KWD+6 ; SAVE IN &RTNTYPE
JSP R5,S$$TMF ; FINISH TIMING ON LAST STATEMENT
POPJ SS, ; GO TO APPROPRIATE FUNCTION RETURN ROUTINE
; STORAGE
S$$RTP+4 ; POINTS TO 'FRETURN' DESCRIPTOR
RTNTYP: S$$RTP ; POINTS TO 'RETURN' DESCRIPTOR
S$$RTP+8 ; POINTS TO 'NRETURN' DESCRIPTOR
PRGEND
SUBTTL S$$BGT BAD GOTO ERROR EXIT
ENTRY S$$BGT
EXTERN S$$PGL
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JRST S$$BGT ; CALLED BY FAILPOINT ROUTINE DURING GOTO EVA
LUATION/
S$$BGT: UFERR 3,S$$PGL ; FAILURE DURING GOTO EVALUATION
PRGEND
SUBTTL S$$NFE FAILURE UNDER 'NOFAIL' ERROR EXIT
ENTRY S$$NFE
EXTERN S$$PGL
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JRST S$$NFE ; CALLED BY FAILPOINT ROUTINE/
S$$NFE: UFERR 13,S$$PGL ; FAILURE UNDER 'NOFAIL'
PRGEND
SUBTTL S$$UDF UNDEFINED FUNCTION ERROR EXIT
ENTRY S$$UDF
EXTERN S$$PGL
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JRST S$$UDF ; CALLED BY FCALV OR FCALN/
S$$UDF: CFERR 5,S$$PGL ; UNDEFINED FUNCTION CALL
PRGEND
SUBTTL S$$CPE COMPILATION ERROR EXIT
ENTRY S$$CPE
RADIX 10
SEARCH S$$NDF
COMMENT/
CALL: JSP R12,S$$CPE ; EXECUTION OF STATEMENT WITH COMPI-
LATION ERROR/
S$$CPE: UFERR 12,R12
END