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
We are also going to look at a couple of other changes to
_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.
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.
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.
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.
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.
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.
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. |