2109 lines
63 KiB
C
2109 lines
63 KiB
C
/****************************************************************************
|
|
* arch/arm/src/samv7/sam_xdmac.c
|
|
*
|
|
* Copyright (C) 2015 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.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
#include <semaphore.h>
|
|
#include <debug.h>
|
|
#include <errno.h>
|
|
|
|
#include <nuttx/irq.h>
|
|
#include <nuttx/arch.h>
|
|
#include <arch/irq.h>
|
|
#include <arch/samv7/chip.h>
|
|
|
|
#include "up_arch.h"
|
|
#include "cache.h"
|
|
#include "up_internal.h"
|
|
#include "sched/sched.h"
|
|
|
|
#include "sam_xdmac.h"
|
|
#include "sam_periphclks.h"
|
|
#include "chip/sam_pmc.h"
|
|
#include "chip/sam_xdmac.h"
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
/* Configuration ************************************************************/
|
|
|
|
/* Condition out the whole file unless DMA is selected in the configuration */
|
|
|
|
#ifdef CONFIG_SAMV7_XDMAC
|
|
|
|
/* If SAMV7 DMA support is enabled, then OS DMA support should also be
|
|
* enabled
|
|
*/
|
|
|
|
#ifndef CONFIG_ARCH_DMA
|
|
# warning "SAMV7 DMA enabled but CONFIG_ARCH_DMA disabled"
|
|
#endif
|
|
|
|
/* Check the number of link list descriptors to allocate */
|
|
|
|
#ifndef CONFIG_SAMV7_NLLDESC
|
|
# define CONFIG_SAMV7_NLLDESC SAMV7_NDMACHAN
|
|
#endif
|
|
|
|
#if CONFIG_SAMV7_NLLDESC < SAMV7_NDMACHAN
|
|
# warning "At least SAMV7_NDMACHAN descriptors must be allocated"
|
|
|
|
# undef CONFIG_SAMV7_NLLDESC
|
|
# define CONFIG_SAMV7_NLLDESC SAMV7_NDMACHAN
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Private Types
|
|
****************************************************************************/
|
|
/* This structure maps a peripheral ID to an DMA channel */
|
|
|
|
struct sam_pidmap_s
|
|
{
|
|
uint8_t pid; /* Peripheral identifier */
|
|
uint8_t pchan; /* DMA channel */
|
|
};
|
|
|
|
/* This structure describes one DMA channel */
|
|
|
|
struct sam_xdmach_s
|
|
{
|
|
uint8_t chan; /* DMA channel number (0-15) */
|
|
bool inuse; /* TRUE: The DMA channel is in use */
|
|
bool rx; /* TRUE: Peripheral to memory transfer */
|
|
uint32_t flags; /* DMA channel flags */
|
|
uint32_t base; /* DMA register channel base address */
|
|
uint32_t cc; /* Pre-calculated CC register for transfer */
|
|
dma_callback_t callback; /* Callback invoked when the DMA completes */
|
|
void *arg; /* Argument passed to callback function */
|
|
uint32_t rxaddr; /* RX memory address to be invalidated */
|
|
size_t rxsize; /* Size of RX memory region */
|
|
struct chnext_view1_s *llhead; /* DMA link list head */
|
|
struct chnext_view1_s *lltail; /* DMA link list head */
|
|
};
|
|
|
|
/* This structure describes the state of one DMA controller */
|
|
|
|
struct sam_xdmac_s
|
|
{
|
|
/* These semaphores protect the DMA channel and descriptor tables */
|
|
|
|
sem_t chsem; /* Protects channel table */
|
|
sem_t dsem; /* Protects descriptor table */
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Data
|
|
****************************************************************************/
|
|
|
|
/* Channel Control (CC) Register field lookups */
|
|
|
|
static const uint32_t g_chanwidth[3] =
|
|
{
|
|
XDMACH_CC_DWIDTH_BYTE,
|
|
XDMACH_CC_DWIDTH_HWORD,
|
|
XDMACH_CC_DWIDTH_WORD,
|
|
};
|
|
|
|
/* These tables map peripheral IDs to channels. A lookup is performed
|
|
* before each DMA transfer in order to map the peripheral IDs to the
|
|
* correct channel. This must be done because the channel can change with
|
|
* the direction of the transfer.
|
|
*/
|
|
|
|
/* RX DMA: */
|
|
|
|
static const struct sam_pidmap_s g_xdmac_rxchan[] =
|
|
{
|
|
{ SAM_PID_HSMCI0, XDMACH_HSMCI }, /* HSMCI Receive/Transmit */
|
|
{ SAM_PID_SPI0, XDMACH_SPI0_RX }, /* SPI0 Receive */
|
|
{ SAM_PID_SPI1, XDMACH_SPI1_RX }, /* SPI1 Receive */
|
|
{ SAM_PID_QSPI, XDMACH_QSPI_RX }, /* QSPI Receive */
|
|
{ SAM_PID_USART0, XDMACH_USART0_RX }, /* USART0 Receive */
|
|
{ SAM_PID_USART1, XDMACH_USART1_RX }, /* USART1 Receive */
|
|
{ SAM_PID_USART2, XDMACH_USART2_RX }, /* USART2 Receive */
|
|
{ SAM_PID_TWIHS0, XDMACH_TWIHS0_RX }, /* TWIHS0 Receive */
|
|
{ SAM_PID_TWIHS1, XDMACH_TWIHS1_RX }, /* TWIHS1 Receive */
|
|
{ SAM_PID_TWIHS2, XDMACH_TWIHS2_RX }, /* TWIHS2 Receive */
|
|
{ SAM_PID_UART0, XDMACH_UART0_RX }, /* UART0 Receive */
|
|
{ SAM_PID_UART1, XDMACH_UART1_RX }, /* UART1 Receive */
|
|
{ SAM_PID_UART2, XDMACH_UART2_RX }, /* UART2 Receive */
|
|
{ SAM_PID_UART3, XDMACH_UART3_RX }, /* UART3 Receive */
|
|
{ SAM_PID_UART4, XDMACH_UART4_RX }, /* UART4 Receive */
|
|
{ SAM_PID_SSC0, XDMACH_SSC_RX }, /* SSC Receive */
|
|
{ SAM_PID_PIOA, XDMACH_PIOA_RX }, /* PIOA Receive */
|
|
{ SAM_PID_AFEC0, XDMACH_AFEC0_RX }, /* AFEC0 Receive */
|
|
{ SAM_PID_AFEC1, XDMACH_AFEC1_RX }, /* AFEC1 Receive */
|
|
{ SAM_PID_AES, XDMACH_AES_RX }, /* AES Receive */
|
|
{ SAM_PID_TC0, XDMACH_TC0_RX }, /* TC0 Receive */
|
|
{ SAM_PID_TC1, XDMACH_TC1_RX }, /* TC1 Receive */
|
|
{ SAM_PID_TC2, XDMACH_TC2_RX }, /* TC2 Receive */
|
|
{ SAM_PID_TC3, XDMACH_TC3_RX } /* TC3 Receive */
|
|
};
|
|
#define NXDMAC_RXCHANNELS (sizeof(g_xdmac_rxchan) / sizeof(struct sam_pidmap_s))
|
|
|
|
/* TX DMA: */
|
|
|
|
static const struct sam_pidmap_s g_xdmac_txchan[] =
|
|
{
|
|
{ SAM_PID_HSMCI0, XDMACH_HSMCI }, /* HSMCI Receive/Transmit */
|
|
{ SAM_PID_SPI0, XDMACH_SPI0_TX }, /* SPI0 Transmit */
|
|
{ SAM_PID_SPI1, XDMACH_SPI1_TX }, /* SPI1 Transmit */
|
|
{ SAM_PID_QSPI, XDMACH_QSPI_TX }, /* QSPI Transmit */
|
|
{ SAM_PID_USART0, XDMACH_USART0_TX }, /* USART0 Transmit */
|
|
{ SAM_PID_USART1, XDMACH_USART1_TX }, /* USART1 Transmit */
|
|
{ SAM_PID_USART2, XDMACH_USART2_TX }, /* USART2 Transmit */
|
|
{ SAM_PID_PWM0, XDMACH_PWM0_TX }, /* PWM0 Transmit */
|
|
{ SAM_PID_TWIHS0, XDMACH_TWIHS0_TX }, /* TWIHS0 Transmit */
|
|
{ SAM_PID_TWIHS1, XDMACH_TWIHS1_TX }, /* TWIHS1 Transmit */
|
|
{ SAM_PID_TWIHS2, XDMACH_TWIHS2_TX }, /* TWIHS2 Transmit */
|
|
{ SAM_PID_UART0, XDMACH_UART0_TX }, /* UART0 Transmit */
|
|
{ SAM_PID_UART1, XDMACH_UART1_TX }, /* UART1 Transmit */
|
|
{ SAM_PID_UART2, XDMACH_UART2_TX }, /* UART2 Transmit */
|
|
{ SAM_PID_UART3, XDMACH_UART3_TX }, /* UART3 Transmit */
|
|
{ SAM_PID_UART4, XDMACH_UART4_TX }, /* UART4 Transmit */
|
|
{ SAM_PID_DACC, XDMACH_DACC_TX }, /* DACC Transmit */
|
|
{ SAM_PID_SSC0, XDMACH_SSC_TX }, /* SSC Transmit */
|
|
{ SAM_PID_AES, XDMACH_AES_TX }, /* AES Transmit */
|
|
{ SAM_PID_PWM1, XDMACH_PWM1_TX } /* PWM01Transmit */
|
|
};
|
|
#define NXDMAC_TXCHANNELS (sizeof(g_xdmac_txchan) / sizeof(struct sam_pidmap_s))
|
|
|
|
/* This array describes the available link list descriptors */
|
|
|
|
struct chnext_view1_s g_lldesc[CONFIG_SAMV7_NLLDESC];
|
|
|
|
/* This array describes the state of each XDMAC channel 0 */
|
|
|
|
static struct sam_xdmach_s g_xdmach[SAMV7_NDMACHAN] =
|
|
{
|
|
#if SAMV7_NDMACHAN > 0
|
|
{
|
|
.chan = 0,
|
|
.base = SAM_XDMACH0_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 1
|
|
{
|
|
.chan = 1,
|
|
.base = SAM_XDMACH1_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 2
|
|
{
|
|
.chan = 2,
|
|
.base = SAM_XDMACH2_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 3
|
|
{
|
|
.chan = 3,
|
|
.base = SAM_XDMACH3_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 4
|
|
{
|
|
.chan = 4,
|
|
.base = SAM_XDMACH4_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 5
|
|
{
|
|
.chan = 5,
|
|
.base = SAM_XDMACH5_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 6
|
|
{
|
|
.chan = 6,
|
|
.base = SAM_XDMACH6_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 7
|
|
{
|
|
.chan = 7,
|
|
.base = SAM_XDMACH7_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 8
|
|
{
|
|
.chan = 8,
|
|
.base = SAM_XDMACH8_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 9
|
|
{
|
|
.chan = 9,
|
|
.base = SAM_XDMACH9_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 10
|
|
{
|
|
.chan = 10,
|
|
.base = SAM_XDMACH10_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 11
|
|
{
|
|
.chan = 11,
|
|
.base = SAM_XDMACH11_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 12
|
|
{
|
|
.chan = 12,
|
|
.base = SAM_XDMACH12_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 13
|
|
{
|
|
.chan = 13,
|
|
.base = SAM_XDMACH13_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 14
|
|
{
|
|
.chan = 14,
|
|
.base = SAM_XDMACH14_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 15
|
|
{
|
|
.chan = 15,
|
|
.base = SAM_XDMACH15_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 16
|
|
{
|
|
.chan = 16,
|
|
.base = SAM_XDMACH16_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 17
|
|
{
|
|
.chan = 17,
|
|
.base = SAM_XDMACH17_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 18
|
|
{
|
|
.chan = 18,
|
|
.base = SAM_XDMACH18_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 19
|
|
{
|
|
.chan = 19,
|
|
.base = SAM_XDMACH19_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 20
|
|
{
|
|
.chan = 20,
|
|
.base = SAM_XDMACH10_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 21
|
|
{
|
|
.chan = 21,
|
|
.base = SAM_XDMACH11_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 22
|
|
{
|
|
.chan = 22,
|
|
.base = SAM_XDMACH22_BASE,
|
|
},
|
|
#endif
|
|
#if SAMV7_NDMACHAN > 23
|
|
{
|
|
.chan = 23,
|
|
.base = SAM_XDMACH23_BASE,
|
|
}
|
|
#endif
|
|
};
|
|
|
|
/* This describes the overall state of DMA controller */
|
|
|
|
static struct sam_xdmac_s g_xdmac;
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: sam_takechsem() and sam_givechsem()
|
|
*
|
|
* Description:
|
|
* Used to get exclusive access to the DMA channel table
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void sam_takechsem(struct sam_xdmac_s *xdmac)
|
|
{
|
|
/* Take the semaphore (perhaps waiting) */
|
|
|
|
while (sem_wait(&xdmac->chsem) != 0)
|
|
{
|
|
/* The only case that an error should occur here is if the wait was
|
|
* awakened by a signal.
|
|
*/
|
|
|
|
ASSERT(errno == EINTR);
|
|
}
|
|
}
|
|
|
|
static inline void sam_givechsem(struct sam_xdmac_s *xdmac)
|
|
{
|
|
(void)sem_post(&xdmac->chsem);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_takedsem() and sam_givedsem()
|
|
*
|
|
* Description:
|
|
* Used to wait for availability of descriptors in the descriptor table.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void sam_takedsem(struct sam_xdmac_s *xdmac)
|
|
{
|
|
/* Take the semaphore (perhaps waiting) */
|
|
|
|
while (sem_wait(&xdmac->dsem) != 0)
|
|
{
|
|
/* The only case that an error should occur here is if the wait was
|
|
* awakened by a signal.
|
|
*/
|
|
|
|
ASSERT(errno == EINTR);
|
|
}
|
|
}
|
|
|
|
static inline void sam_givedsem(struct sam_xdmac_s *xdmac)
|
|
{
|
|
(void)sem_post(&xdmac->dsem);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_getdmac
|
|
*
|
|
* Description:
|
|
* Read a global XDMAC register
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline uint32_t sam_getdmac(struct sam_xdmac_s *xdmac,
|
|
unsigned int offset)
|
|
{
|
|
return getreg32(SAM_XDMAC_BASE + offset);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_putdmac
|
|
*
|
|
* Description:
|
|
* Write a value to a global XDMAC register
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline void sam_putdmac(struct sam_xdmac_s *xdmac, uint32_t value,
|
|
unsigned int offset)
|
|
{
|
|
putreg32(value, SAM_XDMAC_BASE + offset);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_getdmach
|
|
*
|
|
* Description:
|
|
* Read a XDMAC channel register
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline uint32_t sam_getdmach(struct sam_xdmach_s *xdmach,
|
|
unsigned int offset)
|
|
{
|
|
return getreg32(xdmach->base + offset);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_putdmach
|
|
*
|
|
* Description:
|
|
* Write a value to a XDMAC channel register
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline void sam_putdmach(struct sam_xdmach_s *xdmach, uint32_t value,
|
|
unsigned int offset)
|
|
{
|
|
putreg32(value, xdmach->base + offset);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_controller
|
|
*
|
|
* Description:
|
|
* Given a DMA channel instance, return a pointer to the parent DMA
|
|
* controller instance.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline struct sam_xdmac_s *sam_controller(struct sam_xdmach_s *xdmach)
|
|
{
|
|
/* This function is useful only on the SAMV7D4 where there are two
|
|
* XDMAC controllers.
|
|
*/
|
|
|
|
return &g_xdmac;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_physramaddr and sam_virtramaddr
|
|
*
|
|
* Description:
|
|
* Useful only if address mapping is required. If needed, these should
|
|
* problem be moved to file contain common address mapping logic
|
|
* (sam_memories)
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline uintptr_t sam_physramaddr(uintptr_t virtaddr)
|
|
{
|
|
#warning REVISIT -- Do DCTM address need to be remapped for DMA?
|
|
return virtaddr;
|
|
}
|
|
|
|
static inline uintptr_t sam_virtramaddr(uintptr_t physaddr)
|
|
{
|
|
#warning REVISIT -- Do DCTM address need to be remapped for DMA?
|
|
return physaddr;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_channel, sam_source_channel, and sam_sink_channel
|
|
*
|
|
* Description:
|
|
* Return the RX or TX channel associated with a PID. As a clarification:
|
|
*
|
|
* The source channel refers to the source of data for the DMA. This is,
|
|
* either (1) memory for a TX DMA or (2) a peripheral register for an RX
|
|
* DMA.
|
|
*
|
|
* The sink channel is the recipient of the DMA data. This is either
|
|
* (1) memory for an RX DMA, or (2) a peripheral register for a TX DMA.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static uint8_t sam_channel(uint8_t pid, const struct sam_pidmap_s *table,
|
|
unsigned int nentries)
|
|
{
|
|
int i;
|
|
|
|
/* Search until either the entry with the matching PID is found or until
|
|
* all of the table entries have been examined without finding the PID.
|
|
*/
|
|
|
|
for (i = 0; i < nentries; i++, table++)
|
|
{
|
|
if (table->pid == pid)
|
|
{
|
|
return table->pchan;
|
|
}
|
|
}
|
|
|
|
dmadbg("No channel found for pid %d\n", pid);
|
|
DEBUGPANIC();
|
|
return 0x3f;
|
|
}
|
|
|
|
static uint32_t sam_source_channel(struct sam_xdmach_s *xdmach, uint8_t pid)
|
|
{
|
|
return (uint32_t)sam_channel(pid, g_xdmac_rxchan, NXDMAC_RXCHANNELS);
|
|
}
|
|
|
|
static uint32_t sam_sink_channel(struct sam_xdmach_s *xdmach, uint8_t pid)
|
|
{
|
|
return (uint32_t)sam_channel(pid, g_xdmac_txchan, NXDMAC_TXCHANNELS);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_maxtransfer
|
|
*
|
|
* Description:
|
|
* Maximum number of bytes that can be sent in one transfer
|
|
*
|
|
****************************************************************************/
|
|
|
|
static size_t sam_maxtransfer(struct sam_xdmach_s *xdmach)
|
|
{
|
|
unsigned int xfrwidth;
|
|
size_t maxtransfer;
|
|
|
|
/* Get the maximum transfer size in bytes */
|
|
|
|
xfrwidth = (xdmach->flags & DMACH_FLAG_PERIPHWIDTH_MASK) >>
|
|
DMACH_FLAG_PERIPHWIDTH_SHIFT;
|
|
|
|
switch (xfrwidth)
|
|
{
|
|
default:
|
|
case 0: /* 8 bits, 1 byte */
|
|
maxtransfer = XDMACH_CUBC_UBLEN_MAX;
|
|
break;
|
|
|
|
case 1: /* 16 bits, 2 bytes */
|
|
maxtransfer = 2 * XDMACH_CUBC_UBLEN_MAX;
|
|
break;
|
|
|
|
case 2: /* 32 bits 4 bytes */
|
|
maxtransfer = 4 * XDMACH_CUBC_UBLEN_MAX;
|
|
break;
|
|
|
|
case 3: /* 64 bits, 8 bytes */
|
|
maxtransfer = 8 * XDMACH_CUBC_UBLEN_MAX;
|
|
break;
|
|
}
|
|
|
|
return maxtransfer;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_ntxtransfers
|
|
*
|
|
* Description:
|
|
* Number of TX transfers via DMA
|
|
*
|
|
****************************************************************************/
|
|
|
|
static uint32_t sam_ntxtransfers(struct sam_xdmach_s *xdmach,
|
|
uint32_t dmasize)
|
|
{
|
|
unsigned int srcwidth;
|
|
|
|
/* Adjust the source transfer size for the source chunk size (memory
|
|
* chunk size). BTSIZE is "the number of transfers to be performed, that
|
|
* is, for writes it refers to the number of source width transfers
|
|
* to perform when XDMAC is flow controller. For Reads, BTSIZE refers to
|
|
* the number of transfers completed on the Source Interface. ..."
|
|
*/
|
|
|
|
srcwidth = (xdmach->flags & DMACH_FLAG_PERIPHWIDTH_MASK) >>
|
|
DMACH_FLAG_PERIPHWIDTH_SHIFT;
|
|
|
|
switch (srcwidth)
|
|
{
|
|
default:
|
|
case 0: /* 8 bits, 1 byte */
|
|
break;
|
|
|
|
case 1: /* 16 bits, 2 bytes */
|
|
dmasize = (dmasize + 1) >> 1;
|
|
break;
|
|
|
|
case 2: /* 32 bits, 4 bytes */
|
|
dmasize = (dmasize + 3) >> 2;
|
|
break;
|
|
|
|
case 3: /* 64 bits, 8 bytes */
|
|
dmasize = (dmasize + 7) >> 3;
|
|
break;
|
|
}
|
|
|
|
return dmasize;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_cubc
|
|
*
|
|
* Description:
|
|
* Calculate the CUBC transfer size
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline uint32_t sam_cubc(struct sam_xdmach_s *xdmach,
|
|
uint32_t dmasize)
|
|
{
|
|
uint32_t ntransfers;
|
|
|
|
/* Set the buffer transfer size field. This is the number of transfers to
|
|
* be performed, that is, the number of source width transfers to perform.
|
|
*/
|
|
|
|
ntransfers = sam_ntxtransfers(xdmach, dmasize);
|
|
|
|
DEBUGASSERT(ntransfers <= XDMACH_CUBC_UBLEN_MAX);
|
|
return (ntransfers << XDMACH_CUBC_UBLEN_SHIFT);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_txcc
|
|
*
|
|
* Description:
|
|
* Decode the flags to get the correct Channel Control (CC) Register bit
|
|
* settings for a transmit (memory to peripheral) transfer.
|
|
*
|
|
* Single Block:
|
|
* 1. Clear TYPE for a memory to memory transfer, otherwise set
|
|
* this bit for memory to/from peripheral transfer.
|
|
* 2. Program MBSIZE to the memory burst size used.
|
|
* 3. Program SAM/DAM to the memory addressing scheme.
|
|
* 4. Program SYNC to select the peripheral transfer direction.
|
|
* 5. Set PROT to activate a secure channel.
|
|
* 6. Program CSIZE to configure the channel chunk size (only
|
|
* relevant for peripheral synchronized transfer).
|
|
* 7. Program DWIDTH to configure the transfer data width.
|
|
* 8. Program SIF, DIF to configure the master interface
|
|
* used to read data and write data respectively.
|
|
* 9. Program PERID to select the active hardware request line
|
|
* (only relevant for a peripheral synchronized transfer).
|
|
* 10. Set SWREQ to use software request (only relevant for a
|
|
* peripheral synchronized transfer).
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline uint32_t sam_txcc(struct sam_xdmach_s *xdmach)
|
|
{
|
|
uint32_t regval = 0;
|
|
uint32_t field;
|
|
|
|
/* 1. Clear TYPE for a memory to memory transfer, otherwise set
|
|
* this bit for memory to/from peripheral transfer.
|
|
*
|
|
* By convention, TX refers to a memory to peripheral transfer. So the
|
|
* source is memory. Is the "peripheral" destination a peripheral? or
|
|
* it is really memory?
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHISPERIPH) != 0)
|
|
{
|
|
/* Yes.. Use peripheral synchronized mode */
|
|
|
|
regval |= XDMACH_CC_TYPE;
|
|
}
|
|
|
|
/* 2. Program MBSIZE to the memory burst size used.
|
|
*
|
|
* NOTE: This assumes the same encoding in the DMACH flags as in the CC
|
|
* register MBSIZE field.
|
|
*/
|
|
|
|
field = (xdmach->flags & DMACH_FLAG_MEMBURST_MASK) >>
|
|
DMACH_FLAG_MEMBURST_SHIFT;
|
|
regval |= (field << XDMACH_CC_MBSIZE_SHIFT);
|
|
|
|
/* 3. Program SAM/DAM to the memory addressing scheme.
|
|
*
|
|
* NOTE: This assumes that 0 means non-incrementing.
|
|
* TX -> Source is memory.
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_MEMINCREMENT) != 0)
|
|
{
|
|
regval |= XDMACH_CC_SAM_INCR;
|
|
}
|
|
|
|
/* TX -> Destination is peripheral. */
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHINCREMENT) != 0)
|
|
{
|
|
regval |= XDMACH_CC_DAM_INCR;
|
|
}
|
|
|
|
/* 4. Program DSYNC to select the peripheral transfer direction.
|
|
*
|
|
* TX -> Memory to peripheral
|
|
*/
|
|
|
|
regval |= XDMACH_CC_DSYNC;
|
|
|
|
#if 0 /* REVISIT */
|
|
/* 5. Set PROT to activate a secure channel (REVISIT). */
|
|
|
|
regval |= XDMACH_CC_PROT; /* Channel is unsecured */
|
|
#endif
|
|
|
|
/* 6. Program CSIZE to configure the channel chunk size (only
|
|
* relevant for peripheral synchronized transfer).
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHISPERIPH) != 0)
|
|
{
|
|
/* Peripheral synchronized mode -- set chunk size.
|
|
* NOTE that we assume that encoding in the XDMACH flags is the same
|
|
* as in the CC register CSIZE field.
|
|
*/
|
|
|
|
field = (xdmach->flags & DMACH_FLAG_PERIPHCHUNKSIZE_MASK) >>
|
|
DMACH_FLAG_PERIPHCHUNKSIZE_SHIFT;
|
|
regval |= (field << XDMACH_CC_CSIZE_SHIFT);
|
|
}
|
|
|
|
/* 7. Program DWIDTH to configure the transfer data width.
|
|
*
|
|
* NOTE that we assume that encoding in the XDMACH flags is the same as in
|
|
* the CC register CSIZE field.
|
|
*/
|
|
|
|
field = (xdmach->flags & DMACH_FLAG_PERIPHWIDTH_MASK) >>
|
|
DMACH_FLAG_PERIPHWIDTH_SHIFT;
|
|
regval |= (field << XDMACH_CC_DWIDTH_SHIFT);
|
|
|
|
/* 8. Program SIF, DIF to configure the master interface
|
|
* used to read data and write data respectively.
|
|
*
|
|
* TX -> Source is memory
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_MEMAHB_MASK) == DMACH_FLAG_MEMAHB_AHB_IF1)
|
|
{
|
|
regval |= XDMACH_CC_SIF;
|
|
}
|
|
|
|
/* TX -> Destination is peripheral */
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHAHB_MASK) == DMACH_FLAG_PERIPHAHB_AHB_IF1)
|
|
{
|
|
regval |= XDMACH_CC_DIF;
|
|
}
|
|
|
|
/* 9. Program PERID to select the active hardware request line
|
|
* (only relevant for a peripheral synchronized transfer).
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHISPERIPH) != 0)
|
|
{
|
|
int pid;
|
|
|
|
/* Get the PID from the DMACH flags */
|
|
|
|
pid = (xdmach->flags & DMACH_FLAG_PERIPHPID_MASK) >>
|
|
DMACH_FLAG_PERIPHPID_SHIFT;
|
|
|
|
/* Look up the DMA channel code for TX: Peripheral is the sink. */
|
|
|
|
field = sam_sink_channel(xdmach, pid);
|
|
regval |= (field << XDMACH_CC_CSIZE_SHIFT);
|
|
|
|
#if 0 /* Not supported */
|
|
/* 10. Set SWREQ to use software request (only relevant for a
|
|
* peripheral synchronized transfer).
|
|
*/
|
|
|
|
regval |= XDMACH_CC_SWREQ;
|
|
#endif
|
|
}
|
|
|
|
return regval;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_rxcc
|
|
*
|
|
* Description:
|
|
* Decode the flags to get the correct Channel Control (CC) Register bit
|
|
* settings for a receive (peripheral to memory) transfer.
|
|
*
|
|
* 1. Clear TYPE for a memory to memory transfer, otherwise set
|
|
* this bit for memory to/from peripheral transfer.
|
|
* 2. Program MBSIZE to the memory burst size used.
|
|
* 3. Program SAM/DAM to the memory addressing scheme.
|
|
* 4. Program SYNC to select the peripheral transfer direction.
|
|
* 5. Set PROT to activate a secure channel.
|
|
* 6. Program CSIZE to configure the channel chunk size (only
|
|
* relevant for peripheral synchronized transfer).
|
|
* 7. Program DWIDTH to configure the transfer data width.
|
|
* 8. Program SIF, DIF to configure the master interface
|
|
* used to read data and write data respectively.
|
|
* 9. Program PERID to select the active hardware request line
|
|
* (only relevant for a peripheral synchronized transfer).
|
|
* 10. Set SWREQ to use software request (only relevant for a
|
|
* peripheral synchronized transfer).
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline uint32_t sam_rxcc(struct sam_xdmach_s *xdmach)
|
|
{
|
|
uint32_t regval = 0;
|
|
uint32_t field;
|
|
|
|
/* 1. Clear TYPE for a memory to memory transfer, otherwise set
|
|
* this bit for memory to/from peripheral transfer.
|
|
*
|
|
* By convention, RX refers to a peripheral to memory transfer. So the
|
|
* source is peripheral. Is the "peripheral" source a peripheral? or
|
|
* is it also memory?
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHISPERIPH) != 0)
|
|
{
|
|
/* Yes.. Use peripheral synchronized mode */
|
|
|
|
regval |= XDMACH_CC_TYPE;
|
|
}
|
|
|
|
/* 2. Program MBSIZE to the memory burst size used.
|
|
*
|
|
* NOTE: This assumes the same encoding in the DMACH flags as in the CC
|
|
* register MBSIZE field.
|
|
*/
|
|
|
|
field = (xdmach->flags & DMACH_FLAG_MEMBURST_MASK) >>
|
|
DMACH_FLAG_MEMBURST_SHIFT;
|
|
regval |= (field << XDMACH_CC_MBSIZE_SHIFT);
|
|
|
|
/* 3. Program SAM/DAM to the memory addressing scheme.
|
|
*
|
|
* NOTE: This assumes that 0 means non-incrementing.
|
|
* RX -> Source is peripheral.
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHINCREMENT) != 0)
|
|
{
|
|
regval |= XDMACH_CC_SAM_INCR;
|
|
}
|
|
|
|
/* RX -> Destination is memory. */
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_MEMINCREMENT) != 0)
|
|
{
|
|
regval |= XDMACH_CC_DAM_INCR;
|
|
}
|
|
|
|
/* 4. Program DSYNC to select the peripheral transfer direction.
|
|
*
|
|
* RX -> Peripheral to memory (DSYNC == 0)
|
|
*/
|
|
|
|
#if 0 /* REVISIT */
|
|
/* 5. Set PROT to activate a secure channel (REVISIT) */
|
|
|
|
regval |= XDMACH_CC_PROT; /* Channel is unsecured */
|
|
#endif
|
|
|
|
/* 6. Program CSIZE to configure the channel chunk size (only
|
|
* relevant for peripheral synchronized transfer).
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHISPERIPH) != 0)
|
|
{
|
|
/* Peripheral synchronized mode -- set chunk size.
|
|
* NOTE that we assume that encoding in the XDMACH flags is the same
|
|
* as in the CC register CSIZE field.
|
|
*/
|
|
|
|
field = (xdmach->flags & DMACH_FLAG_PERIPHCHUNKSIZE_MASK) >>
|
|
DMACH_FLAG_PERIPHCHUNKSIZE_SHIFT;
|
|
regval |= (field << XDMACH_CC_CSIZE_SHIFT);
|
|
}
|
|
|
|
/* 7. Program DWIDTH to configure the transfer data width.
|
|
*
|
|
* NOTE that we assume that encoding in the XDMACH flags is the same as in
|
|
* the CC register CSIZE field.
|
|
*/
|
|
|
|
field = (xdmach->flags & DMACH_FLAG_PERIPHWIDTH_MASK) >>
|
|
DMACH_FLAG_PERIPHWIDTH_SHIFT;
|
|
regval |= (field << XDMACH_CC_DWIDTH_SHIFT);
|
|
|
|
/* 8. Program SIF, DIF to configure the master interface
|
|
* used to read data and write data respectively.
|
|
*
|
|
* RX -> Source is peripheral
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHAHB_MASK) == DMACH_FLAG_PERIPHAHB_AHB_IF1)
|
|
{
|
|
regval |= XDMACH_CC_SIF;
|
|
}
|
|
|
|
/* RX -> Destination is memory */
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_MEMAHB_MASK) == DMACH_FLAG_MEMAHB_AHB_IF1)
|
|
{
|
|
regval |= XDMACH_CC_DIF;
|
|
}
|
|
|
|
/* 9. Program PERID to select the active hardware request line
|
|
* (only relevant for a peripheral synchronized transfer).
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHISPERIPH) != 0)
|
|
{
|
|
int pid;
|
|
|
|
/* Get the PID from the DMACH flags */
|
|
|
|
pid = (xdmach->flags & DMACH_FLAG_PERIPHPID_MASK) >>
|
|
DMACH_FLAG_PERIPHPID_SHIFT;
|
|
|
|
/* Look up the DMA channel code for RX: Peripheral is the source. */
|
|
|
|
field = sam_source_channel(xdmach, pid);
|
|
regval |= (field << XDMACH_CC_CSIZE_SHIFT);
|
|
|
|
#if 0 /* Not supported */
|
|
/* 10. Set SWREQ to use software request (only relevant for a
|
|
* peripheral synchronized transfer).
|
|
*/
|
|
|
|
regval |= XDMACH_CC_SWREQ;
|
|
#endif
|
|
}
|
|
|
|
return regval;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_allocdesc
|
|
*
|
|
* Description:
|
|
* Allocate and add one descriptor to the DMA channel's link list.
|
|
*
|
|
* NOTE: link list entries are freed by the DMA interrupt handler.
|
|
* However, since the setting/clearing of the 'in use' indication is
|
|
* atomic, no special actions need be performed. It would be a good thing
|
|
* to add logic to handle the case where all of the entries are exhausted
|
|
* and we could wait for some to be freed by the interrupt handler.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static struct chnext_view1_s *
|
|
sam_allocdesc(struct sam_xdmach_s *xdmach, struct chnext_view1_s *prev,
|
|
uint32_t csa, uint32_t cda, uint32_t cubc)
|
|
{
|
|
struct sam_xdmac_s *xdmac = sam_controller(xdmach);
|
|
struct chnext_view1_s *descr = NULL;
|
|
int i;
|
|
|
|
/* Sanity check -- csa == 0 is the indication that the link is unused.
|
|
* Obviously setting it to zero would break that usage.
|
|
*/
|
|
|
|
#ifdef CONFIG_DEBUG
|
|
if (csa != 0)
|
|
#endif
|
|
{
|
|
/* Table a descriptor table semaphore count. When we get one, then there
|
|
* is at least one free descriptor in the table and it is ours.
|
|
*/
|
|
|
|
sam_takedsem(xdmac);
|
|
|
|
/* Examine each link list entry to find an available one -- i.e., one
|
|
* with csa == 0. That csa field is set to zero by the DMA transfer
|
|
* complete interrupt handler. The following should be safe because
|
|
* that is an atomic operation.
|
|
*/
|
|
|
|
sam_takechsem(xdmac);
|
|
for (i = 0; i < CONFIG_SAMV7_NLLDESC; i++)
|
|
{
|
|
if (g_lldesc[i].csa == 0)
|
|
{
|
|
/* We have it. Initialize the new link list entry */
|
|
|
|
descr = &g_lldesc[i];
|
|
descr->cnda = 0; /* Next Descriptor Address */
|
|
descr->cubc = cubc; /* Channel Microblock Control Register */
|
|
descr->csa = csa; /* Source address */
|
|
descr->cda = cda; /* Destination address */
|
|
|
|
/* And then hook it at the tail of the link list */
|
|
|
|
if (!prev)
|
|
{
|
|
/* There is no previous link. This is the new head of
|
|
* the list
|
|
*/
|
|
|
|
DEBUGASSERT(xdmach->llhead == NULL && xdmach->lltail == NULL);
|
|
xdmach->llhead = descr;
|
|
}
|
|
else
|
|
{
|
|
DEBUGASSERT(xdmach->llhead != NULL && xdmach->lltail == prev);
|
|
|
|
/* When the second link is added to the list, that is the
|
|
* cue that we are going to do the link list transfer.
|
|
*
|
|
* Set the NDE field in the previous descriptor; Clear the
|
|
* NDE field in the final descriptor.
|
|
*/
|
|
|
|
prev->cubc |= CHNEXT_UBC_NDE;
|
|
|
|
/* Link the previous tail to the new tail.
|
|
* REVISIT: This assumes that the next description is fetched
|
|
* via AHB IF0.
|
|
*/
|
|
|
|
prev->cnda = (uint32_t)sam_physramaddr((uintptr_t)descr);
|
|
}
|
|
|
|
/* In any event, this is the new tail of the list */
|
|
|
|
xdmach->lltail = descr;
|
|
|
|
/* Assume that we will be doing multiple buffer transfers and that
|
|
* that hardware will be accessing the descriptor via DMA.
|
|
*/
|
|
|
|
arch_clean_dcache((uintptr_t)descr,
|
|
(uintptr_t)descr + sizeof(struct chnext_view1_s));
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Because we hold a count from the counting semaphore, the above
|
|
* search loop should always be successful.
|
|
*/
|
|
|
|
sam_givechsem(xdmac);
|
|
DEBUGASSERT(descr != NULL);
|
|
}
|
|
|
|
return descr;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_freelinklist
|
|
*
|
|
* Description:
|
|
* Free all descriptors in the DMA channel's link list.
|
|
*
|
|
* NOTE: Called from the DMA interrupt handler.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void sam_freelinklist(struct sam_xdmach_s *xdmach)
|
|
{
|
|
struct sam_xdmac_s *xdmac = sam_controller(xdmach);
|
|
struct chnext_view1_s *descr;
|
|
uintptr_t paddr;
|
|
|
|
/* Get the head of the link list and detach the link list from the DMA
|
|
* channel
|
|
*/
|
|
|
|
descr = xdmach->llhead;
|
|
xdmach->llhead = NULL;
|
|
xdmach->lltail = NULL;
|
|
|
|
while (descr != NULL)
|
|
{
|
|
/* Valid, in-use descriptors never have csa == 0 */
|
|
|
|
DEBUGASSERT(descr->csa != 0);
|
|
|
|
/* Get the physical address of the next descriptor in the list */
|
|
|
|
paddr = descr->cnda;
|
|
|
|
/* Free the descriptor by simply nullifying it and bumping up the
|
|
* semaphore count.
|
|
*/
|
|
|
|
memset(descr, 0, sizeof(struct chnext_view1_s));
|
|
sam_givedsem(xdmac);
|
|
|
|
/* Get the virtual address of the next descriptor in the list */
|
|
|
|
descr = (struct chnext_view1_s *)sam_virtramaddr(paddr);
|
|
}
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_txbuffer
|
|
*
|
|
* Description:
|
|
* Configure DMA for transmit of one buffer (memory to peripheral). This
|
|
* function may be called multiple times to handle large and/or dis-
|
|
* continuous transfers.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static int sam_txbuffer(struct sam_xdmach_s *xdmach, uint32_t paddr,
|
|
uint32_t maddr, size_t nbytes)
|
|
{
|
|
uint32_t cubc;
|
|
|
|
/* If we are appending a buffer to a linklist, then re-use the previously
|
|
* calculated CC register value. Otherwise, create the CC register value
|
|
* from the properties of the transfer.
|
|
*/
|
|
|
|
if (!xdmach->llhead)
|
|
{
|
|
xdmach->cc = sam_txcc(xdmach);
|
|
}
|
|
|
|
/* Calculate the number of transfers for CUBC */
|
|
|
|
cubc = sam_cubc(xdmach, nbytes);
|
|
cubc |= (CHNEXT_UBC_NVIEW_1 | CHNEXT_UBC_NSEN);
|
|
|
|
/* Add the new link list entry */
|
|
|
|
if (!sam_allocdesc(xdmach, xdmach->lltail, maddr, paddr, cubc))
|
|
{
|
|
return -ENOMEM;
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_rxbuffer
|
|
*
|
|
* Description:
|
|
* Configure DMA for receipt of one buffer (peripheral to memory). This
|
|
* function may be called multiple times to handle large and/or dis-
|
|
* continuous transfers.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static int sam_rxbuffer(struct sam_xdmach_s *xdmach, uint32_t paddr,
|
|
uint32_t maddr, size_t nbytes)
|
|
{
|
|
uint32_t cubc;
|
|
|
|
/* If we are appending a buffer to a linklist, then re-use the previously
|
|
* calculated CC register value. Otherwise, create the CC register value
|
|
* from the properties of the transfer.
|
|
*/
|
|
|
|
if (!xdmach->llhead)
|
|
{
|
|
xdmach->cc = sam_rxcc(xdmach);
|
|
}
|
|
|
|
/* Calculate the number of transfers for CUBC */
|
|
|
|
cubc = sam_cubc(xdmach, nbytes);
|
|
cubc |= (CHNEXT_UBC_NVIEW_1 | CHNEXT_UBC_NSEN);
|
|
|
|
/* Add the new link list entry */
|
|
|
|
if (!sam_allocdesc(xdmach, xdmach->lltail, paddr, maddr, cubc))
|
|
{
|
|
return -ENOMEM;
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_single
|
|
*
|
|
* Description:
|
|
* Start a single buffer DMA.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline int sam_single(struct sam_xdmach_s *xdmach)
|
|
{
|
|
struct sam_xdmac_s *xdmac = sam_controller(xdmach);
|
|
struct chnext_view1_s *llhead = xdmach->llhead;
|
|
|
|
/* 1. Read the XDMAC Global Channel Status Register (XDMAC_GS) to choose a
|
|
* free channel.
|
|
*
|
|
* In this implementation, the free channel is assigned in a different
|
|
* manner.
|
|
*/
|
|
|
|
/* 2. Clear any pending interrupts from any previous XDMAC transfer by
|
|
* reading the XDMAC Channel Interrupt Status Register (CIS).
|
|
*/
|
|
|
|
(void)sam_getdmach(xdmach, SAM_XDMACH_CIS_OFFSET);
|
|
|
|
/* 3. Write the starting source address in the Channel Source Address (CSA)
|
|
* Register.
|
|
*/
|
|
|
|
DEBUGASSERT(llhead != NULL && llhead->csa != 0);
|
|
sam_putdmach(xdmach, llhead->csa, SAM_XDMACH_CSA_OFFSET);
|
|
|
|
/* 4. Write the starting destination address in the Channel Destination
|
|
* Address Register (CDA).
|
|
*/
|
|
|
|
sam_putdmach(xdmach, llhead->cda, SAM_XDMACH_CDA_OFFSET);
|
|
|
|
/* 5. Program field UBLEN in the XDMAC Channel Microblock Control (CUBC)
|
|
* Register with the number of data.
|
|
*/
|
|
|
|
sam_putdmach(xdmach, llhead->cubc, SAM_XDMACH_CUBC_OFFSET);
|
|
|
|
/* 6. Program the Channel Control (CC) Register */
|
|
|
|
sam_putdmach(xdmach, xdmach->cc, SAM_XDMACH_CC_OFFSET);
|
|
|
|
/* 7. Clear the following five registers:
|
|
*
|
|
* XDMAC Channel Next Descriptor Control (CNDC) Register
|
|
* XDMAC Channel Block Control (CBC) Register
|
|
* XDMAC Channel Data Stride Memory Set Pattern (CDSMSP) Register
|
|
* XDMAC Channel Source Microblock Stride (CSUS) Register
|
|
* XDMAC Channel Destination Microblock Stride (CDUS)Register
|
|
*
|
|
* This respectively indicates that the linked list is disabled, there is
|
|
* only one block and striding is disabled
|
|
*/
|
|
|
|
sam_putdmach(xdmach, 0, SAM_XDMACH_CNDC_OFFSET);
|
|
sam_putdmach(xdmach, 0, SAM_XDMACH_CBC_OFFSET);
|
|
sam_putdmach(xdmach, 0, SAM_XDMACH_CDSMSP_OFFSET);
|
|
sam_putdmach(xdmach, 0, SAM_XDMACH_CSUS_OFFSET);
|
|
sam_putdmach(xdmach, 0, SAM_XDMACH_CDUS_OFFSET);
|
|
|
|
/* 8. Enable the Microblock interrupt by setting the "End of Block"
|
|
* interrupt bit in the XDMAC Channel Interrupt Enable (CIE) Register.
|
|
*/
|
|
|
|
sam_putdmach(xdmach, XDMAC_CHINT_BI | XDMAC_CHINT_ERRORS,
|
|
SAM_XDMACH_CIE_OFFSET);
|
|
|
|
/* Enable the Channel Interrupt Enable bit by setting the corresponding
|
|
* bit in the XDMAC Global Interrupt Enable (GIE) Register.
|
|
*/
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN(xdmach->chan), SAM_XDMAC_GIE_OFFSET);
|
|
|
|
/* 9. Enable the channel by setting the corresponding bit in the
|
|
* XDMAC Global Channel Enable (GE) Register. The channel bit will
|
|
* be set in the GS register by hardware.
|
|
*/
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN(xdmach->chan), SAM_XDMAC_GE_OFFSET);
|
|
|
|
/* 10. The DMA has been started. Once completed, the DMA channel sets the
|
|
* corresponding "End of Block Interrupt" Status bit in the channel
|
|
* CIS register and generates an global interrupt for the channel.
|
|
* The channel bit will be cleared in the GS register by hardware.
|
|
*/
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_multiple
|
|
*
|
|
* Description:
|
|
* Start a multiple buffer DMA.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline int sam_multiple(struct sam_xdmach_s *xdmach)
|
|
{
|
|
struct sam_xdmac_s *xdmac = sam_controller(xdmach);
|
|
#ifdef CONFIG_DEBUG
|
|
struct chnext_view1_s *llhead = xdmach->llhead;
|
|
#endif
|
|
uintptr_t paddr;
|
|
uint32_t regval;
|
|
|
|
DEBUGASSERT(llhead != NULL && llhead->csa != 0);
|
|
|
|
/* 1. Read the XDMAC Global Channel Status Register (XDMAC_GS) to choose a
|
|
* free channel.
|
|
*
|
|
* In this implementation, the free channel is assigned in a different
|
|
* manner.
|
|
*/
|
|
|
|
/* 2. Clear any pending interrupts from any previous XDMAC transfer by
|
|
* reading the XDMAC Channel Interrupt Status Register (CIS).
|
|
*/
|
|
|
|
(void)sam_getdmach(xdmach, SAM_XDMACH_CIS_OFFSET);
|
|
|
|
/* 3. Build a linked list of transfer descriptors in memory. The
|
|
* descriptor view is programmable on a per descriptor basis. The
|
|
* linked list items structure must be word aligned. CUBC NDE
|
|
* must be configured to 0 in the last descriptor to terminate the
|
|
* list.
|
|
*
|
|
* This was done during the RX/TX setup phases of the DMA transfer.
|
|
*/
|
|
|
|
/* Since we are using view1, I imagine that we need to set the Channel
|
|
* Control (CC) Register.
|
|
*/
|
|
|
|
sam_putdmach(xdmach, xdmach->cc, SAM_XDMACH_CC_OFFSET);
|
|
|
|
/* 4. Program field NDA in the XDMAC Channel Next Descriptor Address
|
|
* (CNDA) Register with the first descriptor address and bit NDAIF
|
|
* with the master interface identifier.
|
|
*
|
|
* REVIST: Using NDAIF=0. Is that correct?
|
|
*/
|
|
|
|
paddr = sam_physramaddr((uintptr_t)xdmach->llhead);
|
|
sam_putdmach(xdmach, (uint32_t)paddr, SAM_XDMACH_CNDA_OFFSET);
|
|
|
|
/* 5. Program the CNDC register:
|
|
*
|
|
* a. Set NDE to enable the descriptor fetch.
|
|
* b. Set NDSUP to update the source address at the descriptor fetch
|
|
* time, otherwise clear this bit.
|
|
* c. Set NDDUP to update the destination address at the descriptor
|
|
* fetch time, otherwise clear this bit.
|
|
* d. Program the NDVIEW field to define the length of the first
|
|
* descriptor.
|
|
*/
|
|
|
|
regval = (XDMACH_CNDC_NDE | XDMACH_CNDC_NDVIEW_NDV2);
|
|
|
|
/* Update the source address if this is a memory-to-* transfer.
|
|
*
|
|
* TYPE = 0 -> memory-to-memory
|
|
* TYPE = 1 && DSYNC = 1 -> memory-to-peripheral
|
|
*/
|
|
|
|
if ((xdmach->cc & XDMACH_CC_TYPE) == 0 ||
|
|
(xdmach->cc & XDMACH_CC_DSYNC) != 0)
|
|
{
|
|
regval |= XDMACH_CNDC_NDSUP;
|
|
}
|
|
|
|
/* Update the destination address if this is a *-to-memory transfer.
|
|
*
|
|
* TYPE = 0 -> memory-to-memory
|
|
* TYPE = 1 && DSYNC = 0 -> peripheral-to-memory
|
|
*/
|
|
|
|
if ((xdmach->cc & XDMACH_CC_TYPE) == 0 ||
|
|
(xdmach->cc & XDMACH_CC_TYPE) == 0)
|
|
{
|
|
regval |= XDMACH_CNDC_NDDUP;
|
|
}
|
|
|
|
sam_putdmach(xdmach, regval, SAM_XDMACH_CNDC_OFFSET);
|
|
|
|
/* 6. Enable the End of Linked List interrupt by setting the LI bit in the
|
|
* CIE register.
|
|
*/
|
|
|
|
sam_putdmach(xdmach, XDMAC_CHINT_LI | XDMAC_CHINT_ERRORS,
|
|
SAM_XDMACH_CIE_OFFSET);
|
|
|
|
/* Enable the Channel Interrupt Enable bit by setting the corresponding
|
|
* bit in the XDMAC Global Interrupt Enable (GIE) Register.
|
|
*/
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN(xdmach->chan), SAM_XDMAC_GIE_OFFSET);
|
|
|
|
/* 7. Enable the channel by setting the corresponding bit in the
|
|
* XDMAC Global Channel Enable (GE) Register. The channel bit will
|
|
* be set in the GS register by hardware.
|
|
*/
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN(xdmach->chan), SAM_XDMAC_GE_OFFSET);
|
|
|
|
/* 8. The DMA has been started. Once completed, the DMA channel sets the
|
|
* corresponding "End of Block Interrupt" Status bit in the channel
|
|
* CIS register and generates an global interrupt for the channel.
|
|
* The channel bit will be cleared in the GS register by hardware.
|
|
*/
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmaterminate
|
|
*
|
|
* Description:
|
|
* Terminate the DMA transfer and disable the DMA channel
|
|
*
|
|
****************************************************************************/
|
|
|
|
static void sam_dmaterminate(struct sam_xdmach_s *xdmach, int result)
|
|
{
|
|
struct sam_xdmac_s *xdmac = sam_controller(xdmach);
|
|
uint32_t chanbit = XDMAC_CHAN(xdmach->chan);
|
|
|
|
/* Disable all channel interrupts */
|
|
|
|
sam_putdmac(xdmac, chanbit, SAM_XDMAC_GID_OFFSET);
|
|
sam_putdmach(xdmach, XDMAC_CHINT_ALL, SAM_XDMACH_CID_OFFSET);
|
|
|
|
/* Under normal operation, the software enables a channel by setting the
|
|
* associated bit in the Global Channel Enable (GE) Register. The hardware
|
|
* then disables a channel on transfer completion by clearing channel bit
|
|
* in the GS register. To disable a channel, setting the channel bit in the
|
|
* GD register and poll the channel bit in GS register to determine when
|
|
* the channel has been disabled.
|
|
*/
|
|
|
|
sam_putdmac(xdmac, chanbit, SAM_XDMAC_GD_OFFSET);
|
|
while ((sam_getdmac(xdmac, SAM_XDMAC_GS_OFFSET) & chanbit) != 0);
|
|
|
|
/* Free the linklist */
|
|
|
|
sam_freelinklist(xdmach);
|
|
|
|
/* Perform the DMA complete callback */
|
|
|
|
if (xdmach->callback)
|
|
{
|
|
xdmach->callback((DMA_HANDLE)xdmach, xdmach->arg, result);
|
|
}
|
|
|
|
xdmach->callback = NULL;
|
|
xdmach->arg = NULL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_xdmac_interrupt
|
|
*
|
|
* Description:
|
|
* DMA interrupt handler
|
|
*
|
|
****************************************************************************/
|
|
|
|
static int sam_xdmac_interrupt(int irq, void *context)
|
|
{
|
|
struct sam_xdmac_s *xdmac = &g_xdmac;
|
|
struct sam_xdmach_s *xdmach;
|
|
unsigned int chndx;
|
|
uint32_t gpending;
|
|
uint32_t chpending;
|
|
uint32_t bit;
|
|
|
|
/* Get the set of pending, unmasked global XDMAC interrupts */
|
|
|
|
gpending = sam_getdmac(xdmac, SAM_XDMAC_GIS_OFFSET) &
|
|
sam_getdmac(xdmac, SAM_XDMAC_GIM_OFFSET);
|
|
|
|
/* Yes.. Check each bit to see which channel(s) have interrupted */
|
|
|
|
for (chndx = 0; chndx < SAMV7_NDMACHAN && gpending != 0; chndx++)
|
|
{
|
|
/* Are any interrupts pending for this channel? */
|
|
|
|
bit = XDMAC_CHAN(chndx);
|
|
if ((gpending & bit) != 0)
|
|
{
|
|
xdmach = &g_xdmach[chndx];
|
|
|
|
/* Get the set of interrupts pending for this channel */
|
|
|
|
chpending = sam_getdmach(xdmach, SAM_XDMACH_CIS_OFFSET) &
|
|
sam_getdmach(xdmach, SAM_XDMACH_CIM_OFFSET);
|
|
|
|
/* Yes.. Did an error occur? */
|
|
|
|
if ((chpending & XDMAC_CHINT_ERRORS) != 0)
|
|
{
|
|
/* Yes... Terminate the transfer with an error? */
|
|
|
|
dmalldbg("ERROR: DMA failed: %08x\n", chpending);
|
|
sam_dmaterminate(xdmach, -EIO);
|
|
}
|
|
|
|
/* Is the transfer complete? */
|
|
|
|
else if ((chpending & (XDMAC_CHINT_BI | XDMAC_CHINT_LI)) != 0)
|
|
{
|
|
/* Yes.. Terminate the transfer with success */
|
|
|
|
sam_dmaterminate(xdmach, OK);
|
|
}
|
|
|
|
/* Else what? */
|
|
|
|
else
|
|
{
|
|
dmalldbg("ERROR: Unexpected interrupt: %08x\n", chpending);
|
|
DEBUGPANIC();
|
|
}
|
|
|
|
/* Clear the bit in the sampled set of pending global interrupts */
|
|
|
|
gpending &= !bit;
|
|
}
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmainitialize
|
|
*
|
|
* Description:
|
|
* Initialize the DMA subsystem
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
void sam_dmainitialize(struct sam_xdmac_s *xdmac)
|
|
{
|
|
/* Disable all DMA interrupts */
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN_ALL, SAM_XDMAC_GID_OFFSET);
|
|
|
|
/* Disable all DMA channels */
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN_ALL, SAM_XDMAC_GD_OFFSET);
|
|
|
|
/* Initialize semaphores */
|
|
|
|
sem_init(&xdmac->chsem, 0, 1);
|
|
sem_init(&xdmac->dsem, 0, SAMV7_NDMACHAN);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: up_dmainitialize
|
|
*
|
|
* Description:
|
|
* Initialize the DMA subsystem
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
void weak_function up_dmainitialize(void)
|
|
{
|
|
dmallvdbg("Initialize XDMAC\n");
|
|
|
|
/* Enable peripheral clock */
|
|
|
|
sam_xdmac_enableclk();
|
|
|
|
/* Attach DMA interrupt vector */
|
|
|
|
(void)irq_attach(SAM_IRQ_XDMAC, sam_xdmac_interrupt);
|
|
|
|
/* Initialize the controller */
|
|
|
|
sam_dmainitialize(&g_xdmac);
|
|
|
|
/* Enable the IRQ at the AIC (still disabled at the DMA controller) */
|
|
|
|
up_enable_irq(SAM_IRQ_XDMAC);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmachannel
|
|
*
|
|
* Allocate a DMA channel. This function sets aside a DMA channel then
|
|
* gives the caller exclusive access to the DMA channel.
|
|
*
|
|
* The naming convention in all of the DMA interfaces is that one side is
|
|
* the 'peripheral' and the other is 'memory'. Howerver, the interface
|
|
* could still be used if, for example, both sides were memory although
|
|
* the naming would be awkward.
|
|
*
|
|
* Returned Value:
|
|
* If a DMA channel is available, this function returns a non-NULL, void*
|
|
* DMA channel handle. NULL is returned on any failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
DMA_HANDLE sam_dmachannel(uint8_t dmacno, uint32_t chflags)
|
|
{
|
|
struct sam_xdmac_s *xdmac = &g_xdmac;
|
|
struct sam_xdmach_s *xdmach;
|
|
unsigned int chndx;
|
|
|
|
/* Search for an available DMA channel with at least the requested FIFO
|
|
* size.
|
|
*/
|
|
|
|
xdmach = NULL;
|
|
sam_takechsem(xdmac);
|
|
for (chndx = 0; chndx < SAMV7_NDMACHAN; chndx++)
|
|
{
|
|
struct sam_xdmach_s *candidate = &g_xdmach[chndx];
|
|
if (!candidate->inuse)
|
|
{
|
|
xdmach = candidate;
|
|
xdmach->inuse = true;
|
|
|
|
/* Clear the pending Interrupt Status bits by reading the XDMAC
|
|
* Channel Interrupt Status (CIS) Register
|
|
*/
|
|
|
|
(void)sam_getdmach(xdmach, SAM_XDMACH_CIS_OFFSET);
|
|
|
|
/* Disable the channel by writing one to the write-only Global
|
|
* Channel Disable (GD) Register
|
|
*/
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN(chndx), SAM_XDMAC_GD_OFFSET);
|
|
|
|
/* Set the DMA channel flags. */
|
|
|
|
xdmach->flags = chflags;
|
|
break;
|
|
}
|
|
}
|
|
|
|
sam_givechsem(xdmac);
|
|
|
|
/* Show the result of the allocation */
|
|
|
|
if (xdmach)
|
|
{
|
|
dmavdbg("XDMAC%d CH%d: chflags: %08x returning xdmach: %p\n",
|
|
(int)dmacno, xdmach->chan, (int)chflags, xdmach);
|
|
}
|
|
else
|
|
{
|
|
dmadbg("ERROR: Failed allocate XDMAC%d channel\n", (int)dmacno);
|
|
}
|
|
|
|
return (DMA_HANDLE)xdmach;
|
|
}
|
|
|
|
/************************************************************************************
|
|
* Name: sam_dmaconfig
|
|
*
|
|
* Description:
|
|
* There are two channel usage models: (1) The channel is allocated and configured
|
|
* in one step. This is the typical case where a DMA channel performs a constant
|
|
* role. The alternative is (2) where the DMA channel is reconfigured on the fly.
|
|
* In this case, the chflags provided to sam_dmachannel are not used and
|
|
* sam_dmaconfig() is called before each DMA to configure the DMA channel
|
|
* appropriately.
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
************************************************************************************/
|
|
|
|
void sam_dmaconfig(DMA_HANDLE handle, uint32_t chflags)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
|
|
/* Set the new DMA channel flags. */
|
|
|
|
xdmach->flags = chflags;
|
|
dmavdbg("XDMAC CH%d: chflags: %08x\n", xdmach->chan, (int)chflags);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmafree
|
|
*
|
|
* Description:
|
|
* Release a DMA channel. NOTE: The 'handle' used in this argument must
|
|
* NEVER be used again until sam_dmachannel() is called again to re-gain
|
|
* a valid handle.
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
****************************************************************************/
|
|
|
|
void sam_dmafree(DMA_HANDLE handle)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
struct sam_xdmac_s *xdmac;
|
|
|
|
dmavdbg("xdmach: %p\n", xdmach);
|
|
DEBUGASSERT((xdmach != NULL) && (xdmach->inuse));
|
|
|
|
xdmac = sam_controller(xdmach);
|
|
|
|
/* Make sure that the channel is disabled by writing one to the write-only
|
|
* Global Channel Disable (GD) Register
|
|
*/
|
|
|
|
sam_putdmac(xdmac, XDMAC_CHAN(xdmach->chan), SAM_XDMAC_GD_OFFSET);
|
|
|
|
/* Mark the channel no longer in use. Clearing the inuse flag is an atomic
|
|
* operation and so should be safe.
|
|
*/
|
|
|
|
xdmach->flags = 0;
|
|
xdmach->inuse = false; /* No longer in use */
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmatxsetup
|
|
*
|
|
* Description:
|
|
* Configure DMA for transmit of one buffer (memory to peripheral). This
|
|
* function may be called multiple times to handle large and/or dis-
|
|
* continuous transfers. Calls to sam_dmatxsetup() and sam_dmarxsetup()
|
|
* must not be intermixed on the same transfer, however.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int sam_dmatxsetup(DMA_HANDLE handle, uint32_t paddr, uint32_t maddr,
|
|
size_t nbytes)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
size_t maxtransfer;
|
|
size_t remaining;
|
|
int ret = OK;
|
|
|
|
dmavdbg("xdmach: %p paddr: %08x maddr: %08x nbytes: %d\n",
|
|
xdmach, (int)paddr, (int)maddr, (int)nbytes);
|
|
DEBUGASSERT(xdmach);
|
|
dmavdbg("llhead: %p lltail: %p\n", xdmach->llhead, xdmach->lltail);
|
|
|
|
/* The maximum transfer size in bytes depends upon the maximum number of
|
|
* transfers and the number of bytes per transfer.
|
|
*/
|
|
|
|
maxtransfer = sam_maxtransfer(xdmach);
|
|
remaining = nbytes;
|
|
|
|
/* If this is a large transfer, break it up into smaller buffers */
|
|
|
|
while (remaining > maxtransfer)
|
|
{
|
|
/* Set up the maximum size transfer */
|
|
|
|
ret = sam_txbuffer(xdmach, paddr, maddr, maxtransfer);
|
|
if (ret == OK);
|
|
{
|
|
/* Decrement the number of bytes left to transfer */
|
|
|
|
remaining -= maxtransfer;
|
|
|
|
/* Increment the memory & peripheral address (if it is appropriate to
|
|
* do so).
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHINCREMENT) != 0)
|
|
{
|
|
paddr += maxtransfer;
|
|
}
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_MEMINCREMENT) != 0)
|
|
{
|
|
maddr += maxtransfer;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Then set up the final buffer transfer */
|
|
|
|
if (ret == OK && remaining > 0)
|
|
{
|
|
ret = sam_txbuffer(xdmach, paddr, maddr, remaining);
|
|
}
|
|
|
|
/* Save an indication so that the DMA interrupt completion logic will know
|
|
* that this was not an RX transfer.
|
|
*/
|
|
|
|
xdmach->rx = false;
|
|
|
|
/* Clean caches associated with the DMA memory */
|
|
|
|
arch_clean_dcache(maddr, maddr + nbytes);
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmarxsetup
|
|
*
|
|
* Description:
|
|
* Configure DMA for receipt of one buffer (peripheral to memory). This
|
|
* function may be called multiple times to handle large and/or dis-
|
|
* continuous transfers. Calls to sam_dmatxsetup() and sam_dmarxsetup()
|
|
* must not be intermixed on the same transfer, however.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int sam_dmarxsetup(DMA_HANDLE handle, uint32_t paddr, uint32_t maddr,
|
|
size_t nbytes)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
size_t maxtransfer;
|
|
size_t remaining;
|
|
int ret = OK;
|
|
|
|
dmavdbg("xdmach: %p paddr: %08x maddr: %08x nbytes: %d\n",
|
|
xdmach, (int)paddr, (int)maddr, (int)nbytes);
|
|
DEBUGASSERT(xdmach);
|
|
dmavdbg("llhead: %p lltail: %p\n", xdmach->llhead, xdmach->lltail);
|
|
|
|
/* The maximum transfer size in bytes depends upon the maximum number of
|
|
* transfers and the number of bytes per transfer.
|
|
*/
|
|
|
|
maxtransfer = sam_maxtransfer(xdmach);
|
|
remaining = nbytes;
|
|
|
|
/* If this is a large transfer, break it up into smaller buffers */
|
|
|
|
while (remaining > maxtransfer)
|
|
{
|
|
/* Set up the maximum size transfer */
|
|
|
|
ret = sam_rxbuffer(xdmach, paddr, maddr, maxtransfer);
|
|
if (ret == OK);
|
|
{
|
|
/* Decrement the number of bytes left to transfer */
|
|
|
|
remaining -= maxtransfer;
|
|
|
|
/* Increment the memory & peripheral address (if it is appropriate to
|
|
* do so).
|
|
*/
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_PERIPHINCREMENT) != 0)
|
|
{
|
|
paddr += maxtransfer;
|
|
}
|
|
|
|
if ((xdmach->flags & DMACH_FLAG_MEMINCREMENT) != 0)
|
|
{
|
|
maddr += maxtransfer;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Then set up the final buffer transfer */
|
|
|
|
if (ret == OK && remaining > 0)
|
|
{
|
|
ret = sam_rxbuffer(xdmach, paddr, maddr, remaining);
|
|
}
|
|
|
|
/* Save an indication so that the DMA interrupt completion logic will know
|
|
* that this was an RX transfer and will invalidate the cache.
|
|
*/
|
|
|
|
xdmach->rx = true;
|
|
xdmach->rxaddr = maddr;
|
|
xdmach->rxsize = (xdmach->flags & DMACH_FLAG_MEMINCREMENT) != 0 ? nbytes : sizeof(uint32_t);
|
|
|
|
/* Clean caches associated with the DMA memory */
|
|
|
|
arch_clean_dcache(maddr, maddr + nbytes);
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmastart
|
|
*
|
|
* Description:
|
|
* Start the DMA transfer
|
|
*
|
|
****************************************************************************/
|
|
|
|
int sam_dmastart(DMA_HANDLE handle, dma_callback_t callback, void *arg)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
int ret = -EINVAL;
|
|
|
|
dmavdbg("xdmach: %p callback: %p arg: %p\n", xdmach, callback, arg);
|
|
DEBUGASSERT(xdmach != NULL);
|
|
|
|
/* Verify that the DMA has been setup (i.e., at least one entry in the
|
|
* link list).
|
|
*/
|
|
|
|
if (xdmach->llhead)
|
|
{
|
|
/* Save the callback info. This will be invoked when the DMA completes */
|
|
|
|
xdmach->callback = callback;
|
|
xdmach->arg = arg;
|
|
|
|
/* If this is an RX DMA (peripheral-to-memory), then flush and
|
|
* invalidate the data cache to force reloading from memory when the
|
|
* DMA completes.
|
|
*/
|
|
|
|
if (xdmach->rx)
|
|
{
|
|
arch_flush_dcache(xdmach->rxaddr, xdmach->rxaddr + xdmach->rxsize);
|
|
}
|
|
|
|
/* Is this a single block transfer? Or a multiple block transfer? */
|
|
|
|
if (xdmach->llhead == xdmach->lltail)
|
|
{
|
|
ret = sam_single(xdmach);
|
|
}
|
|
else
|
|
{
|
|
ret = sam_multiple(xdmach);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmastop
|
|
*
|
|
* Description:
|
|
* Cancel the DMA. After sam_dmastop() is called, the DMA channel is
|
|
* reset and sam_dmarx/txsetup() must be called before sam_dmastart() can be
|
|
* called again
|
|
*
|
|
****************************************************************************/
|
|
|
|
void sam_dmastop(DMA_HANDLE handle)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
irqstate_t flags;
|
|
|
|
dmavdbg("xdmach: %p\n", xdmach);
|
|
DEBUGASSERT(xdmach != NULL);
|
|
|
|
flags = irqsave();
|
|
sam_dmaterminate(xdmach, -EINTR);
|
|
irqrestore(flags);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmasample
|
|
*
|
|
* Description:
|
|
* Sample DMA register contents
|
|
*
|
|
* Assumptions:
|
|
* - DMA handle allocated by sam_dmachannel()
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifdef CONFIG_DEBUG_DMA
|
|
void sam_dmasample(DMA_HANDLE handle, struct sam_dmaregs_s *regs)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
struct sam_xdmac_s *xdmac = sam_controller(xdmach);
|
|
irqstate_t flags;
|
|
|
|
/* Sample global registers. NOTE: reading GIS clears interrupts, but
|
|
* that should be okay IF interrupts are enabled when this function is
|
|
* called. But there is a race condition where this instrumentation could
|
|
* cause lost interrupts.
|
|
*/
|
|
|
|
flags = irqsave();
|
|
|
|
regs->gtype = sam_getdmac(xdmac, SAM_XDMAC_GTYPE_OFFSET);
|
|
regs->gcfg = sam_getdmac(xdmac, SAM_XDMAC_GCFG_OFFSET);
|
|
regs->gwac = sam_getdmac(xdmac, SAM_XDMAC_GWAC_OFFSET);
|
|
regs->gim = sam_getdmac(xdmac, SAM_XDMAC_GIM_OFFSET);
|
|
regs->gis = sam_getdmac(xdmac, SAM_XDMAC_GIS_OFFSET);
|
|
regs->gs = sam_getdmac(xdmac, SAM_XDMAC_GS_OFFSET);
|
|
regs->grs = sam_getdmac(xdmac, SAM_XDMAC_GRS_OFFSET);
|
|
regs->gws = sam_getdmac(xdmac, SAM_XDMAC_GWS_OFFSET);
|
|
regs->gsws = sam_getdmac(xdmac, SAM_XDMAC_GSWS_OFFSET);
|
|
|
|
/* Sample channel registers */
|
|
|
|
regs->cim = sam_getdmach(xdmach, SAM_XDMACH_CIM_OFFSET);
|
|
regs->cis = sam_getdmach(xdmach, SAM_XDMACH_CIS_OFFSET);
|
|
regs->csa = sam_getdmach(xdmach, SAM_XDMACH_CSA_OFFSET);
|
|
regs->cda = sam_getdmach(xdmach, SAM_XDMACH_CDA_OFFSET);
|
|
regs->cnda = sam_getdmach(xdmach, SAM_XDMACH_CNDA_OFFSET);
|
|
regs->cndc = sam_getdmach(xdmach, SAM_XDMACH_CNDC_OFFSET);
|
|
regs->cubc = sam_getdmach(xdmach, SAM_XDMACH_CUBC_OFFSET);
|
|
regs->cbc = sam_getdmach(xdmach, SAM_XDMACH_CBC_OFFSET);
|
|
regs->cc = sam_getdmach(xdmach, SAM_XDMACH_CC_OFFSET);
|
|
regs->cdsmsp = sam_getdmach(xdmach, SAM_XDMACH_CDSMSP_OFFSET);
|
|
regs->csus = sam_getdmach(xdmach, SAM_XDMACH_CSUS_OFFSET);
|
|
regs->cdus = sam_getdmach(xdmach, SAM_XDMACH_CDUS_OFFSET);
|
|
|
|
irqrestore(flags);
|
|
}
|
|
#endif /* CONFIG_DEBUG_DMA */
|
|
|
|
/****************************************************************************
|
|
* Name: sam_dmadump
|
|
*
|
|
* Description:
|
|
* Dump previously sampled DMA register contents
|
|
*
|
|
* Assumptions:
|
|
* - DMA handle allocated by sam_dmachannel()
|
|
*
|
|
****************************************************************************/
|
|
|
|
#ifdef CONFIG_DEBUG_DMA
|
|
void sam_dmadump(DMA_HANDLE handle, const struct sam_dmaregs_s *regs,
|
|
const char *msg)
|
|
{
|
|
struct sam_xdmach_s *xdmach = (struct sam_xdmach_s *)handle;
|
|
|
|
dmadbg("%s\n", msg);
|
|
dmadbg(" DMA Global Registers:\n");
|
|
dmadbg(" GTYPE[%08x]: %08x\n", SAM_XDMAC_GTYPE, regs->gtype);
|
|
dmadbg(" GCFG[%08x]: %08x\n", SAM_XDMAC_GCFG, regs->gcfg);
|
|
dmadbg(" GWAC[%08x]: %08x\n", SAM_XDMAC_GWAC, regs->gwac);
|
|
dmadbg(" GIM[%08x]: %08x\n", SAM_XDMAC_GIM, regs->gim);
|
|
dmadbg(" GIS[%08x]: %08x\n", SAM_XDMAC_GIS, regs->gis);
|
|
dmadbg(" GS[%08x]: %08x\n", SAM_XDMAC_GS, regs->gs);
|
|
dmadbg(" GRS[%08x]: %08x\n", SAM_XDMAC_GRS, regs->grs);
|
|
dmadbg(" GWS[%08x]: %08x\n", SAM_XDMAC_GWS, regs->gws);
|
|
dmadbg(" GSWS[%08x]: %08x\n", SAM_XDMAC_GSWS, regs->gsws);
|
|
dmadbg(" DMA Channel Registers:\n");
|
|
dmadbg(" CIM[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CIM_OFFSET, regs->cim);
|
|
dmadbg(" CIS[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CIS_OFFSET, regs->cis);
|
|
dmadbg(" CSA[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CSA_OFFSET, regs->csa);
|
|
dmadbg(" CDA[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CDA_OFFSET, regs->cda);
|
|
dmadbg(" CNDA[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CNDA_OFFSET, regs->cnda);
|
|
dmadbg(" CNDC[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CNDC_OFFSET, regs->cndc);
|
|
dmadbg(" CUBC[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CUBC_OFFSET, regs->cubc);
|
|
dmadbg(" CBC[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CBC_OFFSET, regs->cbc);
|
|
dmadbg(" CC[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CC_OFFSET, regs->cc);
|
|
dmadbg(" CDSMSP[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CDSMSP_OFFSET, regs->cdsmsp);
|
|
dmadbg(" CSUS[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CSUS_OFFSET, regs->csus);
|
|
dmadbg(" CDUS[%08x]: %08x\n", xdmach->base + SAM_XDMACH_CDUS_OFFSET, regs->cdus);
|
|
}
|
|
#endif /* CONFIG_DEBUG_DMA */
|
|
#endif /* CONFIG_SAMV7_XDMAC */
|