Skip to content
This repository was archived by the owner on Aug 5, 2022. It is now read-only.

Latest commit

 

History

History
144 lines (110 loc) · 10.6 KB

spi.md

File metadata and controls

144 lines (110 loc) · 10.6 KB

SPI API

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.

The SPI API object

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.

The SPI open(options) method

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 the options.bus property is undefined, then set options.bus to 0.
  • Otherwise if options.bus is not a number between 0 and 127, throw TypeError and abort these steps.
  • If options.speed is undefined, let options.speed be 10. Otherwise, set options.speed to the closest matching value that is lower than options.speed and is supported by the platform.
  • If options.msbFirst is undefined, set it to true.
  • If options.bits is not in the set {1, 2, 4, 8, 16 }, then set it to the value 4.
  • If options.polarity is undefined, set it to 0. If it is not 0, set it to 2.
  • If options.phase is undefined, 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, set spi.frameGap to undefined.
  • If spi.topology is defined, use its value to initialize SPI on the underlying platform. If that fails, or it is not supported, set spi.topology to the default value "full-duplex". This will define the behaviour of transceive().
  • Request the underlying platform to initialize SPI with options. The implementation will use the board mapping from the value of options.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 an SPI object that controls the initialized SPI bus.
  • Return spi.

The SPI interface

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 to 2^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.

The transceive(target, buffer, direction) method

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, throw TypeError and abort these steps. Depending on SPI topology, multiple slave devices may be possible to select. The value of target 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 and direction values from the following table. If direction is undefined or null, 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, let readBuffer be a new Buffer, otherwise let readBuffer be null.
  • Request the underlying platform to perform the data transfer, i.e. write the bytes specified in writeBuffer to the specified target if direction allows writing and read another Buffer readBuffer if direction allows reading. The implementation maps the value of target to the physical SS (slave select) pins on the board. If the operation fails, throw SystemError and abort these steps.
  • Return readBuffer.

The close() method

Closes the current SPI transaction and cancels all pending operations. The implementation SHOULD NOT explicitly reset or change the existing SPI configuration.

Examples

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);
}