Skip to the content.

Compiling Spatial

# Goto Spatial main directory
cd ~/spatial-lang
# Goto the app directory of Spatial
cd ~/spatial-lang/apps/src
# Simulate an app foo with Scala backend
cd ~/spatial-lang
bin/spatial foo --sim
# Run Scala sim
cd ~/spatial-lang/
./foo.sim "arg0 arg1 ..."
# Synthesize an app foo with VCS backend, with instrumentation and retiming turned on
bin/spatial foo --synth --instrumentation --retime
# View the generated design
cd ~/spatial-lang/gen/foo
ls
# Run VCS sim
make vcs && bash run.sh arg0 arg1 ...
# Remove VCS files and prepare for synthesis
make vcs-clean
# Prepare bitstream that runs on Arria10 SoC:
make arria10 | tee make.log

Spatial Flags

# Generates design for simulation
--sim
# Generates design for synthesis
--synth
# Inserts instrumentation registers for synthesis. You can use this option for improving performance.
--instrumentation
# Inserts retiming registers in your design. This helps your design to meet timing constraints.
--retime
# Splits your design file into multiple smaller files. Usually you will not need this feature; however if your design becomes quite large (e.g. State-of-the-art neural networks), you will need to enable this flag. N is usually set to 5.
--multifile=N

Basic Spatial Template

import spatial.dsl._
import org.virtualized._

object Lab1Part1RegExample extends SpatialApp {

  // Types
  type T = Int

  @virtualize
  def main() {
    // Your host code here
    ...

    Accel {
      // Your accelerator code here
      ...
    }
  }

  // Define a function with return
  @virtualize
  def functionWithReturn[T:Type:Num](a: Array[T], b: T, c: SRAM[T], ...) = { // function body }

  // Define a function without return
  @virtualize
  def functionWithoutReturn[T:Type:Num](a: Array[T], b: T, c: SRAM[T], ...) { // function body }
}

Basic APIs of Spatial Memory Elements

Let’s say that the data type is T.

// Host <-> Accel Registers
val argIn = ArgIn[T]
val argOut = ArgOut[T]

// Register initialized with value 0
val reg = Reg[T](0)

// DRAM
val dm = DRAM[T](N) 
setMem(dm, data)
val readData = getMem(dm)

// SRAM
val sm = SRAM[T](N)
sm load dm(k::k+n0)
dm(k::k+n0) tore sm

// FIFO
val f1 = FIFO[T](tileSize)
f1 load dram(i::i+tileSize)
f1.enq(data)
f1.deq(data)
f1.peek()

// FILO
val f1 = FILO[T](tileSize)
f1 load dram(i::i+tileSize) 
f1.push(data) 
f1.pop(data)
f1.peek()

Controller Examples

# Foreach, non-sequential. Compiler will attempt to parallelize / pipeline the body
Foreach(N by n) { i =>
  ...
}

// Foreach, sequential
Sequential.Foreach(N by n) { i =>
  ...        
}

// Reduce, non-sequential
val accum = Reg[T](0) // create a register to hold the reduced result
Reduce(accum)(N by n) { i =>
  // map body	
}{// binary operator}

// Reduce, sequential
val accum = Reg[T](0) // create a register to hold the reduced result
Sequential.Reduce(accum)(N by n) { i =>
  // map body	
}{// binary operator}

// Fold, non-sequential
val accum = Reg[T](0) // create a register to hold the reduced result
Fold(accum)(N by n) { i =>
  // map body	
}{// binary operator}

// Fold, sequential
val accum = Reg[T](0) // create a register to hold the reduced result
Sequential.Reduce(accum)(N by n) { i =>
  // map body	
}{// binary operator}

Useful Screen Commands

Let’s say you want to start a screen s.

# Start s
screen -S s
# Detach from s
Ctrl-A D
# Show session name
Ctrl-A sessionname
# List all the screen sessions
screen -ls
# Resume s
screen -r s
# Reattach to an attached session
screen -r -d s

ssh and scp

# ssh into tucson
ssh -Y username@tucson.stanford.edu
# scp a file f to /foo directory on the board. Let's say the board name is arria10.
scp f username@arria10.stanford.edu:/foo/
# scp a directory to /foo directory on the board.
scp -r f username@arria10.stanford.edu:/foo/