.. _SPI: =============================== Serial Peripheral Interface SPI =============================== The Serial Peripheral Interface is used for high-speed data exchange between the controller and some peripheral devices. There are several modes available. It consists of three signal lines plus one per peripheral device (called slave). All peripheral devices share the signal lines and use the selecting line exclusivly. For any given data transfer only one of the selecting lines must be at LOW level, all others must be HIGH. The basic data transfer operation is a data exchange of 8 bits. The sender transmits 8 bits and recieves 8 other bits in return from the communication partner. The basic forth word is ``c!@spi`` which translates to character store/fetch via SPI. It uses the hardware SPI module of the controller and thus the pre-defined pins of it. Basic Workflow -------------- The built-in SPI module uses a few pins to establish the communication with any device. To distinguish between different SPI attached devices a separate signalling line is used: slave select. Every slave device is connected with one such line. The SPI communication takes place with the one which signalling line is LOW. All other lines have to be HIGH. The setup of the slave select lines includes two steps: configuring as output and give it HIGH level when idle. Note that a pin that is configured as output will immediatly go to LOW level. This may disturb a SPI slave so after configuring the line direction the port has to go to HIGH exllicitly. When all slave select lines are configured, the remaining SPI setup can take place .. code-block:: forth \ requires bitnames, quotations and spi loaded > PORTB 0 portpin: dev.ss \ define hardware > dev.ss to spi.ss \ assign ss pin to lib > spi.ss is_output \ short LOW pulse > spi.ss high \ de-select slave > +spi \ turn on SPI module Data Exchange ------------- Any SPI transfer starts with pulling the slave select line LOW. Now any number of read/write operations may take place. To stop an exchange, the slace select line goes back to HIGH. This signals the slave device that the communication has ended and it usually goes back to a state that awaits a new commication. The basic ``c!@spi`` is the building block for the next words .. code-block:: forth \ single byte transfers : c@spi ( -- c ) 0 c!@spi ; : c!cpi ( c -- ) c!@spi drop ; \ read len bytes from SPI and store \ them starting at addr : n@spi ( addr len -- ) 0 ?do c@spi over c! 1+ loop drop ; \ write len bytes from addr to SPI : n!spi ( addr len --- ) 0 ?do dup c@ c!spi 1+ loop drop ; The file :file:`core/words/n-spi.asm` contains speed optimized implementations of the ``n@spi`` and ``n!spi`` words. SD-Cards/MMC ------------ MMC and SD-Cards have an SPI mode which is slower than the usual mode used on PC's but is simpler to program. .. code-block:: forth \ standard stuff, only if not already uploaded #require postpone.frt #require marker.frt #require bitnames.frt \ board definitions #include netio.frt \ SPI library #require quotations.frt #require 2rvalue.frt #include spi.frt \ SD Card specific #include mmc.frt The include order of the file is important. The board specfic definitions need to define the words ``+spi``, ``-spi`` for global SPI port setup. In addition the commands ``+mmc`` and ``-mmc`` are used to perform a single communication with the device. The portpin definitions are not used elsewhere but should match the hardware. .. code-block:: forth PORTB 0 portpin: sdcard sdcard to spi.ss : +mmc sdcard low ; : -mmc sdcard high ; After successfully loading these files, the command ``mmc_init`` initializes the communication and enables the remaining access. It has to be issued every time the card has changed. .. code-block:: forth (ATmega640)> mmc_init ok (ATmega640)> mmc_CID . cr 10 0 mmc. 0 1 50 41 53 30 32 47 46 12 39 B6 28 D6 0 B4 99 ok (ATmega640)>