> On Jun 16, 2023, at 1:07 PM, Martin Uecker <uec...@tugraz.at> wrote:
>
> Am Freitag, dem 16.06.2023 um 16:21 +0000 schrieb Joseph Myers:
>> On Fri, 16 Jun 2023, Martin Uecker via Gcc-patches wrote:
>>
>>>> Note that no expressions can start with the '.' token at present. As soon
>>>> as you invent a new kind of expression that can start with that token, you
>>>> have syntactic ambiguity.
>>>>
>>>> struct s1 { int c; char a[(struct s2 { int c; char b[.c]; }) {.c=.c}.c]; };
>>>>
>>>> Is ".c=.c" a use of the existing syntax for designated initializers, with
>>>> the first ".c" being a designator and the second being a use of the new
>>>> kind of expression, or is it an assignment expression, where both the LHS
>>>> and the RHS of the assignment use the new kind of expression? And do
>>>> those .c, when the use the new kind of expression, refer to the inner or
>>>> outer struct definition?
>>>
>>> I would treat this is one integrated feature. Essentially .c is
>>> somthing like this->c for the current struct for designated
>>> initializer *and* size expressions because it is semantically
>>> so close. In the initializer I would allow only
>>> the current use for designated initialization for all names of
>>> member of the currently initialized struct, so .c = .c would
>>> be invalid. It should never refer to the outer struct if there
>>
>> I'm not clear on what the intended disambiguation rule here is, when "."
>> is seen in initializer list context - does this rule depend on whether the
>> following identifier is a member of the struct being initialized, so
>> ".c=.c" would be OK above if the initialized struct didn't have a member
>> called c but the outer struct definition did?
>
> When initializers are parsed it is already clear what
> the names of the members of the inner struct are, so
> one can differentiate between designated initializers
> and potential other uses in an expression.
>
> So the main rule is: if you parse .something in a context
> where a designator is allowed and "something" is a member
> of the current struct, then it is a designator.
So, Limiting the .something ONLY to the CURRENT structure/union might be the
simple and clean rule.
And I guess that this is also the rule for the current designator initializer
syntax in C99?
>
> So for
>
> struct foo { int c; int buf[(struct { int d; }){ .d = .c }]; };
>
> one knows during parsing that the .d is a designator
> and that .c is not.
Therefore, the above should be invalid based on this rule since .c is not a
member in the current structure.
> For
>
> struct foo { int c; int buf[(struct { int d; }){ .c = .c }]; };
>
> one knows that both uses of .c are not.
And this also is invalid since .c is not to a member in the current structure.
>
> Whether these different use cases should be allowed or not
> is a different question, but my point is that there does
> not seem to be a problem directly identifying the uses
> as a designator as usual. To me, this seems to imply that
> it is safe to use the same syntax.
>
>> That seems like a rather
>> messy rule. And does "would allow only" apply other than in the ambiguous
>> context? That seems to be implied by ".c=.c" being invalid above, because
>> to make it invalid you need to disallow the new construct being used for
>> the second .c, not just make the first .c interpreted as a designator.
>
> Yes.
>>
>> Again, this sort of thing needs a detailed written specification, with
>> multiple iterations discussed among different implementations.
>
> Oh, I agree with this.
>
>> The above
>> paragraph doesn't make clear to me any of: the disambiguation rules; what
>> is allowed in what context; how name lookup works (consider tricky cases
>> such as a reference to an identifier declared *later* in the same struct,
>> possibly in the context of C2x tag compatibility where a previous
>> definition of the struct is visible); when these expressions get
>> evaluated; what the underlying principles are behind those choices.
>
> I also agree that all this needs careful consideration and written
> rules. My point is mereley that there does not seem to be a
> fundamental issue differentiating the new feature from
> designators during parsing, so there may not be a risk using
> the same syntax.
Yes, I agree on this.
Extending the existing designated initializer syntax, .member, for the purpose
of the argument of the new attribute seems very natural.
If we can use this syntax in the argument of this new attribute,
1. it will be easy to extend the argument of this attribute to an expression.
2. It will also easy to use this syntax later if we accept the following
struct foo {
...
unsigned int count;
...
int data[.count];
};
thanks.
Qing
>
>> Using a token (existing or new) other than '.' - one that doesn't
>> introduce ambiguity in any context where expressions can be used - would
>> help significantly, although some of the issues would still apply.
>
> The cost of using a new symbol is that one has two different
> syntax for something which is semantically equivalent, i.e.
> a notion to refer to a member of the current struct.
>
> Martin