tag | 276b76e7edb5012ce1207cf74bf1ed18a2c54d4d | |
---|---|---|
tagger | The Android Open Source Project <initial-contribution@android.com> | Tue Nov 12 18:38:29 2024 -0800 |
object | 275569d7d4d5c2d9fb2ffa0db98c8e782a175613 |
Android 14.0.0 release 75
commit | 275569d7d4d5c2d9fb2ffa0db98c8e782a175613 | [log] [tgz] |
---|---|---|
author | Andrew Walbran <qwandor@google.com> | Tue Nov 14 14:25:26 2023 +0000 |
committer | Automerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com> | Tue Nov 14 14:25:26 2023 +0000 |
tree | ba19dc65d179988d7e4cba1952262efafa233084 | |
parent | a597fe051d48ef7b525eb0721a028fdcd30eb673 [diff] | |
parent | fd6693dcfed74a21870f74ba37bd22f10004726b [diff] |
Migrate to cargo_embargo. am: 71431fc1f7 am: 77c4925545 am: fd6693dcfe Original change: https://android-review.googlesource.com/c/platform/external/rust/crates/synstructure/+/2828053 Change-Id: I323ca7b4f67dc0ad32e55b3ad575cfee4609adfd Signed-off-by: Automerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com>
NOTE: What follows is an exerpt from the module level documentation. For full details read the docs on docs.rs
This crate provides helper types for matching against enum variants, and extracting bindings to each of the fields in the deriving Struct or Enum in a generic way.
If you are writing a #[derive]
which needs to perform some operation on every field, then you have come to the right place!
WalkFields
pub trait WalkFields: std::any::Any { fn walk_fields(&self, walk: &mut FnMut(&WalkFields)); } impl WalkFields for i32 { fn walk_fields(&self, _walk: &mut FnMut(&WalkFields)) {} }
#[macro_use] extern crate synstructure; #[macro_use] extern crate quote; extern crate proc_macro2; fn walkfields_derive(s: synstructure::Structure) -> proc_macro2::TokenStream { let body = s.each(|bi| quote!{ walk(#bi) }); s.bound_impl(quote!(example_traits::WalkFields), quote!{ fn walk_fields(&self, walk: &mut FnMut(&example_traits::WalkFields)) { match *self { #body } } }) } decl_derive!([WalkFields] => walkfields_derive); /* * Test Case */ fn main() { test_derive! { walkfields_derive { enum A<T> { B(i32, T), C(i32), } } expands to { #[allow(non_upper_case_globals)] const _DERIVE_example_traits_WalkFields_FOR_A: () = { extern crate example_traits; impl<T> example_traits::WalkFields for A<T> where T: example_traits::WalkFields { fn walk_fields(&self, walk: &mut FnMut(&example_traits::WalkFields)) { match *self { A::B(ref __binding_0, ref __binding_1,) => { { walk(__binding_0) } { walk(__binding_1) } } A::C(ref __binding_0,) => { { walk(__binding_0) } } } } } }; } } }
Interest
pub trait Interest { fn interesting(&self) -> bool; } impl Interest for i32 { fn interesting(&self) -> bool { *self > 0 } }
#[macro_use] extern crate synstructure; #[macro_use] extern crate quote; extern crate proc_macro2; fn interest_derive(mut s: synstructure::Structure) -> proc_macro2::TokenStream { let body = s.fold(false, |acc, bi| quote!{ #acc || example_traits::Interest::interesting(#bi) }); s.bound_impl(quote!(example_traits::Interest), quote!{ fn interesting(&self) -> bool { match *self { #body } } }) } decl_derive!([Interest] => interest_derive); /* * Test Case */ fn main() { test_derive!{ interest_derive { enum A<T> { B(i32, T), C(i32), } } expands to { #[allow(non_upper_case_globals)] const _DERIVE_example_traits_Interest_FOR_A: () = { extern crate example_traits; impl<T> example_traits::Interest for A<T> where T: example_traits::Interest { fn interesting(&self) -> bool { match *self { A::B(ref __binding_0, ref __binding_1,) => { false || example_traits::Interest::interesting(__binding_0) || example_traits::Interest::interesting(__binding_1) } A::C(ref __binding_0,) => { false || example_traits::Interest::interesting(__binding_0) } } } } }; } } }
For more example usage, consider investigating the abomonation_derive
crate, which makes use of this crate, and is fairly simple.