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

Fix double std::move on objects in 'lenses::getset' on set call #165

Open
wants to merge 7 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
158 changes: 148 additions & 10 deletions lager/lenses.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,15 @@
namespace lager {
namespace detail {

template <typename T>
struct should_move :
std::integral_constant<bool,
!std::is_array_v<T> &&
!std::is_lvalue_reference_v<T>> {};

template <typename T>
constexpr bool should_move_v = should_move<T>::value;

template <typename T>
struct const_functor;

Expand All @@ -42,6 +51,15 @@ struct const_functor
}
};

template <typename T>
struct is_const_functor : public std::false_type {};

template <typename T>
struct is_const_functor<const_functor<T>> : public std::true_type {};

template <typename T>
constexpr bool is_const_functor_v = is_const_functor<T>::value;

template <typename T>
struct identity_functor;

Expand All @@ -59,10 +77,135 @@ struct identity_functor
template <typename Fn>
auto operator()(Fn&& f) &&
{
return make_identity_functor(
std::forward<Fn>(f)(std::forward<T>(value)));

if constexpr (!should_move_v<T>) {
return make_identity_functor(
std::forward<Fn>(f)(value));
} else {
return make_identity_functor(
std::forward<Fn>(f)(std::move(value)));
}
}
};

template <typename Part>
struct identity_functor_skip_first
{
template <typename T>
auto operator() (T&&) const & {
return make_identity_functor(part);
}

template <typename T>
auto operator() (T&&) && {
if constexpr (!should_move_v<Part>) {
return make_identity_functor(part);
} else {
return make_identity_functor(std::move(part));
}
}

Part part;
};

template <typename T>
auto make_identity_functor_skip_first(T&& x) -> identity_functor_skip_first<T>
{
return {std::forward<T>(x)};
}

template <typename F, typename Getter, typename Setter>
struct getset_t
{
template <typename Whole>
auto operator() (Whole &&w) {
return functorImpl(*this, LAGER_FWD(w));
}

template <typename Whole>
auto operator() (Whole &&w) const {
return functorImpl(*this, LAGER_FWD(w));
}

F f;
Getter getter;
Setter setter;

private:
/**
* Deduce proper constness of 'this' via the functorImpl() call
*/
template <typename Self, typename Whole>
static auto functorImpl(Self &&self, Whole &&w) {
if constexpr (is_const_functor_v<decltype(self.f(self.getter(LAGER_FWD(w))))>) {
/**
* We don't have a setter here, so it is safe to
* jus pass `w` as an rvalue.
*
* We also know that we are calling the const_functor,
* and it discards the passed argument, so just pass a
* noop to it.
*
* This branch is taken when viewing through the lens.
*/
return self.f(self.getter(LAGER_FWD(w))) // pass `w` into the getter as an rvalue!
(zug::noop);
} else {
/**
* Here we have both, getter and setter, so we pass
* `w` to getter as an lvalue, and to setter as an rvalue.
* The setter has a chance to reuse the resources of
* the passed value.
*
* This branch is taken on all the levels of setting the
* value through except of the tompost level.
*/
return self.f(self.getter(w)) // pass `w` into the getter as an lvalue!
([&](auto&& x) {
return self.setter(LAGER_FWD(w), LAGER_FWD(x));
});
}
}
};

/**
* This specialization is called when a set() method is called over
* the lens. In such a case we can skip calling the getter branch
* of the lens.
*
* This branch is taken on the topmost level of setting the value
* through the lens.
*/
template <typename T, typename Getter, typename Setter>
struct getset_t<identity_functor_skip_first<T>, Getter, Setter>
{
template <typename Part>
auto operator() (Part &&p) {
return std::move(f)(zug::noop)
([&](auto&& x) {
return setter(LAGER_FWD(p), LAGER_FWD(x));
});
}

template <typename Part>
auto operator() (Part &&p) const {
return f(zug::noop)
([&](auto&& x) {
return setter(LAGER_FWD(p), LAGER_FWD(x));
});
}

identity_functor_skip_first<T> &&f;
Getter getter;
Setter setter;
};

template <typename F, typename Getter, typename Setter>
auto make_getset_t(F &&f, const Getter &getter, const Setter &setter)
{
return getset_t<F, Getter, Setter>{std::forward<F>(f), getter, setter};
}

} // namespace detail

//! @defgroup lenses-api
Expand All @@ -80,7 +223,7 @@ decltype(auto) view(LensT&& lens, T&& x)
template <typename LensT, typename T, typename U>
decltype(auto) set(LensT&& lens, T&& x, U&& v)
{
return lens([&v](auto&&) { return detail::make_identity_functor(v); })(
return lens(detail::make_identity_functor_skip_first(std::forward<decltype(v)>(v))) (
std::forward<T>(x))
.value;
}
Expand All @@ -103,15 +246,10 @@ namespace lenses {
//! @{

template <typename Getter, typename Setter>
auto getset(Getter&& getter, Setter&& setter)
auto getset(const Getter& getter, const Setter& setter)
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just a note: the change to const Getter& getter is not a regression, because they were previously bound to a const reference inside a const (immutable) lambda.

{
return zug::comp([=](auto&& f) {
return [&, f = LAGER_FWD(f)](auto&& p) {
return f(getter(std::forward<decltype(p)>(p)))([&](auto&& x) {
return setter(std::forward<decltype(p)>(p),
std::forward<decltype(x)>(x));
});
};
return detail::make_getset_t(LAGER_FWD(f), getter, setter);
});
}

Expand Down
18 changes: 9 additions & 9 deletions lager/lenses/attr.hpp
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
#pragma once

#include <lager/lenses.hpp>
#include <lager/util.hpp>
#include <zug/compose.hpp>

Expand All @@ -17,15 +18,14 @@ namespace lenses {
template <typename Member>
auto attr(Member member)
{
return zug::comp([member](auto&& f) {
return [&, f = LAGER_FWD(f)](auto&& p) {
return f(LAGER_FWD(p).*member)([&](auto&& x) {
auto r = LAGER_FWD(p);
r.*member = LAGER_FWD(x);
return r;
});
};
});
return getset(
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi, @arximboldi!

I had to rewrite attr into (the fixed version of) getset, because otherwise it also caused double-move when nested and passed with a prvalue. Basically, a code like that is enough to crash it:

auto birthday_lens = attr(&debug_person::birthday);
auto person_lens = attr(&debug_freelancer::person);
auto freelancer_birthday_lens = person_lens | birthday_lens;

auto birthday =
    set(freelancer_birthday_lens,
        debug_freelancer(freelancer_copy_info, person_copy_info, birthday_copy_info),
        debug_yearday(birthday_copy_info));

I have a weird feeling like all the lenses in lager have to be rewritten like that... :( Not only they do double-move, capturing functors into immutable lambdas prevents proper forwarding of the identity functors in view and set.

[=](auto &&p) -> decltype(auto) {
return LAGER_FWD(p).*member;
},
[=](auto p, auto &&x) {
p.*member = LAGER_FWD(x);
return p;
});
}

//! @}
Expand Down
2 changes: 1 addition & 1 deletion lager/lenses/unbox.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ namespace lenses {
* `Lens<box<T>, T>`
*/
ZUG_INLINE_CONSTEXPR auto unbox = zug::comp([](auto&& f) {
return [f](auto&& p) {
return [f = LAGER_FWD(f)](auto&& p) {
return f(LAGER_FWD(p).get())(
[&](auto&& x) { return std::decay_t<decltype(p)>{LAGER_FWD(x)}; });
};
Expand Down
Loading