However, rather than having to document how to write a suitable read-value/write-value pair, you can provide a macro to do it automatically. This also has the advantage of making the abstraction created by define-binary-class less leaky. Currently, define-binary-class depends on having methods on read-value and write-value defined in a particular way, but that’s really just an implementation detail. By defining a macro that generates the read-value and write-value methods for primitive types, you hide those details behind an abstraction you control. If you decide later to change the implementation of define-binary-class, you can change your primitive-type-defining macro to meet the new requirements without requiring any changes to code that uses the binary data library.

    So you should define one last macro, define-binary-type, that will generate read-value and write-value methods for reading values represented by instances of existing classes, rather than by classes defined with define-binary-class.

    For a concrete example, consider a type used in the id3-tag class, a fixed-length string encoded in ISO-8859-1 characters. I’ll assume, as I did earlier, that the native character encoding of your Lisp is ISO-8859-1 or a superset, so you can use **CODE-CHAR** and **CHAR-CODE** to translate bytes to characters and back.

    As always, your goal is to write a macro that allows you to express only the essential information needed to generate the required code. In this case, there are four pieces of essential information: the name of the type, iso-8859-1-string; the **&key** parameters that should be accepted by the read-value and write-value methods, in this case; the code for reading from a stream; and the code for writing to a stream. Here’s an expression that contains those four pieces of information:

    1. (defmacro define-binary-type (name (&rest args) &body spec) ...

    then within the macro the parameter spec will be a list containing the reader and writer definitions. You can then use **ASSOC** to extract the elements of spec using the tags :reader and :writer and then use **DESTRUCTURING-BIND** to take apart the **REST** of each element.10

    From there it’s just a matter of interpolating the extracted values into the backquoted templates of the read-value and write-value methods.

    Note how the backquoted templates are nested: the outermost template starts with the backquoted **PROGN** form. That template consists of the symbol **PROGN** and two comma-unquoted **DESTRUCTURING-BIND** expressions. Thus, the outer template is filled in by evaluating the **DESTRUCTURING-BIND** expressions and interpolating their values. Each **DESTRUCTURING-BIND** expression in turn contains another backquoted template, which is used to generate one of the method definitions to be interpolated in the outer template.

    With this macro defined, the define-binary-type form given previously expands to this code:

    1. (progn
    2. (defmethod read-value ((#:g1618 (eql 'iso-8859-1-string)) in &key length)
    3. (let ((string (make-string length)))
    4. (setf (char string i) (code-char (read-byte in))))
    5. string))
    6. (dotimes (i length)

    ID3 tags, like many other binary formats, use lots of primitive types that are minor variations on a theme, such as unsigned integers in one-, two-, three-, and four-byte varieties. You could certainly define each of those types with define-binary-type as it stands. Or you could factor out the common algorithm for reading and writing n-byte unsigned integers into helper functions.

    But suppose you had already defined a binary type, unsigned-integer, that accepts a :bytes parameter to specify how many bytes to read and write. Using that type, you could specify a slot representing a one-byte unsigned integer with a type specifier of (unsigned-integer :bytes 1). But if a particular binary format specifies lots of slots of that type, it’d be nice to be able to easily define a new type—say, u1--that means the same thing. As it turns out, it’s easy to change define-binary-type to support two forms, a long form consisting of a :reader and :writer pair and a short form that defines a new binary type in terms of an existing type. Using a short form define-binary-type, you can define u1 like this:

    which will expand to this:

    1. (progn
    2. (defmethod read-value ((#:g161887 (eql 'u1)) #:g161888 &key)
    3. (read-value 'unsigned-integer #:g161888 :bytes 1))
    4. (defmethod write-value ((#:g161887 (eql 'u1)) #:g161888 #:g161889 &key)

    To support both long- and short-form define-binary-type calls, you need to differentiate based on the value of the spec argument. If spec is two items long, it represents a long-form call, and the two items should be the :reader and :writer specifications, which you extract as before. On the other hand, if it’s only one item long, the one item should be a type specifier, which needs to be parsed differently. You can use **ECASE** to switch on the **LENGTH** of spec and then parse and generate an appropriate expansion for either the long form or the short form.