* Move default config to .c file * Explicitly define PAL modes for boards with custom init * Unify spi_master headers
116 lines
3.5 KiB
C
116 lines
3.5 KiB
C
// Copyright 2025 QMK
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#pragma once
|
|
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
#include "gpio.h"
|
|
|
|
/**
|
|
* \file
|
|
*
|
|
* \defgroup spi_master SPI Master API
|
|
*
|
|
* \brief API to communicate with SPI devices.
|
|
* \{
|
|
*/
|
|
|
|
// Hardware SS pin is defined in the header so that user code can refer to it
|
|
#ifdef __AVR__
|
|
# if defined(__AVR_AT90USB162__) || defined(__AVR_ATmega16U2__) || defined(__AVR_ATmega32U2__) || defined(__AVR_ATmega16U4__) || defined(__AVR_ATmega32U4__) || defined(__AVR_AT90USB646__) || defined(__AVR_AT90USB647__) || defined(__AVR_AT90USB1286__) || defined(__AVR_AT90USB1287__)
|
|
# define SPI_SS_PIN B0
|
|
# elif defined(__AVR_ATmega32A__)
|
|
# define SPI_SS_PIN B4
|
|
# elif defined(__AVR_ATmega328P__) || defined(__AVR_ATmega328__)
|
|
# define SPI_SS_PIN B2
|
|
# endif
|
|
#endif
|
|
|
|
typedef int16_t spi_status_t;
|
|
|
|
#define SPI_STATUS_SUCCESS (0)
|
|
#define SPI_STATUS_ERROR (-1)
|
|
#define SPI_STATUS_TIMEOUT (-2)
|
|
|
|
#define SPI_TIMEOUT_IMMEDIATE (0)
|
|
#define SPI_TIMEOUT_INFINITE (0xFFFF)
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
typedef struct spi_start_config_t {
|
|
pin_t slave_pin;
|
|
bool lsb_first;
|
|
uint8_t mode;
|
|
uint16_t divisor;
|
|
bool cs_active_low;
|
|
} spi_start_config_t;
|
|
|
|
/**
|
|
* \brief Initialize the SPI driver. This function must be called only once, before any of the below functions can be called.
|
|
*/
|
|
void spi_init(void);
|
|
|
|
/**
|
|
* \brief Start an SPI transaction.
|
|
*
|
|
* \param slavePin The GPIO pin connected to the desired device's `SS` line.
|
|
* \param lsbFirst Determines the endianness of the transmission. If `true`, the least significant bit of each byte is sent first.
|
|
* \param mode The SPI mode to use.
|
|
* \param divisor The SPI clock divisor.
|
|
*
|
|
* \return `true` if the operation was successful, otherwise `false` if the supplied parameters are invalid or the SPI peripheral is already in use.
|
|
*/
|
|
bool spi_start(pin_t slavePin, bool lsbFirst, uint8_t mode, uint16_t divisor);
|
|
|
|
bool spi_start_extended(spi_start_config_t *start_config);
|
|
|
|
/**
|
|
* \brief Write a byte to the selected SPI device.
|
|
*
|
|
* \param data The byte to write.
|
|
*
|
|
* \return `SPI_STATUS_TIMEOUT` if the timeout period elapses, or `SPI_STATUS_SUCCESS`.
|
|
*/
|
|
spi_status_t spi_write(uint8_t data);
|
|
|
|
/**
|
|
* \brief Read a byte from the selected SPI device.
|
|
*
|
|
* \return `SPI_STATUS_TIMEOUT` if the timeout period elapses, otherwise the byte read from the device.
|
|
*/
|
|
spi_status_t spi_read(void);
|
|
|
|
/**
|
|
* \brief Send multiple bytes to the selected SPI device.
|
|
*
|
|
* \param data A pointer to the data to write from.
|
|
* \param length The number of bytes to write. Take care not to overrun the length of `data`.
|
|
*
|
|
* \return `SPI_STATUS_TIMEOUT` if the timeout period elapses, `SPI_STATUS_ERROR` if some other error occurs, otherwise `SPI_STATUS_SUCCESS`.
|
|
*/
|
|
spi_status_t spi_transmit(const uint8_t *data, uint16_t length);
|
|
|
|
/**
|
|
* \brief Receive multiple bytes from the selected SPI device.
|
|
*
|
|
* \param data A pointer to a buffer to read into.
|
|
* \param length The number of bytes to read. Take care not to overrun the length of `data`.
|
|
*
|
|
* \return `SPI_STATUS_TIMEOUT` if the timeout period elapses, `SPI_STATUS_ERROR` if some other error occurs, otherwise `SPI_STATUS_SUCCESS`.
|
|
*/
|
|
spi_status_t spi_receive(uint8_t *data, uint16_t length);
|
|
|
|
/**
|
|
* \brief End the current SPI transaction. This will deassert the slave select pin and reset the endianness, mode and divisor configured by `spi_start()`.
|
|
*
|
|
*/
|
|
void spi_stop(void);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
/** \} */
|