Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow macros to expand to multiple items #4375

Closed
bstrie opened this issue Jan 8, 2013 · 13 comments
Closed

Allow macros to expand to multiple items #4375

bstrie opened this issue Jan 8, 2013 · 13 comments
Labels
A-syntaxext Area: Syntax extensions C-enhancement Category: An issue proposing an enhancement or a PR with one.

Comments

@bstrie
Copy link
Contributor

bstrie commented Jan 8, 2013

https://gist.github.com/4480389

The macro here is expanding to only the first statement in its definition. According to pauls, this is because "the mac system is currently only able to expand a macro to a single item", and that this should be fixed. Make it so that a macro instead expands to all the statements in its definition.

@paulstansifer
Copy link
Contributor

Fixing this mainly will entail editing fold.rs so that one item can fold to multiple items. The odd thing is that there are a number of places in the AST where exactly one item is accepted; I haven't investigated what's up with them.

@brendanzab
Copy link
Member

This is sorely missed. Here is a use case where I'd love to have it (more on the way):

pub trait Cast {
    static fn from<T:CastTo<self>>(x: T) -> self;
}

pub trait CastTo<T> {
    fn cast(&self) -> T;
}

macro_rules ! impl_casts {
    {$from:ty -> $($to:ty),+} => {
        pub impl $from: Cast {
            #[inline(always)] static fn from<T:CastTo<$from>>(x: T) -> $from { x.cast() }
        }

        $(
            pub impl $from: CastTo<$to> {
                #[inline(always)] fn cast(&self) -> $to { (*self) as $to }
            }
        )+
    }
}

impl_casts! { u8    -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { u16   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { u32   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { u64   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { uint  -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { i8    -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { i16   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { i32   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { i64   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { int   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { f32   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { f64   -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }
impl_casts! { float -> u8, u16, u32, u64, uint, i8, i16, i32, i64, int, f32, f64, float }

The alternative (as you can imagine) is frighteningly verbose.

@jbclements
Copy link
Contributor

agreed that expanding into multiple items would be very important/useful. Another alternative would be to have an explicit "item-sequence" AST node, that gets flattened at some point during compilation.

@sanxiyn
Copy link
Member

sanxiyn commented Apr 10, 2013

Jeaye suggests using mod to group multiple items as a workaround. Isn't mod exactly "item-sequence" AST node @jbclements wanted?

@bstrie
Copy link
Contributor Author

bstrie commented Apr 22, 2013

Nominating this for a milestone, as per bjz's plea.

@dobkeratops
Copy link

i was looking for this feature too whilst writing something to auto generate a struct alongside some sort of serialization assist. (initially I tried to declare a struct plus a default 'to_str()' that formated it in a JSON-esque manner).

bors added a commit that referenced this issue May 25, 2013
…rson

Changes the int/uint modules to all use macros instead of using the `merge` attribute. It would be nice to have #4375 resolved as well for this, but that can probably come at a later date.

Closes #4219.
@jedestep
Copy link
Contributor

jedestep commented Aug 2, 2013

I've been looking into this; @sanxiyn 's suggestion seems to be the simplest way to get this done. Still work left to do but generally the approach is to have MRAny hold a vector of items instead of just a single one and propagate this change appropriately.

The goofy hack alternative would be to literally inject a module into the macro, __macro__<macroname> or something, and pub use it.

sfackler added a commit to sfackler/rust-postgres that referenced this issue Sep 15, 2013
The enum is stuck inside of an extra module due to rust-lang/rust#4375.
@nejucomo
Copy link

Hello. I've just started exploring @jedestep 's comment by changing MRAny to MRAny(@[@AnyMacro]). I figure that definition will be "easiest" to get working, then later I can see if it makes more sense to have @[AnyMacro] or ~ pointers instead.

Right now I'm waiting for make check to expose all the type mismatches. I'm new to rust, and I've only spent about 20 minutes reading over ast.rs and ext/base.rs so we'll see how this goes. ;-)

@nejucomo
Copy link

nejucomo commented Oct 3, 2013

I haven't had time yet to explore the MRAny redefinition. I thought I'd point out a relevant pull request, #9673, which disallows multi-item expansions explicitly, instead of implicitly dropping after the first.

@alexcrichton
Copy link
Member

I believe that the purpose of #9673 is to stop silently ignoring extra items, but it'd still be awesome to be able to expand to multiple items (meaning that the safeguard in #9673 would be removed).

@bors bors closed this as completed in c144752 Nov 27, 2013
@ntrel
Copy link

ntrel commented Aug 30, 2014

c144752 may fix some cases, e.g.:
https://github.com/rust-lang/rust/blob/master/src/test/run-pass/macro-multiple-items.rs

But I think this issue is not fully resolved, at least with 0.11:

macro_rules! do_bar(
    () => (
        let x = 1u;
        if x == 1 {return}
    )
)

pub fn main() {
    // error: macro expansion ignores token `if` and any following
    do_bar!();
} 

@huonw
Copy link
Member

huonw commented Aug 30, 2014

@ntrel they are statements/expressions, not items (in this context an item is a top-level thing like fn, static or impl), #10681 covers that specifically.

You can work around it in most cases by wrapping the whole contents a pair of {} (since a { ... } block is a single expression):

macro_rules! do_bar(
    () => ({ // extra { here.
        let x = 1u;
        if x == 1 {return}
    })
)

@ntrel
Copy link

ntrel commented Sep 1, 2014

@huonw Great, thanks.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-syntaxext Area: Syntax extensions C-enhancement Category: An issue proposing an enhancement or a PR with one.
Projects
None yet
Development

No branches or pull requests