/* * This file is part of the coreboot project. * * Copyright (C) 2013 Alexandru Gagniuc <mr.nuke.me@gmail.com> * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ #ifndef DEVICE_DRAM_DDR3L_H #define DEVICE_DRAM_DDR3L_H /** * @file ddr3.h * * \brief Utilities for decoding DDR3 SPDs */ #include <stdint.h> #include <spd.h> /** * \brief Convenience definitions for TCK values * * Different values for tCK, representing standard DDR3 frequencies. * These values are in 1/256 ns units. * @{ */ #define TCK_1066MHZ 240 #define TCK_933MHZ 275 #define TCK_800MHZ 320 #define TCK_666MHZ 384 #define TCK_533MHZ 480 #define TCK_400MHZ 640 #define TCK_333MHZ 768 #define TCK_266MHZ 960 #define TCK_200MHZ 1280 /** @} */ /** * \brief Convenience macro for enabling printk with CONFIG_DEBUG_RAM_SETUP * * Use this macro instead of printk(); for verbose RAM initialization messages. * When CONFIG_DEBUG_RAM_SETUP is not selected, these messages are automatically * disabled. * @{ */ #if defined(CONFIG_DEBUG_RAM_SETUP) && (CONFIG_DEBUG_RAM_SETUP) #define printram(x, ...) printk(BIOS_DEBUG, x, ##__VA_ARGS__) #else #define printram(x, ...) #endif /** @} */ /* * Module type (byte 3, bits 3:0) of SPD * This definition is specific to DDR3. DDR2 SPDs have a different structure. */ enum spd_dimm_type { SPD_DIMM_TYPE_UNDEFINED = 0x00, SPD_DIMM_TYPE_RDIMM = 0x01, SPD_DIMM_TYPE_UDIMM = 0x02, SPD_DIMM_TYPE_SO_DIMM = 0x03, SPD_DIMM_TYPE_MICRO_DIMM = 0x04, SPD_DIMM_TYPE_MINI_RDIMM = 0x05, SPD_DIMM_TYPE_MINI_UDIMM = 0x06, SPD_DIMM_TYPE_MINI_CDIMM = 0x07, SPD_DIMM_TYPE_72B_SO_UDIMM = 0x08, SPD_DIMM_TYPE_72B_SO_RDIMM = 0x09, SPD_DIMM_TYPE_72B_SO_CDIMM = 0x0a, SPD_DIMM_TYPE_LRDIMM = 0x0b, SPD_DIMM_TYPE_16B_SO_DIMM = 0x0d, SPD_DIMM_TYPE_32B_SO_DIMM = 0x0e, /* Masks to bits 3:0 to give the dimm type */ SPD_DIMM_TYPE_MASK = 0x0f, }; /** * \brief DIMM flags * * Characteristic flags for the DIMM, as presented by the SPD */ typedef union dimm_flags_st { /* The whole point of the union/struct construct is to allow us to clear * all the bits with one line: flags.raw = 0. * We do not care how these bits are ordered */ struct { /* Indicates if rank 1 of DIMM uses a mirrored pin mapping. See: * Annex K: Serial Presence Detect (SPD) for DDR3 SDRAM */ unsigned pins_mirrored:1; /* Module can work at 1.50V - All DIMMS must be 1.5V operable */ unsigned operable_1_50V:1; /* Module can work at 1.35V */ unsigned operable_1_35V:1; /* Module can work at 1.20V */ unsigned operable_1_25V:1; /* Has an 8-bit bus extension, meaning the DIMM supports ECC */ unsigned is_ecc:1; /* DLL-Off Mode Support */ unsigned dll_off_mode:1; /* Indicates a drive strength of RZQ/6 (40 Ohm) is supported */ unsigned rzq6_supported:1; /* Indicates a drive strength of RZQ/7 (35 Ohm) is supported */ unsigned rzq7_supported:1; /* Partial Array Self Refresh */ unsigned pasr:1; /* On-die Thermal Sensor Readout */ unsigned odts:1; /* Auto Self Refresh */ unsigned asr:1; /* Extended temperature range supported */ unsigned ext_temp_range:1; /* Operating at extended temperature requires 2X refresh rate */ unsigned ext_temp_refresh:1; /* Thermal sensor incorporated */ unsigned therm_sensor:1; }; unsigned raw; } dimm_flags_t; /** * \brief DIMM characteristics * * The characteristics of each DIMM, as presented by the SPD */ typedef struct dimm_attr_st { enum spd_memory_type dram_type; enum spd_dimm_type dimm_type; u16 cas_supported; /* Flags extracted from SPD */ dimm_flags_t flags; /* SDRAM width */ u8 width; /* Number of ranks */ u8 ranks; /* Number or row address bits */ u8 row_bits; /* Number or column address bits */ u8 col_bits; /* Size of module in MiB */ u32 size_mb; /* Latencies are in units of 1/256 ns */ u32 tCK; u32 tAA; u32 tWR; u32 tRCD; u32 tRRD; u32 tRP; u32 tRAS; u32 tRC; u32 tRFC; u32 tWTR; u32 tRTP; u32 tFAW; u8 reference_card; } dimm_attr; /** Result of the SPD decoding process */ enum spd_status { SPD_STATUS_OK = 0, SPD_STATUS_INVALID, SPD_STATUS_CRC_ERROR, SPD_STATUS_INVALID_FIELD, }; typedef u8 spd_raw_data[256]; u16 spd_ddr3_calc_crc(u8 *spd, int len); int spd_decode_ddr3(dimm_attr * dimm, spd_raw_data spd_data); int dimm_is_registered(enum spd_dimm_type type); void dram_print_spd_ddr3(const dimm_attr * dimm); /** * \brief Read double word from specified address * * Should be useful when doing an MRS to the DIMM */ static inline u32 volatile_read(volatile u32 addr) { volatile u32 result; result = *(volatile u32 *)addr; return result; } /** * \brief Representation of an MRS command * * This represents an MRS command as seen by the DIMM. This is not a memory * address that can be read to generate an MRS command. The mapping of CPU * to memory pins is hardware-dependent. * \n * The idea is to generalize the MRS code, and only need a hardware-specific * function to map the MRS bits to CPU address bits. An MRS command can be * sent like: * @code{.c} * u32 addr; * mrs_cmd_t mrs; * chipset_enable_mrs_command_mode(); * mrs = ddr3_get_mr2(rtt_wr, srt, asr, cwl) * if (rank_has_mirrorred_pins) * mrs = ddr3_mrs_mirror_pins(mrs); * addr = chipset_specific_get_mrs_addr(mrs); * volatile_read(addr); * @endcode * * The MRS representation has the following structure: * - cmd[15:0] = Address pins MA[15:0] * - cmd[18:16] = Bank address BA[2:0] */ typedef u32 mrs_cmd_t; enum ddr3_mr0_precharge { DDR3_MR0_PRECHARGE_SLOW = 0, DDR3_MR0_PRECHARGE_FAST = 1, }; enum ddr3_mr0_mode { DDR3_MR0_MODE_NORMAL = 0, DDR3_MR0_MODE_TEST = 1, }; enum ddr3_mr0_dll_reset { DDR3_MR0_DLL_RESET_NO = 0, DDR3_MR0_DLL_RESET_YES = 1, }; enum ddr3_mr0_burst_type { DDR3_MR0_BURST_TYPE_SEQUENTIAL = 0, DDR3_MR0_BURST_TYPE_INTERLEAVED = 1, }; enum ddr3_mr0_burst_length { DDR3_MR0_BURST_LENGTH_8 = 0, DDR3_MR0_BURST_LENGTH_CHOP = 1, DDR3_MR0_BURST_LENGTH_4 = 2, }; mrs_cmd_t ddr3_get_mr0(enum ddr3_mr0_precharge precharge_pd, u8 write_recovery, enum ddr3_mr0_dll_reset dll_reset, enum ddr3_mr0_mode mode, u8 cas, enum ddr3_mr0_burst_type interleaved_burst, enum ddr3_mr0_burst_length burst_length); enum ddr3_mr1_qoff { DDR3_MR1_QOFF_ENABLE = 0, DDR3_MR1_QOFF_DISABLE = 1, }; enum ddr3_mr1_tqds { DDR3_MR1_TQDS_DISABLE = 0, DDR3_MR1_TQDS_ENABLE = 1, }; enum ddr3_mr1_write_leveling { DDR3_MR1_WRLVL_DISABLE = 0, DDR3_MR1_WRLVL_ENABLE = 1, }; enum ddr3_mr1_rtt_nom { DDR3_MR1_RTT_NOM_OFF = 0, DDR3_MR1_RTT_NOM_RZQ4 = 1, DDR3_MR1_RTT_NOM_RZQ2 = 2, DDR3_MR1_RTT_NOM_RZQ6 = 3, DDR3_MR1_RTT_NOM_RZQ12 = 4, DDR3_MR1_RTT_NOM_RZQ8 = 5, }; enum ddr3_mr1_additive_latency { DDR3_MR1_AL_DISABLE = 0, DDR3_MR1_AL_CL_MINUS_1 = 1, DDR3_MR1_AL_CL_MINUS_2 = 2, }; enum ddr3_mr1_ods { DDR3_MR1_ODS_RZQ6 = 0, DDR3_MR1_ODS_RZQ7 = 1, }; enum ddr3_mr1_dll { DDR3_MR1_DLL_ENABLE = 0, DDR3_MR1_DLL_DISABLE = 1, }; mrs_cmd_t ddr3_get_mr1(enum ddr3_mr1_qoff qoff, enum ddr3_mr1_tqds tqds, enum ddr3_mr1_rtt_nom rtt_nom, enum ddr3_mr1_write_leveling write_leveling, enum ddr3_mr1_ods output_drive_strenght, enum ddr3_mr1_additive_latency additive_latency, enum ddr3_mr1_dll dll_disable); enum ddr3_mr2_rttwr { DDR3_MR2_RTTWR_OFF = 0, DDR3_MR2_RTTWR_RZQ4 = 1, DDR3_MR2_RTTWR_RZQ2 = 2, }; enum ddr3_mr2_srt_range { DDR3_MR2_SRT_NORMAL = 0, DDR3_MR2_SRT_EXTENDED = 1, }; enum ddr3_mr2_asr { DDR3_MR2_ASR_MANUAL = 0, DDR3_MR2_ASR_AUTO = 1, }; mrs_cmd_t ddr3_get_mr2(enum ddr3_mr2_rttwr rtt_wr, enum ddr3_mr2_srt_range extended_temp, enum ddr3_mr2_asr self_refresh, u8 cas_cwl); mrs_cmd_t ddr3_get_mr3(char dataflow_from_mpr); mrs_cmd_t ddr3_mrs_mirror_pins(mrs_cmd_t cmd); #endif /* DEVICE_DRAM_DDR3_H */