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

Handles

Before we proceed further in our discussions of I/O operations, we need to discuss the concept of handles. Whenever a process wants to reference a UOS resource (whether a file, a device, or something else), there needs to be a way for it to uniquely identify that resource. This means of that unique identification is what we call a "handle". What do we use for the handle? We could use a string name for the resource, such as the name of the file. But passing strings across the processor rings has a lot of overhead associated with it. Besides that, if an application was to unknowingly use the same file twice (perhaps the user specified the same file for input and output) then it would need two handles. But if the filename was the handle, then the application would use the same handle twice and, thus, in the same context for both cases, which is almost certainly not what is desired. It would be better to use a unique integer value for handles.

We could use a simple integer value that is incremented for each new handle that is needed, but then we would need to translate that into whatever UOS class instance is associated with that handle value. This would require storing both the handle and the associated instance pointer, which requies a larger memory footprint (especially of concern on a system with thousands of active handles). In addition, that translation on every handle access would also cause additional processing overhead. So, instead we will use the address of the class instance as the handle value. Because a valid class instance will never be less than 256, we can use 0 to indicate an invalid handle, and other low values for special cases (such as the redirection handles like RH_SysOutput, discussed below). The disadvantage of this approach is that handle instance addresses (and thus handle values) tend to get reused over time. If a program closes a handle, but "hangs onto" the handle value and uses it again after the handle has been reused for another process, the error result would indicate a handle access violation. But if the handle wasn't reused, the error result from using it would indicate an invalid handle. The first condition could be somewhat confusing to someone trying to debug the application. The "inconsistency" between the two conditions could be avoided if we never reused a handle value in the executive - and some operating systems do this. For the aforementioned performance reasons, UOS does not take that approach. However, programs could use an interface that provides unique handle values and translates them to executive handle values. We will leave such an interface as an exercise for the reader.

First, let's add some new constants.

// Reserved Handles...
const RH_SysInput = 0 ; // SYS$INPUT
const RH_SysOutput = 1 ; // SYS$OUTPUT
const RH_SysCommand = 2 ; // SYS$COMMAND
const RH_SysError = 3 ; // SYS$ERROR
These redirection constants are used to indicate the process' current sys$input, sys$output, sys$command, and sys$error files. We will see how these are used a little later in the article.

When a new process is created, the standard redirection devices are assigned to the process' terminal. After we create the process in the kernel startup (as discussed in article 38), we call the FiP.Attach_Device method.

function TUOS_FiP.Attach_Device( Device : PChar ; PID : TPID ) : TUnified_Exception ;

var D : TDevice ;
    S : TString ;

begin
    // Setup and sanity check...
    Result := Set_Last_Error( nil ) ;
    D := Get_Device( string( Device ) ) ;
    if( D = nil ) then // Device not found
    begin
        Set_Last_Error( Create_Error( UOSErr_Device_Not_Found ) ) ;
        Result := Last_Error ;
        exit ;
    end ;
First we clear any pending exceptions. Next we obtain the device passed to us. If the device doesn't exist, we return an exception.

    // Assign device ownership to Process...
    Result := Assign_Device( Device, PID ) ;
    if( Result <> nil ) then
    begin
        exit ;
    end ;

    D.Attached := PID ; // Attach to PID...
Next we assign the device to the process. If there was an exception, we exit. Then we mark the device as being assigned to the process.

    // Create standard I/O logicals...
    S := TString.Create ;
    try
        S.P := Device ;
        S.Len := length( string( Device ) ) ;
        SSC.Set_Symbol( LNM_JOB, PID, 'sys$input', S ) ;
        SSC.Set_Symbol( LNM_JOB, PID, 'sys$output', S ) ;
        SSC.Set_Symbol( LNM_JOB, PID, 'sys$command', S ) ;
        SSC.Set_Symbol( LNM_JOB, PID, 'sys$error', S ) ;
        Assign( PID, 'sys$input:', RH_SysInput, 3, '', 0 ) ;
        Assign( PID, 'sys$output:', RH_SysOutput, 3, '', 0 ) ;
        Assign( PID, 'sys$error:', RH_SysError, 3, '', 0 ) ;
        Assign( PID, 'sys$command:', RH_SysError, 3, '', 0 ) ;
    finally
        S.Free ;
    end ;
end ; // TUOS_FiP.Attach_Device
Finally we define the four standard I/O logicals for the process (hence the use of the LNM_JOB table). Then we assign the passed device to each of the relocation handles.

The Assign method is used to assign a device to a process handle.

function TUOS_FiP.Assign( PID : TPID ; Name : string ; Channel : word ;
    Access : word ; Mailbox : string ; Flags : word ) : TUnified_Exception ;

var Resource : TResource ;

begin
    Result := nil ;
    case Channel of
        RH_SysInput, RH_SysOutput, RH_SysCommand, RH_SysError: ;
        else
            begin
                Set_Last_Error( Create_Error( UOSErr_Invalid_Channel ) ) ;
                exit ;
            end ;
    end ; // case Channel
The first thing we do is validate the handle. At this point, the only valid handles are the relocation handles. If any other value is passed, we return an error.

    Resource := Create_File_Handle( PID, PChar( Name ), Flags ) ;
    if( Resource = nil ) then
    begin
        Result := Last_Error ;
        exit ;
    end ;
Next we create a file handle and assign it to Resource. The TResource class is used as an indirection to the actual class instance for the device or file. Why this indirection? We will discuss the advantages of it in a future article. We'll look at the TResource class below.

    case Channel of
        RH_SysInput: // SYS$INPUT
            begin
                if( Resource._File.Write_Only ) then
                begin
                    Result := Set_Last_Error( Create_Error( UOSErr_Read_Only ) ) ;
                    exit ;
                end ;
                USC.Assign_Channel( PID, Channel, int64( Resource ) ) ;
            end ;
        RH_SysOutput: // SYS$OUTPUT
            begin
                if( Resource._File.Read_Only ) then
                begin
                    Result := Set_Last_Error( Create_Error( UOSErr_Read_Only ) ) ;
                    exit ;
                end ;
                USC.Assign_Channel( PID, Channel, int64( Resource ) ) ;
            end ;
        RH_SysCommand: // SYS$COMMAND
            begin
                if( Resource._File.Write_Only ) then
                begin
                    Result := Set_Last_Error( Create_Error( UOSErr_Read_Only ) ) ;
                    exit ;
                end ;
                USC.Assign_Channel( PID, Channel, int64( Resource ) ) ;
            end ;
        RH_SysError: // SYS$ERROR
            begin
                if( Resource._File.Read_Only ) then
                begin
                    Result := Set_Last_Error( Create_Error( UOSErr_Read_Only ) ) ;
                    exit ;
                end ;
                USC.Assign_Channel( PID, Channel, int64( Resource ) ) ;
            end ;
    end ; // case Channel
end ; // TUOS_FiP.Assign
Next we check the specific relocation handle and make sure we have the necessary access mode to the device (not read-only for sys$output and sys$error, and not write-only for sys$input and sys$command). Finally we assign the channel via the Assign_Channel method of the USC (discussed below).

Now let's look at the Create_File_Handle method.

function TUOS_FiP.Create_File_Handle( PID : TPID ; Name : PChar ;
    Flags : longint ) : TResource ;

var Device : TDevice ;
    Dummy : integer ;
    F : TUOS_File ;
    S : string ;

begin
    Result := nil ; // Assume failure
    Set_Last_Error( nil ) ;
    S := Resolve_Path( PID, string( Name ), True ) ;
    Dummy := pos( ':', S ) ;
    if( Dummy = 1 ) then // Start with colon
    begin
        Set_Last_Error( Create_Error( UOSErr_Invalid_Filename ) ) ;
        exit ;
    end ;
First, we resolve the passed path/name. If the name starts with a colon, the name is invalid and we exit with an error.

    if( Dummy = 0 ) then // No colon - a file on a file system
    begin 
        .
        .
        .
    end else
    begin
        Device := Get_Device( copy( S, 1, Dummy ) ) ;
        if( Device = nil ) then
        begin
            Set_Last_Error( Create_Error( UOSErr_Device_Not_Found ) ) ;
            exit ;
        end ;
If there is no colon at all, the passed name is a file name. We will handle that case in the future. Otherwise, it is a device and we get the device instance for that name. If it is invalid we exit with an error.

        S := copy( S, Dummy + 1, length( S ) ) ; // Get path/file after device
        if( Device.Device_Class = DFC_Stream ) then // Stream device
        begin
            if( Device._File <> nil ) then
            begin
                F := Device._File ;
            end else
            if( Device.Terminal <> nil ) then
            begin
                F := TFip_Terminal_File.Create( Kernel ) ;
                TFip_Terminal_File( F ).Terminal := Device.Terminal ;
                Device._File := F ;
            end else
            begin
                F := TFiP_Device_File.Create( Kernel ) ;
                TFiP_Device_File( F ).Device := Device ;
                Device._File := F ;
            end ;
        end else
Every device can have a file instance associated with it, although we don't make the association until there is a need. But if a file instance is ever associated, it remains asscciated from then on. So, the first thing we do is see if a file has already been associated. If so, we will use that file. Otherwise, we check to see if the device is a terminal. In that case, we create a TFiP_Terminal_File instance and associate it with the device. If no file is already associated and is not a terminal, we create a TFiP_Device_File instance and associate that with the device. The TFip_Device_File is used for generic access to devices, which we will discuss in a future article.

        if( not Device.Mounted ) then
        begin
            Set_Last_Error( Create_Error( UOSErr_Device_Not_Mounted ) ) ;
            exit ;
        end else
        if( Device.FS = nil ) then // No file structure on device
        begin
            if( ( Flags and FO_Block_Mode ) = 0 ) then // Opening in non-file-structured mode
            begin
                Set_Last_Error( Create_Error( UOSErr_Device_Not_File_Structured ) ) ;
                exit ;
            end ;
            if( Device._File <> nil ) then
            begin
                F := Device._File ;
            end else
            begin
                F := TFiP_Store_File.Create( Kernel ) ;
                TFiP_Store_File( F ).Store := Device.Store ; // Store
                Device._File := F ;
            end ;
        end else
If the device class isn't a stream, and the device is not mounted, we generate an exception. Otherwise, we see if there is a file system associated with the device. If not, then we assume the user wants to open the device in non-file-structured (or "block") mode. But if the user hasn't verified this by using the FO_Block_Mode flag, we generate an error and exit. Otherwise, we use the associated file, if any, or else we create a TFip_Store_File and associate that with the store.

        begin
            F := Open_File( PID, PChar( S ), Flags ) ; // Normal file...
            if( F = nil ) then // Error occurred
            begin
                exit ;
            end ;
        end ;
    end ;
On the other hand, if there is a file system on the device, we simply request an open on the file.

    Result := TResource.Create ;
    Result._File := TFile( F ) ;
    TFiP_File( F ).Add_Handle( Result ) ;
    USC.Add_Handle( PID, int64( Result ) ) ;
end ; // TUOS_FiP.Create_File_Handle
Once we have the proper file (in variable F), we create a resource instance and associate the file with the resource. Then we call USC.Add_Handle to add the handle to the process.
We will address the TFiP_Device_File and TFiP_Store_File classes in a later article.

Now let's look at the Assign_Channel method of the USC. First we need to add some instance data to the TProcess class:

public // Handles...
    _Sys_Input : THandle ; // Handle for sys$input
    _Sys_Output : THandle ; // Handle for sys$output
    _Sys_Command : THandle ; // Handle for sts$command
    _Sys_Error : THandle ; // Handle for sys$error
    _Handles : TInteger_List ; // Handles owned by process

Now the Assign_Channel method itself:
procedure TProcess.Assign_Channel( RC : word ; Handle : THandle ) ;

    procedure Update_Channel( var H : THandle ) ;

    begin
        if( H <> Handle ) then // Handle is being changed
        begin
            if( H <> 0 ) then // Have a handle
            begin
                FiP.Close_Handle( H ) ;
                _Handles.Remove( H ) ;
            end ;
            H := Handle ;
            if( Handle <> 0 ) then
            begin
                _Handles.Add( H ) ;
            end ;
        end ;
    end ;

begin
    case RC of
        RH_SysInput : Update_Channel( _Sys_Input ) ;
        RH_SysOutput : Update_Channel( _Sys_Output ) ;
        RH_SysCommand : Update_Channel( _Sys_Command ) ;
        RH_SysError : Update_Channel( _Sys_Error ) ;
    end ;
end ; // TProcess.Assign_Channel
Note that "channel", an older term, is synonomous with "handle". As you can see from the new instance data, we have a list of handles, and the four standard handles. The four standard handles are stored in the _Handles list, but we also have the handles in separate variables for quick access. We switch based on the relocation handle passed to us, calling the local Update_Channel procedure with the corresponding standard handle. We'll discuss Close_Handle below.
In Update_Channel, we first check that we are actually changing the channel value. If so and the passed handle is not 0, we close the handle and remove it from the handle list. In either case, we add the new handle to _Handles and set the standard handle value.
A THandle is a 64-bit integer value which contains the address of a TResource instance. You can think of it as simply a pointer to an instance.

procedure TUSC.Add_Handle( PID : TPID ; Handle : THandle ) ;

var Process : TProcess ;

begin
    if( PID = 0 ) then
    begin
        Process := Get_Process( Kernel.PID ) ;
    end else
    begin
        Process := Get_Process( PID ) ;
    end ;
    if( Process = nil ) then
    begin
        Set_Error( UOS_User_Security_Error_Invalid_PID ) ;
        exit ;
    end ;
    Process._Handles.Add( Handle ) ;
end ;
This method simply adds a handle to a process' _Handles list. The passed process ID is used, unless it is 0 - in which case, the current process is used. If an invalid process ID is passed, we generate an error.

Now let's look at TResource:

TResource = class( TCommon_COM_Interface )
                public // API...
                    PID : TPID ;
                    Position : int64 ; // Current position
                    _File : TFile ;
            end ;
The class has a process ID, a context (position), and the file object associated with the handle. As mentioned earlier, a TResource instance is an indirection to the actual file instance. This is to handle multiple processes accessing some files (a topic for a future article). Each TResource is owned by a single process*, but each file can be referenced by multiple TResources.

* it is possible for a process to share a handle under some circumstances.

Now let's look at the FiP's Close_Handle method.

function TUOS_FiP.Close_Handle( Handle : THandle ) : TUnified_Exception ;

var Resource : TResource ;

begin
    if( not USC.Remove_Handle( 0, Handle ) ) then
    begin
        Result := Set_Last_Error( Create_Error( UOSErr_Invalid_Handle ) ) ;
        exit ;
    end ;
    Result := Set_Last_Error( nil ) ;
    Resource := TResource( Handle ) ;
    TFiP_File( Resource._File ).Remove_Handle( Resource ) ;
    Resource.Free ;
end ;
The process of closing a handle gets rid of the TResource, releasing any memory for, and references to, the device. After that happens, the handle is no longer valid. The first thing we do is remove the handle from the process via the Remove_Handle method of the USC. If that fails, we return with an execption. Otherwise, we clear any pending exceptions, remove the handle's association with the file, and then we free the TResource instance.

Here is the Remove_Handle method.

function TUSC.Remove_Handle( PID : TPID ; Handle : THandle ) : boolean ;

var Index : integer ;
    Process : TProcess ;

begin
    Result := False ;
    if( PID = 0 ) then
    begin
        Process := Get_Process( Kernel.PID ) ;
    end else
    begin
        Process := Get_Process( PID ) ;
    end ;
    if( Process = nil ) then
    begin
        Set_Error( UOS_User_Security_Error_Invalid_PID ) ;
        exit ;
    end ;
    Index := Process._Handles.ItemIndex( Handle ) ;
    if( Index = -1 ) then // Handle not found
    begin
        exit ;
    end ;
    Result := True ;
    Process._Handles.Delete( Index ) ;
end ;
First we get the process instance and return an error if it is not found. Then we check the _Handles list for the process to see if the process owns that handle. It is possible that some invalid value could be passed to us from the user. If we treat an invalid value as a pointer to a class instance, we could crash the executive. Further, we don't want one process accessing the handles of another process (unless the first process has granted access to the second). Nor do we want to allow access to a handle that has been closed. For these reasons, whenever a handle is referenced, we will check against the process _Handles list to see if it is a valid handle for that process. If the passed handle is anything other than a valid handle in the list, we do nothing. Otherwise, we remove the handle from the list. Regardless of what happens to the handle after this point, it is no longer a valid handle for this process.

In the next article, we will look at the FiP file interfaces and the link between the application and the executive, which transfers output to TTerminal.

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