Abstracting over mutability in Rust macros

Tags:

Macros can help to avoid bugs in repetitive code. It would be nice if we could reuse the same macro for fn(&self) and fn(&mut self) alike, or similar cases. In this note I will show how to do this.

Let’s say we have an enumeration over many types that present an identical interface:

1
2
3
4
5
6
pub enum Socket {
    Udp(UdpSocket),
    Tcp(TcpSocket),
    #[doc(hidden)]
    __Nonexhaustive
}

Naively, code for dispatching methods to individual sockets would look something like this:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
impl Socket {
    pub fn process(&mut self, timestamp: u64, ip_repr: &IpRepr,
                   payload: &[u8]) -> Result<(), Error> {
        match self {
            &mut Socket::Udp(ref mut socket) =>
                socket.process(timestamp, ip_repr, payload),
            &mut Socket::Tcp(ref mut socket) =>
                socket.process(timestamp, ip_repr, payload),
            &mut Socket::__Nonexhaustive => unreachable!()
        }
    }

    pub fn dispatch<F, R>(&mut self, timestamp: u64, emit: &mut F) -> Result<R, Error>
            where F: FnMut(&IpRepr, &IpPayload) -> Result<R, Error> {
        match self {
            &mut Socket::Udp(ref mut socket) =>
                socket.dispatch(timestamp, emit),
            &mut Socket::Tcp(ref mut socket) =>
                socket.dispatch(timestamp, emit),
            &mut Socket::__Nonexhaustive => unreachable!()
        }
    }
}

We could simplify it with a macro:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
macro_rules! dispatch_socket {
    ($self_:expr, |$socket:ident| $code:expr) => ({
        match $self_ {
            &mut Socket::Udp(ref mut $socket) => $code,
            &mut Socket::Tcp(ref mut $socket) => $code,
            &mut Socket::__Nonexhaustive => unreachable!()
        }
    })
}

impl Socket {
    pub fn process(&mut self, timestamp: u64, ip_repr: &IpRepr,
                   payload: &[u8]) -> Result<(), Error> {
        dispatch_socket!(self, |socket| socket.process(timestamp, ip_repr, payload))
    }

    pub fn dispatch<F, R>(&mut self, timestamp: u64, emit: &mut F) -> Result<R, Error>
            where F: FnMut(&IpRepr, &IpPayload) -> Result<R, Error> {
        dispatch_socket!(self, |socket| socket.process(timestamp, ip_repr, payload))
    }
}

However, what do we do to implement a method that accepts a &self? It seems inelegant to duplicate the macro when the entire point of this exercise is reduction in duplication.

Let’s try to parameterize a macro over the mut qualifier. It’s not a valid syntactic entity itself, so we can only represent it as a token tree:

1
2
3
4
5
6
7
8
9
macro_rules! dispatch_socket {
    ($self_:expr, |$socket:ident $mut_:tt| $code:expr) => ({
        match $self_ {
            &$mut_ Socket::Udp(ref $mut_ $socket) => $code,
            &$mut_ Socket::Tcp(ref $mut_ $socket) => $code,
            &$mut_ Socket::__Nonexhaustive => unreachable!()
        }
    })
}

Unfortunately this doesn’t work for the non-mut case as $mut_:tt matcher will eat the delimiting |. We can’t surround it with delimiters, like [ $mut_:tt ] either, for the same reason. We can, however, make it match zero or more muts!

1
2
3
4
5
6
7
8
9
macro_rules! dispatch_socket {
    ($self_:expr, |$socket:ident [$( $mut_:tt )*]| $code:expr) => ({
        match $self_ {
            &$( $mut_ )* Socket::Udp(ref $( $mut_ )* $socket) => $code,
            &$( $mut_ )* Socket::Tcp(ref $( $mut_ )* $socket) => $code,
            &$( $mut_ )* Socket::__Nonexhaustive => unreachable!()
        }
    })
}

I wish Rust added first-class support for a zero-or-one matcher in macros by example, but meanwhile, this works too.


Want to discuss this note? Drop me a letter.