324 lines
12 KiB
Python
324 lines
12 KiB
Python
# This file is part of Fast Servo Software Package.
|
|
#
|
|
# Copyright (C) 2023 Jakub Matyas
|
|
# Warsaw University of Technology <jakubk.m@gmail.com>
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program 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 General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
import os
|
|
|
|
from migen import *
|
|
from misoc.interconnect import csr_bus
|
|
from misoc.interconnect.csr import AutoCSR, CSRStorage
|
|
|
|
from fast_servo.gateware.cores.adc import ADC, AUX_ADC_CTRL
|
|
from fast_servo.gateware.cores.dac import AUX_DAC_CTRL, DAC
|
|
from fast_servo.gateware.cores.pitaya_ps import Axi2Sys, Sys2CSR, SysCDC, SysInterconnect
|
|
from fast_servo.gateware.cores.ps7 import PS7
|
|
from fast_servo.gateware.cores.spi_phy import SpiInterface, SpiPhy
|
|
|
|
|
|
class CRG(Module):
|
|
def __init__(self, platform, dco_freq=250e6):
|
|
self.ps_rst = Signal()
|
|
self.locked = Signal()
|
|
|
|
dco_clk = platform.request("adc_dco_clk")
|
|
dco_clk_buf = Signal()
|
|
self.specials += Instance(
|
|
"IBUFGDS", i_I=dco_clk.p, i_IB=dco_clk.n, o_O=dco_clk_buf
|
|
)
|
|
|
|
self.clock_domains.cd_sys = ClockDomain()
|
|
self.clock_domains.cd_sys_45_degree = ClockDomain()
|
|
self.clock_domains.cd_sys_double = ClockDomain()
|
|
self.clock_domains.cd_idelay = ClockDomain()
|
|
|
|
clk_feedback = Signal()
|
|
clk_feedback_buf = Signal()
|
|
|
|
clk_sys = Signal()
|
|
clk_sys_45_degree = Signal()
|
|
clk_sys_double = Signal()
|
|
clk_idelay = Signal()
|
|
|
|
self.ddr_clk_phase_shift_en = Signal()
|
|
self.ddr_clk_phase_incdec = Signal()
|
|
self.mmcm_ps_psdone = Signal()
|
|
|
|
si5340_nlol = platform.request("si5340_nlol")
|
|
si5340_nlol_buf = Signal()
|
|
self.specials += Instance("IBUF", i_I=si5340_nlol, o_O=si5340_nlol_buf)
|
|
|
|
platform.add_period_constraint(dco_clk.p, 1e9 / dco_freq)
|
|
self.specials += [
|
|
Instance(
|
|
"MMCME2_ADV",
|
|
p_BANDWIDTH="OPTIMIZED",
|
|
p_DIVCLK_DIVIDE=1,
|
|
p_CLKFBOUT_PHASE=0.0,
|
|
p_CLKFBOUT_MULT_F=4, # VCO @ 1000 MHz
|
|
p_CLKIN1_PERIOD=(1e9 / dco_freq),
|
|
p_REF_JITTER1=0.06, # From LTC2195 Datasheet
|
|
p_STARTUP_WAIT="FALSE",
|
|
i_CLKIN1=dco_clk_buf,
|
|
i_PWRDWN=0,
|
|
i_RST=self.ps_rst | ~si5340_nlol_buf,
|
|
i_CLKFBIN=clk_feedback_buf,
|
|
o_CLKFBOUT=clk_feedback,
|
|
|
|
p_CLKOUT0_USE_FINE_PS="True",
|
|
p_CLKOUT0_DIVIDE_F=8,
|
|
p_CLKOUT0_PHASE=45.0,
|
|
p_CLKOUT0_DUTY_CYCLE=0.5,
|
|
o_CLKOUT0=clk_sys_45_degree, # 1000MHz / 8 -> 125MHz
|
|
o_LOCKED=self.locked,
|
|
|
|
p_CLKOUT1_DIVIDE=8,
|
|
p_CLKOUT1_PHASE=0.0,
|
|
p_CLKOUT1_DUTY_CYCLE=0.5,
|
|
o_CLKOUT1=clk_sys, # 1000MHz / 8 -> 120MHz
|
|
|
|
p_CLKOUT2_DIVIDE=4,
|
|
p_CLKOUT2_PHASE=0.0,
|
|
p_CLKOUT2_DUTY_CYCLE=0.5,
|
|
o_CLKOUT2=clk_sys_double, # 1000MHz / 4 -> 250MHz
|
|
|
|
p_CLKOUT3_DIVIDE=5,
|
|
p_CLKOUT3_PHASE=0.0,
|
|
p_CLKOUT3_DUTY_CYCLE=0.5,
|
|
o_CLKOUT3=clk_idelay, # 1000MHz / 5 -> 200MHz
|
|
|
|
i_PSCLK=ClockSignal(),
|
|
i_PSEN=self.ddr_clk_phase_shift_en,
|
|
i_PSINCDEC=self.ddr_clk_phase_incdec,
|
|
o_PSDONE=self.mmcm_ps_psdone,
|
|
)
|
|
]
|
|
|
|
self.specials += Instance("BUFG", i_I=clk_feedback, o_O=clk_feedback_buf)
|
|
self.specials += Instance("BUFG", i_I=clk_sys, o_O=self.cd_sys.clk)
|
|
self.specials += Instance("BUFG", i_I=clk_sys_45_degree, o_O=self.cd_sys_45_degree.clk)
|
|
self.specials += Instance("BUFG", i_I=clk_sys_double, o_O=self.cd_sys_double.clk)
|
|
self.specials += Instance("BUFG", i_I=clk_idelay, o_O=self.cd_idelay.clk)
|
|
|
|
# Ignore sys_clk to pll clkin path created by SoC's rst.
|
|
platform.add_false_path_constraints(self.cd_sys.clk, dco_clk)
|
|
|
|
self.specials += Instance("FD", p_INIT=1, i_D=~self.locked, i_C=self.cd_sys.clk, o_Q=self.cd_sys.rst)
|
|
|
|
|
|
class BaseSoC(PS7, AutoCSR):
|
|
def __init__(self, platform, passthrouh=False):
|
|
PS7.__init__(self, platform)
|
|
|
|
# TODO:
|
|
# LINIEN SPECIFIC csr_map - in the future should be moved
|
|
# to csr_devices list
|
|
self.csr_map = {
|
|
"adc": 9,
|
|
"fp_led0": 10,
|
|
"fp_led1": 11,
|
|
"fp_led2": 12,
|
|
"fp_led3": 13,
|
|
"dac": 14,
|
|
"adc_aux_ctrl": 15,
|
|
"dac_aux_ctrl": 16,
|
|
}
|
|
self.soc_name = "FastServo"
|
|
self.interconnect_slaves = []
|
|
self.csr_devices = []
|
|
|
|
self.platform = platform
|
|
|
|
self.submodules.crg = CRG(platform)
|
|
|
|
# self.comb += self.afe_ctrl.storage[4].eq(self.crg.mmcm_rst)
|
|
# self.comb += self.afe_ctrl.storage[5].eq(self.crg.ddr_clk_phase_shift_en)
|
|
# self.comb += self.afe_ctrl.storage[6].eq(self.crg.ddr_clk_phase_incdec)
|
|
|
|
# # # AXI to system bus bridge
|
|
self.submodules.axi2sys = Axi2Sys()
|
|
self.submodules.sys2csr = Sys2CSR()
|
|
self.submodules.syscdc = SysCDC()
|
|
self.add_axi_gp_master(self.axi2sys.axi)
|
|
|
|
self.comb += [
|
|
self.axi2sys.axi.aclk.eq(ClockSignal("sys")),
|
|
self.axi2sys.axi.arstn.eq(self.frstn),
|
|
self.syscdc.target.connect(self.sys2csr.sys),
|
|
]
|
|
|
|
# ETH LEDS
|
|
self.comb += [
|
|
platform.request("eth_led", 0).eq(platform.request("from_eth_phy", 0)),
|
|
platform.request("eth_led", 1).eq(platform.request("from_eth_phy", 1)),
|
|
]
|
|
|
|
# I2C0 to Si5340 on Fast Servo
|
|
self.add_i2c_emio(platform, "ps7_i2c", 0)
|
|
|
|
# SPI0 - interface to main ADC and auxiliary ADC
|
|
self.add_spi_interface(platform, SpiInterface.ADC)
|
|
|
|
# SPI1 - interface to main DAC and auxiliary DAC
|
|
self.add_spi_interface(platform, SpiInterface.DAC)
|
|
|
|
# self.add_main_adc(platform)
|
|
self.submodules.adc = ADC(platform)
|
|
self.csr_devices.append("adc")
|
|
# platform.add_false_path_constraints(self.crg.cd_sys.clk, self.adc.crg.cd_dco2d.clk)
|
|
|
|
# self.add_main_dac(platform)
|
|
self.submodules.dac = DAC(platform, [self.crg.ddr_clk_phase_shift_en, self.crg.ddr_clk_phase_incdec,])
|
|
self.csr_devices.append("dac")
|
|
|
|
# DEBUG
|
|
if passthrouh:
|
|
DAC_DATA_WIDTH = 14
|
|
for ch in range(2):
|
|
saturate = Signal()
|
|
adc_signal = self.adc.data_out[ch]
|
|
self.comb += [
|
|
saturate.eq(adc_signal[-3:] != Replicate(adc_signal[-1], 3)),
|
|
self.dac.data_in[ch].eq(Mux(saturate,
|
|
Cat(Replicate(~adc_signal[-1], DAC_DATA_WIDTH-1), adc_signal[-1]),
|
|
adc_signal[:-2]))
|
|
]
|
|
|
|
si_5340_nrst = platform.request("nrst")
|
|
self.comb += si_5340_nrst.eq(1)
|
|
|
|
for i in range(4):
|
|
led_pin = platform.request("fp_led", i)
|
|
setattr(self.submodules, f"fp_led{i}", LED(led_pin))
|
|
self.csr_devices.append(f"fp_led{i}")
|
|
|
|
self.submodules.adc_aux_ctrl = AUX_ADC_CTRL(platform)
|
|
self.csr_devices.append("adc_aux_ctrl")
|
|
|
|
self.submodules.dac_aux_ctrl = AUX_DAC_CTRL(platform)
|
|
self.csr_devices.append("dac_aux_ctrl")
|
|
|
|
def add_spi_interface(self, platform, spi_type):
|
|
assert isinstance(spi_type, SpiInterface)
|
|
n = spi_type.value
|
|
|
|
ps7_spi_pads = platform.request("spi", spi_type.value)
|
|
spi_phy = SpiPhy(spi_type, ps7_spi_pads)
|
|
self.submodules += spi_phy
|
|
|
|
ps7_config_spi = {
|
|
f"PCW_SPI{n}_GRP_SS0_ENABLE" : "1",
|
|
f"PCW_SPI{n}_GRP_SS0_IO" : "EMIO",
|
|
f"PCW_SPI{n}_GRP_SS1_ENABLE" : "1",
|
|
f"PCW_SPI{n}_GRP_SS1_IO" : "EMIO",
|
|
f"PCW_SPI{n}_GRP_SS2_ENABLE" : "1",
|
|
f"PCW_SPI{n}_GRP_SS2_IO" : "EMIO",
|
|
f"PCW_SPI{n}_PERIPHERAL_ENABLE" : "1",
|
|
f"PCW_SPI{n}_SPI{n}_IO" : "EMIO",
|
|
f"PCW_SPI_PERIPHERAL_CLKSRC" : "IO PLL",
|
|
f"PCW_SPI_PERIPHERAL_DIVISOR0" : "10",
|
|
f"PCW_SPI_PERIPHERAL_FREQMHZ" : "166.666666",
|
|
}
|
|
self.add_ps7_config(ps7_config_spi)
|
|
|
|
self.cpu_params.update({
|
|
f"o_SPI{n}_MISO_O" : spi_phy.ps_miso_o,
|
|
f"o_SPI{n}_MISO_T" : spi_phy.ps_miso_t,
|
|
f"o_SPI{n}_MOSI_O" : spi_phy.ps_mosi_o,
|
|
f"o_SPI{n}_MOSI_T" : spi_phy.ps_mosi_t,
|
|
f"o_SPI{n}_SCLK_O" : spi_phy.ps_sclk_o,
|
|
f"o_SPI{n}_SCLK_T" : spi_phy.ps_sclk_t,
|
|
|
|
f"i_SPI{n}_SCLK_I" : spi_phy.ps_sclk_i,
|
|
f"i_SPI{n}_MOSI_I" : spi_phy.ps_mosi_i,
|
|
f"i_SPI{n}_MISO_I" : spi_phy.ps_miso_i,
|
|
f"i_SPI{n}_SS_I" : spi_phy.ps_ss_i,
|
|
f"o_SPI{n}_SS_O" : spi_phy.ps_ss[0],
|
|
f"o_SPI{n}_SS1_O" : spi_phy.ps_ss[1],
|
|
f"o_SPI{n}_SS2_O" : spi_phy.ps_ss[2],
|
|
f"o_SPI{n}_SS_T" : spi_phy.ps_ss_t,
|
|
})
|
|
|
|
def add_interconnect_slave(self, slave):
|
|
self.interconnect_slaves.append(slave)
|
|
|
|
def get_csr_dev_address(self, name, memory):
|
|
# TODO: switch to MiSoC-like address retriving from
|
|
# the list of CSR devices
|
|
if memory is not None:
|
|
name = name + "_" + memory.name_override
|
|
try:
|
|
return self.csr_map[name]
|
|
except KeyError:
|
|
return None
|
|
|
|
def soc_finalize(self):
|
|
# Overload this method to customize SystemInterconnect
|
|
# and csrbanks - especially useful in Linien
|
|
self.add_interconnect_slave(self.syscdc.source)
|
|
self.submodules.interconnect = SysInterconnect(
|
|
self.axi2sys.sys,
|
|
*self.interconnect_slaves
|
|
)
|
|
self.submodules.csrbanks = csr_bus.CSRBankArray(self,
|
|
self.get_csr_dev_address)
|
|
self.submodules.csrcon = csr_bus.Interconnect(
|
|
self.sys2csr.csr, self.csrbanks.get_buses()
|
|
)
|
|
|
|
def do_finalize(self):
|
|
self.soc_finalize()
|
|
PS7.do_finalize(self)
|
|
|
|
def build(self, *args, **kwargs):
|
|
self.platform.build(self, *args, **kwargs)
|
|
|
|
class LED(Module, AutoCSR):
|
|
def __init__(self, led):
|
|
|
|
self.led_out = CSRStorage(1)
|
|
|
|
self.comb += led.eq(self.led_out.storage)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
import subprocess
|
|
from fast_servo.gateware.fast_servo_platform import Platform
|
|
import argparse
|
|
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("--debug", action="store_true", default=False, help="Hardwire ADC data to DAC")
|
|
args = parser.parse_args()
|
|
|
|
|
|
root_path = os.getcwd()
|
|
platform = Platform()
|
|
fast_servo = BaseSoC(platform, passthrouh=args.debug)
|
|
|
|
build_dir = "builds/fast_servo_gw_debug" if args.debug else"builds/fast_servo_gw"
|
|
build_name = "top"
|
|
fast_servo.build(build_dir=build_dir, build_name=build_name, run=True)
|
|
os.chdir(os.path.join(root_path, build_dir))
|
|
with open(f"{build_name}.bif", "w") as f:
|
|
f.write(f"all:\n{{\n\t{build_name}.bit\n}}")
|
|
|
|
cmd = f"bootgen -image {build_name}.bif -arch zynq -process_bitstream bin -w on".split(" ")
|
|
subprocess.run(cmd)
|
|
|
|
|
|
|