2021-12-21 08:26:30 +01:00
import os
import argparse
from migen import *
from migen . genlib . fifo import *
from migen . fhdl . specials import Tristate
import litex
from litex . build . generic_platform import *
from litex . build . xilinx . vivado import vivado_build_args , vivado_build_argdict
from litex . soc . integration . soc import *
from litex . soc . integration . soc_core import *
from litex . soc . integration . builder import *
from litex . soc . interconnect import wishbone
from litex . soc . cores . clock import *
from litex . soc . cores . led import LedChaser
import ztex213_nubus
2022-01-29 11:03:47 +01:00
import nubus_to_fpga_export
2021-12-21 08:26:30 +01:00
2022-11-05 14:54:35 +01:00
import nubus_full_unified
2022-06-04 18:56:41 +02:00
import nubus_stat
2021-12-21 08:26:30 +01:00
from litedram . modules import MT41J128M16
from litedram . phy import s7ddrphy
from litedram . frontend . dma import *
2023-04-16 09:04:21 +02:00
from liteeth . phy . rmii import LiteEthPHYRMII
2021-12-21 08:26:30 +01:00
from migen . genlib . cdc import BusSynchronizer
from migen . genlib . resetsync import AsyncResetSynchronizer
2022-04-17 11:25:48 +02:00
from litex . soc . cores . video import VideoS7HDMIPHY
2021-12-21 08:26:30 +01:00
from litex . soc . cores . video import VideoVGAPHY
2022-04-17 11:25:48 +02:00
from litex . soc . cores . video import video_timings
2022-10-31 15:45:24 +01:00
from VintageBusFPGA_Common . goblin_accel import *
2021-12-21 08:26:30 +01:00
# Wishbone stuff
2022-10-31 15:45:24 +01:00
from VintageBusFPGA_Common . cdc_wb import WishboneDomainCrossingMaster
2022-10-31 15:58:47 +01:00
from VintageBusFPGA_Common . fpga_blk_dma import *
2021-12-21 08:26:30 +01:00
from nubus_mem_wb import NuBus2Wishbone
2022-05-30 13:15:20 +02:00
from nubus_memfifo_wb import NuBus2WishboneFIFO
2022-04-17 11:25:48 +02:00
from nubus_cpu_wb import Wishbone2NuBus
2021-12-21 08:26:30 +01:00
# CRG ----------------------------------------------------------------------------------------------
class _CRG ( Module ) :
2023-01-14 11:03:01 +01:00
def __init__ ( self , platform , version , sys_clk_freq ,
2022-02-05 15:32:44 +01:00
goblin = False ,
2021-12-21 08:26:30 +01:00
hdmi = False ,
2023-04-16 09:04:21 +02:00
pix_clk = 0 ,
ethernet = False ) :
2021-12-21 08:26:30 +01:00
self . clock_domains . cd_sys = ClockDomain ( ) # 100 MHz PLL, reset'ed by NuBus (via pll), SoC/Wishbone main clock
self . clock_domains . cd_sys4x = ClockDomain ( reset_less = True )
self . clock_domains . cd_sys4x_dqs = ClockDomain ( reset_less = True )
self . clock_domains . cd_idelay = ClockDomain ( )
self . clock_domains . cd_native = ClockDomain ( reset_less = True ) # 48MHz native, non-reset'ed (for power-on long delay, never reset, we don't want the delay after a warm reset)
self . clock_domains . cd_nubus = ClockDomain ( ) # 10 MHz NuBus, reset'ed by NuBus, native NuBus clock domain (25% duty cycle)
self . clock_domains . cd_nubus90 = ClockDomain ( ) # 20 MHz NuBus90, reset'ed by NuBus, native NuBus90 clock domain (25% duty cycle)
2022-02-05 15:32:44 +01:00
if ( goblin ) :
2021-12-21 08:26:30 +01:00
if ( not hdmi ) :
self . clock_domains . cd_vga = ClockDomain ( reset_less = True )
else :
self . clock_domains . cd_hdmi = ClockDomain ( )
self . clock_domains . cd_hdmi5x = ClockDomain ( )
2023-04-16 09:04:21 +02:00
if ( ethernet ) :
self . clock_domains . cd_eth = ClockDomain ( )
2021-12-21 08:26:30 +01:00
# # #
clk48 = platform . request ( " clk48 " )
###### explanations from betrusted-io/betrusted-soc/betrusted_soc.py
# Note: below feature cannot be used because Litex appends this *after* platform commands! This causes the generated
# clock derived constraints immediately below to fail, because .xdc file is parsed in-order, and the main clock needs
# to be created before the derived clocks. Instead, we use the line afterwards.
platform . add_platform_command ( " create_clock -name clk48 -period 20.8333 [get_nets clk48] " )
# The above constraint must strictly proceed the below create_generated_clock constraints in the .XDC file
# This allows PLLs/MMCMEs to be placed anywhere and reference the input clock
self . clk48_bufg = Signal ( )
self . specials + = Instance ( " BUFG " , i_I = clk48 , o_O = self . clk48_bufg )
self . comb + = self . cd_native . clk . eq ( self . clk48_bufg )
#self.cd_native.clk = clk48
2023-01-14 11:03:01 +01:00
##### V1.2 extra clock for B34
if ( version == " V1.2 " ) :
self . clock_domains . cd_bank34 = ClockDomain ( )
clk54 = platform . request ( " clk54 " )
self . clk54_bufg = Signal ( )
self . specials + = Instance ( " BUFG " , i_I = clk54 , o_O = self . clk54_bufg )
self . comb + = self . cd_native . clk . eq ( self . clk54_bufg )
else :
clk54 = None
2021-12-21 08:26:30 +01:00
clk_nubus = platform . request ( " clk_3v3_n " )
if ( clk_nubus is None ) :
print ( " ***** ERROR ***** Can ' t find the NuBus Clock !!!! \n " ) ;
assert ( false )
self . cd_nubus . clk = clk_nubus
rst_nubus_n = platform . request ( " reset_3v3_n " )
self . comb + = self . cd_nubus . rst . eq ( ~ rst_nubus_n )
platform . add_platform_command ( " create_clock -name nubus_clk -period 100.0 -waveform {{ 0.0 75.0}} [get_ports clk_3v3_n] " )
2022-01-29 11:03:47 +01:00
clk2x_nubus = platform . request ( " clk2x_3v3_n " )
if ( clk2x_nubus is None ) :
print ( " ***** ERROR ***** Can ' t find the NuBus90 Clock !!!! \n " ) ;
assert ( false )
self . cd_nubus90 . clk = clk2x_nubus
self . comb + = self . cd_nubus90 . rst . eq ( ~ rst_nubus_n )
platform . add_platform_command ( " create_clock -name nubus90_clk -period 50.0 -waveform {{ 0.0 37.5}} [get_ports clk2x_3v3_n] " )
2021-12-21 08:26:30 +01:00
num_adv = 0
num_clk = 0
self . submodules . pll = pll = S7MMCM ( speedgrade = platform . speedgrade )
#pll.register_clkin(clk48, 48e6)
pll . register_clkin ( self . clk48_bufg , 48e6 )
pll . create_clkout ( self . cd_sys , sys_clk_freq )
platform . add_platform_command ( " create_generated_clock -name sysclk [get_pins {{ {{ MMCME2_ADV/CLKOUT {} }}}}] " . format ( num_clk ) )
num_clk = num_clk + 1
pll . create_clkout ( self . cd_sys4x , 4 * sys_clk_freq )
platform . add_platform_command ( " create_generated_clock -name sys4xclk [get_pins {{ {{ MMCME2_ADV/CLKOUT {} }}}}] " . format ( num_clk ) )
num_clk = num_clk + 1
pll . create_clkout ( self . cd_sys4x_dqs , 4 * sys_clk_freq , phase = 90 )
platform . add_platform_command ( " create_generated_clock -name sys4x90clk [get_pins {{ {{ MMCME2_ADV/CLKOUT {} }}}}] " . format ( num_clk ) )
num_clk = num_clk + 1
2023-04-16 09:04:21 +02:00
if ( ethernet ) :
pll . create_clkout ( self . cd_eth , 50e6 , phase = 90 ) # fixme: what if sys_clk_feq != 100e6?
platform . add_platform_command ( " create_generated_clock -name ethclk [get_pins {{ {{ MMCME2_ADV/CLKOUT {} }}}}] " . format ( num_clk ) )
num_clk = num_clk + 1
2021-12-21 08:26:30 +01:00
self . comb + = pll . reset . eq ( ~ rst_nubus_n ) # | ~por_done
2023-01-14 11:03:01 +01:00
platform . add_false_path_constraints ( clk48 , self . cd_nubus . clk ) # FIXME?
platform . add_false_path_constraints ( self . cd_nubus . clk , clk48 ) # FIXME?
2021-12-21 08:26:30 +01:00
#platform.add_false_path_constraints(self.cd_sys.clk, self.cd_nubus.clk)
#platform.add_false_path_constraints(self.cd_nubus.clk, self.cd_sys.clk)
##platform.add_false_path_constraints(self.cd_native.clk, self.cd_sys.clk)
num_adv = num_adv + 1
num_clk = 0
self . submodules . pll_idelay = pll_idelay = S7MMCM ( speedgrade = platform . speedgrade )
#pll_idelay.register_clkin(clk48, 48e6)
pll_idelay . register_clkin ( self . clk48_bufg , 48e6 )
pll_idelay . create_clkout ( self . cd_idelay , 200e6 , margin = 0 )
platform . add_platform_command ( " create_generated_clock -name idelayclk [get_pins {{ {{ MMCME2_ADV_ {} /CLKOUT {} }}}}] " . format ( num_adv , num_clk ) )
num_clk = num_clk + 1
self . comb + = pll_idelay . reset . eq ( ~ rst_nubus_n ) # | ~por_done
self . submodules . idelayctrl = S7IDELAYCTRL ( self . cd_idelay )
num_adv = num_adv + 1
num_clk = 0
2022-02-05 15:32:44 +01:00
if ( goblin ) :
2021-12-21 08:26:30 +01:00
self . submodules . video_pll = video_pll = S7MMCM ( speedgrade = platform . speedgrade )
2023-01-14 11:03:01 +01:00
if ( clk54 is None ) :
# no 54 MHz clock, drive hdmi from the main clock
video_pll . register_clkin ( self . clk48_bufg , 48e6 )
else :
# drive hdmi from the 54 MHz clock, easier to generate e.g. 148.5 MHz
video_pll . register_clkin ( self . clk54_bufg , 54e6 )
platform . add_false_path_constraints ( self . cd_bank34 . clk , self . cd_nubus . clk ) # FIXME?
platform . add_false_path_constraints ( self . cd_bank34 . clk , clk48 ) # FIXME?
2021-12-21 08:26:30 +01:00
if ( not hdmi ) :
video_pll . create_clkout ( self . cd_vga , pix_clk , margin = 0.0005 )
platform . add_platform_command ( " create_generated_clock -name vga_clk [get_pins {{ {{ MMCME2_ADV_ {} /CLKOUT {} }}}}] " . format ( num_adv , num_clk ) )
num_clk = num_clk + 1
else :
2022-04-17 11:25:48 +02:00
video_pll . create_clkout ( self . cd_hdmi , pix_clk , margin = 0.005 )
video_pll . create_clkout ( self . cd_hdmi5x , 5 * pix_clk , margin = 0.005 )
2021-12-21 08:26:30 +01:00
platform . add_platform_command ( " create_generated_clock -name hdmi_clk [get_pins {{ {{ MMCME2_ADV_ {} /CLKOUT {} }}}}] " . format ( num_adv , num_clk ) )
num_clk = num_clk + 1
platform . add_platform_command ( " create_generated_clock -name hdmi5x_clk [get_pins {{ {{ MMCME2_ADV_ {} /CLKOUT {} }}}}] " . format ( num_adv , num_clk ) )
num_clk = num_clk + 1
2023-01-08 15:11:22 +01:00
2021-12-21 08:26:30 +01:00
self . comb + = video_pll . reset . eq ( ~ rst_nubus_n )
#platform.add_false_path_constraints(self.cd_sys.clk, self.cd_vga.clk)
platform . add_false_path_constraints ( self . cd_sys . clk , video_pll . clkin )
num_adv = num_adv + 1
num_clk = 0
class NuBusFPGA ( SoCCore ) :
2023-04-16 09:04:21 +02:00
def __init__ ( self , variant , version , sys_clk_freq , goblin , hdmi , goblin_res , ethernet , * * kwargs ) :
2021-12-21 08:26:30 +01:00
print ( f " Building NuBusFPGA for board version { version } " )
kwargs [ " cpu_type " ] = " None "
kwargs [ " integrated_sram_size " ] = 0
kwargs [ " with_uart " ] = False
kwargs [ " with_timer " ] = False
self . sys_clk_freq = sys_clk_freq
self . platform = platform = ztex213_nubus . Platform ( variant = variant , version = version )
2023-04-16 09:04:21 +02:00
if ( ethernet and ( version == " V1.2 " ) ) :
platform . add_extension ( ztex213_nubus . _rmii_eth_extpmod_io_v1_2 )
2022-12-20 08:19:41 +01:00
use_goblin_alt = True
if ( ( not use_goblin_alt ) or ( not hdmi ) ) :
from VintageBusFPGA_Common . goblin_fb import goblin_rounded_size , Goblin
else :
from VintageBusFPGA_Common . goblin_alt_fb import goblin_rounded_size , GoblinAlt
2022-02-05 15:32:44 +01:00
if ( goblin ) :
hres = int ( goblin_res . split ( " @ " ) [ 0 ] . split ( " x " ) [ 0 ] )
vres = int ( goblin_res . split ( " @ " ) [ 0 ] . split ( " x " ) [ 1 ] )
2022-10-31 15:45:24 +01:00
goblin_fb_size = goblin_rounded_size ( hres , vres )
2022-02-05 15:32:44 +01:00
print ( f " Reserving { goblin_fb_size } bytes ( { goblin_fb_size / / 1048576 } MiB) for the goblin " )
2021-12-21 08:26:30 +01:00
else :
hres = 0
vres = 0
2022-02-05 15:32:44 +01:00
goblin_fb_size = 0
2022-10-31 15:45:24 +01:00
# litex.soc.cores.video.video_timings.update(goblin_timings)
2021-12-21 08:26:30 +01:00
SoCCore . __init__ ( self ,
platform = platform ,
sys_clk_freq = sys_clk_freq ,
clk_freq = sys_clk_freq ,
csr_paging = 0x800 , # default is 0x800
2022-11-04 09:28:55 +01:00
bus_interconnect = " crossbar " ,
2021-12-21 08:26:30 +01:00
* * kwargs )
# Quoting the doc:
# * Separate address spaces are reserved for processor access to cards in NuBus slots. For a
# * device in NuBus slot number s, the address space in 32-bit mode begins at address
# * $Fs00 0000 and continues through the highest address, $FsFF FFFF (where s is a constant in
# * the range $9 through $E for the Macintosh II, the Macintosh IIx, and the Macintosh IIfx;
# * $A through $E for the Macintosh Quadra 900; $9 through $B for the Macintosh IIcx;
# * $C through $E for the Macintosh IIci; $D and $E for the Macintosh Quadra 700; and
# * $9 for the Macintosh IIsi).
2022-04-17 11:25:48 +02:00
# the Q650 is $C through $E like the IIci, $E is the one with the PDS.
# So at best we get 16 MiB in 32-bits mode, unless using "super slot space"
2021-12-21 08:26:30 +01:00
# in 24 bits it's only one megabyte, $s0 0000 through $sF FFFF
# they are translated: '$s0 0000-$sF FFFF' to '$Fs00 0000-$Fs0F FFFF' (for s in range $9 through $E)
2022-04-17 11:25:48 +02:00
# let's assume we have 32-bits mode, this can be requested in the DeclROM apparently
2021-12-21 08:26:30 +01:00
self . wb_mem_map = wb_mem_map = {
2022-05-15 14:05:23 +02:00
# master to map the NuBus access to RAM
" master " : 0x00000000 , # to 0x3FFFFFFF
2021-12-21 08:26:30 +01:00
" main_ram " : 0x80000000 , # not directly reachable from NuBus
2022-02-05 15:32:44 +01:00
" video_framebuffer " : 0x80000000 + 0x10000000 - goblin_fb_size , # Updated later
2022-05-15 14:05:23 +02:00
# map everything in slot 0, remapped from the real slot in NuBus2Wishbone
" goblin_mem " : 0xF0000000 , # up to 8 MiB of FB memory
#"END OF FIRST MB" : 0xF00FFFFF,
#"END OF 8 MB": 0xF07FFFFF,
" goblin_bt " : 0xF0900000 , # BT for goblin (regs)
" goblin_accel " : 0xF0901000 , # accel for goblin (regs)
" goblin_accel_ram " : 0xF0902000 , # accel for goblin (scratch ram)
2022-06-04 18:56:41 +02:00
" stat " : 0xF0903000 , # stat
2022-05-15 14:05:23 +02:00
" goblin_accel_rom " : 0xF0910000 , # accel for goblin (rom)
2023-01-08 15:11:22 +01:00
" goblin_audio_ram " : 0xF0920000 , # audio for goblin (RAM buffers)
2022-05-30 19:06:33 +02:00
" csr " : 0xF0A00000 , # CSR
" pingmaster " : 0xF0B00000 ,
2023-04-16 09:04:21 +02:00
" ethmac " : 0xF0C00000 ,
2022-05-15 14:05:23 +02:00
" rom " : 0xF0FF8000 , # ROM at the end (32 KiB of it ATM)
#"END OF SLOT SPACE": 0xF0FFFFFF,
2021-12-21 08:26:30 +01:00
}
self . mem_map . update ( wb_mem_map )
2023-04-16 09:04:21 +02:00
self . submodules . crg = _CRG ( platform = platform , version = version , sys_clk_freq = sys_clk_freq , goblin = goblin , hdmi = hdmi , pix_clk = litex . soc . cores . video . video_timings [ goblin_res ] [ " pix_clk " ] , ethernet = ethernet )
2021-12-21 08:26:30 +01:00
## add our custom timings after the clocks have been defined
xdc_timings_filename = None ;
2022-09-21 23:06:17 +02:00
#if (version == "V1.0"):
# xdc_timings_filename = "/home/dolbeau/nubus-to-ztex-gateware/nubus_fpga_V1_0_timings.xdc"
2021-12-21 08:26:30 +01:00
if ( xdc_timings_filename != None ) :
xdc_timings_file = open ( xdc_timings_filename )
2022-04-17 11:25:48 +02:00
2021-12-21 08:26:30 +01:00
xdc_timings_lines = xdc_timings_file . readlines ( )
for line in xdc_timings_lines :
if ( line [ 0 : 3 ] == " set " ) :
fix_line = line . strip ( ) . replace ( " { " , " {{ " ) . replace ( " } " , " }} " )
#print(fix_line)
platform . add_platform_command ( fix_line )
rom_file = " rom_ {} .bin " . format ( version . replace ( " . " , " _ " ) )
2022-10-09 09:02:30 +02:00
rom_data = soc_core . get_mem_data ( filename_or_regions = rom_file , endianness = " little " ) # "big"
2021-12-21 08:26:30 +01:00
# rom = Array(rom_data)
#print("\n****************************************\n")
#for i in range(len(rom)):
# print(hex(rom[i]))
#print("\n****************************************\n")
2022-02-05 15:32:44 +01:00
self . add_ram ( " rom " , origin = self . mem_map [ " rom " ] , size = 2 * * 15 , contents = rom_data , mode = " r " ) ## 32 KiB, must match mmap
2021-12-21 08:26:30 +01:00
2022-04-17 11:25:48 +02:00
#from wb_test import WA2D
#self.submodules.wa2d = WA2D(self.platform)
#self.bus.add_slave("WA2D", self.wa2d.bus, SoCRegion(origin=0x00C00000, size=0x00400000, cached=False))
2022-11-05 14:59:33 +01:00
# notsimul to signify we're making a real bitstream
# notsimul == False only to produce a verilog implementation to simulate the bus side of things
notsimul = True
2022-04-17 11:25:48 +02:00
if ( notsimul ) :
avail_sdram = 0
self . submodules . ddrphy = s7ddrphy . A7DDRPHY ( platform . request ( " ddram " ) ,
memtype = " DDR3 " ,
nphases = 4 ,
sys_clk_freq = sys_clk_freq )
self . add_sdram ( " sdram " ,
phy = self . ddrphy ,
module = MT41J128M16 ( sys_clk_freq , " 1:4 " ) ,
l2_cache_size = 0 ,
)
avail_sdram = self . bus . regions [ " main_ram " ] . size
2023-01-27 22:25:12 +01:00
#from sdram_init import DDR3FBInit
#self.submodules.sdram_init = DDR3FBInit(sys_clk_freq=sys_clk_freq, bitslip=1, delay=25)
#self.bus.add_master(name="DDR3Init", master=self.sdram_init.bus)
2022-04-17 11:25:48 +02:00
else :
avail_sdram = 256 * 1024 * 1024
2022-05-15 14:05:23 +02:00
self . add_ram ( " ram " , origin = 0x8f800000 , size = 2 * * 16 , mode = " rw " )
2021-12-21 08:26:30 +01:00
2022-06-04 09:53:09 +02:00
if ( not notsimul ) : # otherwise we have no CSRs and litex doesn't like that
self . submodules . leds = ClockDomainsRenamer ( " nubus " ) ( LedChaser (
pads = platform . request_all ( " user_led " ) ,
sys_clk_freq = 10e6 ) )
self . add_csr ( " leds " )
2021-12-21 08:26:30 +01:00
base_fb = self . wb_mem_map [ " main_ram " ] + avail_sdram - 1048576 # placeholder
2022-02-05 15:32:44 +01:00
if ( goblin ) :
if ( avail_sdram > = goblin_fb_size ) :
avail_sdram = avail_sdram - goblin_fb_size
2021-12-21 08:26:30 +01:00
base_fb = self . wb_mem_map [ " main_ram " ] + avail_sdram
self . wb_mem_map [ " video_framebuffer " ] = base_fb
2022-04-17 11:25:48 +02:00
print ( f " FrameBuffer base_fb @ { base_fb : x } " )
2021-12-21 08:26:30 +01:00
else :
print ( " ***** ERROR ***** Can ' t have a FrameBuffer without main ram \n " )
assert ( False )
# don't enable anything on the NuBus side for XX seconds after power up
# this avoids FPGA initialization messing with the cold boot process
# requires us to reset the Macintosh afterward so the FPGA board
# is properly identified
# This is in the 'native' ClockDomain that is never reset
2022-04-17 11:25:48 +02:00
# not needed, FPGA initializes fast enough, works on cold boots
2021-12-21 08:26:30 +01:00
#hold_reset_ctr = Signal(30, reset=960000000)
hold_reset_ctr = Signal ( 5 , reset = 31 )
self . sync . native + = If ( hold_reset_ctr > 0 , hold_reset_ctr . eq ( hold_reset_ctr - 1 ) )
2022-04-17 11:25:48 +02:00
hold_reset = Signal ( )
2021-12-21 08:26:30 +01:00
self . comb + = hold_reset . eq ( ~ ( hold_reset_ctr == 0 ) )
pad_nubus_oe = platform . request ( " nubus_oe " )
self . comb + = pad_nubus_oe . eq ( hold_reset )
2022-04-17 11:25:48 +02:00
#pad_user_led_0 = platform.request("user_led", 0)
#self.comb += pad_user_led_0.eq(~hold_reset)
2021-12-21 08:26:30 +01:00
# Interface NuBus to wishbone
# we need to cross clock domains
2022-11-05 14:59:33 +01:00
# Xibus is the original VErilog implementation I used
# mostly only for testing now, it doesn't have block mode so doesn't support the DMA mode of the RAM disk
# Should be set to False unless for testing (usually with notsimul=False)
2022-11-05 14:54:35 +01:00
xibus = False
2022-06-04 09:53:09 +02:00
if ( xibus ) :
wishbone_master_sys = wishbone . Interface ( data_width = self . bus . data_width )
self . submodules . wishbone_master_nubus = WishboneDomainCrossingMaster ( platform = self . platform , slave = wishbone_master_sys , cd_master = " nubus " , cd_slave = " sys " )
self . bus . add_master ( name = " NuBusBridgeToWishbone " , master = wishbone_master_sys )
2022-11-01 15:42:59 +01:00
if ( version == " V1.0 " ) :
from nubus_V1_0 import NuBus
elif ( version == " V1.2 " ) :
from nubus_V1_2 import NuBus
self . submodules . nubus = NuBus ( platform = platform , cd_nubus = " nubus " )
2022-06-04 09:53:09 +02:00
#self.submodules.nubus2wishbone = ClockDomainsRenamer("nubus")(NuBus2Wishbone(nubus=self.nubus,wb=self.wishbone_master_nubus))
2022-11-01 15:31:34 +01:00
if ( version == " V1.2 " ) :
self . comb + = self . nubus . nubus_oe . eq ( hold_reset ) # improveme
2022-06-04 09:53:09 +02:00
nubus_writemaster_sys = wishbone . Interface ( data_width = self . bus . data_width )
self . submodules . nubus2wishbone = NuBus2WishboneFIFO ( platform = self . platform , nubus = self . nubus , wb_read = self . wishbone_master_nubus , wb_write = nubus_writemaster_sys )
self . bus . add_master ( name = " NuBusBridgeToWishboneWrite " , master = nubus_writemaster_sys )
wishbone_slave_nubus = wishbone . Interface ( data_width = self . bus . data_width )
self . submodules . wishbone2nubus = ClockDomainsRenamer ( " nubus " ) ( Wishbone2NuBus ( nubus = self . nubus , wb = wishbone_slave_nubus ) )
self . submodules . wishbone_slave_sys = WishboneDomainCrossingMaster ( platform = self . platform , slave = wishbone_slave_nubus , cd_master = " sys " , cd_slave = " nubus " )
self . bus . add_slave ( " DMA " , self . wishbone_slave_sys , SoCRegion ( origin = self . mem_map . get ( " master " , None ) , size = 0x40000000 , cached = False ) )
2022-10-31 15:28:08 +01:00
irq_line = self . platform . request ( " nmrq_3v3_n " ) # active low
fb_irq = Signal ( ) # active low
2022-11-01 12:41:58 +01:00
#led0 = platform.request("user_led", 0)
#self.comb += [
# led0.eq(~fb_irq),
#]
2022-10-31 15:28:08 +01:00
self . comb + = irq_line . eq ( fb_irq ) # active low, enable if one is low
2022-06-04 09:53:09 +02:00
else :
2022-11-05 14:59:33 +01:00
# details for usesampling in the NuBus python object
2022-11-05 14:54:35 +01:00
usesampling = True
2022-06-04 09:53:09 +02:00
wishbone_master_sys = wishbone . Interface ( data_width = self . bus . data_width )
2022-11-05 14:54:35 +01:00
if ( not usesampling ) : # we need an extra CDC
2022-09-21 23:06:17 +02:00
self . submodules . wishbone_master_nubus = WishboneDomainCrossingMaster ( platform = self . platform , slave = wishbone_master_sys , cd_master = " nubus " , cd_slave = " sys " ) # for non-sampling only
2022-06-04 09:53:09 +02:00
nubus_writemaster_sys = wishbone . Interface ( data_width = self . bus . data_width )
wishbone_slave_nubus = wishbone . Interface ( data_width = self . bus . data_width )
2022-06-05 18:03:23 +02:00
self . submodules . wishbone_slave_sys = WishboneDomainCrossingMaster ( platform = self . platform , slave = wishbone_slave_nubus , cd_master = " sys " , cd_slave = " nubus " , force_delay = 6 ) # force delay needed to avoid back-to-back transaction running into issue https://github.com/alexforencich/verilog-wishbone/issues/4
2022-07-14 17:17:53 +02:00
burst_size = 4
data_width = burst_size * 4
data_width_bits = burst_size * 32
blk_addr_width = 32 - log2_int ( data_width )
self . tosbus_layout = [
( " address " , 32 ) ,
( " data " , data_width_bits ) ,
]
self . fromsbus_layout = [
( " blkaddress " , blk_addr_width ) ,
( " data " , data_width_bits ) ,
]
self . fromsbus_req_layout = [
( " blkaddress " , blk_addr_width ) ,
( " dmaaddress " , 32 ) ,
]
2022-07-23 12:53:30 +02:00
self . submodules . tosbus_fifo = ClockDomainsRenamer ( { " read " : " nubus " , " write " : " sys " } ) ( AsyncFIFOBuffered ( width = layout_len ( self . tosbus_layout ) , depth = 1024 / / data_width ) )
self . submodules . fromsbus_fifo = ClockDomainsRenamer ( { " write " : " nubus " , " read " : " sys " } ) ( AsyncFIFOBuffered ( width = layout_len ( self . fromsbus_layout ) , depth = 512 / / data_width ) )
self . submodules . fromsbus_req_fifo = ClockDomainsRenamer ( { " read " : " nubus " , " write " : " sys " } ) ( AsyncFIFOBuffered ( width = layout_len ( self . fromsbus_req_layout ) , depth = 512 / / data_width ) )
2022-10-08 18:23:01 +02:00
irq_line = self . platform . request ( " nmrq_3v3_n " ) # active low
2023-01-08 15:11:22 +01:00
fb_irq = Signal ( reset = 1 ) # active low
dma_irq = Signal ( reset = 1 ) # active low
audio_irq = Signal ( reset = 1 ) # active low
#led0 = platform.request("user_led", 0)
#led1 = platform.request("user_led", 1)
#self.comb += [
# led0.eq(~fb_irq),
# led1.eq(~dma_irq),
#]
2022-10-08 10:39:18 +02:00
2023-01-08 15:11:22 +01:00
self . comb + = irq_line . eq ( fb_irq & dma_irq & audio_irq ) # active low, enable if one is low
2022-07-14 17:17:53 +02:00
self . submodules . exchange_with_mem = ExchangeWithMem ( soc = self ,
platform = platform ,
tosbus_fifo = self . tosbus_fifo ,
fromsbus_fifo = self . fromsbus_fifo ,
fromsbus_req_fifo = self . fromsbus_req_fifo ,
dram_native_r = self . sdram . crossbar . get_port ( mode = " read " , data_width = data_width_bits ) ,
dram_native_w = self . sdram . crossbar . get_port ( mode = " write " , data_width = data_width_bits ) ,
mem_size = avail_sdram / / 1048576 ,
burst_size = burst_size ,
2022-07-23 12:53:30 +02:00
do_checksum = False ,
clock_domain = " nubus " )
2022-07-14 17:17:53 +02:00
2022-10-08 10:39:18 +02:00
self . comb + = dma_irq . eq ( self . exchange_with_mem . irq )
2022-11-05 14:54:35 +01:00
self . submodules . nubus = nubus_full_unified . NuBus ( soc = self ,
2023-01-14 11:03:01 +01:00
version = version ,
2022-11-05 14:54:35 +01:00
burst_size = burst_size ,
tosbus_fifo = self . tosbus_fifo ,
fromsbus_fifo = self . fromsbus_fifo ,
fromsbus_req_fifo = self . fromsbus_req_fifo ,
wb_read = ( wishbone_master_sys if usesampling else self . wishbone_master_nubus ) , # CDC or not
wb_write = nubus_writemaster_sys ,
wb_dma = wishbone_slave_nubus ,
usesampling = usesampling ,
cd_nubus = " nubus " )
2022-06-04 09:53:09 +02:00
self . bus . add_master ( name = " NuBusBridgeToWishbone " , master = wishbone_master_sys )
self . bus . add_slave ( " DMA " , self . wishbone_slave_sys , SoCRegion ( origin = self . mem_map . get ( " master " , None ) , size = 0x40000000 , cached = False ) )
self . bus . add_master ( name = " NuBusBridgeToWishboneWrite " , master = nubus_writemaster_sys )
2022-06-04 18:56:41 +02:00
self . submodules . stat = nubus_stat . NuBusStat ( nubus = self . nubus , platform = platform )
self . bus . add_slave ( " Stat " , self . stat . bus_slv , SoCRegion ( origin = self . mem_map . get ( " stat " , None ) , size = 0x1000 , cached = False ) )
2022-06-04 09:53:09 +02:00
2022-02-05 15:32:44 +01:00
if ( goblin ) :
2021-12-21 08:26:30 +01:00
if ( not hdmi ) :
self . submodules . videophy = VideoVGAPHY ( platform . request ( " vga " ) , clock_domain = " vga " )
2022-10-31 15:45:24 +01:00
self . submodules . goblin = Goblin ( soc = self , phy = self . videophy , timings = goblin_res , clock_domain = " vga " , irq_line = fb_irq , endian = " little " , hwcursor = False , truecolor = True ) # clock_domain for the VGA side, goblin is running in cd_sys
2021-12-21 08:26:30 +01:00
else :
2022-12-20 08:19:41 +01:00
if ( not use_goblin_alt ) :
self . submodules . videophy = VideoS7HDMIPHY ( platform . request ( " hdmi " ) , clock_domain = " hdmi " )
self . submodules . goblin = Goblin ( soc = self , phy = self . videophy , timings = goblin_res , clock_domain = " hdmi " , irq_line = fb_irq , endian = " little " , hwcursor = False , truecolor = True ) # clock_domain for the HDMI side, goblin is running in cd_sys
else :
# GoblinAlt contains its own PHY
self . submodules . goblin = GoblinAlt ( soc = self , timings = goblin_res , clock_domain = " hdmi " , irq_line = fb_irq , endian = " little " , hwcursor = False , truecolor = True )
2023-01-08 15:11:22 +01:00
# it also has a bus master so that the audio bit can fetch data from Wishbone
self . bus . add_master ( name = " GoblinAudio " , master = self . goblin . goblin_audio . busmaster )
self . add_ram ( " goblin_audio_ram " , origin = self . mem_map [ " goblin_audio_ram " ] , size = 2 * * 13 , mode = " rw " ) # 8 KiB buffer, planned as 2*4KiB
self . comb + = [ audio_irq . eq ( self . goblin . goblin_audio . irq ) , ]
2022-02-05 15:32:44 +01:00
self . bus . add_slave ( " goblin_bt " , self . goblin . bus , SoCRegion ( origin = self . mem_map . get ( " goblin_bt " , None ) , size = 0x1000 , cached = False ) )
2022-04-22 22:00:25 +01:00
#pad_user_led_0 = platform.request("user_led", 0)
#pad_user_led_1 = platform.request("user_led", 1)
#self.comb += pad_user_led_0.eq(self.goblin.video_framebuffer.underflow)
#self.comb += pad_user_led_1.eq(self.goblin.video_framebuffer.fb_dma.enable)
2022-05-15 14:05:23 +02:00
if ( True ) :
2022-10-31 15:45:24 +01:00
self . submodules . goblin_accel = GoblinAccelNuBus ( soc = self )
2022-05-15 14:05:23 +02:00
self . bus . add_slave ( " goblin_accel " , self . goblin_accel . bus , SoCRegion ( origin = self . mem_map . get ( " goblin_accel " , None ) , size = 0x1000 , cached = False ) )
self . bus . add_master ( name = " goblin_accel_r5_i " , master = self . goblin_accel . ibus )
self . bus . add_master ( name = " goblin_accel_r5_d " , master = self . goblin_accel . dbus )
2022-10-31 16:38:29 +01:00
goblin_rom_file = " VintageBusFPGA_Common/blit_goblin_nubus.raw "
2022-10-09 09:02:30 +02:00
goblin_rom_data = soc_core . get_mem_data ( filename_or_regions = goblin_rom_file , endianness = " little " )
2022-05-15 14:05:23 +02:00
goblin_rom_len = 4 * len ( goblin_rom_data ) ;
rounded_goblin_rom_len = 2 * * log2_int ( goblin_rom_len , False )
print ( f " GOBLIN ROM is { goblin_rom_len } bytes, using { rounded_goblin_rom_len } " )
assert ( rounded_goblin_rom_len < = 2 * * 16 )
self . add_ram ( " goblin_accel_rom " , origin = self . mem_map [ " goblin_accel_rom " ] , size = rounded_goblin_rom_len , contents = goblin_rom_data , mode = " r " )
self . add_ram ( " goblin_accel_ram " , origin = self . mem_map [ " goblin_accel_ram " ] , size = 2 * * 12 , mode = " rw " )
2022-04-17 11:25:48 +02:00
2023-04-16 09:04:21 +02:00
if ( ethernet ) :
# we need the CRG to provide the cd_eth clock: "use refclk_cd as RMII reference clock (provided by user design) (no external clock).
self . ethphy = LiteEthPHYRMII (
clock_pads = self . platform . request ( " eth_clocks " ) ,
pads = self . platform . request ( " eth " ) )
self . add_ethernet ( phy = self . ethphy , data_width = 32 )
print ( f " %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% { self . ethmac . interface . sram . ev . irq } " ) # FIXME HANDLEME
2022-04-17 11:25:48 +02:00
# for testing
2023-04-16 09:04:21 +02:00
if ( False ) :
2022-06-04 09:53:09 +02:00
from nubus_master_tst import PingMaster
self . submodules . pingmaster = PingMaster ( nubus = self . nubus , platform = self . platform )
self . bus . add_slave ( " pingmaster_slv " , self . pingmaster . bus_slv , SoCRegion ( origin = self . mem_map . get ( " pingmaster " , None ) , size = 0x010 , cached = False ) )
self . bus . add_master ( name = " pingmaster_mst " , master = self . pingmaster . bus_mst )
2021-12-21 08:26:30 +01:00
def main ( ) :
2023-04-16 09:04:21 +02:00
parser = argparse . ArgumentParser ( description = " NuBusFPGA " )
2021-12-21 08:26:30 +01:00
parser . add_argument ( " --build " , action = " store_true " , help = " Build bitstream " )
parser . add_argument ( " --variant " , default = " ztex2.13a " , help = " ZTex board variant (default ztex2.13a) " )
parser . add_argument ( " --version " , default = " V1.0 " , help = " NuBusFPGA board version (default V1.0) " )
parser . add_argument ( " --sys-clk-freq " , default = 100e6 , help = " NuBusFPGA system clock (default 100e6 = 100 MHz) " )
2022-02-05 15:32:44 +01:00
parser . add_argument ( " --goblin " , action = " store_true " , help = " add a goblin framebuffer " )
2023-04-16 09:04:21 +02:00
parser . add_argument ( " --hdmi " , action = " store_true " , help = " The framebuffer uses HDMI (default to VGA, required for V1.2) " )
2022-02-05 15:32:44 +01:00
parser . add_argument ( " --goblin-res " , default = " 640x480@60Hz " , help = " Specify the goblin resolution " )
2023-04-16 09:04:21 +02:00
parser . add_argument ( " --ethernet " , action = " store_true " , help = " Add Ethernet (V1.2 w/ custom PMod only) " )
2021-12-21 08:26:30 +01:00
builder_args ( parser )
vivado_build_args ( parser )
args = parser . parse_args ( )
2023-04-16 09:04:21 +02:00
if ( args . ethernet and ( args . version == " V1.0 " ) ) :
print ( " ***** ERROR ***** : Ethernet not supported on V1.0 \n " ) ;
assert ( False )
if ( ( not args . hdmi ) and ( args . version == " V1.2 " ) ) :
print ( " ***** ERROR ***** : VGA not supported on V1.2 \n " ) ;
assert ( False )
2021-12-21 08:26:30 +01:00
soc = NuBusFPGA ( * * soc_core_argdict ( args ) ,
variant = args . variant ,
version = args . version ,
sys_clk_freq = int ( float ( args . sys_clk_freq ) ) ,
2022-02-05 15:32:44 +01:00
goblin = args . goblin ,
2021-12-21 08:26:30 +01:00
hdmi = args . hdmi ,
2023-04-16 09:04:21 +02:00
goblin_res = args . goblin_res ,
ethernet = args . ethernet )
2021-12-21 08:26:30 +01:00
version_for_filename = args . version . replace ( " . " , " _ " )
soc . platform . name + = " _ " + version_for_filename
builder = Builder ( soc , * * builder_argdict ( args ) )
builder . build ( * * vivado_build_argdict ( args ) , run = args . build )
# Generate modified CSR registers definitions/access functions to netbsd_csr.h.
# should be split per-device (and without base) to still work if we have identical devices in different configurations on multiple boards
# now it is split
2022-01-29 11:03:47 +01:00
csr_contents_dict = nubus_to_fpga_export . get_csr_header_split (
regions = soc . csr_regions ,
constants = soc . constants ,
csr_base = soc . mem_regions [ ' csr ' ] . origin )
for name in csr_contents_dict . keys ( ) :
write_to_file ( os . path . join ( " nubusfpga_csr_ {} .h " . format ( name ) ) , csr_contents_dict [ name ] )
2021-12-21 08:26:30 +01:00
if __name__ == " __main__ " :
main ( )