Warp-SE/cpld/CNT.v

144 lines
3.8 KiB
Coq
Raw Normal View History

2021-10-29 06:04:59 -04:00
module CNT(
/* FSB clock and E clock inputs */
2024-10-03 07:59:29 -04:00
input CLK, input C8M, input E,
2021-10-29 06:04:59 -04:00
/* Refresh request */
2024-10-03 19:13:23 -04:00
output reg RefReq, output reg RefUrg,
2023-03-26 04:33:59 -04:00
/* Reset, button */
output reg nRESout, input nRESin, input nIPL2,
2023-03-20 00:53:10 -04:00
/* Mac PDS bus master control outputs */
2023-04-08 05:46:13 -04:00
output reg AoutOE, output reg nBR_IOB,
2024-10-03 07:59:29 -04:00
/* QoS select inputs */
2024-09-06 06:05:06 -04:00
input BACT,
2024-10-03 05:51:10 -04:00
input QoSCS,
2024-10-03 07:59:29 -04:00
input SndQoSCS,
/* QoS outputs */
output reg QoSEN,
output reg SndQoSReady);
2023-03-26 04:33:59 -04:00
/* E clock synchronization */
reg [1:0] Er; always @(posedge CLK) Er[1:0] <= { Er[0], E };
2023-03-26 04:33:59 -04:00
wire EFall = Er[1] && !Er[0];
2024-10-03 07:59:29 -04:00
/* C8M clock synchronization */
reg [3:0] C8Mr; always @(posedge CLK) C8Mr[3:0] <= { C8Mr[2:0], C8M };
/* Timer counts from 0 to 1010 (10) -- 11 states == 14.042 us
2023-03-21 21:11:58 -04:00
* Refresh timer sequence
2023-04-08 04:08:53 -04:00
* | Timer | RefReq | RefUrg |
* |---------|--------|-----------|
2023-03-21 21:11:58 -04:00
* | 0 0000 | 0 | 0 |
2023-04-08 04:08:53 -04:00
* | 1 0001 | 1 | 0 |
2023-03-26 04:33:59 -04:00
* | 2 0010 | 1 | 0 |
2023-03-21 21:11:58 -04:00
* | 3 0011 | 1 | 0 |
* | 4 0100 | 1 | 0 |
* | 5 0101 | 1 | 0 |
* | 6 0110 | 1 | 0 |
* | 7 0111 | 1 | 0 |
2024-10-03 19:13:23 -04:00
* | 8 1000 | 1 | 0 |
* | 9 1001 | 1 | 1 |
* | 10 1010 | 1 | 1 |
2023-03-20 00:53:10 -04:00
* back to timer==0
2022-09-03 21:32:05 -04:00
*/
2023-03-21 21:11:58 -04:00
reg [3:0] Timer = 0;
2024-10-03 07:59:29 -04:00
wire TimerTC = Timer==10;
reg TimerTick;
2023-03-26 04:33:59 -04:00
always @(posedge CLK) begin
if (EFall) begin
if (TimerTC) Timer <= 0;
else Timer <= Timer+1;
RefReq <= Timer!=10;
2024-10-03 19:13:23 -04:00
RefUrg <= Timer==8 || Timer==9;
2024-10-03 05:51:10 -04:00
end
end
2024-10-03 07:59:29 -04:00
always @(posedge CLK) TimerTick <= EFall && TimerTC;
2024-10-03 18:45:35 -04:00
/* QoS select latches */
reg QoSCSr, SndQoSCSr;
always @(posedge CLK) QoSCSr <= (BACT && QoSCS) || !nRESin;
always @(posedge CLK) SndQoSCSr <= BACT && SndQoSCS;
/* Wait state timer */
reg [3:0] Wait;
always @(posedge CLK) begin
if (!BACT) Wait <= 0;
else Wait <= Wait+1;
end
2024-10-03 05:51:10 -04:00
/* QoS timer
* In the absence of a QoS trigger, QS==0.
* When Qos triggered, QS is set to 1 and counts 1, 2, 3, 0.
* While QS!=0, QoS is enabled.
2024-10-03 18:45:35 -04:00
* QoS enable period is 196.588 us - 210.630 us */
2024-10-03 07:59:29 -04:00
reg [3:0] QS;
2023-04-08 05:46:13 -04:00
always @(posedge CLK) begin
2024-10-03 18:45:35 -04:00
if (SndQoSCSr || QoSCSr) QS <= 15;
2024-10-03 07:59:29 -04:00
else if (QS==0) QS <= 0;
else if (TimerTick) QS <= QS-1;
end
2024-10-03 05:51:10 -04:00
/* QoS enable control */
2024-10-03 07:59:29 -04:00
always @(posedge CLK) if (!BACT) QoSEN <= QS!=0;
/* Sound QoS timer */
reg [1:0] SndQS;
always @(posedge CLK) begin
if (SndQoSCSr) SndQS <= 3;
else if (QoSCSr) SndQS <= 0;
else if (SndQS==0) SndQS <= 0;
else if (TimerTick) SndQS <= SndQS-1;
end
/* Sound QoS ready control */
always @(posedge CLK) begin
if (!BACT) SndQoSReady <= SndQS==0;
2024-10-03 18:45:35 -04:00
else if (QoSCSr) SndQoSReady <= 1;
2024-10-03 07:59:29 -04:00
else if (Wait==15) SndQoSReady <= 1;
end
/* Long timer counts from 0 to 4095.
* 4096 states == 57.516 ms */
reg [11:0] LTimer;
wire LTimerTC = LTimer[11:0]==12'hFFF;
reg LTimerTick;
always @(posedge CLK) if (TimerTick) LTimer <= LTimer+1;
always @(posedge CLK) LTimerTick <= TimerTick && LTimerTC;
/* C8M duty cycle check and power-on reset */
reg nPOR = 0;
always @(posedge CLK) begin
if (C8Mr[3:0]==4'b0000 || C8Mr[3:0]==4'b1111) nPOR <= 0;
else if (C8Mr[1:0]==2'b01) nPOR <= 1;
end
2024-09-29 03:29:49 -04:00
2023-04-08 05:49:29 -04:00
/* Startup sequence state control */
2024-10-03 07:59:29 -04:00
reg [1:0] IS = 0;
always @(posedge CLK) begin
2024-10-03 18:45:13 -04:00
if (!nPOR) IS <= 0;
2024-10-03 07:59:29 -04:00
else case (IS[1:0])
0: if (LTimerTick) IS <= 1;
1: if (LTimerTick) IS <= 2;
2024-10-03 18:45:35 -04:00
2: if (LTimerTick && nIPL2) IS[0] <= 1;
2024-10-03 07:59:29 -04:00
3: IS <= 3;
endcase
end
/* Startup sequence */
2023-03-26 04:33:59 -04:00
always @(posedge CLK) begin
2023-04-08 05:46:13 -04:00
case (IS[1:0])
2024-10-03 07:59:29 -04:00
0, 1: begin
2023-03-20 00:53:10 -04:00
AoutOE <= 0; // Tristate PDS address and control
nRESout <= 0; // Hold reset low
2023-03-21 21:11:58 -04:00
nBR_IOB <= 0; // Default to request bus
2023-04-09 22:47:27 -04:00
end 2: begin
2024-10-03 07:59:29 -04:00
AoutOE <= 0;
2023-03-26 04:33:59 -04:00
nRESout <= 0;
2024-10-03 18:45:35 -04:00
if (!nIPL2) nBR_IOB <= 1; // Disable bus request if NMI pressed
2023-04-09 22:47:27 -04:00
end 3: begin
2024-10-03 07:59:29 -04:00
AoutOE <= !nBR_IOB;
if (LTimerTick) nRESout <= 1; // Release reset after a while
2022-09-03 21:32:05 -04:00
end
2022-09-11 17:15:53 -04:00
endcase
end
2024-10-03 05:51:10 -04:00
2021-10-29 06:04:59 -04:00
endmodule