@@ -35,6 +35,7 @@ import config.SourceVersion.*
35
35
import config .SourceVersion
36
36
import dotty .tools .dotc .config .MigrationVersion
37
37
import dotty .tools .dotc .util .chaining .*
38
+ import dotty .tools .dotc .config .Feature .ccEnabled
38
39
39
40
object Parsers {
40
41
@@ -220,6 +221,10 @@ object Parsers {
220
221
def isErased = isIdent(nme.erased) && in.erasedEnabled
221
222
// Are we seeing an `erased` soft keyword that will not be an identifier?
222
223
def isErasedKw = isErased && in.isSoftModifierInParamModifierPosition
224
+ // Are we seeing a `cap` soft keyword for declaring a capture-set member or at the beginning a capture-variable parameter list?
225
+ def isCapKw = Feature .ccEnabled && isIdent(nme.cap)
226
+ // 'cap type' ?
227
+ def isCapTypeKw = isCapKw && in.lookahead.token == TYPE
223
228
def isSimpleLiteral =
224
229
simpleLiteralTokens.contains(in.token)
225
230
|| isIdent(nme.raw.MINUS ) && numericLitTokens.contains(in.lookahead.token)
@@ -1903,7 +1908,7 @@ object Parsers {
1903
1908
refinedTypeRest(atSpan(startOffset(t)) {
1904
1909
RefinedTypeTree (rejectWildcardType(t), refinement(indentOK = true ))
1905
1910
})
1906
- else if Feature .ccEnabled && in.isIdent(nme.UPARROW ) && isCaptureUpArrow then
1911
+ else if Feature .ccEnabled && in.isIdent(nme.UPARROW ) && isCaptureUpArrow then // TODO remove
1907
1912
atSpan(t.span.start):
1908
1913
in.nextToken()
1909
1914
if in.token == LBRACE
@@ -1958,7 +1963,8 @@ object Parsers {
1958
1963
1959
1964
def typeBlockStats (): List [Tree ] =
1960
1965
val tdefs = new ListBuffer [Tree ]
1961
- while in.token == TYPE do tdefs += typeBlockStat()
1966
+ while (in.token == TYPE ) do
1967
+ tdefs += typeBlockStat()
1962
1968
tdefs.toList
1963
1969
1964
1970
/** TypeBlockStat ::= ‘type’ {nl} TypeDef
@@ -2159,11 +2165,14 @@ object Parsers {
2159
2165
* NamesAndTypes ::= NameAndType {‘,’ NameAndType}
2160
2166
* NameAndType ::= id ':' Type
2161
2167
*/
2162
- def argTypes (namedOK : Boolean , wildOK : Boolean , tupleOK : Boolean ): List [Tree ] =
2163
- def argType ( ) =
2164
- val t = typ()
2168
+ def argTypes (namedOK : Boolean , wildOK : Boolean , tupleOK : Boolean ): List [Tree ] = // TOOD grammar doc
2169
+ def withWildCard ( gen : => Tree ) =
2170
+ val t = gen
2165
2171
if wildOK then t else rejectWildcardType(t)
2166
2172
2173
+ def argType () = withWildCard(typ())
2174
+ def argOrCapType () = withWildCard(if in.token == LBRACE then concreteCapsType(captureSet()) else typ())
2175
+
2167
2176
def namedArgType () =
2168
2177
atSpan(in.offset):
2169
2178
val name = ident()
@@ -2174,14 +2183,14 @@ object Parsers {
2174
2183
atSpan(in.offset):
2175
2184
val name = ident()
2176
2185
acceptColon()
2177
- NamedArg (name, argType())
2186
+ NamedArg (name, argType()) // TODO allow capsets here?
2178
2187
2179
- if namedOK && isIdent && in.lookahead.token == EQUALS then
2180
- commaSeparated(() => namedArgType())
2188
+ if namedOK && isIdent && in.lookahead.token == EQUALS then // TOOD support for named cap args
2189
+ commaSeparated(() => namedArgType())
2181
2190
else if tupleOK && isIdent && in.lookahead.isColon && sourceVersion.enablesNamedTuples then
2182
2191
commaSeparated(() => namedElem())
2183
2192
else
2184
- commaSeparated(() => argType ())
2193
+ commaSeparated(() => argOrCapType ())
2185
2194
end argTypes
2186
2195
2187
2196
def paramTypeOf (core : () => Tree ): Tree =
@@ -2240,7 +2249,7 @@ object Parsers {
2240
2249
inBraces(refineStatSeq())
2241
2250
2242
2251
/** TypeBounds ::= [`>:' Type] [`<:' Type]
2243
- * | `^` -- under captureChecking
2252
+ * | `^` -- under captureChecking TODO remove
2244
2253
*/
2245
2254
def typeBounds (): TypeBoundsTree =
2246
2255
atSpan(in.offset):
@@ -2250,10 +2259,29 @@ object Parsers {
2250
2259
else
2251
2260
TypeBoundsTree (bound(SUPERTYPE ), bound(SUBTYPE ))
2252
2261
2262
+ /** CaptureSetBounds ::= [`>:' CaptureSetOrRef ] [`<:' CaptureSetOrRef ] --- under captureChecking
2263
+ */
2264
+ def captureSetBounds (): TypeBoundsTree =
2265
+ atSpan(in.offset):
2266
+ TypeBoundsTree (capsBound(SUPERTYPE ), capsBound(SUBTYPE ))
2267
+
2253
2268
private def bound (tok : Int ): Tree =
2254
2269
if (in.token == tok) { in.nextToken(); toplevelTyp() }
2255
2270
else EmptyTree
2256
2271
2272
+ private def capsBound (refs : List [Tree ], isLowerBound : Boolean = false ): Tree =
2273
+ if isLowerBound && refs.isEmpty then // lower bounds with empty capture sets become a pure CapSet
2274
+ Select (scalaDot(nme.caps), tpnme.CapSet )
2275
+ else
2276
+ makeRetaining(Select (scalaDot(nme.caps), tpnme.CapSet ), refs, if refs.isEmpty then tpnme.retainsCap else tpnme.retains)
2277
+
2278
+ private def capsBound (tok : Int ): Tree =
2279
+ if (in.token == tok) then
2280
+ in.nextToken()
2281
+ capsBound(captureSet(), isLowerBound = tok == SUPERTYPE )
2282
+ else
2283
+ capsBound(Nil , isLowerBound = tok == SUPERTYPE )
2284
+
2257
2285
/** TypeAndCtxBounds ::= TypeBounds [`:` ContextBounds]
2258
2286
*/
2259
2287
def typeAndCtxBounds (pname : TypeName ): Tree = {
@@ -2263,6 +2291,15 @@ object Parsers {
2263
2291
else atSpan((t.span union cbs.head.span).start) { ContextBounds (t, cbs) }
2264
2292
}
2265
2293
2294
+ /** CaptureSetAndCtxBounds ::= CaptureSetBounds [`:` ContextBounds] -- under captureChecking
2295
+ */
2296
+ def captureSetAndCtxBounds (pname : TypeName ): Tree = {
2297
+ val t = captureSetBounds()
2298
+ val cbs = contextBounds(pname)
2299
+ if (cbs.isEmpty) t
2300
+ else atSpan((t.span union cbs.head.span).start) { ContextBounds (t, cbs) }
2301
+ }
2302
+
2266
2303
/** ContextBound ::= Type [`as` id] */
2267
2304
def contextBound (pname : TypeName ): Tree =
2268
2305
val t = toplevelTyp(inContextBound = true )
@@ -2782,7 +2819,10 @@ object Parsers {
2782
2819
in.nextToken()
2783
2820
simpleExprRest(selectorOrMatch(t), location, canApply = true )
2784
2821
case LBRACKET =>
2785
- val tapp = atSpan(startOffset(t), in.offset) { TypeApply (t, typeArgs(namedOK = true , wildOK = false )) }
2822
+ val tapp = atSpan(startOffset(t), in.offset) {
2823
+ val args = typeArgs(namedOK = true , wildOK = false )
2824
+ TypeApply (t, args)
2825
+ }
2786
2826
simpleExprRest(tapp, location, canApply = true )
2787
2827
case LPAREN | LBRACE | INDENT if canApply =>
2788
2828
val app = atSpan(startOffset(t), in.offset) { mkApply(t, argumentExprs()) }
@@ -3305,6 +3345,7 @@ object Parsers {
3305
3345
case nme.transparent => Mod .Transparent ()
3306
3346
case nme.infix => Mod .Infix ()
3307
3347
case nme.tracked => Mod .Tracked ()
3348
+ case nme.cap => Mod .CaptureParam ()
3308
3349
}
3309
3350
}
3310
3351
@@ -3372,7 +3413,7 @@ object Parsers {
3372
3413
* | override
3373
3414
* | opaque
3374
3415
* LocalModifier ::= abstract | final | sealed | open | implicit | lazy | erased |
3375
- * inline | transparent | infix
3416
+ * inline | transparent | infix | cap
3376
3417
*/
3377
3418
def modifiers (allowed : BitSet = modifierTokens, start : Modifiers = Modifiers ()): Modifiers = {
3378
3419
@ tailrec
@@ -3461,7 +3502,6 @@ object Parsers {
3461
3502
recur(numLeadParams, firstClause = true , prevIsTypeClause = false )
3462
3503
end typeOrTermParamClauses
3463
3504
3464
-
3465
3505
/** ClsTypeParamClause::= ‘[’ ClsTypeParam {‘,’ ClsTypeParam} ‘]’
3466
3506
* ClsTypeParam ::= {Annotation} [‘+’ | ‘-’]
3467
3507
* id [HkTypeParamClause] TypeAndCtxBounds
@@ -3486,6 +3526,43 @@ object Parsers {
3486
3526
in.nextToken()
3487
3527
ok
3488
3528
3529
+ def ensureNoHKParams () = // for cap params
3530
+ if in.token == LBRACKET then
3531
+ syntaxError(em " 'cap' parameters cannot have type parameters " )
3532
+ in.nextToken()
3533
+
3534
+ def ensureNoVariance () = // for cap params
3535
+ if isIdent(nme.raw.PLUS ) || isIdent(nme.raw.MINUS ) then
3536
+ syntaxError(em " no `+/-` variance annotation allowed here " )
3537
+ in.nextToken()
3538
+
3539
+ def typeOrCapParam (): TypeDef =
3540
+ if isCapKw then
3541
+ in.nextToken()
3542
+ capParam()
3543
+ else typeParam()
3544
+
3545
+ def capParam (): TypeDef = {
3546
+ val start = in.offset
3547
+ var mods = annotsAsMods() | Param
3548
+ if paramOwner.isClass then
3549
+ mods |= PrivateLocal
3550
+ ensureNoVariance() // TODO: in the future, we might want to support variances on capture params, ruled out for now
3551
+ atSpan(start, nameStart) {
3552
+ val name =
3553
+ if paramOwner.acceptsWildcard && in.token == USCORE then
3554
+ in.nextToken()
3555
+ WildcardParamName .fresh().toTypeName
3556
+ else ident().toTypeName
3557
+ ensureNoHKParams()
3558
+ val bounds =
3559
+ if paramOwner.acceptsCtxBounds then captureSetAndCtxBounds(name)
3560
+ else if sourceVersion.enablesNewGivens && paramOwner == ParamOwner .Type then captureSetAndCtxBounds(name)
3561
+ else captureSetBounds()
3562
+ TypeDef (name, bounds).withMods(mods)
3563
+ }
3564
+ }
3565
+
3489
3566
def typeParam (): TypeDef = {
3490
3567
val start = in.offset
3491
3568
var mods = annotsAsMods() | Param
@@ -3509,11 +3586,14 @@ object Parsers {
3509
3586
TypeDef (name, lambdaAbstract(hkparams, bounds)).withMods(mods)
3510
3587
}
3511
3588
}
3512
- commaSeparated(() => typeParam ())
3589
+ commaSeparated(() => typeOrCapParam ())
3513
3590
}
3514
3591
3515
3592
def typeParamClauseOpt (paramOwner : ParamOwner ): List [TypeDef ] =
3516
- if (in.token == LBRACKET ) typeParamClause(paramOwner) else Nil
3593
+ if (in.token == LBRACKET )
3594
+ typeParamClause(paramOwner)
3595
+ else
3596
+ Nil
3517
3597
3518
3598
/** ContextTypes ::= FunArgType {‘,’ FunArgType}
3519
3599
*/
@@ -3855,25 +3935,29 @@ object Parsers {
3855
3935
* | var VarDef
3856
3936
* | def DefDef
3857
3937
* | type {nl} TypeDef
3938
+ * | cap type {nl} CapDef -- under capture checking
3858
3939
* | TmplDef
3859
3940
* EnumCase ::= `case' (id ClassConstr [`extends' ConstrApps]] | ids)
3860
3941
*/
3861
- def defOrDcl (start : Int , mods : Modifiers ): Tree = in.token match {
3862
- case VAL =>
3863
- in.nextToken()
3864
- patDefOrDcl(start, mods)
3865
- case VAR =>
3866
- val mod = atSpan(in.skipToken()) { Mod .Var () }
3867
- val mod1 = addMod(mods, mod)
3868
- patDefOrDcl(start, mod1)
3869
- case DEF =>
3870
- defDefOrDcl(start, in.skipToken(mods))
3871
- case TYPE =>
3872
- typeDefOrDcl(start, in.skipToken(mods))
3873
- case CASE if inEnum =>
3874
- enumCase(start, mods)
3875
- case _ =>
3876
- tmplDef(start, mods)
3942
+ def defOrDcl (start : Int , mods : Modifiers ): Tree =
3943
+ in.token match {
3944
+ case VAL =>
3945
+ in.nextToken()
3946
+ patDefOrDcl(start, mods)
3947
+ case VAR =>
3948
+ val mod = atSpan(in.skipToken()) { Mod .Var () }
3949
+ val mod1 = addMod(mods, mod)
3950
+ patDefOrDcl(start, mod1)
3951
+ case DEF =>
3952
+ defDefOrDcl(start, in.skipToken(mods))
3953
+ case TYPE if mods.is(CaptureParam ) =>
3954
+ capDefOrDcl(start, in.skipToken(mods))
3955
+ case TYPE =>
3956
+ typeDefOrDcl(start, in.skipToken(mods))
3957
+ case CASE if inEnum =>
3958
+ enumCase(start, mods)
3959
+ case _ =>
3960
+ tmplDef(start, mods)
3877
3961
}
3878
3962
3879
3963
/** PatDef ::= ids [‘:’ Type] [‘=’ Expr]
@@ -4082,6 +4166,43 @@ object Parsers {
4082
4166
}
4083
4167
}
4084
4168
4169
+ private def concreteCapsType (refs : List [Tree ]): Tree =
4170
+ makeRetaining(Select (scalaDot(nme.caps), tpnme.CapSet ), refs, tpnme.retains)
4171
+
4172
+ /** CapDef ::= id CaptureSetAndCtxBounds [‘=’ CaptureSetOrRef] -- under capture checking
4173
+ */
4174
+ def capDefOrDcl (start : Offset , mods : Modifiers ): Tree =
4175
+ newLinesOpt()
4176
+ atSpan(start, nameStart) {
4177
+ val nameIdent = typeIdent()
4178
+ val tname = nameIdent.name.asTypeName
4179
+ if in.token == LBRACKET then syntaxError(em " 'cap type' declarations cannot have type parameters " )
4180
+
4181
+ def makeCapDef (refs : List [Tree ] | Tree ): Tree = {
4182
+ val tdef = TypeDef (nameIdent.name.toTypeName,
4183
+ refs.match
4184
+ case refs : List [Tree ] => concreteCapsType(refs)
4185
+ case bounds : Tree => bounds)
4186
+
4187
+ if (nameIdent.isBackquoted)
4188
+ tdef.pushAttachment(Backquoted , ())
4189
+ finalizeDef(tdef, mods, start)
4190
+ }
4191
+
4192
+ in.token.match
4193
+ case EQUALS =>
4194
+ in.nextToken()
4195
+ makeCapDef(captureSet())
4196
+ case SUBTYPE | SUPERTYPE | SEMI | NEWLINE | NEWLINES | COMMA | RBRACE | OUTDENT | EOF =>
4197
+ makeCapDef(captureSetAndCtxBounds(tname))
4198
+ case _ if (staged & StageKind .QuotedPattern ) != 0
4199
+ || sourceVersion.enablesNewGivens && in.isColon =>
4200
+ makeCapDef(captureSetAndCtxBounds(tname))
4201
+ case _ =>
4202
+ syntaxErrorOrIncomplete(ExpectedCaptureBoundOrEquals (in.token))
4203
+ return EmptyTree // return to avoid setting the span to EmptyTree
4204
+ }
4205
+
4085
4206
/** TmplDef ::= ([‘case’] ‘class’ | ‘trait’) ClassDef
4086
4207
* | [‘case’] ‘object’ ObjectDef
4087
4208
* | ‘enum’ EnumDef
@@ -4675,6 +4796,7 @@ object Parsers {
4675
4796
* | ‘var’ VarDef
4676
4797
* | ‘def’ DefDef
4677
4798
* | ‘type’ {nl} TypeDef
4799
+ * | ‘cap’ ‘type’ {nl} CapDef -- under capture checking
4678
4800
* (in reality we admit class defs and vars and filter them out afterwards in `checkLegal`)
4679
4801
*/
4680
4802
def refineStatSeq (): List [Tree ] = {
@@ -4699,9 +4821,14 @@ object Parsers {
4699
4821
fail(em " this kind of definition cannot be a refinement " )
4700
4822
4701
4823
while
4824
+ val mods =
4825
+ if isCapTypeKw then // allow `cap type` in refinements
4826
+ in.nextToken()
4827
+ addMod(Modifiers (), Mod .CaptureParam ())
4828
+ else Modifiers ()
4702
4829
val dclFound = isDclIntro
4703
4830
if dclFound then
4704
- stats ++= checkLegal(defOrDcl(in.offset, Modifiers () ))
4831
+ stats ++= checkLegal(defOrDcl(in.offset, mods ))
4705
4832
var what = " declaration"
4706
4833
if inFunReturnType then what += " (possible cause: missing `=` in front of current method body)"
4707
4834
statSepOrEnd(stats, noPrevStat = ! dclFound, what)
0 commit comments