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
|
CALL, SUBROUTINE, and ENDSUBROUTINE
In a recent article we discussed the @ command, which is the first type of subroutine
supported by UCL. In the previous article, we discussed the GOSUB command, which is
the second subroutine mechanism supported by UCL. @ executes a different script file
and creates a new scope, whereas GOSUB doesn't create a new scope and only
calls a subroutine in the current script file. CALL is the third subroutine mechanism
and is a cross between @ and GOSUB. Like GOSUB, a local subroutine within the current
script file is called. But like @, parameters can be passed to the subroutine, and
output can be automatically redirected during subroutine execution.
However, because there is no new scope, the P1... symbols are set within the current
scope.
I should note the distinction between a nesting level (or "context") and a scope. A
scope has to do with a set of symbols that are local to that scope and not available
outside of it. Nesting levels can be scoped or unscoped and simply indicate a sublevel
of control that will eventually return to the parent nesting level. Thus, scopes are
nested, but not all nesting levels are scoped.
In actuality, there is a type of scoping that is used for SUBROUTINEs: labels. Labels
used within subroutines are invisible outside of the subroutine. Because a subroutine
has a nesting level, the labels defined in it are seen only when in that nesting
level. So, labels are scoped by SUBROUTINE, but symbols are not.
The specified label is where the subroutine
starts, but that must immediately precede a SUBROUTINE command. Called subroutines
must start with SUBROUTINE and end with ENDSUBROUTINE. If SUBROUTINE is encountered
in normal execution of the script file, that line and all following lines, up through
the matching ENDSUBROUTINE are skipped. The only way to execute the lines within
that subroutine is via the CALL command. Note that SUBROUTINES can be nested.
Here is the user documentation:
CALL
Transfers control to a labeled subroutine in the current script file without creating
a new scope.
Format
CALL label {/OUTPUT=filespec} {parameter {...}}
Parameters
label
Specifies a label that must appear within the command file. The label may follow
or precede the CALL command. If there are duplicate labels, the last one encountered
is used. If it hasn't been encountered yet, the entire script is searched from the
beginning and the first instance found is used. If still not found, an error occurs.
parameter
Specifies one or more optional parameters. The symbols (P1, P2, etc.) are assigned
these values during the execution of the specified script. Each parameter is delimited
by one or more spaces. A null parameter can be provided by two consecutive quotes
(""). Each parameter can contain any characters desired, consistent with the following
rules.
- If the first parameter starts with a slash, the entire parameter must be surrounded by quotes.
- If the parameter includes embedded spaces, the parameter must be enclosed in quotes.
- All alphabetic characters are converted to uppercase. If you wish to preserve lowercase characters, the parameter must be enclosed in quotes.
- You can enclose quotes within the parameter by using double quotes for each quote.
For instance, the following parameters:
Hello there
will be converted to two parameters as if the following were executed:
P1="HELLO"
P2="THERE"
The following:
"Hello there"
will be converted to a single parameter as if:
P1="Hello there"
The following:
"Hello ""there"""
the value assigned to P1 in this case would be:
Hello "there"
Description
If the command procedure is not coming from a random-access device, CALL will generate
an error. Otherwise, control of the script is transferred to the first command immediately
following the SUBROUTINE at the specified label. The CALL command operates similarly
to the @ command, except that the overhead of opening a new command file and creating
a new symbol table is avoided. Execution continues until ENDSUBROUTINE, RETURN,
or EXIT is encountered, at which point control returns to the line following the
CALL command.
CALL creates a new procedure level, but not a new scope. Parameters are assigned
to symbols P1, P2, etc., which are local to the calling level. Local symbols defined
in the current procedure level are available to the subroutine called with CALL.
A subroutine has only one entry point - code cannot jump into the middle of a subroutine.
And each SUBROUTINE command must have a matching ENDSUBROUTINE command. If SUBROUTINE
is encountered during normal execution of the command file (i.e. not as the target
of a CALL), all lines up to, and including, the matching ENDSUBROUTINE are skipped.
Qualifier
/OUTPUT=filespec
Using this switch will direct all output to the specified file or device. If not
specified, the current output setting is used. Wildcards are not allowed in the filespec.
When the subroutine exits, the output is restored to what it was immediately prior
to the CALL.
Example:
$ A = 1
$Test1:
$ CALL Test2
$ IF A.LE.10 THEN GOTO Test1
$ EXIT
$Test2:
$ SUBROUTINE
$ WRITE SYS$OUTPUT "This is Test2"
$ CALL Test3
$ A = A + 1
$Test3:
$ SUBROUTINE
$ WRITE SYS$OUTPUT "This is Test3"
$ ENDSUBROUTINE
$ ENDSUBROUTINE
This sample script shows how to use CALL. A is set to 1 and the lines between
Test1 and EXIT will loop 10 times. Each time through the loop, control is transferred to
the line after Test2, which writes "This is Test2" and then uses CALL to transfer
control to the line after Test3, which write "This is Test3". When ENDSUBROUTINE is
executed, control is returned to the line after the CALL Test3, which adds 1 to A
and then the nested SUBROUTINE is skipped until the ENDSUBROUTINE matching the SUBROUTINE
at Test2, which returns to the line after CALL Test2. The output from this will be:
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
This is Test2
This is Test3
SUBROUTINE
Indicates the start of a local subroutine in a command file. SUBROUTINE must be the
first executable command after the label associated with the subroutine. See the
CALL command for more details.
Format
SUBROUTINE
Parameters
None.
Example:
$Test3:
$ SUBROUTINE
$ WRITE SYS$OUTPUT "This is Test3"
$ ENDSUBROUTINE
ENDSUBROUTINE
Indicates the end of a local subroutine in a command file. ENDSUBROUTINE must be the
last executable command of a subroutine. See the CALL command for more details.
Format
ENDSUBROUTINE
Parameters
None.
Example:
$Test3:
$ SUBROUTINE
$ WRITE SYS$OUTPUT "This is Test3"
$ ENDSUBROUTINE
if( Match( Sym, 'subroutine', 4 ) ) then
begin
Process_Subroutine ;
continue ;
end ;
if( Waiting_For_Subroutine ) then // Next command we should encounter is SUBROUTINE...
begin
Exception( UCL_INVCALL, '' ) ; // ...but it wasn't
exit ;
end ;
if( Sym = 'call' ) then
begin
Process_Call ;
end else
if( Match( Sym, 'endsubroutine', 4 ) ) then
begin
Process_EndSubroutine ;
exit ;
end else
This code is added to the Process routine in UCL_Main . It
handles encountering the SUBROUTINE, ENDSUBROUTINE, and CALL commands. There are two
things to note here. First, the call to Process_Call will set the
Waiting_For_Subroutine flag, which is checked before we look for any
other command other than SUBROUTINE. This is because the first executable command
encountered after a CALL must be SUBROUTINE.
The second thing to note is that the Match routine is used to check the
command values. DCL/UCL commands can be abbreviated to the minimum number of characters
that is unambiguous with other commands. In these cases, 4 characters is the minimum
acceptable length. Match is used to match the command in Sym with
the given UCL command with the specified minimum number of characters.
function Match( S, Command : string ; Min : integer ) : boolean ;
begin
if( length( S ) < Min ) then
begin
Result := False ;
end else
begin
Result := copy( Command, 1, length( S ) ) = S ;
end ;
end ;
This is the new function to do the abbreviated comparison. First we make sure that
the minimum length is met. If not, we indicate that this is not a match. Otherwise
we compare the passed string (S) with the passed command and return true if S is a
subset of the command. If so, we return true, otherwise we return false.
Local : boolean ;
Start : integer ;
This instance data is added to the TUCL_Context class. The way we are going to implement
the CALL mechanism is that we will use the same nesting approach that we used with
the @ command insofar as creating a new context for the CALL. However, as we will
see in a bit, the creation of a new symbol table and command file reassignment does
not happen. But this difference in the use of contexts means that we need to know
if a given context is due to @ or CALL. If the Local flag is set, we
know it is a CALL context, otherwise it is an @ context. Start is used
when searching for labels. Discussed below.
procedure Process_Call ;
var Output, Sym, X : string ;
E, Err, I, Position : int64 ;
Has_Quotes : boolean ;
Loop : integer ;
Previous : TUCL_Context ;
Resname, S : string ;
SL : TStringList ;
Node, Access, Secondary_Node, Device, Path, Name, Extension, Version : string ;
begin
if( Interactive ) then
begin
Exception( UCL_INVCALL, '' ) ;
exit ;
end ;
// Setup...
Position := This_UCL_Context.syscommand_line ; // Save our position
Sym := Get_Token ;
S := Parser.Grab_Line ;
Sym := Find_Target_Label( Sym ) ;
if( Sym = '' ) then
begin
exit ; // Aborted
end ;
This function operates in a similar way to the both the Process_GOSUB and
Process_At routines that we covered in past articles. CALL is not allowed
in interactive mode so we first check that case and exit with an error if so. Otherwise,
we save our current position in the command file, pull the destination label from
the parser, and make sure we can find said label. If not, we exit.
// Process output switch...
Output := '' ;
if( Parse_Switch( 'OUTPUT', '', S, Output ) = 1 ) then
begin
if( Output = '' ) then
begin
Exception( UCL_NULFIL, '' ) ;
exit ;
end ;
end ;
if( Switch_Present( S ) > 0 ) then
begin
Exception( UCL_IVQUAL, '' ) ;
exit ;
end ;
Next we process the optional /OUTPUT switch, as we did for the @ command, exiting
if there was a problem.
try
// Process parameters...
SL := TStringList.Create ;
while( S <> '' ) do
begin
X := Parse_Parameter( ' ', S ) ;
if( X <> '' ) then
begin
Has_Quotes := False ;
if( copy( X, 1, 1 ) = '"' ) then
begin
// Trim leading/trailing quotes...
Has_Quotes := True ;
X := copy( X, 2, length( X ) ) ;
if( copy( X, length( X ), 1 ) = '"' ) then
begin
setlength( X, length( X ) - 1 ) ;
end ;
end ; // if( copy( X, 1, 1 ) = '"' )
// Handle double-quotes
I := 1 ;
while( I < length( X ) ) do
begin
if( copy( X, I, 2 ) = '""' ) then
begin
delete( X, I + 1, 1 ) ;
end ;
inc( I ) ;
end ;
if( not Has_Quotes ) then
begin
X := Edit( X, 32 or 256 ) ; // Uppercase except within quotes
end ;
SL.Add( X ) ;
end ; // if( X <> '' )
end ;
for Loop := 0 to SL.Count - 1 do
begin
if( LIB_Set_Symbol( 'P' + inttostr( Loop + 1 ), SL[ Loop ], LNM_PROCESS ) <> 0 ) then
begin
Set_Exception( LIB_Get_Exception_Text( 0, E ) ) ;
exit ;
end ;
end ;
We process the parameters in almost the exact same way as we did for the @ command.
The difference being that we don't create a new symbol table, since we remain in the
same scope. As noted with the @ command, there is no practical limit to the number
of parameters that can be passed, which differs from DCL's limit of 8.
// Create new context...
Previous := This_UCL_Context ;
_This_UCL_Context := TUCL_Context.Create ;
_This_UCL_Context.Local := True ;
_This_UCL_Context.sysoutput_name := Previous.sysoutput_name ;
_This_UCL_Context.sysinput_name := Previous.sysinput_name ;
_This_UCL_Context.syscommand_name := Previous.syscommand_name ;
_This_UCL_Context.syserror_name := Previous.syserror_name ;
_This_UCL_Context.syscommand_line := Previous.syscommand_line ;
Contexts.Add( This_UCL_Context ) ;
Now, we create the new nesting level. Although we don't care about the various
redirections, other than sys$output, we still have to appropriately set the various
context values since it is possible that the subroutine may use the @ command itself,
which will care about these values.
// Handle output
if( Output <> '' ) then
begin
_This_UCL_Context.sysoutput_name := Output ;
Err := LIB_Set_Symbol( 'sys$output', Output, LNM_PROCESS ) ;
if( Err <> 0 ) then
begin
Process_Return ;
E := LIB_Get_Exception( 0 ) ; // Get handle
Set_Exception( LIB_Get_Exception_Text( 0, E ) ) ;
exit ;
end ;
This_UCL_Context.sysoutput_name := Output ;
end ;
Next we handle the /OUTPUT switch, in the same manner as the @ command.
// Push return address onto previous scope and goto subroutine...
I := Previous.Labels.Indexof( Sym ) ;
Previous.Stack.Add( Position + 1 ) ;
This_UCL_Context.Start := Position + 1 ;
Goto_Line( integer( Previous.Labels.Objects[ I ] ) ) ;
Waiting_For_Subroutine := True ;
finally
SL.Free ;
end ;
end ; // Process_Call
To finish up this routine, we add the saved position to the previous nesting's stack
and position to the destination label. In this way, we are more like the GOSUB command.
We also set the Start value in the new context, which tells the Find_Target_Label
to start label searches at the beginning of the subroutine. This ensures that any
labels within the subroutine take precedence over any duplicates elsewhere in the
command file. Finally we set the Waiting_For_Subroutine flag and clean up.
procedure Process_Endsubroutine ;
var Position : int64 ;
begin
if( Interactive or not This_UCL_Context.Local ) then
begin
Exception( UCL_INVCALL, '' ) ;
exit ;
end ;
Process_Return ;
end ;
This routine handles encountering ENDSUBROUTINE. In the case of being interactive
or not being in local nesting context, this indicates that we encountered this command
without a CALL to the subroutine - most likely indicating a stray ENDSUBROUTINE command
that has no matching SUBROUTINE.
procedure Process_Subroutine ;
begin
if( Interactive ) then
begin
Exception( UCL_INVCALL, '' ) ;
exit ;
end ;
if( Waiting_For_Subroutine ) then
begin
Waiting_For_Subroutine := False ;
exit ;
end ;
Skip_Subroutine_Block ;
end ;
This routine handles encountering SUBROUTINE during normal processing through the
file or in response to a CALL. It is not allowed in interactive mode and we exit
with an error in that case. If we are waiting for SUBROUTINE (according to the flag),
we clear the flag and exit. In that case, we will return to the main Process loop,
to execute the next command - which is the first command within the subroutine.
Otherwise, we came across SUBROUTINE while reading through the command file. In
this case, we skip past the subroutine block.
procedure Skip_Subroutine_Block ;
var Nest_Level : integer ;
S : string ;
begin
Nest_Level := 0 ;
while( true ) do
begin
S := Edit( Get_Command, 8 or 16 or 128 or 256 or 512 ) ;
if( Match( S, 'endsubroutine', 4 ) ) then
begin
dec( Nest_Level ) ;
if( Nest_Level < 0 ) then
begin
exit ;
end ;
continue ;
end else
if( Match( S, 'subroutine', 4 ) ) then
begin
inc( Nest_Level ) ; // In a new SUBROUTINE block
continue ;
end else
if( copy( S, 1, 3 ) = 'if ' ) then
begin
Parser.Put_Token( copy( S, 4, length( S ) ) ) ;
if( Skip_If_Block ) then
begin
exit ;
end ;
end ;
end ; // while( true )
end ; // Skip_Subroutine_Block
This routine is similar to the Skip_If_Block that we previously covered in
article 129. In fact, we use that routine to properly skip
over IF commands that occur within a SUBROUTINE. We exit upon error or upon finding
the matching ENDSUBROUTINE. Because SUBROUTINEs can be nested, we keep track of our
nesting level.
if( Match( S, 'subroutine', 4 ) ) then
begin
Skip_Subroutine_Block ;
end else
if( Match( S, 'endsubroutine', 4 ) ) then
begin
Exception( UCL_INVCALL, '' ) ;
Result := True ;
exit ;
end ;
Speaking of the Skip_If_Block routine, we add the above code to the end
of that routine to handle encountering the SUBROUTINE or ENDSUBROUTINE commands while
looking for the end of an IF block.
if( Contexts.Count > 1 ) then // Nested
begin
//NEW---->
if( This_UCL_Context.Local ) then // Returning from a local call //NEW
begin
Previous := TUCL_Context( Contexts[ Contexts.Count - 2 ] ) ;
if( Previous.sysoutput_name <> This_UCL_Context.sysoutput_name ) then
begin
Err := LIB_Set_Symbol( 'sys$output', Previous.sysoutput_name, LNM_PROCESS ) ;
if( Err <> 0 ) then
begin
E := LIB_Get_Exception( 0 ) ; // Get handle
Set_Exception( LIB_Get_Exception_Text( 0, E ) ) ;
end ;
end ;
// Delete the context...
Contexts.Delete( Contexts.Count - 1 ) ;
_This_UCL_Context.Free ;
_This_UCL_Context := Previous ;
if( This_UCL_Context.Stack.Count = 0 ) then // Nothing to return to - should never happen
begin
Exception( UCL_BADRETURN, '' ) ;
exit ;
end ;
// Pop the return location from the stack and go there...
Position := This_UCL_Context.Stack[ This_UCL_Context.Stack.Count - 1 ] ;
This_UCL_Context.Stack.Count := This_UCL_Context.Stack.Count - 1 ;
Goto_Line( Position ) ;
Parser.Grab_Line ; // Clear any tokens remaining in parser
end else
begin
//<----NEW
// Delete the current scope table and make previous table the default...
if( Contexts.Count = 2 ) then
The code between the comments is added to the Process_Exit routine to
handle the new context of a SUBROUTINE for the EXIT command. If we are in a local subroutine
nesting level, we delete the current nesting context, and point back to the previous
context, changing sys$output if it was altered by the call, and then we pop the return
location from the stack and position to that place. We finish by clearing anything
else that may remain in the parser's buffer so that we begin processing the command
at the destination location.
if( This_UCL_Context.Stack.Count = 0 ) then // Nothing to return to
begin
//NEW---->
if( This_UCL_Context.Local ) then // Returning from a local call
begin
if( Contexts.Count > 1 ) then // Nested (should always be true)
begin
Previous := TUCL_Context( Contexts[ Contexts.Count - 2 ] ) ;
if( Previous.sysoutput_name <> This_UCL_Context.sysoutput_name ) then
begin
Err := LIB_Set_Symbol( 'sys$output', Previous.sysoutput_name, LNM_PROCESS ) ;
if( Err <> 0 ) then
begin
E := LIB_Get_Exception( 0 ) ; // Get handle
Set_Exception( LIB_Get_Exception_Text( 0, E ) ) ;
end ;
end ;
// Delete the context...
Contexts.Delete( Contexts.Count - 1 ) ;
_This_UCL_Context.Free ;
_This_UCL_Context := Previous ;
if( This_UCL_Context.Stack.Count = 0 ) then // Nothing to return to - should never happen
begin
Exception( UCL_BADRETURN, '' ) ;
exit ;
end ;
// Pop the return location from the stack and go there...
Position := This_UCL_Context.Stack[ This_UCL_Context.Stack.Count - 1 ] ;
This_UCL_Context.Stack.Count := This_UCL_Context.Stack.Count - 1 ;
Goto_Line( Position ) ;
Parser.Grab_Line ; // Clear any tokens remaining in parser
exit ;
end ; // if( Contexts.Count > 1 )
end ;
//<-----NEW
Parser.Grab_Line ;
Exception( UCL_BADRETURN, '' ) ;
exit ;
end ;
Likewise, we have to modify RETURN processing to deal with the SUBROUTINE situation.
The lines between the comments above are added to the Process_Return routine.
They do the same thing as the code in Process_Exit above.
if( This_UCL_Context.Start <> 0 ) then
begin
Goto_Line( This_UCL_Context.Start ) ;
end ;
This code is added near the beginning of the Find_Target_Label routine
so that any label searches begin in the current subroutine (Start will be 0 if we
aren't in a SUBROUTINE).
if( Match( Sym, 'subroutine', 4 ) ) then
begin
Skip_Subroutine_Block ;
end ;
Finally, we add this code to the Find_Target_Label routine so that any
searches for a label will skip over SUBROUTINES, since labels within SUBROUTINES are
invisible to any code outside of that subroutine.
In the next article, we will look at the next UCL commands.
Copyright © 2021 by Alan Conroy. This article may be copied
in whole or in part as long as this copyright is included.
|