Warp-SE/cpld/RAM.v

193 lines
4.6 KiB
Coq
Raw Normal View History

2021-10-29 06:04:59 -04:00
module RAM(
/* MC68HC000 interface */
2023-04-10 19:28:13 -04:00
input CLK, input [21:1] A, input nWE,
input nAS, input nLDS, input nUDS, input nDTACK,
2021-10-29 06:04:59 -04:00
/* AS cycle detection */
input BACT, input BACTr,
2021-10-29 06:04:59 -04:00
/* Select and ready signals */
2024-09-21 21:52:46 -04:00
input RAMCS, input RAMCS0X, input ROMCS, input ROMCS4X,
/* RAM ready output */
output RAMReady,
2021-10-29 06:04:59 -04:00
/* Refresh Counter Interface */
input RefReqIn, input RefUrgIn,
2024-10-07 07:53:56 -04:00
/* DRAM interface */
2021-10-29 06:04:59 -04:00
output [11:0] RA, output nRAS, output reg nCAS,
2024-10-07 07:53:56 -04:00
output nLWE, output nUWE, output reg nOE,
/* NOR flash interface */
output nROMOE, output nROMWE);
2023-04-10 19:28:13 -04:00
/* RAM control state */
2024-10-03 11:31:14 -04:00
reg [2:0] RS;
reg RASEN;
reg RASEL;
reg RASrf;
reg RefCAS;
reg CASEndEN;
/* Refresh command generation */
reg RefDone; // Refresh done "remember"
always @(posedge CLK) begin
if (!RefReqIn) RefDone <= 0;
else if (RS[2]) RefDone <= 1;
end
wire RefReq = RefReqIn && !RefDone;
wire RefUrg = RefUrgIn && !RefDone;
/* RAM ready control */
reg RAMReadyReg;
assign RAMReady = RAMReadyReg;//!RS[2];
2024-10-08 21:12:13 -04:00
/* RAM /RAS control */
2024-10-08 07:19:18 -04:00
assign nRAS = !((!nAS && RAMCS && RASEN) || RASrf);
2024-10-07 07:53:56 -04:00
2024-10-08 21:12:13 -04:00
/* RAM /WE control */
2024-10-08 07:19:18 -04:00
assign nLWE = !(!nLDS && RASEL && !nWE);
assign nUWE = !(!nUDS && RASEL && !nWE);
/* RAM /OE control */
always @(posedge CLK, posedge nAS) begin
if (nAS) nOE <= 1;
else nOE <= !(RAMCS && nWE);
end
2021-10-29 06:04:59 -04:00
2024-10-08 21:12:13 -04:00
/* ROM /OE and /WE control */
2024-10-03 11:31:14 -04:00
assign nROMOE = !(!nAS && ROMCS && nWE);
2024-10-08 21:13:29 -04:00
assign nROMWE = !(!nAS && ROMCS4X && !nWE);
2021-10-29 06:04:59 -04:00
2022-09-03 21:32:05 -04:00
/* RAM address mux (and ROM address on RA8) */
// RA11 doesn't do anything so both should be identical.
assign RA[11] = !RASEL ? A[19] : A[20]; // ROM address 19
assign RA[03] = !RASEL ? A[19] : A[20];
// RA10 has only row so different rows but same column.
assign RA[10] = !RASEL ? A[17] : A[07];
assign RA[02] = !RASEL ? A[16] : A[07];
// Remainder of RA bus is unpaired
assign RA[09] = !RASEL ? A[15] : A[08];
assign RA[08] = !RASEL ? A[18] : A[21]; // ROM address 18
assign RA[07] = !RASEL ? A[14] : A[06];
assign RA[06] = !RASEL ? A[13] : A[05];
assign RA[05] = !RASEL ? A[12] : A[04];
assign RA[04] = !RASEL ? A[11] : A[03];
assign RA[01] = !RASEL ? A[10] : A[02];
assign RA[00] = !RASEL ? A[09] : A[01];
2023-04-10 20:33:44 -04:00
2023-04-15 07:12:24 -04:00
wire RS0toRef = // Refresh during first clock of non-RAM access
(RefReq && BACT && !BACTr && !RAMCS0X) ||
2023-04-15 07:12:24 -04:00
// Urgent refresh while bus inactive
(RefUrg && !BACT) ||
// Urgent refresh during non-RAM access
2024-10-03 11:31:14 -04:00
(RefUrg && BACT && !RAMCS0X);
wire RS0toRAM = BACT && RAMCS && RASEN;
2021-10-29 06:04:59 -04:00
always @(posedge CLK) begin
2023-04-10 20:33:44 -04:00
case (RS[2:0])
2023-04-10 19:28:13 -04:00
0: begin // Idle/ready
2024-10-03 11:31:14 -04:00
if (RS0toRAM) RS <= 1; // Access RAM
else if (RS0toRef) RS <= 4; // To refresh
else RS <= 0; // Stay in idle/ready
RASEL <= BACT && RAMCS;
RefCAS <= RS0toRef;
RASEN <= !RS0toRef;
RAMReadyReg <= !RS0toRef;
2023-04-10 19:28:13 -04:00
end 1: begin // RAM access
2024-10-03 11:31:14 -04:00
if (!nDTACK || !BACT) RS <= 2; // Cycle ending
else RS <= 1; // Cycle not ending yet
2021-10-29 06:04:59 -04:00
RASEL <= 1;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
RASEN <= nDTACK;
RAMReadyReg <= 1;
2023-04-10 19:28:13 -04:00
end 2: begin // finish RAM access
2024-10-03 11:31:14 -04:00
RS <= 3;
2023-04-15 00:30:11 -04:00
RASEL <= 0;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
2023-04-15 00:30:11 -04:00
RASEN <= 0;
RAMReadyReg <= 1;
2023-04-10 19:28:13 -04:00
end 3: begin //AS cycle complete
if (RefUrg) begin // Refresh RAS
RS <= 4;
2024-10-03 11:31:14 -04:00
RefCAS <= 1;
2023-04-10 19:28:13 -04:00
RASEN <= 0;
RAMReadyReg <= 0;
2024-10-03 11:31:14 -04:00
end else begin // Cycle ended so go back to idle/ready
RS <= 0;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
RASEN <= 1;
RAMReadyReg <= 1;
end
2024-10-03 11:31:14 -04:00
RASEL <= 0;
end 4: begin // Refresh RAS I
RS <= 5;
RASEL <= 0;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
RASEN <= 0;
RAMReadyReg <= 0;
2024-10-03 11:31:14 -04:00
end 5: begin // Refresh RAS II
2023-04-10 19:28:13 -04:00
RS <= 6;
2021-10-29 06:04:59 -04:00
RASEL <= 0;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
2023-04-15 00:30:11 -04:00
RASEN <= 0;
RAMReadyReg <= 0;
2024-10-03 11:31:14 -04:00
end 6: begin // Refresh precharge I
2023-04-15 00:30:11 -04:00
RS <= 7;
RASEL <= 0;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
RASEN <= 0;
RAMReadyReg <= 0;
2023-04-10 20:33:44 -04:00
end 7: begin // Reenable RAM and go to idle/ready
RS <= 0;
2021-10-29 06:04:59 -04:00
RASEL <= 0;
2024-10-03 11:31:14 -04:00
RefCAS <= 0;
RASEN <= 1;
RAMReadyReg <= 1;
2021-10-29 06:04:59 -04:00
end
endcase
2021-10-29 06:04:59 -04:00
end
2024-10-03 11:31:14 -04:00
2023-04-10 19:28:13 -04:00
always @(negedge CLK) begin
case (RS[2:0])
2024-10-03 11:31:14 -04:00
0: begin
RASrf <= 0;
CASEndEN <= 0;
end 1: begin
RASrf <= 1;
CASEndEN <= 1;
end 2: begin
RASrf <= 0;
CASEndEN <= 1;
end 3: begin
RASrf <= 0;
CASEndEN <= 0;
end 4: begin
RASrf <= 1;
CASEndEN <= 0;
end 5: begin
RASrf <= 1;
CASEndEN <= 0;
end 6: begin
RASrf <= 0;
CASEndEN <= 0;
end 7: begin
RASrf <= 0;
CASEndEN <= 0;
end
endcase
end
wire CASEnd = CASEndEN && nAS;
always @(negedge CLK, posedge RefCAS, posedge CASEnd) begin
if (RefCAS) nCAS <= 0;
else if (CASEnd) nCAS <= 1;
else case (RS[2:0])
0: nCAS <= 1;
2023-04-10 19:28:13 -04:00
1: nCAS <= 0;
2024-10-03 11:31:14 -04:00
2: nCAS <= 0;
3: nCAS <= 1;
4: nCAS <= 0;
2023-04-10 19:28:13 -04:00
5: nCAS <= 1;
6: nCAS <= 1;
7: nCAS <= 1;
endcase
end
2021-10-29 06:04:59 -04:00
endmodule