HPlogo   NetIPC 3000/XL Programmer's Reference Manual:
HP 3000 MPE/iX Computer Systems
> Chapter 4 NetIPC Examples

Example 1

MPE documents

Complete PDF

 

Table of Contents

Glossary

Index

 

⇓ Page Bottom

 
Program 1A
Program 1B
Program 1C
 
Runtime Output

⇑ Page Top

 

Chapter 4 NetIPC Examples

Example 2

In the first two programs (1A and 1B), the lengths of the data messages are not known. The sending side (Program 1A) includes the length of each message as the first two bytes of each message it sends. The receiving side (Program 1B) executes two IPCRECV loops for each message: first to receive the length and then to receive the data.

The first program (Program 1A):
  • looks up the call socket named RALPH located on node JANE and gets back a destination descriptor;
  • creates its own call socket;
  • sends a connection request to RALPH;
  • shuts down its call socket and its destination socket;
  • completes the connection;
  • executes a loop in which it:

    • reads a line of data;
    • stores the length (number of bytes) of the data in the first part of the message;
    • stores the data itself in the second part of the message;
    • sends the message on the connection, including the message length as the first two bytes of the message;
  • sends a "last message" which will be recognized by the receiving program as a termination request;
  • receives a "termination confirmation message" and shuts down the connection by releasing its VC socket.
The second program (Program 1B):
  • creates a call socket and names it RALPH;
  • waits to receive a connection request;
  • shuts down its call socket;
  • executes a loop in which it:

    • calls a procedure that receives a message by executing two IPCRECV loops (the first loop determines the incoming message length and the second loop receives data until all the pieces of the message have been received);
    • prints the message which was received;
  • receives a "last message" termination request;
  • sends a "termination confirmation message" in response to the termination request;
  • receives a result parameter value of 64 ("REMOTE ABORTED CONNECTION") in response to a receive request;
  • releases its VC socket.
Note
NOTE: 3kRanger made changes marked by comment {3k} to correct compile errors and simple enhancements. The PDF manual contains the original code.

Example 1A had a hardcoded Nodename of JANE, you can see below.
3kRanger added a prompt for Nodename, where just a return will lookup the and use the local Nodename.

Example 1B will lookup and display the local Nodename; incase you are running these examples on 2 machines, and don't recall their Nodenames.

Program 1A

Download

 {NetIPC Example-1-Program 1A}
 { MPEV  pascalprep  netipc1a,netipc1a.pub;info="set 'native=false'"}
 { MPEiX pasxllk netipc1a,netipc1x.pub,,,"set 'native=true'" }
 $list off$
 {******************************************************************}
 { Example 1 (programs 1A and 1B) present an example of how to set  }
 { up and use a connection (virtual circuit) for TCP access. The    }
 { two programs, running on different nodes, open communication via }
 { call sockets. They then establish a connection (using VC sockets)}
 { and send and receive data over this connection. Finally, they    }
 { terminate their connection.                                      }
 {                                                                  }
 { To have netipc1A connect, run netipc1B first. netipc1A prompts   }
 { for a Node name. A return looks up the local Node, such that you }
 { can run both program on the same machine.                        }
 {******************************************************************}
 $if 'native'$
    $standard_level 'HP_PASCAL'$                                {3k}
 $else$
    $standard_level 'HP3000'$                                   {3k}
    $uslinit$                                                   {3k}
 $endif$

 program connection_example1 ( input, output );

 const
       maxdata = 2000;
       maxmsg = maxdata + 2;
       maxname = 20;
       maxloc = 52;                                             {3k}

 type
 $if 'not native'$                                              {3k}
       shortint = -32768..32767;                                {3k}
 $endif$                                                        {3k}
       datatype = record
          len   : shortint;
          msg   : packed array [1..maxdata] of char;
       end;
       timeval_type  = record case boolean of
          true  : (int   : shortint);
          false : (chars : packed array [1..30] of char);
       end;

       nametype = packed array [1..maxname] of char;
       loctype  = packed array [1..maxloc] of char;

 var
       calldesc    :     integer;   {2-word integer}
       vcdesc      :     integer;
       protocol    :     integer;
       socket_kind :     integer;

       dest        :     integer;
       result      :     integer;
       data        :     datatype;
       name        :     nametype;
       location    :     loctype;
       y_len       :     integer;
       y_data      :     char;
       num_msgs    :     integer;
       strdata     :     string [maxdata];
       i           :     integer;
       timeval     :     timeval_type;

 procedure terminate;       intrinsic;

 {===== NetIPC intrinsic declarations =====}
 procedure   ipccreate;     intrinsic;
 procedure   ipclookup;     intrinsic;
 procedure   ipcconnect;    intrinsic;
 procedure   ipcrecv;       intrinsic;
 procedure   ipcsend;       intrinsic;
 procedure   ipcshutdown;   intrinsic;
 procedure   ipcerrmsg;     intrinsic;
 procedure   ipccontrol;    intrinsic;

 {===== error handling procedure =====}

 procedure leave ( result : integer; locat : integer );         {3k}
    var
       msg: string [80];
       i, len, newresult: integer;
 begin
    prompt ( '** ErrLoc ', locat:1, ': ' );                     {3k}
    ipcerrmsg ( result, msg, len, newresult );
    if newresult = 0 then begin
       setstrlen (msg, len);
       writeln (msg);
    end else
       writeln ( '** IpcErrMsg result is ', newresult:1 );
    terminate;
 end;

 {===== main of NetIPC Program 1A =====}

 begin

 { look up the call socket RALPH located on node JANE }
    name:= 'RALPH';
    location:= 'JANE';

 {===== prompt for or lookup current node ====================== 3k}
    location := ' ';                                            {3k}
    prompt ( '--Enter Node name or return for local host? ' );  {3k}
                                                                {3k}
    readln ( strdata );                                         {3k}
    i := strlen ( strdata );                                    {3k}
    strmove ( i, strdata, 1, location, 1 );                     {3k}
                                                                {3k}
    if location [1] <> ' ' then begin                           {3k}
       writeln ( '-- Node: ', location:i );                     {3k}
    end else begin                                              {3k}
 {=== get local/current nodename ===}                           {3k}
       ipccreate ( 3, 0, , , calldesc, result );                {3k}
       if result <> 0 then leave ( result, 1 );                 {3k}
       i := maxloc;                                             {3k}
       ipccontrol ( calldesc, 14,,, location, i,, result );     {3k}
       if result <> 0 then leave ( result, 2 );                 {3k}
       ipcshutdown ( calldesc );                                {3k}
       location [i+1] := ' ';                                   {3k}
       writeln ( '--Node: ', location:i, ' --' );               {3k}
    end;                                                        {3k}
                                                                {3k}
    writeln ( '--Looking up socket ',name:5,' on Node ',        {3k}
       location:i,' --');                                       {3k}
 {===== Get Nodename =========================================== 3k}

    ipclookup ( name, 5, location, 3, , dest, protocol, socket_kind,
                result );
    if result <> 0 then leave ( result, 3 );                    {3k}

 { create a call socket;
   then initiate and complete connection to destination socket}

    ipccreate ( socket_kind, protocol, , , calldesc, result );
    if result <> 0 then leave ( result, 4 );                    {3k}
    timeval.int := 100;  {tenth seconds}                        {3k}
    ipccontrol ( calldesc, 3, timeval.chars, 2, , , result );   {3k}
    if result <> 0 then leave ( result, 5 );                    {3k}

 {initiate connection}
    writeln ( '--Connecting-- ', timeval.int:1, ' tenths' );    {3k}
    ipcconnect ( calldesc, dest, , , vcdesc, result );
    if result <> 0 then leave ( result, 6 );                    {3k}

    timeval.int := 0;
    timeval.int := 100;  {tenth seconds}                        {3k}
    ipccontrol ( vcdesc, 3, timeval.chars, 2, , , result );
    if result <> 0 then leave ( result, 7 );                    {3k}

    ipcshutdown ( calldesc );
    ipcshutdown ( dest );

    ipcrecv ( vcdesc, , , , , result );  {complete connection}
    if result <> 0 then leave ( result, 8 );                    {3k}
    writeln ( '--Connected--' );                                {3k}

 { prompt for messages and send them }
    writeln ('Enter "//" to terminate the program.');
    setstrlen (strdata, 0);
    while strdata <> '//' do begin
       prompt ('Message? ');
       readln (strdata);             {read message}
       data.len := strlen (strdata); {store message length}
       strmove ( data.len, strdata, 1, data.msg, 1 ); {store message}

 {send message with length as first 2 bytes}
       ipcsend ( vcdesc, data, data.len +2, , ,result );
       if result <>  0 then leave ( result, 9 );                {3k}
    end;

 {connection shutdown procedure}

    data.len := 4;
    data.msg := 'END?';                      { termination request}
    ipcsend ( vcdesc, data, 6, , , result );
    writeln ('END sent');
    if result <> 0 then leave ( result, 10 );                   {3k}

 {receive 'Y' confirmation}
    y_len := 1;
    ipcrecv ( vcdesc, y_data, y_len, , , result );
    if (y_data = 'Y') then writeln ('Y received' );
    if (y_data = 'Y') and (result = 0) then
       ipcshutdown ( vcdesc )
    else begin
       writeln ( 'Warning: shutdown not confirmed or result  0' );
       leave ( result, 11 );                                    {3k}
    end;
 end.
  

Program 1B

Download

 {NetIPC Example-1-Program 1B}
 { MPEV  pascalprep  netipc1b,netipc1b.pub;info="set 'native=false'"}
 { MPEiX pasxllk netipc1b,netipc1y.pub,,,"set 'native=true'" }
 $list off$
 {******************************************************************}
 { Example 1 (programs 1A and 1B) present an example of how to set  }
 { up and use a connection (virtual circuit) for TCP access. The    }
 { two programs, running on different nodes, open communication via }
 { call sockets. They then establish a connection (using VC sockets)}
 { and send and receive data over this connection. Finally, they    }
 { terminate their connection.                                      }
 {                                                                  }
 { To have netipc1A connect, run netipc1B first. netipc1A prompts   }
 { for a Node name. A return looks up the local Node, such that you }
 { can run both program on the same machine.                        }
 {******************************************************************}
 $if 'native'$
    $standard_level 'HP_PASCAL'$                                {3k}
 $else$
    $standard_level 'HP3000'$                                   {3k}
    $uslinit$                                                   {3k}
 $endif$

 program connection_example2 ( output );

 const
       maxdata = 2000;
       maxname = 20;
       maxloc = 52;                                             {3k}

 type
 $if 'not native'$                                              {3k}
       shortint = -32768..32767;                                {3k}
 $endif$                                                        {3k}
       datatype = packed array [1..maxdata] of char;
       timeval_type = record case boolean of
          true  :  (int   : shortint);
          false :  (chars : packed array [1..30] of char);
       end;
       nametype = packed array [1..maxname] of char;
       loctype  = packed array [1..maxloc] of char;

 var
       calldesc    :  integer;          {2-word integer}
       vcdesc      :  integer;
       dlen        :  integer;
       result      :  integer;
       data        :  datatype;
       name        :  nametype;
       len         :  shortint;
       datastr     :  string [maxdata];
       timeval     :  timeval_type;
       port        :  timeval_type;                             {3k}
       ulen        :  integer;                                  {3k}
       location    :  loctype;                                  {3k}

 procedure   terminate;     intrinsic;

 {===== NetIPC intrinsic declarations =====}
 procedure   ipccreate;     intrinsic;
 procedure   ipcname;       intrinsic;
 procedure   ipcrecvcn;     intrinsic;
 procedure   ipcrecv;       intrinsic;
 procedure   ipcsend;       intrinsic;
 procedure   ipcshutdown;   intrinsic;
 procedure   ipcerrmsg;     intrinsic;
 procedure   ipccontrol;    intrinsic;

 {===== error handling procedure =====}

 procedure leave ( result : integer; locat : integer );         {3k}
    var
       msg: string [80];
       i, len, newresult: integer;
 begin
    prompt ( '** ErrLoc ', locat:1, ': ' );                     {3k}
    ipcerrmsg ( result, msg, len, newresult );
    if newresult = 0 then begin
       setstrlen (msg, len);
       writeln (msg);
    end else
       writeln ( '** IpcErrMsg result is ', newresult:1 );
    terminate;
 end;

 { procedure receive }
 { The following procedure receives one message which was sent via
   an ipcsend call. It assumes that the length (number of bytes) of
   the message was sent as the first two bytes of data and that the
   length value does not include those two bytes. }

 procedure receive ( connection : integer;
                     var   rbfr : datatype;
                     var   rlen : shortint;
                     var errorcode : integer    );

 const
       head_len = 2;

 type
       length_buffer_type = packed array [1..2] of char;
       header_len_type = record case integer of
          0: ( word: shortint );
          1: ( byte: length_buffer_type);
       end;
 var
       i, j        : integer;
       dlen        : integer;
       header_len  : header_len_type;
       tempbfr     : datatype;

 begin      { procedure receive }
    i:=0;
    errorcode := 0;
    while (i <> head_len) and (errorcode = 0) do begin
                                { get length of message }
       dlen := head_len - i;
       ipcrecv ( connection, tempbfr, dlen, , , errorcode );

       if errorcode = 0 then begin                              {3k}
          strmove ( dlen, tempbfr, 1, header_len.byte, i+1 );
          i := i + dlen;
       end;                                                     {3k}
    end;

    rlen := 0;                                                  {3k}
    if errorcode = 0 then begin
       rlen := header_len.word;
       i := 0;
 { get the message }
       while (i <> rlen) and (errorcode = 0) do begin
          dlen := header_len.word - i;
          ipcrecv ( connection, tempbfr, dlen, , ,errorcode );

          if errorcode = 0 then begin                           {3k}
             strmove ( dlen, tempbfr, 1, rbfr, i+1 );
             i := i + dlen;
          end;                                                  {3k}
       end;
    end;
 end;

 {===== main of NetIPC Program 2 or 1B =====}

 begin

 {create a call socket and name it}
    ipccreate ( 3, 4, , , calldesc, result );
    if result <> 0 then  leave ( result ,1 );                   {3k}

    name := 'RALPH';
    ipcname ( calldesc, name, 5, result );
    if result <> 0 then  leave ( result, 2 );                   {3k}

 {===== Get my Nodename ======================================== 3k}
    location := ' ';                                            {3k}
    dlen := maxloc;                                             {3k}
    ipccontrol ( calldesc, 14,,, location, dlen,, result );     {3k}
    if result <> 0 then leave ( result, 3 );                    {3k}
    location [dlen+1] := ' ';                                   {3k}
    writeln ( '--Node: ', location:dlen, ' --' );               {3k}
 {===== Get Nodename =========================================== 3k}
 {===== Get Port number ======================================== 3k}
    ipccontrol ( calldesc, 514,,, port.chars, dlen,, result );  {3k}
    if result <> 0 then leave ( result, 4 );                    {3k}
    ulen := port.int;                                           {3k}
    if ( ulen < 0 ) then ulen := 65536 + ulen;                  {3k}
    writeln ( '--Port: ', ulen:1, ' name=', name:5, ' --' );    {3k}
 {===== Get Port =============================================== 3k}

 {wait for a connection request}
    timeval.int:=0;
    timeval.int:=300; { tenths }                                {3k}
    ipccontrol ( calldesc, 3, timeval.chars, 2, , , result );
    if result <> 0 then  leave ( result, 5 );                   {3k}

    writeln ( '--Waiting for connect ',timeval.int:1,' tenths');{3k}
    ipcrecvcn ( calldesc, vcdesc, , , result );
    if result <> 0 then  leave ( result, 6 );                   {3k}
    ipcshutdown ( calldesc );
    writeln ( '-- Connected --' );                              {3k}

 {wait for a message on the connection and print message received}
 {set timeout}
    timeval.int:=0;
    timeval.int:=100; { tenths }                                {3k}
    ipccontrol ( vcdesc, 3, timeval.chars, 2, , , result );
    if result <> 0 then  leave ( result, 7 );                   {3k}
    writeln ( '-- Wait ', timeval.int:1, ' tenths' );           {3k}

    repeat begin
       receive ( vcdesc, data, len, result );
       if result <> 0 then leave ( result, 8 );                 {3k}

       setstrlen (datastr, len);
       strmove ( len, data, 1, datastr, 1 );
       if datastr <> 'END?' then
          writeln (datastr);    {print data received}
    end until datastr = 'END?';

 {connection shutdown procedure}

    if datastr = 'END?' then writeln ('END received');
    data := 'Y';
    ipcsend ( vcdesc, data, 1, , , result );   {confirmation message}
    writeln ('Y sent');
    if result <> 0 then leave ( result, 9 );                    {3k}

    receive ( vcdesc, data, len, result );
    if result = 64 then
       ipcshutdown ( vcdesc )
    else
       leave ( result, 10 );                                    {3k}
 end.
  

Program 1C

3kRanger provides a modified Program 1B with IOWAIT and Control-Y interrupt allowing you to interrupt IPCRECVCN waiting for a connection. Download

If you run Example 1B, and cannot make a connection from Example 1A, your program is blocked until you [Break] and Abort to get out of the program.

Program 1C uses IOWAIT so as not to block in IPCRECVCN. IOWAIT can be interrupted with a user Control-Y trap and the I/O aborted.

3kRanger Runtime Example

Two sessions on the same machine. 1B should be run before 1A.

 Fox 25:netipc1b.pub
 --Node: FOX.TKRANGER.COM --
 --Port: 59204 name=RALPH --
 --Waiting for connect 300 tenths
 -- Connected --
 -- Wait 100 tenths
 test message
 //
 END received
 Y sent
 Fox 25:
  

 Fox 9:netipc1a.pub
 --Enter Node name or return for local host? [Return]
 --Node: FOX.TKRANGER.COM --
 --Looking up socket RALPH on Node FOX.TKRANGER.COM --
 --Connecting-- 100 tenths
 --Connected--
 Enter "//" to terminate the program.
 Message? test message
 Message? //
 END sent
 Y received
 Fox 9:
  

 Fox 9:netipc1c.pub
 --Node: FOX.TKRANGER.COM --
 --Port: 59238 name=RALPH --
 !! Control-Y set
 --Waiting for connect 300 tenths
 !! Control-Y (1) !!           [Control-Y]
 Fox 9:
  


Chapter 4 NetIPC Examples

Example 2