Trailing-Edge
-
PDP-10 Archives
-
AP-D480B-SB_1978
-
error0.bli
There are 12 other files named error0.bli in the archive. Click here to see a list.
!THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY ONLY BE USED
! OR COPIED IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE.
!COPYRIGHT (C) 1976,1977 BY DIGITAL EQUIPMENT CORPORATION
MODULE IOERR=
BEGIN
%(
THE CHANNEL TABLE
)%
MACRO TABMAX=16$, TABSIZ=15$; % MAX CHANNELS AND SIZE %
STRUCTURE CHTB[I,J]=[(I+1)*(J+1)]((.CHTB+.I*(J+1)+.J)<0,36>);
EXTERNAL CHTB CHNLTAB[TABMAX,TABSIZ];
MACRO BUFSIZ=100$;
MACRO UNPACKIT(X,Y)=REPLACEI(X,SCANI(Y)+#40)$;
STRUCTURE FUNNYBIT[I]=((.FUNNYBIT+(.I/36))<.I MOD 36,1>);
EXTERNAL FUNNYBIT ZIGNORE;
%(
C H A N N E L T A B L E M A C R O S
)%
MACRO FMODE=CHAN[0]$,
FDEV=CHAN[1]$,
FBUF=CHAN[2]$,
FBUFO=CHAN[2]<18,18>$,
FBUFI=CHAN[2]<0,18>$,
FOBUFH=CHAN[3]$,
FOPTR=CHAN[4]$,
FOCNT=CHAN[5]$,
FIBUFH=CHAN[6]$,
FIPTR=CHAN[7]$,
FICNT=CHAN[8]$,
FFILE=CHAN[9]$,
FEXT=CHAN[10]$,
FERR=CHAN[10]<0,12>$,
FBLK=CHAN[10]<0,18>$, % DEVICE BLK NO DSK=0 %
FPROT=CHAN[11]<27,9>$,
FDMODE=CHAN[11]<23,4>$,
FTIME=CHAN[11]<13,10>$,
FDATE=CHAN[11]<0,12>$,
FPPN=CHAN[12]$,
FPRJ=CHAN[12]<18,18>$,
FPRG=CHAN[12]<0,18>$,
FSIZ=CHAN[12]<18,18>$,
FSTATUS=CHAN[13]$,
FSWITCH=CHAN[14]$,
FNEWPR=CHAN[15]<0,10>$,
%(
STATUS WORD BITS
)%
FYEND=0,1$, % ENDFILE ALLOWED %
FYREW=1,1$, % REWIND ALLOWED %
FYREAD=2,1$, % READING ALLOWED %
FYWRITE=3,1$, % WRITING ALLOWED %
FYUPDAT=4,1$, % UPDATING ALLOWED %
FYDUMMY=5,1$, % DUMMY CHANNEL %
FYERR=6,1$, % WRITE ERRORS NOT TRAPPED %
FPRIMARY=7,1$, % PRIMARY DEFAULT VALUE %
% SYSTEM BITS OF STATUS WORD %
FROPEN=18,1$, % FILE OPEN FOR INPUT %
FWOPEN=19,1$, % FILE OPEN FOR OUTPUT %
FEOF=20,1$, % ENDFILE ENCOUNTERED %
FIS35=21,1$, % BIT 35 IS SET %
FSET35=22,1$, % SET BIT 35 ON NEXT WRITE %
FSEQ=24,1$, % PROCESSING SEQUENCE NO. %
FPGMK=25,1$, % PROCESSING PAGE MARK %
FFISTAR=26,1$, % FILE NAME IS "*" %
FXSTAR=27,1$, % EXTENSION NAME IS "*" %
FUOPEN=28,1$, % FILE IS OPEN FOR UPDATE %
FDEVAS=29,1$, % DEVICE ASSIGNED %
FIOERR=30,1$, % I/O ERROR OCCURRED %
FNSET=35,1$; % CHANNEL NOT SET %
MACRO
XFYEND=1^0$, % ENDFILE ALLOWED %
XFYREW=1^1$, % REWIND ALLOWED %
XFYREAD=1^2$, % READING ALLOWED %
XFYWRITE=1^3$, % WRITING ALLOWED %
XFYUPDAT=1^4$, % UPDATE ALLOWED %
XFYDUMMY=1^5$, % DUMMY CHANNEL %
XFYERR=1^6$, % ALLOW WRITE ERROR TRAP %
XFPRIMARY=1^7$; % PRIMARY DEFAULT VALUE %
MACRO OTHER=-1$;
EXTERNAL BINOUT, OCTOUT, DECOUT;
EXTERNAL WRITE, OUTMSG, TTYPUTS;
FORWARD
IOBUFH, %( 6 )%
IODEC, %( 6 )%
IOCTAL, %( 6 )%
IOSPACE, %( 6 )%
IOERR, %(#2 PARAMETERS: THE ERROR NUMBER
)%
IOERRDMP, %( 3
)%
IOERR2DMP, %(
)%
BYTEP, %(
)%
IOPCODE, %( 7
)%
IOPCHAN, %( 9
)%
IODCHAN, %(#10
)%
IOPANIC; %(
)%
EXTERNAL GETSTAT,PUTMSG;
EXTERNAL CRLF;
EXTERNAL CHNLMAX;
EXTERNAL GETSTS;
GLOBAL IOZERROR[10];
GLOBAL ROUTINE IOERR(N)=
BEGIN
%(
THIS IS THE FANTASTIC AMAZING ERROR-RECOVERY ROUTINE
IT DOES EVERYTHING BUT REWRITE THE PROGRAM FOR THE
IDIOT WHO GOT THE ERROR.
WHEN CALLED, THIS ROUTINE INDEXES INTO ERRMSGS BY THE
ERROR NUMBER. IF THE ERROR NUMBER IS TOO LARGE IT
PROVIDES A DEFAULT VALUE OF IOERR(0).
THE ERROR TABLE CONTAINS SPECIFICATIONS OF WHAT TO DO
WITH THE ERROR. THE ENTRY IS TWO WORDS. THE FIRST
HALFWORD IS A RECOVERY OR DIAGNOSTIC ROUTINE (BUT NOT
BOTH). DIAGNOSTIC ROUTINES ARE EXECUTED BEFORE A SELECTED
SOFT HALT (SEE BELOW); RECOVERY ROUTINES ARE EXECUTED
AFTER. THIS IS SO THE SOFT HALT DISPLAYS THE ORIGINAL
ERROR STATE.
THE NEXT HALFWORD CONTAINS THE ACTION FLAGS. THE
DEFAULT IS 0 WHICH @ROVIDES MAXIMUM INFORMATION AND
MINIMUM RECOVERY.
THE FIRST HALFWORD OF THE SEOND WORD POINTS TO THE
SHORT MESSAGE AND THE SECOND HALFWORD TO THE LONG MSSAGE.
IF .ZSHORT THEN WE GIVE THE SHORT MESSAGE.
THE BIT ASSIGNMENTS ARE:
D -> NO CHANNEL TABLE DUMP
H -> NO HARD HALT
S -> GIVE SOFT HALT
I -> IGNORE THIS ERROR
R -> THE ROUTINE IS A DIAGNOSTIC ROUTINE
0<18,18> -> RECOVERY OR DIAGNOSTIC ROUTINE
0<0,1> -> 0 => GIVE CHANNEL TABLE DUMP
1 => NO CHANNEL TABLE DUMP (D-BIT)
0<1,1> -> 0 => GIVE HARD HALT
1 => NO HARD HALT (H-BIT)
0<2,1> -> 0 => NO SOFT HALT
1 => GIVE SOFT (DEBUG) HALT (S-BIT)
0<3,1> -> 0 => PROCESS THIS ERROR
1 => IGNORE THIS ERROR (I-BIT)
0<4,1> -> 0 => RECOVERY ROUTINE
1 => DIAGNOSTIC ROUTINE (R-BIT)
AN ERROR CALL MAY PASS AND RECEIVE INFORMATION VIA
"IOZERROR". THE CONTENTS OF IOZERROR DEPEND UPON WHO ISSUES
THE CALL. WHAT IS RETURNED IN IOZERROR DEPENDS ON THE
DIAGNOSTIC OR RECOVERY ROUTINE SELECTED.
)%
STRUCTURE ERRTAB[I,J]=(((IF .I GTR ..ERRTAB THEN .ERRTAB+1
ELSE .ERRTAB+1+.I*2)+.J)<0,36>);
EXTERNAL ZSHORT;
EXTERNAL ERRTAB ERRMSGS;
LOCAL T,OLDZ;
MACHOP CALLI=#047,HLT=#254;
BIND VECTOR MSGDESC=ERRMSGS[.N,0]<0,36>;
MACRO HALT=HLT(4)$,
D=0,1$,
H=1,1$,
S=2,1$,
I=3,1$,
R=4,1$,
ROUTNAME=18,18$,
MSG=MSGDESC[1]$;
MACRO DUMP=NOT .MSGDESC[0]<D>$,
SOFT=.MSGDESC[0]<S>$,
STOP= NOT .MSGDESC[0]<H>$,
IGNORE=.MSGDESC[0]<I>$,
DIAGNOSE=.MSGDESC[0]<R>$,
RECOVER= NOT .MSGDESC[0]<R>$,
RECRTN=.MSGDESC[0]<ROUTNAME>$,
DIARTN=.MSGDESC[0]<ROUTNAME>$;
MACRO CRUNCH=CALLI(0,#12)$;
IF .ZIGNORE[.N] THEN .N^18 ELSE
IF IGNORE THEN .N^18 ELSE
BEGIN
CRLF(0);
IF .ZSHORT THEN PUTMSG('ERR') ELSE PUTMSG('ERROR');
IOSPACE();
IODEC(.N);
PUTMSG(':');
IF .MSG NEQ 0 THEN TTYPUTS( .MSG<(IF .ZSHORT THEN 18 ELSE 0), 18>);
CRLF(0);
%(
HERE WE ACTUALLY DO SOMETHING ABOUT THE ERROR
)%
IF DUMP THEN IOERRDMP(); % NO D-BIT: DUMP WHOLE WORLD %
OLDZ_.IOZERROR[0]; % SAVE CONTENTS %
IF DIAGNOSE THEN IF (T_DIARTN) NEQ 0 THEN (.T)();
IF SOFT THEN HALT; % S-BIT: SOFT HALT %
% IF WE HAVE A RECOVERY ROUTINE THEN DO IT %
IF RECOVER THEN IF (T_RECRTN) NEQ 0 THEN (.T)();
% IF WE WANT A HARD HALT, DO IT %
IF STOP THEN CRUNCH;
% IF WE HAVE HACKED THRU TO THIS POINT WE CAN DO NO MORE. RETURN TO
THE USER AND HOPE TO HELL HE KNOWS WHAT HE'S DOING %
% IF HE CHANGED IOZERROR[0] IN THE DIAGNOSTIC/RECOVERY ROUTINES
THEN RETURN THE NEW VALUE ELSE RETURN THE ERROR CODE
%
IF .OLDZ NEQ .IOZERROR[0] THEN .IOZERROR[0] ELSE .N^18
END
END;
ROUTINE IOERRDMP=
BEGIN
LOCAL PTR, IND;
INCR PTR TO .CHNLMAX DO IOERR2DMP(.PTR);
END;
ROUTINE IOERR2DMP(N)=
BEGIN
BIND VECTOR CHAN=CHNLTAB[.N,0]<0,36>;
LOCAL WORD[2],PTR1,PTR2,I;
LOCAL T;
! PUT OUT THE CHANNEL
STRUCTURE CHTB[I,J]=[(I+1)*(J+1)]((.CHTB+.I*(J+1)+.J)<0,36>);
IODEC(.N);
PUTMSG('=');
IF .N EQL 0 THEN RETURN (PUTMSG('*TTY*',' ');
CRLF(0));
IF GETSTAT(.N) LSS 0 THEN RETURN (PUTMSG('*INVA','LID*');CRLF(0));
! PUT OUT THE MODE
T_ ( SELECT .FMODE OF NSET
0: '(A)';
1: '(AL)';
#10: '(I)';
#13: '(IB)';
#14: '(B)';
#15: '(ID)';
#16: '(DR)';
#17: '(D)';
OTHERWISE: '*INV*';
TESN);
PUTMSG(.T);
IOSPACE();
! PUT OUT THE DEVICE
WORD[0]_WORD[1]_0;
PTR1_WORD<36,7>;
PTR2_FDEV<36,6>;
DECR I FROM 5 TO 0 DO UNPACKIT(PTR1,PTR2);
TTYPUTS(WORD);
PUTMSG(':');
! PUT OUT THE FILE
WORD[0]_WORD[1]_0;
PTR1_WORD<36,7>;
PTR2_FFILE<36,6>;
DECR I FROM 5 TO 0 DO UNPACKIT(PTR1,PTR2);
TTYPUTS(WORD);
! PUT OUT THE EXTENSION
PUTMSG('.');
WORD[0]_0;
PTR1_WORD<36,7>;
PTR2_FEXT<36,6>;
DECR I FROM 2 TO 0 DO UNPACKIT(PTR1,PTR2);
TTYPUTS(WORD);
IF .FPPN NEQ 0 THEN
(
! PUT OUT THE PPN
PUTMSG('[');
IOCTAL(.FPRJ);
PUTMSG(',');
IOCTAL(.FPRG);
PUTMSG('] ')
) ELSE PUTMSG(' ');
! PUT OUT THE OUTPUT BUFFER HEADER INFO
IOBUFH(.FBUFO,.FOBUFH,.FOPTR,.FOCNT);
! PUT OUT THE INPUT BUFFER HEADER INFO
IOBUFH(.FBUFI,.FIBUFH,.FIPTR,.FICNT);
! PUT OUT THE STATUS INFO
PUTMSG(' <(');
BINOUT(0,0,.FSTATUS<18,18>);
PUTMSG(',,');
BINOUT(0,0,.FSTATUS<0,18>);
PUTMSG(')');
IF .FPROT NEQ 0 THEN ( PUTMSG(' OLD:'); IOCTAL(.FPROT));
IF (PTR1_.FNEWPR) NEQ 0 THEN (PUTMSG(' NEW:'); IOCTAL(.PTR1<0,9>));
PUTMSG('>');
! PUT OUT THE SWITCHES
INCR I TO 35 DO
BEGIN
IF .FSWITCH<.I,1> THEN
BEGIN
PUTMSG('/');
WRITE(0, IF .I GTR 25 THEN .I-26+"0" ELSE .I+"A");
END;
END;
! PUT OUT THE ERROR CODE
IF .FERR NEQ 0 THEN
(PUTMSG(' ERRO','R= #');
IOCTAL(.FERR));
! PUT OUT THE SYSTEM STATUS BITS
IF .FSTATUS<FDEVAS> THEN
BEGIN
PUTMSG(' STS=');
IOCTAL(GETSTS(.N));
END;
! AND RETURN THE CARRIAGE (WHEW!)
CRLF(0);
END;
ROUTINE BYTEP(VAL)=
%( THE FUNCTION OF THIS ROUTINE IS TO PRINT OUT A BYTE POINTER
IN THE FORMAT
ADR<P,S,NDX,IND>
WHERE @ AND (NDX) MAY BE OMITTED IF 0
)%
BEGIN
IOCTAL(.VAL<0,18>);
PUTMSG('<');
IODEC(.VAL<30,6>);
PUTMSG(',');
IODEC(.VAL<24,6>);
IF (.VAL<18,4> NEQ 0) OR .VAL<22,1> THEN
BEGIN
PUTMSG(',');
IODEC(.VAL<18,4>);
IF .VAL<22,1> THEN PUTMSG(',1');
END;
PUTMSG('>');
END;
ROUTINE IOCTAL(X)=OCTOUT(0,0,.X);
ROUTINE IODEC(X)=DECOUT(0,0,.X);
ROUTINE IOSPACE=PUTMSG(' ');
ROUTINE IOBUFH(BUF,HDR,PTR,CNT)=
BEGIN
IOCTAL(.BUF);
PUTMSG(':(');
IOCTAL(.HDR<0,18>);
IOSPACE();
BYTEP(.PTR);
IOSPACE();
IODEC(.CNT);
PUTMSG(') ');
END;
GLOBAL ROUTINE IOPCODE=
BEGIN
PUTMSG('CODE=');
IODEC(.IOZERROR)
END;
GLOBAL ROUTINE IOPCHAN=
BEGIN
PUTMSG('CHNL=');
IODEC(.IOZERROR);
END;
GLOBAL ROUTINE IODCHAN=
BEGIN
IOERR2DMP(.IOZERROR);
END;
GLOBAL ROUTINE IOPANIC(N)=
BEGIN
MACHOP CALLI=#047;
PUTMSG('PANIC',' STOP',' ON E','RROR ',' ');
IODEC(.N);
CRLF(0);
CALLI(0,#12);
END;
ROUTINE DIAGNOSE=
BEGIN
%(
CALL FROM DDT BY PUSHJ DIAGNO$X
)%
LOCAL T;
MACHOP TTCALL=#051;
TTCALL(#11);
DO (PUTMSG('CHNL=');TTCALL(0,T)) UNTIL .T GTR "0" AND .T LEQ "9";
IOERR2DMP(.T-"0");
END;
GLOBAL ROUTINE SETREQ(CODE)=
BEGIN
%%
%
THIS ROUTINE RECEIVES A VARIABLE NUMBDR OF PARAMETERS OF THE
FORM (M1, M2, ... , MN, CODE) AND SETS THE VALUE "CODE" IN THE
IGNORE REQUEST BITS FOR MESSAGES M1-MN.
%
%%
LOCAL N;
N_ IF .(@(CODE+1))<27,9> NEQ #274 THEN 0 ELSE .(@@(CODE+1))<0,18>;
IF .N GTR 1 THEN
BEGIN
BIND VECTOR PLIST=CODE+1-.N;
%%
%
PLIST NOW POINTS TO THE FIRST OF THE ARGUMENTS
%
%%
DECR I FROM .N-2 TO 0 DO ZIGNORE[.PLIST[.I]]_.CODE;
END;
END;
END ELUDOM