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.
178 lines
5.0 KiB
VHDL
178 lines
5.0 KiB
VHDL
--
|
||
-- PhaseGenerator.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 PhaseGenerator 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;
|
||
pm : in PM_TYPE;
|
||
ml : in ML_TYPE;
|
||
blk : in BLK_TYPE;
|
||
fnum : in FNUM_TYPE;
|
||
key : in std_logic;
|
||
|
||
noise : out std_logic;
|
||
pgout : out std_logic_vector( 17 downto 0 )
|
||
);
|
||
end PhaseGenerator;
|
||
|
||
architecture RTL of PhaseGenerator is
|
||
|
||
component PhaseMemory is port (
|
||
clk : in std_logic;
|
||
reset : in std_logic;
|
||
|
||
slot : in SLOT_TYPE;
|
||
memwr : in std_logic;
|
||
|
||
memout : out PHASE_TYPE;
|
||
memin : in PHASE_TYPE
|
||
);
|
||
end component;
|
||
|
||
type ML_TABLE is array (0 to 15) of std_logic_vector(4 downto 0);
|
||
|
||
constant mltbl : ML_TABLE := (
|
||
"00001","00010","00100","00110","01000","01010","01100","01110",
|
||
"10000","10010","10100","10100","11000","11000","11110","11110"
|
||
);
|
||
|
||
constant noise14_tbl : std_logic_vector(63 downto 0) :=
|
||
"1000100010001000100010001000100100010001000100010001000100010000";
|
||
constant noise17_tbl : std_logic_vector(7 downto 0) :=
|
||
"00001010";
|
||
|
||
-- Signals connected to the phase memory.
|
||
signal memwr : std_logic;
|
||
signal memout, memin : PHASE_TYPE;
|
||
|
||
-- Counter for pitch modulation;
|
||
signal pmcount : std_logic_vector(12 downto 0);
|
||
|
||
begin
|
||
|
||
process(clk, reset)
|
||
variable lastkey : std_logic_vector(18-1 downto 0);
|
||
variable dphase : PHASE_TYPE;
|
||
variable noise14 : std_logic;
|
||
variable noise17 : std_logic;
|
||
variable pgout_buf : std_logic_vector( 17 downto 0 ); -- <20>®<EFBFBD>”•” 9bit, <20>¬<EFBFBD>”•” 9bit
|
||
begin
|
||
|
||
if reset = '1' then
|
||
|
||
pmcount <= (others=>'0');
|
||
memwr <= '0';
|
||
lastkey := (others=>'0');
|
||
dphase := (others=>'0');
|
||
noise14 := '0';
|
||
noise17 := '0';
|
||
|
||
elsif rising_edge(clk) then if clkena = '1' then
|
||
|
||
noise <= noise14 xor noise17;
|
||
|
||
if stage = 0 then
|
||
|
||
memwr <= '0';
|
||
|
||
elsif stage = 1 then
|
||
|
||
-- Wait for memory
|
||
|
||
elsif stage = 2 then
|
||
|
||
-- Update pitch LFO counter when slot = 0 and stage = 0 (i.e. increment per 72 clocks)
|
||
if slot = 0 then
|
||
pmcount <= pmcount + '1';
|
||
end if;
|
||
|
||
-- Delta phase
|
||
dphase := (SHL("00000000"&(fnum*mltbl(CONV_INTEGER(ml))),blk)(19 downto 2));
|
||
|
||
if pm ='1' then
|
||
case pmcount(pmcount'high downto pmcount'high-1) is
|
||
when "01" =>
|
||
dphase := dphase + SHR(dphase,"111");
|
||
when "11" =>
|
||
dphase := dphase - SHR(dphase,"111");
|
||
when others => null;
|
||
end case;
|
||
end if;
|
||
|
||
-- Update Phase
|
||
if lastkey(conv_integer(slot)) = '0' and key = '1' and (rhythm = '0' or (slot /= "01110" and slot /= "10001")) then
|
||
memin <= (others=>'0');
|
||
else
|
||
memin <= memout + dphase;
|
||
end if;
|
||
lastkey(conv_integer(slot)) := key;
|
||
|
||
-- Update noise
|
||
if slot = "01110" then
|
||
noise14 := noise14_tbl(CONV_INTEGER(memout(15 downto 10)));
|
||
elsif slot = "10001" then
|
||
noise17 := noise17_tbl(CONV_INTEGER(memout(13 downto 11)));
|
||
end if;
|
||
|
||
pgout_buf := memout;
|
||
pgout <= pgout_buf;
|
||
memwr <= '1';
|
||
|
||
elsif stage = 3 then
|
||
|
||
memwr <= '0';
|
||
|
||
end if;
|
||
|
||
end if; end if;
|
||
|
||
end process;
|
||
|
||
MEM : PhaseMemory port map(clk,reset,slot,memwr,memout,memin);
|
||
|
||
end RTL;
|
||
|