------------------------------------------------------------
-- Copyright: 2010 Integrated Sytems Laboratory, ETH Zurich
--            http://www.iis.ee.ethz.ch/~sha3
------------------------------------------------------------
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
use work.blakePkg.all;
entity controller is
  port (
    CLKxCI      : in  std_logic;
    RSTxRBI     : in  std_logic;
    VALIDINxSI  : in  std_logic;
    VALIDOUTxSO : out std_logic;
    ROUNDxSO    : out unsigned(4 downto 0);
    PROUNDxSO   : out unsigned(4 downto 0);
    NewChainxSO : out std_logic;
    UseIVxSO    : out std_logic);
end controller;
architecture hash of controller is
  type state is (idle, round, fin);
  signal STATExDP, STATExDN : state;
  signal ROUNDxDP, ROUNDxDN : unsigned(4 downto 0);
  
begin  -- hash
  ROUNDxSO  <= ROUNDxDP;
  PROUNDxSO <= ROUNDxDN;
  fsm: process (ROUNDxDP, STATExDP, VALIDINxSI)
  begin  -- process fsm
    
    VALIDOUTxSO <= '0';
    NewChainxSO <= '0';
    UseIVxSO    <= '0';
    ROUNDxDN    <= (others => '0');
    case STATExDP is
      -------------------------------------------------------------------------
      when idle => 
        if VALIDINxSI = '1' then
          UseIVxSO <= '1';
          STATExDN <= round;
        else
          STATExDN <= idle;
          
        end if;
      -------------------------------------------------------------------------
      when round =>
        
        ROUNDxDN <= ROUNDxDP + 1;
        if ROUNDxDP < NROUND-1 then
          STATExDN <= round;
          
        else
          STATExDN <= fin;
          
        end if;
      -------------------------------------------------------------------------
      when fin =>
 
        NewChainxSO <= '1';
        if VALIDINxSI = '1' then
          STATExDN <= round;
        else
          VALIDOUTxSO <= '1';
          STATExDN    <= idle; 
          
        end if; 
      -------------------------------------------------------------------------
      when others =>
        STATExDN <= idle;
      
    end case;
    
  end process fsm;
  
  
  process (CLKxCI, RSTxRBI)
  begin  -- process
    if RSTxRBI = '0' then               -- asynchronous reset (active low)
      STATExDP <= idle;
      ROUNDxDP <= (others => '0');
      
    elsif CLKxCI'event and CLKxCI = '1' then  -- rising clock edge
      STATExDP <= STATExDN;
      ROUNDxDP <= ROUNDxDN;
        
    end if;
  end process;
  
end hash;