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

Ground Rules

In the first article, we described the basic idea behind the Universal Operating System (UOS). In this article, we will discuss some ground rules that we will follow for this series.

Why write UOS? Besides being a fun and education exercise, I've been disappointed by every other Operating System out there and want something better. Perhaps the best O/S I've seen was RSTS/E (within the constraints of the hardware limitations imposed by the 16-bit PDP-11), followed closely by VMS. Both of these Operating Systems were designed by brilliant engineers at Digital Equipment Corporation, and were reliable, feature-rich, and well-documented. In my entire career, I only found one bug in either of those Operating Systems. Not surprisingly, then, those systems have a huge influence on the design of UOS. In fact, we can take the VMS documentation and use it as a specification for UOS - with some alterations.

I see two extremes in software: either it is difficult to use because it provides a lot of flexibility and power, or it is simple to use because it doesn't. This applies to programming interfaces for programmers and user interfaces for users. What we want in UOS is all the power and flexibility possible, but also a simplified user interface that defaults all the options to safe and sane values. The best way to implement this "best of both worlds" approach is to provide two interfaces: the flexible one, and a simplified layer over that which translates the simplified interface into the more flexible one. We will discuss this dual approach as we go.

I want to make it clear that, though this is partially for educational purposes, I will be taking a practical approach instead of an "ivory tower" approach. If we were talking pure theory, we could simplify a great many things. Instead, we are going to tackle the challenges, trade-offs, and decision making required by any software developer who wants to write software for the real world.

Pascal
One of the first things that many will notice is my use of Pascal sources to illustrate the algorithms. This is for several reasons. In olden days, operating systems were written in assembly code. By the 1970s, some of them were written using languages that were glorified assemblers, such as Bliss and C. We will use a high-level language, Pascal, to write UOS. The reasons for avoiding assembly include the fact that it is harder to write assembly than high-level languages, which means it takes longer to code, debug, and test. Second, assembly is specific to a hardware platform. Thus, if you want your operating system to run on multiple hardware platforms, you must rewrite it in an assembly code for each of those platforms. A high-level language allows you to write it once and compile it for different target platforms. Here's an example of some Pascal code and what it would look like if we wrote it in assembly:

In Pascal:


    P := Pos( HT, S ) + 1 ;
    while( S[ P ] <> HT ) do inc( P ) ;

In assembly:


    lea eax,[ebp-$918]
    mov edx,[ebp-$10]
    call @UStrFromLStr
    mov edx,[ebp-$918]
    mov ecx,1
    mov eax,$62f890
    call Pos
    inc eax
    mov [ebp-$8],eax
    jmp A
loop:
    inc dword ptr [ebp-8]
A:
    mov eax,[ebp-$10]
    mov edx,[ebp-8]
    cmp byte ptr [eax+edx-1],9
    jnz loop

Obviously, the Pascal is shorter and easier to read. The assembly code is specific to the Intel iAPX CPUs. I'd have to write different assembly code to accomplish the same on an ARM CPU. Thanks to tools such as FPC and Delphi, the same Pascal code can be compiled to run on multiple different hardware platforms.

Of course, UOS could be coded in any sufficiently modern compilable language. In fact, I expect it will end up being coded in multiple languages. But there are other reasons I will be using Pascal. First, it is easier to read than C or C++. It was designed as a teaching tool and I will be using this series of articles as such. Second, I have a large library of Pascal code that will help speed up the process rather than having to write absolutely everything from scratch. Third, I don't have a lot of spare time to devote to this, so I need to use every possible advantage to work on UOS. One advantage is that I write Pascal code about 10% faster than other languages, such as C++. Finally, one advantage that Delphi and FPC have over C++ is that we can "hook" into certain features, such as memory management.

Not only do compilers target specific hardware platforms, but they also target software platforms - specifically operating systems. When memory is requested for heap, for instance, a call is made to the operating system to allocate physical memory. Which operating system call is made depends upon which operating system is being targeted by the compiler. Thus, code compiled for an Android won't run on Windows. Since there are no compilers that target UOS at this moment, we have to be careful not to use any features of the compiler which would result in a call to the operating system. Our code for UOS has to run stand-alone. Therefore, we have to override the memory manager to use our own memory management code rather than calling to an operating system (we will cover this in a future article). This cannot be easily done with most compilers and is a major reason for using Pascal. But we also have to be careful of other features of Object Pascal provided by Delphi and FPC. For instance, we cannot use the halt statement as that calls the operating system to end the running image. We can't use any of the file statements or functions since they, too, call the operating system. Thus, we cannot use reset, rewrite, append, read, write, blockread, blockwrite, seek, eof, eoln, filepos, filesize, assignfile, closefile, (or any others I've forgotten). Also, we dare not use any Windows-specific units, such as Windows. Exceptions are also handled by Windows, so we cannot use them - and we must be careful not to generate any. Nor can we use any DLLs that come with Windows or the compiler. This means we'll have to be careful in writing our code, in whatever language - and we'll address situations as they come up.

Terminology
Now to define some terms.
Component: Any stand-alone code that can be dynamically loaded and used in UOS. In Windows terms, a component would most likely exist as an EXE or DLL.
Kernel: In most operating systems, this refers to the basic part of the O/S that contains such things as memory management, file systems, and hardware access. In UOS, these are all separate components. The Kernel is responsible for loading the components, coordinating activity between them, and providing an interface for them to the rest of the system.
Shell: The user interface for an operating system. Some systems have multiple shells. In the case of Windows, for instance, there is the Windows GUI and command-line box.
Store: Anything that we can store data on and read it back from. A common example of a store is a hard disk. But there are other stores as well, such as floppy disks, files, and memory. Stores have a generic interface in UOS, which allows us to deal with them in a general fashion. For example, a file system could exist on a store that is a disk partition, or in RAM (eg a RAM Disk), or even within a file that exists on another store (a container file). Stores can be fixed in size (such as a disk), or expandable (such as a file).
Managed Store: A store that keep track of what areas of the store have been used, and allows us to allocate/deallocate areas of the store. Heaps are an example of a managed store.

In the next article, we will dive into our first UOS component, and you might be surprised which one we pick.

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