mirror of
https://github.com/MiSTer-devel/SMS_MiSTer.git
synced 2026-05-17 03:04:32 +00:00
Technically, `Clk’event and clk = ‘1’` can result in non-`0` triggers (e.g. `1`, `U`, `H`, `X`, `Z`, `W`, etc...), whereas `rising_edge(clk)` does not.
161 lines
6.4 KiB
VHDL
161 lines
6.4 KiB
VHDL
--
|
|
-- Operator.vhd
|
|
--
|
|
-- Copyright (c) 2006 Mitsutaka Okazaki (brezza@pokipoki.org)
|
|
-- All rights reserved.
|
|
--
|
|
-- Redistribution and use of this source code or any derivative works, are
|
|
-- permitted provided that the following conditions are met:
|
|
--
|
|
-- 1. Redistributions of source code must retain the above copyright notice,
|
|
-- this list of conditions and the following disclaimer.
|
|
-- 2. Redistributions in binary form must reproduce the above copyright
|
|
-- notice, this list of conditions and the following disclaimer in the
|
|
-- documentation and/or other materials provided with the distribution.
|
|
-- 3. Redistributions may not be sold, nor may they be used in a commercial
|
|
-- product or activity without specific prior written permission.
|
|
--
|
|
-- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
-- "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
-- TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
-- PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
|
|
-- CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
-- EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
-- PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
-- OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
-- WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
-- OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
-- ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
--
|
|
--
|
|
|
|
--
|
|
-- modified by t.hara
|
|
--
|
|
|
|
library ieee;
|
|
use ieee.std_logic_1164.all;
|
|
use ieee.std_logic_unsigned.all;
|
|
use work.vm2413.all;
|
|
|
|
entity Operator is
|
|
port (
|
|
clk : in std_logic;
|
|
reset : in std_logic;
|
|
clkena : in std_logic;
|
|
|
|
slot : in SLOT_TYPE;
|
|
stage : in STAGE_TYPE;
|
|
rhythm : in std_logic;
|
|
|
|
WF : in WF_TYPE;
|
|
FB : in FB_TYPE;
|
|
|
|
noise : in std_logic;
|
|
pgout : in std_logic_vector( 17 downto 0 ); -- 整数部 9bit, 小数部 9bit
|
|
egout : in std_logic_vector( 12 downto 0 );
|
|
|
|
faddr : out CH_TYPE;
|
|
fdata : in SIGNED_LI_TYPE;
|
|
|
|
opout : out std_logic_vector( 13 downto 0 ) -- 整数部 8bit, 小数部 6bit
|
|
);
|
|
end Operator;
|
|
|
|
architecture rtl of Operator is
|
|
|
|
component SineTable
|
|
port (
|
|
clk : in std_logic;
|
|
clkena : in std_logic;
|
|
wf : in std_logic;
|
|
addr : in std_logic_vector( 17 downto 0 ); -- 整数部 9bit, 小数部 9bit
|
|
data : out std_logic_vector( 13 downto 0 ) -- 整数部 8bit, 小数部 6bit
|
|
);
|
|
end component;
|
|
|
|
signal addr : std_logic_vector( 17 downto 0 );
|
|
signal data : std_logic_vector( 13 downto 0 );
|
|
signal w_is_carrier : std_logic;
|
|
signal w_modula_m : std_logic_vector( LI_TYPE'high + 2 + 9 downto 0 );
|
|
signal w_modula_c : std_logic_vector( LI_TYPE'high + 2 + 9 downto 0 );
|
|
signal w_modula : std_logic_vector( LI_TYPE'high + 2 + 9 downto 0 );
|
|
signal ff_egout : std_logic_vector( 12 downto 0 );
|
|
begin
|
|
|
|
-- サイン波(対数表現)--------------------------------------------------
|
|
-- addr 指定した次々サイクルに data が出てくる
|
|
--
|
|
-- stage X 00 X 01 X 10 X 11 X 00
|
|
-- addr X 確定
|
|
-- data X 確定
|
|
-- opout X 確定
|
|
--
|
|
u_sine_table : SineTable
|
|
port map(
|
|
clk => clk,
|
|
clkena => clkena,
|
|
wf => wf,
|
|
addr => addr,
|
|
data => data
|
|
);
|
|
|
|
w_is_carrier <= slot(0);
|
|
w_modula_m <= (others => '0') when( fb = "000" )else
|
|
shr( '0' & fdata.value & '0' & "000000000", "111" xor fb );
|
|
w_modula_c <= fdata.value & "00" & "000000000";
|
|
w_modula <= w_modula_c when( w_is_carrier = '1' )else
|
|
w_modula_m;
|
|
|
|
process( reset, clk )
|
|
variable opout_buf : std_logic_vector( 13 downto 0 ); -- 整数部 8bit, 小数部 6bit
|
|
begin
|
|
if( reset = '1' )then
|
|
opout <= (others => '0');
|
|
ff_egout <= (others => '0');
|
|
elsif rising_edge(clk) then
|
|
if( clkena = '1' )then
|
|
if( stage = "00" )then
|
|
-- サイン波の参照アドレス(位相)を決定するステージ
|
|
if( rhythm = '1' and ( slot = 14 or slot = 17 ))then -- HH or CYM
|
|
addr <= (not noise) & "01111111" & "000000000";
|
|
elsif( rhythm = '1' and slot = 15 )then -- SD
|
|
addr <= (not pgout(pgout'high)) & "01111111" & "000000000";
|
|
elsif( rhythm = '1' and slot = 16 )then -- TOM
|
|
addr <= pgout;
|
|
else
|
|
if( fdata.sign = '0' )then -- modula は fdata の絶対値をシフトした値だから、ここで符号処理してる
|
|
addr <= pgout + w_modula(pgout'range);
|
|
else
|
|
addr <= pgout - w_modula(pgout'range);
|
|
end if;
|
|
end if;
|
|
|
|
elsif( stage = "01" )then
|
|
-- 決定された参照アドレスが u_sine_table へ供給されるステージ
|
|
elsif( stage = "10" )then
|
|
ff_egout <= egout;
|
|
|
|
-- フィードバックメモリのアドレスを決めるステージ
|
|
if( slot(0) = '1' )then
|
|
if( conv_integer(slot)/2 = 8 )then
|
|
faddr <= 0;
|
|
else
|
|
faddr <= conv_integer(slot)/2 + 1; -- 次のモジュレータのアドレスなので +1
|
|
end if;
|
|
end if;
|
|
elsif( stage = "11" )then
|
|
-- SineTable からデータが出てくるステージ
|
|
if ( ( '0' & ff_egout ) + ('0'& data(12 downto 0) ) ) < "10000000000000" then
|
|
opout_buf := data(13) & (ff_egout + data(12 downto 0) );
|
|
else
|
|
opout_buf := data(13) & "1111111111111";
|
|
end if;
|
|
opout <= opout_buf;
|
|
-- 決定されたフィードバックメモリアドレスが FeedBackMemory へ供給されるステージ
|
|
end if;
|
|
end if;
|
|
end if;
|
|
end process;
|
|
end rtl;
|