Skip to content

Commit

Permalink
Improve warning for wildcard matching only null under explicit nulls …
Browse files Browse the repository at this point in the history
…(scala#21577)

Adds a more detailed warning message when a wildcard case is only reachable
by null under explict nulls flag.

Fixes scala#21577
  • Loading branch information
HarrisL2 committed Oct 23, 2024
1 parent f7f51ed commit 0cf41d0
Show file tree
Hide file tree
Showing 4 changed files with 52 additions and 5 deletions.
16 changes: 11 additions & 5 deletions compiler/src/dotty/tools/dotc/transform/patmat/Space.scala
Original file line number Diff line number Diff line change
Expand Up @@ -920,7 +920,7 @@ object SpaceEngine {
then project(OrType(selTyp, ConstantType(Constant(null)), soft = false))
else project(selTyp)
)

var i = 0
val len = cases.length
var prevs = List.empty[Space]
Expand All @@ -942,11 +942,17 @@ object SpaceEngine {
report.warning(MatchCaseUnreachable(), pat.srcPos)
if pat != EmptyTree // rethrow case of catch uses EmptyTree
&& !pat.symbol.isAllOf(SyntheticCase, butNot=Method) // ExpandSAMs default cases use SyntheticCase
&& isSubspace(covered, prev)
&& isSubspace(covered, Or(List(prev, Typ(defn.NullType)))) // for when Null is not subtype of AnyRef under explicit nulls
then {
val nullOnly = isNullable && i == len - 1 && isWildcardArg(pat)
val msg = if nullOnly then MatchCaseOnlyNullWarning() else MatchCaseUnreachable()
report.warning(msg, pat.srcPos)
val nullOnly =
(isNullable || (defn.NullType <:< selTyp))
&& i == len - 1
&& isWildcardArg(pat)
if nullOnly then {
report.warning(MatchCaseOnlyNullWarning(), pat.srcPos)
} else if isSubspace(covered, prev) then {
report.warning(MatchCaseUnreachable(), pat.srcPos)
}
}
deferred = Nil
}
Expand Down
5 changes: 5 additions & 0 deletions compiler/test/dotty/tools/dotc/CompilationTests.scala
Original file line number Diff line number Diff line change
Expand Up @@ -213,6 +213,11 @@ class CompilationTests {
)
}.checkCompile()

@Test def explicitNullsWarn: Unit = {
implicit val testGroup: TestGroup = TestGroup("explicitNullsWarn")
compileFilesInDir("tests/explicit-nulls/warn", explicitNullsOptions)
}.checkWarnings()

@Test def explicitNullsRun: Unit = {
implicit val testGroup: TestGroup = TestGroup("explicitNullsRun")
compileFilesInDir("tests/explicit-nulls/run", explicitNullsOptions)
Expand Down
12 changes: 12 additions & 0 deletions tests/explicit-nulls/warn/i21577.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
-- [E121] Pattern Match Warning: tests/explicit-nulls/warn/i21577.scala:5:9 --------------------------------------------
5 | case _ => println(2) // warn
| ^
| Unreachable case except for null (if this is intentional, consider writing case null => instead).
-- [E121] Pattern Match Warning: tests/explicit-nulls/warn/i21577.scala:12:11 ------------------------------------------
12 | case _ => println(2) // warn
| ^
| Unreachable case except for null (if this is intentional, consider writing case null => instead).
-- [E121] Pattern Match Warning: tests/explicit-nulls/warn/i21577.scala:18:11 ------------------------------------------
18 | case _ => println(2) // warn
| ^
| Unreachable case except for null (if this is intentional, consider writing case null => instead).
24 changes: 24 additions & 0 deletions tests/explicit-nulls/warn/i21577.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
def f(s: String) =
val s2 = s.trim()
s2 match
case s3: String => println(1)
case _ => println(2) // warn


def f2(s: String | Null) =
val s2 = s.nn.trim()
s2 match
case s3: String => println(1)
case _ => println(2) // warn

def f3(s: String | Null) =
val s2 = s
s2 match
case s3: String => println(1)
case _ => println(2) // warn

def f4(s: String | Int) =
val s2 = s
s2 match
case s3: String => println(1)
case _ => println(2)

0 comments on commit 0cf41d0

Please sign in to comment.