Single Transfer PCI Target mit 4 KByte RAM
Die für dieses Projekt modifizerten Dateien des LogiCore PCI Interfaces und die darüber hinaus benötigten Dateien sind:
Dieser Entwurf implementiert ein PCI-Target ohne Burstunterstützung, das ein 4 KByte (1 K x 32 Bit) großes RAM über den PCI-Bus ansteuert. Der PCI Schnittstellenteil ist praktisch identisch mit dem "Board-Test"-Entwurf und wird daher an dieser Stelle nicht nochmals beschrieben.
Anpassung des LogiCore PCI Interfaces
Das LogiCore PCI Interface wird in der Datei cfg.vhd mit symbolischen Konstanten konfiguriert.
Das PCI-Target belegt einen 4 KByte umfassenden Speicherbereich. Außerdem erfüllt das RAM die Prefetch-Bedingungen. Ansonsten ist die Konfiguration identisch mit der des Projektes "Board-Test".
[...] -------------------------------------------------------------- -- Configure Base Address Registers -------------------------------------------------------------- -- BAR0 : 4 KByte prefetchable memory space cfg_int(0) <= ENABLE ; cfg_int(32 downto 1) <= SIZE4K ; cfg_int(33) <= PREFETCH ; -- Hier nun das Prefetch-Attribut gesetzt! cfg_int(35 downto 34) <= TYPE00 ; cfg_int(36) <= MEMORY ; [...]
Aufbau des RAMs
Der Speicher soll aus den Spartan-II internen Block RAMs aufgebaut werden, den gesamten geplanten Adressraum der PCI-Karte (4 KByte) abdecken und natürlich auf voller PCI-Busbreite (32 Bit) arbeiten. Also wird ein RAM-Modul mit 32 Bit Wortbeite und 1 K (10 Bit) Adressraum benötigt, da 1 K x 32 Bit = 4 KByte.
Das Block RAM Primitiv RAMB4_S4 der Xilinx-Bibliothek hat einen 10 Bit Adressraum und eine Wortbreite von 4 Bit:
-- Primitiv aus der Xilinx-Bibliothek entity RAMB4_S4 is [...] port ( ADDR : in std_logic_vector(9 downto 0); -- Adresse CLK : in std_logic; -- Takt DI : in std_logic_vector(4 downto 0); -- Dateneingang EN : in std_logic; -- RAM Enable (Chip Select) RST : in std_logic; -- Reset (Initialisierung des Speichers) WE : in std_logic; -- Write Enable DO : out std_logic_vector(4 downto 0) -- Datenausgang ); end RAMB4_S4;
Bei der Implementierung wird es auf genau eines der 14 Block RAMs im Spartan-II abgebildet. Ausführliche Erklärungen zu den Block RAMs des Spartan-II befinden sich in der Spartan-II Dokumentation (Functional Description). Hier nur die wesentlichen Stichpunkte:
- Das RAM arbeitet taktsynchron, d.h. Eingangssignale werden bei steigender Taktflanke ausgewertet, und die Ausgänge sind bei steigender Taktflanke gültig.
- EN = 1: Der Inhalt der Speicherzelle mit Adresse ADDR wird mit der nächsten steigenden Taktflanke auf DO ausgegeben.
- EN = 1: Zu schreibende Daten werden an DI angelegt und WE auf 1 gesetzt. Mit der nächsten steigenden Flanke erfolgt die Speicherung in die Speicherzelle mit Adresse ADDR.
- EN = 0: Ist EN (Chip Select) nicht gesetzt, so behält der Datenausgang DO seinen letzten Zustand bei. Außerdem kann nichts gespeichert werden.
Durch "Parallelschaltung" von 8 dieser RAMB4_S4 Instanzen (8 x 4 Bit = 32 Bit Wortbreite) erhält man die gewünschte RAM-Struktur bram_1k_32b. Die separaten Dateneingangs- und Datenausgangsbusse werden zum bidirektionalen Bus DIO zusammengefasst, siehe Abbildung.
Als VHDL-Code für das bram_1k_32b Modul ergibt sich damit (gekürzt):
library IEEE; use IEEE.STD_LOGIC_1164.ALL; -- Einbinden der Xilinx Bibliothek library UNISIM; use UNISIM.Vcomponents.ALL; entity bram_1k_32b is port ( ADDR : in std_logic_vector(9 downto 0); -- Adresse CLK : in std_logic; -- Takt DIO : inout std_logic_vector(31 downto 0); -- bidirektionaler Datenbus (Tristate) OE : in std_logic; -- Output Enable RST : in std_logic; -- Reset (Initialisierung des RAMs) WE : in std_logic -- Write Signal ); end bram_1k_32b; architecture rtl of bram_1k_32b is constant LOGIC_1 : std_logic := '1'; -- Datenausgaenge der Block RAMs signal dout : std_logic_vector(31 downto 0); begin -- Daten auf den Bus, wenn OE = '1' und WE = '0' (lesen) DIO <= dout when OE = '1' and WE = '0' else (others => 'Z'); -- Block RAM fuer die Datenbits 0 - 3 blockram_inst0 : RAMB4_S4 port map ( ADDR => ADDR(9 downto 0), CLK => CLK, DI => DIO(3 downto 0), EN => LOGIC_1, RST => RST, WE => WE, DO => dout(3 downto 0) ); -- Block RAM fuer die Datenbits 7 - 4 blockram_inst1 : RAMB4_S4 port map ( ADDR => ADDR(9 downto 0), CLK => CLK, DI => DIO(7 downto 4), EN => LOGIC_1, RST => RST, WE => WE, DO => dout(7 downto 4) ); [...] -- Block RAM fuer die Datenbits 31 - 28 blockram_inst7 : RAMB4_S4 port map ( ADDR => ADDR(9 downto 0), CLK => CLK, DI => DIO(31 downto 28), EN => LOGIC_1, RST => RST, WE => WE, DO => dout(31 downto 28) ); end rtl;
Die Datei bram_1k_32b.vhd wird dem Projekt hinzugefügt.
Der Hardware-Entwurf des PCI-Targets
Da außer dem PCI-Interface keine I/O-Pins verwendet werden, wird neben der Konfigurationsdatei cfg.vhd, siehe oben, nur die Datei userapp.vhd verändert. Das Modul bram_1k_32b wird als Komponente eingebunden und instanziiert. Der PCI-Teil des Entwurfs stimmt mit dem des Projektes "Board-Test" weitgehend überein. Er ist sogar noch einfacher, weil die gesamte Adressdekodierung RAM-intern geschieht. Deshalb wird an dieser Stelle auf eine nochmalige ausführliche Erläuterung verzichtet und direkt auf den VHDL-Code verwiesen.