forked from tomstokes/python-spi
-
Notifications
You must be signed in to change notification settings - Fork 0
/
spi.py
435 lines (345 loc) · 13.9 KB
/
spi.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
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
"""
The MIT License (MIT)
Copyright (c) 2015 Thomas Stokes
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
furnished 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.
"""
__author__ = 'Tom Stokes'
import ctypes
import struct
import fcntl
import array
import os.path
def _ioc(direction, number, structure):
"""
ioctl command encoding helper function
Calculates the appropriate spidev ioctl op argument given the direction,
command number, and argument structure in python's struct.pack format.
Returns a tuple of the calculated op and the struct.pack format
See Linux kernel source file /include/uapi/asm-generic/ioctl.h
"""
ioc_magic = ord('k')
ioc_nrbits = 8
ioc_typebits = 8
ioc_sizebits = 14 # XXX: 13 on PPC, MIPS, Sparc, and Alpha
ioc_nrshift = 0
ioc_typeshift = ioc_nrshift + ioc_nrbits
ioc_sizeshift = ioc_typeshift + ioc_typebits
ioc_dirshift = ioc_sizeshift + ioc_sizebits
size = struct.calcsize(structure)
op = (direction << ioc_dirshift) | (ioc_magic << ioc_typeshift) | \
(number << ioc_nrshift) | (size << ioc_sizeshift)
return direction, op, structure
class SPI(object):
"""
struct spi_ioc_transfer {
__u64 tx_buf;
__u64 rx_buf;
__u32 len;
__u32 speed_hz;
__u16 delay_usecs;
__u8 bits_per_word;
__u8 cs_change;
__u8 tx_nbits;
__u8 rx_nbits;
__u16 pad;
"""
_IOC_TRANSFER_FORMAT = "QQIIHBBBBH"
_IOC_WRITE = 1
_IOC_READ = 2
# _IOC_MESSAGE is a special case, so we ony need the ioctl number
_IOC_MESSAGE = _ioc(_IOC_WRITE, 0, _IOC_TRANSFER_FORMAT)[1]
_IOC_RD_MODE = _ioc(_IOC_READ, 1, "B")
_IOC_WR_MODE = _ioc(_IOC_WRITE, 1, "B")
_IOC_RD_LSB_FIRST = _ioc(_IOC_READ, 2, "B")
_IOC_WR_LSB_FIRST = _ioc(_IOC_WRITE, 2, "B")
_IOC_RD_BITS_PER_WORD = _ioc(_IOC_READ, 3, "B")
_IOC_WR_BITS_PER_WORD = _ioc(_IOC_WRITE, 3, "B")
_IOC_RD_MAX_SPEED_HZ = _ioc(_IOC_READ, 4, "I")
_IOC_WR_MAX_SPEED_HZ = _ioc(_IOC_WRITE, 4, "I")
_IOC_RD_MODE32 = _ioc(_IOC_READ, 5, "I")
_IOC_WR_MODE32 = _ioc(_IOC_WRITE, 5, "I")
CPHA = 0x01
CPOL = 0x02
CS_HIGH = 0x04
LSB_FIRST = 0x08
THREE_WIRE = 0x10
LOOP = 0x20
NO_CS = 0x40
READY = 0x80
TX_DUAL = 0x100
TX_QUAD = 0x200
RX_DUAL = 0x400
RX_QUAD = 0x800
MODE_0 = 0
MODE_1 = CPHA
MODE_2 = CPOL
MODE_3 = CPHA | CPOL
def __init__(self, device, speed=None, bits_per_word=None, phase=None,
polarity=None, cs_high=None, lsb_first=None,
three_wire=None, loop=None, no_cs=None, ready=None):
"""Create spidev interface object.
Args:
device: Tuple of (bus, device) or string of device path
speed: Optional target bus speed in Hz.
bits_per_word: Optional number of bits per word.
Raises:
IOError: The spidev device could not be opened (check permissions)
"""
if isinstance(device, tuple):
(bus, dev) = device
device = "/dev/spidev{:d}.{:d}".format(bus, dev)
if not os.path.exists(device):
raise IOError("{} does not exist".format(device))
self.handle = open(device, "w")
if speed is not None:
self.speed = speed
if bits_per_word is not None:
self.bits_per_word = bits_per_word
if phase is not None:
self.phase = phase
if polarity is not None:
self.polarity = polarity
if cs_high is not None:
self.cs_high = cs_high
if lsb_first is not None:
self.lsb_first = lsb_first
if three_wire is not None:
self.three_wire = three_wire
if self.loop is not None:
self.loop = loop
if self.no_cs is not None:
self.no_cs = no_cs
if self.ready is not None:
self.ready = ready
def _ioctl(self, ioctl_data, data=None):
"""ioctl helper function.
Performs an ioctl on self.handle. If the ioctl is an SPI read type
ioctl, returns the result value.
Args:
ioctl_data: Tuple of (direction, op structure), where direction
is one of SPI._IOC_READ or SPI._IOC_WRITE, op is the
pre-computed ioctl op (see _ioc above) and structure is the
Python format string for the ioctl arguments.
Returns:
If ioctl_data specifies an SPI._IOC_READ, returns the result.
For SPI._IOC_WRITE types, returns None
"""
(direction, ioctl, structure) = ioctl_data
if direction == SPI._IOC_READ:
arg = array.array(structure, [0])
fcntl.ioctl(self.handle, ioctl, arg, True)
return arg[0]
else:
arg = struct.pack("=" + structure, data)
fcntl.ioctl(self.handle, ioctl, arg)
return
def _get_mode(self):
"""Helper function to get spidev mode
Returns:
spidev mode as an integer. Bits correspond to SPI.CPHA,
SPI.CPOL, SPI.CS_HIGH, SPI.LSB_FIRST, SPI.THREE_WIRE, SPI.LOOP,
SPI.NO_CS, and SPI.READY
"""
return self._ioctl(SPI._IOC_RD_MODE)
def _set_mode(self, mode):
"""Helper function to set the spidev mode
Args:
mode: spidev mode as an integer. Bits correspond to SPI.CPHA,
SPI.CPOL, SPI.CS_HIGH, SPI.LSB_FIRST, SPI.THREE_WIRE, SPI.LOOP,
SPI.NO_CS, and SPI.READY
"""
self._ioctl(SPI._IOC_WR_MODE, mode)
def _get_mode_field(self, field):
"""Helper function to get specific spidev mode bits
Args:
field: Bit mask to apply to spidev mode.
Returns:
bool(mode & field). True if specified bit is 1, otherwise False
"""
return True if self._get_mode() & field else False
def _set_mode_field(self, field, value):
"""Helper function to set a spidev mode bit
Args:
field: Bitmask of bit(s) to set to value
value: True to set bit(s) to 1, false to set bit(s) to 0
"""
mode = self._get_mode()
if value:
mode |= field
else:
mode &= ~field
self._set_mode(mode)
@property
def phase(self):
"""SPI clock phase bit
False: Sample at leading edge of clock
True: Sample at trailing edge of clock
"""
return self._get_mode_field(SPI.CPHA)
@phase.setter
def phase(self, phase):
self._set_mode_field(SPI.CPHA, phase)
@property
def polarity(self):
"""SPI polarity bit
False: Data sampled at rising edge, data changes on falling edge
True: Data sampled at falling edge, data changes on rising edge
"""
return self._get_mode_field(SPI.CPOL)
@polarity.setter
def polarity(self, polarity):
self._set_mode_field(SPI.CPOL, polarity)
@property
def cs_high(self):
"""SPI chip select active level
True: Chip select is active high
False: Chip select is active low
"""
return self._get_mode_field(SPI.CS_HIGH)
@cs_high.setter
def cs_high(self, cs_high):
self._set_mode_field(SPI.CS_HIGH, cs_high)
@property
def lsb_first(self):
"""Bit order of SPI word transfers
False: Send MSB first
True: Send LSB first
"""
return self._get_mode_field(SPI.LSB_FIRST)
@lsb_first.setter
def lsb_first(self, lsb_first):
self._set_mode_field(SPI.LSB_FIRST, lsb_first)
@property
def three_wire(self):
"""SPI 3-wire mode
True: Data is read and written on the same line (3-wire mode)
False: Data is read and written on separate lines (MOSI & MISO)
"""
return self._get_mode_field(SPI.THREE_WIRE)
@three_wire.setter
def three_wire(self, three_wire):
self._set_mode_field(SPI.THREE_WIRE, three_wire)
@property
def loop(self):
"""SPI loopback mode"""
return self._get_mode_field(SPI.LOOP)
@loop.setter
def loop(self, loop):
self._set_mode_field(SPI.LOOP, loop)
@property
def no_cs(self):
"""No chipselect. Single device on bus."""
return self._get_mode_field(SPI.NO_CS)
@no_cs.setter
def no_cs(self, no_cs):
self._set_mode_field(SPI.NO_CS, no_cs)
@property
def ready(self):
"""Slave pulls low to pause"""
return self._get_mode_field(SPI.READY)
@ready.setter
def ready(self, ready):
self._set_mode_field(SPI.READY, ready)
@property
def speed(self):
"""Maximum SPI transfer speed in Hz.
Note that the controller cannot necessarily assign the requested
speed.
"""
return self._ioctl(SPI._IOC_RD_MAX_SPEED_HZ)
@speed.setter
def speed(self, speed):
self._ioctl(SPI._IOC_WR_MAX_SPEED_HZ, speed)
@property
def bits_per_word(self):
"""Number of bits per word of SPI transfer.
A value of 0 is equivalent to 8 bits per word
"""
return self._ioctl(SPI._IOC_RD_BITS_PER_WORD)
@bits_per_word.setter
def bits_per_word(self, bits_per_word):
self._ioctl(SPI._IOC_WR_BITS_PER_WORD, bits_per_word)
@property
def mode(self):
return self._get_mode()
@mode.setter
def mode(self, mode):
self._set_mode(mode)
def write(self, data, speed=0, bits_per_word=0, delay=0):
"""Perform half-duplex SPI write.
Args:
data: List of words to write
speed: Optional temporary bitrate override in Hz. 0 (default)
uses existing spidev speed setting.
bits_per_word: Optional temporary bits_per_word override. 0
(default) will use the current bits_per_word setting.
delay: Optional delay in usecs between sending the last bit and
deselecting the chip select line. 0 (default) for no delay.
"""
data = array.array('B', data).tobytes()
length = len(data)
transmit_buffer = ctypes.create_string_buffer(data)
spi_ioc_transfer = struct.pack(SPI._IOC_TRANSFER_FORMAT,
ctypes.addressof(transmit_buffer), 0,
length, speed, delay, bits_per_word, 0,
0, 0, 0)
fcntl.ioctl(self.handle, SPI._IOC_MESSAGE, spi_ioc_transfer)
def read(self, length, speed=0, bits_per_word=0, delay=0):
"""Perform half-duplex SPI read as a binary string
Args:
length: Integer count of words to read
speed: Optional temporary bitrate override in Hz. 0 (default)
uses existing spidev speed setting.
bits_per_word: Optional temporary bits_per_word override. 0
(default) will use the current bits_per_word setting.
delay: Optional delay in usecs between sending the last bit and
deselecting the chip select line. 0 (default) for no delay.
Returns:
List of words read from device
"""
receive_buffer = ctypes.create_string_buffer(length)
spi_ioc_transfer = struct.pack(SPI._IOC_TRANSFER_FORMAT, 0,
ctypes.addressof(receive_buffer),
length, speed, delay, bits_per_word, 0,
0, 0, 0)
fcntl.ioctl(self.handle, SPI._IOC_MESSAGE, spi_ioc_transfer)
return [ord(byte) for byte in ctypes.string_at(receive_buffer, length)]
def transfer(self, data, speed=0, bits_per_word=0, delay=0):
"""Perform full-duplex SPI transfer
Args:
data: List of words to transmit
speed: Optional temporary bitrate override in Hz. 0 (default)
uses existing spidev speed setting.
bits_per_word: Optional temporary bits_per_word override. 0
(default) will use the current bits_per_word setting.
delay: Optional delay in usecs between sending the last bit and
deselecting the chip select line. 0 (default) for no delay.
Returns:
List of words read from SPI bus during transfer
"""
data = array.array('B', data).tobytes()
length = len(data)
transmit_buffer = ctypes.create_string_buffer(data)
receive_buffer = ctypes.create_string_buffer(length)
spi_ioc_transfer = struct.pack(SPI._IOC_TRANSFER_FORMAT,
ctypes.addressof(transmit_buffer),
ctypes.addressof(receive_buffer),
length, speed, delay, bits_per_word, 0,
0, 0, 0)
fcntl.ioctl(self.handle, SPI._IOC_MESSAGE, spi_ioc_transfer)
#return [ord(byte) for byte in ctypes.string_at(receive_buffer, length)]
return [byte for byte in ctypes.string_at(receive_buffer, length)]