Министерство науки и высшего образования Российской Федерации Санкт-Петербургский политехнический университет Петра Великого Институт компьютерных наук и технологий

| « <u> </u> |        | 2022 г.        |
|------------|--------|----------------|
|            |        | _ В.М. Ицыксон |
| дире       | ктор Е | ВШИСиСТ        |
| Рабо       | та доп | ущена к защите |

#### ВЫПУСКНАЯ КВАЛИФИКАЦИОННАЯ РАБОТА БАКАЛАВРА

# РАЗРАБОТКА УПРАВЛЯЮЩЕГО УСТРОЙСТВА КАЛИБРАТОРА СУБНАНОСЕКУНДНОЙ СИНХРОНИЗАЦИИ

по направлению 03.09.01 «Информатика и вычислительная техника» по образовательной программе 03.09.01\_01 «Вычислительные машины, комплексы, системы и сети»

Выполнил Д.В. Пешков студент гр. 3530101/80101 < подпись> Руководитель К.Т.Н., <*nodnucь*> А.А. Лавров доцент, Консультант А.А. Антонов <подпись> ассистент Консультант А.Г. Новопашенный по нормоконтролю <подпись>

> Санкт-Петербург 2022

# САНКТ-ПЕТЕРБУРГСКИЙ ПОЛИТЕХНИЧЕСКИЙ УНИВЕРСИТЕТ ПЕТРА ВЕЛИКОГО

#### Институт компьютерных наук и технологий

| УTЕ              | ВЕРЖДА   | М            |  |  |  |  |
|------------------|----------|--------------|--|--|--|--|
| директор ВШИСиСТ |          |              |  |  |  |  |
|                  |          | В.М. Ицыксон |  |  |  |  |
| <b>«</b>         | <b>»</b> | 2022г.       |  |  |  |  |

#### ЗАДАНИЕ

## на выполнение выпускной квалификационной работы

студенту Пешкову Даниилу Валерьевичу гр. 3530101/80101

- 1. Тема работы: Разработка управляющего устройства калибратора субнаносекундной синхронизации.
- 2. Срок сдачи студентом законченной работы: 23.06.2022.
- 3. Исходные данные по работе: Техническая документация на систему суб-наносекундной синхронизации «White Rabbit Project».
- 4. Содержание работы (перечень подлежащих разработке вопросов):
  - 4.1. Изучить техническую документацию по процедуре калибровки узлов сети White Rabbit.
  - 4.2. Изучить техническую документацию на используемые в калибраторе электронные узлы.
  - 4.3. Портировать используемые открытые модули под целевую ПЛИС.
  - 4.4. Реализовать модули для проведения измерений при помощи стробирования.
  - 4.5. Спроектировать систему на кристалле и интегрировать в неё используемые аппаратные модули.
  - 4.6. Написать драйвера для используемой периферии.
  - 4.7. Протестировать и отладить полученное управляющее устройство на опытном образце калибратора, при необходимости осуществить доработку.
- 5. Перечень графического материала (с указанием обязательных чертежей): Отсутствует.
- 6. Консультанты по работе:

- 6.1. Ассистент, А.А. Антонов.
- 6.2. А.Г. Новопашенный (нормоконтроль).
- 7. Дата выдачи задания: <u>06.05.2022</u>.

Студент Д.В. Пешков

 Руководитель ВКР
 А.А. Лавров

 Консультант
 А.А. Антонов

 Задание принял к исполнению 06.05.2022

#### РЕФЕРАТ

На 33 с., 28 рисунков, 0 таблиц, 2 приложения

КЛЮЧЕВЫЕ СЛОВА: КАЛИБРУЮЩЕЕ УСТРОЙСТВО, ВЫСОКОТОЧ-НАЯ СИНХРОНИЗАЦИЯ, СТРОБИРОВАНИЕ. .

Тема выпускной квалификационной работы: «Разработка управляющего устройства калибратора субнаносекундной синхронизации».

В работе рассмотрен подход к калибровке синхронизации распределённых систем потоковой обработки данных. Рассмотрен способ снятия осциллограмм сигналов при помощи стробирования. Разработано устройство для автоматизация процесса калибровки для средств высокоточной синхронизации распределенных систем потоковой обработки данных

#### **ABSTRACT**

33 pages, 28 figures, 0 tables, 2 appendices

KEYWORDS: CALIBRATION DEVICE, HIGH-PRECISION SYNCHRONIZATION, STROBING, WHITE RABBIT.

The subject of the graduate qualification work is «Title of the thesis».

The paper considers an approach to calibration of synchronization of distributed streaming data processing systems. A method of removing waveforms of signals using strobing is considered. A device has been developed to automate the calibration process for means of high-precision synchronization of distributed streaming data processing systems.

# СОДЕРЖАНИЕ

| Введение                                                         | 6  |
|------------------------------------------------------------------|----|
| Глава 1. Анализ предметной области                               | 8  |
| 1.1. White Rabbit                                                | 8  |
| 1.2. Калибровка                                                  | 8  |
| 1.3. Актуальность                                                | 11 |
| 1.4. Стробоскопический осциллограф                               | 11 |
| Глава 2. Аппаратная платформа и алгоритм проведения измерений    | 13 |
| 2.1. Измерительный канал                                         | 14 |
| 2.2. Алгоритм проведения измерений                               | 15 |
| 2.3. Пороговое напряжение сравнения                              | 16 |
| 2.4. Линия задержки                                              | 16 |
| 2.5. Компаратор                                                  | 17 |
| Глава 3. Разработка аппаратного описания управляющего устройства | 19 |
| 3.1. Описание верхнего уровня                                    | 19 |
| 3.2. Измерительный модуль                                        | 20 |
| 3.2.1. Модуль stb_gen                                            | 21 |
| 3.2.2. Модуль <i>ch_measure_ctl</i>                              | 26 |
| 3.2.3. Программный интерфейс                                     | 28 |
| Заключение                                                       | 31 |
| Список использованных источников                                 | 32 |
| Приложение 1. Описание файлов проекта                            | 34 |
| Приложение 2. Исходные коды измерительного модуля                | 35 |

#### **ВВЕДЕНИЕ**

Для надёжного функционирования распределённых систем потоковой обработки данных, работающих в режиме реального времени, на исполняемые в них операции накладываются строгие временные ограничения. Многочисленные узлы таких систем могут находиться друг от друга на значительных расстояниях, что приводит к длительным (и не всегда одинаковым) задержкам при передаче сигналов между ними. Возникающие задержки приводят к рассинхронизации работы устройств систем, что влечёт за собой возникновение потенциально некорректных результатов выполнения операций.

С целью согласования всех устройств и обеспечения общего представления времени во всей сети применяются системы синхронизации часов. Такие системы гарантируют, что часы всех устройств сети отсчитывают время с одинаковой скоростью (выдают одинаковые показания в каждый момент времени).

Для передачи информации при синхронизации могут использоваться различные протоколы. Наибольшее распространение получили следующие два: NTP (Network Time Protocol) и PTP (Precision Time Protocol). Протокол NTP способен обеспечивать точность синхронизации времени до одной миллисекунды, а протокол PTP – до десяти.

Протоколы NTP и PTP не подходят для случая, когда необходима синхронизация с субнаносекундной точностью. Например, такая высокая точность требуется в распределённых системах, применяемых в экспериментах по физике высоких энергий. Там они используются для потоковой обработки информации, поступающей с детекторов и ускорителей частиц.

Системы субнаносекундной синхронизации применяются в Большом Адронном Коллайдере, расположенным в ЦЕРН, в Швейцарии. Они также планируются к применению в строящемся комплексе «NICA» Объединённого института ядерных исследований (ОИЯИ) в Дубне. Другим приложением субнаносекундной синхронизации являются системы радиочастотного позиционирования, использующие технологию сверхширокополосной связи (UWB) и алгоритм позиционирования TDoA (Time Difference of Arrival).

**Цель работы:** разработка управляющего устройства для устройства, выполняющего калибровку систем суб-наносекундной синхронизации.

**Решаемые в данной работе задачи:** портирование и интеграция существующих открытых модулей в систему на кристалле, реализация модулей для

проведения измерений по принципу стробоскопического осциллографа, написание драйверов для используемой периферии, написания управляющей программы для процессорного ядра, тестирование и отладка, калибровка полученного устройства.

## ГЛАВА 1. АНАЛИЗ ПРЕДМЕТНОЙ ОБЛАСТИ

#### 1.1. White Rabbit

White Rabbit – система синхронизации часов. Разработана при сотрудничестве множества институтов и компаний. Изначально проект был начат для улучшения текущей системы синхронизации в Церне. Предполагалось использование для физических экспериментов, однако в процессе было создано обобщенное решение, которое нашло своё применение в различных сферах.

#### Характеристики:

- Суб-наносекундная точность
- Большое количество синхронизируемых узлов
- Расстояния в десятки километров
- Канал передачи между двумя узлами 1 Gbps
- Открытый исходный код

Достоинствами White Rabbit являются полностью открытый исходный код и аппаратура, а также использование существующих стандартов (Ethernet, PTP и т. д.).

## 1.2. Калибровка

Синхронизация в сети White Rabbit выполняется по протоколу WR PTP — модифицированному протоколу PTP. (IEEE 1588). Однако для достижения суб-наносекундной точности необходима дополнительная калибровка.

Обмен данными между двумя устройствами происходит по одной линии оптоволокна, работающей в полнодуплексном режиме. Для передачи в одну и другую сторону используется свет с разной длиной волны, поэтому возникает асимметричность в задержках распространения сигнала. Из-за этого устройство не может само определить задержки, отправив эхо запрос другому устройству — время распространения сигнала в одну и другую сторону не равны.

Определение коэффициента асимметричности оптоволокна позволит протоколу White Rabbit PTP обеспечить требуемую точность синхронизации устройств сети.



Рис.1.1. Модель соединения между двумя устройствами

На рис.1.1 изображены возникающие задержки, требующие калибровки. Внутри каждого устройства возникают задержки на приёме и отправке  $(\Delta_{TXM}, \Delta_{RXM}, \Delta_{TXS}, \Delta_{RXS}, \varepsilon_M, \varepsilon_S)$ , которые являются результатом задержек в SFP (Small Form-factor Pluggable) модуле, в электрических цепях и электронных компонентах. Эти задержки калибруются отдельно на каждом устройстве и не являются предметом рассмотрения в данной работе.

Суммарная задержка распространения сигнала от ведущего устройства к ведомому устройству и обратно считается по формуле:

$$delay_{MM} = \Delta_{TXM} + \Delta_{RXS} + \varepsilon_S + \Delta_{TXS} + \Delta_{RXM} + \varepsilon_M + \delta_{MS} + \delta_{SM}$$
 (1.1)

В данной работе рассматривается калибровка задержек  $\delta_{MS}$  и  $\delta_{SM}$  – задержек распространения сигнала по оптоволокну.

Когда оптоволокно ещё не установлено, измерить задержки и вычислить коэффициент асимметричности можно без особых усилий. Коэффициент асимметричности определяется, как:

$$\alpha = \frac{\delta_{MS} - \delta_{SM}}{\delta_{SM}} \tag{1.2}$$

Измеряется коэффициент при помощи дополнительного короткого оптоволокна, с известной задержкой (рис.1.2).



Рис.1.2. Измерение асимметричности про помощи дополнительного оптоволокна

Два устройства подключаются калибруемым оптоволокном ( $\delta_2$ ) и дополнительным ( $\delta_1$ ) отдельно, далее два устройства синхронизируются. После этого измеряется разность фаз синхросигналов 1-PPS (Pulse Per Second), генерируемых ведущим и ведомым устройством.

$$skew_{PPS} = t_{PPS_S} - t_{PPS_M} \tag{1.3}$$

По измеренным значениям можно определить коэффициент асимметричности, как:

$$\alpha = \frac{2\left(skew_{PPS2} - skew_{PPS1}\right)}{\frac{1}{2}\delta_2 - \left(skew_{PPS2} - skew_{PPS1}\right)}$$
(1.4)

Однако существуют так же уже установленные линии, нуждающиеся в калибровке. В таких случаях описанный выше метод не подходит.



Рис. 1.3. Измерение асимметричности про помощи дополнительного оптоволокна

В таких случаях применяется немного изменённый метод с подключением петли от выхода 1-PPS одного устройства к другому (рис.1.3). После синхронизации так же измеряются расхождения фронтов синхросигналов и усредняются.

$$skew_{PPS} = \frac{1}{2} \left( skew_{PPS1} + skew_{PPS2} \right) \tag{1.5}$$

Далее это значение может быть использовано для вычисления коэффициента по формуле 1.4.

Разрабатываемое устройство служит для автоматизированного измерения разности фаз и выполнения калибровки.

#### 1.3. Актуальность

Для процесса калибровки предполагается использовать устройство, способное измерять отрезки времени меньше, чем 1 нс, чтобы обеспечить суб-наносекундную точность, т.е. иметь частоту дискретизации 1 GSa. Если применить правило «пятикратного превышения частоты дискретизации», то используемый осциллограф должен иметь частоту дискретизации выше 5 GSa.

Цена осциллографов с такими характеристиками крайне высока. Актуальность данной работы заключается в том, что предлагается разработать относительно бюджетное устройство, работающее по принципу стробоскопического осциллографа.

Сигналы PPS от калибруемых устройств являются периодическими. Это позволяет применить для их обнаружения, захвата и анализа устройство, работающее по принципу стробоскопического осциллографа.

#### 1.4. Стробоскопический осциллограф

Стробоскопические осциллографы предназначены для обнаружения, захвата и анализа периодических сигналов. Принцип работы стробоскопического осциллографа проиллюстрирован на рис.1.4.



Рис.1.4. Измерение асимметричности про помощи дополнительного оптоволокна

Условные обозначения:

 $U_c$  – исследуемый периодический сигнал

 $T_c$  — период исследуемого сигнала

т – длительность исмпульса исследуемого сигнала

 $\Delta t$  — шаг считывания исследуемого сигнала

 $U_2$  – стробы осциллографа

 $U_3$  – снятая осциллограмма сигнала

Таким образом, для снятия очередной точки изменяется смещение  $\Delta t$ . Частота дискретизации определяется минимальным смещением, которое может задать осциллограф.

# ГЛАВА 2. АППАРАТНАЯ ПЛАТФОРМА И АЛГОРИТМ ПРОВЕДЕНИЯ ИЗМЕРЕНИЙ

Управляющее устройство реализовано на базе ПЛИС, так как есть необходимость разрабатывать специфические аппаратные модули.

В качестве аппаратной платформы УУ (управляющее устройство) выступает ПЛИС серии LittleBee китайского производителя Gowin Semiconductor – GW1N-UV9QN88C6/I5.

### Данная ПЛИС имеет следующие характеристики:

- количество логических блоков LUT4 8640 шт.
- количество триггеров 6480 шт.
- объём FLASH 408 Кбит
- количество блоков BSRAM 26 шт.
- объём блоков SRAM 468 Кбит
- количество блоков PLL (ФАПЧ) 2 шт.
- напряжение питания ядра 1.8–3.3 В
- количество доступных для пользователя выходов I/O 77
- среди них дифференциальных пар 19



Рис.2.1. Печатная плата устройства



Рис.2.2. Структурная схема устройства

На рис.2.2 изображена структурная схема устройства. К УУ подключаются два измерительных канала, которыми необходимо управлять для проведения измерений. ПЛИС тактируется высокостабильным тактовым сигналом с частотой 125 МГц. Связь с устройством верхнего уровня осуществляется через FTDI чип FT2232H.

Перед разработкой УУ необходимо определить ОУ (объект управления), которым предстоит управлять.

#### 2.1. Измерительный канал



Рис.2.3. Функциональная схема измерительного канала

На рис.2.3 приведена функциональная схема одного из измерительных каналов.

Threshold – пороговое напряжение сравнения на компараторе Delay – настройка задержки сигнала на линии задержки (Delay line)

Strobe – стробы, выдаваемые УУ и защёлкивающее текущий результат сравнения компаратора.

PPS — измеряемый сигнал 1-PPS (В общем случае, может быть любой частоты, кратной либо 125 МГц, либо частоте внешнего тактового сигнала, подаваемого с SMA разъёма)

#### 2.2. Алгоритм проведения измерений

Перед началом проведения измерения необходимо определить частоты измеряемого сигнала и найти его фронт.



Рис.2.4. Измерение сигнала

Далее необходимо начать генерировать стробы с частотой измеряемого сигнала. Стробы блокируют текущий выход компаратора, и на его выходе остаётся результат сравнения в момент прихода стробы. При помощи изменения задержки можно сдвигать измеряемую точку по оси ОХ. Для поиска значения напряжения в измеряемой точке необходимо изменять пороговое напряжение сравнения на компараторе.

На рис.2.4 приведён пример измерения.  $t_0$  — точка от которой измеряется сигнал (момент прихода стробы),  $t_1$  — измеряемая точка (момент прихода задержанной стробы на компаратор). Временной отрезок  $[t_0, t_1]$  задаётся линией задержки. Если в момент защёлкивания на выходе компаратора логическая единица, значит измеряемое напряжение выше, чем пороговое. Перед приходом следующей стробы пороговое напряжение повышается. Так продолжается до тех

пор, пока на выходе компаратора не будет логического нуля. В таком случае мы считаем, что измеряемая точка имеет значения напряжение равное пороговому. Задержка увеличивается, и измеряется следующая точка. Таким образом снимается осциллограмма входного сигнала.

#### 2.3. Пороговое напряжение сравнения

Для задания порогового напряжения используется DAC (Digital to analog converter) AD5662 от Analog Devices. AD5662 – 16-битный DAC с гарантированной точностью 12 бит.

С опорным напряжением 3.3 В, шаг изменения напряжения —  $\frac{3.3}{2^{16}} \approx 50$  мкВ. Для задания напряжения используется интерфейс SPI (рис.2.5).



Рис.2.5. Интерфейс взаимодействия с AD5662

При напряжении питания 3.3~B максимальная частота сигнала SCLK -20~MГц. Максимальное время установки выходного напряжения -10~Mкс (обычно 8~Mкс).

# 2.4. Линия задержки

Для сдвига задержки строба используется микросхема SY100EP196V от Microchip. Задержка задаётся 10-битным цифровым кодом с шагом  $\approx 10$  пс (рис.2.6). Так же есть аналоговый вход FTUNE, которым можно точнее точнее подстраивать задержку (рис.2.7) (на данный момент не используется).



Рис.2.6. Задержка, задаваемая цифровым входом



Рис.2.7. Подстройка задержки через аналоговый вход FTUNE

## 2.5. Компаратор

Для сравнения используется компаратор ADCMP582 от Analog Devices. Основным параметром, который необходимо учесть является  $t_{PL}$  (рис.2.8).



Рис.2.8. Временная диаграмма ADCMP582

 $t_{PL}$  – минимальное время, в течение которого сигнал защёлки (Latch Enable) должен быть высоким, чтобы получить результат сравнения попал на выход компаратора.

## ГЛАВА 3. РАЗРАБОТКА АППАРАТНОГО ОПИСАНИЯ УПРАВЛЯЮЩЕГО УСТРОЙСТВА

Все исходные коды аппаратных описаний находятся в директории /rtl/.

#### 3.1. Описание верхнего уровня

Описание верхнего уровня находится в файле /rtl/calsoc\_top.sv.

Управляющее устройство реализовано в виде СнК (Система на кристалле) (рис.3.1) на базе открытого процессорного ядра *PicoRV32*, основанного на открытой архитектуре RISC-V.

Все периферийные модули подключаются к ядру через шину *Wishbone*. Арбитраж на шине выполняет открытый модуль *wbxbar*.



Рис.3.1. Структурная схема СнК

Все периферийные устройства разделяют между собой общее адресное пространство.

0x00000000 – 0x00FFFFFF – RAM 0x01000000 – 0x01FFFFFF – ROM загрузичка 0x02000000 - 0x02FFFFFF - GPIO

0x03000000 - 0x03FFFFFF - UART1

0x04000000 – 0x04FFFFFF – память программы

0x05000000 - 0x05FFFFFF - измерительный модуль

#### 3.2. Измерительный модуль

Измерительный модуль выдаёт все необходимые управляющие сигналы для проведения измерений и логически разделён на несколько компонентов (рис.3.2):

- *stb\_gen* модуль, измеряющий частоту сигнала и генерирующий стробы
- ch\_measure\_ctl модуль, непосредственно управляющий измерением одного канала
- spi\_master модуль, реализующий взаимодействие с AD5662
- sc\_fifo FIFO, накапливающее измеренные значения



Рис.3.2. Структурная схема измерительного модуля (для одного канала)

Все описанные ранее модули подключены в модуле верхнего уровня *measure\_unit*. В нём же реализована вся логика управлением проведением измерений через шину Wishbone.

Далее будет отдельно рассмотрен каждый модуль.

#### 3.2.1. Модуль stb\_gen

Для определения частоты измеряемого сигнала необходимо измерить время между двумя фронтами. Для измерения используется 32-разрядный счётчик, один отсчёт которого равняется 8 нс (125 МГц).

Реализация «в лоб» не может работать стабильно на целевой ПЛИС из-за задержек на цепочке переносов в 32-разрядном сумматоре. Для корректной работы на такой частоте необходимо конвейеризировать счётчик (рис.3.3).



Рис.3.3. Конвейерный 32-х разрядный счётчик

К младшим байтам (регистр LSB) каждый такт прибавляется 1, текущее значение младших байт защёлкивается в регистре  $latched\ LSB$ . При переполнении, перенос защёлкивается в регистре  $latched\ carry$ . К старшим байтам (регистр MSB) прибавляется сохранённый перенос из-за младших байт. Выход счётчика – комбинация значений регистров MSB и  $latched\ LSB$ .

Листинг 3.1. Реализация на языке System Verilog

```
// pipelined counter

logic [T_CNT_WIDTH/2-1 : 0] high_bytes = 0;
logic [T_CNT_WIDTH/2-1 : 0] latched_low_bytes = 0;
logic [T_CNT_WIDTH/2-1 : 0] low_bytes = 0;
logic [T_CNT_WIDTH/2-1 : 0] low_bytes_plus_1;
logic carry;

assign {carry, low_bytes_plus_1} = low_bytes + 1;
//incrementing low bytes
```

```
13
    always_ff @(posedge clk_i, negedge arst_i)
14
       if (~arst_i) low_bytes = 0;
15
       else low_bytes <= low_bytes_plus_1;</pre>
16
17
    //latching low bytes for 1 cycle
    always_ff @(posedge clk_i, negedge arst_i)
18
19
       if (~arst_i) latched_low_bytes = 0;
20
       else latched_low_bytes <= low_bytes;</pre>
21
22
    logic latched_carry = 0;
23
24
    //latching carry
25
    always_ff @(posedge clk_i, negedge arst_i)
26
       if (~arst_i) latched_carry = 0;
27
       else latched_carry <= carry;</pre>
28
29
    //adding latched carry to high bytes
    always_ff @(posedge clk_i, negedge arst_i)
30
31
       if (~arst_i) high_bytes = 0;
32
       else high_bytes <= high_bytes + latched_carry;</pre>
33
34
    //seting t_cnt
35
    always_ff @(posedge clk_i, negedge arst_i)
36
       if (~arst_i) t_cnt = 0;
       else t_cnt <= {high_bytes, latched_low_bytes};</pre>
37
```



Рис.3.4. Конечный автомат модуля stb\_gen

Нас рис.3.4 представлен конечный автомат модуля  $stb\_gen$ . Вся логика максимально упрощена, так как, если на одной регистровой передаче будет много комбинаторной логики, при синтезе под целевую ПЛИС не получится выдержать требуемые  $t_{setup}$  и  $t_{hold}$ .

FIND\_EDGE\_1 – состояние после сброса, ожидание первого фронта на входе  $sig\_i$ 

FIND\_EDGE\_2 – ожидание второго фронта на входе  $sig\_i$ 

WRITE\_START – запись текущего значения  $t\_cnt$  в  $t\_start$ 

FIND\_EDGE\_3 – ожидание третьего фронта на входе  $sig\_i$ 

WRITE\_END – запись текущего значения  $t\_cnt$  в  $t\_end$ 

 ${\sf COUNT\_PERIOD}$  — вычисление периода сигнала ( $t\_end - t\_start$ )

WAIT\_COUNT\_PERIOD – задержка на 1 такт

COUNT\_STROBE – начало расчёта времени начала и конца отрицательного импульса на выходе стробы

WAIT\_STB\_END – ожидание конца стробы и переход к расчёту новой

Таким образом, модуль определяет частоту входного сигнала и начинает бесконечно выдавать стробы.

Помимо 32-х разрядного счётчика, конвейеризация была необходима для всех операций с 32-х битными числами. В модуле *two\_cycle\_32\_adder* реализован двухтактный сумматор, однако использованный подход идентичен тому, что используется в счётчике.

Отдельного упоминания стоит оптимизация операции проверки на равенство двух чисел. Для этого необходимо сравнить попарно все биты чисел, что выполняется параллельно и не должно увеличивать максимальный путь, однако при синтезе получалась цепочка с последовательным сравнением всех разрядов (рис.3.5).



Рис.3.5. Реализация сравнения при синтезе

Для исправления этого пришлось явно заменить операцию сравнения на исключающее ИЛИ и свёртку по ИЛИ-НЕ.

Листинг 3.2. Реализация сравнения

```
always_ff @(posedge clk_i) is_zero_hold_start_lo <= ~|(t_cnt
        [15:0] ^ latched_zero_hold_res[15:0]); //t_cnt ==
        latched_zero_hold_res;

always_ff @(posedge clk_i) is_zero_hold_start_hi <= ~|(
        t_cnt[31:16] ^ latched_zero_hold_res[31:16]); //t_cnt ==
        latched_zero_hold_res;

always_ff @(posedge clk_i) is_zero_hold_start <=
        is_zero_hold_start_lo & is_zero_hold_start_hi;</pre>
```

При работе модуль не выдаёт стробы, а держит компаратор в защёлкнутом состоянии. Для запроса стробы используется пара сигналов  $stb\_req\_i$  и  $stb\_valid\_o$ . При выдаче стробы, с компаратора снимается защёлка на короткое время, после чего устанавливается обратно.

Для тестирования модуля были написаны тестбенчи  $tb\_stb\_gen.sv$  и  $tb2\_stb\_gen.sv$ . На рис.3.6 и рис.3.7 представлены результаты моделирования.



Рис. 3.6. Пример определения частоты входного сигнала

```
iverilog -o test_stb_gen.vvp -g2005-sv tb_stb_gen.sv
./stb_gen.sv:108: vvp.tgt sorry: Case unique/unique0 qualities are ignored.
vvp -n test_stb_gen.vvp
VCD info: dumpfile dump.vcd opened for output.
measured signal T=
generated strobe T=
                            100 ns
                            104 ns
            4 ns
                        clk T= 8 ns
counted period=
measured signal T=
                          20000 ns
generated strobe T=
                          20000 ns
             0 ns
counted period=
                    20000
measured signal T=
                         200000 ns
generated strobe T=
                         200000 ns
                        clk T= 8 ns
counted period=
measured signal T=
                        1333333 ns
generated strobe T=
                        1333336 ns
                        clk T= 8 ns
counted period= 1333336
OK!
tb_stb_gen.sv:69: $finish called at 20196276 (1ns)
```

Рис.3.7. Вывод в консоль при симуляции

Видно, что сигналы с периодом, кратным 8 нс измеряются корректно, при измерении других – ошибка меньше 8 нс.



Рис.3.8. Генерация строб для 10 Мгц



Рис.3.9. Генерация строб для 200 кГц

На рис.3.8 и рис.3.9 приведены осциллограммы генерации строб для 10 МГц и 200 кГц. Видно, что точно определить не кратную частоту невозможно – при генерации получается стробы с частотой 9.6 Мгц, а не 10 Мгц.

#### 3.2.2. Модуль ch\_measure\_ctl

Данный модуль управляет измерением одного из каналов — выставляет необходимое пороговое напряжение и задержку, а затем выставляет запрос на стробу. После прохода стробы, в зависимости от текущего и предыдущего выхода компаратора, принимается решение о следующем значении задержки и порогового напряжения.

Для ускорения снятия осциллограммы применяется предсказание направления изменения сигнала. Для этого, после нахождения очередной точки, изменяется только задержка, после чего на основании выхода компаратора принимается решение – измеряемый сигнал нарастает или уменьшается.



Рис.3.10. Конечный автомат модуля *ch\_measure\_ctl* 

На рис.3.10 представлен конечный автомат описываемого модуля.

IDLE – состояние покоя

SET\_THRESHOLD – отправление запроса на установку порогового напряжения ( $threshold\_wre\_o = 1$ )

WAIT\_THRESHOLD – ожидание установки порогового напряжения  $(threshold\_rdy\_i == 1)$ 

REQ\_STROBE – установка запроса на стробу  $(stb\_req\_o = 1)$ 

WAIT\_STROBE – ожидание прохода стробы ( $stb\_valid\_i == 1$ )

SAVE\_CMP\_RES – обновление текущего и предыдущего выхода компаратора

PROCESS\_RES – принятие решения о статусе поиска точки (определение направления поиска или поиск точки) и о выборе направления поиска UPDATE\_CONF – обновление регистров с текущей задержкой и пороговым напряжением на основании принятого ранее решения

В состоянии PROCESS\_RES, в зависимости от выхода компаратора переключаются два других конечных автомата, отвечающих за статус поиска точки и направления поиска.

На рис.3.11 приведён пример снятия осциллограммы сигнала.



Рис.3.11. Пример снятия осциллограммы с угадыванием направления изменения сигнала

#### 3.2.3. Программный интерфейс

Управление измерительным модулем осуществляется через memory-mapped регистры (рис.3.12, рис.3.13, рис.3.14, рис.3.15, рис.3.16).

Register 3.1: STB\_GEN\_CTL

|    |          | il mitur        |
|----|----------|-----------------|
| 31 |          | 3 2 1 0         |
|    | reserved | 1/01/01/0 (w)   |
|    |          | city mitigh     |
| 31 |          | 3 2 1 0         |
|    | reserved | 1/0 1/0 1/0 (r) |

Register 3.2: STB\_GEN\_PERIOD



Рис.3.12. Регистры для управления модулем stb\_gen

 ${f run}$  — запись 1 начинает измерение частоты входного сигнала  ${f mux}$  — выбор канала для измерения (0 — первый канал, 1 — второй)  ${f clk\_sel}$  — выбор источника тактового сигнала для генерации строб (0 — внутренний, 1 — внешний)

rdy – 1 при окончании измерения

**period** – измеренное значение периода (кол-во отсчётов счётчика)

Register 3.3: CH\_CTL\_DELTA\_REG



Рис.3.13. Регистр для задания параметров измерения

threshold delta – шаг изменения порогового напряжения (разрешение по оси OY)

delay delta – шаг изменения задержки (разрешение по оси ОХ)

Register 3.4: W\_THRESHOLD



Рис.3.14. Регистр для установки порогового напряжения

**dac1/2 rdy** – ЦАП на первом/втором канале установил указанное напряжение

**threshold** – запись в это поле устанавливает заданное напряжение на оба канала

Register 3.5: MU\_CTL\_1/2

|          |   | <sup>(1)</sup> | 3.5   |
|----------|---|----------------|-------|
| 31       | 1 | 0              |       |
| reserved |   | 1/0            | (r/w) |

Рис.3.15. Регистр для управление измерением канала

## **run** – запись 1 начинает измерение

Register 3.6: MU\_CH\_1/2\_VAL



Рис.3.16. Регистр для чтения измеренных значений

measured point — значение измеренной точки valid — 1 если FIFO не пустое и прочитано измеренное значение

#### **ЗАКЛЮЧЕНИЕ**

В результате работы были выполнены: портирование под целевую ПЛИС и интеграция в СнК открытых модулей, созданы модули для проведения измерений при помощи стробирования, написано и отлажено встраиваемое программное обеспечение.

В результате работы получено управляющее устройство для калибратора субнаносекундной синхронизации.

Разработка не завершается на данном этапе, в планах имеются доработки для повышения точности измерений.

ДОПИСАТЬ ЕЩЁ

#### СПИСОК ИСПОЛЬЗОВАННЫХ ИСТОЧНИКОВ

- 1. Пешков Д.В., Михайловский В.С. Калибрующее устройство для средств высокоточной синхронизации распределённых систем потоковой обработки данных // Экономика и Индустрия 5.0 в условиях новой реальности (ИНПРОМ-2022): сборник трудов Всероссийской научно-практической конференции с зарубежным участием 28-30 апреля 2022. СПб.: ПОЛИТЕХ-ПРЕСС, 2022. С. 680-684.
- 2. AD5662BRMZ Datasheet [Электронный ресурс] // Analog Devices URL: https://www.analog.com/media/en/technical-documentation/data-sheets/ad5662.pdf (дата обращения: 16.06.2022).
- 3. ADCMP582BCPZ Datasheet [Электронный ресурс] // Analog Devices URL: https://www.analog.com/media/en/technical-documentation/data-sheets/adcmp580\_581\_582.pdf (дата обращения: 16.06.2022).
- 4. FT2232HL Datasheet [Электронный ресурс] // FTDI Chip URL: https://www.ftdichip.com/Support/Documents/DataSheets/ICs/DS\_FT2232H.pdf (дата обращения: 16.06.2022).
- 5. Grzegorz Daniluk. White Rabbit calibration procedure [Электронный ресурс] // CERN BE-CO-HT URL: https://www.ohwr.org/project/white-rabbit/uploads/76cdbdbadccc9d6c54d5caf246550fbf/WR\_Calibration-v1.1-20151109.pdf (дата обращения: 16.06.2022).
- 6. GW1N series of FPGA Products Data Sheet DS100-2.7.1E [Электронный ресурс] // GOWIN Semiconductor Corp. URL: https://gowinsemi.com/upload/database\_doc/1752/document/6296db1b68af5.pdf (дата обращения: 16.06.2022).
- 7. GowinSynthesis User Guide [Электронный ресурс] // GOWIN Semiconductor Corp. URL: http://cdn.gowinsemi.com.cn/SUG550E.pdf (дата обращения: 16.06.2022).
- 8. Gowin FPGA Primitive User Guide [Электронный ресурс] // GOWIN Semiconductor Corp. URL: https://www.gowinsemi.com/upload/database\_doc/39/document/5bfcff2ce0b72.pdf (дата обращения: 16.06.2022).
- 9. SY100EP196V Datasheet [Электронный ресурс] // Microchip URL: https://www1.microchip.com/downloads/aemDocuments/documents/TCG/ProductDocuments/DataSheets/SY100EP196V-3.3V-5V-2.5GHz-Programmable-Delay-Chip-with-Fine-Tune-Copdf (дата обращения: 16.06.2022).
- 10. PicoRV32 A Size-Optimized RISC-V CPU [Электронный ресурс] // Github URL: https://github.com/YosysHQ/picorv32 (дата обращения: 16.06.2022).
- 11. Another Wishbone (or even AXI-lite) Controlled UART [Электронный ресурс] //

- Github URL: https://github.com/ZipCPU/wbuart32 (дата обращения: 16.06.2022).
- 12. WB2AXIP: Bus interconnects, bridges, and other components [Электронный ресурс] // Github URL: https://github.com/ZipCPU/wb2axip (дата обращения: 16.06.2022).
- 13. OpenCores Free and Open Source gateware IP cores [Электронный ресурс] // OpenCores URL: https://opencores.org/ (дата обращения: 16.06.2022).
- 14. WISHBONE System-on-Chip (SoC)Interconnection Architecture for Portable IP Cores [Электронный ресурс] // OpenCores URL: https://cdn.opencores.org/downloads/wbspec\_b4.pdf (дата обращения: 16.06.2022).

## Приложение 1

## Описание файлов проекта

./rtl/ – директория с исходными кодами аппаратных описаний
./firmware/ – директория с исходными кодами управляющей программы
./bootloader/ – директория с исходными кодами загрузчика
./syn/ – директория с файлами для синтеза (назначения выводов ПЛИС, временные ограничения, конфигурации втроенного логического анализатора)
./boot.py – скрипт для прошивки калибратора
./calsoc.gprj – файл для открытия проекта в Gowin EDA

#### Приложение 2

#### Исходные коды измерительного модуля

Листинг  $\Pi 2.1.$  stb\_gen.sv

```
2 module stb_gen #(
3
    parameter OFFSET = 20,
    parameter T_CNT_WIDTH = 32
5) (
6
    input wire clk_i,
7
    input wire arst_i,
8
9
    input wire sig_i,
10
11
    output logic err_o,
12
    output logic rdy_o,
13
    output logic stb_o,
14
    output logic [T_CNT_WIDTH-1:0] stb_period_o,
15
16
    input logic stb_req_i,
17
    output logic stb_valid_o,
    output logic debug_stb_o
18
19);
20
    // localparam T_CNT_WIDTH
                                   = 32;
    localparam ZERO_HOLD_CYCLES = 2;
21
22
    logic int_stb = 0;
23
24
    assign debug_stb_o = int_stb;
25
26
    logic sig_synced;
27
28
    //stb req interface
29
30
    logic stb_oe; // stb_oe == 1 blocks strobe generation
31
32
    assign stb_o = (rdy_o ? int_stb | stb_oe : int_stb);
33
34
    assign stb_valid_o = stb_oe & rdy_o;
35
36
    logic prev_stb_req;
37
38
    always_ff @(posedge clk_i) prev_stb_req <= stb_req_i;</pre>
39
```

```
40
    logic req_posedge;
41
    assign req_posedge = ~prev_stb_req & stb_req_i;
42
43
    logic is_zero_hold_start;
44
    logic is_stb_end;
45
46
    always_ff @(posedge clk_i, negedge arst_i) begin
47
      if (~arst_i) begin
48
        stb_oe = 0;
49
      end else begin
50
         casex ({req_posedge, is_stb_end})
51
           2'bx1: stb_oe <= 1;
52
           2'b1x:
                   stb_oe <= 0;
53
         endcase
54
      end
55
    end
56
57
    sync_ff #(
58
      .WIDTH (1),
59
      .STAGES (2)
    ) sig_i_sync_ff_inst (
60
61
      .clk_i (clk_i),
62
      .data_i(sig_i),
63
      .data_o(sig_synced)
64
    );
65
66
    logic prev_sig; //edge detect
67
68
    always_ff @(posedge clk_i) begin
69
      prev_sig <= sig_synced;</pre>
70
    end
71
72
    logic sig_posedge;
73
    assign sig_posedge = sig_synced & ~prev_sig;
74
75
    typedef enum logic[8:0] {
76
      FIND_EDGE_1
                       = 9,0000000001,
77
      FIND_EDGE_2
                      = 9,000000010,
78
      WRITE_START
                      = 9,000000100,
79
      FIND_EDGE_3
                      = 9,000001000
80
      WRITE_END
                    = 9'b000010000,
81
      COUNT_PERIOD = 9'b000100000,
82
      WAIT_COUNT_PERIOD
                           = 9,001000000,
83
      COUNT_STROBE
                       = 9,0010000000
      WAIT_STB_END = 9'b100000000
84
```

```
85
     } stb_gen_state;
86
87
     stb_gen_state state = FIND_EDGE_1;
88
89
     logic [T_CNT_WIDTH-1 : 0] t_cnt /* synthesis syn_keep=1
        syn_preserve=1 syn_ramstyle="registers" */;
     logic [T_CNT_WIDTH-1 : 0] t_start;
90
91
     logic [T_CNT_WIDTH-1 : 0] t_end;
92
     stb_gen_state next_state;
93
94
95
     logic count_zero_hold_begin, zero_hold_begin_valid;
96
     logic count_stb_end, stb_end_valid;
97
98
     assign count_zero_hold_begin = (state == COUNT_STROBE ? 1 :
        0);
99
     assign count_stb_end = (state == COUNT_STROBE ? 1 : 0);
100
101
     always_comb begin
102
       unique case (state) /* sythesis parallel_case*/
103
         FIND_EDGE_1:
                       if (sig_posedge) next_state =
            FIND_EDGE_2;
104
                      else next state = state;
105
         FIND_EDGE_2:
                            if (sig_posedge) next_state =
            WRITE_START;
                      else next_state = state;
106
107
         WRITE_START:
                            next_state = FIND_EDGE_3;
108
         FIND_EDGE_3:
                            if (sig_posedge) next_state =
            WRITE_END;
109
                      else next_state = state;
110
         WRITE_END:
                            next_state = COUNT_PERIOD;
111
         COUNT_PERIOD:
                            next_state = WAIT_COUNT_PERIOD;
112
         WAIT_COUNT_PERIOD:
                                next_state = COUNT_STROBE;
113
                            next_state = WAIT_STB_END;
         COUNT_STROBE:
114
         WAIT_STB_END:
                            if (is_stb_end) next_state =
            COUNT_STROBE;
115
                      else next_state = state;
116
         default:
                          next_state = state;
117
       endcase
118
     end
119
120
     always_ff @(posedge clk_i, negedge arst_i) begin
121
       if (~arst_i) begin
122
         state = FIND_EDGE_1;
123
       end else begin
```

```
124
         state <= next_state;</pre>
125
       end
126
     end
127
128
     logic [T_CNT_WIDTH-1:0] adder_zero_hold_res;
129
     logic [T_CNT_WIDTH -1:0] adder_stb_end_res;
130
131
     always_ff @(posedge clk_i) begin
132
       stb_period_o <= (state == COUNT_PERIOD ? t_end - t_start :
           stb_period_o);
133
     end
134
135
     localparam MAGIC_CONST = 3;
136
137
     logic [T_CNT_WIDTH-1:0] period_minus_zero_hold;
138
     always_ff @(posedge clk_i) period_minus_zero_hold <=
        stb_period_o - (ZERO_HOLD_CYCLES+MAGIC_CONST); //magic
        constat due to computation pipeline
139
140
     two_cycle_32_adder adder_zero_hold_begin (
141
       .clk_i (clk_i),
142
       .a_i (t_cnt),
143
             (period_minus_zero_hold),
144
       .valid_i(count_zero_hold_begin),
145
       .valid_o(zero_hold_begin_valid),
       .res_o (adder_zero_hold_res)
146
147
     );
148
     two_cycle_32_adder adder_stb_end (
149
150
       .clk_i (clk_i),
151
       .a_i (t_cnt),
152
              (stb_period_o - MAGIC_CONST), //magic constat due to
           computation pipeline
153
       .valid_i(count_stb_end),
154
       .valid_o(stb_end_valid),
155
       .res_o (adder_stb_end_res)
156
     );
157
158
     logic [T_CNT_WIDTH -1:0] latched_zero_hold_res;
159
     logic [T_CNT_WIDTH-1:0] latched_stb_end_res;
160
161
     always_ff @(posedge clk_i) latched_zero_hold_res <= (
        zero_hold_begin_valid ? adder_zero_hold_res :
        latched_zero_hold_res);
```

```
162
     always_ff @(posedge clk_i) latched_stb_end_res <= (
        stb_end_valid ? adder_stb_end_res : latched_stb_end_res);
163
164
165
     logic is_zero_hold_start_lo;
166
     logic is_zero_hold_start_hi;
167
     logic is_stb_end_lo;
168
     logic is_stb_end_hi;
169
170
     always_ff @(posedge clk_i) is_zero_hold_start_lo <= ~|(
        t_cnt[15:0] ^ latched_zero_hold_res[15:0]); //t_cnt ==
        latched_zero_hold_res;
171
     always_ff @(posedge clk_i) is_zero_hold_start_hi <= ~|(
        t_cnt[31:16] ^ latched_zero_hold_res[31:16]); //t_cnt ==
         latched_zero_hold_res;
172
     always_ff @(posedge clk_i) is_zero_hold_start <=
        is_zero_hold_start_lo & is_zero_hold_start_hi;
173
174
     always_ff @(posedge clk_i) is_stb_end_lo <= ~|(t_cnt[15:0]
        ^ latched_stb_end_res[15:0]); //t_cnt ==
        latched_zero_hold_res;
175
     always_ff @(posedge clk_i) is_stb_end_hi <= ~|(t_cnt[31:16]
         ^ latched_stb_end_res[31:16]); //t_cnt ==
        latched_zero_hold_res;
176
     always_ff @(posedge clk_i) is_stb_end <= is_stb_end_hi &
        is_stb_end_lo;
177
178
     always_ff @(posedge clk_i, negedge arst_i) begin
179
       if (~arst_i) begin
180
         int_stb = 0;
181
       end else begin
182
         case (1)
183
           is_zero_hold_start: int_stb <= 0;</pre>
184
           is_stb_end:
                            int_stb <= 1;</pre>
185
           default:
                         int_stb <= int_stb;</pre>
186
         endcase
187
       end
188
     end
189
190
     always_ff @(posedge clk_i, negedge arst_i) begin
191
       if (~arst_i) rdy_o = 0;
192
       else rdy_o <= (state == COUNT_STROBE ? 1 : rdy_o);</pre>
193
     end
194
195
     always_ff @(posedge clk_i) begin
```

```
196
       err_o <= (state == FIND_EDGE_1 ? 0 : err_o);</pre>
197
     end
198
199
     always_ff @(posedge clk_i) t_end <= (state == WRITE_END ?
        t_cnt : t_end);
200
     always_ff @(posedge clk_i) t_start <= (state == WRITE_START
201
        ? t_cnt : t_start);
202
203
     // pipelined counter
204
205
     logic [T_CNT_WIDTH/2-1 : 0] high_bytes = 0
                                                        /* synthesis
         syn_keep=1 syn_preserve=1 syn_ramstyle="registers" */;
206
     logic [T_CNT_WIDTH/2-1 : 0] latched_low_bytes = 0
        synthesis syn_keep=1 syn_preserve=1 syn_ramstyle="
        registers" */;
     logic [T_CNT_WIDTH/2-1 : 0] low_bytes = 0
207
                                                         /* synthesis
         syn_keep=1 syn_preserve=1 syn_ramstyle="registers" */;
     logic [T_CNT_WIDTH/2-1 : 0] low_bytes_plus_1;
208
209
     logic carry;
210
     assign {carry, low_bytes_plus_1} = low_bytes + 1;
211
212
213
     //incrementing low bytes
214
     always_ff @(posedge clk_i, negedge arst_i)
       if (~arst_i) low_bytes = 0;
215
216
       else low_bytes <= low_bytes_plus_1;</pre>
217
218
     //latching low bytes for 1 cycle
219
     always_ff @(posedge clk_i, negedge arst_i)
220
       if (~arst_i) latched_low_bytes = 0;
221
       else latched_low_bytes <= low_bytes;</pre>
222
223
     logic latched_carry = 0;
224
225
     //latching carry
226
     always_ff @(posedge clk_i, negedge arst_i)
227
       if (~arst_i) latched_carry = 0;
228
       else latched_carry <= carry;</pre>
229
230
     //adding latched carry to high bytes
     always_ff @(posedge clk_i, negedge arst_i)
231
232
       if (~arst_i) high_bytes = 0;
233
       else high_bytes <= high_bytes + latched_carry;</pre>
234
```

```
//seting t_cnt
always_ff @(posedge clk_i, negedge arst_i)
if (~arst_i) t_cnt = 0;
else t_cnt <= {high_bytes, latched_low_bytes};
endmodule</pre>
```

Листинг П2.2. two\_cycle\_32\_adder.sv

```
2 module two_cycle_32_adder (
3
       input logic clk_i,
4
       input logic [31:0] a_i,
5
       input logic [31:0] b_i,
6
       input logic valid_i,
7
8
       output logic valid_o,
9
       output logic [31:0] res_o
10);
11
12
       logic [31:0] a, b;
13
14
       always_ff @(posedge clk_i) begin
15
           a <= (valid_i ? a_i : a);
           b <= (valid_i ? b_i : b);
16
17
       end
18
19
       enum logic[2:0] {
20
           IDLE
                             = 3,0001,
21
           FIRST_STAGE
                             = 3,0010,
22
           SECOND_STAGE
                             = 3, b100
23
       } state = IDLE, next_state;
24
25
       always_ff @(posedge clk_i) state <= next_state;</pre>
26
27
       logic [1:0] valid;
28
       integer i;
29
       always_ff @(posedge clk_i) begin
           valid[0] <= valid_i;</pre>
30
31
           for (i = 1; i <= 1; i = i +1) valid[i] <= valid[i-1];
32
           valid_o <= valid[1];</pre>
33
       end
34
35
36
       always_comb begin
```

```
37
           next_state = state;
38
           case (state)
39
                IDLE: if (valid_i) next_state = FIRST_STAGE;
40
                FIRST_STAGE: next_state = SECOND_STAGE;
41
                SECOND_STAGE: next_state = IDLE;
42
           endcase
43
       end
44
45
       logic carry;
46
47
       always_ff @(posedge clk_i) begin
48
           case (state)
49
                FIRST_STAGE: {carry, res_o[15:0]} <= a[15:0] + b
                   [15:0];
                default: {carry, res_o[15:0]} <= {carry, res_o</pre>
50
                   [15:0]};
51
           endcase
52
           case (state)
53
                SECOND_STAGE: res_o[31:16] \le a[31:16] + b[31:16]
                   + carry;
                default: res_o[31:16] <= res_o[31:16];</pre>
54
55
           endcase
56
       end
57
58 endmodule
```

Листинг  $\Pi 2.3.$   $sc_fifo.sv$ 

```
2 module sc_fifo #(
3
      parameter LGFLEN = 10,
4
      parameter WIDTH = 32
5
  ) (
       input logic
6
                        clk_i,
7
       input logic
                        arstn_i,
8
9
       input logic [WIDTH-1:0] data_i,
10
      input logic
                                 wre_i,
11
12
      output logic [WIDTH-1:0] data_o,
13
      input
              logic
                                  re_i,
14
      output logic
                                  n_empty_o
15);
16
17
      logic [WIDTH-1:0] cyc_buf [int'($pow(2, LGFLEN)-1):0];
```

```
18
19
       logic [LGFLEN-1:0] r_addr, w_addr;
20
21
       assign n_empty_o = (r_addr != w_addr);
22
       assign data_o = cyc_buf[r_addr];
23
24
       always_ff @(posedge clk_i/*, negedge arstn_i*/) begin :
          w_addr_ff
25
           if (~arstn_i) begin
26
                w_addr = 0;
27
           end else begin
28
                if (wre_i) begin
29
                    w_addr <= w_addr + 1;
30
                end
31
           end
32
       end
33
34
       always_ff @(posedge clk_i/*, negedge arstn_i*/) begin :
          r_addr_ff
35
           if (~arstn_i) begin
                r_addr = 0;
36
37
           end else begin
38
                if (re_i & n_empty_o) begin
39
                    r_addr \le r_addr + 1;
40
                end
41
           end
42
       end
43
44
       always_ff @(posedge clk_i) begin
45
           if (wre_i) begin
46
                cyc_buf[w_addr] <= data_i;</pre>
47
           end
       end
48
49
50 endmodule
```

Листинг  $\Pi 2.4.$  *ch\_measure\_ctl.sv* 

```
2 module ch_measure_ctl #(
3  parameter DEFAULT_THRESHOLD_DELTA = 1,
4  parameter DEFAULT_D_CODE_DELTA = 1
5 ) (
6  input logic clk_i,
7  input logic arst_i,
```

```
8
9
    //stb request interface
10
    output logic stb_req_o,
11
            logic stb_valid_i,
    input
12
13
    //CMP input
14
    input logic cmp_out_i,
15
16
    //control threshold and delay delta
17
    input logic [15:0] threshold_delta_i,
18
    input logic [9:0] d_code_delta_i,
19
20
    //dac (threshold) output
21
    output logic [15:0] threshold_o,
22
    output logic threshold_wre_o,
23
    input logic threshold_rdy_i,
24
25
    //delay line
    output logic [9:0] d_code_o,
26
27
28
    //ctl
29
    input logic run_i,
30
31
    //TODO output measured value
    output logic point_rdy_o
32
33);
34
35
    enum logic[7:0] {
36
      IDLE
                        = 8,00000001,
37
      SET_THRESHOLD
                        = 8,000000010,
38
      WAIT_THRESHOLD
                         = 8,00000100
39
      REQ_STROBE
                        = 8,00001000
40
      WAIT_STROBE
                       = 8,00010000,
41
                       = 8,000100000,
      SAVE_CMP_RES
42
      PROCESS_RES
                       = 8,001000000,
43
      UPDATE_CONF
                        = 8, b10000000
44
    } ctl_state, next_ctl_state;
45
46
    enum logic [1:0] {
47
      DIR_UP
                 = 2, b01,
48
      DIR_DOWN = 2'b10
49
    } threshold_dir;
50
51
    enum logic [1:0] {
52
      FIND_DIR = 2'b01,
```

```
53
      FIND_POINT = 2'b10
54
    } point_state, next_point_state;
55
56
    logic cur_cmp_out, prev_cmp_out;
57
58
    always_comb begin : next_ctl_state_comb
59
       if (~run_i) begin
60
        next_ctl_state = IDLE;
61
      end else begin
62
         case (ctl_state)
63
           IDLE:
                         next_ctl_state = SET_THRESHOLD;
64
           SET_THRESHOLD:
                              next_ctl_state = WAIT_THRESHOLD;
65
           WAIT_THRESHOLD:
                              if (threshold_rdy_i) next_ctl_state
             = REQ_STROBE;
66
                     else next_ctl_state = ctl_state;
67
           REQ_STROBE:
                           next_ctl_state = WAIT_STROBE;
68
           WAIT_STROBE:
                            if (stb_valid_i) next_ctl_state =
              SAVE_CMP_RES;
69
                     else next_ctl_state = ctl_state;
70
71
           SAVE_CMP_RES:
                           next_ctl_state = PROCESS_RES;
72
           PROCESS_RES:
                           next_ctl_state = UPDATE_CONF;
73
           UPDATE_CONF:
                           next_ctl_state = SET_THRESHOLD;
74
75
         endcase
76
      end
77
    end
78
79
    always_comb begin : next_point_state_comb
80
       if (ctl_state == PROCESS_RES) begin
         case (point_state) /* synthesis full_case */
81
           FIND_POINT: if (cur_cmp_out != prev_cmp_out)
82
              next_point_state = FIND_DIR;
83
                 else next_point_state = point_state;
84
           FIND_DIR: next_point_state = FIND_POINT;
85
         endcase
86
      end else begin
87
         next_point_state = point_state;
88
      end
89
    end
90
91
    always_ff @(posedge clk_i, negedge arst_i) begin
92
      if (~arst_i) begin
93
         cur_cmp_out = 1;
94
       end else if (ctl_state == SAVE_CMP_RES) begin
```

```
95
         cur_cmp_out <= cmp_out_i;</pre>
96
         prev_cmp_out <= cur_cmp_out;</pre>
97
       end
98
     end
99
100
     always_ff @(posedge clk_i, negedge arst_i) begin :
        ctl_state_ff
101
       if (~arst_i) begin
102
          ctl_state = IDLE;
       end else begin
103
104
          ctl_state <= next_ctl_state;</pre>
105
       end
106
     end
107
108
     logic [15:0] next_threshold;
109
     logic [9:0] next_d_code;
110
111
     always_comb begin : next_threshold_comb
       case (ctl_state)
112
113
          IDLE:
                     next_threshold = 0;
114
          UPDATE_CONF: if (point_state == FIND_POINT) begin
115
                     case (threshold_dir) /* synthesis full_case */
116
                       DIR UP:
                                 next_threshold = threshold_o +
                          threshold_delta_i;
117
                       DIR_DOWN: next_threshold = threshold_o -
                          threshold_delta_i;
118
                     endcase
119
                  end else begin
120
                     next_threshold = threshold_o;
121
                  end
122
          default:
                      next_threshold = threshold_o;
123
       endcase
124
     end
125
126
     always_comb begin : next_d_code_comb
127
       case (ctl_state)
128
          IDLE:
                     next_d_code = 0;
129
          UPDATE_CONF: if (point_state == FIND_DIR) begin
                       next_d_code = d_code_o + d_code_delta_i;
130
131
                   end else begin
132
                     next_d_code = d_code_o;
133
                   end
134
          default:
                      next_d_code = d_code_o;
135
       endcase
136
     end
```

```
137
138
     always_ff @(posedge clk_i, negedge arst_i) begin :
        threshold_ff
139
        if (~arst_i) begin
140
          threshold_o = 0;
141
        end else begin
142
          threshold_o <= next_threshold;</pre>
143
        end
144
     end
145
146
     always_ff @(posedge clk_i, negedge arst_i) begin :
        threshold_wre_ff
        if (~arst_i) begin
147
148
          threshold_wre_o = 0;
149
        end else begin
150
          case (ctl_state)
151
            SET_THRESHOLD:
                              threshold_wre_o <= 1;</pre>
152
            default:
                              threshold_wre_o <= 0;</pre>
153
          endcase
154
        end
155
     end
156
157
     always_ff @(posedge clk_i, negedge arst_i) begin :
        stb_req_ff
158
        if (~arst_i) begin
159
          stb_req_o = 0;
160
        end else begin
161
          case (ctl_state)
            REQ_STROBE: stb_req_o <= 1;</pre>
162
163
            default: stb_req_o <= 0;</pre>
164
          endcase
165
        end
166
     end
167
168
     always_ff @(posedge clk_i, negedge arst_i) begin :
        threshold_dir_ff
169
        if (~arst_i) begin
170
          threshold_dir = DIR_UP;
171
        end else begin
172
          if (ctl_state == PROCESS_RES & point_state == FIND_DIR)
             begin
173
            case ({cur_cmp_out, prev_cmp_out})
174
              2'b10:
                          threshold_dir <= DIR_DOWN;</pre>
                         threshold_dir <= DIR_UP;</pre>
175
              2'b01:
```

```
176
              default: threshold_dir <= (threshold_dir ==</pre>
                  DIR_DOWN ? DIR_UP : DIR_DOWN);
177
            endcase
178
          end
179
        end
180
     end
181
182
     always_ff @(posedge clk_i, negedge arst_i) begin :
        point_state_ff
183
        if (~arst_i) begin
184
          point_state = FIND_POINT;
185
        end else begin
186
          point_state <= next_point_state;</pre>
187
188
     end
189
190
     always_ff @(posedge clk_i, negedge arst_i) begin : d_code_ff
191
        if (~arst_i) begin
192
          d_code_o = 0;
193
        end else begin
194
          d_code_o <= next_d_code;</pre>
195
        end
196
     end
197
198
     always_ff @(posedge clk_i, negedge arst_i) begin :
        point_rdy_ff
199
        if (~arst_i) begin
200
          point_rdy_o = 0;
201
        end else begin
202
          if (ctl_state == PROCESS_RES & cur_cmp_out !=
             prev_cmp_out) point_rdy_o <= 1;</pre>
203
          else point_rdy_o <= 0;</pre>
204
        end
205
     end
206
207 endmodule
```