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


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

Base Classes, Testing, and More

Base Classes
Now that we've written our first classes, it is time to take a look at the larger framework under which we are developing the various pieces of UOS. Most classes, including the main UOS components need to be emancipated and need to support exception handling. But rather than including this support in each class by copying the code and instance data, it makes more sense to create a base class from which all of our classes descend. Here is the abstract base class:

type TBase_COM_Interface = class
                               public { API... }
                                   { Initialize the object.  Returns result of
                                     initialization attempt. }
                                   function Initialize : TUnified_Exception ;
                                       virtual ; stdcall ; abstract ;

                                   { Terminate the use of the object.  Returns
                                     result of termination.  This generally
                                     should not be used - instead, use the
                                     Detach method. }
                                   function Terminate : TUnified_Exception ;
                                       virtual ; stdcall ; abstract ;

                                   { Increments the reference count for the
                                     object. }
                                   procedure Attach ; virtual ;
                                       stdcall ; abstract ;

                                   { Decrements the reference count for the
                                     object.  When count reaches 0, the object
                                     is destroyed. }
                                   procedure Detach ; virtual ;
                                       stdcall ; abstract ;

                                   { Returns True if this class is the passed
                                     class name. }
                                   function Is_Class( Name : PChar ) : boolean ;
                                       virtual ; stdcall ; abstract ;

                                   { Version of the COM interface for this
                                     object. }
                                   function Interface_Version : longint ;
                                       virtual ; stdcall ; abstract ;

                                   { Facility ID for the facility represented
                                     by this object.  Returns -1 if no
                                     facility assigned. }
                                   function Facility_ID : longint ; virtual ;
                                       stdcall ; abstract ;

                                   { Facility version.  Result only has
                                     meaning if Facility_ID doesn't return
                                     -1. }
                                   function Facility_Version : longint ;
                                       virtual ; stdcall ; abstract ;

                                   { Returns a debugging interface for the
                                     object.  Returns nil if not supported. }
                                   function Debugger : TDebug_Interface ;
                                       virtual ; stdcall ; abstract ;

                                   { Returns last error. }
                                   function Last_Error : TUnified_Exception ;
                                       virtual ; stdcall ; abstract ;

                                   { Returns a pointer to an object with
                                     extended common COM methods for object.
                                     Always returns nil for now. }
                                   function Extension : pointer ; virtual ;
                                       stdcall ; abstract ;
                           end ; { TBase_COM_Interface }

We use this abstract base class for all our UOS objects. We've already discussed Attach and Detach. Initialize is used to (re)initialize the object after it is constructed, and Terminate is used to free the instance. In our case, our constructor does all the initialization and there is no need to reinitialize the instance, so our Initialize does nothing and our Terminate just calls Free. Note: Terminate should only be called from Detach. Interface_Version is used to indicate the version of the TBase_COM_Interface class. This is intended for use in the future, if we make significant changes to the class, for backwards compatibility. For our purposes, we ignore this. Facility_ID is a numeric value that uniquely identifies our class (more on this in a later article). Facility_Version indicates the version of our class (times 10). Thus, we will return 10, which means we are V1.0 of our class. Debugger returns a debugging interface for our class. This is intended for run-time debugging. For now, we will return nil to indicate that we don't support that feature. Finally, Extension is to allow us to extend the base class capabilities, without requiring a complete build of code that uses it when we do. Is_Class is a way to query the class name. Which we will ignore for now. It may seem like a lot, but we are still only really interested in Attach, Detach, and Facility_Version.

It would be annoying to have to include overrides for all of these methods - especially since they do the same thing in every class, for the most part. So, we have a descendant of this class that includes all the default behavior. Then we only have to override what is different for our class (in our case, this is only the Facility_Version):

type TCommon_COM_Interface = class( TBase_COM_Interface )
                                 private { Instance data... }
                                     Reference_Count : longint ;
                                     _Last_Error : TUnified_Exception ;

                                 public { API... }
                                     { Initialize the object.  Returns result of
                                       initialization attempt. }
                                     function Initialize : TUnified_Exception ;
                                         override ; stdcall ;

                                     { Terminate the use of the object.  Returns
                                       result of termination.  This generally
                                       should not be used - instead, use the
                                       Detach method. }
                                     function Terminate : TUnified_Exception ;
                                         override ; stdcall ;

                                     { Increments the reference count for the
                                       object. }
                                     procedure Attach ; override ; stdcall ;

                                     { Decrements the reference count for the
                                       object.  When count reaches 0, the object
                                       is destroyed. }
                                     procedure Detach ;
                                         override ; stdcall ;

                                     { Version of the COM interface for this
                                       object. }
                                     function Interface_Version : longint ;
                                         override ; stdcall ;

                                     { Facility ID for the facility represented
                                       by this object.  Returns -1 if no
                                       facility assigned. }
                                     function Facility_ID : longint ;
                                         override ; stdcall ;

                                     { Facility version.  Result only has
                                       meaning if Facility_ID doesn't return
                                       -1. }
                                     function Facility_Version : longint ;
                                         override ; stdcall ;

                                     { Returns a debugging interface for the
                                       object.  Returns nil if not supported. }
                                     function Debugger : TDebug_Interface ;
                                         override ; stdcall ;

                                     { Returns last error. }
                                     function Last_Error : TUnified_Exception ;
                                         override ; stdcall ;

                                     { Returns a pointer to an object with
                                       extended common COM methods for object.
                                       Always returns nil for now. }
                                     function Extension : pointer ; override ;
                                         stdcall ;

                                 protected // Internal utility routines...
                                     { Note: Don't make any of these virtual, as
                                       that could mess up VMT layout.  This class
                                       is simply to provide implementation of
                                       the abstract base class. }
                                     procedure Set_Last_Error( E : TUnified_Exception ) ;
                             end ; { TCommon_COM_Interface }

function TCommon_COM_Interface.Initialize : TUnified_Exception ; stdcall ;

begin
    Result := nil ;
    Reference_Count := 0 ;
end ;


function TCommon_COM_Interface.Terminate : TUnified_Exception ; stdcall ;

begin
    Result := nil ;
    Free ;
end ;


procedure TCommon_COM_Interface.Attach ; stdcall ;

begin
    inc( Reference_Count ) ;
end ;


procedure TCommon_COM_Interface.Detach ; stdcall ;

begin
    dec( Reference_Count ) ;
    if( Reference_Count < 1 ) then
    begin
        Terminate ;
    end ;
end ;


function TCommon_COM_Interface.Interface_Version : longint ; stdcall ;

begin
    Interface_Version := 0 ;
end ;


function TCommon_COM_Interface.Facility_ID : longint ; stdcall ;

begin
    Facility_ID := -1 ;
end ;


function TCommon_COM_Interface.Facility_Version : longint ; stdcall ;

begin
    Facility_Version := 0 ;
end ;


function TCommon_COM_Interface.Debugger : TDebug_Interface ; stdcall ;

begin
    Debugger := nil ;
end ;


function TCommon_COM_Interface.Extension : pointer ; stdcall ;

begin
    Extension := nil ;
end ;


// Internal utility routines...

procedure TCommon_COM_Interface.Set_Last_Error( E : TUnified_Exception ) ;

begin
    if( E <> nil ) then
    begin
        E.Attach ;
    end ;
    if( _Last_Error <> nil ) then
    begin
        _Last_Error.Detach ;
    end ;
    _Last_Error := E ;
end ;

We'll keep these in the COMInter unit, and descend our class from TCommon_COM_Interface. In fact, we will descend all of our classes from this base class. Since the base class handles the attach, detach, and Set_Last_Error, we can drop those from our class.

TUnified_Exception is the one exception to descending all things from the base class. That is because the base class references this class and we don't want to create a circular reference. As a consequence the class definition is very similar to TBase_COM_Interface. Here is the actual base class:

type tUnified_Exception = class
                            public
                                function Initialize : TUnified_Exception ;
                                    virtual ; stdcall ; abstract ;
                                procedure Terminate ; virtual ;
                                    stdcall ; abstract ;
                                procedure Attach ;
                                    virtual ; stdcall ; abstract ;
                                procedure Detach ;
                                    virtual ; stdcall ; abstract ;
                                function Is_Class( Name : PChar ) : boolean ;
                                    virtual ; stdcall ; abstract ;
                                function Interface_Version : longint ;
                                    virtual ; stdcall ; abstract ;
                                function Get_Facility : longint ; virtual ;
                                    stdcall ; abstract ;
                                function Get_Facility_Version : longint ;
                                    virtual ; stdcall ; abstract ;
                                function Version : longint ; virtual ;
                                    stdcall ; abstract ;
                                function Debugger : TDebug_Interface ;
                                    virtual ; stdcall ; abstract ;
                                function Severity : longint ; virtual ;
                                    stdcall ; abstract ;
                                function Error_Text( var Size, Typ : longint ) : PChar ;
                                    virtual ; stdcall ; abstract ;
                                function Get_Error : longint ; virtual ;
                                    stdcall ; abstract ;
                                function Get_Previous : tUnified_Exception ;
                                    virtual ; stdcall ; abstract ;
                        end ;

As in the previous case, rather than have each descendant class implement attach, detach, et al, we will descend from a class that implements default methods. Here it is:
type TStandard_Error_Interface = class( tUnified_Exception )
                                     private
                                         _Reference_Count : integer ;
                                         _Additional_Text : string ;
                                         Temp : string ;

                                     public
                                         function Initialize : TUnified_Exception ;
                                             override ;
                                         procedure Terminate ; override ;
                                         procedure Attach ; override ;
                                         procedure Detach ; override ;
                                         function Is_Class( Name : PChar ) : boolean ;
                                             override ;
                                         function Interface_Version : longint ;
                                             override ;
                                         function Get_Facility : longint ;
                                             override ;
                                         function Get_Facility_Version : longint ;
                                             override ;
                                         function Version : longint ; override ;
                                         function Debugger : TDebug_Interface ;
                                             override ;
                                         function Severity : longint ;
                                             override ;
                                         function Error_Text( var Size, Typ : longint ) : PChar ;
                                             override ;
                                         function Get_Error : longint ;
                                             override ;
                                         function Get_Previous : tUnified_Exception ;
                                             override ;
                                 end ; { TStandard_Error_Interface }

function TSStandard_Error_Interface.Get_Facility : longint ;

begin
    Get_Facility := -1 ;
end ;


function TSStandard_Error_Interface.Get_Facility_Version : longint ;

begin
    Get_Facility_Version := 0 ;
end ;


function TSStandard_Error_Interface.Severity : longint ;

begin
    Severity := UE_Error ;
end ;


function TSStandard_Error_Interface.Error_Text( var Size, Typ : longint ) : string ;

var P : string ;

begin
    P := '' ;
    Size := length( P ) ;
    Typ := 0 ; { 7-bit ASCII }
    Error_Text := P ;
end ;


function TSStandard_Error_Interface.Get_Error : longint ;

begin
    Get_Error := 0 ;
end ;


function TSStandard_Error_Interface.Get_Previous : psUnified_Exception ;

begin
    Get_Previous := nil ;
end ;


procedure TSStandard_Error_Interface.Terminate ;

begin
    if( Get_Previous <> nil ) then
    begin
        Get_Previous.Free ;
    end ;
end ;


procedure TSStandard_Error_Interface.Set_Memory_Type( MT : TMemory_Type ) ;

begin
    _Memory_Type := MT ;
end ;


function TSStandard_Error_Interface.Get_Memory_Type : TMemory_Type ;

begin
    Get_Memory_Type := _Memory_Type ;
end ;

Testing
The subject of code testing is a large one and we won't go into gory detail here. But, since it is essential to be able to rely upon the operating system code, we need to be rigorous in our development. I have implemented a three-level certification process for the code. Level 1 is what we call a functional test. The purpose is to test each functional aspect of the code. In this class, it is relatively easy, as there are really only three different functions: expand, truncate, and translate offset to pointer. So, we will construct an instance of the class, assign it a memory store, and try out the various functions, verifying that after each request that the result is as expected.
Level 2 uses "coverage analysis" to make sure that each line of code in the class is covered by at least one test. In the absence of an abend, and loop, if, and call constructs, a series of statements will be executed from start to finish. So, we know that, in that case, if the first statement is exceuted, then all of them will be executed. Because of this, we can quickly set up a test case to help us make sure that all code is covered by our tests: we put a breakpoint at the beginning of each method and at every "if" (inside the block), "for", "while", etc, and in the "else" block of all ifs. Then we run the test. Each time we hit a breakpoint, we remove the breakpoint since we know that code block is being executed as a result of our tests. When the test finishes, any remaining breakpoints indicate code that never executed. We can then design additional test scenarios that exercise those missed code blocks. In some cases, it may not be possible, because it requires failure modes that are not easy to replicate. Except for these exceptional situations, we ought to be able to verify complete code coverage in our test.
Note that it would be ideal to test every path through our code. For example, given the following code:

if( A ) then
begin
  // case 1
end else
begin
  // case 2
end ;
if( B ) then
begin
  // case 3
end else
begin
  // case 4
end ;

There are 4 paths through this code:
ABcase
FalseFalse2, 4
FalseTrue2, 3
TrueFalse1, 4
TrueTrue1, 3

In fact, the number of unique paths through our program will be 2 raised to the number of branches (if statements). So, a class with 30 if statements would have over 1 billion unique paths. Not to mention that all loops count as branches as well. Thus, testing all possible paths is just not feasible. So, simple coverage analysis is the best we can usually manage.
Level 3 is called a "stress test". The idea here is to randomly exercise the class. In the case of our class, we will randomly truncate, extend, translate offsets, and read and write (based on those offsets). We will verify correct operation by reading what was previously written. If we run several hundred million random operations, we can hope to possibly find errors that we missed. There is no guarantee that we will find errors that we missed, and this is not a substitute for good coding. But since everyone makes mistakes, it is nice to have an extra check. In fact, this process did discover a bug that happened under a specific set of circumstances that I hadn't thought of checking in the existing tests. It was in the Truncation code.
Original code (at the end of the truncation loop):
                // Move to next allocation cluster in chain...
                P := Last ;
                if( P <> 0 ) then
                begin
                    inc( Turns ) ;
                    Read( P ) ;
                end ;
            end ; // while( P <> 0 )

Corrected code:
                // Move to next allocation cluster in chain...
                if( ( Value <= 0 ) and ( Last = 0 ) ) then // Done with the truncation
                begin
                    Last := P ;
                    break ;
                end ;
                P := Last ;
                if( P <> 0 ) then
                begin
                    inc( Turns ) ;
                    Read( P ) ;
                end ;
            end ; // while( P <> 0 )

Another benefit of the stress test is that we can get a good estimate as to the performance of the class. In our case, we achieved over 10,000 operations per second on a memory store, on a 3.2 GHz CPU. This includes the random 512-byte reads and writes. We can say, with a fair amount of certainty, that a RAM Disk would support 10,000 file operations per second using our class. That is adequate performance, in my book.
Another benefit of writing a test routine is that we can run it every time we make a change to the code to verify that we didn't break anything. But, we need to remember to add breakpoints to new code to make sure that all new code blocks are covered by our tests. This type of retesting is called "regression testing". That is, we test to make sure the code hasn't regressed to a broken state.
I also added some test "instrumentation" to the class itself to perform various checks, including making sure that all data was freed up when the size was set to 0. All of this test code slowed the speed to about 1,500 operations per second, but I feel much better than about the reliability of the code. I won't bother showing all the instrumentation code, but here is the general test code that I used:
procedure Test ;

var ACM : TCOM_Allocation_Cluster_Manager64 ;
    Buff : array[ 0..511 ] of byte ;
    H : TCOM_Heap ;
    Old_P, P, P1 : int64 ;
    S64 : TCOM_Managed_Store64 ;


    procedure error( s : string ) ;

    begin
        halt ;
    end ;


    procedure Do_Size( P : int64 ) ;

    begin
        ACM.Set_Size( P ) ;
    end ;


    procedure Do_Read( P : int64 ) ;

    var Index : longint ;
        P1 : int64 ;

    begin
        P1 := ACM.Offset_To_Pointer( P * 512 ) ;
        if( P1 <> 0 ) then
        begin
            S64.Read( P1, 512, Buff ) ;
            for Index := 0 to 511 do
            begin
                if( Buff[ Index ] <> P ) then
                begin
                    error( 'Failure' ) ;
                end ;
            end ;
        end ;
    end ;


var CA, Dummy : integer ;
    S : string ;

begin
    // Coverage test...
    ACM := TCOM_Allocation_Cluster_Manager64.Create ;
    S64 := TCOM_Heap64.Create ;
    if( ACM.Get_Store <> nil ) then
    begin
        error( 'Invalid store' ) ;
    end ;
    ACM.Set_Store( S64 ) ;
    if( ACM.Get_Store <> S64 ) then
    begin
        error( 'Invalid store' ) ;
    end ;
    H := TCOM_Heap.Create ;
    ACM.Set_Heap( H ) ;
    if( ACM.Get_Clustersize <> 128 ) then
    begin
        error( 'Invalid clustersize' ) ;
    end ;
    if( ACM.Get_Heap <> H ) then
    begin
        error( 'Invalid heap' ) ;
    end ;
    if( ACM.Get_Size <> 0 ) then
    begin
        error( 'Invalid size' ) ;
    end ;
    if( ACM.Get_Root <> 0 ) then
    begin
        error( 'Invalid root' ) ;
    end ;
    ACM.Set_Clustersize( 512 ) ;
    if( ACM.Get_Clustersize <> 512 ) then
    begin
        error( 'Invalid clustersize' ) ;
    end ;
    if( ACM.Get_Size <> 0 ) then
    begin
        error( 'Invalid size' ) ;
    end ;
    ACM.Set_Size( 65536 ) ;
    if( ACM.Get_Size <> 65536 ) then
    begin
        error( 'Invalid size' ) ;
    end ;
    ACM.Set_Size( 65536 ) ;
    if( ACM.Get_Size <> 65536 ) then
    begin
        error( 'Invalid size' ) ;
    end ;
    ACM.Set_Size( 256 ) ;
    if( ACM.Get_Size <> 512 ) then
    begin
        error( 'Invalid size' ) ;
    end ;
    if( ACM.Offset_To_Pointer( 1024000 ) <> 0 ) then
    begin
        error( 'Invalid translation' ) ;
    end ;
    ACM.Set_Size( 65536 ) ;
    P := ACM.Offset_To_Pointer( 32768 ) ;
    if( P = 0 ) then
    begin
        error( 'Invalid translation' ) ;
    end ;
    fillchar( Buff, 512, 1 ) ;
    S64.Write( P, 512, Buff ) ;
    P := ACM.Offset_To_Pointer( 65500 ) ;
    if( P = 0 ) then
    begin
        error( 'Invalid translation' ) ;
    end ;
    fillchar( Buff, 512, 2 ) ;
    S64.Write( P, 512, Buff ) ;
    P := ACM.Offset_To_Pointer( 32768 ) ;
    S64.Read( P, 512, Buff ) ;
    if( Buff[ 0 ] <> 1 ) then
    begin
        error( 'Invalid data' ) ;
    end ;
    P := ACM.Offset_To_Pointer( 32769 ) ;
    if( P = 0 ) then
    begin
        error( 'Offset_To_Pointer error' ) ;
    end ;
    P := ACM.Offset_To_Pointer( 65500 ) ;
    S64.Read( P, 512, Buff ) ;
    if( Buff[ 0 ] <> 2 ) then
    begin
        error( 'Invalid data' ) ;
    end ;
    if( ACM.Get_Root = 0 ) then
    begin
        error( 'Invalid root' ) ;
    end ;
    ACM.Set_Size( 2*65536 ) ;
    ACM.Set_Size( 0 ) ;

    ACM.Set_Heap( nil ) ;
    ACM.Set_Store( nil ) ;
    ACM.Free ;
    ACM := TCOM_Allocation_Cluster_Manager64.Create ;
    ACM.Free ;

    // Stress test...
{$IFDEF CC_Debug}
    Allocated_CAs.Clear ;
    Other_Allocated.Clear ;
{$ENDIF}
    randomize ;
    ACM := TCOM_Allocation_Cluster_Manager64.Create ;
    S64 := TCOM_Heap64.Create ;
    ACM.Set_Store( S64 ) ;
    H := TCOM_Heap.Create ;
    ACM.Set_Heap( H ) ;
    ACM.Set_Clustersize( 512 ) ;

    try
        while( true ) do
        begin
            case random( 101 ) of
                0..50 : // Set size
                    begin
                        Old_P := ACM.Get_Size ;
                        P := random( 256 ) * 512 ;
                        Do_Size( P ) ;
                        if( P > Old_P ) then // Expand
                        begin
                            Do_Size( P ) ;
                            while( Old_P < P ) do
                            begin
                                P1 := ACM.Offset_To_Pointer( Old_P ) ;
				if( P1 = 0 ) then
				begin
                             	    error( 'Failure' ) ;
				end ;
                                S64.Fill( P1, 512, Old_P div 512 ) ;
                                Old_P := Old_P + 512 ;
                            end ; // while( Old_P < P )
                        end ; // if( P > Old_P )
                    end ;
                51..99 : // Read
                    begin
                        P := random( 256 ) ;
                        Do_Read( P ) ;
                    end ;
                100 : // Zero
                    begin
                        Do_Size( 0 ) ;
                    end ;
            end ;
        end ; // while( true )
    except
    end ;
    ACM.Free ;
end ;

Other applications
There are some other ways we can use our Allocation_Cluster_Manager class, besides using it for managing files on stores. For instance, we can implement a virtual list class that works like the VCL/LCL TList class, but keeps the data in a store (such as a file). There are obviously easier methods of store a list of values in a file. But, what if we want to keep several lists in a file, each of which is updated/resized during the course of program execution? Hence, the TStandard_COM_Store64_List class. I won't bother to discuss the code - I merely present it as an example of the use of our Allocation_Cluster_Manager class which you can examine or skip, as you desire. The point of this exercise is to show that no matter what classes we write for UOS, there are potential other uses for them. Yes, we want things to work optimally for UOS. But we also want to write them so that they can be repurposed for other applications.

type TStandard_COM_Store64_List = class( TCOM_Store64_List )
                              public // Constructors and destructors...
                                  constructor Create ;
                                  destructor Destroy ; override ;

                              private // Instance data...
                                  _ACM : TCOM_Allocation_Cluster_Manager64 ;
                                  _Address : TStore_Address64 ; // Address of header
                                  _Store : TCOM_Managed_Store64 ;
                                  Header : TList_Record ;
                                  _Buffer : PChar ;
                                  _Current_Buffer : TStore_Address64 ; // Current location of Buffer

                              protected { Internal utility routines... }
                                  function Index_To_Offset( Index : longint ) : longint ;
                                  procedure Read( P : TStore_Address64 ) ;
                                  procedure Write( P : TStore_Address64 ) ;

                              public { API... }
                                  function Get_Address : TStore_Address64 ;
                                      override ; stdcall ;
                                  procedure Set_Address( Value : TStore_Address64 ) ;
                                      override ; stdcall ;
                                  function Add( Value : longint ) : longint ;
                                      override ; stdcall ;
                                  function Get_Count : longint ;
                                      override ; stdcall ;
                                  procedure Set_Count( Value : longint ) ;
                                      override ; stdcall ;
                                  function Get_Item( Index : longint ) : longint ;
                                      override ; stdcall ;
                                  procedure Set_Item( Index, Value : longint ) ;
                                      override ; stdcall ;
                                  function Get_Capacity : longint ;
                                      override ; stdcall ;
                                  procedure Set_Capacity( Value : longint ) ;
                                      override ; stdcall ;
                                  function Get_Delta : longint ;
                                      override ; stdcall ;
                                  procedure Set_Delta( Value : longint ) ;
                                      override ; stdcall ;

                                  function Get_Store : TCOM_Managed_Store64 ;
                                      override ; stdcall ;
                                  procedure Set_Store( Value : TCOM_Managed_Store64 ) ;
                                      override ; stdcall ;

                              public // API...
                                  function Is_Class( Name : PChar ) : boolean ;
                                      override ; stdcall ;
                                  function Add_Insert( Value : longint ) : longint ;
                                      override ; stdcall ;
                                  function IndexOf( Value : longint ) : longint ;
                                      override ; stdcall ;
                                  procedure Update_Header ;
                                      override ; stdcall ;
                          end ; // TStandard_COM_Store64_List

// TStandard_COM_Store64_List methods...

// Constructors and destructors...

constructor TStandard_COM_Store64_List.Create ;

begin
    inherited Create ;

    _ACM := TCOM_Allocation_Cluster_Manager64.Create ;
    _ACM.Set_Heap( TCOM_Heap.Create ) ;
end ;


destructor TStandard_COM_Store64_List.Destroy ;

begin
    _ACM.Detach ;
    Set_Store( nil ) ;

    inherited Destroy ;
end ;


{ Internal utility routines... }

procedure TStandard_COM_Store64_List.Read( P : TStore_Address64 ) ;

begin
    if( _Current_Buffer = P ) then // Already have this chunk in the buffer
    begin
        exit ;
    end ;
    _Store.Read( P, Header.Delta * sizeof( longint ), _Buffer^ ) ;
    _Current_Buffer := P ;
end ;


procedure TStandard_COM_Store64_List.Write( P : TStore_Address64 ) ;

begin
    if( P = -1 ) then // Use last address
    begin
        P := _Current_Buffer ;
    end ;
    _Store.Write( P, Header.Delta * sizeof( longint ), _Buffer^ ) ;
    _Current_Buffer := P ;
end ;


{ API... }

function TStandard_COM_Store64_List.Get_Address : TStore_Address64 ;

begin
    Result := _Address ;
end ;


procedure TStandard_COM_Store64_List.Set_Address( Value : TStore_Address64 ) ;

begin
    if( _Address <> Value ) then
    begin
        _Address := Value ;
        if( ( Value <> 0 ) and ( _Store <> nil ) ) then
        begin
            _Store.Read( _Address, sizeof( Header ), Header ) ;
        end else
        begin
            fillchar( Header, sizeof( Header ), 0 ) ;
        end ;
        _ACM.Set_Clustersize( Header.Delta * sizeof( longint ) ) ;
        _ACM.Set_Root( Header.List ) ;
    end ;
end ;


function TStandard_COM_Store64_List.Add( Value : longint ) : longint ;

var P : TStore_Address64 ;
    S : TStore_Address64 ;

begin
    // Make sure enough data is allocated...
    S := Header.Count + 1 ;
    S := S * sizeof( Value ) ; // Offset in data...
    if( Header.Count >= Header.Capacity ) then // Need a larger amount of data
    begin
        Set_Capacity( Header.Capacity + Header.Delta ) ;
        _ACM.Set_Size( S ) ;
        if( Header.Count >= Header.Capacity ) then // Couldn't expand
        begin
            Result := -1 ;
            exit ;
        end ;
    end ;

    // Update header...
    Result := Header.Count ; // This will be the offset we added it at
    inc( Header.Count ) ;
    Update_Header ;

    // Write value...
    P := _ACM.Offset_To_Pointer( S - sizeof( Value ) ) ;
    if( P = 0 ) then
    begin
        Result := -1 ;
        exit ;
    end ;
    Read( P ) ;
    S := Result - ( ( Result div Header.Delta ) * Header.Delta ) ; // Offset in this buffer
    move( Value, _Buffer[ S * sizeof( Value ) ], sizeof( Value ) ) ;
    Write( P ) ;
end ; // TStandard_COM_Store64_List.Add


function TStandard_COM_Store64_List.Get_Count : longint ;

begin
    Result := Header.Count ;
end ;


procedure TStandard_COM_Store64_List.Set_Count( Value : longint ) ;

begin
    Set_Last_Error( nil ) ;
    if( Value < 0 ) then
    begin
        exit ;
    end ;
    if( Value > Capacity ) then
    begin
        Set_Capacity( Value ) ;
    end ;
    if( Value > Capacity ) then // Couldn't expand
    begin
        exit ;
    end ;
    while( Header.Count < Value ) do
    begin
        Add( 0 ) ;
        if( Last_Error <> nil ) then
        begin
            exit ;
        end ;
    end ;
    Header.Count := Value ;
    Update_Header ;
end ;


function TStandard_COM_Store64_List.Index_To_Offset( Index : longint ) : longint ;

var P : TStore_Address64 ;
    S : int64 ;

begin
    S := Index ;
    S := S * sizeof( Result ) ; // Offset in data...
    P := _ACM.Offset_To_Pointer( S ) ;
    Read( P ) ;
    Result := Index - ( ( Index div Header.Delta ) * Header.Delta ) ; // Longint offset in this buffer
    Result := Result * sizeof( longint ) ; // Byte offset in the buffer
end ;


function TStandard_COM_Store64_List.Get_Item( Index : longint ) : longint ;

var S : TStore_Address64 ;

begin
    if( ( Index < 0 ) or ( Index >= Header.Count ) ) then
    begin
        Set_Last_Error( Create_Store_List64_Exception( Store_List64_Err_Invalid_List_Index ) ) ;
        Result := -1 ;
        exit ;
    end ;

    S := Index_To_Offset( Index ) ;
    move( _Buffer[ S ], Result, sizeof( Result ) ) ;
end ;


procedure TStandard_COM_Store64_List.Set_Item( Index, Value : longint ) ;

var S : TStore_Address64 ;

begin
    if( ( Index < 0 ) or ( Index >= Header.Count ) ) then
    begin
        Set_Last_Error( Create_Store_List64_Exception( Store_List64_Err_Invalid_List_Index ) ) ;
        exit ;
    end ;

    S := Index_To_Offset( Index ) ;
    move( Value, _Buffer[ S ], sizeof( Value ) ) ;
    Write( -1 ) ;
end ;


function TStandard_COM_Store64_List.Get_Capacity : longint ;

begin
    Result := Header.Capacity ;
end ;


procedure TStandard_COM_Store64_List.Set_Capacity( Value : longint ) ;

begin
    if( Value < 0 ) then
    begin
        exit ; // Invalid value
    end ;
    if( _Store = nil ) then
    begin
        exit ;
    end ;
    if( Header.Delta = 0 ) then
    begin
        _ACM.Set_Clustersize( 16 ) ;
        Header.Delta := _ACM.Get_Clustersize div sizeof( longint ) ; // Default
    end ;
    if( Value mod Header.Delta <> 0 ) then
    begin
        Value := Value + Header.Delta ;
    end ;
    Value := ( Value div Header.Delta ) * Header.Delta ; // New size
    _ACM.Set_Size( Value * sizeof( longint ) ) ;
    Header.Capacity := _ACM.Get_Size div sizeof( longint ) ;
    Update_Header ;
end ;


function TStandard_COM_Store64_List.Get_Delta : longint ;

begin
    Result := Header.Delta ;
end ;


procedure TStandard_COM_Store64_List.Set_Delta( Value : longint ) ;

begin
    if( Header.List <> 0 ) then // Already have things allocated
    begin
        exit ; // Cannot change delta
    end ;
    if( Value <> Header.Delta ) then
    begin
        _ACM.Set_Clustersize( Value * sizeof( longint ) ) ;
        Header.Delta := _ACM.Get_Clustersize div sizeof( longint ) ;
        Update_Header ;
        Reallocmem( _Buffer, _ACM.Get_Clustersize ) ;
    end ;
end ;


function TStandard_COM_Store64_List.Get_Store : TCOM_Managed_Store64 ;

begin
    Result := _Store ;
end ;


procedure TStandard_COM_Store64_List.Set_Store( Value : TCOM_Managed_Store64 ) ;

begin
    if( Value <> nil ) then
    begin
        Value.Attach ;
    end ;
    if( _Store <> nil ) then
    begin
        _Store.Detach ;
    end ;
    _Store := Value ;
    _Address := 0 ;
    _Current_Buffer := 0 ;
    if( _Store <> nil ) then
    begin
        _ACM.Set_Store( _Store ) ;
        Header.Delta := _Store.Min_Storage div sizeof( longint ) ;
        _ACM.Set_Clustersize( Header.Delta * sizeof( longint ) ) ;
        Reallocmem( _Buffer, _ACM.Get_Clustersize ) ;
    end ;
end ;


function TStandard_COM_Store64_List.Is_Class( Name : PChar ) : boolean ; stdcall ;

var S : string ;

begin
    S := Name ;
    Result := lowercase( S ) = 'tstandard_com_store64_list' ;
end ;


function TStandard_COM_Store64_List.Add_Insert( Value : longint ) : longint ;

begin
    Result := IndexOf( 0 ) ;
    if( Result = -1 ) then
    begin
        Result := Add( Value ) ;
    end else
    begin
        Set_Item( Result, Value ) ;
    end ;
end ;


function TStandard_COM_Store64_List.IndexOf( Value : longint ) : longint ;

var Index : integer ;

begin
    for Index := 0 to Count - 1 do
    begin
        if( Get_Item( Index ) = Value ) then
        begin
            Result := Index ;
            exit ;
        end ;
    end ;
    Result := -1 ;
end ;


procedure TStandard_COM_Store64_List.Update_Header ;

begin
    if( _Store = nil ) then
    begin
        exit ;
    end ;
    if( _Address = 0 ) then
    begin
        _Address := _Store.Allocate( sizeof( Header ) ) ;
    end ;
    if( _Address <> 0 ) then
    begin
        if( Header.List = 0 ) then
        begin
            Header.List := _ACM.Get_Root ;
        end ;
        _Store.Write( _Address, sizeof( Header ), Header ) ;
        if( Header.List <> 0 ) then
        begin
            _ACM.Set_Root( Header.List ) ;
        end ;
    end ;
end ;

In our next article, we will continue building our UOS native file system. Next stop: the native file object.