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

Clean serialization #92

Merged
merged 10 commits into from
Sep 19, 2024
11 changes: 11 additions & 0 deletions doc/components/serialization.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# Serialization / Deserialization

ROHD-HCL implements a `Serializer` and `Deserializer` set of components that enable converting wide structures to a serialized narrow stream of data and vice-versa.

## Serializer

The `Serializer` is a module that accepts a wider `LogicArray` `deserialized` for input data and optionally a `Logic` `readyIn` to allow for pausing serialization. While `readyIn` is high, the `Serializer` sequentially outputs chunks of data on the Logic `serialized` output until the entire `LogicArray` `deserialized` has been transferred to the output. At that point the `Serializer` raises Logic `done`. This process will continue until the Logic `readyIn` is lowered, allowing for back-to-back transfers of wide data over the Logic `serialized` stream. The number of serialization steps in the current transfer is available in Logic `count`. Lowering `readyIn` will pause the transfer and raising it will continue from where it paused.

## Deserializer

The `Deserializer` is a module that accepts a `Logic` `serialized` stream over a pre-defined `length` number of clocks. It outputs a `LogicArray` `deserialized` of the same length of Logic words that match the width of Logic `serialized`. Deserialization runs while Logic `enable` is high, and Logic `validOut` is emitted when deserialization is complete. This process will continue when Logic `enable` is high, allowing for back-to-back deserialization transfers of a narrow stream of data into wider (`length`) `LogicArray`s during `length` number of serialization steps (clocks while `enable` is high) for each transfer. The number of serialization steps in the current transfer is available in Logic `count`. Lowering `enable` will pause the transfer and raising it will continue from where it paused.
2 changes: 2 additions & 0 deletions lib/rohd_hcl.dart
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ export 'src/arithmetic/arithmetic.dart';
export 'src/binary_gray.dart';
export 'src/component_config/component_config.dart';
export 'src/count.dart';
export 'src/deserializer.dart';
export 'src/edge_detector.dart';
export 'src/encodings/encodings.dart';
export 'src/error_checking/error_checking.dart';
Expand All @@ -16,6 +17,7 @@ export 'src/interfaces/interfaces.dart';
export 'src/memory/memories.dart';
export 'src/models/models.dart';
export 'src/rotate.dart';
export 'src/serializer.dart';
export 'src/shift_register.dart';
export 'src/sort.dart';
export 'src/summation/summation.dart';
Expand Down
69 changes: 69 additions & 0 deletions lib/src/deserializer.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
// Copyright (C) 2024 Intel Corporation
// SPDX-License-Identifier: BSD-3-Clause
//
// deserializer.dart
// A deserialization block, deserializing narrow input data onto a wide channel.
//
// 2024 August 27
// Author: desmond Kirkpatrick <[email protected]>

import 'package:meta/meta.dart';
import 'package:rohd/rohd.dart';
import 'package:rohd_hcl/rohd_hcl.dart';

/// Aggregates data from a serialized stream
class Deserializer extends Module {
/// Clk input
@protected
Logic get clk => input('clk');

/// Reset input
@protected
Logic get reset => input('reset');

/// Run deserialization whenever [enable] is true
@protected
Logic? get enable => input('enable');
desmonddak marked this conversation as resolved.
Show resolved Hide resolved

/// Serialized input, one data item per clock
Logic get serialized => input('serialized');
desmonddak marked this conversation as resolved.
Show resolved Hide resolved

/// Aggregated data output
LogicArray get deserialized => output('deserialized') as LogicArray;

/// Valid out when data is reached
Logic get done => output('done');

/// Return the count as an output
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
@protected
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
Logic get count => output('count');

/// Build a Deserializer that takes serialized input [serialized]
/// and aggregates it into one wide output [deserialized], one element per
/// clock while [enable] (if connected) is high, emitting [done] when
/// completing the filling of wide output [deserialized].
Deserializer(Logic serialized, int length,
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
{required Logic clk,
required Logic reset,
Logic? enable,
super.name = 'Deserializer'}) {
clk = addInput('clk', clk);
reset = addInput('reset', reset);
if (enable != null) {
enable = addInput('enable', enable);
} else {
enable = Const(1);
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
}
serialized = addInput('serialized', serialized, width: serialized.width);
final cnt = Counter.simple(
clk: clk, reset: reset, enable: enable, maxValue: length - 1);
addOutput('count', width: cnt.width) <= cnt.count;
addOutput('done') <= cnt.overflowed;
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
addOutputArray('deserialized',
dimensions: [length], elementWidth: serialized.width) <=
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
[
for (var i = 0; i < length; i++)
flop(clk, reset: reset, en: enable & count.eq(i), serialized)
].swizzle();
}
}
92 changes: 92 additions & 0 deletions lib/src/serializer.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
// Copyright (C) 2024 Intel Corporation
// SPDX-License-Identifier: BSD-3-Clause
//
// serializer.dart
// A serialization block, serializing wide input data onto a narrower channel.
//
// 2024 August 27
// Author: desmond Kirkpatrick <[email protected]>

import 'package:meta/meta.dart';
import 'package:rohd/rohd.dart';
import 'package:rohd_hcl/rohd_hcl.dart';

/// Serializes wide aggregated data onto a narrower serialization stream
class Serializer extends Module {
/// Clk input
@protected
Logic get clk => input('clk');
desmonddak marked this conversation as resolved.
Show resolved Hide resolved

/// Reset input
@protected
Logic get reset => input('reset');

/// Allow serialization onto the output stream when [enable] is true
@protected
Logic? get enable => input('enable');
desmonddak marked this conversation as resolved.
Show resolved Hide resolved

/// Return the count as an output
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
Logic get count => output('count');

/// Return [done] = true when we have processed [deserialized] completely
Logic get done => output('done');

/// Aggregated data to serialize out
LogicArray get deserialized => input('deserialized') as LogicArray;
desmonddak marked this conversation as resolved.
Show resolved Hide resolved

/// Serialized output, one data item per clock
Logic get serialized => output('serialized');

/// Build a Serializer that takes the array [deserialized] and sequences it
/// onto the [serialized] output, one element per clock while [enable]
/// is high (if connected). If [flopInput] is true, the
/// [Serializer] is configured to latch the input data and hold it until
/// [done] is asserted after the full [deserialized] is transferred.
Serializer(LogicArray deserialized,
{required Logic clk,
required Logic reset,
Logic? enable,
bool flopInput = false,
super.name = 'Serializer'}) {
clk = addInput('clk', clk);
reset = addInput('reset', reset);
if (enable != null) {
enable = addInput('enable', enable);
} else {
enable = Const(1);
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
}
deserialized = addInputArray('deserialized', deserialized,
dimensions: deserialized.dimensions,
desmonddak marked this conversation as resolved.
Show resolved Hide resolved
elementWidth: deserialized.elementWidth);
addOutput('serialized', width: deserialized.elementWidth);
addOutput('count', width: log2Ceil(deserialized.dimensions[0]));
addOutput('done');

final cnt = Counter.simple(
clk: clk,
reset: reset,
enable: enable,
maxValue: deserialized.elements.length - 1);

final latchInput = enable & ~done;
count <=
(flopInput
? flop(clk, reset: reset, en: enable, cnt.count)
: cnt.count);

final dataOutput =
LogicArray(deserialized.dimensions, deserialized.elementWidth);
for (var i = 0; i < deserialized.elements.length; i++) {
dataOutput.elements[i] <=
(flopInput
? flop(
clk, reset: reset, en: latchInput, deserialized.elements[i])
: deserialized.elements[i]);
}
serialized <= dataOutput.elements.selectIndex(count);
done <=
(flopInput
? flop(clk, reset: reset, en: enable, cnt.equalsMax)
: cnt.equalsMax);
}
}
Loading
Loading