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
|
COPY, part 3
In this article we will continue to look at the code for the COPY CUSP.
// Do the operation...
OutFile := nil ;
if( Operation = Op_Contiguous ) then
begin
F := Open( S, 0 ) ;
if( F = nil ) then
begin
Result := Last_Error ;
exit ;
end ;
F.Contiguous := True ;
F.Free ;
exit ;
end ;
if( Operation = Op_Noncontiguous ) then
begin
F := Open( S, 0 ) ;
if( F = nil ) then
begin
Result := Last_Error ;
exit ;
end ;
F.Contiguous := False ;
F.Free ;
exit ;
end ;
The first two operations have to do with altering the contiguous state of a file.
On VMS, one would have to copy the file to itself, creating a (non)contiguous new file.
If not open elsewhere, the old file would then be deleted when COPY closed it, leaving
the new copy with the desired state. This can be problematic with large files as
there may not be enough space for two copies of the file, even if only for a short
time. The UOS file system provides for a means of simply changing the file layout
in-place. We discussed this in the past, and going from contiguous to non-contiguous
might increase the store requirements slightly, but not as much as creating another
copy of the file. So COPY will make use of this feature and simply change the contiguous
state of the file. Note that a non-UOS native file system might have to accomplish
this operation in a way similar to VMS, but such operations must be encapsulated within
the file system driver.
OS := new( POS_UOS, Init ) ;
for Index := 0 to Inputs.Count - 1 do // Loop through input filespecs
begin
_Out := Inputs[ Index ] ;
Search_Context := 0 ;
The copy and concatenate operations take up the rest of the COPY routine. We will
iterate through the input file list, processing them in the order they were specified.
S := Lookup( _Out, Search_Context ) ;
while( S <> '' ) do // While matching input files...
begin
// Skip exclusions...
Skip_File := False ;
for Index1 := 0 to Exclusions.Count - 1 do
begin
if( Filename_Match( S, Exclusions[ Index1 ] ) ) then
begin
Skip_File := True ;
break ;
end ;
end ;
if( Skip_File ) then
begin
S := Lookup( '', Search_Context ) ;
continue ;
end ;
For each input file specification, we do a wildcard lookup loop to find all matching
files. For each found file, we first check to see if the filename matches any of
the excluded file specifications. If a match was found, we lookup the next file and
loop back to check the exclusions.
// Skip files not matching date criteria...
F := Open( S, FAB_V_GET or FAB_V_SHRALL ) ;
if( F = nil ) then
begin
S := Lookup( '', Search_Context ) ;
continue ;
end ;
if( File_Date_Mode <> Date_None ) then
begin
case File_Date_Match of
Date_Backup : File_Time := F.Last_Backup ;
Date_Created : File_Time := F.Creation ;
Date_Expired : File_Time := F.Expiration ;
Date_Modified : File_Time := F.Last_Modified ;
end ;
end ;
if( File_Date_Mode = Date_Since ) then
begin
if( File_Time < Comparison_Time ) then
begin
S := Lookup( '', Search_Context ) ; // File was before the after date
F.Free ;
continue ;
end ;
end else
if( File_Date_Mode = Date_Before ) then
begin
if( File_Time >= Comparison_Time ) then
begin
S := Lookup( '', Search_Context ) ; // File was after the before date
F.Free ;
continue ;
end ;
end ;
if( Owner <> 0 ) then
begin
if( Owner <> F.Owner ) then
begin
S := Lookup( '', Search_Context ) ; // File is the wrong owner
F.Free ;
continue ;
end ;
end ;
F.Free ;
Next we check to see if the file matches the specified date comparison (if any).
But this requires that we first open the file. So we open it in read-only mode with
all sharing allowed. This is so that COPY will work even if the file is opened
by another process (and so it won't cause file access errors for other processes).
If the open fails, it probably indicates that the file was deleted between the lookup
and the open - always a potential in a pre-emptive multi-tasking system.
Now we can check the appropriate file date with the specified comparison date. If
the file is excluded by this check we skip to the next file. Then we check if the
file's owner matches the specified owner (if one was specified), likewise exiting
if there is a mismatch. Finally, now that we've skipped files that don't match the
selection criteria, we close the file.
// Confirm operation...
if( Confirm ) then // Need to confirm operation with user
begin
while( True ) do
begin
if( Operation = Op_Copy ) then
begin
C := 'Copy' ;
end else
begin
C := 'Append' ;
end ;
C := trim( lowercase( Get_Command_With_Timeout( C + ' ' + S + '? ' ) ) ) ;
if( Is_Prefix( 'yes', C )
or
Is_Prefix( 'true', C )
or
Is_Prefix( 'no', C )
or
Is_Prefix( 'false', C )
or
Is_Prefix( 'quit', C )
or
Is_Prefix( 'all', C )
or
( C = '1' ) or ( C = '0' ) or ( C = Control_Z )
) then
begin
break ; // Valid response
end ;
OS^.OutputLn( RH_SysError, 'COPY-E-COPY_IVRESP, Invalid response' ) ;
end ;
if( Is_Prefix( 'all', C ) ) then
begin
Confirm := False ; // No more confirmation required
end else
if( Is_Prefix( 'no', C )
or Is_Prefix( 'false', C )
or ( C = '' )
or ( C = '0' )
) then
begin
S := Lookup( '', Search_Context ) ;
continue ;
end else
if( Is_Prefix( 'quit', C ) or ( C = Control_Z ) ) then
begin
Inputs.Clear ;
S := '' ;
break ;
end ;
end ; // if( Confirm )
Now that we have a valid candidate input file, we need to confirm the operation with
the user. If confirmation was not specified, we skip this code. Otherwise, we prompt
the user and get his response. We loop until a valid response is provided, showing
an error if the user's response doesn't match one of the valid options. Once we
have a valid response, we act on it. If ALL is specified, we clear the Confirm
flag so that we no longer ask the user for confirmation. If the user QUITs, we break
out of the current file lookup loop after clearing the Inputs list so
that the outer loop ends immediately. If the user doesn't confirm this file, we
look up the next file and loop. The only other option is that the user confirms the
file and we just fall through to the following code (since all invalid responses are
checked for above).
// Perform operation...
O := Substitute_Wildcards( S, Output, DefDev + DefDir ) ;
if( Log ) then
begin
O1 := O ;
S1 := S ;
if( Style = Style_Condensed ) then
begin
if( length( S1 ) > 255 ) then
begin
C := inttostr( Increment ) ;
setlength( S1, 255 - length( C ) ) ;
S1 := S1 + C ;
inc( Increment ) ;
end ;
if( length( O1 ) > 255 ) then
begin
C := inttostr( Increment ) ;
setlength( O1, 255 - length( C ) ) ;
O1 := O1 + C ;
inc( Increment ) ;
end ;
end ;
if( Operation = Op_Copy ) then
begin
C := 'Copying' ;
end else
begin
C := 'Appending' ;
end ;
OS^.OutputLn( RH_SysOutput, C + ' ' + S1 + ' to ' + O1 ) ;
end ;
//TODO:Handle RMS
//TODO:Handle serial devices
//TODO:Handle sequential devices
Having made it through the gauntlet of selection and confirmation, we are now ready
to perform the operation. First, we construct the output file name based on the input
file, defaulting the device and path if necessary. Next we log the operation to the user,
if logging was requested. If condensed style was specified, we limit the filename
length(s) to 255 characters. For names longer than that, we trim off some portion
of the end of the filename and replace it with a numeric value that we increment each
time we have to condense the filename.
Finally, I've included three comments to remind us of things we have yet to address.
All of these situations we will cover in future articles.
// Open input file...
IFlags := 0 ;
OFlags := 0 ;
if( Read_Check ) then
begin
IFlags := FAB_V_RCK ;
end ;
if( SymLink ) then
begin
IFlags := IFlags or FO_Symlink ;
end ;
if( Write_Check ) then
begin
OFlags := FAB_V_WCK ;
end ;
F := Open( S, FAB_V_GET or FAB_V_SHRALL or IFlags ) ;
if( F = nil ) then
begin
S := Lookup( '', Search_Context ) ;
continue ;
end ;
Count := F.Get_Size ;
if( Truncate ) then
begin
Count := F.Get_EOF ;
end ;
Now we determine the open flags for the input and output files and then open the
input file. If the open fails, we move on to the next file. Of special note here
is the FO_SymLink flag which indicates how a symbolic link is to be opened. A link
is simply an indirect reference to a file located elsewhere. When opened, normally,
the referenced file is opened transparently to the code that is opening the file.
This is done by reading the name of the file from the link file. In other words, a
link file's data is the name of the file to actually open. So, when COPY opens a
link, it is unknowingly opening the referenced file. But if we use the FO_Symlink
flag, the link file itself is what is opened. Thus, we can copy the link itself.
Finally, we get the amount of data to copy from the input file. If /TRUNCATE is
specified, we limit the amount to that indicated by the logical end of file, rather
than the physical end of file.
// Prepare output file...
if( Operation = Op_Concatenate ) then
begin
if( OutFile = nil ) then // First one
begin
if( not Open_Output( O ) ) then
begin
OS^.OutputLn( RH_SysError, LIB_Get_Exception_Text( 0,
LIB_Get_Exception_Code( 0, 0 ) ) ) ;
continue ;
end ;
end ;
end else
begin
if( not Open_Output( O ) ) then
begin
OS^.OutputLn( RH_SysError, LIB_Get_Exception_Text( 0,
LIB_Get_Exception_Code( 0, 0 ) ) ) ;
continue ;
end ;
end ;
Now we prepare the output file. In the case of concatenations, we open it once on
the first file (OutFile will be nil in this case). Otherwise,
we open it for each copy operation. Any output open errors are displayed.
// Copy data...
while( Count > 0 ) do
begin
if( not Read_File( Count ) ) then
begin
Result := F.IO_Error.Get_Error ;
exit ;
end ;
if( not Write_File( Count ) ) then
begin
Result := F.IO_Error.Get_Error ;
exit ;
end ;
end ;
Now we loop through the data, reading the input and writing the output. If either
the read or write operations fail, we exit.
// Copy ancillary streams...
I := 1 ;
S := F.Stream_Name( I ) ;
while( S <> '' ) do
begin
if( OutFile.Stream_Index( PChar( S ) ) < 1 ) then
begin
// Stream doesn't exist in output file already
L := Outfile.Create_Stream( PChar( S ) ) ;
Copy_Stream( I, L ) ;
end ;
inc( I ) ;
S := F.Stream_Name( I ) ;
end ;
Now that the data is copied, we need to copy any ancillary data streams. We start
with stream 1 and keep going until we run out of streams to copy. First we get the
name of the stream, then we see if a stream with that name already exists on the output
file. If not, we copy the stream. In this way, the first occurance of a stream name
in the input files is the one written to the output, regardless of how many input
files have a stream with that name.
// Close files...
if( Operation = Op_Copy ) then
begin
if( Contiguous ) then
begin
OutFile.Contiguous := True ;
end ;
OutFile.Free ;
end ;
F.Free ;
// Prepare for next lookup
S := Lookup( '', Search_Context ) ;
end ; // while( S <> '' )
When the data copy is finished, if this was a copy operation then we close the output
file. But first we make sure it is marked contiguous if that was specified. Finally,
we close the input file, look up the next one, and loop back to process it.
SYS_Lookup_Close( Search_Context ) ;
end ; // for I := 0 to Inputs.Count - 1
if( Operation = Op_Concatenate ) then
begin
if( Contiguous ) then
begin
OutFile.Contiguous := True ;
end ;
OutFile.Free ;
end ;
OS.Free ;
finally
Inputs.Free ;
Exclusions.Free ;
end ;
end ; // if( UUI.Execute )
end ; // Run
When we run out of files during the lookup, we close the lookup and loop back to the
next input file specification, repeating the whole process again. Once we are done
with all the input specifications, we can finally close the output file, which is still
open for concatenation operations. Then we free the OS instance and various lists.
In the next article, we will look at the supporting routines that we referenced in
this, and the previous, article's code.
|