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

Update augmentation specification. #4123

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open

Update augmentation specification. #4123

wants to merge 3 commits into from

Conversation

lrhn
Copy link
Member

@lrhn lrhn commented Oct 7, 2024

  • Says that the entire class scope is in the lexical scope for members.

  • Update augemented specification, include constructors.

    • Only reserved inside expressions and bodies of augmenting
      member-like declarations, not type-like declarations.
    • Try to give grammar. It's really a parameterized grammar
      per kind of declaration. We don't have good parameterized grammars,
      so it's said in prose.
  • Changes specification for non-redirecting generative constructors.

    • Adds order of initialization of instance variables with initializers.
    • Says that body cannot use augmented.
    • Therefore no longer needs to reuse variable bindings, so don't.
    • Each augmenting declaration binds actuals to formals,
      remember any super parameters or super initializers,
      executes initializer list entries,
      and then executes its augmented declaration.
      When that comes back, it executes its body.
    • (Layering an augmentation on top of an NRG-constructor
      auto-calls augmented at the end of its initializer list.)

    The initializing formals and super parameters need more work.

  • Some small tweaks to disallow invalid combinations.

*   Says that the entire class scope is in the lexical scope for members.
*   Update `augemented` specification, include constructors.
    *   Only reserved inside expressions and bodies of augmenting
        member-like declarations, not type-like declarations.
    *   Try to give grammar. It's really a parameterized grammar
        per kind of declaration. We don't have good parameterized grammars,
        so it's said in prose.
*   Changes specification for non-redirecting generative constructors.
    *   Adds order of initialization of instance variables with initializers.
    *   Says that body cannot use `augmented`.
    *   Therefore no longer needs to reuse variable bindings, so don't.
    *   Each augmenting declaration binds actuals to formals,
        remember any super parameters or super initializers,
        executes initializer list entries,
        and then executes its augmented declaration.
        When that comes back, it executes its body.
    *   (Layering an augmentation on top of an NRG-constructor
        auto-calls augmented at the end of its initializer list.)

    The initializing formals and super parameters need more work.
*   Some small tweaks to disallow invalid combinations.
@lrhn lrhn requested review from munificent and jakemac53 October 7, 2024 18:01
@lrhn
Copy link
Member Author

lrhn commented Oct 7, 2024

Needs more work. No updated version/changelog, pretty sure the rules for using initializing formals and super parameters are inconsistent, aka. not updated everywhere, and super-parameters need more formal specification.

Still: WDYT? How far is it from what we want?

Copy link
Member

@munificent munificent left a comment

Choose a reason for hiding this comment

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

These changes are all great. Thank you for warming up my half-baked proposal into something edible. :)

@@ -237,16 +238,16 @@ which may be used to provide a body for an abstract member._

An augmentation that replaces the body of a function may also want to
preserve and run the code of the augmented declaration (hence the name
"augmentation"). It may want to run its own code before the augmented
code, after it, or both. To support that, we allow a new expression syntax
Copy link
Member

Choose a reason for hiding this comment

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

Haha, yes. This brings such joy to my dumb proofreader's heart. :)

in the function body refers to the augmented function, and it must
occur only as the function of a function invocation, which means
it is immediately followed by one, or no, type argument list
and then an argument list.
Copy link
Member

Choose a reason for hiding this comment

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

This is a little difficult to read. Maybe:

"which means it is immediately followed by a type argument list (which may be omitted) then an argument list (which must be present but may be empty)."

Copy link
Member Author

Choose a reason for hiding this comment

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

I went through a lot of phrasings here, including "zero or one type parameter lists", but the plural annoyed me.
Maybe just:

it is immediately followed by an argument list, or a type parameter list and then an argument list.

_Tear offs are not possible, neither are member accesses like
`augmented.call()`._
The expression `augmented` must not occur in an optional parameter
default value expressions _It is a reserved word in such expressions,
Copy link
Member

Choose a reason for hiding this comment

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

"default value expressions" -> "default value expression."

Copy link
Member Author

Choose a reason for hiding this comment

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

Going with singular "expression" rather than removing "an".

working/augmentation-libraries/feature-specification.md Outdated Show resolved Hide resolved
* **Augmenting operators**: Operators are methods with special
invocation syntax. When augmenting an operator declaration, `augmented`
refers to the augmented operator *method*, and it must occur only
as the function of a function invocation. For example, when augmenting
`operator +` you could use `augmented(1)` to call the augmented
Copy link
Member

Choose a reason for hiding this comment

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

"could" -> "would"

It's possible to close over `augmented` inside an augmenting function, to,
for example, do `list.forEach((x) => augmented(x));` in a function augmentation,
but doing `List.forEach(augmented)` is not allowed. That would treat the
augmented function as if it had an identity._
Copy link
Member

Choose a reason for hiding this comment

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

This is a good thing to point out. It feels weird at first, but then I realized that super has the same limitation and it's never bothered me.

super-initializer of either the augmented or augmenting constructor.

Invoking a non-redirecting generative constructor, with an argument list and a
new object to initialize, two effects.
Copy link
Member

Choose a reason for hiding this comment

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

"two effects." -> "has two effects:".

declaration has a body, execute that body of this declaration in a
scope that has the parameter scope of the binding-actuals-to-formals
step as parent scope, and with `this` bound to the object being
initialized.
Copy link
Member

Choose a reason for hiding this comment

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

This whole section is really good.

| : |
.-----------------. : .-----------------.
| C.isEven() body | : | C._isOdd() body |
'-----------------' : '-----------------'
```
Copy link
Member

Choose a reason for hiding this comment

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

My ASCII art diagram lives on! :)


_It still is a compile time error for both a static and instance member
of the same name to be defined on the same type, even if they live in
declarations. You cannot work around this restriction by moving the
Copy link
Member

Choose a reason for hiding this comment

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

"live in declarations" -> "live in different declarations for the same class".

Copy link
Contributor

@jakemac53 jakemac53 left a comment

Choose a reason for hiding this comment

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

Looking good so far, will continue my review tomorrow.

The exact grammar and meaning of the `augmented` keyword depends on what is
being augmented.

* **Augmenting variables**: Within an augmenting variable declaration's
Copy link
Contributor

Choose a reason for hiding this comment

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

This sounds repetitive with the new wording, maybe remove all of can only be used in an initializer expression, and it

Copy link
Member Author

@lrhn lrhn Oct 9, 2024

Choose a reason for hiding this comment

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

It is repetetive, and unnecessarily so.
I think one of the parts is a leftover that should have been removed.

An augmenting variable declaration has no other place to put the word than in the initializer expression.

(Well, except in metadata. We can just generally say, once and for all, that augmented is not a reserved word inside metadata annotations on the augmeting declaration.)

working/augmentation-libraries/feature-specification.md Outdated Show resolved Hide resolved
The reserved words parses as a `<primary>` expression.
Evaluating `augmented` invokes the augmented declaration's getter
function, and evaluates to the returned value.
The static type of `augmented` is the declared return type of the augmented
Copy link
Contributor

Choose a reason for hiding this comment

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

Should we specify this as the declared return type or something a bit more general, like how we use the "static type" for initializer expressions?

For instance, does "declared return type" cover the override case correctly?

Copy link
Member Author

Choose a reason for hiding this comment

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

Good point. We want the return type, not the declared return type. Or maybe we want the inferred return type. If there is a difference, it's the type that the declaration is determined to have after (top-level?) type inference has been performed.

We just need to define what the return type of a getter declaration is.

The getter return type of a non-augmenting getter declaration is the type denoted by its declared return type, if any.
If there is no declared return type, the getter is an instance getter of a class, mixin or enum declaration, and any superinterface of the surrounding type has a getter with the same name, then the return type is the return type of the getter with that name in the combined superinterface of the current declaration. It's a compile-time error if the combined superinterface does not have a combined type signature for that getter.
Otherwise the return type of the getter is dynamic.
(We don't infer the type of bodies in top-level inference, right? It's only for variables.)

The getter return type of a variable declaration is the type denoted by the declared type of the variable, if any. Otherwise, if the variable has no initializer and it's an instance variable, it's the return type of the getter or parameter type of the setter in the combined super-interface, if any. Otherwise it's the static type of its initializer expression.

The getter return type of an augmenting getter declaration is the getter return type of the augmented declaration. (Can't change the type. If the augmenting declaration writes a type, it must be that type.)

It gets complicated, because it's basically repeating all of type inference. I hope there is a way to reuse type inference instead.

Copy link
Contributor

Choose a reason for hiding this comment

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

Right, but I think all we need to state here is that it uses the inferred return type? We don't need to explain how the inference works.

working/augmentation-libraries/feature-specification.md Outdated Show resolved Hide resolved
and then an argument list.
_Tear offs are not possible, neither are member accesses like
`augmented.call()`._
The expression `augmented` must not occur in an optional parameter
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
The expression `augmented` must not occur in an optional parameter
The expression `augmented` must not occur in optional parameter

lrhn and others added 2 commits October 9, 2024 11:21
Co-authored-by: Jacob MacDonald <jakemac@google.com>
Co-authored-by: Jacob MacDonald <jakemac@google.com>
no earlier declaration to augment.*
invoked with the same value for `this`. For a non-redirecting factory
constructor, it's invoked with the same type parameter bindings for the
surrounding type.
Copy link
Contributor

Choose a reason for hiding this comment

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

I had a similar comment above... I hadn't considered that the type arguments are on the class and not the function... that does make sense. And in any case it is a very weird thing to return a class with a tighter generic type than was given, it probably shouldn't even be allowed.

@@ -724,11 +793,11 @@ It is a **compile-time error** if:

* An `abstract` variable is augmented with a non-abstract variable.

* An `external` variable is augmented with an `abstract` variable.
* An `external` variable is augmented with a variable declaration.
Copy link
Contributor

Choose a reason for hiding this comment

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

I am not sure if we need this restriction? It is true that the external keyword would be unnecessary though in this case, but I feel like we have gone back and forth on this a bit so and I am curious what the motivation for not allowing this would be?

Do we need to make similar changes for external functions? The CFE today actually requires us to mark certain things external (but this is a bug).

Copy link
Member Author

@lrhn lrhn Oct 9, 2024

Choose a reason for hiding this comment

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

Let's see if I remember my reasoning.
I think it was mainly that an external "variable declaration" doesn't introduce a variable to augment, it only introduces a getter and a setter. The only augmentations you should be allowed to make are to getter and setter.

But I can also see wanting to add metadata and comments.
Allowing an external variable declaration with an initializer expression is definitely wrong.
Allowing an augmenting variable declaration with no external modofier would be weird, because that also looks like it is variable. So it should be marked external, but that feels like it's an external augmentation, that is, the augmenting getter/setter body is supplied by the compiler.

Maybe we should allow @metadata augment id; to apply to any declaration, and not be able to do anything other than add metadata (and comments, which the language has absolutely no opinion on, but then, it has no opinion on metadata either, as long as it compiles).

That is:

external int foo; // valid

@newData
augment external int foo; // Reasonable, but looks like the augmentation itself is external.

@newData
augment int foo; // Possible, looks like it's augmenting a variable.

@newData
augment abstract int foo; // Already not allowed (`abstract` and `external` are usually mutually exclusive).

augment int foo = 1; // Completely unacceptable.

@newData
augment foo; // Nice-to-have, but not currently possible.

Which ones should we allow, and what should they mean?

Copy link
Contributor

Choose a reason for hiding this comment

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

I think it was mainly that an external "variable declaration" doesn't introduce a variable to augment, it only introduces a getter and a setter. The only augmentations you should be allowed to make are to getter and setter.

That is a compelling argument for me. I think we can leave things as you have defined here. Possibly expand to allow adding metadata/comments.

@newData
augment external int foo; // Reasonable, but looks like the augmentation itself is external.

This seems ok to me, it just adds the annotation.

@newData
augment int foo; // Possible, looks like it's augmenting a variable.

I don't like this, because it does seem like it is actually making it non-external. And, we don't allow augmenting getters/setters with variables.

@newData
augment abstract int foo; // Already not allowed (`abstract` and `external` are usually mutually exclusive).

lgtm

augment int foo = 1; // Completely unacceptable.

agreed (now)

@newData
augment foo; // Nice-to-have, but not currently possible.

I think this looks a bit weird so I wouldn't try to allow it


* First it initializes any instance variables in the class which have
initializing expressions:
* For each instance variable declared by the class, in source order
Copy link
Contributor

Choose a reason for hiding this comment

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

Do we need to define "source order" given that fields might be defined in different parts?

Copy link
Member Author

Choose a reason for hiding this comment

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

I think it's defined somewhere as the "before" order from parts-with-imports. If not, it should be.

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

Successfully merging this pull request may close these issues.

3 participants