-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathgpib_device_manager.py
331 lines (291 loc) · 14.3 KB
/
gpib_device_manager.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
# Copyright (C) 2008 Matthew Neeley
#
# 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 2 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 <http://www.gnu.org/licenses/>.
# Changelog
# 1.3 Created serverConnected() to identify devices on a given server once
# it has completed its registration with the LabRAD manager. Before this
# device registration would fail if the server had a custom IDN handling
# function because this setting could not be properly accessed through the
# LabRAD manager at the time of execution.
from twisted.internet.defer import DeferredList, DeferredLock
from twisted.internet.reactor import callLater
from labrad.server import (LabradServer, setting,
inlineCallbacks, returnValue)
from labrad.units import Unit,Value
"""
### BEGIN NODE INFO
[info]
name = GPIB Device Manager
version = 1.3
description = Manages discovery and lookup of GPIB devices
[startup]
cmdline = %PYTHON% %FILE%
timeout = 20
[shutdown]
message = 987654321
timeout = 5
### END NODE INFO
"""
UNKNOWN = '<unknown>'
def parseIDNResponse(s):
"""Parse the response from *IDN? to get mfr and model info."""
mfr, model, ver, rev = s.split(',')
return mfr.strip() + ' ' + model.strip()
class GPIBDeviceManager(LabradServer):
"""Manages autodetection and identification of GPIB devices.
The device manager listens for "GPIB Device Connect" and
"GPIB Device Disconnect" messages coming from GPIB bus servers.
It attempts to identify the connected devices and forward the
messages on to servers interested in particular devices. For
devices that cannot be identified by *IDN? in the usual way,
servers can register an identification setting to be called
by the device manager to properly identify the device.
"""
name = 'GPIB Device Manager'
@inlineCallbacks
def initServer(self):
"""Initialize the server after connecting to LabRAD."""
self.knownDevices = {} # maps (server, channel) to (name, idn)
self.deviceServers = {} # maps device name to list of interested servers.
# each interested server is {'target':<>,'context':<>,'messageID':<>}
self.identFunctions = {} # maps server to (setting, ctx) for ident
self.identLock = DeferredLock()
# named messages are sent with source ID first, which we ignore
connect_func = lambda c, (s, payload): self.gpib_device_connect(*payload)
disconnect_func = lambda c, (s, payload): self.gpib_device_disconnect(*payload)
mgr = self.client.manager
self._cxn.addListener(connect_func, source=mgr.ID, ID=10)
self._cxn.addListener(disconnect_func, source=mgr.ID, ID=11)
yield mgr.subscribe_to_named_message('GPIB Device Connect', 10, True)
yield mgr.subscribe_to_named_message('GPIB Device Disconnect', 11, True)
# do an initial scan of the available GPIB devices
yield self.refreshDeviceLists()
@inlineCallbacks
def refreshDeviceLists(self):
"""Ask all GPIB bus servers for their available GPIB devices."""
servers = [s for n, s in self.client.servers.items()
if (('GPIB Bus' in n) or ('gpib_bus' in n)) and \
(('List Devices' in s.settings) or \
('list_devices' in s.settings))]
serverNames = [s.name for s in servers]
print 'Pinging servers:', serverNames
resp = yield DeferredList([s.list_devices() for s in servers])
for serverName, (success, addrs) in zip(serverNames, resp):
if not success:
print 'Failed to get device list for:', serverName
else:
print 'Server %s has devices: %s' % (serverName, addrs)
for addr in addrs:
self.gpib_device_connect(serverName, addr)
@inlineCallbacks
def gpib_device_connect(self, gpibBusServer, channel):
"""Handle messages when devices connect."""
print 'Device Connect:', gpibBusServer, channel
if (gpibBusServer, channel) in self.knownDevices:
return
device, idnResult = yield self.lookupDeviceName(gpibBusServer, channel)
if device == UNKNOWN:
device = yield self.identifyDevice(gpibBusServer, channel, idnResult)
self.knownDevices[gpibBusServer, channel] = (device, idnResult)
# forward message if someone cares about this device
if device in self.deviceServers:
self.notifyServers(device, gpibBusServer, channel, True)
def gpib_device_disconnect(self, server, channel):
"""Handle messages when devices connect."""
print 'Device Disconnect:', server, channel
if (server, channel) not in self.knownDevices:
return
device, idnResult = self.knownDevices[server, channel]
del self.knownDevices[server, channel]
# forward message if someone cares about this device
if device in self.deviceServers:
self.notifyServers(device, server, channel, False)
@inlineCallbacks
def lookupDeviceName(self, server, channel):
"""Try to send a *IDN? query to lookup info about a device.
Returns the name of the device and the actual response string
to the identification query. If the response cannot be parsed
or the query fails, the name will be listed as '<unknown>'.
"""
p = self.client.servers[server].packet()
p.address(channel).timeout(Value(1,'s')).write('*CLS').write('*IDN?').read()
print 'Sending *IDN? to', server, channel
resp = None
try:
resp = (yield p.send()).read
name = parseIDNResponse(resp)
except Exception, e:
print 'Error sending *IDN? to', server, channel + ':', e
name = UNKNOWN
returnValue((name, resp))
def identifyDevice(self, server, channel, idn):
"""Try to identify a new device with all ident functions.
Returns the first name returned by a successful identification.
"""
@inlineCallbacks
def _doIdentifyDevice():
for identifier in list(self.identFunctions.keys()):
name = yield self.tryIdentFunc(server, channel, idn, identifier)
if name is not None:
returnValue(name)
returnValue(UNKNOWN)
return self.identLock.run(_doIdentifyDevice)
def identifyDevicesWithServer(self, identifier):
"""Try to identify all unknown devices with a new server."""
@inlineCallbacks
def _doServerIdentify():
#yield self.client.refresh()
for (server, channel), (device, idn) in list(self.knownDevices.items()):
if device != UNKNOWN:
continue
name = yield self.tryIdentFunc(server, channel, idn, identifier)
if name is None:
continue
self.knownDevices[server, channel] = (name, idn)
if name in self.deviceServers:
self.notifyServers(name, server, channel, True)
return self.identLock.run(_doServerIdentify)
@inlineCallbacks
def tryIdentFunc(self, server, channel, idn, identifier):
"""Try calling one registered identification function.
If the identification succeeds, returns the new name,
otherwise returns None.
"""
try:
#yield self.client.refresh()
s = self.client[identifier]
setting, context = self.identFunctions[identifier]
print 'Trying to identify device', server, channel,
print 'on server', identifier,
print 'with *IDN?:', repr(idn)
if idn is None:
resp = yield s[setting](server, channel, context=context)
else:
resp = yield s[setting](server, channel, idn, context=context)
if resp is not None:
data = (identifier, server, channel, resp)
print 'Server %s identified device %s %s as "%s"' % data
returnValue(resp)
except Exception, e:
print 'Error during ident:', str(e)
@setting(1, 'Register Server',
devices=['s', '*s'], messageID='w',
returns='*(s{device} s{server} s{address}, b{isConnected})')
def register_server(self, c, devices, messageID):
"""Register as a server that handles a particular GPIB device(s).
Returns a list with information about all matching devices that
have been connected up to this point:
[(device name, gpib server name, gpib channel, bool)]
After registering, messages will be sent to the registered
message ID whenever a matching device connects or disconnects.
The clusters sent in response to this setting and those sent as
messages have the same format. For messages, the final boolean
indicates whether the device has been connected or disconnected,
while in response to this function call, the final boolean is
always true, since we only send info about connected devices.
The device name is determined by parsing the response to a *IDN?
query. To handle devices that don't support *IDN? correctly, use
the 'Register Ident Function' in addition.
"""
#Managed device servers can specify device names as string or a list of strings
if isinstance(devices, str):
devices = [devices]
found = []
for device in devices:
servers = self.deviceServers.setdefault(device, [])
servers.append({'target': c.source,
'context': c.ID,
'messageID': messageID})
# gather info about matching servers already connected
for (server, channel), (known_device, idnResult) in self.knownDevices.items():
if device != known_device:
continue
found.append((device, server, channel, True))
return found
@setting(2, 'Register Ident Function', setting=['s', 'w'])
def register_ident_function(self, c, setting):
"""Specify a setting to be called to identify devices.
This setting must accept either of the following:
s, s, s: server, address, *IDN? response
s, s: server, address
If a device returned a non-standard response to a *IDN? query
(including possibly an empty string), then the first call signature
will be used. If the *IDN? query timed out or otherwise failed,
the second call signature will be used. As a server writer, you
must choose which of these signatures to support. Note that if the
device behavior is unpredictable (sometimes it returns a string,
sometimes it times out), you may need to support both signatures.
"""
self.identFunctions[c.source] = setting, c.ID
@setting(10)
def dump_info(self, c):
"""Returns information about the server status.
This info includes currently known devices, registered device
servers, and registered identification functions.
"""
return (str(self.knownDevices),
str(self.deviceServers),
str(self.identFunctions))
def notifyServers(self, device, server, channel, isConnected):
"""Notify all registered servers about a device status change."""
for s in self.deviceServers[device]:
rec = s['messageID'], (device, server, channel, isConnected)
print 'Sending message:', s['target'], s['context'], [rec]
self.client._sendMessage(s['target'], [rec], context=s['context'])
def serverConnected(self, ID, name):
"""New GPIBManagedServers will register directly with us, before they
have even completed their registration with the LabRAD manager as a server.
We will get this signal once they are accessible through the LabRAD client
so that we can probe them for devices. This ordering matters mainly if
the new server has a custom IDN parsing function"""
recognizeServer = False
for device, serverInfo in list(self.deviceServers.items()):
if serverInfo[0]['target'] == ID:
recognizeServer = True
if recognizeServer:
callLater(0, self.identifyDevicesWithServer, ID)
def serverDisconnected(self, ID, name):
"""Disconnect devices when a bus server disconnects."""
for (server, channel) in list(self.knownDevices.keys()):
if server == name:
self.gpib_device_disconnect(server, channel)
def expireContext(self, c):
"""Stop sending notifications when a context expires."""
print 'Expiring context:', c.ID
# device servers
deletions = []
for device, servers in list(self.deviceServers.items()):
# remove all registrations with this context
servers = [s for s in servers if s['context'] != c.ID]
self.deviceServers[device] = servers
# if no one is left listening to this device, delete the list
if not len(servers):
deletions.append(device)
for device in deletions:
del self.deviceServers[device]
# ident functions
deletions = []
for src, idents in list(self.identFunctions.items()):
# remove all registrations with this context
idents = [i for i in idents if i[1] != c.ID]
self.identFunctions[src] = idents
# if no one is left listening to this device, delete the list
if not len(idents):
deletions.append(src)
for src in deletions:
del self.identFunctions[src]
__server__ = GPIBDeviceManager()
if __name__ == '__main__':
from labrad import util
util.runServer(__server__)