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:
Value | Mnuemonic | Description |
0 | BRK_C_ALLTERMS | Message is sent to all terminals. Messages
are not sent to terminals with AUTOBAUD that are not logged in. sendto is ignored. |
1 | BRK_C_ALLUSERS | Message is sent to all terminals at which
a user are logged in. |
2 | BRK_C_DEVICE | Message is sent to the terminal specified in sendto. |
3 | BRK_C_USERNAME | Message 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.
Value | Mnuemonic | Description |
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. |
32 | BRK_M_NOREFRESH | Do not redisplay last line or a read operation that was interrupted by the message. |
64 | BRK_M_BOTTOM | If BRK_M_SCREEN is also specified, the message is shown at the bottom of the screen. |
128 | BRK_M_CLUSTER | Enable broadcast on other nodes in a cluster. Otherwise the message is sent only terminals on the current node. |
256 | BRK_M_SCREEN | Message 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 Name | Description |
BRK_C_GENERAL | Default value. |
BRK_C_PHONE | Sent by the PHONE utility. |
BRK_C_MAIL | Sent by the MAIL utility. |
BRK_C_UCL | Sent for Control-T status. |
BRK_C_QUEUE | Sent by the queue manager. |
BRK_C_SHUTDOWN | Used by REPLY/ID=SHUTDOWN. |
BRK_C_URGENT | Used by REPLY/ID=URGENT. |
BRK_C_USER1 through BRK_C_USER16 | Reserved 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.
|