In this lab, we start from Lab 20 (the single-channel design) and create a sub-design that implements the logic for a single channel. We then instantiate it multiple times in the top-level design using multiple sub-design instances.

However, we also move the registers and oscilloscopes inside the sub-design itself, rather than keeping them in the top design. This approach yields a cleaner top-level schematic and gives each sub-design instance its own independent register file and oscilloscope.


1. Sub-Design Setup

1.1 Sub-Design with Registers and Oscilloscope

Within the sub-design (let’s call it core_block), we include:

  • Logic for a single channel (trigger, baseline restoration, QDC, etc.).
  • Registers for configuration (e.g., THRESHOLD, GAIN, INT_SAMPLES).
  • Oscilloscope for monitoring signals.

Unlike previous labs, we place all these registers and the oscilloscope inside the sub-design.

Subdesign block diagram
Subdesign block diagram

1.2 Address Assignment for the Sub-Design

While the sub-design is open, we assign addresses for:

  • Registers (e.g., THRESHOLD, GAIN, INT_SAMPLES).
  • Oscilloscope (for data capture).

Because they are inside a sub-design, these addresses are relative to the sub-design’s base address. When we have multiple instances of the same sub-design, each instance will have its own base address, which is added to these relative offsets.

Sub design memory assignment
Sub design memory assignment

By clicking Auto Assign, the tool automatically sets addresses for the registers and oscilloscope, and it allocates address space for the sub-design according to the total size of its internal components.

Auto assign address to sub-design elements
Auto assign address to sub-design elements


2. Top-Level Integration

2.1 Placing the Sub-Design and Defining Base Addresses

In the top design, we place our core_block sub-design multiple times to create multiple channels. Each instance has:

  • A unique name (Designator) to avoid name conflicts.
  • A base address that determines where its internal registers and oscilloscope appear in the global address map.

Import multichannel design
Import multichannel design

Make sure each instance has a different name (e.g., core_block_1 vs. core_block_2) and a different base address if you want separate memory maps.

Create two instances of the sub-design for the two channels
Create two instances of the sub-design for the two channels

After placing multiple instances, each channel (sub-design instance) has its own registers and its own oscilloscope, making the top design cleaner:

Top level design diagram
Top level design diagram

Here, we assign the base addresses to each core_block instance:

Address assignment
Address assignment


3. Testing the Design

3.1 Resource Explorer

Open the Resource Explorer and add the registers from the sub-design to the configuration table. Notice how each register’s name is prefixed with:

  • The sub-design instance name.
  • The channel or instance number.

Hence, if you have multiple core_block instances, you will see multiple copies of the same register name but with different addresses.

Resource Explorer tree
Resource Explorer tree

You can configure these registers independently, since each channel’s base address is different.

Configuration register for the two channels
Configuration register for the two channels

3.2 Independent Oscilloscopes

Each sub-design instance includes its own oscilloscope. Thus, you can open two separate scope windows and configure triggers separately for each channel:

Two oscilloscope for the two challes
Two oscilloscope for the two challes

For instance, set Trigger Channel 1 with a threshold of 2500 on the first sub-design’s oscilloscope, and do the same (or different) for the second sub-design’s oscilloscope. The integration time is different for each channel as you can see in D1

3.3 Single List Block

Despite having multiple channels, we still have one List block in the top design that collects the data. Each sub-design instance sends data to the Round Robin Arbiter, which then outputs to this single List block.

List data download
List data download

Alternating data (last byte) for each channel is shown in the List block dump on file:

Decoded HXD dump of the file
Decoded HXD dump of the file

4. Conclusion

By embedding registers and oscilloscopes within each sub-design instance:

  • We achieve a cleaner top-level schematic (fewer blocks and wires on top).
  • Each channel has independent parameter registers and independent oscilloscope capture capability.
  • The sub-design’s memory addresses are mapped relative to each instance’s base address, allowing multiple channels without collisions.

This concludes Lab 22. You now have a multichannel design where each channel (sub-design instance) contains its own registers and oscilloscope, offering a modular and well-organized system.