Skip to content

Commit 705e10e

Browse files
authored
Constructor proxy is restricted if class is protected (#22563)
Calling a constructor proxy should be at least as restricted as using `new`. Add `Protected` to a constructor companion if the class is protected. If the companion exists but is not protected, then make the constructor proxy protected. Explicit `new` already errored for access. Fixes #22560
2 parents f9080e0 + b6ad24f commit 705e10e

File tree

7 files changed

+77
-9
lines changed

7 files changed

+77
-9
lines changed

compiler/src/dotty/tools/dotc/core/NamerOps.scala

+6-2
Original file line numberDiff line numberDiff line change
@@ -149,9 +149,11 @@ object NamerOps:
149149
*/
150150
def addConstructorApplies(scope: MutableScope, cls: ClassSymbol, modcls: ClassSymbol)(using Context): scope.type =
151151
def proxy(constr: Symbol): Symbol =
152+
var flags = ApplyProxyFlags | (constr.flagsUNSAFE & AccessFlags)
153+
if cls.is(Protected) && !modcls.is(Protected) then flags |= Protected
152154
newSymbol(
153155
modcls, nme.apply,
154-
ApplyProxyFlags | (constr.flagsUNSAFE & AccessFlags),
156+
flags,
155157
ApplyProxyCompleter(constr),
156158
cls.privateWithin,
157159
constr.coord)
@@ -175,9 +177,11 @@ object NamerOps:
175177

176178
/** A new symbol that is the constructor companion for class `cls` */
177179
def classConstructorCompanion(cls: ClassSymbol)(using Context): TermSymbol =
180+
var flags = ConstructorCompanionFlags
181+
if cls.is(Protected) then flags |= Protected
178182
val companion = newModuleSymbol(
179183
cls.owner, cls.name.toTermName,
180-
ConstructorCompanionFlags, ConstructorCompanionFlags,
184+
flags, flags,
181185
constructorCompanionCompleter(cls),
182186
coord = cls.coord,
183187
compUnitInfo = cls.compUnitInfo)

compiler/src/dotty/tools/dotc/typer/Applications.scala

+1-2
Original file line numberDiff line numberDiff line change
@@ -1198,9 +1198,8 @@ trait Applications extends Compatibility {
11981198
//
11991199
// summonFrom {
12001200
// case given A[t] =>
1201-
// summonFrom
1201+
// summonFrom:
12021202
// case given `t` => ...
1203-
// }
12041203
// }
12051204
//
12061205
// the second `summonFrom` should expand only once the first `summonFrom` is

compiler/src/dotty/tools/dotc/typer/Typer.scala

+3-3
Original file line numberDiff line numberDiff line change
@@ -4176,9 +4176,9 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
41764176
readapt(tree.appliedToNone) // insert () to primary constructors
41774177
else
41784178
errorTree(tree, em"Missing arguments for $methodStr")
4179-
case _ => tryInsertApplyOrImplicit(tree, pt, locked) {
4180-
errorTree(tree, MethodDoesNotTakeParameters(tree))
4181-
}
4179+
case _ =>
4180+
tryInsertApplyOrImplicit(tree, pt, locked):
4181+
errorTree(tree, MethodDoesNotTakeParameters(tree))
41824182
}
41834183

41844184
def adaptNoArgsImplicitMethod(wtp: MethodType): Tree = {

docs/_docs/reference/other-new-features/creator-applications.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -39,8 +39,11 @@ The precise rules are as follows:
3939
- the class has a companion object (which might have been generated in step 1), and
4040
- that companion object does not already define a member named `apply`.
4141

42-
Each generated `apply` method forwards to one constructor of the class. It has the
43-
same type and value parameters as the constructor.
42+
Each generated `apply` method forwards to one constructor of the class.
43+
It has the same type and value parameters as the constructor,
44+
as well as the same access restriction as the class.
45+
If the class is `protected`, then either the companion object must be `protected`
46+
or the `apply` method will be made `protected`.
4447

4548
Constructor proxy companions cannot be used as values by themselves. A proxy companion object must
4649
be selected with `apply` (or be applied to arguments, in which case the `apply` is implicitly

tests/neg/i22560.scala

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
2+
class A:
3+
protected class B
4+
5+
// This fails to compile, as expected
6+
val x = A().B() // error
7+
8+
object C:
9+
protected val p = "protected"
10+
protected def getString() = "Hello!"
11+
protected class D:
12+
def d = D() // ok
13+
14+
// This fails to compile
15+
// val y = C.p
16+
17+
// And this also fails to compile
18+
// val z = C.getString()
19+
20+
// However, this compiles just fine.
21+
val alpha = C.D() // error
22+
val beta = new C.D() // error

tests/neg/i22560b.scala

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
2+
class Enumeration:
3+
protected class Val(i: Int):
4+
def this() = this(42)
5+
object Val
6+
7+
class Test extends Enumeration:
8+
val Hearts = Val(27) // error
9+
val Diamonds = Val() // error
10+
11+
package p:
12+
private[p] class C(i: Int) // ctor proxy gets privateWithin of class
13+
private[p] class D(i: Int)
14+
object D
15+
16+
package q:
17+
def f() = p.C(42) // error
18+
def g() = p.D(42) // error

tests/pos/i22560.scala

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
2+
package companionless:
3+
4+
class Enumeration:
5+
protected class Val(i: Int):
6+
def this() = this(42)
7+
8+
class Test extends Enumeration:
9+
val Hearts = Val(27)
10+
val Diamonds = Val()
11+
12+
13+
package companioned:
14+
15+
class Enumeration:
16+
protected class Val(i: Int):
17+
def this() = this(42)
18+
protected object Val
19+
20+
class Test extends Enumeration:
21+
val Hearts = Val(27)
22+
val Diamonds = Val()

0 commit comments

Comments
 (0)