blob: e5df72a4928875e572137878dd6767b7e699d49e [file] [log] [blame]
/* ----------------------------------------------------------------------------
* SAM Software Package License
* ----------------------------------------------------------------------------
* Copyright (c) 2014, Atmel Corporation
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* - Redistributions of source code must retain the above copyright notice,
* this list of conditions and the disclaimer below.
*
* Atmel's name may not be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
* DISCLAIMED. IN NO EVENT SHALL ATMEL 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.
* ----------------------------------------------------------------------------
*/
/** \addtogroup ddrd_module
*
* The DDR/SDR SDRAM Controller (DDRSDRC) is a multiport memory controller. It comprises
* four slave AHB interfaces. All simultaneous accesses (four independent AHB ports) are interleaved
* to maximize memory bandwidth and minimize transaction latency due to SDRAM protocol.
*
* \section ddr2 Configures DDR2
*
* The DDR2-SDRAM devices are initialized by the following sequence:
* <ul>
* <li> EBI Chip Select 1 is assigned to the DDR2SDR Controller, Enable DDR2 clock x2 in PMC.</li>
* <li> Step 1: Program the memory device type</li>
* <li> Step 2:
* -# Program the features of DDR2-SDRAM device into the Configuration Register.
* -# Program the features of DDR2-SDRAM device into the Timing Register HDDRSDRC2_T0PR.
* -# Program the features of DDR2-SDRAM device into the Timing Register HDDRSDRC2_T1PR.
* -# Program the features of DDR2-SDRAM device into the Timing Register HDDRSDRC2_T2PR. </li>
* <li> Step 3: An NOP command is issued to the DDR2-SDRAM to enable clock. </li>
* <li> Step 4: An NOP command is issued to the DDR2-SDRAM </li>
* <li> Step 5: An all banks precharge command is issued to the DDR2-SDRAM. </li>
* <li> Step 6: An Extended Mode Register set (EMRS2) cycle is issued to chose between commercialor high temperature operations.</li>
* <li> Step 7: An Extended Mode Register set (EMRS3) cycle is issued to set all registers to 0. </li>
* <li> Step 8: An Extended Mode Register set (EMRS1) cycle is issued to enable DLL.</li>
* <li> Step 9: Program DLL field into the Configuration Register.</li>
* <li> Step 10: A Mode Register set (MRS) cycle is issued to reset DLL.</li>
* <li> Step 11: An all banks precharge command is issued to the DDR2-SDRAM.</li>
* <li> Step 12: Two auto-refresh (CBR) cycles are provided. Program the auto refresh command (CBR) into the Mode Register.</li>
* <li> Step 13: Program DLL field into the Configuration Register to low(Disable DLL reset).</li>
* <li> Step 14: A Mode Register set (MRS) cycle is issued to program the parameters of the DDR2-SDRAM devices.</li>
* <li> Step 15: Program OCD field into the Configuration Register to high (OCD calibration default). </li>
* <li> Step 16: An Extended Mode Register set (EMRS1) cycle is issued to OCD default value.</li>
* <li> Step 17: Program OCD field into the Configuration Register to low (OCD calibration mode exit).</li>
* <li> Step 18: An Extended Mode Register set (EMRS1) cycle is issued to enable OCD exit.</li>
* <li> Step 19,20: A mode Normal command is provided. Program the Normal mode into Mode Register.</li>
* <li> Step 21: Write the refresh rate into the count field in the Refresh Timer register. The DDR2-SDRAM device requires a refresh every 15.625 or 7.81. </li>
* </ul>
*/
/*@{*/
/*@}*/
/** \addtogroup sdram_module
*
* \section sdram Configures SDRAM
*
* The SDR-SDRAM devices are initialized by the following sequence:
* <ul>
* <li> EBI Chip Select 1 is assigned to the DDR2SDR Controller, Enable DDR2 clock x2 in PMC.</li>
* <li> Step 1. Program the memory device type into the Memory Device Register</li>
* <li> Step 2. Program the features of the SDR-SDRAM device into the Timing Register and into the Configuration Register.</li>
* <li> Step 3. For low-power SDRAM, temperature-compensated self refresh (TCSR), drive strength (DS) and partial array self refresh (PASR) must be set in the Low-power Register.</li>
* <li> Step 4. A NOP command is issued to the SDR-SDRAM. Program NOP command into Mode Register, the application must
* set Mode to 1 in the Mode Register. Perform a write access to any SDR-SDRAM address to acknowledge this command.
* Now the clock which drives SDR-SDRAM device is enabled.</li>
* <li> Step 5. An all banks precharge command is issued to the SDR-SDRAM. Program all banks precharge command into Mode Register, the application must set Mode to 2 in the
* Mode Register . Perform a write access to any SDRSDRAM address to acknowledge this command.</li>
* <li> Step 6. Eight auto-refresh (CBR) cycles are provided. Program the auto refresh command (CBR) into Mode Register, the application must set Mode to 4 in the Mode Register.
* Once in the idle state, two AUTO REFRESH cycles must be performed.</li>
* <li> Step 7. A Mode Register set (MRS) cycle is issued to program the parameters of the SDRSDRAM
* devices, in particular CAS latency and burst length. </li>
* <li> Step 8. For low-power SDR-SDRAM initialization, an Extended Mode Register set (EMRS) cycle is issued to program the SDR-SDRAM parameters (TCSR, PASR, DS). The write
* address must be chosen so that BA[1] is set to 1 and BA[0] is set to 0 </li>
* <li> Step 9. The application must go into Normal Mode, setting Mode to 0 in the Mode Register and perform a write access at any location in the SDRAM to acknowledge this command.</li>
* <li> Step 10. Write the refresh rate into the count field in the DDRSDRC Refresh Timer register </li>
* </ul>
*/
/*@{*/
/*@}*/
/**
* \file
*
* Implementation of memories configuration on board.
*
*/
/*----------------------------------------------------------------------------
* Headers
*----------------------------------------------------------------------------*/
#include "board.h"
/*----------------------------------------------------------------------------
* Definiation
*----------------------------------------------------------------------------*/
#define DDRC2_MODE_NORMAL_CMD (0x0) // (HDDRSDRC2) Normal Mode
#define DDRC2_MODE_NOP_CMD (0x1) // (HDDRSDRC2) Issue a NOP Command at every access
#define DDRC2_MODE_PRCGALL_CMD (0x2) // (HDDRSDRC2) Issue a All Banks Precharge Command at every access
#define DDRC2_MODE_LMR_CMD (0x3) // (HDDRSDRC2) Issue a Load Mode Register at every access
#define DDRC2_MODE_RFSH_CMD (0x4) // (HDDRSDRC2) Issue a Refresh
#define DDRC2_MODE_EXT_LMR_CMD (0x5) // (HDDRSDRC2) Issue an Extended Load Mode Register
#define DDRC2_MODE_DEEP_CMD (0x6) // (HDDRSDRC2) Enter Deep Power Mode
#define DDRC2_MODE_Reserved (0x7) // (HDDRSDRC2) Reserved value
/*----------------------------------------------------------------------------
* Exported functions
*----------------------------------------------------------------------------*/
/**
* \brief Changes the mapping of the chip so that the remap area mirrors the
* internal ROM or the EBI CS0.
*/
void BOARD_RemapRom( void )
{
MATRIX->MATRIX_MRCR = MATRIX_MRCR_RCB0;
AXIMX->AXIMX_REMAP = 0;
}
/**
* \brief Changes the mapping of the chip so that the remap area mirrors the
* internal RAM.
*/
void BOARD_RemapRam( void )
{
MATRIX->MATRIX_MRCR = MATRIX_MRCR_RCB0;
AXIMX->AXIMX_REMAP = AXIMX_REMAP_REMAP0;
}
/**
* \brief Initialize Vdd EBI drive
* \param 0: 1.8V 1: 3.3V
*/
void BOARD_ConfigureVddMemSel( uint8_t VddMemSel )
{
( void ) VddMemSel;
}
#define DDR2_BA0(r) (1 << (25 + r))
#define DDR2_BA1(r) (1 << (26 + r))
/* -------- MPDDRC_DLL_SOR : (MPDDRC Offset: 0x74) MPDDRC DLL Slave Offset Register -------- */
// SxOFF: DLL Slave x Delay Line Offset ([x=0..1][x=0..3])
#define MPDDRC_DLL_SOR_S0_OFF_Pos 0
#define MPDDRC_DLL_SOR_S0_OFF_Msk (0x1fu << MPDDRC_DLL_SOR_S0_OFF_Pos) /**< \brief (MPDDRC_DLL_SOR) DLL Slave 0 Delay Line Offset */
#define MPDDRC_DLL_SOR_S0_OFF(value) ((MPDDRC_DLL_SOR_S0_OFF_Msk & ((value) << MPDDRC_DLL_SOR_S0_OFF_Pos)))
#define MPDDRC_DLL_SOR_S1_OFF_Pos 8
#define MPDDRC_DLL_SOR_S1_OFF_Msk (0x1fu << MPDDRC_DLL_SOR_S1_OFF_Pos) /**< \brief (MPDDRC_DLL_SOR) DLL Slave 1 Delay Line Offset */
#define MPDDRC_DLL_SOR_S1_OFF(value) ((MPDDRC_DLL_SOR_S1_OFF_Msk & ((value) << MPDDRC_DLL_SOR_S1_OFF_Pos)))
#define MPDDRC_DLL_SOR_S2_OFF_Pos 16
#define MPDDRC_DLL_SOR_S2_OFF_Msk (0x1fu << MPDDRC_DLL_SOR_S2_OFF_Pos) /**< \brief (MPDDRC_DLL_SOR) DLL Slave 2 Delay Line Offset */
#define MPDDRC_DLL_SOR_S2_OFF(value) ((MPDDRC_DLL_SOR_S2_OFF_Msk & ((value) << MPDDRC_DLL_SOR_S2_OFF_Pos)))
#define MPDDRC_DLL_SOR_S3_OFF_Pos 24
#define MPDDRC_DLL_SOR_S3_OFF_Msk (0x1fu << MPDDRC_DLL_SOR_S3_OFF_Pos) /**< \brief (MPDDRC_DLL_SOR) DLL Slave 3 Delay Line Offset */
#define MPDDRC_DLL_SOR_S3_OFF(value) ((MPDDRC_DLL_SOR_S3_OFF_Msk & ((value) << MPDDRC_DLL_SOR_S3_OFF_Pos)))
/**
* \brief Configures DDR2 (MT47H128M16RT 128MB/ MT47H64M16HR)
MT47H64M16HR : 8 Meg x 16 x 8 banks
Refresh count: 8K
Row address: A[12:0] (8K)
Column address A[9:0] (1K)
Bank address BA[2:0] a(24,25) (8)
*/
void BOARD_ConfigureDdram( uint8_t device )
{
volatile uint8_t *pDdr = (uint8_t *) DDR_CS_ADDR;
volatile uint32_t i;
volatile uint32_t cr = 0;
volatile uint32_t dummy_value;
dummy_value = 0x00000000;
/* Enable DDR2 clock x2 in PMC */
PMC->PMC_PCER1 = (1 << (ID_MPDDRC-32));
PMC->PMC_SCER |= PMC_SCER_DDRCK;
MPDDRC->MPDDRC_LPR = 0;
MPDDRC->MPDDRC_DLL_SOR = 0x101 | MPDDRC_DLL_SOR_S1_OFF(0x0) | MPDDRC_DLL_SOR_S2_OFF(0x1) | MPDDRC_DLL_SOR_S3_OFF(0x1);
MPDDRC->MPDDRC_DLL_MOR = (0xC5000000) | MPDDRC_DLL_MOR_MOFF(7) | MPDDRC_DLL_MOR_CLK90OFF(0x1F) | MPDDRC_DLL_MOR_SELOFF; // Key = 0xc5000000
dummy_value = MPDDRC->MPDDRC_IO_CALIBR;
dummy_value &= ~MPDDRC_IO_CALIBR_RDIV_Msk;
dummy_value &= ~MPDDRC_IO_CALIBR_TZQIO_Msk;
dummy_value |= MPDDRC_IO_CALIBR_RDIV_RZQ_48;
dummy_value |= MPDDRC_IO_CALIBR_TZQIO(3);
MPDDRC->MPDDRC_IO_CALIBR = dummy_value;
*(uint32_t *)0xFFFFEA80 = 0x1100;
*(uint32_t *)0xFFFFEA84 = 0x1100;
*(uint32_t *)0xFFFFEA88 = 0x1100;
*(uint32_t *)0xFFFFEA8C = 0x1100;
/* Step 1: Program the memory device type */
/* DBW = 0 (32 bits bus wide); Memory Device = 6 = DDR2-SDRAM = 0x00000006*/
MPDDRC->MPDDRC_MD = MPDDRC_MD_MD_DDR2_SDRAM;
/* Step 2: Program the features of DDR2-SDRAM device into the Timing Register.*/
if (device == DDRAM_MT47H128M16RT)
{
MPDDRC->MPDDRC_CR = MPDDRC_CR_NR_14 |
MPDDRC_CR_NC_10 |
MPDDRC_CR_CAS_4_DDR2 |
MPDDRC_CR_NB_8 |
MPDDRC_CR_DLL_RESET_DISABLED |
MPDDRC_CR_DQMS_NOT_SHARED |
MPDDRC_CR_ENRDM_OFF |
MPDDRC_CR_UNAL_SUPPORTED |
MPDDRC_CR_NDQS_DISABLED |
MPDDRC_CR_OCD(0x0);
}
if (device == DDRAM_MT47H64M16HR)
{
MPDDRC->MPDDRC_CR = MPDDRC_CR_NR_13 |
MPDDRC_CR_NC_10 |
MPDDRC_CR_CAS_3_DDR2|
MPDDRC_CR_NB_8 |
MPDDRC_CR_DLL_RESET_DISABLED |
MPDDRC_CR_DQMS_NOT_SHARED |
MPDDRC_CR_ENRDM_OFF |
MPDDRC_CR_UNAL_SUPPORTED |
MPDDRC_CR_NDQS_DISABLED |
MPDDRC_CR_OCD(0x0);
}
MPDDRC->MPDDRC_TPR0 = MPDDRC_TPR0_TRAS(6) // 6 * 7.5 = 45 ns
| MPDDRC_TPR0_TRCD(2) // 2 * 7.5 = 15 ns
| MPDDRC_TPR0_TWR(3) // 3 * 7.5 = 22.5 ns
| MPDDRC_TPR0_TRC(8) // 8 * 7.5 = 60 ns
| MPDDRC_TPR0_TRP(2) // 2 * 7.5 = 15 ns
| MPDDRC_TPR0_TRRD(2) // 2 * 7.5 = 15 ns
| MPDDRC_TPR0_TWTR(2) // 2 clock cycle
| MPDDRC_TPR0_TMRD(2); // 2 clock cycles
MPDDRC->MPDDRC_TPR1 = MPDDRC_TPR1_TRFC(0x1A) // 18 * 7.5 = 135 ns (min 127.5 ns for 1Gb DDR)
| MPDDRC_TPR1_TXSNR(0x1C) // 20 * 7.5 > 142.5ns TXSNR: Exit self refresh delay to non read command
| MPDDRC_TPR1_TXSRD(0xC8) // min 200 clock cycles, TXSRD: Exit self refresh delay to Read command
| MPDDRC_TPR1_TXP(0x2); // 2 * 7.5 = 15 ns
MPDDRC->MPDDRC_TPR2 = MPDDRC_TPR2_TXARD(8) // min 2 clock cycles
| MPDDRC_TPR2_TXARDS(7)// min 7 clock cycles
| MPDDRC_TPR2_TRPA(2) // min 18ns
| MPDDRC_TPR2_TRTP(2) // 2 * 7.5 = 15 ns (min 7.5ns)
| MPDDRC_TPR2_TFAW(7) ;
/* DDRSDRC Low-power Register */
for (i = 0; i < 13300; i++) {
asm("nop");
}
MPDDRC->MPDDRC_LPR = MPDDRC_LPR_LPCB_DISABLED |
MPDDRC_LPR_CLK_FR_DISABLED |
MPDDRC_LPR_TIMEOUT_0 |
MPDDRC_LPR_APDE_SLOW ;
/* Step 3: An NOP command is issued to the DDR2-SDRAM. Program the NOP command into
the Mode Register, the application must set MODE to 1 in the Mode Register. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_NOP_CMD;
/* Perform a write access to any DDR2-SDRAM address to acknowledge this command */
*pDdr = 0; /* Now clocks which drive DDR2-SDRAM device are enabled.*/
/* A minimum pause of 200 ¦Ìs is provided to precede any signal toggle. (6 core cycles per iteration, core is at 396MHz: min 13200 loops) */
for (i = 0; i < 13300; i++) {
asm("nop");
}
/* Step 4: An NOP command is issued to the DDR2-SDRAM */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_NOP_CMD;
/* Perform a write access to any DDR2-SDRAM address to acknowledge this command.*/
*pDdr = 0; /* Now CKE is driven high.*/
/* wait 400 ns min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 5: An all banks precharge command is issued to the DDR2-SDRAM. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_PRCGALL_CMD;
/* Perform a write access to any DDR2-SDRAM address to acknowledge this command.*/
*pDdr = 0;
/* wait 400 ns min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 6: An Extended Mode Register set (EMRS2) cycle is issued to chose between commercialor high temperature operations. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_EXT_LMR_CMD;
*((uint8_t *)(pDdr + DDR2_BA1(device))) = 0; /* The write address must be chosen so that BA[1] is set to 1 and BA[0] is set to 0. */
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 7: An Extended Mode Register set (EMRS3) cycle is issued to set all registers to 0. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_EXT_LMR_CMD;
*((uint8_t *)(pDdr + DDR2_BA1(device) + DDR2_BA0(device))) = 0; /* The write address must be chosen so that BA[1] is set to 1 and BA[0] is set to 1.*/
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 8: An Extended Mode Register set (EMRS1) cycle is issued to enable DLL. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_EXT_LMR_CMD;
*((uint8_t *)(pDdr + DDR2_BA0(device))) = 0; /* The write address must be chosen so that BA[1] is set to 0 and BA[0] is set to 1. */
/* An additional 200 cycles of clock are required for locking DLL */
for (i = 0; i < 10000; i++) {
asm("nop");
}
/* Step 9: Program DLL field into the Configuration Register.*/
cr = MPDDRC->MPDDRC_CR;
MPDDRC->MPDDRC_CR = cr | MPDDRC_CR_DLL_RESET_ENABLED;
/* Step 10: A Mode Register set (MRS) cycle is issued to reset DLL. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LMR_CMD;
*(pDdr) = 0; /* The write address must be chosen so that BA[1:0] bits are set to 0. */
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 11: An all banks precharge command is issued to the DDR2-SDRAM. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_PRCGALL_CMD;
*(pDdr) = 0; /* Perform a write access to any DDR2-SDRAM address to acknowledge this command */
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 12: Two auto-refresh (CBR) cycles are provided. Program the auto refresh command (CBR) into the Mode Register. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_RFSH_CMD;
*(pDdr) = 0; /* Perform a write access to any DDR2-SDRAM address to acknowledge this command */
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Configure 2nd CBR. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_RFSH_CMD;
*(pDdr) = 0; /* Perform a write access to any DDR2-SDRAM address to acknowledge this command */
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 13: Program DLL field into the Configuration Register to low(Disable DLL reset). */
cr = MPDDRC->MPDDRC_CR;
MPDDRC->MPDDRC_CR = cr & (~MPDDRC_CR_DLL_RESET_ENABLED);
/* Step 14: A Mode Register set (MRS) cycle is issued to program the parameters of the DDR2-SDRAM devices. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LMR_CMD;
*(pDdr) = 0; /* The write address must be chosen so that BA[1:0] are set to 0. */
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 15: Program OCD field into the Configuration Register to high (OCD calibration default). */
cr = MPDDRC->MPDDRC_CR;
MPDDRC->MPDDRC_CR = cr | MPDDRC_CR_OCD(0x07);
/* Step 16: An Extended Mode Register set (EMRS1) cycle is issued to OCD default value. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_EXT_LMR_CMD;
*((uint8_t *)(pDdr + DDR2_BA0(device))) = 0; /* The write address must be chosen so that BA[1] is set to 0 and BA[0] is set to 1.*/
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 17: Program OCD field into the Configuration Register to low (OCD calibration mode exit). */
// cr = MPDDRC->MPDDRC_CR;
// MPDDRC->MPDDRC_CR = cr & (~ MPDDRC_CR_OCD(0x07));
/* Step 18: An Extended Mode Register set (EMRS1) cycle is issued to enable OCD exit.*/
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_EXT_LMR_CMD;
*((uint8_t *)(pDdr + DDR2_BA0(device))) = 0; /* The write address must be chosen so that BA[1] is set to 0 and BA[0] is set to 1.*/
/* wait 2 cycles min */
for (i = 0; i < 100; i++) {
asm("nop");
}
/* Step 19,20: A mode Normal command is provided. Program the Normal mode into Mode Register. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_NORMAL_CMD;
*(pDdr) = 0;
/* Step 21: Write the refresh rate into the count field in the Refresh Timer register. The DDR2-SDRAM device requires a refresh every 15.625 ¦Ìs or 7.81 ¦Ìs.
With a 100MHz frequency, the refresh timer count register must to be set with (15.625 /100 MHz) = 1562 i.e. 0x061A or (7.81 /100MHz) = 781 i.e. 0x030d. */
/* For MT47H64M16HR, The refresh period is 64ms (commercial), This equates to an average
refresh rate of 7.8125¦Ìs (commercial), To ensure all rows of all banks are properly
refreshed, 8192 REFRESH commands must be issued every 64ms (commercial) */
/* ((64 x 10(^-3))/8192) x133 x (10^6) */
//MPDDRC->MPDDRC_RTR = MPDDRC_RTR_COUNT(300); /* Set Refresh timer 7.8125 us*/
MPDDRC->MPDDRC_RTR = 0x00300408;
/* OK now we are ready to work on the DDRSDR */
/* wait for end of calibration */
for (i = 0; i < 500; i++) {
asm(" nop");
}
}
/**
* \brief Configures the EBI for Sdram (LPSDR Micron MT48H8M16) access.
*/
void BOARD_ConfigureSdram( void )
{
}
/** \brief Configures the EBI for NandFlash access at 133Mhz.
*/
void BOARD_ConfigureNandFlash( uint8_t busWidth )
{
PMC_EnablePeripheral(ID_SMC);
SMC->SMC_CS_NUMBER[3].SMC_SETUP = 0
| SMC_SETUP_NWE_SETUP(1)
| SMC_SETUP_NCS_WR_SETUP(1)
| SMC_SETUP_NRD_SETUP(2)
| SMC_SETUP_NCS_RD_SETUP(1);
SMC->SMC_CS_NUMBER[3].SMC_PULSE = 0
| SMC_PULSE_NWE_PULSE(5)
| SMC_PULSE_NCS_WR_PULSE(7)
| SMC_PULSE_NRD_PULSE(5)
| SMC_PULSE_NCS_RD_PULSE(7);
SMC->SMC_CS_NUMBER[3].SMC_CYCLE = 0
| SMC_CYCLE_NWE_CYCLE(8)
| SMC_CYCLE_NRD_CYCLE(9);
SMC->SMC_CS_NUMBER[3].SMC_TIMINGS = SMC_TIMINGS_TCLR(3)
| SMC_TIMINGS_TADL(10)
| SMC_TIMINGS_TAR(3)
| SMC_TIMINGS_TRR(4)
| SMC_TIMINGS_TWB(5)
| SMC_TIMINGS_RBNSEL(3)
|(SMC_TIMINGS_NFSEL);
SMC->SMC_CS_NUMBER[3].SMC_MODE = SMC_MODE_READ_MODE |
SMC_MODE_WRITE_MODE |
((busWidth == 8 )? SMC_MODE_DBW_BIT_8 :SMC_MODE_DBW_BIT_16) |
SMC_MODE_TDF_CYCLES(1);
}
void BOARD_ConfigureNorFlash( uint8_t busWidth )
{
uint32_t dbw;
PMC_EnablePeripheral(ID_SMC);
if (busWidth == 8)
{
dbw = SMC_MODE_DBW_BIT_8;
}
else {
dbw = SMC_MODE_DBW_BIT_16;
}
/* Configure SMC, NCS0 is assigned to a norflash */
SMC->SMC_CS_NUMBER[0].SMC_SETUP = 0x00020001;
SMC->SMC_CS_NUMBER[0].SMC_PULSE = 0x0B0B0A0A;
SMC->SMC_CS_NUMBER[0].SMC_CYCLE = 0x000E000B;
SMC->SMC_CS_NUMBER[0].SMC_TIMINGS = 0x00000000;
SMC->SMC_CS_NUMBER[0].SMC_MODE = SMC_MODE_WRITE_MODE
| SMC_MODE_READ_MODE
| dbw
| SMC_MODE_EXNW_MODE_DISABLED
| SMC_MODE_TDF_CYCLES(1);
}
// -----------------------------------------------------------------------------
// Function Name : LPDDR2_Initialize
// Object :
// -----------------------------------------------------------------------------
void BOARD_ConfigureLpDdram( void)
{
volatile uint32_t i;
volatile uint32_t dummy_value;
PMC->PMC_PCER1 = (1 << (ID_MPDDRC-32));
PMC->PMC_SCER |= PMC_SCER_DDRCK;
/* -------------------- Additional DDR2 setting ------------------------ */
MPDDRC->MPDDRC_DLL_SOR = MPDDRC_DLL_SOR_S0_OFF(0x4) | MPDDRC_DLL_SOR_S1_OFF(0x3) | MPDDRC_DLL_SOR_S2_OFF(0x4) | MPDDRC_DLL_SOR_S3_OFF(0x4); // design recommendation
MPDDRC->MPDDRC_DLL_MOR = (0xC5000000) | MPDDRC_DLL_MOR_MOFF(7) | MPDDRC_DLL_MOR_CLK90OFF(0x1F) | MPDDRC_DLL_MOR_SELOFF; // Key = 0xc5000000
dummy_value = MPDDRC->MPDDRC_IO_CALIBR;
dummy_value &= ~MPDDRC_IO_CALIBR_RDIV_Msk;
dummy_value &= ~MPDDRC_IO_CALIBR_TZQIO_Msk;
dummy_value |= (0x4 << 0);
dummy_value |= MPDDRC_IO_CALIBR_TZQIO(3);
MPDDRC->MPDDRC_IO_CALIBR = dummy_value;
/* DDRSDRC High Speed Register (MPDDRC_HS) : hidden option -> calibration during autorefresh */
*(uint32_t *)0xFFFFEA24 |= (1 << 5);
/* SFR_DDRCFG DDR Configuration Force DDR_DQ and DDR_DQS input buffer always on */
*(uint32_t *)0xF0038004 |= (0x3 << 16);
/* Initialization sequence STEP 1
Program the memory device type into the Memory Device Register */
/* Memory device = LPDDR2 => MPDDRC_MD_MD_LPDDR2_SDRAM
Data bus width = 32 bits => 0x0 (The system is in 64 bits, thus memory data bus width should be 32 bits) */
MPDDRC->MPDDRC_MD = MPDDRC_MD_MD_LPDDR2_SDRAM;
/* Initialization sequence STEP 2
Program the features of Low-power DDR2-SDRAM device into the Timing Register
(asynchronous timing, trc, tras, etc.) and into the Configuration Register (number of
columns, rows, banks, CAS latency and output drive strength) (see Section 8.3 on
page 35, Section 8.4 on page 39 and Section 80.5 on page 41). */
MPDDRC->MPDDRC_CR = MPDDRC_CR_NC_10 |
MPDDRC_CR_NR_14 |
MPDDRC_CR_CAS_5_LPDDR2 |
MPDDRC_CR_NB_8 |
MPDDRC_CR_UNAL_SUPPORTED |
MPDDRC_CR_ENRDM_ON;
MPDDRC->MPDDRC_LPDDR2_LPR |= MPDDRC_LPDDR2_LPR_DS(0x3);
MPDDRC->MPDDRC_TPR0 = MPDDRC_TPR0_TRAS (6) | // 03 - TRAS tRAS Row active time
MPDDRC_TPR0_TRCD (2) | // 04 - TRC tRCD RAS-to-CAS delay
MPDDRC_TPR0_TWR (3) | // 05 - TWR tWR WRITE recovery time
MPDDRC_TPR0_TRC (8) | // 06 - TRC tRC ACTI-to-ACTIVT command period
MPDDRC_TPR0_TRP (3) | // 07 - TRP tRPpb Row precharge time
MPDDRC_TPR0_TRRD (2) | // 08 - TRRD tRRD Active bank a to active bank b
MPDDRC_TPR0_TWTR (2) | // 09 - TWTR-tWTR Internal WRITE-to-READcommand delay
MPDDRC_TPR0_TMRD (3); // 10 - TMRD-tMRD
MPDDRC->MPDDRC_TPR1 = MPDDRC_TPR1_TRFC (17) | // 11 - TRFC tRFCab Refresh cycle time
MPDDRC_TPR1_TXSNR (18) | // 12 - TXSNR SELF REFRESH exit to next valid delay
MPDDRC_TPR1_TXSRD (14) | // 13 - TXSRD Exit Self Refresh
MPDDRC_TPR1_TXP (2); // 14 - TXP-tXP Exit power-down
MPDDRC->MPDDRC_TPR2 = MPDDRC_TPR2_TXARD (1) | // 15 - TXARD-txARD
MPDDRC_TPR2_TXARDS (1) | // 16 - TXARDS-txARDs
MPDDRC_TPR2_TRPA (3) | // 17 - TRPA-tRPpab Row precharge time (all banks)
MPDDRC_TPR2_TRTP (2) | // 18 - TRTP-tRTP
MPDDRC_TPR2_TFAW (8); // 19 - TFAW-tFAW
/* Initialization sequence STEP 3
An NOP command is issued to the Low-power DDR2-SDRAM. Program the NOP
command into the Mode Register, the application must set the MODE (MDDRC Command
Mode) field to 1 in the Mode Register (see Section 8.1 on page 32). Perform a
write access to any Low-power DDR2-SDRAM address to acknowledge this command.
Now, clocks which drive Low-power DDR2-SDRAM devices are enabled.
A minimum pause of 100 ns must be observed to precede any signal toggle. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_NOP_CMD; // NOP to ENABLE CLOCK output
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 100; i++) {
asm(" nop");
} // Delay loop (at least 100 ns)
/* Initialization sequence STEP 4
An NOP command is issued to the Low-power DDR2-SDRAM. Program the NOP
command into the Mode Register, the application must set MODE to 1 in the Mode
Register (see Section 8.1 on page 32). Perform a write access to any Low-power
DDR2-SDRAM address to acknowledge this command. Now, CKE is driven high.
A minimum pause of 200 us must be satisfied before Reset Command.
*/
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_NOP_CMD; // NOP to drive CKE high
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 10000; i++) {
asm(" nop");
} // Delay loop (at least 200 us)
/* Initialization sequence STEP 5
A reset command is issued to the Low-power DDR2-SDRAM. Program
LPDDR2_CMD in the MODE (MDDRC Command Mode) and MRS (Mode Register
Select LPDDR2) field of the Mode Register, the application must set MODE to 7 and
MRS to 63. (see Section 8.1 on page 32). Perform a write access to any Low-power
DDR2-SDRAM address to acknowledge this command. Now, the reset command is issued.
A minimum pause of 1us must be satisfied before any commands. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MRS(0x3F) | MPDDRC_MR_MODE_LPDDR2_CMD;
*(unsigned int *)DDR_CS_ADDR = 0x00000000;
for (i = 0; i < 500; i++) { asm(" nop"); }
/* Initialization sequence STEP 6
A Mode Register Read command is issued to the Low-power DDR2-SDRAM. Program
LPPDR2_CMD in the MODE and MRS field of the Mode Register, the
application must set MODE to 7 and must set MRS field to 0. (see Section 8.1 on
page 32). Perform a write access to any Low-power DDR2-SDRAM address to
acknowledge this command. Now, the Mode Register Read command is issued.
A minimum pause of 10 us must be satisfied before any commands. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LPDDR2_CMD | MPDDRC_MR_MRS(0x00);
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 5000; i++) {asm(" nop"); }
/* Initialization sequence STEP 7
A calibration command is issued to the Low-power DDR2-SDRAM. Program the type
of calibration into the Configuration Register, ZQ field, RESET value (see Section 8.3
”MPDDRC Configuration Register?on page 37). In the Mode Register, program the
MODE field to LPDDR2_CMD value, and the MRS field; the application must set
MODE to 7 and MRS to 10 (see Section 8.1 LPDDRC Mode Register?on page 34).
Perform a write access to any Low-power DDR2-SDRAM address to acknowledge
this command. Now, the ZQ Calibration command is issued. Program the type of calibration
into the Configuration Register, ZQ field */
MPDDRC->MPDDRC_CR &= ~MPDDRC_CR_ZQ_Msk;
MPDDRC->MPDDRC_CR |= MPDDRC_CR_ZQ_RESET;
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LPDDR2_CMD | MPDDRC_MR_MRS(0x0A);
// Mode Register Read command. MODE = 0x7 and MRS = 0x0A
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 500; i++) {asm(" nop"); }
MPDDRC->MPDDRC_CR &= ~MPDDRC_CR_ZQ_Msk;
MPDDRC->MPDDRC_CR |= MPDDRC_CR_ZQ_SHORT;
/* Initialization sequence STEP 8
A Mode Register Write command is issued to the Low-power DDR2-SDRAM. Program
LPPDR2_CMD in the MODE and MRS field in the Mode Register, the
application must set MODE to 7 and must set MRS field to 0.5 (see Section 8.1 on
page 32). The Mode Register Write command cycle is issued to program the parameters
of the Low-power DDR2-SDRAM devices, in particular burst length. Perform a
write access to any Low-power DDR2-SDRAM address to acknowledge this command.
Now, the Mode Register Write command is issued. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LPDDR2_CMD | MPDDRC_MR_MRS(0x01);
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 500; i++) {
asm(" nop");
}
/* Initialization sequence STEP 9
Mode Register Write Command is issued to the Low-power DDR2-SDRAM. Program
LPPDR2_CMD in the MODE and MRS field in the Mode Register, the
application must set MODE to 7 and must set MRS field to 2. (see Section 8.1 on
page 32). The Mode Register Write command cycle is issued to program the parameters
of the Low-power DDR2-SDRAM devices, in particular CAS latency. Perform a
write access to any Low-power DDR2-SDRAM address to acknowledge this command.
Now, the Mode Register Write command is issued. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LPDDR2_CMD | MPDDRC_MR_MRS(0x02);
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 500; i++) {
asm(" nop");
}
/* Initialization sequence STEP 10
A Mode Register Write Command is issued to the Low-power DDR2-SDRAM. Program
LPPDR2_CMD in the MODE and MRS field of the Mode Register, the
application must set MODE to 7 and must set MRS field to 3. (see Section 8.1 on
page 32). The Mode Register Write command cycle is issued to program the parameters
of the Low-power DDR2-SDRAM devices, in particular Drive Strength and Slew
Rate. Perform a write access to any Low-power DDR2-SDRAM address to acknowledge
this command. Now, the Mode Register Write command is issued. */
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LPDDR2_CMD | MPDDRC_MR_MRS(0x03); //0x00000307;
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 500; i++) {
asm(" nop");
}
/* Initialization sequence STEP 11
A Mode Register Write Command is issued to the Low-power DDR2-SDRAM. Program
LPPDR2_CMD in the MODE and MRS field of the Mode Register, the
application must set MODE to 7 and must set MRS field to 16. (see Section 8.1 on
page 32). Mode Register Write command cycle is issued to program the parameters
of the Low-power DDR2-SDRAM devices, in particular Partial Array Self Refresh
(PASR). Perform a write access to any Low-power DDR2-SDRAM address to
acknowledge this command. Now, the Mode Register Write command is issued.*/
MPDDRC->MPDDRC_MR = MPDDRC_MR_MODE_LPDDR2_CMD | MPDDRC_MR_MRS(0x10);// 0x00001007;
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Access to memory
for (i = 0; i < 500; i++) {
asm(" nop");
}
/* Initialization sequence STEP 12
Write the refresh rate into the COUNT field in the Refresh Timer register (see page
33). (Refresh rate = delay between refresh cycles). The Low-power DDR2-SDRAM
device requires a refresh every 7.81 ìs. With a 100 MHz frequency, the refresh timer
count register must to be set with (7.81/100 MHz) = 781 i.e. 0x030d. */
MPDDRC->MPDDRC_RTR &= ~MPDDRC_RTR_COUNT_Msk;
MPDDRC->MPDDRC_RTR |= MPDDRC_RTR_COUNT(1030);
MPDDRC->MPDDRC_MR = 0x00000000; // Set Normal mode
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Perform
for (i = 0; i < 500; i++) {
asm(" nop");
}
/* Launch short ZQ calibration */
MPDDRC->MPDDRC_CR &= ~MPDDRC_CR_ZQ_Msk; // Enable short calibration in the CR
MPDDRC->MPDDRC_CR |= MPDDRC_CR_ZQ_SHORT;
MPDDRC->MPDDRC_CR |= MPDDRC_CR_DLL_RESET_ENABLED;
*(unsigned int *)DDR_CS_ADDR = 0x00000000; // Perform
/* Calculate ZQS: search for tZQCS in the memory datasheet => tZQCS = 180 ns*/
MPDDRC->MPDDRC_LPDDR2_TIM_CAL = MPDDRC_LPDDR2_TIM_CAL_ZQCS(12);
}