-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfinal_memory.v
150 lines (134 loc) · 4.5 KB
/
final_memory.v
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
/* $Author: karu $ */
/* $LastChangedDate: 2009-03-04 23:09:45 -0600 (Wed, 04 Mar 2009) $ */
/* $Rev: 45 $ */
////////////////////////////////////////////////
//
// final_memory -- four cycle version of memory
// for use in the four-banked memory
//
// written for CS/ECE 552, Spring '06
// Andy Phelps, 1 May 2006
//
// modified 30 Oct 2006 by Derek Hower
// - byte-addressable, word aligned
// modified 23 Apr 2008 by karu
// - addr_1c must be 14 bits wide
// This is a word-addressable,
// 16-bit wide, 16K-byte memory.
//
// | | | | | |
// | addr | busy | read data | | new addr |
// | data_in | | | | etc. |
// | wr, rd | | | | |
// | enable | | | | |
// | | | | | |
// <----bank busy; do not try new req--->
//
// Requests may be presented at most every 4th cycle;
// a new request before this time will result in an error.
// Read requests presented in cycle N will deliver data in cycle N+2.
// Concurrent read and write not allowed.
//
// On reset, memory loads from file "loadfile_0.img",
// "loadfile_1.img", "loadfile_2.img", or "loadfile_3.img", depending
// on the "bank_id" input.
//
// File format:
// @0
// <hex data 0>
// <hex data 1>
// ...etc
//
// If input "create_dump" is true on rising clock,
// contents of memory will be dumped to
// file "dumpfile_0", "dumpfile_1", etc, depending on
// "bank_id". It will dump from location 0 up through
// the highest location modified by a write.
//
// File names for loading and dumping is the only purpose of
// the "bank_id" input. (You may change the name of the file
// in the $readmemh statement below.)
//
//////////////////////////////////////
module final_memory (
output [15:0] data_out,
output err,
input [15:0] data_in,
input [12:0] addr,
input wr,
input rd,
input enable,
input create_dump,
input [1:0] bank_id,
input clk,
input rst
);
reg [7:0] mem [0:16384];
reg loaded;
reg [15:0] largest;
wire [13:0] addr_1c;
wire [15:0] data_in_1c;
integer mcd;
integer largeout;
integer i;
assign rd0 = rd & ~wr & enable & ~rst;
assign wr0 = ~rd & wr & enable & ~rst;
// clkrst clkmod(clk, rst, err);
dff ff0 (rd1, rd0, clk, rst);
dff ff1 (wr1, wr0, clk, rst);
dff reg0 [12:0] (addr_1c[12:0], addr, clk, rst);
dff reg1 [15:0] (data_in_1c, data_in, clk, rst);
assign addr_1c[13]=1'b0;
wire [15:0] data_out_1c = rd1 ? {mem[addr_1c<<1], mem[(addr_1c<<1)+1]} : 0;
dff reg2 [15:0] (data_out, data_out_1c, clk, rst);
dff ff2 (rd2, rd1, clk, rst);
dff ff3 (wr2, wr1, clk, rst);
dff ff4 (rd3, rd2, clk, rst);
dff ff5 (wr3, wr2, clk, rst);
assign busy = rd1 | rd2 | rd3 | wr1 | wr2 | wr3;
assign err = ((rd0 | wr0) & busy)
| (rd & wr & enable & ~rst);
initial begin
loaded = 0;
largest = 1;
end
always @(posedge clk) begin
if (rst) begin
if (!loaded) begin
for (i = 0; i <= 16384; i=i+1) begin
mem[i] = 0;
end
case (bank_id)
0: $readmemh("loadfile_0.img", mem);
1: $readmemh("loadfile_1.img", mem);
2: $readmemh("loadfile_2.img", mem);
3: $readmemh("loadfile_3.img", mem);
endcase
loaded = 1;
end
end
else begin
if (wr1) begin
mem[addr_1c<<1] = data_in_1c[15:8]; // The actual write
mem[(addr_1c<<1)+1] = data_in_1c[7:0];
if ({1'b0, (addr_1c<<1)+1} > largest) largest = (addr_1c<<1)+1; // avoid negative numbers
end
if (create_dump) begin
case (bank_id)
0: mcd = $fopen("dumpfile_0", "w");
1: mcd = $fopen("dumpfile_1", "w");
2: mcd = $fopen("dumpfile_2", "w");
3: mcd = $fopen("dumpfile_3", "w");
endcase
for (i=0; i<=largest; i=i+1) begin
$fdisplay(mcd,"%4h %2h", i, mem[i]);
end
largeout = $fopen("largest");
$fdisplay(largeout,"%4h",largest);
$fclose(largeout);
$fclose(mcd);
end
end
end
endmodule // final_memory
// DUMMY LINE FOR REV CONTROL :0: