630 lines
19 KiB
C
630 lines
19 KiB
C
/****************************************************************************
|
|
* drivers/spi/spi_slave_driver.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 <nuttx/config.h>
|
|
|
|
#include <semaphore.h>
|
|
#include <sys/types.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <debug.h>
|
|
|
|
#include <nuttx/kmalloc.h>
|
|
#include <nuttx/fs/fs.h>
|
|
#include <nuttx/semaphore.h>
|
|
#include <nuttx/spi/slave.h>
|
|
|
|
#ifdef CONFIG_SPI_SLAVE_DRIVER
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
#define DEVNAME_FMT "/dev/spislv%d"
|
|
#define DEVNAME_FMTLEN (11 + 3 + 1)
|
|
|
|
#ifndef MIN
|
|
# define MIN(a, b) (((a) < (b)) ? (a) : (b))
|
|
#endif
|
|
|
|
#define WORDS2BYTES(_wn) ((_wn) * (CONFIG_SPI_SLAVE_DRIVER_WIDTH / 8))
|
|
#define BYTES2WORDS(_bn) ((_bn) / (CONFIG_SPI_SLAVE_DRIVER_WIDTH / 8))
|
|
|
|
/****************************************************************************
|
|
* Private Types
|
|
****************************************************************************/
|
|
|
|
struct spislv_driver_s
|
|
{
|
|
/* Externally visible part of the SPI Slave device interface */
|
|
|
|
struct spi_sdev_s dev;
|
|
|
|
/* Reference to SPI Slave controller interface */
|
|
|
|
struct spi_sctrlr_s *sctrlr;
|
|
|
|
/* Receive buffer */
|
|
|
|
uint8_t rx_buffer[CONFIG_SPI_SLAVE_DRIVER_BUFFER_SIZE];
|
|
uint32_t rx_length; /* Location of next RX value */
|
|
|
|
/* Transmit buffer */
|
|
|
|
uint8_t tx_buffer[CONFIG_SPI_SLAVE_DRIVER_BUFFER_SIZE];
|
|
uint32_t tx_length; /* Location of next TX value */
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
sem_t exclsem; /* Mutual exclusion */
|
|
int16_t crefs; /* Number of open references */
|
|
bool unlinked; /* Indicates if the driver has been unlinked */
|
|
#endif
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Function Prototypes
|
|
****************************************************************************/
|
|
|
|
/* Character driver methods */
|
|
|
|
static int spislv_open(FAR struct file *filep);
|
|
static int spislv_close(FAR struct file *filep);
|
|
static ssize_t spislv_read(FAR struct file *filep, FAR char *buffer,
|
|
size_t buflen);
|
|
static ssize_t spislv_write(FAR struct file *filep, FAR const char *buffer,
|
|
size_t buflen);
|
|
static int spislv_unlink(FAR struct inode *inode);
|
|
|
|
/* SPI Slave driver methods */
|
|
|
|
static void spislv_select(FAR struct spi_sdev_s *sdev, bool selected);
|
|
static void spislv_cmddata(FAR struct spi_sdev_s *sdev, bool data);
|
|
static size_t spislv_getdata(FAR struct spi_sdev_s *sdev,
|
|
FAR const void **data);
|
|
static size_t spislv_receive(FAR struct spi_sdev_s *sdev,
|
|
FAR const void *data, size_t nwords);
|
|
|
|
/****************************************************************************
|
|
* Private Data
|
|
****************************************************************************/
|
|
|
|
static const struct file_operations g_spislvfops =
|
|
{
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
spislv_open, /* open */
|
|
spislv_close, /* close */
|
|
#else
|
|
NULL, /* open */
|
|
NULL, /* close */
|
|
#endif
|
|
spislv_read, /* read */
|
|
spislv_write, /* write */
|
|
NULL, /* seek */
|
|
NULL, /* ioctl */
|
|
NULL /* poll */
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
, spislv_unlink /* unlink */
|
|
#endif
|
|
};
|
|
|
|
static const struct spi_sdevops_s g_spisdev_ops =
|
|
{
|
|
spislv_select, /* select */
|
|
spislv_cmddata, /* cmddata */
|
|
spislv_getdata, /* getdata */
|
|
spislv_receive, /* receive */
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_open
|
|
*
|
|
* Description:
|
|
* This function is called whenever the SPI Slave device is opened.
|
|
*
|
|
* Input Parameters:
|
|
* filep - File structure instance
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) is returned on success. A negated errno value is returned on
|
|
* any failure to indicate the nature of the failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
static int spislv_open(FAR struct file *filep)
|
|
{
|
|
FAR struct inode *inode;
|
|
FAR struct spislv_driver_s *priv;
|
|
int ret;
|
|
|
|
DEBUGASSERT(filep != NULL);
|
|
DEBUGASSERT(filep->f_inode != NULL);
|
|
DEBUGASSERT(filep->f_inode->i_private != NULL);
|
|
|
|
spiinfo("filep: %p\n", filep);
|
|
|
|
/* Get our private data structure */
|
|
|
|
inode = filep->f_inode;
|
|
priv = (FAR struct spislv_driver_s *)inode->i_private;
|
|
|
|
/* Get exclusive access to the SPI Slave driver state structure */
|
|
|
|
ret = nxsem_wait(&priv->exclsem);
|
|
if (ret < 0)
|
|
{
|
|
spierr("Failed to get exclusive access to the driver: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Increment the count of open references on the driver */
|
|
|
|
priv->crefs++;
|
|
DEBUGASSERT(priv->crefs > 0);
|
|
|
|
nxsem_post(&priv->exclsem);
|
|
return OK;
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_close
|
|
*
|
|
* Description:
|
|
* This routine is called when the SPI Slave device is closed.
|
|
*
|
|
* Input Parameters:
|
|
* filep - File structure instance
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) is returned on success; A negated errno value is returned on
|
|
* any failure to indicate the nature of the failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
static int spislv_close(FAR struct file *filep)
|
|
{
|
|
FAR struct inode *inode;
|
|
FAR struct spislv_driver_s *priv;
|
|
int ret;
|
|
|
|
DEBUGASSERT(filep != NULL);
|
|
DEBUGASSERT(filep->f_inode != NULL);
|
|
DEBUGASSERT(filep->f_inode->i_private != NULL);
|
|
|
|
spiinfo("filep: %p\n", filep);
|
|
|
|
/* Get our private data structure */
|
|
|
|
inode = filep->f_inode;
|
|
priv = (FAR struct spislv_driver_s *)inode->i_private;
|
|
|
|
/* Get exclusive access to the SPI Slave driver state structure */
|
|
|
|
ret = nxsem_wait(&priv->exclsem);
|
|
if (ret < 0)
|
|
{
|
|
spierr("Failed to get exclusive access to the driver: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Decrement the count of open references on the driver */
|
|
|
|
DEBUGASSERT(priv->crefs > 0);
|
|
priv->crefs--;
|
|
|
|
/* If the count has decremented to zero and the driver has been already
|
|
* unlinked, then dispose of the driver resources.
|
|
*/
|
|
|
|
if (priv->crefs <= 0 && priv->unlinked)
|
|
{
|
|
nxsem_destroy(&priv->exclsem);
|
|
kmm_free(priv);
|
|
inode->i_private = NULL;
|
|
return OK;
|
|
}
|
|
|
|
nxsem_post(&priv->exclsem);
|
|
return OK;
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_read
|
|
*
|
|
* Description:
|
|
* This routine is called when the application requires to read the data
|
|
* received by the SPI Slave device.
|
|
*
|
|
* Input Parameters:
|
|
* filep - File structure instance
|
|
* buffer - User-provided to save the data
|
|
* buflen - The maximum size of the user-provided buffer
|
|
*
|
|
* Returned Value:
|
|
* The positive non-zero number of bytes read on success, 0 on if an
|
|
* end-of-file condition, or a negated errno value on any failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static ssize_t spislv_read(FAR struct file *filep, FAR char *buffer,
|
|
size_t buflen)
|
|
{
|
|
FAR struct inode *inode;
|
|
FAR struct spislv_driver_s *priv;
|
|
size_t read_bytes;
|
|
size_t remaining_words;
|
|
|
|
spiinfo("filep=%p buffer=%p buflen=%zu\n", filep, buffer, buflen);
|
|
|
|
/* Get our private data structure */
|
|
|
|
inode = filep->f_inode;
|
|
priv = (FAR struct spislv_driver_s *)inode->i_private;
|
|
|
|
if (buffer == NULL)
|
|
{
|
|
spierr("ERROR: Buffer is NULL\n");
|
|
return -ENOBUFS;
|
|
}
|
|
|
|
remaining_words = SPI_SCTRLR_QPOLL(priv->sctrlr);
|
|
if (remaining_words == 0)
|
|
{
|
|
spiinfo("All words retrieved!\n");
|
|
}
|
|
else
|
|
{
|
|
spiinfo("%zu words left in the buffer\n", remaining_words);
|
|
}
|
|
|
|
read_bytes = MIN(buflen, priv->rx_length);
|
|
|
|
memcpy(buffer, priv->rx_buffer, read_bytes);
|
|
|
|
return (ssize_t)read_bytes;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_write
|
|
*
|
|
* Description:
|
|
* This routine is called when the application needs to enqueue data to be
|
|
* transferred at the next leading clock edge of the SPI Slave controller.
|
|
*
|
|
* Input Parameters:
|
|
* filep - Instance of struct file to use with the write
|
|
* buffer - Data to write
|
|
* buflen - Length of data to write in bytes
|
|
*
|
|
* Returned Value:
|
|
* On success, the number of bytes written are returned (zero indicates
|
|
* nothing was written). On any failure, a negated errno value is returned.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static ssize_t spislv_write(FAR struct file *filep, FAR const char *buffer,
|
|
size_t buflen)
|
|
{
|
|
FAR struct inode *inode;
|
|
FAR struct spislv_driver_s *priv;
|
|
size_t num_words;
|
|
size_t enqueued_bytes;
|
|
|
|
spiinfo("filep=%p buffer=%p buflen=%zu\n", filep, buffer, buflen);
|
|
|
|
/* Get our private data structure */
|
|
|
|
inode = filep->f_inode;
|
|
priv = (FAR struct spislv_driver_s *)inode->i_private;
|
|
|
|
memcpy(priv->tx_buffer, buffer, buflen);
|
|
priv->tx_length = buflen;
|
|
num_words = BYTES2WORDS(priv->tx_length);
|
|
|
|
enqueued_bytes = WORDS2BYTES(SPI_SCTRLR_ENQUEUE(priv->sctrlr,
|
|
priv->tx_buffer,
|
|
num_words));
|
|
|
|
spiinfo("%zu bytes enqueued\n", enqueued_bytes);
|
|
|
|
return (ssize_t)enqueued_bytes;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_unlink
|
|
*
|
|
* Description:
|
|
* This routine is called when the SPI Slave device is unlinked.
|
|
*
|
|
* Input Parameters:
|
|
* inode - The inode associated with the SPI Slave device
|
|
*
|
|
* Returned Value:
|
|
* Zero is returned on success; a negated value is returned on any failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
static int spislv_unlink(FAR struct inode *inode)
|
|
{
|
|
FAR struct spislv_driver_s *priv;
|
|
int ret;
|
|
|
|
DEBUGASSERT(inode != NULL);
|
|
DEBUGASSERT(inode->i_private != NULL);
|
|
|
|
/* Get our private data structure */
|
|
|
|
priv = (FAR struct spislv_driver_s *)inode->i_private;
|
|
|
|
/* Get exclusive access to the SPI Slave driver state structure */
|
|
|
|
ret = nxsem_wait(&priv->exclsem);
|
|
if (ret < 0)
|
|
{
|
|
spierr("Failed to get exclusive access to the driver: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Are there open references to the driver data structure? */
|
|
|
|
if (priv->crefs <= 0)
|
|
{
|
|
nxsem_destroy(&priv->exclsem);
|
|
kmm_free(priv);
|
|
inode->i_private = NULL;
|
|
return OK;
|
|
}
|
|
|
|
/* No... just mark the driver as unlinked and free the resources when the
|
|
* last client closes their reference to the driver.
|
|
*/
|
|
|
|
priv->unlinked = true;
|
|
nxsem_post(&priv->exclsem);
|
|
return ret;
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_select
|
|
*
|
|
* Description:
|
|
* This is a SPI device callback that is used when the SPI controller
|
|
* driver detects any change in the chip select pin.
|
|
*
|
|
* Input Parameters:
|
|
* sdev - SPI device interface instance
|
|
* selected - Indicates whether the chip select is in active state
|
|
*
|
|
* Returned Value:
|
|
* None.
|
|
*
|
|
* Assumptions:
|
|
* May be called from an interrupt handler. Processing should be as
|
|
* brief as possible.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void spislv_select(FAR struct spi_sdev_s *sdev, bool selected)
|
|
{
|
|
spiinfo("sdev: %p CS: %s\n", sdev, selected ? "select" : "free");
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_cmddata
|
|
*
|
|
* Description:
|
|
* This is a SPI device callback that is used when the SPI controller
|
|
* driver detects any change in the command/data condition.
|
|
*
|
|
* Normally only LCD devices distinguish command and data. For devices
|
|
* that do not distinguish between command and data, this method may be
|
|
* a stub. For devices that do make that distinction, they should treat
|
|
* all subsequent calls to getdata() or receive() appropriately for the
|
|
* current command/data selection.
|
|
*
|
|
* Input Parameters:
|
|
* sdev - SPI device interface instance
|
|
* data - True: Data is selected
|
|
*
|
|
* Returned Value:
|
|
* None.
|
|
*
|
|
* Assumptions:
|
|
* May be called from an interrupt handler. Processing should be as
|
|
* brief as possible.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void spislv_cmddata(FAR struct spi_sdev_s *sdev, bool data)
|
|
{
|
|
spiinfo("sdev: %p CMD: %s\n", sdev, data ? "data" : "command");
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_getdata
|
|
*
|
|
* Description:
|
|
* This is a SPI device callback that is used when the SPI controller
|
|
* requires data be shifted out at the next leading clock edge. This
|
|
* is necessary to "prime the pump" so that the SPI controller driver
|
|
* can keep pace with the shifted-in data.
|
|
*
|
|
* The SPI controller driver will prime for both command and data
|
|
* transfers as determined by a preceding call to the device cmddata()
|
|
* method. Normally only LCD devices distinguish command and data.
|
|
*
|
|
* Input Parameters:
|
|
* sdev - SPI device interface instance
|
|
* data - Pointer to the data buffer pointer to be shifed out.
|
|
* The device will set the data buffer pointer to the actual data
|
|
*
|
|
* Returned Value:
|
|
* The number of data units to be shifted out from the data buffer.
|
|
*
|
|
* Assumptions:
|
|
* May be called from an interrupt handler and the response is usually
|
|
* time-critical.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static size_t spislv_getdata(FAR struct spi_sdev_s *sdev,
|
|
FAR const void **data)
|
|
{
|
|
FAR struct spislv_driver_s *priv = (FAR struct spislv_driver_s *)sdev;
|
|
|
|
*data = priv->tx_buffer;
|
|
|
|
return BYTES2WORDS(priv->tx_length);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_receive
|
|
*
|
|
* Description:
|
|
* This is a SPI device callback that is used when the SPI controller
|
|
* receives a new value shifted in. Notice that these values may be out of
|
|
* synchronization by several words.
|
|
*
|
|
* Input Parameters:
|
|
* sdev - SPI device interface instance
|
|
* data - Pointer to the new data that has been shifted in
|
|
* len - Length of the new data in units of nbits wide,
|
|
* nbits being the data width previously provided to the bind()
|
|
* method.
|
|
*
|
|
* Returned Value:
|
|
* Number of units accepted by the device. In other words,
|
|
* number of units to be removed from controller's receive queue.
|
|
*
|
|
* Assumptions:
|
|
* May be called from an interrupt handler and in time-critical
|
|
* circumstances. A good implementation might just add the newly
|
|
* received word to a queue, post a processing task, and return as
|
|
* quickly as possible to avoid any data overrun problems.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static size_t spislv_receive(FAR struct spi_sdev_s *sdev,
|
|
FAR const void *data, size_t len)
|
|
{
|
|
FAR struct spislv_driver_s *priv = (FAR struct spislv_driver_s *)sdev;
|
|
size_t recv_bytes = MIN(len, sizeof(priv->rx_buffer));
|
|
|
|
memcpy(priv->rx_buffer, data, recv_bytes);
|
|
|
|
priv->rx_length = recv_bytes;
|
|
|
|
return BYTES2WORDS(recv_bytes);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: spislv_register
|
|
*
|
|
* Description:
|
|
* Register the SPI Slave character device as 'devpath'.
|
|
*
|
|
* Input Parameters:
|
|
* sctrlr - An instance of the SPI Slave interface to use to communicate
|
|
* with the SPI Slave device
|
|
* bus - The SPI Slave bus number. This will be used as the SPI device
|
|
* minor number. The SPI Slave character device will be
|
|
* registered as /dev/spislvN where N is the minor number
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) on success; a negated errno value on failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int spislv_register(FAR struct spi_sctrlr_s *sctrlr, int bus)
|
|
{
|
|
FAR struct spislv_driver_s *priv;
|
|
char devname[DEVNAME_FMTLEN];
|
|
int ret;
|
|
|
|
/* Sanity check */
|
|
|
|
DEBUGASSERT(sctrlr != NULL && (unsigned int)bus < 1000);
|
|
|
|
/* Initialize the SPI Slave device structure */
|
|
|
|
priv = (FAR struct spislv_driver_s *)
|
|
kmm_zalloc(sizeof(struct spislv_driver_s));
|
|
if (!priv)
|
|
{
|
|
spierr("ERROR: Failed to allocate instance\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
priv->dev.ops = &g_spisdev_ops;
|
|
priv->sctrlr = sctrlr;
|
|
|
|
#ifdef CONFIG_SPI_SLAVE_DRIVER_COLORIZE_TX_BUFFER
|
|
memset(priv->tx_buffer,
|
|
CONFIG_SPI_SLAVE_DRIVER_COLORIZE_PATTERN,
|
|
CONFIG_SPI_SLAVE_DRIVER_COLORIZE_NUM_BYTES);
|
|
priv->tx_length = CONFIG_SPI_SLAVE_DRIVER_COLORIZE_NUM_BYTES;
|
|
#endif
|
|
|
|
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
|
|
nxsem_init(&priv->exclsem, 0, 1);
|
|
#endif
|
|
|
|
/* Create the character device name */
|
|
|
|
snprintf(devname, DEVNAME_FMTLEN, DEVNAME_FMT, bus);
|
|
|
|
/* Register the character driver */
|
|
|
|
ret = register_driver(devname, &g_spislvfops, 0666, priv);
|
|
if (ret < 0)
|
|
{
|
|
spierr("ERROR: Failed to register driver: %d\n", ret);
|
|
kmm_free(priv);
|
|
}
|
|
|
|
SPI_SCTRLR_BIND(priv->sctrlr, (FAR struct spi_sdev_s *)priv,
|
|
CONFIG_SPI_SLAVE_DRIVER_MODE,
|
|
CONFIG_SPI_SLAVE_DRIVER_WIDTH);
|
|
|
|
spiinfo("SPI Slave driver loaded successfully!\n");
|
|
|
|
return ret;
|
|
}
|
|
|
|
#endif /* CONFIG_SPI_SLAVE_DRIVER */
|