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

Remove ^ in capture set syntax #22725

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions compiler/src/dotty/tools/dotc/core/Mode.scala
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,14 @@ object Mode {
*/
val ImplicitExploration: Mode = newMode(12, "ImplicitExploration")

/** We are currently inside a capture set.
* A term name could be a capture variable, so we need to
* check that it is valid to use as type name.
* Since this mode is only used during annotation typing,
* we can reuse the value of `ImplicitExploration` to save bits.
*/
val InCaptureSet: Mode = ImplicitExploration

/** We are currently unpickling Scala2 info */
val Scala2Unpickling: Mode = newMode(13, "Scala2Unpickling")

Expand Down
8 changes: 1 addition & 7 deletions compiler/src/dotty/tools/dotc/parsing/Parsers.scala
Original file line number Diff line number Diff line change
Expand Up @@ -1594,20 +1594,14 @@ object Parsers {
}

/** CaptureRef ::= { SimpleRef `.` } SimpleRef [`*`]
* | [ { SimpleRef `.` } SimpleRef `.` ] id `^`
* | [ { SimpleRef `.` } SimpleRef `.` ] id
*/
def captureRef(): Tree =
val ref = dotSelectors(simpleRef())
if isIdent(nme.raw.STAR) then
in.nextToken()
atSpan(startOffset(ref)):
PostfixOp(ref, Ident(nme.CC_REACH))
else if isIdent(nme.UPARROW) then
in.nextToken()
atSpan(startOffset(ref)):
convertToTypeId(ref) match
case ref: RefTree => makeCapsOf(ref)
case ref => ref
else ref

/** CaptureSet ::= `{` CaptureRef {`,` CaptureRef} `}` -- under captureChecking
Expand Down
5 changes: 4 additions & 1 deletion compiler/src/dotty/tools/dotc/typer/Applications.scala
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ import annotation.threadUnsafe

import scala.util.control.NonFatal
import dotty.tools.dotc.inlines.Inlines
import dotty.tools.dotc.cc.isRetains

object Applications {
import tpd.*
Expand Down Expand Up @@ -1114,7 +1115,9 @@ trait Applications extends Compatibility {
val fun2 = Applications.retypeSignaturePolymorphicFn(fun1, methType)
simpleApply(fun2, proto)
case funRef: TermRef =>
val app = ApplyTo(tree, fun1, funRef, proto, pt)
// println(i"typedApply: $funRef, ${tree.args}, ${funRef.symbol.maybeOwner.isRetains}")
val applyCtx = if funRef.symbol.maybeOwner.isRetains then ctx.addMode(Mode.InCaptureSet) else ctx
val app = ApplyTo(tree, fun1, funRef, proto, pt)(using applyCtx)
convertNewGenericArray(
widenEnumCase(
postProcessByNameArgs(funRef, app).computeNullable(),
Expand Down
12 changes: 12 additions & 0 deletions compiler/src/dotty/tools/dotc/typer/Typer.scala
Original file line number Diff line number Diff line change
Expand Up @@ -715,6 +715,10 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
&& ctx.owner.owner.unforcedDecls.lookup(tree.name).exists
then // we are in the arguments of a this(...) constructor call
errorTree(tree, em"$tree is not accessible from constructor arguments")
else if name.isTermName && ctx.mode.is(Mode.InCaptureSet) then
// If we are in a capture set and the identifier is not a term name,
// try to type it with the same name but as a type
typed(untpd.makeCapsOf(untpd.cpy.Ident(tree)(name.toTypeName)), pt)
else
errorTree(tree, MissingIdent(tree, kind, name, pt))
end typedIdent
Expand Down Expand Up @@ -920,6 +924,13 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
typedCBSelect(tree0, pt, qual)
else EmptyTree

// Otherwise, if we are in a capture set, try to type it as a capture variable
// reference (as selecting a type name).
def trySelectTypeInCaptureSet() =
if tree0.name.isTermName && ctx.mode.is(Mode.InCaptureSet) then
typedSelectWithAdapt(untpd.cpy.Select(tree0)(qual, tree0.name.toTypeName), pt, qual)
else EmptyTree

// Otherwise, report an error
def reportAnError() =
assignType(tree,
Expand All @@ -941,6 +952,7 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
.orElse(tryDynamic())
.orElse(trySelectable())
.orElse(tryCBCompanion())
.orElse(trySelectTypeInCaptureSet())
.orElse(reportAnError())
end typedSelectWithAdapt

Expand Down
2 changes: 1 addition & 1 deletion tests/neg-custom-args/captures/capset-bound.scala
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ class IO
case class File(io: IO^)

def test(io1: IO^, io2: IO^) =
def f[C >: CapSet^{io1} <: CapSet^](file: File^{C^}) = ???
def f[C >: CapSet^{io1} <: CapSet^](file: File^{C}) = ???
val f1: File^{io1} = ???
val f2: File^{io2} = ???
val f3: File^{io1, io2} = ???
Expand Down
3 changes: 1 addition & 2 deletions tests/neg-custom-args/captures/capset-bound2.scala
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,11 @@ import caps.*

class IO

def f[C^](io: IO^{C^}) = ???
def f[C^](io: IO^{C}) = ???

def test =
f[CapSet](???)
f[CapSet^{}](???)
f[CapSet^](???)
f[Nothing](???) // error
f[String](???) // error

3 changes: 1 addition & 2 deletions tests/neg-custom-args/captures/capset-members.scala
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import caps.*
trait Abstract[X^]:
type C >: X <: CapSet^
// Don't test the return type using Unit, because it is a pure type.
def boom(): AnyRef^{C^}
def boom(): AnyRef^{C}

class Concrete extends Abstract[CapSet^{}]:
type C = CapSet^{}
Expand All @@ -27,4 +27,3 @@ class Concrete5(a: AnyRef^, b: AnyRef^) extends Abstract[CapSet^{a}]:

class Concrete6(a: AnyRef^, b: AnyRef^) extends Abstract[CapSet^{a}]:
def boom(): AnyRef^{b} = b // error

9 changes: 4 additions & 5 deletions tests/neg-custom-args/captures/capture-parameters.scala
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,7 @@ import caps.*

class C

def test[X^, Y^, Z >: X <: Y](x: C^{X^}, y: C^{Y^}, z: C^{Z^}) =
val x2z: C^{Z^} = x
val z2y: C^{Y^} = z
val x2y: C^{Y^} = x // error

def test[X^, Y^, Z >: X <: Y](x: C^{X}, y: C^{Y}, z: C^{Z}) =
val x2z: C^{Z} = x
val z2y: C^{Y} = z
val x2y: C^{Y} = x // error
8 changes: 4 additions & 4 deletions tests/neg-custom-args/captures/capture-poly.scala
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,8 @@ trait Foo extends Capability
trait CaptureSet:
type C >: CapSet <: CapSet^

def capturePoly[C^](a: Foo^{C^}): Foo^{C^} = a
def capturePoly2(c: CaptureSet)(a: Foo^{c.C^}): Foo^{c.C^} = a
def capturePoly[C^](a: Foo^{C}): Foo^{C} = a
def capturePoly2(c: CaptureSet)(a: Foo^{c.C}): Foo^{c.C} = a

def test =
val x: Foo^ = ???
Expand All @@ -15,8 +15,8 @@ def test =
object X extends CaptureSet:
type C = CapSet^{x}

val z1: Foo^{X.C^} = x
val z2: Foo^{X.C^} = y // error
val z1: Foo^{X.C} = x
val z2: Foo^{X.C} = y // error

val z3: Foo^{x} = capturePoly(x)
val z4: Foo^{x} = capturePoly(y) // error
Expand Down
26 changes: 13 additions & 13 deletions tests/neg-custom-args/captures/capture-vars-subtyping.scala
Original file line number Diff line number Diff line change
Expand Up @@ -3,32 +3,32 @@ import caps.*

def test[C^] =
val a: C = ???
val b: CapSet^{C^} = a
val b: CapSet^{C} = a
val c: C = b
val d: CapSet^{C^, c} = a
val d: CapSet^{C, c} = a

// TODO: make "CapSet-ness" of type variables somehow contagious?
// Then we don't have to spell out the bounds explicitly...
def testTrans[C^, D >: CapSet <: C, E >: CapSet <: D, F >: C <: CapSet^] =
val d1: D = ???
val d2: CapSet^{D^} = d1
val d2: CapSet^{D} = d1
val d3: D = d2
val e1: E = ???
val e2: CapSet^{E^} = e1
val e2: CapSet^{E} = e1
val e3: E = e2
val d4: D = e1
val c1: C = d1
val c2: C = e1
val f1: F = c1
val d_e_f1: CapSet^{D^,E^,F^} = d1
val d_e_f2: CapSet^{D^,E^,F^} = e1
val d_e_f3: CapSet^{D^,E^,F^} = f1
val d_e_f1: CapSet^{D,E,F} = d1
val d_e_f2: CapSet^{D,E,F} = e1
val d_e_f3: CapSet^{D,E,F} = f1
val f2: F = d_e_f1
val c3: C = d_e_f1 // error
val c4: C = f1 // error
val e4: E = f1 // error
val e5: E = d1 // error
val c5: CapSet^{C^} = e1
val c5: CapSet^{C} = e1


trait A[+T]
Expand All @@ -37,12 +37,12 @@ trait B[-C]

def testCong[C^, D^] =
val a: A[C] = ???
val b: A[CapSet^{C^}] = a
val c: A[CapSet^{D^}] = a // error
val d: A[CapSet^{C^,D^}] = a
val b: A[CapSet^{C}] = a
val c: A[CapSet^{D}] = a // error
val d: A[CapSet^{C,D}] = a
val e: A[C] = d // error
val f: B[C] = ???
val g: B[CapSet^{C^}] = f
val g: B[CapSet^{C}] = f
val h: B[C] = g
val i: B[CapSet^{C^,D^}] = h // error
val i: B[CapSet^{C,D}] = h // error
val j: B[C] = i
30 changes: 15 additions & 15 deletions tests/neg-custom-args/captures/capture-vars-subtyping2.scala
Original file line number Diff line number Diff line change
Expand Up @@ -10,32 +10,32 @@ trait BoundsTest:

def testTransMixed[A^,
B >: CapSet <: A,
C >: CapSet <: CapSet^{B^},
C >: CapSet <: CapSet^{B},
D >: CapSet <: C,
E >: CapSet <: CapSet^{D^},
F >: CapSet <: CapSet^{A^,b},
X >: CapSet <: CapSet^{F^,D^},
Y >: CapSet^{F^} <: CapSet^{F^,A^,b},
Z >: CapSet^{b} <: CapSet^{b,Y^}] =
E >: CapSet <: CapSet^{D},
F >: CapSet <: CapSet^{A,b},
X >: CapSet <: CapSet^{F,D},
Y >: CapSet^{F} <: CapSet^{F,A,b},
Z >: CapSet^{b} <: CapSet^{b,Y}] =
val e: E = ???
val e2: CapSet^{E^} = e
val e2: CapSet^{E} = e
val ed: D = e
val ed2: CapSet^{D^} = e
val ed2: CapSet^{D} = e
val ec: C = e
val ec2: CapSet^{C^} = e
val ec2: CapSet^{C} = e
val eb: B = e
val eb2: CapSet^{B^} = e
val eb2: CapSet^{B} = e
val ea: A = e
val ea2: CapSet^{A^} = e
val ea2: CapSet^{A} = e
val ex: X = e // error
val ex2: CapSet^{X^} = e // error
val ex2: CapSet^{X} = e // error
val f: F = ???
val f2: CapSet^{F^} = f
val f2: CapSet^{F} = f
val y: Y = f
val y2: CapSet^{Y^} = f
val y2: CapSet^{Y} = f
val cb: CapSet^{b} = ???
val z: Z = cb
val z2: CapSet^{Z^} = cb
val z2: CapSet^{Z} = cb

def callTransMixed =
val x, y, z: Bar^ = ???
Expand Down
2 changes: 1 addition & 1 deletion tests/neg-custom-args/captures/cc-poly-1.scala
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ object Test:
class C extends Capability
class D

def f[X^](x: D^{X^}): D^{X^} = x
def f[X^](x: D^{X}): D^{X} = x

def test(c1: C, c2: C) =
f[Any](D()) // error
Expand Down
2 changes: 1 addition & 1 deletion tests/neg-custom-args/captures/cc-poly-2.scala
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ object Test:
class C extends Capability
class D

def f[X^](x: D^{X^}): D^{X^} = x
def f[X^](x: D^{X}): D^{X} = x

def test(c1: C, c2: C) =
val d: D^ = D()
Expand Down
2 changes: 1 addition & 1 deletion tests/neg-custom-args/captures/i21313.scala
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ trait Async:
def foo(x: Async) = x.await(???) // error

trait Source[+T, Cap^]:
final def await(using ac: Async^{Cap^}) = ac.await[T, Cap](this) // Contains[Cap, ac] is assured because {ac} <: Cap.
final def await(using ac: Async^{Cap}) = ac.await[T, Cap](this) // Contains[Cap, ac] is assured because {ac} <: Cap.

def test(using ac1: Async^, ac2: Async^, x: String) =
val src1 = new Source[Int, CapSet^{ac1}] {}
Expand Down
2 changes: 1 addition & 1 deletion tests/neg-custom-args/captures/i21347.scala
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import language.experimental.captureChecking

def runOps[C^](ops: List[() ->{C^} Unit]): Unit =
def runOps[C^](ops: List[() ->{C} Unit]): Unit =
ops.foreach: op => // error
op()

Expand Down
4 changes: 2 additions & 2 deletions tests/neg-custom-args/captures/i21868.scala
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,11 @@ import caps.*

trait AbstractWrong:
type C <: CapSet
def f(): Unit^{C^} // error
def f(): Unit^{C} // error

trait Abstract1:
type C >: CapSet <: CapSet^
def f(): Unit^{C^}
def f(): Unit^{C}

// class Abstract2:
// type C^
Expand Down
6 changes: 3 additions & 3 deletions tests/neg-custom-args/captures/i21868b.scala
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ class File

trait Abstract:
type C >: CapSet <: CapSet^
def f(file: File^{C^}): Unit
def f(file: File^{C}): Unit

class Concrete1 extends Abstract:
type C = CapSet
Expand All @@ -23,7 +23,7 @@ class Concrete3(io: IO^) extends Abstract:

trait Abstract2(tracked val io: IO^):
type C >: CapSet <: CapSet^{io}
def f(file: File^{C^}): Unit
def f(file: File^{C}): Unit

class Concrete4(io: IO^) extends Abstract2(io):
type C = CapSet
Expand All @@ -35,7 +35,7 @@ class Concrete5(io1: IO^, io2: IO^) extends Abstract2(io1):

trait Abstract3[X^]:
type C >: CapSet <: X
def f(file: File^{C^}): Unit
def f(file: File^{C}): Unit

class Concrete6(io: IO^) extends Abstract3[CapSet^{io}]:
type C = CapSet
Expand Down
4 changes: 2 additions & 2 deletions tests/neg-custom-args/captures/i22005.scala
Original file line number Diff line number Diff line change
Expand Up @@ -4,5 +4,5 @@ class IO
class File(io: IO^)

class Handler[C^]:
def f(file: File^): File^{C^} = file // error
def g(file: File^{C^}): File^ = file // ok
def f(file: File^): File^{C} = file // error
def g(file: File^{C}): File^ = file // ok
4 changes: 2 additions & 2 deletions tests/neg-custom-args/captures/polyCaptures.check
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
-- Error: tests/neg-custom-args/captures/polyCaptures.scala:4:22 -------------------------------------------------------
4 |val runOpsCheck: [C^] -> (ops: List[() ->{C^} Unit]) ->{C^} Unit = runOps // error
4 |val runOpsCheck: [C^] -> (ops: List[() ->{C} Unit]) ->{C} Unit = runOps // error
| ^
| Implementation restriction: polymorphic function types cannot wrap function types that have capture sets
-- Error: tests/neg-custom-args/captures/polyCaptures.scala:5:23 -------------------------------------------------------
5 |val runOpsCheck2: [C^] => (ops: List[() ->{C^} Unit]) ->{C^} Unit = runOps // error
5 |val runOpsCheck2: [C^] => (ops: List[() ->{C} Unit]) ->{C} Unit = runOps // error
| ^
| Implementation restriction: polymorphic function types cannot wrap function types that have capture sets
6 changes: 3 additions & 3 deletions tests/neg-custom-args/captures/polyCaptures.scala
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
class Box[X](val elem: X)

val runOps = [C^] => (b: Box[() ->{C^} Unit]) => b.elem()
val runOpsCheck: [C^] -> (ops: List[() ->{C^} Unit]) ->{C^} Unit = runOps // error
val runOpsCheck2: [C^] => (ops: List[() ->{C^} Unit]) ->{C^} Unit = runOps // error
val runOps = [C^] => (b: Box[() ->{C} Unit]) => b.elem()
val runOpsCheck: [C^] -> (ops: List[() ->{C} Unit]) ->{C} Unit = runOps // error
val runOpsCheck2: [C^] => (ops: List[() ->{C} Unit]) ->{C} Unit = runOps // error


10 changes: 5 additions & 5 deletions tests/neg-custom-args/captures/use-capset.check
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
-- Error: tests/neg-custom-args/captures/use-capset.scala:5:50 ---------------------------------------------------------
5 |private def g[C^] = (xs: List[Object^{C^}]) => xs.head // error
| ^^^^^^^
| Capture set parameter C leaks into capture scope of method g.
| To allow this, the type C should be declared with a @use annotation
-- Error: tests/neg-custom-args/captures/use-capset.scala:5:49 ---------------------------------------------------------
5 |private def g[C^] = (xs: List[Object^{C}]) => xs.head // error
| ^^^^^^^
| Capture set parameter C leaks into capture scope of method g.
| To allow this, the type C should be declared with a @use annotation
-- [E007] Type Mismatch Error: tests/neg-custom-args/captures/use-capset.scala:11:22 -----------------------------------
11 | val _: () -> Unit = h // error: should be ->{io}
| ^
Expand Down
Loading
Loading