mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 07:52:44 +01:00
890 lines
29 KiB
C
890 lines
29 KiB
C
/*
|
||
* Copyright (C) 2021 Otto-von-Guericke Universität Magdeburg
|
||
*
|
||
* This file is subject to the terms and conditions of the GNU Lesser
|
||
* General Public License v2.1. See the file LICENSE in the top level
|
||
* directory for more details.
|
||
*/
|
||
|
||
/**
|
||
* @ingroup cpu_rpx0xx
|
||
* @ingroup drivers_periph_pio
|
||
* @{
|
||
*
|
||
* @file
|
||
* @brief PIO implementation for the RPX0XX
|
||
* @details The RPX0XX has 2 PIOs with 4 state machines each
|
||
*
|
||
* @author Fabian Hüßler <fabian.huessler@ovgu.de>
|
||
*
|
||
* @}
|
||
*/
|
||
|
||
#include <assert.h>
|
||
#include <errno.h>
|
||
#include <stdbool.h>
|
||
#include "periph_conf.h"
|
||
#include "periph_cpu.h"
|
||
#include "bitarithm.h"
|
||
#include "io_reg.h"
|
||
#include "pio/pio.h"
|
||
#include "periph/pio/i2c.h"
|
||
#define ENABLE_DEBUG 0
|
||
#include "debug.h"
|
||
|
||
/* ISR vectors */
|
||
static const pio_isr_vec_t *_isr[PIO_NUMOF][PIO_SM_NUMOF];
|
||
/* ISR vectors for shared interrupts 0-3 raised by any state machine */
|
||
static const pio_isr_sm_vec_t *_isr_sm[PIO_NUMOF][4];
|
||
/* locked state machines mask */
|
||
static volatile uint32_t _sm_mask[PIO_NUMOF];
|
||
/* occupied instruction memory mask */
|
||
static volatile uint32_t _instr_mask[PIO_NUMOF];
|
||
|
||
/* behaviour is only atomic if only one CPU core is used */
|
||
static inline bool _atomic_set_mask_u32(volatile uint32_t *dst, uint32_t msk)
|
||
{
|
||
bool exch = false;
|
||
unsigned irq = irq_disable();
|
||
if (!(*dst & msk)) {
|
||
*dst |= msk;
|
||
exch = true;
|
||
}
|
||
irq_restore(irq);
|
||
return exch;
|
||
}
|
||
|
||
/* behaviour is only atomic if only one CPU core is used */
|
||
static inline bool _atomic_clear_mask_u32(volatile uint32_t *dst, uint32_t msk)
|
||
{
|
||
bool exch = false;
|
||
unsigned irq = irq_disable();
|
||
if (*dst & msk) {
|
||
*dst &= ~msk;
|
||
exch = true;
|
||
}
|
||
irq_restore(irq);
|
||
return exch;
|
||
}
|
||
|
||
static void _irq(pio_t pio, pio_irq_line_t irq, uint32_t status)
|
||
{
|
||
(void)irq;
|
||
if ((status & PIO0_IRQ0_INTS_SM0_RXNEMPTY_Msk) && _isr[pio][0]
|
||
&& _isr[pio][0]->rx_ready) {
|
||
_isr[pio][0]->rx_ready(pio, 0);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM1_RXNEMPTY_Msk) && _isr[pio][1]
|
||
&& _isr[pio][1]->rx_ready) {
|
||
_isr[pio][1]->rx_ready(pio, 1);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM2_RXNEMPTY_Msk) && _isr[pio][2]
|
||
&& _isr[pio][2]->rx_ready) {
|
||
_isr[pio][2]->rx_ready(pio, 2);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM3_RXNEMPTY_Msk) && _isr[pio][3]
|
||
&& _isr[pio][3]->rx_ready) {
|
||
_isr[pio][3]->rx_ready(pio, 3);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM0_TXNFULL_Msk) && _isr[pio][0]
|
||
&& _isr[pio][0]->tx_ready) {
|
||
_isr[pio][0]->tx_ready(pio, 0);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM1_TXNFULL_Msk) && _isr[pio][1]
|
||
&& _isr[pio][1]->tx_ready) {
|
||
_isr[pio][1]->tx_ready(pio, 1);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM2_TXNFULL_Msk) && _isr[pio][2]
|
||
&& _isr[pio][2]->tx_ready) {
|
||
_isr[pio][2]->tx_ready(pio, 2);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM3_TXNFULL_Msk) && _isr[pio][3]
|
||
&& _isr[pio][3]->tx_ready) {
|
||
_isr[pio][3]->tx_ready(pio, 3);
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM0_Msk)) {
|
||
assert(pio_irq_get(pio) & PIO_IRQ_MASK(0));
|
||
if (_isr_sm[pio][0] && _isr_sm[pio][0]->sm) {
|
||
_isr_sm[pio][0]->sm(pio, 0);
|
||
}
|
||
pio_irq_clear(pio, PIO_IRQ_MASK(0));
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM1_Msk)) {
|
||
assert(pio_irq_get(pio) & PIO_IRQ_MASK(1));
|
||
if (_isr_sm[pio][1] && _isr_sm[pio][1]->sm) {
|
||
_isr_sm[pio][1]->sm(pio, 1);
|
||
}
|
||
pio_irq_clear(pio, PIO_IRQ_MASK(1));
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM2_Msk)) {
|
||
assert(pio_irq_get(pio) & PIO_IRQ_MASK(2));
|
||
if (_isr_sm[pio][2] && _isr_sm[pio][2]->sm) {
|
||
_isr_sm[pio][2]->sm(pio, 2);
|
||
}
|
||
pio_irq_clear(pio, PIO_IRQ_MASK(2));
|
||
}
|
||
if ((status & PIO0_IRQ0_INTS_SM3_Msk)) {
|
||
assert(pio_irq_get(pio) & PIO_IRQ_MASK(3));
|
||
if (_isr_sm[pio][3] && _isr_sm[pio][3]->sm) {
|
||
_isr_sm[pio][3]->sm(pio, 3);
|
||
}
|
||
pio_irq_clear(pio, PIO_IRQ_MASK(3));
|
||
}
|
||
cortexm_isr_end();
|
||
}
|
||
|
||
void pio_init(pio_t pio)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
NVIC_EnableIRQ(pio_config[pio].irqn0);
|
||
NVIC_EnableIRQ(pio_config[pio].irqn1);
|
||
}
|
||
|
||
void pio_start_programs(void)
|
||
{
|
||
if (IS_USED(MODULE_PIO_AUTOSTART_I2C)) {
|
||
pio_i2c_start_programs();
|
||
}
|
||
}
|
||
|
||
pio_sm_t pio_sm_lock(pio_t pio)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
uint32_t pos = 0;
|
||
bool exch;
|
||
while ((pos < PIO_SM_NUMOF) && !(exch = _atomic_set_mask_u32(&_sm_mask[pio], 1u << pos))) {
|
||
pos++;
|
||
}
|
||
return exch ? (pio_sm_t)pos : -1;
|
||
}
|
||
|
||
void pio_sm_unlock(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
_atomic_clear_mask_u32(&_sm_mask[pio], (1u << sm));
|
||
}
|
||
|
||
void pio_sm_start(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
io_reg_atomic_set(&dev->CTRL,
|
||
((1u << sm) << PIO0_CTRL_CLKDIV_RESTART_Pos) |
|
||
((1u << sm) << PIO0_CTRL_SM_ENABLE_Pos));
|
||
}
|
||
|
||
void pio_sm_stop(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
io_reg_atomic_clear(&dev->CTRL, (1u << sm) << PIO0_CTRL_SM_ENABLE_Pos);
|
||
}
|
||
|
||
int pio_alloc_program(pio_t pio, pio_program_t *prog)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
if (!prog->instr_numof) {
|
||
return 0;
|
||
}
|
||
if (prog->instr_numof > PIO_INSTR_NUMOF) {
|
||
return -ENOMEM;
|
||
}
|
||
/* don´t ((uint32_t)1 << 32) */
|
||
uint32_t mask = ((((uint32_t)1 << (prog->instr_numof - 1)) - 1) << 1) | 1;
|
||
bool exch = false;
|
||
unsigned i = 0;
|
||
while ((i <= PIO_INSTR_NUMOF - prog->instr_numof) &&
|
||
!(exch = _atomic_set_mask_u32(&_instr_mask[pio], mask << i))) {
|
||
i++;
|
||
}
|
||
if (!exch) {
|
||
return -ENOMEM;
|
||
}
|
||
prog->location = i;
|
||
prog->written = false;
|
||
return 0;
|
||
}
|
||
|
||
int pio_alloc_program_sm_lock_any(pio_t *pio_ptr, pio_sm_t *sm_ptr, pio_program_t *program)
|
||
{
|
||
pio_t pio;
|
||
pio_sm_t sm = -1;
|
||
int alloc = 0;
|
||
if (!pio_ptr || !sm_ptr || !program) {
|
||
return -EFAULT;
|
||
}
|
||
for (pio = 0; pio < PIO_NUMOF; pio++) {
|
||
if ((alloc = pio_alloc_program(pio, program))) {
|
||
continue;
|
||
}
|
||
if ((sm = pio_sm_lock(pio)) < 0) {
|
||
pio_free_program(pio, program);
|
||
continue;
|
||
}
|
||
break;
|
||
}
|
||
if (sm >= 0) {
|
||
*pio_ptr = pio;
|
||
*sm_ptr = sm;
|
||
return 0;
|
||
}
|
||
return alloc ? alloc : (int)sm;
|
||
}
|
||
|
||
void pio_free_program(pio_t pio, pio_program_t *prog)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
if (!prog->instr_numof || prog->instr_numof > PIO_INSTR_NUMOF ||
|
||
prog->location < 0 || prog->location >= PIO_INSTR_NUMOF) {
|
||
return;
|
||
}
|
||
/* don´t ((uint32_t)1 << 32) */
|
||
uint32_t mask = ((((uint32_t)1 << (prog->instr_numof - 1)) - 1) << 1) | 1;
|
||
mask <<= prog->location;
|
||
_atomic_clear_mask_u32(&_instr_mask[pio], mask);
|
||
prog->location = PIO_PROGRAM_NOT_LOADED;
|
||
prog->written = false;
|
||
}
|
||
|
||
int pio_sm_exec(pio_t pio, pio_sm_t sm, pio_instr_t inst) {
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
if (ctrl->execctrl & PIO0_SM0_EXECCTRL_EXEC_STALLED_Msk) {
|
||
return -EBUSY;
|
||
}
|
||
ctrl->instr = inst;
|
||
return 0;
|
||
}
|
||
|
||
void pio_sm_exec_block(pio_t pio, pio_sm_t sm, pio_instr_t inst) {
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
while (ctrl->execctrl & PIO0_SM0_EXECCTRL_EXEC_STALLED_Msk) {}
|
||
ctrl->instr = inst;
|
||
}
|
||
|
||
int pio_write_program(pio_t pio, pio_program_t *prog, const pio_instr_t *prog_instr)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
if (prog->location < 0 || prog->location > PIO_INSTR_NUMOF) {
|
||
return -EFAULT;
|
||
}
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
for (unsigned i = 0; i < prog->instr_numof; i++) {
|
||
pio_instr_t inst = prog_instr[i];
|
||
/* JMPs are absolute addresses and must be adjusted to the program offset */
|
||
if ((inst & PIO_INST_JMP_MASK) == PIO_INST_JMP) {
|
||
inst += prog->location;
|
||
}
|
||
(&dev->INSTR_MEM0 + prog->location)[i] = inst;
|
||
}
|
||
prog->written = true;
|
||
return 0;
|
||
}
|
||
|
||
void pio_sm_reset(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
ctrl->clkdiv = 1u << PIO0_SM0_CLKDIV_INT_Pos;
|
||
ctrl->execctrl = 0x1f << PIO0_SM0_EXECCTRL_WRAP_TOP_Pos;
|
||
ctrl->shiftctrl = (1u << PIO0_SM0_SHIFTCTRL_OUT_SHIFTDIR_Pos) |
|
||
(1u << PIO0_SM0_SHIFTCTRL_IN_SHIFTDIR_Pos);
|
||
ctrl->pinctrl = 5u << PIO0_SM0_PINCTRL_SET_COUNT_Pos;
|
||
}
|
||
|
||
void pio_sm_restart(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
io_reg_atomic_set(&dev->CTRL,
|
||
((1u << sm) << PIO0_CTRL_SM_RESTART_Pos) |
|
||
((1u << sm) << PIO0_CTRL_CLKDIV_RESTART_Pos));
|
||
}
|
||
|
||
void pio_set_isr_vec(pio_t pio, pio_sm_t sm, const pio_isr_vec_t *vec)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
unsigned irq = irq_disable();
|
||
/* something is wrong when something overrides an existing isr */
|
||
assert(!_isr[pio][sm] || _isr[pio][sm] == vec);
|
||
_isr[pio][sm] = vec;
|
||
irq_restore(irq);
|
||
}
|
||
|
||
void pio_set_isr_sm_vec(pio_t pio, unsigned irq_num, const pio_isr_sm_vec_t *vec)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert(irq_num < 4); /* irq 0 to 3 are routed to NVIC */
|
||
unsigned irq = irq_disable();
|
||
/* something is wrong when something overrides an existing isr */
|
||
assert(!_isr_sm[pio][irq_num] || _isr_sm[pio][irq_num] == vec);
|
||
_isr_sm[pio][irq_num] = vec;
|
||
irq_restore(irq);
|
||
}
|
||
|
||
void pio_sm_set_out_pins(pio_t pio, pio_sm_t sm, gpio_t pin_base, unsigned pin_count)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->pinctrl,
|
||
(pin_base << PIO0_SM0_PINCTRL_OUT_BASE_Pos) |
|
||
(pin_count << PIO0_SM0_PINCTRL_OUT_COUNT_Pos),
|
||
PIO0_SM0_PINCTRL_OUT_BASE_Msk | PIO0_SM0_PINCTRL_OUT_COUNT_Msk);
|
||
}
|
||
|
||
void pio_sm_set_in_pins(pio_t pio, pio_sm_t sm, gpio_t pin_base)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->pinctrl,
|
||
pin_base << PIO0_SM0_PINCTRL_IN_BASE_Pos,
|
||
PIO0_SM0_PINCTRL_IN_BASE_Msk);
|
||
}
|
||
|
||
void pio_sm_set_set_pins(pio_t pio, pio_sm_t sm, gpio_t pin_base, unsigned pin_count)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->pinctrl,
|
||
(pin_base << PIO0_SM0_PINCTRL_SET_BASE_Pos) |
|
||
(pin_count << PIO0_SM0_PINCTRL_SET_COUNT_Pos),
|
||
PIO0_SM0_PINCTRL_SET_BASE_Msk | PIO0_SM0_PINCTRL_SET_COUNT_Msk);
|
||
}
|
||
|
||
void pio_sm_set_sideset_pins(pio_t pio, pio_sm_t sm, gpio_t pin_base)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->pinctrl,
|
||
(pin_base << PIO0_SM0_PINCTRL_SIDESET_BASE_Pos),
|
||
PIO0_SM0_PINCTRL_SIDESET_BASE_Msk);
|
||
}
|
||
|
||
void pio_sm_set_sideset_count(pio_t pio, pio_sm_t sm, unsigned pin_count, bool enable)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->pinctrl,
|
||
((pin_count + !!enable) << PIO0_SM0_PINCTRL_SIDESET_COUNT_Pos),
|
||
PIO0_SM0_PINCTRL_SIDESET_COUNT_Msk);
|
||
io_reg_write_dont_corrupt(&ctrl->execctrl,
|
||
(!!enable) << PIO0_SM0_EXECCTRL_SIDE_EN_Pos,
|
||
PIO0_SM0_EXECCTRL_SIDE_EN_Msk);
|
||
}
|
||
|
||
void pio_sm_set_sideset_target(pio_t pio, pio_sm_t sm, bool pindir)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->execctrl,
|
||
(!!pindir) << PIO0_SM0_EXECCTRL_SIDE_PINDIR_Pos,
|
||
PIO0_SM0_EXECCTRL_SIDE_PINDIR_Msk);
|
||
}
|
||
|
||
pio_sm_clkdiv_t pio_sm_clkdiv(uint32_t f_hz)
|
||
{
|
||
uint32_t div = CLOCK_CORECLOCK / f_hz;
|
||
uint32_t frac = (((uint64_t)100 * CLOCK_CORECLOCK) / f_hz) - (100 * div);
|
||
|
||
assert(div > 0);
|
||
assert((div < PIO_SM_CLKDIV_MAX) || (div == PIO_SM_CLKDIV_MAX && frac == 0));
|
||
|
||
if (div >= PIO_SM_CLKDIV_MAX) {
|
||
div = 0;
|
||
frac = 0;
|
||
}
|
||
return (pio_sm_clkdiv_t){ .div = div, .frac_100 = frac };
|
||
}
|
||
|
||
void pio_sm_set_clkdiv(pio_t pio, pio_sm_t sm, pio_sm_clkdiv_t clk)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
if (clk.div == 0) {
|
||
clk.frac_100 = 0;
|
||
/* 0 --> 65536 */
|
||
}
|
||
if (clk.frac_100 >= 100) {
|
||
clk.frac_100 /= 10;
|
||
}
|
||
uint32_t frac = ((((uint16_t)clk.frac_100) * 256) + 50) / 100;
|
||
uint32_t val = (clk.div << PIO0_SM0_CLKDIV_INT_Pos) | (frac << PIO0_SM0_CLKDIV_FRAC_Pos);
|
||
uint32_t msk = PIO0_SM0_CLKDIV_INT_Msk | PIO0_SM0_CLKDIV_FRAC_Msk;
|
||
io_reg_write_dont_corrupt(&ctrl->clkdiv, val, msk);
|
||
}
|
||
|
||
void pio_sm_clkdiv_restart(pio_t pio, unsigned sm_mask)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
io_reg_atomic_set(&dev->CTRL, (sm_mask & PIO_SM_ALL) << PIO0_CTRL_CLKDIV_RESTART_Pos);
|
||
}
|
||
|
||
void pio_sm_set_wrap(pio_t pio, pio_sm_t sm, unsigned prog_loc, uint8_t top, uint8_t bottom)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->execctrl,
|
||
((prog_loc + bottom) << PIO0_SM0_EXECCTRL_WRAP_BOTTOM_Pos) |
|
||
((prog_loc + top) << PIO0_SM0_EXECCTRL_WRAP_TOP_Pos),
|
||
PIO0_SM0_EXECCTRL_WRAP_BOTTOM_Msk | PIO0_SM0_EXECCTRL_WRAP_TOP_Msk);
|
||
}
|
||
|
||
void pio_sm_set_jmp_pin(pio_t pio, pio_sm_t sm, gpio_t pin)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->execctrl,
|
||
pin << PIO0_SM0_EXECCTRL_JMP_PIN_Pos,
|
||
PIO0_SM0_EXECCTRL_JMP_PIN_Msk);
|
||
}
|
||
|
||
void pio_sm_set_in_shift(pio_t pio, pio_sm_t sm, bool right, bool autopush, unsigned threshold)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->shiftctrl,
|
||
((!!right) << PIO0_SM0_SHIFTCTRL_IN_SHIFTDIR_Pos) |
|
||
((!!autopush) << PIO0_SM0_SHIFTCTRL_AUTOPUSH_Pos) |
|
||
((threshold % 32) << PIO0_SM0_SHIFTCTRL_PUSH_THRESH_Pos),
|
||
PIO0_SM0_SHIFTCTRL_IN_SHIFTDIR_Msk |
|
||
PIO0_SM0_SHIFTCTRL_AUTOPUSH_Msk |
|
||
PIO0_SM0_SHIFTCTRL_PUSH_THRESH_Msk);
|
||
}
|
||
|
||
void pio_sm_set_out_shift(pio_t pio, pio_sm_t sm, bool right, bool autopull, unsigned threshold)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->shiftctrl,
|
||
((!!right) << PIO0_SM0_SHIFTCTRL_OUT_SHIFTDIR_Pos) |
|
||
((!!autopull) << PIO0_SM0_SHIFTCTRL_AUTOPULL_Pos) |
|
||
((threshold % 32) << PIO0_SM0_SHIFTCTRL_PULL_THRESH_Pos),
|
||
PIO0_SM0_SHIFTCTRL_OUT_SHIFTDIR_Msk |
|
||
PIO0_SM0_SHIFTCTRL_AUTOPULL_Msk |
|
||
PIO0_SM0_SHIFTCTRL_PULL_THRESH_Msk);
|
||
}
|
||
|
||
void pio_sm_set_fifo_join_rx(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->shiftctrl,
|
||
1u << PIO0_SM0_SHIFTCTRL_FJOIN_RX_Pos,
|
||
PIO0_SM0_SHIFTCTRL_FJOIN_RX_Msk | PIO0_SM0_SHIFTCTRL_FJOIN_TX_Msk);
|
||
}
|
||
|
||
void pio_sm_set_fifo_join_tx(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_write_dont_corrupt(&ctrl->shiftctrl,
|
||
1u << PIO0_SM0_SHIFTCTRL_FJOIN_TX_Pos,
|
||
PIO0_SM0_SHIFTCTRL_FJOIN_RX_Msk | PIO0_SM0_SHIFTCTRL_FJOIN_TX_Msk);
|
||
}
|
||
|
||
void pio_sm_reset_fifos(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_atomic_clear(&ctrl->shiftctrl,
|
||
(1u << PIO0_SM0_SHIFTCTRL_FJOIN_TX_Pos) |
|
||
(1u << PIO0_SM0_SHIFTCTRL_FJOIN_RX_Pos));
|
||
}
|
||
|
||
void pio_sm_clear_fifos(pio_t pio, pio_sm_t sm)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
io_reg_atomic_xor(&ctrl->shiftctrl, (1u << PIO0_SM0_SHIFTCTRL_FJOIN_RX_Pos));
|
||
io_reg_atomic_xor(&ctrl->shiftctrl, (1u << PIO0_SM0_SHIFTCTRL_FJOIN_RX_Pos));
|
||
}
|
||
|
||
void pio_irq_enable(pio_t pio, pio_irq_line_t irq, pio_irq_source_t irq_mask)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)irq < PIO_IRQ_LINE_NUMOF);
|
||
|
||
uint32_t val = irq_mask & PIO_IRQ_ALL;
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
if (irq == PIO_IRQ_LINE_0) {
|
||
io_reg_atomic_set(&dev->IRQ0_INTE, val);
|
||
}
|
||
else if (irq == PIO_IRQ_LINE_1) {
|
||
io_reg_atomic_set(&dev->IRQ1_INTE, val);
|
||
}
|
||
}
|
||
|
||
void pio_irq_disable(pio_t pio, pio_irq_line_t irq, pio_irq_source_t irq_mask)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)irq < PIO_IRQ_LINE_NUMOF);
|
||
|
||
uint32_t val = irq_mask & PIO_IRQ_ALL;
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
if (irq == PIO_IRQ_LINE_0) {
|
||
io_reg_atomic_clear(&dev->IRQ0_INTE, val);
|
||
}
|
||
else if (irq == PIO_IRQ_LINE_1) {
|
||
io_reg_atomic_clear(&dev->IRQ1_INTE, val);
|
||
}
|
||
}
|
||
|
||
int pio_sm_transmit_word(pio_t pio, pio_sm_t sm, uint32_t word)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
if (pio_sm_tx_fifo_full(pio, sm)) {
|
||
return -EBUSY;
|
||
}
|
||
(&dev->TXF0)[sm] = word;
|
||
return 0;
|
||
}
|
||
|
||
void pio_sm_transmit_word_block(pio_t pio, pio_sm_t sm, uint32_t word)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
while (pio_sm_tx_fifo_full(pio, sm)) {}
|
||
(&dev->TXF0)[sm] = word;
|
||
}
|
||
|
||
void pio_sm_transmit_words_block(pio_t pio, pio_sm_t sm, const uint32_t *words, unsigned count)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
while (count--) {
|
||
pio_sm_transmit_word_block(pio, sm, *words++);
|
||
}
|
||
}
|
||
|
||
int pio_sm_receive_word(pio_t pio, pio_sm_t sm, uint32_t *word)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
if (pio_sm_rx_fifo_empty(pio, sm)) {
|
||
return -ENODATA;
|
||
}
|
||
uint32_t w = (&dev->RXF0)[sm];
|
||
if (word) {
|
||
*word = w;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
void pio_sm_receive_word_block(pio_t pio, pio_sm_t sm, uint32_t *word)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
while (pio_sm_rx_fifo_empty(pio, sm)) {}
|
||
uint32_t w = (&dev->RXF0)[sm];
|
||
if (word) {
|
||
*word = w;
|
||
}
|
||
}
|
||
|
||
void pio_sm_receive_words_block(pio_t pio, pio_sm_t sm, uint32_t *word, unsigned count)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
while (count--) {
|
||
pio_sm_receive_word_block(pio, sm, !word ? word : word++);
|
||
}
|
||
}
|
||
|
||
uint32_t pio_irq_get(pio_t pio)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
return pio_config[pio].dev->IRQ;
|
||
}
|
||
|
||
void pio_irq_clear(pio_t pio, unsigned irq_flags)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
pio_config[pio].dev->IRQ = irq_flags & ((1u << PIO_IRQ_NUMOF) - 1);
|
||
}
|
||
|
||
int pio_sm_init_common(pio_t pio, pio_sm_t sm,
|
||
const pio_program_t *prog,
|
||
const pio_program_conf_t *conf)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
if (prog->location < 0 ||
|
||
prog->location >= PIO_INSTR_NUMOF ||
|
||
prog->instr_numof > PIO_INSTR_NUMOF ||
|
||
!prog->written ||
|
||
conf->pc_start >= prog->instr_numof ||
|
||
conf->wrap_top >= prog->instr_numof ||
|
||
conf->wrap_bottom >= prog->instr_numof ||
|
||
conf->wrap_bottom > conf->wrap_top) {
|
||
return -ECANCELED;
|
||
}
|
||
pio_sm_reset(pio, sm);
|
||
pio_sm_clear_debug_txstall(pio, PIO_SM_MASK(sm));
|
||
pio_sm_clear_debug_txover(pio, PIO_SM_MASK(sm));
|
||
pio_sm_clear_debug_rxunder(pio, PIO_SM_MASK(sm));
|
||
pio_sm_clear_debug_rxstall(pio, PIO_SM_MASK(sm));
|
||
pio_sm_set_wrap(pio, sm, prog->location, conf->wrap_top, conf->wrap_bottom);
|
||
pio_sm_set_sideset_count(pio, sm, conf->sideset_count, conf->sideset_optional);
|
||
pio_sm_set_sideset_target(pio, sm, conf->sideset_pindirs);
|
||
pio_set_isr_vec(pio, sm, NULL);
|
||
/* cannot disable SM interrupts because they are globally used per PIO and
|
||
some program might be using it */
|
||
pio_irq_disable(pio, PIO_IRQ_LINE_0, (PIO_IRQ_RXNEMPTY_SM0 << sm) |
|
||
(PIO_IRQ_TXNFULL_SM0 << sm));
|
||
pio_irq_disable(pio, PIO_IRQ_LINE_1, (PIO_IRQ_RXNEMPTY_SM0 << sm) |
|
||
(PIO_IRQ_TXNFULL_SM0 << sm));
|
||
pio_sm_exec(pio, sm, pio_inst_jmp(PIO_INST_JMP_COND_NONE, (prog->location + conf->pc_start)));
|
||
return 0;
|
||
}
|
||
|
||
void pio_sm_set_pindirs_with_mask(pio_t pio, pio_sm_t sm, gpio_t values, gpio_t mask)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
uint32_t pinctrl = ctrl->pinctrl;
|
||
while (mask) {
|
||
uint8_t pos;
|
||
mask = bitarithm_test_and_clear(mask, &pos);
|
||
ctrl->pinctrl = (1u << PIO0_SM0_PINCTRL_SET_COUNT_Pos) |
|
||
((uint32_t)pos << PIO0_SM0_PINCTRL_SET_BASE_Pos);
|
||
pio_sm_exec(pio, sm, pio_inst_set(PIO_INST_SET_DST_PINDIRS, (values >> pos) & 0x1u));
|
||
}
|
||
ctrl->pinctrl = pinctrl;
|
||
}
|
||
|
||
void pio_sm_set_pins_with_mask(pio_t pio, pio_sm_t sm, gpio_t values, gpio_t mask)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
pio_sm_ctrl_regs_t *ctrl = &PIO_SM_CTRL_BASE(dev)[sm];
|
||
uint32_t pinctrl = ctrl->pinctrl;
|
||
while (mask) {
|
||
uint8_t pos;
|
||
mask = bitarithm_test_and_clear(mask, &pos);
|
||
ctrl->pinctrl = (1u << PIO0_SM0_PINCTRL_SET_COUNT_Pos) |
|
||
((uint32_t)pos << PIO0_SM0_PINCTRL_SET_BASE_Pos);
|
||
pio_sm_exec(pio, sm, pio_inst_set(PIO_INST_SET_DST_PINS, (values >> pos) & 0x1u));
|
||
}
|
||
ctrl->pinctrl = pinctrl;
|
||
}
|
||
|
||
static void _pio_pins_configure(pio_t pio, pio_sm_t sm, const pio_gpio_init_t *pin_init)
|
||
{
|
||
for (unsigned i = 0; i < pin_init->gpio_count; i++) {
|
||
gpio_pad_ctrl_t pad = pin_init->pad;
|
||
gpio_io_ctrl_t io = pin_init->io;
|
||
io.function_select = pio ? FUNCTION_SELECT_PIO1 : FUNCTION_SELECT_PIO0;
|
||
if (pin_init->gpio_direction & (1u << i)) { /* pin is output */
|
||
pad.output_disable = 0;
|
||
}
|
||
else {
|
||
pad.input_enable = 1;
|
||
}
|
||
gpio_set_pad_config(pin_init->gpio_base + i, pad);
|
||
gpio_set_io_config(pin_init->gpio_base + i, io);
|
||
pio_sm_set_pindirs_with_mask(pio, sm,
|
||
pin_init->gpio_direction << (pin_init->gpio_base + i),
|
||
0x1u << (pin_init->gpio_base + i));
|
||
pio_sm_set_pins_with_mask(pio, sm,
|
||
pin_init->gpio_state << (pin_init->gpio_base + i),
|
||
0x1u << (pin_init->gpio_base + i));
|
||
}
|
||
}
|
||
|
||
void pio_sm_set_set_pins_init(pio_t pio, pio_sm_t sm, const pio_gpio_init_t *pin_init)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
assert(pin_init->gpio_base < 32);
|
||
assert(pin_init->gpio_count <= 5);
|
||
assert(pin_init->gpio_base + pin_init->gpio_count <= 32);
|
||
|
||
pio_sm_set_set_pins(pio, sm, pin_init->gpio_base, pin_init->gpio_count);
|
||
_pio_pins_configure(pio, sm, pin_init);
|
||
|
||
}
|
||
|
||
void pio_sm_set_out_pins_init(pio_t pio, pio_sm_t sm, const pio_gpio_init_t *pin_init)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
assert(pin_init);
|
||
assert(pin_init->gpio_base < 32);
|
||
assert(pin_init->gpio_count <= 32);
|
||
assert(pin_init->gpio_base + pin_init->gpio_count <= 32);
|
||
|
||
pio_sm_set_out_pins(pio, sm, pin_init->gpio_base, pin_init->gpio_count);
|
||
_pio_pins_configure(pio, sm, pin_init);
|
||
|
||
}
|
||
|
||
void pio_sm_set_sideset_pins_init(pio_t pio, pio_sm_t sm, const pio_gpio_init_t *pin_init)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
assert((unsigned)sm < PIO_SM_NUMOF);
|
||
assert(pin_init);
|
||
assert(pin_init->gpio_base < 32);
|
||
assert(pin_init->gpio_count <= 5);
|
||
assert(pin_init->gpio_base + pin_init->gpio_count <= 32);
|
||
|
||
pio_sm_set_sideset_pins(pio, sm, pin_init->gpio_base);
|
||
_pio_pins_configure(pio, sm, pin_init);
|
||
}
|
||
|
||
void pio_print_status(pio_t pio)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
/* FIFO status */
|
||
uint32_t stat = dev->FSTAT;
|
||
uint8_t set;
|
||
set = (stat & (PIO_SM_ALL << PIO0_FSTAT_TXEMPTY_Pos) >> PIO0_FSTAT_TXEMPTY_Pos);
|
||
printf("TXEMPTY: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
set = (stat & (PIO_SM_ALL << PIO0_FSTAT_TXFULL_Pos) >> PIO0_FSTAT_TXFULL_Pos);
|
||
printf("TXFULL: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
set = (stat & (PIO_SM_ALL << PIO0_FSTAT_RXEMPTY_Pos) >> PIO0_FSTAT_RXEMPTY_Pos);
|
||
printf("RXEMPTY: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
set = (stat & (PIO_SM_ALL << PIO0_FSTAT_RXFULL_Pos) >> PIO0_FSTAT_RXFULL_Pos);
|
||
printf("RXFULL: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
/* program status */
|
||
printf("ADDR: SM0=%"PRIu32" SM1=%"PRIu32" SM2=%"PRIu32" SM3=%"PRIu32"\n",
|
||
dev->SM0_ADDR, dev->SM1_ADDR, dev->SM2_ADDR, dev->SM3_ADDR);
|
||
printf("INSTR: SM0=%"PRIu32" SM1=%"PRIu32" SM2=%"PRIu32" SM3=%"PRIu32"\n",
|
||
dev->SM0_INSTR, dev->SM1_INSTR, dev->SM2_INSTR, dev->SM3_INSTR);
|
||
}
|
||
|
||
void pio_print_debug(pio_t pio)
|
||
{
|
||
assert(pio <= PIO_NUMOF);
|
||
|
||
PIO0_Type *dev = pio_config[pio].dev;
|
||
uint32_t debug = dev->FDEBUG;
|
||
uint8_t set;
|
||
set = (debug & (PIO_SM_ALL << PIO0_FDEBUG_TXSTALL_Pos) >> PIO0_FDEBUG_TXSTALL_Pos);
|
||
printf("TXSTALL: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
set = (debug & (PIO_SM_ALL << PIO0_FDEBUG_TXOVER_Pos) >> PIO0_FDEBUG_TXOVER_Pos);
|
||
printf("TXOVER: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
set = (debug & (PIO_SM_ALL << PIO0_FDEBUG_RXUNDER_Pos) >> PIO0_FDEBUG_RXUNDER_Pos);
|
||
printf("RXUNDER: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
set = (debug & (PIO_SM_ALL << PIO0_FDEBUG_RXSTALL_Pos) >> PIO0_FDEBUG_RXSTALL_Pos);
|
||
printf("RXSTALL: SM0=%d SM1=%d SM2=%d SM3=%d\n",
|
||
set & PIO_SM0, set & PIO_SM1, set & PIO_SM2, set & PIO_SM3);
|
||
}
|
||
|
||
void PIO_0_ISR0(void)
|
||
{
|
||
_irq(0, 0, PIO0->IRQ0_INTS); /* calls cortexm_isr_end() */
|
||
}
|
||
|
||
void PIO_0_ISR1(void)
|
||
{
|
||
_irq(0, 1, PIO0->IRQ1_INTS); /* calls cortexm_isr_end() */
|
||
}
|
||
|
||
void PIO_1_ISR0(void)
|
||
{
|
||
_irq(1, 0, PIO1->IRQ0_INTS); /* calls cortexm_isr_end() */
|
||
}
|
||
|
||
void PIO_1_ISR1(void)
|
||
{
|
||
_irq(1, 1, PIO1->IRQ1_INTS); /* calls cortexm_isr_end() */
|
||
}
|