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:
(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:
(progn
(defmethod read-value ((#:g1618 (eql 'iso-8859-1-string)) in &key length)
(let ((string (make-string length)))
(setf (char string i) (code-char (read-byte in))))
string))
(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:
(progn
(defmethod read-value ((#:g161887 (eql 'u1)) #:g161888 &key)
(read-value 'unsigned-integer #:g161888 :bytes 1))
(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.