parent
27b786e92f
commit
7376218e01
|
@ -0,0 +1,438 @@
|
|||
------------------------------------------------------------------------------
|
||||
------------------------------------------------------------------------------
|
||||
-- Copyright 2011-2013(c) Analog Devices, Inc.
|
||||
--
|
||||
-- All rights reserved.
|
||||
--
|
||||
-- Redistribution and use in source and binary forms, with or without modification,
|
||||
-- are permitted provided that the following conditions are met:
|
||||
-- - Redistributions of source code must retain the above copyright
|
||||
-- notice, this list of conditions and the following disclaimer.
|
||||
-- - 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.
|
||||
-- - Neither the name of Analog Devices, Inc. nor the names of its
|
||||
-- contributors may be used to endorse or promote products derived
|
||||
-- from this software without specific prior written permission.
|
||||
-- - The use of this software may or may not infringe the patent rights
|
||||
-- of one or more patent holders. This license does not release you
|
||||
-- from the requirement that you obtain separate licenses from these
|
||||
-- patent holders to use this software.
|
||||
-- - Use of the software either in source or binary form, must be run
|
||||
-- on or directly connected to an Analog Devices Inc. component.
|
||||
--
|
||||
-- THIS SOFTWARE IS PROVIDED BY ANALOG DEVICES "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
||||
-- INCLUDING, BUT NOT LIMITED TO, NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS FOR A
|
||||
-- PARTICULAR PURPOSE ARE DISCLAIMED.
|
||||
--
|
||||
-- IN NO EVENT SHALL ANALOG DEVICES BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||||
-- EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, INTELLECTUAL PROPERTY
|
||||
-- RIGHTS, 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.
|
||||
------------------------------------------------------------------------------
|
||||
------------------------------------------------------------------------------
|
||||
-- istvan.csomortani@analog.com (c) Analog Devices Inc.
|
||||
------------------------------------------------------------------------------
|
||||
------------------------------------------------------------------------------
|
||||
|
||||
library ieee;
|
||||
use ieee.std_logic_1164.all;
|
||||
use ieee.std_logic_arith.all;
|
||||
use ieee.std_logic_unsigned.all;
|
||||
use ieee.numeric_std.all;
|
||||
|
||||
library work;
|
||||
use work.rx_package.all;
|
||||
use work.axi_ctrlif;
|
||||
use work.axi_streaming_dma_rx_fifo;
|
||||
use work.pl330_dma_fifo;
|
||||
|
||||
entity axi_spdif_rx is
|
||||
generic
|
||||
(
|
||||
C_S_AXI_DATA_WIDTH : integer := 32;
|
||||
C_S_AXI_ADDR_WIDTH : integer := 32;
|
||||
C_DMA_TYPE : integer := 0
|
||||
);
|
||||
port
|
||||
(
|
||||
--SPDIF ports
|
||||
rx_int_o : out std_logic;
|
||||
spdif_rx_i : in std_logic;
|
||||
spdif_rx_i_osc : out std_logic;
|
||||
|
||||
--AXI Lite inter face
|
||||
S_AXI_ACLK : in std_logic;
|
||||
S_AXI_ARESETN : in std_logic;
|
||||
S_AXI_AWADDR : in std_logic_vector(C_S_AXI_ADDR_WIDTH-1 downto 0);
|
||||
S_AXI_AWVALID : in std_logic;
|
||||
S_AXI_WDATA : in std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
|
||||
S_AXI_WSTRB : in std_logic_vector((C_S_AXI_DATA_WIDTH/8)-1 downto 0);
|
||||
S_AXI_WVALID : in std_logic;
|
||||
S_AXI_BREADY : in std_logic;
|
||||
S_AXI_ARADDR : in std_logic_vector(C_S_AXI_ADDR_WIDTH-1 downto 0);
|
||||
S_AXI_ARVALID : in std_logic;
|
||||
S_AXI_RREADY : in std_logic;
|
||||
S_AXI_ARREADY : out std_logic;
|
||||
S_AXI_RDATA : out std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
|
||||
S_AXI_RRESP : out std_logic_vector(1 downto 0);
|
||||
S_AXI_RVALID : out std_logic;
|
||||
S_AXI_WREADY : out std_logic;
|
||||
S_AXI_BRESP : out std_logic_vector(1 downto 0);
|
||||
S_AXI_BVALID : out std_logic;
|
||||
S_AXI_AWREADY : out std_logic;
|
||||
|
||||
--AXI STREAM interface
|
||||
M_AXIS_ACLK : in std_logic;
|
||||
M_AXIS_TREADY : in std_logic;
|
||||
M_AXIS_TDATA : out std_logic_vector(31 downto 0);
|
||||
M_AXIS_TLAST : out std_logic;
|
||||
M_AXIS_TVALID : out std_logic;
|
||||
M_AXIS_TKEEP : out std_logic_vector(3 downto 0);
|
||||
|
||||
--PL330 DMA interface
|
||||
DMA_REQ_ACLK : in std_logic;
|
||||
DMA_REQ_RSTN : in std_logic;
|
||||
DMA_REQ_DAVALID : in std_logic;
|
||||
DMA_REQ_DATYPE : in std_logic_vector(1 downto 0);
|
||||
DMA_REQ_DAREADY : out std_logic;
|
||||
DMA_REQ_DRVALID : out std_logic;
|
||||
DMA_REQ_DRTYPE : out std_logic_vector(1 downto 0);
|
||||
DMA_REQ_DRLAST : out std_logic;
|
||||
DMA_REQ_DRREADY : in std_logic
|
||||
);
|
||||
end entity axi_spdif_rx;
|
||||
|
||||
------------------------------------------------------------------------------
|
||||
-- Architecture section
|
||||
------------------------------------------------------------------------------
|
||||
|
||||
architecture IMP of axi_spdif_rx is
|
||||
|
||||
signal wr_data : std_logic_vector(31 downto 0);
|
||||
signal rd_data : std_logic_vector(31 downto 0);
|
||||
signal wr_addr : integer range 0 to 3;
|
||||
signal rd_addr : integer range 0 to 3;
|
||||
signal wr_stb : std_logic;
|
||||
signal rd_ack : std_logic;
|
||||
|
||||
signal version_reg : std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
|
||||
signal control_reg : std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
|
||||
signal chstatus_reg : std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
|
||||
|
||||
signal sampled_data : std_logic_vector(31 downto 0);
|
||||
|
||||
signal sample_ack : std_logic;
|
||||
signal sample_din : std_logic_vector(31 downto 0);
|
||||
signal sample_wr : std_logic;
|
||||
|
||||
signal conf_rxen : std_logic;
|
||||
signal conf_sample : std_logic;
|
||||
signal evt_en : std_logic;
|
||||
signal conf_chas : std_logic;
|
||||
signal conf_valid : std_logic;
|
||||
signal conf_blken : std_logic;
|
||||
signal conf_valen : std_logic;
|
||||
signal conf_useren : std_logic;
|
||||
signal conf_staten : std_logic;
|
||||
signal conf_paren : std_logic;
|
||||
signal config_rd : std_logic;
|
||||
signal config_wr : std_logic;
|
||||
|
||||
signal conf_mode : std_logic_vector(3 downto 0);
|
||||
signal conf_bits : std_logic_vector(C_S_AXI_DATA_WIDTH - 1 downto 0);
|
||||
signal conf_dout : std_logic_vector(C_S_AXI_DATA_WIDTH - 1 downto 0);
|
||||
|
||||
signal fifo_data_out : std_logic_vector(31 downto 0);
|
||||
signal fifo_data_ack : std_logic;
|
||||
signal fifo_reset : std_logic;
|
||||
signal tx_fifo_stb : std_logic;
|
||||
|
||||
signal enable : boolean;
|
||||
|
||||
signal lock : std_logic;
|
||||
|
||||
signal rx_data : std_logic;
|
||||
signal rx_data_en : std_logic;
|
||||
signal rx_block_start : std_logic;
|
||||
signal rx_channel_a : std_logic;
|
||||
signal rx_error : std_logic;
|
||||
signal lock_evt : std_logic;
|
||||
signal ud_a_en : std_logic;
|
||||
signal ud_b_en : std_logic;
|
||||
signal cs_a_en : std_logic;
|
||||
signal cs_b_en : std_logic;
|
||||
signal rx_frame_start : std_logic;
|
||||
|
||||
signal istat_lsbf : std_logic;
|
||||
signal istat_hsbf : std_logic;
|
||||
signal istat_paritya : std_logic;
|
||||
signal istat_parityb : std_logic;
|
||||
|
||||
signal sbuf_wr_adr : std_logic_vector(C_S_AXI_ADDR_WIDTH - 2 downto 0);
|
||||
|
||||
begin
|
||||
|
||||
-------------------------------------------------------------------------------
|
||||
-- Version Register'w
|
||||
-------------------------------------------------------------------------------
|
||||
version_reg(31 downto 20) <= (others => '0');
|
||||
version_reg(19 downto 16) <= "0001";
|
||||
version_reg(15 downto 12) <= (others => '0');
|
||||
version_reg(11 downto 5) <= std_logic_vector(to_unsigned(C_S_AXI_ADDR_WIDTH,7));
|
||||
version_reg(4) <= '1';
|
||||
version_reg(3 downto 0) <= "0001";
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Control Register
|
||||
--------------------------------------------------------------------------------
|
||||
conf_mode(3 downto 0) <= control_reg(23 downto 20);
|
||||
conf_paren <= control_reg(19);
|
||||
conf_staten <= control_reg(18);
|
||||
conf_useren <= control_reg(17);
|
||||
conf_valen <= control_reg(16);
|
||||
conf_blken <= control_reg(5);
|
||||
conf_valid <= control_reg(4);
|
||||
conf_chas <= control_reg(3);
|
||||
evt_en <= control_reg(2);
|
||||
conf_sample <= control_reg(1);
|
||||
conf_rxen <= control_reg(0);
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
fifo_reset <= not conf_sample;
|
||||
enable <= conf_sample = '1';
|
||||
|
||||
streaming_dma_gen: if C_DMA_TYPE = 0 generate
|
||||
fifo: entity axi_streaming_dma_rx_fifo
|
||||
generic map (
|
||||
RAM_ADDR_WIDTH => 3,
|
||||
FIFO_DWIDTH => 32
|
||||
)
|
||||
port map (
|
||||
clk => S_AXI_ACLK,
|
||||
resetn => S_AXI_ARESETN,
|
||||
fifo_reset => fifo_reset,
|
||||
|
||||
enable => enable,
|
||||
period_len => 11,
|
||||
|
||||
M_AXIS_ACLK => M_AXIS_ACLK,
|
||||
M_AXIS_TREADY => M_AXIS_TREADY,
|
||||
M_AXIS_TDATA => M_AXIS_TDATA,
|
||||
M_AXIS_TLAST => M_AXIS_TLAST,
|
||||
M_AXIS_TVALID => M_AXIS_TVALID,
|
||||
M_AXIS_TKEEP => M_AXIS_TKEEP,
|
||||
|
||||
-- Write port
|
||||
in_stb => sample_wr,
|
||||
in_ack => sample_ack,
|
||||
in_data => sample_din
|
||||
);
|
||||
end generate;
|
||||
|
||||
no_streaming_dma_gen: if C_DMA_TYPE /= 0 generate
|
||||
M_AXIS_TVALID <= '0';
|
||||
M_AXIS_TLAST <= '0';
|
||||
M_AXIS_TKEEP <= "0000";
|
||||
end generate;
|
||||
|
||||
pl330_dma_gen: if C_DMA_TYPE = 1 generate
|
||||
tx_fifo_stb <= '1' when wr_addr = 3 and wr_stb = '1' else '0';
|
||||
|
||||
fifo: entity pl330_dma_fifo
|
||||
generic map(
|
||||
RAM_ADDR_WIDTH => 3,
|
||||
FIFO_DWIDTH => 32,
|
||||
FIFO_DIRECTION => 0
|
||||
)
|
||||
port map (
|
||||
clk => S_AXI_ACLK,
|
||||
resetn => S_AXI_ARESETN,
|
||||
fifo_reset => fifo_reset,
|
||||
enable => enable,
|
||||
|
||||
in_data => sample_din,
|
||||
in_stb => tx_fifo_stb,
|
||||
|
||||
out_ack => '1',
|
||||
out_data => sampled_data,
|
||||
|
||||
dclk => DMA_REQ_ACLK,
|
||||
dresetn => DMA_REQ_RSTN,
|
||||
davalid => DMA_REQ_DAVALID,
|
||||
daready => DMA_REQ_DAREADY,
|
||||
datype => DMA_REQ_DATYPE,
|
||||
drvalid => DMA_REQ_DRVALID,
|
||||
drready => DMA_REQ_DRREADY,
|
||||
drtype => DMA_REQ_DRTYPE,
|
||||
drlast => DMA_REQ_DRLAST
|
||||
);
|
||||
end generate;
|
||||
|
||||
no_pl330_dma_gen: if C_DMA_TYPE /= 1 generate
|
||||
DMA_REQ_DAREADY <= '0';
|
||||
DMA_REQ_DRVALID <= '0';
|
||||
DMA_REQ_DRTYPE <= (others => '0');
|
||||
DMA_REQ_DRLAST <= '0';
|
||||
end generate;
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Status Register
|
||||
--------------------------------------------------------------------------------
|
||||
STAT: rx_status_reg
|
||||
generic map
|
||||
(
|
||||
DATA_WIDTH => C_S_AXI_DATA_WIDTH
|
||||
)
|
||||
port map
|
||||
(
|
||||
up_clk => S_AXI_ACLK,
|
||||
status_rd => rd_ack,
|
||||
lock => lock,
|
||||
chas => conf_chas,
|
||||
rx_block_start => rx_block_start,
|
||||
ch_data => rx_data,
|
||||
cs_a_en => cs_a_en,
|
||||
cs_b_en => cs_b_en,
|
||||
status_dout => chstatus_reg
|
||||
);
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Phase decoder
|
||||
--------------------------------------------------------------------------------
|
||||
PDET: rx_phase_det
|
||||
generic map
|
||||
(
|
||||
AXI_FREQ => 100 -- WishBone frequency in MHz
|
||||
)
|
||||
port map
|
||||
(
|
||||
up_clk => S_AXI_ACLK,
|
||||
rxen => conf_rxen,
|
||||
spdif => spdif_rx_i,
|
||||
lock => lock,
|
||||
lock_evt => lock_evt,
|
||||
rx_data => rx_data,
|
||||
rx_data_en => rx_data_en,
|
||||
rx_block_start => rx_block_start,
|
||||
rx_frame_start => rx_frame_start,
|
||||
rx_channel_a => rx_channel_a,
|
||||
rx_error => rx_error,
|
||||
ud_a_en => ud_a_en,
|
||||
ud_b_en => ud_b_en,
|
||||
cs_a_en => cs_a_en,
|
||||
cs_b_en => cs_b_en
|
||||
);
|
||||
spdif_rx_i_osc <= spdif_rx_i;
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Rx Decoder
|
||||
--------------------------------------------------------------------------------
|
||||
FDEC: rx_decode
|
||||
generic map
|
||||
(
|
||||
DATA_WIDTH => C_S_AXI_DATA_WIDTH,
|
||||
ADDR_WIDTH => C_S_AXI_ADDR_WIDTH
|
||||
)
|
||||
port map
|
||||
(
|
||||
up_clk => S_AXI_ACLK,
|
||||
conf_rxen => conf_rxen,
|
||||
conf_sample => conf_sample,
|
||||
conf_valid => conf_valid,
|
||||
conf_mode => conf_mode,
|
||||
conf_blken => conf_blken,
|
||||
conf_valen => conf_valen,
|
||||
conf_useren => conf_useren,
|
||||
conf_staten => conf_staten,
|
||||
conf_paren => conf_paren,
|
||||
lock => lock,
|
||||
rx_data => rx_data,
|
||||
rx_data_en => rx_data_en,
|
||||
rx_block_start => rx_block_start,
|
||||
rx_frame_start => rx_frame_start,
|
||||
rx_channel_a => rx_channel_a,
|
||||
wr_en => sample_wr,
|
||||
wr_addr => sbuf_wr_adr,
|
||||
wr_data => sample_din,
|
||||
stat_paritya => istat_paritya,
|
||||
stat_parityb => istat_parityb,
|
||||
stat_lsbf => istat_lsbf,
|
||||
stat_hsbf => istat_hsbf
|
||||
);
|
||||
rx_int_o <= sample_wr;
|
||||
|
||||
ctrlif: entity axi_ctrlif
|
||||
generic map (
|
||||
C_S_AXI_ADDR_WIDTH => C_S_AXI_ADDR_WIDTH,
|
||||
C_S_AXI_DATA_WIDTH => C_S_AXI_DATA_WIDTH,
|
||||
C_NUM_REG => 4
|
||||
)
|
||||
port map(
|
||||
S_AXI_ACLK => S_AXI_ACLK,
|
||||
S_AXI_ARESETN => S_AXI_ARESETN,
|
||||
S_AXI_AWADDR => S_AXI_AWADDR,
|
||||
S_AXI_AWVALID => S_AXI_AWVALID,
|
||||
S_AXI_WDATA => S_AXI_WDATA,
|
||||
S_AXI_WSTRB => S_AXI_WSTRB,
|
||||
S_AXI_WVALID => S_AXI_WVALID,
|
||||
S_AXI_BREADY => S_AXI_BREADY,
|
||||
S_AXI_ARADDR => S_AXI_ARADDR,
|
||||
S_AXI_ARVALID => S_AXI_ARVALID,
|
||||
S_AXI_RREADY => S_AXI_RREADY,
|
||||
S_AXI_ARREADY => S_AXI_ARREADY,
|
||||
S_AXI_RDATA => S_AXI_RDATA,
|
||||
S_AXI_RRESP => S_AXI_RRESP,
|
||||
S_AXI_RVALID => S_AXI_RVALID,
|
||||
S_AXI_WREADY => S_AXI_WREADY,
|
||||
S_AXI_BRESP => S_AXI_BRESP,
|
||||
S_AXI_BVALID => S_AXI_BVALID,
|
||||
S_AXI_AWREADY => S_AXI_AWREADY,
|
||||
|
||||
rd_addr => rd_addr,
|
||||
rd_data => rd_data,
|
||||
rd_ack => rd_ack,
|
||||
rd_stb => '1',
|
||||
|
||||
wr_addr => wr_addr,
|
||||
wr_data => wr_data,
|
||||
wr_ack => '1',
|
||||
wr_stb => wr_stb
|
||||
);
|
||||
|
||||
process (S_AXI_ACLK)
|
||||
begin
|
||||
if rising_edge(S_AXI_ACLK) then
|
||||
if S_AXI_ARESETN = '0' then
|
||||
version_reg <= (others => '0');
|
||||
control_reg <= (others => '0');
|
||||
else
|
||||
if wr_stb = '1' then
|
||||
case wr_addr is
|
||||
when 0 => version_reg <= wr_data;
|
||||
when 1 => control_reg <= wr_data;
|
||||
when others => null;
|
||||
end case;
|
||||
end if;
|
||||
end if;
|
||||
end if;
|
||||
end process;
|
||||
|
||||
process (rd_addr, version_reg, control_reg, chstatus_reg)
|
||||
begin
|
||||
case rd_addr is
|
||||
when 0 => rd_data <= version_reg;
|
||||
when 1 => rd_data <= control_reg;
|
||||
when 2 => rd_data <= chstatus_reg;
|
||||
when 3 => rd_data <= sampled_data;
|
||||
when others => rd_data <= (others => '0');
|
||||
end case;
|
||||
end process;
|
||||
|
||||
end IMP;
|
|
@ -0,0 +1,52 @@
|
|||
# ip
|
||||
|
||||
source ../scripts/adi_env.tcl
|
||||
source $ad_hdl_dir/library/scripts/adi_ip.tcl
|
||||
|
||||
adi_ip_create axi_spdif_rx
|
||||
adi_ip_files axi_spdif_rx [list \
|
||||
"$ad_hdl_dir/library/common/axi_ctrlif.vhd" \
|
||||
"$ad_hdl_dir/library/common/axi_streaming_dma_rx_fifo.vhd" \
|
||||
"$ad_hdl_dir/library/common/pl330_dma_fifo.vhd" \
|
||||
"$ad_hdl_dir/library/common/dma_fifo.vhd" \
|
||||
"rx_phase_det.vhd" \
|
||||
"rx_package.vhd" \
|
||||
"rx_decode.vhd" \
|
||||
"rx_status_reg.vhd" \
|
||||
"axi_spdif_rx.vhd" \
|
||||
"axi_spdif_rx_constr.xdc"]
|
||||
|
||||
adi_ip_properties_lite axi_spdif_rx
|
||||
adi_ip_constraints axi_spdif_tx axi_spdif_rx_constr.xdc
|
||||
|
||||
adi_add_bus "DMA_ACK" "slave" \
|
||||
"xilinx.com:interface:axis_rtl:1.0" \
|
||||
"xilinx.com:interface:axis:1.0" \
|
||||
[list {"DMA_REQ_DAVALID" "TVALID"} \
|
||||
{"DMA_REQ_DAREADY" "TREADY"} \
|
||||
{"DMA_REQ_DATYPE" "TUSER"} ]
|
||||
adi_add_bus "DMA_REQ" "master" \
|
||||
"xilinx.com:interface:axis_rtl:1.0" \
|
||||
"xilinx.com:interface:axis:1.0" \
|
||||
[list {"DMA_REQ_DRVALID" "TVALID"} \
|
||||
{"DMA_REQ_DRREADY" "TREADY"} \
|
||||
{"DMA_REQ_DRTYPE" "TUSER"} \
|
||||
{"DMA_REQ_DRLAST" "TLAST"} ]
|
||||
|
||||
# Clock and reset are for both DMA_REQ and DMA_ACK
|
||||
adi_add_bus_clock "DMA_REQ_ACLK" "DMA_REQ:DMA_ACK" "DMA_REQ_RSTN"
|
||||
|
||||
adi_set_bus_dependency "M_AXIS" "M_AXIS" \
|
||||
"(spirit:decode(id('MODELPARAM_VALUE.C_DMA_TYPE')) = 0)"
|
||||
|
||||
adi_set_bus_dependency "DMA_ACK" "DMA_REQ_DA" \
|
||||
"(spirit:decode(id('MODELPARAM_VALUE.C_DMA_TYPE')) = 1)"
|
||||
adi_set_bus_dependency "DMA_REQ" "DMA_REQ_DR" \
|
||||
"(spirit:decode(id('MODELPARAM_VALUE.C_DMA_TYPE')) = 1)"
|
||||
adi_set_ports_dependency "DMA_REQ_ACLK" \
|
||||
"(spirit:decode(id('MODELPARAM_VALUE.C_DMA_TYPE')) = 1)"
|
||||
adi_set_ports_dependency "DMA_REQ_RSTN" \
|
||||
"(spirit:decode(id('MODELPARAM_VALUE.C_DMA_TYPE')) = 1)"
|
||||
|
||||
ipx::save_core [ipx::current_core]
|
||||
|
|
@ -0,0 +1,176 @@
|
|||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- WISHBONE SPDIF IP Core ----
|
||||
---- ----
|
||||
---- This file is part of the SPDIF project ----
|
||||
---- http://www.opencores.org/cores/spdif_interface/ ----
|
||||
---- ----
|
||||
---- Description ----
|
||||
---- SPDIF receiver channel status capture module ----
|
||||
---- ----
|
||||
---- ----
|
||||
---- To Do: ----
|
||||
---- - ----
|
||||
---- ----
|
||||
---- Author(s): ----
|
||||
---- - Geir Drange, gedra@opencores.org ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- Copyright (C) 2004 Authors and OPENCORES.ORG ----
|
||||
---- ----
|
||||
---- This source file may be used and distributed without ----
|
||||
---- restriction provided that this copyright statement is not ----
|
||||
---- removed from the file and that any derivative work contains ----
|
||||
---- the original copyright notice and the associated disclaimer. ----
|
||||
---- ----
|
||||
---- This source file is free software; you can redistribute it ----
|
||||
---- and/or modify it under the terms of the GNU Lesser General ----
|
||||
---- Public License as published by the Free Software Foundation; ----
|
||||
---- either version 2.1 of the License, or (at your option) any ----
|
||||
---- later version. ----
|
||||
---- ----
|
||||
---- This source 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 Lesser General Public License for more ----
|
||||
---- details. ----
|
||||
---- ----
|
||||
---- You should have received a copy of the GNU Lesser General ----
|
||||
---- Public License along with this source; if not, download it ----
|
||||
---- from http://www.opencores.org/lgpl.shtml ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
--
|
||||
-- CVS Revision History
|
||||
--
|
||||
-- $Log: not supported by cvs2svn $
|
||||
-- Revision 1.4 2004/07/19 16:58:37 gedra
|
||||
-- Fixed bug.
|
||||
--
|
||||
-- Revision 1.3 2004/06/27 16:16:55 gedra
|
||||
-- Signal renaming and bug fix.
|
||||
--
|
||||
-- Revision 1.2 2004/06/26 14:14:47 gedra
|
||||
-- Converted to numeric_std and fixed a few bugs.
|
||||
--
|
||||
-- Revision 1.1 2004/06/05 17:16:46 gedra
|
||||
-- Channel status/user data capture register
|
||||
--
|
||||
--
|
||||
|
||||
library ieee;
|
||||
use ieee.std_logic_1164.all;
|
||||
use ieee.numeric_std.all;
|
||||
use work.rx_package.all;
|
||||
|
||||
entity rx_cap_reg is
|
||||
port (
|
||||
clk: in std_logic; -- clock
|
||||
rst: in std_logic; -- reset
|
||||
--cap_ctrl_wr: in std_logic; -- control register write
|
||||
--cap_ctrl_rd: in std_logic; -- control register read
|
||||
--cap_data_rd: in std_logic; -- data register read
|
||||
cap_reg: in std_logic_vector(31 downto 0);
|
||||
cap_din: in std_logic_vector(31 downto 0); -- write data
|
||||
rx_block_start: in std_logic; -- start of block signal
|
||||
ch_data: in std_logic; -- channel status/user data
|
||||
ud_a_en: in std_logic; -- user data ch. A enable
|
||||
ud_b_en: in std_logic; -- user data ch. B enable
|
||||
cs_a_en: in std_logic; -- channel status ch. A enable
|
||||
cs_b_en: in std_logic; -- channel status ch. B enable
|
||||
cap_dout: out std_logic_vector(31 downto 0); -- read data
|
||||
cap_evt: out std_logic); -- capture event (interrupt)
|
||||
end rx_cap_reg;
|
||||
|
||||
architecture rtl of rx_cap_reg is
|
||||
|
||||
signal cap_ctrl_bits, cap_ctrl_dout: std_logic_vector(31 downto 0);
|
||||
signal cap_reg_1, cap_new : std_logic_vector(31 downto 0);
|
||||
signal bitlen, cap_len : integer range 0 to 63;
|
||||
signal bitpos, cur_pos : integer range 0 to 255;
|
||||
signal chid, cdata, compared : std_logic;
|
||||
signal d_enable : std_logic_vector(3 downto 0);
|
||||
|
||||
begin
|
||||
|
||||
-- Data bus or'ing
|
||||
cap_dout <= cap_reg_1;-- when cap_data_rd = '1' else cap_ctrl_dout;
|
||||
|
||||
chid <= cap_reg(6);
|
||||
cdata <= cap_reg(7);
|
||||
|
||||
-- capture data register
|
||||
CDAT: process (clk, rst)
|
||||
begin
|
||||
if rst = '1' then
|
||||
cap_reg_1 <= (others => '0'); --
|
||||
cap_new <= (others => '0');
|
||||
cur_pos <= 0;
|
||||
cap_len <= 0;
|
||||
cap_evt <= '0';
|
||||
compared <= '0';
|
||||
bitpos <= 0;
|
||||
bitlen <= 0;
|
||||
else
|
||||
if rising_edge(clk) then
|
||||
bitlen <= to_integer(unsigned(cap_reg(5 downto 0)));
|
||||
bitpos <= to_integer(unsigned(cap_reg(15 downto 8)));
|
||||
if bitlen > 0 then -- bitlen = 0 disables the capture function
|
||||
-- bit counter, 0 to 191
|
||||
if rx_block_start = '1' then
|
||||
cur_pos <= 0;
|
||||
cap_len <= 0;
|
||||
cap_new <= (others => '0');
|
||||
compared <= '0';
|
||||
elsif cs_b_en = '1' then -- ch. status #2 comes last, count then
|
||||
cur_pos <= cur_pos + 1;
|
||||
end if;
|
||||
-- capture bits
|
||||
if cur_pos >= bitpos and cap_len < bitlen then
|
||||
case d_enable is
|
||||
when "0001" => -- user data channel A
|
||||
if cdata = '0' and chid = '0' then
|
||||
cap_new(cap_len) <= ch_data;
|
||||
cap_len <= cap_len + 1;
|
||||
end if;
|
||||
when "0010" => -- user data channel B
|
||||
if cdata = '0' and chid = '1' then
|
||||
cap_new(cap_len) <= ch_data;
|
||||
cap_len <= cap_len + 1;
|
||||
end if;
|
||||
when "0100" => -- channel status ch. A
|
||||
if cdata = '1' and chid = '0' then
|
||||
cap_new(cap_len) <= ch_data;
|
||||
cap_len <= cap_len + 1;
|
||||
end if;
|
||||
when "1000" => -- channel status ch. B
|
||||
if cdata = '1' and chid = '1' then
|
||||
cap_new(cap_len) <= ch_data;
|
||||
cap_len <= cap_len + 1;
|
||||
end if;
|
||||
when others => null;
|
||||
end case;
|
||||
end if;
|
||||
-- if all bits captured, check with previous data
|
||||
if cap_len = bitlen and compared = '0' then
|
||||
compared <= '1';
|
||||
-- event generated if captured bits differ
|
||||
if cap_reg_1 /= cap_new then
|
||||
cap_evt <= '1';
|
||||
end if;
|
||||
cap_reg_1 <= cap_new;
|
||||
else
|
||||
cap_evt <= '0';
|
||||
end if;
|
||||
end if;
|
||||
end if;
|
||||
end if;
|
||||
end process CDAT;
|
||||
|
||||
d_enable(0) <= ud_a_en;
|
||||
d_enable(1) <= ud_b_en;
|
||||
d_enable(2) <= cs_a_en;
|
||||
d_enable(3) <= cs_b_en;
|
||||
|
||||
end rtl;
|
|
@ -0,0 +1,265 @@
|
|||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- WISHBONE SPDIF IP Core ----
|
||||
---- ----
|
||||
---- This file is part of the SPDIF project ----
|
||||
---- http://www.opencores.org/cores/spdif_interface/ ----
|
||||
---- ----
|
||||
---- Description ----
|
||||
---- Sample decoder. Extract sample words and write to sample ----
|
||||
---- buffer. ----
|
||||
---- ----
|
||||
---- ----
|
||||
---- To Do: ----
|
||||
---- - ----
|
||||
---- ----
|
||||
---- Author(s): ----
|
||||
---- - Geir Drange, gedra@opencores.org ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- Copyright (C) 2004 Authors and OPENCORES.ORG ----
|
||||
---- ----
|
||||
---- This source file may be used and distributed without ----
|
||||
---- restriction provided that this copyright statement is not ----
|
||||
---- removed from the file and that any derivative work contains ----
|
||||
---- the original copyright notice and the associated disclaimer. ----
|
||||
---- ----
|
||||
---- This source file is free software; you can redistribute it ----
|
||||
---- and/or modify it under the terms of the GNU Lesser General ----
|
||||
---- Public License as published by the Free Software Foundation; ----
|
||||
---- either version 2.1 of the License, or (at your option) any ----
|
||||
---- later version. ----
|
||||
---- ----
|
||||
---- This source 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 Lesser General Public License for more ----
|
||||
---- details. ----
|
||||
---- ----
|
||||
---- You should have received a copy of the GNU Lesser General ----
|
||||
---- Public License along with this source; if not, download it ----
|
||||
---- from http://www.opencores.org/lgpl.shtml ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
--
|
||||
-- CVS Revision History
|
||||
--
|
||||
-- $Log: not supported by cvs2svn $
|
||||
-- Revision 1.4 2004/07/11 16:19:50 gedra
|
||||
-- Bug-fix.
|
||||
--
|
||||
-- Revision 1.3 2004/06/26 14:14:47 gedra
|
||||
-- Converted to numeric_std and fixed a few bugs.
|
||||
--
|
||||
-- Revision 1.2 2004/06/16 19:04:09 gedra
|
||||
-- Fixed a few bugs.
|
||||
--
|
||||
-- Revision 1.1 2004/06/13 18:07:47 gedra
|
||||
-- Frame decoder and sample extractor
|
||||
--
|
||||
--
|
||||
|
||||
library ieee;
|
||||
use ieee.std_logic_1164.all;
|
||||
use ieee.numeric_std.all;
|
||||
|
||||
entity rx_decode is
|
||||
generic (DATA_WIDTH: integer range 16 to 32;
|
||||
ADDR_WIDTH: integer range 8 to 64);
|
||||
port (
|
||||
up_clk: in std_logic;
|
||||
conf_rxen: in std_logic;
|
||||
conf_sample: in std_logic;
|
||||
conf_valid: in std_logic;
|
||||
conf_mode: in std_logic_vector(3 downto 0);
|
||||
conf_blken: in std_logic;
|
||||
conf_valen: in std_logic;
|
||||
conf_useren: in std_logic;
|
||||
conf_staten: in std_logic;
|
||||
conf_paren: in std_logic;
|
||||
lock: in std_logic;
|
||||
rx_data: in std_logic;
|
||||
rx_data_en: in std_logic;
|
||||
rx_block_start: in std_logic;
|
||||
rx_frame_start: in std_logic;
|
||||
rx_channel_a: in std_logic;
|
||||
wr_en: out std_logic;
|
||||
wr_addr: out std_logic_vector(ADDR_WIDTH - 2 downto 0);
|
||||
wr_data: out std_logic_vector(DATA_WIDTH - 1 downto 0);
|
||||
stat_paritya: out std_logic;
|
||||
stat_parityb: out std_logic;
|
||||
stat_lsbf: out std_logic;
|
||||
stat_hsbf: out std_logic);
|
||||
end rx_decode;
|
||||
|
||||
architecture rtl of rx_decode is
|
||||
|
||||
signal adr_cnt : integer range 0 to 2**(ADDR_WIDTH - 1) - 1;
|
||||
type samp_states is (IDLE, CHA_SYNC, GET_SAMP, PAR_CHK);
|
||||
signal sampst : samp_states;
|
||||
signal bit_cnt, par_cnt : integer range 0 to 31;
|
||||
signal samp_start : integer range 0 to 15;
|
||||
signal tmp_data : std_logic_vector(31 downto 0);
|
||||
signal tmp_stat : std_logic_vector(4 downto 0);
|
||||
signal valid, next_is_a, blk_start : std_logic;
|
||||
|
||||
begin
|
||||
|
||||
-- output data
|
||||
OD32: if DATA_WIDTH = 32 generate
|
||||
--wr_data(31 downto 27) <= tmp_stat;
|
||||
wr_data(31 downto 0) <= tmp_data(31 downto 0);
|
||||
end generate OD32;
|
||||
OD16: if DATA_WIDTH = 16 generate
|
||||
wr_data(15 downto 0) <= tmp_data(15 downto 0);
|
||||
end generate OD16;
|
||||
|
||||
-- State machine extracting audio samples
|
||||
SAEX: process (up_clk, conf_rxen)
|
||||
begin -- process SAEX
|
||||
if conf_rxen = '0' then
|
||||
adr_cnt <= 0;
|
||||
next_is_a <= '1';
|
||||
wr_en <= '0';
|
||||
wr_addr <= (others => '0');
|
||||
tmp_data <= (others => '0');
|
||||
par_cnt <= 0;
|
||||
blk_start <= '0';
|
||||
stat_paritya <= '0';
|
||||
stat_parityb <= '0';
|
||||
stat_lsbf <= '0';
|
||||
stat_hsbf <= '0';
|
||||
valid <= '0';
|
||||
bit_cnt <= 0;
|
||||
sampst <= IDLE;
|
||||
tmp_stat <= (others => '0');
|
||||
elsif rising_edge(up_clk) then
|
||||
--extract and store samples
|
||||
case sampst is
|
||||
when IDLE =>
|
||||
next_is_a <= '1';
|
||||
if lock = '1' and conf_sample = '1' then
|
||||
sampst <= CHA_SYNC;
|
||||
end if;
|
||||
when CHA_SYNC =>
|
||||
wr_addr <= std_logic_vector(to_unsigned(adr_cnt, ADDR_WIDTH - 1));
|
||||
wr_en <= '0';
|
||||
bit_cnt <= 0;
|
||||
valid <= '0';
|
||||
par_cnt <= 0;
|
||||
stat_paritya <= '0';
|
||||
stat_parityb <= '0';
|
||||
stat_lsbf <= '0';
|
||||
stat_hsbf <= '0';
|
||||
--tmp_data(31 downto 0) <= (others => '0');
|
||||
if rx_block_start = '1' and conf_blken = '1' then
|
||||
blk_start <= '1';
|
||||
end if;
|
||||
if rx_frame_start = '1' then --and rx_channel_a = '1' then --next_is_a then
|
||||
next_is_a <= rx_channel_a;
|
||||
if(rx_channel_a = '1') then
|
||||
tmp_data(31 downto 0) <= (others => '0');
|
||||
end if;
|
||||
sampst <= GET_SAMP;
|
||||
end if;
|
||||
when GET_SAMP =>
|
||||
tmp_stat(0) <= blk_start;
|
||||
if rx_data_en = '1' then
|
||||
bit_cnt <= bit_cnt + 1;
|
||||
-- audio part
|
||||
if bit_cnt >= samp_start and bit_cnt <= 23 then
|
||||
if(next_is_a = '1') then
|
||||
tmp_data(bit_cnt - samp_start) <= rx_data;
|
||||
else
|
||||
tmp_data(bit_cnt + 16 - samp_start) <= rx_data;
|
||||
end if;
|
||||
end if;
|
||||
-- status bits
|
||||
case bit_cnt is
|
||||
when 24 => -- validity bit
|
||||
valid <= rx_data;
|
||||
if conf_valen = '1' then
|
||||
tmp_stat(1) <= rx_data;
|
||||
else
|
||||
tmp_stat(1) <= '0';
|
||||
end if;
|
||||
when 25 => -- user data
|
||||
if conf_useren = '1' then
|
||||
tmp_stat(2) <= rx_data;
|
||||
else
|
||||
tmp_stat(2) <= '0';
|
||||
end if;
|
||||
when 26 => -- channel status
|
||||
if conf_staten = '1' then
|
||||
tmp_stat(3) <= rx_data;
|
||||
else
|
||||
tmp_stat(3) <= '0';
|
||||
end if;
|
||||
when 27 => -- parity bit
|
||||
if conf_paren = '1' then
|
||||
tmp_stat(4) <= rx_data;
|
||||
else
|
||||
tmp_stat(4) <= '0';
|
||||
end if;
|
||||
when others =>
|
||||
null;
|
||||
end case;
|
||||
-- parity: count number of 1's
|
||||
if rx_data = '1' then
|
||||
par_cnt <= par_cnt + 1;
|
||||
end if;
|
||||
end if;
|
||||
if bit_cnt = 28 then
|
||||
sampst <= PAR_CHK;
|
||||
end if;
|
||||
when PAR_CHK =>
|
||||
blk_start <= '0';
|
||||
if (((valid = '0' and conf_valid = '1') or conf_valid = '0') and (next_is_a = '0')) then
|
||||
wr_en <= '1';
|
||||
end if;
|
||||
-- parity check
|
||||
if par_cnt mod 2 /= 0 then
|
||||
if rx_channel_a = '1' then
|
||||
stat_paritya <= '1';
|
||||
else
|
||||
stat_parityb <= '1';
|
||||
end if;
|
||||
end if;
|
||||
-- address counter
|
||||
if adr_cnt < 2**(ADDR_WIDTH - 1) - 1 then
|
||||
adr_cnt <= adr_cnt + 1;
|
||||
else
|
||||
adr_cnt <= 0;
|
||||
stat_hsbf <= '1'; -- signal high buffer full
|
||||
end if;
|
||||
if adr_cnt = 2**(ADDR_WIDTH - 2) - 1 then
|
||||
stat_lsbf <= '1'; -- signal low buffer full
|
||||
end if;
|
||||
sampst <= CHA_SYNC;
|
||||
when others =>
|
||||
sampst <= IDLE;
|
||||
end case;
|
||||
end if;
|
||||
end process SAEX;
|
||||
|
||||
-- determine sample resolution from mode bits in 32bit mode
|
||||
M32: if DATA_WIDTH = 32 generate
|
||||
samp_start <= 8 when conf_mode = "0000" else
|
||||
7 when conf_mode = "0001" else
|
||||
6 when conf_mode = "0010" else
|
||||
5 when conf_mode = "0011" else
|
||||
4 when conf_mode = "0100" else
|
||||
3 when conf_mode = "0101" else
|
||||
2 when conf_mode = "0110" else
|
||||
1 when conf_mode = "0111" else
|
||||
0 when conf_mode = "1000" else
|
||||
8;
|
||||
end generate M32;
|
||||
-- in 16bit mode only 16bit of audio is supported
|
||||
M16: if DATA_WIDTH = 16 generate
|
||||
samp_start <= 8;
|
||||
end generate M16;
|
||||
|
||||
|
||||
end rtl;
|
|
@ -0,0 +1,248 @@
|
|||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- WISHBONE SPDIF IP Core ----
|
||||
---- ----
|
||||
---- This file is part of the SPDIF project ----
|
||||
---- http://www.opencores.org/cores/spdif_interface/ ----
|
||||
---- ----
|
||||
---- Description ----
|
||||
---- SPDIF receiver component package. ----
|
||||
---- ----
|
||||
---- ----
|
||||
---- To Do: ----
|
||||
---- - ----
|
||||
---- ----
|
||||
---- Author(s): ----
|
||||
---- - Geir Drange, gedra@opencores.org ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- Copyright (C) 2004 Authors and OPENCORES.ORG ----
|
||||
---- ----
|
||||
---- This source file may be used and distributed without ----
|
||||
---- restriction provided that this copyright statement is not ----
|
||||
---- removed from the file and that any derivative work contains ----
|
||||
---- the original copyright notice and the associated disclaimer. ----
|
||||
---- ----
|
||||
---- This source file is free software; you can redistribute it ----
|
||||
---- and/or modify it under the terms of the GNU Lesser General ----
|
||||
---- Public License as published by the Free Software Foundation; ----
|
||||
---- either version 2.1 of the License, or (at your option) any ----
|
||||
---- later version. ----
|
||||
---- ----
|
||||
---- This source 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 Lesser General Public License for more ----
|
||||
---- details. ----
|
||||
---- ----
|
||||
---- You should have received a copy of the GNU Lesser General ----
|
||||
---- Public License along with this source; if not, download it ----
|
||||
---- from http://www.opencores.org/lgpl.shtml ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
--
|
||||
-- CVS Revision History
|
||||
--
|
||||
-- $Log: not supported by cvs2svn $
|
||||
-- Revision 1.8 2004/06/27 16:16:55 gedra
|
||||
-- Signal renaming and bug fix.
|
||||
--
|
||||
-- Revision 1.7 2004/06/26 14:14:47 gedra
|
||||
-- Converted to numeric_std and fixed a few bugs.
|
||||
--
|
||||
-- Revision 1.6 2004/06/23 18:10:17 gedra
|
||||
-- Added Wishbone bus cycle decoder.
|
||||
--
|
||||
-- Revision 1.5 2004/06/16 19:03:45 gedra
|
||||
-- Changed status reg. declaration
|
||||
--
|
||||
-- Revision 1.4 2004/06/13 18:08:09 gedra
|
||||
-- Added frame decoder and sample extractor
|
||||
--
|
||||
-- Revision 1.3 2004/06/10 18:57:36 gedra
|
||||
-- Cleaned up lint warnings.
|
||||
--
|
||||
-- Revision 1.2 2004/06/09 19:24:50 gedra
|
||||
-- Added dual port ram.
|
||||
--
|
||||
-- Revision 1.1 2004/06/07 18:06:00 gedra
|
||||
-- Receiver component declarations.
|
||||
--
|
||||
--
|
||||
|
||||
library IEEE;
|
||||
use IEEE.std_logic_1164.all;
|
||||
|
||||
package rx_package is
|
||||
|
||||
-- type declarations
|
||||
type bus_array is array (0 to 7) of std_logic_vector(31 downto 0);
|
||||
|
||||
-- components
|
||||
component rx_ver_reg
|
||||
generic (DATA_WIDTH: integer := 32;
|
||||
ADDR_WIDTH: integer := 8;
|
||||
CH_ST_CAPTURE: integer := 1);
|
||||
port (
|
||||
ver_rd: in std_logic; -- version register read
|
||||
ver_dout: out std_logic_vector(DATA_WIDTH - 1 downto 0)); -- read data
|
||||
end component;
|
||||
|
||||
component gen_control_reg
|
||||
generic (DATA_WIDTH: integer;
|
||||
-- note that this vector is (0 to xx), reverse order
|
||||
ACTIVE_BIT_MASK: std_logic_vector);
|
||||
port (
|
||||
clk: in std_logic; -- clock
|
||||
rst: in std_logic; -- reset
|
||||
ctrl_wr: in std_logic; -- control register write
|
||||
ctrl_rd: in std_logic; -- control register read
|
||||
ctrl_din: in std_logic_vector(DATA_WIDTH - 1 downto 0);
|
||||
ctrl_dout: out std_logic_vector(DATA_WIDTH - 1 downto 0);
|
||||
ctrl_bits: out std_logic_vector(DATA_WIDTH - 1 downto 0));
|
||||
end component;
|
||||
|
||||
component rx_status_reg
|
||||
generic (DATA_WIDTH: integer := 32);
|
||||
port (
|
||||
up_clk: in std_logic; -- clock
|
||||
status_rd: in std_logic; -- status register read
|
||||
lock: in std_logic; -- signal lock status
|
||||
chas: in std_logic; -- channel A or B select
|
||||
rx_block_start: in std_logic; -- start of block signal
|
||||
ch_data: in std_logic; -- channel status/user data
|
||||
cs_a_en: in std_logic; -- channel status ch. A enable
|
||||
cs_b_en: in std_logic; -- channel status ch. B enable
|
||||
status_dout: out std_logic_vector(DATA_WIDTH - 1 downto 0));
|
||||
end component;
|
||||
|
||||
component gen_event_reg
|
||||
generic (DATA_WIDTH: integer := 32);
|
||||
port (
|
||||
clk: in std_logic; -- clock
|
||||
rst: in std_logic; -- reset
|
||||
evt_wr: in std_logic; -- event register write
|
||||
evt_rd: in std_logic; -- event register read
|
||||
evt_din: in std_logic_vector(DATA_WIDTH - 1 downto 0); -- write data
|
||||
event: in std_logic_vector(DATA_WIDTH - 1 downto 0); -- event vector
|
||||
evt_mask: in std_logic_vector(DATA_WIDTH - 1 downto 0); -- irq mask
|
||||
evt_en: in std_logic; -- irq enable
|
||||
evt_dout: out std_logic_vector(DATA_WIDTH - 1 downto 0); -- read data
|
||||
evt_irq: out std_logic); -- interrupt request
|
||||
end component;
|
||||
|
||||
component rx_cap_reg
|
||||
port (
|
||||
clk: in std_logic; -- clock
|
||||
rst: in std_logic; -- reset
|
||||
--cap_ctrl_wr: in std_logic; -- control register write
|
||||
--cap_ctrl_rd: in std_logic; -- control register read
|
||||
--cap_data_rd: in std_logic; -- data register read
|
||||
cap_reg: in std_logic_vector(31 downto 0);
|
||||
cap_din: in std_logic_vector(31 downto 0); -- write data
|
||||
rx_block_start: in std_logic; -- start of block signal
|
||||
ch_data: in std_logic; -- channel status/user data
|
||||
ud_a_en: in std_logic; -- user data ch. A enable
|
||||
ud_b_en: in std_logic; -- user data ch. B enable
|
||||
cs_a_en: in std_logic; -- channel status ch. A enable
|
||||
cs_b_en: in std_logic; -- channel status ch. B enable
|
||||
cap_dout: out std_logic_vector(31 downto 0); -- read data
|
||||
cap_evt: out std_logic); -- capture event (interrupt)
|
||||
end component;
|
||||
|
||||
component rx_phase_det
|
||||
generic (AXI_FREQ: natural := 33); -- WishBone frequency in MHz
|
||||
port (
|
||||
up_clk: in std_logic;
|
||||
rxen: in std_logic;
|
||||
spdif: in std_logic;
|
||||
lock: out std_logic;
|
||||
lock_evt: out std_logic; -- lock status change event
|
||||
rx_data: out std_logic;
|
||||
rx_data_en: out std_logic;
|
||||
rx_block_start: out std_logic;
|
||||
rx_frame_start: out std_logic;
|
||||
rx_channel_a: out std_logic;
|
||||
rx_error: out std_logic;
|
||||
ud_a_en: out std_logic; -- user data ch. A enable
|
||||
ud_b_en: out std_logic; -- user data ch. B enable
|
||||
cs_a_en: out std_logic; -- channel status ch. A enable
|
||||
cs_b_en: out std_logic); -- channel status ch. B enable);
|
||||
end component;
|
||||
|
||||
component dpram
|
||||
generic (DATA_WIDTH: positive := 32;
|
||||
RAM_WIDTH: positive := 8);
|
||||
port (
|
||||
clk: in std_logic;
|
||||
rst: in std_logic; -- reset is optional, not used here
|
||||
din: in std_logic_vector(DATA_WIDTH - 1 downto 0);
|
||||
wr_en: in std_logic;
|
||||
rd_en: in std_logic;
|
||||
wr_addr: in std_logic_vector(RAM_WIDTH - 1 downto 0);
|
||||
rd_addr: in std_logic_vector(RAM_WIDTH - 1 downto 0);
|
||||
dout: out std_logic_vector(DATA_WIDTH - 1 downto 0));
|
||||
end component;
|
||||
|
||||
component rx_decode
|
||||
generic (DATA_WIDTH: integer range 16 to 32 := 32;
|
||||
ADDR_WIDTH: integer range 8 to 64 := 8);
|
||||
port (
|
||||
up_clk: in std_logic;
|
||||
conf_rxen: in std_logic;
|
||||
conf_sample: in std_logic;
|
||||
conf_valid: in std_logic;
|
||||
conf_mode: in std_logic_vector(3 downto 0);
|
||||
conf_blken: in std_logic;
|
||||
conf_valen: in std_logic;
|
||||
conf_useren: in std_logic;
|
||||
conf_staten: in std_logic;
|
||||
conf_paren: in std_logic;
|
||||
lock: in std_logic;
|
||||
rx_data: in std_logic;
|
||||
rx_data_en: in std_logic;
|
||||
rx_block_start: in std_logic;
|
||||
rx_frame_start: in std_logic;
|
||||
rx_channel_a: in std_logic;
|
||||
wr_en: out std_logic;
|
||||
wr_addr: out std_logic_vector(ADDR_WIDTH - 2 downto 0);
|
||||
wr_data: out std_logic_vector(DATA_WIDTH - 1 downto 0);
|
||||
stat_paritya: out std_logic;
|
||||
stat_parityb: out std_logic;
|
||||
stat_lsbf: out std_logic;
|
||||
stat_hsbf: out std_logic);
|
||||
end component;
|
||||
|
||||
component rx_wb_decoder
|
||||
generic (DATA_WIDTH: integer := 32;
|
||||
ADDR_WIDTH: integer := 8);
|
||||
port (
|
||||
up_clk: in std_logic; -- wishbone clock
|
||||
wb_rst_i: in std_logic; -- reset signal
|
||||
wb_sel_i: in std_logic; -- select input
|
||||
wb_stb_i: in std_logic; -- strobe input
|
||||
wb_we_i: in std_logic; -- write enable
|
||||
wb_cyc_i: in std_logic; -- cycle input
|
||||
wb_bte_i: in std_logic_vector(1 downto 0); -- burts type extension
|
||||
wb_adr_i: in std_logic_vector(ADDR_WIDTH - 1 downto 0); -- address
|
||||
wb_cti_i: in std_logic_vector(2 downto 0); -- cycle type identifier
|
||||
data_out: in std_logic_vector(DATA_WIDTH - 1 downto 0); -- internal bus
|
||||
wb_ack_o: out std_logic; -- acknowledge
|
||||
wb_dat_o: out std_logic_vector(DATA_WIDTH - 1 downto 0); -- data out
|
||||
version_rd: out std_logic; -- Version register read
|
||||
config_rd: out std_logic; -- Config register read
|
||||
config_wr: out std_logic; -- Config register write
|
||||
status_rd: out std_logic; -- Status register read
|
||||
intmask_rd: out std_logic; -- Interrupt mask register read
|
||||
intmask_wr: out std_logic; -- Interrupt mask register write
|
||||
intstat_rd: out std_logic; -- Interrupt status register read
|
||||
intstat_wr: out std_logic; -- Interrupt status register read
|
||||
mem_rd: out std_logic; -- Sample memory read
|
||||
mem_addr: out std_logic_vector(ADDR_WIDTH - 2 downto 0); -- memory addr.
|
||||
ch_st_cap_rd: out std_logic_vector(7 downto 0); -- Ch. status cap. read
|
||||
ch_st_cap_wr: out std_logic_vector(7 downto 0); -- Ch. status cap. write
|
||||
ch_st_data_rd: out std_logic_vector(7 downto 0)); -- Ch. status data read
|
||||
end component;
|
||||
|
||||
end rx_package;
|
|
@ -0,0 +1,398 @@
|
|||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- WISHBONE SPDIF IP Core ----
|
||||
---- ----
|
||||
---- This file is part of the SPDIF project ----
|
||||
---- http://www.opencores.org/cores/spdif_interface/ ----
|
||||
---- ----
|
||||
---- Description ----
|
||||
---- Oversampling phase detector. Decodes bi-phase mark encoded ----
|
||||
---- signal. Clock must be at least 8 times higher than bit rate. ----
|
||||
---- The SPDIF bitrate must be minimum 100kHz. ----
|
||||
---- ----
|
||||
---- To Do: ----
|
||||
---- - ----
|
||||
---- ----
|
||||
---- Author(s): ----
|
||||
---- - Geir Drange, gedra@opencores.org ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- Copyright (C) 2004 Authors and OPENCORES.ORG ----
|
||||
---- ----
|
||||
---- This source file may be used and distributed without ----
|
||||
---- restriction provided that this copyright statement is not ----
|
||||
---- removed from the file and that any derivative work contains ----
|
||||
---- the original copyright notice and the associated disclaimer. ----
|
||||
---- ----
|
||||
---- This source file is free software; you can redistribute it ----
|
||||
---- and/or modify it under the terms of the GNU Lesser General ----
|
||||
---- Public License as published by the Free Software Foundation; ----
|
||||
---- either version 2.1 of the License, or (at your option) any ----
|
||||
---- later version. ----
|
||||
---- ----
|
||||
---- This source 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 Lesser General Public License for more ----
|
||||
---- details. ----
|
||||
---- ----
|
||||
---- You should have received a copy of the GNU Lesser General ----
|
||||
---- Public License along with this source; if not, download it ----
|
||||
---- from http://www.opencores.org/lgpl.shtml ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
--
|
||||
-- CVS Revision History
|
||||
--
|
||||
-- $Log: not supported by cvs2svn $
|
||||
-- Revision 1.5 2004/07/19 16:58:37 gedra
|
||||
-- Fixed bug.
|
||||
--
|
||||
-- Revision 1.4 2004/07/12 17:06:41 gedra
|
||||
-- Fixed bug with lock event generation.
|
||||
--
|
||||
-- Revision 1.3 2004/07/11 16:19:50 gedra
|
||||
-- Bug-fix.
|
||||
--
|
||||
-- Revision 1.2 2004/06/13 18:08:50 gedra
|
||||
-- Renamed generic and cleaned some lint's
|
||||
--
|
||||
-- Revision 1.1 2004/06/06 15:43:02 gedra
|
||||
-- Early version of the bi-phase mark decoder.
|
||||
--
|
||||
--
|
||||
|
||||
library ieee;
|
||||
use ieee.std_logic_1164.all;
|
||||
|
||||
entity rx_phase_det is
|
||||
generic (AXI_FREQ: natural := 100); -- WishBone frequency in MHz
|
||||
port (
|
||||
up_clk: in std_logic; -- wishbone clock
|
||||
rxen: in std_logic; -- phase detector enable
|
||||
spdif: in std_logic; -- SPDIF input signal
|
||||
lock: out std_logic; -- true if locked to spdif input
|
||||
lock_evt: out std_logic; -- lock status change event
|
||||
rx_data: out std_logic; -- recevied data
|
||||
rx_data_en: out std_logic; -- received data enable
|
||||
rx_block_start: out std_logic; -- start-of-block pulse
|
||||
rx_frame_start: out std_logic; -- start-of-frame pulse
|
||||
rx_channel_a: out std_logic; -- 1 if channel A frame is recevied
|
||||
rx_error: out std_logic; -- signal error was detected
|
||||
ud_a_en: out std_logic; -- user data ch. A enable
|
||||
ud_b_en: out std_logic; -- user data ch. B enable
|
||||
cs_a_en: out std_logic; -- channel status ch. A enable
|
||||
cs_b_en: out std_logic); -- channel status ch. B enable);
|
||||
end rx_phase_det;
|
||||
|
||||
architecture rtl of rx_phase_det is
|
||||
|
||||
constant TRANSITIONS : integer := 70;
|
||||
constant FRAMES_FOR_LOCK : integer := 3;
|
||||
signal maxpulse, maxp, mp_cnt: integer range 0 to 16 * AXI_FREQ;
|
||||
signal last_cnt, max_thres : integer range 0 to 16 * AXI_FREQ;
|
||||
signal minpulse, minp, min_thres: integer range 0 to 8 * AXI_FREQ;
|
||||
signal zspdif, spdif_in, zspdif_in, trans, ztrans : std_logic;
|
||||
signal trans_cnt : integer range 0 to TRANSITIONS;
|
||||
signal valid, p_long, p_short: std_logic;
|
||||
type pulse_type is (ZERO, SHORT, MED, LONG);
|
||||
type pulse_array is array (0 to 3) of pulse_type;
|
||||
signal preamble: pulse_array;
|
||||
signal new_pulse, short_idx, ilock: std_logic;
|
||||
type frame_state is (IDLE, HUNT, FRAMESTART, FRAME_RX);
|
||||
signal framerx : frame_state;
|
||||
signal frame_cnt : integer range 0 to FRAMES_FOR_LOCK;
|
||||
signal bit_cnt : integer range 0 to 63;
|
||||
signal pre_cnt : integer range 0 to 7;
|
||||
type preamble_types is (NONE, PRE_X, PRE_Y, PRE_Z);
|
||||
signal new_preamble, last_preamble : preamble_types;
|
||||
signal irx_channel_a, zilock : std_logic;
|
||||
|
||||
begin
|
||||
|
||||
-- Pulse width analyzer
|
||||
PHDET: process (up_clk, rxen)
|
||||
begin
|
||||
if rxen = '0' then -- reset by configuration register bit
|
||||
maxpulse <= 0;
|
||||
maxp <= 0;
|
||||
mp_cnt <= 0;
|
||||
zspdif <= '0';
|
||||
zspdif_in <= '0';
|
||||
spdif_in <= '0';
|
||||
trans_cnt <= 0;
|
||||
minpulse <= 0;
|
||||
minp <= 8 * AXI_FREQ;
|
||||
last_cnt <= 0;
|
||||
trans <= '0';
|
||||
valid <= '0';
|
||||
preamble <= (ZERO, ZERO, ZERO, ZERO);
|
||||
max_thres <= 0;
|
||||
min_thres <= 0;
|
||||
new_preamble <= NONE;
|
||||
ztrans <= '0';
|
||||
new_pulse <= '0';
|
||||
else
|
||||
if rising_edge(up_clk) then
|
||||
-- sync spdif signal to wishbone clock
|
||||
zspdif <= spdif;
|
||||
spdif_in <= zspdif;
|
||||
-- find the longest pulse, which is the bi-phase violation
|
||||
-- also find the shortest pulse
|
||||
zspdif_in <= spdif_in;
|
||||
if zspdif_in /= spdif_in then -- input transition
|
||||
mp_cnt <= 0;
|
||||
trans <= '1';
|
||||
last_cnt <= mp_cnt;
|
||||
if trans_cnt > 0 then
|
||||
if mp_cnt > maxp then
|
||||
maxp <= mp_cnt;
|
||||
end if;
|
||||
if mp_cnt < minp then
|
||||
minp <= mp_cnt;
|
||||
end if;
|
||||
end if;
|
||||
else
|
||||
trans <= '0';
|
||||
if mp_cnt < 16 * AXI_FREQ then
|
||||
mp_cnt <= mp_cnt + 1;
|
||||
end if;
|
||||
end if;
|
||||
-- transition counting
|
||||
if trans = '1' then
|
||||
if trans_cnt < TRANSITIONS then
|
||||
trans_cnt <= trans_cnt + 1;
|
||||
else
|
||||
-- the max/min pulse length is updated after given # of transitions
|
||||
trans_cnt <= 0;
|
||||
maxpulse <= maxp;
|
||||
maxp <= 0;
|
||||
minpulse <= minp;
|
||||
minp <= 8 * AXI_FREQ;
|
||||
min_thres <= maxp / 2;
|
||||
if maxp < 11 then
|
||||
max_thres <= maxp - 1;
|
||||
else
|
||||
max_thres <= maxp - 3;
|
||||
end if;
|
||||
end if;
|
||||
end if;
|
||||
-- detection of valid SPDIF signal
|
||||
if maxpulse > 6 then
|
||||
valid <= '1';
|
||||
else
|
||||
valid <= '0';
|
||||
end if;
|
||||
-- bit decoding
|
||||
if trans = '1' then
|
||||
if (last_cnt < min_thres) and (last_cnt > 0) then
|
||||
p_short <= '1';
|
||||
preamble(0) <= SHORT;
|
||||
else
|
||||
p_short <= '0';
|
||||
end if;
|
||||
if last_cnt >= max_thres then
|
||||
p_long <= '1';
|
||||
preamble(0) <= LONG;
|
||||
else
|
||||
p_long <= '0';
|
||||
end if;
|
||||
if last_cnt = 0 then
|
||||
preamble(0) <= ZERO;
|
||||
end if;
|
||||
if (last_cnt < max_thres) and (last_cnt >= min_thres) then
|
||||
preamble(0) <= MED;
|
||||
end if;
|
||||
preamble(3) <= preamble(2);
|
||||
preamble(2) <= preamble(1);
|
||||
preamble(1) <= preamble(0);
|
||||
end if;
|
||||
-- preamble detection
|
||||
if preamble(3) = LONG and preamble(2) = LONG and preamble(1) = SHORT
|
||||
and preamble(0) = SHORT then
|
||||
new_preamble <= PRE_X;
|
||||
elsif preamble(3) = LONG and preamble(2) = MED and preamble(1) = SHORT
|
||||
and preamble(0) = MED then
|
||||
new_preamble <= PRE_Y;
|
||||
elsif preamble(3) = LONG and preamble(2) = SHORT and preamble(1) = SHORT
|
||||
and preamble(0) = LONG then
|
||||
new_preamble <= PRE_Z;
|
||||
else
|
||||
new_preamble <= NONE;
|
||||
end if;
|
||||
-- delayed transition pulse for the state machine
|
||||
ztrans <= trans;
|
||||
new_pulse <= ztrans;
|
||||
end if;
|
||||
end if;
|
||||
end process;
|
||||
|
||||
lock <= ilock;
|
||||
rx_channel_a <= irx_channel_a;
|
||||
|
||||
-- State machine that hunt for and lock onto sub-frames
|
||||
FRX: process (up_clk, rxen)
|
||||
begin
|
||||
if rxen = '0' then
|
||||
framerx <= IDLE;
|
||||
ilock <= '0';
|
||||
zilock <= '0';
|
||||
rx_data <= '0';
|
||||
rx_data_en <= '0';
|
||||
rx_block_start <= '0';
|
||||
rx_frame_start <= '0';
|
||||
irx_channel_a <= '0';
|
||||
ud_a_en <= '0';
|
||||
ud_b_en <= '0';
|
||||
cs_a_en <= '0';
|
||||
cs_b_en <= '0';
|
||||
rx_error <= '0';
|
||||
lock_evt <= '0';
|
||||
bit_cnt <= 0;
|
||||
pre_cnt <= 0;
|
||||
short_idx <= '0';
|
||||
frame_cnt <= 0;
|
||||
last_preamble <= NONE;
|
||||
elsif rising_edge(up_clk) then
|
||||
zilock <= ilock;
|
||||
if zilock /= ilock then -- generate event for event reg.
|
||||
lock_evt <= '1';
|
||||
else
|
||||
lock_evt <= '0';
|
||||
end if;
|
||||
case framerx is
|
||||
when IDLE => -- wait for recevier to be enabled
|
||||
if valid = '1' then
|
||||
framerx <= HUNT;
|
||||
end if;
|
||||
when HUNT => -- wait for preamble detection
|
||||
frame_cnt <= 0;
|
||||
ilock <= '0';
|
||||
rx_error <= '0';
|
||||
if new_pulse = '1' then
|
||||
if new_preamble /= NONE then
|
||||
framerx <= FRAMESTART;
|
||||
end if;
|
||||
end if;
|
||||
when FRAMESTART => -- reset sub-frame bit counter
|
||||
bit_cnt <= 0;
|
||||
pre_cnt <= 0;
|
||||
if frame_cnt < FRAMES_FOR_LOCK then
|
||||
frame_cnt <= frame_cnt + 1;
|
||||
else
|
||||
ilock <= '1';
|
||||
end if;
|
||||
last_preamble <= new_preamble;
|
||||
short_idx <= '0';
|
||||
rx_frame_start <= '1';
|
||||
rx_block_start <= '0';
|
||||
framerx <= FRAME_RX;
|
||||
when FRAME_RX => -- receive complete sub-frame
|
||||
if new_pulse = '1' then
|
||||
if bit_cnt < 28 then
|
||||
case preamble(0) is
|
||||
when ZERO =>
|
||||
short_idx <= '0';
|
||||
when SHORT =>
|
||||
if short_idx = '0' then
|
||||
short_idx <= '1';
|
||||
else
|
||||
-- two short pulses is a logical '1'
|
||||
bit_cnt <= bit_cnt + 1;
|
||||
short_idx <= '0';
|
||||
rx_data <= '1';
|
||||
rx_data_en <= ilock;
|
||||
-- user data enable for the capture register
|
||||
if bit_cnt = 25 and ilock = '1' then
|
||||
ud_a_en <= irx_channel_a;
|
||||
ud_b_en <= not irx_channel_a;
|
||||
end if;
|
||||
-- channel status enable for the capture register
|
||||
if bit_cnt = 26 and ilock = '1' then
|
||||
cs_a_en <= irx_channel_a;
|
||||
cs_b_en <= not irx_channel_a;
|
||||
end if;
|
||||
end if;
|
||||
when MED =>
|
||||
-- medium pulse is logical '0'
|
||||
bit_cnt <= bit_cnt + 1;
|
||||
rx_data <= '0';
|
||||
rx_data_en <= ilock;
|
||||
short_idx <= '0';
|
||||
-- user data enable for the capture register
|
||||
if bit_cnt = 25 and ilock = '1' then
|
||||
ud_a_en <= irx_channel_a;
|
||||
ud_b_en <= not irx_channel_a;
|
||||
end if;
|
||||
-- channel status enable for the capture register
|
||||
if bit_cnt = 26 and ilock = '1' then
|
||||
cs_a_en <= irx_channel_a;
|
||||
cs_b_en <= not irx_channel_a;
|
||||
end if;
|
||||
when LONG =>
|
||||
short_idx <= '0';
|
||||
when others =>
|
||||
framerx <= HUNT;
|
||||
end case;
|
||||
else
|
||||
-- there should be 4 pulses in preamble
|
||||
if pre_cnt < 7 then
|
||||
pre_cnt <= pre_cnt + 1;
|
||||
else
|
||||
rx_error <= '1';
|
||||
framerx <= HUNT;
|
||||
end if;
|
||||
-- check for correct preamble here
|
||||
if pre_cnt = 3 then
|
||||
case last_preamble is
|
||||
when PRE_X =>
|
||||
if new_preamble = PRE_Y then
|
||||
framerx <= FRAMESTART;
|
||||
irx_channel_a <= '0';
|
||||
else
|
||||
rx_error <= '1';
|
||||
framerx <= HUNT;
|
||||
end if;
|
||||
when PRE_Y =>
|
||||
if new_preamble = PRE_X or new_preamble = PRE_Z then
|
||||
irx_channel_a <= '1';
|
||||
-- start of new block?
|
||||
if new_preamble = PRE_Z then
|
||||
rx_block_start <= '1';
|
||||
end if;
|
||||
framerx <= FRAMESTART;
|
||||
else
|
||||
rx_error <= '1';
|
||||
framerx <= HUNT;
|
||||
end if;
|
||||
when PRE_Z =>
|
||||
if new_preamble = PRE_Y then
|
||||
irx_channel_a <= '0';
|
||||
framerx <= FRAMESTART;
|
||||
else
|
||||
rx_error <= '1';
|
||||
framerx <= HUNT;
|
||||
end if;
|
||||
when others =>
|
||||
rx_error <= '1';
|
||||
framerx <= HUNT;
|
||||
end case;
|
||||
end if;
|
||||
end if;
|
||||
else
|
||||
rx_data_en <= '0';
|
||||
rx_block_start <= '0';
|
||||
rx_frame_start <= '0';
|
||||
ud_a_en <= '0';
|
||||
ud_b_en <= '0';
|
||||
cs_a_en <= '0';
|
||||
cs_b_en <= '0';
|
||||
end if;
|
||||
when others =>
|
||||
framerx <= IDLE;
|
||||
end case;
|
||||
end if;
|
||||
end process FRX;
|
||||
|
||||
end rtl;
|
|
@ -0,0 +1,150 @@
|
|||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- WISHBONE SPDIF IP Core ----
|
||||
---- ----
|
||||
---- This file is part of the SPDIF project ----
|
||||
---- http://www.opencores.org/cores/spdif_interface/ ----
|
||||
---- ----
|
||||
---- Description ----
|
||||
---- SPDIF receiver status register ----
|
||||
---- ----
|
||||
---- ----
|
||||
---- To Do: ----
|
||||
---- - ----
|
||||
---- ----
|
||||
---- Author(s): ----
|
||||
---- - Geir Drange, gedra@opencores.org ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
---- ----
|
||||
---- Copyright (C) 2004 Authors and OPENCORES.ORG ----
|
||||
---- ----
|
||||
---- This source file may be used and distributed without ----
|
||||
---- restriction provided that this copyright statement is not ----
|
||||
---- removed from the file and that any derivative work contains ----
|
||||
---- the original copyright notice and the associated disclaimer. ----
|
||||
---- ----
|
||||
---- This source file is free software; you can redistribute it ----
|
||||
---- and/or modify it under the terms of the GNU Lesser General ----
|
||||
---- Public License as published by the Free Software Foundation; ----
|
||||
---- either version 2.1 of the License, or (at your option) any ----
|
||||
---- later version. ----
|
||||
---- ----
|
||||
---- This source 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 Lesser General Public License for more ----
|
||||
---- details. ----
|
||||
---- ----
|
||||
---- You should have received a copy of the GNU Lesser General ----
|
||||
---- Public License along with this source; if not, download it ----
|
||||
---- from http://www.opencores.org/lgpl.shtml ----
|
||||
---- ----
|
||||
----------------------------------------------------------------------
|
||||
--
|
||||
-- CVS Revision History
|
||||
--
|
||||
-- $Log: not supported by cvs2svn $
|
||||
-- Revision 1.4 2004/06/27 16:16:55 gedra
|
||||
-- Signal renaming and bug fix.
|
||||
--
|
||||
-- Revision 1.3 2004/06/26 14:14:47 gedra
|
||||
-- Converted to numeric_std and fixed a few bugs.
|
||||
--
|
||||
-- Revision 1.2 2004/06/16 19:03:10 gedra
|
||||
-- Added channel status decoding.
|
||||
--
|
||||
-- Revision 1.1 2004/06/05 17:17:12 gedra
|
||||
-- Recevier status register
|
||||
--
|
||||
--
|
||||
|
||||
library ieee;
|
||||
use ieee.std_logic_1164.all;
|
||||
|
||||
entity rx_status_reg is
|
||||
generic (DATA_WIDTH: integer);
|
||||
port (
|
||||
up_clk: in std_logic; -- clock
|
||||
status_rd: in std_logic; -- status register read
|
||||
lock: in std_logic; -- signal lock status
|
||||
chas: in std_logic; -- channel A or B select
|
||||
rx_block_start: in std_logic; -- start of block signal
|
||||
ch_data: in std_logic; -- channel status/user data
|
||||
cs_a_en: in std_logic; -- channel status ch. A enable
|
||||
cs_b_en: in std_logic; -- channel status ch. B enable
|
||||
status_dout: out std_logic_vector(DATA_WIDTH - 1 downto 0));
|
||||
end rx_status_reg;
|
||||
|
||||
architecture rtl of rx_status_reg is
|
||||
|
||||
signal status_vector : std_logic_vector(DATA_WIDTH - 1 downto 0);
|
||||
signal cur_pos : integer range 0 to 255;
|
||||
signal pro_mode : std_logic;
|
||||
|
||||
begin
|
||||
|
||||
status_dout <= status_vector when status_rd = '1' else (others => '0');
|
||||
|
||||
D32: if DATA_WIDTH = 32 generate
|
||||
status_vector(31 downto 16) <= (others => '0');
|
||||
end generate D32;
|
||||
|
||||
status_vector(0) <= lock;
|
||||
status_vector(15 downto 7) <= (others => '0');
|
||||
|
||||
-- extract channel status bits to be used
|
||||
CDAT: process (up_clk, lock)
|
||||
begin
|
||||
if lock = '0' then
|
||||
cur_pos <= 0;
|
||||
pro_mode <= '0';
|
||||
status_vector(6 downto 1) <= (others => '0');
|
||||
else
|
||||
if rising_edge(up_clk) then
|
||||
-- bit counter, 0 to 191
|
||||
if rx_block_start = '1' then
|
||||
cur_pos <= 0;
|
||||
elsif cs_b_en = '1' then -- ch. status #2 comes last, count then
|
||||
cur_pos <= cur_pos + 1;
|
||||
end if;
|
||||
-- extract status bits used in status register
|
||||
if (chas = '0' and cs_b_en = '1') or
|
||||
(chas = '1' and cs_a_en = '1') then
|
||||
case cur_pos is
|
||||
when 0 => -- PRO bit
|
||||
status_vector(1) <= ch_data;
|
||||
pro_mode <= ch_data;
|
||||
when 1 => -- AUDIO bit
|
||||
status_vector(2) <= not ch_data;
|
||||
when 2 => -- emphasis/copy bit
|
||||
if pro_mode = '1' then
|
||||
status_vector(5) <= ch_data;
|
||||
else
|
||||
status_vector(6) <= ch_data;
|
||||
end if;
|
||||
when 3 => -- emphasis
|
||||
if pro_mode = '1' then
|
||||
status_vector(4) <= ch_data;
|
||||
else
|
||||
status_vector(5) <= ch_data;
|
||||
end if;
|
||||
when 4 => -- emphasis
|
||||
if pro_mode = '1' then
|
||||
status_vector(3) <= ch_data;
|
||||
else
|
||||
status_vector(4) <= ch_data;
|
||||
end if;
|
||||
when 5 => -- emphasis
|
||||
if pro_mode = '0' then
|
||||
status_vector(3) <= ch_data;
|
||||
end if;
|
||||
when others =>
|
||||
null;
|
||||
end case;
|
||||
end if;
|
||||
end if;
|
||||
end if;
|
||||
end process CDAT;
|
||||
|
||||
end rtl;
|
Loading…
Reference in New Issue