| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469 |
- /*
- * Copyright (C) 2012 by Simon Budig (simon@budig.de)
- *
- * 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 of the copyright holders nor the names of
- * 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.
- *
- * For additional information see http://www.ethernut.de/
- *
- */
- /*
- * \verbatim
- * $Id$
- * \endverbatim
- */
- #include <cfg/arch.h>
- #include <sys/timer.h>
- #include <sys/nutdebug.h>
- #include <dev/spibus.h>
- #include <dev/gpio.h>
- #include <arch/cm3.h>
- #include <arch/cm3/nxp/mach/lpc_sc.h>
- #include <arch/cm3/nxp/mach/lpc_ssp.h>
- #if defined(MCU_LPC176x)
- #include <arch/cm3/nxp/lpc176x.h>
- #include <arch/cm3/nxp/lpc176x_clk.h>
- #elif defined(MCU_LPC177x_8x)
- #include <arch/cm3/nxp/lpc177x_8x.h>
- #include <arch/cm3/nxp/lpc177x_8x_clk.h>
- #elif defined(MCU_LPC407x_8x)
- #include <arch/cm3/nxp/lpc407x_8x.h>
- #include <arch/cm3/nxp/lpc407x_8x_clk.h>
- #else
- #warning "Unknown LPC familiy"
- #endif
- #include <errno.h>
- #include <stdlib.h>
- #if defined(MCU_LPC176x)
- #define SSP0BUS_SCK_PORT NUTGPIO_PORT0
- #define SSP0BUS_MISO_PORT NUTGPIO_PORT0
- #define SSP0BUS_MOSI_PORT NUTGPIO_PORT0
- #define SSP0BUS_SCK_PIN 15
- #define SSP0BUS_MISO_PIN 17
- #define SSP0BUS_MOSI_PIN 18
- #define SSP0BUS_SCK_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL2)
- #define SSP0BUS_MISO_PIN_CFG (GPIO_CFG_PERIPHERAL2)
- #define SSP0BUS_MOSI_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL2)
- #define SSP1BUS_SCK_PORT NUTGPIO_PORT0
- #define SSP1BUS_MISO_PORT NUTGPIO_PORT0
- #define SSP1BUS_MOSI_PORT NUTGPIO_PORT0
- #define SSP1BUS_SCK_PIN 7
- #define SSP1BUS_MISO_PIN 8
- #define SSP1BUS_MOSI_PIN 9
- #define SSP1BUS_SCK_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL2)
- #define SSP1BUS_MISO_PIN_CFG (GPIO_CFG_PERIPHERAL2)
- #define SSP1BUS_MOSI_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL2)
- #elif defined(MCU_LPC177x_8x) || defined(MCU_LPC407x_8x)
- #define SSP0BUS_SCK_PORT NUTGPIO_PORT2
- #define SSP0BUS_MISO_PORT NUTGPIO_PORT2
- #define SSP0BUS_MOSI_PORT NUTGPIO_PORT2
- #define SSP0BUS_SCK_PIN 22
- #define SSP0BUS_MISO_PIN 26
- #define SSP0BUS_MOSI_PIN 27
- #define SSP0BUS_SCK_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL2)
- #define SSP0BUS_MISO_PIN_CFG (GPIO_CFG_PERIPHERAL2)
- #define SSP0BUS_MOSI_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL2)
- #define SSP1BUS_SCK_PORT NUTGPIO_PORT4
- #define SSP1BUS_MISO_PORT NUTGPIO_PORT4
- #define SSP1BUS_MOSI_PORT NUTGPIO_PORT4
- #define SSP1BUS_SCK_PIN 20
- #define SSP1BUS_MISO_PIN 22
- #define SSP1BUS_MOSI_PIN 23
- #define SSP1BUS_SCK_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL3)
- #define SSP1BUS_MISO_PIN_CFG (GPIO_CFG_PERIPHERAL3)
- #define SSP1BUS_MOSI_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL3)
- #if defined(LPC_SSP2_BASE)
- #define SSP2BUS_SCK_PORT NUTGPIO_PORT1
- #define SSP2BUS_MISO_PORT NUTGPIO_PORT1
- #define SSP2BUS_MOSI_PORT NUTGPIO_PORT1
- #define SSP2BUS_SCK_PIN 0
- #define SSP2BUS_MISO_PIN 4
- #define SSP2BUS_MOSI_PIN 1
- #define SSP2BUS_SCK_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL4)
- #define SSP2BUS_MISO_PIN_CFG (GPIO_CFG_PERIPHERAL4)
- #define SSP2BUS_MOSI_PIN_CFG (GPIO_CFG_OUTPUT | GPIO_CFG_PERIPHERAL4)
- #endif
- #endif
- static int Lpc17xxSspSetup (NUTSPINODE * node);
- static int Lpc17xxSspBusNodeInit (NUTSPINODE * node);
- static int Lpc17xxSspBusSelect (NUTSPINODE * node, uint32_t tmo);
- static int Lpc17xxSspBusDeselect (NUTSPINODE * node);
- static int Lpc17xxSspBusTransfer (NUTSPINODE * node, const void *txbuf, void *rxbuf, int xlen);
- NUTSPIBUS spiBus0Lpc17xxSsp = {
- NULL, /*!< Bus mutex semaphore (bus_mutex). */
- NULL, /*!< Bus ready signal (bus_ready). */
- LPC_SSP0_BASE, /*!< Bus base address (bus_base). */
- NULL, /*!< Bus interrupt handler (bus_sig). */
- Lpc17xxSspBusNodeInit, /*!< Initialize the bus (bus_initnode). */
- Lpc17xxSspBusSelect, /*!< Select the specified device (bus_alloc). */
- Lpc17xxSspBusDeselect, /*!< Deselect the specified device (bus_release). */
- Lpc17xxSspBusTransfer,
- NutSpiBusWait,
- NutSpiBusSetMode, /*!< Set SPI mode of a specified device (bus_set_mode). */
- NutSpiBusSetRate, /*!< Set clock rate of a specified device (bus_set_rate). */
- NutSpiBusSetBits /*!< Set number of data bits of a specified device (bus_set_bits). */
- };
- NUTSPIBUS spiBus1Lpc17xxSsp = {
- NULL, /*!< Bus mutex semaphore (bus_mutex). */
- NULL, /*!< Bus ready signal (bus_ready). */
- LPC_SSP1_BASE, /*!< Bus base address (bus_base). */
- NULL, /*!< Bus interrupt handler (bus_sig). */
- Lpc17xxSspBusNodeInit, /*!< Initialize the bus (bus_initnode). */
- Lpc17xxSspBusSelect, /*!< Select the specified device (bus_alloc). */
- Lpc17xxSspBusDeselect, /*!< Deselect the specified device (bus_release). */
- Lpc17xxSspBusTransfer,
- NutSpiBusWait,
- NutSpiBusSetMode, /*!< Set SPI mode of a specified device (bus_set_mode). */
- NutSpiBusSetRate, /*!< Set clock rate of a specified device (bus_set_rate). */
- NutSpiBusSetBits /*!< Set number of data bits of a specified device (bus_set_bits). */
- };
- #if defined(LPC_SSP2_BASE)
- NUTSPIBUS spiBus2Lpc17xxSsp = {
- NULL, /*!< Bus mutex semaphore (bus_mutex). */
- NULL, /*!< Bus ready signal (bus_ready). */
- LPC_SSP2_BASE, /*!< Bus base address (bus_base). */
- NULL, /*!< Bus interrupt handler (bus_sig). */
- Lpc17xxSspBusNodeInit, /*!< Initialize the bus (bus_initnode). */
- Lpc17xxSspBusSelect, /*!< Select the specified device (bus_alloc). */
- Lpc17xxSspBusDeselect, /*!< Deselect the specified device (bus_release). */
- Lpc17xxSspBusTransfer,
- NutSpiBusWait,
- NutSpiBusSetMode, /*!< Set SPI mode of a specified device (bus_set_mode). */
- NutSpiBusSetRate, /*!< Set clock rate of a specified device (bus_set_rate). */
- NutSpiBusSetBits /*!< Set number of data bits of a specified device (bus_set_bits). */
- };
- #endif
- /*!
- * \brief Set the specified chip select to a given level.
- */
- static int Lpc17xxSspChipSelect(uint_fast8_t cs, uint_fast8_t hi)
- {
- if (cs != 0xFF) {
- GpioPinSet(cs / 32, cs % 32, hi);
- }
- return 0;
- }
- /*! \brief Deselect a device on the SSP bus.
- *
- * Deactivates the chip select and unlocks the bus.
- *
- * \param node Specifies the SSP bus node.
- *
- * \return Always 0.
- */
- static int Lpc17xxSspBusDeselect(NUTSPINODE * node)
- {
- /* Sanity check. */
- NUTASSERT(node != NULL);
- NUTASSERT(node->node_bus != NULL);
- NutSpiBusWait(node, NUT_WAIT_INFINITE);
- /* Deactivate the node's chip select. */
- Lpc17xxSspChipSelect(node->node_cs, (node->node_mode & SPI_MODE_CSHIGH) == 0);
- /* Release the bus. */
- NutEventPostAsync(&node->node_bus->bus_mutex);
- return 0;
- }
- /*! \brief Select a device on the SSP bus.
- *
- * Locks and activates the bus for the specified node.
- *
- * \param node Specifies the SSP bus node.
- * \param tmo Timeout in milliseconds. To disable timeout, set this
- * parameter to NUT_WAIT_INFINITE.
- *
- * \return 0 on success. In case of an error, -1 is returned and the bus
- * is not locked.
- */
- static int Lpc17xxSspBusSelect(NUTSPINODE * node, uint32_t tmo)
- {
- int rc;
- LPC_SSP_TypeDef* base;
- /* Sanity check. */
- NUTASSERT(node != NULL);
- NUTASSERT(node->node_bus != NULL);
- NUTASSERT(node->node_stat != NULL);
- base = (LPC_SSP_TypeDef*) (node->node_bus->bus_base);
- /* Allocate the bus. */
- rc = NutEventWait(&node->node_bus->bus_mutex, tmo);
- if (rc) {
- errno = EIO;
- } else {
- LPC_SSP_TypeDef *sspreg = node->node_stat;
- /* If the mode update bit is set, then update our shadow registers. */
- if (node->node_mode & SPI_MODE_UPDATE) {
- Lpc17xxSspSetup(node);
- }
- /* Set SPI mode. */
- base->CR0 = sspreg->CR0;
- base->CR1 = sspreg->CR1;
- base->CPSR = sspreg->CPSR;
-
- /* Finally activate the node's chip select. */
- rc = Lpc17xxSspChipSelect(node->node_cs, (node->node_mode & SPI_MODE_CSHIGH) != 0);
- if (rc) {
- /* Release the bus in case of an error. */
- NutEventPost(&node->node_bus->bus_mutex);
- }
- }
- return rc;
- }
- /*!
- * \brief Update SSP shadow registers.
- *
- * \param node Specifies the SPI bus node.
- *
- * \return Always 0.
- */
- static int Lpc17xxSspSetup(NUTSPINODE * node)
- {
- uint32_t clk;
- uint32_t prescale;
- uint32_t cr0_div;
- uint32_t cmp_clk;
-
- LPC_SSP_TypeDef *sspreg;
- NUTASSERT(node != NULL);
- NUTASSERT(node->node_stat != NULL);
- NUTASSERT(node->node_bus != NULL);
- NUTASSERT(node->node_bus->bus_base != 0);
- sspreg = node->node_stat;
- sspreg->CR0 &= ~(SSP_CR0_DSS_MSK | SSP_CR0_FRF_MSK | SSP_CR0_CPOL | SSP_CR0_CPHA | SSP_CR0_SCR_MSK);
- sspreg->CR0 |= SSP_CR0_DSS (node->node_bits);
- if (node->node_mode & SPI_MODE_CPOL) {
- sspreg->CR0 |= SSP_CR0_CPOL;
- }
- if (node->node_mode & SPI_MODE_CPHA) {
- sspreg->CR0 |= SSP_CR0_CPHA;
- }
- sspreg->CR0 |= SSP_CR0_FRF_SPI;
- /* master only for now */
- sspreg->CR1 &= ~(SSP_CR1_LBM | SSP_CR1_SSE | SSP_CR1_MS | SSP_CR1_SOD);
- clk = NutClockGet(NUT_HWCLK_PCLK);
- #if defined(MCU_LPC176x)
- if (node->node_bus->bus_base == LPC_SSP0_BASE) {
- clk /= Lpc176x_PclkDivGet(CLKPWR_PCLKSEL_SSP0);
- } else if (node->node_bus->bus_base == LPC_SSP1_BASE) {
- clk /= Lpc176x_PclkDivGet(CLKPWR_PCLKSEL_SSP1);
- }
- #if defined(LPC_SSP2_BASE)
- else if (node->node_bus->bus_base == LPC_SSP2_BASE) {
- clk /= Lpc176x_PclkDivGet(CLKPWR_PCLKSEL_SSP2);
- }
- #endif
- #endif
- /* Find closest divider to get at or under the target frequency.
- Use smallest prescale possible and rely on the divider to get
- the closest target frequency */
- cr0_div = 0;
- cmp_clk = 0xFFFFFFFF;
- prescale = 2;
- while (cmp_clk > node->node_rate)
- {
- cmp_clk = clk / ((cr0_div + 1) * prescale);
- if (cmp_clk > node->node_rate)
- {
- cr0_div++;
- if (cr0_div > 0xFF)
- {
- cr0_div = 0;
- prescale += 2;
- }
- }
- }
- /* Write computed prescaler and divider back to register */
- sspreg->CR0 &= ~SSP_CR0_SCR_MSK;
- sspreg->CR0 |= (cr0_div << SSP_CR0_SCR_LSB) & SSP_CR0_SCR_MSK;
- sspreg->CPSR = prescale & 0xFF;
- /* Update interface parameters. */
- node->node_rate = (clk / prescale) / (cr0_div + 1);
- node->node_mode &= ~SPI_MODE_UPDATE;
- return 0;
- }
- /*!
- * \brief Initialize an SSP bus node.
- *
- * This routine is called for each SSP node, which is registered via
- * NutRegisterSpiDevice().
- *
- * \param node Specifies the SSP bus node.
- *
- * \return 0 on success or -1 if there is no valid chip select.
- */
- static int Lpc17xxSspBusNodeInit(NUTSPINODE * node)
- {
- int rc = -1;
- /* Sanity check. */
- NUTASSERT(node != NULL);
- NUTASSERT(node->node_bus != NULL);
- if (node->node_bus->bus_base == LPC_SSP0_BASE) {
- GpioPinConfigSet(SSP0BUS_SCK_PORT, SSP0BUS_SCK_PIN, SSP0BUS_SCK_PIN_CFG); // SCK
- GpioPinConfigSet(SSP0BUS_MISO_PORT, SSP0BUS_MISO_PIN, SSP0BUS_MISO_PIN_CFG); // MISO
- GpioPinConfigSet(SSP0BUS_MOSI_PORT, SSP0BUS_MOSI_PIN, SSP0BUS_MOSI_PIN_CFG); // MOSI
- } else if (node->node_bus->bus_base == LPC_SSP1_BASE) {
- GpioPinConfigSet(SSP1BUS_SCK_PORT, SSP1BUS_SCK_PIN, SSP1BUS_SCK_PIN_CFG); // SCK
- GpioPinConfigSet(SSP1BUS_MISO_PORT, SSP1BUS_MISO_PIN, SSP1BUS_MISO_PIN_CFG); // MISO
- GpioPinConfigSet(SSP1BUS_MOSI_PORT, SSP1BUS_MOSI_PIN, SSP1BUS_MOSI_PIN_CFG); // MOSI
- }
- #if defined(LPC_SSP2_BASE)
- else if (node->node_bus->bus_base == LPC_SSP2_BASE) {
- GpioPinConfigSet(SSP2BUS_SCK_PORT, SSP2BUS_SCK_PIN, SSP2BUS_SCK_PIN_CFG); // SCK
- GpioPinConfigSet(SSP2BUS_MISO_PORT, SSP2BUS_MISO_PIN, SSP2BUS_MISO_PIN_CFG); // MISO
- GpioPinConfigSet(SSP2BUS_MOSI_PORT, SSP2BUS_MOSI_PIN, SSP2BUS_MOSI_PIN_CFG); // MOSI
- }
- #endif
- if (node->node_cs != 0xFF) {
- GpioPinConfigSet(node->node_cs / 32, node->node_cs % 32, GPIO_CFG_OUTPUT);
- }
-
- /* Try to deactivate the node's chip select. */
- rc = Lpc17xxSspChipSelect(node->node_cs, (node->node_mode & SPI_MODE_CSHIGH) == 0);
- /* It should not hurt us being called more than once. Thus, we
- ** check wether any initialization had been taken place already. */
- if (rc == 0 && node->node_stat == NULL)
- {
- /* Allocate and set our shadow registers. */
- LPC_SSP_TypeDef *sspreg = malloc(sizeof(LPC_SSP_TypeDef));
- if (sspreg) {
- /* Update with node's defaults. */
- node->node_stat = (void *)sspreg;
- Lpc17xxSspSetup(node);
- }
- else {
- /* Out of memory? */
- rc = -1;
- }
- }
- return rc;
- }
- /*!
- * \brief Transfer data on the SSP bus using single buffered interrupt mode.
- *
- * A device must have been selected by calling At91SpiSelect().
- *
- * \param node Specifies the SSP bus node.
- * \param txbuf Pointer to the transmit buffer. If NULL, undetermined
- * byte values are transmitted.
- * \param rxbuf Pointer to the receive buffer. If NULL, then incoming
- * data is discarded.
- * \param xlen Number of bytes to transfer.
- *
- * \return Always 0.
- */
- static int Lpc17xxSspBusTransfer(NUTSPINODE * node, const void *txbuf, void *rxbuf, int xlen)
- {
- LPC_SSP_TypeDef* base;
- /* Sanity check. */
- NUTASSERT(node != NULL);
- NUTASSERT(node->node_bus != NULL);
- NUTASSERT(node->node_bus->bus_base != 0);
- base = (LPC_SSP_TypeDef*) node->node_bus->bus_base;
- unsigned char *tx = (unsigned char*) txbuf;
- unsigned char *rx = (unsigned char*) rxbuf;
- base->CR1 |= SSP_CR1_SSE;
- while (xlen-- > 0) {
- unsigned char b = tx ? (*tx++) : 0xff;
- base->DR = b;
- /* wait until receive buffer no longer empty */
- while ((base->SR & SSP_SR_RNE) == 0)
- ;
- b = base->DR;
- if (rx) {
- *rx++ = b;
- }
- }
- base->CR1 &= ~SSP_CR1_SSE;
- return 0;
- }
|