aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPhilipp Haller <hallerp@gmail.com>2014-03-27 16:01:58 +0100
committerPhilipp Haller <hallerp@gmail.com>2014-03-27 16:01:58 +0100
commit5a49226d34bfd2a9ec9f8a5cbae5df4c1b4bc8db (patch)
tree9bf7b4b58828f93d04700d55ec6ff54d8d02b64e
parentbdb1686c00a7fc9304c91eb5418a83529ab935fc (diff)
parent5c6ea29966fa80faae13892da50fc68ed1bf9ae7 (diff)
downloadscala-async-5a49226d34bfd2a9ec9f8a5cbae5df4c1b4bc8db.tar.gz
scala-async-5a49226d34bfd2a9ec9f8a5cbae5df4c1b4bc8db.tar.bz2
scala-async-5a49226d34bfd2a9ec9f8a5cbae5df4c1b4bc8db.zip
Merge pull request #67 from retronym/ticket/52-lazy-val
Allow lazy vals without await in the initializer
-rw-r--r--src/main/scala/scala/async/internal/AsyncAnalysis.scala5
-rw-r--r--src/main/scala/scala/async/internal/AsyncTransform.scala11
-rw-r--r--src/test/scala/scala/async/neg/NakedAwait.scala4
-rw-r--r--src/test/scala/scala/async/run/lazyval/LazyValSpec.scala29
4 files changed, 42 insertions, 7 deletions
diff --git a/src/main/scala/scala/async/internal/AsyncAnalysis.scala b/src/main/scala/scala/async/internal/AsyncAnalysis.scala
index ffbc04d..6540bdb 100644
--- a/src/main/scala/scala/async/internal/AsyncAnalysis.scala
+++ b/src/main/scala/scala/async/internal/AsyncAnalysis.scala
@@ -60,9 +60,8 @@ trait AsyncAnalysis {
super.traverse(tree)
case Return(_) =>
c.abort(tree.pos, "return is illegal within a async block")
- case ValDef(mods, _, _, _) if mods.hasFlag(Flag.LAZY) =>
- // TODO lift this restriction
- c.abort(tree.pos, "lazy vals are illegal within an async block")
+ case DefDef(mods, _, _, _, _, _) if mods.hasFlag(Flag.LAZY) && containsAwait =>
+ reportUnsupportedAwait(tree, "lazy val initializer")
case CaseDef(_, guard, _) if guard exists isAwait =>
// TODO lift this restriction
reportUnsupportedAwait(tree, "pattern guard")
diff --git a/src/main/scala/scala/async/internal/AsyncTransform.scala b/src/main/scala/scala/async/internal/AsyncTransform.scala
index cf9dd1c..aca8c7e 100644
--- a/src/main/scala/scala/async/internal/AsyncTransform.scala
+++ b/src/main/scala/scala/async/internal/AsyncTransform.scala
@@ -57,8 +57,7 @@ trait AsyncTransform {
val template = Template(List(tryToUnit, typeOf[() => Unit]).map(TypeTree(_)), emptyValDef, body)
val t = ClassDef(NoMods, name.stateMachineT, Nil, template)
- typingTransform(atPos(macroPos)(Block(t :: Nil, Literal(Constant(())))))((tree, api) => api.typecheck(tree))
- t
+ typecheckClassDef(t)
}
val stateMachineClass = stateMachine.symbol
@@ -211,4 +210,12 @@ trait AsyncTransform {
}
result
}
+
+ def typecheckClassDef(cd: ClassDef): ClassDef = {
+ val Block(cd1 :: Nil, _) = typingTransform(atPos(macroPos)(Block(cd :: Nil, Literal(Constant(())))))(
+ (tree, api) =>
+ api.typecheck(tree)
+ )
+ cd1.asInstanceOf[ClassDef]
+ }
}
diff --git a/src/test/scala/scala/async/neg/NakedAwait.scala b/src/test/scala/scala/async/neg/NakedAwait.scala
index 9778c72..2af691f 100644
--- a/src/test/scala/scala/async/neg/NakedAwait.scala
+++ b/src/test/scala/scala/async/neg/NakedAwait.scala
@@ -163,10 +163,10 @@ class NakedAwait {
@Test
def lazyValIllegal() {
- expectError("lazy vals are illegal") {
+ expectError("await must not be used under a lazy val initializer") {
"""
| import _root_.scala.async.internal.AsyncId._
- | def foo(): Any = async { val x = { lazy val y = 0; y } }
+ | def foo(): Any = async { val x = { lazy val y = await(0); y } }
| ()
|
|""".stripMargin
diff --git a/src/test/scala/scala/async/run/lazyval/LazyValSpec.scala b/src/test/scala/scala/async/run/lazyval/LazyValSpec.scala
new file mode 100644
index 0000000..701d5d6
--- /dev/null
+++ b/src/test/scala/scala/async/run/lazyval/LazyValSpec.scala
@@ -0,0 +1,29 @@
+/*
+ * Copyright (C) 2012-2014 Typesafe Inc. <http://www.typesafe.com>
+ */
+
+package scala.async
+package run
+package lazyval
+
+import org.junit.Test
+import scala.async.internal.AsyncId._
+
+class LazyValSpec {
+ @Test
+ def lazyValAllowed() {
+ val result = async {
+ var x = 0
+ lazy val y = { x += 1; 42 }
+ assert(x == 0, x)
+ val z = await(1)
+ val result = y + x
+ assert(x == 1, x)
+ identity(y)
+ assert(x == 1, x)
+ result
+ }
+ result mustBe 43
+ }
+}
+