aboutsummaryrefslogtreecommitdiffstats
path: root/mpm/python
diff options
context:
space:
mode:
Diffstat (limited to 'mpm/python')
-rw-r--r--mpm/python/usrp_mpm/nijesdcore.py134
1 files changed, 101 insertions, 33 deletions
diff --git a/mpm/python/usrp_mpm/nijesdcore.py b/mpm/python/usrp_mpm/nijesdcore.py
index c3f1342db..00c557fdd 100644
--- a/mpm/python/usrp_mpm/nijesdcore.py
+++ b/mpm/python/usrp_mpm/nijesdcore.py
@@ -31,22 +31,34 @@ class NIMgJESDCore(object):
Arguments:
regs -- regs class to use for peek/poke
"""
-
+
+ DB_ID = 0x0630
+ MGT_QPLL_CONTROL = 0x2000
+ MGT_PLL_POWER_DOWN_CONTROL = 0x200C
+ MGT_TX_RESET_CONTROL = 0x2020
+ MGT_RX_RESET_CONTROL = 0x2024
MGT_RECEIVER_CONTROL = 0x2040
MGT_RX_DESCRAMBLER_CONTROL = 0x2050
MGT_TRANSMITTER_CONTROL = 0x2060
MGT_TX_TRANSCEIVER_CONTROL = 0x2064
MGT_TX_SCRAMBLER_CONTROL = 0x2068
+ LMK_SYNC_CONTROL = 0x206C
SYSREF_CAPTURE_CONTROL = 0x2078
JESD_SIGNATURE_REG = 0x2100
JESD_REVISION_REG = 0x2104
-
-
+
+
def __init__(self, regs, slot_idx=0):
self.regs = regs
self.log = get_logger("NIMgJESDCore-{}".format(slot_idx))
assert hasattr(self.regs, 'peek32')
assert hasattr(self.regs, 'poke32')
+ # FutureWork: The following are constants for the Magnesium board. These need
+ # to change to variables to support other interfaces.
+ self.qplls_used = 1
+ self.cplls_used = 0
+ # Number of FPGA clock cycles per LMFC period.
+ self.lmfc_divider = 20
def unreset_qpll(self):
# new_val = self.regs.peek32(0x0) & ~0x8
@@ -63,10 +75,22 @@ class NIMgJESDCore(object):
#if self.regs.peek32(JESD_REVISION_REG) != 0xFF
#error here for date revision mismatch
self.log.trace("JESD Core build code: {0}".format(hex(self.regs.peek32(self.JESD_REVISION_REG))))
- self.log.trace("DB Slot #: {}".format( (self.regs.peek32(0x630) & 0x10000) >> 16 ))
- self.log.trace("DB PID: {:X}".format( self.regs.peek32(0x630) & 0xFFFF ))
+ self.log.trace("DB Slot #: {}".format( (self.regs.peek32(self.DB_ID) & 0x10000) >> 16 ))
+ self.log.trace("DB PID: {:X}" .format( self.regs.peek32(self.DB_ID) & 0xFFFF ))
return True
+ def reset(self):
+ """
+ Reset to the core. Places the PLLs, TX MGTs and RX MGTs (along with the glue
+ logic) in reset. Also disables the SYSREF sampler.
+ """
+ self.log.trace("Resetting the JESD204B FPGA core(s)...")
+ self._gt_reset('tx', reset_only=True)
+ self._gt_reset('rx', reset_only=True)
+ self._gt_pll_lock_control(self.qplls_used, self.cplls_used, reset_only=True)
+ # Disable SYSREF Sampler
+ self.enable_lmfc(False)
+
def init_deframer(self):
" Initialize deframer "
self.log.trace("Initializing deframer...")
@@ -121,21 +145,26 @@ class NIMgJESDCore(object):
def init(self):
"""
- Initializes to the core. Needs to happen after the clock signal is ready.
+ Initializes the core. Must happen after the reference clock is stable.
"""
- self.log.trace("Initializing core...")
- self._gt_pll_power_control()
+ self.log.trace("Initializing JESD204B FPGA core(s)...")
+ self._gt_pll_power_control(self.qplls_used, self.cplls_used)
self._gt_reset('tx', reset_only=True)
self._gt_reset('rx', reset_only=True)
- self._gt_pll_lock_control()
+ self._gt_pll_lock_control(self.qplls_used, self.cplls_used, reset_only=False)
# Disable SYSREF Sampler
- self.regs.poke32(self.SYSREF_CAPTURE_CONTROL, 0x9800040)
+ self.enable_lmfc(False)
- def enable_lmfc(self):
+ def enable_lmfc(self, enable=False):
"""
- Enable LMFC generator in FPGA. This step is woefully incomplete, but this call will work for now.
+ Enable/disable LMFC generator in FPGA.
"""
- self.regs.poke32(self.SYSREF_CAPTURE_CONTROL, 0x9800000)
+ disable_bit = 0b1
+ if enable:
+ disable_bit = 0b0
+ reg_val = ((self.lmfc_divider-1) << 23) | (disable_bit << 6)
+ self.log.trace("Setting SYSREF Capture reg: 0x{:08X}".format(reg_val))
+ self.regs.poke32(self.SYSREF_CAPTURE_CONTROL, reg_val)
def send_sysref_pulse(self):
"""
@@ -143,12 +172,12 @@ class NIMgJESDCore(object):
Note: SYSREFs must be enabled on LMK separately beforehand.
"""
self.log.trace("Sending SYSREF pulse...")
- self.regs.poke32(0x206C, 0x40000000) # Bit 30. Self-clears.
+ self.regs.poke32(self.LMK_SYNC_CONTROL, 0b1 << 30) # Bit 30. Self-clears.
def _gt_reset(self, tx_or_rx, reset_only=False):
" Put MGTs into reset. Optionally unresets and enables them "
assert tx_or_rx.lower() in ('rx', 'tx')
- mgt_reg = {'tx': 0x2020, 'rx': 0x2024}[tx_or_rx]
+ mgt_reg = {'tx': self.MGT_TX_RESET_CONTROL, 'rx': self.MGT_RX_RESET_CONTROL}[tx_or_rx]
self.log.trace("Resetting %s MGTs..." % tx_or_rx.upper())
self.regs.poke32(mgt_reg, 0x10)
if not reset_only:
@@ -165,28 +194,67 @@ class NIMgJESDCore(object):
))
return True
- def _gt_pll_power_control(self):
+ def _gt_pll_power_control(self, qplls = 0, cplls = 0):
" Power down unused CPLLs and QPLLs "
+ assert qplls in range(4+1) # valid is 0 - 4
+ assert cplls in range(8+1) # valid is 0 - 8
self.log.trace("Powering down unused CPLLs and QPLLs...")
- self.regs.poke32(0x200C, 0xFFF000E)
+ self.log.trace("Using {} CPLLs and {} QPLLs!".format(cplls, qplls))
+ reg_val = 0xFFFF000F
+ reg_val_on = 0x0
+ # Power down state is when the corresponding bit is set. For the PLLs we wish to
+ # use, clear those bits.
+ for x in range(qplls):
+ reg_val_on = reg_val_on | 0x1 << x # QPLL bits are 0-3
+ for y in range(16, 16 + cplls):
+ reg_val_on = reg_val_on | 0x1 << y # CPLL bits are 16-23, others are reserved
+ reg_val = reg_val ^ reg_val_on
+ self.regs.poke32(self.MGT_PLL_POWER_DOWN_CONTROL, reg_val)
- def _gt_pll_lock_control(self):
+ def _gt_pll_lock_control(self, qplls = 0, cplls = 0, reset_only=False):
"""
Turn on the PLLs we're using, and make sure lock bits are set.
+ QPLL bitfield mapping: the following nibble is repeated for each QPLL. For
+ example, QPLL0 get bits 0-3, QPLL1 get bits 4-7, etc.
+ [0] = reset
+ [1] = locked
+ [2] = unlocked sticky
+ [3] = ref clock lost sticky
+ ...
+ [16] = sticky reset (strobe)
"""
- self.regs.poke32(0x2000, 0x1111) # Reset QPLLs
- self.regs.poke32(0x2000, 0x1110) # Unreset the ones we're using
- time.sleep(0.002) # alternatively, poll on the locked bit below
- self.regs.poke32(0x2000, 0x10000) # Clear all QPLL sticky bits
- rb = self.regs.peek32(0x2000) # Read QPLL locked and no unlocked stickies.
- self.log.trace("Reading QPLL lock bit: {0}".format(hex(rb & 0xF)))
- # Error: GT PLL failed to lock.
- if rb & 0xF != 0x2:
- raise Exception("GT PLL failed to lock!")
-
- def reset_mykonos(self):
- " Toggle reset line on Mykonos "
- self.regs.poke32(0x0008, 0) # Active low reset
- time.sleep(0.001)
- self.regs.poke32(0x0008, 1) # No longer in reset
+ # FutureWork: CPLLs are NOT supported yet!!!
+ assert cplls == 0
+ assert qplls in range(4+1) # valid is 0 - 4
+
+ # Reset QPLLs.
+ reg_val = 0x1111 # by default assert all resets
+ self.regs.poke32(self.MGT_QPLL_CONTROL, reg_val)
+
+ # Unreset the PLLs in use and check for lock.
+ if not reset_only:
+ if qplls > 0:
+ # Unreset only the QPLLs we are using.
+ reg_val_on = 0x0
+ for nibble in range(qplls):
+ reg_val_on = reg_val_on | 0x1 << nibble*4
+ reg_val = reg_val ^ reg_val_on
+ self.regs.poke32(self.MGT_QPLL_CONTROL, reg_val)
+
+ # Check for lock a short time later.
+ time.sleep(0.010)
+ # Clear all QPLL sticky bits
+ self.regs.poke32(self.MGT_QPLL_CONTROL, 0b1 << 16)
+ rb = self.regs.peek32(self.MGT_QPLL_CONTROL)
+ self.log.trace("Reading QPLL status register: {:04X}".format(rb & 0xFFFF))
+ # Check for lock on active quads only.
+ rb_mask = 0x0
+ locked_val = 0x0
+ for nibble in range(qplls):
+ if (rb & (0xF << nibble*4)) != (0x2 << nibble*4):
+ self.log.warning("GT QPLL {} failed to lock!".format(nibble))
+ locked_val = locked_val | 0x2 << nibble*4
+ rb_mask = rb_mask | 0xF << nibble*4
+ if (rb & rb_mask) != locked_val:
+ raise Exception("One or more GT QPLLs failed to lock!")