1
0
mirror of https://github.com/arduino/Arduino.git synced 2025-01-10 00:46:09 +01:00
Arduino/hardware/arduino/sam/libraries/SPI/SPI.h

149 lines
4.6 KiB
C
Raw Normal View History

2011-06-21 00:03:22 +02:00
/*
2015-02-05 16:47:22 +01:00
* Copyright (c) 2010 by Cristian Maglie <c.maglie@arduino.cc>
2014-08-01 14:57:13 +02:00
* Copyright (c) 2014 by Paul Stoffregen <paul@pjrc.com> (Transaction API)
2011-06-21 00:03:22 +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.
*/
#ifndef _SPI_H_INCLUDED
#define _SPI_H_INCLUDED
#include "variant.h"
2011-06-21 00:03:22 +02:00
#include <stdio.h>
// SPI_HAS_TRANSACTION means SPI has
// - beginTransaction()
// - endTransaction()
// - usingInterrupt()
// - SPISetting(clock, bitOrder, dataMode)
#define SPI_HAS_TRANSACTION 1
// SPI_HAS_EXTENDED_CS_PIN_HANDLING means SPI has automatic
// CS pin handling and provides the following methods:
// - begin(pin)
// - end(pin)
// - setBitOrder(pin, bitorder)
// - setDataMode(pin, datamode)
// - setClockDivider(pin, clockdiv)
// - transfer(pin, data, SPI_LAST/SPI_CONTINUE)
// - beginTransaction(pin, SPISettings settings) (if transactions are available)
#define SPI_HAS_EXTENDED_CS_PIN_HANDLING 1
2012-04-24 18:43:33 +02:00
#define SPI_MODE0 0x02
#define SPI_MODE1 0x00
#define SPI_MODE2 0x03
#define SPI_MODE3 0x01
2011-06-21 00:03:22 +02:00
2012-05-18 01:30:54 +02:00
enum SPITransferMode {
SPI_CONTINUE,
SPI_LAST
};
2014-08-01 14:57:13 +02:00
class SPISettings {
public:
SPISettings(uint32_t clock, BitOrder bitOrder, uint8_t dataMode) {
if (__builtin_constant_p(clock)) {
init_AlwaysInline(clock, bitOrder, dataMode);
} else {
init_MightInline(clock, bitOrder, dataMode);
}
}
2014-09-10 16:37:23 +02:00
SPISettings() { init_AlwaysInline(4000000, MSBFIRST, SPI_MODE0); }
2014-08-01 14:57:13 +02:00
private:
void init_MightInline(uint32_t clock, BitOrder bitOrder, uint8_t dataMode) {
init_AlwaysInline(clock, bitOrder, dataMode);
}
2014-09-10 16:37:23 +02:00
void init_AlwaysInline(uint32_t clock, BitOrder bitOrder, uint8_t dataMode) __attribute__((__always_inline__)) {
2014-08-01 14:57:13 +02:00
border = bitOrder;
2014-09-10 16:37:23 +02:00
uint8_t div;
if (clock < (F_CPU / 255)) {
div = 255;
} else if (clock >= (F_CPU / 2)) {
div = 2;
2014-08-01 14:57:13 +02:00
} else {
2014-09-10 16:37:23 +02:00
div = (F_CPU / (clock + 1)) + 1;
2014-08-01 14:57:13 +02:00
}
config = (dataMode & 3) | SPI_CSR_CSAAT | SPI_CSR_SCBR(div) | SPI_CSR_DLYBCT(1);
}
uint32_t config;
BitOrder border;
friend class SPIClass;
};
2011-12-02 16:29:50 +01:00
class SPIClass {
public:
2012-05-18 01:30:54 +02:00
SPIClass(Spi *_spi, uint32_t _id, void(*_initCb)(void));
2011-06-21 00:03:22 +02:00
// Transfer functions
byte transfer(byte _pin, uint8_t _data, SPITransferMode _mode = SPI_LAST);
2015-11-04 21:49:08 +01:00
uint16_t transfer16(byte _pin, uint16_t _data, SPITransferMode _mode = SPI_LAST);
void transfer(byte _pin, void *_buf, size_t _count, SPITransferMode _mode = SPI_LAST);
// Transfer functions on default pin BOARD_SPI_DEFAULT_SS
2012-06-09 15:02:51 +02:00
byte transfer(uint8_t _data, SPITransferMode _mode = SPI_LAST) { return transfer(BOARD_SPI_DEFAULT_SS, _data, _mode); }
2015-11-04 21:49:08 +01:00
uint16_t transfer16(uint16_t _data, SPITransferMode _mode = SPI_LAST) { return transfer16(BOARD_SPI_DEFAULT_SS, _data, _mode); }
void transfer(void *_buf, size_t _count, SPITransferMode _mode = SPI_LAST) { transfer(BOARD_SPI_DEFAULT_SS, _buf, _count, _mode); }
2011-06-21 00:03:22 +02:00
2014-08-01 14:57:13 +02:00
// Transaction Functions
void usingInterrupt(uint8_t interruptNumber);
2014-09-10 15:21:18 +02:00
void beginTransaction(SPISettings settings) { beginTransaction(BOARD_SPI_DEFAULT_SS, settings); }
2014-08-01 14:57:13 +02:00
void beginTransaction(uint8_t pin, SPISettings settings);
void endTransaction(void);
2011-06-21 00:03:22 +02:00
2014-08-01 14:57:13 +02:00
// SPI Configuration methods
2011-12-02 16:29:50 +01:00
void attachInterrupt(void);
void detachInterrupt(void);
2011-06-21 00:03:22 +02:00
2012-06-09 15:02:51 +02:00
void begin(void);
2011-12-02 16:29:50 +01:00
void end(void);
2011-06-21 00:03:22 +02:00
// Attach/Detach pin to/from SPI controller
void begin(uint8_t _pin);
void end(uint8_t _pin);
// These methods sets a parameter on a single pin
void setBitOrder(uint8_t _pin, BitOrder);
void setDataMode(uint8_t _pin, uint8_t);
void setClockDivider(uint8_t _pin, uint8_t);
// These methods sets the same parameters but on default pin BOARD_SPI_DEFAULT_SS
void setBitOrder(BitOrder _order) { setBitOrder(BOARD_SPI_DEFAULT_SS, _order); };
2012-06-09 15:02:51 +02:00
void setDataMode(uint8_t _mode) { setDataMode(BOARD_SPI_DEFAULT_SS, _mode); };
void setClockDivider(uint8_t _div) { setClockDivider(BOARD_SPI_DEFAULT_SS, _div); };
2011-12-02 16:29:50 +01:00
private:
void init();
2011-12-02 16:29:50 +01:00
Spi *spi;
uint32_t id;
BitOrder bitOrder[SPI_CHANNELS_NUM];
uint32_t divider[SPI_CHANNELS_NUM];
uint32_t mode[SPI_CHANNELS_NUM];
void (*initCb)(void);
bool initialized;
2014-08-02 01:44:47 +02:00
uint8_t interruptMode; // 0=none, 1-15=mask, 16=global
uint8_t interruptSave; // temp storage, to restore state
uint32_t interruptMask[4];
2011-12-02 16:29:50 +01:00
};
#if SPI_INTERFACES_COUNT > 0
2012-05-18 01:30:54 +02:00
extern SPIClass SPI;
2011-12-02 16:29:50 +01:00
#endif
2011-06-21 00:03:22 +02:00
2014-08-01 14:57:13 +02:00
// For compatibility with sketches designed for AVR @ 16 MHz
// New programs should use SPI.beginTransaction to set the SPI clock
#define SPI_CLOCK_DIV2 11
#define SPI_CLOCK_DIV4 21
#define SPI_CLOCK_DIV8 42
#define SPI_CLOCK_DIV16 84
#define SPI_CLOCK_DIV32 168
#define SPI_CLOCK_DIV64 255
#define SPI_CLOCK_DIV128 255
2011-06-21 00:03:22 +02:00
#endif