diff options
Diffstat (limited to 'cpu/arm926ejs/at91sam926x/spi.c')
-rw-r--r-- | cpu/arm926ejs/at91sam926x/spi.c | 243 |
1 files changed, 243 insertions, 0 deletions
diff --git a/cpu/arm926ejs/at91sam926x/spi.c b/cpu/arm926ejs/at91sam926x/spi.c new file mode 100644 index 00000000000..bd855283b92 --- /dev/null +++ b/cpu/arm926ejs/at91sam926x/spi.c @@ -0,0 +1,243 @@ +/* + * (c) Copyright 2005 + * M. Amine SAYA, ATMEL Rousset, France. + * Driver for AT91SAM9261 SPI + * Based on at91rm9200 spi driver from H. Ikdoumi, ATMEL Rousset. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, + * MA 02111-1307 USA + * + */ + +#include <config.h> +#include <common.h> + +#ifdef CONFIG_HAS_DATAFLASH +#include <dataflash.h> +#include <asm/arch/pio.h> + +#ifdef CONFIG_AT91SAM9263EK +#define AT91C_SPI_CS0_CLK 8000000 /* SPI Clock for Dataflash cards -> SAM9263EK */ +#else +#define AT91C_SPI_CS0_CLK 33000000 /* SPI Clock for Dataflash devices */ +#endif + +#define AT91C_SPI_CS1_CLK 33000000 /* SPI Clock for Dataflash devices -> SAM9260EK */ +#define AT91C_SPI_CS3_CLK 8000000 /* SPI Clock for Dataflash cards -> SAM9261EK */ + +#define AT91C_SPI_PCS0_DATAFLASH_CARD 0xE /* Chip Select 0 : NPCS0 %1110 */ +#define AT91C_SPI_PCS1_DATAFLASH_CARD 0xD /* Chip Select 1 : NPCS1 %1101 */ +#define AT91C_SPI_PCS3_DATAFLASH_CARD 0x7 /* Chip Select 3 : NPCS3 %0111 */ + + +/*----------------------------------------------------------------------------*/ +/* \fn AT91F_SpiInit */ +/* \brief SPI Init */ +/*----------------------------------------------------------------------------*/ +void AT91F_SpiInit(void) { + + volatile unsigned int dummy; + +#ifdef CONFIG_AT91SAM9260EK + /* Configure PIO controllers to periph mode */ + AT91F_PIO_CfgPeriph(AT91C_BASE_PIOA, // PIO controller base address + ((unsigned int) AT91C_PA1_SPI0_MOSI) | + ((unsigned int) AT91C_PA3_SPI0_NPCS0) | + ((unsigned int) AT91C_PA0_SPI0_MISO) | + ((unsigned int) AT91C_PA2_SPI0_SPCK), /* Peripheral A */ + 0); /* Peripheral B */ + /* Configure PIO controllers to periph mode */ + AT91F_PIO_CfgPeriph(AT91C_BASE_PIOC, /* PIO controller base address */ + 0, /* Peripheral A */ + ((unsigned int) AT91C_PC11_SPI0_NPCS1)); /* Peripheral B */ +#endif + +#ifdef CONFIG_AT91SAM9261EK + /* Configure PIOs for SPI0 (internal DataFlash) */ + AT91F_PIO_CfgPeriph(AT91C_BASE_PIOA, + (AT91C_PA0_MISO0 | AT91C_PA1_MOSI0 | + AT91C_PA2_SPCK0 | AT91C_PA3_NPCS00), + 0); +#endif + +#ifdef CONFIG_AT91SAM9263EK + /* Configure PIO controllers to periph mode */ + AT91F_PIO_CfgPeriph(AT91C_BASE_PIOA, /* PIO controller base address */ + 0, /* Peripheral A */ + ((unsigned int) AT91C_PA2_SPI0_SPCK) | + ((unsigned int) AT91C_PA1_SPI0_MOSI) | + ((unsigned int) AT91C_PA5_SPI0_NPCS0) | + ((unsigned int) AT91C_PA0_SPI0_MISO)); /* Peripheral B */ +#endif + +#ifdef CONFIG_AT91SAM9RLEK + /* Configure PIOs for SPI0 (internal DataFlash) */ + AT91F_PIO_CfgPeriph(AT91C_BASE_PIOA, + ((unsigned int) AT91C_PA25_MISO) | + ((unsigned int) AT91C_PA26_MOSI) | + ((unsigned int) AT91C_PA27_SPCK) | + ((unsigned int) AT91C_PA28_NPCS0), /* Peripheral A */ + 0 /* Peripheral B */); +#endif + + + /* Enable Clock */ + AT91C_BASE_PMC->PMC_PCER = 1 << AT91C_ID_SPI0; + + /* Reset the SPI */ + AT91C_BASE_SPI0->SPI_CR = AT91C_SPI_SWRST; + + /* Configure SPI in Master Mode with No CS selected !!! */ + AT91C_BASE_SPI0->SPI_MR = AT91C_SPI_MSTR | AT91C_SPI_MODFDIS | AT91C_SPI_PCS; + + /* Configure CS0 in Mode 0 */ + /* to avoid pb with DataFlash AT45DB642D */ + AT91C_BASE_SPI0->SPI_CSR[0] = (AT91C_SPI_NCPHA | (AT91C_SPI_DLYBS & DATAFLASH_TCSS) | + (AT91C_SPI_DLYBCT & DATAFLASH_TCHS) | + (AT91C_MASTER_CLOCK / AT91C_SPI_CS0_CLK) << 8); + + +#ifdef CONFIG_AT91SAM9260EK + AT91C_BASE_SPI0->SPI_CSR[1] = (AT91C_SPI_NCPHA | (AT91C_SPI_DLYBS & DATAFLASH_TCSS) | + (AT91C_SPI_DLYBCT & DATAFLASH_TCHS) | + (AT91C_MASTER_CLOCK / AT91C_SPI_CS1_CLK) << 8); +#endif +#ifdef CONFIG_AT91SAM9261EK + AT91C_BASE_SPI0->SPI_CSR[3] = (AT91C_SPI_NCPHA | (AT91C_SPI_DLYBS & DATAFLASH_TCSS) | + (AT91C_SPI_DLYBCT & DATAFLASH_TCHS) | + (AT91C_MASTER_CLOCK / AT91C_SPI_CS3_CLK) << 8); +#endif + + /* SPI_Enable */ + AT91C_BASE_SPI0->SPI_CR = AT91C_SPI_SPIEN; + + while(!(AT91C_BASE_SPI0->SPI_SR & AT91C_SPI_SPIENS)); + + /* Add tempo to get SPI in a safe state. + Should be removed for new silicon (Rev B) + */ + udelay(500000); + dummy = AT91C_BASE_SPI0->SPI_SR; + dummy = AT91C_BASE_SPI0->SPI_RDR; +} + + +/*----------------------------------------------------------------------------*/ +/* \fn AT91F_SpiEnable */ +/* \brief SPI Enable */ +/*----------------------------------------------------------------------------*/ +void AT91F_SpiEnable(int cs) { + AT91C_BASE_SPI0->SPI_MR &= 0xFFF0FFFF; + + switch(cs) { + case 0: + /* Configure SPI CS0 for DataFlash card AT45DBxx */ + AT91C_BASE_SPI0->SPI_MR |= ((AT91C_SPI_PCS0_DATAFLASH_CARD<<16) & AT91C_SPI_PCS); + break; + case 1: + /* Configure SPI CS1 for dataflash card AT45DBxx */ + AT91C_BASE_SPI0->SPI_MR |= ((AT91C_SPI_PCS1_DATAFLASH_CARD<<16) & AT91C_SPI_PCS); + break; + case 2: + case 3: + /* Configure SPI CS3 for dataflash card AT45DBxx */ + AT91C_BASE_SPI0->SPI_MR |= ((AT91C_SPI_PCS3_DATAFLASH_CARD<<16) & AT91C_SPI_PCS); + break; + default: + /* Configure PCS */ + AT91C_BASE_SPI0->SPI_MR |= ((AT91C_SPI_PCS0_DATAFLASH_CARD<<16) & AT91C_SPI_PCS); + break; + } + + /* SPI_Enable */ + AT91C_BASE_SPI0->SPI_CR = AT91C_SPI_SPIEN; +} + + +/*----------------------------------------------------------------------------*/ +/* \fn AT91F_SpiDisablePDC */ +/* \brief Disable PDC registers */ +/*----------------------------------------------------------------------------*/ +void __inline__ AT91F_SpiDisablePDC(AT91PS_SPI spi_ptr) +{ + spi_ptr->SPI_PTCR = AT91C_PDC_RXTDIS | AT91C_PDC_TXTDIS; +} + + +/*----------------------------------------------------------------------------*/ +/* \fn AT91F_SpiEnablePDC */ +/* \brief Enable PDC registers */ +/*----------------------------------------------------------------------------*/ +void __inline__ AT91F_SpiEnablePDC(AT91PS_SPI spi_ptr) +{ + spi_ptr->SPI_PTCR = AT91C_PDC_RXTEN | AT91C_PDC_TXTEN; +} + + +/*----------------------------------------------------------------------------*/ +/* \fn AT91F_SpiWrite */ +/* \brief Set the PDC registers for a transfert */ +/*----------------------------------------------------------------------------*/ +unsigned int AT91F_SpiWrite ( AT91PS_DataflashDesc pDesc ) +{ + unsigned int timeout; + + pDesc->state = BUSY; + + /* Disable PDC TX and RX channels */ + AT91C_BASE_SPI0->SPI_PTCR = AT91C_PDC_TXTDIS + AT91C_PDC_RXTDIS; + + /* Initialize the Transmit and Receive Pointer */ + AT91C_BASE_SPI0->SPI_RPR = (unsigned int)pDesc->rx_cmd_pt ; + AT91C_BASE_SPI0->SPI_TPR = (unsigned int)pDesc->tx_cmd_pt ; + + /* Intialize the Transmit and Receive Counters */ + AT91C_BASE_SPI0->SPI_RCR = pDesc->rx_cmd_size; + AT91C_BASE_SPI0->SPI_TCR = pDesc->tx_cmd_size; + + /* If there is another transfer to perform */ + if(pDesc->tx_data_size){ + /* Initialize the Next Transmit and Next Receive Pointer */ + AT91C_BASE_SPI0->SPI_RNPR = (unsigned int)pDesc->rx_data_pt ; + AT91C_BASE_SPI0->SPI_TNPR = (unsigned int)pDesc->tx_data_pt ; + + /* Initialize the Next Transmit and Next Receive Counters */ + AT91C_BASE_SPI0->SPI_RNCR = pDesc->rx_data_size ; + AT91C_BASE_SPI0->SPI_TNCR = pDesc->tx_data_size ; + } + + /* arm simple, non interrupt dependent timer */ + reset_timer_masked(); + timeout = 0; + + AT91F_SpiEnablePDC(AT91C_BASE_SPI0); + + while(!(AT91C_BASE_SPI0->SPI_SR & AT91C_SPI_RXBUFF) && ((timeout = get_timer_masked() ) < CFG_SPI_WRITE_TOUT)); + + /* Disable PDC TX and RX channels */ + AT91F_SpiDisablePDC(AT91C_BASE_SPI0); + + pDesc->state = IDLE; + + + if (timeout >= CFG_SPI_WRITE_TOUT){ + printf("AT91F_SpiWrite : Error Timeout\n\r"); + return DATAFLASH_ERROR; + } + + return DATAFLASH_OK; +} + +#endif |