56248d9da8
make use of it where possible. This primarily brings in support for newer hardware, and FreeBSD is not yet able to support the abundance of IRQs on new hardware and many features in the Ethernet driver. Because of the changes to IRQs in the Simple Executive, we have to maintain our own list of Octeon IRQs now, which probably can be pared-down and be specific to the CIU interrupt unit soon, and when other interrupt mechanisms are added they can maintain their own definitions. Remove unmasking of interrupts from within the UART device now that the function used is no longer present in the Simple Executive. The unmasking seems to have been gratuitous as this is more properly handled by the buses above the UART device, and seems to work on that basis.
8422 lines
532 KiB
C
8422 lines
532 KiB
C
/***********************license start***************
|
|
* Copyright (c) 2003-2012 Cavium Inc. (support@cavium.com). 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 following disclaimer.
|
|
*
|
|
* * 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.
|
|
|
|
* * Neither the name of Cavium Inc. 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, including technical data, may be subject to U.S. export control
|
|
* laws, including the U.S. Export Administration Act and its associated
|
|
* regulations, and may be subject to export or import regulations in other
|
|
* countries.
|
|
|
|
* TO THE MAXIMUM EXTENT PERMITTED BY LAW, THE SOFTWARE IS PROVIDED "AS IS"
|
|
* AND WITH ALL FAULTS AND CAVIUM INC. MAKES NO PROMISES, REPRESENTATIONS OR
|
|
* WARRANTIES, EITHER EXPRESS, IMPLIED, STATUTORY, OR OTHERWISE, WITH RESPECT TO
|
|
* THE SOFTWARE, INCLUDING ITS CONDITION, ITS CONFORMITY TO ANY REPRESENTATION OR
|
|
* DESCRIPTION, OR THE EXISTENCE OF ANY LATENT OR PATENT DEFECTS, AND CAVIUM
|
|
* SPECIFICALLY DISCLAIMS ALL IMPLIED (IF ANY) WARRANTIES OF TITLE,
|
|
* MERCHANTABILITY, NONINFRINGEMENT, FITNESS FOR A PARTICULAR PURPOSE, LACK OF
|
|
* VIRUSES, ACCURACY OR COMPLETENESS, QUIET ENJOYMENT, QUIET POSSESSION OR
|
|
* CORRESPONDENCE TO DESCRIPTION. THE ENTIRE RISK ARISING OUT OF USE OR
|
|
* PERFORMANCE OF THE SOFTWARE LIES WITH YOU.
|
|
***********************license end**************************************/
|
|
|
|
|
|
/**
|
|
* cvmx-lmcx-defs.h
|
|
*
|
|
* Configuration and status register (CSR) type definitions for
|
|
* Octeon lmcx.
|
|
*
|
|
* This file is auto generated. Do not edit.
|
|
*
|
|
* <hr>$Revision$<hr>
|
|
*
|
|
*/
|
|
#ifndef __CVMX_LMCX_DEFS_H__
|
|
#define __CVMX_LMCX_DEFS_H__
|
|
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_BIST_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_BIST_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000F0ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_BIST_CTL(block_id) (CVMX_ADD_IO_SEG(0x00011800880000F0ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_BIST_RESULT(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_BIST_RESULT(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000F8ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_BIST_RESULT(block_id) (CVMX_ADD_IO_SEG(0x00011800880000F8ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CHAR_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CHAR_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000220ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CHAR_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000220ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CHAR_MASK0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CHAR_MASK0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000228ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CHAR_MASK0(block_id) (CVMX_ADD_IO_SEG(0x0001180088000228ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CHAR_MASK1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CHAR_MASK1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000230ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CHAR_MASK1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000230ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CHAR_MASK2(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CHAR_MASK2(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000238ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CHAR_MASK2(block_id) (CVMX_ADD_IO_SEG(0x0001180088000238ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CHAR_MASK3(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CHAR_MASK3(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000240ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CHAR_MASK3(block_id) (CVMX_ADD_IO_SEG(0x0001180088000240ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CHAR_MASK4(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CHAR_MASK4(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000318ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CHAR_MASK4(block_id) (CVMX_ADD_IO_SEG(0x0001180088000318ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_COMP_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_COMP_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000028ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_COMP_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000028ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_COMP_CTL2(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_COMP_CTL2(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001B8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_COMP_CTL2(block_id) (CVMX_ADD_IO_SEG(0x00011800880001B8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CONFIG(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CONFIG(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000188ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CONFIG(block_id) (CVMX_ADD_IO_SEG(0x0001180088000188ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CONTROL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CONTROL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000190ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CONTROL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000190ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000010ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000010ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_CTL1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_CTL1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000090ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_CTL1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000090ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DCLK_CNT(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DCLK_CNT(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001E0ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DCLK_CNT(block_id) (CVMX_ADD_IO_SEG(0x00011800880001E0ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DCLK_CNT_HI(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DCLK_CNT_HI(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000070ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DCLK_CNT_HI(block_id) (CVMX_ADD_IO_SEG(0x0001180088000070ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DCLK_CNT_LO(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DCLK_CNT_LO(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000068ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DCLK_CNT_LO(block_id) (CVMX_ADD_IO_SEG(0x0001180088000068ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DCLK_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_DCLK_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000B8ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DCLK_CTL(block_id) (CVMX_ADD_IO_SEG(0x00011800880000B8ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DDR2_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DDR2_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000018ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DDR2_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000018ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DDR_PLL_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DDR_PLL_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000258ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DDR_PLL_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000258ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DELAY_CFG(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DELAY_CFG(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000088ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DELAY_CFG(block_id) (CVMX_ADD_IO_SEG(0x0001180088000088ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DIMMX_PARAMS(unsigned long offset, unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && (((offset <= 1)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && (((offset <= 1)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && (((offset <= 1)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && (((offset <= 1)) && ((block_id <= 3)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && (((offset <= 1)) && ((block_id == 0))))))
|
|
cvmx_warn("CVMX_LMCX_DIMMX_PARAMS(%lu,%lu) is invalid on this chip\n", offset, block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000270ull) + (((offset) & 1) + ((block_id) & 3) * 0x200000ull) * 8;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DIMMX_PARAMS(offset, block_id) (CVMX_ADD_IO_SEG(0x0001180088000270ull) + (((offset) & 1) + ((block_id) & 3) * 0x200000ull) * 8)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DIMM_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DIMM_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000310ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DIMM_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000310ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DLL_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_DLL_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000C0ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DLL_CTL(block_id) (CVMX_ADD_IO_SEG(0x00011800880000C0ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DLL_CTL2(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DLL_CTL2(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001C8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DLL_CTL2(block_id) (CVMX_ADD_IO_SEG(0x00011800880001C8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_DLL_CTL3(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_DLL_CTL3(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000218ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_DLL_CTL3(block_id) (CVMX_ADD_IO_SEG(0x0001180088000218ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
static inline uint64_t CVMX_LMCX_DUAL_MEMCFG(unsigned long block_id)
|
|
{
|
|
switch(cvmx_get_octeon_family()) {
|
|
case OCTEON_CNF71XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN50XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN58XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN66XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN52XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN61XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN63XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id == 0))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000098ull) + ((block_id) & 0) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN56XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 1))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000098ull) + ((block_id) & 1) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN68XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 3))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000098ull) + ((block_id) & 3) * 0x1000000ull;
|
|
break;
|
|
}
|
|
cvmx_warn("CVMX_LMCX_DUAL_MEMCFG (block_id = %lu) not supported on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000098ull) + ((block_id) & 0) * 0x60000000ull;
|
|
}
|
|
static inline uint64_t CVMX_LMCX_ECC_SYND(unsigned long block_id)
|
|
{
|
|
switch(cvmx_get_octeon_family()) {
|
|
case OCTEON_CN30XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN50XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN38XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN31XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN58XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN66XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN52XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN61XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CNF71XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN63XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id == 0))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000038ull) + ((block_id) & 0) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN56XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 1))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000038ull) + ((block_id) & 1) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN68XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 3))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000038ull) + ((block_id) & 3) * 0x1000000ull;
|
|
break;
|
|
}
|
|
cvmx_warn("CVMX_LMCX_ECC_SYND (block_id = %lu) not supported on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000038ull) + ((block_id) & 0) * 0x60000000ull;
|
|
}
|
|
static inline uint64_t CVMX_LMCX_FADR(unsigned long block_id)
|
|
{
|
|
switch(cvmx_get_octeon_family()) {
|
|
case OCTEON_CN30XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN50XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN38XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN31XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN58XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN66XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN52XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN61XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CNF71XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN63XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id == 0))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000020ull) + ((block_id) & 0) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN56XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 1))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000020ull) + ((block_id) & 1) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN68XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 3))
|
|
return CVMX_ADD_IO_SEG(0x0001180088000020ull) + ((block_id) & 3) * 0x1000000ull;
|
|
break;
|
|
}
|
|
cvmx_warn("CVMX_LMCX_FADR (block_id = %lu) not supported on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000020ull) + ((block_id) & 0) * 0x60000000ull;
|
|
}
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_IFB_CNT(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_IFB_CNT(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001D0ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_IFB_CNT(block_id) (CVMX_ADD_IO_SEG(0x00011800880001D0ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_IFB_CNT_HI(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_IFB_CNT_HI(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000050ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_IFB_CNT_HI(block_id) (CVMX_ADD_IO_SEG(0x0001180088000050ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_IFB_CNT_LO(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_IFB_CNT_LO(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000048ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_IFB_CNT_LO(block_id) (CVMX_ADD_IO_SEG(0x0001180088000048ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_INT(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_INT(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001F0ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_INT(block_id) (CVMX_ADD_IO_SEG(0x00011800880001F0ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_INT_EN(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_INT_EN(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001E8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_INT_EN(block_id) (CVMX_ADD_IO_SEG(0x00011800880001E8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_MEM_CFG0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_MEM_CFG0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000000ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_MEM_CFG0(block_id) (CVMX_ADD_IO_SEG(0x0001180088000000ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_MEM_CFG1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_MEM_CFG1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000008ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_MEM_CFG1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000008ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_MODEREG_PARAMS0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_MODEREG_PARAMS0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001A8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_MODEREG_PARAMS0(block_id) (CVMX_ADD_IO_SEG(0x00011800880001A8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_MODEREG_PARAMS1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_MODEREG_PARAMS1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000260ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_MODEREG_PARAMS1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000260ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
static inline uint64_t CVMX_LMCX_NXM(unsigned long block_id)
|
|
{
|
|
switch(cvmx_get_octeon_family()) {
|
|
case OCTEON_CNF71XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN61XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN66XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN52XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN58XX & OCTEON_FAMILY_MASK:
|
|
case OCTEON_CN63XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id == 0))
|
|
return CVMX_ADD_IO_SEG(0x00011800880000C8ull) + ((block_id) & 0) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN56XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 1))
|
|
return CVMX_ADD_IO_SEG(0x00011800880000C8ull) + ((block_id) & 1) * 0x60000000ull;
|
|
break;
|
|
case OCTEON_CN68XX & OCTEON_FAMILY_MASK:
|
|
if ((block_id <= 3))
|
|
return CVMX_ADD_IO_SEG(0x00011800880000C8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
break;
|
|
}
|
|
cvmx_warn("CVMX_LMCX_NXM (block_id = %lu) not supported on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000C8ull) + ((block_id) & 0) * 0x60000000ull;
|
|
}
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_OPS_CNT(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_OPS_CNT(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001D8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_OPS_CNT(block_id) (CVMX_ADD_IO_SEG(0x00011800880001D8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_OPS_CNT_HI(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_OPS_CNT_HI(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000060ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_OPS_CNT_HI(block_id) (CVMX_ADD_IO_SEG(0x0001180088000060ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_OPS_CNT_LO(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_OPS_CNT_LO(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000058ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_OPS_CNT_LO(block_id) (CVMX_ADD_IO_SEG(0x0001180088000058ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_PHY_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_PHY_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000210ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_PHY_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000210ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_PLL_BWCTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_PLL_BWCTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000040ull);
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_PLL_BWCTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000040ull))
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_PLL_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_PLL_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000A8ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_PLL_CTL(block_id) (CVMX_ADD_IO_SEG(0x00011800880000A8ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_PLL_STATUS(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_PLL_STATUS(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000B0ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_PLL_STATUS(block_id) (CVMX_ADD_IO_SEG(0x00011800880000B0ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_READ_LEVEL_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_READ_LEVEL_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000140ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_READ_LEVEL_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000140ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_READ_LEVEL_DBG(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_READ_LEVEL_DBG(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000148ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_READ_LEVEL_DBG(block_id) (CVMX_ADD_IO_SEG(0x0001180088000148ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_READ_LEVEL_RANKX(unsigned long offset, unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && (((offset <= 3)) && ((block_id <= 1))))))
|
|
cvmx_warn("CVMX_LMCX_READ_LEVEL_RANKX(%lu,%lu) is invalid on this chip\n", offset, block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000100ull) + (((offset) & 3) + ((block_id) & 1) * 0xC000000ull) * 8;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_READ_LEVEL_RANKX(offset, block_id) (CVMX_ADD_IO_SEG(0x0001180088000100ull) + (((offset) & 3) + ((block_id) & 1) * 0xC000000ull) * 8)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RESET_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_RESET_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000180ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RESET_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000180ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RLEVEL_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_RLEVEL_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880002A0ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RLEVEL_CTL(block_id) (CVMX_ADD_IO_SEG(0x00011800880002A0ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RLEVEL_DBG(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_RLEVEL_DBG(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880002A8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RLEVEL_DBG(block_id) (CVMX_ADD_IO_SEG(0x00011800880002A8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RLEVEL_RANKX(unsigned long offset, unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && (((offset <= 3)) && ((block_id <= 3)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && (((offset <= 3)) && ((block_id == 0))))))
|
|
cvmx_warn("CVMX_LMCX_RLEVEL_RANKX(%lu,%lu) is invalid on this chip\n", offset, block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000280ull) + (((offset) & 3) + ((block_id) & 3) * 0x200000ull) * 8;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RLEVEL_RANKX(offset, block_id) (CVMX_ADD_IO_SEG(0x0001180088000280ull) + (((offset) & 3) + ((block_id) & 3) * 0x200000ull) * 8)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RODT_COMP_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_RODT_COMP_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880000A0ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RODT_COMP_CTL(block_id) (CVMX_ADD_IO_SEG(0x00011800880000A0ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RODT_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_RODT_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000078ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RODT_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000078ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_RODT_MASK(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_RODT_MASK(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000268ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_RODT_MASK(block_id) (CVMX_ADD_IO_SEG(0x0001180088000268ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_SCRAMBLED_FADR(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_SCRAMBLED_FADR(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000330ull);
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_SCRAMBLED_FADR(block_id) (CVMX_ADD_IO_SEG(0x0001180088000330ull))
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_SCRAMBLE_CFG0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_SCRAMBLE_CFG0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000320ull);
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_SCRAMBLE_CFG0(block_id) (CVMX_ADD_IO_SEG(0x0001180088000320ull))
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_SCRAMBLE_CFG1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_SCRAMBLE_CFG1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000328ull);
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_SCRAMBLE_CFG1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000328ull))
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_SLOT_CTL0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_SLOT_CTL0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001F8ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_SLOT_CTL0(block_id) (CVMX_ADD_IO_SEG(0x00011800880001F8ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_SLOT_CTL1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_SLOT_CTL1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000200ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_SLOT_CTL1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000200ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_SLOT_CTL2(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_SLOT_CTL2(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000208ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_SLOT_CTL2(block_id) (CVMX_ADD_IO_SEG(0x0001180088000208ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_TIMING_PARAMS0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_TIMING_PARAMS0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000198ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_TIMING_PARAMS0(block_id) (CVMX_ADD_IO_SEG(0x0001180088000198ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_TIMING_PARAMS1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_TIMING_PARAMS1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001A0ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_TIMING_PARAMS1(block_id) (CVMX_ADD_IO_SEG(0x00011800880001A0ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_TRO_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_TRO_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000248ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_TRO_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000248ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_TRO_STAT(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_TRO_STAT(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000250ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_TRO_STAT(block_id) (CVMX_ADD_IO_SEG(0x0001180088000250ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_WLEVEL_CTL(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_WLEVEL_CTL(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000300ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_WLEVEL_CTL(block_id) (CVMX_ADD_IO_SEG(0x0001180088000300ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_WLEVEL_DBG(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_WLEVEL_DBG(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000308ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_WLEVEL_DBG(block_id) (CVMX_ADD_IO_SEG(0x0001180088000308ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_WLEVEL_RANKX(unsigned long offset, unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && (((offset <= 3)) && ((block_id == 0)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && (((offset <= 3)) && ((block_id <= 3)))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && (((offset <= 3)) && ((block_id == 0))))))
|
|
cvmx_warn("CVMX_LMCX_WLEVEL_RANKX(%lu,%lu) is invalid on this chip\n", offset, block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880002B0ull) + (((offset) & 3) + ((block_id) & 3) * 0x200000ull) * 8;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_WLEVEL_RANKX(offset, block_id) (CVMX_ADD_IO_SEG(0x00011800880002B0ull) + (((offset) & 3) + ((block_id) & 3) * 0x200000ull) * 8)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_WODT_CTL0(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN38XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN50XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN58XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_WODT_CTL0(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000030ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_WODT_CTL0(block_id) (CVMX_ADD_IO_SEG(0x0001180088000030ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_WODT_CTL1(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN30XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN31XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN52XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN56XX) && ((block_id <= 1)))))
|
|
cvmx_warn("CVMX_LMCX_WODT_CTL1(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x0001180088000080ull) + ((block_id) & 1) * 0x60000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_WODT_CTL1(block_id) (CVMX_ADD_IO_SEG(0x0001180088000080ull) + ((block_id) & 1) * 0x60000000ull)
|
|
#endif
|
|
#if CVMX_ENABLE_CSR_ADDRESS_CHECKING
|
|
static inline uint64_t CVMX_LMCX_WODT_MASK(unsigned long block_id)
|
|
{
|
|
if (!(
|
|
(OCTEON_IS_MODEL(OCTEON_CN61XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN63XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN66XX) && ((block_id == 0))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CN68XX) && ((block_id <= 3))) ||
|
|
(OCTEON_IS_MODEL(OCTEON_CNF71XX) && ((block_id == 0)))))
|
|
cvmx_warn("CVMX_LMCX_WODT_MASK(%lu) is invalid on this chip\n", block_id);
|
|
return CVMX_ADD_IO_SEG(0x00011800880001B0ull) + ((block_id) & 3) * 0x1000000ull;
|
|
}
|
|
#else
|
|
#define CVMX_LMCX_WODT_MASK(block_id) (CVMX_ADD_IO_SEG(0x00011800880001B0ull) + ((block_id) & 3) * 0x1000000ull)
|
|
#endif
|
|
|
|
/**
|
|
* cvmx_lmc#_bist_ctl
|
|
*
|
|
* Notes:
|
|
* This controls BiST only for the memories that operate on DCLK. The normal, chip-wide BiST flow
|
|
* controls BiST for the memories that operate on ECLK.
|
|
*/
|
|
union cvmx_lmcx_bist_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_bist_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_1_63 : 63;
|
|
uint64_t start : 1; /**< A 0->1 transition causes BiST to run. */
|
|
#else
|
|
uint64_t start : 1;
|
|
uint64_t reserved_1_63 : 63;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_bist_ctl_s cn50xx;
|
|
struct cvmx_lmcx_bist_ctl_s cn52xx;
|
|
struct cvmx_lmcx_bist_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_bist_ctl_s cn56xx;
|
|
struct cvmx_lmcx_bist_ctl_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_bist_ctl cvmx_lmcx_bist_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_bist_result
|
|
*
|
|
* Notes:
|
|
* Access to the internal BiST results
|
|
* Each bit is the BiST result of an individual memory (per bit, 0=pass and 1=fail).
|
|
*/
|
|
union cvmx_lmcx_bist_result {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_bist_result_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_11_63 : 53;
|
|
uint64_t csrd2e : 1; /**< BiST result of CSRD2E memory (0=pass, !0=fail) */
|
|
uint64_t csre2d : 1; /**< BiST result of CSRE2D memory (0=pass, !0=fail) */
|
|
uint64_t mwf : 1; /**< BiST result of MWF memories (0=pass, !0=fail) */
|
|
uint64_t mwd : 3; /**< BiST result of MWD memories (0=pass, !0=fail) */
|
|
uint64_t mwc : 1; /**< BiST result of MWC memories (0=pass, !0=fail) */
|
|
uint64_t mrf : 1; /**< BiST result of MRF memories (0=pass, !0=fail) */
|
|
uint64_t mrd : 3; /**< BiST result of MRD memories (0=pass, !0=fail) */
|
|
#else
|
|
uint64_t mrd : 3;
|
|
uint64_t mrf : 1;
|
|
uint64_t mwc : 1;
|
|
uint64_t mwd : 3;
|
|
uint64_t mwf : 1;
|
|
uint64_t csre2d : 1;
|
|
uint64_t csrd2e : 1;
|
|
uint64_t reserved_11_63 : 53;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_bist_result_cn50xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_9_63 : 55;
|
|
uint64_t mwf : 1; /**< BiST result of MWF memories (0=pass, !0=fail) */
|
|
uint64_t mwd : 3; /**< BiST result of MWD memories (0=pass, !0=fail) */
|
|
uint64_t mwc : 1; /**< BiST result of MWC memories (0=pass, !0=fail) */
|
|
uint64_t mrf : 1; /**< BiST result of MRF memories (0=pass, !0=fail) */
|
|
uint64_t mrd : 3; /**< BiST result of MRD memories (0=pass, !0=fail) */
|
|
#else
|
|
uint64_t mrd : 3;
|
|
uint64_t mrf : 1;
|
|
uint64_t mwc : 1;
|
|
uint64_t mwd : 3;
|
|
uint64_t mwf : 1;
|
|
uint64_t reserved_9_63 : 55;
|
|
#endif
|
|
} cn50xx;
|
|
struct cvmx_lmcx_bist_result_s cn52xx;
|
|
struct cvmx_lmcx_bist_result_s cn52xxp1;
|
|
struct cvmx_lmcx_bist_result_s cn56xx;
|
|
struct cvmx_lmcx_bist_result_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_bist_result cvmx_lmcx_bist_result_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_char_ctl
|
|
*
|
|
* LMC_CHAR_CTL = LMC Characterization Control
|
|
* This register is an assortment of various control fields needed to charecterize the DDR3 interface
|
|
*/
|
|
union cvmx_lmcx_char_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_char_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_44_63 : 20;
|
|
uint64_t dr : 1; /**< Pattern at Data Rate (not Clock Rate) */
|
|
uint64_t skew_on : 1; /**< Skew adjacent bits */
|
|
uint64_t en : 1; /**< Enable characterization */
|
|
uint64_t sel : 1; /**< Pattern select
|
|
0 = PRBS
|
|
1 = Programmable pattern */
|
|
uint64_t prog : 8; /**< Programmable pattern */
|
|
uint64_t prbs : 32; /**< PRBS Polynomial */
|
|
#else
|
|
uint64_t prbs : 32;
|
|
uint64_t prog : 8;
|
|
uint64_t sel : 1;
|
|
uint64_t en : 1;
|
|
uint64_t skew_on : 1;
|
|
uint64_t dr : 1;
|
|
uint64_t reserved_44_63 : 20;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_char_ctl_s cn61xx;
|
|
struct cvmx_lmcx_char_ctl_cn63xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_42_63 : 22;
|
|
uint64_t en : 1; /**< Enable characterization */
|
|
uint64_t sel : 1; /**< Pattern select
|
|
0 = PRBS
|
|
1 = Programmable pattern */
|
|
uint64_t prog : 8; /**< Programmable pattern */
|
|
uint64_t prbs : 32; /**< PRBS Polynomial */
|
|
#else
|
|
uint64_t prbs : 32;
|
|
uint64_t prog : 8;
|
|
uint64_t sel : 1;
|
|
uint64_t en : 1;
|
|
uint64_t reserved_42_63 : 22;
|
|
#endif
|
|
} cn63xx;
|
|
struct cvmx_lmcx_char_ctl_cn63xx cn63xxp1;
|
|
struct cvmx_lmcx_char_ctl_s cn66xx;
|
|
struct cvmx_lmcx_char_ctl_s cn68xx;
|
|
struct cvmx_lmcx_char_ctl_cn63xx cn68xxp1;
|
|
struct cvmx_lmcx_char_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_char_ctl cvmx_lmcx_char_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_char_mask0
|
|
*
|
|
* LMC_CHAR_MASK0 = LMC Characterization Mask0
|
|
* This register is an assortment of various control fields needed to charecterize the DDR3 interface
|
|
*/
|
|
union cvmx_lmcx_char_mask0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_char_mask0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t mask : 64; /**< Mask for DQ0[63:0] */
|
|
#else
|
|
uint64_t mask : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_char_mask0_s cn61xx;
|
|
struct cvmx_lmcx_char_mask0_s cn63xx;
|
|
struct cvmx_lmcx_char_mask0_s cn63xxp1;
|
|
struct cvmx_lmcx_char_mask0_s cn66xx;
|
|
struct cvmx_lmcx_char_mask0_s cn68xx;
|
|
struct cvmx_lmcx_char_mask0_s cn68xxp1;
|
|
struct cvmx_lmcx_char_mask0_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_char_mask0 cvmx_lmcx_char_mask0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_char_mask1
|
|
*
|
|
* LMC_CHAR_MASK1 = LMC Characterization Mask1
|
|
* This register is an assortment of various control fields needed to charecterize the DDR3 interface
|
|
*/
|
|
union cvmx_lmcx_char_mask1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_char_mask1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_8_63 : 56;
|
|
uint64_t mask : 8; /**< Mask for DQ0[71:64] */
|
|
#else
|
|
uint64_t mask : 8;
|
|
uint64_t reserved_8_63 : 56;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_char_mask1_s cn61xx;
|
|
struct cvmx_lmcx_char_mask1_s cn63xx;
|
|
struct cvmx_lmcx_char_mask1_s cn63xxp1;
|
|
struct cvmx_lmcx_char_mask1_s cn66xx;
|
|
struct cvmx_lmcx_char_mask1_s cn68xx;
|
|
struct cvmx_lmcx_char_mask1_s cn68xxp1;
|
|
struct cvmx_lmcx_char_mask1_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_char_mask1 cvmx_lmcx_char_mask1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_char_mask2
|
|
*
|
|
* LMC_CHAR_MASK2 = LMC Characterization Mask2
|
|
* This register is an assortment of various control fields needed to charecterize the DDR3 interface
|
|
*/
|
|
union cvmx_lmcx_char_mask2 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_char_mask2_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t mask : 64; /**< Mask for DQ1[63:0] */
|
|
#else
|
|
uint64_t mask : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_char_mask2_s cn61xx;
|
|
struct cvmx_lmcx_char_mask2_s cn63xx;
|
|
struct cvmx_lmcx_char_mask2_s cn63xxp1;
|
|
struct cvmx_lmcx_char_mask2_s cn66xx;
|
|
struct cvmx_lmcx_char_mask2_s cn68xx;
|
|
struct cvmx_lmcx_char_mask2_s cn68xxp1;
|
|
struct cvmx_lmcx_char_mask2_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_char_mask2 cvmx_lmcx_char_mask2_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_char_mask3
|
|
*
|
|
* LMC_CHAR_MASK3 = LMC Characterization Mask3
|
|
* This register is an assortment of various control fields needed to charecterize the DDR3 interface
|
|
*/
|
|
union cvmx_lmcx_char_mask3 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_char_mask3_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_8_63 : 56;
|
|
uint64_t mask : 8; /**< Mask for DQ1[71:64] */
|
|
#else
|
|
uint64_t mask : 8;
|
|
uint64_t reserved_8_63 : 56;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_char_mask3_s cn61xx;
|
|
struct cvmx_lmcx_char_mask3_s cn63xx;
|
|
struct cvmx_lmcx_char_mask3_s cn63xxp1;
|
|
struct cvmx_lmcx_char_mask3_s cn66xx;
|
|
struct cvmx_lmcx_char_mask3_s cn68xx;
|
|
struct cvmx_lmcx_char_mask3_s cn68xxp1;
|
|
struct cvmx_lmcx_char_mask3_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_char_mask3 cvmx_lmcx_char_mask3_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_char_mask4
|
|
*
|
|
* LMC_CHAR_MASK4 = LMC Characterization Mask4
|
|
* This register is an assortment of various control fields needed to charecterize the DDR3 interface
|
|
*/
|
|
union cvmx_lmcx_char_mask4 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_char_mask4_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_33_63 : 31;
|
|
uint64_t reset_n_mask : 1; /**< Mask for RESET_L */
|
|
uint64_t a_mask : 16; /**< Mask for A[15:0] */
|
|
uint64_t ba_mask : 3; /**< Mask for BA[2:0] */
|
|
uint64_t we_n_mask : 1; /**< Mask for WE_N */
|
|
uint64_t cas_n_mask : 1; /**< Mask for CAS_N */
|
|
uint64_t ras_n_mask : 1; /**< Mask for RAS_N */
|
|
uint64_t odt1_mask : 2; /**< Mask for ODT1 */
|
|
uint64_t odt0_mask : 2; /**< Mask for ODT0 */
|
|
uint64_t cs1_n_mask : 2; /**< Mask for CS1_N */
|
|
uint64_t cs0_n_mask : 2; /**< Mask for CS0_N */
|
|
uint64_t cke_mask : 2; /**< Mask for CKE* */
|
|
#else
|
|
uint64_t cke_mask : 2;
|
|
uint64_t cs0_n_mask : 2;
|
|
uint64_t cs1_n_mask : 2;
|
|
uint64_t odt0_mask : 2;
|
|
uint64_t odt1_mask : 2;
|
|
uint64_t ras_n_mask : 1;
|
|
uint64_t cas_n_mask : 1;
|
|
uint64_t we_n_mask : 1;
|
|
uint64_t ba_mask : 3;
|
|
uint64_t a_mask : 16;
|
|
uint64_t reset_n_mask : 1;
|
|
uint64_t reserved_33_63 : 31;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_char_mask4_s cn61xx;
|
|
struct cvmx_lmcx_char_mask4_s cn63xx;
|
|
struct cvmx_lmcx_char_mask4_s cn63xxp1;
|
|
struct cvmx_lmcx_char_mask4_s cn66xx;
|
|
struct cvmx_lmcx_char_mask4_s cn68xx;
|
|
struct cvmx_lmcx_char_mask4_s cn68xxp1;
|
|
struct cvmx_lmcx_char_mask4_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_char_mask4 cvmx_lmcx_char_mask4_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_comp_ctl
|
|
*
|
|
* LMC_COMP_CTL = LMC Compensation control
|
|
*
|
|
*/
|
|
union cvmx_lmcx_comp_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_comp_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t nctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t nctl_clk : 4; /**< Compensation control bits */
|
|
uint64_t nctl_cmd : 4; /**< Compensation control bits */
|
|
uint64_t nctl_dat : 4; /**< Compensation control bits */
|
|
uint64_t pctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t pctl_clk : 4; /**< Compensation control bits */
|
|
uint64_t reserved_0_7 : 8;
|
|
#else
|
|
uint64_t reserved_0_7 : 8;
|
|
uint64_t pctl_clk : 4;
|
|
uint64_t pctl_csr : 4;
|
|
uint64_t nctl_dat : 4;
|
|
uint64_t nctl_cmd : 4;
|
|
uint64_t nctl_clk : 4;
|
|
uint64_t nctl_csr : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_comp_ctl_cn30xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t nctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t nctl_clk : 4; /**< Compensation control bits */
|
|
uint64_t nctl_cmd : 4; /**< Compensation control bits */
|
|
uint64_t nctl_dat : 4; /**< Compensation control bits */
|
|
uint64_t pctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t pctl_clk : 4; /**< Compensation control bits */
|
|
uint64_t pctl_cmd : 4; /**< Compensation control bits */
|
|
uint64_t pctl_dat : 4; /**< Compensation control bits */
|
|
#else
|
|
uint64_t pctl_dat : 4;
|
|
uint64_t pctl_cmd : 4;
|
|
uint64_t pctl_clk : 4;
|
|
uint64_t pctl_csr : 4;
|
|
uint64_t nctl_dat : 4;
|
|
uint64_t nctl_cmd : 4;
|
|
uint64_t nctl_clk : 4;
|
|
uint64_t nctl_csr : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn30xx;
|
|
struct cvmx_lmcx_comp_ctl_cn30xx cn31xx;
|
|
struct cvmx_lmcx_comp_ctl_cn30xx cn38xx;
|
|
struct cvmx_lmcx_comp_ctl_cn30xx cn38xxp2;
|
|
struct cvmx_lmcx_comp_ctl_cn50xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t nctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t reserved_20_27 : 8;
|
|
uint64_t nctl_dat : 4; /**< Compensation control bits */
|
|
uint64_t pctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t reserved_5_11 : 7;
|
|
uint64_t pctl_dat : 5; /**< Compensation control bits */
|
|
#else
|
|
uint64_t pctl_dat : 5;
|
|
uint64_t reserved_5_11 : 7;
|
|
uint64_t pctl_csr : 4;
|
|
uint64_t nctl_dat : 4;
|
|
uint64_t reserved_20_27 : 8;
|
|
uint64_t nctl_csr : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn50xx;
|
|
struct cvmx_lmcx_comp_ctl_cn50xx cn52xx;
|
|
struct cvmx_lmcx_comp_ctl_cn50xx cn52xxp1;
|
|
struct cvmx_lmcx_comp_ctl_cn50xx cn56xx;
|
|
struct cvmx_lmcx_comp_ctl_cn50xx cn56xxp1;
|
|
struct cvmx_lmcx_comp_ctl_cn50xx cn58xx;
|
|
struct cvmx_lmcx_comp_ctl_cn58xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t nctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t reserved_20_27 : 8;
|
|
uint64_t nctl_dat : 4; /**< Compensation control bits */
|
|
uint64_t pctl_csr : 4; /**< Compensation control bits */
|
|
uint64_t reserved_4_11 : 8;
|
|
uint64_t pctl_dat : 4; /**< Compensation control bits */
|
|
#else
|
|
uint64_t pctl_dat : 4;
|
|
uint64_t reserved_4_11 : 8;
|
|
uint64_t pctl_csr : 4;
|
|
uint64_t nctl_dat : 4;
|
|
uint64_t reserved_20_27 : 8;
|
|
uint64_t nctl_csr : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_comp_ctl cvmx_lmcx_comp_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_comp_ctl2
|
|
*
|
|
* LMC_COMP_CTL2 = LMC Compensation control
|
|
*
|
|
*/
|
|
union cvmx_lmcx_comp_ctl2 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_comp_ctl2_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_34_63 : 30;
|
|
uint64_t ddr__ptune : 4; /**< DDR PCTL from compensation circuit
|
|
The encoded value provides debug information for the
|
|
compensation impedance on P-pullup */
|
|
uint64_t ddr__ntune : 4; /**< DDR NCTL from compensation circuit
|
|
The encoded value provides debug information for the
|
|
compensation impedance on N-pulldown */
|
|
uint64_t m180 : 1; /**< Cap impedance at 180 Ohm (instead of 240 Ohm) */
|
|
uint64_t byp : 1; /**< Bypass mode
|
|
When set, PTUNE,NTUNE are the compensation setting.
|
|
When clear, DDR_PTUNE,DDR_NTUNE are the compensation setting. */
|
|
uint64_t ptune : 4; /**< PCTL impedance control in bypass mode */
|
|
uint64_t ntune : 4; /**< NCTL impedance control in bypass mode */
|
|
uint64_t rodt_ctl : 4; /**< NCTL RODT impedance control bits
|
|
This field controls ODT values during a memory read
|
|
on the Octeon side
|
|
0000 = No ODT
|
|
0001 = 20 ohm
|
|
0010 = 30 ohm
|
|
0011 = 40 ohm
|
|
0100 = 60 ohm
|
|
0101 = 120 ohm
|
|
0110-1111 = Reserved */
|
|
uint64_t cmd_ctl : 4; /**< Drive strength control for CMD/A/RESET_L drivers
|
|
0001 = 24 ohm
|
|
0010 = 26.67 ohm
|
|
0011 = 30 ohm
|
|
0100 = 34.3 ohm
|
|
0101 = 40 ohm
|
|
0110 = 48 ohm
|
|
0111 = 60 ohm
|
|
0000,1000-1111 = Reserved */
|
|
uint64_t ck_ctl : 4; /**< Drive strength control for CK/CS*_L/ODT/CKE* drivers
|
|
0001 = 24 ohm
|
|
0010 = 26.67 ohm
|
|
0011 = 30 ohm
|
|
0100 = 34.3 ohm
|
|
0101 = 40 ohm
|
|
0110 = 48 ohm
|
|
0111 = 60 ohm
|
|
0000,1000-1111 = Reserved */
|
|
uint64_t dqx_ctl : 4; /**< Drive strength control for DQ/DQS drivers
|
|
0001 = 24 ohm
|
|
0010 = 26.67 ohm
|
|
0011 = 30 ohm
|
|
0100 = 34.3 ohm
|
|
0101 = 40 ohm
|
|
0110 = 48 ohm
|
|
0111 = 60 ohm
|
|
0000,1000-1111 = Reserved */
|
|
#else
|
|
uint64_t dqx_ctl : 4;
|
|
uint64_t ck_ctl : 4;
|
|
uint64_t cmd_ctl : 4;
|
|
uint64_t rodt_ctl : 4;
|
|
uint64_t ntune : 4;
|
|
uint64_t ptune : 4;
|
|
uint64_t byp : 1;
|
|
uint64_t m180 : 1;
|
|
uint64_t ddr__ntune : 4;
|
|
uint64_t ddr__ptune : 4;
|
|
uint64_t reserved_34_63 : 30;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_comp_ctl2_s cn61xx;
|
|
struct cvmx_lmcx_comp_ctl2_s cn63xx;
|
|
struct cvmx_lmcx_comp_ctl2_s cn63xxp1;
|
|
struct cvmx_lmcx_comp_ctl2_s cn66xx;
|
|
struct cvmx_lmcx_comp_ctl2_s cn68xx;
|
|
struct cvmx_lmcx_comp_ctl2_s cn68xxp1;
|
|
struct cvmx_lmcx_comp_ctl2_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_comp_ctl2 cvmx_lmcx_comp_ctl2_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_config
|
|
*
|
|
* LMC_CONFIG = LMC Configuration Register
|
|
*
|
|
* This register controls certain parameters of Memory Configuration
|
|
*
|
|
* Notes:
|
|
* a. Priority order for hardware writes to LMC*_CONFIG/LMC*_FADR/LMC*_SCRAMBLED_FADR/LMC*_ECC_SYND: DED error >= NXM error > SEC error
|
|
* b. The self refresh entry sequence(s) power the DLL up/down (depending on LMC*_MODEREG_PARAMS0[DLL])
|
|
* when LMC*_CONFIG[SREF_WITH_DLL] is set
|
|
* c. Prior to the self-refresh exit sequence, LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 should be re-programmed (if needed) to the
|
|
* appropriate values
|
|
*
|
|
* LMC Bringup Sequence:
|
|
* 1. SW must ensure there are no pending DRAM transactions and that the DDR PLL and the DLL have been initialized.
|
|
* 2. Write LMC*_COMP_CTL2, LMC*_CONTROL, LMC*_WODT_MASK, LMC*_DUAL_MEMCFG, LMC*_TIMING_PARAMS0, LMC*_TIMING_PARAMS1,
|
|
* LMC*_MODEREG_PARAMS0, LMC*_MODEREG_PARAMS1, LMC*_RESET_CTL (with DDR3RST=0), LMC*_CONFIG (with INIT_START=0)
|
|
* with appropriate values, if necessary.
|
|
* 3. Wait 200us, then write LMC*_RESET_CTL[DDR3RST] = 1.
|
|
* 4. Initialize all ranks at once by writing LMC*_CONFIG[RANKMASK][n] = 1, LMC*_CONFIG[INIT_STATUS][n] = 1, and LMC*_CONFIG[INIT_START] = 1
|
|
* where n is a valid rank index for the specific board configuration.
|
|
* 5. for each rank n to be write-leveled [
|
|
* if auto write-leveling is desired [
|
|
* write LMC*_CONFIG[RANKMASK][n] = 1, LMC*_WLEVEL_CTL appropriately and LMC*_CONFIG[INIT_START] = 1
|
|
* wait until LMC*_WLEVEL_RANKn[STATUS] = 3
|
|
* ] else [
|
|
* write LMC*_WLEVEL_RANKn with appropriate values
|
|
* ]
|
|
* ]
|
|
* 6. for each rank n to be read-leveled [
|
|
* if auto read-leveling is desired [
|
|
* write LMC*_CONFIG[RANKMASK][n] = 1, LMC*_RLEVEL_CTL appropriately and LMC*_CONFIG[INIT_START] = 1
|
|
* wait until LMC*_RLEVEL_RANKn[STATUS] = 3
|
|
* ] else [
|
|
* write LMC*_RLEVEL_RANKn with appropriate values
|
|
* ]
|
|
* ]
|
|
*/
|
|
union cvmx_lmcx_config {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_config_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_61_63 : 3;
|
|
uint64_t mode32b : 1; /**< 32b Datapath Mode NS
|
|
Set to 1 if we use only 32 DQ pins
|
|
0 for 64b DQ mode. */
|
|
uint64_t scrz : 1; /**< Hide LMC*_SCRAMBLE_CFG0 and LMC*_SCRAMBLE_CFG1 when set */
|
|
uint64_t early_unload_d1_r1 : 1; /**< When set, unload the PHY silo one cycle early for Rank 3
|
|
reads
|
|
The recommended EARLY_UNLOAD_D1_R1 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK3[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 3 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK3[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D1_R1
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D1_R1 = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d1_r0 : 1; /**< When set, unload the PHY silo one cycle early for Rank 2
|
|
reads
|
|
The recommended EARLY_UNLOAD_D1_RO value can be calculated
|
|
after the final LMC*_RLEVEL_RANK2[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 2 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK2[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D1_RO
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D1_RO = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d0_r1 : 1; /**< When set, unload the PHY silo one cycle early for Rank 1
|
|
reads
|
|
The recommended EARLY_UNLOAD_D0_R1 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK1[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 1 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK1[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D0_R1
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D0_R1 = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d0_r0 : 1; /**< When set, unload the PHY silo one cycle early for Rank 0
|
|
reads.
|
|
The recommended EARLY_UNLOAD_D0_R0 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK0[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 0 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK0[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D0_R0
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D0_R0 = (maxset<1:0>!=3)). */
|
|
uint64_t init_status : 4; /**< Indicates status of initialization
|
|
INIT_STATUS[n] = 1 implies rank n has been initialized
|
|
SW must set necessary INIT_STATUS bits with the
|
|
same LMC*_CONFIG write that initiates
|
|
power-up/init and self-refresh exit sequences
|
|
(if the required INIT_STATUS bits are not already
|
|
set before LMC initiates the sequence).
|
|
INIT_STATUS determines the chip-selects that assert
|
|
during refresh, ZQCS, and precharge power-down and
|
|
self-refresh entry/exit SEQUENCE's. */
|
|
uint64_t mirrmask : 4; /**< Mask determining which ranks are address-mirrored.
|
|
MIRRMASK<n> = 1 means Rank n addresses are mirrored
|
|
for 0 <= n <= 3
|
|
A mirrored read/write has these differences:
|
|
- DDR_BA<1> is swapped with DDR_BA<0>
|
|
- DDR_A<8> is swapped with DDR_A<7>
|
|
- DDR_A<6> is swapped with DDR_A<5>
|
|
- DDR_A<4> is swapped with DDR_A<3>
|
|
When RANK_ENA=0, MIRRMASK<1> and MIRRMASK<3> MBZ */
|
|
uint64_t rankmask : 4; /**< Mask to select rank to be leveled/initialized.
|
|
To write-level/read-level/initialize rank i, set RANKMASK<i>
|
|
RANK_ENA=1 RANK_ENA=0
|
|
RANKMASK<0> = DIMM0_CS0 DIMM0_CS0
|
|
RANKMASK<1> = DIMM0_CS1 MBZ
|
|
RANKMASK<2> = DIMM1_CS0 DIMM1_CS0
|
|
RANKMASK<3> = DIMM1_CS1 MBZ
|
|
For read/write leveling, each rank has to be leveled separately,
|
|
so RANKMASK should only have one bit set.
|
|
RANKMASK is not used during self-refresh entry/exit and
|
|
precharge power-down entry/exit instruction sequences.
|
|
When RANK_ENA=0, RANKMASK<1> and RANKMASK<3> MBZ */
|
|
uint64_t rank_ena : 1; /**< RANK ena (for use with dual-rank DIMMs)
|
|
For dual-rank DIMMs, the rank_ena bit will enable
|
|
the drive of the CS*_L[1:0] and ODT_<1:0> pins differently based on the
|
|
(pbank_lsb-1) address bit.
|
|
Write 0 for SINGLE ranked DIMM's. */
|
|
uint64_t sref_with_dll : 1; /**< Self-refresh entry/exit write MR1 and MR2
|
|
When set, self-refresh entry and exit instruction sequences
|
|
write MR1 and MR2 (in all ranks). (The writes occur before
|
|
self-refresh entry, and after self-refresh exit.)
|
|
When clear, self-refresh entry and exit instruction sequences
|
|
do not write any registers in the DDR3 parts. */
|
|
uint64_t early_dqx : 1; /**< Send DQx signals one CK cycle earlier for the case when
|
|
the shortest DQx lines have a larger delay than the CK line */
|
|
uint64_t sequence : 3; /**< Selects the sequence that LMC runs after a 0->1
|
|
transition on LMC*_CONFIG[INIT_START].
|
|
SEQUENCE=0=power-up/init:
|
|
- RANKMASK selects participating ranks (should be all ranks with attached DRAM)
|
|
- INIT_STATUS must equal RANKMASK
|
|
- DDR_DIMM*_CKE signals activated (if they weren't already active)
|
|
- RDIMM register control words 0-15 will be written to RANKMASK-selected
|
|
RDIMM's when LMC(0)_CONTROL[RDIMM_ENA]=1 and corresponding
|
|
LMC*_DIMM_CTL[DIMM*_WMASK] bits are set. (Refer to LMC*_DIMM*_PARAMS and
|
|
LMC*_DIMM_CTL descriptions below for more details.)
|
|
- MR0, MR1, MR2, and MR3 will be written to selected ranks
|
|
SEQUENCE=1=read-leveling:
|
|
- RANKMASK selects the rank to be read-leveled
|
|
- MR3 written to selected rank
|
|
SEQUENCE=2=self-refresh entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- MR1 and MR2 will be written to selected ranks if SREF_WITH_DLL=1
|
|
- DDR_DIMM*_CKE signals de-activated
|
|
SEQUENCE=3=self-refresh exit:
|
|
- INIT_STATUS must be set to indicate participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_DIMM*_CKE signals activated
|
|
- MR0, MR1, MR2, and MR3 will be written to participating ranks if SREF_WITH_DLL=1
|
|
SEQUENCE=4=precharge power-down entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_DIMM*_CKE signals de-activated
|
|
SEQUENCE=5=precharge power-down exit:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_DIMM*_CKE signals activated
|
|
SEQUENCE=6=write-leveling:
|
|
- RANKMASK selects the rank to be write-leveled
|
|
- INIT_STATUS must indicate all ranks with attached DRAM
|
|
- MR1 and MR2 written to INIT_STATUS-selected ranks
|
|
SEQUENCE=7=illegal
|
|
Precharge power-down entry and exit SEQUENCE's may also
|
|
be automatically generated by the HW when IDLEPOWER!=0.
|
|
Self-refresh entry SEQUENCE's may also be automatically
|
|
generated by hardware upon a chip warm or soft reset
|
|
sequence when LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT] are set.
|
|
LMC writes the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 CSR field values
|
|
to the Mode registers in the DRAM parts (i.e. MR0, MR1, MR2, and MR3) as part of some of these sequences.
|
|
Refer to the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 descriptions for more details.
|
|
If there are two consecutive power-up/init's without
|
|
a DRESET assertion between them, LMC asserts DDR_DIMM*_CKE as part of
|
|
the first power-up/init, and continues to assert DDR_DIMM*_CKE
|
|
through the remainder of the first and the second power-up/init.
|
|
If DDR_DIMM*_CKE deactivation and reactivation is needed for
|
|
a second power-up/init, a DRESET assertion is required
|
|
between the first and the second. */
|
|
uint64_t ref_zqcs_int : 19; /**< Refresh & ZQCS interval represented in \#of 512 CK cycle
|
|
increments. A Refresh sequence is triggered when bits
|
|
[24:18] are equal to 0, and a ZQCS sequence is triggered
|
|
when [36:18] are equal to 0.
|
|
Program [24:18] to RND-DN(tREFI/clkPeriod/512)
|
|
Program [36:25] to RND-DN(ZQCS_Interval/clkPeriod/(512*64)). Note
|
|
that this value should always be greater than 32, to account for
|
|
resistor calibration delays.
|
|
000_00000000_00000000: RESERVED
|
|
Max Refresh interval = 127 * 512 = 65024 CKs
|
|
Max ZQCS interval = (8*256*256-1) * 512 = 268434944 CKs ~ 335ms for a 800 MHz CK
|
|
LMC*_CONFIG[INIT_STATUS] determines which ranks receive
|
|
the REF / ZQCS. LMC does not send any refreshes / ZQCS's
|
|
when LMC*_CONFIG[INIT_STATUS]=0. */
|
|
uint64_t reset : 1; /**< Reset oneshot pulse for refresh counter,
|
|
and LMC*_OPS_CNT, LMC*_IFB_CNT, and LMC*_DCLK_CNT
|
|
CSR's. SW should write this to a one, then re-write
|
|
it to a zero to cause the reset. */
|
|
uint64_t ecc_adr : 1; /**< Include memory reference address in the ECC calculation
|
|
0=disabled, 1=enabled */
|
|
uint64_t forcewrite : 4; /**< Force the oldest outstanding write to complete after
|
|
having waited for 2^FORCEWRITE CK cycles. 0=disabled. */
|
|
uint64_t idlepower : 3; /**< Enter precharge power-down mode after the memory
|
|
controller has been idle for 2^(2+IDLEPOWER) CK cycles.
|
|
0=disabled.
|
|
This field should only be programmed after initialization.
|
|
LMC*_MODEREG_PARAMS0[PPD] determines whether the DRAM DLL
|
|
is disabled during the precharge power-down. */
|
|
uint64_t pbank_lsb : 4; /**< DIMM address bit select
|
|
Reverting to the explanation for ROW_LSB,
|
|
PBank_LSB would be Row_LSB bit + \#rowbits + \#rankbits
|
|
In the 512MB DIMM Example, assuming no rank bits:
|
|
pbank_lsb=mem_addr[15+13] for 64b mode
|
|
=mem_addr[14+13] for 32b mode
|
|
Decoding for pbank_lsb
|
|
- 0000:DIMM = mem_adr[28] / rank = mem_adr[27] (if RANK_ENA)
|
|
- 0001:DIMM = mem_adr[29] / rank = mem_adr[28] "
|
|
- 0010:DIMM = mem_adr[30] / rank = mem_adr[29] "
|
|
- 0011:DIMM = mem_adr[31] / rank = mem_adr[30] "
|
|
- 0100:DIMM = mem_adr[32] / rank = mem_adr[31] "
|
|
- 0101:DIMM = mem_adr[33] / rank = mem_adr[32] "
|
|
- 0110:DIMM = mem_adr[34] / rank = mem_adr[33] "
|
|
- 0111:DIMM = 0 / rank = mem_adr[34] "
|
|
- 1000-1111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16]
|
|
With rank_ena = 0, pbank_lsb = 2
|
|
With rank_ena = 1, pbank_lsb = 3 */
|
|
uint64_t row_lsb : 3; /**< Row Address bit select
|
|
Encoding used to determine which memory address
|
|
bit position represents the low order DDR ROW address.
|
|
The processor's memory address[34:7] needs to be
|
|
translated to DRAM addresses (bnk,row,col,rank and DIMM)
|
|
and that is a function of the following:
|
|
1. Datapath Width (64 or 32)
|
|
2. \# Banks (8)
|
|
3. \# Column Bits of the memory part - spec'd indirectly
|
|
by this register.
|
|
4. \# Row Bits of the memory part - spec'd indirectly
|
|
5. \# Ranks in a DIMM - spec'd by RANK_ENA
|
|
6. \# DIMM's in the system by the register below (PBANK_LSB).
|
|
Col Address starts from mem_addr[2] for 32b (4Bytes)
|
|
dq width or from mem_addr[3] for 64b (8Bytes) dq width
|
|
\# col + \# bank = 12. Hence row_lsb is mem_adr[15] for
|
|
64bmode or mem_adr[14] for 32b mode. Hence row_lsb
|
|
parameter should be set to 001 (64b) or 000 (32b).
|
|
Decoding for row_lsb
|
|
- 000: row_lsb = mem_adr[14]
|
|
- 001: row_lsb = mem_adr[15]
|
|
- 010: row_lsb = mem_adr[16]
|
|
- 011: row_lsb = mem_adr[17]
|
|
- 100: row_lsb = mem_adr[18]
|
|
- 101: row_lsb = mem_adr[19]
|
|
- 110: row_lsb = mem_adr[20]
|
|
- 111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16] */
|
|
uint64_t ecc_ena : 1; /**< ECC Enable: When set will enable the 8b ECC
|
|
check/correct logic. Should be 1 when used with DIMMs
|
|
with ECC. 0, otherwise.
|
|
When this mode is turned on, DQ[71:64]
|
|
on writes, will contain the ECC code generated for
|
|
the 64 bits of data which will
|
|
written in the memory and then later on reads, used
|
|
to check for Single bit error (which will be auto-
|
|
corrected) and Double Bit error (which will be
|
|
reported). When not turned on, DQ[71:64]
|
|
are driven to 0. Please refer to SEC_ERR, DED_ERR,
|
|
LMC*_FADR, LMC*_SCRAMBLED_FADR and LMC*_ECC_SYND registers
|
|
for diagnostics information when there is an error. */
|
|
uint64_t init_start : 1; /**< A 0->1 transition starts the DDR memory sequence that is
|
|
selected by LMC*_CONFIG[SEQUENCE]. This register is a
|
|
oneshot and clears itself each time it is set. */
|
|
#else
|
|
uint64_t init_start : 1;
|
|
uint64_t ecc_ena : 1;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t pbank_lsb : 4;
|
|
uint64_t idlepower : 3;
|
|
uint64_t forcewrite : 4;
|
|
uint64_t ecc_adr : 1;
|
|
uint64_t reset : 1;
|
|
uint64_t ref_zqcs_int : 19;
|
|
uint64_t sequence : 3;
|
|
uint64_t early_dqx : 1;
|
|
uint64_t sref_with_dll : 1;
|
|
uint64_t rank_ena : 1;
|
|
uint64_t rankmask : 4;
|
|
uint64_t mirrmask : 4;
|
|
uint64_t init_status : 4;
|
|
uint64_t early_unload_d0_r0 : 1;
|
|
uint64_t early_unload_d0_r1 : 1;
|
|
uint64_t early_unload_d1_r0 : 1;
|
|
uint64_t early_unload_d1_r1 : 1;
|
|
uint64_t scrz : 1;
|
|
uint64_t mode32b : 1;
|
|
uint64_t reserved_61_63 : 3;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_config_s cn61xx;
|
|
struct cvmx_lmcx_config_cn63xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_59_63 : 5;
|
|
uint64_t early_unload_d1_r1 : 1; /**< When set, unload the PHY silo one cycle early for Rank 3
|
|
reads
|
|
The recommended EARLY_UNLOAD_D1_R1 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK3[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 3 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK3[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D1_R1
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D1_R1 = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d1_r0 : 1; /**< When set, unload the PHY silo one cycle early for Rank 2
|
|
reads
|
|
The recommended EARLY_UNLOAD_D1_RO value can be calculated
|
|
after the final LMC*_RLEVEL_RANK2[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 2 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK2[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D1_RO
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D1_RO = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d0_r1 : 1; /**< When set, unload the PHY silo one cycle early for Rank 1
|
|
reads
|
|
The recommended EARLY_UNLOAD_D0_R1 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK1[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 1 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK1[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D0_R1
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D0_R1 = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d0_r0 : 1; /**< When set, unload the PHY silo one cycle early for Rank 0
|
|
reads.
|
|
The recommended EARLY_UNLOAD_D0_R0 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK0[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 0 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK0[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D0_R0
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D0_R0 = (maxset<1:0>!=3)). */
|
|
uint64_t init_status : 4; /**< Indicates status of initialization
|
|
INIT_STATUS[n] = 1 implies rank n has been initialized
|
|
SW must set necessary INIT_STATUS bits with the
|
|
same LMC*_CONFIG write that initiates
|
|
power-up/init and self-refresh exit sequences
|
|
(if the required INIT_STATUS bits are not already
|
|
set before LMC initiates the sequence).
|
|
INIT_STATUS determines the chip-selects that assert
|
|
during refresh, ZQCS, and precharge power-down and
|
|
self-refresh entry/exit SEQUENCE's. */
|
|
uint64_t mirrmask : 4; /**< Mask determining which ranks are address-mirrored.
|
|
MIRRMASK<n> = 1 means Rank n addresses are mirrored
|
|
for 0 <= n <= 3
|
|
A mirrored read/write has these differences:
|
|
- DDR_BA<1> is swapped with DDR_BA<0>
|
|
- DDR_A<8> is swapped with DDR_A<7>
|
|
- DDR_A<6> is swapped with DDR_A<5>
|
|
- DDR_A<4> is swapped with DDR_A<3>
|
|
When RANK_ENA=0, MIRRMASK<1> and MIRRMASK<3> MBZ */
|
|
uint64_t rankmask : 4; /**< Mask to select rank to be leveled/initialized.
|
|
To write-level/read-level/initialize rank i, set RANKMASK<i>
|
|
RANK_ENA=1 RANK_ENA=0
|
|
RANKMASK<0> = DIMM0_CS0 DIMM0_CS0
|
|
RANKMASK<1> = DIMM0_CS1 MBZ
|
|
RANKMASK<2> = DIMM1_CS0 DIMM1_CS0
|
|
RANKMASK<3> = DIMM1_CS1 MBZ
|
|
For read/write leveling, each rank has to be leveled separately,
|
|
so RANKMASK should only have one bit set.
|
|
RANKMASK is not used during self-refresh entry/exit and
|
|
precharge power-down entry/exit instruction sequences.
|
|
When RANK_ENA=0, RANKMASK<1> and RANKMASK<3> MBZ */
|
|
uint64_t rank_ena : 1; /**< RANK ena (for use with dual-rank DIMMs)
|
|
For dual-rank DIMMs, the rank_ena bit will enable
|
|
the drive of the CS*_L[1:0] and ODT_<1:0> pins differently based on the
|
|
(pbank_lsb-1) address bit.
|
|
Write 0 for SINGLE ranked DIMM's. */
|
|
uint64_t sref_with_dll : 1; /**< Self-refresh entry/exit write MR1 and MR2
|
|
When set, self-refresh entry and exit instruction sequences
|
|
write MR1 and MR2 (in all ranks). (The writes occur before
|
|
self-refresh entry, and after self-refresh exit.)
|
|
When clear, self-refresh entry and exit instruction sequences
|
|
do not write any registers in the DDR3 parts. */
|
|
uint64_t early_dqx : 1; /**< Send DQx signals one CK cycle earlier for the case when
|
|
the shortest DQx lines have a larger delay than the CK line */
|
|
uint64_t sequence : 3; /**< Selects the sequence that LMC runs after a 0->1
|
|
transition on LMC*_CONFIG[INIT_START].
|
|
SEQUENCE=0=power-up/init:
|
|
- RANKMASK selects participating ranks (should be all ranks with attached DRAM)
|
|
- INIT_STATUS must equal RANKMASK
|
|
- DDR_CKE* signals activated (if they weren't already active)
|
|
- RDIMM register control words 0-15 will be written to RANKMASK-selected
|
|
RDIMM's when LMC(0)_CONTROL[RDIMM_ENA]=1 and corresponding
|
|
LMC*_DIMM_CTL[DIMM*_WMASK] bits are set. (Refer to LMC*_DIMM*_PARAMS and
|
|
LMC*_DIMM_CTL descriptions below for more details.)
|
|
- MR0, MR1, MR2, and MR3 will be written to selected ranks
|
|
SEQUENCE=1=read-leveling:
|
|
- RANKMASK selects the rank to be read-leveled
|
|
- MR3 written to selected rank
|
|
SEQUENCE=2=self-refresh entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- MR1 and MR2 will be written to selected ranks if SREF_WITH_DLL=1
|
|
- DDR_CKE* signals de-activated
|
|
SEQUENCE=3=self-refresh exit:
|
|
- INIT_STATUS must be set to indicate participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals activated
|
|
- MR0, MR1, MR2, and MR3 will be written to participating ranks if SREF_WITH_DLL=1
|
|
SEQUENCE=4=precharge power-down entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals de-activated
|
|
SEQUENCE=5=precharge power-down exit:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals activated
|
|
SEQUENCE=6=write-leveling:
|
|
- RANKMASK selects the rank to be write-leveled
|
|
- INIT_STATUS must indicate all ranks with attached DRAM
|
|
- MR1 and MR2 written to INIT_STATUS-selected ranks
|
|
SEQUENCE=7=illegal
|
|
Precharge power-down entry and exit SEQUENCE's may also
|
|
be automatically generated by the HW when IDLEPOWER!=0.
|
|
Self-refresh entry SEQUENCE's may also be automatically
|
|
generated by hardware upon a chip warm or soft reset
|
|
sequence when LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT] are set.
|
|
LMC writes the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 CSR field values
|
|
to the Mode registers in the DRAM parts (i.e. MR0, MR1, MR2, and MR3) as part of some of these sequences.
|
|
Refer to the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 descriptions for more details.
|
|
If there are two consecutive power-up/init's without
|
|
a DRESET assertion between them, LMC asserts DDR_CKE* as part of
|
|
the first power-up/init, and continues to assert DDR_CKE*
|
|
through the remainder of the first and the second power-up/init.
|
|
If DDR_CKE* deactivation and reactivation is needed for
|
|
a second power-up/init, a DRESET assertion is required
|
|
between the first and the second. */
|
|
uint64_t ref_zqcs_int : 19; /**< Refresh & ZQCS interval represented in \#of 512 CK cycle
|
|
increments. A Refresh sequence is triggered when bits
|
|
[24:18] are equal to 0, and a ZQCS sequence is triggered
|
|
when [36:18] are equal to 0.
|
|
Program [24:18] to RND-DN(tREFI/clkPeriod/512)
|
|
Program [36:25] to RND-DN(ZQCS_Interval/clkPeriod/(512*64)). Note
|
|
that this value should always be greater than 32, to account for
|
|
resistor calibration delays.
|
|
000_00000000_00000000: RESERVED
|
|
Max Refresh interval = 127 * 512 = 65024 CKs
|
|
Max ZQCS interval = (8*256*256-1) * 512 = 268434944 CKs ~ 335ms for a 800 MHz CK
|
|
LMC*_CONFIG[INIT_STATUS] determines which ranks receive
|
|
the REF / ZQCS. LMC does not send any refreshes / ZQCS's
|
|
when LMC*_CONFIG[INIT_STATUS]=0. */
|
|
uint64_t reset : 1; /**< Reset oneshot pulse for refresh counter,
|
|
and LMC*_OPS_CNT, LMC*_IFB_CNT, and LMC*_DCLK_CNT
|
|
CSR's. SW should write this to a one, then re-write
|
|
it to a zero to cause the reset. */
|
|
uint64_t ecc_adr : 1; /**< Include memory reference address in the ECC calculation
|
|
0=disabled, 1=enabled */
|
|
uint64_t forcewrite : 4; /**< Force the oldest outstanding write to complete after
|
|
having waited for 2^FORCEWRITE CK cycles. 0=disabled. */
|
|
uint64_t idlepower : 3; /**< Enter precharge power-down mode after the memory
|
|
controller has been idle for 2^(2+IDLEPOWER) CK cycles.
|
|
0=disabled.
|
|
This field should only be programmed after initialization.
|
|
LMC*_MODEREG_PARAMS0[PPD] determines whether the DRAM DLL
|
|
is disabled during the precharge power-down. */
|
|
uint64_t pbank_lsb : 4; /**< DIMM address bit select
|
|
Reverting to the explanation for ROW_LSB,
|
|
PBank_LSB would be Row_LSB bit + \#rowbits + \#rankbits
|
|
Decoding for pbank_lsb
|
|
- 0000:DIMM = mem_adr[28] / rank = mem_adr[27] (if RANK_ENA)
|
|
- 0001:DIMM = mem_adr[29] / rank = mem_adr[28] "
|
|
- 0010:DIMM = mem_adr[30] / rank = mem_adr[29] "
|
|
- 0011:DIMM = mem_adr[31] / rank = mem_adr[30] "
|
|
- 0100:DIMM = mem_adr[32] / rank = mem_adr[31] "
|
|
- 0101:DIMM = mem_adr[33] / rank = mem_adr[32] "
|
|
- 0110:DIMM = mem_adr[34] / rank = mem_adr[33] "
|
|
- 0111:DIMM = 0 / rank = mem_adr[34] "
|
|
- 1000-1111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16]
|
|
With rank_ena = 0, pbank_lsb = 2
|
|
With rank_ena = 1, pbank_lsb = 3 */
|
|
uint64_t row_lsb : 3; /**< Row Address bit select
|
|
Encoding used to determine which memory address
|
|
bit position represents the low order DDR ROW address.
|
|
The processor's memory address[34:7] needs to be
|
|
translated to DRAM addresses (bnk,row,col,rank and DIMM)
|
|
and that is a function of the following:
|
|
1. Datapath Width (64)
|
|
2. \# Banks (8)
|
|
3. \# Column Bits of the memory part - spec'd indirectly
|
|
by this register.
|
|
4. \# Row Bits of the memory part - spec'd indirectly
|
|
5. \# Ranks in a DIMM - spec'd by RANK_ENA
|
|
6. \# DIMM's in the system by the register below (PBANK_LSB).
|
|
Decoding for row_lsb
|
|
- 000: row_lsb = mem_adr[14]
|
|
- 001: row_lsb = mem_adr[15]
|
|
- 010: row_lsb = mem_adr[16]
|
|
- 011: row_lsb = mem_adr[17]
|
|
- 100: row_lsb = mem_adr[18]
|
|
- 101: row_lsb = mem_adr[19]
|
|
- 110: row_lsb = mem_adr[20]
|
|
- 111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16] */
|
|
uint64_t ecc_ena : 1; /**< ECC Enable: When set will enable the 8b ECC
|
|
check/correct logic. Should be 1 when used with DIMMs
|
|
with ECC. 0, otherwise.
|
|
When this mode is turned on, DQ[71:64]
|
|
on writes, will contain the ECC code generated for
|
|
the 64 bits of data which will
|
|
written in the memory and then later on reads, used
|
|
to check for Single bit error (which will be auto-
|
|
corrected) and Double Bit error (which will be
|
|
reported). When not turned on, DQ[71:64]
|
|
are driven to 0. Please refer to SEC_ERR, DED_ERR,
|
|
LMC*_FADR, and LMC*_ECC_SYND registers
|
|
for diagnostics information when there is an error. */
|
|
uint64_t init_start : 1; /**< A 0->1 transition starts the DDR memory sequence that is
|
|
selected by LMC*_CONFIG[SEQUENCE]. This register is a
|
|
oneshot and clears itself each time it is set. */
|
|
#else
|
|
uint64_t init_start : 1;
|
|
uint64_t ecc_ena : 1;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t pbank_lsb : 4;
|
|
uint64_t idlepower : 3;
|
|
uint64_t forcewrite : 4;
|
|
uint64_t ecc_adr : 1;
|
|
uint64_t reset : 1;
|
|
uint64_t ref_zqcs_int : 19;
|
|
uint64_t sequence : 3;
|
|
uint64_t early_dqx : 1;
|
|
uint64_t sref_with_dll : 1;
|
|
uint64_t rank_ena : 1;
|
|
uint64_t rankmask : 4;
|
|
uint64_t mirrmask : 4;
|
|
uint64_t init_status : 4;
|
|
uint64_t early_unload_d0_r0 : 1;
|
|
uint64_t early_unload_d0_r1 : 1;
|
|
uint64_t early_unload_d1_r0 : 1;
|
|
uint64_t early_unload_d1_r1 : 1;
|
|
uint64_t reserved_59_63 : 5;
|
|
#endif
|
|
} cn63xx;
|
|
struct cvmx_lmcx_config_cn63xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_55_63 : 9;
|
|
uint64_t init_status : 4; /**< Indicates status of initialization
|
|
INIT_STATUS[n] = 1 implies rank n has been initialized
|
|
SW must set necessary INIT_STATUS bits with the
|
|
same LMC*_CONFIG write that initiates
|
|
power-up/init and self-refresh exit sequences
|
|
(if the required INIT_STATUS bits are not already
|
|
set before LMC initiates the sequence).
|
|
INIT_STATUS determines the chip-selects that assert
|
|
during refresh, ZQCS, and precharge power-down and
|
|
self-refresh entry/exit SEQUENCE's. */
|
|
uint64_t mirrmask : 4; /**< Mask determining which ranks are address-mirrored.
|
|
MIRRMASK<n> = 1 means Rank n addresses are mirrored
|
|
for 0 <= n <= 3
|
|
A mirrored read/write has these differences:
|
|
- DDR_BA<1> is swapped with DDR_BA<0>
|
|
- DDR_A<8> is swapped with DDR_A<7>
|
|
- DDR_A<6> is swapped with DDR_A<5>
|
|
- DDR_A<4> is swapped with DDR_A<3>
|
|
When RANK_ENA=0, MIRRMASK<1> and MIRRMASK<3> MBZ */
|
|
uint64_t rankmask : 4; /**< Mask to select rank to be leveled/initialized.
|
|
To write-level/read-level/initialize rank i, set RANKMASK<i>
|
|
RANK_ENA=1 RANK_ENA=0
|
|
RANKMASK<0> = DIMM0_CS0 DIMM0_CS0
|
|
RANKMASK<1> = DIMM0_CS1 MBZ
|
|
RANKMASK<2> = DIMM1_CS0 DIMM1_CS0
|
|
RANKMASK<3> = DIMM1_CS1 MBZ
|
|
For read/write leveling, each rank has to be leveled separately,
|
|
so RANKMASK should only have one bit set.
|
|
RANKMASK is not used during self-refresh entry/exit and
|
|
precharge power-down entry/exit instruction sequences.
|
|
When RANK_ENA=0, RANKMASK<1> and RANKMASK<3> MBZ */
|
|
uint64_t rank_ena : 1; /**< RANK ena (for use with dual-rank DIMMs)
|
|
For dual-rank DIMMs, the rank_ena bit will enable
|
|
the drive of the CS*_L[1:0] and ODT_<1:0> pins differently based on the
|
|
(pbank_lsb-1) address bit.
|
|
Write 0 for SINGLE ranked DIMM's. */
|
|
uint64_t sref_with_dll : 1; /**< Self-refresh entry/exit write MR1 and MR2
|
|
When set, self-refresh entry and exit instruction sequences
|
|
write MR1 and MR2 (in all ranks). (The writes occur before
|
|
self-refresh entry, and after self-refresh exit.)
|
|
When clear, self-refresh entry and exit instruction sequences
|
|
do not write any registers in the DDR3 parts. */
|
|
uint64_t early_dqx : 1; /**< Send DQx signals one CK cycle earlier for the case when
|
|
the shortest DQx lines have a larger delay than the CK line */
|
|
uint64_t sequence : 3; /**< Selects the sequence that LMC runs after a 0->1
|
|
transition on LMC*_CONFIG[INIT_START].
|
|
SEQUENCE=0=power-up/init:
|
|
- RANKMASK selects participating ranks (should be all ranks with attached DRAM)
|
|
- INIT_STATUS must equal RANKMASK
|
|
- DDR_CKE* signals activated (if they weren't already active)
|
|
- RDIMM register control words 0-15 will be written to RANKMASK-selected
|
|
RDIMM's when LMC(0)_CONTROL[RDIMM_ENA]=1 and corresponding
|
|
LMC*_DIMM_CTL[DIMM*_WMASK] bits are set. (Refer to LMC*_DIMM*_PARAMS and
|
|
LMC*_DIMM_CTL descriptions below for more details.)
|
|
- MR0, MR1, MR2, and MR3 will be written to selected ranks
|
|
SEQUENCE=1=read-leveling:
|
|
- RANKMASK selects the rank to be read-leveled
|
|
- MR3 written to selected rank
|
|
SEQUENCE=2=self-refresh entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- MR1 and MR2 will be written to selected ranks if SREF_WITH_DLL=1
|
|
- DDR_CKE* signals de-activated
|
|
SEQUENCE=3=self-refresh exit:
|
|
- INIT_STATUS must be set to indicate participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals activated
|
|
- MR0, MR1, MR2, and MR3 will be written to participating ranks if SREF_WITH_DLL=1
|
|
SEQUENCE=4=precharge power-down entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals de-activated
|
|
SEQUENCE=5=precharge power-down exit:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals activated
|
|
SEQUENCE=6=write-leveling:
|
|
- RANKMASK selects the rank to be write-leveled
|
|
- INIT_STATUS must indicate all ranks with attached DRAM
|
|
- MR1 and MR2 written to INIT_STATUS-selected ranks
|
|
SEQUENCE=7=illegal
|
|
Precharge power-down entry and exit SEQUENCE's may also
|
|
be automatically generated by the HW when IDLEPOWER!=0.
|
|
Self-refresh entry SEQUENCE's may also be automatically
|
|
generated by hardware upon a chip warm or soft reset
|
|
sequence when LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT] are set.
|
|
LMC writes the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 CSR field values
|
|
to the Mode registers in the DRAM parts (i.e. MR0, MR1, MR2, and MR3) as part of some of these sequences.
|
|
Refer to the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 descriptions for more details.
|
|
If there are two consecutive power-up/init's without
|
|
a DRESET assertion between them, LMC asserts DDR_CKE* as part of
|
|
the first power-up/init, and continues to assert DDR_CKE*
|
|
through the remainder of the first and the second power-up/init.
|
|
If DDR_CKE* deactivation and reactivation is needed for
|
|
a second power-up/init, a DRESET assertion is required
|
|
between the first and the second. */
|
|
uint64_t ref_zqcs_int : 19; /**< Refresh & ZQCS interval represented in \#of 512 CK cycle
|
|
increments. A Refresh sequence is triggered when bits
|
|
[24:18] are equal to 0, and a ZQCS sequence is triggered
|
|
when [36:18] are equal to 0.
|
|
Program [24:18] to RND-DN(tREFI/clkPeriod/512)
|
|
Program [36:25] to RND-DN(ZQCS_Interval/clkPeriod/(512*64)). Note
|
|
that this value should always be greater than 32, to account for
|
|
resistor calibration delays.
|
|
000_00000000_00000000: RESERVED
|
|
Max Refresh interval = 127 * 512 = 65024 CKs
|
|
Max ZQCS interval = (8*256*256-1) * 512 = 268434944 CKs ~ 335ms for a 800 MHz CK
|
|
LMC*_CONFIG[INIT_STATUS] determines which ranks receive
|
|
the REF / ZQCS. LMC does not send any refreshes / ZQCS's
|
|
when LMC*_CONFIG[INIT_STATUS]=0. */
|
|
uint64_t reset : 1; /**< Reset oneshot pulse for refresh counter,
|
|
and LMC*_OPS_CNT, LMC*_IFB_CNT, and LMC*_DCLK_CNT
|
|
CSR's. SW should write this to a one, then re-write
|
|
it to a zero to cause the reset. */
|
|
uint64_t ecc_adr : 1; /**< Include memory reference address in the ECC calculation
|
|
0=disabled, 1=enabled */
|
|
uint64_t forcewrite : 4; /**< Force the oldest outstanding write to complete after
|
|
having waited for 2^FORCEWRITE CK cycles. 0=disabled. */
|
|
uint64_t idlepower : 3; /**< Enter precharge power-down mode after the memory
|
|
controller has been idle for 2^(2+IDLEPOWER) CK cycles.
|
|
0=disabled.
|
|
This field should only be programmed after initialization.
|
|
LMC*_MODEREG_PARAMS0[PPD] determines whether the DRAM DLL
|
|
is disabled during the precharge power-down. */
|
|
uint64_t pbank_lsb : 4; /**< DIMM address bit select
|
|
Reverting to the explanation for ROW_LSB,
|
|
PBank_LSB would be Row_LSB bit + \#rowbits + \#rankbits
|
|
Decoding for pbank_lsb
|
|
- 0000:DIMM = mem_adr[28] / rank = mem_adr[27] (if RANK_ENA)
|
|
- 0001:DIMM = mem_adr[29] / rank = mem_adr[28] "
|
|
- 0010:DIMM = mem_adr[30] / rank = mem_adr[29] "
|
|
- 0011:DIMM = mem_adr[31] / rank = mem_adr[30] "
|
|
- 0100:DIMM = mem_adr[32] / rank = mem_adr[31] "
|
|
- 0101:DIMM = mem_adr[33] / rank = mem_adr[32] "
|
|
- 0110:DIMM = mem_adr[34] / rank = mem_adr[33] "
|
|
- 0111:DIMM = 0 / rank = mem_adr[34] "
|
|
- 1000-1111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16]
|
|
With rank_ena = 0, pbank_lsb = 2
|
|
With rank_ena = 1, pbank_lsb = 3 */
|
|
uint64_t row_lsb : 3; /**< Row Address bit select
|
|
Encoding used to determine which memory address
|
|
bit position represents the low order DDR ROW address.
|
|
The processor's memory address[34:7] needs to be
|
|
translated to DRAM addresses (bnk,row,col,rank and DIMM)
|
|
and that is a function of the following:
|
|
1. Datapath Width (64)
|
|
2. \# Banks (8)
|
|
3. \# Column Bits of the memory part - spec'd indirectly
|
|
by this register.
|
|
4. \# Row Bits of the memory part - spec'd indirectly
|
|
5. \# Ranks in a DIMM - spec'd by RANK_ENA
|
|
6. \# DIMM's in the system by the register below (PBANK_LSB).
|
|
Decoding for row_lsb
|
|
- 000: row_lsb = mem_adr[14]
|
|
- 001: row_lsb = mem_adr[15]
|
|
- 010: row_lsb = mem_adr[16]
|
|
- 011: row_lsb = mem_adr[17]
|
|
- 100: row_lsb = mem_adr[18]
|
|
- 101: row_lsb = mem_adr[19]
|
|
- 110: row_lsb = mem_adr[20]
|
|
- 111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16] */
|
|
uint64_t ecc_ena : 1; /**< ECC Enable: When set will enable the 8b ECC
|
|
check/correct logic. Should be 1 when used with DIMMs
|
|
with ECC. 0, otherwise.
|
|
When this mode is turned on, DQ[71:64]
|
|
on writes, will contain the ECC code generated for
|
|
the 64 bits of data which will
|
|
written in the memory and then later on reads, used
|
|
to check for Single bit error (which will be auto-
|
|
corrected) and Double Bit error (which will be
|
|
reported). When not turned on, DQ[71:64]
|
|
are driven to 0. Please refer to SEC_ERR, DED_ERR,
|
|
LMC*_FADR, and LMC*_ECC_SYND registers
|
|
for diagnostics information when there is an error. */
|
|
uint64_t init_start : 1; /**< A 0->1 transition starts the DDR memory sequence that is
|
|
selected by LMC*_CONFIG[SEQUENCE]. This register is a
|
|
oneshot and clears itself each time it is set. */
|
|
#else
|
|
uint64_t init_start : 1;
|
|
uint64_t ecc_ena : 1;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t pbank_lsb : 4;
|
|
uint64_t idlepower : 3;
|
|
uint64_t forcewrite : 4;
|
|
uint64_t ecc_adr : 1;
|
|
uint64_t reset : 1;
|
|
uint64_t ref_zqcs_int : 19;
|
|
uint64_t sequence : 3;
|
|
uint64_t early_dqx : 1;
|
|
uint64_t sref_with_dll : 1;
|
|
uint64_t rank_ena : 1;
|
|
uint64_t rankmask : 4;
|
|
uint64_t mirrmask : 4;
|
|
uint64_t init_status : 4;
|
|
uint64_t reserved_55_63 : 9;
|
|
#endif
|
|
} cn63xxp1;
|
|
struct cvmx_lmcx_config_cn66xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_60_63 : 4;
|
|
uint64_t scrz : 1; /**< Hide LMC*_SCRAMBLE_CFG0 and LMC*_SCRAMBLE_CFG1 when set */
|
|
uint64_t early_unload_d1_r1 : 1; /**< When set, unload the PHY silo one cycle early for Rank 3
|
|
reads
|
|
The recommended EARLY_UNLOAD_D1_R1 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK3[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 3 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK3[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D1_R1
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D1_R1 = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d1_r0 : 1; /**< When set, unload the PHY silo one cycle early for Rank 2
|
|
reads
|
|
The recommended EARLY_UNLOAD_D1_RO value can be calculated
|
|
after the final LMC*_RLEVEL_RANK2[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 2 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK2[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D1_RO
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D1_RO = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d0_r1 : 1; /**< When set, unload the PHY silo one cycle early for Rank 1
|
|
reads
|
|
The recommended EARLY_UNLOAD_D0_R1 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK1[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 1 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK1[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D0_R1
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D0_R1 = (maxset<1:0>!=3)). */
|
|
uint64_t early_unload_d0_r0 : 1; /**< When set, unload the PHY silo one cycle early for Rank 0
|
|
reads.
|
|
The recommended EARLY_UNLOAD_D0_R0 value can be calculated
|
|
after the final LMC*_RLEVEL_RANK0[BYTE*] values are
|
|
selected (as part of read-leveling initialization).
|
|
Then, determine the largest read-leveling setting
|
|
for rank 0 (i.e. calculate maxset=MAX(LMC*_RLEVEL_RANK0[BYTEi])
|
|
across all i), then set EARLY_UNLOAD_D0_R0
|
|
when the low two bits of this largest setting is not
|
|
3 (i.e. EARLY_UNLOAD_D0_R0 = (maxset<1:0>!=3)). */
|
|
uint64_t init_status : 4; /**< Indicates status of initialization
|
|
INIT_STATUS[n] = 1 implies rank n has been initialized
|
|
SW must set necessary INIT_STATUS bits with the
|
|
same LMC*_CONFIG write that initiates
|
|
power-up/init and self-refresh exit sequences
|
|
(if the required INIT_STATUS bits are not already
|
|
set before LMC initiates the sequence).
|
|
INIT_STATUS determines the chip-selects that assert
|
|
during refresh, ZQCS, and precharge power-down and
|
|
self-refresh entry/exit SEQUENCE's. */
|
|
uint64_t mirrmask : 4; /**< Mask determining which ranks are address-mirrored.
|
|
MIRRMASK<n> = 1 means Rank n addresses are mirrored
|
|
for 0 <= n <= 3
|
|
A mirrored read/write has these differences:
|
|
- DDR_BA<1> is swapped with DDR_BA<0>
|
|
- DDR_A<8> is swapped with DDR_A<7>
|
|
- DDR_A<6> is swapped with DDR_A<5>
|
|
- DDR_A<4> is swapped with DDR_A<3>
|
|
When RANK_ENA=0, MIRRMASK<1> and MIRRMASK<3> MBZ */
|
|
uint64_t rankmask : 4; /**< Mask to select rank to be leveled/initialized.
|
|
To write-level/read-level/initialize rank i, set RANKMASK<i>
|
|
RANK_ENA=1 RANK_ENA=0
|
|
RANKMASK<0> = DIMM0_CS0 DIMM0_CS0
|
|
RANKMASK<1> = DIMM0_CS1 MBZ
|
|
RANKMASK<2> = DIMM1_CS0 DIMM1_CS0
|
|
RANKMASK<3> = DIMM1_CS1 MBZ
|
|
For read/write leveling, each rank has to be leveled separately,
|
|
so RANKMASK should only have one bit set.
|
|
RANKMASK is not used during self-refresh entry/exit and
|
|
precharge power-down entry/exit instruction sequences.
|
|
When RANK_ENA=0, RANKMASK<1> and RANKMASK<3> MBZ */
|
|
uint64_t rank_ena : 1; /**< RANK ena (for use with dual-rank DIMMs)
|
|
For dual-rank DIMMs, the rank_ena bit will enable
|
|
the drive of the CS*_L[1:0] and ODT_<1:0> pins differently based on the
|
|
(pbank_lsb-1) address bit.
|
|
Write 0 for SINGLE ranked DIMM's. */
|
|
uint64_t sref_with_dll : 1; /**< Self-refresh entry/exit write MR1 and MR2
|
|
When set, self-refresh entry and exit instruction sequences
|
|
write MR1 and MR2 (in all ranks). (The writes occur before
|
|
self-refresh entry, and after self-refresh exit.)
|
|
When clear, self-refresh entry and exit instruction sequences
|
|
do not write any registers in the DDR3 parts. */
|
|
uint64_t early_dqx : 1; /**< Send DQx signals one CK cycle earlier for the case when
|
|
the shortest DQx lines have a larger delay than the CK line */
|
|
uint64_t sequence : 3; /**< Selects the sequence that LMC runs after a 0->1
|
|
transition on LMC*_CONFIG[INIT_START].
|
|
SEQUENCE=0=power-up/init:
|
|
- RANKMASK selects participating ranks (should be all ranks with attached DRAM)
|
|
- INIT_STATUS must equal RANKMASK
|
|
- DDR_CKE* signals activated (if they weren't already active)
|
|
- RDIMM register control words 0-15 will be written to RANKMASK-selected
|
|
RDIMM's when LMC(0)_CONTROL[RDIMM_ENA]=1 and corresponding
|
|
LMC*_DIMM_CTL[DIMM*_WMASK] bits are set. (Refer to LMC*_DIMM*_PARAMS and
|
|
LMC*_DIMM_CTL descriptions below for more details.)
|
|
- MR0, MR1, MR2, and MR3 will be written to selected ranks
|
|
SEQUENCE=1=read-leveling:
|
|
- RANKMASK selects the rank to be read-leveled
|
|
- MR3 written to selected rank
|
|
SEQUENCE=2=self-refresh entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- MR1 and MR2 will be written to selected ranks if SREF_WITH_DLL=1
|
|
- DDR_CKE* signals de-activated
|
|
SEQUENCE=3=self-refresh exit:
|
|
- INIT_STATUS must be set to indicate participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals activated
|
|
- MR0, MR1, MR2, and MR3 will be written to participating ranks if SREF_WITH_DLL=1
|
|
SEQUENCE=4=precharge power-down entry:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals de-activated
|
|
SEQUENCE=5=precharge power-down exit:
|
|
- INIT_STATUS selects participating ranks (should be all ranks with attached DRAM)
|
|
- DDR_CKE* signals activated
|
|
SEQUENCE=6=write-leveling:
|
|
- RANKMASK selects the rank to be write-leveled
|
|
- INIT_STATUS must indicate all ranks with attached DRAM
|
|
- MR1 and MR2 written to INIT_STATUS-selected ranks
|
|
SEQUENCE=7=illegal
|
|
Precharge power-down entry and exit SEQUENCE's may also
|
|
be automatically generated by the HW when IDLEPOWER!=0.
|
|
Self-refresh entry SEQUENCE's may also be automatically
|
|
generated by hardware upon a chip warm or soft reset
|
|
sequence when LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT] are set.
|
|
LMC writes the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 CSR field values
|
|
to the Mode registers in the DRAM parts (i.e. MR0, MR1, MR2, and MR3) as part of some of these sequences.
|
|
Refer to the LMC*_MODEREG_PARAMS0 and LMC*_MODEREG_PARAMS1 descriptions for more details.
|
|
If there are two consecutive power-up/init's without
|
|
a DRESET assertion between them, LMC asserts DDR_CKE* as part of
|
|
the first power-up/init, and continues to assert DDR_CKE*
|
|
through the remainder of the first and the second power-up/init.
|
|
If DDR_CKE* deactivation and reactivation is needed for
|
|
a second power-up/init, a DRESET assertion is required
|
|
between the first and the second. */
|
|
uint64_t ref_zqcs_int : 19; /**< Refresh & ZQCS interval represented in \#of 512 CK cycle
|
|
increments. A Refresh sequence is triggered when bits
|
|
[24:18] are equal to 0, and a ZQCS sequence is triggered
|
|
when [36:18] are equal to 0.
|
|
Program [24:18] to RND-DN(tREFI/clkPeriod/512)
|
|
Program [36:25] to RND-DN(ZQCS_Interval/clkPeriod/(512*64)). Note
|
|
that this value should always be greater than 32, to account for
|
|
resistor calibration delays.
|
|
000_00000000_00000000: RESERVED
|
|
Max Refresh interval = 127 * 512 = 65024 CKs
|
|
Max ZQCS interval = (8*256*256-1) * 512 = 268434944 CKs ~ 335ms for a 800 MHz CK
|
|
LMC*_CONFIG[INIT_STATUS] determines which ranks receive
|
|
the REF / ZQCS. LMC does not send any refreshes / ZQCS's
|
|
when LMC*_CONFIG[INIT_STATUS]=0. */
|
|
uint64_t reset : 1; /**< Reset oneshot pulse for refresh counter,
|
|
and LMC*_OPS_CNT, LMC*_IFB_CNT, and LMC*_DCLK_CNT
|
|
CSR's. SW should write this to a one, then re-write
|
|
it to a zero to cause the reset. */
|
|
uint64_t ecc_adr : 1; /**< Include memory reference address in the ECC calculation
|
|
0=disabled, 1=enabled */
|
|
uint64_t forcewrite : 4; /**< Force the oldest outstanding write to complete after
|
|
having waited for 2^FORCEWRITE CK cycles. 0=disabled. */
|
|
uint64_t idlepower : 3; /**< Enter precharge power-down mode after the memory
|
|
controller has been idle for 2^(2+IDLEPOWER) CK cycles.
|
|
0=disabled.
|
|
This field should only be programmed after initialization.
|
|
LMC*_MODEREG_PARAMS0[PPD] determines whether the DRAM DLL
|
|
is disabled during the precharge power-down. */
|
|
uint64_t pbank_lsb : 4; /**< DIMM address bit select
|
|
Reverting to the explanation for ROW_LSB,
|
|
PBank_LSB would be Row_LSB bit + \#rowbits + \#rankbits
|
|
Decoding for pbank_lsb
|
|
- 0000:DIMM = mem_adr[28] / rank = mem_adr[27] (if RANK_ENA)
|
|
- 0001:DIMM = mem_adr[29] / rank = mem_adr[28] "
|
|
- 0010:DIMM = mem_adr[30] / rank = mem_adr[29] "
|
|
- 0011:DIMM = mem_adr[31] / rank = mem_adr[30] "
|
|
- 0100:DIMM = mem_adr[32] / rank = mem_adr[31] "
|
|
- 0101:DIMM = mem_adr[33] / rank = mem_adr[32] "
|
|
- 0110:DIMM = mem_adr[34] / rank = mem_adr[33] "
|
|
- 0111:DIMM = 0 / rank = mem_adr[34] "
|
|
- 1000-1111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16]
|
|
With rank_ena = 0, pbank_lsb = 2
|
|
With rank_ena = 1, pbank_lsb = 3 */
|
|
uint64_t row_lsb : 3; /**< Row Address bit select
|
|
Encoding used to determine which memory address
|
|
bit position represents the low order DDR ROW address.
|
|
The processor's memory address[34:7] needs to be
|
|
translated to DRAM addresses (bnk,row,col,rank and DIMM)
|
|
and that is a function of the following:
|
|
1. Datapath Width (64)
|
|
2. \# Banks (8)
|
|
3. \# Column Bits of the memory part - spec'd indirectly
|
|
by this register.
|
|
4. \# Row Bits of the memory part - spec'd indirectly
|
|
5. \# Ranks in a DIMM - spec'd by RANK_ENA
|
|
6. \# DIMM's in the system by the register below (PBANK_LSB).
|
|
Decoding for row_lsb
|
|
- 000: row_lsb = mem_adr[14]
|
|
- 001: row_lsb = mem_adr[15]
|
|
- 010: row_lsb = mem_adr[16]
|
|
- 011: row_lsb = mem_adr[17]
|
|
- 100: row_lsb = mem_adr[18]
|
|
- 101: row_lsb = mem_adr[19]
|
|
- 110: row_lsb = mem_adr[20]
|
|
- 111: RESERVED
|
|
For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
DDR3 parts, the column address width = 10, so with
|
|
10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16] */
|
|
uint64_t ecc_ena : 1; /**< ECC Enable: When set will enable the 8b ECC
|
|
check/correct logic. Should be 1 when used with DIMMs
|
|
with ECC. 0, otherwise.
|
|
When this mode is turned on, DQ[71:64]
|
|
on writes, will contain the ECC code generated for
|
|
the 64 bits of data which will
|
|
written in the memory and then later on reads, used
|
|
to check for Single bit error (which will be auto-
|
|
corrected) and Double Bit error (which will be
|
|
reported). When not turned on, DQ[71:64]
|
|
are driven to 0. Please refer to SEC_ERR, DED_ERR,
|
|
LMC*_FADR, LMC*_SCRAMBLED_FADR and LMC*_ECC_SYND registers
|
|
for diagnostics information when there is an error. */
|
|
uint64_t init_start : 1; /**< A 0->1 transition starts the DDR memory sequence that is
|
|
selected by LMC*_CONFIG[SEQUENCE]. This register is a
|
|
oneshot and clears itself each time it is set. */
|
|
#else
|
|
uint64_t init_start : 1;
|
|
uint64_t ecc_ena : 1;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t pbank_lsb : 4;
|
|
uint64_t idlepower : 3;
|
|
uint64_t forcewrite : 4;
|
|
uint64_t ecc_adr : 1;
|
|
uint64_t reset : 1;
|
|
uint64_t ref_zqcs_int : 19;
|
|
uint64_t sequence : 3;
|
|
uint64_t early_dqx : 1;
|
|
uint64_t sref_with_dll : 1;
|
|
uint64_t rank_ena : 1;
|
|
uint64_t rankmask : 4;
|
|
uint64_t mirrmask : 4;
|
|
uint64_t init_status : 4;
|
|
uint64_t early_unload_d0_r0 : 1;
|
|
uint64_t early_unload_d0_r1 : 1;
|
|
uint64_t early_unload_d1_r0 : 1;
|
|
uint64_t early_unload_d1_r1 : 1;
|
|
uint64_t scrz : 1;
|
|
uint64_t reserved_60_63 : 4;
|
|
#endif
|
|
} cn66xx;
|
|
struct cvmx_lmcx_config_cn63xx cn68xx;
|
|
struct cvmx_lmcx_config_cn63xx cn68xxp1;
|
|
struct cvmx_lmcx_config_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_config cvmx_lmcx_config_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_control
|
|
*
|
|
* LMC_CONTROL = LMC Control
|
|
* This register is an assortment of various control fields needed by the memory controller
|
|
*/
|
|
union cvmx_lmcx_control {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_control_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t scramble_ena : 1; /**< When set, will enable the scramble/descramble logic */
|
|
uint64_t thrcnt : 12; /**< Fine Count */
|
|
uint64_t persub : 8; /**< Offset for DFA rate-matching */
|
|
uint64_t thrmax : 4; /**< Fine Rate Matching Max Bucket Size
|
|
0 = Reserved
|
|
In conjunction with the Coarse Rate Matching Logic, the Fine Rate
|
|
Matching Logic gives SW the ability to prioritize DFA Rds over
|
|
L2C Writes. Higher PERSUB values result in a lower DFA Rd
|
|
bandwidth. */
|
|
uint64_t crm_cnt : 5; /**< Coarse Count */
|
|
uint64_t crm_thr : 5; /**< Coarse Rate Matching Threshold */
|
|
uint64_t crm_max : 5; /**< Coarse Rate Matching Max Bucket Size
|
|
0 = Reserved
|
|
The Coarse Rate Matching Logic is used to control the bandwidth
|
|
allocated to DFA Rds. CRM_MAX is subdivided into two regions
|
|
with DFA Rds being preferred over LMC Rd/Wrs when
|
|
CRM_CNT < CRM_THR. CRM_CNT increments by 1 when a DFA Rd is
|
|
slotted and by 2 when a LMC Rd/Wr is slotted, and rolls over
|
|
when CRM_MAX is reached. */
|
|
uint64_t rodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
RD cmd is delayed an additional CK cycle. */
|
|
uint64_t wodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
WR cmd is delayed an additional CK cycle. */
|
|
uint64_t bprch : 2; /**< Back Porch Enable: When set, the turn-on time for
|
|
the default DDR_DQ/DQS drivers is delayed an additional BPRCH
|
|
CK cycles.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = 3 CKs */
|
|
uint64_t ext_zqcs_dis : 1; /**< Disable (external) auto-zqcs calibration
|
|
When clear, LMC runs external ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t int_zqcs_dis : 1; /**< Disable (internal) auto-zqcs calibration
|
|
When clear, LMC runs internal ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t auto_dclkdis : 1; /**< When 1, LMC will automatically shut off its internal
|
|
clock to conserve power when there is no traffic. Note
|
|
that this has no effect on the DDR3 PHY and pads clocks. */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[2:0]=address[9:7] ^ address[14:12]
|
|
else
|
|
bank[2:0]=address[9:7] */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
forcing reads to interrupt. */
|
|
uint64_t nxm_write_en : 1; /**< NXM Write mode
|
|
When clear, LMC discards writes to addresses that don't
|
|
exist in the DRAM (as defined by LMC*_NXM configuration).
|
|
When set, LMC completes writes to addresses that don't
|
|
exist in the DRAM at an aliased address. */
|
|
uint64_t elev_prio_dis : 1; /**< Disable elevate priority logic.
|
|
When set, writes are sent in
|
|
regardless of priority information from L2C. */
|
|
uint64_t inorder_wr : 1; /**< Send writes in order(regardless of priority) */
|
|
uint64_t inorder_rd : 1; /**< Send reads in order (regardless of priority) */
|
|
uint64_t throttle_wr : 1; /**< When set, use at most one IFB for writes */
|
|
uint64_t throttle_rd : 1; /**< When set, use at most one IFB for reads */
|
|
uint64_t fprch2 : 2; /**< Front Porch Enable: When set, the turn-off
|
|
time for the default DDR_DQ/DQS drivers is FPRCH2 CKs earlier.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = RESERVED */
|
|
uint64_t pocas : 1; /**< Enable the Posted CAS feature of DDR3.
|
|
This bit must be set whenever LMC*_MODEREG_PARAMS0[AL]!=0,
|
|
and clear otherwise. */
|
|
uint64_t ddr2t : 1; /**< Turn on the DDR 2T mode. 2 CK cycle window for CMD and
|
|
address. This mode helps relieve setup time pressure
|
|
on the Address and command bus which nominally have
|
|
a very large fanout. Please refer to Micron's tech
|
|
note tn_47_01 titled "DDR2-533 Memory Design Guide
|
|
for Two Dimm Unbuffered Systems" for physical details. */
|
|
uint64_t bwcnt : 1; /**< Bus utilization counter Clear.
|
|
Clears the LMC*_OPS_CNT, LMC*_IFB_CNT, and
|
|
LMC*_DCLK_CNT registers. SW should first write this
|
|
field to a one, then write this field to a zero to
|
|
clear the CSR's. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require address and
|
|
control bits to be registered in the controller. */
|
|
#else
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t bwcnt : 1;
|
|
uint64_t ddr2t : 1;
|
|
uint64_t pocas : 1;
|
|
uint64_t fprch2 : 2;
|
|
uint64_t throttle_rd : 1;
|
|
uint64_t throttle_wr : 1;
|
|
uint64_t inorder_rd : 1;
|
|
uint64_t inorder_wr : 1;
|
|
uint64_t elev_prio_dis : 1;
|
|
uint64_t nxm_write_en : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t auto_dclkdis : 1;
|
|
uint64_t int_zqcs_dis : 1;
|
|
uint64_t ext_zqcs_dis : 1;
|
|
uint64_t bprch : 2;
|
|
uint64_t wodt_bprch : 1;
|
|
uint64_t rodt_bprch : 1;
|
|
uint64_t crm_max : 5;
|
|
uint64_t crm_thr : 5;
|
|
uint64_t crm_cnt : 5;
|
|
uint64_t thrmax : 4;
|
|
uint64_t persub : 8;
|
|
uint64_t thrcnt : 12;
|
|
uint64_t scramble_ena : 1;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_control_s cn61xx;
|
|
struct cvmx_lmcx_control_cn63xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_24_63 : 40;
|
|
uint64_t rodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
RD cmd is delayed an additional CK cycle. */
|
|
uint64_t wodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
WR cmd is delayed an additional CK cycle. */
|
|
uint64_t bprch : 2; /**< Back Porch Enable: When set, the turn-on time for
|
|
the default DDR_DQ/DQS drivers is delayed an additional BPRCH
|
|
CK cycles.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = 3 CKs */
|
|
uint64_t ext_zqcs_dis : 1; /**< Disable (external) auto-zqcs calibration
|
|
When clear, LMC runs external ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t int_zqcs_dis : 1; /**< Disable (internal) auto-zqcs calibration
|
|
When clear, LMC runs internal ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t auto_dclkdis : 1; /**< When 1, LMC will automatically shut off its internal
|
|
clock to conserve power when there is no traffic. Note
|
|
that this has no effect on the DDR3 PHY and pads clocks. */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[2:0]=address[9:7] ^ address[14:12]
|
|
else
|
|
bank[2:0]=address[9:7] */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
forcing reads to interrupt. */
|
|
uint64_t nxm_write_en : 1; /**< NXM Write mode
|
|
When clear, LMC discards writes to addresses that don't
|
|
exist in the DRAM (as defined by LMC*_NXM configuration).
|
|
When set, LMC completes writes to addresses that don't
|
|
exist in the DRAM at an aliased address. */
|
|
uint64_t elev_prio_dis : 1; /**< Disable elevate priority logic.
|
|
When set, writes are sent in
|
|
regardless of priority information from L2C. */
|
|
uint64_t inorder_wr : 1; /**< Send writes in order(regardless of priority) */
|
|
uint64_t inorder_rd : 1; /**< Send reads in order (regardless of priority) */
|
|
uint64_t throttle_wr : 1; /**< When set, use at most one IFB for writes */
|
|
uint64_t throttle_rd : 1; /**< When set, use at most one IFB for reads */
|
|
uint64_t fprch2 : 2; /**< Front Porch Enable: When set, the turn-off
|
|
time for the default DDR_DQ/DQS drivers is FPRCH2 CKs earlier.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = RESERVED */
|
|
uint64_t pocas : 1; /**< Enable the Posted CAS feature of DDR3.
|
|
This bit must be set whenever LMC*_MODEREG_PARAMS0[AL]!=0,
|
|
and clear otherwise. */
|
|
uint64_t ddr2t : 1; /**< Turn on the DDR 2T mode. 2 CK cycle window for CMD and
|
|
address. This mode helps relieve setup time pressure
|
|
on the Address and command bus which nominally have
|
|
a very large fanout. Please refer to Micron's tech
|
|
note tn_47_01 titled "DDR2-533 Memory Design Guide
|
|
for Two Dimm Unbuffered Systems" for physical details. */
|
|
uint64_t bwcnt : 1; /**< Bus utilization counter Clear.
|
|
Clears the LMC*_OPS_CNT, LMC*_IFB_CNT, and
|
|
LMC*_DCLK_CNT registers. SW should first write this
|
|
field to a one, then write this field to a zero to
|
|
clear the CSR's. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require address and
|
|
control bits to be registered in the controller. */
|
|
#else
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t bwcnt : 1;
|
|
uint64_t ddr2t : 1;
|
|
uint64_t pocas : 1;
|
|
uint64_t fprch2 : 2;
|
|
uint64_t throttle_rd : 1;
|
|
uint64_t throttle_wr : 1;
|
|
uint64_t inorder_rd : 1;
|
|
uint64_t inorder_wr : 1;
|
|
uint64_t elev_prio_dis : 1;
|
|
uint64_t nxm_write_en : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t auto_dclkdis : 1;
|
|
uint64_t int_zqcs_dis : 1;
|
|
uint64_t ext_zqcs_dis : 1;
|
|
uint64_t bprch : 2;
|
|
uint64_t wodt_bprch : 1;
|
|
uint64_t rodt_bprch : 1;
|
|
uint64_t reserved_24_63 : 40;
|
|
#endif
|
|
} cn63xx;
|
|
struct cvmx_lmcx_control_cn63xx cn63xxp1;
|
|
struct cvmx_lmcx_control_cn66xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t scramble_ena : 1; /**< When set, will enable the scramble/descramble logic */
|
|
uint64_t reserved_24_62 : 39;
|
|
uint64_t rodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
RD cmd is delayed an additional CK cycle. */
|
|
uint64_t wodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
WR cmd is delayed an additional CK cycle. */
|
|
uint64_t bprch : 2; /**< Back Porch Enable: When set, the turn-on time for
|
|
the default DDR_DQ/DQS drivers is delayed an additional BPRCH
|
|
CK cycles.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = 3 CKs */
|
|
uint64_t ext_zqcs_dis : 1; /**< Disable (external) auto-zqcs calibration
|
|
When clear, LMC runs external ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t int_zqcs_dis : 1; /**< Disable (internal) auto-zqcs calibration
|
|
When clear, LMC runs internal ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t auto_dclkdis : 1; /**< When 1, LMC will automatically shut off its internal
|
|
clock to conserve power when there is no traffic. Note
|
|
that this has no effect on the DDR3 PHY and pads clocks. */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[2:0]=address[9:7] ^ address[14:12]
|
|
else
|
|
bank[2:0]=address[9:7] */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
forcing reads to interrupt. */
|
|
uint64_t nxm_write_en : 1; /**< NXM Write mode
|
|
When clear, LMC discards writes to addresses that don't
|
|
exist in the DRAM (as defined by LMC*_NXM configuration).
|
|
When set, LMC completes writes to addresses that don't
|
|
exist in the DRAM at an aliased address. */
|
|
uint64_t elev_prio_dis : 1; /**< Disable elevate priority logic.
|
|
When set, writes are sent in
|
|
regardless of priority information from L2C. */
|
|
uint64_t inorder_wr : 1; /**< Send writes in order(regardless of priority) */
|
|
uint64_t inorder_rd : 1; /**< Send reads in order (regardless of priority) */
|
|
uint64_t throttle_wr : 1; /**< When set, use at most one IFB for writes */
|
|
uint64_t throttle_rd : 1; /**< When set, use at most one IFB for reads */
|
|
uint64_t fprch2 : 2; /**< Front Porch Enable: When set, the turn-off
|
|
time for the default DDR_DQ/DQS drivers is FPRCH2 CKs earlier.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = RESERVED */
|
|
uint64_t pocas : 1; /**< Enable the Posted CAS feature of DDR3.
|
|
This bit must be set whenever LMC*_MODEREG_PARAMS0[AL]!=0,
|
|
and clear otherwise. */
|
|
uint64_t ddr2t : 1; /**< Turn on the DDR 2T mode. 2 CK cycle window for CMD and
|
|
address. This mode helps relieve setup time pressure
|
|
on the Address and command bus which nominally have
|
|
a very large fanout. Please refer to Micron's tech
|
|
note tn_47_01 titled "DDR2-533 Memory Design Guide
|
|
for Two Dimm Unbuffered Systems" for physical details. */
|
|
uint64_t bwcnt : 1; /**< Bus utilization counter Clear.
|
|
Clears the LMC*_OPS_CNT, LMC*_IFB_CNT, and
|
|
LMC*_DCLK_CNT registers. SW should first write this
|
|
field to a one, then write this field to a zero to
|
|
clear the CSR's. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require address and
|
|
control bits to be registered in the controller. */
|
|
#else
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t bwcnt : 1;
|
|
uint64_t ddr2t : 1;
|
|
uint64_t pocas : 1;
|
|
uint64_t fprch2 : 2;
|
|
uint64_t throttle_rd : 1;
|
|
uint64_t throttle_wr : 1;
|
|
uint64_t inorder_rd : 1;
|
|
uint64_t inorder_wr : 1;
|
|
uint64_t elev_prio_dis : 1;
|
|
uint64_t nxm_write_en : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t auto_dclkdis : 1;
|
|
uint64_t int_zqcs_dis : 1;
|
|
uint64_t ext_zqcs_dis : 1;
|
|
uint64_t bprch : 2;
|
|
uint64_t wodt_bprch : 1;
|
|
uint64_t rodt_bprch : 1;
|
|
uint64_t reserved_24_62 : 39;
|
|
uint64_t scramble_ena : 1;
|
|
#endif
|
|
} cn66xx;
|
|
struct cvmx_lmcx_control_cn68xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_63_63 : 1;
|
|
uint64_t thrcnt : 12; /**< Fine Count */
|
|
uint64_t persub : 8; /**< Offset for DFA rate-matching */
|
|
uint64_t thrmax : 4; /**< Fine Rate Matching Max Bucket Size
|
|
0 = Reserved
|
|
In conjunction with the Coarse Rate Matching Logic, the Fine Rate
|
|
Matching Logic gives SW the ability to prioritize DFA Rds over
|
|
L2C Writes. Higher PERSUB values result in a lower DFA Rd
|
|
bandwidth. */
|
|
uint64_t crm_cnt : 5; /**< Coarse Count */
|
|
uint64_t crm_thr : 5; /**< Coarse Rate Matching Threshold */
|
|
uint64_t crm_max : 5; /**< Coarse Rate Matching Max Bucket Size
|
|
0 = Reserved
|
|
The Coarse Rate Matching Logic is used to control the bandwidth
|
|
allocated to DFA Rds. CRM_MAX is subdivided into two regions
|
|
with DFA Rds being preferred over LMC Rd/Wrs when
|
|
CRM_CNT < CRM_THR. CRM_CNT increments by 1 when a DFA Rd is
|
|
slotted and by 2 when a LMC Rd/Wr is slotted, and rolls over
|
|
when CRM_MAX is reached. */
|
|
uint64_t rodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
RD cmd is delayed an additional CK cycle. */
|
|
uint64_t wodt_bprch : 1; /**< When set, the turn-off time for the ODT pin during a
|
|
WR cmd is delayed an additional CK cycle. */
|
|
uint64_t bprch : 2; /**< Back Porch Enable: When set, the turn-on time for
|
|
the default DDR_DQ/DQS drivers is delayed an additional BPRCH
|
|
CK cycles.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = 3 CKs */
|
|
uint64_t ext_zqcs_dis : 1; /**< Disable (external) auto-zqcs calibration
|
|
When clear, LMC runs external ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t int_zqcs_dis : 1; /**< Disable (internal) auto-zqcs calibration
|
|
When clear, LMC runs internal ZQ calibration
|
|
every LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t auto_dclkdis : 1; /**< When 1, LMC will automatically shut off its internal
|
|
clock to conserve power when there is no traffic. Note
|
|
that this has no effect on the DDR3 PHY and pads clocks. */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[2:0]=address[9:7] ^ address[14:12]
|
|
else
|
|
bank[2:0]=address[9:7] */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
forcing reads to interrupt. */
|
|
uint64_t nxm_write_en : 1; /**< NXM Write mode
|
|
When clear, LMC discards writes to addresses that don't
|
|
exist in the DRAM (as defined by LMC*_NXM configuration).
|
|
When set, LMC completes writes to addresses that don't
|
|
exist in the DRAM at an aliased address. */
|
|
uint64_t elev_prio_dis : 1; /**< Disable elevate priority logic.
|
|
When set, writes are sent in
|
|
regardless of priority information from L2C. */
|
|
uint64_t inorder_wr : 1; /**< Send writes in order(regardless of priority) */
|
|
uint64_t inorder_rd : 1; /**< Send reads in order (regardless of priority) */
|
|
uint64_t throttle_wr : 1; /**< When set, use at most one IFB for writes */
|
|
uint64_t throttle_rd : 1; /**< When set, use at most one IFB for reads */
|
|
uint64_t fprch2 : 2; /**< Front Porch Enable: When set, the turn-off
|
|
time for the default DDR_DQ/DQS drivers is FPRCH2 CKs earlier.
|
|
00 = 0 CKs
|
|
01 = 1 CKs
|
|
10 = 2 CKs
|
|
11 = RESERVED */
|
|
uint64_t pocas : 1; /**< Enable the Posted CAS feature of DDR3.
|
|
This bit must be set whenever LMC*_MODEREG_PARAMS0[AL]!=0,
|
|
and clear otherwise. */
|
|
uint64_t ddr2t : 1; /**< Turn on the DDR 2T mode. 2 CK cycle window for CMD and
|
|
address. This mode helps relieve setup time pressure
|
|
on the Address and command bus which nominally have
|
|
a very large fanout. Please refer to Micron's tech
|
|
note tn_47_01 titled "DDR2-533 Memory Design Guide
|
|
for Two Dimm Unbuffered Systems" for physical details. */
|
|
uint64_t bwcnt : 1; /**< Bus utilization counter Clear.
|
|
Clears the LMC*_OPS_CNT, LMC*_IFB_CNT, and
|
|
LMC*_DCLK_CNT registers. SW should first write this
|
|
field to a one, then write this field to a zero to
|
|
clear the CSR's. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require address and
|
|
control bits to be registered in the controller. */
|
|
#else
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t bwcnt : 1;
|
|
uint64_t ddr2t : 1;
|
|
uint64_t pocas : 1;
|
|
uint64_t fprch2 : 2;
|
|
uint64_t throttle_rd : 1;
|
|
uint64_t throttle_wr : 1;
|
|
uint64_t inorder_rd : 1;
|
|
uint64_t inorder_wr : 1;
|
|
uint64_t elev_prio_dis : 1;
|
|
uint64_t nxm_write_en : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t auto_dclkdis : 1;
|
|
uint64_t int_zqcs_dis : 1;
|
|
uint64_t ext_zqcs_dis : 1;
|
|
uint64_t bprch : 2;
|
|
uint64_t wodt_bprch : 1;
|
|
uint64_t rodt_bprch : 1;
|
|
uint64_t crm_max : 5;
|
|
uint64_t crm_thr : 5;
|
|
uint64_t crm_cnt : 5;
|
|
uint64_t thrmax : 4;
|
|
uint64_t persub : 8;
|
|
uint64_t thrcnt : 12;
|
|
uint64_t reserved_63_63 : 1;
|
|
#endif
|
|
} cn68xx;
|
|
struct cvmx_lmcx_control_cn68xx cn68xxp1;
|
|
struct cvmx_lmcx_control_cn66xx cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_control cvmx_lmcx_control_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ctl
|
|
*
|
|
* LMC_CTL = LMC Control
|
|
* This register is an assortment of various control fields needed by the memory controller
|
|
*/
|
|
union cvmx_lmcx_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 4; /**< DDR nctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pulldns. */
|
|
uint64_t ddr__pctl : 4; /**< DDR pctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pullup. */
|
|
uint64_t slow_scf : 1; /**< Should be cleared to zero */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[n:0]=address[n+7:7] ^ address[n+7+5:7+5]
|
|
else
|
|
bank[n:0]=address[n+7:7]
|
|
where n=1 for a 4 bank part and n=2 for an 8 bank part */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
allowing reads to interrupt. */
|
|
uint64_t pll_div2 : 1; /**< PLL Div2. */
|
|
uint64_t pll_bypass : 1; /**< PLL Bypass. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require Write
|
|
data to be registered in the controller. */
|
|
uint64_t r2r_slot : 1; /**< R2R Slot Enable: When set, all read-to-read trans
|
|
will slot an additional 1 cycle data bus bubble to
|
|
avoid DQ/DQS bus contention. This is only a CYA bit,
|
|
in case the "built-in" DIMM and RANK crossing logic
|
|
which should auto-detect and perfectly slot
|
|
read-to-reads to the same DIMM/RANK. */
|
|
uint64_t inorder_mwf : 1; /**< Reads as zero */
|
|
uint64_t inorder_mrf : 1; /**< Always clear to zero */
|
|
uint64_t reserved_10_11 : 2;
|
|
uint64_t fprch2 : 1; /**< Front Porch Enable: When set, the turn-off
|
|
time for the DDR_DQ/DQS drivers is 1 dclk earlier.
|
|
This bit should typically be set. */
|
|
uint64_t bprch : 1; /**< Back Porch Enable: When set, the turn-on time for
|
|
the DDR_DQ/DQS drivers is delayed an additional DCLK
|
|
cycle. This should be set to one whenever both SILO_HC
|
|
and SILO_QC are set. */
|
|
uint64_t sil_lat : 2; /**< SILO Latency: On reads, determines how many additional
|
|
dclks to wait (on top of TCL+1+TSKW) before pulling
|
|
data out of the pad silos.
|
|
- 00: illegal
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: illegal
|
|
This should always be set to 1. */
|
|
uint64_t tskw : 2; /**< This component is a representation of total BOARD
|
|
DELAY on DQ (used in the controller to determine the
|
|
R->W spacing to avoid DQS/DQ bus conflicts). Enter
|
|
the largest of the per byte Board delay
|
|
- 00: 0 dclk
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: 3 dclks */
|
|
uint64_t qs_dic : 2; /**< DDR2 Termination Resistor Setting
|
|
A non Zero value in this register
|
|
enables the On Die Termination (ODT) in DDR parts.
|
|
These two bits are loaded into the RTT
|
|
portion of the EMRS register bits A6 & A2. If DDR2's
|
|
termination (for the memory's DQ/DQS/DM pads) is not
|
|
desired, set it to 00. If it is, chose between
|
|
01 for 75 ohm and 10 for 150 ohm termination.
|
|
00 = ODT Disabled
|
|
01 = 75 ohm Termination
|
|
10 = 150 ohm Termination
|
|
11 = 50 ohm Termination
|
|
Octeon, on writes, by default, drives the 4/8 ODT
|
|
pins (64/128b mode) based on what the masks
|
|
(LMC_WODT_CTL) are programmed to.
|
|
LMC_DDR2_CTL->ODT_ENA enables Octeon to drive ODT pins
|
|
for READS. LMC_RODT_CTL needs to be programmed based
|
|
on the system's needs for ODT. */
|
|
uint64_t dic : 2; /**< Drive Strength Control:
|
|
DIC[0] is
|
|
loaded into the Extended Mode Register (EMRS) A1 bit
|
|
during initialization.
|
|
0 = Normal
|
|
1 = Reduced
|
|
DIC[1] is used to load into EMRS
|
|
bit 10 - DQSN Enable/Disable field. By default, we
|
|
program the DDR's to drive the DQSN also. Set it to
|
|
1 if DQSN should be Hi-Z.
|
|
0 - DQSN Enable
|
|
1 - DQSN Disable */
|
|
#else
|
|
uint64_t dic : 2;
|
|
uint64_t qs_dic : 2;
|
|
uint64_t tskw : 2;
|
|
uint64_t sil_lat : 2;
|
|
uint64_t bprch : 1;
|
|
uint64_t fprch2 : 1;
|
|
uint64_t reserved_10_11 : 2;
|
|
uint64_t inorder_mrf : 1;
|
|
uint64_t inorder_mwf : 1;
|
|
uint64_t r2r_slot : 1;
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t pll_bypass : 1;
|
|
uint64_t pll_div2 : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t slow_scf : 1;
|
|
uint64_t ddr__pctl : 4;
|
|
uint64_t ddr__nctl : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ctl_cn30xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 4; /**< DDR nctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pulldns. */
|
|
uint64_t ddr__pctl : 4; /**< DDR pctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pullup. */
|
|
uint64_t slow_scf : 1; /**< 1=SCF has pass1 latency, 0=SCF has 1 cycle lower latency
|
|
when compared to pass1 */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[n:0]=address[n+7:7] ^ address[n+7+5:7+5]
|
|
else
|
|
bank[n:0]=address[n+7:7]
|
|
where n=1 for a 4 bank part and n=2 for an 8 bank part */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
allowing reads to interrupt. */
|
|
uint64_t pll_div2 : 1; /**< PLL Div2. */
|
|
uint64_t pll_bypass : 1; /**< PLL Bypass. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require Write
|
|
data to be registered in the controller. */
|
|
uint64_t r2r_slot : 1; /**< R2R Slot Enable: When set, all read-to-read trans
|
|
will slot an additional 1 cycle data bus bubble to
|
|
avoid DQ/DQS bus contention. This is only a CYA bit,
|
|
in case the "built-in" DIMM and RANK crossing logic
|
|
which should auto-detect and perfectly slot
|
|
read-to-reads to the same DIMM/RANK. */
|
|
uint64_t inorder_mwf : 1; /**< Reads as zero */
|
|
uint64_t inorder_mrf : 1; /**< Always set to zero */
|
|
uint64_t dreset : 1; /**< Dclk domain reset. The reset signal that is used by the
|
|
Dclk domain is (DRESET || ECLK_RESET). */
|
|
uint64_t mode32b : 1; /**< 32b data Path Mode
|
|
Set to 1 if we use only 32 DQ pins
|
|
0 for 16b DQ mode. */
|
|
uint64_t fprch2 : 1; /**< Front Porch Enable: When set, the turn-off
|
|
time for the DDR_DQ/DQS drivers is 1 dclk earlier.
|
|
This bit should typically be set. */
|
|
uint64_t bprch : 1; /**< Back Porch Enable: When set, the turn-on time for
|
|
the DDR_DQ/DQS drivers is delayed an additional DCLK
|
|
cycle. This should be set to one whenever both SILO_HC
|
|
and SILO_QC are set. */
|
|
uint64_t sil_lat : 2; /**< SILO Latency: On reads, determines how many additional
|
|
dclks to wait (on top of TCL+1+TSKW) before pulling
|
|
data out of the pad silos.
|
|
- 00: illegal
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: illegal
|
|
This should always be set to 1. */
|
|
uint64_t tskw : 2; /**< This component is a representation of total BOARD
|
|
DELAY on DQ (used in the controller to determine the
|
|
R->W spacing to avoid DQS/DQ bus conflicts). Enter
|
|
the largest of the per byte Board delay
|
|
- 00: 0 dclk
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: 3 dclks */
|
|
uint64_t qs_dic : 2; /**< QS Drive Strength Control (DDR1):
|
|
& DDR2 Termination Resistor Setting
|
|
When in DDR2, a non Zero value in this register
|
|
enables the On Die Termination (ODT) in DDR parts.
|
|
These two bits are loaded into the RTT
|
|
portion of the EMRS register bits A6 & A2. If DDR2's
|
|
termination (for the memory's DQ/DQS/DM pads) is not
|
|
desired, set it to 00. If it is, chose between
|
|
01 for 75 ohm and 10 for 150 ohm termination.
|
|
00 = ODT Disabled
|
|
01 = 75 ohm Termination
|
|
10 = 150 ohm Termination
|
|
11 = 50 ohm Termination
|
|
Octeon, on writes, by default, drives the 8 ODT
|
|
pins based on what the masks (LMC_WODT_CTL1 & 2)
|
|
are programmed to. LMC_DDR2_CTL->ODT_ENA
|
|
enables Octeon to drive ODT pins for READS.
|
|
LMC_RODT_CTL needs to be programmed based on
|
|
the system's needs for ODT. */
|
|
uint64_t dic : 2; /**< Drive Strength Control:
|
|
For DDR-I/II Mode, DIC[0] is
|
|
loaded into the Extended Mode Register (EMRS) A1 bit
|
|
during initialization. (see DDR-I data sheet EMRS
|
|
description)
|
|
0 = Normal
|
|
1 = Reduced
|
|
For DDR-II Mode, DIC[1] is used to load into EMRS
|
|
bit 10 - DQSN Enable/Disable field. By default, we
|
|
program the DDR's to drive the DQSN also. Set it to
|
|
1 if DQSN should be Hi-Z.
|
|
0 - DQSN Enable
|
|
1 - DQSN Disable */
|
|
#else
|
|
uint64_t dic : 2;
|
|
uint64_t qs_dic : 2;
|
|
uint64_t tskw : 2;
|
|
uint64_t sil_lat : 2;
|
|
uint64_t bprch : 1;
|
|
uint64_t fprch2 : 1;
|
|
uint64_t mode32b : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t inorder_mrf : 1;
|
|
uint64_t inorder_mwf : 1;
|
|
uint64_t r2r_slot : 1;
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t pll_bypass : 1;
|
|
uint64_t pll_div2 : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t slow_scf : 1;
|
|
uint64_t ddr__pctl : 4;
|
|
uint64_t ddr__nctl : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn30xx;
|
|
struct cvmx_lmcx_ctl_cn30xx cn31xx;
|
|
struct cvmx_lmcx_ctl_cn38xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 4; /**< DDR nctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pulldns. */
|
|
uint64_t ddr__pctl : 4; /**< DDR pctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pullup. */
|
|
uint64_t slow_scf : 1; /**< 1=SCF has pass1 latency, 0=SCF has 1 cycle lower latency
|
|
when compared to pass1
|
|
NOTE - This bit has NO effect in PASS1 */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[n:0]=address[n+7:7] ^ address[n+7+5:7+5]
|
|
else
|
|
bank[n:0]=address[n+7:7]
|
|
where n=1 for a 4 bank part and n=2 for an 8 bank part */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
allowing reads to interrupt. */
|
|
uint64_t reserved_16_17 : 2;
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require Write
|
|
data to be registered in the controller. */
|
|
uint64_t r2r_slot : 1; /**< R2R Slot Enable: When set, all read-to-read trans
|
|
will slot an additional 1 cycle data bus bubble to
|
|
avoid DQ/DQS bus contention. This is only a CYA bit,
|
|
in case the "built-in" DIMM and RANK crossing logic
|
|
which should auto-detect and perfectly slot
|
|
read-to-reads to the same DIMM/RANK. */
|
|
uint64_t inorder_mwf : 1; /**< When set, forces LMC_MWF (writes) into strict, in-order
|
|
mode. When clear, writes may be serviced out of order
|
|
(optimized to keep multiple banks active).
|
|
This bit is ONLY to be set at power-on and
|
|
should not be set for normal use.
|
|
NOTE: For PASS1, set as follows:
|
|
DDR-I -> 1
|
|
DDR-II -> 0
|
|
For Pass2, this bit is RA0, write ignore (this feature
|
|
is permanently disabled) */
|
|
uint64_t inorder_mrf : 1; /**< When set, forces LMC_MRF (reads) into strict, in-order
|
|
mode. When clear, reads may be serviced out of order
|
|
(optimized to keep multiple banks active).
|
|
This bit is ONLY to be set at power-on and
|
|
should not be set for normal use.
|
|
NOTE: For PASS1, set as follows:
|
|
DDR-I -> 1
|
|
DDR-II -> 0
|
|
For Pass2, this bit should be written ZERO for
|
|
DDR I & II */
|
|
uint64_t set_zero : 1; /**< Reserved. Always Set this Bit to Zero */
|
|
uint64_t mode128b : 1; /**< 128b data Path Mode
|
|
Set to 1 if we use all 128 DQ pins
|
|
0 for 64b DQ mode. */
|
|
uint64_t fprch2 : 1; /**< Front Porch Enable: When set, the turn-off
|
|
time for the DDR_DQ/DQS drivers is 1 dclk earlier.
|
|
This bit should typically be set. */
|
|
uint64_t bprch : 1; /**< Back Porch Enable: When set, the turn-on time for
|
|
the DDR_DQ/DQS drivers is delayed an additional DCLK
|
|
cycle. This should be set to one whenever both SILO_HC
|
|
and SILO_QC are set. */
|
|
uint64_t sil_lat : 2; /**< SILO Latency: On reads, determines how many additional
|
|
dclks to wait (on top of TCL+1+TSKW) before pulling
|
|
data out of the pad silos.
|
|
- 00: illegal
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: illegal
|
|
This should always be set to 1. */
|
|
uint64_t tskw : 2; /**< This component is a representation of total BOARD
|
|
DELAY on DQ (used in the controller to determine the
|
|
R->W spacing to avoid DQS/DQ bus conflicts). Enter
|
|
the largest of the per byte Board delay
|
|
- 00: 0 dclk
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: 3 dclks */
|
|
uint64_t qs_dic : 2; /**< QS Drive Strength Control (DDR1):
|
|
& DDR2 Termination Resistor Setting
|
|
When in DDR2, a non Zero value in this register
|
|
enables the On Die Termination (ODT) in DDR parts.
|
|
These two bits are loaded into the RTT
|
|
portion of the EMRS register bits A6 & A2. If DDR2's
|
|
termination (for the memory's DQ/DQS/DM pads) is not
|
|
desired, set it to 00. If it is, chose between
|
|
01 for 75 ohm and 10 for 150 ohm termination.
|
|
00 = ODT Disabled
|
|
01 = 75 ohm Termination
|
|
10 = 150 ohm Termination
|
|
11 = 50 ohm Termination
|
|
Octeon, on writes, by default, drives the 4/8 ODT
|
|
pins (64/128b mode) based on what the masks
|
|
(LMC_WODT_CTL) are programmed to.
|
|
LMC_DDR2_CTL->ODT_ENA enables Octeon to drive ODT pins
|
|
for READS. LMC_RODT_CTL needs to be programmed based
|
|
on the system's needs for ODT. */
|
|
uint64_t dic : 2; /**< Drive Strength Control:
|
|
For DDR-I/II Mode, DIC[0] is
|
|
loaded into the Extended Mode Register (EMRS) A1 bit
|
|
during initialization. (see DDR-I data sheet EMRS
|
|
description)
|
|
0 = Normal
|
|
1 = Reduced
|
|
For DDR-II Mode, DIC[1] is used to load into EMRS
|
|
bit 10 - DQSN Enable/Disable field. By default, we
|
|
program the DDR's to drive the DQSN also. Set it to
|
|
1 if DQSN should be Hi-Z.
|
|
0 - DQSN Enable
|
|
1 - DQSN Disable */
|
|
#else
|
|
uint64_t dic : 2;
|
|
uint64_t qs_dic : 2;
|
|
uint64_t tskw : 2;
|
|
uint64_t sil_lat : 2;
|
|
uint64_t bprch : 1;
|
|
uint64_t fprch2 : 1;
|
|
uint64_t mode128b : 1;
|
|
uint64_t set_zero : 1;
|
|
uint64_t inorder_mrf : 1;
|
|
uint64_t inorder_mwf : 1;
|
|
uint64_t r2r_slot : 1;
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t reserved_16_17 : 2;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t slow_scf : 1;
|
|
uint64_t ddr__pctl : 4;
|
|
uint64_t ddr__nctl : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn38xx;
|
|
struct cvmx_lmcx_ctl_cn38xx cn38xxp2;
|
|
struct cvmx_lmcx_ctl_cn50xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 4; /**< DDR nctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pulldns. */
|
|
uint64_t ddr__pctl : 4; /**< DDR pctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pullup. */
|
|
uint64_t slow_scf : 1; /**< Should be cleared to zero */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[n:0]=address[n+7:7] ^ address[n+7+5:7+5]
|
|
else
|
|
bank[n:0]=address[n+7:7]
|
|
where n=1 for a 4 bank part and n=2 for an 8 bank part */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
allowing reads to interrupt. */
|
|
uint64_t reserved_17_17 : 1;
|
|
uint64_t pll_bypass : 1; /**< PLL Bypass. */
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require Write
|
|
data to be registered in the controller. */
|
|
uint64_t r2r_slot : 1; /**< R2R Slot Enable: When set, all read-to-read trans
|
|
will slot an additional 1 cycle data bus bubble to
|
|
avoid DQ/DQS bus contention. This is only a CYA bit,
|
|
in case the "built-in" DIMM and RANK crossing logic
|
|
which should auto-detect and perfectly slot
|
|
read-to-reads to the same DIMM/RANK. */
|
|
uint64_t inorder_mwf : 1; /**< Reads as zero */
|
|
uint64_t inorder_mrf : 1; /**< Always clear to zero */
|
|
uint64_t dreset : 1; /**< Dclk domain reset. The reset signal that is used by the
|
|
Dclk domain is (DRESET || ECLK_RESET). */
|
|
uint64_t mode32b : 1; /**< 32b data Path Mode
|
|
Set to 1 if we use 32 DQ pins
|
|
0 for 16b DQ mode. */
|
|
uint64_t fprch2 : 1; /**< Front Porch Enable: When set, the turn-off
|
|
time for the DDR_DQ/DQS drivers is 1 dclk earlier.
|
|
This bit should typically be set. */
|
|
uint64_t bprch : 1; /**< Back Porch Enable: When set, the turn-on time for
|
|
the DDR_DQ/DQS drivers is delayed an additional DCLK
|
|
cycle. This should be set to one whenever both SILO_HC
|
|
and SILO_QC are set. */
|
|
uint64_t sil_lat : 2; /**< SILO Latency: On reads, determines how many additional
|
|
dclks to wait (on top of TCL+1+TSKW) before pulling
|
|
data out of the pad silos.
|
|
- 00: illegal
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: illegal
|
|
This should always be set to 1. */
|
|
uint64_t tskw : 2; /**< This component is a representation of total BOARD
|
|
DELAY on DQ (used in the controller to determine the
|
|
R->W spacing to avoid DQS/DQ bus conflicts). Enter
|
|
the largest of the per byte Board delay
|
|
- 00: 0 dclk
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: 3 dclks */
|
|
uint64_t qs_dic : 2; /**< DDR2 Termination Resistor Setting
|
|
When in DDR2, a non Zero value in this register
|
|
enables the On Die Termination (ODT) in DDR parts.
|
|
These two bits are loaded into the RTT
|
|
portion of the EMRS register bits A6 & A2. If DDR2's
|
|
termination (for the memory's DQ/DQS/DM pads) is not
|
|
desired, set it to 00. If it is, chose between
|
|
01 for 75 ohm and 10 for 150 ohm termination.
|
|
00 = ODT Disabled
|
|
01 = 75 ohm Termination
|
|
10 = 150 ohm Termination
|
|
11 = 50 ohm Termination
|
|
Octeon, on writes, by default, drives the ODT
|
|
pins based on what the masks
|
|
(LMC_WODT_CTL) are programmed to.
|
|
LMC_DDR2_CTL->ODT_ENA enables Octeon to drive ODT pins
|
|
for READS. LMC_RODT_CTL needs to be programmed based
|
|
on the system's needs for ODT. */
|
|
uint64_t dic : 2; /**< Drive Strength Control:
|
|
DIC[0] is
|
|
loaded into the Extended Mode Register (EMRS) A1 bit
|
|
during initialization.
|
|
0 = Normal
|
|
1 = Reduced
|
|
DIC[1] is used to load into EMRS
|
|
bit 10 - DQSN Enable/Disable field. By default, we
|
|
program the DDR's to drive the DQSN also. Set it to
|
|
1 if DQSN should be Hi-Z.
|
|
0 - DQSN Enable
|
|
1 - DQSN Disable */
|
|
#else
|
|
uint64_t dic : 2;
|
|
uint64_t qs_dic : 2;
|
|
uint64_t tskw : 2;
|
|
uint64_t sil_lat : 2;
|
|
uint64_t bprch : 1;
|
|
uint64_t fprch2 : 1;
|
|
uint64_t mode32b : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t inorder_mrf : 1;
|
|
uint64_t inorder_mwf : 1;
|
|
uint64_t r2r_slot : 1;
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t pll_bypass : 1;
|
|
uint64_t reserved_17_17 : 1;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t slow_scf : 1;
|
|
uint64_t ddr__pctl : 4;
|
|
uint64_t ddr__nctl : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn50xx;
|
|
struct cvmx_lmcx_ctl_cn52xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 4; /**< DDR nctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pulldns. */
|
|
uint64_t ddr__pctl : 4; /**< DDR pctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pullup. */
|
|
uint64_t slow_scf : 1; /**< Always clear to zero */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[n:0]=address[n+7:7] ^ address[n+7+5:7+5]
|
|
else
|
|
bank[n:0]=address[n+7:7]
|
|
where n=1 for a 4 bank part and n=2 for an 8 bank part */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
allowing reads to interrupt. */
|
|
uint64_t reserved_16_17 : 2;
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require Write
|
|
data to be registered in the controller. */
|
|
uint64_t r2r_slot : 1; /**< R2R Slot Enable: When set, all read-to-read trans
|
|
will slot an additional 1 cycle data bus bubble to
|
|
avoid DQ/DQS bus contention. This is only a CYA bit,
|
|
in case the "built-in" DIMM and RANK crossing logic
|
|
which should auto-detect and perfectly slot
|
|
read-to-reads to the same DIMM/RANK. */
|
|
uint64_t inorder_mwf : 1; /**< Reads as zero */
|
|
uint64_t inorder_mrf : 1; /**< Always set to zero */
|
|
uint64_t dreset : 1; /**< MBZ
|
|
THIS IS OBSOLETE. Use LMC_DLL_CTL[DRESET] instead. */
|
|
uint64_t mode32b : 1; /**< 32b data Path Mode
|
|
Set to 1 if we use only 32 DQ pins
|
|
0 for 64b DQ mode. */
|
|
uint64_t fprch2 : 1; /**< Front Porch Enable: When set, the turn-off
|
|
time for the DDR_DQ/DQS drivers is 1 dclk earlier.
|
|
This bit should typically be set. */
|
|
uint64_t bprch : 1; /**< Back Porch Enable: When set, the turn-on time for
|
|
the DDR_DQ/DQS drivers is delayed an additional DCLK
|
|
cycle. This should be set to one whenever both SILO_HC
|
|
and SILO_QC are set. */
|
|
uint64_t sil_lat : 2; /**< SILO Latency: On reads, determines how many additional
|
|
dclks to wait (on top of TCL+1+TSKW) before pulling
|
|
data out of the pad silos.
|
|
- 00: illegal
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: illegal
|
|
This should always be set to 1.
|
|
THIS IS OBSOLETE. Use READ_LEVEL_RANK instead. */
|
|
uint64_t tskw : 2; /**< This component is a representation of total BOARD
|
|
DELAY on DQ (used in the controller to determine the
|
|
R->W spacing to avoid DQS/DQ bus conflicts). Enter
|
|
the largest of the per byte Board delay
|
|
- 00: 0 dclk
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: 3 dclks
|
|
THIS IS OBSOLETE. Use READ_LEVEL_RANK instead. */
|
|
uint64_t qs_dic : 2; /**< DDR2 Termination Resistor Setting
|
|
When in DDR2, a non Zero value in this register
|
|
enables the On Die Termination (ODT) in DDR parts.
|
|
These two bits are loaded into the RTT
|
|
portion of the EMRS register bits A6 & A2. If DDR2's
|
|
termination (for the memory's DQ/DQS/DM pads) is not
|
|
desired, set it to 00. If it is, chose between
|
|
01 for 75 ohm and 10 for 150 ohm termination.
|
|
00 = ODT Disabled
|
|
01 = 75 ohm Termination
|
|
10 = 150 ohm Termination
|
|
11 = 50 ohm Termination
|
|
Octeon, on writes, by default, drives the 4/8 ODT
|
|
pins (64/128b mode) based on what the masks
|
|
(LMC_WODT_CTL0 & 1) are programmed to.
|
|
LMC_DDR2_CTL->ODT_ENA enables Octeon to drive ODT pins
|
|
for READS. LMC_RODT_CTL needs to be programmed based
|
|
on the system's needs for ODT. */
|
|
uint64_t dic : 2; /**< Drive Strength Control:
|
|
DIC[0] is
|
|
loaded into the Extended Mode Register (EMRS) A1 bit
|
|
during initialization.
|
|
0 = Normal
|
|
1 = Reduced
|
|
DIC[1] is used to load into EMRS
|
|
bit 10 - DQSN Enable/Disable field. By default, we
|
|
program the DDR's to drive the DQSN also. Set it to
|
|
1 if DQSN should be Hi-Z.
|
|
0 - DQSN Enable
|
|
1 - DQSN Disable */
|
|
#else
|
|
uint64_t dic : 2;
|
|
uint64_t qs_dic : 2;
|
|
uint64_t tskw : 2;
|
|
uint64_t sil_lat : 2;
|
|
uint64_t bprch : 1;
|
|
uint64_t fprch2 : 1;
|
|
uint64_t mode32b : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t inorder_mrf : 1;
|
|
uint64_t inorder_mwf : 1;
|
|
uint64_t r2r_slot : 1;
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t reserved_16_17 : 2;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t slow_scf : 1;
|
|
uint64_t ddr__pctl : 4;
|
|
uint64_t ddr__nctl : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn52xx;
|
|
struct cvmx_lmcx_ctl_cn52xx cn52xxp1;
|
|
struct cvmx_lmcx_ctl_cn52xx cn56xx;
|
|
struct cvmx_lmcx_ctl_cn52xx cn56xxp1;
|
|
struct cvmx_lmcx_ctl_cn58xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 4; /**< DDR nctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pulldns. */
|
|
uint64_t ddr__pctl : 4; /**< DDR pctl from compensation circuit
|
|
The encoded value on this will adjust the drive strength
|
|
of the DDR DQ pullup. */
|
|
uint64_t slow_scf : 1; /**< Should be cleared to zero */
|
|
uint64_t xor_bank : 1; /**< If (XOR_BANK == 1), then
|
|
bank[n:0]=address[n+7:7] ^ address[n+7+5:7+5]
|
|
else
|
|
bank[n:0]=address[n+7:7]
|
|
where n=1 for a 4 bank part and n=2 for an 8 bank part */
|
|
uint64_t max_write_batch : 4; /**< Maximum number of consecutive writes to service before
|
|
allowing reads to interrupt. */
|
|
uint64_t reserved_16_17 : 2;
|
|
uint64_t rdimm_ena : 1; /**< Registered DIMM Enable - When set allows the use
|
|
of JEDEC Registered DIMMs which require Write
|
|
data to be registered in the controller. */
|
|
uint64_t r2r_slot : 1; /**< R2R Slot Enable: When set, all read-to-read trans
|
|
will slot an additional 1 cycle data bus bubble to
|
|
avoid DQ/DQS bus contention. This is only a CYA bit,
|
|
in case the "built-in" DIMM and RANK crossing logic
|
|
which should auto-detect and perfectly slot
|
|
read-to-reads to the same DIMM/RANK. */
|
|
uint64_t inorder_mwf : 1; /**< Reads as zero */
|
|
uint64_t inorder_mrf : 1; /**< Always clear to zero */
|
|
uint64_t dreset : 1; /**< Dclk domain reset. The reset signal that is used by the
|
|
Dclk domain is (DRESET || ECLK_RESET). */
|
|
uint64_t mode128b : 1; /**< 128b data Path Mode
|
|
Set to 1 if we use all 128 DQ pins
|
|
0 for 64b DQ mode. */
|
|
uint64_t fprch2 : 1; /**< Front Porch Enable: When set, the turn-off
|
|
time for the DDR_DQ/DQS drivers is 1 dclk earlier.
|
|
This bit should typically be set. */
|
|
uint64_t bprch : 1; /**< Back Porch Enable: When set, the turn-on time for
|
|
the DDR_DQ/DQS drivers is delayed an additional DCLK
|
|
cycle. This should be set to one whenever both SILO_HC
|
|
and SILO_QC are set. */
|
|
uint64_t sil_lat : 2; /**< SILO Latency: On reads, determines how many additional
|
|
dclks to wait (on top of TCL+1+TSKW) before pulling
|
|
data out of the pad silos.
|
|
- 00: illegal
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: illegal
|
|
This should always be set to 1. */
|
|
uint64_t tskw : 2; /**< This component is a representation of total BOARD
|
|
DELAY on DQ (used in the controller to determine the
|
|
R->W spacing to avoid DQS/DQ bus conflicts). Enter
|
|
the largest of the per byte Board delay
|
|
- 00: 0 dclk
|
|
- 01: 1 dclks
|
|
- 10: 2 dclks
|
|
- 11: 3 dclks */
|
|
uint64_t qs_dic : 2; /**< DDR2 Termination Resistor Setting
|
|
A non Zero value in this register
|
|
enables the On Die Termination (ODT) in DDR parts.
|
|
These two bits are loaded into the RTT
|
|
portion of the EMRS register bits A6 & A2. If DDR2's
|
|
termination (for the memory's DQ/DQS/DM pads) is not
|
|
desired, set it to 00. If it is, chose between
|
|
01 for 75 ohm and 10 for 150 ohm termination.
|
|
00 = ODT Disabled
|
|
01 = 75 ohm Termination
|
|
10 = 150 ohm Termination
|
|
11 = 50 ohm Termination
|
|
Octeon, on writes, by default, drives the 4/8 ODT
|
|
pins (64/128b mode) based on what the masks
|
|
(LMC_WODT_CTL) are programmed to.
|
|
LMC_DDR2_CTL->ODT_ENA enables Octeon to drive ODT pins
|
|
for READS. LMC_RODT_CTL needs to be programmed based
|
|
on the system's needs for ODT. */
|
|
uint64_t dic : 2; /**< Drive Strength Control:
|
|
DIC[0] is
|
|
loaded into the Extended Mode Register (EMRS) A1 bit
|
|
during initialization.
|
|
0 = Normal
|
|
1 = Reduced
|
|
DIC[1] is used to load into EMRS
|
|
bit 10 - DQSN Enable/Disable field. By default, we
|
|
program the DDR's to drive the DQSN also. Set it to
|
|
1 if DQSN should be Hi-Z.
|
|
0 - DQSN Enable
|
|
1 - DQSN Disable */
|
|
#else
|
|
uint64_t dic : 2;
|
|
uint64_t qs_dic : 2;
|
|
uint64_t tskw : 2;
|
|
uint64_t sil_lat : 2;
|
|
uint64_t bprch : 1;
|
|
uint64_t fprch2 : 1;
|
|
uint64_t mode128b : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t inorder_mrf : 1;
|
|
uint64_t inorder_mwf : 1;
|
|
uint64_t r2r_slot : 1;
|
|
uint64_t rdimm_ena : 1;
|
|
uint64_t reserved_16_17 : 2;
|
|
uint64_t max_write_batch : 4;
|
|
uint64_t xor_bank : 1;
|
|
uint64_t slow_scf : 1;
|
|
uint64_t ddr__pctl : 4;
|
|
uint64_t ddr__nctl : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn58xx;
|
|
struct cvmx_lmcx_ctl_cn58xx cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ctl cvmx_lmcx_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ctl1
|
|
*
|
|
* LMC_CTL1 = LMC Control1
|
|
* This register is an assortment of various control fields needed by the memory controller
|
|
*/
|
|
union cvmx_lmcx_ctl1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ctl1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_21_63 : 43;
|
|
uint64_t ecc_adr : 1; /**< Include memory reference address in the ECC calculation
|
|
0=disabled, 1=enabled */
|
|
uint64_t forcewrite : 4; /**< Force the oldest outstanding write to complete after
|
|
having waited for 2^FORCEWRITE cycles. 0=disabled. */
|
|
uint64_t idlepower : 3; /**< Enter power-down mode after the memory controller has
|
|
been idle for 2^(2+IDLEPOWER) cycles. 0=disabled. */
|
|
uint64_t sequence : 3; /**< Instruction sequence that is run after a 0->1 transition
|
|
on LMC_MEM_CFG0[INIT_START].
|
|
0=DDR2 power-up/init, 1=read-leveling
|
|
2=self-refresh entry, 3=self-refresh exit,
|
|
4=power-down entry, 5=power-down exit, 6=7=illegal */
|
|
uint64_t sil_mode : 1; /**< Read Silo mode. 0=envelope, 1=self-timed. */
|
|
uint64_t dcc_enable : 1; /**< Duty Cycle Corrector Enable.
|
|
0=disable, 1=enable
|
|
If the memory part does not support DCC, then this bit
|
|
must be set to 0. */
|
|
uint64_t reserved_2_7 : 6;
|
|
uint64_t data_layout : 2; /**< Logical data layout per DQ byte lane:
|
|
In 32b mode, this setting has no effect and the data
|
|
layout DQ[35:0] is the following:
|
|
[E[3:0], D[31:24], D[23:16], D[15:8], D[7:0]]
|
|
In 16b mode, the DQ[35:0] layouts are the following:
|
|
0 - [0[3:0], 0[7:0], [0[7:2], E[1:0]], D[15:8], D[7:0]]
|
|
1 - [0[3:0], [0[7:2], E[1:0]], D[15:8], D[7:0], 0[7:0]]
|
|
2 - [[0[1:0], E[1:0]], D[15:8], D[7:0], 0[7:0], 0[7:0]]
|
|
where E means ecc, D means data, and 0 means unused
|
|
(ignored on reads and written as 0 on writes) */
|
|
#else
|
|
uint64_t data_layout : 2;
|
|
uint64_t reserved_2_7 : 6;
|
|
uint64_t dcc_enable : 1;
|
|
uint64_t sil_mode : 1;
|
|
uint64_t sequence : 3;
|
|
uint64_t idlepower : 3;
|
|
uint64_t forcewrite : 4;
|
|
uint64_t ecc_adr : 1;
|
|
uint64_t reserved_21_63 : 43;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ctl1_cn30xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_2_63 : 62;
|
|
uint64_t data_layout : 2; /**< Logical data layout per DQ byte lane:
|
|
In 32b mode, this setting has no effect and the data
|
|
layout DQ[35:0] is the following:
|
|
[E[3:0], D[31:24], D[23:16], D[15:8], D[7:0]]
|
|
In 16b mode, the DQ[35:0] layouts are the following:
|
|
0 - [0[3:0], 0[7:0], [0[7:2], E[1:0]], D[15:8], D[7:0]]
|
|
1 - [0[3:0], [0[7:2], E[1:0]], D[15:8], D[7:0], 0[7:0]]
|
|
2 - [[0[1:0], E[1:0]], D[15:8], D[7:0], 0[7:0], 0[7:0]]
|
|
where E means ecc, D means data, and 0 means unused
|
|
(ignored on reads and written as 0 on writes) */
|
|
#else
|
|
uint64_t data_layout : 2;
|
|
uint64_t reserved_2_63 : 62;
|
|
#endif
|
|
} cn30xx;
|
|
struct cvmx_lmcx_ctl1_cn50xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_10_63 : 54;
|
|
uint64_t sil_mode : 1; /**< Read Silo mode. 0=envelope, 1=self-timed. */
|
|
uint64_t dcc_enable : 1; /**< Duty Cycle Corrector Enable.
|
|
0=disable, 1=enable
|
|
If the memory part does not support DCC, then this bit
|
|
must be set to 0. */
|
|
uint64_t reserved_2_7 : 6;
|
|
uint64_t data_layout : 2; /**< Logical data layout per DQ byte lane:
|
|
In 32b mode, this setting has no effect and the data
|
|
layout DQ[35:0] is the following:
|
|
[E[3:0], D[31:24], D[23:16], D[15:8], D[7:0]]
|
|
In 16b mode, the DQ[35:0] layouts are the following:
|
|
0 - [0[3:0], 0[7:0], [0[7:2], E[1:0]], D[15:8], D[7:0]]
|
|
1 - [0[3:0], [0[7:2], E[1:0]], D[15:8], D[7:0], 0[7:0]]
|
|
2 - [[0[1:0], E[1:0]], D[15:8], D[7:0], 0[7:0], 0[7:0]]
|
|
where E means ecc, D means data, and 0 means unused
|
|
(ignored on reads and written as 0 on writes) */
|
|
#else
|
|
uint64_t data_layout : 2;
|
|
uint64_t reserved_2_7 : 6;
|
|
uint64_t dcc_enable : 1;
|
|
uint64_t sil_mode : 1;
|
|
uint64_t reserved_10_63 : 54;
|
|
#endif
|
|
} cn50xx;
|
|
struct cvmx_lmcx_ctl1_cn52xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_21_63 : 43;
|
|
uint64_t ecc_adr : 1; /**< Include memory reference address in the ECC calculation
|
|
0=disabled, 1=enabled */
|
|
uint64_t forcewrite : 4; /**< Force the oldest outstanding write to complete after
|
|
having waited for 2^FORCEWRITE cycles. 0=disabled. */
|
|
uint64_t idlepower : 3; /**< Enter power-down mode after the memory controller has
|
|
been idle for 2^(2+IDLEPOWER) cycles. 0=disabled. */
|
|
uint64_t sequence : 3; /**< Instruction sequence that is run after a 0->1 transition
|
|
on LMC_MEM_CFG0[INIT_START].
|
|
0=DDR2 power-up/init, 1=read-leveling
|
|
2=self-refresh entry, 3=self-refresh exit,
|
|
4=power-down entry, 5=power-down exit, 6=7=illegal */
|
|
uint64_t sil_mode : 1; /**< Read Silo mode. 0=envelope, 1=self-timed. */
|
|
uint64_t dcc_enable : 1; /**< Duty Cycle Corrector Enable.
|
|
0=disable, 1=enable
|
|
If the memory part does not support DCC, then this bit
|
|
must be set to 0. */
|
|
uint64_t reserved_0_7 : 8;
|
|
#else
|
|
uint64_t reserved_0_7 : 8;
|
|
uint64_t dcc_enable : 1;
|
|
uint64_t sil_mode : 1;
|
|
uint64_t sequence : 3;
|
|
uint64_t idlepower : 3;
|
|
uint64_t forcewrite : 4;
|
|
uint64_t ecc_adr : 1;
|
|
uint64_t reserved_21_63 : 43;
|
|
#endif
|
|
} cn52xx;
|
|
struct cvmx_lmcx_ctl1_cn52xx cn52xxp1;
|
|
struct cvmx_lmcx_ctl1_cn52xx cn56xx;
|
|
struct cvmx_lmcx_ctl1_cn52xx cn56xxp1;
|
|
struct cvmx_lmcx_ctl1_cn58xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_10_63 : 54;
|
|
uint64_t sil_mode : 1; /**< Read Silo mode. 0=envelope, 1=self-timed. */
|
|
uint64_t dcc_enable : 1; /**< Duty Cycle Corrector Enable.
|
|
0=disable, 1=enable
|
|
If the memory part does not support DCC, then this bit
|
|
must be set to 0. */
|
|
uint64_t reserved_0_7 : 8;
|
|
#else
|
|
uint64_t reserved_0_7 : 8;
|
|
uint64_t dcc_enable : 1;
|
|
uint64_t sil_mode : 1;
|
|
uint64_t reserved_10_63 : 54;
|
|
#endif
|
|
} cn58xx;
|
|
struct cvmx_lmcx_ctl1_cn58xx cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ctl1 cvmx_lmcx_ctl1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dclk_cnt
|
|
*
|
|
* LMC_DCLK_CNT = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_dclk_cnt {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dclk_cnt_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t dclkcnt : 64; /**< Performance Counter
|
|
64-bit counter that increments every CK cycle */
|
|
#else
|
|
uint64_t dclkcnt : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dclk_cnt_s cn61xx;
|
|
struct cvmx_lmcx_dclk_cnt_s cn63xx;
|
|
struct cvmx_lmcx_dclk_cnt_s cn63xxp1;
|
|
struct cvmx_lmcx_dclk_cnt_s cn66xx;
|
|
struct cvmx_lmcx_dclk_cnt_s cn68xx;
|
|
struct cvmx_lmcx_dclk_cnt_s cn68xxp1;
|
|
struct cvmx_lmcx_dclk_cnt_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_dclk_cnt cvmx_lmcx_dclk_cnt_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dclk_cnt_hi
|
|
*
|
|
* LMC_DCLK_CNT_HI = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_dclk_cnt_hi {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t dclkcnt_hi : 32; /**< Performance Counter that counts dclks
|
|
Upper 32-bits of a 64-bit counter. */
|
|
#else
|
|
uint64_t dclkcnt_hi : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn30xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn31xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn38xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn38xxp2;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn50xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn52xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn52xxp1;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn56xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn56xxp1;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn58xx;
|
|
struct cvmx_lmcx_dclk_cnt_hi_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_dclk_cnt_hi cvmx_lmcx_dclk_cnt_hi_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dclk_cnt_lo
|
|
*
|
|
* LMC_DCLK_CNT_LO = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_dclk_cnt_lo {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t dclkcnt_lo : 32; /**< Performance Counter that counts dclks
|
|
Lower 32-bits of a 64-bit counter. */
|
|
#else
|
|
uint64_t dclkcnt_lo : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn30xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn31xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn38xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn38xxp2;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn50xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn52xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn52xxp1;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn56xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn56xxp1;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn58xx;
|
|
struct cvmx_lmcx_dclk_cnt_lo_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_dclk_cnt_lo cvmx_lmcx_dclk_cnt_lo_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dclk_ctl
|
|
*
|
|
* LMC_DCLK_CTL = LMC DCLK generation control
|
|
*
|
|
*
|
|
* Notes:
|
|
* This CSR is only relevant for LMC1. LMC0_DCLK_CTL is not used.
|
|
*
|
|
*/
|
|
union cvmx_lmcx_dclk_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dclk_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_8_63 : 56;
|
|
uint64_t off90_ena : 1; /**< 0=use global DCLK (i.e. the PLL) directly for LMC1
|
|
1=use the 90 degree DCLK DLL to offset LMC1 DCLK */
|
|
uint64_t dclk90_byp : 1; /**< 0=90 degree DCLK DLL uses sampled delay from LMC0
|
|
1=90 degree DCLK DLL uses DCLK90_VLU
|
|
See DCLK90_VLU. */
|
|
uint64_t dclk90_ld : 1; /**< The 90 degree DCLK DLL samples the delay setting
|
|
from LMC0's DLL when this field transitions 0->1 */
|
|
uint64_t dclk90_vlu : 5; /**< Manual open-loop delay setting.
|
|
The LMC1 90 degree DCLK DLL uses DCLK90_VLU rather
|
|
than the delay setting sampled from LMC0 when
|
|
DCLK90_BYP=1. */
|
|
#else
|
|
uint64_t dclk90_vlu : 5;
|
|
uint64_t dclk90_ld : 1;
|
|
uint64_t dclk90_byp : 1;
|
|
uint64_t off90_ena : 1;
|
|
uint64_t reserved_8_63 : 56;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dclk_ctl_s cn56xx;
|
|
struct cvmx_lmcx_dclk_ctl_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_dclk_ctl cvmx_lmcx_dclk_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ddr2_ctl
|
|
*
|
|
* LMC_DDR2_CTL = LMC DDR2 & DLL Control Register
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ddr2_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ddr2_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t bank8 : 1; /**< For 8 bank DDR2 parts
|
|
1 - DDR2 parts have 8 internal banks (BA is 3 bits
|
|
wide).
|
|
0 - DDR2 parts have 4 internal banks (BA is 2 bits
|
|
wide). */
|
|
uint64_t burst8 : 1; /**< 8-burst mode.
|
|
1 - DDR data transfer happens in burst of 8
|
|
0 - DDR data transfer happens in burst of 4
|
|
BURST8 should be set when DDR2T is set
|
|
to minimize the command bandwidth loss. */
|
|
uint64_t addlat : 3; /**< Additional Latency for posted CAS
|
|
When Posted CAS is on, this configures the additional
|
|
latency. This should be set to
|
|
1 .. LMC_MEM_CFG1[TRCD]-2
|
|
(Note the implication that posted CAS should not
|
|
be used when tRCD is two.) */
|
|
uint64_t pocas : 1; /**< Enable the Posted CAS feature of DDR2. */
|
|
uint64_t bwcnt : 1; /**< Bus utilization counter Clear.
|
|
Clears the LMC_OPS_CNT_*, LMC_IFB_CNT_*, and
|
|
LMC_DCLK_CNT_* registers. SW should first write this
|
|
field to a one, then write this field to a zero to
|
|
clear the CSR's. */
|
|
uint64_t twr : 3; /**< DDR Write Recovery time (tWR). Last Wr Brst to Pre delay
|
|
This is not a direct encoding of the value. Its
|
|
programmed as below per DDR2 spec. The decimal number
|
|
on the right is RNDUP(tWR(ns) / tCYC(ns))
|
|
TYP=15ns
|
|
- 000: RESERVED
|
|
- 001: 2
|
|
- 010: 3
|
|
- 011: 4
|
|
- 100: 5
|
|
- 101: 6
|
|
- 110: 7
|
|
- 111: 8 */
|
|
uint64_t silo_hc : 1; /**< Delays the read sample window by a Half Cycle. */
|
|
uint64_t ddr_eof : 4; /**< Early Fill Counter Init.
|
|
L2 needs to know a few cycle before a fill completes so
|
|
it can get its Control pipe started (for better overall
|
|
performance). This counter contains an init value which
|
|
is a function of Eclk/Dclk ratio to account for the
|
|
asynchronous boundary between L2 cache and the DRAM
|
|
controller. This init value will
|
|
determine when to safely let the L2 know that a fill
|
|
termination is coming up.
|
|
Set DDR_EOF according to the following rule:
|
|
eclkFreq/dclkFreq = dclkPeriod/eclkPeriod = RATIO
|
|
RATIO < 6/6 -> illegal
|
|
6/6 <= RATIO < 6/5 -> DDR_EOF=3
|
|
6/5 <= RATIO < 6/4 -> DDR_EOF=3
|
|
6/4 <= RATIO < 6/3 -> DDR_EOF=2
|
|
6/3 <= RATIO < 6/2 -> DDR_EOF=1
|
|
6/2 <= RATIO < 6/1 -> DDR_EOF=0
|
|
6/1 <= RATIO -> DDR_EOF=0 */
|
|
uint64_t tfaw : 5; /**< tFAW - Cycles = RNDUP[tFAW(ns)/tcyc(ns)] - 1
|
|
Four Access Window time. Relevant only in DDR2 AND in
|
|
8-bank parts.
|
|
tFAW = 5'b0 in DDR2-4bank
|
|
tFAW = RNDUP[tFAW(ns)/tcyc(ns)] - 1
|
|
in DDR2-8bank */
|
|
uint64_t crip_mode : 1; /**< Cripple Mode - When set, the LMC allows only
|
|
1 inflight transaction (.vs. 8 in normal mode).
|
|
This bit is ONLY to be set at power-on and
|
|
should not be set for normal use. */
|
|
uint64_t ddr2t : 1; /**< Turn on the DDR 2T mode. 2 cycle window for CMD and
|
|
address. This mode helps relieve setup time pressure
|
|
on the Address and command bus which nominally have
|
|
a very large fanout. Please refer to Micron's tech
|
|
note tn_47_01 titled "DDR2-533 Memory Design Guide
|
|
for Two Dimm Unbuffered Systems" for physical details.
|
|
BURST8 should be set when DDR2T is set to minimize
|
|
add/cmd loss. */
|
|
uint64_t odt_ena : 1; /**< Enable Obsolete ODT on Reads
|
|
Obsolete Read ODT wiggles DDR_ODT_* pins on reads.
|
|
Should normally be cleared to zero.
|
|
When this is on, the following fields must also be
|
|
programmed:
|
|
LMC_CTL->QS_DIC - programs the termination value
|
|
LMC_RODT_CTL - programs the ODT I/O mask for Reads */
|
|
uint64_t qdll_ena : 1; /**< DDR Quad DLL Enable: A 0->1 transition on this bit after
|
|
DCLK init sequence will reset the DDR 90 DLL. Should
|
|
happen at startup before any activity in DDR.
|
|
DRESET should be asserted before and for 10 usec
|
|
following the 0->1 transition on QDLL_ENA. */
|
|
uint64_t dll90_vlu : 5; /**< Contains the open loop setting value for the DDR90 delay
|
|
line. */
|
|
uint64_t dll90_byp : 1; /**< DDR DLL90 Bypass: When set, the DDR90 DLL is to be
|
|
bypassed and the setting is defined by DLL90_VLU */
|
|
uint64_t rdqs : 1; /**< DDR2 RDQS mode. When set, configures memory subsystem to
|
|
use unidirectional DQS pins. RDQS/DM - Rcv & DQS - Xmit */
|
|
uint64_t ddr2 : 1; /**< Should be set */
|
|
#else
|
|
uint64_t ddr2 : 1;
|
|
uint64_t rdqs : 1;
|
|
uint64_t dll90_byp : 1;
|
|
uint64_t dll90_vlu : 5;
|
|
uint64_t qdll_ena : 1;
|
|
uint64_t odt_ena : 1;
|
|
uint64_t ddr2t : 1;
|
|
uint64_t crip_mode : 1;
|
|
uint64_t tfaw : 5;
|
|
uint64_t ddr_eof : 4;
|
|
uint64_t silo_hc : 1;
|
|
uint64_t twr : 3;
|
|
uint64_t bwcnt : 1;
|
|
uint64_t pocas : 1;
|
|
uint64_t addlat : 3;
|
|
uint64_t burst8 : 1;
|
|
uint64_t bank8 : 1;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ddr2_ctl_cn30xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t bank8 : 1; /**< For 8 bank DDR2 parts
|
|
1 - DDR2 parts have 8 internal banks (BA is 3 bits
|
|
wide).
|
|
0 - DDR2 parts have 4 internal banks (BA is 2 bits
|
|
wide). */
|
|
uint64_t burst8 : 1; /**< 8-burst mode.
|
|
1 - DDR data transfer happens in burst of 8
|
|
0 - DDR data transfer happens in burst of 4
|
|
BURST8 should be set when DDR2T is set to minimize
|
|
add/cmd bandwidth loss. */
|
|
uint64_t addlat : 3; /**< Additional Latency for posted CAS
|
|
When Posted CAS is on, this configures the additional
|
|
latency. This should be set to
|
|
1 .. LMC_MEM_CFG1[TRCD]-2
|
|
(Note the implication that posted CAS should not
|
|
be used when tRCD is two.) */
|
|
uint64_t pocas : 1; /**< Enable the Posted CAS feature of DDR2. */
|
|
uint64_t bwcnt : 1; /**< Bus utilization counter Clear.
|
|
Clears the LMC_OPS_CNT_*, LMC_IFB_CNT_*, and
|
|
LMC_DCLK_CNT_* registers. SW should first write this
|
|
field to a one, then write this field to a zero to
|
|
clear the CSR's. */
|
|
uint64_t twr : 3; /**< DDR Write Recovery time (tWR). Last Wr Brst to Pre delay
|
|
This is not a direct encoding of the value. Its
|
|
programmed as below per DDR2 spec. The decimal number
|
|
on the right is RNDUP(tWR(ns) / tCYC(ns))
|
|
TYP=15ns
|
|
- 000: RESERVED
|
|
- 001: 2
|
|
- 010: 3
|
|
- 011: 4
|
|
- 100: 5
|
|
- 101: 6
|
|
- 110-111: RESERVED */
|
|
uint64_t silo_hc : 1; /**< Delays the read sample window by a Half Cycle. */
|
|
uint64_t ddr_eof : 4; /**< Early Fill Counter Init.
|
|
L2 needs to know a few cycle before a fill completes so
|
|
it can get its Control pipe started (for better overall
|
|
performance). This counter contains an init value which
|
|
is a function of Eclk/Dclk ratio to account for the
|
|
asynchronous boundary between L2 cache and the DRAM
|
|
controller. This init value will
|
|
determine when to safely let the L2 know that a fill
|
|
termination is coming up.
|
|
DDR_EOF = RNDUP (DCLK period/Eclk Period). If the ratio
|
|
is above 3, set DDR_EOF to 3.
|
|
DCLK/ECLK period DDR_EOF
|
|
Less than 1 1
|
|
Less than 2 2
|
|
More than 2 3 */
|
|
uint64_t tfaw : 5; /**< tFAW - Cycles = RNDUP[tFAW(ns)/tcyc(ns)] - 1
|
|
Four Access Window time. Relevant only in
|
|
8-bank parts.
|
|
TFAW = 5'b0 for DDR2-4bank
|
|
TFAW = RNDUP[tFAW(ns)/tcyc(ns)] - 1 in DDR2-8bank */
|
|
uint64_t crip_mode : 1; /**< Cripple Mode - When set, the LMC allows only
|
|
1 inflight transaction (.vs. 8 in normal mode).
|
|
This bit is ONLY to be set at power-on and
|
|
should not be set for normal use. */
|
|
uint64_t ddr2t : 1; /**< Turn on the DDR 2T mode. 2 cycle window for CMD and
|
|
address. This mode helps relieve setup time pressure
|
|
on the Address and command bus which nominally have
|
|
a very large fanout. Please refer to Micron's tech
|
|
note tn_47_01 titled "DDR2-533 Memory Design Guide
|
|
for Two Dimm Unbuffered Systems" for physical details.
|
|
BURST8 should be used when DDR2T is set to minimize
|
|
add/cmd bandwidth loss. */
|
|
uint64_t odt_ena : 1; /**< Enable ODT for DDR2 on Reads
|
|
When this is on, the following fields must also be
|
|
programmed:
|
|
LMC_CTL->QS_DIC - programs the termination value
|
|
LMC_RODT_CTL - programs the ODT I/O mask for writes
|
|
Program as 0 for DDR1 mode and ODT needs to be off
|
|
on Octeon Reads */
|
|
uint64_t qdll_ena : 1; /**< DDR Quad DLL Enable: A 0->1 transition on this bit after
|
|
erst deassertion will reset the DDR 90 DLL. Should
|
|
happen at startup before any activity in DDR. */
|
|
uint64_t dll90_vlu : 5; /**< Contains the open loop setting value for the DDR90 delay
|
|
line. */
|
|
uint64_t dll90_byp : 1; /**< DDR DLL90 Bypass: When set, the DDR90 DLL is to be
|
|
bypassed and the setting is defined by DLL90_VLU */
|
|
uint64_t reserved_1_1 : 1;
|
|
uint64_t ddr2 : 1; /**< DDR2 Enable: When set, configures memory subsystem for
|
|
DDR-II SDRAMs. */
|
|
#else
|
|
uint64_t ddr2 : 1;
|
|
uint64_t reserved_1_1 : 1;
|
|
uint64_t dll90_byp : 1;
|
|
uint64_t dll90_vlu : 5;
|
|
uint64_t qdll_ena : 1;
|
|
uint64_t odt_ena : 1;
|
|
uint64_t ddr2t : 1;
|
|
uint64_t crip_mode : 1;
|
|
uint64_t tfaw : 5;
|
|
uint64_t ddr_eof : 4;
|
|
uint64_t silo_hc : 1;
|
|
uint64_t twr : 3;
|
|
uint64_t bwcnt : 1;
|
|
uint64_t pocas : 1;
|
|
uint64_t addlat : 3;
|
|
uint64_t burst8 : 1;
|
|
uint64_t bank8 : 1;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn30xx;
|
|
struct cvmx_lmcx_ddr2_ctl_cn30xx cn31xx;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn38xx;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn38xxp2;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn50xx;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn52xx;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn56xx;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn56xxp1;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn58xx;
|
|
struct cvmx_lmcx_ddr2_ctl_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ddr2_ctl cvmx_lmcx_ddr2_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ddr_pll_ctl
|
|
*
|
|
* LMC_DDR_PLL_CTL = LMC DDR PLL control
|
|
*
|
|
*
|
|
* Notes:
|
|
* DDR PLL Bringup sequence:
|
|
* 1. Write CLKF, DDR_PS_EN, DFM_PS_EN, DIFFAMP, CPS, CPB.
|
|
* If test mode is going to be activated, then also write jtg__ddr_pll_tm_en1, jtg__ddr_pll_tm_en2, jtg__ddr_pll_tm_en3,
|
|
* jtg__ddr_pll_tm_en4, jtg__dfa_pll_tm_en1, jtg__dfa_pll_tm_en2, jtg__dfa_pll_tm_en3, jtg__dfa_pll_tm_en4, JTAG_TEST_MODE
|
|
* 2. Wait 128 ref clock cycles (7680 rclk cycles)
|
|
* 3. Write 1 to RESET_N
|
|
* 4. Wait 1152 ref clocks (1152*16 rclk cycles)
|
|
* 5. Write 0 to DDR_DIV_RESET and DFM_DIV_RESET
|
|
* 6. Wait 10 ref clock cycles (160 rclk cycles) before bringing up the DDR interface
|
|
* If test mode is going to be activated, wait an additional 8191 ref clocks (8191*16 rclk cycles) to allow PLL
|
|
* clock alignment
|
|
*/
|
|
union cvmx_lmcx_ddr_pll_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_27_63 : 37;
|
|
uint64_t jtg_test_mode : 1; /**< JTAG Test Mode
|
|
Clock alignment between DCLK & REFCLK as well as FCLK &
|
|
REFCLK can only be performed after the ddr_pll_divider_reset
|
|
is deasserted. SW need to wait atleast 10 reference clock
|
|
cycles after deasserting pll_divider_reset before asserting
|
|
LMC(0)_DDR_PLL_CTL[JTG_TEST_MODE]. During alignment (which can
|
|
take upto 160 microseconds) DCLK and FCLK can exhibit some
|
|
high frequency pulses. Therefore, all bring up activities in
|
|
that clock domain need to be delayed (when the chip operates
|
|
in jtg_test_mode) by about 160 microseconds to ensure that
|
|
lock is achieved. */
|
|
uint64_t dfm_div_reset : 1; /**< DFM postscalar divider reset */
|
|
uint64_t dfm_ps_en : 3; /**< DFM postscalar divide ratio
|
|
Determines the DFM CK speed.
|
|
0x0 : Divide LMC+DFM PLL output by 1
|
|
0x1 : Divide LMC+DFM PLL output by 2
|
|
0x2 : Divide LMC+DFM PLL output by 3
|
|
0x3 : Divide LMC+DFM PLL output by 4
|
|
0x4 : Divide LMC+DFM PLL output by 6
|
|
0x5 : Divide LMC+DFM PLL output by 8
|
|
0x6 : Divide LMC+DFM PLL output by 12
|
|
0x7 : Divide LMC+DFM PLL output by 12
|
|
DFM_PS_EN is not used when DFM_DIV_RESET = 1 */
|
|
uint64_t ddr_div_reset : 1; /**< DDR postscalar divider reset */
|
|
uint64_t ddr_ps_en : 3; /**< DDR postscalar divide ratio
|
|
Determines the LMC CK speed.
|
|
0x0 : Divide LMC+DFM PLL output by 1
|
|
0x1 : Divide LMC+DFM PLL output by 2
|
|
0x2 : Divide LMC+DFM PLL output by 3
|
|
0x3 : Divide LMC+DFM PLL output by 4
|
|
0x4 : Divide LMC+DFM PLL output by 6
|
|
0x5 : Divide LMC+DFM PLL output by 8
|
|
0x6 : Divide LMC+DFM PLL output by 12
|
|
0x7 : Divide LMC+DFM PLL output by 12
|
|
DDR_PS_EN is not used when DDR_DIV_RESET = 1 */
|
|
uint64_t diffamp : 4; /**< PLL diffamp input transconductance */
|
|
uint64_t cps : 3; /**< PLL charge-pump current */
|
|
uint64_t cpb : 3; /**< PLL charge-pump current */
|
|
uint64_t reset_n : 1; /**< PLL reset */
|
|
uint64_t clkf : 7; /**< Multiply reference by CLKF
|
|
32 <= CLKF <= 64
|
|
LMC+DFM PLL frequency = 50 * CLKF
|
|
min = 1.6 GHz, max = 3.2 GHz */
|
|
#else
|
|
uint64_t clkf : 7;
|
|
uint64_t reset_n : 1;
|
|
uint64_t cpb : 3;
|
|
uint64_t cps : 3;
|
|
uint64_t diffamp : 4;
|
|
uint64_t ddr_ps_en : 3;
|
|
uint64_t ddr_div_reset : 1;
|
|
uint64_t dfm_ps_en : 3;
|
|
uint64_t dfm_div_reset : 1;
|
|
uint64_t jtg_test_mode : 1;
|
|
uint64_t reserved_27_63 : 37;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cn61xx;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cn63xx;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cn63xxp1;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cn66xx;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cn68xx;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_ddr_pll_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_ddr_pll_ctl cvmx_lmcx_ddr_pll_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_delay_cfg
|
|
*
|
|
* LMC_DELAY_CFG = Open-loop delay line settings
|
|
*
|
|
*
|
|
* Notes:
|
|
* The DQ bits add OUTGOING delay only to dq, dqs_[p,n], cb, cbs_[p,n], dqm. Delay is approximately
|
|
* 50-80ps per setting depending on process/voltage. There is no need to add incoming delay since by
|
|
* default all strobe bits are delayed internally by 90 degrees (as was always the case in previous
|
|
* passes and past chips.
|
|
*
|
|
* The CMD add delay to all command bits DDR_RAS, DDR_CAS, DDR_A<15:0>, DDR_BA<2:0>, DDR_n_CS<1:0>_L,
|
|
* DDR_WE, DDR_CKE and DDR_ODT_<7:0>. Again, delay is 50-80ps per tap.
|
|
*
|
|
* The CLK bits add delay to all clock signals DDR_CK_<5:0>_P and DDR_CK_<5:0>_N. Again, delay is
|
|
* 50-80ps per tap.
|
|
*
|
|
* The usage scenario is the following: There is too much delay on command signals and setup on command
|
|
* is not met. The user can then delay the clock until setup is met.
|
|
*
|
|
* At the same time though, dq/dqs should be delayed because there is also a DDR spec tying dqs with
|
|
* clock. If clock is too much delayed with respect to dqs, writes will start to fail.
|
|
*
|
|
* This scheme should eliminate the board need of adding routing delay to clock signals to make high
|
|
* frequencies work.
|
|
*/
|
|
union cvmx_lmcx_delay_cfg {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_delay_cfg_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_15_63 : 49;
|
|
uint64_t dq : 5; /**< Setting for DQ delay line */
|
|
uint64_t cmd : 5; /**< Setting for CMD delay line */
|
|
uint64_t clk : 5; /**< Setting for CLK delay line */
|
|
#else
|
|
uint64_t clk : 5;
|
|
uint64_t cmd : 5;
|
|
uint64_t dq : 5;
|
|
uint64_t reserved_15_63 : 49;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_delay_cfg_s cn30xx;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_14_63 : 50;
|
|
uint64_t dq : 4; /**< Setting for DQ delay line */
|
|
uint64_t reserved_9_9 : 1;
|
|
uint64_t cmd : 4; /**< Setting for CMD delay line */
|
|
uint64_t reserved_4_4 : 1;
|
|
uint64_t clk : 4; /**< Setting for CLK delay line */
|
|
#else
|
|
uint64_t clk : 4;
|
|
uint64_t reserved_4_4 : 1;
|
|
uint64_t cmd : 4;
|
|
uint64_t reserved_9_9 : 1;
|
|
uint64_t dq : 4;
|
|
uint64_t reserved_14_63 : 50;
|
|
#endif
|
|
} cn38xx;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn50xx;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn52xx;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn52xxp1;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn56xx;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn56xxp1;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn58xx;
|
|
struct cvmx_lmcx_delay_cfg_cn38xx cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_delay_cfg cvmx_lmcx_delay_cfg_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dimm#_params
|
|
*
|
|
* LMC_DIMMX_PARAMS = LMC DIMMX Params
|
|
* This register contains values to be programmed into each control word in the corresponding (registered) DIMM. The control words allow
|
|
* optimization of the device properties for different raw card designs.
|
|
*
|
|
* Notes:
|
|
* LMC only uses this CSR when LMC*_CONTROL[RDIMM_ENA]=1. During a power-up/init sequence, LMC writes
|
|
* these fields into the control words in the JEDEC standard SSTE32882 registering clock driver on an
|
|
* RDIMM when corresponding LMC*_DIMM_CTL[DIMM*_WMASK] bits are set.
|
|
*/
|
|
union cvmx_lmcx_dimmx_params {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dimmx_params_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t rc15 : 4; /**< RC15, Reserved */
|
|
uint64_t rc14 : 4; /**< RC14, Reserved */
|
|
uint64_t rc13 : 4; /**< RC13, Reserved */
|
|
uint64_t rc12 : 4; /**< RC12, Reserved */
|
|
uint64_t rc11 : 4; /**< RC11, Encoding for RDIMM Operating VDD */
|
|
uint64_t rc10 : 4; /**< RC10, Encoding for RDIMM Operating Speed */
|
|
uint64_t rc9 : 4; /**< RC9 , Power Savings Settings Control Word */
|
|
uint64_t rc8 : 4; /**< RC8 , Additional IBT Settings Control Word */
|
|
uint64_t rc7 : 4; /**< RC7 , Reserved */
|
|
uint64_t rc6 : 4; /**< RC6 , Reserved */
|
|
uint64_t rc5 : 4; /**< RC5 , CK Driver Characterstics Control Word */
|
|
uint64_t rc4 : 4; /**< RC4 , Control Signals Driver Characteristics Control Word */
|
|
uint64_t rc3 : 4; /**< RC3 , CA Signals Driver Characterstics Control Word */
|
|
uint64_t rc2 : 4; /**< RC2 , Timing Control Word */
|
|
uint64_t rc1 : 4; /**< RC1 , Clock Driver Enable Control Word */
|
|
uint64_t rc0 : 4; /**< RC0 , Global Features Control Word */
|
|
#else
|
|
uint64_t rc0 : 4;
|
|
uint64_t rc1 : 4;
|
|
uint64_t rc2 : 4;
|
|
uint64_t rc3 : 4;
|
|
uint64_t rc4 : 4;
|
|
uint64_t rc5 : 4;
|
|
uint64_t rc6 : 4;
|
|
uint64_t rc7 : 4;
|
|
uint64_t rc8 : 4;
|
|
uint64_t rc9 : 4;
|
|
uint64_t rc10 : 4;
|
|
uint64_t rc11 : 4;
|
|
uint64_t rc12 : 4;
|
|
uint64_t rc13 : 4;
|
|
uint64_t rc14 : 4;
|
|
uint64_t rc15 : 4;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dimmx_params_s cn61xx;
|
|
struct cvmx_lmcx_dimmx_params_s cn63xx;
|
|
struct cvmx_lmcx_dimmx_params_s cn63xxp1;
|
|
struct cvmx_lmcx_dimmx_params_s cn66xx;
|
|
struct cvmx_lmcx_dimmx_params_s cn68xx;
|
|
struct cvmx_lmcx_dimmx_params_s cn68xxp1;
|
|
struct cvmx_lmcx_dimmx_params_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_dimmx_params cvmx_lmcx_dimmx_params_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dimm_ctl
|
|
*
|
|
* LMC_DIMM_CTL = LMC DIMM Control
|
|
*
|
|
*
|
|
* Notes:
|
|
* This CSR is only used when LMC*_CONTROL[RDIMM_ENA]=1. During a power-up/init sequence, this CSR
|
|
* controls LMC's writes to the control words in the JEDEC standard SSTE32882 registering clock driver
|
|
* on an RDIMM.
|
|
*/
|
|
union cvmx_lmcx_dimm_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dimm_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_46_63 : 18;
|
|
uint64_t parity : 1; /**< Parity
|
|
The PAR_IN input of a registered DIMM should be
|
|
tied off. LMC adjusts the value of the DDR_WE_L (DWE#)
|
|
pin during DDR3 register part control word writes to
|
|
ensure the parity is observed correctly by the receiving
|
|
SSTE32882 register part.
|
|
When PAR_IN is grounded, PARITY should be cleared to 0. */
|
|
uint64_t tcws : 13; /**< LMC waits for this time period before and after a RDIMM
|
|
Control Word Access during a power-up/init SEQUENCE.
|
|
TCWS is in multiples of 8 CK cycles.
|
|
Set TCWS (CSR field) = RNDUP[tcws(ns)/(8*tCYC(ns))],
|
|
where tCWS is the desired time (ns), and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=0x4e0 (equivalent to 15us) when changing
|
|
clock timing (RC2.DBA1, RC6.DA4, RC10.DA3, RC10.DA4,
|
|
RC11.DA3, and RC11.DA4)
|
|
TYP=0x8, otherwise
|
|
0x0 = Reserved */
|
|
uint64_t dimm1_wmask : 16; /**< DIMM1 Write Mask
|
|
if (DIMM1_WMASK[n] = 1)
|
|
Write DIMM1.RCn */
|
|
uint64_t dimm0_wmask : 16; /**< DIMM0 Write Mask
|
|
if (DIMM0_WMASK[n] = 1)
|
|
Write DIMM0.RCn */
|
|
#else
|
|
uint64_t dimm0_wmask : 16;
|
|
uint64_t dimm1_wmask : 16;
|
|
uint64_t tcws : 13;
|
|
uint64_t parity : 1;
|
|
uint64_t reserved_46_63 : 18;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dimm_ctl_s cn61xx;
|
|
struct cvmx_lmcx_dimm_ctl_s cn63xx;
|
|
struct cvmx_lmcx_dimm_ctl_s cn63xxp1;
|
|
struct cvmx_lmcx_dimm_ctl_s cn66xx;
|
|
struct cvmx_lmcx_dimm_ctl_s cn68xx;
|
|
struct cvmx_lmcx_dimm_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_dimm_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_dimm_ctl cvmx_lmcx_dimm_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dll_ctl
|
|
*
|
|
* LMC_DLL_CTL = LMC DLL control and DCLK reset
|
|
*
|
|
*/
|
|
union cvmx_lmcx_dll_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dll_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_8_63 : 56;
|
|
uint64_t dreset : 1; /**< Dclk domain reset. The reset signal that is used by the
|
|
Dclk domain is (DRESET || ECLK_RESET). */
|
|
uint64_t dll90_byp : 1; /**< DDR DLL90 Bypass: When set, the DDR90 DLL is to be
|
|
bypassed and the setting is defined by DLL90_VLU */
|
|
uint64_t dll90_ena : 1; /**< DDR Quad DLL Enable: A 0->1 transition on this bit after
|
|
DCLK init sequence resets the DDR 90 DLL. Should
|
|
happen at startup before any activity in DDR. QDLL_ENA
|
|
must not transition 1->0 outside of a DRESET sequence
|
|
(i.e. it must remain 1 until the next DRESET).
|
|
DRESET should be asserted before and for 10 usec
|
|
following the 0->1 transition on QDLL_ENA. */
|
|
uint64_t dll90_vlu : 5; /**< Contains the open loop setting value for the DDR90 delay
|
|
line. */
|
|
#else
|
|
uint64_t dll90_vlu : 5;
|
|
uint64_t dll90_ena : 1;
|
|
uint64_t dll90_byp : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t reserved_8_63 : 56;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dll_ctl_s cn52xx;
|
|
struct cvmx_lmcx_dll_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_dll_ctl_s cn56xx;
|
|
struct cvmx_lmcx_dll_ctl_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_dll_ctl cvmx_lmcx_dll_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dll_ctl2
|
|
*
|
|
* LMC_DLL_CTL2 = LMC (Octeon) DLL control and DCLK reset
|
|
*
|
|
*
|
|
* Notes:
|
|
* DLL Bringup sequence:
|
|
* 1. If not done already, set LMC*_DLL_CTL2 = 0, except when LMC*_DLL_CTL2[DRESET] = 1.
|
|
* 2. Write 1 to LMC*_DLL_CTL2[DLL_BRINGUP]
|
|
* 3. Wait for 10 CK cycles, then write 1 to LMC*_DLL_CTL2[QUAD_DLL_ENA]. It may not be feasible to count 10 CK cycles, but the
|
|
* idea is to configure the delay line into DLL mode by asserting DLL_BRING_UP earlier than [QUAD_DLL_ENA], even if it is one
|
|
* cycle early. LMC*_DLL_CTL2[QUAD_DLL_ENA] must not change after this point without restarting the LMC and/or DRESET initialization
|
|
* sequence.
|
|
* 4. Read L2D_BST0 and wait for the result. (L2D_BST0 is subject to change depending on how it called in o63. It is still ok to go
|
|
* without step 4, since step 5 has enough time)
|
|
* 5. Wait 10 us.
|
|
* 6. Write 0 to LMC*_DLL_CTL2[DLL_BRINGUP]. LMC*_DLL_CTL2[DLL_BRINGUP] must not change after this point without restarting the LMC
|
|
* and/or DRESET initialization sequence.
|
|
* 7. Read L2D_BST0 and wait for the result. (same as step 4, but the idea here is the wait some time before going to step 8, even it
|
|
* is one cycle is fine)
|
|
* 8. Write 0 to LMC*_DLL_CTL2[DRESET]. LMC*_DLL_CTL2[DRESET] must not change after this point without restarting the LMC and/or
|
|
* DRESET initialization sequence.
|
|
*/
|
|
union cvmx_lmcx_dll_ctl2 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dll_ctl2_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_16_63 : 48;
|
|
uint64_t intf_en : 1; /**< Interface Enable */
|
|
uint64_t dll_bringup : 1; /**< DLL Bringup */
|
|
uint64_t dreset : 1; /**< Dclk domain reset. The reset signal that is used by the
|
|
Dclk domain is (DRESET || ECLK_RESET). */
|
|
uint64_t quad_dll_ena : 1; /**< DLL Enable */
|
|
uint64_t byp_sel : 4; /**< Bypass select
|
|
0000 : no byte
|
|
0001 : byte 0
|
|
- ...
|
|
1001 : byte 8
|
|
1010 : all bytes
|
|
1011-1111 : Reserved */
|
|
uint64_t byp_setting : 8; /**< Bypass setting
|
|
DDR3-1600: 00100010
|
|
DDR3-1333: 00110010
|
|
DDR3-1066: 01001011
|
|
DDR3-800 : 01110101
|
|
DDR3-667 : 10010110
|
|
DDR3-600 : 10101100 */
|
|
#else
|
|
uint64_t byp_setting : 8;
|
|
uint64_t byp_sel : 4;
|
|
uint64_t quad_dll_ena : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t dll_bringup : 1;
|
|
uint64_t intf_en : 1;
|
|
uint64_t reserved_16_63 : 48;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dll_ctl2_s cn61xx;
|
|
struct cvmx_lmcx_dll_ctl2_cn63xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_15_63 : 49;
|
|
uint64_t dll_bringup : 1; /**< DLL Bringup */
|
|
uint64_t dreset : 1; /**< Dclk domain reset. The reset signal that is used by the
|
|
Dclk domain is (DRESET || ECLK_RESET). */
|
|
uint64_t quad_dll_ena : 1; /**< DLL Enable */
|
|
uint64_t byp_sel : 4; /**< Bypass select
|
|
0000 : no byte
|
|
0001 : byte 0
|
|
- ...
|
|
1001 : byte 8
|
|
1010 : all bytes
|
|
1011-1111 : Reserved */
|
|
uint64_t byp_setting : 8; /**< Bypass setting
|
|
DDR3-1600: 00100010
|
|
DDR3-1333: 00110010
|
|
DDR3-1066: 01001011
|
|
DDR3-800 : 01110101
|
|
DDR3-667 : 10010110
|
|
DDR3-600 : 10101100 */
|
|
#else
|
|
uint64_t byp_setting : 8;
|
|
uint64_t byp_sel : 4;
|
|
uint64_t quad_dll_ena : 1;
|
|
uint64_t dreset : 1;
|
|
uint64_t dll_bringup : 1;
|
|
uint64_t reserved_15_63 : 49;
|
|
#endif
|
|
} cn63xx;
|
|
struct cvmx_lmcx_dll_ctl2_cn63xx cn63xxp1;
|
|
struct cvmx_lmcx_dll_ctl2_cn63xx cn66xx;
|
|
struct cvmx_lmcx_dll_ctl2_s cn68xx;
|
|
struct cvmx_lmcx_dll_ctl2_s cn68xxp1;
|
|
struct cvmx_lmcx_dll_ctl2_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_dll_ctl2 cvmx_lmcx_dll_ctl2_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dll_ctl3
|
|
*
|
|
* LMC_DLL_CTL3 = LMC DLL control and DCLK reset
|
|
*
|
|
*/
|
|
union cvmx_lmcx_dll_ctl3 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dll_ctl3_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_41_63 : 23;
|
|
uint64_t dclk90_fwd : 1; /**< Forward setting
|
|
0 : disable
|
|
1 : forward (generates a 1 cycle pulse to forward setting)
|
|
This register is oneshot and clears itself each time
|
|
it is set */
|
|
uint64_t ddr_90_dly_byp : 1; /**< Bypass DDR90_DLY in Clock Tree */
|
|
uint64_t dclk90_recal_dis : 1; /**< Disable periodic recalibration of DDR90 Delay Line in */
|
|
uint64_t dclk90_byp_sel : 1; /**< Bypass Setting Select for DDR90 Delay Line */
|
|
uint64_t dclk90_byp_setting : 8; /**< Bypass Setting for DDR90 Delay Line */
|
|
uint64_t dll_fast : 1; /**< DLL lock
|
|
0 = DLL locked */
|
|
uint64_t dll90_setting : 8; /**< Encoded DLL settings. Works in conjuction with
|
|
DLL90_BYTE_SEL */
|
|
uint64_t fine_tune_mode : 1; /**< DLL Fine Tune Mode
|
|
0 = disabled
|
|
1 = enable.
|
|
When enabled, calibrate internal PHY DLL every
|
|
LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t dll_mode : 1; /**< DLL Mode */
|
|
uint64_t dll90_byte_sel : 4; /**< Observe DLL settings for selected byte
|
|
0001 : byte 0
|
|
- ...
|
|
1001 : byte 8
|
|
0000,1010-1111 : Reserved */
|
|
uint64_t offset_ena : 1; /**< Offset enable
|
|
0 = disable
|
|
1 = enable */
|
|
uint64_t load_offset : 1; /**< Load offset
|
|
0 : disable
|
|
1 : load (generates a 1 cycle pulse to the PHY)
|
|
This register is oneshot and clears itself each time
|
|
it is set */
|
|
uint64_t mode_sel : 2; /**< Mode select
|
|
00 : reset
|
|
01 : write
|
|
10 : read
|
|
11 : write & read */
|
|
uint64_t byte_sel : 4; /**< Byte select
|
|
0000 : no byte
|
|
0001 : byte 0
|
|
- ...
|
|
1001 : byte 8
|
|
1010 : all bytes
|
|
1011-1111 : Reserved */
|
|
uint64_t offset : 6; /**< Write/read offset setting
|
|
[4:0] : offset
|
|
[5] : 0 = increment, 1 = decrement
|
|
Not a 2's complement value */
|
|
#else
|
|
uint64_t offset : 6;
|
|
uint64_t byte_sel : 4;
|
|
uint64_t mode_sel : 2;
|
|
uint64_t load_offset : 1;
|
|
uint64_t offset_ena : 1;
|
|
uint64_t dll90_byte_sel : 4;
|
|
uint64_t dll_mode : 1;
|
|
uint64_t fine_tune_mode : 1;
|
|
uint64_t dll90_setting : 8;
|
|
uint64_t dll_fast : 1;
|
|
uint64_t dclk90_byp_setting : 8;
|
|
uint64_t dclk90_byp_sel : 1;
|
|
uint64_t dclk90_recal_dis : 1;
|
|
uint64_t ddr_90_dly_byp : 1;
|
|
uint64_t dclk90_fwd : 1;
|
|
uint64_t reserved_41_63 : 23;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dll_ctl3_s cn61xx;
|
|
struct cvmx_lmcx_dll_ctl3_cn63xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_29_63 : 35;
|
|
uint64_t dll_fast : 1; /**< DLL lock
|
|
0 = DLL locked */
|
|
uint64_t dll90_setting : 8; /**< Encoded DLL settings. Works in conjuction with
|
|
DLL90_BYTE_SEL */
|
|
uint64_t fine_tune_mode : 1; /**< DLL Fine Tune Mode
|
|
0 = disabled
|
|
1 = enable.
|
|
When enabled, calibrate internal PHY DLL every
|
|
LMC*_CONFIG[REF_ZQCS_INT] CK cycles. */
|
|
uint64_t dll_mode : 1; /**< DLL Mode */
|
|
uint64_t dll90_byte_sel : 4; /**< Observe DLL settings for selected byte
|
|
0001 : byte 0
|
|
- ...
|
|
1001 : byte 8
|
|
0000,1010-1111 : Reserved */
|
|
uint64_t offset_ena : 1; /**< Offset enable
|
|
0 = disable
|
|
1 = enable */
|
|
uint64_t load_offset : 1; /**< Load offset
|
|
0 : disable
|
|
1 : load (generates a 1 cycle pulse to the PHY)
|
|
This register is oneshot and clears itself each time
|
|
it is set */
|
|
uint64_t mode_sel : 2; /**< Mode select
|
|
00 : reset
|
|
01 : write
|
|
10 : read
|
|
11 : write & read */
|
|
uint64_t byte_sel : 4; /**< Byte select
|
|
0000 : no byte
|
|
0001 : byte 0
|
|
- ...
|
|
1001 : byte 8
|
|
1010 : all bytes
|
|
1011-1111 : Reserved */
|
|
uint64_t offset : 6; /**< Write/read offset setting
|
|
[4:0] : offset
|
|
[5] : 0 = increment, 1 = decrement
|
|
Not a 2's complement value */
|
|
#else
|
|
uint64_t offset : 6;
|
|
uint64_t byte_sel : 4;
|
|
uint64_t mode_sel : 2;
|
|
uint64_t load_offset : 1;
|
|
uint64_t offset_ena : 1;
|
|
uint64_t dll90_byte_sel : 4;
|
|
uint64_t dll_mode : 1;
|
|
uint64_t fine_tune_mode : 1;
|
|
uint64_t dll90_setting : 8;
|
|
uint64_t dll_fast : 1;
|
|
uint64_t reserved_29_63 : 35;
|
|
#endif
|
|
} cn63xx;
|
|
struct cvmx_lmcx_dll_ctl3_cn63xx cn63xxp1;
|
|
struct cvmx_lmcx_dll_ctl3_cn63xx cn66xx;
|
|
struct cvmx_lmcx_dll_ctl3_s cn68xx;
|
|
struct cvmx_lmcx_dll_ctl3_s cn68xxp1;
|
|
struct cvmx_lmcx_dll_ctl3_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_dll_ctl3 cvmx_lmcx_dll_ctl3_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_dual_memcfg
|
|
*
|
|
* LMC_DUAL_MEMCFG = LMC Dual Memory Configuration Register
|
|
*
|
|
* This register controls certain parameters of Dual Memory Configuration
|
|
*
|
|
* Notes:
|
|
* This register enables the design to have two, separate memory configurations, selected dynamically
|
|
* by the reference address. Note however, that both configurations share
|
|
* LMC*_CONTROL[XOR_BANK], LMC*_CONFIG[PBANK_LSB], LMC*_CONFIG[RANK_ENA], and all timing parameters.
|
|
* In this description, "config0" refers to the normal memory configuration that is defined by the
|
|
* LMC*_CONFIG[ROW_LSB] parameters and "config1" refers to the dual (or second)
|
|
* memory configuration that is defined by this register.
|
|
*
|
|
* Enable mask to chip select mapping is shown below:
|
|
* CS_MASK[3] -> DIMM1_CS_<1>
|
|
* CS_MASK[2] -> DIMM1_CS_<0>
|
|
*
|
|
* CS_MASK[1] -> DIMM0_CS_<1>
|
|
* CS_MASK[0] -> DIMM0_CS_<0>
|
|
*
|
|
* DIMM n uses the pair of chip selects DIMMn_CS_<1:0>.
|
|
*
|
|
* Programming restrictions for CS_MASK:
|
|
* when LMC*_CONFIG[RANK_ENA] == 0, CS_MASK[2n + 1] = CS_MASK[2n]
|
|
*/
|
|
union cvmx_lmcx_dual_memcfg {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_dual_memcfg_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_20_63 : 44;
|
|
uint64_t bank8 : 1; /**< See LMC_DDR2_CTL[BANK8] */
|
|
uint64_t row_lsb : 3; /**< See LMC*_CONFIG[ROW_LSB] */
|
|
uint64_t reserved_8_15 : 8;
|
|
uint64_t cs_mask : 8; /**< Chip select mask.
|
|
This mask corresponds to the 8 chip selects for a memory
|
|
configuration. Each reference address will assert one of
|
|
the chip selects. If that chip select has its
|
|
corresponding CS_MASK bit set, then the "config1"
|
|
parameters are used, otherwise the "config0" parameters
|
|
are used. See additional notes below.
|
|
[7:4] *UNUSED IN 6xxx* */
|
|
#else
|
|
uint64_t cs_mask : 8;
|
|
uint64_t reserved_8_15 : 8;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t bank8 : 1;
|
|
uint64_t reserved_20_63 : 44;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_dual_memcfg_s cn50xx;
|
|
struct cvmx_lmcx_dual_memcfg_s cn52xx;
|
|
struct cvmx_lmcx_dual_memcfg_s cn52xxp1;
|
|
struct cvmx_lmcx_dual_memcfg_s cn56xx;
|
|
struct cvmx_lmcx_dual_memcfg_s cn56xxp1;
|
|
struct cvmx_lmcx_dual_memcfg_s cn58xx;
|
|
struct cvmx_lmcx_dual_memcfg_s cn58xxp1;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_19_63 : 45;
|
|
uint64_t row_lsb : 3; /**< See LMC*_CONFIG[ROW_LSB] */
|
|
uint64_t reserved_8_15 : 8;
|
|
uint64_t cs_mask : 8; /**< Chip select mask.
|
|
This mask corresponds to the 8 chip selects for a memory
|
|
configuration. Each reference address will assert one of
|
|
the chip selects. If that chip select has its
|
|
corresponding CS_MASK bit set, then the "config1"
|
|
parameters are used, otherwise the "config0" parameters
|
|
are used. See additional notes below.
|
|
[7:4] *UNUSED IN 6xxx* */
|
|
#else
|
|
uint64_t cs_mask : 8;
|
|
uint64_t reserved_8_15 : 8;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t reserved_19_63 : 45;
|
|
#endif
|
|
} cn61xx;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx cn63xx;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx cn63xxp1;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx cn66xx;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx cn68xx;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx cn68xxp1;
|
|
struct cvmx_lmcx_dual_memcfg_cn61xx cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_dual_memcfg cvmx_lmcx_dual_memcfg_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ecc_synd
|
|
*
|
|
* LMC_ECC_SYND = MRD ECC Syndromes
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ecc_synd {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ecc_synd_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t mrdsyn3 : 8; /**< MRD ECC Syndrome Quad3
|
|
MRDSYN3 corresponds to DQ[63:0]_c1_p1
|
|
In 32b mode, ecc is calculated on 4 cycle worth of data
|
|
MRDSYN3 corresponds to [DQ[31:0]_c3_p1, DQ[31:0]_c3_p0]
|
|
where _cC_pP denotes cycle C and phase P */
|
|
uint64_t mrdsyn2 : 8; /**< MRD ECC Syndrome Quad2
|
|
MRDSYN2 corresponds to DQ[63:0]_c1_p0
|
|
In 32b mode, ecc is calculated on 4 cycle worth of data
|
|
MRDSYN2 corresponds to [DQ[31:0]_c2_p1, DQ[31:0]_c2_p0]
|
|
where _cC_pP denotes cycle C and phase P */
|
|
uint64_t mrdsyn1 : 8; /**< MRD ECC Syndrome Quad1
|
|
MRDSYN1 corresponds to DQ[63:0]_c0_p1
|
|
In 32b mode, ecc is calculated on 4 cycle worth of data
|
|
MRDSYN1 corresponds to [DQ[31:0]_c1_p1, DQ[31:0]_c1_p0]
|
|
where _cC_pP denotes cycle C and phase P */
|
|
uint64_t mrdsyn0 : 8; /**< MRD ECC Syndrome Quad0
|
|
MRDSYN0 corresponds to DQ[63:0]_c0_p0
|
|
In 32b mode, ecc is calculated on 4 cycle worth of data
|
|
MRDSYN0 corresponds to [DQ[31:0]_c0_p1, DQ[31:0]_c0_p0]
|
|
where _cC_pP denotes cycle C and phase P */
|
|
#else
|
|
uint64_t mrdsyn0 : 8;
|
|
uint64_t mrdsyn1 : 8;
|
|
uint64_t mrdsyn2 : 8;
|
|
uint64_t mrdsyn3 : 8;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ecc_synd_s cn30xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn31xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn38xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn38xxp2;
|
|
struct cvmx_lmcx_ecc_synd_s cn50xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn52xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn52xxp1;
|
|
struct cvmx_lmcx_ecc_synd_s cn56xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn56xxp1;
|
|
struct cvmx_lmcx_ecc_synd_s cn58xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn58xxp1;
|
|
struct cvmx_lmcx_ecc_synd_s cn61xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn63xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn63xxp1;
|
|
struct cvmx_lmcx_ecc_synd_s cn66xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn68xx;
|
|
struct cvmx_lmcx_ecc_synd_s cn68xxp1;
|
|
struct cvmx_lmcx_ecc_synd_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_ecc_synd cvmx_lmcx_ecc_synd_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_fadr
|
|
*
|
|
* LMC_FADR = LMC Failing Address Register (SEC/DED/NXM)
|
|
*
|
|
* This register only captures the first transaction with ecc/nxm errors. A DED/NXM error can
|
|
* over-write this register with its failing addresses if the first error was a SEC. If you write
|
|
* LMC*_CONFIG->SEC_ERR/DED_ERR/NXM_ERR then it will clear the error bits and capture the
|
|
* next failing address.
|
|
*
|
|
* If FDIMM is 2 that means the error is in the higher bits DIMM.
|
|
*
|
|
* Notes:
|
|
* LMC*_FADR captures the failing pre-scrambled address location (split into dimm, bunk, bank, etc). If
|
|
* scrambling is off, then LMC*_FADR will also capture the failing physical location in the DRAM parts.
|
|
*
|
|
* LMC*_SCRAMBLED_FADR captures the actual failing address location in the physical DRAM parts, i.e.,
|
|
* a. if scrambling is on, LMC*_SCRAMBLE_FADR contains the failing physical location in the DRAM parts (split
|
|
* into dimm, bunk, bank, etc)
|
|
* b. if scrambling is off, the pre-scramble and post-scramble addresses are the same, and so the contents of
|
|
* LMC*_SCRAMBLED_FADR match the contents of LMC*_FADR
|
|
*/
|
|
union cvmx_lmcx_fadr {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_fadr_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_0_63 : 64;
|
|
#else
|
|
uint64_t reserved_0_63 : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_fadr_cn30xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t fdimm : 2; /**< Failing DIMM# */
|
|
uint64_t fbunk : 1; /**< Failing Rank */
|
|
uint64_t fbank : 3; /**< Failing Bank[2:0] */
|
|
uint64_t frow : 14; /**< Failing Row Address[13:0] */
|
|
uint64_t fcol : 12; /**< Failing Column Start Address[11:0]
|
|
Represents the Failing read's starting column address
|
|
(and not the exact column address in which the SEC/DED
|
|
was detected) */
|
|
#else
|
|
uint64_t fcol : 12;
|
|
uint64_t frow : 14;
|
|
uint64_t fbank : 3;
|
|
uint64_t fbunk : 1;
|
|
uint64_t fdimm : 2;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn30xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn31xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn38xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn38xxp2;
|
|
struct cvmx_lmcx_fadr_cn30xx cn50xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn52xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn52xxp1;
|
|
struct cvmx_lmcx_fadr_cn30xx cn56xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn56xxp1;
|
|
struct cvmx_lmcx_fadr_cn30xx cn58xx;
|
|
struct cvmx_lmcx_fadr_cn30xx cn58xxp1;
|
|
struct cvmx_lmcx_fadr_cn61xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_36_63 : 28;
|
|
uint64_t fdimm : 2; /**< Failing DIMM# */
|
|
uint64_t fbunk : 1; /**< Failing Rank */
|
|
uint64_t fbank : 3; /**< Failing Bank[2:0] */
|
|
uint64_t frow : 16; /**< Failing Row Address[15:0] */
|
|
uint64_t fcol : 14; /**< Failing Column Address[13:0]
|
|
Technically, represents the address of the 128b data
|
|
that had an ecc error, i.e., fcol[0] is always 0. Can
|
|
be used in conjuction with LMC*_CONFIG[DED_ERR] to
|
|
isolate the 64b chunk of data in error */
|
|
#else
|
|
uint64_t fcol : 14;
|
|
uint64_t frow : 16;
|
|
uint64_t fbank : 3;
|
|
uint64_t fbunk : 1;
|
|
uint64_t fdimm : 2;
|
|
uint64_t reserved_36_63 : 28;
|
|
#endif
|
|
} cn61xx;
|
|
struct cvmx_lmcx_fadr_cn61xx cn63xx;
|
|
struct cvmx_lmcx_fadr_cn61xx cn63xxp1;
|
|
struct cvmx_lmcx_fadr_cn61xx cn66xx;
|
|
struct cvmx_lmcx_fadr_cn61xx cn68xx;
|
|
struct cvmx_lmcx_fadr_cn61xx cn68xxp1;
|
|
struct cvmx_lmcx_fadr_cn61xx cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_fadr cvmx_lmcx_fadr_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ifb_cnt
|
|
*
|
|
* LMC_IFB_CNT = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ifb_cnt {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ifb_cnt_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t ifbcnt : 64; /**< Performance Counter
|
|
64-bit counter that increments every
|
|
CK cycle there is something in the in-flight buffer. */
|
|
#else
|
|
uint64_t ifbcnt : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ifb_cnt_s cn61xx;
|
|
struct cvmx_lmcx_ifb_cnt_s cn63xx;
|
|
struct cvmx_lmcx_ifb_cnt_s cn63xxp1;
|
|
struct cvmx_lmcx_ifb_cnt_s cn66xx;
|
|
struct cvmx_lmcx_ifb_cnt_s cn68xx;
|
|
struct cvmx_lmcx_ifb_cnt_s cn68xxp1;
|
|
struct cvmx_lmcx_ifb_cnt_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_ifb_cnt cvmx_lmcx_ifb_cnt_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ifb_cnt_hi
|
|
*
|
|
* LMC_IFB_CNT_HI = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ifb_cnt_hi {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ifbcnt_hi : 32; /**< Performance Counter to measure Bus Utilization
|
|
Upper 32-bits of 64-bit counter that increments every
|
|
cycle there is something in the in-flight buffer. */
|
|
#else
|
|
uint64_t ifbcnt_hi : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn30xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn31xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn38xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn38xxp2;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn50xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn52xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn52xxp1;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn56xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn56xxp1;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn58xx;
|
|
struct cvmx_lmcx_ifb_cnt_hi_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ifb_cnt_hi cvmx_lmcx_ifb_cnt_hi_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ifb_cnt_lo
|
|
*
|
|
* LMC_IFB_CNT_LO = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ifb_cnt_lo {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ifbcnt_lo : 32; /**< Performance Counter
|
|
Low 32-bits of 64-bit counter that increments every
|
|
cycle there is something in the in-flight buffer. */
|
|
#else
|
|
uint64_t ifbcnt_lo : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn30xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn31xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn38xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn38xxp2;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn50xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn52xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn52xxp1;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn56xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn56xxp1;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn58xx;
|
|
struct cvmx_lmcx_ifb_cnt_lo_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ifb_cnt_lo cvmx_lmcx_ifb_cnt_lo_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_int
|
|
*
|
|
* LMC_INT = LMC Interrupt Register
|
|
*
|
|
*/
|
|
union cvmx_lmcx_int {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_int_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_9_63 : 55;
|
|
uint64_t ded_err : 4; /**< Double Error detected (DED) of Rd Data
|
|
[0] corresponds to DQ[63:0]_c0_p0
|
|
[1] corresponds to DQ[63:0]_c0_p1
|
|
[2] corresponds to DQ[63:0]_c1_p0
|
|
[3] corresponds to DQ[63:0]_c1_p1
|
|
In 32b mode, ecc is calculated on 4 cycle worth of data
|
|
[0] corresponds to [DQ[31:0]_c0_p1, DQ[31:0]_c0_p0]
|
|
[1] corresponds to [DQ[31:0]_c1_p1, DQ[31:0]_c1_p0]
|
|
[2] corresponds to [DQ[31:0]_c2_p1, DQ[31:0]_c2_p0]
|
|
[3] corresponds to [DQ[31:0]_c3_p1, DQ[31:0]_c3_p0]
|
|
where _cC_pP denotes cycle C and phase P
|
|
Write of 1 will clear the corresponding error bit */
|
|
uint64_t sec_err : 4; /**< Single Error (corrected) of Rd Data
|
|
[0] corresponds to DQ[63:0]_c0_p0
|
|
[1] corresponds to DQ[63:0]_c0_p1
|
|
[2] corresponds to DQ[63:0]_c1_p0
|
|
[3] corresponds to DQ[63:0]_c1_p1
|
|
In 32b mode, ecc is calculated on 4 cycle worth of data
|
|
[0] corresponds to [DQ[31:0]_c0_p1, DQ[31:0]_c0_p0]
|
|
[1] corresponds to [DQ[31:0]_c1_p1, DQ[31:0]_c1_p0]
|
|
[2] corresponds to [DQ[31:0]_c2_p1, DQ[31:0]_c2_p0]
|
|
[3] corresponds to [DQ[31:0]_c3_p1, DQ[31:0]_c3_p0]
|
|
where _cC_pP denotes cycle C and phase P
|
|
Write of 1 will clear the corresponding error bit */
|
|
uint64_t nxm_wr_err : 1; /**< Write to non-existent memory
|
|
Write of 1 will clear the corresponding error bit */
|
|
#else
|
|
uint64_t nxm_wr_err : 1;
|
|
uint64_t sec_err : 4;
|
|
uint64_t ded_err : 4;
|
|
uint64_t reserved_9_63 : 55;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_int_s cn61xx;
|
|
struct cvmx_lmcx_int_s cn63xx;
|
|
struct cvmx_lmcx_int_s cn63xxp1;
|
|
struct cvmx_lmcx_int_s cn66xx;
|
|
struct cvmx_lmcx_int_s cn68xx;
|
|
struct cvmx_lmcx_int_s cn68xxp1;
|
|
struct cvmx_lmcx_int_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_int cvmx_lmcx_int_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_int_en
|
|
*
|
|
* LMC_INT_EN = LMC Interrupt Enable Register
|
|
*
|
|
*/
|
|
union cvmx_lmcx_int_en {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_int_en_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_3_63 : 61;
|
|
uint64_t intr_ded_ena : 1; /**< ECC Double Error Detect(DED) Interrupt Enable bit
|
|
When set, the memory controller raises a processor
|
|
interrupt on detecting an uncorrectable Dbl Bit ECC
|
|
error. */
|
|
uint64_t intr_sec_ena : 1; /**< ECC Single Error Correct(SEC) Interrupt Enable bit
|
|
When set, the memory controller raises a processor
|
|
interrupt on detecting a correctable Single Bit ECC
|
|
error. */
|
|
uint64_t intr_nxm_wr_ena : 1; /**< Non Write Error Interrupt Enable bit
|
|
When set, the memory controller raises a processor
|
|
interrupt on detecting an non-existent memory write */
|
|
#else
|
|
uint64_t intr_nxm_wr_ena : 1;
|
|
uint64_t intr_sec_ena : 1;
|
|
uint64_t intr_ded_ena : 1;
|
|
uint64_t reserved_3_63 : 61;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_int_en_s cn61xx;
|
|
struct cvmx_lmcx_int_en_s cn63xx;
|
|
struct cvmx_lmcx_int_en_s cn63xxp1;
|
|
struct cvmx_lmcx_int_en_s cn66xx;
|
|
struct cvmx_lmcx_int_en_s cn68xx;
|
|
struct cvmx_lmcx_int_en_s cn68xxp1;
|
|
struct cvmx_lmcx_int_en_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_int_en cvmx_lmcx_int_en_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_mem_cfg0
|
|
*
|
|
* Specify the RSL base addresses for the block
|
|
*
|
|
* LMC_MEM_CFG0 = LMC Memory Configuration Register0
|
|
*
|
|
* This register controls certain parameters of Memory Configuration
|
|
*/
|
|
union cvmx_lmcx_mem_cfg0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_mem_cfg0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t reset : 1; /**< Reset oneshot pulse for refresh counter,
|
|
and LMC_OPS_CNT_*, LMC_IFB_CNT_*, and LMC_DCLK_CNT_*
|
|
CSR's. SW should write this to a one, then re-write
|
|
it to a zero to cause the reset. */
|
|
uint64_t silo_qc : 1; /**< Adds a Quarter Cycle granularity to generate
|
|
dqs pulse generation for silo.
|
|
Combination of Silo_HC and Silo_QC gives the
|
|
ability to position the read enable with quarter
|
|
cycle resolution. This is applied on all the bytes
|
|
uniformly. */
|
|
uint64_t bunk_ena : 1; /**< Bunk Enable aka RANK ena (for use with dual-rank DIMMs)
|
|
For dual-rank DIMMs, the bunk_ena bit will enable
|
|
the drive of the CS_N[1:0] pins based on the
|
|
(pbank_lsb-1) address bit.
|
|
Write 0 for SINGLE ranked DIMM's. */
|
|
uint64_t ded_err : 4; /**< Double Error detected (DED) of Rd Data
|
|
In 128b mode, ecc is calulated on 1 cycle worth of data
|
|
[25] corresponds to DQ[63:0], Phase0
|
|
[26] corresponds to DQ[127:64], Phase0
|
|
[27] corresponds to DQ[63:0], Phase1
|
|
[28] corresponds to DQ[127:64], Phase1
|
|
In 64b mode, ecc is calculated on 2 cycle worth of data
|
|
[25] corresponds to DQ[63:0], Phase0, cycle0
|
|
[26] corresponds to DQ[63:0], Phase0, cycle1
|
|
[27] corresponds to DQ[63:0], Phase1, cycle0
|
|
[28] corresponds to DQ[63:0], Phase1, cycle1
|
|
Write of 1 will clear the corresponding error bit */
|
|
uint64_t sec_err : 4; /**< Single Error (corrected) of Rd Data
|
|
In 128b mode, ecc is calulated on 1 cycle worth of data
|
|
[21] corresponds to DQ[63:0], Phase0
|
|
[22] corresponds to DQ[127:64], Phase0
|
|
[23] corresponds to DQ[63:0], Phase1
|
|
[24] corresponds to DQ[127:64], Phase1
|
|
In 64b mode, ecc is calculated on 2 cycle worth of data
|
|
[21] corresponds to DQ[63:0], Phase0, cycle0
|
|
[22] corresponds to DQ[63:0], Phase0, cycle1
|
|
[23] corresponds to DQ[63:0], Phase1, cycle0
|
|
[24] corresponds to DQ[63:0], Phase1, cycle1
|
|
Write of 1 will clear the corresponding error bit */
|
|
uint64_t intr_ded_ena : 1; /**< ECC Double Error Detect(DED) Interrupt Enable bit
|
|
When set, the memory controller raises a processor
|
|
interrupt on detecting an uncorrectable Dbl Bit ECC
|
|
error. */
|
|
uint64_t intr_sec_ena : 1; /**< ECC Single Error Correct(SEC) Interrupt Enable bit
|
|
When set, the memory controller raises a processor
|
|
interrupt on detecting a correctable Single Bit ECC
|
|
error. */
|
|
uint64_t tcl : 4; /**< This register is not used */
|
|
uint64_t ref_int : 6; /**< Refresh interval represented in \#of 512 dclk increments.
|
|
Program this to RND-DN(tREFI/clkPeriod/512)
|
|
- 000000: RESERVED
|
|
- 000001: 1 * 512 = 512 dclks
|
|
- ...
|
|
- 111111: 63 * 512 = 32256 dclks */
|
|
uint64_t pbank_lsb : 4; /**< Physical Bank address select
|
|
Reverting to the explanation for ROW_LSB,
|
|
PBank_LSB would be Row_LSB bit + \#rowbits
|
|
+ \#rankbits
|
|
In the 512MB DIMM Example, assuming no rank bits:
|
|
pbank_lsb=mem_addr[15+13] for 64 b mode
|
|
=mem_addr[16+13] for 128b mode
|
|
Hence the parameter
|
|
0000:pbank[1:0] = mem_adr[28:27] / rank = mem_adr[26] (if bunk_ena)
|
|
0001:pbank[1:0] = mem_adr[29:28] / rank = mem_adr[27] "
|
|
0010:pbank[1:0] = mem_adr[30:29] / rank = mem_adr[28] "
|
|
0011:pbank[1:0] = mem_adr[31:30] / rank = mem_adr[29] "
|
|
0100:pbank[1:0] = mem_adr[32:31] / rank = mem_adr[30] "
|
|
0101:pbank[1:0] = mem_adr[33:32] / rank = mem_adr[31] "
|
|
0110:pbank[1:0] =[1'b0,mem_adr[33]] / rank = mem_adr[32] "
|
|
0111:pbank[1:0] =[2'b0] / rank = mem_adr[33] "
|
|
1000-1111: RESERVED */
|
|
uint64_t row_lsb : 3; /**< Encoding used to determine which memory address
|
|
bit position represents the low order DDR ROW address.
|
|
The processor's memory address[33:7] needs to be
|
|
translated to DRAM addresses (bnk,row,col,rank and dimm)
|
|
and that is a function of the following:
|
|
1. \# Banks (4 or 8) - spec'd by BANK8
|
|
2. Datapath Width(64 or 128) - MODE128b
|
|
3. \# Ranks in a DIMM - spec'd by BUNK_ENA
|
|
4. \# DIMM's in the system
|
|
5. \# Column Bits of the memory part - spec'd indirectly
|
|
by this register.
|
|
6. \# Row Bits of the memory part - spec'd indirectly
|
|
by the register below (PBANK_LSB).
|
|
Illustration: For Micron's MT18HTF6472A,512MB DDR2
|
|
Unbuffered DIMM which uses 256Mb parts (8M x 8 x 4),
|
|
\# Banks = 4 -> 2 bits of BA
|
|
\# Columns = 1K -> 10 bits of Col
|
|
\# Rows = 8K -> 13 bits of Row
|
|
Assuming that the total Data width is 128, this is how
|
|
we arrive at row_lsb:
|
|
Col Address starts from mem_addr[4] for 128b (16Bytes)
|
|
dq width or from mem_addr[3] for 64b (8Bytes) dq width
|
|
\# col + \# bank = 12. Hence row_lsb is mem_adr[15] for
|
|
64bmode or mem_adr[16] for 128b mode. Hence row_lsb
|
|
parameter should be set to 001 (64b) or 010 (128b).
|
|
- 000: row_lsb = mem_adr[14]
|
|
- 001: row_lsb = mem_adr[15]
|
|
- 010: row_lsb = mem_adr[16]
|
|
- 011: row_lsb = mem_adr[17]
|
|
- 100: row_lsb = mem_adr[18]
|
|
- 101-111:row_lsb = RESERVED */
|
|
uint64_t ecc_ena : 1; /**< ECC Enable: When set will enable the 8b ECC
|
|
check/correct logic. Should be 1 when used with DIMMs
|
|
with ECC. 0, otherwise.
|
|
When this mode is turned on, DQ[71:64] and DQ[143:137]
|
|
on writes, will contain the ECC code generated for
|
|
the lower 64 and upper 64 bits of data which will
|
|
written in the memory and then later on reads, used
|
|
to check for Single bit error (which will be auto-
|
|
corrected) and Double Bit error (which will be
|
|
reported). When not turned on, DQ[71:64] and DQ[143:137]
|
|
are driven to 0. Please refer to SEC_ERR, DED_ERR,
|
|
LMC_FADR, and LMC_ECC_SYND registers
|
|
for diagnostics information when there is an error. */
|
|
uint64_t init_start : 1; /**< A 0->1 transition starts the DDR memory initialization
|
|
sequence. */
|
|
#else
|
|
uint64_t init_start : 1;
|
|
uint64_t ecc_ena : 1;
|
|
uint64_t row_lsb : 3;
|
|
uint64_t pbank_lsb : 4;
|
|
uint64_t ref_int : 6;
|
|
uint64_t tcl : 4;
|
|
uint64_t intr_sec_ena : 1;
|
|
uint64_t intr_ded_ena : 1;
|
|
uint64_t sec_err : 4;
|
|
uint64_t ded_err : 4;
|
|
uint64_t bunk_ena : 1;
|
|
uint64_t silo_qc : 1;
|
|
uint64_t reset : 1;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_mem_cfg0_s cn30xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn31xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn38xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn38xxp2;
|
|
struct cvmx_lmcx_mem_cfg0_s cn50xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn52xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn52xxp1;
|
|
struct cvmx_lmcx_mem_cfg0_s cn56xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn56xxp1;
|
|
struct cvmx_lmcx_mem_cfg0_s cn58xx;
|
|
struct cvmx_lmcx_mem_cfg0_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_mem_cfg0 cvmx_lmcx_mem_cfg0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_mem_cfg1
|
|
*
|
|
* LMC_MEM_CFG1 = LMC Memory Configuration Register1
|
|
*
|
|
* This register controls the External Memory Configuration Timing Parameters. Please refer to the
|
|
* appropriate DDR part spec from your memory vendor for the various values in this CSR.
|
|
* The details of each of these timing parameters can be found in the JEDEC spec or the vendor
|
|
* spec of the memory parts.
|
|
*/
|
|
union cvmx_lmcx_mem_cfg1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_mem_cfg1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t comp_bypass : 1; /**< Compensation bypass. */
|
|
uint64_t trrd : 3; /**< tRRD cycles: ACT-ACT timing parameter for different
|
|
banks. (Represented in tCYC cycles == 1dclks)
|
|
TYP=15ns (66MHz=1,167MHz=3,200MHz=3)
|
|
For DDR2, TYP=7.5ns
|
|
- 000: RESERVED
|
|
- 001: 1 tCYC
|
|
- 010: 2 tCYC
|
|
- 011: 3 tCYC
|
|
- 100: 4 tCYC
|
|
- 101: 5 tCYC
|
|
- 110: 6 tCYC
|
|
- 111: 7 tCYC */
|
|
uint64_t caslat : 3; /**< CAS Latency Encoding which is loaded into each DDR
|
|
SDRAM device (MRS[6:4]) upon power-up (INIT_START=1).
|
|
(Represented in tCYC cycles == 1 dclks)
|
|
000 RESERVED
|
|
001 RESERVED
|
|
010 2.0 tCYC
|
|
011 3.0 tCYC
|
|
100 4.0 tCYC
|
|
101 5.0 tCYC
|
|
110 6.0 tCYC
|
|
111 RESERVED
|
|
eg). The parameters TSKW, SILO_HC, and SILO_QC can
|
|
account for 1/4 cycle granularity in board/etch delays. */
|
|
uint64_t tmrd : 3; /**< tMRD Cycles
|
|
(Represented in dclk tCYC)
|
|
For DDR2, its TYP 2*tCYC)
|
|
- 000: RESERVED
|
|
- 001: 1
|
|
- 010: 2
|
|
- 011: 3
|
|
- 100: 4
|
|
- 101-111: RESERVED */
|
|
uint64_t trfc : 5; /**< Indicates tRFC constraints.
|
|
Set TRFC (CSR field) = RNDUP[tRFC(ns)/4*tcyc(ns)],
|
|
where tRFC is from the DDR2 spec, and tcyc(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
For example, with 2Gb, DDR2-667 parts,
|
|
typ tRFC=195ns, so TRFC (CSR field) = 0x11.
|
|
TRFC (binary): Corresponding tRFC Cycles
|
|
----------------------------------------
|
|
- 00000-00001: RESERVED
|
|
- 00010: 0-8
|
|
- 00011: 9-12
|
|
- 00100: 13-16
|
|
- ...
|
|
- 11110: 117-120
|
|
- 11111: 121-124 */
|
|
uint64_t trp : 4; /**< tRP Cycles = RNDUP[tRP(ns)/tcyc(ns)]
|
|
(Represented in tCYC cycles == 1dclk)
|
|
TYP=15ns (66MHz=1,167MHz=3,400MHz=6 for TYP)
|
|
- 0000: RESERVED
|
|
- 0001: 1
|
|
- ...
|
|
- 1001: 9
|
|
- 1010-1111: RESERVED
|
|
When using parts with 8 banks (LMC_DDR2_CTL->BANK8
|
|
is 1), load tRP cycles + 1 into this register. */
|
|
uint64_t twtr : 4; /**< tWTR Cycles = RNDUP[tWTR(ns)/tcyc(ns)]
|
|
Last Wr Data to Rd Command time.
|
|
(Represented in tCYC cycles == 1dclks)
|
|
TYP=15ns (66MHz=1,167MHz=3,400MHz=6, for TYP)
|
|
- 0000: RESERVED
|
|
- 0001: 1
|
|
- ...
|
|
- 0111: 7
|
|
- 1000-1111: RESERVED */
|
|
uint64_t trcd : 4; /**< tRCD Cycles = RNDUP[tRCD(ns)/tcyc(ns)]
|
|
(Represented in tCYC cycles == 1dclk)
|
|
TYP=15ns (66MHz=1,167MHz=3,400MHz=6 for TYP)
|
|
- 0000: RESERVED
|
|
- 0001: 2 (2 is the smallest value allowed)
|
|
- 0002: 2
|
|
- ...
|
|
- 1001: 9
|
|
- 1010-1111: RESERVED
|
|
In 2T mode, make this register TRCD-1, not going
|
|
below 2. */
|
|
uint64_t tras : 5; /**< tRAS Cycles = RNDUP[tRAS(ns)/tcyc(ns)]
|
|
(Represented in tCYC cycles == 1 dclk)
|
|
- 00000-0001: RESERVED
|
|
- 00010: 2
|
|
- ...
|
|
- 11111: 31 */
|
|
#else
|
|
uint64_t tras : 5;
|
|
uint64_t trcd : 4;
|
|
uint64_t twtr : 4;
|
|
uint64_t trp : 4;
|
|
uint64_t trfc : 5;
|
|
uint64_t tmrd : 3;
|
|
uint64_t caslat : 3;
|
|
uint64_t trrd : 3;
|
|
uint64_t comp_bypass : 1;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_mem_cfg1_s cn30xx;
|
|
struct cvmx_lmcx_mem_cfg1_s cn31xx;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_31_63 : 33;
|
|
uint64_t trrd : 3; /**< tRRD cycles: ACT-ACT timing parameter for different
|
|
banks. (Represented in tCYC cycles == 1dclks)
|
|
TYP=15ns (66MHz=1,167MHz=3,200MHz=3)
|
|
For DDR2, TYP=7.5ns
|
|
- 000: RESERVED
|
|
- 001: 1 tCYC
|
|
- 010: 2 tCYC
|
|
- 011: 3 tCYC
|
|
- 100: 4 tCYC
|
|
- 101: 5 tCYC
|
|
- 110-111: RESERVED */
|
|
uint64_t caslat : 3; /**< CAS Latency Encoding which is loaded into each DDR
|
|
SDRAM device (MRS[6:4]) upon power-up (INIT_START=1).
|
|
(Represented in tCYC cycles == 1 dclks)
|
|
000 RESERVED
|
|
001 RESERVED
|
|
010 2.0 tCYC
|
|
011 3.0 tCYC
|
|
100 4.0 tCYC
|
|
101 5.0 tCYC
|
|
110 6.0 tCYC (DDR2)
|
|
2.5 tCYC (DDR1)
|
|
111 RESERVED
|
|
eg). The parameters TSKW, SILO_HC, and SILO_QC can
|
|
account for 1/4 cycle granularity in board/etch delays. */
|
|
uint64_t tmrd : 3; /**< tMRD Cycles
|
|
(Represented in dclk tCYC)
|
|
For DDR2, its TYP 2*tCYC)
|
|
- 000: RESERVED
|
|
- 001: 1
|
|
- 010: 2
|
|
- 011: 3
|
|
- 100: 4
|
|
- 101-111: RESERVED */
|
|
uint64_t trfc : 5; /**< Indicates tRFC constraints.
|
|
Set TRFC (CSR field) = RNDUP[tRFC(ns)/4*tcyc(ns)],
|
|
where tRFC is from the DDR2 spec, and tcyc(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
For example, with 2Gb, DDR2-667 parts,
|
|
typ tRFC=195ns, so TRFC (CSR field) = 0x11.
|
|
TRFC (binary): Corresponding tRFC Cycles
|
|
----------------------------------------
|
|
- 00000-00001: RESERVED
|
|
- 00010: 0-8
|
|
- 00011: 9-12
|
|
- 00100: 13-16
|
|
- ...
|
|
- 11110: 117-120
|
|
- 11111: 121-124 */
|
|
uint64_t trp : 4; /**< tRP Cycles = RNDUP[tRP(ns)/tcyc(ns)]
|
|
(Represented in tCYC cycles == 1dclk)
|
|
TYP=15ns (66MHz=1,167MHz=3,400MHz=6 for TYP)
|
|
- 0000: RESERVED
|
|
- 0001: 1
|
|
- ...
|
|
- 0111: 7
|
|
- 1000-1111: RESERVED
|
|
When using parts with 8 banks (LMC_DDR2_CTL->BANK8
|
|
is 1), load tRP cycles + 1 into this register. */
|
|
uint64_t twtr : 4; /**< tWTR Cycles = RNDUP[tWTR(ns)/tcyc(ns)]
|
|
Last Wr Data to Rd Command time.
|
|
(Represented in tCYC cycles == 1dclks)
|
|
TYP=15ns (66MHz=1,167MHz=3,400MHz=6, for TYP)
|
|
- 0000: RESERVED
|
|
- 0001: 1
|
|
- ...
|
|
- 0111: 7
|
|
- 1000-1111: RESERVED */
|
|
uint64_t trcd : 4; /**< tRCD Cycles = RNDUP[tRCD(ns)/tcyc(ns)]
|
|
(Represented in tCYC cycles == 1dclk)
|
|
TYP=15ns (66MHz=1,167MHz=3,400MHz=6 for TYP)
|
|
- 0000: RESERVED
|
|
- 0001: 2 (2 is the smallest value allowed)
|
|
- 0002: 2
|
|
- ...
|
|
- 0111: 7
|
|
- 1110-1111: RESERVED
|
|
In 2T mode, make this register TRCD-1, not going
|
|
below 2. */
|
|
uint64_t tras : 5; /**< tRAS Cycles = RNDUP[tRAS(ns)/tcyc(ns)]
|
|
(Represented in tCYC cycles == 1 dclk)
|
|
For DDR-I mode:
|
|
TYP=45ns (66MHz=3,167MHz=8,400MHz=18
|
|
- 00000-0001: RESERVED
|
|
- 00010: 2
|
|
- ...
|
|
- 10100: 20
|
|
- 10101-11111: RESERVED */
|
|
#else
|
|
uint64_t tras : 5;
|
|
uint64_t trcd : 4;
|
|
uint64_t twtr : 4;
|
|
uint64_t trp : 4;
|
|
uint64_t trfc : 5;
|
|
uint64_t tmrd : 3;
|
|
uint64_t caslat : 3;
|
|
uint64_t trrd : 3;
|
|
uint64_t reserved_31_63 : 33;
|
|
#endif
|
|
} cn38xx;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn38xxp2;
|
|
struct cvmx_lmcx_mem_cfg1_s cn50xx;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn52xx;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn52xxp1;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn56xx;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn56xxp1;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn58xx;
|
|
struct cvmx_lmcx_mem_cfg1_cn38xx cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_mem_cfg1 cvmx_lmcx_mem_cfg1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_modereg_params0
|
|
*
|
|
* Notes:
|
|
* These parameters are written into the DDR3 MR0, MR1, MR2 and MR3 registers.
|
|
*
|
|
*/
|
|
union cvmx_lmcx_modereg_params0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_modereg_params0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_25_63 : 39;
|
|
uint64_t ppd : 1; /**< DLL Control for precharge powerdown
|
|
0 = Slow exit (DLL off)
|
|
1 = Fast exit (DLL on)
|
|
LMC writes this value to MR0[PPD] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
This value must equal the MR0[PPD] value in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t wrp : 3; /**< Write recovery for auto precharge
|
|
Should be programmed to be equal to or greater than
|
|
RNDUP[tWR(ns)/tCYC(ns)]
|
|
000 = 5
|
|
001 = 5
|
|
010 = 6
|
|
011 = 7
|
|
100 = 8
|
|
101 = 10
|
|
110 = 12
|
|
111 = 14
|
|
LMC writes this value to MR0[WR] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
This value must equal the MR0[WR] value in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t dllr : 1; /**< DLL Reset
|
|
LMC writes this value to MR0[DLL] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
The MR0[DLL] value must be 0 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t tm : 1; /**< Test Mode
|
|
LMC writes this value to MR0[TM] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
The MR0[TM] value must be 0 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t rbt : 1; /**< Read Burst Type
|
|
1 = interleaved (fixed)
|
|
LMC writes this value to MR0[RBT] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
The MR0[RBT] value must be 1 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t cl : 4; /**< CAS Latency
|
|
0010 = 5
|
|
0100 = 6
|
|
0110 = 7
|
|
1000 = 8
|
|
1010 = 9
|
|
1100 = 10
|
|
1110 = 11
|
|
0001 = 12
|
|
0011 = 13
|
|
0101 = 14
|
|
0111 = 15
|
|
1001 = 16
|
|
0000, 1011, 1101, 1111 = Reserved
|
|
LMC writes this value to MR0[CAS Latency / CL] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
This value must equal the MR0[CAS Latency / CL] value in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t bl : 2; /**< Burst Length
|
|
0 = 8 (fixed)
|
|
LMC writes this value to MR0[BL] in the selected DDR3 parts
|
|
during power-up/init and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
The MR0[BL] value must be 0 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t qoff : 1; /**< Qoff Enable
|
|
0 = enable
|
|
1 = disable
|
|
LMC writes this value to MR1[Qoff] in the DDR3 parts in the selected ranks
|
|
during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK,INIT_STATUS] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
The MR1[Qoff] value must be 0 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t tdqs : 1; /**< TDQS Enable
|
|
0 = disable
|
|
LMC writes this value to MR1[TDQS] in the DDR3 parts in the selected ranks
|
|
during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK,INIT_STATUS] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t wlev : 1; /**< Write Leveling Enable
|
|
0 = disable
|
|
LMC writes MR1[Level]=0 in the DDR3 parts in the selected ranks
|
|
during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
(Write-leveling can only be initiated via the
|
|
write-leveling instruction sequence.)
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK,INIT_STATUS] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t al : 2; /**< Additive Latency
|
|
00 = 0
|
|
01 = CL-1
|
|
10 = CL-2
|
|
11 = Reserved
|
|
LMC writes this value to MR1[AL] in the selected DDR3 parts
|
|
during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
This value must equal the MR1[AL] value in all the DDR3
|
|
parts attached to all ranks during normal operation.
|
|
See also LMC*_CONTROL[POCAS]. */
|
|
uint64_t dll : 1; /**< DLL Enable
|
|
0 = enable
|
|
1 = disable.
|
|
LMC writes this value to MR1[DLL] in the selected DDR3 parts
|
|
during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
This value must equal the MR1[DLL] value in all the DDR3
|
|
parts attached to all ranks during normal operation.
|
|
In dll-off mode, CL/CWL must be programmed
|
|
equal to 6/6, respectively, as per the DDR3 specifications. */
|
|
uint64_t mpr : 1; /**< MPR
|
|
LMC writes this value to MR3[MPR] in the selected DDR3 parts
|
|
during power-up/init, read-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
(LMC also writes MR3[MPR]=1 at the beginning of the
|
|
read-leveling instruction sequence. Read-leveling should only be initiated via the
|
|
read-leveling instruction sequence.)
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
The MR3[MPR] value must be 0 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t mprloc : 2; /**< MPR Location
|
|
LMC writes this value to MR3[MPRLoc] in the selected DDR3 parts
|
|
during power-up/init, read-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh exit instruction sequences.
|
|
(LMC also writes MR3[MPRLoc]=0 at the beginning of the
|
|
read-leveling instruction sequence.)
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK].
|
|
The MR3[MPRLoc] value must be 0 in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
uint64_t cwl : 3; /**< CAS Write Latency
|
|
- 000: 5
|
|
- 001: 6
|
|
- 010: 7
|
|
- 011: 8
|
|
- 100: 9
|
|
- 101: 10
|
|
- 110: 11
|
|
- 111: 12
|
|
LMC writes this value to MR2[CWL] in the selected DDR3 parts
|
|
during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
This value must equal the MR2[CWL] value in all the DDR3
|
|
parts attached to all ranks during normal operation. */
|
|
#else
|
|
uint64_t cwl : 3;
|
|
uint64_t mprloc : 2;
|
|
uint64_t mpr : 1;
|
|
uint64_t dll : 1;
|
|
uint64_t al : 2;
|
|
uint64_t wlev : 1;
|
|
uint64_t tdqs : 1;
|
|
uint64_t qoff : 1;
|
|
uint64_t bl : 2;
|
|
uint64_t cl : 4;
|
|
uint64_t rbt : 1;
|
|
uint64_t tm : 1;
|
|
uint64_t dllr : 1;
|
|
uint64_t wrp : 3;
|
|
uint64_t ppd : 1;
|
|
uint64_t reserved_25_63 : 39;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_modereg_params0_s cn61xx;
|
|
struct cvmx_lmcx_modereg_params0_s cn63xx;
|
|
struct cvmx_lmcx_modereg_params0_s cn63xxp1;
|
|
struct cvmx_lmcx_modereg_params0_s cn66xx;
|
|
struct cvmx_lmcx_modereg_params0_s cn68xx;
|
|
struct cvmx_lmcx_modereg_params0_s cn68xxp1;
|
|
struct cvmx_lmcx_modereg_params0_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_modereg_params0 cvmx_lmcx_modereg_params0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_modereg_params1
|
|
*
|
|
* Notes:
|
|
* These parameters are written into the DDR3 MR0, MR1, MR2 and MR3 registers.
|
|
*
|
|
*/
|
|
union cvmx_lmcx_modereg_params1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_modereg_params1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_48_63 : 16;
|
|
uint64_t rtt_nom_11 : 3; /**< RTT_NOM Rank 3
|
|
LMC writes this value to MR1[Rtt_Nom] in the rank 3 (i.e. DIMM1_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
Per JEDEC DDR3 specifications, if RTT_Nom is used during writes,
|
|
only values MR1[Rtt_Nom] = 1 (RQZ/4), 2 (RQZ/2), or 3 (RQZ/6) are allowed.
|
|
Otherwise, values MR1[Rtt_Nom] = 4 (RQZ/12) and 5 (RQZ/8) are also allowed. */
|
|
uint64_t dic_11 : 2; /**< Output Driver Impedance Control Rank 3
|
|
LMC writes this value to MR1[D.I.C.] in the rank 3 (i.e. DIMM1_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_wr_11 : 2; /**< RTT_WR Rank 3
|
|
LMC writes this value to MR2[Rtt_WR] in the rank 3 (i.e. DIMM1_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t srt_11 : 1; /**< Self-refresh temperature range Rank 3
|
|
LMC writes this value to MR2[SRT] in the rank 3 (i.e. DIMM1_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t asr_11 : 1; /**< Auto self-refresh Rank 3
|
|
LMC writes this value to MR2[ASR] in the rank 3 (i.e. DIMM1_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t pasr_11 : 3; /**< Partial array self-refresh Rank 3
|
|
LMC writes this value to MR2[PASR] in the rank 3 (i.e. DIMM1_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_nom_10 : 3; /**< RTT_NOM Rank 2
|
|
LMC writes this value to MR1[Rtt_Nom] in the rank 2 (i.e. DIMM1_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
Per JEDEC DDR3 specifications, if RTT_Nom is used during writes,
|
|
only values MR1[Rtt_Nom] = 1 (RQZ/4), 2 (RQZ/2), or 3 (RQZ/6) are allowed.
|
|
Otherwise, values MR1[Rtt_Nom] = 4 (RQZ/12) and 5 (RQZ/8) are also allowed. */
|
|
uint64_t dic_10 : 2; /**< Output Driver Impedance Control Rank 2
|
|
LMC writes this value to MR1[D.I.C.] in the rank 2 (i.e. DIMM1_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_wr_10 : 2; /**< RTT_WR Rank 2
|
|
LMC writes this value to MR2[Rtt_WR] in the rank 2 (i.e. DIMM1_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t srt_10 : 1; /**< Self-refresh temperature range Rank 2
|
|
LMC writes this value to MR2[SRT] in the rank 2 (i.e. DIMM1_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t asr_10 : 1; /**< Auto self-refresh Rank 2
|
|
LMC writes this value to MR2[ASR] in the rank 2 (i.e. DIMM1_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t pasr_10 : 3; /**< Partial array self-refresh Rank 2
|
|
LMC writes this value to MR2[PASR] in the rank 2 (i.e. DIMM1_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_nom_01 : 3; /**< RTT_NOM Rank 1
|
|
LMC writes this value to MR1[Rtt_Nom] in the rank 1 (i.e. DIMM0_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
Per JEDEC DDR3 specifications, if RTT_Nom is used during writes,
|
|
only values MR1[Rtt_Nom] = 1 (RQZ/4), 2 (RQZ/2), or 3 (RQZ/6) are allowed.
|
|
Otherwise, values MR1[Rtt_Nom] = 4 (RQZ/12) and 5 (RQZ/8) are also allowed. */
|
|
uint64_t dic_01 : 2; /**< Output Driver Impedance Control Rank 1
|
|
LMC writes this value to MR1[D.I.C.] in the rank 1 (i.e. DIMM0_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_wr_01 : 2; /**< RTT_WR Rank 1
|
|
LMC writes this value to MR2[Rtt_WR] in the rank 1 (i.e. DIMM0_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t srt_01 : 1; /**< Self-refresh temperature range Rank 1
|
|
LMC writes this value to MR2[SRT] in the rank 1 (i.e. DIMM0_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t asr_01 : 1; /**< Auto self-refresh Rank 1
|
|
LMC writes this value to MR2[ASR] in the rank 1 (i.e. DIMM0_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t pasr_01 : 3; /**< Partial array self-refresh Rank 1
|
|
LMC writes this value to MR2[PASR] in the rank 1 (i.e. DIMM0_CS1) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_nom_00 : 3; /**< RTT_NOM Rank 0
|
|
LMC writes this value to MR1[Rtt_Nom] in the rank 0 (i.e. DIMM0_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT].
|
|
Per JEDEC DDR3 specifications, if RTT_Nom is used during writes,
|
|
only values MR1[Rtt_Nom] = 1 (RQZ/4), 2 (RQZ/2), or 3 (RQZ/6) are allowed.
|
|
Otherwise, values MR1[Rtt_Nom] = 4 (RQZ/12) and 5 (RQZ/8) are also allowed. */
|
|
uint64_t dic_00 : 2; /**< Output Driver Impedance Control Rank 0
|
|
LMC writes this value to MR1[D.I.C.] in the rank 0 (i.e. DIMM0_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t rtt_wr_00 : 2; /**< RTT_WR Rank 0
|
|
LMC writes this value to MR2[Rtt_WR] in the rank 0 (i.e. DIMM0_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t srt_00 : 1; /**< Self-refresh temperature range Rank 0
|
|
LMC writes this value to MR2[SRT] in the rank 0 (i.e. DIMM0_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t asr_00 : 1; /**< Auto self-refresh Rank 0
|
|
LMC writes this value to MR2[ASR] in the rank 0 (i.e. DIMM0_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
uint64_t pasr_00 : 3; /**< Partial array self-refresh Rank 0
|
|
LMC writes this value to MR2[PASR] in the rank 0 (i.e. DIMM0_CS0) DDR3 parts
|
|
when selected during power-up/init, write-leveling, and, if LMC*_CONFIG[SREF_WITH_DLL] is set,
|
|
self-refresh entry and exit instruction sequences.
|
|
See LMC*_CONFIG[SEQUENCE,INIT_START,RANKMASK] and
|
|
LMC*_RESET_CTL[DDR3PWARM,DDR3PSOFT]. */
|
|
#else
|
|
uint64_t pasr_00 : 3;
|
|
uint64_t asr_00 : 1;
|
|
uint64_t srt_00 : 1;
|
|
uint64_t rtt_wr_00 : 2;
|
|
uint64_t dic_00 : 2;
|
|
uint64_t rtt_nom_00 : 3;
|
|
uint64_t pasr_01 : 3;
|
|
uint64_t asr_01 : 1;
|
|
uint64_t srt_01 : 1;
|
|
uint64_t rtt_wr_01 : 2;
|
|
uint64_t dic_01 : 2;
|
|
uint64_t rtt_nom_01 : 3;
|
|
uint64_t pasr_10 : 3;
|
|
uint64_t asr_10 : 1;
|
|
uint64_t srt_10 : 1;
|
|
uint64_t rtt_wr_10 : 2;
|
|
uint64_t dic_10 : 2;
|
|
uint64_t rtt_nom_10 : 3;
|
|
uint64_t pasr_11 : 3;
|
|
uint64_t asr_11 : 1;
|
|
uint64_t srt_11 : 1;
|
|
uint64_t rtt_wr_11 : 2;
|
|
uint64_t dic_11 : 2;
|
|
uint64_t rtt_nom_11 : 3;
|
|
uint64_t reserved_48_63 : 16;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_modereg_params1_s cn61xx;
|
|
struct cvmx_lmcx_modereg_params1_s cn63xx;
|
|
struct cvmx_lmcx_modereg_params1_s cn63xxp1;
|
|
struct cvmx_lmcx_modereg_params1_s cn66xx;
|
|
struct cvmx_lmcx_modereg_params1_s cn68xx;
|
|
struct cvmx_lmcx_modereg_params1_s cn68xxp1;
|
|
struct cvmx_lmcx_modereg_params1_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_modereg_params1 cvmx_lmcx_modereg_params1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_nxm
|
|
*
|
|
* LMC_NXM = LMC non-existent memory
|
|
*
|
|
*
|
|
* Notes:
|
|
* Decoding for mem_msb/rank
|
|
* - 0000: mem_msb = mem_adr[25]
|
|
* - 0001: mem_msb = mem_adr[26]
|
|
* - 0010: mem_msb = mem_adr[27]
|
|
* - 0011: mem_msb = mem_adr[28]
|
|
* - 0100: mem_msb = mem_adr[29]
|
|
* - 0101: mem_msb = mem_adr[30]
|
|
* - 0110: mem_msb = mem_adr[31]
|
|
* - 0111: mem_msb = mem_adr[32]
|
|
* - 1000: mem_msb = mem_adr[33]
|
|
* - 1001: mem_msb = mem_adr[34]
|
|
* 1010-1111 = Reserved
|
|
* For example, for a DIMM made of Samsung's k4b1g0846c-f7 1Gb (16M x 8 bit x 8 bank)
|
|
* DDR3 parts, the column address width = 10, so with
|
|
* 10b of col, 3b of bus, 3b of bank, row_lsb = 16. So, row = mem_adr[29:16] and
|
|
* mem_msb = 4
|
|
*
|
|
* Note also that addresses greater the max defined space (pbank_msb) are also treated
|
|
* as NXM accesses
|
|
*/
|
|
union cvmx_lmcx_nxm {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_nxm_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_40_63 : 24;
|
|
uint64_t mem_msb_d3_r1 : 4; /**< Max Row MSB for DIMM3, RANK1/DIMM3 in Single Ranked
|
|
*UNUSED IN 6xxx* */
|
|
uint64_t mem_msb_d3_r0 : 4; /**< Max Row MSB for DIMM3, RANK0
|
|
*UNUSED IN 6xxx* */
|
|
uint64_t mem_msb_d2_r1 : 4; /**< Max Row MSB for DIMM2, RANK1/DIMM2 in Single Ranked
|
|
*UNUSED IN 6xxx* */
|
|
uint64_t mem_msb_d2_r0 : 4; /**< Max Row MSB for DIMM2, RANK0
|
|
*UNUSED IN 6xxx* */
|
|
uint64_t mem_msb_d1_r1 : 4; /**< Max Row MSB for DIMM1, RANK1/DIMM1 in Single Ranked */
|
|
uint64_t mem_msb_d1_r0 : 4; /**< Max Row MSB for DIMM1, RANK0 */
|
|
uint64_t mem_msb_d0_r1 : 4; /**< Max Row MSB for DIMM0, RANK1/DIMM0 in Single Ranked */
|
|
uint64_t mem_msb_d0_r0 : 4; /**< Max Row MSB for DIMM0, RANK0 */
|
|
uint64_t cs_mask : 8; /**< Chip select mask.
|
|
This mask corresponds to the 8 chip selects for a memory
|
|
configuration. If LMC*_CONFIG[RANK_ENA]==0 then this
|
|
mask must be set in pairs because each reference address
|
|
will assert a pair of chip selects. If the chip
|
|
select(s) have a corresponding CS_MASK bit set, then the
|
|
reference is to non-existent memory (NXM). LMC will alias a
|
|
NXM read reference to use the lowest, legal chip select(s)
|
|
and return 0's. LMC normally discards NXM writes, but will
|
|
also alias them when LMC*_CONTROL[NXM_WRITE_EN]=1.
|
|
CS_MASK<7:4> MBZ in 6xxx */
|
|
#else
|
|
uint64_t cs_mask : 8;
|
|
uint64_t mem_msb_d0_r0 : 4;
|
|
uint64_t mem_msb_d0_r1 : 4;
|
|
uint64_t mem_msb_d1_r0 : 4;
|
|
uint64_t mem_msb_d1_r1 : 4;
|
|
uint64_t mem_msb_d2_r0 : 4;
|
|
uint64_t mem_msb_d2_r1 : 4;
|
|
uint64_t mem_msb_d3_r0 : 4;
|
|
uint64_t mem_msb_d3_r1 : 4;
|
|
uint64_t reserved_40_63 : 24;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_nxm_cn52xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_8_63 : 56;
|
|
uint64_t cs_mask : 8; /**< Chip select mask.
|
|
This mask corresponds to the 8 chip selects for a memory
|
|
configuration. If LMC_MEM_CFG0[BUNK_ENA]==0 then this
|
|
mask must be set in pairs because each reference address
|
|
will assert a pair of chip selects. If the chip
|
|
select(s) have a corresponding CS_MASK bit set, then the
|
|
reference is to non-existent memory. LMC will alias the
|
|
reference to use the lowest, legal chip select(s) in
|
|
that case. */
|
|
#else
|
|
uint64_t cs_mask : 8;
|
|
uint64_t reserved_8_63 : 56;
|
|
#endif
|
|
} cn52xx;
|
|
struct cvmx_lmcx_nxm_cn52xx cn56xx;
|
|
struct cvmx_lmcx_nxm_cn52xx cn58xx;
|
|
struct cvmx_lmcx_nxm_s cn61xx;
|
|
struct cvmx_lmcx_nxm_s cn63xx;
|
|
struct cvmx_lmcx_nxm_s cn63xxp1;
|
|
struct cvmx_lmcx_nxm_s cn66xx;
|
|
struct cvmx_lmcx_nxm_s cn68xx;
|
|
struct cvmx_lmcx_nxm_s cn68xxp1;
|
|
struct cvmx_lmcx_nxm_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_nxm cvmx_lmcx_nxm_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ops_cnt
|
|
*
|
|
* LMC_OPS_CNT = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ops_cnt {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ops_cnt_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t opscnt : 64; /**< Performance Counter
|
|
64-bit counter that increments when the DDR3 data bus
|
|
is being used
|
|
DRAM bus utilization = LMC*_OPS_CNT/LMC*_DCLK_CNT */
|
|
#else
|
|
uint64_t opscnt : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ops_cnt_s cn61xx;
|
|
struct cvmx_lmcx_ops_cnt_s cn63xx;
|
|
struct cvmx_lmcx_ops_cnt_s cn63xxp1;
|
|
struct cvmx_lmcx_ops_cnt_s cn66xx;
|
|
struct cvmx_lmcx_ops_cnt_s cn68xx;
|
|
struct cvmx_lmcx_ops_cnt_s cn68xxp1;
|
|
struct cvmx_lmcx_ops_cnt_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_ops_cnt cvmx_lmcx_ops_cnt_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ops_cnt_hi
|
|
*
|
|
* LMC_OPS_CNT_HI = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ops_cnt_hi {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ops_cnt_hi_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t opscnt_hi : 32; /**< Performance Counter to measure Bus Utilization
|
|
Upper 32-bits of 64-bit counter
|
|
DRAM bus utilization = LMC_OPS_CNT_* /LMC_DCLK_CNT_* */
|
|
#else
|
|
uint64_t opscnt_hi : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn30xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn31xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn38xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn38xxp2;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn50xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn52xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn52xxp1;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn56xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn56xxp1;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn58xx;
|
|
struct cvmx_lmcx_ops_cnt_hi_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ops_cnt_hi cvmx_lmcx_ops_cnt_hi_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_ops_cnt_lo
|
|
*
|
|
* LMC_OPS_CNT_LO = Performance Counters
|
|
*
|
|
*/
|
|
union cvmx_lmcx_ops_cnt_lo {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_ops_cnt_lo_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t opscnt_lo : 32; /**< Performance Counter
|
|
Low 32-bits of 64-bit counter
|
|
DRAM bus utilization = LMC_OPS_CNT_* /LMC_DCLK_CNT_* */
|
|
#else
|
|
uint64_t opscnt_lo : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn30xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn31xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn38xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn38xxp2;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn50xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn52xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn52xxp1;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn56xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn56xxp1;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn58xx;
|
|
struct cvmx_lmcx_ops_cnt_lo_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_ops_cnt_lo cvmx_lmcx_ops_cnt_lo_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_phy_ctl
|
|
*
|
|
* LMC_PHY_CTL = LMC PHY Control
|
|
*
|
|
*/
|
|
union cvmx_lmcx_phy_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_phy_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_15_63 : 49;
|
|
uint64_t rx_always_on : 1; /**< Disable dynamic DDR3 IO Rx power gating */
|
|
uint64_t lv_mode : 1; /**< Low Voltage Mode (1.35V) */
|
|
uint64_t ck_tune1 : 1; /**< Clock Tune */
|
|
uint64_t ck_dlyout1 : 4; /**< Clock delay out setting */
|
|
uint64_t ck_tune0 : 1; /**< Clock Tune */
|
|
uint64_t ck_dlyout0 : 4; /**< Clock delay out setting */
|
|
uint64_t loopback : 1; /**< Loopback enable */
|
|
uint64_t loopback_pos : 1; /**< Loopback pos mode */
|
|
uint64_t ts_stagger : 1; /**< TS Staggermode
|
|
This mode configures output drivers with 2-stage drive
|
|
strength to avoid undershoot issues on the bus when strong
|
|
drivers are suddenly turned on. When this mode is asserted,
|
|
Octeon will configure output drivers to be weak drivers
|
|
(60 ohm output impedance) at the first CK cycle, and
|
|
change drivers to the designated drive strengths specified
|
|
in $LMC(0)_COMP_CTL2 [CMD_CTL/CK_CTL/DQX_CTL] starting
|
|
at the following cycle */
|
|
#else
|
|
uint64_t ts_stagger : 1;
|
|
uint64_t loopback_pos : 1;
|
|
uint64_t loopback : 1;
|
|
uint64_t ck_dlyout0 : 4;
|
|
uint64_t ck_tune0 : 1;
|
|
uint64_t ck_dlyout1 : 4;
|
|
uint64_t ck_tune1 : 1;
|
|
uint64_t lv_mode : 1;
|
|
uint64_t rx_always_on : 1;
|
|
uint64_t reserved_15_63 : 49;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_phy_ctl_s cn61xx;
|
|
struct cvmx_lmcx_phy_ctl_s cn63xx;
|
|
struct cvmx_lmcx_phy_ctl_cn63xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_14_63 : 50;
|
|
uint64_t lv_mode : 1; /**< Low Voltage Mode (1.35V) */
|
|
uint64_t ck_tune1 : 1; /**< Clock Tune */
|
|
uint64_t ck_dlyout1 : 4; /**< Clock delay out setting */
|
|
uint64_t ck_tune0 : 1; /**< Clock Tune */
|
|
uint64_t ck_dlyout0 : 4; /**< Clock delay out setting */
|
|
uint64_t loopback : 1; /**< Loopback enable */
|
|
uint64_t loopback_pos : 1; /**< Loopback pos mode */
|
|
uint64_t ts_stagger : 1; /**< TS Staggermode
|
|
This mode configures output drivers with 2-stage drive
|
|
strength to avoid undershoot issues on the bus when strong
|
|
drivers are suddenly turned on. When this mode is asserted,
|
|
Octeon will configure output drivers to be weak drivers
|
|
(60 ohm output impedance) at the first CK cycle, and
|
|
change drivers to the designated drive strengths specified
|
|
in $LMC(0)_COMP_CTL2 [CMD_CTL/CK_CTL/DQX_CTL] starting
|
|
at the following cycle */
|
|
#else
|
|
uint64_t ts_stagger : 1;
|
|
uint64_t loopback_pos : 1;
|
|
uint64_t loopback : 1;
|
|
uint64_t ck_dlyout0 : 4;
|
|
uint64_t ck_tune0 : 1;
|
|
uint64_t ck_dlyout1 : 4;
|
|
uint64_t ck_tune1 : 1;
|
|
uint64_t lv_mode : 1;
|
|
uint64_t reserved_14_63 : 50;
|
|
#endif
|
|
} cn63xxp1;
|
|
struct cvmx_lmcx_phy_ctl_s cn66xx;
|
|
struct cvmx_lmcx_phy_ctl_s cn68xx;
|
|
struct cvmx_lmcx_phy_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_phy_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_phy_ctl cvmx_lmcx_phy_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_pll_bwctl
|
|
*
|
|
* LMC_PLL_BWCTL = DDR PLL Bandwidth Control Register
|
|
*
|
|
*/
|
|
union cvmx_lmcx_pll_bwctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_pll_bwctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_5_63 : 59;
|
|
uint64_t bwupd : 1; /**< Load this Bandwidth Register value into the PLL */
|
|
uint64_t bwctl : 4; /**< Bandwidth Control Register for DDR PLL */
|
|
#else
|
|
uint64_t bwctl : 4;
|
|
uint64_t bwupd : 1;
|
|
uint64_t reserved_5_63 : 59;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_pll_bwctl_s cn30xx;
|
|
struct cvmx_lmcx_pll_bwctl_s cn31xx;
|
|
struct cvmx_lmcx_pll_bwctl_s cn38xx;
|
|
struct cvmx_lmcx_pll_bwctl_s cn38xxp2;
|
|
};
|
|
typedef union cvmx_lmcx_pll_bwctl cvmx_lmcx_pll_bwctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_pll_ctl
|
|
*
|
|
* LMC_PLL_CTL = LMC pll control
|
|
*
|
|
*
|
|
* Notes:
|
|
* This CSR is only relevant for LMC0. LMC1_PLL_CTL is not used.
|
|
*
|
|
* Exactly one of EN2, EN4, EN6, EN8, EN12, EN16 must be set.
|
|
*
|
|
* The resultant DDR_CK frequency is the DDR2_REF_CLK
|
|
* frequency multiplied by:
|
|
*
|
|
* (CLKF + 1) / ((CLKR + 1) * EN(2,4,6,8,12,16))
|
|
*
|
|
* The PLL frequency, which is:
|
|
*
|
|
* (DDR2_REF_CLK freq) * ((CLKF + 1) / (CLKR + 1))
|
|
*
|
|
* must reside between 1.2 and 2.5 GHz. A faster PLL frequency is desirable if there is a choice.
|
|
*/
|
|
union cvmx_lmcx_pll_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_pll_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_30_63 : 34;
|
|
uint64_t bypass : 1; /**< PLL Bypass */
|
|
uint64_t fasten_n : 1; /**< Should be set, especially when CLKF > ~80 */
|
|
uint64_t div_reset : 1; /**< Analog pll divider reset
|
|
De-assert at least 500*(CLKR+1) reference clock
|
|
cycles following RESET_N de-assertion. */
|
|
uint64_t reset_n : 1; /**< Analog pll reset
|
|
De-assert at least 5 usec after CLKF, CLKR,
|
|
and EN* are set up. */
|
|
uint64_t clkf : 12; /**< Multiply reference by CLKF + 1
|
|
CLKF must be <= 128 */
|
|
uint64_t clkr : 6; /**< Divide reference by CLKR + 1 */
|
|
uint64_t reserved_6_7 : 2;
|
|
uint64_t en16 : 1; /**< Divide output by 16 */
|
|
uint64_t en12 : 1; /**< Divide output by 12 */
|
|
uint64_t en8 : 1; /**< Divide output by 8 */
|
|
uint64_t en6 : 1; /**< Divide output by 6 */
|
|
uint64_t en4 : 1; /**< Divide output by 4 */
|
|
uint64_t en2 : 1; /**< Divide output by 2 */
|
|
#else
|
|
uint64_t en2 : 1;
|
|
uint64_t en4 : 1;
|
|
uint64_t en6 : 1;
|
|
uint64_t en8 : 1;
|
|
uint64_t en12 : 1;
|
|
uint64_t en16 : 1;
|
|
uint64_t reserved_6_7 : 2;
|
|
uint64_t clkr : 6;
|
|
uint64_t clkf : 12;
|
|
uint64_t reset_n : 1;
|
|
uint64_t div_reset : 1;
|
|
uint64_t fasten_n : 1;
|
|
uint64_t bypass : 1;
|
|
uint64_t reserved_30_63 : 34;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_pll_ctl_cn50xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_29_63 : 35;
|
|
uint64_t fasten_n : 1; /**< Should be set, especially when CLKF > ~80 */
|
|
uint64_t div_reset : 1; /**< Analog pll divider reset
|
|
De-assert at least 500*(CLKR+1) reference clock
|
|
cycles following RESET_N de-assertion. */
|
|
uint64_t reset_n : 1; /**< Analog pll reset
|
|
De-assert at least 5 usec after CLKF, CLKR,
|
|
and EN* are set up. */
|
|
uint64_t clkf : 12; /**< Multiply reference by CLKF + 1
|
|
CLKF must be <= 256 */
|
|
uint64_t clkr : 6; /**< Divide reference by CLKR + 1 */
|
|
uint64_t reserved_6_7 : 2;
|
|
uint64_t en16 : 1; /**< Divide output by 16 */
|
|
uint64_t en12 : 1; /**< Divide output by 12 */
|
|
uint64_t en8 : 1; /**< Divide output by 8 */
|
|
uint64_t en6 : 1; /**< Divide output by 6 */
|
|
uint64_t en4 : 1; /**< Divide output by 4 */
|
|
uint64_t en2 : 1; /**< Divide output by 2 */
|
|
#else
|
|
uint64_t en2 : 1;
|
|
uint64_t en4 : 1;
|
|
uint64_t en6 : 1;
|
|
uint64_t en8 : 1;
|
|
uint64_t en12 : 1;
|
|
uint64_t en16 : 1;
|
|
uint64_t reserved_6_7 : 2;
|
|
uint64_t clkr : 6;
|
|
uint64_t clkf : 12;
|
|
uint64_t reset_n : 1;
|
|
uint64_t div_reset : 1;
|
|
uint64_t fasten_n : 1;
|
|
uint64_t reserved_29_63 : 35;
|
|
#endif
|
|
} cn50xx;
|
|
struct cvmx_lmcx_pll_ctl_s cn52xx;
|
|
struct cvmx_lmcx_pll_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_pll_ctl_cn50xx cn56xx;
|
|
struct cvmx_lmcx_pll_ctl_cn56xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_28_63 : 36;
|
|
uint64_t div_reset : 1; /**< Analog pll divider reset
|
|
De-assert at least 500*(CLKR+1) reference clock
|
|
cycles following RESET_N de-assertion. */
|
|
uint64_t reset_n : 1; /**< Analog pll reset
|
|
De-assert at least 5 usec after CLKF, CLKR,
|
|
and EN* are set up. */
|
|
uint64_t clkf : 12; /**< Multiply reference by CLKF + 1
|
|
CLKF must be <= 128 */
|
|
uint64_t clkr : 6; /**< Divide reference by CLKR + 1 */
|
|
uint64_t reserved_6_7 : 2;
|
|
uint64_t en16 : 1; /**< Divide output by 16 */
|
|
uint64_t en12 : 1; /**< Divide output by 12 */
|
|
uint64_t en8 : 1; /**< Divide output by 8 */
|
|
uint64_t en6 : 1; /**< Divide output by 6 */
|
|
uint64_t en4 : 1; /**< Divide output by 4 */
|
|
uint64_t en2 : 1; /**< Divide output by 2 */
|
|
#else
|
|
uint64_t en2 : 1;
|
|
uint64_t en4 : 1;
|
|
uint64_t en6 : 1;
|
|
uint64_t en8 : 1;
|
|
uint64_t en12 : 1;
|
|
uint64_t en16 : 1;
|
|
uint64_t reserved_6_7 : 2;
|
|
uint64_t clkr : 6;
|
|
uint64_t clkf : 12;
|
|
uint64_t reset_n : 1;
|
|
uint64_t div_reset : 1;
|
|
uint64_t reserved_28_63 : 36;
|
|
#endif
|
|
} cn56xxp1;
|
|
struct cvmx_lmcx_pll_ctl_cn56xxp1 cn58xx;
|
|
struct cvmx_lmcx_pll_ctl_cn56xxp1 cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_pll_ctl cvmx_lmcx_pll_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_pll_status
|
|
*
|
|
* LMC_PLL_STATUS = LMC pll status
|
|
*
|
|
*/
|
|
union cvmx_lmcx_pll_status {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_pll_status_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ddr__nctl : 5; /**< DDR nctl from compensation circuit */
|
|
uint64_t ddr__pctl : 5; /**< DDR pctl from compensation circuit */
|
|
uint64_t reserved_2_21 : 20;
|
|
uint64_t rfslip : 1; /**< Reference clock slip */
|
|
uint64_t fbslip : 1; /**< Feedback clock slip */
|
|
#else
|
|
uint64_t fbslip : 1;
|
|
uint64_t rfslip : 1;
|
|
uint64_t reserved_2_21 : 20;
|
|
uint64_t ddr__pctl : 5;
|
|
uint64_t ddr__nctl : 5;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_pll_status_s cn50xx;
|
|
struct cvmx_lmcx_pll_status_s cn52xx;
|
|
struct cvmx_lmcx_pll_status_s cn52xxp1;
|
|
struct cvmx_lmcx_pll_status_s cn56xx;
|
|
struct cvmx_lmcx_pll_status_s cn56xxp1;
|
|
struct cvmx_lmcx_pll_status_s cn58xx;
|
|
struct cvmx_lmcx_pll_status_cn58xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_2_63 : 62;
|
|
uint64_t rfslip : 1; /**< Reference clock slip */
|
|
uint64_t fbslip : 1; /**< Feedback clock slip */
|
|
#else
|
|
uint64_t fbslip : 1;
|
|
uint64_t rfslip : 1;
|
|
uint64_t reserved_2_63 : 62;
|
|
#endif
|
|
} cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_pll_status cvmx_lmcx_pll_status_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_read_level_ctl
|
|
*
|
|
* Notes:
|
|
* The HW writes and reads the cache block selected by ROW, COL, BNK and the rank as part of a read-leveling sequence for a rank.
|
|
* A cache block write is 16 72-bit words. PATTERN selects the write value. For the first 8
|
|
* words, the write value is the bit PATTERN<i> duplicated into a 72-bit vector. The write value of
|
|
* the last 8 words is the inverse of the write value of the first 8 words.
|
|
* See LMC*_READ_LEVEL_RANK*.
|
|
*/
|
|
union cvmx_lmcx_read_level_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_read_level_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_44_63 : 20;
|
|
uint64_t rankmask : 4; /**< Selects ranks to be leveled
|
|
to read-level rank i, set RANKMASK<i> */
|
|
uint64_t pattern : 8; /**< All DQ driven to PATTERN[burst], 0 <= burst <= 7
|
|
All DQ driven to ~PATTERN[burst-8], 8 <= burst <= 15 */
|
|
uint64_t row : 16; /**< Row address used to write/read data pattern */
|
|
uint64_t col : 12; /**< Column address used to write/read data pattern */
|
|
uint64_t reserved_3_3 : 1;
|
|
uint64_t bnk : 3; /**< Bank address used to write/read data pattern */
|
|
#else
|
|
uint64_t bnk : 3;
|
|
uint64_t reserved_3_3 : 1;
|
|
uint64_t col : 12;
|
|
uint64_t row : 16;
|
|
uint64_t pattern : 8;
|
|
uint64_t rankmask : 4;
|
|
uint64_t reserved_44_63 : 20;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_read_level_ctl_s cn52xx;
|
|
struct cvmx_lmcx_read_level_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_read_level_ctl_s cn56xx;
|
|
struct cvmx_lmcx_read_level_ctl_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_read_level_ctl cvmx_lmcx_read_level_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_read_level_dbg
|
|
*
|
|
* Notes:
|
|
* A given read of LMC*_READ_LEVEL_DBG returns the read-leveling pass/fail results for all possible
|
|
* delay settings (i.e. the BITMASK) for only one byte in the last rank that the HW read-leveled.
|
|
* LMC*_READ_LEVEL_DBG[BYTE] selects the particular byte.
|
|
* To get these pass/fail results for another different rank, you must run the hardware read-leveling
|
|
* again. For example, it is possible to get the BITMASK results for every byte of every rank
|
|
* if you run read-leveling separately for each rank, probing LMC*_READ_LEVEL_DBG between each
|
|
* read-leveling.
|
|
*/
|
|
union cvmx_lmcx_read_level_dbg {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_read_level_dbg_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t bitmask : 16; /**< Bitmask generated during deskew settings sweep
|
|
BITMASK[n]=0 means deskew setting n failed
|
|
BITMASK[n]=1 means deskew setting n passed
|
|
for 0 <= n <= 15 */
|
|
uint64_t reserved_4_15 : 12;
|
|
uint64_t byte : 4; /**< 0 <= BYTE <= 8 */
|
|
#else
|
|
uint64_t byte : 4;
|
|
uint64_t reserved_4_15 : 12;
|
|
uint64_t bitmask : 16;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_read_level_dbg_s cn52xx;
|
|
struct cvmx_lmcx_read_level_dbg_s cn52xxp1;
|
|
struct cvmx_lmcx_read_level_dbg_s cn56xx;
|
|
struct cvmx_lmcx_read_level_dbg_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_read_level_dbg cvmx_lmcx_read_level_dbg_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_read_level_rank#
|
|
*
|
|
* Notes:
|
|
* This is four CSRs per LMC, one per each rank.
|
|
* Each CSR is written by HW during a read-leveling sequence for the rank. (HW sets STATUS==3 after HW read-leveling completes for the rank.)
|
|
* Each CSR may also be written by SW, but not while a read-leveling sequence is in progress. (HW sets STATUS==1 after a CSR write.)
|
|
* Deskew setting is measured in units of 1/4 DCLK, so the above BYTE* values can range over 4 DCLKs.
|
|
* SW initiates a HW read-leveling sequence by programming LMC*_READ_LEVEL_CTL and writing INIT_START=1 with SEQUENCE=1.
|
|
* See LMC*_READ_LEVEL_CTL.
|
|
*/
|
|
union cvmx_lmcx_read_level_rankx {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_read_level_rankx_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_38_63 : 26;
|
|
uint64_t status : 2; /**< Indicates status of the read-levelling and where
|
|
the BYTE* programmings in <35:0> came from:
|
|
0 = BYTE* values are their reset value
|
|
1 = BYTE* values were set via a CSR write to this register
|
|
2 = read-leveling sequence currently in progress (BYTE* values are unpredictable)
|
|
3 = BYTE* values came from a complete read-leveling sequence */
|
|
uint64_t byte8 : 4; /**< Deskew setting */
|
|
uint64_t byte7 : 4; /**< Deskew setting */
|
|
uint64_t byte6 : 4; /**< Deskew setting */
|
|
uint64_t byte5 : 4; /**< Deskew setting */
|
|
uint64_t byte4 : 4; /**< Deskew setting */
|
|
uint64_t byte3 : 4; /**< Deskew setting */
|
|
uint64_t byte2 : 4; /**< Deskew setting */
|
|
uint64_t byte1 : 4; /**< Deskew setting */
|
|
uint64_t byte0 : 4; /**< Deskew setting */
|
|
#else
|
|
uint64_t byte0 : 4;
|
|
uint64_t byte1 : 4;
|
|
uint64_t byte2 : 4;
|
|
uint64_t byte3 : 4;
|
|
uint64_t byte4 : 4;
|
|
uint64_t byte5 : 4;
|
|
uint64_t byte6 : 4;
|
|
uint64_t byte7 : 4;
|
|
uint64_t byte8 : 4;
|
|
uint64_t status : 2;
|
|
uint64_t reserved_38_63 : 26;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_read_level_rankx_s cn52xx;
|
|
struct cvmx_lmcx_read_level_rankx_s cn52xxp1;
|
|
struct cvmx_lmcx_read_level_rankx_s cn56xx;
|
|
struct cvmx_lmcx_read_level_rankx_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_read_level_rankx cvmx_lmcx_read_level_rankx_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_reset_ctl
|
|
*
|
|
* Specify the RSL base addresses for the block
|
|
*
|
|
*
|
|
* Notes:
|
|
* DDR3RST - DDR3 DRAM parts have a new RESET#
|
|
* pin that wasn't present in DDR2 parts. The
|
|
* DDR3RST CSR field controls the assertion of
|
|
* the new 6xxx pin that attaches to RESET#.
|
|
* When DDR3RST is set, 6xxx asserts RESET#.
|
|
* When DDR3RST is clear, 6xxx de-asserts
|
|
* RESET#.
|
|
*
|
|
* DDR3RST is set on a cold reset. Warm and
|
|
* soft chip resets do not affect the DDR3RST
|
|
* value. Outside of cold reset, only software
|
|
* CSR writes change the DDR3RST value.
|
|
*
|
|
* DDR3PWARM - Enables preserve mode during a warm
|
|
* reset. When set, the DDR3 controller hardware
|
|
* automatically puts the attached DDR3 DRAM parts
|
|
* into self refresh (see LMC*CONFIG[SEQUENCE] below) at the beginning of a warm
|
|
* reset sequence, provided that the DDR3 controller
|
|
* is up. When clear, the DDR3 controller hardware
|
|
* does not put the attached DDR3 DRAM parts into
|
|
* self-refresh during a warm reset sequence.
|
|
*
|
|
* DDR3PWARM is cleared on a cold reset. Warm and
|
|
* soft chip resets do not affect the DDR3PWARM
|
|
* value. Outside of cold reset, only software
|
|
* CSR writes change the DDR3PWARM value.
|
|
*
|
|
* Note that if a warm reset follows a soft reset,
|
|
* DDR3PWARM has no effect, as the DDR3 controller
|
|
* is no longer up after any cold/warm/soft
|
|
* reset sequence.
|
|
*
|
|
* DDR3PSOFT - Enables preserve mode during a soft
|
|
* reset. When set, the DDR3 controller hardware
|
|
* automatically puts the attached DDR3 DRAM parts
|
|
* into self refresh (see LMC*CONFIG[SEQUENCE] below) at the beginning of a soft
|
|
* reset sequence, provided that the DDR3 controller
|
|
* is up. When clear, the DDR3 controller hardware
|
|
* does not put the attached DDR3 DRAM parts into
|
|
* self-refresh during a soft reset sequence.
|
|
*
|
|
* DDR3PSOFT is cleared on a cold reset. Warm and
|
|
* soft chip resets do not affect the DDR3PSOFT
|
|
* value. Outside of cold reset, only software
|
|
* CSR writes change the DDR3PSOFT value.
|
|
*
|
|
* DDR3PSV - May be useful for system software to
|
|
* determine when the DDR3 contents have been
|
|
* preserved.
|
|
*
|
|
* Cleared by hardware during a cold reset. Never
|
|
* cleared by hardware during a warm/soft reset.
|
|
* Set by hardware during a warm/soft reset if
|
|
* the hardware automatically put the DDR3 DRAM
|
|
* into self-refresh during the reset sequence.
|
|
*
|
|
* Can also be written by software (to any value).
|
|
*/
|
|
union cvmx_lmcx_reset_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_reset_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_4_63 : 60;
|
|
uint64_t ddr3psv : 1; /**< Memory Reset
|
|
1 = DDR contents preserved */
|
|
uint64_t ddr3psoft : 1; /**< Memory Reset
|
|
1 = Enable Preserve mode during soft reset */
|
|
uint64_t ddr3pwarm : 1; /**< Memory Reset
|
|
1 = Enable Preserve mode during warm reset */
|
|
uint64_t ddr3rst : 1; /**< Memory Reset
|
|
0 = Reset asserted
|
|
1 = Reset de-asserted */
|
|
#else
|
|
uint64_t ddr3rst : 1;
|
|
uint64_t ddr3pwarm : 1;
|
|
uint64_t ddr3psoft : 1;
|
|
uint64_t ddr3psv : 1;
|
|
uint64_t reserved_4_63 : 60;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_reset_ctl_s cn61xx;
|
|
struct cvmx_lmcx_reset_ctl_s cn63xx;
|
|
struct cvmx_lmcx_reset_ctl_s cn63xxp1;
|
|
struct cvmx_lmcx_reset_ctl_s cn66xx;
|
|
struct cvmx_lmcx_reset_ctl_s cn68xx;
|
|
struct cvmx_lmcx_reset_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_reset_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_reset_ctl cvmx_lmcx_reset_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_rlevel_ctl
|
|
*/
|
|
union cvmx_lmcx_rlevel_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_rlevel_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_22_63 : 42;
|
|
uint64_t delay_unload_3 : 1; /**< When set, unload the PHY silo one cycle later
|
|
during read-leveling if LMC*_RLEVEL_RANKi[BYTE*<1:0>] = 3
|
|
DELAY_UNLOAD_3 should normally be set, particularly at higher speeds. */
|
|
uint64_t delay_unload_2 : 1; /**< When set, unload the PHY silo one cycle later
|
|
during read-leveling if LMC*_RLEVEL_RANKi[BYTE*<1:0>] = 2
|
|
DELAY_UNLOAD_2 should normally not be set. */
|
|
uint64_t delay_unload_1 : 1; /**< When set, unload the PHY silo one cycle later
|
|
during read-leveling if LMC*_RLEVEL_RANKi[BYTE*<1:0>] = 1
|
|
DELAY_UNLOAD_1 should normally not be set. */
|
|
uint64_t delay_unload_0 : 1; /**< When set, unload the PHY silo one cycle later
|
|
during read-leveling if LMC*_RLEVEL_RANKi[BYTE*<1:0>] = 0
|
|
DELAY_UNLOAD_0 should normally not be set. */
|
|
uint64_t bitmask : 8; /**< Mask to select bit lanes on which read-leveling
|
|
feedback is returned when OR_DIS is set to 1 */
|
|
uint64_t or_dis : 1; /**< Disable or'ing of bits in a byte lane when computing
|
|
the read-leveling bitmask
|
|
OR_DIS should normally not be set. */
|
|
uint64_t offset_en : 1; /**< When set, LMC attempts to select the read-leveling
|
|
setting that is LMC*RLEVEL_CTL[OFFSET] settings earlier than the
|
|
last passing read-leveling setting in the largest
|
|
contiguous sequence of passing settings.
|
|
When clear, or if the setting selected by LMC*RLEVEL_CTL[OFFSET]
|
|
did not pass, LMC selects the middle setting in the
|
|
largest contiguous sequence of passing settings,
|
|
rounding earlier when necessary. */
|
|
uint64_t offset : 4; /**< The offset used when LMC*RLEVEL_CTL[OFFSET] is set */
|
|
uint64_t byte : 4; /**< 0 <= BYTE <= 8
|
|
Byte index for which bitmask results are saved
|
|
in LMC*_RLEVEL_DBG */
|
|
#else
|
|
uint64_t byte : 4;
|
|
uint64_t offset : 4;
|
|
uint64_t offset_en : 1;
|
|
uint64_t or_dis : 1;
|
|
uint64_t bitmask : 8;
|
|
uint64_t delay_unload_0 : 1;
|
|
uint64_t delay_unload_1 : 1;
|
|
uint64_t delay_unload_2 : 1;
|
|
uint64_t delay_unload_3 : 1;
|
|
uint64_t reserved_22_63 : 42;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_rlevel_ctl_s cn61xx;
|
|
struct cvmx_lmcx_rlevel_ctl_s cn63xx;
|
|
struct cvmx_lmcx_rlevel_ctl_cn63xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_9_63 : 55;
|
|
uint64_t offset_en : 1; /**< When set, LMC attempts to select the read-leveling
|
|
setting that is LMC*RLEVEL_CTL[OFFSET] settings earlier than the
|
|
last passing read-leveling setting in the largest
|
|
contiguous sequence of passing settings.
|
|
When clear, or if the setting selected by LMC*RLEVEL_CTL[OFFSET]
|
|
did not pass, LMC selects the middle setting in the
|
|
largest contiguous sequence of passing settings,
|
|
rounding earlier when necessary. */
|
|
uint64_t offset : 4; /**< The offset used when LMC*RLEVEL_CTL[OFFSET] is set */
|
|
uint64_t byte : 4; /**< 0 <= BYTE <= 8
|
|
Byte index for which bitmask results are saved
|
|
in LMC*_RLEVEL_DBG */
|
|
#else
|
|
uint64_t byte : 4;
|
|
uint64_t offset : 4;
|
|
uint64_t offset_en : 1;
|
|
uint64_t reserved_9_63 : 55;
|
|
#endif
|
|
} cn63xxp1;
|
|
struct cvmx_lmcx_rlevel_ctl_s cn66xx;
|
|
struct cvmx_lmcx_rlevel_ctl_s cn68xx;
|
|
struct cvmx_lmcx_rlevel_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_rlevel_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_rlevel_ctl cvmx_lmcx_rlevel_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_rlevel_dbg
|
|
*
|
|
* Notes:
|
|
* A given read of LMC*_RLEVEL_DBG returns the read-leveling pass/fail results for all possible
|
|
* delay settings (i.e. the BITMASK) for only one byte in the last rank that the HW read-leveled.
|
|
* LMC*_RLEVEL_CTL[BYTE] selects the particular byte.
|
|
*
|
|
* To get these pass/fail results for another different rank, you must run the hardware read-leveling
|
|
* again. For example, it is possible to get the BITMASK results for every byte of every rank
|
|
* if you run read-leveling separately for each rank, probing LMC*_RLEVEL_DBG between each
|
|
* read-leveling.
|
|
*/
|
|
union cvmx_lmcx_rlevel_dbg {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_rlevel_dbg_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t bitmask : 64; /**< Bitmask generated during deskew settings sweep
|
|
BITMASK[n]=0 means deskew setting n failed
|
|
BITMASK[n]=1 means deskew setting n passed
|
|
for 0 <= n <= 63 */
|
|
#else
|
|
uint64_t bitmask : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_rlevel_dbg_s cn61xx;
|
|
struct cvmx_lmcx_rlevel_dbg_s cn63xx;
|
|
struct cvmx_lmcx_rlevel_dbg_s cn63xxp1;
|
|
struct cvmx_lmcx_rlevel_dbg_s cn66xx;
|
|
struct cvmx_lmcx_rlevel_dbg_s cn68xx;
|
|
struct cvmx_lmcx_rlevel_dbg_s cn68xxp1;
|
|
struct cvmx_lmcx_rlevel_dbg_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_rlevel_dbg cvmx_lmcx_rlevel_dbg_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_rlevel_rank#
|
|
*
|
|
* Notes:
|
|
* This is four CSRs per LMC, one per each rank.
|
|
*
|
|
* Deskew setting is measured in units of 1/4 CK, so the above BYTE* values can range over 16 CKs.
|
|
*
|
|
* Each CSR is written by HW during a read-leveling sequence for the rank. (HW sets STATUS==3 after HW read-leveling completes for the rank.)
|
|
* If HW is unable to find a match per LMC*_RLEVEL_CTL[OFFSET_ENA] and LMC*_RLEVEL_CTL[OFFSET], then HW will set LMC*_RLEVEL_RANKi[BYTE*<5:0>]
|
|
* to 0.
|
|
*
|
|
* Each CSR may also be written by SW, but not while a read-leveling sequence is in progress. (HW sets STATUS==1 after a CSR write.)
|
|
*
|
|
* SW initiates a HW read-leveling sequence by programming LMC*_RLEVEL_CTL and writing INIT_START=1 with SEQUENCE=1.
|
|
* See LMC*_RLEVEL_CTL.
|
|
*
|
|
* LMC*_RLEVEL_RANKi values for ranks i without attached DRAM should be set such that
|
|
* they do not increase the range of possible BYTE values for any byte
|
|
* lane. The easiest way to do this is to set
|
|
* LMC*_RLEVEL_RANKi = LMC*_RLEVEL_RANKj,
|
|
* where j is some rank with attached DRAM whose LMC*_RLEVEL_RANKj is already fully initialized.
|
|
*/
|
|
union cvmx_lmcx_rlevel_rankx {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_rlevel_rankx_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_56_63 : 8;
|
|
uint64_t status : 2; /**< Indicates status of the read-levelling and where
|
|
the BYTE* programmings in <35:0> came from:
|
|
0 = BYTE* values are their reset value
|
|
1 = BYTE* values were set via a CSR write to this register
|
|
2 = read-leveling sequence currently in progress (BYTE* values are unpredictable)
|
|
3 = BYTE* values came from a complete read-leveling sequence */
|
|
uint64_t byte8 : 6; /**< Deskew setting
|
|
When ECC DRAM is not present (i.e. when DRAM is not
|
|
attached to chip signals DDR_CBS_0_* and DDR_CB[7:0]),
|
|
SW should write BYTE8 to a value that does
|
|
not increase the range of possible BYTE* values. The
|
|
easiest way to do this is to set
|
|
LMC*_RLEVEL_RANK*[BYTE8] = LMC*_RLEVEL_RANK*[BYTE0]
|
|
when there is no ECC DRAM, using the final BYTE0 value. */
|
|
uint64_t byte7 : 6; /**< Deskew setting */
|
|
uint64_t byte6 : 6; /**< Deskew setting */
|
|
uint64_t byte5 : 6; /**< Deskew setting */
|
|
uint64_t byte4 : 6; /**< Deskew setting */
|
|
uint64_t byte3 : 6; /**< Deskew setting */
|
|
uint64_t byte2 : 6; /**< Deskew setting */
|
|
uint64_t byte1 : 6; /**< Deskew setting */
|
|
uint64_t byte0 : 6; /**< Deskew setting */
|
|
#else
|
|
uint64_t byte0 : 6;
|
|
uint64_t byte1 : 6;
|
|
uint64_t byte2 : 6;
|
|
uint64_t byte3 : 6;
|
|
uint64_t byte4 : 6;
|
|
uint64_t byte5 : 6;
|
|
uint64_t byte6 : 6;
|
|
uint64_t byte7 : 6;
|
|
uint64_t byte8 : 6;
|
|
uint64_t status : 2;
|
|
uint64_t reserved_56_63 : 8;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_rlevel_rankx_s cn61xx;
|
|
struct cvmx_lmcx_rlevel_rankx_s cn63xx;
|
|
struct cvmx_lmcx_rlevel_rankx_s cn63xxp1;
|
|
struct cvmx_lmcx_rlevel_rankx_s cn66xx;
|
|
struct cvmx_lmcx_rlevel_rankx_s cn68xx;
|
|
struct cvmx_lmcx_rlevel_rankx_s cn68xxp1;
|
|
struct cvmx_lmcx_rlevel_rankx_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_rlevel_rankx cvmx_lmcx_rlevel_rankx_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_rodt_comp_ctl
|
|
*
|
|
* LMC_RODT_COMP_CTL = LMC Compensation control
|
|
*
|
|
*/
|
|
union cvmx_lmcx_rodt_comp_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_17_63 : 47;
|
|
uint64_t enable : 1; /**< 0=not enabled, 1=enable */
|
|
uint64_t reserved_12_15 : 4;
|
|
uint64_t nctl : 4; /**< Compensation control bits */
|
|
uint64_t reserved_5_7 : 3;
|
|
uint64_t pctl : 5; /**< Compensation control bits */
|
|
#else
|
|
uint64_t pctl : 5;
|
|
uint64_t reserved_5_7 : 3;
|
|
uint64_t nctl : 4;
|
|
uint64_t reserved_12_15 : 4;
|
|
uint64_t enable : 1;
|
|
uint64_t reserved_17_63 : 47;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn50xx;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn52xx;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn56xx;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn56xxp1;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn58xx;
|
|
struct cvmx_lmcx_rodt_comp_ctl_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_rodt_comp_ctl cvmx_lmcx_rodt_comp_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_rodt_ctl
|
|
*
|
|
* LMC_RODT_CTL = Obsolete LMC Read OnDieTermination control
|
|
* See the description in LMC_WODT_CTL1. On Reads, Octeon only supports turning on ODT's in
|
|
* the lower 2 DIMM's with the masks as below.
|
|
*
|
|
* Notes:
|
|
* When a given RANK in position N is selected, the RODT _HI and _LO masks for that position are used.
|
|
* Mask[3:0] is used for RODT control of the RANKs in positions 3, 2, 1, and 0, respectively.
|
|
* In 64b mode, DIMMs are assumed to be ordered in the following order:
|
|
* position 3: [unused , DIMM1_RANK1_LO]
|
|
* position 2: [unused , DIMM1_RANK0_LO]
|
|
* position 1: [unused , DIMM0_RANK1_LO]
|
|
* position 0: [unused , DIMM0_RANK0_LO]
|
|
* In 128b mode, DIMMs are assumed to be ordered in the following order:
|
|
* position 3: [DIMM3_RANK1_HI, DIMM1_RANK1_LO]
|
|
* position 2: [DIMM3_RANK0_HI, DIMM1_RANK0_LO]
|
|
* position 1: [DIMM2_RANK1_HI, DIMM0_RANK1_LO]
|
|
* position 0: [DIMM2_RANK0_HI, DIMM0_RANK0_LO]
|
|
*/
|
|
union cvmx_lmcx_rodt_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_rodt_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t rodt_hi3 : 4; /**< Read ODT mask for position 3, data[127:64] */
|
|
uint64_t rodt_hi2 : 4; /**< Read ODT mask for position 2, data[127:64] */
|
|
uint64_t rodt_hi1 : 4; /**< Read ODT mask for position 1, data[127:64] */
|
|
uint64_t rodt_hi0 : 4; /**< Read ODT mask for position 0, data[127:64] */
|
|
uint64_t rodt_lo3 : 4; /**< Read ODT mask for position 3, data[ 63: 0] */
|
|
uint64_t rodt_lo2 : 4; /**< Read ODT mask for position 2, data[ 63: 0] */
|
|
uint64_t rodt_lo1 : 4; /**< Read ODT mask for position 1, data[ 63: 0] */
|
|
uint64_t rodt_lo0 : 4; /**< Read ODT mask for position 0, data[ 63: 0] */
|
|
#else
|
|
uint64_t rodt_lo0 : 4;
|
|
uint64_t rodt_lo1 : 4;
|
|
uint64_t rodt_lo2 : 4;
|
|
uint64_t rodt_lo3 : 4;
|
|
uint64_t rodt_hi0 : 4;
|
|
uint64_t rodt_hi1 : 4;
|
|
uint64_t rodt_hi2 : 4;
|
|
uint64_t rodt_hi3 : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_rodt_ctl_s cn30xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn31xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn38xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn38xxp2;
|
|
struct cvmx_lmcx_rodt_ctl_s cn50xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn52xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn52xxp1;
|
|
struct cvmx_lmcx_rodt_ctl_s cn56xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn56xxp1;
|
|
struct cvmx_lmcx_rodt_ctl_s cn58xx;
|
|
struct cvmx_lmcx_rodt_ctl_s cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_rodt_ctl cvmx_lmcx_rodt_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_rodt_mask
|
|
*
|
|
* LMC_RODT_MASK = LMC Read OnDieTermination mask
|
|
* System designers may desire to terminate DQ/DQS lines for higher frequency DDR operations
|
|
* especially on a multi-rank system. DDR3 DQ/DQS I/O's have built in
|
|
* Termination resistor that can be turned on or off by the controller, after meeting tAOND and tAOF
|
|
* timing requirements. Each Rank has its own ODT pin that fans out to all the memory parts
|
|
* in that DIMM. System designers may prefer different combinations of ODT ON's for reads
|
|
* into different ranks. Octeon supports full programmability by way of the mask register below.
|
|
* Each Rank position has its own 8-bit programmable field.
|
|
* When the controller does a read to that rank, it sets the 4 ODT pins to the MASK pins below.
|
|
* For eg., When doing a read from Rank0, a system designer may desire to terminate the lines
|
|
* with the resistor on DIMM0/Rank1. The mask RODT_D0_R0 would then be [00000010].
|
|
* Octeon drives the appropriate mask values on the ODT pins by default. If this feature is not
|
|
* required, write 0 in this register. Note that, as per the DDR3 specifications, the ODT pin
|
|
* for the rank that is being read should always be 0.
|
|
*
|
|
* Notes:
|
|
* When a given RANK is selected, the RODT mask for that RANK is used. The resulting RODT mask is
|
|
* driven to the DIMMs in the following manner:
|
|
* RANK_ENA=1 RANK_ENA=0
|
|
* Mask[3] -> DIMM1_ODT_1 MBZ
|
|
* Mask[2] -> DIMM1_ODT_0 DIMM1_ODT_0
|
|
* Mask[1] -> DIMM0_ODT_1 MBZ
|
|
* Mask[0] -> DIMM0_ODT_0 DIMM0_ODT_0
|
|
*
|
|
* LMC always reads entire cache blocks and always reads them via two consecutive
|
|
* read CAS operations to the same rank+bank+row spaced exactly 4 CK's apart.
|
|
* When a RODT mask bit is set, LMC asserts the OCTEON ODT output
|
|
* pin(s) starting (CL - CWL) CK's after the first read CAS operation. Then, OCTEON
|
|
* normally continues to assert the ODT output pin(s) for 9+LMC*_CONTROL[RODT_BPRCH] more CK's
|
|
* - for a total of 10+LMC*_CONTROL[RODT_BPRCH] CK's for the entire cache block read -
|
|
* through the second read CAS operation of the cache block,
|
|
* satisfying the 6 CK DDR3 ODTH8 requirements.
|
|
* But it is possible for OCTEON to issue two cache block reads separated by as few as
|
|
* RtR = 8 or 9 (10 if LMC*_CONTROL[RODT_BPRCH]=1) CK's. In that case, OCTEON asserts the ODT output pin(s)
|
|
* for the RODT mask of the first cache block read for RtR CK's, then asserts
|
|
* the ODT output pin(s) for the RODT mask of the second cache block read for 10+LMC*_CONTROL[RODT_BPRCH] CK's
|
|
* (or less if a third cache block read follows within 8 or 9 (or 10) CK's of this second cache block read).
|
|
* Note that it may be necessary to force LMC to space back-to-back cache block reads
|
|
* to different ranks apart by at least 10+LMC*_CONTROL[RODT_BPRCH] CK's to prevent DDR3 ODTH8 violations.
|
|
*/
|
|
union cvmx_lmcx_rodt_mask {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_rodt_mask_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t rodt_d3_r1 : 8; /**< Read ODT mask DIMM3, RANK1/DIMM3 in SingleRanked
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d3_r0 : 8; /**< Read ODT mask DIMM3, RANK0
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d2_r1 : 8; /**< Read ODT mask DIMM2, RANK1/DIMM2 in SingleRanked
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d2_r0 : 8; /**< Read ODT mask DIMM2, RANK0
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d1_r1 : 8; /**< Read ODT mask DIMM1, RANK1/DIMM1 in SingleRanked
|
|
if (RANK_ENA) then
|
|
RODT_D1_R1[3] must be 0
|
|
else
|
|
RODT_D1_R1[3:0] is not used and MBZ
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d1_r0 : 8; /**< Read ODT mask DIMM1, RANK0
|
|
if (RANK_ENA) then
|
|
RODT_D1_RO[2] must be 0
|
|
else
|
|
RODT_D1_RO[3:2,1] must be 0
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d0_r1 : 8; /**< Read ODT mask DIMM0, RANK1/DIMM0 in SingleRanked
|
|
if (RANK_ENA) then
|
|
RODT_D0_R1[1] must be 0
|
|
else
|
|
RODT_D0_R1[3:0] is not used and MBZ
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t rodt_d0_r0 : 8; /**< Read ODT mask DIMM0, RANK0
|
|
if (RANK_ENA) then
|
|
RODT_D0_RO[0] must be 0
|
|
else
|
|
RODT_D0_RO[1:0,3] must be 0
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
#else
|
|
uint64_t rodt_d0_r0 : 8;
|
|
uint64_t rodt_d0_r1 : 8;
|
|
uint64_t rodt_d1_r0 : 8;
|
|
uint64_t rodt_d1_r1 : 8;
|
|
uint64_t rodt_d2_r0 : 8;
|
|
uint64_t rodt_d2_r1 : 8;
|
|
uint64_t rodt_d3_r0 : 8;
|
|
uint64_t rodt_d3_r1 : 8;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_rodt_mask_s cn61xx;
|
|
struct cvmx_lmcx_rodt_mask_s cn63xx;
|
|
struct cvmx_lmcx_rodt_mask_s cn63xxp1;
|
|
struct cvmx_lmcx_rodt_mask_s cn66xx;
|
|
struct cvmx_lmcx_rodt_mask_s cn68xx;
|
|
struct cvmx_lmcx_rodt_mask_s cn68xxp1;
|
|
struct cvmx_lmcx_rodt_mask_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_rodt_mask cvmx_lmcx_rodt_mask_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_scramble_cfg0
|
|
*
|
|
* LMC_SCRAMBLE_CFG0 = LMC Scramble Config0
|
|
*
|
|
*/
|
|
union cvmx_lmcx_scramble_cfg0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_scramble_cfg0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t key : 64; /**< Scramble Key for Data */
|
|
#else
|
|
uint64_t key : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_scramble_cfg0_s cn61xx;
|
|
struct cvmx_lmcx_scramble_cfg0_s cn66xx;
|
|
struct cvmx_lmcx_scramble_cfg0_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_scramble_cfg0 cvmx_lmcx_scramble_cfg0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_scramble_cfg1
|
|
*
|
|
* LMC_SCRAMBLE_CFG1 = LMC Scramble Config1
|
|
*
|
|
*
|
|
* Notes:
|
|
* Address scrambling usually maps addresses into the same rank. Exceptions are when LMC_NXM[CS_MASK] requires
|
|
* aliasing that uses the lowest, legal chip select(s).
|
|
*/
|
|
union cvmx_lmcx_scramble_cfg1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_scramble_cfg1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t key : 64; /**< Scramble Key for Addresses */
|
|
#else
|
|
uint64_t key : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_scramble_cfg1_s cn61xx;
|
|
struct cvmx_lmcx_scramble_cfg1_s cn66xx;
|
|
struct cvmx_lmcx_scramble_cfg1_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_scramble_cfg1 cvmx_lmcx_scramble_cfg1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_scrambled_fadr
|
|
*
|
|
* LMC_SCRAMBLED_FADR = LMC Scrambled Failing Address Register (SEC/DED/NXM)
|
|
*
|
|
* This register only captures the first transaction with ecc/nxm errors. A DED/NXM error can
|
|
* over-write this register with its failing addresses if the first error was a SEC. If you write
|
|
* LMC*_CONFIG->SEC_ERR/DED_ERR/NXM_ERR then it will clear the error bits and capture the
|
|
* next failing address.
|
|
*
|
|
* If FDIMM is 2 that means the error is in the higher bits DIMM.
|
|
*
|
|
* Notes:
|
|
* LMC*_FADR captures the failing pre-scrambled address location (split into dimm, bunk, bank, etc). If
|
|
* scrambling is off, then LMC*_FADR will also capture the failing physical location in the DRAM parts.
|
|
*
|
|
* LMC*_SCRAMBLED_FADR captures the actual failing address location in the physical DRAM parts, i.e.,
|
|
* a. if scrambling is on, LMC*_SCRAMBLE_FADR contains the failing physical location in the DRAM parts (split
|
|
* into dimm, bunk, bank, etc)
|
|
* b. if scrambling is off, the pre-scramble and post-scramble addresses are the same, and so the contents of
|
|
* LMC*_SCRAMBLED_FADR match the contents of LMC*_FADR
|
|
*/
|
|
union cvmx_lmcx_scrambled_fadr {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_scrambled_fadr_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_36_63 : 28;
|
|
uint64_t fdimm : 2; /**< Failing DIMM# */
|
|
uint64_t fbunk : 1; /**< Failing Rank */
|
|
uint64_t fbank : 3; /**< Failing Bank[2:0] */
|
|
uint64_t frow : 16; /**< Failing Row Address[15:0] */
|
|
uint64_t fcol : 14; /**< Failing Column Address[13:0]
|
|
Technically, represents the address of the 128b data
|
|
that had an ecc error, i.e., fcol[0] is always 0. Can
|
|
be used in conjuction with LMC*_CONFIG[DED_ERR] to
|
|
isolate the 64b chunk of data in error */
|
|
#else
|
|
uint64_t fcol : 14;
|
|
uint64_t frow : 16;
|
|
uint64_t fbank : 3;
|
|
uint64_t fbunk : 1;
|
|
uint64_t fdimm : 2;
|
|
uint64_t reserved_36_63 : 28;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_scrambled_fadr_s cn61xx;
|
|
struct cvmx_lmcx_scrambled_fadr_s cn66xx;
|
|
struct cvmx_lmcx_scrambled_fadr_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_scrambled_fadr cvmx_lmcx_scrambled_fadr_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_slot_ctl0
|
|
*
|
|
* LMC_SLOT_CTL0 = LMC Slot Control0
|
|
* This register is an assortment of various control fields needed by the memory controller
|
|
*
|
|
* Notes:
|
|
* If SW has not previously written to this register (since the last DRESET),
|
|
* HW updates the fields in this register to the minimum allowed value
|
|
* when any of LMC*_RLEVEL_RANKn, LMC*_WLEVEL_RANKn, LMC*_CONTROL and
|
|
* LMC*_MODEREG_PARAMS0 CSR's change. Ideally, only read this register
|
|
* after LMC has been initialized and LMC*_RLEVEL_RANKn, LMC*_WLEVEL_RANKn
|
|
* have valid data.
|
|
*
|
|
* The interpretation of the fields in this CSR depends on LMC*_CONFIG[DDR2T]:
|
|
* - If LMC*_CONFIG[DDR2T]=1, (FieldValue + 4) is the minimum CK cycles
|
|
* between when the DRAM part registers CAS commands of the 1st and 2nd types
|
|
* from different cache blocks.
|
|
* - If LMC*_CONFIG[DDR2T]=0, (FieldValue + 3) is the minimum CK cycles
|
|
* between when the DRAM part registers CAS commands of the 1st and 2nd types
|
|
* from different cache blocks. FieldValue = 0 is always illegal in this
|
|
* case.
|
|
*
|
|
* The hardware-calculated minimums are:
|
|
*
|
|
* min R2R_INIT = 1 - LMC*_CONFIG[DDR2T]
|
|
* min R2W_INIT = 5 - LMC*_CONFIG[DDR2T] + (RL + MaxRdSkew) - (WL + MinWrSkew) + LMC*_CONTROL[BPRCH]
|
|
* min W2R_INIT = 2 - LMC*_CONFIG[DDR2T] + LMC*_TIMING_PARAMS1[TWTR] + WL
|
|
* min W2W_INIT = 1 - LMC*_CONFIG[DDR2T]
|
|
*
|
|
* where
|
|
*
|
|
* RL = CL + AL (LMC*_MODEREG_PARAMS0[CL] selects CL, LMC*_MODEREG_PARAMS0[AL] selects AL)
|
|
* WL = CWL + AL (LMC*_MODEREG_PARAMS0[CWL] selects CWL)
|
|
* MaxRdSkew = max(LMC*_RLEVEL_RANKi[BYTEj]/4) + 1 (max is across all ranks i (0..3) and bytes j (0..8))
|
|
* MinWrSkew = min(LMC*_WLEVEL_RANKi[BYTEj]/8) - LMC*_CONFIG[EARLY_DQX] (min is across all ranks i (0..3) and bytes j (0..8))
|
|
*
|
|
* R2W_INIT has 1 CK cycle built in for OCTEON-internal ODT settling/channel turnaround time.
|
|
*/
|
|
union cvmx_lmcx_slot_ctl0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_slot_ctl0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_24_63 : 40;
|
|
uint64_t w2w_init : 6; /**< Write-to-write spacing control
|
|
for back to back write followed by write cache block
|
|
accesses to the same rank and DIMM */
|
|
uint64_t w2r_init : 6; /**< Write-to-read spacing control
|
|
for back to back write followed by read cache block
|
|
accesses to the same rank and DIMM */
|
|
uint64_t r2w_init : 6; /**< Read-to-write spacing control
|
|
for back to back read followed by write cache block
|
|
accesses to the same rank and DIMM */
|
|
uint64_t r2r_init : 6; /**< Read-to-read spacing control
|
|
for back to back read followed by read cache block
|
|
accesses to the same rank and DIMM */
|
|
#else
|
|
uint64_t r2r_init : 6;
|
|
uint64_t r2w_init : 6;
|
|
uint64_t w2r_init : 6;
|
|
uint64_t w2w_init : 6;
|
|
uint64_t reserved_24_63 : 40;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_slot_ctl0_s cn61xx;
|
|
struct cvmx_lmcx_slot_ctl0_s cn63xx;
|
|
struct cvmx_lmcx_slot_ctl0_s cn63xxp1;
|
|
struct cvmx_lmcx_slot_ctl0_s cn66xx;
|
|
struct cvmx_lmcx_slot_ctl0_s cn68xx;
|
|
struct cvmx_lmcx_slot_ctl0_s cn68xxp1;
|
|
struct cvmx_lmcx_slot_ctl0_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_slot_ctl0 cvmx_lmcx_slot_ctl0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_slot_ctl1
|
|
*
|
|
* LMC_SLOT_CTL1 = LMC Slot Control1
|
|
* This register is an assortment of various control fields needed by the memory controller
|
|
*
|
|
* Notes:
|
|
* If SW has not previously written to this register (since the last DRESET),
|
|
* HW updates the fields in this register to the minimum allowed value
|
|
* when any of LMC*_RLEVEL_RANKn, LMC*_WLEVEL_RANKn, LMC*_CONTROL and
|
|
* LMC*_MODEREG_PARAMS0 CSR's change. Ideally, only read this register
|
|
* after LMC has been initialized and LMC*_RLEVEL_RANKn, LMC*_WLEVEL_RANKn
|
|
* have valid data.
|
|
*
|
|
* The interpretation of the fields in this CSR depends on LMC*_CONFIG[DDR2T]:
|
|
* - If LMC*_CONFIG[DDR2T]=1, (FieldValue + 4) is the minimum CK cycles
|
|
* between when the DRAM part registers CAS commands of the 1st and 2nd types
|
|
* from different cache blocks.
|
|
* - If LMC*_CONFIG[DDR2T]=0, (FieldValue + 3) is the minimum CK cycles
|
|
* between when the DRAM part registers CAS commands of the 1st and 2nd types
|
|
* from different cache blocks. FieldValue = 0 is always illegal in this
|
|
* case.
|
|
*
|
|
* The hardware-calculated minimums are:
|
|
*
|
|
* min R2R_XRANK_INIT = 2 - LMC*_CONFIG[DDR2T] + MaxRdSkew - MinRdSkew + LMC*_CONTROL[RODT_BPRCH]
|
|
* min R2W_XRANK_INIT = 5 - LMC*_CONFIG[DDR2T] + (RL + MaxRdSkew) - (WL + MinWrSkew) + LMC*_CONTROL[BPRCH]
|
|
* min W2R_XRANK_INIT = 3 - LMC*_CONFIG[DDR2T] + MaxWrSkew + LMC*_CONTROL[FPRCH2]
|
|
* min W2W_XRANK_INIT = 4 - LMC*_CONFIG[DDR2T] + MaxWrSkew - MinWrSkew
|
|
*
|
|
* where
|
|
*
|
|
* RL = CL + AL (LMC*_MODEREG_PARAMS0[CL] selects CL, LMC*_MODEREG_PARAMS0[AL] selects AL)
|
|
* WL = CWL + AL (LMC*_MODEREG_PARAMS0[CWL] selects CWL)
|
|
* MinRdSkew = min(LMC*_RLEVEL_RANKi[BYTEj]/4) (min is across all ranks i (0..3) and bytes j (0..8))
|
|
* MaxRdSkew = max(LMC*_RLEVEL_RANKi[BYTEj]/4) + 1 (max is across all ranks i (0..3) and bytes j (0..8))
|
|
* MinWrSkew = min(LMC*_WLEVEL_RANKi[BYTEj]/8) - LMC*_CONFIG[EARLY_DQX] (min is across all ranks i (0..3) and bytes j (0..8))
|
|
* MaxWrSkew = max(LMC*_WLEVEL_RANKi[BYTEj]/8) - LMC*_CONFIG[EARLY_DQX] + 1 (max is across all ranks i (0..3) and bytes j (0..8))
|
|
*
|
|
* R2W_XRANK_INIT has 1 extra CK cycle built in for OCTEON-internal ODT settling/channel turnaround time.
|
|
*
|
|
* W2R_XRANK_INIT has 1 extra CK cycle built in for channel turnaround time.
|
|
*/
|
|
union cvmx_lmcx_slot_ctl1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_slot_ctl1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_24_63 : 40;
|
|
uint64_t w2w_xrank_init : 6; /**< Write-to-write spacing control
|
|
for back to back write followed by write cache block
|
|
accesses across ranks of the same DIMM */
|
|
uint64_t w2r_xrank_init : 6; /**< Write-to-read spacing control
|
|
for back to back write followed by read cache block
|
|
accesses across ranks of the same DIMM */
|
|
uint64_t r2w_xrank_init : 6; /**< Read-to-write spacing control
|
|
for back to back read followed by write cache block
|
|
accesses across ranks of the same DIMM */
|
|
uint64_t r2r_xrank_init : 6; /**< Read-to-read spacing control
|
|
for back to back read followed by read cache block
|
|
accesses across ranks of the same DIMM */
|
|
#else
|
|
uint64_t r2r_xrank_init : 6;
|
|
uint64_t r2w_xrank_init : 6;
|
|
uint64_t w2r_xrank_init : 6;
|
|
uint64_t w2w_xrank_init : 6;
|
|
uint64_t reserved_24_63 : 40;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_slot_ctl1_s cn61xx;
|
|
struct cvmx_lmcx_slot_ctl1_s cn63xx;
|
|
struct cvmx_lmcx_slot_ctl1_s cn63xxp1;
|
|
struct cvmx_lmcx_slot_ctl1_s cn66xx;
|
|
struct cvmx_lmcx_slot_ctl1_s cn68xx;
|
|
struct cvmx_lmcx_slot_ctl1_s cn68xxp1;
|
|
struct cvmx_lmcx_slot_ctl1_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_slot_ctl1 cvmx_lmcx_slot_ctl1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_slot_ctl2
|
|
*
|
|
* LMC_SLOT_CTL2 = LMC Slot Control2
|
|
* This register is an assortment of various control fields needed by the memory controller
|
|
*
|
|
* Notes:
|
|
* If SW has not previously written to this register (since the last DRESET),
|
|
* HW updates the fields in this register to the minimum allowed value
|
|
* when any of LMC*_RLEVEL_RANKn, LMC*_WLEVEL_RANKn, LMC*_CONTROL and
|
|
* LMC*_MODEREG_PARAMS0 CSR's change. Ideally, only read this register
|
|
* after LMC has been initialized and LMC*_RLEVEL_RANKn, LMC*_WLEVEL_RANKn
|
|
* have valid data.
|
|
*
|
|
* The interpretation of the fields in this CSR depends on LMC*_CONFIG[DDR2T]:
|
|
* - If LMC*_CONFIG[DDR2T]=1, (FieldValue + 4) is the minimum CK cycles
|
|
* between when the DRAM part registers CAS commands of the 1st and 2nd types
|
|
* from different cache blocks.
|
|
* - If LMC*_CONFIG[DDR2T]=0, (FieldValue + 3) is the minimum CK cycles
|
|
* between when the DRAM part registers CAS commands of the 1st and 2nd types
|
|
* from different cache blocks. FieldValue = 0 is always illegal in this
|
|
* case.
|
|
*
|
|
* The hardware-calculated minimums are:
|
|
*
|
|
* min R2R_XDIMM_INIT = 3 - LMC*_CONFIG[DDR2T] + MaxRdSkew - MinRdSkew + LMC*_CONTROL[RODT_BPRCH]
|
|
* min R2W_XDIMM_INIT = 6 - LMC*_CONFIG[DDR2T] + (RL + MaxRdSkew) - (WL + MinWrSkew) + LMC*_CONTROL[BPRCH]
|
|
* min W2R_XDIMM_INIT = 3 - LMC*_CONFIG[DDR2T] + MaxWrSkew + LMC*_CONTROL[FPRCH2]
|
|
* min W2W_XDIMM_INIT = 5 - LMC*_CONFIG[DDR2T] + MaxWrSkew - MinWrSkew
|
|
*
|
|
* where
|
|
*
|
|
* RL = CL + AL (LMC*_MODEREG_PARAMS0[CL] selects CL, LMC*_MODEREG_PARAMS0[AL] selects AL)
|
|
* WL = CWL + AL (LMC*_MODEREG_PARAMS0[CWL] selects CWL)
|
|
* MinRdSkew = min(LMC*_RLEVEL_RANKi[BYTEj]/4) (min is across all ranks i (0..3) and bytes j (0..8))
|
|
* MaxRdSkew = max(LMC*_RLEVEL_RANKi[BYTEj]/4) + 1 (max is across all ranks i (0..3) and bytes j (0..8))
|
|
* MinWrSkew = min(LMC*_WLEVEL_RANKi[BYTEj]/8) - LMC*_CONFIG[EARLY_DQX] (min is across all ranks i (0..3) and bytes j (0..8))
|
|
* MaxWrSkew = max(LMC*_WLEVEL_RANKi[BYTEj]/8) - LMC*_CONFIG[EARLY_DQX] + 1 (max is across all ranks i (0..3) and bytes j (0..8))
|
|
*
|
|
* R2W_XDIMM_INIT has 2 extra CK cycles built in for OCTEON-internal ODT settling/channel turnaround time.
|
|
*
|
|
* R2R_XDIMM_INIT, W2R_XRANK_INIT, W2W_XDIMM_INIT have 1 extra CK cycle built in for channel turnaround time.
|
|
*/
|
|
union cvmx_lmcx_slot_ctl2 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_slot_ctl2_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_24_63 : 40;
|
|
uint64_t w2w_xdimm_init : 6; /**< Write-to-write spacing control
|
|
for back to back write followed by write cache block
|
|
accesses across DIMMs */
|
|
uint64_t w2r_xdimm_init : 6; /**< Write-to-read spacing control
|
|
for back to back write followed by read cache block
|
|
accesses across DIMMs */
|
|
uint64_t r2w_xdimm_init : 6; /**< Read-to-write spacing control
|
|
for back to back read followed by write cache block
|
|
accesses across DIMMs */
|
|
uint64_t r2r_xdimm_init : 6; /**< Read-to-read spacing control
|
|
for back to back read followed by read cache block
|
|
accesses across DIMMs */
|
|
#else
|
|
uint64_t r2r_xdimm_init : 6;
|
|
uint64_t r2w_xdimm_init : 6;
|
|
uint64_t w2r_xdimm_init : 6;
|
|
uint64_t w2w_xdimm_init : 6;
|
|
uint64_t reserved_24_63 : 40;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_slot_ctl2_s cn61xx;
|
|
struct cvmx_lmcx_slot_ctl2_s cn63xx;
|
|
struct cvmx_lmcx_slot_ctl2_s cn63xxp1;
|
|
struct cvmx_lmcx_slot_ctl2_s cn66xx;
|
|
struct cvmx_lmcx_slot_ctl2_s cn68xx;
|
|
struct cvmx_lmcx_slot_ctl2_s cn68xxp1;
|
|
struct cvmx_lmcx_slot_ctl2_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_slot_ctl2 cvmx_lmcx_slot_ctl2_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_timing_params0
|
|
*/
|
|
union cvmx_lmcx_timing_params0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_timing_params0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_47_63 : 17;
|
|
uint64_t trp_ext : 1; /**< Indicates tRP constraints.
|
|
Set [TRP_EXT[0:0], TRP[3:0]] (CSR field) = RNDUP[tRP(ns)/tCYC(ns)]
|
|
+ (RNDUP[tRTP(ns)/tCYC(ns)]-4)-1,
|
|
where tRP, tRTP are from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP tRP=10-15ns
|
|
TYP tRTP=max(4nCK, 7.5ns) */
|
|
uint64_t tcksre : 4; /**< Indicates tCKSRE constraints.
|
|
Set TCKSRE (CSR field) = RNDUP[tCKSRE(ns)/tCYC(ns)]-1,
|
|
where tCKSRE is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(5nCK, 10ns) */
|
|
uint64_t trp : 4; /**< Indicates tRP constraints.
|
|
Set [TRP_EXT[0:0], TRP[3:0]] (CSR field) = RNDUP[tRP(ns)/tCYC(ns)]
|
|
+ (RNDUP[tRTP(ns)/tCYC(ns)])-4)-1,
|
|
where tRP, tRTP are from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP tRP=10-15ns
|
|
TYP tRTP=max(4nCK, 7.5ns) */
|
|
uint64_t tzqinit : 4; /**< Indicates tZQINIT constraints.
|
|
Set TZQINIT (CSR field) = RNDUP[tZQINIT(ns)/(256*tCYC(ns))],
|
|
where tZQINIT is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=2 (equivalent to 512) */
|
|
uint64_t tdllk : 4; /**< Indicates tDLLK constraints.
|
|
Set TDLLK (CSR field) = RNDUP[tDLLK(ns)/(256*tCYC(ns))],
|
|
where tDLLK is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=2 (equivalent to 512)
|
|
This parameter is used in self-refresh exit
|
|
and assumed to be greater than tRFC */
|
|
uint64_t tmod : 4; /**< Indicates tMOD constraints.
|
|
Set TMOD (CSR field) = RNDUP[tMOD(ns)/tCYC(ns)]-1,
|
|
where tMOD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(12nCK, 15ns) */
|
|
uint64_t tmrd : 4; /**< Indicates tMRD constraints.
|
|
Set TMRD (CSR field) = RNDUP[tMRD(ns)/tCYC(ns)]-1,
|
|
where tMRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=4nCK */
|
|
uint64_t txpr : 4; /**< Indicates tXPR constraints.
|
|
Set TXPR (CSR field) = RNDUP[tXPR(ns)/(16*tCYC(ns))],
|
|
where tXPR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(5nCK, tRFC+10ns) */
|
|
uint64_t tcke : 4; /**< Indicates tCKE constraints.
|
|
Set TCKE (CSR field) = RNDUP[tCKE(ns)/tCYC(ns)]-1,
|
|
where tCKE is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(3nCK, 7.5/5.625/5.625/5ns) */
|
|
uint64_t tzqcs : 4; /**< Indicates tZQCS constraints.
|
|
Set TZQCS (CSR field) = RNDUP[tZQCS(ns)/(16*tCYC(ns))],
|
|
where tZQCS is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=4 (equivalent to 64) */
|
|
uint64_t tckeon : 10; /**< Reserved. Should be written to zero. */
|
|
#else
|
|
uint64_t tckeon : 10;
|
|
uint64_t tzqcs : 4;
|
|
uint64_t tcke : 4;
|
|
uint64_t txpr : 4;
|
|
uint64_t tmrd : 4;
|
|
uint64_t tmod : 4;
|
|
uint64_t tdllk : 4;
|
|
uint64_t tzqinit : 4;
|
|
uint64_t trp : 4;
|
|
uint64_t tcksre : 4;
|
|
uint64_t trp_ext : 1;
|
|
uint64_t reserved_47_63 : 17;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_timing_params0_cn61xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_47_63 : 17;
|
|
uint64_t trp_ext : 1; /**< Indicates tRP constraints.
|
|
Set [TRP_EXT[0:0], TRP[3:0]] (CSR field) = RNDUP[tRP(ns)/tCYC(ns)]
|
|
+ (RNDUP[tRTP(ns)/tCYC(ns)]-4)-1,
|
|
where tRP, tRTP are from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP tRP=10-15ns
|
|
TYP tRTP=max(4nCK, 7.5ns) */
|
|
uint64_t tcksre : 4; /**< Indicates tCKSRE constraints.
|
|
Set TCKSRE (CSR field) = RNDUP[tCKSRE(ns)/tCYC(ns)]-1,
|
|
where tCKSRE is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(5nCK, 10ns) */
|
|
uint64_t trp : 4; /**< Indicates tRP constraints.
|
|
Set [TRP_EXT[0:0], TRP[3:0]] (CSR field) = RNDUP[tRP(ns)/tCYC(ns)]
|
|
+ (RNDUP[tRTP(ns)/tCYC(ns)])-4)-1,
|
|
where tRP, tRTP are from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP tRP=10-15ns
|
|
TYP tRTP=max(4nCK, 7.5ns) */
|
|
uint64_t tzqinit : 4; /**< Indicates tZQINIT constraints.
|
|
Set TZQINIT (CSR field) = RNDUP[tZQINIT(ns)/(256*tCYC(ns))],
|
|
where tZQINIT is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=2 (equivalent to 512) */
|
|
uint64_t tdllk : 4; /**< Indicates tDLLK constraints.
|
|
Set TDLLK (CSR field) = RNDUP[tDLLK(ns)/(256*tCYC(ns))],
|
|
where tDLLK is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=2 (equivalent to 512)
|
|
This parameter is used in self-refresh exit
|
|
and assumed to be greater than tRFC */
|
|
uint64_t tmod : 4; /**< Indicates tMOD constraints.
|
|
Set TMOD (CSR field) = RNDUP[tMOD(ns)/tCYC(ns)]-1,
|
|
where tMOD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(12nCK, 15ns) */
|
|
uint64_t tmrd : 4; /**< Indicates tMRD constraints.
|
|
Set TMRD (CSR field) = RNDUP[tMRD(ns)/tCYC(ns)]-1,
|
|
where tMRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=4nCK */
|
|
uint64_t txpr : 4; /**< Indicates tXPR constraints.
|
|
Set TXPR (CSR field) = RNDUP[tXPR(ns)/(16*tCYC(ns))],
|
|
where tXPR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(5nCK, tRFC+10ns) */
|
|
uint64_t tcke : 4; /**< Indicates tCKE constraints.
|
|
Set TCKE (CSR field) = RNDUP[tCKE(ns)/tCYC(ns)]-1,
|
|
where tCKE is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(3nCK, 7.5/5.625/5.625/5ns) */
|
|
uint64_t tzqcs : 4; /**< Indicates tZQCS constraints.
|
|
Set TZQCS (CSR field) = RNDUP[tZQCS(ns)/(16*tCYC(ns))],
|
|
where tZQCS is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=4 (equivalent to 64) */
|
|
uint64_t reserved_0_9 : 10;
|
|
#else
|
|
uint64_t reserved_0_9 : 10;
|
|
uint64_t tzqcs : 4;
|
|
uint64_t tcke : 4;
|
|
uint64_t txpr : 4;
|
|
uint64_t tmrd : 4;
|
|
uint64_t tmod : 4;
|
|
uint64_t tdllk : 4;
|
|
uint64_t tzqinit : 4;
|
|
uint64_t trp : 4;
|
|
uint64_t tcksre : 4;
|
|
uint64_t trp_ext : 1;
|
|
uint64_t reserved_47_63 : 17;
|
|
#endif
|
|
} cn61xx;
|
|
struct cvmx_lmcx_timing_params0_cn61xx cn63xx;
|
|
struct cvmx_lmcx_timing_params0_cn63xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_46_63 : 18;
|
|
uint64_t tcksre : 4; /**< Indicates tCKSRE constraints.
|
|
Set TCKSRE (CSR field) = RNDUP[tCKSRE(ns)/tCYC(ns)]-1,
|
|
where tCKSRE is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(5nCK, 10ns) */
|
|
uint64_t trp : 4; /**< Indicates tRP constraints.
|
|
Set TRP (CSR field) = RNDUP[tRP(ns)/tCYC(ns)]
|
|
+ (RNDUP[tRTP(ns)/tCYC(ns)])-4)-1,
|
|
where tRP, tRTP are from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP tRP=10-15ns
|
|
TYP tRTP=max(4nCK, 7.5ns) */
|
|
uint64_t tzqinit : 4; /**< Indicates tZQINIT constraints.
|
|
Set TZQINIT (CSR field) = RNDUP[tZQINIT(ns)/(256*tCYC(ns))],
|
|
where tZQINIT is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=2 (equivalent to 512) */
|
|
uint64_t tdllk : 4; /**< Indicates tDLLK constraints.
|
|
Set TDLLK (CSR field) = RNDUP[tDLLK(ns)/(256*tCYC(ns))],
|
|
where tDLLK is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=2 (equivalent to 512)
|
|
This parameter is used in self-refresh exit
|
|
and assumed to be greater than tRFC */
|
|
uint64_t tmod : 4; /**< Indicates tMOD constraints.
|
|
Set TMOD (CSR field) = RNDUP[tMOD(ns)/tCYC(ns)]-1,
|
|
where tMOD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(12nCK, 15ns) */
|
|
uint64_t tmrd : 4; /**< Indicates tMRD constraints.
|
|
Set TMRD (CSR field) = RNDUP[tMRD(ns)/tCYC(ns)]-1,
|
|
where tMRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=4nCK */
|
|
uint64_t txpr : 4; /**< Indicates tXPR constraints.
|
|
Set TXPR (CSR field) = RNDUP[tXPR(ns)/(16*tCYC(ns))],
|
|
where tXPR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(5nCK, tRFC+10ns) */
|
|
uint64_t tcke : 4; /**< Indicates tCKE constraints.
|
|
Set TCKE (CSR field) = RNDUP[tCKE(ns)/tCYC(ns)]-1,
|
|
where tCKE is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(3nCK, 7.5/5.625/5.625/5ns) */
|
|
uint64_t tzqcs : 4; /**< Indicates tZQCS constraints.
|
|
Set TZQCS (CSR field) = RNDUP[tZQCS(ns)/(16*tCYC(ns))],
|
|
where tZQCS is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=4 (equivalent to 64) */
|
|
uint64_t tckeon : 10; /**< Reserved. Should be written to zero. */
|
|
#else
|
|
uint64_t tckeon : 10;
|
|
uint64_t tzqcs : 4;
|
|
uint64_t tcke : 4;
|
|
uint64_t txpr : 4;
|
|
uint64_t tmrd : 4;
|
|
uint64_t tmod : 4;
|
|
uint64_t tdllk : 4;
|
|
uint64_t tzqinit : 4;
|
|
uint64_t trp : 4;
|
|
uint64_t tcksre : 4;
|
|
uint64_t reserved_46_63 : 18;
|
|
#endif
|
|
} cn63xxp1;
|
|
struct cvmx_lmcx_timing_params0_cn61xx cn66xx;
|
|
struct cvmx_lmcx_timing_params0_cn61xx cn68xx;
|
|
struct cvmx_lmcx_timing_params0_cn61xx cn68xxp1;
|
|
struct cvmx_lmcx_timing_params0_cn61xx cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_timing_params0 cvmx_lmcx_timing_params0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_timing_params1
|
|
*/
|
|
union cvmx_lmcx_timing_params1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_timing_params1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_47_63 : 17;
|
|
uint64_t tras_ext : 1; /**< Indicates tRAS constraints.
|
|
Set [TRAS_EXT[0:0], TRAS[4:0]] (CSR field) = RNDUP[tRAS(ns)/tCYC(ns)]-1,
|
|
where tRAS is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=35ns-9*tREFI
|
|
- 000000: RESERVED
|
|
- 000001: 2 tCYC
|
|
- 000010: 3 tCYC
|
|
- ...
|
|
- 111111: 64 tCYC */
|
|
uint64_t txpdll : 5; /**< Indicates tXPDLL constraints.
|
|
Set TXPDLL (CSR field) = RNDUP[tXPDLL(ns)/tCYC(ns)]-1,
|
|
where tXPDLL is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(10nCK, 24ns) */
|
|
uint64_t tfaw : 5; /**< Indicates tFAW constraints.
|
|
Set TFAW (CSR field) = RNDUP[tFAW(ns)/(4*tCYC(ns))],
|
|
where tFAW is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=30-40ns */
|
|
uint64_t twldqsen : 4; /**< Indicates tWLDQSEN constraints.
|
|
Set TWLDQSEN (CSR field) = RNDUP[tWLDQSEN(ns)/(4*tCYC(ns))],
|
|
where tWLDQSEN is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(25nCK) */
|
|
uint64_t twlmrd : 4; /**< Indicates tWLMRD constraints.
|
|
Set TWLMRD (CSR field) = RNDUP[tWLMRD(ns)/(4*tCYC(ns))],
|
|
where tWLMRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(40nCK) */
|
|
uint64_t txp : 3; /**< Indicates tXP constraints.
|
|
Set TXP (CSR field) = RNDUP[tXP(ns)/tCYC(ns)]-1,
|
|
where tXP is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(3nCK, 7.5ns) */
|
|
uint64_t trrd : 3; /**< Indicates tRRD constraints.
|
|
Set TRRD (CSR field) = RNDUP[tRRD(ns)/tCYC(ns)]-2,
|
|
where tRRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(4nCK, 10ns)
|
|
- 000: RESERVED
|
|
- 001: 3 tCYC
|
|
- ...
|
|
- 110: 8 tCYC
|
|
- 111: 9 tCYC */
|
|
uint64_t trfc : 5; /**< Indicates tRFC constraints.
|
|
Set TRFC (CSR field) = RNDUP[tRFC(ns)/(8*tCYC(ns))],
|
|
where tRFC is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=90-350ns
|
|
- 00000: RESERVED
|
|
- 00001: 8 tCYC
|
|
- 00010: 16 tCYC
|
|
- 00011: 24 tCYC
|
|
- 00100: 32 tCYC
|
|
- ...
|
|
- 11110: 240 tCYC
|
|
- 11111: 248 tCYC */
|
|
uint64_t twtr : 4; /**< Indicates tWTR constraints.
|
|
Set TWTR (CSR field) = RNDUP[tWTR(ns)/tCYC(ns)]-1,
|
|
where tWTR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(4nCK, 7.5ns)
|
|
- 0000: RESERVED
|
|
- 0001: 2
|
|
- ...
|
|
- 0111: 8
|
|
- 1000-1111: RESERVED */
|
|
uint64_t trcd : 4; /**< Indicates tRCD constraints.
|
|
Set TRCD (CSR field) = RNDUP[tRCD(ns)/tCYC(ns)],
|
|
where tRCD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=10-15ns
|
|
- 0000: RESERVED
|
|
- 0001: 2 (2 is the smallest value allowed)
|
|
- 0002: 2
|
|
- ...
|
|
- 1110: 14
|
|
- 1111: RESERVED
|
|
In 2T mode, make this register TRCD-1, not going
|
|
below 2. */
|
|
uint64_t tras : 5; /**< Indicates tRAS constraints.
|
|
Set [TRAS_EXT[0:0], TRAS[4:0]] (CSR field) = RNDUP[tRAS(ns)/tCYC(ns)]-1,
|
|
where tRAS is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=35ns-9*tREFI
|
|
- 000000: RESERVED
|
|
- 000001: 2 tCYC
|
|
- 000010: 3 tCYC
|
|
- ...
|
|
- 111111: 64 tCYC */
|
|
uint64_t tmprr : 4; /**< Indicates tMPRR constraints.
|
|
Set TMPRR (CSR field) = RNDUP[tMPRR(ns)/tCYC(ns)]-1,
|
|
where tMPRR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=1nCK */
|
|
#else
|
|
uint64_t tmprr : 4;
|
|
uint64_t tras : 5;
|
|
uint64_t trcd : 4;
|
|
uint64_t twtr : 4;
|
|
uint64_t trfc : 5;
|
|
uint64_t trrd : 3;
|
|
uint64_t txp : 3;
|
|
uint64_t twlmrd : 4;
|
|
uint64_t twldqsen : 4;
|
|
uint64_t tfaw : 5;
|
|
uint64_t txpdll : 5;
|
|
uint64_t tras_ext : 1;
|
|
uint64_t reserved_47_63 : 17;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_timing_params1_s cn61xx;
|
|
struct cvmx_lmcx_timing_params1_s cn63xx;
|
|
struct cvmx_lmcx_timing_params1_cn63xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_46_63 : 18;
|
|
uint64_t txpdll : 5; /**< Indicates tXPDLL constraints.
|
|
Set TXPDLL (CSR field) = RNDUP[tXPDLL(ns)/tCYC(ns)]-1,
|
|
where tXPDLL is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(10nCK, 24ns) */
|
|
uint64_t tfaw : 5; /**< Indicates tFAW constraints.
|
|
Set TFAW (CSR field) = RNDUP[tFAW(ns)/(4*tCYC(ns))],
|
|
where tFAW is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=30-40ns */
|
|
uint64_t twldqsen : 4; /**< Indicates tWLDQSEN constraints.
|
|
Set TWLDQSEN (CSR field) = RNDUP[tWLDQSEN(ns)/(4*tCYC(ns))],
|
|
where tWLDQSEN is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(25nCK) */
|
|
uint64_t twlmrd : 4; /**< Indicates tWLMRD constraints.
|
|
Set TWLMRD (CSR field) = RNDUP[tWLMRD(ns)/(4*tCYC(ns))],
|
|
where tWLMRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(40nCK) */
|
|
uint64_t txp : 3; /**< Indicates tXP constraints.
|
|
Set TXP (CSR field) = RNDUP[tXP(ns)/tCYC(ns)]-1,
|
|
where tXP is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(3nCK, 7.5ns) */
|
|
uint64_t trrd : 3; /**< Indicates tRRD constraints.
|
|
Set TRRD (CSR field) = RNDUP[tRRD(ns)/tCYC(ns)]-2,
|
|
where tRRD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(4nCK, 10ns)
|
|
- 000: RESERVED
|
|
- 001: 3 tCYC
|
|
- ...
|
|
- 110: 8 tCYC
|
|
- 111: 9 tCYC */
|
|
uint64_t trfc : 5; /**< Indicates tRFC constraints.
|
|
Set TRFC (CSR field) = RNDUP[tRFC(ns)/(8*tCYC(ns))],
|
|
where tRFC is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=90-350ns
|
|
- 00000: RESERVED
|
|
- 00001: 8 tCYC
|
|
- 00010: 16 tCYC
|
|
- 00011: 24 tCYC
|
|
- 00100: 32 tCYC
|
|
- ...
|
|
- 11110: 240 tCYC
|
|
- 11111: 248 tCYC */
|
|
uint64_t twtr : 4; /**< Indicates tWTR constraints.
|
|
Set TWTR (CSR field) = RNDUP[tWTR(ns)/tCYC(ns)]-1,
|
|
where tWTR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=max(4nCK, 7.5ns)
|
|
- 0000: RESERVED
|
|
- 0001: 2
|
|
- ...
|
|
- 0111: 8
|
|
- 1000-1111: RESERVED */
|
|
uint64_t trcd : 4; /**< Indicates tRCD constraints.
|
|
Set TRCD (CSR field) = RNDUP[tRCD(ns)/tCYC(ns)],
|
|
where tRCD is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=10-15ns
|
|
- 0000: RESERVED
|
|
- 0001: 2 (2 is the smallest value allowed)
|
|
- 0002: 2
|
|
- ...
|
|
- 1001: 9
|
|
- 1010-1111: RESERVED
|
|
In 2T mode, make this register TRCD-1, not going
|
|
below 2. */
|
|
uint64_t tras : 5; /**< Indicates tRAS constraints.
|
|
Set TRAS (CSR field) = RNDUP[tRAS(ns)/tCYC(ns)]-1,
|
|
where tRAS is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=35ns-9*tREFI
|
|
- 00000: RESERVED
|
|
- 00001: 2 tCYC
|
|
- 00010: 3 tCYC
|
|
- ...
|
|
- 11111: 32 tCYC */
|
|
uint64_t tmprr : 4; /**< Indicates tMPRR constraints.
|
|
Set TMPRR (CSR field) = RNDUP[tMPRR(ns)/tCYC(ns)]-1,
|
|
where tMPRR is from the DDR3 spec, and tCYC(ns)
|
|
is the DDR clock frequency (not data rate).
|
|
TYP=1nCK */
|
|
#else
|
|
uint64_t tmprr : 4;
|
|
uint64_t tras : 5;
|
|
uint64_t trcd : 4;
|
|
uint64_t twtr : 4;
|
|
uint64_t trfc : 5;
|
|
uint64_t trrd : 3;
|
|
uint64_t txp : 3;
|
|
uint64_t twlmrd : 4;
|
|
uint64_t twldqsen : 4;
|
|
uint64_t tfaw : 5;
|
|
uint64_t txpdll : 5;
|
|
uint64_t reserved_46_63 : 18;
|
|
#endif
|
|
} cn63xxp1;
|
|
struct cvmx_lmcx_timing_params1_s cn66xx;
|
|
struct cvmx_lmcx_timing_params1_s cn68xx;
|
|
struct cvmx_lmcx_timing_params1_s cn68xxp1;
|
|
struct cvmx_lmcx_timing_params1_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_timing_params1 cvmx_lmcx_timing_params1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_tro_ctl
|
|
*
|
|
* LMC_TRO_CTL = LMC Temperature Ring Osc Control
|
|
* This register is an assortment of various control fields needed to control the temperature ring oscillator
|
|
*
|
|
* Notes:
|
|
* To bring up the temperature ring oscillator, write TRESET to 0, and follow by initializing RCLK_CNT to desired
|
|
* value
|
|
*/
|
|
union cvmx_lmcx_tro_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_tro_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_33_63 : 31;
|
|
uint64_t rclk_cnt : 32; /**< rclk counter */
|
|
uint64_t treset : 1; /**< Reset ring oscillator */
|
|
#else
|
|
uint64_t treset : 1;
|
|
uint64_t rclk_cnt : 32;
|
|
uint64_t reserved_33_63 : 31;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_tro_ctl_s cn61xx;
|
|
struct cvmx_lmcx_tro_ctl_s cn63xx;
|
|
struct cvmx_lmcx_tro_ctl_s cn63xxp1;
|
|
struct cvmx_lmcx_tro_ctl_s cn66xx;
|
|
struct cvmx_lmcx_tro_ctl_s cn68xx;
|
|
struct cvmx_lmcx_tro_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_tro_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_tro_ctl cvmx_lmcx_tro_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_tro_stat
|
|
*
|
|
* LMC_TRO_STAT = LMC Temperature Ring Osc Status
|
|
* This register is an assortment of various control fields needed to control the temperature ring oscillator
|
|
*/
|
|
union cvmx_lmcx_tro_stat {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_tro_stat_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t ring_cnt : 32; /**< ring counter */
|
|
#else
|
|
uint64_t ring_cnt : 32;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_tro_stat_s cn61xx;
|
|
struct cvmx_lmcx_tro_stat_s cn63xx;
|
|
struct cvmx_lmcx_tro_stat_s cn63xxp1;
|
|
struct cvmx_lmcx_tro_stat_s cn66xx;
|
|
struct cvmx_lmcx_tro_stat_s cn68xx;
|
|
struct cvmx_lmcx_tro_stat_s cn68xxp1;
|
|
struct cvmx_lmcx_tro_stat_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_tro_stat cvmx_lmcx_tro_stat_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_wlevel_ctl
|
|
*/
|
|
union cvmx_lmcx_wlevel_ctl {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_wlevel_ctl_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_22_63 : 42;
|
|
uint64_t rtt_nom : 3; /**< RTT_NOM
|
|
LMC writes a decoded value to MR1[Rtt_Nom] of the rank during
|
|
write leveling. Per JEDEC DDR3 specifications,
|
|
only values MR1[Rtt_Nom] = 1 (RQZ/4), 2 (RQZ/2), or 3 (RQZ/6)
|
|
are allowed during write leveling with output buffer enabled.
|
|
000 : LMC writes 001 (RZQ/4) to MR1[Rtt_Nom]
|
|
001 : LMC writes 010 (RZQ/2) to MR1[Rtt_Nom]
|
|
010 : LMC writes 011 (RZQ/6) to MR1[Rtt_Nom]
|
|
011 : LMC writes 100 (RZQ/12) to MR1[Rtt_Nom]
|
|
100 : LMC writes 101 (RZQ/8) to MR1[Rtt_Nom]
|
|
101 : LMC writes 110 (Rsvd) to MR1[Rtt_Nom]
|
|
110 : LMC writes 111 (Rsvd) to MR1[Rtt_Nom]
|
|
111 : LMC writes 000 (Disabled) to MR1[Rtt_Nom] */
|
|
uint64_t bitmask : 8; /**< Mask to select bit lanes on which write-leveling
|
|
feedback is returned when OR_DIS is set to 1 */
|
|
uint64_t or_dis : 1; /**< Disable or'ing of bits in a byte lane when computing
|
|
the write-leveling bitmask */
|
|
uint64_t sset : 1; /**< Run write-leveling on the current setting only. */
|
|
uint64_t lanemask : 9; /**< One-hot mask to select byte lane to be leveled by
|
|
the write-leveling sequence
|
|
Used with x16 parts where the upper and lower byte
|
|
lanes need to be leveled independently */
|
|
#else
|
|
uint64_t lanemask : 9;
|
|
uint64_t sset : 1;
|
|
uint64_t or_dis : 1;
|
|
uint64_t bitmask : 8;
|
|
uint64_t rtt_nom : 3;
|
|
uint64_t reserved_22_63 : 42;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_wlevel_ctl_s cn61xx;
|
|
struct cvmx_lmcx_wlevel_ctl_s cn63xx;
|
|
struct cvmx_lmcx_wlevel_ctl_cn63xxp1 {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_10_63 : 54;
|
|
uint64_t sset : 1; /**< Run write-leveling on the current setting only. */
|
|
uint64_t lanemask : 9; /**< One-hot mask to select byte lane to be leveled by
|
|
the write-leveling sequence
|
|
Used with x16 parts where the upper and lower byte
|
|
lanes need to be leveled independently */
|
|
#else
|
|
uint64_t lanemask : 9;
|
|
uint64_t sset : 1;
|
|
uint64_t reserved_10_63 : 54;
|
|
#endif
|
|
} cn63xxp1;
|
|
struct cvmx_lmcx_wlevel_ctl_s cn66xx;
|
|
struct cvmx_lmcx_wlevel_ctl_s cn68xx;
|
|
struct cvmx_lmcx_wlevel_ctl_s cn68xxp1;
|
|
struct cvmx_lmcx_wlevel_ctl_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_wlevel_ctl cvmx_lmcx_wlevel_ctl_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_wlevel_dbg
|
|
*
|
|
* Notes:
|
|
* A given write of LMC*_WLEVEL_DBG returns the write-leveling pass/fail results for all possible
|
|
* delay settings (i.e. the BITMASK) for only one byte in the last rank that the HW write-leveled.
|
|
* LMC*_WLEVEL_DBG[BYTE] selects the particular byte.
|
|
* To get these pass/fail results for another different rank, you must run the hardware write-leveling
|
|
* again. For example, it is possible to get the BITMASK results for every byte of every rank
|
|
* if you run write-leveling separately for each rank, probing LMC*_WLEVEL_DBG between each
|
|
* write-leveling.
|
|
*/
|
|
union cvmx_lmcx_wlevel_dbg {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_wlevel_dbg_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_12_63 : 52;
|
|
uint64_t bitmask : 8; /**< Bitmask generated during deskew settings sweep
|
|
if LMCX_WLEVEL_CTL[SSET]=0
|
|
BITMASK[n]=0 means deskew setting n failed
|
|
BITMASK[n]=1 means deskew setting n passed
|
|
for 0 <= n <= 7
|
|
BITMASK contains the first 8 results of the total 16
|
|
collected by LMC during the write-leveling sequence
|
|
else if LMCX_WLEVEL_CTL[SSET]=1
|
|
BITMASK[0]=0 means curr deskew setting failed
|
|
BITMASK[0]=1 means curr deskew setting passed */
|
|
uint64_t byte : 4; /**< 0 <= BYTE <= 8 */
|
|
#else
|
|
uint64_t byte : 4;
|
|
uint64_t bitmask : 8;
|
|
uint64_t reserved_12_63 : 52;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_wlevel_dbg_s cn61xx;
|
|
struct cvmx_lmcx_wlevel_dbg_s cn63xx;
|
|
struct cvmx_lmcx_wlevel_dbg_s cn63xxp1;
|
|
struct cvmx_lmcx_wlevel_dbg_s cn66xx;
|
|
struct cvmx_lmcx_wlevel_dbg_s cn68xx;
|
|
struct cvmx_lmcx_wlevel_dbg_s cn68xxp1;
|
|
struct cvmx_lmcx_wlevel_dbg_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_wlevel_dbg cvmx_lmcx_wlevel_dbg_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_wlevel_rank#
|
|
*
|
|
* Notes:
|
|
* This is four CSRs per LMC, one per each rank.
|
|
*
|
|
* Deskew setting is measured in units of 1/8 CK, so the above BYTE* values can range over 4 CKs.
|
|
*
|
|
* Assuming LMC*_WLEVEL_CTL[SSET]=0, the BYTE*<2:0> values are not used during write-leveling, and
|
|
* they are over-written by the hardware as part of the write-leveling sequence. (HW sets STATUS==3
|
|
* after HW write-leveling completes for the rank). SW needs to set BYTE*<4:3> bits.
|
|
*
|
|
* Each CSR may also be written by SW, but not while a write-leveling sequence is in progress. (HW sets STATUS==1 after a CSR write.)
|
|
*
|
|
* SW initiates a HW write-leveling sequence by programming LMC*_WLEVEL_CTL and writing RANKMASK and INIT_START=1 with SEQUENCE=6 in LMC*_CONFIG.
|
|
* LMC will then step through and accumulate write leveling results for 8 unique delay settings (twice), starting at a delay of
|
|
* LMC*_WLEVEL_RANKn[BYTE*<4:3>]*8 CK increasing by 1/8 CK each setting. HW will then set LMC*_WLEVEL_RANKi[BYTE*<2:0>] to indicate the
|
|
* first write leveling result of '1' that followed a reslt of '0' during the sequence by searching for a '1100' pattern in the generated
|
|
* bitmask, except that LMC will always write LMC*_WLEVEL_RANKi[BYTE*<0>]=0. If HW is unable to find a match for a '1100' pattern, then HW will
|
|
* set LMC*_WLEVEL_RANKi[BYTE*<2:0>] to 4.
|
|
* See LMC*_WLEVEL_CTL.
|
|
*
|
|
* LMC*_WLEVEL_RANKi values for ranks i without attached DRAM should be set such that
|
|
* they do not increase the range of possible BYTE values for any byte
|
|
* lane. The easiest way to do this is to set
|
|
* LMC*_WLEVEL_RANKi = LMC*_WLEVEL_RANKj,
|
|
* where j is some rank with attached DRAM whose LMC*_WLEVEL_RANKj is already fully initialized.
|
|
*/
|
|
union cvmx_lmcx_wlevel_rankx {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_wlevel_rankx_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_47_63 : 17;
|
|
uint64_t status : 2; /**< Indicates status of the write-leveling and where
|
|
the BYTE* programmings in <44:0> came from:
|
|
0 = BYTE* values are their reset value
|
|
1 = BYTE* values were set via a CSR write to this register
|
|
2 = write-leveling sequence currently in progress (BYTE* values are unpredictable)
|
|
3 = BYTE* values came from a complete write-leveling sequence, irrespective of
|
|
which lanes are masked via LMC*WLEVEL_CTL[LANEMASK] */
|
|
uint64_t byte8 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE8 must be zero during normal operation.
|
|
When ECC DRAM is not present (i.e. when DRAM is not
|
|
attached to chip signals DDR_CBS_0_* and DDR_CB[7:0]),
|
|
SW should write BYTE8 with a value that does
|
|
not increase the range of possible BYTE* values. The
|
|
easiest way to do this is to set
|
|
LMC*_WLEVEL_RANK*[BYTE8] = LMC*_WLEVEL_RANK*[BYTE0]
|
|
when there is no ECC DRAM, using the final BYTE0 value. */
|
|
uint64_t byte7 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE7 must be zero during normal operation */
|
|
uint64_t byte6 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE6 must be zero during normal operation */
|
|
uint64_t byte5 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE5 must be zero during normal operation */
|
|
uint64_t byte4 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE4 must be zero during normal operation */
|
|
uint64_t byte3 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE3 must be zero during normal operation */
|
|
uint64_t byte2 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE2 must be zero during normal operation */
|
|
uint64_t byte1 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE1 must be zero during normal operation */
|
|
uint64_t byte0 : 5; /**< Deskew setting
|
|
Bit 0 of BYTE0 must be zero during normal operation */
|
|
#else
|
|
uint64_t byte0 : 5;
|
|
uint64_t byte1 : 5;
|
|
uint64_t byte2 : 5;
|
|
uint64_t byte3 : 5;
|
|
uint64_t byte4 : 5;
|
|
uint64_t byte5 : 5;
|
|
uint64_t byte6 : 5;
|
|
uint64_t byte7 : 5;
|
|
uint64_t byte8 : 5;
|
|
uint64_t status : 2;
|
|
uint64_t reserved_47_63 : 17;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_wlevel_rankx_s cn61xx;
|
|
struct cvmx_lmcx_wlevel_rankx_s cn63xx;
|
|
struct cvmx_lmcx_wlevel_rankx_s cn63xxp1;
|
|
struct cvmx_lmcx_wlevel_rankx_s cn66xx;
|
|
struct cvmx_lmcx_wlevel_rankx_s cn68xx;
|
|
struct cvmx_lmcx_wlevel_rankx_s cn68xxp1;
|
|
struct cvmx_lmcx_wlevel_rankx_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_wlevel_rankx cvmx_lmcx_wlevel_rankx_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_wodt_ctl0
|
|
*
|
|
* LMC_WODT_CTL0 = LMC Write OnDieTermination control
|
|
* See the description in LMC_WODT_CTL1.
|
|
*
|
|
* Notes:
|
|
* Together, the LMC_WODT_CTL1 and LMC_WODT_CTL0 CSRs control the write ODT mask. See LMC_WODT_CTL1.
|
|
*
|
|
*/
|
|
union cvmx_lmcx_wodt_ctl0 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_wodt_ctl0_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_0_63 : 64;
|
|
#else
|
|
uint64_t reserved_0_63 : 64;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_wodt_ctl0_cn30xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t wodt_d1_r1 : 8; /**< Write ODT mask DIMM1, RANK1 */
|
|
uint64_t wodt_d1_r0 : 8; /**< Write ODT mask DIMM1, RANK0 */
|
|
uint64_t wodt_d0_r1 : 8; /**< Write ODT mask DIMM0, RANK1 */
|
|
uint64_t wodt_d0_r0 : 8; /**< Write ODT mask DIMM0, RANK0 */
|
|
#else
|
|
uint64_t wodt_d0_r0 : 8;
|
|
uint64_t wodt_d0_r1 : 8;
|
|
uint64_t wodt_d1_r0 : 8;
|
|
uint64_t wodt_d1_r1 : 8;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn30xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn30xx cn31xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn38xx {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t wodt_hi3 : 4; /**< Write ODT mask for position 3, data[127:64] */
|
|
uint64_t wodt_hi2 : 4; /**< Write ODT mask for position 2, data[127:64] */
|
|
uint64_t wodt_hi1 : 4; /**< Write ODT mask for position 1, data[127:64] */
|
|
uint64_t wodt_hi0 : 4; /**< Write ODT mask for position 0, data[127:64] */
|
|
uint64_t wodt_lo3 : 4; /**< Write ODT mask for position 3, data[ 63: 0] */
|
|
uint64_t wodt_lo2 : 4; /**< Write ODT mask for position 2, data[ 63: 0] */
|
|
uint64_t wodt_lo1 : 4; /**< Write ODT mask for position 1, data[ 63: 0] */
|
|
uint64_t wodt_lo0 : 4; /**< Write ODT mask for position 0, data[ 63: 0] */
|
|
#else
|
|
uint64_t wodt_lo0 : 4;
|
|
uint64_t wodt_lo1 : 4;
|
|
uint64_t wodt_lo2 : 4;
|
|
uint64_t wodt_lo3 : 4;
|
|
uint64_t wodt_hi0 : 4;
|
|
uint64_t wodt_hi1 : 4;
|
|
uint64_t wodt_hi2 : 4;
|
|
uint64_t wodt_hi3 : 4;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} cn38xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn38xx cn38xxp2;
|
|
struct cvmx_lmcx_wodt_ctl0_cn38xx cn50xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn30xx cn52xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn30xx cn52xxp1;
|
|
struct cvmx_lmcx_wodt_ctl0_cn30xx cn56xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn30xx cn56xxp1;
|
|
struct cvmx_lmcx_wodt_ctl0_cn38xx cn58xx;
|
|
struct cvmx_lmcx_wodt_ctl0_cn38xx cn58xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_wodt_ctl0 cvmx_lmcx_wodt_ctl0_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_wodt_ctl1
|
|
*
|
|
* LMC_WODT_CTL1 = LMC Write OnDieTermination control
|
|
* System designers may desire to terminate DQ/DQS/DM lines for higher frequency DDR operations
|
|
* (667MHz and faster), especially on a multi-rank system. DDR2 DQ/DM/DQS I/O's have built in
|
|
* Termination resistor that can be turned on or off by the controller, after meeting tAOND and tAOF
|
|
* timing requirements. Each Rank has its own ODT pin that fans out to all the memory parts
|
|
* in that DIMM. System designers may prefer different combinations of ODT ON's for read and write
|
|
* into different ranks. Octeon supports full programmability by way of the mask register below.
|
|
* Each Rank position has its own 8-bit programmable field.
|
|
* When the controller does a write to that rank, it sets the 8 ODT pins to the MASK pins below.
|
|
* For eg., When doing a write into Rank0, a system designer may desire to terminate the lines
|
|
* with the resistor on Dimm0/Rank1. The mask WODT_D0_R0 would then be [00000010].
|
|
* If ODT feature is not desired, the DDR parts can be programmed to not look at these pins by
|
|
* writing 0 in QS_DIC. Octeon drives the appropriate mask values on the ODT pins by default.
|
|
* If this feature is not required, write 0 in this register.
|
|
*
|
|
* Notes:
|
|
* Together, the LMC_WODT_CTL1 and LMC_WODT_CTL0 CSRs control the write ODT mask.
|
|
* When a given RANK is selected, the WODT mask for that RANK is used. The resulting WODT mask is
|
|
* driven to the DIMMs in the following manner:
|
|
* BUNK_ENA=1 BUNK_ENA=0
|
|
* Mask[7] -> DIMM3, RANK1 DIMM3
|
|
* Mask[6] -> DIMM3, RANK0
|
|
* Mask[5] -> DIMM2, RANK1 DIMM2
|
|
* Mask[4] -> DIMM2, RANK0
|
|
* Mask[3] -> DIMM1, RANK1 DIMM1
|
|
* Mask[2] -> DIMM1, RANK0
|
|
* Mask[1] -> DIMM0, RANK1 DIMM0
|
|
* Mask[0] -> DIMM0, RANK0
|
|
*/
|
|
union cvmx_lmcx_wodt_ctl1 {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_wodt_ctl1_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t reserved_32_63 : 32;
|
|
uint64_t wodt_d3_r1 : 8; /**< Write ODT mask DIMM3, RANK1/DIMM3 in SingleRanked */
|
|
uint64_t wodt_d3_r0 : 8; /**< Write ODT mask DIMM3, RANK0 */
|
|
uint64_t wodt_d2_r1 : 8; /**< Write ODT mask DIMM2, RANK1/DIMM2 in SingleRanked */
|
|
uint64_t wodt_d2_r0 : 8; /**< Write ODT mask DIMM2, RANK0 */
|
|
#else
|
|
uint64_t wodt_d2_r0 : 8;
|
|
uint64_t wodt_d2_r1 : 8;
|
|
uint64_t wodt_d3_r0 : 8;
|
|
uint64_t wodt_d3_r1 : 8;
|
|
uint64_t reserved_32_63 : 32;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_wodt_ctl1_s cn30xx;
|
|
struct cvmx_lmcx_wodt_ctl1_s cn31xx;
|
|
struct cvmx_lmcx_wodt_ctl1_s cn52xx;
|
|
struct cvmx_lmcx_wodt_ctl1_s cn52xxp1;
|
|
struct cvmx_lmcx_wodt_ctl1_s cn56xx;
|
|
struct cvmx_lmcx_wodt_ctl1_s cn56xxp1;
|
|
};
|
|
typedef union cvmx_lmcx_wodt_ctl1 cvmx_lmcx_wodt_ctl1_t;
|
|
|
|
/**
|
|
* cvmx_lmc#_wodt_mask
|
|
*
|
|
* LMC_WODT_MASK = LMC Write OnDieTermination mask
|
|
* System designers may desire to terminate DQ/DQS lines for higher frequency DDR operations
|
|
* especially on a multi-rank system. DDR3 DQ/DQS I/O's have built in
|
|
* Termination resistor that can be turned on or off by the controller, after meeting tAOND and tAOF
|
|
* timing requirements. Each Rank has its own ODT pin that fans out to all the memory parts
|
|
* in that DIMM. System designers may prefer different combinations of ODT ON's for writes
|
|
* into different ranks. Octeon supports full programmability by way of the mask register below.
|
|
* Each Rank position has its own 8-bit programmable field.
|
|
* When the controller does a write to that rank, it sets the 4 ODT pins to the MASK pins below.
|
|
* For eg., When doing a write into Rank0, a system designer may desire to terminate the lines
|
|
* with the resistor on DIMM0/Rank1. The mask WODT_D0_R0 would then be [00000010].
|
|
* Octeon drives the appropriate mask values on the ODT pins by default. If this feature is not
|
|
* required, write 0 in this register.
|
|
*
|
|
* Notes:
|
|
* When a given RANK is selected, the WODT mask for that RANK is used. The resulting WODT mask is
|
|
* driven to the DIMMs in the following manner:
|
|
* RANK_ENA=1 RANK_ENA=0
|
|
* Mask[3] -> DIMM1_ODT_1 MBZ
|
|
* Mask[2] -> DIMM1_ODT_0 DIMM1_ODT_0
|
|
* Mask[1] -> DIMM0_ODT_1 MBZ
|
|
* Mask[0] -> DIMM0_ODT_0 DIMM0_ODT_0
|
|
*
|
|
* LMC always writes entire cache blocks and always writes them via two consecutive
|
|
* write CAS operations to the same rank+bank+row spaced exactly 4 CK's apart.
|
|
* When a WODT mask bit is set, LMC asserts the OCTEON ODT output
|
|
* pin(s) starting the same CK as the first write CAS operation. Then, OCTEON
|
|
* normally continues to assert the ODT output pin(s) for 9+LMC*_CONTROL[WODT_BPRCH] more CK's
|
|
* - for a total of 10+LMC*_CONTROL[WODT_BPRCH] CK's for the entire cache block write -
|
|
* through the second write CAS operation of the cache block,
|
|
* satisfying the 6 CK DDR3 ODTH8 requirements.
|
|
* But it is possible for OCTEON to issue two cache block writes separated by as few as
|
|
* WtW = 8 or 9 (10 if LMC*_CONTROL[WODT_BPRCH]=1) CK's. In that case, OCTEON asserts the ODT output pin(s)
|
|
* for the WODT mask of the first cache block write for WtW CK's, then asserts
|
|
* the ODT output pin(s) for the WODT mask of the second cache block write for 10+LMC*_CONTROL[WODT_BPRCH] CK's
|
|
* (or less if a third cache block write follows within 8 or 9 (or 10) CK's of this second cache block write).
|
|
* Note that it may be necessary to force LMC to space back-to-back cache block writes
|
|
* to different ranks apart by at least 10+LMC*_CONTROL[WODT_BPRCH] CK's to prevent DDR3 ODTH8 violations.
|
|
*/
|
|
union cvmx_lmcx_wodt_mask {
|
|
uint64_t u64;
|
|
struct cvmx_lmcx_wodt_mask_s {
|
|
#ifdef __BIG_ENDIAN_BITFIELD
|
|
uint64_t wodt_d3_r1 : 8; /**< Write ODT mask DIMM3, RANK1/DIMM3 in SingleRanked
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d3_r0 : 8; /**< Write ODT mask DIMM3, RANK0
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d2_r1 : 8; /**< Write ODT mask DIMM2, RANK1/DIMM2 in SingleRanked
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d2_r0 : 8; /**< Write ODT mask DIMM2, RANK0
|
|
*UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d1_r1 : 8; /**< Write ODT mask DIMM1, RANK1/DIMM1 in SingleRanked
|
|
if (!RANK_ENA) then WODT_D1_R1[3:0] MBZ
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d1_r0 : 8; /**< Write ODT mask DIMM1, RANK0
|
|
if (!RANK_ENA) then WODT_D1_R0[3,1] MBZ
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d0_r1 : 8; /**< Write ODT mask DIMM0, RANK1/DIMM0 in SingleRanked
|
|
if (!RANK_ENA) then WODT_D0_R1[3:0] MBZ
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
uint64_t wodt_d0_r0 : 8; /**< Write ODT mask DIMM0, RANK0
|
|
if (!RANK_ENA) then WODT_D0_R0[3,1] MBZ
|
|
*Upper 4 bits UNUSED IN 6xxx, and MBZ* */
|
|
#else
|
|
uint64_t wodt_d0_r0 : 8;
|
|
uint64_t wodt_d0_r1 : 8;
|
|
uint64_t wodt_d1_r0 : 8;
|
|
uint64_t wodt_d1_r1 : 8;
|
|
uint64_t wodt_d2_r0 : 8;
|
|
uint64_t wodt_d2_r1 : 8;
|
|
uint64_t wodt_d3_r0 : 8;
|
|
uint64_t wodt_d3_r1 : 8;
|
|
#endif
|
|
} s;
|
|
struct cvmx_lmcx_wodt_mask_s cn61xx;
|
|
struct cvmx_lmcx_wodt_mask_s cn63xx;
|
|
struct cvmx_lmcx_wodt_mask_s cn63xxp1;
|
|
struct cvmx_lmcx_wodt_mask_s cn66xx;
|
|
struct cvmx_lmcx_wodt_mask_s cn68xx;
|
|
struct cvmx_lmcx_wodt_mask_s cn68xxp1;
|
|
struct cvmx_lmcx_wodt_mask_s cnf71xx;
|
|
};
|
|
typedef union cvmx_lmcx_wodt_mask cvmx_lmcx_wodt_mask_t;
|
|
|
|
#endif
|