aboutsummaryrefslogtreecommitdiffstats
path: root/mpm/python/usrp_mpm/chips/ds125df410.py
diff options
context:
space:
mode:
Diffstat (limited to 'mpm/python/usrp_mpm/chips/ds125df410.py')
-rw-r--r--mpm/python/usrp_mpm/chips/ds125df410.py107
1 files changed, 107 insertions, 0 deletions
diff --git a/mpm/python/usrp_mpm/chips/ds125df410.py b/mpm/python/usrp_mpm/chips/ds125df410.py
new file mode 100644
index 000000000..5cea7e67a
--- /dev/null
+++ b/mpm/python/usrp_mpm/chips/ds125df410.py
@@ -0,0 +1,107 @@
+#
+# Copyright 2018 Ettus Research, a National Instruments Company
+#
+# SPDX-License-Identifier: GPL-3.0-or-later
+#
+"""
+DS125DF410 driver class
+
+For use with TI's retimer chip
+"""
+
+import math
+from builtins import object
+from usrp_mpm.mpmlog import get_logger
+
+LINE_RATE_PRESETS = {'Ethernet': 0xF6, 'CPRI1': 0x36, 'CPRI2': 0x46}
+ALL_CHANS = range(4)
+
+
+class DS125DF410(object):
+ """
+ Driver class for DS125DF410 access.
+
+ Inputs:
+ regs_iface : regs_iface bus driver to access device
+ parent_log : logger of parent
+ """
+
+ # (deemphasis, swing)
+ DRIVER_PRESETS = {}
+
+ def __init__(self, regs_iface, parent_log=None):
+ self.log = \
+ parent_log.getChild("DS125DF410") if parent_log is not None \
+ else get_logger("DS125DF410")
+ self.regs_iface = regs_iface
+ # Set channel select register to control set
+ self.regs_iface.poke8(0xFF, 0)
+ # Probe chip ID
+ chip_id = self.regs_iface.peek8(0x01)
+ assert chip_id == 0xd1
+ self.log.debug("Probed DS125DF410 retimer")
+ for chan in ALL_CHANS:
+ self._set_chan_select(chan)
+ # Reset channel registers
+ self.regs_iface.poke8(0x00, 0x04)
+ # Enable DFE mode
+ self.regs_iface.poke8(0x1E, 0xE1)
+ self.regs_iface.poke8(0x31, 0x40)
+
+ def _rmw(self, addr, data, mask):
+ """ Read, modify, write """
+ base = self.regs_iface.peek8(addr) & ~mask
+ data = (data & mask) | base
+ self.regs_iface.poke8(addr, data)
+
+ def _set_chan_select(self, chan):
+ """
+ Channel select
+ """
+ assert chan in ALL_CHANS
+ self.regs_iface.poke8(0xFF, chan + 4)
+
+ def set_rate_preset(self, preset, channels=None):
+ """
+ Set rate preset
+ """
+ channels = channels or ALL_CHANS
+ assert preset in LINE_RATE_PRESETS.keys()
+ for chan in channels:
+ self._set_chan_select(chan)
+ self.regs_iface.poke8(0x2F, LINE_RATE_PRESETS[preset])
+
+ def set_rate(self, rate, channels=None):
+ """
+ Set rate
+ """
+ channels = channels or ALL_CHANS
+ self.log.trace("Writing custom line rate {}".format(rate))
+ ppm_val = int(math.ceil(rate*1280.0))
+ assert ppm_val <= 0x7FFF
+ for chan in channels:
+ assert chan in ALL_CHANS
+ self._set_chan_select(chan)
+ # Set VCO divider to 1
+ self.regs_iface.poke8(0x2F, 0xC6)
+ # Set frequency range detection
+ self.regs_iface.poke8(0x60, (ppm_val & 0x00FF))
+ self.regs_iface.poke8(0x62, (ppm_val & 0x00FF))
+ self.regs_iface.poke8(0x61, 0x80 | ((ppm_val >> 8) & 0x00FF))
+ self.regs_iface.poke8(0x63, 0x80 | ((ppm_val >> 8) & 0x00FF))
+ # Set VCO tolerance range to max
+ self.regs_iface.poke8(0x64, 0xFF)
+
+ def set_driver_preset(self, preset, channels=None):
+ """
+ Set driver preset
+ """
+ channels = channels or ALL_CHANS
+ assert preset in self.DRIVER_PRESETS.keys()
+ self.log.trace("Setting retimer's driver for " + preset + " preset")
+ deemphasis, swing = self.DRIVER_PRESETS[preset]
+ for chan in channels:
+ self._set_chan_select(chan)
+ self._rmw(0x15, deemphasis, 0x47)
+ self._rmw(0x2D, swing, 0x07)
+