forked from M-Labs/artiq
1
0
Fork 0
artiq/artiq/dashboard/moninj.py

375 lines
13 KiB
Python
Raw Normal View History

import asyncio
import logging
from PyQt5 import QtCore, QtWidgets, QtGui
2015-06-12 17:41:04 +08:00
from artiq.protocols.sync_struct import Subscriber
2017-02-27 00:59:31 +08:00
from artiq.coredevice.moninj import MonInjComm
from artiq.gui.tools import LayoutWidget
2016-03-17 18:18:31 +08:00
from artiq.gui.flowlayout import FlowLayout
logger = logging.getLogger(__name__)
2016-11-04 16:20:46 +08:00
class _TTLWidget(QtWidgets.QFrame):
2017-02-27 00:59:31 +08:00
def __init__(self, channel, set_mode, force_out, title):
QtWidgets.QFrame.__init__(self)
2015-06-06 00:03:30 +08:00
self.channel = channel
2017-02-27 00:59:31 +08:00
self.set_mode = set_mode
2015-06-05 19:11:41 +08:00
self.force_out = force_out
2016-11-04 16:20:46 +08:00
self.setFrameShape(QtWidgets.QFrame.Box)
self.setFrameShadow(QtWidgets.QFrame.Raised)
2015-06-05 19:11:41 +08:00
2016-02-15 07:23:47 +08:00
grid = QtWidgets.QGridLayout()
2016-11-04 16:20:46 +08:00
grid.setContentsMargins(0, 0, 0, 0)
grid.setHorizontalSpacing(0)
grid.setVerticalSpacing(0)
2015-06-05 19:11:41 +08:00
self.setLayout(grid)
2016-02-15 07:23:47 +08:00
label = QtWidgets.QLabel(title)
2015-06-05 19:11:41 +08:00
label.setAlignment(QtCore.Qt.AlignCenter)
2016-11-04 16:20:46 +08:00
label.setSizePolicy(QtWidgets.QSizePolicy.Ignored,
QtWidgets.QSizePolicy.Preferred)
2015-06-05 19:11:41 +08:00
grid.addWidget(label, 1, 1)
self.stack = QtWidgets.QStackedWidget()
grid.addWidget(self.stack, 2, 1)
self.direction = QtWidgets.QLabel()
self.direction.setAlignment(QtCore.Qt.AlignCenter)
self.stack.addWidget(self.direction)
grid_cb = LayoutWidget()
2016-11-04 16:20:46 +08:00
grid_cb.layout.setContentsMargins(0, 0, 0, 0)
grid_cb.layout.setHorizontalSpacing(0)
grid_cb.layout.setVerticalSpacing(0)
self.override = QtWidgets.QToolButton()
self.override.setText("OVR")
self.override.setCheckable(True)
self.override.setToolTip("Override")
grid_cb.addWidget(self.override, 3, 1)
2016-11-04 16:20:46 +08:00
self.level = QtWidgets.QToolButton()
self.level.setText("LVL")
self.level.setCheckable(True)
self.level.setToolTip("Level")
grid_cb.addWidget(self.level, 3, 2)
self.stack.addWidget(grid_cb)
self.value = QtWidgets.QLabel()
self.value.setAlignment(QtCore.Qt.AlignCenter)
grid.addWidget(self.value, 3, 1)
2015-06-06 00:03:30 +08:00
2015-09-02 04:25:20 +08:00
grid.setRowStretch(1, 1)
grid.setRowStretch(2, 0)
grid.setRowStretch(3, 0)
grid.setRowStretch(4, 1)
self.programmatic_change = False
2016-11-04 16:20:46 +08:00
self.override.clicked.connect(self.override_toggled)
self.level.clicked.connect(self.level_toggled)
2015-09-02 04:25:20 +08:00
2017-02-27 00:59:31 +08:00
self.cur_value = False
self.cur_oe = False
self.cur_override = False
self.refresh_display()
2015-06-06 00:03:30 +08:00
def enterEvent(self, event):
self.stack.setCurrentIndex(1)
2016-11-04 16:20:46 +08:00
QtWidgets.QFrame.enterEvent(self, event)
def leaveEvent(self, event):
if not self.override.isChecked():
self.stack.setCurrentIndex(0)
2016-11-04 16:20:46 +08:00
QtWidgets.QFrame.leaveEvent(self, event)
def override_toggled(self, override):
if self.programmatic_change:
return
if override:
if self.level.isChecked():
2017-02-27 00:59:31 +08:00
self.set_mode(self.channel, "1")
else:
2017-02-27 00:59:31 +08:00
self.set_mode(self.channel, "0")
else:
2017-02-27 00:59:31 +08:00
self.set_mode(self.channel, "exp")
def level_toggled(self, level):
if self.programmatic_change:
return
if self.override.isChecked():
if level:
2017-02-27 00:59:31 +08:00
self.set_mode(self.channel, "1")
else:
2017-02-27 00:59:31 +08:00
self.set_mode(self.channel, "0")
2017-02-27 00:59:31 +08:00
def refresh_display(self):
value_s = "1" if self.cur_value else "0"
if self.cur_override:
2015-06-06 00:03:30 +08:00
value_s = "<b>" + value_s + "</b>"
2015-06-05 19:11:41 +08:00
color = " color=\"red\""
else:
color = ""
2016-11-04 16:20:46 +08:00
self.value.setText("<font size=\"5\"{}>{}</font>".format(
2015-06-06 00:03:30 +08:00
color, value_s))
2017-02-27 00:59:31 +08:00
oe = self.cur_oe or self.force_out
2015-06-05 19:11:41 +08:00
direction = "OUT" if oe else "IN"
2016-11-04 16:20:46 +08:00
self.direction.setText("<font size=\"2\">" + direction + "</font>")
self.programmatic_change = True
try:
2017-02-27 00:59:31 +08:00
self.override.setChecked(self.cur_override)
if self.cur_override:
self.stack.setCurrentIndex(1)
2017-02-27 00:59:31 +08:00
self.level.setChecked(self.cur_value)
finally:
self.programmatic_change = False
2015-06-05 19:11:41 +08:00
def sort_key(self):
return self.channel
2015-06-05 19:11:41 +08:00
2016-11-04 16:20:46 +08:00
class _DDSWidget(QtWidgets.QFrame):
2017-02-27 00:59:31 +08:00
def __init__(self, bus_channel, channel, title):
2016-11-04 16:20:46 +08:00
QtWidgets.QFrame.__init__(self)
self.bus_channel = bus_channel
self.channel = channel
2015-06-12 17:41:04 +08:00
2016-11-04 16:20:46 +08:00
self.setFrameShape(QtWidgets.QFrame.Box)
self.setFrameShadow(QtWidgets.QFrame.Raised)
2015-06-12 17:41:04 +08:00
2016-02-15 07:23:47 +08:00
grid = QtWidgets.QGridLayout()
2016-11-04 16:20:46 +08:00
grid.setContentsMargins(0, 0, 0, 0)
grid.setHorizontalSpacing(0)
grid.setVerticalSpacing(0)
2015-06-12 17:41:04 +08:00
self.setLayout(grid)
2016-02-15 07:23:47 +08:00
label = QtWidgets.QLabel(title)
2015-06-12 17:41:04 +08:00
label.setAlignment(QtCore.Qt.AlignCenter)
2016-11-04 16:20:46 +08:00
label.setSizePolicy(QtWidgets.QSizePolicy.Ignored,
QtWidgets.QSizePolicy.Preferred)
2015-06-12 17:41:04 +08:00
grid.addWidget(label, 1, 1)
self.value = QtWidgets.QLabel()
self.value.setAlignment(QtCore.Qt.AlignCenter)
grid.addWidget(self.value, 2, 1, 6, 1)
2015-06-12 17:41:04 +08:00
2015-09-02 04:25:20 +08:00
grid.setRowStretch(1, 1)
grid.setRowStretch(2, 0)
grid.setRowStretch(3, 1)
2017-02-27 00:59:31 +08:00
self.cur_frequency = 0
self.refresh_display()
2015-06-12 17:41:04 +08:00
2017-02-27 00:59:31 +08:00
def refresh_display(self):
2016-11-04 16:20:46 +08:00
self.value.setText("<font size=\"4\">{:.7f}</font><font size=\"2\"> MHz</font>"
2017-02-27 00:59:31 +08:00
.format(self.cur_frequency/1e6))
2015-06-12 17:41:04 +08:00
def sort_key(self):
return (self.bus_channel, self.channel)
2015-06-12 17:41:04 +08:00
2017-02-27 00:59:31 +08:00
TTL_PROBE_LEVEL = 0
TTL_PROBE_OE = 1
TTL_OVERRIDE_ENABLE = 0
TTL_OVERRIDE_LEVEL = 1
TTL_OVERRIDE_OE = 2
class _DeviceManager:
2017-02-27 00:59:31 +08:00
def __init__(self, init):
self.core_addr = None
self.new_core_addr = asyncio.Event()
self.core_connection = None
self.core_connector_task = asyncio.ensure_future(self.core_connector())
self.dds_sysclk = 0
2015-06-05 19:11:41 +08:00
self.ttl_cb = lambda: None
self.ttl_widgets = dict()
2017-02-27 00:59:31 +08:00
self.ttl_widgets_by_channel = dict()
2015-06-12 17:41:04 +08:00
self.dds_cb = lambda: None
self.dds_widgets = dict()
2017-02-27 00:59:31 +08:00
self.dds_widgets_by_channel = dict()
2015-06-05 19:11:41 +08:00
for k, v in init.items():
self[k] = v
def __setitem__(self, k, v):
2015-06-05 19:11:41 +08:00
if k in self.ttl_widgets:
del self[k]
if k in self.dds_widgets:
del self[k]
2015-06-05 19:11:41 +08:00
if not isinstance(v, dict):
return
try:
2015-06-12 17:41:04 +08:00
if v["type"] == "local":
widget = None
2017-02-27 00:59:31 +08:00
if v["module"] == "artiq.coredevice.comm_tcp":
self.core_addr = v["arguments"]["host"]
self.new_core_addr.set()
elif v["module"] == "artiq.coredevice.ttl":
2015-06-12 17:41:04 +08:00
channel = v["arguments"]["channel"]
force_out = v["class"] == "TTLOut"
widget = _TTLWidget(
2017-02-27 00:59:31 +08:00
channel, self.ttl_set_mode, force_out, k)
self.ttl_widgets[k] = widget
2017-02-27 00:59:31 +08:00
self.ttl_widgets_by_channel[channel] = widget
2015-06-12 17:41:04 +08:00
self.ttl_cb()
2017-02-27 00:59:31 +08:00
self.setup_ttl_monitoring(True, channel)
elif (v["module"] == "artiq.coredevice.dds"
and v["class"] == "DDSGroupAD9914"):
self.dds_sysclk = v["arguments"]["sysclk"]
2017-02-27 00:59:31 +08:00
elif (v["module"] == "artiq.coredevice.dds"
and v["class"] == "DDSChannelAD9914"):
bus_channel = v["arguments"]["bus_channel"]
2015-06-12 17:41:04 +08:00
channel = v["arguments"]["channel"]
2017-02-27 00:59:31 +08:00
widget = _DDSWidget(bus_channel, channel, k)
self.dds_widgets[k] = widget
self.dds_widgets_by_channel[(bus_channel, channel)] = widget
2015-06-12 17:41:04 +08:00
self.dds_cb()
2017-02-27 00:59:31 +08:00
self.setup_dds_monitoring(True, bus_channel, channel)
if widget is not None and "comment" in v:
widget.setToolTip(v["comment"])
2015-06-05 19:11:41 +08:00
except KeyError:
pass
def __delitem__(self, k):
if k in self.ttl_widgets:
2017-02-27 00:59:31 +08:00
widget = self.ttl_widgets[k]
self.setup_ttl_monitoring(False, widget.channel)
widget.deleteLater()
del self.ttl_widgets_by_channel[widget.channel]
2015-06-05 19:11:41 +08:00
del self.ttl_widgets[k]
self.ttl_cb()
if k in self.dds_widgets:
2017-02-27 00:59:31 +08:00
widget = self.dds_widgets[k]
self.setup_dds_monitoring(False, widget.bus_channel, widget.channel)
widget.deleteLater()
del self.dds_widgets_by_channel[(widget.bus_channel, widget.channel)]
del self.dds_widgets[k]
self.dds_cb()
2017-02-27 00:59:31 +08:00
def ttl_set_mode(self, channel, mode):
if self.core_connection is not None:
widget = self.ttl_widgets_by_channel[channel]
if mode == "0":
widget.cur_override = True
self.core_connection.inject(channel, TTL_OVERRIDE_LEVEL, 0)
self.core_connection.inject(channel, TTL_OVERRIDE_OE, 1)
self.core_connection.inject(channel, TTL_OVERRIDE_EN, 1)
elif mode == "1":
widget.cur_override = True
self.core_connection.inject(channel, TTL_OVERRIDE_LEVEL, 1)
self.core_connection.inject(channel, TTL_OVERRIDE_OE, 1)
self.core_connection.inject(channel, TTL_OVERRIDE_EN, 1)
elif mode == "exp":
widget.cur_override = False
self.core_connection.inject(channel, TTL_OVERRIDE_EN, 0)
else:
raise ValueError
def setup_ttl_monitoring(self, enable, channel):
if self.core_connection is not None:
self.core_connection.monitor(enable, channel, TTL_PROBE_LEVEL)
self.core_connection.monitor(enable, channel, TTL_PROBE_OE)
def setup_dds_monitoring(self, enable, bus_channel, channel):
if self.core_connection is not None:
self.core_connection.monitor(enable, bus_channel, channel)
def monitor_cb(self, channel, probe, value):
if channel in self.ttl_widgets_by_channel:
widget = self.ttl_widgets_by_channel[channel]
if probe == TTL_PROBE_LEVEL:
widget.cur_level = bool(value)
elif probe == TTL_PROBE_OE:
widget.cur_oe = bool(value)
widget.refresh_display()
if (bus_channel, channel) in self.dds_widgets_by_channel:
widget = self.dds_widgets_by_channel[(channel, probe)]
widget.cur_frequency = value*self.dds_sysclk/2**32
widget.refresh_display()
def injection_status_cb(self, channel, override, value):
if channel in self.ttl_widgets_by_channel:
self.ttl_widgets_by_channel[channel].cur_override = bool(value)
async def core_connector(self):
while True:
await self.new_core_addr.wait()
self.new_core_addr.clear()
if self.core_connection is not None:
await self.core_connection.close()
self.core_connection = None
new_core_connection = MonInjComm(self.monitor_cb, self.injection_status_cb,
lambda: logger.error("lost connection to core device moninj"))
try:
await new_core_connection.connect(self.core_addr, 1383)
except:
logger.error("failed to connect to core device moninj", exc_info=True)
else:
self.core_connection = new_core_connection
for ttl_channel in self.ttl_widgets_by_channel.keys():
self.setup_ttl_monitoring(True, ttl_channel)
for bus_channel, channel in self.dds_widgets_by_channel.keys():
self.setup_dds_monitoring(True, bus_channel, channel)
async def close(self):
self.core_connector_task.cancel()
try:
2017-02-27 00:59:31 +08:00
await asyncio.wait_for(self.core_connector_task, None)
except asyncio.CancelledError:
pass
if self.core_connection is not None:
await self.core_connection.close()
class _MonInjDock(QtWidgets.QDockWidget):
2015-06-12 17:41:04 +08:00
def __init__(self, name):
QtWidgets.QDockWidget.__init__(self, name)
2016-02-14 20:46:15 +08:00
self.setObjectName(name)
self.setFeatures(QtWidgets.QDockWidget.DockWidgetMovable |
QtWidgets.QDockWidget.DockWidgetFloatable)
2015-06-05 19:11:41 +08:00
2015-06-12 17:41:04 +08:00
def layout_widgets(self, widgets):
2016-02-15 07:57:15 +08:00
scroll_area = QtWidgets.QScrollArea()
self.setWidget(scroll_area)
2016-03-17 18:18:31 +08:00
grid = FlowLayout()
2016-02-15 07:57:15 +08:00
grid_widget = QtWidgets.QWidget()
grid_widget.setLayout(grid)
for _, w in sorted(widgets, key=lambda i: i[1].sort_key()):
2016-03-17 18:18:31 +08:00
grid.addWidget(w)
2016-02-15 07:57:15 +08:00
2016-03-17 18:18:31 +08:00
scroll_area.setWidgetResizable(True)
2016-02-15 07:57:15 +08:00
scroll_area.setWidget(grid_widget)
2015-06-12 17:41:04 +08:00
2017-02-27 00:59:31 +08:00
class MonInj:
2015-06-12 17:41:04 +08:00
def __init__(self):
self.ttl_dock = _MonInjDock("TTL")
self.dds_dock = _MonInjDock("DDS")
self.subscriber = Subscriber("devices", self.init_devices)
self.dm = None
2015-06-12 17:41:04 +08:00
2015-10-03 19:28:57 +08:00
async def start(self, server, port):
await self.subscriber.connect(server, port)
2015-10-03 19:28:57 +08:00
async def stop(self):
await self.subscriber.close()
2017-02-27 00:59:31 +08:00
if self.dm is not None:
await self.dm.close()
def init_devices(self, d):
2017-02-27 00:59:31 +08:00
self.dm = _DeviceManager(d)
2015-06-12 17:41:04 +08:00
self.dm.ttl_cb = lambda: self.ttl_dock.layout_widgets(
self.dm.ttl_widgets.items())
self.dm.dds_cb = lambda: self.dds_dock.layout_widgets(
self.dm.dds_widgets.items())
self.dm.ttl_cb()
self.dm.dds_cb()
return self.dm