blob: 0520ead4b7dcf878d2da70bd1cfccf83021f8374 [file] [log] [blame]
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -05001/*
2 * This file is part of the coreboot project.
3 *
4 * Copyright (C) 2013 Alexandru Gagniuc <mr.nuke.me@gmail.com>
5 *
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -050015 */
16
17#ifndef DEVICE_DRAM_DDR3L_H
18#define DEVICE_DRAM_DDR3L_H
19
20/**
21 * @file ddr3.h
22 *
23 * \brief Utilities for decoding DDR3 SPDs
24 */
25
26#include <stdint.h>
27#include <spd.h>
28
29/**
30 * \brief Convenience definitions for TCK values
31 *
32 * Different values for tCK, representing standard DDR3 frequencies.
33 * These values are in 1/256 ns units.
34 * @{
35 */
36#define TCK_1066MHZ 240
Vladimir Serbinenko7686a562014-05-18 11:05:56 +020037#define TCK_933MHZ 275
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -050038#define TCK_800MHZ 320
39#define TCK_666MHZ 384
40#define TCK_533MHZ 480
41#define TCK_400MHZ 640
42#define TCK_333MHZ 768
43#define TCK_266MHZ 960
44#define TCK_200MHZ 1280
45/** @} */
46
47/**
48 * \brief Convenience macro for enabling printk with CONFIG_DEBUG_RAM_SETUP
49 *
50 * Use this macro instead of printk(); for verbose RAM initialization messages.
51 * When CONFIG_DEBUG_RAM_SETUP is not selected, these messages are automatically
52 * disabled.
53 * @{
54 */
Martin Rothc4e49f62015-07-11 13:42:54 -060055#if IS_ENABLED(CONFIG_DEBUG_RAM_SETUP)
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -050056#define printram(x, ...) printk(BIOS_DEBUG, x, ##__VA_ARGS__)
57#else
58#define printram(x, ...)
59#endif
60/** @} */
61
62/*
63 * Module type (byte 3, bits 3:0) of SPD
Martin Roth0cb07e32013-07-09 21:46:01 -060064 * This definition is specific to DDR3. DDR2 SPDs have a different structure.
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -050065 */
66enum spd_dimm_type {
67 SPD_DIMM_TYPE_UNDEFINED = 0x00,
68 SPD_DIMM_TYPE_RDIMM = 0x01,
69 SPD_DIMM_TYPE_UDIMM = 0x02,
70 SPD_DIMM_TYPE_SO_DIMM = 0x03,
71 SPD_DIMM_TYPE_MICRO_DIMM = 0x04,
72 SPD_DIMM_TYPE_MINI_RDIMM = 0x05,
73 SPD_DIMM_TYPE_MINI_UDIMM = 0x06,
74 SPD_DIMM_TYPE_MINI_CDIMM = 0x07,
75 SPD_DIMM_TYPE_72B_SO_UDIMM = 0x08,
76 SPD_DIMM_TYPE_72B_SO_RDIMM = 0x09,
77 SPD_DIMM_TYPE_72B_SO_CDIMM = 0x0a,
78 SPD_DIMM_TYPE_LRDIMM = 0x0b,
79 SPD_DIMM_TYPE_16B_SO_DIMM = 0x0d,
80 SPD_DIMM_TYPE_32B_SO_DIMM = 0x0e,
81 /* Masks to bits 3:0 to give the dimm type */
82 SPD_DIMM_TYPE_MASK = 0x0f,
83};
84
85/**
86 * \brief DIMM flags
87 *
88 * Characteristic flags for the DIMM, as presented by the SPD
89 */
90typedef union dimm_flags_st {
91 /* The whole point of the union/struct construct is to allow us to clear
92 * all the bits with one line: flags.raw = 0.
93 * We do not care how these bits are ordered */
94 struct {
95 /* Indicates if rank 1 of DIMM uses a mirrored pin mapping. See:
96 * Annex K: Serial Presence Detect (SPD) for DDR3 SDRAM */
97 unsigned pins_mirrored:1;
98 /* Module can work at 1.50V - All DIMMS must be 1.5V operable */
99 unsigned operable_1_50V:1;
100 /* Module can work at 1.35V */
101 unsigned operable_1_35V:1;
102 /* Module can work at 1.20V */
103 unsigned operable_1_25V:1;
104 /* Has an 8-bit bus extension, meaning the DIMM supports ECC */
105 unsigned is_ecc:1;
106 /* DLL-Off Mode Support */
107 unsigned dll_off_mode:1;
108 /* Indicates a drive strength of RZQ/6 (40 Ohm) is supported */
109 unsigned rzq6_supported:1;
110 /* Indicates a drive strength of RZQ/7 (35 Ohm) is supported */
111 unsigned rzq7_supported:1;
112 /* Partial Array Self Refresh */
113 unsigned pasr:1;
114 /* On-die Thermal Sensor Readout */
115 unsigned odts:1;
116 /* Auto Self Refresh */
117 unsigned asr:1;
118 /* Extended temperature range supported */
119 unsigned ext_temp_range:1;
120 /* Operating at extended temperature requires 2X refresh rate */
121 unsigned ext_temp_refresh:1;
122 /* Thermal sensor incorporated */
123 unsigned therm_sensor:1;
124 };
125 unsigned raw;
126} dimm_flags_t;
127
128/**
129 * \brief DIMM characteristics
130 *
131 * The characteristics of each DIMM, as presented by the SPD
132 */
133typedef struct dimm_attr_st {
134 enum spd_memory_type dram_type;
Vladimir Serbinenko0e675f72014-12-07 13:56:48 +0100135 enum spd_dimm_type dimm_type;
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500136 u16 cas_supported;
137 /* Flags extracted from SPD */
138 dimm_flags_t flags;
Vladimir Serbinenko7686a562014-05-18 11:05:56 +0200139 /* SDRAM width */
140 u8 width;
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500141 /* Number of ranks */
142 u8 ranks;
143 /* Number or row address bits */
144 u8 row_bits;
145 /* Number or column address bits */
146 u8 col_bits;
147 /* Size of module in MiB */
148 u32 size_mb;
149 /* Latencies are in units of 1/256 ns */
150 u32 tCK;
151 u32 tAA;
152 u32 tWR;
153 u32 tRCD;
154 u32 tRRD;
155 u32 tRP;
156 u32 tRAS;
157 u32 tRC;
158 u32 tRFC;
159 u32 tWTR;
160 u32 tRTP;
161 u32 tFAW;
Vladimir Serbinenko7686a562014-05-18 11:05:56 +0200162
163 u8 reference_card;
Patrick Rudolphbd1fdc62016-01-26 08:45:21 +0100164 /* XMP: Module voltage in mV */
165 u16 voltage;
166 /* XMP: max DIMMs per channel supported (1-4) */
167 u8 dimms_per_channel;
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500168} dimm_attr;
169
170/** Result of the SPD decoding process */
171enum spd_status {
172 SPD_STATUS_OK = 0,
173 SPD_STATUS_INVALID,
174 SPD_STATUS_CRC_ERROR,
175 SPD_STATUS_INVALID_FIELD,
176};
177
Patrick Rudolphbd1fdc62016-01-26 08:45:21 +0100178enum ddr3_xmp_profile {
179 DDR3_XMP_PROFILE_1 = 0,
180 DDR3_XMP_PROFILE_2 = 1,
181};
182
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500183typedef u8 spd_raw_data[256];
184
Alexandru Gagniuc4c37e582013-12-17 13:08:01 -0500185u16 spd_ddr3_calc_crc(u8 *spd, int len);
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500186int spd_decode_ddr3(dimm_attr * dimm, spd_raw_data spd_data);
187int dimm_is_registered(enum spd_dimm_type type);
188void dram_print_spd_ddr3(const dimm_attr * dimm);
Patrick Rudolphbd1fdc62016-01-26 08:45:21 +0100189int spd_xmp_decode_ddr3(dimm_attr * dimm,
190 spd_raw_data spd,
191 enum ddr3_xmp_profile profile);
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500192
193/**
194 * \brief Read double word from specified address
195 *
196 * Should be useful when doing an MRS to the DIMM
197 */
Stefan Reinauer1e2500e2015-06-19 14:59:06 -0700198static inline u32 volatile_read(volatile uintptr_t addr)
Alexandru Gagniucf97ff3f2013-05-21 14:43:45 -0500199{
200 volatile u32 result;
201 result = *(volatile u32 *)addr;
202 return result;
203}
204
Alexandru Gagniuc78706fd2013-06-03 13:58:10 -0500205/**
206 * \brief Representation of an MRS command
207 *
208 * This represents an MRS command as seen by the DIMM. This is not a memory
209 * address that can be read to generate an MRS command. The mapping of CPU
210 * to memory pins is hardware-dependent.
211 * \n
212 * The idea is to generalize the MRS code, and only need a hardware-specific
213 * function to map the MRS bits to CPU address bits. An MRS command can be
214 * sent like:
215 * @code{.c}
216 * u32 addr;
217 * mrs_cmd_t mrs;
218 * chipset_enable_mrs_command_mode();
219 * mrs = ddr3_get_mr2(rtt_wr, srt, asr, cwl)
220 * if (rank_has_mirrorred_pins)
221 * mrs = ddr3_mrs_mirror_pins(mrs);
222 * addr = chipset_specific_get_mrs_addr(mrs);
223 * volatile_read(addr);
224 * @endcode
225 *
226 * The MRS representation has the following structure:
227 * - cmd[15:0] = Address pins MA[15:0]
228 * - cmd[18:16] = Bank address BA[2:0]
229 */
230typedef u32 mrs_cmd_t;
231
232enum ddr3_mr0_precharge {
233 DDR3_MR0_PRECHARGE_SLOW = 0,
234 DDR3_MR0_PRECHARGE_FAST = 1,
235};
236enum ddr3_mr0_mode {
237 DDR3_MR0_MODE_NORMAL = 0,
238 DDR3_MR0_MODE_TEST = 1,
239};
240enum ddr3_mr0_dll_reset {
241 DDR3_MR0_DLL_RESET_NO = 0,
242 DDR3_MR0_DLL_RESET_YES = 1,
243};
244enum ddr3_mr0_burst_type {
245 DDR3_MR0_BURST_TYPE_SEQUENTIAL = 0,
246 DDR3_MR0_BURST_TYPE_INTERLEAVED = 1,
247};
248enum ddr3_mr0_burst_length {
249 DDR3_MR0_BURST_LENGTH_8 = 0,
250 DDR3_MR0_BURST_LENGTH_CHOP = 1,
251 DDR3_MR0_BURST_LENGTH_4 = 2,
252};
253mrs_cmd_t ddr3_get_mr0(enum ddr3_mr0_precharge precharge_pd,
254 u8 write_recovery,
255 enum ddr3_mr0_dll_reset dll_reset,
256 enum ddr3_mr0_mode mode,
257 u8 cas,
258 enum ddr3_mr0_burst_type interleaved_burst,
259 enum ddr3_mr0_burst_length burst_length);
260
261enum ddr3_mr1_qoff {
262 DDR3_MR1_QOFF_ENABLE = 0,
263 DDR3_MR1_QOFF_DISABLE = 1,
264};
265enum ddr3_mr1_tqds {
266 DDR3_MR1_TQDS_DISABLE = 0,
267 DDR3_MR1_TQDS_ENABLE = 1,
268};
269enum ddr3_mr1_write_leveling {
270 DDR3_MR1_WRLVL_DISABLE = 0,
271 DDR3_MR1_WRLVL_ENABLE = 1,
272};
273enum ddr3_mr1_rtt_nom {
274 DDR3_MR1_RTT_NOM_OFF = 0,
275 DDR3_MR1_RTT_NOM_RZQ4 = 1,
276 DDR3_MR1_RTT_NOM_RZQ2 = 2,
277 DDR3_MR1_RTT_NOM_RZQ6 = 3,
278 DDR3_MR1_RTT_NOM_RZQ12 = 4,
279 DDR3_MR1_RTT_NOM_RZQ8 = 5,
280};
281enum ddr3_mr1_additive_latency {
282 DDR3_MR1_AL_DISABLE = 0,
283 DDR3_MR1_AL_CL_MINUS_1 = 1,
284 DDR3_MR1_AL_CL_MINUS_2 = 2,
285};
286enum ddr3_mr1_ods {
287 DDR3_MR1_ODS_RZQ6 = 0,
288 DDR3_MR1_ODS_RZQ7 = 1,
289};
290enum ddr3_mr1_dll {
291 DDR3_MR1_DLL_ENABLE = 0,
292 DDR3_MR1_DLL_DISABLE = 1,
293};
294
295mrs_cmd_t ddr3_get_mr1(enum ddr3_mr1_qoff qoff,
296 enum ddr3_mr1_tqds tqds,
297 enum ddr3_mr1_rtt_nom rtt_nom,
298 enum ddr3_mr1_write_leveling write_leveling,
299 enum ddr3_mr1_ods output_drive_strenght,
300 enum ddr3_mr1_additive_latency additive_latency,
301 enum ddr3_mr1_dll dll_disable);
302
303enum ddr3_mr2_rttwr {
304 DDR3_MR2_RTTWR_OFF = 0,
305 DDR3_MR2_RTTWR_RZQ4 = 1,
306 DDR3_MR2_RTTWR_RZQ2 = 2,
307};
308enum ddr3_mr2_srt_range {
309 DDR3_MR2_SRT_NORMAL = 0,
310 DDR3_MR2_SRT_EXTENDED = 1,
311};
312enum ddr3_mr2_asr {
313 DDR3_MR2_ASR_MANUAL = 0,
314 DDR3_MR2_ASR_AUTO = 1,
315};
316
317mrs_cmd_t ddr3_get_mr2(enum ddr3_mr2_rttwr rtt_wr,
318 enum ddr3_mr2_srt_range extended_temp,
319 enum ddr3_mr2_asr self_refresh, u8 cas_cwl);
320
321mrs_cmd_t ddr3_get_mr3(char dataflow_from_mpr);
322mrs_cmd_t ddr3_mrs_mirror_pins(mrs_cmd_t cmd);
323
Martin Rothfd277d82016-01-11 12:47:30 -0700324#endif /* DEVICE_DRAM_DDR3L_H */