Skip to content

Commit ef97824

Browse files
committed
Use other tree for actual symbol of Assign
1 parent 701fa8a commit ef97824

File tree

8 files changed

+112
-18
lines changed

8 files changed

+112
-18
lines changed

compiler/src/dotty/tools/dotc/ast/Trees.scala

+3-4
Original file line numberDiff line numberDiff line change
@@ -1821,16 +1821,15 @@ object Trees {
18211821
}
18221822
}
18231823

1824-
def rename(tree: NameTree, newName: Name)(using Context): tree.ThisTree[T] = {
1825-
tree match {
1824+
def rename(tree: NameTree, newName: Name)(using Context): tree.ThisTree[T] =
1825+
tree.match
18261826
case tree: Ident => cpy.Ident(tree)(newName)
18271827
case tree: Select => cpy.Select(tree)(tree.qualifier, newName)
18281828
case tree: Bind => cpy.Bind(tree)(newName, tree.body)
18291829
case tree: ValDef => cpy.ValDef(tree)(name = newName.asTermName)
18301830
case tree: DefDef => cpy.DefDef(tree)(name = newName.asTermName)
18311831
case tree: TypeDef => cpy.TypeDef(tree)(name = newName.asTypeName)
1832-
}
1833-
}.asInstanceOf[tree.ThisTree[T]]
1832+
.asInstanceOf[tree.ThisTree[T]]
18341833

18351834
object TypeDefs:
18361835
def unapply(xs: List[Tree]): Option[List[TypeDef]] = xs match

compiler/src/dotty/tools/dotc/reporting/messages.scala

+8-6
Original file line numberDiff line numberDiff line change
@@ -1528,14 +1528,16 @@ class AmbiguousExtensionMethod(tree: untpd.Tree, expansion1: tpd.Tree, expansion
15281528

15291529
class ReassignmentToVal(name: Name)(using Context)
15301530
extends TypeMsg(ReassignmentToValID) {
1531-
def msg(using Context) = i"""Reassignment to val $name"""
1531+
def msg(using Context) = i"""Reassignment to value $name"""
15321532
def explain(using Context) =
1533-
i"""|You can not assign a new value to $name as values can't be changed.
1534-
|Keep in mind that every statement has a value, so you may e.g. use
1535-
| ${hl("val")} $name ${hl("= if (condition) 2 else 5")}
1536-
|In case you need a reassignable name, you can declare it as
1537-
|variable
1533+
i"""|Members defined using `val` or `def` can't be assigned to.
1534+
|If you need to change the value of $name, use `var` instead:
15381535
| ${hl("var")} $name ${hl("=")} ...
1536+
|However, it's more common to initialize a variable just once
1537+
|with a complex expression or even a block with many statements:
1538+
| ${hl("val")} $name ${hl("= if (condition) 1 else -1")}
1539+
|Also, assignment syntax can be used if there is a corresponding setter:
1540+
| ${hl("def")} ${name}${hl("_=(x: Int): Unit = _v = x")}
15391541
|"""
15401542
}
15411543

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

+8-5
Original file line numberDiff line numberDiff line change
@@ -1373,9 +1373,11 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
13731373

13741374
def typedAssign(tree: untpd.Assign, pt: Type)(using Context): Tree =
13751375
tree.lhs match {
1376-
case lhs @ Apply(fn, args) =>
1377-
typed(untpd.Apply(untpd.Select(fn, nme.update), args :+ tree.rhs), pt)
1378-
case untpd.TypedSplice(Apply(MaybePoly(Select(fn, app), targs), args)) if app == nme.apply =>
1376+
case Apply(fn, args) =>
1377+
val appliedUpdate =
1378+
untpd.Apply(untpd.Select(fn, nme.update), args :+ tree.rhs)
1379+
typed(appliedUpdate, pt)
1380+
case untpd.TypedSplice(Apply(MaybePoly(Select(fn, nme.apply), targs), args)) =>
13791381
val rawUpdate: untpd.Tree = untpd.Select(untpd.TypedSplice(fn), nme.update)
13801382
val wrappedUpdate =
13811383
if (targs.isEmpty) rawUpdate
@@ -1389,7 +1391,7 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
13891391
def lhs1 = adapt(lhsCore, LhsProto, locked)
13901392

13911393
def reassignmentToVal =
1392-
report.error(ReassignmentToVal(lhsCore.symbol.name), tree.srcPos)
1394+
report.error(ReassignmentToVal(lhs1.symbol.name), tree.srcPos)
13931395
cpy.Assign(tree)(lhsCore, typed(tree.rhs, lhs1.tpe.widen)).withType(defn.UnitType)
13941396

13951397
def canAssign(sym: Symbol) =
@@ -1478,8 +1480,9 @@ class Typer(@constructorOnly nestingLevel: Int = 0) extends Namer
14781480
typedDynamicAssign(tree, pt)
14791481
case tpe =>
14801482
reassignmentToVal
1481-
}
1483+
}
14821484
}
1485+
end typedAssign
14831486

14841487
def typedBlockStats(stats: List[untpd.Tree])(using Context): (List[tpd.Tree], Context) =
14851488
index(stats)

tests/neg/i11561.check

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,6 @@
1111
-- [E052] Type Error: tests/neg/i11561.scala:3:30 ----------------------------------------------------------------------
1212
3 | val updateText2 = copy(text = (_: String)) // error
1313
| ^^^^^^^^^^^^^^^^^^
14-
| Reassignment to val text
14+
| Reassignment to value text
1515
|
1616
| longer explanation available when compiling with `-explain`

tests/neg/i16655.check

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
-- [E052] Type Error: tests/neg/i16655.scala:3:4 -----------------------------------------------------------------------
22
3 | x = 5 // error
33
| ^^^^^
4-
| Reassignment to val x
4+
| Reassignment to value x
55
|
66
| longer explanation available when compiling with `-explain`

tests/neg/i20338c.check

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
-- [E052] Type Error: tests/neg/i20338c.scala:9:6 ----------------------------------------------------------------------
22
9 | f.x = 42 // error
33
| ^^^^^^^^
4-
| Reassignment to val x
4+
| Reassignment to value x
55
|
66
| longer explanation available when compiling with `-explain`

tests/neg/i22671.check

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
-- [E007] Type Mismatch Error: tests/neg/i22671.scala:37:22 ------------------------------------------------------------
2+
37 | names_times(fields(0)) += fields(1).toLong // error
3+
| ^^^^^^^^^
4+
| Found: Char
5+
| Required: String
6+
|
7+
| longer explanation available when compiling with `-explain`
8+
-- [E008] Not Found Error: tests/neg/i22671.scala:41:6 -----------------------------------------------------------------
9+
41 | x() += "42" // error
10+
| ^^^^^^
11+
| value += is not a member of Int - did you mean Int.!=? or perhaps Int.<=?
12+
-- [E052] Type Error: tests/neg/i22671.scala:8:6 -----------------------------------------------------------------------
13+
8 | X.x = 27 // error
14+
| ^^^^^^^^
15+
| Reassignment to value x
16+
|
17+
| longer explanation available when compiling with `-explain`
18+
-- [E052] Type Error: tests/neg/i22671.scala:12:4 ----------------------------------------------------------------------
19+
12 | x = 27 // error
20+
| ^^^^^^
21+
| Reassignment to value x
22+
|
23+
| longer explanation available when compiling with `-explain`
24+
-- [E052] Type Error: tests/neg/i22671.scala:16:4 ----------------------------------------------------------------------
25+
16 | y = 27 // error
26+
| ^^^^^^
27+
| Reassignment to value x
28+
|
29+
| longer explanation available when compiling with `-explain`
30+
-- [E052] Type Error: tests/neg/i22671.scala:20:4 ----------------------------------------------------------------------
31+
20 | y = 27 // error
32+
| ^^^^^^
33+
| Reassignment to value z
34+
|
35+
| longer explanation available when compiling with `-explain`
36+
-- [E052] Type Error: tests/neg/i22671.scala:24:4 ----------------------------------------------------------------------
37+
24 | x = 27 // error
38+
| ^^^^^^
39+
| Reassignment to value x
40+
|
41+
| longer explanation available when compiling with `-explain`
42+
-- [E008] Not Found Error: tests/neg/i22671.scala:27:6 -----------------------------------------------------------------
43+
27 | X.x += 27 // error
44+
| ^^^^^^
45+
| value += is not a member of Int - did you mean Int.!=? or perhaps Int.<=?
46+
-- [E008] Not Found Error: tests/neg/i22671.scala:28:4 -----------------------------------------------------------------
47+
28 | 1 += 1 // error
48+
| ^^^^
49+
| value += is not a member of Int - did you mean (1 : Int).!=? or perhaps (1 : Int).<=?

tests/neg/i22671.scala

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
2+
object X:
3+
def x: Int = 42
4+
def x(y: Int): Int = x + y
5+
val z = 26
6+
7+
def f =
8+
X.x = 27 // error
9+
10+
def g =
11+
import X.x
12+
x = 27 // error
13+
14+
def h =
15+
import X.x as y
16+
y = 27 // error
17+
18+
def i =
19+
import X.z as y
20+
y = 27 // error
21+
22+
def j =
23+
val x = 42
24+
x = 27 // error
25+
26+
def k =
27+
X.x += 27 // error
28+
1 += 1 // error
29+
30+
31+
object t8763:
32+
import collection.mutable
33+
def bar(): Unit =
34+
val names_times = mutable.Map.empty[String, mutable.Set[Long]]
35+
val line = ""
36+
val Array(fields) = line.split("\t")
37+
names_times(fields(0)) += fields(1).toLong // error
38+
39+
object t9834:
40+
object x { def apply() = 42 ; def update(i: Int) = () }
41+
x() += "42" // error

0 commit comments

Comments
 (0)