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
|
SYS_TRNLNM, Part 2
We introduced the TRNLNM system service back in article 68.
In this article, we will flesh out the rest of its features. Here is the definition
of the service:
TRNLNM returns information about a symbol name.
Format
TRNLNM flags table name acmod itemlst
Arguments
flags
Address of 64-bit flag to control the operation of TRNLNM. If this is 0, no flags
are considered to be set. The following flags can be specified:
Flag | Description |
LNM_M_CASE_BLIND | If set, TRNLNM doesn't distinguish between upper and lower case characters in the symbol name. By default, TRNLNM matches by case. |
LNM_INTERLOCKED | If set, TRNLNM will wait to complete until any current clusterwide symbols are complete. |
table
The address of an SRB structure pointing to the name of a symbol table. If this is
0 or the symbol table name is null, the process, job, group, system, and cluster symbol
tables are searched, in that order.
name
Address of an SRB structure that points to the symbol name to translate.
acmode
The address of the 64-bit access mode to be used in the translation. If this is 0,
the access mode is assumed to the outermost mode (USER). When provided, all symbols
with an access mode less privileged than this are ignored. The valid access modes
are:
Mneumonic | Description |
PSL_C_KERNEL | Kernel (executive) mode (ring 0) |
PSL_C_DRIVER | Driver mode (ring 1) |
PSL_C_EXEC | Synonymous with PSL_C_DRIVER |
PSL_C_USER | User mode (ring 3) |
PSL_C_SUPER | Supervisor mode (ring 2) |
itemlst
A pointer to a descriptor list containing items defining what the service should return.
A descriptor list is an array of descriptors with the following layout:
Byte offset | Byte length | Description |
0 | 2 | Must be -1. |
2 | 2 | Item code. |
4 | 4 | Must be -1. |
8 | 8 | Length of result buffer. |
16 | 8 | Address of result buffer. |
24 | 8 | Address where to write result length. |
Codes
LNM_ACMODE | Returns the access mode associated with the specified symbol. |
LNM_ATTRIBUTES | Returns the attributes of the specified symbol. The following
mnuemonics can be used to compare with these attributes.
LNM_M_CONCEALED | The matching symbol is a concealed logical. |
LNM_M_CONFINE | The symbol will not be copied to a spawned process. |
LNM_M_CRELOG | The symbol was created using the CRELOG system service. |
LNM_M_EXISTS | The symbol exists. |
LNM_M_NO_ALIAS | The symbol cannot be defined in the same table with an outer access mode. |
LNM_M_TABLE | The symbol is a symbol table name. |
LNM_M_CLUSTERWIDE | The symbol was found in a clusterwide table. |
LNM_M_TERMINAL | The symbol is not subject to recursive translation. |
|
LNM_CHAIN | This is ignored by UOS. It is provided for VMS compatibility. |
LNM_INDEX | Defines which equivalence value is to be returned for following
descriptors. If not specified, index 0 is used. This can be used to return several
equivalence values by alternating LNM_INDEX and LNM_STRING items, changing the index
with each one. |
LNM_LENGTH | Returns the length of the equivalence value for the specified symbol. |
LNM_MAX_INDEX | The maximum index for equivalence values for the specified symbol. |
LNM_NAME | Returns the case-sensitive name of the symbol matching the one specified. |
LNM_STRING | Returns the equivalence name (value) for the specified symbol. |
LNM_TABLE | Returns the name of the table containing the symbol name being translated. |
Description
The TRNLNM system service returns information about the given symbol name.
Condition Codes
SS_NORMAL SS_BADPARAM SS_BUFFEROVF SS_IVLOGTAB SS_NOLOGNAM
function SYS_TRNLNM( attr : int64 ; tabnam, lognam : int64 ;
acmode, itmlst : int64 ) : int64 ;
var Status : int64 ;
SysRequest : TS2I3_Request ;
STabName, SLogName : PSRB ;
begin
Status := 0 ;
if( Lognam = 0 ) then
begin
Result := SS_BADPARAM ;
exit ;
end ;
STabName := PSRB( pointer( Tabnam ) ) ;
SLogName := PSRB( pointer( Lognam ) ) ;
fillchar( SysRequest, sizeof( SysRequest ), 0 ) ;
SysRequest.Request.Subsystem := UOS_Subsystem_SSC ;
SysRequest.Request.Request := UOS_SSC_TRNLNM ;
SysRequest.Request.Length := sizeof( SysRequest ) - sizeof( TSystem_Request ) ;
SysRequest.Request.Status := integer( @Status ) ;
SysRequest.SRB1.Buffer := STabname.Buffer ;
SysRequest.SRB1.Length := STabname.Length ;
SysRequest.SRB2.Buffer := SLogName.Buffer ;
SysRequest.SRB2.Length := SLogName.Length ;
SysRequest.Integer1 := Attr ;
SysRequest.Integer2 := AcMode ;
SysRequest.Integer3 := ItmLst ;
Call_To_Ring0( integer( @SysRequest ) ) ;
Result := Status ;
end ; // SYS_TRNLNM
We've made one small change to this function: we check for a missing symbol name
and return an error in that case. The TRNLNM system service also checks for this
condition, but by doing it here, we avoid a call across rings for this error.
function TUOS_SSC.TRNLNM( Tabname, SymName : TUOS_string ;
Attr, AcMode, ItemLst : int64 ) : int64 ;
var aName : PChar ;
Buff : PAnsiChar ;
CS : boolean ;
Descriptor : TSYS_Descriptor ;
L : longint ;
Flags, I, Index, Indexes, Len, Mode : int64 ;
Offset : int64 ;
P : PChar ;
PID : TPID ;
Res : int64 ;
Status : longint ;
S, SName : string ;
Table : TUOS_Symbol_Table ;
Tname : string ;
begin
// Setup...
Result := SS_NORMAL ;
PID := Kernel.PID ;
if( ( SymName = nil ) or ( SymName.Contents = '' ) ) then
begin
Generate_Exception( UOSErr_Missing_Name ) ;
Result := SS_BADPARAM ;
exit ;
end ;
SName := SymName.Contents ;
CS := ( Attr and LNM_M_CASE_BLIND ) <> 0 ;
We've made several changes to this routine since we last looked at it, so we'll cover
almost the entire function. In the set up, we set the CS boolean based on the
LNM_M_CASE_BLIND flag, and we pull the symbol name from the parameter.
// Determine table...
if( TabName = nil ) then
begin
TName := '' ;
end else
begin
TName := TabName.Contents ;
end ;
if( TName = '' ) then
begin
Table := Resolve_Table( 0, PID, SName, CS, AcMode ) ;
end else
begin
Table := Get_Symbol_Table( TName ) ;
end ;
if( Table = nil ) then
begin
Generate_Exception( UOSErr_Resource_Not_Found ) ;
Result := SS_IVLOGTAB ;
exit ;
end ;
This code is largely the same, with the exception of returning the condition code, and
the extra parameters for the symbol table functions.
// Get value (in S), and other symbol information...
Flags := 0 ;
S := '' ;
if( Table.Exists( PAnsiChar( SName ), CS, AcMode ) ) then
begin
P := Table.Info( PAnsiChar( SName ), L, CS, AcMode, Index, Flags, Mode, Indexes, aName ) ;
Flags := Flags or LNM_M_EXISTS ;
setlength( S, L ) ;
move( P[ 0 ], PChar( S )[ 0 ], L ) ;
end else
begin
Result := SS_NOLOGNAM ;
end ; // if( Table.Exists( SName ) )
We obtain all the information about the symbol, or set the condition code if the symbol
wasn't found. If found, we add the LNM_M_EXISTS flag to the symbol flags.
LNM_LENGTH:
begin
Res := length( S ) ;
if( Len > sizeof( int64 ) ) then
begin
Len := sizeof( int64 ) ;
end ;
Status := Write_User( Kernel, PID, Descriptor.Buffer_Address, Len, Res ) ;
if( Status = 0 ) then
begin
Status := Write_User( Kernel, PID, Descriptor.Return_Length_Address,
sizeof( int64 ), Len ) ;
end ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
The iteration of descriptors hasn't changed, so we'll simply look at the items we've
added to the processing.
This item returns the length of the symbol value, writing it and it's length to the
user memory and exiting if there was an error.
LNM_ACMODE:
begin
if( Len > sizeof( int64 ) ) then
begin
Len := sizeof( int64 ) ;
end ;
Status := Write_User( Kernel, PID, Descriptor.Buffer_Address, Len, Mode ) ;
if( Status = 0 ) then
begin
Status := Write_User( Kernel, PID, Descriptor.Return_Length_Address,
sizeof( int64 ), Len ) ;
end ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
This item returns the access mode, writing it to the user memory and exiting on error.
LNM_CHAIN: ; // Ignore these
LNM_INDEX:
begin
Index := Get_User_Integer( Kernel, PID, Descriptor.Buffer_Address, Status ) ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
We simply skip past any LNM_CHAIN item, which does nothing in UOS. In
the case of LNM_INDEX , we read the index from the user's memory.
LNM_MAX_INDEX:
begin
if( Len > sizeof( int64 ) ) then
begin
Len := sizeof( int64 ) ;
end ;
Status := Write_User( Kernel, PID, Descriptor.Buffer_Address, Len, Indexes ) ;
if( Status = 0 ) then
begin
Status := Write_User( Kernel, PID, Descriptor.Return_Length_Address,
sizeof( int64 ), Len ) ;
end ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
This item returns the maximum index of the equivalence names for the symbol - writing
it to the user memory and exiting if there is an error.
LNM_TABLE:
begin
S := ansistring( Table.Name ) ;
if( S = '$' + inttostr( PID ) ) then
begin
S := 'LNM$PROCESS' ;
end ;
if( Len > length( S ) ) then
begin
Len := length( S ) ;
end ;
if( Len < length( S ) ) then
begin
Result := SS_BUFFEROVF ;
end ;
Status := Write_User( Kernel, PID, Descriptor.Buffer_Address, Len,
PChar( S )[ 0 ] ) ;
if( Status = 0 ) then
begin
Status := Write_User( Kernel, PID, Descriptor.Return_Length_Address,
sizeof( int64 ), Len ) ;
end ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
This item returns the name of the table in which the symbol was found, and the length
of that name. These are written to the user memory and we exit if either of those
writes fail. We also set the buffer override condition code, if the length of the
target buffer is less than the table name. Note that the table name is converted to
"LNM$PROCESS" if it is the main process symbol table. The actual name of the main
process symbol table is a dollar sign followed by the process ID.
LNM_TABLELENGTH:
begin
S := ansistring( Table.Name ) ;
if( S = '$' + inttostr( PID ) ) then
begin
S := 'LNM$PROCESS' ;
end ;
if( Len > sizeof( int64 ) ) then
begin
Len := sizeof( int64 ) ;
end ;
I := length( S ) ;
Status := Write_User( Kernel, PID, Descriptor.Buffer_Address, Len, I ) ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
This is not an item supported by VMS. We add it to UOS because table names are not
limited in length, as they are on VMS. We write the table name length to the user
memory and exit if there is an error.
LNM_NAME:
begin
S := ansistring( aName ) ;
if( Len > length( S ) ) then
begin
Len := length( S ) ;
end ;
if( Len < length( S ) ) then
begin
Result := SS_BUFFEROVF ;
end ;
Status := Write_User( Kernel, PID, Descriptor.Buffer_Address, Len,
PChar( S )[ 0 ] ) ;
if( Status = 0 ) then
begin
Status := Write_User( Kernel, PID, Descriptor.Return_Length_Address,
sizeof( int64 ), Len ) ;
end ;
if( Status = UE_Error ) then // Error reading user value
begin
exit ;
end ;
end ;
This item is also UOS-specific. This returns the case-sensitive name of the found
symbol. This is useful if the symbol lookup was case-insensitive. We write the name
and length to the user memory and exit if there was an error. If the result buffer
is too small to hold the name, we set the overflow condition (but we don't exit).
In the next article, we will examine the next lexical function.
Copyright © 2020 by Alan Conroy. This article may be copied
in whole or in part as long as this copyright is included.
|