1
0
mirror of https://github.com/Gehstock/Mist_FPGA.git synced 2026-04-28 05:05:19 +00:00
Files
Gehstock.Mist_FPGA/common/Sound/JT12/hdl/jt10_acc.v
Gyorgy Szombathelyi e64964ce9f Konami Iron Horse
2022-01-02 01:01:15 +01:00

174 lines
5.3 KiB
Verilog

/* This file is part of JT12.
JT12 program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
JT12 program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with JT12. If not, see <http://www.gnu.org/licenses/>.
Author: Jose Tejada Gomez. Twitter: @topapate
Version: 1.0
Date: 21-03-2019
Each channel can use the full range of the DAC as they do not
get summed in the real chip.
Operator data is summed up without adding extra bits. This is
the case of real YM3438, which was used on Megadrive 2 models.
*/
// YM2610
// ADPCM inputs
// Full OP resolution
// No PCM
// 4 OP channels
// ADPCM-A input is added for the time assigned to FM channel 0_10 (i.e. 3)
module jt10_acc(
input clk,
input clk_en /* synthesis direct_enable */,
input signed [13:0] op_result,
input [ 1:0] rl,
input zero,
input s1_enters,
input s2_enters,
input s3_enters,
input s4_enters,
input [2:0] cur_ch,
input [1:0] cur_op,
input [2:0] alg,
input signed [15:0] adpcmA_l,
input signed [15:0] adpcmA_r,
input signed [15:0] adpcmB_l,
input signed [15:0] adpcmB_r,
// combined output
output signed [15:0] left,
output signed [15:0] right
);
reg sum_en;
always @(*) begin
case ( alg )
default: sum_en = s4_enters;
3'd4: sum_en = s2_enters | s4_enters;
3'd5,3'd6: sum_en = ~s1_enters;
3'd7: sum_en = 1'b1;
endcase
end
wire left_en = rl[1];
wire right_en= rl[0];
wire signed [15:0] opext = { {2{op_result[13]}}, op_result };
reg signed [15:0] acc_input_l, acc_input_r;
reg acc_en_l, acc_en_r;
// YM2610 mode:
// uses channels 0 and 4 for ADPCM data, throwing away FM data for those channels
// reference: YM2610 Application Notes.
always @(*)
case( {cur_op,cur_ch} )
{2'd0,3'd0}: begin // ADPCM-A:
acc_input_l = (adpcmA_l <<< 2) + (adpcmA_l <<< 1);
acc_input_r = (adpcmA_r <<< 2) + (adpcmA_r <<< 1);
`ifndef NOMIX
acc_en_l = 1'b1;
acc_en_r = 1'b1;
`else
acc_en_l = 1'b0;
acc_en_r = 1'b0;
`endif
end
{2'd0,3'd4}: begin // ADPCM-B:
acc_input_l = adpcmB_l >>> 1; // Operator width is 14 bit, ADPCM-B is 16 bit
acc_input_r = adpcmB_r >>> 1; // accumulator width per input channel is 14 bit
`ifndef NOMIX
acc_en_l = 1'b1;
acc_en_r = 1'b1;
`else
acc_en_l = 1'b0;
acc_en_r = 1'b0;
`endif
end
default: begin
// Note by Jose Tejada:
// I don't think we should divide down the FM output
// but someone was looking at the balance of the different
// channels and made this arrangement
// I suppose ADPCM-A would saturate if taken up a factor of 8 instead of 4
// I'll leave it as it is but I think it is worth revisiting this:
acc_input_l = opext >>> 1;
acc_input_r = opext >>> 1;
acc_en_l = sum_en & left_en;
acc_en_r = sum_en & right_en;
end
endcase
// Continuous output
jt12_single_acc #(.win(16),.wout(16)) u_left(
.clk ( clk ),
.clk_en ( clk_en ),
.op_result ( acc_input_l ),
.sum_en ( acc_en_l ),
.zero ( zero ),
.snd ( left )
);
jt12_single_acc #(.win(16),.wout(16)) u_right(
.clk ( clk ),
.clk_en ( clk_en ),
.op_result ( acc_input_r ),
.sum_en ( acc_en_r ),
.zero ( zero ),
.snd ( right )
);
`ifdef SIMULATION
// Dump each channel independently
// It dumps values in decimal, left and right
integer f0,f1,f2,f4,f5,f6;
reg signed [15:0] sum_l[7], sum_r[7];
initial begin
f0=$fopen("fm0.raw","w");
f1=$fopen("fm1.raw","w");
f2=$fopen("fm2.raw","w");
f4=$fopen("fm4.raw","w");
f5=$fopen("fm5.raw","w");
f6=$fopen("fm6.raw","w");
end
always @(posedge clk) begin
if(cur_op==2'b0) begin
sum_l[cur_ch] <= acc_en_l ? acc_input_l : 16'd0;
sum_r[cur_ch] <= acc_en_r ? acc_input_r : 16'd0;
end else begin
sum_l[cur_ch] <= sum_l[cur_ch] + (acc_en_l ? acc_input_l : 16'd0);
sum_r[cur_ch] <= sum_r[cur_ch] + (acc_en_r ? acc_input_r : 16'd0);
end
end
always @(posedge zero) begin
$fwrite(f0,"%d,%d\n", sum_l[0], sum_r[0]);
$fwrite(f1,"%d,%d\n", sum_l[1], sum_r[1]);
$fwrite(f2,"%d,%d\n", sum_l[2], sum_r[2]);
$fwrite(f4,"%d,%d\n", sum_l[4], sum_r[4]);
$fwrite(f5,"%d,%d\n", sum_l[5], sum_r[5]);
$fwrite(f6,"%d,%d\n", sum_l[6], sum_r[6]);
end
`endif
endmodule