/**************************************************************************** * drivers/rpmsg/rpmsg_port_spi.c * * SPDX-License-Identifier: Apache-2.0 * * 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 #include #include #include #include #include #include #include #include #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; }