Go to the previous, next section.

Using ILU with Common Lisp

Introduction

This document is for the Common Lisp programmer who wishes to use ILU. The following sections will show how ILU is mapped into Common Lisp constructs and how both Common Lisp clients and servers are generated and built.

The ISL Mapping to Common Lisp

Runtime code is in the Common Lisp package ilu.

Names

Names from interface specifications are transformed into Lisp names (case-insensitive) by inserting hyphens at lower-to-upper case transitions. Hyphens that are already present are maintained as is.(7) Method names are mapped to INTERFACE:TYPE.METHOD.

Interface

A separate package is defined for each interface with defpackage. The name of this package is taken from the name of the interface. This package uses the packages common-lisp and ilu. The Common Lisp names of all entities defined in the ISL are exported from the package, including types, classes, constants, accessors, type predicates, generic functions, exceptions, etc. Such symbols are also shadowed, to avoid conflicts with used packages. For example, given the following interface:

INTERFACE MyInterface END;
EXCEPTION TotalWinner : Person;
TYPE Person = OBJECT
  METHODS
     Enemies (someone : Person) : Cardinal
       RAISES TotalWinner END
  END;

the stubber generates the following defpackage:

(defpackage :my-interface
  (:use :common-lisp :ilu)
  (:shadow #:total-winner #:person #:person.enemies)
  (:export #:total-winner #:person #:person.enemies)

This allows symbols defined in the commonlisp package to be used by the automatically generated code in the generated package, but it also means that the user needs to be careful about using any generated package. In general, we recommend that you explicitly specify the full name of symbols from ILU interfaces.

Basic Types

The basic ISL types have the following mapping to Common Lisp types:

Constant

Constants are implemented in CL by a value of the appropriate type, defined with defconstant.

Strings

Arrays and sequences of CHARACTER (regular or SHORT) are implemented as simple-strings.

Pickles and Typecodes

Pickles are represented with the CLOS class ilu:pickle

Instances of a pickle may be created by calling cl:make-instance on ILU:PICKLE, with the :VALUE and :TYPE keywords, as in
(cl:make-instance 'ilu:pickle :type 'ilu:cardinal :value 3456)

Pickle has three reader functions defined on it:

Constructed Types

Enumeration

Enumerations are implemented with symbols, as in
(deftype answer () `(member 'yes 'no 'maybe))

Array

Arrays are implemented as simple-arrays.

Sequence

Sequences are implemented as lists, except for sequences of characters, which are implemented as simple-strings.

Record

Record types are implemented with CL defstruct.

Union

Unions are implemented as a cons'ed value, with the cdr containing the union type discriminant, and the cdr containing the actual value.

Optional

Object Types

Classes are implemented with CLOS defclass.

Private slots are created for methods which are specified as functional, and the runtime caches the value of this method in such slots after the first call to the method.

Instances are always subtypes of ilu:ilu-object.

Methods always take as their first argument the object which they are a method on. Subsequent arguments are those specified in the `.isl' file. Methods that have OUT or INOUT arguments may return multiple values. In general, the parameters to a method are the IN and INOUT parameters specified in the ISL interface, but not the OUT parameters. The return values from a method are the specified return value for the ISL method, if any, followed by the INOUT and OUT parameters for the method, if any, in the order in which they appear in the ISL specification of the method.

OMG IDL attributes map to a CLOS method of the same name, and a setf method with the same name (unless the attribute is readonly).

Surrogate and True Object Types

Methods, Parameters, and Exceptions

Exceptions are represented with CL conditions, defined by define-condition. All ILU conditions are subtypes of ilu:rpc-exception, which is a serious-condition. If an associated value is specified for an exception it may be accessed in one of the following two ways:

  1. If the name of the value type begins with "-ilu-prefix-idlExceptionType-", the value type is a generated type from an OMG IDL exception description, and is a record type. In this case, each of the fields of the record type are placed in the condition individually, and an accessor with that field name is declared for that field.
  2. In all other cases, there is a single accessor called value through which the associated value may be read.

Garbage Collection and COLLECTIBLE

Access to standard ILU features

Servers and Ports

Object Tables

Server Relocation

Threading and Event Loops

Custom Records

Custom Surrogates

Custom surrogates allow the user to specify custom surrogate object types which may have additional functionality in terms of caching or other side effects, and have them created instead of the default ILU surrogate object type when an instance is received. This functionality is provided in the Common Lisp runtime with the function ilu:register-custom-surrogate.

String Binding Handle Formation

Simple Binding

Principal Identities and Passports

It's also possible to find out who is making the call by examining the value of ilu:*caller-identity*.

Building Common Lisp/ILU Applications

Stub Generation

The program ILU lisp-stubber takes a interface specification (an `.isl' file) and generates lisp code to provide both client-side and server-side support for the interface. The files are generated in the current working directory. In particular, the following files are generated:

Implementing an ILU module in Common Lisp

For each ILU class interface.otype, ILU will define, in the file `interface-server-procs.lisp', a CLOS class called interface:otype.IMPL. To implement a true object for interface.otype, one should further subclass this CLOS class, and override all of its methods. In particular, do not let any of the default methods for the class be called from your methods for it.

ILU supports, in each address space, multiple instances of something called a kernel server, each of which in turn supports some set of object instances. A kernel server exports its objects by making them available to other modules. It may do so via one or more ports, which are abstractly a tuple of (rpc protocol, transport type, transport address). For example, a typical port might provide access to a kernel server's objects via (Sun RPC, TCP/IP, UNIX port 2076). Another port on the same kernel server might provide access to the objects via (Xerox Courier, XNS SPP, XNS port 1394).

When creating an instance of a true object, a kernel server for it, and an instance id (the name by which the kernel server knows it) for it must be determined. These may be specified explicitly by use of the keyword arguments to commonlisp:make-instance :ilu-kernel-server and :ilu-instance-handle, respectively. If they are not specified explicitly, the variable ilu:*default-server* will be bound, and its value will be used; a default instance handle, unique relative to the kernel server, will be generated.

A kernel server may be created by instantiating the class ilu:kernel-server. The keyword argument :id may be specified to select a name for the server. Note that ILU object IDs, which consist of the kernel server ID, plus the instance handle of the object on that server, must be unique "across space and time", as the saying goes. If no kernel server id is specified, ILU will generate one automatically, using an algorithm that provides a high probability of uniqueness. If you explicitly specify a kernel server ID, a good technique is to use a prefix or suffix which uniquely identifies some domain in which you can assure the uniqueness of the remaining part of the ID. For example, when using ILU at some project called NIFTY at some internet site in the IP domain department.company.com, one might use kernel server IDs with names like something.NIFTY.department.company.com.

=> (make-instance 'ilu:kernel-server :id "FOO-SERVER-1")
#<ILU:KERNEL-SERVER "FOO-SERVER-1">
=> (make-instance 'ilu:kernel-server)
#<ILU:KERNEL-SERVER "121.2.100.231.1404.2c7577eb.3e5a28f">
=>

Implementation Inheritance

Exporting Objects

To export a module for use by other modules, simply instantiate one or more instances of your subtypes of interface:otype.IMPL (which will inherit from ilu:ilu-true-object.

=> (make-instance 'foo:my-bar.impl :ilu-kernel-server s)
#<FOO:MY-BAR.IMPL 0x3b32e8 "1">
=>

The simplest Common Lisp "server" code would look something like:

(defun start-server ()
  (make-instance 'foo:my-bar.impl))

which will create an instance of FOO:MY-BAR.IMPL and export it via a default server.

To enable users of your module find the exported objects, you may register the string binding handle of the object or objects, along with their type IDs, in any name service or registry that is convenient for you. In release 1.6 of ILU, we are supporting an experimental simple binding method that allows you to "publish" an object, which registers it in a domain-wide registry, and then to withdraw the object, if necessary. Potential clients can find the string binding handle and type ID of the object by calling a lookup function. Note that this interface and service is experimental, and may be supported differently in future releases of the ILU system.

If you wanted to create an instance, and publish it, the code for starting a service might look something like this:

(defun start-server ()
  (let* ((ks (make-instance 'ilu:kernel-server
                ;; specify the service id
                :id "service.localdomain.company.com"))
         (si (make-instance 'foo:my-bar.impl
                ;; specify the server
                :ilu-kernel-server ks
                ;; specify the instance handle
                :ilu-instance-handle "theServer")))
    ;; the OID for "si" is now "theServer@service.localdomain.company.com"
    (ilu:publish si)
    si))

Someone who wanted to use this service could then find it with the following:

(defun find-server ()
  (ilu:lookup 'foo:bar "theServer@service.localdomain.company.com"))

Debugging

To help with finding errors in your methods, the variable *debug-uncaught-conditions* is provided.

Using an ILU module in Common Lisp

To use a module from Common Lisp, you must first have loaded the PDEFSYS "system" that describes the module. Typically, for an ILU interface called Foo, the system can be loaded by invoking (pdefsys:load-system :foo). Next, you must bind an instance of an object from that interface. The most common way of doing this is to receive an instance of an object from a method called on another object. But to get the first object exported by that module, one can use either ilu:sbh->instance or ilu:lookup.

Dumping an image with ILU

ILU has dynamic runtime state. In particular, after it is initialized, it uses several Common Lisp threads to maintain part of its state, and may also keep open connections on operating system communication interfaces. If you wish to dump an image containing ILU, you must dump the image before initializing the ILU module.

Initialization occurs automatically whenever a instance of ilu:ilu-object or ilu:rpc-server is created. Thus you should not create any instances of either true or surrogate ILU objects before dumping the image. However, you may load all the interface code for any interfaces that you are using, before dumping the image.

Initialization may also be accomplished by an explicit call to ilu:initialize-ilu. You may check to see whether the system has been initialized by examining the variable ilu::*ilu-initialized*, which is t iff ilu:initialize-ilu has been invoked.

Notes for Microsoft Windows Users

Installation on the MS Windows platform

To install the Lisp binding on the MS Windows platform proceed as follows: Copy or rename the file `ilu-non-threaded-sysdcl.lisp' in directory `ILUSRC\runtime\lisp' to `ilu-sysdcl.lisp'. Compile the Lisp runtime files; i.e., start Allegro and type:

(load "c:\\ilu\\src\\runtime\\lisp\\compile-files.lisp")
Copy the resulting `*.fsl' files and the files `ilu-sysdcl.lisp' and `pdefsys.lisp' to the Lisp installation directory (`ILUHOME\lisp'). Copy the ILU kernel and Lisp DLLs into a directory that is on your PATH.

Allegro 3.0.1 Needs ilu:run-main-loop

Because Allegro 3.0.1 is single-threaded, servers on Windows 95/NT must run the ILU mainloop. To run it indefinitely, use

(ilu:run-main-loop)
Or allocate a handle, which can later, presumably in a method call, be used to exit the event loop:
(setf *handle* (ilu:create-main-loop-handle))
(ilu:run-main-loop *handle*)
...
(ilu:exit-main-loop *handle*)
For example, to run the example server in directory `examples/test1', start Allegro 3.0.1 for Windows, and type the following:
(load "c:\\ilu\\examples\\test1\\load-lisp-example.lisp")
(test1-server:start-server)
(ilu:run-main-loop)

The Portable DEFSYSTEM Module

ILU support uses a portable implementation of DEFSYSTEM to specify modules to Common Lisp. See section The ILU Common Lisp Portable DEFSYSTEM Module, for details of this system.

ILU Common Lisp Lightweight Processes

ILU currently assumes the existence of lightweight process, or thread, support in your Common Lisp implementation. It uses these internally via a generic veneer, described fully in section The ILU Common Lisp Lightweight Process System.

Porting ILU to a New Common Lisp Implementation

The Lisp support provided with ILU includes support for the Franz Allegro Common Lisp 4.x implementation. To use ILU with other Common Lisp implementations, please see section Porting ILU to Common Lisp Implementations.

Common Lisp/ILU API Reference

Method: ilu:ilu-class-info (DISC (or ilu:ilu-object type-name)) (WHAT keyword) => (or string boolean list)

This routine will return the specified piece of information about the ILU class specified with DISC, which may be either a CLOS class name, or an instance of the class, and with WHAT, which identifies which piece of information to return. WHAT may have the following values:

Method: cl:make-instance 'ilu:kernel-server &key {(id string nil)} {(unix-port fixnum 0)} {(object-table list of 2 elements nil)} {(protocol string "sunrpc")} {(transport list of string ("sunrpcrm" "tcp_0_0"))} => ilu:kernel-server

Creates and returns an instance of ilu:kernel-server. If id is specified, the server has that value for its server ID. If unix-port is specified, the server attempts to `listen' on that UNIX port, if the notion of a UNIX port is applicable. If object-table is specified, it must consist of a list of two functions. The first function must take a string, which is the instance handle of a desired object on this kernel server, and return a value of type ilu:ilu-true-object. The second funtion must free up any resources used by this object table. Specific protocols and transport stacks can be specified with the protocol and transport keywords; these default to whatever defaults were selected when your ILU installation was built.

Method: cl:make-instance 'ilu:ilu-true-object &key {(ilu-kernel-server ilu:kernel-server nil)} {(ilu-instance-handle string nil)} => ilu:ilu-true-object

Creates and returns an instance of ilu:ilu-true-object. If ilu-true-server is specified, the instance is created on the specified server. If ilu-instance-handle is specified, that instance handle is used.

Variable: ilu:*caller-identity*

The identity of the caller is bound to the special variable ilu:*caller-identity*. It is a string which begins with the name of an identity scheme, followed by an identity in that scheme. For example, an identity in the SunRPC UNIX identity scheme would be something like "sunrpc-unix:2345,67@13.12.11.10" (i.e., "sunrpc-unix:<uid>,<gid>@<hostname>"). If no identity is furnished, a zero-length string is bound.

Function: ilu:publish (OBJ ilu:ilu-object) => boolean

Accepts an ilu:ilu-object instance and registers it with some domain-wide registration service. The object is known by its object ID (OID), which is composed of the ID of its kernel server, plus a server-relative instance ID, typically composed as instance-ID@server-ID. Clients may find the object by looking up the OID via the ilu:lookup function. The function returns non-cl:nil if the publication succeeded.

Function: ilu:withdraw (OBJ ilu:ilu-object) => boolean

If OBJ is registered, and if it was published by the same address space that is calling withdraw, its registration is withdrawn. The function returns non-cl:nil if the object is no longer published.

Variable: ilu:*debug-uncaught-conditions*

If cl:t, causes a server to invoke the debugger when an unhandled error in user code is encountered, rather than the default action of signalling an exception back to the caller. The default value is cl:nil.

Function: ilu:register-custom-surrogate (CLASS-NAME symbol) (CUSTOM-CLASS clos:standard-class)

Instructs the runtime to create an instance of CUSTOM-CLASS whenever it would normally create a new instance of the ILU object type named by CLASS-NAME, which should be the Common Lisp name for the object type. CUSTOM-CLASS must be a subtype of the class named by CLASS-NAME.

Function: ilu:initialize-ilu

Initializes the ILU module.

Function: ilu:sbh->instance (PUTATIVE-TYPE-NAME symbol) (SBH string) &optional (MOST-SPECIFIC-TYPE-ID simple-string mstid of specified PUTATIVE-TYPE) => ilu:ilu-object

Accepts an ILU string binding handle and Common Lisp type name, and attempts to locally bind an instance of that type with the OID specified in the string binding handle. If no such instance exists locally, a surrogate instance is created and returned. If a true instance exists locally, that instance will be returned.

Function: ilu:lookup (server-id simple-string) (instance-handle simple-string) (PUTATIVE-TYPE-NAME symbol) => (or nil ilu:ilu-object)

This routine will find and return an object with a server ID of server-id and instance handle of instance-handle, if such an object has been registered in the local domain via the ILU simple binding protocol. See the section on "Exporting Objects" for an example.

Method: ilu:ping (DISC ilu:ilu-object) => (or t nil)

Returns cl:t if the true object for DISC exists, and the process serving it can be contacted; cl:nil otherwise.

Go to the previous, next section.