summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPaul Phillips <paulp@improving.org>2010-02-25 02:54:04 +0000
committerPaul Phillips <paulp@improving.org>2010-02-25 02:54:04 +0000
commit46e78e4589fe6d68703194e71cd34b5accb0942b (patch)
treef54913da3a380b984553360d024626b55c82687f
parentebe8a875e57bf562d152519a274e644268c24ecc (diff)
downloadscala-46e78e4589fe6d68703194e71cd34b5accb0942b.tar.gz
scala-46e78e4589fe6d68703194e71cd34b5accb0942b.tar.bz2
scala-46e78e4589fe6d68703194e71cd34b5accb0942b.zip
Tweaking the sealed logic in light of #3097.
Reorganizes children a little so they always come back sorted the same way the pickler does. Taking advantage of -Yfatal-warnings in the test case. Review by community.
-rw-r--r--src/compiler/scala/tools/nsc/matching/MatrixAdditions.scala21
-rw-r--r--src/compiler/scala/tools/nsc/symtab/Symbols.scala15
-rw-r--r--test/files/jvm/interpreter.check2
-rw-r--r--test/files/neg/patmatexhaust.check2
-rw-r--r--test/files/pos/bug3097.flags1
-rw-r--r--test/files/pos/bug3097.scala31
6 files changed, 58 insertions, 14 deletions
diff --git a/src/compiler/scala/tools/nsc/matching/MatrixAdditions.scala b/src/compiler/scala/tools/nsc/matching/MatrixAdditions.scala
index fd55945d50..23944c91d9 100644
--- a/src/compiler/scala/tools/nsc/matching/MatrixAdditions.scala
+++ b/src/compiler/scala/tools/nsc/matching/MatrixAdditions.scala
@@ -160,10 +160,20 @@ trait MatrixAdditions extends ast.TreeDSL
singleType(sym.tpe.prefix, lmoc) // e.g. None, Nil
else sym.tpe
+ /** Note to Martin should you come through this way: this
+ * logic looks way overcomplicated for the intention, but a little
+ * experimentation showed that at least most of it is serving
+ * some necessary purpose. It doesn't seem like much more than
+ * "sym.tpe matchesPattern tpe" ought to be necessary though.
+ *
+ * For the time being I tacked the matchesPattern test onto the
+ * end to address #3097.
+ */
(tpe.typeSymbol == sym) ||
(symtpe <:< tpe) ||
(symtpe.parents exists (x => cmpSymbols(x, tpe))) || // e.g. Some[Int] <: Option[&b]
- ((tpe.prefix memberType sym) <:< tpe) // outer, see combinator.lexical.Scanner
+ ((tpe.prefix memberType sym) <:< tpe) || // outer, see combinator.lexical.Scanner
+ (symtpe matchesPattern tpe)
}
cond(p.tree) {
@@ -180,18 +190,13 @@ trait MatrixAdditions extends ast.TreeDSL
private def requiresExhaustive(s: Symbol) =
(s hasFlag MUTABLE) && // indicates that have not yet checked exhaustivity
!(s hasFlag TRANS_FLAG) && // indicates @unchecked
- (s.tpe.typeSymbol hasFlag SEALED) &&
+ (s.tpe.typeSymbol.isSealed) &&
{ s resetFlag MUTABLE ; true } // side effects MUTABLE flag
- private def sealedSymsFor(s: Symbol): Set[Symbol] = {
- val kids = s.children flatMap sealedSymsFor
- if (s hasFlag ABSTRACT) kids else kids + s
- }
-
private lazy val inexhaustives: List[List[Combo]] = {
val collected =
for ((pv, i) <- tvars.zipWithIndex ; val sym = pv.lhs ; if requiresExhaustive(sym)) yield
- i -> sealedSymsFor(sym.tpe.typeSymbol)
+ i -> sym.tpe.typeSymbol.sealedDescendants
val folded =
collected.foldRight(List[List[Combo]]())((c, xs) => {
diff --git a/src/compiler/scala/tools/nsc/symtab/Symbols.scala b/src/compiler/scala/tools/nsc/symtab/Symbols.scala
index d878839fea..1ab0632e50 100644
--- a/src/compiler/scala/tools/nsc/symtab/Symbols.scala
+++ b/src/compiler/scala/tools/nsc/symtab/Symbols.scala
@@ -24,7 +24,6 @@ trait Symbols extends reflect.generic.Symbols { self: SymbolTable =>
def symbolCount = ids // statistics
val emptySymbolArray = new Array[Symbol](0)
- val emptySymbolSet = Set.empty[Symbol]
/** Used for deciding in the IDE whether we can interrupt the compiler */
protected var activeLocks = 0
@@ -1425,7 +1424,15 @@ trait Symbols extends reflect.generic.Symbols { self: SymbolTable =>
(if (isModule) moduleClass else toplevelClass).isFromClassFile
/** If this is a sealed class, its known direct subclasses. Otherwise Set.empty */
- def children: Set[Symbol] = emptySymbolSet
+ def children: List[Symbol] = Nil
+
+ /** Recursively finds all sealed descendants and returns a sorted list. */
+ def sealedDescendants: List[Symbol] = {
+ val kids = children flatMap (_.sealedDescendants)
+ val all = if (this hasFlag ABSTRACT) kids else this :: kids
+
+ all.distinct sortBy (_.sealedSortName)
+ }
// ToString -------------------------------------------------------------------
@@ -1951,8 +1958,8 @@ trait Symbols extends reflect.generic.Symbols { self: SymbolTable =>
override def sourceModule =
if (isModuleClass) linkedModuleOfClass else NoSymbol
- private var childSet: Set[Symbol] = emptySymbolSet
- override def children: Set[Symbol] = childSet
+ private var childSet: Set[Symbol] = Set()
+ override def children: List[Symbol] = childSet.toList sortBy (_.sealedSortName)
override def addChild(sym: Symbol) { childSet = childSet + sym }
incCounter(classSymbolCount)
diff --git a/test/files/jvm/interpreter.check b/test/files/jvm/interpreter.check
index 1a6fcba5fe..03c773e2cb 100644
--- a/test/files/jvm/interpreter.check
+++ b/test/files/jvm/interpreter.check
@@ -218,8 +218,8 @@ defined class Fact
defined class Term
scala> | | <console>:16: warning: match is not exhaustive!
-missing combination Term
missing combination Exp
+missing combination Term
def f(e: Exp) = e match { // non-exhaustive warning here
^
diff --git a/test/files/neg/patmatexhaust.check b/test/files/neg/patmatexhaust.check
index 1c46b6c9e5..ca769300c0 100644
--- a/test/files/neg/patmatexhaust.check
+++ b/test/files/neg/patmatexhaust.check
@@ -15,8 +15,8 @@ missing combination Qult Qult
def ma3(x:Mult) = (x,x) match { // not exhaustive
^
patmatexhaust.scala:49: warning: match is not exhaustive!
-missing combination Gu
missing combination Gp
+missing combination Gu
def ma4(x:Deep) = x match { // missing cases: Gu, Gp
^
diff --git a/test/files/pos/bug3097.flags b/test/files/pos/bug3097.flags
new file mode 100644
index 0000000000..570b15929d
--- /dev/null
+++ b/test/files/pos/bug3097.flags
@@ -0,0 +1 @@
+-unchecked -Yfatal-warnings
diff --git a/test/files/pos/bug3097.scala b/test/files/pos/bug3097.scala
new file mode 100644
index 0000000000..a034b960f7
--- /dev/null
+++ b/test/files/pos/bug3097.scala
@@ -0,0 +1,31 @@
+package seal
+
+sealed trait ISimpleValue
+
+sealed trait IListValue extends ISimpleValue {
+ def items: List[IAtomicValue[_]]
+}
+sealed trait IAtomicValue[O] extends ISimpleValue {
+ def data: O
+}
+
+sealed trait IAbstractDoubleValue[O] extends IAtomicValue[O] { }
+sealed trait IDoubleValue extends IAbstractDoubleValue[Double]
+
+case class ListValue(val items: List[IAtomicValue[_]]) extends IListValue
+class DoubleValue(val data: Double) extends IDoubleValue {
+ def asDouble = data
+}
+
+object Test {
+ /**
+ * @param args the command line arguments
+ */
+ def main(args: Array[String]): Unit = {
+ val v: ISimpleValue = new DoubleValue(1)
+ v match {
+ case m: IListValue => println("list")
+ case a: IAtomicValue[_] => println("atomic")
+ }
+ }
+}