-
Notifications
You must be signed in to change notification settings - Fork 1.5k
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
Save WhereOperand node instead of performing a scan #4795
base: trunk
Are you sure you want to change the base?
Conversation
Note, this is based on #4772 but with the paren expr fix for the test. Also a few small style things (FIXME -> TODO since I think it doesn't need to be addressed here, NodeId -> WhereOperandId because I think we prefer that (?), CARBON_CHECK that |
toolchain/check/handle_impl.cpp
Outdated
// Stop at the `where` node, if present, to exclude it from syntactic match. | ||
auto name = PopImplIntroducerAndParamsAsNameComponent( | ||
context, name_context.where_operand_node_id.is_valid() | ||
? Parse::NodeId(name_context.where_operand_node_id) | ||
: node_id); |
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.
In open discussion, we talked about checking the last node kind to make sure it is a WhereExpr
.
// Stop at the `where` node, if present, to exclude it from syntactic match. | |
auto name = PopImplIntroducerAndParamsAsNameComponent( | |
context, name_context.where_operand_node_id.is_valid() | |
? Parse::NodeId(name_context.where_operand_node_id) | |
: node_id); | |
// Find the end node of the declaration for syntactic match. | |
Parse::Tree::PostorderIterator last_param_iter(node_id); | |
auto node_kind = context.parse_tree().node_kind(node_id); | |
CARBON_CHECK(node_kind == Parse::NodeKind::ImplDefinitionStart || | |
node_kind == Parse::NodeKind::ImplDecl); | |
// Subtracting 1 since we don't want to include the final `{`, `;`. | |
--last_param_iter; | |
// Stop at the `where` node, if present, to exclude it from syntactic match. | |
if (context.parse_tree().node_kind(*last_param_iter) == | |
Parse::NodeKind::WhereExpr && | |
name_context.where_operand_node_id.is_valid()) { | |
last_param_iter = name_context.where_operand_node_id; | |
--last_param_iter; | |
} | |
// Pop the `impl` introducer and any `forall` parameters as a "name". | |
auto name = PopImplIntroducerAndParamsAsNameComponent( | |
context, *last_param_iter); |
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 the discussed check was something like:
CARBON_CHECK(name_context.where_operand_node_id.is_valid() ==
(context.parse_tree().node_kind(constraint_node) ==
Parse::NodeKind::WhereExpr),
"where_operand_node_id: {0}; constraint node kind: {1}",
name_context.where_operand_node_id,
context.parse_tree().node_kind(constraint_node));
Note, one difference with your suggestion is that it validates the WhereExpr
when where_operand_node_id
is set, essentially a if-and-only-if. But I've gone a slightly different approach now, because I was realizing a similar thing could be achieved through the node stack at negligible storage cost (no cost for non-impls).
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.
Was looking at this a little further, I've at least moved out the param iter logic as requested.
Given the type of node_id
is AnyImplDeclId
, is the CARBON_CHECK
still needed? I was thinking on edit that it could be removed because the validation comes from the type already.
// parameters. | ||
auto name_context = context.decl_name_stack().FinishImplName(); | ||
CARBON_CHECK(name_context.state == DeclNameStack::NameContext::State::Empty); | ||
|
||
// Pop the `impl` introducer and any `forall` parameters as a "name". |
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.
// Pop the `impl` introducer and any `forall` parameters as a "name". |
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'm confused by this, can you clarify why this no longer applies?
#include "toolchain/parse/typed_nodes.h" | ||
|
||
namespace Carbon::Check { | ||
|
||
auto HandleParseNode(Context& /*context*/, Parse::ParenExprStartId /*node_id*/) | ||
auto HandleParseNode(Context& context, Parse::ParenExprStartId node_id) | ||
-> bool { | ||
// The open paren is unused. | ||
// Push the start to help track nesting. | ||
context.node_stack().Push(node_id); | ||
return true; | ||
} | ||
|
||
auto HandleParseNode(Context& context, Parse::ParenExprId node_id) -> bool { | ||
// We re-push because the ParenExpr is valid for member expressions, whereas | ||
// the child expression might not be. | ||
context.node_stack().Push(node_id, context.node_stack().PopExpr()); | ||
auto expr = context.node_stack().PopExpr(); | ||
context.node_stack().PopForSoloNodeId<Parse::NodeKind::ParenExprStart>(); | ||
// We push with the ParenExpr node because it's valid for member expressions, | ||
// whereas the child expression might not be. | ||
context.node_stack().Push(node_id, expr); |
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.
With the suggested change to toolchain/check/handle_impl.cpp, I think these changes should no longer be needed.
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.
If this is removed, how does WhereExpr handling know when or when not to mark the WhereOperand
? It cannot do that in normal expression handling, and this was added to distinguish between a nested expression and the impl
form.
If you have an approach that works without this, maybe it'd make sense to update #4772 since this PR was mainly adding the paren expr tracking on top of your work?
Note, I'm suggesting a slightly different approach now, keeping the |
The "templated for consistency" variants felt a little confusing when I was working on #4795, so suggesting to remove templating where it's not helpful to instantiate (particularly when there were both templated and non-templated variants). Note this leaves a `PeekIs<IdT>` because there's indirection there, but that's more the exception than the rule.
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.
FYI, I'm working on a change to distinguish these top-level where
nodes in the parser, which will interact with this change (making it a little simpler, I think).
Would it make sense for me to close this PR, and you can merge in any necessary changes? |
Follow-up to #4762
This changes to pushing ParenExprStart to make it easy for expressions to tell that they're nested (by breaking up sequencing on the node stack).