This tutorial is intended for experienced programmers who want to learn RPG IV (also known as ILE RPG).
This tutorial assumes.
The coding examples are intended to show the feature being discussed, but they sometimes have a "Bonus features for this example" section following the example that lists additional interesting things to note about the example, usually things that have not been discussed yet. Readers can choose to ignore those "Bonus features" sections without worrying about missing something, since the material is covered in later chapters.
Chapter 1: Hello world
This chapter takes you through the steps to create and run an RPG IV program.
It assumes you are using one of two ways to do your coding:
If you are using RDi:
If you are using SEU:
Entering the RPG program statements in "fully-free" form
**free dsply 'Hello World'; return;
"Fully free form" vs "column-limited free form"
RPG has two modes of free-form code. The historical mode (column-limited mode) requires that all free-form code must be coded between columns 8 and 80. Fully-free mode allows free-form code in any column, with no limit on line-length.
Fully free-form source must have **FREE in column 1 of line 1. All the code must be free-form.
Non-fully-free source, or column-limited source does not have **FREE in line 1. You can mix fixed-form code (code that uses columns 6 and 7) and free-form code in column-limited source.
This tutorial assumes you have **FREE at the beginning of your source, but it also works if you use column-limited source, if you ensure that columns 1-7 are blank and that you don't go past column 80. The examples have 7 blanks at the beginning to make it easy to copy and paste into column-limited source, but if you have **FREE at the beginning of your source, you can remove those blanks if you like.
Dealing with compiler errors
dsply name; return;
dcl-s name char(10) inz('Jim'); dsply name; return;
Compile your program again, specifying a debug view other than *NONE or *STMT. For example, specify DBGVIEW(*LIST). (But *ALL, *SOURCE, or *COPY would all work fine here.)
Now, start the debugger. For now, I'll just mention the system debugger, but there is a more intuitive debugger associated with RDi that you can investigate separately.
Assuming your program is called QTEMP/MYPGM, do the following commands:
===> STRDBG QTEMP/MYPGM - at this point, you can just use F10 to exit the screen - when you call the program, it stops on the first executable statement - or you could also use F6 to set a breakpoint on a particular statement ===> CALL QTEMP/MYPGM - when you get a breakpoint, step (F10) through the program - display variables (F11) along the way ===> ENDDBG
From now on, I'll assume you know
Chapter 2: General info about RPG
This chapter introduces you to general information about RPG source code.
RPG source code
For free-form statements, the RPG compiler uses only columns 8 - 80 of a source member. Columns 1 -5 and columns 81+ are used for comments. Columns 7 and 8 are only used for fixed form statements. The only fixed-form statements you need to use are I and O statements, and they are rarely used in RPG nowadays.
There are several types of RPG statements.
The specifications must appear in the order given above, but it is not necessary to code all the specifications. A module could contain just a single C spec.
There are fixed-form versions of all the statements, but the only ones you might need to use are I and O statements, or a few historical types of File or Calculation statements. This tutorial does not cover fixed-form code. You can see many examples in the ILE RPG Reference.
RPG was originally created to handle files. The intention was that the program would read a record from the "primary file" and for each record, it would perform the calculations. When it reached the last record, it would close the file and end the program.
To artificially create the "last record" situation, the program could set on the "Last Record" indicator, called *INLR.
For programs that do not have a primary file, it is still necessary to stop the program from looping through the calculations. This can be done two ways:
*inlr = '1';
There are subtle differences in these two mechanisms that are discussed in later chapters.
Chapter 3: Variables and procedures
This chapter introduces you to defining constants, variables and prototypes.
Defining constants, variables and prototypes
The Definition statement, often called the "D spec" is used to define constants, variables, prototypes, and "procedure interfaces". The statement begins with DCL-x where x is C, S, DS, PR, or PI, depending on the type of item being defined. This is followed by the name, and then one or more keywords, then a semicolon.
Most of the keywords can go in any order you like, but if you have a data-type keyword such as CHAR or PACKED, it must be the first keyword.
Define a constant
The constant is the simplest RPG definition. It just has one keyword, CONST, and it's optional to actually specify the keyword. 100 and CONST(100) mean exactly the same thing for a constant definition.
Paste the following code into a source member, then compile and run it.
dcl-c MAX_ELEMS 100; dcl-c default_city_name 'London'; dsply max_elems; dsply default_city_name; return;
Define a standalone field
TThe standalone field is another simple RPG definition. A standalone field is an ordinary scalar variable. It has a name, a definition type of 'S', and a type.
Paste the following code into a source member, then compile and run it.
dcl-s num int(10); for num = 1 to 3; dsply ('i = ' + %char(num)); endfor; return;
The definition statement defines a field named "num" . The INT keyword indicates that the field is an integer. The "10" indicates that it has 10 digits; this is a 4-byte integer. The "s" in DCL-S indicates that it is a standalone field.
Bonus features in this example:
Define a data structure
A data structure is defined with one statement the data structure itself, using DCL-DS. This is followed by one statement for each subfield. Subfield statements just start with the name of the subfield. The data structure usually ends with an END-DS statement. If you are defining the data structure like another data structure by using the LIKEDS keyword, or like a record format by using the LIKEREC keyword, then you don't code END-DS.
Paste the following code into a source member, then compile and run it.
Note: This example uses the INZ keyword, which provides an initialization for the subfield. You can also code the INZ keyword for a standalone field.
dcl-ds info qualified; name char(10) inz('Sam'); salary packed(9 : 2) inz(50000.25); end-ds; dcl-ds otherInfo likeds(info) inz(*likeds); dsply (info.name + ' has a salary of' + %char(info.salary)); otherInfo.name = 'Joe'; otherInfo.salary += 10000; dsply (otherInfo.name + ' has a salary of' + %char(otherInfo.salary)); return;
Bonus features in this example:
Define an array
You can define an array of scalars or an array of data structures.
RPG supports only one dimension for arrays. Multiple-dimension arrays can be simulated by using data structure arrays with array subfields; instead of coding cell(i j k) you would code table(i).row(j).col(k).
The dimension of the array is specified by using the DIM keyword.
Array indexes are specified with parentheses.
Define a scalar array
Paste the following code into a source member, then compile and run it.
dcl-s dates date(*iso) dim(3); dates(1) = %date(); // the current date dates(2) = %date() + %days(1); // tomorrow dates(3) = %date() + %years(1); // next year dsply (%char(dates(1)) + ' ' + %char(dates(2)) + ' ' + %char(dates(3))); return;
Bonus features in this example:
Define a data structure array
Note: This example requires a 7.2 compiler, and you might need a PTF. See http://ibm.biz/spring_2017_rpg_enhancements for PTF details.
Paste the following code into a source member, then compile and run it.
dcl-ds families qualified dim(5); address varchar(50); numPeople uns(3); dcl-ds people dim(8); name varchar(25); age packed(5); end-ds; end-ds; dcl-s numFamilies uns(5) inz(0); dcl-s i int(10); dcl-s j int(10); families(1).address = '10 Mockingbird Lane'; families(1).people(1).name = 'Alice'; families(1).people(1).age = 3; families(1).people(2).name = 'Bill'; families(1).people(2).age = 15; families(1).numPeople = 2; numFamilies = 1; for i = 1 to numFamilies; dsply (families(i).address); for j = 1 to families(i).numPeople; dsply (families(i).people(j).name + ' is ' + %char(families(i).people(j).age) + ' years old.'); endfor; endfor; return;
Bonus features in this example:
Define a prototype
RPG prototypes describe how to call a program, procedure, or Java method.
The definition statement starts with DCL-PR. Similar to data stuctures, the DCL-PR statement is followed by parameter definitions, and then the prototype is ended with an END-PR statement. You always need the END-PR statement for a prototype.
The EXTPROC or EXTPGM keyword indicates whether it is calling a procedure or program and it also indicates exactly which procedure or program to call. (Calls to Java methods also use the EXTPROC keyword.)
If neither the EXTPROC nor EXTPGM keyword is coded, the EXTPROC keyword is assumed.
Example 1: Call a program
A common program to call is QCMDEXC. This program runs a system command.
dcl-pr qcmdexc extpgm('QCMDEXC'); theCmd char(3000) const; cmdLen packed(15 : 5) const; dbcs char(3) const options(*nopass); end-pr; dcl-s cmd varchar(100); cmd = 'DSPJOB OUTPUT(*PRINT)'; qcmdexc (cmd : %len(cmd)); qcmdexc ('WRKSPLF' : 7); return;
Bonus features in this example:
Note: If the RPG prototype name is the same as the actual program name, you can just code EXTPGM with no parameter. The RPG compiler uppercases the prototype name to determine the actual program name. Here is an alternate version of the QCMDEXC prototype.
dcl-pr qcmdexc extpgm; theCmd char(3000) const; cmdLen packed(15 : 5) const; dbcs char(3) const options(*nopass); end-pr;
Example 2: Call a procedure
For this example, we call the C runtime printf() function to print a message to the standard output instead of to the external message queue.
/if defined(*CRTBNDRPG) ctl-opt dftactgrp(*no) actgrp(*new); /endif ctl-opt option(*srcstmt); dcl-s num int(10) inz(25); print ('This message is much longer than the 52 ' + 'characters that DSPLY allows. ' + 'The value of variable "num" is ' + %char(num)); return; dcl-proc print; dcl-pi *n; msg varchar(5000) const; end-pi; dcl-pr printf extproc(*dclcase); template pointer value options(*string); dummy int(10) value options(*nopass); end-pr; dcl-c NEWLINE x'15'; printf(msg + NEWLINE); end-proc print;
Bonus features in this example:
Chapter 4: Introduction to files
This chapter introduces you to using files.
The file statement
The File statement is used to define a file to be used in the RPG module. File statements start with DCL-F (declare file). Like definition statements, the file statement starts with the name to be used for the file in the RPG module. The name is followed by keywords, then a semicolon. If you code a device keyword to say what type of file it is (DISK, PRINTER, WORKSTN), then the device keyword must be the first keyword.
A simple example
Let's start with a little example where we just read all the records of a file.
First, let's get a file to read. Enter the following command on the command line. The command produces a file MYLIB/RPGTESTF that lists the *FILE objects in QGPL whose names start with QRPG. (For this example, change "MYLIB" to the name of your own library, in both the DSPOBJD command and the RPG program)
===> DSPOBJD OBJ(QGPL/QRPG*) OBJTYPE(*FILE) OUTPUT(*OUTFILE) OUTFILE(MYLIB/RPGTESTF)
Now, compile and run the following RPG program.
When you run the program, just press ENTER on each DSPLY that shows up.
dcl-f rpgtestf usropn extdesc('MYLIB/RPGTESTF') extfile(*extdesc); open rpgtestf; read rpgtestf; dow not %eof; dsply ODOBNM; read rpgtestf; enddo; close rpgtestf; return;
If MYLIB is not in your library list at compile time and runtime, change your DCL-F command to the following, adding the EXTDESC and EXTFILE keywords so that the system can find the file. Add those keywords to all the examples throughout this chapter that use file RPGTESTF.
dcl-f rpgtestf usropn extdesc('MYLIB/RPGTESTF') extfile(*extdesc);
If you haven't seen the power of RPG before, you might be wondering where ODOBNM comes from.
Try running it under debug by using the listing view. (If you forgot to compile with DBGVIEW(*ALL) or DBGVIEW(*LIST), compile it again.)
If you compile with DBGVIEW(*ALL), you have to choose the listing view while you are debugging.
When you first see the debug listing view, it looks very different from your original code. You see several RPG statements that were generated by the RPG compiler. These are "Input specifications", and they describe the input buffer of the RPGTESTF file. There is one I spec for each field in the file.
When you step through the program, you notice that you only get a breakpoint on the ODOBNM I spec. That is because the RPG program didn't use any of the other fields, so the RPG compiler did an optimization to avoid loading the data for those other fields.
You also notice that you step to the I spec and DSPLY opcode twice (at least it was twice on my system, once for QRPGLESRC and once for QRPGSRC).
Using externally-described files
The previous program defined RPGTESTF as an externally-described file (to make it a program described file, we would have had to code DISK(100) or whatever the record length is). The RPG compiler "extracts" the file definition as part of the compile, so it knows the names and buffer positions of all the fields in the file.
Most RPG programmers use explicit loops to read a file, but they allow the RPG compiler to implicitly open and close the file. Let's try that.
dcl-f rpgtestf; read rpgtestf; dow not %eof; dsply ODOBNM; read rpgtestf; enddo;
The RPG compiler implicitly opens the file when you call your program.
But what about closing the file? The RPG compiler does not always close files when a program ends by using the RETURN operation. It only closes files when it finds the "Last Record" indicator, *INLR, to be on. You can simply set *INLR on at some point before reaching the end of calculations, or you can set on *INLR and immediately return. Many RPG programmers set *INLR on as the very first calculation, as a visible clue that the calculations are only meant to be run once. Other RPG programmers set *INLR on at the end of calculations. Either way works fine to cause the calculations to end and to cause the file to be closed.
Try compiling the program. Note that it doesn't have a RETURN operation or code to set *INLR on. The RPG compiler issues message RNF7023 saying that it doesn't know how the program will end. Without a RETURN operation, or *INLR on, the program just loops doing the calculations over and over.
Here is the final corrected program, with *INLR set on as the first statement in the calculations. (Remember that it doesn't matter where *INLR is set on, as long as it is on at the end of the calculations.)
dcl-f rpgtestf; *inlr = '1'; read rpgtestf; dow not %eof; dsply ODOBNM; read rpgtestf; enddo;
Other types of files
The previous examples all used a database file. The RPG device type for a database file is DISK. This is the default, so we didn't have to code it.
The other two most commonly used device types in RPG are printer files (PRINTER) and display files (WORKSTN). Display files are handled in a later chapter. Here is a little example of a program-described printer file.
dcl-c QPRINT_LEN 132; dcl-f qprint printer(QPRINT_LEN); dcl-ds qprint_ds len(QPRINT_LEN) end-ds; *inlr = *on; qprint_ds = 'Hello'; write qprint qprint_ds; qprint_ds = 'world'; write qprint qprint_ds;
Output and update
The previous examples all opened the file for input only. The default usage for a DISK file is USAGE(*INPUT). If you want to write to the file, code USAGE(*OUTPUT). If you want to be able to update the file, code USAGE(*UPDATE). If you want to do both, code USAGE(*OUTPUT:*UPDATE).
dcl-f rpgtestf usage(*update); read rpgtestf; dow not %eof; dsply 'new name' '' ODOBNM; update QLIDOBJD; read rpgtestf; enddo; *inlr = *on;
This program uses the UPDATE operation. Unlike the READ operation, which can be used with either a file name or a record format name, the UPDATE operation can only be used with a record format name. You can use DSPFD to find out the name of the format, or you can just look in one of the RPG compiler listings for the earlier versions of the program.
When you run this program, the DSPLY operation waits until you enter a value. The value you enter becomes the new value for the ODOBNM variable, and that value is used when the record is updated by the UPDATE operation. If you display the file by using DSPPFM, you see that the file name has been changed.
dcl-f rpgtestf usage(*output); ODOBNM = 'ABCDE'; write QLIDOBJD; *inlr = '1';
The WRITE operation also needs to have the record format name rather than the file name.
When you run this program, it adds a new record to the end of the file. In the new record, all the values are defaulted to blanks or zeros except the file name. Use DSPPFM to display the file again to see the new record.
To see the names of other fields you could set before you do the WRITE, use command DSPFFD RPGTESTF.
Exercises related to implicit open and close
Using the example from 'Implicitly opening and closing the file' as an example, remove the assignment to *INLR, and add a RETURN operation at the end of the calculations.
dcl-f rpgtestf; read rpgtestf; dow not %eof; dsply ODOBNM; read rpgtestf; enddo; return;
Call the program twice.
Why does the program only produce output the first time it is called?
Using the version of the program with the USROPN keyword as an example, remove the CLOSE operation (you can just comment it out by using //).
dcl-f rpgtestf usropn; open rpgtestf; read rpgtestf; dow not %eof; dsply ODOBNM; read rpgtestf; enddo; // close rpgtestf; return;
Call the program twice.
Why does the program get an error the second time it is called?