-- KCPSM3 reference design -- PicoBlaze performing programming of Intel StrataFlash NOR Flash Memory. -- -- Design provided and tested on the Spartan-3E Starter Kit (Revision C). -- -- Ken Chapman - Xilinx Ltd - 28th March 2006. -- -- The JTAG loader utility is also available for rapid program development. -- -- The design is set up for a 50MHz system clock and UART communications of 115200 baud -- 8-bit, no parity, 1 stop-bit. IMPORTANT note: Soft flow control XON/XOFF is used. -- ------------------------------------------------------------------------------------ -- -- NOTICE: -- -- Copyright Xilinx, Inc. 2006. This code may be contain portions patented by other -- third parties. By providing this core as one possible implementation of a standard, -- Xilinx is making no representation that the provided implementation of this standard -- is free from any claims of infringement by any third party. Xilinx expressly -- disclaims any warranty with respect to the adequacy of the implementation, including -- but not limited to any warranty or representation that the implementation is free -- from claims of any third party. Furthermore, Xilinx is providing this core as a -- courtesy to you and suggests that you contact all third parties to obtain the -- necessary rights to use this implementation. -- ------------------------------------------------------------------------------------ -- -- Library declarations -- -- Standard IEEE libraries -- library IEEE; use IEEE.STD_LOGIC_1164.ALL; use IEEE.STD_LOGIC_ARITH.ALL; use IEEE.STD_LOGIC_UNSIGNED.ALL; -- ------------------------------------------------------------------------------------ -- -- entity parallel_flash_memory_uart_programmer is Port ( tx_female : out std_logic; rx_female : in std_logic; strataflash_oe : out std_logic; strataflash_ce : out std_logic; strataflash_we : out std_logic; strataflash_a : out std_logic_vector(20 downto 0); strataflash_d : inout std_logic_vector(7 downto 0); clk : in std_logic); end parallel_flash_memory_uart_programmer; -- ------------------------------------------------------------------------------------ -- -- Start of test architecture -- architecture Behavioral of parallel_flash_memory_uart_programmer is -- ------------------------------------------------------------------------------------ -- -- declaration of KCPSM3 -- component kcpsm3 Port ( address : out std_logic_vector(9 downto 0); instruction : in std_logic_vector(17 downto 0); port_id : out std_logic_vector(7 downto 0); write_strobe : out std_logic; out_port : out std_logic_vector(7 downto 0); read_strobe : out std_logic; in_port : in std_logic_vector(7 downto 0); interrupt : in std_logic; interrupt_ack : out std_logic; reset : in std_logic; clk : in std_logic); end component; -- -- declaration of program ROM -- component progctrl Port ( address : in std_logic_vector(9 downto 0); instruction : out std_logic_vector(17 downto 0); proc_reset : out std_logic; --JTAG Loader version clk : in std_logic); end component; -- -- declaration of UART transmitter with integral 16 byte FIFO buffer -- Note this is a modified version of the standard 'uart_tx' in which -- the 'data_present' signal has also been brought out to better support -- the XON/XOFF flow control. -- component uart_tx_plus Port ( data_in : in std_logic_vector(7 downto 0); write_buffer : in std_logic; reset_buffer : in std_logic; en_16_x_baud : in std_logic; serial_out : out std_logic; buffer_data_present : out std_logic; buffer_full : out std_logic; buffer_half_full : out std_logic; clk : in std_logic); end component; -- -- declaration of UART Receiver with integral 16 byte FIFO buffer -- component uart_rx Port ( serial_in : in std_logic; data_out : out std_logic_vector(7 downto 0); read_buffer : in std_logic; reset_buffer : in std_logic; en_16_x_baud : in std_logic; buffer_data_present : out std_logic; buffer_full : out std_logic; buffer_half_full : out std_logic; clk : in std_logic); end component; -- ------------------------------------------------------------------------------------ -- -- Signals used to connect KCPSM3 to program ROM and I/O logic -- signal address : std_logic_vector(9 downto 0); signal instruction : std_logic_vector(17 downto 0); signal port_id : std_logic_vector(7 downto 0); signal out_port : std_logic_vector(7 downto 0); signal in_port : std_logic_vector(7 downto 0); signal write_strobe : std_logic; signal read_strobe : std_logic; signal interrupt : std_logic :='0'; signal interrupt_ack : std_logic; signal kcpsm3_reset : std_logic; -- -- Signals for connection of peripherals -- signal status_port : std_logic_vector(7 downto 0); -- -- -- Signals for UART connections -- signal baud_count : integer range 0 to 35 :=0; signal en_16_x_baud : std_logic; signal write_to_uart : std_logic; signal tx_data_present : std_logic; signal tx_full : std_logic; signal tx_half_full : std_logic; signal read_from_uart : std_logic; signal rx_data : std_logic_vector(7 downto 0); signal rx_data_present : std_logic; signal rx_full : std_logic; signal rx_half_full : std_logic; -- -- -- Signals used to generate interrupt -- signal previous_rx_half_full : std_logic; signal rx_half_full_event : std_logic; -- -- -- Signals to connect to StrataFLASH memory -- signal strataflash_read : std_logic; signal write_data : std_logic_vector(7 downto 0); -- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------ -- -- Start of circuit description -- begin ---------------------------------------------------------------------------------------------------------------------------------- -- Set 8-bit mode of operation for StrataFLASH memory ---------------------------------------------------------------------------------------------------------------------------------- -- -- The StrataFLASH memory can be used in 8-bit or 16-bit modes. Since PicoBlaze is an 8-bit -- processor and the configuration from parallel flash is conducted using an 8-bit interface, -- this design forces the 8-bit data mode. -- -- As a result, the 128Mbit memory is organised as 16,777,216 bytes accessed using a 24-bit address. -- -- -- ---------------------------------------------------------------------------------------------------------------------------------- -- Bidirectional data interface for StrataFLASH memory ---------------------------------------------------------------------------------------------------------------------------------- -- -- To read the StrataFLASH memory the output enable (OE) signal must be driven Low on the memory and -- the pins on the Spartan-3E must become inputs (i.e. the output buffers must be high impedance). -- -- strataflash_oe <= not(strataflash_read); --active Low output enable -- strataflash_d <= write_data when (strataflash_read='0') else "ZZZZZZZZ"; -- ---------------------------------------------------------------------------------------------------------------------------------- -- KCPSM3 and the program memory ---------------------------------------------------------------------------------------------------------------------------------- -- processor: kcpsm3 port map( address => address, instruction => instruction, port_id => port_id, write_strobe => write_strobe, out_port => out_port, read_strobe => read_strobe, in_port => in_port, interrupt => interrupt, interrupt_ack => interrupt_ack, reset => kcpsm3_reset, clk => clk); program_rom: progctrl port map( address => address, instruction => instruction, proc_reset => kcpsm3_reset, clk => clk); -- ---------------------------------------------------------------------------------------------------------------------------------- -- Interrupt ---------------------------------------------------------------------------------------------------------------------------------- -- -- -- Interrupt is used to detect when the UART receiver FIFO reaches half full and this is -- then used to send XON and XOFF flow control characters back to the PC. -- -- If 'rx_half_full' goes High, an interrupt is generated and the subsequent ISR will transmit -- an XOFF character to stop the flow of new characters from the PC and allow the FIFO to start to empty. -- -- If 'rx_half_full' goes Low, an interrupt is generated and the subsequent ISR will transmit -- an XON character which will allow the PC to send new characters and allow the FIFO to start to fill. -- interrupt_control: process(clk) begin if clk'event and clk='1' then -- detect change in state of the 'rx_half_full' flag. previous_rx_half_full <= rx_half_full; rx_half_full_event <= previous_rx_half_full xor rx_half_full; -- processor interrupt waits for an acknowledgement if interrupt_ack='1' then interrupt <= '0'; elsif rx_half_full_event='1' then interrupt <= '1'; else interrupt <= interrupt; end if; end if; end process interrupt_control; -- ---------------------------------------------------------------------------------------------------------------------------------- -- KCPSM3 input ports ---------------------------------------------------------------------------------------------------------------------------------- -- -- -- UART FIFO status signals to form a bus -- Also the status signal (STS) from the StrataFlash memory -- status_port <= strataflash_sts & '0' & rx_full & rx_half_full & rx_data_present & tx_full & tx_half_full & tx_data_present; status_port <= '0' & '0' & rx_full & rx_half_full & rx_data_present & tx_full & tx_half_full & tx_data_present; -- -- The inputs connect via a pipelined multiplexer -- input_ports: process(clk) begin if clk'event and clk='1' then case port_id(1 downto 0) is -- read status signals at address 00 hex when "00" => in_port <= status_port; -- read UART receive data at address 01 hex when "01" => in_port <= rx_data; -- read StrataFLASH memory data at address 02 hex when "10" => in_port <= strataflash_d; -- Don't care used for all other addresses to ensure minimum logic implementation when others => in_port <= "XXXXXXXX"; end case; -- Form read strobe for UART receiver FIFO buffer at address 01 hex. -- The fact that the read strobe will occur after the actual data is read by -- the KCPSM3 is acceptable because it is really means 'I have read you'! if (read_strobe='1' and port_id(1 downto 0)="01") then read_from_uart <= '1'; else read_from_uart <= '0'; end if; end if; end process input_ports; -- ---------------------------------------------------------------------------------------------------------------------------------- -- KCPSM3 output ports ---------------------------------------------------------------------------------------------------------------------------------- -- -- adding the output registers to the processor output_ports: process(clk) begin if clk'event and clk='1' then if write_strobe='1' then -- The 24-bit address to the StrataFLASH memory requires 3 ports. -- Address [23:16] at port 80 hex if port_id(7)='1' then strataflash_a(20 downto 16) <= out_port(4 downto 0); end if; -- Address [15:8] at port 40 hex if port_id(6)='1' then strataflash_a(15 downto 8) <= out_port; end if; -- Address [7:0] at port 20 hex if port_id(5)='1' then strataflash_a(7 downto 0) <= out_port; end if; -- Data to be written to StrataFlash at port 10 hex if port_id(4)='1' then write_data <= out_port; end if; -- StrataFlash control signals at port 08 hex if port_id(3)='1' then strataflash_read <= out_port(0); --Active High and used to control data bus direction and OE strataflash_ce <= out_port(1); --Active Low StrataFLASH device enable strataflash_we <= out_port(2); --Active Low StrataFLASH write enable end if; end if; end if; end process output_ports; -- -- write to UART transmitter FIFO buffer at address 04 hex. -- This is a combinatorial decode because the FIFO is the 'port register'. -- write_to_uart <= '1' when (write_strobe='1' and port_id(2)='1') else '0'; -- ---------------------------------------------------------------------------------------------------------------------------------- -- UART ---------------------------------------------------------------------------------------------------------------------------------- -- -- Connect the 8-bit, 1 stop-bit, no parity transmit and receive macros. -- Each contains an embedded 16-byte FIFO buffer. -- transmit: uart_tx_plus port map ( data_in => out_port, write_buffer => write_to_uart, reset_buffer => '0', en_16_x_baud => en_16_x_baud, serial_out => tx_female, buffer_data_present => tx_data_present, buffer_full => tx_full, buffer_half_full => tx_half_full, clk => clk ); receive: uart_rx port map ( serial_in => rx_female, data_out => rx_data, read_buffer => read_from_uart, reset_buffer => '0', en_16_x_baud => en_16_x_baud, buffer_data_present => rx_data_present, buffer_full => rx_full, buffer_half_full => rx_half_full, clk => clk ); -- -- Set baud rate to 115200 for the UART communications -- Requires en_16_x_baud to be 1843200Hz which is a single cycle pulse every 27 cycles at 50MHz -- baud_timer: process(clk) begin if clk'event and clk='1' then if baud_count=34 then baud_count <= 0; en_16_x_baud <= '1'; else baud_count <= baud_count + 1; en_16_x_baud <= '0'; end if; end if; end process baud_timer; -- ---------------------------------------------------------------------------------------------------------------------------------- end Behavioral; ------------------------------------------------------------------------------------------------------------------------------------ -- -- END OF FILE parallel_flash_memory_uart_programmer.vhd -- ------------------------------------------------------------------------------------------------------------------------------------