si_dma.v
5.65 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
/************************************************************************\
* *
* Copyright (C) 1994, Silicon Graphics, Inc. *
* *
* These coded instructions, statements, and computer programs contain *
* unpublished proprietary information of Silicon Graphics, Inc., and *
* are protected by Federal copyright law. They may not be disclosed *
* to third parties or copied or duplicated in any form, in whole or *
* in part, without the prior written consent of Silicon Graphics, Inc. *
* *
\************************************************************************/
// $Id: si_dma.v,v 1.1.1.1 2002/05/17 06:07:48 blythe Exp $
module si_dma(clk, reset_l,
cbus_read_enable, cbus_write_enable, cbus_select, cbus_command, read_grant,
dma_read, dma_address, dma_length, reg_read_data, io_busy,
read_request, reg_write_data, reg_address, io_address, reg_write_enable,
cbus_data);
`include "si.vh"
parameter READ_DELAY = 8'd1;
parameter WRITE_DELAY = -8'd0;
input clk; // system clock
input reset_l; // system reset_l
input cbus_read_enable; // enable cbus tristate drivers
input cbus_write_enable; // enable cbus tristate drivers
input [CBUS_SELECT_SIZE-1:0] cbus_select; // cbus data select
input [CBUS_COMMAND_SIZE-1:0] cbus_command; // cbus data type
input read_grant; // read response request granted
input dma_read; // DMA read/write select
input [DRAM_ADDRESS_SIZE-1:0] dma_address; // DMA slave address
input [DMA_LENGTH_SIZE-1:0] dma_length; // DMA length in words
input [CBUS_DATA_SIZE-1:0] reg_read_data; // register read data to cbus
input io_busy; // operation in progress
output read_request; // request a read response cycle
output [SI_REG_WRITE_SIZE-1:0] reg_write_data; // register write data from cbus
output [SI_REG_ADDRESS_SIZE-1:0] reg_address;// register read/write address
output [SI_IO_ADDRESS_SIZE-1:2] io_address; // cbus read request address
output reg_write_enable; // register write enable
inout [CBUS_DATA_SIZE-1:0] cbus_data; // IO bus
// input/output registers
reg read_request;
reg reg_write_enable;
reg [SI_REG_ADDRESS_SIZE-1:0] reg_address;
reg [SI_IO_ADDRESS_SIZE-1:2] io_address;
reg [CBUS_COMMAND_SIZE-1:0] cbus_command_reg;
reg [CBUS_DATA_SIZE-1:0] cbus_data_reg;
// internal registers
reg io_request_0, io_request_1;
// cbus tristate drivers
cbus_driver cbus_driver_0(cbus_data_reg, cbus_write_enable, cbus_data);
assign reg_write_data = cbus_data_reg;
always @(posedge clk) begin : cbus_block
reg [CBUS_DATA_SIZE-1:0] cbus_data_out;
case (cbus_select) // synopsys full_case parallel_case
CBUS_ADDRESS_SELECT :
cbus_data_out = dma_address;
CBUS_LENGTH_SELECT :
if (dma_read) begin
cbus_data_out = {BUS_DEVICE_SI, READ_DELAY, HIGH, dma_length};
end
else begin
cbus_data_out = {BUS_DEVICE_SI, WRITE_DELAY, LOW, dma_length};
end
CBUS_DATA_SELECT :
cbus_data_out = reg_read_data;
default :
cbus_data_out = 'bx;
endcase
cbus_command_reg <= cbus_command;
cbus_data_reg <= cbus_read_enable ? cbus_data : cbus_data_out;
end
always @(posedge clk or negedge reset_l) begin
if (!reset_l) begin
// resetable registers
read_request <= LOW;
reg_write_enable <= LOW;
io_request_0 <= LOW;
io_request_1 <= LOW;
reg_address <= 'bx;
io_address <= 'bx;
end
else begin : main_loop
reg next_reg_write_enable;
reg [BUS_ID_SIZE-1:0] next_id;
reg next_si_select;
reg next_pif_select;
reg next_write_command;
reg next_read_command;
reg next_read_request;
reg next_io_request;
next_reg_write_enable = LOW;
next_id = cbus_data_reg >> BUS_ID_OFFSET;
next_si_select = next_id == BUS_ID_SI;
next_pif_select = next_id == BUS_ID_PIF;
next_write_command = cbus_command_reg == CBUS_WRITE_COMMAND;
next_read_command = cbus_command_reg == CBUS_READ_COMMAND;
next_read_request = read_request;
next_io_request = io_request_0;
if (next_si_select) begin
if (next_write_command) begin
next_reg_write_enable = HIGH;
reg_address <= cbus_data_reg >> 2;
end
else if (next_read_command) begin
next_read_request = HIGH;
reg_address <= cbus_data_reg >> 2;
end
end
else if (next_pif_select) begin
if (next_write_command) begin
next_reg_write_enable = HIGH;
io_address <= cbus_data_reg >>2;
reg_address <= SI_IO_WRITE_REG;
end
else if (next_read_command) begin
next_io_request = HIGH;
next_reg_write_enable = HIGH;
io_address <= cbus_data_reg >>2;
reg_address <= SI_IO_READ_REG;
end
end
if (io_request_0 && io_request_1 && !io_busy) begin
next_io_request = LOW;
next_read_request = HIGH;
end
if (read_request && read_grant) begin
next_read_request = LOW;
end
reg_write_enable <= next_reg_write_enable;
read_request <= next_read_request;
io_request_0 <= next_io_request;
io_request_1 <= io_request_0;
end
end
endmodule