-
Notifications
You must be signed in to change notification settings - Fork 0
/
test.py
387 lines (328 loc) · 16.6 KB
/
test.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
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
#! /usr/bin/env python
# encoding: utf-8
# Copyright (c) 2015-2019 Stanford Research Systems
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnshished to do so, subject to the following conditions:
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
# DEALINGS IN THE SOFTWARE.
""" Example python script to capture streaming data from an SR865.
Tested with python 3.7.3
Typical installation will require vxi11 and docopt installation in the usual way.
**** Your host computer firewall MUST allow incoming UDP on the streaming port !!! ****
See your IT person if you need help with this. Streaming cannot work without the open port.
python stream.py -h to see the list of options
Default options can be changed by editing the useStr (below).
You will certainly need to change the IP address to match your SR865 and
for compatability with your network.
"""
import csv
import math
import pandas as pd
import socket
from struct import unpack_from
import signal
import sys
import time
import threading
import multiprocessing
import queue
# you may need to install these python modules
try:
import vxi11 # required
except ImportError:
print('required python vxi11 library not found. Please install vxi11')
#try:
# import docopt # handy command line parser.
#except ImportError:
# print('python docopt library not found. Please install docopt')
USE_STR = """
--Stream Data from an SR865 to a file--
Usage:
stream [--address=<A>] [--length=<L>] [--port=<P>] [--duration=<D>] [--vars=<V>] [--rate=<R>] [--silent] [--thread] [--file=<F>] [--ints]
stream -h | --help
Options:
-a --address <A> IP address of SR865 [default: 172.25.98.253]
-d --duration <D> How long to transfer in seconds [default: 10]
-f --file <F> Name for file output. No file output without a file name.
-h --help Show this screen
-i --ints Data in 16-bit ints instead of 32-bit floats
-l --length <L> Packet length enum (0 to 3) [default: 0]
-p --port <P> UDP Port [default: 1865]
-r --rate <R> Sample rate per second. Actual will be less and depends on filter settings [default: 1e5]
-s --silent Refrain from printing packet count and data until complete
-t --thread Decouple output from ethernet stream using threads
-v --vars <V> Lock-in variables to stream [default: X] XY, RT, or XYRT are also allowed
"""
def show_status(left_text='', right_text=''):
""" Simple text status line that overwrites itself to prevent scrolling.
"""
print(' %-30s %48s\r'%(left_text[:30], right_text[:48]), end=' ')
# globals get assigned the udp and vxi11 objects to allow SIGINT to cleanup properly
# pylint wants me to name these in all caps, as if they are constants. They're not.
the_udp_socket = None #pylint: disable=global-statement, invalid-name
the_vx_ifc = None #pylint: disable=global-statement, invalid-name
def cleanup_ifcs():
""" Stop the stream and close the socket and vxi11.
"""
# global the_udp_socket
# global the_vx_ifc
print("\n cleaning up...", end=' ')
the_vx_ifc.write('STREAM OFF')
the_vx_ifc.close()
the_udp_socket.close()
print('connections closed\n')
def open_interfaces(ipadd, port):
""" open a UDP socket and a vxi11 instrument and assign them to the globals
"""
global the_udp_socket #pylint: disable=global-statement, invalid-name
global the_vx_ifc #pylint: disable=global-statement, invalid-name
print('\nopening incoming UDP Socket at %d ...' % port, end=' ')
the_udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
the_udp_socket.bind(('', port)) # listen to anything arriving on this port from anyone
print('done')
print('opening VXI-11 at %s ...' % ipadd, end=' ')
the_vx_ifc = vxi11.Instrument(ipadd)
the_vx_ifc.write('STREAMPORT %d'%port)
print('done')
def dut_config(vx_ifc, s_channels, idx_pkt_len, f_rate_req, b_integers):
""" Setup the SR865 for streaming. Return the rate (samples/sec)
"""
vx_ifc.write('STREAM OFF') # turn off streaming while we set it up
vx_ifc.write('STREAMCH %s'%s_channels)
if b_integers:
vx_ifc.write('STREAMFMT 1') # 16 bit int
else:
vx_ifc.write('STREAMFMT 0') # 32 bit float
vx_ifc.write('STREAMOPTION 2') # use big-endian (~1) and data integrity checking (2)
vx_ifc.write('STREAMPCKT %d'%idx_pkt_len)
f_rate_max = float(vx_ifc.ask('STREAMRATEMAX?')) # filters determine the max data rate
# calculate a decimation to stay under f_rate_req
i_decimate = int(math.ceil(math.log(f_rate_max/f_rate_req, 2.0)))
if i_decimate < 0:
i_decimate = 0
if i_decimate > 20:
i_decimate = 20
f_rate = f_rate_max/(2.0**i_decimate)
print('Max rate is %.3f kS/S.'%(f_rate_max*1e-3))
print('Decimating by 2^%d down to %.3f kS/S'%(i_decimate, f_rate*1e-3))
vx_ifc.write('STREAMRATE %d'%i_decimate) # bring the rate under our target rate
return f_rate
def write_to_file(f_name, s_channels, lst_stream):
""" Save data to a comma separated file. We could also use the csv python module...
s_channels is "X" or "XY", etc indicating how many values in a lst_sample
lst_stream[][] is a list of sample lists
"""
show_status('writing %s ...'%f_name)
with open(f_name, 'w') as f_ptr:
f_ptr.write(''.join(['%s,'%str.upper(v) for v in s_channels])+'\n')
if isinstance(lst_stream[0][0], float):
s_val_fmt = '%+12.6e,'*len(s_channels)
else:
s_val_fmt = '%+d,'*len(s_channels)
for lst_sample in lst_stream:
for i in range(0, len(lst_sample), len(s_channels)):
smpl = lst_sample[i:i+len(s_channels)]
f_ptr.write(s_val_fmt%tuple(smpl)+'\n')
show_status('%s written'%f_name)
def alternatewrite_to_file(f_name, lst_stream):
show_status('writing %s ...' % f_name)
with open(f_name, "w", newline="") as f:
writer = csv.writer(f)
writer.writerow(["timestamp", "X", "Y"])
writer.writerows(lst_stream)
# socket seems to catch the KeyboardInterrupt exception if I don't grab it explicitly here
def interrupt_handler(signum, frame): #pylint: disable=unused-argument
""" call my cleanup_ifcs when something bad happens
"""
cleanup_ifcs()
# catching the signal removes the close process behaviour of Ctrl-C
sys.exit(-2) # so Terminate process here
signal.signal(signal.SIGINT, interrupt_handler)
# thread functions ----------------------------------------------
def fill_queue(sock_udp, q_data, count_packets, bytes_per_packet):
""" Pump packets from the socket (SR865) to the python dataQueue
"""
for _ in range(count_packets):
buf, _ = sock_udp.recvfrom(bytes_per_packet)
q_data.put(buf)
cleanup_ifcs()
def process_packet(buf, fmt_unpk, prev_pkt_cntr):
""" Unpack the header and data froma packet, checking for dropped packets.
return the data, the header, the number of packets missed, and the current packet number.
Only the packet counter is checked - other, possibly important information in the header
(such as overload, unlock status, data type, streamed variables, and sample rate)
are ignored.
"""
# convert to floats or ints after skipping 4 bytes of header
vals = list(unpack_from(fmt_unpk, buf, 4))
vals.insert(0,time.time_ns())
head = unpack_from('>I', buf)[0] # convert the header to an 32 bit int
cntr = head & 0xff # extract the packet counter from the header
# check for missed packets
# if this isn't the 1st and the difference isn't 1 then
if prev_pkt_cntr is not None and ((prev_pkt_cntr+1)&0xff) != cntr:
n_dropped = cntr - prev_pkt_cntr # calculate how many we missed
if n_dropped < 0:
n_dropped += 0xff
else:
n_dropped = 0
return vals, head, n_dropped, cntr
def empty_queue(q_data, q_drop, count_packets, bytes_per_packet, fmt_unpk, s_prt_fmt, s_channels, fname, bshow_status): #pylint: disable=too-many-arguments, too-many-locals, line-too-long
""" myThreads[1] calls this to pull data out of the dataQueue.
When all the packets have been processed:
writes to a file (optional)
displays the dropped packet stats
writes the drop list (maybe empty) to q_drop.
"""
prev_pkt_cntr = None # init the packet counter
lst_dropped = [] # make a list of any missing packets
count_dropped = 0
lst_stream = []
count_vars = len(s_channels)
for i in range(count_packets):
buf = q_data.get()
vals, _, n_dropped, prev_pkt_cntr = process_packet(buf, fmt_unpk, prev_pkt_cntr)
lst_stream += [vals]
if n_dropped:
lst_dropped += [(n_dropped, i)]
count_dropped += n_dropped
if bshow_status:
show_status('dropped %4d of %d'%(count_dropped, i+1), s_prt_fmt%tuple(lst_stream[-1][-count_vars:])) #pylint: disable=line-too-long
if fname is not None:
alternatewrite_to_file(fname, lst_stream)
#write_to_file(fname, s_channels, lst_stream)
show_results(count_dropped, count_packets, lst_dropped, count_packets*bytes_per_packet/(4*count_vars)) #pylint: disable=line-too-long
q_drop.put(lst_dropped) # signal to main thread that we finished the post-processing
def show_results(count_dropped, count_packets, lst_dropped, count_samples):
""" print indicating OK, or some dropped packets"""
if count_dropped:
print('\nFAIL: Dropped %d out of %d packets in %d gaps:'%(count_dropped, count_packets, len(lst_dropped)), end=' ') #pylint: disable=line-too-long
print(''.join('%d at %d, '%(x[0], x[1]) for x in lst_dropped[:5]))
else:
print('\npass: No packets dropped out of %d. %d samples captured.'%(count_packets, count_samples)) #pylint: disable=line-too-long
# the main program -----------------------------------------------
def test(opts): #pylint: disable=too-many-locals, too-many-statements
""" example main()
"""
# global the_udp_socket
# global the_vx_ifc
# group the docopt stuff to make it easier to remove, if desired
dut_add = opts['--address'] # IP address and streaming port of the SR865
dut_port = int(opts['--port'])
# sample rate that host wants.
# Actual rate will be below this and depends on filter settings
f_rate_req = float(opts['--rate'])
# select the packet size: 0 to 3 select 1024..128 byte packets
idx_pkt_len = int(opts['--length'])
duration_stream = float(opts['--duration']) # in seconds
bshow_status = not opts['--silent']
fname = opts['--file']
s_channels = str(opts['--vars']) # what to stream. X, XY, RT, or XYRT allowed
lst_vars_allowed = ['X', 'XY', 'RT', 'XYRT']
b_integers = opts['--ints']
b_use_threads = opts['--thread']
if s_channels.upper() not in lst_vars_allowed:
print('bad --vars option (%s). Must be one of'%s_channels.upper(), ', '.join(lst_vars_allowed)) #pylint: disable=line-too-long
sys.exit(-1)
open_interfaces(dut_add, dut_port)
f_total_samples = duration_stream * dut_config(the_vx_ifc, s_channels, idx_pkt_len, f_rate_req, b_integers) #pylint: disable=line-too-long
# translate the packet size enumeration into an actual byte count
bytes_per_pkt = [1024, 512, 256, 128][idx_pkt_len]
if b_integers:
fmt_unpk = '>%dh'%(bytes_per_pkt//2) # create an unpacking format string.
fmt_live_printing = '%12d'*len(s_channels) # create status format string.
else:
fmt_unpk = '>%df'%(bytes_per_pkt//4)
fmt_live_printing = '%12.6f'*len(s_channels)
total_packets = int(math.ceil(f_total_samples*4*len(s_channels)/bytes_per_pkt))
prev_pkt_cntr = None # init the packet counter
lst_stream = [] # make a list of lists of the float data
headers = [] # make a list of the packet headers
dropped = [] # make a list of any gaps in the packets
show_status('streaming ...')
time_start = time.perf_counter()
the_vx_ifc.write('STREAM ON')
if b_use_threads:
the_threads = []
queue_drops = queue.Queue()
queue_data = queue.Queue() # decouple the printing/saving from the UDP socket
for queue_func, queue_args in [(fill_queue, (the_udp_socket, queue_data, total_packets, bytes_per_pkt+4)), #pylint: disable=line-too-long
(empty_queue, (queue_data, queue_drops, total_packets, bytes_per_pkt, fmt_unpk, fmt_live_printing, s_channels, fname, bshow_status))]: #pylint: disable=line-too-long
the_threads.append(threading.Thread(target=queue_func, args=queue_args))
# the_threads[-1].setDaemon(True)
the_threads[-1].start()
s_no_printing = '' if bshow_status else 'silently'
print('threads started %s\n'%s_no_printing)
the_threads[0].join(duration_stream+2) # time out 2 seconds after the expected duration
the_threads[1].join(duration_stream*2) # time out 2x the duration more
# queue_drops.get() blocks until empty_queue() writes to queue_drops showing it finished.
dropped = queue_drops.get()
print('threads done')
else: # don't use threads. "block" instead
for i in range(total_packets):
# .recvfrom "blocks" program execution until all the bytes have been received.
buf, _ = the_udp_socket.recvfrom(bytes_per_pkt+4)
vals, head, n_dropped, prev_pkt_cntr = process_packet(buf, fmt_unpk, prev_pkt_cntr)
lst_stream += [vals] #vals include hardcode timestamp
headers += [head]
dropped += [n_dropped]
if bshow_status:
show_status('dropped %4d of %d'%(sum(dropped), i), fmt_live_printing%tuple(lst_stream[-1][-len(s_channels):])) #pylint: disable=line-too-long
if fname is not None:
alternatewrite_to_file(fname, lst_stream)
#write_to_file(fname, s_channels, lst_stream)
cleanup_ifcs()
print("avoided cleanup")
#show_results(sum(dropped), total_packets, dropped, total_packets*bytes_per_pkt//(4*len(s_channels))) #pylint: disable=line-too-long
time_end = time.perf_counter()
print('Time elapsed: %.3f seconds'%(time_end-time_start))
if __name__ == '__main__':
# group the docopt stuff to make it easier to remove, if desired
#dict_options = docopt.docopt(USE_STR, version='0.0.2') #pylint: disable=invalid-name
dict_options1 = {
'--address': '10.0.0.3',
'--duration': 120,
'--file': 'thread1short.csv',
'--ints':False,
'--length': 3,
'--port': 1865,
'--rate': 100000,
'--silent':False,
'--thread': True,
'--vars': 'XY'
}
dict_options2 = {
'--address': '10.0.0.4',
'--duration': 120,
'--file': 'thread2short.csv',
'--ints': False,
'--length': 3,
'--port': 1866,
'--rate': 100000,
'--silent': False,
'--thread': True,
'--vars': 'XY'
}
process1 = multiprocessing.Process(target=test, args=(dict_options1,))
process2 = multiprocessing.Process(target=test, args=(dict_options2,))
process1.start()
print("process1 started")
#time.sleep(2)
process2.start()
print("process2 started")
process1.join()
process2.join()
print("simultaneous threads are done!")
#cleanup_ifcs() #commenting out to see if there are erros
print("cleaned up")