The SPI API supports the Serial Peripheral Interface, a synchronous serial protocol that allows multiple slave chips to communicate with a master chip. A single SPI bus uses the following pins: SCLK for clock, MOSI (Master Out, Slave In) for write, MISO (Master In, Slave Out) for read, and one or more SS (Slave Select) for selecting the slave device.
For each clock signal one bit is written from the master to the selected slave and one bit is read by the master from the selected slave, so there is no separate read and write, but one transceive operation.
When a slave device's chip select is 0 (low), then it communicates with the master, otherwise it ignores the master. The master can select multiple slaves in a write-only configuration; in this case no slave is writing data, they only read.
Since the SS pins may be connected to slave chip select through a demultiplexer and thereby work as an address bus, slave devices are identified by an index in this API, rather than by SS pins. Also, since multiple SPI buses may be present on a board, these are identified by an index in this API. Implementations SHOULD encapsulate the mapping from SPI bus number and device number to the physical SPI pins.
Commmunication with SPI devices is defined in the terms of SPI transactions:
- configure SPI
- start the transaction
- data transfer (one or more times)
- close the transaction.
This API uses a Buffer
object for both read and write data. An SPI transaction is started by invoking the open()
method that configures and starts SPI and returns a control object on which applications can invoke the transceive()
method multiple times, then are supposed to invoke close()
in order to close the SPI transaction.
When requiring "spi"
, the following steps are run:
- If there is no permission for using the functionality, throw
SecurityError
. - If the SPI functionality is not supported on the board, throw
"NotSupportedError"
. - Return an object that implements the following method.
Method | Description |
---|---|
open() |
synchronous open |
See also the Web IDL definition.
Configures an SPI bus using data provided by the options
argument.
Property | Type | Optional | Default value | Represents |
---|---|---|---|---|
bus |
octet | no | 0 | SPI bus |
speed |
long | yes | undefined |
SPI bus speed in MHz |
msbFirst |
boolean | yes | true |
MSB (or LSB) first |
bits |
number | yes | 8 | number of data bits |
polarity |
long | yes | 0 | clock polarity, 0 or 2 |
phase |
long | yes | 0 | clock phase, 0 or 1 |
frameGap |
unsigned long | yes | undefined |
inter-frame gap in nanoseconds |
topology |
string | yes | "full-duplex" |
SPI master-slave topology |
The open()
method runs the following steps:
- If
options
is a dictionary and theoptions.bus
property isundefined
, then setoptions.bus
to 0. - Otherwise if
options.bus
is not a number between 0 and 127, throwTypeError
and abort these steps. - If
options.speed
isundefined
, letoptions.speed
be 10. Otherwise, setoptions.speed
to the closest matching value that is lower thanoptions.speed
and is supported by the platform. - If
options.msbFirst
isundefined
, set it totrue
. - If
options.bits
is not in the set {1, 2, 4, 8, 16 }, then set it to the value 4. - If
options.polarity
isundefined
, set it to 0. If it is not 0, set it to 2. - If
options.phase
isundefined
, set it to 0. If it is not 0, set it to 1. - If
spi.frameGap
is defined, request the underlying platform to set the SPI inter-frame delay value expressed in nanoseconds. If the request fails or it is not supported, setspi.frameGap
toundefined
. - If
spi.topology
is defined, use its value to initialize SPI on the underlying platform. If that fails, or it is not supported, setspi.topology
to the default value"full-duplex"
. This will define the behaviour oftransceive()
. - Request the underlying platform to initialize SPI with
options
. The implementation will use the board mapping from the value ofoptions.bus
to the set of physical pins used for the bus. - In case of failure in any of the steps above, throw
SystemError
. - Let
spi
be anSPI
object that controls the initialized SPI bus. - Return
spi
.
Exposes SPI functionality and represents one SPI transaction that has begun by calling open()
. Applications can invoke the transceive()
method multiple times, then are supposed to invoke close()
in order to release the SPI bus.
Method | Description |
---|---|
transceive() |
read or write device(s) |
close() |
close the SPI bus |
The bus
property denotes the SPI bus number between 0 and 127.
The speed
property is a floating point number that denotes the SPI bus speed in MHz. Usually it is between 10 and 66 MHz.
The msbFirst
property is a boolean denoting whether the most significant bit (MSB) is sent first, or the least significant bit (LSB). The default value is true
(MSB first).
The bits
property is a number denoting the number of data bits (word size). Usually it is 1, 2, 4, 8 or 16. The default value is 4.
The polarity
property is a number denoting the clock polarity (CPOL). The value 0 means clock is active on high (rising edge), the value 2 means clock is active on low (falling edge).
The phase
property is a number denoting the clock phase (CPHA), i.e. when the data is sampled on the MOSI (Master Out Slave In) and MISO (Master In Slave Out) pins of the SPI bus. When phase
is 0, MISO and MOSI data is sampled on the leading clock edge, whether it is rising (polarity 0) or falling (polarity 2). When phase
is 1, MISO and MOSI data is sampled on the trailing clock edge (whether it is rising or falling).
Often, SPI mode is being referred to as the combination of clock polarity and phase values:
- mode 0: polarity normal (value: 0), phase 0, data is sampled on leading (and rising) clock
- mode 1: polarity normal (value: 0), phase 1, data is sampled on trailing (and rising) clock
- mode 2: polarity inverse (value: 2), phase 0, data is sampled on leading (and falling) clock
- mode 3: polarity inverse (value: 2), phase 1, data is sampled on trailing (and falling) clock.
The sum of the polarity
and phase
property values provides the SPI mode (this is the reason polarity
takes the values 0 or 2).
The frameGap
property denotes the inter-frame gap in milliseconds on the platforms this is supported. The default value is 0.
The topology
property describes the SPI master-slave connection type. The values can be the following:
"full-duplex"
: the slave devices are connected to the master via separate SS (Slave Select) lines that each activate one slave device. Prior to communication, the master must activate one device by driving the corresponding SS to 0, then data transfer is bidirectional. This is the default value."read"
: the slaves are connected to the master as above. When a slave is activated, the data is read by the master."write"
: the slaves are connected to the master as above. The master can activate any number of SS lines and the data is written to all activated slaves."multiplexed"
: a numbern
of SS lines are connected to a decoder that can activate up to2^n - 1
number of slave devices. This works as full-duplex."daisy-chain"
: the master uses one SS and one SCLK (clock) line for all slaves. The MOSI line from the master goes to the first slave's MOSI pin, the MISO line of that slave goes to the MOSI pin of the next slave, and so forth. The last slave's MISO line is connected to the master's MISO pin. This is also bidirectional.
Writes buffer
buffer
using SPI to the slave identified by the target
argument, and reads from the slave device into a readBuffer
that is returned. The method runs the following steps:
- If
target
is not a number between 0 and 127, throwTypeError
and abort these steps. Depending on SPI topology, multiple slave devices may be possible to select. The value oftarget
may denote one single address, or an index of a device, or a multicast address (e.g.0b110
could enable slaves 2 and 1 and leave slave 0 off). - Check the valid combinations of
this.topology
anddirection
values from the following table. Ifdirection
isundefined
ornull
, let it take the default value.
this.topology |
direction |
---|---|
"read" |
"read" (default) |
"write" |
"write" (default) |
"full-duplex" |
"read-write" (default), "read" , "write" |
"multiplexed" |
"read-write" (default), "read" , "write" |
"daisy-chain" |
"read-write" (default), "read" , "write" |
In any other case, throw SystemError
and abort these steps.
- If
direction
allows reading, letreadBuffer
be a newBuffer
, otherwise letreadBuffer
benull
. - Request the underlying platform to perform the data transfer, i.e. write the bytes specified in
writeBuffer
to the specified target ifdirection
allows writing and read anotherBuffer
readBuffer
ifdirection
allows reading. The implementation maps the value oftarget
to the physical SS (slave select) pins on the board. If the operation fails, throwSystemError
and abort these steps. - Return
readBuffer
.
Closes the current SPI
transaction and cancels all pending operations. The implementation SHOULD NOT explicitly reset or change the existing SPI configuration.
try {
var spi = require("spi").open(); // open the default SPI bus
console.log("SPI bus " + spi.bus + " opened with bus speed " + spi.speed);
console.log("SPI mode: " + spi.mode);
console.log("Data bits: " + spi.bits);
console.log("Speed [MHz]: " + spi.speed);
console.log("MSB first: " + (spi.msbFirst ? "true" : "false"));
var buffer = spi.transceive(0, [1, 2, 3]);
console.log("From SPI device 0: " + buffer.toString());
spi.close();
} catch (err) {
console.log("SPI error: " + err.message);
}