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

SETTERM utility, part 3

In the previous articles, we started looking at SETTERM. We will finish up in this article.

First we'll consider the terminal definition file. sys$system:setterm.txt is a text file that contains options for different terminals, one per line. Each line starts with the name of a terminal model, an equal sign, and a series of switches that tell SETTERM how to configure the terminal. One of the names provided in the default file is "unknown". Additional terminals, and/or alterations to existing terminals, can be added to the file by the system administrator. I won't include the whole file here, but here's a couple of (truncated) lines:

la120=/noadvanced_video/noansi_crt/crfill=0/nodec_crt/nodec_crt5/form/hardcopy/page=66 vt100=/ansi_crt/crfill=0/dec_crt/nodec_crt2/noeight_bit/noform/scope/lffill=0/lowercase/page=24

This file both defines which terminal types are known, but also what characteristics to apply to a given terminal type. The following two functions in setterm use this file.

function Get_Device_Switches( S : string ) : string ;

var Buffer : PAnsiChar ;
    F : TCOM_UOS_File ;
    I : integer ;
    S1 : string ;       

begin
    Result := '' ;
    s := lowercase( S ) ;
    F := Open_Binary_File( 'sys$system:setterm.txt', FAB_V_GET or FAB_V_SHRALL ) ;
    while( not F.EOF ) do
    begin
        F.Readln( Buffer ) ;
        S1 := string( Buffer ) ;
        I := pos( '=', S1 ) ;
        if( S = lowercase( copy( S1, 1, I - 1 ) ) ) then
        begin
            Result := copy( S1, I + 1, length( S1 ) ) ;
            F.Free ;
            exit ;
        end ;
    end ;
    F.Free ;
end ;
This routine returns the switches associated with the passed terminal type. If no match is found in the setterm.txt file, a null string is returned.

function Get_Devices : string ;

var Buffer : PAnsiChar ;
    F : TCOM_UOS_File ;
    I : integer ;
    S1 : string ;

begin
    Result := '' ;
    F := Open_Binary_File( 'sys$system:setterm.txt', FAB_V_GET or FAB_V_SHRALL ) ;
    while( not F.EOF ) do
    begin
        F.Readln( Buffer ) ;
        S1 := string( Buffer ) ;
        I := pos( '=', S1 ) ;
        Result := Result + '"' + copy( S1, 1, I - 1 ) + '",' ;
    end ;
    F.Free ;
    setlength( Result, length( Result ) - 1 ) ; // Trim trailing comma
end ;
This routine iterates through the setterm.txt file, creating a comma-delimited list of known terminal types. Each type is surrounded with quotes so that the list is appropriate for a UUI list component.

Next, we will look at the QIO changes to support SETTERM. I should note that there are some differences from VMS here, in addition to the ones noted in the previous article. For instance, IO_SENSEMODE doesn't return information via p3-p6 in VMS, but does in UOS. Also, we use a single buffer size, whereas VMS is backwards compatible to earlier versions that had a smaller buffer size. UOS has a single buffer size, with a slightly modified layout. Here's the user documentation for the new QIO features (this expands upon the previous documentation):

IO_SENSEMODE function:
This function returns device-dependant information. The information is written to a buffer whose address is passed in parameter p1. This buffer has the following layout:

MnuemonicByte offsetByte lengthDescription
Devclass01Device class to use. DC_TERM indicates a terminal.
DevType11unused.
PageWidth22Number of columns.
PageLength42Number of lines/rows.
Characteristics68Device flags.

p2 is the length of the buffer pointed to by p1. But since the buffer has a specific size in UOS, we will simply ignore it. If p3-p6 are not 0, they are interpreted as addresses of 8-byte buffers to receive additional terminal values. The following table indicates what each of the buffers receives, if the corresponding parameter is provided.

ParameterDescription
p2unused/ignored.
p3Baud rate. The low 8 bits indicate the output baud rate. If bits 7-5 are not 0, they indicate the input baud rate (for split baud rates only).
p4Number of fill characters. The lowest 8 bits are the CR fill and bits 7-15 are the LF fill.
p5Parity. The low 4 bits contain the number of data bits. If the TT_M_DISPARERR flag is set then parity errors are ignored on this device. If the TT_M_PARITY flag is not set, no party bit is used. If the TT_M_ODD flag is set then the parity is odd, otherwise the parity is even (this bit has no meaning if no parity is being used).
p6Type-ahead buffer size, in bytes.

IO_SETMODE and IO_SETCHAR functions:
IO_SETMODE is used to set current terminal characteristics, while IO_SETCHAR is used to set permanent terminal characteristics. For subfunctions, such as IOM_CTRLCAST, both functions operate the same. When no subfunction is used, terminal flags can be set. Parameter p1 points to a buffer that defines which characteristics to set. The buffer has the following layout:
MnuemonicByte offsetByte lengthDescription
Devclass01Device class to use. DC_TERM indicates a terminal.
DevType11unused.
PageWidth22Number of columns.
PageLength42Number of lines/rows.
Characteristics68Device flags.
p3-p6 contain additional terminal attributes. The following table indicates what attributes are passed in p2-p6:
ParameterDescription
p2unused/ignored.
p3Baud rate. The low 8 bits indicate the output baud rate. If bits 7-15 are not 0, they indicate the input baud rate (for split baud rates only).
p4Number of fill characters. The lowest 8 bits are the CR fill and bits 7-15 are the LF fill.
p5Parity. If the TT_M_ALTRPAR flag is included in this value, the parity is altered according to the TT_M_PARITY and TT_M_ODD flags (otherwise those flags are ignored). If the TT_M_DISPARERR flag is set in this parameter value then parity errors are ignored on this device. If the TT_M_ALTFRAME flag is included in this parameter value, the low 4 bits contain the number of data bits.
p6If the TT_M_ALTYPEAHD flag is included in this value, the low 16 bits indicates the new type-ahead buffer size, in bytes.
The baud rates are defined as follows:
MnuemonicValue
TT_C_BAUD_501
TT_C_BAUD_752
TT_C_BAUD_1103
TT_C_BAUD_1344
TT_C_BAUD_1505
TT_C_BAUD_3006
TT_C_BAUD_6007
TT_C_BAUD_12008
TT_C_BAUD_18009
TT_C_BAUD_200010
TT_C_BAUD_240011
TT_C_BAUD_360012
TT_C_BAUD_480013
TT_C_BAUD_720014
TT_C_BAUD_960015
TT_C_BAUD_1920016
TT_C_BAUD_3840017
TT_C_BAUD_5760018
TT_C_BAUD_7680019
TT_C_BAUD_11520020

Note that any of the attributes specified by p3-p6 are left unchanged if the corresponding parameter is 0.

Note that PHY_IO privilege is required to set permanent characteristics, and LOG_IO is required to set current settings unless speed settings are enabled (changing that setting also requires PHY_IO or LOG_IO). PHY_IO is also required to alter the type-ahead buffer size. However, virtual terminals have no permanent characteristics, thus setting permanent characteristics is the same as setting current characteristics, and requires no privileges.

            if( p3 <> 0 ) then // Return speed
            begin
                I64 := TFiP_Terminal_File( Resource._File ).Terminal.Speed ;
                I64 := Convert_to_Baud_Ordinal( I64 ) ;
                if(
                    ( TFiP_Terminal_File( Resource._File ).Terminal.Speed <> 0 )
                    and
                    ( TFiP_Terminal_File( Resource._File ).Terminal.Speed <> 
                        TFiP_Terminal_File( Resource._File ).Terminal.Input_Speed )
                  ) then
                begin
                    I64 := TFiP_Terminal_File( Resource._File ).Terminal.Input_Speed 
                        or ( TFiP_Terminal_File( Resource._File ).Terminal.Speed shl 8 ) ;
                end ;
                IOSB.r_io_64.w_status := Write_User( Kernel, PID, p3, sizeof( I64 ), I64 ) ;
                if( IOSB.r_io_64.w_status <> 0 ) then
                begin
                    exit ;
                end ;
            end ;
            if( p4 <> 0 ) then // Return fill counts
            begin
                I64 := TFiP_Terminal_File( Resource._File ).Terminal.CRFill or
                    ( TFiP_Terminal_File( Resource._File ).Terminal.LFFill shl 8 ) ;
                IOSB.r_io_64.w_status := Write_User( Kernel, PID, p4, sizeof( I64 ), I64 ) ;
                if( IOSB.r_io_64.w_status <> 0 ) then
                begin
                    exit ;
                end ;
            end ;
            if( p5 <> 0 ) then // Return parity information
            begin
                I64 := TFiP_Terminal_File( Resource._File ).Terminal.Bits ;
                case TFiP_Terminal_File( Resource._File ).Terminal.Parity of
                    1 : I64 := I64 or TT_M_PARITY or TT_M_ODD ;
                    2 : I64 := I64 or TT_M_PARITY ;
                end ;
                if( TFiP_Terminal_File( Resource._File ).Terminal.Ignore_Errors ) then
                begin
                    I64 := I64 or TT_M_DISPARERR ;
                end ;
                IOSB.r_io_64.w_status := Write_User( Kernel, PID, p5, sizeof( I64 ), I64 ) ;
                if( IOSB.r_io_64.w_status <> 0 ) then
                begin
                    exit ;
                end ;
            end ;
            if( p6 <> 0 ) then // Return type-ahead buffer size
            begin
                I64 := TFiP_Terminal_File( Resource._File ).Terminal._Input_Filter.Buffer_Size ;
                IOSB.r_io_64.w_status := Write_User( Kernel, PID, p6, sizeof( I64 ), I64 ) ;
                if( IOSB.r_io_64.w_status <> 0 ) then
                begin
                    exit ;
                end ;
            end ;
This code is added to the IO_SENSEMODE function processing the FIP.QIO method. It checks each of the parameters p3 through p6 for non-zero values. If any of the values are non-zero, the addresses are used as destinations to write the information for the corresponding data, as described in the documentation.

    if( ( Func and IO_Function_Mask ) = IO_SETCHAR ) then
    begin
        case Mode of
            0 : // Set permanent characteristics
                begin
                    Get_User_Data( Kernel, 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 // Only terminals are supported at present
                    begin
                        IOSB.r_io_64.w_status := SS_ACCVIO ;
                        Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                        exit ;
                    end ;
                    if( TFiP_Terminal_File( Resource._File ).Terminal = nil ) then // Not a terminal
                    begin
                        IOSB.r_io_64.w_status :=  UOSErr_Invalid_Operation ; // Probably ACCVIO in VMS
                        Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                        exit ;
                    end ;
                    if( ( USC.Get_Process_Info( PID, JPI_CURPRIV ) and PHY_IO ) = 0 ) then
                    begin
                        IOSB.r_io_64.w_status := UOSErr_Protection_Violation ;
                        Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                        exit ;
                    end ;
We've updated and expanded the IO_SETCHAR function of QIO to handle the other terminal characteristics. First we get the buffer and exit with an error, as before. Next we check that the device class in the buffer is DC_TERM and exit with an error if not. In the future, we will support other device types for IO_SETCHAR, but for now, only terminals are handled. Next we make sure that the device actually is a terminal. If not, we exit with an error. Finally, we check that the user has the PHY_IO privilege - which is required for the IO_SETCHAR function.

                    TFiP_Terminal_File( Resource._File ).Terminal.Permanent_Terminal_Flags :=
                        Buffer.Characteristics ;
                    if( Buffer.PageWidth <> 0 ) then
                    begin
                        TFiP_Terminal_File( Resource._File ).Terminal.Permanent_Max_Char_Rows :=
                            Buffer.PageWidth ;
                    end ;
                    if( Buffer.PageLength <> 0 ) then
                    begin
                        TFiP_Terminal_File( Resource._File ).Terminal.Permanent_Max_Char_Columns :=
                            Buffer.PageLength ;
                    end ;
                    if( p3 <> 0 ) then // Set speed
                    begin
                        Get_User_Data( Kernel, PID, P3, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        I64 := TFiP_Terminal_File( Resource._File ).Terminal.Speed ;
                        I64 := Convert_to_Baud_Ordinal( I64 ) ;
                        if(
                            ( TFiP_Terminal_File( Resource._File ).Terminal.Speed <> 0 )
                            and
                            ( TFiP_Terminal_File( Resource._File ).Terminal.Speed <> 
                                TFiP_Terminal_File( Resource._File ).Terminal.Input_Speed )
                          ) then
                        begin
                            I64 := TFiP_Terminal_File( Resource._File ).Terminal.Input_Speed 
                                or ( TFiP_Terminal_File( Resource._File ).Terminal.Speed shl 8 ) ;
                        end ;
                    end ;
                    if( p4 <> 0 ) then // Set fill counts
                    begin
                        Get_User_Data( Kernel, PID, P4, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        TFiP_Terminal_File( Resource._File ).Terminal.CRFill := I64 and 255 ;
                        TFiP_Terminal_File( Resource._File ).Terminal.LFFill := I64 shl 8 ;
                    end ;
                    if( p5 <> 0 ) then // Set parity information
                    begin
                        Get_User_Data( Kernel, PID, P5, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        if( ( I64 and TT_M_PARITY ) <> 0 ) then
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Bits := I64 and 15 ;
                            if( ( I64 and TT_M_ODD ) <> 0 ) then
                            begin
                                TFiP_Terminal_File( Resource._File ).Terminal.Parity := Parity_Odd ;
                            end else
                            begin
                                TFiP_Terminal_File( Resource._File ).Terminal.Parity := Parity_Even ;
                            end ;
                        end else
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Parity := Parity_None ;
                        end ;
                        if( ( I64 and TT_M_DISPARERR ) <> 0 ) then
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Ignore_Errors := True ;
                        end ;
                    end ;
                    if( p6 <> 0 ) then // Set type-ahead buffer size
                    begin
                        if( ( USC.Get_Process_Info( PID, JPI_CURPRIV ) and PHY_IO ) = 0 ) then
                        begin
                            // PHY_IO required to set the buffer size...
                            IOSB.r_io_64.w_status := UOSErr_Protection_Violation ;
                            Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                            exit ;
                        end ;
                        Get_User_Data( Kernel, PID, P6, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        if( I64 < 256 ) then
                        begin
                            I64 := 256 ; // Minimum buffer size
                        end ;
                        TFiP_Terminal_File( Resource._File ).Terminal._Input_Filter.Buffer_Size := I64 ;
                    end ;
                end ;
        end ; // case Mode
Next, we update the terminal characteristics. For parameters p3-p6, we essentially do the inverse of the IO_SENSEMODE function, reading the new value from the user memory, unpacking them, and then setting the terminal attributes.

            0 : // Set current 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 // Only terminals are supported at present
                    begin
                        IOSB.r_io_64.w_status := SS_ACCVIO ;
                        Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                        exit ;
                    end ;
                    if( TFiP_Terminal_File( Resource._File ).Terminal = nil ) then // Not a terminal
                    begin
                        IOSB.r_io_64.w_status :=  UOSErr_Invalid_Operation ; // Probably ACCVIO in VMS
                        Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                        exit ;
                    end ;
                    if( USC.Get_Process_Info( PID, JPI_CURPRIV ) and ( PHY_IO or LOG_IO ) = 0 ) then
                    begin
                        if( ( TFiP_Terminal_File( Resource._File ).Terminal.Terminal_Flags and TT2_M_SETSPEED ) = 0 ) then
                        begin
                            // If terminal doesn't have SETSPEED flag set, the user needs either PHY_IO or LOG_IO
                            IOSB.r_io_64.w_status := UOSErr_Protection_Violation ;
                            Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                            exit ;
                        end ;
                    end ;
                    if(
                        ( Buffer.Characteristics and TT2_M_SETSPEED ) <>
                        ( TFiP_Terminal_File( Resource._File ).Terminal.Permanent_Terminal_Flags and TT2_M_SETSPEED )
                      ) then // Changing setspeed flag
                    begin
                        if(
                            ( USC.Get_Process_Info( PID, JPI_CURPRIV )
                            and
                            ( PHY_IO or LOG_IO ) ) = 0 ) then
                        begin
                            IOSB.r_io_64.w_status := UOSErr_Protection_Violation ;
                            Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                            exit ;
                        end ;
                    end ;
This code is added to the IO_SETMODE function in QIO. It is similar to the code for IO_SETCHAR above except for the privilege check. In IO_SETCHAR, we require PHY_IO. Here, we only require privileges if the terminal doesn't have the the SETSPEED flag set. Since we are setting the current settings, rather than the permanent settings, either the PHY_IO or LOG_IO privilege will suffice. The user must also have either PHY_IO or LOG_IO to change the setspeed flag.

                    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 ;
                    if( p3 <> 0 ) then // Set speed
                    begin
                        Get_User_Data( Kernel, PID, P3, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        TFiP_Terminal_File( Resource._File ).Terminal.Input_Speed :=
                            Convert_to_Baud_Rate( I64 and 255 ) ;
                        TFiP_Terminal_File( Resource._File ).Terminal.Speed :=
                            Convert_to_Baud_Rate( I64 shr 8 ) ;
                    end ;
                    if( p4 <> 0 ) then // Set fill counts
                    begin
                        Get_User_Data( Kernel, PID, P4, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        TFiP_Terminal_File( Resource._File ).Terminal.CRFill := I64 and 255 ;
                        TFiP_Terminal_File( Resource._File ).Terminal.LFFill := I64 shl 8 ;
                    end ;
                    if( p5 <> 0 ) then // Set parity information
                    begin
                        Get_User_Data( Kernel, PID, P5, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        if( ( I64 and TT_M_PARITY ) <> 0 ) then
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Bits := I64 and 15 ;
                            if( ( I64 and TT_M_ODD ) <> 0 ) then
                            begin
                                TFiP_Terminal_File( Resource._File ).Terminal.Parity := Parity_Odd ;
                            end else
                            begin
                                TFiP_Terminal_File( Resource._File ).Terminal.Parity := Parity_Even ;
                            end ;
                        end else
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Parity := Parity_None ;
                        end ;
                        if( ( I64 and TT_M_DISPARERR ) <> 0 ) then
                        begin
                            TFiP_Terminal_File( Resource._File ).Terminal.Ignore_Errors := True ;
                        end ;
                    end ;
                    if( p6 <> 0 ) then // Set type-ahead buffer size
                    begin
                        if( ( USC.Get_Process_Info( PID, JPI_CURPRIV ) and PHY_IO ) = 0 ) then
                        begin
                            // PHY_IO required to set the buffer size...
                            IOSB.r_io_64.w_status := UOSErr_Protection_Violation ;
                            Set_Last_Error( Create_Error( IOSB.r_io_64.w_status ) ) ;
                            exit ;
                        end ;
                        Get_User_Data( Kernel, PID, P6, sizeof( I64 ), I64, IOSB.r_io_64.w_status ) ;
                        if( IOSB.r_io_64.w_status = UE_ERROR ) then
                        begin
                            exit ;
                        end ;
                        if( I64 < 256 ) then
                        begin
                            I64 := 256 ; // Minimum buffer size
                        end ;
                        TFiP_Terminal_File( Resource._File ).Terminal._Input_Filter.Buffer_Size := I64 ;
                    end ;
                end ;
This code is almost exactly the same as the IO_SETCHAR code, except that it sets the current settings instead of the permanent settings.

function Convert_to_Baud_Rate( I : integer ) : integer ;

begin
    case I of
        TT_C_BAUD_50 : Result := 50 ;
        TT_C_BAUD_75 : Result := 75 ;
        TT_C_BAUD_110 : Result := 110 ;
        TT_C_BAUD_134 : Result := 134 ;
        TT_C_BAUD_150 : Result := 150 ;
        TT_C_BAUD_300 : Result := 300 ;
        TT_C_BAUD_600 : Result := 600 ;
        TT_C_BAUD_1200 : Result := 1200 ;
        TT_C_BAUD_1800 : Result := 1800 ;
        TT_C_BAUD_2000 : Result := 2000 ;
        TT_C_BAUD_2400 : Result := 2400 ;
        TT_C_BAUD_3600 : Result := 3600 ;
        TT_C_BAUD_4800 : Result := 4800 ;
        TT_C_BAUD_7200 : Result := 7200 ;
        TT_C_BAUD_9600 : Result := 9600 ;
        TT_C_BAUD_19200 : Result := 19200 ;
        TT_C_BAUD_38400 : Result := 38400 ;
        TT_C_BAUD_57600 : Result := 57600 ;
        TT_C_BAUD_76800 : Result := 76800 ;
        TT_C_BAUD_115200 : Result := 115200 ;
        else Result := 0 ;
    end ;
end ;


function Convert_to_Baud_Ordinal( I : integer ) : integer ;

begin
    case I of
        50 : Result := TT_C_BAUD_50 ;
        75 : Result := TT_C_BAUD_75 ;
        110 : Result := TT_C_BAUD_110 ;
        134 : Result := TT_C_BAUD_134 ;
        150 : Result := TT_C_BAUD_150 ;
        300 : Result := TT_C_BAUD_300 ;
        600 : Result := TT_C_BAUD_600 ;
        1200 : Result := TT_C_BAUD_1200;
        1800 : Result := TT_C_BAUD_1800;
        2000 : Result := TT_C_BAUD_2000 ;
        2400 : Result := TT_C_BAUD_2400 ;
        3600 : Result := TT_C_BAUD_3600 ;
        4800 : Result := TT_C_BAUD_4800 ;
        7200 : Result := TT_C_BAUD_7200 ;
        9600 : Result := TT_C_BAUD_9600 ;
        19200 : Result := TT_C_BAUD_19200 ;
        38400 : Result := TT_C_BAUD_38400 ;
        57600 : Result := TT_C_BAUD_57600 ;
        76800 : Result := TT_C_BAUD_76800 ;
        115200 : Result := TT_C_BAUD_115200 ;
        else Result := 0 ;
    end ;
end ;
These two functions convert between the baud rate constants used by VMS and the actual baud rate value used by UOS.

This concludes the changes to support SETTERM. However, with modern computer equipment, it is unlikely that most UOS users will ever need to deal with any terminal characteristics other than turning echo on and off. I'll end with a brief discussion of character echoing.

You might wonder why we have two different terminal characteristics that deal with echoing (LOCALECHO and HALFDUP). Either one will turn off the echo. Why two rather than a single characteristic? They should be viewed this way: HALFDUP indicates whether the terminal echoes its own characters and we don't want to modify this value to turn echo on or off. ECHO/LOCALECHO, on the other hand, indicates whether or not we want to echo characters to the terminal (regardless of the duplex), usually for security purposes. We want to be able to change the echo without altering the HALFDUP setting. If a terminal is half-duplex, turning ECHO back on will still not echo to the terminal. But, if we modified the HALFDUP setting to turn the echo back on and the terminal hardware is in local echo mode, we will essentially cause a doubling of each character (one from the terminal's local echo and one from the UOS echo). To put it another way, HALFDUP tells UOS how the hardware is configured, whereas ECHO indicates what UOS should do, regardless of the duplex setting. Thus, UCL scripts can safely turn the echo on and off via SETTERM/LOCALECHO without causing actual UOS echoing when there shouldn't be any. Admittedly, the opposite of ECHO being LOCALECHO somewhat muddies the terminology (it really should be NOECHO), but such is the cost of using the VMS specification.

In the next article, we will begin our look at the AUTHORIZE utility.

 

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