1 Introduction
2 Ground Rules

Building a File System
3 File Systems
4 File Content Data Structure
5 Allocation Cluster Manager
6 Exceptions and Emancipation
7 Base Classes, Testing, and More
8 File Meta Data
9 Native File Class
10 Our File System
11 Allocation Table
12 File System Support Code
13 Initializing the File System
14 Contiguous Files
15 Rebuilding the File System
16 Native File System Support Methods
17 Lookups, Wildcards, and Unicode, Oh My
18 Finishing the File System Class

The Init Program
19 Hardware Abstraction and UOS Architecture
20 Init Command Mode
21 Using Our File System
22 Hardware and Device Lists
23 Fun with Stores: Partitions
24 Fun with Stores: RAID
25 Fun with Stores: RAM Disks
26 Init wrap-up

The Executive
27 Overview of The Executive
28 Starting the Kernel
29 The Kernel
30 Making a Store Bootable
31 The MMC
32 The HMC
33 Loading the components
34 Using the File Processor
35 Symbols and the SSC
36 The File Processor and Device Management
37 The File Processor and File System Management
38 Finishing Executive Startup

Users and Security
39 Introduction to Users and Security
40 More Fun With Stores: File Heaps
41 File Heaps, part 2
42 SysUAF
43 TUser
44 SysUAF API

Terminal I/O
45 Shells and UCL
46 UOS API, the Application Side
47 UOS API, the Executive Side
48 I/O Devices
49 Streams
50 Terminal Output Filters
51 The TTerminal Class
52 Handles
53 Putting it All Together
54 Getting Terminal Input
55 QIO
56 Cooking Terminal Input
57 Putting it all together, part 2
58 Quotas and I/O

UCL
59 UCL Basics
60 Symbol Substitution
61 Command execution
62 Command execution, part 2
63 Command Abbreviation
64 ASTs
65 Expressions, Part 1
66 Expressions, Part 2: Support code
67 Expressions, part 3: Parsing
68 SYS_GETJPIW and SYS_TRNLNM
69 Expressions, part 4: Evaluation

UCL Lexical Functions
70 PROCESS_SCAN
71 PROCESS_SCAN, Part 2
72 TProcess updates
73 Unicode revisted
74 Lexical functions: F$CONTEXT
75 Lexical functions: F$PID
76 Lexical Functions: F$CUNITS
77 Lexical Functions: F$CVSI and F$CVUI
78 UOS Date and Time Formatting
79 Lexical Functions: F$CVTIME
80 LIB_CVTIME
81 Date/Time Contexts
82 SYS_GETTIM, LIB_Get_Timestamp, SYS_ASCTIM, and LIB_SYS_ASCTIM
83 Lexical Functions: F$DELTA_TIME
84 Lexical functions: F$DEVICE
85 SYS_DEVICE_SCAN
86 Lexical functions: F$DIRECTORY
87 Lexical functions: F$EDIT and F$ELEMENT
88 Lexical functions: F$ENVIRONMENT
89 SYS_GETUAI
90 Lexical functions: F$EXTRACT and F$IDENTIFIER
91 LIB_FAO and LIB_FAOL
92 LIB_FAO and LIB_FAOL, part 2
93 Lexical functions: F$FAO
94 File Processing Structures
95 Lexical functions: F$FILE_ATTRIBUTES
96 SYS_DISPLAY
97 Lexical functions: F$GETDVI
98 Parse_GetDVI
99 GetDVI
100 GetDVI, part 2
101 GetDVI, part 3
102 Lexical functions: F$GETJPI
103 GETJPI
104 Lexical functions: F$GETSYI
105 GETSYI
106 Lexical functions: F$INTEGER, F$LENGTH, F$LOCATE, and F$MATCH_WILD
107 Lexical function: F$PARSE
108 FILESCAN
109 SYS_PARSE
110 Lexical Functions: F$MODE, F$PRIVILEGE, and F$PROCESS
111 File Lookup Service
112 Lexical Functions: F$SEARCH
113 SYS_SEARCH
114 F$SETPRV and SYS_SETPRV
115 Lexical Functions: F$STRING, F$TIME, and F$TYPE
116 More on symbols
117 Lexical Functions: F$TRNLNM
118 SYS_TRNLNM, Part 2
119 Lexical functions: F$UNIQUE, F$USER, and F$VERIFY
120 Lexical functions: F$MESSAGE
121 TUOS_File_Wrapper
122 OPEN, CLOSE, and READ system services

UCL Commands
123 WRITE
124 Symbol assignment
125 The @ command
126 @ and EXIT
127 CRELNT system service
128 DELLNT system service
129 IF...THEN...ELSE
130 Comments, labels, and GOTO
131 GOSUB and RETURN
132 CALL, SUBROUTINE, and ENDSUBROUTINE
133 ON, SET {NO}ON, and error handling
134 INQUIRE
135 SYS_WRITE Service
136 OPEN
137 CLOSE
138 DELLNM system service
139 READ
140 Command Recall
141 RECALL
142 RUN
143 LIB_RUN
144 The Data Stream Interface
145 Preparing for execution
146 EOJ and LOGOUT
147 SYS_DELPROC and LIB_GET_FOREIGN

CUSPs and utilities
148 The I/O Queue
149 Timers
150 Logging in, part one
151 Logging in, part 2
152 System configuration
153 SET NODE utility
154 UUI
155 SETTERM utility
156 SETTERM utility, part 2
157 SETTERM utility, part 3
158 AUTHORIZE utility
159 AUTHORIZE utility, UI
160 AUTHORIZE utility, Access Restrictions
161 AUTHORIZE utility, Part 4
162 AUTHORIZE utility, Reporting
163 AUTHORIZE utility, Part 6
164 Authentication
165 Hashlib
166 Authenticate, Part 7
167 Logging in, part 3
168 DAY_OF_WEEK, CVT_FROM_INTERNAL_TIME, and SPAWN
169 DAY_OF_WEEK and CVT_FROM_INTERNAL_TIME
170 LIB_SPAWN
171 CREPRC
172 CREPRC, Part 2
173 COPY
174 COPY, part 2
175 COPY, part 3
176 COPY, part 4
177 LIB_Get_Default_File_Protection and LIB_Substitute_Wildcards
178 CREATESTREAM, STREAMNAME, and Set_Contiguous
179 Help Files
180 LBR Services
181 LBR Services, Part 2
182 LIBRARY utility
183 LIBRARY utility, Part 2
184 FS Services
185 FS Services, Part 2
186 Implementing Help
187 HELP
188 HELP, Part 2
189 DMG_Get_Key and LIB_Put_Formatted_Output
190 LIBRARY utility, Part 3
191 Shutting Down UOS
192 SHUTDOWN
193 WAIT
194 SETIMR
195 WAITFR and Scheduling
196 REPLY, OPCOM, and Mailboxes
197 REPLY utility
198 Mailboxes
199 BRKTHRU
200 OPCOM
201 Mailbox Services
202 Mailboxes, Part 2
203 DEFINE
204 CRELNM
205 DISABLE
206 STOP
207 OPCCRASH and SHUTDOWN
208 APPEND

Glossary/Index


Downloads

IF...THEN...ELSE

Being able to do calculations and output things is essential to any scripting language, but very few scripts are useful unless they can also use conditional logic. In this article we'll look at the IF statement, and associated commands, in UCL. Here is the user documentation:

IF

IF tests the value of an expression and, depending upon the form uses, executes one of the following:

  • If THEN follows IF on the same line: if the expression is true, the command after THEN is executed; if the expression is false, execution continues with the next line.
  • If THEN follows IF on the next line: if the expression is true, all of the commands after that point are executed until ELSE or ENDIF is encountered; if the expression is false, execution begins after the next ELSE (or ENDIF if ELSE is not specified).
Note that the expression evaluates to false if it has an even numeric value or is a string that starts with any letter other than Y, y, T, or t. The expression evaluates to true if it has an odd numeric value or is a string that starts with Y, y, T, or t.

Format
Form 1:


$ IF expression THEN {$} command

Form 2:

$ IF expression
$ THEN {{$} command}
command
.
.
.
$ ENDIF

Form 3:

$ IF expression
$ THEN {{$} command}
command
.
.
.
$ ELSE {command}
command
.
.
.
$ ENDIF

Parameters
expression

Defines the comparison to be made. The result of the comparison is either 0 (false) or non-zero (true). Comparisons can be numeric or string literals, symbol names, or lexical functions separated by operators. Any string not enclosed within quotes (") are assumed to be symbol names. If an unknown symbol name is specified, an error will result and the next command is executed.

command

Specifies command(s) to be executed under the appropriate circumstances based on the true or false value of the expression.

Examples:
1. Form 1


$ IF P1 .EQS. "" THEN EXIT
$ IF P1 .EQS. "X" THEN WRITE SYS$OUTPUT "Invalid parameter value"

2. Form 2

$ IF F$ENVIRONMENT("INTERACTIVE")
$ THEN
$ WRITE SYS$OUTPUT "UOS version: ", F$GETSYI("VERSION")
$ ENDIF

3. Nested IFs

$ IF AVAILABLE .EQ. 0
$ THEN
$   IF AMOUNT .GE. 0
$   THEN WRITE SYS$OUTPUT "It doesn't fit"
$   ELSE WRITE SYS$OUTPUT "Copying next item..."
$   ENDIF
$ ENDIF
Note: the extra indentation is not required, but using it makes the script more easily understood.

Processing the second form of the IF command requires that we keep track of additional context so we can properly handle nested IFs. It also requires a state machine approach to determine where we are in the IF...THEN...ELSE...ENDIF structure. For example, before we hit an IF, we are in the "normal" state. When we find an IF without a THEN on the same line, we enter the waiting-for-then state if the condition is true, or the waiting-for-else state otherwise. If we are waiting for THEN, the next command must be one. Otherwise, we skip commands until we come to the ELSE or ENDIF. If we hit ELSE while waiting for it, we enter the "normal" state again and start executing commands. But if we were processing the THEN, we move to the waiting-for-endif state. ENDIF ends either the waiting-for-else or the waiting-for-endif states. But we also have to handle nested IFs. It wouldn't do to end the outer THEN because we hit an inner ENDIF. This is where the nesting comes in. Each IF (without a THEN on the same line) puts us into a new nesting level with it's own state (starting in the "normal" state). IFs can be nested indefinitely this way.

                            Nesting : TList ;

                        public // Constructor and destructor...
                            constructor Create ;
                            destructor Destroy ; override ;

                        public // API...
                            procedure Nest ;
                            procedure Unnest ;
                            function Get_Awaiting_Then : boolean ;
                            procedure Set_Awaiting_Then( Value : boolean ) ;
                            function Get_Awaiting_Else : boolean ;
                            procedure Set_Awaiting_Else( Value : boolean ) ;
                            function Get_Awaiting_Endif : boolean ;
                            procedure Set_Awaiting_Endif( Value : boolean ) ;

                        public // Properties...
                            property Awaiting_Then : boolean
                                read Get_Awaiting_Then
                                write Set_Awaiting_Then ;
                            property Awaiting_Else : boolean
                                read Get_Awaiting_Else
                                write Set_Awaiting_Else ;
                            property Awaiting_Endif : boolean
                                read Get_Awaiting_Endif
                                write Set_Awaiting_Endif ;
The TUCL_Context class is now more than merely a container of context - we have added some properties as well. The Nesting instance data will hold our nesting contexts, which we will describe below. The Nest and Unnest methods are used to create a new nest level and exit the current nest level, respectively. The rest of the methods are property handlers. The properties access the current state of the current nested level.

// Constructor and destructor...

constructor TUCL_Context.Create ;

begin
    inherited Create ;

    Nesting := TList.Create ;
end ;


destructor TUCL_Context.Destroy ;

var I : integer ;

begin
    for I := 0 to Nesting.Count - 1 do
    begin
        TUCL_Nest_Context( Nesting[ I ] ).Free ;
        Nesting[ I ] := nil ;
    end ;
    Nesting.Free ;
    Nesting := nil ;
    inherited Destroy ;
end ;


// API...

procedure TUCL_Context.Nest ;

begin
    Nesting.Add( TUCL_Nest_Context.Create ) ;
end ;


procedure TUCL_Context.Unnest ;

begin
    if( Nesting.Count > 0 ) then
    begin
        TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Free ;
        Nesting.Count := Nesting.Count - 1 ;
    end ;
end ;


function TUCL_Context.Get_Awaiting_Then : boolean ;

begin
    if( Nesting.Count = 0 ) then
    begin
        Result := False ;
        exit ;
    end ;
    Result := TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Awaiting_Then ;
end ;


procedure TUCL_Context.Set_Awaiting_Then( Value : boolean ) ;

begin
    if( Nesting.Count > 0 ) then
    begin
        TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Awaiting_Then := Value ;
    end ;
end ;


function TUCL_Context.Get_Awaiting_Else : boolean ;

begin
    if( Nesting.Count = 0 ) then
    begin
        Result := False ;
        exit ;
    end ;
    Result := TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Awaiting_Else ;
end ;


procedure TUCL_Context.Set_Awaiting_Else( Value : boolean ) ;

begin
    if( Nesting.Count > 0 ) then
    begin
        TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Awaiting_Else := Value ;
    end ;
end ;


function TUCL_Context.Get_Awaiting_Endif : boolean ;

begin
    if( Nesting.Count = 0 ) then
    begin
        Result := False ;
        exit ;
    end ;
    Result := TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Awaiting_Endif ;
end ;


procedure TUCL_Context.Set_Awaiting_Endif( Value : boolean ) ;

begin
    if( Nesting.Count > 0 ) then
    begin
        TUCL_Nest_Context( Nesting[ Nesting.Count - 1 ] ).Awaiting_Endif := Value ;
    end ;
end ;
This is fairly straight-forward code. The destructor must destruct all the nesting contexts when the object is freed, but that will only happen if UCL terminates. More likely is that when we exit a nest level, we will need to free that nest level's context. None of the property handlers should be called if we aren't nested, but we check because it is better to prevent the situation (in our code) or report the error (in the script) than to generate an exception.

type TUCL_Nest_Context = class
                             Awaiting_Then : boolean ;
                             Awaiting_Else : boolean ;
                             Awaiting_Endif : boolean ;
                         end ;
This new class simply holds our state for the current nesting level.

            if( This_UCL_Context.Awaiting_Then ) then
            begin
                This_UCL_Context.Awaiting_Then := False ;
                if( Sym <> 'then' ) then // Expecting THEN, but found something else
                begin
                    Exception( UCL_INVIFNEST, '' ) ;
                    exit ;
                end ;
                Sym := Edit( Get_Token, 512, 0 ) ; // Get first command following THEN
                if( Sym = '' ) then // All done with this command
                begin
                    exit ;
                end ;
            end ;
            if( This_UCL_Context.Awaiting_Else ) then
            begin
                if( Sym = 'if' ) then
                begin
                    Skip_If_Block ;
                end ;
                if( ( Sym <> 'else' ) and ( Sym <> 'endif' ) ) then
                begin
                    exit ;
                end ;
                This_UCL_Context.Awaiting_Else := False ;
                continue ;
            end ;
This code is added to the Process code just prior to parsing individual commands. Here we handle the various states described above. If we're in the waiting-for-then state, the very next command must be THEN. First we clear the state. If the next command isn't THEN, we exit with an error. Next we get the command following the THEN. If there isn't any, we return and the next command will be executed. If there is a command, we are now positioned in the input line at that command, so we just drop through to process the new command.

If we are in the waiting-for-else states, we check for a nested IF. If found, we call Skip_If_Block (described below) to skip to the end of that IF structure. If the next symbol isn't ELSE, we exit and wait for the next command to come along. Otherwise, we clear the waiting-for-else state, returning us to "normal" state and then continue to process the next command.

          if( Sym = 'if' ) then
          begin
              Process_If ;
          end else
          if( Sym = 'else' ) then // We've processed the THEN block and need to skip the ELSE block
          begin
              if( This_UCL_Context.Nesting.Count = 0 ) then // Not nested
              begin
                  Exception( UCL_INVIFNEST, '' ) ;
                  exit ;
              end ;
              Skip_If_Block ; // Skips to the endif (inclusive)
              exit ;
          end else
          if( Sym = 'endif' ) then
          begin
              if( This_UCL_Context.Nesting.Count = 0 ) then // Not nested
              begin
                  Exception( UCL_INVIFNEST, '' ) ;
                  exit ;
              end ;
              This_UCL_Context.Unnest ;
              exit ;
          end else
          if( Sym = 'then' ) then // Encountered by itself
          begin
              Exception( UCL_INVIFNEST, '' ) ;
              exit ;
          end else
This code is also added to the Process routine. These handle the various IF...THEN...ELSE...ENDIF commands. Process_If is called to process the IF command. In the case of ELSE, if we are not nested, it is an ELSE independent of any IF structure so we generate an error and exit. We also do that if we encounter a THEN. Otherwise, this is the ELSE portion of the IF structure we are currently in. If we get here, it means we've found the ELSE that ends the THEN portion of the IF structure that we've been processing. In that case, we call Skip_If_Block to skip to the ENDIF. If we encounter ENDIF, we again check that we are nested. If not, it is a lone ENDIF and we exit with an error. Otherwise, we unnest.

function Skip_If_Block : boolean ;

var Found : boolean ;
    Nest_Level : integer ;
    S : string ;

begin
    Result := False ;
    Nest_Level := 0 ;
    S := Get_Token ;
    while( S <> '' ) do
    begin
        if( lowercase( S ) = 'then' ) then // Single line if-then
        begin
            exit ;
        end ;
        S := Get_Token ;
    end ;
    while( true ) do
    begin
        S := Edit( Get_Command, 8 or 16 or 128 or 256 or 512 ) ;
        if( copy( S, 1, 3 ) = 'if ' ) then
        begin
            Found := False ;
            while( S <> '' ) do
            begin
                if( Parse_Parameter( ' ', S ) = 'then' ) then
                begin
                    Found := True ; // single-line IF
                    break ;
                end ;
            end ; // while( S <> '' )
            if( not Found ) then
            begin
                inc( Nest_Level ) ; // In a new IF block
            end ;
            continue ;
        end ;
        if( copy( S, 1, 5 ) = 'else ' ) then
        begin
            if( Nest_Level <= 0 ) then
            begin
                exit ;
            end ;
        end ;
        if( ( S = 'endif' ) or ( copy( S, 1, 6 ) = 'endif ' ) ) then
        begin
            dec( Nest_Level ) ;
            if( Nest_Level < 0 ) then
            begin
                exit ;
            end ;
            continue ;
        end ;
    end ;
end ; // Skip_If_Block
This routine will skip to the end of the current THEN or ELSE block in an IF structure. It is a little more complicated than just reading commands until ELSE or ENDIF is found, since we have to take nested IFs into consideration. If we encounter a multiple-line IF statement (form 2), we increment our internal nesting count, decrementing it when we hit the ENDIF. If we reduce our nesting count to 0, we are back to the outermost level of the block we are skipping. If the nesting count goes negative, that means we hit the ENDIF for the current block. When that happens, we exit, having consumed all the commands in the block.

procedure Process_If ;

var Condition, Err : integer ;
    Context, S, S1 : string ;

begin
    S := Get_Expression( Err, Context ) ; // Get conditional expression
    if(
        ( Err = UCL_EXPSYN )
        and
        ( ( lowercase( Parser.Peek ) = 'then' ) or ( Parser.Peek = '' ) )
      ) then
    begin
        Err := 0 ;
    end ;
    if( ( Err <> 0 ) or ( S = '' ) ) then // Error
    begin
        Exception( UCL_EXPSYN, Context ) ;
        exit ;
    end ;
    Condition := strtoint( UCL_Strtoint( S ) ) and 1 ; // Get integer version of conditional (low bit only)
This new routine will process the IF command. First we get the value of the expression following the IF. If Get_Expression returns a syntax error (which, in properly constructed single-line IF commands, it will) we check that the next token is THEN. If it is, we clear the error, otherwise there is a real syntax error and we generate an error and exit. If everything is fine, we convert the result of the expression to an integer and trim off all but the least significant bit. So, at this point, Condition is either 0 (false) or 1 (true).

    S1 := lowercase( Get_Token ) ;
    if( S1 = '' ) then // End of line before THEN
    begin
        if( Interactive ) then
        begin
            Exception( UCL_INVIFNEST, '' ) ;
            exit ;
        end ;
        This_UCL_Context.Nest ;
        if( Condition = 0 ) then
        begin
            This_UCL_Context.Awaiting_Else := True ;
        end else
        begin
            This_UCL_Context.Awaiting_Then := True ;
        end ;
        exit ;
    end ;
We get the next token, which must either be null (nothing more on the command line) or THEN. If there is no THEN following, this is a multi-line IF structure (form 2). However form 2 is not allowed outside of command files. Thus, if we are running interactively, we exit with an error. Otherwise, we nest. Then we set our state appropriately based on the value of the condition and exit.

    if( S1 <> 'then' ) then
    begin
        Exception( UCL_INVIFNEST, '' ) ;
        exit ;
    end ; // if( S1 <> 'then' )
    if( Parser.Peek = '$' ) then
    begin
        Get_Token ; // Swallow "$"
    end ;

    // Then portion of command
    if( Condition = 0 ) then
    begin
        S := Parser.Grab_Line ; // Eat rest of line
    end ;
If we get here, this is a single-line IF...THEN structure (form 1). In this case, the next token must be THEN. If it isn't, we exit with an error. If the next token is a dollar sign ($), we eat that. In this case, preceding a UCL command with a dollar sign is optional. Finally, if the condition was false, we grab the rest of the line (essentially throwing it away). Otherwise, when the routine ends, it goes back to the Process routine to process the rest of the line, which is the command following the THEN.

Now that we have IF commands working, we can create a regression test file. Regression tests are simply tests that make sure that things that used to work still work. So, we now have a RTest.com file which checks the operation of all the UCL features we've covered so far. As we continue, we will add new things to test. I won't include the entire file here (the current file can be viewed here), but the following is a sample of the file contents:

$ if f$match_wild("ABCDE","*BC*").nes."TRUE" then write sys$output "f$match_wild('ABCDE','*BC*') failed"
$ if f$match_wild("ABCDE","*").nes."TRUE" then write sys$output "f$match_wild('ABCDE','*') failed"
$ if f$match_wild("ABCDE","F").nes."FALSE" then write sys$output "f$match_wild('ABCDE','F') failed"
$ if f$locate("2","12345").nes."1" then write sys$output "f$locate('2','12345') failed"
$ if f$locate("1","12345").nes."0" then write sys$output "f$locate('1','12345') failed"
$ if f$locate("A","12345").nes."5" then write sys$output "f$locate('A','12345') failed"
$ if f$locate("5","12345").nes."4" then write sys$output "f$locate('5','12345') failed"
$ if f$locate("1","").nes."0" then write sys$output "f$locate('1','') failed"
When I ran this file, I found that there were some things that broke since the original code was written. I don't know when this happened, so the last binary release might have a few broken lexical functions. Fortunately, with the test file I can verify the operation of UCL before the next binary release. I actually found more errors in the test file itself than in UOS. That is typical: test code itself needs to be debugged as well as the code that it tests. Of course, once it is debugged, we can use it over and over to test for proper operation.

Note that I don't usually include corrected code in articles after the fact, unless there is something to be learned from it. So, there may be slight issues with code covered in some of these articles, but it still serves its instructive purpose. With each binary release, you can get the latest code.

In the next article, we will look at more UCL commands.

 

Copyright © 2021 by Alan Conroy. This article may be copied in whole or in part as long as this copyright is included.