Expand description
§Implementation approach - how does this work?
You do not need to understand this in order to use derive-deftly.
Also, you should not rely on the details here. They don’t form part of the public interface.
§Introduction
It is widely understood that proc macro invocations cannot communicate with each other. (Some people have tried sneaking round the back with disk files etc. but this can break due to incremental and concurrent compilation.)
But, a proc macro can define a macro_rules
macro.
Then, later proc macros can invoke that macro.
The expansion can even invoke further macros.
In this way, a proc macro invocation can communicate with
subsequent proc macro invocations.
(This trick is also used by
ambassador
.)
There is a further complication.
One macro X cannot peer into and dismantle
the expansion of another macro Y.
(This is necessary for soundness, when macros can generate unsafe
.)
So we must sometimes define macros that apply other macros
(whose name is supplied as an argument)
to what is conceptually the first macro’s output.
§Implementation approach - reusable template macros
§Expansion chaining
When a #[derive(Deftly)]
call site invokes multiple templates,
we don’t emit separate macro calls for each template.
Instead, we generate a macro call for the first template,
and pass it the names of subsequent templates.
Each template expansion is responsible for invoking the next.
This allows output from the expansion to be
threaded through the chain,
collecting additional information as we go,
and eventually analysed after all the expansions are done.
(This is done by puttingderive_deftly_engine
as the final entry
in the list of macros to chain to.)
Currently, this data collection is used for detecting unused meta attributes.
This does mean that specifying a completely unknown template (for example, one that’s misspelled or not in scope) will prevent error reports from subsequent templates, which is a shame.
§Overall input syntax for derive_deftly_engine!
and templates
Because of expansion chaining, calls to the engine, and to templates, overlap.
macro! {
{ DRIVER.. }
[ 1 0 AOPTIONS.. ] // not present for d_d_dengine expanding ad-hoc
// replaced with just "." at end of chain
// always present, but always immediately ignored:
( .... )
// these parts passed to d_d_engine only, and not in last call:
{ TEMPLATE.. }
( CRATE; [TOPTIONS..]
TEMPLATE_NAME /* omitted if not available */;
.... )
// always present:
// after here is simply passed through by templates:
[
// usually one or more of these; none at end, or ad-hoc
CHAIN_TO ( CHAIN_PASS_AFTER_DRIVER .... )
..
]
[ACCUM..]
....
}
(We use the notation ....
for room we are leaving
for future expansion;
these are currently empty.)
§1. define_derive_deftly!
macro for defining a reuseable template
Implemented in define.rs::define_derive_deftly_func_macro
.
When used like this
define_derive_deftly! {
MyMacro TOPTIONS..:
TEMPLATE..
}
Expands to
macro_rules! derive_deftly_template_Template { {
{ $($driver:tt)* }
[ $($aoptions:tt)* ]
( $($future:tt)* )
$($tpassthrough:tt)*
} => {
derive_deftly_engine! {
{ $($driver)* }
[ $(aoptions)* ]
( )
{ TEMPLATE.. }
( $crate; [TOPTIONS..] Template; )
$($tpassthrough)*
}
} }
Except, every $
in the TEMPLATE is replaced with $orig_dollar
.
This is because a macro_rules!
template
is not capable of generating a
literal in the expansion $
.
(With the still-unstable
decl_macro
feature, $$
does this.)
macro_rules!
simply passes through $orig_dollar
(as it does with all unrecognised variables), and
the template expansion engine treats $orig_dollar
as just a single $
.
(The extra ( )
parts after the driver and template
include space for future expansion.)
§2. #[derive_deftly(Template)]
, implemented in #[derive(Deftly)]
Template use is implemented in derive.rs::derive_deftly
.
This
#[derive(Deftly)]
#[derive_deftly(Template[AOPTIONS..], Template2)]
pub struct StructName { .. }
Generates:
derive_deftly_template_Template! {
{ #[derive_deftly(..)] struct StructName { .. } }
[ 1 0 AOPTIONS ]
( )
[
derive_deftly_template_Template2 ( [1 0 AOPTIONS2] () )
derive_deftly_engine ( . () )
]
[] // (nothing accumulated yet)
}
§3. Actual expansion
The first call to derive_deftly_template_Template!
is expanded according to the macro_rules!
definition,
resulting in a call to derive_deftly_engine
:
derive_deftly_engine! {
{ #[derive_deftly(..)] pub struct StructName { .. } }
[ 1 0 AOPTIONS ]
( )
{ TEMPLATE.. }
( $crate; [TOPTIONS..] Template; )
[
derive_deftly_template_Template2 ( [1 0 AOPTIONS2] () )
derive_deftly_engine ( . () )
]
[] // ACCUM
}
Implemented in engine.rs::derive_deftly_engine_func_macro
,
this performs the actual template expansion.
§4. Chaining to the next macro
derive_deftly_engine!
then invokes the next template.
In the above example, it outputs:
derive_deftly_template_Template2! {
{ #[derive_deftly(..)] pub struct StructName { .. } }
[ 1 0 AOPTIONS2 ]
( ) // threaded through from the call site
[ derive_deftly_engine ( . () ) ]
[_name Template _meta_used [..] _meta_recog [..]] // ACCUM
}
ACCUM
accumulates information from successive expansions,
and is actually parsed only in the next step.
§Used meta attribute syntax
_meta_used
introduces the meta attributes from the driver,
which were used by this template,
in the following syntax:
::VARIANT // entries until next :: are for this variant
.field // entries until next :: or . are for this field
( // each (..) corresponds to one #[deftly()], and matches its tree
somemeta(
value_used =, // value was used
bool_tested ?, // boolean was tested
, // skip a node (subtree) not used at all
.. // trailing skipped notes are omitted
)
) // empty trailing groups are omitted
So for example this
struct S {
#[adhoc(foo(bar, baz))]
#[adhoc(wombat, zoo = "Berlin")]
f: T,
}
might correspond to this
_meta [
// nothing (don't need field name even)
.f () (, zoo?)
.f () (wombat?, zoo=)
.f (foo=(bar?, baz?)) (wombat?, zoo?)
]
This representation helps minimise the amount of text which needs to be passed through all the macro chains, when only a few attributes are used by each template, while still being unambiguous and detecting desynchronisation.
Instead of a [..]
list, _meta_used
can also be *
,
meaning that no checking should be done.
§Recognised meta attribute syntax
_meta_recog
introduces the meta attributes which
might conceivably be recognised by this template.
This is determined statically, by scanning the template.
The information is used only if there are unused attributes, to assist with producing good error messages.
_meta_recog
takes a [ ]
list containing items like fmeta(foo(bar))
,
or ``fmeta(foo(bar))?` when recognised only as a boolean.
§5. Reporting unused meta attributes
At the end of the list of chained templates,
is (the driver’s version of) derive_deftly_engine!
.
So after all the templates have been processed,
instead of invoking the next template,
we return directly to the engine:
derive_deftly_engine! {
{ #[derive_deftly(..)] pub struct StructName { .. } }
.
( )
[] // end of the chain
[_name Template _meta_used [..] _meta_recog [..] _name Template2 ..]
}
The accumulated parts are all of the form: KEYWORD DATA
;
_KEYWORD DATA
for a part which can be safely ignored,
if unrecognised.
DATA
is a single tt.
Each template’s parts start with a _name
part.
There can also be error
parts which
appear when the template couldn’t be parsed
(which is used to suppress “unrecognised attribute” errors).
The actually supplied attributes are compared with the union of the used attributes, and errors are reported for unused ones.
§Implementation approach - ad-hoc macro applications
§1. #[derive(Deftly)]
feature for saving struct definitions
Also implemented in derive.rs::derive_deftly
.
When applied to (e.g.) pub struct StructName
,
with #[derive_deftly_adhoc]
specified
generates this
macro_rules! derive_deftly_driver_StructName { {
{ $($template:tt)* }
{ ($orig_dollar:tt) $(future:tt)* }
$($dpassthrough:tt)*
} => {
derive_deftly_engine!{
{ pub struct StructName { /* original struct definition */ } }
/* no AOPTIONS since there's no derive() application */
( )
{ $($template)* }
$($dpassthrough)*
}
} }
(Again, the extra { }
parts after the driver and template
include space for future expansion.)
In the pub struct
part every $
is replaced with $orig_dollar
,
to use the $
passed in at the invocation site.
(This is only relevant if the driver contains $
somehow,
for example in helper attributes.)
§2. derive_deftly_adhoc!
function-like proc macro for applying to a template
Implemented in adhoc.rs::derive_deftly_adhoc
.
When applied like this
derive_deftly_adhoc!{
StructName TOPTIONS..:
TEMPLATE..
}
Expands to
derive_deftly_driver_StructName! {
{ TEMPLATE.. }
{ ($) }
( crate; [TOPTIONS..] /*no template name*/; )
[]
[_meta_used *]
}
The literal $
is there to work around a limitation in macro_rules!
,
see above.
§3. Function-like proc macro to do the actual expansion
The result of expanding the above is this:
derive_deftly_engine!{
{ pub struct StructName { /* original struct definition */ } }
( )
{ TEMPLATE.. }
( crate; [TOPTIONS..] /*no template name*/; )
[]
[_meta_used *]
}
derive_deftly_engine
parses pub struct StructName
,
and implements a bespoke template expander,
whose template syntax resembles the expansion syntax from macro_rules
.
crate
is just used as the expansion for ${crate}
.
(For an ad-hoc template, the local crate is correct.)