perm filename TOTAL[10X,AIL] blob
sn#342546 filedate 1978-03-26 generic text, type T, neo UTF8
00100 COMMENT ⊗ VALID 00052 PAGES
00200 C REC PAGE DESCRIPTION
00300 C00001 00001
00400 C00005 00002 HISTORY
00500 C00013 00003 DATA for Total (Low-level Code Production) Routines
00600 C00016 00004 Description of Total Routines
00700 C00026 00005 CONV, PRE, POST -- Type-Conversion routines
00800 C00032 00006
00900 C00036 00007
01000 C00043 00008
01100 C00045 00009 PUT
01200 C00049 00010 ACCESS,GETSDR,GETDR,DISBLK,ZOTDIS--last four only for dis
01300 C00059 00011 GET
01400 C00066 00012
01500 C00069 00013
01600 C00073 00014
01700 C00079 00015 STACK -- Issue Instrs. to Stack Anything on Approp. Stack
01800 C00084 00016 MARK, MARKINT, MARKME -- Mark Semblk with Correct Temp Semantics
01900 C00088 00017 INCOR -- Issue Code to Clear this Entity from ACs
02000 C00089 00018 REMOPs, CLEARs -- Remove Temps, ACs, from Use
02100 C00101 00019 DSCR CLEAR,CLEARL,CLEARA
02200 C00104 00020 STROP -- Bit-Driven String Operation Code Generator
02300 C00111 00021 GETTEM, etc. -- Temp Semblk Allocators
02400 C00115 00022 GETAC, GETAN0 -- AC Allocators
02500 C00121 00023 AC Store routines -- BOLSTO, FORSTO, STORIX, GOSTO, STORZ
02600 C00126 00024 STORA -- main AC-storing subr. -- called by above
02700 C00133 00025 EMITER -- Descriptions of Routine and Control Bits
02800 C00136 00026 EMITER Routine
02900 C00140 00027
03000 C00146 00028 SUBI TEMP,1 FIX IT
03100 C00152 00029
03200 C00155 00030 Qstack Routines -- BPUSH, etc.
03300 C00159 00031
03400 C00162 00032
03500 C00165 00033 PWR2
03600 C00166 00034 GBOUT Description, Loader Block Format Description
03700 C00169 00035 Control Variables for Loader Block Output
03800 C00172 00036 Loader Output Blocks-- Entry, Program Name, Initial Stuff
03900 C00176 00037 Code, Boolean Code, Fixups, Links
04000 C00180 00038 Space Allocation Block
04100 C00184 00039 Request Blocks -- RELfile, Libraries
04200 C00186 00040 Ending Code, Symbols -- END Block
04300 C00190 00041 RELINI -- Loader Block Initialization
04400 C00191 00042 GBOUT Routine
04500 C00194 00043 CODOUT Routine -- Output Code or Data
04600 C00198 00044
04700 C00199 00045 FBOUT, etc. -- Output Fixups
04800 C00202 00046 SCOUT, etc. -- Output Symbols
04900 C00206 00047 LNKOUT -- Output Linkage Block
05000 C00208 00048 PRGOUT, FILSCN -- Output Request Blocks, Scan for Source!file Rqst
05100 C00218 00049
05200 C00221 00050 >NOTENX
05300 C00223 00051 RAD50, RAD52 -- Radix-50 Functions for Scout Routines
05400 C00227 00052
05500 C00228 ENDMK
05600 C⊗;
00100 COMMENT ⊗HISTORY
00200 AUTHOR,REASON
00300 021 102100000044 ⊗;
00400
00500
00600 COMMENT ⊗
00700 VERSION 17-1(36) 3-7-75 BY RHT BUG #UE# STRING ARRAY ISNT A STRING
00800 VERSION 17-1(35) 2-16-75 BY JFR BAIL P.35 DEFINE RESIDENCE OF RUNTIME PROCEDURE DESCRIPTORS
00900 VERSION 17-1(34) 12-7-74 BY JFR DEFINE RESIDENCE OF BAIL LOADMODULE
01000 VERSION 17-1(33) 11-3-74 BY RHT BUG TR MAKE GBOUT HONEST ABOUT WORD COUNT
01100 VERSION 17-1(32) 10-10-74 BY RHT FEAT %BR% (REMOVE HACKS)
01200 VERSION 17-1(31) 7-24-74 BY RHT BUG #SV# GET SPAC OF RECORD WAS LOSING
01300 VERSION 17-1(30) 7-22-74 BY RHT BUG #SU# CONV(ARITH) FOR PNTVAR
01400 VERSION 17-1(29) 7-7-74 BY RHT MANY EDITS FOR RECGC
01500 VERSION 17-1(28) 7-7-74
01600 VERSION 17-1(27) 7-7-74
01700 VERSION 17-1(26) 7-7-74
01800 VERSION 17-1(25) 5-30-74 BY RLS TENEX BUG #SK# DONT MESS UP DEVICE NAME FOR LOAD!MODULE
01900 VERSION 17-1(24) 5-20-74 BY RHT BUG #SA# SHOULD NOT BUMP REF CNT ON GET ADDR
02000 VERSION 17-1(23) 5-14-74 BY RHT BUG #RY# RECUUO (AC) S/B RECUUO 0,AC
02100 VERSION 17-1(22) 4-18-74
02200 VERSION 17-1(21) 4-12-74 BY RHT %BI% MAKE EMITTER KNOW ABOUT RECORD CLASSES
02300 VERSION 17-1(20) 4-12-74 BY RHT %BI% ADD SOME LOW LEVEL RECORD STUFF
02400 VERSION 17-1(19) 4-12-74
02500 VERSION 17-1(18) 4-12-74
02600 VERSION 17-1(17) 4-12-74
02700 VERSION 17-1(16) 4-12-74
02800 VERSION 17-1(15) 4-12-74
02900 VERSION 17-1(14) 4-6-74 BY RLS TENEX
03000 VERSION 17-1(13) 3-17-74 BY RLS TENEX ADDITIONS
03100 VERSION 17-1(13) 2-13-74 BY JRL BUG #RE# STRING ITEMVAR ARRAY NOT STRING ARRAY
03200 VERSION 17-1(12) 1-11-74 BY JRL CMU CHANGE COMVER (UNDER NOHACK)
03300 VERSION 17-1(11) 12-8-73 BY JRL REMOVE SPECIAL STANFORD CHARACTERS(WHERE POSSIBLE)
03400 VERSION 17-1(10) 11-24-73 BY RHT %AL% TAKE HRLOI 12, OUT OF S. SEQUENCE
03500 VERSION 17-1(9) 11-24-73 BY RFS RADIX50 TYPE BITS NOT INSTALLED IN RAD5$, RAD5%
03600 VERSION 17-1(8) 11-13-73 BY JRL FORCE PUT TO ALWAYS DO AN ACCESS
03700 VERSION 17-1(7) 11-13-73 BY JRL BUG #PA# GET ADDR OF MPPARM WAS DESTROYING AC C
03800 VERSION 17-1(6) 11-13-73 BY JRL BUG #OZ# FIX GET FOR INSISTED ITEMVARS
03900 VERSION 17-1(5) 11-4-73 BY JRL BUG #OX# LET PUT KNOW ABOUT ? ITEMVARS
04000 VERSION 17-1(4) 10-26-73 BY JRL BUG #OR# A STRING ITEM IS NOT A STRING
04100 VERSION 17-1(3) 10-23-73 BY JRL FEATURE %AG& ITEM OVERLAP STUFF
04200 VERSION 17-1(2) 8-16-73 BY JRL REMOVE REFERENCES TO LEAPSW
04300 VERSION 17-1(1) 8-2-73 BY JRL BUG #NK# TEMPS SHOULD NOT HAVE DISPLAY LEVELS
04400 VERSION 17-1(0) 7-26-73 BY RHT **** VERSION IS 17 ****
04500 VERSION 16-2(31) 7-13-73 BY RHT MODIFY SOUT FOR FNYNAM
04600 VERSION 16-2(30) 7-13-73 BY RHT BUG #MN# A DREADFUL KLUGE TO FIX ACCESS BUG
04700 VERSION 16-2(29) 7-13-73
04800 VERSION 16-2(28) 6-28-73 BY JRL BUG #KA#B IMMEDIATE INSTRUCTIONS NOT USED FOR OR,AND
04900 VERSION 16-2(27) 3-19-73 BY RHT CHANGE SOUT SO STACK SYMBOLS WORK RIGHT
05000 VERSION 16-2(26) 3-13-73 BY JRL REMOVE REFERENCES TO WOM,SLS,GAG,NODIS
05100 VERSION 16-2(25) 2-26-73 BY JRL DO A SKIPE NOEMIT IN XCALLQ
05200 VERSION 16-2(24) 2-6-73 BY JRL MAKE GET HONEST FOR QPARS
05300 VERSION 16-2(23) 1-31-73 BY HJS DISABLE CODOUT, EMITER, AND FBOUT FOR EXPR!TYPE
05400 VERSION 16-2(22) 12-13-72 BY JRL BUG #KS# ADD LOADVR SWITCH
05500 VERSION 16-2(21) 12-13-72
05600 VERSION 16-2(20) 11-30-72 BY JRL MAKE GET HONEST FOR ? ITEMVARS
05700 VERSION 16-2(19) 11-30-72 BY JRL BUG #KQ# IGNORE FIXARRS IN STORA
05800 VERSION 16-2(18) 11-21-72 BY RHT BUG #KH# DEL FORMFX STUFF FROM SIMPROC FORMALS
05900 VERSION 16-2(17) 10-17-72 BY JRL BUG #JR# STRING ITEMVARS NOT STRING
06000 VERSION 16-2(16) 8-29-72 BY KVL ADD CKECK FOR UNTYPED IN PRE
06100 VERSION 16-2(15) 7-17-72 BY RHT BUG #IO# EVAR MESSED UP BY INDEXED STRING TEMP
06200 VERSION 16-2(14) 7-8-72 BY RHT BUG ##I#L# GET ACCESS TO A VARIABLE IN PRE BEFORE INSISTING
06300 VERSION 16-2(13) 6-30-72 BY DCS BUG #IA# PROTECT PTRAC AC OVER FIX, FLOAT, STRING to INTEGER
06400 VERSION 16-2(12) 6-25-72 BY DCS BUG #HX# PARAMETERIZE LIBRARY NAMES (OTHER THINGS)
06500 VERSION 16-2(11) 6-20-72 BY DCS BUG #HU# BETTER TTY PRINTOUT
06600 VERSION 16-2(10) 6-14-72 BY JRL BUG #HS# AN ITEMVAR IS NOT ITS DATUM(MARK).
06700 VERSION 16-2(9) 5-13-72 BY DCS BUG #HF# MAKE GETAC MUCH MORE HONEST
06800 VERSION 15-2(8) 3-25-72 BY DCS BAD ARRAY ADDRESS PROBLEM
06900 VERSION 15-2(7) 3-10-72 BY DCS REPLACE RING, ULINK MACROS WITH ROUTINES
07000 VERSION 15-2(6) 2-9-72 BY DCS BUG #GQ# MAKE ! = UNDERLINE IN RADIX50
07100 VERSION 15-2(5) 2-5-72 BY DCS BUG #GJ# ADD LSTON LISTING CONTROL STUFF
07200 VERSION 15-2(4) 2-1-72 BY DCS ISSUE %ALLOC SPACE REQUESTS IN NEW WAY (SEE GOGOL FOR FORMAT)
07300 VERSION 15-2(3) 1-10-72 BY DCS BUG #FP# FIX A NEGAT BUG
07400 VERSION 15-2(2) 1-7-72 BY DCS BUG #FY# Fix Strvar←INAC-Intvar bookkeeping problem
07500 VERSION 15-2(1) 12-2-71 BY DCS INSTALL VERSION NUMBER
07600
07700 ⊗;
07800
00100 COMMENT ⊗DATA for Total (Low-level Code Production) Routines⊗
00200 LSTON (TOTAL)
00300
00400 SUBTTL WIZARD'S DEN -- Generator Called Routines.
00500 BEGIN TOTAL
00600
00700 ZERODATA (TOTAL ROUTINE VARIABLES)
00800
00900 ;ACKPNT -- next AC # GETAC should try -- used to distribute
01000 ; AC usages among the ACs -- used by GETAC only
01100 ?ACKPNT: 0
01200
01300 COMMENT ⊗
01400 FORMFX -- QSTACK descriptor for formal fixups. Until a recursive
01500 Procedure has been completely compiled, it is not known how
01600 many local strings and non-strings will be saved in the runtime
01700 stacks between the stack tops and the formal parameters. Therefore
01800 as instructions accessing parameters are issued, the address
01900 field displacements (assuming 0 locals) are saved, along with
02000 the addresses where they are issued, in the FORMFX stack.
02100 The left half of each entry is the address of the instruction--
02200 the right half is the desired relative displacement (high-order
02300 bit specifies String stack or System stack). After the procedure
02400 is compiled, these entries are QPOPed off and used, along with
02500 the ALOCALS, SLOCALS counts (see PROCED variables) to issue
02600 fixups for these instructions. This Qstack is not used
02700 for non-recursive Procedures
02800 ⊗
02900 ↑↑FORMFX: 0
03000
03100 ?POSSIB: 0 ;TEMP USED BY GETAC WHEN GETTING 2
03200
03300 ;TEMPNO -- each temp Semblk allocated is assigned a unique
03400 ; number, by incrementing TEMPNO -- a temp Semblk may
03500 ; be used several times in the same procedure. See GETTEM
03600 ; for description of the mysteries of temps.
03700 ?TEMPNO: 0
03800
03900 ENDDATA
04000
00100 COMMENT ⊗Description of Total Routines⊗
00200
00300 DSCR CONV,ACCESS,GET,PUT,STACK,MARK
00400 DES This is the generalized move code. (i.e. called by macro GENMOV).
00500 It consists of several routines which are called in a uniform
00600 fashion. This fashion stipulates that "directive" bits be passed
00700 in the right half of FF which specify modifiers on the operation
00800 of the routine called. Each routine is preceded by a standard
00900 preamble (PRE) and followed by a standard epilog (POST).
01000
01100 Some of the directive bits control PRE and POST. They are:
01200
01300 PAR
01400 PRE:
01500 1. If the GETD bit is on, we do a GETAD first (i.e. use PNT
01600 as the pointer to a symbol table entry, and fill TBITS
01700 and SBITS. This is useful since many of the GENMOV routines
01800 require that TBITS and SBITS be set up.
01900 2. If the PROTECT bit is set, then register D is assumed to have
02000 an accumulator number in it. That accumulator table entry
02100 is "protected". I.e. calls on GETAC and STORA will not affect
02200 the status of anything marked in that accumulator.
02300 3. If the EXCHIN bit is set, we do an EXCHOP.
02400 4. If the INSIST bit is on, type conversions are performed.
02500 These conversions convert from the type specified in the
02600 TBITS word to the type specified in register B (bits
02700 passed to the INSISTer).
02800 5. If the ARITH bit is on, we make sure that the type is
02900 an arithmetic type, performing conversions if necessary.
03000
03100
03200 POST:
03300 1. Put the current contents of the ac's TBITS and SBITS
03400 down in the symbol table entry pointed to by PNT
03500 2. If the REM bit is set, do a REMOP on the thing in PNT
03600 3. If the BITS2 bit is set, we execute MOVE SBITS2,$SBITS(PNT2)
03700 This is useful when an operation on one argument of a binary
03800 op. may change the semantics of another.
03900 4. If the UNPROTECT bit is set, then register D is assumed to
04000 contain an ac number. The ac table entry is unprotected.
04100 5. If the EXCHOUT bit is set, we do an EXCHOP.
04200
04300 NOW FOR A DESCRIPTION OF THE ROUTINES WHICH ACTUALLY USE PRE AND POST:
04400
04500 CONV:
04600 This is really a no-op. It is here for the purposes of calling
04700 the type-conversion routines in PRE, and for the purpose of
04800 making sure that an argument is positive if in an accumulator
04900 (e.g. if we had CVF(-(A+B)), then the result would be in an
05000 accumulator in negated fashion. We now want to push it onto the
05100 stack for the call on CVF. We want to make sure it is REAL and
05200 positive. We use the POSIT bit: GENMOV (CONV,INSIST!POSIT,REAL)
05300
05400
05500 PUT
05600 This issues a store of accumulator mentioned in register D
05700 into the thing described in TBITS, SBITS, PNT. The accumulator
05800 table is updated to reflect this store (i.e. the thing talked about
05900 by PNT is marked as "inac").
06000
06100 If the PNT entry is a string, then D is assumed to be an ac.
06200 into which a HRROI was done, or the SP stack. At any rate, two
06300 POP's are emitted.
06400
06500 ACCESS:
06600 This routine makes sure that we can have access to the entry
06700 mentioned in PNT. That is, if the thing is indexed (result of
06800 an array calculation) and if it requires that some index accumulator
06900 be loaded with a good number, then the load will happen, so
07000 that an effective address can be generated which points at
07100 the thing talked about by PNT.
07200
07300 GET:
07400 This is the generalized "get this entity in an ac" routine.
07500 It makes many checks (i.e. is it already in an ac?) and
07600 finally returns in register D the number of the ac which
07700 has been loaded, and returns in SBITS the updated semantics
07800 information that now reflects the loaded state.
07900 (By the way, to "get" a string means to do HRROI ac,second word
08000 of string.. This is so that POP's can be done later). There
08100 are many modifier bits to this routine:
08200
08300 DBL -- make sure that the ac following the one loaded
08400 is free (for a double ac operation such as IDIV)
08500 INDX -- make sure entity is loaded in an AC which can be
08600 used for indexing (i.e. not 0 or 1. The reason
08700 for including 1 in this is a bit vague -- since
08800 runtime routines often return results in 1, we
08900 try to avoid its use for things thay may have
09000 to be stored as temps).
09100 SPAC -- load this into a special accumulator. That accumulator
09200 number is passed in D.
09300 ADDR -- load the address of this entity, not the value.
09400 POSIT -- make sure the entity is in the ac in positive form.
09500 NEGAT -- make sure in negative form.
09600 NONSTD -- if indxed temp, do not remop it as someone wants
09700 to use it again. (see SWPR for instance). The
09800 problem is not so much remopping, but that GET
09900 likes to make the semantic entries as "inac" on
10000 exit. This fouls up any index calculations that
10100 may have been stored in the PNT entity.
10200 MRK -- when done with the GET, call MARK (see below).
10300
10400 STACK:
10500 The entity mentioned in PNT is stacked on an appropriate
10600 stack. Strings (except arrays) are stacked on the SP
10700 stack, all others on the P stack. ADEPTH or SDEPTH is
10800 updated.
10900
11000 MARK:
11100 This uses the bits in TBITS and SBITS, and the ac number
11200 in D as prototypes for making up a temp descriptor, and
11300 marking the ac full with that temp. Return is a valid
11400 temp descriptor in PNT. If STRING is on in TBITS,
11500 a stacked-string descriptor will be generated
11600 (and of course, no accumulator will be marked).
11700 WARNING ***** MARK masks off some bits in SBITS and
11800 TBITS. PTRAC,CORTMP,INDXED,FIXARR are turned off in SBITS
11900 and the only bits honored by TBITS are:
12000 LPARRAY,SET,ITEM,ITMVAR,INTEGR,FLOTNG,STRING
12100
12200 SID
12300 ACCUMULATORS:
12400 FF -- RIGHT HALF SAVED.
12500 A --THIS MAY BE CHANGED
12600 B --SAVED, I BELIEVE.
12700 C --SAVED, I BELIEVE.
12800 D --OCCASIONALLY FILLED UP (E.G. GET,ACCESS)
12900 TBITS -- THESE ARE THE SEMANTIC BITS -- THEY MAY BE CHANGED.
13000 SBITS -- "
13100 PNT -- " (IN CASE OF MARK OR CONVERSIONS)
13200 LPSA CLOBBERED
13300 USER CLOBBERED
13400 TEMP CLOBBERED
13500 SP --SAVED
13600 SBITS2 --SAVED (modulo what is done in PRE).
13700 TBITS2 --SAVED
13800 PNT2 --SAVED
13900
14000 SEE GENMOV MACRO
14100 ⊗;
14200
00100 COMMENT ⊗CONV, PRE, POST -- Type-Conversion routines⊗
00200
00300 MASK←← 0+LPARRAY+SET+LSTBIT+ITEM+ITMVAR+INTEGR+FLOTNG+STRING!DBLPRC
00400 ;GENMOVE KNOWS ABOUT THESE TYPES
00500 REC <
00600 MASK ←← MASK+PNTVAR
00700 >;REC
00800
00900 ;THIS IS THE PREAMBLE FOR ALL OF THE ROUTINES WHICH
01000 ;USE DIRECTIVE BITS TO SPECIFY COERCIONS, EXCHOPS, ETC.
01100
01200 PREMASK ←← GETD!EXCHIN!INSIST!ARITH!PROTECT
01300
01400
01500 ↑↑CONV: TRNE FF,PREMASK
01600 PUSHJ P,PRE ;DO EVERYTHING HERE.
01700 TLNE SBITS,NEGAT ;IF NOT NEGAT OR
01800 TRNN FF,POSIT ;NOT NEED THINGS POSITIVE?
01900 JRST POST ;ALL DONE.
02000 JRST GETOPE ;DO THE GET.
02100
02200
02300
02400 PRE: TRNE FF,GETD ;DO A GETAD?
02500 PUSHJ P,GETAD ;YES
02600 TRNE FF,EXCHIN ;EXCHOP ON WAY IN?
02700 JRST [EXCHOP ;YES
02800 JRST .+1]
02900 TRNE FF,PROTECT
03000 JRST [HRROS ACKTAB(D) ;PROTECT THIS AC
03100 TDNN TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
03200 TRNN TBITS,DBLPRC
03300 JRST .+1 ;THESE ARE NOT DOUBLE
03400 MOVEI TEMP,(D)
03500 CAIN TEMP,RF-1
03600 ERR <DRYROT PRE>,1
03700 CAIE TEMP,RF-1
03800 HRROS ACKTAB+1(D) ;2ND AC OF LONG
03900 JRST .+1]
04000 TRNN FF,INSIST!ARITH ;ANY COERCIONS TO DO?
04100 POPJ P, ;NO -- ALL DONE.
04200 PUSHJ P,QTYPCK
04300 ;CHECK FOR UNTYPED AND TYPE IF NEC. (SEE ERRORS)
04400 ;#IL# 7-8-72 RHT ! GET ACCESS BEFORE YOU CONVERT
04500 PUSHJ P,ACCOP ;GET ACCESS -- YOU MAY NEED IT
04600 TRNE FF,ARITH ;WANT TO BE SURE OF ARITH ARG?
04700 JRST AGET ;YES
04800 LEPPRE: TRNN TBITS,ITEM!ITMVAR ;IF EITHER HAS ITEM BITS ON.
04900 TRNE B,ITEM!ITMVAR ;ALL THESE ARE GOOD GUYS.
05000 JRST [ ;.... ;KEEP GOING.
05100 TRNE B,ITEM!ITMVAR
05200 TRNN TBITS,ITEM!ITMVAR
05300 ERR <ITEM TYPE MISMATCH >,1
05400 POPJ P,] ;THIS IS ALL THE CHECKING!
05500 TRNE B,SET ;A SET OR LIST DESIRED?
05600 JRST [TRNN TBITS,SET ;IF NOT LIST OR A SET CAN'T BE DONE
05700 ERR <TYPE CAN'T BE CONVERTED TO SET OR LIST>,1
05800 TRNE B,LSTBIT ;IF WANTED LIST CAN RETURN
05900 JRST MAKLST ;MAY HAVE TO COPY LIST.
06000 TRNN TBITS,LSTBIT ;IF WANTED SET AND HAVE SET CAN RETURN
06100 POPJ P,
06200 JRST MAKEST] ;WILL HAVE TO CALL CVSET
06300 ;;#YQ# JFR 2-2-77 DO RCLASS CHECKING FOR ASSIGNMENT NOW
06400 REC <
06500 TRNE TBITS,PNTVAR ;IF RECORDS & INSISTING
06600 TRNE TBITS,SHORT!ITEM!ITMVAR ;THEN BETTER BE SURE CLASSES MATCH
06700 JRST LEPP.1 ;NOT THAT CASE, ANYHOW
06800 PUSH P,PNT2 ;NOTE DON'T CHECK ITEMS
06900 PUSH P,LPSA
07000 HLRZ PNT2,$ACNO(PNT) ; THE CLASSID
07100 SKIPN LPSA,RCLASS ;
07200 ;; MWK This message used to be "RCLASS=0 ON INSISTING GET"
07300 ERR <ATTEMPT TO COERCE RECORD POINTER>,1
07400 PUSHJ P,SUBFOK ;CHECK CLASS ID
07500 ERR <CLASS DISAGREEMENT FOR RECORD COERCION>,1
07600 TRNN FF,MRK ;ASKED FOR A MARK
07700 SETZM RCLASS ;NO, ALWAYS CLEAR THIS OUT
07800 POP P,LPSA
07900 POP P,PNT2
08000 LEPP.1:
08100 >;REC
08200 ;;#YQ# ↑
08300 MOVEI TEMP,(TBITS) ;ALWAYS INSIST ON CORRECT PRECISION
08400 XORI TEMP,(B)
08500 TRNN TBITS,DBLPRC
08600 TRNE B,DBLPRC
08700 TRNN TEMP,DBLPRC ;ONE IS DBL. ARE BOTH?
08800 SKIPA TEMP,TBITS ;BOTH ARE SAME PRECISON
08900 JRST AGOTH ;DIFFERENT PRECISION. MUST CONVERT
09000 MOVE USER,B ;COPY OFF.
09100 AND TEMP,[XWD SBSCRP,MASK≠(ITEM!ITMVAR)]
09200 ORCB USER,[XWD SBSCRP,MASK≠(ITEM!ITMVAR)]
09300 TDNN TEMP,USER ;ARE ALL BITS IN B ON IN TBITS?
09400 POPJ P, ;THEY MATCH !!
09500 AGOTH:
09600 PUSH P,FF
09700 TRZ FF,-1≠NONSTD ;IN CASE ANY OTHER ROUTINES CALLED.
09800 PUSH P,D
09900 TRNE B,INTEGR+FLOTNG
10000 JRST RESAR ;INSISTS ON ARITHMETIC TYPE
10100 TRNE B,STRING
10200 JRST RESSTR ;INSISTS ON STRING
10300 ERR <IMPOSSIBLE TYPE COERCION>,1
10400 JRST GEMGO ;GO ON ANYWAY
10500
00100
00200
00300 RESSTR: TRNN TBITS,INTEGR ;INSIST ON INTEGER ARGUMENT.
00400 ERR <STRINGS OF NON-INTEGERS?>
00500 TLNN TBITS,CNST ;CONSTANT?
00600 JRST STR1 ;NO
00700 EXCH SP,STPSAV ;GET A GOOD STACK POINTER.
00800 MOVSS POVTAB+6 ;ENABLE FOR STRING PDLOV
00900 PUSH P,$VAL(PNT)
01000 PUSHJ P,PUTCH ;MAKE A STRING (SLOWLY)
01100 POP SP,PNAME+1
01200 POP SP,PNAME
01300 EXCH SP,STPSAV ;AND RESTORE EVERYONE.
01400 MOVSS POVTAB+6 ;RE-ENABLE FOR PARSE PDLOV
01500 PUSHJ P,STRINS ;INSERT A STRING CONSTANT
01600 ;THIS DOES A GETAD.
01700 JRST GEMGO ;ALL DONE
01800
01900 STR1: ;PREPARE TO STACK THE INTEGER
02000 PUSHJ P,STACK1 ;DO THE STACK.
02100 SOS ADEPTH ;SINCE THE RUNTIM ROUTINES ADJUST.
02200 MOVEI TEMP,2
02300 ADDM TEMP,SDEPTH ;INCREASE DUE TO CALL.
02400 XCALL <PUTCH> ;FUNCTION CALL
02500 MOVEI SBITS,0 ;START WITH CLEAN DYNAMIC SLATE
02600 JRST TGO ;GO MAKE A TEMP.
02700
02800
02900
03000 ;;#SU# ! ADD PNTVAR TO THIS LIST
03100 AGET: TRNE TBITS,INTEGR+FLOTNG+PNTVAR ;IS IT ALREADY ARITHMETIC TYPE?
03200 POPJ P, ;YES
03300 PUSH P,FF
03400 TRZ FF,-1≠NONSTD ; SAVE ALL THIS FOR OTHER
03500 PUSH P,D ; EMBEDDED OPERATIONS
03600 MOVEI B,INTEGR ;THIS FOR THE BENEFIT OF ARSTR.
03700 RESAR: TRNE TBITS,STRING ;HERE TO GET ARITHMETIC RESULTS
03800 JRST ARSTR ;CONVERT FROM STRING
03900 TRNE TBITS,INTEGR+FLOTNG
04000 JRST FIXFL
04100 ERR <IMPOSSIBLE TYPE COERCION>,1
04200 JRST TGO ;MAKE A TEMP FOR IT ANYWAY...
04300
04400 ARSTR: TLNE TBITS,CNST ;CONSTANT?
04500 JRST STRCNS
04600 ;;#IA# 6-30-72 DCS (3-6) PROTECT PTRAC AC OVER GETAC
04700 HRLI PNT,-1 ;FLAG, ASSUME PROTECTION
04800 HRRZ TEMP,$ACNO(PNT) ;PTRAC AC #, IF ANY
04900 TLNN SBITS,PTRAC ;NEED PROTECTION?
05000 TLZA PNT,-1 ;NO, UNMARK
05100 HRROS ACKTAB(TEMP) ;YES, PROTECT
05200 ;;#IA# (3-6)
05300 PUSH P,B ;SAVE TYPE WORD
05400 PUSHJ P,GETAN0 ;NON-0 AC NUMBER
05500 JUMPGE PNT,.+3 ;NEED TO UNPROTECT?
05600 ;;#IA# 6-30-72 (4-6)
05700 HRRZ TEMP,$ACNO(PNT) ;YES, DO
05800 HRRZS ACKTAB(TEMP) ; IT
05900 ;;#IA# (4-6)
06000 MOVE A,[HRRZ LNWORD] ;CALCULATE LENGTH TO THIS AC
06100 PUSHJ P,STROP ;VIA STROP
06200 HRL B,PCNT ;SAVE PC FOR FIXUP
06300 HRLI C,0
06400 EMIT (<JUMPE USADDR!NORLC>) ;0 IF STRING EMPTY
06500 TLNE SBITS,STTEMP ;NO NEED TO COPY BP IF TEMP STRING
06600 JRST [MOVE A,[ILDB BPWORD]
06700 PUSHJ P,STROP ;SO DO ILDB DIRECTLY
06800 JRST NOCOP] ;AND GET OUT
06900 MOVE A,[MOVE BPWORD] ;GET COPY OF BP
07000 PUSHJ P,STROP ;IN SAME AC
07100 HRL C,D
07200 EMIT (<ILDB USADDR!NORLC>) ;ILDB AC,AC
07300 NOCOP: HRR B,PCNT ;FIXUP WORD
07400 PUSHJ P,FBOUT
07500 MOVEI A,UNDO!REM
07600 PUSHJ P,STROP ;NOW ISSUE SUB IF NECESSARY
07700 PUSHJ P,MARKINT ;MARK INT. RETS RIGHT THING IN PNT
07800 POP P,B
07900 TRNE B,INTEGR ;CONVERT ONLY TO INTEGER?
08000 JRST GEMGO ;YES, OK.
08100 JRST FIXFL ;GO ON FARTHER
08200
00100
00200 STRCNS: HRRZ TEMP,$PNAME(PNT) ;THIS IS THE SAME CODE AS
00300 JUMPE TEMP,.+3 ; SAIL GENERATES TO DO
00400 MOVE TEMP,$PNAME+1(PNT) ; STRING to INTEGER AT
00500 ILDB TEMP,TEMP ; RUNTIME
00600 TRNN B,INTEGR ;DOES HE WANT AN INTEGER CONST
00700 FLOAT TEMP,TEMP ;NO -- ASSUME FLOATING
00800 JRST CONGO ;GO INSERT A CONSTANT.
00900
01000 FIXFL:
01100 TRNN TBITS,DBLPRC ;INPUT LONG?
01200 JRST FXFLS1 ;NO
01300 EXCH B,-1(P) ;ULTIMATE DESIRES
01400 MOVEI TEMP,FXFLL1 ;RETURN ADDR
01500 EXCH TEMP,(P)
01600 PUSH P,B ;ORIGINAL DIRECTIVE BITS
01700 PUSH P,TEMP ;OLD AC D
01800 MOVSI A,(<SNGL>) ;WHAT TO EMIT, IF WE HAVE TO
01900 MOVEI B,FLOTNG ;WHAT WE WANT, TEMPORARILY
02000 TLNN TBITS,CNST
02100 JRST UUOGO ;CONVERT TO SINGLE REAL, COME BACK TO FXFLL1
02200 SNGL TEMP,$VAL(PNT) ;DO THE OP ON A CONSTANT
02300 JRST CONGO ;RECORD RESULTS
02400 FXFLL1: POP P,B ;ULTIMATE DESIRES ARE BACK
02500 JRST LEPPRE ;PROBLEM HAS BEEN REDUCED ONE NOTCH
02600
02700 FXFLS1:
02800 TRNN B,DBLPRC ;OUTPUT LONG?
02900 JRST FXFLS2 ;NO
03000 TLNE TBITS,CNST ;CONSTANT?
03100 JRST [MOVE TEMP,$VAL(PNT) ;YES, GET VALUE
03200 ;;#YB# ! JFR 1-3-77 COMPLETE TYPO
03300 TRNN TBITS,FLOTNG ;ALREADY REAL?
03400 FLOAT TEMP,TEMP ;NO
03500 SETZM DBLVAL ;SECOND WORD IS ZERO
03600 JRST CONGO] ;THE EASY WAY
03700 GENMOV (GET,DBL!INSIST,FLOTNG) ;LOAD IT FLOTNG
03800 PUSHJ P,CLEARA ;THEN FORGET YOU DID IT
03900 MOVE FF,-1(P) ;ORIGINAL DIRECTIVE BITS
04000 IORI B,DBLPRC
04100 ADDI D,1 ;AND ZERO THE NEXT
04200 FXFLL2: EMIT (<SETZ NOADDR>)
04300 SOJA D,TGO1 ;MARK RESULT
04400 FXFLS2:
04500 ;;%DN% JFR 7-1-76
04600 MOVE TEMP,ASWITCH ;OPTION BITS
04700 MOVSI A,(<FIX>) ;ASSUME STANDARD
04800 TRNE TEMP,AFIXR
04900 MOVSI A,(<FIXR>)
05000 TRNE TEMP,AKIFIX
05100 MOVSI A,(<KIFIX>)
05200 MOVE USER,A ;COPY THE DECISION
05300 OR USER,[TEMP,TEMP] ;INSERT AC AND ADDR FIELDS
05400 ;;%DN% ↑
05500 MOVE TEMP,TBITS ;GET OR OF `SHORT' BITS
05600 OR TEMP,B
05700 TRNE B,INTEGR ;RESULT FIXED?
05800 JRST FIX ;YES
05900 ;;%DN%
06000 MOVE TEMP,ASWITCH
06100 MOVSI A,(<FLOAT>)
06200 TRNE TEMP,AFLTR
06300 MOVSI A,(<FLTR>)
06400 MOVE USER,A
06500 OR USER,[TEMP,TEMP]
06600 MOVE TEMP,TBITS ;GET OR OF `SHORT' BITS
06700 OR TEMP,B
06800 ;;%DN% ↑
06900 TLNE TBITS,CNST ;CONSTANT?
07000 JRST FLC
07100 TRNN TEMP,SHORT ;SHORT INTEGER BEGIN FLOATED?
07200 JRST UUOGO ;NO, USE UUO
07300 PUSH P,[FSC USADDR!NORLC] ;INSTR TO FLOAT
07400 HRLI C,233 ;ARGUMENT OF FLOAT INSTR
07500 SHRTCV: MOVE TEMP,-2(P) ;FF BITS COMING INTO TOTAL
07600 TRNE TEMP,SPAC ;WAS SPECIFIC AC REQUIRED
07700 TRO FF,SPAC ;YES, RETAIN IT
07800 PUSHJ P,GET ;GET THE THING
07900 POP P,A ;INSTR
08000 JRST JSTEST ;ALREADY KNOW WHAT AC
08100
08200
08300 FIX: TLNE TBITS,CNST ;CONSTANT?
08400 JRST FLC
08500 NOEXPO<
08600 TRNN TEMP,SHORT ;CONVERT TO SHORT INTEGER?
08700 JRST UUOGO ;NO
08800 PUSH P,[PDPFIX USADDR!NORLC] ;YES, USE PDP-10 INSTR
08900 HRLI C,233000 ;MAGIC ADDR FIELD FOR PDPFIX INSTR
09000 JRST SHRTCV ;DO SHORT CONVERSION
09100 >;NOEXPO
09200
09300 UUOGO: MOVE TEMP,-1(P) ;DIRECTIVE BITS WORD FROM STACK.
09400 TRNE TEMP,SPAC ;IS HE GOING TO WANT A SPECIAL ONE?
09500 JRST JSTEST ;YES
09600 HRR D,$ACNO(PNT)
09700 ;;#IA# 6-30-72 DCS (5-6) PROTECT PTRAC AC OVER GETAC
09800 HRLI PNT,-1 ;FLAG, ETC., SEE PART (3-6)
09900 TLNN SBITS,PTRAC
10000 TLZA PNT,-1
10100 HRROS ACKTAB(D)
10200 ;;#IA# (5-6)
10300 TLNN SBITS,INAC ;IF NOT IN AN AC, THEN GET ONE.
10400 PUSHJ P,GETAC
10500 ;;#IA# 6-30-72 (6-6)
10600 JUMPGE PNT,.+3
10700 HRRZ TEMP,$ACNO(PNT)
10800 HRRZS ACKTAB(TEMP)
10900 ;;#IA# (6-6)
11000 GOTACB:
11100 JSTEST:
11200 DPB D,[POINT 4,A,12] ; STORE AC NUMBER IN INSTRUCTION.
11300 PUSHJ P,EMITER
11400 TGO1: HRRZ TEMP,FF ;ORIGINAL FF
11500 TRNE TEMP,NONSTD ;IF NON-STANDARD (SEE SWAP OPER),
11600 JRST [POP P,(P) ; DON'T REMOP OR MARK
11700 JRST GEMGO1] ;BUT RETAIN THE AC USED
11800 PUSHJ P,REMOP ;REMOP THE OPERAND.
11900 TGO: HRRZ TBITS,B ;MAKE TBITS CONFORM TO THE DESIRED TYPE
12000 ANDI TBITS,MASK ;MAKE RESULT LOOK LIKE THE REQUESTS
12100 TLZ SBITS,-1≠NEGAT ;CLEAR AWAY THE CHAFF
12200 PUSHJ P,MARK1 ;GO DO A MARK.
12300 JRST GEMGO
12400
12500 FLC: MOVE TEMP,$VAL(PNT) ;HERE FOR A CONSTANT.
12600 XCT USER ;DO THE CONVERSION
12700 CONGO: MOVEM TEMP,SCNVAL ;SET UP FOR SYMBOL TABLE INSERTION
12800 HRRZ TBITS,B ;COME HERE TO INSERT A CONSTANT.
12900 ANDI TBITS,MASK
13000 TLO TBITS,CNST
13100 MOVEM TBITS,BITS ;FOR CONINS
13200 PUSHJ P,REMOP ;ALWAYS REMOVE THE OLD GUY
13300 PUSHJ P,CONINS
13400 GEMGO: POP P,D
13500 GEMGO1: POP P,FF ;AT LAST DO THE POP AND
13600 POPJ P, ;ALL DONE -- FULL SPEED AHEAD.
13700
00100
00200 ; NOW FOR THE POSTAMBLE (WE WILL AMBLE THROUGH THE COMPILATION).
00300
00400
00500 ↑↑POST: MOVEM SBITS,$SBITS(PNT) ;PUT DOWN SEMANTICS WORDS.
00600 MOVEM TBITS,$TBITS(PNT)
00700 TRNN FF,EXCHOUT!BITS2!REM!UNPROTECT ;THESE ARE THINGS TO DO.
00800 POPJ P, ;ALL DONE.
00900 TRNE FF,REM ;REMOP THE THING?
01000 JRST [PUSHJ P,REMOP ;YES
01100 MOVE SBITS,$SBITS(PNT)
01200 JRST .+1]
01300 TRNE FF,BITS2 ;UPDATE SBITS2?
01400 MOVE SBITS2,$SBITS2(PNT2) ;DONE.
01500
01600 TRNE FF,UNPROTECT
01700 JRST [HRRZS ACKTAB(D)
01800 TDNN TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
01900 TRNN TBITS,DBLPRC
02000 JRST .+1
02100 MOVEI TEMP,(D)
02200 CAIN TEMP,RF-1
02300 ERR <DRYROT POST>,1
02400 CAIE TEMP,RF-1
02500 HRRZS ACKTAB+1(D)
02600 JRST .+1]
02700
02800 TRNN FF,EXCHOUT ;EXCHANGE ON WAY OUT?
02900 POPJ P, ;NO --DONE.
03000 EXCHOP
03100 POPJ P,
03200
00100 COMMENT ⊗PUT⊗
00200
00300 ↑↑PUT: TRNE FF,PREMASK ;ANY PREAMBLE TO BE DONE
00400 PUSHJ P,PRE ;YES -- DO IT.
00500 PUSH P,FF ;HERE TO STORE AN ACCUMULATOR INTO
00600 ; HAVE PUT ALWAYS DO AN ACCESS
00700 ; TLNE SBITS,INDXED ;A DESCRIPTOR
00800 PUSHJ P,ACCOP ;GET ACCESS TO THE TARGET.
00900 TRNE TBITS,STRING ;IF NOT A STRING
01000 TDNE TBITS,[XWD SBSCRP,ITEM!ITMVAR] ;OR NOT REALLY A STRING,THEN
01100 JRST APUT ;USE A MOVEM OR THE LIKE.
01200
01300 MOVE A,[POP BPWORD!LNWORD!SBOP!BPFIRST]
01400 PUSHJ P,STROP ;USE THE STRING OPERATION TO PUT OUT POPS.
01500 CAIE D,RSP ;IF IT WAS NOT THE STACK, THEN
01600 PUSHJ P,CLEARA ;CLEAR OUT THIS ACCUMULATOR ENTRY.
01700 ;IT WAS CHANGED WHEN THE POPS WERE DONE ANYWAY.
01800 JRST PUTFIN ;ALL DONE. MY THAT WAS SIMPLE.
01900
02000 APUT: PUSHJ P,CLEARA ;CLEAR OUT THE DESTINATION ACCUMULATOR.
02100 TLNE SBITS,INAC ;IF THE DESTINATION OF THE STORE IS ALREADY
02200 PUSHJ P,CLEAR ;IN AN AC, THEN CLEAR IT OUT.
02300 REC <
02400 NORGC <
02500 TRNE TBITS,PNTVAR ;A RECORD ?
02600 TRNE TBITS,777777-(PNTVAR!GLOBL)
02700 JRST APUT2 ;NOPE, JUST DO THE PUT
02800 PUSH P,C ;IT MAY BE USED, NOT SURE
02900 MOVNI C,1 ;DEREFERENCE THE THING IN PNT
03000 PUSHJ P,RFCADJ ;LIKE SO
03100 POP P,C
03200
03300 APUT2:
03400 >;NORGC
03500 >;REC
03600 HRLZI A,(<MOVEM>) ;THE ORDINARY STORE INSTRUCTION.
03700 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
03800 JRST .+3 ;SUFFICES IN ANY CASE FOR AN ARRAY, ITEM, ITEMVAR
03900 TRNE TBITS,DBLPRC
04000 HRLZI A,(<DMOVEM>)
04100 TLNN SBITS,NEGAT ; BUT IF NEGATED, USE THE OTHER
04200 JRST .+4 ;NOT NEGATED
04300 TRNN TBITS,DBLPRC ;DOUBLE?
04400 TLCA A,(<MOVNM>≠<MOVEM>) ;NO. MAKE INTO MOVNM
04500 TLC A,(<DMOVNM>≠<DMOVEM>) ;YES. MAKE INTO DMOVNM
04600 ;; #OX# TREAT ? ITEMVARS SPECIALLY
04700 TLNE TBITS,MPBIND
04800 JRST [HRR C,D ;SAVE AC NUMBER
04900 GENMOVE (GET,ADDR!INDX)
05000 MOVSS D
05100 HRR D,C ;XWD INDX,,AC
05200 MOVE A,[MOVEM USX+NORLC+NOADDR]
05300 JRST .+1
05400 ] ;GO AWAY
05500 ;; #OX#
05600 PUSHJ P,EMITER ;AND PUT OUT THE INSTRUCTION.
05700
05800 TLNE SBITS,INDXED ;WE DO NOT WANT TO MARK *********
05900 JRST PUTFN1 ;GO AWAY.
06000
06100 HRRM D,$ACNO(PNT) ;AND THE AC IT IS IN
06200 HRRM PNT,ACKTAB(D) ;IN TWO PLACES.
06300 ;THIS UNPROTECTS THIS ACCUMULATOR.
06400 TLNN SBITS,PTRAC
06500 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
06600 JRST .+3
06700 TRNE TBITS,DBLPRC
06800 JRST [SKIPE ACKTAB+1(D)
06900 ERR <DRYROT PUT DOUBLE>,1
07000 MOVEI TEMP,(D)
07100 CAIE TEMP,RF-1 ;DO NOT CLOBBER RF!
07200 HRRM PNT,ACKTAB+1(D)
07300 JRST .+1]
07400 TLOA SBITS,INAC ;AND NOW MARK THE DESCRIPTOR BITS
07500
07600 PUTFN1: TLZ SBITS,NEGAT ;SUBSCRIPTED, NEGAT GETS IN WAY (BELIEVE!)
07700 PUTFIN: POP P,FF ;ALL DONE
07800 JRST POST ;AND FINISH OUT.
07900
00100 COMMENT ⊗ACCESS,GETSDR,GETDR,DISBLK,ZOTDIS--last four only for dis
00200
00300 Call ACCOP when you need to reference a thing and don't know whether you
00400 can get at it in a single instruction (i.e. an indexed thing).
00500 GENMOV(ACCESS) will cause ACCOP to be called for you.
00600 People like GET and STACKOP do it automatically.
00700 ⊗
00800
00900 ↑↑ACCESS: TRNE FF,PREMASK
01000 PUSHJ P,PRE
01100 PUSHJ P,ACCOP
01200 JRST POST
01300
01400 ACCOP: TDNN SBITS,[XWD INDXED,DLFLDM]; ONLY CARE IF INDEXED OR NEED A DISPLY
01500 POPJ P,
01600 TLNE SBITS,INAC!PTRAC ;IF IN AN AC WE CAN ACCESS IT
01700 POPJ P,
01800 TRNN SBITS,DLFLDM ;IF DISPLAY LEV=0 ONLY CARE ABOUT INDEXED
01900 JRST INXSTF ;NO WORRY ABOUT THE DIAPLAY
02000 LDB TEMP,[LEVPOINT<SBITS>] ;PICK UP DISPLY LEV
02100 TRNE TBITS,STRING ;IS ITT A STRING
02200 JRST [
02300 ;; #JR# BY JRL 10-17-72 ITEMVARS,ARRAYS DON'T USE STRING STACK
02400 TDNN TBITS,[REFRNC!SBSCRP,,ITEM!ITMVAR];THESE THINGS DON'T USE
02500 TLNE SBITS,INDXED ;INDEXED? ;STRING STACK
02600 JRST .+1
02700 ;; #JR#
02800 JRST GETSDR ;GET STRING DR
02900 ]
03000 PUSHJ P,GETDR ;GET A DISPLAY REG LOADED
03100 ;;%DU% JFR 1-4-77
03200 TRNE FF,ACESS2 ;DIRECT TO 2ND WD
03300 TRNN TBITS,DBLPRC ; OF LONG?
03400 JRST ACCOP1 ;NO
03500 TLNE TBITS,REFRNC
03600 JRST ACMOP ;APPARENTLY ONLY REF. FORMALS NEED THIS
03700 ACCOP1:
03800 ;;%DU% ↑
03900 TRNN SBITS,INDXED ;INDEXED TOO?
04000 POPJ P, ;NO
04100 INXSTF:
04200 ;;#JR#
04300 TRNN TBITS,ITEM!ITMVAR
04400 TRNN TBITS,STRING ;ALWAYS NEED STRING GUYS
04500 JRST .+2
04600 ;;#JR#
04700 JRST ACMOP
04800 HRRZ TEMP,$VAL(PNT) ; ONLY NEED IT IF NON-ZERO
04900 JUMPE TEMP,CPOPJ ; DISPLACEMENT
05000
05100 ACMOP: TLNE SBITS,PTRAC ;IS IT ALREADY ACCEPTABLE (IN AC)?
05200 POPJ P, ; YES, WHY HAVE WE WORRIED?
05300
05400 PUSH P,D ;HAVE TO SAVE CURRENT AC
05500 PUSH P,A
05600 PUSH P,FF
05700 MOVE TEMP,FF
05800 HRRI FF,INDX ;SO THAT NOTHING NONSTD WILL HAPPEN.
05900 MOVE A,[XWD 40!2!1,ADDR] ;SET NECESSARY BITS
06000 ;(SPECIAL BIT, MOVE, GET AC, USE INDXBLE AC, GET ADDR)
06100 ;;%DU%
06200 TRNE TEMP,ACESS2
06300 TRNN TBITS,DBLPRC
06400 CAIA
06500 TLO A,400 ;DO NOT CHANGE SBITS(PNT)
06600 ;;%DU% ↑
06700 PUSHJ P,GETWD
06800 POP P,FF
06900 ;;%DU% JFR 1-4-77
07000 TRNE FF,ACESS2
07100 TRNN TBITS,DBLPRC
07200 JRST ACMOP1 ;NOT SPECIAL 2ND WD ACCESS
07300 PUSH P,TBITS
07400 SETZB TBITS,SBITS
07500 PUSHJ P,GETTEM ;WE HAVE GOTTEN THE ADDR INTO A TEMP
07600 MOVEM D,$ACNO(LPSA) ;REMEMBER INTO WHICH AC
07700 HRRM LPSA,ACKTAB(D)
07800 POP P,$TBITS(LPSA)
07900 HRRZS TBITS,$TBITS(LPSA) ;SANITIZE THE TYPE
08000 MOVSI SBITS,ARTEMP!PTRAC!INDXED
08100 MOVEM SBITS,$SBITS(LPSA)
08200 SETZM $VAL(LPSA) ;DISPLACEMENT IS ZERO
08300 MOVEI PNT,(LPSA) ;OUR NEW OPERAND
08400 ACMOP1:
08500 ;;%DU% ↑
08600 POP P,A
08700 POP P,D
08800 POPJ P,
08900
09000
09100 COMMENT⊗
09200 DSCR GETSDR,GETDR
09300 DES ROUTINES TO LOAD UP STRING (ARITHMETIC) DISPLAYS
09400 LOADS UP LPSA WITH THE AC NO TO USE & FIXES UP ACTAB,DISTAB,&DISLST
09500 PARM TEMP=LEVEL DESIRED
09600 SID MANGLE TEMP,LPSA
09700 STORES LEVEL IN LSDRLV, STORES DR # IN LSDRNM (LH FOR SDR & RH FOR DR)
09800 ⊗
09900
10000 ;;#MN# 7-13-73 FIX ACCESS PROBLEM
10100 ZERODATA(EMITTER DATA)
10200 LSDRLV: 0 ;SEE ABOVE
10300 LSDRNM: 0
10400 ENDDATA
10500
10600 ↑↑GETSDR:
10700 HRLM TEMP,LSDRLV ;REMEMBER LEVEL OF STRING REQUEST
10800 HLRZ LPSA,DISTAB(TEMP) ;DO WE HAVE IT ALREADY
10900 HRLM LPSA,LSDRNM ;IF SO REMEMBER
11000 ;;#MN#
11100 JUMPN LPSA,CPOPJ ;YES
11200 PUSHJ P,GETDR ;GET THE P-DISPLY
11300 PUSH P,FF ;WHAT A PITY WE MIGHT HAVE JUST POPPED
11400 PUSH P,A ;BUT THIS IS QUICKER IN THE LONG
11500 PUSH P,B ;RUN THAN MESSING WITH FLAGS
11600 PUSH P,C ;
11700 PUSH P,D
11800 TRZ FF,DBL ;ONLY ONE AC
11900 HRL D,LPSA ;USE P-DR AS INDEX
12000 MOVE B,TEMP ;WE WILL NEED THIS
12100 HRLI C,2 ;DISPL OF 2
12200 PUSHJ P,GETAN0 ;GET AN AC FOR DISPLY
12300 EMIT (<MOVE ,USX!USADDR!NORLC>) ;LOAD THE DR
12400 HRLM D,DISTAB(B) ;ENTER INTO DISPLAY TABLE
12500 PUSHJ P,DISBLK ;SET UP MOST OF BLOCK
12600 MOVEI TEMP,STRING ;
12700 HRRZM TEMP,$TBITS(LPSA) ;MAKE TYPE RIGHT
12800 MOVSS $VAL(LPSA) ;FIX UP AND MASK
12900 ;;#MN# !
13000 HRLM LPSA,LSDRNM
13100
13200 JRST RETSEQ ;GO POP STUFF & RETURN
13300 ↑↑GETDR:
13400 ;;#MN# !
13500 HRRM TEMP,LSDRLV
13600 HRRZ LPSA,DISTAB(TEMP) ;PICK UP THE PUTATIVE REGISTER
13700 ;;#MN# !
13800 HRRM LPSA,LSDRNM
13900 JUMPN LPSA,CPOPJ ;IF THERE,RETURN
14000 PUSH P,FF
14100 PUSH P,A
14200 PUSH P,B
14300 PUSH P,C
14400 PUSH P,D
14500 PUSH P,TEMP ;GETDR MUST SAVE IT
14600 TRZ FF,DBL ;ONLY ONE AC
14700 HRRZI B,1(TEMP) ;NEXT LEVEL DEEPER
14800
14900 GDR1: HRLZ D,DISTAB(B) ;PICK IT UP
15000 CAIN D,0 ;IS IT LOADED
15100 AOJA B,GDR1 ;NO
15200 HRLI C,1 ;SET TO SELECT STATIC LINK
15300 MOVE A,[<MOVE 0,USX!NORLC!USADDR>]
15400 GDR2: PUSHJ P,GETAN0 ;THIS BETTER LEAVE LH(D) ALONE -- IT DOES
15500 PUSHJ P,EMITER ;UP ONE STATIC LINK
15600 SOS B ;BACK A LEVEL
15700 HRRM D,DISTAB(B) ;SAY WE HAVE IT
15800 PUSHJ P,DISBLK ;TO DO STUFF FOR DISPLAY BLOCK&ACKTAB
15900 CAMN B,(P) ;IS THIS THE ONE WE WANT
16000 JRST GDR4 ;YES
16100 GDR3: HRL D,D ;USE AS INDEX PERHAPS
16200 HRR D,DISTAB-1(B) ;NEXT AC BACK
16300 TRNE D,-1 ;IS IT THERE
16400 SOJA B,GDR3 ;YES
16500 JRST GDR2 ;NO--FETCH IT
16600 GDR4: HRRZ LPSA,D ;AC NO OD DISPLY
16700 ;;#MN# !
16800 HRRM LPSA,LSDRNM ;REMEMBER NUMBER
16900 POP P,TEMP
17000 ;;#UW# ! JFR 8-17-75 CALL TO EMITER AT GDR2+1 WIPED OUT LSDRLV
17100 HRRM TEMP,LSDRLV
17200 RETSEQ: POP P,D
17300 POP P,C
17400 POP P,B
17500 POP P,A
17600 POP P,FF
17700 POPJ P, ;RETURN
17800
17900 COMMENT ⊗
18000 DSCR DISBLK
18100 DES THIS PROCEDURE SETS UP DISPLAY SEMBLK STUFF & UPDATES ACKTAB
18200 IT SETS LPSA TO POINT ATE THE NEW SEMBLK
18300 THE BLOCK IS SET UP FOR A LPSA TYPE SEMBLK
18400 PARM B = DISPLAY LEBEL
18500 D= ACNO OF DISPLAY REG
18600 ⊗
18700 ↑↑DISBLK:
18800 GETBLK ;GET A BLOCK
18900 HRRM D,$ACNO(LPSA) ;SAVE AC NO
19000 HRRM B,$ADR(LPSA) ;LEVEL GOES HERE
19100 SETOM TEMP
19200 HRLZM TEMP,$VAL(LPSA) ;SETS UP ANDING MASK
19300 MOVE TEMP,[XWD PTRAC!INAC!DISTMP,INTEGR]
19400 HRRZM TEMP,$TBITS(LPSA) ;$TBITS WORD
19500 HLLZM TEMP,$SBITS(LPSA) ;$SBITS WORD
19600 PUSHJ P,RNGDIS ;PUT IT ON DISLST LIST
19700 HRRZM LPSA,ACKTAB(D) ;MARK AC FULL OF IT
19800 POPJ P, ;RETURN
19900
20000 COMMENT ⊗
20100 DSCR ZOTDIS
20200 DES this procedure will wipe out your current display
20300 PARM None
20400 SID LPSA,TEMP used
20500 ⊗
20600 ↑↑ZOTDIS:
20700 PUSH P,D ;SAVE
20800 PUSH P,A
20900 MOVE A,CDLEV ;CURRENT DISPLAY LEVEL
21000 ZDIS.1: SOJL A,ZDIS.2
21100 HRRZ D,DISTAB+1(A)
21200 CAIE D,RF ;DONT ZONK RF
21300 CAIN D, ;DONT DO ANYTHING IF NOT THERE
21400 SKIPA
21500 PUSHJ P,STORZ
21600 HLRZ D,DISTAB+1(A)
21700 CAILE D,
21800 PUSHJ P,STORZ
21900 SETZM DISTAB+1(A)
22000 JRST ZDIS.1
22100 ZDIS.2: POP P,A
22200 POP P,D
22300 POPJ P,
22400
22500
22600
00100 COMMENT ⊗GET
00200
00300 GENMOV(GET) generally invokes this routine.
00400 It has many purposes, depending on the entity to be "getted".
00500 Briefly, however, it loads an AC with the thing one
00600 wants in order to store or compute using the entity in
00700 question. For strings, it loads a string address
00800 with the left half negative (for popping). For
00900 INDXED guys (with ADDR turned on), it loads
01000 the result of the index calc to an ac if it was not
01100 there. For regular variables, it simply picks them
01200 up if they are not in an AC. The bits
01300 ADDR, INDX, DBL, POSIT, NEGAT, and MARK
01400 may be used to modify the action of GETOPE.
01500
01600 ⊗
01700
01800 ↑↑GET: TRNE FF,PREMASK ;ANYTHING TO DO??
01900 PUSHJ P,PRE
02000 TRC FF,INSIST!NONSTD ;IF NO MARKING TO BE DONE, AND
02100 TRCE FF,INSIST!NONSTD ; A TYPE CONVERSION WAS DONE,
02200 JRST GETOPE
02300 ;; #OZ# (1 OF 1) PRE DOESN'T DO A GET OF ITEMS OR ITEMVARS
02400 TRNE B,ITMVAR!ITEM ; PRE DID NOT DO A GET
02500 ; IF ITEMVARS OR ITEMS
02600 JRST GETOPE
02700 ;; #OZ#
02800 HRRZ TEMP,B ; (COMPARE INSISTED TYPE WITH
02900 CAIE TEMP,(TBITS) ; ACTUAL TYPE), THEN DON'T GET
03000 JRST POST ; AGAIN
03100 ↑GETOPE:
03200 PUSHJ P,ACCOP ; ESTABLISH ACCESS TO THE EFFECTIVE ADDRESS.
03300
03400 COMMENT ⊗ IF STTEMP, NO MORE WORK NECESSARY
03500 (ASSUME STRING IS ON) ⊗
03600
03700 TLNN SBITS,STTEMP
03800 JRST GETOPC
03900 TRNN FF,ADDR ;MUST GO THRU WITH IT IF ADDR
04000 JRST TMPRET
04100
04200 COMMENT ⊗ USE LEFT HALF OF A TO HOLD SOME EXTRA BITS:
04300
04400 1 -- NEED AN AC (GETAC)
04500 2 -- DO A MOVE OF SOME SORT
04600 4 -- DO A MOVN
04700 10 - MAKE IT A HRRO
04800 20 - MAKE IT A HRROI, FOR STRING INDXED GUYS (SEE BELOW)
04900 40 - SPECIAL ACCOP BIT, SEE GETRET BELOW
05000 100 - SEEMS TO MEAN MOVEI -- WASN'T DOCUMENTED, DAMMIT
05100 200 - SPECIAL KLUGERY FOR RECORDS
05200 400 - KLUGE TO PREVENT GET ADDR OF REGULAR THING
05300 FROM MARKING INAC.
05400 1000 - USE DOUBLE WORD INSTRUCTIONS (DMOVE, ...) INSTEAD OF SINGLE (MOVE,...)
05500
05600 NEED EXTRA CHECKS IF ENTITY IS ALREADY IN AN AC
05700 ⊗
05800
05900 GETOPC: HRLZI A,3 ;ASSUME NEED A MOVE
06000 TRNE FF,SPAC ;UNLESS AC # PROVIDED,
06100 TLZ A,1 ; ASSUME AC NEEDED
06200 TLNN SBITS,INDXED ;IF ¬INDEXED, THEN TURN OFF NONSTD.
06300 TRZ FF,NONSTD ;SO AS NOT TO FOUL UP.
06400 REC <
06500 TRNE TBITS,PNTVAR ;MAKE SURE ONLY DO KLUGE IF A RECORD
06600 TRNE TBITS,777777-(PNTVAR!GLOBL) ;ONLY THAT BIT IS ALLOWED TO BE ON
06700 JRST NOSPAC ; NOT A RECORD
06800
06900 ;;#SA# ! GET ADDR IS JUST NORMAL
07000 TRNE FF,ADDR ;WELL??
07100 JRST NOSPAC
07200
07300 HLRZ TEMP,$ACNO(PNT) ;IN CASE WE WANT A MARK (USUALLY WILL)
07400 TRNE FF,MRK ;TEST IT OUT
07500 HRRZM TEMP,RCLASS ;NOW THE MARK WON'T DRYROT
07600 TLNE TBITS,CNST ;CONSTANTS ARE GETTABLE DIRECTLY
07700 JRST [ CAME PNT,NLRCBK ;THIS SHOULD BE THE ONLY ONE POSSIBLE
07800 ERR <RECORD CLASS TEMP OTHER THAN NULL RECORD?>,1
07900 JRST NOSPAC
08000 ]
08100
08200 TLNN SBITS,ARTEMP ;IF NOT A TEMP
08300 JRST RECKL1 ;THEN DO THE FIRST PART OF RECORD KLUGE
08400 TLNN SBITS,INDXED ;IF NOT INDEXED TEMP BUT A TEMP
08500 JRST NOSPAC ;DON'T DO ANYTHING ABOUT THIS
08600 ;WE WILL PERFORM THE INCREMENT OF
08700 ;THE REF CNT FOR ANY VARIABLE OR
08800 ;INDXED TEMP, WHETHER A SUBFIELD OR NOT
08900 RECKL1:
09000 NORGC <
09100 TLO A,200 ;BIT THAT SAYS TO DO RECORD ACCESS
09200 >;NORGC
09300 TRNN FF,SPAC ;KLUGE TO GET AC # GOOD
09400 TRO FF,INDX ;IF WE GET ONE, IT BETTER BE INDEXABLE
09500
09600 IFN 0,<
09700 TLNE SBITS,INAC ;IF INAC, WE WILL FORGET IT FOR THIS PURPOSE
09800 PUSHJ P,[ TLNE SBITS,INDXED ;BIG SURPRIZE IF THIS IS ON
09900 ERR <DRYROT: INDXED INAC?>,1
10000 ;;#SV# RHT MUST PRESERVE D
10100 PUSH P,D
10200 HRR D,$ACNO(PNT) ;GET OUT OF THE AC & THEN WILL WIN
10300 PUSHJ P,CLEARA ;FORGET INACITUDE
10400 POP P,D
10500 JRST GETAD ;REFURBISH THE BITS & RETURN FROM LITERAL
10600 ]
10700 >;FALSE
10800
10900 ;; FALL INTO NOSPAC
11000 >;REC
11100 NOSPAC: TLNN SBITS,INAC!PTRAC;IF IN AC, HAVE TO BE SURE IT'S RIGHT
11200 JRST STCHK ; IF NOT, MUST CHECK
11300 ; FOR STRINGS (HAVE TO LOAD)
11400
11500 ;DBLPRC PTRAC MUST TURN ON DBL UNLESS ADDR
11600 TRNN FF,ADDR
11700 TLNN SBITS,PTRAC
11800 JRST NOSPA1
11900 TDNN TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
12000 TRNN TBITS,DBLPRC
12100 JRST NOSPA1
12200 IORI FF,DBL
12300 NOSPA1:
00100
00200 Comment ⊗ INAC -- if DBL or INDX or SPAC,
00300 find out if thing can stay in this AC -- otherwise
00400 must get another. ⊗
00500
00600 ; FIRST CHECK SPAC GUYS
00700
00800 TLZ A,1!2 ;ASSUME NOTHING YET
00900 TRNN FF,SPAC ;PROVIDED WITH SPECIFIC AC?
01000 JRST DBCHK ; NO, CHECK DBL WANTED
01100 HRRZ TEMP,$ACNO(PNT) ;GET CURRENT AC #
01200 CAIN TEMP,(D) ;DID WE LUCK OUT (SAME ONE)?
01300 JRST SBSCHK ;YES, GO CHECK SPECIAL INDXED THING
01400
01500 ;DCS 8/16/70 IF SPAC AC BEING REPLACED,
01600 ; STORE AND CLEAR WHAT'S IN IT
01700 SKIPLE ACKTAB(D) ;PROTECTED OR NOTHING THERE?
01800 PUSHJ P,STORZ ; NO, GET RID OF IT
01900 ;DCS 8/16/70
02000
02100 TLO A,2 ;WILL HAVE TO DO A MOVE
02200 JRST WPCHK1 ;AND MAKE SEMANTICS CHANGES
02300
02400 ; IF DBL IS ON, SEE IF NEXT AC IS FREE, SET UP TO MOVE IF NOT
02500
02600 DBCHK:
02700 HRR D,$ACNO(PNT) ;GET CURRENT AC NUMBER
02800 TRNN FF,DBL ;WELL
02900 JRST IDXCHK ;NO DBL REQUESTED
03000
03100 SKIPGE ACKTAB+1(D) ;NEXT ONE NOT USABLE?
03200 JRST WIPCHK ; CANNOT BE USED, MAKE SEMANTIC CHANGES
03300
03400 HRRI D,1(D) ;STORE THE NEXT
03500 PUSHJ P,STORZ
03600 HRRI D,-1(D) ;RESTORE AC #
03700
03800
03900 IDXCHK: TRNE FF,INDX ;NEED INDX?
04000 TRNE D,-2 ; AND NOT IN ONE ALREADY?
04100 JRST SBSCHK ;OK, 'TWOULD SEEM
04200
04300
04400 Comment ⊗ If AC # is being changed (INAC and NEEDAC or SPAC and MOVE)
04500 clear right half of ACKTAB(AC), but first be sure nothing will be
04600 wiped out ⊗
04700
04800 WIPCHK: TLO A,1!2 ;HAVE TO MOVE IT
04900 WPCHK1: HRRZ TEMP,$ACNO(PNT) ;IT IS HERE CURRENTLY
05000 SKIPGE ACKTAB(TEMP) ;WAS THIS AC PROTECTED?
05100 ERR <DRYROT --AC CLOBBER>,1
05200 SETZM ACKTAB(TEMP) ;"STORR" (STORL DONE BEFORE)
05300 TDNN TBITS,[SBSCRP,,ITEM!ITMVAR!PROCED]
05400 TRNN TBITS,DBLPRC
05500 JRST WPCHK2 ;NOT VALUE LONG
05600 TLNE SBITS,PTRAC
05700 JRST WPCHK2 ;NEITHER IS THIS
05800 CAIN TEMP,RF-1
05900 ERR <DRYROT WIPCHK>,1
06000 CAIE TEMP,RF-1
06100 SETZM ACKTAB+1(TEMP) ;CLEAR 2ND AC OF LONG
06200 WPCHK2:
00100
00200 Comment ⊗ for STRING INDXED quantities (or non-STRING with ADDR)
00300 (guaranteed INAC by now) requiring a displacement,
00400 a "HRROI" FXTWO (or MOVEI)must be done --
00500 "HRRO" ("MOVE") with ADDR would yield a no-op
00600 ⊗
00700
00800 SBSCHK: TLNN SBITS,INDXED ;TEST THE CONDITONS
00900 JRST POSN ; NOT INDEXED
01000 HRRZ TEMP,$VAL(PNT) ;≠0 DISPLACEMENT?
01100 JUMPE TEMP,POSN ; NO DISPLACEMENT, NO PROBLEM
01200 ;; #OR# ! A STRING ITEMVAR IS NOT A STRING
01300 ;; #UE# ! NOR IS A STRING ARRAY
01400 TDNN TBITS,[XWD SBSCRP,ITMVAR!ITEM]; A STRING ITEM IS NOT A STRING
01500 TRNN TBITS,STRING ;INDXED STRING?
01600 JRST CHKNUM ; NO, CHECK GET!ADDR FOR NUMERIC ARRAY
01700 TRZ FF,ADDR ;JUST IN CASE
01800 TLO A,2!20 ;MOVE, HRROI, NO ADDR
01900 JRST POSN
02000
02100 CHKNUM: TRZE FF,ADDR ;WANT THE ADDRESS ALL TOGETHER?
02200 TLO A,100!2 ; YES, MOVE, MOVEI
02300 JRST POSN
02400
02500
02600 Comment ⊗ for strings, we must do a HRRO with ADDR
02700 turned ON (except for SBSCRP strings) ⊗
02800
02900 STCHK: TRNE FF,SPAC ;STORE AC IF SPAC
03000 PUSHJ P,STORZ
03100 TRNE TBITS,STRING ;STRING, NOT SBSCRP?
03200 ;;#VJ# ! JFR 10-17-75 A STRING PROCEDURE IS NOT A STRING, EITHER
03300 TDNE TBITS,[XWD SBSCRP,ITEM!ITMVAR!PROCED] ;NOT REALLY A STRING?
03400 JRST POSN
03500 TDO A,[XWD 2!10,ADDR] ;DO A "HRRO" ADDR
03600
03700 ; IF (POSIT(A) and NEGAT(SBITS)) or (NEGAT(A) and ¬ NEGAT(SBITS)) MUST
03800 ; DO SOMETHING ABOUT IT
03900
04000 POSN: TRNE FF,POSIT ;FIRST CONDITION
04100 TLNN SBITS,NEGAT
04200 JRST CHNGAT ; UNSATISFIED
04300 TLZ SBITS,NEGAT ;NO LONGER NEGAT
04400 TLO A,2!4 ;DO "MOVN"
04500 JRST CHKDX ;GO CHECK INDEXED
04600
04700 CHNGAT: TRNE FF,NEGAT ;SECOND CONDITION
04800 TLNE SBITS,NEGAT
04900 JRST CHKDX ; UNSATISFIED
05000 TLO SBITS,NEGAT ;NOW NEGAT
05100 TLO A,2!4 ;DO A "MOVN"
05200
05300 CHKDX: TLNN SBITS,INDXED ;IF INDXED, NOT STRING, NOT ADDR, BE
05400 JRST ADRCK
05500 ;; #RE# (1 OF 1) A STRING ITEMVAR ARRAY NOT A STRING ARRAY
05600 ;; #UE# ! (2 OF 3) STRING ARRAY INDXED TEMPS EXIST TOO
05700 TDNE TBITS,[XWD SBSCRP,ITMVAR!ITEM]
05800 JRST CHKDX2
05900 TRNE TBITS,STRING
06000 JRST ADRCK ;DOES NOT NEED A HRRO, HRROI
06100 CHKDX2:
06200 ;; # RE#
06300 ;;#TE# DAMNED CODE WAS PUTTING RESULT OF GET ADDR INAC
06400 TRNN FF,ADDR
06500 TLOA A,2 ; SURE SOME SORT OF MOVE GETS DONE
06600 TRO A,ADDR ;IN CASE OF INDXED THING, OK TO SAY "INAC"
06700 JRST ADRCKD ;(IF WAS STRING, MARKING INAC DOESN'T HURT)
06800 ADRCK: TRNE FF,ADDR ;NOW COPY THIS INTO A
06900 ;;#TE# ! USED TO BE A TRO A,ADDR
07000 TDO A,[400,,ADDR] ;LIKE ALL CPA'S.
07100 TRNE A,ADDR
07200 JRST GETWD ;WANT THE ADDRESS
07300 ADRCKD:
07400 TDNN TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR] ;THESE ARE BOGUS
07500 TRNN TBITS,DBLPRC
07600 JRST GETWD ;NOT DOUBLE
07700 TLO A,1000 ;USE DBL MOVES
07800 IORI FF,DBL ;AND DBL ACS
07900
00100
00200 GETWD: TRNN FF,NONSTD ;THE NON-STANDARD TYPE WILL
00300 ;**ALWAYS** GET AN AC.
00400 TLNE A,1 ;NEED AC?
00500 PUSHJ P,GETAC ; YES, GOT IT
00600 TLNN A,2 ;NEED TO MOVE?
00700 JRST [TLNN SBITS,INAC!PTRAC ;STRIVE TO PUT BITS BACK RIGHT
00800 JRST TMPRET
00900 TLNE SBITS,INDXED
01000 JRST IDXRET
01100 JRST GETRET] ;BEST AS POSSIBLE THE SAME AS ON ENTRY
01200 MOVE TEMP,A ;SAVE BITS SO YOU CAN TEST THEM
01300 PUSH P,A ;SAVE LH BITS
01400 HRLI A,(<MOVE>) ;ASSUME "MOVE"
01500 TLNE TEMP,1000 ;DOUBLE?
01600 HRLI A,(<DMOVE>)
01700 TLNE TEMP,4 ;MOVN?
01800 JRST [HRLI A,(<MOVN>) ; YES
01900 TLNE TEMP,1000
02000 HRLI A,(<DMOVN>) ;DOUBLE MOVN
02100 JRST .+1]
02200 TLNN TEMP,20!10 ;HRRO OR HRROI?
02300 JRST NOHRRO ;NO
02400 TRO A,FXTWO
02500 HRLI A,(<HRRO>)
02600 TLNE TEMP,20 ;ETC.
02700 HRLI A,(<HRROI>)
02800 NOHRRO:
02900 PUSH P,PNT
03000 TRNE TBITS,ITMVAR
03100 TLNN TBITS,MPBIND ;IF NOT ?ITEMVAR
03200 JRST NOTMPP ;CONTINUE
03300 TRZ A,ADDR
03400 ;; JRL FOLLOWING WAS MISTAKENLY A HRRI
03500 HRLI A,(<MOVEI @>)
03600 TRNE TEMP,ADDR ;ADDR REQUESTED
03700 ;; JRL FOLLOWING WAS MISTAKENLY A HRRI
03800 HRLI A,(<MOVE>)
03900 JRST EMTMOV ;EMIT THE MOVE
04000 NOTMPP: TLNE TEMP,100 ;FOR GET ADDR
04100 HRLI A,(<MOVEI>)
04200 TRO A,IMMOVE ;IF POSSIBLE
04300
04400 TRNE TBITS,ITEM ;OH MY GOSH AROODIES.
04500 JRST [TLNN TBITS,FORMAL!SBSCRP
04600 MOVE PNT,$VAL2(PNT) ; IT WILL BE AN INTEGER....
04700 JRST EMTMOV]
04800 REC <
04900 NORGC <
05000 TLNN TEMP,100 ;SPECIFIED IMMEDIATENESS
05100 TLNN TEMP,200 ;NO, THEN RECORD KLUGE IS A LIVE OPTION
05200 JRST EMTMOV ;NOT A RECORD KLUGERY INSTANCE
05300 RECKL2: HRLI A,(<SKIPE>) ;NEED TO BUMP REF CNT IF NOT NULL
05400 ;;BUG TRAP
05500 TRZE A,USCOND
05600 ERR <DRYROT: USCOND ON AT RECKL2>,1
05700 PUSHJ P,EMITER
05800 HRLOI A,(<AOS>) ;
05900 TLO A,(D) ;PUT AC NUMBER IN PLACE
06000 TLZ FF,RELOC ; NOT A RELOCATABLE -1 !
06100 PUSHJ P,CODOUT ; AOS -1(AC)
06200
06300 SKIPA ;SKIP OVER EMITER CALL AT EMTMOV
06400
06500 >;NORGC
06600 >;REC
06700 EMTMOV: PUSHJ P,EMITER
06800 POP P,PNT ;IN CASE OF ITEM.
06900
07000 POP P,A
07100 TLNE TBITS,MPBIND
07200 JRST [TLO SBITS,INAC
07300 TRNN A,ADDR ;ADDR?
07400 JRST ALLRET ;NO.
07500 ;; #PA#!(1OF 2) SAVE C ON CALL TO GET
07600 PUSH P,C
07700 HRLZI C,20 ;INDIRECT BIT
07800 EMIT <TLZN ,USADDR!NORLC>
07900 ;; #PA#!(2 OF 2) RESTORE C
08000 POP P,C
08100 EMIT <MOVEI ,0>
08200 TLZ SBITS,INAC
08300 JRST TMPRET] ;DON'T REMEMBER ADDR IS IN AC
08400
08500
08600 ;;#TE# DONT WANT TO ALWAYS REMEMBER THIS AC IN $ACNO
08700 GETRET: TLNN A,400 ;WAS IT REGULAR VBL, GET (ADDR)
08800 TRNE FF,NONSTD ;SPECIAL CASE OF PRESERVING INDXD TEMPS
08900 JRST [MOVE SBITS,$SBITS(PNT) ;RESTORE OLD MARKING.
09000 JRST TMPRT1] ;AND FINISH OUT.
09100 TLZ SBITS,PTRAC!INDXED!INAC ;START FROM SCRATCH
09200 TLNN A,20!40!100 ;INAC MARKING?
09300 JRST STDRET ; YES, DO IT
09400
09500 IDXRET: TLO SBITS,PTRAC!INDXED;KEEP INDXED BITS
09600 TLNN A,20!100 ;HRROI (MOVEI) THING?
09700 JRST ALLRET ; NO
09800 TLZ TBITS,OWN
09900 HLLZS $VAL(PNT) ; NO DISPL ANYMORE
10000 JRST ALLRET
10100
10200 STDRET: TDNN TBITS,[XWD SBSCRP,ITEM!ITMVAR] ;NOT REALLY A STRING?
10300 TRNN TBITS,STRING ;KEEP BITS OFF IF STRING
10400 TLO SBITS,INAC
10500 ALLRET: HRRM PNT,ACKTAB(D) ;UPDATE SEMANTICS AND
10600 HRRM D,$ACNO(PNT) ; ACKTAB
10700 TLNN SBITS,PTRAC
10800 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR] ;THESE ARE BOGUS
10900 JRST .+3
11000 TRNE TBITS,DBLPRC
11100 JRST [MOVEI TEMP,(D)
11200 CAIN TEMP,RF-1
11300 ERR <DRYROT ALLRET>,1
11400 CAIE TEMP,RF-1
11500 HRRM PNT,ACKTAB+1(D) ;SECOND AC OF DOUBLE
11600 JRST .+1]
11700
11800 TMPRET: MOVEM SBITS,$SBITS(PNT) ;IF ACCOP, THIS WILL BE NECESSARY
11900 TMPRT1: TRNN FF,MRK ;DOES HE WANT A MARK?
12000 JRST POST ;ALL DONE.
12100 PUSHJ P,REMOP ;AFTER ALL THAT?
12200 JRST MARK1 ;AH, WELL
12300
00100 COMMENT ⊗STACK -- Issue Instrs. to Stack Anything on Approp. Stack⊗
00200
00300 ↑↑STACK: TRNE FF,PREMASK ;ANY TO DO?
00400 PUSHJ P,PRE
00500 PUSHJ P,STACK1
00600 TRNN FF,MRK ;HAS HE ASKED FOR A MARK?
00700 JRST POST ;FINISH OUT.
00800 JRST MARK1 ;AND DO A MARK.
00900
01000
01100 STACK1: PUSH P,FF ;SAVE
01200 TRNN SBITS,DLFLDM ;DOES HE LIVE IN THE STACK?
01300 TLNE SBITS,INDXED
01400 PUSHJ P,ACCOP ;GET ACCESS.
01500 TDNE TBITS,[XWD SBSCRP,ITEM!ITMVAR] ;ALWAYS STACK ARRAYS ON P-STACK
01600 JRST ASTACK ; NO MATTER WHAT
01700 TRNN FF,ADDR ;MUST BE A CALL BY REF.
01800 TRNN TBITS,STRING ;STRING STACK?
01900 JRST ASTACK ;NO -- ARITHMETIC
02000 TLNE SBITS,STTEMP ;IF STTEMP and INUSE,
02100 ; ALREADY STACKED, DON'T DO AGAIN
02200 JRST MARTK ;JUST MARK AND QUIT
02300
02400
02500 MOVEI D,RSP ;TO AVOID CLOBBERING CORE.
02600 MOVE A,[PUSH RSP,STAK!BPWORD!LNWORD!ADOP!REM]
02700 TRNE FF,REM ; IF REM BIT IS ON IN FF THEN DON'T REMOP IN
02800 TRZ A,REM ; STROP1 SINCE POST WILL DO IT
02900 PUSHJ P,STROP1 ;THIS IS REALLY EASY. DO TWO PUSHES.
03000 ;; FOLLOWING WAS ERRONEOUSLY TO MARTK THUS REMOPING BLOCK TWICE
03100 ;; #ML ACTUALLY NEED TO LOAD SBITS AGAIN SHOULD BE MARTJ
03200 JRST MARTJ ;AND NOW MARK THINGS.
03300
03400
03500
03600 ASTACK: TLZN SBITS,NEGAT ;ARE THINGS CURRENTLY NEGATIVE?
03700 JRST OKPO ;NO
03800 TLNN SBITS,INAC!PTRAC
03900 ERR <DRYROT -- STACK NEGAT IN CORE?>,1
04000 HRL C,$ACNO(PNT)
04100 TRNE TBITS,DBLPRC
04200 JRST [HLR D,C ;SAME AC AS ADDR
04300 EMIT (DMOVN USADDR!NORLC) ;DMOVN AC,AC
04400 JRST ASTA.1]
04500 EMIT (MOVNS USADDR!NORLC!NOUSAC)
04600 ASTA.1: MOVEM SBITS,$SBITS(PNT);FOR THE EMITER.
04700 OKPO:
04800 REC <
04900 NORGC <
05000 TRNE TBITS,PNTVAR ;IS IT A PNTVAR (IE RECORD)
05100 TRNE TBITS,777777-(PNTVAR!GLOBL) ;
05200 JRST OKPO.1 ;NO
05300 TRNE FF,ADDR ;WANT ADDRESS?
05400 JRST OKPO.1 ;WON'T WORK ANYHOW
05500 PUSH P,FF ;
05600 GENMOV (GET,MRK) ;GET IT & MARK IT
05700 POP P,FF
05800 OKPO.1: ;
05900 >;NORGC
06000 >;REC
06100
06200 TLNE TBITS,MPBIND ;A ?ITEMVAR
06300 JRST [TRNE FF,ADDR ;ADDRESS REQUIRED?
06400 ERR <DRYROT -STACK ADDR ? ITEMVAR>
06500 TLNE SBITS,PTRAC!INAC
06600 JRST .+1
06700 PUSH P,D
06800 PUSHJ P,GETAC
06900 EMIT <MOVEI @,>
07000 PUSHJ P,MARKINT
07100 POP P,D
07200 JRST .+1]
07300 HRLZI A,(<PUSH RP,>)
07400 TRNE FF,ADDR ;COPY THIS BIT.
07500 TRO A,ADDR
07600 TRO A,NOUSAC ;WE HAVE SPECIFIED IT.
07700 PUSHJ P,EMITER ;PUT OUT THE PUSH.
07800 AOS ADEPTH ;SINCE WE USED THE PSTACK
07900 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR] ;THESE ARE NOT DOUBLE
08000 JRST MARTK
08100 TRNE TBITS,DBLPRC
08200 TRNE FF,ADDR
08300 JRST MARTK ;ADDR OR NOT DOUBLE
08400 EMIT (<PUSH RP,NOUSAC!FXTWO>) ;SECOND WORD
08500 AOS ADEPTH
08600 MARTK: TRNN FF,REM ;IF REM BIT IS ON THEN DON'T DO REMOP SINCE POST
08700 ; WILL DO IT
08800 PUSHJ P,REMOP ;REMOVE THE THING YOU'RE STACKING
08900 MARTJ: MOVE SBITS,$SBITS(PNT);GET ITS BITS BACK FOR THE REST OF THIS
09000 MARTH: POP P,FF ;RESTORE
09100 POPJ P,
09200
00100 COMMENT ⊗MARK, MARKINT, MARKME -- Mark Semblk with Correct Temp Semantics
00200 This marks the AC (D) with a temp descriptor of type in SBITS, TBITS⊗
00300
00400 ↑↑MARK: TRNE FF,PREMASK ;
00500 PUSHJ P,[TRNE FF,657777
00600 ERR <MARK>,1
00700 JRST PRE]
00800 PUSHJ P,MARK1
00900 JRST POST ;ALL DONE.
01000
01100 MARK1: ANDI TBITS,MASK ;WANT ONLY THE TYPE BITS (NOT FORMAL,ETC.)
01200 ;;#NK# ! (1 OF 2) TEMPS SHOULD NOT HAVE DISPLAY LEVELS
01300 TDZ SBITS,[CORTMP!PTRAC!INDXED!FIXARR,,DLFLDM]
01400 ;;#HS# JRL AN ITEMVAR IS NOT ITS DATUM
01500 TRNE TBITS,ITMVAR!ITEM
01600 JRST .+3
01700 ;;#HS#
01800 TRNE TBITS,STRING ;IF STRING TYPE, THEN
01900 JRST STMARK
02000 TLO SBITS,INAC!ARTEMP!INUSE ;SINCE HE MAY NOT HAVE SET THEM.
02100 TLZ SBITS,STTEMP
02200 HRRE LPSA,ACKTAB(D) ;PICK UP TEMP DESCIRIPTOR
02300 JUMPLE LPSA,NOTEM ;IF NO TEMP OR REMOPPED TEMP
02400 MOVE USER,$SBITS(LPSA) ;GET SEMANTIC BITS
02500 TLNN USER,INUSE ;A TEMP?
02600 JRST REMM ;NO
02700 TLNN USER,CORTMP ;A CORE TEMP?
02800 JRST USOLD ;NO -- USE THE TEMP THAT IS THERE.
02900 TLNE USER,INAC ;IS IT STILL IN THE ACCUMULATOR?
03000 PUSHJ P,STORA ;YES --STORE IT.
03100
03200 SKIPA
03300 REMM: PUSHJ P,CLEARL ;DO THE REMOP
03400 NOTEM: PUSHJ P,GETTEM ;GET A NEW TEMPORARY
03500 USOLD: HRRM LPSA,ACKTAB(D) ;INSERT IN AC TABLE RIGHT HALF
03600 TLNN SBITS,PTRAC
03700 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
03800 JRST .+3
03900 TRNE TBITS,DBLPRC
04000 JRST [
04100 ;;#YX# 3! JFR 2-10-77 SPURIOUS MESSAGE FOR X←-((Y+Z)+Z);
04200 MOVE TEMP,ACKTAB(D)
04300 CAMN TEMP,ACKTAB+1(D)
04400 JRST .+1 ;ALREADY SAME, ASSUME RE-USE
04500 SKIPE ACKTAB+1(D)
04600 ERR <DRYROT MARK DOUBLE>,1
04700 MOVEI TEMP,(D)
04800 CAIE TEMP,RF-1 ;DO NOT CLOBBER RF!
04900 HRRM LPSA,ACKTAB+1(D);SECOND AC OF DOUBLE
05000 JRST .+1]
05100 HRRM D,$ACNO(LPSA) ;AND THE LOGICAL INVERSE.
05200 REC <
05300 TRNE TBITS,PNTVAR ;A RECORD TEMP
05400 TRNE TBITS,777777-(PNTVAR!GLOBL) ;
05500 JRST MARKT ;NOPE
05600 SKIPN TEMP,RCLASS ;BUG TRAP
05700 ERR <DRYROT: RCLASS=0 WHEN TRYING TO MARK RECORD TEMP>,1
05800 HRLM TEMP,$ACNO(LPSA) ;MARK IT
05900 SETZM RCLASS ;
06000 ;FALL INTO MARKT
06100 >;REC
06200 MARKT: HRRZM LPSA,PNT ;
06300 SETZM $VAL(PNT)
06400 MARTS: POPJ P,
06500 STMARK: TLO SBITS,STTEMP ;IN CASE IT SKIPS AND NOONE ELSE DID
06600 TLZ SBITS,ARTEMP
06700 HRRZ LPSA,PNT ;IN CASE STRTMP NOT CALLED
06800 TLNN SBITS,INUSE ;ALREADY HAS A TEMP?
06900 PUSHJ P,STRTMP ;GET A STRING TEMP.
07000 JRST MARKT
07100
07200 DSCR MARKINT, MARKME
07300 DES THESE ARE ROUTINES TO HELP YOU CALL "MARK"
07400 MARKINT -- ALWAYS MARKS A VANILLA INTEGER, RETURNS DESCR. IN PNT,SBITS,TBITS.
07500 MARKME -- YOU SPECIFY TBITS, SBITS=0 IS ASSUMED
07600 ⊗;
07700 ↑↑MARKINT: MOVEI TBITS,INTEGR ;MARK AN INTEGR,
07800 ↑↑MARKME: HRRI FF,0
07900 SETZ SBITS,
08000 JRST MARK1
08100
00100 COMMENT ⊗INCOR -- Issue Code to Clear this Entity from ACs⊗
00200
00300 DSCR INCOR
00400 DES makes sure that the entity mentioned in PNT,TBITS,SBITS is really
00500 in core. If not, the AC entry for that entity is cleared.
00600 The updated Semantics bits are returned in SBITS.
00700 ⊗;
00800
00900 ↑↑INCOR:
01000 TLZN SBITS,INAC!PTRAC ;GONE?
01100 POPJ P, ;ALL DONE!
01200 PUSH P,D ;SAVE THIS.
01300 HRRZ D,$ACNO(PNT) ;PICK UP RELEVANT AC.
01400 PUSHJ P,STORZ
01500 POP P,D
01600 JRST GETAD ;ALAS, SINCE STORZ WILL CHANGE THINGS.
01700
00100 COMMENT ⊗REMOPs, CLEARs -- Remove Temps, ACs, from Use⊗
00200
00300 DSCR REMOP,REMOPA,REMOPL,REMOP2
00400 DES These are the REMOP routines. They say, in effect, "I am
00500 finished with this argument. If it was a temp descriptor, then I
00600 am really finished, and the temp may be returned to the pool of
00700 such temps. If it was a simple variable or constant, etc. then no
00800 action is taken.
00900
01000 PAR The differences among the routines are only in the call form:
01100 REMOP -- PNT has pointer to entity.
01200 REMOPL -- LPSA has pointer to entity
01300 REMOPA -- D has AC number of entity.
01400 REMOP2 -- PNT2 has pointer to entity.
01500
01600 SID AC'S USED: LPSA,TEMP,USER
01700 ⊗;
01800
01900
02000 ↑REMOP2: MOVE LPSA,PNT2
02100 JRST REMOPL
02200 ↑REMOPA: SKIPA LPSA,ACKTAB(D) ;REMOP BY ACCUMULATOR NUMBER
02300 ↑REMOP: MOVE LPSA,PNT ;OH WELL.
02400 ↑REMOPL: TRNN LPSA,-1
02500 POPJ P, ;NONE THERE.
02600 MOVE TEMP,$SBITS(LPSA);THE STANDARD REMOP
02700 TLNN TEMP,STTEMP!ARTEMP!INUSE ;A REAL TEMP?
02800 JRST STCNST ;NO, CHECK IF A STRING CONSTANT
02900 DELAL:
03000 REC <
03100 NORGC <
03200 TLNN TEMP,INDXED
03300 JRST DRFDON ;DONT HAVE TO DEREFERENCE IT
03400 HRRZ USER,$VAL2(LPSA);WAS THIS GUY A RECORD SUBFIELD
03500 JUMPE USER,DRFDON ;IF NOT, THEN NOTHING TO WORRY ABOUT
03600
03700 SKIPN USER,%RVARB(LPSA);UNLINK SELF FROM SUBFIELD CHAIN
03800 ERR <DRYROT: REMOP OF SUBFIELD NOT ON SUBFIELD CHAIN>,1,SFULKD
03900 TRNE USER,-1 ;ASS END OF CHAIN ?
04000 HLLM USER,%RVARB(USER) ;NO, MAKE THE RIGHT GUY POINT AT MY LEFT
04100 MOVS USER,USER ;NOW LINK THE OTHER WAY
04200 HLRM USER,%RVARB(USER) ;MY LEFT POINTER NOW POINTS AT MY RIGHT
04300 SETZM %RVARB(LPSA) ;TIDY UP
04400 SFULKD: ;UNLINKING DONE NOW
04500
04600 HLLZS $VAL2(LPSA) ;MAKE SUBFIELD FLAG ZERO AGAIN
04700 HLRZ USER,%TLINK(LPSA); WAS THIS THING HANGING RECD REF
04800 CAIN USER,-1 ;IF SO,THIS IS -1
04900 JRST DREFIT ;IT WAS, MUST DE-REFERENCE THIS ONE
05000
05100 PUSH P,USER ;I AM A SUBFIELD OF A FIELD
05200 PUSHJ P,DRFDON ;KILL MYSELF OFF
05300 POP P,LPSA ;THEN REMOP THE FIELD I HUNG OFF OF
05400 JRST REMOPL ;
05500
05600 DREFIT:
05700 SETZM $VAL(LPSA) ;SO THAT THE DEREF WORKS
05800
05900 PUSH P,A ;SAVE SOME ACS
06000 PUSH P,C ;
06100 PUSH P,PNT
06200 PUSH P,TEMP
06300 PUSH P,LPSA
06400 ;;#RY# MUST DO RECUUO ON AC, NOT (AC)
06500 MOVE TEMP,$SBITS(LPSA)
06600 MOVEI C,ARTEMP+INAC+INUSE
06700 TLNE TEMP,CORTMP
06800 TRC C,INAC+CORTMP
06900 HRLM C,$SBITS(LPSA)
07000 ;;#RY#
07100 MOVNI C,1 ;TO DO DEREFERENCING BY 1, SET C TO -1
07200 MOVE PNT,LPSA ;THE THING TO DEREFERENCE
07300 PUSHJ P,RFCADJ ;ADJUST REFERENCE COUNT
07400 ;**** NOTE: MAY BE SAFER TO PUT THESE
07500 ; ONTO SOME "HANG LIST" UNTIL STATEMENT LEVEL
07600 ; THIS IS BETTER, THOUGH, IF NOTHING BAD HAPPENS
07700
07800 POP P,LPSA ;RECOVER THESE FROM EARLIER
07900 POP P,TEMP
08000 POP P,PNT ;
08100 POP P,C
08200 POP P,A
08300
08400 DRFDON:
08500
08600 ;; HERE CAN FALL INTO THE REST OF THE DELALL CODE. THIS WILL BE OK
08700 ;; SO LONG AS (1) DON'T SUFFER THE LOSSAGE I FEAR ABOUT ROUTINES ASSUMING
08800 ;; REMOP LEAVES PCNT THE SAME (ONE KLUGE WOULD BE TO SET A FLAG TO ALLOW
08900 ;; THE NEXT CALL TO ACCESS TO DO THE "RIGHT" THING, BUT UGH!
09000 ;; (2) THE CODE ABOVE ONLY GOBBLES THE SORT OF INDEXED TEMPS I EXPECT IT TO
09100 ;; IF NOT, MORE TESTING & MARKING IS REQUIRED
09200 >;NORGC
09300 RGC <
09400 ;;#WD# RHT 1-25-76 MAKE SURE THAT DEPENDENT TEMP TO STRING GOES, TOO.
09500 MOVE USER,$TBITS(LPSA) ;
09600 TLNE TEMP,INDXED ;INDEXED
09700 TRNN USER,STRING!DBLPRC ;STRING, TOO
09800 JRST RMP.00 ;NOPE
09900 TDNE USER,[XWD SBSCRP,PROCED!ITEM!ITMVAR] ;EXCEPT FOR THESE
10000 JRST RMP.00
10100 HRRZ USER,$VAL2(LPSA) ;IS IT SUBFIELD
10200 JUMPE USER,RMP.00 ;NO
10300 PUSH P,LPSA ;SAVE STATE
10400 PUSH P,TEMP ;
10500 HLRZ LPSA,$ACNO(LPSA) ;HAVE WE A DEPENDENT?
10600 SKIPE LPSA
10700 PUSHJ P,REMOPL ;YUP, FLUSH HIM, TOO
10800 POP P,TEMP
10900 POP P,LPSA
11000 JRST RMP.1 ;STRING SUBFIELD INDXED TEMPS
11100 ;DO NOT HAVE RECORDS AT ALL
11200 RMP.00:
11300 ;;#WD# ↑
11400 TLNN TEMP,CORTMP ;ONLY CORTMPS ARE SPECIAL
11500 JRST RMP.1 ;
11600 TLNN TEMP,INDXED ;INDXED CORTMP??
11700 JRST RMP.0 ;NOPE
11800 HRRZ USER,$VAL2(LPSA) ;RECORD SUBFIELD??
11900 JUMPE USER,RMP.1 ;NOPE
12000 MOVSI USER,CORTMP!INUSE!ARTEMP;MAKE INTO A RECORD CORTMP
12100 MOVEM USER,$SBITS(LPSA)
12200 MOVEI USER,PNTVAR
12300 MOVEM USER,$TBITS(LPSA) ;LIKE SO
12400 JRST RMP.RC ;PUT IT ONTO THE RIGHT RING
12500
12600 RMP.0:
12700 MOVE USER,$TBITS(LPSA) ;
12800 TRNE USER,PNTVAR ;WAS IT A RECORD CORTMP
12900 ;;#VQ# ! RECORD ARRAYS ARE ALSO OK
13000 TDNE USER,[XWD SBSCRP,ITEM!ITMVAR] ;THESE ARE OK
13100 JRST RMP.1 ;NOPE
13200 RMP.RC:
13300 ;;%##% BUG TRAP
13400 HRRZ USER,RCTEMP ;WAS THIS GUY ALREADY ON THE CHAIN
13500 JUMPE USER,RMP.0R ;NO CHAIN
13600 CAIN USER,(LPSA) ;WELL?
13700 ERR <DRYROT: RECORD CORTMP REMOP>,1
13800 HRRZ USER,(USER) ;CHAIN
13900 JUMPN USER,.-3
14000 RMP.0R: ;;%??% INSERTED HERE BY JFR 11-16-75
14100 ;;%##% ↑
14200 HRRZ USER,LPSA ;
14300 EXCH USER,RCTEMP ;
14400 HRRZM USER,%TLINK(LPSA) ;REMEMBER IT AS AN AVAILABLE
14500 JRST IACCHK ;RECORD TEMP
14600 ;(NOTICE THAT INUSE WAS LEFT ON)
14700 RMP.1:
14800 >;RGC
14900 >;REC
15000 MOVSI USER,INUSE!STTEMP!INAC!PTRAC!NEGAT!FIXARR ;TURN THESE OFF
15100 ANDCAM USER,$SBITS(LPSA) ;IN MEMORY.
15200 IACCHK: HRRZ USER,$ACNO(LPSA) ;GET THE AC IT WAS IN
15300 TLNN TEMP,INAC!PTRAC ;WAS IT IN AN AC?
15400 JRST CTCHK ;NO -- ALL DONE.
15500 SKIPGE ACKTAB(USER) ;YES --TURN IT OFF.
15600 ERR <DRYROT -- REMOP>,1
15700 SETZM ACKTAB(USER)
15800 TLNN TEMP,PTRAC
15900 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
16000 JRST .+3
16100 TRNE TBITS,DBLPRC
16200 JRST [CAIN USER,RF-1
16300 ERR <DRYROT IACCHK>,1
16400 CAIE USER,RF-1
16500 SETZM ACKTAB+1(USER)
16600 JRST .+1]
16700 CTCHK: TLNE TEMP,INUSE ;If this was still an alive temp, and
16800 TLNE TEMP,CORTMP ; was not a CORTMP, thus contains no fixups
16900 POPJ P, ; or anything, we can release it to free
17000 PUSH P,LPSA ; storage. Otherwise, leave it on the TTEMP
17100 PUSHJ P,BLKFRE ; list (where it MUST be), and forget it.
17200 POPJ P,
17300
17400
17500 STCNST: MOVE TEMP,$TBITS(LPSA) ;
17600 TLNE TEMP,CNST ;
17700 TRNN TEMP,STRING ;
17800 POPJ P, ; RETURN IF NOT A STRING CONSTANT
17900 MOVE TEMP,$PNAME(LPSA) ; CHECK IF TRYING TO REMOP NULL STRING WHICH IS
18000 TRNN TEMP,-1 ; ONLY STRINS'ED ONCE
18100 POPJ P, ; YES, DON'T REMOP
18200 SOSLE TEMP,$VAL2(LPSA) ; DECREMENT REFERENCE COUNT AND GET OUT IF NOT
18300 POPJ P, ; ZERO
18400 JUMPE TEMP,.+2 ; ZERO COUNT?
18500 ERR <DRYROT REMOP:STCNST> ;
18600 SKIPN $VAL(LPSA) ; USED IN PRELOAD?
18700 SKIPE $ADR(LPSA) ; USED IN FIXUP?
18800 POPJ P, ; YES, RETURN
18900 PUSHJ P,URGCST ; REMOVE FROM STRING CONSTANT RING
19000 PUSHJ P,URGSTR ; REMOVE FROM STRING RING
19100 PUSH P,PNAME ; SAVE PNAME AND PNAME+1 (OK ON P STACK SINCE NO
19200 PUSH P,PNAME+1 ; GARBAGE COLLECTION CAN HAPPEN)
19300 HRROI TEMP,$PNAME+1(LPSA) ; GET STRING DESCRIPTOR FOR HASH LOOKUP SO THE
19400 POP TEMP,PNAME+1 ; STRING CAN BE REMOVED FROM THE HASHED SYMBOL
19500 POP TEMP,PNAME ; TABLE
19600 PUSH P,TBITS ; SAVE AC'S WHICH SHASH WILL DESTROY
19700 PUSH P,A ;
19800 PUSH P,B ;
19900 PUSH P,C ;
20000 PUSH P,D ;
20100 PUSH P,PNT ;
20200 PUSH P,LPSA ;
20300 MOVE LPSA,STRCON ; USE STRING HASH TABLE
20400 PUSHJ P,SHASH ;
20500 MOVE B,HPNT ; INSTRUCTION TO LOAD FIRST IN CONFLICT LIST
20600 XCT B ; FIRST IN CONFLICT LIST INTO LPSA
20700 HRRZ PNT,(P) ; THE ONE WE ARE LOOKING FOR
20800 MOVEI A,LPSA ;
20900 SCOMLP: HRRZ TEMP,(A) ; CANDIDATE?
21000 JUMPE TEMP,ERRSTC ; NOT THERE - ERROR
21100 CAMN TEMP,PNT ;
21200 JRST SFNDIT ;
21300 MOVE A,TEMP ; CHAIN DOWN CONFLICT LIST
21400 JRST SCOMLP ;
21500 SFNDIT: HRRZ TEMP,(TEMP) ; NEXT IN LIST
21600 HRRM TEMP,(A) ; CHAIN AROUND DELETED ELEMENT
21700 TLO B,2000 ; CHANGE FROM LOAD TO STORE
21800 XCT B ;
21900 FREBLK (PNT) ;
22000 POP P,LPSA ; RESTORE AC'S
22100 POP P,PNT ;
22200 POP P,D ;
22300 POP P,C ;
22400 POP P,B ;
22500 POP P,A ;
22600 POP P,TBITS ;
22700 POP P,PNAME+1 ;
22800 POP P,PNAME ;
22900 POPJ P, ;
23000 ERRSTC: ERR <DRYROT AT REMOP>,1 ;
23100
00100 DSCR CLEAR,CLEARL,CLEARA
00200 DES These are routines to clear an entry in the AC table (ACKTAB)
00300 That is, all memory of what is in the AC is lost. The difference
00400 among the routines is the call form:
00500
00600 PAR CLEAR -- PNT has pointer to entity to be "cleared"
00700 If it turns out not to be in an AC, no action is taken.
00800 CLEARL -- LPSA has pointer; same deal.
00900 CLEARA -- D has AC number to be cleared.
01000
01100 SID AC'S USED: LPSA,TEMP
01200 ⊗;
01300
01400 ↑CLEAR: MOVEI LPSA,(PNT) ;CLEAR OUT AN AC TABLE ENTRY.
01500 ↑CLEARL: MOVE TEMP,$SBITS(LPSA) ;SEE IF IT IS IN AN AC.
01600 TLNN TEMP,INAC!PTRAC ;IF NOT -- ALL DONE.
01700 POPJ P, ;DONE.
01800 MOVE TEMP,$ACNO(LPSA) ;AC IT IS IN.
01900 ;;#YJ# 2! JFR 1-13-77 QUIT IF NOTHING THERE
02000 SKIPN ACKTAB(TEMP)
02100 JRST CLR1
02200 SETZM ACKTAB(TEMP) ;AND ZERO THE ENTRY.
02300 MOVE TEMP,$SBITS(LPSA)
02400 TLNE TEMP,PTRAC
02500 JRST CLR1 ;POINTERS ARE NOT LONG
02600 MOVE TEMP,$TBITS(LPSA)
02700 TDNN TEMP,[SBSCRP,,PROCED!ITEM!ITMVAR] ;LEAP IS NOT LONG
02800 TRNN TEMP,DBLPRC ;LONG?
02900 JRST CLR1 ;NO
03000 HRRZ TEMP,$ACNO(LPSA);YES, REFETCH AC
03100 JRST CLR2
03200
03300 ↑CLEARA:
03400 SKIPN LPSA,ACKTAB(D)
03500 POPJ P, ;NOTHING THERE
03600 CAMN LPSA,ACKTAB-1(D) ;POSSIBLE DBLPRC SCREWUP?
03700 SOJA D,[PUSHJ P,CLEARA ;YES, CLEAR FIRST AC INSTEAD
03800 AOJA D,CPOPJ] ;RESTORE D
03900 SETZM ACKTAB(D) ;ZERO AC TABLE ENTRY
04000 MOVE TEMP,$SBITS(LPSA)
04100 TLNE TEMP,PTRAC
04200 JRST CLR1 ;POINTERS ARE NOT LONG
04300 MOVE TEMP,$TBITS(LPSA)
04400 TDNE TEMP,[SBSCRP,,PROCED!ITEM!ITMVAR]
04500 JRST CLR1
04600 TRNE TEMP,DBLPRC ;DOUBLE?
04700 JRST [MOVEI TEMP,(D)
04800 CLR2: HLL LPSA,ACKTAB+1(TEMP) ;MAKE COMPARISON ON RIGHT HALF ONLY
04900 CAIE TEMP,RF-1 ;MAKE THIS IRON-CLAD
05000 CAME LPSA,ACKTAB+1(TEMP);NEXT AC SHOULD BE SAME
05100 ERR <DRYROT CLEAR DOUBLE>,1
05200 CAIE TEMP,RF-1
05300 SETZM ACKTAB+1(TEMP) ;CLEAR SECOND AC
05400 JRST .+1]
05500 CLR1: MOVSI TEMP,INAC!PTRAC!NEGAT
05600 TRNE LPSA,-1 ;ANYTHING THERE? (DCS -- 8/16/70)
05700 ANDCAM TEMP,$SBITS(LPSA) ;TURN THESE OFF IN MEMORY.
05800 POPJ P,
05900
00100 COMMENT ⊗STROP -- Bit-Driven String Operation Code Generator⊗
00200
00300 DSCR STROP
00400 DES This routine is willing to do lots of twiddling on strings.
00500 It knows about reference strings, etc.
00600 PAR A is an instruction for the EMITTER, with some bits in
00700 it to say what things should be done with this instruction.
00800 Bits in A: bpword -- issue the instruction for
00900 the byte pointer word.
01000 lnword -- or for the length word.
01100 bpfirst -- issue the byte pointer inst. first.
01200 adop -- this is an instruction which adds to stack.
01300 sbop -- this is an instruction which subs from stack.
01400 undo -- so a SUB SP,X22 at end.
01500 rem -- do a remop when done.
01600
01700 stak -- used internally.
01800 bpinc -- byte pointer instruction is in c(rh)
01900
02000 PNT,TBITS,SBITS -- semantics of string.
02100
02200 D -- accumulator to use for ac field of op.
02300 Thus, it must be RSP if that stack is to be used.
02400 ⊗;
02500
02600
02700 ↑STROP: CAIN D,RSP ;IF THE STACK,
02800 TRO A,STAK ;THEN MARK AS SUCH.
02900 DPB D,[POINT 4,A,12] ;SAVE IN AC FIELD OF INSTRUCTION.
03000 PUSHJ P,ACCOP ;AND GET ACCESS TO THE ROUTINE.
03100 ;THIS UPDATES SBITS IN CORE.
03200 STROP1: PUSH P,ACKTAB(D) ;PROTECT.
03300 SETOM ACKTAB(D)
03400 PUSH P,D ;SAVE AC.
03500 TLNN TBITS,REFRNC ;THE HARD CASE.
03600 JRST OPPP1 ;
03700 PUSH P,A ;SINCE GETOPE DOES NOT PRESEVE.
03800 HRRI FF,ADDR!INDX
03900 PUSHJ P,GETOPE ;GET THE ADDRESS OF THE BP WORD IN AN AC.
04000 ;THIS UPDATES SBITS IN CORE.
04100 SETZM ACKTAB(D) ;WE DO NOT WANT TO SEE THIS AGAIN.
04200 HRLZS D ;READY FOR INDEXING.
04300 POP P,A
04400 OPPP1: TLNE SBITS,STTEMP ;IF STACKED, THEN NEED
04500 HRLI D,RSP ;THE STACK
04600 HRRI FF,(A) ;SAVE BITS.
04700 TRNE FF,BPFIRST ;IF BYTE POINTER WORD FIRST, DO IT
04800 PUSHJ P,BP
04900 PUSHJ P,LN ;NOW THE LENGTH
05000 TRNN FF,BPFIRST
05100 PUSHJ P,BP
05200
05300 TRNE FF,UNDO
05400 TLNN SBITS,STTEMP ;IF UNDO AND A STACKED STRING.
05500 JRST OP2 ;
05600 PUSHJ P,SUBIT
05700 OP2: POP P,D ;RESTORE.
05800 POP P,ACKTAB(D)
05900 TRNE FF,REM ;IF REMOP ASKED FOR.
06000 JRST REMOP
06100 POPJ P, ;ALL DONE.
06200
06300
06400 DSCR SUBIT
06500 DES Emits a SUB SP,[XWD 2,2], and subtracts two from SDEPTH.
06600 ⊗;
06700 ↑SUBIT:
06800 ;;%DN% JFR 7-2-76
06900 ;; PUSH P,A
07000 MOVNI A,2
07100 ADDM A,SDEPTH
07200 HRLI C,-2
07300 JRST ESPADJ
07400 ;; MOVE A,X22 ;SUBTRACT TWO FROM THE STACK.
07500 ;; PUSH P,PNT
07600 ;; PUSHJ P,CREINT
07700 ;; EMIT (<SUB RSP,NOUSAC>) ;THEN ISSUE THE SUBS.
07800 ;; PUSHJ P,REMOP ;JUST IN CASE
07900 ;; POP P,PNT
08000 ;; MOVNI A,2
08100 ;; ADDM A,SDEPTH ;UPDATE COUNT.
08200 ;; POP P,A
08300 ;; JRST GETAD ;RESTORE TBITS,SBITS.
08400 ;;%DN% ↑
08500
08600 BP: TRNN FF,BPWORD ;ONLY IF ASKED FOR.
08700 POPJ P,
08800 PUSH P,A ;SAVE
08900 TRNE FF,BPINC ;IF ANOTHER INSTRUCTION AROUND.
09000 DPB C,[POINT 9,A,8] ;IN INSTRUCTION PARTS.
09100 HRRI A,NOUSAC!FXTWO ;TENTATIVE BITS TO EMITER.
09200 TLNN SBITS,STTEMP ;IF ON STACK OR
09300 TLNE TBITS,REFRNC ;BUT IF THIS CASE, THEN
09400 TRC A,FXTWO!NORLC!USX!USADDR
09500 HRLI C,0 ;WITH NO DISCPLACEMENT.
09600 PUSHJ P,EMITER
09700 POP P,A
09800 JRST FINBP
09900
10000 LN: TRNN FF,LNWORD ;ONLY IF ASKED
10100 POPJ P,
10200 HRRI A,NOUSAC
10300 TLNN SBITS,STTEMP ;IF TEMP OR
10400 TLNE TBITS,REFRNC ;REFERENCE, THEN MUST USE
10500 TRO A,NORLC!USX!USADDR ;INDEXING ETC.
10600 HRLI C,-1 ;ANO THIS TIME A DISPLACEMENT.
10700 PUSHJ P,EMITER
10800
10900 FINBP: TRNE FF,ADOP!SBOP ;PREPARE TO ADJUST STACK.
11000 TRNN FF,STAK ;ONLY IF ON STACK.
11100 POPJ P, ;NONE.
11200 TRNE FF,ADOP
11300 AOSA SDEPTH
11400 SOS SDEPTH ;OUR BOOKKEEPING DONE,
11500 POPJ P, ;WE DEPART.
11600
11700
11800 ;;%DN% JFR 7-4-76
11900 DSCR EADJSP, EPADJ, ESPADJ
12000 DES Emits instruction to alter stack depth
12100 PAR LH(C) proper constant for ADJSP
12200 RH(D) stack ac for EADJSP.
12300 RES ADJSP emitted if allowed, else proper ADD or SUB
12400 SID A, TEMP clobbered. PNT,TBITS,SBITS saved
12500 ⊗;
12600
12700 ↑EADJSP:MOVEI TEMP,(D) ;AC
12800 CAIE TEMP,RP ;FIGURE OUT WHICH STACK
12900 ↑ESPADJ:SKIPA A,[ADJSP RSP,NOUSAC!USADDR!NORLC]
13000 ↑EPADJ: MOVE A,[ADJSP RP,NOUSAC!USADDR!NORLC]
13100 MOVE TEMP,ASWITCH
13200 TRNE TEMP,AADJSP
13300 JRST EMITER ;EASY WAY
13400 PUSH P,PNT ;SAVE THIS GUY
13500 PUSH P,TBITS
13600 PUSH P,SBITS
13700 JUMPL C,.+2 ;FIGURE OUT ADD OR SUB
13800 TLCA A,(<ADJSP>≠<ADD>)
13900 TLC A,(<ADJSP>≠<SUB>)
14000 TRZ A,USADDR!NORLC
14100 PUSH P,A ;SAVE INSTR FOR LATER
14200 HLRE A,C ;COMPUTE CONSTANT
14300 MOVM A,A
14400 HRLI A,(A)
14500 PUSHJ P,CREINT ;MAKE AN XWD
14600 POP P,A ;GET INSTR BACK
14700 PUSHJ P,EMITER ;PUT OUT INSTR, PNT POINTS TO XWD
14800 POP P,SBITS
14900 POP P,TBITS
15000 POP P,PNT ;GET IT BACK
15100 POPJ P,
15200 ;;%DN% ↑
00100 COMMENT ⊗GETTEM, etc. -- Temp Semblk Allocators⊗
00200
00300 DSCR GETTEM,GETCRTMP,STRTMP
00400 DES Routines for getting temp descriptor Semblks. The list of
00500 free temps is searched for an appropriately free one. If found,
00600 a masked form of TBITS, and a masked form of SBITS are stored
00700 in the Semblk for this temp. A pointer to it is returned in LPSA
00800 INCL more descriptions about temps, their numbers, how they're
00900 moved, kept track of, deleted, depend on procedures, etc.
01000
01100 GETTEM -- get a non-core temp
01200 STRTMP -- get a String temp (i.e. turn on the STTEMP bit in SBITS)
01300 GETCRTMP -- get a core temp.
01400
01500 SID AC'S USED: USER,LPSA,TEMP
01600 ⊗;
01700
01800 STRTMP: TLOA SBITS,INUSE!STTEMP
01900 ↑GETTEM: TLO SBITS,INUSE!ARTEMP ;TURN ON TEMP BITS.
02000 ;;#NK# ! (2 OF 2) TEMPS SHOULD NOT HAVE DISPLAYS LEVELS
02100 TDZ SBITS,[CORTMP,,DLFLDM]
02200 GETBLK ;GET A NEW BLOCK
02300 GTT1: MOVEM SBITS,$SBITS(LPSA)
02400 ANDI TBITS,MASK
02500 MOVEM TBITS,$TBITS(LPSA) ;GOOD BITS IN MEMORY
02600 POPJ P, ;NOTHING ELSE TO DO
02700
02800 ↑GETCRTMP: ;GET A CORE TEMP
02900 SKIPA LPSA,TTEMP
03000 STRG: LEFT ,%RVARB,NOFF
03100 MOVE TEMP,$SBITS(LPSA)
03200 TLNE SBITS,CORTMP
03300 TLOE TEMP,INUSE
03400 JRST STRG
03500 TLNE SBITS,PTRAC
03600 JRST DDRET ;POINTERS ARE NOT LONG
03700 TDNN TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
03800 TRNN TBITS,DBLPRC
03900 JRST DDRET ;NOT DBL
04000 MOVE TEMP,$TBITS(LPSA)
04100 TDNN TEMP,[SBSCRP,,PROCED!ITEM!ITMVAR]
04200 TRNN TEMP,DBLPRC
04300 JRST STRG ;NEED DBL TEMP FOR DBL QTY
04400 DDRET: MOVSI SBITS,INUSE!CORTMP!ARTEMP
04500 JRST GTT1 ;FINISH OUT AS ABOVE.
04600
04700 NOFF: PUSHJ P,GETTEM
04800 AOS TEMP,TEMPNO ;INCREMENT TEMP ID NO
04900 MOVEM TEMP,$PNAME(LPSA) ;STORE IN $PNAME FOR ADCON AND SCOUT
05000 SETZM $ADR(LPSA) ;AND ZERO THE FIXUP.......
05100 PUSHJ P,RNGTMP
05200 JRST DDRET
05300
05400 RGC <
05500 ↑GETRCT:SKIPE SIMPSW ;SIMPLE PROCEDURE??
05600 ERR <ATTEMPT TO CREATE A RECORD TEMP INSIDE A SIMPLE PROCEDURE>,1
05700 HRRZ LPSA,RCTEMP ;GET NEXT OFF RECORD TEMP CHAIN
05800 JUMPE LPSA,GRCT.1 ;NONE THERE
05900 HRRZ TEMP,%TLINK(LPSA);
06000 MOVEM TEMP,RCTEMP
06100 POPJ P,
06200 GRCT.1: GETBLK
06300 PUSHJ P,RNGTMP
06400 AOS TEMP,TEMPNO
06500 MOVEM TEMP,$PNAME(LPSA)
06600 MOVSI TEMP,ARTEMP!INUSE!CORTMP
06700 MOVEM TEMP,$SBITS(LPSA)
06800 MOVEI TEMP,PNTVAR
06900 MOVEM TEMP,$TBITS(LPSA)
07000 POPJ P,
07100 >;RGC
07200
07300
00100 COMMENT ⊗GETAC, GETAN0 -- AC Allocators⊗
00200
00300 DSCR GETAC,GETAN0
00400 DES These are the "get a free AC routines".
00500 PAR FF(rh) -- two modifier bits:
00600 DBL -- get a double AC (i.e. next one free too)
00700 INDX -- get an indexable AC (not 0 or 1 -- 1 is avoided since
00800 Procedures tend to return values in 1).
00900 RES in D is returned the free (first free) AC number
01000 Note that no ACKTAB marking has been done yet, so the AC
01100 need not be used.
01200
01300 GETAN0: same as GETAC, but INDX is autimatically turned on.
01400
01500 AC'S USED: TEMP,LPSA
01600 ⊗;
01700
01800 ↑GETAN0: TRO FF,INDX ;HERE IF YOU DON'T WANT TO SET THE BIT
01900 ↑GETAC:
02000 HRR D,ACKPNT ;LAST AC USED
02100 SETOM ACKPNT ;CLEAR IT
02200 SETZM POSSIB ;MASK OF POSSIBILITIES
02300 MOVNI TEMP,20 ;NUMBER OF AC'S TO SEARCH
02400
02500 ;;#HF# 5-13-72 DCS RETURN OLDEST AVAILABLE AC IF NONE FREE, FIX DBL
02600 GET1: AOJG TEMP,GET7 ;For each AC, starting with the one
02700 ADDI D,1 ; after the last allocated, and wrapping
02800 TRZ D,777760 ; around to 0 (2 if GETAN0), if the AC
02900 TRNE FF,INDX ; is not protected (ACKTAB(AC)<0),
03000 TRNE D,-2 ; record the (oldest) first one seen in
03100 SKIPGE LPSA,ACKTAB(D) ; ACKPNT -- if the entry is free (0),
03200 JRST GET1 ; try to terminate. Otherwise, continue
03300 SKIPGE ACKPNT ; looking for a free one.
03400 HRRZM D,ACKPNT
03500 TRNN LPSA,-1
03600 JRST GET4
03700 JRST GET1
03800
03900 ; ONE FREE ONE EXISTS -- JUST RECORD IF DBL (NEED TWO)
04000
04100 GET4: TRNN FF,DBL ;If only one AC is needed, it's number
04200 JRST DSTORZ ; is in D.
04300
04400 GET3: MOVEI LPSA,1 ;Otherwise, record its number in the
04500 LSH LPSA,(D) ; bit array POSSIB. This is not the
04600 IORM LPSA,POSSIB ; most efficient method, but it allows
04700 JRST GET1 ; the fun below.
04800
04900 ; LIST EXHAUSTED -- TAKE WHAT WE COULD GET
05000
05100 GET7: TRNE FF,DBL ;If two were needed, we must work
05200 JRST GET9 ; harder.
05300
05400 ; TAKE A DISPLAY TEMP FIRST
05500
05600 SKIPE DISLST ;ONLY ANY GOOD IF HAVE SOME
05700 SKIPG LPSA,CDLEV ;CURRENT DISPLAY LEV
05800 JRST GET7.1
05900 HRRI D,1 ; COULD NEVER BE ZERO OR 1
06000 GET7.2: SKIPE DISTAB(D)
06100 JRST GET7.3 ;THIS THING HAS AN AC
06200 AOS D ;TRY THE NEXT ONE UP
06300 SOJG LPSA,GET7.2
06400 ERR <DRYROT AT GETAC> ;YOU REALLY BLEW IT, SAM
06500 GET7.3: MOVE LPSA,DISTAB(D) ;PICK IT UP
06600 TLNE LPSA,-1 ;USE STRING DISPLY IF WE CAN
06700 MOVSS LPSA ;US STRING -HURRAH
06800 CAIN LPSA,RF ;
06900 JRST GET7.1 ;IF RF, THEN NO GO
07000 HRR D,LPSA ;WE CAN GRAB THIS ONE
07100 SKIPG ACKTAB(D)
07200 ERR <GETAC GRABBED SAFE AC -- DRYROT AND WORMS>
07300 JRST DSTORZ ;RECORD IT, CLEAR IT OUT
07400 GET7.1:
07500
07600 ; NO DISPLAY TEMP, CLEAR SOMETHING ELSE OUT AND USE IT.
07700
07800 HRR D,ACKPNT ;Use the first one recorded, which
07900 JRST STORZ ; is also the oldest found
08000
08100 ; WE NEED TWO -- TRY FOR TWO UNUSED IN A ROW
08200
08300 GET9: MOVE LPSA,POSSIB ;If any two in a row were free,
08400 LSH LPSA,1 ; the AND of the bits and 2*bits
08500 AND LPSA,POSSIB ; will yield a bit for each pair.
08600 JUMPE LPSA,G10 ;No bits implies no pairs.
08700 FSC LPSA,231 ;The FSC shifts the first match
08800 LDB LPSA,[POINT 4,LPSA,8] ; to a normalized position, and
08900 MOVEM LPSA,ACKPNT ; records its index in the exponent
09000 HRR D,LPSA ; field.
09100 POPJ P,
09200
09300
09400 G10: HRRI D,21 ;As a last resort, take the first
09500 G11: SUBI D,2 ; two unprotected ACs available.
09600 TRNE D,777000 ;If none are found, complain bitterly.
09700 ERR <DRYROT AT DBL GETAC> ;This could be improved by
09800 SKIPL LPSA,ACKTAB(D) ; looking for the oldest pair, and/or
09900 SKIPGE ACKTAB-1(D) ; a pair with one free AC, but at
10000 JRST G11 ; this point, we're sort of beyond
10100 JUMPE LPSA,.+2 ; caring.
10200 PUSHJ P,STORZ ;Store the second, if it needs it.
10300 SUBI D,1 ;This is the result.
10400
10500 DSTORZ: HRRZM D,ACKPNT ;Allocating this one. Now go make
10600 JRST STORZ ; sure it's ready for new action.
10700 ;;#HF#
10800
00100 COMMENT ⊗AC Store routines -- BOLSTO, FORSTO, STORIX, GOSTO, STORZ⊗
00200
00300 DSCR BOLSTO
00400 DES Special Boolean store. It does not remove from ACs any
00500 of the arguments to the Boolean compare.
00600 PAR PNT and PNT2 must point to Semantics of the two arguments.
00700 RES All other ACs are stored. The Semantics of the parameters
00800 are not necessarily guaranteed over the call, since either
00900 may have been marked for storing.
01000 SEE STORZ, which it calls for each AC cleared
01100 ⊗;
01200
01300
01400 ↑BOLSTO: PUSH P,[PUSHJ P,[
01500 HRRZ TEMP,LPSA
01600 CAIE TEMP,(PNT2)
01700 CAIN TEMP,(PNT)
01800 POPJ P,
01900 JRST STORZ]] ;DO TURN OFF ACSAME FOR THESE GUYS.
02000 ; THIS STORZ IS NEEDED BECAUSE A PARTICULAR BOOLEAN MAY LOOK LIKE:
02100 ; MOVE 4,I
02200 ; SKIPN J
02300 ; JRST FOO1
02400 ; MOVE 4,J+K
02500 ; SKIPE GH
02600 ; JRST SHIT
02700 ;FOO1: ..... HERE THE COMPILER THINKS J+K IS IN 4, WHERE I MIGHT BE!!!
02800 ;
02900
03000 JRST GG0
03100
03200 DSCR FORSTO
03300 DES Special AC dumper for FOR Loops. This protects the index
03400 AC from being cleared. Other variables are not cleared, just
03500 stored if temps.
03600 PAR PNT and PNT2 should point to anything to be preserved
03700 over this operation (e.g. FOR I← <EXP> STEP .... want to preserve
03800 I and the Semantics of <EXP> from storing before the test.
03900 SEE STORA, which it calls for each AC stored.
04000 ⊗;
04100
04200 ↑FORSTO: PUSH P,[PUSHJ P,[HRRZ TEMP,ACKTAB(D) ;FOR FOR LOOPS.
04300 CAIE TEMP,(PNT)
04400 CAIN TEMP,(PNT2)
04500 POPJ P,
04600 ;DCS -- 8/16/70
04700 PUSHJ P,STORA ;STORE IT FOR SURE
04800 JUMPE LPSA,NSBSC ;NOTHING TO CLEAR
04900 ;;#MU# RHT 6-25-73 I THINK THE FOLLOWING DISTINCTION IS POINTLESS
05000 ; MOVE TEMP,$TBITS(LPSA) ;IF AN INAC ARRAY,
05100 ; TLNE TEMP,SBSCRP ;CLEAR IT, BECAUSE WILL
05200 ;;#MU#
05300 JRST CLEARL ;STILL BE ASSUMED INAC AT
05400 NSBSC: POPJ P, ; LOOP TOP OTHERWISE
05500 ]] ;DCS -- 8/16/70
05600
05700 JRST GG0
05800
05900
06000 DSCR STORIX
06100 DES "Store" all INTERNALs and EXTERNALs, i.e. forget that
06200 they are in ACs.
06300 ⊗;
06400 ↑STORIX: PUSH P, [PUSHJ P,[
06500 HRRZ LPSA,ACKTAB(D)
06600 JUMPE LPSA,CPOPJ ;NOTHING THERE.
06700 MOVE LPSA,$TBITS(LPSA)
06800 TLNE LPSA,INTRNL!EXTRNL
06900 JRST CLEARA
07000 POPJ P,]]
07100 JRST GG0
07200
07300
07400 DSCR ALLSTO
07500 DES Dump all ACs in the most permanent of ways. Do not
07600 retain any marking of the AC's at all.
07700
07800 SEE STORZ, which it calls for each AC gronked.
07900 ⊗;
08000
08100 ↑ALLSTO:OPTSYM %ALSTO
08200 PUSH P,[PUSHJ P,STORZ] ;TO CLEAR INAC" BITS.
08300 SKIPA
08400
08500 DSCR GOSTO
08600 DES Store any AC's marked with temps (as opposed to variables).
08700 Leave the AC markings as they are.
08800 Storing in forward direction makes life easier for LONG (double) things.
08900 ⊗;
09000
09100 ↑GOSTO: PUSH P,[PUSHJ P,STORA]
09200 GG0: PUSH P,D
09300 MOVSI D,-20 ;D, WHO WILL HAVE A COUNT
09400 SKIPLE LPSA,ACKTAB(D) ;DO WE HAVE A STORE TO DO?
09500 XCT -1(P) ;EXECUTE STORING ROUTINE.
09600 AOBJN D,.-2
09700
09800 ALLD: POP P,D
09900 POP P,(P) ;THROW AWAY
10000 POPJ P, ;AND RETURN
10100
10200
10300 DSCR STORZ
10400 DES "Store" this AC and wipe out the ACKTAB entry -- clear
10500 INAC-type SBITS in the Semantics which were there.
10600 PAR AC # in D
10700 SEE STORA,CLEARA routines, which it calls
10800 ⊗;
10900
11000 ↑STORZ: PUSHJ P,STORA
11100 JRST CLEARA
11200
00100 COMMENT ⊗ STORA -- main AC-storing subr. -- called by above⊗
00200
00300 DSCR STORA
00400 DES Stores temp results that are in a specified AC into
00500 a core temp. If a temp exists in that AC, an appropriate core
00600 temp is found, and the Stoe is EMITted.
00700 Then the SBITS word in the Semantics is updated to
00800 reflect the "In Core" status (e.g. CORTMP bit, fixup
00900 chain addr, etc.) The fixup chain may have originated
01000 in another temp entry, but was moved here to avoid searching
01100 up the Semantic stack for all who refer to this temp and
01200 changing the addresses of the entry they point to. WHAT????
01300
01400 PAR D contains AC # affected.
01500 SID LPSA, TEMP used
01600 ⊗;
01700
01800 ↑STORA: SKIPG LPSA,ACKTAB(D)
01900 POPJ P, ;NOTHING THERE.
02000 CAMN LPSA,ACKTAB-1(D) ;POSSIBLE DBLPRC SCREWUP?
02100 SOJA D,[PUSHJ P,STORA ;YES, STORE PRECEDING AC INSTEAD
02200 AOJA D,CPOPJ] ;AND RESTORE D
02300 PUSH P,SBITS
02400 PUSH P,TBITS ;SAVE YET ANOTHER AC
02500 MOVE SBITS,$SBITS(LPSA);GET SEMANTIC BITS.
02600 TLNN SBITS,INAC!PTRAC ;IF NOT IN AC, THEN TROUBLE
02700 ERR <STORA A THING NOT IN AC>,1
02800 ;; #KQ BY JRL (11-30-72) IGNORE FIXARS
02900 TLNN SBITS,FIXARR ;A FIXARR SHOULDN'T GET STORED
03000 TLNN SBITS,ARTEMP!DISTMP ;OTHERWISE A NOOP
03100 JRST ZER
03200 PUSH P,PNT
03300 PUSH P,A
03400 MOVEI PNT,(LPSA)
03500
03600 ;BUG TRAP
03700 HRRZ TEMP,$ACNO(PNT) ;THIS IS THE AC IT THINKS ITS IN.
03800 CAIE TEMP,(D) ;THE SAME
03900 ERR <STORA>,1
04000
04100 TLNE SBITS,DISTMP ;DISPLAY????
04200 JRST ZERDR ;YES
04300
04400 TLNE SBITS,CORTMP ;CAN WE PUT IT WHERE WE PUT IT BEFORE?
04500 JRST DEP ; YES (USUALLY ONLY HAPPENS WHEN SOME
04600 ; BUG PROVOKES IT --LIKE MISSING REMOP)
04700 RGC <
04800 TLNN SBITS,INDXED ;IF NOT INDXED TEMP
04900 JRST RCTCHK ;GO CHECK IF RECORD TEMP
05000 HRRZ TEMP,$VAL2(PNT) ;A SUBFIELD INDXED TEMP??
05100 JUMPE TEMP,NRML ;NO, JUST TREAT NORMALLY
05200 ;;#WX# ! JFR 6-5-76 FORGOT TO FETCH TBITS
05300 MOVE TBITS,$TBITS(PNT)
05400 ;;#WD# STRING SUBFIELD INDXED TEMPS ARE SPECIAL
05500 TDNN TBITS,[XWD SBSCRP,PROCED!ITEM!ITMVAR]
05600 ;;#WW# ! JFR 6-1-76 used to be TRNE (typo)
05700 TRNN TBITS,STRING
05800 JRST RCTMAK ;YES, DO THE OTHER SORT OF MOVEM
05900 JRST NRML ;HERE IF STR SUBF INDX TEMP
06000 ;;#WD# ↑
06100 RCTCHK: MOVE TBITS,$TBITS(PNT)
06200 TRNN TBITS,ITEM!ITMVAR ;THESE ARE ALWAYS NORMAL
06300 TRNN TBITS,PNTVAR ;A RECORD TEMP
06400 JRST NRML ;NOPE NORMAL
06500 RCTMAK: PUSHJ P,GETRCT ;GET A PNTVAR CORTMP
06600 JRST TMPCPY ;GO COPY FIXUPS,ETC
06700 NRML:
06800 >;RGC
06900
07000 SKIPA LPSA,TTEMP ;PREPARE TO SEARCH TEMP LIST
07100 TEML: LEFT ,%RVARB,NOFND ;GO DOWN TEMP LIST
07200 MOVE TEMP,$SBITS(LPSA)
07300 TLZE TEMP,INUSE ;NEED ONE NOT IN USE
07400 JRST TEML
07500 TLZN TEMP,CORTMP ;AND IN CORE
07600 JRST TEML ;REALLY AN ERROR
07700 TMPCPY: MOVE TEMP,$ADR(LPSA)
07800 MOVEM TEMP,$ADR(PNT) ; HO HO.
07900 MOVE TEMP,$PNAME(LPSA) ;ID NUMBER OF THIS CORTMP
08000 MOVEM TEMP,$PNAME(PNT) ;SO ADRINS AND SCOUT DON'T GET CONFUSED
08100 PUSHJ P,URGTMP ;REMOVE FROM RING
08200 FREBLK () ;THE OLD ONE
08300 JRST DEP1
08400
08500 NOFND: SETZM $ADR(PNT) ;WITH ZERO FIXUP
08600 ;; #JRL ALWAYS GIVE CORTMPS ID NO.
08700 AOS TEMP,TEMPNO ;CORTMP ID
08800 MOVEM TEMP,$PNAME(PNT)
08900 ;; #JRL
09000 DEP1: MOVE LPSA,PNT
09100 PUSHJ P,RNGTMP ;PUT ON RING
09200 DEP: MOVSI SBITS,CORTMP!INUSE!ARTEMP
09300 IORB SBITS,$SBITS(PNT) ;INDICATE THE NEW STATUS
09400 TURNOF: MOVSI LPSA,INAC!PTRAC!NEGAT ;TEMP NO LONGER IN AC
09500 ANDCAM LPSA,$SBITS(PNT)
09600 HRRM D,$ACNO(PNT) ;RECORD THE AC NUMBER
09700 HRLZI A,(<MOVEM>)
09800 TDNE TBITS,[SBSCRP,,PROCED!ITEM!ITMVAR]
09900 JRST .+3
10000 TRNE TBITS,DBLPRC
10100 MOVSI A,(<DMOVEM>)
10200 TLNE SBITS,INDXED ;A CALCULATED SUBSCRIPT?
10300 TRO A,ADDR ;YES -- DO NOT STORE INDIRECT.
10400 TLNE SBITS,NEGAT ;IS THE AC AROUND NEGATIVELY?
10500 JRST [HRLI A,(<MOVNM>) ;YES
10600 TRNE TBITS,DBLPRC
10700 HRLI A,(<DMOVNM>)
10800 JRST .+1]
10900 ;; #MD# ONLY STORE RIGHT HALF OF PTRAC
11000 TLNE SBITS,PTRAC
11100 HRLI A,(<HRRZM>) ;ONLY RIGHT HALF, IN CASE LATER AN
11200 ;INDIRECT MOVE IS DONE
11300 PUSHJ P,EMITER
11400 ;NOTE THOUGH THAT NEGAT MAY STILL
11500 ;BE ON. THIS MAY BE DANGEROUS.
11600 MOVEM SBITS,$SBITS(PNT)
11700 ZRET: POP P,A
11800 POP P,PNT
11900
12000 ZER:
12100 POP P,TBITS
12200 POP P,SBITS
12300 POPJ P, ;RETURN
12400 ZERDR: MOVE A,$VAL(PNT) ;ZEROING MASK
12500 HRR LPSA,$ADR(PNT) ;PICK UP DISPLAY LEVEL
12600 ANDM A,DISTAB(LPSA) ;ZERO APPROPRIATE SIDE OF DISTAB WORD
12700 HLLZS ACKTAB(D) ;ZONK THE ACKTAB ENTRY
12800 MOVE LPSA,PNT
12900 PUSHJ P,URGDIS ;UNLINK FROM DISPLAY VARB RING
13000 FREBLK (PNT)
13100 JRST ZRET
13200 SUBTTL CODE EMITTER
13300
00100 COMMENT ⊗EMITER -- Descriptions of Routine and Control Bits⊗
00200
00300 DSCR EMITER -- code emitting routine.
00400
00500 DES From input parameters and symbol table information,
00600 generate a word of real live code.
00700
00800 PAR
00900 A -- OPCODE in LH, bits in RH:
01000 NOUSAC←←400000 ;DON'T USE D(RH) AS AC #
01100 USCOND←←200000 ;USE C(RH) AS 3 BITS OF CONDITION
01200 USADDR←←100000 ;USE C(LH) AS DISPLACEMENT PART
01300 USX ←← 40000 ;USE D(LH) AS INDEX REG
01400 NORLC ←← 20000 ;RELOCATE NOT!
01500 IMMOVE←← 10000 ;IF OPERAND CONSTANT, LOAD IT ANY WAY POSSIBLE
01600 INDRCT←← 4000 ;INDIRECT ADDRESSING REQUIRED
01700 JSFIX ←← 2000 ;JUST DO A FIXUP (DON'T GET SEMANTICS).
01800 NOADDR←← 1000 ;NO EFFECTIVE ADDRESS PART
01900 ADDR ←← 400 ;WE WANT THE ADDRESS OF THIS ENTITY
02000 FXTWO←← 100 ;USE SECOND FIXUP WORD
02100
02200 C -- DISPLACEMENT (if provided) in LH, condition bits in RH
02300 D -- Index number in LH, AC number in RH (both optional)
02400 PNT -- symbol table pointer, if required
02500
02600 RES Code is written, RELOC bit is set to final value;
02700 Formal fixup list (FORMFX) has been updated, if necessary.
02800
02900 SID All Ac's are saved except TEMP and LPSA.
03000 ⊗;
03100
03200 BIT2DATA (EMITTER)
03300 INDIR ←← 20 ;THE INDIRECT BIT!!
03400 ;PNTROP ←← 200 ;THIS OPERATION WILL DO POINTER INDEXING
03500 ; (PURELY LOCAL BIT, BUT DON'T SEND IT IN)
03600 IMMED ←← 1000 ;THE IMMEDIATE BIT (FOR SOME THINGS).
03700
03800
03900 ↑XCALLQ:
04000 SKIPE NOEMIT
04100 POPJ P,
04200 PUSH P,C ;LITTLE ROUTINE
04300 HRL C,PCNT ;FOR CALLING LIBRARY ROUTINES.
04400 EXCH C,(A) ;FIXUP INTO LIBRARY TABLE.
04500 EMIT (<PUSHJ RP,NOUSAC!USADDR>)
04600 POP P,C
04700 POPJ P,
04800
04900
00100 COMMENT ⊗ EMITER Routine⊗
00200
00300 ↑EMITER:
00400 SKIPE NOEMIT ; GET OUT IF NO CODE IS TO BE EMITTED (I.E.
00500 POPJ P, ; EXPR!TYPE)
00600 PUSH P,A ;SAVE THOSE THINGS WHICH MIGHT CHANGE
00700 PUSH P,C
00800 PUSH P,D
00900 PUSH P,TBITS
01000 PUSH P,SBITS
01100 TRZ A,PNTROP ;ASSUME NO POINTER OP
01200 ;;# # DCS 3-25-72 Eliminate bad array address problem
01300 ;;# # When [0,0,0]-word of array (location known, no fixup) falls
01400 ;;# # on reladr 0 of .REL file, CODOUT will mistake the 0 addr field
01500 ;;# # for end of fixup chain, will inhibit RELOC -- want RELOC in this
01600 ;;# # case. A bad fix, should be more generally solved.
01700 TLO FF,RELOC!FFTMP1 ;AND RELOC (FFTMP1 FOR CODOUT 0-TEST)
01800 ;;# #
01900 TRNE A,USADDR ;ADDR IN C(LH)?
02000 JRST EAC ;YES, BYPASS SEMANTICS TESTING
02100 TLZ FF,RELOC ;NOW ASSUME NO RELOCATION
02200 HRRZS C ;CLEAR DISPLACEMENT FLD -- C(LH)
02300 TRNE A,NOADDR ;IS THERE AN ADDRESS FLD AT ALL?
02400 JRST EAC ;NO, FINISH UP
02500 TRNE A,JSFIX
02600 JRST EVAR ;GO DO A FIXUP
02700
02800 ; NOW GET SEMANTICS AND DISPATCH TO CORRECT ROUTINE TO OUTPUT INSTR
02900
03000 MOVE SBITS,$SBITS(PNT)
03100 MOVE TBITS,$TBITS(PNT)
03200 ;; #JR# BY JRL 10-17-72 A STRING ITEM IS NOT A STRING
03300 TRNE TBITS,ITEM!ITMVAR
03400 TRZ TBITS,STRING!DBLPRC ;FORGET ABOUT STRING TYPE FOR ITEMS
03500 ;; #JR#
03600 NOSBS:
03700 NOREC <
03800 TRNN TBITS,PNTVAR ;IF PNTVAR OR INDXED OR
03900 >;NOREC
04000 TLNE SBITS,INDXED ; REFERENCE FORMAL,
04100 TRO A,PNTROP ;INDICATE A POINTER OPERATION
04200 TLNE TBITS,REFRNC
04300 TRO A,PNTROP
04400 TRNE A,ADDR ;IF ADDR and PNTROP, TURN OFF BOTH
04500 TRZE A,PNTROP ;(THE IMMEDIATENESS
04600 TRZ A,ADDR ; OF ADDR CANCELS THE INDIRECTNESS OF PNTROP
04700 TLNE TBITS,SBSCRP ;ELIMINATE FXTWO IF
04800 TRZ A,FXTWO ; ARRAY NAME
04900
05000 ;;#FP# 1-10-72 DCS (1-2)
05100 TLNE SBITS,INAC ;IN ACCUMULATOR?
05200 JRST EINAC
05300 ;;#FP#
05400 TLNE TBITS,FORMAL ;FORMAL PARAMETER (ACTUAL)?
05500 JRST EFORM ;
05600 TRNE A,PNTROP ;INDIRECTNESS DESIRED?
05700 JRST EPNT
05800 ;;#FP# 1-10-72 DCS (2-2)
05900 TLNE SBITS,PTRAC ;IN ACCUMULATOR? (WAS INAC TOO)
06000 JRST EINAC
06100 ;;#FP#
06200 TRNE A,ADDR ;SHOULD WE CONSIDER CONSTANT IMMED?
06300 JRST EVAR ;NO
06400 TLNE TBITS,CNST ;NUMERIC CONSTANT?
06500 TRNE TBITS,STRING ;
06600 JRST EVAR ; NO
06700
00100
00200 ECONST: TRNN TBITS,DBLPRC ;CANT OPTIMIZE LONG PRECISION
00300 SKIPE OPDUN ;NEVER OPTIMIZE USER INLINE CODE
00400 JRST EVAR ; BUT REFER TO MEMORY
00500 MOVE TEMP,$VAL(PNT) ;GET VALUE
00600 TRNN A,IMMOVE ;IMMEDIATE MOVE REQUESTED?
00700 JRST OPCON1 ; NO, TEST LH0
00800 HRLI A,(<MOVE >) ;ASSUME MOVEI
00900
01000 TLC TEMP,-1 ;TEST LEFT HALF -1
01100 TLCN TEMP,-1 ;IS IT?
01200 JRST [HRL C,TEMP ;YES, SET UP
01300 HRLI A,(<HRROI>) ; INSTR
01400 JRST EAC] ;AND EMIT IT
01500 TRNE TEMP,-1 ;RIGHT HALF ZERO?
01600 JRST OPCON1 ; NO
01700 MOVSS TEMP ;YES, SWAP HALVES
01800 TLO A,4000 ; AND TURN ON MOVSI BIT
01900 OPCON1: TLNE TEMP,-1 ;LEFT HALF ZERO?
02000 JRST EVAR ;NO
02100 HRL C,TEMP
02200 LDB TEMP,[POINT 9,A,8] ;GET OP-CODE
02300 SUBI TEMP,200 ;ONLY OPCODES IN RANGE <MOVE> (200)
02400 JUMPL TEMP,EVAR ; TO <OR> (434) WILL
02500 CAILE TEMP,234 ; BE CONSIDERED
02600 JRST EVAR
02700 PUSH P,USER
02800 IDIVI TEMP,=36 ;WORD # TO TEMP, BIT # TO USER
02900 MOVE TEMP,OPBTS(TEMP);SOME BITS
03000
03100 TABCONDATA (OPCODE BITS TABLE FOR EMITER OPTIMIZER)
03200 OPBTS: 421042004000 ;BIT ON IF
03300 000000104000 ;CORRESPONDING OPCODE
03400 776000000000 ;CAN BE IMMEDIATE
03500 ; OLD WORD OBPTS+3 ;REPLACED (6-27-73)
03600 ; 001040000000
03700 ;; #KAB INCORRECT AND MISSING OBPTS ENTRIES
03800 000000004200
03900 401040000000
04000 ;; #KAB#
04100 ENDDATA
04200
04300 LSH TEMP,(USER) ;THE RIGHT ONE
04400 POP P,USER
04500 JUMPGE TEMP,EVAR ;CAN'T OPTIMIZE, CODE WRONG
04600 CAML A,[CAM] ;THE COMPARES ARE MADE
04700 CAML A,[JUMP] ; IMMEDIATE BY TURNING OFF
04800 TLOA A,IMMED ; THE 10000 BIT, ALL OTHERS
04900 TLZ A,10000 ; BY TURNING ON THE 1000 BIT
05000 JRST EAC ;PUT OUT OPTIMIZED INSTR
05100
05200
05300
05400 EPNT: HRRE TEMP,$VAL(PNT) ;GET DISPLACEMENT IF ANY
05500 SUBI TEMP,1 ;ASSUME STRING AND ¬FXTWO
05600 ;;#UE# (3 OF 3) INDEXED STRING ARRAY TEMPS ARE LOSERS
05700 TLNE TBITS,SBSCRP ;IF AN ARRAY
05800 AOJA TEMP,EPNT.1 ;JUST REVERSE ASSUMPTION QUAM CELERIME
05900 ;;#UE# ↑
06000 TRNN TBITS,STRING
06100 ADDI TEMP,1 ;WAS NOT STRING
06200 TRZE A,FXTWO
06300 ADDI TEMP,1 ;WAS FXTWO
06400 EPNT.1: HRL C,TEMP ;GET TO DISPLACEMENT PLACE
06500 TLNE SBITS,PTRAC ;POINTER IN AC?
06600 JRST EACX ; YES
06700 TLNE C,-1 ;MAKE INDIRECT
06800 ERR <DRYROT AT EPNT>,1 ;UNLESS WE WANTED A DISPLACEMENT
06900 TRO A,INDRCT ;MAKE IT INDIRECT
07000 JRST EVAR ;GO DO FIXUPS
07100
07200 EACX: HRL D,$ACNO(PNT) ;USE AC AS INDEX
07300 TLNE TBITS,OWN ;IF ARRAY NAME COMES INTO IT,
07400 ;;# # DCS 3-25-72 Bad array address problem.
07500 TLC FF,RELOC!FFTMP1;RELOCATABLE, SHOUDN'T 0-TEST IN CODOUT
07600 ;;# #
07700 TRO A,USX ;DENOTE THAT IT SHLD BE DONE
07800 JRST CHKIMM
07900
08000 EINAC: HRL C,$ACNO(PNT) ;INAC, GET ACNO AS DISPL.
08100 TRNE TBITS,DBLPRC ;LONG
08200 TRNN A,FXTWO ;AND FXTWO
08300 JRST CHKIMM ; SEE IF ADDR IS ON
08400 ADD C,[1,,0] ;MEANS AC+1
08500 JRST CHKIMM
08600
08700 EFORM: TRO A,USX ;WILL NEED TO USE A STACK AS INDEX
08800 HRRZ TEMP,$ADR(PNT) ;GET DISPL FROM STACK TOP
08900 TDNN TBITS,[SBSCRP!REFRNC,,PROCED!ITEM!ITMVAR] ;THESE ARE NOT VALUE LONGS
09000 TRNN TBITS,DBLPRC ;LONG?
09100 JRST .+3 ;NO
09200 TRZE A,FXTWO
09300 SUBI TEMP,1 ;FXTWO ON LONG FORMAL MOVES CLOSER TO STACK TOP
09400 TLNE TBITS,REFRNC ;REFERENCE PARAM?
09500 JRST REFPRM ; YES
09600 VALPRM: TRNN TBITS,STRING ;STRING
09700 JRST REFPRM ;NO
09800 SKIPN SIMPSW
09900 TRNN SBITS,DLFLDM ;IF SIMPLE OR DL 0 THEN DO IT THE OOLD WAY
10000 JRST USERSP
10100 LDB LPSA,[LEVPOINT(SBITS)]; PICK UP LEVEL
10200 HLL D,DISTAB(LPSA) ;PICK UP REGISTER
10300 TLNN D,17
10400 ;;#MN# 7-13-73 THE FRIDAY 13 ACCESS KLUGE
10500 JRST [
10600 PUSH P,TEMP
10700 HLRZ TEMP,LSDRLV ;MAYBE THE THING IS STILL AROUND
10800 CAIE TEMP,(LPSA)
10900 ERR <DRYROT AT EFORM FOR STRING> ;BETTER NOT BE 0
11000 HLL D,LSDRNM ;GET THE OLD THING
11100 POP P,TEMP
11200 JRST .+1]
11300 ;;# #
11400 TRZE A,FXTWO ;IF SECONG WORD
11500
00100 SUBI TEMP,1 ;FIX IT
00200 MOVN TEMP,TEMP
00300 HRL C,TEMP ;USE THIS DISPL
00400 JRST CHKIMM ;GO CHECK
00500
00600 REFPRM: TLNN TBITS,SBSCRP ;IF SUBSCRIPTED AND
00700 JRST .+3 ; REFERENCE,
00800 TLNE TBITS,REFRNC ;
00900 TRZ A,PNTROP ;DO NOT GO INDIRECT.
01000 TRZE A,PNTROP ;WANT TO GET VALUE?
01100 TRO A,INDRCT ; YES, GO INDIRECT, FIND ON RP STACK
01200 LDB LPSA,[LEVPOINT(SBITS)];PICK UP DISPLY LEVEL
01300 CAIE LPSA,0 ;IF HAVE A DISPLAY
01400 JRST USEDRF ;USE IT
01500 MOVE LPSA,TPROC ;PICK UP PROC ID
01600 HRRZ LPSA,$SBITS(LPSA);PICK UP RH OF SBITS FOR PROC
01700 ADDI LPSA,1 ;WANT LEVEL OF FORMLS
01800 XOR LPSA,SBITS ;ALL THIS IS A FANCY TEST TO SEE IF THIS PROC'S
01900 TRNE LPSA,LLFLDM ;IS IT THE SAME
02000 ERR <INACCESSABLE FORMAL> ;NO
02100 SKIPN SIMPSW ;BETTER BE SIMPLE PROC
02200 ERR <DRYROT AT EPNT -- SIMPLE?> ;YOU FUCKED UP
02300
02400
02500 USERP: HRLI D,RP ;MARK THIS STACK
02600 ADD TEMP,ADEPTH ;TOTAL ARITH STACK DEPTH
02700 JRST MAKFRM ;GO CREATE FORMAL REF INSTR
02800
02900 USERSP: HRLI D,RSP
03000 ADD TEMP,SDEPTH
03100 TRZE A,FXTWO ;SECOND WORD?
03200 SUBI TEMP,1 ;YES, DON'T GO SO FAR
03300
03400 MAKFRM: MOVNS TEMP ;NEGATIVE STACK DISPLACEMENT
03500 HRL C,TEMP ;USE THIS DISPLACEMENT
03600 ;;#KH# RHT (11-21-72) DELETED LARGE HUNKOF LEFT OVER STUFF FROM FORMFX
03700 JRST CHKIMM ;FINISH OUT
03800 USEDRF: HRL D,DISTAB(LPSA) ;PICK UP DISPLAY REGISTER
03900 TLNN D,-1 ;WAS IT LOADED
04000 ;;#MN# FRIDAY 13 JULY
04100 PUSHJ P,[DRKLUG:
04200 PUSH P,TEMP
04300 HRRZ TEMP,LSDRLV
04400 CAIE TEMP,(LPSA) ;OLD LEVEL THERE???
04500 ERR <DRYROT AT EFORM>,1;NO
04600 POP P,TEMP
04700 HRL D,LSDRNM
04800 POPJ P, ]
04900 ;;# #
05000 MOVN TEMP,TEMP ;NEGATE DISPL
05100 SUBI TEMP,1 ;SINCE RF IS ONE MORE AWAY
05200 HRL C,TEMP ;USE IT
05300 JRST CHKIMM ;GO FINISH UP
05400
05500 EVAR:
05600 TLO FF,RELOC ;NOW ASSUME RELOC AGAIN
05700 ;;#VM# ! JFR 10-30-75 PARANOIA THAT PROCEDURES COULD SLIP THROUGH
05800 TRNN TBITS,PROCED
05900 TRNE A,JSFIX ;IF JUST WANT A FIXUP
06000 JRST USECR ;THEN THATS ALL YOU GET
06100 TLNE SBITS,CORTMP ;IS IT A CORE TEMP
06200 JRST [ ;YES
06300 SKIPN RECSW ;IF NOT RECURSIVE PROC THEN
06400 JRST USECR ;USE A CORE LOCN -- NO DR NEEDED
06500 MOVE LPSA,CDLEV ;USE THIS LEVEL
06600 JRST USED.1 ;NO LDB ALLOWED
06700 ]
06800 TRNE SBITS,DLFLDM ;STACK VAR?
06900 JRST USEDR ;YES
07000 USECR:
07100 HRL C,$ADR(PNT) ;ADDR OR LAST FIXUP
07200 DCDFX: TRNN A,JSFIX
07300 TRNE TBITS,FORWRD!INPROG ;MUST FIXUP IF EITHER IS ON
07400 JRST DOFIX
07500 TLNN SBITS,FIXARR ;DON'T FIXUP IF FIXARR ON
07600 TRNE TBITS,PROCED!LABEL ;ELSE ONLY IF NEITHER OF THESE
07700 JRST DONTFX
07800 REC <
07900 TRNE TBITS,PNTVAR ;CHECK FOR CLASS ID
08000 TRNN TBITS,SHORT ; IE SHORT PNTVAR
08100 JRST DOFIX
08200 JRST DONTFX ;CLASS ID NOT FIXED UP
08300 >;REC
08400 NOREC <
08500 JRST DOFIX ;HERE DO IT
08600 >;NOREC
08700
08800 USEDR: LDB LPSA,[LEVPOINT<SBITS>] ;GET DISPLAY LEVEL
08900 USED.1: HRL D,DISTAB(LPSA) ;USE DISPLY REG
09000 TRNE TBITS,STRING ;UNLESS STRING
09100 JRST [
09200 ;#IO# RHT 7-17-72 ATTEMPT TO USE STR DR FOR A INDEXED TEMP
09300 TLNE SBITS,INDXED ;DONT IF RESULT OF ARRAY CALC
09400 JRST .+1 ;
09500 ;# #
09600 TLNN TBITS,SBSCRP ;DONT FOR ARRAYS
09700 HLL D,DISTAB(LPSA) ;CODED THIS WAY TO HANDLE USUAL CASE
09800 JRST .+1]
09900 TRNE A,USX ;BETTER NOT PLAN TO INDEX THIS
10000 ERR <DRYROT AT EVAR>,1 ;NO
10100 TLNN D,-1 ;WAS IT LOADER
10200 ;;#UV# JFR 8-16-75 WHAT A HACK.
10300 PUSHJ P,DRKLUG ;FIX RACE CONDITION. GET (ACCESS)
10400 ;FOUND THE DISPLAY REG, BUT ACCOP USED THE REG SINCE
10500 ;ALL OTHERS WERE BUSY. HACK, HACK.
10600 HRL C,$ADR(PNT) ;PICK UP DISPL
10700 TRO A,USX ;USE THE MOTHER
10800 JRST DCDFX ;GO THINK ABOUT FIXING UP
10900
11000
00100
00200 DOFIX: HRRZ TEMP,PCNT ;READY TO DO FIXUP CHAINING
00300 TRZE A,FXTWO ;USE SECOND FIXUP ADDR
00400 JRST [HLL C,$ADR(PNT)
00500 HRLM TEMP,$ADR(PNT) ;YES, MATTER OF FACT
00600 JRST CHKIMM]
00700 HRRM TEMP,$ADR(PNT) ;FINISH FIXUP CHAINING
00800
00900 DONTFX:
01000 TLNN SBITS,FIXARR
01100 JRST CHKIMM
01200 ;;#YY# JFR 2-12-77 MAKE LOGIC CLEARER AND FXTWO OF LONG DO THE RIGHT THING
01300 ;; SUB C,[XWD 1,0] ;ASSUME STRING, NOT FXTWO
01400 ;; TRNE TBITS,STRING ;IF NOT STRING OR IF FXTWO,
01500 ;; TRZE A,FXTWO
01600 ;; ADD C,[XWD 1,0] ; NULLIFY ASSUMPTION
01700 TRZE A,FXTWO
01800 ADD C,[XWD 1,0] ;FXTWO, SO 2ND WD
01900 TRNE TBITS,STRING
02000 SUB C,[XWD 1,0] ;EXCEPT STRINGS HAVE ORIGIN AT -1
02100 ;;#YY# ↑
02200 CHKIMM:
02300
02400 TRNN A,ADDR ;DO WE WANT THIS POINTER RAW?
02500 JRST EAC ; NO, FINISH UP
02600 TLO A,IMMED ;THE ONLY WAY TO DO IT HERE IS TO
02700 TRNE A,USCOND ; MAKE THE INSTR IMMEDIATE
02800 HRLI A,(<CAI>) ; (CONDITIONAL MUST BE A CAM)
02900
03000 EAC: TRNE A,INDRCT ;INDIRECT BIT WANTED?
03100 TLO A,INDIR
03200 TRNN A,NOUSAC ;AC FLD PROHIBITED?
03300 DPB D,[POINT 4,A,12] ;NO, PUT IT IN
03400 TRNE A,NORLC ;RELOCATION PROHIBITED?
03500 TLZ FF,RELOC ; YES, TAKE IT OUT
03600 TRNE A,USCOND ;CONDITION BITS NEEDED TO FINISH OPCODE
03700 DPB C,[POINT 3,A,8] ;YES, DO IT
03800 TRNE A,USX ;D(LH) TO BE USED AS INDEX FLD?
03900 TDO A,D ;YES (WIPES OUT A(RH))
04000 HLR A,C ;GET DISPL (SO DOES THIS)
04100 ;;# # DCS 3-25-72 bad array address problem
04200 MOVEI TEMP,CODOUT ;STANDARD CASE
04300 TLNN FF,FFTMP1 ;IF THIS BIT GOT TURNED OFF, CODREL SHOULD
04400 MOVEI TEMP,CODREL ; BE CALLED TO AVOID THE 0-TEST WHICH
04500 PUSHJ P,(TEMP) ; WOULD INHIBIT RELOC -- PUT OUT THE CODE
04600 ;;# #
04700 POP P,SBITS
04800 POP P,TBITS
04900 POP P,D
05000 POP P,C
05100 POP P,A
05200 ;;#MN# 7-13-73
05300 SETZM LSDRLV ;REALLY ONLY NEED TO ZERO THIS
05400 SETZM LSDRNM ;REALLY WILL DO THIS ANYHOW
05500 ;;# #
05600 POPJ P, ;RESTORE AND RETURN
05700 SUBTTL Generalized push and pop.
05800
00100 COMMENT ⊗Qstack Routines -- BPUSH, etc.⊗
00200
00300 DSCR QSTACK ROUTINES
00400 DES These are routines to provide generalized, expandable push-
00500 down stacks (buffers? queues?) for use by algorithms which need
00600 widely varying storage, accessed in simple ways. Such structures
00700 are called QSTACKS, and are built out of Semblks as follows --
00800
00900 WORD1 -- ptr to PREV,,ptr to NEXT
01000 WORDS 2-11 -- up to 10 words of "stack" data
01100
01200 A stack is identified by its QPDP, or Qstack Descriptor, which is --
01300 ptr TOP,,ptr Semblk containing TOP
01400
01500 Most Qstack operations reference the address where this QPDP (there
01600 should be one QPDP which always refers to the TOP) is stored. Others
01700 may also be used in conjunction with Qstack operations
01800
01900 Qstack operations are provided to PUSH data on, POP data off (these
02000 allocate and release Semblks, if necessary, and change the TOP QPDP),
02100 access data non-destructively in forward and reverse directions, and
02200 to clear a given Qstack.
02300 ⊗
02400
02500 DSCR BPUSH
02600 CAL PUSHJ via QPUSH macro
02700 PAR LPSA ptr to QPDP for Qstack
02800 A is data to be pushed
02900 RES QPDP is updated, A is stored in Qstack, new Semblk if necessary
03000 DES if QPDP is 0, an initial Semblk is created, QPDP constructed.
03100 SID only TEMP is changed
03200 SEE QPUSH
03300 ⊗
03400
03500 ↑BPUSH: PUSH P,A ;SAVE IT.
03600 SKIPN TEMP,(LPSA) ;THE CURRENT POINTER
03700 JRST NEWONE ;NONE YET, GUYS.
03800 HLRZ A,TEMP
03900 CAIL A,BLKLEN-1(TEMP) ;GONE OVER BLOCK BOUNDARY?
04000 JRST NOTHER ;YES
04100 PUSH1: PUSH A,(P) ;SEE !!!
04200 HRLM A,(LPSA) ;CURRENT POINTER UPDATED.
04300 POP P,A ;RESTORE
04400 POPJ P, ;DONE
04500
04600 NEWONE: PUSH P,LPSA
04700 GETBLK ;GET A NEW BLOCK.
04800 SETZM (LPSA)
04900 MOVE TEMP,LPSA ;POINTER TO NEW BLOCK.
05000 POP P,LPSA
05100 MORBLK: HRRM TEMP,(LPSA) ;UPDATE PDP POINTER.
05200 HRRZ A,TEMP
05300 JRST PUSH1 ;FINISH OUT.
05400
05500 NOTHER: PUSH P,LPSA ;SAVE IT
05600 GETBLK
05700 MOVE TEMP,LPSA ;POINTER TO NEW ONE.
05800 POP P,LPSA
05900 HRRZ A,(LPSA) ;PDP POINTER.
06000 HRLZM A,(TEMP) ;SAVE LINKS IN NEW BLOCK.
06100 HRRM TEMP,(A) ;AND IN PDP
06200 JRST MORBLK
06300
00100
00200 DSCR BPOP
00300 CAL PUSHJ via QPOP macro
00400 PAR LPSA ptr to QPDP
00500 RES A ← data from TOP, QPDP is updated
00600 DES Semblks are released as they are emptied
00700 SID only TEMP, A are changed
00800 ERR if there is no QPDP, or if no more data, error
00900 SEE QPOP
01000 ⊗
01100
01200 ↑BPOP: SKIPN TEMP,(LPSA) ;PDP POINTER
01300 ERR <DRYROT -- BPOP>
01400 HLRZ A,TEMP
01500 POPMOR: SUBI A,1 ;THIS IS A POP
01600 CAIGE A,(TEMP) ;GONE BELOW THIS BLOCK?
01700 JRST POPBAK ;YES ALAS
01800 HRLM A,(LPSA) ;UPDATE PDP
01900 MOVE A,1(A) ;THIS IS THE RESULT.
02000 POPJ P,
02100
02200 POPBAK: PUSH P,TEMP
02300 HLRZ TEMP,(TEMP) ;BACKWARD POINTER.
02400 PUSH P,TEMP
02500 FREBLK <-1(P)> ;DELETE THE BLOCK.
02600 POP P,TEMP
02700 POP P,(P) ;INGNORE THIS.
02800 SKIPN TEMP ;IS IT THERE?
02900 ERR <DRYROT -- BPOP>
03000 HLLZS (TEMP) ;ZERO FORWARD POINTER
03100 MOVEM TEMP,(LPSA) ;UPDATE PDP
03200 MOVEI A,BLKLEN-1(TEMP) ;NEW MAX.
03300 JRST POPMOR ;FINISH OUT.
03400
03500
03600 DSCR QTAK
03700 CAL PUSHJ, via QTAKE macro
03800 PAR B is QPDP for data word preceding one desired
03900 LPSA ptr QPDP for this QSTACK
04000 RES if there is more data (check via LPSA ptr):
04100 B is updated as if it were a BPUSH QPDP
04200 A receives value of TOP
04300 BTAK skips
04400
04500 if there is no more data:
04600 nothing is changed
04700 BTAK does not skip
04800 SID only A,B, TEMP changed
04900 SEE QTAKE macro
05000 ⊗
05100 ↑QTAK: CAMN B,(LPSA) ;OVERFLOW?
05200 POPJ P, ;YUP
05300 HLRZ TEMP,B
05400 CAIL TEMP,BLKLEN-1(B) ;OVERFLOW OF OTHER TYPE?
05500 JRST NEXTBL ;YES
05600 TAKMOR: MOVE A,1(TEMP)
05700 HRLI B,1(TEMP)
05800 AOS (P)
05900 POPJ P,
06000
06100 NEXTBL: HRRZ B,(B) ;GO FORWARD
06200 HRRZ TEMP,B ;NOTE THAT THE BLOCKS ARE
06300 JRST TAKMOR ;NOT DELETED !!!!!!
06400
00100
00200 DSCR BBACK
00300 CAL PUSHJ via QBACK macro
00400 PAR B contains QPDP
00500 RES B is "popped"
00600 A receives data from TOP word
00700 if there was data left, skip-returns -- else no-skip
00800 SID only A, TEMP, B changed
00900 SEE QBACK
01000 ⊗
01100 ↑↑BBACK: HLRZ A,B ;ptr to TOP, ACCORDING TO B'S QPDP
01200 BTMOR: SUBI A,1 ;TRY THE "POP"
01300 CAIGE A,(B) ;WAS THERE DATA LEFT HERE?
01400 JRST BTBAK ;NO, BACK UP
01500 HRLM A,B ;UPDATE B'S QPDP
01600 MOVE A,1(A) ;FETCH "TOP" ELEMENT
01700 AOS (P) ;SUCCESS UNLESS SOSED BY BTBAK
01800 QPOPJ: POPJ P, ;DONE
01900
02000 BTBAK: HLRZ B,(B) ;BACK UP
02100 JUMPE B,QPOPJ ; NO MORE DATA
02200 MOVEI A,BLKLEN-1(B) ;RESET LH PTR
02300 JRST BTMOR ;FINISH UP
02400
02500 DSCR BFLUSH
02600 CAL PUSHJ, via QFLUSH macro
02700 PAR LPSA ptr to QPDP
02800 RES all Semblks cleared, QPDP zeroed
02900 SID A, B, TEMP changed
03000 SEE QFLUSH
03100 ⊗
03200 ↑↑BFLUSH: SKIPN A,(LPSA)
03300 POPJ P, ;NO STACK
03400 FLSHLP: HLRZ B,(A) ;GET NEXT PTR
03500 FREBLK (A) ;RELEASE TOP SEMBLK
03600 MOVE A,B
03700 JUMPN A,FLSHLP ;MAKE NEXT ONE BACK TOP ONE
03800 SETZM (LPSA) ;ALL DONE
03900 POPJ P,
04000
04100 DSCR BBEG
04200 CAL PUSHJ, via QBEGIN macro
04300 PAR B is QPDP
04400 RES B is QPDP which, when BTAKEd, returns first element in Qstack
04500 B is 0 if no Qstack exists
04600 SID only B, TEMP changed
04700 SEE QBEGIN
04800 ⊗
04900 ↑↑BBEG: SKIPN B,(LPSA) ;IS THERE A STACK?
05000 POPJ P, ; NO
05100 LOPPP: HRLS B ;MAKE INIT QPDP FOR THIS SEMBLK
05200 HLRZ TEMP,(B) ;GET BACK PTR
05300 JUMPE TEMP,CPOPJ ;WHEN HAVE REACHED FIRST SEMBLK, QUIT
05400 MOVE B,TEMP ;TRY AGAIN
05500 JRST LOPPP
05600
00100 COMMENT ⊗PWR2⊗
00200
00300 DSCR PWR2
00400 DES Tests number in register B for being a power of 2.
00500 if so, it skip-returns (********) and C
00600 has a small integer representing the power.
00700
00800 SID AC'S: uses TEMP
00900 ⊗;
01000 ↑PWR2: JUMPLE B,CPOPJ ;ROUTINE TO TEST B FOR A POWER OF TWO.
01100 MOVN TEMP,B ;TWO'S COMPLEMENT.
01200 AND TEMP,B ;AND THE AND
01300 TLNN B,777000 ;TOO BIG ?
01400 CAME TEMP,B ;THE MAGIC TEST FOR POWER OF TWO.
01500 POPJ P, ;NO DICE.
01600 FSC B,233 ;NOW THE NORMALIZE.
01700 ASHC B,-=44 ;NOW CORRECTLY IN C. (LEFT HALF)
01800 SUB C,[XWD 201,400000]
01900 AOS (P)
02000 POPJ P,
02100
02200
02300 SUBTTL Generator Output Routines.
02400
00100 COMMENT ⊗GBOUT Description, Loader Block Format Description⊗
00200
00300 DSCR GBOUT -- write a block of binary output
00400 DES
00500 One of the specialized output routines has produced
00600 a loader block, ready for output. These
00700 routines are:
00800
00900 CODOUT -- prepares a code block. Each call
01000 puts a word of code into a buffer and sets relocation
01100 appropriately.
01200
01300 FBOUT -- prepares a fixup block. Each call puts a fixup word into
01400 a buffer.
01500
01600 SOUT -- for outputting symbols. Each call puts a symbol
01700 name (in RADIX50) and an address into a buffer.
01800
01900 Other parts of the generators also call GBOUT for special functions
02000 (entry block, prog name block, etc). The routines
02100 call GBOUT when their buffers are full or when they
02200 wish to force out all of a given block.
02300
02400 Each block outputted by GBOUT has the same general format:
02500 WD1: BLOCK TYPE,,COUNT
02600 0 LEQ COUNT (WDn-WD3+1) LEQ 18
02700 WD2: relocation bits
02800 18 2-bit bytes (left-justified) corresponding
02900 to the 18 (maximum) data words in the block.
03000 The first bit of each is on if the left
03100 half is to be relocated. The second bit
03200 of each corresponds to the right half
03300 of its data word.
03400 WD3: first data word
03500 .
03600 .
03700 .
03800 WDn: last data word 2 LEQ n LEQ 20
03900
04000 The Binary file is opened and initialized in the command
04100 scanner (outer block of SAIL). The FF bit BINARY
04200 is on if a binary output is desired (if the file is open).
04300
04400 PAR B -- SIZE,,address of loader block
04500 SIZE is size of ENTIRE block (2 + WD1's count)
04600 It is zero if WD1's COUNT is to be believed.
04700
04800 RES The block is written if SIZE is GEQ 3
04900
05000 SID All ACS are preserved
05100 ⊗;
05200
00100 COMMENT ⊗ Control Variables for Loader Block Output⊗
00200
00300 ZERODATA (REL-FILE OUTPUT VARIABLES)
00400
00500 ;CODPNT -- bp for relocation bits in BINTAB CODE block
00600 ; see GBOUT for details about relocation bits -- initted to --
00700 ?CODPNT: POINT 2,BINTAB+1
00800
00900 ;FRSTSW -- off until first word of code goes out -- used to
01000 ; trigger output of program name block, initial code, etc.
01100 ; in CODOUT -- set on in CODOUt
01200 ?FRSTSW: 0
01300
01400 ;FXPNT -- reloc bits bp for FXTAB FIXUP block -- see FBOUT, GBOUT
01500 ?FXPNT: POINT 2,FXTAB+1
01600
01700 ;LSTRAD, LSTRLC, LSTWRD -- last radix50 word output, last code
01800 ; word output, last relocation bits output -- used by Boolean
01900 ; and ALLOT code, for repeating some of it
02000 ↑↑LSTRAD: 0
02100 ↑↑LSTRLC: 0
02200 ↑↑LSTWRD: 0
02300
02400 ;OUTADR -- bp set up by GBOUT for fetching words from LODBLKs
02500 ; for transfer to output buffer
02600 ?OUTADR: 0
02700
02800 ;RAD5. -- RADIX50 creates a value corresponding to a symbol comprising
02900 ; the first 5 characters of the identifier, followed by ".", in
03000 ; addition to each value it creates. It is saved here, used sometimes.
03100 ↑↑RAD5.: 0
03200 ↑↑RAD5$: 0 ;SIMILAR, BUT WITH A $
03300 ↑↑RAD5%: 0 ;GUESS WHAT
03400 ;SMPNT -- reloc bits pb for SMTAB SYMBOLS block -- see SCOUT, GBOUT
03500 ?SMPNT: 0
03600
03700 DATA (REL-FILE OUTPUT VARIABLES)
03800
03900 ;SALIB -- used to place main SAIL library request in LBTAB output
04000 ; loader block -- see DONES, PRGOUT
04100 ;SALIH -- re-entrant version of library
04200
04300 ↑SALIB: LIBLEN ;STRING CONSTANT, LIBLEN LONG
04400 ;;#HX# 6-24-72 DCS PARAMETERIZE LIBRARY NAMES
04500 POINT 7,[LIBLOW]
04600 REN <
04700 ↑SALIBH:LIBLEN
04800 POINT 7,[LIBHI]
04900 ;;#HX#
05000 >;REN
05100
05200 BAIL<
05300 ↑BAIREL: BALENG ;STRING CONSTANT, BALENG LONG
05400 POINT 7,[BAILOD]
05500 ↑BAIPD: BPDALN ;STRING CONSTANT
05600 POINT 7,[BAIPDS]
05700 >;BAIL
05800
00100 COMMENT ⊗ Loader Output Blocks-- Entry, Program Name, Initial Stuff⊗
00200
00300 DATA (LOADER OUTPUT BLOCKS)
00400 COMMENT ⊗
00500 Here are the loader output blocks. They are formatted as described
00600 in SAILON ;;.; by Bill Weiher. The general routine GBOUT handles
00700 the actual output of these (filled) blocks to the .REL file. For
00800 several of the block types, special routines exist below (CODOUT,
00900 FBOUT, etc.) to place individual words (and their relocation) into
01000 the blocks, and to call GBOUT when a block is full
01100 ⊗
01200
01300
01400 COMMENT ⊗
01500 ENTTAB -- ENTRY block -- names included in SAIL ENTRY statements.
01600 This must be the first block out (due both to syntax and
01700 necessity. It allows the .REL file to be used as part
01800 of a library.
01900 ⊗
02000 LODBLK (ENTRY,4,ENTTAB,,=18)
02100
02200
02300 COMMENT ⊗
02400 PROGNAM -- PROGRAM NAME BLOCK -- output of this block is delayed until
02500 first word of code goes out, to give user longest possible time
02600 to come up with a program name. Must go out before code to name
02700 outer block symbols and labels and stuff.
02800 ⊗
02900 ;;%CL% JFR 7-22-75 IDENTIFY OURSELVES TO LINK-10
03000 LODBLK (PROGNAM,6,BEGNAM,BEGCNT,2)
03100 RELOC .-2
03200 ↑↑PRGTTL: RADIX50 0,M ;DEFAULT NAME, IF NO OTHER COMES
03300 XWD 7,0 ;7 means SAIL, bits 0-5 tell hardware assumptions
03400 ;;%CL% ↑
03500
03600 COMMENT ⊗
03700 HBLK -- High Segment Block -- Denotes Re-entrant Output
03800 ⊗
03900 REN <
04000 LODBLK (HIGH,3,HBLK,HBLK2,1,,<XWD 200000,0>)
04100 RELOC .-1
04200 XWD 400000,400000 ;TWOSEG
04300 >;REN
04400
04500
04600
04700 COMMENT ⊗
04800 BEGOUT -- STANDARD INITIAL CODE SEQUENCE
04900 This code is always put out, but is only executed (and fixups
05000 are only correct) for Main Programs. Sample fixed-up code is
05100 included in the comments
05200 ⊗
05300
05400
05500 LODBLK (CODE,1,BEGOUT,BEGCT2,10,,<XWD 200000,0>)
05600 RELOC .-10
05700
05800 ↑↑BEGPC:0 ;PC ALWAYS 0 OR 400000
05900 SKIPA ;NOT STARTED IN RPG
06000 SETOM ;RPGSW
06100 JSR ;SAILOR
06200 ;;%AL% THE HRLOI IS NOW DONE BY SAILOR
06300 ;; HRLOI RF,1 ;FOR FIRST LINK
06400 PUSH P,RF
06500 PUSH P, ;[PDA,,0]
06600 PUSH P,SP
06700 HRRI RF,-2(P); SET F
06800
06900
07000
00100 COMMENT ⊗ Code, Boolean Code, Fixups, Links⊗
00200
00300 COMMENT ⊗
00400 BINTAB -- MAIN CODE BLOCK
00500 All generated instructions are output via CODOUT-GBOUT
00600 to this block. See CODOUT for details
00700 ⊗
00800 LODBLK (CODE,1,BINTAB,,=18)
00900
01000
01100 COMMENT ⊗
01200 BOLOUT -- SPECIAL BOOLEAN CODE BLOCK
01300 Conditionals are output once when a condition is seen, and
01400 again (with fixups and compare op codes correct) when the
01500 entire Boolean expression has been parsed and analyzed.
01600 See BOOLEAN for details.
01700 ⊗
01800 LODBLK (CODE,1,BOLOUT,,0,,<XWD 200000,0>)
01900 ↑↑BRELC←.-1 ;TO ACCESS RELOCATION BITS
02000 ↑↑BPCNT: 0 ;PROGRAM COUNTER -- SAME AS WHEN INSTRS FIRST OUT
02100 ↑↑BWRD1: 0 ;COMPARE, SKIP, OR CONDITIONAL JUMP
02200 ↑↑BWRD2: 0 ;UNCONDITIONAL JUMP IF BWRD1 WAS A COMPARE OR SKIP
02300 ↑↑BWRD3: 0
02400
02500
02600 COMMENT ⊗
02700 FXTAB -- FIXUPS
02800 Each word contains in its right half the address or stack
02900 displacement (reloc bits adj. accordingly) of a variable
03000 or instruction. The left half contains the address
03100 (relative to 0, of course) of the last instruction or data
03200 which requires this address field. This location, in turn,
03300 was compiled to refer to the next previous use of the variable
03400 or whatever... in other words, a fixup chain (terminates in 0).
03500 The LOADER uses these fixups to handle forward references to
03600 things. See FBOUT for details
03700 ⊗
03800 LODBLK (FIXUPS,10,FXTAB,,=18,-1)
03900
04000
04100 COMMENT ⊗
04200 SMTAB -- SYMBOLS
04300 All local and internal symbols, and global requests, are output
04400 through this block. See SCOUT and friends for details.
04500 ⊗
04600 LODBLK (SYMBOLS,2,SMTAB,,=18,<XWD 42104,210421>)
04700 ;(RELOCATE EVERY OTHER WORD -- GENERALLY)
04800
04900
05000 COMMENT ⊗
05100 SLNKBK -- LINK BLOCKS
05200 The string link, space link, and other links are output
05300 through this block. These links provide inter-RELfile
05400 communication (best example is link that chains all string
05500 variables together, so that STRNGC can get at them. See
05600 LNKOUT for details.
05700 ⊗
05800 LODBLK (LINK,12,SLNKBK,SDSCRP,2,,<XWD 40000,0>)
05900 RELOC .-2
06000 ↑↑LNKNM: 1 ;USUALLY STRING LINK, BY CONVENTION #1
06100 ;SPACE LINK IS #2
06200 ;SET LINK IS #3
06300 ;STRNGC ROUTINE NAMES LINK IS #4
06400 ; THESE ARE SAIL CONVENTIONS ONLY
06500 ↑↑SLNKWD: 0 ;ADDRESS OF ELEMENT OF CHAIN
06600
00100 COMMENT ⊗ Space Allocation Block
00200
00300 SBCTBL -- SPACE ALLOCATION BLOCK
00400 In this block is collected all REQUIRE specifications
00500 (except LOAD!MODULES, LIBRARIES, SOURCE!FILES) and
00600 space limits (string space, system pdl, new items, etc.)
00700 It is output as a code block. Also output is a link
00800 block tying this space block to all the others loaded
00900 together. The SAILOR (initialization) routine uses this
01000 information to provide an environment pleasing to the user.
01100 See DONES and the REQUIRE code for more details. Also GOGOL
01200 (%ALLOC) for block format explanations
01300 ⊗
01400 ;;%BR% RHT ALLOW COMVER THING FOR EVERYONE (BUT KEEP EXPO FOR NOW)
01500 ↑↑SPCSIZ←←=17 ;$SPREQ+1 ;IF EVER MAKE 18 OR MORE, MUST CHANGE SOME THINGS
01600 ;;↑↑SPCSIZ←←=14 ;BAD OLD VALUE *****
01700 ;;%BR% ↑
01800
01900 ↑↑SPCTBL:XWD 1,SPCSIZ ;CODE BLOCK, AT LEAST SPCSIZ LONG
02000 BYTE (2) 1,0,0,0,0,1,0,0,0,0,0,0,1,1,0,1
02100 ;PC WORD,MESLNK,TINIT,PINIT,OBPDA(RELOC)
02200 ↑SPCPC: 0 ;PC LOCATION
02300 0 ;LINK BLOCK PROVIDES CHAIN THROUGH THIS LOC
02400 ;; %AG% HAVE ITEMNO KEEP BOTH MIN AND MAX
02500 ↑ITEMNO:0 ;MIN,,MAX ITEM NUMBER DECLARED THIS COMPILATION
02600 ↑NWITM: 0 ;REQUIRE n NEW!ITEMS PUTS n HERE
02700 ;; %AG% ! HAVE GITEMNO CONTAIN LEAPIS FLAG
02800 ↑GITEMNO:0 ;XWD LEAPIS,MAX (MIN?) GLOBAL ITEM NUMBER DECLARED
02900 ↑MESLNK:0 ;POINTER TO MESSAGE PROCEDURE LIST PUT HERE
03000 ↑PNAMNO:0 ;REQUIRE n PNAMES PUTS n HERE
03100 ↑VERNO: 0 ;REQUIRE n VERSION PUTS n HERE
03200 ↑SEGNAM:0 ;REQUIRE "name" SEGMENT!NAME PUTS "name" HERE IN SIXBIT
03300 ↑SEGDEV:0 ;REQUIRE "dev:file[p,pn]" SEGMENT!FILE PUTS
03400 ↑SEGFIL:0 ; dev, file, ppn IN THESE LOCS IN SIXBIT
03500 ↑SEGPPN:0 ;(LOW BIT OF DEV IS SEGMENT PROTECT BIT, NOT USED NOW)
03600 ↑TINIT: 0 ;INITIALIZATION BLOCK ADDRESS FOR DECLARED ITEM TYPES
03700 ↑PINIT: 0 ;INIT. BLOCK FOR PNAMES(DECLARED ITEMS)
03800 ;;%BR%
03900 ↑↑COMVER: 0 ;NICE THING, BUT SUAI
04000 ;;%BV% !
04100 ↑↑OBPDA:0 ;OUTER BLOCK PDA
04200 0 ;SPARE
04300 ;;%BR% ↑
04400 BLOCK 50 ;ROOM FOR MORE REQUESTS
04500 ↑SPCEND←←.-1
04600
04700
04800
00100 COMMENT ⊗ Request Blocks -- RELfile, Libraries⊗
00200
00300 COMMENT ⊗
00400 PRGTAB -- RELFILE REQUEST BLOCK
00500 REQUIRE "...." LOAD!MODULE generates one of these. The LOADER
00600 loads all requested .REL files after loading all the explicit
00700 stuff. See REQUIRE code for details
00800 ⊗
00900 ;; #KS# ADD LOADVR SWITCH
01000 IFN (LOADVR-=54), <
01100 LODBLK (RELREQ,15,PRGTAB,,=18)
01200 >
01300 IFE (LOADVR-=54), <
01400 LODBLK (RELREQ,16,PRGTAB,,=18)
01500 >
01600 ;; #KS#
01700
01800 COMMENT ⊗
01900 LBTAB -- LIBRARY REQUEST BLOCK
02000 REQUIRE "...." LIBRARY generates one of these (SAIL main programs
02100 automatically request SYS:LIBSAI.REL). The LOADER searches these
02200 libraries, if necessary, after searching all the others except the
02300 automatic F4 search.
02400 ⊗
02500
02600 ;; #KS# LOADVR SWITCH
02700 IFN (LOADVR-=54), <
02800 LODBLK (LIBREQ,16,LBTAB,,=18)
02900 >
03000 IFE (LOADVR-=54), <
03100 LODBLK (LIBREQ,17,LBTAB,,=18)
03200 >
03300 ;; #KS#
03400
03500
00100 COMMENT ⊗ Ending Code, Symbols -- END Block
00200
00300 STAROT ETC. -- ENDING STUFF.
00400 These include some constant ending code, some extra standard
00500 symbols, the starting address block, if there is one, and so on.
00600 It's too messy to use the LODBLK macro on, so here it is in
00700 all its glory--
00800 ⊗
00900 EBLEN←←. ;COLLECT LENGTH.
01000
01100 ;If this is a Main Program, a starting address block is issued
01200 ; (via the GBOUT descriptor EBDSC); else EBDSC1 is used to issue
01300 ; all but the starting address block. Starting address is always
01400 ; relative 0 (addr of the BEGOUT code--see above)
01500 ?STAROT: XWD 7,1 ;STARTING ADDR BLOCK -- 1 DATA WORD
01600 XWD 200000,0 ;RELOCATE ADDRESS (RH)
01700 ↑STRDDR:0 ;STARTING ADDRESS ALWAYS REL 0
01800
01900 ; If Main Program, global requests must be issued to fill in
02000 ; the RPGSW and SAILOR blanks in the BEGOUT block (above)
02100 XWD 2,4 ;SYMBOL BLOCK
02200 XWD 42104,210421 ;EVERY OTHER WORD.
02300 ↑CONSYM:RADIX50 60,SAILOR;JSR REQUEST.
02400 2 ;JSR IS IN LOC 2
02500 RADIX50 60,RPGSW;FOR SETOM RPGSW BUSINESS
02600 1 ;SETOM IS IN 1
02700
02800 ; This part is always issued -- standard symbol names, end block
02900 NOSTAR: XWD 2,STRCT-NOSTAR-2;SYMBOLS
03000 XWD 40000,0;RELOCATE ONLY S.
03100 RADIX50 10,S. ;FIRST EXECUTABLE LOC IN PROG
03200 0 ;ALWAYS 0
03300 RADIX50 10,P ;SYSTEM PDP ADDR
03400 RP ;USUALLY 17
03500 RADIX50 10,SP ;STRING PDP ADDR
03600 RSP ;USUALLY 16
03700 RADIX50 10,ARERR;UUO FOR ARRAY INDEX OV/UNDERFLOW
03800 ARERR ;THE UUO OPCODE
03900 RADIX50 10,FLOAT;UUO FOR INTEGER to REAL
04000 FLOAT
04100 RADIX50 10,FIX ;UUO FOR REAL to INTEGER
04200 FIX
04300 RADIX50 10,SNGL ;UUO FOR LONG REAL to REAL
04400 SNGL
04500 STRCT: ;END OF EXTRA SYMBOLS
04600
04700 ; END BLOCK
04800 NOREN <
04900 XWD 5,1 ;END BLOCK.
05000 XWD 200000,0 ;RELOCATE PROGRAM BREAK WORD
05100 ↑↑PRGBRK: 0 ;PROGRAM BREAK-- FIRST NON-USED ADDR
05200 >;NOREN
05300 REN <
05400 XWD 5,2 ;TWO PROGRAM BREAKS
05500 XWD 240000,0 ;RELOCATE PROGRAM BREAK WORD
05600 ↑↑PRGBRK: 0 ;HIGH-SEG PROGRAM BREAK
05700 0 ;LOW-SEG PROGRAM BREAK
05800 >;REN
05900
06000 EBLEN←← .-EBLEN ;LENGTH OF ENTIRE OUTPUT RITUAL
06100
06200 ↑EBDSC: XWD EBLEN,STAROT ;IF MAIN PROGRAM
06300 ↑EBDSC1:XWD EBLEN+STAROT-NOSTAR,NOSTAR ;IF NOT
06400 ENDDATA
06500
00100 COMMENT ⊗ RELINI -- Loader Block Initialization⊗
00200
00300 DSCR RELINI
00400 CAL PUSHJ FROM GENINI
00500 DES SETS UP ALL REL-FILE OUTPUT STUFF BEFORE EACH COMPILATION
00600 ⊗
00700
00800 ↑↑RELINI:
00900 HLLZS BINTAB
01000 HLLZS FXTAB
01100 SETOM FXTAB+1 ;ALL RELOCATABLE
01200 HLLZS SMTAB ;CLEARS OUTPUT BUFFER COUNTS
01300 HLLZS PRGTAB ;PROGRAM AND LIBRARY REQUEST BLOCKS
01400 HLLZS LBTAB
01500 MOVE A,[XWD SPCPC,SPCPC+1] ;CLEAR SPACE ALLOCATION BLOCK
01600 SETZM SPCPC
01700 BLT A,SPCEND ;SIZE ALLOCATION BLOCK.
01800 HRRI TEMP,SPCSIZ
01900 HRRM TEMP,SPCTBL
02000 POPJ P, ;RETURN TO GENINI
02100
00100 COMMENT ⊗ GBOUT Routine⊗
00200
00300 ↑GBOUT:
00400 PUSH P,A ;SAVE A
00500 PUSH P,B ;SAVE ADDRESS OF BUFFER
00600 HLRZ A,B ;GET COUNT IF NONSTANDARD
00700
00800 TLO FF,IREGCT ;SET NON-STANDARD COUNT BIT
00900 HRLI B,(<POINT 36,0>) ;FOR PICKING UP WORDS
01000 MOVEM B,OUTADR ;SAVE TABLE ADDRESS
01100 JUMPN A,GBOUTA ;NOT STANDARD (FROM TABLE) COUNT
01200 HRRZ A,(B) ;GET COUNT FROM BLOCK
01300 ;;#TR# BE MORE HONEST ON COMPUTING THIS
01400 ; ADDI A,2 ; +2 FOR BLOCK TYPE & RELOC
01500 ADDI A,=35 ; CNT ← CNT+1+FLOOR((A+17)/18)
01600 IDIVI A,=18
01700 HRRZ B,@OUTADR ;WORD CNT AGAIN
01800 ADD A,B ; CORRECT VALUE (I HOPE)
01900
02000 TLZ FF,IREGCT ;RESET NON-STANDARD COUNT BIT
02100
02200 ; OUTPUT ROUTINE
02300
02400 GBOUTA: TLNN FF,BINARY ;IS THERE A BINARY FILE?
02500 JRST OUTDUN ;NO, DON'T WRITE
02600 CAIGE A,3 ;IS THERE ANYTHING TO WRITE?
02700 JRST OUTDUN ;NO, DON'T DO IT
02800
02900 NOTENX <
03000 BQN: SOSLE BINCNT ;FULL?
03100 JRST OKOUT ;NO
03200 OUTPUT BIN,0 ;EMPTY BUFFER, ON TO NEXT
03300 TSTERR BIN ;ERRORS?
03400 ERR <OUTPUT ERROR ON BINARY FILE>
03500
03600 OKOUT: ILDB B,OUTADR ;BLOCK WORD
03700 IDPB B,BINPNT
03800 SOJG A,BQN ;WRITE THEM ALL
03900 >;NOTENX
04000 TENX <
04100 PUSH P,C
04200 MOVNI C,(A)
04300 MOVE B,OUTADR
04400 SKIPL A,BINJFN ;JUST IN CASE IT'S -1 (DUMMY)
04500 JSYS SOUT
04600 MOVEM B,OUTADR ;UPDATE OUTADR
04700 POP P,C
04800 >;TENX
04900
05000 OUTDUN: POP P,B ;GET BUFFER ADDR BACK
05100 TLZN FF,IREGCT ;DON-'T CLEAR IF NON-STANDARD COUNT
05200 HLLZS (B) ;CLEAR COUNT
05300 POP P,A ;RESTORE A
05400 POPJ P,
05500
00100 COMMENT ⊗ CODOUT Routine -- Output Code or Data⊗
00200
00300 DSCR CODOUT -- WRITE DATA (ALSO CODREL)
00400
00500 PAR WORD IN "A"
00600 relocatable if RELOC in in "FF"
00700 (if rh of A is zero, then never RELOC. If you want to
00800 TO BYPASS THIS TEST, CALL "CODREL").
00900
01000 RES Writes word, increments program counter (PCNT)
01100
01200 SID Uses A,B,C -- Saves all
01300 ⊗;
01400
01500 ↑CODOUT:
01600 SKIPE NOEMIT ; GET OUT IF NO CODE IS TO BE EMITTED (I.E.
01700 POPJ P, ; EXPR!TYPE)
01800 PUSH P,A
01900 PUSH P,B
02000
02100 SKIPE FRSTSW ;HAVE WE DONE THIS BEFORE
02200 JRST COD1 ; YES, DON'T DO AGAIN
02300 SETOM FRSTSW
02400 PUSH P,LPSA ;AND SOME OTHERS
02500 MOVEI LPSA,IPROC ;GET PROGRAM NAME.
02600 PUSHJ P,RAD50 ;IN RADIX50
02700 TLZ A,740000 ;RADIX50 0,NAME
02800 MOVEM A,PRGTTL
02900 MOVE B,BEGCNT
03000 PUSHJ P,GBOUT ;WRITE NAME BLOCK
03100 REN <
03200 MOVEI A,0
03300 SKIPN HISW ;TWO-SEGMENT PROGRAM?
03400 JRST JUST1 ;NO
03500 MOVE B,HBLK2 ;YES, WRITE HISEG (TYPE 3) BLOCK
03600 PUSHJ P,GBOUT
03700 MOVEI A,400000 ;BEGINNING PC
03800 JUST1:
03900 MOVEM A,BEGPC ;IN WHICH SEGMENT
04000 >;REN
04100 MOVE B,BEGCT2 ;CALL TO INIT & LINKAGE
04200 PUSHJ P,GBOUT
04300 COD2: POP P,LPSA
04400 MOVE A,-1(P) ;RESTORE A.
04500
04600 COD1: TRNN A,-1 ;ZERO ADDRESS?
04700 TLZ FF,RELOC ;YES, NO RELOC
04800 JRST CDRL1
04900 ↑CODREL:
05000 PUSH P,A ;ENTER HERE TO BYPASS ZERO TEST
05100 PUSH P,B
05200 CDRL1:
05300 HRRZ B,BINTAB ;GET COUNT
05400 JUMPN B,BAQ ;FIRST WORD OF BLOCK?
05500
05600 AOS BINTAB ;YES, SET UP BLOCK
05700 MOVE B,PCNT ;SET LOCATION WORD
05800 MOVEM B,BINTAB+2 ;INTO 3D WORD OF BLOCK
05900 SETZM BINTAB+1 ;CLEAR RELOCATION BITS
06000 MOVE B,[POINT 2,BINTAB+1] ;BYTE POINTER FOR RELOC BITS
06100 MOVEM B,CODPNT ;TO RIGHT PLACE
06200 MOVEI B,1 ;RELOCATE THE LOC COUNTER WORD
06300 IDPB B,CODPNT
06400
06500 BAQ: AOS B,BINTAB ;INCREMENT COUNT
06600 HRRZS B ;AND MOVE TO B
06700 MOVEM A,BINTAB+1(B) ;DEPOSIT WORD
06800 MOVEM A,LSTWRD ;SAVE LAST WORD OUTPUT
06900 LDB A,[POINT 1,FF,RLCPOS] ;RELOC?
07000 SKIPE LHRELC ;RELOC LEFT HALF?
07100 ADDI A,2 ;SAY SO
07200 MOVEM A,LSTRLC ;AND LAST RELOCATION BIT.
07300 IDPB A,CODPNT ;SET RELOC BITS
07400
07500 AOS PCNT ;INCREMENT COUNT
07600
07700 CAIGE B,22 ;FULL?
07800 JRST CDRET ;NO, RETURN
07900
08000 MOVEI B,BINTAB ;INDICATE STANDARD COUNT AND WHICH TABLE
08100 PUSHJ P,GBOUT ;WRITE BLOCK
08200 ; JRST CDRET
08300
08400 CDRET: POP P,B
08500 POP P,A
08600 POPJ P,
08700
08800 ↑CODLRL: ;RELOCATE LEFT HALF -- FF SAYS ABOUT RIGHT HALF
08900 TLNE A,-1 ;NEVER RELOCATE 0
09000 SETOM LHRELC ;SET FLAG
09100 PUSHJ P,CODOUT
09200 SETZM LHRELC
09300 POPJ P,
09400
09500 ZERODATA( DISPLAY STUFF)
09600 LHRELC: 0
09700 ENDDATA
09800
09900
10000
00100
00200 DSCR FRBT
00300 DES Force out current binary (BINTAB) code block,
00400 even if it's not full yet. This is done whenever
00500 symbols or fixups which might refer to this code
00600 are put out, so that there is something to fixup
00700 or refer to symbolically. It is also called from DONES.
00800 SID Saves all ACS
00900 ⊗
01000
01100 ↑FRBT: PUSH P,B
01200 MOVEI B,BINTAB
01300 PUSHJ P,GBOUT ;CLEAR BINARY BUFFER
01400 POP P,B
01500 POPJ P,
01600
01700
00100 COMMENT ⊗ FBOUT, etc. -- Output Fixups⊗
00200
00300 DSCR FBOUT,FIXOUT,FBOSWP
00400 DES Put word of fixup information into output file.
00500 PAR B contains fixup specification:
00600 lh -- PCNT of actual location of entity
00700 rh -- PCNT of last word in fixup chain.
00800 FBOSWP takes the above B value, swapped.
00900 RES This word is written into the FXTAB fixup Loader
01000 block via GBOUT (when there are enough).
01100 FBOUT always assumes both halves reloatable
01200 FIXOUT always assumes the actual (lh) address is not
01300 relocatable
01400 FBOSWP is included for convenience
01500 SID Saves all ACs
01600 ⊗;
01700
01800 ↑FXOSW2: MOVSS B
01900 PUSHJ P,FIXOUT
02000 MOVSS B
02100 POPJ P,
02200 ↑FBOSW2: MOVSS B
02300 PUSHJ P,FBOUT
02400 MOVSS B
02500 POPJ P,
02600
02700 ↑FBOSWP: MOVSS B
02800 ↑FBOUT: SKIPE NOEMIT ; GET OUT IF NO CODE IS TO BE EMITTED (I.E.
02900 POPJ P, ; EXPR!TYPE)
03000 TLNN B,-1 ;IS LEFT HALF ZERO?
03100 ERR <DRYROT -- FBOUT>,1
03200 TLOA FF,FFTEMP ;USE RELOCATION IN FIXUP SIDE
03300 ↑FIXOUT:
03400 TLZ FF,FFTEMP ;DO NOT RELOCATE FIXUP PART
03500 PUSH P,B
03600 PUSH P,A ;SAVE A
03700 HRRZ A,FXTAB
03800 JUMPN A,FAQ ;FIRST WORD OF BLOCK?
03900 MOVE A,[POINT 2,FXTAB+1] ;YES, RESET RELOCATION BIT POINTER
04000 MOVEM A,FXPNT ; (SEE CODOUT FOR SIMILARITIES)
04100 FAQ:
04200 AOS A,FXTAB ;INCREMENT AND FETCH COUNT
04300 HRRZS A
04400 MOVEM B,FXTAB+1(A) ;DEPOSIT WORD
04500 MOVEI B,3 ;ASSUME BOTH HALVES RELOC
04600 TLNN FF,FFTEMP ;TEST ASSUMPTION
04700 MOVEI B,2 ; WRONG
04800 IDPB B,FXPNT ;INSERT RELOCATION BITS
04900
05000 CAIGE A,22 ;FULL?
05100 JRST FXRET ;NO, RETURN
05200
05300 PUSHJ P,FRBT ;FORCE OUT ANY BINARY
05400 ;(BECAUSE FIXUPS HAVE TO COME AFTER)
05500
05600 MOVEI B,FXTAB
05700 PUSHJ P,GBOUT ;WRITE BLOCK
05800
05900 FXRET: POP P,A
06000 POP P,B
06100 POPJ P,
06200
06300
06400
00100 COMMENT ⊗ SCOUT, etc. -- Output Symbols⊗
00200
00300 DSCR SOUT,SCOUT,SHOUT,SCOUT0
00400 DES Output symbols in RADIX50 -- many ways exist for
00500 obtaining symbols for output, thus the proliferation.
00600
00700 PAR
00800 SOUT: LPSA -- Semantics ptr. $PNAME and $ADR are used to
00900 obtain the symbol and address.
01000 SHOUT: LPSA -- descriptor of the form:
01100 bits 0-5 DDT symbol type
01200 6-17 #characters
01300 18-35 address of string in ASCII (assumed justified)
01400 B -- address for symbol
01500 SCOUT: A -- RADIX50 for symbol
01600 B -- address for symbol
01700 SCOUT0: SAME AS SCOUT, BUT MAKES SYMBOL NON-RELOCATABLE.
01800
01900 SID A, TEMP, may be different on exit
02000 ⊗;
02100
02200 ↑SHOUT: PUSHJ P,RAD52
02300 JRST SCOUT ;MAKE RADIX50 FROM DESCRIPTOR
02400
02500 ↑SCOUT0: PUSH P,B ;NON-RELOCATED SYMBOL
02600 MOVEI TEMP,0
02700 JRST SASS
02800
02900
03000 ↑SOUT: PUSHJ P,RAD50 ;GET RADIX50 FOR SYMBOL
03100 PUSH P,B ;SAVE IT
03200 ;;# # RHT 3-19-73 MAKE RECSV SYMBOLS GO OUT UNRELOC
03300 HRRZ B,$ADR(LPSA) ;GET ADDRESS
03400 MOVE TEMP,$SBITS(LPSA);DOES THIS SYMBOL USE THE STACK?
03500 TRNN TEMP,DLFLDM ;
03600 JRST SOUT.0 ;NO
03700 CAIGE B,20 ;HALF KILL??
03800 TLO A,400000 ;YES
03900 MOVEI TEMP,0 ;
04000 JRST SASS
04100 ;;# # RHT
04200
04300 ;;# # RHT -- 7-13-73 EXTRA KLUGE TO USE THE RAD5$ SYMBOL FOR
04400 ;; SPECIAL BILTIN RUNTIMES
04500 SOUT.0: SETCM TEMP,$TBITS(LPSA)
04600 TLNE TEMP,FNYNAM+OWN+EXTRNL
04700 JRST SOUT.1 ;REGULAR
04800 SKIPA A,RAD5$ ;PREMIUM
04900 ;;# #
05000 ↑SCOUT: PUSH P,B ;SAVE
05100 SOUT.1: MOVEI TEMP,1 ;RELOCATION BIT.
05200 SASS: PUSH P,C
05300 HRRZ C,SMTAB
05400 JUMPN C,SAQ
05500 MOVE C,[POINT 4,SMTAB+1]
05600 MOVEM C,SMPNT
05700 SAQ:
05800 CAMN A,LSTRAD ;RADIX50 FOR LAST BLOCK NAME.
05900 JRST SYMRET ;DO NOT PUT IT OUT.
06000 AOS C,SMTAB ;BINARY DOES NOT HAVE TO BE
06100 HRRZS B ;FORCED OUT
06200 MOVEM A,SMTAB+1(C)
06300 MOVEM B,SMTAB+2(C)
06400 AOS C,SMTAB
06500 HRRZS C
06600 LDB B,[POINT 4,A,3] ;DON'T RELOCATE BLOCK LEVELS
06700 CAIN B,3 ;BLOCK TYPE 14
06800 MOVEI TEMP,0
06900 IDPB TEMP,SMPNT
07000 CAIGE C,22
07100 JRST SYMRET
07200
07300 PUSHJ P,FRBT ;MAKE BINARY GO FIRST
07400 MOVEI B,SMTAB
07500 PUSHJ P,GBOUT
07600
07700 SYMRET: POP P,C
07800 POP P,B
07900 POPJ P,
08000
00100 COMMENT ⊗ LNKOUT -- Output Linkage Block⊗
00200
00300 DSCR LNKOUT --
00400 DES Put out a (type 12) Link block via GBOUT. These blocks
00500 allow chains of addresses to be created through separate
00600 .REL files. STRINGC uses LINK 1 to find all its strings.
00700 Other uses are for SETS, STRINGC routine names, and the
00800 space allocation block.
00900 PAR B -- link number
01000 PCNT -- decremented by one; that is address for LINK rqst.
01100 ⊗
01200
01300 ↑LNKOUT: MOVEM B,LNKNM ;SAVE LINK NUMBER
01400 PUSHJ P,FRBT ;NOTE DOES NOT SAVE ACS
01500 HRRZ TEMP,PCNT
01600 SUBI TEMP,1 ;LAST WORD OUTPUT WILL HOLD LINK
01700 HRRZM TEMP,SLNKWD ;PLACE IN ADDR WORD OF LINK BLOCK TEMPLATE
01800 MOVE B,SDSCRP ;DESCRIPTOR OF LINK BLOCK [COUNT,ADDR OF TEMPLATE]
01900 PUSHJ P,GBOUT
02000 POPJ P, ;RETURN AFTER WRITING BLOCK
02100
00100 COMMENT ⊗ PRGOUT, FILSCN -- Output Request Blocks, Scan for Source!file Rqst⊗
00200
00300 DSCR FILSCN -- CONVERT ASCII FILE-STRING TO SIXBIT
00400 PAR PNAME, PNAME+1 describe a String representing the file
00500 name.
00600 RES A, C, D return DEVICE, FILENAME, and PPN in SIXBIT
00700 DES Converts String to SIXBIT via FILNAM routine (approp-
00800 riately informed) in Command Scanner (SAIL). Extension
00900 not returned, because there's currenlty no need.
01000 SID Nothing much saved
01100 SEE FILNAM, PRGOUT, RQSET, SRCSWT
01200 ⊗
01300 NOTENX <
01400 ↑↑FILSCN: SETOM TYICORE ;TYI IN COMND WILL GET CHARS FRM STRNG
01500 PUSH P,DEVICE ;SAVE FILE DATA
01600 PUSH P,EXTEN
01700 PUSH P,SAVTYI
01800 PUSH P,EOL
01900 SETZM SAVTYI ;NO SCAN-AHEAD
02000 MOVSI TEMP,(<SIXBIT /DSK/>) ;DEFAULT DEVICE
02100 MOVEM TEMP,DEVICE
02200 PUSHJ P,FILNAM ;GET SIXBITS IN NAME, EXTEN, ETC.
02300 MOVE A,DEVICE ;LOAD RESULTS
02400 MOVE C,NAME
02500 MOVE D,PPN
02600 POP P,EOL
02700 POP P,SAVTYI
02800 POP P,EXTEN
02900 POP P,DEVICE ;RESTORE OLD VALUES
03000 POPJ P,
03100 >;NOTENX
03200
03300 TENX <
03400 TFLSCN:
03500 BEGIN TFLSCN
03600
03700 CTRLV←←"V"-100 ;TENEX QUOTING CHARACTER
03800 FIND←←D
03900
04000 SETZM FIND
04100 PUSH SP,PNAME ;ORIGINAL NAME -- COPY ONTO STACK
04200 PUSH SP,PNAME+1
04300 PUSH SP,[0] ;DEVICE TEMPORARY
04400 PUSH SP,[0]
04500 PUSH SP,[0] ;DIR TEMPORARY
04600 PUSH SP,[0]
04700 PUSH SP,[0] ;NAM TEMPORARY
04800 PUSH SP,[0]
04900
05000 DEFINE ORIG <-7(SP)>
05100 DEFINE ORIG1 <-6(SP)>
05200 DEFINE DEV <-5(SP)>
05300 DEFINE DEV1 <-4(SP)>
05400 DEFINE DIR <-3(SP)>
05500 DEFINE DIR1 <-2(SP)>
05600 DEFINE NAM <-1(SP)>
05700 DEFINE NAM1 <0(SP)>
05800
05900 ;SIMPLE SINCE NAME IS AT THE TOP OF SP
06000 DEFINE CATNAM (X) <
06100 PUSH P,X
06200 PUSHJ P,CATCHR
06300 >
06400 DEFINE CATDIR (X) <
06500 PUSH P,X
06600 PUSH SP,DIR
06700 PUSH SP,DIR
06800 PUSHJ P,CATCHR
06900 POP SP,-4(SP)
07000 POP SP,-4(SP)
07100 >
07200
07300 DEFINE GCH <
07400 HRRZ A,ORIG
07500 JUMPE A,TENDUN
07600 ILDB C,ORIG1
07700 SOS ORIG
07800 >
07900
08000 TENX1: GCH
08100 CAIE C,CTRLV
08200 JRST NOQUOTE
08300 SKIPE FIND
08400 JRST QUODIR
08500 PUSHJ P,CATNA3
08600 GCH
08700 PUSHJ P,CATNA3 ;AND THE CHAR FOLLOWING THE CTRLV
08800 JRST TENX1
08900 QUODIR: PUSHJ P,CATDI3
09000 GCH
09100 PUSHJ P,CATDI3
09200 JRST TENX1 ;AND CONTINUE
09300
09400 NOQUOTE:
09500 CAIN C,":" ;COLON -- DEVICE
09600 JRST ISDEV ;ITS BEEN A DEVICE ALL ALONG!!
09700 CAIN C,","
09800 JRST TENX1 ;IGNORE COMMA
09900 CAIE C,40 ;SPACE
10000 CAIN C,11 ;OR TAB
10100 JRST TENX1
10200 CAIE C,"<" ;THESE START THE DIRECTORY NAME
10300 CAIN C,"["
10400 JRST STTDIR
10500 CAIE C,">" ;THESE FINISH THE DIR. NAME
10600 CAIN C,"]"
10700 JRST ENDDIR
10800 SKIPE FIND ;DOING DIRECTORY?
10900 JRST .+3 ;YES
11000 PUSHJ P,CATNA3
11100 JRST TENX1
11200 PUSHJ P,CATDI3
11300 JRST TENX1
11400
11500 STTDIR: SETOM FIND
11600 JRST TENX1
11700
11800 ENDDIR: SETZM FIND
11900 JRST TENX1
12000
12100 ;;#SK# 5-30-74 RLS DONT MESS UP DEVICE NAME IF PRESENT
12200 ISDEV:
12300 MOVE C,NAM ;THE "NAME" HAS REALLY BEEN A DEV
12400 MOVEM C,DEV
12500 MOVE C,NAM1
12600 MOVEM C,DEV1
12700
12800 SETZM NAM ;SO CLEAR THE NAME -- START OVER
12900 SETZM NAM1
13000 JRST TENX1
13100
13200 TENDUN:
13300 ;NOW STACK HAS ORIG,DEV,DIR,NAM
13400 GOTDIR:
13500 ;NOW FIND ONLY THE NAME -- IGNORE EXTENSION, VERSION, ETC.
13600 PUSH SP,[0] ;NEW TEMPORARY
13700 PUSH SP,[0]
13800 NAMLUP: HRRZ A,-3(SP)
13900 SOS -3(SP) ;DECREMENT
14000 JUMPE A,GOTDI1
14100 ILDB C,-2(SP)
14200 CAIE C,"." ;QUIT ON PERIOD
14300 CAIN C,";" ;OR SEMICOLON
14400 JRST GOTDI1
14500 PUSH P,C
14600 PUSHJ P,CATCHR
14700 JRST NAMLUP
14800 GOTDI1: POP SP,-2(SP) ;REMOVE TEMPORARY
14900 POP SP,-2(SP)
15000 HRRZ A,-1(SP) ;CHECK LENGTH OF NAME
15100 CAILE A,6
15200 ERR <Name too long for loader.>,1
15300 PUSHJ P,CVSIX ;GET SIXBIT FOR NAME
15400 MOVEM A,C ;INTO C
15500
15600 PUSHJ P,DIRPPN ;TRANSLATE DIRECTORY STRING TO PPN
15700 MOVEM A,D ;PLACE PPN IN D
15800
15900 HRRZ A,-1(SP) ;NOW DO THE DEVICE
16000 CAILE A,6
16100 ERR <Device name too long for loader.>,1
16200 PUSHJ P,CVSIX ;SIXBIT FOR DEVICE INTO A
16300 SKIPN A ;ANYTHING THERE?
16400 MOVE A,[SIXBIT/DSK/] ;ASSUME DEVICE DSK
16500 SUB SP,X22 ;CLEAR OFF COPY OF PNAME
16600 POPJ P,
16700
16800 ;CALL CAT MACROS WITH AC C AS THE ARG
16900 CATNA3: CATNAM C
17000 POPJ P,
17100
17200 CATDI3: CATDIR C
17300 POPJ P,
17400
17500
17600 DIRPPN:
17700 ; DIRPPN -- CONVERT ASCII DIRECTORY NAME TO PPN (NEEDED FOR THE LOADER)
17800 ; PAR STRING DEVICE-NAME, DIRECTORY-NAME (ON SP STACK)
17900 ; (DEVICE-NAME IS NOT REMOVED)
18000 ; RES A: PPN
18100 ; SID SAME AS SIXBIT (EVERYTHING BUT A)
18200 HRRZ A,-1(SP) ;IF DIRECTORY NAME LENGTH = 0
18300 JUMPE A,DIRP.X ; RETURN(0)-ASSUME CONNECTED DIR
18400 IFN SIXSW,< ;USE SIXBIT(DIRNAME) AS PPN
18500 CAILE A,6 ;VERIFY THAT NAME FITS IN SIXBIT
18600 ERR <DIRECTORY TOO LONG FOR LOADER.>,1
18700 JRST CVSIX ; RETURN( CVSIX(ARG) )
18800 >;IFN SIXSW
18900 IFE SIXSW,< ;NOT SIXBIT, MORE FINAGGLING NECESSARY
19000 ; Modifications made to support TOPS20 v. 3
19100 ; Robert Smith
19200 ; Rutgers University
19300 ; March 12, 1978
19400 ; DEC has blown it! They removed the STDIR jsys, and
19500 ;changed the conventions in the emulator about directory-to-ppn
19600 ;conversion. However, there is a new JSYS, STPPN, that will
19700 ;now suffice.
19800 ; Algorithm for the following code:
19900 ;IF STDEV("NIL") fails, THEN
20000 ; BEGIN "assume TOPS20"
20100 ; AC[D] ← STPPN("<" & directory " ">" & 0);
20200 ; END ELSE
20300 ; BEGIN "assume TENEX"
20400 ; rh of AC[D] ← STDIR(directory & 0);
20500 ; END;
20600 BEGIN DIRNAME
20650 EXTERN CHRCAT
20700 PUSH P,B ;SAVE B
20800 HRROI A,[ASCIZ/NIL/]
20900 JSYS STDEV ;IS IT TENEX?
21000 JRST TOPS20 ;ERROR RETURN--MEANS TOPS20
21100 PUSH P,[0] ;MAKE STRING ASCIZ BY APPENDING
21200 PUSHJ P,CATCHR ;NULL BYTE TO END OF STRING
21300 MOVE B,(SP) ;BP TO ASCIZ
21400 MOVEI A,1 ;POSITIVE -- NO RECOGNIZE
21500 JSYS STDIR ;TWO ERROR RETURNS (+1 AND +2)
21600 JFCL ;HANDLE IDENTICALLY
21700 JRST [ERR <This directory does not exist on this system.>,1
21800 SETZ A, ;RESULT 0(CONNECTED DIRECTORY)
21900 JRST CLNUP ;AND CLEANUP
22000 ]
22100 HRLI A,4 ;4,,DIRNO
22200 JRST CLNUP ;CLEANUP STACK AND RETURN
22300
22400 ;HERE WITH TOPS20
22500 ;FIRST BUILD A STRUCTURE NAME
22600 ;SP STACK:
22700 ; DEVICE-NAME
22800 ; DIRECTORY-NAME
22900 TOPS20: ;
23000 ;FIRST SURROUND DIRECTORY-NAME WITH "<>" PAIR
23100 PUSH P,[74] ;LEFT BROKET
23200 PUSHJ P,CHRCAT
23300 PUSH P,[76] ;RIGHT BROKET
23400 PUSHJ P,CATCHR
23500 ;NOW SEE IF THERE IS A DEVICE NAME
23600 HRRZ B,-3(SP) ;IF NOT LENGTH(DEVICE-NAME)
23700 JUMPE B,NODEVN ;THEN GOTO NODEVN
23800
23900 PUSH SP,-3(SP) ;DEVICE-NAME
24000 PUSH SP,-3(SP)
24100 PUSH P,[":"] ;WITH A COLON
24200 PUSHJ P,CATCHR
24300 PUSH SP,-3(SP) ;DIRECTORY-NAME
24400 PUSH SP,-3(SP)
24500 PUSHJ P,CAT ;
24600 POP SP,-2(SP) ;CLOBBERS DIRECTORY-NAME ON
24700 POP SP,-2(SP) ;STACK
24800
24900 NODEVN:
25000 PUSH P,[0] ;PREPARE FOR STPPN
25100 PUSHJ P,CATCHR ;AND PUT A NULL ON THE END ;
25200 MOVE A,0(SP) ;STRING POINTER
25300 OPDEF STPPN [104000000556]
25400 STPPN ;PPN IN REG A
25450 MOVEM B,A ;FOR RETURN
25500 CLNUP: POP P,B ;RESTORE B
25600 BEND DIRNAME
25700 >;IFE SIXSW
25800 DIRP.X: SUB SP,X22 ;REMOVE ARGUMENT FROM SP
25900 POPJ P, ;AND LEAVE
26000
26100 BEND TFLSCN
26200 >;TENX
00100
00200
00300 DSCR PRGOUT -- OUTPUT PROGRAM AND LIBRARY REQUEST BLOCKS
00400 DES Output (via GBOUT) Program and Libraray REQUEST BLOCKS.
00500 PAR B ptr to PRGTAB or LBTAB (program or library request)
00600 PNAME, PNAME+1 as in FILSCN
00700 Defaults as in FILSCN; DEVICE, FILE and PPN will be passed
00800 to the loader.
00900 RES FILSCN is called to make SIXBIT representations of DEVICE,
01000 FILE, and PPN; these are placed in the output block.
01100 SID Saves the world
01200 ⊗;
01300
01400 ↑↑PRGOUT:
01500 NOTENX<
01600 MOVE USER,GOGTAB ;SAVE ACS IN USER TABLE AREA
01700 HRRZI TEMP,RACS(USER)
01800 BLT TEMP,SBITS2+RACS(USER) ;FILNAME USES MANY ACS
01900 PUSHJ P,FILSCN ;GET SIXBITS IN A,C,D
02000 MOVE B,RACS+2(USER) ;GET TABLE ADDRESS BACK
02100 MOVEI TEMP,3 ;PREPARE TO COUNT UP BLOCK COUNT
02200 ADDB TEMP,(B)
02300 ADDI TEMP,(B) ;ptr to AREAS TO BE FILLED
02400 MOVEM C,-1(TEMP) ;STORE NAME
02500 ;;=I10= SFD PATCH - BE SURE WE HAVE A REAL PPN
02600 SFDS<
02700 JUMPE D,.+3 ;ZERO PPN IS OK
02800 TLNN D,777777 ;SO IS A REAL PPN
02900 ;;JOHN - POSSIBLE YOU WANT TO ISSUE AN ERROR MESSAGE HERE INSTEAD
03000 MOVE D,2(D) ;IF PATH PTR, USE PPN FROM PATH
03100 > ;SFDS
03200 MOVEM D,00(TEMP) ;STORE PPN
03300 MOVEM A,01(TEMP) ;STORE DEVICE
03400 TYMSHR<
03500 JUMPE D,PRGOU2 ;IF NO PPN
03600 TLNN D,-1 ;OR IF REAL PPN
03700 CAME A,[SIXBIT /DSK/] ;OR NOT DISK
03800 JRST PRGOU2
03900 MOVE C,AVLSRC
04000 JFFO C,.+2
04100 JRST PRGOU2
04200 CAILE D,17
04300 JRST PRGOU2 ;FIND CHANNEL
04400 PUSH P,B
04500 MOVSI A,(<RELEASE>)
04600 DPB D,[POINT 4,A,12]
04700 PUSH P,A
04800 MOVE A,[LOOKUP A]
04900 DPB D,[POINT 4,A,12]
05000 PUSH P,A
05100 MOVE A,[OPEN B]
05200 DPB D,[POINT 4,A,12]
05300 MOVEI B,16
05400 MOVSI C,'DSK'
05500 MOVEI D,0
05600 XCT A
05700 JRST PRGOU3
05800 MOVEI A,3 ;NOW LOOKUP
05900 MOVE B,(TEMP)
06000 MOVE C,-1(TEMP)
06100 MOVEI D,0
06200 XCT (P)
06300 JFCL
06400 MOVEM B,(TEMP) ;SAVE PPN
06500 PRGOU3: POP P,A
06600 POP P,A ;THE RELEASE
06700 XCT A
06800 POP P,B
06900 PRGOU2:>
07000
00100 >;NOTENX
00200 TENX<
00300 PUSH P,A ;MUST PUSH SINCE TFLSCN CALLS RUNTIMES
00400 PUSH P,C
00500 PUSH P,D
00600 EXCH SP,STPSAV ;GET A STRING STACK
00700 PUSHJ P,TFLSCN ;DOES NOT MODIFY B
00800 EXCH SP,STPSAV ;RESTORE IT
00900 MOVEI TEMP,3
01000 ADDB TEMP,(B)
01100 ADDI TEMP,(B) ;ptr to AREAS to be filled
01200 MOVEM C,-1(TEMP) ;STORE NAME
01300 MOVEM D,00(TEMP) ;STORE PPN
01400 MOVEM A,01(TEMP) ;STORE DEVICE
01500 POP P,D
01600 POP P,C
01700 POP P,A ;RESTORE
01800 PUSH P,TEMP
01900 MOVE USER,GOGTAB ;SAVE FOR KROCK BELOW
02000 HRRZI TEMP,RACS(USER)
02100 BLT TEMP,SBITS2+RACS(USER)
02200 POP P,TEMP
02300 >;TENX
02400 HRRZS TEMP
02500 CAIL TEMP,22(B) ;BLOCK FULL?
02600 PUSHJ P,GBOUT ;YES, PUT IT OUT
02700 HRLZI TEMP,RACS(USER)
02800 BLT TEMP,SBITS2
02900 POPJ P, ;TRA 0,4?
03000
03100 SUBTTL Generator Miscellaneous.
03200
00100 COMMENT ⊗ RAD50, RAD52 -- Radix-50 Functions for Scout Routines⊗
00200
00300 DSCR RAD50,RAD52 -- create a RADIX50 symbol
00400 PAR RAD50 -- LPSA pntr to block head -- string is in $PNAME, etc.
00500 RAD52 -- LPSA(lh) is count, LPSA (rh) is address of string,
00600 assumed aligned.
00700 RES RADIX50 for symbol in A
00800 SID Results in A, all other ACS saved (except TEMP)
00900 ⊗;
01000
01100 ↑RAD50:
01200 EXCH SP,STPSAV
01300 MOVSS POVTAB+6 ;ENABLE FOR STRING PDL OV
01400 PUSH SP,$PNAME(LPSA) ;COLLECT POINTERS IN COMMON SPOT
01500 PUSH SP,$PNAME+1(LPSA)
01600 HRRZS -1(SP) ;CLEAR STRNO, SAVE COUNT
01700 MOVE A,$TBITS(LPSA) ;CONTROLS MODE BITS IN RAD50 SYMBOL
01800 MOVEI TEMP,10/4 ;ASSUME LOCAL
01900 TLNE A,INTRNL ;INTERNAL IS TYPE 4
02000 MOVEI TEMP,4/4
02100 TLNE A,EXTRNL
02200 MOVEI TEMP,60/4 ;EXTERNAL IS TYPE 60
02300 MOVEI A,0 ;INITIALIZE A
02400 JRST RAD5
02500
02600
02700 ↑RAD52:
02800 LDB TEMP,[POINT 12,LPSA,17] ;COUNT
02900 EXCH SP,STPSAV
03000 MOVSS POVTAB+6 ;ENABLE FOR STRING PDLOV
03100 PUSH SP,TEMP
03200 PUSH SP,LPSA ;MAKE IT LOOK LIKE STRING
03300 HRRI TEMP,(<POINT 7,0>) ; DESCRIPTOR
03400 HRLM TEMP,(SP)
03500 MOVEI A,0
03600 LDB TEMP,[POINT 4,LPSA,3]
03700
03800 RAD5: PUSH P,TEMP
03900 PUSH P,B ;SAVE IT
04000 MOVEI TEMP,6
04100
04200 R50LUP: SOSGE -1(SP) ;QUIT IF NO MORE STRING
04300 JRST R5OUT
04400 ILDB B,(SP) ;CHARACTER
04500 CAIN B," " ;IGNORE BLANKS ABSOLUTELY!
04600 JRST R50LUP ; THIS RUNS ALL THE CHARACTERS TOGETHER
04700 CAIL B,"a"
04800 CAILE B,"z"
04900 JRST .+2
05000 SUBI B,40 ;CONVERT TO UPPER CASE
05100 CAIE B,30 ;UNDERLINE:THESE CHARS HAVE TO BE CREATED INDIVIDUALLY
05200 CAIN B,"."
05300 MOVEI B,66+45 ;RAD50 CHAR FOR "." + 66 TO BE SUBTRACTED
05400 ;;#GQ# DCS 2-8-72 (1-1) ! SAME AS UNDERLINE
05500 CAIN B,"!" ;! SAME AS UNDERLINE
05600 MOVEI B,66+45 ;"."
05700 ;;#GQ# (1)
05800 CAIN B,"$"
05900 MOVEI B,66+46
06000 CAIN B,"%"
06100 MOVEI B,66+47
06200 SUBI B,66 ;OK IF A LETTER
06300 CAIG B,12 ;<12 IF A NUMBER
06400 ADDI B,7 ; THIS MAKES IT RIGHT
06500 IMULI A,50 ;THAT'S THE NUMBER ALL RIGHT
06600 ADD A,B ;COLLECT RADIX50
06700 SOJN TEMP,R50LUP ;QUIT AT 6
06800
06900 R5OUT: MOVEM A,RAD5. ;NOW CREATE SAME SYMBOL WITH
07000 JUMPLE TEMP,MORFIV ;MORE THAN FIVE CHARS?
07100 IMULI A,50 ;MAKE IT "SYMB".
07200 SKIPA
07300 MORFIV: SUB A,B ;"." IN PLACE OF THE LAST
07400 POP P,B ;RESTORE B
07500 POP P,TEMP ;TYPE BITS.
07600 DPB TEMP,[POINT 4,A,3] ;TYPE BITS TO RADIX 50
07700 ADDI A,46 ;$
07800 MOVEM A,RAD5$
07900 ADDI A,1 ;%
08000 MOVEM A,RAD5% ;
08100 SUBI A,2 ;"."
08200 EXCH A,RAD5. ; AND STORE IT IN RAD5. FOR STRINGS
08300 DPB TEMP,[POINT 4,A,3] ;TYPE BITS TO RADIX 50
08400 SUB SP,X22
08500 EXCH SP,STPSAV ;RESTORE REGS
08600 MOVSS POVTAB+6 ;RE-ENABLE FOR PARSE PDLOV
08700 POPJ P,
08800
08900 BEND TOTAL
09000 IFN FTDEBUG, <↑INNA←INNA>
09100
09200
09300
00100
00200