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

Save WhereOperand node instead of performing a scan #4795

Open
wants to merge 11 commits into
base: trunk
Choose a base branch
from

Conversation

jonmeow
Copy link
Contributor

@jonmeow jonmeow commented Jan 13, 2025

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).

@github-actions github-actions bot requested a review from chandlerc January 13, 2025 19:26
@jonmeow jonmeow requested review from josh11b and zygoloid and removed request for chandlerc January 13, 2025 19:26
@jonmeow
Copy link
Contributor Author

jonmeow commented Jan 13, 2025

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 where is only found once because that's part of how I diagnosed -- prior code was "finding" it twice)

Comment on lines 298 to 302
// 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);
Copy link
Contributor

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.

Suggested change
// 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);

Copy link
Contributor Author

@jonmeow jonmeow Jan 14, 2025

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).

Copy link
Contributor Author

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".
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
// Pop the `impl` introducer and any `forall` parameters as a "name".

Copy link
Contributor Author

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?

toolchain/check/handle_impl.cpp Outdated Show resolved Hide resolved
Comment on lines +7 to +23
#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);
Copy link
Contributor

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.

Copy link
Contributor Author

@jonmeow jonmeow Jan 14, 2025

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?

@jonmeow
Copy link
Contributor Author

jonmeow commented Jan 14, 2025

Note, I'm suggesting a slightly different approach now, keeping the WhereOperand on the node_stack instead of popping it. This comes from talking about the side-channeling and how we couldn't just add an arg, I was thinking why pop the node off in the first place if we want to use it again? It also takes advantage of Pop checks to enforce the parity if we have an accident in tracking.

github-merge-queue bot pushed a commit that referenced this pull request Jan 15, 2025
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.
Copy link
Contributor

@josh11b josh11b left a 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).

@jonmeow
Copy link
Contributor Author

jonmeow commented Jan 17, 2025

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?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants