Google
 

Trailing-Edge - PDP-10 Archives - ap-c800d-sb - pscan.mac
There are 21 other files named pscan.mac in the archive. Click here to see a list.
; UPD ID= 1110 on 6/29/78 at 2:52 PM
TITLE	PSCAN FOR COBOL V12
SUBTTL	COBOL PICTURE SCANNER	AL BLACKINGTON/CAM



;THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY ONLY BE USED
;  OR COPIED IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE.
;
;COPYRIGHT (C) 1974, 1979 BY DIGITAL EQUIPMENT CORPORATION

	SEARCH	P
	%%P==:%%P

;EDITS
;V10*****************
;NAME	DATE		COMMENTS
;	30-JAN-76	;[404] FIX COUNT OF ALLOWABLE CHARS IN EDITTED PICTURE
;	9-DEC-76	;[310] RE-INSERTED FIX ERROR CHECKING OF PIC CLAUSES.
;DBT	4/8/75		- DETECT  PIC +$++++..... AS ILLEGAL
;			- CHANGE ALL 9'S IN ALPHANUMERIC EDITED PICTURES
;			  INTO X'S
;DBT	4/14/75		- DETECT ZZZZ AS WELL AS ++++ AS BEING
;			  BLANK WHEN ZERO
;			- DETECT NON LEFT JUSTIFIED SIGN INSERT AS ILLEGAL
;DBT	6/6/75		- ALLOW +PPP999 AND 999PPPCR
;********************


;AT EDIT 211 ALLOW LC LETTERS IN A PICTURE CLAUSE

TWOSEG
RELOC	400000
SALL

;A PICTURE IS CONVERTED INTO A MASK.
;INTERNAL SIZE, EXTERNAL SIZE AND DECIMAL LOCATIONS ARE CALCULATED.
;THE FEDIT, FSIGN, FBWZ AND CLASS FLAGS ARE SET IN "SW" AS APPROPRIATE.
;CONTENTS OF ALL OTHER ACCUMULATORS, EXCEPT LN & CP,  ARE SAVED.

;LN & CP ARE SET TO THE POSTION OF THE FIRST CHARACTER OF THE WORD.

ENTRY		PSCAN
EXTERNAL	GETKAR, GETCH, FATAL, FATALW
;THE VALIDITY OF THE MASK IS DETERMINED BY USING A
;  TRUTH TABLE. WHEN A CHARACTER IS PICKED UP, A
;  BIT IS SET IN ACCUMULATOR "TC" TO REPRESENT
;  ONE OF THE CONDITIONS SHOWN BELOW; AND ACCUMULATOR
;  "TA" IS SET TO POINT TO THE APPROPRIATE ENTRY IN
;  THE TRUTH TABLE. IF THAT ENTRY CONTAINS ANY BITS
;  MATCHING THOSE IN "PC", THE CHARACTER IS INVALID.
;"TC" IS THEN ORED INTO "PC".

;FLAGS FOR ACCUMULATOR "PC"
; 0 + OR - INSERT IS NOT LEADING	;18 P BEFORE 9,POINT
; 1 + FLOAT AFTER 9, POINT		;19 Z BEFORE 9, POINT
; 2 + INSERT AFTER 9, POINT		;20 * BEFORE 9, POINT
; 3 - FLOAT AFTER 9, POINT		;21 $ FLOAT BEFORE 9, POINT
; 4 - INSERT AFTER 9, POINT		;22 $ INSERT BEFORE 9, POINT
; 5 P AFTER POINT			;23 + FLOAT BEFORE 9, POINT
; 6 Z AFTER POINT			;24 + INSERT BEFORE 9, POINT
; 7 * AFTER POINT			;25 - FLOAT BEFORE 9, POINT
; 8 $ FLOAT AFTER POINT			;26 - INSERT BEFORE 9, POINT
; 9 $ INSERT AFTER POINT		;27 9
;10 + FLOAT BEFORE 9, AFTER POINT	;28 V
;11 + INSERT BEFORE 9, AFTER POINT	;29 S
;12 - FLOAT BEFORE 9, AFTER POINT	;30 A,X
;13 - INSERT BEFORE 9, AFTER POINT	;31 CR,DB
;14 + FLOAT AFTER 9, BEFORE POINT	;32 DECIMAL POINT
;15 + INSERT AFTER 9, BEFORE POINT	;33 COMMA
;16 - FLOAT AFTER 9, BEFORE POINT	;34 0
;17 - INSERT AFTER 9, BEFORE POINT	;35 B


;NON-STANDARD ACCUMULATORS USED
PC=3	;FLAGS REPRESENTING CHARACTERS ALREADY SEEN
PR=5	;CONTAINS REPEAT COUNTER
MP=6	;BYTE POINTER TO MASK WORDS
SP=7	;SAVE MP WHEN FIRST FLOATER SEEN
PICCNT=10	;COUNT NO. CHARACTERS IN PICTURE - 30 MAX
ST=11	;SAVE POINTER TO PICT2 TABLE

HI==TA	;HIGHEST ACCUMULATOR SAVED BY THIS ROUTINE

FOFF==FFLOAT!FSIGN!FCLAS1!FEDIT!FBWZ
PSCAN:	MOVEM	HI,SAVEAC+HI	;SAVE THE ACCUMULATORS
	MOVEI	HI,SAVEAC
	BLT	HI,SAVEAC+HI-1

	SWOFF	FOFF		;TURN OFF FLAGS
	SWON	FCLAS2;		;TURN ON ALPHABETIC FLAG
	MOVE	PC,[POINT 7,PICBUF##]
	MOVEM	PC,PICPTR##
	SETZM	PICBUF
	MOVE	PC,[PICBUF,,PICBUF+1]
	BLT	PC,PICBUF+6

	SETZB	PC,EXSIZE	;CLEAR COUNTERS
	SETZB	PR,INSIZE
	SETZM	DPSIZE
	MOVE	MP,MSKPTR
	SETZM	(MP)		;CLEAR SIGN AND FLOAT CHARACTERS, &
	MOVEI	PICCNT,^D30	;MAX NUMBER OF CHARACTERS IN PIC SOURCE
				;NO CHARACTERS TO REPEAT
	SETZM	FLOTBZ##
	TSWF	FREGCH		;[310] PASSED PIC TERMINATOR?
	JRST	PSCAN1		;[310] YES-OKAY SO FAR
	CAIE	CH," "		;[310] DID WE GET HERE VIA BLANK?
	JRST	PICERL		;[310] NO- THEN WE HAVE NO PICTURE

;SKIP OVER LEADING SPACES

PSCAN1:	PUSHJ	PP,GETCH	;GET A CHARACTER
	TSWF	FEOF;		;END OF INPUT?
	JRST	FINISH		;YES

	CAIN	CH," "		;NO--SPACE?
	JRST	PSCAN1		;YES--SKIP IT

	PUSHJ	PP,PCONVL	; CONVERT LC-TO UPPER [211]
	MOVEM	CP,WORDCP
	MOVEM	LN,WORDLN

;CHECK FOR THE WORD "IS"

	CAIE	CH,"I"
	JRST	PSCN1B
	PUSHJ	PP,GETCPY
	CAIE	CH,"S"
	JRST	PICERD
	PUSHJ	PP,GETCPY
	CAIE	CH," "
	JRST	PICERD
PSCN1A:	PUSHJ	PP,GETCH
	CAIN	CH," "
	JRST	PSCN1A
	PUSHJ	PP,PCONVL	; CONVERT LC-TO UPPER [211]

PSCN1B:	MOVEM	CP,SAVEAC+CP	;SAVE CP,LN
	MOVEM	CP,SAVCP1
	MOVEM	CP,WORDCP
	MOVEM	LN,SAVEAC+LN
	MOVEM	LN,WORDLN
	MOVEM	LN,SAVLN1
	MOVEM	CH,PICSAV	;SAVE CHARACTER
	JRST	PSCN2A
;GET A PICTURE CHARACTER

PSCAN2:	MOVE	CH,PICSAV##	;RESTORE LOOKAHEAD CHARACTER
	PUSHJ	PP,GETCP0	;NOTE THAT WE ARE TAKING CHARACTER
	CAIE	CH," "		;END OF PICTURE?
	JRST	PSCN2A		;NO
	TSWF	FREGCH		;IF LOOKAHEAD IS ON (IT SHOULD BE)
	IDPB	CH,PICPTR	;STORE THE LOOKAHEAD CHAR, IT WILL BE REMOVED LATER
	JRST	FINISH		;GO TO END OF PIC STRING


PSCN2A:	TSWF	FEOF;		;END OF FILE?
	JRST	FINISH		;YES

	CAIN	CH,";"		;IS IT ";"
	JRST	FINISH		;YES

	CAIN	CH,","		;NO--IS IT ","
	JRST	PSCN2B		;YES
	CAIN	CH,"="		;COULD IT BE PSEUDO-TEXT?
	JRST	PSCAN3		;YES
	CAIE	CH,"."		;IS IT A PERIOD?
	JRST	PICS		;NO

;IT IS PERIOD OR COMMA--SEE IF LAST CHARACTER

PSCN2B:
	PUSHJ	PP,GETKAR	;LOOK AT NEXT CHARACTER

	CAIN	CH," "		;IS IT A SPACE?
	JRST	PSCAN4		;YES -- DONE
	TSWT	FEOF;		;NO--END OF FILE?
	JRST	PICS00

;TERMINATING CHARACTER SEEN

PSCAN4:	MOVE	CH,PICSAV	;GET CHAR BACK
	CAIN	CH,"."		;WAS IT PERIOD?
	SWON	FGTPER;		;YES--TELL GETWRD ROUTINE
	JRST	FINISH


PSCAN3:	SWON	FREGCH		;REGET = ON RETURN
	MOVEI	CH," "		;FAKE A SPACE
	JRST	FINISH		;AND GET OUT
;	WE HAVE A CHARACTER LETS LOOK AHEAD AND SEE IF THERE IS
;	A REPEAT FORTH COMMING

PICS:
	PUSHJ	PP,GETKAR	;GET NEXT INPUT CHARACTER

PICS00:	;ENTER HERE IF YOU ALREADY HAVE LOOKAHEAD CHARACTER

	MOVEI	PR,1		;INITIALIZE REPEAT COUNT
	CAIN	CH,"("		;REPEAT???
	PUSHJ	PP,GETRP	;YES
	EXCH	CH,PICSAV	;SAVE LOOKAHEAD CHARACTER AND RESTORE CURRENT
;LOOK FOR CHARACTERS WITHOUT SPECIAL CHARACTERISTICS

	MOVSI	TA,-PICT1S	;SET UP IOWD TO SEARCH PICT1
PICS0:	HRRZ	TB,PICT1(TA)	;PICK UP CHARACTER FROM TABLE
	CAME	CH,TB		;DOES IT MATCH INPUT CHARACTER?
	AOBJN	TA,PICS0	;NO--LOOP UNTIL ALL OF TABLE SEEN

	JUMPG	TA,PICS1	;DID WE FIND ONE?
	HLRZ	TA,PICT1(TA)	;YES--RESET TA
	CAIE	CH,"A"		;IS IT "A"
IFN ANS74,<
	CAIN	CH,"B"		; OR "B"?
	 CAIA			;YES--THIS CHARACTER DOESN'T MAKE IT NON-ALPHABETIC
>;END IFN ANS74
	SWOFF	FCLAS2;		;NO--RESET FLAG
IFN ANS74,<
	CAIE	CH,"*"		;IS IT "*"
	JRST	PICF		;NO
	PUSH	PP,TA		;SAVE IT
	SKIPN	TA,CURDAT##
	JRST	PICS01
	LDB	TA,DA.BWZ##	;SEE IF BLANK WHEN ZERO SET?
	SKIPE	TA		;WAS IT?
	EWARNW	E.701		;YES
PICS01:	POP	PP,TA
>
	JRST	PICF

;LOOK FOR "$" "," "."

PICS1:	SWOFF	FCLAS2;		;CANNOT BE ALPHABETIC
	MOVNI	TA,1
	CAMN	CH,DOLLR.##	;IS IT A CURRENCY SIGN?
	MOVEI	TA,16		;YES
	CAMN	CH,DCPNT.##	;NO--IS IT DECIMAL POINT?
	MOVEI	TA,07		;YES
	CAMN	CH,COMA.##	;NO--IS IT COMMA?
	MOVEI	TA,06		;YES
	JUMPG	TA,PICF		;WAS IT ANY OF THOSE?

;LOOK FOR "CR" OR "DB"

	CAIN	CH,"C"		;IS IT "C"?
	JRST	PICS2		;YES

	CAILE	PR,1		;REPEATING??
	JRST	PICERF		;YES -- ERROR
	CAIE	CH,"D"		;NO--IS IT "D"?
	JRST	PICERA		;NO--ERROR

	PUSHJ	PP,GETKAR	;YES--GET NEXT CHARACTER
	EXCH	CH,PICSAV	;SAVE IT AS LOOKAHEAD
	PUSHJ	PP,GETCP0	;NOTE THAT WE TOOK LAST CHAR
	CAIE	CH,"B"		;IS IT "B"?
	JRST	PICERF		;NO--ERROR
	MOVEI	TA,11		;YES--SET TABLE POINTER
	JRST	PICF

PICS2:
	CAILE	PR,1		;REPEATING??
	JRST	PICERF		;YES -- ERROR
	PUSHJ	PP,GETKAR	;IT IS "C"--GET NEXT CHARACTER
	EXCH	CH,PICSAV	;EXCHANGE WITH LOOKAHEAD
	PUSHJ	PP,GETCP0	;NOTE THAT WE TOOK IT
	CAIE	CH,"R"		;IS IT "R"?
	JRST	PICERF		;NO--ERROR
	MOVEI	TA,10		;YES--SET TABLE POINTER
;IT IS A PICTURE CHARACTER

PICF:	HLRZ	TB,PICT2(TA)	;GET BIT SETTINGS

	TRNN	TB,%P		;IS IT "P"?
	JRST	.+3		;NO
	SKIPN	INSIZE		;YES--ANY LEADING CHARACTERS?
	IORI	PC,%V		;NO--"V" IMPLIED

	SUBI	CH,40		;CONVERT CHARACTER TO SIXBIT


	MOVE	TC,TB

	TRNE	TB,%FLOAT	;CAN IT BE A FLOATER?
	PUSHJ	PP,FLOAT	;YES--GO SEE IF IT IS

	MOVE	ST,TA		;SAVE TABLE POINTER
IFN ANS74,<
	CAIN	TA,23		;CHECK FOR "/"
	MOVEI	TA,5		;AND HANDLE AS IF "0"
>

	TRNE	TB,%MINUS!%PLUS	;IS THIS A SIGN?
	TRNN	PC,%9		;YES--HAS 9 BEEN SEEN?
	JRST	PICF2		;NO
	LSH	TC,11		;YES--SHIFT BIT
	ADDI	TA,11		;ADD TO TABLE POINTER

PICF2:	TRNE	TB,%SPEC.	;IS THIS SPECIAL AFTER POINT?
	TRNN	PC,%.!%V	;YES--HAS POINT BEEN SEEN?
	JRST	PICF3		;NO
	LSH	TC,15		;YES--SHIFT BIT
	ADDI	TA,15		;ADD TO TABLE POINTER

PICF3:	TDNE	PC,PICT3(TA)	;IS CHARACTER LEGAL HERE?
	PUSHJ	PP,PICERB	;NO--ERROR
;IT IS A LEGAL CHARACTER IN THIS POSITION


	TDNN	TC,[%SB9]	;IS IT AN INSERT SIGN BEFORE  9
	JRST	PICF3A		;NO
	MOVE	TD,SAVCP1	;SAVE CHARACTER POS FOR FINISH: ERROR CHECK
	MOVEM	TD,SAVBCP	;BORROW SAVBCP TEMPORIARILY
	TDNE	PC,[%CHGEN]	;YES-- IS IT LEADING?
	TLO	PC,(%NOTLD)	;NO - SET FLAG

PICF3A:	IOR	PC,TC		;PUT FLAG IN PC

	TRNE	TB,%PLUS!%MINUS	;IS IT A SIGN?
	DPB	CH,MSKSYN	;YES--PUT IT IN SIGN POSITION

	TDNE	TC,[%FCHAR]	;NO--IS IT SUPPRESSION OR FLOAT?
	DPB	CH,MSKFLT	;YES--PUT IT IN FLOAT POSITION

	TRNN	TB,%S!%V!%P	;IS THIS S, P OR V?
	ADDM	PR,EXSIZE	;NO--ADD TO EXTERNAL SIZE

	TDNE	TC,[%CNTIN]	;DOES IT COUNT AS AN INTERNAL CHARACTER?
	ADDM	PR,INSIZE	;YES--ADD TO INTERNAL SIZE

	TRNE	PC,%.!%V	;IS IT AFTER DECIMAL POINT?
	TDNN	TC,[%CNTDP]	;IS IT A DECIMAL PLACE?
	JRST	.+2		;NO
	ADDM	PR,DPSIZE	;YES--ADD TO DECIMAL PLACES

	EXCH	PR,DPSIZE
	TRNE	TC,%PBP		;IS THIS A "P" BEFORE A "V"?
	SUB	PR,DPSIZE	;YES--DECREMENT DECIMAL PLACES
	EXCH	PR,DPSIZE

	TRNE	TB,%CRDB	;IS IT "CR" OR "DB"?
	AOS	EXSIZE		;YES--ADD TO EXTERNAL SIZE AGAIN

	MOVE	TE,EXSIZE	;PICTURE TOO LARGE?
	CAILE	TE,MAXWSS
	JRST	PICERC		;YES--ERROR

	TLNN	TC,%FAP		;NO--IS IT FLOAT OR SUPPRESS AFTER POINT?
	SKIPA	TE,PICT2(ST)	;NO--GET MASK BYTE FROM TABLE
	MOVEI	TE,PIC9		;YES--USE "9"
	TRNE	TB,%S!%P	;S AND P DON'T GO INTO MASK
	JRST	[		; CHECK FOR IMPLIED V AND IF SO PUT IN MASK
		TRNE	TB,%P		;IS IT P
		CAME	PR,INSIZE	;FIRST CHARACTER?
		JRST	PSCAN2		;NOT P OR NOT FIRST
					;HAVE IMPLIED V
		MOVEI	TE,PICV		;GET V MASK CODE
		MOVEI	PR,1		;BE SURE REPEAT IS 1
		JRST	.+1]
	; DO WE HAVE A REPEAT TO PUT IN
	CAIG	PR,1		;???
	JRST	PICF7		;NO REPEAT

	; INSERT REPEAT CODE
	; <REPEAT CODE><#BYTES FOR BINARY COUNT><BYTE 1>...

	TRNE	TE,NORPT	;CAN IT BE REPEATED???
	PUSHJ	PP,PICERB	;NO - ERROR

	MOVEI	TD,PICRPT	;REPEAT MASK CODE
	SOJL	PICCNT,PICERH	; [404] WILL IT FIT??
	IDPB	TD,MP		;OK - STORE IT

	PUSH	PP,TA		;SAVE A COUPLE REGS
	PUSH	PP,TB	
	MOVEI	TA,9		;MAX NO BYTES FOR BINARY NUMBER (4 BITS)
	MOVE	TB,[POINT 4,PR]	;POINTER TO NUMBER
PICF6:	ILDB	TD,TB		;GET A BYTE
	CAIN	TD,0		;IS IT NON-ZERO??
	SOJN	TA,PICF6	;NO - GET NEXT ONE

	;PUT COUNT OF BYTES IN
	SOJL	PICCNT,PICERJ	; [404] WILL IT FIT??
	IDPB	TA,MP		;OK  - STORE IT

	; NOW THE NUMBER ITSELF
PICF6A:
	SOJL	PICCNT,PICERJ	; [404] WILL BYTE FIT??
	IDPB	TD,MP		;OK - STORE IT
	ILDB	TD,TB		;GET ANOTHER
	SOJG	TA,PICF6A	;ANY LEFT??
	POP	PP,TB		;THATS ALL - RESTORE
	POP	PP,TA

	; STORE MASK CHARACTER CODE
PICF7:
	SOJL	PICCNT,PICERH	; [404] WILL IT FIT???
	IDPB	TE,MP		;OK - STORE IT
	JRST	PSCAN2		;TAKE IT FROM THE TOP
;ROUTINE TO GET AN INTEGER WITHIN PARENTHESES

GETRP:
	PUSHJ	PP,GETCP0	;TAKE LOOKAHEAD CHARACTER
	MOVEI	PR,0		;CLEAR RESULT

GETRP1:	PUSHJ	PP,GETCPY	;GET A CHARACTER
	CAIG	CH,"9"		;IS IT A DIGIT?
	CAIGE	CH,"0"
	JRST	GETRP2		;NO

	IMULI	PR,12		;YES--ADD TO RESULT
	ADDI	PR,-"0"(CH)

	CAIG	PR,MAXWSS	;IS IT > LIMIT?
	JRST	GETRP1		;NO--LOOP

	SETZM	PICSAV		;CLEAR OLD CHARACTER ON ERROR
	JRST	PICERC		;YES--ERROR

GETRP2:	CAIN	CH,")"		;WAS TERMINATOR A RIGHT PAREN?
	SKIPG	PR		;YES--IS INTEGER POSITIVE?
	JRST	GETRP3		;NOT ")",  OR INTEGER = 0  --ERROR
	PJRST	GETKAR		;JUST TO GET IN SYNC

GETRP3:	POP	PP,(PP)		;RETURN TO CALLER'S CALLER.
	SETZM	PICSAV		;CLEAR OLD CHAR
	SKIPG	PR
	MOVEI	PR,1
	JRST	PICERA


;THIS ROUTINE DECIDES IF THIS CHARACTER IS A FLOATER.
;IF IT IS, APPROPRIATE ACTION IS TAKEN


FLOAT:	MOVEI	TE,%ALLSH	;HAS THIS FLOATER BEEN SEEN BEFORE?
	IMUL	TE,TB
	TDNN	PC,TE
	JRST	FLOAT5		;NO

FLOAT0:
	TSWFS	FFLOAT;		;YES--ARE WE ALREADY FLOATING?
	JRST	FLOAT4		;YES

	; CHECK FOR THE ILLEGAL CASE OF +$+++ 
	; AND CLEAR THE LEADING SIGN BIT WHILE YOU ARE AT IT

	TRNN	TB,%MINUS!%PLUS	;IS IT A SIGN?
	JRST	FLOAT1		;NO - MOVE ON
	TLZE	PC,(%NOTLD)	;YES CLEAR AND TEST NOT LEADING FLAG
	JRST	FLOAT1		;ITS NOT A LEADING SIGN
	TRNE	PC,%$		;LEADING- WAS THERE A $ INSERT
	JRST	PICERB		;YES-- ERROR

FLOAT1:	ANDCMI	PC,(TB)		;TURN OFF INSERT FLAG
	DPB	CH,MSKFLT	;PUT CHARACTER IN FLOAT POSITION


	TRNN	PC,%V!%.	;HAS DECIMAL POINT BEEN SEEN?
	CAIN	SP,0		;DO WE NEED TO SHIFT MASK
	JRST	FLOAT4		;YES--LEAVE MASK ALONE

;SHIFT MASK LEFT ONE POSITION TO GET RID OF FIRST FLOATER.

	MOVE	TD,SP		;GET POSITION OF FIRST FLOATER
	IBP	TD		;ADVANCE ONE BYTE

FLOAT2:	CAMN	TD,MP		;CURRENT POSITION?
	JRST	FLOAT3		;YES--DONE
	ILDB	TE,TD		;NO--MOVE A BYTE
	IDPB	TE,SP
	JRST	FLOAT2		;LOOP

FLOAT3:	MOVE	MP,SP		;RESET MP

FLOAT4:	LSH	TC,1		;SHIFT BIT
	AOJA	TA,CPOPJ##	;KICK UP TABLE POINTER AND SKIP

FLOAT5:	MOVE	SP,MP		;SAVE POINTER TO MASK POSITION

	CAIG	PR,1		;IN A REPEAT??
	POPJ	PP,		; N0 - RETURN
	AOS	EXSIZE		;ADD 1 TO MAKE UP FOR TI HERE
	MOVEI	SP,0		;CLEAR SO NO MASK SHIFT
	SOJA	PR,FLOAT0	;SUBTRACT 1 FOR FLOATER
;PICTURE HAS BEEN SCANNED

FINISH:
	SKIPG	EXSIZE		;[310] DOES THIS PIC PRODUCE ANY SIZE?
	JRST	PICERK		;[310] NO - THEN WE HAVE NO PICTURE.
	;CHECK FOR SIGN IN MIDDLE WHICH COULD NOT BE CAUGHT BEFORE
	;LIKE 00+00

	TDNE	PC,[%SB9]	;SIGN INSERT BEFORE 9 ?
	TLNN	PC,(%NOTLD)	;YES - WAS IT LEADING?
	JRST	FINSH1		;OK - NO SIGN OR LEADING SIGN
	LDB	CH,MP		;NOT LEADING BUT IS IT TRAILING?
	CAIN	CH,PICIS	;CHECK LAST MASK CHARACTER AGAINST INSERT SIGN
	JRST	FINSH1		;OK - ITS TRAILING
	MOVE	CP,SAVBCP	;NAUGHTY NAUGHTY - GET SIGN POSITION
	MOVE	LN,SAVLN1
	MOVEI	DW,E.51		;INVALID PICTURE
	PUSHJ	PP,FATAL

FINSH1:	MOVE	CH,SAVCP1	;GET CHARACTER POSITION OF LAST NON-BLANK
	ADDI	CH,1		;BUMP IT BY 1
	MOVEM	CH,SAVBCP	;THAT IS WHERE LAST BLANK WAS
	MOVE	CH,SAVLN1	;THIS IS WHERE
	MOVEM	CH,SAVBLN	;  LAST BLANK WAS

	MOVEI	CH,ENDPIC	;STASH A TERMINATOR IN MASK
	IDPB	CH,MP

	TSWF	FCLAS2;		;STILL ALPHABETIC?
	JRST	FINSH2		;YES

	TRNN	PC,%X		;NO--ANY "X" OR "A"?
	SWONS	FCLAS1;		;NO--IT IS NUMERIC
	SWOFFS	FCLAS1!FCLAS2	;YES--IT IS ALPHANUMERIC
	JRST	FINSH2		;FOR NUMERICS

;	FOR ALPHANUMERICS CHECK AND SEE IF THEY ARE EDITED AND THEN
;	CHANGE ALL 9'S TO X'S IF SO

	TDNN	PC,[%EDIT]	;HAVE WE SEEN ANY EDITING CHARACTERS?
	JRST	FINSH2		;NO

	MOVE	TA,MSKPTR	;GET STARTING MASK POINTER
	MOVEI	TB,PICXA	;SET UP X CODE
FINSH3:	ILDB	CH,TA		;GET MASK CHARACTER
	CAIE	CH,PICRPT	;REPEAT??
	JRST	FINSH4		;NO
	ILDB	CH,TA		;SKIP REPEAT - GET COUNT
	IBP	TA		;SKIP OVER IT
	SOJG	CH,.-1
	ILDB	CH,TA		;GET THE CHARACTER
FINSH4:	CAIN	CH,PIC9		;IS IT A 9 ??
	DPB	TB,TA		;YES - CHANGE IT TO AN X
	CAIE	CH,ENDPIC	; ARE WE DONE?
	JRST	FINSH3		;NOT YET


FINSH2:	TDNE	PC,[%SIGN]	;ANY SIGN?
	SWON	FSIGN;		;YES--SET FLAG

	TDNE	PC,[%FCHAR]	;ANY FLOAT OR SUPRESSION CHARS?
	SETOM	FLOTBZ		;SET POSSIBLE BLANK WHEN ZERO

	TRNE	PC,%9		;ANY 9'S SEEN?
	SETZM	FLOTBZ		;YES


	TDNE	PC,[%EDIT]	;ANY EDITING CHARACTERS?
	SWON	FEDIT;		;YES--SET FLAG

	MOVEM	SW,SAVEAC+SW	;SAVE SW

	MOVEI	TB,0
	LDB	TA,MSKSYN	;GET SIGN CHARACTER
	CAIN	TA,"-"-40	;IS IT MINUS?
	DPB	TB,MSKSYN	;YES--REPLACE WITH SPACE
	LDB	TA,MSKFLT	;GET FLOAT CHARACTER
	CAIE	TA,"-"-40	;IS IT MINUS?
	CAIN	TA,"Z"-40	;NO--IS IT "Z"?
	DPB	TB,MSKFLT	;YES--REPLACE WITH SPACE

	SUB	MP,MSKPTR	;COMPUTE MSKSIZ
	MOVEI	MP,1(MP)
	MOVEM	MP,MSKSIZ

	TSWT	FCLAS1		;IS IT NUMERIC?
	JRST	FINSH9		;NO
	MOVE	CH,INSIZE	;YES-- > 18 DIGITS?
	CAIG	CH,^D18
	JRST	FINSH5		;NO
	MOVEI	DW,E.330	;YES--ERROR
	PUSHJ	PP,FATALW
	MOVEI	CH,^D18		;JAM SIZE OF 18
	MOVEM	CH,INSIZE
	MOVE	CH,DPSIZE	;MORE THAN 18 DECIMAL PLACES, TOO?
	CAIG	CH,^D18
	JRST	FINSH9		;NO, OK NOW
	MOVEI	CH,^D18		;JUST QUIETLY STICK 18 IN HERE, TOO
	MOVEM	CH,DPSIZE	; 'CAUSE WE ALREADY COMPLAINED ABOUT IT
	JRST	FINSH9

FINSH5:	MOVE	CH,DPSIZE	;CHECK FOR TOO MANY DECIMAL PLACES IN ITEM
	CAIG	CH,^D18
	JRST	FINSH9		;NO, OK
	MOVEI	DW,E.602	; ?TOO MANY DECIMAL PLACES
	PUSHJ	PP,FATALW
	MOVEI	CH,^D18		;MAKE IT 18
	MOVEM	CH,DPSIZE
FINSH9:	MOVSI	HI,SAVEAC	;RESTORE ACCUMULATORS
	BLT	HI,HI-1
	MOVE	HI,SAVEAC+HI
	POPJ	PP,
;ERROR--INVALID PICTURE CHARACTER

PICERA:	MOVEI	DW,E.51
	JRST	PICER1

;ERROR--INVALID COMBINATION OF CHARACTERS

PICERB:	POP	PP,DW		;THROW AWAY A PP ENTRY
	MOVEI	DW,E.52
	JRST	PICER1

;ERROR--TOO LARGE A FIELD

PICERC:	MOVEI	DW,E.316

PICER1:	MOVE	LN,SAVLN1
	MOVE	CP,SAVCP1
PICERR:	PUSHJ	PP,FATAL	;[310] PUT OUT A DIAGNOSTIC
PICER2:	MOVEM	CH,SAVEAC+CH	;SAVE THE CHARACTER
	SKIPN	CH,PICSAV	;LOOKAHEAD CHAR??
	PUSHJ	PP,GETCPY	;SCAN UNTIL SPACE SEEN
	SETZM	PICSAV		;DON'T LOOK AT THIS ANYMORE
PICER3:	CAIE	CH," "
	JRST	PICER2
	MOVE	CH,SAVEAC+CH	;GET BACK NEXT TO LAST CHARACTER
	CAIN	CH,"."		;WAS IT A PERIOD?
	SWON	FGTPER		;YES--SET "GET A PERIOD" FLAG
	JRST	FINSH1		;SKIP ERROR CHECK AT BEGINNING OF FINISH

;ERROR--WORD STARTED WITH "I" BUT WAS NOT "IS"

PICERD:	MOVEM	CH,SAVEAC+CH
	MOVEI	DW,E.182
	PUSHJ	PP,FATALW
	SETZM	PICSAV		;NO LOOKAHEAD CHARACTER
	JRST	PICERG



;ERROR--"C" OR "D" NOT FOLLOWED BY PROPER CHARACTER

PICERF:	MOVE	CP,SAVCP1
	MOVE	LN,SAVLN1
	MOVEM	CH,SAVEAC+CH
	MOVEI	DW,E.51
	PUSHJ	PP,FATAL
PICERG:	MOVE	CH,SAVEAC+CH
	JRST	PICER3

;ERROR--PICTURE TOO LARGE

PICERJ:	POP	PP,TB		; [402] RESTORE PUSH DOWN POINTER
	POP	PP,TA		; [402] BEFORE GIVING ERROR
PICERH:	MOVEI	DW,E.72
	JRST	PICER1
	; ERROR NO PICTURE - BECAUSE PIC DELIMITER NOT A SPACE - PROBABLY A PERIOD.
PICERL:	MOVE	LN,SAVELN##	;[310] GET SOURCE LINE
	MOVE	CP,SAVECP##	;[310] GET SOURCE POSITION
	MOVEI	DW,E.220	;[310] PICTURE REQUIRED
	JRST	PICERR		;[310] PUT OUT DIAGNOSTIC

; ERROR NO PICTURE BECAUSE DESCRIPTION PRODUCED NO SIZE.
PICERK:	MOVEI	DW,E.220	;[310] PICTURE REQUIRED
	MOVE	LN,SAVLN1	;[310] GET LINE #
	MOVE	CP,SAVCP1	;[310] GET POSITION #
	PUSHJ	PP,FATAL	;[310] GIVE FATAL ERROR.
	JRST	FINSH1		;[310] FINISH UP

;GET A CHARACTER, IF THE CHARACTER IS NOT A SPACE
;CONVERT TO UPPER CASE, SAVE LINE AND CHARACTER POSITION.

GETCPY:	PUSHJ	PP,GETKAR
GETCP0:
	;ENTRY TO SAY YOU HAVE TAKEN CHARACTER WHHEN YOU ALREADY HAVE IT
	CAIN	CH," "
	POPJ	PP,

PCONVL:				; [211]
	IDPB	CH,PICPTR	;STORE ACTUAL CHAR.
	CAIL	CH,"a"		;CONVERT LC A-Z TO UC [211]
	CAILE	CH,"z"	 	;[211]
	CAIA			; [211]
	TRZ	CH,40		; CHANGE LC TO UPPER [211]
	MOVEM	LN,SAVLN1
	MOVEM	CP,SAVCP1
	POPJ	PP,
;CONSTANTS USED BY ROUTINE

FLTIND=3			;THE MASK BYTE FOR A FLOAT
ENDPIC=17			;THE MASK BYTE FOR PICTURE TERMINATION
PICXA=00			;THE MASK BYTE FOR "X" OR "A"
PICIS=10			;THE MASK BYTE FOR SIGN INSERT
PIC9=01				;THE MASK BYTE FOR "9"
PICRPT=16			;REPEAT MASK CODE
PICV=14				;V MASK CODE
	
MSKPTR:	POINT	4,MSKWRD,11	;BYTE POINTER TO FIRST MASK BYTE
MSKSYN:	POINT	6,MSKWRD,5	;BYTE POINTER TO SIGN CHARACTER
MSKFLT:	POINT	6,MSKWRD,11	;BYTE POINTER TO FLOAT OR SUPPRESSION CHARACTER


EXTERNAL	INSIZE, EXSIZE, DPSIZE, MSKWRD, MSKSIZ, SAVEAC, MAXWSS
EXTERNAL	WORDCP, WORDLN, SAVCP1, SAVLN1, SAVBCP, SAVBLN
;THE FOLLOWING ARE BITS SET BY THE SCANNER

%NOTLD=1B0	;IF SET THE FIXED INSERT SIGN BEFORE 9 OR . IS NOT LEADING
%P=1B18		;FOR "P"
%Z=1B19		;FOR "Z"
%STAR=1B20	;FOR "*"
%$=1B22		;FOR CURRENCY SIGN
%PLUS=1B24	;FOR "+"
%MINUS=1B26	;FOR "-"
%9=1B27		;FOR "9"
%V=1B28		;FOR "V"
%S=1B29		;FOR "S"
%X=1B30		;FOR "X" AND "A"
%CRDB=1B31	;FOR "CR" AND "DB"
%.=1B32		;FOR "."
%COMMA=1B33	;FOR ","
%0=1B34		;FOR "0"
%B=1B35		;FOR "B"

;THE FOLLOWING ARE USEFUL COMBINATIONS

%ZSTAR=%Z!%STAR			;THE SUPPRESSION CHARACTERS
%FLOAT=%PLUS!%MINUS!%$		;THE FLOATING CHARACTERS
%SPECI=%B!%0!%COMMA		;SPECIAL INSERTION CHARACTERS
%SPEC.=%ZSTAR!%FLOAT!%P		;TREATED DIFFERENTLY AFTER DECIMAL POINT

;SOMETIMES THE BITS ARE SHIFTED BY FOLLOWING AMOUNTS

SHIFTF=2			;WHEN A FLOAT CHARACTER
SHIFT9=1000			;WHEN AFTER A "9"
SHIFT.=20000			;WHEN AFTER THE DECIMAL PLACE
%ALLSH=<SHIFT.+1>*<SHIFTF+1>	;POSSIBLE SHIFTS FOR FLOATER

;MORE USEFUL COMBINATIONS

%SB9=<%PLUS!%MINUS>*<1!SHIFT.>	;SIGN BEFORE 9
%SA9=%SB9*SHIFT9		;SIGN AFTER 9
%SBP=<%PLUS!%MINUS>*SHIFTF	;SIGN FLOATED BEFORE DECIMAL POINT
%SAP=%SBP*SHIFT.		;SIGN FLOATED AFTER DECIMAL POINT
%ZBP=%Z!%STAR			;SUPPRESSION BEFORE DECIMAL POINT
%ZAP=%ZBP*SHIFT.		;SUPPRESSION AFTER DECIMAL POINT
%$BP=%$*SHIFTF			;CURRENCY SIGN FLOATING
%$AP=%$BP*SHIFT.		;CURRENCY SIGN FLOATING AFTER POINT (ILLEGAL)
%PAP=%P*SHIFT.			;"P" AFTER A DECIMAL POINT
%PBP=%P				;"P" BEFORE A DECIMAL POINT
%FCHAR=%ZBP!%ZAP!%SBP!%SAP!%$BP!%$AP	;A SUPPRESSION OR FLOAT CHARACTER
%FAP=<%SAP!%ZAP!%$AP>/1000000	;FLOATING OR SUPPRESSION AFTER POINT
%SIGN=%SB9!%SA9!%SBP!%SAP!%CRDB!%S	;ALL SIGNS
%EDIT=-1-<%PBP!%PAP!%9!%V!%S!%X>	;CHARACTERS WHICH CAUSE EDITING
%CNTIN=%9!%X!%FCHAR		;THESE COUNT AS INTERNAL CHARACTERS
%CNTDP=%9!%FCHAR!%PAP		;THESE COUNT AS DECIMAL PLACES
;%FB9AP=%FLOAT*SHIFT./1000000	;+, - OR $ BEFORE 9 BUT AFTER POINT
%CHGEN=-1-<%PAP!%PBP!%V!%S>	;THINGS WHICH DEFINE CHARACTERS IN DESTINATION FIELD
;A TABLE  OF ALLOWABLE CHARACTERS WITHOUT SPECIAL CHARACTERISTICS
;	LH IS RELATIVE POSITION OF ENTRY IN PICT2 AND PICT3.
;	RH IS THE CHARACTER IN ASCII

PICT1:	XWD	00,"X"	;X
	XWD	01,"9"	;9
	XWD	02,"V"	;V
	XWD	14,"-"	;-
	XWD	03,"S"	;S
	XWD	20,"Z"	;Z
	XWD	21,"*"	;*
	XWD	12,"+"	;+
	XWD	04,"B"	;B
	XWD	05,"0"	;0
	XWD	00,"A"	;A
	XWD	22,"P"	;P
IFN ANS74,<
	XWD	23,"/"	;/
>
PICT1S=.-PICT1


;A TABLE OF MASK VALUES AND INITIAL BIT SETTINGS
;	LH IS INITIAL BIT SETTING FOR "PC"
;	RH IS THE MASK VALUE FOR THE CHARACTER.

NORPT=1B18			;CHARACTER CANNOT BE REPEATED

PICT2:	XWD	%X,00		;X AND A
	XWD	%9,01		;9
	XWD	%V,14!NORPT	;V
	XWD	%S,00!NORPT	;S (NOT USED IN MASK)
	XWD	%B,05		;B
	XWD	%0,06		;0
	XWD	%COMMA,04	;COMMA
	XWD	%.,11!NORPT	;DECIMAL POINT
	XWD	%CRDB,12!NORPT	;CREDIT
	XWD	%CRDB,13!NORPT	;DEBIT
	XWD	%PLUS,10!NORPT	;+ INSERT
	XWD	0,FLTIND	;+ FLOATED
	XWD	%MINUS,10!NORPT	;- INSERT
	XWD	0,FLTIND	;- FLOATED
	XWD	%$,07!NORPT	;CURRENCY SIGN INSERT
	XWD	0,FLTIND	;$ FLOATED
	XWD	%Z,02		;Z
	XWD	%STAR,02	;*
	XWD	%P,00		;P (NOT USED IN MASK)
IFN ANS74,<
	XWD	%0,15		;/
>
VALAP==%PAP!%ZAP!%SAP!%$AP
VALBP==%PBP!%ZBP!%SBP!%$BP
VALS1==1+SHIFT.
VALS2==SHIFTF*VALS1
VALX1==%SPECI!%.!%$!%PAP!%V
VALX2==%SPECI!%.!%SB9!%PAP!%V

; THE TRUTH TABLE

DEFINE FALSE (A), <EXP A>			;THESE FLAGS ARE NOT ALLOWED

DEFINE TRUE (A),<EXP -1-A>			;THESE FLAGS ARE ALLOWED

DEFINE TPICT3,<
XLIST

TRUE  <%B!%0!%X!%9>				;X,A
FALSE <%SA9!%CRDB!%PBP!%ZAP!%SAP!%$AP!%NOTLD>	;9
FALSE <%.!%X!%PAP!%V!%ZAP!%SAP!%$AP>		;V
TRUE 0						;S
FALSE <%SA9!%CRDB!%PBP!%S>			;B
;I'M NOT REALLY SURE THAT ANSI 68 SHOULDN'T ALSO PROHIBIT
; THE ZERO, BUT DID IT THIS WAY TO BE SAFE.
IFN ANS68,<
FALSE <%SA9!%CRDB!%PBP>				;0
>
IFN ANS74,<
FALSE <%SA9!%CRDB!%PBP!%S>			;0
>
FALSE <%SA9!%CRDB!%X!%PBP!%S>			;COMMA
TRUE <%SPECI!%SB9!%$!VALBP!%9!%NOTLD>		;DECIMAL POINT
FALSE <%SIGN!%X>				;CR
FALSE <%SIGN!%X>				;DB
FALSE <%SIGN!%X>				;INSERT + BEFORE 9,.
TRUE  <%SPECI!%$!<%PLUS*SHIFTF>!%NOTLD>		;FLOAT + BEFORE 9,.
FALSE <%SIGN!%X>				;INSERT - BEFORE 9,.
TRUE  <%SPECI!%$!<%MINUS*SHIFTF>!%NOTLD>	;FLOAT - BEFORE 9,.
TRUE  <%SPECI!%SB9!%PAP!%V>			;INSERT $ BEFORE 9,.
TRUE  <%SPECI!%SB9!%$BP>			;FLOAT $ BEFORE 9,.
TRUE  <%SPECI!%SB9!%$!%Z>			;Z BEFORE POINT
TRUE  <%SPECI!%SB9!%$!%STAR>			;* BEFORE POINT
FALSE <%.!%X!%V!VALAP!%NOTLD>			;P BEFORE POINT
FALSE <%SIGN!%X>				;INSERT + AFTER 9, BEFORE .
TRUE 0						;FLOAT + AFTER 9, BEFORE .
FALSE <%SIGN!%X>				;INSERT - AFTER 9, BEFORE .
TRUE 0						;FLOAT - AFTER 9, BEFORE .
FALSE <%SIGN!%X>				;INSERT + BEFORE 9, AFTER .
TRUE	<VALX1!<%PLUS*VALS2>!%NOTLD>		;FLOAT + BEFORE 9, AFTER .
FALSE <%SIGN!%X>				;INSERT - BEFORE 9, AFTER .
TRUE	<VALX1!<%MINUS*VALS2>!%NOTLD>		;FLOAT - BEFORE 9, AFTER .
TRUE  <%SB9!%PAP!%V!%.>				;INSERT $ BEFORE 9, AFTER .
TRUE	<VALX2!%$BP!%$AP>			;FLOAT $ BEFORE 9, AFTER .
TRUE	<VALX2!%$!<%Z*VALS1>>			;Z AFTER POINT
TRUE	<VALX2!%$!<%STAR*VALS1>>		;* AFTER POINT
TRUE  <%PAP!%S!%V!%$!%PLUS!%MINUS>		;P AFTER POINT
FALSE <%SIGN!%X>				;INSERT + AFTER 9,.
TRUE 0						;FLOAT + AFTER 9,.
FALSE <%SIGN!%X>				;INSERT - AFTER 9,.
TRUE 0						;FLOAT - AFTER 9,.


LIST>

PICT3:	TPICT3;

END