#operator #overload #automatic #implementation #macro

macro auto-impl-ops

semi-automatic implementation proc-macro for binary operations

5 releases

0.2.1 Mar 2, 2023
0.2.0 Feb 27, 2023
0.1.2 Feb 21, 2023
0.1.1 Nov 27, 2022
0.1.0 Jul 16, 2022

#9 in #overload

Download history 51/week @ 2024-01-01 63/week @ 2024-01-08 56/week @ 2024-01-15 55/week @ 2024-01-22 60/week @ 2024-01-29 59/week @ 2024-02-05 60/week @ 2024-02-12 129/week @ 2024-02-19 111/week @ 2024-02-26 102/week @ 2024-03-04 128/week @ 2024-03-11 103/week @ 2024-03-18 109/week @ 2024-03-25 139/week @ 2024-04-01 86/week @ 2024-04-08 79/week @ 2024-04-15

426 downloads per month
Used in 4 crates (2 directly)

AGPL-3.0-or-later

43KB
1K SLoC

semi-automatic implementation proc-macro for binary operations

#[auto_ops] make implementation for T += U, T + U, T + &U, &T + U, &T + &U from implementation for T += &U.

supported list (@ is +, -, *, /, %, &, |, ^, << or >>.)

  • T @= &U => T @= U, &T @ &U, &T @ U, T @ &U, T @ U
  • T @= U => T @= &U, &T @ &U, &T @ U, T @ &U, T @ U
  • &T @ &U => T @= &U, T @= U, &T @ U, T @ &U, T @ U
  • &T @ U => T @= &U, T @= U, &T @ &U, T @ &U, T @ U
  • T @ &U => T @= &U, T @= U, &T @ &U, &T @ U, T @ U
  • T @ U => T @= &U, T @= U, &T @ U, T @ &U, T @ U

Example

use std::ops::*;
# 
# #[derive(Clone, Default)]
# struct A<T>(T);

#[auto_impl_ops::auto_ops]
impl<M> AddAssign<&A<M>> for A<M>
where
    for<'x> &'x M: Add<Output = M>,
{
    fn add_assign(&mut self, other: &Self) {
        self.0 = &self.0 + &other.0;
    }
}

Above code is expanded into below code. For more examples see examples/a.rs.

use std::ops::*;
# 
# #[derive(Clone, Default)]
# struct A<T>(T);

impl<M> AddAssign<&A<M>> for A<M>
where
    for<'x> &'x M: Add<Output = M>,
{
    fn add_assign(&mut self, other: &Self) {
        self.0 = &self.0 + &other.0;
    }
}
#[allow(clippy::extra_unused_lifetimes)]
impl<M> AddAssign<A<M>> for A<M>
where
    for<'x> &'x M: Add<Output = M>,
{
    fn add_assign(&mut self, rhs: A<M>) {
        let rhs = &rhs;
        self.add_assign(rhs);
    }
}
impl<M> Add<&A<M>> for &A<M>
where
    for<'x> &'x M: Add<Output = M>,
    A<M>: Clone,
{
    type Output = A<M>;
    fn add(self, rhs: &A<M>) -> Self::Output {
        let mut lhs = self.clone();
        lhs.add_assign(rhs);
        lhs
    }
}
impl<M> Add<A<M>> for &A<M>
where
    for<'x> &'x M: Add<Output = M>,
    A<M>: Clone,
{
    type Output = A<M>;
    fn add(self, rhs: A<M>) -> Self::Output {
        let mut lhs = self.clone();
        let rhs = &rhs;
        lhs.add_assign(rhs);
        lhs
    }
}
impl<M> Add<&A<M>> for A<M>
where
    for<'x> &'x M: Add<Output = M>,
{
    type Output = A<M>;
    fn add(self, rhs: &A<M>) -> Self::Output {
        let mut lhs = self;
        lhs.add_assign(rhs);
        lhs
    }
}
#[allow(clippy::extra_unused_lifetimes)]
impl<M> Add<A<M>> for A<M>
where
    for<'x> &'x M: Add<Output = M>,
{
    type Output = A<M>;
    fn add(self, rhs: A<M>) -> Self::Output {
        let mut lhs = self;
        let rhs = &rhs;
        lhs.add_assign(rhs);
        lhs
    }
}

License

auto-impl-ops is AGPL-3.0-or-later. The code generated by this proc-macro is exception of AGPL. You can choose its license as you like.

Dependencies

~1.5MB
~35K SLoC