Fix buffer overrun and memory leak on smartfs and improvement to cxd56xx

Author: Alan Carvalho de Assis <acassis@gmail.com>

    Check all .c and .h against nxstyle and fix it.

Author: Alin Jerpelea <alin.jerpelea@sony.com>

    fs: smartfs: Fix over capacity write

    When the remaining capacity of flash is one sector, if a new root
    directory is created by file open, then the root directory's chain is
    broken and it causes to SmartFS filesystem crash. Once this fatal
    problem occurs, it's impossible to recover even if the system reboot.
    Fix it by finally update link of root directory.

    fs: smartfs: Fix buffer overrun

    fs: smartfs: Fix uninitialized variable warnings

    fs: smartfs: Memory leak fix

    boards: cxd56xx: Update spresense board.h

    - Fix PMIC assignment
    - Add specific pin configurations for spresense
    - Remove unnecessary definitions

    arch: cxd56xx: Add ITM syslog init at startup

    arch: cxd56xx: Enable DMA settings dynamically
This commit is contained in:
Alin Jerpelea 2020-01-08 07:51:11 -03:00 committed by Alan Carvalho de Assis
parent 3abdc352a8
commit 3e45517599
13 changed files with 443 additions and 224 deletions

View File

@ -89,6 +89,10 @@ CMN_CSRCS += up_copyarmstate.c
endif endif
endif endif
ifeq ($(CONFIG_ARMV7M_ITMSYSLOG),y)
CMN_CSRCS += up_itm_syslog.c
endif
CHIP_ASRCS += cxd56_farapistub.S CHIP_ASRCS += cxd56_farapistub.S
CHIP_CSRCS = cxd56_allocateheap.c cxd56_idle.c CHIP_CSRCS = cxd56_allocateheap.c cxd56_idle.c

View File

@ -46,6 +46,7 @@
#include <semaphore.h> #include <semaphore.h>
#include <errno.h> #include <errno.h>
#include <debug.h> #include <debug.h>
#include <string.h>
#include <arch/board/board.h> #include <arch/board/board.h>
#include <nuttx/irq.h> #include <nuttx/irq.h>
@ -58,7 +59,7 @@
#include "chip.h" #include "chip.h"
#include "cxd56_spi.h" #include "cxd56_spi.h"
#include "chip/cxd56_spi.h" #include "hardware/cxd56_spi.h"
#include "cxd56_clock.h" #include "cxd56_clock.h"
#include "cxd56_pinconfig.h" #include "cxd56_pinconfig.h"
#include "cxd56_powermgr.h" #include "cxd56_powermgr.h"
@ -99,6 +100,7 @@ struct cxd56_spidev_s
uint8_t port; /* Port number */ uint8_t port; /* Port number */
int initialized; /* Initialized flag */ int initialized; /* Initialized flag */
#ifdef CONFIG_CXD56_DMAC #ifdef CONFIG_CXD56_DMAC
bool dmaenable; /* Use DMA or not */
DMA_HANDLE rxdmach; /* RX DMA channel handle */ DMA_HANDLE rxdmach; /* RX DMA channel handle */
DMA_HANDLE txdmach; /* TX DMA channel handle */ DMA_HANDLE txdmach; /* TX DMA channel handle */
sem_t dmasem; /* Wait for DMA to complete */ sem_t dmasem; /* Wait for DMA to complete */
@ -187,19 +189,10 @@ static const struct spi_ops_s g_spi4ops =
#endif #endif
.send = spi_send, .send = spi_send,
#ifdef CONFIG_SPI_EXCHANGE #ifdef CONFIG_SPI_EXCHANGE
# if defined(CONFIG_CXD56_DMAC_SPI4_TX) || defined(CONFIG_CXD56_DMAC_SPI4_RX)
.exchange = spi_dmaexchange,
# else
.exchange = spi_exchange, .exchange = spi_exchange,
# endif
#else #else
# if defined(CONFIG_CXD56_DMAC_SPI4_TX) || defined(CONFIG_CXD56_DMAC_SPI4_RX)
.sndblock = spi_dmasndblock,
.recvblock = spi_dmarecvblock,
# else
.sndblock = spi_sndblock, .sndblock = spi_sndblock,
.recvblock = spi_recvblock, .recvblock = spi_recvblock,
# endif
#endif #endif
#ifdef CONFIG_SPI_CALLBACK #ifdef CONFIG_SPI_CALLBACK
.registercallback = cxd56_spi4register, /* Provided externally */ .registercallback = cxd56_spi4register, /* Provided externally */
@ -210,7 +203,10 @@ static const struct spi_ops_s g_spi4ops =
static struct cxd56_spidev_s g_spi4dev = static struct cxd56_spidev_s g_spi4dev =
{ {
.spidev = { &g_spi4ops }, .spidev =
{
&g_spi4ops
},
.spibase = CXD56_IMG_SPI_BASE, .spibase = CXD56_IMG_SPI_BASE,
.spibasefreq = 0, .spibasefreq = 0,
.port = 4, .port = 4,
@ -236,19 +232,10 @@ static const struct spi_ops_s g_spi5ops =
#endif #endif
.send = spi_send, .send = spi_send,
#ifdef CONFIG_SPI_EXCHANGE #ifdef CONFIG_SPI_EXCHANGE
# if defined(CONFIG_CXD56_DMAC_SPI5_TX) || defined(CONFIG_CXD56_DMAC_SPI5_RX)
.exchange = spi_dmaexchange,
# else
.exchange = spi_exchange, .exchange = spi_exchange,
# endif
#else #else
# if defined(CONFIG_CXD56_DMAC_SPI5_TX) || defined(CONFIG_CXD56_DMAC_SPI5_RX)
.sndblock = spi_dmasndblock,
.recvblock = spi_dmarecvblock,
# else
.sndblock = spi_sndblock, .sndblock = spi_sndblock,
.recvblock = spi_recvblock, .recvblock = spi_recvblock,
# endif
#endif #endif
#ifdef CONFIG_SPI_CALLBACK #ifdef CONFIG_SPI_CALLBACK
.registercallback = cxd56_spi5register, /* Provided externally */ .registercallback = cxd56_spi5register, /* Provided externally */
@ -259,7 +246,10 @@ static const struct spi_ops_s g_spi5ops =
static struct cxd56_spidev_s g_spi5dev = static struct cxd56_spidev_s g_spi5dev =
{ {
.spidev = { &g_spi5ops }, .spidev =
{
&g_spi5ops
},
.spibase = CXD56_IMG_WSPI_BASE, .spibase = CXD56_IMG_WSPI_BASE,
.spibasefreq = 0, .spibasefreq = 0,
.port = 5, .port = 5,
@ -298,7 +288,10 @@ static const struct spi_ops_s g_spi0ops =
static struct cxd56_spidev_s g_spi0dev = static struct cxd56_spidev_s g_spi0dev =
{ {
.spidev = { &g_spi0ops }, .spidev =
{
&g_spi0ops
},
.spibase = CXD56_SPIM_BASE, .spibase = CXD56_SPIM_BASE,
.spibasefreq = 0, .spibasefreq = 0,
.port = 0, .port = 0,
@ -337,7 +330,10 @@ static const struct spi_ops_s g_spi3ops =
static struct cxd56_spidev_s g_spi3dev = static struct cxd56_spidev_s g_spi3dev =
{ {
.spidev = { &g_spi3ops }, .spidev =
{
&g_spi3ops
},
.spibase = CXD56_SCU_SPI_BASE, .spibase = CXD56_SCU_SPI_BASE,
.spibasefreq = 0, .spibasefreq = 0,
.port = 3, .port = 3,
@ -623,6 +619,7 @@ static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
*/ */
priv->nbits = nbits; priv->nbits = nbits;
#ifdef CONFIG_CXD56_DMAC
if (priv->nbits > 8) if (priv->nbits > 8)
{ {
priv->txconfig.dest_width = CXD56_DMAC_WIDTH16; priv->txconfig.dest_width = CXD56_DMAC_WIDTH16;
@ -637,6 +634,7 @@ static void spi_setbits(FAR struct spi_dev_s *dev, int nbits)
priv->rxconfig.dest_width = CXD56_DMAC_WIDTH8; priv->rxconfig.dest_width = CXD56_DMAC_WIDTH8;
priv->rxconfig.src_width = CXD56_DMAC_WIDTH8; priv->rxconfig.src_width = CXD56_DMAC_WIDTH8;
} }
#endif
} }
} }
@ -706,10 +704,10 @@ static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t wd)
} }
/**************************************************************************** /****************************************************************************
* Name: spi_exchange * Name: spi_do_exchange
* *
* Description: * Description:
* Exahange a block of data from SPI. Required. * Exchange a block of data from SPI. Required.
* *
* Input Parameters: * Input Parameters:
* dev - Device-specific state data * dev - Device-specific state data
@ -726,8 +724,8 @@ static uint16_t spi_send(FAR struct spi_dev_s *dev, uint16_t wd)
* *
****************************************************************************/ ****************************************************************************/
static void spi_exchange(FAR struct spi_dev_s *dev, FAR const void *txbuffer, static void spi_do_exchange(FAR struct spi_dev_s *dev, FAR const void *txbuffer,
FAR void *rxbuffer, size_t nwords) FAR void *rxbuffer, size_t nwords)
{ {
FAR struct cxd56_spidev_s *priv = (FAR struct cxd56_spidev_s *)dev; FAR struct cxd56_spidev_s *priv = (FAR struct cxd56_spidev_s *)dev;
uint32_t regval = 0; uint32_t regval = 0;
@ -738,12 +736,14 @@ static void spi_exchange(FAR struct spi_dev_s *dev, FAR const void *txbuffer,
FAR const uint16_t *p16; FAR const uint16_t *p16;
FAR const void *pv; FAR const void *pv;
} tx; } tx;
union union
{ {
FAR uint8_t *p8; FAR uint8_t *p8;
FAR uint16_t *p16; FAR uint16_t *p16;
FAR void *pv; FAR void *pv;
} rx; } rx;
uint32_t data; uint32_t data;
uint32_t datadummy = (priv->nbits > 8) ? 0xffff : 0xff; uint32_t datadummy = (priv->nbits > 8) ? 0xffff : 0xff;
uint32_t rxpending = 0; uint32_t rxpending = 0;
@ -828,6 +828,44 @@ static void spi_exchange(FAR struct spi_dev_s *dev, FAR const void *txbuffer,
cxd56_spi_clock_gate_enable(priv->port); cxd56_spi_clock_gate_enable(priv->port);
} }
/****************************************************************************
* Name: spi_exchange
*
* Description:
* Wrapper function of 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)
{
#ifdef CONFIG_CXD56_DMAC
FAR struct cxd56_spidev_s *priv = (FAR struct cxd56_spidev_s *)dev;
if (priv->dmaenable)
{
spi_dmaexchange(dev, txbuffer, rxbuffer, nwords);
}
else
#endif
{
spi_do_exchange(dev, txbuffer, rxbuffer, nwords);
}
}
/**************************************************************************** /****************************************************************************
* Name: spi_sndblock * Name: spi_sndblock
* *
@ -909,6 +947,7 @@ static void cxd56_spi_pincontrol(int ch, bool on)
CXD56_PIN_CONFIGS(PINCONFS_SPI0_GPIO); CXD56_PIN_CONFIGS(PINCONFS_SPI0_GPIO);
} }
#endif #endif
#ifdef CONFIG_CXD56_SPI0_PINMAP_SPIFLASH #ifdef CONFIG_CXD56_SPI0_PINMAP_SPIFLASH
if (on) if (on)
{ {
@ -932,6 +971,7 @@ static void cxd56_spi_pincontrol(int ch, bool on)
{ {
CXD56_PIN_CONFIGS(PINCONFS_SPI3_GPIO); CXD56_PIN_CONFIGS(PINCONFS_SPI3_GPIO);
} }
#ifdef CONFIG_CXD56_SPI3_CS0 #ifdef CONFIG_CXD56_SPI3_CS0
if (on) if (on)
{ {
@ -942,6 +982,7 @@ static void cxd56_spi_pincontrol(int ch, bool on)
CXD56_PIN_CONFIGS(PINCONFS_SPI3_CS0_X_GPIO); CXD56_PIN_CONFIGS(PINCONFS_SPI3_CS0_X_GPIO);
} }
#endif #endif
#ifdef CONFIG_CXD56_SPI3_CS1 #ifdef CONFIG_CXD56_SPI3_CS1
if (on) if (on)
{ {
@ -952,6 +993,7 @@ static void cxd56_spi_pincontrol(int ch, bool on)
CXD56_PIN_CONFIGS(PINCONFS_SPI3_CS1_X_GPIO); CXD56_PIN_CONFIGS(PINCONFS_SPI3_CS1_X_GPIO);
} }
#endif #endif
#ifdef CONFIG_CXD56_SPI3_CS2 #ifdef CONFIG_CXD56_SPI3_CS2
if (on) if (on)
{ {
@ -990,6 +1032,7 @@ static void cxd56_spi_pincontrol(int ch, bool on)
CXD56_PIN_CONFIGS(PINCONFS_EMMCA_GPIO); CXD56_PIN_CONFIGS(PINCONFS_EMMCA_GPIO);
} }
#endif #endif
#ifdef CONFIG_CXD56_SPI5_PINMAP_SDIO #ifdef CONFIG_CXD56_SPI5_PINMAP_SDIO
if (on) if (on)
{ {
@ -1144,54 +1187,10 @@ FAR struct spi_dev_s *cxd56_spibus_initialize(int port)
/* DMA settings */ /* DMA settings */
#if defined(CONFIG_CXD56_DMAC_SPI4_TX) || defined(CONFIG_CXD56_DMAC_SPI4_RX) #ifdef CONFIG_CXD56_DMAC
if (port == 4) priv->dmaenable = false;
{ priv->txdmach = NULL;
#if defined(CONFIG_CXD56_DMAC_SPI4_TX) priv->rxdmach = NULL;
priv->txconfig.channel_cfg = CXD56_DMA_PERIPHERAL_SPI4_TX;
priv->txdmach = cxd56_dmachannel(CONFIG_CXD56_DMAC_SPI4_TX_CH,
CONFIG_CXD56_DMAC_SPI4_TX_MAXSIZE);
if (priv->txdmach == NULL)
{
return NULL;
}
#endif
#if defined(CONFIG_CXD56_DMAC_SPI4_RX)
priv->rxconfig.channel_cfg = CXD56_DMA_PERIPHERAL_SPI4_RX;
priv->rxdmach = cxd56_dmachannel(CONFIG_CXD56_DMAC_SPI4_RX_CH,
CONFIG_CXD56_DMAC_SPI4_RX_MAXSIZE);
if (priv->rxdmach == NULL)
{
return NULL;
}
#endif
nxsem_init(&priv->dmasem, 0, 0);
}
#endif
#if defined(CONFIG_CXD56_DMAC_SPI5_TX) || defined(CONFIG_CXD56_DMAC_SPI5_RX)
if (port == 5)
{
#if defined(CONFIG_CXD56_DMAC_SPI5_TX)
priv->txconfig.channel_cfg = CXD56_DMA_PERIPHERAL_SPI5_TX;
priv->txdmach = cxd56_dmachannel(CONFIG_CXD56_DMAC_SPI5_TX_CH,
CONFIG_CXD56_DMAC_SPI5_TX_MAXSIZE);
if (priv->txdmach == NULL)
{
return NULL;
}
#endif
#if defined(CONFIG_CXD56_DMAC_SPI5_RX)
priv->rxconfig.channel_cfg = CXD56_DMA_PERIPHERAL_SPI5_RX;
priv->rxdmach = cxd56_dmachannel(CONFIG_CXD56_DMAC_SPI5_RX_CH,
CONFIG_CXD56_DMAC_SPI5_RX_MAXSIZE);
if (priv->rxdmach == NULL)
{
return NULL;
}
#endif
nxsem_init(&priv->dmasem, 0, 0);
}
#endif #endif
/* CS control */ /* CS control */
@ -1263,6 +1262,81 @@ FAR struct spi_dev_s *cxd56_spibus_initialize(int port)
return &priv->spidev; return &priv->spidev;
} }
#ifdef CONFIG_CXD56_DMAC
/****************************************************************************
* Name: cxd56_spi_dmaconfig
*
* Description:
* Enable DMA configuration.
*
* Input Parameter:
* port - Port number
* chtype - Channel type(TX or RX)
* handle - DMA channel handle
* conf - DMA configuration
*
* Returned Value:
* None
*
****************************************************************************/
void cxd56_spi_dmaconfig(int port, int chtype, DMA_HANDLE handle,
FAR dma_config_t *conf)
{
FAR struct cxd56_spidev_s *priv = NULL;
switch (port)
{
#if defined(CONFIG_CXD56_SPI4) && defined(CONFIG_CXD56_DMAC_SPI4_TX)
case 4:
priv = &g_spi4dev;
break;
#endif
#if defined(CONFIG_CXD56_SPI5) && defined(CONFIG_CXD56_DMAC_SPI5_TX)
case 5:
priv = &g_spi5dev;
break;
#endif
default:
break;
}
if (priv && priv->initialized)
{
if ((chtype == CXD56_SPI_DMAC_CHTYPE_TX) && (!priv->txdmach))
{
/* TX DMA setting */
priv->txdmach = handle;
memcpy(&priv->txconfig, conf, sizeof(dma_config_t));
if (!priv->dmaenable)
{
sem_init(&priv->dmasem, 0, 0);
priv->dmaenable = true;
}
}
else if ((chtype == CXD56_SPI_DMAC_CHTYPE_RX) && (!priv->rxdmach))
{
/* RX DMA setting */
priv->rxdmach = handle;
memcpy(&priv->rxconfig, conf, sizeof(dma_config_t));
if (!priv->dmaenable)
{
sem_init(&priv->dmasem, 0, 0);
priv->dmaenable = true;
}
}
}
}
#endif
/**************************************************************************** /****************************************************************************
* Name: spi_flush * Name: spi_flush
* *

View File

@ -44,6 +44,9 @@
#include <nuttx/config.h> #include <nuttx/config.h>
#include <nuttx/spi/spi.h> #include <nuttx/spi/spi.h>
#include "hardware/cxd56_spi.h" #include "hardware/cxd56_spi.h"
#ifdef CONFIG_CXD56_DMAC
#include "cxd56_dmac.h"
#endif
#if defined(CONFIG_CXD56_SPI0) || defined(CONFIG_CXD56_SPI3) || \ #if defined(CONFIG_CXD56_SPI0) || defined(CONFIG_CXD56_SPI3) || \
defined(CONFIG_CXD56_SPI4) || defined(CONFIG_CXD56_SPI5) defined(CONFIG_CXD56_SPI4) || defined(CONFIG_CXD56_SPI5)
@ -51,6 +54,7 @@
/**************************************************************************** /****************************************************************************
* Pre-processor Definitions * Pre-processor Definitions
****************************************************************************/ ****************************************************************************/
/* This header file defines interfaces to common SPI logic. /* This header file defines interfaces to common SPI logic.
* To use this common SPI logic on your board: * To use this common SPI logic on your board:
* *
@ -68,6 +72,9 @@
* for example, will bind the SPI driver to the SPI MMC/SD driver). * for example, will bind the SPI driver to the SPI MMC/SD driver).
*/ */
#define CXD56_SPI_DMAC_CHTYPE_TX (0)
#define CXD56_SPI_DMAC_CHTYPE_RX (1)
/**************************************************************************** /****************************************************************************
* Public Types * Public Types
****************************************************************************/ ****************************************************************************/
@ -107,6 +114,28 @@ extern "C"
FAR struct spi_dev_s *cxd56_spibus_initialize(int port); FAR struct spi_dev_s *cxd56_spibus_initialize(int port);
/****************************************************************************
* Name: cxd56_spi_dmaconfig
*
* Description:
* Enable DMA configuration.
*
* Input Parameter:
* port - Port number
* chtype - Channel type(TX or RX)
* handle - DMA channel handle
* conf - DMA configuration
*
* Returned Value:
* None
*
****************************************************************************/
#ifdef CONFIG_CXD56_DMAC
void cxd56_spi_dmaconfig(int port, int chtype, DMA_HANDLE handle,
FAR dma_config_t *conf);
#endif
/**************************************************************************** /****************************************************************************
* Name: cxd56_spiXselect, cxd56_spiXstatus, and cxd56_spiXcmddata * Name: cxd56_spiXselect, cxd56_spiXstatus, and cxd56_spiXcmddata
* *

View File

@ -320,6 +320,12 @@ void __start(void)
fpuconfig(); fpuconfig();
#ifdef CONFIG_ARMV7M_ITMSYSLOG
/* Perform ARMv7-M ITM SYSLOG initialization */
itm_syslog_initialize();
#endif
/* Perform early serial initialization */ /* Perform early serial initialization */
#ifdef USE_EARLYSERIALINIT #ifdef USE_EARLYSERIALINIT

View File

@ -50,6 +50,7 @@
#include <nuttx/modem/altmdm.h> #include <nuttx/modem/altmdm.h>
#include <arch/board/cxd56_altmdm.h> #include <arch/board/cxd56_altmdm.h>
#include "cxd56_spi.h" #include "cxd56_spi.h"
#include "cxd56_dmac.h"
#include "cxd56_pinconfig.h" #include "cxd56_pinconfig.h"
/**************************************************************************** /****************************************************************************
@ -57,29 +58,27 @@
****************************************************************************/ ****************************************************************************/
#if defined(CONFIG_CXD56_LTE_SPI4) #if defined(CONFIG_CXD56_LTE_SPI4)
#define SPI_CH (4) # define SPI_CH (4)
#if defined(CONFIG_CXD56_LTE_SPI4_DMAC) # if defined(CONFIG_CXD56_LTE_SPI4_DMAC)
# if defined(CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE) # define DMA_TXCH (2)
# if (CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE > CONFIG_CXD56_DMAC_SPI4_TX_MAXSIZE) # define DMA_RXCH (3)
# error CONFIG_CXD56_DMAC_SPI4_TX_MAXSIZE too small # define DMA_TXCHCHG (CXD56_DMA_PERIPHERAL_SPI4_TX)
# endif # define DMA_RXCHCFG (CXD56_DMA_PERIPHERAL_SPI4_RX)
# if (CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE > CONFIG_CXD56_DMAC_SPI4_RX_MAXSIZE) # if !defined(CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE)
# error CONFIG_CXD56_DMAC_SPI4_RX_MAXSIZE too small # error CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE is not set
# endif # endif
# endif # endif
#endif
#elif defined(CONFIG_CXD56_LTE_SPI5) #elif defined(CONFIG_CXD56_LTE_SPI5)
#define SPI_CH (5) # define SPI_CH (5)
#if defined(CONFIG_CXD56_LTE_SPI5_DMAC) # if defined(CONFIG_CXD56_LTE_SPI5_DMAC)
# if defined(CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE) # define DMA_TXCH (4)
# if (CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE > CONFIG_CXD56_DMAC_SPI5_TX_MAXSIZE) # define DMA_RXCH (5)
# error CONFIG_CXD56_DMAC_SPI5_TX_MAXSIZE too small # define DMA_TXCHCHG (CXD56_DMA_PERIPHERAL_SPI5_TX)
# endif # define DMA_RXCHCFG (CXD56_DMA_PERIPHERAL_SPI5_RX)
# if (CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE > CONFIG_CXD56_DMAC_SPI5_RX_MAXSIZE) # if !defined(CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE)
# error CONFIG_CXD56_DMAC_SPI5_RX_MAXSIZE too small # error CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE is not set
# endif # endif
# endif # endif
#endif
#else #else
# error "Select LTE SPI 4 or 5" # error "Select LTE SPI 4 or 5"
#endif #endif
@ -110,44 +109,45 @@ static void spi_pincontrol(int bus, bool on)
switch (bus) switch (bus)
{ {
#ifdef CONFIG_CXD56_SPI4 #ifdef CONFIG_CXD56_SPI4
case 4: case 4:
if (on) if (on)
{ {
CXD56_PIN_CONFIGS(PINCONFS_SPI4); CXD56_PIN_CONFIGS(PINCONFS_SPI4);
} }
else else
{ {
CXD56_PIN_CONFIGS(PINCONFS_SPI4_GPIO); CXD56_PIN_CONFIGS(PINCONFS_SPI4_GPIO);
} }
break; break;
#endif /* CONFIG_CXD56_SPI4 */ #endif /* CONFIG_CXD56_SPI4 */
#ifdef CONFIG_CXD56_SPI5 #ifdef CONFIG_CXD56_SPI5
case 5: case 5:
#ifdef CONFIG_CXD56_SPI5_PINMAP_EMMC #ifdef CONFIG_CXD56_SPI5_PINMAP_EMMC
if (on) if (on)
{ {
CXD56_PIN_CONFIGS(PINCONFS_EMMCA_SPI5); CXD56_PIN_CONFIGS(PINCONFS_EMMCA_SPI5);
} }
else else
{ {
CXD56_PIN_CONFIGS(PINCONFS_EMMCA_GPIO); CXD56_PIN_CONFIGS(PINCONFS_EMMCA_GPIO);
} }
#endif /* CONFIG_CXD56_SPI5_PINMAP_EMMC */ #endif /* CONFIG_CXD56_SPI5_PINMAP_EMMC */
#ifdef CONFIG_CXD56_SPI5_PINMAP_SDIO #ifdef CONFIG_CXD56_SPI5_PINMAP_SDIO
if (on) if (on)
{ {
CXD56_PIN_CONFIGS(PINCONFS_SDIOA_SPI5); CXD56_PIN_CONFIGS(PINCONFS_SDIOA_SPI5);
} }
else else
{ {
CXD56_PIN_CONFIGS(PINCONFS_SDIOA_GPIO); CXD56_PIN_CONFIGS(PINCONFS_SDIOA_GPIO);
} }
#endif /* CONFIG_CXD56_SPI5_PINMAP_SDIO */ #endif /* CONFIG_CXD56_SPI5_PINMAP_SDIO */
break; break;
#endif /* CONFIG_CXD56_SPI5 */ #endif /* CONFIG_CXD56_SPI5 */
default: default:
break; break;
} }
} }
@ -166,7 +166,11 @@ static void spi_pincontrol(int bus, bool on)
int board_altmdm_initialize(FAR const char *devpath) int board_altmdm_initialize(FAR const char *devpath)
{ {
FAR struct spi_dev_s *spi; FAR struct spi_dev_s *spi;
int spi_ch = SPI_CH; int spi_ch = SPI_CH;
#if defined(CONFIG_CXD56_LTE_SPI4_DMAC) || defined(CONFIG_CXD56_LTE_SPI5_DMAC)
DMA_HANDLE hdl;
dma_config_t conf;
#endif
m_info("Initializing ALTMDM..\n"); m_info("Initializing ALTMDM..\n");
@ -181,6 +185,26 @@ int board_altmdm_initialize(FAR const char *devpath)
return -ENODEV; return -ENODEV;
} }
#if defined(CONFIG_CXD56_LTE_SPI4_DMAC) || defined(CONFIG_CXD56_LTE_SPI5_DMAC)
hdl = cxd56_dmachannel(DMA_TXCH, CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE);
if (hdl)
{
conf.channel_cfg = DMA_TXCHCHG;
conf.dest_width = CXD56_DMAC_WIDTH8;
conf.src_width = CXD56_DMAC_WIDTH8;
cxd56_spi_dmaconfig(spi_ch, CXD56_SPI_DMAC_CHTYPE_TX, hdl, &conf);
}
hdl = cxd56_dmachannel(DMA_RXCH, CONFIG_MODEM_ALTMDM_MAX_PACKET_SIZE);
if (hdl)
{
conf.channel_cfg = DMA_RXCHCFG;
conf.dest_width = CXD56_DMAC_WIDTH8;
conf.src_width = CXD56_DMAC_WIDTH8;
cxd56_spi_dmaconfig(spi_ch, CXD56_SPI_DMAC_CHTYPE_RX, hdl, &conf);
}
#endif
spi_pincontrol(spi_ch, false); spi_pincontrol(spi_ch, false);
g_devhandle = altmdm_register(devpath, spi); g_devhandle = altmdm_register(devpath, spi);

View File

@ -334,6 +334,10 @@ int board_lcd_initialize(void)
{ {
FAR struct ili93404ws_lcd_s *priv = &g_lcddev; FAR struct ili93404ws_lcd_s *priv = &g_lcddev;
FAR struct spi_dev_s *spi; FAR struct spi_dev_s *spi;
#if defined(CONFIG_CXD56_DMAC)
DMA_HANDLE hdl;
dma_config_t conf;
#endif
lcdinfo("Initializing lcd\n"); lcdinfo("Initializing lcd\n");
@ -345,8 +349,33 @@ int board_lcd_initialize(void)
lcderr("ERROR: Failed to initialize spi bus.\n"); lcderr("ERROR: Failed to initialize spi bus.\n");
return -ENODEV; return -ENODEV;
} }
priv->spi = spi; priv->spi = spi;
#if defined(CONFIG_CXD56_DMAC)
/* DMA settings */
hdl = cxd56_dmachannel(DISPLAY_DMA_TXCH, DISPLAY_DMA_TX_MAXSIZE);
if (hdl)
{
conf.channel_cfg = DISPLAY_DMA_TXCH_CFG;
conf.dest_width = CXD56_DMAC_WIDTH8;
conf.src_width = CXD56_DMAC_WIDTH8;
cxd56_spi_dmaconfig(DISPLAY_SPI, CXD56_SPI_DMAC_CHTYPE_TX,
hdl, &conf);
}
hdl = cxd56_dmachannel(DISPLAY_DMA_RXCH, DISPLAY_DMA_RX_MAXSIZE);
if (hdl)
{
conf.channel_cfg = DISPLAY_DMA_RXCH_CFG;
conf.dest_width = CXD56_DMAC_WIDTH8;
conf.src_width = CXD56_DMAC_WIDTH8;
cxd56_spi_dmaconfig(DISPLAY_SPI, CXD56_SPI_DMAC_CHTYPE_RX,
hdl, &conf);
}
#endif
/* Reset ILI9340 */ /* Reset ILI9340 */
up_mdelay(10); up_mdelay(10);
@ -394,5 +423,6 @@ FAR struct lcd_dev_s *board_lcd_getdev(int lcddev)
{ {
return g_lcd; return g_lcd;
} }
return NULL; return NULL;
} }

View File

@ -334,6 +334,10 @@ int board_lcd_initialize(void)
{ {
FAR struct lpm013m091a_lcd_s *priv = &g_lcddev; FAR struct lpm013m091a_lcd_s *priv = &g_lcddev;
FAR struct spi_dev_s *spi; FAR struct spi_dev_s *spi;
#if defined(CONFIG_CXD56_DMAC)
DMA_HANDLE hdl;
dma_config_t conf;
#endif
lcdinfo("Initializing lcd\n"); lcdinfo("Initializing lcd\n");
@ -345,8 +349,33 @@ int board_lcd_initialize(void)
lcderr("ERROR: Failed to initialize spi bus.\n"); lcderr("ERROR: Failed to initialize spi bus.\n");
return -ENODEV; return -ENODEV;
} }
priv->spi = spi; priv->spi = spi;
#if defined(CONFIG_CXD56_DMAC)
/* DMA settings */
hdl = cxd56_dmachannel(DISPLAY_DMA_TXCH, DISPLAY_DMA_TX_MAXSIZE);
if (hdl)
{
conf.channel_cfg = DISPLAY_DMA_TXCH_CFG;
conf.dest_width = CXD56_DMAC_WIDTH8;
conf.src_width = CXD56_DMAC_WIDTH8;
cxd56_spi_dmaconfig(DISPLAY_SPI, CXD56_SPI_DMAC_CHTYPE_TX,
hdl, &conf);
}
hdl = cxd56_dmachannel(DISPLAY_DMA_RXCH, DISPLAY_DMA_RX_MAXSIZE);
if (hdl)
{
conf.channel_cfg = DISPLAY_DMA_RXCH_CFG;
conf.dest_width = CXD56_DMAC_WIDTH8;
conf.src_width = CXD56_DMAC_WIDTH8;
cxd56_spi_dmaconfig(DISPLAY_SPI, CXD56_SPI_DMAC_CHTYPE_RX,
hdl, &conf);
}
#endif
/* Reset LPM013M091A */ /* Reset LPM013M091A */
up_mdelay(10); up_mdelay(10);
@ -394,5 +423,6 @@ FAR struct lcd_dev_s *board_lcd_getdev(int lcddev)
{ {
return g_lcd; return g_lcd;
} }
return NULL; return NULL;
} }

View File

@ -59,12 +59,16 @@ choice
config LCD_ON_EXTENSION_BOARD config LCD_ON_EXTENSION_BOARD
bool "Extension board: SPI4" bool "Extension board: SPI4"
select CXD56_SPI4 select CXD56_SPI4
select CXD56_DMAC_SPI4_TX
select CXD56_DMAC_SPI4_RX
---help--- ---help---
Display connected to extension board. Display connected to extension board.
config LCD_ON_MAIN_BOARD config LCD_ON_MAIN_BOARD
bool "Main board: SPI5" bool "Main board: SPI5"
select CXD56_SPI5 select CXD56_SPI5
select CXD56_DMAC_SPI5_TX
select CXD56_DMAC_SPI5_RX
---help--- ---help---
Display connected to main board. Display connected to main board.

View File

@ -80,63 +80,19 @@
/* Clocking ****************************************************************/ /* Clocking ****************************************************************/
#define BOARD_XTAL_FREQUENCY (26000000) /* XTAL oscillator frequency */
#define BOARD_RTCCLK_FREQUENCY (32768) /* RTC oscillator frequency */
#define BOARD_INTRCOSC_FREQUENCY (8192000) /* Internal RC oscillator frequency */
#ifdef CONFIG_CXD56_80MHz #ifdef CONFIG_CXD56_80MHz
# define BOARD_FCLKOUT_FREQUENCY (80000000) # define BOARD_FCLKOUT_FREQUENCY (80000000)
#else #else
# define BOARD_FCLKOUT_FREQUENCY (97500000) # define BOARD_FCLKOUT_FREQUENCY (100000000)
#endif
#define CXD56_CCLK BOARD_FCLKOUT_FREQUENCY
/* USB0 ********************************************************************/
#define BOARD_USB0_CLKSRC PLL0USB_CLKSEL_XTAL
#define BOARD_USB0_MDIV 0x06167ffa /* Table 149 datsheet, valid for 12Mhz Fclkin */
#define BOARD_USB0_NP_DIV 0x00302062 /* Table 149 datsheet, valid for 12Mhz Fclkin */
/* SPIFI clocking **********************************************************/
/* The SPIFI will receive clocking from a divider per the settings provided
* in this file. The NuttX code will configure PLL1 as the input clock
* for the selected divider
*/
#undef BOARD_SPIFI_PLL1 /* No division */
#undef BOARD_SPIFI_DIVA /* Supports division by 1-4 */
#undef BOARD_SPIFI_DIVB /* Supports division by 1-16 */
#undef BOARD_SPIFI_DIVC /* Supports division by 1-16 */
#undef BOARD_SPIFI_DIVD /* Supports division by 1-16 */
#undef BOARD_SPIFI_DIVE /* Supports division by 1-256 */
#if BOARD_FCLKOUT_FREQUENCY < 20000000
# define BOARD_SPIFI_PLL1 1 /* Use PLL1 directly */
#else
# define BOARD_SPIFI_DIVB 1 /* Use IDIVB */
#endif
/* We need to configure the divider so that its output is as close to the
* desired SCLK value. The peak data transfer rate will be about half of
* this frequency in bytes per second.
*/
#if BOARD_FCLKOUT_FREQUENCY < 20000000
# define BOARD_SPIFI_FREQUENCY BOARD_FCLKOUT_FREQUENCY /* 72Mhz? */
#else
# define BOARD_SPIFI_DIVIDER (14) /* 204MHz / 14 = 14.57MHz */
# define BOARD_SPIFI_FREQUENCY (102000000) /* 204MHz / 14 = 14.57MHz */
#endif #endif
/* UART clocking ***********************************************************/ /* UART clocking ***********************************************************/
/* Configure all UARTs to use the XTAL input frequency */ /* Configure all UARTs to use the XTAL input frequency */
#define BOARD_UART0_BASEFREQ BOARD_XTAL_FREQUENCY #define BOARD_UART0_BASEFREQ CONFIG_CXD56_XOSC_CLOCK
#define BOARD_UART1_BASEFREQ 48750000 #define BOARD_UART1_BASEFREQ BOARD_FCLKOUT_FREQUENCY
#define BOARD_UART2_BASEFREQ BOARD_XTAL_FREQUENCY #define BOARD_UART2_BASEFREQ CONFIG_CXD56_XOSC_CLOCK
/* LED definitions *********************************************************/ /* LED definitions *********************************************************/
@ -175,19 +131,24 @@
/* Power Control definitions ***********************************************/ /* Power Control definitions ***********************************************/
/* Switch Device /* For SPRESENSE board:
*
* Switch Device
* --------- ------------------------------- * --------- -------------------------------
* LSW2 AcaPulco Audio Digital VDD * LDO_EMMC GNSS A-ANT
* LSW3 SPI-Flash & TCXO * DDC_ANA N/A
* LSW4 GNSS LNA * LDO_PERI N/A
* GPO0 AcaPulco Audio Analog VDD * LSW2 CXD5247 Audio Digital VDD
* GPO1 Sensor 1.8V * LSW3 SPI-Flash
* GPO2 Sensor 3.3V * LSW4 TCXO & GNSS LNA
* GPO3 Bluetooth/Bluetooth Low Energy * GPO0
* GPO4 Image Sensor 1.2V * GPO1 CXD5247 Audio Analog VDD
* GPO5 Image Sensor 3.3V * GPO2
* GPO6 eMMC 3.3V/1.8V * GPO3
* GPO7 Image Sensor 1.8V * GPO4 Camera
* GPO5 Camera
* GPO6 Audio External Amp.
* GPO7 Camera
* *
*/ */
@ -223,16 +184,12 @@ enum board_power_device
POWER_AUDIO_AVDD = PMIC_GPO(1), POWER_AUDIO_AVDD = PMIC_GPO(1),
POWER_AUDIO_MUTE = PMIC_GPO(6), POWER_AUDIO_MUTE = PMIC_GPO(6),
POWER_SENSOR_18V = PMIC_GPO(1),
POWER_SENSOR_33V = PMIC_GPO(2),
POWER_BMI160 = POWER_SENSOR_18V,
POWER_SENSOR = POWER_SENSOR_18V | POWER_SENSOR_33V,
POWER_BTBLE = PMIC_GPO(3),
POWER_EINK = PMIC_NONE,
POWER_EMMC = PMIC_GPO(6),
POWER_LFOUR = PMIC_NONE,
POWER_LTE = PMIC_NONE,
POWER_IMAGE_SENSOR = PMIC_GPO(4) | PMIC_GPO(5) | PMIC_GPO(7), POWER_IMAGE_SENSOR = PMIC_GPO(4) | PMIC_GPO(5) | PMIC_GPO(7),
POWER_BTBLE = PMIC_NONE,
POWER_SENSOR = PMIC_NONE,
POWER_EMMC = PMIC_NONE,
POWER_LTE = PMIC_GPO(2),
}; };
/* CXD5247 audio control definitions ***************************************/ /* CXD5247 audio control definitions ***************************************/
@ -254,6 +211,13 @@ enum board_power_device
#define DISPLAY_SPI 5 #define DISPLAY_SPI 5
#define DISPLAY_DMA_TXCH (4)
#define DISPLAY_DMA_RXCH (5)
#define DISPLAY_DMA_TXCH_CFG CXD56_DMA_PERIPHERAL_SPI5_TX
#define DISPLAY_DMA_RXCH_CFG CXD56_DMA_PERIPHERAL_SPI5_RX
#define DISPLAY_DMA_TX_MAXSIZE (192000)
#define DISPLAY_DMA_RX_MAXSIZE (192000)
#else /* Display is connected through extension board. */ #else /* Display is connected through extension board. */
#define DISPLAY_RST PIN_SPI2_MISO #define DISPLAY_RST PIN_SPI2_MISO
@ -261,6 +225,13 @@ enum board_power_device
#define DISPLAY_SPI 4 #define DISPLAY_SPI 4
#define DISPLAY_DMA_TXCH (2)
#define DISPLAY_DMA_RXCH (3)
#define DISPLAY_DMA_TXCH_CFG CXD56_DMA_PERIPHERAL_SPI4_TX
#define DISPLAY_DMA_RXCH_CFG CXD56_DMA_PERIPHERAL_SPI4_RX
#define DISPLAY_DMA_TX_MAXSIZE (192000)
#define DISPLAY_DMA_RX_MAXSIZE (192000)
#endif #endif
/* Sensor device bus definitions *******************************************/ /* Sensor device bus definitions *******************************************/

View File

@ -53,14 +53,20 @@
* 4mA : Drive Current 1=4mA, 0=2mA * 4mA : Drive Current 1=4mA, 0=2mA
* Pull: 0=HiZ floating, PINCONF_PULLUP, PINCONF_PULLDOWN * Pull: 0=HiZ floating, PINCONF_PULLUP, PINCONF_PULLDOWN
* *
* M E P * M E P
* P o N 4 u * P o N 4 u
* i d Z m l * i d Z m l
* n e I A l * n e I A l
*/ */
#undef PINCONF_UART2_TXD
#undef PINCONF_UART2_RXD
#undef PINCONF_UART2_CTS #undef PINCONF_UART2_CTS
#define PINCONF_UART2_CTS PINCONF(PIN_UART2_CTS, 1, 1, 0, PINCONF_PULLDOWN) #undef PINCONF_UART2_RTS
#define PINCONF_UART2_TXD PINCONF(PIN_UART2_TXD, 1, 0, 1, 0)
#define PINCONF_UART2_RXD PINCONF(PIN_UART2_RXD, 1, 1, 1, 0)
#define PINCONF_UART2_CTS PINCONF(PIN_UART2_CTS, 1, 1, 1, PINCONF_PULLDOWN)
#define PINCONF_UART2_RTS PINCONF(PIN_UART2_RTS, 1, 0, 1, 0)
#undef PINCONF_SPI4_CS_X #undef PINCONF_SPI4_CS_X
#undef PINCONF_SPI4_SCK #undef PINCONF_SPI4_SCK
@ -69,6 +75,19 @@
#define PINCONF_SPI4_SCK PINCONF(PIN_SPI4_SCK, 1, 0, 1, 0) #define PINCONF_SPI4_SCK PINCONF(PIN_SPI4_SCK, 1, 0, 1, 0)
#define PINCONF_SPI4_MOSI PINCONF(PIN_SPI4_MOSI, 1, 0, 1, 0) #define PINCONF_SPI4_MOSI PINCONF(PIN_SPI4_MOSI, 1, 0, 1, 0)
#undef PINCONF_SPI3_CS0_X
#undef PINCONF_SPI3_CS1_X
#undef PINCONF_SPI3_CS2_X
#undef PINCONF_SPI3_SCK
#undef PINCONF_SPI3_MOSI
#undef PINCONF_SPI3_MISO
#define PINCONF_SPI3_CS0_X PINCONF(PIN_SPI3_CS0_X, 1, 0, 1, 0)
#define PINCONF_SPI3_CS1_X PINCONF(PIN_SPI3_CS1_X, 1, 0, 1, 0)
#define PINCONF_SPI3_CS2_X PINCONF(PIN_SPI3_CS2_X, 1, 0, 1, 0)
#define PINCONF_SPI3_SCK PINCONF(PIN_SPI3_SCK, 1, 0, 1, 0)
#define PINCONF_SPI3_MOSI PINCONF(PIN_SPI3_MOSI, 1, 0, 1, 0)
#define PINCONF_SPI3_MISO PINCONF(PIN_SPI3_MISO, 1, 1, 1, 0)
#undef PINCONF_PWM0 #undef PINCONF_PWM0
#undef PINCONF_PWM1 #undef PINCONF_PWM1
#undef PINCONF_PWM2 #undef PINCONF_PWM2

View File

@ -53,12 +53,6 @@
#include "cxd56_clock.h" #include "cxd56_clock.h"
#include "cxd56_gpio.h" #include "cxd56_gpio.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define MMCSD_DETECT PIN_AP_CLK
/**************************************************************************** /****************************************************************************
* Public Functions * Public Functions
****************************************************************************/ ****************************************************************************/

View File

@ -383,6 +383,9 @@ errout_with_buffer:
sf->entry.name = NULL; sf->entry.name = NULL;
} }
#ifdef CONFIG_SMARTFS_USE_SECTOR_BUFFER
kmm_free(sf->buffer);
#endif /* CONFIG_SMARTFS_USE_SECTOR_BUFFER */
kmm_free(sf); kmm_free(sf);
errout_with_semaphore: errout_with_semaphore:

View File

@ -56,6 +56,12 @@
#include "smartfs.h" #include "smartfs.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define WORKBUFFER_SIZE 256
/**************************************************************************** /****************************************************************************
* Private Data * Private Data
****************************************************************************/ ****************************************************************************/
@ -269,7 +275,7 @@ int smartfs_mount(struct smartfs_mountpt_s *fs, bool writeable)
if (nextfs == NULL) if (nextfs == NULL)
{ {
fs->fs_rwbuffer = (char *) kmm_malloc(fs->fs_llformat.availbytes); fs->fs_rwbuffer = (char *) kmm_malloc(fs->fs_llformat.availbytes);
fs->fs_workbuffer = (char *) kmm_malloc(256); fs->fs_workbuffer = (char *) kmm_malloc(WORKBUFFER_SIZE);
} }
/* Now add ourselves to the linked list of SMART mounts */ /* Now add ourselves to the linked list of SMART mounts */
@ -292,12 +298,12 @@ int smartfs_mount(struct smartfs_mountpt_s *fs, bool writeable)
g_mounthead = fs; g_mounthead = fs;
#endif #endif
#endif /* CONFIG_SMARTFS_MULTI_ROOT_DIRS */
fs->fs_rwbuffer = (char *) kmm_malloc(fs->fs_llformat.availbytes); fs->fs_rwbuffer = (char *) kmm_malloc(fs->fs_llformat.availbytes);
fs->fs_workbuffer = (char *) kmm_malloc(256); fs->fs_workbuffer = (char *) kmm_malloc(WORKBUFFER_SIZE);
fs->fs_rootsector = SMARTFS_ROOT_DIR_SECTOR; fs->fs_rootsector = SMARTFS_ROOT_DIR_SECTOR;
#endif /* CONFIG_SMARTFS_MULTI_ROOT_DIRS */
/* We did it! */ /* We did it! */
fs->fs_mounted = TRUE; fs->fs_mounted = TRUE;
@ -486,6 +492,11 @@ int smartfs_finddirentry(struct smartfs_mountpt_s *fs,
struct smart_read_write_s readwrite; struct smart_read_write_s readwrite;
struct smartfs_entry_header_s *entry; struct smartfs_entry_header_s *entry;
/* Set the initial value of the output */
*parentdirsector = 0xffff;
*filename = NULL;
/* Initialize directory level zero as the root sector */ /* Initialize directory level zero as the root sector */
dirstack[0] = fs->fs_rootsector; dirstack[0] = fs->fs_rootsector;
@ -523,6 +534,14 @@ int smartfs_finddirentry(struct smartfs_mountpt_s *fs,
ptr++; ptr++;
} }
/* Check to avoid buffer overflow */
if (seglen >= WORKBUFFER_SIZE)
{
ret = -ENAMETOOLONG;
goto errout;
}
strncpy(fs->fs_workbuffer, segment, seglen); strncpy(fs->fs_workbuffer, segment, seglen);
fs->fs_workbuffer[seglen] = '\0'; fs->fs_workbuffer[seglen] = '\0';
@ -832,6 +851,9 @@ int smartfs_createentry(FAR struct smartfs_mountpt_s *fs,
uint16_t entrysize; uint16_t entrysize;
struct smartfs_entry_header_s *entry; struct smartfs_entry_header_s *entry;
struct smartfs_chain_header_s *chainheader; struct smartfs_chain_header_s *chainheader;
int update_chain = 0;
struct smart_read_write_s update_readwrite;
struct smartfs_chain_header_s update_header;
/* Start at the 1st sector in the parent directory */ /* Start at the 1st sector in the parent directory */
@ -924,19 +946,15 @@ int smartfs_createentry(FAR struct smartfs_mountpt_s *fs,
nextsector = (uint16_t) ret; nextsector = (uint16_t) ret;
/* Chain the next sector into this sector sector */ /* Chain the next sector into this sector. */
*((FAR uint16_t *)chainheader->nextsector) = nextsector; *((uint16_t *)update_header.nextsector) = nextsector;
readwrite.offset = offsetof(struct smartfs_chain_header_s, update_readwrite.logsector = psector;
nextsector); update_readwrite.offset = offsetof(struct smartfs_chain_header_s,
readwrite.count = sizeof(uint16_t); nextsector);
readwrite.buffer = chainheader->nextsector; update_readwrite.count = sizeof(uint16_t);
ret = FS_IOCTL(fs, BIOC_WRITESECT, (unsigned long) &readwrite); update_readwrite.buffer = update_header.nextsector;
if (ret < 0) update_chain = 1;
{
ferr("ERROR: Error chaining sector %d\n", nextsector);
goto errout;
}
} }
/* Now update to the next sector */ /* Now update to the next sector */
@ -1048,6 +1066,19 @@ int smartfs_createentry(FAR struct smartfs_mountpt_s *fs,
goto errout; goto errout;
} }
if (update_chain)
{
/* Update chain header after the next sector was written */
ret = FS_IOCTL(fs, BIOC_WRITESECT, (unsigned long) &update_readwrite);
if (ret < 0)
{
ferr("ERROR: Error chaining sector %d\n",
update_readwrite.logsector);
goto errout;
}
}
/* Now fill in the entry */ /* Now fill in the entry */
direntry->firstsector = nextsector; direntry->firstsector = nextsector;