$ goto start $!*************************************************************************** $!* * $!* DCL_CHECK.COM * $!* * $!* Copyright 1995, 2003 Hewlett-Packard Development Company, L.P. * $!* * $!* Confidential computer software. Valid license from HP and/or * $!* its subsidiaries required for possession, use, or copying. * $!* * $!* Consistent with FAR 12.211 and 12.212, Commercial Computer Software, * $!* Computer Software Documentation, and Technical Data for Commercial * $!* Items are licensed to the U.S. Government under vendor's standard * $!* commercial license. * $!* * $!* Neither HP nor any of its subsidiaries shall be liable for technical * $!* or editorial errors or omissions contained herein. The information * $!* in this document is provided "as is" without warranty of any kind and * $!* is subject to change without notice. The warranties for HP products * $!* are set forth in the express limited warranty statements accompanying * $!* such products. Nothing herein should be construed as constituting an * $!* additional warranty. * $!* * $!*************************************************************************** $! $! Abstract: Check DCL procedures for certain errors that are easy $! to make and/or potentially difficult to locate, as well $! as some others. Note that this procedure cannot $! guarantee total correctness. $! $! Author: Charles W Hammond $! $! Created: October/November 1996 $! $! Inputs: P1 is the procedure file to be checked. $! Current default directory is assumed if not included. $! .COM extension is assumed if not included. $! $! If P1 is not entered or if the file is not found $! the procedure prompts for the file. $! $! If P1 contains "HELP" (not case sensitive) the DCL_CHECK $! help subsystem is accessed. $! $! If P1 is a file name (i.e. not "HELP) then P2 may contain $! the name for a listing file. If p2 is not supplied, the $! listing is displayed on SYS$OUTPUT. $! $! if P1 contains "HELP", then P2 through P8 may contain $! help topics and sub-topics. $! $! SYS$SCRATCH is used for intermediate work files. $! $! Outputs: Listing of errors. $! $! $! MODIFICATION HISTORY $! $! V3.3 2004-Nov-22 Charlie Hammond $! Change version to V. $! $! B3.3 2004-Nov-15 Charlie Hammond $! Add $! DCL_CHECK COMMENT "tag" to add comments to diagnostic list. $! $! A3.3 2004-Nov-10 Charlie Hammond $! Improve CCN to avoid some unnecessary diagnostics. $! Improved check for perceives to eliminate some false PSQ diagnostics. $! $! V3.2 2004-Aug-23 Charlie Hammond $! No change from K3.2. V3.2 will be sumitted to the FREEWARE CD. $! $! K3.2 2004-Aug-19 Charlie Hammond $! Implement Ed Millers SLAC-J3.2 version. $! This is another fine piece of work by Ed. Thanks! $! SLAC-J3.2 2004-AUG-12 Ed Miller (esm@slac.stanford.edu) $! Split UMP into two checks: UMP and USP (for parentheses $! outside and inside quoted strings). Similarly for $! bracket tests UMB and USB. $! $! J3.2 2004-APR-19 Charlie Hammond $! Fix problems with similar lables in different subroutines $! $! I3.2 2004-APR-07 Charlie Hammond $! -- Thanks to Paddy O'Brien for reporting these problems. $! Fix parsing of DECK w/ /DOLLARS $! Do check for UMB in quoted strings (previously did not) $! Avoid LNS for label on CALL $! Avoid ONW for file with READ/DELETE $! Don't create a shadow for EXIT = and GOTO = assignments. $! $! H3.2 2004-APR-05 Charlie Hammond $! [no verison change] $! Fix one check for " "")""" $! $! H3.2 2004-APR-02 Charlie Hammond $! [no verison change] $! Don't use PIPE command on pre V7.1 systems $! $! H3.2 2004-MAR-05 Charlie Hammond $! Add LNS-S errlr -- $! referenced label is not in this subroutine $! $! H3.1-B 2004-FEB-18 Charlie Hammond $! Fix problem detecting apparent labels that contain ( or " $! $! H3.1-A 2003-DEC-30 Charlie Hammond $! (version not changed) $! Dislay elapsed time on systems that support F$DELTA_TIME $! $! H3.1-A 2003-DEC-10 Charlie Hammond $! Updated help to reflect improvements made in H3.1. $! Change code to eliminate a UMP diagnostic when DCL_CHECK $! is used to check itself. $! Improved error message (file name) for some cases where $! the input file is not found. $! $! H3.1 2003-DEC-03 Ed Miller $! Add or remove blanks from dcl_record to avoid invalid diagnostics $! $! Many thanks to Ed Miller, Stanford University, for his efforts to $! report many of the problems fixed in versions F3.1 and G3.1, $! and for testing (ofter re-testing) these fixes. $! $! G3.1 2003-NOV-03 Charlie Hammond $! Parse input file better to allow for logical search list $! Catch and warn about wildcard characters $! Improve check for SUBROUTINE and ENDSUBROUTINE $! Handle a label NOT followed by a space $! Avoid confusing @: as a lable $! $! F3.1 2003-OCT-08 Charlie Hammond $! Most of the fixes in these version are the result of errors $! reported by Ed Miller -- esm@slac.stanford.edu -- Thanks, Ed! $! $! Use F$LOCATE to indentify SUBROUTINE and ENDSUBROUTINE $! to avoind confusion with symbols starting with SUBRxxx or ENDSxxx $! out-> if $extract(0,4,token) .eqs. "SUBR" $! out-> if $extract(0,4,token) .eqs. "ENDS" $! new-> if f$locate(token,"SUBROUTINE") .eq. 0 $! new-> if f$locate(token,"ENDSUBROUTINE") .eq. 0 $! $! Convert a PIPE to inline code so that we don't need to $! redefine SYS$OUTPUT in other than user mode. $! (This preserves a customer SYS$OUTPUT definition.) $! $! Add a display of DCL_CHECK$* logicals to file listing. $! $! Allow for the possibility that OPEN, READ, WRITE and CLOSE $! may be used as variables. $! $! remove handling of /OUT -- this is done by DCL, which sees $! /out as a qualifier on the @ command. $! $! Look for "$GOTO" as well as "GOTO" to find labels $! $! Add OCE diagnostic for /END on OPEN or CLOSE $! $! Avoid possible infinite loop trying to prompt for input file $! $! Find I/O verbs on lines with labels $! $! E3.1 2003-SEP-17 Charlie Hammond $! Fixed false "END IF" such as $! $ Read/Error=3DCheck_MEP_End IF Rec $! ^^^^^^ $! Fixed problem with operators not preceded/followed by spaces $! $ If.not.condition $! Fixed some false truncated lexical problems $! $! D3.1 2003-AUG-21 Charlie Hammond $! Fixed some false instances of UMP and PSQ/PSR errors. $! $! C3.1 2003-AUG-20 Charlie Hammond $! Avoid flagging possessives ("'s " in quoted string) as $! PSQ-W possible error using single-quote (') in quoted string $! $! B3.1 2003-AUG-20 Charlie Hammond $! Re-write code that finds the file-name logical-name on I/P verbs $! so that it can handle quoted strings and symbol substitution $! within quoted strings. $! $! A3.1 2003-JUN-05 Charlie Hammond $! Improved checking for paired brackets $! by not checking in quoted strings $! $! V3.0 2003-JUN-03 Charlie Hammond $! Add checking for paired brackets $! $! H3.0 2003-APR-03 Charlie Hammond $! Added the contraction "They've". $! Added all contractions in all upper case. $! Added possessives for common accounting periods. $! "Day's", "Week's", "Month's", "Period's", "Quarter's", $! "Half's" and "Year's" (also in all lower and all upper case). $! Also names of weekdays and months $! $! G3.0 2003-APR-01 Charlie Hammond $! Fixed various spelling errors/typos $! Fixed aproblem with ICO (Help from Norm Raphael @metso.com) $! $! F3.0 30-Sep-2002 Charlie Hammond $! Don't do file checking on SYS$COMMAND, SYS$ERROR or SYS$PIPE $! (In addition to SYS$INPUT and SYS$OUTPUT) $! Fix a problem with double quotes in symbol used w/ READ/KEY $! $! E3.0 16-Sep-2002 Charlie Hammond $! Avoid false diagnostics ICO-S for ">" and "<" in PIPEs $! Only check /END and /ERR labels on /END_OF_FILE and /ERROR $! $! D3.0 04-Sep-2002 Charlie Hammond $! added " .ea ", " ea. " and " .ea. " to list of invaled $! comparison operators. $! $! C3.0 18-Mar-2002 Charlie Hammond $! Fix problem getting file logical names when there is an $! /END or /ERR qualifier with spaces preceding of following the "=" $! $! B3.0 13-Mar-2002 Charlie Hammond $! Fix problem with /ERROR on RUN statement $! $! A3.0 18-Feb-2002 Charlie Hammond $! Add file verb checking $! NLN-S An OPEN, READ, WRITE or CLOSE statement has no logical name $! ONC-E A file that is opened has no close statement $! INR-A file that is opened for read is not read $! ANR-E A file that is opened for append is not written $! ONW-E A file that is opened for write is not written $! RNR-S A file that is not opened for read is being read $! WNW-S A file that is not opened for write or append is being written $! UNU-S A file that is not opened for read and write is being updated $! CNO-S A file that is closed has no open statement $! $! NOTE: There is no V2.2 -- A3.0 added sufficient functionality $! that the major version was bumped. $! $! G2.2 20-Nov-2001 Charlie Hammond $! Make an "&" in a PIPE statement a warning. $! Re-word several ICF diagnostics. $! Include this in help. $! $! F2.2 02-Aug-2001 Charlie Hammond $! Spelling corrections, mostly in comments, suggested $! by Norm.Raphael@jamesbury.com $! $! E2.2 28-Jun-2001 Charlie Hammond $! Improve line number display in pass 2 $! Display 1000, 2000, etc. instead of 500, 1500, etc. $! Also display line numbers in round 100's instead $! of actual numbers that could be off a bit. $! $! D2.2 08-Jun-2001 Charlie Hammond $! Fix problem mistaking "CALL", "GOTO" and "GOSUB" $! at the end of a symbol name as a DCL command. $! $! C2.2 31-May-2001 Charlie Hammond $! Fix problem with /END and /ERR on same line. $! $! B2.2 22-Feb-2001 Charlie Hammond $! Fix problems with detecting various forms of $DECK and $EOD $! $! A2.2 15-Aug-2000 Charlie Hammond $! Fix problem when first blank delimited token on line $! ends with ":" but is not a label. $! $! V2.1 15-Aug-2000 Charlie Hammond $! Updated freeware release $! $! H2.1 07-Aug-2000 Charlie Hammond $! Make ICF as warning for & and % $! Add information to ICF help. $! $! G2.1 03-Aug-2000 Charlie Hammond $! Allow for a $ in a continued, "one-line" if statement. $! Avoid incorrect INT err that could happen if a PSQ error $! is found in a nested "one-line" if statement. $! $! F2.1 02-May-2000 Charlie Hammond $! Add ICF invalid character found (#, %, ^ or &) $! $! E2.1 24-Mar-2000 Charlie Hammond $! Improved EFB detection ("=" found between IF and THEN) $! $! D2.1 13-Dec-1999 Charlie Hammond $! Improve detection of single quote errors $! Allow /OUT= on output file (P2) $! $! B2.1 04-Dec-1999 Charlie Hammond $! Add LDS error -- Label defined by symbol substitution (warning) $! Fix single quote (') in definition of valid_lexicals $! It should be and now is a comma (,). $! A2.1 23-Oct-1999 Charlie Hammond $! Correct calculation of code_lines. $! We had been subtracting deck_lines twice. $! $! V2.0 17-Sep-1999 Charlie Hammond $! For FREEWARE release $! $! B2.0 31-Jul-1999 Charlie Hammond $! Correct handling of continuation comment that starts $! with only a "!" rather than "$!" $! Correct handling of SRT and INT errors. $! (SUBROUTINE/IF not terminated) $! "Beef up" a few help entries. $! $! A2.0 28 July 1999 Charlie Hammond $! Improve handling for TNA/ENA/DNA $! (THEN/ELSE/ENDIF statement not allowed here) $! Make ENDSUBROUTINE cancel goto and exit shadows $! Handle SUBROUTINE and ENDSUBROUTINE $! IF/ENDIF and SUBR/ENDS may be disjoint or $! strictly nested -- they may not overlap $! $! A1.0 - R1.0 $! through October 1996 Charlie Hammond $! Many changes/additions $! $! X-1 dd-mmm-1996 Charlie Hammond $! Original procedure created. $! $! --------------------------------------------------------------------------- $! This $DECK remains after DCLDIET is run to give a hint... $DECK ************************************************************** Enter the command @DCL_CHECK HELP for help with this procedure. ************************************************************** $EOD $start: $! $! Set defautl SAVED_LINE for possible errors that occur early $ saved_line = 0 $! $! Make sure DCL verbs aren't unexpected symbols. $! $ set = "set" $ set symbol /scope=(nolocal,noglobal) $! $! Ensure that message text is being displayed $! $ sav_msg = f$environment("MESSAGE") $ set message /TEXT $! $! Set up to handle CTRL_Y and errors. $! $ sav_status = 1 ! Default to success $! $ on control_y then goto y_exit $ on warning then goto err_exit $! $ pid = f$getjpi("","PID") ! for unique file names $! $ say = "WRITE SYS$OUTPUT" $! $! Define a symbol since the quote character (") is hard to $! handle as a literal $ quote[0,8]=34 $! $! Also create a symbol for the single quote character (') $ s_quote[0,8]=39 $! $! And a symbol for TWO single quotes in a row ('') $ s2_quote = s_quote+s_quote $! $! And, everybody's favorite, a symbol for "''F$FA" $ s2_fao = s2_quote + "F$FA" $ s2_upr = s2_quote + "F$" $ s2_lwr = s2_quote + "f$" $! $! Plus the exclamation point and form-feed character () $ exclaim[0,8]=33 $ form_feed[0,8] = 12 $! $! And a couple to help eliminate false UMP when DCL_CHECKing DCL_CHECK $ if_lpar = "IF" + "(" $ rpar_then = ")" + "THEN" $! $! Set a default for $EOD -- i.e. assume no /DOLLARS is in effect $ d$w = "$EOD" $ d$dollars = 0 $! $! $! ------------------------------------------------------------------ $! $ dcl_ck_vers = "V3.3" $ say "" $ say - "-*- Charlie Hammond's unsupported DCL checker (Version ''dcl_ck_vers') -*-" $! $! Set up symbols to indicate that we are suppressing diagnostic messages. $! This avoids repeatedly translating the logicals. $! $ suppress_ANR = f$trnlnm("DCL_CHECK$SUPPRESS_ANR") $ suppress_BL = f$trnlnm("DCL_CHECK$SUPPRESS_BL") $ suppress_CCN = f$trnlnm("DCL_CHECK$SUPPRESS_CCN") $ suppress_CLD = f$trnlnm("DCL_CHECK$SUPPRESS_CLD") $ suppress_CLS = f$trnlnm("DCL_CHECK$SUPPRESS_CLS") $ suppress_CNO = f$trnlnm("DCL_CHECK$SUPPRESS_CNO") $ suppress_CRE = f$trnlnm("DCL_CHECK$SUPPRESS_CRE") $ suppress_CRG = f$trnlnm("DCL_CHECK$SUPPRESS_CRG") $ suppress_DFB = f$trnlnm("DCL_CHECK$SUPPRESS_DFB") $ suppress_DL = f$trnlnm("DCL_CHECK$SUPPRESS_DL") $ suppress_DNA = f$trnlnm("DCL_CHECK$SUPPRESS_DNA") $ suppress_EFB = f$trnlnm("DCL_CHECK$SUPPRESS_EFB") $ suppress_EFN = f$trnlnm("DCL_CHECK$SUPPRESS_EFN") $ suppress_ENA = f$trnlnm("DCL_CHECK$SUPPRESS_ENA") $ suppress_ICF = f$trnlnm("DCL_CHECK$SUPPRESS_ICF") $ suppress_ICO = f$trnlnm("DCL_CHECK$SUPPRESS_ICO") $ suppress_INR = f$trnlnm("DCL_CHECK$SUPPRESS_INR") $ suppress_INT = f$trnlnm("DCL_CHECK$SUPPRESS_INT") $ suppress_LC = f$trnlnm("DCL_CHECK$SUPPRESS_LC") $ suppress_LDS = f$trnlnm("DCL_CHECK$SUPPRESS_LDS") $ suppress_LFF = f$trnlnm("DCL_CHECK$SUPPRESS_LFF") $ suppress_LND = f$trnlnm("DCL_CHECK$SUPPRESS_LND") $ suppress_LNF = f$trnlnm("DCL_CHECK$SUPPRESS_LNF") $ suppress_LNR = f$trnlnm("DCL_CHECK$SUPPRESS_LNR") $ suppress_LNS = f$trnlnm("DCL_CHECK$SUPPRESS_LNS") $ suppress_LOD = f$trnlnm("DCL_CHECK$SUPPRESS_LOD") $ suppress_MEC = f$trnlnm("DCL_CHECK$SUPPRESS_MEC") $ suppress_NCL = f$trnlnm("DCL_CHECK$SUPPRESS_NCL") $ suppress_NED = f$trnlnm("DCL_CHECK$SUPPRESS_NED") $ suppress_NLN = f$trnlnm("DCL_CHECK$SUPPRESS_NLN") $ suppress_OCE = f$trnlnm("DCL_CHECK$SUPPRESS_OCE") $ suppress_ONA = f$trnlnm("DCL_CHECK$SUPPRESS_ONA") $ suppress_ONC = f$trnlnm("DCL_CHECK$SUPPRESS_ONC") $ suppress_ONW = f$trnlnm("DCL_CHECK$SUPPRESS_ONW") $ suppress_PML = f$trnlnm("DCL_CHECK$SUPPRESS_PML") $ suppress_PRQ = f$trnlnm("DCL_CHECK$SUPPRESS_PRQ") $ suppress_PSQ = f$trnlnm("DCL_CHECK$SUPPRESS_PSQ") $ suppress_PTL = f$trnlnm("DCL_CHECK$SUPPRESS_PTL") $ suppress_RLI = f$trnlnm("DCL_CHECK$SUPPRESS_RLI") $ suppress_RLS = f$trnlnm("DCL_CHECK$SUPPRESS_RLS") $ suppress_RNA = f$trnlnm("DCL_CHECK$SUPPRESS_RNA") $ suppress_RNR = f$trnlnm("DCL_CHECK$SUPPRESS_RNR") $ suppress_SNT = f$trnlnm("DCL_CHECK$SUPPRESS_SNT") $ suppress_TLS = f$trnlnm("DCL_CHECK$SUPPRESS_TLS") $ suppress_TML = f$trnlnm("DCL_CHECK$SUPPRESS_TML") $ suppress_TNA = f$trnlnm("DCL_CHECK$SUPPRESS_TNA") $ suppress_TRH = f$trnlnm("DCL_CHECK$SUPPRESS_TRH") $ suppress_UMB = f$trnlnm("DCL_CHECK$SUPPRESS_UMB") $ suppress_UMP = f$trnlnm("DCL_CHECK$SUPPRESS_UMP") $ suppress_UNU = f$trnlnm("DCL_CHECK$SUPPRESS_UNU") $ suppress_UPQ = f$trnlnm("DCL_CHECK$SUPPRESS_UPQ") $ suppress_USB = f$trnlnm("DCL_CHECK$SUPPRESS_USB") $ suppress_USP = f$trnlnm("DCL_CHECK$SUPPRESS_USP") $ suppress_WCT = f$trnlnm("DCL_CHECK$SUPPRESS_WCT") $ suppress_WNW = f$trnlnm("DCL_CHECK$SUPPRESS_WNW") $! $get_help: $! $! Make certain that P2 exists $! $ if f$type(p2) .eqs. "" then p2 = "" $! $ if f$edit(p1,"UPCASE") .nes. "HELP" then goto get_filename1 $! $! Invoke the DCL HELP utility $! $ help_dir = f$parse(F$ENVIRONMENT("PROCEDURE"),,,"DEVICE") - + f$parse(F$ENVIRONMENT("PROCEDURE"),,,"DIRECTORY") $ define/user sys$input sys$output $ if p2 .eqs. "" $ then $ help/libr='help_dir'dcl_check/nouser/prompt DCL_CHECK $ else $ help/libr='help_dir'dcl_check/nouser/prompt - 'p2' 'p3' 'p4' 'p5' 'p6' 'p7' 'p8' $ endif $! $! Reset Params $ p1 = "" $ p2 = "" $ p3 = "" $ p4 = "" $ p5 = "" $ p6 = "" $ p7 = "" $ p8 = "" $! $! $ say "" $ say - "-*- Charlie Hammond's unsupported DCL checker (Version ''dcl_ck_vers') -*-" $! $! Get the name of the file to be checked $! $get_filename1: $ get_file_count = 0 ! initialize counter $get_filename: $ if p1 .eqs. "" $ then $ if get_file_count .gt. 99 $ then $ say "" $ say "Too many attempts to input filename -- aborting..." $ say "" $ say "Do you have SYS$OUTPUT re-defined?" $ say - " If so, you MUST supply the file to be checked on the command line!" $ say "" $ goto common_exit $ endif $ say "" $ say "You can follow the name of the file to be checked with a name" $ say "for the report file (default is SYS$OUTPUT). (blank separated)" $ say "" $! Count how many times we get here $ get_file_count = get_file_count + 1 $ read sys$output /end=common_exit /err=get_filename in$ - /prompt= - "enter name of file (or HELP or EXIT): " $ in$ = f$edit(in$,"TRIM,COMPRESS") $ if f$element(0," ",in$) .gts. " " then p1 = f$element(0," ",in$) $ if f$element(1," ",in$) .gts. " " then p2 = f$element(1," ",in$) $ if f$element(2," ",in$) .gts. " " then p3 = f$element(2," ",in$) $ if f$element(3," ",in$) .gts. " " then p4 = f$element(2," ",in$) $ if f$element(4," ",in$) .gts. " " then p5 = f$element(2," ",in$) $ if f$element(5," ",in$) .gts. " " then p6 = f$element(2," ",in$) $ if f$element(6," ",in$) .gts. " " then p7 = f$element(2," ",in$) $ if f$element(7," ",in$) .gts. " " then p8 = f$element(2," ",in$) $ goto get_filename $ endif $! $! create a usage log for DCL_CHECK $! $!$ comm_dir = f$parse(F$ENVIRONMENT("PROCEDURE"),,,"DEVICE") - $! + f$parse(F$ENVIRONMENT("PROCEDURE"),,,"DIRECTORY") $!$ if f$search("''comm_dir'dcl_check.access_log") .eqs. "" - $! then create 'comm_dir'dcl_check.access_log $!$ open/error=start/append/share access$log 'comm_dir'dcl_check.access_log $!$ write access$log f$fao("!%D !12AS !AS",0,f$getjpi(0,"USERNAME"),p1) $!$ close access$log $! $ if f$edit(p1,"UPCASE") .eqs. "HELP" then goto get_help $ if f$edit(p1,"UPCASE") .eqs. "EXIT" then goto common_exit $! $! P1 may not contain wildcard characters (*, % or ?) $! Note that ^% is not a wildcard. $! $ wild$work = p1 $wild$loop: $ if f$locate("^%",wild$work) .lt. f$length(wild$work) $ then $ wild$work = wild$work - "^%" $ goto wild$loop $ endif $ if ( (f$locate("*",wild$work) .lt. f$length(wild$work) ) - .or. (f$locate("%",wild$work) .lt. f$length(wild$work) ) - .or. (f$locate("?",wild$work) .lt. f$length(wild$work) ) ) $ then $ say "" $ say "*** ''P1'" $ say "*** wildcards not suported!" $ p1 = "" $ goto get_filename $ endif $! $! dcl$file = f$parse(p1,".COM") $!! $ dcl$file = p1 $ if f$search (dcl$file,111) .eqs. "" $ then $ dcl$file = p1 + ".COM" $ if f$search (dcl$file,222) .eqs. "" $ then $ dcl$file = f$parse (p1,".COM",,,"SYNTAX_ONLY") $ endif $ endif $!! $ if f$search(dcl$file) .eqs. "" $ then $ say "*** Cannot find ''dcl$file'" $ p1 = "" $ goto get_filename $ endif $ dcl$file = f$search(dcl$file) $! $ if p2 .nes. "" $ then $ if f$parse(p2,".LIS") .eqs. "" $ then $ say "" $ say "*** ""''p2'"" is not a valid report filename" $ p1 = "" $ p2 = "" $ goto get_filename $ endif $ endif $! $! Check if PIPE is available on this verisn of OpenVMS $! PIPE was introduced in OpenVMS V7.1 $! $ use_pipe = 0 ! default to zero (0) -- no pipe $! Get OpenVMS major version $ mvmsv = f$element(0,".",f$extract(1,8,f$getsyi("VERSION"))) $ if mvmsv .lt. 7 $ then $ use_pipe = 0 ! set to zero (0) -- no pipe $ goto after_pipe_check $ endif $! $ if mvmsv .eq. 7 $ then $! get Openvms minor version $ nvmsv = - f$element(0,"-",f$element(1,".",f$extract(1,8,f$getsyi("VERSION")))) $ if nvmsv .eq. 0 $ then $ use_pipe = 0 ! set to zero (0) -- no pipe $ else $ use_pipe = 1 ! set to one (1) -- use pipe $ endif $ goto after_pipe_check $ endif $! $ if mvmsv .gt. 7 $ then $ use_pipe = 1 ! set to one (1) -- use pipe $ goto after_pipe_check $ endif $! $after_pipe_check: $! $ dcl_ck_time = f$time() $ say "Checking file ''dcl$file'" $ say "''dcl_ck_time'" $ say "" $! $ say "Checking for DCL_CHECK$ logicals..." $ if use_pipe $ then $ pipe sho log DCL_CHECK$* | sear sys$input DCL_CHECK$ $ else $ sho log DCL_CHECK$* $ endif $ say "" $! $! $! Open the error file -- this will later be used to create the listing. $! $! Attempt to close it in case it was left open $ close/err=open_error1 err_file $open_error1: $! delete any "left over" error files $ if f$search("sys$scratch:dcl$error_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$error_'pid'.tmp;* $! $! Use CREATE and OPEN/APPEND instead of OPEN/WRITE because CREATE $! results in the desired file characteristics. $ create sys$scratch:dcl$error_'pid'.tmp $ open /append err_file sys$scratch:dcl$error_'pid'.tmp $! $! Create and open an indexed file to contain names of all labels defined $! Format of this file is: $! $! xnnnnnlll... $! $! x -- 0 initially; 1 when label is referenced $! nnnnn -- Five digit (ASCII) line number $! lll... -- The label (max of 255 chars) $! This is the key (Ascending String) $! $! Attempt to close it in case it was left open $ close/err=open_label1 label_file $open_label1: $! delete any "left over" label files $ if f$search("sys$scratch:dcl$label_''pid'.idx") .nes. "" then - delete /nolog sys$scratch:dcl$label_'pid'.idx;* $! $ create /fdl=sys$input sys$scratch:dcl$label_'pid'.idx $DECK FILE ORGANIZATION indexed RECORD CARRIAGE_CONTROL carriage_return FORMAT variable SIZE 261 AREA 0 ALLOCATION 54 BEST_TRY_CONTIGUOUS yes BUCKET_SIZE 9 EXTENSION 27 AREA 1 ALLOCATION 9 BEST_TRY_CONTIGUOUS yes BUCKET_SIZE 9 EXTENSION 9 KEY 0 CHANGES no DATA_AREA 0 DATA_FILL 100 DATA_KEY_COMPRESSION yes DATA_RECORD_COMPRESSION yes DUPLICATES yes INDEX_AREA 1 INDEX_COMPRESSION yes INDEX_FILL 100 LEVEL1_INDEX_AREA 1 PROLOG 3 SEG0_LENGTH 255 SEG0_POSITION 6 TYPE string $EOD $! $ open /read /write label_file sys$scratch:dcl$label_'pid'.idx $! $! $! Create and open an indexed file to contain information about $! "file operations" -- OPEN, CLOSE, READ and WRITE $! Format of this file is: $! $! iaorwuclll... $! (iaorwuc are 0 if the action not found, and 1 if it is) $! i -- OPEN for Input (/READ -- default) $! a -- OPEN for APPEND $! o -- OPEN for Output (/WRITE) $! r -- READ $! w -- WRITE (w/o /UPDATE) $! u -- WRITE /UPDATE $! c -- Close $! lll... -- The logical name used to reference the file $! (max of 255 chars) $! This is the key (Ascending String) $! $! Attempt to close it in case it was left open $ close/err=open_file_file file_file $open_file_file: $! delete any "left over" file files $ if f$search("sys$scratch:dcl$file_''pid'.idx") .nes. "" then - delete /nolog sys$scratch:dcl$file_'pid'.idx;* $! $ create /fdl=sys$input sys$scratch:dcl$file_'pid'.idx $DECK FILE ORGANIZATION indexed RECORD CARRIAGE_CONTROL carriage_return FORMAT variable SIZE 262 AREA 0 ALLOCATION 30 BEST_TRY_CONTIGUOUS yes BUCKET_SIZE 6 EXTENSION 12 AREA 1 ALLOCATION 6 BEST_TRY_CONTIGUOUS yes BUCKET_SIZE 6 EXTENSION 6 KEY 0 CHANGES no DATA_AREA 0 DATA_FILL 100 DATA_KEY_COMPRESSION yes DATA_RECORD_COMPRESSION yes DUPLICATES no INDEX_AREA 1 INDEX_COMPRESSION yes INDEX_FILL 100 LEVEL1_INDEX_AREA 1 PROLOG 3 SEG0_LENGTH 255 SEG0_POSITION 7 TYPE string $EOD $! $ open /read /write file_file sys$scratch:dcl$file_'pid'.idx $! $! Create and open an indexed file to contain the line number range $! of all multi-line IF/THEN/ELSE statements. $! Format of this file is: $! $! iiiiieeeee $! $! iiiii -- Five digit (ASCII) line number of the IF $! This is the key (Descending String) $! eeeee -- Five digit (ASCII) line number of the ENDIF $! $! Attempt to close it in case it was left open $ close/err=open_if1 if_file $open_if1: $! delete any "left over" if files $ if f$search("sys$scratch:dcl$if_''pid'.idx") .nes. "" then - delete /nolog sys$scratch:dcl$if_'pid'.idx;* $! $ create /fdl=sys$input sys$scratch:dcl$if_'pid'.idx $DECK FILE ORGANIZATION indexed RECORD CARRIAGE_CONTROL carriage_return FORMAT fixed SIZE 11 AREA 0 ALLOCATION 6 BEST_TRY_CONTIGUOUS yes BUCKET_SIZE 3 EXTENSION 3 AREA 1 ALLOCATION 3 BEST_TRY_CONTIGUOUS yes BUCKET_SIZE 3 EXTENSION 3 KEY 0 CHANGES no DATA_AREA 0 DATA_FILL 100 DATA_KEY_COMPRESSION no DATA_RECORD_COMPRESSION no DUPLICATES no INDEX_AREA 1 INDEX_COMPRESSION no INDEX_FILL 100 LEVEL1_INDEX_AREA 1 PROLOG 3 SEG0_LENGTH 5 SEG0_POSITION 0 TYPE dstring $EOD $! $ open /read /write if_file sys$scratch:dcl$if_'pid'.idx $! $! Open the intermediate work file $! This will be a copy of the input file with the statement number $! added at the beginning, with comments removed, with quoted strings $! removed, with $DECK/$EODgroups removed, and with any records that $! are blank (except for starting $, $! or "$ !) removed. $! $! Attempt to close it in case it was left open $ close/err=open_work1 work_file $open_work1: $! delete any "left over" work files $ if f$search("sys$scratch:dcl$work_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$work_'pid'.tmp;* $! $! Use CREATE and OPEN/APPEND instead of OPEN/WRITE because CREATE $! results in the desired file characteristics. $ create sys$scratch:dcl$work_'pid'.tmp $ open /append work_file sys$scratch:dcl$work_'pid'.tmp $! $! Now open the DCL file to be checked for the first pass. $! $! Attempt to close it in case it was left open $ close/err=open_dcl1 dcl_file $open_dcl1: $ open /read dcl_file 'dcl$file' $! $! "Initialize" some symbols $! $ exit_shadow = 0 ! lines are not reachable due to preceding EXIT $ goto_shadow = 0 ! lines are not reachable due to preceding GOTO $ call$call = 0 ! Assume it is NOT a "CALL" $! $ line_number = 0 ! line (record) number w/i DCL procedure $ total_lines = 0 ! save count of total lines $ code_lines = 0 ! save count of code lines $ commented_code_lines = 0 ! save count of code lines w/comments $ continuation_lines = 0 ! save count of continuation lines $ deck_lines = 0 ! save count of lines w/i $DECK/$EOD pairs $ comment_lines = 0 ! save count of comment lines $ blank_lines = 0 ! save count of blank lines $ in_deck = 0 ! indicates that we aren't within a $DECK/$EOD $ deck_line = 0 ! line number of last valid $DECK $! $ if_level = 0 ! IF nesting level (index into then_ and else_level) $! IF_LEVEL tracks the combined levels of IF and SUBROUTINES. $! SR_LEVEL counts how many of IF_LEVEL are SUBROUTINES. $ sr_level = 0 $! $ if_lines = "" ! line numbers of prior if statements $! then_ and else_level[if_level,1] = 1 when if/then is found $ then_level = "000000000000000000000" ! (level 1 is always satisfied) $ else_level = "000000000000000000000" ! (level 1 is always satisfied) $! ifsr_level[if_level,1] = "I" or "S" for IF or SUBROUTINE $ ifsr_level = "000000000000000000000" $! $ valid_lexicals = - "F$CONTEXT,F$CSID,F$CVSI,F$CVTIME,F$CVUI,F$DELTA_TIME,F$DEVICE,F$DIRECTORY," + - "F$EDIT,F$ELEMENT,F$ENVIRONMENT,F$EXTRACT,F$FAO,F$FILE_ATTRIBUTES," + - "F$GETENV,F$GETDVI,F$GETJPI,F$GETQUI,F$GETSYI,F$IDENTIFIER,F$INTEGER," + - "F$LENGTH,F$LOCATE,F$LOGICAL,F$MESSAGE,F$MODE,F$PARSE,F$PID," + - "F$PRIVILEGE,F$PROCESS,F$SEARCH,F$SETPRV,F$STRING,F$TIME,F$TRNLNM," + - "F$TYPE,F$USER,F$VERIFY" $ length_valid_lexicals = f$length(valid_lexicals) $! $! $ Say "Starting Pass 1 -- ''f$time()' ..." $! $! This pass reads the DCL file and does the following: $! $! Copies the file leaving out $! comments $! $DECKs $! quoted strings $! Blank lines $! $! collects LABEL info and FILE info (open/close/read/write) $! $! Checks for $! $! BL--I blank line (warning) $! CCN-I Continuation character (""-"") not preceded by space (warning) $! CLD-S continuation line starts with ""$"" $! CLS-I comment line separates continuation line (warning) $! CRE-W code cannot be reached due to EXIT at line !UL $! CRG-W code cannot be reached due to GOTO at line !UL $! DFB-I $DECK found between $DECK and $EOD $! DL--E duplicate label ""!AS"" $! DNA-S ENDIF statement not allowed here $! EFB-S ""="" found between IF and THEN $! EFN-W $EOD found with no corresponding $DECK $! ENA-S ELSE statement not allowed here $! INT-S IF statement not terminated $! LDS-I Label ""!AS"" defined by symbol substitution (warning) $! LFF-I line contains only form-feed () $! LND-E line does not start with ""$"" $! LOD-I line contains only ""$"" (warning) $! MEC-S missing expression for comparison $! NCL-S no continuation line at EOF $! NED-S no $EOD for $DECK at !UL $! PML-E possible misspelled lexical (!AS) $! PRQ-E probable error using single-quote (') $! PSQ-W possible error using single-quote (') in quoted string $! PTL-E possible truncated lexical (!AS) $! RNA-S ENDSUBROUTINE statement not allowed here $! SNT-S SUBROUTINE statement not terminated $! TML-S too many levels of IF statements $! TMS-S too many levels of IF and SUBROUTINE statements $! TNA-S THEN statement not allowed here $! TRH-S THEN statement required here $! UMP-S unmatched parentheses $! UPQ-S unpaired quotation marks ("") $! WCT-E wrong constant type for comparison $! $ line_disply_increment = 500 $ line_to_display = line_disply_increment $read_dcl1: $! $ read /end=end_dcl1 dcl_file dcl_record $! count line (record) numbers $ line_number = line_number + 1 ! count every record $ saved_line = line_number ! save number in case line is continued $! $ if line_number .gt. line_to_display $ then $ say f$fao("...processing line number !UL...",line_to_display) $ line_to_display = line_to_display + line_disply_increment $ endif $! $! Insert any diagnostic comments into err_file. $! $! Lines that look like this: $! $! $! DCL_CHECK COMMENT: $! 111111111122 $! 0123456789012345678901 $! $! are diagnostic comments to be inserted into the error file. $! $ if f$edit(f$element(0,":",dcl_record),"TRIM,COLLAPSE,UPCASE") - .EQS. "$!DCL_CHECKCOMMENT" $ then $ dcl_comment = dcl_record - f$element(0,":",dcl_record) - ":" $ write err_file f$fao( - "!5UL -CMNT-!AS",saved_line,dcl_comment) $ goto read_dcl1 $ endif $! $! remove comments and trim/compress/upcase the line, check if in a deck $! $ if in_deck $ then $ dcl_record = f$edit(dcl_record,"COMPRESS,UPCASE") $ deck_lines = deck_lines + 1 ! Count lines in $DECK/$EOD pair $! $ else ! not in deck $! $ dcl_record = f$edit(dcl_record,"TRIM,COMPRESS,UPCASE") $! $! Check for line containing only "$" $ if dcl_record .eqs. "$" $ then $ if .not. suppress_LOD then write err_file f$fao( - "!5UL LOD-I line contains only ""$"" (warning)", - saved_line) $ goto read_dcl1 ! No more checking required $ endif $! $! Check for a blank lines $ if dcl_record .eqs. "" $ then $ if .not. suppress_BL then write err_file f$fao( - "!5UL BL--I blank line (warning)", - saved_line) $ blank_lines = blank_lines + 1 ! Count blank_lines $ goto read_dcl1 ! No more checking required $ endif $! $! Check for lines that contain only form-feed $ if dcl_record .eqs. form_feed $ then $ if .not. suppress_LFF then write err_file f$fao( - "!5UL LFF-I line contains only form-feed ()", - saved_line) $ goto read_dcl1 ! No more checking required $ endif $! $! Check for lines that are only comments $ if ( (f$extract(0,2,dcl_record) .eqs. "$!") - .or. (f$extract(0,3,dcl_record) .eqs. "$ !") ) $ then $! Is that ALL the line contains? $ if ( (dcl_record .eqs. "$!") - .or. (dcl_record .eqs. "$ !") ) $ then ! Its a blank line $ blank_lines = blank_lines + 1 ! Count blank lines $ else ! Its a comment line $ comment_lines = comment_lines + 1 ! Count comment lines $ endif $ goto read_dcl1 ! No more checking required $ endif $! $ endif $! $! Get the first two tokens on the line $! "$" and leading space has been removed from DCL_RECORD at this point $ t0 = f$element(0," ",dcl_record) $ t0 = f$element(0,"/",t0) $ t1 = f$edit(dcl_record - t0,"TRIM") $! $! Check for $DECK and $EOD -- don't write these lines $! NOTE: $DECK and $EOD command may NOT include labels $! and may NOT be nested. $! $ if ( (t0 .eqs. "$DECK") - .or. ( (t0 .eqs. "$") .and. (t1 .eqs. "DECK") ) ) $ then $ if in_deck $ then $ if .not. suppress_DFB then write err_file f$fao( - "!5UL DFB-I $DECK found between $DECK and ''d$w'", - saved_line) $ if (.not. suppress_NED) .and. (.not. d$dollars) - then write err_file f$fao( - "!5UL NED-S no ''d$w' for $DECK at line !UL", - saved_line,deck_line) $ else $ in_deck = 1 $ deck_line = saved_line $! $! Check for /DOLLARS $ d$w = f$edit(dcl_record,"COLLAPSE,UNCOMMENT") - "$DECK" $ d$w = f$element(1,"=",d$w) - """" - """" $ if d$w .eqs. "=" $ then ! no /DOLLARS $ d$w = "$EOD" ! assume the default $ d$dollars = 0 ! Remember we found /DOLLARS $ else ! /DOLLARS FOUND $! ! user the value found in d$w $ d$dollars = 1 ! remember we did not find /DOLLARS $ endif $! $ endif $! $DECK, $EOD and lines in a "deck" are not continued and are $! not written to the intermediate file. $ goto read_dcl1 $ endif $! $ if ( ( (d$dollars) .and. (t0 .eqs. "''d$w'") ) - .or. ( (.not. d$dollars) .and. (t0 .eqs. "''d$w'") ) - .or. ( (.not. d$dollars) .and. (t0 .eqs. "$") .and. (t1 .eqs. "EOD") ) ) $ then $ if in_deck $ then $ in_deck = 0 $ d$w = "EOD" $ deck_lines = deck_lines - 1 ! already counted but should not be $ else $ if .not. suppress_EFN then write err_file f$fao( - "!5UL EFN-W $EOD found with no corresponding $DECK", - saved_line) $ endif $! $DECK, $EOD and lines in a "deck" are not continued and are $! not written to the intermediate file. $ goto read_dcl1 $ endif $! $! $DECK, $EOD and lines in a "deck" are not continued and are $! not written to the intermediate file. $ if in_deck then goto read_dcl1 $! $! $read_dcl1_c: $! $! Loose "!" from ''F$FA....' $! $! We want to uncomment the dcl record, $! But first we must deal with a problem that F$EDIT has with $! exclamation marks used for formatting directives in F$FAO control strings. $! This only occurs when the F$FAO is preceded by two single quotes $! indicating symbol substitution within a quoted string. e.g. $! $! $ write sys$output "Value is: ''f$fao("!4UL",value)'" $! $! (This may not be good coding practice, but it IS used.) $! $! If a line contains the string "''F$FA" then we will remove any and ALL $! exclamation marks from the "''F$FA" to the second occurrence of a $! single double-quote (") character. The first double-quote starts the $! control string; the next SINGLE double-quote ends the control string. $! (The control string could contain DOUBLE double-quotes ("") representing $! output of a single double-quote.) $! $! Of course we only need to do all this if the line contains at least $! one instance of "''F$FA" -- so, lets check. $! Since is in a quoted string, it may be any combination of upper/lower case. $! $ x = f$locate(s2_lwr,dcl_record) $ if f$edit(f$extract(x,6,dcl_record),"UPCASE") .eqs. s2_fao - then goto do_fao $! $ x = f$locate(s2_upr,dcl_record) $ if f$edit(f$extract(x,6,dcl_record),"UPCASE") .eqs. s2_fao - then goto do_fao $! $! If no more instances of ;''F$+, we're done $ goto after_fao_check $! $! Otherwise we gotta do it... $! $do_fao: $! $ x = x + 2 ! to get all through '' $ work = f$extract (0,x,dcl_record) + "X$" ! get the beginning of the rec $ x = x + 2 ! to allow for the "F$" replaced by "X$" $! $ q_count = 0 $ next_char = f$extract(x,1,dcl_record) $ x = x + 1 $fao_loop: $! $! Are we done? $! $ if q_count .ge. 2 $ then ! we're done with this ''F$ $! Get the rest of the line $ work = work + f$extract(x,f$length(dcl_record)-x,dcl_record) $ dcl_record = work $! Loop back to check if there is another F$F to check $ goto read_dcl1_c $ endif $! $! Check the character $ char = next_char $ next_char = f$extract(x,1,dcl_record) $ x = x + 1 $! $ if char .eqs. quote $ then $! $ if next_char .eqs. quote $ then ! it's a double double-quote -- copy it $ work = work + quote + quote $! Cycle through the second quote $ next_char = f$extract(x,1,dcl_record) $ x = x + 1 $! $ else ! it's a single double-quote -- copy and increment q_count $ work = work + quote $ q_count = q_count + 1 $! $ endif $! $ else ! just copy the char, except exclamation point $ if char .nes. exclaim then work = work + char $! $ endif $! $! This loops if there is a missing closing quote, so... $! (a smaller number than 1024 would probably do...) $ if x .ge. 1024 then q_count = q_count + 1 $! $ goto fao_loop $! $after_fao_check: $! $! Now we can safely uncomment the line $ work = f$edit(dcl_record,"UNCOMMENT,TRIM") $ if work .nes. dcl_record $ then $ commented_code_lines = commented_code_lines + 1 $ dcl_record = work $ endif $! $! check for continuation $read_continuation: $ if f$extract(f$length(dcl_record)-1,1,dcl_record) .eqs. "-" $ then ! the line IS continued $ yes_cont = 1 $ read /end=no_cont dcl_file dcl_c_record $ line_number = line_number + 1 $ work = f$edit(dcl_c_record,"TRIM,COMPRESS") $ goto yes_cont $no_cont: $ work = "" $ yes_cont = 0 $yes_cont: $ if f$extract(0,2,work) .nes. "/" $ then $! DCL_CHECK COMMENT: USP O.K. in hext line $ if ( (f$extract(f$length(dcl_record)-2,2,dcl_record) .nes. " -") - .and. (f$extract(f$length(dcl_record)-2,2,dcl_record) .nes. ",-") - .and. (f$extract(f$length(dcl_record)-2,2,dcl_record) .nes. ")-") ) $ then $! $! If there is no space preceding the continuation $! character ("-"), and there is no space at the beginning of the $! continuation line, this can result in keywords, qualifiers or $! parameters being incorrectly concatenated. $! DCLDIETing a command procedure can cause these condition because $! DCLDIET removes all space at the beginning of the continuation line. $! $ if .not. suppress_CCN then write err_file f$fao( - "!5UL CCN-I Continuation character (""-"") not preceded by space (warning)", - line_number) $ endif $ endif $ if .not. yes_cont then goto end_dcl1_c $! Remove the trailing "-" $ dcl_record = f$extract(0,f$length(dcl_record)-1,dcl_record) $! $! Check if it is a comment only $ if ( (f$extract(0,2,work) .eqs. "$!") - .or. (f$extract(0,3,work) .eqs. "$ !") - .or. (f$extract(0,1,work) .eqs. "!") ) $ then ! It is a comment only -- issue warning and skip it. $ ! This condition causes DCLDIET to output invalid code. $ if .not. suppress_CLS then write err_file f$fao( - "!5UL CLS-I comment line separates continuation line (warning)", - line_number) $ comment_lines = comment_lines + 1 $ work = f$edit(dcl_c_record,"TRIM,COMPRESS") $ goto read_continuation $ endif $ continuation_lines = continuation_lines + 1 ! count continuation lines $! $! if f$extract(0,1,dcl_c_record) .eqs. "$" .and. - $! f$edit(f$extract(f$length(dcl_record)-6,6,dcl_record),"upcase,trim") - $! .nes. "THEN" $! $ if f$extract(0,1,dcl_c_record) .eqs. "$" $ then $ if .not. suppress_CLD then write err_file f$fao( - "!5UL CLD-S continuation line starts with ""$""", - line_number) $ endif $ dcl_record = dcl_record + dcl_c_record $! trim/compress/upcase the line $ dcl_record = f$edit(dcl_record,"TRIM,COMPRESS,UPCASE") $ goto read_dcl1_c $ endif $! $! Check DCL line for balanced (left/right) parentheses and $! brackets. We check the parts of the statement inside and outside quoted $! strings separately. $! $! For statement outside of quoted string: $! Demand left and right balance, and demand each left precede $! its matching right $! For part of statement within quoted string: $! Ignore singleton strings (see details below); $! Demand left and right balance, but make no requirement that $! each left precede a matching right $! $! If there are any parens/brackets in the string, we go thru and $! collect characters from non-quoted string and quoted-string $! part separately, omitting any singleton strings from the $! the quoted-string collection. $! $! Note on singleton strings: $! x=a+")"+a ! is considered singleton ")" $! x=a+"..."")"+a ! is NOT considered singleton ")" $! x=a+"(""..."+a ! is NOT considered singleton "(" $! $ if (dcl_record - "(" - ")" - "[" - "]") .nes. dcl_record $ then $ ump_fail = 0 ! tentatively assume UMP test passes $ umb_fail = 0 ! tentatively assume UMB test passes $ usp_fail = 0 ! tentatively assume USP test passes $ usb_fail = 0 ! tentatively assume USB test passes $! $ dcl_work1 = "" ! string to accumulate non-quoted parts of DCL line $ dcl_work2 = "" ! string to accumulate quoted parts (except singletons) $ n = 0 $! $! Separate the command line into two strings, that part inside and outside $! quoted strings. We ignore 'singleton' quoted strings. $! $split_quotstr: $ str0 = f$element (n,"""",dcl_record) ! an element of non-quoted part $ if str0 .nes. """" $ then $ dcl_work1 = dcl_work1 + str0 $! $ str1 = f$element (n+1,"""",dcl_record) ! an element of quoted part $ if str1 .nes. """" $ then $! $ if f$length(str1) .ne. 1 $ then $ dcl_work2 = dcl_work2 + str1 $ else $ if str0 .eqs. "" .or. - (f$element (n+2,"""",dcl_record) .eqs. "" .and. - f$element (n+3,"""",dcl_record) .nes. """") $ then $ dcl_work2 = dcl_work2 + str1 $ endif $ endif $! $ n = n + 2 $ goto split_quotstr $ endif $ endif $! $! For the non-quoted string, we flag an error if either the $! count of left and right delimiters does not match or $! if a left delimiter does not precede its matching right delimiter. $! For the quoted string, we do only the count test, not the $! order test. $! $! First, check the non-quoted string for parentheses balance, if necessary. $! $ if (dcl_work1 - "(" - ")") .nes. dcl_work1 $ then $ nl = 0 ! count of left delimiter so far $ nr = 0 ! count of right delimiter so far $! $next_lparen_elem: $ str = f$element (nl,"(",dcl_work1) $ if str .nes. "(" $ then $! $ if f$locate (")",str) .ne. f$length (str) $ then $ n = 2 $! $next_rparen_subelem: $ if f$element (n,")",str) .nes. ")" $ then $ n = n + 1 $ goto next_rparen_subelem $ endif $! $ nr = nr + n-1 $ if nr .gt. nl then ump_fail = 1 ! parentheses out of order $! $ endif ! if there are any ")" in this element of dcl_work1 $! $ nl = nl + 1 $ goto next_lparen_elem $! $ endif ! if there is more text to scan in dcl_work1 $! $ if nr .ne. nl-1 then ump_fail = 1 ! parentheses unbalanced $! $ endif ! if there are any parentheses in dcl_work1 $! $! $! Now check the non-quoted string for bracket balance, if necessary. $! $ if (dcl_work1 - "[" - "]") .nes. dcl_work1 $ then $ nl = 0 ! count of left delimiter so far $ nr = 0 ! count of right delimiter so far $! $next_lbracket_elem: $ str = f$element (nl,"[",dcl_work1) $ if str .nes. "[" $ then $! $ if f$locate ("]",str) .ne. f$length (str) $ then $ n = 2 $! $next_rbracket_subelem: $ if f$element (n,"]",str) .nes. "]" $ then $ n = n + 1 $ goto next_rbracket_subelem $ endif $! $ nr = nr + n-1 $ if nr .gt. nl then umb_fail = 1 ! brackets out of order $! $ endif ! if there are any "]" in this element of dcl_work1 $! $ nl = nl + 1 $ goto next_lbracket_elem $! $ endif ! if there is more text to scan in dcl_work1 $! $ if nr .ne. nl-1 then umb_fail = 1 ! brackets unbalanced $! $ endif ! if there are any brackets in dcl_work1 $! $! $! Now check the quoted string for parenthesis balance, if necessary. $! $ if (dcl_work2 - "(" - ")") .nes. dcl_work2 $ then $ n = 1 $count_lparen: $ if f$element (n,"(",dcl_work2) .nes. "(" $ then $ n = n + 1 $ goto count_lparen $ endif $! $ usp_fail = f$element (n, ")",dcl_work2) .nes. ")" .or. - f$element (n-1,")",dcl_work2) .eqs. ")" $! $ endif ! if any parentheses in quoted string part of dcl line $! $! Now check the quoted string for bracket balance, if necessary. $! $ if (dcl_work2 - "[" - "]") .nes. dcl_work2 $ then $ n = 1 $count_lbrack: $ if f$element (n,"[",dcl_work2) .nes. "[" $ then $ n = n + 1 $ goto count_lbrack $ endif $! $ usb_fail = f$element (n, "]",dcl_work2) .nes. "]" .or. - f$element (n-1,"]",dcl_work2) .eqs. "]" $! $ endif ! if any brackets in quoted string part of dcl line $! $! $! All four unbalanced delimiter tests have been made; report any failures. $! $ if ump_fail $ then $ if .not. suppress_UMP then write err_file f$fao( - "!5UL UMP-S unmatched parentheses (not in quoted string)", - saved_line) $ endif $! $ if umb_fail $ then $ if .not. suppress_UMB then write err_file f$fao( - "!5UL UMB-S unmatched brackets (not in quoted string)", - saved_line) $ endif $! $ if usp_fail $ then $ if .not. suppress_USP then write err_file f$fao( - "!5UL USP-W unmatched parentheses in quoted string", - saved_line) $ endif $! $ if usb_fail $ then $ if .not. suppress_USB then write err_file f$fao( - "!5UL USB-W unmatched brackets in quoted string", - saved_line) $ endif $! $ endif ! if any paren/brack in full dcl line $! $! $! Normaly there is space before and after the comparison operators, $! ".eq.", ".ge.", ".gt.", ".le.", ".lt.", ".ne." (numeric), $! and $! ".eqs.", ".ges.", ".gts.", ".les.", ".lts.", ".nes.", (string), $! and the logical operators, $! ".and.", ".or.", and ".not.". $! However, DCL does not require this and some code takes advantage $! of this, which "confuses" DCL_CHECK. $! So we will make sure there is a space before and after $! each of these operators. $! $! $! (The following code provided by Ed Miller -- esm@slac.stanford.edu) $! (code has been reformated slightly.) $! $! The code following may add or remove blanks from dcl_record $! in order to avoid invalid error reports that may otherwise $! occur in the syntax checking which follows. $! $! The first such change is applied only to statements with lexicals. $! If unnecessary blanks are not removed from lexical function $! references, under certain circumstances invalid PRQ or WCT $! diagnostics will be generated. The removal of unnecessary blanks $! here is applied to the whole statement, since it causes no harm. $! It is not applied in the absence of a lexical in the statement, $! so as not to needlessly extend execution time. $! $! The final pair of changes (changing "IF(" TO "IF (" and $! ")THEN" to ") THEN" is applied to all statements. These may $! appear in legal DCL code (and may also be generated when $! unnecessary blanks are removed from statements with lexicals). $! $ if f$locate ("F$",dcl_record) .ne. f$length(dcl_record) $ then $ dclxrecord = dcl_record $ lx = f$length (dcl_record) - 1 $loopx: $ n = f$locate (" ",dclxrecord) $ if n .lt. lx $ then $ if (f$loca (f$extr(n-1,1,dcl_record),"-+/*,()") .lt. 7 .or. - f$loca (f$extr(n+1,1,dcl_record),"-+/*,()") .lt. 7) $ then $ dcl_record = f$extr(0,n,dcl_record) + f$extr(n+1,-1,dcl_record) $ dclxrecord = f$extr(0,n,dclxrecord) + f$extr(n+1,-1,dclxrecord) $ lx = lx - 1 $ else $ dclxrecord = f$extr(0,n,dclxrecord) + "X" + - f$extr(n+1,-1,dclxrecord) $ endif $ goto loopx $ endif $ endif $! $ if f$locate (if_lpar,dcl_record) .ne. f$length(dcl_record) $ then $ p = f$locate (if_lpar,dcl_record) $ dcl_record = f$extr (0,p+2,dcl_record) + " " + f$extr(p+2,-1,dcl_record) $ endif $! $ if f$locate (rpar_then,dcl_record) .ne. f$length(dcl_record) $ then $ p = f$locate (rpar_then,dcl_record) $ dcl_record = f$extr (0,p+1,dcl_record) + " " + f$extr(p+1,-1,dcl_record) $ endif $! $! (End of code provided by Ed Miller.) $! $! $ oper = ".EQ." $ gosub oper_space $ oper = ".GE." $ gosub oper_space $ oper = ".GT." $ gosub oper_space $ oper = ".LE." $ gosub oper_space $ oper = ".LT." $ gosub oper_space $ oper = ".NE." $ gosub oper_space $ oper = ".EQS." $ gosub oper_space $ oper = ".GES." $ gosub oper_space $ oper = ".GTS." $ gosub oper_space $ oper = ".LES." $ gosub oper_space $ oper = ".LTS." $ gosub oper_space $ oper = ".NES." $ gosub oper_space $ oper = ".AND." $ gosub oper_space $ oper = ".OR." $ gosub oper_space $ oper = ".NOT." $ gosub oper_space $! $ goto after_oper $! $! $oper_space: $! $ oper_length = f$length(oper) $ oper_work = "" $! $! Can't fix this condition so skip it $ if f$locate("""''oper'""",dcl_record) .lt. f$length(dcl_record) - then return $! $oper_loop: $ oper_loc = f$locate(oper,dcl_record) $ if oper_loc .ge. f$length(dcl_record) $ then $ oper_work = oper_work + dcl_record $ dcl_record = oper_work $ return $ endif $ oper_work = oper_work + - f$edit(f$extract(0,oper_loc,dcl_record),"TRIM") + - " " + oper + " " $ dcl_record = f$edit(f$extract(oper_loc+oper_length,511,dcl_record),"TRIM") $ goto oper_loop $! $! $after_oper: $! $! $! check for lines not starting w/$ (and not deck or continuation) $! $ if f$extract(0,1,dcl_record) .nes. "$" $ then $ if .not. suppress_LND then write err_file f$fao( - "!5UL LND-E line does not start with ""$""", - saved_line) $! not written to the intermediate file; not checked any further. $ goto read_dcl1 $ endif $! $! Loose the "$" -- to make things simpler... $! $! Record starts w/ "$ " $ if f$extract(0,2,dcl_record) .eqs. "$ " $ then $ dcl_record = dcl_record - "$ " $! $! Record starts w/ "$" (no space after $) $ else $ if f$extract(0,1,dcl_record) .eqs. "$" then - dcl_record = dcl_record - "$" $ endif $! $! $! Find labels and add to label_file $! $! Save the full line for file checking -- label removed below $ file_work = dcl_record $! $ if f$locate("@",dcl_record) .eq. 0 then goto label_found $ label = f$element(0," ",dcl_record) $ label = f$element(0,"/",label) $ label = f$element(0,"""",label) $ label = f$element(0,"(",label) $ lc = f$locate(":",label) $ if lc .eq. f$locate(":=",label) then goto label_found $ if lc .gt. f$locate("=",label) then goto label_found $ If f$locate(":",label) .lt. (f$length(label)-1) $ then $ label = f$element(0,":",label) + ":" $! now put a space after the label in DCL_RECORD $ dcl_record = label + " " + (dcl_record - label) $ endif $ If f$locate(":",label) .eq. (f$length(label)-1) $ then ! it really is a label $! $! Remember that we are no longer in an EXIT or GOTO shadow $ exit_shadow = 0 $ goto_shadow = 0 $! $! Remove the label from file_work (":" and leading space removed) $ file_work = f$edit(file_work - label,"TRIM") $! $ label = label - ":" $! $! Check to see if the label is defined by symbol substitution $ if f$locate(s_quote,label) .lt. f$length(label) $ then ! symbol substitution $ if .not. suppress_LDS then write err_file f$fao( - "!5UL LDS-I Label ""!AS"" defined by symbol substitution (warning)", - saved_line,label) $ goto label_found ! Don't check dup or add to label file $ endif $! $! Check to see if its already in the label file $ read /err=add_the_label /key="''f$fao("!255AS",label)'" label_file x $! $! If its found, it is a possible duplicate $! (Or it could be a different subroutine...) $! Write it to label_file with a D (Duplicate) in position one $! (We overwrite the "D" later if it is in a different subroutine) $! $ x = f$fao("D!5UL!255AS",saved_line,label) $ write/symbol label_file x $ endif $! $ goto label_found ! it's already there -- O.K. $! $add_the_label: $ work_status = $status $ if work_status .eq. %X000182B2 $ then ! not found so add it $ x = f$fao("0!5UL!255AS",saved_line,label) $ write/symbol label_file x $ else ! it is some unexpected error $ goto err_exit_w_status $ endif $label_found: $! $! Check for OPEN/READ/WRITE/CLOSE statements $! and write/update entries in file_file $! $! Labels are already stripped. $! However, we can still have a DCL verb in a THEN statement/clause. $! We need to create a VERB_WORK variable, which is FILE_WORK $! with any THEN and whatever precedes it stripped. $! $ t$l = f$locate("THEN",file_work) $ w$l = f$length(file_work) $ if t$l .eq. w$l $ then $ verb_work = file_work $ else $ verb_work = f$extract(t$l+5,w$l,file_work) $ endif $! $ verb_loc = 0 $ verb = f$element(0," ",verb_work) $ verb = f$element(0,"/",verb) $ verb_work = f$edit(verb_work - verb,"TRIM") $! $ if f$length(verb) .lt. 3 then goto after_verb $! $! 1 1 2 $! 1 6 1 7 3 (23 is length) $ verb_list = "\OPEN\READ\WRITE\CLOSE\" $ verb_loc = f$locate(verb,verb_list) $ if verb_loc .eq. 23 then goto after_verb $! $! If verb_work starts with "=" or ":=", then this is a situation in which $! an I/O verb is also being used as a symbol. Skip this. $ if ( (f$extract(0,1,verb_work) .eqs. "=") - .or. (f$extract(0,2,verb_work) .eqs. ":=") ) - then goto after_verb $! $! Now parse out the logical name used to reference the file $! $! We already removed the verb from verb_work. $! The logical will be at the start of the first, blank-delimited $! element of verb_work that does NOT start with "/" -- $! BUT we need to look out for blanks before and/or after the $! = sign following a qualifier. $! $ log_num = 0 $ log_name = verb_work $file_log_loop: $! $ if f$extract(0,1,log_name) .eqs. "/" $ then ! This is a qualifier, not the logical name $! It will be terminated by either an equal sign, or a blank, $! or by another qualifier. $ log_name = f$edit(log_name - "/","TRIM") ! remove the / $ log_work = f$element(0," ",f$element(0,"=",f$element(0,"/",log_name))) $! Strip the qualifier and try again $ log_name = f$edit(log_name - log_work,"TRIM") $ goto file_log_loop $ endif $! $ if f$extract(0,1,log_name) .eqs. "=" $ then ! This introduces a qualifier value (string or symbol) or keyword $ log_name = f$edit(log_name - "=","TRIM") ! remove the equal sign $ if f$extract(0,1,log_name) .eqs. """" $ then ! remove the quoted string $ log_name = f$edit(log_name - """","TRIM") $ if ( ( f$extract(0,1,log_name) .eqs. "'" ) - .and. ( f$extract(1,1,log_name) .eqs. "'" ) ) $ then $ log_name = log_name - "'" - "'" $ log_work = f$element(0,"'",log_name) $ log_name = f$edit(log_name - log_work,"TRIM") - "'" $ endif $ log_work = f$element(0,"""",log_name) $ log_name = f$edit(log_name - log_work - """","TRIM") $ goto file_log_loop $ else ! remove the keyword/symbol $! ! It can be terminated by a space or another "/" $ log_work = f$element(0," ",f$element(0,"/",log_name)) $ log_name = f$edit(log_name - log_work,"TRIM") $ goto file_log_loop $ endif $! $ endif $! $ log_name = f$edit(log_name,"TRIM") ! remove leading spaces $ log_name = f$element(0,"/",log_name) ! strip any qualifiers $ log_name = f$element(0," ",log_name) ! strip any other stuff $! $ if log_name .eqs. "SYS$OUTPUT" then goto after_verb $ if log_name .eqs. "SYS$INPUT" then goto after_verb $ if log_name .eqs. "SYS$PIPE" then goto after_verb $ if log_name .eqs. "SYS$COMMAND" then goto after_verb $ if log_name .eqs. "SYS$ERROR" then goto after_verb $! $ if log_name .eqs. "" $ then $ if .not. suppress_NLN then write err_file f$fao( - "!5UL NLN-S An OPEN, READ, WRITE or CLOSE statement has no logical name", - saved_line) $ goto after_Verb $ endif $! $! Setup default output $ i$o = "0" ! input (opened for read) (default) $ a$o = "0" ! output (opened for append) $ o$o = "0" ! output (opened for write) $ r$o = "0" ! read $ w$o = "0" ! write (w/o /UPDATE) (or READ/DELETE) $ u$o = "0" ! write /UPDATE $ c$o = "0" ! close $ logical = "" ! blank initially $! $ if verb_loc .eq. 1 $ then ! OPEN -- CHECK FOR /READ/WRITE/APPEND $ v$l = f$length(verb_work) $ i$l = f$locate("/REA",verb_work) $ a$l = f$locate("/APP",verb_work) $ o$l = f$locate("/WRI",verb_work) $! $ if i$l .lt. v$l then i$o = "1" $ if a$l .lt. v$l then a$o = "1" $ if o$l .lt. v$l then o$o = "1" $ if ( (i$l .eq. v$l) .and. (a$l .eq. v$l) .and. (o$l .eq. v$l) ) - then i$o = "1" ! default to read $ endif $! $ if verb_loc .eq. 6 ! READ $ then $ v$l = f$length(verb_work) $ d$l = f$locate("/DEL",verb_work) $ if d$l .lt. v$l then w$o = "1" ! /READ with /DELETE => write $ r$o = "1" ! r$o always set for READ $ endif $! $ if verb_loc .eq. 11 ! WRITE $ then $ v$l = f$length(verb_work) $ u$l = f$locate("/UPD",verb_work) $ if u$l .eq. v$l $ then $ w$o = "1" $ else $ u$o = "1" $ endif $ endif $! $ if verb_loc .eq. 17 then c$o = "1" ! CLOSE $! $! $! Check to see if it's already in the file file $! $! Replace any double quotes ("...") with {*} $! This prevents miss-interpretation in the /KEY qualifier $! and is consistent with quoted string checking $! $file_quote_loop: $ log_work = f$element(0,"""",log_name) $ if log_work .eqs. log_name then goto after_file_quote $ log_name = log_work + "{*}" + f$element(2,"""",log_name) $ goto file_quote_loop $after_file_quote: $! $ file_key = f$fao("!255AS",log_name) $ read /err=add_the_file /key="''file_key'" file_file x $! $! If its found, then update it $! $ if f$extract(0,1,x) then i$o = "1" $ if f$extract(1,1,x) then a$o = "1" $ if f$extract(2,1,x) then o$o = "1" $ if f$extract(3,1,x) then r$o = "1" $ if f$extract(4,1,x) then w$o = "1" $ if f$extract(5,1,x) then u$o = "1" $ if f$extract(6,1,x) then c$o = "1" $! $ x = i$o + a$o + o$o + r$o + w$o + u$o + c$o + file_key $! $ write/update/symbol file_file x $! $ goto after_verb $! $add_the_file: $! $ work_status = $status $ if work_status .eq. %X000182B2 $ then ! not found so add it $ x = i$o + a$o + o$o + r$o + w$o + u$o + c$o + file_key $ write/symbol file_file x $ else ! it is some unexpected error $ goto err_exit_w_status $ endif $! $! $after_verb: $! $! $! Check for possible misspelled lexical function $! (not in valid_lexicals) $! possible truncated lexical function $! (4 chars (6 including "F$") required for uniqueness) $! $! Check for possible misspelled/typed lexical functions. $! If we find "F$..." that doesn't match up with the leading $! characters in a valid lexical function, it is an error. $! If it matches but has less that 4 characters (6, include "F$) $! then that is also an error. $! $! Check if the line contains a lexical function $! We must take into account that because the lexical function could $! be within a quoted string it could be in lower (or mixed) case. $! So we must check for both "F$" and "f$". $ l = f$locate("F$",dcl_record) $ if l .ge. f$length(dcl_record) $ then $ l = f$locate("f$",dcl_record) $ if l .ge. f$length(dcl_record) then goto after_lexical $ endif $! $! Save the character that precedes the F$ $ if l .gt. 0 $ then $ prior$char = f$extract(l-1,1,dcl_record) $ else $ prior$char = " " $ endif $! $ work = f$extract(l,999,dcl_record) $! $lexical_loop: $! $ work1 = f$edit(f$extract(0,2,work),"UPCASE") ! get the F$ $ l = 2 $! $! If what we now have is a "F$" preceded by a letter, a number or $! a "$" or "_", or """", then it isn't a lexical. $! $ if prior$char .ges. "0" .and. prior$char .les. "9" then goto lexical_next $ if prior$char .ges. "a" .and. prior$char .les. "z" then goto lexical_next $ if prior$char .ges. "A" .and. prior$char .les. "Z" then goto lexical_next $ if prior$char .eqs. "$" then goto lexical_next $ if prior$char .eqs. "_" then goto lexical_next $ if prior$char .eqs. """" then goto lexical_next $! $lexical_loop_2: $ char = f$edit(f$extract(l,1,work),"UPCASE") ! get the next character $ l = l + 1 $! If the character is alphabetic, it is part of the lexical function name $ if char .ges. "a" .and. char .les. "z" then goto lexical_char $ if char .ges. "A" .and. char .les. "Z" then goto lexical_char $! If the character is not alphabetic, it is not part of the name $ goto lexical_check ! not a letter $lexical_char: $ work1 = work1 + char $ goto lexical_loop_2 $! $lexical_check: $! $! Since it could be in lower or mixed case, upshift it $ work1 = f$edit(work1,"UPCASE") $! $ if l .lt. 6 ! **TOO SHORT** $ then $ if work1 .nes. "F$FAO" .and. work1 .nes. "F$PID" $ then $ if .not. suppress_PTL then write err_file f$fao( - "!5UL PTL-E possible truncated lexical (!AS)", - saved_line,work1) $ endif $ endif $! $ if f$locate(work1,valid_lexicals) .eq. length_valid_lexicals $ then $ if .not. suppress_PML then write err_file f$fao( - "!5UL PML-E possible misspelled lexical (!AS)", - saved_line,work1) $ endif $! $lexical_next: $ work = f$extract(l-1,999,work) ! get rid of what we just checked $ l = f$locate("F$",work) ! look for more $ if l .ge. f$length(work) then goto after_lexical ! no more $! $! Save the character that precedes the F$ $ if l .gt. 0 $ then $ prior$char = f$extract(l-1,1,work) $ else $ prior$char = " " $ endif $! $ work = f$extract(l,999,work) $! $ goto lexical_loop ! go do the next F$ $! $! $after_lexical: $! $! Check for apparent numeric constants with string comparison, $! or apparent string constants with numeric comparison $! $! Walk through the line looking for comparison operators $! $ l = 0 $wct_loop: $ work = f$element(l," ",dcl_record) $ if work .eqs. " " then got after_wct $! $! Look for numeric comparison operators $ if work .eqs. ".EQ." - .or. work .eqs. ".GE." - .or. work .eqs. ".GT." - .or. work .eqs. ".LE." - .or. work .eqs. ".LT." - .or. work .eqs. ".NE." $ then $! $ if l .eq. 0 $ then $ if .not. suppress_MEC then write err_file f$fao( - "!5UL MEC-S missing expression for comparison", - saved_line) $ l = l + 1 $ goto wct_loop $ endif $! $ if f$element(l+1," ",dcl_record) .eqs. " " $ then $ if .not. suppress_MEC then write err_file f$fao( - "!5UL MEC-S missing expression for comparison", - saved_line) $ goto after_wct $ endif $! $! Numeric comparison operators must not be preceded or followed $! by quoted strings. $ if ( ( (f$extract(0,1,f$element(l-1," ",dcl_record)) .eqs. quote ) - .and. (f$extract(0,2,f$element(l-1," ",dcl_record)) .nes. """," ) - .and. (f$extract(0,2,f$element(l-1," ",dcl_record)) .nes. """""") ) - .or. ( (f$extract(0,1,f$element(l+1," ",dcl_record)) .eqs. quote ) - .and. (f$extract(0,2,f$element(l+1," ",dcl_record)) .nes. """""") ) ) $ then $ if .not. suppress_WCT then write err_file f$fao( - "!5UL WCT-E wrong constant type for comparison", - saved_line) $ endif $ endif $! $! Look for string comparison operators $ if work .eqs. ".EQS." - .or. work .eqs. ".GES." - .or. work .eqs. ".GTS." - .or. work .eqs. ".LES." - .or. work .eqs. ".LTS." - .or. work .eqs. ".NES." $ then $! $ if l .eq. 0 $ then $ if .not. suppress_MEC then write err_file f$fao( - "!5UL MEC-S missing expression for comparison", - saved_line) $ l = l + 1 $ goto wct_loop $ endif $! $ if f$element(l+1," ",dcl_record) .eqs. " " $ then $ if .not. suppress_MEC then write err_file f$fao( - "!5UL MEC-S missing expression for comparison", - saved_line) $ goto after_wct $ endif $! $! String comparison operators must not be preceded or followed $! by numeric constants $ work1 = f$extract(0,1,f$element(l-1," ",dcl_record)) $ work2 = f$extract(0,1,f$element(l+1," ",dcl_record)) $ if (work1 .ges. "0" .and. work1 .les. "9") - .or. (work2 .ges. "0" .and. work2 .les. "9") $ then $ if .not. suppress_WCT then write err_file f$fao( - "!5UL WCT-E wrong constant type for comparison", - saved_line) $ endif $ endif $! $ l = l + 1 $! $ goto wct_loop $! $after_wct: $! $! ----------------------------------- $! $! Check for unpaired (i.e., an odd number of) quotes (") $! $ l = 0 $count_quotes: $ work = f$element(l,quote,dcl_record) $ if work .eqs. quote then goto quotes_counted $ l = l + 1 $ goto count_quotes $! $quotes_counted: $ l = l - 1 $ if l .ne. l/2*2 $ then $ if .not. suppress_UPQ then write err_file f$fao( - "!5UL UPQ-S unpaired quotation marks ("")", - saved_line) $ endif $! $! (2)------------------------------------------------------------ $! $! Checking for properly paired double quotes has already been done. $! Checking for properly paired single quotes outside quoted strings $! is done later. $! $ work = "" ! Blank "work" so we can build the output in it $! $! If dcl_rec contains contractions, they will provide an incorrect $! possible error using single quote in quoted strings. $! So we'll remove any of the following contractions. $! $! Contractions ending in "'s" are removed from this list $! because we check for possessives below. $! $! (multiple statements to avoid "expression too complex") $! $ dcl_record = dcl_record - - "aren't" - - "can't" - - "couldn't" - - "didn't" - - "doesn't" - - "don't" - - "hadn't" - - "hasn't" - - "haven't" - - "he'd" - - "he'll" - - "I'd" - - "I'll" - - "I'm" - - "I've" - - "isn't" - - "she'd" - - "she'll" $ dcl_record = dcl_record - - "shouldn't" - - "they'd" - - "they'll" - - "they're" - - "they've" - - "wasn't" - - "we'd" - - "we'll" - - "we're" - - "we've" - - "weren't" - - "won't" - - "wouldn't" - - "you'd" - - "you'll" - - "you're" - - "you've" $! $! And they could be capitalized... $ dcl_record = dcl_record - - "Aren't" - - "Can't" - - "Couldn't" - - "Didn't" - - "Doesn't" - - "Don't" - - "Hadn't" - - "Hasn't" - - "Haven't" - - "He'd" - - "He'll" - - "Isn't" - - "She'd" - - "She'll" $ dcl_record = dcl_record - - "Shouldn't" - - "They'd" - - "They'll" - - "They're" - - "They've" - - "Wasn't" - - "We'd" - - "We'll" - - "We're" - - "We've" - - "Weren't" - - "Won't" - - "Wouldn't" - - "You'd" - - "You'll" - - "You're" - - "You've" $! $! And they could be in all capitals... $ dcl_record = dcl_record - - "AREN'T" - - "CAN'T" - - "COULDN'T" - - "DIDN'T" - - "DOESN'T" - - "DON'T" - - "HADN'T" - - "HASN'T" - - "HAVEN'T" - - "HE'D" - - "HE'LL" - - "ISN'T" - - "SHE'D" - - "SHE'LL" $ dcl_record = dcl_record - - "SHOULDN'T" - - "THEY'D" - - "THEY'LL" - - "THEY'RE" - - "THEY'VE" - - "WASN'T" - - "WE'D" - - "WE'LL" - - "WE'RE" - - "WE'VE" - - "WEREN'T" - - "WON'T" - - "WOULDN'T" - - "YOU'D" - - "YOU'LL" - - "YOU'RE" - - "YOU'VE" $! $! Also get rid of "'" and (') to avoid PSQ or PRQ errors $! $dsd_quote_remove: $ if f$locate("""'""",dcl_record) .lt. f$length(dcl_record) $ then $ dcl_record = dcl_record - """'""" $ goto dsd_quote_remove $ endif $! $osc_quote_remove: $ if f$locate("(')",dcl_record) .lt. f$length(dcl_record) $ then $ dcl_record = dcl_record - "(')" $ goto osc_quote_remove $ endif $! $! And to eliminate most possessives, get rid of "%'s " $! where % is any non-blank. (Note the trailing blank.) $! $pos_quote_remove1: $ if ( ( f$locate("'s ",dcl_record) .lt. f$length(dcl_record) ) - .and. ( f$extract(f$locate("'s ",dcl_record)-1,1,dcl_record) .gts. " " ) ) $ then $ dcl_record = dcl_record - "'s " $ goto pos_quote_remove1 $ endif $! $pos_quote_remove2: $! DCL_CHECK COMMENT: PSQ O.K. in next line (2 times) $ if ( ( f$locate("'s""",dcl_record) .lt. f$length(dcl_record) ) - .and. ( f$locate("'s""",dcl_record) .eq. ( f$length(dcl_record) - 3 ) ) - .and. ( f$extract(f$locate("'s ",dcl_record)-1,1,dcl_record) .gts. " " ) ) $ then $! DCL_CHECK COMMENT: PSQ O.K. in next line $ dcl_record = dcl_record - "'s""" + """" $ goto pos_quote_remove2 $ endif $! $! At the start of the line we are NOT in a quoted string. $! (Continuation lines have already been concatenated.) $! $not_in_quote: $! $! Everything before the first/next quote is not in a quoted string $! $ temp = f$element(0,quote,dcl_record) $ if temp .eqs. quote then goto end_check_quotes $! $! Add what we've just got to WORK... $ work = work + temp $! $! .. and remove it from DCL_RECORD $ dcl_record = dcl_record - temp $! $! If what is left in DCL_RECORD starts with a quote, $! replace it with a placeholder $ if f$extract(0,1,dcl_record) .eqs. quote $ then $ dcl_record = dcl_record - quote $ work = work + "{*}" $ endif $! $ i = 0 ! We'll use 0-based F$EXTRACT to search the quoted string $! $!in_quote: $! $! The contents of quoted strings is NOT copied to "work". $! At the start of a quoted string we are NOT in symbol substitution $! $not_in_symbol_subs: $! $! If we encounter "'s " () [lower case s] $! or () [upper case S] $! in a quoted string, we'll take it as a possive and skip it $! $ if ( (f$extract(i,3,dcl_record) .eqs. "'s ") - .or. (f$extract(i,3,dcl_record) .eqs. "'S ") ) $ then $ i = i + 3 $ goto not_in_symbol_subs $ endif $! $! Symbol substitution starts with two single quotes ('') $ if f$extract(i,2,dcl_record) .eqs. s2_quote $ then $ i = i + 2 $ goto in_symbol_subs $ endif $! $! Get the next character $ char = f$extract(i,1,dcl_record) $ i = i + 1 $! $ if char .eqs. "" then goto end_check_quotes ! shouldn't happen, but... $! $! If we encounter a single single-quote that is NOT terminating a $! symbol substitution, it could be a valid possessive or contraction, or $! it could be an error. $! $ if char .eqs. s_quote $ then $ if .not. suppress_PSQ then write err_file f$fao( - "!5UL PSQ-W possible error using single-quote (') in quoted string", - saved_line) $ goto not_in_symbol_subs $ endif $! $! The first double quote not within symbol substitution ends $! the quoted string. $ if char .eqs. quote $ then $! When the quoted string ends, go back to the not_in_quote routine. $! First, reset what is in DCL_RECORD $ dcl_record = f$extract(i,9999,dcl_record) $ goto not_in_quote $ endif $! $ goto not_in_symbol_subs $! $in_symbol_subs: $! $! Get the next character $ char = f$extract(i,1,dcl_record) $ i = i + 1 $! $! If the line ends while still in symbol substitution, it is an error; $! (Since we keep going until we find a single-quote, we WILL get to line $! end if symbol substitution is not terminated.) $! output the error and exit the quote checking $ if char .eqs. "" $ then $ if .not. suppress_PSQ then write err_file f$fao( - "!5UL PSQ-W possible error using single-quote (') in quoted string", - saved_line) $ goto end_check_quotes $ endif $! $! Symbol substitution ends with one single quote ('). $ if f$extract(i,1,dcl_record) .eqs. s_quote $ then $ i = i + 1 $ goto not_in_symbol_subs $ endif $! $! Double quotes within symbol substitution are ignored. $! $ goto in_symbol_subs $! $! $end_check_quotes: $! $! "work" now contains DCL_RECORD with quoted strings removed. $! Put it back in DCL_RECORD $! $ dcl_record = work $! $! $! (2)------------------------------------------------------------ $! $! Check for unpaired (i.e., an odd number of) single-quotes ('). $! Since we have at this point removed any quoted strings, which could $! contain an odd number of single-quotes for contractions (e.g. "can't"), $! possessives (e.g. "Mary's") or symbol substitution (e.g. "''count'") $! any single-quotes left must be paired, 1-1, for symbol substitution $! (e.g. 'file_name') $! Since a symbol being used for substitution cannot contain a space, $! We will look at each blank separated element of the line. $! This will catch errors similar to $! $ rename 'old 'new -- s/b $ rename 'old' 'new' $! $ j = 0 $s_quotes_loop: $ work1 = f$element(j," ",dcl_record) $ if work1 .eqs. " " then goto s_quotes_counted $ j = j + 1 $! $ l = 0 $count_s_quotes: $ work = f$element(l,s_quote,work1) $ if work .eqs. s_quote $ then $ l = l - 1 $! $ if l .ne. l/2*2 $ then $ if .not. suppress_PRQ then write err_file f$fao( - "!5UL PRQ-E probable error using single-quote (')", - saved_line) $ endif $! $ goto s_quotes_loop $ endif $ l = l + 1 $ goto count_s_quotes $! $s_quotes_counted: $! $! now write the intermediate work file before we $! mess with if/then/else/endif $! $! Because the record can be "too big", we build it in a symbol... $ temp_rec = f$fao("!5UL !AS",saved_line,dcl_record) $! ...and then use write/symbol to handle large records. $ write /symbol work_file temp_rec $! $! Unconditional GOTO and EXIT lines cast a "shadow" -- following lines $! are unreachable until a LABEL or ELSE is found. $! $ token = f$edit(f$element(0," ",dcl_record),"UPCASE") $ If f$locate(":",token) .eq. (f$length(token)-1) $ then ! it is a label $ token = f$edit(f$element(1," ",dcl_record),"UPCASE") $ endif $! $ if token .eqs. "GOTO" $ then $ wrk = f$edit(dcl_record - "GOTO","TRIM,UPCASE") $ if ( (f$extract(0,1,wrk) .nes. "=") - .and. (f$extract(0,2,wrk) .nes. ":=") ) - then goto_shadow = saved_line $ endif $! $ if token .eqs. "EXIT" $ then $ wrk = f$edit(dcl_record - "EXIT","TRIM,UPCASE") $ if ( (f$extract(0,1,wrk) .nes. "=") - .and. (f$extract(0,2,wrk) .nes. ":=") ) - then exit_shadow = saved_line $ endif $! $! $! A single command can contain either IF, or both IF and THEN. $! $! If it contains IF and THEN, then we are O.K. and only need $! to check for "=" between the IF and the THEN. $! $! If it contains only IF, then there may not be an "=" on the line $! and the next statement must be THEN, and there must eventually be $! an ENDIF. There may also be an ELSE between the THEN and the ENDIF. $! $! IF statements may be nested, so we must track all this at levels. $! $if_then_else: $! $ token = f$edit(f$element(0," ",dcl_record),"UPCASE") $ If f$locate(":",token) .eq. (f$length(token)-1) $ then ! it is a label $ token = f$edit(f$element(1," ",dcl_record),"UPCASE") $ endif $! $ if ( (token .nes. "THEN") - .and. (f$extract(if_level,1,then_level) .eqs. "0") - .and. (if_level .gt. 0) ) $ then $ then_level[if_level,1] := 1 $ if .not. suppress_TRH then write err_file f$fao( - "!5UL TRH-S THEN statement required here", - saved_line) $ endif $! $ if token .eqs. "IF" $ then $ if f$locate(" THEN ",dcl_record) .lt. f$length(dcl_record) $ then $ if f$locate("=",dcl_record) .lt. f$locate(" THEN ",dcl_record) $ then $ if .not. suppress_EFB then write err_file f$fao( - "!5UL EFB-S ""="" found between IF and THEN", - saved_line) $ endif $ else $! $ if f$locate("=",dcl_record) .lt. f$length(dcl_record) $ then $ if .not. suppress_EFB then write err_file f$fao( - "!5UL EFB-S ""="" found between IF and THEN", - saved_line) $ endif $! $ if if_level - sr_level .gt. 14 $ then $ if .not. suppress_TML then write err_file f$fao( - "!5UL TML-S too many levels of IF statements", - saved_line) $ endif $ if_lines = f$fao("!UL/!AS",saved_line,if_lines) $ if_level = if_level + 1 $ ifsr_level[if_level,1] := "I" ! remember it's an IF $ endif $ endif $! $ if (f$length(token) .ge. 4) - .and. (f$locate(token,"SUBROUTINE") .eq. 0) $ then $ if if_level .gt. 20 $ then $ if .not. suppress_TMS then write err_file f$fao( - "!5UL TMS-S too many levels of IF and SUBROUTINE statements", - saved_line) $ endif $ if_lines = f$fao("!UL/!AS",saved_line,if_lines) $ if_level = if_level + 1 $ sr_level = sr_level + 1 $ ifsr_level[if_level,1] := "S" ! remember it's a SUBROUTINE $ then_level[if_level,1] := 1 ! Don't allow THEN after SUBROUTINE $ else_level[if_level,1] := 1 ! Don't allow ELSE after SUBROUTINE $ endif $! $ if token .eqs. "THEN" $ then $ if f$extract(if_level,1,then_level) .eqs. "1" - .or. if_level .eq. 0 $ then $ if .not. suppress_TNA then write err_file f$fao( - "!5UL TNA-S THEN statement not allowed here", - saved_line) $ else $ then_level[if_level,1] := 1 $ endif $ dcl_record = dcl_record - "THEN" $ if f$extract(0,1,dcl_record) .eqs. " " then - dcl_record = dcl_record - " " $ goto if_then_else $ endif $! $ if token .eqs. "ELSE" $ then $! ELSE cancels goto_shadow and exit_shadow $ goto_shadow = 0 $ exit_shadow = 0 $! $ if f$extract(if_level,1,else_level) .eqs. "1" - .or. if_level .eq. 0 $ then $ if .not. suppress_ENA then write err_file f$fao( - "!5UL ENA-S ELSE statement not allowed here", - saved_line) $ else $ else_level[if_level,1] := 1 $ endif $ dcl_record = dcl_record - "ELSE" $ if f$extract(0,1,dcl_record) .eqs. " " then - dcl_record = dcl_record - " " $ goto if_then_else $ endif $! $ if token .eqs. "ENDIF" $ then $! ENDIF cancels goto_shadow and exit_shadow $ goto_shadow = 0 $ exit_shadow = 0 $! $ if if_level .eq. 0 - .or. f$extract(if_level,1,ifsr_level) .nes. "I" $ then $ if .not. suppress_DNA then write err_file f$fao( - "!5UL DNA-S ENDIF statement not allowed here", - saved_line) $ else $! $! Write record in if_file $ if_start = f$element(0,"/",if_lines) $ if_start_num = f$integer(if_start) $ write if_file f$fao("!5UL!5ULI",if_start_num,saved_line) $! $! Reset if-level info $ then_level[if_level,1] := 0 $ else_level[if_level,1] := 0 $ ifsr_level[if_level,1] := 0 $ if_level = if_level - 1 $ if_lines = if_lines - if_start - "/" $! $ endif $ endif $! $ if (f$length(token) .ge. 4) - .and. (f$locate(token,"ENDSUBROUTINE") .eq. 0) $ then $! ENDSUBROUTINE cancels goto_shadow and exit_shadow $ goto_shadow = 0 $ exit_shadow = 0 $! $ if if_level .eq. 0 - .or. f$extract(if_level,1,ifsr_level) .nes. "S" $ then $ if .not. suppress_RNA then write err_file f$fao( - "!5UL RNA-S ENDSUBROUTINE statement not allowed here", - saved_line) $ else $! $! Write record in if_file $ if_start = f$element(0,"/",if_lines) $ if_start_num = f$integer(if_start) $ write if_file f$fao("!5UL!5ULS",if_start_num,saved_line) $! $! Reset if-level info $ then_level[if_level,1] := 0 $ else_level[if_level,1] := 0 $ ifsr_level[if_level,1] := 0 $ if_level = if_level - 1 $ sr_level = if_level - 1 $ if_lines = if_lines - if_start - "/" $! $ endif $ endif $! $! $! Check for lines that are obscured by an EXIT or GOTO $! $ if exit_shadow .gt. 0 .and. saved_line .gt. exit_shadow $ then $ if .not. suppress_CRE then write err_file f$fao( - "!5UL CRE-W code cannot be reached due to EXIT at line !UL", - saved_line,exit_shadow) $ exit_shadow = 0 $ endif $! $ if goto_shadow .gt. 0 .and. saved_line .gt. goto_shadow $ then $ if .not. suppress_CRG then write err_file f$fao( - "!5UL CRG-W code cannot be reached due to GOTO at line !UL", - saved_line,goto_shadow) $ goto_shadow = 0 $ endif $! $ goto read_dcl1 $! $end_dcl1_c: $! $ if .not. suppress_NCL then write err_file f$fao( - "!5UL NCL-S no continuation line at EOF", - line_number) $end_dcl1: $ work2 = 0 $end_dcl1_a: $! $ if if_level .gt. 0 $ then $! work = f$integer(f$element(if_level-1,"/",if_lines)) $ work = f$integer(f$element(work2,"/",if_lines)) $ if f$extract(if_level,1,ifsr_level) .eqs. "I" $ then $ if .not. suppress_INT then write err_file f$fao( - "!5UL INT-S IF statement not terminated", - work) $ endif $ if f$extract(if_level,1,ifsr_level) .eqs. "S" $ then $ if .not. suppress_SNT then write err_file f$fao( - "!5UL SNT-S SUBROUTINE statement not terminated", - work) $ endif $ if_level = if_level - 1 $ work2 = work2 + 1 $ goto end_dcl1_a $ endif $! $ if in_deck $ then $ if .not. suppress_NED then write err_file f$fao( - "!5UL NED-S no ''d$w' for $DECK at line !UL", - saved_line,deck_line) $ endif $! $! Close the DCL file and the WORK file $! $ close work_file $ close dcl_file $! $! comment_lines already contains number of comment lines $! continuation_lines already contains number of continuation lines $! deck_lines already contains number of lines within $DECK/$EOD pairs $! Save number of code lines $ total_lines = line_number $ code_lines = total_lines - continuation_lines - deck_lines - - comment_lines - blank_lines $! $ Say "Starting Pass 2 -- ''f$time()' ..." $! $! This pass checks labels referenced $! $! LNF-S label ""!AS"" not found $! LNR-I label ""!AS"" not referenced (warning) $! RLI-S referenced label ""!AS"" is in if-group at lines !UL-!UL $! RLS-W referenced label ""!AS"" is in subroutine at lines !UL-!UL $! TLS-I target label ""!AS"" provided by symbol substitution (warning)" $! $! by CALL, GOTO and GOSUB commands, and $! by /ERROR and /END qualifiers. $! $! Open the work file for read $ open /read work_file sys$scratch:dcl$work_'pid'.tmp $! $ line_to_display = (line_disply_increment * 2) $read_dcl2: $! $ read /end=end_dcl2 work_file dcl_record $! $! The original line number was saved in the work file pass 1. $! Separate the line number and the "compressed" dcl record $! $ line_number = f$integer(f$extract(0,5,dcl_record)) $ saved_line = line_number $ dcl_record = f$extract(6,(f$length(dcl_record)-5),dcl_record) $! $ if line_number .gt. line_to_display $ then $ say f$fao("...processing line number !UL...",line_to_display) $ line_to_display = line_to_display + (line_disply_increment * 2) $ endif $! $! $! Check for OPEN/READ/WRITE/CLOSE statements $! check entries in file_file and write errors $! $! Labels are still there; we need to strip them $! $ verb_work = dcl_record $ label = f$element(0," ",verb_work) $ label = f$element(0,"/",label) $ If f$locate(":",label) .eq. (f$length(label)-1) $ then ! it really is a label $! Remove the label from verb_work (":" and leading space removed) $ verb_work = verb_work - label - " " $ endif $! $! Now labels are removed $! However, we can still have a DCL verb in a THEN statement/clause. $! We need to create a VERB_WORK variable, which is DCL_RECORD $! with any THEN and whatever precedes it stripped. $! $ t$l = f$locate("THEN",verb_work) $ w$l = f$length(verb_work) $ if t$l .ne. w$l $ then $ verb_work = f$extract(t$l+5,w$l,verb_work) $ endif $! $ verb = f$element(0," ",verb_work) $ verb = f$element(0,"/",verb) $ verb_work = f$edit(verb_work - verb,"TRIM") $! $ if f$length(verb) .lt. 3 then goto scan_for_labels_used $! $! 1 1 2 $! 1 6 1 7 3 (23 is length) $ verb_list = "\OPEN\READ\WRITE\CLOSE\" $ verb_loc = f$locate(verb,verb_list) $ if verb_loc .eq. 23 then goto scan_for_labels_used $! $! If verb_work starts with a "=", then this is a situation in which $! an I/O verb is also being used as a symbol. Skip this. $ if ( (f$extract(0,1,verb_work) .eqs. "=") - .or. (f$extract(0,2,verb_work) .eqs. ":=") ) - then goto scan_for_labels_used $! $! Now parse out the logical name used to reference the file $! $! We already removed the verb from verb_work. $! The logical will be at the start of the first, blank-delimited $! element of verb_work that does NOT start with "/" $ log_num = 0 $file_log_loop2: $ log_name = f$element(log_num," ",verb_work) $! $ if f$extract(0,1,log_name) .eqs. "/" $ then ! this is a qualifier, not the logical name $! $! If the next element IS an "=" sign... $ if f$element(log_num + 1," ",verb_work) .eqs. "=" $ then $ log_num = log_num + 3 $ goto file_log_loop2 $ endif $! $! If this element ends with an "=" sign... $! If the next element STARTS with an "=" sign... $ if ( (f$extract(f$length(log_name)-1,1,log_name) .eqs. "=") - .or. (f$extract(0,1,f$element(log_num+1," ",verb_work)) .eqs. "=") ) $ then $ log_num = log_num + 2 $ goto file_log_loop2 $ endif $! $! Otherwise... $ log_num = log_num + 1 $ goto file_log_loop2 $! $ endif $! $ log_name = f$element(0,"/",log_name) ! strip any qualifiers $ log_name = f$element(0," ",log_name) ! strip any other stuff $! $ if log_name .eqs. "SYS$OUTPUT" then goto scan_for_labels_used $ if log_name .eqs. "SYS$INPUT" then goto scan_for_labels_used $ if log_name .eqs. "SYS$PIPE" then goto scan_for_labels_used $ if log_name .eqs. "SYS$COMMAND" then goto scan_for_labels_used $ if log_name .eqs. "SYS$ERROR" then goto scan_for_labels_used $ if log_name .eqs. "" then goto scan_for_labels_used ! Already caught NLN $! $! Read file_file $! $ file_key = f$fao("!255AS",log_name) $! Will abort on read error $ read /key="''file_key'" file_file x $! $ i$o = f$extract(0,1,x) $ a$o = f$extract(1,1,x) $ o$o = f$extract(2,1,x) $ r$o = f$extract(3,1,x) $ w$o = f$extract(4,1,x) $ u$o = f$extract(5,1,x) $ c$o = f$extract(6,1,x) $! $ if verb_loc .eq. 1 $ then ! OPEN $! $! ONC-E A file that is opened has no close statement $ if .not. suppress_ONC .and. .not. c$o then write err_file f$fao( - "!5UL ONC-E A file that is OPENed has no CLOSE statement", - line_number) $! $! INR-E A file that is opened for read is not read $ if .not. suppress_INR .and. i$o .and. .not. r$o - then write err_file f$fao( - "!5UL INR-E A file that is opened for read is not read", - line_number) $! $! ANR-E A file that is opened for append is not written $ if .not. suppress_ANR .and. a$o .and. .not. w$o - then write err_file f$fao( - "!5UL ANR-E A file that is opened for append is not written", - line_number) $! $! ONW-E A file that is opened for write is not written $ if .not. suppress_ONW .and. o$o .and. .not. w$o - then write err_file f$fao( - "!5UL ONW-E A file that is opened for write is not written", - line_number) $! $ endif $! $ if verb_loc .eq. 6 $ then ! READ $ if .not. suppress_RNR .and. .not. i$o then write err_file f$fao( - "!5UL RNR-S A file that is not opened for read is being read", - line_number) $ endif $! $ if verb_loc .eq. 11 $ then ! WRITE $ v$l = f$length(verb_work) $ u$l = f$locate("/UPD",verb_work) $ if u$l .eq. v$l $ then ! not /UPDATE $ if .not. suppress_WNW .and. .not. o$o .and. .not. a$o - then write err_file f$fao( - "!5UL WNW-S A file that is not opened for write or append is being written", - line_number) $ else !/Update $ if ( (.not. suppress_UNU) - .and. (.not. i$o .or. .not. o$o) ) - then write err_file f$fao( - "!5UL UNU-S A file that is not oppend for read and write is being updated", - line_number) $ endif $ endif $! $ if verb_loc .eq. 17 $ then ! CLOSE $ if .not. suppress_CNO .and. .not. i$o .and. .not. a$o .and. .not. o$o - then write err_file f$fao( - "!5UL CNO-S A file that is closed has no open statement", - line_number) $ endif $! $! $! Now scan the line for "/END","/ERR[OR]","GOTO ","GOSUB " "CALL " - $! (Remember -- the line was UPCASEd in pass 1 -- and compressed) $! $scan_for_labels_used: $! $ length = f$length(dcl_record) $! $ if f$locate("/END",dcl_record) .lt. length $ then $ if verb_loc .eq. 6 ! READ $ then $! set l to point to the first character of the label $ l = f$locate("/END",dcl_record) + 4 $! Increment l to the "=" $end_qual_loop: $ if f$extract(l,1,dcl_record) .nes. "=" $ then $ l = l + 1 $ if l .lt. length then goto end_qual_loop $ endif $! increment one more for the "=" itself $ l = l + 1 $! Increment l past the space, if there is one $ if f$extract(l,1,dcl_record) .eqs. " " $ then $ l = l + 1 $ endif $ gosub check_for_label $ endif $! $ if (verb_loc .eq. 1) .or. (verb_loc .eq.17) ! OPEN or CLOSE $ then $ if .not. suppress_OCE then write err_file f$fao( - "!5UL OCE-I OPEN or CLOSE statement with /END qualifier", - line_number) $ endif $ endif $! $ if ( (f$locate("/ERR",dcl_record) .lt. length) - .and. ( ( verb_loc .eq. 1 ) - .or. ( verb_loc .eq. 6 ) - .or. ( verb_loc .eq. 11 ) - .or. ( verb_loc .eq. 17 ) ) ) $ then ! found $! set l to point to the first character of the label $ l = f$locate("/ERR",dcl_record) + 4 $! Increment l to the "=" $err_qual_loop: $ if f$extract(l,1,dcl_record) .nes. "=" $ then $ l = l + 1 $ if l .lt. length then goto err_qual_loop $ endif $! increment one more for the "=" itself $ l = l + 1 $! Increment l past the space, if there is one $ if f$extract(l,1,dcl_record) .eqs. " " $ then $ l = l + 1 $ endif $ gosub check_for_label $ endif $! $ if f$locate("GOTO ",dcl_record) .lt. length $ then ! found $! if "GOTO" is not preceded by a space or a $, then it is not a command $ l = f$locate("GOTO",dcl_record) $ if (l .gt. 0) - .and. (f$extract(l-1,1,dcl_record) .nes. " ") - .and. (f$extract(l-1,1,dcl_record) .nes. "$") - then goto after_goto $! set l to point to the first character of the label $ l = l + 4 $! Increment l past any spaces $goto_qual_loop: $ if f$extract(l,1,dcl_record) .eqs. " " $ then $ l = l + 1 $ if l .lt. length then goto goto_qual_loop $ endif $ gosub check_for_label $ endif $after_goto: $! $ if f$locate("GOSUB ",dcl_record) .lt. length $ then ! found $! if "GOSUB" is not preceded by a space or a $, then it is not a command $ l = f$locate("GOSUB",dcl_record) $ if (l .gt. 0) - .and. (f$extract(l-1,1,dcl_record) .nes. " ") - .and. (f$extract(l-1,1,dcl_record) .nes. "$") - then goto after_gosub $! set l to point to the first character of the label $ l = l + 5 $! Increment l past any spaces $gosub_qual_loop: $ if f$extract(l,1,dcl_record) .eqs. " " $ then $ l = l + 1 $ if l .lt. length then goto gosub_qual_loop $ endif $ gosub check_for_label $ endif $after_gosub: $! $ if f$locate("CALL",dcl_record) .lt. length $ then ! found $ l = f$locate("CALL",dcl_record) $! if "CALL" is not preceded by a space, then it is not a command $ if (l .gt. 0) .and. (f$extract(l-1,1,dcl_record) .nes. " ") - then goto after_call $! $! set l to point to the first character of the label $ l = l + 4 $! $! if "CALL" is not followed by a space or a "/", $! then it is not a command $ if ( (f$extract(l,1,dcl_record) .nes. " ") - .and. (f$extract(l,1,dcl_record) .nes. "/") ) - then goto after_call $! $ call$call = 1 ! remember that it IS a "CALL" $! $! Increment l past any spaces $call_qual_loop: $ if f$extract(l,1,dcl_record) .eqs. " " $ then $ l = l + 1 $ if l .lt. length then goto call_qual_loop $ endif $! $! Now check for /OUTPUT on the CALL $ if f$extract(l,4,dcl_record) .eqs. "/OUT" $ then $ l = f$locate("=",dcl_record) $ l = l + 1 $ if f$extract(l,1,dcl_record) .eqs. " " then l = l + 1 $call_out_loop: $ if f$extract(l,1,dcl_record) .nes. " " $ then $ l = l + 1 $ if l .lt. length then goto call_out_loop $ endif $ l = l + 1 $ endif $ gosub check_for_label $! $ call$call = 0 ! Assume it is NOT a "CALL" $ endif $after_call: $! $ goto read_dcl2 $! $! ======================================================================= $check_for_label: $! $ labrefinsr = 0 ! Start with assumption label referenced in a subroutine $ labfoundsr = 0 ! and not found in the same subroutine $! $! At this point a l points to the first character of the label. $! it could be terminated by a space (" ") or slash ("/") $ label = "" $extract_label_loop: $ x = f$extract(l,1,dcl_record) $ if ( (x .eqs. " ") .or. (x .eqs. "/") .or. (x .eqs. "") ) then - goto now_have_label $ label = label + x $ l = l + 1 $ goto extract_label_loop $! $now_have_label: $! If it contains a "'" it is symbol substitution $ if f$locate("'",label) .lt. f$length(label) $ then $ if .not. suppress_TLS then write err_file f$fao( - "!5UL TLS-I target label ""!AS"" provided by symbol substitution (warning)", - line_number,label) $ return $ endif $! ----------------------------------------------------------------------- $! $ if_rec_found = 0 ! assume no if-rec found $! $! Check to see if it's in the label file $ label_key = f$fao("!255AS",label) $ read /err=no_such_label /key="''label_key'" label_file label_rec $ lab_marked = 0 $! $next_label: $! Extract the line number of the label $ label_line = f$extract(1,5,label_rec) $! $! Check if the label's line number is in an if-group $! If it is, check that the reference is in the same if-group $! $! N.B. -- Because the key in IF_FILE is in descending order, the $! /match=gt has the effect of finding the first record that is $! LESS THAN the /key value. (i.e. "greater" really means "next") $ read /err=endif_range - /key="''label_line'" /match=gt if_file if_rec $ if_rec_found = 1 $! $read_next_if: $! get the start and end of the range $ if_start = f$extract(0,5,if_rec) $ if_end = f$extract(5,5,if_rec) $ if label_line .gt. if_end $ then $ if ( .not. lab_marked ) - .and. ( f$extract(0,1,label_rec) .nes. "D" ) $ then $! Mark the label as having been referenced $ label_rec[0,1] := "1" $ write/symbol/update label_file label_rec $ lab_marked = 1 $ endif $! $! goto endif_range $! Try again $ read /err=endif_range if_file if_rec $ if_rec_found = 1 $ goto read_next_if $! $ endif $! $ if_sr = f$extract(10,1,if_rec) $! $! Is the label_line in the if range? $! Note: if the label_line is in multiple, nested if ranges, then the $! first one we find is the outermost one -- Honest! $ if ( (label_line .gt. if_start) .and. (label_line .lt. if_end) ) $ then ! this is the one $ if ( (line_number .lt. if_start) .or. (line_number .gt. if_end) ) $ then ! reference is not in the same if range with the label $ if if_sr .eqs. "I" $ then $ if .not. suppress_RLI then write err_file f$fao( - "!5UL RLI-S referenced label ""!AS"" is in if-group at lines !UL-!UL", - line_number,label,f$integer(if_start),f$integer(if_end)) $ endif $ if if_sr .eqs. "S" $ then $ if .not. suppress_RLS then write err_file f$fao( - "!5UL RLS-W referenced label ""!AS"" is in subroutine at lines !UL-!UL", - line_number,label,f$integer(if_start),f$integer(if_end)) $ endif $ endif $ else $ if ( .not. lab_marked ) - .and. ( f$extract(0,1,label_rec) .nes. "D" ) $ then $! Mark the label as having been referenced $ label_rec[0,1] := "1" $ write/symbol/update label_file label_rec $ lab_marked = 1 $ endif $ endif $! Try again $ read /err=endif_range if_file if_rec $ if_rec_found = 1 $ goto read_next_if $! $endif_range: $! $! ........................................................................ $! Now, if we are in a subroutine, we need to check that the $! target label is in the same subroutine $! $! ... but not if this label was on a "CALL $ if call$call $ then $ if ( .not. lab_marked ) - .and. ( f$extract(0,1,label_rec) .nes. "D" ) $ then $! Mark the label as having been referenced $ label_rec[0,1] := "1" $ write/symbol/update label_file label_rec $ lab_marked = 1 $ endif $! $ goto after_lns $ endif $! $! Zero the sr_start and _end symbols $ sr_start = 0 $ sr_end = 0 $! $ read /err=endsr_range - /key="''line_number'" /match=gt if_file if_rec $ if_rec_found = 1 $read_next_sr: $! get the start and end ONLY of a subroutine range $ if_start = f$extract(0,5,if_rec) $ if_end = f$extract(5,5,if_rec) $ if label_line .gt. if_end then goto endsr_range $ if_sr = f$extract(10,1,if_rec) $! Is the line in this range? $ if (line_number .ge. if_start) .and. (line_number .lt. if_end) $ then ! We're in the range $! Is it a subroutine range? $ if if_sr .eqs. "S" $ then ! It is a subroutine range $ sr_start = if_start $ sr_end = if_end $ labrefinsr = 1 ! Remember label is referneced in a subroutine $ endif $ endif $! Try again $ read /err=endsr_range if_file if_rec $ if_rec_found = 1 $ goto read_next_sr $! $endsr_range: $! $ if ( (sr_end .gt. 0 ) .and. - ( (label_line .ge. sr_start) .and. (label_line .le. sr_end) ) ) $ then ! we found the label in the same subroutine $! $ if (.not. lab_marked) .and. (.not. labfoundsr) $ then $! Mark the label as having been referenced $ label_rec[0,1] := "1" $ write/symbol/update label_file label_rec $ lab_marked = 1 $ endif $ labfoundsr = 1 $ endif $! $! Read the next label_file entry $ read /err=after_lns label_file label_rec $ lab_marked = 0 $ if f$extract(6,255,label_rec) .eqs. label_key then goto next_label $! $after_lns: $! $ if labrefinsr .and. .not. labfoundsr $ then ! the target label is not in the same subroutine $ if .not. suppress_LNS then write err_file f$fao( - "!5UL LNS-S referenced label ""!AS"" is not in this subroutine", - line_number,label) $ endif $! $ if .not. lab_marked .and. ((.not. if_rec_found) .or. (.not. labfoundsr)) $ then $! Read label_file to re-establish a a current record $ read /err=no_such_label /key="''label_key'" label_file label_rec $! Mark the label as having been referenced $ label_rec[0,1] := "1" $ write/symbol/update label_file label_rec $ lab_marked = 1 $ return $! $ endif $! $ return $! ---------------------------------------------------------------------- $! $no_such_label: $! $ if .not. suppress_LNF then write err_file f$fao( - "!5UL LNF-S label ""!AS"" not found", - line_number,label) $! $ return $! ========================================================================= $! $end_dcl2: $! $ close work_file $ close label_file $ close file_file $ close if_file $! $ open /read label_file sys$scratch:dcl$label_'pid'.idx $read_label: $ read/end=end_label label_file x $! $ dup_lab = f$extract(0,1,x) $ if dup_lab .eqs. "1" then goto read_label $! $ if dup_lab .eqs. "0" $ then $ line_number = f$integer(f$extract(1,5,x)) $ label = f$edit(f$extract(6,2555,x),"TRIM") $ if .not. suppress_LNR then write err_file f$fao( - "!5UL LNR-I label ""!AS"" not referenced (warning)", - line_number,label) $ goto read_label $ endif $! $ if dup_lab .eqs. "D" $ then $ line_number = f$integer(f$extract(1,5,x)) $ label = f$edit(f$extract(6,2555,x),"TRIM") $ if .not. suppress_DL then write err_file f$fao( - "!5UL DL--E duplicate label ""!AS""", - line_number,label) $ goto read_label $ endif $! $ goto read_label $! $end_label: $ close label_file $!$ delete /nolog sys$scratch:dcl$label_'pid'.idx;* $! $! $ Say "Starting Pass 3 -- ''f$time()' ..." $! $! This pass searches for $! $! ICF-W invalid character found (#, %, ^ or &) $! ICO-S invalid comparison operator", - $! LC--S line contains END_IF, END IF, GO_TO, GO TO, GO_SUB or GO SUB", - $! $ saved_line = 0 $! $! delete any "leftover" files $ if f$search("sys$scratch:dcl$inv_char_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$inv_char_'pid'.tmp;* $ if f$search("sys$scratch:dcl$comp_op_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op_'pid'.tmp;* $ if f$search("sys$scratch:dcl$comp_op1_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op1_'pid'.tmp;* $ if f$search("sys$scratch:dcl$spell_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$spell_'pid'.tmp;* $! $! Search for invalid characters -- $! #, %, ^ and & are not valid in commands, symbols or expressions $! (At this point, all quoted strings have been removed) $! $ define /user sys$output nl: $ define /user sys$error nl: $ search sys$scratch:dcl$work_'pid'.tmp - /out=sys$scratch:dcl$inv_char_'pid'.tmp - "#", "%", "^", "&" $ work_status = $status $! $status %X08D78053 indicates "No strings found" $ if work_status .ne. %X08D78053 $ then $ open search_file sys$scratch:dcl$inv_char_'pid'.tmp $! $read_search0: $ read /end=end_search0 search_file dcl_record $! $! %X and %O (HEX and OCTAL radix) are O.K. $ if f$locate("%X",dcl_record) .lt. f$length(dcl_record) - then goto read_search0 $ if f$locate("%O",dcl_record) .lt. f$length(dcl_record) - then goto read_search0 $! $! The original line number was saved in the file we searched $! Separate the line number. $ line_number = f$integer(f$extract(0,5,dcl_record)) $! $! "&" is usually O.K. in a PIPE statement; make it a warning. $ if f$locate("&",dcl_record) .lt. f$length(dcl_record) - .and. f$locate("PIPE",dcl_record) .lt. f$length(dcl_record) $ then ! make it a warning. $ if .not. suppress_ICF then write err_file f$fao( - "!5UL ICF-W possible invalid character found (& in PIPE) (warning)", - line_number) $ goto read_search0 $ endif $! $! % and & may be used correctly (wild card and symbol substitution) $! so if they were found, make it a warning. $! $ if f$locate("%",dcl_record) .lt. f$length(dcl_record) - .or. f$locate("&",dcl_record) .lt. f$length(dcl_record) $ then ! make it a warning. $ if .not. suppress_ICF then write err_file f$fao( - "!5UL ICF-W possible invalid character found ( % or &) (warning)", - line_number) $ goto read_search0 $ endif $! $! ! Else not a warning $ if .not. suppress_ICF then write err_file f$fao( - "!5UL ICF-W invalid character found (# or ^)", - line_number) $ goto read_search0 $! $end_search0: $ close search_file $ endif $! $! $! Search for invalid comparison operators $! $ define /user sys$output nl: $ define /user sys$error nl: $ search sys$scratch:dcl$work_'pid'.tmp - /out=sys$scratch:dcl$comp_op_'pid'.tmp - " .eq ", " eq. ", " eq ", - " .ge ", " ge. ", " ge ", - " .gt ", " gt. ", " gt ", - " .le ", " le. ", " le ", - " .lt ", " lt. ", " lt ", - " .ne ", " ne. ", " ne ", - " .or ", " or. ", " or ", - " .not "," not. "," not ", - " .and "," and. "," and ", - " .eqs "," eqs. "," eqs ", - " .ges "," ges. "," ges ", - " .gts "," gts. "," gts ", - " .les "," les. "," les ", - " .lts "," lts. "," lts ", - " .nes "," nes. "," nes ", - " .new "," new. "," .new. ", - " .eas "," eas. "," .eas. ", - " .ea "," ea. "," .ea. ", - " .gs. "," .ls. "," .es. "," .ns. ", - ">=","=>","<=","=<"," _ " $ work_status = $status $! $status %X08D78053 indicates "No strings found" $ if work_status .ne. %X08D78053 $ then $ open search_file sys$scratch:dcl$comp_op_'pid'.tmp $! $read_search1: $ read /end=end_search1 search_file dcl_record $! $! The original line number was saved in the file we searched $! Separate the line number. $! $ line_number = f$integer(f$extract(0,5,dcl_record)) $ if .not. suppress_ICO then write err_file f$fao( - "!5UL ICO-S invalid comparison operator", - line_number) $ goto read_search1 $end_search1: $ close search_file $ endif $! $ if f$search("sys$scratch:dcl$comp_op_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op_'pid'.tmp;* $! $ define /user sys$output nl: $ define /user sys$error nl: $ search sys$scratch:dcl$work_'pid'.tmp " > "," < " - /out=sys$scratch:dcl$comp_op1_'pid'.tmp $! $ define /user sys$output nl: $ define /user sys$error nl: $ search sys$scratch:dcl$comp_op1_'pid'.tmp "PIP" /match=nor - /out=sys$scratch:dcl$comp_op_'pid'.tmp $ if f$search("sys$scratch:dcl$comp_op1_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op1_'pid'.tmp;* $! $ open search_file sys$scratch:dcl$comp_op_'pid'.tmp $read_search1a: $ read /end=end_search1a search_file dcl_record $ if f$locate("NO STRINGS MATCHED",(f$edit(dcl_record,"UPCASE"))) - .lt. f$length(dcl_record) - then goto end_search1a $ line_number = f$integer(f$extract(0,5,dcl_record)) $ if .not. suppress_ICO then write err_file f$fao( - "!5UL ICO-S invalid comparison operator", - line_number) $ goto read_search1a $end_search1a: $ close search_file $ if f$search("sys$scratch:dcl$comp_op_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op_'pid'.tmp;* $! $ define /user sys$output nl: $ define /user sys$error nl: $ search sys$scratch:dcl$work_'pid'.tmp - /out=sys$scratch:dcl$spell_'pid'.tmp - " end_if"," end if", - " go_to"," go to", - " go_sub"," go sub" $ work_status = $status $! $status %X08D78053 indicates "No strings found" $ if work_status .eq. %X08D78053 $ then $ spell_found = 0 $ else $ spell_found = 1 $ open search_file sys$scratch:dcl$spell_'pid'.tmp $! $read_search2: $ read /end=end_search2 search_file dcl_record $! $! The original line number was saved in the file we searched $! Separate the line number. $! $ line_number = f$integer(f$extract(0,5,dcl_record)) $ if .not. suppress_LC then write err_file f$fao( - "!5UL LC--S line contains END_IF, END IF, GO_TO, GO TO, GO_SUB or GO SUB", - line_number) $ goto read_search2 $end_search2: $ close search_file $ endif $! $! $ close err_file $! $ sort /stable /key=(pos:1,size:5) - sys$scratch:dcl$error_'pid'.tmp - sys$scratch:dcl$error_'pid'.tmp $! $! count lines in err_file $ open /read err_file sys$scratch:dcl$error_'pid'.tmp $ open /write err_out sys$scratch:dcl$error_'pid'.tmp $ diag_count = 0 $ is_comment = 1 $ comment_line = 0 $ prior_blank = 1 $read_count: $ read /end=end_count err_file err_rec $ diag_count = diag_count + 1 $! $! set prior_comment $ if is_comment $ then $ prior_comment = 1 $ endif $! $! Check for DCL_COMMENT lines $ if f$extract(7,6,err_rec) .eqs. "-CMNT-" $ then $ is_comment = 1 $ else $ is_comment = 0 $ endif $! $! if this is a comment and the prior line wasn't, output a blank line $ if is_comment .and. (.not. prior_comment) .and. (.not. prior_blank) $ then $ write err_out "" $ prior_blank = 1 $ prior_comment = 0 $ endif $! $! get the line number $ line_number = f$integer(f$extract(0,5,err_rec)) $! $! save the line number of a comment line $ if is_comment $ then $ comment_line = comment_line + 1 $ if (line_number .ne. comment_line) .and. (.not. prior_blank) $ then $ write err_out "" $ prior_blank = 1 $ prior_comment = 0 $ endif $ comment_line = line_number $ else ! check if it is a consecutive with a comment line $ if prior_comment $ then $ if (line_number .ne. comment_line + 1) .and. (.not. prior_blank) $ then $ write err_out "" $ prior_blank = 1 $ prior_comment = 0 $ endif $ endif $ endif $ write err_out err_rec $ prior_blank = 0 $ goto read_count $end_count: $ close err_file $ close err_out $ purge sys$scratch:dcl$error_'pid'.tmp $! $ dcl_end_time = f$time() $ if f$file_ATTRIBUTES("sys$scratch:dcl$error_''pid'.tmp","eof") .gt. 0 $ then $ if ( (p2 .nes. "") .and. (p2 .nes. "$") ) $ then $ report$file = f$parse(p2,".LIS") $ say "" $ say "Creating errors listing in ''report$file'" $ say "''dcl_end_time'" $ say "" $! open the report file $! Attempt to close it in case it was left open $ close/err=open_report rep_file $open_report: $ create 'report$file' ! create gets file characteristics right $ open /append rep_file 'report$file' $ write rep_file - "-*- Charlie Hammond's unsupported DCL checker (Version ''dcl_ck_vers') -*-" $ write rep_file "Checking file ''dcl$file'" $ write rep_file "''dcl_ck_time'" $ write rep_file "" $ write rep_file "Checking for DCL_CHECK$ logicals..." $ close rep_file $! $ if use_pipe then - pipe sho log DCL_CHECK$* | - sear sys$input DCL_CHECK$ | - append sys$input 'report$file' $! $ open /append rep_file 'report$file' $ write rep_file "" $ write rep_file f$fao( - "Procedure contains:!7UL total lines",total_lines) $ write rep_file f$fao( - " !7UL code lines (including !UL lines w/ comments)", - code_lines, commented_code_lines) $ write rep_file f$fao( - " !7UL additional continuation lines",continuation_lines) $ write rep_file f$fao( - " !7UL lines w/i $DECK/$EOD pairs",deck_lines) $ write rep_file f$fao( - " !7UL comment only lines",comment_lines) $ write rep_file f$fao( - " !7UL blank lines",blank_lines) $ write rep_file f$fao( - " !7UL diagnostics",diag_count) $ write rep_file "" $ write rep_file " LINE CODE --DIAGNOSTIC MESSAGE--" $ write rep_file "" $ close rep_file $ append sys$scratch:dcl$error_'pid'.tmp,sys$input 'report$file' $DECK -*- END OF LISTING -*- $EOD $ else $ say "" $ say f$fao( - "Procedure contains:!7UL total lines",total_lines) $ say f$fao( - " !7UL code lines (including !UL lines w/ comments)", - code_lines, commented_code_lines) $ say f$fao( - " !7UL additional continuation lines",continuation_lines) $ say f$fao( - " !7UL lines w/i $DECK/$EOD pairs",deck_lines) $ say f$fao( - " !7UL comment only lines",comment_lines) $ say f$fao( - " !7UL blank lines",blank_lines) $ say f$fao( - " !7UL diagnostics",diag_count) $ say "" $ say " LINE CODE --DIAGNOSTIC MESSAGE--" $ say "" $ type sys$scratch:dcl$error_'pid'.tmp $ say "" $ say "-*- END OF LISTING -*- ''dcl_end_time'" $ say "" $ endif $ else $ say "" $ say f$fao( - "Procedure contains:!7UL total lines",total_lines) $ say f$fao( - " !7UL code lines (including !UL lines w/ comments)", - code_lines, commented_code_lines) $ say f$fao( - " !7UL additional continuation lines",continuation_lines) $ say f$fao( - " !7UL lines w/i $DECK/$EOD pairs",deck_lines) $ say f$fao( - " !7UL comment only lines",comment_lines) $ say f$fao( - " !7UL blank lines",blank_lines) $ say f$fao( - " !7UL diagnostics",diag_count) $ say "" $ say "-*- No errors found -*- ''f$time()'" $ if ( (p2 .nes. "") .and. (p2 .nes. "$") ) then - say "...listing file not created ''dcl_end_time'" $ endif $! $ say "" $ set noon $ define /user sys$output nl: $ define /user sys$error nl: $ elapsed_time = f$delta_time(dcl_ck_time,dcl_end_time) $ delta_status = $status $ set on $ deassign /user sys$output $ deassign /user sys$error $ if delta_status .nes. "%X00030001" $ then $ if delta_status .eqs. "%X000381C0" $ then $!$ say f$fao - $! ("*** F$DELTA_TIME not implemented in this version of OpenVMS!!!/") $ else $ say f$fao ("!/!AS!/",f$message(delta_status)) $ endif $ else $ say f$fao("Elapsed Time: !AS!/",elapsed_time) $ endif $! $ goto common_exit $! $! ------------------------------------------------------------ $! Exit routines $! $y_exit: ! Ctrl_y exit routine $! $! Display Ctrl_y message $! $ say "Exiting due to Ctrl_y entry" $! $ goto 1_exit $! $err_exit: ! error/warning exit routine $! $ sav_status = $status $err_exit_w_status: $ say f$message(sav_status) $! Add %x10000000 to set the bit that suppresses display of the message. $! This prevents re-displaying the message when we EXIT SAV_STATUS. $ if sav_status .lt. %x10000000 then sav_status = sav_status + %x10000000 $! $ say f$fao - ("!/Error occured processing at or near source line !5UL",saved_line) $! $ goto common_exit $! $1_exit: ! Go here to force exit with status 1 = success $! %SYSTEM-S-NORMAL, normal successful completion $! $! Display success message, if desired $! $ say f$message(sav_status) $! $! Note: success messages aren't displayed by EXIT SAV_STATUS. $! Don't need to use %x10000001. $ sav_status = 1 $! $ say f$fao - ("!/Error occured processing at or near source line !5UL",saved_line) $! $ goto common_exit $! $common_exit: ! common exit $! $! Disable control_y and error handling $ on control_y then continue $ on warning then continue $! $! Cleanup code $! $! Close any file left open... $! $ close/error=y_1 err_file $y_1: $ close/error=y_2 label_file $y_2: $ close/error=y_2a file_file $y_2a: $ close/error=y_3 work_file $y_3: $ close/error=y_4 dcl_file $y_4: $ close/error=y_5 rep_file $y_5: $ close/error=y_6 if_file $y_6: $! ... $! $! Deassign logicals $! $! Delete temporary files $! $ if f$search("sys$scratch:dcl$error_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$error_'pid'.tmp;* $ if f$search("sys$scratch:dcl$label_''pid'.idx") .nes. "" then - delete /nolog sys$scratch:dcl$label_'pid'.idx;* $ if f$search("sys$scratch:dcl$file_''pid'.idx") .nes. "" then - delete /nolog sys$scratch:dcl$file_'pid'.idx;* $ if f$search ("sys$scratch:dcl$work_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$work_'pid'.tmp;* $ if f$search("sys$scratch:dcl$inv_char_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$inv_char_'pid'.tmp;* $ if f$search("sys$scratch:dcl$comp_op_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op_'pid'.tmp;* $ if f$search("sys$scratch:dcl$comp_op1_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$comp_op1_'pid'.tmp;* $ if f$search("sys$scratch:dcl$spell_''pid'.tmp") .nes. "" then - delete /nolog sys$scratch:dcl$spell_'pid'.tmp;* $ if f$search("sys$scratch:dcl$if_''pid'.idx") .nes. "" then - delete /nolog sys$scratch:dcl$if_'pid'.idx;* $! $! ... $! $! Any other cleanup required... $! $! Restore message facility to original state $! $ set message 'sav_msg' $! $! Exit with status $! $ exit sav_status