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

BRKTHRU

The REPLY utility uses the BRKTHRU system service to send a message to all (or some) terminals. There used to be a BRDCST (broadcast) service in VMS which was superseded by BRKTHRU. The current VMS documentation doesn't even include the old service, other than a passing mention to it. Since we follow the documentation, we don't include the older service.

The following code has TODO comments in regard to carriage control and flags parameters passed to the service. In both cases, this has to do with how messages are displayed on terminals. This is handled by the display management facility, which (as mentioned in previous articles) which we will cover sometime in the future. For now, message text is just spit out without any formatting considerations. Also, we leave the cluster processing for a future set of articles. Finally of note, there is a reference to the SET BROADCAST utility, which we will address at some point in the future. Now, the documentation for BRKTHRU:

BRKTHRU
BRKTHRUW
Breakthrough

Sends a message to one or more terminals. BRKTHRU is asynchonous, BRKTHRUW is synchronous.

Format

SYS$BRKTHRU efn msgbuf sendto sendtyp iosb carcon flags reqid timout astadr astprm

Parameters

efn
A 64-bit integer that contains the event flag that will cause the process to exit a wait state when it is set.
msgbuf
A 64-bit address of the SRB structure defining the message to send.
sendto
A 64-bit address of the SRB structure defining the terminal or user to send the message to. This value may be ignored or required, depending on the value of the sendtyp parameter.
sendtyp
A 64-bit value indicating which terminal(s) should recieve the message. The values are:
ValueMnuemonicDescription
0BRK_C_ALLTERMSMessage is sent to all terminals. Messages are not sent to terminals with AUTOBAUD that are not logged in. sendto is ignored.
1BRK_C_ALLUSERSMessage is sent to all terminals at which a user are logged in.
2BRK_C_DEVICEMessage is sent to the terminal specified in sendto.
3BRK_C_USERNAMEMessage is sent to all terminals at which the user specified by sendto are logged in.
iosb
A 64-bit address of an IOSB structure.
carcon
A 64-bit value indicating carriage control sequence is to follow the message. By default this is 32, indicating that a LF precedes the message text and a CR follows it.
flags
A 64-bit value indicating processing options, as followed.
ValueMnuemonicDescription
0-31-Display 0 to 31 blank lines before the message is shown. Only applies if BRK_M_SCREEN is specified. Note that this is not a bit flag like the other flags, rather the lower 5 bits are treated as an integer indicating the number of lines.
32BRK_M_NOREFRESHDo not redisplay last line or a read operation that was interrupted by the message.
64BRK_M_BOTTOMIf BRK_M_SCREEN is also specified, the message is shown at the bottom of the screen.
128BRK_M_CLUSTEREnable broadcast on other nodes in a cluster. Otherwise the message is sent only terminals on the current node.
256BRK_M_SCREENMessage is sent using display formatting. In the absence of other flags, the message is shown at the top of the display, assuming the display filter supports that operation.
reqid
A 64-bit value indicating the application that is using the service. These correspond to the optional value that can be used with SET BROADCAST=. For instance, SET BROADCAST=NOPHONE would prevent the PHONE utility from sending messages to the terminal.
Class NameDescription
BRK_C_GENERALDefault value.
BRK_C_PHONESent by the PHONE utility.
BRK_C_MAILSent by the MAIL utility.
BRK_C_UCLSent for Control-T status.
BRK_C_QUEUESent by the queue manager.
BRK_C_SHUTDOWNUsed by REPLY/ID=SHUTDOWN.
BRK_C_URGENTUsed by REPLY/ID=URGENT.
BRK_C_USER1 through BRK_C_USER16Reserved for user programs.
timout
A 64-bit value indicating the number of seconds that must elapse before a write to a terminal is considered to have failed. This timeout applies to each individual terminal if the message is sent to multiple terminals. If not specified, or 0, there is no timeout (timeout is infinite). Note, however, that values or 1, 2, 3, and 4 are illegal. The terminal can timeout if it sent an XOFF to the computer, for instance.
astadr
The 64-bit address of a AST service routine to call when the BRKTHRU message has been delivered to the specified terminals. If 0, no callback is made.
astprm
The 64-bit value to pass to the AST service routine when it is called.

Description

BRKTHRU sends a message to one or more terminals. BRKTHRU completes asychronously and returns immediately without waiting for the message to be written. BRKTHRUW is identical to BRKTHRU, but only returns to the caller after the message has been written. The passed message should not be altered until BRKTHRU completes or else what is output to terminals may be changed.
The service operates by assigning a channel (via ASSIGN) to the terminal(s) and then writing to the terminal (via QIO). When calling QIO, the IO_WRITEVBLK function code is used, in addition to IO_M_BREAKTHRU, IO_M_CANCTRLO, and optionally IO_M_REFRESH function modifiers.
If a target terminal has the NOBROADCAST characteristic set for the ID (or all IDs), the operation is skipped for that terminal. In this circumstance, the operation completes without error.
If the terminal is performing a read operation at the time, the read operation is suspended, the message is displayed, and then the line that was being read is redisplayed (as if Control-R was used) and the read operation is resumed.
If the terminal is performing a write operation, the message is displayed after the current write operation completes.
If the BRDCSTMBX flag is set for a terminal, the message is instead sent to the associated mailbox.

Required Privileges

A user is allowed to send a message to a terminal where the same user account is logged in without any privileges. Otherwise, the OPER privilege is required. Note that if directed to all terminals, only those terminals where the same user account is logged in are affected if the user doesn't have OPER.

Required Quotas

The same quotas as are used for any write operation apply to BRKTHRU.

Condition Codes

SS_NORMAL The timer was successfully created.
SS_ACCVIO Values could not be read from the user address space.
SS_BADPARAM An invalid parameter value was passed: null message address, invalid timeout value, reqid contains an invalid valid flag, or sendtyp is not one of the valid values.
SS_EXQUOTA The process has exceeded a quota.
SS_ILLEFC An illegal event flag number was specified.
SS_INSFMEM System resources were exceeded.
SS_NONLOCAL The device is on a different node.
SS_NOOPER The process does not have the OPER privilege and is trying to send a message to a terminal of another user account.
SS_NOSUCHDEV The specified terminal does not exist or is not available for output.
Any other condition codes returned by ASSIGN, QIO, GETJPI, or GETDVI can also be returned.


function SYS_BRKTHRU( efn, msgbuf, sendto, sndtyp, iosb, carcon, flags, reqid,
    timout, astadr, astprm : int64 ) : int64 ;

var Status : int64 ;
    SysRequest : TS2I9_Request ;

begin
    fillchar( SysRequest, sizeof( SysRequest ), 0 ) ;
    SysRequest.Request.Subsystem :=  UOS_Subsystem_FIP ;
    SysRequest.Request.Request := UOS_FIP_BRKTHRU ;
    SysRequest.Request.Length := sizeof( SysRequest ) - sizeof( Sysrequest.Request ) ;
    SysRequest.Request.Status := integer( @Status ) ;
    SysRequest.SRB1 := PSRB( Msgbuf )^ ;
    SysRequest.SRB2 := PSRB( Sendto )^ ;
    SysRequest.Integer1 := EFN ;
    SysRequest.Integer2 := sndtyp ;
    SysRequest.Integer3 := IOSB ;
    SysRequest.Integer4 := carcon ;
    SysRequest.Integer5 := Flags ;
    SysRequest.Integer6 := ReqID ;
    SysRequest.Integer7 := timout ;
    SysRequest.Integer8 := astadr ;
    SysRequest.Integer9 := astprm ;

    Call_To_Ring0( integer( @SysRequest ) ) ;
    Result := Status ;
end ;


function SYS_BRKTHRUW( efn, msgbuf, sendto, sndtyp, iosb, carcon, flags, reqid,
    timout, astadr, astprm : int64 ) : int64 ;

var Status : int64 ;
    SysRequest : TS2I9_Request ;

begin
    fillchar( SysRequest, sizeof( SysRequest ), 0 ) ;
    SysRequest.Request.Subsystem :=  UOS_Subsystem_FIP ;
    SysRequest.Request.Request := UOS_FIP_BRKTHRUW ;
    SysRequest.Request.Length := sizeof( SysRequest ) - sizeof( Sysrequest.Request ) ;
    SysRequest.Request.Status := integer( @Status ) ;
    SysRequest.SRB1 := PSRB( Msgbuf )^ ;
    SysRequest.SRB2 := PSRB( Sendto )^ ;
    SysRequest.Integer1 := EFN ;
    SysRequest.Integer2 := sndtyp ;
    SysRequest.Integer3 := IOSB ;
    SysRequest.Integer4 := carcon ;
    SysRequest.Integer5 := Flags ;
    SysRequest.Integer6 := ReqID ;
    SysRequest.Integer7 := timout ;
    SysRequest.Integer8 := astadr ;
    SysRequest.Integer9 := astprm ;

    Call_To_Ring0( integer( @SysRequest ) ) ;
    Result := Status ;
end ;
These system service wrappers are similar to those we've examined in the past. There are two since once is the asychronous version and one is the synchronous version.

        UOS_FIP_BRKTHRU, UOS_FIP_BRKTHRUW:
            begin
                UE := Enter_System_Call( Request, SReq, PID, MMC,
                    sizeof( TS2I9_Request ) - sizeof( SReq ), Address ) ;
                if( UE <> nil ) then
                begin
                    Set_Last_Error( UE ) ;
                    exit ;
                end ;
                try
                    S2I9_Request := PS2I9_Request( Address ) ;

                    UE := Broadcast( S2I9_Request.Integer1, S2I9_Request.Integer2,
                        S2I9_Request.Integer3, S2I9_Request.Integer4,
                        S2I9_Request.Integer5, S2I9_Request.Integer6,
                        S2I9_Request.Integer7, S2I9_Request.Integer8,
                        S2I9_Request.Integer9, S2I9_Request.SRB1,
                        S2I9_Request.SRB2, SReq.Request = UOS_FIP_BRKTHRUW ) ;
                    Status := 0 ;
                    if( UE <> nil ) then
                    begin
                        Status := UE.Get_Error ;
                    end ;
                    Write_User_int64( Kernel, PID, S2I9_Request.Request.Status,
                        Status ) ;
                finally
                    Exit_System_Call( integer( S2I9_Request ), PID, MMC,
                        sizeof( TS2I9_Request ) - sizeof( SReq ) ) ;
                end ;
                if( UE <> nil ) then
                begin
                    Set_Last_Error( UE ) ;
                    exit ;
                end ;
            end ;
This code is added to the File Processor's .API method. It handles both forms of the service, the only difference being the last parameter of the call to Broadcast. Otherwise, it is similar to all other system service handlers found in this method.

function TUOS_FiP.Broadcast( efn, sendtyp, IOSB, carcon, Flags, Reqid, timout, astadr, 
    astprm : int64 ; Msg, Sendto : TSRB ; Wait : boolean ) : TUnified_Exception ;

var Chars, Chars2 : int64 ;
    Context : int64 ;
    Descriptors : array[ 0..7 ] of TDVI_Descriptor ;
    DevLen : int64 ;
    Devname : string ;
    Have_OPER : boolean ;
    I : integer ;
    _IOSB : TIOSB ;
    Mes : string ;
    PID : TPID ;
    Res : integer ;
    RetLen : int64 ;
    Send : string ;
    R, SRB : TSRB ;
    ReqID_Mask : int64 ;
    S : TUOS_String ;
    St : string ;
    Status : integer ;
    Tag : int64 ;
    TermPID : TPID ;
    UIC, UIC_of_Caller : int64 ;
    US : TUOS_String ;
    Caller_Username, Username : string ;

begin
    // Get and validate the parameters...
    Result := nil ;
    PID := Kernel.PID ;
    US := Get_User_String( Kernel, PID, Msg, Status ) ;
    Mes := '' ;
    if( ( US <> nil ) and ( Status = 0 ) ) then
    begin
        Mes := trim( US.Contents ) ;
    end ;
    US.Free ;
    if( Status <> 0 ) then
    begin
        exit ;
    end ;
    if( Mes = '' ) then
    begin
        exit ; // Null message - do nothing
    end ;
    US := Get_User_String( Kernel, PID, Sendto, Status ) ;
    Send := '' ;
    if( ( US <> nil ) and ( Status = 0 ) ) then
    begin
        Send := trim( US.Contents ) ;
    end ;
    US.Free ;
    if( Status <> 0 ) then
    begin
        exit ;
    end ;
    if( ( efn < 0 ) or ( efn > 128 ) ) then
    begin
        Result := Generate_Exception( UOSErr_Illegal_Event_Flag ) ;
        exit ;
    end ;
    if( ( sendtyp < BRK_C_ALLTERMS ) or ( sendtyp > BRK_C_USERNAME ) ) then
    begin
        Result := Generate_Exception( UOSErr_Bad_Parameter ) ;
        exit ;
    end ;
    if( ( Flags < 0 ) or ( Flags > 511 ) ) then
    begin
        Result := Generate_Exception( UOSErr_Bad_Parameter ) ;
        exit ;
    end ;
    if( ( Reqid < BRK_C_GENERAL ) or ( Reqid > BRK_C_USER16 ) ) then
    begin
        Result := Generate_Exception( UOSErr_Bad_Parameter ) ;
        exit ;
    end ;
    if( ( Sendtyp = BRK_C_DEVICE ) or ( Sendtyp = BRK_C_USERNAME ) ) then
    begin
        if( Send = '' ) then // sendto must be specified for these types
        begin
            Result := Generate_Exception( UOSErr_Bad_Parameter ) ;
            exit ;
        end ;
    end ;
    Have_OPER := ( USC.Get_Process_Info( PID, JPI_CURPRIV ) and OPER ) <> 0 ;
    UIC_of_Caller := Kernel.USC.Get_Process_Info( PID, JPI_UIC ) ;
    Caller_Username := lowercase( UIC_Name( UIC_of_Caller ) ) ;
    if( SendTyp = BRK_C_USERNAME ) then
    begin
        if( Caller_Username <> Send ) then
        begin
            if( not Have_OPER ) then
            begin
                Result := Generate_Exception( UOSErr_OPER_Required ) ;
                exit ;
            end ;
        end ;
    end ;
    if( ReqID > 0 ) then
    begin
        ReqID_Mask := 1 shl ( ReqID - 1 ) ;
    end ;
    //TODO: Handle Flags
This new method of the File Processor handles the broadcasting of text to terminals. First we get and validate the parameters. If the message is null, we exit with no errors. If any of the parameter values are out of range or we cannot read data from the user memory, we exit with an error. We also set up a few variables for later use. Have_OPER is set to true if the current user has the OPER privilege. UIC_of_Caller is set to the caller's UIC, and Caller_Username is set to the caller's account name. Then we make sure that if a target username was specified that it either matches the caller or that the caller has OPER. If not, we exit with a error. Finally, we create a mask based on the ReqID - essentially we convert from a number to a bit offset. Thus, for example, if the reqID is 3, we set the mask to 4 (the 3rd bit, or bit 2, or 22, or 1 shl 2). This will be used in code to follow.

    // Iterate through terminals...
    setlength( Devname, 256 ) ;
    Set_String( Devname, R ) ;
    fillchar( Descriptors, sizeof( Descriptors ), 0 ) ;
    Descriptors[ 0 ].Item_Code := DVI_DEVCHAR ;
    Descriptors[ 0 ].Buffer_Address := int64( @Chars ) ;
    Descriptors[ 0 ].Buffer_Length := sizeof( Chars ) ;
    Descriptors[ 1 ].Item_Code := DVI_PID ;
    Descriptors[ 1 ].Buffer_Address := int64( @TermPID ) ;
    Descriptors[ 1 ].Buffer_Length := sizeof( TermPID ) ;
    Descriptors[ 2 ].Item_Code := DVI_DEVNAM ;
    Descriptors[ 2 ].Buffer_Address := R.Buffer ;
    Descriptors[ 2 ].Buffer_Length := 256 ;
    Descriptors[ 3 ].Item_Code := DVI_OWNUIC ;
    Descriptors[ 3 ].Buffer_Address := int64( @UIC ) ;
    Descriptors[ 3 ].Buffer_Length := sizeof( UIC ) ;
    Descriptors[ 4 ].Item_Code := DVI_DEVNAMLEN ;
    Descriptors[ 4 ].Buffer_Address := int64( @DevLen ) ;
    Descriptors[ 4 ].Buffer_Length := sizeof( DevLen ) ;
    Descriptors[ 5 ].Item_Code := DVI_TAG ;
    Descriptors[ 5 ].Buffer_Address := int64( @Tag ) ;
    Descriptors[ 5 ].Buffer_Length := sizeof( Tag ) ;
    Descriptors[ 6 ].Item_Code := DVI_DEVDEPEND3 ;
    Descriptors[ 6 ].Buffer_Address := int64( @Chars2 ) ;
    Descriptors[ 6 ].Buffer_Length := sizeof( Chars2 ) ;
    Context := 0 ;
    UIC := 0 ;
Next we prepare to iterate through terminals, by setting up a descriptor list that we can use to get information about each terminal to see if it is eligible to receive a message. We also initialize a couple other variables.

    if( Sendtyp = BRK_C_DEVICE ) then // Send to specific terminal
    begin
        Send := lowercase( Send ) ;
        if( ( copy( Send, 1, 4 ) <> 'term' ) and ( copy( Send, 1, 5 ) <> '_term' ) ) then
        begin
            Result := Generate_Exception( UOSErr_Bad_Parameter ) ;
            exit ;
        end ;
        St := send ;
        I := pos( 'term', St ) ;
        St := copy( St, I + 4, length( St ) ) ;
        if( copy( St, length( St ), 1 ) = ':' ) then
        begin
            setlength( St, length( St ) - 1 ) ;
        end ;
        if( ( St = '' ) or ( not trystrtoint( St, I ) ) ) then
        begin
            result := Generate_Exception( UOSErr_Bad_Parameter ) ;
            exit ;
        end ;
        Send_Message( Send ) ;
        if( IOSB <> 0 ) then
        begin
            Status := Write_User( Kernel, PID, IOSB, sizeof( _IOSB ), _IOSB ) ;
            if( Status <> 0 ) then
            begin
                exit ;
            end ;
        end ;
        exit ;
    end ;
If a single device is the target of the message, there is no need to iterate through terminals, so in this case, we validate that the target is a valid terminal specification. Then we call the local Send_Message procedure, then update the IOSB (if an address for one was passed).

    S := Pascal_To_UOS_String( 'term*' ) ;
    RetLen := 0 ;
    while( Kernel.USC.Device_Scan_Context( PID, S, R.Buffer, int64( @RetLen ), 0,
      int64( @Context ), _IOSB ) = 0 ) do
    begin
        Send_Message( copy( UOS_Util.Get_String( R ), 1, RetLen ) ) ;
    end ; // while
    if( IOSB <> 0 ) then
    begin
        Status := Write_User( Kernel, PID, IOSB, sizeof( _IOSB ), _IOSB ) ;
        if( Status <> 0 ) then
        begin
            exit ;
        end ;
    end ;
end ; // TUOS_FiP.Broadcast
If we are sending to (potentially) multiple terminals, we construct a terminal wildcard to use with Device_Scan_Context ("term*"). when we use that routine to iterate through all of the terminals. For each one, we call Send_Message. After the loop, we write the IOSB to the user-supplied address, if one was provided.

    procedure Send_Message( Term : string ) ;

    var F : int64 ;
        Handle : THandle ;
        MB : TMailbox ;
        SRB : TSRB ;
        Status : int64 ;

    begin
        if( Term[ length( Term ) ] <> ':' ) then
        begin
            Term := Term + ':' ;
        end ;
        Set_String( Term, SRB ) ;
        Get_Device_Info( 0, int64( @Descriptors ), High( Descriptors ) - 1, SRB ) ;

        // Check for conditions that prevent broadcast on this device...
        if( ( Chars and TT_M_NOBRDCST ) <> 0 ) then
        begin
            exit ; // Ignore terminals with nobroadcast set
        end ;

        if( not Have_OPER ) then // Check restrictions if no OPER privilege
        begin
            if( UIC <> UIC_Of_Caller ) then
            begin
                exit ; // Different user on this terminal
            end ;
        end ;
        if( TermPID <> 0 ) then // Terminal is logged in or otherwise assigned
        begin
            if( Username <> '' ) then // Filter by username
            begin
                if( lowercase( Username ) <> Caller_Username ) then
                begin
                    exit ; // Different user
                end ;
            end ;
        end else
        if( ( Chars and TT2_M_AUTOBAUD ) <> 0 ) then
        begin
            exit ; // Non-logged-in autobaud terminals are ignored
        end ;
        if( ReqID > 0 ) then
        begin
            if( ( Chars2 and ReqID_Mask ) <> 0 ) then
            begin
                exit ; // This class of message disabled for terminal
            end ;
        end ;
This local routine handles sending the message to the passed terminal. The first thing we do is make sure the terminal specification ends in a colon. If it doesn't then the handle assigned that we do below will attempt to reference a file rather than the terminal device. The rest of the routine can be divided into two parts. The first part checks for the eligibility of the terminal to receive the message. If the terminal has NOBRDCST set, no broadcasts are allowed and we exit. If we are selecting terminals based on username and the username for the process assigned to the terminal doesn't match, we exit. If the UIC for the process assigned to the terminal doesn't match the caller's UIC and doesn't have the OPER privilege, we exit. If not logged in and the terminal has autobaud (TT2_M_AUTOBAUD) set, we exit. Finally, if the ReqID has been specified and the corresponding characteristic flag is set (meaning that the specific message class has been disabled), we exit.

The second set of terminal characteristic flags has a bit for each message class in the low bits, which is why we use the ReqID_Mask value.

        // See if terminal has associated mailbox
        if( Tag <> 0 ) then
        begin
            MB := TMailbox( Tag ) ;
            Term := 'MAILB' + inttostr( Temporary_Mailboxes.Indexof( MB ) ) + ':' ;
        end ;

        // Assign a handle...
        Handle := 0 ;
        Assign_Handle( Term, Handle, 0, '', 0 ) ;
        if( Handle <> 0 ) then // Successfully assigned a handle
        begin
            F := IO_WRITEVBLK or IOM_BREAKTHRU or IOM_CANCTRLO or IOM_CLOSE ;
            if( ( Flags and BRK_M_NOREFRESH ) = 0 ) then
            begin
                F := F or IOM_REFRESH ;
            end ;
            if( timout <> 0 ) then
            begin
                F := F or IOM_TIMED ;
            end ;
            if( Wait ) then
            begin
                QIOW( efn, Handle, F, _IOSB, Astadr, Astprm, Msg.Buffer,
                    Msg.Length, timout, carcon, 0, 0 ) ;
            end else
            begin
                Status := QIO( efn, Handle, F, _IOSB, Astadr, Astprm, Msg.Buffer,
                    Msg.Length, timout, carcon, 0, 0 ) ;
            end ;
            if( Status <> 0 ) then
            begin
                Result := Generate_Exception( Status ) ;
            end ;
        end ;
    end ; // .Send_Message
Next we check to see if there is a mailbox associated with the terminal. This is accomplished with the ASSIGN service, which we'll cover in a future article. In this case, we alter the terminal to be the mailbox device name.

In either case, we assign a handle to the device, create the flags for our QIO call, and then call either QIO or QIOW, depending upon the Wait parameter. If an error is returned, we exit with an error.

    function UIC_Name( UIC : integer ) : string ;

    var IOSB : TIOSB ;
        Len : int64 ;
        SRB : TSRB ;

    begin
        setlength( Result, 256 ) ;
        Len := 0 ;
        Set_String( Result, SRB ) ;
        Kernel.USC.Get_UIC_Name( Kernel, PID, SRB, UIC, int64( @Len ), IOSB ) ;
        setlength( Result, Len ) ;
    end ;
This local helper function is used to get the username for the passed UIC.

There were are few other changes needed in various places to support this. The first being the addition of the following line to the TMailbox class to provide a link to the associated terminal (if one):

Terminal : TTerminal ;

We also added a Tag integer variable to the TTerminal class to keep track of the associated mailbox from that side. The next change was to add a set of secondary terminal characteristics to hold the broadcast-by-message-class disable bits. There weren't enough bits left over in the primary characteristics, hence the need for another set. Then we needed a way to obtain them and also the terminal's tag (ie associated mailbox).

                    DVI_DEVDEPEND3 : begin
                                         Res := 0 ;
                                         if( Device.Terminal <> nil ) then
                                         begin
                                             Res := Device.Terminal.Terminal_Flags2 ;
                                         end ;
                                         if( Write_Integer( Res ) = UE_Error ) then
                                         begin
                                             exit ;
                                         end ;
                                     end ;
                    DVI_DEVNAMLEN : begin
                                        Res := system.length( This_Device_Name ) ;
                                        if( Write_Integer( Res ) = UE_Error ) then
                                        begin
                                            exit ;
                                        end ;
                                    end ;
                    DVI_TAG : begin
                                  Res := 0 ;
                                  if( Device.Terminal <> nil ) then
                                  begin
                                      Res := Device.Terminal.Tag ;
                                  end ;
                                  if( Write_Integer( Res ) = UE_Error ) then
                                  begin
                                      exit ;
                                  end ;
                              end ;
Three new item codes are added to the GetDVI service. DVI_DEVDEPEND3 returns the secondary characteristics from a terminal (0 returned if not a terminal). DVI_DEVNAMLEN returns the length of a device's name. DVI_TAG returns the tag value associated with a terminal. All of these are 64-bit integer values.

    end else
    if( ( Func and IO_Function_Mask ) = IO_WRITEVBLK ) then 
    begin
        if( ( Mode and IOM_TIMED ) <> 0 ) then
        begin
            Timeout := p3 ;
        end ;
        Request := Create_Request( Handle, 1, p1, p2 ) ;
        Pending_IOs.Add( Request ) ;
        Result := int64( Request ) ;
    end ;
This code is added to the end of the QIO method of the File Processor. It handles the IO_WRITEVBLK function for terminals. It simply sets up the output request. Note that, unlike VMS, the UOS version allows for a timeout to be associated with this request.

    if( Terminal <> nil ) then
    begin
        Terminal.Tag := 0 ; // Disassociate mailbox from terminal
    end ;
This code is added to the end of the TMailbox destructor to disassociate the mailbox with a terminal, if there is the association. Obviously, we don't want a non-existant mailbox to be associated with a terminal. Normally, this deassignment would be specifically requested, but we make sure to do it here just in case it wasn't.

In the next article, we will look at OPCOM.