nuttx/drivers/rpmsg/rpmsg_port_spi.c
liaoao 4d35c60ba6 rpmsg_port_spi: set mreq to high to trigger next transmission
rpmsg_port_spi_connect can not be used here because peer may have not finished
the last transmission which will keep the sreq gpio in high level, and it will
read an error data frame.

Signed-off-by: liaoao <liaoao@xiaomi.com>
2024-09-17 01:55:51 +08:00

595 lines
17 KiB
C

/****************************************************************************
* drivers/rpmsg/rpmsg_port_spi.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <debug.h>
#include <errno.h>
#include <stdio.h>
#include <nuttx/crc16.h>
#include <nuttx/kmalloc.h>
#include <nuttx/kthread.h>
#include <nuttx/irq.h>
#include <nuttx/mutex.h>
#include "rpmsg_port.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#ifdef CONFIG_RPMSG_PORT_SPI_CRC
# define rpmsg_port_spi_crc16(hdr) crc16((FAR uint8_t *)&(hdr)->cmd, \
(hdr)->len - sizeof((hdr)->crc))
#else
# define rpmsg_port_spi_crc16(hdr) 0
#endif
#define BYTES2WORDS(s,b) ((b) / ((s)->nbits >> 3))
/****************************************************************************
* Private Types
****************************************************************************/
enum rpmsg_port_spi_cmd_e
{
RPMSG_PORT_SPI_CMD_CONNECT = 0x01,
RPMSG_PORT_SPI_CMD_AVAIL,
RPMSG_PORT_SPI_CMD_DATA,
};
struct rpmsg_port_spi_s
{
struct rpmsg_port_s port;
FAR struct spi_dev_s *spi;
FAR struct ioexpander_dev_s *ioe;
/* GPIOs used for handshake */
uint8_t mreq;
uint8_t sreq;
/* SPI devices' configuration */
uint32_t devid;
int nbits;
/* Reserved for cmd send */
FAR struct rpmsg_port_header_s *cmdhdr;
/* Used for sync data state between sreq_handler and complete_handler */
FAR struct rpmsg_port_header_s *txhdr;
FAR struct rpmsg_port_header_s *rxhdr;
rpmsg_port_rx_cb_t rxcb;
bool connected;
/* Used for flow control */
uint16_t txavail;
uint16_t rxavail;
uint16_t rxthres;
atomic_int transferring;
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static void rpmsg_port_spi_notify_tx_ready(FAR struct rpmsg_port_s *port);
static void rpmsg_port_spi_notify_rx_free(FAR struct rpmsg_port_s *port);
static void rpmsg_port_spi_register_cb(FAR struct rpmsg_port_s *port,
rpmsg_port_rx_cb_t callback);
/****************************************************************************
* Private Data
****************************************************************************/
static const struct rpmsg_port_ops_s g_rpmsg_port_spi_ops =
{
rpmsg_port_spi_notify_tx_ready,
rpmsg_port_spi_notify_rx_free,
rpmsg_port_spi_register_cb,
};
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: rpmsg_port_spi_drop_packets
****************************************************************************/
static void rpmsg_port_spi_drop_packets(FAR struct rpmsg_port_spi_s *rpspi)
{
FAR struct rpmsg_port_header_s *hdr;
while (!!(hdr = rpmsg_port_queue_get_buffer(&rpspi->port.txq, false)))
{
rpmsg_port_queue_return_buffer(&rpspi->port.txq, hdr);
}
while (!!(hdr = rpmsg_port_queue_get_buffer(&rpspi->port.rxq, false)))
{
rpmsg_port_queue_return_buffer(&rpspi->port.rxq, hdr);
}
}
/****************************************************************************
* Name: rpmsg_port_spi_notify_tx_ready
****************************************************************************/
static void rpmsg_port_spi_notify_tx_ready(FAR struct rpmsg_port_s *port)
{
FAR struct rpmsg_port_spi_s *rpspi =
container_of(port, struct rpmsg_port_spi_s, port);
if (rpspi->connected)
{
IOEXP_WRITEPIN(rpspi->ioe, rpspi->mreq, 1);
}
else
{
rpmsg_port_spi_drop_packets(rpspi);
}
}
/****************************************************************************
* Name: rpmsg_port_spi_notify_rx_free
****************************************************************************/
static void rpmsg_port_spi_notify_rx_free(FAR struct rpmsg_port_s *port)
{
FAR struct rpmsg_port_spi_s *rpspi =
container_of(port, struct rpmsg_port_spi_s, port);
if (rpmsg_port_queue_navail(&port->rxq) - rpspi->rxavail >= rpspi->rxthres)
{
IOEXP_WRITEPIN(rpspi->ioe, rpspi->mreq, 1);
}
}
/****************************************************************************
* Name: rpmsg_port_spi_register_cb
****************************************************************************/
static void rpmsg_port_spi_register_cb(FAR struct rpmsg_port_s *port,
rpmsg_port_rx_cb_t callback)
{
FAR struct rpmsg_port_spi_s *rpspi =
container_of(port, struct rpmsg_port_spi_s, port);
rpspi->rxcb = callback;
}
/****************************************************************************
* Name: rpmsg_port_spi_exchange
****************************************************************************/
static void rpmsg_port_spi_exchange(FAR struct rpmsg_port_spi_s *rpspi)
{
FAR struct rpmsg_port_header_s *txhdr;
IOEXP_WRITEPIN(rpspi->ioe, rpspi->mreq, 0);
if (atomic_fetch_add(&rpspi->transferring, 1))
{
return;
}
if (!rpspi->connected)
{
txhdr = rpspi->cmdhdr;
txhdr->cmd = RPMSG_PORT_SPI_CMD_CONNECT;
strlcpy((FAR char *)(txhdr + 1), rpspi->port.cpuname, RPMSG_NAME_SIZE);
}
else if (rpspi->txavail > 0 &&
rpmsg_port_queue_nused(&rpspi->port.txq) > 0)
{
txhdr = rpmsg_port_queue_get_buffer(&rpspi->port.txq, false);
DEBUGASSERT(txhdr != NULL);
txhdr->cmd = RPMSG_PORT_SPI_CMD_DATA;
rpspi->txhdr = txhdr;
}
else
{
txhdr = rpspi->cmdhdr;
txhdr->cmd = RPMSG_PORT_SPI_CMD_AVAIL;
}
txhdr->avail = rpmsg_port_queue_navail(&rpspi->port.rxq);
txhdr->avail = txhdr->avail > 1 ? txhdr->avail - 1 : 0;
txhdr->crc = rpmsg_port_spi_crc16(txhdr);
rpmsginfo("irq send cmd:%u avail:%u\n", txhdr->cmd, txhdr->avail);
SPI_SELECT(rpspi->spi, rpspi->devid, true);
SPI_EXCHANGE(rpspi->spi, txhdr, rpspi->rxhdr,
BYTES2WORDS(rpspi, rpspi->cmdhdr->len));
rpspi->rxavail = txhdr->avail;
}
/****************************************************************************
* Name: rpmsg_port_spi_complete_handler
****************************************************************************/
static void rpmsg_port_spi_complete_handler(FAR void *arg)
{
FAR struct rpmsg_port_spi_s *rpspi = arg;
SPI_SELECT(rpspi->spi, rpspi->devid, false);
rpmsginfo("received cmd:%u avail:%u\n",
rpspi->rxhdr->cmd, rpspi->rxhdr->avail);
if (rpspi->txhdr != NULL)
{
rpmsg_port_queue_return_buffer(&rpspi->port.txq, rpspi->txhdr);
rpspi->txhdr = NULL;
}
if (rpspi->rxhdr->crc != 0)
{
uint16_t crc = rpmsg_port_spi_crc16(rpspi->rxhdr);
if (rpspi->rxhdr->crc != crc)
{
rpmsgerr("crc check fail received: %u calculated: %u\n",
rpspi->rxhdr->crc, crc);
goto out;
}
}
/* Skip any data received when connection is not established until a
* connect req data packet has been received.
*/
if (!rpspi->connected)
{
if (rpspi->rxhdr->cmd != RPMSG_PORT_SPI_CMD_CONNECT)
{
goto out;
}
rpspi->txavail = rpspi->rxhdr->avail;
rpspi->connected = true;
}
else
{
rpspi->txavail = rpspi->rxhdr->avail;
if (rpspi->rxhdr->cmd == RPMSG_PORT_SPI_CMD_CONNECT)
{
rpspi->connected = false;
rpmsg_port_spi_drop_packets(rpspi);
}
}
if (rpspi->rxhdr->cmd != RPMSG_PORT_SPI_CMD_AVAIL)
{
rpmsg_port_queue_add_buffer(&rpspi->port.rxq, rpspi->rxhdr);
rpspi->rxhdr = rpmsg_port_queue_get_available_buffer(
&rpspi->port.rxq, false);
DEBUGASSERT(rpspi->rxhdr != NULL);
}
out:
if (atomic_exchange(&rpspi->transferring, 0) > 1)
{
rpmsg_port_spi_exchange(rpspi);
}
else if (rpspi->connected &&
rpspi->txavail > 0 && rpmsg_port_queue_nused(&rpspi->port.txq) > 0)
{
IOEXP_WRITEPIN(rpspi->ioe, rpspi->mreq, 1);
}
}
/****************************************************************************
* Name: rpmsg_port_spi_sreq_handler
****************************************************************************/
static int rpmsg_port_spi_sreq_handler(FAR struct ioexpander_dev_s *dev,
ioe_pinset_t pinset, FAR void *arg)
{
FAR struct rpmsg_port_spi_s *rpspi = arg;
rpmsginfo("sreq enter\n");
rpmsg_port_spi_exchange(rpspi);
return 0;
}
/****************************************************************************
* Name: rpmsg_port_spi_connect
****************************************************************************/
static void rpmsg_port_spi_connect(FAR struct rpmsg_port_spi_s *rpspi)
{
bool val;
IOEXP_READPIN(rpspi->ioe, rpspi->sreq, &val);
rpmsginfo("sreq level is %d\n", val);
/* If sreq gpio is high, that means peer has sent connect req and receive
* the connect data packet directly. if req gpio is low, then current
* side send the req.
*/
if (val)
{
rpmsg_port_spi_exchange(rpspi);
}
else
{
IOEXP_WRITEPIN(rpspi->ioe, rpspi->mreq, 1);
}
}
/****************************************************************************
* Name: rpmsg_port_spi_process_packet
****************************************************************************/
static void
rpmsg_port_spi_process_packet(FAR struct rpmsg_port_spi_s *rpspi,
FAR struct rpmsg_port_header_s *rxhdr)
{
rpmsginfo("received cmd: %u avail: %u", rxhdr->cmd, rxhdr->avail);
switch (rxhdr->cmd)
{
case RPMSG_PORT_SPI_CMD_CONNECT:
if (!rpspi->connected)
{
rpmsg_port_unregister(&rpspi->port);
/* Trigger a transmission for reconnection */
IOEXP_WRITEPIN(rpspi->ioe, rpspi->mreq, 1);
}
else
{
rpmsg_port_register(&rpspi->port, (FAR const char *)(rxhdr + 1));
}
rpmsg_port_queue_return_buffer(&rpspi->port.rxq, rxhdr);
break;
case RPMSG_PORT_SPI_CMD_DATA:
rpspi->rxcb(&rpspi->port, rxhdr);
break;
default:
rpmsgerr("received a unexpected frame, dropped\n");
rpmsg_port_queue_return_buffer(&rpspi->port.rxq, rxhdr);
break;
}
}
/****************************************************************************
* Name: rpmsg_port_spi_thread
****************************************************************************/
static int rpmsg_port_spi_thread(int argc, FAR char *argv[])
{
FAR struct rpmsg_port_spi_s *rpspi =
(FAR struct rpmsg_port_spi_s *)((uintptr_t)strtoul(argv[2], NULL, 0));
FAR struct rpmsg_port_queue_s *queue = &rpspi->port.rxq;
FAR struct rpmsg_port_header_s *rxhdr;
rpmsg_port_spi_connect(rpspi);
for (; ; )
{
while ((rxhdr = rpmsg_port_queue_get_buffer(queue, true)) != NULL)
{
rpmsg_port_spi_process_packet(rpspi, rxhdr);
}
}
return 0;
}
/****************************************************************************
* Name: rpmsg_port_spi_gpio_init
****************************************************************************/
static int
rpmsg_port_spi_init_gpio(FAR struct ioexpander_dev_s *ioe,
FAR uint8_t *gpio, uint8_t pin, int invert,
ioe_callback_t callback, FAR void *args)
{
int direction = callback ?
IOEXPANDER_DIRECTION_IN_PULLDOWN : IOEXPANDER_DIRECTION_OUT;
int ret;
ret = IOEXP_SETOPTION(ioe, pin, IOEXPANDER_OPTION_INVERT,
(FAR void *)invert);
if (ret < 0)
{
rpmsgerr("gpio set invert option error: %d\n", ret);
return ret;
}
ret = IOEXP_SETDIRECTION(ioe, pin, direction);
if (ret < 0)
{
rpmsgerr("gpio set direction %d error: %d\n", direction, ret);
return ret;
}
if (direction == IOEXPANDER_DIRECTION_IN_PULLDOWN)
{
int intcfg = invert == IOEXPANDER_VAL_INVERT ?
IOEXPANDER_VAL_FALLING : IOEXPANDER_VAL_RISING;
FAR void *ptr;
ret = IOEXP_SETOPTION(ioe, pin, IOEXPANDER_OPTION_INTCFG,
(FAR void *)intcfg);
if (ret < 0)
{
rpmsgerr("gpio set int option %d error: %d\n", intcfg, ret);
return ret;
}
ptr = IOEP_ATTACH(ioe, pin, callback, args);
if (ptr == NULL)
{
rpmsgerr("gpio attach error: %d\n", ret);
return -EINVAL;
}
}
*gpio = pin;
return ret;
}
/****************************************************************************
* Name: rpmsg_port_spi_init_hardware
****************************************************************************/
static int
rpmsg_port_spi_init_hardware(FAR struct rpmsg_port_spi_s *rpspi,
FAR const struct rpmsg_port_spi_config_s *spicfg,
FAR struct spi_dev_s *spi, FAR struct ioexpander_dev_s *ioe)
{
int ret;
/* Init mreq gpio */
ret = rpmsg_port_spi_init_gpio(ioe, &rpspi->mreq, spicfg->mreq_pin,
spicfg->mreq_invert, NULL, NULL);
if (ret < 0)
{
rpmsgerr("mreq init failed\n");
return ret;
}
/* Init sreq gpio */
ret = rpmsg_port_spi_init_gpio(ioe, &rpspi->sreq, spicfg->sreq_pin,
spicfg->sreq_invert,
rpmsg_port_spi_sreq_handler, rpspi);
if (ret < 0)
{
rpmsgerr("sreq init failed\n");
return ret;
}
SPI_SETBITS(spi, spicfg->nbits);
SPI_SETMODE(spi, spicfg->mode);
SPI_SETFREQUENCY(spi, spicfg->freq);
SPI_REGISTERCALLBACK(spi, rpmsg_port_spi_complete_handler, rpspi);
SPI_SELECT(spi, spicfg->devid, false);
rpspi->spi = spi;
rpspi->ioe = ioe;
rpspi->devid = spicfg->devid;
rpspi->nbits = spicfg->nbits;
return 0;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: rpmsg_port_spi_initialize
****************************************************************************/
int
rpmsg_port_spi_initialize(FAR const struct rpmsg_port_config_s *cfg,
FAR const struct rpmsg_port_spi_config_s *spicfg,
FAR struct spi_dev_s *spi,
FAR struct ioexpander_dev_s *ioe)
{
FAR struct rpmsg_port_spi_s *rpspi;
FAR char *argv[3];
char arg1[32];
int ret;
if (spi == NULL || ioe == NULL || spicfg == NULL)
{
rpmsgerr("invalid params\n");
return -EINVAL;
}
rpspi = kmm_zalloc(sizeof(*rpspi));
if (rpspi == NULL)
{
rpmsgerr("malloc rpmsg spi failed\n");
return -ENOMEM;
}
DEBUGASSERT(cfg->txlen == cfg->rxlen);
ret = rpmsg_port_initialize(&rpspi->port, cfg, &g_rpmsg_port_spi_ops);
if (ret < 0)
{
rpmsgerr("rpmsg port initialize failed\n");
goto rpmsg_err;
}
/* Always reserve one buffer for sending/receiving cmd packet */
rpspi->cmdhdr = rpmsg_port_queue_get_available_buffer(
&rpspi->port.txq, true);
rpspi->rxhdr = rpmsg_port_queue_get_available_buffer(
&rpspi->port.rxq, true);
DEBUGASSERT(rpspi->cmdhdr != NULL && rpspi->rxhdr != NULL);
rpspi->rxthres = rpmsg_port_queue_navail(&rpspi->port.rxq) *
CONFIG_RPMSG_PORT_SPI_RX_THRESHOLD / 100;
ret = rpmsg_port_spi_init_hardware(rpspi, spicfg, spi, ioe);
if (ret < 0)
{
rpmsgerr("rpmsg port spi hardware init failed\n");
goto out;
}
snprintf(arg1, sizeof(arg1), "%p", rpspi);
argv[0] = (FAR void *)cfg->remotecpu;
argv[1] = arg1;
argv[2] = NULL;
ret = kthread_create("rpmsg-spi",
CONFIG_RPMSG_PORT_SPI_THREAD_PRIORITY,
CONFIG_RPMSG_PORT_SPI_THREAD_STACKSIZE,
rpmsg_port_spi_thread, argv);
if (ret < 0)
{
rpmsgerr("rpmsg port spi create thread failed\n");
goto out;
}
return 0;
out:
rpmsg_port_uninitialize(&rpspi->port);
rpmsg_err:
kmm_free(rpspi);
return ret;
}