summaryrefslogtreecommitdiff
path: root/test/pending/run/reify_brainf_ck.scala
blob: 2af3bca1c763c51252f03eca6bbdfc6eb8148d59 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
import scala.reflect.runtime.universe._
import scala.tools.reflect.Eval

object Test extends App {
  reify {
    import scala.annotation._

    trait Func[T] {
        val zero: T
        def inc(t: T): T
        def dec(t: T): T
        def in: T
        def out(t: T): Unit
    }

    object ByteFunc extends Func[Byte] {
      override val zero: Byte = 0
      override def inc(t: Byte) = ((t + 1) & 0xFF).toByte
      override def dec(t: Byte) = ((t - 1) & 0xFF).toByte
      override def in: Byte = readByte
      override def out(t: Byte) { print(t.toChar) }
    }

    case class Tape[T](left: List[T], cell: T, right: List[T])(implicit func: Func[T]) {
      private def headOf(list:List[T]) = if (list.isEmpty) func.zero else list.head
      private def tailOf(list:List[T]) = if (list.isEmpty) Nil else list.tail
      def isZero = cell == func.zero
      def execute(ch: Char) = (ch: @switch) match {
       case '+' => copy(cell = func.inc(cell))
       case '-' => copy(cell = func.dec(cell))
       case '<' => Tape(tailOf(left), headOf(left), cell :: right)
       case '>' => Tape(cell :: left, headOf(right), tailOf(right))
       case '.' => func.out(cell); this
       case ',' => copy(cell = func.in)
       case '[' | ']' => this
       case _ => error("Unexpected token: " + ch)
      }
    }

    object Tape {
      def empty[T](func: Func[T]) = Tape(Nil, func.zero, Nil)(func)
    }

    class Brainfuck[T](func:Func[T]) {

      def execute(p: String) {
        val prog = p.replaceAll("[^\\+\\-\\[\\]\\.\\,\\>\\<]", "")

        @tailrec def braceMatcher(pos: Int, stack: List[Int], o2c: Map[Int, Int]): Map[Int,Int] =
          if(pos == prog.length) o2c else (prog(pos): @switch) match {
            case '[' => braceMatcher(pos + 1, pos :: stack, o2c)
            case ']' => braceMatcher(pos + 1, stack.tail, o2c + (stack.head -> pos))
            case _ => braceMatcher(pos + 1, stack, o2c)
          }

        val open2close = braceMatcher(0, Nil, Map())
        val close2open = open2close.map(_.swap)

        @tailrec def ex(pos:Int, tape:Tape[T]): Unit =
          if(pos < prog.length) ex((prog(pos): @switch) match {
              case '[' if tape.isZero => open2close(pos)
              case ']' if ! tape.isZero => close2open(pos)
              case _ => pos + 1
            }, tape.execute(prog(pos)))

        println("---running---")
        ex(0, Tape.empty(func))
        println("\n---done---")
      }
    }

    val bf = new Brainfuck(ByteFunc)
    bf.execute(""">+++++++++[<++++++++>-]<.>+++++++[<++
                  ++>-]<+.+++++++..+++.[-]>++++++++[<++++>-]
                  <.#>+++++++++++[<+++++>-]<.>++++++++[<++
                  +>-]<.+++.------.--------.[-]>++++++++[<++++>
                  -]<+.[-]++++++++++.""")
  }.eval
}