# **LCTES 2022 Artifact**

This document provides evaluation guidelines for the following paper:

• paper #14: Implicit State Machines

It will help evaluate the following claims in the paper (Section 5):

- We implement an embedded DSL in Scala based on implicit state machines.
- The DSL examples given in the paper work as expected.
- We implement a simple microcontroller in the DSL.

# **Getting Started Guide**

The artifact is a Docker image. The reviewer is invited to install docker and make sure it is working normally. The following official guide can help with the installation of Docker: https://docs.docker.com/get-docker/.

#### 1. Get the Docker image:

You may pull the docker image with the following command:

```
1 docker pull liufengyun/lctes22:1.0
```

Or, if you already have the container image, you can import it as follows:

```
1 docker load < lctes22-1.0.tar.gz
```

# 2. Run the Docker image:

```
1 docker run -it liufengyun/lctes22:1.0
```

#### 3. Play with examples

Start the sbt:

```
1 cd /home/zeno && sbt
```

Compile the project:

```
1 compile
```

Now you can run the following example:

```
1 run asm/hello.s
```

You should be able to see "hello, world!" printed.

To exit SBT, type:

```
1 exit
```

# **Step-by-Step Instructions**

In this section, the following claims are supported by the artifact:

- We implement an embedded DSL in Scala based on implicit state machines.
- The examples given in the paper work in the paper.
- We implement a simple microcontroller in the DSL.

#### 1. Implementation of DSL

```
src/main/scala/zeno
  - lang.scala
  - core
     — Trees.scala
      - Types.scala
      Values.scala
  - examples
      Adder.scala
      - Controller.scala
      - Filter.scala
      - Misc.scala
   rewrite
      - Interpreter.scala
      - Phases.scala
      - TreeAccumulator.scala
      - TreeMap.scala
      Verilog.scala
   util
      printing.scala
      - Tracing.scala
```

The ASTs (abstract syntax trees) and compiler phases are defined in the following directories respectively:

- src/main/scala/zeno/core
- src/main/scala/zeno/rewrite

The file src/main/scala/zeno/lang.scala provides APIs for programmers to construct circuits using the DSL.

### 2. DSL Example: Adder

The source code for the example Adder can be found below:

• src/main/scala/zeno/examples/Adder.scala

First start the interactive Scala console:

```
1 cd /home/zeno && sbt console
```

Now import packages:

```
import zeno.lang.*
import zeno.examples.*
import scala.language.implicitConversions
```

We can create an instance of the adder as follows:

```
1 val a = input[Vec[2]]("a")
2 val b = input[Vec[2]]("b")
3 val circuit = Adder.adder2(a, b)
```

We can inspect the representation of the circuit as follows:

```
1 println(show(circuit))
```

We can generate the Verilog:

```
1 circuit.toVerilog("Adder", a, b)
```

We can create a simulator:

```
1 val add2 = circuit.eval(a, b)
```

Now, we can simulate the circuit with input:

```
1 add2(Value(1, 0) :: Value(0, 1) :: Nil)
2 add2(Value(1, 0) :: Value(1, 1) :: Nil)
```

To exit the Scala console, type:

```
1 :exit
```

#### 3. DSL Example: Filter

The source code for the example Filter can be found below:

• src/main/scala/zeno/examples/Filter.scala

First start the interactive Scala console:

```
1 cd /home/zeno && sbt console
```

Now import packages:

```
import zeno.lang.*
import zeno.examples.*
import scala.language.implicitConversions
```

We can create an instance of the adder as follows:

```
1 val a = input[Vec[8]]("a")
2 val circuit = Filter.movingAverage(a)
```

We can inspect the representation of the circuit as follows:

```
1 println(show(circuit))
```

We can generate the Verilog:

```
1 circuit.toVerilog("Filter", a)
```

We can create a simulator:

```
1 val avg = circuit.eval(a)
```

Now, we can simulate the circuit with input:

```
1 avg(10.toValue(8) :: Nil)
2 avg(20.toValue(8) :: Nil)
3 avg(20.toValue(8) :: Nil)
```

You should be able to see the following values:

```
1 VecV(List(0, 0, 0, 0, 0, 0, 1, 0))
2 VecV(List(0, 0, 0, 1, 0, 1, 0))
3 VecV(List(0, 0, 0, 1, 0, 0, 0, 1))
```

To exit the Scala console, type:

```
1 :exit
```

#### 4. Micro-controller

The implementation of the controller can be found in src/main/scala/zeno/examples/Filter .scala. The test assembly programs can be found in asm/.

The simplest way to test the microcontroller is to run it with simulation.

First start the interactive Scala console:

```
1 cd /home/zeno && sbt console
```

Now import packages:

```
import zeno.lang.*
import zeno.examples.controller.*
import scala.language.implicitConversions
```

Then we can run the simulation with a test assembly file as follows:

```
1 Controller.test("asm/jump.s")
```

You should be able to see the following string gets printed:

```
1 ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz
```

The method Controller.test performs the following steps:

- 1. Use the assembler to translate the test into machine code.
- 2. Contruct an instance of the microcontroller with the machine code in the ROM for instructions.
- 3. Create a circuit simulator for the microcontroller.
- 4. Drive the simulator bus with a simulated memory.
- 5. Accumulate all memory stores to address 0.
- 6. When the program finishes, return the accumulated writes to address 0.

You can play with other assembly programs located under asm/.

We can generate Verilog for the micro-controller as follows:

```
val busIn = input[Controller.BusIn]("busIn")
val instructions = Assembler.assemble("asm/jump.s")
val circuit = Controller.processor(instructions, busIn)
circuit.toVerilog("Controller", busIn)
```

The generated code is a little large, thus the output is truncated.

To exit the Scala console, type:

1 :exit

# Afterword

Thank you for getting to the end of the guideline.

If you have any questions, please feel free to contact us.