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

The I/O Queue

In the last article, we dicussed logging a process out. Next, we will address the complementary operation: logging in. However, we need to first address the concept of queuing up I/O requests. This is due to the fact that the LOGIN program uses a timer on its input, which can cause problems on a multi-user system if there is no asynchronous I/O support. So far we haven't had a need for asynchronous I/O. But now that we do, we need to take a trip down a side road before we continue with the logging-in operation, since I think that things will make more sense in this order.

Why do we need asynchronous I/O, in general terms? First, if there are multiple processes running that each do I/O operations, they may all request I/O on the same store device. Although a virutal store might work with parallel read access, it could get corrupted if there are simultaneous write operationes. And a physical store simply cannot do more than one operation at a time. So, we may end up with a situation where process 1 issues a read request from a disk, and while it is waiting for the disk to return the information, process 2 is running and also requests a read from the disk. Since the first operation hasn't finished yet, we need to wait until it does before we can start the request from process 2. On a public disk, with many processes running, this could result in quite a few pending requests. Since we want to generally answer requests in the order they are given, the solution is to add each request to a queue as it comes in. As each operation completes, the request is removed from the queue and the next one is started.

But programs may also request asychronous I/O, such as requesting input from the user, but going on to perform background tasks while it waits for the user to provide the input. And some system calls generate multiple requests. For instance the IO_READPROMPT function of QIO issues a request to write a prompt to a terminal and also a request to read input from the user. Generally, we don't want the read to happen before the prompt, so these two requests are queued up in order.

Now, add to the foregoing the need for the executive to do I/Os to support the system. For instance, a login request may require a read of the UAF file. Or if processes are being swapped in and out of memory, data will have to be read from and written to the disk, quite apart from specific user application I/O requests.

All of this could result in rather complicated code, but we can simplify things by placing requests into a queue and then processing each request when the corresponding device is ready for the next operation. This is why we have the QIO routine, which means "Queue an I/O", which all of the I/O system calls eventually resolve down to.

By default, I/O operations (and some other system services) are done asychronously, and the program has to request to wait (be suspended) until the operation is complete. Many system services have two versions. For instance, QIO and QIOW. QIO does an asychronous I/O operation, returning immediately - probably before the operation completes, while QIOW suspends the calling process until the operation completes. In implementation, QIOW simply calls QIO and then immediately calls the Wait service.

The File Processor maintains a master I/O queue, but each device has a separate I/O queue that is specific to that device. The first one is so we can easily look up an I/O operation by handle and/or PID (such as for the purpose of cancelling it before it completes). The per-device queue is used to serialize the requests to a given device without serializing them to all devices. In other words, we don't want DISKA0 requests to wait until TERMA0 requests have completed if the terminal requests came before the disk requests. But if there are multiple requests to a single device, we generally want to handle them in the order received (there are some exceptions we'll cover in the future).

So far I've been a little lazy in implementing the I/O routines for the sake of making sure that the code and concepts were as clear as possible. This means that we've been doing the I/O synchronously. For instance, when a read is requested from a terminal, we go into a loop in the Input_Filter.Read method that doesn't return until a delimiter is encountered or the operation is cancelled. However, we can no longer do this. We won't revisit that code here - you can look at the changes when the next source release is made. But to summarize: Read only processes a single character, and builds the input string which can be retrieved via the new Get_Input method. Read is now called when a new character is received while a read request is active for that terminal.

We are also going to look at a couple of other changes to QIO, which will be used by LOGIN.

const IOState_Pending = 0 ; // Not sent to device yet
      IOState_Waiting = 1 ; // Sent to device - awaiting response
      IOState_Done = 2 ; // Device has completed operation

type TUOS_IO_Request = class
                           public
                               PID : TPID ; // Process making request
                               Handle : THandle ; // Handle for request
                               IO : integer ; // 0 = read, 1 = write, 2 = control
                               IOSB : int64 ; // Address of IOSB
                               efn : int64 ;
                               astadr : int64 ;
                               astctx : int64 ;
                               Timeout : int64 ; // Timeout period
                               Timer : int64 ; // Timer ID for timeouts
                               DDst : int64 ; // Data source/destination address
                               LDst : int64 ; // Length destination address
                               Size : int64 ; // Data transfer size, in bytes
                               Blocking : boolean ; // True if process is blocked on this operation
                               State : integer ; // See IOState_*
                       end ; // TUOS_IO_Request
This class implements an I/O request. It contains all the information necessary to track the progress of an I/O operation. The I/O queue contains a queue of instances of this class.

                        _Pending_IOs : TList ;

function TUOS_FiP.Pending_IOs : TList ;

begin
    if( _Pending_IOs = nil ) then
    begin
        _Pending_IOs := TList.Create ;
    end ;
    Result := _Pending_IOs ;
end ;
The new instance data (_Pending_IOs) is added to the FIP and contains the I/O queue. The new method is used to create (if needed) and return the FIP's I/O queue. As mentioned above, this is a list of TUOS_IO_Request instances.

                        _Pending_IOs : TList ;

function TDevice.Pending_IOs : TList ;

begin
    if( _Pending_IOs = nil ) then
    begin
        _Pending_IOs := TList.Create ;
    end ;
    Result := _Pending_IOs ;
end ;
An identical method and list are added to the TDevice class. The requests in this list are also found on the FIP's I/O queue (in fact, the same instance is in both places). But whereas the FIP's queue has all current I/O requests, each device's list only contains the I/O requests for that specific device.

procedure TUOS_FiP.QIOW( Event_Flags : cardinal ; Handle : THandle ; Func : cardinal ;
    var IOSB : TIOSB ; Astadr, Astprm, p1, p2, p3, p4, p5, p6 : int64 ) ;

var ID : int64 ;

begin
    ID := QIO( Event_Flags, Handle, Func, IOSB, Astadr, Astprm, p1, p2, p3, p4, p5, p6 ) ;
    if( ID <> 0 ) then
    begin
        TUOS_IO_Request( pointer( ID ) ).Blocking := True ;
        Wait ;
    end ;
end ;
The QIOW method that we previously covered has been renamed to QIO since it is asynchronous. This new method is added to the FIP to handle synchronous I/O. As mentioned above, it simply calls the asychronous QIO, and then waits for the operation to complete. We will cover the wait operation in the future. QIO returns an integer which is the address of the I/O request instance that was generated (or nil if QIO failed). Note that in the case where multiple I/Os are queued up (such as for IO_READPROMPT), this is the last of the queued I/Os. If QIO failed, we simply exit. Otherwise, we set the Blocking flag in the I/O request to indicate that this request is blocking the process.

    function Create_Request( Handle, IO, BufAdr, BufLen : int64 ) : TUOS_IO_Request ;

    begin
        Result := TUOS_IO_Request.Create ;
        Result.PID := Kernel.PID ;
        Result.Handle := Handle ;
        Result.IO := IO ;
        Result.DDst := BufAdr ;
        Result.Size := BufLen ;
        Result.IOSB := int64( @IOSB ) ;
        if( Timeout <> 0 ) then
        begin
            Result.Timeout := HAL.Timestamp + Timeout * One_Second ;
            Result.Timer := SSC.Create_Timer( 0, Result.Timeout ) ;
        end ;
        Pending_IOs.Add( Result ) ;
    end ;
QIO has been modified to not make I/O calls itself, but to create an appropriate request and add it to the FIP's I/O queue. This local function does just that. The timeout value comes from the p3 parameter on QIO and indicates the number of seconds until the operation times out. It is currently only implemented for the IO_READPROMPT function of QIO. If p3 is 0, there is no timeout on the operation. The Create_Timer function will be covered in the next article.

    if( ( Func and IO_Function_Mask ) = IO_SENSEMODE ) then
    begin
        if( TFiP_Terminal_File( Resource._File ).Terminal <> nil ) then
        begin
            fillchar( Buffer, sizeof( Buffer ), 0 ) ;
            Buffer.DevClass := DC_TERM ;
            Buffer.Characteristics := TFiP_Terminal_File( Resource._File ).Terminal.Terminal_Flags ;
            Buffer.PageWidth := TFiP_Terminal_File( Resource._File ).Terminal.Max_Char_Rows ;
            Buffer.PageLength := TFiP_Terminal_File( Resource._File ).Terminal.Max_Char_Columns ;
            IOSB.r_io_64.w_status := Write_User( Kernel, PID, p1, sizeof( Buffer ), Buffer ) ;
        end ;
    end else
This new code is added to QIO to provide a means of getting a terminal's flags. You may recall that some of this information is available via the GETDVI system call (see the TT_* codes), but this provides a means of obtaining all of the terminal flags in a single call, plus the page width and length.

    if( ( Func and IO_Function_Mask ) = IO_SETCHAR ) then
    begin
        case Mode of
            0 : // Set characteristics
                begin
                    Get_User_Data( Kernel, PID, P1, sizeof( Buffer ), Buffer, IOSB.r_io_64.w_status ) ;
                    if( IOSB.r_io_64.w_status = UE_ERROR ) then
                    begin
                        exit ;
                    end ;
                    If( Buffer.DevClass = DC_TERM ) then
                    begin
                        if( TFiP_Terminal_File( Resource._File ).Terminal <> nil ) then
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Terminal_Flags :=
                                Buffer.Characteristics ;
                            if( Buffer.PageWidth <> 0 ) then
                            begin
                                TFiP_Terminal_File( Resource._File ).Terminal.Max_Char_Rows :=
                                    Buffer.PageWidth ;
                            end ;
                            if( Buffer.PageLength <> 0 ) then
                            begin
                                TFiP_Terminal_File( Resource._File ).Terminal.Max_Char_Columns :=
                                    Buffer.PageLength ;
                            end ;
                        end ;
                    end ;
                end ;
        end ; // case Mode
    end else
This addition to QIO provides a means of setting a terminal's characteristics (it's flags and page width/height). This is the inverse of the IO_SENSEMODE function above.

        Timeout := 0 ;
        if( ( Mode and IOM_TIMED ) <> 0 ) then
        begin
            Timeout := p3 ;
        end ;

        Request := Create_Request( Handle, 0, p1, p2 ) ;
        Result := int64( Request ) ;
This code is added to the IO_READPROMPT function of QIO, setting the timeout value, if requested - which is used in the Create_Request method.

In the next article, we will discuss UOS timers.

 

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