forked from NeuralEnsemble/pypl2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpypl2api.py
313 lines (244 loc) · 11.6 KB
/
pypl2api.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
# pypl2api.py - High level functions for accessing
# .pl2 files. Mimics Plexon's Matlab PL2 SDK
#
# (c) 2016 Plexon, Inc., Dallas, Texas
# www.plexon.com
#
# This software is provided as-is, without any warranty.
# You are free to modify or share this file, provided that the above
# copyright notice is kept intact.
from collections import namedtuple
from pypl2lib import *
def print_error(pypl2_file_reader_instance):
error_message = (ctypes.c_char * 256)()
pypl2_file_reader_instance.pl2_get_last_error(error_message, 256)
print(error_message.value)
def pl2_ad(filename, channel):
"""
Reads continuous data from specific file and channel.
Usage:
>>>adfrequency, n, timestamps, fragmentcounts, ad = pl2_ad(filename, channel)
>>>res = pl2_ad(filename, channel)
Args:
filename - full path and filename of .pl2 file
channel - zero-based channel index, or channel name
Returns (named tuple fields):
adfrequency - digitization frequency for the channel
n - total number of data points
timestamps - tuple of fragment timestamps (one timestamp per fragment, in seconds)
fragmentcounts - tuple of fragment counts
ad - tuple of raw a/d values in volts
The returned data is in a named tuple object, so it can be accessed as a normal tuple:
>>>res = pl2_ad('data/file.pl2', 0)
>>>res[0]
40000
or as a named tuple:
>>>res.adfrequency
40000
If any error is detected, an error message is printed and the function returns 0
"""
# Create an instance of PyPL2FileReader.
p = PyPL2FileReader()
# Open the file.
p.pl2_open_file(filename)
p.pl2_get_file_info()
# Check if channel is an integer or string, and call appropriate function
if type(channel) is int:
achannel_info = p.pl2_get_analog_channel_info(channel)
if type(channel) in (str, bytes):
achannel_info = p.pl2_get_analog_channel_info_by_name(channel)
# Check if channel is an integer or string, and call appropriate function
if type(channel) is int:
fragment_timestamps, fragment_counts, values = p.pl2_get_analog_channel_data(channel)
if type(channel) in (str, bytes):
fragment_timestamps, fragment_counts, values = p.pl2_get_analog_channel_data_by_name(channel)
# Close the file
p.pl2_close_file()
# Create a named tuple called PL2Ad.
PL2Ad = namedtuple('PL2Ad', 'adfrequency n timestamps fragmentcounts ad')
# Fill in and return named tuple.
return PL2Ad(achannel_info.m_SamplesPerSecond,
len(values),
to_array_nonzero(fragment_timestamps) / p.pl2_file_info.m_TimestampFrequency,
to_array_nonzero(fragment_counts),
to_array(values) * achannel_info.m_CoeffToConvertToUnits)
def pl2_spikes(filename, channel, unit=[]):
"""
Reads spike data from a specific file and channel.
Usage:
>>>n, timestamps, units, waveforms = pl2_spikes(filename, channel)
>>>res = pl2_spikes(filename, channel)
Args:
filename - full path and filename of .pl2 file
channel - zero-based channel index, or channel name
Returns (named tuple fields):
n - number of spike waveforms
timestamps - tuple of spike waveform timestamps in seconds
units - tuple of spike waveform unit assignments (0 = unsorted, 1 = Unit A, 2 = Unit B, etc)
waveforms - tuple of tuples with raw waveform a/d values in volts
The returned data is in a named tuple object, so it can be accessed as a normal tuple:
>>>res = pl2_spikes('data/file.pl2', 0)
>>>res[0]
6589
or as a named tuple:
>>>res.n
6589
To access individual waveform values, address res.waveforms with the index of the waveform.
Result shortened for example:
>>>res.waveforms[49]
(0.000345643, 0.000546342, ... , -0.03320040)
If any error is detected, an error message is printed and the function returns 0
"""
# Create an instance of PyPL2FileReader.
p = PyPL2FileReader()
# Verify that the file passed exists first.
# Open the file.
p.pl2_open_file(filename)
# Create instance of PL2SpikeChannelInfo.
schannel_info = PL2SpikeChannelInfo()
# Check if channel is an integer or string, and call appropriate function
if type(channel) is int:
schannel_info = p.pl2_get_spike_channel_info(channel)
if type(channel) in (str, bytes):
schannel_info = p.pl2_get_spike_channel_info_by_name(channel)
if type(channel) is int:
res = p.pl2_get_spike_channel_data(channel)
if type(channel) in (str, bytes):
res = p.pl2_get_spike_channel_data_by_name(channel)
spike_timestamps, units, values = res
# Close the file
p.pl2_close_file()
waveforms = (values * schannel_info.m_CoeffToConvertToUnits)
# Create a named tuple called PL2Spikes
PL2Spikes = namedtuple('PL2Spikes', 'n timestamps units waveforms')
return PL2Spikes(waveforms.size,
spike_timestamps / p.pl2_file_info.m_TimestampFrequency,
units,
waveforms)
def pl2_events(filename, channel):
"""
Reads event channel data from a specific file and event channel
Usage:
>>>n, timestamps, values = pl2_events(filename, channel)
>>>res = pl2_events(filename, channel)
Args:
filename - full path of the file
channel - 1-based event channel index, or event channel name;
Returns (named tuple fields):
n - number of events
timestamps - array of timestamps (in seconds)
values - array of event values (when event is a strobed word)
The returned data is in a named tuple object, so it can be accessed as a normal tuple:
>>>res = pl2_events('data/file.pl2', 1)
>>>res[0]
784
or as a named tuple:
>>>res.n
784
"""
# Create an instance of PyPL2FileReader.
p = PyPL2FileReader()
# Verify that the file passed exists first.
# Open the file.
p.pl2_open_file(filename)
p.pl2_get_file_info()
# Set up instances of ctypes classes needed by pl2_get_digital_channel_data().
if type(channel) is int:
event_timestamps, even_values = p.pl2_get_digital_channel_data(channel)
if type(channel) in (str, bytes):
event_timestamps, event_values = p.pl2_get_digital_channel_data_by_name(channel)
# Close the file
p.pl2_close_file()
# Create a named tuple called PL2DigitalEvents
PL2DigitalEvents = namedtuple('PL2DigitalEvents', 'n timestamps values')
return PL2DigitalEvents(len(event_values),
event_timestamps / p.pl2_file_info.m_TimestampFrequency,
event_values)
def pl2_info(filename):
"""
Reads a PL2 file and returns information about the file.
Usage:
>>>spkcounts, evtcounts, adcounts = pl2_info(filename)
>>>res = pl2_info(filename)
Args:
filename - Full path of the file
Returns (named tuple fields):
spikes - tuple the length of enabled spike channels with tuples
consisting of the spike channel number, name, and tuple of
unit counts. The returned named tuple fields are:
channel - channel number
name - channel name
units - tuple with number of waveforms assigned to units
0 (unsorted) through 255
events - tuple the length of event channels that contain data with tuples
consisting of the event channel number, name, and number of events.
The returned named tuple fields are:
channel - channel number
name - channel name
n - number of events in the channel
ad - tuple the length of enabled ad channels with tuples consisting of the ad
channel number, name, and number of samples. The returned named tuple fields
are:
channel - channels name
name - channel name
n - number of samples in the channel
The returned data is in a named tuple object, and it's filled with more named tuple objects.
There are several ways to access returned data.
>>>spikecounts, eventcounts, adcounts = pl2_info('data/file.pl2')
>>>len(spikecounts)
>>>4
pl2_info returns a named tuple object, but in the above example the three elements have been
unpacked already into tuples called spikecounts, eventcounts, and adcounts. The length of
these tuples indicate how many channels were enabled, or had values in them in the case of
event channels (since events are always enabled).
>>>spikecounts[2].name
>>>'SPK03'
Continuing the example, the third element returned in spikecounts (2 is the third element
because tuple indexing starts from 0) has a field called name, which contains the channel's
name, SPK03. Because you can treat named tuples like normal tuples, you could also get that
information by the index (because you read the documentation and know that the name field is
the second element of the tuple).
>>>spikecounts[2][1]
>>>'SPK03'
If you don't unpack the returned tuple, it's still easy to get the information. The unpacked
named tuple has the fields spikes, events, and ad.
>>>res = pl2_info('data/file.pl2')
>>>res.spikes[2].name
>>>'SPK03'
"""
# Create an instance of PyPL2FileReader.
p = PyPL2FileReader()
# Verify that the file passed exists first.
# Open the file.
p.pl2_open_file(filename)
p.pl2_get_file_info()
# Lists that will be filled with tuples
spike_counts = []
event_counts = []
ad_counts = []
# Named tuples that will be appended into lists
spike_info = namedtuple('spike_info', 'channel name units')
event_info = namedtuple('event_info', 'channel name n')
ad_info = namedtuple('ad_info', 'channel name n')
# Get channel numbers, names, and unit counts for all enabled spike channels
for i in range(p.pl2_file_info.m_TotalNumberOfSpikeChannels):
schannel_info = p.pl2_get_spike_channel_info(i)
if schannel_info.m_ChannelEnabled:
spike_counts.append(spike_info(schannel_info.m_Channel, schannel_info.m_Name.decode('ascii'),
tuple(schannel_info.m_UnitCounts)))
# Get channel numbers, names, and counts for all event channels with data
for i in range(p.pl2_file_info.m_NumberOfDigitalChannels):
echannel_info = p.pl2_get_digital_channel_info(i)
if echannel_info.m_NumberOfEvents:
event_counts.append(event_info(echannel_info.m_Channel, echannel_info.m_Name.decode('ascii'),
echannel_info.m_NumberOfEvents))
# Get channel numbers, names, and counts for all enabled spike channels
for i in range(p.pl2_file_info.m_TotalNumberOfAnalogChannels):
achannel_info = p.pl2_get_analog_channel_info(i)
if achannel_info.m_ChannelEnabled:
ad_counts.append(ad_info(achannel_info.m_Channel, achannel_info.m_Name.decode('ascii'),
achannel_info.m_NumberOfValues))
# Close the file
p.pl2_close_file()
PL2Info = namedtuple('PL2Info', 'spikes events ad')
return PL2Info(tuple(spike_counts), tuple(event_counts), tuple(ad_counts))