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.