mirror of
https://github.com/apache/nuttx.git
synced 2026-06-06 08:36:24 +08:00
Add a general bit-bang SPI lower-half driver and implement the bit-bang driver for the Arduino ITEAD TFT shield
This commit is contained in:
@@ -0,0 +1,52 @@
|
||||
############################################################################
|
||||
# drivers/spi/Make.defs
|
||||
#
|
||||
# Copyright (C) 2013 Gregory Nutt. All rights reserved.
|
||||
# Author: Gregory Nutt <gnutt@nuttx.org>
|
||||
#
|
||||
# Redistribution and use in source and binary forms, with or without
|
||||
# modification, are permitted provided that the following conditions
|
||||
# are met:
|
||||
#
|
||||
# 1. Redistributions of source code must retain the above copyright
|
||||
# notice, this list of conditions and the following disclaimer.
|
||||
# 2. Redistributions in binary form must reproduce the above copyright
|
||||
# notice, this list of conditions and the following disclaimer in
|
||||
# the documentation and/or other materials provided with the
|
||||
# distribution.
|
||||
# 3. Neither the name NuttX nor the names of its contributors may be
|
||||
# used to endorse or promote products derived from this software
|
||||
# without specific prior written permission.
|
||||
#
|
||||
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||
# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||||
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
||||
# OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||||
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||||
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||
# POSSIBILITY OF SUCH DAMAGE.
|
||||
#
|
||||
############################################################################
|
||||
|
||||
# Don't build anything if there is no SPI support
|
||||
|
||||
ifeq ($(CONFIG_SPI),y)
|
||||
|
||||
# Include the selected SPI drivers
|
||||
|
||||
ifeq ($(CONFIG_SPI_BITBANG),y)
|
||||
CSRCS += spi_bitbang.c
|
||||
endif
|
||||
|
||||
# Include SPI device driver build support
|
||||
|
||||
DEPPATH += --dep-path spi
|
||||
VPATH += :spi
|
||||
CFLAGS += ${shell $(INCDIR) $(INCDIROPT) "$(CC)" $(TOPDIR)$(DELIM)drivers$(DELIM)spi}
|
||||
endif
|
||||
|
||||
+67
-21
@@ -42,6 +42,7 @@
|
||||
#include <semaphore.h>
|
||||
#include <assert.h>
|
||||
#include <errno.h>
|
||||
#include <debug.h>
|
||||
|
||||
#include <nuttx/spi/spi.h>
|
||||
#include <nuttx/spi/spi_bitbang.h>
|
||||
@@ -123,17 +124,17 @@ static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t ch);
|
||||
static void spi_exchange(FAR struct spi_dev_s *dev,
|
||||
FAR const void *txbuffer, FAR void *rxbuffer,
|
||||
size_t nwords);
|
||||
static uint8_t spi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid);
|
||||
#ifdef CONFIG_SPI_CMDDATA
|
||||
static int spi_cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid,
|
||||
bool cmd);
|
||||
#endif
|
||||
#ifndef CONFIG_SPI_EXCHANGE
|
||||
static void spi_sndblock(FAR struct spi_dev_s *dev,
|
||||
FAR const void *buffer, size_t nwords);
|
||||
static void spi_recvblock(FAR struct spi_dev_s *dev, FAR void *buffer,
|
||||
size_t nwords);
|
||||
#endif
|
||||
static uint8_t spi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid);
|
||||
#ifdef CONFIG_SPI_CMDDATA
|
||||
static int spi_cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid,
|
||||
bool cmd);
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Private Data
|
||||
@@ -259,22 +260,12 @@ static void spi_select(FAR struct spi_dev_s *dev, enum spi_dev_e devid,
|
||||
static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency)
|
||||
{
|
||||
FAR struct spi_bitbang_s *priv = (FAR struct spi_bitbang_s *)dev;
|
||||
uint32_t actual;
|
||||
|
||||
/* SPI frequency cannot be precisely controlled with a bit-bang interface.
|
||||
* Freqency corresponds to delay in toggle the SPI clock line: Set high,
|
||||
* wait, set low, wait, set high, wait, etc.
|
||||
*
|
||||
* Here we calcalute the half period of the frequency in nanoseconds (i.e.,
|
||||
* the amount of time that the clock should remain in the high or low state).
|
||||
*
|
||||
* frequency = psec / 1 sec - psec = full period in seconds
|
||||
* psec = 1 sec / frequency
|
||||
* hpsec = 1 sec / (2 * frequency) - hpsec = half period in seconds
|
||||
* hpnsec = 1000000000 / (2 * frequency) - hpnsec = half period in nanoseconds
|
||||
*/
|
||||
|
||||
priv->hpnsec = (1000000000ul + frequency) / (frequency << 2);
|
||||
return frequency;
|
||||
DEBUGASSERT(priv && priv->low->setfrequency);
|
||||
actual = priv->low->setfrequency(priv, frequency);
|
||||
spivdbg("frequency=%d holdtime=%d actual=%d\n",
|
||||
frequency, priv->holdtime, actual);
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
@@ -297,6 +288,7 @@ static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode)
|
||||
FAR struct spi_bitbang_s *priv = (FAR struct spi_bitbang_s *)dev;
|
||||
DEBUGASSERT(priv && priv->low->setmode);
|
||||
priv->low->setmode(priv, mode);
|
||||
spivdbg("mode=%d exchange=%p\n", mode, priv->exchange);
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
@@ -327,6 +319,7 @@ static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
|
||||
DEBUGASSERT(nbits == 8);
|
||||
#endif
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: spi_send
|
||||
*
|
||||
@@ -382,6 +375,7 @@ static void spi_exchange(FAR struct spi_dev_s *dev,
|
||||
uint16_t dataout;
|
||||
uint16_t datain;
|
||||
|
||||
spivdbg("txbuffer=%p rxbuffer=%p nwords=%d\n", txbuffer, rxbuffer, nwords);
|
||||
DEBUGASSERT(priv && priv->low && priv->low->exchange);
|
||||
|
||||
/* If there is no data source, send 0xff */
|
||||
@@ -496,6 +490,56 @@ static void spi_recvblock(FAR struct spi_dev_s *dev, FAR void *buffer, size_t nw
|
||||
}
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Name: spi_status
|
||||
*
|
||||
* Description:
|
||||
* Get status bits associated with the device associated with 'devid'
|
||||
*
|
||||
* Input Parameters:
|
||||
* dev - Device-specific state data
|
||||
* devid - Identifies the device of interest
|
||||
*
|
||||
* Returned Value:
|
||||
* Bit encoded status byte
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
static uint8_t spi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid)
|
||||
{
|
||||
FAR struct spi_bitbang_s *priv = (FAR struct spi_bitbang_s *)dev;
|
||||
DEBUGASSERT(priv && priv->low && priv->low->status);
|
||||
|
||||
return priv->low->status(priv, devid);
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: spi_cmddata
|
||||
*
|
||||
* Description:
|
||||
* Control the SPI CMD/DATA like for the device associated with 'devid'
|
||||
*
|
||||
* Input Parameters:
|
||||
* dev - Device-specific state data
|
||||
* devid - Identifies the device of interest
|
||||
* cmd - True:CMD False:DATA
|
||||
*
|
||||
* Returned Value:
|
||||
* OK on success; a negated errno value on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
#ifdef CONFIG_SPI_CMDDATA
|
||||
static int spi_cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid,
|
||||
bool cmd)
|
||||
{
|
||||
FAR struct spi_bitbang_s *priv = (FAR struct spi_bitbang_s *)dev;
|
||||
DEBUGASSERTcmddata(priv && priv->low && priv->low->status);
|
||||
|
||||
return priv->low->cmddata(priv, devid, cmd);
|
||||
}
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
@@ -531,10 +575,12 @@ FAR struct spi_dev_s *spi_create_bitbang(FAR const struct spi_bitbang_ops_s *low
|
||||
priv->nbits = 8;
|
||||
#endif
|
||||
|
||||
sem_init(&priv->exclsem, 0, 1);
|
||||
|
||||
/* Select an initial state of mode 0, 8-bits, and 400KHz */
|
||||
|
||||
low->setmode(priv, SPIDEV_MODE0);
|
||||
spi_setfrequency(&priv->dev, 400000);
|
||||
low->setfrequency(priv, 400000);
|
||||
|
||||
/* And return the initialized driver structure */
|
||||
|
||||
|
||||
Reference in New Issue
Block a user