### Университет ИТМО

# ФАКУЛЬТЕТ ПРОГРАММНОЙ ИНЖЕНЕРИИ И КОМПЬЮТЕРНОЙ ТЕХНИКИ

Кафедра вычислительной техники

## Системы ввода-вывода и периферийные устройства Лабораторная работа №3

«Разработка высокоуровневой модели микропроцессорной системы» Вариант №6 РmodJSTK

Выполнили:

Милосердов А. О.

Калугин Ф. И.

Группа Р3410

Преподаватель:

Быковский С. В.

Санкт-Петербург 2017 г.

## Содержание

| 1 Описание работы |                                                                                          |   |  |  |
|-------------------|------------------------------------------------------------------------------------------|---|--|--|
| 2                 | Описание функциональности реализуемой системы                                            | 2 |  |  |
| 3                 | Описание функций и регистровая карта разработанного контроллера периферийного устройства | 2 |  |  |
|                   | 3.1 Регистровая карта                                                                    | 2 |  |  |
|                   | 3.2 Временные диаграммы шины АНВ и периферийного интерфейса                              | 3 |  |  |
| 4                 | Листинги                                                                                 | 3 |  |  |
| 5                 | Временные диаграммы                                                                      | 9 |  |  |
| 6                 | Отладочный вывод                                                                         | 9 |  |  |

#### 1. Описание работы

Разработать на языке Verilog синтезируемое описание контроллера интерфейса ввода-вывода (SPI) с интерфейсом АНВ. Контроллер должен подключаться к шине АНВ процессора MIPSfpga. Подключить разработанный контроллер к коммутатору шины в шаблоне проекта. Провести тестирование контроллера периферийного интерфейса с использованием имитатора ведущего устройства шины АНВ. Получить временные диаграммы обмена по шине АНВ и периферийному интерфейсу.

#### 2. Описание функциональности реализуемой системы

Основная функция контроллера — управление обменом с PmodJSTK по интерфейсу SPI. Обмен ведется пятью байтами в связи с форматом данных для данного устройства. Контроллер подключен к коммутатору шины AHB процессорного ядра MIPSfpga. В качестве системной шины используется упрощенный стандарт AMBA 3 AHB-Lite. Роль ведущего системной шины в разрабатываемой системе играет процессорное ядро, роль ведомых - контроллеры ввода-вывода.

# 3. Описание функций и регистровая карта разработанного контроллера периферийного устройства

В качестве периферийного устройства используется джойстик PmodJSTK:

- Джойстик, передающий 10 битные значения координат X и Y, и значения кнопок;
- 6-контактный порт с интерфейсом SPI.

| Name                                  | <u>۸</u> 1 | Slice LUTs<br>(63400) | Slice Registers<br>(126800) |
|---------------------------------------|------------|-----------------------|-----------------------------|
| ahb_lite_matrix (mfp_ahb_lite_matrix) |            | 140                   | 223                         |
| <pre>gpio (mfp_ahb_gpio_slave)</pre>  |            | 14                    | 51                          |
| ✓ ■ spi (ahb_spi)                     |            | 42                    | 42                          |
| <pre>spi (spi_master_driver)</pre>    |            | 37                    | 24                          |
| <pre>uart_tx (ahb_uart_tx)</pre>      |            | 84                    | 127                         |

Рис. 1: Данные о занятых ресурсах ПЛИС (Implemented Design -> Report Utilization)

#### 3.1. Регистровая карта

| Адрес      | Название | Биты  | Назначение                                 |  |
|------------|----------|-------|--------------------------------------------|--|
| 0×40001000 | SA_ST    | [0]   | начало обмена с периферийным устройством   |  |
| 0×40001004 | SA_SS    | [0]   | управление сигналом Slave Select           |  |
| 0×40001008 | SA_RE    | [0]   | сигнал состояния периферийного контроллера |  |
| 0×40001012 | SA_DA    | [7:0] | данные                                     |  |

#### 3.2. Временные диаграммы шины АНВ и периферийного интерфейса



Рис. 2: Временная диаграмма обмена пятью байтами

### 4. Листинги

Листинг 1: ahb spi.v

```
`timescale 100ps/1ps
   module ahb_spi
    (
                             HCLK,
        input
        input
                             HRESETn,
        input
                    [ 31: 0] HADDR,
                    [ 31: 0] HWDATA,
        input
        input
                             HWRITE.
        input
                             HSEL,
        output reg [ 31: 0] HRDATA,
12
        input
                             SPI MISO,
        output
                             SPI_MOSI,
        output
                             SPI_SCLK,
15
        output reg
                             SPI_SS
16
   );
17
18
        localparam START_REG_ADDR = 8'h0;
19
        localparam SS_REG_ADDR
                                   = 8'h4;
        localparam READY_REG_ADDR = 8'h8;
21
        localparam DATA REG ADDR = 8'h12;
22
        reg [ 7:0] data_buf_w;
        wire [ 7:0] data buf r;
                    ss_flag;
        reg
        reg
                     start_flag;
        wire
                     ready_flag;
```

```
spi_master_driver spi(
30
            .clk_i(HCLK),
            .rst_i(~HRESETn),
32
            .start_i(start_flag),
            .data_in_bi(data_buf_w),
            .busy o(ready flag),
            .data_out_bo(data_buf_r),
            .spi_miso_i(SPI_MISO),
            .spi_mosi_o(SPI_MOSI),
            .spi_sclk_o(SPI_SCLK),
            .spi_cs_i(ss_flag)
        );
        // AHB bus adapter
        reg [ 11:0] HADDR_dly;
                     HWRITE_dly;
        reg
        reg
                     HSEL dly;
        always @(posedge HCLK) begin
50
            HADDR_dly <= HADDR[11:0];</pre>
            HWRITE_dly <= HWRITE;</pre>
            HSEL_dly <= HSEL;</pre>
        end
        wire [ 11:0] reg addr = HADDR dly;
        always @(posedge HCLK or negedge HRESETn) begin
            if (~HRESETn) begin
                data_buf_w <= 8'b0;</pre>
                start_flag <= 1'b0;
                ss flag
                            <= 1'b1;
            end
            else begin
                if (HSEL dly) begin
                    // Bus interface logic, write operation
                    if (HWRITE_dly) begin
                         case (reg addr)
                         DATA_REG_ADDR: data_buf_w <= HWDATA[7:0];</pre>
                         START_REG_ADDR: start_flag <= HWDATA[0];</pre>
                         SS REG ADDR:
                                          ss flag
                                                     <= HWDATA[0];
                         default:; /* READY_REG_ADDR: do nothing */
                         endcase
                    end
                     // Reset start flag automatically after one tick
                     if (!(HWRITE_dly && reg_addr == START_REG_ADDR)) begin
                         start_flag <= 1'b0;
                     end
                end
            end
        end
        // Bus interface logic, data for read operation
83
        always @(*) begin
            SPI_SS = ss_flag;
```

```
case (reg_addr)
                READY_REG_ADDR: HRDATA = ~ready_flag;
                DATA_REG_ADDR: HRDATA = data_buf_r;
                default:
                                HRDATA = 32'b0;
            endcase
        end
   endmodule
                                                   Листинг 2: ahb feeder.v
    `timescale 100ps/1ps
   module ahb_feeder
   (
        input
                             HCLK,
        output reg
                             HRESETn,
        output
                    [ 31: 0] HADDR,
        output
                    [ 2: 0] HBURST,
                              HMASTLOCK,
        output
        output
                    [ 3: 0] HPROT,
                    [ 2: 0] HSIZE,
        output
        output
                    [ 1: 0] HTRANS,
        output
                    [ 31: 0] HWDATA,
13
                             HWRITE,
        output
        input
                    [ 31: 0] HRDATA,
15
        input
                              HREADY,
16
        input
                              HRESP
   );
18
19
        ahb_master ahb_master (
            .HCLK(HCLK),
21
            .HRESETn(HRESETn),
22
            .HADDR(HADDR),
            .HBURST(HBURST),
            .HMASTLOCK(HMASTLOCK),
25
            .HPROT(HPROT),
            .HSIZE(HSIZE),
            .HTRANS(HTRANS),
            .HWDATA(HWDATA),
            .HWRITE(HWRITE),
            .HRDATA(HRDATA),
31
            .HREADY(HREADY),
            .HRESP(HRESP)
        );
34
        reg
           [31:0] data_buf;
        reg
                    spi_ready_flag;
        localparam UART_DATA_ADDR = 32'hbf400000; // register for data to transmit (only one byte is used)
        localparam UART_CTRL_ADDR = 32'hbf400004; // control regster
        localparam UART_DVDR_ADDR = 32'hbf400008; // clock divider register
41
        localparam SPI_START_ADDR = 32'hbf000000;
        localparam SPI_SS_ADDR
                                   = 32 hbf000004;
44
        localparam SPI_READY_ADDR = 32'hbf000008;
```

45

```
localparam SPI_DATA_ADDR = 32'hbf000012;
       initial begin
48
           // wait for end of reset
           repeat (35) @(posedge HCLK);
           // Test PMODJSTK controller
           ahb_master.ahb_write(SPI_SS_ADDR, 1'b0);
           ahb_master.ahb_write(SPI_DATA_ADDR, 32'b10101010);
           ahb_master.ahb_write(SPI_START_ADDR, 1'b1);
           spi_ready_flag = 1'b0;
           while (spi_ready_flag == 1'b0) begin
               ahb master.ahb read(SPI READY ADDR, spi ready flag);
           end
           ahb_master.ahb_read(SPI_DATA_ADDR, data_buf);
           ahb master.ahb write(SPI DATA ADDR, 32'b00110011);
           ahb_master.ahb_write(SPI_START_ADDR, 1'b1);
           spi_ready_flag = 1'b0;
           while (spi_ready_flag == 1'b0) begin
               ahb_master.ahb_read(SPI_READY_ADDR, spi_ready_flag);
           end
           ahb_master.ahb_read(SPI_DATA_ADDR, data_buf);
           ahb master.ahb write(SPI DATA ADDR, 32'b11111111);
           ahb_master.ahb_write(SPI_START_ADDR, 1'b1);
           spi_ready_flag = 1'b0;
           while (spi_ready_flag == 1'b0) begin
               ahb_master.ahb_read(SPI_READY_ADDR, spi_ready_flag);
           end
           ahb_master.ahb_read(SPI_DATA_ADDR, data_buf);
           ahb_master.ahb_write(SPI_START_ADDR, 1'b1);
           spi ready flag = 1'b0;
           while (spi_ready_flag == 1'b0) begin
               ahb_master.ahb_read(SPI_READY_ADDR, spi_ready_flag);
           end
           ahb_master.ahb_read(SPI_DATA_ADDR, data_buf);
           ahb master.ahb write(SPI START ADDR, 1'b1);
           spi_ready_flag = 1'b0;
           while (spi_ready_flag == 1'b0) begin
               ahb_master.ahb_read(SPI_READY_ADDR, spi_ready_flag);
           end
           ahb_master.ahb_read(SPI_DATA_ADDR, data_buf);
           ahb_master.ahb_write(SPI_SS_ADDR, 1'b1);
       end
   endmodule
```

Листинг 3: spi master driver.v

```
`timescale 1ns / 1ps
  // Simple SPI master controller with CPOL=0. CPHA=0
   module spi_master_driver(
      input
      input
                       rst_i,
      // system interface
                                   // signal to start transaction
      input
                       start_i,
11
      input
                  [7:0] data in bi, // data that master will write to slave
12
                                 // transaction is being processed
      output reg
                       busy_o,
      output reg [7:0] data_out_bo, // data recevied from slave in last transaction
      // SPI interface
      input
                       spi_cs_i,
      input
                       spi_miso_i,
      output reg
                       spi mosi o,
      output reg
                       spi_sclk_o
      );
21
      localparam CLK NOPS = 1;
            in_progress;
       reg
            [2:0] counter;
       reg
            [7:0] clk div;
       reg
            clk_div_pulse;
       reg
            [7:0] shiftreg;
       reg
                  bit_buffer;
       reg
      localparam STATE IDLE
                                      = 0; // wait for transaction begin
      localparam STATE WAIT SCLK 1
                                       = 1; // wait for SCLK to become 1
                                       = 2; // wait for SCLK to become 0
      localparam STATE WAIT SCLK 0
      localparam STATE_WAIT_IDLE
                                      = 3; // wait one SCLK and swith to idle
      reg
            [2:0] state;
       always @(posedge clk_i) begin
          if (rst i) begin
              counter
                         <= 0;
              shiftreg
                         <= 0;
              bit_buffer <= 0;</pre>
              in progress <= 0;
              clk div
                         <= 0;
              clk_div_pulse <= 0;</pre>
              state
                         <= STATE_IDLE;
          end
          else begin
              case (state)
                 STATE_IDLE: begin
                     if (start_i) begin
                         in progress = 1;
                         shiftreg <= data in bi;</pre>
                         state <= STATE_WAIT_SCLK_1;</pre>
                         clk_div <= 0;</pre>
```

```
end else begin
                                in progress = 0;
                           end
59
                       end
                       STATE_WAIT_SCLK_1: begin
                            if (clk_div == CLK_NOPS) begin
                                bit buffer <= spi miso i;</pre>
                                state <= STATE_WAIT_SCLK_0;</pre>
                                clk_div <= 0;</pre>
                                clk_div_pulse <= ~clk_div_pulse;</pre>
                            end else begin
                                clk_div <= clk_div + 1;</pre>
                            end
                       end
                       STATE WAIT SCLK 0: begin
                            if (clk_div == CLK_NOPS) begin
                                shiftreg <= { bit_buffer, shiftreg[7:1] };</pre>
                                if (counter == 7) begin
                                     in progress \leftarrow 0;
                                     state <= STATE_WAIT_IDLE;</pre>
                                     counter <= 0;</pre>
                                end else begin
                                     state <= STATE_WAIT_SCLK_1;</pre>
                                     counter <= counter + 1;</pre>
                                end
                                clk_div <= 0;</pre>
                                clk_div_pulse <= ~clk_div_pulse;</pre>
                            end else begin
                                clk_div <= clk_div + 1;</pre>
                            end
                       end
                       STATE_WAIT_IDLE: begin
                            if (clk_div == CLK_NOPS) begin
                                state <= STATE_IDLE;</pre>
                                clk_div <= 0;</pre>
                                clk_div_pulse <= 0;</pre>
                            end else begin
                                clk_div <= clk_div + 1;</pre>
                            end
                       end
                       default: begin
                            state <= STATE IDLE;</pre>
                       end
                  endcase
100
              end
101
         end
102
103
         always @* begin
104
             busy_o
                           = (state != STATE_IDLE);
              data_out_bo = shiftreg;
106
              spi_sclk_o = clk_div_pulse && !spi_cs_i;
107
              if (in_progress)
                  spi_mosi_o = shiftreg[0] && !spi_cs_i;
109
              else
110
                  spi_mosi_o = 0;
         end
112
```

endmodule