84 lines
3.2 KiB
C
84 lines
3.2 KiB
C
|
/**
|
||
|
* Copyright (c) 2020 Raspberry Pi (Trading) Ltd.
|
||
|
*
|
||
|
* SPDX-License-Identifier: BSD-3-Clause
|
||
|
*/
|
||
|
|
||
|
#include "pio_spi.h"
|
||
|
|
||
|
// Just 8 bit functions provided here. The PIO program supports any frame size
|
||
|
// 1...32, but the software to do the necessary FIFO shuffling is left as an
|
||
|
// exercise for the reader :)
|
||
|
//
|
||
|
// Likewise we only provide MSB-first here. To do LSB-first, you need to
|
||
|
// - Do shifts when reading from the FIFO, for general case n != 8, 16, 32
|
||
|
// - Do a narrow read at a one halfword or 3 byte offset for n == 16, 8
|
||
|
// in order to get the read data correctly justified.
|
||
|
|
||
|
void __time_critical_func(pio_spi_write8_blocking)(const pio_spi_inst_t *spi, const uint8_t *src, size_t len) {
|
||
|
size_t tx_remain = len, rx_remain = len;
|
||
|
// Do 8 bit accesses on FIFO, so that write data is byte-replicated. This
|
||
|
// gets us the left-justification for free (for MSB-first shift-out)
|
||
|
io_rw_8 *txfifo = (io_rw_8 *) &spi->pio->txf[spi->sm];
|
||
|
io_rw_8 *rxfifo = (io_rw_8 *) &spi->pio->rxf[spi->sm];
|
||
|
while (tx_remain || rx_remain) {
|
||
|
if (tx_remain && !pio_sm_is_tx_fifo_full(spi->pio, spi->sm)) {
|
||
|
*txfifo = *src++;
|
||
|
--tx_remain;
|
||
|
}
|
||
|
if (rx_remain && !pio_sm_is_rx_fifo_empty(spi->pio, spi->sm)) {
|
||
|
(void) *rxfifo;
|
||
|
--rx_remain;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void __time_critical_func(pio_spi_read8_blocking)(const pio_spi_inst_t *spi, uint8_t *dst, size_t len) {
|
||
|
size_t tx_remain = len, rx_remain = len;
|
||
|
io_rw_8 *txfifo = (io_rw_8 *) &spi->pio->txf[spi->sm];
|
||
|
io_rw_8 *rxfifo = (io_rw_8 *) &spi->pio->rxf[spi->sm];
|
||
|
while (tx_remain || rx_remain) {
|
||
|
if (tx_remain && !pio_sm_is_tx_fifo_full(spi->pio, spi->sm)) {
|
||
|
*txfifo = 0;
|
||
|
--tx_remain;
|
||
|
}
|
||
|
if (rx_remain && !pio_sm_is_rx_fifo_empty(spi->pio, spi->sm)) {
|
||
|
*dst++ = *rxfifo;
|
||
|
--rx_remain;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void __time_critical_func(pio_spi_write8_read8_blocking)(const pio_spi_inst_t *spi, uint8_t *src, uint8_t *dst,
|
||
|
size_t len) {
|
||
|
size_t tx_remain = len, rx_remain = len;
|
||
|
io_rw_8 *txfifo = (io_rw_8 *) &spi->pio->txf[spi->sm];
|
||
|
io_rw_8 *rxfifo = (io_rw_8 *) &spi->pio->rxf[spi->sm];
|
||
|
while (tx_remain || rx_remain) {
|
||
|
if (tx_remain && !pio_sm_is_tx_fifo_full(spi->pio, spi->sm)) {
|
||
|
*txfifo = *src++;
|
||
|
--tx_remain;
|
||
|
}
|
||
|
if (rx_remain && !pio_sm_is_rx_fifo_empty(spi->pio, spi->sm)) {
|
||
|
*dst++ = *rxfifo;
|
||
|
--rx_remain;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void __time_critical_func(pio_spi_repeat8_read8_blocking)(const pio_spi_inst_t *spi, uint8_t src, uint8_t *dst,
|
||
|
size_t len) {
|
||
|
size_t tx_remain = len, rx_remain = len;
|
||
|
io_rw_8 *txfifo = (io_rw_8 *) &spi->pio->txf[spi->sm];
|
||
|
io_rw_8 *rxfifo = (io_rw_8 *) &spi->pio->rxf[spi->sm];
|
||
|
while (tx_remain || rx_remain) {
|
||
|
if (tx_remain && !pio_sm_is_tx_fifo_full(spi->pio, spi->sm)) {
|
||
|
*txfifo = src;
|
||
|
--tx_remain;
|
||
|
}
|
||
|
if (rx_remain && !pio_sm_is_rx_fifo_empty(spi->pio, spi->sm)) {
|
||
|
*dst++ = *rxfifo;
|
||
|
--rx_remain;
|
||
|
}
|
||
|
}
|
||
|
}
|