Pup Package




The  Pup  package  consists  of a  large  body  of  Alto  software that
implements communication by means of Pups (Parc Universal  Packets) and
Pup-based  protocols.   This  software  is  broken  into  a  number  of
independent  modules implementing  various  "levels" of  protocol  in a
hierarchical  fashion.  Each  level  depends on  primitives  defined at
lower   levels,  and   defines  new   primitives   (e.g,  inter-network
addressing, process-to-process connections, byte streams)  available to
levels above it.  A program making use of the Pup package  need include
only those components implementing primitives utilized by that program.



1. Overview


This  document  is  organized   as  a  general  overview   followed  by
descriptions of each of the components of the package, with  the lowest
levels described first.  A history  of revisions to the package  may be
found at the end.

Before beginning the  real documentation, we  should like to  mention a
number of points worth bearing  in mind throughout, as well  as various
caveats and suggestions for use.

a.  This document concerns itself only with external program interfaces
and   not  with   protocol  specifications,   internal  implementation,
motivations for  design choices, etc.   The Pup package  implements the
protocols  described  in  the  memo  "Pup  Specifications"  (Maxc  file
<Pup>PupSpec.Press)  and in  other documents  also to  be found  in the
<Pup> directory.  A higher-level  overview of the Pup protocols  may be
found  in  the   report  "Pup:  An  Internetwork   Architecture",  file
<Pup>PupPaper.press.   Users  interested  in  protocol  information are
referred  to  those documents.   Knowledge  of these  protocols  is not
required  when  writing   programs  making  use  of   the  higher-level
primitives provided by  the Pup package (specifically,  connections and
byte streams), but is  essential when dealing directly with  the lower-
level primitives.

b.  Since both  the software and the  protocols are still  under active
development,  users  are  requested  to  avoid  making  changes  to the
package, if at  all possible.  This is  so that subsequent  releases of
the package  may be  incorporated into  existing programs  with minimum
fuss.  We have attempted to provide as general-purpose a package  as is
reasonable (consistent with clean programming practices and considering
Alto memory limitations),  so if you come  up with an  application that
simply can't  be accomodated  without modifying  the package,  we would
like to know about it.  There are a small number of parameters  that we
have designated as "user-adjustable"  and separated out into  a special
declaration file (PupParams.decl).  The intention is that users be able
to  change these  parameters and  recompile the  package;  however, one
should be aware that the  software has not been tested  with parameters
set to values other than the ones in the released version.

                             ------------
                   Copyright Xerox Corporation 1981


Pup Package                November 21, 1981                          2




c.  One of  the design goals has  been to implement software  that will
also  run  on  a  Nova.   All  Alto-specific  code  has  been carefully
separated  out  into  modules containing  "Al"  in  their  names (e.g.,
PupAlEthb.bcpl for the Alto Ethernet driver).  The Nova  equivalents of
the Alto-specific modules (released as a separate package) contain "Nv"
in their  names.  Source  files not  containing "Al"  or "Nv"  in their
names may be recompiled on the  Nova (with BCPL or the Nova  version of
Asm) and run without change; either they are completely free of machine
dependencies or (in a few cases) they enclose machine-dependent code in
conditional  compilation.   People  writing  general-purpose subsystems
making use of this package are encouraged to adopt the same approach.

d.  The Pup package makes extensive use of primitives provided  in four
other  software packages:  Context, Interrupt,  Queue, and  Timer.  The
dependence  on the  Context package  means that  calling  programs must
operate in a manner  compatible with contexts.  In particular,  the Pup
package  initiates a  number of  independent background  processes that
must  be given  an opportunity  to run  fairly frequently.   Hence, the
user's "main  program" must run  within a context,  and wait  loops and
very long computations in the main program should be  interspersed with
calls to Block.  For example, a call such as "Gets(keys)" (which causes
busy-waiting  inside  the  operating  system)  might  be   replaced  by
something like "GetKeys()", where the latter function is defined as:

        let GetKeys() = valof
           [
           while Endofs(keys) do Block()
           resultis Gets(keys)
           ]

Alternatively, you may  change the Operating System's  "Idle" procedure
to call Block, if you  understand what you are doing.  Consult  the the
Context Package writeup for further information.


1.1. Organization

The Pup software is  divided into three major levels,  corresponding to
levels 0 through 2 of the Pup protocol hierarchy.  Software at  a given
level depends on primitives provided at all levels below it.

At level 0 is the "transport mechanism" software necessary for  an Alto
to send  and receive  Pups on an  Ethernet.  This  consists of  a small
Ethernet interrupt handler that appends received Pups to an input queue
and transmits Pups taken from an output queue.  It is the  only portion
of the  Pup package specific  to the Ethernet  or to  the Alto-Ethernet
interface.   Corresponding  drivers  are  included  for  the  XEOS  EIA
interface and the  ASD Communication Processor,  for use in  Altos that
have those special interfaces installed.

Level 1 defines a number of important and generally  useful primitives.
A program desiring to send and receive "raw Pups"  (without sequencing,
retransmissions, flow control, etc.) would interface to the Pup package
at this level.  The level 1 module includes the following:

    a.   Procedures  for   creating,  maintaining,  and   destroying  a
    "socket", a process's logical connection to the Pup inter-network.

    b.  Procedures for managing  "Packet Buffer Items" (PBIs),  each of


Pup Package                November 21, 1981                          3




    which holds  a Pup  and some associated  information while  the Pup
    resides in Alto memory.

    c.   A background  process that  distributes received  Pups  to the
    correct sockets.   This includes checking  port address  fields and
    optionally verifying the Pup checksum.

    d.   Procedures for  allocating PBIs,  building Pups,  and queueing
    them for transmission.

    e.  A background process that dynamically maintains a routing table
    for transmission of Pups to arbitrary inter-network addresses.

    f.  Optional procedures permitting  the local host to be  a gateway
    (not ordinarly used).

At level 2 are  modules implementing three higher-level  protocols: the
Rendezvous/Termination Protocol (RTP), the Byte Stream  Protocol (BSP),
and  the Network  Directory  Lookup Protocol.   These  are independent,
parallel  protocols, each  built on  top of  the primitives  defined at
level 1; however, the RTP and  the BSP interact in a way such  that, in
this implementation, BSP depends on the existence of RTP.

The  RTP  module  contains   procedures  for  opening  and   closing  a
"connection" with a foreign process.  These have options permitting the
local  process  to  operate  in  the  role  of  either  "initiator"  or
"listener".

The BSP module  contains mechanisms for  sending and receiving  data by
means of error-free, flow-controlled "byte streams" between a local and
a foreign process.   These are true "streams"  in the sense  defined by
the  Alto  operating  system.   Additionally,  means  are  provided for
sending  and  receiving Marks  and  Interrupts, which  are  special in-
sequence  and  out-of-sequence  signals  defined  by  the  Byte  Stream
Protocol.  A  separate, optional module  permits sending  and receiving
blocks of data in memory an order of magnitude more efficiently than by
use of the basic "Puts" and "Gets" operations.

The Network Directory Lookup module contains procedures  for converting
between  inter-network "names"  (e.g.,  host names)  and  Pup addresses
(ports).   When necessary,  it finds  and interacts  with  some network
directory lookup  server on the  directly connected network.   The Name
Lookup  module  converts  only  from names  to  addresses,  and  may be
substituted for Network Directory Lookup when its other  facilities are
not required.


1.2. File Conventions

The Pup  package is distributed  as file PupPackage.dm,  which contains
the following binary files:


Pup Package                November 21, 1981                          4




Level 0
    PupAlEthb.br             Alto Ethernet driver (BCPL portion)
    PupAlEtha.br             Assembly code for Ethernet driver
    PupAlEthInit.br          Alto Ethernet initialization
    PupAlEIAb.br             Driver for EIA interface
    PupAlEIAa.br
    PupAlEIAInit.br
    PupAlComProcb.br         Driver for Communication Processor
    PupAlComProca.br
    PupAlComProcInit.br

Level 1
    Pup1b.br                 Main level 1 code (BCPL portion)
    PupAl1a.br               Assembly-language code for level 1
    Pup1OpenClose.br         Opening and closing Pup sockets
    PupRoute.br              Routing table maintenance and access
    PupDummyGate.br          Dummy substitute for gateway code
    Pup1Init.br              Level 1 initialization

Level 2
    PupRTP.br                Rendezvous/Termination Protocol
    PupRTPOpenClose.br       Opening and closing RTP sockets
    PupBSPStreams.br         Byte Stream Protocol (BCPL portion)
    PupBSPProt.br            Additional BSP code
    PupBSPOpenClose.br       Opening and closing BSP sockets
    PupBSPa.br               Assembly-language code for BSP
    PupBSPBlock.br           Fast BSP block transfer procedures
    PupNetDirLookup.br       Network directory lookup module
    PupNameLookup.br         Name lookup module (subset of PupNetDirLoo

The  files with  "Init"  in their  names, as  well  as PupDummyGate.br,
contain initialization  code that  need be executed  only once  and may
then be  thrown away.   (Note, however, that  the level  1 and  level 0
"Destroy" procedures are included in the "Init" modules.)

Files PupNetDirLookup, PupNameLookup.br, and the files with "OpenClose"
in their names contain  code that is infrequently executed  (i.e., only
when  particular  types  of  sockets  are  opened  or  closed)  and may
therefore  be  loaded  into  overlays (to  be  managed  by  the Overlay
package) without significant performance penalties.  All  other modules
must remain resident while any part of the Pup package is active, since
they contain main-line  code or code  that is executed  by continually-
running contexts.

Additionally,  the following  "get" files  are included.   They contain
declarations of  all structures  and other parameters  likely to  be of
interest to calling programs (as well as some others of no  interest to
callers).  We  suggest that the  user make listings  of these  files to
accompany this documentation.


Pup Package                November 21, 1981                          5




    Pup0.decl                Level 0 definitions (network-independent)
    Pup1.decl                Level 1 definitions
    PupRTP.decl              Definitions for RTP
    PupBSP.decl              Definitions for BSP

    Pup.decl                 Does "get" of all the above
    PupParams.decl           User-adjustable parameters
    PupNetDir.decl           Definitions for PupNetDirLookup
    PupStats.decl            Statistics definitions
    PupAlEth.decl            Definitions specific to Alto Ethernet
    PupAlEIA.decl            Definitions specific to EIA driver
    PupAlComProc.decl        Definitions specific to ComProc driver

A program that  does a "get"  of any of the  first group of  files must
also "get" all files earlier on  the list, and in the same  order.  (We
were not able to make  this happen automatically because of a  limit on
the number of simultaneous  open files at compilation time).   The file
Pup.decl is provided for  the convenience of programs dealing  with the
package at  the BSP level.   A "get" of  PupParams.decl is  included in
Pup0.decl, and  PupAlEth.decl and PupStats.decl  are not  ordinarily of
interest to outside programs.

The  following  table  shows,  for  each  module   (including  external
packages), what .br files constitute that module and what other modules
are also required.

Module Name             Files               Other Modules Required

BSP Block Transfer      PupBSPBlock.br      BSP
                                            ByteBlt

ByteBlt (external)      AltoByteBlt.br

BSP                     PupBSPStreams.br    RTP
                        PupBSPProt.br
                        PupBSPOpenClose.br
                        PupBSPa.br

RTP                     PupRTP.br           Level 1
                        PupRTPOpenClose.br

Net Dir Lookup          PupNetDirLookup.br  Level 1
Name Lookup             PupNameLookup.br    Level 1

Level 1                 Pup1b.br            Level 0
                        Pup1OpenClose.br    Timer
                        PupAl1a.br
                        PupRoute.br
                        PupDummyGate.br
                        Pup1Init.br

Level 0                 PupAlEthb.br        Context
                        PupAlEtha.br        Interrupt
                        PupAlEthInit.br     Queue

Context (external)      Context.br
                        ContextInit.br

Interrupt (external)    Interrupt.br
                        InterruptInit.br


Pup Package                November 21, 1981                          6




Queue (external)        AltoQueue.br

Timer (external)        AltoTimer.br

There are a few global parameters that may be changed by  modifying the
PupParams.decl file and then  recompiling the entire Pup  package.  The
most interesting  parameter is "pupDebug",  which, if true  (default is
false) causes some additional consistency checking code to be compiled.

The sources for  the Pup package  are contained in  file PupSources.dm,
and consist of the following files:

    PupAlEthb.bcpl      PupAlEtha.asm       PupAlEthInit.bcpl
    PupAlEIAb.bcpl      PupAlEIAa.asm       PupAlEIAInit.bcpl
    PupAlComProcb.bcpl  PupAlComProca.asm   PupAlComProcInit.bcpl
    Pup1b.bcpl          PupAl1a.asm         Pup1OpenClose.bcpl
    PupRoute.decl       PupRoute.bcpl
    Pup1Init.bcpl       PupDummyGate.bcpl
    PupRTPInternal.decl PupRTP.bcpl         PupRTPOpenClose.bcpl
    PupBSPStreams.bcpl  PupBSPProt.bcpl     PupBSPa.asm
    PupBSPOpenClose.bcplPupBSPBlock.bcpl
    PupNameLookup.bcpl

Additionally, there are several command files:

    CompilePup.cm            Compiles all the source files
    DumpPupPackage.cm        Creates PupPackage.dm
    DumpPupSources.cm        Creates PupSources.dm
    Pup.cm                   A list of all the source files

The source files are formatted for printing in a small fixed-pitch font
such as Gacha8 (the normal default for Empress).


1.3. Glossary of Data Types

Name     Defined in     Meaning

BSPSoc   PupBSP.decl    BSP-level  Pup  socket,  consisting  of  an RTP
                        socket   (RTPSoc)   followed    by   additional
                        information about a byte stream.  This includes
                        byte  IDs   (sequence  numbers),   queues,  and
                        allocations for incoming and outgoing  data and
                        interrupts, and a BSP stream block (BSPStr).

BSPStr   PupBSP.decl    BSP stream (part of a BSPSoc),  for interfacing
                        the  BSPSoc  to  the  Alto  operating  system's
                        stream mechanism.

HTP      Pup1.decl      Hash  Table  Preamble,  defining  the publicly-
                        accessible  operations on  a  dictionary object
                        (specifically, the  Pup routing  table).  These
                        operations  are  Lookup,  Insert,  Delete,  and
                        Enumerate.  This object is misnamed in  that it
                        need not actually be implemented by means  of a
                        hash table; at  present, the Pup  routing table
                        is not.

NDB      Pup0.decl      Network  Data  Block,   containing  information


Pup Package                November 21, 1981                          7




                        specific to each network physically attached to
                        the local host.  (A standard Alto has  only one
                        of these, for the directly-connected Ethernet.)

PBI      Pup0.decl      Packet  Buffer  Item,  which  holds  a  Pup and
                        various associated information.

PF       Pup0.decl      Packet   Filter,   controlling   acceptance  of
                        incoming packets on a given network.

Port     Pup0.decl      An   inter-network   address,   consisting   of
                        network, host,  and socket numbers,  as defined
                        by protocol.

PSIB     Pup1.decl      Pup Socket Info  Block, contains data  used for
                        setting initial default values when a PupSoc is
                        created.

Pup      Pup0.decl      An   inter-network   packet,   as   defined  by
                        protocol.

PupSoc   Pup1.decl      Level  1  Pup  socket,  defining   a  process's
                        logical  connection to  the  inter-network.  It
                        contains   default  local   and   foreign  port
                        addresses, PBI  allocation information,  and an
                        input queue header.

RT       --             Routing Table, containing information necessary
                        to route outgoing Pups to destination  hosts or
                        to gateways.  There is only one instance  of an
                        RT, called  pupRT.  The structure  of an  RT is
                        not public, but object procedures (see HTP) are
                        provided   for   accessing    and   enumerating
                        individual Routing Table Entries  (RTEs), which
                        are public structures.

RTE      Pup1.decl      Routing  Table Entry  (routing  information for
                        one network).

RTPSoc   PupRTP.decl    RTP-level Pup socket,  consisting of a  level 1
                        socket   (PupSoc)   followed    by   additional
                        information about a connection.   This includes
                        state,  connection  ID, timers,  and  a higher-
                        level Pup-handling procedure.

soc      --             An  instance of  a PupSoc,  RTPSoc,  or BSPSoc,
                        depending on context.   Note that a  PupSoc may
                        be the initial portion of an RTPSoc,  which may
                        in  turn be  the initial  portion of  a BSPSoc;
                        hence, a given soc  may be an instance  of more
                        than one of these structures.

str      --             An  instance  of  a  stream  (most   likely,  a
                        BSPStr).


Pup Package                November 21, 1981                          8




2. Level 0 Interface


Only the level 0 driver for the Ethernet is described here.  There also
exist drivers for the EIA and ComProc interfaces, but they are somewhat
specialized and are not  documented here.  Their function  is analogous
to the Ethernet driver and their operation is quite similar.

The  level  0  module (files  PupAlEthb,  PupAlEtha,  and PupAlEthInit)
serves only to interface  the Alto Ethernet to  the network-independent
Pup level 1  module.  Assuming the  level 1 code  is being used,  as is
normally the case, external programs will generally have no occasion to
deal directly with  the level 0 module.   Provisions are also  made for
sending  and receiving  non-Pup Ethernet  packets, for  use  in unusual
applications.

This module  requires the existence  of the following  external statics
(all of which are defined in level 1):

    ndbQ      A pointer to a two-word queue header  (hereafter referred
              to as  "a queue"; see  Queue Package  documentation) upon
              which the Ethernet NDB  (etherNDB) may be queued  by this
              module.   In  a  machine  with  more  than   one  network
              interface, this queue contains an NDB for each network.

    pbiFreeQ  A  queue  from  which  free  PBIs  may  be  obtained, for
              buffering received Pups.

    pbiIQ     A  queue  to  which  PBIs  are  appended  when  Pups  are
              received.

    lenPup    The maximum length of a Pup (in words).

The externally-callable procedures in this module are the following:

InitAltoEther(zone, ctxQ, device)
    Initializes  the  Alto  Ethernet  interface  and   associated  data
    structures.  "zone" is a free-storage zone from which space  may be
    obtained  for permanent  data structures  (currently less  than 100
    words).   "ctxQ" is  a queue  on which  a context  created  by this
    procedure  may  be queued.   This  procedure allocates  an  NDB and
    appends it to ndbQ;  allocates an interrupt context  (see Interrupt
    Package documentation) and sets it up to field Ethernet interrupts;
    and  allocates  and  initiates  an  ordinary  context  (see Context
    Package documentation) which  runs forever and  whose job it  is to
    restart  the Ethernet  interface  if it  is  ever shut  off  due to
    running out of free  PBIs for input.  InitAltoEther  returns having
    done  nothing  if  the  Alto  doesn't  have  an  Ethernet interface
    installed (the level 1 initialization detects the condition of ndbQ
    being empty after all interface initialization procedures have been
    called).

    "device"  should  normally be  0,  referring to  the  standard Alto
    Ethernet interface.  However, in Altos with more than  one Ethernet
    interface installed, the driver may be initialized  multiple times,
    once for each interface; in this case, device numbers 1 and 2 refer
    to the first and second additional interfaces.

EncapsulateEtherPup(pbi, pdh)


Pup Package                November 21, 1981                          9




    Encapsulates  the  Pup  contained  in  "pbi"  for  transmission  to
    physical destination host "pdh" on the directly-connected Ethernet.
    The   PBI   should   contain   a   completely    well-formed   Pup.
    EncapsulateEtherPup sets the Ethernet destination, source, and type
    fields in the  encapsulation portion of  the packet, and  also sets
    the packetLength  word in the  PBI.  SendEtherPup is  the procedure
    called from level  1 via the  encapsulatePup entry in  the Ethernet
    NDB.

SendEtherPacket(pbi)
    Queues "pbi" for  transmission on the  directly-connected Ethernet,
    and  initiates  transmission if  the  interface is  idle.   The PBI
    should contain a completely well-formed Ethernet packet (which need
    not be a  Pup), the packetLength word  in the PBI must  contain the
    physical length of the packet in words, pbi>>PBI.queue must contain
    a pointer to a queue to which the PBI will be appended after it has
    been transmitted,  and pbi>>PBI.ndb must  contain a pointer  to the
    NDB associated with the Ethernet interface through which the packet
    is to be sent.  SendEtherPacket is the procedure called  from level
    1 via the level0Transmit entry in the Ethernet NDB.

SendEtherStats(pbi, ndb) = true or false
    If the debugging version of PupAlEthb is loaded (pupDebug on), this
    procedure  copies  the  statistics  accumulated  by   the  Ethernet
    interface (described  by ndb)  into pbi and  returns true.   If the
    module  was   not  compiled   with  debugging   on,  SendEtherStats
    immediately returns false.

DestroyAltoEther(ndb)
    Turns off  the Ethernet interface  designated by ndb,  and releases
    all  storage allocated  by  InitAltoEther.  This  is  the procedure
    called from level 1 via the NDB.destroy procedure.   This procedure
    is in the PupAlEthInit module, which must therefore be  retained if
    the "destroy" operation is actually to be utilized.

When  a  packet is  received  from the  Ethernet,  the  input interrupt
routine  first  verifies that  the  hardware and  microcode  status are
correct, and discards the  packet without error indication if  not.  It
then tests the packet for acceptance by each Packet Filter (PF)  on the
Ethernet packet filter queue, as will be described shortly.  If some PF
accepts the packet, the PBI is then enqueued on the queue designated in
the PF; otherwise  it is discarded.  A  free PBI is then  obtained from
pbiFreeQ, and the receiver is restarted. (Actually, an attempt  is made
to restart the receiver before  any other processing so as  to minimize
the interval during which a packet could be missed because the receiver
isn't listening to the Ethernet.)

When an  output PBI  is passed to  SendEtherPacket, it  is queued  on a
local Ethernet output queue (eOQ,  part of the NDB).  If  the interface
is currently  idle, transmission  is initiated  immediately; otherwise,
the  PBI  is simply  left  on the  queue  for action  by  the interrupt
routine.  When an output completion interrupt occurs (or a  fatal error
indication such  as a  "load overflow", or  a 100  millisecond software
timeout), the PBI  is then enqueued on  the queue specified in  the PBI
(typically pbiFreeQ or a level 1 queue called pbiTQ).

Garden-variety errors (e.g.,  collisions, bad Ethernet CRCs,  etc.) are
handled automatically: input errors cause the received packet simply to
be discarded, while  output errors cause  retransmission.  "Impossible"


Pup Package                November 21, 1981                         10




errors (suggesting that the interface or the Alto is broken)  result in
a call to SysErr(@ePLoc, ecBadEtherStatus).

In the  debugging version  of this  module (pupDebug  on), a  number of
Ethernet performance statistics  are gathered.  These are  intended for
experimental   purposes   and   measurements.    One   should   consult
PupStats.decl to see what is collected.

Though the primary  purpose of the  Pup level 0  module is to  send and
receive Pups on a particular directly-connected network, means are also
provided for sending and receiving arbitrary  network-dependent packets
(i.e., Ethernet packets in an Alto).

Sending  a  non-Pup   packet  is  straightforward:  one   simply  calls
SendEtherPacket after constructing  the desired Ethernet packet  in the
PBI, as described above.

Discrimination among  received packets is  accomplished by one  or more
objects called Packet  Filters (PFs), which  reside on a  Packet Filter
Queue (pfQ) whose head is in the NDB.  Each PF contains a predicate and
a pointer to a queue.  When a packet is received, the predicate in each
PF in turn  is called with  the PBI as  an argument.  If  the predicate
returns true, the PBI is enqueued on the queue pointed to by the PF; if
it returns false, the next PF  is tried.  If no PF accepts  the packet,
the PBI is discarded.

The pfQ initially  contains a single PF  that accepts Pups  and appends
them to  pbiIQ (the level  1 Pup input  queue).  A program  desiring to
receive other kinds of Ethernet packets should construct its own PF and
enqueue it on the Ethernet pfQ.



3. Level 1 Interface


The  level 1  module  (files Pup1b,  PupAl1a,  PupRoute, Pup1OpenClose,
PupDummyGate, and Pup1Init) contains the mechanisms enabling  a process
to send  and receive  individual Pups  to and  from other  processes at
arbitrary inter-network addresses.   Concepts such as  "connection" and
"stream",  however,  are  not  defined at  this  level,  so  it  is the
process's  responsibility  to perform  initial  connection, sequencing,
retransmission, duplicate detection, etc., where required.

A process deals  with the level  1 module through  a PupSoc, a  level 1
socket  structure  (see  Pup1.decl),  which  completely  describes that
process's  interface  to  the  inter-network  at  the  first  level  of
protocol.  The information in the socket is as follows:

    iQ             Input  queue.   PBIs  received  for  the  socket are
                   appended to this  queue.  The two-word  queue header
                   is included  in the socket  structure itself,  so to
                   remove  a  packet  from  the  iQ  one   would  write
                   "Dequeue(lv soc>>PupSoc.iQ)".

    lclPort        Local port address (a Port structure).   This serves
                   two  purposes.  First,  the "socket  number"  in the
                   port  enables  the  level  1  Pup  input  handler to
                   distribute each incoming  Pup to the  correct PupSoc


Pup Package                November 21, 1981                         11




                   by  comparing  pbi>>PBI.pup.dPort.socket   (the  Pup
                   destination        socket        number)        with
                   soc>>PupSoc.lclPort.socket  of  each  active  PupSoc
                   until  a match  is found.   Second, the  source port
                   fields of each outgoing Pup generated by the process
                   are defaulted (if zero)  to the values given  in the
                   local port address.

    frnPort        Foreign  port  address  (a  Port  structure).   This
                   provides information for defaulting  the destination
                   port fields of outgoing Pups, in the same  manner as
                   described for lclPort.

    psib           Pup  Socket Info  Block (PSIB),  which  contains the
                   information described below.  Since it  is generally
                   the same for all sockets, there is a  "default PSIB"
                   (dPSIB) whose contents are copied into the  psib for
                   each socket when the socket is created.

    maxTPBI        The  maximum  total  number  of  PBIs  that  may  be
                   assigned to the  socket.  Since free PBIs  are taken
                   from  a  common  pool, some  means  is  required for
                   ensuring that no single socket can usurp more than a
                   certain share  of the  total available  PBIs (which,
                   aside from  reducing performance for  other sockets,
                   could lead to deadlocks in higher-level protocols if
                   the free pool became exhausted).  This  is discussed
                   further  in  the  descriptions  of  the  GetPBI  and
                   ReleasePBI procedures.

    numTPBI        The  total number  of  additional PBIs  that  may be
                   assigned  to  the socket  (i.e.,  maxTPBI  minus the
                   number of PBIs already assigned).

    maxIPBI        The maximum number of  PBIs that may be  assigned to
                   the socket for input use.

    numIPBI        The number of  additional PBIs that may  be assigned
                   for input  (i.e., maxIPBI minus  the number  of PBIs
                   already assigned for input).

    maxOPBI        The maximum number of  PBIs that may be  assigned to
                   the socket for output use.

    numOPBI        The number of  additional PBIs that may  be assigned
                   for output (i.e.,  maxOPBI minus the number  of PBIs
                   already assigned for output).

    doChecksum     If true, the Pup software checksum is checked by the
                   level  1  software in  incoming  Pups  (before being
                   given  to  the process)  and  generated  in outgoing
                   Pups.  The default value is true.

The following statics are defined within the level 1 module and  may be
referenced externally (though only a few are likely to be of interest):

    dPSIB     Pointer  to default  socket info  block, used  to provide
              initial values in part of each PupSoc when it is created.


Pup Package                November 21, 1981                         12




    gatewayIQ Pointer to queue on which received Pups not  addressed to
              this  host are  placed.   Unless the  Gateway  package is
              loaded, gatewayIQ is initialized to pbiFreeQ.

    lenPup    The length of the largest possible Pup, in words (derived
              from maxPupDataBytes).

    lenPBI    The  length of  a PBI,  in words  (derived  from lenPup).
              Note that  all PBIs  are of  the same  size and  can each
              contain a Pup of maximum length.

    lPupSoc   The length of a PupSoc, in words.

    maxPupDataBytes  The maximum  number of data  (content) bytes  in a
              Pup.  This is initialized to the  "pupDataBytes" argument
              to InitPupLevel1 and remains constant thereafter.

    ndbQ      Pointer to queue of NDBs for all the physically connected
              networks  (see level  0 description).   The first  NDB on
              ndbQ is considered to be the "default" network, i.e., the
              one  sent to  if a  process specifies  a  Pup destination
              network of zero.

    numNets   The number of directly connected networks (always 1 in an
              Alto).

    pbiFreeQ  Pointer to queue of free PBIs.

    pbiIQ     Pointer to  queue on  which incoming  Pups are  placed by
              level 0 interrupt routines.

    pbiTQ     Pointer to  queue on which  outgoing Pups  are ordinarily
              placed after transmission.

    pupCtxQ   Default context queue onto which new contexts  created by
              the Pup package will be appended.  This is initialized to
              the "ctxQ" argument to InitPupLevel1.

    pupRT     Pointer to routing table (described later).

    pupZone   Default zone from which  allocations will be made  by the
              Pup package.  This is initialized to the  "zone" argument
              to InitPupLevel1.

    socketQ   Pointer to queue of all active PupSocs.

The level  1 module  must be initialized  by calling  InitPupLevel1, as
follows:

InitPupLevel1(zone,  ctxQ, numPBI,  pupDataBytes [defaultPupDataBytes],
    numRTE [9])
    Initializes  all  the  level   1  software,  and  also   calls  the
    appropriate  level  0  initialization  (InitAltoEther  in  the Alto
    version).   "zone"  is  a free-storage  zone  from  which permanent
    allocations  may  be done.   "ctxQ"  is  a pointer  to  a  queue of
    contexts to  which the  contexts created by  this procedure  may be
    appended.  "numPBI"  is the  number of PBIs  to be  allocated (from
    "zone") and appended to the pbiFreeQ.


Pup Package                November 21, 1981                         13




    The optional argument  "pupDataBytes" specifies the  maximum number
    of data (content) bytes to be permitted in any Pup; it must be even
    and  by  convention should  not  be greater  than  532.   A smaller
    maximum Pup  length is  useful in  some applications  not requiring
    high throughput,  since the  PBIs are thereby  smaller and  one can
    have more of them at the same cost in memory.  The default value of
    this parameter is 532.

    The optional argument "numRTE"  specifies the number of  entries to
    allocate in the Pup routing  table.  These are used as a  cache for
    routing information, and there  should be at least as  many entries
    as there are likely to be simultaneous conversations with  hosts on
    different networks.

    InitPupLevel1  does the  following:  it creates  the  queues pbiIQ,
    pbiTQ,  pbiFreeQ, socketQ,  and ndbQ;  allocates "numPBI"  PBIs and
    appends them to pbiFreeQ; creates the routing table  pupRT; creates
    the  default  Pup  socket  info  block  dPSIB;  calls  the  level 0
    initialization    procedure(s);   creates    the    PupLevel1   and
    GatewayListener background  contexts (to  be described  later); and
    broadcasts  requests for  gateway routing  information.   The total
    amount of  storage taken  from "zone"  (in words)  is approximately
    numPBI*290 +  lenPSIB +  lenPupSoc +  numRTE*5 +  250 +  the amount
    needed  by level  0 initialization.   InitPupLevel1 also  calls the
    external   procedure    InitForwarder   (ordinarily    defined   in
    PupDummyGate.br),  initializes  the static  pupZone  to  "zone" and
    pupCtxQ  to  "ctxQ",  and sets  up  the  constants maxPupDataBytes,
    lenPup, and lenPBI on the basis of "pupDataBytes".

    InitPupLevel1 does not call Block, so it is permissible to  call it
    from initialization code that is not running as a context.

DestroyPupLevel1()
    Undoes the actions of InitPupLevel1.  This includes calling all the
    level 0 "destroy" procedures (via the NDB.destroy operations in all
    NDBs on  ndbQ) and  releasing all  storage allocated  from pupZone.
    DestroyPupLevel1 is in the Pup1Init module, which must therefore be
    retained rather than discarded if this procedure is to be used.

The  following  procedures  are provided  for  creating  and destroying
sockets:

OpenLevel1Socket(soc, lclPort [defaulted], frnPort  [zeroes], transient
    [false])
    Creates a PupSoc.  "soc" should point to a block of size lenPupSoc.
    "lclPort", if  specified and  nonzero, points  to a  Port structure
    describing the desired local port address.  "frnPort", if specified
    and  nonzero, points  to a  Port structure  describing  the desired
    foreign  port  address.  The  "soc"  is then  appended  to socketQ,
    thereby enabling reception of Pups directed to it.

    Each field in  the local port address  is subject to  defaulting if
    either the "lclPort"  is unspecified or the  field is zero,  in the
    following  manner.  If  the socket  number is  unspecified,  one is
    chosen at random  (it is guaranteed  unique).  If both  the network
    and  host  numbers  are  unspecified, they  are  filled  in  with a
    reasonable  local  host  address  (perhaps  based  on  the supplied
    "frnPort").  Ordinarily, one should  allow the socket number  to be
    defaulted unless one intends the process to reside at a "well-known


Pup Package                November 21, 1981                         14




    socket" (as in a server),  and one should always allow  the network
    and host numbers to be defaulted.

    If "frnPort" is unspecified, the  foreign port in the "soc"  is set
    to zeroes.  Then, if the foreign network number is  zero (generally
    for the purpose  of designating the "directly  connected" network),
    it is set to the connected network's actual number, if known.  Note
    that the  "lclPort" and  "frnPort" fields in  the "soc"  are copied
    from the corresponding arguments to OpenLevel1Socket;  the argument
    ports are not modified and are not needed thereafter.

    If  "transient"  is  true, the  caller  asserts  that  the socket's
    lifetime  will be  very short.   This modifies  the  disposition of
    incoming  Pups  that  arrive  after  the  socket  has  been closed:
    ordinarily such Pups are  answered with Error pups (saying  that no
    such  socket exists),  but  if the  socket was  transient  they are
    simply ignored.  This eliminates needless extra traffic in the case
    that the  caller broadcasts a  request through the  socket, accepts
    the  first   reply,  and  immediately   closes  the   socket.   The
    "transient"  feature  works  only if  the  local  socket  number is
    defaulted.

CloseLevel1Socket(soc)
    Causes "soc"  to be  removed from  socketQ.  This  procedure blocks
    until  all PBIs  assigned  to the  socket have  been  recovered and
    released.  If "soc" is not in fact on socketQ, this procedure calls
    SysErr(soc, ecNoSuchSocket).

Control over assignment of PBIs to sockets is accomplished in  a manner
that is  more complicated  to describe  than to  implement.  Associated
with each socket are three numbers that determine the maximum number of
PBIs  that may  be assigned  to a  socket simultaneously.   The "total"
(soc>>PupSoc.maxTPBI) is  the maximum total  number of  PBIs permitted,
while   the   "input"   and   "output"   values   (soc>>PupSoc.maxIPBI.
soc>>PupSoc.maxOPBI) determine (independent  of the overall  total) the
maximum  number  of PBIs  that  may be  assigned  for  those respective
purposes.  The "total" maximum  prevents a single socket  from usurping
more than a fixed share of  the total PBIs in the system;  within that,
the "input"  and "output"  limits, if  properly set,  prevent all  of a
socket's  allocation  from  being  devoted  to  packets  going  in  one
direction (with resultant potential deadlocks).  The "total" allocation
must be greater than either "input" or "output", but need not  be equal
to their sum, since in  most applications one expects heavy  demands on
PBIs in only a single direction.

The actual number  of PBIs assigned  to a socket  at a given  moment is
reflected  in three  other  cells in  the  socket: soc>>PupSoc.numTPBI,
soc>>PupSoc.numIPBI, and soc>>PupSoc.numOPBI.  These are initialized to
the corresponding "max" values, decremented whenever a PBI  is assigned
to the  socket, and  incremented when  the PBI  is released.   The code
responsible for allocating and releasing PBIs (the PupLevel1 background
process for input PBIs and the GetPBI procedure for output PBIs) do not
permit any of these counts to go below zero; if allocating  another PBI
would cause a count to be decremented below zero, PupLevel1 will simply
discard the Pup  and release the PBI,  and GetPBI will either  block or
fail (see below).

The  allocations in  the  socket are  also useful  when  destroying the
socket.  At  the time  CloseLevel1Socket is called,  there may  be PBIs


Pup Package                November 21, 1981                         15




that  are assigned  to the  socket but  that cannot  be located  at the
moment because they  reside on some other  queue (such as  the Ethernet
output  queue or  the  pbiTQ).  CloseLevel1Socket  simply  blocks until
soc>>PupSoc.numTPBI equals  soc>>PupSoc.maxTPBI, at  which point  it is
known that all PBIs have "returned" to the socket and been released.

PBIs may be added to the free pool simply by allocating blocks  of size
lenPBI and "Enqueue"ing them  on pbiFreeQ.  One could also  remove PBIs
from the system  by "Dequeue"ing them  from pbiFreeQ and  freeing them,
but of  course one  has no control  over which  PBIs are  available for
release.  Note that  such changes in the  total number of PBIs  are not
automatically reflected  in any  socket allocations  or in  the default
allocations contained in dPSIB.

SetAllocation(soc, total, input, output)
    Changes the  number of  PBIs that  may be  assigned to  the socket.
    "total", "input",  and "output"  are the  new maximum  values.  The
    "total"  must  be  greater than  either  the  "input"  or "output".
    SetAllocation need be called only if the desired allocations differ
    from the defaults in dPSIB.  Alternatively, one may manually change
    the contents of dPSIB; note  that the "num" and "max" values  for a
    given allocation must be  the same and that the  "total" allocation
    must  be  greater  than  or  equal  to  the  "input"  and  "output"
    allocations.  Changing dPSIB does not affect allocations in sockets
    that have already been  opened.  The initial "total"  allocation in
    dPSIB  is   numPBI-numNets,  where  numPBI   is  the   argument  to
    InitPupLevel1 that determines the number of PBIs  initially created
    and numNets is the number of directly-connected  networks (normally
    one in an Alto).  The initial "input" and "output"  allocations are
    each one less than the "total".

GetPBI(soc, returnOnFail [false]) = PBI
    Assigns  a PBI  from pbiFreeQ  and charges  it to  the  socket, for
    output use (that is, it decrements soc>>PupSoc.numTPBI  (total) and
    soc>>PupSoc.numOPBI  (output)).  If  the socket  has  exhausted its
    total or output  allocation or the  pbiFreeQ is empty,  then GetPBI
    blocks unless returnOnFail is true, in which case it  returns zero.
    The PBI returned  has its Pup header  zeroed so that if  the caller
    later transmits the Pup  without setting up source  and destination
    port addresses, the addresses will be correctly defaulted  from the
    socket.  The PBI's  "queue" pointer is  set to pbiTQ,  resulting in
    automatic release of  the PBI after  it is transmitted.   The PBI's
    "socket" pointer is set  to "soc", thereby recording the  socket to
    which it has been assigned.

ReleasePBI(pbi)
    Releases the "pbi" and appropriately credits the allocations in the
    socket to which it was assigned.

CompletePup(pbi, type [], length [])
    Causes  "pbi"  to  be  completed  and   transmitted.   "Completion"
    consists  of  the  following operations:  "type"  and  "length", if
    supplied, are stored  in the Pup type  and length fields;  any zero
    fields in the Pup source or destination ports are defaulted  to the
    values  given  in  the  owning  socket's  local  and  foreign  port
    addresses,  respectively;  the  transport  control  byte  (used  by
    gateways) is zeroed;  then, if the  socket's doChecksum flag  is on
    (the default unless changed explicitly), a software Pup checksum is
    computed and stored in the Pup.  The caller is expected to have set


Pup Package                November 21, 1981                         16




    up the Pup's ID, and contents  (if any) and its type and  length if
    not  supplied in  the  call.  Finally,  the  PBI is  routed  to its
    destination and queued for transmission.

    After transmission,  the PBI is  appended to  pbi>>PBI.queue, which
    (unless changed explicitly by the caller) will be  pbiTQ, resulting
    in automatic release of the PBI.  If a different queue is specified
    for  disposal  of the  PBI  (as is  done  in the  BSP  package, for
    example), then the caller  is responsible for keeping track  of the
    PBI, and, in particular, for ensuring that all PBIs assigned to the
    socket have been released before destroying the socket.

    A special mechanism exists for broadcasting a Pup on  all directly-
    connected networks.  If  the allNets bit is  set in the  PBI status
    word, then instead of routing the Pup to the destination  stated in
    the Pup  header, CompletePup  sends the Pup  out on  each directly-
    connected network.   For each  network, the  local host  address on
    that network is substituted for the network and host numbers in the
    Pup source port, and  the local network number is  also substituted
    for the destination network field (the checksum is  recomputed each
    time this is done).  The "queue" word in the PBI must be pbiTQ (the
    default) for this feature to work properly.

    The allNets mechanism  ordinarily causes a Pup  to be sent  on each
    directly-connected network, whether  or not the  network's identity
    is known.  However, if the  bypassZeroNet bit is also set,  the Pup
    will not be sent on networks whose identity is not known.

Distribution  of   received  Pups  to   the  correct  sockets   is  the
responsibility of  a background process  called PupLevel1.  When  a PBI
appears on  pbiIQ (where  it was left  by the  level 0  input handler),
PupLevel1 first  performs some checks  on the Pup  destination address,
and discards the PBI if it  is not destined for a process in  the local
host  (actually,  it  enqueues it  on  gatewayIQ,  which,  assuming the
PupDummyGate module has been loaded, is the same as pbiFreeQ).  It then
searches the socketQ for a socket whose local socket number matches the
Pup destination socket number.  If no such socket is found, the  PBI is
passed  to  SocketNotFound(pbi),  which  generates  an  Error  Pup  and
discards  the  packet  (but  could be  made  to  do  something  else by
clobbering  the  SocketNotFound  procedure  static  with   a  different
handling procedure).

Assuming the destination socket is found, PupLevel1 then checks the Pup
checksum (assuming the socket's doChecksum flag is on),  discarding the
PBI if it is incorrect.  Finally, the socket's "total" and  "input" PBI
allocations are checked.  If either is exhausted, the PBI  is discarded
(causing an Error Pup to  be returned to the Pup's  source); otherwise,
the allocations are updated and the PBI is appended to the socket's iQ.

PupLevel1 is also responsible for releasing PBIs on the pbiTQ, which is
the  default  queue  to  which  outgoing  packets  are  appended  after
transmission.

Another  process,  GatewayListener,  is  responsible   for  dynamically
maintaining the  routing table pupRT  and updating it  with information
periodically received from  gateways.  While routing and  routing table
maintenance are operations performed automatically (by  CompletePup and
GatewayListener),  the  format  of the  routing  table  is  of possible
interest to callers in certain cases--for example, in deciding which of


Pup Package                November 21, 1981                         17




several possible remote servers is the best choice in terms  of network
topology (see the  PupNameLookup module for  an example of  this).  The
following description is much  more than most programmers will  wish to
know about.

The  RT is  a  dictionary object  consisting of  routing  table entries
(RTEs) keyed  by network  number, each  containing information  about a
specific network.  For  a given RTE, if  the "hops" field is  zero, the
network  is  one  to  which  the  local  host  is  directly  connected;
otherwise, the network may be reached via the gateway whose host number
is given in the "host" field (the "hops" field indicates the  number of
gateways believed to lie along  the route to the destination  net).  In
either  case, the  "ndb"  field points  to  the NDB  for  the immediate
destination network (see "Pup Specifications").  If the "hops" field is
greater  than "maxHops"  (currently  15), the  network is  known  to be
inaccessible, and the remainder of the RTE should not be believed.

If no RTE exists for  a particular network, then we know  nothing about
that network and can't route Pups to it.  The routing table  is treated
as a cache  of recently-used routing  information.  When an  attempt is
made to  transmit a  Pup to a  network not  represented in  the routing
table, new routing information is obtained from a nearby gateway and an
RTE  for that  network  is inserted  into the  routing  table (possibly
displacing  some other  RTE that  has not  been used  recently).  Note,
however, that  RTEs for directly-connected  networks are  never removed
from the routing table.

Network number zero  in the routing table  is special.  It refers  to a
network known  to be directly  connected to the  local host  (but whose
identity may  or may not  be known, i.e.,  we may or  may not  know its
network  number).   Pups  handed  to  CompletePup  for  transmission to
network  zero  will  be  sent  over  this  network.   This  facility is
essential during initialization, before any gateways have  been located
and the remainder of the RT filled out.  It also  permits communication
among hosts on a network  whose identity is unknown due to  there being
no connected gateways.

The routing table as a  whole is treated as an "object",  with standard
operations  defined by  a Hash  Table Preamble  (HTP).  This  object is
misnamed, since it  need not be implemented  by means of a  hash table,
and is not in the present implementation of the Pup routing table.  The
procedures described below are merely renamed versions of the Alto OS's
Call0, Call1,  etc.  The  operations return pointers  to RTEs,  and the
caller may operate on the individual RTE by means of ordinary structure
references.  The defined operations are:

HLookup(rt, net, dontPromote [...false]) = RTE or 0
    Looks up "net"  in the routing table  "rt", returning a  pointer to
    the RTE if  it is found and  zero if not.  Unless  "dontPromote" is
    supplied and true, the RTE is marked as having been referenced most
    recently.

HInsert(rt, net) = RTE
    Inserts an RTE for "net" into "rt", setting the "net" field  of the
    RTE and zeroing the rest of the entry.  If an entry  already exists
    for "net", it  is overwritten.  If no  entry already exists,  a new
    one is created,  possibly displacing the least  recently referenced
    RTE.

HDelete(rt, net)


Pup Package                November 21, 1981                         18




    Deletes the RTE for "net" in "rt", if one exists.

HEnumerate(rt, proc, arg)
    Enumerates all RTEs in "rt", calling proc(rte, arg) for each one.

The  following miscellaneous  procedures  are of  possible  interest to
callers:

LocateNet(net) = rte or 0
    Attempts to locate  a route to "net".   If an RTE for  "net" exists
    and is valid (i.e., hops not greater than maxHops), a pointer to it
    is returned.  Otherwise, activity is initiated to locate a route to
    "net" and zero is returned.

PupError(pbi, errorType, string)
    Causes  an "Error"  Pup  to be  returned  to the  sender  of "pbi",
    containing  the specified  "errorType"  and "string".   The  PBI is
    released  in  the  process.   Consult  the  "Pup   Error  Protocol"
    specification  for  more  information.   PupError  is  called  from
    several places inside PupLevel1 when incoming Pups are rejected for
    one reason or another.

ExchangePorts(pbi)
    Exchanges the  Pup source and  destination ports in  "pbi".  Useful
    when  sending a  packet  back where  it came  from  (possibly after
    modifying its contents).

AppendStringToPup(pbi, firstByte, string)
    Appends the supplied "string" to the Pup in "pbi", starting at byte
    position pbi>>PBI.pup.bytes↑firstByte, then sets the Pup  length to
    include the data so stored.  Useful for generating Pups that end in
    (or  consist  entirely of)  a  string, such  as  Error,  Abort, and
    Interrupt Pups.

SetPupDPort(pbi, port)
    Copies the specified "port" into the Pup destination port  field of
    "pbi".

SetPupSPort(pbi, port)
    Copies  the specified  "port"  into the  Pup source  port  field of
    "pbi".

SetPupID(pbi, pupID)
    Copies the two words pointed to by "pupID" into the Pup ID field of
    "pbi".

FlushQueue(queue)
    Dequeues and releases all PBIs presently on "queue".

OnesComplementAdd(a, b)
    Returns the ones-complement sum of "a" and "b".

OnesComplementSubtract(a, b)
    Returns the ones-complement difference between "a" and "b".

LeftCycle(word, count) = result
    Returns the result of left-cycling "word" by "count" mod 16 bits.

MultEq(adr1, adr2, nWords [...2]) = true or false


Pup Package                November 21, 1981                         19




    Compares the nWords words  starting at adr1 with  the corresponding
    words starting at adr2, returning true iff they all match.

Max(a, b); Min(a, b)
    Return the arithmetic maximum or minimum, respectively, of  "a" and
    "b".  These are treated as signed integers and must differ  by less
    than 2↑15.

DoubleIncrement(adr, offset)
    Adds  the  signed  16-bit integer  "offset"  to  the  32-bit number
    pointed to by "adr".  Note that a negative "offset" will  cause the
    32-bit number to be decremented.

DoubleDifference(adr1, adr2) = value
    Returns as a  16-bit signed integer  the result of  subtracting the
    32-bit  number pointed  to by  "adr2" from  the one  pointed  to by
    "adr1".  If the two numbers differ by more than 2↑15, the result is
    either  2↑15-1  or  -2↑15,  depending on  the  sign  of  the 32-bit
    difference.

DoubleSubtract(adr1, adr2)
    Subtracts  the 32-bit  number  pointed to  by "adr2"  from  the one
    pointed to by "adr1", and leaves the result in "adr1".



4. Rendezvous/Termination Protocol Interface


The RTP module (file  PupRTP) contains primitives for  establishing and
breaking   connections  with   foreign  processes   according   to  the
Rendezvous/Termination Protocol.

The local end of a  connection is maintained within the confines  of an
RTPSoc, an RTP socket structure (defined in PupRTP.decl).   This begins
with  a  level  1  Pup  socket  (PupSoc),  but  includes  the following
additional information:

    ctx                 A pointer to the background context maintaining
                        the connection.

    state               The state of the connection (see below).

    connID              The connection ID (see "Pup Specifications").

    rtpOtherPupProc     A procedure called upon receipt of any Pup that
                        is  not  part  of   the  Rendezvous/Termination
                        Protocol.

    rtpOtherTimer       A timer for use by higher levels of protocol.

    rtpOtherTimerProc   A procedure called when rtpOtherTimer expires.

There is  some other information  (wasListening, rtpTimer) used  by the
RTP module but not of interest to external programs.

At a given moment, an RTPSoc may be in one of a number of  "states".  A
detailed explanation of  the meanings of these  states may be  found in
the memo "Pup Connection State Diagram" (file <Pup>RTPStates.press).


Pup Package                November 21, 1981                         20




    stateClosed    No  connection  exists: either  none  has  ever been
                   created  or  a  previously  existing  connection has
                   terminated.

    stateRFCOut    The  local  process  has  initiated  a  request  for
                   connection (RFC) to  some foreign process.   A reply
                   is expected from the remote process.

    stateListening The local process is "listening" for an RFC from any
                   foreign process.

    stateOpen      The connection is considered by both parties to have
                   been established.  What the cooperating processes do
                   with  this connection  is a  matter  of higher-level
                   protocol (e.g., BSP).

    stateEndIn     The   foreign   process  has   requested   that  the
                   connection   be  terminated,   and  is   awaiting  a
                   confirmation from the local process.

    stateEndOut    The local process has requested that  the connection
                   be terminated, and  is awaiting a  confirmation from
                   the foreign process.

    stateDally     A transitory state having to do with the termination
                   handshake (see "Pup Specifications").

    stateAbort     The connection  has been  aborted abnormally  by the
                   foreign process.

An RTPSoc is created  by calling OpenRTPSocket, which  performs various
initialization, creates a background process to manage  the connection,
and  interacts with  some foreign  process in  one of  three  ways (see
below) to  open a  connection.  Once  the connection  is open,  the RTP
background process monitors the  socket for arrival of  Pups requesting
that the connection be closed or aborted, and updates the state  of the
socket appropriately.   The local process  may also  request explicitly
that the connection be terminated, by calling CloseRTPSocket.

The procedures defined in the RTP module are the following:

OpenRTPSocket(soc, ctxQ [pupCtxQ], openMode [modeInitAndWait], connID
          [random], otherProc [DefaultOtherPupProc], timeout
          [defaultTimeout], zone [pupZone]) = true or false
    Causes an RTP socket to be created and optional interactions with a
    foreign  process  to be  initiated.   "soc" is  a  block  of length
    lenRTPSoc which  must already  have been initialized  as a  level 1
    socket (PupSoc) by a prior call to OpenLevel1Socket.   (An external
    static "lRTPSoc" exists whose value  is the length of an  RTPSoc in
    words.) Both  the local and  foreign port addresses  (the "lclPort"
    and "frnPort" fields in the PupSoc) must be completely established,
    unless "openMode" is "listenAndWait" or "listenAndReturn", in which
    case only the local socket number (soc>>PupSoc.lclPort.socket) need
    be established.

    "ctxQ"  is a  context  queue to  which  a context  created  by this
    procedure  may be  appended.  It  defaults to  pupCtxQ  (the "ctxQ"
    passed to InitPupLevel1).


Pup Package                November 21, 1981                         21




    "openMode" specifies the  manner in which  the connection is  to be
    opened.  If  it is "modeInitAndWait",  a request for  connection to
    the  foreign process  is initiated,  and OpenRTPSocket  then blocks
    until either  the answering  RFC is  received and  the connection's
    state becomes  open (in  which case  it returns  true) or  an error
    occurs  (in  which  case the  RTPSoc  is  closed  and OpenRTPSocket
    returns  false).   If  it is  "modeInitAndReturn",  the  request is
    initiated in a similar manner, but then OpenRTPSocket  returns true
    immediately and  it is the  caller's responsibility to  monitor the
    subsequent state of the connection.

    If "openMode"  is "modeListenAndWait",  the socket  is placed  in a
    "listening" state.  When a request for connection is  received from
    some  foreign  process, a  reply  is generated  and  the connection
    becomes  open,  and OpenRTPSocket  returns  true.  If  the  mode is
    "modeListenAndReturn", OpenRTPSocket  returns true  immediately and
    it is the caller's  responsibility to monitor the  subsequent state
    of the connection.

    If  "openMode" is  "modeImmediateOpen", the  socket  is immediately
    placed in the open state (it is assumed that the caller has already
    performed  a  rendezvous with  the  foreign process  in  some other
    manner) and OpenRTPSocket returns true.

    "connID"  is  a  pointer  to  a  two-word  vector   specifying  the
    connection  ID (see  "Pup  Specifications").  If  not  specified, a
    connection  ID  is  chosen  at  random.   "connID"  need  never  be
    specified if "openMode" is one of the listening modes.

    "otherProc"  is a  procedure to  be called  when a  non-RTP  Pup is
    received  by the  socket.  This  will be  described in  more detail
    later.     If    not    specified,    "otherProc"    defaults    to
    DefaultOtherPupProc, a procedure that simply releases any PBI it is
    passed   (one   may   change   the   default   by   clobbering  the
    DefaultOtherPupProc static with something else).

    "timeout" specifies  the maximum time  OpenRTPSocket will  wait (if
    "openMode"  is  "modeInitAndWait"  or  "modeListenAndWait")  before
    timing  out  and  returning false.   It  (and  all  other "timeout"
    arguments in the Pup package) is in units of 10  milliseconds, with
    a maximum legal value of 2↑15 (a little over 5  minutes), according
    to  the   conventions  established  in   the  Timer   Package.   If
    unspecified,  "timeout"  defaults  to  "defaultTimeout",  a  static
    defined in this module, whose value in the released package is 6000
    (i.e.,    60   seconds;    this   is    set   by    the   parameter
    "defaultDefaultTimeout" in PupParams.decl).

    "zone" is a free-storage zone  from which a context block  (of size
    rtpStackSize) may  be allocated.  If  it is not  specified, pupZone
    (the "zone" passed to InitPupLevel1) is used.   Note: OpenRTPSocket
    calls InitializeContext, so the ContextInit module must be resident
    (despite what the Context Package writeup says).

CloseRTPSocket(soc, timeout [...defaultTimeout]) = true or false
    Requests  that  the  connection  rooted  in  the  RTPSoc  "soc"  be
    terminated.   If  "timeout"  is nonzero,  a  normal  termination is
    attempted if possible; if zero (or the attempted normal termination
    times  out),  the connection  is  aborted  (terminated abnormally).
    When  the  connection  has  been  closed,  the  context  created by


Pup Package                November 21, 1981                         22




    OpenRTPSocket is destroyed and  returned to the zone from  which it
    was allocated.  CloseRTPSocket then returns true if  the connection
    was  terminated  normally and  false  if abnormally.   The  level 1
    PupSoc pointed  to by "soc"  still exists, and  it is  the caller's
    responsibility to dispose of it appropriately (generally by calling
    CloseLevel1Socket).

The  process  created by  OpenRTPSocket  (called  RTPSocketProcess) has
several responsibilities.  First, all Pups arriving on the  socket's iQ
are  dequeued  and  inspected.   Ones  whose  types  are  part  of  the
Rendezvous/Termination protocol are processed internally.  All protocol
interactions  (including  replies,  retransmissions,  and  local  state
changes) are handled automatically.

Received  Pups  that  are  not  part  of  the  RTP  are  passed  to the
"rtpOtherPupProc" procedure,  which is  initialized to  the "otherProc"
argument in OpenRTPSocket.  More specifically, the statement

    (soc>>RTPSoc.rtpOtherPupProc)(pbi)

is executed,  and it  is up  to the  called procedure  to appropriately
process and  dispose of the  PBI.  Since this  call is made  within the
context of the RTPSocketProcess, which has only "rtpStackSize"  (130 as
released) words of stack space, the called procedure cannot  make heavy
demands  on  the  stack  without risk  of  stack  overflow.   One might
increase rtpStackSize (a static  defined in this module,  whose initial
value  is given  in PupParams.decl  as "defaultRTPStackSize"),  but the
safest course of action is  for the called procedure simply  to enqueue
the PBI  on some  queue looked at  by another  process with  more stack
space  available   to  it.   (One   should  note,  however,   that  the
"rtpOtherPupProc" procedure defined by the BSP module, to  be described
in the next section, manages to do all its work--a significant amount--
without  overflowing  the  RTP  process's  stack.   The  main potential
pitfall is in  calling system procedures such  as Ws that  require very
large amounts of stack space in some cases.)

"Abort" and "Error" Pups, while handled by RTPSocketProcess  (for their
effects  on   the  socket's   state),  are  also   passed  on   to  the
"rtpOtherPupProc" procedure, for purposes such as displaying  the Pup's
text to  the user.   The RTP module  distinguishes between  "fatal" and
"non-fatal" sub-types  of Errors,  treating the former  the same  as an
Abort  (thereby  placing  the  connection  in  the  "Abort"  state) and
ignoring   the   latter;   both   kinds,   however,   are   passed   to
"rtpOtherPupProc".

Additionally,  the RTPSocketProcess  checks for  expiration of  a timer
called "rtpOtherTimer"  in the  RTPSoc.  If  it expires,  the procedure
given  in  "rtpOtherTimerProc"  is  called,  with  the  socket  as  its
argument.  This facility is used in the BSP module, which also requires
the  ability  to do  asynchronous  processing.   "rtpOtherTimerProc" is
initialized to Noop when OpenRTPSocket is called.

The following miscellaneous procedures defined in the RTP module are of
possible interest to callers:

RTPFilter(pbi, checkFrnPort, checkID) = true or false
    Does selective filtering of "pbi" against parameters in  the socket
    to  which the  PBI is  assigned,  and returns  true if  the  PBI is
    accepted and false if rejected.  First, broadcast Pups (destination


Pup Package                November 21, 1981                         23




    host zero) are always rejected.  Then, if checkFrnPort is true, the
    source port  address of the  PBI is checked  for equality  with the
    foreign port address given  in the socket.  Finally, if  checkID is
    true,  the Pup  ID in  the  PBI is  checked for  equality  with the
    connection ID in the socket.

CompleteRTPPup(pbi, type, length)
    Stores "type"  and "length"  in the respective  fields of  the Pup,
    copies the connection  ID from the socket  to the Pup,  and finally
    calls CompletePup(pbi) to send it on its way.



5. Byte Stream Protocol Interface


The BSP module (files PupBSPStreams, PupBSPProt, and  PupBSPa) contains
procedures for sending  and receiving error-free,  flow-controlled byte
streams to and from a  foreign process, and for dealing with  the other
primitives defined by the BSP (namely Marks and Interrupts).

A process's interface to  the BSP module is by  way of a BSPSoc,  a BSP
socket structure, which is a further extension of an RTPSoc  (which, it
will be recalled, is an extension of a PupSoc).  The BSPSoc  contains a
large amount  of additional information,  most of which  fortunately is
not of interest to external  programs.  The items that are  of interest
are the following:

    bspStatus           A   word   containing   various   status  bits,
                        including the following three:

    markPending         A Mark has  been encountered while  reading the
                        incoming  byte  stream.   Further  attempts  at
                        input  (via  Gets  or  BSPReadBlock)  will fail
                        until this bit is cleared (either explicitly or
                        by calling BSPGetMark).

    interruptIn         An Interrupt has been received.  If  the caller
                        depends on this bit for noticing the arrival of
                        Interrupts,   then  it   must  clear   the  bit
                        explicitly after doing so.  Interrupts arriving
                        in close succession will not be distinguishable
                        as separate events unless they  are intercepted
                        via the "bspOtherPupProc"  mechanism, described
                        later.

    noInactivityTimeout This flag, normally  false, may be set  to true
                        to disable an automatic timeout  mechanism that
                        aborts  the  BSP  connection  if   the  foreign
                        process does  not respond  to any  BSP protocol
                        requests for two minutes.  The purpose  of this
                        is to detect that a connection has died (due to
                        network   failure   or   the   foreign  process
                        crashing).  Being able to disable  this timeout
                        mechanism is handy during debugging.

    bspOtherPupProc     A procedure called upon receipt of any  Pup not
                        part of the BSP (or RTP).


Pup Package                November 21, 1981                         24




    bspStr              A  block  containing  a  BSPStr,  a  BSP stream
                        structure.   This contains  the  dispatches for
                        interfacing to  the operating  system's generic
                        stream-handling  procedures (Gets,  Puts), plus
                        some information specific to the BSP stream.

A BSP  stream is  created by first  opening a  connection to  a foreign
process (by means of the RTP), then calling the following procedure:

CreateBSPStream(soc) = str
    Creates and initializes a BSP socket, and returns a pointer  to the
    stream block  within it.  "soc"  must point to  a region  of length
    lenBSPSoc (which is the  value of an external static  lBSPSoc), and
    it  must already  support one  end of  an open  RTP  connection (by
    having been passed to OpenLevel1Socket and then OpenRTPSocket).  If
    the  state  of  the  connection  is  not  stateOpen  or stateEndIn,
    CreateBSPStream returns zero.  Otherwise, the stream  is completely
    initialized  and the  pointer to  it is  returned.  See  the sample
    program at the  end of this document  for an example of  the proper
    sequence of operations for opening a BSP stream from scratch.

All the  generic stream  procedures (Gets, Puts,  etc.) must  be passed
"str"  as  an  argument,  as  should  the  procedures  BSPReadBlock and
BSPWriteBlock.  However, all other operations on the  socket (including
specialized BSP  functions such  as BSPGetMark)  must be  passed "soc".
When necessary, "str" and "soc" may be computed from each other  by the
following statements:

    str = soc+offsetBSPStr
    soc = str-offsetBSPStr

where offsetBSPStr is an external static defined in the BSP package.

The defined generic stream procedures are as follows.  The descriptions
of  Gets  and  Puts  assume  that  the  default  stream  error-handling
procedure  (invoked  by Errors(str,  ec))  is in  use;  the  real truth
appears in the description of Errors.

Gets(str, timeout [...-1]) = byte or -1
    Attempts to return the next byte from the BSP stream "str"; returns
    -1 on  any failure.  A  failure will result  if the  connection has
    become  closed  or a  Mark  has been  encountered  in  the incoming
    stream.  If "timeout" is -1 (the default), Gets  waits indefinitely
    for data to arrive (or  some failure condition to arise);  if other
    than -1, it  waits up to "timeout"  (units of 10  milliseconds) and
    then gives the failure return.

    Note  that  occurrence  of the  timeout  condition  does  not imply
    anything about the health of the connection; the timeout feature is
    provided entirely for the caller's convenience, and has  nothing to
    do  with  the  internal  connection  inactivity  timeout.   If  the
    connection  fails,  the connection  state  (soc>>RTPSoc.state) will
    change to something other than stateOpen.

Puts(str, byte, timeout [...-1]) = true or false
    Attempts to output "byte" to the BSP stream "str"; returns  true on
    success  and  false  on  failure.  A  failure  will  result  if the
    connection  has  become closed  or  the operation  times  out.  The
    "timeout"  is   defined  as   for  Gets,   with  -1   meaning  wait


Pup Package                November 21, 1981                         25




    indefinitely.  Note  that in  general, outputting a  byte to  a BSP
    stream  merely causes  that  byte to  be appended  to  a partially-
    constructed Pup  in memory;  only when a  Pup is  filled up  is any
    packet  actually  sent  over  the  net.   BSPForceOutput (described
    below) must be called to cause a partially-filled Pup to  be closed
    out and transmitted immediately.

Endofs(str) = true or false
    Returns true if there is not presently any data to be read from the
    BSP stream "str"  or a Mark has  been encountered.  Note  that this
    definition of Endofs is analogous to that for "keys" as  opposed to
    that for disk files; i.e., so long as the connection is still open,
    Endofs(str) being true says only that there is not now any  data to
    be read, not that there won't be data at some time in the future.

Closes(str) = true or false
    Closes the BSP stream "str" and destroys the associated  socket, as
    detailed in the description of CloseBSPSocket (below).

Errors(str, ec) = value
    The stream  error procedure (which  is initialized to  BSPErrors by
    CreateBSPStream) is called  under various error  conditions arising
    in  Gets  and  Puts.   The  error code  "ec"  will  be  one  of the
    following:

    ecBadStateForGets   Gets has  failed because  the connection  is no
                        longer open.  This can occur either  because an
                        Abort or fatal Error is received or because the
                        connection's  inactivity  timeout  (2  minutes)
                        expires.   (The  timeout  may  be  disabled for
                        debugging       purposes       by       setting
                        soc>>BSPSoc.noInactivityTimeout to true.)

    ecGetsTimeout       Gets   has  failed   because  no   data  became
                        available  for   reading  within   the  timeout
                        specified in the call to Gets.

    ecMarkEncountered   Gets has  failed because  it has  encountered a
                        Mark in the stream.

    ecBadStateForPuts   Puts has  failed because  the connection  is no
                        longer open.

    ecPutsTimeout       Puts has failed because it was not  possible to
                        output the byte within the timeout specified in
                        the call to Puts.

    In each case, the Gets or Puts returns the result of calling Errors
    with the  corresponding error code.   The default  Errors procedure
    returns -1 when passed any  of the Gets error codes and  false when
    passed one of the  Puts error codes, thereby obtaining  the failure
    behavior presented earlier in the descriptions of Gets and Puts.

The  remaining procedures  operate on  a "soc"  (BSPSoc) rather  than a
"str", since they are peculiar to BSP.

CloseBSPSocket(soc, timeout [...defaultTimeout]) = true or false
    Closes the connection and destroys the BSPSoc pointed to  by "soc".
    First,  if  the connection  is  still in  a  reasonable  state, any


Pup Package                November 21, 1981                         26




    pending  output  is  transmitted; CloseBSPSocket  will  wait  up to
    "timeout" for  successful acknowledgment of  this data.   Next, the
    connection  is  terminated   by  a  call  to   CloseRTPSocket  (the
    description  of which  includes the  interpretation  of "timeout").
    Then all  PBIs still  residing on the  BSPSoc's various  queues are
    released.   Finally,  the  socket   is  destroyed  by  a   call  to
    CloseLevel1Socket.  The result  returned is true if  the connection
    was closed normally, false if abnormally.

BSPGetMark(soc) = byte
    Returns the value of the pending Mark byte in the  incoming stream,
    and clears  the markPending flag  so as to  permit future  calls to
    Gets to read data past the Mark in the stream.  This procedure will
    call SysErr(soc, ecBadBSPGetMark)  if a Mark  has not in  fact been
    encountered.

BSPPutMark(soc, markByte, timeout [...-1], sendNow [false]) = true or
          false
    Inserts the specified "markByte" into the outgoing stream.  Calling
    this procedure causes all data up to and including the Mark byte to
    actually  be   transmitted  immediately.   The   interpretation  of
    "timeout" and the result returned by the procedure are the  same as
    for Puts; "sendNow" is described under BSPForceOutput (below).

BSPForceOutput(soc, sendNow [false])
    Forces   any  partially-filled   output  Pup   to   be  transmitted
    immediately.   This procedure  will never  block.  If  "sendNow" is
    true,  the BSP  package  will elicit  an  immediate acknowledgment,
    thereby expediting the process  of flushing the local  output queue
    of unacknowledged  Pups.  The  caller should  set this  argument to
    true  when  it  expects  not  to  send  more  data  for   a  while,
    particularly if it  is about to turn  around and receive  some data
    over the same stream.

BSPPutInterrupt(soc, code, string, timeout [...-1]) = true or false
    Generates a BSP Interrupt Pup (see "Pup Specifications")  using the
    specified  "code"  for  the Interrupt  Code  and  "string"  for the
    Interrupt Text.   The procedure  returns true  unless it  failed to
    send the  Interrupt due  either to the  connection no  longer being
    open or to exhausting the specified "timeout".

The BSP module accomplishes much of its work as a result of being given
control by the socket's RTPSocketProcess context through two paths: the
"rtpOtherPupProc" procedure, called when a non-RTP Pup  is encountered,
and the "rtpOtherTimerProc" procedure, called when  the "rtpOtherTimer"
expires.   These  three  cells  in  the  RTPSoc  structure  are renamed
"bspPupProc", "bspTimer", and "bspTimerProc" within the BSP module.  By
this means, the management  of both incoming and outgoing  byte streams
is   accomplished   automatically   (including   the    generation   of
acknowledgments and retransmissions).

Received Pups that are not part of either the RTP or the BSP are handed
to the  procedure given  in the "bspOtherPupProc"  cell in  the socket.
This  is  initialized  to   the  previous  contents  of   the  socket's
"rtpOtherPupProc" by  CreateBSPStream (which then  stores a  pointer to
the BSP  module's own  BSPPupProc into the  latter cell).   The earlier
description of  "rtpOtherPupProc" (in  the section  on the  RTP module)
applies to "bspOtherPupProc".


Pup Package                November 21, 1981                         27




Received Interrupt packets  are also passed to  "bspOtherPupProc" after
being processed by  the BSP module.  Note  that an Interrupt  passed in
this manner has been verified to conform to protocol (this is  the case
also for Abort and Error packets passed up from the RTP module) and may
therefore be "believed".  Any other type of packet, on the  other hand,
has had no checking done on it beyond the level 1 interface  (where the
destination port and checksum were verified).

A note on allocations:  this BSP implementation probably will  not work
at all unless the socket's PBI allocations are at least 3, 2, and 2 for
"total", "input", and  "output" respectively.  High throughput  will be
gained only by giving the socket somewhat larger allocations (say, 6 to
10 PBIs) for the direction(s) in which high throughput is desired.

In a program with at most one active BSP connection, that socket should
be allocated  all of the  PBIs in the  system except one  per directly-
connected network  (there must  always be one  extra PBI  available for
receiving  incoming  packets  on each  network);  this  is  the default
allocation established  in dPSIB by  InitPupLevel1.  In a  program with
several  active  connections,  one  should  adjust   individual  socket
allocations appropriately (though  probably not simply by  dividing the
total PBIs by the number of sockets, since doing so typically  leads to
underutilization of PBIs).   Assuming there are  plenty of PBIs  in the
system,  it  is  generally  safe  to  overcommit  the  system resources
(relying  on  the  statistical  unlikelihood  that  all   sockets  will
simultaneously  tie up  all  the PBIs  to which  they  are individually
entitled).   One  should  be  aware,  however,  that  the  higher-level
protocols  can  get into  deadlock  conditions if  the  system pbiFreeQ
becomes exhausted.  For  the same reason, a  PBI passed to  an external
program via the "bspOtherPupProc" entry in the socket must  be released
as  quickly  as possible,  since  it is  charged  against  the socket's
allocation.

The BSP module includes a static "bspVersion" whose value  is (protocol
version * 1000) + package version.



6. BSP Block Transfer Procedures


The BSP stream mechanism just presented, while being a "fast stream" in
the sense defined by the operating system, is still relatively slow and
is  therefore not  well suited  to transferring  large volumes  of data
(such  as file  transfers  between disk  and net).   A  separate module
(PupBSPBlock) is provided for accomplishing block transfers at least an
order of magnitude faster than by iterated calls on Gets or Puts.  This
module requires  that the  AltoByteBlt module  (released as  a separate
package) be loaded as well.

Two procedures are defined in this module:

BSPReadBlock(str, wordP, byteP, count, timeout [...-1]) = count
    Reads a maximum of "count" bytes from the BSP stream "str", storing
    them in memory starting  at byte position "byteP" relative  to word
    address "wordP" (for example, byteP = 0 means the left byte  of the
    word referenced by "wordP").  The transfer terminates under  any of
    the  conditions that  would cause  Gets(soc,timeout) to  return -1.
    The procedure returns the actual number of bytes transferred.


Pup Package                November 21, 1981                         28




BSPWriteBlock(str, wordP, byteP, count, timeout [...-1]) = count
    Writes  a  maximum  of  "count"  bytes  to  the  BSP  stream "str",
    obtaining  them  from  memory  starting  at  byte  position "byteP"
    relative to  word address "wordP".   The transfer  terminates under
    any of  the conditions that  would cause  Puts(soc,byte,timeout) to
    return false.   The procedure  returns the  actual number  of bytes
    transferred.



7. Network Directory Lookup Module


This  module  (file  PupNetDirLookup)  contains  procedures  to convert
between  a string  consisting of  any legal  inter-network name/address
expression and a  Port structure containing the  corresponding address.
See  the  memo  "Naming  and  Addressing  Conventions  for  Pup"  (file
<Pup>PupName.press) for information on legal expressions.

The PupNetDirLookup module contains all the procedures described below.
The alternative PupNameLookup module,  included in the Pup  package for
historical reasons, contains only GetPartner and  ParseAddressConst; it
is  somewhat  smaller  than  PupNetDirLookup, and  it  may  be  used in
situations where the other functions are not needed.

GetPartner(name, stream [none], port, s1 [...none], s2 [...none]) =
          true or false
    Parses  the BCPL  string "name"  and stores  the  resulting address
    value in  the Port structure  "port", returning true  if successful
    and false otherwise.  "stream", if nonzero, is used  for publishing
    an error message if the conversion is unsuccessful.  "s1" and "s2",
    if supplied, specify the  high- and low-order parts of  the default
    socket number, which is  substituted into the "port" if  the socket
    number is unspecified in the "name".

    If the "name" consists  entirely of address constants (in  the form
    "net#host#socket" or some subset thereof, where the  components are
    octal numbers), then  it is parsed locally.   Otherwise, GetPartner
    attempts  to  establish  contact with  a  Network  Directory Lookup
    server, to which it passes the "name" for evaluation.  If the reply
    consists of several alternative addresses, the "best" one is chosen
    on the basis of information in the local routing table.  Regardless
    of whether  or not  the string is  an address  constant, GetPartner
    will return false (with the message "Can't get there from here") if
    no routing table entry exists for the resulting network and several
    calls to LocateNet discover no way of reaching that network.

ParseAddressConst(name, port) = true or false
    Attempts to parse the BCPL string "name" as an address  constant of
    the  form  "net#host#socket".   Stores  the  result  in  "port" and
    returns true if successful; returns false if unsuccessful.

EnumeratePupAddresses(name, filter [TakeTheFirst], arg [], dontCheckRT
          [false]) = 0 or error code
    Evaluates "name" to one or more addresses.  For each address, calls
    filter(port,  arg), where  port is  a pointer  to a  Port structure
    containing  that   address  and   arg  is   the  "arg"   passed  to
    EnumeratePupAddresses.  If filter returns true,  enumeration stops;
    if false, enumeration continues.


Pup Package                November 21, 1981                         29




    Ordinarily  the addresses  are  enumerated in  order  of increasing
    distance away from here (hop count), and inaccessible addresses are
    omitted;  however,  if  "dontCheckRT"  is  supplied  and  true, the
    addresses are not sorted and are all returned.  Filter  defaults to
    an internal procedure TakeTheFirst, which treats "arg" as a pointer
    to a Port and copies the first Pup address to that port, discarding
    the rest.

    EnumeratePupAddresses returns zero if successful and an  error code
    (from  PupNetDir.decl)  if  unsuccessful.   The  error   codes  are
    ecNameNotFound,  meaning that  the name  is not  registered  in the
    Network Directory  server data  base; ecCantGetThere,  meaning that
    the name exists but none of the addresses are  currently accessible
    in the internet;  and ecNoServerResponded, meaning that  no Network
    Directory server could be located.

PupAddressLookup(port, lvEC [], zone [pupZone]) = string or 0
    Interacts with  a network  directory server to  find a  name string
    corresponding  to  the  Pup  address  pointed  to  by  "port".   If
    successful, returns a string allocated from "zone"  (which defaults
    to pupZone,  the zone passed  to InitPupLevel1).   If unsuccessful,
    stores an error code  (from PupNetDir.decl) at @lvEC (which  may be
    omitted) and returns zero.  The error codes  are ecAddressNotFound,
    meaning that the address is not registered in the Network Directory
    server data base; and ecNoServerResponded, meaning that  no Network
    Directory server could be located.



8. Example


The  following example  program  makes use  of most  of  the facilities
provided in  the Pup  package.  It is  basically a  rock-bottom minimal
user Telnet (like Chat) with no redeeming features whatsoever.

The main procedure  PupExample performs initialization,  which consists
of  creating a  large zone,  initializing the  Pup package,  creating a
large display window, and  creating and starting a context  running the
procedure TopLevel.

TopLevel first requests the user to type in a foreign port  name, which
it  parses  by  calling  GetPartner (note  that  the  socket  number is
defaulted to 1,  the server Telnet socket).   Then a socket  is created
and a connection is opened.  Two new contexts are now  created, running
the  procedures KeysToNet  and  NetToDsp.  TopLevel  then  blocks until
either the connection is no longer open or the second blank key  on the
right of the  keyboard is pressed, at  which point it destroys  the two
contexts  it created,  closes  the connection,  and loops  back  to the
beginning.

The KeysToNet procedure blocks waiting for keyboard input, then outputs
the typed-in character  to the BSP  stream and calls  BSPForceOutput to
force  immediate transmission.   If  the Puts  fails,  KeysToNet simply
blocks forever, in the  expectation that TopLevel will detect  that the
connection is no longer open and take appropriate action.

The NetToDsp procedure  blocks waiting for  input from the  BSP stream.
When a normal character is  received, it is output to the  display.  If


Pup Package                November 21, 1981                         30




Gets returns -1,  then either a Mark  is pending or the  connection has
ended; if the former, a message is printed and BSPGetMark is  called to
clear  the  Mark  pending  status;  if  the  latter,   NetToDsp  blocks
indefinitely.


// PupExample.bcpl

// Bldr PupExample PupBSPStreams PupBSPProt PupBSPa PupBSPOpenClose ↑
//  PupRTP PupRTPOpenClose PupNameLookup ↑
//  Pup1b PupAl1a Pup1OpenClose PupRoute ↑
//  PupAlEthb PupAlEtha ↑
//  Context ContextInit Interrupt AltoQueue AltoTimer ↑
//  Pup1Init PupDummyGate PupAlEthInit InterruptInit

get "Pup.decl"

external
[
InitPupLevel1; OpenLevel1Socket; CloseLevel1Socket; SetAllocation
OpenRTPSocket; CreateBSPStream; GetPartner
BSPForceOutput; BSPGetMark
InitializeContext; CallContextList; Block; Enqueue; Unqueue
InitializeZone; CreateDisplayStream; ShowDisplayStream
Gets; Puts; Closes; Endofs; Ws
keys; dsp
]
static [ ctxQ; myDsp; bspSoc; bspStr ]


let PupExample() be  // initialization
[
let myZone = vec 10000; InitializeZone(myZone, 10000)
let q = vec 1; ctxQ = q; ctxQ!0 = 0
InitPupLevel1(myZone, ctxQ, 20)
let v = vec 10000
myDsp = CreateDisplayStream(40, v, 10000)
ShowDisplayStream(myDsp)
let v = vec 3000
Enqueue(ctxQ, InitializeContext(v, 3000, TopLevel))
CallContextList(ctxQ!0) repeat
]


and TopLevel() be  // top-level process
[
Ws("*nConnect to: ")
let name = vec 127; GetString(name)
if name>>String.length eq 0 then finish
let frnPort = vec lenPort
unless GetPartner(name, dsp, frnPort, 0, 1) do loop
let v = vec lenBSPSoc; bspSoc = v
OpenLevel1Socket(bspSoc, 0, frnPort)
unless OpenRTPSocket(bspSoc, ctxQ) do
   [ Ws("*nFailed to connect"); CloseLevel1Socket(bspSoc); loop]
Ws("*nOpen!")
bspStr = CreateBSPStream(bspSoc)
let keysToNetCtx, netToDspCtx = vec 1000, vec 1000
Enqueue(ctxQ, InitializeContext(keysToNetCtx, 1000, KeysToNet))
Enqueue(ctxQ, InitializeContext(netToDspCtx, 1000, NetToDsp))


Pup Package                November 21, 1981                         31




Block() repeatuntil bspSoc>>BSPSoc.state ne stateOpen %
   @#177035 eq #177775  //second blank key pressed
Unqueue(ctxQ, keysToNetCtx); Unqueue(ctxQ, netToDspCtx)
Closes(bspStr)
Ws("*nClosed!")
] repeat


and KeysToNet() be
[
test Puts(bspStr, GetKeys())
   ifso BSPForceOutput(bspSoc)
   ifnot Block() repeat
] repeat


and NetToDsp() be
[
let char = Gets(bspStr)
if char eq -1 then
   test bspSoc>>BSPSoc.markPending
      ifso
         [
         Ws("*nI saw a Mark!")
         BSPGetMark(bspSoc)
         loop
         ]
      ifnot Block() repeat
Puts(myDsp, char)
] repeat


and GetKeys() = valof
[
while Endofs(keys) do Block()
resultis Gets(keys)
]


and GetString(string) be
[
for i = 1 to 255 do
   [
   let char = GetKeys(); Puts(dsp, char)
   test char eq $*n
      ifnot string>>String.char↑i = char
      ifso [ string>>String.length = i-1; return ]
   ]
]



9. Revision History


March 25, 1976

Various  minor bugs  in both  code and  documentation were  fixed.  One
serious  error  in  the   documentation  was  in  the   description  of


Pup Package                November 21, 1981                         32




CreateBSPStream, where "lenBSPStr"  should have been  "lenBSPSoc".  The
level 1, RTP,  and BSP modules  each became slightly  smaller.  Various
calls to CallSwat were changed to SysErr with registered error codes.

Level 0: External change: file PupAlEth.bcpl replaced by PupAlEthb.bcpl
and  PupAlEtha.asm.  Internal  change: fast  (~20-instruction) Ethernet
receiver turnaround implemented.

Level  1:  External   changes:  statics  pupZone  and   pupCtxQ  added;
procedures SetPupDPort, SetPupSPort, SetPupSPort, and FlushQueue added;
RT structure definition changed; default pupErrSt is now a "nil" stream
rather than "dsp".

RTP:  External changes:  defaultTimeout and  rtpStackSize  changed from
manifests  to statics  (with default  values  defaultDefaultTimeout and
defaultRTPStackSize); DefaultOtherPupProc added.

BSP: External  change: static bspVersion  added.  Internal  change: the
transmission strategy was  modified to elicit an  acknowledgment before
allocation is completely exhausted, hence reducing lost  throughput due
to round-trip delay.

April 16, 1976

The released package Pup.dm was renamed PupPackage.dm, and  a debugging
version  of the  package  released as  PupDebug.dm.  A  number  of bugs
(particularly in level 1) were uncovered while bringing up the software
on the Nova.

Level 0: External change:  lenPup and lenPBI changed from  manifests to
statics  (defined  in level  1)  to permit  changing  PBI  size without
recompiling  the  package.  Internal  change:  100-millisecond transmit
timeout and discard added (eliminating deadlocks caused by  things like
disconnecting the Alto from the Ethernet).

Level 1: External changes:  gateway code split out into  separate files
PupGateway and PupDummyGate, one  of which must be loaded  (usually the
latter); optional extra argument "pupDataBytes" added to InitPupLevel1;
default allocations in dPSIB changed  to permit a socket to  assign all
but  one  of the  PBIs  in the  system;  OpenLevel1Socket  defaults the
foreign net in some  circumstances.  Internal change: if  "pupDebug" is
on, PupLevel1 checks for the pbiFreeQ being exhausted for more  than 20
seconds and calls Swat (this usually indicates a deadlock).

BSP: External change:  PupBSPb.bcpl replaced by  PupBSPStreams.bcpl and
PupBSPProt.bcpl  (necessitated  by  Nova  BCPL's  inability  to compile
PupBSPb in one gulp).

May 18, 1976

Mostly  bug  fixes   and  performance  improvements.    Some  structure
definitions were changed, so recompilation of user programs is advised.

Level 0: Internal changes: more assembly code included to reduce packet
loss rate; performance statistics gathered if pupDebug on.

Level 1: External change: optional "type" and "length"  arguments added
to CompletePup.

October 6, 1976


Pup Package                November 21, 1981                         33




Significant internal changes were made  at levels 0 and 1,  and several
new capabilities were  added.  However, for  the most part  the changes
are  upward-compatible.    Many  structure  declarations   changed,  so
recompilation of programs that "get" any Pup .decl files is required.

Level  0: External  changes: SendEtherPup  removed; EncapsulateEtherPup
and  SendEtherPacket  added;  ability  to  send  and  receive  non-Pups
implemented.

Level  1:  External  changes: PupRoute  file  added;  PupGateway module
deleted  from  public  Pup package  release;  routing  table completely
reorganized;  new  procedures  HLookup,  HInsert,  HDelete, HEnumerate,
HHash added; pupErrSt removed; mechanism added for broadcasting  to all
connected   networks;  procedures   DoubleIncrement,  DoubleDifference,
Double subtract  included (formerly in  BSP module).   Internal change:
GatewayListener dynamically maintains the best path to each network and
purges  RTEs of  networks  for which  no routing  information  has been
received recently.

BSP: Internal  change: adaptive  retransmission timeout  implemented to
reduce packet loss rate when  sending through slow networks or  to slow
destinations (e.g., Maxc).

March 21, 1977

Mostly bug fixes.  Some structure definitions at level 0  were changed,
so recompilation of user programs is advised.

Level 0: SendStats operation added to the NDB object.

July 11, 1977

No external  changes.  The Ethernet  driver was rewritten  to eliminate
several  low-probability   race  conditions  and   improve  performance
slightly.   The  driver  now  uses  the  "input  under  output" feature
unconditionally, so problems may be encountered on Alto-Is  running old
microcode.

March 20, 1978

External changes:  Several source files  have been broken  into smaller
pieces to  permit much  of the code  to be  included in  overlays.  The
added modules  are Pup1OpenClose, PupRTPOpenClose,  and PupBSPOpenClose
(see section 1.2 for revised packaging information).   Recompilation is
required of any programs that get Pup.decl or PupBSP.decl.

Internal changes: BSP performance  through slow links and  gateways has
been improved.  GetPartner's timeout  has been increased.  A  few minor
bugs have been fixed.

November 6, 1978

Level 0: External changes: The Ethernet driver can now control multiple
interfaces  connected  to  the  same  Alto.   InitAltoEther  is  called
differently.  Drivers are now available for the XEOS EIA  interface and
the ASD Communication Processor, though they aren't documented here.

Level  1:  Internal  change: The  routing  module  data  structures and
algorithms have  geen modified  to conform to  some minor  Pup protocol
changes.


Pup Package                November 21, 1981                         34




BSP:  External  change:  An  inactivity  timeout  has  been   added  to
automatically abort connections that have died; this may be disabled by
setting the  noInactivityTimeout bit in  the BSPSoc.   Recompilation is
required of any programs that get Pup.decl or PupBSP.decl.

February 19, 1979

Level 0:  Internal change:  The format of  the statistics  collected by
drivers changed.

Level  1:  Internal  change: The  interface  between  PupRoute  and the
forwarder changed.   This is  only of  interest to  gateways.  External
change: The definitions of  pup types and well-known sockets  have been
removed from Pup1.decl.  We now  feel that these belong in  less global
declaration  files  closer   to  the  code  implementing   the  various
protocols.   Recompilation  of   user  programs  will   probably  cause
undefined symbols which you will have to add to your declaration files.

May 27, 1979

Level 1: External  changes: InitPupLevel1 takes an  additional optional
argument   "numRTE";   LocateNet   procedure   added;   HHash  removed;
PupDummyGate  module  moved from  resident  to  initialization; lPupSoc
static added.  Internal  changes: the routing table  is now a  cache of
routing information rather than a hash table of all accessible networks
in the internet; RTEs may  be "invalid" (hops greater than  maxHops and
ndb equal to zero); new PupRoute.decl includes definitions  internal to
the routing module.

RTP: External change: lRTPSoc static added.  Internal change: more code
moved from resident (PupRTP) to swappable (PupRTPOpenClose) modules.

BSP: External change: lBSPSoc and offsetBSPStr statics added.  Internal
changes: minor bugs fixed; more code moved from resident (PupBSPStreams
and PupBSPProt) to swappable (PupBSPOpenClose) modules.

NameLookup: External  changes: RequestNameLookup  and ParseAddressConst
procedures now exported.

March 9, 1980

Levels 0 & 1: External change: a "destroy" operation has been  added to
the NDB, and  the procedure DestroyPupLevel1  is included to  shut down
the Pup package.  A few bugs have been fixed.

December 30, 1980

BSP:  External  change:   The  default  timeouts  for   Puts,  PutMark,
PutInterrupt and BSPWriteBlock were changed from 'defaultTimeout' to '-
1' (infinity).   This makes the  'put' operations symmetrical  with the
'get' operations: so long as the other end of the connection  is alive,
the BSP will wait indefinitely to put or get a byte.   Recompilation of
client programs is not necessary.

January 25, 1981

BSP:   External   change:   optional   'sendNow'   argument   added  to
BSPForceOutput and BSPPutMark.  Recompilation of client programs is not
necessary.


Pup Package                November 21, 1981                         35




November 21, 1981

Level  1:  External  change:  optional  "transient"  argument  added to
OpenLevel1Socket.

NameLookup: External change: a new module PupNetDirLookup  provides the
same functions as PupNameLookup (which still exists),  and additionally
contains  new  procedures  EnumeratePupAddresses  and PupAddressLookup.
Recompilation of client programs is not necessary.