-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathnewport.py
291 lines (224 loc) · 9 KB
/
newport.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
"""Python Driver for NewFocus controllers
__author__ = "Ben Hammel"
__email__ = "[email protected]"
__status__ = "Development"
Requirements:
python (version > 2.7)
re
pyusb: $ pip install pyusb
libusb-compat (USB backend): $ brew install libusb-compat
References:
[1] pyusb tutorial, https://github.com/walac/pyusb/blob/master/docs/tutorial.rst
5/13/2015
[2] user manual, http://assets.newport.com/webDocuments-EN/images/8742_User_Manual_revB.pdf
5/13/2015
Further Information:
Notes:
1) Only tested with 1 8742 Model Open Loop Picomotor Controller.
2) Current code will not support multiple controllers
TODO:
1) Add in multi controller functionality
2) Block illegal commands, not just commands with an invalid format
3) Develop GUI
4) Add in connection check.
Execute the following commands ever .5 s
1>1 MD?\r
1>1 TP?\r
1>2 TP?\r
1>3 TP?\r
1>4 TP?\r
1>ERRSTR?\r
Use http://stackoverflow.com/questions/6357850/how-do-i-run-some-python-code-in-another-process
"""
import usb.core
import usb.util
import re
NEWFOCUS_COMMAND_REGEX = re.compile("([0-9]{0,1})([a-zA-Z?]{2,})([0-9+-]*)")
MOTOR_TYPE = {
"0":"No motor connected",
"1":"Motor Unknown",
"2":"'Tiny' Motor",
"3":"'Standard' Motor"
}
class Controller(object):
"""Picomotor Controller
Example:
>>> controller = Controller(idProduct=0x4000, idVendor=0x104d)
>>> controller.command('VE?')
>>> controller.start_console()
"""
def __init__(self, idProduct, idVendor):
"""Initialize the Picomotor class with the spec's of the attached device
Call self._connect to set up communication with usb device and endpoints
Args:
idProduct (hex): Product ID of picomotor controller
idVendor (hex): Vendor ID of picomotor controller
"""
self.idProduct = idProduct
self.idVendor = idVendor
self._connect()
def _connect(self):
"""Connect class to USB device
Find device from Vendor ID and Product ID
Setup taken from [1]
Raises:
ValueError: if the device cannot be found by the Vendor ID and Product
ID
Assert False: if the input and outgoing endpoints can't be established
"""
# find the device
self.dev = usb.core.find(
idProduct=self.idProduct,
idVendor=self.idVendor
)
if self.dev is None:
raise ValueError('Device not found')
# set the active configuration. With no arguments, the first
# configuration will be the active one
self.dev.set_configuration()
# get an endpoint instance
cfg = self.dev.get_active_configuration()
intf = cfg[(0,0)]
self.ep_out = usb.util.find_descriptor(
intf,
# match the first OUT endpoint
custom_match = \
lambda e: \
usb.util.endpoint_direction(e.bEndpointAddress) == \
usb.util.ENDPOINT_OUT)
self.ep_in = usb.util.find_descriptor(
intf,
# match the first IN endpoint
custom_match = \
lambda e: \
usb.util.endpoint_direction(e.bEndpointAddress) == \
usb.util.ENDPOINT_IN)
assert (self.ep_out and self.ep_in) is not None
# Confirm connection to user
resp = self.command('VE?')
print("Connected to Motor Controller Model {}. Firmware {} {} {}\n".format(
*resp.split(' ')
))
for m in range(1,5):
resp = self.command("{}QM?".format(m))
print("Motor #{motor_number}: {status}".format(
motor_number=m,
status=MOTOR_TYPE[resp[-1]]
))
def send_command(self, usb_command, get_reply=False):
"""Send command to USB device endpoint
Args:
usb_command (str): Correctly formated command for USB driver
get_reply (bool): query the IN endpoint after sending command, to
get controller's reply
Returns:
Character representation of returned hex values if a reply is
requested
"""
self.ep_out.write(usb_command)
if get_reply:
return self.ep_in.read(100)
def parse_command(self, newfocus_command):
"""Convert a NewFocus style command into a USB command
Args:
newfocus_command (str): of the form xxAAnn
> The general format of a command is a two character mnemonic (AA).
Both upper and lower case are accepted. Depending on the command,
it could also have optional or required preceding (xx) and/or
following (nn) parameters.
cite [2 - 6.1.2]
"""
m = NEWFOCUS_COMMAND_REGEX.match(newfocus_command)
# Check to see if a regex match was found in the user submitted command
if m:
# Extract matched components of the command
driver_number, command, parameter = m.groups()
usb_command = command
# Construct USB safe command
if driver_number:
usb_command = '1>{driver_number} {command}'.format(
driver_number=driver_number,
command=usb_command
)
if parameter:
usb_command = '{command} {parameter}'.format(
command=usb_command,
parameter=parameter
)
usb_command += '\r'
return usb_command
else:
print("ERROR! Command {} was not a valid format".format(
newfocus_command
))
def parse_reply(self, reply):
"""Take controller's reply and make human readable
Args:
reply (list): list of bytes returns from controller in hex format
Returns:
reply (str): Cleaned string of controller reply
"""
# convert hex to characters
reply = ''.join([chr(x) for x in reply])
return reply.rstrip()
def command(self, newfocus_command):
"""Send NewFocus formated command
Args:
newfocus_command (str): Legal command listed in usermanual [2 - 6.2]
Returns:
reply (str): Human readable reply from controller
"""
usb_command = self.parse_command(newfocus_command)
# if there is a '?' in the command, the user expects a response from
# the driver
if '?' in newfocus_command:
get_reply = True
else:
get_reply = False
reply = self.send_command(usb_command, get_reply)
# if a reply is expected, parse it
if get_reply:
return self.parse_reply(reply)
def start_console(self):
"""Continuously ask user for a command
"""
print('''
Picomotor Command Line
---------------------------
Enter a valid NewFocus command, or 'quit' to exit the program.
Common Commands:
xMV[+-]: .....Indefinitely move motor 'x' in + or - direction
ST: .....Stop all motor movement
xPRnn: .....Move motor 'x' 'nn' steps
\n
''')
while True:
command = raw_input("Input > ")
if command.lower() in ['q', 'quit', 'exit']:
break
else:
rep = self.command(command)
if rep:
print("Output: {}".format(rep))
if __name__ == '__main__':
print('\n\n')
print('#'*80)
print('#\tPython controller for NewFocus Picomotor Controller')
print('#'*80)
print('\n')
idProduct = None # '0x4000'
idVendor = None # '0x104d'
if not (idProduct or idVendor):
print('Run the following command in a new terminal window:')
print('\t$ system_profiler SPUSBDataType\n')
print('Enter Product ID:')
idProduct = raw_input('> ')
print('Enter Vendor ID:')
idVendor = raw_input('> ')
print('\n')
# convert hex value in string to hex value
idProduct = int(idProduct, 16)
idVendor = int(idVendor, 16)
# Initialize controller and start console
controller = Controller(idProduct=idProduct, idVendor=idVendor)
controller.start_console()