aboutsummaryrefslogtreecommitdiff
path: root/library
diff options
context:
space:
mode:
authorMartin Odersky <odersky@gmail.com>2017-01-05 16:02:09 +0700
committerMartin Odersky <odersky@gmail.com>2017-01-05 18:00:06 +0700
commitaa6ebe938639f07dd6f5612e645f1449f37a86eb (patch)
tree24d257c1f12d0f5cb9b76102043046e3a904d98a /library
parent42eb864dc752254fc3b8b0428570fe94aa1dafc7 (diff)
downloaddotty-aa6ebe938639f07dd6f5612e645f1449f37a86eb.tar.gz
dotty-aa6ebe938639f07dd6f5612e645f1449f37a86eb.tar.bz2
dotty-aa6ebe938639f07dd6f5612e645f1449f37a86eb.zip
Implement structural type member access
New scheme for implementing structural type member access.
Diffstat (limited to 'library')
-rw-r--r--library/src/scala/Projector.scala10
-rw-r--r--library/src/scala/reflect/Projector.scala71
2 files changed, 81 insertions, 0 deletions
diff --git a/library/src/scala/Projector.scala b/library/src/scala/Projector.scala
new file mode 100644
index 000000000..cff73f84d
--- /dev/null
+++ b/library/src/scala/Projector.scala
@@ -0,0 +1,10 @@
+package scala
+import scala.reflect.ClassTag
+import scala.annotation.implicitNotFound
+
+@implicitNotFound("no projector instance found to implement reflective access to structural type ${T}")
+trait Projector[-T] extends Any {
+ def get(receiver: T, name: String): Any
+ def getMethod(receiver: T, name: String, paramClasses: ClassTag[_]*): Any =
+ new UnsupportedOperationException("getMethod")
+}
diff --git a/library/src/scala/reflect/Projector.scala b/library/src/scala/reflect/Projector.scala
new file mode 100644
index 000000000..68240da76
--- /dev/null
+++ b/library/src/scala/reflect/Projector.scala
@@ -0,0 +1,71 @@
+package scala.reflect
+
+class Projector extends scala.Projector[Any] {
+ import Projector._
+ def get(receiver: Any, name: String): Any = {
+ val rcls = receiver.getClass
+ try {
+ val fld = rcls.getField(name)
+ fld.get(receiver)
+ }
+ catch {
+ case ex: NoSuchFieldError =>
+ getMethod(receiver, name).asInstanceOf[() => Any]()
+ }
+ }
+
+ override def getMethod(receiver: Any, name: String, paramTypes: ClassTag[_]*): Any = {
+ val rcls = receiver.getClass
+ val paramClasses = paramTypes.map(_.runtimeClass)
+ val mth = rcls.getMethod(name, paramClasses: _*)
+ paramTypes.length match {
+ case 0 => () =>
+ mth.invoke(receiver)
+ case 1 => (x0: Any) =>
+ mth.invoke(receiver, x0.asInstanceOf[Object])
+ case 2 => (x0: Any, x1: Any) =>
+ mth.invoke(receiver,
+ x0.asInstanceOf[Object],
+ x1.asInstanceOf[Object])
+ case 3 => (x0: Any, x1: Any, x2: Any) =>
+ mth.invoke(receiver,
+ x0.asInstanceOf[Object],
+ x1.asInstanceOf[Object],
+ x2.asInstanceOf[Object])
+ case 4 => (x0: Any, x1: Any, x2: Any, x3: Any) =>
+ mth.invoke(receiver,
+ x0.asInstanceOf[Object],
+ x1.asInstanceOf[Object],
+ x2.asInstanceOf[Object],
+ x3.asInstanceOf[Object])
+ case 5 => (x0: Any, x1: Any, x2: Any, x3: Any, x4: Any) =>
+ mth.invoke(receiver,
+ x0.asInstanceOf[Object],
+ x1.asInstanceOf[Object],
+ x2.asInstanceOf[Object],
+ x3.asInstanceOf[Object],
+ x4.asInstanceOf[Object])
+ case 6 => (x0: Any, x1: Any, x2: Any, x3: Any, x4: Any, x5: Any) =>
+ mth.invoke(receiver,
+ x0.asInstanceOf[Object],
+ x1.asInstanceOf[Object],
+ x2.asInstanceOf[Object],
+ x3.asInstanceOf[Object],
+ x4.asInstanceOf[Object],
+ x5.asInstanceOf[Object])
+ case 7 => (x0: Any, x1: Any, x2: Any, x3: Any, x4: Any, x5: Any, x6: Any) =>
+ mth.invoke(receiver,
+ x0.asInstanceOf[Object],
+ x1.asInstanceOf[Object],
+ x2.asInstanceOf[Object],
+ x3.asInstanceOf[Object],
+ x4.asInstanceOf[Object],
+ x5.asInstanceOf[Object],
+ x6.asInstanceOf[Object])
+ }
+ }
+}
+
+object Projector {
+ implicit def reflectiveProjector: scala.Projector[Any] = new Projector
+} \ No newline at end of file