-
Notifications
You must be signed in to change notification settings - Fork 258
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
feat(reflect): add function_declaration
API to set initializer and inspect parameters
#831
feat(reflect): add function_declaration
API to set initializer and inspect parameters
#831
Conversation
This comment was marked as resolved.
This comment was marked as resolved.
57217d0
to
4de962b
Compare
Here's another bug. A member Although |
This comment was marked as off-topic.
This comment was marked as off-topic.
A minor inconsistency is that most built-in metafunctions are found by ADL. |
4de962b
to
66e83d2
Compare
Johel, can you give this a try in your test code and make sure your test code still works as expected? I had to create a little test to exercise it, since the new functions aren't used in this PR.
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.
Thanks! Here's a set of review comments that go with the commit I just pushed. Please check that it still works for your use cases, and any other feedback?
I think this is ready to merge if you have nothing more to add/tweak?
while !ret.emplace_back(get_argument(count)).empty() { count++; } // No `next` (#832). | ||
ret.pop_back(); | ||
return ret; | ||
} |
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.
Would the following be equivalent?
get_arguments: (inout this) -> std::vector<std::string> = {
metafunction_used = true;
return metafunction_args;
}
I tried a quick example and it seems to be the same.
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.
Definitely.
To think it was so easy.
@@ -272,7 +279,7 @@ declaration: @polymorphic_base @copyable type = | |||
as_type : (this) -> type_declaration = type_declaration(n, this); | |||
as_alias : (this) -> alias_declaration = alias_declaration(n, this); | |||
|
|||
get_parent : (this) -> declaration = declaration(n, this); | |||
get_parent : (this) -> declaration = declaration(n*.parent_declaration, this); |
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.
Wups! Thanks. 😌
source/parse.h
Outdated
@@ -3383,6 +3383,16 @@ struct declaration_node | |||
return false; | |||
} | |||
|
|||
auto get_function_parameters() | |||
-> std::span<std::unique_ptr<parameter_declaration_node>> |
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 usually don't expose the unique_ptr
. Also I usually as const
.
Perhaps -> std::vector<parameter_declaration_node const*>
?
source/reflect.h2
Outdated
|
||
is_binary_comparison_function: (this) -> bool = n*.is_binary_comparison_function(); | ||
|
||
default_to_virtual : (inout this) = _ = n*.make_function_virtual(); | ||
|
||
make_virtual : (inout this) -> bool = n*.make_function_virtual(); | ||
|
||
add_initializer: (move this, source: std::string_view) |
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.
Is move this
intentional?
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.
Yes, it was.
It's can be considered a manifestation of Value category is not lifetime.
Since my implementation did remove the member,
this was necessary as an indication of safety.
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.
Oh right, and I even briefly thought about that move-from-this-member original characteristic, then forgot it when writing the comment. OK, anyway it's good now, thanks. 👍
p.add_member(prefix + source); | ||
_ = p; | ||
mark_for_removal_from_enclosing_type(); | ||
} |
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.
Two things:
- I wouldn't use duplicate-and-remove here, because here it's easy to graft a new node into the existing parse tree.
- That also avoids the need to visualize-and-reparse existing code that's already been parsed, which so far I've avoided doing. (I hope visualizing round-trips, but I don't thoroughly test for that yet... I have a test case to exercise it, but it's not exhaustive and I only occasionally manually try to compile the visualization test case's output to make sure it compiles and I don't compare the before/after trees.)
I would just make add_initializer
follow the model of add_member
, which is this:
add_member: (inout this, source: std::string_view)
= {
decl := parse_statement(source);
if !decl || !n*.add_type_member(decl) {
error( std::string("error attempting to add member:\n") + source );
}
}
So add_initializer
would similarly be:
add_initializer: (inout this, source: std::string_view)
= {
stmt := parse_statement(source); // only parsing `source`, not the whole thing
if !stmt || !n*.add_function_initializer(stmt) { // need to add this function
error( std::string("error attempting to add initializer:\n") + source );
}
}
I'll push a commit to this PR that makes these changes, and provides the new add_function_initializer
function besides the existing add_type_member
function in parse.h
.
Alternatively, using preconditions it could be:
add_initializer: (inout this, source: std::string_view)
pre(!has_initializer(), "cannot add an initializer to a function that already has one")
pre(parent_is_type() , "cannot add an initializer to a function that isn't in a type scope")
= {
stmt := parse_statement(source); // only parsing `source`, not the whole thing
if !stmt || !n*.add_function_initializer(stmt) { // need to add this function
error( std::string("error attempting to add initializer:\n") + source );
}
}
I'm a fan of preconditions, but in this case the parse.h
add_type_member
function (and so similarly the new add_function_initializer
function) also acts as a precondition test because it tests those conditions and returns bool
. Note add_type_member
pretty much has to check the preconditions internally to navigate the tree correctly, and since it returns false
if it failed we just check that and emit a single diagnostic, which would be hidden by preconditions.
I did try this precondition version, and found that another drawback was that by default it aborted the compiler which was inelegant. So maybe it's worth using preconditions, but with a <compiler>
contract_group and installing a log-and-continue error handler, or using t.require
instead which exists for this. I'll think about it more but maybe not do it right now.
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 this has convinced me that both add_member
and the newadd_intializer
should be using require
more:
add_member: (inout this, source: std::string_view)
= {
decl := parse_statement(source);
require( decl as bool,
"the provided source string is not a valid statement");
require( decl*.is_declaration(),
"cannot add a member that is not a declaration");
require( n*.add_type_member(decl),
std::string("error attempting to add member:\n") + source );
}
add_initializer: (inout this, source: std::string_view)
= {
require( !has_initializer(),
"cannot add an initializer to a function that already has one");
require( parent_is_type(),
"cannot add an initializer to a function that isn't in a type scope");
stmt := parse_statement(source);
require( stmt as bool,
std::string("cannot add an initializer that is not a valid statement: ") + source);
require (n*.add_function_initializer(stmt),
std::string("error attempting to add initializer:\n") + source);
}
This improves the error message. For now I'll update the PR to go that direction. In the future we can see about integrating require
with preconditions. Interesting exercise, thanks!
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, weird how some of the review comments got interleaved with the parallel commit. These comments were intended to go on the previous commit. Ah well!
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.
Thank you for the insightful comments.
One thing these versions don't have is the ability to add a constexpr
initializer
(some relevant discussions include #387 (comment) and #761).
For using contracts, #751 will be necessary.
if !d*.has_initializer() | ||
&& !d*.is_virtual_function() | ||
&& !d*.is_defaultable_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.
I still keep noticing how nice (IMO) for
-do
-if
looks to read vertically.
I'm hitting a segfault due to #830. Now that members who have an initializer added aren't added at the end of the type scope, I already had this problem for types whose member functions weren't all generated. After that (and adjusting the call to (Previously #789 (reply in thread).) |
source/reflect.h2
Outdated
decl := parse_statement(source); | ||
if !decl || !n*.add_type_member(decl) { | ||
error( std::string("error attempting to add member:\n") + source ); | ||
} | ||
require( decl as bool, | ||
"the provided source string is not a valid statement"); | ||
require( decl*.is_declaration(), | ||
"cannot add a member that is not a declaration"); |
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.
source/reflect.h2
Outdated
require( !has_initializer(), | ||
"cannot add an initializer to a function that already has one"); | ||
require( parent_is_type(), | ||
"cannot add an initializer to a function that isn't in a type scope"); | ||
|
||
stmt := parse_statement(source); |
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.
Not using contracts also means that the parse_statement
will be executed regardless of the preconditions.
Contracts seems like the better tool here.
It would still be nice to integrate the custom error handling of the reflection API with contracts.
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 have similar issues in my code.
I'm using it as I would a contract without continuation.
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 may have mentioned this somewhere else,
but I also expect the string argument to be a lazy parameter.
I can't remember what I wrote,
but I expected the condition to guard an invalid use in the interpolated string.
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 remember now.
It was something like
require(v.empty() || …, "(v.front())$");
which the expectation that the string remains unevaluated when v
is empty.
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.
Oh right, I should be at least using if
...error
instead of bunding them in require
. Will fix...
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 would still be nice to integrate the custom error handling of the reflection API with contracts.
Hold that thought... I'm trying it now and discovering some cool things seem to already work.
Spoiler: TIL that pre<this>(expr, msg)
already invokes (*this).expects(expr, msg);
. I discovered that when I tried to add support for <this>
as a special-cased name only to discover it already worked. And there doesn't even have to be a contract_group
object around as this
behaves as its own custom contract group... +1 for duck typing and generic code.
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.
However, this is making me realize there's a reason not to turn all those requires
checks into preconditions or assertions: Contracts shouldn't be used for control flow. Today, the way to turn off contract checks is to install a do-nothing handler, but if in the future there's a way to turn off contract checks, it would disable not only the checks but also the compiler diagnostics that would be emitted.
And: - Add proper early returns for failed null checks - Disable the `contract_group` default constructor, since we aren't checking the handler before trying to invoke it.
OK, I'm just pushing a commit that restores those two preconditions to try that out with diagnose-and-continue semantics, but keeps the I'm not sure I addressed all your comments above. Please take a look -- how is this now? For the preconditions: Did you prefer diagnose-and-quit semantics? |
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.
For the preconditions: Did you prefer diagnose-and-quit semantics?
That's my preference, yes.
if !decl || !n*.add_type_member(decl) { | ||
error( std::string("error attempting to add member:\n") + source ); | ||
if !(decl as bool) { | ||
error("the provided source string is not a valid statement"); |
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.
error("the provided source string is not a valid statement"); | |
error("the provided source string is not a valid statement:\n(source)$" ); |
This one used to include the source fragment.
This is where I'm getting the error, but I can't as easily see what's wrong:
algorithms.test.cpp2: error: unexpected text '
lgorithms.test.cpp2: error: unexpected text '
lgorithms.test.cpp2: error: invalid else branch body (at '{')
algorithms.test.cpp2: error: ill-formed initializer (at '{')
algorithms.test.cpp2(37,1): error: while applying @array - the provided source string is not a valid statement
With the suggestion above (add_initializer
already does it right):
algorithms.test.cpp2: error: unexpected text '
lgorithms.test.cpp2: error: unexpected text '
lgorithms.test.cpp2: error: invalid else branch body (at '{')
algorithms.test.cpp2: error: ill-formed initializer (at '{')
algorithms.test.cpp2(37,1): error: while applying @array - the provided source string is not a valid statement:
operator(): <I, S> (this, copy first: I, copy last: S)
-> std::iter_difference_t<I> pre<Bounds>(0 <= n) requires std::sentinel_for<S, I> && std::totally_ordered_with<decltype((n)), std::iter_difference_t<I>> = {
if constexpr std::sized_sentinel_for<S, I> {
if constexpr std::same_as<S, I> {
last = std::ranges::next(first, n, last);
} else {
return operator()(first, std::ranges::next(first, n, last));
}
}
take := n;
if constexpr std::sized_sentinel_for<S, I> {
return last - first;
} else {
return take - std::ranges::advance(first, take, last);error: expected a single string_ = first;
}
}
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.
By the way, to print in colors, I use https://stackoverflow.com/questions/5947742/how-to-change-the-output-color-of-echo-in-linux
It's very useful, since I'm programmatically generating the source code.
warn: (forward s) = _ = std::cerr << "\033[1;35mwarning:\033[0m (s)$\n";
error: (forward s) -> std::string = "\033[1;31m(s)$\033[0m";
single_string: type == split_strings<"\033[1;31merror: expected a single string\033[0m">;
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.
By the way, to print in colors, I use https://stackoverflow.com/questions/5947742/how-to-change-the-output-color-of-echo-in-linux
It's very useful, since I'm programmatically generating the source code.
Thanks. Are you suggesting that cppfront have something like this, possibly under a new -fcolor-diagnostics
or -fdiagnostics-color
switch?
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.
Sorry, I missed your reply.
No, that wasn't my intention.
But I might come back later for that.
} | ||
require( n*.add_type_member(decl), | ||
std::string("unexpected error while attempting to add member:\n") + source ); |
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.
std::string("unexpected error while attempting to add member:\n") + source ); | |
"unexpected error while attempting to add member:\n(source)$") ); |
source/reflect.h2
Outdated
|
||
stmt := parse_statement(source); | ||
if !(stmt as bool) { | ||
error( std::string("cannot add an initializer that is not a valid statement: ") + source ); |
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.
error( std::string("cannot add an initializer that is not a valid statement: ") + source ); | |
error( "cannot add an initializer that is not a valid statement:\n(source)$" ); |
source/reflect.h2
Outdated
return; | ||
} | ||
require (n*.add_function_initializer(stmt), | ||
std::string("unexpected error while attempting to add initializer:\n") + source); |
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.
std::string("unexpected error while attempting to add initializer:\n") + source); | |
"unexpected error while attempting to add initializer:\n(source)$" ); |
Everything compiles and runs fine after I fixed my bug. |
…on violation And restore `source` string to error output, and make it just come out routinely as soon as a parse error is detected
Does this now look ready to merge to you, @JohelEGP ? |
source/reflect.h2
Outdated
generated_tokens* | ||
); | ||
if !ret { | ||
error( "parse failed - the source string is not a valid statement:\n(source)$"); |
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.
The source
string view is modified.
You need to add an original_source := source;
at the beginning.
This is the diagnostic for my previous bug:
algorithms.test.cpp2: error: unexpected text '
lgorithms.test.cpp2: error: unexpected text '
lgorithms.test.cpp2: error: invalid else branch body (at '{')
algorithms.test.cpp2: error: ill-formed initializer (at '{')
algorithms.test.cpp2(37,1): error: while applying @array - parse failed - the source string is not a valid statement:
}
algorithms.test.cpp2(37,1): error: while applying @array - the provided source string is not a valid statement
Note how the source string is just }
.
Yes. |
Signed-off-by: Herb Sutter <herb.sutter@gmail.com>
There were |
Yeah, looks like regenerating just changes line numbers. |
I've been playing around with program-defined metafunctions in #797.
It's been a terrific experience.
This PR enhances the reflection API on functions.
get_parameters
returns the objects in the parameter list.add_initializer
sets the function's initializer.main
's reflection API on a function has very limited queries.Returning the parameters permits generating code based on inspection.
To make full use of that, I allow setting the function's initializer.
It also convenience members:
declaration::get_arguments
.type_declaration::get_member_functions_needing_initializer
.Here's some of the things that can be done with this PR:
For an example, see
@sfml
at Playing with the meta functions / writing own meta functions #789 (reply in thread).It's on my TODO list to also play with the visitor of
@union
s discussed at How to use union metafunction to replace variant member and member functions that call std::visit. #766.the highly wanted feature mentioned at Informal meetup at CppCon 2023 @ 12.30 Monday 2 Oct #650 (comment):
Incidentally, fix a bug in
declaration::get_parent
.I've also noticed an inconsistency
declaration
's API.type
returns astd::string
, whereasname
returns astd::string_view
.Testing summary:
Acknowledgements: