[cfe-dev] [PATCH] C++ nested-name-specifier (Parser)
Argiris Kirtzidis
akyrtzi at gmail.com
Sat Aug 9 18:29:02 PDT 2008
Chris Lattner wrote:.
>
> However, I think I am starting to understand what you mean.
> isDeclarationSpecifier() (for example) contains this code right now:
>
> switch (Tok.getKind()) {
> ...
> // typedef-name
> case tok::identifier:
> return Actions.isTypeName(*Tok.getIdentifierInfo(), CurScope) != 0;
>
> This means that following my approach would take us back down the
> route of having a "parse expression with leading identifier expression
> already eaten" method, and things like that. This is because you'd
> have to do something like:
>
>
> case tok::identifier:
> // eats the current identifier and related type stuff as a whole,
> D = ConsumeAndResolveIdentifier();
> if (Actions.isTypeName(D))
> ... it's a decl spec ...
> else
> .. it's an 'identifier expression' ..
>
> Is that the problem?
Yes, exactly! I want to avoid the "leading stuff.." route.
> It also means that isDeclarationSpecifier would need backtracking
> and significant other stuff to work as a predicate. In practice, this
> means we'd want to refactor all callers to not use it like it does.
Here's another way to make it work, without any backtracking:
Either Parser or Sema, keeps a scope-spec state (like Parser keeps a
'current scope' state and Sema keeps 'current DeclContext' state), which
indicates the Decl that we need to do lookup into.
As you suggested, the Parser, calls actions during parsing of scope
specifiers.
For "A::T<int>::":
-Parses 'A::' and calls ActOnNestedNameSpecifier for 'A' (updates
scope-spec)
-Parses, instantiates, etc. 'T<int>' through action calls (updates
scope-spec)
Sema actions like isTypeName and ActOnIdentifier, check the scope-spec
to see whether it needs to lookup a name inside the scope-spec decl, or
do a normal lookup.
Now, the questions is whether Sema should keep the scope-spec state, or
whether the Parser should keep the scope-spec state and pass it along to
the actions (isTypeName, etc.)
When I say passing the scope-spec to actions, I don't mean like the
CXXScopeSpec in my patch which contained only parsed tokens, I mean
passing the decl that represents the scope where names should be looked
up into.
If the Sema keeps the scope-spec state, it is cleaner, but how will the
scope-state be cleared in case of an error like "A:: ;" ?
Should the parser call an ActOnErrorAfterNestedName or something ?
>
>>> Is this approach achievable? I'd prefer to avoid parsing and
>>> rewinding unless absolutely necessary.
>>
>> The advantage of the rewinding part is that there's no need for the
>> Parser or Sema to keep and look after a 'C++ scoping' state (a state
>> that is not just local to functions, but part of the Parser/Sema class).
>
> I'm not sure I believe that. How do you propose to handle more
> complex things (like the T<int>::x case) in the future? It seems that
> we're going to need to invoke sema to do template instantiation and
> other stuff at some point: passing a pre-parsed thing like this to
> sema as one big action call seems difficult.
Yes, you are right that for templates, passing them as parsed tokens to
one action is not a good idea.
>>
>> This also means that we can drop the rewinding part if the Parser or
>> Sema keep and look after a 'C++ scoping' state.
>> If the Parser keeps the state, it will build and pass 'CXXScopeSpec'
>> objects to action methods, (like in my patch).
>> If the Sema keeps the state, it will make error recovery awkward.
>
> I actually think that things are more awkward with having Sema work
> this out. Specifically, if you have: "A :: B :: C" you might have one
> reference or it might be "A::B ::C" which can occur in a few places in
> the grammar:
> http://www.cs.berkeley.edu/~smcpeak/elkhound/sources/elsa/doc/coloncolon.txt
>
>
> If the parser just passed all of A/B/C to Sema, I'm not sure how it
> would handle this. It seems really easy if you invoke actions for
> "A::" then "B" (which returns a type). When the parser got a type, it
> would naturally ratchet on and parse ::C as a qualified name.
But "B" type could contain "C" as nested type, in which case you may
want "A::B::C".
Anyway, I don't think we need to worry about whether it should be
"A::B ::C" or "A ::B::C" or whatever.
GCC treats it as "A::B::C" always:
struct S {};
namespace A { S f(); }
S ::A::f(); // error: 'S::A' has not been declared
I think this is correct since the spec says nothing about "'::'
associativity".
-Argiris
More information about the cfe-dev
mailing list