2013-05-15 10:47:17 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2010 by Cristian Maglie <c.maglie@bug.st>
|
2014-08-01 14:38:27 +02:00
|
|
|
* Copyright (c) 2014 by Paul Stoffregen <paul@pjrc.com> (Transaction API)
|
2014-10-21 12:52:06 +02:00
|
|
|
* Copyright (c) 2014 by Matthijs Kooijman <matthijs@stdin.nl> (SPISettings AVR)
|
2014-10-21 12:57:41 +02:00
|
|
|
* Copyright (c) 2014 by Andrew J. Kroll <xxxajk@gmail.com> (atomicity fixes)
|
2013-05-15 10:47:17 +02:00
|
|
|
* SPI Master library for arduino.
|
|
|
|
*
|
|
|
|
* This file is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of either the GNU General Public License version 2
|
|
|
|
* or the GNU Lesser General Public License version 2.1, both as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*/
|
|
|
|
|
2013-11-21 12:35:31 +01:00
|
|
|
#include "SPI.h"
|
2013-05-15 10:47:17 +02:00
|
|
|
|
|
|
|
SPIClass SPI;
|
|
|
|
|
2014-10-21 12:57:41 +02:00
|
|
|
uint8_t SPIClass::initialized = 0;
|
2014-08-01 14:38:27 +02:00
|
|
|
uint8_t SPIClass::interruptMode = 0;
|
|
|
|
uint8_t SPIClass::interruptMask = 0;
|
|
|
|
uint8_t SPIClass::interruptSave = 0;
|
|
|
|
#ifdef SPI_TRANSACTION_MISMATCH_LED
|
|
|
|
uint8_t SPIClass::inTransactionFlag = 0;
|
|
|
|
#endif
|
2013-05-15 10:47:17 +02:00
|
|
|
|
2014-08-01 14:38:27 +02:00
|
|
|
void SPIClass::begin()
|
|
|
|
{
|
2014-10-21 12:57:41 +02:00
|
|
|
uint8_t sreg = SREG;
|
|
|
|
noInterrupts(); // Protect from a scheduler and prevent transactionBegin
|
|
|
|
if (!initialized) {
|
|
|
|
// Set SS to high so a connected chip will be "deselected" by default
|
|
|
|
digitalWrite(SS, HIGH);
|
2013-05-15 10:47:17 +02:00
|
|
|
|
2014-10-21 12:57:41 +02:00
|
|
|
// When the SS pin is set as OUTPUT, it can be used as
|
|
|
|
// a general purpose output port (it doesn't influence
|
|
|
|
// SPI operations).
|
|
|
|
pinMode(SS, OUTPUT);
|
2013-05-15 10:47:17 +02:00
|
|
|
|
2014-10-21 12:57:41 +02:00
|
|
|
// Warning: if the SS pin ever becomes a LOW INPUT then SPI
|
|
|
|
// automatically switches to Slave, so the data direction of
|
|
|
|
// the SS pin MUST be kept as OUTPUT.
|
|
|
|
SPCR |= _BV(MSTR);
|
|
|
|
SPCR |= _BV(SPE);
|
2013-05-15 10:47:17 +02:00
|
|
|
|
2014-10-21 12:57:41 +02:00
|
|
|
// Set direction register for SCK and MOSI pin.
|
|
|
|
// MISO pin automatically overrides to INPUT.
|
|
|
|
// By doing this AFTER enabling SPI, we avoid accidentally
|
|
|
|
// clocking in a single bit since the lines go directly
|
|
|
|
// from "input" to SPI control.
|
|
|
|
// http://code.google.com/p/arduino/issues/detail?id=888
|
|
|
|
pinMode(SCK, OUTPUT);
|
|
|
|
pinMode(MOSI, OUTPUT);
|
|
|
|
}
|
|
|
|
initialized++; // reference count
|
|
|
|
SREG = sreg;
|
2013-05-15 10:47:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SPIClass::end() {
|
2014-10-21 12:57:41 +02:00
|
|
|
uint8_t sreg = SREG;
|
|
|
|
noInterrupts(); // Protect from a scheduler and prevent transactionBegin
|
|
|
|
// Decrease the reference counter
|
|
|
|
if (initialized)
|
|
|
|
initialized--;
|
|
|
|
// If there are no more references disable SPI
|
|
|
|
if (!initialized) {
|
|
|
|
SPCR &= ~_BV(SPE);
|
|
|
|
interruptMode = 0;
|
|
|
|
#ifdef SPI_TRANSACTION_MISMATCH_LED
|
|
|
|
inTransactionFlag = 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
SREG = sreg;
|
2013-05-15 10:47:17 +02:00
|
|
|
}
|
|
|
|
|
2014-08-01 14:38:27 +02:00
|
|
|
// mapping of interrupt numbers to bits within SPI_AVR_EIMSK
|
|
|
|
#if defined(__AVR_ATmega32U4__)
|
|
|
|
#define SPI_INT0_MASK (1<<INT0)
|
|
|
|
#define SPI_INT1_MASK (1<<INT1)
|
|
|
|
#define SPI_INT2_MASK (1<<INT2)
|
|
|
|
#define SPI_INT3_MASK (1<<INT3)
|
|
|
|
#define SPI_INT4_MASK (1<<INT6)
|
|
|
|
#elif defined(__AVR_AT90USB646__) || defined(__AVR_AT90USB1286__)
|
|
|
|
#define SPI_INT0_MASK (1<<INT0)
|
|
|
|
#define SPI_INT1_MASK (1<<INT1)
|
|
|
|
#define SPI_INT2_MASK (1<<INT2)
|
|
|
|
#define SPI_INT3_MASK (1<<INT3)
|
|
|
|
#define SPI_INT4_MASK (1<<INT4)
|
|
|
|
#define SPI_INT5_MASK (1<<INT5)
|
|
|
|
#define SPI_INT6_MASK (1<<INT6)
|
|
|
|
#define SPI_INT7_MASK (1<<INT7)
|
|
|
|
#elif defined(EICRA) && defined(EICRB) && defined(EIMSK)
|
|
|
|
#define SPI_INT0_MASK (1<<INT4)
|
|
|
|
#define SPI_INT1_MASK (1<<INT5)
|
|
|
|
#define SPI_INT2_MASK (1<<INT0)
|
|
|
|
#define SPI_INT3_MASK (1<<INT1)
|
|
|
|
#define SPI_INT4_MASK (1<<INT2)
|
|
|
|
#define SPI_INT5_MASK (1<<INT3)
|
|
|
|
#define SPI_INT6_MASK (1<<INT6)
|
|
|
|
#define SPI_INT7_MASK (1<<INT7)
|
|
|
|
#else
|
|
|
|
#ifdef INT0
|
|
|
|
#define SPI_INT0_MASK (1<<INT0)
|
|
|
|
#endif
|
|
|
|
#ifdef INT1
|
|
|
|
#define SPI_INT1_MASK (1<<INT1)
|
|
|
|
#endif
|
|
|
|
#ifdef INT2
|
|
|
|
#define SPI_INT2_MASK (1<<INT2)
|
|
|
|
#endif
|
|
|
|
#endif
|
2013-05-15 10:47:17 +02:00
|
|
|
|
2014-08-01 14:38:27 +02:00
|
|
|
void SPIClass::usingInterrupt(uint8_t interruptNumber)
|
2013-05-15 10:47:17 +02:00
|
|
|
{
|
2014-08-01 14:38:27 +02:00
|
|
|
uint8_t mask;
|
2013-05-15 10:47:17 +02:00
|
|
|
|
2014-08-01 14:38:27 +02:00
|
|
|
if (interruptMode > 1) return;
|
|
|
|
|
|
|
|
noInterrupts();
|
|
|
|
switch (interruptNumber) {
|
|
|
|
#ifdef SPI_INT0_MASK
|
|
|
|
case 0: mask = SPI_INT0_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT1_MASK
|
|
|
|
case 1: mask = SPI_INT1_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT2_MASK
|
|
|
|
case 2: mask = SPI_INT2_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT3_MASK
|
|
|
|
case 3: mask = SPI_INT3_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT4_MASK
|
|
|
|
case 4: mask = SPI_INT4_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT5_MASK
|
|
|
|
case 5: mask = SPI_INT5_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT6_MASK
|
|
|
|
case 6: mask = SPI_INT6_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT7_MASK
|
|
|
|
case 7: mask = SPI_INT7_MASK; break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
interruptMode = 2;
|
|
|
|
interrupts();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
interruptMode = 1;
|
|
|
|
interruptMask |= mask;
|
|
|
|
interrupts();
|
2013-05-15 10:47:17 +02:00
|
|
|
}
|
|
|
|
|
2014-10-21 13:12:25 +02:00
|
|
|
void SPIClass::notUsingInterrupt(uint8_t interruptNumber)
|
|
|
|
{
|
|
|
|
// Once in mode 2 we can't go back to 0 without a proper reference count
|
|
|
|
if (interruptMode == 2)
|
|
|
|
return;
|
|
|
|
uint8_t mask = 0;
|
|
|
|
uint8_t sreg = SREG;
|
|
|
|
noInterrupts(); // Protect from a scheduler and prevent transactionBegin
|
|
|
|
switch (interruptNumber) {
|
|
|
|
#ifdef SPI_INT0_MASK
|
|
|
|
case 0: mask = SPI_INT0_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT1_MASK
|
|
|
|
case 1: mask = SPI_INT1_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT2_MASK
|
|
|
|
case 2: mask = SPI_INT2_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT3_MASK
|
|
|
|
case 3: mask = SPI_INT3_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT4_MASK
|
|
|
|
case 4: mask = SPI_INT4_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT5_MASK
|
|
|
|
case 5: mask = SPI_INT5_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT6_MASK
|
|
|
|
case 6: mask = SPI_INT6_MASK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef SPI_INT7_MASK
|
|
|
|
case 7: mask = SPI_INT7_MASK; break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
// this case can't be reached
|
|
|
|
}
|
|
|
|
interruptMask &= ~mask;
|
|
|
|
if (!interruptMask)
|
|
|
|
interruptMode = 0;
|
|
|
|
SREG = sreg;
|
|
|
|
}
|