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

A first, incomplete implementation of polyadic capture calculus #18566

Closed
wants to merge 58 commits into from

Conversation

odersky
Copy link
Contributor

@odersky odersky commented Sep 18, 2023

it's been a big revamp. Most things are in place now.

The major omission is that we currently don't check for level inversion, so there are soundness holes. We know how to do it in principle, but it will be some work to push an implementation through.

@odersky odersky force-pushed the change-root-levels branch from 890a2ba to 0534202 Compare October 1, 2023 15:05
odersky added 29 commits October 1, 2023 18:33
This is needed if we want to do setup transforms as part of denotation transformers.
 1. Update symbols owner before recursing to children, so that new owners
    taking account of try block owners are installed.
 2. Query owner chaines at phase checkCpatures when computing enclosing level owners.
I tried several other strategies, but nothing worked.

 - update after Setup: This means we cannot to levelOwner during setup.
 - running levelOwner after Setup causes CyclicErrors when transforming symbols
   in Setup
 - having a seperate notion of ccOwner maintained in a CCState table does
   not work since there are too many hidden uses of owner everywhere that
   would ignore that table.
(except if they are updated). This is the first step towards converting
to capturing types in SymTransformer, where we have the context.
Need to copy the denotation, since denotations come with next pointers
which would get scrambled otherwise.
Makes things more regular and allows for a
non-identity mapping between declared types and infos of vals.
The new check is that a publicly visible inferred type after capture checking
must conform to the type before capture checking (which is also the type seen
by other separately compiled units).
Widen singleton types when instantiating local roots in checkConforms
Don't force info when checking whether something is a root capability while printing
Refactor everything so that now vals as well as defs can be level owners
Innstead of traversing old types at postCheck, note what needs to be done
when these types are first transformed at Setup.
With the new way to construct capturing types at Setup, we can add the correct
boxing annotations there, so the previous unmodular hack can be dropped.
Break out operation to add a single element. This makes widenCaptures
redundant.
Avoid the mutable state in CapturSet.Var
# Conflicts:
#	compiler/src/dotty/tools/dotc/cc/Setup.scala
@odersky odersky force-pushed the change-root-levels branch from e1858db to 2d07bd5 Compare October 1, 2023 16:36
The problem arises if we have a class like the one in pos-custom-args/captures/refs.scala:

```scala
class MonoRef(init: Proc):
  type MonoProc = Proc
  var x: MonoProc = init
  def getX: MonoProc = x
  def setX(x: MonoProc): Unit = this.x = x
```

The type of `getX` and `setX` refer to the local root capability of class `MonoRef`.
When we call `m.getX` or `m.setX` in `m: MonoRef`, these occurrences have to be adapted to
capture roots in the scope of the selection. We determine these roots by inspecting the
capture set of `m` and picking a root that corresponds to it.
@odersky odersky changed the title Run Setup code one phase before CheckCaptures A first, incomplete implementation of polyadic capture calculus Oct 3, 2023
@odersky odersky marked this pull request as ready for review October 3, 2023 21:38
 - Drop valsCanBeLevelOwners and levelOwnersNeedCapParam config settings
 - Drop !isCaseClassSynthetic condition in isLevelOwner
 - Drop handling of anonymous functions in levelOwnerNamed

Neither are needed anymore with the the conflg flags gone.
When comparing refined types

   C{x: T}^cs  <:<  C{x: T'}^cs'

we need to remember the capture set `cs` on the left hand side when we
get to ask whether the refined type has a member `x: T'`, so that we
can correctly instantiate any local root `cap[C]` in the member type of `x`.
Same if the LHS is simply `C^cs`. In both cases we strip the capture set before
we get to take the compute `x`, so we have to re-add it at the point of that
member computation.
Without that step, class refinements can contain references to the generic
root `cap`.
Check that instantiated root variable of a method inside a class
is nested in the instance variable of the class.

For the moment I was not able to construct a counter example where this
makes a difference.
@odersky odersky force-pushed the change-root-levels branch from 2e1ed26 to b7d8047 Compare October 7, 2023 17:24
odersky added 10 commits October 8, 2023 12:38
We need to do it for compiled as well as imported types. We only
keep adding variables to selftypes of local classes in postProcess.

This allows us to drop one occurrence of LooseRootChecking
That's needed since a subclass might be a level owner.
Also: Print local roots with id of owner under -uniqid
The only time a nested object class could have a self type with a non-empty
capture set is when it declared in the current compilation unit. Then that
capture set is a Var that gets installed via an updateInfo. But then we do
also an updateInfo of the module val, so no need to do the same thing in
transformSym.
@odersky
Copy link
Contributor Author

odersky commented Oct 15, 2023

In the end, after getting everything to compile again, I conclude that the scheme is too complicated to be seriously considered. Mapping cap to local roots means a lot of machinery is active under the surface, and errors require too much expertise from the user to diagnose them.

So I decided to put this on hold in favor of #18699, which is overall simpler and hopefully more robust.

@SethTisue SethTisue marked this pull request as draft October 17, 2023 22:09
odersky added a commit that referenced this pull request Oct 23, 2023
- Start with `sealed` type parameters, extended so that class parameters
can also be `sealed`.
- Tighten the rules so that type parameter arguments for `sealed` type
parameters must themselves be `sealed`, except if the argument is from
some outer scope.
- Introduce a coercion that maps a universal capture set to a local root
that is enclosed by the owners of all free variables of the coerced
expression.


Supersedes #18566
@odersky
Copy link
Contributor Author

odersky commented Oct 25, 2023

Withdrawn in favor of #18699

@odersky odersky closed this Oct 25, 2023
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.

1 participant