summaryrefslogtreecommitdiff
path: root/nuttx/arch/z16
diff options
context:
space:
mode:
authorGregory Nutt <gnutt@nuttx.org>2014-04-24 12:52:34 -0600
committerGregory Nutt <gnutt@nuttx.org>2014-04-24 12:52:34 -0600
commite18ac0d6899d10b9d25c1e40f62d46d18cf71b30 (patch)
treed4f46b44f0edf2672bd8d658d888caf70e6a4de7 /nuttx/arch/z16
parent8ac8a70d3896f58a0190ed05bfe93b185a36ac63 (diff)
downloadnuttx-e18ac0d6899d10b9d25c1e40f62d46d18cf71b30.tar.gz
nuttx-e18ac0d6899d10b9d25c1e40f62d46d18cf71b30.tar.bz2
nuttx-e18ac0d6899d10b9d25c1e40f62d46d18cf71b30.zip
ZNEO: Add ESPI driver
Diffstat (limited to 'nuttx/arch/z16')
-rw-r--r--nuttx/arch/z16/src/z16f/Kconfig20
-rw-r--r--nuttx/arch/z16/src/z16f/Make.defs25
-rw-r--r--nuttx/arch/z16/src/z16f/chip.h109
-rw-r--r--nuttx/arch/z16/src/z16f/z16f_espi.c890
4 files changed, 1031 insertions, 13 deletions
diff --git a/nuttx/arch/z16/src/z16f/Kconfig b/nuttx/arch/z16/src/z16f/Kconfig
index eaa490057..d825701e3 100644
--- a/nuttx/arch/z16/src/z16f/Kconfig
+++ b/nuttx/arch/z16/src/z16f/Kconfig
@@ -6,6 +6,13 @@
if ARCH_CHIP_Z16F
comment "Z16F Configuration Options"
+menu "Z16F Peripheral Selection"
+
+config Z16F_ESPI
+ bool "ESPI"
+ default n
+ select SPI
+
# UART0/1 always enabled
config Z16F_UART0
@@ -18,4 +25,15 @@ config Z16F_UART1
default y
select ARCH_HAVE_UART1
-endif
+endmenu # Z16F Peripheral Selection
+
+menu "Z16F ESPI Configuration"
+ depends on Z16F_ESPI
+
+config Z16F_ESPI_REGDEBUG
+ bool "ESPI register-level debug"
+ default n
+ depends on DEBUG
+
+endmenu # Z16F ESPI Configuration
+endif # ARCH_CHIP_Z16F
diff --git a/nuttx/arch/z16/src/z16f/Make.defs b/nuttx/arch/z16/src/z16f/Make.defs
index 4146c1221..99a9fdda6 100644
--- a/nuttx/arch/z16/src/z16f/Make.defs
+++ b/nuttx/arch/z16/src/z16f/Make.defs
@@ -1,7 +1,7 @@
############################################################################
# arch/z16/src/z16f/Make.defs
#
-# Copyright (C) 2008 Gregory Nutt. All rights reserved.
+# Copyright (C) 2008, 2014 Gregory Nutt. All rights reserved.
# Author: Gregory Nutt <gnutt@nuttx.org>
#
# Redistribution and use in source and binary forms, with or without
@@ -33,16 +33,19 @@
#
############################################################################
-HEAD_SSRC = z16f_head.S
+HEAD_SSRC = z16f_head.S
-CMN_SSRCS =
-CMN_CSRCS = up_allocateheap.c up_initialize.c up_schedulesigaction.c \
- up_assert.c up_initialstate.c up_sigdeliver.c up_blocktask.c \
- up_interruptcontext.c up_stackdump.c up_copystate.c \
- up_mdelay.c up_udelay.c up_createstack.c up_registerdump.c \
- up_unblocktask.c up_doirq.c up_releasepending.c up_usestack.c \
- up_exit.c up_releasestack.c up_idle.c up_reprioritizertr.c
+CMN_SSRCS =
+CMN_CSRCS = up_allocateheap.c up_initialize.c up_schedulesigaction.c
+CMN_CSRCS += up_assert.c up_initialstate.c up_sigdeliver.c up_blocktask.c
+CMN_CSRCS += up_interruptcontext.c up_stackdump.c up_copystate.c
+CMN_CSRCS += up_mdelay.c up_udelay.c up_createstack.c up_registerdump.c
+CMN_CSRCS += up_unblocktask.c up_doirq.c up_releasepending.c up_usestack.c
+CMN_CSRCS += up_exit.c up_releasestack.c up_idle.c up_reprioritizertr.c
-CHIP_SSRCS = z16f_lowuart.S z16f_saveusercontext.S z16f_restoreusercontext.S
-CHIP_CSRCS = z16f_clkinit.c z16f_sysexec.c z16f_irq.c z16f_timerisr.c z16f_serial.c
+CHIP_SSRCS = z16f_lowuart.S z16f_saveusercontext.S z16f_restoreusercontext.S
+CHIP_CSRCS = z16f_clkinit.c z16f_sysexec.c z16f_irq.c z16f_timerisr.c z16f_serial.c
+ifeq ($(CONFIG_Z16F_ESPI),y)
+CHIP_CSRCS += z16f_espi.c
+endif
diff --git a/nuttx/arch/z16/src/z16f/chip.h b/nuttx/arch/z16/src/z16f/chip.h
index 05ce812c0..484977b2e 100644
--- a/nuttx/arch/z16/src/z16f/chip.h
+++ b/nuttx/arch/z16/src/z16f/chip.h
@@ -41,10 +41,16 @@
************************************************************************************/
#include <nuttx/config.h>
+
#ifndef __ASSEMBLY__
# include <stdint.h>
#endif
+
#include <arch/irq.h>
+#if !defined(__ASSEMBLY__) && defined(CONFIG_Z16F_ESPI)
+# include <nuttx/spi/spi.h>
+#endif
+
#include "common/up_internal.h"
/************************************************************************************
@@ -454,6 +460,87 @@
#define Z16F_UARTMDSEL_HWREV _HX8(e0) /* Bits 5-7=7: LIN-UART Hardware Revision */
/* Bits 0-4: Mode dependent status */
+/* ESPI registers *******************************************************************/
+
+#define Z16F_ESPI_DATA _HX32(ffffe260) /* 8-bit: ESPI Data */
+#define Z16F_ESPI_DCR _HX32(ffffe261) /* 8-bit: ESPI Transmit Data Command */
+#define Z16F_ESPI_CTL _HX32(ffffe262) /* 8-bit: ESPI Control */
+#define Z16F_ESPI_MODE _HX32(ffffe263) /* 8-bit: ESPI Mode */
+#define Z16F_ESPI_STAT _HX32(ffffe264) /* 8-bit: ESPI Status */
+#define Z16F_ESPI_STATE _HX32(ffffe265) /* 8-bit: ESPI State */
+#define Z16F_ESPI_BR _HX32(ffffe266) /* 16-bit: ESPI Baud Rate High Byte */
+# define Z16F_ESPI_BRH _HX32(ffffe266) /* 8-bit: ESPI Baud Rate High Byte */
+# define Z16F_ESPI_BRL _HX32(ffffe267) /* 8-bit: ESPI Baud Rate Low Byte */
+
+/* ESPI register bit definitions ****************************************************/
+
+#define Z16F_ESPI_DCR_SSV _HX8(01) /* Bit 0: Slave Select Value */
+#define Z16F_ESPI_DCR_TEOF _HX8(02) /* Bit 1: Transmit End of Frame */
+
+#define Z16F_ESPI_CTL_ESPIEN0 _HX8(01) /* Bit 0: ESPI Enable and Direction Control */
+#define Z16F_ESPI_CTL_MMEN _HX8(02) /* Bit 1: ESPI Master Mode Enable */
+#define Z16F_ESPI_CTL_WOR _HX8(04) /* Bit 2: Wire-OR (Open-Drain) Mode Enabled */
+#define Z16F_ESPI_CTL_CLKPOL _HX8(08) /* Bit 3: Clock Polarity */
+#define Z16F_ESPI_CTL_PHASE _HX8(10) /* Bit 4: Phase Select */
+#define Z16F_ESPI_CTL_BRGCTL _HX8(20) /* Bit 5: Baud Rate Generator Control */
+#define Z16F_ESPI_CTL_ESPIEN1 _HX8(40) /* Bit 6: ESPI Enable and Direction Control */
+#define Z16F_ESPI_CTL_DIRQE _HX8(80) /* Bit 7: Data Interrupt Request Enable */
+
+#define Z16F_ESPI_MODE_SSPO _HX8(01) /* Bit 0: Slave Select Polarity */
+#define Z16F_ESPI_MODE_SSIO _HX8(02) /* Bit 1: Slave Select I/O */
+#define Z16F_ESPI_MODE_NUMBITS_SHIFT (2) /* Bits 2-4: Number of Data Bits Per Character */
+#define Z16F_ESPI_MODE_NUMBITS_MASK (7 << Z16F_ESPI_MODE_NUMBITS_SHIFT)
+# define Z16F_ESPI_MODE_NUMBITS_8BITS (0 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 8 bits */
+# define Z16F_ESPI_MODE_NUMBITS_1BIT (1 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 1 bit */
+# define Z16F_ESPI_MODE_NUMBITS_2BITS (2 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 2 bits */
+# define Z16F_ESPI_MODE_NUMBITS_3BITS (3 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 3 bits */
+# define Z16F_ESPI_MODE_NUMBITS_4BITS (4 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 4 bits */
+# define Z16F_ESPI_MODE_NUMBITS_5BITS (5 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 5 bits */
+# define Z16F_ESPI_MODE_NUMBITS_6BITS (6 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 6 bits */
+# define Z16F_ESPI_MODE_NUMBITS_7BITS (7 << Z16F_ESPI_MODE_NUMBITS_SHIFT) /* 7 bits */
+#define Z16F_ESPI_MODE_SSMD_SHIFT (5) /* Bits 5-7: SLAVE SELECT Mode */
+#define Z16F_ESPI_MODE_SSMD_MASK (7 << Z16F_ESPI_MODE_SSMD_SHIFT)
+# define Z16F_ESPI_MODE_SSMD_SPI (0 << Z16F_ESPI_MODE_SSMD_SHIFT) /* SPI mode */
+# define Z16F_ESPI_MODE_SSMD_LPBK (1 << Z16F_ESPI_MODE_SSMD_SHIFT) /* LOOPBACK Mode */
+# define Z16F_ESPI_MODE_SSMD_I2S (2 << Z16F_ESPI_MODE_SSMD_SHIFT) /* I2S Mode */
+
+#define Z16F_ESPI_STAT_SLAS _HX8(01) /* Bit 0: Slave Select */
+#define Z16F_ESPI_STAT_TFST _HX8(02) /* Bit 1: Transfer Status */
+#define Z16F_ESPI_STAT_RDRF _HX8(04) /* Bit 2: Receive Data Register Full */
+#define Z16F_ESPI_STAT_ROVR _HX8(08) /* Bit 3: Receive Overrun */
+#define Z16F_ESPI_STAT_ABT _HX8(10) /* Bit 4: Slave mode transaction abort */
+#define Z16F_ESPI_STAT_COL _HX8(20) /* Bit 5: Collision */
+#define Z16F_ESPI_STAT_TUND _HX8(40) /* Bit 6: Transmit Underrun */
+#define Z16F_ESPI_STAT_TDRE _HX8(80) /* Bit 7: Transmit Data Register Empty */
+
+#define Z16F_ESPI_STATE_SHIFT (0) /* Bits 0-5: ESPI State Machine */
+#define Z16F_ESPI_STATE_MASK (0x3f << Z16F_ESPI_STATE_SHIFT)
+# define Z16F_ESPI_STATE_IDLE (0x00 << Z16F_ESPI_STATE_SHIFT) /* Idle */
+# define Z16F_ESPI_STATE_SWAIT (0x01 << Z16F_ESPI_STATE_SHIFT) /* Slave Wait For SCK */
+# define Z16F_ESPI_STATE_I2SSD0 (0x02 << Z16F_ESPI_STATE_SHIFT) /* I2S slave mode start delay */
+# define Z16F_ESPI_STATE_I2SSD1 (0x03 << Z16F_ESPI_STATE_SHIFT) /* I2S slave mode start delay */
+# define Z16F_ESPI_STATE_SPIMD (0x10 << Z16F_ESPI_STATE_SHIFT) /* SPI master mode start delay */
+# define Z16F_ESPI_STATE_I2SMD0 (0x31 << Z16F_ESPI_STATE_SHIFT) /* I2S master mode start delay */
+# define Z16F_ESPI_STATE_I2SMD1 (0x32 << Z16F_ESPI_STATE_SHIFT) /* I2S master mode start delay */
+# define Z16F_ESPI_STATE_B7RCV (0x2e << Z16F_ESPI_STATE_SHIFT) /* Bit 7 Receive */
+# define Z16F_ESPI_STATE_B7XMT (0x2f << Z16F_ESPI_STATE_SHIFT) /* Bit 7 Transmit */
+# define Z16F_ESPI_STATE_B6RCV (0x2c << Z16F_ESPI_STATE_SHIFT) /* Bit 6 Receive */
+# define Z16F_ESPI_STATE_B6XMT (0x2d << Z16F_ESPI_STATE_SHIFT) /* Bit 6 Transmit */
+# define Z16F_ESPI_STATE_B5RCV (0x2a << Z16F_ESPI_STATE_SHIFT) /* Bit 5 Receive */
+# define Z16F_ESPI_STATE_B5XMT (0x2b << Z16F_ESPI_STATE_SHIFT) /* Bit 5 Transmit */
+# define Z16F_ESPI_STATE_B4RCV (0x28 << Z16F_ESPI_STATE_SHIFT) /* Bit 4 Receive */
+# define Z16F_ESPI_STATE_B4XMT (0x29 << Z16F_ESPI_STATE_SHIFT) /* Bit 4 Transmit */
+# define Z16F_ESPI_STATE_B3RCV (0x26 << Z16F_ESPI_STATE_SHIFT) /* Bit 3 Receive */
+# define Z16F_ESPI_STATE_B3XMT (0x27 << Z16F_ESPI_STATE_SHIFT) /* Bit 3 Transmit */
+# define Z16F_ESPI_STATE_B2RCV (0x24 << Z16F_ESPI_STATE_SHIFT) /* Bit 2 Receive */
+# define Z16F_ESPI_STATE_B2XMT (0x25 << Z16F_ESPI_STATE_SHIFT) /* Bit 2 Transmit */
+# define Z16F_ESPI_STATE_B1RCV (0x22 << Z16F_ESPI_STATE_SHIFT) /* Bit 1 Receive */
+# define Z16F_ESPI_STATE_B1XMT (0x23 << Z16F_ESPI_STATE_SHIFT) /* Bit 1 Transmit */
+# define Z16F_ESPI_STATE_B0RCV (0x20 << Z16F_ESPI_STATE_SHIFT) /* Bit 0 Receive */
+# define Z16F_ESPI_STATE_B0XMT (0x21 << Z16F_ESPI_STATE_SHIFT) /* Bit 0 Transmit */
+#define Z16F_ESPI_STATE_SDI _HX8(40) /* Bit 6: Serial Data Input */
+#define Z16F_ESPI_STATE_SCKI _HX8(80) /* Bit 7: Serial Clock Input */
+
/* Timer0/1/2 registers *************************************************************/
#define Z16F_TIMER0_HL _HX32(ffffe300) /* 16-bit: Timer 0 */
@@ -575,7 +662,7 @@ void z16f_lowinit(void);
void z16f_lowuartinit(void);
#endif
-/* This function handles Z16F system execeptions */
+/* This function handles Z16F system exceptions */
void z16f_sysexec(FAR chipreg_t *regs);
@@ -583,6 +670,26 @@ void z16f_sysexec(FAR chipreg_t *regs);
void z16f_reset(void);
+/* The following must be provided by board-specific logic that uses the ZNeo
+ * ESPI
+ */
+
+#ifdef CONFIG_Z16F_ESPI
+/* Select an SPI device (see include/nuttx/spi/spi.h) */
+
+void z16f_espi_select(FAR struct spi_dev_s *dev, enum spi_dev_e devid, bool selected);
+
+/* Provide SPI device status (see include/nuttx/spi/spi.h) */
+
+uint8_t z16f_espi_status(FAR struct spi_dev_s *dev, enum spi_dev_e devid);
+
+/* Select CMD/DATA options (often used with LCD devices) */
+
+#ifdef CONFIG_SPI_CMDDATA
+int z16f_espi_cmddata(FAR struct spi_dev_s *dev, enum spi_dev_e devid, bool cmd);
+#endif
+#endif
+
#undef EXTERN
#ifdef __cplusplus
}
diff --git a/nuttx/arch/z16/src/z16f/z16f_espi.c b/nuttx/arch/z16/src/z16f/z16f_espi.c
new file mode 100644
index 000000000..e0366f3b3
--- /dev/null
+++ b/nuttx/arch/z16/src/z16f/z16f_espi.c
@@ -0,0 +1,890 @@
+/****************************************************************************
+ * arch/z16/src/z16f/z16f_espi.c
+ *
+ * Copyright (C) 2014 Gregory Nutt. All rights reserved.
+ * Authors: 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.
+ *
+ ****************************************************************************/
+
+/****************************************************************************
+ * Included Files
+ ****************************************************************************/
+
+#include <nuttx/config.h>
+
+#include <stdint.h>
+#include <stdbool.h>
+#include <semaphore.h>
+#include <assert.h>
+#include <errno.h>
+#include <debug.h>
+
+#include <arch/board/board.h>
+#include <nuttx/spi/spi.h>
+
+#include "up_arch.h"
+#include "chip.h"
+
+#ifdef CONFIG_Z16F_ESPI
+
+/****************************************************************************
+ * Pre-processor Definitions
+ ****************************************************************************/
+/* Debug *******************************************************************/
+/* Check if SPI debug is enabled (non-standard.. no support in
+ * include/debug.h
+ */
+
+#ifndef CONFIG_DEBUG
+# undef CONFIG_DEBUG_VERBOSE
+# undef CONFIG_DEBUG_SPI
+# undef CONFIG_Z16F_ESPI_REGDEBUG
+#endif
+
+#ifdef CONFIG_DEBUG_SPI
+# define spidbg lldbg
+# ifdef CONFIG_DEBUG_VERBOSE
+# define spivdbg lldbg
+# else
+# define spivdbg (void)
+# endif
+#else
+# define spidbg (void)
+# define spivdbg (void)
+#endif
+
+/****************************************************************************
+ * Private Types
+ ****************************************************************************/
+
+/* The overall state of one SPI controller */
+
+struct z16f_spi_s
+{
+ struct spi_dev_s spi; /* Externally visible part of the SPI interface */
+ bool initialized; /* TRUE: Controller has been initialized */
+#ifndef CONFIG_SPI_OWNBUS
+ uint8_t nbits; /* Width of word in bits (1-8) */
+ uint8_t mode; /* Mode 0,1,2,3 */
+ sem_t exclsem; /* Assures mutually exclusive access to SPI */
+ uint32_t frequency; /* Requested clock frequency */
+ uint32_t actual; /* Actual clock frequency */
+#endif
+
+ /* Debug stuff */
+
+#ifdef CONFIG_Z16F_ESPI_REGDEBUG
+ bool wr; /* Last was a write */
+ uint16_t regval; /* Last value */
+ int ntimes; /* Number of times */
+ uintptr_t regaddr; /* Last address */
+#endif
+};
+
+/****************************************************************************
+ * Private Function Prototypes
+ ****************************************************************************/
+
+/* Helpers */
+
+#ifdef CONFIG_Z16F_ESPI_REGDEBUG
+static bool spi_checkreg(FAR struct z16f_spi_s *priv, bool wr,
+ uint16_t regval, uintptr_t regaddr);
+static uint8_t spi_getreg8(FAR struct z16f_spi_s *priv, uintptr_t regaddr);
+static void spi_putreg8(FAR struct z16f_spi_s *priv, uint8_t regval,
+ uintptr_t regaddr);
+static void spi_putreg16(FAR struct z16f_spi_s *priv, uint16_t regval,
+ uintptr_t regaddr);
+#else
+# define spi_getreg8(priv,regaddr) getreg8(regaddr)
+# define spi_putreg8(priv,regval,regaddr) putreg8(regval, regaddr)
+# define spi_putreg16(priv,regval,regaddr) putreg16(regval, regaddr)
+#endif
+
+#if defined(CONFIG_DEBUG_SPI) && defined(CONFIG_DEBUG_VERBOSE)
+static void spi_dumpregs(FAR struct z16f_spi_s *priv, const char *msg);
+#else
+# define spi_dumpregs(priv,msg)
+#endif
+
+static void spi_flush(FAR struct z16f_spi_s *priv);
+
+/* SPI methods */
+
+#ifndef CONFIG_SPI_OWNBUS
+static int spi_lock(FAR struct spi_dev_s *dev, bool lock);
+#endif
+static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency);
+static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode);
+static void spi_setbits(FAR struct spi_dev_s *dev, int nbits);
+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);
+#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
+
+/****************************************************************************
+ * Private Data
+ ****************************************************************************/
+
+/* SEPI driver operations */
+
+static const struct spi_ops_s g_epsiops =
+{
+#ifndef CONFIG_SPI_OWNBUS
+ spi_lock,
+#endif
+ z16f_espi_select,
+ spi_setfrequency,
+ spi_setmode,
+ spi_setbits,
+ z16f_espi_status,
+#ifdef CONFIG_SPI_CMDDATA
+ z16f_espi_cmddata,
+#endif
+ spi_send,
+#ifdef CONFIG_SPI_EXCHANGE
+ spi_exchange,
+#else
+ spi_sndblock,
+ spi_recvblock,
+#endif
+ NULL /* registercallback: Not implemented */
+};
+
+/* ESPI driver state */
+
+static struct z16f_spi_s g_espi;
+
+/****************************************************************************
+ * Public Data
+ ****************************************************************************/
+
+/****************************************************************************
+ * Private Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: spi_checkreg
+ *
+ * Description:
+ * Check if the current register access is a duplicate of the preceding.
+ *
+ * Input Parameters:
+ * wr - true:write false:read
+ * regval - The value to be written
+ * regaddr - The address of the register to write to
+ *
+ * Returned Value:
+ * true: This is the first register access of this type.
+ * flase: This is the same as the preceding register access.
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_Z16F_ESPI_REGDEBUG
+static bool spi_checkreg(FAR struct z16f_spi_s *priv, bool wr, uint16_t regval,
+ uintptr_t regaddr)
+{
+ if (wr == priv->wr && /* Same kind of access? */
+ regval == priv->regval && /* Same value? */
+ regaddr == priv->regaddr) /* Same address? */
+ {
+ /* Yes, then just keep a count of the number of times we did this. */
+
+ priv->ntimes++;
+ return false;
+ }
+ else
+ {
+ /* Did we do the previous operation more than once? */
+
+ if (priv->ntimes > 0)
+ {
+ /* Yes... show how many times we did it */
+
+ lldbg("...[Repeats %d times]...\n", priv->ntimes);
+ }
+
+ /* Save information about the new access */
+
+ priv->wr = wr;
+ priv->regval = regval;
+ priv->regaddr = regaddr;
+ priv->ntimes = 0;
+ }
+
+ /* Return true if this is the first time that we have done this operation */
+
+ return true;
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_getreg8
+ *
+ * Description:
+ * Read an SPI register
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_Z16F_ESPI_REGDEBUG
+static uint8_t spi_getreg8(FAR struct z16f_spi_s *priv, uintptr_t regaddr)
+{
+ uint8_t regval = getreg8(regaddr);
+
+ if (spi_checkreg(priv, false, (uint16_t)regval, regaddr))
+ {
+ lldbg("%06x->%02x\n", regaddr, regval);
+ }
+
+ return regval;
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_putreg8
+ *
+ * Description:
+ * Write a value to an SPI register
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_Z16F_ESPI_REGDEBUG
+static void spi_putreg8(FAR struct z16f_spi_s *priv, uint8_t regval,
+ uintptr_t regaddr)
+{
+ if (spi_checkreg(priv, true, (uint16_t)regval, regaddr))
+ {
+ lldbg("%06x<-%02x\n", regaddr, regval);
+ }
+
+ putreg8(regval, regaddr);
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_putreg16
+ *
+ * Description:
+ * Write a value to an SPI register
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_Z16F_ESPI_REGDEBUG
+static void spi_putreg16(FAR struct z16f_spi_s *priv, uint16_t regval,
+ uintptr_t regaddr)
+{
+ if (spi_checkreg(priv, true, regval, regaddr))
+ {
+ lldbg("%06x<-%04x\n", regaddr, regval);
+ }
+
+ putreg8(regval, regaddr);
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_dumpregs
+ *
+ * Description:
+ * Dump the contents of all SPI registers
+ *
+ * Input Parameters:
+ * priv - The SPI controller to dump
+ * msg - Message to print before the register data
+ *
+ * Returned Value:
+ * None
+ *
+ ****************************************************************************/
+
+#if defined(CONFIG_DEBUG_SPI) && defined(CONFIG_DEBUG_VERBOSE)
+static void spi_dumpregs(FAR struct z16f_spi_s *priv, FAR const char *msg)
+{
+ spivdbg("%s:\n", msg);
+ spivdbg(" DCR: %02x CTL: %02x MODE: %02x STAT: %02x\n",
+ getreg8(Z16F_ESPI_DCR), getreg8(Z16F_ESPI_CTL),
+ getreg8(Z16F_ESPI_MODE), getreg8(Z16F_ESPI_STAT));
+ spivdbg(" STATE: %02x BR: %02x %02x\n",
+ getreg8(Z16F_ESPI_STATE), getreg8(Z16F_ESPI_BRH),
+ getreg8(Z16F_ESPI_BRL));
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_flush
+ *
+ * Description:
+ * Make sure that there are now dangling SPI transfer in progress
+ *
+ * Input Parameters:
+ * priv - SPI controller state
+ *
+ * Returned Value:
+ * None
+ *
+ ****************************************************************************/
+
+static void spi_flush(FAR struct z16f_spi_s *priv)
+{
+ /* Make sure the no transfer is in progress... waiting if necessary */
+
+ while ((spi_getreg8(priv, Z16F_ESPI_STAT) & Z16F_ESPI_STAT_TFST) != 0);
+
+ /* Then make sure that there is no pending RX data .. reading as
+ * discarding as necessary.
+ */
+
+ while ((spi_getreg8(priv, Z16F_ESPI_STAT) & Z16F_ESPI_STAT_RDRF) != 0)
+ {
+ (void)spi_getreg8(priv, Z16F_ESPI_DATA);
+ }
+}
+
+/****************************************************************************
+ * Name: spi_lock
+ *
+ * Description:
+ * On SPI buses where there are multiple devices, it will be necessary to
+ * lock SPI to have exclusive access to the buses for a sequence of
+ * transfers. The bus should be locked before the chip is selected. After
+ * locking the SPI bus, the caller should then also call the setfrequency,
+ * setbits, and setmode methods to make sure that the SPI is properly
+ * configured for the device. If the SPI buss is being shared, then it
+ * may have been left in an incompatible state.
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * lock - true: Lock SPI bus, false: unlock SPI bus
+ *
+ * Returned Value:
+ * None
+ *
+ ****************************************************************************/
+
+#ifndef CONFIG_SPI_OWNBUS
+static int spi_lock(FAR struct spi_dev_s *dev, bool lock)
+{
+ FAR struct z16f_spi_s *priv = (FAR struct z16f_spi_s *)dev;
+
+ spivdbg("lock=%d\n", lock);
+ if (lock)
+ {
+ /* Take the semaphore (perhaps waiting) */
+
+ while (sem_wait(&priv->exclsem) != 0)
+ {
+ /* The only case that an error should occur here is if the wait was awakened
+ * by a signal.
+ */
+
+ ASSERT(errno == EINTR);
+ }
+ }
+ else
+ {
+ (void)sem_post(&priv->exclsem);
+ }
+
+ return OK;
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_setfrequency
+ *
+ * Description:
+ * Set the SPI frequency.
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * frequency - The SPI frequency requested
+ *
+ * Returned Value:
+ * Returns the actual frequency selected
+ *
+ ****************************************************************************/
+
+static uint32_t spi_setfrequency(FAR struct spi_dev_s *dev, uint32_t frequency)
+{
+ FAR struct z16f_spi_s *priv = (FAR struct z16f_spi_s *)dev;
+ uint32_t actual;
+ uint32_t brg;
+
+ spivdbg("frequency=%d\n", frequency);
+
+ /* Check if the requested frequency is the same as the frequency selection */
+
+#ifndef CONFIG_SPI_OWNBUS
+ if (priv->frequency == frequency)
+ {
+ /* We are already at this frequency. Return the actual. */
+
+ return priv->actual;
+ }
+#endif
+
+ /* Fbaud = Fsystem / (2 * BRG)
+ * BRG = Fsystem / (2 * Fbaud)
+ *
+ * Example, Fsystem = 18.432MHz, Fbaud = 9600
+ * BRG = 960
+ */
+
+ brg = (BOARD_SYSTEM_FREQUENCY >> 1) / frequency;
+ if (brg > 0xffff)
+ {
+ brg = 0xffff;
+ }
+
+ /* Save the new BRG setting */
+
+ spi_putreg16(priv, (uint16_t)brg, Z16F_ESPI_BR);
+
+ /* Calculate the new actual frequency */
+
+ actual = (BOARD_SYSTEM_FREQUENCY >> 1) / brg;
+ spivdbg("BR=%04x actual=%ld\n", (unsigned int)brg, (long)actual);
+
+ /* Save the frequency setting */
+
+#ifndef CONFIG_SPI_OWNBUS
+ priv->frequency = frequency;
+ priv->actual = actual;
+#endif
+
+ spidbg("Frequency %d->%d\n", frequency, actual);
+ return actual;
+}
+
+/****************************************************************************
+ * Name: spi_setmode
+ *
+ * Description:
+ * Set the SPI mode. Optional. See enum spi_mode_e for mode definitions
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * mode - The SPI mode requested
+ *
+ * Returned Value:
+ * none
+ *
+ ****************************************************************************/
+
+static void spi_setmode(FAR struct spi_dev_s *dev, enum spi_mode_e mode)
+{
+ FAR struct z16f_spi_s *priv = (FAR struct z16f_spi_s *)dev;
+ uint8_t regval;
+
+ spivdbg("mode=%d\n", mode);
+
+ /* Has the mode changed? */
+
+#ifndef CONFIG_SPI_OWNBUS
+ if (mode != priv->mode)
+ {
+#endif
+ /* Yes... Set the mode appropriately:
+ *
+ * SPI CPOL CPHA
+ * MODE
+ * 0 0 0
+ * 1 0 1
+ * 2 1 0
+ * 3 1 1
+ */
+
+ regval = spi_getreg8(priv, Z16F_ESPI_CTL);
+ regval &= ~(Z16F_ESPI_CTL_PHASE | Z16F_ESPI_CTL_CLKPOL);
+
+ switch (mode)
+ {
+ case SPIDEV_MODE0: /* CPOL=0; NCPHA=1 */
+ break;
+
+ case SPIDEV_MODE1: /* CPOL=0; NCPHA=0 */
+ regval |= Z16F_ESPI_CTL_PHASE;
+ break;
+
+ case SPIDEV_MODE2: /* CPOL=1; NCPHA=1 */
+ regval |= Z16F_ESPI_CTL_CLKPOL;
+ break;
+
+ case SPIDEV_MODE3: /* CPOL=1; NCPHA=0 */
+ regval |= (Z16F_ESPI_CTL_PHASE | Z16F_ESPI_CTL_CLKPOL);
+ break;
+
+ default:
+ DEBUGASSERT(false);
+ return;
+ }
+
+ spi_putreg8(priv, regval, Z16F_ESPI_CTL);
+ spivdbg("ESPI CTL: %02x\n", regval);
+
+ /* Save the mode so that subsequent re-configurations will be faster */
+
+#ifndef CONFIG_SPI_OWNBUS
+ priv->mode = mode;
+ }
+#endif
+}
+
+/****************************************************************************
+ * Name: spi_setbits
+ *
+ * Description:
+ * Set the number if bits per word.
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * nbits - The number of bits requests
+ *
+ * Returned Value:
+ * none
+ *
+ ****************************************************************************/
+
+static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
+{
+ FAR struct z16f_spi_s *priv = (FAR struct z16f_spi_s *)dev;
+ uint8_t regval;
+
+ spivdbg("nbits=%d\n", nbits);
+ DEBUGASSERT(priv && nbits > 0 && nbits <= 8);
+
+ /* Has the number of bits changed? */
+
+#ifndef CONFIG_SPI_OWNBUS
+ if (nbits != priv->nbits)
+#endif
+ {
+ /* Yes... Set number of bits appropriately */
+
+ regval = spi_getreg8(priv, Z16F_ESPI_MODE);
+ regval &= ~Z16F_ESPI_MODE_NUMBITS_MASK;
+
+ /* The register value of zero is 8-bit */
+
+ if (nbits < 8)
+ {
+ regval |= ((uint8_t)nbits << Z16F_ESPI_MODE_NUMBITS_SHIFT);
+ }
+
+ spi_putreg8(priv, regval, Z16F_ESPI_MODE);
+ spivdbg("ESPI MODE: %02x\n", regval);
+
+#ifndef CONFIG_SPI_OWNBUS
+ /* Save the selection so the subsequence re-configurations will be faster */
+
+ priv->nbits = nbits;
+#endif
+ }
+}
+
+/****************************************************************************
+ * Name: spi_send
+ *
+ * Description:
+ * Exchange one word on SPI
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * wd - The word to send. the size of the data is determined by the
+ * number of bits selected for the SPI interface.
+ *
+ * Returned Value:
+ * response
+ *
+ ****************************************************************************/
+
+static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t wd)
+{
+ uint8_t txbyte;
+ uint8_t rxbyte;
+
+ /* spi_exchange can do this. Note: right now, this only deals with 8-bit
+ * words. If the SPI interface were configured for words of other sizes,
+ * this would fail.
+ */
+
+ txbyte = (uint8_t)wd;
+ rxbyte = (uint8_t)0;
+ spi_exchange(dev, &txbyte, &rxbyte, 1);
+
+ spivdbg("Sent %02x received %02x\n", txbyte, rxbyte);
+ return (uint16_t)rxbyte;
+}
+
+/****************************************************************************
+ * Name: spi_exchange
+ *
+ * Description:
+ * Exchange a block of data from SPI.
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * txbuffer - A pointer to the buffer of data to be sent
+ * rxbuffer - A pointer to the buffer in which to receive data
+ * nwords - the length of data that to be exchanged in units of words.
+ * The wordsize is determined by the number of bits-per-word
+ * selected for the SPI interface. If nbits <= 8, the data is
+ * packed into uint8_t's; if nbits >8, the data is packed into
+ * uint16_t's
+ *
+ * Returned Value:
+ * None
+ *
+ ****************************************************************************/
+
+static void spi_exchange(FAR struct spi_dev_s *dev, FAR const void *txbuffer,
+ FAR void *rxbuffer, size_t nwords)
+{
+ FAR struct z16f_spi_s *priv = (struct z16f_spi_s *)dev;
+ uint8_t data;
+ FAR uint8_t *rxptr = rxbuffer;
+ FAR const uint8_t *txptr = txbuffer;
+
+ spivdbg("txbuffer=%p rxbuffer=%p nwords=%d\n", txbuffer, rxbuffer, nwords);
+
+ /* Make sure that any previous transfer is flushed from the hardware */
+
+ spi_flush(priv);
+
+ /* Make sure the the TEOF bit is not set (SSV must also be zero) */
+
+ spi_putreg8(priv, 0, Z16F_ESPI_CTL);
+
+ /* Loop, sending each word in the user-provided data buffer.
+ *
+ * Note 2: This loop might be made more efficient. Would logic
+ * like the following improve the throughput? Or would it
+ * just add the risk of overruns?
+ *
+ * Get byte 1;
+ * Send byte 1; Now word 1 is "in flight"
+ * nwords--;
+ * for ( ; byte > 0; byte--)
+ * {
+ * Get byte N.
+ * Wait for TDRE meaning that byte N-1 has moved to the shift
+ * register.
+ * Disable interrupts to keep the following atomic
+ * Send byte N. Now both work N-1 and N are "in flight"
+ * Wait for RDRF meaning that byte N-1 is available
+ * Read byte N-1.
+ * Re-enable interrupts.
+ * Save byte N-1.
+ * }
+ * Wait for RDRF meaning that the final byte is available
+ * Read the final byte.
+ * Save the final byte.
+ */
+
+ for ( ; nwords > 0; nwords--)
+ {
+ /* Get the data to send (0xff if there is no data source). */
+
+ if (txptr)
+ {
+ data = *txptr++;
+ }
+ else
+ {
+ data = 0xff;
+ }
+
+ /* Do we need to set the TEOF bit in the CTL register too? */
+
+ if (nwords == 1)
+ {
+ spi_putreg8(priv, Z16F_ESPI_DCR_TEOF, Z16F_ESPI_CTL);
+ }
+
+ /* Wait for any transmit data register to be empty. */
+
+ while ((spi_getreg8(priv, Z16F_ESPI_STAT) & Z16F_ESPI_STAT_TDRE) == 0);
+
+ /* Write the data to transmitted to the Transmit Data Register (TDR) */
+
+ spi_putreg8(priv, data, Z16F_ESPI_DATA);
+
+ /* Wait for the read data to be available in the data regsiter */
+
+ while ((spi_getreg8(priv, Z16F_ESPI_STAT) & Z16F_ESPI_STAT_RDRF) == 0);
+
+ /* Read the received data from the SPI Data Register. */
+
+ data = spi_getreg8(priv, Z16F_ESPI_DATA);
+ if (rxptr)
+ {
+ *rxptr++ = (uint8_t)data;
+ }
+ }
+}
+
+/***************************************************************************
+ * Name: spi_sndblock
+ *
+ * Description:
+ * Send a block of data on SPI
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * buffer - A pointer to the buffer of data to be sent
+ * nwords - the length of data to send from the buffer in number of words.
+ * The wordsize is determined by the number of bits-per-word
+ * selected for the SPI interface. If nbits <= 8, the data is
+ * packed into uint8_t's; if nbits >8, the data is packed into
+ * uint16_t's
+ *
+ * Returned Value:
+ * None
+ *
+ ****************************************************************************/
+
+#ifndef CONFIG_SPI_EXCHANGE
+static void spi_sndblock(FAR struct spi_dev_s *dev, FAR const void *buffer,
+ size_t nwords)
+{
+ /* spi_exchange can do this. */
+
+ spi_exchange(dev, buffer, NULL, nwords);
+}
+#endif
+
+/****************************************************************************
+ * Name: spi_recvblock
+ *
+ * Description:
+ * Revice a block of data from SPI
+ *
+ * Input Parameters:
+ * dev - Device-specific state data
+ * buffer - A pointer to the buffer in which to receive data
+ * nwords - the length of data that can be received in the buffer in number
+ * of words. The wordsize is determined by the number of bits-per-word
+ * selected for the SPI interface. If nbits <= 8, the data is
+ * packed into uint8_t's; if nbits >8, the data is packed into
+ * uint16_t's
+ *
+ * Returned Value:
+ * None
+ *
+ ****************************************************************************/
+
+#ifndef CONFIG_SPI_EXCHANGE
+static void spi_recvblock(FAR struct spi_dev_s *dev, FAR void *buffer,
+ size_t nwords)
+{
+ /* spi_exchange can do this. */
+
+ spi_exchange(dev, NULL, buffer, nwords);
+}
+#endif
+
+/****************************************************************************
+ * Public Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: up_spiinitialize
+ *
+ * Description:
+ * Initialize the selected SPI port
+ *
+ * Input Parameter:
+ * port - Identifies the "logical" SPI port. Must be zero in this case.
+ *
+ * Returned Value:
+ * Valid SPI device structure reference on success; a NULL on failure
+ *
+ ****************************************************************************/
+
+struct spi_dev_s *up_spiinitialize(int port)
+{
+ FAR struct z16f_spi_s *priv;
+ irqstate_t flags;
+#ifndef CONFIG_SPI_OWNBUS
+ uint8_t regval;
+ unsigned int offset;
+#endif
+
+ spivdbg("port: %d\n", port);
+ DEBUGASSERT(port == 0);
+
+ /* Check if we have already initialized the ESPI */
+
+ priv = &g_espi;
+ if (priv->initialized)
+ {
+ /* Initialize the ESPI state structure */
+
+ flags = irqsave();
+ priv->spi.ops = &g_epsiops;
+#ifndef CONFIG_SPI_OWNBUS
+ sem_init(&priv->exclsem, 0, 1);
+#endif
+
+ /* Initialize the hardware. Mode 0, 8-bits, 400KHz */
+
+ spi_putreg8(priv, 0x00, Z16F_ESPI_CTL); /* Disabled the ESPI */
+ spi_putreg8(priv, 0x00, Z16F_ESPI_DCR); /* Disabled slave select; clear TEOF */
+
+ regval = Z16F_ESPI_MODE_SSIO | Z16F_ESPI_MODE_NUMBITS_8BITS | Z16F_ESPI_MODE_SSMD_SPI;
+ spi_putreg8(priv, regval, Z16F_ESPI_MODE); /* SPI mode, 8-bit */
+
+ regval = Z16F_ESPI_CTL_ESPIEN0 | Z16F_ESPI_CTL_MMEN | Z16F_ESPI_CTL_ESPIEN1;
+ spi_putreg8(priv, 0x00, Z16F_ESPI_CTL); /* TX/RX mode, Master mode */
+
+ /* Make sure that we are all in agreement about the configuration and set
+ * the BRG for 400KHz operation.
+ */
+
+ (void)spi_setfrequency(&priv->spi, 400000);
+ spi_setmode(&priv->spi, SPIDEV_MODE0);
+ spi_setbits(&priv->spi, 8);
+
+ /* Now we are initialized */
+
+ priv->initialized = true;
+ irqrestore(flags);
+ }
+
+ spi_dumpregs(priv, "After initialization");
+ return &priv->spi;
+}
+
+#endif /* CONFIG_Z16F_ESPI */