Ascential Datastage Basic Guide
Ascential Datastage Basic Guide
BASIC Guide
Version 7.0
August 2003
Part No. 00D-0006DS70
Published by Ascential Software Corporation.
©1997-2003 Ascential Software Corporation. All rights reserved. Ascential, DataStage, QualityStage,
AuditStage,, ProfileStage, and MetaStage ar are trademarks of Ascential Software Corporation or its affiliates
and may be registered in the United States or other jurisdictions. Windows is a trademark of Microsoft
Corporation. Unix is a registered trademark of The Open Group. Adobe and Acrobat are registered trademarks
of Adobe Systems Incorporated. Other marks are the property of the owners of those marks.
This product may contain or utilize third party components subject to the user documentation previously
provided by Ascential Software Corporation or contained herein.
                                                                                                                                 iii
Cataloging a BASIC Program ..................................................................................3-10
Catalog Shared Memory ...........................................................................................3-13
iv                                                                                                                BASIC Guide
Appendix B. ASCII and Hex Equivalents
Appendix C. Correlative and Conversion Codes
Appendix D. BASIC Reserved Words
Appendix E. @Variables
Appendix F. BASIC Subroutines
Index
                                               v
vi   BASIC Guide
                                                            Preface
Preface                                                                         vii
Organization of This Manual
       This manual contains the following:
          Chapter 1 covers information you should know before you begin to
          use BASIC, such as initial procedures, terminology, and features that
          are unique to this implementation of BASIC.
          Chapter 2 describes types of data, such as constants and variables, and
          types of operators.
          Chapter 3 describes the DataStage BASIC compiler. The discussion
          includes instructions on how to run the compiler, compiling options,
          warnings and error messages, and other related commands.
          Chapter 4 describes how to use locks, transaction processing, and
          isolation levels to prevent data loss and other data conflicts.
          Chapter 5 describes the tools available for debugging DataStage
          BASIC programs. Included is an interactive debugger, RAID, and the
          program listing command, VLIST.
          Chapter 6 contains statements and functions in alphabetical order. At
          the top of each page is the syntax for the statement or function,
          followed by a detailed description of its use, often including refer-
          ences to other statements or functions that can be used with it or are
          helpful to know about. Examples illustrate the application of the state-
          ment or function in a program.
          Appendix A is a quick reference for BASIC statements and functions
          grouped according to use.
          Appendix B is a table of ASCII character codes and equivalents and
          hexadecimal equivalents.
          Appendix C describes the syntax and use of correlative and conver-
          sion codes.
          Appendix D lists DataStage BASIC reserved words.
          Appendix E is a quick reference for DataStage BASIC @variables.
          Appendix F describes subroutines you can call from DataStage BASIC
          programs.
Convention       Usage
Bold             In syntax, bold indicates commands, function names, and
                 options. In text, bold indicates keys to press, function names,
                 menu selections, and MS-DOS commands.
UPPERCASE        In syntax, uppercase indicates DataStage commands,
                 keywords, and options; BASIC statements and functions;
                 and SQL statements and keywords. In text, uppercase also
                 indicates DataStage identifiers such as filenames, account
                 names, schema names, and Windows NT filenames and
                 pathnames.
Italic           In syntax, italic indicates information that you supply. In text,
                 italic also indicates UNIX commands and options, filenames,
                 and pathnames.
Courier          Courier indicates examples of source code and system
                 output.
Courier Bold     In examples, courier bold indicates characters that the user
                 types or keys the user presses (for example, <Return>).
[]               Brackets enclose optional items. Do not type the brackets
                 unless indicated.
{}               Braces enclose nonoptional items from which you must
                 select at least one. Do not type the braces.
itemA | itemB    A vertical bar separating items indicates that you can choose
                 only one item. Do not type the vertical bar.
...              Three periods indicate that more of the same type of item can
                 optionally follow.
‰                A right arrow between menu options indicates you should
                 choose each option in sequence. For example, “Choose
                 File ‰ Exit” means you should choose File from the menu
                 bar, then choose Exit from the File pull-down menu.
 I               Item mark. For example, the item mark ( I ) in the following
                 string delimits elements 1 and 2, and elements 3 and 4:
                 1I2F3I4V5
 F               Field mark. For example, the field mark ( F ) in the following
                 string delimits elements FLD1 and VAL1:
                 FLD1FVAL1VSUBV1SSUBV2
Preface                                                                              ix
Convention      Usage
V               Value mark. For example, the value mark ( V ) in the following
                string delimits elements VAL1 and SUBV1:
                FLD1FVAL1VSUBV1SSUBV2
S               Subvalue mark. For example, the subvalue mark ( S ) in the
                following string delimits elements SUBV1 and SUBV2:
                FLD1FVAL1VSUBV1SSUBV2
T               Text mark. For example, the text mark ( T ) in the following
                string delimits elements 4 and 5: 1F2S3V4T5
DataStage Documentation
         DataStage documentation includes the following:
         Client Guides
             DataStage Designer Guide. This guide describes the DataStage
             Designer, and gives a general description of how to create, design, and
             develop a DataStage application.
             DataStage Director Guide. This guide describes the DataStage
             Director and how to validate, schedule, run, and monitor DataStage
             server jobs.
             DataStage Manager Guide. This guide describes how to view and edit
             the contents of the Repository. It also shows how to import and export
             DataStage components.
             DataStage Administrator Guide. This guide describes DataStage
             setup, routine housekeeping, and administration.
Preface                                                                               xi
xii   Ascential DataStage BASIC Guide
                                                                                1
                                     Introduction to
                                  DataStage BASIC
BASIC Terminology
      DataStage BASIC programmers should understand the meanings of the
      following terms:
          •   BASIC program
          •   Source code
          •   Object code
          •   Variable
          •   Function
          •   Keyword.
      Source Code. Source code is the original form of the program written by
      the programmer.
Subroutines
            A subroutine is a set of instructions that perform a specific task. It is a small
            program that can be embedded in a program and accessed with a GOSUB
            statement, or it can be external to the program and accessed with a CALL
            statement. Common processes are often kept as external subroutines. This
            lets the programmer access them from many different programs without
            having to rewrite them.
            When a GOSUB or CALL statement is encountered, program control
            branches to the referenced subroutine. An internal subroutine must begin
            with a statement label. An external subroutine must begin with a
            SUBROUTINE statement.
            ARETURNstatement can be used at the end of a subroutine to return
            program flow to the statement following the last referenced GOSUB or
            CALL statement. If there is no corresponding CALL or GOSUB statement,
            the program halts and returns to the DataStage command level. If an
            external subroutine ends before a RETURN statement is encountered, a
            RETURN is provided automatically.
Source Syntax
      A BASIC source line has the following syntax:
Statement Types
      BASIC statements can be used for any of the following purposes:
          •   Input and output control
          •   Program control
          •   Assignment (assigning a value to a variable)
          •   Specification (specifying the value of a constant)
          •   Documentation
      Input statements indicate where the computer can expect data to come from
      (for example, the keyboard, a particular file, and so on). Output statements
      control where the data is displayed or stored.
      In general, BASIC statements are executed in the order in which they are
      entered. Control statements alter the sequence of execution by branching to
      a statement other than the next statement, by conditionally executing
      statements, or by passing control to a subroutine.
      Assignment statements assign values to variables, and specification state-
      ments assign names to constants.
      Program documentation is accomplished by including optional comments
      that explain or document various parts of the program. Comments are
Statement Labels
            A statement label is a unique identifier for a program line. A statement
            label consists of a string of characters followed by a colon. The colon is
            optional when the statement label is completely numeric. Like variable
            names, alphanumeric statement labels begin with an alphabetic character
            and can include periods ( . ), dollar signs ( $ ), and percent signs ( % ).
            Upper- and lowercase letters are interpreted as different; that is, ABC and
            Abc are different labels. Statement labels, like variable names, can be as
            long as the length of the physical line, but only the first 64 characters are
            significant. A statement label can be put either in front of a BASIC state-
Spaces or Tabs
          In a program line, spaces or tabs that are not part of a data item are
          ignored. Therefore you can use spaces or tabs to improve the program’s
          appearance and readability.
Special Characters
          The DataStage BASIC character set comprises alphabetic, numeric, and
          special characters. The alphabetic characters are the upper- and lowercase
          letters of the alphabet. The numeric characters are the digits 0 through 9.
          The special characters are as follows. Most of the special characters are not
          permitted in a numeric constant or a variable name.
          Space
          Tab
      =   Equal sign or assignment symbol
      +   Plus sign
Storing Programs
            BASIC programs are stored as records in type 1 or type 19 files. The
            program file must exist before you invoke an editor to create a new record
            to hold your program. Record IDs must follow the conventions for type 1
            and type 19 files.
Editing Programs
            You can use the DataStage Editor or any suitable editor, such as vi on UNIX
            or edit on Windows NT, to write your programs. You can edit programs in
            the DataStage environment or at the operating system level.
Getting Started
      To create and use a BASIC program, follow these steps:
      1.   Use the CREATE.FILE command to create a type 1 or type 19
           DataStage file to store your BASIC program source. The RUN
           command uses the filename BP if a filename is not specified, so many
           people use BP as the name of their general BASIC program file.
      2.   Use the DataStage Editor or some other editor to create the source for
           your BASIC program as a record in the file you created in step 1.
      3.   Once you have created the record containing your BASIC program
           source statements, use the BASIC command to compile your
           program. The BASIC command creates a file to contain the object
Types of Data
            Although many program languages distinguish different types of data,
            the DataStage BASIC compiler does not. All data is stored internally as
            character strings, and data typing is done contextually at run time. There
            are three main types of data: character string, numeric, and unknown
            (that is, the null value).
            Numeric Constants
            Numeric constants can be represented in either fixed-point or floating-
            point form. Commas and spaces are not allowed in numeric constants.
            Note: Do not confuse the null value with the empty string. The empty
                  string is a character string of zero length which is known to have
                  no value. Unlike null, whose value is defined as unknown, the
      Like all other data in DataStage BASIC, the null value is represented
      internally as a character string. The string is made up of the single byte
      CHAR(128). At run time when explicit or implicit dynamic array
      extractions are executed on this character, it is assigned the data type
      “null.” DataStage BASIC programs can reference the null value using the
      system variable @NULL. They can test whether a value is the null value
      using the ISNULL and ISNULLS functions.
      There is no printable representation of the null value. In this manual the
      symbol λ (lambda) is sometimes used to denote the null value.
      Here is an example of the difference between an empty string and the null
      value. If you concatenate a string value with an empty string, the string
      value is returned, but if you concatenate a string value with the null
      value, null is returned.
          A   =   @NULL
          B   =   ""
          C   =   "JONES"
          X   =   C:B
          Y   =   C:A
      The resulting value of X is "JONES", but the value of Y is the null value.
      When you concatenate known data with unknown data, the result is
      unknown.
      Programmers should also note the difference between the null value—a
      special constant whose type is “null”—and the stored representation of
      the null value—the special character CHAR(128) whose type is “string.”
      BASIC programs can reference the stored representation of null using the
      system variable @NULL.STR instead of @NULL.
Variables
            Variables are symbolic names that represent stored data values. The value
            of a variable can be:
                 • Unassigned
                 • A string, which can be an alphanumeric character string, a number,
                   or a dynamic array
                 • A number, which can be fixed-point (an integer) or floating-point
                 • The null value
                 • A dimensioned array (that is, a vector or matrix)
                 • A subroutine name
                 • A file
                 • A select list
            The value of a variable can be explicitly assigned by the programmer, or it
            can be the result of operations performed by the program during
            execution. Variables can change in value during program execution. At
            the start of program execution, all variables are unassigned. Any attempt
            to use an unassigned variable produces an error message.
            A variable name must begin with an alphabetic character. It can also
            include one or more digits, letters, periods, dollar signs, or percent signs.
            Spaces and tabs are not allowed. A variable name can be any length up to
            the length of the physical line, but only the first 64 characters are
            significant. A variable name cannot be any of the BASIC Reserved Words
            listed in Appendix D. In DataStage, upper- and lowercase characters in a
            variable name are interpreted differently.
            DataStage BASIC also provides a set of system variables called
            @Variables. Many of these are read-only variables. Read-only @variables
            cannot be changed by the programmer.
Array Variables
        An array is a variable that represents more than one data value. There are
        two types of array: dimensioned and dynamic. Dimensioned arrays can
        be either standard or fixed. Fixed arrays are provided in PICK, IN2, and
        REALITY flavor accounts for compatibility with other Pick systems.
        Dimensioned Arrays
        Each value in a dimensioned array is called an element of the array.
        Dimensioned arrays can be one- or two-dimensional.
        A one-dimensional array is called a vector. Its elements are arranged
        sequentially in memory. An element of a vector is specified by the
        variable name followed by the index of the element enclosed in
        parentheses. The index of the first element is 1. The index can be a
        constant or an expression. Two examples of valid vector element
        specifiers are:
            A(1)
            COST(35)
        A two-dimensional array is called a matrix. The elements of the first row
        are arranged sequentially in memory, followed by the elements of the
        second row, and so on. An element of a matrix is specified by the variable
        name followed by two indices enclosed in parentheses. The indices
        represent the row and column position of the element. The indices of the
        first element are (1,1). Indices can be constants or expressions. The indices
        used to specify the elements of a matrix that has four columns and three
        rows are illustrated by the following:
            1,1       1,2       1,3        1,4
            Dynamic Arrays
            Dynamic arrays map the structure of DataStage file records to character
            string data. Any character string can be a dynamic array. A dynamic array
            is a character string containing elements that are substrings separated by
            delimiters. At the highest level these elements are fields separated by
            field marks ( F ) (ASCII 254). Each field can contain values separated by
            value marks ( V ) (ASCII 253). Each value can contain subvalues separated
            by subvalue marks ( S ) (ASCII 252).
            A common use of dynamic arrays is to store data that is either read in
            from or written out to a DataStage file record. However, DataStage BASIC
            includes facilities for manipulating dynamic array elements that make
File Variables
         A file variable is created by a form of the OPEN statement. Once opened,
         a file variable is used in I/O statements to access the file. There are two
         types of file variable: hashed file variable and sequential file variable. File
         variables can be scalars or elements of a dimensioned array.
Format Expressions
            A format expression formats variables for output. It specifies the size of
            the field in which data is displayed or printed, the justification (left, right,
            or text), the number of digits to the right of the decimal point to display,
            and so on. Format expressions work like the FMT function. The syntax is:
                 variable format
            format is a valid string expression that evaluates to:
Operators
        Operators perform mathematical, string, and logical operations on
        values. Operands are expressions on which operations are performed.
        BASIC operators are divided into the following categories:
            •   Arithmetic
            •   String
            •   Relational
            •   Pattern matching
            •   IF operator
            •   Logical
            •   Assignment
            •   Dynamic array
Arithmetic Operators
        Arithmetic operators combine operands comprising one or more
        variables, constants, or intrinsic functions. Resulting arithmetic
        expressions can be combined with other expressions almost indefinitely.
        The syntax of arithmetic expressions is:
            expression operator expression
String Operators
            The concatenation operator ( : or CAT) links string expressions to form
            compound string expressions, as follows:
                  'HELLO.    ' : 'MY NAME IS ' : X : ".        WHAT'S YOURS?"
            or:
                  'HELLO.    ' CAT 'MY NAME IS ' CAT X CAT ".          WHAT'S
                  YOURS?"
If, for instance, the current value of X is JANE, these string expressions both have the
following value:
                   "HELLO.    MY NAME IS JANE.      WHAT'S YOURS?"
            Multiple concatenation operations are performed from left to right.
            Parenthetical expressions are evaluated before operations outside the
            parentheses.
            Substring Operator
            A substring is a subset of contiguous characters of a character string. For
            example, JAMES is a substring of the string JAMES JONES. JAMES JON
            is also a substring of JAMES JONES.
            You can specify a substring as a variable name or an array element
            specifier, followed by two values separated by a comma and enclosed in
            square brackets. The two values specify the starting character position
            and the length of the substring. The syntax is:
Relational Operators
        Relational operators compare numeric, character string, or logical data.
        The result of the comparison, either true ( 1 ) or false ( 0 ), can be used to
        make a decision regarding program flow (see the IF statement). Table 2-2
        lists the relational operators.
            A space is evaluated as less than zero. Leading and trailing spaces are
            significant. If two strings can be converted to numeric, then the
            comparison is always made numerically.
Some examples of true comparisons are:
                 "AA" < "AB"
                 "FILENAME" = "FILENAME"
                 "X&" > "X#"
                 "CL " > "CL"
                 "kg" > "KG"
IF Operator
        The IF operator lets you indicate a value conditional upon the truth of
        another value. The IF operator has the following syntax:
              variable = IF expression THEN expression ELSE expression
Logical Operators
            Numeric data, string data, and the null value can function as logical data.
            Numeric and string data can have a logical value of true or false. The
            numeric value 0 (zero), is false; all other numeric values are true.
            Character string data other than an empty string is true; an empty string
            is false. The null value is neither true nor false. It has the special logical
            value of null.
            Logical operators perform tests on logical expressions. Logical
            expressions that evaluate to zero or an empty string are false. Logical
            expressions that evaluate to null are null. Expressions that evaluate to any
            other value are true.
            The logical operators in DataStage BASIC are:
                 • AND (or the equivalent &)
                 • OR (or the equivalent !)
                 • NOT
            The NOT function inverts a logical value.
NOT
TRUE FALSE
NULL NULL
FALSE TRUE
       Note: The logical value NULL takes the action of false, because the condi-
             tion is not known to be true.
Example      Interpretation
X=5          This statement assigns the value 5 to the variable X.
X += 5       This statement is equivalent to X=X+5. It adds 5 to the value of the
             variable X, changing the value of X to 10 if it was originally 5.
X −= 3       This statement is equivalent to X=X−3. It subtracts 3 from the value
             of the variable X, changing the value of X to 2 if it was originally 5.
X := Y       This statement is equivalent to X=X:Y. If the value of X is 'CON', and
             the value of Y is 'CATENATE', then the new value of the variable X is
             'CONCATENATE'.
                                       Corresponding Instruction
       Vector Function
                                       for Single-Valued Field
       ADDS (m1, m2)                   s1 + s2
       ANDS (m1, m2)                   s1 AND s2
       CATS (m1, m2)                   s1 : s2
       CHARS(m1)                       CHAR (s1)
       COUNTS (m1, p1)                 COUNT (s1, p1)
       DIVS (m1, m2)                   s1 / s2
       EQS (m1, m2)                    s1 EQ s2
                                             Corresponding Instruction
             Vector Function
                                             for Single-Valued Field
             ISNULLS (m1)                    ISNULL (s1)
             NES (m1, m2)                    s1 NE s2
             LES (m1, m2)                    s1 LE s2
             LTS (m1, m2)                    s1 LT s2
             GES (m1, m2)                    s1 GE s2
             GTS (m1, m2)                    s1 GT s2
             NOTS (m1)                       NOT (s1)
             FIELDS (m1, p1, p2, p3)         FIELD (s1, p1, p2, p3)
             FMTS (m1, p1)                   FMT (s1, p1)
             ICONVS (m1, p1)                 ICONV (s1, p1)
             IFS (m1, m2, m3)                IF s1 THEN s2 ELSE s3
             INDEXS (m1, p1, p2)             INDEX (s1, p1, p2)
             LENS (m1)                       LEN (s1)
             MODS (m1, m2)                   MOD (s1, s2)
             MULS (m1, m1)                   s1 * s2
             NUMS (m1)                       NUM (s1)
             OCONVS (m1, p1)                 OCONV (s1, p1)
             ORS (m1, m2)                    s1 OR s2
             SEQS (m1)                       SEQ (s1)
             STRS (m1, p1)                   STR (s1, p1)
             SPACES (m1)                     SPACE (s1)
             SPLICE (m1, p1, m2)             s1 : p1 : s2
             SUBSTRINGS (m1, p1, p2)         s1 [p1, p2]
             SUBS (m1, m1)                   s1 – s2
             TRIMS (m1)                      TRIM (s1)
Before you can run a BASIC program, you must compile it with the DataStage
BASIC compiler. The compiler takes your source code as input and produces
executable object code.
Use the DataStage command CREATE.FILE to create a type 1 or type 19 file in
which to store the source code of your BASIC programs. You can create and edit
the source code with an operating system editor (such as vi), the DataStage Editor,
or a combination of the two.
BASIC
BASIC Options
You can use the following options with the BASIC command:
A listing produced with either the −LIST or the −XREF option is saved in a file
whose name is made up of the source filename and a suffixed .L . The record ID of
the program listing in the listing file (filename.L) is the same as the record ID in the
program file (filename).
The +$ Option
The +$ option specifies the $OPTIONS options you want to turn on, or the flavor
you want the program to use. See the $OPTIONS statement for the list of options
and flavors. You must specify all options you want to turn on before the options
you want to turn off.
The –$ Option
The –$ option specifies the $OPTIONS options, or the flavor, you want to turn off.
See the $OPTIONS statement for the list of options and flavors. You must specify
all options you want to turn off after the options you want to turn on.
The –I Option
The −I option inhibits the execution of RAID or VLIST on your BASIC program.
This lets you bypass subroutines already debugged and provides security to your
subroutines.
    Compilation Complete.
    >ED BP.L DATE.INT
    13 lines long.
    ----: P
    0001: BP.L/DATE.INT Source Listing
    0002:
    0003:
    0004: Cross Reference Listing
    0005:
    0006: Variable....... Type.......... References........................
     ............
    0007:
    0008: DATE            Local Scalar   0003=    0004
    0009:
    0010: * Definition of symbol
    0011: = Assignment of variable
    0012: ! Dimension
    0013: @ Argument to CALL
The listing shows three columns: Variable, Type, and References. Variable is the
name of the variable or symbol. Type is one of the following symbol types:
    Local Scalar
    Local Array
    Common Scalar
      *           Definition of symbol
      =           Assignment of variable
      !           Dimension of array
      @           Argument to CALL statement
The –T Option
The −T option suppresses the table of symbols and the table of line numbers that
are appended to the end of object files. These tables are used for handling run-time
error messages. Suppressing them results in somewhat smaller object files, but
run-time error messages do not know the line number or variable involved in the
error.
Compiler Directives
Compiler directives are BASIC statements that direct the behavior of the compiler.
Functions performed by compiler directives include: inserting source code from
one program into another program during compilation, setting compile-time
compatibility with another flavor, and specifying a condition for compiling certain
    /                    converts to      ?\
    ?                    converts to      ??
    An ASCII NUL         converts to      ?0
    An initial .         converts to      ?.
Any leading *> is ignored. If a full pathname is specified, the > between directory
names changes to a / to yield the following:
[ pathname/ ] program
Conditional Compilation
You can specify the conditions under which all or part of a BASIC program is to be
compiled, using:
      • A modified version of the IF statement
      • $IFDEF
      • $IFNDEF
Conditional compilation with the modified IF statement is useful for customizing
large programs that are to be used by more than one kind of user. It can also reduce
the size of the object code and increase program efficiency.
You can use the compiler directives $IFDEF and $IFNDEF to control whether or
not sections of a program are compiled. Both of these compiler directives test a
given identifier to see if it is currently defined (that is, has appeared in a $DEFINE
compiler directive and has not been undefined). If the identifier that appears in a
$IFDEF is defined, all the program source lines appearing between the $IFDEF
compiler directive and the closing $ENDIF compiler directive are compiled. If the
identifier is not defined, all the lines between the $IFDEF compiler directive and
the $ENDIF compiler directive are ignored.
IF Statements
The syntax of the conditional compilation statement is the same as that of the IF
statement with the exception of the test expression, which must be one of the
following: $TRUE, $T, $FALSE, or $F. The syntaxes are as follows:
     IF $TRUE THEN statements ELSE statements
     IF $T THEN statements ELSE statements
     IF $FALSE THEN statements ELSE statements
     IF $F THEN statements ELSE statements
The conditional compilation statement can specify a variable name rather than one
of the test specifications listed previously. If it does, an EQUATE statement
equating the variable to the test specification must appear at the beginning of the
source code. For example:
     EQUATE USER.A LIT "$T"
     IF USER.A THEN statements ELSE statements
Consider a program that contains debugging statements in its source code. Using
the conditional compilation statement, you could create two versions of the object
code from the same source: a test version that contains the debugging statements,
and a release version without the debugging statements. The following steps
produce the two required versions of the program:
1.   Include a conditional debugging statement throughout the source code:
     IF TEST PRINT X,Y
2.   Put the following statement in the source code before any conditional
     statements:
     EQUATE TEST LIT "$TRUE"
3.   Compile the source code to produce object code that contains the debugging
     statements (the test version).
4.   Change the EQUATE statement to:
     EQUATE TEST LIT "$FALSE"
Successful Compilation
When all errors in the source code are corrected, the compiler successfully
completes the compilation by producing an object code record. The object code
record is stored in a file whose name is made up of the source filename suffixed
with .O (sourcename.O). The object code record ID is the same as the source file
record ID (program name).
For example, if source code record MAIN is stored in a file called BP, executing the
following compile statement:
    >BASIC BP MAIN
compiles the source code in record MAIN in file BP, producing object code that is
stored in record MAIN in file BP.O . The compiler creates the object code file if it
does not exist.
For example, the following command executes the record MAIN in the BP.O file:
    >RUN BP MAIN
Run-time error messages are printed on the terminal screen as they are encoun-
tered, unless the NO.WARN keyword was specified.
Note: Cataloged programs are considered executable files (that is, the RUN
      command is not required). To run a cataloged program, enter its catalog
      name at the system prompt.
Local Cataloging
Local cataloging creates a VOC entry for the program. This entry is a verb that
points to the file and record containing the object code for the cataloged program.
A locally cataloged program can be accessed only from the account in which it was
cataloged, unless you copy the VOC entry for the catalog name to another account.
Since cataloging a program locally only creates a VOC entry pointing to the object
file, you need not recatalog the program every time you recompile it.
Normal Cataloging
Normal cataloging copies the specified object record to the system catalog space
and gives it a name of the form:
    *account*catalog.name
Normal cataloging also creates a VOC entry for the program. This entry is a verb
that contains the name *account*catalog in field 2. A normally cataloged program
can be accessed only from the account in which it was cataloged, unless you copy
the VOC entry for the catalog name to another account or specify the full catalog
name, including the account prefix.
Since cataloging a program normally copies the object code to the system catalog
space, you must recatalog the program every time you recompile it.
Global Cataloging
Global cataloging copies the specified object record into the system catalog space
and gives it a name in one of the following forms:
    *catalog.name
    −catalog.name
    $catalog.name
    !catalog.name
VOC entries are not created for globally cataloged programs. They are available to
all accounts on the system as soon as they are cataloged. The DataStage command
processor looks in the system catalog space for verbs or external subroutines that
have an initial *. The run machine looks in the system catalog space for verbs or
subroutines whose names begin with *, −, $, or ! .
Note: Because the command processor interprets any line beginning with an
      asterisk and containing blanks as a comment, you cannot use command
      parameters when you invoke a globally cataloged program. That is, you
      can use the following command to run the globally catalog program
      *GLOBAL, but you cannot include arguments in the command line:
       >*GLOBAL
DELETE.CATALOG
The DELETE.CATALOG command removes locally, normally, or globally cata-
loged programs. It has the following syntax:
    DELETE.CATALOG catalog.name
DECATALOG
The DECATALOG command removes a locally cataloged program. It deletes the
object code and removes the catalog entry from the user’s VOC file. It has the
following syntax:
This chapter describes the DataStage BASIC mechanisms that prevent lost updates
and other problems caused by data conflicts among concurrent users:
    • Locks
    • Transactions
    • Isolation levels
Locks
DataStage locks control access to records and files among concurrent users. To
provide this control, DataStage supports the following two levels of lock
granularity:
    • Fine granularity (record locks)
    • Coarse granularity (file locks)
The level at which you acquire a lock is known as granularity. Record locks affect a
smaller component (the record) and provide a fine level of granularity, whereas file
locks affect a larger component (the file) and provide a coarse level of granularity.
Lock compatibility determines what your process can access when other processes
have locks on records or files. Record locks allow more compatibility because they
coexist with other record locks, thus allowing more transactions to take place
concurrently. However, these “finer-grained” locks provide a lower isolation level.
File locks enforce a higher isolation level, providing more concurrency control but
less compatibility.
Lock compatibility decreases and isolation level increases as strength and granu-
larity increase. This can increase the possibility of deadlocks at high isolation
Note: An update record lock you own is incompatible with a shared file lock you
      own. Be sure to use a LOCKED clause to avoid deadlocks.
Note: A shared file lock you own is incompatible with an update record lock you
      own. Be sure to use a LOCKED clause to avoid deadlocks.
In DataStage BASIC a shared file lock can be acquired or promoted with a FILE-
LOCK statement and released with aCLOSE, FILEUNLOCK, RELEASE, or STOP
statement.
In DataStage BASIC, an intent file lock can be acquired or promoted from a shared
file lock with a FILELOCK statement, and released with a CLOSE, FILEUNLOCK,
RELEASE, or STOP statement.
In DataStage BASIC an exclusive file lock can be acquired from a shared file lock
with a FILELOCK statement and released with a CLOSE, FILEUNLOCK,
RELEASE, or STOP statement.
Deadlocks
Deadlocks occur when two users who acquire locks incrementally try to acquire a
lock that the other user owns, and the existing lock is incompatible with the
requested lock. The following situations can lead to deadlocks:
    • Lock promotion from a shared record or shared file lock to a stronger lock
    • Lock escalation to file locks when two users try to escalate at the same time
You can configure DataStage to automatically identify and resolve deadlocks as
they occur through the Deadlock Daemon Administration menu, or you can
manually fix a deadlock by selecting and aborting one of the deadlocked user
processes. You use the deadlock daemon dsdlockd to identify and resolve dead-
locks. For more information, see Administering DataStage.
Active Transactions
DataStage BASIC supports nested transactions. Any transaction can include:
      • Read and write operations
      • Other transactions or subtransactions that can contain other operations or
        other transactions
When a transaction begins, it is active. If a second transaction begins before the first
transaction is committed or rolled back, the new (child) transaction becomes the
active transaction while the first (parent) transaction continues to exist but inac-
tively. The child transaction remains active until:
      • It is committed or rolled back, when the parent transaction becomes active
        again
      • Another transaction (child) begins and becomes the active transaction
Only one transaction can be active at any time, although many transactions can
exist concurrently. Only one transaction can exist at each transaction nesting level.
The top-level transaction is at nesting level 1. When no transactions exist, the
nesting level is 0.
Transaction Properties
Each transaction must possess properties commonly referred to as the ACID
properties:
    •   Atomicity
    •   Consistency
    •   Isolation
    •   Durability
In nested transactions, child transactions exhibit atomicity, consistency, and isola-
tion properties. They do not exhibit the durability property since the parent adopts
its operation when it is committed. The operations affect the database only when
the top-level transaction is committed. Therefore, even though a child transaction
is committed, its operations and locks can be lost if the parent transaction is rolled
back.
Atomicity
Either all the actions of a transaction occur successfully or the transaction is nulli-
fied by rolling back all operations. The transaction system ensures that all
operations performed by a successfully committed transaction are reflected in the
database, and the effects of a failed transaction are completely undone.
Consistency
A transaction moves the database from one valid state to another valid state, and
if the transaction is prematurely terminated, the database is returned to its
previous valid state.
Isolation
The actions carried out by a transaction cannot become visible to another transac-
tion until the transaction is committed. Also, a transaction should not be affected
by the actions of other concurrent transactions. DataStage provides different isola-
tion levels among concurrently executing transactions.
Serializability
In addition to the ACID properties, SQL standards stipulate that transactions be
serializable. Serializability means that the effects of a set of concurrent transactions
should produce the same results as though the individual transactions were
executed in a serial order, and as if each transaction had exclusive use of the
system. In DataStage, serializability can be achieved by using isolation level 4 for
all transactions.
@Variables
You can use the following @Variables to track transaction activity:
    •   @ISOLATION
    •   @TRANSACTION
    •   @TRANSACTION.ID
    •   @TRANSACTION.LEVEL
@ISOLATION indicates the current transaction isolation level (0, 1, 2, 3, or 4) for
the active transaction, or the current default isolation level if no transaction exists.
Transaction Restrictions
Other than memory and disk space, there is no restriction on the number of nesting
levels in transactions. However, it is important to remember that having many
levels in a transaction affects system performance.
You cannot use the following statements while a transaction is active. Doing so
causes a fatal error.
      • CLEARFILE
      • SET TRANSACTION ISOLATION LEVEL
You cannot use the EXECUTE or PERFORM statements in a transaction to execute
most DataStage commands and SQL statements. However, you can use EXECUTE
and PERFORM to execute the following DataStage commands and SQL statements
within a transaction:
Isolation Levels
isolation level
Setting a transaction’s isolation level helps avoid various data anomalies.
DataStage BASIC lets you set different isolation levels depending on which data
anomalies you want to avoid. Your transaction runs at the specified isolation level
 Level        Type
 0            NO.ISOLATION
 1            READ.UNCOMMITTED
 2            READ.COMMITTED
 3            REPEATABLE.READ
 4            SERIALIZABLE
Data Anomalies
Isolation levels provide protection against the following data anomalies or
conflicts that can occur when two processes concurrently access the data:
     • Lost updates occur when two processes try to update an object at the same
       time. For example, Process A reads a record. Process B reads the same
       record, adds 10, and rewrites it. Process A adds 20 to the record that it
       previously read, and rewrites the record. Thus, Process B’s update is lost.
 Setting            Description
 0                  Provides backward compatibility. Transactions are not required
                    to use well-formed writes.
 1                  Enforces well-formed writes in BASIC transactions. This is the
                    default.
 2                  Enforces well-formed writes in BASIC programs, whether or
                    not they are in a transaction.
Example
The following example illustrates how isolation levels affect transactions. A trans-
action running at isolation level 2 deletes records for Customer 100 from the file
CUST. The transaction scans the file ORDERS for all orders placed by this customer
and deletes each order. The part of the transaction that deletes the orders does not
want to lock the ORDERS file unnecessarily.
The following program illustrates how lock escalation takes place:
    OPEN "CUST" TO CUST ELSE
        STOP "Cannot open CUST file"
    END
    OPEN "ORDERS" TO ORDERS ELSE
        CLOSE CUST
        STOP "Cannot open ORDERS file"
    END
    LOCK.COUNT = 0
DataStage provides two debugging tools: RAID and VLIST. RAID is an interactive
debugger. VLIST is a diagnostic tool that lists source code followed by object code,
as well as statistics about your program.
Note: You cannot run RAID or VLIST on programs compiled with the −I option.
RAID
RAID
You can use RAID with your DataStage BASIC programs. RAID is both an object
code and a source code debugger—a powerful tool for detecting errors in
DataStage BASIC code. RAID lets you do the following:
       • Set and delete breakpoints. You can suspend execution of the program at
         specified lines by setting breakpoints in the source code. Once RAID
         suspends program execution, you can examine the source code, change or
         display variable values, set additional breakpoints, or delete breakpoints.
       • Set watchpoints. You can keep track of changes to variable values by
         setting watchpoints in the source code. When a variable’s value changes,
         RAID can print both the old and new values and the source instruction that
         caused the change.
       • Step through and display source code, line by line or in segments.
       • Examine object addresses.
       • Display and modify variables.
       • Display all the elements of an array in succession.
You can invoke RAID from the command processor, from within a BASIC
program, or by pressing the Break key while your BASIC program is executing.
Use RAID the same way you use RUN. This causes RAID to be invoked just before
program execution. For example, the following command executes the file
BP.O/MAIN using the RAID debugger:
      >RAID BP MAIN
When you invoke RAID from the command processor, RAID displays the first
executable source code instruction, followed by a double colon ( :: ). Enter a RAID
command at the :: prompt. To run the program, enter R at the :: prompt. To quit
RAID, enter Q. RAID commands are discussed in detail in “RAID Commands” on
page 5-4.
Note: Unreferenced variables are not carried in the symbol table of BASIC object
      code. Therefore, RAID can only display the contents of variables referenced
      in the current subroutine. RAID ignores all unreferenced variables, and
      treats them as unknown symbols.
Registers hold intermediate values in a program. For example, for the following
statement the sum of 3 and 4 is evaluated and placed in $R0:
    A=B:3+4
RAID Commands
You can enter any RAID command from the double colon ( :: ) prompt. RAID
commands have the following general syntax:
      position command qualifier
position      Tells where and how often to execute the RAID command in the
              program. You can provide one of the following:
              line         The decimal number of a line of the source code.
              address      The hexadecimal address of an object code instruction,
                           indicated by a leading 0X.
              procedure    The name of a procedure in the source code.
              variable     The name of a variable in the source code. You must
                           specify the variable exactly as it appears in the source
                           code. Variable names are case-sensitive, so “A” is not the
                           same as “a”. Subscript variable to indicate an element of
                           an array. For example, A[1,2].
              n            Indicates the number of times to execute the command.
qualifier     Can be either of the following:
              string       A string of characters to search for or to replace the value
                           of a variable.
              *            Indicates a special form of the specified command.
Command           Description
line              Displays the specified line of the source code.
/[string]         Searches the source code for string.
B                 Sets a RAID breakpoint.
C                 Continues program execution.
D                 Deletes a RAID breakpoint.
G                 Goes to a line or address, and continues program execution.
H                 Displays statistics for the program.
I                 Displays and executes the next object code instruction.
L                 Displays the next line to be executed.
M                 Sets watchpoints.
Q                 Quits RAID.
R                 Runs the program.
S                 Steps through the BASIC source code.
T                 Displays the call stack trace.
V                 Enters verbose mode for the M command.
V*                Prints the compiler version that generated the object code.
W                 Displays the current window.
X                 Displays the current object code instruction and address.
X*                Displays local run machine registers and variables.
Z                 Displays the next 10 lines of source code.
$                 Turns on instruction counting.
#                 Turns on program timing.
+                 Increments the current line.
−                 Decrements the current line.
.                 Displays object code instruction and address before execution.
variable/         Prints the value of variable.
variable!string   Changes the value of variable to string.
      / [ string ]
The search begins at the line following the current line. If string is found, RAID sets
the current line to the line containing string and displays that line. If you issue the
/ command without specifying string, RAID searches for the last-searched string.
This is an empty string if you have not yet specified a string during this RAID
session. If RAID encounters the end of file without finding string, it starts at the
first line and continues the search until it finds string or returns to the current line.
B: Setting Breakpoints
Use the B command to set or list RAID breakpoints. There are two syntaxes:
D: Deleting Breakpoints
Use the D command to delete RAID breakpoints. There are two syntaxes:
    [ address | line ] D
    D*
You can delete a RAID breakpoint at the current line, an object code address, or a
BASIC source line number. If you use the * option, this command deletes all break-
points. Some BASIC statements produce multiple run machine statements. If you
delete a breakpoint by line number, RAID tries to match the starting address of the
BASIC source number. A breakpoint set at anything other than the starting address
of a BASIC source line must be deleted by its address.
address | line G
    [n]I
If you use the n option, RAID displays and executes the next n instructions.
      variable M [ ; [ variable ] M …]
      variable =VALUE M
variable is a variable found in the symbol table.
VALUE is the value that you want to break.
The second syntax lets you set a watchpoint for a variable set to a specific value.
A watchpoint condition occurs when RAID monitors a variable until the variable’s
value changes. The program then suspends operation and displays the variable’s
old value, its new value, and the source instruction that caused the change to occur.
If no change occurs, no display appears. This command accepts up to eight vari-
ables. To continue monitoring a variable after RAID has already displayed a
change to that variable’s value, you need only enter M again, not variable M.
Q: Quitting RAID
Use the Q command to quit RAID.
      [n]S [*]
If the line includes a subroutine call, RAID steps into the subroutine. If you use the
n option, RAID steps through the next n lines. If you use the * option, RAID steps
around any subroutine call, essentially treating the entire subroutine as a single
line. That is, the S* command instructs RAID to display and execute a source line.
If the line includes a subroutine call, RAID executes the subroutine and displays
the first source line occurring after the subroutine returns.
    [ line ] W
A window comprises a group of 10 lines of source code, centered around the
current line. For example, if the current line is 4 when you issue the W command,
RAID displays the first 10 lines. The W command by itself does not change the
current line. If you use the line option, RAID changes the current line to line and
displays a window centered around that line. For example, if the current line is 14,
RAID displays the lines 9−18. Note that this command affects only the current
display lines; it does not affect the executable lines. That is, if you are stepping
through the code using the S command, RAID considers the current display line to
be the last line displayed before issuing the first S command.
      X[*]
      X!
The second syntax lets you set a variable to the empty string. The value of X! is
displayed with a carriage return immediately following.
RAID displays the contents. The run machine variables are the following:
      [ line ] Z
For example, if the current line is 4 when you issue the Z command, RAID displays
lines 4−13, and line 13 becomes the current line.
The current window changes each time this command is used, since the last line
printed becomes the current line. For example, if the current line is 14 when you
issue the Z command, RAID displays lines 14−23. The current line becomes 23.
If you use the line option, the current line becomes line, and RAID displays a
window with the specified line first. Regardless of the syntax used, the last line
printed becomes the current line once you issue this command. Note that this
command affects only the current display lines. It does not affect the executable
lines.
    [n]+
If you use the n option, RAID adds n to the current line. However, the command
only displays valid lines. For example, if you start at line 4 and use the command
3+, the command moves to line 7 only if line 7 is a valid line for the code. If line 7
is invalid, RAID moves to the first valid line after line 7. Note that this command
does not affect the currently executed line. That is, if you are stepping through the
code using the S command, RAID considers the current line to be the line you
displayed before issuing the first S command.
    [n]−
If you use the n option, RAID subtracts n from the current line. However, the
command only displays valid lines. For example, if you start at line 14 and use the
command 3−, the command moves to line 11 only if line 11 is a valid address for
the code. If line 11 is invalid, RAID moves backward to the first valid address
before address 11.
VLIST
Use VLIST to display a listing of the object code of a BASIC program. The syntax
for VLIST is as follows:
filename     The name of the file containing the source code of the BASIC
             program. The default filename is BP.
program      The name of the program to list.
R            Displays internal reference numbers for variables and constants
             rather than source code names and values.
    0001: FOR I = 1 TO 10
    0001 0000 : 0F8 move                  0 => I
    0001 0006 : 098 forincr               I 10 1 0020:
    0002: PRINT I
    0002 0014 : 130 printcrlf             I
    0003: NEXT I
    0003 001A : 0C2 jump                  0006:
    0004: END
    0004 0020 : 190 stop
This chapter describes the BASIC statements and functions. Each statement and
function is listed on a separate page. The sample shows a typical statement or func-
tion reference page.
                                                Statement or function
                    XXXX statement                              name
                                                Statement or function
                                                               syntax
   Syntax
     STATEMENT qualifiers
   Example
                                              Example showing how to
     OPEN 'DICT','FILE' TO FILE.V            use statement or function
     ELSE GOTO OPEN.ERR:
     CLEARFILE FILE.V
     CLOSE FILE.V
!
Syntax
      ! [comment.text]
Description
Use the ! statement to insert a comment in a BASIC program. Comments explain
or document various parts of a program. They are part of the source code only and
are nonexecutable. They do not affect the size of the object code.
A comment must be a separate BASIC statement and can appear anywhere in a
program. A comment must begin with one of the following comment designators:
      REM    *   !   $*
Any text that appears between a comment designator and the end of a physical line
is treated as part of the comment, not as part of the executable program. If a
comment does not fit on one physical line, you can continue it on the next physical
line only by starting the new line with a comment designator. If a comment
appears at the end of a physical line containing an executable statement, you must
put a semicolon ( ; ) before the comment designator.
Example
The PRINT statement at the end of the third line is not executed because it follows
the exclamation point on the same line and is treated as part of the comment. Lines
4, 5, and 6 show how to include a comment in the same sequence of executable
statements.
      001:   PRINT "HI THERE"; ! Anything after the ! is a comment.
      002:   ! This line is also a comment and does not print.
      003:   IF 5<6 THEN PRINT "YES"; ! A comment; PRINT "PRINT ME"
      004:   IF 5<6 THEN
      005:     PRINT "YES"; ! A comment
      006:     PRINT "PRINT ME"
      007:   END
This is the program output:
      HI THERE
      YES
      YES
      PRINT ME
#INCLUDE
Syntax
    #INCLUDE [filename] program
    #INCLUDE program FROM filename
Description
Use the #INCLUDE statement to direct the compiler to insert the source code in the
record program and compile it with the main program. The #INCLUDE statement
differs from the $CHAIN statement in that the compiler returns to the main
program and continues compiling with the statement following the #INCLUDE
statement.
When program is specified without filename, program must be a record in the same
file as the program containing the #INCLUDE statement.
If program is a record in a different file, the filename must be specified in the
#INCLUDE statement, followed by the name of the program. The filename must
specify a type 1 or type 19 file defined in the VOC file.
You can nest #INCLUDE statements.
The #INCLUDE statement is a synonym for the $INCLUDE and INCLUDE
statements.
Example
    PRINT "START"
    #INCLUDE END
    PRINT "FINISH"
When this program is compiled, the #INCLUDE statement inserts code from the
program END (see the example on theEND statement page). This is the program
output:
    START
    THESE TWO LINES WILL PRINT ONLY
    WHEN THE VALUE OF 'A' IS 'YES'.
$*
Syntax
      $* [comment.text]
Description
Use the $* statement to insert a comment in a BASIC program. Comments explain
or document various parts of a program. They are part of the source code only and
are nonexecutable. They do not affect the size of the object code.
A comment must be a separate BASIC statement and can appear anywhere in a
program. A comment must begin with one of the following comment designators:
      REM    *   !   $*
Any text that appears between a comment designator and the end of a physical line
is treated as part of the comment, not as part of the executable program. If a
comment does not fit on one physical line, you can continue it on the next physical
line only by starting the new line with a comment designator. If a comment
appears at the end of a physical line containing an executable statement, you must
put a semicolon ( ; ) before the comment designator.
Example
The PRINT statement at the end of the third line is not executed because it follows
the exclamation point on the same line and is treated as part of the comment. Lines
4, 5, and 6 show how to include a comment in the same sequence of executable
statements.
      001:   PRINT "HI THERE"; $* Anything after the $* is a comment.
      002:   $* This line is also a comment and does not print.
      003:   IF 5<6 THEN PRINT "YES"; $* A comment; PRINT "PRINT ME"
      004:   IF 5<6 THEN
      005:     PRINT "YES"; $* A comment
      006:     PRINT "PRINT ME"
      007:   END
This is the program output:
      HI THERE
      YES
      YES
      PRINT ME
$CHAIN
Syntax
    $CHAIN [filename] program
Description
Use the $CHAIN statement to direct the compiler to read source code from program
and compile it as if it were part of the current program. The $CHAIN statement
differs from the$INCLUDE, #INCLUDE, and INCLUDE statements in that the
compiler does not return to the main program. Any statements appearing after the
$CHAIN statement are not compiled or executed.
When the program name is specified without a filename, the source code to insert
must be in the same file as the current program.
If the source code to insert is in a different file, the $CHAIN statement must specify
the name of the remote file followed by the program name. filename must specify a
type 1 or type 19 file defined in the VOC file.
When statements in program generate error messages, the messages name the
program containing the $CHAIN statement.
Example
    PRINT "START"
    $CHAIN END
    PRINT "FINISH"
When this program is compiled, the $CHAIN statement inserts code from the
program END (see the example on theEND statement page). This is the program
output:
    START
    THESE TWO LINES WILL PRINT ONLY
    WHEN THE VALUE OF 'A' IS 'YES'.
$COPYRIGHT
Syntax
      $COPYRIGHT "copyright.notice"
Description
Use the $COPYRIGHT statement to specify the copyright information that is
inserted in the copyright field of the object header code.
copyright.notice must be enclosed by single or double quotation marks.
The copyright field in the object code header is set to the empty string at the begin-
ning of compilation. It remains empty until it encounters a $COPYRIGHT
statement in the program.
If more than one $COPYRIGHT statement is included in the program, only the
information included in the last one encountered is inserted in the object code
header.
This statement is included for compatibility with existing software.
$DEFINE
Syntax
    $DEFINE identifier [replacement.text]
Description
Use the $DEFINE statement to define identifiers that control program compilation.
$DEFINE has two functions:
    • Defining an identifier
    • Supplying replacement text for an identifier
identifier is the symbol to be defined. It can be any valid identifier.
replacement.text is a string of characters that the compiler uses to replace identifier
everywhere it appears in the program containing the $DEFINE statement.
When used as a replacement text supplier, $DEFINE adds the specified identifier
and its associated replacement.text to the symbol table. Each time identifier is found
in the program following the $DEFINE statement in which its value was set, it is
replaced by replacement.text. If replacement.text is not specified, identifier is defined
and has a null value.
Separate replacement.text from identifier with one or more blanks. Every character
typed after this blank is added to replacement.text up to, but not including, the
Return character that terminates the replacement.text.
Conditional Compilation
You can use $DEFINE with the $IFDEF or $IFNDEF statement to define an identi-
fier that controls conditional compilation. The syntax is as follows:
Example
In this example the identifier NAME.SUFFIX is defined to have a value of
PROGRAM.NAME[5]. When the compiler processes the next line, it finds the
symbol NAME.SUFFIX, substitutes PROGRAM.NAME[5] in its place and
continues processing with the first character of the replacement text.
      $DEFINE NAME.SUFFIX PROGRAM.NAME[5]
      IF NAME.SUFFIX = '.B' THEN
          .
          .
          .
      END
          .
          .
          .
$EJECT
Syntax
    $EJECT
Description
Use the $EJECT statement to begin a new page in the listing record.
This statement is a synonym for the $PAGE statement.
$IFDEF
Syntax
    $IFDEF identifier
           [ statements ]
    [[ $ELSE ]
           [ statements ]]
    $ENDIF
Description
Use the $IFDEF statement to test for the definition of a compile-time symbol.
$IFDEF tests to see if identifier is currently defined (that is, has appeared in a
$DEFINE statement and has not been undefined).
If identifier is currently defined and the $ELSE clause is omitted, the statements
between the $IFDEF and $ENDIF statements are compiled. If the $ELSE clause is
included, only the statements between $IFDEF and $ELSE are compiled.
If identifier is not defined and the $ELSE clause is omitted, all the lines between the
$IFDEF and $ENDIF statements are ignored. If the $ELSE clause is included, only
the statements between $ELSE and $ENDIF are compiled.
$IFDEF and $IFNDEFstatements can be nested up to 10 deep.
Example
The following example determines if the identifier “modified” is defined:
    $DEFINE modified 0
    $IFDEF modified
       PRINT "modified is defined."
    $ELSE
       PRINT "modified is not defined."
    $ENDIF
$IFNDEF
Syntax
    $IFNDEF identifier
          [ statements ]
    [[ $ELSE ]
           [ statements ]]
    $ENDIF
Description
Use the $IFNDEF statement to test for the definition of a compile-time symbol. The
$IFNDEF statement complements the $IFDEF statement.
If identifier is currently not defined and the $ELSE clause is omitted, the statements
between the $IFNDEF and $ENDIF statements are compiled. If the $ELSE clause is
included, only the statements between $IFNDEF and $ELSE are compiled.
If identifier is defined and the $ELSE clause is omitted, all the lines between the
$IFNDEF and $ENDIF statements are ignored. If the $ELSE clause is included,
only the statements between $ELSE and $ENDIF are compiled.
$IFDEF and $IFNDEF statements can be nested up to 10 deep.
Example
The following example determines if the identifier “modified” is not defined:
    $DEFINE modified 0
    $IFNDEF modified
       PRINT "modified is not defined."
    $ELSE
       PRINT "modified is defined."
    $ENDIF
$INCLUDE
Syntax
    $INCLUDE [filename] program
    $INCLUDE program FROM filename
Description
Use the $INCLUDE statement to direct the compiler to insert the source code in the
record program and compile it with the main program. The $INCLUDE statement
differs from the $CHAIN statement in that the compiler returns to the main
program and continues compiling with the statement following the $INCLUDE
statement.
When program is specified without filename, program must be a record in the same
file as the program currently containing the $INCLUDE statement.
If program is a record in a different file, the filename must be specified in the
$INCLUDE statement, followed by the name of the program. The filename must
specify a type 1 or type 19 file defined in the VOC file.
You can nest $INCLUDE statements.
The $INCLUDE statement is a synonym for the #INCLUDE and INCLUDE
statements.
Example
    PRINT "START"
    $INCLUDE END
    PRINT "FINISH"
When this program is compiled, the $INCLUDE statement inserts code from the
program END (see the example on the END statement page). This is the program
output:
    START
    THESE TWO LINES WILL PRINT ONLY
    WHEN THE VALUE OF 'A' IS 'YES'.
$INSERT
Syntax
    $INSERT primos.pathname
Description
Use the $INSERT statement to direct the compiler to insert the source code
contained in the file specified by primos.pathname and compile it with the main
program. The difference between $INSERT and $INCLUDE (and its synonyms
#INCLUDE and INCLUDE) is that $INSERT takes a PRIMOS pathname as an
argument, whereas $INCLUDE takes a DataStage filename and record ID. The
PRIMOS pathname is converted to a pathname; any leading *> is ignored.
$INSERT is included for compatibility with Prime INFORMATION programs; the
$INCLUDE statement is recommended for general use.
If primos.pathname is the name of the program only, it is interpreted as a relative
pathname. In this case, the program must be a file in the same directory as the
program containing the $INSERT statement.
You can nest $INSERT statements.
primos.pathname is converted to a valid pathname using the following conversion
rules:
    / is converted to ?\
    ? is converted to ??
    ASCII CHAR 0 (NUL) is converted to ?0
     . (period) is converted to ?.
If you specify a full pathname, the > between directory names changes to a / to
yield:
    [pathname/] program
$INSERT uses the transformed argument directly as a pathname of the file
containing the source to be inserted. It does not use the file definition in the VOC
file.
Example
   PRINT "START"
   $INSERT END
   PRINT "FINISH"
When this program is compiled, the $INSERT statement inserts code from the
program END (see the example on the END statement page). This is the program
output:
   START
   THESE TWO LINES WILL PRINT ONLY
   WHEN THE VALUE OF 'A' IS 'YES'.
$MAP
Syntax
    $MAP mapname
Description
In NLS mode, use the $MAP statement to direct the compiler to specify the map
for the source code. Use the $MAP statement if you use embedded literal strings
that contain non-ASCII characters.
mapname must be the name of a map that has been built and installed.
You can use only one $MAP statement during compilation.
Note: You can execute programs that contain only ASCII characters whether NLS
      mode is on or off. You cannot execute programs that contain non-ASCII
      characters that were compiled in NLS mode if NLS mode is switched off.
Example
The following example assigns a string containing the three characters alpha, beta,
and gamma to the variable GREEKABG:
    $MAP MNEMONICS
    .
    .
    .
    GREEKABG = "<A*><B*><G*>"
$OPTIONS
Syntax
    $OPTIONS [flavor] [options]
Description
Use the $OPTIONS statement to set compile-time emulation of any flavor. This
does not allow object code compiled in one flavor to execute in another flavor. You
can select individual options in a program to override the default setting.
Use the following keywords to specify flavor:
Keyword              Flavor
PICK                 Generic Pick emulation
INFORMATION          Prime INFORMATION emulation
REALITY              REALITY emulation
IN2                  Intertechnique emulation
DEFAULT              IDEAL flavor
PIOPEN               PI/open emulation
For instance, the following statement instructs the compiler to treat all BASIC
syntax as if it were running in a PICK flavor account:
    $OPTIONS PICK
Another way to select compile-time emulation is to specify one of the following
keywords in field 6 of the VOC entry for the BASIC command:
    INFORMATION.FORMAT
    PICK.FORMAT
    REALITY.FORMAT
    IN2.FORMAT
    PIOPEN.FORMAT
By default the VOC entry for the BASIC command corresponds with the account
flavor specified when your account was set up.
options are specified by the keywords listed in following table. To turn off an
option, prefix it with a minus sign ( − ).
                          Option
Option Name                      Description
                          Letter
CASE                        none    Differentiates between uppercase and lower-
                                    case identifiers and keywords.
COMP.PRECISION              none    Rounds the number at the current precision
                                    value in any comparison.
COUNT.OVLP                   O      For INDEX and COUNT functions, the count
                                    overlaps.
END.WARN                     R      Prints a warning message if there is no final
                                    END statement.
EXEC.EQ.PERF                 P      Compiles EXECUTE asPERFORM.
EXTRA.DELIM                  W      For INSERT and REPLACE functions, the
                                    compiler handles fields, values, and
                                    subvalues that contain the empty string
                                    differently from the way they are handled in
                                    the IDEAL flavor. In particular, if you specify
                                    a negative one (−1) parameter, INFORMA-
                                    TION and IN2 flavors add another delimiter,
                                    except when starting with an empty string.
FOR.INCR.BEF                 F      Increments the index for FOR…NEXT loop
                                    before instead of after the bound checking.
FORMAT.OCONV                none    Lets output conversion codes be used as
                                    format masks (see the FMT function).
FSELECT                     none    Makes the SELECT statement return the total
                                    number of records selected to the
                                    @SELECTED variable. Using this option can
                                    result in slower performance for the SELECT
                                    statement.
HEADER.BRK                  none    Specifies the PIOPEN flavor for the I and P
                                    options to the HEADING and FOOTING
                                    keywords. This is the default for the PIOPEN
                                    flavor.
                      Option
Option Name                  Description
                      Letter
HEADER.DATE              D     Displays times and dates in headings or foot-
                               ings in fixed format (that is, they do not
                               change from page to page). Dates are
                               displayed in 'D2−' format instead of 'D'
                               format. Allows page number field specifica-
                               tion by multiple invocations of 'P' in a single
                               set of quotation marks.
HEADER.EJECT             H     HEADING statement causes initial page
                               eject.
IN2.SUBSTR               T     Uses IN2 definitions for BASIC substring
                               handling (string[n,m]). If a single parameter is
                               specified, a length of 1 is assumed. The size of
                               the string expands or contracts according to
                               the length of the replacement string.
INFO.ABORT               J     ABORT syntax follows Prime INFORMA-
                               TION instead of PICK.
INFO.CONVERT            none   Specifies that the FMT, ICONV, and OCONV
                               functions perform PI/open style conversions.
INFO.ENTER              none   Specifies the PIOPEN flavor of the ENTER
                               statement.
INFO.INCLUDE            none   Processes any PRIMOS pathnames specified
                               with the $INSERT statement.
INFO.LOCATE              L     LOCATE syntax follows Prime
                               INFORMATION instead of REALITY. The
                               Pick format of the LOCATE statement is
                               always supported.
INFO.MARKS              none   Specifies that the LOWER, RAISE, and
                               REMOVE functions use a smaller range of
                               delimiters for PI/open compatibility.
INFO.MOD                none   Specifies the PIOPEN flavor for the MOD
                               function. This is the default for the PIOPEN
                               flavor.
                        Option
Option Name                    Description
                        Letter
INPUTAT                   none   Specifies the PIOPEN flavor for the INPUT @
                                 statement. This is the default for the PIOPEN
                                 flavor.
INPUT.ELSE                 Y     Accepts an optional THEN…ELSE clause on
                                 INPUT statement.
INT.PRECISION             none   Rounds the integer at the current precision
                                 value in an INT function.
LOCATE.R83                none   A LOCATE statement returns an “AR” or
                                 “DR” sequence value compatible with Pick,
                                 Prime INFORMATION, and PI/open
                                 systems.
NO.CASE                   none   Does not differentiate between uppercase
                                 and lowercase in identifiers or keywords.
                                 This is the default for the PIOPEN flavor.
NO.RESELECT                U     For SELECT and SSELECT statements, active
                                 select list 0 remains active; another selection
                                 or sort is not performed. The next READ-
                                 NEXT statement uses select list 0.
ONGO.RANGE                 G     If the value used in an ON GOTO or ON
                                 GOSUB is out of range, executes the next
                                 statement rather than the first or last branch.
PCLOSE.ALL                 Z     The PRINTER CLOSE statement closes all
                                 print channels.
PERF.EQ.EXEC               C     PERFORM compiles as EXECUTE.
PIOPEN.INCLUDE            none   Processes any PRIMOS pathnames specified
                                 with the $INSERT and $INCLUDE
                                 statements.
PIOPEN.MATREAD            none   Sets the elements of the matrix to empty
                                 strings when the record ID is not found.
                                 MATREAD, MATREADL, and MATREADU
                                 will behave as they do on PI/open systems.
                      Option
Option Name                  Description
                      Letter
PIOPEN.SELIDX           none   In the SELECTINDEX statement, removes
                               multiple occurrences of the same record ID in
                               an index with a multivalued field.
RADIANS                 none   Calculates trigonometric operations using
                               radians instead of degrees.
RAW.OUTPUT              none   Suppresses automatic mapping of system
                               delimiters on output. When an application
                               handles terminal control directly,
                               RAW.OUTPUT turns off this automatic
                               mapping.
READ.RETAIN              Q     If a READ, READU, READV, READVL, or
                               READVU fails, the resulting variable retains
                               its value. The variable is not set to an empty
                               string.
REAL.SUBSTR              K     Uses REALITY flavor definitions for
                               substring handling (string[n,m]). If m or n is
                               less than 0, the starting position for substring
                               extraction is defined as the right side (the
                               end) of the string.
RNEXT.EXPL               X     READNEXT returns an exploded select list.
SEQ.255                  N     SEQ(" ") = 255 (instead of 0).
STATIC.DIM               M     Creates arrays at compile time, not at run
                               time. The arrays are not redimensioned, and
                               they do not have a zero element.
STOP.MSG                 E     Causes STOP andABORT to use the ERRMSG
                               file to produce error messages instead of
                               using the specified text.
SUPP.DATA.ECHO           I     Causes input statements to suppress echo
                               from data.
TIME.MILLISECOND        none   Causes the SYSTEM (12) function to return
                               the current system time in milliseconds, and
                               the TIME function to return the current
                               system time in seconds.
                          Option
Option Name                      Description
                          Letter
ULT.FORMAT                 none    Format operations are compatible with
                                   Ult/ix. For example, FMT("","MR2") returns
                                   an empty string, not 0.00.
USE.ERRMSG                   B     PRINTERR prints error messages from
                                   ERRMSG.
VAR.SELECT                   S     SELECT TO variable creates a local select vari-
                                   able instead of using numbered select lists,
                                   and READLIST reads a saved select list
                                   instead of an active numbered select list.
VEC.MATH                     V     Uses vector arithmetic instructions for oper-
                                   ating on multivalued data. For performance
                                   reasons the IDEAL flavor uses single-valued
                                   arithmetic.
WIDE.IF                    none    Testing numeric values for true or false uses
                                   the wide zero test. In Release 6 of UniVerse,
                                   the WIDE.IF option is OFF by default. In
                                   Release 7, WIDE.IF is ON by default.
You can also set individual options by using special versions of some statements to
override the current setting. These are listed as follows:
The default settings for each flavor are listed in the following table:
Example
   >ED BP OPT
   4 lines long.
   ----: P
   0001: $OPTIONS INFORMATION
   0002: A='12'
   0003: B='14'
   0004: PRINT A,B
   Bottom at line 4
   ----: Q
   >BASIC BP OPT
   Compiling: Source = 'BP/OPT', Object = 'BP.O/OPT'
   Compilation Complete.
   >ED BP OPT
   4 lines long.
   ----: P
   0001: $OPTIONS PICK
   0002: A='12'
   0003: B='14'
   0004: PRINT A,B
   Bottom at line 4
   ----: Q
   >BASIC BP OPT
   Compiling: Source = 'BP/OPT', Object = 'BP.O/OPT'
   Compilation Complete.
$PAGE
The $PAGE statement is a synonym for the $EJECT statement.
$UNDEFINE
Syntax
    $UNDEFINE identifier
Description
Use the $UNDEFINE statement to remove the definition of identifiers set with the
$DEFINE statement. The $UNDEFINE statement removes the definition of identi-
fier from the symbol table if it appeared in a previous $DEFINE statement. If the
identifier was not previously defined, $UNDEFINE has no effect.
identifier is the identifier whose definition is to be deleted from the symbol table.
You can use $UNDEFINE with the $IFDEF or $IFNDEF statement to undefine an
identifier that controls conditional compilation. The syntax is as follows:
    $UNDEFINE identifier
        .
        .
        .
    { $IFDEF | $IFNDEF } identifier
        [ statements ]
    $ELSE
       [ statements ]
    $ENDIF
The $IFDEF statement that begins the conditional compilation block tests identifier
to determine whether it is currently defined. Using this syntax, the $UNDEFINE
statement deletes the definition of identifier from the symbol table, and the state-
ments between the $ELSE and the $ENDIF statements are compiled.
If you use the $IFNDEF statement, on the other hand, and identifier is undefined,
the statements between $IFDEF and $ENDIF are compiled. If identifier is not
defined, the statements between $IFDEF and $ELSE are compiled.
*
Syntax
    * [comment.text]
Description
Use the * statement to insert a comment in a BASIC program. Comments explain
or document various parts of a program. They are part of the source code only and
are nonexecutable. They do not affect the size of the object code.
A comment must be a separate BASIC statement, and can appear anywhere in a
program. A comment must begin with one of the following comment designators:
    REM    *   !   $*
Any text that appears between a comment designator and the end of a physical line
is treated as part of the comment, not as part of the executable program. If a
comment does not fit on one physical line, you can continue it on the next physical
line only by starting the new line with a comment designator. If a comment
appears at the end of a physical line containing an executable statement, you must
put a semicolon ( ; ) before the comment designator.
Example
The PRINT statement at the end of the third line is not executed because it follows
the asterisk on the same line and is treated as part of the comment. Lines 4, 5, and
6 show how to include a comment in the same sequence of executable statements.
    001:   PRINT "HI THERE"; * Anything after the * is a comment
    002:   * This line is also a comment and does not print.
    003:   IF 5<6 THEN PRINT "YES"; * A comment; PRINT "PRINT ME"
    004:   IF 5<6 THEN
    005:     PRINT "YES"; * A comment
    006:     PRINT "PRINT ME"
    007:   END
This is the program output:
    HI THERE
    YES
    YES
    PRINT ME
<>
Syntax
     variable < field# [ ,value# [ ,subvalue# ] ] >
Description
Use the < > operator (angle brackets) to extract or replace elements of a dynamic
array.
variable specifies the dynamic array containing the data to be changed.
field#, value#, and subvalue# are delimiter expressions.
Angle brackets to the left of an assignment operator change the specified data in
the dynamic array according to the assignment operator. For examples, see the
REPLACE function.
Angle brackets to the right of an assignment operator indicate that an EXTRACT
function is to be performed. For examples, see the EXTRACT function.
@
Syntax
    @ (column [,row])
@ (–code [ ,arg ])
Description
Use the @ function with the PRINT statement to control display attributes, screen
display, and cursor postioning.
Note: You can save processing time by assigning the result of a commonly used
      @ function, such as @ (–1), to a variable, rather than reevaluating the func-
      tion each time it is used.
Cursor Positioning
You position the cursor by specifying a screen column and row position using the
syntax @ (column [,row ]). If you do not specify a row, the current row is the default.
The top line is row 0, the leftmost column is column 0. If you specify a column or
row value that is out of range, the effect of the function is undefined.
If you use the @ function to position the cursor, automatic screen pagination is
disabled.
If you want to use mnemonics rather than the code numbers, you can use an insert
file of equate names by specifying either of the following options when you
compile your program:
     $INCLUDE UNIVERSE.INCLUDE ATFUNCTIONS.H
     $INCLUDE SYSCOM ATFUNCTIONS.INS.IBAS (PIOPEN flavor only)
Note: Not all terminal control codes are supported by all terminal types. If the
      current terminal type does not support the code you specified, the function
      returns an empty string. You can use this to test whether your program
      operates correctly on a particular terminal, and whether you need to code
      any alternative actions.
       If you issue multiple video attributes (such as blink and reverse video) at
       the same time, the result is undefined. See the description of the
       @(IT$VIDEO) function for details of additive attributes.
The following table summarizes the characteristics of the terminal control codes,
and the sections following the table give more information on each equate name:
For example:
     PRINT @(IT$VIDEO,IT$HALF+IT$ULINE+IT$REVERSE)
In this example, m is set to 74 (2 + 8 + 64) for half-intensity underline display in
reverse video. Bold, italic, fast blink, and concealed are not supported on all termi-
nals. To set the video attributes half-intensity and underline, specify the following:
     @(-35,10)
In this example, 10 is an additive key composed of 2 (half-intensity) plus 8
(underline).
The color attributes are not additive. Only one foreground color at a time can be
displayed. If a terminal does not support a particular color, a request for that color
should return an empty string.
Scroll Up @(IT$SU)
Moves the entire contents of the display up one line. For m greater than 0, the func-
tion @(IT$SU, m) moves the display up m lines or until the bottom of the display is
reached, whichever occurs first. For each line that is scrolled, the first line is
removed from sight and another line is moved into the last line. This function
works only if the terminal is capable of addressing character positions that do not
all fit on the screen, such that some lines are not displayed. This normally requires
the terminal to be set to vertical two-page mode in the initialization string. The
effect of attempting to scroll the terminal too far is undefined.
initialization string. The effect of attempting to scroll the terminal too far is
undefined.
is received. Programs driving such terminals must not change an attribute in the
middle of a contiguous piece of text. You must leave at least one blank character
position at the point where the attribute changes. The field in the terminal defini-
tion record called xmc is used to specify the number of character positions required for
video attributes. A program can examine this field, and take appropriate action. To do
this, the program must execute GET.TERM.TYPE and examine the @SYSTEM-
.RETURN.CODE variable, or use the definition VIDEO.SPACES from the TERM
INFO.H file.
Many terminals do not clear video attributes automatically when the data on a line
is cleared or deleted. The recommended programming practice is to reposition to
the point at which a start attribute was emitted, and overwrite it with an end
attribute, before clearing the line.
On some terminals you can set up the Clear to End of Line sequence to clear both
data and video attributes. This is done by combining the strings for erase data from
active position to end of line, selecting Graphic Rendition normal, and changing all
video attributes from active position to end of line. Sending the result of the
@(IT$CLEOL) function causes both the visible data on the line to be cleared, and
all video attributes to be set to normal, after the cursor position.
Note: Where possible, you should try to ensure that any sequences that clear data
      also clear video attributes. This may not be the case for all terminal types.
       An exception is @(IT$CS) clear screen. The sequence associated with this
       function should always clear not only all data on the screen but also reset
       any video attributes to normal.
Examples
The following example displays “Demonstration” at column 5, line 20:
    PRINT @(5,20):"Demonstration"
In the next example, the PRINT statement positions the cursor to home, at the top-
left corner of the screen, and clears the screen:
    PRINT @(IT$CS):
The $INCLUDE statement is used to include the ATFUNCTIONS insert file of
equate names. Assignment statements are used to assign the evaluated @ functions
to variables. The variables are used in PRINT statements to produce code that
clears the screen and returns the cursor to its original position; positions the cursor
at column 5, line 20; turns on the reverse video mode; prints the string; and turns
off the reverse video mode.
    $INCLUDE UNIVERSE.INCLUDE ATFUNCTIONS.H
    CLS = @(IT$CS)
    REVERSE.ON = @(IT$SREV)
    REVERSE.OFF = @(IT$EREV)
    .
    .
    .
    PRINT CLS: @(5,20):
    PRINT REVERSE.ON:"THIS IS REVERSE VIDEO":REVERSE.OFF
The next example displays any following text in yellow letters:
    PRINT @(IT$FCOLOR, IT$YELLOW)
The next example displays any following text on a cyan background:
    PRINT @(IT$BCOLOR, IT$CYAN)
The next example gives a yellow foreground, not a green foreground, because
color changes are not additive:
    PRINT @(IT$FCOLOR, IT$BLUE):@(IT$FCOLOR, IT$YELLOW)
If you have a terminal that supports colored letters on a colored background, the
next example displays the text “Hello” in yellow on a cyan background. All subse-
quent output is in yellow on cyan until another color @ function is used. If your
color terminal cannot display colored foreground on colored background, only the
last color command is used, so that this example displays the text “Hello” in
yellow on a black background.
    PRINT @(IT$BCOLOR,IT$CYAN):@(IT$FCOLOR,IT$YELLOW):"Hello"
If your color terminal cannot display colored foreground on colored background,
the previous example displays the text “Hello” in black on a cyan background.
The next example gives the same result as the previous example for a terminal that
supports colored letters on a colored background. Strings containing the @ func-
tions can be interpreted as a sequence of instructions, which can be stored for
subsequent frequent reexecution.
    PRINT @(IT$FCOLOR,IT$YELLOW):@(IT$BCOLOR,IT$CYAN):"Hello"
In the last example, the screen is cleared, the cursor is positioned to the tenth
column in the tenth line, and the text “Hello” is displayed in foreground color
cyan. The foreground color is then changed to white for subsequent output. This
sequence of display instructions can be executed again, whenever it is required, by
a further PRINT SCREEN statement.
    SCREEN = @(IT$CS):@(10,10):@(IT$FCOLOR,IT$CYAN):"Hello"
    SCREEN = SCREEN:@(IT$FCOLOR,IT$WHITE)
    PRINT SCREEN
[]
Syntax
     expression [ [ start, ] length ]
Description
Use the [ ] operator (square brackets) to extract a substring from a character string.
The bold brackets are part of the syntax and must be typed.
expression evaluates to any character string.
start is an expression that evaluates to the starting character position of the
substring. If start is 0 or a negative number, the starting position is assumed to be
1. If you omit start, the starting position is calculated according to the following
formula:
     string.length – substring.length + 1
This lets you specify a substring consisting of the last n characters of a string
without having to calculate the string length.
If start exceeds the number of characters in expression, an empty string results. An
empty string also results if length is 0 or a negative number. If the sum of start and
length exceeds the number of characters in the string, the substring ends with the
last character of the string.
length is an expression that evaluates to the length of the substring.
Use the second syntax to return a substring located between the specified number
of occurrences of the specified delimiter. This syntax performs the same function
as the FIELD function.
delimiter can be any string, including field mark, value mark, and subvalue mark
characters. It delimits the start and end of the substring (all that appears within the
two delimiters). If delimiter consists of more than one character, only the first char-
acter is used.
occurrence specifies which occurrence of the delimiter is to be used as a terminator.
If occurrence is less than 1, 1 is assumed.
fields specifies the number of successive fields after the delimiter specified by occur-
rence that are to be returned with the substring. If the value of fields is less than 1, 1
is assumed. The delimiter is part of the returned value in the successive fields.
If the delimiter or the occurrence specified does not exist within the string, an
empty string is returned. If occurrence specifies 1 and no delimiter is found, the
entire string is returned.
If expression is the null value, any substring extracted from it will also be the null
value.
Examples
In the following example (using the second syntax) the fourth # is the terminator
of the substring to be extracted, and one field is extracted:
    A="###DHHH#KK"
    PRINT A["#",4,1]
This is the result:
    DHHH
The following syntaxes specify substrings that start at character position 1:
expression [ 0, length ]
A[3] replaces the last three characters of A (345) with the newly assigned value for
that substring (1212).
The FIELDSTORE function provides the same functionality as assigning the three-
argument syntax of the [ ] operator.
ABORT
Syntax
    ABORT [expression …]
ABORTE [expression …]
ABORTM [expression …]
Description
Use the ABORT statement to terminate execution of a BASIC program and return
to the command prompt. ABORT differs from STOP in that a STOP statement
returns to the calling environment (for example, a menu, a paragraph, another
BASIC program following an EXECUTE statement, and so on), whereas ABORT
terminates all calling environments as well as the BASIC program. You can use it
as part of an IF…THEN statement to terminate processing if certain conditions
exist.
If expression is used, it is printed when the program terminates. If expression evalu-
ates to the null value, nothing is printed.
The ABORTE statement is the same as the ABORT statement except that it behaves
as if $OPTIONS STOP.MSG were in force. This causes ABORT to use the ERRMSG
file to produce error messages instead of using the specified text. If expression in the
ABORTE statement evaluates to the null value, the default error message is
printed:
    Message ID is NULL:         undefined error
For information about the ERRMSG file, see the ERRMSG statement.
The ABORTM statement is the same as the ABORT statement except that it
behaves as if $OPTIONS −STOP.MSG were in force. This causes ABORT to use the
specified text instead of text from the ERRMSG file.
Example
    PRINT "DO YOU WANT TO CONTINUE?":
    INPUT A
    IF A="NO" THEN ABORT
This is the program output:
    DO YOU WANT TO CONTINUE?NO
    Program "TEST": Line 3, Abort.
ABS
Syntax
      ABS (expression)
Description
Use the ABS function to return the absolute value of any numeric expression. The
absolute value of an expression is its unsigned magnitude. If expression is negative,
the value returned is:
      −expression
For example, the absolute value of −6 is 6.
If expression is positive, the value of expression is returned. If expression evaluates to
the null value, null is returned.
Example
      Y = 100
      X = ABS(43-Y)
      PRINT X
This is the program output:
      57
ABSS
Syntax
       ABSS (dynamic.array)
Description
Use the ABSS function to return the absolute values of all the elements in a
dynamic array. If an element in dynamic.array is the null value, null is returned for
that element.
Example
       Y = REUSE(300)
       Z = 500:@VM:400:@VM:300:@SM:200:@SM:100
       A = SUBS(Z,Y)
       PRINT A
       PRINT ABSS(A)
This is the program output:
       200V100V0S-100S-200
       200V100V0S100S200
ACOS
Syntax
    ACOS (expression)
Description
Use the ACOS function to return the trigonometric arc-cosine of expression. expres-
sion must be a numeric value. The result is expressed in degrees. If expression
evaluates to the null value, null is returned. The ACOS function is the inverse of
the COS function.
Example
    PRECISION 5
    PRINT "ACOS(0.707106781) = ":ACOS(0.707106781):" degrees"
This is the program output:
    ACOS(0.707106781) = 45 degrees
ADDS
Syntax
    ADDS (array1, array2)
    CALL −ADDS (return.array, array1, array2)
    CALL !ADDS (return.array, array1, array2)
Description
Use the ADDS function to create a dynamic array of the element-by-element addi-
tion of two dynamic arrays.
Each element of array1 is added to the corresponding element of array2. The result
is returned in the corresponding element of a new dynamic array. If an element of
one array has no corresponding element in the other array, the existing element is
returned. If an element of one array is the null value, null is returned for the sum
of the corresponding elements.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    A = 2:@VM:4:@VM:6:@SM:10
    B = 1:@VM:2:@VM:3:@VM:4
    PRINT ADDS(A,B)
This is the program output:
    3V6V9S10V4
ALPHA
Syntax
    ALPHA (expression)
Description
Use the ALPHA function to determine whether expression is an alphabetic or
nonalphabetic string. If expression contains the characters a through z or A through
Z, it evaluates to true and a value of 1 is returned. If expression contains any other
character or an empty string, it evaluates to false and a value of 0 is returned. If
expression evaluates to the null value, null is returned.
If NLS is enabled, the ALPHA function uses the characters in the Alphabetics field
in the NLS.LC.CTYPE file. For more information, see the DataStage NLS Guide.
Example
    PRINT   "ALPHA('ABCDEFG') = ":ALPHA('ABCDEFG')
    PRINT   "ALPHA('abcdefg') = ":ALPHA('abcdefg')
    PRINT   "ALPHA('ABCDEFG.') = ":ALPHA('ABCDEFG.')
    PRINT   "ALPHA('SEE DICK') = ":ALPHA('SEE DICK')
    PRINT   "ALPHA('4 SCORE') = ":ALPHA('4 SCORE')
    PRINT   "ALPHA('') = ":ALPHA('')
This is the program output:
    ALPHA('ABCDEFG') = 1
    ALPHA('abcdefg') = 1
    ALPHA('ABCDEFG.') = 0
    ALPHA('SEE DICK') = 0
    ALPHA('4 SCORE') = 0
    ALPHA('') = 0
ANDS
Syntax
    ANDS (array1, array2)
    CALL −ANDS (return.array, array1, array2)
    CALL !ANDS (return.array, array1, array2)
Description
Use the ANDS function to create a dynamic array of the logical AND of corre-
sponding elements of two dynamic arrays.
Each element of the new dynamic array is the logical AND of the corresponding
elements of array1 and array2. If an element of one dynamic array has no corre-
sponding element in the other dynamic array, a false (0) is returned for that
element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If both corresponding elements of array1 and array2 are the null value, null is
returned for those elements. If one element is the null value and the other is 0 or an
empty string, a false is returned for those elements.
Example
    A = 1:@SM:4:@VM:4:@SM:1
    B = 1:@SM:1-1:@VM:2
    PRINT ANDS(A,B)
This is the program output:
    1S0V1S0
ASCII
Syntax
    ASCII (expression)
Description
Use the ASCII function to convert each character of expression from its EBCDIC
representation value to its ASCII representation value. If expression evaluates to the
null value, null is returned.
The ASCII function and the EBCDIC function perform complementary operations.
Example
    X = EBCDIC('ABC 123')
    Y = ASCII(X)
    PRINT "EBCDIC", "ASCII", " Y "
    PRINT "------", "-----", "---"
    FOR I = 1 TO LEN (X)
    PRINT SEQ(X[I,1]) , SEQ(Y[I,1]),Y[I,1]
    NEXT I
This is the program output:
    EBCDIC     ASCII         Y
    ------     -----        ---
    193        65           A
    194        66           B
    195        67           C
    64         32
    241        49           1
    242        50           2
    243        51           3
ASIN
Syntax
       ASIN (expression)
Description
Use the ASIN function to return the trigonometric arc-sine of expression. expression
must be a numeric value. The result is expressed in degrees. If expression evaluates
to the null value, null is returned. The ASIN function is the inverse of the SIN
function.
Example
       PRECISION 5
       PRINT "ASIN(0.707106781) = ":ASIN(0.707106781):" degrees"
This is the program output:
       ASIN(0.707106781) = 45 degrees
ASSIGNED
Syntax
    ASSIGNED (variable)
Description
Use the ASSIGNED function to determine if variable is assigned a value.
ASSIGNED returns 1 (true) if variable is assigned a value, including common vari-
ables and the null value. It returns 0 (false) if variable is not assigned a value.
PICK Flavor
When you run in a PICK flavor account, all common variables are initially unas-
signed. ASSIGNED returns 0 (false) for common variables until the program
explicitly assigns them a value.
Example
    A = "15 STATE STREET"
    C = 23
    X = ASSIGNED(A)
    Y = ASSIGNED(B)
    Z = ASSIGNED(C)
    PRINT X,Y,Z
This is the program output:
    1      0   1
Assignment Statements
Syntax
     variable = expression
=
     variable += expression
     variable −= expression
     variable := expression
Description
Use assignment statements to assign a value to a variable. The variable can be
currently unassigned (that is, one that has not been assigned a value by an assign-
ment statement, a READ statement, or any other statement that assigns values to
variables) or have an old value that is to be replaced. The assigned value can be a
constant or an expression. It can be any data type (that is, numeric, character string,
or the null value).
Use the operators += , −= , and := to alter the value of a variable. The += operator
adds the value of expression to variable. The −= operator subtracts the value of
expression from variable. The := operator concatenates the value of expression to the
end of variable.
Use the system variable @NULL to assign the null value to a variable:
     variable = @NULL
Use the system variable @NULL.STR to assign a character string containing only
the null value (more accurately, the character used to represent the null value) to a
variable:
     variable = @NULL.STR
Example
     EMPL=86
     A="22 STAGECOACH LANE"
     X='$4,325'
     B=999
     PRINT "A= ":A,"B= ":B,"EMPL= ":EMPL
     B+=1
     PRINT "X= ":X,"B= ":B
ATAN
Syntax
    ATAN (expression)
Description
Use the ATAN function to return the trigonometric arc-tangent of expression. expres-
sion must be a numeric value. The result is expressed in degrees. If expression
evaluates to the null value, null is returned. The ATAN function is the inverse of
the TAN function.
Examples
The following example prints the numeric value 135 and the angle, in degrees, that
has an arc-tangent of 135:
    PRINT 135, ATAN(135)
The next example finds what angle has an arc-tangent of 1:
    X = ATAN(1)
    PRINT 1, X
This is the program output:
    135        89.5756
    1          45
AUTHORIZATION
Syntax
    AUTHORIZATION "username"
Description
Use the AUTHORIZATION statement to specify or change the effective run-time
user of a program. After an AUTHORIZATION statement is executed, any SQL
security checking acts as if username is running the program.
username is a valid login name on the machine where the program is run. username
must be a constant. username is compiled as a character string whose user identifi-
cation (UID) number is looked up in the /etc/passwd file at run time.
An AUTHORIZATION statement changes only the user name that is used for SQL
security checking while the program is running. It does not change the actual user
name, nor does it change the user’s effective UID at the operating system level. If
a program does not include an AUTHORIZATION statement, it runs with the user
name of the user who invokes it.
You can change the effective user of a program as many times as you like. The user-
name specified by the most recently executed AUTHORIZATION statement
remains in effect for subsequent EXECUTE and PERFORM statements as well as
for subroutines.
When a file is opened, the effective user’s permissions are stored in the file vari-
able. These permissions apply whenever the file variable is referenced, even if a
subsequent AUTHORIZATION statement changes the effective user name.
The effective user name is stored in the system variable @AUTHORIZATION.
A program using the AUTHORIZATION statement must be compiled on the
machine where the program is to run. To compile the AUTHORIZATION state-
ment, SQL DBA privilege is required. If the user compiling the program does not
have DBA privilege, the program will not be compiled. You cannot run the
program on a machine different from the one where it was compiled. If you try, the
program terminates with a fatal error message.
Example
    AUTHORIZATION "susan"
    OPEN "","SUES.FILE" TO FILE.S ELSE PRINT "CAN'T OPEN SUES.FILE"
    AUTHORIZATION "bill"
AUXMAP
Syntax
    AUXMAP { ON | OFF | expression }
Description
In NLS mode, use the AUXMAP statement to associate an auxiliary device with a
terminal.
AUXMAP ON causes subsequent PRINT statements directed to print channel 0 to
use the auxiliary map. If no auxiliary map is defined, the terminal map is used.
AUXMAP OFF causes subsequent PRINT statements to use the terminal map. OFF
is the default. If expression evaluates to true, AUXMAP is turned on. If expression
evaluates to false, AUXMAP is turned off.
A program can access the map for an auxiliary device only by using the AUXMAP
statement. Other statements used for printing to the terminal channel, such as
CRT, PRINT, or INPUTERR, use the terminal map.
If NLS is not enabled and you execute the AUXMAP statement, the program
displays a run-time error message. For more information, see the DataStage NLS
Guide.
Use the BEGIN CASE statement to begin a set of CASE statements. For details, see
the statement.
BEGIN TRANSACTION
Syntax
     BEGIN TRANSACTION [ISOLATION LEVEL level]
             [statements]
Description
Use the BEGIN TRANSACTION statement to indicate the beginning of a
transaction.
The ISOLATION LEVEL clause sets the transaction isolation level for the duration
of that transaction. The isolation level reverts to the original value at the end of the
transaction.
level is an expression that evaluates to one of the following:
     • An integer from 0 through 4
     • One of the following keywords
Examples
The following examples both start a transaction at isolation level 3:
     BEGIN TRANSACTION ISOLATION LEVEL REPEATABLE.READ
     BEGIN TRANSACTION ISOLATION LEVEL 3
BITAND
Syntax
    BITAND (expression1, expression2)
Description
Use the BITAND function to perform the bitwise AND comparison of two integers
specified by numeric expressions. The bitwise AND operation compares two inte-
gers bit by bit. It returns a bit of 1 if both bits are 1; otherwise it returns a bit of 0.
If either expression1 or expression2 evaluates to the null value, null is returned.
Noninteger values are truncated before the operation is performed.
The BITAND operation is performed on a 32-bit twos-complement word.
Note: Differences in hardware architecture can make the use of the high-order bit
      nonportable.
Example
    PRINT BITAND(6,12)
    * The binary value of 6 =             0110
    * The binary value of 12 =            1100
This results in 0100, and the following output is displayed:
    4
BITNOT
Syntax
    BITNOT (expression [,bit#])
Description
Use the BITNOT function to return the bitwise negation of an integer specified by
any numeric expression.
bit# is an expression that evaluates to the number of the bit to invert. If bit# is
unspecified, BITNOT inverts each bit. It changes each bit of 1 to a bit of 0 and each
bit of 0 to a bit of 1. This is equivalent to returning a value equal to the following:
    (−expression)−1
If expression evaluates to the null value, null is returned. If bit# evaluates to the null
value, the BITNOT function fails and the program terminates with a run-time error
message.
Noninteger values are truncated before the operation is performed.
The BITNOT operation is performed on a 32-bit twos-complement word.
Note: Differences in hardware architecture can make the use of the high-order bit
      nonportable.
Example
    PRINT BITNOT(6),BITNOT(15,0),BITNOT(15,1),BITNOT(15,2)
This is the program output:
    −7 14 13 11
BITOR
Syntax
    BITOR (expression1, expression2)
Description
Use the BITOR function to perform the bitwise OR comparison of two integers
specified by numeric expressions. The bitwise OR operation compares two inte-
gers bit by bit. It returns the bit 1 if the bit in either or both numbers is 1; otherwise
it returns the bit 0.
If either expression1 or expression2 evaluates to the null value, null is returned.
Noninteger values are truncated before the operation is performed.
The BITOR operation is performed on a 32-bit twos-complement word.
Note: Differences in hardware architecture can make the use of the high-order bit
      nonportable.
Example
    PRINT BITOR(6,12)
    * Binary value of 6 = 0110
    * Binary value of 12 = 1100
This results in 1110, and the following output is displayed:
    14
BITRESET
Syntax
    BITRESET (expression, bit#)
Description
Use the BITRESET function to reset to 0 the bit number of the integer specified by
expression. Bits are counted from right to left. The number of the rightmost bit is 0.
If the bit is 0, it is left unchanged.
If expression evaluates to the null value, null is returned. If bit# evaluates to the null
value, the BITRESET function fails and the program terminates with a run-time
error message.
Noninteger values are truncated before the operation is performed.
Example
    PRINT   BITRESET(29,0),BITRESET(29,3)
    * The   binary value of 29 = 11101
    * The   binary value of 28 = 11100
    * The   binary value of 21 = 10101
    PRINT BITRESET(2,1),BITRESET(2,0)
    * The binary value of 2 = 10
    * The binary value of 0 = 0
This is the program output:
    28 21
    0 2
BITSET
Syntax
    BITSET (expression, bit#)
Description
Use the BITSET function to set to 1 the bit number of the integer specified by expres-
sion. The number of the rightmost bit is 0. If the bit is 1, it is left unchanged.
If expression evaluates to the null value, null is returned. If bit# evaluates to the null
value, the BITSET function fails and the program terminates with a run-time error
message.
Noninteger values are truncated before the operation is performed.
Example
    PRINT   BITSET(20,0),BITSET(20,3)
    * The   binary value of 20 = 10100
    * The   binary value of 21 = 10101
    * The   binary value of 28 = 11100
    PRINT BITSET(2,0),BITSET(2,1)
    * The binary value of 2 = 10
    * The binary value of 3 = 11
This is the program output:
    21 28
    3 2
BITTEST
Syntax
    BITTEST (expression, bit#)
Description
Use the BITTEST function to test the bit number of the integer specified by expres-
sion. The function returns 1 if the bit is set; it returns 0 if it is not. Bits are counted
from right to left. The number of the rightmost bit is 0.
If expression evaluates to the null value, null is returned. If bit# evaluates to null, the
BITTEST function fails and the program terminates with a run-time error message.
Noninteger values are truncated before the operation is performed.
Example
    PRINT BITTEST(11,0),BITTEST(11,1),BITTEST(11,2),BITTEST(11,3)
    * The binary value of 11 = 1011
This is the program output:
    1     1   0   1
BITXOR
Syntax
    BITXOR (expression1, expression2)
Description
Use the BITXOR function to perform the bitwise XOR comparison of two integers
specified by numeric expressions. The bitwise XOR operation compares two inte-
gers bit by bit. It returns a bit 1 if only one of the two bits is 1; otherwise it returns
a bit 0.
If either expression1 or expression2 evaluates to the null value, null is returned.
Noninteger values are truncated before the operation is performed.
The BITXOR operation is performed on a 32-bit twos-complement word.
Note: Differences in hardware architecture can make the use of the high-order bit
      nonportable.
Example
    PRINT BITXOR(6,12)
    * Binary value of 6 = 0110
    * Binary value of 12 = 1100
This results in 1010, and the following output is displayed:
    10
BREAK
Syntax
    BREAK [KEY] { ON | OFF | expression }
Description
Use the BREAK statement to enable or disable the Intr, Quit, and Susp keys on the
keyboard.
When the BREAK ON statement is in effect, pressing Intr, Quit, or Susp causes
operations to pause.
When the BREAK OFF statement is in effect, pressing Intr, Quit, or Susp has no
effect. This prevents a break in execution of programs that you do not want
interrupted.
When expression is used with the BREAK statement, the value of expression deter-
mines the status of the Intr, Quit, and Susp keys. If expression evaluates to false (0,
an empty string, or the null value), the Intr, Quit, and Susp keys are disabled. If
expression evaluates to true (not 0, an empty string, or the null value), the Intr, Quit,
and Susp keys are enabled.
A counter is maintained for the BREAK statement. It counts the number of
executed BREAK ON and BREAK OFF commands. When program control
branches to a subroutine, the value of the counter is maintained; it is not set back
to 0. For each BREAK ON statement executed, the counter decrements by 1; for
each BREAK OFF statement executed, the counter increments by 1. The counter
cannot go below 0. The Intr, Quit, and Susp keys are enabled only when the value
of the counter is 0. The following example illustrates the point:
Examples
The following example increases the counter by 1:
    BREAK KEY OFF
The following example decreases the counter by 1:
    BREAK KEY ON
The following example disables the Intr, Quit, and Susp keys if QTY is false, 0, an
empty string, or the null value; it enables them if QTY is true, not 0, not an empty
string, or not the null value:
    BREAK QTY ;*
BSCAN
Syntax
    BSCAN ID.variable [ ,rec.variable ] [FROM [ file.variable [ ,record ]
             [USING indexname] [RESET] [BY seq]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the BSCAN statement to scan the leaf nodes of a B-tree file (type 25) or of a
secondary index. The record ID returned by the current scan operation is assigned
to ID.variable. If you specify rec.variable, the contents of the record whose ID is
ID.variable is assigned to it.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information on default files, see the OPEN statement). If the file
is neither accessible nor open, the program terminates with a run-time error
message.
record is an expression that evaluates to a record ID of a record in the B-tree file. If
the USING clause is used, record is a value in the specified index. record specifies
the relative starting position of the scan.
record need not exactly match an existing record ID or value. If it does not, the scan
finds the next or previous record ID or value, depending on whether the scan is in
ascending or descending order. For example, depending on how precisely you
want to specify the starting point at or near the record ID or value SMITH, record
can evaluate to SMITH, SMIT, SMI, SM, or S.
If you do not specify record, the scan starts at the leftmost slot of the leftmost leaf,
or the rightmost slot of the rightmost leaf, depending on the value of the seq expres-
sion. The scan then moves in the direction specified in the BY clause.
indexname is an expression that evaluates to the name of a secondary index associ-
ated with the file.
RESET resets the internal B-tree scan pointer. If the scanning order is ascending, the
pointer is set to the leftmost slot of the leftmost leaf; if the order is descending, the
pointer is set to the rightmost slot of the rightmost leaf. If you do not specify seq,
the scan is done in ascending order. If you specify record in the FROM clause,
RESET is ignored.
0   The scan proceeded beyond the leftmost or rightmost leaf node. ID.variable
    and rec.variable are set to empty strings.
1   The scan returned an existing record ID, or a record ID that matches the
    record ID specified by record.
2   The scan returned a record ID that does not match record. ID.variable is either
    the next or the previous record ID in the B-tree, depending on the direction of
    the scan.
3   The file is not a B-tree (type 25) file, or, if the USING clause is used, the file has
    no active secondary indexes.
4   indexname does not exist.
5   seq does not evaluate to A or D.
6   The index specified by indexname needs to be built.
10 An internal error was detected.
If NLS is enabled, the BSCAN statement retrieves record IDs in the order deter-
mined by the active collation locale; otherwise, BSCAN uses the default order,
which is simple byte ordering that uses the standard binary value for characters;
the Collate convention as specified in the NLS.LC.COLLATE file for the current
locale is ignored. For more information about collation, see the DataStage NLS
Guide.
Example
The following example shows how you might indicate that the ELSE statements
were executed because the contents of the leaf nodes were exhausted:
   BSCAN ID,REC FROM FILE,MATCH USING "PRODUCT" BY "A" THEN
      PRINT ID,REC
   END ELSE
      ERR = STATUS()
      BEGIN CASE
          CASE ERR = 0
             PRINT "Exhausted leaf node contents."
          CASE ERR = 3
             PRINT "No active indices, or file is not type 25."
            CASE ERR = 4
                PRINT "Index name does not exist."
            CASE ERR = 5
                PRINT "Invalid BY clause value."
            CASE ERR = 6
                PRINT "Index must be built."
            CASE ERR = 10
                PRINT "Internal error detected."
         END CASE
         GOTO EXIT.PROGRAM:
   END
BYTE
Syntax
       BYTE (expression)
Description
In NLS mode, use the BYTE function to generate a byte from the numeric value of
expression. BYTE returns a string containing a single byte.
If expression evaluates to a value in the range 0 to 255, a single-byte character is
returned. If expression evaluates to a value in the range 0x80 to 0xF7, a byte that is
part of a multibyte character is returned.
If NLS is not enabled, BYTE works like the CHAR function. For more information,
see the DataStage NLS Guide.
Example
When NLS is enabled, the BYTE and CHAR functions return the following:
BYTELEN
Syntax
    BYTELEN (expression)
Description
In NLS mode, use the BYTELEN function to generate the number of bytes
contained in the ASCII string value in expression.
The bytes in expression are counted, and the count is returned. If expression evalu-
ates to the null value, null is returned.
If NLS is not enabled, BYTELEN works like the LEN function. For more informa-
tion, see the DataStage NLS Guide.
BYTETYPE
Syntax
     BYTETYPE (value)
Description
In NLS mode, use the BYTETYPE function to determine the function of a byte in
value.
If value is from 0 to 255, the BYTETYPE function returns a number that corresponds
to the following:
BYTEVAL
Syntax
     BYTEVAL (expression [, n ] )
Description
In NLS mode, use the BYTEVAL function to examine the bytes contained in the
internal string value of expression. The BYTEVAL function returns a number from
0 through 255 as the byte value of n in expression. If you omit n, 1 is assumed.
If an error occurs, the BYTEVAL function returns a number that corresponds to the
following conditions:
BYTEVAL behaves the same whether NLS is enabled or not. For more information,
see the DataStage NLS Guide.
CALL
Syntax
    CALL name [ ( [MAT] argument [ , [MAT] argument …] ) ]
    variable = 'name'
    CALL @variable [ ( [MAT] argument [ , [MAT] argument …] ) ]
Description
Use the CALL statement to transfer program control from the calling program to
an external subroutine or program that has been compiled and cataloged.
Locally cataloged subroutines can be called directly. Specify name using the exact
name under which it was cataloged. For more details, see The CATALOG
Command on page 3-12.
External subroutines can be called directly or indirectly. To call a subroutine indi-
rectly, the name under which the subroutine is cataloged must be assigned to a
variable or to an element of an array. This variable name or array element specifier,
prefixed with an at sign (@), is used as the operand of the CALL statement.
The first time a CALL is executed, the system searches for the subroutine in a cata-
loged library and changes a variable that contains the subroutine name to contain
its location information instead. This procedure eliminates the need to search the
catalog again if the same subroutine is called later in the program. For indirect
calls, the variable specified in the CALL as the @variable is used; for direct calls, an
internal variable is used. With the indirect method, it is best to assign the subrou-
tine name to the variable only once in the program, not every time the indirect
CALL statement is used.
arguments are variables, arrays, array variables, expressions, or constants that
represent actual values. You can pass one or more arguments from the calling
program to a subroutine. The number of arguments passed in a CALL statement
must equal the number of arguments specified in the SUBROUTINE statement
that identifies the subroutine. If multiple arguments are passed, they must be sepa-
rated by commas. If an argument requires more than one physical line, use a
comma at the end of the line to indicate that the list continues.
If argument is an array, it must be preceded by the MAT keyword, and the array
should be named and dimensioned in both the calling program and the subroutine
before using this statement. If the array is not dimensioned in the subroutine, it
must be declared using the MAT keyword in the SUBROUTINE statement. Other
arguments can be passed at the same time regardless of the size of the array.
The actual values of arguments are not passed to the subroutine. Instead, a pointer
to the location of each argument is passed. Passing a pointer instead of the values
is more efficient when many values need to be passed to the subroutine. This
method of passing arguments is called passing by reference; passing actual values is
called passing by value.
All scalar and matrix variables are passed to subroutines by reference. If you want
to pass variables by value, enclose them in parentheses. When data is passed by
value, the contents of the variable in the main program do not change as a result of
manipulations to the data in the subroutine. When data is passed by reference, the
memory location of the variable is changed by manipulations in both the main
program and the subroutines. Constants are passed to subroutines by value.
When an array is passed to an external subroutine as an argument in a CALL state-
ment, any dimensions assigned to the array in the subroutine are ignored. The
dimensions of the original array as it exists in the calling program are maintained.
Therefore, it is a common and acceptable practice to dimension the array in the
subroutine with subscripts or indices of one. For example, you could dimension
the arrays in the subroutine as follows:
    DIM A (1), B (1, 1), C (1, 1)
When the corresponding array arguments are passed from the calling program to
the subroutine at run time, arrays A, B, and C inherit the dimensions of the arrays
in the calling program. The indices in the DIMENSION statement are ignored.
A better way to declare array arguments in a subroutine is to use the MAT
keyword of the SUBROUTINE statement in the first line of the subroutine. The
following example tells the subroutine to expect the three arrays A, B, and C:
    SUBROUTINE X(MAT A, MAT B, MAT C)
When a RETURN statement is encountered in the subroutine, or when execution
of the subroutine ends without encountering a RETURN statement, control returns
to the statement following the CALL statement in the calling program. For more
details, see the RETURN statement.
Examples
The following example calls the local subroutine SUB. It has no arguments.
    CALL SUB
The following example calls the local subroutine QTY.ROUTINE with three
arguments:
    CALL QTY.ROUTINE(X,Y,Z)
The following example calls the subroutine cataloged as *PROGRAM.1 with six
arguments. The argument list can be expressed on more than one line.
    AAA="*PROGRAM.1"
    CALL @AAA(QTY,SLS,ORDER,ANS,FILE.O,SEQ)
The following example calls the subroutine *MA with three arguments. Its index
and three arguments are passed.
    STATE.TAX(1,2)='*MA'
    CALL @STATE.TAX(1,2)(EMP.NO,GROSS,NET)
The following example calls the subroutine cataloged as *SUB and two matrices
are passed to two subroutine matrices. A third, scalar, argument is also passed.
    GET.VALUE="*SUB"
    DIM QTY(10)
    DIM PRICE(10)
    CALL @GET.VALUE( MAT QTY,MAT PRICE,COST )
The following example shows the SUBROUTINE statement in the subroutine SUB
that is called by the preceding example. The arrays Q and P need not be dimen-
sioned in the subroutine.
    SUBROUTINE SUB( MAT Q,MAT P,C )
Syntax
    BEGIN CASE
       CASE expression
          statements
       [ CASE expression
          statements
               .
               .
               .         ]
    END CASE
Description
Use the CASE statement to alter the sequence of instruction execution based on the
value of one or more expressions. If expression in the first CASE statement is true,
the following statements up to the next CASE statement are executed. Execution
continues with the statement following the END CASE statement.
If the expression in a CASE statement is false, execution continues by testing the
expression in the next CASE statement. If it is true, the statements following the
CASE statement up to the next CASE or END CASE statement are executed. Execu-
tion continues with the statement following the END CASE statement.
If more than one CASE statement contains a true expression, only the statements
following the first such CASE statement are executed. If no CASE statements are
true, none of the statements between the BEGIN CASE and END CASE statements
are executed.
If an expression evaluates to the null value, the CASE statement is considered false.
Use the ISNULL function with the CASE statement when you want to test whether
the value of a variable is the null value. This is the only way to test for the null
value since null cannot be equal to any value, including itself. The syntax is:
    CASE ISNULL (expression)
Use an expression of the constant "1" to specify a default CASE to be executed if
none of the other CASE expressions evaluate to true.
Examples
In the following example NUMBER is equal to 3. CASE 1 is always true, therefore
control is transferred to subroutine 30. Once the subroutine RETURN is executed,
control proceeds to the statement following the END CASE statement.
    NUMBER=3
    BEGIN CASE
        CASE NUMBER=1
           GOTO 10
        CASE 1
           GOSUB 30
        CASE NUMBER<3
           GOSUB 20
    END CASE
    PRINT 'STATEMENT FOLLOWING END CASE'
    GOTO 50
    10*
    PRINT 'LABEL 10'
    STOP
    20*
    PRINT 'LABEL 20'
    RETURN
    30*
    PRINT 'LABEL 30'
    RETURN
    50*
This is the program output:
    LABEL 30
    STATEMENT FOLLOWING END CASE
In the following example, control proceeds to the statement following the END
CASE because 'NAME' does not meet any of the conditions:
    NAME="MICHAEL"
    BEGIN CASE
       CASE NAME[1,2]='DA'
           PRINT NAME
           GOTO 10
       CASE NAME[1,2]='RI'
           PRINT NAME
           GOSUB 20
       CASE NAME[1,2]='BA'
           PRINT NAME
           GOSUB 30
    END CASE
    PRINT 'NO MATCH'
    STOP
This is the program output:
    NO MATCH
CATS
Syntax
       CATS (array1, array2)
       CALL −CATS (return.array, array1, array2)
       CALL !CATS (return.array, array1, array2)
Description
Use the CATS function to create a dynamic array of the element-by-element
concatenation of two dynamic arrays.
Each element of array1 is concatenated with the corresponding element of array2.
The result is returned in the corresponding element of a new dynamic array. If an
element of one dynamic array has no corresponding element in the other dynamic
array, the existing element is returned. If an element of one dynamic array is the
null value, null is returned for the concatenation of the corresponding elements.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
       A="A":@VM:"B":@SM:"C"
       B="D":@SM:"E":@VM:"F"
       PRINT CATS(A,B)
This is the program output:
       ADSEVBFSC
CHAIN
Syntax
    CHAIN command
Description
Use the CHAIN statement to terminate execution of a BASIC program and to
execute the value of command. command is an expression that evaluates to any valid
DataStage command. If command evaluates to the null value, the CHAIN statement
fails and the program terminates with a run-time error message.
Local variables belonging to the current program are lost when you chain from one
program to another. Named and unnamed common variables are retained.
CHAIN differs from the EXECUTE or PERFORM statements in that CHAIN does
not return control to the calling program. If a program chains to a proc, any nested
calling procs are removed.
Example
The following program clears the screen, initializes the common area, and then
runs the main application:
    PRINT   @(-1)
    PRINT   "INITIALIZING COMMON, PLEASE WAIT"
    GOSUB   INIT.COMMON
    CHAIN   "RUN BP APP.MAIN KEEP.COMMON"
CHANGE
Syntax
    CHANGE (expression, substring, replacement [ ,occurrence [ ,begin] ] )
Description
Use the CHANGE function to replace a substring in expression with another
substring. If you do not specify occurrence, each occurrence of the substring is
replaced.
occurrence specifies the number of occurrences of substring to replace. To change all
occurrences, specify occurrence as a number less than 1.
begin specifies the first occurrence to replace. If begin is omitted or less than 1, it
defaults to 1.
If substring is an empty string, the value of expression is returned. If replacement is
an empty string, all occurrences of substring are removed.
If expression evaluates to the null value, null is returned. If substring, replacement,
occurrence, or begin evaluates to the null value, the CHANGE function fails and the
program terminates with a run-time error message.
The CHANGE function behaves like the EREPLACE function except when
substring evaluates to an empty string.
Example
    A = "AAABBBCCCDDDBBB"
    PRINT CHANGE (A,"BBB","ZZZ")
    PRINT CHANGE (A,"","ZZZ")
    PRINT CHANGE (A,"BBB","")
This is the program output:
    AAAZZZCCCDDDZZZ
    AAABBBCCCDDDBBB
    AAACCCDDD
CHAR
Syntax
    CHAR (expression)
Description
Use the CHAR function to generate an ASCII character from the numeric value of
expression.
If expression evaluates to the null value, null is returned. If expression evaluates to
128, CHAR(128) is returned, not the null value. CHAR(128) is the equivalent of the
system variable @NULL.STR.
The CHAR function is the inverse of the SEQ function.
If NLS mode is enabled, and if expression evaluates to a number from 129 through
247, the CHAR function generates Unicode characters from x0081 through x00F7.
These values correspond to the equivalent ISO 8859-1 (Latin 1) multibyte charac-
ters. The evaluation of numbers from 0 through 127, 128, and 248 through 255
remains the same whether NLS is enabled or not.
The UNICHAR function is the recommended method for generating Unicode
characters. FFor more information, see the DataStage NLS Guide.
Note: In order to run programs using the CHAR function in NLS mode, you must
      first recompile them in NLS mode.
Example
    X = CHAR(38)
    Y = CHAR(32)
    PRINT X:Y:X
CHAR(38) is an ampersand ( & ). CHAR(32) is a space. This is the program output:
    & &
CHARS
Syntax
    CHARS (dynamic.array)
    CALL −CHARS (return.array, dynamic.array)
    CALL !CHARS (return.array, dynamic.array)
Description
Use the CHARS function to generate a dynamic array of ASCII characters from the
decimal numeric value of each element of dynamic.array.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If any element in the dynamic array is the null value, null is returned for that
element. If any element in the dynamic array evaluates to 128, CHAR(128) is
returned, not the null value. CHAR(128) is the equivalent of the system variable
@NULL.STR.
If NLS mode is enabled, and if any element in the dynamic array evaluates to a
number from 129 through 247, the CHARS function generates Unicode characters
from x0081 through x00F7. These values correspond to the equivalent ISO 8859-1
(Latin 1) multibyte characters. The evaluation of numbers from 0 through 127, 128,
and 248 through 255 remains the same whether NLS is enabled or not.
The UNICHARS function is the recommended method for generating a dynamic
array of Unicode characters. For more information, see the DataStage NLS Guide.
Example
    X = CHARS(38:@VM:32:@VM:38)
    PRINT X
The dynamic array X comprises three elements: CHAR(38) (an ampersand ( & )),
CHAR(32) (a space), and another CHAR(38). The program prints a dynamic array
of these elements separated by value marks:
    &V   V&
CHECKSUM
Syntax
    CHECKSUM (string)
Description
Use the CHECKSUM function to return a cyclical redundancy code (a checksum
value).
If string is the null value, null is returned.
Example
    A = "THIS IS A RECORD TO BE SENT VIA SOME PROTOCOL"
    REC = A:@FM:CHECKSUM(A)
    PRINT REC
This is the program output:
    THIS IS A RECORD TO BE SENT VIA SOME PROTOCOLF30949
CLEAR
Syntax
    CLEAR [COMMON]
Description
Use the CLEAR statement at the beginning of a program to set all assigned and
unassigned values of variables outside of the common area of the program to 0.
This procedure avoids run-time errors for unassigned variables. If you use the
CLEAR statement later in the program, any values assigned to noncommon vari-
ables (including arrays) are lost.
Use the COMMON option to reset the values of all the variables in the unnamed
common area to 0. Variables outside the common area or in the named common
area are unaffected.
Example
    A=100
    PRINT "The value of A before the CLEAR statement:"
    PRINT A
    CLEAR
    PRINT "The value of A after the CLEAR statement:"
    PRINT A
    PRINT
    *
    COMMON B,C,D
    D="HI"
    PRINT "The values of B, C, and D"
    PRINT B,C,D
    CLEAR COMMON
    PRINT B,C,D
This is the program output:
    The value of A before the CLEAR statement: 100
    The value of A after the CLEAR statement:    0
    The values of B, C, and D
    0        0         HI
    0        0         0
CLEARDATA
Syntax
    CLEARDATA
Description
Use the CLEARDATA statement to flush all data that has been loaded in the input
stack by the DATA statement. No expressions or spaces are allowed with this state-
ment. Use the CLEARDATA statement when an error is detected, to prevent data
placed in the input stack from being used incorrectly.
Example
The following program is invoked from a paragraph. A list of filenames and record
IDs is passed to it from the paragraph with DATA statements. If a file cannot be
opened, the CLEARDATA statement clears the data stack since the DATA state-
ments would no longer be valid to the program.
    TEN:
    INPUT FILENAME
    IF FILENAME="END" THEN STOP
    OPEN FILENAME TO FILE ELSE
        PRINT "CAN'T OPEN FILE ":FILENAME
        PRINT "PLEASE ENTER NEW FILENAME "
        CLEARDATA
        GOTO TEN:
    END
    TWENTY:
    INPUT RECORD
    READ REC FROM FILE,RECORD ELSE GOTO TEN:
    PRINT REC<1>
    GOTO TEN:
    TEST.FILE.
    0 records listed.
CLEARFILE
Syntax
    CLEARFILE [file.variable] [ON ERROR statements] [LOCKED statements]
Description
Use the CLEARFILE statement to delete all records in an open dictionary or data
file. You cannot use this statement to delete the file itself. Each file to be cleared
must be specified in a separate CLEARFILE statement.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information on default files, see the OPEN statement).
The CLEARFILE statement fails and the program terminates with a run-time error
message if:
    • The file is neither accessible nor open.
    • file.variable evaluates to the null value.
    • A distributed file contains a part file that cannot be accessed, but the
      CLEARFILE statement clears those part files still available.
    • A transaction is active. That is, you cannot execute this statement between
      a BEGIN TRANSACTION (or TRANSACTION START) statement and the
      COMMIT (or TRANSACTION START) or ROLLBACK statement that ends
      the transaction.
Example
    OPEN "","TEST.FILE" ELSE PRINT "NOT OPEN"
    EXECUTE "LIST TEST.FILE"
    CLEARFILE
    CHAIN "LIST TEST.FILE"
This is the program output:
    LIST TEST.FILE 11:37:45am       03-22-94 PAGE        1
    TEST.FILE
    ONE
    TWO
    THREE
    3 records listed.
    LIST TEST.FILE 11:37:46am       03-22-94 PAGE        1
    TEST.FILE.
    0 records listed.
CLEARPROMPTS
Syntax
    CLEARPROMPTS
    CALL !CLEAR.PROMPTS
Description
Use the CLEARPROMPTS statement to clear the value of the in-line prompt. Once
a value is entered for an in-line prompt, the prompt continues to have that value
until a CLEARPROMPTS statement is executed, unless the in-line prompt control
option A is specified. CLEARPROMPTS clears all values that have been entered for
in-line prompts.
For information about in-line prompts, see the ILPROMPT function.
CLEARSELECT
Syntax
    CLEARSELECT [ALL | list.number]
Description
Use the CLEARSELECT statement to clear an active select list. This statement is
normally used when one or more select lists have been generated but are no longer
needed. Clearing select lists prevents remaining select list entries from being used
erroneously.
Use the keyword ALL to clear all active select lists. Use list.number to specify a
numbered select list to clear. list.number must be a numeric value from 0 through
10. If neither ALL nor list.number is specified, select list 0 is cleared.
If list.number evaluates to the null value, the CLEARSELECT statement fails and
the program terminates with a run-time error message.
Example
The following program illustrates the use of CLEARSELECT to clear a partially
used select list. The report is designed to display the first 40-odd hours of lessons.
A CLEARSELECT is used so that all the selected records are not printed. Once the
select list is cleared, the READNEXT ELSE clause is executed.
    OPEN 'SUN.SPORT' TO FILE ELSE STOP "CAN'T OPEN FILE"
    HOURS=0
    *
    EXECUTE 'SSELECT SUN.SPORT BY START BY INSTRUCTOR'
    *
    START:
    READNEXT KEY ELSE
       PRINT 'FIRST WEEK', HOURS
       STOP
    END
    READ MEMBER FROM FILE,KEY ELSE GOTO START:
    HOURS=HOURS+MEMBER<4>
    PRINT MEMBER<1>,MEMBER<4>
    IF HOURS>40 THEN
        ******
        CLEARSELECT
        ******
        GOTO START:
    END
    GOTO START:
    END
This is the program output:
    14 records selected to Select List #0
    4309     1
    6100     4
    3452     3
    6783     12
    5390     9
    4439     4
    6203     14
    FIRST WEEK         47
CLOSE
Syntax
    CLOSE [file.variable] [ON ERROR statements]
Description
Use the CLOSE statement after opening and processing a file. Any file locks or
record locks are released.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed. If the file is neither accessible nor open, or if file.variable evaluates to the
null value, the CLOSE statement fails and the program terminates with a run-time
error message.
Example
   CLEAR
   OPEN '','EX.BASIC' TO DATA ELSE STOP
   READ A FROM DATA, 'XYZ' ELSE STOP
   A<3>='*'
   WRITE A ON DATA, 'XYZ'
   CLOSE DATA
CLOSESEQ
Syntax
    CLOSESEQ file.variable [ON ERROR statements]
Description
Use the CLOSESEQ statement after opening and processing a file opened for
sequential processing. CLOSESEQ makes the file available to other users.
file.variable specifies a file previously opened with an OPENSEQ statement. If the
file is neither accessible nor open, the program terminates with a run-time error
message. If file.variable is the null value, the CLOSESEQ statement fails and the
program terminates with a run-time error message.
Example
In this example, the CLOSESEQ statement closes FILE.E, making it available to
other users:
   OPENSEQ 'FILE.E', 'RECORD1' TO FILE ELSE ABORT
   READSEQ A FROM FILE THEN PRINT A ELSE STOP
   CLOSESEQ FILE
   END
COL1
Syntax
    COL1 ( )
Description
Use the COL1 function after the execution of a FIELD function to return the
numeric value for the character position that immediately precedes the selected
substring (see the FIELD function). Although the COL1 function takes no argu-
ments, parentheses are required to identify it as a function.
The value obtained from COL1 is local to the program or subroutine executing the
FIELD function. Before entering a subroutine, the current value of COL1 in the
main program is saved. The value of COL1 in the subroutine is initialized as 0.
When control is returned to the calling program, the saved value of COL1 is
restored.
If no FIELD function precedes the COL1 function, a value of 0 is returned. If the
delimiter expression of the FIELD function is an empty string or the null value, or
if the string is not found, the COL1 function returns a 0 value.
Examples
The FIELD function in the following example returns the substring CCC. COL1( )
returns 8, the position of the delimiter ( $ ) that precedes CCC.
    SUBSTRING=FIELD("AAA$BBB$CCC",'$',3)
    POS=COL1()
In the following example, the FIELD function returns a substring of 2 fields with
the delimiter ( . ) that separates them: 4.5. COL1( ) returns 6, the position of the
delimiter that precedes 4.
    SUBSTRING=FIELD("1.2.3.4.5",'.',4,2)
    POS=COL1()
COL2
Syntax
    COL2 ( )
Description
Use the COL2 function after the execution of a FIELD function to return the
numeric value for the character position that immediately follows the selected
substring (see the FIELD function). Although the COL2 function takes no argu-
ments, parentheses are required to identify it as a function.
The value obtained from COL2 is local to the program or subroutine executing the
FIELD function. Before entering a subroutine, the current value of COL2 in the
main program is saved. The value of COL2 in the subroutine is initialized as 0.
When control is returned to the calling program, the saved value of COL2 is
restored.
If no FIELD function precedes the COL2 function, a value of 0 is returned. If the
delimiter expression of the FIELD function is an empty string or the null value, or
if the string is not found, the COL2 function returns a 0 value.
Examples
The FIELD function in the following example returns the substring 111. COL2( )
returns 4, the position of the delimiter ( # ) that follows 111.
    SUBSTRING=FIELD("111#222#3","#",1)
    P=COL2()
In the following example, the FIELD function returns a substring of two fields with
the delimiter ( & ) that separates them: 7&8. COL2( ) returns 5, the position of the
delimiter that follows 8.
    SUBSTRING=FIELD("&7&8&B&","&",2,2)
    S=COL2()
In the next example, FIELD( ) returns the whole string, because the delimiter ( . ) is
not found. COL2( ) returns 6, the position after the last character of the string.
    SUBSTRING=FIELD("9*8*7",".",1)
    Y=COL2()
In the next example, FIELD( ) returns an empty string, because there is no tenth
occurrence of the substring in the string. COL2( ) returns 0 because the substring
was not found.
    SUBSTRING=FIELD("9*8*7","*",10)
    O=COL2()
COMMIT
Syntax
    COMMIT [ WORK ] [ THEN statements ] [ ELSE statements ]
Description
Use the COMMIT statement to commit all file I/O changes made during a trans-
action. The WORK keyword is provided for compatibility with SQL syntax
conventions; it is ignored by the compiler.
A transaction includes all statements between a BEGIN TRANSACTION state-
ment and the COMMIT or ROLLBACK statement that ends the transaction. Either
a COMMIT or a ROLLBACK statement ends the current transaction.
The COMMIT statement can either succeed or fail.
When a subtransaction commits, it makes the results of its database operations
accessible to its parent transaction. The subtransaction commits to the database
only if all of its predecessors up to the top-level transaction are committed.
If a top-level transaction succeeds, all changes to files made during the active trans-
action are committed to disk.
If a subtransaction fails, all its changes are rolled back and do not affect the parent
transaction. If the top-level transaction fails, none of the changes made during the
active transaction are committed, and the database remains unaffected by the
failed transaction. This ensures that the database is maintained in a consistent
state.
If the COMMIT statement succeeds, the THEN statements are executed; any ELSE
statements are ignored. If COMMIT fails, any ELSE statements are executed. After
the THEN or the ELSE statements are executed, control is transferred to the state-
ment following the next END TRANSACTION statement.
All Locks obtained during a transaction remain in effect for the duration of the
active transaction; they are not released by a RELEASE,WRITE, WRITEV, or
MATWRITE statement that is part of the transaction. The parent transaction
adopts the acquired or promoted locks. If a subtransaction rolls back, any locks
that have been acquired or promoted within that transaction are demoted or
released.
The COMMIT statement that ends the top-level transaction releases locks set
during that transaction. Locks obtained outside the transaction are not affected by
the COMMIT statement.
If no transaction is active, the COMMIT statement generates a run-time warning,
and the ELSE statements are executed.
Example
This example begins a transaction that applies locks to rec1 and rec2. If no errors
occur, the COMMIT statement ensures that the changes to rec1 and rec2 are written
to the file. The locks on rec1 and rec2 are released, and control is transferred to the
statement following the END TRANSACTION statement.
    BEGIN TRANSACTION
       READU data1 FROM file1,rec1 ELSE ROLLBACK
       READU data2 FROM file2,rec2, ELSE ROLLBACK
           .
           .
           .
       WRITE new.data1 ON file1,rec1 ELSE ROLLBACK
       WRITE new.data2 ON file2,rec2 ELSE ROLLBACK
       COMMIT WORK
    END TRANSACTION
The update record lock on rec1 is not released on completion of the first WRITE
statement but on completion of the COMMIT statement.
COMMON
Syntax
    COM[MON] [/name/] variable [ ,variable …]
Description
Use the COMMON statement to provide a storage area for variables. Variables in
the common area are accessible to main programs and external subroutines. Corre-
sponding variables can have different names in the main program and in external
subroutines, but they must be defined in the same order. The COMMON statement
must precede any reference to the variables it names.
A common area can be either named or unnamed. An unnamed common area is
lost when the program completes its execution and control returns to the
DataStage command level. A named common area remains available for as long as
the user remains in the DataStage environment.
The common area name can be of any length, but only the first 31 characters are
significant.
Arrays can be dimensioned and named with a COMMON statement. They can be
redimensioned later with a DIMENSION statement, but the COMMON statement
must appear before the DIMENSION statement. When an array is dimensioned in
a subroutine, it takes on the dimensions of the array in the main program regard-
less of the dimensions stated in the COMMON statement. For a description of
dimensioning array variables in a subroutine, see the CALL statement.
When programs share a common area, use the $INCLUDE statement to define the
common area in each program.
Example
Program:
    COMMON NAME, ADDRESS (15, 6), PHONE
Subroutine:
    COMMON A, B (15, 6), C
In this example the variable pairs NAME and A, ADDRESS and B, PHONE and C
are stored in the same memory location.
COMPARE
Syntax
     COMPARE (string1, string2 [ ,justification ])
Description
Use the COMPARE function to compare two strings and return a numeric value
indicating the result.
string1, string2 specify the strings to be compared.
justification is either L for left-justified comparison or R for right-justified compar-
ison. (Any other value causes a run-time warning, and 0 is returned.)
The comparison can be left-justified or right-justified. A right-justified comparison
compares numeric substrings within the specified strings as numbers. The
numeric strings must occur at the same character position in each string. For
example, a right-justified comparison of the strings AB100 and AB99 indicates that
AB100 is greater than AB99 since 100 is greater than 99. A right-justified compar-
ison of the strings AC99 and AB100 indicates that AC99 is greater since C is greater
than B.
If neither L nor R is specified, the default comparison is left-justified.
The following list shows the values returned:
If NLS is enabled, the COMPARE function uses the sorting algorithm and the
Collate convention specified in the NLS.LC.COLLATE file in order to compare the
strings. For more information about collation, see the DataStage NLS Guide.
Examples
In the following example, the strings AB99 and AB100 are compared with the
right-justified option and the result displayed. In this case the result displayed is
–1.
     PRINT COMPARE('AB99','AB100','R')
An example in NLS mode follows. It compares the strings anilno and anillo,
returning the result as 1. It sets the locale to Spanish and compares the strings
again. In this case, the result displayed is –1.
         $INCLUDE UNIVERSE.INCLUDE UVNLSLOC.H
         x=SETLOCALE( UVLC$ALL, 'OFF' )
         PRINT COMPARE( 'anilno', 'anillo', 'L' )
         x=SETLOCALE( UVLC$ALL, 'ES-SPANISH' )
         PRINT COMPARE( 'anilno', 'anillo', 'L' )
This is the program output:
    1
    -1
CONTINUE
The CONTINUE statement is a loop-controlling statement. For syntax details, see
the statement and the LOOP statement.
CONVERT
Syntax
    CONVERT (expression1, expression2, variable)
Description
Use the CONVERT function to return a copy of variable with every occurrence of
specified characters in variable replaced with other specified characters. Every time
a character to be converted appears in variable, it is replaced by the replacement
character.
expression1 specifies a list of characters to be converted. expression2 specifies the
corresponding replacement characters. The first character of expression2 replaces
all instances of the first character of expression1, the second character of expression2
replaces all instances of the second character of expression1, and so on.
If expression2 contains more characters than expression1, the extra characters are
ignored. If expression1 contains more characters than expression2, the characters
with no corresponding expression2 characters are deleted from the result.
If variable is the null value, null is returned. If either expression1 or expression2 is the
null value, the CONVERT function fails and the program terminates with a run-
time error message.
The CONVERT function works similarly to the CONVERT statement.
Example
    A="NOW IS THE TIME"
    PRINT A
    A=CONVERT('TI','XY',A)
    PRINT A
    A=CONVERT('XY','T',A)
    PRINT A
This is the program output:
    NOW IS THE TIME
    NOW YS XHE XYME
    NOW S THE TME
CONVERT statement
Syntax
    CONVERT expression1 TO expression2 IN variable
Description
Use the CONVERT statement to replace every occurrence of specific characters in
a string with other characters. Every time the character to be converted appears in
the string, it is replaced by the replacement character.
expression1 specifies a list of characters to be converted. expression2 specifies a list
of replacement characters. The first character of expression2 replaces all instances of
the first character of expression1, the second character of expression2 replaces all
instances of the second character of expression1, and so on.
If expression2 contains more characters than expression1, the extra characters are
ignored. If expression1 contains more characters than expression2, the characters
with no corresponding expression2 characters are deleted from the variable.
If variable is the null value, null is returned. If either expression1 or expression2 eval-
uates to the null value, the CONVERT statement fails and the program terminates
with a run-time error message.
Example
    A="NOW IS THE TIME"
    PRINT A
    CONVERT 'TI' TO 'XY' IN A
    PRINT A
    CONVERT 'XY' TO 'T' IN A
    PRINT A
This is the program output:
    NOW IS THE TIME
    NOW YS XHE XYME
    NOW S THE TME
COS
Syntax
      COS (expression)
Description
Use the COS function to return the trigonometric cosine of an angle. expression is
an angle expressed as a numeric value in degrees. The COS function is the inverse
of the ACOS function.
Values outside the range of 0 to 360 degrees are interpreted as modulo 360.
Numbers greater than 1E17 produce a warning message and 0 is returned. If expres-
sion evaluates to the null value, null is returned.
Example
      PRINT "COS(45) = " : COS(45)
      END
This is the program output:
      COS(45) = 0.7071
COSH
Syntax
    COSH (expression)
Description
Use the COSH function to return the hyperbolic cosine of expression. expression
must be a numeric value.
If expression evaluates to the null value, null is returned.
Example
    PRINT "COSH(2) = ":COSH(2)
This is the program output:
    COSH(2) = 3.7622
COUNT
Syntax
    COUNT (string, substring)
Description
Use the COUNT function to return the number of times a substring is repeated in
a string value.
string is an expression that evaluates to the string value to be searched. substring is
an expression that evaluates to the substring to be counted. substring can be a char-
acter string, a constant, or a variable.
If substring does not appear in string, a 0 value is returned. If substring is an empty
string, the number of characters in string is returned. If string is the null value, null
is returned. If substring is the null value, the COUNT function fails and the
program terminates with a run-time error message.
By default, each character in string is matched to substring only once. Therefore,
when substring is longer than one character and a match is found, the search
continues with the character following the matched substring. No part of the
matched string is recounted toward another match. For example, the following
statement counts two occurrences of substring TT and assigns the value 2 to vari-
able C:
    C = COUNT ('TTTT', 'TT')
Example
    A=COUNT('ABCAGHDALL','A')
    PRINT "A= ",A
    *
    Z='S#FF##G#JJJJ#'
    Q=COUNT(Z,'#')
    PRINT "Q= ",Q
    *
    Y=COUNT('11111111','11')
    PRINT "Y= ",Y
This is the program output:
    A=        3
    Q=        5
    Y=        4
COUNTS
Syntax
    COUNTS (dynamic.array, substring)
    CALL −COUNTS (return.array, dynamic.array, substring)
    CALL !COUNTS (return.array, dynamic.array, substring)
Description
Use the COUNTS function to count the number of times a substring is repeated in
each element of a dynamic array. The result is a new dynamic array whose
elements are the counts corresponding to the elements in dynamic.array.
dynamic.array specifies the dynamic array whose elements are to be searched.
substring is an expression that evaluates to the substring to be counted. substring
can be a character string, a constant, or a variable.
Each character in an element is matched to substring only once. Therefore, when
substring is longer than one character and a match is found, the search continues
with the character following the matched substring. No part of the matched
element is recounted toward another match.
If substring does not appear in an element, a 0 value is returned. If substring is an
empty string, the number of characters in the element is returned. If substring is the
null value, the COUNTS function fails and the program terminates with a run-time
error message.
If any element in dynamic.array is the null value, null is returned.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    ARRAY="A":@VM:"AA":@SM:"AAAAA"
    PRINT COUNTS(ARRAY, "A")
    PRINT COUNTS(ARRAY, "AA")
This is the program output:
    1V2S5
    0V1S2
CREATE
Syntax
    CREATE file.variable {THEN statements [ELSE statements] | ELSE statements}
Description
Use the CREATE statement after an OPENSEQ statement to create a record in a
type 1 or type 19 file or to create a UNIX or DOS file. CREATE creates the record or
file if the OPENSEQ statement fails. An OPENSEQ statement for the specified
file.variable must be executed before the CREATE statement to associate the path-
name or record ID of the file to be created with the file.variable. If file.variable is the
null value, the CREATE statement fails and the program terminates with a run-
time error message.
Use the CREATE statement when OPENSEQ cannot find a record or file to open
and the next operation is to be a NOBUF, READSEQ, or READBLK. You need not
use the CREATE statement if the first file operation is aWRITESEQ, since
WRITESEQ creates the record or file if it does not exist.
If the record or file is created, the THEN statements are executed, and the ELSE
statements are ignored. If no THEN statements are specified, program execution
continues with the next statement.
If the record or file is not created, the ELSE statements are executed; any THEN
statements are ignored.
File Buffering
Normally DataStage uses buffering for sequential input and output operations.
Use the NOBUF statement after an OPENSEQ statement to turn off buffering and
cause all writes to the file to be performed immediately. For more information
about file buffering, see the NOBUF statement.
Example
In the following example, RECORD4 does not yet exist. When OPENSEQ fails to
open RECORD4 to the file variable FILE, the CREATE statement creates RECORD4
in the type 1 file FILE.E and opens it to the file variable FILE.
    OPENSEQ 'FILE.E', 'RECORD4' TO FILE
       ELSE CREATE FILE ELSE ABORT
    WEOFSEQ FILE
    WRITESEQ 'HELLO, UNIVERSE' TO FILE ELSE STOP
CRT
Syntax
      CRT [print.list]
Description
Use the CRT statement to print data on the screen, regardless of whether a
PRINTER ON statement has been executed. The syntax for print.list is the same as
for PRINT statements.
print.list can contain any BASIC expression. The elements of the list can be numeric
or character strings, variables, constants, or literal strings; the null value, however,
cannot be output. The list can consist of a single expression or a series of expres-
sions separated by commas ( , ) or colons ( : ) for output formatting. If no print.list
is designated, a blank line is output.
Expressions separated by commas are printed at preset tab positions. You can use
multiple commas together to cause multiple tabulation between expressions.
Expressions separated by colons are concatenated. That is, the expression
following the colon is printed immediately after the expression preceding the
colon. To print a list without a LINEFEED and RETURN, end the print.list with a
colon ( : ).
The CRT statement works similarly to the DISPLAY statement.
If NLS is enabled, the CRT statement uses the terminal map in order to print. For
more information about maps and devices, see the DataStage NLS Guide.
Example
      CRT "This can be used to print something on the"
      CRT "terminal while"
      CRT "the PRINTER ON statement is in effect."
The program output on the terminal is:
      This can be used to print something on the
      terminal while
      the PRINTER ON statement is in effect.
DATA
Syntax
    DATA expression [ ,expression …]
Description
Use the DATA statement to place values in an input stack. These values can be
used as responses to INPUT statements executed later in the program or in a
subroutine (see the INPUT statement). The values can also serve as responses to
DataStage commands that request input.
Expressions used in DATA statements can be numeric or character string data. The
null value cannot be stored in the input stack. If expression evaluates to null, the
DATA statement fails and the program terminates with a run-time error message.
Put a comma at the end of each line of a DATA statement to indicate that more data
expressions follow on the next line.
The order in which expressions are specified in the DATA statement is the order in
which the values are accessed by subsequent INPUT statements: first-in, first-out.
When all DATA values have been exhausted, the INPUT statement prompts the
user for a response at the terminal.
The DATA statement must be executed before an INPUT statement that is to use
expression for input.
You can store up to 512 characters in a data stack.
You can list the current data in the stack from your program by accessing the
@DATA.PENDING variable with the statement:
    PRINT @DATA.PENDING
Example
In the following example, the INPUT NBR statement uses the first value placed in
the input stack by the DATA statement, 33, as the value of NBR. The INPUT
DESCR statement uses the second value, 50, as the value of DESCR. The INPUT
PRICE statement uses the third value, 21, as the value of PRICE.
    X=33; Y=50; Z=21
    DATA X,Y,Z
    X=Y+Z
    *
    INPUT   NBR
    INPUT   DESCR
    INPUT   PRICE
    INPUT   QTY
    PRINT   NBR,DESCR,PRICE,QTY
This is the program output:
    ?33
    ?50
    ?21
    ?2
    33         50         21          2
The value of NBR is the value of X when the DATA statement is executed, not the
current value of X (namely, Y+Z). The INPUT QTY statement has no corre-
sponding value in the input stack, so it prompts the user for input.
DATE
Syntax
    DATE ( )
Description
Use the DATE function to return the numeric value of the internal system date.
Although the DATE function takes no arguments, parentheses are required to
identify it as a function.
The internal format for the date is based on a reference date of December 31, 1967,
which is day 0. All dates thereafter are positive numbers representing the number
of days elapsed since day 0. All dates before day 0 are negative numbers repre-
senting the number of days before day 0. For example:
Example
    PRINT DATE()
    PRINT OCONV(DATE(),"D2/")
This is the program output:
    9116
    12/15/92
DCFLUSH
Syntax
    DCFLUSH (file.variable, opt)
Description
Use the DCFLUSH function to flush the disk cache file buffers.
file.variable is the file descriptor of an open uniVerse file.
opt is the flush mode. If opt is set to 1, the disk cache file buffer is flushed directly
to disk, otherwise it is flushed from the disk cache to the operating system file
buffer for later flushing to disk by the operating system.
DCFLUSH will wait until the file has been flushed either to disk or to the operating
system buffer, and then returns 0.
DCOUNT
Syntax
    DCOUNT (string, delimiter)
Description
Use the DCOUNT function to return the number of delimited fields in a data
string.
string is an expression that evaluates to the data string to be searched.
delimiter is an expression that evaluates to the delimiter separating the fields to be
counted. delimiter can be a character string of 0, 1, or more characters.
DCOUNT differs from COUNT in that it returns the number of values separated
by delimiters rather than the number of occurrences of a character string. Two
consecutive delimiters in string are counted as one field. If delimiter evaluates to an
empty string, a count of 1 plus the number of characters in the string is returned.
If string evaluates to an empty string, 0 is returned.
If string evaluates to the null value, null is returned. If delimiter evaluates to the null
value, the DCOUNT function fails and the program terminates with a run-time
error message.
Example
    REC="88.9.B.7"
    Q=DCOUNT(REC,'.')
    PRINT "Q= ",Q
    REC=34:@VM:55:@VM:88:@VM:"FF":@VM:99:@VM:"PP"
    R=DCOUNT(REC,@VM)
    PRINT "R= ",R
This is the program output:
    Q=          4
    R=          6
DEBUG
Syntax
     DEBUG
Description
Use the DEBUG statement to invoke RAID, the interactive BASIC debugger. The
DEBUG statement takes no arguments. When this statement is encountered,
program execution stops and the double colon ( :: ) prompt appears, waiting for a
RAID command. The following table summarizes the RAID commands:
RAID Commands
Command            Action
line               Displays the specified line of the source code.
/[string]          Searches the source code for string.
B                  Set a RAID breakpoint.
C                  Continue program execution.
D                  Delete a RAID breakpoint.
G                  Go to a specified line or address and continue program
                   execution.
H                  Display statistics for the program.
I                  Display and execute the next object code instruction.
L                  Print the next line to be executed.
M                  Set watchpoints.
Q                  Quit RAID.
R                  Run the program.
S                  Step through the BASIC source code.
T                  Display the call stack trace.
V                  Enter verbose mode for the M command.
V*                Print the compiler version that generated the object code.
W                  Display the current window.
X                  Display the current object code instruction and address.
Command           Action
X*                Display local run machine registers and variables.
Z                 Display the next 10 lines of source code.
$                 Turn on instruction counting.
#                 Turn on program timing.
+                 Increment the current line or address.
−                 Decrement the current line or address.
.                 Display the last object code instruction executed.
variable/         Print the value of variable.
variable!string   Change the value of variable to string.
DEFFUN
Syntax
    DEFFUN function [ ( [MAT] argument [ , [MAT] argument …] ) ]
             [CALLING call.ID]
Description
Use the DEFFUN statement to define a user-written function. You must declare a
user-defined function before you can use it in a program. The DEFFUN statement
provides the compiler with information such as the function name and the number
and type of arguments. You can define a user-written function only once in a
program. A subsequent DEFFUN statement for an already defined user-written
function causes a fatal error.
function is the name of the user-written function.
arguments supply up to 254 arguments in the DEFFUN statement. To pass an array,
you must precede the array name with the keyword MAT. An extra argument is
hidden so that the user-defined function can use it to return a value. An extra argu-
ment is retained by the user-written function so that a value is returned by a
RETURN (value) statement (for more information see the RETURN(value) state-
ment). If the RETURN (value) statement specifies no value, an empty string is
returned. The extra argument is reported by the MAP and MAKE.MAP.FILE
commands.
call.ID is an expression that evaluates to the name by which the function is called
if it is not the same as the function name. It can be a quoted string (the call ID itself)
or a variable that evaluates to the call ID. If you do not use the CALLING clause,
the user-defined function is presumed to be defined in the VOC file and cataloged
without any prefix.
Examples
The following example defines a user-written function called MYFUNC with the
arguments or formal parameters A, B, and C:
    FUNCTION MYFUNC(A, B, C)
    Z = ...
    RETURN (Z)
    END
The next example declares the function MYFUNC. It uses the function with the
statement T = MYFUNC (X, Y, Z). The actual parameters held in X, Y, and Z are
referenced by the formal parameters A, B, and C, so the value assigned to T can be
calculated.
    DEFFUN MYFUNC(X, Y, Z)
    T = MYFUNC(X, Y, Z)
    END
DEL
Syntax
      DEL dynamic.array < field# [ ,value# [ ,subvalue#] ] >
Description
Use the DEL statement to delete a field, value, or subvalue from a dynamic array.
The DEL statement works similarly to the DELETE function.
dynamic.array is an expression that evaluates to a dynamic array. If dynamic.array
evaluates to the null value, null is returned.
field# is an expression that evaluates to the field in dynamic.array. value# is an
expression that evaluates to the value in the field. subvalue# is an expression that
evaluates to the subvalue in the value. These expressions are called delimiter
expressions. The numeric values of the delimiter expressions specify which field,
value, or subvalue to delete. The entire position is deleted, including its delimiter
characters.
value# and subvalue# are optional. If they are equal to 0, the entire field is deleted.
If subvalue# is equal to 0 and value# and field# are greater than 0, the specified value
in the specified field is deleted. If all three delimiter expressions are greater than 0,
only the specified subvalue is deleted.
If any delimiter expression is the null value, the DEL statement fails and the
program terminates with a run-time error message.
If a higher-level delimiter expression has a value of 0 when a lower-level delimiter
expression is greater than 0, the 0 delimiter is treated as if it were equal to 1. The
delimiter expressions are, from highest to lowest: field, value, and subvalue.
If the DEL statement references a subelement of a higher element whose value is
the null value, the dynamic array is unchanged. Similarly, if all delimiter expres-
sions are 0, the original string is returned.
Examples
In the following examples a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S.
The next example deletes the first subvalue in field 4 and sets the value of Q to
FLD1FVAL1VSUBV1SSUBV2FFSUBV4:
    Q=R
    DEL Q<4,1,1>
The next example deletes the second value in field 2 and sets the value of Q to
FLD1FVAL1FFSUBV3SSUBV4:
    Q=R
    DEL Q<2,2,0>
The next example deletes field 3 entirely and sets the value of Q to
FLD1FVAL1VSUBV1SSUBV2FSUBV3SSUBV4:
    Q=R
    DEL Q<3,0,0>
The next example deletes the second subvalue in field 4 and sets the value of Q to
FLD1FVAL1VSUBV1SSUBV2FFSUBV3:
    Q=R
    DEL Q<4,1,2>
DELETE
Syntax
    DELETE (dynamic.array, field#[ ,value#[ ,subvalue#] ] )
Description
Use the DELETE function to erase the data contents of a specified field, value, or
subvalue and its corresponding delimiter from a dynamic array. The DELETE
function returns the contents of the dynamic array with the specified data removed
without changing the actual value of the dynamic array.
dynamic.array is an expression that evaluates to the array in which the field, value,
or subvalue to be deleted can be found. If dynamic.array evaluates to the null value,
null is returned.
field# is an expression that evaluates to the field in the dynamic array; value# is an
expression that evaluates to the value in the field; subvalue# is an expression that
evaluates to the subvalue in the value. The numeric values of the delimiter expres-
sions specify which field, value, or subvalue to delete. The entire position is
deleted, including its delimiting characters.
value# and subvalue# are optional. If they are equal to 0, the entire field is deleted.
If subvalue# is equal to 0 and value# and field# are greater than 0, the specified value
in the specified field is deleted. If all three delimiter expressions are greater than 0,
only the specified subvalue is deleted.
If any delimiter expression is the null value, the DELETE function fails and the
program terminates with a run-time error message.
If a higher-level delimiter expression has a value of 0 when a lower-level delimiter
is greater than 0, the 0 delimiter is treated as if it were equal to 1. The delimiter
expressions are, from highest to lowest: field, value, and subvalue.
If the DELETE function references a subelement of a higher element whose value
is the null value, the dynamic array is unchanged. Similarly, if all delimiter expres-
sions are 0, the original string is returned.
Examples
In the following examples a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S.
The next example deletes the first subvalue in field 4 and sets the value of Q to
FLD1FVAL1VSUBV1SSUBV2FFSUBV4:
    Q=DELETE (R,4,1,1)
The next example deletes the second value in field 2 and sets the value of Q to
FLD1FVAL1FFSUBV3SSUBV4:
    Q=DELETE (R,2,2)
The next example deletes field 3 entirely and sets the value of Q to
FLD1FVAL1VSUBV1SSUBV2FSUBV3SSUBV4:
    Q=DELETE (R,3,0,0)
The next example deletes the second subvalue in field 4 and sets the value of Q to
FLD1FVAL1VSUBV1SSUBV2FFSUBV3:
    Q=DELETE (R,4,1,2)
DELETE
Syntax
    DELETE [ file.variable, ] record.ID [ ON ERROR statements ]
             [ LOCKED statements ]
             [ THEN statements ] [ ELSE statements ]
    DELETEU [ file.variable, ] record.ID [ ON ERROR statements ]
             [ LOCKED statements ]
             [ THEN statements ] [ ELSE statements ]
Description
Use the DELETE statements to delete a record from a DataStage file. If you specify
a file variable, the file must be open when the DELETE statement is encountered
(see the OPEN statement).
file.variable is a file variable from a previous OPEN statement.
record.ID is an expression that evaluates to the record ID of the record to be deleted.
If the file does not exist or is not open, the program terminates and a run-time error
results. If you do not specify a file variable, the most recently opened default file is
used (see the OPEN statement for more information on default files). If you specify
both a file variable and a record ID, you must use a comma to separate them.
If the file is an SQL table, the effective user of the program must have SQL DELETE
privilege to delete records in the file. For information about the effective user of a
program, see the AUTHORIZATION statement.
The record is deleted, and any THEN statements are executed. If the deletion fails,
the ELSE statements are executed; any THEN statements are ignored.
If a record is locked, it is not deleted, and an error message is produced. The ELSE
statements are not executed.
If either file.variable or record.ID evaluates to the null value, the DELETE statement
fails and the program terminates with a run-time error message.
The file must have been previously opened with an OPEN statement. If a file vari-
able was specified in the OPEN statement, it can be used in the DELETEU
statement. You must place a comma between the file variable and the record ID
expression. If no file variable is specified in the DELETEU statement, the statement
applies to the default file. See the OPEN statement for a description of the default
file.
Examples
    OPEN "","MLIST" TO MALIST ELSE STOP
    PRINT "FILE BEFORE DELETE STATEMENT:"
    EXECUTE "COUNT MLIST"
    PRINT
    DELETE MALIST, "JONES"
    PRINT "FILE AFTER DELETE STATMENT:"
    EXECUTE "LIST MLIST"
This is the program output:
    FILE BEFORE DELETE STATEMENT:
3 records listed.
    2 records listed.
In the following example, the data portion of the SUBSIDIARIES files is opened to
the file variable SUBS. If the file cannot be opened an appropriate message is
printed. The record MADRID is read and then deleted from the file. An update
record lock had been set and is maintained by the DELETEU statement.
    OPEN "","SUBSIDIARIES" TO SUBS
    READU REC FROM SUBS, 'MADRID'
       ELSE STOP 'Sorry, cannot open Subsidiaries file'
    DELETEU SUBS, "MADRID"
DELETELIST
Syntax
      DELETELIST listname
Description
Use the DELETELIST statement to delete a select list saved in the &SAVEDLISTS&
file.
listname can evaluate to the form:
      record.ID
or:
      record.ID account.name
record.ID is the name of a select list in the &SAVEDLISTS& file. If you specify
account.name, the &SAVEDLISTS& file of the specified account is used instead of
the local &SAVEDLISTS& file.
If listname evaluates to the null value, the DELETELIST statement fails and the
program terminates with a run-time error message.
DELETEU
Use the DELETEU statement to maintain an update record lock while performing
the DELETE statement.
DIMENSION
Syntax
    DIM[ENSION] matrix (rows, columns) [ , matrix (rows, columns) …]
Description
Use the DIMENSION statement to define the dimensions of an array variable
before referencing the array in the program. For a matrix (a two-dimensional
array), use the DIMENSION statement to set the maximum number of rows and
columns available for the elements of the array. For a vector (a one-dimensional
array), use the DIMENSION statement to set the maximum value of the subscript
(the maximum elements) in the array.
matrix and vector can be any valid variable name. The maximum dimension can be
any valid numeric expression. When specifying the two dimensions of a matrix,
you must use a comma to separate the row and column expressions. These expres-
sions are called indices.
You can use a single DIMENSION statement to define multiple arrays. If you
define more than one array with a DIMENSION statement, you must use commas
to separate the array definitions.
The DIMENSION statement declares only the name and size of the array. It does
not assign values to the elements of the array. Assignment of values to the elements
is done with the MAT, MATPARSE, MATREAD, MATREADU, and Assignment
Statements.
The DIMENSION statement in an IDEAL or INFORMATION flavor account is
executed at run time. The advantage of the way DataStage handles this statement
is that the amount of memory allocated is not determined until the DIM statement
is executed. This means that arrays can be redimensioned at run time.
When redimensioning an array, you can change the maximum number of
elements, rows, columns, or any combination thereof. You can even change the
dimensionality of an array (that is, from a one-dimensional to a two-dimensional
array or vice versa).
The values of the array elements are affected by redimensioning as follows:
    • Common elements (those with the same indices) are preserved.
    • New elements (those that were not indexed in the original array) are initial-
      ized as unassigned.
    • Abandoned elements (those that can no longer be referenced in the altered
      array) are lost, and the memory space is returned to the operating system.
The DIMENSION statement fails if there is not enough memory available for the
array. When this happens, the INMAT function is set to a value of 1.
An array variable that is passed to a subroutine in its entirety as an argument in a
CALL statement cannot be redimensioned in the subroutine. Each array in a
subroutine must be dimensioned once. The dimensions declared in the subroutine
DIMENSION statement are ignored, however, when an array is passed to the
subroutine as an argument (for more information, see the CALL statement).
Examples
    DIM ARRAY(2,2)
    ARRAY(1,1)="KK"
    ARRAY(1,2)="GG"
    ARRAY(2,1)="MM"
    ARRAY(2,2)="NN"
In the next example warning messages are printed for the unassigned elements in
the matrix. The elements are assigned empty strings as their values.
    DIM ARRAY(2,3)
    *
    PRINT
    FOR X=1 TO 2
       FOR Y=1 TO 3
           PRINT "ARRAY(":X:",":Y:")", ARRAY(X,Y)
       NEXT Y
    NEXT X
    DIM S(3,2)
    S(1,1)=1
    S(1,2)=2
    S(2,1)=3
    S(2,2)=4
    S(3,1)=5
    S(3,2)=6
In the next example the common elements are preserved. Those elements that
cannot be referenced in the new matrix (S(3,1), S(3,2) ) are lost.
    DIM S(2,2)
    *
    PRINT
    FOR X=1 TO 2
    FOR Y=1 TO 2
    PRINT "S(":X:",":Y:")", S(X,Y)
    NEXT Y
    NEXT X
This is the program output:
    ARRAY(1,1)         KK
    ARRAY(1,2)         GG
    ARRAY(1,3)         Program 'DYNAMIC.DIMENSION':
    Line 12, Variable previously undefined, empty string used.
    ARRAY(2,1)         MM
    ARRAY(2,2)         NN
    ARRAY(2,3)         Program 'DYNAMIC.DIMENSION':
    Line 12, Variable previously undefined, empty string used.
    S(1,1)     1
    S(1,2)     2
    S(2,1)     3
    S(2,2)     4
DISPLAY
Syntax
    DISPLAY [print.list]
Description
Use the DISPLAY statement to print data on the screen, regardless of whether a
PRINTER ON statement has been executed. The syntax for print.list is the same as
forPRINT statements.
The elements of the list can be numeric or character strings, variables, constants, or
literal strings; the null value, however, cannot be output. The list can consist of a
single expression or a series of expressions separated by commas ( , ) or colons ( : )
for output formatting. If no print.list is designated, a blank line is output.
Expressions separated by commas are printed at preset tab positions. You can use
multiple commas together to cause multiple tabulation between expressions.
Expressions separated by colons are concatenated. That is, the expression
following the colon is printed immediately after the expression preceding the
colon. To print a list without a LINEFEED and RETURN, end the print list with a
colon ( : ).
The DISPLAY statement works similarly to the CRT statement.
Example
    DISPLAY "This can be used to print something on the
    DISPLAY "terminal while"
    DISPLAY "the PRINTER ON statement is in effect."
The program output on the terminal is:
    This can be used to print something on the
    terminal while
    the PRINTER ON statement is in effect.
DIV
Syntax
      DIV (dividend, divisor)
Description
Use the DIV function to calculate the value of the quotient after dividend is divided
by divisor.
The dividend and divisor expressions can evaluate to any numeric value. The only
exception is that divisor cannot be 0. If either dividend or divisor evaluates to the null
value, null is returned.
Example
      X=100; Y=25
      Z = DIV (X,Y)
      PRINT Z
This is the program output:
      4
DIVS
Syntax
       DIVS (array1, array2)
       CALL −DIVS (return.array, array1, array2)
       CALL !DIVS (return.array, array1, array2)
Description
Use the DIVS function to create a dynamic array containing the result of the
element-by-element division of two dynamic arrays.
Each element of array1 is divided by the corresponding element of array2 with the
result being returned in the corresponding element of a new dynamic array. If
elements of array1 have no corresponding elements in array2, array2 is padded with
ones and the array1 elements are returned. If an element of array2 has no corre-
sponding element in array1, 0 is returned. If an element of array2 is 0, a run-time
error message is printed and a 0 is returned. If either element of a corresponding
pair is the null value, null is returned.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
       A=10:@VM:15:@VM:9:@SM:4
       B=2:@VM:5:@VM:9:@VM:2
       PRINT DIVS(A,B)
This is the program output:
       5V3V1S4V0
DOWNCASE
Syntax
    DOWNCASE (expression)
Description
Use the DOWNCASE function to change all uppercase letters in expression to
lowercase. If expression evaluates to the null value, null is returned.
DOWNCASE is equivalent to OCONV("MCL").
If NLS is enabled, the DOWNCASE function uses the conventions specified by the
Ctype category for the Lowercase field of the NLS.LC.CTYPE file to change the
letters in expression. For more information about the NLS.LC.CTYPE, see the
DataStage NLS Guide.
Example
    A="DOWN CASE DOES THIS:      "
    PRINT A:DOWNCASE(A)
    B="Down Case Does This:      "
    PRINT B:DOWNCASE(B)
This is the program output:
    DOWN CASE DOES THIS:      down case does this:
    Down Case Does This:      down case does this:
DQUOTE
Syntax
    DQUOTE (expression)
Description
Use the DQUOTE function to enclose an expression in double quotation marks. If
expression evaluates to the null value, null is returned.
Example
    PRINT DQUOTE(12 + 5) : " IS THE ANSWER."
    END
This is the program output:
    "17" IS THE ANSWER.
Syntax
    JobHandle = DSAttachJob (JobName, ErrorMode)
JobHandle is the name of a variable to hold the return value which is subsequently
used by any other function or routine when referring to the job. Do not assume
that this value is an integer.
JobName is a string giving the name of the job to be attached to.
ErrorMode is a value specifying how other routines using the handle should report
errors. It is one of:
    DSJ.ERRFATAL          Log a fatal message and abort the controlling
                          job (default).
    DSJ.ERRWARNINGLog a warning message but carry on.
    DSJ.ERRNONE           No message logged - caller takes full
                          responsibility (failure of DSAttachJob itself
                          will be logged, however).
Remarks
A job cannot attach to itself.
The JobName parameter can specify either an exact version of the job in the form
job%Reln.n.n, or the latest version of the job in the form job. If a controlling job is
itself released, you will get the latest released version of job. If the controlling job
is a development version, you will get the latest development version of job.
Example
This is an example of attaching to Release 11 of the job Qsales:
    Qsales_handle = DSAttachJob ("Qsales%Rel1",
    ➥ DSJ.ERRWARN)
Syntax
    Found = DSCheckRoutine(RoutineName)
RoutineName is the name of BASIC routine to check.
Found      Boolean. @False if RoutineName not findable, else @True.
Example
    rtn$ok = DSCheckRoutine(“DSU.DSSendMail”)
    If(NOT(rtn$ok)) Then
        * error handling here
    End.
Syntax
    ErrCode = DSDetachJob (JobHandle)
JobHandle is the handle for the job as derived from DSAttachJob.
ErrCode is 0 if DSStopJob is successful, otherwise it may be the following:
    DSJE.BADHANDLE            Invalid JobHandle.
The only possible error is an attempt to close DSJ.ME. Otherwise, the call always
succeeds.
Example
The following command detaches the handle for the job qsales:
    Deterr = DSDetachJob (qsales_handle)
Syntax
     Call DSExecute (ShellType, Command, Output, SystemReturnCode)
ShellType (input) specifies the type of command you want to execute and is either
NT or UV (for DataStage Engine).
Command (input) is the command to execute. Command should not prompt for
input when it is executed.
Output (output) is any output from the command. Each line of output is separated
by a field mark, @FM. Output is added to the job log file as an information
message.
SystemReturnCode (output) is a code indicating the success of the command. A
value of 0 means the command executed successfully. A value of 1 (for a DOS
command) indicates that the command was not found. Any other value is a
specific exit code from the command.
Remarks
Do not use DSExecute from a transform; the overhead of running a command for
each row processed by a stage will degrade performance of the job.
Obtains information reported at the end of execution of certain parallel stages. The
information collected, and available to be interrogated, is specified at design time.
For example, transformer stage information is specified in the Triggers tab of the
Transformer stage Properties dialog box.
DSGetProjectInfo
Syntax
     Result = DSGetCustInfo (JobHandle, StageName, CustInfoName, InfoType)
JobHandle is the handle for the job as derived from DSAttachJob, or it may be
DSJ.ME to refer to the current job.
StageName is the name of the stage to be interrogated. It may also be DSJ.ME to
refer to the current stage if necessary.
VarName is the name of the variable to be interrogated.
Provides a method of obtaining information about a job, which can be used gener-
ally as well as for job control. It can refer to the current job or a controlled job,
depending on the value of JobHandle.
DSGetJobInfo
Syntax
     Result = DSGetJobInfo (JobHandle, InfoType)
JobHandle is the handle for the job as derived from DSAttachJob, or it may be
DSJ.ME to refer to the current job.
InfoType specifies the information required and can be one of:
     DSJ.JOBSTATUS
     DSJ.JOBNAME
     DSJ.JOBCONTROLLER
     DSJ.JOBSTARTTIMESTAMP
     DSJ.JOBWAVENO
     DSJ.PARAMLIST
     DSJ.STAGELIST
     DSJ.USERSTATUS
     DSJ.JOBCONTROL
     DSJ.JOBPID
     DSJ.JPBLASTTIMESTAMP
     DSJ.JOBINVOCATIONS
     DSJ.JOBINTERIMSTATUS
     DSJ.JOBINVOCATIONID
     DSJ.JOBDESC
     DSJ.STAGELIST2
     DSJ.JOBELAPSED
Result depends on the specified InfoType, as follows:
Remarks
When referring to a controlled job, DSGetJobInfo can be used either before or
after a DSRunJob has been issued. Any status returned following a successful call
to DSRunJob is guaranteed to relate to that run of the job.
Examples
The following command requests the job status of the job qsales:
Syntax
    Result = DSGetLinkInfo (JobHandle, StageName, LinkName, InfoType)
JobHandle is the handle for the job as derived from DSAttachJob, or it can be
DSJ.ME to refer to the current job.
StageName is the name of the active stage to be interrogated. May also be DSJ.ME
to refer to the current stage if necessary.
LinkName is the name of a link (input or output) attached to the stage. May also be
DSJ.ME to refer to current link (e.g. when used in a Transformer expression or
transform function called from link code).
InfoType specifies the information required and can be one of:
    DSJ.LINKLASTERR
    DSJ.LINKNAME
    DSJ.LINKROWCOUNT
    DSJ.LINKSQLSTATE
    DSJ.LINKDBMSCODE
    DSJ.LINKDESC
    DSJ.LINKSTAGE
    DSJ.INSTROWCOUNT
Result depends on the specified InfoType, as follows:
    • DSJ.LINKLASTERR String – last error message (if any) reported from the
      link in question.
    • DSJ.LINKNAME String – returns the name of the link, most useful when
      used with JobHandle = DSJ.ME and StageName = DSJ.ME and LinkName =
      DSJ.ME to discover your own name.
    • DSJ.LINKROWCOUNT Integer – number of rows that have passed down
      a link so far.
    • DSJ.LINKSQLSTATE – the SQL state for the last error occurring on this
      link.
    • DSJ.LINKDBMSCODE – the DBMS code for the last error occurring on
      this link.
    • DSJ.LINKDESC – description of the link.
    • DSJ.LINKSTAGE – name of the stage at the other end of the link.
    • DSJ.INSTROWCOUNT – comma-separated list of rowcounts, one per
      instance (parallel jobs)
Result may also return error conditions as follows:
    DSJE.BADHANDLE JobHandle was invalid.
    DSJE.BADTYPE           InfoType was unrecognized.
    DSJE.BADSTAGE          StageName does not refer to a known stage
                           in the job.
    DSJE.NOTINSTAGE StageName was DSJ.ME and the caller is
                    not running within a stage.
    DSJE.BADLINK           LinkName does not refer to a known link
                           for the stage in question.
Remarks
When referring to a controlled job, DSGetLinkInfo can be used either before or
after a DSRunJob has been issued. Any status returned following a successful call
to DSRunJob is guaranteed to relate to that run of the job.
Example
The following command requests the number of rows that have passed down the
order_feed link in the loader stage of the job qsales:
    link_status = DSGetLinkInfo(qsales_handle, "loader", ➥
    "order_feed", DSJ.LINKROWCOUNT)
Syntax
    EventDetail = DSGetLogEntry (JobHandle, EventId)
JobHandle is the handle for the job as derived from DSAttachJob.
EventId is an integer that identifies the specific log event for which details are
required. This is obtained using the DSGetNewestLogId function.
EventDetail is a string containing substrings separated by \. The substrings are as
follows:
    Substring1       Timestamp in form YYYY-MM-DD HH:MM:SS
    Substring2       User information
    Substring3       EventType – see DSGetNewestLogId
    Substring4 – n Event message
If any of the following errors are found, they are reported via a fatal log event:
    DSJE.BADHANDLE            Invalid JobHandle.
    DSJE.BADVALUE             Error accessing EventId.
Example
The following command reads full event details of the log event identified by
LatestLogid into the string LatestEventString:
    LatestEventString =
    ➥ DSGetLogEntry(qsales_handle,latestlogid)
Returns a list of short log event details. The details returned are determined by the
setting of some filters. (Care should be taken with the setting of the filters, other-
wise a large amount of information can be returned.)
DSGetLogSummary
Syntax
    SummaryArray = DSGetLogSummary (JobHandle, EventType, StartTime,
    EndTime, MaxNumber)
JobHandle is the handle for the job as derived from DSAttachJob.
EventType is the type of event logged and is one of:
    DSJ.LOGINFO               Information message
    DSJ.LOGWARNING            Warning message
    DSJ.LOGFATAL              Fatal error
    DSJ.LOGREJECT             Reject link was active
    DSJ.LOGSTARTED            Job started
    DSJ.LOGRESET              Log was reset
    DSJ.LOGANY                Any category (the default)
StartTime is a string in the form YYYY-MM-DD HH:MM:SS or YYYY-MM-DD.
EndTime is a string in the form YYYY-MM-DD HH:MM:SS or YYYY-MM-DD.
MaxNumber is an integer that restricts the number of events to return. 0 means no
restriction. Use this setting with caution.
SummaryArray is a dynamic array of fields separated by @FM. Each field
comprises a number of substrings separated by \, where each field represents a
separate event, with the substrings as follows:
    Substring1       EventId as per DSGetLogEntry
    Substring2       Timestamp in form YYYY-MM-DD
                     HH:MM:SS
    Substring3       EventType – see DSGetNewestLogId
    Substring4 – n Event message
If any of the following errors are found, they are reported via a fatal log event:
Example
The following command produces an array of reject link active events recorded
for the qsales job between 18th August 1998, and 18th September 1998, up to a
maximum of MAXREJ entries:
   RejEntries = DSGetLogSummary (qsales_handle,
   ➥ DSJ.LOGREJECT, "1998-08-18 00:00:00", "1998-09-18
   ➥ 00:00:00", MAXREJ)
Gets the ID of the most recent log event in a particular category, or in any category.
DSGetNewestLogId
Syntax
    EventId = DSGetNewestLogId (JobHandle, EventType)
JobHandle is the handle for the job as derived from DSAttachJob.
EventType is the type of event logged and is one of:
    DSJ.LOGINFO             Information message
    DSJ.LOGWARNING Warning message
    DSJ.LOGFATAL            Fatal error
    DSJ.LOGREJECT           Reject link was active
    DSJ.LOGSTARTED Job started
    DSJ.LOGRESET            Log was reset
    DSJ.LOGANY              Any category (the default)
EventId is an integer that identifies the specific log event. EventId can also be
returned as an integer, in which case it contains an error code as follows:
    DSJE.BADHANDLE Invalid JobHandle.
    DSJE.BADTYPE            Invalid EventType.
Example
The following command obtains an ID for the most recent warning message in the
log for the qsales job:
    Warnid = DSGetNewestLogId (qsales_handle,
    ➥ DSJ.LOGWARNING)
Syntax
    Result = DSGetParamInfo (JobHandle, ParamName, InfoType)
JobHandle is the handle for the job as derived from DSAttachJob, or it may be
DSJ.ME to refer to the current job.
ParamName is the name of the parameter to be interrogated.
InfoType specifies the information required and may be one of:
    DSJ.PARAMDEFAULT
    DSJ.PARAMHELPTEXT
    DSJ.PARAMPROMPT
    DSJ.PARAMTYPE
    DSJ.PARAMVALUE
    DSJ.PARAMDES.DEFAULT
    DSJ.PARAMLISTVALUES
    DSJ.PARAMDES.LISTVALUES
    DSJ.PARAMPROMPT.AT.RUN
Result depends on the specified InfoType, as follows:
    • DSJ.PARAMDEFAULT String – Current default value for the parameter in
      question. See also DSJ.PARAMDES.DEFAULT.
    • DSJ.PARAMHELPTEXT String – Help text (if any) for the parameter in
      question.
    • DSJ.PARAMPROMPT String – Prompt (if any) for the parameter in
      question.
    • DSJ.PARAMTYPE Integer – Describes the type of validation test that
      should be performed on any value being set for this parameter. Is one of:
       DSJ.PARAMTYPE.STRING
        DSJ.PARAMTYPE.ENCRYPTED
        DSJ.PARAMTYPE.INTEGER
        DSJ.PARAMTYPE.FLOAT (the parameter may contain periods and E)
        DSJ.PARAMTYPE.PATHNAME
        DSJ.PARAMTYPE.LIST (should be a string of Tab-separated strings)
        DSJ.PARAMTYPE.DATE (should be a string in form YYYY-MM-DD)
        DSJ.PARAMTYPE.TIME (should be a string in form HH:MM)
    • DSJ.PARAMVALUE String – Current value of the parameter for the
      running job or the last job run if the job is finished.
    • DSJ.PARAMDES.DEFAULT String – Original default value of the param-
      eter - may differ from DSJ.PARAMDEFAULT if the latter has been
      changed by an administrator since the job was installed.
    • DSJ.PARAMLISTVALUES String – Tab-separated list of allowed values
      for the parameter. See also DSJ.PARAMDES.LISTVALUES.
    • DSJ.PARAMDES.LISTVALUES String – Original Tab-separated list of
      allowed values for the parameter – may differ from DSJ.PARAM-
      LISTVALUES if the latter has been changed by an administrator since the
      job was installed.
    • DSJ.PROMPT.AT.RUN String – 1 means the parameter is to be prompted
      for when the job is run; anything else means it is not (DSJ.PARAMDE-
      FAULT String to be used directly).
Result may also return error conditions as follows:
        DSJE.BADHANDLE           JobHandle was invalid.
        DSJE.BADPARAM            ParamName is not a parameter
                                 name in the job.
        DSJE.BADTYPE             InfoType was unrecognized.
Remarks
When referring to a controlled job, DSGetParamInfo can be used either before or
after a DSRunJob has been issued. Any status returned following a successful call
to DSRunJob is guaranteed to relate to that run of the job.
Example
The following command requests the default value of the quarter parameter for
the qsales job:
   Qs_quarter = DSGetparamInfo(qsales_handle, "quarter",
   ➥ DSJ.PARAMDEFAULT)
Syntax
     Result = DSGetProjectInfo (InfoType)
InfoType specifies the information required and can be one of:
     DSJ.JOBLIST
     DSJ.PROJECTNAME
     DSJ.HOSTNAME
Result depends on the specified InfoType, as follows:
     • DSJ.JOBLIST String - comma-separated list of names of all jobs known to
       the project (whether the jobs are currently attached or not).
     • DSJ.PROJECTNAME String - name of the current project.
     • DSJ.HOSTNAME String - the host name of the server holding the current
       project.
Result may also return an error condition as follows:
     DSJE.BADTYPE        InfoType was unrecognized.
Syntax
     Result = DSGetStageInfo (JobHandle, StageName, InfoType)
JobHandle is the handle for the job as derived from DSAttachJob, or it may be
DSJ.ME to refer to the current job.
StageName is the name of the stage to be interrogated. It may also be DSJ.ME to
refer to the current stage if necessary.
InfoType specifies the information required and may be one of:
     DSJ.LINKLIST
     DSJ.STAGELASTERR
     DSJ.STAGENAME
     DSJ.STAGETYPE
     DSJ.STAGEINROWNUM
     DSJ.VARLIST
     DSJ.STAGESTARTTIMESTAMP
     DSJ.STAGEENDTIMESTAMP
     DSJ.STAGEDESC
     DSJ.STAGEINST
     DSJ.STAGECPU
     DSJ.LINKTYPES
     DSJ.STAGEELAPSED
     DSJ.STAGEPID
     DSJ.STAGESTATUS
     DSJ.CUSTINFOLIST
Result depends on the specified InfoType, as follows:
Remarks
When referring to a controlled job, DSGetStageInfo can be used either before or
after a DSRunJob has been issued. Any status returned following a successful call
to DSRunJob is guaranteed to relate to that run of the job.
Example
The following command requests the last error message for the loader stage of the
job qsales:
    stage_status = DSGetStageInfo(qsales_handle, "loader", ➥
    DSJ.STAGELASTERR)
Syntax
     Result = DSGetVarInfo (JobHandle, StageName, VarName, InfoType)
JobHandle is the handle for the job as derived from DSAttachJob, or it may be
DSJ.ME to refer to the current job.
StageName is the name of the stage to be interrogated. It may also be DSJ.ME to
refer to the current stage if necessary.
VarName is the name of the variable to be interrogated.
InfoType specifies the information required and can be one of:
     DSJ.VARVALUE
     DSJ.VARDESCRIPTION
Result depends on the specified InfoType, as follows:
     • DSJ.VARVALUE String - the value of the specified variable.
     • DSJ.VARDESCRIPTION String - description of the specified variable.
Result may also return an error condition as follows:
     DSJE.BADHANDLE          JobHandle was invalid.
     DSJE.BADTYPE            InfoType was not recognized.
     DSJE.NOTINSTAGE         StageName was DSJ.ME and the caller is
                             not running within a stage.
     DSJE.BADVAR             VarName was not recognized.
     DSJE.BADSTAGE           StageName does not refer to a known
                             stage in the job.
Logs an event message to a job other than the current one. (Use DSLogInfo,
DSLogFatal, or DSLogWarn to log an event to the current job.)
DSLogEvent
Syntax
    ErrCode = DSLogEvent (JobHandle, EventType, EventMsg)
JobHandle is the handle for the job as derived from DSAttachJob.
EventType is the type of event logged and is one of:
    DSJ.LOGINFO               Information message
    DSJ.LOGWARNING            Warning message
EventMsg is a string containing the event message.
ErrCode is 0 if there is no error. Otherwise it contains one of the following errors:
    DSJE.BADHANDLE            Invalid JobHandle.
    DSJE.BADTYPE              Invalid EventType (particularly note that
                              you cannot place a fatal message in
                              another job’s log).
Example
The following command, when included in the msales job, adds the message
“monthly sales complete” to the log for the qsales job:
    Logerror = DsLogEvent (qsales_handle, DSJ.LOGINFO,
    ➥ "monthly sales complete")
Logs a fatal error message in a job's log file and aborts the job.
DSLogFatal
Syntax
    Call DSLogFatal (Message, CallingProgName)
Message (input) is the warning message you want to log. Message is automatically
prefixed with the name of the current stage and the calling before/after
subroutine.
CallingProgName (input) is the name of the before/after subroutine that calls the
DSLogFatal subroutine.
Remarks
DSLogFatal writes the fatal error message to the job log file and aborts the job.
DSLogFatal never returns to the calling before/after subroutine, so it should be
used with caution. If a job stops with a fatal error, it must be reset using the
DataStage Director before it can be rerun.
In a before/after subroutine, it is better to log a warning message (using
DSLogWarn) and exit with a nonzero error code, which allows DataStage to stop
the job cleanly.
DSLogFatal should not be used in a transform. Use DSTransformError instead.
Example
    Call DSLogFatal("Cannot open file", "MyRoutine")
Syntax
    Call DSLogInfo (Message, CallingProgName)
Message (input) is the information message you want to log. Message is
automatically prefixed with the name of the current stage and the calling
program.
CallingProgName (input) is the name of the transform or before/after subroutine
that calls the DSLogInfo subroutine.
Remarks
DSLogInfo writes the message text to the job log file as an information message
and returns to the calling routine or transform. If DSLogInfo is called during the
test phase for a newly created routine in the DataStage Manager, the two
arguments are displayed in the results window.
Unlimited information messages can be written to the job log file. However, if a
lot of messages are produced the job may run slowly and the DataStage Director
may take some time to display the job log file.
Example
    Call DSLogInfo("Transforming: ":Arg1, "MyTransform")
This routine may be used to put an info message in the log file of the job controlling
this job, if any. If there isn't one, the call is just ignored.
Syntax
    Call DSLogToController(MsgString)
MsgString is the text to be logged. The log event is of type Information.
Remarks
If the current job is not under control, a silent exit is performed.
Example
    Call DSLogToController(“This is logged to parent”)
Syntax
    Call DSLogWarn (Message, CallingProgName)
Message (input) is the warning message you want to log. Message is automatically
prefixed with the name of the current stage and the calling before/after
subroutine.
CallingProgName (input) is the name of the before/after subroutine that calls the
DSLogWarn subroutine.
Remarks
DSLogWarn writes the message to the job log file as a warning and returns to the
calling before/after subroutine. If the job has a warning limit defined for it, when
the number of warnings reaches that limit, the call does not return and the job is
aborted.
DSLogWarn should not be used in a transform. Use DSTransformError instead.
Example
    If InputArg > 100 Then
        Call DSLogWarn("Input must be =< 100; received
           ":InputArg,"MyRoutine")
    End Else
        * Carry on processing unless the job aborts
    End
Syntax
    ReportText = DSMakeJobReport(JobHandle, ReportLevel, LineSeparator)
JobHandle is the string as returned from DSAttachJob.
ReportLevel specifies the type of report and is one of the following:
    • 0 – basic report. Text string containing start/end time, time elapsed and
      status of job.
    • 1 – stage/link detail. As basic report, but also contains information about
      individual stages and links within the job.
    • 2 – text string containing full XML report.
LineSeparator is the string used to separate lines of the report. Special values
recognised are:
    "CRLF" => CHAR(13):CHAR(10)
    "LF" => CHAR(10)
    "CR" => CHAR(13)
The default is CRLF if on Windows NT, else LF.
Remarks
If a bad job handle is given, or any other error is encountered, information is
added to the ReportText.
Example
    h$ = DSAttachJob(“MyJob”, DSJ.ERRNONE)
    rpt$ = DSMakeJobReport(h$,0,”CRLF”)
Syntax
    FullText = DSMakeMsg(Template, ArgList)
FullText is the message with parameters substituted
Template is the message template, in which %1, %2 etc. are to be substituted with
values from the equivalent position in ArgList. If the template string starts with a
number followed by "\", that is assumed to be part of a message id to be looked
up in the DataStage message file.
Note: If an argument token is followed by "[E]", the value of that argument is
assumed to be a job control error code, and an explanation of it will be inserted in
place of "[E]". (See the DSTranslateCode function.)
ArgList is the dynamic array, one field per argument to be substituted.
Remarks
This routine is called from job control code created by the JobSequence Generator.
It is basically an interlude to call DSRMessage which hides any runtime includes.
It will also perform local job parameter substitution in the message text. That is, if
called from within a job, it looks for substrings such as "#xyz#" and replaces them
with the value of the job parameter named "xyz".
Example
    t$ = DSMakeMsg(“Error calling DSAttachJob(%1)<L>%2”,
    ➥jb$:@FM:DSGetLastErrorMsg())
Used to ensure that a compiled job is in the correct state to be run or validated.
Syntax
    JobHandle = DSPrepareJob(JobHandle)
JobHandle is the handle, as returned from DSAttachJob(), of the job to be
prepared.
JobHandle is either the original handle or a new one. If returned as 0, an error
occurred and a message is logged.
Example
    h$ = DSPrepareJob(h$)
Starts a job running. Note that this call is asynchronous; the request is passed to the
run-time engine, but you are not informed of its progress.
DSRunJob
Syntax
    ErrCode = DSRunJob (JobHandle, RunMode)
JobHandle is the handle for the job as derived from DSAttachJob.
RunMode is the name of the mode the job is to be run in and is one of:
    DSJ.RUNNORMAL             (Default) Standard job run.
    DSJ.RUNRESET              Job is to be reset.
    DSJ.RUNVALIDATE           Job is to be validated only.
ErrCode is 0 if DSRunJob is successful, otherwise it is one of the following
negative integers:
    DSJE.BADHANDLE            Invalid JobHandle.
    DSJE.BADSTATE             Job is not in the right state (compiled,
                              not running).
    DSJE.BADTYPE              RunMode is not a known mode.
Remarks
If the controlling job is running in validate mode, then any calls of DSRunJob will
act as if RunMode was DSJ.RUNVALIDATE, regardless of the actual setting.
A job in validate mode will run its JobControl routine (if any) rather than just
check for its existence, as is the case for before/after routines. This allows you to
examine the log of what jobs it started up in validate mode.
After a call of DSRunJob, the controlled job’s handle is unloaded. If you require
to run the same job again, you must use DSDetachJob and DSAttachJob to set a
new handle. Note that you will also need to use DSWaitForJob, as you cannot
attach to a job while it is running.
Example
The following command starts the job qsales in standard mode:
Syntax
    Reply = DSSendMail(Parameters)
Parameters is a set of name:value parameters, separated by either a mark character
or "\n".
Currently recognized names (case-insensitive) are:
    "From"     Mail address of sender, e.g. Me@SomeWhere.com
               Can only be left blank if the local template file does not contain a
               "%from%" token.
    "To"       Mail address of recipient, e.g. You@ElseWhere.com
               Can only be left blank if the local template file does not contain a
               "%to%" token.
    "Subject" Something to put in the subject line of the message.
               Refers to the "%subject%" token. If left as "", a standard subject line
               will be created, along the lines of "From DataStage job: jobname"
    "Server" Name of host through which the mail should be sent.
               May be omitted on systems (such as Unix) where the SMTP host
               name can be and is set up externally, in which case the local
               template file presumably will not contain a "%server%" token.
    "Body"     Message body.
               Can be omitted. An empty message will be sent. If used, it must be
               the last parameter, to allow for getting multiple lines into the
               message, using "\n" for line breaks. Refers to the "%body%" token.
               Note: The text of the body may contain the tokens "%report% or
               %fullreport% anywhere within it, which will cause a report on the
               current job status to be inserted at that point. A full report contains
               stage and link information as well as job status.
Reply. Possible replies are:
    DSJE.NOERROR            (0) OK
    DSJE.NOPARAM            Parameter name missing - field does not look like
                            'name:value'
    DSJE.NOTEMPLATE Cannot find template file
    DSJE.BADTEMPLATE Error in template file
Remarks
The routine looks for a local file, in the current project directory, with a well-
known name. That is, a template to describe exactly how to run the local sendmail
command.
Example
    code = DSSendMail("From:me@here\nTo:You@there\nSubject:Hi
    ya\nBody:Line1\nLine2")
By default a controlled job inherits any row or warning limits from the controlling
job. These can, however, be overridden using the DSSetJobLimit function.
DSSetJobLimit
Syntax
     ErrCode = DSSetJobLimit (JobHandle, LimitType, LimitValue)
JobHandle is the handle for the job as derived from DSAttachJob.
LimitType is the name of the limit to be applied to the running job and is one of:
     DSJ.LIMITWARN             Job to be stopped after LimitValue
                               warning events.
     DSJ.LIMITROWS             Stages to be limited to LimitValue rows.
LimitValue is an integer specifying the value to set the limit to. Set this to 0 to
specify unlimited warnings.
ErrCode is 0 if DSSetJobLimit is successful, otherwise it is one of the following
negative integers:
     DSJE.BADHANDLE            Invalid JobHandle.
     DSJE.BADSTATE             Job is not in the right state (compiled,
                               not running).
     DSJE.BADTYPE              LimitType is not a known limiting
                               condition.
     DSJE.BADVALUE             LimitValue is not appropriate for the
                               limiting condition type.
Example
The following command sets a limit of 10 warnings on the qsales job before it is
stopped:
     LimitErr = DSSetJobLimit(qsales_handle,
     ➥ DSJ.LIMITWARN, 10)
Specifies job parameter values before running a job. Any parameter not set will be
defaulted.
DSSetParam
Syntax
    ErrCode = DSSetParam (JobHandle, ParamName, ParamValue)
JobHandle is the handle for the job as derived from DSAttachJob.
ParamName is a string giving the name of the parameter.
ParamValue is a string giving the value for the parameter.
ErrCode is 0 if DSSetParam is successful, otherwise it is one of the following
negative integers:
    DSJE.BADHANDLE           Invalid JobHandle.
    DSJE.BADSTATE            Job is not in the right state (compiled,
                             not running).
    DSJE.BADPARAM            ParamName is not a known parameter of
                             the job.
    DSJE.BADVALUE            ParamValue is not appropriate for that
                             parameter type.
Example
The following commands set the quarter parameter to 1 and the startdate
parameter to 1/1/97 for the qsales job:
    paramerr = DSSetParam (qsales_handle, "quarter", "1")
    paramerr = DSSetParam (qsales_handle, "startdate",
    ➥ "1997-01-01")
Applies only to the current job, and does not take a JobHandle parameter. It can be
used by any job in either a JobControl or After routine to set a termination code for
interrogation by another job. In fact, the code may be set at any point in the job, and
the last setting is the one that will be picked up at any time. So to be certain of
getting the actual termination code for a job the caller should use DSWaitForJob
and DSGetJobInfo first, checking for a successful finishing status.
Note: This routine is defined as a subroutine not a function because there are no
      possible errors.
DSSetUserStatus
Syntax
     Call DSSetUserStatus (UserStatus)
UserStatus String is any user-defined termination message. The string will be
logged as part of a suitable "Control" event in the calling job’s log, and stored for
retrieval by DSGetJobInfo, overwriting any previous stored string.
This string should not be a negative integer, otherwise it may be indistinguishable
from an internal error in DSGetJobInfo calls.
Example
The following command sets a termination code of “sales job done”:
     Call DSSetUserStatus("sales job done")
This routine should only be used after a DSRunJob has been issued. It immedi-
ately sends a stop request to the run-time engine. The call is asynchronous. If you
need to know that the job has actually stopped, you must call DSWaitForJob or use
the Sleep statement and poll for DSGetJobStatus. Note that the stop request gets
sent regardless of the job's current status.
DSStopJob
Syntax
    ErrCode = DSStopJob (JobHandle)
JobHandle is the handle for the job as derived from DSAttachJob.
ErrCode is 0 if DSStopJob is successful, otherwise it may be the following:
    DSJE.BADHANDLE           Invalid JobHandle.
Example
The following command requests that the qsales job is stopped:
    stoperr = DSStopJob(qsales_handle)
Logs a warning message to a job log file. This function is called from transforms
only.
DSTransformError
Syntax
     Call DSTransformError (Message, TransformName)
Message (input) is the warning message you want to log. Message is automatically
prefixed with the name of the current stage and the calling transform.
TransformName (input) is the name of the transform that calls the
DSTransformError subroutine.
Remarks
DSTransformError writes the message (and other information) to the job log file
as a warning and returns to the transform. If the job has a warning limit defined
for it, when the number of warnings reaches that limit, the call does not return
and the job is aborted.
In addition to the warning message, DSTransformError logs the values of all
columns in the current rows for all input and output links connected to the
current stage.
Example
     Function MySqrt(Arg1)
     If Arg1 < 0 Then
         Call DSTransformError("Negative value:"Arg1, "MySqrt")
         Return("0")    ;*transform produces 0 in this case
     End
         Result = Sqrt(Arg1) ;* else return the square root
     Return(Result)
Converts a job control status or error code into an explanatory text message.
Syntax
    Ans = DSTranslateCode(Code)
Code is:
           If Code > 0, it's assumed to be a job status.
           If Code < 0, it's assumed to be an error code.
           (0 should never be passed in, and will return "no error")
Remarks
If Code is not recognized, then Ans will report it.
Example
    code$ = DSGetLastErrorMsg()
    ans$ = DSTranslateCode(code$)
Suspend a job until a named file either exists or does not exist.
Syntax
    Reply = DSWaitForFile(Parameters)
Parameters is the full path of file to wait on. No check is made as to whether this is
a reasonable path (for example, whether all directories in the path exist). A path
name starting with "-", indicates a flag to check the non-existence of the path. It is
not part of the path name.
Parameters may also end in the form " timeout:NNNN" (or "timeout=NNNN")
This indicates a non-default time to wait before giving up. There are several
possible formats, case-insensitive:
    nnn              number of seconds to wait (from now)
    nnnS             ditto
    nnnM             number of minutes to wait (from now)
    nnnH             number of hours to wait (from now)
    nn:nn:nn         wait until this time in 24HH:MM:SS. If this or nn:nn time has
                     passed, will wait till next day.
The default timeout is the same as "12H".
The format may optionally terminate "/nn", indicating a poll delay time in
seconds. If omitted, a default poll time is used.
Reply may be:
    DSJE.NOERROR             (0) OK - file now exists or does not exist, depending on
                             flag.
    DSJE.BADTIME             Unrecognized Timeout format
    DSJE.NOFILEPATH File path missing
    DSJE.TIMEOUT             Waited too long
Examples
    Reply = DSWaitForFile("C:\ftp\incoming.txt timeout:2H")
    (wait 7200 seconds for file on C: to exist before it gives up.)
This function is only valid if the current job has issued a DSRunJob on the given
JobHandle(s). It returns if the/a job has started since the last DSRunJob has since
finished.
DSWaitForJob
Syntax
    ErrCode = DSWaitForJob (JobHandle)
JobHandle is the string returned from DSAttachJob. If commas are contained, it's a
comma-delimited set of job handles, representing a list of jobs that are all to be
waited for.
ErrCode is 0 if no error, else possible error values (<0) are:
    DSJE.BADHANDLE             Invalid JobHandle.
    DSJE.WRONGJOB              Job for this JobHandle was not run from within this
                               job.
ErrCode is >0 => handle of the job that finished from a multi-job wait.
Remarks
DSWaitForJob will wait for either a single job or multiple jobs.
Example
To wait for the return of the qsales job:
    WaitErr = DSWaitForJob(qsales_handle)
DTX
Syntax
      DTX (number [ ,size] )
Description
Use the DTX function to convert a decimal integer to its hexadecimal equivalent.
size indicates the minimum size which the hexadecimal character string should
have. This field is supplemented with zeros if appropriate.
If number evaluates to the null value, null is returned. If size is the null value, the
DTX function fails and the program terminates with a run-time error message.
Example
      X = 25
      Y = DTX (X)
      PRINT Y
      Y = DTX (X,4)
      PRINT Y
      END
This is the program output:
      19
      0019
EBCDIC
Syntax
    EBCDIC (expression)
Description
Use the EBCDIC function to convert each character of expression from its ASCII
representation value to its EBCDIC representation value. The EBCDIC and ASCII
functions perform complementary operations. Data that is not represented in
ASCII code produces undefined results.
If expression evaluates to the null value, the EBCDIC function fails and the program
terminates with a run-time error message.
Example
    X = 'ABC 123'
    Y = EBCDIC(X)
    PRINT "ASCII", "EBCDIC", " X "
    PRINT "------", "-----", "---"
    FOR I = 1 TO LEN (X)
    PRINT SEQ(X[I,1]) , SEQ(Y[I,1]),X[I,1]
    NEXT I
This is the program output:
    ASCII       EBCDIC      X
    ------      -----       ---
    65          193         A
    66          194         B
    67          195         C
    32          64
    49          241         1
    50          242         2
    51          243         3
ECHO
Syntax
    ECHO {ON | OFF | expression}
Description
Use the ECHO statement to control the display of input characters on the screen.
If ECHO ON is specified, subsequent input characters are displayed, or echoed, on
the screen. If ECHO OFF is specified, subsequent input characters are assigned to
the INPUT statement variables but are not displayed on the screen.
The ability to turn off character display is useful when the keyboard is to be used
for cursor movement or for entering password information. If expression evaluates
to true, ECHO is turned ON. If expression evaluates to false, ECHO is turned OFF.
If expression evaluates to the null value, it is treated as false, and ECHO is turned
OFF.
Example
    PROMPT ""
    ECHO OFF
    PRINT "ENTER YOUR PASSWORD"
    INPUT PWORD
    ECHO ON
This is the program output:
    ENTER YOUR PASSWORD
END
Syntax
      END
Description
Use the END statement to terminate a BASIC program or a section of an IF, READ,
or OPEN statement.
An END statement is the last statement in a BASIC program; it indicates the logical
end of the program. When an END statement that is not associated with an IF,
READ, or OPEN statement is encountered, execution of the program terminates.
You can use comments after the END statement.
You can also use the END statement with conditional statements in the body of a
program. In this case END indicates the end of a multistatement conditional
clause.
Example
      A="YES"
      IF A="YES" THEN
          PRINT "THESE TWO LINES WILL PRINT ONLY"
          PRINT "WHEN THE VALUE OF 'A' IS 'YES'."
      END
      *
      PRINT
      PRINT "THIS IS THE END OF THE PROGRAM"
      END ; * END IS THE LAST STATEMENT EXECUTED
This is the program output:
      THESE TWO LINES WILL PRINT ONLY
      WHEN THE VALUE OF 'A' IS 'YES'.
END CASE
Use the END CASE statement to end a set of statements
END TRANSACTION
Use the END TRANSACTION statement to specify where processing is to
continue after a transaction ends.
ENTER
Syntax
    ENTER subroutine
    variable = 'subroutine'
    ENTER @variable
Description
Use the ENTER statement to transfer program control from the calling program to
an external subroutine without returning to the calling program. The subroutine
must have been compiled and cataloged.
The ENTER statement is similar to the CALL statement, except that with the
ENTER statement, program flow does not return from the entered program to the
calling program (see the CALL statement). The ENTER statement also does not
accept arguments.
In the PIOPEN flavor, the ENTER statement is a synonym for the CALL statement.
It takes arguments and returns control to the calling program.
External subroutines can be entered directly or indirectly. To enter a subroutine
indirectly, assign the name of the cataloged subroutine to a variable or to an
element of an array. Use the name of this variable or array element, prefixed with
an at sign (@), as the operand of the ENTER statement.
If subroutine evaluates to the null value, the ENTER statement fails and the
program terminates with a run-time error message.
Example
The following program transfers control to the cataloged program PROGRAM2:
    ENTER PROGRAM2
EOF(ARG.)
Syntax
    EOF(ARG.)
Description
Use the EOF(ARG.) function to check if the command line argument pointer is past
the last command line argument. ARG. is part of the syntax of the EOF(ARG.) func-
tion and must be specified. EOF(ARG.) returns 1 (true) if the pointer is past the last
command line argument, otherwise it returns 0 (false).
The arg# argument of the GET(ARG.) and SEEK(ARG.) statements affect the value
of the EOF(ARG.) function.
EQS
Syntax
      EQS (array1, array2)
      CALL −EQS (return.array, array1, array2)
      CALL !EQS (return.array, array1, array2)
Description
Use the EQS function to test if elements of one dynamic array are equal to the
elements of another dynamic array.
Each element of array1 is compared with the corresponding element of array2. If the
two elements are equal, a 1 is returned in the corresponding element of a dynamic
array. If the two elements are not equal, a 0 is returned. If an element of one
dynamic array has no corresponding element in the other dynamic array, a 0 is
returned. If either element of a corresponding pair is the null value, null is returned
for that element.
If you use the subroutine syntax, the resulting dynamic array returns as
return.array.
Example
      A=1:@VM:45:@SM:3:@VM:"one"
      B=0:@VM:45:@VM:1
      PRINT EQS(A,B)
This is the program output:
      0V1S0V0
EQUATE
Syntax
    EQU[ATE] symbol TO expression [ ,symbol TO expression …]
Description
In an EQUATE statement, symbol represents the value of expression or string. You
can use the two interchangeably in the program. When the program is compiled,
each occurrence of symbol is replaced by the value of expression or string. The value
is compiled as object code and does not have to be reassigned each time the
program is executed.
You can define multiple symbols in a single EQUATE statement. symbol cannot be
a number.
You can define symbol only once. Any subsequent EQUATE state generates a
compiler error because the compiler interprets the symbol before the statement is
parsed.
If you use TO as a connector, the object can be any BASIC expression. If you use
LIT or LITERALLY as a connector, the object must be a literal string.
RAID does not recognize EQUATE symbols. You must use the object value in
RAID sessions.
There is no limit on the number of EQUATE statements allowed by the BASIC
compiler, except that of memory.
If symbol is the same as the name of a BASIC function, the function is disabled in
the program. If a statement exists with the same name as a disabled function, the
statement is also disabled.
Examples
In the following example, A is made equivalent to the string JANE:
    JANE="HI"
    EQUATE A TO "JANE"
EREPLACE
Syntax
    EREPLACE (expression, substring, replacement [,occurrence [,begin] ] )
Description
Use the EREPLACE function to replace substring in expression with another
substring. If you do not specify occurrence, each occurrence of substring is replaced.
occurrence specifies the number of occurrences of substring to replace. To replace all
occurrences, specify occurrence as a number less than 1.
begin specifies the first occurrence to replace. If begin is omitted or less than 1, it
defaults to 1.
If substring is an empty string, replacement is prefixed to expression. If replacement is
an empty string, all occurrences of substring are removed.
If expression evaluates to the null value, null is returned. If substring, replacement,
occurrence, or begin evaluates to the null value, the EREPLACE function fails and
the program terminates with a run-time error message.
The EREPLACE function behaves like the CHANGE function except when
substring evaluates to an empty string.
Example
    A = "AAABBBCCCDDDBBB"
    PRINT EREPLACE (A,"BBB","ZZZ")
    PRINT EREPLACE (A,"","ZZZ")
    PRINT EREPLACE (A,"BBB","")
This is the program output:
    AAAZZZCCCDDDZZZ
    ZZZAAABBBCCCDDDBBB
    AAACCCDDD
ERRMSG
Syntax
    ERRMSG message.ID [ ,message.ID …]
Description
Use the ERRMSG statement to print a formatted error message from the ERRMSG
file.
message.ID is an expression evaluating to the record ID of a message to be printed
on the screen. Additional expressions are evaluated as arguments that can be
included in the error message.
If message.ID evaluates to the null value, the default error message is printed:
    Message ID is NULL:         undefined error
A standard Pick ERRMSG file is supplied with DataStage. Users can construct a
local ERRMSG file using the following syntax in the records. Each field must start
with one of these codes shown in the following table:
Code          Action
A[(n)]        Display next argument left-justified; n specifies field length.
D             Display system date.
E [string]    Display record ID of message in brackets; string displayed after ID.
H [string]    Display string.
L [(n)]       Output a newline; n specifies number of newlines.
R [(n)]       Display next argument right-justified; n specifies field length.
S [(n)]       Output n blank spaces from beginning of line.
T             Display system time.
Example
    >ED ERRMSG 1
    7 lines long.
    ----: P
    0001: HBEGINNING OF ERROR MESSAGE
    0002: L
EXCHANGE
Syntax
    EXCHANGE (string, xx, yy)
Description
Use the EXCHANGE function to replace one character with another or to delete all
occurrences of the specified character.
string is an expression evaluating to the string whose characters are to be replaced
or deleted. If string evaluates to the null value, null is returned.
xx is an expression evaluating to the character to be replaced, in hexadecimal
notation.
yy is an expression evaluating to the replacement character, also in hexadecimal
notation.
If yy is FF, all occurrences of xx are deleted. If xx or yy consist of fewer than two
characters, no conversion is done. If xx or yy is the null value, the EXCHANGE
function fails and the program terminates with a run-time error message.
If NLS is enabled, EXCHANGE uses only the first two bytes of xx and yy in order
to evaluate the characters. Note how the EXCHANGE function evaluates the
following characters:
For more information about character values, see the DataStage NLS Guide.
Example
In the following example, 41 is the hexadecimal value for the character A and 2E is
the hexadecimal value for the period character ( . ):
    PRINT EXCHANGE('ABABC','41','2E')
EXECUTE
Syntax
    EXECUTE commands [CAPTURING variable] [PASSLIST [dynamic.array] ]
            [RTNLIST [variable] ] [ {SETTING | RETURNING} variable]
    EXECUTE commands [ ,IN < expression] [ ,OUT > variable]
            [ ,SELECT[ (list) ] < dynamic.array] [ ,SELECT[ (list) ] > variable]
            [ ,PASSLIST [ (dynamic.array) ] ] [ ,STATUS > variable]
    EXECUTE commands [ ,//IN. < expression] [ ,//OUT. > variable]
            [ ,//SELECT.[ (list) ] < dynamic.array] [ ,//SELECT.[ (list) ] > variable]
            [ ,//PASSLIST.[ (dynamic.array) ] ] [ ,//STATUS. > variable]
Description
Use the EXECUTE statement to execute DataStage commands from within the
BASIC program and then return execution to the statement following the
EXECUTE statement.
EXECUTE creates a new environment for the executed command. This new envi-
ronment is initialized with the values of the current prompt, current printer state,
Break key counter, the values of in-line prompts, KEYEDITs, KEYTRAPs, and
KEYEXITs. If any of these values change in the new environment, the changes are
not passed back to the calling environment. In the new environment, stacked
@Variables are either initialized to 0 or set to reflect the new environment.
Nonstacked @variables are shared between the EXECUTE and calling
environments.
commands can be sentences, paragraphs, verbs, procs, menus, or BASIC programs.
You can specify multiple commands in the EXECUTE statement in the same way
they are specified in a UniVerse paragraph. Each command or line must be sepa-
rated by a field mark (ASCII CHAR 254).
The EXECUTE statement has two main syntaxes. The first syntax requires options
to be separated by spaces. The second and third syntaxes require options to be
separated by commas. In the third syntax, the "//" preceding the keywords and
the periods following them are optional; the compiler ignores these marks. Except
for the slashes and periods, the second and third syntaxes are the same.
In the first syntax the CAPTURING clause assigns the output of the executed
commands to variable. The PASSLIST clause passes the current active select list or
expression to the commands for use as select list 0. The RTNLIST option assigns
select list 0, created by the commands, to variable. If you do not specify variable, the
RTNLIST clause is ignored. Using the SETTING or RETURNING clause causes the
@SYSTEM.RETURN.CODE of the last executed command to be placed in variable.
In the second syntax the executed commands use the value of expression in the IN
clause as input. When the IN clause is used, the DATA queue is passed back to the
calling program, otherwise data is shared between environments. The OUT clause
assigns the output of the commands to variable. The SELECT clauses let you supply
the select list stored in expression as a select list to the commands, or to assign a
select list created by the commands to variable. If list is not specified, select list 0 is
used. The PASSLIST clause passes the currently active select list to the commands.
If you do not specify list, select list 0 in the current program’s environment is
passed as select list 0 in the executed command’s environment. The STATUS clause
puts the @SYSTEM.RETURN.CODE of the last executed command in variable.
The EXECUTE statement fails and the program terminates with a run-time error
message if:
    • dynamic.array or expression in the IN clause evaluates to the null value.
    • The command expression evaluates to the null value.
In transactions you can use only the following DataStage commands and SQL
statements with EXECUTE:
    CHECK.SUM              INSERT            SEARCH                      SSELECT
    COUNT                  LIST              SELECT (RetrieVe)           STAT
    DELETE (SQL)           LIST.ITEM         SELECT (SQL)                SUM
    DISPLAY                LIST.LABEL        SORT                        UPDATE
    ESEARCH                RUN               SORT.ITEM
INFORMATION Flavor
In INFORMATION flavor accounts, the EXECUTE statement without any options
is the same as the PERFORM statement. In this case executed commands keep the
same environment as the BASIC program that called them. Use the
EXEC.EQ.PERF option of the $OPTIONS statement to cause EXECUTE to behave
like PERFORM in other flavors.
Example
The following example performs a nested SELECT, demonstrating the use of the
CAPTURING, RTNLIST, and PASSLIST keywords:
    CMD = "SELECT VOC WITH TYPE = V"
    EXECUTE CMD RTNLIST VERBLIST1
    CMD = "SELECT VOC WITH NAME LIKE ...LIST..."
    EXECUTE CMD PASSLIST VERBLIST1 RTNLIST VERBLIST2
    CMD = "LIST VOC NAME"
    EXECUTE CMD CAPTURING RERUN PASSLIST VERBLIST2
    PRINT RERUN
The program first selects all VOC entries that define verbs, passing the select list to
the variable VERBLIST1. Next, it selects from VERBLIST1 all verbs whose names
contain the string LIST and passes the new select list to VERBLIST2. The list in
VERBLIST2 is passed to the LIST command, whose output is captured in the vari-
able RERUN, which is then printed.
EXIT
Syntax
       EXIT
Description
Use the EXIT statement to quit execution of a FOR…NEXT loop or a
LOOP…REPEAT loop and branch to the statement following the NEXT or
REPEAT statement of the loop. The EXIT statement quits exactly one loop. When
loops are nested and the EXIT statement is executed in an inner loop, the outer loop
remains in control.
Example
       COUNT = 0
       LOOP
       WHILE COUNT < 100 DO
          INNER = 0
          LOOP
          WHILE INNER < 100 DO
              COUNT += 1
              INNER += 1
              IF INNER = 50 THEN EXIT
          REPEAT
          PRINT "COUNT = ":COUNT
       REPEAT
This is the program output:
       COUNT = 50
       COUNT = 100
EXP
Syntax
      EXP (expression)
Description
Use the EXP function to return the value of "e" raised to the power designated by
expression. The value of "e" is approximately 2.71828. expression must evaluate to a
numeric value.
If expression is too large or small, a warning message is printed and 0 is returned. If
expression evaluates to the null value, null is returned.
The formula used by the EXP function to perform the calculations is
      value of EXP function = 2.71828**(expression)
Example
      X=5
      PRINT EXP(X-1)
This is the program output:
      54.5982
EXTRACT
Syntax
    EXTRACT (dynamic.array, field# [,value# [,subvalue#] ] )
Description
Use the EXTRACT function to access the data contents of a specified field, value,
or subvalue from a dynamic array. You can use either syntax shown to extract data.
The first syntax uses the EXTRACT keyword, the second uses angle brackets.
dynamic.array is an expression that evaluates to the array in which the field, value,
or subvalue to be extracted is to be found. If dynamic.array evaluates to the null
value, null is returned.
field# specifies the field in the dynamic array; value# specifies the value in the field;
subvalue# specifies the subvalue in the value. These arguments are called delimiter
expressions. The numeric values of the delimiter expressions determine whether a
field, a value, or a subvalue is to be extracted. value# and subvalue# are optional.
Angle brackets used as an EXTRACT function appear on the right side of an
assignment statement. Angle brackets on the left side of the assignment statement
indicate that a REPLACE function is to be performed (for examples, see the
REPLACE function).
The second syntax uses angle brackets to extract data from dynamic arrays. variable
specifies the dynamic array containing the data to be extracted. field#, value#, and
subvalue# are delimiter expressions.
Here are the five outcomes that can result from the different uses of delimiter
expressions:
Case 4:     If field#, value#, and subvalue# are all specified and are all nonzero, the
            specified subvalue is extracted.
Case 5:     If field#, value#, or subvalue# evaluates to the null value, the EXTRACT
            function fails and the program terminates with a run-time error
            message.
If a higher-level delimiter expression has a value of 0 when a lower-level delimiter
is greater than 0, a 1 is assumed. The delimiter expressions are from highest to
lowest: field, value, and subvalue.
If the EXTRACT function references a subelement of an element whose value is the
null value, null is returned.
Example
In the following example a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S:
    VAR=1:@FM:4:@VM:9:@SM:3:@SM:5:@FM:1:@VM:0:@SM:7:@SM:3
    Z=EXTRACT(VAR,1,0,0)
    PRINT Z
    *
    Z=VAR<1,1,1>
    PRINT Z
    *
    Z=EXTRACT(VAR,2,1,1)
    PRINT Z
    *
    Z=VAR<3,2,3>
    PRINT Z
    *
    Z=EXTRACT(VAR,10,0,0)
    PRINT Z
    *
    Z=EXTRACT(VAR,2,2,0)
    PRINT Z
    *
This is the program output:
    1
    1
    4
   3
   9S3S5
FADD
Syntax
    FADD (number1, number2)
    CALL !FADD (return.array, number1, number2)
Description
Use the FADD function to perform floating-point addition on two numeric values.
If either number evaluates to the null value, null is returned. If either number1 or
number2 evaluates to the null value, null is returned. return.array equates to
number1 plus number2.
This function is provided for compatibility with existing software. You can also use
the + operator to perform floating-point addition.
Example
    PRINT FADD(.234,.567)
This is the program output:
    0.801
FDIV
Syntax
       FDIV (number1, number2)
       CALL !FDIV (return.array, number1, number2)
Description
Use the FDIV function to perform floating-point division on two numeric values.
number1 is divided by number2. return.array equates to number1 divided by
number2. If number2 is 0, a run-time error message is produced and a 0 is returned
for the function. If either number evaluates to the null value, null is returned.
This function is provided for compatibility with existing software. You can also use
the / operator to perform floating-point division.
Example
       PRINT FDIV(.234,.567)
This is the program output:
       0.4127
FFIX
Syntax
       FFIX (number)
Description
Use the FFIX function to convert a floating-point number to a numeric string with
fixed precision. If number evaluates to the null value, null is returned.
This function is provided for compatibility with existing software.
FFLT
Syntax
       FFLT (number)
Description
Use the FFLT function to round a number to a string with a precision of 13. The
number also converts to scientific notation when required for precision. If number
evaluates to the null value, null is returned.
FIELD
Syntax
    FIELD (string, delimiter, occurrence [ ,num.substr] )
Description
Use the FIELD function to return one or more substrings located between specified
delimiters in string.
delimiter evaluates to any character, including field mark, value mark, and
subvalue marks. It delimits the start and end of the substring. If delimiter evaluates
to more than one character, only the first character is used. Delimiters are not
returned with the substring.
occurrence specifies which occurrence of the delimiter is to be used as a terminator.
If occurrence is less than 1, 1 is assumed.
num.substr specifies the number of delimited substrings to return. If the value of
num.substr is an empty string or less than 1, 1 is assumed. When more than one
substring is returned, delimiters are returned along with the successive substrings.
If either delimiter or occurrence is not in the string, an empty string is returned,
unless occurrence specifies 1. If occurrence is 1 and delimiter is not found, the entire
string is returned. If delimiter is an empty string, the entire string is returned.
If string evaluates to the null value, null is returned. If string contains CHAR(128)
(that is, @NULL.STR), it is treated like any other character in a string. If delimiter,
occurrence, or num.substr evaluate to the null value, the FIELD function fails and the
program terminates with a run-time error message.
The FIELD function works identically to the GROUP function.
Examples
    D=FIELD("###DHHH#KK","#",4)
    PRINT "D= ",D
The variable D is set to DHHH because the data between the third and fourth
occurrence of the delimiter # is DHHH.
    REC="ACADABA"
    E=FIELD(REC,"A",2)
    PRINT "E= ",E
FIELDS
Syntax
    FIELDS (dynamic.array, delimiter, occurrence [ ,num.substr] )
    CALL −FIELDS (return.array, dynamic.array, delimiter, occurrence, num.substr)
    CALL !FIELDS (return.array, dynamic.array, delimiter, occurrence, num.substr)
Description
Use the FIELDS function to return a dynamic array of substrings located between
specified delimiters in each element of dynamic.array.
delimiter evaluates to any character, including value and subvalue characters. It
marks the start and end of the substring. If delimiter evaluates to more than one
character, the first character is used.
occurrence specifies which occurrence of the delimiter is to be used as a terminator.
If occurrence is less than 1, 1 is assumed.
num.substr specifies the number of delimited substrings to return. If the value of
num.substr is an empty string or less than 1, 1 is assumed. In this case delimiters are
returned along with the successive substrings. If delimiter or occurrence does not
exist in the string, an empty string is returned, unless occurrence specifies 1. If occur-
rence is 1 and the specified delimiter is not found, the entire element is returned. If
occurrence is 1 and delimiter is an empty string, an empty string is returned.
If dynamic.array is the null value, null is returned. If any element in dynamic.array is
the null value, null is returned for that element. If delimiter, occurrence, or num.substr
evaluates to the null value, the FIELDS function fails and the program terminates
with a run-time error message.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    A="000-P-0":@VM:"-H--O-":@SM:"N-I-T":@VM:"BC":@SM:"-L-"
    PRINT FIELDS(A,"-",2)
This is the program output:
    PVHSIVSL
FIELDSTORE
Syntax
    FIELDSTORE (string, delimiter, start, n, new.string)
Description
Use the FIELDSTORE function to modify character strings by inserting, deleting,
or replacing fields separated by specified delimiters.
string is an expression that evaluates to the character string to be modified.
delimiter evaluates to any single ASCII character, including field, value, and
subvalue marks.
start evaluates to a number specifying the starting field position. Modification
begins at the field specified by start. If start is greater than the number of fields in
string, the required number of empty fields is generated before the FIELDSTORE
function is executed.
n specifies the number of fields of new.string to insert in string. n determines how
the FIELDSTORE operation is executed. If n is positive, n fields in string are
replaced with the first n fields of new.string. If n is negative, n fields in string are
replaced with all the fields in new.string. If n is 0, all the fields in new.string are
inserted in string before the field specified by start.
If string evaluates to the null value, null is returned. If delimiter, start, n, or
new.string is null, the FIELDSTORE function fails and the program terminates with
a run-time error message.
Example
    Q='1#2#3#4#5'
    *
    TEST1=FIELDSTORE(Q,"#",2,2,"A#B")
    PRINT "TEST1= ",TEST1
    *
    TEST2=FIELDSTORE(Q,"#",2,-2,"A#B")
    PRINT "TEST2= ",TEST2
    *
    TEST3=FIELDSTORE(Q,"#",2,0,"A#B")
    PRINT "TEST3= ",TEST3
    *
    TEST4=FIELDSTORE(Q,"#",1,4,"A#B#C#D")
FILEINFO
Syntax
    FILEINFO (file.variable, key)
file.variable is the file variable of an open file.
key is a number that indicates the particular information required. These key
numbers are described in the table “Keys and Values Supplied to the FILEINFO
Function.”
Description
Use the FILEINFO function to return information about the specified file’s config-
uration, such as the specified file’s parameters, its modulus and load, its operating
system filename, and its VOC name. The information returned depends on the file
type and the value of the key.
If the first argument is not a file variable, all keys except 0 return an empty string.
A warning message is also displayed. A fatal error results if an invalid key is
supplied.
Note: The first time that an I/O operation fails for a part file in a distributed file,
      the FILEINFO function returns an error code for that part file. For any
      subsequent I/O operations on the distributed file with the same unavail-
      able part file, the FILEINFO function returns –1.
NLS Mode
The FILEINFO function determines the map name of a file by using the value of
FINFO$NLSMAP. NLS uses the insert file called FILEINFO.H. For more informa-
tion about maps, see the DataStage NLS Guide.
Examples
In the following example, the file containing the key equate names is inserted with
the $INCLUDE statement. The file FILMS is opened and its file type displayed.
    $INCLUDE SYSCOM FILEINFO.INS.IBAS
    OPEN '','FILMS' TO FILMS
            ELSE STOP 'CANT OPEN FILE'
    PRINT FILEINFO(FILMS,FINFO$TYPE)
In the next example, the file FILMS is opened and its file type displayed by speci-
fying the numeric key:
    OPEN '','FILMS' TO FILMS
            ELSE STOP 'CANT OPEN FILE'
    PRINT FILEINFO(FILMS,3)
FILELOCK
Syntax
    FILELOCK [ file.variable ] [ ,lock.type ]
             [ON ERROR statements] [LOCKED statements]
Description
Use the FILELOCK statement to acquire a lock on an entire file. This prevents other
users from updating the file until the program releases it. A FILELOCK statement
that does not specify lock.type is equivalent to obtaining an update record lock on
every record of the file.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information on default files, see the OPEN statement). If the file
is neither accessible nor open, the program terminates with a run-time error
message. If file.variable evaluates to the null value, the FILELOCK statement fails
and the program terminates with a run-time error message.
lock.type is an expression that evaluates to one of the following keywords:
    • SHARED (to request an FS lock)
    • INTENT (to request an IX lock)
    • EXCLUSIVE (to request an FX lock)
If the FILELOCK statement does not include a LOCKED clause and a conflicting
lock exists, the program pauses until the lock is released.
If a LOCKED clause is used, the value returned by the STATUS function is the
terminal number of the user who owns the conflicting lock.
Releasing Locks
A shared, intent, or exclusive file lock can be released by a FILEUNLOCK,
RELEASE, or STOP statement.
Locks acquired or promoted within a transaction are not released when previous
statements are processed.
Examples
    OPEN '','SUN.MEMBER' TO DATA ELSE STOP "CAN'T OPEN FILE"
    FILELOCK DATA LOCKED STOP 'FILE IS ALREADY LOCKED'
    FILEUNLOCK DATA
    OPEN '','SUN.MEMBER' ELSE STOP "CAN'T OPEN FILE"
    FILELOCK LOCKED STOP 'FILE IS ALREADY LOCKED'
    PRINT "The file is locked."
    FILEUNLOCK
This is the program output:
    The file is locked.
The following example acquires an intent file lock:
    FILELOCK fvar, "INTENT" LOCKED
        owner = STATUS( )
        PRINT "File already locked by":owner
        STOP
    END
FILEUNLOCK
Syntax
    FILEUNLOCK [file.variable] [ON ERROR statements]
Description
Use the FILEUNLOCK statement to release a file lock set by the FILELOCK
statement.
file.variable specifies a file previously locked with a FILELOCK statement. If
file.variable is not specified, the default file with the FILELOCK statement is
assumed (for more information on default files, see the OPEN statement). If
file.variable evaluates to the null value, the FILEUNLOCK statement fails and the
program terminates with a run-time error message.
The FILEUNLOCK statement releases only file locks set with the FILELOCK state-
ment. Update record locks must be released with one of the other unlocking
statements (for example, WRITE, WRITEV, and so on).
If the ON ERROR clause is used, the value returned by the STATUS function is the
error number. The ON ERROR clause is not supported if the FILEUNLOCK state-
ment is within a transaction.
Example
In the following example, the first FILEUNLOCK statement unlocks the default
file. The second FILEUNLOCK statement unlocks the file variable FILE.
    OPEN '','SUN.MEMBER' ELSE STOP "CAN'T OPEN SUN.MEMBER"
    FILELOCK
       .
       .
       .
    FILEUNLOCK
    OPEN 'EX.BASIC' TO FILE ELSE STOP
    FILELOCK FILE
       .
       .
       .
    FILEUNLOCK FILE
FIND
Syntax
       FIND element IN dynamic.array [ ,occurrence] SETTING fmc [ ,vmc [ ,smc] ]
               {THEN statements [ELSE statements] | ELSE statements}
Description
Use the FIND statement to locate an element in dynamic.array. The field, value, and
subvalue positions of element are put in the variables fmc, vmc, and smc respectively.
element evaluates to a character string. FIND succeeds only if the string matches an
element in its entirety. If element is found in dynamic.array, any THEN statements
are executed. If element is not found, or if dynamic.array evaluates to the null value,
fmc, vmc, and smc are unchanged, and the ELSE statements are executed.
If occurrence is unspecified, it defaults to 1. If occurrence is the null value, the FIND
statement fails and the program terminates with a run-time error message.
Example
       A="THIS":@FM:"IS":@FM:"A":@FM:"DYNAMIC":@FM:"ARRAY"
       FIND "IS" IN A SETTING FM,VM,SM ELSE ABORT
       PRINT "FM=",FM
       PRINT "VM=",VM
       PRINT "SM=",SM
This is the program output:
       FM=     2
       VM=     1
       SM=     1
FINDSTR
Syntax
    FINDSTR substring IN dynamic.array [ ,occurrence]
             SETTING fmc [ ,vmc [ ,smc] ]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the FINDSTR statement to locate substring in dynamic.array. The field, value,
and subvalue positions of substring are placed in the variables fmc, vmc, and smc
respectively.
FINDSTR succeeds if it finds substring as part of any element in dynamic array. If
substring is found in dynamic.array, any THEN statements are executed. If substring
is not found, or if dynamic.array evaluates to the null value, fmc, vmc, and smc are
unchanged, and the ELSE statements are executed.
If occurrence is unspecified, it defaults to 1. If occurrence is the null value, FINDSTR
fails and the program terminates with a run-time error message.
Example
    A="THIS":@FM:"IS":@FM:"A":@FM:"DYNAMIC":@FM:"ARRAY"
    FINDSTR "IS" IN A SETTING FM,VM,SM ELSE ABORT
    PRINT "FM=",FM
    PRINT "VM=",VM
    PRINT "SM=",SM
This is the program output:
    FM=      1
    VM=      1
    SM=      1
FIX
Syntax
      FIX (number [ ,precision [ ,mode ] ] )
Description
Use the FIX function to convert a numeric value to a floating-point number with a
specified precision. FIX lets you control the accuracy of computation by elimi-
nating excess or unreliable data from numeric results. For example, a bank
application that computes the interest accrual for customer accounts does not need
to deal with credits expressed in fractions of cents. An engineering application
needs to throw away digits that are beyond the accepted reliability of
computations.
number is an expression that evaluates to the numeric value to be converted.
precision is an expression that evaluates to the number of digits of precision in the
floating-point number. If you do not specify precision, the precision specified by the
PRECISION statement is used. The default precision is 4.
mode is a flag that specifies how excess digits are handled. If mode is either 0 or not
specified, excess digits are rounded off. If mode is anything other than 0, excess
digits are truncated.
If number evaluates to the null value, null is returned.
Examples
The following example calculates a value to the default precision of 4:
      REAL.VALUE = 37.73629273
      PRINT FIX (REAL.VALUE)
This is the program output:
      37.7363
The next example calculates the same value to two digits of precision. The first
result is rounded off, the second is truncated:
      PRINT FIX (REAL.VALUE, 2)
      PRINT FIX (REAL.VALUE, 2, 1)
FLUSH
Syntax
    FLUSH file.variable {THEN statements [ELSE statements] | ELSE statements}
Description
The FLUSH statement causes all the buffers for a sequential I/O file to be written
immediately. Normally, sequential I/O uses UNIX "stdio" buffering for
input/output operations, and writes are not performed immediately.
file.variable specifies a file previously opened for sequential processing. If file.vari-
able evaluates to the null value, the FLUSH statement fails and the program
terminates with a run-time error message.
After the buffer is written to the file, the THEN statements are executed, and the
ELSE statements are ignored. If THEN statements are not present, program execu-
tion continues with the next statement.
If the file cannot be written to or does not exist, the ELSE statements are executed;
any THEN statements are ignored.
See the OPENSEQ and WRITESEQ statements for more information on sequential
file processing.
Example
    OPENSEQ 'FILE.E', 'RECORD1' TO FILE THEN
        PRINT "'FILE.E' OPENED FOR SEQUENTIAL PROCESSING"
    END ELSE STOP
    WEOFSEQ FILE
    *
    WRITESEQ 'NEW LINE' ON FILE THEN
        FLUSH FILE THEN
           PRINT "BUFFER FLUSHED"
        END ELSE PRINT "NOT FLUSHED"
    ELSE ABORT
    *
    CLOSESEQ FILE
    END
FMT
Syntax
      FMT (expression, format)
      expression format
Description
Use the FMT function or a format expression to format data for output. Any BASIC
expression can be formatted for output by following it with a format expression.
expression evaluates to the numeric or string value to be formatted.
format is an expression that evaluates to a string of formatting codes. The syntax of
the format expression is:
Exponential notation:
mask lets literals be intermixed with numerics in the formatted output field. The
mask can include any combination of literals and the following three special
format mask characters:
If you want to use numeric characters or any of the special characters as literals,
you must escape the character with a backslash ( \ ).
A #, %, or * character followed by digits causes the background fill character to be
repeated n times. Other characters followed by digits cause those characters to
appear in the output data n times.
mask can be enclosed in parentheses ( ) for clarity. If mask contains parentheses, you
must include the whole mask in another set of parentheses. For example:
     ((###) ###-####)
You must specify either width or mask in the FMT function. You can specify both in
the same function. When you specify width, the string is formatted according to the
following rules:
If string is smaller than width n, it is padded with fill characters.
If string is larger than width n, a text mark (CHAR(251)) is inserted every nth char-
acter and each field is padded with the fill character to width.
The STATUS function reflects the result of edit as follows:
REALITY Flavor
In REALITY flavor accounts, you can use conversion codes in format expressions.
Examples
    X="555666898"                         X=       *****$555,666,898.00
    X=FMT(X,"20*R2$,")
    Y="DAVID"                             Y=       DAVID.....
    Y=FMT(Y,"10.L")
    V="24500"                             V=          $24500.00
    V=FMT(V,"10R2$Z")
R=FMT(77777,"R#10") R= 77777
    B="0.12345678E1"                      B=       *1.2346E0
    B=FMT(B,"9*Q")
FMTDP
Syntax
    FMTDP (expression, format [ , mapname ] )
Description
In NLS mode, use the FMTDP function to format data for output in display posi-
tions rather than character lengths.
expression evaluates to the numeric or string value to be formatted. Any unmap-
pable characters in expression are assumed to have a display length of 1.
format is an expression that evaluates to a string of formatting codes. The syntax of
the format expression is:
FMTS
Syntax
    FMTS (dynamic.array, format)
    CALL −FMTS (return.array, dynamic.array, format)
    CALL !FMTS (return.array, dynamic.array, format)
Description
Use the FMTS function to format elements of dynamic.array for output. Each
element of the array is acted upon independently and is returned as an element in
a new dynamic array.
format is an expression that evaluates to a string of formatting codes. The syntax of
the format expression is:
FMTSDP
Syntax
    FMTSDP (dynamic.array, format [ , mapname ] )
Description
In NLS mode, use the FMTSDP function to format elements of dynamic.array for
output in display positions rather than character lengths. Each element of the array
is acted upon independently and is returned as an element in a new dynamic array.
Any unmappable characters in dynamic.array are assumed to have a display length
of 1.
format is an expression that evaluates to a string of formatting codes. The syntax of
the format expression is:
Note: If you execute FMTSDP when NLS is disabled, the behavior is the same as
      for FMTS.
For more information about display length, see the DataStage NLS Guide.
FMUL
Syntax
    FMUL (number1, number2)
    CALL !FMUL (return.array, number1, number2)
Description
Use the FMUL function to perform floating-point multiplication on two numeric
values. If either number evaluates to the null value, null is returned. return.array
equates to number1 multiplied by number2.
This function is provided for compatibility with existing software. You can also use
the * operator to perform floating-point multiplication.
Example
    PRINT FMUL(.234,.567)
This is the program output:
    0.1327
FOLD
Syntax
    FOLD (string, length )
    CALL !FOLD (subdivided.string, string, length)
Description
Use the FOLD function to divide a string into a number of substrings separated by
field marks.
string is separated into substrings of length less than or equal to length. string is
separated on blanks, if possible, otherwise it is separated into substrings of the
specified length.
subdivided.string contains the result of the FOLD operation.
If string evaluates to the null value, null is returned. If length is less than 1, an empty
string is returned. If length is the null value, the FOLD function fails and the
program terminates with a run-time error message.
Examples
    PRINT FOLD("THIS IS A FOLDED STRING.",5)
This is the program output:
    THISFIS AFFOLDEFDFSTRINFG.
In the following example, the blanks are taken as substring delimiters, and as no
substring exceeds the specified length of six characters, the output would be:
    RED FM ORANGEFM YELLOWFM GREENFMBLUEFMINDIGOFMVIOLET
The field mark replaces the space in the string:
    A="RED ORANGE YELLOW GREEN BLUE INDIGO VIOLET"
    CALL !FOLD (RESULT,A,6)
    PRINT RESULT
FOLDDP
Syntax
    FOLDDP (string, length [ , mapname ] )
Description
In NLS mode, use the FOLDDP function to divide a string into a number of
substrings separated by field marks. The division is in display positions rather
than character lengths.
string is separated into substrings of display length less than or equal to length.
string is separated on blanks, if possible, otherwise it is separated into substrings
of the specified length.
If string evaluates to the null value, null is returned. If length is less than 1, an empty
string is returned. If length is the null value, the FOLDDP function fails and the
program terminates with a run-time error message.
If you execute FOLDDP when NLS is disabled, the behavior is the same as for
FOLD. For more information about display length, see the DataStage NLS Guide.
FOOTING
Syntax
    FOOTING [ON print.channel] footing
Description
Use the FOOTING statement to specify the text and format of the footing to print
at the bottom of each page of output.
The ON clause specifies the logical print channel to use for output. print.channel is
an expression that evaluates to a number from –1 through 255. If you do not use
the ON clause, logical print channel 0 is used, which prints to the user’s terminal
if PRINTER OFF is set (see the PRINTER statement). Logical print channel –1
prints the data on the screen, regardless of whether a PRINTER ON statement has
been executed.
footing is an expression that evaluates to the footing text and the control characters
that specify the footing’s format. You can use the following format control charac-
ters, enclosed in single quotation marks, in the footing expression:
See the following examples, in which the vertical bars represent the left and right
margins:
Specification                      Result
"Hello there"                      |Hello there                                     |
"'G'Hello there"                   |                                Hello there|
"'G'Hello there'G'"                |                 Hello there                    |
"Hello'G'there"                    |Hello                                   there|
"'G'Hello'G'there'G'"              |           Hello             there              |
The minimum gap size is 0 blanks. If a line is wider than the device width even
when all the gaps are 0, the line wraps, and all gaps remain 0.
If NLS is enabled, FOOTING calculates gaps using varying display positions
rather than character lengths. For more information about display length, see the
DataStage NLS Guide.
INFORMATION Flavor
immediately precedes a P option. For example, 'PPP' prints the page number right-
justified in a field of three blanks.
Note: In all other flavors, 'PPP' prints three identical page numbers, each in the
      default field of four.
Date Format. In an INFORMATION flavor account the default date format is mm-
dd-yy, and the default time format is 24-hour style.
In PICK, IN2, REALITY, and IDEAL flavor accounts, use the HEADER.DATE
option of the $OPTIONS statement to cause HEADING, FOOTING, and PAGE
statements to behave as they do in INFORMATION flavor accounts.
PIOPEN Flavor
Resetting the Page Number and the Date. The control character I (for initialize)
resets the page number to 1, and resets the date.
FOR WHILE…UNTIL
Syntax
      FOR variable = start TO end [STEP increment]
         [loop.statements]
         [CONTINUE | EXIT]
      [ {WHILE | UNTIL} expression]
         [loop.statements]
         [CONTINUE | EXIT]
      NEXT [variable]
FOR…NEXT
Description
Use the FOR statement to create a FOR…NEXT program loop. A program loop is
a series of statements that execute repeatedly until the specified number of repeti-
tions have been performed or until specified conditions are met.
variable is assigned the value of start, which is the initial value of the counter. end is
the end value of the counter.
The loop.statements that follow the FOR clause execute until the NEXT statement is
encountered. Then the counter is adjusted by the amount specified by the STEP
clause.
At this point a check is performed on the value of the counter. If it is less than or
equal to end, program execution branches back to the statement following the FOR
clause and the process repeats. If it is greater than end, execution continues with the
statement following the NEXT statement.
The WHILE condition specifies that as long as the WHILE expression evaluates to
true, the loop continues to execute. When the WHILE expression evaluates to false,
the loop ends, and program execution continues with the statement following the
NEXT statement. If a WHILE or UNTIL expression evaluates to the null value, the
condition is false.
The UNTIL condition specifies that the loop continues to execute only as long as
the UNTIL expression evaluates to false. When the UNTIL expression evaluates to
true, the loop ends and program execution continues with the statement following
the NEXT statement.
expression can also contain a conditional statement. As expression you can use any
statement that takes a THEN or an ELSE clause, but without the THEN or ELSE
clause. When the conditional statement would execute the ELSE clause, expression
evaluates to false; when the conditional statement would execute the THEN
clause, expression evaluates to true. The LOCKED clause is not supported in this
context.
You can use multiple WHILE and UNTIL clauses in a FOR…NEXT loop.
Use the CONTINUE statement within FOR…NEXT to transfer control to the next
iteration of the loop, from any point in the loop.
Use the EXIT statement within FOR…NEXT to terminate the loop from any point
within the loop.
If STEP is not specified, increment is assumed to be 1. If increment is negative, the
end value of the counter is less than the initial value. Each time the loop is
processed, the counter is decreased by the amount specified in the STEP clause.
Execution continues to loop until the counter is less than end.
The body of the loop is skipped if start is greater than end, and increment is not
negative. If start, end, or increment evaluates to the null value, the FOR statement
fails and the program terminates with a run-time error message.
Nested Loops
You can nest FOR…NEXT loops. That is, you can put a FOR…NEXT loop inside
another FOR…NEXT loop. When loops are nested, each loop must have a unique
variable name as its counter. The NEXT statement for the inside loop must appear
before the NEXT statement for the outside loop.
If you omit the variables in the NEXT statement, the NEXT statement corresponds
to the most recent FOR statement. If a NEXT statement is encountered without a
previous FOR statement, an error occurs during compilation.
INFORMATION Flavor
In an INFORMATION flavor account the FOR variable is checked to see if it
exceeds end before increment is added to it. That means that the value of the FOR
variable does not exceed end at the termination of the loop. In IDEAL, PICK, IN2,
and REALITY flavors the increment is made before the bound checking. In this
case it is possible for variable to exceed end. Use the FOR.INCR.BEF option of the
$OPTIONS statement to get IDEAL flavor behavior in an INFORMATION flavor
account.
Examples
In the following example, the loop is executed 100 times or until control is trans-
ferred by one of the statements in the loop:
    FOR VAR=1 TO 100
    NEXT VAR
Here are more examples of FOR…NEXT loops:
FORMLIST
Syntax
    FORMLIST [variable] [TO list.number] [ON ERROR statements]
Description
The FORMLIST statement is the same as the SELECT statement.
FSUB
Syntax
       FSUB (number1, number2)
       CALL !FSUB (result, number1, number2)
Description
Use the FSUB function to perform floating-point subtraction on two numeric
values. number2 is subtracted from number1. If either number evaluates to the null
value, null is returned. result equates to number1 minus number2.
This function is provided for compatibility with existing software. You can also use
the − operator to perform floating-point subtraction.
Example
       PRINT FSUB(.234,.567)
This is the program output:
       -0.333
FUNCTION
Syntax
    FUNCTION [name] [ ( [MAT] variable [ , [MAT] variable …] ) ]
Description
Use the FUNCTION statement to identify a user-written function and to specify
the number and names of the arguments to be passed to it. The FUNCTION state-
ment must be the first noncomment line in the user-written function. A user-
written function can contain only one FUNCTION statement.
name is specified for documentation purposes; it need not be the same as the func-
tion name or the name used to reference the function in the calling program. name
can be any valid variable name.
variable is an expression that passes values between the calling programs and the
function. variables are the formal parameters of the user-written function. When
actual parameters are specified as arguments to a user-written function, the actual
parameters are referenced by the formal parameters so that calculations performed
in the user-written function use the actual parameters.
Separate variables by commas. Up to 254 variables can be passed to a user-written
function. To pass an array, you must precede the array name with the keyword
MAT. When a user-written function is called, the calling function must specify the
same number of variables as are specified in the FUNCTION statement.
An extra variable is hidden so that the user-written function can use it to return a
value. An extra variable is retained by the user-written function so that a value is
returned by the RETURN (value) statement. This extra variable is reported by the
MAP and MAKE.MAP.FILE commands. If you use the RETURN statement in a
user-written function and you do not specify a value to return, an empty string is
returned by default.
The program that calls a user-written function must contain a DEFFUN statement
that defines the user-written function before it uses it. The user-written function
must be cataloged in either a local catalog or the system catalog, or it must be a
record in the same object file as the calling program.
If the user-defined function recursively calls itself within the function, a DEFFUN
statement must precede it in the user-written function.
Examples
The following user-defined function SHORT compares the length of two argu-
ments and returns the shorter:
   FUNCTION SHORT(A,B)
   AL = LEN(A)
   BL = LEN(B)
   IF AL < BL THEN RESULT = A ELSE RESULT = B
   RETURN(RESULT)
The following example defines a function called MYFUNC with the arguments or
formal parameters A, B, and C. It is followed by an example of the DEFFUN state-
ment declaring and using the MYFUNC function. The actual parameters held in X,
Y, and Z are referenced by the formal parameters A, B, and C so that the value
assigned to T can be calculated.
   FUNCTION MYFUNC(A, B, C)
   Z = ...
   RETURN (Z)
       .
       .
       .
   END
   DEFFUN MYFUNC(X, Y, Z)
   T = MYFUNC(X, Y, Z)
   END
GES
Syntax
      GES (array1, array2)
      CALL −GES (return.array, array1, array2)
      CALL !GES (return.array, array1, array2)
Description
Use the GES function to test if elements of one dynamic array are greater than or
equal to corresponding elements of another dynamic array.
Each element of array1 is compared with the corresponding element of array2. If the
element from array1 is greater than or equal to the element from array2, a 1 is
returned in the corresponding element of a new dynamic array. If the element from
array1 is less than the element from array2, a 0 is returned. If an element of one
dynamic array has no corresponding element in the other dynamic array, the unde-
fined element is evaluated as empty, and the comparison continues.
If either element of a corresponding pair is the null value, null is returned for that
element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
GET
Syntax
      GET[X] read.var[ ,length ] [ SETTING read.count ] FROM device
              [ UNTIL eop.char.list ] [ RETURNING last.char.read ]
              [ WAITING seconds ] [ THEN statements ] [ ELSE statements ]
Description
Use GET statements to read a block of data from an input stream associated with
a device, such as a serial line or terminal. The device must be opened with the
OPENDEV or OPENSEQ statement. Once the device has been opened, the GET
statements read data from the device. The GET statements do not perform any pre-
or postprocessing of the data stream, nor do they control local echo characteristics.
These aspects of terminal control are handled either by the application or by the
device driver. The behavior of certain devices can be managed through the
TTYSET/TTYGET interface.
Use the GETX statement to return the characters in ASCII hexadecimal format. For
example, the sequence of 8-bit character “abcde” is returned as the character string
“6162636465”. However, the value returned in the last.char.read variable is in stan-
dard ASCII character form.
read.var is the variable into which the characters read from device are stored. If no
data is read, read.var is set to the empty string.
length is the expression evaluating to the number of characters read from the data
stream; if length and timeout are not specified, the default length is 1. If length is not
specified, but an eop.char.list value is included, no length limit is imposed on the
input.
read.count is the variable that records the actual count of characters read and stored
in read.var. This may differ from length when a timeout condition occurs or when a
recognized end-of-packet character is detected.
device is a valid file variable resulting from a successful OPENDEV or OPENSEQ
statement. This is the handle to the I/O device that supplies the data stream for the
operation of the GET statements.
Terminating Conditions
GET statements read data from the device’s input stream until the first terminating
condition is encountered. The following table lists the possible terminating
conditions:
Condition                        Description
Requested read length has        The read is fully satisfied. read.var contains the
been satisfied                   characters read, and last.char.read contains an
                                 empty string. Program control passes to the
                                 THEN clause if present. The default requested
                                 read length is one character unless an end-of-
                                 packet value has been selected (in which case, no
                                 length limit is used).
Recognized end-of-packet         The read is terminated by a special application-
character has been               defined character in the data stream. The data
processed                        read to this point, excluding the end-of-packet
                                 character, is stored in read.var. The end-of-packet
                                 character is stored in last.char.read. Program
                                 control passes to the THEN clause if present. This
                                 terminating condition is only possible if the
                                 UNTIL clause has been specified. If there is no
                                 UNTIL clause, no end-of-packet characters are
                                 recognized.
Condition                      Description
Timeout limit has expired      The read could not be satisfied within the speci-
                               fied time limitation. If no characters have been
                               read, read.var and last.char.read are set to the
                               empty string, and read.count is set to 0. The
                               system status code is set to 0 and may be checked
                               with the STATUS function. Control passes to the
                               ELSE clause if present. This condition is only
                               possible if the WAITING clause is specified. In
                               the absence of a WAITING clause, the applica-
                               tion waits until one of the other terminating
                               conditions is met.
Device input error             An unrecoverable error occurred on the device.
                               Unrecoverable errors can include EOF conditions
                               and operating system reported I/O errors. In this
                               case, the data read to this point is stored in
                               read.var, and the empty string is stored in
                               last.char.read. If no characters have been read,
                               read.var and last.char.read are set to the empty
                               string, and read.count is set to 0. The system
                               status code is set to a nonzero value and may
                               checked with the STATUS function. Control
                               passes to the ELSE clause if present.
Note: Under all termination conditions, read.count is set to the number of charac-
      ters read from the input data stream.
    • If the WAITING clause is used, the GET statement behaves normally, and
      the ELSE clause is executed only if the number of seconds for timeout has
      elapsed. If the input terminates for any other reason, it executes the THEN
      clause.
    • If the WAITING clause is not used and there is a finite number of characters
      to expect from the input, then only the type-ahead buffer is examined for
      input. If the type-ahead buffer contains the expected number of characters,
      it executes the THEN clause; otherwise it executes the ELSE clause. If the
      type-ahead feature is turned off, the ELSE clause is always executed.
    • In a special case, the ELSE clause is executed if the line has not been
      attached before executing the GET statement.
In summary, unless the WAITING clause is used, specifying the THEN and ELSE
clauses causes the GET statement to behave like an INPUTIF…FROM statement.
The exception to this is the UNTIL clause without a maximum length specified, in
which case the GET statement behaves normally and the ELSE clause is never
used.
Example
The following code fragment shows how the GET statement reads a number of
data buffers representing a transaction message from a device:
    DIM SAVEBUFFER(10)
    SAVELIMIT = 10
    OPENDEV "TTY10" TO TTYLINE ELSE STOP "CANNOT OPEN TTY10"
    I = 1
    LOOP
           GET BUFFER,128 FROM TTYLINE UNTIL CHAR(10) WAITING 10
           ELSE
               IF STATUS()
               THEN PRINT "UNRECOVERABLE ERROR DETECTED ON DEVICE,
    "IM SAVEBUFFER(10)
    SAVELIMIT = 10
    OPENDEV "TTY10" TO TTYLINE ELSE STOP "CANNOT OPEN TTY10"
    I = 1
    LOOP
           GET BUFFER,128 FROM TTYLINE UNTIL CHAR(10)
    WAITING 10
           ELSE
               IF STATUS()
GETX
Use the GETX statement to read a block of data from an input stream and return
the characters in ASCII hexadecimal format. For details, see the GET statements.
GET(ARG.)
Syntax
    GET(ARG. [ ,arg#] ) variable [THEN statements] [ELSE statements]
Description
Use the GET(ARG.) statement to retrieve the next command line argument. The
command line is delimited by blanks, and the first argument is assumed to be the
first word after the program name. When a cataloged program is invoked, the
argument list starts with the second word in the command line.
Blanks in quoted strings are not treated as delimiters and the string is treated as a
single argument. For example, "54 76" returns 54 76.
arg# specifies the command line argument to retrieve. It must evaluate to a number.
If arg# is not specified, the next command line argument is retrieved. The retrieved
argument is assigned to variable.
THEN and ELSE statements are both optional. The THEN clause is executed if the
argument is found. The ELSE clause is executed if the argument is not found. If the
argument is not found and no ELSE clause is present, variable is set to an empty
string.
If no arg# is specified or if arg# evaluates to 0, the argument to the right of the last
argument retrieved is assigned to variable. The GET statement fails if arg# evaluates
to a number greater than the number of command line arguments or if the last
argument has been assigned and a GET with no arg# is used. To move to the begin-
ning of the argument list, set arg# to 1.
If arg# evaluates to the null value, the GET statement fails and the program termi-
nates with a run-time error message.
Example
In the following example, the command is:
    RUN BP PROG ARG1 ARG2 ARG3
and the program is:
    A=5;B=2
    GET(ARG.)FIRST
    GET(ARG.,B)SECOND
    GET(ARG.)THIRD
    GET(ARG.,1)FOURTH
    GET(ARG.,A-B)FIFTH
    PRINT FIRST
    PRINT   SECOND
    PRINT   THIRD
    PRINT   FOURTH
    PRINT   FIFTH
This is the program output:
    ARG1
    ARG2
    ARG3
    ARG1
    ARG3
If the command line is changed to RUN PROG, the system looks in the file PROG
for the program with the name of the first argument. If PROG is a cataloged
program, the command line would have to be changed to PROG ARG1 ARG2
ARG3 to get the same results.
GETLIST
Syntax
      GETLIST listname [TO list.number] [SETTING variable]
              {THEN statements [ELSE statements] | ELSE statements}
Description
Use the GETLIST statement to activate a saved select list so that a READNEXT
statement can use it.
listname is an expression that evaluates to the form:
      record.ID
or:
      record.ID account.name
record.ID is the record ID of a select list in the &SAVEDLISTS& file. If account.name
is specified, the &SAVEDLISTS& file of that account is used instead of the one in
the local account.
If listname evaluates to the null value, the GETLIST statement fails and the program
terminates with a run-time error message.
The TO clause puts the list in a select list numbered 0 through 10. If list.number is
not specified, the list is saved as select list 0.
The SETTING clause assigns the count of the elements in the list to variable. The
system variable @SELECTED is also assigned this count whether or not the
SETTING clause is used. If the list is retrieved successfully, even if the list is empty,
the THEN statements execute; if not, the ELSE statements execute.
GETLOCALE
Syntax
    GETLOCALE (category)
Description
In NLS mode use the GETLOCALE function to return the names of specified cate-
gories of the current locale. The GETLOCALE function also returns the details of
any saved locale that differs from the current one.
category is one of the following tokens that are defined in the DataStage include file
UVNLSLOC.H:
If the GETLOCALE function fails, it returns one of the following error tokens:
For more information about locales, see the DataStage NLS Guide.
GETREM
Syntax
    GETREM (dynamic.array)
Description
Use the GETREM function after the execution of a REMOVE Statement statement,
a REMOVE function, or a REVREMOVE statement, to return the numeric value for
the character position of the pointer associated with dynamic.array.
dynamic.array evaluates to the name of a variable containing a dynamic array.
The returned value is an integer. The integer returned is one-based, not zero-based.
If no REMOVE statements have been executed on dynamic.array, 1 is returned. At
the end of dynamic.array, GETREM returns the length of dynamic array plus 1. The
offset returned by GETREM indicates the first character of the next dynamic array
element to be removed.
Example
    DYN = "THIS":@FM:"HERE":@FM:"STRING"
    REMOVE VAR FROM DYN SETTING X
    PRINT GETREM(DYN)
This is the program output:
    5
GOSUB
Syntax
    GOSUB statement.label [ : ]
GO SUB statement.label [ : ]
Description
Use the GOSUB statement to transfer program control to an internal subroutine
referenced by statement.label. A colon ( : ) is optional in GOSUB statements, even
though it is required after nonnumeric statement labels at the beginning of
program lines.
Use the RETURN statement at the end of the internal subroutine referenced by the
GOSUB statement, to transfer program control to the statement following the
GOSUB statement.
Use the RETURN TO statement at the end of an internal subroutine to transfer
control to a location in the program other than the line following the GOSUB state-
ment. In this case, use statement.label to refer to the target location.
Be careful with the RETURN TO statement, because all other GOSUBs or CALLs
active when the GOSUB is executed remain active, and errors can result.
A program can call a subroutine any number of times. A subroutine can also be
called from within another subroutine; this process is called nesting subroutines.
You can nest up to 256 GOSUB calls.
Subroutines can appear anywhere in the program but should be readily distin-
guishable from the main program. To prevent inadvertent entry into the
subroutine, precede it with a STOP, END, or GOTO statement that directs program
control around the subroutine.
Example
    VAR='ABKL1234'
    FOR X=1 TO LEN(VAR)
       Y=VAR[X,1]
       GOSUB 100
    NEXT X
    STOP
    100*
    IF Y MATCHES '1N' THEN RETURN TO 200
GOTO
Syntax
    GO[TO] statement.label [ : ]
GO TO statement.label [ : ]
Description
Use the GOTO statement to transfer program control to the statement specified by
statement.label. A colon ( : ) is optional in GOTO statements.
If the statement referenced is an executable statement, that statement and those
that follow are executed. If it is a nonexecutable statement, execution proceeds at
the first executable statement encountered after the referenced statement.
Example
    X=80
    GOTO 10
    STOP
    *
    10*
    IF X>20 THEN GO 20 ELSE STOP
    *
    20*
    PRINT 'AT LABEL 20'
    GO TO CALCULATE:
    STOP
    *
    CALCULATE:
    PRINT 'AT LABEL CALCULATE'
This is the program output:
    AT LABEL 20
    AT LABEL CALCULATE
GROUP
Syntax
    GROUP (string, delimiter, occurrence [ ,num.substr] )
Description
Use the GROUP function to return one or more substrings located between speci-
fied delimiters in string.
delimiter evaluates to any character, including field mark, value mark, and
subvalue marks. It delimits the start and end of the substring. If delimiter evaluates
to more than one character, only the first character is used. Delimiters are not
returned with the substring.
occurrence specifies which occurrence of the delimiter is to be used as a terminator.
If occurrence is less than 1, 1 is assumed.
num.substr specifies the number of delimited substrings to return. If the value of
num.substr is an empty string or less than 1, 1 is assumed. When more than one
substring is returned, delimiters are returned along with the successive substrings.
If either delimiter or occurrence is not in the string, an empty string is returned,
unless occurrence specifies 1. If occurrence is 1 and delimiter is not found, the entire
string is returned. If delimiter is an empty string, the entire string is returned.
If string evaluates to the null value, null is returned. If string contains CHAR(128)
(that is, @NULL.STR), it is treated like any other character in a string. If delimiter,
occurrence, or num.substr evaluates to the null value, the GROUP function fails and
the program terminates with a run-time error message.
The GROUP function works identically to the FIELD function.
Examples
    D=GROUP("###DHHH#KK","#",4)
    PRINT "D= ",D
The variable D is set to DHHH because the data between the third and fourth
occurrence of the delimiter # is DHHH.
    REC="ACADABA"
    E=GROUP(REC,"A",2)
    PRINT "E= ",E
GROUPSTORE
Syntax
    GROUPSTORE new.string IN string USING start, n [ ,delimiter]
Description
Use the GROUPSTORE statement to modify character strings by inserting,
deleting, or replacing fields separated by specified delimiters.
new.string is an expression that evaluates to the character string to be inserted in
string.
string is an expression that evaluates to the character string to be modified.
delimiter evaluates to any single ASCII character, including field, value, and
subvalue marks. If you do not specify delimiter, the field mark is used.
start evaluates to a number specifying the starting field position. Modification
begins at the field specified by start. If start is greater than the number of fields in
string, the required number of empty fields is generated before the GROUPSTORE
statement is executed.
n specifies the number of fields of new.string to insert in string. n determines how
the GROUPSTORE operation is executed. If n is positive, n fields in string are
replaced with the first n fields of new.string. If n is negative, n fields in string are
replaced with all the fields in new.string. If n is 0, all the fields in new.string are
inserted in string before the field specified by start.
If string evaluates to the null value, null is returned. If new.string, start, n, or delim-
iter is null, the GROUPSTORE statement fails and the program terminates with a
run-time error message.
Example
    Q='1#2#3#4#5'
    GROUPSTORE "A#B" IN Q USING 2,2,"#"
    PRINT "TEST1= ",Q
    *
    Q='1#2#3#4#5'
    GROUPSTORE "A#B" IN Q USING 2,-2,"#"
    PRINT "TEST2= ",Q
    *
    Q='1#2#3#4#5'
GTS
Syntax
      GTS (array1, array2)
      CALL −GTS (return.array, array1, array2)
      CALL !GTS (return.array, array1, array2)
Description
Use the GTS function to test if elements of one dynamic array are greater than
elements of another dynamic array.
Each element of array1 is compared with the corresponding element of array2. If the
element from array1 is greater than the element from array2, a 1 is returned in the
corresponding element of a new dynamic array. If the element from array1 is less
than or equal to the element from array2, a 0 is returned. If an element of one
dynamic array has no corresponding element in the other dynamic array, the unde-
fined element is evaluated as an empty string, and the comparison continues.
If either of a corresponding pair of elements is the null value, null is returned for
that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
HEADING
Syntax
    HEADING [ON print.channel] heading
Description
Use the HEADING statement to specify the text and format of the heading to print
at the top of each page of output.
The ON clause specifies the logical print channel to use for output. print.channel is
an expression that evaluates to a number from –1 through 255. If you do not use
the ON clause, logical print channel 0 is used, which prints to the user’s terminal
if PRINTER OFF is set (see the PRINTER statement). Logical print channel –1
prints the data on the screen, regardless of whether a PRINTER ON statement has
been executed.
heading is an expression that evaluates to the heading text and the control charac-
ters that specify the heading’s format. You can use the following format control
characters, enclosed in single quotation marks, in the heading expression:
]           Starts a new line. Do not put the right bracket inside single quotation
            marks.
N           Suppresses automatic paging.
P[n]        Prints current page number right-justified in a field of n blanks. The
            default value for n is 4.
S           Left-justified, inserted page number.
^           Prints current page number right-justified in a field of n blanks. The
            default value for n is 4. Do not put the caret inside single quotation
            marks.
Two single quotation marks ( ' ' ) print one single quotation mark in heading text.
When the program is executed, the format control characters produce the specified
results. You can specify multiple options in a single set of quotation marks.
If either print.channel or heading evaluates to the null value, the HEADING state-
ment fails and the program terminates with a run-time error message.
Pagination begins with page 1 and increments automatically on generation of each
new page or upon encountering the PAGE statement.
Output to a terminal or printer is paged automatically. Use the N option in either
a HEADING or a FOOTING statement to turn off automatic paging.
once in any heading or footing, but it must appear before any occurrence of the
characters ] ^ and \.
Specification                      Result
"Hello there"                      |Hello there                                     |
"'G'Hello there"                   |                                Hello there|
"'G'Hello there'G'"                |                 Hello there                    |
"Hello'G'there"                    |Hello                                   there|
"'G'Hello'G'there'G'"              |           Hello              there             |
The minimum gap size is 0 blanks. If a line is wider than the device width even
when all the gaps are 0, the line wraps, and all gaps remain 0.
If NLS is enabled, HEADING calculates gaps using varying display positions
rather than character lengths. For more information about display length, see the
DataStage NLS Guide.
INFORMATION Flavor
Note: In all other flavors, 'PPP' prints three identical page numbers, each in the
      default field of four.
Date Format. In an INFORMATION flavor account the default date format is mm-
dd-yy, and the default time format is 24-hour style.
In PICK, IN2, REALITY, and IDEAL flavor accounts, use the HEADER.DATE
option of the $OPTIONS statement to cause HEADING, FOOTING, and PAGE
statements to behave as they do in INFORMATION flavor accounts.
PIOPEN Flavor
Resetting the Page Number and the Date. The control character I (for initialize)
resets the page number to 1, and resets the date.
Example
    HEADING "'C' LIST PRINTED: 'D'"
    FOR N=1 TO 10
       PRINT "THIS IS ANOTHER LINE"
    NEXT
This is the program output:
                       LIST PRINTED:   04 Jun 1994
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
    THIS   IS   ANOTHER LINE
HUSH
Syntax
    HUSH { ON | OFF | expression}      [ SETTING status ]
Description
Use the HUSH statement to suppress the display of all output normally sent to a
terminal during processing. HUSH also suppresses output to a COMO file or
TANDEM display.
SETTING status sets the value of a variable to the value of the HUSH state before
the HUSH statement was executed. It can be used instead of the STATUS function
to save the state so that it can be restored later. STATUS has a value of 1 if the
previous state was HUSH ON or a value of 0 if the previous state was HUSH OFF.
You might use this statement when you are transmitting information over phone
lines or when you are sending data to a hard-copy terminal. Both these situations
result in slower transmission speeds. The unnecessary data display makes the task
even slower.
HUSH acts as a toggle. If it is used without a qualifier, it changes the current state.
Do not use this statement to shut off output display unless you are sure the display
is unnecessary. When you use HUSH ON, all output is suppressed including error
messages and requests for information.
Example
In the following example, terminal output is disabled with the HUSH statement
and the previous state was saved in the variable USER.HUSH.STATE.
After executing some other statements, the program returns the user’s process to
the same HUSH state as it was in previous to the execution of the first HUSH
statement:
    HUSH ON
    USER.HUSH.STATE = STATUS()
    ...
    HUSH USER.HUSH.STATE
The example could have been written as follows:
    HUSH ON SETTING USER.HUSH.STATE
       .
       .
       .
    HUSH USER.HUSH.STATE
ICHECK
Syntax
    ICHECK (dynamic.array [ ,file.variable ] ,key [ ,column# ] )
Description
Use the ICHECK function to check if data you intend to write to an SQL table
violates any SQL integrity constraints. ICHECK verifies that specified data and
primary keys satisfy the defined SQL integrity constraints for an SQL table.
dynamic.array is an expression that evaluates to the data you want to check against
any integrity constraints.
file.variable specifies an open file. If file.variable is not specified, the default file vari-
able is assumed (for more information on default files, see the OPEN statement).
key is an expression that evaluates to the primary key you want to check against
any integrity constraints.
column# is an expression that evaluates to the number of the column in the table
whose data is to be checked. If you do not specify column#, all columns in the file
are checked. Column 0 specifies the primary key (record ID).
If dynamic.array, file.variable, key, or column# evaluates to the null value, the ICHECK
function fails and the program terminates with a run-time error message.
You might use the ICHECK function to limit the amount of integrity checking that
is done and thus improve performance. If you do this, however, you are assuming
responsibility for data integrity. For example, you might want to use ICHECK with
a program that changes only a few columns in a file. To do this, turn off the
OPENCHK configurable parameter, open the file with the OPEN statement rather
than the OPENCHECK statement, and use the ICHECK function before you write
the updated record to verify, for each column you are updating, that you are not
violating the table’s integrity checks.
If the ON UPDATE clause of a referential constraint specifies an action, ICHECK
always validates data being written to the referenced table; it does not check the
referencing table. Therefore, ICHECK can succeed, but when the actual write is
done, it can have a constraint failure while attempting to update the referencing
table. If the referential constraint does not have an ON UPDATE clause, or if these
clauses specify NO ACTION, the referencing table is checked to ensure that no row
in it contains the old value of the referenced column.
ICHECK does not check triggers when it checks other SQL integrity constraints.
Therefore, a write that fires a trigger can fail even if the ICHECK succeeds.
ICHECK returns a dynamic array of three elements separated by field marks:
    error.codeFcolumn#Fconstraint
error.code   A code that indicates the type of failure. Error codes can be any of the
             following:
             0   No failure
             1   SINGLEVALUED failure
             2   NOT NULL failure
             3   NOT EMPTY failure
             4   ROWUNIQUE failure (including single-column association
                 KEY)
             5   UNIQUE (column constraint) failure
             6   UNIQUE (table constraint) failure
             7   Association KEY ROWUNIQUE failure when association has
                 multiple KEY fields.
             8   CHECK constraint failure
             9   Primary key has too many parts
             10 Referential constraint failure
             11 Referential constraint failure that occurs when a numeric column
                references a nonnumeric column in the referenced table.
column#      The number of the column where the failure occurred. If any part of
             a primary key fails, 0 is returned. If the violation involves more than
             one column, -1 is returned.
constraint   This element is returned only when error.code is 7 or 8. For code 7, the
             association name is returned. For code 8, the name of the CHECK
             constraint is returned if it has a name; otherwise, the CHECK
             constraint itself is returned.
If the record violates more than one integrity constraint, ICHECK returns a
dynamic array only for the first constraint that causes a failure.
The ICHECK function is advisory only. That is, if two programs try to write the
same data to the same column defined as UNIQUE (see error 5), an ICHECK in the
first program may pass. If the second program writes data to the file before the first
program writes its ICHECKed data, the first program’s write fails even though the
ICHECK did not fail.
ICONV
Syntax
      ICONV (string, conversion)
Description
Use the ICONV function to convert string to a specified internal storage format.
string is an expression that evaluates to the string to be converted.
conversion is an expression that evaluates to one or more valid conversion codes,
separated by value marks (ASCII 253).
string is converted to the internal format specified by conversion. If multiple codes
are used, they are applied from left to right. The first conversion code converts the
value of string. The second conversion code converts the output of the first conver-
sion, and so on.
If string evaluates to the null value, null is returned. If conversion evaluates to the
null value, the ICONV function fails and the program terminates with a run-time
error message.
The STATUS function reflects the result of the conversion:
For information about converting strings to an external format, see the OCONV
function.
Examples
The following are examples of date conversions:
DATE=ICONV("30/9/67","DE") -92
DATE=ICONV("19850625","D") 6386
DATE=ICONV("85161","D") 6371
OCT=ICONV("3001","MO") 1537
BIN=ICONV(1111,"MB") 15
 X=563.888                                -564
 X=ICONV(X,"MD0")
X=ICONV(1988.28,"MD24") 19882800
ICONVS
Syntax
    ICONVS (dynamic.array, conversion)
    CALL −ICONVS (return.array, dynamic.array, conversion)
    CALL !ICONVS (return.array, dynamic.array, conversion)
Description
Use the ICONVS function to convert each element of dynamic.array to a specified
internal storage format.
conversion is an expression that evaluates to one or more valid conversion codes,
separated by value marks (ASCII 253).
Each element of dynamic.array is converted to the internal format specified by
conversion and is returned in a dynamic array. If multiple codes are used, they are
applied from left to right. The first conversion code converts the value of each
element of dynamic.array. The second conversion code converts the value of each
element of the output of the first conversion, and so on.
If dynamic.array evaluates to the null value, null is returned. If an element of
dynamic.array is the null value, null is returned for that element. If conversion eval-
uates to the null value, the ICONV function fails and the program terminates with
a run-time error message.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
The STATUS function reflects the result of the conversion:
IF
Syntax
     IF expression {THEN statements [ELSE statements] | ELSE statements}
     IF expression
     {THEN statements
     [ELSE statements] |
     ELSE statements}
     IF expression {THEN
          statements
     END [ELSE
          statements
     END] | ELSE
          statements
     END}
     IF expression
     {THEN
          statements
     END
     [ELSE
          statements
     END] |
     ELSE
          statements
     END }
Description
Use the IF statement to determine program flow based on the evaluation of expres-
sion. If the value of expression is true, the THEN statements are executed. If the
value of expression is false, the THEN statements are ignored and the ELSE state-
ments are executed. If expression is the null value, expression evaluates to false. If no
ELSE statements are present, program execution continues with the next execut-
able statement.
The IF statement must contain either a THEN clause or an ELSE clause. It need not
include both.
Use the ISNULL function with the IF statement when you want to test whether the
value of a variable is the null value. This is the only way to test for the null value
since null cannot be equal to any value, including itself. The syntax is:
    IF ISNULL (expression) …
You can write IF…THEN statements on a single line or separated onto several
lines. Separating statements onto several lines can improve readability. Either way,
the statements are executed identically.
You can nest IF…THEN statements. If the THEN or ELSE statements are written
on more than one line, you must use an END statement as the last statement of the
THEN or ELSE statements.
Conditional Compilation
You can specify the conditions under which all or part of a BASIC program is to be
compiled, using a modified version of the IF statement. The syntax of the condi-
tional compilation statement is the same as that of the IF statement except for the
test expression, which must be one of the following: $TRUE, $T, $FALSE, or $F.
Example
    X=10
    IF X>5 THEN PRINT 'X IS GREATER THAN 5';Y=3
    *
    IF Y>5 THEN STOP ELSE Z=9; PRINT 'Y IS LESS THAN 5'
    *
    IF Z=9 THEN PRINT 'Z EQUALS 9'
    ELSE PRINT 'Z DOES NOT EQUAL 9' ; STOP
    *
    IF Z=9 THEN
        GOTO 10
    END ELSE
        STOP
    END
    *
    10*
    IF Y>4
        THEN
           PRINT 'Y GREATER THAN 4'
         END
         ELSE
             PRINT 'Y IS LESS THAN 4'
         END
This is the program output:
    X   IS GREATER THAN 5
    Y   IS LESS THAN 5
    Z   EQUALS 9
    Y   IS LESS THAN 4
IFS
Syntax
      IFS (dynamic.array, true.array, false.array)
      CALL −IFS (return.array, dynamic.array, true.array, false.array)
      CALL !IFS (return.array, dynamic.array, true.array, false.array)
Description
Use the IFS function to return a dynamic array whose elements are chosen individ-
ually from one of two dynamic arrays based on the contents of a third dynamic
array.
Each element of dynamic.array is evaluated. If the element evaluates to true, the
corresponding element from true.array is returned to the same element of a new
dynamic array. If the element evaluates to false, the corresponding element from
false.array is returned. If there is no corresponding element in the correct response
array, an empty string is returned for that element. If an element is the null value,
that element evaluates to false.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
ILPROMPT
Syntax
     ILPROMPT (in.line.prompt)
Description
Use the ILPROMPT function to evaluate a string containing in-line prompts.
in.line.prompt is an expression that evaluates to a string containing in-line prompts.
In-line prompts have the following syntax:
Sn           Takes the nth word from the command but uses the most recent
             command entered at the command prompt to execute the paragraph,
             rather than an argument in the paragraph. Use this option in nested
             paragraphs.
@(CLR)       Clears the screen.
@(BELL)      Rings the terminal bell.
@(TOF)       Positions the prompt at the top left of the screen.
@(col, row) Prompts at this column and row number on the terminal.
text is the prompt text to display. If you want to include quotation marks (single or
double) or backslashes as delimiters within the prompt text, you must enclose the
entire text string in a set of delimiters different from the delimiters you are using
within the text string. For example, to print the following prompt text:
     'P'RINTER OR 'T'ERMINAL
you must specify the prompt text as
     \'P'RINTER OR 'T'ERMINAL\
or
     "'P'RINTER OR 'T'ERMINAL"
option can be any valid ICONV conversion or matching pattern (see the MATCH
operator). A conversion must be in parentheses.
If in.line.prompt evaluates to the null value, the ILPROMPT function fails and the
program terminates with a run-time error.
If the in-line prompt has a value, that value is substituted for the prompt. If the in-
line prompt does not have a value, the prompt is displayed to request an input
value when the sentence is executed. The value entered at the prompt is then
substituted for the in-line prompt.
Once a value has been entered for a particular prompt, the prompt will continue to
have that value until a CLEARPROMPTS statement is executed, unless the control
option A is specified. CLEARPROMPTS clears all values entered for in-line
prompts.
You can enclose prompts within prompts.
Example
    A="This is your number. - <<number>>"
    PRINT ILPROMPT(A)
    PRINT ILPROMPT("Your number is <<number>>, and your letter is
       <<letter>>.")
This is the program output:
    number=5
    This is your number. - 5
    letter=K
    Your number is 5, and your letter is K.
INCLUDE
Syntax
    INCLUDE [filename] program
    INCLUDE program FROM filename
Description
Use the INCLUDE statement to direct the compiler to insert the source code in the
record program and compile it along with the main program. The INCLUDE state-
ment differs from the $CHAIN statement in that the compiler returns to the main
program and continues compiling with the statement following the INCLUDE
statement.
When program is specified without filename, program must be a record in the same
file as the program currently containing the INCLUDE statement.
If program is a record in a different file, the name of the file in which it is located
must be specified in the INCLUDE statement, followed by the name of the
program. The filename must specify a type 1 or type 19 file defined in the VOC file.
You can nest INCLUDE statements.
The INCLUDE statement is a synonym for the $INCLUDE and #INCLUDE
statements.
Example
    PRINT "START"
    INCLUDE END
    PRINT "FINISH"
When this program is compiled, the INCLUDE statement inserts code from the
program END (see the example on the END statement page). This is the program
output:
    START
    THESE TWO LINES WILL PRINT ONLY
    WHEN THE VALUE OF 'A' IS 'YES'.
INDEX
Syntax
    INDEX (string, substring, occurrence)
Description
Use the INDEX function to return the starting character position for the specified
occurrence of substring in string.
string is an expression that evaluates to any valid string. string is examined for the
substring expression.
occurrence specifies which occurrence of substring is to be located.
When substring is found and if it meets the occurrence criterion, the starting char-
acter position of the substring is returned. If substring is an empty string, 1 is
returned. If the specified occurrence of the substring is not found, or if string or
substring evaluate to the null value, 0 is returned.
If occurrence evaluates to the null value, the INDEX function fails and the program
terminates with a run-time error message.
Example
    Q='AAA11122ABB1619MM'
    P=INDEX(Q,1,4)
    PRINT "P= ",P
    *
    X='XX'
    Y=2
    Q='P1234XXOO1299XX00P'
    TEST=INDEX(Q,X,Y)
    PRINT "TEST= ",TEST
    *
    Q=INDEX("1234",'A',1)
    PRINT "Q= ",Q
INDEXS
Syntax
    INDEXS (dynamic.array, substring, occurrence)
    CALL −INDEXS (return.array, dynamic.array, substring, occurrence)
    CALL !INDEXS (return.array, dynamic.array, substring, occurrence)
Description
Use the INDEXS function to return a dynamic array of the starting column posi-
tions for a specified occurrence of a substring in each element of dynamic.array.
Each element is examined for substring.
occurrence specifies which occurrence of substring is to be located.
When substring is found, and if it meets the occurrence criterion, the starting
column position of the substring is returned. If substring is an empty string, 1 is
returned. If the specified occurrence of substring cannot be found, 0 is returned.
If dynamic.array evaluates to the null value, 0 is returned. If any element of
dynamic.array is null, 0 is returned for that element. If occurrence is the null value,
the INDEXS function fails and the program terminates with a run-time error
message.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
INDICES
Syntax
    INDICES (file.variable [ ,indexname])
Description
Use the INDICES function to return information about the secondary key indexes
in a file.
file.variable specifies an open file.
indexname is the name of a secondary index in the specified file.
If only file.variable is specified, a dynamic array is returned that contains the index
names of all secondary indexes for the file. The index names are separated by field
marks. If file.variable has no indexes, an empty string is returned.
If indexname is specified, information is returned in a dynamic array for indexname.
Field 1 of the dynamic array contains the following information:
If Value 1 of Field 1 is D, A, or S, Field 2 contains the field location (that is, the field
number), and Field 6 contains either S (single-valued field) or M (multivalued
field).
If Value 1 of Field 1 is I or SQL, the other fields of the dynamic array contain the
following information, derived from the I-descriptor in the file dictionary:
If Value 1 of Field 1 is C, the other fields of the dynamic array contain the following
information, derived from the A- or S-descriptor in the file dictionary:
If either file.variable or indexname is the null value, the INDICES function fails and
the program terminates with a run-time error message.
Any file updates executed in a transaction (that is, between a BEGIN TRANSAC-
TION statement and a COMMIT statement) are not accessible to the INDICES
function until after the COMMIT statement has been executed.
If NLS is enabled, the INDICES function reports the name of the current Collate
convention (as specified in the NLS.LC.COLLATE file) in force when the index was
created. See Value 17 in Field 1 for the name of the Collate convention of the index.
For more information about the collate convention, see the DataStage NLS Guide.
INMAT
Syntax
    INMAT ( [array] )
Description
Use the INMAT function to return the number of array elements that have been
loaded after the execution of a MATREAD, MATREADL, MATREADU, or
MATPARSE statement, or to return the modulo of a file after the execution of an
OPEN statement. You can also use the INMAT function after a DIM statement to
determine whether the DIM statement failed due to lack of available memory. If a
preceding DIM statement fails, INMAT returns a value of 1.
If the matrix assignment exceeds the number of elements specified in its dimen-
sioning statement, the zero element is loaded by the MATREAD, MATREADL,
MATREADU, or MATPARSE statement. If the array dimensioning is too small and
the zero element has been loaded, the INMAT function returns a value of 0.
If array is specified, the INMAT function returns the current dimensions of the
array. If array is the null value, the INMAT function fails and the program termi-
nates with a run-time error message.
Example
    DIM X(6)
    D='123456'
    MATPARSE X FROM D,''
    Y=INMAT()
    PRINT 'Y= ':Y
    *
    DIM X(5)
    A='CBDGFH'
    MATPARSE X FROM A,''
    C=INMAT()
    PRINT 'C= ':C
    *
    OPEN '','VOC' TO FILE ELSE STOP
    T=INMAT()
    PRINT 'T= ':T
INPUT
Syntax
    INPUT variable [ ,length] [ : ] [ _ ]
Description
Use the INPUT statement to halt program execution and prompt the user to enter
a response. Data entered at the terminal or supplied by a DATA statement in
response to an INPUT statement is assigned to variable. Input supplied by a DATA
statement is echoed to the terminal. If the response is a RETURN with no preceding
data, an empty string is assigned to variable.
The INPUT statement has two syntaxes. The first syntax displays a prompt and
assigns the input to variable. The second syntax specifies the location of the input
field on the screen and lets you display the current value of variable. Both the
current value and the displayed input can be formatted.
Use the INPUTIF statement to assign the contents of the type-ahead buffer to a
variable. If the type-ahead buffer is empty, the ELSE statements are executed,
otherwise any THEN statements are executed.
Use the @ expression to specify the position of the input field. The prompt is
displayed one character to the left of the beginning of the field, and the current
value of variable is displayed as the value in the input field. The user can edit the
displayed value or enter a new value. If the first character typed in response to the
prompt is an editing key, the user can edit the contents of the field. If the first char-
acter typed is anything else, the field’s contents are deleted and the user can enter
a new value. Editing keys are defined in the terminfo files; they can also be defined
by the KEYEDIT statement. Calculations are based on display length rather than
character length.
col and row are expressions that specify the column and row positions of the input
prompt. The prompt is positioned one character to the left of the input field.
Because the prompt character is positioned to the left of the col position, you must
set the prompt to the empty string if you want to use column 0. Otherwise, the
screen is erased before the prompt appears.
length specifies the maximum number of characters allowed as input. When the
maximum number of characters is entered, input is terminated. If the @ expression
is used, the newline is suppressed.
If length evaluates to less than 0 (for example, −1), the input buffer is tested for the
presence of characters. If characters are present, variable is set to 1, otherwise it is
set to 0. No input is performed.
If you use the underscore ( _ ) with the length expression, the user must enter the
RETURN manually at the terminal when input is complete. Only the specified
number of characters is accepted.
Use a format expression to validate input against a format mask and to format the
displayed input field. The syntax of the format expression is the same as that for
the FMT function. If you specify a length expression together with a format expres-
sion, length checking is performed. If input does not conform to the format mask,
an error message appears at the bottom of the screen, prompting the user for the
correct input.
The colon ( : ) suppresses the newline after input is terminated. This allows
multiple input prompts on a single line.
The default prompt character is a question mark. Use the PROMPT statement to
reassign the prompt character.
The INPUT statement prints only the prompt character on the screen. To print a
variable name or prompt text along with the prompt, precede the INPUT statement
with a PRINT statement.
The INPUT statement lets the user type ahead when entering a response. Users
familiar with a sequence of prompts can save time by entering data at their own
speed, not waiting for all prompts to be displayed. Responses to a sequence of
INPUT prompts are accepted in the order in which they are entered.
If col, row, length, or format evaluate to the null value, the INPUT statement fails and
the program terminates with a run-time error message. If variable is the null value
and the user types the TRAP key, null is retained as the value of variable.
If NLS is enabled, INPUT @ displays the initial value of an external multibyte char-
acter set through the mask as best as possible. If the user enters a new value, mask
disappears, and an input field of the approximate length (not including any
inserted characters) is entered. For details about format and mask, see the
FMTDPfunction.
Only backspace and kill are supported for editing functions when using a format
mask with input. When the user finishes the input, the new value is redisplayed
through the mask in the same way as the original value. For more information
about NLS in BASIC programs, see the DataStage NLS Guide.
PICK Flavor
In a PICK flavor account, the syntax of the INPUT and INPUT @ statements
includes THEN and ELSE clauses:
Examples
In the following examples of program output, bold type indicates words the user
types. In the first example the value entered is assigned to the variable NAME:
In the next example the value entered is assigned to the variable CODE. Only the
first seven characters are recognized. A RETURN and a LINEFEED automatically
occur.
In the next example the user can enter more than two characters. The program
waits for a RETURN to end input, but only the first two characters are assigned to
the variable YES.
In the next example the colon inhibits the automatic LINEFEED after the RETURN:
In the next example the input buffer is tested for the presence of characters. If char-
acters are present, VAR is set to 1, otherwise it is set to 0. No input is actually done.
In the next example the PRINT statement puts INPUT NAME before the input
prompt:
In the next example the contents of X are displayed at column 5, row 5 in a field of
10 characters. The user edits the field, replacing its original contents (CURRENT)
with new contents (NEW). The new input is displayed. If the PRINT statement
after the INPUT statement were not used, X would be printed immediately
following the input field on the same line, since INPUT with the @ expression does
not execute a LINEFEED after a RETURN.
INPUTCLEAR
Syntax
    INPUTCLEAR
Description
Use the INPUTCLEAR statement to clear the type-ahead buffer. You can use this
statement before input prompts so input is not affected by unwanted characters.
Example
    PRINT "DO YOU WANT TO CONTINUE (Y/N)?"
    INPUTCLEAR
    INPUT ANSWER, 1
INPUTDISP
Syntax
    INPUTDISP [@(col, row) [ , | : ] ] variable [format]
Description
Use the INPUTDISP statement with an @ expression to position the cursor at a
specified location and define a format for the variable to print. The current contents
of variable are displayed as the value in the defined field. Calculations are based on
display length rather than character length.
col specifies the column position, and row specifies the row position.
format is an expression that defines how the variable is to be displayed in the
output field. The syntax of the format expression is the same as that for the FMT
function.
Example
    PRINT @(-1)
    X = "CURRENT LINE"
    INPUTDISP @(5,5),X"10T"
The program output on a cleared screen is:
            CURRENT
            LINE
INPUTDP
Syntax
    INPUTDP variable [ , length]   [ : ] [ _ ] [THEN statements] [ELSE statements]
Description
In NLS mode, use the INPUTDP statement to let the user enter data. The INPUTDP
statement is similar to the INPUT, INPUTIF, and INPUTDISPstatements, but it
calculates display positions rather than character lengths.
variable contains the input from a user prompt.
length specifies the maximum number of characters in display length allowed as
input. INPUTDP calculates the display length of the input field based on the
current terminal map. When the specified number of characters is entered, an auto-
matic newline is executed.
The colon ( : ) executes the RETURN, suppressing the newline. This allows
multiple input prompts on a single line.
If you use the underscore ( _ ), the user must enter the RETURN manually when
input is complete, and the newline is not executed.
For more information about display length, see the DataStage NLS Guide.
INPUTERR
Syntax
    INPUTERR [error.message]
Description
Use the INPUTERR statement to print a formatted error message on the bottom
line of the terminal. error.message is an expression that evaluates to the error
message text. The message is cleared by the next INPUT statement or is over-
written by the next INPUTERR or PRINTERR statement. INPUTERR clears the
type-ahead buffer.
error.message can be any BASIC expression. The elements of the expression can be
numeric or character strings, variables, constants, or literal strings. The null value
cannot be output. The expression can be a single expression or a series of expres-
sions separated by commas ( , ) or colons ( : ) for output formatting. If no error
message is designated, a blank line is printed. If error.message evaluates to the null
value, the default error message is printed:
    Message ID is NULL:        undefined error
Expressions separated by commas are printed at preset tab positions. The default
tabstop setting is 10 characters. For information about changing the default setting,
see the TABSTOP statement. Multiple commas can be used together to cause
multiple tabulations between expressions.
Expressions separated by colons are concatenated: that is, the expression following
the colon is printed immediately after the expression preceding the colon.
INPUTIF
Use the INPUTIF statement to assign the contents of the type-ahead buffer to a
variable. For details, see the INPUT statement.
INPUTNULL
Syntax
    INPUTNULL character
Description
Use the INPUTNULL statement to define a character to be recognized as an empty
string when it is input in response to an INPUT statement. If the only input to the
INPUT statement is character, that character is recognized as an empty string. char-
acter replaces the default value of the INPUT variable with an empty string. If
character evaluates to the null value, the INPUTNULL statement fails and the
program terminates with a run-time error message.
You can also assign an empty string to the variable used in theINPUT @ statement
before executing the INPUT @. In this case entering a RETURN leaves the variable
set to the empty string.
Note: Although the name of this statement is INPUTNULL, it does not define
      character to be recognized as the null value. It defines it to be recognized as
      an empty string.
INPUTTRAP
Syntax
    INPUTTRAP [trap.chars] {GOTO | GOSUB} label [ ,label …]
Description
Use the INPUTTRAP statement to branch to a program label or subroutine when
a trap character is input. Execution is passed to the statement label which corre-
sponds to the trap number of the trap character. If the trap number is larger than
the number of labels, execution is passed to the statement specified by the last label
in the list.
trap.chars is an expression that evaluates to a string of characters, each of which
defines a trap character. The first character in the string is defined as trap one.
Additional characters are assigned consecutive trap numbers. Each trap character
corresponds to one of the labels in the label list. If trap.chars evaluates to the null
value, the INPUTTRAP statement fails and the program terminates with a run-
time error message.
Using GOTO causes execution to be passed to the specified statement label.
Control is not returned to the INPUTTRAP statement except by the use of another
trap. Using GOSUB causes execution to be passed to the specified subroutine, but
control can be returned to the INPUTTRAP statement by a RETURN (value) state-
ment. Control is returned to the statement following the INPUTTRAP statement,
not the INPUT @ statement that received the trap.
INS
Syntax
      INS expression BEFORE dynamic.array < field# [ ,value# [ ,subvalue#] ] >
Description
Use the INS statement to insert a new field, value, or subvalue into the specified
dynamic.array.
expression specifies the value of the new element to be inserted.
dynamic.array is an expression that evaluates to the dynamic array to be modified.
field#, value#, and subvalue# specify the type and position of the new element to be
inserted and are called delimiter expressions.
There are three possible outcomes of the INS statement, depending on the delim-
iter expressions specified.
Case 1:       If both value# and subvalue# are omitted or are 0, INS inserts a new
              field with the value of expression into the dynamic array.
                  • If field# is positive and less than or equal to the number of
                    fields in dynamic.array, the value of expression followed by a
                    field mark is inserted before the field specified by field#.
                  • If field# is −1, a field mark followed by the value of expression is
                    appended to the last field in dynamic.array.
                  • If field# is positive and greater than the number of fields in
                    dynamic.array, the proper number of field marks followed by
                    the value of expression are appended so that the value of field#
                    is the number of the new field.
Case 2:       If value# is nonzero and subvalue# is omitted or is 0, INS inserts a new
              value with the value of expression into the dynamic array.
                  • If value# is positive and less than or equal to the number of
                    values in the field, the value of expression followed by a value
                    mark is inserted before the value specified by value#.
                  • If value# is −1, a value mark followed by the value of expression
                    is appended to the last value in the field.
end of a value, a delimiter is appended to the dynamic array, field, or value. Use
the −EXTRA.DELIM option of the $OPTIONS statement to make the INS statement
work as it does in IDEAL, PICK, and REALITY flavor accounts.
Examples
In the following examples a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S.
The first example inserts the character # before the first field and sets Q to
#FFF1V2V3S6F9F5F7V3:
    R=@FM:@FM:1:@VM:2:@VM:3:@SM:6:@FM:9:@FM:5:@FM:7:@VM:3
    Q=R
    INS "#" BEFORE Q<1,0,0>
The next example inserts a # before the third value of field 3 and sets the value of
Q to FF1V2V#V3S6F9F5F7V3:
    Q=R
    INS "#" BEFORE Q<3,3,0>
The next example inserts a value mark followed by a # after the last value in the
field and sets Q to FF1V2V3S6F9V#F5F7V3:
    Q=R
    INS "#" BEFORE Q<4,-1,0>
The next example inserts a # before the second subvalue of the second value of field
3 and sets Q to FF1V2S#V3S6F9F5F7V3:
    Q=R
    INS "#" BEFORE Q<3,2,2>
INSERT
Syntax
    INSERT (dynamic.array, field#, value#, subvalue#, expression)
Description
Use the INSERT function to return a dynamic array that has a new field, value, or
subvalue inserted into the specified dynamic array.
dynamic.array is an expression that evaluates to a dynamic array.
field#, value#, and subvalue# specify the type and position of the new element to be
inserted and are called delimiter expressions. value# and subvalue# are optional, but
if either is omitted, a semicolon ( ; ) must precede expression, as shown in the
second syntax line.
expression specifies the value of the new element to be inserted.
There are three possible outcomes of the INSERT function, depending on the
delimiter expressions specified.
Case 1:     If both value# and subvalue# are omitted or are 0, INSERT inserts a
            new field with the value of expression into the dynamic array.
                 • If field# is positive and less than or equal to the number of
                   fields in dynamic.array, the value of expression followed by a
                   field mark is inserted before the field specified by field#.
                 • If field# is −1, a field mark followed by the value of expression is
                   appended to the last field in dynamic.array.
                 • If field# is positive and greater than the number of fields in
                   dynamic.array, the proper number of field marks followed by
                   the value of expression are appended so that the value of field#
                   is the number of the new field.
Case 2:     If value# is nonzero and subvalue# is omitted or is 0, INSERT inserts a
            new value with the value of expression into the dynamic array.
                 • If value# is positive and less than or equal to the number of
                   values in the field, the value of expression followed by a value
                   mark is inserted before the value specified by value#.
the −EXTRA.DELIM option of the $OPTIONS statement to make the INSERT func-
tion work as it does in IDEAL, PICK, and REALITY flavor accounts.
Examples
In the following examples a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S.
The first example inserts the character # before the first field and sets Q to
#FFF1V2V3S6F9F5F7V:
    R=@FM:@FM:1:@VM:2:@VM:3:@SM:6:@FM:9:@FM:5:@FM:7:@VM:3
    Q=INSERT(R,1,0,0,"#")
The next example inserts a # before the third value of field 3 and sets the value of
Q to FF1V2V#V3S6F9F5F7V3:
    Q=INSERT(R,3,3;"#")
The next example inserts a value mark followed by a # after the last value in the
field and sets Q to FF1V2V3S6F9V#F5F7V3:
    Q=INSERT(R,4,-1,0,"#")
The next example inserts a # before the second subvalue of the second value of field
3 and sets Q to FF1V2S#V3S6F9F5F7V3:
    Q=INSERT(R,3,2,2;"#")
INT
Syntax
      INT (expression)
Description
Use the INT function to return the integer portion of an expression.
expression must evaluate to a numeric value. Any arithmetic operations specified
are calculated using the full accuracy of the system. The fractional portion of the
value is truncated, not rounded, and the integer portion remaining is returned.
If expression evaluates to the null value, null is returned.
Example
      PRINT "123.45 ", INT(123.45)
      PRINT "454.95 ", INT(454.95)
This is the program output:
      123.45     123
      454.95     454
ISNULL
Syntax
    ISNULL (variable)
Description
Use the ISNULL function to test whether a variable is the null value. If variable is
the null value, 1 (true) is returned, otherwise 0 (false) is returned. This is the only
way to test for the null value since the null value is not equal to any value,
including itself.
Example
    X = @NULL
    Y = @NULL.STR
    PRINT ISNULL(X), ISNULL(Y)
This is the program output:
    1    0
ISNULLS
Syntax
    ISNULLS (dynamic.array)
    CALL –ISNULLS (return.array, dynamic.array)
Description
Use the ISNULLS function to test whether any element of dynamic.array is the null
value. A dynamic array is returned, each of whose elements is either 1 (true) or 0
(false). If an element in dynamic.array is the null value, 1 is returned, otherwise 0 is
returned. This is the only way to test for the null value since the null value is not
equal to any value, including itself.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    DA = ""
    FOR I = 1 TO 7
       DA := I:@FM
       IF I = 5 THEN DA := @NULL.STR:@FM
    NEXT I
    PRINT ISNULLS(DA)
This is the program output:
    0F0F0F0F0F1F0F0F0
ITYPE
Syntax
    ITYPE (i.type)
Description
Use the ITYPE function to return the value resulting from the evaluation of an
I-descriptor expression in a DataStage file dictionary.
i.type is an expression evaluating to the contents of the compiled I-descriptor. The
I-descriptor must have been compiled before the ITYPE function uses it, otherwise
you get a run-time error message.
i.type can be set to the I-descriptor to be evaluated in several ways. One way is to
read the I-descriptor from a file dictionary into a variable, then use the variable as
the argument to the ITYPE function. If the I-descriptor references a record ID, the
current value of the system variable @ID is used. If the I-descriptor references field
values in a data record, the data is taken from the current value of the system vari-
able @RECORD.
To assign field values to @RECORD, read a record from the data file into
@RECORD before invoking the ITYPE function.
If i.type evaluates to the null value, the ITYPE function fails and the program termi-
nates with a run-time error message.
Example
This is the SUN.MEMBER file contents:
    AW
    F1: ACCOUNTING
    TRX
    F1: MARKETING
    JXA
    F1: SALES
This is the DICT.ITME contents:
    DEPARTMENT
    F1:D
    2:1
    3:
    4:
    5:10L
    6:L
This is the program source code:
    OPEN 'SUN.MEMBER' TO FILE ELSE STOP
    OPEN 'DICT','SUN.MEMBER' TO D.FILE ELSE STOP
    *
    READ ITEM.ITYPE FROM D.FILE, 'DEPARTMENT' ELSE STOP
    *
    EXECUTE 'SELECT SUN.MEMBER'
    LOOP
    READNEXT @ID DO
    *
        READ @FRECORD FROM FILE, @ID THEN
        *
        PRINT @ID: "WORKS IN DEPARTMENT" ITYPE(ITEM.ITYPE)
        END
    REPEAT
    STOP
    END
This is the program output:
    3 records   selected to Select List #0
    FAW WORKS   IN DEPARTMENT ACCOUNTING
    TRX WORKS   IN DEPARTMENT MARKETING
    JXA WORKS   IN DEPARTMENT SALES
KEYEDIT
Syntax
    KEYEDIT (function, key) [ ,(function, key) ] …
Description
Use the KEYEDIT statement to assign specific keyboard keys to the editing func-
tions of the INPUT @ statement, and to the !EDIT.INPUT and !GET.KEY
subroutines. KEYEDIT supports the following editing functions:
    •   Left arrow (<—)
    •   Enter (Return)
    •   Back space
    •   Right arrow (—>)
    •   Insert character
    •   Delete character
    •   Insert mode on
    •   Insert mode off
    •   Clear field
    •   Erase to end-of-line
    •   Insert mode toggle
In addition to the supported editing functions, two codes exist to designate the Esc
and function keys.
function is an expression that evaluates to a numeric code assigned to a particular
editing function.
Code      Function
1         Function key
2         Left arrow (<—)
3         Return key
4         Back space
5         Esc key
6         Right arrow (—>)
7         Insert character
8         Delete character
 Code      Function
 9         Insert mode ON
 10        Insert mode OFF
 11        Clear from current position to end-of-line
 12        Erase entire line
 13        Insert mode toggle
key is an expression evaluating to a decimal value that designates the keyboard key
to assign to the editing function. There are three key types, described in the
following table:
If either function or key evaluates to the null value or an empty string, the KEYEDIT
statement fails, the program terminates, and a run-time error message is produced.
To define key, you must know the ASCII value generated by the keyboard on the
terminal being used. Once you know the ASCII code sequence generated by a
particular keyboard key, you can use one of the following three methods for
deriving the numeric key value.
First you define the function key value. Do this by issuing the KEYEDIT statement
with a function value of 1 and with a key value defined as the ASCII value of the
preamble character, i.e., KEYEDIT (1, 1).
Once you define the function key, the following formula is applied to the
remaining characters in the sequence:
The results of each calculation are then added together. Finally, the constant 160 is
added to insure that the final key parameter value falls within the range of 160
through 2,139,062,303. For our example above, this would yield 176,043,613 + 160,
or 176,043,773. To complete this example and assign this key to the Clear Field
functionality, use the following KEYEDIT statement:
    KEYEDIT (11, 176043773)
Historically, key values falling in the range of 160 through 287 included an implied
Return, as there was no method for supporting multiple character sequences. With
the support of multiple character sequences, you must now include the Return in
the calculation for proper key recognition, with one exception. For legacy key
values that fall within the range of 160 through 287, a Return is automatically
appended to the end of the character sequence, yielding an internal key parameter
of greater value.
Value Description
1        A KEYEDIT value
2        A KEYTRAP value
3        A KEYEXIT value
4        The INPUTNULL value
5        An unsupported value
Example
The following example illustrates the use of the KEYEDIT statement and the
SYSTEM(1050) function:
    KEYEDIT (1,1), (2,21), (3,13), (4,8), (6,6), (12,176043773)
    KEYTRAP (1,2)
    keys.dfn=SYSTEM(1050)
    PRINT "#","Type","Value","Key"
    XX=DCOUNT(keys.dfn,@FM)
    FOR I=1 TO XX
    print I-1,keys.dfn<I,1>,keys.dfn<I,2>,keys.dfn<I,3>
    NEXT I
The program output is:
    #            Type          Value        Key
    0            1             3            10
    1            1             3            13
    2            1             4            8
    3            1             1            1
    4            1             2            21
   5           1           6     6
   6           1           12    176043773
   7           2           1     2
KEYEXIT
Syntax
    KEYEXIT (value, key) [ , (value, key) ] …
Description
Use the KEYEXIT statement to specify exit traps for the keys assigned specific
functions by the KEYEDIT statement. When an exit trap key is typed, the variable
being edited with the INPUT@ statement or the !EDIT.INPUT subroutine remains
in its last edited state. Use the KEYTRAP statement to restore the variable to its
initial state.
value is an expression that specifies a user-defined trap number for each key
assigned by the KEYEDIT statement.
key is a decimal value that designates the specific keyboard key assigned to the
editing function. There are three key types, described in the following table:
See the KEYEDITstatement for how to derive the decimal value of control, escape,
and function keys.
If either the value or key expression evaluates to the null value or an empty string,
the KEYEXIT statement fails, the program terminates, and a run-time error
message is produced.
KEYEXIT sets the STATUS function to the trap number of any trap key typed by
the user.
Examples
The following example sets up Ctrl-B as an exit trap key. The STATUS function is
set to 1 when the user types the key.
    KEYEXIT (1,2)
The next example sets up Ctrl-K as an exit trap key. The STATUS function is set to
2 when the user types the key.
    KEYEXIT (2,11)
KEYIN
Syntax
    KEYIN ( )
Description
Use the KEYIN function to read a single character from the input buffer and return
it. All special character handling (such as case inversion, erase, kill, and so on) is
disabled. UNIX special character handling (processing of interrupts, XON/XOFF,
conversion of CR to LF, and so on) still takes place.
Calculations are based on display length rather than character length.
No arguments are required with the KEYIN function; however, parentheses are
required.
KEYTRAP
Syntax
    KEYTRAP (value, key) [ , (value, key) ] …
Description
Use the KEYTRAP statement to specify traps for the keys assigned specific func-
tions by the KEYEDIT statement. When a trap key is typed, the variable being
edited with the INPUT @ statement or the !EDIT.INPUT subroutine is restored to
its initial state. Use the KEYEXIT statement to leave the variable in its last edited
state.
value is an expression that evaluates to a user-defined trap number for each key
assigned by the KEYEDIT statement.
key is a decimal value which designates the specific keyboard key assigned to the
editing function. There are three key types, described in the following table:
See the KEYEDIT statement for how to derive the decimal value of control, escape,
and function keys.
If either the value or key expression evaluates to the null value or an empty string,
the KEYEXIT statement fails, the program terminates, and a run-time error
message is produced.
KEYTRAP sets the STATUS function to the trap number of any trap key typed by
the user.
Examples
The following example sets up Ctrl-B as a trap key. The STATUS function is set to
1 when the user types the key.
    KEYTRAP (1, 2)
The next example defines function key values for the F1, F2, F3, and F4 keys on a
Wyse 50 terminal:
    KEYEDIT (1,1)
    KEYTRAP (1,224), (2,225), (3,226), (4,227)
    PRINT @(-1)
    VALUE = "KEY"
    INPUT @ (10,10):VALUE
    X=STATUS()
    BEGIN CASE
        CASE X = 1
         PRINT "FUNCTION KEY 1"
        CASE X =2
         PRINT "FUNCTION KEY 2"
        CASE X =3
         PRINT "FUNCTION KEY 3"
        CASE X =4
         PRINT "FUNCTION KEY 4"
    END CASE
    PRINT VALUE
    STOP
    END
LEFT
Syntax
       LEFT (string, n)
Description
Use the LEFT function to extract a substring comprising the first n characters of a
string, without specifying the starting character position. It is equivalent to the
following substring extraction operation:
       string [ 1, length ]
If string evaluates to the null value, null is returned. If n evaluates to the null value,
the LEFT function fails and the program terminates with a run-time error message.
Example
       PRINT LEFT("ABCDEFGH",3)
This is the program output:
       ABC
LEN
Syntax
      LEN (string)
Description
Use the LEN function to return the number of characters in string. Calculations are
based on character length rather than display length.
string must be a string value. The characters in string are counted, and the count is
returned.
The LEN function includes all blank spaces, including trailing blanks, in the
calculation.
If string evaluates to the null value, 0 is returned.
If NLS is enabled, use the LENDP function to return the length of a string in
display positions rather than character length. FFor more information about
display length, see the DataStage NLS Guide.
Example
      P="PORTLAND, OREGON"
      PRINT "LEN(P)= ",LEN(P)
      *
      NUMBER=123456789
      PRINT "LENGTH OF NUMBER IS ",LEN(NUMBER)
This is the program output:
      LEN(P)= 16
      LENGTH OF NUMBER IS                 9
LENDP
Syntax
    LENDP (string [ ,mapname ] )
Description
In NLS mode, use the LENDP function to return the number of display positions
occupied by string when using the specified map. Calculations are based on
display length rather than character length.
string must be a string value. The display length of string is returned.
mapname is the name of an installed map. If mapname is not installed, the character
length of string is returned.
If mapname is omitted, the map associated with the channel activated by PRINTER
ON is used, otherwise it uses the map for print channel 0. You can also specify
mapname as CRT, AUX, LPTR, and OS. These values use the maps associated with
the terminal, auxiliary printer, print channel 0, or the operating system, respec-
tively. If you specify mapname as NONE, the string is not mapped.
Any unmappable characters in string have a display length of 1.
The LENDP function includes all blank spaces, including trailing blanks, in the
calculation.
If string evaluates to the null value, 0 is returned.
If you use the LENDP function with NLS disabled, the program behaves as if the
LEN function is used. See the LEN function to return the length of a string in char-
acter rather than display positions.
For more information about display length, see the DataStage NLS Guide.
LENS
Syntax
       LENS (dynamic.array)
       CALL −LENS (return.array, dynamic.array)
       CALL !LENS (return.array, dynamic.array)
Description
Use the LENS function to return a dynamic array of the number of display posi-
tions in each element of dynamic.array. Calculations are based on character length
rather than display length.
Each element of dyamic.array must be a string value. The characters in each element
of dynamic.array are counted, and the counts are returned.
The LENS function includes all blank spaces, including trailing blanks, in the
calculation.
If dynamic.array evaluates to the null value, 0 is returned. If any element of
dynamic.array is null, 0 is returned for that element.
If NLS is enabled, use the LENSDP function to return a dynamic array of the
number of characters in each element of dynamic.array in display positions. For
more information about display length, see the DataStage NLS Guide.
LENSDP
Syntax
    LENSDP (dynamic.array [, mapname ] )
Description
In NLS mode, use the LENSDP function to return a dynamic array of the number
of display positions occupied by each element of dynamic.array. Calculations are
based on display length rather than character length.
Each element of dynamic.array must be a string value. The display lengths of each
element of dynamic.array are counted, and the counts are returned.
mapname is the name of an installed map. If mapname is not installed, the character
length of string is returned.
If mapname is omitted, the map associated with the channel activated by PRINTER
ON is used, otherwise it uses the map for print channel 0. You can also specify
mapname as CRT, AUX, LPTR, and OS. These values use the maps associated with
the terminal, auxiliary printer, print channel 0, or the operating system, respec-
tively. If you specify mapname as NONE, the string is not mapped.
Any unmappable characters in dynamic.array have a display length of 1.
The LENSDP function includes all blank spaces, including trailing blanks, in the
calculation.
If dynamic.array evaluates to the null value, 0 is returned. If any element of
dynamic.array is null, 0 is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If you use the LENSDP function with NLS disabled, the program behaves as if the
LENS function is used. See the LENS function to return the length of a string in
character length rather than display length.
For more information about display length, see the DataStage NLS Guide.
LES
Syntax
      LES (array1, array2)
      CALL −LES (return.array, array1, array2)
      CALL !LES (return.array, array1, array2)
Description
Use the LES function to test if elements of one dynamic array are less than or equal
to the elements of another dynamic array.
Each element of array1 is compared with the corresponding element of array2. If the
element from array1 is less than or equal to the element from array2, a 1 is returned
in the corresponding element of a new dynamic array. If the element from array1 is
greater than the element from array2, a 0 is returned. If an element of one dynamic
array has no corresponding element in the other dynamic array, the undefined
element is evaluated as empty, and the comparison continues.
If either of a corresponding pair of elements is the null value, null is returned for
that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
LET
Syntax
      [LET] variable = expression
Description
Use the LET statement to assign the value of expression to variable. See “assignment
statements” for more information about assigning values to variables.
Example
      LET A=55
      LET B=45
      LET C=A+B
      LET D="55+45="
      LET E=D:C
      PRINT E
This is the program output:
      55+45=100
LN
Syntax
     LN (expression)
Description
Use the LN function to calculate the natural logarithm of the value of an expres-
sion, using base "e". The value of "e" is approximately 2.71828. expression must
evaluate to a numeric value greater than 0.
If expression is 0 or negative, 0 is returned and a warning is printed. If expression
evaluates to the null value, null is returned.
Example
     PRINT LN(6)
This is the program output:
     1.7918
LOCALEINFO
Syntax
    LOCALEINFO (category)
Description
In NLS mode, use the LOCALEINFO function to retrieve the settings of the current
locale.
category is one of the following tokens that are defined in the DataStage include file
UVNLSLOC.H:
If the specified category is set to OFF, LOCALEINFO returns the string OFF.
If the LOCALEINFO function fails to execute, LOCALEINFO returns one of the
following:
For more information about locales, see the DataStage NLS Guide.
Example
The following example shows the contents of the multivalued DAYS field when
the locale FR-FRENCH is current. Information for LCT$DAYS is contained in the
UVNLSLOC.H file in the INCLUDE directory in the UV account directory.
    category.info = LOCALEINFO(LC$TIME)
    PRINT category.info<LCT$DAYS>
This is the program output:
lundi}mardi}mercredi}jeudi}vendredi}samedi}dimanche
LOCATE
Syntax (IDEAL, REALITY)
    LOCATE expression IN dynamic.array [ < field# [ ,value#] > ] [ ,start] [BY seq]
         SETTING variable
          {THEN statements [ELSE statements] | ELSE statements}
Syntax (PICK)
    LOCATE (expression, dynamic.array[ ,field# [ ,value#] ] ; variable [ ;seq] )
             {THEN statements [ELSE statements] | ELSE statements}
Syntax (INFORMATION)
    LOCATE expression IN dynamic.array < field# [ ,value# [ ,subvalue#] ] >
             [BY seq] SETTING variable
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use a LOCATE statement to search dynamic.array for expression and to return a
value indicating one of the following:
    • Where expression was found in dynamic.array
    • Where expression should be inserted in dynamic.array if it was not found
The search can start anywhere in dynamic.array.
expression evaluates to the string to be searched for in dynamic.array. If expression or
dynamic.array evaluate to the null value, variable is set to 0 and the ELSE statements
are executed. If expression and dynamic.array both evaluate to empty strings, variable
is set to 1 and the THEN statements are executed.
field#, value#, and subvalue# are delimiter expressions, specifying:
    • Where the search is to start in dynamic.array
    • What kind of element is being searched for
start evaluates to a number specifying the field, value, or subvalue from which to
start the search.
In the IDEAL and PICK syntaxes the delimiter expressions specify the level of the
search, and start specifies the starting position of the search. In INFORMATION
syntax the delimiter expressions specify the starting position of the search.
If any delimiter expression or start evaluates to the null value, the LOCATE state-
ment fails and the program terminates with a run-time error message.
variable stores the index of expression. variable returns a field number, value number,
or a subvalue number, depending on the delimiter expressions used. variable is set
to a number representing one of the following:
    • The index of the element containing expression, if such an element is found
    • An index that can be used in an INSERT function to create a new element
      with the value specified by expression
During the search, fields are processed as single-valued fields even if they contain
value or subvalue marks. Values are processed as single values, even if they
contain subvalue marks.
The search stops when one of the following conditions is met:
    • A field containing expression is found.
    • The end of the dynamic array is reached.
    • A field that is higher or lower, as specified by seq, is found.
If the elements to be searched are sorted in one of the ascending or descending
ASCII sequences listed below, you can use the BY seq expression to end the search.
The search ends at the place where expression should be inserted to maintain the
ASCII sequence, rather than at the end of the list of specified elements.
Use the following values for seq to describe the ASCII sequence being searched:
seq does not reorder the elements in dynamic.array; it specifies the terminating
conditions for the search. If a seq expression is used and the elements are not in the
sequence indicated by seq, an element with the value of expression may not be
found. If seq evaluates to the null value, the statement fails and the program
terminates.
The ELSE statements are executed if expression is not found. The format of the ELSE
statement is the same as that used in the IF…THEN statement.
Use the INFO.LOCATE option of the $OPTIONS statement to use the INFORMA-
TION syntax in other flavors.
If NLS is enabled, the LOCATE statement with a BY seq expression uses the Collate
convention as specified in the NLS.LC.COLLATE file to determine the sort order
for characters with ascending or descending sequences. The Collate convention
defines rules for casing, accents, and ordering. For more information about how
NLS calculates the order, see the DataStage NLS Guide.
Case 1:      If field# and value# are omitted, the search starts at the first field in
             dynamic.array.
Case 2:      If only field# is specified and it is greater than 0, the search starts at
             the first value in the field indicated by field#. If field# is less than or
             equal to 0, both field# and value# are ignored.
Case 3:      If both field# and value# are specified, the search starts at the first
             subvalue in the value specified by value#, in the field specified by
             field#. If field# is greater than 0, but value# is less than or equal to 0,
             LOCATE behaves as though only field# is specified.
INFORMATION Syntax
When you use the INFORMATION flavor syntax of LOCATE, three outcomes can
result depending on how the delimiter expressions are used. The results are
described as case 1, case 2, and case 3.
Case 1:      If both value# and subvalue# are omitted or are both less than or equal
             to 0, the search starts at the field indicated by field#.
Examples
The examples show the IDEAL and REALITY flavor LOCATE statement. A field
mark is shown by F, a value mark is shown by V, and a subvalue mark is shown by S.
    Q='X':@SM:"$":@SM:'Y':@VM:'Z':@SM:4:@SM:2:@VM:'B':@VM
    PRINT "Q= ":Q
    LOCATE "$" IN Q <1> SETTING WHERE ELSE PRINT 'ERROR'
    PRINT "WHERE= ",WHERE
    LOCATE "$" IN Q <1,1> SETTING HERE ELSE PRINT 'ERROR'
    PRINT "HERE= ", HERE
    NUMBERS=122:@FM:123:@FM:126:@FM:130:@FM
    PRINT "BEFORE INSERT, NUMBERS= ",NUMBERS
    NUM= 128
    LOCATE NUM IN NUMBERS <2> BY "AR" SETTING X ELSE
    NUMBERS = INSERT(NUMBERS,X,0,0,NUM)
    PRINT "AFTER INSERT, NUMBERS= ",NUMBERS
    END
This is the program output:
   Q= XS$SYVZS4S2VBV
   ERROR
   WHERE= 5
   HERE=    2
   BEFORE INSERT, NUMBERS= 122F123F126F130F
   AFTER INSERT, NUMBERS= 122F128F123F126F130F
LOCK
Syntax
    LOCK expression [THEN statements] [ELSE statements]
Description
Use the LOCK statement to protect specified user-defined resources or events
against unauthorized use or simultaneous data file access by different users.
There are 64 public semaphore locks in the DataStage system. They are task
synchronization tools but have no intrinsic definitions. You must define the
resource or event associated with each semaphore, ensuring that there are no
conflicts in definition or usage of these semaphores throughout the entire system.
expression evaluates to a number in the range of 0 through 63 that specifies the lock
to be set. A program can reset a lock any number of times and with any frequency
desired. If expression evaluates to the null value, the LOCK statement fails and the
program terminates with a run-time error message.
If program B tries to set a lock already set by program A, execution of program B
is suspended until the first lock is released by program A; execution of program B
then continues.
The ELSE clause provides an alternative to this procedure. When a LOCK state-
ment specifies a lock that has already been set, the ELSE clause is executed rather
than program execution being suspended.
Program termination does not automatically release locks set in the program. Each
LOCK statement must have a corresponding UNLOCK statement. If a program
locks the same semaphore more than once during its execution, a single UNLOCK
statement releases that semaphore.
The UNLOCK statement can specify the expression used in the LOCK statement
to be released. If no expression is used in the UNLOCK statement, all locks set by
the program are released.
Alternatively, locks can be released by logging off the system or by executing either
the QUIT command or the CLEAR.LOCKS command.
You can check the status of locks with the LIST.LOCKS command; this lists the
locks on the screen. The unlocked state is indicated by 0. The locked state is indi-
cated by a number other than 0 (including both positive and negative numbers).
The number is the unique signature of the user who has set the lock.
Note: The LOCK statement protects user-defined resources only. The READL,
      READU, READL, READVU, MATREADL, and MATREADU statements
      use a different method of protecting files and records.
Example
The following example sets lock 60, executes the LIST.LOCKS command, then
unlocks all locks set by the program:
   LOCK 60 ELSE PRINT "ALREADY LOCKED"
   EXECUTE "LIST.LOCKS"
   UNLOCK
The program displays the LIST.LOCKS report. Lock 60 is set by user 4.
    0:--    1:--    2:--    3:--    4:--    5:--    6:--    7:--
    8:--    9:--   10:--   11:--   12:--   13:--   14:--   15:--
   16:--   17:--   18:--   19:--   20:--   21:--   22:--   23:--
   24:--   25:--   26:--   27:--   28:--   29:--   30:--   31:--
   32:--   33:--   34:--   35:--   36:--   37:--   38:--   39:--
   40:--   41:--   42:--   43:--   44:--   45:--   46:--   47:--
   48:--   49:--   50:--   51:--   52:--   53:--   54:--   55:--
   56:--   57:--   58:--   59:--   60:4    61:--   62:--   63:--
LOOP
Syntax
    LOOP
       [loop.statements]
       [CONTINUE | EXIT]
    [{WHILE | UNTIL} expression [DO] ]
       [loop.statements]
       [CONTINUE | EXIT]
    REPEAT
LOOP…REPEAT
Description
Use the LOOP statement to start a LOOP…REPEAT program loop. A program
loop is a series of statements that executes for a specified number of repetitions or
until specified conditions are met.
Use the WHILE clause to indicate that the loop should execute repeatedly as long
as the WHILE expression evaluates to true (1). When the WHILE expression eval-
uates to false (0), repetition of the loop stops, and program execution continues
with the statement following the REPEAT statement.
Use the UNTIL clause to put opposite conditions on the LOOP statement. The
UNTIL clause indicates that the loop should execute repeatedly as long as the
UNTIL expression evaluates to false (0). When the UNTIL expression evaluates to
true (1), repetition of the loop stops, and program execution continues with the
statement following the REPEAT statement.
If a WHILE or UNTIL expression evaluates to the null value, the condition is false.
expression can also contain a conditional statement. Any statement that takes a
THEN or an ELSE clause can be used as expression, but without the THEN or ELSE
clause. When the conditional statement would execute the ELSE clause, expression
evaluates to false; when the conditional statement would execute the THEN
clause, expression evaluates to true. A LOCKED clause is not supported in this
context.
You can use multiple WHILE and UNTIL clauses in a LOOP…REPEAT loop. You
can also nest LOOP…REPEAT loops. If a REPEAT statement is encountered
without a previous LOOP statement, an error occurs during compilation.
Examples
 A=20                                 A=         20
 LOOP                                 A=         19
    PRINT "A= ", A                    A=         18
    A=A-1                             A=         17
 UNTIL A=15 REPEAT                    A=         16
 Q=3                                  Q=         3
 LOOP                                 Q=         2
     PRINT "Q= ",Q                    Q=         1
 WHILE Q DO                           Q=         0
     Q=Q-1
 REPEAT
LOWER
Syntax
     LOWER (expression)
Description
Use the LOWER function to return a value equal to expression, except that system
delimiters which appear in expression are converted to the next lower-level delim-
iter: field marks are changed to value marks, value marks are changed to subvalue
marks, and so on. If expression evaluates to the null value, null is returned.
The conversions are:
IM      CHAR(255)         to        FM        CHAR(254)
FM      CHAR(254)         to        VM        CHAR(253)
VM      CHAR(253)         to        SM        CHAR(252)
SM      CHAR(252)         to        TM        CHAR(251)
TM      CHAR(251)         to                  CHAR(250)
        CHAR(250)         to                  CHAR(249)
        CHAR(249)         to                  CHAR(248)
PIOPEN Flavor
In PIOPEN flavor, the delimiters that can be lowered are CHAR(255) through
CHAR(252). All other characters are left unchanged. You can obtain PIOPEN
flavor for the LOWER function by:
     • Compiling your program in a PIOPEN flavor account
     • Specifying the $OPTIONS INFO.MARKS statement
Examples
In the following examples an item mark is shown by I, a field mark is shown by F,
a value mark is shown by V, a subvalue mark is shown by S, and a text mark is
shown by T. CHAR(250) is shown as Z.
The following example sets A to DDFEEV123V77:
     A= LOWER('DD':IM'EE':FM:123:FM:777)
LTS
Syntax
      LTS (array1, array2)
      CALL −LTS (return.array, array1, array2)
      CALL !LTS (return.array, array1, array2)
Description
Use the LTS function to test if elements of one dynamic array are less than elements
of another dynamic array.
Each element of array1 is compared with the corresponding element of array2. If the
element from array1 is less than the element from array2, a 1 is returned in the corre-
sponding element of a new dynamic array. If the element from array1 is greater
than or equal to the element from array2, a 0 is returned. If an element of one
dynamic array has no corresponding element in the other dynamic array, the unde-
fined element is evaluated as an empty string, and the comparison continues.
If either of a corresponding pair of elements is the null value, null is returned for
that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
MAT
Syntax
      MAT array = expression
      MAT array1 = MAT array2
Description
Use the MAT statement to assign one value to all of the elements in the array or to
assign all the values of one array to the values of another array.
Use the first syntax to assign the same value to all array elements. Use any valid
expression. The value of expression becomes the value of each array element.
Use the second syntax to assign values from the elements of array2 to the elements
of array1. Both arrays must previously be named and dimensioned. The dimen-
sioning of the two arrays can be different. The values of the elements of the new
array are assigned in consecutive order, regardless of whether the dimensions of
the arrays are the same or not. If array2 has more elements than in array1, the extra
elements are ignored. If array2 has fewer elements, the extra elements of array1 are
not assigned.
Note: Do not use the MAT statement to assign individual elements of an array.
Examples
The following example sets all elements in ARRAY to the empty string:
   MAT ARRAY=''
MATBUILD
Syntax
    MATBUILD dynamic.array FROM array [ ,start         [ ,end] ] [USING delimiter]
Description
Use the MATBUILD statement to build a dynamic array from a dimensioned array.
dynamic.array is created by concatenating the elements of array beginning with start
and finishing with end. If start and end are not specified or are out of range, they
default to 1 and the size of the array respectively.
array must be named and dimensioned in a DIMENSION or COMMON statement
before it is used in this statement.
delimiter specifies characters to be inserted between fields of the dynamic array. If
delimiter is not specified, it defaults to a field mark. To specify no delimiter, specify
USING without delimiter.
If an element of array is the null value, the dynamic array will contain CHAR(128)
for that element. If start, end, or delimiter is the null value, the MATBUILD state-
ment fails and the program terminates with a run-time error.
Overflow Elements
PICK, IN2, and REALITY flavor dimensioned arrays contain overflow elements in
the last element. INFORMATION and IDEAL flavor dimensioned arrays contain
overflow elements in element 0.
In PICK, IN2, and REALITY flavor accounts, if end is not specified, dynamic.array
contains the overflow elements of array. In IDEAL and INFORMATION flavor
accounts, to get the overflow elements you must specify end as less than or equal
to 0, or as greater than the size of array.
REALITY flavor accounts use only the first character of delimiter, and if USING is
specified without a delimiter, delimiter defaults to a field mark rather than an
empty string.
MATCH
Syntax
      string MATCH[ES] pattern
Description
Use the MATCH operator or its synonym MATCHES to compare a string expres-
sion with a pattern.
pattern is a general description of the format of string. It can consist of text or the
special characters X, A, and N preceded by an integer used as a repeating factor.
For example, nN is the pattern for strings of n numeric characters.
The following table lists the pattern codes and their definitions:
 Pattern     Definition
 …           Any number of any characters (including none).
 0X          Any number of any characters (including none).
 nX          n number of any characters.
 0A          Any number of alphabetic characters (including none).
 nA          n number of alphabetic characters.
 0N          Any number of numeric characters (including none).
 nN          n number of numeric characters.
 'text'      Exact text; any literal string (quotation marks required).
 "text"      Exact text; any literal string (quotation marks required).
If n is longer than nine digits, it is used as text in a pattern rather than as a repeating
factor for a special character. For example, the pattern "1234567890N" is treated as
a literal string, not as a pattern of 1,234,567,890 numeric characters.
If the string being evaluated matches the pattern, the expression evaluates as true
( 1 ); otherwise, it evaluates as false ( 0 ). If either string or pattern is the null value,
the match evaluates as false.
A tilde ( ~ ) placed immediately before pattern specifies a negative match. That is,
it specifies a pattern or a part of a pattern that does not match the expression or a
part of the expression. The match is true only if string and pattern are of equal
length and differ in at least one character. An example of a negative match pattern
is:
    "'A'~'X'5N
This pattern returns a value of true if the expression begins with the letter A, which
is not followed by the letter X, and which is followed by any five numeric charac-
ters. Thus AB55555 matches the pattern, but AX55555, A55555, AX5555, and A5555
do not.
You can specify multiple patterns by separating them with value marks (ASCII
CHAR(253) ). The following expression is true if the address is either 16 alphabetic
characters or 4 numeric characters followed by 12 alphabetic characters; otherwise,
it is false:
    ADDRESS MATCHES "16A": CHAR(253): "4N12A"
An empty string matches the following patterns: "0A", "0X", "0N", "…", "", '', or \\.
If NLS is enabled, the MATCH operator uses the current values for alphabetic and
numeric characters specified in the NLS.LC.CTYPE file. For more information
about the NLS.LC.CTYPE file, see the DataStage NLS Guide.
MATCHFIELD
Syntax
    MATCHFIELD (string, pattern, field)
Description
Use the MATCHFIELD function to check a string against a match pattern (see the
MATCH operator for information about pattern matching).
field is an expression that evaluates to the portion of the match string to be
returned.
If string matches pattern, the MATCHFIELD function returns the portion of string
that matches the specified field in pattern. If string does not match pattern, or if
string or pattern evaluates to the null value, the MATCHFIELD function returns an
empty string. If field evaluates to the null value, the MATCHFIELD function fails
and the program terminates with a run-time error.
pattern must contain specifiers to cover all characters contained in string. For
example, the following statement returns an empty string because not all parts of
string are specified in the pattern:
    MATCHFIELD ("XYZ123AB", "3X3N", 1)
To achieve a positive pattern match on string above, the following statement might
be used:
    MATCHFIELD ("XYZ123AB", "3X3N0X", 1)
This statement returns a value of "XYZ".
Examples
In the following example the string does not match the pattern:
In the following example the entire string does not match the pattern:
MATPARSE
Syntax
    MATPARSE array FROM dynamic.array [ ,delimiter]
Description
Use the MATPARSE statement to separate the fields of dynamic.array into consecu-
tive elements of array.
array must be named and dimensioned in a DIMENSION or COMMON statement
before it is used in this statement.
start specifies the starting position in array. If start is less than 1, it defaults to 1.
end specifies the ending position in array. If end is less than 1 or greater than the
length of array, it defaults to the length of array.
delimiter is an expression evaluating to the characters used to delimit elements in
dynamic.array. Use a comma or USING to separate delimiter from dynamic.array.
delimiter can have no characters (an empty delimiter), one character, or more than
one character with the following effects:
    • An empty delimiter (a pair of quotation marks) parses dynamic.array so that
      each character becomes one element of array (see the second example). The
      default delimiter is a field mark. This is different from the empty delimiter.
      To use the default delimiter, omit the comma or USING following
      dynamic.array.
    • A single character delimiter parses dynamic.array into fields delimited by
      that character by storing the substrings that are between successive delim-
      iters as elements in the array. The delimiters are not stored in the array (see
      the first example).
    • A multicharacter delimiter parses dynamic.array by storing as elements both
      the substrings that are between any two successive delimiters and the
      substrings consisting of one or more consecutive delimiters in the
      following way: dynamic.array is searched until any of the delimiter charac-
      ters are found. All of the characters up to but not including the delimiter
      character are stored as an element of array. The delimiter character and any
Examples
MATREAD
Syntax
    MATREAD array FROM [file.variable,] record.ID [ON ERROR statements]
             {THEN statements [ELSE statements] | ELSE statements}
    { MATREADL | MATREADU } array FROM [file.variable,] record.ID
          [ON ERROR statements] [LOCKED statements]
          {THEN statements [ELSE statements] | ELSE statements}
Description
Use the MATREAD statement to assign the contents of the fields of a record from
a DataStage file to consecutive elements of array. The first field of the record
becomes the first element of array, the second field of the record becomes the
second element of array, and so on. The array must be named and dimensioned in
a DIMENSION or COMMON statement before it is used in this statement.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information about default files, see the OPEN statement). If the
file is neither accessible nor open, the program terminates with a run-time error
message.
If record.ID exists, array is set to the contents of the record, and the THEN state-
ments are executed; any ELSE statements are ignored. If no THEN statements are
specified, program execution continues with the next sequential statement. If
record.ID does not exist, the elements of array are not changed, and the ELSE state-
ments are executed; any THEN statements are ignored.
If either file.variable or record.ID evaluates to the null value, the MATREAD state-
ment fails and the program terminates with a run-time error. If any field in the
record is the null value, null becomes an element in array. If a value or a subvalue
in a multivalued field is the null value, it is read into the field as the stored repre-
sentation of null (CHAR(128)).
If the file is an SQL table, the effective user of the program must have SQL SELECT
privilege to read records in the file. For information about the effective user of a
program, see the AUTHORIZATION statement.
A MATREAD statement does not set an update record lock on the specified record.
That is, the record remains available for update to other users. To prevent other
Releasing Locks. A shared record lock can be released with a CLOSE, RELEASE,
or STOP statement. An update record lock can be released with a CLOSE, DELETE,
MATWRITE, RELEASE, STOP, WRITE, or WRITEV statement.
Locks acquired or promoted within a transaction are not released when the
previous statements are processed.
Example
    DIM ARRAY(10)
    OPEN 'SUN.MEMBER' TO SUN.MEMBER ELSE STOP
    MATREAD ARRAY FROM SUN.MEMBER, 6100 ELSE STOP
    *
    FOR X=1 TO 10
    PRINT "ARRAY(":X:")",ARRAY(X)
    NEXT X
    *
    PRINT
    *
    DIM TEST(4)
    OPEN '','SUN.SPORT' ELSE STOP 'CANNOT OPEN SUN.SPORT'
    MATREAD TEST FROM 851000 ELSE STOP
    *
    FOR X=0 TO 4
    PRINT "TEST(":X:")",TEST(X)
    NEXT X
This is the program output:
    ARRAY(1)   MASTERS
    ARRAY(2)   BOB
    ARRAY(3)   55 WESTWOOD ROAD
    ARRAY(4)   URBANA
    ARRAY(5)   IL
    ARRAY(6)   45699
    ARRAY(7)   1980
    ARRAY(8)   SAILING
   ARRAY(9)
   ARRAY(10)     II
   TEST(0) 6258
   TEST(1) 6100
   TEST(2) HARTWELL
   TEST(3) SURFING
   TEST(4) 4
MATREADL
Use the MATREADL statement to set a shared record lock and perform the
MATREAD statement. For details, see the MATREAD statement.
MATREADU
Use the MATREADU statement to set an update record lock and perform the
MATREAD statement. For details, see the MATREAD statement.
MATWRITE
Syntax
    MATWRITE[U] array ON | TO [file.variable,] record.ID
             [ON ERROR statements] [LOCKED statements]
             [THEN statements] [ELSE statements]
Description
Use the MATWRITE statement to write data from the elements of a dimensioned
array to a record in a DataStage file. The elements of array replace any data stored
in the record. MATWRITE strips any trailing empty fields from the record.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information on default files, see the OPEN statement). If the file
is neither accessible nor open, the program terminates with a run-time message,
unless ELSE statements are specified.
If the file is an SQL table, the effective user of the program must have SQL INSERT
and UPDATE privileges to read records in the file. For information about the effec-
tive user of a program, see the AUTHORIZATION statement.
If the OPENCHK configurable parameter is set to TRUE, or if the file is opened
with the OPENCHECK statement, all SQL integrity constraints are checked for
every MATWRITE to an SQL table. If an integrity check fails, the MATWRITE
statement uses the ELSE clause. Use the ICHECK function to determine what
specific integrity constraint caused the failure.
The system searches the file for the record specified by record.ID. If the record is not
found, MATWRITE creates a new record.
If NLS is enabled, MATWRITE and other BASIC statements that perform I/O oper-
ations always map internal data to the external character set using the appropriate
map for the output file. For details, see the WRITE statement. For more informa-
tion about maps, see the DataStage NLS Guide.
If a fatal error occurs, and the ON ERROR clause was not specified, or was ignored
(as in the case of an active transaction), the following occurs:
    • An error message appears.
    • Any uncommitted transactions begun within the current execution envi-
      ronment roll back.
    • The current program terminates.
    • Processing continues with the next statement of the previous execution
      environment, or the program returns to the command prompt.
A fatal error can occur if any of the following occur:
    • A file is not open.
    • file.variable is the null value.
    • A distributed file contains a part file that cannot be accessed.
If the ON ERROR clause is used, the value returned by the STATUS function is the
error number.
If either file.variable or record.ID evaluates to the null value, the MATWRITE state-
ment fails and the program terminates with a run-time error message. Null
elements of array are written to record.ID as the stored representation of the null
value, CHAR(128).
Example
    DIM ARRAY(5)
    OPEN 'EX.BASIC' TO EX.BASIC ELSE STOP 'CANNOT OPEN'
    MATREADU ARRAY FROM EX.BASIC, 'ABS' ELSE STOP
    ARRAY(1)='Y = 100'
    MATWRITE ARRAY TO EX.BASIC, 'ABS'
    PRINT 'STATUS()= ',STATUS()
This is the program output:
    STATUS()=       0
MATWRITEU
Use the MATWRITEU statement to maintain an update record lock and perform
the MATWRITE statement. For details, see the MATWRITE statement.
MAXIMUM
Syntax
    MAXIMUM (dynamic.array)
    CALL !MAXIMUM (result, dynamic.array)
Description
Use the MAXIMUM function to return the element with the highest numeric value
in dynamic.array. Nonnumeric values, except the null value, are treated as 0. If
dynamic.array evaluates to the null value, null is returned. Any element that is the
null value is ignored, unless all elements of dynamic.array are null, in which case
null is returned.
result is the variable that contains the largest element found in dynamic.array.
dynamic.array is the array to be tested.
Examples
    A=1:@VM:"ZERO":@SM:20:@FM:-25
    PRINT "MAX(A)=",MAXIMUM(A)
This is the program output:
    MAX(A)=20
In the following example, the !MAXIMUM subroutine is used to obtain the
maximum value contained in array A. The nonnumeric value, Z, is treated as 0.
    A=1:@VM:25:@VM:'Z':@VM:7
    CALL !MAXIMUM (RESULT,A)
    PRINT RESULT
This is the program output:
    0
MINIMUM
Syntax
    MINIMUM (dynamic.array)
    CALL !MINIMUM (result, dynamic.array)
Description
Use the MINIMUM function to return the element with the lowest numeric value
in dynamic.array. Nonnumeric values, except the null value, are treated as 0. If
dynamic.array evaluates to the null value, null is returned. Any element that is the
null value is ignored, unless all elements of dynamic.array are null, in which case
null is returned.
result is the variable that contains the smallest element found in dyamic.array.
dynamic.array is the array to be tested.
Examples
    A=1:@VM:"ZERO":@SM:20:@FM:-25
    PRINT "MIN(A)=",MINIMUM(A)
This is the program output:
    MIN(A)=-25
In the following example, the !MINIMUM subroutine is used to obtain the
minimum value contained in array A. The nonnumeric value, Q, is treated as 0.
    A=2:@VM:19:@VM:6:@VM:'Q'
    CALL !MINIMUM (RESULT,A)
    PRINT RESULT
This is the program output:
    0
MOD
Syntax
      MOD (dividend, divisor)
Description
Use the MOD function to calculate the value of the remainder after integer division
is performed on the dividend expression by the divisor expression.
The MOD function calculates the remainder using the following formula:
      MOD (X, Y) = X − (INT (X / Y) * Y)
dividend and divisor can evaluate to any numeric value, except that divisor cannot
be 0. If divisor is 0, a division by 0 warning message is printed, and 0 is returned. If
either dividend or divisor evaluates to the null value, null is returned.
The MOD function works like the REM function.
Example
      X=85; Y=3
      PRINT 'MOD (X,Y)= ',MOD (X,Y)
This is the program output:
      MOD (X,Y)=        1
MODS
Syntax
    MODS (array1, array2)
    CALL −MODS (return.array, array1, array2)
    CALL !MODS (return.array, array1, array2)
Description
Use the MODS function to create a dynamic array of the remainder after the integer
division of corresponding elements of two dynamic arrays.
The MODS function calculates each element according to the following formula:
    XY.element = X − (INT (X / Y) * Y)
X is an element of array1 and Y is the corresponding element of array2. The
resulting element is returned in the corresponding element of a new dynamic
array. If an element of one dynamic array has no corresponding element in the
other dynamic array, 0 is returned. If an element of array2 is 0, 0 is returned. If either
of a corresponding pair of elements is the null value, null is returned for that
element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    A=3:@VM:7
    B=2:@SM:7:@VM:4
    PRINT MODS(A,B)
This is the program output:
    1S0V3
MULS
Syntax
    MULS (array1, array2)
    CALL −MULS (return.array, array1, array2)
    CALL !MULS (return.array, array1, array2)
Description
Use the MULS function to create a dynamic array of the element-by-element multi-
plication of two dynamic arrays.
Each element of array1 is multiplied by the corresponding element of array2 with
the result being returned in the corresponding element of a new dynamic array. If
an element of one dynamic array has no corresponding element in the other
dynamic array, 0 is returned. If either of a corresponding pair of elements is the null
value, null is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    A=1:@VM:2:@VM:3:@SM:4
    B=4:@VM:5:@VM:6:@VM:9
    PRINT MULS(A,B)
This is the program output:
    4V10V18S0V0
NAP
Syntax
      NAP [milliseconds]
Description
Use the NAP statement to suspend the execution of a BASIC program, pausing for
a specified number of milliseconds.
milliseconds is an expression evaluating to the number of milliseconds for the
pause. If milliseconds is not specified, a value of 1 is used. If milliseconds evaluates
to the null value, the NAP statement is ignored.
NEG
Syntax
      NEG (number)
Description
Use the NEG function to return the arithmetic inverse of the value of the argument.
number is an expression evaluating to a number.
Example
In the following example, A is assigned the value of 10, and B is assigned the value
of NEG(A), which evaluates to –10:
      A = 10
      B = NEG(A)
NEGS
Syntax
    NEGS (dynamic.array)
    CALL −NEGS (return.array, dynamic.array)
Description
Use the NEGS function to return the negative values of all the elements in a
dynamic array. If the value of an element is negative, the returned value is positive.
If dynamic.array evaluates to the null value, null is returned. If any element is null,
null is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
NES
Syntax
      NES (array1, array2)
      CALL −NES (return.array, array1, array2)
      CALL !NES (return.array, array1, array2)
Description
Use the NES function to test if elements of one dynamic array are equal to the
elements of another dynamic array.
Each element of array1 is compared with the corresponding element of array2. If the
two elements are equal, a 0 is returned in the corresponding element of a new
dynamic array. If the two elements are not equal, a 1 is returned. If an element of
one dynamic array has no corresponding element in the other dynamic array, a 1
is returned. If either of a corresponding pair of elements is the null value, null is
returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
NEXT
Syntax
    NEXT [variable]
Description
Use the NEXT statement to end a FOR…NEXT loop, causing the program to
branch back to the FOR statement and execute the statements that follow it.
Each FOR statement must have exactly one corresponding NEXT statement.
variable is the name of the variable given as the index counter in the statement. If
the variable is not named, the most recently named index counter variable is
assumed.
Example
    FOR I=1 TO 10
       PRINT I:" ":
    NEXT I
    PRINT
This is the program output:
    1 2 3 4 5 6 7 8 9 10
NOBUF
Syntax
    NOBUF file.variable {THEN statements [ELSE statements] | ELSE statements}
Description
Use the NOBUF statement to turn off buffering for a file previously opened for
sequential processing. Normally DataStage uses buffering for sequential input and
output operations. The NOBUF statement turns off this buffering and causes all
writes to the file to be performed immediately. It eliminates the need for FLUSH
operations but also eliminates the benefits of buffering. The NOBUF statement
must be executed after a successful OPENSEQ or CREATE statement and before
any input or output operation is performed on the record.
If the NOBUF operation is successful, the THEN statements are executed; the ELSE
statements are ignored. If THEN statements are not present, program execution
continues with the next statement.
If the specified file cannot be accessed or does not exist, the ELSE statements are
executed; the THEN statements are ignored. If file.variable evaluates to the null
value, the NOBUF statement fails and the program terminates with a run-time
error message.
Example
In the following example, if RECORD1 in FILE.E can be opened, buffering is
turned off:
    OPENSEQ 'FILE.E', 'RECORD1' TO DATA THEN NOBUF DATA
    ELSE ABORT
NOT
Syntax
      NOT (expression)
Description
Use the NOT function to return the logical complement of the value of expression.
If the value of expression is true, the NOT function returns a value of false (0). If the
value of expression is false, the NOT function returns a value of true (1).
A numeric expression that evaluates to 0 is a logical value of false. A numeric
expression that evaluates to anything else, other than the null value, is a logical
true.
An empty string is logically false. All other string expressions, including strings
that include an empty string, spaces, or the number 0 and spaces, are logically true.
If expression evaluates to the null value, null is returned.
Example
      X=5; Y=5
      PRINT NOT(X-Y)
      PRINT NOT(X+Y)
This is the program output:
      1
      0
NOTS
Syntax
    NOTS (dynamic.array)
    CALL −NOTS (return.array, dynamic.array)
    CALL !NOTS (return.array, dynamic.array)
Description
Use the NOTS function to return a dynamic array of the logical complements of
each element of dynamic.array. If the value of the element is true, the NOTS function
returns a value of false (0) in the corresponding element of the returned array. If the
value of the element is false, the NOTS function returns a value of true (1) in the
corresponding element of the returned array.
A numeric expression that evaluates to 0 has a logical value of false. A numeric
expression that evaluates to anything else, other than the null value, is a logical
true.
An empty string is logically false. All other string expressions, including strings
which consist of an empty string, spaces, or the number 0 and spaces, are logically
true.
If any element in dynamic.array is the null value, null is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    X=5; Y=5
    PRINT NOTS(X-Y:@VM:X+Y)
This is the program output:
    1V0
NULL
Syntax
   NULL
Description
Use the NULL statement when a statement is required but no operation is to be
performed. For example, you can use it with the ELSE clause if you do not want
any operation performed when the ELSE clause is executed.
Example
   OPEN '','SUN.MEMBER' TO FILE ELSE STOP
   FOR ID=5000 TO 6000
      READ MEMBER FROM FILE, ID THEN PRINT ID ELSE NULL
   NEXT ID
NUM
Syntax
      NUM (expression)
Description
Use the NUM function to determine whether expression is a numeric or nonnu-
meric string. If expression is a number, a numeric string, or an empty string, it
evaluates to true and a value of 1 is returned. If expression is a nonnumeric string,
it evaluates to false and a value of 0 is returned.
A string that contains a period used as a decimal point ( . ) evaluates to numeric. A
string that contains any other character used in formatting numeric or monetary
amounts, for example, a comma ( , ) or a dollar sign ( $ ) evaluates to nonnumeric.
If expression evaluates to the null value, null is returned.
If NLS is enabled, NUM uses the Numeric category of the current locale to deter-
mine the decimal separator. For more information about locales, see the DataStage
NLS Guide.
Example
      X=NUM(2400)
      Y=NUM("Section 4")
      PRINT "X= ",X,"Y= ",Y
This is the program output:
      X=    1       Y=      0
NUMS
Syntax
    NUMS (dynamic.array)
    CALL −NUMS (return.array, dynamic.array)
    CALL !NUMS (return.array, dynamic.array)
Description
Use the NUMS function to determine whether the elements of a dynamic array are
numeric or nonnumeric strings. If an element is numeric, a numeric string, or an
empty string, it evaluates to true, and a value of 1 is returned to the corresponding
element in a new dynamic array. If the element is a nonnumeric string, it evaluates
to false, and a value of 0 is returned.
The NUMS of a numeric element with a decimal point ( . ) evaluates to true; the
NUMS of a numeric element with a comma ( , ) or dollar sign ( $ ) evaluates to
false.
If dynamic.array evaluates to the null value, null is returned. If an element of
dynamic.array is null, null is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If NLS is enabled, NUMS uses the Numeric category of the current locale to deter-
mine the decimal separator. For more information about locales, see the DataStage
NLS Guide.
OCONV
Syntax
    OCONV (string, conversion)
Description
Use the OCONV function to convert string to a specified format for external
output. The result is always a string expression.
string is converted to the external output format specified by conversion.
conversion must evaluate to one or more conversion codes separated by value
marks (ASCII 253).
If multiple codes are used, they are applied from left to right as follows: the left-
most conversion code is applied to string, the next conversion code to the right is
then applied to the result of the first conversion, and so on.
If string evaluates to the null value, null is returned. If conversion evaluates to the
null value, the OCONV function fails and the program terminates with a run-time
error message.
The OCONV function also allows PICK flavor exit codes.
The STATUS function reflects the result of the conversion:
For information about converting strings to an internal format, see the ICONV
function.
Examples
The following examples show date conversions:
DATE=OCONV(9166,'D/E') 3/2/1993
DATE=OCONV(9166,'DI')1 3/2/1993
DATE=OCONV('9166',"D2-") 2-3-93
TIME=OCONV("10000","MTHS") 02:46:40am
TIME=OCONV(10000,"MTH") 02:46am
TIME=OCONV(10000,"MT.") 02.46
TIME=OCONV(10000,"MTS") 02:46:40
HEX=OCONV('CDE',"MX0C") 434445
OCT=OCONV(1024,"MO") 2000
OCT=OCONV('CDE',"MO0C") 103104105
BIN=OCONV(1024,"MB") 10000000000
X=OCONV(987654,"MD0") 987654
X=OCONV(987654,"MD2,$") $9,876.54
X=OCONV(987654,"MD24$") $98.77
X=OCONV(987654,"MD2-Z") 9876.54
X=OCONV(987654,"MD2,D") 9,876.54
X=OCONV(987654,"MD3,$CPZ") $987.654
X=OCONV(987654,"MD2,ZP12#") ####9,876.54
OCONVS
Syntax
    OCONVS (dynamic.array, conversion)
    CALL −OCONVS (return.array, dynamic.array, conversion)
    CALL !OCONVS (return.array, dynamic.array, conversion)
Description
Use the OCONVS function to convert the elements of dynamic.array to a specified
format for external output.
The elements are converted to the external output format specified by conversion
and returned in a dynamic array. conversion must evaluate to one or more conver-
sion codes separated by value marks (ASCII 253).
If multiple codes are used, they are applied from left to right as follows: the left-
most conversion code is applied to the element, the next conversion code to the
right is then applied to the result of the first conversion, and so on.
If dynamic.array evaluates to the null value, null is returned. If any element of
dynamic.array is null, null is returned for that element. If conversion evaluates to the
null value, the OCONVS function fails and the program terminates with a run-
time error message.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
The STATUS function reflects the result of the conversion:
ON ON GOSUB ON GOTO
Syntax
     ON expression GOSUB statement.label [ : ] [,statement.label [ : ] … ]
Description
Use the ON statement to transfer program control to one of the internal subrou-
tines named in the GOSUB clause or to one of the statements named in the GOTO
clause.
clause, execution continues with the next statement rather than the last subroutine.
To get this characteristic in other flavors, use the ONGO.RANGE option of the
$OPTIONS statement.
Examples
 VAR=1234                           AT   LABEL   20
 Y=1                                AT   LABEL   30
 10*                                AT   LABEL   40
 X=VAR[Y,1]                         AT   LABEL   40
 IF X='' THEN STOP
 ON X GOTO 20,30,40
 20*
 PRINT 'AT LABEL 20'
 Y=Y+1
 GOTO 10
 30*
 PRINT 'AT LABEL 30'
 Y=Y+1
 GOTO 10
 40*
 PRINT 'AT LABEL 40'
 Y=Y+1
 GOTO 10
OPEN
Syntax
    OPEN [dict,] filename [TO file.variable] [ON ERROR statements]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the OPEN statement to open a DataStage file for use by BASIC programs. All
file references in a BASIC program must be preceded by either an OPEN statement
or an OPENCHECK statement for that file. You can open several DataStage files at
the same point in a program, but you must use a separate OPEN statement for each
file.
dict is an expression that evaluates to a string specifying whether to open the file
dictionary or the data file. Use the string DICT to open the file dictionary, or use
PDICT to open an associated Pick-style dictionary. Any other string opens the data
file. By convention an empty string or the string DATA is used when you are
opening the data file. If the dict expression is omitted, the data file is opened. If dict
is the null value, the data file is opened.
filename is an expression that evaluates to the name of the file to be opened. If the
file exists, the file is opened, and the THEN statements are executed; the ELSE
statements are ignored. If no THEN statements are specified, program execution
continues with the next statement. If the file cannot be accessed or does not exist,
the ELSE statements are executed; any THEN statements are ignored. If filename
evaluates to the null value, the OPEN statement fails and the program terminates
with a run-time error message.
Use the TO clause to assign the opened file to file.variable. All statements that read,
write to, delete, or clear that file must refer to it by the name of the file variable. If
you do not assign the file to a file variable, an internal default file variable is used.
File references that do not specify a file variable access the default file variable,
which contains the most recently opened file. The file opened to the current default
file variable is assigned to the system variable @STDFIL.
Default file variables are not local to the program from which they are executed.
When a subroutine is called, the current default file variable is shared with the
calling program.
When opening an SQL table, the OPEN statement enforces SQL security. The
permissions granted to the program’s effective user ID are loaded when the file is
opened. If no permissions have been granted, the OPEN statement fails, and the
ELSE statements are executed.
All writes to an SQL table opened with the OPEN statement are subject to SQL
integrity checking unless the OPENCHK configurable parameter has been set to
FALSE. Use the OPENCHECK statement instead of the OPEN statement to enable
automatic integrity checking for all writes to a file, regardless of whether the
OPENCHK configurable parameter is true or false.
Use the INMAT function after an OPEN statement to determine the modulo of the
file.
Value         Description
–1            Filename not found in the VOC file.
     1
–2            Null filename or file.
–3            Operating system access error that occurs when you do not have
              permission to access a DataStage file in a directory. For example,
              this may occur when trying to access a type 1 or type 30 file.
–41           Access error when you do not have operating system permissions
              or if DATA.30 is missing for a type 30 file.
–5            Read error detected by the operating system.
–6            Unable to lock file header.
–7            Invalid file revision or wrong byte-ordering for the platform.
     1
–8            Invalid part file information.
–91           Invalid type 30 file information in a distributed file.
–10           A problem occurred while the file was being rolled forward during
              warmstart recovery. Therefore, the file is marked “inconsistent.”
–11           The file is a view, therefore it cannot be opened by a BASIC
              program.
–12           No SQL privileges to open the table.
         1
–13           Index problem.
–14           Cannot open the NFS file.
  1. A generic error that can occur for various reasons.
Examples
     OPEN "SUN.MEMBER" TO DATA ELSE STOP "CAN'T OPEN SUN.MEMBER"
     OPEN "FOOBAR" TO FOO ELSE STOP "CAN'T OPEN FOOBAR"
     PRINT "ALL FILES OPEN OK"
This is the program output:
     CAN'T OPEN FOOBAR
The following example opens the same file as in the previous example. The OPEN
statement includes an empty string for the dict argument.
   OPEN "","SUN.MEMBER" TO DATA ELSE STOP "CAN'T OPEN SUN.MEMBER"
   OPEN "","FOO.BAR" TO FOO ELSE STOP "CAN'T OPEN FOOBAR"
   PRINT "ALL FILES OPEN OK"
OPENCHECK
Syntax
     OPENCHECK [dict,] filename [TO file.variable]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the OPENCHECK statement to open an SQL table for use by BASIC programs,
enforcing SQL integrity checking. All file references in a BASIC program must be
preceded by either an OPENCHECK statement or an OPEN statement for that file.
The OPENCHECK statement works like the OPEN statement, except that SQL
integrity checking is enabled if the file is an SQL table. All field integrity checks for
an SQL table are stored in the security and integrity constraints area (SICA). The
OPENCHECK statement loads the compiled form of these integrity checks into
memory, associating them with the file variable. All writes to the file are subject to
SQL integrity checking.
Value         Description
–1            Filename not found in the VOC file.
     1
–2            Null filename or file.
–3            Operating system access error that occurs when you do not have
              permission to access a DataStage file in a directory. For example,
              this may occur when trying to access a type 1 or type 30 file.
–41           Access error when you do not have operating system permissions
              or if DATA.30 is missing for a type 30 file.
–5            Read error detected by the operating system.
–6            Unable to lock file header.
–7            Invalid file revision or wrong byte-ordering for the platform.
     1
–8            Invalid part file information.
     1
–9            Invalid type 30 file information in a distributed file.
Value         Description
–10           A problem occurred while the file was being rolled forward during
              warmstart recovery. Therefore, the file is marked “inconsistent.”
–11           The file is a view, therefore it cannot be opened by a BASIC
              program.
–12           No SQL privileges to open the table.
      1
–13           Index problem.
–14           Cannot open the NFS file.
 1. A generic error that can occur for various reasons.
OPENDEV
Syntax
    OPENDEV device TO file.variable [LOCKED statements]
            {THEN statements [ELSE statements] | ELSE statements}
Description
Use the OPENDEV statement to open a device for sequential processing.
OPENDEV also sets a record lock on the opened device or file. See the READSEQ
and WRITESEQ statements for more details on sequential processing.
device is an expression that evaluates to the record ID of a device definition record
in the &DEVICE& file. If device evaluates to the null value, the OPENDEV state-
ment fails and the program terminates with a run-time error message. For more
information, see “Devices on Windows NT.”
The TO clause assigns a file.variable to the device being opened. All statements used
to read to or write from that device must refer to it by the assigned file.variable.
If the device exists and is not locked, the device is opened and any THEN state-
ments are executed; the ELSE statements are ignored. If no THEN statements are
specified, program execution continues with the next statement.
If the device is locked, the LOCKED statements are executed; THEN statements
and ELSE statements are ignored.
If the device does not exist or cannot be opened, the ELSE statements are executed;
any THEN statements are ignored. The device must have the proper access permis-
sions for you to open it.
If NLS is enabled, you can use OPENDEV to open a device that uses a map defined
in the &DEVICE& file. If there is no map defined in the &DEVICE& file, the default
mapname is the name in the NLSDEFDEVMAP parameter in the uvconfig file. For
more information about maps, see the DataStage NLS Guide.
Devices on Windows NT
On Windows NT systems, you may need to change to block size defined for a
device in the &DEVICE& file before you can use OPENDEV to reference the
device. On some devices there are limits to the type of sequential processing that
is available once you open the device. The following table summarizes the limits:
Example
The following example opens TTY30 for sequential input and output operations:
    OPENDEV 'TTY30' TO TERM THEN PRINT 'TTY30 OPENED'
    ELSE ABORT
This is the program output:
    TTY30 OPENED
OPENPATH
Syntax
    OPENPATH pathname [TO file.variable] [ON ERROR statements]
             {THEN statements [ELSE statements] | ELSE statements}
Description
The OPENPATH statement is similar to the OPEN statement, except that the path-
name of the file is specified. This file is opened without reference to the VOC file.
The file must be a hashed DataStage file or a directory (types 1 and 19).
pathname specifies the relative or absolute pathname of the file to be opened. If the
file exists, it is opened and the THEN statements are executed; the ELSE statements
are ignored. If pathname evaluates to the null value, the OPENPATH statement fails
and the program terminates with a run-time error message.
If the file cannot be accessed or does not exist, the ELSE statements are executed;
any THEN statements are ignored.
Use the TO clause to assign the file to a file.variable. All statements used to read,
write, delete, or clear that file must refer to it by the assigned file.variable name. If
you do not assign the file to a file.variable, an internal default file variable is used.
File references that do not specify file.variable access the most recently opened
default file. The file opened to the default file variable is assigned to the system
variable @STDFIL.
 Value         Description
–1             Filename not found in the VOC file.
     1
–2             Null filename or file.
–3             Operating system access error that occurs when you do not have
               permission to access a DataStage file in a directory. For example,
               this may occur when trying to access a type 1 or type 30 file.
–41            Access error when you do not have operating system permissions
               or if DATA.30 is missing for a type 30 file.
–5             Read error detected by the operating system.
–6             Unable to lock file header.
–7             Invalid file revision or wrong byte-ordering for the platform.
–81            Invalid part file information.
–91            Invalid type 30 file information in a distributed file.
–10            A problem occurred while the file was being rolled forward during
               warmstart recovery. Therefore, the file is marked “inconsistent.”
–11            The file is a view, therefore it cannot be opened by a BASIC
               program.
–12            No SQL privileges to open the table.
 Value       Description
      1
–13          Index problem.
–14          Cannot open the NFS file.
 1. A generic error that can occur for various reasons.
Example
The following example opens the file SUN.MEMBER. The pathname specifies the
file.
   OPENPATH '/user/members/SUN.MEMBER' ELSE ABORT
OPENSEQ
Syntax
    OPENSEQ filename, record.ID TO file.variable [USING dynamic.array]
             [ON ERROR statements] [LOCKED statements]
             {THEN statements [ELSE statements] | ELSE statements}
    OPENSEQ pathname TO file.variable [USING dynamic.array]
             [ON ERROR statements] [LOCKED statements]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the OPENSEQ statement to open a file for sequential processing. All sequen-
tial file references in a BASIC program must be preceded by an OPENSEQ or
OPENDEV statement for that file. Although you can open several files for sequen-
tial processing at the same point in the program, you must issue a separate
OPENSEQ statement for each. See the READSEQ and WRITESEQ statements for
more details on sequential processing.
ELSE statements are ignored. If the pathname does not exist, the ELSE statements
are executed; any THEN statements are ignored.
If the file does not exist, the OPENSEQ statement fails. The file can also be explic-
itly created with the CREATE statement.
OPENSEQ sets an update record lock on the specified record or file. This lock is
reset by a CLOSESEQ statement. This prevents any other program from changing
the record while you are processing it.
If filename, record.ID, or pathname evaluate to the null value, the OPENSEQ state-
ment fails and the program terminates with a run-time error message.
The TO clause is required. It assigns the record, file, or device to file.variable. All
statements used to sequentially read, write, delete, or clear that file must refer to it
by the assigned file variable name.
If NLS is enabled, you can use the OPENSEQ filename, record.ID statement to open
a type 1 or type 19 file that uses a map defined in the .uvnlsmap file in the directory
containing the type 1 or type 19 file. If there is no .uvnlsmap file in the directory, the
default mapname is the name in the NLSDEFDIRMAP parameter in the uvconfig
file.
Use the OPENSEQ pathname statement to open a UNIX pipe, file, or a file specified
by a device that uses a map defined in the .uvnlsmap file in the directory holding
pathname. If there is no .uvnlsmap file in the directory, the default mapname is the
name in the NLSDEFSEQMAP parameter in the uvconfig file, or you can use the
SET.SEQ.MAP command to assign a map.
For more information about maps, see the DataStage NLS Guide.
File Buffering
Normally DataStage uses buffering for sequential input and output operations.
Use the NOBUF statement after an OPENSEQ statement to turn off buffering and
cause all writes to the file to be performed immediately. For more information
about file buffering, see the NOBUF statement.
record. Field 17 of the dynamic array controls inclusion in the rotating file pool
with the following values:
    • Y removes the opened file.
    • N includes the opened file.
Value         Description
–1            Filename not found in the VOC file.
–21           Null filename or file.
–3            Operating system access error that occurs when you do not have
              privileges to access a DataStage file in a directory. For example, this
              may occur when trying to access a type 1 or type 30 file.
–41           Access error when you do not have operating system permissions or
              if DATA.30 is missing for a type 30 file.
–5            Read error detected by the operating system.
–6            Unable to lock file header.
–7            Invalid file revision or wrong byte-ordering for the platform.
     1
–8            Invalid part file information.
–91           Invalid type 30 file information in a distributed file.
–10           A problem occurred while the file was being rolled forward during
              warmstart recovery. Therefore, the file is marked “inconsistent.”
–11           The file is a view, therefore it cannot be opened by a BASIC program.
–12           No SQL privileges to open the table.
–131          Index problem.
–14           Cannot open the NFS file.
Examples
The following example reads RECORD1 from the nonhashed file FILE.E:
    OPENSEQ 'FILE.E', 'RECORD1' TO FILE THEN
       PRINT "'FILE.E' OPENED FOR PROCESSING"
       END ELSE ABORT
    READSEQ A FROM FILE THEN PRINT A ELSE STOP
The next example writes the record read from FILE.E to the file /usr/depta/file1:
    OPENSEQ '/usr/depta/file1' TO OUTPUT THEN
        PRINT "usr/depta/file1 OPENED FOR PROCESSING"
    END ELSE ABORT
    WRITESEQ A ON OUTPUT ELSE PRINT "CANNOT WRITE TO OUTPUT"
        .
        .
        .
    CLOSESEQ FILE
    CLOSESEQ OUTPUT
    END
This is the program output:
    FILE.E OPENED FOR PROCESSING
    HI THERE
       .
       .
       .
    /usr/depta/file1 OPENED FOR PROCESSING
The next example includes the USING clause to remove an opened file from the
rotating file pool:
    DEVREC = "1"@FM
    FOR I = 2 TO 16
       DEVREC = DEVREC:I:@FM
    NEXT I
    DEVREC=DEVREC:'Y'
    *
    OPENSEQ 'SEQTEST', 'TESTDATA' TO TESTFILE USING DEVREC
    THEN PRINT "OPENED 'TESTDATA' OK...."
    ELSE PRINT "COULD NOT OPEN TESTDATA"
    CLOSESEQ TESTFILE
ORS
Syntax
      ORS (array1, array2)
      CALL −ORS (return.array, array1, array2)
      CALL !ORS (return.array, array1, array2)
Description
Use the ORS function to create a dynamic array of the logical OR of corresponding
elements of two dynamic arrays.
Each element of the new dynamic array is the logical OR of the corresponding
elements of array1 and array2. If an element of one dynamic array has no corre-
sponding element in the other dynamic array, a false is assumed for the missing
element.
If both corresponding elements of array1 and array2 are the null value, null is
returned for those elements. If one element is the null value and the other is 0 or an
empty string, null is returned. If one element is the null value and the other is any
value other than 0 or an empty string, a true is returned.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
      A="A":@SM:0:@VM:4:@SM:1
      B=0:@SM:1-1:@VM:2
      PRINT ORS(A,B)
This is the program output:
      1S0V1S1
PAGE
Syntax
    PAGE [ ON print.channel ] [ page# ]
Description
Use the PAGE statement to print headings, footings, and page advances at the
appropriate places on the specified output device. You can specify headings and
footings before execution of the PAGE statement (see the HEADING and
FOOTING statements). If there is no heading or footing, PAGE clears the screen.
The ON clause specifies the logical print channel to use for output. print.channel is
an expression that evaluates to a number from –1 through 255. If you do not use
the ON clause, logical print channel 0 is used, which prints to the user’s terminal
if PRINTER OFF is set (see the PRINTER statement). Logical print channel –1
prints the data on the screen, regardless of whether a PRINTER ON statement has
been executed.
page# is an expression that specifies the next page number. If a heading or footing
is in effect when the page number is specified, the heading or footing on the current
page contains a page number equal to one less than the value of page#.
If either print.channel or page# evaluates to the null value, the PAGE statement fails
and the program terminates with a run-time error message.
Example
In the following example the current value of X provides the next page number:
    PAGE ON 5 X
PERFORM
Syntax
    PERFORM command
Description
Use the PERFORM statement to execute a DataStage sentence, paragraph, menu,
or command from within the BASIC program, then return execution to the state-
ment following the PERFORM statement. The commands are executed in the same
environment as the BASIC program that called them; that is, unnamed common
variables, @variables, and in-line prompts retain their values, and select lists and
the DATA stack remain active. If these values change, the new values are passed
back to the calling program.
You can specify multiple commands in the PERFORM statement in the same way
you specify them in the body of a DataStage paragraph. Each command or line
must be separated by a field mark (ASCII CHAR(254)).
If command evaluates to the null value, the PERFORM statement fails and the
program terminates with a run-time error message.
You cannot use the PERFORM statement within a transaction to execute most
DataStage commands and SQL statements. However, you can use PERFORM to
execute the following DataStageDataStage commands and SQL statements within
a transaction:
CHECK.SUM              INSERT                 SEARCH                 SSELECT
COUNT                  LIST                   SELECT (RetrieVe)      STAT
DELETE (SQL)           LIST.ITEM              SELECT (SQL)           SUM
DISPLAY                LIST.LABEL             SORT                   UPDATE
ESEARCH                RUN                    SORT.ITEM
GET.LIST               SAVE.LIST              SORT.LABEL
REALITY Flavor
In a REALITY flavor account PERFORM can take all the clauses of the EXECUTE
statement. To get these PERFORM characteristics in other flavor accounts, use the
PERF.EQ.EXEC option of the $OPTIONS statement.
Example
In the following example multiple commands are separated by field marks:
   PERFORM 'RUN BP SUB'
   FM=CHAR(254)
   COMMAND = 'SSELECT EM':FM
   COMMAND := 'RUN BP PAY':FM
   COMMAND := 'DATA 01/10/85'
   PERFORM COMMAND
   A = 'SORT EM '
   A := 'WITH PAY.CODE EQ'
   A := '10 AND WITH DEPT'
   A := 'EQ 45'
   PERFORM A
PRECISION
Syntax
    PRECISION expression
Description
Use the PRECISION statement to control the maximum number of decimal places
that are output when the system converts a numeric value from internal binary
format to an ASCII character string value.
expression specifies a number from 0 through 9. Any fractional digits in the result
of such a conversion that exceed the precision setting are rounded off.
If you do not include a PRECISION statement, a default precision of 4 is assumed.
Precisions are stacked so that a BASIC program can change its precision and call a
subroutine whose precision is the default unless the subroutine executes a PRECI-
SION statement. When the subroutine returns to the calling program, the calling
program has the same precision it had when it called the subroutine.
Trailing fractional zeros are dropped during output. Therefore, when an internal
number is converted to an ASCII string, the result might appear to have fewer
decimal places than the precision setting allows. However, regardless of the preci-
sion setting, the calculation always reflects the maximum accuracy of which the
computer is capable (that is, slightly more than 17 total digits, including integers).
If expression evaluates to the null value, the PRECISION statement fails and the
program terminates with a run-time error message.
Example
    A = 12.123456789
    PRECISION 8
    PRINT A
    PRECISION 4
    PRINT A
This is the program output:
    12.12345679
    12.1235
PRINT
Syntax
    PRINT [ON print.channel] [print.list]
Description
Use the PRINT statement to send data to the screen, a line printer, or another print
file.
The ON clause specifies the logical print channel to use for output. print.channel is
an expression that evaluates to a number from –1 through 255. If you do not use
the ON clause, logical print channel 0 is used, which prints to the user’s terminal
if PRINTER OFF is set (see the PRINTER statement). If print.channel evaluates to
the null value, the PRINT statement fails and the program terminates with a run-
time error message. Logical print channel –1 prints the data on the screen, regard-
less of whether a PRINTER ON statement has been executed.
You can specifyHEADING, FOOTING, PAGE, and PRINTER CLOSE statements
for each logical print channel. The contents of the print files are printed in order by
logical print channel number.
print.list can contain any BASIC expression. The elements of the list can be numeric
or character strings, variables, constants, or literal strings; the null value, however,
cannot be printed. The list can consist of a single expression or a series of expres-
sions separated by commas ( , ) or colons ( : ) for output formatting. If no print.list
is designated, a blank line is printed.
Expressions separated by commas are printed at preset tab positions. The default
tabstop setting is 10 characters. Calculations for tab characters are based on char-
acter length rather than display length. For information about changing the default
setting, see the TABSTOP statement. Use multiple commas together for multiple
tabulations between expressions.
Expressions separated by colons are concatenated. That is, the expression
following the colon is printed immediately after the expression preceding the
colon. To print a list without a LINEFEED and RETURN, end print.list with a colon
( : ).
If NLS is enabled, calculations for the PRINT statement are based on character
length rather than display length. If print.channel has a map associated with it, data
is mapped before it is output to the device. For more information about maps, see
the DataStage NLS Guide.
Examples
    A=25;B=30
    C="ABCDE"
    PRINT A+B
    PRINT
    PRINT "ALPHA ":C
    PRINT "DATE ":PRINT "10/11/93"
    *
    PRINT ON 1 "FILE 1"
    * The string "FILE 1" is printed on print file 1.
This is the program output:
    55
    ALPHA ABCDE
    DATE 10/11/93
The following example clears the screen:
    PRINT @(–1)
The following example prints the letter X at location column 10, row 5:
    PRINT @(10,5) 'X'
PRINTER
Syntax
    PRINTER { ON | OFF | RESET }
PRINTER CLOSE
    PRINTER CLOSE [ON print.channel]
Description
Use the PRINTER statement to direct output either to the screen or to a printer. By
default, all output is sent to the screen unless a PRINTER ON is executed or the P
option to the RUN command is used. See the SETPTR command for more details
about redirecting output.
PRINTER ON sends output to the system line printer via print channel 0. The
output is stored in a buffer until a PRINTER CLOSE statement is executed or the
program terminates; the output is then printed (see the PRINTER CLOSE
statement).
PRINTER OFF sends output to the screen via print channel 0. When the program
is executed, the data is immediately printed on the screen.
The PRINTER ON or PRINTER OFF statement must precede the PRINT statement
that starts the print file.
Use the PRINTER RESET statement to reset the printing options. PRINTER RESET
removes the header and footer, resets the page count to 1, resets the line count to 1,
and restarts page waiting.
Note: Use TPRINT to set a delay before printing. See also the TPARM statement.
If print.channel evaluates to the null value, the PRINTER CLOSE statement fails
and the program terminates with a run-time error message.
In PICK, IN2, and REALITY flavor accounts, the PRINTER CLOSE statement
closes all print channels.
Example
    PRINTER ON
    PRINT "OUTPUT IS PRINTED ON PRINT FILE 0"
    PRINTER OFF
    PRINT "OUTPUT IS PRINTED ON THE TERMINAL"
    *
    PRINT ON 1 "OUTPUT WILL BE PRINTED ON PRINT FILE 1"
    PRINT ON 2 "OUTPUT WILL BE PRINTED ON PRINT FILE 2"
This is the program output:
    OUTPUT IS PRINTED ON THE TERMINAL
PRINTERR
Syntax
    PRINTERR [error.message]
Description
Use the PRINTERR statement to print a formatted error message on the bottom
line of the terminal. The message is cleared by the next INPUT @ statement or is
overwritten by the next PRINTERR or INPUTERR statement. PRINTERR clears
the type-ahead buffer.
error.message is an expression that evaluates to the error message text. The elements
of the expression can be numeric or character strings, variables, constants, or literal
strings. The null value cannot be an element because it cannot be output. The
expression can be a single expression or a series of expressions separated by
commas ( , ) or colons ( : ) for output formatting. If no error message is designated,
a blank line is printed. If error.message evaluates to the null value, the default
message is printed:
    Message ID is NULL: undefined error
Expressions separated by commas are printed at preset tab positions. The default
tabstop setting is 10 characters. For information about changing the default setting,
see the TABSTOP statement. Use multiple commas together to cause multiple
tabulations between expressions.
Expressions separated by colons are concatenated: that is, the expression following
the colon is printed immediately after the expression preceding the colon.
See also the INPUTERR statement.
REALITY Flavor
In a REALITY flavor account the PRINTERR statement prints a formatted error
message from the ERRMSG file on the bottom line of the terminal. REALITY
syntax is:
If dynamic.array evaluates to the null value, the default error message is printed:
    Message ID is NULL: undefined error
The FROM clause lets you read the error message from an open file. If file.variable
evaluates to the null value, the PRINTERR statement fails and the program termi-
nates with a run-time error message.
This statement is similar to the STOP statement on a Pick system except that it does
not terminate the program upon execution. You can use it wherever you can use a
STOP or ABORT statement.
To use the REALITY version of the PRINTERR statement in PICK, IN2, INFOR-
MATION, and IDEAL flavor accounts, use the USE.ERRMSG option of the
$OPTIONS statement.
DataStage provides a standard Pick ERRMSG file. You can construct a local
ERRMSG file using the following syntax in the records. Each field must start with
one of these codes, as shown in the following table:
Code          Action
A[(n)]        Display next argument left-justified; n specifies field length.
D             Display system date.
E [string]    Display record ID of message in brackets; string displayed after ID.
H [string]    Display string.
L [(n)]       Output newline; n specifies number of newlines.
R [(n)]       Display next argument right-justified; n specifies field length.
S [(n)]       Output n blank spaces from beginning of line.
T             Display system time.
PROCREAD
Syntax
    PROCREAD variable
         {THEN statements [ELSE statements] | ELSE statements}
Description
Use the PROCREAD statement to assign the contents of the primary input buffer
to a variable. Your BASIC program must be called by a proc. If your program was
not called from a proc, the ELSE statements are executed; otherwise the THEN
statements are executed.
If variable evaluates to the null value, the PROCREAD statement fails and the
program terminates with a run-time error message.
PROCWRITE
Syntax
    PROCWRITE string
Description
Use the PROCWRITE statement to write string to the primary input buffer. Your
program must be called by a proc.
If string evaluates to the null value, the PROCWRITE statement fails and the
program terminates with a run-time error message.
PROGRAM
Syntax
    PROG[RAM] [name]
Description
Use the PROGRAM statement to identify a program. The PROGRAM statement is
optional; if you use it, it must be the first noncomment line in the program.
name can be specified for documentation purposes; it need not be the same as the
actual program name.
Example
    PROGRAM BYSTATE
PROMPT
Syntax
    PROMPT character
Description
Use the PROMPT statement to specify the character to be displayed on the screen
when user input is required. If no PROMPT statement is issued, the default prompt
character is the question mark ( ? ).
If character evaluates to more than one character, only the first character is signifi-
cant; all others are ignored.
The prompt character becomes character when the PROMPT statement is executed.
Although the value of character can change throughout the program, the prompt
character remains the same until a new PROMPT statement is issued or the
program ends.
Generally, data the user enters in response to the prompt appears on the screen. If
the source of the input is something other than the keyboard (for example, a DATA
statement), the data is displayed on the screen after the prompt character. Use
PROMPT " " to prevent any prompt from being displayed. PROMPT " " also
suppresses the display of input from DATA statements.
If character evaluates to the null value, no prompt appears.
Examples
PWR
Syntax
      PWR (expression, power)
Description
Use the PWR function to return the value of expression raised to the power speci-
fied by power.
The PWR function operates like exponentiation (that is, PWR(X,Y) is the same as
X**Y).
A negative value cannot be raised to a noninteger power. If it is, the result of the
function is PWR(−X,Y) and an error message is displayed.
If either expression or power is the null value, null is returned.
On overflow or underflow, a warning is printed and 0 is returned.
Example
      A=3
      B=PWR(5,A)
      PRINT "B= ",B
This is the program output:
      B=     125
QUOTE
Syntax
    QUOTE (expression)
Description
Use the QUOTE function to enclose an expression in double quotation marks. If
expression evaluates to the null value, null is returned, without quotation marks.
Example
    PRINT QUOTE(12 + 5) : " IS THE ANSWER."
    END
This is the program output:
    "17" IS THE ANSWER.
RAISE
Syntax
     RAISE (expression)
Description
Use the RAISE function to return a value equal to expression, except that system
delimiters in expression are converted to the next higher-level delimiter: value
marks are changed to field marks, subvalue marks are changed to value marks,
and so on. If expression evaluates to the null value, null is returned.
The conversions are:
IM       CHAR(255)          to    IM      CHAR(255)
FM       CHAR(254)          to    IM      CHAR(255)
VM       CHAR(253)          to    FM      CHAR(254)
SM       CHAR(252)          to    VM      CHAR(253)
TM       CHAR(251)          to    SM      CHAR(252)
         CHAR(250)          to            CHAR(251)
         CHAR(249)          to            CHAR(250)
         CHAR(248)          to            CHAR(249)
PIOPEN Flavor
In PIOPEN flavor, the delimiters that can be raised are CHAR(255) through
CHAR(252). All other characters are left unchanged. You can obtain PIOPEN
flavor for the RAISE function by:
     • Compiling your program in a PIOPEN flavor account
     • Specifying the $OPTIONS INFO.MARKS statement
Examples
In the following examples an item mark is shown by I, a field mark is shown by F,
a value mark is shown by V, and a subvalue mark is shown by S.
The following example sets A to DDIEEI123I777:
     A= RAISE('DD':FM'EE':FM:123:FM:777)
RANDOMIZESyntax
    RANDOMIZE (expression)
Description
Use the RANDOMIZE statement with an expression to make the RND function
generate the same sequence of random numbers each time the program is run. If
no expression is supplied, or if expression evaluates to the null value, the internal
time of day is used (the null value is ignored). In these cases the sequence is
different each time the program is run.
Example
    RANDOMIZE (0)
    FOR N=1 TO 10
       PRINT RND(4):' ':
    NEXT N
    PRINT
    *
    RANDOMIZE (0)
    FOR N=1 TO 10
       PRINT RND(4):' ':
    NEXT
    PRINT
    *
    RANDOMIZE (3)
    FOR N=1 TO 10
       PRINT RND(4):' ':
    NEXT N
    PRINT
This is the program output:
    0 2 1 2 0 2 1 2 1 1
    0 2 1 2 0 2 1 2 1 1
    2 0 1 1 2 1 0 1 2 3
Syntax
Description
Use READ statements to assign the contents of a record from a DataStage file to
dynamic.array.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information on default files, see the OPEN statement). If the file
is neither accessible nor open, the program terminates with a run-time error
message.
If record.ID exists on the specified file, dynamic.array is set to the contents of the
record, and the THEN statements are executed; any ELSE statements are ignored.
If no THEN statements are specified, program execution continues with the next
statement. If record.ID does not exist, dynamic.array is set to an empty string, and
the ELSE statements are executed; any THEN statements are ignored.
If file.variable, record.ID, or field# evaluate to the null value, the READ statement
fails and the program terminates with a run-time error message.
Tables. If the file is a table, the effective user of the program must have SQL
SELECT privilege to read records in the file. For information about the effective
user of a program, see the AUTHORIZATION statement.
Distributed Files. If the file is a distributed file, use the STATUS function after a
READ statement to determine the results of the operation, as follows:
NLS Mode. If NLS is enabled, READ and other BASIC statements that perform
I/O operations map external data to the internal character set using the appro-
priate map for the input file.
If the file contains unmappable characters, the ELSE statements are executed.
The results of the READ statements depend on all of the following:
      • The inclusion of the ON ERROR clause
      • The setting of the NLSREADELSE parameter in the uvconfig file
      • The location of the unmappable character
The values returned by the STATUS function are as follows:
3     The unmappable character is in the record ID.
4     The unmappable character is in the record’s data.1
For more information about maps, see the DataStage NLS Guide.
If a READ statement does not include a LOCKED clause, and a conflicting lock
exists, the program pauses until the lock is released.
If a LOCKED clause is used, the value returned by the STATUS function is the
terminal number of the user who owns the conflicting lock.
Releasing Locks. A shared record lock can be released with a CLOSE, RELEASE,
or STOP statement. An update record lock can be released with a CLOSE, DELETE,
MATWRITE, RELEASE, STOP, WRITE, or WRITEV statement.
Locks acquired or promoted within a transaction are not released when the
previous statements are processed.
All record locks are released when you return to the command prompt.
Use the READVL statement to acquire a shared record lock and then read a field
from the record. The READVL statement conforms to all the specifications of the
READL and READV statements.
Use the READVU statement to acquire an update record lock and then read a field
from the record. The READVU statement conforms to all the specifications of the
READU and READV statements.
You can specify field# only with the READV, READVL, and READVU statements.
It specifies the index number of the field to be read from the record. You can use a
field# of 0 to determine whether the record exists. If the field does not exist,
dynamic.array is assigned the value of an empty string.
Examples
    OPEN '','SUN.MEMBER' TO FILE ELSE STOP
    FOR ID=5000 TO 6000
       READ MEMBER FROM FILE, ID THEN PRINT ID ELSE NULL
    NEXT ID
    OPEN '','SUN.SPORT' ELSE STOP 'CANT OPEN "SUN.SPORT"'
    READ ID FROM "853333" ELSE
        PRINT 'CANT READ ID "853333" ON FILE "SUN.SPORT"'
    END
    X="6100"
    READ PERSON FROM FILE,X THEN PRINT PERSON<1> ELSE
        PRINT "PERSON ":X:" NOT ON FILE"
    END
The next example locks the record N in the file SUN.MEMBER, reads field 3
(STREET) from it, and prints the value of the field:
    OPEN '','SUN.MEMBER' TO FILE ELSE STOP
    FOR N=5000 TO 6000
       READVU STREET FROM FILE,N,3 THEN PRINT STREET ELSE NULL
       RELEASE
    NEXT
    OPEN "DICT","MYFILE" TO DICT.FILE ELSE STOP
    OPEN "","MYFILE" ELSE STOP ; *USING DEFAULT FILE VARIABLE
    READU ID.ITEM FROM DICT.FILE,"@ID" ELSE
        PRINT "NO @ID"
        STOP
    END
This is the program output:
    5205
    5390
    CANT READ ID "853333" ON FILE "SUN.SPORT"
    MASTERS
    4646 TREMAIN DRIVE
    670 MAIN STREET
READBLK
Syntax
    READBLK variable FROM file.variable, blocksize
         { THEN statements [ ELSE statements ] | ELSE statements }
Description
Use the READBLK statement to read a block of data of a specified length from a
file opened for sequential processing and assign it to a variable. The READBLK
statement reads a block of data beginning at the current position in the file and
continuing for blocksize bytes and assigns it to variable. The current position is reset
to just beyond the last byte read.
file.variable specifies a file previously opened for sequential processing.
If the data can be read from the file, the THEN statements are executed; any ELSE
statements are ignored. If the file is not readable or if the end of file is encountered,
the ELSE statements are executed and the THEN statements are ignored. If the
ELSE statements are executed, variable is set to an empty string.
If either file.variable or blocksize evaluates to the null value, the READBLK statement
fails and the program terminates with a run-time error message.
Note: A newline in UNIX files is one byte long, whereas in Windows NT it is two
      bytes long. This means that for a file with newlines, the same READBLK
      statement may return a different set of data depending on the operating
      system the file is stored under.
In the event of a timeout, READBLK returns no bytes from the buffer, and the
entire I/O operation must be retried.
The difference between the READSEQ statement and the READBLK statement is
that the READBLK statement reads a block of data of a specified length, whereas
the READSEQ statement reads a single line of data.
On Windows NT systems, if you use READBLK to read data from a 1/4-inch
cartridge drive (60 or 150 MB) that you open with the OPENDEV statement, you
must use a block size of 512 bytes or a multiple of 512 bytes.
For more information about sequential file processing, see the OPENSEQ,
READSEQ, and WRITESEQ statements.
If NLS is enabled and file.variable has a map associated with it, the data is mapped
accordingly. For more information about maps, see the DataStage NLS Guide.
Example
    OPENSEQ 'FILE.E', 'RECORD4' TO FILE ELSE ABORT
    READBLK VAR1 FROM FILE, 15 THEN PRINT VAR1
    PRINT
    READBLK VAR2 FROM FILE, 15 THEN PRINT VAR2
This is the program output:
    FIRST LINE
    SECO
    ND LINE
    THIRD L
READL
Use the READL statement to acquire a shared record lock and perform the READ
statement. For details, see the READ statement.
READLIST
Syntax
    READLIST dynamic.array [FROM list.number]
             { THEN statements [ ELSE statements ] | ELSE statements }
Description
Use the READLIST statement to read the remainder of an active select list into a
dynamic array.
list.number is an expression that evaluates to the number of the select list to be read.
It can be from 0 through 10. If you do not use the FROM clause, select list 0 is used.
READLIST reads all elements in the active select list. If a READNEXT statement is
used on the select list before the READLIST statement, only the elements not read
by the READNEXT statement are stored in dynamic.array. READLIST empties the
select list.
If one or more elements are read from list.number, the THEN statements are
executed. If there are no more elements in the select list or if a select list is not
active, the ELSE statements are executed; any THEN statements are ignored.
If list.number evaluates to the null value, the READLIST statement fails and the
program terminates with run-time error message.
In IDEAL and INFORMATION flavor accounts, use the VAR.SELECT option of the
$OPTIONS statement to get READLIST to behave as it does in PICK flavor
accounts.
The select list saved in listname in the &SAVEDLISTS& file is put in dynamic.array.
The elements of the list are separated by field marks.
listname can be of the form
     record.ID
or
     record.ID account.name
record.ID specifies the record ID of the list in &SAVEDLISTS&, and account.name
specifies the name of another DataStage account in which to look for the
&SAVEDLISTS& file.
The SETTING clause assigns the count of the elements in the list to variable.
If the list is retrieved successfully (the list must not be empty), the THEN state-
ments are executed; if not, the ELSE statements are executed. If listname evaluates
to the null value, the READLIST statement fails and the program terminates with
a run-time error message.
In PICK, REALITY, and IN2 flavor accounts, use the −VAR.SELECT option of the
$OPTIONS statement to get READLIST to behave as it does in IDEAL flavor
accounts.
READNEXT
Syntax
    READNEXT dynamic.array [ ,value [ ,subvalue ]        ] [FROM list]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the READNEXT statement to assign the next record ID from an active select
list to dynamic.array.
list specifies the select list. If none is specified, select list 0 is used. list can be a
number from 0 through 10 indicating a numbered select list, or the name of a select
list variable.
The BASIC SELECT statement or the DataStage GET.LIST, FORM.LIST, SELECT, or
SSELECT commands create an active select list; these commands build the list of
record IDs. The READNEXT statement reads the next record ID on the list speci-
fied in the FROM clause and assigns it to the dynamic.array.
When the select list is exhausted, dynamic.array is set to an empty string, and the
ELSE statements are executed; any THEN statements are ignored.
If list evaluates to the null value, the READNEXT statement fails and the program
terminates with a run-time error message.
A READNEXT statement with value and subvalue specified accesses an exploded
select list. The record ID is stored in dynamic.array, the value number in value, and
the subvalue number in subvalue. If only dynamic.array is specified, it is set to a
multivalued field consisting of the record ID, value number, and subvalue number,
separated by value marks.
INFORMATION Flavor
In INFORMATION flavor accounts READNEXT returns an exploded select list.
Use the RNEXT.EXPL option of the $OPTIONS statement to return exploded select
lists in other flavors.
Example
    OPEN '','SUN.MEMBER' ELSE STOP "CAN'T OPEN FILE"
    SELECT TO 1
    10: READNEXT MEM FROM 1 THEN PRINT MEM ELSE GOTO 15:
    GOTO 10:
    *
    15: PRINT
    OPEN '','SUN.SPORT' TO FILE ELSE STOP
    SELECT FILE
    COUNT=0
    20*
    READNEXT ID ELSE
    PRINT 'COUNT= ',COUNT
    STOP
    END
    COUNT=COUNT+1
    GOTO 20
This is the program output:
    4108
    6100
    3452
    5390
    7100
    4500
    2430
    2342
    6783
    5205
    4439
    6203
    7505
    4309
    1111
    COUNT= 14
READSEQ
Syntax
    READSEQ variable FROM file.variable [ON ERROR statements]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the READSEQ statement to read a line of data from a file opened for sequential
processing. Sequential processing lets you process data one line at a time.
DataStage keeps a pointer at the current position in the file. The OPENSEQ state-
ment sets this pointer to the first byte of the file, and it is advanced by READSEQ,
READBLK, WRITESEQ, and WRITEBLK statements.
Each READSEQ statement reads data from the current position in the file up to a
newline and assigns it to variable. The pointer is then set to the position following
the newline. The newline is discarded.
file.variable specifies a file previously opened for sequential processing. The FROM
clause is required. If the file is neither accessible nor open, or if file.variable evalu-
ates to the null value, the READSEQ statement fails and the program terminates
with a run-time error message.
If data is read from the file, the THEN statements are executed, and the ELSE state-
ments are ignored. If the file is not readable, or the end of file is encountered, the
ELSE statements are executed; any THEN statements are ignored.
In the event of a timeout, READSEQ returns no bytes from the buffer, and the
entire I/O operation must be retried.
READSEQ affects the STATUS function in the following way:
If NLS is enabled, the READSEQ and other BASIC statements that perform I/O
operations always map external data to the internal character set using the appro-
priate map for the input file if the file has a map associated with it. For more
information about maps, see the DataStage NLS Guide.
Example
    OPENSEQ 'FILE.E', 'RECORD4' TO FILE ELSE ABORT
    FOR N=1 TO 3
       READSEQ A FROM FILE THEN PRINT A
    NEXT N
    CLOSESEQ FILE
This is the program output:
    FIRST LINE
    SECOND LINE
    THIRD LINE
READT
Syntax
    READT [UNIT (mtu)] variable
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the READT statement to read the next tape record from a magnetic tape unit
and assign its contents to a variable.
The UNIT clause specifies the number of the tape drive unit. Tape unit 0 is used if
no unit is specified.
mtu is an expression that evaluates to a code made up of three decimal digits, as
shown in the following table:
mtu Codes
The mtu expression is read from right to left. Therefore, if mtu evaluates to a one-
digit code, it represents the tape unit number. If mtu evaluates to a two-digit code,
the rightmost digit represents the unit number and the digit to its left is the track
number; and so on.
If either mtu or variable evaluates to the null value, the READT statement fails and
the program terminates with a run-time error message.
Each tape record is read and processed completely before the next record is read.
The program waits for the completion of data transfer from the tape before
continuing.
If the next tape record exists, variable is set to the contents of the record, and the
THEN statements are executed. If no THEN statements are specified, program
execution continues with the next statement.
PIOPEN Flavor
If you have a program that specifies the syntax UNIT ndmtu, the nd elements are
ignored by the compiler and no errors are reported.
Examples
The following example reads a tape record from tape drive 0:
    READT RECORD ELSE PRINT "COULD NOT READ FROM TAPE"
The next example reads a record from tape drive 3, doing an EBCDIC conversion
in the process:
    READT UNIT(103) RECORD ELSE PRINT "COULD NOT READ"
Use the READU statement to set an update record lock and perform the READ
statement. For details, see the READ statement.
Use the READV statement to read the contents of a specified field of a record in a
DataStage file. For details, see the READ statement.
READVL
Use the READVL statement to set a shared record lock and perform the READV
statement. For details, see the READ statement.
Use the READVU statement to set an update record lock and read the contents of
a specified field of a record in a DataStage file. For details, see the READ statement.
REAL
Syntax
   REAL (number)
Description
Use the REAL function to convert number into a floating-point number without
loss of accuracy. If number evaluates to the null value, null is returned.
RECORDLOCK
Syntax
    RECORDLOCKL file.variable, record.ID      [ ON ERROR statements ]
             [ LOCKED statements ]
    RECORDLOCKU file.variable, record.ID       [ ON ERROR statements ]
             [ LOCKED statements ]
Description
Use RECORDLOCK statements to acquire a record lock on a record without
reading the record.
                                                 Conflicts with
In this statement…          This requested lock… these locks…
RECORDLOCKL                 Shared record lock           Exclusive file lock
                                                         Update record lock
RECORDLOCKU                 Update record lock           Exclusive file lock
                                                         Intent file lock
                                                         Shared file lock
                                                         Update recordlock
                                                         Shared record lock
Releasing Locks
A shared record lock can be released with a CLOSE, RELEASE, or STOP statement.
An update record lock can be released with a CLOSE,DELETE, MATWRITE,
RELEASE, STOP, WRITE, or WRITEV statement.
Locks acquired or promoted within a transaction are not released when the
previous statements are processed.
All record locks are released when you return to the command prompt.
Example
In the following example, the file EMPLOYEES is opened. Record 23694 is locked.
If the record was already locked, the program terminates, and an appropriate
message is displayed. The RECORDLOCKL statement allows other users to read
the record with READL or lock it with another RECORDLOCKL, but prevents any
other user from gaining exclusive control over the record.
   OPEN '','EMPLOYEES' TO EMPLOYEES ELSE STOP 'Cannot open file'
   RECORDLOCKL EMPLOYEES,'23694'
    LOCKED STOP 'Record previously locked by user ':STATUS( )
RECORDLOCKED
Syntax
    RECORDLOCKED (file.variable, record.ID)
Description
Use the RECORDLOCKED function to return the status of a record lock.
file.variable is a file variable from a previous OPEN statement.
record.ID is an expression that evaluates to the record ID of the record that is to be
checked.
An insert file of equate names is provided to let you use mnemonics (see the
following table). The insert file is called RECORDLOCKED.INS.IBAS, and is
located in the INCLUDE directory in the DataStage engine directory. In PIOPEN
flavor accounts, the VOC file has a file pointer called SYSCOM. SYSCOM refer-
ences the INCLUDE directory in the DataStage engine directory.
To use the insert file, specify $INCLUDE SYSCOM RECORDLOCKED.INS.IBAS
when you compile the program.
If you have locked the file, the RECORDLOCKED function indicates only that you
have the file lock for that record. It does not indicate any update record or shared
record lock that you also have on the record.
>0      A positive value is the terminal number of the owner of the lock (or the first
        terminal number encountered, if more than one user has locked records in
        the specified file).
<0      A negative value is –1 times the terminal number of the remote user who
        has locked the record or file.
Examples
The following program checks to see if there is an update record lock or FILELOCK
held by the current user on the record. If the locks are not held by the user, the ELSE
clause reminds the user that an update record lock or FILELOCK is required on the
record. This example using the SYSCOM file pointer, only works in PI/open flavor
accounts.
     $INCLUDE SYSCOM RECORDLOCKED.INS.IBAS
     OPEN '','EMPLOYEES' TO EMPLOYEES
      ELSE STOP 'CANNOT OPEN FILE'
      .
      .
      .
     IF RECORDLOCKED(EMPLOYEES,RECORD.ID) >= LOCK$MY.READU THEN
      GOSUB PROCESS.THIS.RECORD:
     ELSE PRINT 'Cannot process record : ':RECORD.ID :', READU or
     FILELOCK required.'
The next program checks to see if the record lock is held by another user and prints
a message where the STATUS function gives the terminal number of the user who
holds the record lock:
     $INCLUDE SYSCOM RECORDLOCKED.INS.IBAS
     OPEN '','EMPLOYEES' TO EMPLOYEES
     ELSE STOP 'CANNOT OPEN FILE'
      .
      .
      .
     IF RECORDLOCKED(EMPLOYEES,RECORD.ID) < LOCK$NO.LOCK THEN
      PRINT 'Record locked by user' : STATUS()
     END
RELEASE
Syntax
    RELEASE [ file.variable [ ,record.ID ] ] [ ON ERROR statements ]
Description
Use the RELEASE statement to unlock, or release, locks set by a FILELOCK,
MATREADL, MATREADU, READL, READU, READVL, READVU, and
OPENSEQ statement. These statements lock designated records to prevent concur-
rent updating by other users. If you do not explicitly release a lock that you have
set, it is unlocked automatically when the program terminates.
file.variable specifies an open file. If file.variable is not specified and a record ID is
specified, the default file is assumed (for more information on default files, see the
OPEN statement). If the file is neither accessible nor open, the program terminates
with a run-time error message.
record.ID specifies the lock to be released. If it is not specified, all locks in the spec-
ified file (that is, either file.variable or the default file) are released. If either
file.variable or record.ID evaluates to the null value, the RELEASE statement fails
and the program terminates with a run-time error message.
When no options are specified, all locks in all files set by any FILELOCK, READL,
READU, READVL, READVU, WRITEU, WRITEVU, MATREADL, MATREADU,
MATWRITEU, or OPENSEQ statements during the current login session are
released.
A RELEASE statement within a transaction is ignored.
Examples
The following example releases all locks set in all files by the current program:
    RELEASE
The next example releases all locks set in the NAMES file:
    RELEASE NAMES
The next example releases the lock set on the record QTY in the PARTS file:
    RELEASE PARTS, "QTY"
REM function
Syntax
     REM (dividend, divisor)
Description
Use the REM function to calculate the remainder after integer division is
performed on the dividend expression by the divisor expression.
The REM function calculates the remainder using the following formula:
     REM (X, Y) = X − (INT (X / Y) * Y)
dividend and divisor can evaluate to any numeric value, except that divisor cannot
be 0. If divisor is 0, a division by 0 warning message is printed, and 0 is returned. If
either dividend or divisor evaluates to the null value, null is returned.
The REM function works like the MOD function.
Example
     X=85; Y=3
     PRINT 'REM (X,Y)= ',REM (X,Y)
This is the program output:
     REM (X,Y)=         1
REM
Syntax
      REM [comment.text]
Description
Use the REM statement to insert a comment in a BASIC program. Comments
explain or document various parts of a program. They are part of the source code
only and are nonexecutable. They do not affect the size of the object code.
A comment must be a separate BASIC statement, and can appear anywhere in a
program. A comment must be one of the following comment designators:
      REM   *    !   $*
Any text that appears between a comment designator and the end of a physical line
is treated as part of the comment. If a comment does not fit on one physical line, it
can be continued on the next physical line only by starting the new line with a
comment designator. If a comment appears at the end of a physical line containing
an executable statement, you must treat it as if it were a new statement and put a
semicolon ( ; ) after the executable statement, before the comment designator.
Example
      PRINT "HI THERE"; REM This part is a comment.
      REM This is also a comment and does not print.
      REM
      IF 5<6 THEN PRINT "YES"; REM A comment; PRINT "PRINT ME"
      REM BASIC thinks PRINT "PRINT ME" is also part
      REM of the comment.
      IF 5<6 THEN
          PRINT "YES"; REM Now it doesn't.
          PRINT "PRINT ME"
      END
This is the program output:
      HI THERE
      YES
      YES
      PRINT ME
REMOVE function
Syntax
    REMOVE (dynamic.array, variable)
Description
Use the REMOVE function to successively extract and return dynamic array
elements that are separated by system delimiters, and to indicate which system
delimiter was found. When a system delimiter is encountered, the value of the
extracted element is returned. The REMOVE function is more efficient than the
EXTRACT function for extracting successive fields, values, and so on, for multi-
value list processing.
dynamic.array is the dynamic array from which to extract elements.
variable is set to a code corresponding to the system delimiter which terminates the
extracted element. The contents of variable indicate which system delimiter was
found, as follows:
0   End of string
1   Item mark ASCII CHAR(255)
2   Field mark ASCII CHAR(254)
3   Value mark ASCII CHAR(253)
4   Subvalue mark ASCII CHAR(252)
5   Text mark ASCII CHAR(251)
6   ASCII CHAR(250) (Not available in the PIOPEN flavor)
7   ASCII CHAR(249) (Not available in the PIOPEN flavor)
8   ASCII CHAR(248) (Not available in the PIOPEN flavor)
The REMOVE function extracts one element each time it is executed, beginning
with the first element in dynamic.array. The operation can be repeated until all
elements of dynamic.array are extracted. The REMOVE function does not change
the dynamic array.
As each successive element is extracted from dynamic.array, a pointer associated
with dynamic.array is set to the beginning of the next element to be extracted. Thus
the pointer is advanced every time the REMOVE function is executed.
Examples
The first example sets the variable FIRST to the string MIKE and the variable X to
2 (field mark). The second example executes the REMOVE function and PRINT
statement until all the elements have been extracted, at which point A = 0. Printed
lines are 12, 4, 5, 7654, and 00.
 VM=CHAR(253)                                   Y   =   3   A   =   12
 A = 1                                          Y   =   3   A   =   4
 Z=12:VM:4:VM:5:VM:7654:VM:00                   Y   =   3   A   =   5
 FOR X=1 TO 20 UNTIL A=0                        Y   =   3   A   =   7654
    A = REMOVE(Z,Y)                             Y   =   0   A   =   0
    PRINT 'Y = ':Y, 'A = ':A
 NEXT X
REMOVE Statement
Syntax
    REMOVE element FROM dynamic.array SETTING variable
Description
Use the REMOVE statement to successively extract dynamic array elements that
are separated by system delimiters. When a system delimiter is encountered, the
extracted element is assigned to element. The REMOVE statement is more efficient
than the EXTRACT function for extracting successive fields, values, and so on, for
multivalue list processing.
dynamic.array is the dynamic array from which to extract elements.
variable is set to a code value corresponding to the system delimiter terminating the
element just extracted. The delimiter code settings assigned to variable are as
follows:
0   End of string
1   Item mark ASCII CHAR(255)
2   Field mark ASCII CHAR(254)
3   Value mark ASCII CHAR(253)
4   Subvalue mark ASCII CHAR(252)
5   Text mark ASCII CHAR(251)
6   ASCII CHAR(250) – Not supported in the PIOPEN flavor
7   ASCII CHAR(249) – Not supported in the PIOPEN flavor
8   ASCII CHAR(248) – Not supported in the PIOPEN flavor
The REMOVE statement extracts one element each time it is executed, beginning
with the first element in dynamic.array. The operation can be repeated until all
elements of dynamic.array are extracted. The REMOVE statement does not change
the dynamic array.
As each element is extracted from dynamic.array to element, a pointer associated
with dynamic.array is set to the beginning of the next element to be extracted. Thus,
the pointer is advanced every time the REMOVE statement is executed.
Examples
The first example sets the variable FIRST to the string MIKE and the variable X to
2 (field mark). The second example executes the REMOVE and PRINT statements
until all the elements have been extracted, at which point A = 0. Printed lines are
12, 4, 5, 7654, and 00.
 VM=CHAR(253)                                     Y=   12 A= 3
 A=1                                              Y=   4   A= 3
 Z=12:VM:4:VM:5:VM:7654:VM:00                     Y=   5   A= 3
 FOR X=1 TO 20 UNTIL A=0                          Y=   7654 A= 3
     REMOVE Y FROM Z SETTING A                    Y=   0   A= 0
     PRINT 'Y= ':Y, 'A= ':A
 NEXT X
REPEAT
The REPEAT statement is a loop-controlling statement. For syntax details, see the
LOOP statement.
REPLACE
Syntax
    REPLACE (expression, field#, value#, subvalue# { , | ; } replacement)
Description
Use the REPLACE function to return a copy of a dynamic array with the specified
field, value, or subvalue replaced with new data.
expression specifies a dynamic array.
The expressions field#, value#, and subvalue# specify the type and position of the
element to be replaced. These expressions are called delimiter expressions.
replacement specifies the value that the element is given.
The value# and subvalue# are optional. However, if either subvalue# or both value#
and subvalue# are omitted, a semicolon ( ; ) must precede replacement, as shown in
the second syntax.
You can use angle brackets to replace data in dynamic arrays. Angle brackets to the
left of an assignment operator change the specified data in the dynamic array
according to the assignment operator. Angle brackets to the right of an assignment
operator indicate that an EXTRACT function is to be performed (for examples, see
the EXTRACT function).
variable specifies the dynamic array containing the data to be changed.
The three possible results of delimiter expressions are described as case 1, case 2,
and case 3.
Case 1:      Both value# and subvalue# are omitted or are specified as 0. A field is
             replaced by the value of replacement.
                 • If field# is positive and less than or equal to the number of
                   fields in the dynamic array, the field specified by field# is
                   replaced by the value of replacement.
Examples
In the following examples a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S.
The first example replaces field 1 with # and sets Q to #FAVBVDSEFDFFF:
    R=@FM:"A":@VM:"B":@VM:"D":@SM:"E":@FM:"D":@FM:@FM:"F"
    Q=R
    Q=REPLACE(Q,1;"#")
The next example replaces the first subvalue of the third value in field 2 with # and
sets Q to FAVBV#SEFDFFF:
    Q=R
    Q<2,3,1>="#"
The next example replaces field 4 with # and sets Q to FAVBVDSEFDF#FF:
    Q=R
    Q=REPLACE(Q,4,0,0;"#")
The next example replaces the first value in fields 1 through 4 with # and sets Q to
#F#VBVDSEF#F#FF:
    Q=R
    FOR X=1 TO 4
    Q=REPLACE(Q,X,1,0;"#")
    NEXT
The next example appends a value mark and # to the last value in field 2 and sets
Q to FAVBVDSEV#FDFFF:
    Q=R
    Q=REPLACE(Q,2,-1;"#")
RETURN
Syntax
    RETURN [TO statement.label]
Description
Use the RETURN statement to terminate a subroutine and return control to the
calling program or statement.
If the TO clause is not specified, the RETURN statement exits either an internal
subroutine called by a GOSUB statement or an external subroutine called by a
CALL statement. Control returns to the statement that immediately follows the
CALL or GOSUB statement.
Use a RETURN statement to terminate an internal subroutine called with a GOSUB
statement to ensure that the program proceeds in the proper sequence.
Use a RETURN statement or an END statement to terminate an external subrou-
tine called with a CALL statement. When you exit an external subroutine called by
CALL, all files opened by the subroutine are closed, except files that are open to
common variables.
Use the TO clause to exit only an internal subroutine; control passes to the specified
statement label. If you use the TO clause and statement.label does not exist, an error
message appears when the program is compiled.
Note: Using the TO clause can make program debugging and modification
      extremely difficult. Be careful when you use the RETURN TO statement,
      because all other GOSUBs or CALLs active at the time the GOSUB is
      executed remain active, and errors can result.
If the RETURN or RETURN TO statement does not have a place to return to,
control is passed to the calling program or to the command language.
Example
In the following example, subroutine XYZ prints the message “THIS IS THE
EXTERNAL SUBROUTINE” and returns to the main program:
    20: GOSUB 80:
    25: PRINT "THIS LINE WILL NOT PRINT"
RETURN (value)
Syntax
    RETURN (expression)
Description
Use the RETURN (value) statement to return a value from a user-written function.
expression evaluates to the value you want the user-written function to return. If
you use a RETURN (value) statement in a user-written function and you do not
specify expression, an empty string is returned by default.
You can use the RETURN (value) statement only in user-written functions. If you
use one in a program or subroutine, an error message appears.
REUSE
Syntax
     REUSE (expression)
Description
Use the REUSE function to specify that the value of the last field, value, or
subvalue be reused in a dynamic array operation.
expression is either a dynamic array or an expression whose value is considered to
be a dynamic array.
When a dynamic array operation processes two dynamic arrays in parallel, the
operation is always performed on corresponding subvalues. This is true even for
corresponding fields, each of which contains a single value. This single value is
treated as the first and only subvalue in the first and only value in the field.
A dynamic array operation isolates the corresponding fields, values, and
subvalues in a dynamic array. It then operates on them in the following order:
1.   The subvalues in the values
2.   The values in the fields
3.   The fields of each dynamic array
A dynamic array operation without the REUSE function adds zeros or empty
strings to the shorter array until the two arrays are equal. (The DIVS function is an
exception. If a divisor element is absent, the divisor array is padded with ones, so
that the dividend value is returned.)
The REUSE function reuses the last value in the shorter array until all elements in
the longer array are exhausted or until the next higher delimiter is encountered.
After all subvalues in a pair of corresponding values are processed, the dynamic
array operation isolates the next pair of corresponding values in the corresponding
fields and repeats the procedure.
After all values in a pair of corresponding fields are processed, the dynamic array
operation isolates the next pair of corresponding fields in the dynamic arrays and
repeats the procedure.
If expression evaluates to the null value, the null value is replicated, and null is
returned for each corresponding element.
Example
    B = (1:@SM:6:@VM:10:@SM:11)
    A = ADDS(REUSE(5),B)
    PRINT "REUSE(5) + 1:@SM:6:@VM:10:@SM:11 = ": A
    *
    PRINT "REUSE(1:@SM:2) + REUSE(10:@VM:20:@SM:30) = ":
    PRINT ADDS(REUSE(1:@SM:2),REUSE(10:@VM:20:@SM:30))
    *
    PRINT "(4:@SM:7:@SM:8:@VM:10)*REUSE(10) = ":
    PRINT MULS((4:@SM:7:@SM:8:@VM:10 ),REUSE(10))
This is the program output:
    REUSE(5) + 1:@SM:6:@VM:10:@SM:11 = 6S11V15S16
    REUSE(1:@SM:2) + REUSE(10:@VM:20:@SM:30) = 11S12V22S32
    (4:@SM:7:@SM:8:@VM:10)*REUSE(10) = 40S70S80V100
REVREMOVE statement
Syntax
    REVREMOVE element FROM dynamic.array SETTING variable
Description
Use the REVREMOVE statement to successively extract dynamic array elements
that are separated by system delimiters. The elements are extracted from right to
left, in the opposite order from those extracted by the REMOVE Statement. When
a system delimiter is encountered, the extracted element is assigned to element.
dynamic.array is an expression that evaluates to the dynamic array from which to
extract elements.
variable is set to a code value corresponding to the system delimiter terminating the
element just extracted. The delimiter code settings assigned to variable are as
follows:
0   End of string
1   Item mark ASCII CHAR(255)
2   Field mark ASCII CHAR(254)
3   Value mark ASCII CHAR(253)
4   Subvalue mark ASCII CHAR(252)
5   Text mark ASCII CHAR(251)
6   ASCII CHAR(250)
7   ASCII CHAR(249)
8   ASCII CHAR(248)
The REVREMOVE statement extracts one element each time it is executed, begin-
ning with the last element in dynamic.array. The operation can be repeated until all
elements of dynamic.array are extracted. The REVREMOVE statement does not
change the dynamic array.
As each element is extracted from dynamic.array to element, a pointer associated
with dynamic.array moves back to the beginning of the element just extracted.
The pointer is reset to the beginning of dynamic.array whenever dynamic.array is
reassigned. Therefore, dynamic.array should not be assigned a new value until all
elements have been extracted (that is, until variable = 0).
If an element in dynamic.array is the null value, null is returned for that element.
Use REVREMOVE with the REMOVE statement. After a REMOVE statement,
REVREMOVE returns the same string as the preceding REMOVE, setting the
pointer to the delimiter preceding the extracted element. Thus, a subsequent
REMOVE statement extracts the same element yet a third time.
If no REMOVE statement has been performed on dynamic.array or if the leftmost
dynamic array element has been returned, element is set to the empty string and
variable indicates end of string (that is, 0).
Example
    DYN = "THIS":@FM:"HERE":@FM:"STRING"
    REMOVE VAR FROM DYN SETTING X
    PRINT VAR
    REVREMOVE NVAR FROM DYN SETTING X
    PRINT NVAR
    REMOVE CVAR FROM DYN SETTING X
    PRINT CVAR
The program output is:
    THIS
    THIS
    THIS
REWIND
Syntax
    REWIND [UNIT (mtu)]
            {THEN statements [ELSE statements] | ELSE statements}
Description
Use the REWIND statement to rewind a magnetic tape to the beginning-of-tape
position.
The UNIT clause specifies the number of the tape drive unit. Tape unit 0 is used if
no unit is specified. If the UNIT clause is used, mtu is an expression that evaluates
to a code made up of three decimal digits. Although the mtu expression is a func-
tion of the UNIT clause, the REWIND statement uses only the third digit (the u).
Its value must be in the range of 0 through 7. If mtu evaluates to the null value, the
REWIND statement fails and the program terminates with a run-time error
message.
Before a REWIND statement is executed, a tape drive unit must be attached to the
user. Use the ASSIGN command to assign a tape unit to a user. If no tape unit is
attached or if the unit specification is incorrect, the ELSE statements are executed.
The STATUS function returns 1 if REWIND takes the ELSE clause, otherwise it
returns 0.
PIOPEN Flavor
If you have a program that specifies the syntax UNIT ndmtu, the nd elements are
ignored by the compiler and no errors are reported.
Example
    REWIND UNIT(002) ELSE PRINT "UNIT NOT ATTACHED"
RIGHT
Syntax
    RIGHT (string, n)
Description
Use the RIGHT function to extract a substring comprising the last n characters of a
string. It is equivalent to the following substring extraction operation:
    string [ length ]
If you use this function, you need not calculate the string length.
If string evaluates to the null value, null is returned. If n evaluates to the null value,
the RIGHT function fails and the program terminates with a run-time error
message.
Example
    PRINT RIGHT("ABCDEFGH",3)
This is the program output:
    FGH
RND
Syntax
      RND (expression)
Description
Use the RND function to generate any positive or negative random integer or 0.
expression evaluates to the total number of integers, including 0, from which the
random number can be selected. That is, if n is the value of expression, the random
number is generated from the numbers 0 through (n − 1).
If expression evaluates to a negative number, a random negative number is gener-
ated. If expression evaluates to 0, 0 is the random number. If expression evaluates to
the null value, the RND function fails and the program terminates with a run-time
error message.
See the RANDOMIZESyntax statement for details on generating repeatable
sequences of random numbers.
Example
      A=20
      PRINT   RND(A)
      PRINT   RND(A)
      PRINT   RND(A)
      PRINT   RND(A)
This is the program output:
      10
      3
      6
      10
ROLLBACK
Syntax
    ROLLBACK [ WORK ] [ THEN statements ] [ ELSE statements ]
Description
Use the ROLLBACK statement to cancel all file I/O changes made during a trans-
action. The WORK keyword provides compatibility with SQL syntax conventions;
it is ignored by the compiler.
A transaction includes all statements executed since the most recent BEGIN
TRANSACTION statement. The ROLLBACK statement rolls back all changes
made to files during the active transaction. If a subtransaction rolls back, none of
the changes resulting from the active subtransaction affect the parent transaction.
If the top-level transaction rolls back, none of the changes made are committed to
disk, and the database remains unaffected by the transaction.
Use the ROLLBACK statement in a transaction without a COMMIT statement to
review the results of a possible change. Doing so does not affect the parent trans-
action or the database. Executing a ROLLBACK statement ends the current
transaction. After the transaction ends, execution continues with the statement
following the next END TRANSACTION statement.
If no transaction is active, the ROLLBACK statement generates a run-time
warning, and the ELSE statements are executed.
Example
This example begins a transaction that applies locks to rec1 and rec2. If errors occur
(such as a failed READU or a failed WRITE), the ROLLBACK statements ensure
that no changes are written to the file.
    BEGIN TRANSACTION
       READU data1 FROM file1,rec1 ELSE ROLLBACK
       READU data2 FROM file2,rec2 ELSE ROLLBACK
           .
           .
           .
       WRITE new.data1 ON file1,rec1 ELSE ROLLBACK
       WRITE new.data2 ON file2,rec2 ELSE ROLLBACK
       COMMIT WORK
    END TRANSACTION
The update record lock on rec1 is not released on successful completion of the first
WRITE statement.
RPC.CALL
Syntax
    RPC.CALL (connection.ID, procedure, #args, MAT arg.list, #values,
          MAT return.list)
Description
Use the RPC.CALL function to make requests of a connected server. The request is
packaged and sent to the server using the C client RPC library. RPC.CALL returns
the results of processing the remote request: 1 for success, 0 for failure.
connection.ID is the handle of the open server connection on which to issue the RPC
request. The RPC.CONNECT function gets the connection.ID.
procedure is a string identifying the operation requested of the server.
#args is the number of elements of arg.list to pass to the RPC server.
arg.list is a two-dimensional array (matrix) containing the input arguments to pass
to the RPC server. The elements of this array represent ordered pairs of values. The
first value is the number of the argument to the server operation, the second value
is an argument-type declarator. (Data typing generalizes the RPC interface to work
with servers that are data-type sensitive.)
#values is the number of values returned by the server.
return.list is a dimensioned array containing the results of the remote operation
returned by RPC.CALL. Like arg.list, the results are ordered pairs of values.
RPC.CALL builds an RPC packet from #args and arg.list. Functions in the C client
RPC library transmit the packet to the server and wait for the server to respond.
When a response occurs, the RPC packet is separated into its elements and stored
in the array return.list.
Use the STATUS function after an RPC.CALL function is executed to determine the
result of the operation, as follows:
Example
The following example looks for jobs owned by fred. The server connection was
made using the RPC.CONNECT function.
   args (1,1) = "fred"; args (1,2) = UVRPC.STRING
   IF (RPC.CALL (server.handle, "COUNT.USERS", 1, MAT args,
          return.count, MAT res)) ELSE
       PRINT "COUNT.JOBS request failed, error code is: " STATUS()
       GOTO close.connection:
   END
RPC.CONNECT
Syntax
    RPC.CONNECT (host, server)
Description
Use the RPC.CONNECT function to establish a connection to a server process.
Once the host and server are identified, the local daemon tries to connect to the
remote server. If the attempt succeeds, RPC.CONNECT returns a connection ID. If
it fails, RPC.CONNECT returns 0. The connection ID is a nonzero integer used to
refer to the server in subsequent calls to RPC.CALL and RPC.DISCONNECT.
host is the name of the host where the server resides.
    UNIX. This is defined in the local /etc/hosts file.
    Windows NT. This is defined in the system32\drivers\etc\hosts file.
server is the name, as defined in the remote /etc/services file, of the RPC server class
on the target host.
If host is not in the /etc/hosts file, or if server is not in the remote /etc/services file, the
connection attempt fails.
Use the STATUS function after an RPC.CONNECT function is executed to deter-
mine the result of the operation, as follows:
Example
The following example connects to a remote server called MONITOR on HOST.A:
   MAT args(1,2), res(1,2)
   server.handle = RPC.CONNECT ("HOST.A", "MONITOR")
   IF (server.handle = 0) THEN
       PRINT "Connection failed, error code is: ": STATUS()
       STOP
   END
RPC.DISCONNECT
Syntax
    RPC.DISCONNECT (connection.ID)
Description
Use the RPC.DISCONNECT function to end an RPC session.
connection.ID is the RPC server connection you want to close.
RPC.DISCONNECT sends a request to end a connection to the server identified by
connection.ID. When the server gets the request to disconnect, it performs any
required termination processing. If the call is successful, RPC.DISCONNECT
returns 1. If an error occurs, RPC.DISCONNECT returns 0.
Use the STATUS function after an RPC.DISCONNECT function is executed to
determine the result of the operation, as follows:
Example
The following example closes the connection to a remote server called MONITOR
on HOST.A:
    MAT args(1,2), res(1,2)
    server.handle = RPC.CONNECT ("HOST.A", "MONITOR")
    IF (server.handle = 0) THEN
        PRINT "Connection failed, error code is: ": STATUS()
        STOP
    END
        .
        .
        .
    close.connection:
    IF (RPC.DISCONNECT (server.handle)) ELSE
        PRINT "Bizarre disconnect error, result code is: " STATUS()
    END
SADD
Syntax
    SADD (string.number.1, string.number.2)
Description
Use the SADD function to add two string numbers and return the result as a string
number. You can use this function in any expression where a string or string
number is valid, but not necessarily where a standard number is valid, because
string numbers can exceed the range of numbers that standard arithmetic opera-
tors can handle.
Either string number can evaluate to any valid number or string number.
If either string number contains nonnumeric data, an error message is generated,
and 0 replaces the nonnumeric data. If either string number evaluates to the null
value, null is returned.
Example
    A = 88888888888888888
    B = 77777777777777777
    X = "88888888888888888"
    Y = "77777777777777777"
    PRINT A + B
    PRINT SADD(X,Y)
This is the program output:
    166666666666667000
    166666666666666665
SCMP
Syntax
    SCMP (string.number.1, string.number.2)
Description
Use the SCMP function to compare two string numbers and return one of the
following three numbers: −1 (less than), 0 (equal), or 1 (greater than). If
string.number.1 is less than string.number.2, the result is 1. If they are equal, the
result is 0. If string.number.1 is greater than string.number.2, the result is 1. You can
use this function in any expression where a string or string number is valid.
Either string number can be a valid number or string number. Computation is
faster with string numbers.
If either string number contains nonnumeric data, an error message is generated
and 0 is used instead of the nonnumeric data. If either string number evaluates to
the empty string, null is returned.
Example
    X = "123456789"
    Y = "123456789"
    IF SCMP(X,Y) = 0 THEN PRINT "X is equal to Y"
        ELSE PRINT "X is not equal to Y"
    END
This is the program output:
    X is equal to Y
SDIV
Syntax
       SDIV (string.number.1, string.number.2 [ ,precision])
Description
Use the SDIV function to divide string.number.1 by string.number.2 and return the
result as a string number. You can use this function in any expression where a
string or string number is valid, but not necessarily where a standard number is
valid, because string numbers can exceed the range of numbers which standard
arithmetic operators can handle. Either string number can be a valid number or a
string number.
precision specifies the number of places to the right of the decimal point. The
default precision is 14.
If either string number contains nonnumeric data, an error message is generated
and 0 is used for that number. If either string number evaluates to the null value,
null is returned.
Example
       X = "1"
       Y = "3"
       Z = SDIV (X,Y)
       ZZ = SDIV (X,Y,20)
       PRINT Z
       PRINT ZZ
This is the program output:
       0.33333333333333
       0.33333333333333333333
SEEK
Syntax
       SEEK file.variable [ , offset [ , relto] ]
                {THEN statements [ ELSE statements ] | ELSE statements}
Description
Use the SEEK statement to move the file pointer by an offset specified in bytes,
relative to the current position, the beginning of the file, or the end of the file.
file.variable specifies a file previously opened for sequential access.
offset is the number of bytes before or after the reference position. A negative offset
results in the pointer being moved before the position specified by relto. If offset is
not specified, 0 is assumed.
Note: On Windows NT systems, line endings in files are denoted by the character
      sequence RETURN + LINEFEED rather than the single LINEFEED used in
      UNIX files. The value of offset should take into account this extra byte on
      each line in Windows NT file systems.
        can move the file pointer only to the beginning or the end of the data. For
        diskette drives, you can move the file pointer only to the start of the data.
Seeking beyond the end of the file and then writing creates a gap, or hole, in the
file. This hole occupies no physical space, and reads from this part of the file return
as ASCII CHAR 0 (neither the number nor the character 0).
For more information about sequential file processing, see the OPENSEQ,
READSEQ, and WRITESEQ statements.
Example
The following example reads and prints the first line of RECORD4. Then the SEEK
statement moves the pointer five bytes from the front of the file, then reads and
prints the rest of the current line.
    OPENSEQ 'FILE.E', 'RECORD4' TO FILE ELSE ABORT
    READSEQ B FROM FILE THEN PRINT B
    SEEK FILE,5, 0 THEN READSEQ A FROM FILE
    THEN PRINT A ELSE ABORT
This is the program output:
    FIRST LINE
     LINE
SEEK(ARG.)
Syntax
    SEEK(ARG. [ ,arg#] ) [THEN statements] [ELSE statements]
Description
Use the SEEK(ARG.) statement to move the command line argument pointer to the
next command line argument from left to right, or to a command line argument
specified by arg#. The command line is delimited by blanks, and the first argument
is assumed to be the first word after the program name. When a cataloged program
is invoked, the argument list starts with the second word in the command line.
Blanks in quoted strings are not treated as delimiters. A quoted string is treated as
a single argument.
arg# specifies the command line argument to move to. It must evaluate to a
number. If arg# is not specified, the pointer moves to the next command line argu-
ment. SEEK(ARG.) works similarly to GET(ARG.) except that SEEK(ARG.) makes
no assignments.
THEN and ELSE statements are both optional. The THEN clause is executed if the
argument is found. The ELSE clause is executed if the argument is not found. The
SEEK(ARG.) statement fails if arg# evaluates to a number greater than the number
of command line arguments or if the last argument has been assigned and a
SEEK(ARG.) with no arg# is used. To move to the beginning of the argument list,
set arg# to 1.
If arg# evaluates to the null value, the SEEK(ARG.) statement fails and the program
terminates with a run-time error message.
Example
If the command line is:
    RUN BP PROG ARG1 ARG2 ARG3
and the program is:
    A=5;B=2
    SEEK(ARG.)
    SEEK(ARG.,B)
    SEEK(ARG.)
   SEEK(ARG.,A-B)
   SEEK(ARG.,1)
the system pointer moves as follows:
   ARG2
   ARG2
   ARG3
   ARG3
   ARG1
SELECT
Syntax
    SELECT [variable] [TO list.number] [ON ERROR statements]
SELECT TO
    SELECTN [variable] [TO list.number] [ON ERROR statements]
Description
Use a SELECT statement to create a numbered select list of record IDs from a
DataStage file or a dynamic array. A subsequent READNEXT statement can access
this select list, removing one record ID at a time from the list. READNEXT instruc-
tions can begin processing the select list immediately.
variable can specify a dynamic array or a file variable. If it specifies a dynamic array,
the record IDs must be separated by field marks (ASCII 254). If variable specifies a
file variable, the file variable must have previously been opened. If variable is not
specified, the default file is assumed (for more information on default files, see the
OPEN statement). If the file is neither accessible nor open, or if variable evaluates
to the null value, the SELECT statement fails and the program terminates with a
run-time error message.
If the file is an SQL table, the effective user of the program must have SQL SELECT
privilege to read records in the file. For information about the effective user of a
program, see the AUTHORIZATION statement.
You must use a file lock with the SELECT statement when it is within a transaction
running at isolation level 4 (serializable). This prevents phantom reads.
The TO clause specifies the select list that is to be used. list.number is an integer
from 0 through 10. If no list.number is specified, select list 0 is used.
The record IDs of all the records in the file, in their stored order, form the list. Each
record ID is one entry in the list.
The SELECT statement does not process the entire file at once. It selects record IDs
group by group. The @SELECTED variable is set to the number of elements in the
group currently being processed.
You often want a select list with the record IDs in an order different from their
stored order or with a subset of the record IDs selected by some specific criteria. To
Example
The following example opens the file SUN.MEMBER to the file variable
MEMBER.F, then creates an active select list of record IDs. The READNEXT state-
ment assigns the first record ID in the select list to the variable @ID, then prints it.
Next, the file SUN.SPORT is opened to the file variable SPORT.F, and a select list
of its record IDs is stored as select list 1. The READNEXT statement assigns the first
record ID in the select list to the variable A, then prints DONE.
    OPEN '','SUN.MEMBER' TO MEMBER.F ELSE PRINT "NOT OPEN"
    SELECT
    READNEXT @ID THEN PRINT @ID
    *
    OPEN '','SUN.SPORT' TO SPORT.F ELSE PRINT "NOT OPEN"
    SELECT TO 1
    READNEXT A FROM 1 THEN PRINT "DONE" ELSE PRINT "NOT"
This is the program output:
    4108
    DONE
SELECTE
Syntax
    SELECTE TO list.variable
Description
Use the SELECTE statement to assign the contents of select list 0 to list.variable.
list.variable is activated in place of select list 0 and can be read with the READNEXT
statement.
SELECTINDEX
Syntax
    SELECTINDEX index [ ,alt.key] FROM file.variable [TO list.number]
Description
Use the SELECTINDEX statement to create select lists from secondary indexes.
file.variable specifies an open file.
index is an expression that evaluates to the name of an indexed field in the file. index
must be the name of the field that was used in the CREATE.INDEX command
when the index was built.
list.number is an expression that evaluates to the select list number. It can be a
number from 0 through 10. The default list number is 0.
alt.key is an expression that evaluates to an alternate key value contained in the
index. If alt.key is specified, a select list is created of the record IDs referenced by
alt.key. If alt.key is not specified, a select list is created of all of the index’s alternate
keys.
If the field is not indexed, the select list is empty, and the value of the STATUS func-
tion is 1; otherwise the STATUS function is 0. If index, alt.key, or file.variable
evaluates to the null value, the SELECTINDEX statement fails and the program
terminates with a run-time error message.
PIOPEN Flavor
In a PIOPEN flavor account, the SELECTINDEX statement creates a select list from
the secondary indexes without duplicate keys. To implement this functionality in
other flavors, use the PIOPEN.SELIDX option with the $OPTIONS statement.
Example
In the following example, the first SELECTINDEX selects all data values to list 1.
The second SELECTINDEX selects record IDs referenced by STOREDVAL to list 2.
    OPEN "", "DB" TO FV ELSE STOP "OPEN FAILED"
    SELECTINDEX "F1" FROM FV TO 1
    EOV = 0
    LOOP
   UNTIL EOV DO
          SELECTINDEX "F1", STOREDVAL FROM FV TO 2
          EOK = 0
          LOOP
          READNEXT KEY FROM 2 ELSE EOK=1
          UNTIL EOK DO
          PRINT "KEY IS ":KEY:" STOREDVAL IS ":STOREDVAL
          REPEAT
          REPEAT
          END
SELECTINFO
Syntax
    SELECTINFO (list, key)
Description
Use the SELECTINFO function to determine whether a select list is active, or to
determine the number of items it contains.
list is an expression evaluating to the number of the select list for which you require
information. The select list number must be in the range of 0 through 10.
key specifies the type of information you require. You can use equate names for the
keys as follows:
Equate Names
An insert file of equate names is provided for the SELECTINFO keys. To use the
equate names, specify the directive $INCLUDE SYSCOM INFO_KEYS.INS.IBAS
when you compile your program.
Example
In the following example, the insert file containing the equate name is inserted by
the $INCLUDE statement. The conditional statement tests if select list 0 is active.
    $INCLUDE SYSCOM INFO_KEYS.INS.IBAS
       IF SELECTINFO(0,IK$SLACTIVE)
         THEN PRINT 'SELECT LIST ACTIVE'
         ELSE PRINT 'SELECT LIST NOT ACTIVE'
       END
SEND
Syntax
    SEND output [:] TO device
             { THEN statements [ ELSE statements ] | ELSE statements }
Description
Use the SEND statement to write a block of data to a device. The SEND statement
can be used to write data to a device that has been opened for I/O using the
OPENDEV or OPENSEQ statement.
output is an expression evaluating to a data string that will be written to device. If
the optional colon is used after output, the terminating newline is not generated.
device is a valid file variable resulting from a successful OPENDEV or OPENSEQ
statement. This is the handle to the I/O device that supplies the data stream for the
operation of the SEND statement.
The SEND syntax requires that either a THEN or an ELSE clause, or both, be spec-
ified. If data is successfully sent, the SEND statement executes the THEN clause. If
data cannot be sent, it executes the ELSE clause.
The data block specified by output is written to the device followed by a newline.
Upon successful completion of the SEND operation, program control is passed to
the THEN clause if specified. If an error occurs during the SEND operation,
program control is passed to the ELSE clause if specified.
Example
The following code fragment shows how the SEND statement is used to write a
series of messages on a connected device:
    OPENDEV "TTY10" TO TTYLINE ELSE STOP "CANNOT OPEN TTY10"
    LOOP
           INPUT MESSAGE
       WHILE MESSAGE # "QUIT" DO
           SEND MESSAGE TO TTYLINE
           ELSE
               STOP "ERROR WRITING DATA TO TTY10"
           END
    REPEAT
SENTENCE
Syntax
    SENTENCE ( )
Description
Use the SENTENCE function to return the stored sentence that invoked the current
process. Although the SENTENCE function uses no arguments, parentheses are
required to identify it as a function. The SENTENCE function is a synonym for the
@SENTENCE system variable.
A PERFORM statement in a program updates the system variable, @SENTENCE,
with the command specified in the PERFORM statement.
Example
    PRINT SENTENCE()
This is the program output:
    RUN BP TESTPROGRAM
SEQ
Syntax
      SEQ (expression)
Description
Use the SEQ function to convert an ASCII character to its numeric string
equivalent.
expression evaluates to the ASCII character to be converted. If expression evaluates
to the null value, null is returned.
The SEQ function is the inverse of the CHAR function.
In NLS mode, use the UNISEQ function to return Unicode values in the range
x0080 through x00F8.
Using the SEQ function to convert a character outside its range results in a run-
time message, and the return of an empty string.
For more information about these ranges, see the DataStage NLS Guide.
Example
      G="T"
      A=SEQ(G)
      PRINT A, A+1
      PRINT SEQ("G")
This is the program output:
      84     85
      71
SEQS
Syntax
       SEQS (dynamic.array)
       CALL −SEQS (return.array, dynamic.array)
       CALL !SEQS (return.array, dynamic.array)
Description
Use the SEQS function to convert a dynamic array of ASCII characters to their
numeric string equivalents.
dynamic.array specifies the ASCII characters to be converted. If dynamic.array eval-
uates to the null value, null is returned. If any element of dynamic.array is the null
value, null is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
In NLS mode, you can use the UNISEQS function to return Unicode values in the
range x0080 through x00F8.
Using the SEQS function to convert a character outside its range results in a run-
time message, and the return of an empty string.
For more information about these ranges, see the DataStage NLS Guide.
Example
       G="T":@VM:"G"
       A=SEQS(G)
       PRINT A
       PRINT SEQS("G")
This is the program output:
       84V71
       71
Description
Use the SET TRANSACTION ISOLATION LEVEL statement to set the default
transaction isolation level you need for your program.
Note: The isolation level you set with this statement remains in effect until
      another such statement is issued. This affects all activities in the session,
      including DataStage commands and SQL transactions.
      {n | keyword | expression}
level is an expression that evaluates to 0 through 4, or one of the following
keywords:
Examples
The following example sets the default isolation level to 3 then starts a transaction
at isolation level 4. The isolation level is reset to 3 after the transaction finishes.
    SET TRANSACTION ISOLATION LEVEL REPEATABLE.READ
    PRINT "We are at isolation level 3."
    BEGIN TRANSACTION ISOLATION LEVEL SERIALIZABLE
              PRINT "We are at isolation level 4."
              COMMIT WORK
    END TRANSACTION
    PRINT "We are at isolation level 3"
The next example uses an expression to set the transaction level:
    PRINT "Enter desired transaction isolation level:":
    INPUT TL
       SET TRANSACTION LEVEL TL
       BEGIN TRANSACTION
           .
           .
           .
       END TRANSACTION
SETLOCALE
Syntax
    SETLOCALE (category, value)
Description
In NLS mode, use the SETLOCALE function to enable or disable a locale for a spec-
ified category or change its setting.
category is one of the following tokens that are defined in the UVNLSLOC.H file:
value specifies either a dynamic array whose elements are separated by field marks
or the string OFF. An array can have one or five elements:
    • If the array has one element, all categories are set or unset to that value.
    • If the array has five elements, it specifies the following values in this order:
      TIME, NUMERIC, MONETARY, CTYPE, and COLLATE.
The MD, MR, and ML conversions require both Numeric and Monetary categories
to be set in order for locale information to be used.
Examples
The following example sets all the categories in the locale to FR-FRENCH:
    status = SETLOCALE(UVLC$ALL,"FR-FRENCH")
The next example saves the current locale. This is the equivalent of executing the
SAVE.LOCALE command.
    status = SETLOCALE(UVLC$SAVE,"")
The next example sets the Monetary category to DE-GERMAN:
    status = SETLOCALE(UVLC$MONETARY,"DE-GERMAN")
The next example disables the Monetary category. DataStage behaves as though
there were no locales for the Monetary category only.
    status = SETLOCALE(UVLC$MONETARY,"OFF")
The next example completely disables locale support for all categories:
    status = SETLOCALE(UVLC$ALL,"OFF")
The next example restores the locale setting saved earlier:
    status = SETLOCALE(UVLC$RESTORE,"")
SETREM
Syntax
    SETREM position ON dynamic.array
Description
Use the SETREM statement to set the remove pointer in dynamic.array to the posi-
tion specified by position.
position is an expression that evaluates to the number of bytes you want to move
the pointer forward. If it is larger than the length of dynamic.array, the length of
dynamic.array is used. If it is less than 0, 0 is used.
dynamic.array must be a variable that evaluates to a string. If it does not evaluate to
a string, an improper data type warning is issued.
If the pointer does not point to the first character after a system delimiter, subse-
quent REMOVE Statement and REVREMOVE statement act as follows:
    • A REMOVE statement returns a substring, starting from the pointer and
      ending at the next delimiter.
    • A REVREMOVE statement returns a substring, starting from the previous
      delimiter and ending at the pointer.
If NLS is enabled and you use a multibyte character set, use GETREM to ensure
that position is at the start of a character. For more information about locales, see the
DataStage NLS Guide.
Example
    DYN = "THIS":@FM:"HERE":@FM:"STRING"
    REMOVE VAR FROM DYN SETTING X
    A = GETREM(DYN)
    REMOVE VAR FROM DYN SETTING X
    PRINT VAR
    SETREM A ON DYN
    REMOVE VAR FROM DYN SETTING X
    PRINT VAR
The program output is:
    HERE
    HERE
SIN
Syntax
      SIN (expression)
Description
Use the SIN function to return the trigonometric sine of an expression. expression
represents the angle expressed in degrees. Numbers greater than 1E17 produce a
warning message, and 0 is returned. If expression evaluates to the null value, null is
returned.
Example
      PRINT SIN(45)
This is the program output:
      0.7071
SINH
Syntax
       SINH (expression)
Description
Use the SINH function to return the hyperbolic sine of expression. expression must
be numeric and represents the angle expressed in degrees. If expression evaluates to
the null value, null is returned.
Example
       PRINT "SINH(2) = ":SINH(2)
This is the program output:
       SINH(2) = 3.6269
SLEEP
Syntax
    SLEEP [seconds]
Description
Use the SLEEP statement to suspend execution of a BASIC program, pausing for a
specified number of seconds.
seconds is an expression evaluating to the number of seconds for the pause. If
seconds is not specified, a value of 1 is used. If seconds evaluates to the null value, it
is ignored and 1 is used.
Example
In the following example the program pauses for three seconds before executing
the statement after the SLEEP statement. The EXECUTE statement clears the
screen.
    PRINT "STUDY THE FOLLOWING SENTENCE CLOSELY:"
    PRINT
    PRINT
    PRINT "There are many books in the"
    PRINT "the library."
    SLEEP 3
    EXECUTE 'CS'
    PRINT "DID YOU SEE THE MISTAKE?"
This is the program output:
    STUDY THE FOLLOWING SENTENCE CLOSELY:
SMUL
Syntax
    SMUL (string.number.1, string.number.2)
Description
Use the SMUL function to multiply two string numbers and return the result as a
string number. You can use this function in any expression where a string or string
number is valid, but not necessarily where a standard number is valid, because
string numbers can exceed the range of numbers that standard arithmetic opera-
tors can handle.
Either string number can be any valid number or string number.
If either string number contains nonnumeric data, an error message is generated
and 0 is used for that number. If either string number evaluates to the null value,
null is returned.
Example
    X = "5436"
    Y = "234"
    Z = SMUL (X,Y)
    PRINT Z
This is the program output:
    1272024
SOUNDEX
Syntax
    SOUNDEX (expression)
Description
The SOUNDEX function evaluates expression and returns the most significant letter
in the input string followed by a phonetic code. Nonalphabetic characters are
ignored. If expression evaluates to the null value, null is returned.
This function uses the soundex algorithm (the same as the one used by the SAID
keyword in RetrieVe) to analyze the input string. The soundex algorithm returns
the first letter of the alphabetic string followed by a one- to three-digit phonetic
code.
Example
SPACE
Syntax
    SPACE (expression)
Description
Use the SPACE function to return a string composed of blank spaces. expression
specifies the number of spaces in the string. If expression evaluates to the null value,
the SPACE function fails and the program terminates with a run-time error
message.
There is no limit to the number of blank spaces that can be generated.
Example
    PRINT "HI":SPACE(20):"THERE"
    *
    *
    VAR=SPACE(5)
    PRINT "TODAY IS":VAR:OCONV(DATE(),"D")
This is the program output:
    HI             THERE
    TODAY IS      18 JUN 1992
SPACES
Syntax
    SPACES (dynamic.array)
    CALL −SPACES (return.array, dynamic.array)
    CALL !SPACES (return.array, dynamic.array)
Description
Use the SPACES function to return a dynamic array with elements composed of
blank spaces. dynamic.array specifies the number of spaces in each element. If
dynamic.array or any element of dynamic.array evaluates to the null value, the
SPACES function fails and the program terminates with a run-time error message.
There is no limit to the number of blank spaces that can be generated except avail-
able memory.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
SPLICE
Syntax
    SPLICE (array1, expression, array2)
    CALL −SPLICE (return.array, array1, expression, array2)
    CALL !SPLICE (return.array, array1, expression, array2)
Description
Use the SPLICE function to create a dynamic array of the element-by-element
concatenation of two dynamic arrays, separating concatenated elements by the
value of expression.
Each element of array1 is concatenated with expression and with the corresponding
element of array2. The result is returned in the corresponding element of a new
dynamic array. If an element of one dynamic array has no corresponding element
in the other dynamic array, the element is returned properly concatenated with
expression. If either element of a corresponding pair is the null value, null is
returned for that element. If expression evaluates to the null value, null is returned
for the entire dynamic array.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    A="A":@VM:"B":@SM:"C"
    B="D":@SM:"E":@VM:"F"
    C='-'
    PRINT SPLICE(A,C,B)
This is the program output:
    A-DS-EVB-FSC-
SQRT
Syntax
       SQRT (expression)
Description
Use the SQRT function to return the square root of expression. expression must eval-
uate to a numeric value that is greater than or equal to 0. If expression evaluates to
a negative value, the result of the function is SQRT(−n) and an error message is
printed. If expression evaluates to the null value, null is returned.
Example
       A=SQRT(144)
       PRINT A
       *
       PRINT "SQRT(45) IS ":SQRT(45)
This is the program output:
       12
       SQRT(45) IS 6.7082
SQUOTE
Syntax
    SQUOTE (expression)
    CALL !SQUOTE (quoted.expression, expression)
Description
Use the SQUOTE function to enclose an expression in single quotation marks. If
expression evaluates to the null value, null is returned, without quotation marks.
quoted.expression is the quoted string.
expression is the input string.
Example
    PRINT SQUOTE(12 + 5) : " IS THE ANSWER."
    END
This is the program output:
    '17' IS THE ANSWER.
SSELECT
Syntax
    SSELECT [variable] [TO list.number] [ON ERROR statements]
Description
Use an SSELECT statement to create a numbered select list of record IDs in sorted
order from a DataStage file or a dynamic array. You can then access this select list
by a subsequent READNEXT statement which removes one record ID at a time
from the list.
variable can specify a dynamic array or a file variable. If it specifies a dynamic array,
the record IDs must be separated by field marks (ASCII 254). If variable specifies a
file variable, the file variable must have previously been opened. If variable is not
specified, the default file is assumed (for more information on default files, see the
OPEN statement). If the file is neither accessible nor open, or if variable evaluates
to the null value, the SSELECT statement fails and the program terminates with a
run-time error message.
If the file is an SQL table, the effective user of the program must have SQL SELECT
privilege to read records in the file. For information about the effective user of a
program, see the AUTHORIZATION statement.
You must use a file lock with the SSELECT statement when it is within a transac-
tion running at isolation level 4 (serializable). This prevents phantom reads.
The TO clause specifies the select list that is to be used. list.number is an integer
from 0 through 10. If no list.number is specified, select list 0 is used.
The record IDs of all the records in the file form the list. The record IDs are listed
in ascending order. Each record ID is one entry in the list.
You often want a select list with the record IDs in an order different from their
stored order or with a subset of the record IDs selected by some specific criteria. To
do this, use the SELECT or SSELECT commands in a BASIC EXECUTE statement.
Processing the list by READNEXT is the same, regardless of how the list is created.
Use the SSELECTV statement to store the select list in a named list variable instead
of to a numbered select list. list.variable is an expression that evaluates to a valid
variable name. This is the default behavior of the SSELECT statement in PICK,
REALITY, and IN2 flavor accounts. You can also use the VAR.SELECT option of
the $OPTIONS statement to make the SSELECT statement act as it does in PICK,
REALITY, and IN2 flavor accounts.
In NLS mode when locales are enabled, the SSELECT statements use the Collate
convention of the current locale to determine the collating order. For more infor-
mation about locales, see the DataStage NLS Guide.
You can use either the SSELECT or the SSELECTV statement to create a select list
and store it in a named list variable. The only useful thing you can do with a list
variable is use a READNEXT statement to read the next element of the select list.
Use the SSELECTN statement to store the select list in a numbered select list.
list.number is an expression that evaluates to a number from 0 through 10. You can
also use the −VAR.SELECT option of the $OPTIONS statement to make the
SSELECT statement act as it does in IDEAL and INFORMATION flavor accounts.
Example
The following example opens the file SUN.MEMBER to the file variable
MEMBER.F, then creates an active sorted select list of record IDs. The READNEXT
statement assigns the first record ID in the select list to the variable @ID, then
prints it. Next, the file SUN.SPORT is opened to the file variable SPORT.F, and a
sorted select list of its record IDs is stored as select list 1. The READNEXT state-
ment assigns the first record ID in the select list to the variable A, then prints
DONE.
    OPEN '','SUN.MEMBER' ELSE PRINT "NOT OPEN"
    SSELECT
    READNEXT @ID THEN PRINT @ID
    *
    OPEN '','SUN.SPORT' ELSE PRINT "NOT OPEN"
    SSELECT TO 1
    READNEXT A FROM 1 THEN PRINT "DONE" ELSE PRINT "NOT"
This is the program output:
    0001
    DONE
SSUB
Syntax
       SSUB (string.number.1, string.number.2)
Description
Use the SSUB function to subtract string.number.2 from string.number.1 and return
the result as a string number. You can use this function in any expression where a
string or string number is valid, but not necessarily where a standard number is
valid, because string numbers can exceed the range of numbers that standard
arithmetic operators can handle.
Either string number can be any valid number or string number.
If either string number contains nonnumeric data, an error message is generated,
and 0 replaces the nonnumeric data. If either string number evaluates to the null
value, null is returned.
Example
       X = "123456"
       Y = "225"
       Z = SSUB (X,Y)
       PRINT Z
This is the program output:
       123231
STATUS
Syntax
    STATUS ( )
Description
Use the STATUS function to determine the results of the operations performed by
certain statements and functions.
The parentheses must be used with the STATUS function to distinguish it from
potential user-named variables called STATUS. However, no arguments are
required with the STATUS function.
The following sections describe STATUS function values.
After a BSCAN statement:
0   The scan proceeded beyond the leftmost or rightmost leaf node. ID.variable
    and rec.variable are set to empty strings.
1   The scan returned an existing record ID, or a record ID that matches record.
2   The scan returned a record ID that does not match record. ID.variable is either
    the next or the previous record ID in the B-tree, depending on the direction of
    the scan.
3   The file is not a B-tree (type 25) file, or, if the USING clause is used, the file has
    no active secondary indexes.
4   indexname does not exist.
5   seq does not evaluate to A or D.
6   The index specified by indexname needs to be built.
10 An internal error was detected.
–6 Failed to write to a published file while the subsystem was shut down.
 Value       Description
–1           The filename was not found in the VOC file.
–21          The filename or file is null.
–3           An operating system access error occurs when you do not have
             permission to access a DataStage file in a directory. For example,
             this error may occur when trying to access a type 1 or type 30 file.
–41          An access error appears when you do not have operating system
             permissions or if DATA.30 is missing for a type 30 file.
–5           The operating system detected a read error.
–6           The lock file header cannot be unlocked.
–7           Invalid file revision or wrong byte-ordering exists for the platform.
–81          Invalid part file information exists.
     1
–9           Invalid type 30 file information exists in a distributed file.
–10          A problem occurred while the file was being rolled forward during
             warmstart recovery. Therefore, the file is marked “inconsistent.”
–11          The file is a view; therefore it cannot be opened by a BASIC program.
–12          No SQL privileges exist to open the table.
         1
–13          An index problem exists.
–14          The NFS file cannot be opened.
  1. A generic error that can occur for various reasons.
After a READ statement: If the file is a distributed file, the STATUS function
returns the following:
81007      Connection refused because the server cannot accept more clients.
81008      Error occurred because of a bad parameter in arg.list.
81009      An unspecified RPC error occurred.
81010      #args does not match the expected argument count on the remote
           machine.
81011      Host was not found in the local /etc/hosts file.
81012      Remote dsrpcd cannot start the service because it could not fork the
           process.
81013      The remote dsrpcservices file cannot be opened.
81014      Service was not found in the remote dsrpcservices file.
81015      A timeout occurred while waiting for a response from the server.
Example
STATUS
Syntax
    STATUS dynamic.array FROM file.variable
          {THEN statements [ELSE statements] | ELSE statements}
Description
Use the STATUS statement to determine the status of an open file. The STATUS
statement returns the file status as a dynamic array and assigns it to dynamic.array.
The following table lists the values of the dynamic array returned by the STATUS
statement:
file.variable specifies an open file. If file.variable evaluates to the null value, the
STATUS statement fails and the program terminates with a run-time error
message.
If the STATUS array is assigned to dynamic.array, the THEN statements are
executed and the ELSE statements are ignored. If no THEN statements are present,
program execution continues with the next statement. If the attempt to assign the
array fails, the ELSE statements are executed; any THEN statements are ignored.
Example
      OPENSEQ '/etc/passwd' TO test THEN PRINT "File Opened" ELSE
      ABORT
      STATUS stat FROM test THEN PRINT stat
    field5 = stat<5,1,1>
    field6 = stat<6,1,1>
    field8 = stat<8,1,1>
    PRINT "permissions:": field5
    PRINT "filesize:": field6
    PRINT "userid:": field8
    CLOSESEQ test
This is the program output:
    File Opened
    0F0F0F4164F33188F4164F1F0F2F2303F
        0F6856F59264F6590F42496F6588F42496F6588
        F0F/etc/passwdF0F0F0
    permissions:33188
    filesize:4164
    userid:0
STOP
Syntax
       STOP [expression]
STOPE [expression]
STOPM [expression]
Description
Use the STOP statement to terminate program execution and return system control
to the calling environment, which can be a menu, a paragraph, another BASIC
program, or the DataStage command processor.
When expression is specified, its value is displayed before the STOP statement is
executed. If expression evaluates to the null value, nothing is printed.
To stop all processes and return to the command level, use the ABORT statement.
Use the ERRMSG statement if you want to display a formatted error message from
the ERRMSG file when the program stops.
Example
       PRINT "1+2=":1+2
       STOP "THIS IS THE END"
STORAGE
Syntax
   STORAGE arg1 arg2 arg3
Description
The STORAGE statement performs no function. It is provided for compatibility
with other Pick systems.
STR
Syntax
      STR (string, repeat)
Description
Use the STR function to produce a specified number of repetitions of a particular
character string.
string is an expression that evaluates to the string to be generated.
repeat is an expression that evaluates to the number of times string is to be repeated.
If repeat does not evaluate to a value that can be truncated to a positive integer, an
empty string is returned.
If string evaluates to the null value, null is returned. If repeat evaluates to the null
value, the STR function fails and the program terminates with a run-time error
message.
Example
      PRINT STR('A',10)
      *
      X=STR(5,2)
      PRINT X
      *
      X="HA"
      PRINT STR(X,7)
This is the program output:
      AAAAAAAAAA
      55
      HAHAHAHAHAHAHA
STRS
Syntax
       STRS (dynamic.array, repeat)
       CALL −STRS (return.array, dynamic.array, repeat)
       CALL !STRS (return.array, dynamic.array, repeat)
Description
Use the STRS function to produce a dynamic array containing the specified
number of repetitions of each element of dynamic.array.
dynamic.array is an expression that evaluates to the strings to be generated.
repeat is an expression that evaluates to the number of times the elements are to be
repeated. If it does not evaluate to a value that can be truncated to a positive
integer, an empty string is returned for dynamic.array.
If dynamic.array evaluates to the null value, null is returned. If any element of
dynamic.array is the null value, null is returned for that element. If repeat evaluates
to the null value, the STRS function fails and the program terminates with a run-
time error message.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
       ABC="A":@VM:"B":@VM:"C"
       PRINT STRS(ABC,3)
This is the program output:
       AAAVBBBVCCC
SUBR
Syntax
    SUBR (name, [argument [ ,argument … ] ] )
Description
Use the SUBR function to return the value of an external subroutine. The SUBR
function is commonly used in I-descriptors.
name is an expression that evaluates to the name of the subroutine to be executed.
This subroutine must be cataloged in either a local catalog or the system catalog,
or it must be a record in the same object file as the calling program. If name evalu-
ates to the null value, the SUBR function fails and the program terminates with a
run-time error message.
argument is an expression evaluating to a variable name whose value is passed to
the subroutine. You can pass up to 254 variables to the subroutine.
Subroutines called by the SUBR function must have a special syntax. The
SUBROUTINE statement defining the subroutine must specify a dummy variable
as the first parameter. The value of the subroutine is the value of the dummy vari-
able when the subroutine finishes execution. Because the SUBROUTINE statement
has this dummy parameter, the SUBR function must specify one argument less
than the number of parameters in the SUBROUTINE statement. In other words,
the SUBR function does not pass any argument to the subroutine through the first
dummy parameter. The first argument passed by the SUBR function is referenced
in the subroutine by the second parameter in the SUBROUTINE statement, and so
on.
Example
The following example uses the globally cataloged subroutine *TEST:
    OPEN "","SUN.MEMBER" TO FILE ELSE STOP "CAN'T OPEN DD"
    EXECUTE "SELECT SUN.MEMBER"
    10*
    READNEXT KEY ELSE STOP
    READ ITEM FROM FILE,KEY ELSE GOTO 10
    X=ITEM<7> ;* attribute 7 of file contains year
    Z=SUBR("*TEST",X)
    PRINT "YEARS=", Z
    GOTO 10
SUBROUTINE
Syntax
    SUBROUTINE [name] [ ( [MAT] variable [ , [MAT] variable … ] ) ]
Description
Use the SUBROUTINE statement to identify an external subroutine. The SUBROU-
TINE statement must be the first noncomment line in the subroutine. Each external
subroutine can contain only one SUBROUTINE statement.
An external subroutine is a separate program or set of statements that can be
executed by other programs or subroutines (called calling programs) to perform a
task. The external subroutine must be compiled and cataloged before another
program can call it.
The SUBROUTINE statement can specify a subroutine name for documentation
purposes; it need not be the same as the program name or the name by which it is
called. The CALL statement must reference the subroutine by its name in the
catalog, in the VOC file, or in the object file.
variables are variable names used in the subroutine to pass values between the
calling programs and the subroutine. To pass an array, you must precede the array
name with the keyword MAT. When an external subroutine is called, the CALL
statement must specify the same number of variables as are specified in the
SUBROUTINE statement. See the CALL statement for more information.
Example
The following SUBROUTINE statements specify three variables, EM, GROSS, and
TAX, the values of which are passed to the subroutine by the calling program:
    SUBROUTINE ALONE(EM, GROSS, TAX)
SUBROUTINE STATE(EM,GROSS,TAX)
SUBS
Syntax
       SUBS (array1, array2)
       CALL −SUBS (return.array, array1, array2)
       CALL !SUBS (return.array, array1, array2)
Description
Use the SUBS function to create a dynamic array of the element-by-element
subtraction of two dynamic arrays.
Each element of array2 is subtracted from the corresponding element of array1 with
the result being returned in the corresponding element of a new dynamic array.
If an element of one dynamic array has no corresponding element in the other
dynamic array, the missing element is evaluated as 0. If either of a corresponding
pair of elements is the null value, null is returned for that element.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
       A=2:@VM:4:@VM:6:@SM:18
       B=1:@VM:2:@VM:3:@VM:9
       PRINT SUBS(A,B)
This is the program output:
       1V2V3S18V-9
SUBSTRINGS
Syntax
    SUBSTRINGS (dynamic.array, start, length)
    CALL −SUBSTRINGS (return.array, dynamic.array, start, length)
    CALL !SUBSTRINGS (return.array, dynamic.array, start, length)
Description
Use the SUBSTRINGS function to create a dynamic array each of whose elements
are substrings of the corresponding elements of dynamic.array.
start indicates the position of the first character of each element to be included in
the substring. If start is 0 or a negative number, the starting position is assumed to
be 1. If start is greater than the number of characters in the element, an empty string
is returned.
length specifies the total length of the substring. If length is 0 or a negative number,
an empty string is returned. If the sum of start and length is larger than the element,
the substring ends with the last character of the element.
If an element of dynamic.array is the null value, null is returned for that element. If
start or length evaluates to the null value, the SUBSTRINGS function fails and the
program terminates with a run-time error message.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
Example
    A="ABCDEF":@VM:"GH":@SM:"IJK"
    PRINT SUBSTRINGS(A,3,2)
This is the program output:
    CDVSK
SUM
Syntax
      SUM (dynamic.array)
Description
Use the SUM function to calculate the sum of numeric data. Only elements at the
lowest delimiter level of a dynamic array are summed. The total is returned as a
single element at the next highest delimiter level.
The delimiters from highest to lowest are field, value, and subvalue.
There are seven levels of delimiters from CHAR(254) to CHAR(248): field mark,
value mark, subvalue mark, text mark, CHAR(250), CHAR(249), and CHAR(248).
The SUM function removes the lowest delimiter level from a dynamic array. In a
dynamic array that contains fields, values, and subvalues, the SUM function sums
only the subvalues, returning the sums as values. In a dynamic array that contains
fields and values, the SUM function sums only the values, returning the sums as
fields. In a dynamic array that contains only fields, the SUM function sums the
fields, returning the sum as the only field of the array. SUM functions can be
applied repeatedly to raise multilevel data to the highest delimiter level or to a
single value.
Nonnumeric values, except the null value, are treated as 0. If dynamic.array evalu-
ates to the null value, null is returned. Any element that is the null value is ignored,
unless all elements of dynamic.array are null, in which case null is returned.
Examples
In the following examples a field mark is shown by F, a value mark is shown by V,
and a subvalue mark is shown by S.
 X=17:@FM:18:@FM:15                                 SUM(X)+SUM(Y)=           80
 Y=10:@FM:20
 PRINT "SUM(X)+SUM(Y)= ",SUM(X)+SUM(Y)
 X=3:@SM:4:@SM:10:@VM:3:@VM:20   Y=   17V3V20
 Y=SUM(X)                        Z=   40
 PRINT "Y= ",Y
 Z=SUM(Y)
 PRINT "Z= ",Z
SUMMATION
Syntax
    SUMMATION (dynamic.array)
    CALL !SUMMATION (result, dynamic.array)
Description
Use the SUMMATION function to return the sum of all the elements in
dynamic.array. Nonnumeric values, except the null value, are treated as 0.
result is a variable containing the result of the sum.
dynamic.array is the dynamic array whose elements are to be added together.
Example
    A=1:@VM:"ZERO":@SM:20:@FM:-25
    PRINT "SUMMATION(A)=",SUMMATION(A)
This is the program output:
    SUMMATION(A)=-4
SYSTEM
Syntax
      SYSTEM (expression)
Description
Use the SYSTEM function to check on the status of a system function. Use the
SYSTEM function to test whether NLS is on when you run a program, and to
display information about NLS settings.
expression evaluates to the number of the system function you want to check. If
expression evaluates to the null value, the SYSTEM function fails and the program
terminates with a run-time error message.
The following table lists the values for expression and their meanings. Values 100
through 107 (read-only) for the SYSTEM function contain NLS information. See the
include file UVNLS.H for their tokens.
Value          Action
1              Checks to see if the PRINTER ON statement has turned the printer
               on. Returns 1 if the printer is on and 0 if it is not.
2              Returns the page width as defined by the terminal characteristic
               settings.
3              Returns the page length as defined by the terminal characteristic
               settings.
4              Returns the number of lines remaining on the current page.
5              Returns the current page number.
6              Returns the current line number.
7              Returns the terminal code for the type of terminal the system
               believes you are using.
8,n            Checks whether the tape is attached. Returns the current block size
               if it is and −1 if it is not. n is the number of the tape unit. If it is not
               specified, tape unit 0 is assumed.
9              Returns the current CPU millisecond count.
10             Checks whether the DATA stack is active. Returns 1 if it is active
               and 0 if it is not.
Value   Action
11      Checks whether select list 0 is active. Returns 1 if select list 0 is
        active and 0 if it is not.
12      By default, returns the current system time in seconds (local time).
        If the TIME.MILLISECOND option is set (see $OPTIONS), returns
        the current system time in milliseconds.
13      Not used. Returns 0.
14      Not used. Returns 0.
15      Not used. Returns 0.
16      Returns 1 if running from a proc, otherwise returns 0.
17      Not used. Returns 0.
18      Returns the terminal number.
19      Returns the login name.
20      Not used. Returns 0.
21      Not used. Returns 0.
22      Not used. Returns 0.
23      Checks whether the Break key is enabled. Returns 1 if the Break
        key is enabled and 0 if it is not.
24      Checks whether character echoing is enabled. Returns 1 if char-
        acter echoing is enabled and 0 if it is not.
25      Returns 1 if running from a phantom process, otherwise returns 0.
26      Returns the current prompt character.
27      Returns the user ID of the person using the routine.
28      Returns the effective user ID of the person using the routine.
        Windows NT. This is the same value as 27.
29      Returns the group ID of the person using the routine.
        Windows NT. This value is 0.
30      Returns the effective group ID of the person using the routine.
        Windows NT. This value is 0.
31      Returns the DataStage serial number.
32      Returns the location of the UV account directory.
Value        Action
33           Returns the last command on the command stack.
34           Returns data pending.
35           Returns the number of users currently in DataStage.
36           Returns the maximum number of DataStage users.
37           Returns the number of UNIX users; on Windows NT systems
             returns same value as 35.
38           Returns the pathname of the temporary directory.
42           Returns an empty string. On Windows NT systems returns the
             current value of the telnet client’s IP address, or an empty string if
             the process evaluating the SYSTEM function is not the main telnet
             process.
43           Returns 1 if db suspension is on, returns 0 if it is not.
50           Returns the field number of the last READNEXT statement when
             reading an exploded select list.
60           Returns the current value of the DataStage configurable parameter
             TXMODE. The value can be either 1 or 0.
61           Returns the status of the transaction log daemon. 1 indicates the
             daemon is active; 0 indicates it is inactive.
91           Returns 0; on Windows NT, returns 1.
99           Returns the system time in the number of seconds since midnight
             Greenwich Mean Time (GMT), January 1, 1970.
100          Returns 1 if NLS is enabled, otherwise returns 0.
101          Returns the value of the NLSLCMODE parameter, otherwise
             returns 0.
102          Reserved for future NLS extensions.
103          Returns the terminal map name assigned to the current terminal
             print channel, otherwise returns 0.
104          Returns the auxiliary printer map name assigned to the current
             terminal print channel, otherwise returns 0.
Value      Action
105        Returns a dynamic array, with field marks separating the
           elements, containing the current values of the uvconfig file parame-
           ters for NLS maps, otherwise returns 0. See the UVNLS.H include
           file for a list of tokens that define the field order.
106        Returns the current map name used for sequential I/O. Token is
           NLS$SEQMAP unless overridden by a SET.SEQ.MAP command.
107        Returns the current map name for GCI string arguments unless
           overridden by a SET.GCI.MAP command.
1001       Returns the DataStage flavor: 1 for IDEAL, 2 for PICK, 4 for
           INFORMATION, 8 for REALITY, 16 for IN2, and 64 for PIOPEN.
1017       Returns the user’s supplementary UNIX groups in a dynamic
           array.
1021       Returns the GCI error number.
1200,      Returns the link number associated with hostname. If there is an
hostname   internal error adding hostname, 0 returns. hostname is an expression
           that contains the host name from a file opened remotely. It refers to
           the host name portion of the file’s pathname. For example, in the
           pathname ORION!/u1/filename, hostname is ORION.
1201,      Returns the RPC connection number associated with hostname. The
hostname   REMOTE.B interface program uses this number. If there is an
           internal error adding hostname, or if RPC has not yet opened, 0
           returns. If the RPC connection was opened but is now closed, –1
           returns.
1202,      Returns the timeout associated with hostname. If there is no
hostname   timeout associated with hostname, 0 returns.
1203       Returns the last RPC connection error number. This number is in
           the range 81000 through 81999. 81015 indicates that a timeout
           occurred. These error numbers correspond to error messages in the
           SYS.MESSAGE file.
Examples
The first example returns the number of lines left to print on a page, with the
maximum defined by the TERM command. The second example returns the
current page number.
PRINT 'X=',SYSTEM(5) X= 0
The next example sets a 30-second timeout for the network connection to the
system ORION:
    TIMEOUT SYSTEM(1200, "ORION"), 30
TABSTOP
Syntax
    TABSTOP expression
Description
Use the TABSTOP statement to set the current tabstop width for PRINT state-
ments. The initial tabstop setting is 10.
If expression evaluates to the null value, the TABSTOP statement fails and the
program terminates with a run-time error message.
Example
    A="FIRST"
    B="LAST"
    PRINT A,B
    TABSTOP 15
    PRINT A,B
This is the program output:
    FIRST   LAST
    FIRST      LAST
TAN
Syntax
      TAN (expression)
Description
Use the TAN function to return the trigonometric tangent of expression. expression
represents an angle expressed in degrees.
Trying to take the tangent of a right angle results in a warning message, and a
return value of 0. Numbers greater than 1E17 produce a warning message, and 0
is returned. If expression evaluates to the null value, null is returned.
Example
      PRINT TAN(45)
This is the program output:
      1
TANH
Syntax
    TANH (expression)
Description
Use the TANH function to return the hyperbolic tangent of expression. expression
must be numeric and represents the angle expressed in degrees. If expression eval-
uates to the null value, null is returned.
Example
    PRINT TANH(45)
This is the program output:
    1
TERMINFO
Syntax
    TERMINFO (argument)
Description
Use the TERMINFO function to access the device-independent terminal handler
string defined for the current terminal type. The TERMINFO function returns a
dynamic array containing the terminal characteristics for the terminal type set by
TERM or SET.TERM.TYPE.
argument can be 0 or 1, depending on whether the terminal characteristics are
returned as stored, or converted to printable form. If argument is 0, the function
returns the terminal characteristics in the form usable by BASIC applications for
device-independent terminal handling with the TPARM function and the TPRINT-
statement. If argument is 1, the function returns characteristics in terminfo source
format. Boolean values are returned as Y = true and N = false. The terminfo files
contain many unprintable control characters that may adversely affect your
terminal.
If argument evaluates to the null value, the TERMINFO function fails and the
program terminates with a run-time error message.
The easiest way to access the terminfo characteristics is by including the BASIC file
UNIVERSE.INCLUDE TERMINFO in your program. The syntax is:
    $INCLUDE UNIVERSE.INCLUDE TERMINFO
The file contains lines that equate each dynamic array element returned by
TERMINFO with a name, so that each element can be easily accessed in your
program. Once this file has been included in your program, you can use the
defined names to access terminal characteristics. The following table lists the
contents of this file:
                              TERMINFO EQUATEs
 terminfo$ = terminfo(0)
 EQU TERMINAL.NAME                                        TO terminfo$<1>
 EQU COLUMNS                                              TO terminfo$<2>
 EQU LINES                                                TO terminfo$<3>
 EQU CARRIAGE.RETURN                                      TO terminfo$<4>
 EQU LINE.FEED                                            TO terminfo$<5>
Example
    $INCLUDE UNIVERSE.INCLUDE TERMINFO
    PRINT AT.NEGATIVE.1
    PRINT "Your terminal type is":TAB:TERMINAL.NAME
The program output on the cleared screen is:
    Your terminal type is icl6404|ICL 6404CG Color Video Display
TIME
TIME
Syntax
       TIME ( )
Description
Use the TIME function to return a string value expressing the internal time of day.
The internal time is the number of seconds that have passed since midnight to the
nearest thousandth of a second (local time).
The parentheses must be used with the TIME function to distinguish it from a user-
named variable called TIME. However, no arguments are required with the TIME
function.
UNIX System V. The time is returned only to the nearest whole second.
If the TIME.MILLISECOND option of the $OPTIONS statement is set, the TIME
function returns the system time in whole seconds.
Example
       PRINT TIME()
This is the program output:
       40663.842
TIMEDATE
Syntax
     TIMEDATE ( )
Description
Use the TIMEDATE function to return the current system time and date in the
following format:
     hh:mm:ss dd mmm yyyy
Examples
     PRINT TIMEDATE()
This is the program output:
     11:19:07 18 JUN 1996
If the TMD0001 message contains four #s, the program output is:
     11:19:07       18 JUN 1996
TIMEOUT
Syntax
    TIMEOUT     {file.variable | link.number}, time
Description
Use the TIMEOUT statement to terminate a READSEQ or READBLK statement if
no data is read in the specified time. You can also use the TIMEOUT statement to
set a time limit for a network link. Use the TTYGET and TTYSET statements to set
a timeout value for a file open on a serial communications port.
The TIMEOUT statement is not supported on Windows NT.
file.variable specifies a file opened for sequential access.
time is an expression that evaluates to the number of seconds the program should
wait before terminating the READSEQ or READBLK statement or the network
connections.
link.number is the network link. It is a positive number from 1 through 255 (or the
number set in the NET_MAXCONNECT VALUE for network connections).
TIMEOUT causes subsequent READSEQ and READBLK statement to terminate
and execute their ELSE statements if the number of seconds specified by time
elapses while waiting for data. Use the STATUS function to determine if time has
elapsed. In the event of a timeout, neither READBLK nor READSEQ returns any
bytes from the buffer, and the entire I/O operation must be retried.
If either file.variable or time evaluates to the null value, the TIMEOUT statement
fails and the program terminates with a run-time error message.
Examples
    TIMEOUT SUN.MEMBER, 10
    READBLK VAR1 FROM SUN.MEMBER, 15 THEN PRINT VAR1 ELSE
        IF STATUS() = 2 THEN
         PRINT "TIMEOUT OCCURRED"
        END ELSE
         PRINT "CANNOT OPEN FILE"
        END
        GOTO EXIT.PROG
    END
TPARM
Syntax
     TPARM (terminfo.string, [arg1], [arg2], [arg3], [arg4], [arg5], [arg6], [arg7],
             [arg8] )
Description
Use the TPARM function to evaluate a parameterized terminfo string.
terminfo.string represents a string of characters to be compiled by the terminfo
compiler, tic. These terminal descriptions define the sequences of characters to
send to the terminal to perform special functions. terminfo.string evaluates to one
of four types of capability: numeric, Boolean, string, or parameterized string. If
terminfo.string or any of the eight arguments evaluates to the null value, the
TPARM function fails and the program terminates with a run-time error message.
Numeric capabilities are limited to a length of five characters that must form a
valid number. Only nonnegative numbers 0 through 32,767 are allowed. If a value
for a particular capability does not apply, the field should be left blank.
Boolean capabilities are limited to a length of one character. The letter Y (in either
uppercase or lowercase) indicates that the specified capability is present. Any
value other than Y indicates that the specified capability is not present.
String capabilities are limited to a length of 44 characters. You can enter special
characters as follows:
%+ %− %* %/
            The top two elements are popped off the stack and added, subtracted,
            multiplied, or divided. The result is pushed back on the stack. The
            fractional portion of a quotient is discarded.
%m          The second element on the stack is taken modulo of the first element,
            and the result is pushed onto the stack.
%& % | %^ The top two elements are popped off the stack and a bitwise AND,
          OR, or XOR operation is performed. The result is pushed onto the
          stack.
%= %< %> The second element on the stack is tested for being equal to, less then,
         or greater than the first element. If the comparison is true, a 1 is
         pushed onto the stack, otherwise a 0 is pushed.
%! %~       The stack is popped, and either the logical or the bitwise NOT of the
            first element is pushed onto the stack.
%i          One (1) is added to the first two parameters. This is useful for termi-
            nals that use a one-based cursor address, rather than a zero-based.
%Px         Pop the stack, and put the result into variable x, where x is a lower-
            case letter (a − z).
%gx         Push the value of variable x on the top of the stack.
%? exp %t exp [%e exp] %;
            Form an if-then-else expression, with "%?" representing "IF", "%t"
            representing "THEN", "%e" representing "ELSE", and "%;" termi-
            nating the expression. The else expression is optional. Else-If
            expressions are possible. For example:
                %? C1 %t B1 %e C2 %t B2 %e C3 %t B3 %e C4 %t B4 %e %
            Cn are conditions, and Bn are bodies.
%%          Output a percent sign (%).
A delay in milliseconds can appear anywhere in a string capability. A delay is spec-
ified by $<nnn>, where nnn is a decimal number indicating the number of
milliseconds (one thousandth of a second) of delay desired. A proper number of
delay characters will be output, depending on the current baud rate.
TPRINT
Syntax
    TPRINT [ON print.channel] [print.list]
Description
Use the TPRINT statement to send data to the screen, a line printer, or another print
file. TPRINT is similar to the PRINT statement, except that TPRINT lets you
specify time delay expressions in the print list.
The ON clause specifies the logical print channel to use for output. print.channel is
an expression that evaluates to a number from –1 through 255. If you do not use
the ON clause, logical print channel 0 is used, which prints to the user’s terminal
if PRINTER OFF is set (see the PRINTER statement). If print.channel evaluates to
the null value, the TPRINT statement fails and the program terminates with a run-
time error message. Logical print channel –1 prints the data on the screen, regard-
less of whether a PRINTER ON statement has been executed.
You can specify HEADING, FOOTING, PAGE, and PRINTER CLOSE statements
for each logical print channel. The contents of the print files are printed in order by
logical print channel number.
print.list can contain any BASIC expression. The elements of the list can be numeric
or character strings, variables, constants, or literal strings. The list can consist of a
single expression or a series of expressions separated by commas ( , ) or colons ( : )
for output formatting. If no print.list is designated, a blank line is printed. The null
value cannot be printed.
print.list can also contain time delays of the form $<time>. time is specified in milli-
seconds to the tenth of a millisecond. As the print list is processed, each time delay
is executed as it is encountered.
Expressions separated by commas are printed at preset tab positions. The default
tabstop setting is 10 characters. See the TABSTOP statement for information about
changing the default setting. Use multiple commas together for multiple tabula-
tions between expressions.
Expressions separated by colons are concatenated. That is, the expression
following the colon is printed immediately after the expression preceding the
colon. To print a list without a LINEFEED and RETURN, end print.list with a colon
( : ).
If NLS is enabled, the TPRINT statement maps data in the same way as the PRINT
statement. For more information about maps, see the DataStage NLS Guide.
Example
The following example prints the string ALPHA followed by a delay of 1 second,
then the letters in the variable X. The printing of each letter is followed by a delay
of one tenth of a second.
    X="A$<100>B$<100>C$<100>D$<100>E"
    TPRINT "ALPHA$<1000.1> ":X
This is the program output:
    ALPHA ABCDE
TRANS
Syntax
    TRANS ( [DICT] filename, record.ID, field#, control.code)
Description
Use the TRANS function to return the contents of a field or a record in a DataStage
file. TRANS opens the file, reads the record, and extracts the specified data.
filename is an expression that evaluates to the name of the remote file. If TRANS
cannot open the file, a run-time error occurs, and TRANS returns an empty string.
record.ID is an expression that evaluates to the ID of the record to be accessed. If
record.ID is multivalued, the translation occurs for each record ID and the result is
multivalued (system delimiters separate data translated from each record).
field# is an expression that evaluates to the number of the field from which the data
is to be extracted. If field# is −1, the entire record is returned, except for the
record ID.
control.code is an expression that evaluates to a code specifying what action to take
if data is not found or is the null value. The possible control codes are:
X   (default) Returns an empty string if the record does not exist or data cannot
    be found.
V   Returns an empty string and produces an error message if the record does not
    exist or data cannot be found.
C   Returns the value of record.ID if the record does not exist or data cannot be
    found.
N Returns the value of record.ID if the null value is found.
The returned value is lowered. For example, value marks in the original field
become subvalue marks in the returned value. For more information, see the
LOWER function.
If filename, record.ID, or field# evaluates to the null value, the TRANS function fails
and the program terminates with a run-time error message. If control.code evaluates
to the null value, null is ignored and X is used.
The TRANS function is the same as the XLATE function.
Example
    X=TRANS("VOC","EX.BASIC",1,"X")
    PRINT "X= ":X
    *
    FIRST=TRANS("SUN.MEMBER","6100",2,"X")
    LAST=TRANS("SUN.MEMBER","6100",1,"X")
    PRINT "NAME IS ":FIRST:" ":LAST
This is the program output:
    X= F BASIC examples file
    NAME IS BOB MASTERS
BEGIN TRANSACTION
Syntax
    BEGIN TRANSACTION
       [statements]
    { COMMIT [WORK] | ROLLBACK [WORK] }
        [statements]
    [{ COMMIT [WORK] | ROLLBACK [WORK] }
        [statements]
      .
      .
      .      ]
    END TRANSACTION
Syntax (PIOPEN)
    TRANSACTION START
       {THEN statements [ELSE statements] | ELSE statements}
    TRANSACTION COMMIT
       {THEN statements [ELSE statements] | ELSE statements}
    TRANSACTION ABORT
Description
Use transaction statements to treat a sequence of file I/O operations as one logical
operation with respect to recovery and visibility to other users. These operations
can include file I/O operations or subtransactions.
Note: BASIC accepts PI/open syntax in addition to DataStage syntax. You cannot
      mix both types of syntax within a program.
TRANSACTION ABORT
Syntax
    TRANSACTION ABORT
Description
Use the TRANSACTION ABORT statement to cancel all file I/O changes made
during a transaction.
You can use the TRANSACTION ABORT statement in a transaction without a
TRANSACTION COMMIT statement to review the results of a possible change.
Doing so does not affect the parent transaction or the database.
After the transaction ends, execution continues with the statement following the
TRANSACTION ABORT statement.
Example
The following example shows the use of the TRANSACTION ABORT statement to
terminate a transaction if both the ACCOUNTS RECEIVABLE file and the INVEN-
TORY file cannot be successfully updated:
    PROMPT ''
    OPEN 'ACC.RECV' TO ACC.RECV ELSE STOP 'NO OPEN ACC.RECV'
    OPEN 'INVENTORY' TO INVENTORY ELSE STOP 'NO OPEN INVENTORY'
END
TRANSACTION COMMIT
Syntax
    TRANSACTION COMMIT
            {THEN statements [ELSE statements] | ELSE statements}
Description
Use the TRANSACTION COMMIT statement to commit all file I/O changes made
during a transaction.
The TRANSACTION COMMIT statement can either succeed or fail. If the TRANS-
ACTION COMMIT statement succeeds, the THEN statements are executed; any
ELSE statements are ignored. If the TRANSACTION COMMIT statement fails, the
ELSE statements, if present, are executed, and control is transferred to the state-
ment following the TRANSACTION COMMIT statement.
TRANSACTION START
Syntax
    TRANSACTION START
            {THEN statements [ELSE statements] | ELSE statements}
Description
Use the TRANSACTION START statement to begin a new transaction.
TRIM
Syntax
       TRIM (expression [ ,character [ ,option] ] )
Description
Use the TRIM function to remove unwanted characters in expression. If only expres-
sion is specified, multiple occurrences of spaces and tabs are reduced to a single tab
or space, and all leading and trailing spaces and tabs are removed. If expression
evaluates to one or more space characters, TRIM returns an empty string.
character specifies a character other than a space or a tab. If only expression and char-
acter are specified, multiple occurrences of character are replaced with a single
occurrence, and leading and trailing occurrences of character are removed.
option specifies the type of trim operation to be performed:
If expression evaluates to the null value, null is returned. If option evaluates to the
null value, null is ignored and option R is assumed. If character evaluates to the null
value, the TRIM function fails and the program terminates with a run-time error
message.
If NLS is enabled, you can use TRIM to remove other white space characters such
as Unicode values 0x2000 through 0x200B, 0x00A0, and 0x3000, marked as TRIM-
MABLE in the NLS.LC.CTYPE file entry for the specified locale. For more
information about Unicode values, see the DataStage NLS Guide.
Example
    A=" Now is the time       for   all good men to"
    PRINT A
    PRINT TRIM(A)
This is the program output:
      Now is the time for all good men to
    Now is the time for all good men to
TRIMB
Syntax
    TRIMB (expression)
Description
Use the TRIMB function to remove all trailing spaces and tabs from expression. All
other spaces or tabs in expression are left intact. If expression evaluates to the null
value, null is returned.
If NLS is enabled, you can use TRIMB to remove white space characters such as
Unicode values 0x2000 through 0x200B, 0x00A0, and 0x3000, marked as TRIM-
MABLE in the NLS.LC.CTYPE file entry for the specified locale. For more
information about Unicode values, see the DataStage NLS Guide.
Example
    A=" THIS IS A SAMPLE STRING "
    PRINT "'":A:"'": " IS THE STRING"
    PRINT "'":TRIMB(A):"'":" IS WHAT TRIMB DOES"
    END
This is the program output:
    '   THIS IS A     SAMPLE STRING ' IS THE STRING
    '   THIS IS A     SAMPLE STRING' IS WHAT TRIMB DOES
TRIMBS
Syntax
    TRIMBS (dynamic.array)
    CALL −TRIMBS (return.array, dynamic.array)
Description
Use the TRIMBS function to remove all trailing spaces and tabs from each element
of dynamic.array.
TRIMBS removes all trailing spaces and tabs from each element and reduces
multiple occurrences of spaces and tabs to a single space or tab.
If dynamic.array evaluates to the null value, null is returned. If any element of
dynamic.array is null, null is returned for that value.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If NLS is enabled, you can use TRIMBS to remove white space characters such as
Unicode values 0x2000 through 0x200B, 0x00A0, and 0x3000, marked as TRIM-
MABLE in the NLS.LC.CTYPE file entry for the specified locale. For more
information about Unicode values, see the DataStage NLS Guide.
TRIMF
Syntax
    TRIMF (expression)
Description
Use the TRIMF function to remove all leading spaces and tabs from expression. All
other spaces or tabs in expression are left intact. If expression evaluates to the null
value, null is returned.
If NLS is enabled, you can use TRIMF to remove white space characters such as
Unicode values 0x2000 through 0x200B, 0x00A0, and 0x3000, marked as TRIM-
MABLE in the NLS.LC.CTYPE file entry for the specified locale. For more
information about Unicode values, see the DataStage NLS Guide.
Example
    A=" THIS IS A SAMPLE STRING "
    PRINT "'":A:"'":" IS THE STRING"
    PRINT "'":TRIMF(A):"'":" IS WHAT TRIMF DOES"
    END
This is the program output:
    ' THIS IS A SAMPLE STRING ' IS THE STRING
    'THIS IS A SAMPLE STRING ' IS WHAT TRIMF DOES
TRIMFS
Syntax
    TRIMFS (dynamic.array)
    CALL −TRIMFS (return.array, dynamic.array)
Description
Use the TRIMFS function to remove all leading spaces and tabs from each element
of dynamic.array.
TRIMFS removes all leading spaces and tabs from each element and reduces
multiple occurrences of spaces and tabs to a single space or tab.
If dynamic.array evaluates to the null value, null is returned. If any element of
dynamic.array is null, null is returned for that value.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If NLS is enabled, you can use TRIMFS to remove white space characters such as
Unicode values 0x2000 through 0x200B, 0x00A0, and 0x3000, marked as TRIM-
MABLE in the NLS.LC.CTYPE file entry for the specified locale. For more
information about Unicode values, see the DataStage NLS Guide.
TRIMS
Syntax
    TRIMS (dynamic.array)
    CALL −TRIMS (return.array, dynamic.array)
Description
Use the TRIMS function to remove unwanted spaces and tabs from each element
of dynamic.array.
TRIMS removes all leading and trailing spaces and tabs from each element and
reduces multiple occurrences of spaces and tabs to a single space or tab.
If dynamic.array evaluates to the null value, null is returned. If any element of
dynamic.array is null, null is returned for that value.
If you use the subroutine syntax, the resulting dynamic array is returned as
return.array.
If NLS is enabled, you can use TRIMS to remove white space characters such as
Unicode values 0x2000 through 0x200B, 0x00A0, and 0x3000, marked as TRIM-
MABLE in the NLS.LC.CTYPE file entry for the specified locale. For more
information about Unicode values, see the DataStage NLS Guide.
TTYCTL
Syntax
     TTYCTL file.variable, code#
           {THEN statements [ELSE statements] | ELSE statements}
Description
Use the TTYCTL statement to set terminal device characteristics on Berkeley
terminal drivers. code# specifies the action to take.
This statement is not supported on UNIX System V or Windows NT.
The following table lists the available actions:
Argument        Action
0               No operation, determines if a device is a TTY.
1               Sets HUP (hang up data line) on close of file.
2               Clears HUP on close of file.
3               Sets exclusive use flag for TTY.
4               Resets exclusive use flag.
5               Sets the BREAK.
6               Clears the BREAK.
7               Turns on DTR (Data Terminal Ready).
8               Turns off DTR.
9               Flushes input and output buffers.
10              Waits for the output buffer to drain.
Example
    OPENSEQ 'FILE.E', 'RECORD4' TO FILE ELSE ABORT
    *
    TTYCTL FILE, 0
       THEN PRINT 'THE FILE IS A TTY'
       ELSE PRINT 'THE FILE IS NOT A TTY'
This is the program output:
    THE FILE IS NOT A TTY
TTYGET
Syntax
    TTYGET variable [FROM {file.variable | LPTR [n] | MTU [n] } ]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the TTYGET statement to assign the characteristics of a terminal, line printer
channel, or tape unit as a dynamic array to variable. If the FROM clause is omitted,
a dynamic array of the terminal characteristics for your terminal is assigned to
variable.
file.variable is a terminal opened for sequential processing with the OPENDEV or
OPENSEQ statement. If file.variable is specified, the terminal characteristics for the
specified terminal are retrieved.
n specifies a logical print channel with LPTR or a tape unit with MTU. (You cannot
specify a tape unit on Windows NT.) If n is specified, the characteristics for the
print channel or tape unit are retrieved. For logical print channels n is in the range
of 0 through 225; the default is 0. For tape units n is in the range of 0 through 7; the
default is 0.
If the terminal characteristics are retrieved, the THEN statements are executed.
If the device does not exist or cannot be opened, or if no dynamic array is returned,
the ELSE statements are executed; any THEN statements are ignored.
If either file.variable or n evaluates to the null value, the TTYGET statement fails and
the program terminates with a run-time error message.
The best way to access the information in the dynamic array is to include the
BASIC code UNIVERSE.INCLUDE TTY. The syntax for including this file is:
    $INCLUDE UNIVERSE.INCLUDE TTY
This file equates each value of the dynamic array to a name, so that each value can
be easily accessed in your program. To take advantage of this code you must call
variable tty$. Once this code has been included in your program, you can use the
names to access the values of the dynamic array. To set values for a terminal line,
use the TTYSET statement.
The following table lists the equate names to the values of the dynamic array, and
describes each value. The final columns indicate which values are available on
                                                                              Avail-
Value     Name              Description
                                                                              ability
                                                                          S             N
                                                                                B
                                                                          V             T
Field 1
1         MODE.TYPE          One of these modes:
                             MODE$LINE or 0 = line                        ✓     ✓
                             MODE$RAW or 1 = raw                          ✓     ✓       ✓
                             MODE$CHAR or 2 = character                   ✓     ✓       ✓
                             MODE$EMULATE or 3 = emulated                 ✓     ✓
2         MODE.MIN           Minimum number of characters before input.   ✓     ✓       ✓
3         MODE.TIME          Minimum time in milliseconds before input.   ✓     ✓       ✓
Field 2
1         CC.INTR            Interrupt character. −1 undefined.           ✓     ✓       ✓
2         CC.QUIT            Quit character. −1 undefined.                ✓     ✓
3         CC.SUSP            Suspend character. −1 undefined.             ✓     ✓
4         CC.DSUSP           dsusp character. −1 undefined.                     ✓
5         CC.SWITCH          Switch character. −1 undefined.              ✓
6         CC.ERASE           erase character. −1 undefined.               ✓     ✓       ✓
7         CC.WERASE          werase character. −1 undefined.                    ✓
8         CC.KILL            Kill character. −1 undefined.                ✓     ✓       ✓
9         CC.LNEXT           lnext character. −1 undefined.                     ✓
10        CC.RPRINT          rprint character. −1 undefined.                    ✓       ✓
11        CC.EOF             eof character. −1 undefined.                 ✓     ✓
12        CC.EOL             eol character. −1 undefined.                 ✓     ✓
13        CC.EOL2            eol2 character. −1 undefined.                ✓
14        CC.FLUSH           Flush character. −1 undefined.                     ✓
                                                                                  Avail-
Value     Name                Description
                                                                                  ability
                                                                              S             N
                                                                                    B
                                                                              V             T
15        CC.START            Start character. −1 undefined.                  ✓     ✓       ✓
                              On System V, ^Q only.
16        CC.STOP             Stop character. −1 undefined.                   ✓     ✓       ✓
                              On System V, ^S only.
17        CC.LCONT            lcont character. −1 undefined. Emulated only.   ✓     ✓       ✓
18        CC.FMC              fmc character. −1 undefined. Emulated only.     ✓     ✓       ✓
19        CC.VMC              vmc character. −1 undefined. Emulated only.     ✓     ✓       ✓
20        CC.SMC              smc character. −1 undefined. Emulated only.     ✓     ✓       ✓
21        CCDEL               Delete character.                               ✓     ✓
Field 3
1         CARRIER.RECEIVE     Terminal can receive data.                      ✓     ✓       ✓
2         CARRIER.HANGUP      Hang up upon close of terminal.                 ✓     ✓
3         CARRIER.LOCAL       Terminal is a local line.                       ✓     ✓       ✓
Field 4
1         CASE.UCIN           Convert lowercase to uppercase on input.        ✓     ✓
2         CASE.UCOUT          Convert lowercase to uppercase on output.       ✓     ✓
3         CASE.XCASE          Uppercase is preceded by a backslash ( \ ) to   ✓     ✓
                              distinguish it from lowercase.
4         CASE.INVERT         Invert case on input. Emulated only.            ✓     ✓       ✓
Field 5
1         CRMODE.INLCR        Convert LINEFEED to RETURN on input.            ✓     ✓
2         CRMODE.IGNCR        Ignore RETURN on input.                         ✓     ✓
3         CRMODE.ICRNL        Convert RETURN to LINEFEED on input.            ✓     ✓
4         CRMODE.ONLCR        Convert LINEFEED to LINEFEED, RETURN            ✓     ✓
                              on output.
5         CRMODE.OCRNL        Convert RETURN to LINEFEED on output.           ✓     ✓
                                                                               Avail-
Value     Name               Description
                                                                               ability
                                                                           S             N
                                                                                 B
                                                                           V             T
6         CRMODE.ONOCR       Prohibit output of RETURN when cursor         ✓     ✓
                             is in column 0.
7         CRMODE.ONLRET      LINEFEED performs RETURN function.            ✓     ✓
Field 6
1         DELAY.BS           Set backspace delay.                          ✓     ✓
2         DELAY.CR           Set RETURN delay.                             ✓     ✓
3         DELAY.FF           Set formfeed delay.                           ✓     ✓
4         DELAY.LF           Set LINEFEED delay.                           ✓     ✓
5         DELAY.VT           Set vertical tab delay.                       ✓     ✓
6         DELAY.TAB          Set tab delay.                                ✓     ✓
7         DELAY.FILL         0 = time delay                                ✓     ✓
                             1 = fill with empty strings
                             2 = fill with DELETEs
Field 7
1         ECHO.ON            Set terminal echo on.                         ✓     ✓       ✓
2         ECHO.ERASE         ECHOE$ERASE or 0 = print echo character       ✓     ✓
                             ECHOE$BS or 1 = echo as backspace
                             ECHOE$BSB or 2 = echo as backspace, space,
                             backspace
                             ECHOE$PRINTER or 3 = echo as a printer
3         ECHO.KILL          ECHOK$KILL or 0 = kill as kill character      ✓     ✓
                             ECHOK$LF or 1 = kill as RETURN,
                             LINEFEED
                             ECHOK$ERASE or 2 = kill as series of erases
4         ECHO.CTRL          Set control to echo as ^ character            ✓     ✓
5         ECHO.LF            When echo is off, echo RETURN as              ✓     ✓       ✓
                             RETURN, LINEFEED
                                                                                Avail-
Value      Name               Description
                                                                                ability
                                                                            S             N
                                                                                  B
                                                                            V             T
Field 8
1          HANDSHAKE.XON      1 = turns on X-ON/X-OFF protocol              ✓     ✓       ✓
                              0 = turns off X-ON/X-OFF protocol
2          HANDSHAKE.         1 = any characters acts as X-ON               ✓     ✓
           STARTANY           0 = only X-ON character acts as X-ON
3          HANDSHAKE.         1 = when input buffer is nearly full,         ✓     ✓       ✓
           TANDEM             X-OFF is sent
                              0 = turns off automatic X-OFF, X-ON mode
4          HANDSHAKE.DTR      1 = turns on DTR                              ✓     ✓
                              0 = turns off DTR
Field 9
1          OUTPUT.POST        Output postprocessing occurs.                 ✓     ✓
2          OUTPUT.TILDE       Special output processing for tilde.          ✓     ✓
3          OUTPUT.BG          Stop background processes at output.          ✓     ✓
4          OUTPUT.CS          Output clearscreen before reports. Emulated   ✓     ✓
                              only.
5          OUTPUT.TAB         Set output tab expansion.                     ✓     ✓
Field 10
1          PROTOCOL.LINE      Line protocol                                 ✓     ✓
2          PROTOCOL.BAUD      1 = 50     9 = 1200                           ✓     ✓       ✓
                              2 = 75    10 = 1800
                              3 = 110   11 = 2400
                              4 = 134   12 = 4800
                              5 = 150   13 = 9600
                              6 = 200   14 or EXTA = 19200
                              7 = 300   15 = EXTB
                              8 = 600
                                                                                 Avail-
Value      Name              Description
                                                                                 ability
                                                                             S             N
                                                                                   B
                                                                             V             T
3          PROTOCOL.DATA     Character size:                                 ✓     ✓       ✓
                             5 = 5 bits 7 = 7 bits
                             6 = 6 bits 8 = 8 bits
4          PROTOCOL.STOP     2 = 2 stopbits      1 = 1 stopbit               ✓     ✓       ✓
5          PROTOCOL.OUTPUT   Output parity:
                             0 = no parity                                                 ✓
                             1 = even parity                                 ✓     ✓       ✓
                             2 = odd parity                                  ✓     ✓       ✓
6          PROTOCOL.INPUT    Input parity:
                             0 = disable input parity checking               ✓     ✓       ✓
                             1 = enable input parity checking                ✓     ✓       ✓
                             2 = mark parity errors                          ✓     ✓       ✓
                             3 = mark parity errors with a null              ✓     ✓       ✓
                             4 = ignore parity errors                        ✓     ✓
7          PROTOCOL.STRIP    1 = strip to 7 bits0 = 8 bits                   ✓     ✓
Field 11
1          SIGNALS.ENABLE    Enable signal keys: Interrupt, Suspend, Quit.   ✓     ✓
2          SIGNALS.FLUSH     Flush type-ahead buffer.                        ✓     ✓
3          SIGNALS.BRKKEY    0 = break ignored                               ✓     ✓
                             1 = break as interrupt
                             2 = break as null
TTYSET
Syntax
    TTYSET dynamic.array [ON {file.variable | LPTR [n] | MTU [n] } ]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the TTYSET statement to set the characteristics of a terminal, line printer
channel, or tape unit. If only dynamic.array is specified, the terminal characteristics
for your terminal are set based on the contents of dynamic.array. dynamic.array is a
dynamic array of eleven fields, each of which has multiple values. A description of
the expected contents of each value of dynamic.array is given in the TTYGET
statement.
file.variable is a terminal opened for sequential processing with the OPENDEV or
OPENSEQ statement. If file.variable is specified, the terminal characteristics for the
specified terminal are set.
n specifies a logical print channel with LPTR or a tape unit with MTU. If n is spec-
ified, the characteristics for the print channel or tape unit are set. n is in the range
of 0 through 225 for logical print channels; the default is 0. n is in the range of 0
through 7 for tape units; the default is 0. On Windows NT you cannot specify a tape
unit.
If the terminal characteristics are set, the THEN statements are executed.
If the device does not exist or cannot be opened, or if no dynamic array is returned,
the ELSE statements are executed; any THEN statements are ignored.
If dynamic.array, file.variable, or n evaluates to the null value, the TTYSET statement
fails and the program terminates with a run-time error message.
To build dynamic.array, get the current values of the terminal line using the
TTYGET statement, manipulate the values, and reset them with the TTYSET state-
ment. The best way to access the information in the dynamic array is to include the
BASIC code UNIVERSE.INCLUDE TTY. The syntax for including this file is:
    $INCLUDE UNIVERSE.INCLUDE TTY
This file equates each value of variable from the TTYGET statement with a name, so
that each value can be easily accessed in your program. To take advantage of this
code you must call variable tty$. Once this code is included in your program, you
can use the names to access the values of the dynamic array. The TTYGET State-
ment Values table lists the names equated to the values of the dynamic array and
describes the values.
Timeout Handling
You can set the MODE.MIN and MODE.TIME values to define timeouts for read
operations over a communications line. MODE.MIN specifies the minimum
number of characters to be received. MODE.TIME specifies time in tenths of a
second. The two values interact to provide four cases that can be used as follows.
Intercharacter Timer. When you set the values of both MODE.MIN and
MODE.TIME to greater than 0, MODE.TIME specifies the maximum time interval
allowed between successive characters received by the communication line in
tenths of a second. Timing begins only after the first character is received.
Blocking Read. When you set the value of MODE.MIN to greater than 0 and
MODE.TIME to 0, no time limit is set, so the read operation waits until the speci-
fied number of characters have been received (or a newline in the case of
READSEQ).
Read Timer. When you set the value of MODE.MIN to 0 and MODE.TIME to
greater than 0, MODE.TIME specifies how long the read operation waits for a char-
acter to arrive before timing out. If no characters are received in the time specified,
the READSEQ and READBLK statements use the ELSE clause if there is one. If you
use the NOBUF statement to turn off buffering, the timer is reset after each char-
acter is received.
Nonblocking Read. When you set the values of both MODE.MIN and
MODE.TIME to 0, data is read as it becomes available. The read operation returns
immediately.
    • If any characters are received:
        – READBLK returns as many characters as specified in the blocksize argu-
          ment, or all the characters received, whichever is fewer.
        – READSEQ returns characters up to the first newline, or all the characters
          received if no newline is received.
    • If no characters are received, READSEQ and READBLK use the ELSE
      clause if there is one.
UNASSIGNED
Syntax
    UNASSIGNED (variable)
Description
Use the UNASSIGNED function to determine if variable is unassigned. UNAS-
SIGNED returns 1 (true) if variable is unassigned. It returns 0 (false) if variable is
assigned a value, including the null value.
Example
    A = "15 STATE STREET"
    C = 23
    X = UNASSIGNED(A)
    Y = UNASSIGNED(B)
    Z = UNASSIGNED(C)
    PRINT X,Y,Z
This is the program output:
    0   1    0
UNICHAR
Syntax
    UNICHAR (unicode)
Description
Use the UNICHAR function to generate a single character from a Unicode value.
unicode is a decimal number from 0 through 65535 that is the value of the character
you want to generate. If unicode is invalid, an empty string is returned. If unicode
evaluates to the null value, null is returned.
The UNICHAR function operates the same way whether NLS mode is enabled or
not.
Note: Use BASIC @Variables to generate DataStage system delimiters. Do not use
      the UNICHAR function.
UNICHARS
Syntax
    UNICHARS (dynamic.array)
Description
Use the UNICHARS function to generate a dynamic array of characters from a
dynamic array of Unicode values.
dynamic.array is an array of decimal Unicode values separated by system delim-
iters. If any element of dynamic.array is invalid, an empty string is returned for that
element. If dynamic.array evaluates to the null value, null is returned. If any
element of dynamic.array is null, null is returned for that element.
The UNICHARS function operates the same way whether NLS mode is enabled or
not.
Note: Use BASIC @Variables to generate DataStage system delimiters. Do not use
      the UNICHARS function.
UNISEQ
Syntax
    UNISEQ (expression)
Description
Use the UNISEQ function to generate a Unicode value from expression.
The first character of expression is converted to its Unicode value, that is, a hexadec-
imal value in the range 0x0000 through 0x1FFFF. If expression is invalid, for
example, an incomplete internal string, an empty string is returned. If expression
evaluates to the null value, null is returned.
The UNISEQ function operates the same way whether NLS mode is enabled or not.
For more information about Unicode values and tokens defined for system delim-
iters, see the DataStage NLS Guide.
UNISEQS
Syntax
    UNISEQS (dynamic.array)
Description
Use the UNISEQS function to generate an array of Unicode values from a dynamic
array of characters.
dynamic.array specifies an array of characters with the elements separated by
system delimiters. The first character of each element of dynamic.array is converted
to its Unicode value, a hexadecimal value in the range 0x0000 through 0x1FFFF. If
any element of dynamic.array is invalid, an empty string is returned for that
element. If dynamic.array evaluates to the null value, null is returned. If any
element of dynamic.array is the null value, null is returned for that element.
The UNISEQS function operates the same way whether NLS mode is enabled or
not.
For more information about Unicode values and tokens defined for system delim-
iters, see DataStage NLS Guide.
UNLOCK
Syntax
    UNLOCK [expression]
Description
Use the UNLOCK statement to release a process lock set by the LOCK statement.
expression specifies an integer from 0 through 63. If expression is not specified, all
locks are released (see the LOCK statement).
If expression evaluates to an integer outside the range of 0 through 63, an error
message appears and no action is taken.
If expression evaluates to the null value, the UNLOCK statement fails and the
program terminates with a run-time error message.
Examples
The following example unlocks execution lock 60:
    UNLOCK 60
The next example unlocks all locks set during the current login session:
    UNLOCK
The next example unlocks lock 50:
    X=10
    UNLOCK 60-X
UPCASE
Syntax
    UPCASE (expression)
Description
Use the UPCASE function to change all lowercase letters in expression to uppercase.
If expression evaluates to the null value, null is returned.
UPCASE is equivalent to OCONV ("MCU").
If NLS is enabled, the UPCASE function uses the conventions specified by the
Ctype category for the NLS.LC.CTYPE file to determine what constitutes upper-
case and lowercase. For more information about the NLS.LC.CTYPE file, see the
DataStage NLS Guide.
Example
    A="This is an example of the UPCASE function: "
    PRINT A
    PRINT UPCASE(A)
This is the program output:
    This is an example of the UPCASE function:
    THIS IS AN EXAMPLE OF THE UPCASE FUNCTION:
UPRINT
Syntax
    UPRINT [ ON print.channel ]     [ print.list ]
Description
In NLS mode, use the UPRINT statement to print data that was mapped to an
external format using OCONV mapname. The UPRINT statement subsequently
sends the mapped data to the screen, a line printer, or another print file with no
further mapping.
The ON clause specifies the logical print channel to use for output. print.channel is
an expression that evaluates to a number from –1 through 255. If you do not use
the ON clause, logical print channel 0 is used, which prints to the user’s terminal
if PRINTER OFF is set (see the PRINTER statement). If print.channel evaluates to
the null value, the PRINT statement fails and the program terminates with a run-
time error message. Logical print channel –1 prints the data on the screen, regard-
less of whether a PRINTER ON statement has been executed.
You can specify HEADING, FOOTING, PAGE, andPRINTER CLOSE statements
for each logical print channel. The contents of the print files are printed in order by
logical print channel number.
print.list can contain any BASIC expression. The elements of the list can be numeric
or character strings, variables, constants, or literal strings; the null value, however,
cannot be printed. The list can consist of a single expression or a series of expres-
sions separated by commas ( , ) or colons ( : ) for output formatting. If no print.list
is designated, a blank line is printed.
Expressions separated by commas are printed at preset tab positions. The default
tabstop setting is 10 characters. For information about changing the default setting,
see the TABSTOP statement. Use multiple commas together for multiple tabula-
tions between expressions.
Expressions separated by colons are concatenated. That is, the expression
following the colon is printed immediately after the expression preceding the
colon. To print a list without a LINEFEED and RETURN, end print.list with a colon
( : ).
If NLS is disabled, the UPRINT statement behaves like the PRINT statement.
For more information about maps, see the DataStage NLS Guide.
WEOF
Syntax
    WEOF [UNIT (mtu) ] {THEN statements [ELSE statements] | ELSE statements}
Description
Use the WEOF statement to write an end-of-file (EOF) mark to tape.
The UNIT clause specifies the number of the tape drive unit. Tape unit 0 is used if
no unit is specified.
mtu is an expression that evaluates to a three-digit code (decimal). Although the
mtu expression is a function of the UNIT clause, the WEOF statement uses only the
third digit (the u). Its value must be in the range of 0 through 7 (see the READT
statement for details on the mtu expression). If mtu evaluates to the null value, the
WEOF statement fails and the program terminates with a run-time error message.
Before a WEOF statement is executed, a tape drive unit must be attached
(assigned) to the user. Use the ASSIGN command to assign a tape unit to a user. If
no tape unit is attached or if the unit specification is incorrect, the ELSE statements
are executed.
The STATUS function returns 1 if WEOF takes the ELSE clause, otherwise it returns
0.
Example
    WEOF UNIT(007) ELSE PRINT "OPERATION NOT COMPLETED."
WEOFSEQ
Syntax
    WEOFSEQ file.variable [ON ERROR statements]
Description
Use the WEOFSEQ statement to write an end-of-file (EOF) mark in a file opened
for sequential access. The end-of-file mark is written at the current position and has
the effect of truncating the file at this point. Any subsequent READSEQ statement
has its ELSE statements executed.
file.variable specifies a file opened for sequential access. If file.variable evaluates to
the null value, the WEOFSEQ statement fails and the program terminates with a
run-time error message.
Note: On Windows NT systems, you cannot use the WEOFSEQ statement with a
      diskette drive that you opened with the OPENDEV statement. For 1/4-
      inch cartridge tape drives (60 MB or 150 MB) you can use WEOFSEQ to
      write an end-of-file (EOF) mark at the beginning of the data or after a write.
If the ON ERROR clause is used, the value returned by the STATUS function is the
error number.
See the OPENSEQ, READSEQ, and WRITESEQ statements for more information
about sequential file processing.
Note: Some systems do not support the truncation of disk files. WEOFSEQ is
      ignored on these systems, except that WEOFSEQ always works at the
      beginning of a file.
Example
The following example writes an end-of-file mark on the record RECORD in the
file TYPE1:
    OPENSEQ 'TYPE1','RECORD' TO FILE ELSE STOP
    WEOFSEQ FILE
WRITE
Syntax
    WRITE[U] expression {ON | TO} [file.variable,] record.ID
             [ON ERROR statements] [LOCKED statements]
             [THEN statements] [ELSE statements]
    WRITEV[U] expression {ON | TO} [file.variable,] record.ID, field#
             [ON ERROR statements] [LOCKED statements]
             [THEN statements] [ELSE statements]
Description
Use WRITE statements to write new data to a record in a DataStage file. The value
of expression replaces any data previously stored in the record.
If expression evaluates to the null value, the WRITE statement fails and the program
terminates with a run-time error message.
file.variable specifies an open file. If file.variable is not specified, the default file is
assumed (for more information on default files, see the OPEN statement. If the file
is neither accessible nor open, the program terminates with a run-time error
message, unless ELSE statements are specified.
The system searches the file for the record specified by record.ID. If the record is not
found, WRITE creates a new record.
If file.variable, record.ID, or field# evaluates to the null value, all WRITE statements
(WRITE, WRITEU, WRITEV, WRITEVU) fail and the program terminates with a
run-time error message.
The new value is written to the record, and the THEN statements are executed. If
no THEN statements are specified, execution continues with the statement
following the WRITE statement. If WRITE fails, the ELSE statements are executed;
any THEN statements are ignored.
When updating a record, the WRITE statement releases the update record lock set
with a READU statement. To maintain the update record lock set by the READU
statement, use a WRITEU statement instead of a WRITE statement.
The WRITE statement does not strip trailing field marks enclosing empty strings
from expression. Use the MATWRITE statement if that operation is required.
Tables. If the file is a table, the effective user of the program must have SQL
INSERT and UPDATE privileges to read records in the file. For information about
the effective user of a program, see the AUTHORIZATION statement.
If the OPENCHK configurable parameter is set to TRUE, or if the file is opened
with the OPENCHECK statement, all SQL integrity constraints are checked for
every write to an SQL table. If an integrity check fails, the WRITE statement uses
the ELSE clause. Use the ICHECK function to determine what specific integrity
constraint caused the failure.
NLS Mode. WRITE and other BASIC statements that perform I/O operations map
internal data to the external character set using the appropriate map for the output
file.
DataStage substitutes the file map’s unknown character for any unmappable char-
acter. The results of the WRITE statements depend on the following:
    • The inclusion of the ON ERROR clause
    • The setting of the NLSWRITEELSE parameter in the uvconfig file
    • The location of the unmappable character
The values returned by the STATUS function and the results are as follows:
For more information about unmappable characters , see the DataStage NLS Guide.
Use the STATUS function after a WRITE statement is executed, to determine the
result of the operation, as follows:
Remote Files. If in a transaction you try to write to a remote file over and the write
statement fails, the transaction is rolled back, and the program terminates with a
run-time error message.
Example
    CLEAR
    DATA "ELLEN","KRANZER","3 AMES STREET","CAMBRIDGE"
    DATA "MA","02139","SAILING"
    OPEN '','SUN.MEMBER' TO FILE ELSE
        PRINT "COULD NOT OPEN FILE"
        STOP
    END
    PRINT "ENTER YOUR FIRST NAME"
    INPUT FNAME
    PRINT "ENTER YOUR LAST NAME"
    INPUT LNAME
    PRINT "ENTER YOUR ADDRESS (PLEASE WAIT FOR PROMPTS)"
    PRINT "STREET ADDRESS"
    INPUT STREET
    PRINT "ENTER CITY"
    INPUT CITY
    PRINT "ENTER STATE"
    INPUT STATE
    PRINT "ENTER ZIP CODE"
    INPUT ZIP
    PRINT "ENTER YOUR INTERESTS"
    INPUT INTERESTS
    RECORD<1>=LNAME
    RECORD<2>=FNAME
    RECORD<3>=STREET
    RECORD<4>=CITY
    RECORD<5>=STATE
    RECORD<6>=ZIP
    RECORD<7>=1989
    RECORD<8>=INTERESTS
    WRITE RECORD TO FILE, 1111
    PRINT
    EXECUTE 'LIST SUN.MEMBER LNAME WITH FNAME EQ ELLEN'
This is the program output:
    ENTER YOUR FIRST NAME
    ?ELLEN
    ENTER YOUR LAST NAME
    ?KRANZER
    ENTER YOUR ADDRESS (PLEASE WAIT FOR PROMPTS)
    STREET ADDRESS
    ?3 AMES STREET
    ENTER CITY
    ?CAMBRIDGE
    ENTER STATE
    ?MA
    ENTER ZIP CODE
    ?02139
    ENTER YOUR INTEREST
    ?SAILING
1 records listed.
WRITEBLK
Syntax
    WRITEBLK expression ON file.variable
          {THEN statements [ELSE statements] | ELSE statements}
Description
Use the WRITEBLK statement to write a block of data to a file opened for sequen-
tial processing. Each WRITEBLK statement writes the value of expression starting
at the current position in the file. The current position is incremented to beyond the
last byte written. WRITEBLK does not add a newline at the end of the data.
file.variable specifies a file opened for sequential processing.
The value of expression is written to the file, and the THEN statements are executed.
If no THEN statements are specified, program execution continues with the next
statement. If the file cannot be accessed or does not exist, the ELSE statements are
executed; any THEN statements are ignored.
If either expression or file.variable evaluates to the null value, the WRITEBLK state-
ment fails and the program terminates with a run-time error message.
If NLS is enabled, the data written is mapped using the appropriate output file
map. For more information about maps, see the DataStage NLS Guide.
Example
    OPENSEQ 'FILE.E','RECORD4' TO FILE ELSE ABORT
    WEOFSEQ FILE
    DATA1='ONE'
    DATA2='TWO'
    *
    WRITEBLK DATA1 ON FILE ELSE ABORT
    WRITEBLK DATA2 ON FILE ELSE ABORT
    * These two lines write two items to RECORD4 in FILE.E without
    * inserting a newline between them.
    WEOFSEQ FILE
    SEEK FILE,0,0 ELSE STOP
    READSEQ A FROM FILE THEN PRINT A
    * This reads and prints the line just written to the file.
This is the program output:
    ONETWO
WRITELIST
Syntax
     WRITELIST dynamic.array ON listname
Description
Use the WRITELIST statement to save a list as a record in the &SAVEDLISTS& file.
dynamic.array is an expression that evaluates to a string made up of elements sepa-
rated by field marks. It is the list to be saved.
listname is an expression that evaluates to
     record.ID
or
     record.ID account.name
record.ID is the record ID of the select list created in the &SAVEDLISTS& file. If list-
name includes account.name, the &SAVEDLISTS& file of the specified account is
used instead of the one in the local account. If record.ID exists, WRITELIST over-
writes the contents of the record.
If either dynamic.array or listname evaluates to the null value, the WRITELIST state-
ment fails and the program terminates with a run-time error message.
WRITESEQ
Syntax
    WRITESEQ expression {ON | TO} file.variable [ON ERROR statements]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the WRITESEQ statement to write new lines to a file opened for sequential
processing. DataStage keeps a pointer to the current position in the file while it is
open for sequential processing. The OPENSEQ statement sets this pointer to the
first byte of the file, and it is advanced by theREADSEQ, READBLK, WRITESEQ,
and WRITEBLK statements.
WRITESEQ writes the value of expression followed by a newline to the file. The
data is written at the current position in the file. The pointer is set to the position
following the newline. If the pointer is not at the end of the file, WRITESEQ over-
writes any existing data byte by byte (including the newline), starting from the
current position.
file.variable specifies a file opened for sequential access.
The value of expression is written to the file as the next line, and the THEN state-
ments are executed. If THEN statements are not specified, program execution
continues with the next statement. If the specified file cannot be accessed or does
not exist, the ELSE statements are executed; any THEN statements are ignored.
If expression or file.variable evaluates to the null value, the WRITESEQ statement
fails and the program terminates with a run-time error message.
After executing a WRITESEQ statement, you can use the STATUS function to
determine the result of the operation:
File Buffering
Normally DataStage uses buffering for sequential input and output operations. If
you use the NOBUF statement after an OPENSEQ statement, buffering is turned
off and writes resulting from the WRITESEQ statement are performed right away.
You can also use the FLUSH statement after a WRITESEQ statement to cause all
buffers to be written right away.
For more information about buffering, see the FLUSH and NOBUF statements.
Example
    DATA 'NEW ITEM 1', 'NEW ITEM 2'
    OPENSEQ 'FILE.E', 'RECORD1' TO FILE ELSE ABORT
    READSEQ A FROM FILE ELSE STOP
    *
    FOR I=1 TO 2
       INPUT B
WRITESEQF
Syntax
    WRITESEQF expression {ON | TO} file.variable [ON ERROR statements]
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the WRITESEQF statement to write new lines to a file opened for sequential
processing, and to ensure that data is physically written to disk (that is, not buff-
ered) before the next statement in the program is executed. The sequential file must
be open, and the end-of-file marker must be reached before you can write to the
file. You can use the FILEINFO function to determine the number of the line about
to be written.
Normally, when you write a record using the WRITESEQ statement, the record is
moved to a buffer that is periodically written to disk. If a system failure occurs, you
could lose all the updated records in the buffer. The WRITESEQF statement forces
the buffer contents to be written to disk; the program does not execute the state-
ment following the WRITESEQF statement until the buffer is successfully written
to disk. A WRITESEQF statement following several WRITESEQ statements
ensures that all buffered records are written to disk.
WRITESEQF is intended for logging applications and should not be used for
general programming. It increases the disk I/O of your program and therefore
degrades performance.
file.variable specifies a file opened for sequential access.
The value of expression is written to the file as the next line, and the THEN state-
ments are executed. If THEN statements are not specified, program execution
continues with the next statement. If the specified file cannot be accessed or does
not exist, the ELSE statements are executed; any THEN statements are ignored.
If expression or file.variable evaluates to the null value, the WRITESEQF statement
fails and the program terminates with a run-time error message.
If NLS is enabled, WRITESEQF and other BASIC statements that perform I/O
operations always map internal data to the external character set using the appro-
priate map for the output file. For more information about maps, see the DataStage
NLS Guide.
Example
In the following example, the print statement following the WRITESEQF state-
ment is not executed until the record is physically written to disk:
    WRITESEQF ACCOUNT.REC TO ACCOUNTS.FILE
       THEN WRITTEN = TRUE
       ELSE STOP "ACCOUNTS.FILE FORCE WRITE ERROR"
    PRINT "Record written to disk."
WRITET
Syntax
    WRITET [UNIT (mtu)] variable
             {THEN statements [ELSE statements] | ELSE statements}
Description
Use the WRITET statement to write a tape record on a magnetic tape. The value of
variable becomes the next tape record. variable is an expression that evaluates to the
text to be written to tape.
The UNIT clause specifies the number of the tape drive unit. Tape unit 0 is used if
no unit is specified. If the UNIT clause is used, mtu is an expression that evaluates
to a code made up of three decimal digits, as shown in the following table:
mtu Codes
The mtu expression is read from right to left. If mtu evaluates to a one-digit code, it
represents the tape unit number. If mtu evaluates to a two-digit code, the rightmost
digit represents the unit number and the digit to its left is the track number.
If either mtu or variable evaluates to the null value, the WRITET statement fails and
the program terminates with a run-time error message.
Each tape record is written completely before the next record is written. The
program waits for the completion of data transfer to the tape before continuing.
Before a WRITET statement is executed, a tape drive unit must be attached
(assigned) to the user. Use the ASSIGN command to assign a tape unit to a user. If
no tape drive unit is attached or if the unit specification is incorrect, the ELSE state-
ments are executed.
The largest record that the WRITET statement can write is system-dependent. If
the actual record is larger, bytes beyond the system byte limit are not written.
Note: DataStage BASIC does not generate tape labels for the tape file produced
      with the WRITET statement.
The STATUS function returns 1 if READT takes the ELSE clause, otherwise it
returns 0.
If NLS is enabled, WRITET and other BASIC statements that perform I/O opera-
tions always map external data to the internal character set using the appropriate
map for the file. The map defines the external character set for the file that is used
to input data on a keyboard, display data on a screen, and so on. For more infor-
mation about maps, see the DataStage NLS Guide.
PIOPEN Flavor
If you have a program that specifies the syntax UNIT ndmtu, the nd elements are
ignored by the compiler and no errors are reported.
Examples
The following example writes a record to tape drive 0:
    RECORD=1S2S3S4
    WRITET RECORD ELSE PRINT "COULD NOT WRITE TO TAPE"
The next example writes the numeric constant 50 to tape drive 2, a 9-track tape
with no conversion:
    WRITET UNIT(002) "50" ELSE PRINT "COULD NOT WRITE"
WRITEU
Use the WRITEU statement to maintain an update record lock while performing
the WRITE statement. For details, see the WRITE statement.
Use the WRITEV statement to write on the contents of a specified field of a record
of a DataStage file. For details, see the WRITE statement.
Use the WRITEVU statement to maintain an update record lock while writing on
the contents of a specified field of a record of a DataStage file. For details, see the
WRITE statement.
Description
Use the XLATE function to return the contents of a field or a record in a DataStage
file. XLATE opens the file, reads the record, and extracts the specified data.
filename is an expression that evaluates to the name of the remote file. If XLATE
cannot open the file, a run-time error occurs, and XLATE returns an empty string.
record.ID is an expression that evaluates to the ID of the record to be accessed. If
record.ID is multivalued, the translation occurs for each record ID and the result is
multivalued (system delimiters separate data translated from each record).
field# is an expression that evaluates to the number of the field from which the data
is to be extracted. If field# is −1, the entire record is returned, except for the
record ID.
control.code is an expression that evaluates to a code specifying what action to take
if data is not found or is the null value. The possible control codes are:
X   (default) Returns an empty string if the record does not exist or data cannot
    be found.
V   Returns an empty string and produces an error message if the record does not
    exist or data cannot be found.
C   Returns the value of record.ID if the record does not exist or data cannot be
    found.
N Returns the value of record.ID if the null value is found.
The returned value is lowered. For example, value marks in the original field
become subvalue marks in the returned value. For more information, see the
LOWER function.
If filename, record.ID, or field# evaluates to the null value, the XLATE function fails
and the program terminates with a run-time error message. If control.code evaluates
to the null value, null is ignored and X is used.
The XLATE function is the same as the TRANS function.
Description
Use the XTD function to convert a string of hexadecimal characters to an integer. If
string evaluates to the null value, null is returned.
Example
      Y = "0019"
      Z = XTD (Y)
      PRINT Z
This is the program output:
      25
This appendix is a quick reference for all DataStage BASIC statements and func-
tions. The statements and functions are grouped according to their uses:
    •   Compiler directives
    •   Declarations
    •   Assignments
    •   Program flow control
    •   File I/O
    •   Sequential file I/O
    •   Printer and terminal I/O
    •   Tape I/O
    •   Select lists
    •   String handling
    •   Data conversion and formatting
    •   NLS
    •   Mathematical functions
    •   Relational functions
    •   System
    •   Remote procedure calls
    •   Miscellaneous
Compiler Directives
*                   Identifies a line as a comment line. Same as the !, $*, and REM
                    statements.
!                   Identifies a line as a comment line. Same as the *, $*, and REM
                    statements.
Declarations
COMMON            Defines a storage area in memory for variables commonly
                  used by programs and external subroutines.
DEFFUN            Defines a user-written function.
DIMENSION         Declares the name, dimensionality, and size constraints of an
                  array variable.
FUNCTION          Identifies a user-written function.
PROGRAM           Identifies a program.
SUBROUTINE        Identifies a series of statements as a subroutine.
Assignments
ASSIGNED( )       Determines if a variable is assigned a value.
CLEAR             Assigns a value of 0 to specified variables.
LET               Assigns a value to a variable.
MAT               Assigns a new value to every element of an array with one
                  statement.
UNASSIGNED( )     Determines if a variable is unassigned.
Tape I/O
READT           Assigns the contents of the next record from a magnetic tape
                unit to the named variable.
REWIND          Rewinds the magnetic tape to the beginning of the tape.
WEOF            Writes an end-of-file mark to a magnetic tape.
WRITET          Writes the contents of a record onto magnetic tape.
Select Lists
CLEARSELECT     Sets a select list to empty.
DELETELIST      Deletes a select list saved in the &SAVEDLISTS& file.
GETLIST         Activates a saved select list so it can be used by a READNEXT
                statement.
READLIST        Assigns an active select list to a variable.
READNEXT        Assigns the next record ID from an active select list to a
                variable.
SELECT          Creates a list of all record IDs in a DataStage file for use by a
                subsequent READNEXT statement.
SELECTE         Assigns the contents of select list 0 to a variable.
SELECTINDEX     Creates select lists from secondary key indexes.
SELECTINFO( )   Returns the activity status of a select list.
SSELECT         Creates a sorted list of all record IDs from a DataStage file.
String Handling
ALPHA( )          Determines whether the expression is an alphabetic or nonal-
                  phabetic string.
CATS( )           Concatenates elements of two dynamic arrays.
CHANGE( )         Substitutes an element of a string with a replacement element.
CHECKSUM( )       Returns a cyclical redundancy code (a checksum value).
COL1( )           Returns the column position immediately preceding the
                  selected substring after a BASIC FIELD function is executed.
COL2( )           Returns the column position immediately following the
                  selected substring after a BASIC FIELD function is executed.
COMPARE( )        Compares two strings for sorting.
CONVERT           Converts specified characters in a string to designated
                  replacement characters.
CONVERT( )        Replaces every occurrence of specified characters in a variable
                  with other specified characters.
COUNT( )          Evaluates the number of times a substring is repeated in a
                  string.
COUNTS( )         Evaluates the number of times a substring is repeated in each
                  element of a dynamic array.
DCOUNT( )         Evaluates the number of delimited fields contained in a string.
DEL               Deletes the specified field, value, or subvalue from a dynamic
                  array.
DELETE( )         Deletes a field, value, or subvalue from a dynamic array.
DOWNCASE( )       Converts all uppercase letters in an expression to lowercase.
DQUOTE( )         Encloses an expression in double quotation marks.
EREPLACE( )       Substitutes an element of a string with a replacement element.
EXCHANGE( )       Replaces one character with another or deletes all occurrences
                  of a specific character.
NLS
$MAP              Directs the compiler to specify the map for the source code.
AUXMAP            Assigns the map for the auxiliary printer to print unit 0 (i.e.,
                  the terminal).
BYTE( )           Generates a string made up of a single byte.
BYTELEN( )        Generates the number of bytes contained in the string value in
                  an expression.
BYTETYPE( )       Determines the function of a byte in a character.
BYTEVAL( )        Retrieves the value of a byte in a string value in an expression.
Mathematical Functions
ABS( )          Returns the absolute (positive) numeric value of an
                expression.
ABSS( )         Creates a dynamic array containing the absolute values of a
                dynamic array.
ACOS( )         Calculates the trigonometric arc-cosine of an expression.
ADDS( )         Adds elements of two dynamic arrays.
ASIN( )         Calculates the trigonometric arc-sine of an expression.
Relational Functions
ANDS( )           Performs a logical AND on elements of two dynamic arrays.
EQS( )            Compares the equality of corresponding elements of two
                  dynamic arrays.
GES( )            Indicates when elements of one dynamic array are greater
                  than or equal to corresponding elements of another dynamic
                  array.
GTS( )            Indicates when elements of one dynamic array are greater
                  than corresponding elements of another dynamic array.
IFS( )            Evaluates a dynamic array and creates another dynamic array
                  on the basis of the truth or falsity of its elements.
ISNULL( )         Indicates when a variable is the null value.
ISNULLS( )        Indicates when an element of a dynamic array is the null
                  value.
LES( )            Indicates when elements of one dynamic array are less than or
                  equal to corresponding elements of another dynamic array.
LTS( )            Indicates when elements of one dynamic array are less than
                  corresponding elements of another dynamic array.
NES( )            Indicates when elements of one dynamic array are not equal
                  to corresponding elements of another dynamic array.
NOT( )            Returns the complement of the logical value of an expression.
NOTS( )           Returns the complement of the logical value of each element
                  of a dynamic array.
System
DATE( )       Returns the internal system date.
DEBUG         Invokes RAID, the interactive DataStage BASIC debugger.
ERRMSG        Prints a formatted error message from the ERRMSG file.
INMAT( )      Used with the MATPARSE, MATREAD, and MATREADU
              statements to return the number of array elements or with the
              OPEN statement to return the modulo of a file.
ITYPE( )      Returns the value resulting from the evaluation of an
              I-descriptor.
LOCK          Sets an execution lock to protect user-defined resources or
              events from being used by more than one concurrently
              running program.
NAP           Suspends execution of a BASIC program, pausing for a speci-
              fied number of milliseconds.
SENTENCE( )   Returns the stored sentence that invoked the current process.
SLEEP         Suspends execution of a BASIC program, pausing for a speci-
              fied number of seconds.
STATUS( )     Reports the results of a function or statement previously
              executed.
SYSTEM( )     Checks the status of a system function.
TIME( )       Returns the time in internal format.
TIMEDATE( )   Returns the time and date.
UNLOCK        Releases an execution lock that was set with the LOCK
              statement.
Miscellaneous
CLEARPROMPTS Clears the value of the in-line prompt.
EOF(ARG.)         Checks whether the command line argument pointer is past
                  the last command line argument.
FILEINFO( )       Returns information about the specified file’s configuration.
ILPROMPT( )       Evaluates strings containing in-line prompts.
GET(ARG.)         Retrieves a command line argument.
SEEK(ARG.)        Moves the command line argument pointer.
Table B-1 lists binary, octal, hexadecimal, and ASCII equivalents of decimal
numbers.
conversion codes
This appendix describes the correlative and conversion codes used in dictionary
entries and with the ICONV, ICONVS, OCONV, and OCONVS functions in
BASIC. Use conversion codes with the ICONV function when converting data to
internal storage format and with the OCONV function when converting data from
its internal representation to an external output format. Read this entire appendix
and both the ICONV function and OCONV function sections before attempting to
perform internal or external data conversion.
Note: If you try to convert the null value, null is returned and the STATUS func-
      tion returns 1 (invalid data).
The NLS extended syntax is supported only for Release 9.4.1 and above.
Table C-1 lists correlative and conversion codes.
 Code                 Description
 A                    Algebraic functions
 C                    Concatenation
 D                    Date conversion
 DI                   International date conversion
 ECS                  Extended character set conversion
 F                    Mathematical functions
 G                    Group extraction
Code            Description
L               Length function
MB              Binary conversion
MCA             Masked alphabetic conversion
MC/A            Masked nonalphabetic conversion
MCD             Decimal to hexadecimal conversion
MCDX            Decimal to hexadecimal conversion
MCL             Masked lowercase conversion
MCM             Masked multibyte conversion
MC/M            Masked single-byte conversion
MCN             Masked numeric conversion
MC/N            Masked nonnumeric conversion
MCP             Masked unprintable character conversion
MCT             Masked initial capitals conversion
MCU             Masked uppercase conversion
MCW             Masked wide-character conversion
MCX             Hexadecimal to decimal conversion
MCXD            Hexadecimal to decimal conversion
MD              Masked decimal conversion
ML              Masked left conversion
MM              NLS monetary conversion
MO              Octal conversion
MP              Packed decimal conversion
MR              Masked right conversion
MT              Time conversion
MU0C            Hexadecimal Unicode character conversion
MX              Hexadecimal conversion
MY              ASCII conversion
NL              NLS Arabic numeral conversion
NLSmapname      Conversion using NLS map name
NR              Roman numeral conversion
P               Pattern matching
Code             Description
Q                Exponential conversion
R                Range function
S                Soundex
S                Substitution
T                Text extraction
T filename       File translation
TI               International time conversion
Format
      A [ ; ] expression
The A code converts A codes into F codes in order to perform mathematical oper-
ations on the field values of a record, or to manipulate strings. The A code
functions in the same way as the F code but is easier to write and to understand.
expression can be one or more of the following:
A data location or string
A function
R(exp)       Remainder after integer division of the first operand by the second.
             For example, R(2,"5") returns the remainder when field 2’s value is
             divided by 5.
S(exp)       Sum all multivalues in exp. For example, S(6) sums the multivalues of
             field 6.
IN(exp)      Test for the null value.
[]           Extract substring. Field numbers, literal numbers, or expressions can
             be used as arguments within the brackets. For example, if the value
             of field 3 is 9, then 7["2",3] returns the second through ninth charac-
             ters of field 7. The brackets are part of the syntax and must be typed.
IF(expression) | THEN(expression) | ELSE(expression)
(conv)       Conversion expression in parentheses (except A and F conversions).
An arithmetic operator
*            Multiply operands.
/            Divide operands. Division always returns an integer result: for
             example, "3" / "2" evaluates to 1, not to 1.5.
+            Add operands.
−            Subtract operands.
:            Concatenate operands.
A relational operator
=            Equal to
<            Less than
>            Greater than
# or <>      Not equal to
<=           Less than or equal to
>=           Greater than or equal to
A conditional operator
In most cases F and A codes do not act on a data string passed to them. The code
specification itself contains all the necessary data (or at least the names of fields
that contain the necessary data). So the following A codes produce identical F
codes, which in turn assign identical results to X:
      X   =    OCONV( "123", "A;'1' + '2'" )
      X   =    OCONV( "", "A;'1' + '2'" )
      X   =    OCONV( @ID, "A;'1' + '2'" )
      X   =    OCONV( "The quick brown fox jumped over a lazy dog's
              back","A;'1' + '2'" )
The data strings passed to the A code—123, the empty string, the record ID, and
“The quick brown fox…” string—simply do not come into play. The only possible
exception occurs when the user includes the LPV (load previous value) special
operand in the A or F code. The following example adds the value 5 and the
previous value 123 to return the sum 128:
      X = OCONV( "123", "A;'5' + LPV" )
It is almost never right to call an A or F code using the vector conversion functions
OCONVS and ICONVS. In the following example, Y = 123V456V789:
      X = OCONVS( Y, "A;'5' + '2' )
The statement says, “For each value of Y, call the A code to add 5 and 2.” (V repre-
sents a value mark.) The A code gets called three times, and each time it returns the
value 7. X, predictably, gets assigned 7. The scalar OCONV function returns the
same result in much less time.
What about correlatives and conversions within an A or F code? Since any string
in the A or F code can be multivalued, the F code calls the vector functions
OCONVS or ICONVS any time it encounters a secondary correlative or conver-
sion. In the following example, the F code—itself called only once—calls OCONVS
to ensure that the G code gets performed on each value of @RECORD< 1 >. X is
assigned the result cccVfff:
      @RECORD< 1 > = aaa*bbb*cccVddd*eee*fff
      X = OCONV( "", "A;1(G2*1)"
The value mark is reserved to separate individual code specifications where
multiple successive conversions must be performed.
The following dictionary entry specifies that the substring between the first and
second asterisks of the record ID should be extracted, then the first four characters
of that substring should be extracted, then the masked decimal conversion should
be applied to that substring:
    001:   D
    002:   0
    003:   G1*1VT1,4VMD2
    004:   Foo
    005:   6R
    006:   S
To attempt to define a multivalued string as part of the A or F code itself rather
than as part of the @RECORD produces invalid code. For instance, both:
    X = OCONV( "", "A;'aaa*bbb*cccVddd*eee*fff'(G2*1)" )
and the dictionary entry:
    001:   D
    002:   0
    003:   A;'aaa*bbb*cccVddd*eee*fff'(G2*1)
    004:   Bar
    005:   7L
    006:   S
are invalid. The first returns an empty string (the original value) and a status of 2.
The second returns the record ID; if the STATUS function were accessible from
dictionary entries, it would also be set to 2.
Format
      C [ ; ] expression1 c expression2 [ c expression3 ] …
The C code chains together field values or quoted strings, or both.
The semicolon is optional and is ignored.
c is the character to be inserted between the fields. Any nonnumeric character
(except system delimiters) is valid, including a blank. A semicolon ( ; ) is a reserved
character that means no separation character is to be used. Two separators cannot
follow in succession, with the exceptions of semicolons and blanks.
expression is a field number and requests the contents of that field; or any string
enclosed in single quotation marks ( ' ), double quotation marks ( " ), or back-
slashes ( \ ); or an asterisk ( * ), which specifies the data value being converted.
You can include any number of delimiters or expressions in a C code.
Note: When the C conversion is used in a field descriptor in a file dictionary, the
      field number in the LOC or A/AMC field of the descriptor should be 0. If
      it is any other number and the specified field contains an empty string, the
      concatenation is not performed.
Examples
Assume a BASIC program with @RECORD = "oneFtwoFthreeVfour":
Statement                                            Output
PRINT      OCONV("x","C;1;'xyz';2")                 onexyztwo
PRINT      ICONV("x","C;2;'xyz';3")                 twoxyzthreeVfour
PRINT      OCONV("","C;2;'xyz';3")
PRINT      ICONV(x,"C;1***2")                       one*x*two
PRINT      OCONV(0,"C;1:2+3")                       one:two+threeVfour
There is one anomaly of the C code (as implemented by ADDS Mentor, at least)
that the C code does not reproduce:
Format
    D [ n ] [ *m ] [ s ] [ fmt [ [f1, f2, f3, f4, f5] ]] [ E ] [ L ]
The D code converts input dates from conventional formats to an internal format
for storage. It also converts internal dates back to conventional formats for output.
When converting an input date to internal format, date conversion specifies the
format you use to enter the date. When converting internal dates to external
format, date conversion defines the external format for the date.
If the D code does not specify a year, the current year is assumed. If the code spec-
ifies the year in two-digit form, the years from 0 through 29 mean 2000 through
2029, and the years from 30 through 99 mean 1930 through 1999.
You can set the default date format with the DATE.FORMAT command. A system-
wide default date format can be set in the msg.text file of the UV account directory.
Date conversions specified in file dictionaries or in ICONV or OCONV functions
use the default date format except where they specifically override it. When NLS
locales are enabled, the locale overrides any value set in the msg.text file.
Each format modifier must correspond to a format option. The following table
shows which modifiers can modify which options:
Example
The following example shows how to use the format modifiers:
    D DMY[Z,A3,Z2]
Z modifies the day format option (D) by suppressing leading zeros (05 becomes 5).
A3 modifies the month format option (M) so that the month is represented by the
first three alphabetic characters (APRIL becomes APR). Z2 modifies the year
format option (Y) by suppressing leading zeros and displaying two digits. This
conversion converts April 5, 1993 to 5 APR 93.
Format
    DI
The international date conversion lets you convert dates in internal format to the
default local convention format and vice versa. If NLS locales are not enabled, the
DI conversion defaults to D. If NLS locales are enabled, DI uses the date conversion
in the DI_FMT field. The DI_FMT field can contain any valid D code.
Format
    ECS
The ECS code resolves clashes between the DataStage system delimiters and the
ASCII characters CHAR(251) through CHAR(255). It converts the system delim-
iters and ASCII characters to alternative characters using an appropriate
localization procedure. If no localization library is in use, the input string is
returned without character conversion. This code is used with an ICONV or
OCONV function.
Format
     F [ ; ] element [ ; element …]
The F code performs mathematical operations on the data values of a record, or
manipulates strings. It comprises any number of operands or operators in reverse
Polish format (Lukasiewicz) separated by semicolons.
The program parses the F code from left to right, building a stack of operands.
Whenever it encounters an operator, it performs the requested operation, puts the
result on the top of the stack, and pops the lower stack elements as necessary.
The semicolon ( ; ) is the element separator.
element can be one or more of the items from the following categories:
A data location or string
loc[R]       Numeric location specifying a data value to be pushed onto the stack,
             optionally followed by an R (repeat code).
Cn           n is a constant to be pushed onto the stack.
string       Literal string enclosed in pairs of double quotation marks ( " ), single
             quotation marks ( ' ), or backslashes ( \ ).
number       Constant number enclosed in pairs of double quotation marks ( " ),
             single quotation marks ( ' ), or backslashes ( \ ). Any integer, positive,
             negative, or zero can be specified.
D            System date (in internal format).
T            System time (in internal format).
An operator
Operators specify an operation to be performed on top stack entries. stack1 refers
to the value on the top of the stack, stack2 refers to the value just below it, stack3
refers to the value below stack2, and so on.
*[n]         Multiply stack1 by stack2. The optional n is the descaling factor (that
             is, the result is divided by 10 raised to the nth power).
/            Divide stack1 into stack2, result to stack1.
R            Same as /, but instead of the quotient, the remainder is returned to
             the top of the stack.
+            Add stack1 to stack2.
−            Subtract stack1 from stack2, result to stack1 (except for REALITY
             flavor, which subtracts stack2 from stack1).
:            Concatenate stack1 string onto the end of stack2 string.
[]           Extract substring. stack3 string is extracted, starting at the character
             specified by stack2 and continuing for the number of characters spec-
             ified in stack1. This is equivalent to the BASIC [m,n] operator, where
             m is in stack2 and n is in stack1.
S            Sum of multivalues in stack1 is placed at the top of the stack.
_            Exchange stack1 and stack2 values.
P or \       Push stack1 back onto the stack (that is, duplicate stack1).
^            Pop the stack1 value off the stack.
(conv)       Standard conversion operator converts data in stack1, putting the
             result into stack1.
A logical operator
Logical operators compare stack1 to stack2. Each returns 1 for true and 0 for false:
=         Equal to.
<         Less than.
>         Greater than.
# or <>   Not equal to.
[         Less than or equal to.
]         Greater than or equal to.
Format
    G   [ skip ] delim #fields
The G code extracts one or more values, separated by the specified delimiter, from
a field.
skip specifies the number of fields to skip; if it is not specified, 0 is assumed and no
fields are skipped.
delim is any single nonnumeric character (except IM, FM, VM, SM, and TM) used
as the field separator.
#fields is the decimal number of contiguous delimited values to extract.
Format
    L [ n [ ,m ] ]
The L code places length constraints on the data to be returned.
If Ln is specified, selection is met if the value’s length is less than or equal to n char-
acters; otherwise an empty string is returned.
If Ln,m is specified, selection is met if the value’s length is greater than or equal to
n characters, and less than or equal to m characters; otherwise an empty string is
returned.
If n is omitted or 0, the length of the value is returned.
Formats
    MCA
    MC/A
    MCD[X]
    MCL
    MCM
    MC/M
    MCN
    MC/N
    MCP
    MCT
    MCU
    MCW
    MCX[D]
The MC codes let you change a field’s data to upper- or lowercase, to extract
certain classes of characters, to capitalize words in the field, and to change unprint-
able characters to periods.
MCA          Extracts all alphabetic characters in the field, both upper- and lower-
             case. Nonalphabetic characters are not printed. In NLS mode, uses
             the ALPHABETICS field in the NLS.LC.CTYPE file.
MC/A         Extracts all nonalphabetic characters in the field. Alphabetic charac-
             ters are not printed. In NLS mode, uses the NON-ALPHABETICS
             field in the NLS.LC.CTYPE file.
MCD[X]       Converts decimal to hexadecimal equivalents.
MCL          Converts all uppercase letters to lowercase. Does not affect lowercase
             letters or nonalphabetic characters. In NLS mode, uses the UPPER-
             CASE and DOWNCASED fields in the NLS.LC.CTYPE file.
MCM          Use only if NLS is enabled. Extracts all NLS multibyte characters in
             the field. Multibyte characters are all those outside the Unicode range
             (x0000–x007F), the DataStage system delimiters, and the null value.
             As long as NLS is enabled, the conversion still works if locales are off.
             If NLS mode is disabled, the code returns a STATUS of 2, that is, an
             invalid conversion code.
MC/M     Use only if NLS is enabled. Extracts all NLS single-byte characters in
         the field. Single-byte characters are all those in the Unicode range
         x0000–x007F. As long as NLS is enabled, the conversion still works if
         locales are off. If NLS mode is disabled, the code returns a STATUS of
         2, that is, an invalid conversion code.
MCN      Extracts all numeric characters in the field. Alphabetic characters are
         not printed. In NLS mode, uses the NUMERICS field in the
         NLS.LC.CTYPE file.
MC/N     Extracts all nonnumeric characters in the field. Numeric characters
         are not printed. In NLS mode, uses the NON-NUMERICS field in the
         NLS.LC.CTYPE file.
MCP      Converts each unprintable character to a period. In NLS mode, uses
         the PRINTABLE and NON_PRINTABLE fields in the NLS.LC.CTYPE
         file.
MCT      Capitalizes the first letter of each word in the field (the remainder of
         the word is converted to lowercase). In NLS mode, uses the LOWER-
         CASE and UPCASED fields of the NLS.LC.CTYPE file.
MCU      Converts all lowercase letters to uppercase. Does not affect uppercase
         letters or nonalphabetic characters. In NLS mode, uses the LOWER-
         CASE and UPCASED fields in the NLS.LC.CTYPE file.
MCW      Use only if NLS is enabled. Converts between 7-bit standard ASCII
         (0021 007E range) and their corresponding double-byte characters,
         which are two display positions in width (FF01 FF5E fullwidth
         range). As long as NLS is enabled, the conversion still works if locales
         are off. If NLS mode is disabled, the code returns a STATUS of 2, that
         is, an invalid conversion code.
MCX[D]   Converts hexadecimal to decimal equivalents.
Format
    MD [ n [ m ]] [ , ] [ $ ] [ F ] [ I ] [ Y ] [ intl ] [ – | < | C | D ] [ P ] [ Z ] [ T ] [ fx ]
The MD code converts numeric input data to a format appropriate for internal
storage. If the code includes the $, F, I, or Y option, the conversion is monetary,
otherwise it is numeric. The MD code must appear in either an ICONV or an
OCONV expression. When converting internal representation of data to external
output format, masked decimal conversion inserts the decimal point and other
appropriate formats into the data.
Note: If NLS is enabled and either the Numeric or Monetary categories are set to
      OFF, the MD code behaves as if NLS locales were turned off.
If the value of n is 0, the decimal point does not appear in the output.
The optional m specifies the power of 10 used to scale the input or output data. On
input, the decimal point is moved m places to the right before storing. On output,
the decimal point is moved m places to the left. For example, if m is 2 in an input
conversion and the input data is 123, it would be stored as 12300. If m is 2 in an
output conversion and the stored data is 123, it would be output as 1.23. If m is not
specified, it is assumed to be the same as n. In both cases, the last required decimal
place is rounded off before excess digits are truncated. Zeros are added if not
enough decimal places exist in the original expression.
If NLS is enabled and the conversion is monetary, the thousands separator comes
from the THOU_SEP field of the Monetary category of the current locale, and the
decimal separator comes from the DEC_SEP field. If the conversion is numeric, the
thousands separator comes from the THOU_SEP field of the Numeric category,
and the decimal separator comes from the DEC_SEP field.
I      Used with the OCONV function, the international monetary symbol for
       the locale is used (INTL_CURR_SYMBOL in the Monetary category).
       Used with the ICONV function, the international monetary symbol for
       the locale is removed. If NLS is disabled or the Monetary category is
       turned off, the default symbol is USD.
Y      Used with the OCONV function: if NLS is enabled, the yen/yuan char-
       acter (Unicode 00A5) is used. If NLS is disabled or the Monetary locale
       category is turned off, the ASCII character xA5 is used.
intl   An expression that customizes numeric output according to different
       international conventions, allowing multibyte characters. The intl expres-
       sion can specify a prefix, a suffix, and the characters to use as a thousands
       delimiter and as the decimal delimiter, using the locale definition from
       the NLS.LC.NUMERIC file. The intl expression has the following syntax:
           [ prefix , thousands , decimal , suffix ]
       The bold brackets are part of the syntax and must be typed. The four
       elements are positional parameters and must be separated by commas.
       Each element is optional, but its position must be held by a comma. For
       example, to specify a suffix only, type [,,,suffix ].
       prefix       Character string to prefix to the number. If prefix contains
                    spaces, commas, or right square brackets, enclose it in quota-
                    tion marks.
       thousands    Character string that separates thousands. If thousands
                    contains spaces, commas, or right square brackets, enclose it
                    in quotation marks.
       decimal      Character string to use as a decimal delimiter. If decimal
                    contains spaces, commas, or right square brackets, enclose it
                    in quotation marks.
       suffix       Character string to append to the number. If suffix contains
                    spaces, commas, or right square brackets, enclose it in quota-
                    tion marks.
−      Specifies that negative data be suffixed with a minus sign and positive
       data be suffixed with a blank space.
<      Specifies that negative data be enclosed in angle brackets for output; posi-
       tive data is prefixed and suffixed with a blank space.
Format
    ML [ n [ m ]] [ Z ] [ , ] [ C | D | M | E | N ] [ $ ] [ F ] [ intl ] [ (fx) ]
    MR [ n [ m ]] [ Z ] [ , ] [ C | D | M | E | N ] [ $ ] [ F ] [ intl ] [ (fx) ]
The ML and MR codes allow special processing and formatting of numbers and
monetary amounts. If the code includes the F or I option, the conversion is mone-
tary, otherwise it is numeric. ML specifies left justification; MR specifies right
justification.
Note: If NLS is enabled and either the Numeric or Monetary categories are set to
      OFF, the ML and MR codes behave as if locales were turned off.
If NLS is enabled and the conversion is monetary, the thousands separator comes
from the THOU_SEP field of the Monetary category of the current locale, and the
decimal separator comes from the DEC_SEP field. If the conversion is numeric, the
thousands separator comes from the THOU_SEP field of the Numeric category,
and the decimal separator comes from the DEC_SEP field.
When NLS locales are enabled, the <, –, C, and D options define numbers intended
for monetary use. These options override any specified monetary formatting. If the
conversion is monetary and no monetary formatting is specified, it uses the
POS_FMT, NEG_FMT, POS_SIGN, and NEG_SIGN fields from the Monetary cate-
gory of the current locale.
They are unaffected by the Numeric or Monetary categories. If no options are set,
the value is returned unchanged.
Format
    MM [ n ] [ I [ L ]]
The MM code provides for local conventions for monetary formatting.
Note: If NLS is enabled and either the Numeric or Monetary categories are set to
      OFF, the MM code behaves as if locales were turned off.
If NLS is enabled and the Monetary category is turned on, the MM code uses the
local monetary conventions for decimal and thousands separators. The format
options are as follows:.
If you specify MM with no arguments, the decimal and thousands separators come
from the Monetary category of the current locale, and the currency symbol comes
from the CURR_SYMBOL field. If you specify MM with the I option, the decimal
and thousands separators are . (period) and , (comma), and the currency symbol
comes from the INTL_CURR_SYMBOL field. If you specify MM with both the I
and the L options, the decimal and thousands separators come from the Monetary
category of the current locale, and the currency symbol comes from the
INTL_CURR_SYMBOL field. The I and L options are ignored when used in the
ICONV function.
If NLS is disabled or the category is turned off, the default decimal and thousands
separators are the period and the comma.
Format
    MP
The MP code allows decimal numbers to be packed two-to-the-byte for storage.
Packed decimal numbers occupy approximately half the disk storage space
required by unpacked decimal numbers.
Leading + signs are ignored. Leading − signs cause a hexadecimal D to be stored
in the lower half of the last internal digit. If there is an odd number of packed
halves, four leading bits of 0 are added. The range of the data bytes in internal
format expressed in hexadecimal is 00 through 99 and 0D through 9D. Only valid
decimal digits (0−9) and signs ( +, − ) should be input. Other characters cause no
conversion to take place.
Packed decimal numbers should always be unpacked for output, since packed
values that are output unconverted are not displayed on terminals in a recogniz-
able format.
Format
    MT [ H ] [ P ] [ Z ] [ S ] [ c ] [ [f1, f2, f3] ]
The MT code converts times from conventional formats to an internal format for
storage. It also converts internal times back to conventional formats for output.
When converting input data to internal storage format, time conversion specifies
the format that is to be used to enter the time. When converting internal represen-
tation of data to external output format, time conversion defines the external
output format for the time.
MT is required when you specify time in either the ICONV or the OCONV func-
tion. The remaining specifiers are meaningful only in the OCONV function; they
are ignored when used in the ICONV function.
The internal representation of time is the numeric value of the number of seconds
since midnight.
If used with ICONV in an IDEAL, INFORMATION, or PIOPEN flavor account, the
value of midnight is 0. In all other account flavors, the value of midnight is 86400.
To separate hours, minutes, and seconds, you can use any nonnumeric character
that is not a system delimiter. Enclose the separator in quotation marks. If no
minutes or seconds are entered, they are assumed to be 0. You can use a suffix of
AM, A, PM, or P to specify that the time is before or after noon. If an hour larger
than 12 is entered, a 24-hour clock is assumed. 12:00 AM is midnight and 12:00 PM
is noon.
If NLS is enabled and the Time category is active, the locale specifies the AM and
PM strings, and the separator comes from the T_FMT or TI_FMT fields in the Time
category.
c         Specifies the character used to separate the hours, minutes, and seconds
          in the output. The colon ( : ) is the default. If NLS is enabled and you do
          not specify c, and if the Time category is active, c uses the
          DEFAULT_TIME_SEP field.
[f1, f2, f3] Specify format modifiers. You must include the brackets, as they are
             part of the syntax. You can specify from 1 through 3 modifiers, which
             correspond to the hours, minutes, and seconds, in that order. The
             format modifiers are positional parameters: if you want to specify f3
             only, you must include two commas as placeholders. Each format
             modifier must correspond to a format option. Use the following value
             for the format modifiers:
          ‘text’   Any text you enclose in single or double quotation marks is
                   output without the quotation marks and placed after the
                   appropriate number for the hours, minutes, or seconds.
Formats
    MX [0C ]      Hexadecimal conversion (base 16)
    MO [0C ]      Octal conversion (base 8)
    MB [0C ]      Binary conversion (base 2)
    MU0C          Hexadecimal Unicode character conversion
The MX, MO, and MB codes convert data from hexadecimal, octal, and binary
format to decimal (base 10) format and vice versa.
With ICONV. The decimal, or ASCII, format is the internal format for data repre-
sentation. When used with the ICONV function, MX, MO, and MB without the 0C
extension convert hexadecimal, octal, or binary data values (respectively) to their
equivalent decimal values. MX, MO, and MB with the 0C extension convert hexa-
decimal, octal, or binary data values to the equivalent ASCII characters rather than
to decimal values.
Use the MU0C code only if NLS is enabled. When used with ICONV, MU0C
converts data in Unicode hexadecimal format to its equivalent in the internal char-
acter set.
Characters outside of the range for each of the bases produce conversion errors.
The ranges are as follows:
With OCONV. When used with the OCONV function, MX, MO, and MB without
the 0C extension convert decimal values to their equivalent hexadecimal, octal, or
binary equivalents for output, respectively. Nonnumeric data produces a conver-
sion error if the 0C extension is not used.
MX, MO, and MB with the 0C extension convert an ASCII character or character
string to hexadecimal, octal, or binary output format. Each character in the string
is converted to the hexadecimal, octal, or binary equivalent of its ASCII character
code.
Use the MU0C code only if NLS is enabled. When used with OCONV, MU0C
converts characters from their internal representation to their Unicode hexadec-
imal equivalents for output. The data to convert must be a character or character
string in the internal character set; each character in the string is converted to its
4-digit Unicode hexadecimal equivalent. Data is converted from left to right, one
character at a time, until all data is exhausted.
Format
    MY
The MY code specifies conversion from hexadecimal to ASCII on output, and
ASCII to hexadecimal on input. When used with the OCONV function, MY
converts from hexadecimal to ASCII. When used with the ICONV function, MY
converts from ASCII to hexadecimal.
Characters outside of the range for each of the bases produce conversion errors.
The ranges are as follows:
Format
    NL
The NL code allows conversion from a locale-dependent set of alternative charac-
ters (representing digits in the local language) to Arabic numerals. The alternative
characters are the external set, the Arabic characters are the internal set.
If NLS is not enabled, characters are checked to ensure only that they are valid
ASCII digits 0 through 9, but no characters are changed.
The STATUS function returns one of the following:
Format
    NLSmapname
The NLSmapname code converts data from internal format to external format and
vice versa using the specified map. mapname is either a valid map name or one of
the following: LPTR, CRT, AUX, or OS.
The STATUS function returns one of the following:
  0 Conversion successful
  1 mapname invalid, string returned empty
  2 Conversion invalid
  3 Data converted, but result may be invalid (map could not deal with some
    characters)
Format
    NR
The NR code converts Roman numerals into Arabic numerals when used with the
ICONV function. The decimal, or ASCII, format is the internal format for
representation.
When used with the OCONV function, the NR code converts Arabic numerals into
Roman numerals.
The following is a table of Roman/Arabic numeral equivalents:
    Roman           Arabic
         i                   1
         v                   5
         x                 10
         l                 50
         c               100
         d               500
       m                1000
       V                5000
       X               10,000
         L             50,000
       C             100,000
       D             500,000
       M            1,000,000
Format
    P(pattern)   [{ ; | / } (pattern) ] …
The P code extracts data whose values match one or more patterns. If the data does
not match any of the patterns, an empty string is returned.
pattern can contain one or more of the following codes:
Format
    QR [ n { E | . } m ] [ edit ] [ mask ]
    QL [ n { E | . } m ] [ edit ] [ mask ]
    QX
The Q code converts numeric input data from exponential notation to a format
appropriate for internal storage. When converting internal representation of data
to external output format, the Q code converts the data to exponential notation by
determining how many places to the right of the decimal point are to be displayed
and by specifying the exponent.
Q alone and QR both specify right justification. QL specifies left justification. QX
specifies right justification. QX is synonymous with QR0E0 as input and MR as
output.
n specifies the number of fractional digits to the right of the decimal point. It can
be a number from 0 through 9.
m specifies the exponent. It can be a number from 0 through 9. When used with E,
m can also be a negative number from –1 through –9.
Separate n and m with either the letter E or a period ( . ). Use E if you want to
specify a negative exponent.
edit can be any of the following:
mask allows literals to be intermixed with numerics in the formatted output field.
The mask can include any combination of literals and the following three special
format mask characters:
If NLS is enabled, the Q code formats numeric and monetary values as the ML and
MR codes do, except that the intl format cannot be specified. See the ML and MR
codes for more information.
See the FMT function for more information about formatting numbers.
Format
    Rn,m [ { ; | / } n,m ] …
The R code limits returned data to that which falls within specified ranges. n is the
lower bound, m is the upper bound.
Separate multiple ranges by a semicolon ( ; ) or a slash ( / ).
If range specifications are not met, an empty string is returned.
Format
    S
The S code with no arguments specifies a soundex conversion. Soundex is a
phonetic converter that converts ordinary English words into a four-character
abbreviation comprising one alphabetic character followed by three digits.
Soundex conversions are frequently used to build indexes for name lookups.
Format
    S ; nonzero.substitute ; zero.substitute ; null.substitute
The S code substitutes one of three values depending on whether the data to
convert evaluates to zero or an empty string, to the null value, or to something else.
If the data to convert evaluates to zero or an empty string, zero.substitute is
returned. If the data is nonzero, nonempty, and nonnull, nonzero.substitute is
returned. If the data is the null value, null.substitute is returned. If null.substitute is
omitted, null values are not replaced.
All three substitute expressions can be one of the following:
    • A quoted string
    • A field number
    • An asterisk
If it is an asterisk and the data evaluates to something other than zero, the empty
string, or the null value, the data value itself is returned.
Example
Assume a BASIC program where @RECORD is:
    AFBFCVD
 Statement                                                   Output
 PRINT    OCONV("x","S;2;'zero'")                            B
 PRINT    OCONV("x","S;*;'zero'")                            x
 PRINT    OCONV(0,"S;2;'zero'")                              zero
 PRINT    OCONV('',"S;*;'zero'")                             zero
Format
    T [ start, ] length
The T code extracts a contiguous string of characters from a field.
If you specify length only, the extraction is either from the left or from the right
depending on the justification specified in line 5 of the dictionary definition item.
In a BASIC program if you specify length only, the extraction is from the right. In
this case the starting position is calculated according to the following formula:
    string.length − substring.length + 1
This lets you extract the last n characters of a string without having to calculate the
string length.
If start is specified, extraction is always from left to right.
Format
       T[DICT] filename ; c [vloc] ; [iloc] ; [oloc] [ ;bloc]
DICT            Specifies the lookup file’s dictionary. (In REALITY flavor accounts,
                you can use an asterisk ( * ) to specify the dictionary: for instance,
                T*filename … .)
filename        Name of the lookup file.
c               Translation subcode, which must be one of the following:
                V   Conversion item must exist on file, and the specified field must
                    have a value, otherwise an error message is returned.
                C   If conversion is impossible, return the original value-to-be-
                    translated.
                I   Input verify only. Functions like V for input and like C for output.
                N Returns the original value-to-be-translated if the null value is
                  found.
                O   Output verify only. Functions like C for input and like V for
                    output.
                X   If conversion is impossible, return an empty string.
vloc            Number of the value to be returned from a multivalued field. If you
                do not specify vloc and the field is multivalued, the whole field is
                returned with all system delimiters turned into blanks. If the vloc
                specification follows the oloc or bloc specification, enclose vloc in
                square brackets or separate vloc from oloc or bloc with a comma.
iloc    Field number (decimal) for input conversion. The input value is used
        as a record ID in the lookup file, and the translated value is retrieved
        from the field specified by the iloc. If the iloc is omitted, no input
        translation takes place.
oloc    Field number (decimal) for output translation. When RetrieVe creates
        a listing, data from the field specified by oloc in the lookup file are
        listed instead of the original value.
bloc    Field number (decimal) which is used instead of oloc during the
        listing of BREAK.ON and TOTAL lines.
Format
    TI
The international time conversion lets you convert times in internal format to the
default local convention format and vice versa. If NLS locales are not enabled, the
TI conversion defaults to MT. If NLS locales are enabled, TI uses the date conver-
sion in the TI_FMT field of the Time category. The TI_FMT field can contain any
valid MT code.
Table E-1 lists BASIC @variables. The @variables denoted by an asterisk ( * ) are
read-only. All others can be changed by the user.
The EXECUTE statement initializes the values of stacked @variables either to 0 or
to values reflecting the new environment. These values are not passed back to the
calling environment. The values of nonstacked @variables are shared between the
EXECUTE and calling environments. All @variables listed here are stacked unless
otherwise indicated.
                                Read-
Variable                              Value
                                Only
@ABORT.CODE                       *     A numeric value indicating the type of
                                        condition that caused the ON.ABORT
                                        paragraph to execute. The values are:
                                        1 - An ABORT statement was executed.
                                        2 - An abort was requested after pressing
                                        the Break key followed by option A.
                                        3 - An internal or fatal error occurred.
@ACCOUNT                          *     User login name. Same as @LOGNAME.
                                        Nonstacked.
@AM                               *     Field mark: CHAR(254). Same as @FM.
@ANS                                    Last I-type answer, value indeterminate.
@AUTHORIZATION                    *     Current effective user name.
@COMMAND                          *     The last command executed or entered at
                                        the command prompt.
@COMMAND.STACK                    *     Dynamic array containing the last 99
                                        commands executed.
@CONV                                   For future use.
@Variables                                                                     E-1
             Table E-1. BASIC @Variables (Continued)
                        Read-
Variable                      Value
                        Only
@CRTHIGH                  *    Number of lines on the terminal.
@CRTWIDE                  *    Number of columns on the terminal.
@DATA.PENDING             *    Dynamic array containing input gener-
                               ated by the DATA statement. Values in
                               the dynamic array are separated by field
                               marks.
@DATE                          Internal date.
@DAY                           Day of month from @DATE.
@DICT                          For future use.
@FALSE                    *    Compiler replaces the value with 0.
@FILE.NAME                     Current filename. Same as @FILENAME.
@FILENAME                      Current filename. Same as
                               @FILE.NAME.
@FM                       *    Field mark: CHAR(254). Same as @AM.
@FORMAT                        For future use.
@HDBC                     *    ODBC connection environment on the
                               local DataStage server. Nonstacked.
@HEADER                        For future use.
@HENV                     *    ODBC environment on the local
                               DataStage server. Nonstacked.
@HSTMT                    *    ODBC statement environment on the
                               local DataStage server. Nonstacked.
@ID                            Current record ID.
@IM                       *    Item mark: CHAR(255).
@ISOLATION                *    Current transaction isolation level for the
                               active transaction or the current default
                               isolation level if no transaction exists.
@LEVEL                    *    The nesting level of execution state-
                               ments. Nonstacked.
@LOGNAME                  *    User login name. Same as @ACCOUNT.
@LPTRHIGH                 *    Number of lines on the device to which
                               you are printing (that is, terminal or
                               printer).
                        Read-
Variable                      Value
                        Only
@LPTRWIDE                 *    Number of columns on the device to
                               which you are printing (that is, terminal
                               or printer).
@MONTH                         Current month.
@MV                            The current value counter for columnar
                               listing only. Used only in I-descriptors.
                               Same as @NV.
@NB                            The current BREAK level number. 1 is
                               the lowest-level break. @NB has a value
                               of 255 on the grand total line. Used only
                               in I-descriptors.
@ND                            The number of detail lines since the last
                               BREAK on a break line. Used only in
                               I-descriptors.
@NEW                      *    The new contents of the current record.
                               Use in trigger programs. Nonstacked.
@NI                            The current item counter (the number of
                               items listed or selected). Used only in
                               I-descriptors. Same as @RECCOUNT.
@NS                            The current subvalue counter for
                               columnar listing only. Used only in
                               I-descriptors.
@NULL                     *    The null value. Nonstacked.
@NULL.STR                 *    The internal representation of the null
                               value, which is CHAR(128). Nonstacked.
@NV                            The current value counter for columnar
                               listing only. Used only in I-descriptors.
                               Same as @MV.
@OLD                      *    The original contents of the current
                               record. Use in trigger programs.
                               Nonstacked.
@OPTION                        The value of field 5 in the VOC for the
                               calling verb.
@PARASENTENCE             *    The last sentence or paragraph that
                               invoked the current process.
@PATH                     *    Pathname of the current account.
@Variables                                                           E-3
             Table E-1. BASIC @Variables (Continued)
                        Read-
Variable                      Value
                        Only
@RECCOUNT                      The current item counter (the number of
                               items listed or selected). Used only in
                               I-descriptors. Same as @NI.
@RECORD                        Entire current record.
@RECUR0                        Reserved.
@RECUR1                        Reserved.
@RECUR2                        Reserved.
@RECUR3                        Reserved.
@RECUR4                        Reserved.
@SCHEMA                   *    Schema name of the current DataStage
                               account. Nonstacked. When users create
                               a new schema, @SCHEMA is not set until
                               the next time they log in to DataStage.
@SELECTED                      Number of elements selected from the
                               last select list. Nonstacked.
@SENTENCE                 *    The sentence that invoked the current
                               BASIC program. Any EXECUTE updates
                               @SENTENCE.
@SM                       *    Subvalue mark: CHAR(252). Same as
                               @SVM.
@SQL.CODE                 *    For future use.
@SQL.DATE                 *    The current system date. Use in trigger
                               programs. Nonstacked.
@SQL.ERROR                *    For future use.
@SQL.STATE                *    For future use.
@SQL.TIME                 *    The current system time. Use in trigger
                               programs. Nonstacked.
@SQL.WARNING              *    For future use.
@SQLPROC.NAME             *    The name of the current SQL procedure.
@SQLPROC.TX.LEVEL         *    The transaction level at which the
                               current SQL procedure began.
@STDFIL                        Default file variable.
@SVM                      *    Subvalue mark: CHAR(252). Same as
                               @SM.
@SYS.BELL                 *    Bell character. Nonstacked.
                         Read-
Variable                       Value
                         Only
@SYSTEM.RETURN.CODE             Status codes returned by system
                                processes. Same as @SYSTEM.SET.
@SYSTEM.SET                     Status codes returned by system
                                processes. Same as
                                @SYSTEM.RETURN.CODE.
@TERM.TYPE                 *    The terminal type. Nonstacked.
@TIME                           Internal time.
@TM                        *    Text mark: CHAR(251).
@TRANSACTION               *    A numeric value. Any nonzero value
                                indicates that a transaction is active; the
                                value 0 indicates that no transaction
                                exists.
@TRANSACTION.ID            *    Transaction number of the active transac-
                                tion. An empty string indicates that no
                                transaction exists.
@TRANSACTION.LEVEL         *    Transaction nesting level of the active
                                transaction. A 0 indicates that no transac-
                                tion exists.
@TRUE                           Compiler replaces the value with 1.
@TTY                            Terminal device name. If the process is a
                                phantom, @TTY returns the value
                                ‘phantom’. If the process is a DataStage
                                API, it returns ‘uvcs’.
@USER0                          User-defined.
@USER1                          User-defined.
@USER2                          User-defined.
@USER3                          User-defined.
@USER4                          User-defined.
@USERNO                    *    User number. Nonstacked. Same as
                                @USER.NO.
@USER.NO                   *    User number. Nonstacked. Same as
                                @USERNO.
@USER.RETURN.CODE               Status codes created by the user.
@VM                        *    Value mark CHAR(253).
@WHO                       *    The name of the current DataStage
                                account directory. Nonstacked.
@Variables                                                              E-5
           Table E-1. BASIC @Variables (Continued)
                      Read-
Variable                    Value
                      Only
@YEAR                        Current year.
This appendix describes the following subroutines you can call from a DataStage
BASIC program:
!ASYNC (!AMLC)
!EDIT.INPUT
!ERRNO
!FCMP
!GET.KEY
!GET.PARTNUM
!GET.PATHNAME
!GET.USER.COUNTS
!GETPU
!INLINE.PROMPTS
!INTS
!MAKE.PATHNAME
!MATCHES
!MESSAGE
!PACK.FNKEYS
!REPORT.ERROR
!SET.PTR
!SETPU
!TIMDAT
!USER.TYPE
!VOC.PATHNAME
Syntax
      CALL !ASYNC (key, line, data, count, carrier)
Description
Use the !ASYNC subroutine (or its synonym !AMLC) to send data to, and receive
data from an asynchronous device.
key defines the action to be taken (1 through 5). The values for key are defined in the
following list:
line is the number portion from the &DEVICE& entry TTY##, where ## represents
a decimal number.
data is the data being sent to or received from the line.
count is an output variable containing the character count.
carrier is an output variable that returns a value dependent on the value of key. If
key is 1, 2, or 3, carrier returns the variable specified by the user. If key has a value
of 4 or 5, carrier returns 1.
You must first assign an asynchronous device using the ASSIGN command. A
entry must be in the &DEVICE& file for the device to be assigned with the
record ID format of TTY##, where ## represents a decimal number. The actions
associated with each key value are as follows:
 key       Action
 1         Inputs the number of characters indicated by the value of count.
 2         Inputs the number of characters indicated by the value of count or until
           a linefeed character is encountered.
 3         Outputs the number of characters indicated by the value of count.
 4         Returns the number of characters in the input buffer to count. On oper-
           ating systems where the FIONREAD key is not supported, 0 is returned
           in count. When the value of key is 4, 1 is always returned to carrier.
 5         Returns 0 in count if there is insufficient space in the output buffer. On
           operating systems where the TIOCOUTQ key is not supported, 0 is
           returned in count. When the value of key is 5, 1 is always returned to
           carrier.
Example
The !ASYNC subroutine returns the first 80 characters from the device defined by
ASYNC10 in the &DEVICE& file to the variable data.
    data=
    count= 80
    carrier= 0
    call !ASYNC(1,10,data,count,carrier)
Syntax
       CALL !EDIT.INPUT (keys, wcol, wrow, wwidth, buffer, startpos, bwidth, ftable,
              code)
Qualifiers
keys         Controls certain operational characteristics. keys can take the additive
             values (the token names can be found in the GTI.FNKEYS.IH include
             file) shown here:
             Value      Token        Description
             0          IK$NON       None of the keys below are required.
             1          IK$OCR       Output a carriage return.
             2          IK$ATM       Terminate editing as soon as the user has entered
                                     bwidth characters.
             4          IK$TCR       Toggle cursor-visible state.
             8          IK$DIS       Display contents of buffer string on entry.
             16         IK$HDX       Set terminal to half-duplex mode (restored on
                                     exit).
             32         IK$INS       Start editing in insert mode. Default is overlay
                                     mode.
             64         IK$BEG       Separate Begin Line/End Line functionality
                                     required.
wcol         The screen column of the start of the window (x coordinate).
wrow         The screen row for the window (y coordinate).
wwidth       The number of screen columns the window occupies.
buffer       Contains the following:
             on entry   The text to display (if key IK$DIS is set).
             on exit    The final edited value of the text.
startpos     Indicates the cursor position as follows:
             on entry   The initial position of the cursor (from start of buffer).
             on exit    The position of the cursor upon exit.
bwidth    The maximum number of positions allowed in buffer. bwidth can be more
          than wwidth, in which case the contents of buffer scroll horizontally as
          required.
ftable    A packed function key trap table, defining which keys cause exit from
          the !EDIT.INPUT function. The !PACK.FNKEYS function creates the
          packed function key trap table.
code      The reply code:
          =0          User pressed Return or entered bwidth characters and
                      IK$ATM was set.
          >0          The function key number that terminated !EDIT.INPUT.
Description
Use the !EDIT.INPUT subroutine to request editable terminal input within a
single-line window on the terminal. Editing keys are defined in the terminfo files
and can be set up using the KEYEDIT, KEYTRAP and KEYEXIT statements. To ease
the implementation, the UNIVERSE.INCLUDE file GTI.FNKEYS.IH can be
included to automatically define the editing keys from the current terminfo defini-
tion. We recommend that you use the INCLUDE file.
All input occurs within a single-line window of the terminal screen, defined by the
parameters wrow, wcol, and wwidth. If the underlying buffer length bwidth is greater
than wwidth and the user performs a function that moves the cursor out of the
window horizontally, the contents of buffer are scrolled so as to keep the cursor
always in the window.
If the specified starting cursor position would take the cursor out of the window,
the buffer’s contents are scrolled immediately so as to keep the cursor visible.
!EDIT.INPUT does not let the user enter more than bwidth characters into the
buffer, regardless of the value of wwidth.
!EDIT.INPUT Functions
!EDIT.INPUT performs up to eight editing functions, as follows:
Unsupported Functions
This implementation does not support a number of functions originally available
in the Prime INFORMATION version. Because of this, sequences can be generated
that inadvertently cause the !EDIT.INPUT function to terminate. For this reason,
you can create a user-defined terminal keystroke definition file so that
!EDIT.INPUT recognizes the unsupported sequences. Unsupported sequences
cause the !EDIT.INPUT subroutine to ring the terminal bell, indicating the recog-
nition of an invalid sequence.
The file CUSTOM.GTI.DEFS defines a series of keystroke sequences for this
purpose. You can create the file in each account or in a central location, with VOC
entries in satellite accounts referencing the remote file. There is no restriction on
how the file can be created. For instance, you can use the command:
      >CREATE.FILE CUSTOM.GTI.DEFS 2 17 1 /* Information style */
or:
      >CREATE-FILE CUSTOM.GTI.DEFS (1,1,3 17,1,2) /* Pick style */
to create the definition file. A terminal keystroke definition record assumes the
name of the terminal which the definitions are associated with, i.e., for vt100 termi-
nals the CUSTOM.GTI.DEFS file record ID would be vt100 (case-sensitive). Each
terminal keystroke definition record contains a maximum of 82 fields (attributes)
Example
The following BASIC program sets up three trap keys (using the !PACK.FNKEYS
subroutine), waits for the user to enter input, then reports how the input was
terminated:
$INCLUDE UNIVERSE.INCLUDE GTI.FNKEYS.IH
* Set up trap keys of FINISH, UPCURSOR and DOWNCURSOR
TRAP.LIST = FK$FIN:@FM:FK$UP:@FM:FK$DOWN
CALL !PACK.FNKEYS(TRAP.LIST, Ftable)
* Start editing in INPUT mode, displaying contents in window
KEYS = IK$INS + IK$DIS
* Window edit is at x=20, y=2, of length 10 characters;
* the user can enter up to 30 characters of input into TextBuffer,
* and the cursor is initially placed on the first character of the
* window.
TextBuffer=""
CursorPos = 1
CALL !EDIT.INPUT(KEYS, 20, 2, 10, TextBuffer, CursorPos, 30, Ftable,
    ReturnCode)
* On exit, the user's input is within TextBuffer,
* CursorPos indicates the location of the cursor upon exiting,
* and ReturnCode contains the reason for exiting.
BEGIN CASE
   CASE CODE = 0           * User pressed RETURN key
   CASE CODE = FK$FIN      * User pressed the defined FINISH key
   CASE CODE = FK$UP       * User pressed the defined UPCURSOR key
   CASE CODE = FK$DOWN     * User pressed the defined DOWNCURSOR key
   CASE 1                  * Should never happen
END CASE
Syntax
    CALL !ERRNO (variable)
Description
Use the !ERRNO subroutine to return the current value of the operating system
errno variable.
variable is the name of a BASIC variable.
The !ERRNO subroutine returns the value of the system errno variable after the last
call to a GCI subroutine in variable. If you call a system routine with the GCI, and
the system call fails, you can use !ERRNO to determine what caused the failure. If
no GCI routine was called prior to its execution, !ERRNO returns 0. The values of
errno that apply to your system are listed in the system include file errno.h.
Syntax
    CALL !FCMP (result, number1, number2)
Description
Use the !FCMP subroutine to compare the equality of two floating-point
numeric values as follows:
If number1 is less than number2, result is –1.
If number1 is equal to number2, result is 0.
If number1 is greater than number2, result is 1.
Syntax
    CALL !GET.KEY (string, code)
Qualifiers
string Returns the character sequence of the next key pressed at the keyboard.
code Returns the string interpretation value:
      Code    String Value
      0       A single character that is not part of any function key sequence. For
              example, if A is pressed, code = 0 and string = CHAR(65).
      >0      The character sequence associated with the function key defined by
              that number in the GTI.FNKEYS.IH include file. For example, on a
              VT100 terminal, pressing the key labelled --> (right cursor move)
              returns code = 5 and string = CHAR(27):CHAR(79):CHAR(67).
      <0      A character sequence starting with an escape or control character
              that does not match any sequence in either the terminfo entry or the
              CUSTOM.GCI.DEFS file.
Description
Use the !GET.KEY subroutine to return the next key pressed at the keyboard. This
can be either a printing character, the Return key, a function key as defined by the
current terminal type, or a character sequence that begins with an escape or control
character not defined as a function key.
Function keys can be automatically initialized by including the $INCLUDE
UNIVERSE.INCLUDES GTI.FNKEYS.IH statement in the application program
that uses the !GET.KEY subroutine.
Example
The following BASIC program waits for the user to enter input, then reports the
type of input entered:
      $INCLUDE GTI.FNKEYS.IH
       STRING = ' ' ; * initial states of call variables
       CODE = -999
       * Now ask for input until user hits a "Q"
       LOOP
Syntax
    CALL !GET.PARTNUM (file, record.ID, partnum, status)
Description
Use the !GET.PARTNUM subroutine with distributed files to determine the
number of the part file to which a given record ID belongs.
file (input) is the file variable of the open distributed file.
record.ID (input) is the record ID.
partnum (output) is the part number of the part file of the distributed file to which
the given record ID maps.
status (output) is 0 for a valid part number or an error number for an invalid part
number. An insert file of equate tokens for the error numbers is available.
An insert file of equate names is provided to allow you to use mnemonics for the
error numbers. The insert file is called INFO_ERRORS.INS.IBAS, and is located in
the INCLUDE subdirectory. To use the insert file, specify $INCLUDE SYSCOM
INFO_ERRORS.INS.IBAS when you compile the program.
Note: !GET.PARTNUM does not check that the returned part number corresponds
      to one of the available part files of the currently opened file.
Example
In the following example, a distributed file SYS has been defined with parts and
part numbers S1, 5, S2, 7, and S3, 3, respectively. The file uses the default SYSTEM
partitioning algorithm.
PROMPT ''
GET.PARTNUM = '!GET.PARTNUM'
STATUS = 0
PART.NUM = 0
OPEN '', 'SYS' TO FVAR ELSE STOP 'NO OPEN SYS'
PATHNAME.LIST = FILEINFO(FVAR, FINFO$PATHNAME)
PARTNUM.LIST = FILEINFO(FVAR, FINFO$PARTNUM)
LOOP
     PRINT 'ENTER Record ID : ':
     INPUT RECORD.ID
 WHILE RECORD.ID
     CALL @GET.PARTNUM(FVAR, RECORD.ID, PART.NUM, STATUS)
     LOCATE PART.NUM IN PARTNUM.LIST<1> SETTING PART.INDEX THEN
PATHNAME = PATHNAME.LIST <PART.INDEX>
     END ELSE
           PATHNAME = ''
   END
     PRINT 'PART.NUM = ':PART.NUM:' STATUS = ':STATUS :'
         PATHNAME = ': PATHNAME
 REPEAT
 END
!GET.PARTNUM returns part number 5 for input record ID 5-1, with status code 0,
and part number 7 for input record ID 7-1, with status code 0, and part number 3
for input record ID 3-1, with status code 0. These part numbers are valid and corre-
spond to available part files of file SYS.
!GET.PARTNUM returns part number 1200 for input record ID 1200-1, with status
code 0. This part number is valid but does not correspond to an available part file
of file SYS.
!GET.PARTNUM returns part number 0 for input record ID 5-1, with status code
IE$NO.MAP.TO.PARTNUM, and part number 0 for input record ID A-1, with
status code IE$NO.MAP.TO.PARTNUM, and part number 0 for input record ID 12-
4, with status code IE$NO.MAP.TO.PARTNUM. These part numbers are not valid
and do not correspond to available part files of the file SYS.
Syntax
    CALL !GET.PATHNAME (pathname, directoryname, filename, status)
Description
Use the !GET.PATHNAME subroutine to return the directory name and filename
parts of a pathname.
pathname (input) is the pathname from which the details are required.
directoryname (output) is the directory name portion of the pathname, that is, the
pathname with the last entry name stripped off.
filename (output) is the filename portion of the pathname.
status (output) is the returned status of the operation. A 0 indicates success, another
number is an error code indicating that the supplied pathname was not valid.
Example
If pathname is input as /usr/accounts/ledger, directoryname is returned as /usr/accounts,
and filename is returned as ledger.
    PATHNAME = "/usr/accounts/ledger "
    CALL !GET.PATHNAME(PATHNAME,DIR,FNAME,STATUS)
    IF STATUS = 0
    THEN
       PRINT "Directory portion = ":DIR
       PRINT "Entryname portion = ":FNAME
    END
Syntax
    CALL !GETPU (key, print.channel, set.value, return.code)
Description
Use the !GETPU subroutine to read individual parameters of any logical print
channel.
key is a number indicating the parameter to be read.
print.channel is the logical print channel, designated by –1 through 255.
set.value is the value to which the parameter is currently set.
return.code is the code returned.
The !GETPU subroutine allows you to read individual parameters of logical print
channels as designated by print.channel. Print channel 0 is the terminal unless a
PRINTER ON statement has been executed to send output to the default printer. If
you specify print channel –1, the output is directed to the terminal, regardless of
the status of PRINTER ON or OFF. See the description of the !SETPU subroutine
later in this appendix for a means of setting individual print.channel parameters.
Bit        Description
1          Uses FORTRAN-format mode. This allows the attaching of
           vertical format information to each line of the data file. The first
           character position of each line from the file does not appear in the
           printed output, and is interpreted as follows:
           Character       Meaning
           0               Advances two lines.
           1               Ejects to the top of the next page.
           +               Overprints the last line.
           Space           Advances one line.
           –               Advances three lines (skip two lines). Any other
                           character is interpreted as advance one line.
3          Generates line numbers at the left margin.
4          Suppresses header page.
5          Suppresses final page eject after printing.
12         Spools the number of copies specified in an earlier !SETPU call.
21         Places the job in the spool queue in the hold state.
22         Retains jobs in the spool queue in the hold state after they have
           been printed.
other      All the remaining bits are reserved.
$INCLUDE statement to insert this file if you want to use equate names. The
following list shows the codes returned in the argument return.code:
Code         Meaning
0            No error
E$BKEY       Bad key (key is out of range)
E$BPAR       Bad parameter (value of new.value is out of range)
E$BUNT       Bad unit number (value of print.channel is out of range)
E$NRIT       No write (attempt to set a read-only parameter)
Examples
In this example, the file containing the parameter key equate names is inserted
with the $INCLUDE compiler directive. Later the top margin parameter for logical
print channel 0 is interrogated. Print channel 0 is the terminal unless a prior
PRINTER ON statement has been executed to direct output to the default printer.
The top margin setting is returned in the argument TM.SETTING. Return codes are
returned in the argument RETURN.CODE.
    $INCLUDE SYSCOM GETPU.INS.IBAS
    CALL !GETPU(PU$TOPMARGIN,0,TM.SETTING,RETURN.CODE)
The next example does the same as the previous example but uses the key 4 instead
of the equate name PU$TOPMARGIN. Because the key number is used, it is not
necessary for the insert file GETPU.INS.IBAS to be included.
    CALL !GETPU(4,0,TM.SETTING,RETURN.CODE)
The next example returns the current deferred time on print channel 0 in the vari-
able TIME.RET:
    CALL !GETPU(PU$DEFERTIME,0,TIME.RET,RETURN.CODE)
Syntax
    CALL !GET.USER.COUNTS (uv.users, max.uv.users, os.users)
Description
Use the !GET.USER.COUNTS subroutine to return a count of DataStage and
system users. If any value cannot be retrieved, a value of –1 is returned.
uv.users (output) is the current number of DataStage users.
max.uv.users (output) is the maximum number of licensed DataStage users allowed
on your system.
os.users (output) is the current number of operating system users.
Syntax
     CALL !INLINE.PROMPTS (result, string)
Description
Use the !INLINE.PROMPTS subroutine to evaluate a string that contains in-line
prompts. In-line prompts have the following syntax:
P                 Saves the input from an in-line prompt. The input is then used for
                  all in-line prompts with the same prompt text. This is done until
                  the saved input is overwritten by a prompt with the same prompt
                  text and with a control option of A, C, I, or S, or until control
                  returns to the command prompt. The P option saves the input from
                  an in-line prompt in the current paragraph, or in other paragraphs.
Sn                Takes the nth word from the command (as in the In control option),
                  but uses the most recent command entered at the command level
                  to execute the paragraph, rather than an argument in the para-
                  graph. This is useful where paragraphs are nested.
text              The prompt to be displayed.
option            A valid conversion code or pattern match. A valid conversion code
                  is one that can be used with the ICONV function. Conversion
                  codes must be enclosed in parentheses. A valid pattern match is
                  one that can be used with the MATCHING keyword.
If the in-line prompt has a value, that value is substituted for the prompt. If the in-
line prompt does not have a value, the prompt is displayed to request an input
value when the function is executed. The value entered at the prompt is then
substituted for the in-line prompt.
Note: Once a value has been entered for a particular prompt, the prompt
      continues to have that value until a !CLEAR.PROMPTS subroutine is
      called, or control option A is specified. A !CLEAR.PROMPTS subroutine
      clears all the values that have been entered for in-line prompts.
          You can enclose prompts within prompts.
Example
       A = ""
       CALL !INLINE.PROMPTS(A,"You have requested the <<Filename>> file")
       PRINT "A"
Syntax
    CALL !INTS (result, dynamic.array)
Description
Use the !INTS subroutine to retrieve the integer portion of elements in a dynamic
array.
result (output) contains a dynamic array that comprises the integer portions of the
elements of dynamic.array.
dynamic.array (input) is the dynamic array to process.
The !INTS subroutine returns a dynamic array, each element of which contains the
integer portion of the numeric value in the corresponding element of the input
dynamic.array.
Example
    A=33.0009:@VM:999.999:@FM:-4.66:@FM:88.3874
    CALL !INTS(RESULT,A)
The following output is displayed:
    33VM999FM–4FM88
Syntax
    CALL !MAKE.PATHNAME (path1, path2, result, status)
Description
Use the !MAKE.PATHNAME subroutine to construct the full pathname of a file.
The !MAKE.PATHNAME subroutine can be used to:
    • Concatenate two strings to form a pathname. The second string must be a
      relative path.
    • Obtain the fully qualified pathname of a file. Where only one of path1 or
      path2 is given, !MAKE.PATHNAME returns the pathname in its fully quali-
      fied state. In this case, any filename you specify does not have to be an
      existing filename.
    • Return the current working directory. To do this, specify both path1 and
      path2 as empty strings.
path1 (input) is a filename or partial pathname. If path1 is an empty string, the
current working directory is used.
path2 (input) is a relative pathname. If path2 is an empty string, the current working
directory is used.
result (output) is the resulting pathname.
status (output) is the returned status of the operation. 0 indicates success. Any other
number indicates either of the following errors:
Example
In this example, the user’s working directory is /usr/accounts:
    ENT = "ledger"
    CALL !MAKE.PATHNAME(ENT,"",RESULT,STATUS)
    IF STATUS = 0
    THEN PRINT "Full name = ":RESULT
Syntax
    CALL !MATCHES (result, dynamic.array, match.pattern)
Description
Use the !MATCHES subroutine to test whether each element of one dynamic array
matches the patterns specified in the elements of the second dynamic array. Each
element of dynamic.array is compared with the corresponding element of
match.pattern. If the element in dynamic.array matches the pattern specified in
match.pattern, 1 is returned in the corresponding element of result. If the element
from dynamic.array is not matched by the specified pattern, 0 is returned.
result (output) is a dynamic array containing the result of the comparison on each
element in dynamic array1.
dynamic.array (input) is the dynamic array to be tested.
match.pattern (input) is a dynamic array containing the match patterns.
When dynamic.array and match.pattern do not contain the same number of elements,
the behavior of !MATCHES is as follows:
    • result always contains the same number of elements as the longer of
      dynamic.array or match.pattern.
    • If there are more elements in dynamic.array than in match.pattern, the
      missing elements are treated as though they contained a pattern that
      matched an empty string.
    • If there are more elements in match.pattern than in dynamic.array, the
      missing elements are treated as though they contained an empty string.
Examples
The following example returns the value of the dynamic array as 1VM1VM1:
    A='AAA4A4':@VM:2398:@VM:'TRAIN'
    B='6X':@VM:'4N':@VM:'5A'
    CALL !MATCHES(RESULT,A,B)
In the next example, there are missing elements in match.pattern that are treated as
though they contain a pattern that matches an empty string. The result is
0VM0SM0FM1FM1.
    R='AAA':@VM:222:@SM:'CCCC':@FM:33:@FM:'DDDDDD'
    S='4A':@FM:'2N':@FM:'6X'
    CALL !MATCHES(RESULT,R,S)
In the next example, the missing element in match.pattern is used as a test for an
empty string in dynamic.array, and the result is 1VM1FM1:
    X='AAA':@VM:@FM:''
    Y='3A':@FM:'3A'
    CALL !MATCHES(RESULT,X,Y)
Syntax
    CALL !MESSAGE (key, username, usernum, message, status)
Description
Use the !MESSAGE subroutine to send a message to another user on the system.
!MESSAGE lets you change and report on the current user’s message status.
key (input) specifies the operation to be performed. You specify the option you
require with the key argument, as follows:
username (input) is the name of the user, or the TTY name, for send or status
operations.
usernum (input) is the number of the user for send/status operations.
message (input) is the message to be sent.
status (output) is the returned status of the operation as follows:
Example
   CALL !MESSAGE (KEY,USERNAME,USERNUMBER,MESSAGE,CODE)
   IF CODE # 0
   THEN CALL !REPORT.ERROR ('MY.COMMAND','!MESSAGE',CODE)
Syntax
     CALL !PACK.FNKEYS (trap.list, ftable)
Qualifiers
trap.list   A list of function numbers delimited by field marks (CHAR(254)),
            defining the specific keys that are to be used as trap keys by the
            !EDIT.INPUT subroutine.
ftable      A bit-significant string of trap keys used in the ftable parameter of the
            !EDIT.INPUT subroutine. This string should not be changed in any
            way before calling the !EDIT.INPUT subroutine.
Description
The !PACK.FNKEYS subroutine converts a list of function key numbers into a bit
string suitable for use with the !EDIT.INPUT subroutine. This bit string defines the
keys which cause !EDIT.INPUT to exit, enabling the program to handle the specific
keys itself.
trap.list can be a list of function key numbers delimited by field marks
(CHAR(254)). Alternatively, the mnemonic key name, listed below and in the
UNIVERSE.INCLUDE file GTI.FNKEYS.IH, can be used:
If ftable is returned as an empty string, an error in the trap.list array is detected, such
as an invalid function number. Otherwise ftable is a bit-significant string which
should not be changed in any way before its use with the !EDIT.INPUT subroutine.
Example
The following program sets up three trap keys using the !PACK.FNKEYS function,
then uses the bit string within the !EDIT.INPUT subroutine:
$INCLUDE UNIVERSE.INCLUDE GTI.FNKEYS.IH
* Set up trap keys of FINISH, UPCURSOR and DOWNCURSOR
TRAP.LIST = FK$FIN:@FM:FK$UP:@FM:FK$DOWN
CALL !PACK.FNKEYS(TRAP.LIST, Ftable)
* Start editing in INPUT mode, displaying contents in window
KEYS = IK$INS + IK$DIS
* Window edit is at x=20, y=2, of length 10 characters;
* the user can enter up to 30 characters of input into TextBuffer,
* and the cursor is initially placed on the first character of the
* window.
TextBuffer=""
CursorPos = 1
CALL !EDIT.INPUT(KEYS,20,2,10,TextBuffer,CursorPos,30,Ftable,ReturnCode)
* On exit, the user's input is within TextBuffer,
* CursorPos indicates the location of the cursor upon exiting,
* and ReturnCode contains the reason for exiting.
BEGIN CASE
   CASE CODE = 0
                       * User pressed RETURN key
   CASE CODE = FK$FIN
                       * User pressed the defined FINISH key
   CASE CODE = FK$UP
                       * User pressed the defined UPCURSOR key
   CASE CODE = FK$DOWN
                       * User pressed the defined DOWNCURSOR key
   CASE 1              * Should never happen
 END CASE
Syntax
    CALL !REPORT.ERROR (command, subroutine, code)
Description
Use the !REPORT.ERROR subroutine to print explanatory text for a DataStage or
operating system error code.
command is the name of the command that used the subroutine in which an error
was reported.
subroutine is the name of the subroutine that returned the error code.
code is the error code.
The general format of the message printed by !REPORT.ERROR is as follows:
    Error: Calling subroutine from command. system error code:
    message.text.
system is the operating system, or DataStage.
Text for values of code in the range 0 through 9999 is retrieved from the operating
system. Text for values of code over 10,000 is retrieved from the SYS.MESSAGES file.
If the code has no associated text, a message to that effect is displayed. Some
DataStage error messages allow text to be inserted in them. In this case, code can be
a dynamic array of the error number, followed by one or more parameters to be
inserted into the message text.
Examples
    CALL !MESSAGE (KEY,USERNAME,USERNUMBER,MESSAGE,CODE)
    IF CODE # 0
    THEN CALL !REPORT.ERROR ('MY.COMMAND','!MESSAGE',CODE)
If code was IE$SEND.REQ.REC, !REPORT.ERROR would display the following:
    Error calling "!MESSAGE" from "MY.COMMAND" UniVerse error 1914:
    Warning: Sender requires "receive" enabled!
The next example shows an error message with additional text:
    CALL !MESSAGE (KEY,USERNAME,USERNUMBER,MESSAGE,CODE)
    IF CODE # 0
    THEN CALL !REPORT.ERROR
    ('MY.COMMAND','!MESSAGE',CODE:@FM:USERNAME)
Syntax
    CALL !SET.PTR (print.channel, width, length, top.margin, bottom.margin,
          mode, options)
Description
Use the !SET.PTR subroutine to set options for a logical print channel. This subrou-
tine provides the same functionality as the SETPTR command.
print.channel is the logical printer number, –1 through 255. The default is 0.
width is the page width. The default is 132.
length is the page length. The default is 66.
top.margin is the number of lines left at the top of the page. The default is 3.
bottom.margin is the number of lines left at the bottom of the page. The default is 3.
mode is a number 1 through 5 that indicates the output medium, as follows:
    1 - Line Printer Spooler Output (default).
    2, 4, 5 - Assigned Device. To send output to an assigned device, you must first
    assign the device to a logical print channel, using the ASSIGN command. The
    ASSIGN command issues an automatic SETPTR command using the default
    parameters, except for mode, which it sets to 2. Use !SET.PTR only if you have
    to change the default parameters.
    3 - Hold File Output. Mode 3 directs all printer output to a file called
    &HOLD&. If a &HOLD& file does not exist in your account, !SET.PTR creates
    the file and its dictionary (D_&HOLD&). You must execute !SET.PTR with
    mode 3 before each report to create unique report names in &HOLD&. If the
    report exists with the same name, the new report overwrites.
options are any of the printer options that are valid for the SETPTR command.
These must be separated by commas and enclosed by valid quotation marks.
If you want to leave a characteristic unchanged, supply an empty string argument
and specify the option NODEFAULT. If you want the default to be selected, supply
an empty string argument without specifying the NODEFAULT option.
Example
The following example sets the options so that printing is deferred until 12:00, and
the job is retained in the queue:
    CALL !SET.PTR (0,80,60,3,3,1,'DEFER 12:00,RETAIN')
Syntax
    CALL !SETPU (key, print.channel, new.value, return.code)
Description
Use the !SETPU subroutine to set individual parameters of any logical print
channel.
Unlike !SET.PTR, you can specify only individual parameters to change; you need
not specify parameters you do not want to change. See the description of the
!GETPU subroutine for a way to read individual print.channel parameters.
key is a number indicating the parameter to be set (see “Equate Names for Keys”).
print.channel is the logical print channel, designated by –1 through 255.
new.value is the value to which you want to set the parameter.
return.code is the returned error code (see “Equate Names for Return Code”).
The !SETPU subroutine lets you change individual parameters of logical print
channels as designated by print.channel. Print channel 0 is the terminal unless a
PRINTER ON statement has been executed to send output to the default printer. If
you specify print channel –1, the output is directed to the terminal, regardless of
the status of PRINTER ON or OFF.
Bit        Description
1          Uses FORTRAN-format mode. This allows the attaching of vertical
           format information to each line of the data file. The first character
           position of each line from the file does not appear in the printed
           output, and is interpreted as follows:
           Character       Meaning
           0               Advances two lines.
           1               Ejects to the top of the next page.
           +               Overprints the last line.
           Space           Advances one line.
           –               Advances three lines (skip two lines). Any other char-
                           acter is interpreted as advance one line.
3          Generates line numbers at the left margin.
4          Suppresses header page.
Bit        Description
5          Suppresses final page eject after printing.
12         Spools the number of copies specified in an earlier !SETPU call.
21         Places the job in the spool queue in the hold state.
22         Retains jobs in the spool queue in the hold state after they have been
           printed.
other      All the remaining bits are reserved.
Code         Meaning
0            No error
E$BKEY       Bad key (key is out of range)
E$BPAR       Bad parameter (value of new.value is out of range)
E$BUNT       Bad unit number (value of print.channel is out of range)
E$NRIT       No write (attempt to set a read-only parameter)
Examples
In the following example, the file containing the parameter key equate names is
inserted with the $INCLUDE compiler directive. Later, the top margin parameter
for logical print channel 0 is set to 10 lines. Return codes are returned in the argu-
ment RETURN.CODE.
Syntax
      CALL !TIMDAT (variable)
Description
Use the !TIMDAT subroutine to return a dynamic array containing the time, date,
and other related information. The !TIMDAT subroutine returns a 13-element
dynamic array containing information shown in the following list.
variable is the name of the variable to which the dynamic array is to be assigned.
 Field              Description
 1                  Month (two digits).
 2                  Day of month (two digits).
 3                  Year (two digits).
 4                  Minutes since midnight (integer).
 5                  Seconds into the minute (integer).
 6                  Ticks1 of last second since midnight (integer). Always
                    returns 0.
 7                  CPU seconds used since entering DataStage.
 8                  Ticks of last second used since login (integer).
 9                  Disk I/O seconds used since entering DataStage. Always
                    returns –1.
 10                 Ticks of last disk I/O second used since login (integer).
                    Always returns –1.
 11                 Number of ticks per second.
 12                 User number.
 13                 Login ID (user ID).
1. Tick refers to the unit of time your system uses to measure real time.
Use the following functions for alternative ways of obtaining time and date
information:
Example
    CALL !TIMDAT(DYNARRAY)
    FOR X = 1 TO 13
       PRINT 'ELEMENT ':X:', DYNARRAY
    NEXT X
Syntax
    CALL !USER.TYPE (type, admin)
Description
Use the !USER.TYPE subroutine to return the user type of the current process and
a flag to indicate if the user is an Administrator.
type is a value that indicates the type of process making the subroutine call. type can
be either of the following:
admin is a value that indicates if the user making the call is an Administrator.
Possible values of admin are 1, if the user is an Administrator, and 0, if the user is
not an Administrator.
An insert file of equate names is provided for the !USER.TYPE values. To use the
equate names, specify the directive $INCLUDE SYSCOM USER_TYPES.H when
you compile your program. (For PI/open compatibility you can specify
$INCLUDE SYSCOM USER_TYPES.INS.IBAS.)
Example
In this example, the !USER.TYPE subroutine is called to determine the type of user.
If the user is a phantom, the program stops. If the user is not a phantom, the
program sends a message to the terminal and continues processing.
    ERROR.ACCOUNTS.FILE: CALL !USER.TYPE(TYPE, ADMIN)
    IF TYPE = U&PH THEN STOP
      ELSE PRINT 'Error on opening ACCOUNTS file'
Description
Use the !VOC.PATHNAME subroutine to extract the pathnames for the data file or
the file dictionary of a specified VOC entry.
data/dict (input) indicates the file dictionary or data file, as follows:
     IK$DICT or 'DICT' returns the pathname of the file dictionary of the specified
     VOC entry.
     IK$DATA or ' ' returns the pathname (or pathnames for distributed files) of the
     data file of the specified VOC entry.
voc.entry is the record ID in the VOC.
result (output) is the resulting pathnames.
status (output) is the returned status of the operation.
An insert file of equate names is provided for the data/dict values. To use the equate
names, specify the directive $INCLUDE SYSCOM INFO_KEYS.H when you
compile your program. (For PI/open compatibility you can specify $INCLUDE
SYSCOM INFO_KEYS.INS.IBAS.)
The result of the operation is returned in the status argument, and has one of the
following values:
 Value             Result
 0                 The operation executed successfully.
 IE$PAR            A bad parameter was used in data/dict or voc.entry.
 IE$RNF            The VOC entry record cannot be found.
Example
     CALL !VOC.PATHNAME (IK$DATA,"VOC",VOC.PATH,STATUS)
     IF STATUS = 0
     THEN PRINT "VOC PATHNAME = ":VOC.PATH
If the user’s current working directory is /usr/account, the output is:
       VOC PATHNAME =       /usr/accounts/VOC
                                                                  Index-1
@COMMAND.STACK variable E-1           @RECUR1 variable E-4
@CONV variable E-1                    @RECUR2 variable E-4
@CRTHIGH variable E-2                 @RECUR3 variable E-4
@CRTWIDE variable E-2                 @RECUR4 variable E-4
@DATA.PENDING variable E-2            @SCHEMA variable E-4
@DATE variable E-2                    @SELECTED variable E-4
@DAY variable E-2                     @SENTENCE variable 6-533, E-4
@DICT variable E-2                    @SM variable E-4
@FILE.NAME variable E-2               @SQL.CODE variable E-4
@FILENAME variable E-2                @SQL.DATE variable E-4
@FM variable E-2                      @SQL.ERROR variable E-4
@FORMAT variable E-2                  @SQL.STATE variable E-4
@HDBC variable E-2                    @SQL.TIME variable E-4
@HEADER variable E-2                  @SQL.WARNING variable E-4
@HENV variable E-2                    @SQLPROC.NAME variable E-4
@HSTMT variable E-2                   @SQLPROC.TX.LEVEL variable E-4
@ID variable E-2                      @STDFIL variable 6-423, 6-432, E-4
@IM variable E-2                      @SVM variable E-4
@ISOLATION variable 4-9, E-2          @SYS.BELL variable E-4
@LEVEL variable E-2                   @SYSTEM.RETURN.CODE
@LOGNAME variable E-2                        variable E-5
@LPTRHIGH variable E-2                @SYSTEM.SET variable E-5
@LPTRWIDE variable E-3                @TERM.TYPE variable E-5
@MONTH variable E-3                   @TIME variable E-5
@MV variable E-3                      @TM variable E-5
@NB variable E-3                      @TRANSACTION variable 4-10, E-5
@ND variable E-3                      @TRANSACTION.ID variable 4-10,
@NEW variable E-3                            E-5
@NI variable E-3                      @TRANSACTION.LEVEL
@NS variable E-3                             variable 4-10, E-5
@NULL variable 2-4, 2-15, 6-62, E-3   @TRUE variable E-5
@NULL.STR variable 2-4, 2-15, 6-62,   @TTY variable E-5
        6-95, 6-96, E-3               @USER.NO variable E-5
@NV variable E-3                      @USER.RETURN.CODE variable E-5
@OLD variable E-3                     @USER0 variable E-5
@OPTION variable E-3                  @USER1 variable E-5
@PARASENTENCE variable E-3            @USER2 variable E-5
@PATH variable E-3                    @USER3 variable E-5
@RECCOUNT variable E-4                @USER4 variable E-5
@RECORD variable E-4                  @USERNO variable E-5
   and ITYPE function 6-341           @variables 4-9, E-1–E-6
@RECUR0 variable E-4                  @VM variable E-5
                                                                        Index-3
   local cataloging 3-11            CAT operator 2-14
   normal cataloging 3-11           CATALOG command 3-12
   object code 3-9                  catalog shared memory 3-13
      listing 5-12                  cataloging BASIC programs
   printing 3-2, 3-4                   globally 3-11
   running 3-10                        locally 3-11
   storing 1-7                         normally 3-11
BASIC statements 1-2                CATS function 6-92
BEGIN CASE statement 6-89           CHAIN statement 6-93
BEGIN TRANSACTION                   CHANGE function 6-94
         statement 6-69             CHAR function 6-95
binary conversion C-36              CHAR(0) 2-2
BITAND function 6-70                CHAR(10) 2-2
BITNOT function 6-71                CHAR(128) 6-95, 6-96, 6-226
BITOR function 6-72                 CHAR(252) 2-7
BITRESET function 6-73              CHAR(253) 2-7
BITSET function 6-74                CHAR(254) 2-7
BITTEST function 6-75               character conversion C-23
BITXOR function 6-76                character set conversion C-17
blank spaces, see spaces            character strings 2-1–2-2
brackets                               ASCII 2-2
   angle (< >) 6-28, 6-219, 6-497      constants 2-2
   square ([ ]) 6-49                   empty 2-3, 2-5
Break key 5-3                          numeric 2-13
BREAK statement 6-77                   substrings 2-15
BSCAN statement 6-79                characters
BYTE function 6-82                     alphabetic 1-6
BYTELEN function 6-83                  numeric 1-6
BYTETYPE function 6-84                 special 1-6
BYTEVAL function 6-85               CHARS function 6-96
                                    CHECKSUM function 6-97
C                                   CLEAR statement 6-98
                                    CLEARDATA statement 6-99
C conversion C-8                    CLEARFILE statement 6-100
CALL statement 1-3, 6-86            clearing
    and RETURN statement 6-501         in-line prompts 6-102
    and SUBROUTINE                     select lists 6-103
         statement 6-571            CLEARPROMPTS statement 6-102,
calling subroutines 6-86                     6-309
CASE                                CLEARSELECT statement 6-103
    option 6-17                     CLOSE statement 6-105
    statement 6-89                  CLOSESEQ statement 6-107
                                                                 Index-5
CRT statement 6-127             DELETEU statement 6-141, 6-145
cursors, positioning 6-29       delimiters, system 2-7
                                DI conversion C-16
D                               DIM statement 6-146
                                DIMENSION statement 6-146
D conversion C-10               dimensioned arrays 2-6–2-7
data                            dirty reads 4-12
   anomalies 4-11               display length 2-12
   character string 2-1–2-2     DISPLAY statement 6-149
   null value 2-3               distributed files, status 6-561
   numeric 2-3                  DIV function 6-150
   preventing loss of 4-1       DIVS function 6-151
   visibility 4-6               documentation conventions 1-ix
DATA statement 6-128            DOWNCASE function 6-152
   and INPUT statements 6-320   DQUOTE function 6-153
data types 2-1–2-4              DSDetachJob function 6-156
   logical 2-19                 DSExecute subroutione 6-157
   null value 2-3               DSGetCustInfo function 6-157
date conversion C-10, C-16      DSGetJobInfo function 6-159
date format, default C-10       DSGetLinkInfo function 6-163
DATE function 6-130             DSGetLogEntry function 6-165
dates, internal system 6-130    DSGetLogSummary function 6-166
DCFLUSH function 6-131          DSGetNewestLogId function 6-168
DCOUNT function 6-132           DSGetParamInfo function 6-169
deadlocks 4-5                   DSGetProjectInfo function 6-172
DEBUG statement 5-3, 6-133      DSGetStageInfo function 6-173
debugger 5-1, 6-133             DSGetVarInfo function 6-176
DECATALOG command 3-13          DSLogEvent function 6-177
decimal equivalents B-5         DSLogFatal function 6-178
DEFFUN statement 6-135, 6-269   DSLogInfo function 6-179
defining                        DSLogWarn function 6-181
   control keys 6-344           DSSetJobLimit function 6-189
   escape keys 6-345            DSSetParam function 6-190
   function keys 6-345          DSSetUserStatus subroutine 6-191
   identifiers 3-6              DSStopJob function 6-192
   unsupported keys 6-347       DSTransformError function 6-193
DEL statement 6-137             DTX function 6-198
DELETE                          durability property 4-8
   function 6-139               dynamic arrays 2-7–2-10
   statement 6-141                 and arithmetic operators 2-22
DELETE.CATALOG command 3-12        and the null value 2-27
DELETELIST statement 6-144         and operators 2-21–2-28
                                                                         Index-7
    type 1 1-7, 3-1                        G
    type 19 1-7, 3-1
FILEUNLOCK statement 6-240                 G conversion C-21
FIND statement 6-242                       GE operator 2-17
FINDSTR statement 6-243                    GES function 6-271
FIX function 6-244                         GET statement 6-272
fixed-point constants 2-3                  GET(ARG.) statement 6-278
flavors 6-16                               GETLIST statement 6-280
    compatibility 3-6                      GETLOCALE function 6-281, 6-363
floating-point constants 2-3               GETREM function 6-282
floating-point numbers 2-3                 GETX statement 6-272, 6-277
FLUSH statement 6-246                      global cataloging 3-11
    after WRITESEQ statement 6-658         GOSUB statement 1-3, 6-283
FMT function 6-247                            with ON statement 6-420
FMTDP function 6-252                          and RETURN statement 6-501
FMTS function 6-253                        GOTO statement 6-285
FMTSDP function 6-254                         with ON statement 6-421
FMUL function 6-256                        granularity 4-1
FOLD function 6-257                        group extraction C-21
FOLDDP function 6-258                      GROUP function 6-286
FOOTING statement 6-259                    GROUPSTORE statement 6-288
FOR statement 6-263                        GT operator 2-17
FOR.INCR.BEF option 6-17, 6-265            GTS function 6-290
format expressions 2-11, 6-247–6-250
    in INPUT statements 6-321              H
FORMAT.OCONV option 6-17
formatting numbers C-28                    HEADER.BRK option 6-17
FORMLIST statement 6-267                   HEADER.DATE option 6-18, 6-262,
FSELECT option 6-17                               6-294
FSUB function 6-268                        HEADER.EJECT option 6-18, 6-292
function keys 6-345, 6-350, 6-353, F-15,   HEADING statement 6-291
          F-34                             HEADINGE statement 6-21, 6-292
    defining 6-345                         HEADINGN statement 6-21, 6-292
FUNCTION statement 6-269                   hexadecimal conversion C-36
functions                                  hexadecimal equivalents B-5
    intrinsic 1-2                          host name and SYSTEM
    numeric 1-3                                   function 6-580
    range C-45                             HUSH statement 6-296
    string 1-3
    user-written 6-503                     I
    vector 2-22
                                           ICHECK function 6-298
                                                                    Index-9
LEFT function 6-355                         logical operators 2-19
LEN function 6-356                              AND 2-19
LENDP function 6-357                            NOT 2-19
length function C-22                            and the null value 2-19
LENS function 6-358                             OR 2-19
LENSDP function 6-359                       LOOP statement 6-372
LES function 6-360                          loops
LET statement 6-361                             FOR...NEXT 6-263, 6-409
levels, see isolation levels                    LOOP...REPEAT 6-372
line number table, suppressing 3-2, 3-4     lost updates 4-11
list variables, see select list variables   LOWER function 6-375
listing object code 5-12                    LT operator 2-17
LN function 6-362                           LTS function 6-377
local cataloging 3-11
local variables 6-93                        M
LOCALEINFO function 6-363
LOCATE statement 6-365                      masked character conversions C-23
LOCATE.R83 option 6-19                      MAT statement 6-378
lock escalation, example 4-14               MATBUILD statement 6-380
LOCK statement 6-370                        MATCH operator 2-18, 6-381
locks                                       MATCHFIELD function 6-383
    compatibility 4-1                       mathematical functions C-4, C-18
    deadlocks 4-5                           MATPARSE statement 6-385
    exclusive file lock 4-5                   and INMAT function 6-318
    file lock 6-240                         MATREAD statement 6-388
    granularity 4-1                           and INMAT function 6-318
    intent file lock 4-4                    MATREADL statement 6-391
    and MATREADL statement 6-391              and INMAT function 6-318
    and MATREADU statement 6-391            matrices 2-6
    process lock 6-642                        zero element 2-7
    and READL statement 6-464               MATWRITE statement 6-395
    and READU statement 6-397,              MATWRITEU statement 6-397
          6-464, 6-651                      MAXIMUM function 6-400
    and READVL statement 6-465              MAXRLOCK parameter 4-14
    releasing 6-397, 6-488, 6-642, 6-651    MB conversion C-36
    semaphore lock 6-370                    MC conversion C-23
    shared file lock 4-4                    MD conversion C-25
    shared record lock 4-2                  messages
    transactions and 4-8                      error 3-9, 6-52, 6-210, 6-450, 6-564
    types 4-2, 6-237                          warning 3-9
    update record lock 4-3                  MINIMUM function 6-401
    well-formed writes and 4-13             ML conversion C-28
                                                             Index-11
   concatenation 2-14                   process locks 6-642
   and dynamic arrays 2-21–2-28         PROCREAD statement 6-452
   logical 2-19                         PROCWRITE statement 6-453
   pattern matching 2-18                PROGRAM statement 6-454
   relational 2-16–2-18                 PROMPT statement 6-455
   string 2-14                             and INPUT statements 6-321
   substring 2-15                       prompts, see in-line prompts
operators, assignment 6-28, 6-62        PWR function 6-456
OR operator 2-19, 6-72, 6-441
ORS function 6-441                      Q
P                                       Q conversion C-43
                                        quotation marks in character
P conversion C-42                               strings 2-2
packed decimal conversion C-33          QUOTE function 6-457
PAGE statement 6-442
part files, status 6-561                R
part numbers, status 6-561
passing                                 R conversion C-45
   arrays to subroutines 6-87, 6-147,   RADIANS option 6-20
         6-571                          RAID 5-5
   variables to subroutines 6-147,      RAID (debugger) 6-133
         6-571                             commands 5-4, 6-133
pattern matching 2-18, 6-381, C-42         description 5-1
   and empty strings 2-18               RAID command 5-2
   codes 6-381                             options 5-2
PCLOSE.ALL option 6-19                     suppressing execution of 3-2
PERF.EQ.EXEC option 6-19, 6-443         RAISE function 6-458
PERFORM statement 6-443                 RANDOMIZE statement 6-460
PHANTOM command 3-1                     range function C-45
phantom writes 4-12                     RAW.OUTPUT option 6-20
PIOPEN.INCLUDE option 6-19              READ statement 6-461
PIOPEN.MATREAD option 6-19              READ.RETAIN option 6-20, 6-465
PIOPEN.SELIDX option 6-20, 6-529        READBLK statement 6-467
pointer (REMOVE) 6-282, 6-506, 6-540       and TIMEOUT statement 6-606
PRECISION statement 6-445               READL locks, see shared record locks
PRINT statement 6-446                   READL statement 6-461
   and INPUT statements 6-321           READLIST statement 6-470
   and TABSTOP statement 6-582          READNEXT statement 6-472
PRINTER CLOSE statement 6-448              and READLIST statement 6-470
PRINTER statement 6-448                    and SELECT statement 6-525
PRINTERR statement 6-450                READSEQ statement 6-474
                                                                        Index-13
serializability                            cross-reference table of 3-2, 3-3
    lost updates and 4-12                  definition 1-5
    property 4-8                       statements 1-2
SET TRANSACTION ISOLATION                  assignment 1-4
          LEVEL statement 4-8, 6-536       control 1-4
SETLOCALE function 6-538                   types 1-4
SETPTR command F-41                    STATIC.DIM option 2-7, 6-20, 6-147
SETREM statement 6-540                 STATUS function 6-555
shared file locks 4-4                      after BSCAN statement 6-80, 6-555
shared memory 3-13                         after DELETE statement 6-555
shared record locks 4-2                    after FILELOCK statement 6-556
SIN function 6-541                         after FMT function 6-556
SINH function 6-542                        after GET statement 6-556
SLEEP statement 6-543                      after GETX statement 6-556
SMUL function 6-544                        after HUSH statement 6-296
soundex conversion C-46                    after ICONV function 6-301, 6-556
SOUNDEX function 6-545                     after ICONVS function 6-303
source code 1-2                            after INPUT @ statement 6-556
    comments 1-4                           after MATWRITE statement 6-556
    spaces in 1-6                          after OCONV function 6-416, 6-556
    syntax 1-4                             after OCONVS function 6-419
    tabs in 1-6                            after OPEN statements 6-557
SPACE function 6-546                       after OPENPATH statement 6-433
spaces                                     after READ statement 6-557
    in numeric constants 2-3               after READBLK statement 6-558
    in source code 1-6                     after READL statement 6-558
    removing 6-620, 6-622, 6-623,          after READSEQ statement 6-474,
          6-624, 6-625, 6-626, 6-640             6-558
SPACES function 6-547                      after READT statement 6-477,
special characters 1-6                           6-508, 6-558, 6-645, 6-663
SPLICE function 6-548                      after READU statement 6-558
SQRT function 6-549                        after READVL statement 6-558
square brackets ([ ]) 6-49                 after READVU statement 6-558
SQUOTE function 6-550                      after RECORDLOCKED
SSELECT statement 6-551                          statement 6-487
SSELECTN statement 6-553                   after REWIND statement 6-558
SSELECTV statement 6-551                   after RPC.CALL function 6-513,
SSUB function 6-554                              6-558
standard arrays 2-7                        after RPC.CONNECT
    matrices 2-6                                 function 6-515, 6-558
    vectors 2-6                            after RPC.DISCONNECT
statement labels                                 function 6-517, 6-558
                                                                 Index-15
TRANSACTION COMMIT                UNISEQ function 6-640
         statement 6-618          UNISEQS function 6-641
TRANSACTION START                 UNIX vi editor 1-8
         statement 6-619          UNLOCK statement 6-642
transaction statements 6-615         and LOCK statement 6-370
transaction variables             unnamed common variables 2-6
   @ISOLATION 4-9                 unnamed common, saving variable
   @TRANSACTION 4-10                      values 6-93
   @TRANSACTION.ID 4-10           unsupported keys, defining 6-347
   @TRANSACTION.LEVEL 4-10        UNTIL statement 6-263, 6-372
transactions 4-6                  UPCASE function 6-643
   @variables 4-9                 update record locks 4-3
   active 4-6                     UPRINT statement 6-644
   and data visibility 4-6        USE.ERRMSG option 6-21, 6-451
   example 4-9                    user ID 6-65
   isolation levels 4-8           user-written functions 6-503
   locks and 4-8                  UVLOCALE.H include file 6-281,
   nested 4-6                             6-363
      committing 6-112
   properties 4-7                 V
   and RELEASE statements 6-488
   subtransactions 4-6            value marks 2-7
TRIM function 6-620               VAR.SELECT option 6-21, 6-103,
TRIMB function 6-622                       6-280, 6-470, 6-471, 6-526,
TRIMBS function 6-623                      6-527, 6-552, 6-553
TRIMF function 6-624              variables 1-2, 2-5–2-10
TRIMFS function 6-625                array 2-6–2-10
TRIMS function 6-626                 assigning 6-62, 6-361
TTYCTL statement 6-627               common 6-114
TTYGET statement 6-629               definition 1-2
   values 6-630                      file 2-10
TTYSET statement 6-635               in RAID 5-3
type 1 files 1-7, 3-1                in user-written functions 6-269
type 19 files 1-7, 3-1               local 6-93
                                     named common 2-6
U                                    names 1-2, 2-5
                                     passing to subroutines 6-147, 6-571
UID, effective 6-65                  saving in unnamed common 6-93
ULT.FORMAT option 6-21               select list 2-10, 6-103, 6-528, 6-551
UNASSIGNED function 6-637            transaction 4-9
UNICHAR function 6-638               unnamed common 2-6, 6-93
UNICHARS function 6-639           VEC.MATH option 2-14, 2-22, 6-21
W
warning messages 3-9
well-formed write 4-13
WEOF statement 6-645
WEOFSEQ statement 6-645, 6-646
WHILE statement 6-263, 6-372
WIDE.IF option 6-21
WRITE statement 6-648
WRITEBLK statement 6-654
WRITELIST statement 6-656
WRITESEQ statement 6-657, 6-660
WRITESEQF statement 6-660
WRITET statement 6-662
WRITEU statement 6-648
WRITEV statement 6-648
WRITEVU statement 6-648
writing
   data, see loading data
X
XLATE function 6-667
XTD function 6-669
Z
zero element 2-7
                                   Index-17
Index-18   Ascential DataStage BASIC Guide