# Лабораторный практикум

«Проектирование цифровых устройств с помощью Verilog HDL»

# Лабораторная работа №1 Введение в Verilog HDL

## 1.1 Возникновение языков описания цифровой аппаратуры

Цифровые устройства — это устройства, предназначенные для приёма и обработки цифровых сигналов. Цифровыми называются сигналы, которые можно рассматривать в виде набора дискретных уровней. В цифровых сигналах информация кодируется в виде конкретного уровня напряжения. Как правило выделяется два уровня — логический «0» и логическая «1».

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

Графические схемы, которые применялись для проектирования цифровых устройств на ранних этапах развития, уже не могли эффективно использоваться. Потребовался новый инструмент разработки, и таким инструментом стали языки описания аппаратной части цифровых устройств (Hardware Description Languages, HDL), которые описывали цифровые структуры формализованным языком, чем-то похожим на язык программирования.

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

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

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

### 1.2 HDL описания логических вентилей

Логические вентили реализуют функции алгебры логики: И, ИЛИ, Исключающее ИЛИ, НЕ. Напомним их таблицы истинности:

| a | $\mid b \mid$ | $a \cdot b$ |
|---|---------------|-------------|
| 0 | 0             | 0           |
| 0 | 1             | 0           |
| 1 | 0             | 0           |
| 1 | 1             | 1           |

 $\begin{array}{c|cccc} a & b & a|b \\ \hline 0 & 0 & 0 \\ 0 & 1 & 1 \\ 1 & 0 & 1 \\ 1 & 1 & 1 \\ \end{array}$ 

Таблица 1.1: И

| a | b | $a\oplus b$ |
|---|---|-------------|
| 0 | 0 | 0           |
| 0 | 1 | 1           |
| 1 | 0 | 1           |
| 1 | 1 | 0           |

Таблица 1.2: ИЛИ

| a | $\bar{a}$ |
|---|-----------|
| 0 | 1         |
| 1 | 0         |

Таблица 1.3: Исключающее ИЛИ

Таблица 1.4: НЕ

Начнём знакомиться с Verilog HDL с описания логического вентиля «И». Ниже приведен код, описывающий вентиль с точки зрения его структуры:

Листинг 1.1: Модуль, описывающий вентиль «И»

Описанный выше модуль можно представить как некоторый «ящик», в который входит 2 провода с названиями (a) и (b) и из которого выходит один провод с названием (b) внутри этого блока результат выполнения операции (b) (в синтаксисе Verilog записывается как (b)) над входами соединяют с выходом.

Схемотично изобразим этот модуль:



Рис. 1.1: Структура модуля «and\_gate»

Аналогично опишем все оставшиеся вентили:

```
1 module or_gate(
2          input a,
3          input b,
4          output result)
5
6 assign result = a | b;
7
8 endmodule
```

Листинг 1.2: Модуль, описывающий вентиль «ИЛИ»

```
1 module xor_gate(
2          input a,
3          input b,
4          output result)
5
6 assign result = a ^ b;
7
8 endmodule
```

Листинг 1.3: Модуль, описывающий вентиль «Исключающее ИЛИ»

```
module not_gate(
input a,
output result)

assign result = ~a;
endmodule
```

Листинг 1.4: Модуль, описывающий вентиль «НЕ»

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

```
1 if ( (a & b) | (~c) ) begin
2 ...
3 end
```

Листинг 1.5: Пример использования логических вентилей

Условие будет выполняться либо когда не выполнено условие «с», либо когда одновременно выполняются условия «а» и «b». Здесь и далее под условием понимается логический сигнал, отражающий его истинность.

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

Листинг 1.6: Модуль, описывающий побитовое «ИЛИ» между двумя шинами

Это описание описывает побитовое «ИЛИ» между двумя шинами по 8 бит. То есть описываются восемь логических вентилей «ИЛИ», каждый из которых имеет на входе соответствующие разряды из шины «x» и шины «y».

При использовании шин можно в описании использовать конкретные биты шины и группы битов. Для этого используют квадратные скобки после имени шины:

```
1
   module bitwise ops(
 2
            input [7:0] \times,
3
            output [4:0] a,
4
            output
                          b,
5
            output [2:0] c);
6
7 assign a = x[5:1];
8 assign b = x[5] | x[7];
   assign c = x[7:5] ^ x[2:0];
9
10
11
   endmodule
```

Листинг 1.7: Модуль, демонстрирующий битовую адресацию шин

Такому описанию соответствует следующая структурная схема, приведённая на Рис. 1.2



Рис. 1.2: Структура модуля «bitwise ops»

Впрочем, реализация ФАЛ с помощью логических вентилей не всегда представляется удобной. Допустим нам нужно описать таблично-заданную ФАЛ. Тогда описания этой функции при помощи логических вентилей нам придётся сначала минимизировать её и только после этого, получив логическое выражение (которое, несмотря на свою минимальность, не обязательно является коротким), сформулировать его с помощью языка Verilog HDL. Как видно, ошибку легко допустить на любом из этих этапов.

Одно из главных достоинств Verilog HDL — это возможность описывать поведение цифровых устройств вместо описания их структуры.

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

Используя эту возможность, опишем таблично-заданную ФАЛ на Verilog HDL:

```
1 module function(
2 input x0,
3 input x1,
4 input x2,
```

```
5
     output reg y);
 6
 7 wire [2:0] x bus;
   assign x bus = \{x2, x1, x0\};
 8
 9
10
   always @(x bus) begin
      case (x bus)
11
12
       3'b000: v \le 1'b0;
       3'b010: y <= 1'b0;
13
       3'b101: y <= 1'b0;
14
       3'b110: y <= 1'b0;
15
       3'b111: v <= 1'b0;
16
        default: v <= 1'b1;</pre>
17
18
      endcase
19 end
20
21
   endmodule
```

Листинг 1.8: Пример описания таблично-заданной ФАЛ на Verilog HDL

Описание, приведённое выше, определяет y, как табличнозаданную функцию, которая равна нулю на наборах 0, 2, 5, 6, 7 и единице на всех остальных наборах.

Остановимся подробнее на новых синтаксических конструкциях:

Описание нашего модуля начинается с создания трёхбитной шины «x bus» на строке 7.

После создания шины «x\_bus», на она подключается к объединению проводов «x2», «x1» и «x0» с помощью оператора assign как показано на Puc. 1.3.



Рис. 1.3: Действие оператора **assign** 

Затем начинается функциональный блок **always**, на котором мы остановимся подробнее.

Verilog HDL описывает цифровую аппаратуру, которая су-

ществует вся одновременно, но инструменты анализа и синтеза описаний являются программами и выполняются последовательно на компьютере. Так возникла необходимость последовательной программе «рассказать» про то, какие события приводят к срабатыванию тех или иных участков кода. Сами эти участки назвали процессами. Процессы обозначаются ключевым словом always.

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

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

Новое ключевое слово **reg** здесь необходимо потому, что в выходной вектор происходит запись, а запись в языке Verilog HDL разрешена только в «регистры» — специальные «переменные», предусмотренные в языке. Данная концепция и ключевое слово reg будет рассмотрено гораздо подробнее в следующей лабораторной работе.

Оператор <= называется оператором неблокирующего присваивания. В результате выполнения этого оператора то, что стоит справа от него, «помещается» («кладется», «перекладывается») в регистр, который записан слева от него. Операции неблокирующего присваивания происходят одновременно по всему процессу.

Оператор **case** описывает выбор действия в зависимости от анализируемого значения. В нашем случае анализируется значение шины «x\_bus». Ключевое слово **default** используется для обозначения всех остальных (не перечисленных) вариантов значений.

Константы и значения в языке Verilog HDL описываются следующим образом: сначала указывается количество бит, затем после апострофа с помощью буквы указывается формат и, сразу за ним, записывается значение числа в этом формате.

Возможные форматы:

• b - бинарный, двоичный;

- h шестнадцатеричный;
- d десятичный.

Немного расширив это описание, легко можно определить не одну, а сразу несколько ФАЛ одновременно. Для упрощения записи сразу объединим во входную шину все переменные. В выходную шину объединим значения функций:

```
1
   module decoder(
 2
      input [2:0] \times,
 3
      output [3:0] y);
 4
   reg [3:0] decoder output;
 5
   always @(x) begin
 6
 7
      case (x)
        3'b000: decoder output <= 4'b0100;
 8
        3'b001: decoder output <= 4'b1010;
 9
        3'b010: decoder output <= 4'b0111;
10
        3'b011: decoder output <= 4'b1100;
11
12
        3'b100: decoder output <= 4'b1001;
13
        3'b101: decoder output <= 4'b1101;
        3'b110: decoder output <= 4'b0000;
14
        3'b111: decoder output <= 4'b0010;
15
16
      endcase
17
   end
18
   assign y = decoder output;
19
20
21
   endmodule:
```

Листинг 1.9: Описание дешифратора на языке Verilog HDL.

Теперь нам удалось компактно записать четыре функции, каждая от трёх переменных:

```
y_0 = f(x_2, x_1, x_0);

y_1 = f(x_2, x_1, x_0);

y_2 = f(x_2, x_1, x_0);

y_3 = f(x_2, x_1, x_0).
```

Но, если мы посмотрим на только что описанную конструк-

цию под другим углом, мы увидим, что это описание можно трактовать следующим образом: «поставить каждому возможному входному вектору x в соответствие заранее определенный выходной вектор y». Такое цифровое устройство называют  $\partial e u u d p a mopo m$ .

На Рис. ?? показано принятое в цифровой схемотехнике обозначение дешифратора.



Рис. 1.4: Графическое обозначение дешифратора

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

Дешифраторы активно применяются при разработке цифровых устройств. В большинстве цифровых устройств в явном или неявном виде можно встретить дешифратор.

Рассмотрим еще один интересный набор ФАЛ:

```
module decoder(
 2
      input [2:0] a,
 3
      input [2:0] b,
      input [2:0] c,
 4
 5
      input [2:0] d,
 6
      input [1:0] s,
 7
      output reg [2:0] y);
 8
    always @(a,b,c,d,s) begin
 9
10
      case (s)
        3'b00: y \le a;
11
12
        3'b01: y <= b;
3'b10: y <= c;
13
                y <= d;
14
        3'b11:
15
        default: y <= a;</pre>
```

```
16 endcase
17 end
18
19 endmodule;
```

Листинг 1.10: Описание мультиплексора на языке Verilog HDI.

Что можно сказать об этом описании? Выходной вектор y — это результат работы трёх ФАЛ, каждая из которых является функцией 6 переменных. Так,  $y_0 = f(a_0, b_0, c_0, d_0, s_1, s_0)$ .

Анализируя оператор **case**, можно увидеть, что главную роль в вычислении значения  $\Phi$ АЛ играет вектор s, в результате проверки которого выходу  $\Phi$ АЛ присваивается значение «выбранной» переменной.

Получившееся устройство называется мультиплексор.

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

Графическое изображение мультиплексора приведено на Рис. 1.5



Рис. 1.5: Графическое обозначение мультиплексора

Особенно хочется отметить, что на самом деле никакой «проверки» сигнала управления не существует и уж тем более не существует «коммутации», ведь мультиплексор — это таблично-заданная ФАЛ. Результат выполнения этой ФАЛ выглядит так, как будто происходит «подключение» «выбранной» входной шины к выходной.

Приведём для наглядности таблицу, задающую ФАЛ для одного бита выходного вектора (число ФАЛ в мультиплексоре и,

следовательно, число таблиц, равняется числу бит в выходном векторе). Для краткости выпишем таблицу наборами строк вида:  $f(s_1, s_0, a_0, b_0, c_0, d_0) = y_0$  в четыре столбца.

Обратите внимание, что в качестве старших двух бит входного вектора для удобства записи и анализа мы выбрали переменные «управляющего» сигнала, а выделение показывает какая переменная «поступает» на выход функции f:

```
f(000000) = 0
                f(010000) = 0
                                f(100000) = 0
                                                 f(110000) = 0
f(000001) = 0
                f(010001) = 0
                                f(100001) = 0
                                                 f(110001) = 1
f(000010) = 0
                f(010010) = 0
                                f(100010) = 1
                                                 f(11001\mathbf{0}) = 0
f(000011) = 0
                f(010011) = 0
                                f(100011) = 1
                                                 f(110011) = 1
f(000100) = 0
                f(010100) = 1
                                f(1001\mathbf{0}0) = 0
                                                 f(11010\mathbf{0}) = 0
f(000101) = 0
                                f(100101) = 0
                                                 f(110101) = 1
                f(010101) = 1
f(000110) = 0
                f(010110) = 1
                                f(1001\mathbf{1}0) = 1
                                                 f(110110) = 0
                                                 f(110111) = 1
f(000111) = 0
                f(0101111) = 1
                                f(100111) = 1
f(001000) = 1
                                                 f(111000) = 0
                f(011000) = 0
                                f(101000) = 0
                                                 f(111001) = 1
f(001001) = 1
                f(011\mathbf{0}01) = 0
                                f(101001) = 0
f(001010) = 1
                f(011010) = 0
                                f(1010\mathbf{1}0) = 1
                                                 f(111010) = 0
f(001011) = 1
                f(011011) = 0
                                f(101011) = 1
                                                 f(111011) = 1
                f(011100) = 1
                                f(101100) = 0
                                                 f(111100) = 0
f(001100) = 1
f(001101) = 1
                f(011101) = 1
                                f(101101) = 0
                                                 f(111101) = 1
                                f(101110) = 1
                                                 f(1111110) = 0
f(001110) = 1
                f(011110) = 1
f(001111) = 1
                f(011111) = 1
                                f(101111) = 1
                                                 f(1111111) = 1
```

## 1.3 Задание лабораторной работы

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

- 1. Ввод информации происходит с переключателей SW[9:0];
- 2. SW[3:0] должны обрабатываться дешифратором «DC1», согласно индивидуальному заданию;
- 3. SW[7:4] должны обрабатываться дешифратором «DC2», согласно индивидуальному заданию;
- 4. Реализовать дешифратор «DC-DEC», преобразующий число, представленное в двоичном коде в цифру, отображаемую на семи сегментном индикаторе. Руководство-

ваться при этом нужно следующими соображениями:

- Семи сегментный индикатор подключается к шине HEX0[6:0]
- Диоды на семи сегментном индикаторе загораются при подаче на них низкого напряжения (0 горит, 1 не горит)
- Соответствие линий диодам семисегментного индикатора приведено ниже:



- 5. С помощью мультиплексора реализовать следующую схему подключения:
  - Если SW[9:8] = 00, на дешифратор DC-DEC поступает выход DC1;
  - Если SW[9:8] = 01, на дешифратор DC-DEC поступает выход DC2;
  - Если SW[9:8] = 10, на дешифратор DC-DEC поступает выход логической функции f;
  - Если SW[9:8] = 11, на дешифратор DC-DEC поступает SW[3:0].

Выполнив описание модуля на языке Verilog необходимо построить временные диаграммы его работы с помощью САПР Altera Quartus.

Привязать входы модуля к переключателям SW, отладочной платы, а выход к шине HEX0[6:0], получить прошивку для ПЛИС и продемонстрировать её работу.

# Лабораторная работа №2 Регистры и счётчики

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

Элемент памяти — один из самых важных элементов цифровых устройств. Чтобы не делать ошибок при разработке цифровых устройств, необходимо понять место этого узла, его идею и инструменты языка Verilog, связанные с ним.

Первый элемент памяти, который мы рассмотрим — это **за-щелка** (англ. latch).

Защелка является основой всех элементов памяти. Она состоит из двух элементов И-НЕ (или из двух элементов ИЛИ-НЕ, в зависимости от базиса, выбранного при проектировании), соединенных по следующей схеме:



Рис. 2.1: Структура RS-защелки

У защелки два входа и два выхода. Входами являются сигналы «сброс» и «установка в единицу» или по-английски «reset»

и «set». В зависимости от элементов, из которых состоит защелка, полярность входных сигналов будет меняться. В базисе И-НЕ сброс и установка происходят, когда соответственно сигналы R или S находятся в нуле, поэтому их обозначают как «не-сброс» и «не-установка», чтобы отразить этот факт. Выход защелки — это тот бит данных, который она хранит. Два выхода отличаются полярностью — один из них инвертирует хранимый бит. Ниже приведена таблица со всеми возможными комбинациями входных сигналов и временная диаграмма работы защелки.



Рис. 2.2: Временная диаграмма работы RS-защелки

Опишем защелку на языке Verilog, опираясь на её структуру, которую мы рассмотрели выше. Нам понадобятся два входа, два выхода и два элемента И-НЕ, которые мы опишем с помощью операций И (оператор &) и НЕ (оператор ~).

```
1
    module latch struct(
 2
   input nR,
   input nS,
 3
 4
   output Q,
 5
    output n();
 6
    assign Q = \sim (nS \& nQ);
 7
    assign nQ = \sim (nR \& Q);
 8
 9
10
    endmodule:
```

Листинг 2.1: Описание RS-защелки на языке Verilog HDL

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

лов.



Рис. 2.3: Структура D-защелки

Защелка теперь будет работать следующим образом: при высоком уровне на входе «разрешить работу» («enable») данные со входа «данные» («data») будут проходить через защелку на выход, при низком уровне на входе «разрешить работу» защелка будет сохранять на выходе последнее значение со входа «данные», которое было до переключения сигнала «разрешить работу». Работа такой защелки показана на временной диаграмме ниже.



Рис. 2.4: Временная диаграмма работы D-защелки

Как мы уже говорили, использовать структурные описания не всегда удобно. В большинстве случаев использовать поведенческое описание намного эффективнее. Поведенческое описание часто формулируется гораздо лаконичнее, и, так как его легче понять человеку, улучшается читаемость кода и уменьшается вероятность ошибок при его написании.

```
1 module d-latch_beahv(
2 input d,
3 input en,
4 output reg q);
```

```
5
6 always @(en, d) begin
7 if (en) q <= d;
8 end
9
10 endmodule;
```

Листинг 2.2: Поведенческое описание D-защелки на языке Verilog HDL

Если добавить к этой схеме еще две защелки, то можно привязать изменение «содержимого» защелки к переходу управляющего сигнала из «0» в «1», то получим следующую структуру:



Рис. 2.5: Структура D-триггера

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



Рис. 2.6: Графическое обозначение D-триггера

Эта схема получила широчайшее применение в цифровой схемотехнике и называется d-триггер (от слова «data» — данные). Ниже приведена временная диаграмма работы d-триггера.



Рис. 2.7: Пример работы регистра

Заметим, что сигнал С называют «тактирующим» сигналом или «сигналом синхронизации». Обычно в роли этого сигнала выступает сигнал от внешнего источника (чаще всего кварцевого резонатора) со стабильной частотой. А сами цифровые устройства, для работы которых необходим сигнал синхронизации, называют синхронными.

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

```
module d-flipflop beahv(
 1
  input d,
 2
3 input clk,
4 input rst,
 5
   input en.
 6
   output reg q);
 7
8
   always @(posedge clk or posedge rst) begin
       if (rst) q <= 0;
9
       else if (en) q <= d;</pre>
10
11
   end
12
13
   endmodule;
```

Листинг 2.3: Описание D-триггера на языке Verilog HDL

В описании появилось новое ключевое слово posedge. Оно используется только в списке чувствительности блока always и означает событие перехода сигнала, имя которого стоит после этого ключевого слова, из состояния «0» в состояние «1».

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

Добавление в список чувствительности события posedge rst позволяет описать поведение триггера в момент асинхронного сброса: как только случается переход rst из «0» в «1» срабатывает блок always и проверка условия **if** (rst) дает положительный результат, триггер сбрасывается в «0».

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

```
module register_behav(
input [7:0] d,
input clk,
input rst,
input en,
output reg [7:0] q);
```

```
8 always @(posedge clk or posedge rst) begin
9   if (rst) q <= 0;
10   else if (en) q <= d;
11 end
12
13 endmodule;</pre>
```

Листинг 2.4: Описание регистра на языке Verilog HDL

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

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

Необходимо отметить важную концепцию языка Verilog. Переменные типа reg могут быть изменены только в пределах одного блока always. Переменные доступны для проверки в любом из блоков, но изменять их значение можно только в одном из них.

```
1
    req a;
2
    req b;
 3
4
    always @(posedge clk) begin
 5
        if (in < 5) a <= in;
6
    end
 7
 8
    always @(posedge clk) begin
 9
        if (n > 5) begin
10
            b <= in:
            a <= in - 5; //ошибка!!!
11
12
        end
13
        else b <= a;</pre>
14 end
```

Листинг 2.5: Пример присвоения значения переменной а разных блоках always на языке Verilog HDL

Одной из простейших, и в тоже время широко распространённой, цифровой схемой на основе регистров является счёт-

#### чик.

Счётчик считает количество тактов, которое прошло с момента его обнуления.

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

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

В результате получим следующую схему:



Рис. 2.8: Структура восьмибитного счетчика

На временной диаграмме ниже хорошо видно как работает счётчик:



Рис. 2.9: Пример работы регистра

Опишем поведение такого счётчика на Verilog.

```
module counter 8bit(
   input clk,
   input en,
 3
   input rst,
 4
 5
   output reg [7:0] counter);
 6
 7
   always @(posedge clk or posedge rst) begin
       if (rst) counter <= 0;</pre>
 8
       else if (en) counter <= counter + 1;</pre>
 9
10
   end
11
12 endmodule:
```

Листинг 2.6: Описание восьмиитного счетчика на языке Verilog HDL

Для того чтобы можно было подсчитывать события, а не переходы сигнала синхронизации из «0» в «1» понадобится ввести еще одну схему. Её смысл и назначение заключается в следующем: нам необходимо из асинхронного события получить синхронный сигнал единичной длительности. Тогда, подавая такой сигнал на вход enable счётчика, мы сможем считать количество произошедших событий.

Ниже представлена схема, позволяющая сделать это:



Рис. 2.10: Структура регистра на D-триггерах



Рис. 2.11: Пример работы регистра

Естественно такая схема работает только тогда, когда входной сигнал изменяется с частотой меньшей, чем частота синхронизации.

Сигнал out в таком случае подключается к входу enable счётчика.

## 2.1 Задание лабораторной работы

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

- 1. Ввод информации происходит с переключателей SW[9:0] и кнопок KEY[0], KEY[1]. Внешний источник сигнала синхронизации: CLK50;
- 2. КЕҮ[1] должна функционировать как общий асинхронный сброс устройства;
- 3. При нажатии на KEY[0] записывать данные с SW[9:0] в десятиразрядный регистр;
- 4. Содержимое десятиразрядного регистра выводить на LEDR[9:0];
- 5. При нажатии на KEY[0] увеличивать 8-ми разрядный счётчик нажатий на 1, если произошло событие, указанное в индивидуальном задании студента;
- 6. Содержимое счётчика выводить в шестнадцатеричной форме на HEX0 и HEX1 (цифры с 0 до 9 и буквы A, B, C, D, E, F)

Выполнив описание модуля на языке Verilog необходимо построить временные диаграммы его работы с помощью САПР Altera Ouartus.

Привязать входы модуля к переключателям SW, отладочной платы, а выход к шине HEX0[6:0], получить прошивку для

## 2.2 Пример индивидуального задания

Событием является наличие 3 и более единиц на SW[9:0] в момент записи в регистр.

```
reg sw event;
 2
   always @(SW) begin
 3
        if ((SW[0] + SW[1] + SW[2] + SW[3])
 4
           + SW[4] + SW[5] + SW[6] + SW[7]
 5
           + SW[8] + SW[9]) > '4d3) sw event <= '1b1;
 6
        else sw event <= '1b0;</pre>
 7
   end
 8
   reg [2:0] event sync reg;
   wire synced event;
10
   assign synced event = event sync reg[1]
11
12
                        & ~event sync reg[0];
13
   always @(posedge CLK50) begin
14
15
       event sync reg[2] <= sw event;
       event sync reg[1:0] <= event sync reg[2:1];</pre>
16
17 end
```

Листинг 2.7: Решение индивидуального задания (фрагмент кода лабораторной работы)

# 2.3 Вопросы к защите лабораторной работы

- 1. Какие элементы памяти вы изучили в данной лабораторной работе?
- 2. Чем отличается RS-защелка от D-защелки?
- 3. Какие входы могут быть у триггера? Перечислите все и назовите их функции.

- 4. Какие блоки вашего цифрового устройства синхронные? Какие нет? Почему?
- 5. Какой фрагмент вашего кода описывает вывод значения счетчика на семи сегментный индикатор? Как называется эта цифровая схема?
- 6. Продемонстрируйте код реализующий индивидуальное задание.
- 7. Покажите в коде лабораторной код счётчика.
- 8. Что такое сигнал синхронизации?

# Лабораторная работа №3 Секундомер

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

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

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

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

В случае секундомера справедливы такие рассуждения:

Чтобы управлять работой секундомера нам понадобятся два входа: «старт/стоп» и «сброс».

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

Для отображения времени выделим 2 индикатора для отображения количества прошедших секунд и 2 индикатора для

отображения количества прошедших десятых и сотых долей секунды.

Значит выходом секундомера будут четыре семибитные шины для управления индикаторами.

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

Т.е. нам потребуется сигнал синхронизации со стабильной частотой.

Больше никаких входов и выходов не требуется. Общая схема на данный момент выглядит так:



Рис. 3.1: Описание входов и выходов секундомера

Начнём описывать модуль на языке Verilog:

```
module stopwatch (
input start_stop,
input reset,
input clk,
output [6:0] hex0,
output [6:0] hex1,
output [6:0] hex2,
output [6:0] hex3);
endmodule;
```

Листинг 3.1: Описание входов и выходов модуля на языке Verilog HDL

Теперь приступим к описанию «внутренностей» модуля.

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

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

Например, если в устройстве установлен кварцевый генератор на 26 МГц, то одной секунде соответствует 26 миллионов тактовых импульсов, а одной сотой секунды соответствует 260 тысяч тактовых импульсов.

Для того, чтобы отсчитать это количество импульсов подходит единственный из известных нам «строительных блоков» - счётчик:



Рис. 3.2: Структура счетчика

Как мы уже говорили, счётчик состоит из регистра и сумматора. Чтобы счётчик циклически отсчитывал одну сотую секунды его необходимо обнулить после того, как он отсчитает 260 тысяч тактовых импульсов. В этот же момент нужно выработать сигнал для остальной схемы, что прошла одна сотая секунды.

Из всех цифровых блоков, которые мы рассмотрели, для реализации задачи сравнения текущего значения счётчика с константой подходит только компаратор. На один из входов компаратора подадим текущее значение счётчика, а на другой вход - константу 260 000.



Рис. 3.3: Структура счетчика с компаратором

Пока значения на входах компаратора будут отличаться, на выходе компаратора будет значение «0». Когда значения будут равны, компаратор изменит выход с «0» на «1», это и будет признак того, что прошло 0.01 секунды. Для того, чтобы можно было эффективно использовать сигнал «прошло 0.01с», этот сигнал должен иметь длительность равную 1 такту.

Этот же сигнал мы будет использовать для управления сбросом счётчика.

Итак, счётчик должен после достижения значения  $260\,000$  принять значение <0>, но переход должен случиться, как и все остальные переходы, в момент перехода тактового сигнала из <0> в <1>.

Сброс, отвечающий таким условиям, называется «синхронный сброс».

Посмотрите, как будет выглядеть на временной диаграмме как будет работать счётчик если выход компаратора, подключить как сигнал синхронного сброса:



Рис. 3.4: Временная диаграмма работы синхронного счетчика

А так выглядит временная диаграмма, если подключить выход компаратора к входу асинхронного сброса триггера:



Рис. 3.5: Временная диаграмма работы асинхронного счетчика

Выход компаратора, установившись в единицу, моментально сбросит счётчик и, так как значение счётчика изменилось, а значит, изменился и один из входов компаратора, выход компаратора сразу же перейдет в значение «0».

Обратите внимание, что длительность сигнала с выхода компаратора должна быть равна одному такту. Ведь в дальнейшем нам необходимо будет считать события «прошла одна сотая секунды», а значит подготовить сигнал единичной длительности, который соответствует этому событию (см. лабораторную работу №3).

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

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



Рис. 3.6: Схема реализации синхронного сброса

Если подключить sync\_reset к выходу компаратора, то когда счётчик достигнет порогового значения, выход компаратора изменится и переключит мультиплексор. Теперь на выход мультиплексора будет подаваться «0». Этот сигнал будет

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

Для общего сброса секундомера при нажатии кнопки «сброс» как раз можно воспользоваться входом асинхронного сброса регистра. Ведь при нажатии кнопки «сброс» можно обнулять регистр мгновенно.

Теперь надо выбрать правильный сигнал управления работой счётчика - сигнал разрешения работы (Enable, EN). Ведь счётчик должен начинать считать после нажатия кнопки «старт/стоп», а после её повторного нажатия должен останавпиваться.

Для управления работой счётчика можно использовать сигнал, который будет единицей, пока счётчик должен работать и нулём, если отсчёт времени остановлен. Как раз такой сигнал можно подать на вход разрешения работы регистра. Назовём этот сигнал «device running».

Посмотрите, как выглядит остановка и запуск счётчика в таком случае:



Рис. 3.7: Временная диаграмма работы сигнала device running

К проектированию и описанию схемы, которая вырабатывала бы сигнал «device\_running», мы вернемся позднее.

Стоит обратить внимание на следующий момент: что будет, если счётчик остановить в тот момент времени, когда его значение стало равно 259999?

Взгляните на временную диаграмму:



Рис. 3.8: Временная диаграмма работы счетчика

Для того чтобы не допустить такого поведения, можно немного изменить условие, запрещающее работу счётчика. Теперь мы будем дополнительно проверять сигнал с компаратора. И если счётчик в данный момент равен 259999, то запретить его работу будет невозможно.

Для решения этой задачи подойдет вентиль «или». Условие будет таким: «работа разрешена, если сигнал «device\_running» равен единице **ИЛИ** когда текущее значение счётчика равно 259999».

Теперь схема счётчика выглядит следующим образом:



Рис. 3.9: Схема счетчика тысячных долей секунды

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

```
8
   //описание счётчика
   always @(posedge clk or posedge reset) begin
 9
       if (reset) pulse counter <= 0;</pre>
10
           //асинхронный сброс
11
12
13
       //сигнал" разрешения работы счётчика"
14
       else if (device running |
15
                   hundredth of second passed)
16
           //синхронный сброс
17
           if (hundredth of second passed)
18
19
                   pulse counter <= 0;</pre>
20
           //увеличение" счётчика на единицу"
21
           else pulse counter <= pulse counter + 1;</pre>
22
23
   end:
```

Листинг 3.2: Описание счетчика тактовых импульсов на языке Verilog HDL

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

Перед нами встаёт выбор.

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



Рис. 3.10: Первый пример реализации секундомера

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

Т.е. первый счетчик подсчитывает количество прошедших сотых долей секунды от 0 до 9, и, затем обнуляется, вырабатывая сигнал «прошла десятая доля секунды». Следующий счётчик, точно также считает уже десятые доли и вырабатывает сигнал «прошла одна секунда» и так далее.



Рис. 3.11: Второй пример реализации секундомера

Второй вариант для нас проще в реализации, компактнее, удобнее и понятнее.

Поэтому выберем именно его.

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

Счетчики подойдут нам потому, что функция их идентична - подсчёт событий с ограничением диапазона. Изменить нужно будет только разрядность счётчика с 16 на 4 и верхнюю границу счёта с 260000 на 9. Тогда счётчик будет выдавать признак переполнения (достижения границы отсчёта, когда его значение будет становиться) девяткой.

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

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



Рис. 3.12: Временная диаграмма работы секундомера

Счетчик будет переключаться каждую 0,01 секунды, 0,1 секунды ,1 секунду или 10 секунд. И выход компаратора будет устанавливаться в 1 на всё время, которое потребуется для переключения счётчика из 9 в ноль. Т.е. в случае счётчика сотых долей секунды потребуется 259999 тактов.

Как выделить из всего времени, пока счётчик имеет значение «9» сигнал длительностью в один такт, который возникает в нужный момент времени? На временной диаграмме этот сигнал отмечен как «прошла 0,1с.»

Сигнал «прошла 0,1с» можно получить из сигналов, представленных на временной диаграмме следующим образом:

«прошла 0.1с» правда, когда выход компаратора равен единице  ${\bf M}$  «прошла 0.01с».

Схема счётчика практически не изменилась:



Рис. 3.13: Схема счётчика сотых долей секунды

Скорректируем описание её работы на Verilog:

```
//регистр счётчика
 2
   reg [3:0] hundredth counter = "4'd0";
 3
 4
   //описание компаратора
   wire tenth of second passed =
 5
 6
           ((hundredths counter == "'4d9") &
 7
             hundredths of second passed);
 8
   //описание счётчика
 9
   always @(posedge clk or posedge reset) begin
10
       if (reset) hundredths counter <= 0;</pre>
11
           //асинхронный сброс
12
13
       //сигнал" разрешения работы счётчика"
14
       else if (hundredth of second passed)
15
16
17
           //синхронный сброс
           if (tenth of second passed)
18
                  hundredths counter <= 0;
19
20
21
           //увеличение" счётчика на единицу"
22
           else hundredths counter <=</pre>
                  hundredths counter + 1;
23
```

#### 24 end:

## Листинг 3.3: Описание счетчика сотых долей секунды на языке Verilog HDL

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

Теперь вернёмся к вопросам, которые мы отложили ранее.

В нашем устройстве пока нет описания схемы, которая вырабатывает сигнал «device\_stopped». Сигнал должен управляться кнопкой, поэтому как мы уже говорили, в лабораторной работе №3 потребуется схема, синхронизирующая сигнал, поступающий с кнопки с внутренним сигналом clk (тактовые импульсы).

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

Тогда схема будет абсолютно такой же, как и в лабораторной работе №3 и будет выглядеть следующим образом:



Рис. 3.14: Структура схемы синхронизации сигнала

Поведение такой схемы описывается на языке Verilog следующим образом:

```
reg [2:0] button_syncroniser;
wire button_was_pressed;
3
```

```
4 always @(posedge clk) begin
5    button_syncroniser[0] <= in;
6    button_syncroniser[1] <= button_syncroniser[0];
7    button_syncroniser[2] <= button_syncroniser[1];
8 end;
9
10 assign button_was_pressed <= ~button_syncroniser[2]
11    & button_syncroniser[1];</pre>
```

Листинг 3.4: Описание схемы синхранизации на языке Verilog HDL

Эта схема и её описание подробно рассмотрены в лабораторной работе №3

Теперь нам нужно построить схему, которая по нажатию кнопки переключала бы сигнал «device\_stopped» из «0» в «1» и из «1» в «0».

Что нам понадобится? Триггер, чтобы хранить значение «device\_stopped». Чтобы менять значение на противоположное надо знать противоположное значение, значит, нужен инвертор. Событие должно случаться по сигналу «button\_was\_pressed», а значит речь, скорее всего, идет о входе разрешения работы триггера.

Немного подумав над этими вводными, нетрудно составить следующую схему:



Рис. 3.15: Схема переключения сигнала «device\_running»

Временная диаграмма, которая соответствует работе этого устройства:



Рис. 3.16: Временная диаграмма переключения сигнала «device running»

Описание поведения этой схемы на Verilog также не представляет сложности. Выполните его самостоятельно.

Теперь приведем полное описание секундомера (за исключением схемы, вырабатывающей сигнал «device\_stopped»), выполненное на языке Verilog:

```
module stopwatch (
 1
 2 input start stop,
 3 input reset,
 4 input clk,
 5 output [6:0] hex0, //индикатор" сотых долей секунды"
 6 output [6:0] hex1, //десятых" долей секунды"
   output [6:0] hex2, //секунд""
 7
  output [6:0] hex3); //десятков" секунд"
 8
 9
   // Часть" I - синхронизация обработки"
10
   // нажатия" кнопки «СтартСтоп»/"
11
   reg [2:0] button syncroniser;
12
          wire button was pressed:
13
14
15
   always @(posedge clk) begin
16
       button syncroniser[0] <= start stop;</pre>
       button syncroniser[1] <= button syncroniser[0];</pre>
17
       button syncroniser[2] <= button syncroniser[1];</pre>
18
19
   end
20
   assign button was pressed = ~button syncroniser[2]
21
22
                                & button syncroniser[1];
23
24
   // Часть" II - выработка признака «"device running»""
25
   // Самостоятельная" работа студента!"
26
```

```
27
    reg device running;
28
29
30
   // Часть" III - счётчик импульсов"
31
32
   // и" признак истечения 0.01 сек"
33
    reg [16:0] pulse counter = "17'd0";
    wire hundredth of second passed =
34
           (pulse counter == "'17d259999");
35
36
    always @(posedge clk or posedge reset) begin
37
       if (reset) pulse counter <= 0;</pre>
38
           //асинхронный сброс
39
       else if (
                  device running |
40
                  hundredth of second passed)
41
           if (hundredth of second passed)
                  pulse counter <= 0;</pre>
42
43
           else pulse counter <= pulse counter + 1;</pre>
44
          end
45
46
    // Часть" IV - основные счётчики"
47
48
    reg [3:0] hundredths counter = "4'd0";
    wire tenth of second passed =
49
           ((hundredths counter == "4'd9") &
50
             hundredth of second passed);
51
    always @(posedge clk or posedge reset) begin
52
53
       if (reset) hundredths counter <= 0;</pre>
54
       else if (hundredth of second passed)
55
           if (tenth of second passed)
                  hundredths counter <= 0;
56
57
           else hundredths counter <=</pre>
58
                  hundredths counter + 1;
59
          end
60
61
    reg [3:0] tenths counter = "4'd0";
62
63
    wire second passed = ((tenths counter == "'4d9") &
64
                            tenth of second passed);
65
    always @(posedge clk or posedge reset) begin
       if (reset) tenths counter <= 0;</pre>
66
```

```
67
        else if (tenth of second passed)
 68
            if (second passed) tenths counter <= 0:</pre>
            else tenths counter <= tenths counter + 1;</pre>
 69
 70
           end
 71
 72
     reg [3:0] seconds counter = "4'd0";
 73
    wire ten seconds passed =
 74
            ((seconds counter == "'4d9") &
 75
              second passed);
    always @(posedge clk or posedge reset) begin
 76
 77
        if (reset) seconds counter <= 0;</pre>
 78
        else if (second passed)
 79
            if (ten seconds passed) seconds counter <= 0;</pre>
 80
            else seconds counter <= seconds counter + 1;</pre>
 81
           end
 82
 83
     reg [3:0] ten seconds counter = "4'd0";
 84
     always @(posedge clk or posedge reset) begin
 85
        if (reset) ten seconds counter <= 0;</pre>
        else if (ten seconds passed)
 86
            if (ten seconds counter == "'4d9")
 87
 88
                   ten seconds counter <= 0;
            else ten seconds counter <=</pre>
 89
 90
                   ten seconds counter + 1;
 91
           end
 92
 93
 94
95
    // Часть" V - дешифраторы для отображения"
 96
 97
    // содержимого" основных регистров"
98
    // на" семисегментных индикаторах"
 99
     reg [6:0] decoder ten seconds;
    always @(*) begin
100
101
        case (ten seconds counter)
            "'4d0": decoder ten seconds <= "'7b0000001";
102
            "'4d1": decoder ten seconds <= "'7b1001111"
103
104
            "'4d2": decoder ten seconds <= "'7b0010010"
            "'4d3": decoder ten seconds <= "'7b0000110"
105
            "'4d4": decoder ten seconds <= "'7b1001100";
106
```

```
107
            "'4d5": decoder ten seconds <= "'7b0100100";
            "'4d6": decoder ten seconds <= "'7b0100000";
108
            "'4d7": decoder ten seconds <= "'7b0001111";
109
            "'4d8": decoder ten seconds <= "'7b0000000";
110
            "'4d9": decoder ten seconds <= "'7b0000100":
111
            default: decoder ten seconds <= "'7b1111111";</pre>
112
113
        endcase;
114
     end
115
     assign hex3 = decoder ten seconds;
     reg [6:0] decoder seconds;
116
117
     always @(*) begin
118
        case (seconds counter)
            "'4d0": decoder seconds <= "'7b0000001";
119
120
            "'4d1": decoder seconds <= "'7b1001111";
            "'4d2": decoder seconds <= "'7b0010010";
121
            "'4d3": decoder seconds <= "'7b0000110";
122
            "'4d4": decoder seconds <= "'7b1001100"
123
            "'4d5": decoder seconds <= "'7b0100100"
124
            "'4d6": decoder seconds <= "'7b0100000";
125
126
            "'4d7": decoder seconds <= "'7b0001111";
127
            "'4d8": decoder seconds <= "'7b0000000"
            "'4d9": decoder seconds <= "'7b0000100";
128
            default: decoder seconds <= "'7b1111111";</pre>
129
130
        endcase;
131
     end
132
     assign hex2 = decoder seconds;
133
     reg [6:0] decoder tenths;
134
135
     always @(*) begin
        case (tenths counter)
136
137
            "'4d0": decoder tenths <= "'7b00000000";
138
            "'4d1": decoder tenths <= "'7b1001111";
139
            "'4d2": decoder tenths <= "'7b0010010"
            "'4d3": decoder tenths <= "'7b0000110"
140
            "'4d4": decoder tenths <= "'7b1001100"
141
            "'4d5": decoder tenths <= "'7b0100100"
142
            "'4d6": decoder tenths <= "'7b0100000"
143
144
            "'4d7": decoder tenths <= "'7b0001111"
            "'4d8": decoder tenths <= "'7b0000000"
145
            "'4d9": decoder tenths <= "'7b0000100";
146
```

```
147
            default: decoder tenths <= "'7b11111111";</pre>
148
        endcase:
     end
149
     assign hex1 = decoder tenths;
150
151
152
     reg [6:0] decoder hundredths;
153
     always @(*) begin
154
        case (hundredths counter)
            "'4d0": decoder hundredths <= "'7b00000000";
155
            "'4d1": decoder hundredths <= "'7b1001111";
156
            "'4d2": decoder hundredths <= "'7b0010010";
157
158
            "'4d3": decoder hundredths <= "'7b0000110";
            "'4d4": decoder hundredths <= "'7b1001100";
159
            "'4d5": decoder hundredths <= "'7b0100100"
160
            "'4d6": decoder hundredths <= "'7b0100000"
161
            "'4d7": decoder hundredths <= "'7b0001111"
162
            "'4d8": decoder hundredths <= "'7b0000000"
163
            "'4d9": decoder hundredths <= "'7b0000100";
164
            default: decoder hundredths <= "'7b1111111":</pre>
165
166
        endcase:
167
     end
     assign hex0 = decoder hundredths;
168
169
170
    endmodule
```

Листинг 3.5: Описание секундомера на языке Verilog HDL

#### 3.1 Задание лабораторной работы:

Изучить разработку к лабораторной работе.

Самостоятельно выполнить описание схемы, вырабатывающей сигнал «device\_stopped».

Выполнить синтез и моделирование работы счётчика.

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

Изучить работу устройства, реализованного в ПЛИС учебного стенда.

Подготовить ответы на вопросы к защите лабораторной работы.

# 3.2 Вопросы к защите лабораторной работы

\*in progress\*

### Лабораторная работа №5 RAM-память

Мы уже познакомились с многими блоками, из который состоят цифровые устройства. Последний базовый «строительный» блок, который мы рассмотрим в этом курсе — память с произвольным доступом (Random Access Memory).

RAM память в цифровых устройствах делится на три типа:

- Статическая память
- Динамическая память
- Энергонезависимая память

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

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

Динамическая память использует другой принцип хранения информации. В качестве элемента памяти в динамической памяти используются конденсаторы. Когда конденсатор заряжен — ячейка хранит «1». Когда разряжен — «0». Эта схема гораздо проще в реализации и требует существенно меньше ресурсов по сравнению со статической памятью. А значит на одном кристалле можно разместить большее количество памяти.

Но конденсатор неизбежно со временем разряжается через сопротивления утечки. Поэтому динамическая память требует постоянного обновления данных, которые были туда записаны. Обновление происходит благодаря внутренне-

му контроллеру, который проходит по всем адресам в памяти, считывая данные, а затем записывая эти же данные обратно и таким образом «обновляет» заряд конденсаторов.

Минусом динамической памяти является большое потребление тока, по сравнению со статической.

Наиболее известным представителем энергонезависимой памяти на данный момент является Flash память. Flash память базируется на транзисторах с плавающим затвором.

Выделяют два вида Flash памяти — NOR и NAND. Последняя получила наибольшее настроение.

У Flash памяти есть недостатки:

- высокая технологическая сложность и, соответственно, стоимость
- деградация ячеек памяти, при повторной перезаписи информации (сотни тысяч операций записи для коммерческих продуктов, миллионы 2012 год и сотни миллионов 2014 год)
- деградация ячеек памяти при чтении (аналогично записи)
- блочная организация (удалить можно только блока информации целиком)

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

Возможно, вы уже знаете основные принципы функционирования памяти в цифровой технике. Тем не менее, напомним, каким образом работает RAM память.

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

# RAM-память Данные считанные для записи данные Адрес записи чтения Разрешение Разрешение чтения записи Тактовые импульсы

Рис. 4.1: Входные и выходные сигралы RAM-памяти

Как происходит запись данных в память и чтение из неё?

На вход «данные для записи» подаются данные, которые мы хотели бы записать в память. Одновременно с этим на вход «адрес записи» подается адрес ячейки, в которую мы хотели поместить входные данные. Запись происходит по положительному фронту сигнала синхронизации, когда вход «запись разрешена» находится в «1».

Посмотрите на временную диаграмму записи данных в память:



Рис. 4.2: Временная диаграмма записи в память

Чтение из RAM памяти происходит следующим образом: на вход «адрес чтения» подается адрес ячейки, из которой мы хотим получить данные, затем на вход «разрешение чтения» подается «1». По положительному фронту сигнала синхронизации данные выгружаются из памяти и становятся доступны для чтения:



Рис. 4.3: Временная диаграмма чтения из памяти

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

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



Результат записи: «ОА» по адресу «О1» «ОВ» по адресу «О2» «ОС» по адресу «О3» «ОD» по адресу «О4»

Рис. 4.4: Временная диаграмма последовательной записи данных

Обратите внимание на смещение считанных данных при последовательном чтении информации из памяти:



Рис. 4.5: Временная диаграмма последовательного чтения из памяти

Как же устроена статическая RAM-память?

Общую структуру статической памяти можно представить следующим образом:



Рис. 4.6: Общая структура статической памяти

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

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

Чтение происходит подобным образом: адрес «выбирает» по одной ячейке из каждого столбца и из значений этих ячеек формируется вектор выходных данных.

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

Обратите внимание, что схема управления тоже имеет регистр для хранения сигналов разрешения записи и разрешения чтения.

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

Ha основе RAM памяти строятся некоторые другие виды памяти.

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



Рис. 4.7: Организация записи данных в буфер

В англоязычной литературе такой блок памяти носит название FIFO от словосочетания First In First Out - Первый Пришел Первый Ушел. Название FIFO настолько широко распространено, что оно практически вытеснило русское название «очередь».

Так как данные всегда записываются в конец очереди, а читаются всегда из её начала, то интерфейс FIFO не содержит адресной шины.

В основе FIFO, как мы уже говорили, лежит RAM память, а

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

Запись происходит с нулевого адреса. После помещения информации в память, управляющий блок увеличивает адрес на единицу. Чтение происходит точно также, начиная с нулевого адреса. Когда происходит чтение, адрес чтения также увеличивается на единицу. Управляющему блоку необходимо следить, чтобы адрес чтения не «перегонял» адрес записи.

Все ячейки, адреса которых находятся «между» адресом чтения и адресом записи — заполнены, а все ячейки, адреса которых находятся «снаружи» этих адресов — пусты.



Рис. 4.8: Порядок чтения и записи в память

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

Подумайте, каким образом устройство управления может выработать эти сигналы?

Также часто в FIFO добавляют дополнительные признаки: «почти пуста» - означает, что в очереди только одно значение и «почти полна» - означает, что есть только одна свободная ячейка. Эти сигналы в некоторых случаях облегчают работу с FIFO.

С точки зрения управления записью и чтением, FIFO работает также, как и блок RAM памяти, т.е. требует сигналов разрешения записи и разрешения чтения.

FIFO часто применяют в качестве буфера, для того, что-

бы сохранить быстро поступающую информацию и, в дальнейшем, обработать её. Такой блок часто можно встретить в контроллерах интерфейсов, где обмен данными часто происходит «волнами».



Рис. 4.9: Запись патетов данных в FIFO

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

Продемонстрируем решение типовой задачи: спроектируем конечный автомат, задачей которого будет управление FIFO и передатчиком некоторого интерфейса. Автомат будет по готовности передатчика при наличии данных в FIFO выгружать из него новое значение и инициировать его передачу. Таким образом, мы реализуем буфер передачи.

Вот структурная схема устройства, которое мы хотим реализовать:



Рис. 4.10: Структура утройства управления передачей данных

Граф для конечного автомата будет выглядеть следующим образом:



Рис. 4.11: Граф работы конечного автомата

В состоянии «ожидание» автомат ждёт готовности передатчика и появления данных в FIFO. Затем, когда данные появляются и передатчик свободен, автомат переходит в состояние «запросить данные у FIFO», затем сразу же в состояние «Начать передачу» и, затем, снова в состояние «ожидание».

Временная диаграмма будет выглядеть следующим образом:



Рис. 4.12: Временная диаграмма последовательного чтения из памяти

Внимательно изучите временную диаграмму. Обратите внимание на то, каким образом спроектирован конечный ав-

томат. Отметьте, что сигнал инициирования передачи совпадает с состоянием «start», а сигнал запроса чтения для FIFO совпадает с состоянием «read». Также заметьте, что инициирование передачи происходит как раз в тот момент, когда данные с FIFO уже готовы для чтения.

Приступим к описанию нашего модуля, используя FIFO и передатчик, в качестве компонентов.

```
module example lab6 (
 2
            input clk,
 3
            input rst,
 4
            input [7:0] data,
 5
            input we,
 6
            output full,
 7
            output transmit lane);
 8
 9
   localparam idle = 2'b00;
10
   localparam load = 2'b01;
11
    localparam transmit = 2'b10:
    localparam wait trasnaction to complete = 2'b11;
12
13
14
   reg [1:0] state;
15
   wire [7:0] data from fifo for transmitter;
16 wire fifo is empty;
17
   wire fifo re;
    reg transmitter is busy;
18
19
    reg start transaction;
20
21
    always @(posedge clk) begin
22
            case (state) is
23
                     idle:
24
                              if (fifo is empty |
25
                                  transmitter is busy)
26
                                      state <= idle:</pre>
27
                              else state <= load;</pre>
28
                              end if:
29
                     load: state <= transmit:</pre>
30
                     transmit: state <= idle;</pre>
31
            endcase:
32 end
```

```
33
   assign fifo re = (state == load);
34
   assign start transaction = (state == transmit);
35
36
37
    fifo fifo input buffer(
38
            .we(we),
39
            .re(fifo re),
40
            .data in(data),
            .data out(data from fifo for transmitter),
41
            .empty(fifo is empty),
42
43
            .full(full)
44
    );
45
   tansmitter my transmitter(
46
47
            .start(start),
48
            .busy(busy),
49
            .data(data from fifo for transmitter),
            .tx(transmit lane)
50
51
   ):
52
53
   end:
```

Листинг 4.1: Описание проектируемого устройства на языке Verilog HDL

#### 4.1 Задание лабораторной работы:

Изучить разработку к лабораторной работе.

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

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

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

Получить файл конфигурации для ПЛИС учебного стенда и продемонстрировать работу устройства.

# 4.2 Вопросы к защите лабораторной работы

Что такое RAM-память?

Изобразите обобщенную структуру RAM-памяти.

RAM-память это синхронное или асинхронное устройство?

Опишите все входные и выходные сигналы RAM памяти, известные вам.

Нарисуйте временную диаграмму записи значения в RAM память.

Нарисуйте временную диаграмму чтения значения из RAM памяти.

Как функционирует буфер FIFO?

Опишите все входные и выходные сигналы FIFO памяти, известные вам.

# Лабораторная работа №6 FLASH память

#### 5.1 Виды энергонезависимой памяти

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

Чтобы решить эту проблему, на заре вычислительной техники, данные в цифровое устройство после подачи питания загружали с таких носителей, как перфокарты и, позже, магнитные ленты. Ещё позже для этих целей были разработаны накопители на гибких магнитных дисках — дискетах и жёстких магнитных дисках — «HDD». На данный момент для хранения данных при отсутствии питания наиболее широко применяется FLASH-память.

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

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

#### 5.2 Принципы работы FLASH-памяти

В качестве элемента хранения информации FLASH-память

использует транзистор с плавающим затвором. Состояние затвора определяет бит хранимой информации.

На Рис.5.1 схематично изображена структура такого транзистора.



Рис. 5.1: Транзистор с плавающим затвором

Как вы видите, он содержит два затвора: управляющий и плавающий. Плавающий затвор — это полупроводник, который полностью окружён диэлектриком. При этом плавающий затвор способен накапливать электроны.

От величины накопленного заряда меняется «лёгкость», с которой транзистор открывается — т.е. величина напряжения «управляющий затвор—исток», при котором через транзистор начнёт течь ток.

Чем больше электронов находятся в плавающем затворе, тем «легче» открывается транзистор — ток начинает про-

текать через него при меньшем напряжении «управляющий затвор—исток».

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

Для хранения информации используют следующий принцип: чтобы считать информацию, на управляющий затвор подаётся напряжение чтения — среднее между самым сильным и самым слабым напряжением, способным открыть затвор. Если транзистор открывается, значит в плавающем затворе были электроны и мы считаем, что в нём записано значение «0», если не открывается, значит электронов в плавающем затворе нет и записано значение «1».



Рис. 5.2: Чтение значение из транзистора с плавающим затвором

Осталось понять как можно «заставить» электроны попадать в плавающий затвор, ведь он изолирован диэлектриком. Не вдаваясь в подробности скажем, что если подать достаточно высокое напряжение «управляющий затвор—сток», то у электронов хватит энергии, чтобы «перескочить» диэлектрик и попасть в плавающий затвор. А если изменить полярность этого напряжения, то можно «выгнать» электроны наружу (см. Рис.5.3).



Рис. 5.3: Изменение состояния плавающего затвора

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

Теперь мы знаем, что для того чтобы записать или считать информацию из FLASH-памяти надо использовать большую разность потенциалов. Но на самом деле транзистор устроен таким образом, что энергия, необходимая чтобы «загнать» электроны в плавающий затвор меньше энергии, необходимой, чтобы их «выгнать». Это делается чтобы при чтении значения электроны не покидали плавающий затвор.

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

#### 5.3 Особенности FLASH-памяти

Из-за особенностей транзистора с плавающим затвором, которые мы рассмотрели можно выделить следующие характерные черты FLASH-памяти:

- Запись значения возможна только из логической «1» в логический «0»;
- Удаление информации возможно только из группы ячеек одновременно (сектора);
- Удаление и запись информации приводят к деградации ячеек памяти;
- Чтение также приводит к деградации ячеек памяти, но в меньшей степени.

#### 5.4 Структура FLASH-памяти

На Рисунке 5.4 изображена общая структура FLASHпамяти. Как видно она практически не отличается от RAMпамяти: из ячеек строится матрица, контролируемая управляющим блоком. А сам управляющий блок обеспечивает коммуникацию с внешними устройствами, дешифрацию адреса и управление записью и чтением массива элементов памяти. Подключение FLASH-памяти и управление ей со стороны цифрового устройства полностью зависит от того, как реализован блок управления — доступ к содержимому FLASH-памяти может быть синхронный или асинхронный, по последовательной или параллельной шине, с разделением шин адреса и данных или без него.



Рис. 5.4: Структура flash-памяти S29AL032D

# 5.5 Микросхема FLASH-памяти S29AL032D

Для практического знакомства с FLASH-памятью мы спроектируем контроллер микросхемы S29AL032D. Именно эта микросхема установлена на отладочной плате Altera DE1.

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

В нашем случае микросхема уже подключена, поэтому из всего datasheet нас, в первую очередь, интересует каким образом необходимо взаимодействовать с этой микросхемой, чтобы записать или считать данные.

Для разработки контроллера следует ознакомиться со следующими разделами документа:

- 11. Commands Definitions;
- 12. Write Operation Status;
- 17. AC Characteristics.

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

#### 5.5.1 Проектирование контроллера S29AL032D

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

Чтобы начать разработку контроллера, нужно ответить важные вопросы: как должен работать наш контроллер и как он должен управляться?

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

Теперь, когда мы определились с тем, как мы будем управлять контроллером, нам нужно понять как он должен взаимодействовать с самой микросхемой FLASH-памяти. Для этого изучим информацию о самой микросхеме и операциях записи и чтения, описанные в datasheet S29AL032D.

#### 5.5.2 Операция чтения

Важной особенностью микросхемы S29AL032D является использование двунаправленной шины данных. Это сделано для минимизации разводки печатной платы при её использовании. В дальнейшем нам придется разработать механизм,

позволяющий использовать одну шину и для операции чтения и для операции записи.

Чтение данных из микросхемы S29AL032D не требует никакой дополнительной подготовки. Временная диаграмма чтения приведена в пункте 17.2 datasheet и представлена на Рисунке5.5.

Времена, указанные на диаграмме, приведены в Таблице 5.1.



Рис. 5.5: Временная диаграмма чтения S29AL032D

| Обозн.     | Описание                                  | Min   | Max   |
|------------|-------------------------------------------|-------|-------|
| $t_{RC}$   | Продолжительность цикла чтения            | 70 нс | _     |
| $t_{ACC}$  | Задержка Адрес — Данные                   | _     | 70 нс |
| $t_{CE}$   | От активного СЕ до формирования выхода    | -     | 70 нс |
| $t_{OE}$   | От активного ОЕ до формирования выхода    | _     | 30 нс |
| $t_{DF}$   | От снятия СЕ до Z-состояния выхода        | _     | 16 нс |
| $t_{SR/W}$ | Задержка между операциями чтения и записи | 20 нс | _     |
| $t_{OEH}$  | Время активного сигнала ОЕ                | 10 нс | _     |
| $t_{OH}$   | Время удержания данных                    | 0 нс  | _     |

Таблица 5.1: Временные характеристики операции чтения S29AL032D в режиме 70 нс.

Чтобы выполнить операцию чтения, нам нужно повторить эту временную диаграмму и соблюсти все временные интервалы. Но как это сделать?

Каким образом можно выдержать указанные временные интервалы?

Мы уже знаем, что единственным источником информации

о времени для цифрового устройства может являться только сигнал синхронизации, частота которого заранее известна.

Привяжем времена, упомянутые в Таблице 5.1 к тактовому сигналу частоты 50 МГц, которым тактируется устройство. При этом учтём, что некоторые задержки могут быть равны нулю.

Полученная временная диаграмма показана на Рисунке 5.6



Рис. 5.6: Временная диаграмма операции чтения S29AL032D, привязанная к тактовому сигналу, частотой 50 МГц.

#### 5.5.3 Операция записи

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

Согласно datasheet S29AL032D (разделы 7 и 11) для того, чтобы осуществить запись нужного значения во FLASH-память, необходимо выполнить следующую последовательность из 4-х операций записи:

Записать данные АА по адресу ААА;

Записать данные 55 по адресу 555;

Записать данные А0 по адресу ААА;

Записать нужные данные по нужному адресу.

Временная диаграмма завершающей части одной операции записи приведена в пункте 17.4 datasheet и представлена на Рисунке 5.7, а её временные характеристики приведены в Табл. 5.2.



Рис. 5.7: Временная диаграмма фрагмента операции записи S29AL032D

| Обозн.      | Описание                            | Min     | Max      |
|-------------|-------------------------------------|---------|----------|
| $t_{WC}$    | Продолжительность цикла записи      | 70 нс   | _        |
| $t_{AS}$    | Время установки адреса              | 0 нс    | <u>—</u> |
| $t_{AH}$    | Время удерживания адреса            | 45      | _        |
| $t_{DS}$    | Время установки данных              | 35 нс   | _        |
| $t_{DH}$    | Время удержания данных              | 0 нс    | _        |
| $t_{CS}$    | Время установки сигнала СЕ          | 0 нс    | _        |
| $t_{CH}$    | Время удержания сигнала СЕ          | 0 нс    | _        |
| $t_{WP}$    | Ширина импульса записи              | 35 нс   | _        |
| $t_{WPH}$   | Интервал между записью              | 30 нс   | _        |
| $t_{WHWH1}$ | Время программирования байта        | ≈ 9 мкс |          |
| $t_{BUSY}$  | От активного СЕ до установка выхода | 35 нс   | 70 нс    |
| $t_{RB}$    | От активного ОЕ до установки выхода | –       | 30 нс    |

Таблица 5.2: Временные характеристики операции записи S29AL032D в режиме 70 нс.

Аналогично операции чтения, привяжем форму и времена временной диаграммы записи к тактовому сигналу, частотой 50 МГц. Полученная диаграмма, представлена на Рис. 5.8



Рис. 5.8: Временная диаграмма одного цикла операции записи S29AL032D, привязанная к тактовому сигналу, частотой 50 МГц.

Также согласно datasheet, данные записываются не мгновенно. На то, чтобы провести операцию записи одного слова требуется порядка 11 мкс. Что приблизительно соответствует 550 тактам на частоте 50 МГи.

Также крайне важно, что при записи данных микросхема S29AL032D может менять значение с «1» на «0», но не наоборот!

Чтобы поменять значение с «0» на «1» требуется очистка целого фрагмента памяти, называемого сектором, либо полная очистка всей микросхемы!

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

#### 5.5.4 Операция очистки

Для очистки выбранного сектора необходимо выполнить следующую последовательность операций:

Записать данные «AA» по адресу «AAA»;

Записать данные «55» по адресу «555»;

Записать данные «80» по адресу «ААА»;

Записать данные «AA» по адресу «AAA»;

Записать данные «55» по адресу «555»;

Записать данные «30» по адресу сектора, который необходимо очистить.

Операция очистки сектора занимает существенное время, и пока она не закончится, невозможно произвести запись или чтение из FLASH-памяти.

Операция полной очистки отличается только последним значением: для полной очистки данные 30 записываются по адресу AAA.

B datasheet на S29AL032D приведены следующие значения: Очистка сектора — приблизительно 0.7 сек.

Полная очистка микросхемы — приблизительно 45 сек.

#### 5.5.5 Статус операции

Для того, чтобы контролировать завершение операций записи и очистки, а также отслеживать ошибки, которые могут возникнуть в процессе их выполнения необходимо получить информацию о статусе операции. Способы получения этой информации и её содержание приведены в разделе 12 datasheet. Далее мы отметим наиболее важные для нас моменты.

Так как на отладочном стенде Altera DE1, которым мы пользуемся для проведения лабораторных работ, не разведён сигнал BUSY микросхемы S29AL032D, то единственным способом получения статуса является чтение информации из адреса, по которому производилась запись.

Если операция удачно завершена — то будет получено значение, из указанного адреса. Для операции записи оно должно совпадать с тем, которое мы хотели записать, а для операции очистки должно содержать только единицы (8'hFF).

Если операция ещё не завершена, то биты [7:2] считанного значения будут содержать информацию о статусе операции. До окончания операции записи DQ[7] будет иметь значение противоположное записываемому («0» при очистке) - это основной признак того, что полученные данные отражают статус операции. Информация о битах статусного пакета приведена в Таблице 5.3

Обратите внимание, что при повторном чтении некоторые биты статусного пакета меняют своё значение на противоположное. Это сделано, чтобы убедиться, что операция чтения выполняется корректно и микросхема не «зависла».

| Операция | DQ7  | DQ6                | DQ5 | DQ4     | DQ3 | DQ2                   |
|----------|------|--------------------|-----|---------|-----|-----------------------|
| Запись   | ~DQ7 | меняет<br>значение | 0   | таймаут | _   | не меняет<br>значение |
| Очистка  | 0    | меняет<br>значение | 0   | таймаут | 1   | меняет<br>значение    |

Таблица 5.3: Значение разрядов статуса.

В информации о статусе операции есть важный признак: бит DQ[4] является признаком того, что время операции превысило максимально допустимое. Если этот бит принимает значение «1», то во время операции произошла какая-то ошибка. В подавляющем большинстве случаев это происходит при попытке записи в ячейку памяти, уже содержавшую какое-то значение.

## 5.5.6 Проектирование контроллера Flash (продолжение)

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

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

Начнём проектировать конечный автомат с начального состояния— состояния бездействия. Будем постепенно наращивать его сложность и степень детализации, уточняя некоторые особенности.

Из состояния бездействия возможны три различных перехода: операция чтения, операция записи и операция очистки. После окончания этих операций автомат снова возвращается в состояние бездействия.



Рис. 5.9: Управляющий конечный автомат контроллера \$29AL032D

Прежде всего, нас интересуют сложные операции «запись» и «очистка». Выделим их основные этапы.

Как уже говорилось, чтобы записать данные во FLASHпамять требуется провести с ней четыре обмена. Но на этом нельзя заканчивать операцию, ведь необходимо дождаться окончания записи. Также надо учесть, что во время записи могут возникнуть ошибки.

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

Аналогичные рассуждения справедливы и для операции очистки. Отразим все эти замечания в состояниях конечного автомата (см. Рисунок 5.10).



Рис. 5.10: Доработанный управляющий конечный автомат контроллера S29AL032D

Раньше мы не разделяли эти состояния  $W_1...W_4, ST_w$  и всё вместе называли «запись». Но, постепенно уточняя детали, мы разбили сложную операцию на более простые этапы.

Первое, что бросается в глаза в получившемся графе— это многократное повторение операций записи в состояниях  $W_1...W_4$  и  $E_1...E_6$ . Повторение операции чтения менее заметно: она происходит в сотояниях R и  $ST_W, ST_E$  (так как используется для проверки статуса).

Также можно постараться выделить анализ статуса  $ST_W$  и  $ST_E$  в состояние, общее для веток записи и очистки.

Тогда структура конечного автомата приобретает вид, показанный на Рисунке 5.11



Рис. 5.11: Конечный автомат контроллера S29AL032D с общим состоянием проверки статуса

#### 5.5.7 Реализация операций чтения и записи

Можем ли мы выделить операции чтения и записи и реализовать их отдельно, чтобы затем использовать их как показано на графе переходов?

Чтобы понять это, сначала ответим на вопрос как вообще возможно реализовать эти операции.

Для того, чтобы провести чтение, необходимо развернуть временную диаграмму, привязанную к тактовому сигналу, которую мы получили на Рисунке 5.6. Аналогичная временная диаграмма для записи была представлена на Рисунке 5.8.

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

томат. Например, чтобы реализовать операцию чтения, разобьём временную диаграмму чтения на этапы, и поставим каждому этапу в соответствие уникальное состояние, как представлено на Рисунке 5.12



Рис. 5.12: Соответствие состояний этапам операции чтения S29AL032D

Мы могли бы добавить эти состояния в конечный автомат, который мы уже начали проектировать, но тогда нам пришлось бы каждое из состояний ST, RD,  $W_1...W_4$  и  $E_1...E_6$ , разбить на несколько состояний. Это привело бы к ненужному усложнению структуры конечного автомата и аппаратной избыточности.

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

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

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

Теперь, когда мы оформили все основные идеи и общую структуру контроллера, можно преступить к его реализации на Verilog HDL.

# 5.5.8 Реализация контроллера микросхемы S29AL032D на языке Verilog HDL

#### ИНТЕРФЕЙС

Как всегда, начнём проектировать с интерфейса будущего контроллера — его входов и выходов. Так как контроллер будет обеспечивать доступ к памяти, мы хотели сделать его интерфейс похожим на интерфейс RAM-памяти. Но нам придётся ввести дополнительные сигналы для того, чтобы реализовать операцию очистки и индикацию ошибок.

Нам будет достаточно одного входа для адреса, так как мы не можем одновременно производить чтение и запись во flash-память. Но для данных, по аналогии с RAM-памятью потребуются два отдельных входа — для записи и для чтения.

```
module flash controller
 1
 2
    ( //module interface
 3
      input
                   clk.
      input [7:0] data for flash,
 4
 5
      output [7:0] data from_flash,
 6
      input [21:0] address,
 7
      input
                   we.
 8
      input
                   oe,
 9
      input
                   erase,
10
      output
                   error,
11
      output
                    ready,
12
13
     //flash interface
      inout [7:0] flash data,
14
     output [21:0] flash address,
15
16
                    flash nwe,
      output
17
     output
                    flash nrst,
18
      output
                    flash nce,
19
      output
                    flash noe );
20
   endmodule
```

Листинг 5.1: Описание интерфейса контроллера \$29AL032D

#### ОСНОВНОЙ УПРАВЛЯЮШИЙ КОНЕЧНЫЙ АВТОМАТ

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

Сразу учтём, что эти же данные можно будет использовать для проверки статуса. Поэтому для операции очистки сектора, запишем в регистр 8'hFF.

```
reg [3:0] main state;
 1
2
3 localparam idle = 4'd0;
4 localparam r = 4'd1;
5 localparam w1 = 4'd2;
6 localparam w2 = 4'd3:
7 localparam w3 = 4'd4;
8 localparam w4 = 4'd5;
9 localparam e1 = 4'd6;
                  = 4'd7;
10 localparam e2
11 localparam e3 = 4'd8:
12 localparam e4 = 4'd9:
13 localparam e5 = 4'd10;
14 localparam e6 = 4'd11;
15 localparam rs
                   = 4'd12;
16 localparam st
                   = 4'd13;
17
                   = 4'd14;
   localparam err
18
19 wire r done, w done;
20 wire wire status ok;
21
   wire error;
22
23
   always @(posedge clk or posedge rst) begin
24
     if (rst) main state <= idle;</pre>
25
     else begin
26
       case (main state)
27
         idle: if (we)
                               main state <= w1:
28
               else if (oe)
                               main state <= r;</pre>
29
               else if (erase) main state <= el;</pre>
```

```
30
31
          w1: if (w done) main state <= w2;
          w2: if (w done) main state <= w3;
32
33
          w3: if (w done) main state <= w4;
34
          w4: if (w done) main state <= st:
35
36
          e1: if (w done) main state <= e2;
37
          e2: if (w done) main state <= e3;
38
          e3: if (w done) main state <= e4;
39
          e4: if (w done) main state <= e5;
40
          e5: if (w done) main state <= e6;
41
          e6: if (w done) main state <= st;
42
43
              if (r done) main state <= idle;</pre>
          r:
44
          rs: if (r done) main state <= st;
          st: begin
45
46
                 if (status ok) main state <= idle;</pre>
47
                 else if (error) main state <= err;</pre>
48
                 else
                                  main state <= rs:</pre>
49
              end
50
51
          err: main state <= idle;
52
        endcase
53
      end
54
    end
55
56
    req [7:0] w data;
    always @(posedge clk or posedge rst) begin
57
58
      if (rst) w data <= 8'h00;
59
      else begin
60
        if ( main stat == idle) begin
61
          if ( we ) w data <= data for flash;</pre>
62
          if ( erase ) w data <= 8'hFF;</pre>
63
        end
64
      end
65
    end
```

Листинг 5.2: Описание управляющего конечного автомата контроллера S29AL032D

#### КОНЕЧНЫЙ АВТОМАТ ЧТЕНИЯ

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

```
1 reg [2:0] read state;
 2 localparam rr1 = 3'd0;
 3 localparam rr2
                     = 3'd1;
 4 localparam rr3 = 3'd2;
                    = 3'd3:
 5 localparam rr4
 6
   always @(posedge clk or posedge rst) begin
 7
 8
     if (rst) read state <= r idle;</pre>
 9
     else begin
10
        case (read state) begin
          rr1: if (start rea) state <= rr2;
11
12
          rr2: state <= rr3:
13
          rr3: state <= rr4:
14
          rr4: state <= rr1:
15
       endcase
16
     end
17
   end
18
19 assign r done = (read state == rr4);
20 wire read nce = (read state == rr1);
21 wire read noe = ((read state == rr1) |
22
                      (read state == rr2));
23 wire read nwe = 1'b1;
24 wire start read = (main state == r) |
25
                      (main state == rs);
26
27 reg [7:0] r data;
28
   always @(posedge clk or posedge rst) begin
29
     if (rst)
                      r data <= 8'h00;
30
     else if (r done) r data <= flash data;</pre>
31
   end
```

Листинг 5.3: Описание конечного автомата чтения контроллера S29AL032D

#### ПРОВЕРКА СТАТУСА ОПЕРАЦИИ

Мы заранее позаботились о том, чтобы проверка статуса не требовала от нас больших усилий: мы защелкиваем данные при начале операции записи и очистки. Теперь нам достаточно сравнить полученное значение с эталонным:

```
1 wire timeout = data_from_flash[4];
2 assign status_ok = ( r_data == w_data )
3 assign error = ~status ok && timeout;
```

Листинг 5.4: Проверка статуса операции в контроллере \$29AL032D

#### УПРАВЛЕНИЕ МИКРОСХЕМОЙ S29AL032D

Опишем поведение двунаправленной шины данных и сигналов управления для S29AL032D. В состояниях чтения, шиной данный должна управлять микросхема, в состояниях записи — контроллер:

```
wire writing = ((main_state == w1) ||
(main_state == w2) ||
(main_state == w3) ||
(main_state == w4));

assign flash_data = writing? w_data : 8'hZZ;
assign flash_noe = writing? write_noe : read_noe;
assign flash_nce = (main_state == idle);
assign flash_nwe = writing? write_nwe : read_nwe;
```

Листинг 5.5: Управление микросхемой S29AL032D

#### 5.6 Задание лабораторной работы

- Разработать конечный автомат записи;
- Разработать механизм передачи различных данных в состояниях  $W_1, ..., W_4$  и  $E_1, ..., E_6$ ;
- Выполнить полную интеграцию контроллера;
- Создать проект, для демонстрации возможностей контроллера, удовлетворяющий требованиям, представленным ниже.

#### 5.6.1 Требования к демонстрационному проекту

#### РЕЖИМЫ РАБОТЫ

С помощью SW[1], SW[2] задается режим — запись, чтение или очистка.

#### СБРОС

Сброс устройства происходит по нажатию BNT[0]

#### РЕЖИМ ЗАПИСИ

В режиме записи с помощью кнопок BTN[3], BTN[2] задается адрес записи, затем ввод подтверждается нажатием BTN[1]. Здесь и далее — во время ввода, вводимое значение отображается на семисегментных индикаторах.

После подтверждения ввода адреса записи, с помощью кнопок BTN[3], BTN[2] вводится записываемое значение и подтверждается нажатием BTN[1].

После подтверждения ввода значения происходит запись в микросхему S29AL032D. В случае успешной записи подсвечивается LEDG[0] и демо-проект возвращается к вводу адреса записи, а в случае ошибки подсвечивается LEDR[0].

#### РЕЖИМ ЧТЕНИЯ

В режиме чтения с помощью кнопок BTN[3], BTN[2] задается адрес чтения, затем ввод подтверждается нажатием BTN[1].

После подтверждения ввода адреса чтения, происходит чтение значения из микросхемы S29AL032D. Считанное значение выводится на семисегментные индикаторы. После нажатия BTN[1], устройство переходит в выбранный режим.

#### РЕЖИМ ОЧИСТКИ

В режиме очистки, с помощью кнопок BTN[3], BTN[2] задается номер очищаемого сектора. Послед подтверждения ввода с помощью кнопки BTN[1], производится выбранная операция.

После окончания операции в случае успешной очистки подсвечивается LEDG[0], а в случае ошибки LEDR[0].