| Document revision date: 19 July 1999 | |
![]() |
|
|
|
| Previous | Contents | Index |
The Create service constructs a new file according to the attributes you specify in the FAB for the file, whereas the Open service makes an existing file available for processing by your program. Both of these services, invoked by the $CREATE and $OPEN macros respectively, allocate resources within the system to establish access (a path) to a file. You must open or create a file to perform most file operations and any record operations on that file. Applications designed for shared access must declare the type of sharing at this time. The user specifies the various types of shared access by setting bits in the file access control (FAB$B_FAC) and share (FAB$B_SHR) fields in the appropriate FAB.
RMS provides several file-processing options for the Create service. The create-if option (FAB$V_CIF option in the FAB$L_FOP field) requests that the file be created only if it does not exist. If the file does exist in the specified directory, the file is opened, not created. The Open and Create services both establish access to the desired file, but the Create service additionally allocates disk space and performs functions related to allocation.
When you are finished processing a file, you invoke the Close service
($CLOSE macro) to close the file, disconnect all record streams
associated with the file, and free all resources allocated to the file.
If you do not explicitly invoke the Close service when the program
image exits, RMS attempts an implicit close. All resources associated
with open files are returned when the files are deaccessed at image
rundown time. However, process permanent files are not implicitly
closed when an image exits. These are special files that the current
CLI opens outside the context of a normal image.
B.3.2 Example of Opening and Creating Files
Example B-2 illustrates the use of the Open, Create, Connect, Get, Put, and Close services to access and copy records from one file to another. Note that the arguments to the $FAB and $RAB macros are listed vertically on separate lines for ease in reading them. However, the argument list must be contiguous and a common programming error is omission of required delimiters and continuation characters when the arguments are listed in this manner.
| Example B-2 Use of the Create, Open, and Close Services | |||
|---|---|---|---|
.TITLE COPYFILE
;
; This program copies the input file to the output file.
;
.PSECT DATA,WRT,NOEXE
INFAB: $FAB FNM = <INFILE:>,- ; Primary input file name
DNM = <.INV> ; Default input file type
INRAB: $RAB FAB = INFAB,- ; Pointer to FAB
ROP = RAH,- ; Read-ahead option
UBF = REC_BUFF,- ; Record buffer
USZ = REC_SIZE ; and size
OUTFAB: $FAB FNM = <OUTFILE:>,- ; Primary output file name
DNM = <.INV>,- ; Default output file name
FOP = CTG,- ; Make contiguous file
FAC = <PUT>,- ; Open for PUT operations
SHR = <NIL>,- ; Exclusive file access
MRS = REC_SIZE,- ; Maximum record size
RAT = CR ; Implied carriage control
OUTRAB: $RAB FAB = OUTFAB,- ; Pointer to FAB
ROP = WBH,- ; Write-behind option
RBF = REC_BUFF ; Output uses same buffer
; as input
;
REC_SIZE = 132 ; Maximum record size
REC_BUFF:
.BLKB REC_SIZE ; Record buffer
.PSECT CODE,NOWRT,EXE
;
; Initialization - Open input and output files and connect streams
;
.ENTRY COPYFILE,^M<R6> ; Save R6
$OPEN FAB=INFAB ; Open input file
BLBC R0,EXIT1 ; Quit on error
$CONNECT RAB=INRAB ; Connect to input
BLBC R0,EXIT2 ; Quit on error
MOVL INFAB+FAB$L_ALQ,- ; Set proper size for output
OUTFAB+FAB$L_ALQ
$CREATE FAB=OUTFAB ; Create output file
BLBC R0,EXIT3 ; Quit on error
$CONNECT RAB=OUTRAB ; Connect to output
BLBS R0,READ ; Branch to READ loop
BRB EXIT4 ; Trap error
EXIT1: MOVAL INFAB,R6 ; Error: Keep FAB address
BRB F_ERR ; Signal file error
EXIT2: MOVAL INRAB,R6 ; Keep RAB address
BRB R_ERR ; Signal record error
EXIT3: MOVAL OUTFAB,R6 ; Keep FAB address
BRB F_ERR ; Signal record error
EXIT4: MOVAL OUTRAB,R6 ; If error, retain RAB addr.
BRB R_ERR ; Signal record error
;
; Copy records loop
;
READ: $GET RAB=INRAB ; Get a record
BLBS R0,WRITE ; Write the record
CMPL R0,#RMS$_EOF ; Was error end-of-file?
BEQL DONE ; Successful completion
BRB EXIT2 ; Error otherwise
WRITE: MOVW INRAB+RAB$W_RSZ, - ; Input RAB sets record
OUTRAB+RAB$W_RSZ ; size for output RAB
$PUT RAB=OUTRAB ; Write the record
BLBC R0,EXIT4 ; Quit on error
BRB READ ; Go back for more
;
; Close files, signal any errors, and exit
;
F_ERR: PUSHL FAB$L_STV(R6) ; Push STV and STS of FAB
PUSHL FAB$L_STS(R6) ; on the stack
CALLS #2, G^LIB$SIGNAL ; Signal error
BRB EXIT
R_ERR: PUSHL RAB$L_STV(R6) ; Push STV and STS of RAB
PUSHL RAB$L_STS(R6) ; on the stack
CALLS #2, G^LIB$SIGNAL ; Signal error
DONE: $CLOSE FAB=INFAB ; Close input
$CLOSE FAB=OUTFAB ; and output
EXIT: RET ; Return with status in R0
.END COPYFILE
|
This example illustrates how you can use the sequential file organization to create a new file by copying records from an existing file. The newly created file and the source file have variable-length records.
This example assumes that an external program has identified the input file as a search list logical name using this statement:
$ ASSIGN [INV]30JUN85,[INV.OLD]30JUN85 INFILE: |
The program also specifies the default file type .INV for the input file using this statement:
DNM=<.INV> ; Default input file name |
Next the program configures the RAB used for the input file (labeled INRAB). The first argument links the RAB to the associated FAB (INFAB) and this is the only required argument to a RAB. The rest of the arguments specify the read-ahead option (described in later text) and the record buffer for the input file. The Get service uses the user record buffer address (UBF) field and the user record buffer size (USZ) field as inputs to specify the record buffer and the record size, respectively.
When you invoke the Get service, RMS takes control of the record buffer and may modify it. RMS returns the record size and only guarantees the contents from where it accessed the record to the completion of the record. |
The program then configures the FAB for the output file. The first argument uses the FNM field to equate the file name to the externally defined logical name OUTFILE. After the program specifies the default file specification extension for the output file, it specifies three additional FAB fields.
First it directs RMS to allocate contiguous space for the output file by setting the CTG bit in the FAB$L_FOP field of the FAB.
Next the program uses a program-defined variable to store the value 132 in the MRS field:
MRS=REC_SIZE REC_SIZE= 132 |
RAT=CR |
Because the program alternately reads and then writes each record, the input file and the output file may share the same buffer. However, because the Put service does not have access to the UBF and UBZ fields, the output RAB defines the buffer using the RBF and the RSZ fields.
Note that the UBF, USZ, and RBF values are set prior to run time, but that the RSZ value is set at run time, just prior to invocation of the Put service. This is done because the input file contains variable-length records and the Put service relies on the Get service to supply each record's size by way of the RSZ field, an INRAB output field.
The following statement from the sample program illustrates this feature:
WRITE: MOVW INRAB+RAB$W_RSZ, - ; Input RAB sets record
OUTRAB+RAB$W_RSZ ; size for output RAB
|
The run-time processing macros for the input file consist of a $OPEN, a $CONNECT, a $GET, and a $CLOSE macro. Because the input file already exists, the program accesses it with a $OPEN macro. The sole argument to this macro identifies the FAB to the Open service:
$OPEN FAB=INFAB |
Next, the program connects a record stream to the input file by calling the Connect service and specifying INRAB as the appropriate RAB:
$CONNECT RAB=INRAB |
Note that upon completion of each service call, the program tests the condition value in R0 returned by the service before proceeding to the next call. If the call fails, the program exits with the appropriate control block address in R6.
After creating the output file and establishing its record stream, the program begins a processing loop in which the Get service reads a record from the input file and the Put service writes the record to the output file. When all file records are copied, as indicated by the detection of the end of the file, the program exits to label DONE, which closes both files.
The Close service disconnects the record stream for all RABs connected
to the specified FAB. In a multistream environment (more than one RAB
can be connected to a single FAB), a program may disconnect individual
record streams using the Disconnect service.
B.3.3 Example of Creating a Multiple-Key Indexed File
Example B-3 creates an indexed file on a remote node from a sequential file on the local node. The indexed file contains three keys: a segmented primary key and two simple alternate keys. The segmented primary key includes the customer's last name, the first letter of the customer's first name, and the customer's middle initial.
| Example B-3 Use of the Create Service for an Indexed File | |||
|---|---|---|---|
.TITLE CREATEIDX - CREATE INDEXED FILE
.IDENT /V001/
;
; This program creates an indexed file with three keys from a
; sequential file containing a name and address list. The record
; format of the input file is shown below:
;
; First Name Column 00-10
; Middle Initial Column 11-11
; Last Name Column 12-26
; Street Column 27-46
; City Column 47-58
; State Column 59-60
; Zip Code Column 61-65
; Reserved for
; new data Column 66-end of record
;
; The input and output files are specified by the logical names SRC
; and DST, respectively. For example:
;
; $ DEFINE SRC DBB1:[TEST]INPUT.DAT
; $ DEFINE DST TRNTO::DRA4:[RMS.FILES]OUTPUT.DAT
; $ RUN CREATEIDX
;
;********************************************************************
.SBTTL Control block and buffer storage
.PSECT DATA NOEXE,LONG
;
; Define the source file FAB and RAB control blocks.
;
SRC_FAB:
$FAB FAC=<GET>,- ; File access for GET only
FOP=<SQO>,- ; DAP file transfer mode
FNM=<SRC:> ; Name of input file
SRC_RAB:
$RAB FAB=SRC_FAB,- ; Address of associated FAB
RAC=SEQ,- ; Sequential record access
UBF=BUFFER,- ; Buffer address
USZ=BUFFER_SIZE ; Buffer size
;
; Define the destination file FAB and RAB control blocks.
;
DST_FAB:
$FAB FAC=<PUT>,- ; File access for PUT only
FOP=CTG,- ; Allocate contiguous
SHR = <NIL>,- ; Exclusive file access
FNM=<DST:>,- ; Name of output file
MRS=128,- ; Maximum record size
RFM=VAR,- ; Variable length records
RAT=<CR>,- ; Implied carriage control
ORG=IDX,- ; Indexed file organization
XAB=DST_KEY0 ; Address of start of XAB chain
DST_RAB:
$RAB FAB=DST_FAB,- ; Address of associated FAB
MBF=3,- ; Use 3 buffers
RAC=KEY,- ; Random record writes
RBF=BUFFER,- ; Buffer address
ROP=LOA,- ; Specify initial fill size
RSZ=BUFFER_SIZE ; Buffer size
;
; Define a key definition XAB for the primary key.
;
DST_KEY0: ; Primary key is Name
$XABKEY REF=0,- ; Key reference number
DAN=0,- ; Define data XABALL
DFL=1536,- ; Define data fill of 75%
FLG=<DUP>,- ; Allow duplicate keys
DTP=DSTG,- ; Descending sort order
IAN=1,- ; Define index XABALL
IFL=1536,- ; Initial index fill 75%
PROLOG=3,- ; Request prolog 3
POS=<12,0,11>,- ; Key segment positions
SIZ=<15,1,1>,- ; Key segment lengths
NXT=DST_KEY1 ; Address of next XAB in chain
;
; Define key definition XABs for the alternate keys.
;
DST_KEY1: ; 1st alternate key is City
$XABKEY REF=1,- ; Key reference number
DAN=2,- ; Data level (SIDR) XABALL
IAN=2,- ; Index XABALL
IFL=768,- ; Initial index fill 75%
POS=47,- ; Starting key position
SIZ=12,- ; Key size
FLG=<CHG,DUP>,- ; Duplicates and changes
NXT=DST_KEY2 ; Address of next XAB in chain
DST_KEY2: ; 2nd alternate key is State
$XABKEY REF=2,- ; Key reference number
DAN=2,- ; Data level (SIDR) XABALL
IAN=2,- ; Index XABALL
IFL=768,- ; Initial index fill 75%
POS=59,- ; Starting key position
FLG=<CHG,DUP>,- ; Duplicates and changes
SIZ=2,- ; Key size
NXT=DST_ALL0 ; Designate next XAB
;
; Define allocation control XABs to define multiple areas
;
DST_ALL0:
$XABALL AID=0,- ; Data area definition
ALQ=328,- ; Allocation quantity and
AOP=<CBT>,- ; contiguous best try
BKZ=4,- ; Bucket size of 4 blocks
DEQ=112,- ; Default extension quantity
NXT=DST_ALL1 ; Designate next XAB
DST_ALL1:
$XABALL AID=1,- ; Primary key index area
ALQ=8,- ; Allocation quantity and
AOP=<CBT>,- ; contiguous best try
BKZ=4,- ; Bucket size of 4 blocks
DEQ=4,- ; Default extension quantity
NXT=DST_ALL2 ; Designate next XAB
DST_ALL2:
$XABALL AID=2,- ; Alternate key data area
ALQ=112,- ; Allocation quantity and
AOP=<CBT>,- ; contiguous best try
BKZ=2,- ; Bucket size of 2 blocks
DEQ=38,- ; Default extension quantity
NXT=0 ; No more XABs
;
; Allocate buffer to the size of the largest record being read.
;
BUFFER: .BLKB 66 ; Buffer for input and output
BUFFER_SIZE=.-BUFFER ; Buffer size
;*********************************************************************
.SBTTL Mainline
.PSECT CODE NOWRT,BYTE
;
; Start of program
;
.ENTRY CREATEIDX,^M<R6> ; Entry point
;
; Open the source and destination files.
;
$OPEN FAB=SRC_FAB ; Open input file
BLBC R0,EXIT1 ; Branch on failure
$CONNECT RAB=SRC_RAB ; Connect input record stream
BLBC R0,EXIT2 ; Branch on failure
$CREATE FAB=DST_FAB ; Create output file
BLBC R0,EXIT3 ; Branch on failure
$CONNECT RAB=DST_RAB ; Connect output record stream
BLBC R0,EXIT4 ; Branch on failure
BRB LOOP ; Bypass signaling code
EXIT1: MOVAL SRC_FAB,R6 ; Keep FAB address
BRB F_ERR ; Signal error
EXIT2: MOVAL SRC_RAB,R6 ; Keep RAB address
BRB R_ERR ; Signal error
EXIT3: MOVAL DST_FAB,R6 ; Keep FAB address
BRB F_ERR ; Signal error
EXIT4: MOVAL DST_RAB,R6 ; Keep RAB address
BRB R_ERR ; Signal error
;
; Transfer records until end-of-file is reached.
;
LOOP: $GET RAB=SRC_RAB ; Read next rec from input file
BLBS R0,PUT ; Branch on success
CMPL R0,#RMS$_EOF ; Was it end-of-file (EOF)?
BNEQ EXIT2 ; Branch if not EOF error
BRB CLOSE ; Close and exit if EOF
PUT: $PUT RAB=DST_RAB ; Write 66-byte record to output
BLBS R0,LOOP ; On success, continue loop
BRB EXIT4 ; On error, signal and exit
;
; Close the source and destination files.
;
F_ERR: PUSHL FAB$L_STV(R6) ; Push STV and STS fields
PUSHL FAB$L_STS(R6) ; on stack
CALLS #2, G^LIB$SIGNAL ; Signal file error
BRB EXIT ; Exit
R_ERR: PUSHL RAB$L_STV(R6) ; Push STV and STS fields
PUSHL RAB$L_STS(R6) ; on stack
CALLS #2, G^LIB$SIGNAL ; Signal file error
CLOSE: $CLOSE FAB=DST_FAB ; Close output file
$CLOSE FAB=SRC_FAB ; Close input file
EXIT: $EXIT_S ; Exit
.END CREATEIDX ; Specify starting address
|
This example program creates an indexed file with a primary key and two alternate keys that are defined by appropriate key definition control blocks (XABKEY). For efficiency, the file is divided into areas consisting of a data area and an index area for each key using multiple allocation control blocks (XABALL).
In each XABKEY, the DAN and IAN arguments (XAB$B_DAN and XAB$B_IAN fields) indicate the area identification number (AID) of the corresponding XABALL. By setting the RAB$V_LOA bit in RAB field RAB$L_ROP, the program directs RMS to use the DFL and IFL arguments (XAB$W_DFL and XAB$W_IFL fields) to determine the maximum initial fill size (in bytes) for data and index buckets (each bucket contains the number of blocks specified in the XABALL BKZ argument, XAB$B_BKZ field).
These are the XABKEY and XABALL control blocks for the primary key (the NAME key) in this example:
DST_KEY0: ; Primary key is Name
$XABKEY REF=0,- ; Key reference number
DAN=0,- ; Define data XABALL
DFL=1536,- ; Define data fill of 75%
FLG=<DUP>,- ; Allow duplicate keys
DTP=DSTG,- ; Descending sort order
IAN=1,- ; Define index XABALL
IFL=1536,- ; Initial index fill 75%
PROLOG=3,- ; Request prolog 3
.
.
.
DST_ALL0:
$XABALL AID=0,- ; Data area definition
ALQ=328,- ; Allocation quantity and
AOP=<CBT>,- ; contiguous best try
BKZ=4,- ; Bucket size of 4 blocks
DEQ=112,- ; Default extension quantity
NXT=DST_ALL1 ; Designate next XAB
DST_ALL1:
$XABALL AID=1,- ; Primary key index area
ALQ=8,- ; Allocation quantity and
AOP=<CBT>,- ; contiguous best try
BKZ=4,- ; Bucket size of 4 blocks
DEQ=4,- ; Default extension quantity
NXT=DST_ALL2 ; Designate next XAB
|
Fixed-length records are copied from the sequential input file on the local node to the indexed file on the remote node. Each variable-length output record is initially 66 bytes long and may be extended to a maximum of 128 bytes.
| Previous | Next | Contents | Index |
|
| privacy and legal statement | ||
| 4523PRO_038.HTML | ||