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:
FlagDescription
LNM_M_CASE_BLINDIf set, TRNLNM doesn't distinguish between upper and lower case characters in the symbol name. By default, TRNLNM matches by case.
LNM_INTERLOCKEDIf 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:
MneumonicDescription
PSL_C_KERNELKernel (executive) mode (ring 0)
PSL_C_DRIVERDriver mode (ring 1)
PSL_C_EXECSynonymous with PSL_C_DRIVER
PSL_C_USERUser mode (ring 3)
PSL_C_SUPERSupervisor 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 offsetByte lengthDescription
02Must be -1.
22Item code.
44Must be -1.
88Length of result buffer.
168Address of result buffer.
248Address where to write result length.

Codes

LNM_ACMODEReturns the access mode associated with the specified symbol.
LNM_ATTRIBUTESReturns the attributes of the specified symbol. The following mnuemonics can be used to compare with these attributes.
LNM_M_CONCEALEDThe matching symbol is a concealed logical.
LNM_M_CONFINEThe symbol will not be copied to a spawned process.
LNM_M_CRELOGThe symbol was created using the CRELOG system service.
LNM_M_EXISTSThe symbol exists.
LNM_M_NO_ALIASThe symbol cannot be defined in the same table with an outer access mode.
LNM_M_TABLEThe symbol is a symbol table name.
LNM_M_CLUSTERWIDEThe symbol was found in a clusterwide table.
LNM_M_TERMINALThe symbol is not subject to recursive translation.
LNM_CHAINThis is ignored by UOS. It is provided for VMS compatibility.
LNM_INDEXDefines 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_LENGTHReturns the length of the equivalence value for the specified symbol.
LNM_MAX_INDEXThe maximum index for equivalence values for the specified symbol.
LNM_NAMEReturns the case-sensitive name of the symbol matching the one specified.
LNM_STRINGReturns the equivalence name (value) for the specified symbol.
LNM_TABLEReturns 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.