Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Readme update #14

Merged
merged 5 commits into from
Aug 14, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
278 changes: 265 additions & 13 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,34 +1,286 @@
# Tydi-Chisel

Tydi-Chisel allows you to transfer simple to very complex nested variable length sequences between components over a stream bus in a structured way, using familiar Chisel concepts.

## What is Tydi?
Tydi-Chisel is an implementation of Tydi interfaces and concepts in Chisel.

[Tydi](https://abs-tudelft.github.io/tydi) (Typed dataflow interface) is an open specification for streaming dataflow designs in digital circuits, allowing designers to express how composite and variable-length data structures are transferred over streams using clear, data-centric types.

[Chisel](https://www.chisel-lang.org/) (Constructing Hardware in a Scala Embedded Language) is a high-level open-source hardware description language (HDL).

With Tydi as data-streaming communication flow specification and Chisel as flexible implementation, an interface-driven design method can be followed.

Tydi-Chisel is an implementation of Tydi interfaces and concepts in Chisel.

Concretely, it contains:
## Why Tidy?

Developing hardware is notoriously difficult. Many solutions exist to lighten the load of developing implementations for hardware components. Yet, a gap exists between the easy use of complex nested (sequence) data structures in software and trying to work with them in hardware.

Tydi aims to solve this issue by proposing an open standard for streams of arbitrary variable length data-structures. A data-structure is created by nesting `Bits`, `Group`, `Union` and `Stream` elements. See the following table for the meaning of the used terms.

<table style="text-align: left" id="tydi-terms">
<thead>
<tr style="border-bottom-width: 3px">
<th style="border-right-width: 3px;">Term</th>
<th>Type</th>
<th>Software equivalent</th>
<th>Chisel equivalent</th>
<th>Meaning</th>
</tr>
</thead>
<tbody>
<tr>
<th scope="row" style="border-right-width: 3px;">Null</th>
<td>Logical type</td>
<td><code>Null</code></td>
<td><code>Bits(0)</code></td>
<td>Empty data</td>
</tr>
<tr>
<th scope="row" style="border-right-width: 3px;">Bit</th>
<td>Logical type</td>
<td>Any primary datatype</td>
<td>Ground hardware type</td>
<td>Primary datatype of <code>x</code> bits</td>
</tr>
<tr>
<th scope="row" style="border-right-width: 3px;">Group</th>
<td>Logical type</td>
<td><code>Struct</code>/<code>dict</code>/<code>map</code></td>
<td><code>Bundle</code></td>
<td>Aggregate of several logic types</td>
</tr>
<tr>
<th scope="row" style="border-right-width: 3px;">Union</th>
<td>Logical type</td>
<td><code>Union</code></td>
<td><code>Bundle</code> with tag</td>
<td>“pick one” of several logic types</td>
</tr>
<tr style="border-bottom-width: 2px">
<th scope="row" style="border-right-width: 3px;">Stream</th>
<td>Logical type</td>
<td>Bus to transport sequence of instance</td>
<td>–</td>
<td>Specify how to transport logic type</td>
</tr>
<tr>
<th scope="row" style="border-right-width: 3px;">Streamlet</th>
<td>Hardware element</td>
<td><code>Interface</code></td>
<td><code>Trait</code> with IO defs</td>
<td>IO specification of component</td>
</tr>
<tr>
<th scope="row" style="border-right-width: 3px;">Impl</th>
<td>Hardware element</td>
<td><code>Class</code> with functionality</td>
<td><code>Module</code></td>
<td>Inner structure of component</td>
</tr>
</tbody>
</table>

By being a super-set of ready-valid communication, like Chisel's `DecoupledIO`, and the AXI-Stream standard, Tydi-interfaces stay close to existing streaming implementations.

See the work in [publications](#publications) for more details.

## Example
### Using Tydi data-structures and streams
The following example provides an overview of the declaration of a nested Tydi data structure and its usage in streams.

In software, the declared data structure would look like

```json
[
{
"time": 123456789,
"nested": {
"a": 5,
"b": true
},
"message": [
"Hello", "this", "is", "a", "string."
]
}
]
```

```scala
// Declaration of the data-structure
class Character extends BitsEl(8.W)

class NestedBundle extends Group {
val a: UInt = UInt(8.W)
val b: Bool = Bool()
}

class TimestampedMessageBundle extends Group {
val time: UInt = UInt(64.W)
val nested: NestedBundle = new NestedBundle
// Create a 2D sub-stream of character data with 3 data-lanes
val message = new PhysicalStreamDetailed(Character, n = 3, d = 2, c = 7)
}

// Declaration of the module
class TimestampedMessageModule extends TydiModule {
// Create Tydi logical stream object
val stream = PhysicalStreamDetailed(new TimestampedMessageBundle, 1, c = 7)

// Create and connect physical streams as IO
// following the Tydi standard with concatenated data bitvector
val tydi_port_top = stream.toPhysical
val tydi_port_child = stream.el.message.toPhysical

// → Assign values to logical stream group elements directly
stream.el.time := System.currentTimeMillis().U
stream.el.nested.a := 5.U
stream.el.nested.b := true.B
stream.el.message.data(0).value := 'H'.U
stream.el.message.data(1).value := 'e'.U
stream.el.message.data(2).value := 'l'.U
...
}
```
See the [timestamped_message](library/src/main/scala/nl/tudelft/tydi_chisel/examples/timestamped_message/TimestampedMessage.scala) file for the full example. This file is just for showing the syntax of declaring and using Tydi-interfaces. It does not contain an implementation. For that, look at the [pipeline examples](library/src/main/scala/nl/tudelft/tydi_chisel/examples/pipeline). There you will also find advanced syntax for the creation of modules and chaining of submodules through their streams:

```scala
...

class PipelineExampleModule extends SimpleProcessorBase(new NumberGroup, new Stats) {
out := in.processWith(new NonNegativeFilter).processWith(new Reducer)
}
```

### Testing a TydiModule
The code below shows a snippet from the [pipeline example test](testing/src/test/scala/nl/tudelft/tydi_chisel/pipeline/PipelineExampleTest.scala) code. It shows how to use the TydiStreamDriver to easily enqueue and dequeue data to sink and from source streams. Currently, this is only available for [ChiselTest](https://github.com/ucb-bar/chiseltest), and not ChiselSim.
```scala
// Import stuff

class PipelineExampleTest extends AnyFlatSpec with ChiselScalatestTester {
behavior of "PipelineExample"

class PipelineWrap extends TydiTestWrapper(new PipelineExampleModule, new NumberGroup, new Stats)

it should "process a sequence" in {
// Test code
test(new PipelineWrap) { c =>
// Initialize signals
c.in.initSource()
c.out.initSink()

// Generate list of random numbers
val nums = randomSeq(n = 50)
val stats = processSeq(nums) // Software impl.

// Test component
parallel({
for ((elem, i) <- nums.zipWithIndex) {
c.in.enqueueElNow(_.time -> i.U, _.value -> elem.S)
}
c.in.enqueueEmptyNow(last = Some(c.in.lastLit(0 -> 1.U)))
}, {
c.out.waitForValid()
// Utility for comprehensively printing stream state
println(c.out.printState(statsRenderer))
c.out.expectDequeue(_.min -> stats.min.U, _.max -> stats.max.U, _.sum -> stats.sum.U, _.average -> stats.average.U)
})
}
}
}
```
Look through the [test examples](testing/src/test/scala/nl/tudelft/tydi_chisel) for all functionality and syntax.


## Getting started

### Acquire the lib
To get started with Tydi-Chisel, you can either run it in your local development environment, or use the [Tydi-tools](https://github.com/abs-tudelft/Tydi-tools) docker container.

For usage with `sbt` or `scala-cli`, the project package must first be built. Make sure you have `git`, `java`, and `sbt` installed and available in your path.

```shell
git clone https://github.com/abs-tudelft/Tydi-Chisel.git
sbt publishLocal
```

### Use the lib
After this, the library can be added to your project's dependencies in your `build.sbt` like so.

```scala
libraryDependencies += "nl.tudelft" %% "tydi-chisel" % "0.1.0"
libraryDependencies += "nl.tudelft" %% "tydi-chisel-test" % "0.1.0" % Test
```

When using `scala-cli`, use the following directive:
```scala
//> using dep "nl.tudelft::tydi-chisel::0.1.0"
```

Then, one can import the functionality, and start to write Tydi-Chisel code!

```scala
import nl.tudelft.tydi_chisel._
```

Look through the [examples](library/src/main/scala/nl/tudelft/tydi_chisel/examples) for inspiration for the functionality and syntax.

## Features

Concretely, this project contains:

- Expressing Tydi stream interfaces in Chisel
- Including nested stream support
- Being able to work with the detailed bundles inside your components
- Compliant with Tydi-standard for communication with components created outside of Chisel
- Compliant with [Tydi-standard](https://abs-tudelft.github.io/tydi/specification/physical.html) for communication with components created outside of Chisel
- Simple stream connection
- Helper functions for common signal use-cases
- A [Tydi-lang-2-Chisel](https://github.com/ccromjongh/tydi-lang-2-chisel) transpiler
to convert Tydi interfaces described in Tydi-lang-2 to Chisel code utilizing the Tydi-Chisel library.
- A reverse-transpiler to share your Tydi-Chisel code as Tydi-lang-2 code
- A stream-processing component chaining syntax
- Testing utilities
- `chisel-test` driver for Tydi stream interfaces.
- Helper components
- A _complexity converter_ that can convert any incoming stream to the lowest source complexity
\* Not all complexity conversion steps are included yet
- A _stream complexity converter_ that can convert any incoming stream to the lowest source complexity
- A _multi-processor_ or _interleaving_ component that splits a multi-lane stream into multiple single lane streams for easy processing, merging them again after.
- A [Tydi-lang-2-Chisel](https://github.com/ccromjongh/tydi-lang-2-chisel) transpiler\
to convert Tydi interfaces described in Tydi-lang-2 to Chisel code utilizing the Tydi-Chisel library.
- A reverse-transpiler to share your Tydi-Chisel code as Tydi-lang-2 code

### Future work

- Projects
- Provide more worked out real-life examples
- Create interoperability with [Fletcher](https://github.com/abs-tudelft/fletcher)
- Investigate adoption of other hardware description languages
- Library
- Stream compatibility checks
- Better Union support
- Interoperability with other streaming standards
- Improved error handling and design rule checks
- Additional helper components
- Testing and debugging
- Change testing front-end to the new ChiselSim framework.\
This can currently not happen yet as ChiselSim still lacks support for `parallel` calls, which makes it hard to do asynchronous component testing.
- Improved tooling for enqueueing and dequeueing of test data
- Stream protocol compliance verification tooling
- Enhanced support for [TyWaves](https://github.com/rameloni/tywaves-chisel-demo)
- Provide more time-domain information

## Publications

- _C. Cromjongh, Y. Tian, Z. Al-Ars and H. P. Hofstee_
**Enabling Collaborative and Interface-Driven Data-Streaming Accelerator Design with Tydi-Chisel**
2023 IEEE Nordic Circuits and Systems Conference (NorCAS), 1 November 2023. DOI: [10.1109/NorCAS58970.2023.10305451](https://doi.org/10.1109/NorCAS58970.2023.10305451).

## License

Tydi and this library are licensed under the [Apache-2.0](https://www.apache.org/licenses/LICENSE-2.0) license. See [LICENSE](./LICENSE).

## Related Tools
* [tydi-lang-2](https://github.com/twoentartian/tydi-lang-2)
A tool for converting Tydi-lang code to a `json` representation of all logic types, streamlets, and implementations
* [tydi-lang-2-chisel](https://github.com/ccromjongh/Tydi-lang-2-chisel)
Consequtively converts the `json` tydi definitions to Chisel code utilising Tydi-Chisel
* [Tydi repository](https://github.com/abs-tudelft/tydi)
* [Tydi-Lang-2](https://github.com/twoentartian/tydi-lang-2)\
A tool for converting Tydi-lang code to a `json` representation of all logic types, streamlets, and implementations.
* [Tydi-Lang-2-Chisel](https://github.com/ccromjongh/Tydi-lang-2-chisel)\
Consequtively converts the `json` tydi definitions to Chisel code utilising Tydi-Chisel.
* [TyWaves](https://github.com/rameloni/tywaves-chisel-demo)\
A type based waveform viewer for Chisel and Tydi-Chisel, combining CIRCT debug output with the [Surfer](https://gitlab.com/surfer-project/surfer) waveform viewer.
* [Tydi repository](https://github.com/abs-tudelft/tydi)\
Contains Tydi standard documentation and first implementation of a compiler.
Original file line number Diff line number Diff line change
Expand Up @@ -109,7 +109,7 @@ class NumberModuleIn extends TydiModule {
}*/

class PipelineExampleModule extends SimpleProcessorBase(new NumberGroup, new Stats) {
out := in.processWith(new NonNegativeFilter).processWith(new Reducer())
out := in.processWith(new NonNegativeFilter).processWith(new Reducer)
}

object PipelineExample extends App {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,7 @@ import chisel3._
import circt.stage.ChiselStage.{emitCHIRRTL, emitSystemVerilog}
import nl.tudelft.tydi_chisel._

////// End lib, start user code //////

// Declaration of the data-structure
class NestedBundle extends Union(2) {
val a: UInt = UInt(8.W)
val b: Bool = Bool()
Expand All @@ -28,6 +27,7 @@ class TimestampedMessageBundle extends Group {
val message = new PhysicalStreamDetailed(BitsEl(charWidth), n = 3, d = 1, c = 7)
}

// Declaration of the module
class TimestampedMessageModuleOut extends TydiModule {
private val timestampedMessageBundle = new TimestampedMessageBundle // Can also be inline

Expand Down
Loading