-
Notifications
You must be signed in to change notification settings - Fork 3k
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
Reduce Event.h and EventQueue.h using C++11 #10895
Conversation
Variadic templates can reduce Event.h from 4,100 lines to 300, and EventQueue.h from 3,400 to 1,000, so 6,000 lines saved total. End result isn't totally variadic, as we still need specialisations for storing 0-5 values in contexts, but that specialisation is now in exactly one place. Only change other from switching to variadic templates is using delegating constructors instead of the `new (this)` trick. That trick is still used in the assignment operator. Minor documentation correction. It's possible that the separate simplified variadic Doxygen version not be needed now, but I've left it.
Second commit makes a change to template deduction which I think is required to make all variadic cases work, but it will be a subtle behaviour change. Full details in commit message - review especially from @geky appreciated. |
Previous commit just replaced instances of "class B0, class B1, class C0, class C1" with "class... BoundArgs, class... ContextArgs". This loses the requirement that the numbers must match. Now, the original code was also inconsistent as to whether it used separate types for the target function and the call input parameters. Some forms just used B0, B1 as parameters rather than separate C0, C1. I believe the separate parameters would have been primarily to avoid template deduction confusion - eg if int was supplied to a B0 parameter but the function took char as B0, there would be an ambiguity. But the fix didn't seem to be fully applied. Rewritten all templates parameterising on function pointer type and input arguments so that they use `type_identity_t<BoundArgTs>...` as input parameters to match the target function. This has the subtle effect that any conversion happens at invocation, before storing to the context, rather than when the context calls the target.
Hmm, still have a problem with the template deduction - I see now that if you don't let the passed arguments be freely-typed, any need for conversion makes the But then if they're freely typed, I don't see how I can use the variadic form for
There's nothing to associate the number of parameters passed to how many bound args we want. It ends up putting nothing in BoundArgTs and everything in ArgTs. It may be necessary to do these binding calls non-variadically. |
template <typename R, typename B0, typename C0, typename A0, typename A1, typename A2> | ||
Event<void(A0, A1, A2)> event(mbed::Callback<R(B0, A0, A1, A2)> cb, C0 c0); | ||
|
||
/** Creates a |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe the variadic form of this would be:
template <typename F, typename... ContextTypes>
struct context {
F f;
std::tuple<ContextTypes...> c;
context(F f, ContextTypes... c)
: f(f), c(c...) {}
template <size_t... I, typename... ArgTs>
void do_call(std::index_sequence<I...>, ArgTs... args)
{
f(get<I>(c)..., args...);
}
template <typename... ArgTs>
void operator()(ArgTs... args)
{
do_call(std::index_sequence_for<ContextTypes>(), args...);
}
};
But that would involve writing std::tuple
for ARM C 5.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It just requires a small subset of tuple that shouldn't be hard to implement:
template<typename... Ts>
struct Tuple : TupleImpl<std:: index_sequence_for <Ts...>, Ts...>;
template<typename Indexes, typename... Ts>
struct TupleImpl;
template<size_t... Indexes, typename... Ts>
struct TupleImpl<std::index_sequence<Indexes...>, Ts...> : : TupleLeaf<Indexes, Ts>... { };
template<size_t Index, typename T>
struct TupleLeaf {
T value;
T& get();
};
template<size_t Index, typename T, typename... Ts>
struct nth_element_impl {
using type = typename nth_element_impl<Index-1, Ts...>::type;
};
template<typename T, typename... Ts>
struct nth_element_impl<0, T, Ts...> {
using type = T;
};
template<size_t Index, typename... Ts>
using nth_element = typename nth_element_impl<Index, Ts...>::type;
template<size_t index, typename... Ts>
auto get(const Tuple<Ts...>& t) -> nth_element<index, Ts...> {
return static_cast<TupleLeaf<index, nth_element<index, Ts...>>>(t).get();
}
We can do that latter if it is necessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Don't see any pressing need here - the non-variadic repetition is small and self-contained, and easier to understand.
But I would appreciate any tips on avoiding the bound argument repetition I've just put back, or the (-,C,V,CV) quads on every member function pointer. Those two multiply together nastily.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we could use a traits to test the compatibility between the pointer and the function to bind.
Rules are fairly simple:
- ptr const: reject non const member function
- ptr volatile: reject non volatile member function
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm, I guess maybe you get it nearly for free with is_invocable
. (Having paid the cost there).
As I've implemented that now, I'll look at applying it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It seems to work: https://godbolt.org/z/wGCijo
Unwind previous commit and restore original behaviour for binding, pending further investigation. Some functions like `EventQueue::call` require precisely matching argument types to get the correct overload, as before. Others like `EventQueue::event` permit compatible types for binding, and those handle the 0-5 bound arguments non-variadically in order to correctly locate the free arguments in deduction.
Okay, I've gone back to non-variadic binding, putting back all the type deduction logic as it was. Did it as a separate commit so the interested can still see what didn't work. If this is the final version, we can squash it all together. Means an extra 500 lines, but it's still a 5,500 line saving. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
dd55353
to
013377a
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good to me. We can add optimisations (perfect forwarding can be added to many places) once this is in.
Yes, I'll be sending more updates presently. My immediate priority is to rework "mbed_cxxsupport.h" before it gets locked in for 5.14 - want to get some better structure there. (See #10274 (comment)) |
CI started |
Test run: SUCCESSSummary: 11 of 11 test jobs passed |
Sorry I didn't get a chance to review this, I would general trust @pan- more than me with advanced C++ anyways 😁 Glad these made it in! I don't know if this is directly related, but someone's broken GitHub's code frequency graph: |
Are there any examples or tests which use these declarations? I have been struggling to use them to define events with a class method as the handler. Forward declarations seem to be left non-variadic, which makes the compiler to expect one argument in the template. The respective declarations: mbed-os/events/include/events/Event.h Lines 34 to 35 in 0db72d0
mbed-os/events/include/events/EventQueue.h Lines 48 to 49 in 0db72d0
I found that using |
This PR wasn't introducing any functionality - it was just replacing the existing implementation's multiple 0,1,2,3,4,5 argument binding forms with a variadic equivalent. (I guess the new functionality is that you could now bind 20 arguments...) You can see examples in the docs here: https://os.mbed.com/docs/mbed-os/v6.2/apis/scheduling-tutorials.html |
But simplest basic usage for a class method callback would be:
That will call the |
To make an
Then That's assuming you want to bind the 101 into the event. If you want to not bind, and specify parameter at call time, then
should work. The variadic stuff should deduce by looking at the prototype of So in the first case |
Thank you for the detailed explanation. I needed to declare an Therefore I used this approach:
Please correct me if I am wrong. |
Yes, that also seems fine. I believe in the current implementation they come out as the same thing, because (Personally, I've never made much use of |
Description
Variadic templates can reduce Event.h from 4,100 lines to 300, and EventQueue.h from 3,400 to 1,000, so 6,000 lines saved total.
End result isn't totally variadic, as we still need specialisations for storing 0-5 values in contexts, but that specialisation is now in exactly one place.
Only change other from switching to variadic templates is using delegating constructors instead of the
new (this)
trick. That trick is still used in the assignment operator.Minor documentation correction. It's possible that the separate simplified variadic Doxygen version not be needed now, but I've left it.
Pull request type
Reviewers
@pan-, @bulislaw, @geky, @evedon