#include "expression.hpp" #include #include template using term = boost::proto17::terminal; namespace bp17 = boost::proto17; namespace user { struct number { double value; friend number operator+ (number lhs, number rhs) { return number{lhs.value + rhs.value}; } friend number operator- (number lhs, number rhs) { return number{lhs.value - rhs.value}; } friend number operator* (number lhs, number rhs) { return number{lhs.value * rhs.value}; } friend std::ostream & operator<< (std::ostream & os, number x) // TODO { return os << x.value; } }; number naxpy (number a, number x, number y) { return number{a.value * x.value + y.value + 10.0}; } struct eval_xform { decltype(auto) operator() (term const & expr) { return expr.value(); } }; struct plus_to_minus_xform { template decltype(auto) operator() (bp17::expression const & expr) { return bp17::make_expression(expr.left(), expr.right()); } }; decltype(auto) naxpy_eager_nontemplate_xform ( bp17::expression< bp17::expr_kind::plus, bp17::expression< bp17::expr_kind::multiplies, term, term >, term > const & expr ) { auto a = evaluate(expr.left().left()); auto x = evaluate(expr.left().right()); auto y = evaluate(expr.right()); return bp17::make_terminal(naxpy(a, x, y)); } decltype(auto) naxpy_lazy_nontemplate_xform ( bp17::expression< bp17::expr_kind::plus, bp17::expression< bp17::expr_kind::multiplies, term, term >, term > const & expr ) { auto a = expr.left().left(); auto x = expr.left().right(); auto y = expr.right(); return bp17::make_terminal(naxpy)(a, x, y); } struct naxpy_xform { template decltype(auto) operator() ( bp17::expression< bp17::expr_kind::plus, bp17::expression< bp17::expr_kind::multiplies, Expr1, Expr2 >, Expr3 > const & expr ) { auto a = transform(expr.left().left(), naxpy_xform{}); auto x = transform(expr.left().right(), naxpy_xform{}); auto y = transform(expr.right(), naxpy_xform{}); return bp17::make_terminal(naxpy)(a, x, y); } }; } // TODO: Add a unit test for moved expressions and expressions containing // move-only types or rvalue refs. // TODO: Use move-only type in other tests that test moves as well. TEST(user_expression_transform_3, test_user_expression_transform_3) { term a{{1.0}}; term x{{42.0}}; term y{{3.0}}; { auto expr = a; { user::number result = evaluate(expr); EXPECT_EQ(result.value, 1); } auto transformed_expr = transform(expr, user::eval_xform{}); { EXPECT_EQ(transformed_expr.value, 1); } } { auto expr = x + y; { user::number result = evaluate(expr); EXPECT_EQ(result.value, 45); } auto transformed_expr = transform(expr, user::plus_to_minus_xform{}); { user::number result = evaluate(transformed_expr); EXPECT_EQ(result.value, 39); } } { auto expr = a * x + y; { user::number result = evaluate(expr); EXPECT_EQ(result.value, 45); } auto transformed_expr = transform(expr, user::naxpy_eager_nontemplate_xform); { user::number result = evaluate(transformed_expr); EXPECT_EQ(result.value, 55); } } { auto expr = a + (a * x + y); { user::number result = evaluate(expr); EXPECT_EQ(result.value, 46); } auto transformed_expr = transform(expr, user::naxpy_eager_nontemplate_xform); { user::number result = evaluate(transformed_expr); EXPECT_EQ(result.value, 56); } } { auto expr = a * x + y; { user::number result = evaluate(expr); EXPECT_EQ(result.value, 45); } auto transformed_expr = transform(expr, user::naxpy_lazy_nontemplate_xform); { user::number result = evaluate(transformed_expr); EXPECT_EQ(result.value, 55); } } { auto expr = a + (a * x + y); { user::number result = evaluate(expr); EXPECT_EQ(result.value, 46); } auto transformed_expr = transform(expr, user::naxpy_lazy_nontemplate_xform); { user::number result = evaluate(transformed_expr); EXPECT_EQ(result.value, 56); } } { auto expr = (a * x + y) * (a * x + y) + (a * x + y); { user::number result = evaluate(expr); EXPECT_EQ(result.value, 45 * 45 + 45); } auto transformed_expr = transform(expr, user::naxpy_xform{}); { user::number result = evaluate(transformed_expr); EXPECT_EQ(result.value, 55 * 55 + 55 + 10); } } }