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

Drop @capability annotations #20396

Merged
merged 10 commits into from
Jun 4, 2024
Merged

Conversation

odersky
Copy link
Contributor

@odersky odersky commented May 13, 2024

Replace with references that inherit trait Capability.

Also: Handle tracked vals in classes.

As a next step, addCaptureRefinements should use the tracked logic for all capturing parameters. Right now, we create a fresh capture set variable instead, but that is too loose.

@@ -11,4 +11,6 @@ import annotation.experimental
* THere, the capture set of any instance of `CanThrow` is assumed to be
* `{*}`.
*/
@experimental final class capability extends StaticAnnotation
@experimental
@deprecated("To make a class a capability, let it derive from the `Capability` trait instead")
Copy link
Contributor

Choose a reason for hiding this comment

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

Why not phrase this as, ...let it extend the...? derives has an established meaning which isn't what's intended here.

@odersky odersky requested a review from noti0na1 May 19, 2024 11:24
@odersky odersky requested a review from Linyxus May 19, 2024 11:24
@odersky
Copy link
Contributor Author

odersky commented May 19, 2024

Summary of changes:

  • Drop @capability
  • Instead allow a class to extend a new trait caps.Capablity. So instead of @capability class IO it would be class IO extends caps.Capability.
  • The treatment of the two is different, though. @capability added ^ to references during setup. The new capability classes are instead treated as primitive maximal capabilities without having cap in their capture set.
  • We also retract the convention that capture sets in the parents of a class are somehow inherited by references to those classes.
  • We also refine box adaptation not to create spurious captureset variables in function types, which leads to more references with function typed being kept as singletons instead of being widened.
  • Finally, we refine the scheme that adds parameter refinements to class types. We now refine only if the parameter has a capture set (for the others it does not matter) and is not tracked (since tracked parameter dependencies are already handled anyway)

Copy link
Member

@noti0na1 noti0na1 left a comment

Choose a reason for hiding this comment

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

There are still some import of annotation.capability can be deleted; the document should be updated as well.

compiler/src/dotty/tools/dotc/cc/CaptureOps.scala Outdated Show resolved Hide resolved
| of an enclosing function literal with expected type () ?->{cap1} I
-- [E007] Type Mismatch Error: tests/neg-custom-args/captures/byname.scala:4:2 -----------------------------------------
4 | def f() = if cap1 == cap1 then g else g // error
| ^
| Found: (x$0: Int) ->{cap2} Int
| Required: (x$0: Int) -> Int
| Found: ((x$0: Int) ->{cap2} Int)^{}
Copy link
Member

Choose a reason for hiding this comment

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

Why is there additional empty cs added to the function type?

@odersky odersky force-pushed the add-max-captureref branch from 2d75aaa to e801dcd Compare May 26, 2024 14:22
odersky added 10 commits May 27, 2024 16:43
A maximal capability is one that derives from `caps.Cap`.

Also: drop the given caps.Cap. It's not clear why there needs to be a given for it.
The current handling of class type refinements is unsound. We cannot simply
use a variable for the capture set of a class argument. What we need to do
instead is treat class arguments as tracked.

In this commit we at least allow explicitly declared tracked arguments. This needed
two modifications:

 - Don't additionally add a capture set for tracked arguments
 - Handle the case where a capture reference is of a singleton type which
   is another capture reference.

As a next step we should treat all class arguments as implicitly tracked.
Replace with references that inherit trait `Capability`.
Drop convention that classes inheriting from universal capturing types are capability classes. Capture sets of parents
are instead ignored.

The convention led to algebraic anomalies. For instance if

    class C extends A => B, Serializable

then C <: (A => B) & Serializable, which has an empty capture set. Yet we treat every occurrence of C as
implicitly carrying `cap`.
Only enrich classes with capture refinements for a parameter if the deep capture set of the parameter's type is nonempty.
… adaptation

This change lets more ref trees with underlying function types keep their singleton
types.
 - Avoid creating capture sets of untrackable references
 - Refine disallowRootCapability to consider only explicit captures
When an expression has a type that derives from caps.Capability, add an
explicit capture set.

Also: Address other review comments
Copy link
Member

@noti0na1 noti0na1 left a comment

Choose a reason for hiding this comment

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

LGTM

@odersky odersky merged commit c51345f into scala:main Jun 4, 2024
19 checks passed
@odersky odersky deleted the add-max-captureref branch June 4, 2024 13:23
@Kordyjan Kordyjan added this to the 3.5.1 milestone Jul 3, 2024
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.

4 participants