1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Copyright (c) 2014-2020, NVIDIA CORPORATION. All rights reserved.
4*4882a593Smuzhiyun */
5*4882a593Smuzhiyun
6*4882a593Smuzhiyun #include <linux/kernel.h>
7*4882a593Smuzhiyun #include <linux/io.h>
8*4882a593Smuzhiyun #include <linux/clk.h>
9*4882a593Smuzhiyun #include <linux/delay.h>
10*4882a593Smuzhiyun #include <linux/of.h>
11*4882a593Smuzhiyun
12*4882a593Smuzhiyun #include <soc/tegra/mc.h>
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun #include "tegra210-emc.h"
15*4882a593Smuzhiyun #include "tegra210-mc.h"
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun /*
18*4882a593Smuzhiyun * Enable flags for specifying verbosity.
19*4882a593Smuzhiyun */
20*4882a593Smuzhiyun #define INFO (1 << 0)
21*4882a593Smuzhiyun #define STEPS (1 << 1)
22*4882a593Smuzhiyun #define SUB_STEPS (1 << 2)
23*4882a593Smuzhiyun #define PRELOCK (1 << 3)
24*4882a593Smuzhiyun #define PRELOCK_STEPS (1 << 4)
25*4882a593Smuzhiyun #define ACTIVE_EN (1 << 5)
26*4882a593Smuzhiyun #define PRAMP_UP (1 << 6)
27*4882a593Smuzhiyun #define PRAMP_DN (1 << 7)
28*4882a593Smuzhiyun #define EMA_WRITES (1 << 10)
29*4882a593Smuzhiyun #define EMA_UPDATES (1 << 11)
30*4882a593Smuzhiyun #define PER_TRAIN (1 << 16)
31*4882a593Smuzhiyun #define CC_PRINT (1 << 17)
32*4882a593Smuzhiyun #define CCFIFO (1 << 29)
33*4882a593Smuzhiyun #define REGS (1 << 30)
34*4882a593Smuzhiyun #define REG_LISTS (1 << 31)
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun #define emc_dbg(emc, flags, ...) dev_dbg(emc->dev, __VA_ARGS__)
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun #define DVFS_CLOCK_CHANGE_VERSION 21021
39*4882a593Smuzhiyun #define EMC_PRELOCK_VERSION 2101
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun enum {
42*4882a593Smuzhiyun DVFS_SEQUENCE = 1,
43*4882a593Smuzhiyun WRITE_TRAINING_SEQUENCE = 2,
44*4882a593Smuzhiyun PERIODIC_TRAINING_SEQUENCE = 3,
45*4882a593Smuzhiyun DVFS_PT1 = 10,
46*4882a593Smuzhiyun DVFS_UPDATE = 11,
47*4882a593Smuzhiyun TRAINING_PT1 = 12,
48*4882a593Smuzhiyun TRAINING_UPDATE = 13,
49*4882a593Smuzhiyun PERIODIC_TRAINING_UPDATE = 14
50*4882a593Smuzhiyun };
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun /*
53*4882a593Smuzhiyun * PTFV defines - basically just indexes into the per table PTFV array.
54*4882a593Smuzhiyun */
55*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C0D0U0_INDEX 0
56*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C0D0U1_INDEX 1
57*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C0D1U0_INDEX 2
58*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C0D1U1_INDEX 3
59*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C1D0U0_INDEX 4
60*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C1D0U1_INDEX 5
61*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C1D1U0_INDEX 6
62*4882a593Smuzhiyun #define PTFV_DQSOSC_MOVAVG_C1D1U1_INDEX 7
63*4882a593Smuzhiyun #define PTFV_DVFS_SAMPLES_INDEX 9
64*4882a593Smuzhiyun #define PTFV_MOVAVG_WEIGHT_INDEX 10
65*4882a593Smuzhiyun #define PTFV_CONFIG_CTRL_INDEX 11
66*4882a593Smuzhiyun
67*4882a593Smuzhiyun #define PTFV_CONFIG_CTRL_USE_PREVIOUS_EMA (1 << 0)
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun /*
70*4882a593Smuzhiyun * Do arithmetic in fixed point.
71*4882a593Smuzhiyun */
72*4882a593Smuzhiyun #define MOVAVG_PRECISION_FACTOR 100
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun /*
75*4882a593Smuzhiyun * The division portion of the average operation.
76*4882a593Smuzhiyun */
77*4882a593Smuzhiyun #define __AVERAGE_PTFV(dev) \
78*4882a593Smuzhiyun ({ next->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] = \
79*4882a593Smuzhiyun next->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] / \
80*4882a593Smuzhiyun next->ptfv_list[PTFV_DVFS_SAMPLES_INDEX]; })
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun /*
83*4882a593Smuzhiyun * Convert val to fixed point and add it to the temporary average.
84*4882a593Smuzhiyun */
85*4882a593Smuzhiyun #define __INCREMENT_PTFV(dev, val) \
86*4882a593Smuzhiyun ({ next->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] += \
87*4882a593Smuzhiyun ((val) * MOVAVG_PRECISION_FACTOR); })
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun /*
90*4882a593Smuzhiyun * Convert a moving average back to integral form and return the value.
91*4882a593Smuzhiyun */
92*4882a593Smuzhiyun #define __MOVAVG_AC(timing, dev) \
93*4882a593Smuzhiyun ((timing)->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] / \
94*4882a593Smuzhiyun MOVAVG_PRECISION_FACTOR)
95*4882a593Smuzhiyun
96*4882a593Smuzhiyun /* Weighted update. */
97*4882a593Smuzhiyun #define __WEIGHTED_UPDATE_PTFV(dev, nval) \
98*4882a593Smuzhiyun do { \
99*4882a593Smuzhiyun int w = PTFV_MOVAVG_WEIGHT_INDEX; \
100*4882a593Smuzhiyun int dqs = PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX; \
101*4882a593Smuzhiyun \
102*4882a593Smuzhiyun next->ptfv_list[dqs] = \
103*4882a593Smuzhiyun ((nval * MOVAVG_PRECISION_FACTOR) + \
104*4882a593Smuzhiyun (next->ptfv_list[dqs] * \
105*4882a593Smuzhiyun next->ptfv_list[w])) / \
106*4882a593Smuzhiyun (next->ptfv_list[w] + 1); \
107*4882a593Smuzhiyun \
108*4882a593Smuzhiyun emc_dbg(emc, EMA_UPDATES, "%s: (s=%lu) EMA: %u\n", \
109*4882a593Smuzhiyun __stringify(dev), nval, next->ptfv_list[dqs]); \
110*4882a593Smuzhiyun } while (0)
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun /* Access a particular average. */
113*4882a593Smuzhiyun #define __MOVAVG(timing, dev) \
114*4882a593Smuzhiyun ((timing)->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX])
115*4882a593Smuzhiyun
update_clock_tree_delay(struct tegra210_emc * emc,int type)116*4882a593Smuzhiyun static u32 update_clock_tree_delay(struct tegra210_emc *emc, int type)
117*4882a593Smuzhiyun {
118*4882a593Smuzhiyun bool periodic_training_update = type == PERIODIC_TRAINING_UPDATE;
119*4882a593Smuzhiyun struct tegra210_emc_timing *last = emc->last;
120*4882a593Smuzhiyun struct tegra210_emc_timing *next = emc->next;
121*4882a593Smuzhiyun u32 last_timing_rate_mhz = last->rate / 1000;
122*4882a593Smuzhiyun u32 next_timing_rate_mhz = next->rate / 1000;
123*4882a593Smuzhiyun bool dvfs_update = type == DVFS_UPDATE;
124*4882a593Smuzhiyun s32 tdel = 0, tmdel = 0, adel = 0;
125*4882a593Smuzhiyun bool dvfs_pt1 = type == DVFS_PT1;
126*4882a593Smuzhiyun unsigned long cval = 0;
127*4882a593Smuzhiyun u32 temp[2][2], value;
128*4882a593Smuzhiyun unsigned int i;
129*4882a593Smuzhiyun
130*4882a593Smuzhiyun /*
131*4882a593Smuzhiyun * Dev0 MSB.
132*4882a593Smuzhiyun */
133*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
134*4882a593Smuzhiyun value = tegra210_emc_mrr_read(emc, 2, 19);
135*4882a593Smuzhiyun
136*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++) {
137*4882a593Smuzhiyun temp[i][0] = (value & 0x00ff) << 8;
138*4882a593Smuzhiyun temp[i][1] = (value & 0xff00) << 0;
139*4882a593Smuzhiyun value >>= 16;
140*4882a593Smuzhiyun }
141*4882a593Smuzhiyun
142*4882a593Smuzhiyun /*
143*4882a593Smuzhiyun * Dev0 LSB.
144*4882a593Smuzhiyun */
145*4882a593Smuzhiyun value = tegra210_emc_mrr_read(emc, 2, 18);
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++) {
148*4882a593Smuzhiyun temp[i][0] |= (value & 0x00ff) >> 0;
149*4882a593Smuzhiyun temp[i][1] |= (value & 0xff00) >> 8;
150*4882a593Smuzhiyun value >>= 16;
151*4882a593Smuzhiyun }
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun
154*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
155*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
156*4882a593Smuzhiyun cval *= 1000000;
157*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[0][0];
158*4882a593Smuzhiyun }
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun if (dvfs_pt1)
161*4882a593Smuzhiyun __INCREMENT_PTFV(C0D0U0, cval);
162*4882a593Smuzhiyun else if (dvfs_update)
163*4882a593Smuzhiyun __AVERAGE_PTFV(C0D0U0);
164*4882a593Smuzhiyun else if (periodic_training_update)
165*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C0D0U0, cval);
166*4882a593Smuzhiyun
167*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
168*4882a593Smuzhiyun tdel = next->current_dram_clktree[C0D0U0] -
169*4882a593Smuzhiyun __MOVAVG_AC(next, C0D0U0);
170*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
171*4882a593Smuzhiyun adel = tmdel;
172*4882a593Smuzhiyun
173*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
174*4882a593Smuzhiyun next->tree_margin)
175*4882a593Smuzhiyun next->current_dram_clktree[C0D0U0] =
176*4882a593Smuzhiyun __MOVAVG_AC(next, C0D0U0);
177*4882a593Smuzhiyun }
178*4882a593Smuzhiyun
179*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
180*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
181*4882a593Smuzhiyun cval *= 1000000;
182*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[0][1];
183*4882a593Smuzhiyun }
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun if (dvfs_pt1)
186*4882a593Smuzhiyun __INCREMENT_PTFV(C0D0U1, cval);
187*4882a593Smuzhiyun else if (dvfs_update)
188*4882a593Smuzhiyun __AVERAGE_PTFV(C0D0U1);
189*4882a593Smuzhiyun else if (periodic_training_update)
190*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C0D0U1, cval);
191*4882a593Smuzhiyun
192*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
193*4882a593Smuzhiyun tdel = next->current_dram_clktree[C0D0U1] -
194*4882a593Smuzhiyun __MOVAVG_AC(next, C0D0U1);
195*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun if (tmdel > adel)
198*4882a593Smuzhiyun adel = tmdel;
199*4882a593Smuzhiyun
200*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
201*4882a593Smuzhiyun next->tree_margin)
202*4882a593Smuzhiyun next->current_dram_clktree[C0D0U1] =
203*4882a593Smuzhiyun __MOVAVG_AC(next, C0D0U1);
204*4882a593Smuzhiyun }
205*4882a593Smuzhiyun
206*4882a593Smuzhiyun if (emc->num_channels > 1) {
207*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
208*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
209*4882a593Smuzhiyun cval *= 1000000;
210*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[1][0];
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun if (dvfs_pt1)
214*4882a593Smuzhiyun __INCREMENT_PTFV(C1D0U0, cval);
215*4882a593Smuzhiyun else if (dvfs_update)
216*4882a593Smuzhiyun __AVERAGE_PTFV(C1D0U0);
217*4882a593Smuzhiyun else if (periodic_training_update)
218*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C1D0U0, cval);
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
221*4882a593Smuzhiyun tdel = next->current_dram_clktree[C1D0U0] -
222*4882a593Smuzhiyun __MOVAVG_AC(next, C1D0U0);
223*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun if (tmdel > adel)
226*4882a593Smuzhiyun adel = tmdel;
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
229*4882a593Smuzhiyun next->tree_margin)
230*4882a593Smuzhiyun next->current_dram_clktree[C1D0U0] =
231*4882a593Smuzhiyun __MOVAVG_AC(next, C1D0U0);
232*4882a593Smuzhiyun }
233*4882a593Smuzhiyun
234*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
235*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
236*4882a593Smuzhiyun cval *= 1000000;
237*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[1][1];
238*4882a593Smuzhiyun }
239*4882a593Smuzhiyun
240*4882a593Smuzhiyun if (dvfs_pt1)
241*4882a593Smuzhiyun __INCREMENT_PTFV(C1D0U1, cval);
242*4882a593Smuzhiyun else if (dvfs_update)
243*4882a593Smuzhiyun __AVERAGE_PTFV(C1D0U1);
244*4882a593Smuzhiyun else if (periodic_training_update)
245*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C1D0U1, cval);
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
248*4882a593Smuzhiyun tdel = next->current_dram_clktree[C1D0U1] -
249*4882a593Smuzhiyun __MOVAVG_AC(next, C1D0U1);
250*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun if (tmdel > adel)
253*4882a593Smuzhiyun adel = tmdel;
254*4882a593Smuzhiyun
255*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
256*4882a593Smuzhiyun next->tree_margin)
257*4882a593Smuzhiyun next->current_dram_clktree[C1D0U1] =
258*4882a593Smuzhiyun __MOVAVG_AC(next, C1D0U1);
259*4882a593Smuzhiyun }
260*4882a593Smuzhiyun }
261*4882a593Smuzhiyun
262*4882a593Smuzhiyun if (emc->num_devices < 2)
263*4882a593Smuzhiyun goto done;
264*4882a593Smuzhiyun
265*4882a593Smuzhiyun /*
266*4882a593Smuzhiyun * Dev1 MSB.
267*4882a593Smuzhiyun */
268*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
269*4882a593Smuzhiyun value = tegra210_emc_mrr_read(emc, 1, 19);
270*4882a593Smuzhiyun
271*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++) {
272*4882a593Smuzhiyun temp[i][0] = (value & 0x00ff) << 8;
273*4882a593Smuzhiyun temp[i][1] = (value & 0xff00) << 0;
274*4882a593Smuzhiyun value >>= 16;
275*4882a593Smuzhiyun }
276*4882a593Smuzhiyun
277*4882a593Smuzhiyun /*
278*4882a593Smuzhiyun * Dev1 LSB.
279*4882a593Smuzhiyun */
280*4882a593Smuzhiyun value = tegra210_emc_mrr_read(emc, 2, 18);
281*4882a593Smuzhiyun
282*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++) {
283*4882a593Smuzhiyun temp[i][0] |= (value & 0x00ff) >> 0;
284*4882a593Smuzhiyun temp[i][1] |= (value & 0xff00) >> 8;
285*4882a593Smuzhiyun value >>= 16;
286*4882a593Smuzhiyun }
287*4882a593Smuzhiyun }
288*4882a593Smuzhiyun
289*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
290*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
291*4882a593Smuzhiyun cval *= 1000000;
292*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[0][0];
293*4882a593Smuzhiyun }
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun if (dvfs_pt1)
296*4882a593Smuzhiyun __INCREMENT_PTFV(C0D1U0, cval);
297*4882a593Smuzhiyun else if (dvfs_update)
298*4882a593Smuzhiyun __AVERAGE_PTFV(C0D1U0);
299*4882a593Smuzhiyun else if (periodic_training_update)
300*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C0D1U0, cval);
301*4882a593Smuzhiyun
302*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
303*4882a593Smuzhiyun tdel = next->current_dram_clktree[C0D1U0] -
304*4882a593Smuzhiyun __MOVAVG_AC(next, C0D1U0);
305*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
306*4882a593Smuzhiyun
307*4882a593Smuzhiyun if (tmdel > adel)
308*4882a593Smuzhiyun adel = tmdel;
309*4882a593Smuzhiyun
310*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
311*4882a593Smuzhiyun next->tree_margin)
312*4882a593Smuzhiyun next->current_dram_clktree[C0D1U0] =
313*4882a593Smuzhiyun __MOVAVG_AC(next, C0D1U0);
314*4882a593Smuzhiyun }
315*4882a593Smuzhiyun
316*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
317*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
318*4882a593Smuzhiyun cval *= 1000000;
319*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[0][1];
320*4882a593Smuzhiyun }
321*4882a593Smuzhiyun
322*4882a593Smuzhiyun if (dvfs_pt1)
323*4882a593Smuzhiyun __INCREMENT_PTFV(C0D1U1, cval);
324*4882a593Smuzhiyun else if (dvfs_update)
325*4882a593Smuzhiyun __AVERAGE_PTFV(C0D1U1);
326*4882a593Smuzhiyun else if (periodic_training_update)
327*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C0D1U1, cval);
328*4882a593Smuzhiyun
329*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
330*4882a593Smuzhiyun tdel = next->current_dram_clktree[C0D1U1] -
331*4882a593Smuzhiyun __MOVAVG_AC(next, C0D1U1);
332*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
333*4882a593Smuzhiyun
334*4882a593Smuzhiyun if (tmdel > adel)
335*4882a593Smuzhiyun adel = tmdel;
336*4882a593Smuzhiyun
337*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
338*4882a593Smuzhiyun next->tree_margin)
339*4882a593Smuzhiyun next->current_dram_clktree[C0D1U1] =
340*4882a593Smuzhiyun __MOVAVG_AC(next, C0D1U1);
341*4882a593Smuzhiyun }
342*4882a593Smuzhiyun
343*4882a593Smuzhiyun if (emc->num_channels > 1) {
344*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
345*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
346*4882a593Smuzhiyun cval *= 1000000;
347*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[1][0];
348*4882a593Smuzhiyun }
349*4882a593Smuzhiyun
350*4882a593Smuzhiyun if (dvfs_pt1)
351*4882a593Smuzhiyun __INCREMENT_PTFV(C1D1U0, cval);
352*4882a593Smuzhiyun else if (dvfs_update)
353*4882a593Smuzhiyun __AVERAGE_PTFV(C1D1U0);
354*4882a593Smuzhiyun else if (periodic_training_update)
355*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C1D1U0, cval);
356*4882a593Smuzhiyun
357*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
358*4882a593Smuzhiyun tdel = next->current_dram_clktree[C1D1U0] -
359*4882a593Smuzhiyun __MOVAVG_AC(next, C1D1U0);
360*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
361*4882a593Smuzhiyun
362*4882a593Smuzhiyun if (tmdel > adel)
363*4882a593Smuzhiyun adel = tmdel;
364*4882a593Smuzhiyun
365*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
366*4882a593Smuzhiyun next->tree_margin)
367*4882a593Smuzhiyun next->current_dram_clktree[C1D1U0] =
368*4882a593Smuzhiyun __MOVAVG_AC(next, C1D1U0);
369*4882a593Smuzhiyun }
370*4882a593Smuzhiyun
371*4882a593Smuzhiyun if (dvfs_pt1 || periodic_training_update) {
372*4882a593Smuzhiyun cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
373*4882a593Smuzhiyun cval *= 1000000;
374*4882a593Smuzhiyun cval /= last_timing_rate_mhz * 2 * temp[1][1];
375*4882a593Smuzhiyun }
376*4882a593Smuzhiyun
377*4882a593Smuzhiyun if (dvfs_pt1)
378*4882a593Smuzhiyun __INCREMENT_PTFV(C1D1U1, cval);
379*4882a593Smuzhiyun else if (dvfs_update)
380*4882a593Smuzhiyun __AVERAGE_PTFV(C1D1U1);
381*4882a593Smuzhiyun else if (periodic_training_update)
382*4882a593Smuzhiyun __WEIGHTED_UPDATE_PTFV(C1D1U1, cval);
383*4882a593Smuzhiyun
384*4882a593Smuzhiyun if (dvfs_update || periodic_training_update) {
385*4882a593Smuzhiyun tdel = next->current_dram_clktree[C1D1U1] -
386*4882a593Smuzhiyun __MOVAVG_AC(next, C1D1U1);
387*4882a593Smuzhiyun tmdel = (tdel < 0) ? -1 * tdel : tdel;
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun if (tmdel > adel)
390*4882a593Smuzhiyun adel = tmdel;
391*4882a593Smuzhiyun
392*4882a593Smuzhiyun if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
393*4882a593Smuzhiyun next->tree_margin)
394*4882a593Smuzhiyun next->current_dram_clktree[C1D1U1] =
395*4882a593Smuzhiyun __MOVAVG_AC(next, C1D1U1);
396*4882a593Smuzhiyun }
397*4882a593Smuzhiyun }
398*4882a593Smuzhiyun
399*4882a593Smuzhiyun done:
400*4882a593Smuzhiyun return adel;
401*4882a593Smuzhiyun }
402*4882a593Smuzhiyun
periodic_compensation_handler(struct tegra210_emc * emc,u32 type,struct tegra210_emc_timing * last,struct tegra210_emc_timing * next)403*4882a593Smuzhiyun static u32 periodic_compensation_handler(struct tegra210_emc *emc, u32 type,
404*4882a593Smuzhiyun struct tegra210_emc_timing *last,
405*4882a593Smuzhiyun struct tegra210_emc_timing *next)
406*4882a593Smuzhiyun {
407*4882a593Smuzhiyun #define __COPY_EMA(nt, lt, dev) \
408*4882a593Smuzhiyun ({ __MOVAVG(nt, dev) = __MOVAVG(lt, dev) * \
409*4882a593Smuzhiyun (nt)->ptfv_list[PTFV_DVFS_SAMPLES_INDEX]; })
410*4882a593Smuzhiyun
411*4882a593Smuzhiyun u32 i, adel = 0, samples = next->ptfv_list[PTFV_DVFS_SAMPLES_INDEX];
412*4882a593Smuzhiyun u32 delay;
413*4882a593Smuzhiyun
414*4882a593Smuzhiyun delay = tegra210_emc_actual_osc_clocks(last->run_clocks);
415*4882a593Smuzhiyun delay *= 1000;
416*4882a593Smuzhiyun delay = 2 + (delay / last->rate);
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun if (!next->periodic_training)
419*4882a593Smuzhiyun return 0;
420*4882a593Smuzhiyun
421*4882a593Smuzhiyun if (type == DVFS_SEQUENCE) {
422*4882a593Smuzhiyun if (last->periodic_training &&
423*4882a593Smuzhiyun (next->ptfv_list[PTFV_CONFIG_CTRL_INDEX] &
424*4882a593Smuzhiyun PTFV_CONFIG_CTRL_USE_PREVIOUS_EMA)) {
425*4882a593Smuzhiyun /*
426*4882a593Smuzhiyun * If the previous frequency was using periodic
427*4882a593Smuzhiyun * calibration then we can reuse the previous
428*4882a593Smuzhiyun * frequencies EMA data.
429*4882a593Smuzhiyun */
430*4882a593Smuzhiyun __COPY_EMA(next, last, C0D0U0);
431*4882a593Smuzhiyun __COPY_EMA(next, last, C0D0U1);
432*4882a593Smuzhiyun __COPY_EMA(next, last, C1D0U0);
433*4882a593Smuzhiyun __COPY_EMA(next, last, C1D0U1);
434*4882a593Smuzhiyun __COPY_EMA(next, last, C0D1U0);
435*4882a593Smuzhiyun __COPY_EMA(next, last, C0D1U1);
436*4882a593Smuzhiyun __COPY_EMA(next, last, C1D1U0);
437*4882a593Smuzhiyun __COPY_EMA(next, last, C1D1U1);
438*4882a593Smuzhiyun } else {
439*4882a593Smuzhiyun /* Reset the EMA.*/
440*4882a593Smuzhiyun __MOVAVG(next, C0D0U0) = 0;
441*4882a593Smuzhiyun __MOVAVG(next, C0D0U1) = 0;
442*4882a593Smuzhiyun __MOVAVG(next, C1D0U0) = 0;
443*4882a593Smuzhiyun __MOVAVG(next, C1D0U1) = 0;
444*4882a593Smuzhiyun __MOVAVG(next, C0D1U0) = 0;
445*4882a593Smuzhiyun __MOVAVG(next, C0D1U1) = 0;
446*4882a593Smuzhiyun __MOVAVG(next, C1D1U0) = 0;
447*4882a593Smuzhiyun __MOVAVG(next, C1D1U1) = 0;
448*4882a593Smuzhiyun
449*4882a593Smuzhiyun for (i = 0; i < samples; i++) {
450*4882a593Smuzhiyun tegra210_emc_start_periodic_compensation(emc);
451*4882a593Smuzhiyun udelay(delay);
452*4882a593Smuzhiyun
453*4882a593Smuzhiyun /*
454*4882a593Smuzhiyun * Generate next sample of data.
455*4882a593Smuzhiyun */
456*4882a593Smuzhiyun adel = update_clock_tree_delay(emc, DVFS_PT1);
457*4882a593Smuzhiyun }
458*4882a593Smuzhiyun }
459*4882a593Smuzhiyun
460*4882a593Smuzhiyun /*
461*4882a593Smuzhiyun * Seems like it should be part of the
462*4882a593Smuzhiyun * 'if (last_timing->periodic_training)' conditional
463*4882a593Smuzhiyun * since is already done for the else clause.
464*4882a593Smuzhiyun */
465*4882a593Smuzhiyun adel = update_clock_tree_delay(emc, DVFS_UPDATE);
466*4882a593Smuzhiyun }
467*4882a593Smuzhiyun
468*4882a593Smuzhiyun if (type == PERIODIC_TRAINING_SEQUENCE) {
469*4882a593Smuzhiyun tegra210_emc_start_periodic_compensation(emc);
470*4882a593Smuzhiyun udelay(delay);
471*4882a593Smuzhiyun
472*4882a593Smuzhiyun adel = update_clock_tree_delay(emc, PERIODIC_TRAINING_UPDATE);
473*4882a593Smuzhiyun }
474*4882a593Smuzhiyun
475*4882a593Smuzhiyun return adel;
476*4882a593Smuzhiyun }
477*4882a593Smuzhiyun
tegra210_emc_r21021_periodic_compensation(struct tegra210_emc * emc)478*4882a593Smuzhiyun static u32 tegra210_emc_r21021_periodic_compensation(struct tegra210_emc *emc)
479*4882a593Smuzhiyun {
480*4882a593Smuzhiyun u32 emc_cfg, emc_cfg_o, emc_cfg_update, del, value;
481*4882a593Smuzhiyun u32 list[] = {
482*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_0,
483*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_1,
484*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_2,
485*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_3,
486*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_0,
487*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_1,
488*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_2,
489*4882a593Smuzhiyun EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_3,
490*4882a593Smuzhiyun EMC_DATA_BRLSHFT_0,
491*4882a593Smuzhiyun EMC_DATA_BRLSHFT_1
492*4882a593Smuzhiyun };
493*4882a593Smuzhiyun struct tegra210_emc_timing *last = emc->last;
494*4882a593Smuzhiyun unsigned int items = ARRAY_SIZE(list), i;
495*4882a593Smuzhiyun unsigned long delay;
496*4882a593Smuzhiyun
497*4882a593Smuzhiyun if (last->periodic_training) {
498*4882a593Smuzhiyun emc_dbg(emc, PER_TRAIN, "Periodic training starting\n");
499*4882a593Smuzhiyun
500*4882a593Smuzhiyun value = emc_readl(emc, EMC_DBG);
501*4882a593Smuzhiyun emc_cfg_o = emc_readl(emc, EMC_CFG);
502*4882a593Smuzhiyun emc_cfg = emc_cfg_o & ~(EMC_CFG_DYN_SELF_REF |
503*4882a593Smuzhiyun EMC_CFG_DRAM_ACPD |
504*4882a593Smuzhiyun EMC_CFG_DRAM_CLKSTOP_PD);
505*4882a593Smuzhiyun
506*4882a593Smuzhiyun
507*4882a593Smuzhiyun /*
508*4882a593Smuzhiyun * 1. Power optimizations should be off.
509*4882a593Smuzhiyun */
510*4882a593Smuzhiyun emc_writel(emc, emc_cfg, EMC_CFG);
511*4882a593Smuzhiyun
512*4882a593Smuzhiyun /* Does emc_timing_update() for above changes. */
513*4882a593Smuzhiyun tegra210_emc_dll_disable(emc);
514*4882a593Smuzhiyun
515*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++)
516*4882a593Smuzhiyun tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
517*4882a593Smuzhiyun EMC_EMC_STATUS_DRAM_IN_POWERDOWN_MASK,
518*4882a593Smuzhiyun 0);
519*4882a593Smuzhiyun
520*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++)
521*4882a593Smuzhiyun tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
522*4882a593Smuzhiyun EMC_EMC_STATUS_DRAM_IN_SELF_REFRESH_MASK,
523*4882a593Smuzhiyun 0);
524*4882a593Smuzhiyun
525*4882a593Smuzhiyun emc_cfg_update = value = emc_readl(emc, EMC_CFG_UPDATE);
526*4882a593Smuzhiyun value &= ~EMC_CFG_UPDATE_UPDATE_DLL_IN_UPDATE_MASK;
527*4882a593Smuzhiyun value |= (2 << EMC_CFG_UPDATE_UPDATE_DLL_IN_UPDATE_SHIFT);
528*4882a593Smuzhiyun emc_writel(emc, value, EMC_CFG_UPDATE);
529*4882a593Smuzhiyun
530*4882a593Smuzhiyun /*
531*4882a593Smuzhiyun * 2. osc kick off - this assumes training and dvfs have set
532*4882a593Smuzhiyun * correct MR23.
533*4882a593Smuzhiyun */
534*4882a593Smuzhiyun tegra210_emc_start_periodic_compensation(emc);
535*4882a593Smuzhiyun
536*4882a593Smuzhiyun /*
537*4882a593Smuzhiyun * 3. Let dram capture its clock tree delays.
538*4882a593Smuzhiyun */
539*4882a593Smuzhiyun delay = tegra210_emc_actual_osc_clocks(last->run_clocks);
540*4882a593Smuzhiyun delay *= 1000;
541*4882a593Smuzhiyun delay /= last->rate + 1;
542*4882a593Smuzhiyun udelay(delay);
543*4882a593Smuzhiyun
544*4882a593Smuzhiyun /*
545*4882a593Smuzhiyun * 4. Check delta wrt previous values (save value if margin
546*4882a593Smuzhiyun * exceeds what is set in table).
547*4882a593Smuzhiyun */
548*4882a593Smuzhiyun del = periodic_compensation_handler(emc,
549*4882a593Smuzhiyun PERIODIC_TRAINING_SEQUENCE,
550*4882a593Smuzhiyun last, last);
551*4882a593Smuzhiyun
552*4882a593Smuzhiyun /*
553*4882a593Smuzhiyun * 5. Apply compensation w.r.t. trained values (if clock tree
554*4882a593Smuzhiyun * has drifted more than the set margin).
555*4882a593Smuzhiyun */
556*4882a593Smuzhiyun if (last->tree_margin < ((del * 128 * (last->rate / 1000)) / 1000000)) {
557*4882a593Smuzhiyun for (i = 0; i < items; i++) {
558*4882a593Smuzhiyun value = tegra210_emc_compensate(last, list[i]);
559*4882a593Smuzhiyun emc_dbg(emc, EMA_WRITES, "0x%08x <= 0x%08x\n",
560*4882a593Smuzhiyun list[i], value);
561*4882a593Smuzhiyun emc_writel(emc, value, list[i]);
562*4882a593Smuzhiyun }
563*4882a593Smuzhiyun }
564*4882a593Smuzhiyun
565*4882a593Smuzhiyun emc_writel(emc, emc_cfg_o, EMC_CFG);
566*4882a593Smuzhiyun
567*4882a593Smuzhiyun /*
568*4882a593Smuzhiyun * 6. Timing update actally applies the new trimmers.
569*4882a593Smuzhiyun */
570*4882a593Smuzhiyun tegra210_emc_timing_update(emc);
571*4882a593Smuzhiyun
572*4882a593Smuzhiyun /* 6.1. Restore the UPDATE_DLL_IN_UPDATE field. */
573*4882a593Smuzhiyun emc_writel(emc, emc_cfg_update, EMC_CFG_UPDATE);
574*4882a593Smuzhiyun
575*4882a593Smuzhiyun /* 6.2. Restore the DLL. */
576*4882a593Smuzhiyun tegra210_emc_dll_enable(emc);
577*4882a593Smuzhiyun }
578*4882a593Smuzhiyun
579*4882a593Smuzhiyun return 0;
580*4882a593Smuzhiyun }
581*4882a593Smuzhiyun
582*4882a593Smuzhiyun /*
583*4882a593Smuzhiyun * Do the clock change sequence.
584*4882a593Smuzhiyun */
tegra210_emc_r21021_set_clock(struct tegra210_emc * emc,u32 clksrc)585*4882a593Smuzhiyun static void tegra210_emc_r21021_set_clock(struct tegra210_emc *emc, u32 clksrc)
586*4882a593Smuzhiyun {
587*4882a593Smuzhiyun /* state variables */
588*4882a593Smuzhiyun static bool fsp_for_next_freq;
589*4882a593Smuzhiyun /* constant configuration parameters */
590*4882a593Smuzhiyun const bool save_restore_clkstop_pd = true;
591*4882a593Smuzhiyun const u32 zqcal_before_cc_cutoff = 2400;
592*4882a593Smuzhiyun const bool cya_allow_ref_cc = false;
593*4882a593Smuzhiyun const bool cya_issue_pc_ref = false;
594*4882a593Smuzhiyun const bool opt_cc_short_zcal = true;
595*4882a593Smuzhiyun const bool ref_b4_sref_en = false;
596*4882a593Smuzhiyun const u32 tZQCAL_lpddr4 = 1000000;
597*4882a593Smuzhiyun const bool opt_short_zcal = true;
598*4882a593Smuzhiyun const bool opt_do_sw_qrst = true;
599*4882a593Smuzhiyun const u32 opt_dvfs_mode = MAN_SR;
600*4882a593Smuzhiyun /*
601*4882a593Smuzhiyun * This is the timing table for the source frequency. It does _not_
602*4882a593Smuzhiyun * necessarily correspond to the actual timing values in the EMC at the
603*4882a593Smuzhiyun * moment. If the boot BCT differs from the table then this can happen.
604*4882a593Smuzhiyun * However, we need it for accessing the dram_timings (which are not
605*4882a593Smuzhiyun * really registers) array for the current frequency.
606*4882a593Smuzhiyun */
607*4882a593Smuzhiyun struct tegra210_emc_timing *fake, *last = emc->last, *next = emc->next;
608*4882a593Smuzhiyun u32 tRTM, RP_war, R2P_war, TRPab_war, deltaTWATM, W2P_war, tRPST;
609*4882a593Smuzhiyun u32 mr13_flip_fspwr, mr13_flip_fspop, ramp_up_wait, ramp_down_wait;
610*4882a593Smuzhiyun u32 zq_wait_long, zq_latch_dvfs_wait_time, tZQCAL_lpddr4_fc_adj;
611*4882a593Smuzhiyun u32 emc_auto_cal_config, auto_cal_en, emc_cfg, emc_sel_dpd_ctrl;
612*4882a593Smuzhiyun u32 tFC_lpddr4 = 1000 * next->dram_timings[T_FC_LPDDR4];
613*4882a593Smuzhiyun u32 bg_reg_mode_change, enable_bglp_reg, enable_bg_reg;
614*4882a593Smuzhiyun bool opt_zcal_en_cc = false, is_lpddr3 = false;
615*4882a593Smuzhiyun bool compensate_trimmer_applicable = false;
616*4882a593Smuzhiyun u32 emc_dbg, emc_cfg_pipe_clk, emc_pin;
617*4882a593Smuzhiyun u32 src_clk_period, dst_clk_period; /* in picoseconds */
618*4882a593Smuzhiyun bool shared_zq_resistor = false;
619*4882a593Smuzhiyun u32 value, dram_type;
620*4882a593Smuzhiyun u32 opt_dll_mode = 0;
621*4882a593Smuzhiyun unsigned long delay;
622*4882a593Smuzhiyun unsigned int i;
623*4882a593Smuzhiyun
624*4882a593Smuzhiyun emc_dbg(emc, INFO, "Running clock change.\n");
625*4882a593Smuzhiyun
626*4882a593Smuzhiyun /* XXX fake == last */
627*4882a593Smuzhiyun fake = tegra210_emc_find_timing(emc, last->rate * 1000UL);
628*4882a593Smuzhiyun fsp_for_next_freq = !fsp_for_next_freq;
629*4882a593Smuzhiyun
630*4882a593Smuzhiyun value = emc_readl(emc, EMC_FBIO_CFG5) & EMC_FBIO_CFG5_DRAM_TYPE_MASK;
631*4882a593Smuzhiyun dram_type = value >> EMC_FBIO_CFG5_DRAM_TYPE_SHIFT;
632*4882a593Smuzhiyun
633*4882a593Smuzhiyun if (last->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX] & BIT(31))
634*4882a593Smuzhiyun shared_zq_resistor = true;
635*4882a593Smuzhiyun
636*4882a593Smuzhiyun if ((next->burst_regs[EMC_ZCAL_INTERVAL_INDEX] != 0 &&
637*4882a593Smuzhiyun last->burst_regs[EMC_ZCAL_INTERVAL_INDEX] == 0) ||
638*4882a593Smuzhiyun dram_type == DRAM_TYPE_LPDDR4)
639*4882a593Smuzhiyun opt_zcal_en_cc = true;
640*4882a593Smuzhiyun
641*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_DDR3)
642*4882a593Smuzhiyun opt_dll_mode = tegra210_emc_get_dll_state(next);
643*4882a593Smuzhiyun
644*4882a593Smuzhiyun if ((next->burst_regs[EMC_FBIO_CFG5_INDEX] & BIT(25)) &&
645*4882a593Smuzhiyun (dram_type == DRAM_TYPE_LPDDR2))
646*4882a593Smuzhiyun is_lpddr3 = true;
647*4882a593Smuzhiyun
648*4882a593Smuzhiyun emc_readl(emc, EMC_CFG);
649*4882a593Smuzhiyun emc_readl(emc, EMC_AUTO_CAL_CONFIG);
650*4882a593Smuzhiyun
651*4882a593Smuzhiyun src_clk_period = 1000000000 / last->rate;
652*4882a593Smuzhiyun dst_clk_period = 1000000000 / next->rate;
653*4882a593Smuzhiyun
654*4882a593Smuzhiyun if (dst_clk_period <= zqcal_before_cc_cutoff)
655*4882a593Smuzhiyun tZQCAL_lpddr4_fc_adj = tZQCAL_lpddr4 - tFC_lpddr4;
656*4882a593Smuzhiyun else
657*4882a593Smuzhiyun tZQCAL_lpddr4_fc_adj = tZQCAL_lpddr4;
658*4882a593Smuzhiyun
659*4882a593Smuzhiyun tZQCAL_lpddr4_fc_adj /= dst_clk_period;
660*4882a593Smuzhiyun
661*4882a593Smuzhiyun emc_dbg = emc_readl(emc, EMC_DBG);
662*4882a593Smuzhiyun emc_pin = emc_readl(emc, EMC_PIN);
663*4882a593Smuzhiyun emc_cfg_pipe_clk = emc_readl(emc, EMC_CFG_PIPE_CLK);
664*4882a593Smuzhiyun
665*4882a593Smuzhiyun emc_cfg = next->burst_regs[EMC_CFG_INDEX];
666*4882a593Smuzhiyun emc_cfg &= ~(EMC_CFG_DYN_SELF_REF | EMC_CFG_DRAM_ACPD |
667*4882a593Smuzhiyun EMC_CFG_DRAM_CLKSTOP_SR | EMC_CFG_DRAM_CLKSTOP_PD);
668*4882a593Smuzhiyun emc_sel_dpd_ctrl = next->emc_sel_dpd_ctrl;
669*4882a593Smuzhiyun emc_sel_dpd_ctrl &= ~(EMC_SEL_DPD_CTRL_CLK_SEL_DPD_EN |
670*4882a593Smuzhiyun EMC_SEL_DPD_CTRL_CA_SEL_DPD_EN |
671*4882a593Smuzhiyun EMC_SEL_DPD_CTRL_RESET_SEL_DPD_EN |
672*4882a593Smuzhiyun EMC_SEL_DPD_CTRL_ODT_SEL_DPD_EN |
673*4882a593Smuzhiyun EMC_SEL_DPD_CTRL_DATA_SEL_DPD_EN);
674*4882a593Smuzhiyun
675*4882a593Smuzhiyun emc_dbg(emc, INFO, "Clock change version: %d\n",
676*4882a593Smuzhiyun DVFS_CLOCK_CHANGE_VERSION);
677*4882a593Smuzhiyun emc_dbg(emc, INFO, "DRAM type = %d\n", dram_type);
678*4882a593Smuzhiyun emc_dbg(emc, INFO, "DRAM dev #: %u\n", emc->num_devices);
679*4882a593Smuzhiyun emc_dbg(emc, INFO, "Next EMC clksrc: 0x%08x\n", clksrc);
680*4882a593Smuzhiyun emc_dbg(emc, INFO, "DLL clksrc: 0x%08x\n", next->dll_clk_src);
681*4882a593Smuzhiyun emc_dbg(emc, INFO, "last rate: %u, next rate %u\n", last->rate,
682*4882a593Smuzhiyun next->rate);
683*4882a593Smuzhiyun emc_dbg(emc, INFO, "last period: %u, next period: %u\n",
684*4882a593Smuzhiyun src_clk_period, dst_clk_period);
685*4882a593Smuzhiyun emc_dbg(emc, INFO, " shared_zq_resistor: %d\n", !!shared_zq_resistor);
686*4882a593Smuzhiyun emc_dbg(emc, INFO, " num_channels: %u\n", emc->num_channels);
687*4882a593Smuzhiyun emc_dbg(emc, INFO, " opt_dll_mode: %d\n", opt_dll_mode);
688*4882a593Smuzhiyun
689*4882a593Smuzhiyun /*
690*4882a593Smuzhiyun * Step 1:
691*4882a593Smuzhiyun * Pre DVFS SW sequence.
692*4882a593Smuzhiyun */
693*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 1\n");
694*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 1.1: Disable DLL temporarily.\n");
695*4882a593Smuzhiyun
696*4882a593Smuzhiyun value = emc_readl(emc, EMC_CFG_DIG_DLL);
697*4882a593Smuzhiyun value &= ~EMC_CFG_DIG_DLL_CFG_DLL_EN;
698*4882a593Smuzhiyun emc_writel(emc, value, EMC_CFG_DIG_DLL);
699*4882a593Smuzhiyun
700*4882a593Smuzhiyun tegra210_emc_timing_update(emc);
701*4882a593Smuzhiyun
702*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++)
703*4882a593Smuzhiyun tegra210_emc_wait_for_update(emc, i, EMC_CFG_DIG_DLL,
704*4882a593Smuzhiyun EMC_CFG_DIG_DLL_CFG_DLL_EN, 0);
705*4882a593Smuzhiyun
706*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 1.2: Disable AUTOCAL temporarily.\n");
707*4882a593Smuzhiyun
708*4882a593Smuzhiyun emc_auto_cal_config = next->emc_auto_cal_config;
709*4882a593Smuzhiyun auto_cal_en = emc_auto_cal_config & EMC_AUTO_CAL_CONFIG_AUTO_CAL_ENABLE;
710*4882a593Smuzhiyun emc_auto_cal_config &= ~EMC_AUTO_CAL_CONFIG_AUTO_CAL_START;
711*4882a593Smuzhiyun emc_auto_cal_config |= EMC_AUTO_CAL_CONFIG_AUTO_CAL_MEASURE_STALL;
712*4882a593Smuzhiyun emc_auto_cal_config |= EMC_AUTO_CAL_CONFIG_AUTO_CAL_UPDATE_STALL;
713*4882a593Smuzhiyun emc_auto_cal_config |= auto_cal_en;
714*4882a593Smuzhiyun emc_writel(emc, emc_auto_cal_config, EMC_AUTO_CAL_CONFIG);
715*4882a593Smuzhiyun emc_readl(emc, EMC_AUTO_CAL_CONFIG); /* Flush write. */
716*4882a593Smuzhiyun
717*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 1.3: Disable other power features.\n");
718*4882a593Smuzhiyun
719*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
720*4882a593Smuzhiyun emc_writel(emc, emc_cfg, EMC_CFG);
721*4882a593Smuzhiyun emc_writel(emc, emc_sel_dpd_ctrl, EMC_SEL_DPD_CTRL);
722*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
723*4882a593Smuzhiyun
724*4882a593Smuzhiyun if (next->periodic_training) {
725*4882a593Smuzhiyun tegra210_emc_reset_dram_clktree_values(next);
726*4882a593Smuzhiyun
727*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++)
728*4882a593Smuzhiyun tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
729*4882a593Smuzhiyun EMC_EMC_STATUS_DRAM_IN_POWERDOWN_MASK,
730*4882a593Smuzhiyun 0);
731*4882a593Smuzhiyun
732*4882a593Smuzhiyun for (i = 0; i < emc->num_channels; i++)
733*4882a593Smuzhiyun tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
734*4882a593Smuzhiyun EMC_EMC_STATUS_DRAM_IN_SELF_REFRESH_MASK,
735*4882a593Smuzhiyun 0);
736*4882a593Smuzhiyun
737*4882a593Smuzhiyun tegra210_emc_start_periodic_compensation(emc);
738*4882a593Smuzhiyun
739*4882a593Smuzhiyun delay = 1000 * tegra210_emc_actual_osc_clocks(last->run_clocks);
740*4882a593Smuzhiyun udelay((delay / last->rate) + 2);
741*4882a593Smuzhiyun
742*4882a593Smuzhiyun value = periodic_compensation_handler(emc, DVFS_SEQUENCE, fake,
743*4882a593Smuzhiyun next);
744*4882a593Smuzhiyun value = (value * 128 * next->rate / 1000) / 1000000;
745*4882a593Smuzhiyun
746*4882a593Smuzhiyun if (next->periodic_training && value > next->tree_margin)
747*4882a593Smuzhiyun compensate_trimmer_applicable = true;
748*4882a593Smuzhiyun }
749*4882a593Smuzhiyun
750*4882a593Smuzhiyun emc_writel(emc, EMC_INTSTATUS_CLKCHANGE_COMPLETE, EMC_INTSTATUS);
751*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
752*4882a593Smuzhiyun emc_writel(emc, emc_cfg, EMC_CFG);
753*4882a593Smuzhiyun emc_writel(emc, emc_sel_dpd_ctrl, EMC_SEL_DPD_CTRL);
754*4882a593Smuzhiyun emc_writel(emc, emc_cfg_pipe_clk | EMC_CFG_PIPE_CLK_CLK_ALWAYS_ON,
755*4882a593Smuzhiyun EMC_CFG_PIPE_CLK);
756*4882a593Smuzhiyun emc_writel(emc, next->emc_fdpd_ctrl_cmd_no_ramp &
757*4882a593Smuzhiyun ~EMC_FDPD_CTRL_CMD_NO_RAMP_CMD_DPD_NO_RAMP_ENABLE,
758*4882a593Smuzhiyun EMC_FDPD_CTRL_CMD_NO_RAMP);
759*4882a593Smuzhiyun
760*4882a593Smuzhiyun bg_reg_mode_change =
761*4882a593Smuzhiyun ((next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
762*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD) ^
763*4882a593Smuzhiyun (last->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
764*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD)) ||
765*4882a593Smuzhiyun ((next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
766*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD) ^
767*4882a593Smuzhiyun (last->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
768*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD));
769*4882a593Smuzhiyun enable_bglp_reg =
770*4882a593Smuzhiyun (next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
771*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD) == 0;
772*4882a593Smuzhiyun enable_bg_reg =
773*4882a593Smuzhiyun (next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
774*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD) == 0;
775*4882a593Smuzhiyun
776*4882a593Smuzhiyun if (bg_reg_mode_change) {
777*4882a593Smuzhiyun if (enable_bg_reg)
778*4882a593Smuzhiyun emc_writel(emc, last->burst_regs
779*4882a593Smuzhiyun [EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
780*4882a593Smuzhiyun ~EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD,
781*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0);
782*4882a593Smuzhiyun
783*4882a593Smuzhiyun if (enable_bglp_reg)
784*4882a593Smuzhiyun emc_writel(emc, last->burst_regs
785*4882a593Smuzhiyun [EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
786*4882a593Smuzhiyun ~EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD,
787*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0);
788*4882a593Smuzhiyun }
789*4882a593Smuzhiyun
790*4882a593Smuzhiyun /* Check if we need to turn on VREF generator. */
791*4882a593Smuzhiyun if ((((last->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
792*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF) == 0) &&
793*4882a593Smuzhiyun ((next->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
794*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF) == 1)) ||
795*4882a593Smuzhiyun (((last->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
796*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF) == 0) &&
797*4882a593Smuzhiyun ((next->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
798*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF) != 0))) {
799*4882a593Smuzhiyun u32 pad_tx_ctrl =
800*4882a593Smuzhiyun next->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX];
801*4882a593Smuzhiyun u32 last_pad_tx_ctrl =
802*4882a593Smuzhiyun last->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX];
803*4882a593Smuzhiyun u32 next_dq_e_ivref, next_dqs_e_ivref;
804*4882a593Smuzhiyun
805*4882a593Smuzhiyun next_dqs_e_ivref = pad_tx_ctrl &
806*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF;
807*4882a593Smuzhiyun next_dq_e_ivref = pad_tx_ctrl &
808*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF;
809*4882a593Smuzhiyun value = (last_pad_tx_ctrl &
810*4882a593Smuzhiyun ~EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF &
811*4882a593Smuzhiyun ~EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF) |
812*4882a593Smuzhiyun next_dq_e_ivref | next_dqs_e_ivref;
813*4882a593Smuzhiyun emc_writel(emc, value, EMC_PMACRO_DATA_PAD_TX_CTRL);
814*4882a593Smuzhiyun udelay(1);
815*4882a593Smuzhiyun } else if (bg_reg_mode_change) {
816*4882a593Smuzhiyun udelay(1);
817*4882a593Smuzhiyun }
818*4882a593Smuzhiyun
819*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
820*4882a593Smuzhiyun
821*4882a593Smuzhiyun /*
822*4882a593Smuzhiyun * Step 2:
823*4882a593Smuzhiyun * Prelock the DLL.
824*4882a593Smuzhiyun */
825*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 2\n");
826*4882a593Smuzhiyun
827*4882a593Smuzhiyun if (next->burst_regs[EMC_CFG_DIG_DLL_INDEX] &
828*4882a593Smuzhiyun EMC_CFG_DIG_DLL_CFG_DLL_EN) {
829*4882a593Smuzhiyun emc_dbg(emc, INFO, "Prelock enabled for target frequency.\n");
830*4882a593Smuzhiyun value = tegra210_emc_dll_prelock(emc, clksrc);
831*4882a593Smuzhiyun emc_dbg(emc, INFO, "DLL out: 0x%03x\n", value);
832*4882a593Smuzhiyun } else {
833*4882a593Smuzhiyun emc_dbg(emc, INFO, "Disabling DLL for target frequency.\n");
834*4882a593Smuzhiyun tegra210_emc_dll_disable(emc);
835*4882a593Smuzhiyun }
836*4882a593Smuzhiyun
837*4882a593Smuzhiyun /*
838*4882a593Smuzhiyun * Step 3:
839*4882a593Smuzhiyun * Prepare autocal for the clock change.
840*4882a593Smuzhiyun */
841*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 3\n");
842*4882a593Smuzhiyun
843*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
844*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config2, EMC_AUTO_CAL_CONFIG2);
845*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config3, EMC_AUTO_CAL_CONFIG3);
846*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config4, EMC_AUTO_CAL_CONFIG4);
847*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config5, EMC_AUTO_CAL_CONFIG5);
848*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config6, EMC_AUTO_CAL_CONFIG6);
849*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config7, EMC_AUTO_CAL_CONFIG7);
850*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config8, EMC_AUTO_CAL_CONFIG8);
851*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
852*4882a593Smuzhiyun
853*4882a593Smuzhiyun emc_auto_cal_config |= (EMC_AUTO_CAL_CONFIG_AUTO_CAL_COMPUTE_START |
854*4882a593Smuzhiyun auto_cal_en);
855*4882a593Smuzhiyun emc_writel(emc, emc_auto_cal_config, EMC_AUTO_CAL_CONFIG);
856*4882a593Smuzhiyun
857*4882a593Smuzhiyun /*
858*4882a593Smuzhiyun * Step 4:
859*4882a593Smuzhiyun * Update EMC_CFG. (??)
860*4882a593Smuzhiyun */
861*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 4\n");
862*4882a593Smuzhiyun
863*4882a593Smuzhiyun if (src_clk_period > 50000 && dram_type == DRAM_TYPE_LPDDR4)
864*4882a593Smuzhiyun ccfifo_writel(emc, 1, EMC_SELF_REF, 0);
865*4882a593Smuzhiyun else
866*4882a593Smuzhiyun emc_writel(emc, next->emc_cfg_2, EMC_CFG_2);
867*4882a593Smuzhiyun
868*4882a593Smuzhiyun /*
869*4882a593Smuzhiyun * Step 5:
870*4882a593Smuzhiyun * Prepare reference variables for ZQCAL regs.
871*4882a593Smuzhiyun */
872*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 5\n");
873*4882a593Smuzhiyun
874*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4)
875*4882a593Smuzhiyun zq_wait_long = max((u32)1, div_o3(1000000, dst_clk_period));
876*4882a593Smuzhiyun else if (dram_type == DRAM_TYPE_LPDDR2 || is_lpddr3)
877*4882a593Smuzhiyun zq_wait_long = max(next->min_mrs_wait,
878*4882a593Smuzhiyun div_o3(360000, dst_clk_period)) + 4;
879*4882a593Smuzhiyun else if (dram_type == DRAM_TYPE_DDR3)
880*4882a593Smuzhiyun zq_wait_long = max((u32)256,
881*4882a593Smuzhiyun div_o3(320000, dst_clk_period) + 2);
882*4882a593Smuzhiyun else
883*4882a593Smuzhiyun zq_wait_long = 0;
884*4882a593Smuzhiyun
885*4882a593Smuzhiyun /*
886*4882a593Smuzhiyun * Step 6:
887*4882a593Smuzhiyun * Training code - removed.
888*4882a593Smuzhiyun */
889*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 6\n");
890*4882a593Smuzhiyun
891*4882a593Smuzhiyun /*
892*4882a593Smuzhiyun * Step 7:
893*4882a593Smuzhiyun * Program FSP reference registers and send MRWs to new FSPWR.
894*4882a593Smuzhiyun */
895*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 7\n");
896*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Step 7.1: Bug 200024907 - Patch RP R2P");
897*4882a593Smuzhiyun
898*4882a593Smuzhiyun /* WAR 200024907 */
899*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
900*4882a593Smuzhiyun u32 nRTP = 16;
901*4882a593Smuzhiyun
902*4882a593Smuzhiyun if (src_clk_period >= 1000000 / 1866) /* 535.91 ps */
903*4882a593Smuzhiyun nRTP = 14;
904*4882a593Smuzhiyun
905*4882a593Smuzhiyun if (src_clk_period >= 1000000 / 1600) /* 625.00 ps */
906*4882a593Smuzhiyun nRTP = 12;
907*4882a593Smuzhiyun
908*4882a593Smuzhiyun if (src_clk_period >= 1000000 / 1333) /* 750.19 ps */
909*4882a593Smuzhiyun nRTP = 10;
910*4882a593Smuzhiyun
911*4882a593Smuzhiyun if (src_clk_period >= 1000000 / 1066) /* 938.09 ps */
912*4882a593Smuzhiyun nRTP = 8;
913*4882a593Smuzhiyun
914*4882a593Smuzhiyun deltaTWATM = max_t(u32, div_o3(7500, src_clk_period), 8);
915*4882a593Smuzhiyun
916*4882a593Smuzhiyun /*
917*4882a593Smuzhiyun * Originally there was a + .5 in the tRPST calculation.
918*4882a593Smuzhiyun * However since we can't do FP in the kernel and the tRTM
919*4882a593Smuzhiyun * computation was in a floating point ceiling function, adding
920*4882a593Smuzhiyun * one to tRTP should be ok. There is no other source of non
921*4882a593Smuzhiyun * integer values, so the result was always going to be
922*4882a593Smuzhiyun * something for the form: f_ceil(N + .5) = N + 1;
923*4882a593Smuzhiyun */
924*4882a593Smuzhiyun tRPST = (last->emc_mrw & 0x80) >> 7;
925*4882a593Smuzhiyun tRTM = fake->dram_timings[RL] + div_o3(3600, src_clk_period) +
926*4882a593Smuzhiyun max_t(u32, div_o3(7500, src_clk_period), 8) + tRPST +
927*4882a593Smuzhiyun 1 + nRTP;
928*4882a593Smuzhiyun
929*4882a593Smuzhiyun emc_dbg(emc, INFO, "tRTM = %u, EMC_RP = %u\n", tRTM,
930*4882a593Smuzhiyun next->burst_regs[EMC_RP_INDEX]);
931*4882a593Smuzhiyun
932*4882a593Smuzhiyun if (last->burst_regs[EMC_RP_INDEX] < tRTM) {
933*4882a593Smuzhiyun if (tRTM > (last->burst_regs[EMC_R2P_INDEX] +
934*4882a593Smuzhiyun last->burst_regs[EMC_RP_INDEX])) {
935*4882a593Smuzhiyun R2P_war = tRTM - last->burst_regs[EMC_RP_INDEX];
936*4882a593Smuzhiyun RP_war = last->burst_regs[EMC_RP_INDEX];
937*4882a593Smuzhiyun TRPab_war = last->burst_regs[EMC_TRPAB_INDEX];
938*4882a593Smuzhiyun
939*4882a593Smuzhiyun if (R2P_war > 63) {
940*4882a593Smuzhiyun RP_war = R2P_war +
941*4882a593Smuzhiyun last->burst_regs[EMC_RP_INDEX] - 63;
942*4882a593Smuzhiyun
943*4882a593Smuzhiyun if (TRPab_war < RP_war)
944*4882a593Smuzhiyun TRPab_war = RP_war;
945*4882a593Smuzhiyun
946*4882a593Smuzhiyun R2P_war = 63;
947*4882a593Smuzhiyun }
948*4882a593Smuzhiyun } else {
949*4882a593Smuzhiyun R2P_war = last->burst_regs[EMC_R2P_INDEX];
950*4882a593Smuzhiyun RP_war = last->burst_regs[EMC_RP_INDEX];
951*4882a593Smuzhiyun TRPab_war = last->burst_regs[EMC_TRPAB_INDEX];
952*4882a593Smuzhiyun }
953*4882a593Smuzhiyun
954*4882a593Smuzhiyun if (RP_war < deltaTWATM) {
955*4882a593Smuzhiyun W2P_war = last->burst_regs[EMC_W2P_INDEX]
956*4882a593Smuzhiyun + deltaTWATM - RP_war;
957*4882a593Smuzhiyun if (W2P_war > 63) {
958*4882a593Smuzhiyun RP_war = RP_war + W2P_war - 63;
959*4882a593Smuzhiyun if (TRPab_war < RP_war)
960*4882a593Smuzhiyun TRPab_war = RP_war;
961*4882a593Smuzhiyun W2P_war = 63;
962*4882a593Smuzhiyun }
963*4882a593Smuzhiyun } else {
964*4882a593Smuzhiyun W2P_war = last->burst_regs[
965*4882a593Smuzhiyun EMC_W2P_INDEX];
966*4882a593Smuzhiyun }
967*4882a593Smuzhiyun
968*4882a593Smuzhiyun if ((last->burst_regs[EMC_W2P_INDEX] ^ W2P_war) ||
969*4882a593Smuzhiyun (last->burst_regs[EMC_R2P_INDEX] ^ R2P_war) ||
970*4882a593Smuzhiyun (last->burst_regs[EMC_RP_INDEX] ^ RP_war) ||
971*4882a593Smuzhiyun (last->burst_regs[EMC_TRPAB_INDEX] ^ TRPab_war)) {
972*4882a593Smuzhiyun emc_writel(emc, RP_war, EMC_RP);
973*4882a593Smuzhiyun emc_writel(emc, R2P_war, EMC_R2P);
974*4882a593Smuzhiyun emc_writel(emc, W2P_war, EMC_W2P);
975*4882a593Smuzhiyun emc_writel(emc, TRPab_war, EMC_TRPAB);
976*4882a593Smuzhiyun }
977*4882a593Smuzhiyun
978*4882a593Smuzhiyun tegra210_emc_timing_update(emc);
979*4882a593Smuzhiyun } else {
980*4882a593Smuzhiyun emc_dbg(emc, INFO, "Skipped WAR\n");
981*4882a593Smuzhiyun }
982*4882a593Smuzhiyun }
983*4882a593Smuzhiyun
984*4882a593Smuzhiyun if (!fsp_for_next_freq) {
985*4882a593Smuzhiyun mr13_flip_fspwr = (next->emc_mrw3 & 0xffffff3f) | 0x80;
986*4882a593Smuzhiyun mr13_flip_fspop = (next->emc_mrw3 & 0xffffff3f) | 0x00;
987*4882a593Smuzhiyun } else {
988*4882a593Smuzhiyun mr13_flip_fspwr = (next->emc_mrw3 & 0xffffff3f) | 0x40;
989*4882a593Smuzhiyun mr13_flip_fspop = (next->emc_mrw3 & 0xffffff3f) | 0xc0;
990*4882a593Smuzhiyun }
991*4882a593Smuzhiyun
992*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
993*4882a593Smuzhiyun emc_writel(emc, mr13_flip_fspwr, EMC_MRW3);
994*4882a593Smuzhiyun emc_writel(emc, next->emc_mrw, EMC_MRW);
995*4882a593Smuzhiyun emc_writel(emc, next->emc_mrw2, EMC_MRW2);
996*4882a593Smuzhiyun }
997*4882a593Smuzhiyun
998*4882a593Smuzhiyun /*
999*4882a593Smuzhiyun * Step 8:
1000*4882a593Smuzhiyun * Program the shadow registers.
1001*4882a593Smuzhiyun */
1002*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 8\n");
1003*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing burst_regs\n");
1004*4882a593Smuzhiyun
1005*4882a593Smuzhiyun for (i = 0; i < next->num_burst; i++) {
1006*4882a593Smuzhiyun const u16 *offsets = emc->offsets->burst;
1007*4882a593Smuzhiyun u16 offset;
1008*4882a593Smuzhiyun
1009*4882a593Smuzhiyun if (!offsets[i])
1010*4882a593Smuzhiyun continue;
1011*4882a593Smuzhiyun
1012*4882a593Smuzhiyun value = next->burst_regs[i];
1013*4882a593Smuzhiyun offset = offsets[i];
1014*4882a593Smuzhiyun
1015*4882a593Smuzhiyun if (dram_type != DRAM_TYPE_LPDDR4 &&
1016*4882a593Smuzhiyun (offset == EMC_MRW6 || offset == EMC_MRW7 ||
1017*4882a593Smuzhiyun offset == EMC_MRW8 || offset == EMC_MRW9 ||
1018*4882a593Smuzhiyun offset == EMC_MRW10 || offset == EMC_MRW11 ||
1019*4882a593Smuzhiyun offset == EMC_MRW12 || offset == EMC_MRW13 ||
1020*4882a593Smuzhiyun offset == EMC_MRW14 || offset == EMC_MRW15 ||
1021*4882a593Smuzhiyun offset == EMC_TRAINING_CTRL))
1022*4882a593Smuzhiyun continue;
1023*4882a593Smuzhiyun
1024*4882a593Smuzhiyun /* Pain... And suffering. */
1025*4882a593Smuzhiyun if (offset == EMC_CFG) {
1026*4882a593Smuzhiyun value &= ~EMC_CFG_DRAM_ACPD;
1027*4882a593Smuzhiyun value &= ~EMC_CFG_DYN_SELF_REF;
1028*4882a593Smuzhiyun
1029*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
1030*4882a593Smuzhiyun value &= ~EMC_CFG_DRAM_CLKSTOP_SR;
1031*4882a593Smuzhiyun value &= ~EMC_CFG_DRAM_CLKSTOP_PD;
1032*4882a593Smuzhiyun }
1033*4882a593Smuzhiyun } else if (offset == EMC_MRS_WAIT_CNT &&
1034*4882a593Smuzhiyun dram_type == DRAM_TYPE_LPDDR2 &&
1035*4882a593Smuzhiyun opt_zcal_en_cc && !opt_cc_short_zcal &&
1036*4882a593Smuzhiyun opt_short_zcal) {
1037*4882a593Smuzhiyun value = (value & ~(EMC_MRS_WAIT_CNT_SHORT_WAIT_MASK <<
1038*4882a593Smuzhiyun EMC_MRS_WAIT_CNT_SHORT_WAIT_SHIFT)) |
1039*4882a593Smuzhiyun ((zq_wait_long & EMC_MRS_WAIT_CNT_SHORT_WAIT_MASK) <<
1040*4882a593Smuzhiyun EMC_MRS_WAIT_CNT_SHORT_WAIT_SHIFT);
1041*4882a593Smuzhiyun } else if (offset == EMC_ZCAL_WAIT_CNT &&
1042*4882a593Smuzhiyun dram_type == DRAM_TYPE_DDR3 && opt_zcal_en_cc &&
1043*4882a593Smuzhiyun !opt_cc_short_zcal && opt_short_zcal) {
1044*4882a593Smuzhiyun value = (value & ~(EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_MASK <<
1045*4882a593Smuzhiyun EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_SHIFT)) |
1046*4882a593Smuzhiyun ((zq_wait_long & EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_MASK) <<
1047*4882a593Smuzhiyun EMC_MRS_WAIT_CNT_SHORT_WAIT_SHIFT);
1048*4882a593Smuzhiyun } else if (offset == EMC_ZCAL_INTERVAL && opt_zcal_en_cc) {
1049*4882a593Smuzhiyun value = 0; /* EMC_ZCAL_INTERVAL reset value. */
1050*4882a593Smuzhiyun } else if (offset == EMC_PMACRO_AUTOCAL_CFG_COMMON) {
1051*4882a593Smuzhiyun value |= EMC_PMACRO_AUTOCAL_CFG_COMMON_E_CAL_BYPASS_DVFS;
1052*4882a593Smuzhiyun } else if (offset == EMC_PMACRO_DATA_PAD_TX_CTRL) {
1053*4882a593Smuzhiyun value &= ~(EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQSP_TX_E_DCC |
1054*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQSN_TX_E_DCC |
1055*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_TX_E_DCC |
1056*4882a593Smuzhiyun EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_CMD_TX_E_DCC);
1057*4882a593Smuzhiyun } else if (offset == EMC_PMACRO_CMD_PAD_TX_CTRL) {
1058*4882a593Smuzhiyun value |= EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQ_TX_DRVFORCEON;
1059*4882a593Smuzhiyun value &= ~(EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQSP_TX_E_DCC |
1060*4882a593Smuzhiyun EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQSN_TX_E_DCC |
1061*4882a593Smuzhiyun EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQ_TX_E_DCC |
1062*4882a593Smuzhiyun EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_CMD_TX_E_DCC);
1063*4882a593Smuzhiyun } else if (offset == EMC_PMACRO_BRICK_CTRL_RFU1) {
1064*4882a593Smuzhiyun value &= 0xf800f800;
1065*4882a593Smuzhiyun } else if (offset == EMC_PMACRO_COMMON_PAD_TX_CTRL) {
1066*4882a593Smuzhiyun value &= 0xfffffff0;
1067*4882a593Smuzhiyun }
1068*4882a593Smuzhiyun
1069*4882a593Smuzhiyun emc_writel(emc, value, offset);
1070*4882a593Smuzhiyun }
1071*4882a593Smuzhiyun
1072*4882a593Smuzhiyun /* SW addition: do EMC refresh adjustment here. */
1073*4882a593Smuzhiyun tegra210_emc_adjust_timing(emc, next);
1074*4882a593Smuzhiyun
1075*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
1076*4882a593Smuzhiyun value = (23 << EMC_MRW_MRW_MA_SHIFT) |
1077*4882a593Smuzhiyun (next->run_clocks & EMC_MRW_MRW_OP_MASK);
1078*4882a593Smuzhiyun emc_writel(emc, value, EMC_MRW);
1079*4882a593Smuzhiyun }
1080*4882a593Smuzhiyun
1081*4882a593Smuzhiyun /* Per channel burst registers. */
1082*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing burst_regs_per_ch\n");
1083*4882a593Smuzhiyun
1084*4882a593Smuzhiyun for (i = 0; i < next->num_burst_per_ch; i++) {
1085*4882a593Smuzhiyun const struct tegra210_emc_per_channel_regs *burst =
1086*4882a593Smuzhiyun emc->offsets->burst_per_channel;
1087*4882a593Smuzhiyun
1088*4882a593Smuzhiyun if (!burst[i].offset)
1089*4882a593Smuzhiyun continue;
1090*4882a593Smuzhiyun
1091*4882a593Smuzhiyun if (dram_type != DRAM_TYPE_LPDDR4 &&
1092*4882a593Smuzhiyun (burst[i].offset == EMC_MRW6 ||
1093*4882a593Smuzhiyun burst[i].offset == EMC_MRW7 ||
1094*4882a593Smuzhiyun burst[i].offset == EMC_MRW8 ||
1095*4882a593Smuzhiyun burst[i].offset == EMC_MRW9 ||
1096*4882a593Smuzhiyun burst[i].offset == EMC_MRW10 ||
1097*4882a593Smuzhiyun burst[i].offset == EMC_MRW11 ||
1098*4882a593Smuzhiyun burst[i].offset == EMC_MRW12 ||
1099*4882a593Smuzhiyun burst[i].offset == EMC_MRW13 ||
1100*4882a593Smuzhiyun burst[i].offset == EMC_MRW14 ||
1101*4882a593Smuzhiyun burst[i].offset == EMC_MRW15))
1102*4882a593Smuzhiyun continue;
1103*4882a593Smuzhiyun
1104*4882a593Smuzhiyun /* Filter out second channel if not in DUAL_CHANNEL mode. */
1105*4882a593Smuzhiyun if (emc->num_channels < 2 && burst[i].bank >= 1)
1106*4882a593Smuzhiyun continue;
1107*4882a593Smuzhiyun
1108*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1109*4882a593Smuzhiyun next->burst_reg_per_ch[i], burst[i].offset);
1110*4882a593Smuzhiyun emc_channel_writel(emc, burst[i].bank,
1111*4882a593Smuzhiyun next->burst_reg_per_ch[i],
1112*4882a593Smuzhiyun burst[i].offset);
1113*4882a593Smuzhiyun }
1114*4882a593Smuzhiyun
1115*4882a593Smuzhiyun /* Vref regs. */
1116*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing vref_regs\n");
1117*4882a593Smuzhiyun
1118*4882a593Smuzhiyun for (i = 0; i < next->vref_num; i++) {
1119*4882a593Smuzhiyun const struct tegra210_emc_per_channel_regs *vref =
1120*4882a593Smuzhiyun emc->offsets->vref_per_channel;
1121*4882a593Smuzhiyun
1122*4882a593Smuzhiyun if (!vref[i].offset)
1123*4882a593Smuzhiyun continue;
1124*4882a593Smuzhiyun
1125*4882a593Smuzhiyun if (emc->num_channels < 2 && vref[i].bank >= 1)
1126*4882a593Smuzhiyun continue;
1127*4882a593Smuzhiyun
1128*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1129*4882a593Smuzhiyun next->vref_perch_regs[i], vref[i].offset);
1130*4882a593Smuzhiyun emc_channel_writel(emc, vref[i].bank, next->vref_perch_regs[i],
1131*4882a593Smuzhiyun vref[i].offset);
1132*4882a593Smuzhiyun }
1133*4882a593Smuzhiyun
1134*4882a593Smuzhiyun /* Trimmers. */
1135*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing trim_regs\n");
1136*4882a593Smuzhiyun
1137*4882a593Smuzhiyun for (i = 0; i < next->num_trim; i++) {
1138*4882a593Smuzhiyun const u16 *offsets = emc->offsets->trim;
1139*4882a593Smuzhiyun
1140*4882a593Smuzhiyun if (!offsets[i])
1141*4882a593Smuzhiyun continue;
1142*4882a593Smuzhiyun
1143*4882a593Smuzhiyun if (compensate_trimmer_applicable &&
1144*4882a593Smuzhiyun (offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_0 ||
1145*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_1 ||
1146*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_2 ||
1147*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_3 ||
1148*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_0 ||
1149*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_1 ||
1150*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_2 ||
1151*4882a593Smuzhiyun offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_3 ||
1152*4882a593Smuzhiyun offsets[i] == EMC_DATA_BRLSHFT_0 ||
1153*4882a593Smuzhiyun offsets[i] == EMC_DATA_BRLSHFT_1)) {
1154*4882a593Smuzhiyun value = tegra210_emc_compensate(next, offsets[i]);
1155*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1156*4882a593Smuzhiyun value, offsets[i]);
1157*4882a593Smuzhiyun emc_dbg(emc, EMA_WRITES, "0x%08x <= 0x%08x\n",
1158*4882a593Smuzhiyun (u32)(u64)offsets[i], value);
1159*4882a593Smuzhiyun emc_writel(emc, value, offsets[i]);
1160*4882a593Smuzhiyun } else {
1161*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1162*4882a593Smuzhiyun next->trim_regs[i], offsets[i]);
1163*4882a593Smuzhiyun emc_writel(emc, next->trim_regs[i], offsets[i]);
1164*4882a593Smuzhiyun }
1165*4882a593Smuzhiyun }
1166*4882a593Smuzhiyun
1167*4882a593Smuzhiyun /* Per channel trimmers. */
1168*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing trim_regs_per_ch\n");
1169*4882a593Smuzhiyun
1170*4882a593Smuzhiyun for (i = 0; i < next->num_trim_per_ch; i++) {
1171*4882a593Smuzhiyun const struct tegra210_emc_per_channel_regs *trim =
1172*4882a593Smuzhiyun &emc->offsets->trim_per_channel[0];
1173*4882a593Smuzhiyun unsigned int offset;
1174*4882a593Smuzhiyun
1175*4882a593Smuzhiyun if (!trim[i].offset)
1176*4882a593Smuzhiyun continue;
1177*4882a593Smuzhiyun
1178*4882a593Smuzhiyun if (emc->num_channels < 2 && trim[i].bank >= 1)
1179*4882a593Smuzhiyun continue;
1180*4882a593Smuzhiyun
1181*4882a593Smuzhiyun offset = trim[i].offset;
1182*4882a593Smuzhiyun
1183*4882a593Smuzhiyun if (compensate_trimmer_applicable &&
1184*4882a593Smuzhiyun (offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_0 ||
1185*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_1 ||
1186*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_2 ||
1187*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_3 ||
1188*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_0 ||
1189*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_1 ||
1190*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_2 ||
1191*4882a593Smuzhiyun offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_3 ||
1192*4882a593Smuzhiyun offset == EMC_DATA_BRLSHFT_0 ||
1193*4882a593Smuzhiyun offset == EMC_DATA_BRLSHFT_1)) {
1194*4882a593Smuzhiyun value = tegra210_emc_compensate(next, offset);
1195*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1196*4882a593Smuzhiyun value, offset);
1197*4882a593Smuzhiyun emc_dbg(emc, EMA_WRITES, "0x%08x <= 0x%08x\n", offset,
1198*4882a593Smuzhiyun value);
1199*4882a593Smuzhiyun emc_channel_writel(emc, trim[i].bank, value, offset);
1200*4882a593Smuzhiyun } else {
1201*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1202*4882a593Smuzhiyun next->trim_perch_regs[i], offset);
1203*4882a593Smuzhiyun emc_channel_writel(emc, trim[i].bank,
1204*4882a593Smuzhiyun next->trim_perch_regs[i], offset);
1205*4882a593Smuzhiyun }
1206*4882a593Smuzhiyun }
1207*4882a593Smuzhiyun
1208*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing burst_mc_regs\n");
1209*4882a593Smuzhiyun
1210*4882a593Smuzhiyun for (i = 0; i < next->num_mc_regs; i++) {
1211*4882a593Smuzhiyun const u16 *offsets = emc->offsets->burst_mc;
1212*4882a593Smuzhiyun u32 *values = next->burst_mc_regs;
1213*4882a593Smuzhiyun
1214*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1215*4882a593Smuzhiyun values[i], offsets[i]);
1216*4882a593Smuzhiyun mc_writel(emc->mc, values[i], offsets[i]);
1217*4882a593Smuzhiyun }
1218*4882a593Smuzhiyun
1219*4882a593Smuzhiyun /* Registers to be programmed on the faster clock. */
1220*4882a593Smuzhiyun if (next->rate < last->rate) {
1221*4882a593Smuzhiyun const u16 *la = emc->offsets->la_scale;
1222*4882a593Smuzhiyun
1223*4882a593Smuzhiyun emc_dbg(emc, SUB_STEPS, "Writing la_scale_regs\n");
1224*4882a593Smuzhiyun
1225*4882a593Smuzhiyun for (i = 0; i < next->num_up_down; i++) {
1226*4882a593Smuzhiyun emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
1227*4882a593Smuzhiyun next->la_scale_regs[i], la[i]);
1228*4882a593Smuzhiyun mc_writel(emc->mc, next->la_scale_regs[i], la[i]);
1229*4882a593Smuzhiyun }
1230*4882a593Smuzhiyun }
1231*4882a593Smuzhiyun
1232*4882a593Smuzhiyun /* Flush all the burst register writes. */
1233*4882a593Smuzhiyun mc_readl(emc->mc, MC_EMEM_ADR_CFG);
1234*4882a593Smuzhiyun
1235*4882a593Smuzhiyun /*
1236*4882a593Smuzhiyun * Step 9:
1237*4882a593Smuzhiyun * LPDDR4 section A.
1238*4882a593Smuzhiyun */
1239*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 9\n");
1240*4882a593Smuzhiyun
1241*4882a593Smuzhiyun value = next->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX];
1242*4882a593Smuzhiyun value &= ~EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_MASK;
1243*4882a593Smuzhiyun
1244*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
1245*4882a593Smuzhiyun emc_writel(emc, 0, EMC_ZCAL_INTERVAL);
1246*4882a593Smuzhiyun emc_writel(emc, value, EMC_ZCAL_WAIT_CNT);
1247*4882a593Smuzhiyun
1248*4882a593Smuzhiyun value = emc_dbg | (EMC_DBG_WRITE_MUX_ACTIVE |
1249*4882a593Smuzhiyun EMC_DBG_WRITE_ACTIVE_ONLY);
1250*4882a593Smuzhiyun
1251*4882a593Smuzhiyun emc_writel(emc, value, EMC_DBG);
1252*4882a593Smuzhiyun emc_writel(emc, 0, EMC_ZCAL_INTERVAL);
1253*4882a593Smuzhiyun emc_writel(emc, emc_dbg, EMC_DBG);
1254*4882a593Smuzhiyun }
1255*4882a593Smuzhiyun
1256*4882a593Smuzhiyun /*
1257*4882a593Smuzhiyun * Step 10:
1258*4882a593Smuzhiyun * LPDDR4 and DDR3 common section.
1259*4882a593Smuzhiyun */
1260*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 10\n");
1261*4882a593Smuzhiyun
1262*4882a593Smuzhiyun if (opt_dvfs_mode == MAN_SR || dram_type == DRAM_TYPE_LPDDR4) {
1263*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4)
1264*4882a593Smuzhiyun ccfifo_writel(emc, 0x101, EMC_SELF_REF, 0);
1265*4882a593Smuzhiyun else
1266*4882a593Smuzhiyun ccfifo_writel(emc, 0x1, EMC_SELF_REF, 0);
1267*4882a593Smuzhiyun
1268*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4 &&
1269*4882a593Smuzhiyun dst_clk_period <= zqcal_before_cc_cutoff) {
1270*4882a593Smuzhiyun ccfifo_writel(emc, mr13_flip_fspwr ^ 0x40, EMC_MRW3, 0);
1271*4882a593Smuzhiyun ccfifo_writel(emc, (next->burst_regs[EMC_MRW6_INDEX] &
1272*4882a593Smuzhiyun 0xFFFF3F3F) |
1273*4882a593Smuzhiyun (last->burst_regs[EMC_MRW6_INDEX] &
1274*4882a593Smuzhiyun 0x0000C0C0), EMC_MRW6, 0);
1275*4882a593Smuzhiyun ccfifo_writel(emc, (next->burst_regs[EMC_MRW14_INDEX] &
1276*4882a593Smuzhiyun 0xFFFF0707) |
1277*4882a593Smuzhiyun (last->burst_regs[EMC_MRW14_INDEX] &
1278*4882a593Smuzhiyun 0x00003838), EMC_MRW14, 0);
1279*4882a593Smuzhiyun
1280*4882a593Smuzhiyun if (emc->num_devices > 1) {
1281*4882a593Smuzhiyun ccfifo_writel(emc,
1282*4882a593Smuzhiyun (next->burst_regs[EMC_MRW7_INDEX] &
1283*4882a593Smuzhiyun 0xFFFF3F3F) |
1284*4882a593Smuzhiyun (last->burst_regs[EMC_MRW7_INDEX] &
1285*4882a593Smuzhiyun 0x0000C0C0), EMC_MRW7, 0);
1286*4882a593Smuzhiyun ccfifo_writel(emc,
1287*4882a593Smuzhiyun (next->burst_regs[EMC_MRW15_INDEX] &
1288*4882a593Smuzhiyun 0xFFFF0707) |
1289*4882a593Smuzhiyun (last->burst_regs[EMC_MRW15_INDEX] &
1290*4882a593Smuzhiyun 0x00003838), EMC_MRW15, 0);
1291*4882a593Smuzhiyun }
1292*4882a593Smuzhiyun
1293*4882a593Smuzhiyun if (opt_zcal_en_cc) {
1294*4882a593Smuzhiyun if (emc->num_devices < 2)
1295*4882a593Smuzhiyun ccfifo_writel(emc,
1296*4882a593Smuzhiyun 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT
1297*4882a593Smuzhiyun | EMC_ZQ_CAL_ZQ_CAL_CMD,
1298*4882a593Smuzhiyun EMC_ZQ_CAL, 0);
1299*4882a593Smuzhiyun else if (shared_zq_resistor)
1300*4882a593Smuzhiyun ccfifo_writel(emc,
1301*4882a593Smuzhiyun 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT
1302*4882a593Smuzhiyun | EMC_ZQ_CAL_ZQ_CAL_CMD,
1303*4882a593Smuzhiyun EMC_ZQ_CAL, 0);
1304*4882a593Smuzhiyun else
1305*4882a593Smuzhiyun ccfifo_writel(emc,
1306*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_CAL_CMD,
1307*4882a593Smuzhiyun EMC_ZQ_CAL, 0);
1308*4882a593Smuzhiyun }
1309*4882a593Smuzhiyun }
1310*4882a593Smuzhiyun }
1311*4882a593Smuzhiyun
1312*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
1313*4882a593Smuzhiyun value = (1000 * fake->dram_timings[T_RP]) / src_clk_period;
1314*4882a593Smuzhiyun ccfifo_writel(emc, mr13_flip_fspop | 0x8, EMC_MRW3, value);
1315*4882a593Smuzhiyun ccfifo_writel(emc, 0, 0, tFC_lpddr4 / src_clk_period);
1316*4882a593Smuzhiyun }
1317*4882a593Smuzhiyun
1318*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4 || opt_dvfs_mode != MAN_SR) {
1319*4882a593Smuzhiyun delay = 30;
1320*4882a593Smuzhiyun
1321*4882a593Smuzhiyun if (cya_allow_ref_cc) {
1322*4882a593Smuzhiyun delay += (1000 * fake->dram_timings[T_RP]) /
1323*4882a593Smuzhiyun src_clk_period;
1324*4882a593Smuzhiyun delay += 4000 * fake->dram_timings[T_RFC];
1325*4882a593Smuzhiyun }
1326*4882a593Smuzhiyun
1327*4882a593Smuzhiyun ccfifo_writel(emc, emc_pin & ~(EMC_PIN_PIN_CKE_PER_DEV |
1328*4882a593Smuzhiyun EMC_PIN_PIN_CKEB |
1329*4882a593Smuzhiyun EMC_PIN_PIN_CKE),
1330*4882a593Smuzhiyun EMC_PIN, delay);
1331*4882a593Smuzhiyun }
1332*4882a593Smuzhiyun
1333*4882a593Smuzhiyun /* calculate reference delay multiplier */
1334*4882a593Smuzhiyun value = 1;
1335*4882a593Smuzhiyun
1336*4882a593Smuzhiyun if (ref_b4_sref_en)
1337*4882a593Smuzhiyun value++;
1338*4882a593Smuzhiyun
1339*4882a593Smuzhiyun if (cya_allow_ref_cc)
1340*4882a593Smuzhiyun value++;
1341*4882a593Smuzhiyun
1342*4882a593Smuzhiyun if (cya_issue_pc_ref)
1343*4882a593Smuzhiyun value++;
1344*4882a593Smuzhiyun
1345*4882a593Smuzhiyun if (dram_type != DRAM_TYPE_LPDDR4) {
1346*4882a593Smuzhiyun delay = ((1000 * fake->dram_timings[T_RP] / src_clk_period) +
1347*4882a593Smuzhiyun (1000 * fake->dram_timings[T_RFC] / src_clk_period));
1348*4882a593Smuzhiyun delay = value * delay + 20;
1349*4882a593Smuzhiyun } else {
1350*4882a593Smuzhiyun delay = 0;
1351*4882a593Smuzhiyun }
1352*4882a593Smuzhiyun
1353*4882a593Smuzhiyun /*
1354*4882a593Smuzhiyun * Step 11:
1355*4882a593Smuzhiyun * Ramp down.
1356*4882a593Smuzhiyun */
1357*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 11\n");
1358*4882a593Smuzhiyun
1359*4882a593Smuzhiyun ccfifo_writel(emc, 0x0, EMC_CFG_SYNC, delay);
1360*4882a593Smuzhiyun
1361*4882a593Smuzhiyun value = emc_dbg | EMC_DBG_WRITE_MUX_ACTIVE | EMC_DBG_WRITE_ACTIVE_ONLY;
1362*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_DBG, 0);
1363*4882a593Smuzhiyun
1364*4882a593Smuzhiyun ramp_down_wait = tegra210_emc_dvfs_power_ramp_down(emc, src_clk_period,
1365*4882a593Smuzhiyun 0);
1366*4882a593Smuzhiyun
1367*4882a593Smuzhiyun /*
1368*4882a593Smuzhiyun * Step 12:
1369*4882a593Smuzhiyun * And finally - trigger the clock change.
1370*4882a593Smuzhiyun */
1371*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 12\n");
1372*4882a593Smuzhiyun
1373*4882a593Smuzhiyun ccfifo_writel(emc, 1, EMC_STALL_THEN_EXE_AFTER_CLKCHANGE, 0);
1374*4882a593Smuzhiyun value &= ~EMC_DBG_WRITE_ACTIVE_ONLY;
1375*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_DBG, 0);
1376*4882a593Smuzhiyun
1377*4882a593Smuzhiyun /*
1378*4882a593Smuzhiyun * Step 13:
1379*4882a593Smuzhiyun * Ramp up.
1380*4882a593Smuzhiyun */
1381*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 13\n");
1382*4882a593Smuzhiyun
1383*4882a593Smuzhiyun ramp_up_wait = tegra210_emc_dvfs_power_ramp_up(emc, dst_clk_period, 0);
1384*4882a593Smuzhiyun ccfifo_writel(emc, emc_dbg, EMC_DBG, 0);
1385*4882a593Smuzhiyun
1386*4882a593Smuzhiyun /*
1387*4882a593Smuzhiyun * Step 14:
1388*4882a593Smuzhiyun * Bringup CKE pins.
1389*4882a593Smuzhiyun */
1390*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 14\n");
1391*4882a593Smuzhiyun
1392*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
1393*4882a593Smuzhiyun value = emc_pin | EMC_PIN_PIN_CKE;
1394*4882a593Smuzhiyun
1395*4882a593Smuzhiyun if (emc->num_devices <= 1)
1396*4882a593Smuzhiyun value &= ~(EMC_PIN_PIN_CKEB | EMC_PIN_PIN_CKE_PER_DEV);
1397*4882a593Smuzhiyun else
1398*4882a593Smuzhiyun value |= EMC_PIN_PIN_CKEB | EMC_PIN_PIN_CKE_PER_DEV;
1399*4882a593Smuzhiyun
1400*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_PIN, 0);
1401*4882a593Smuzhiyun }
1402*4882a593Smuzhiyun
1403*4882a593Smuzhiyun /*
1404*4882a593Smuzhiyun * Step 15: (two step 15s ??)
1405*4882a593Smuzhiyun * Calculate zqlatch wait time; has dependency on ramping times.
1406*4882a593Smuzhiyun */
1407*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 15\n");
1408*4882a593Smuzhiyun
1409*4882a593Smuzhiyun if (dst_clk_period <= zqcal_before_cc_cutoff) {
1410*4882a593Smuzhiyun s32 t = (s32)(ramp_up_wait + ramp_down_wait) /
1411*4882a593Smuzhiyun (s32)dst_clk_period;
1412*4882a593Smuzhiyun zq_latch_dvfs_wait_time = (s32)tZQCAL_lpddr4_fc_adj - t;
1413*4882a593Smuzhiyun } else {
1414*4882a593Smuzhiyun zq_latch_dvfs_wait_time = tZQCAL_lpddr4_fc_adj -
1415*4882a593Smuzhiyun div_o3(1000 * next->dram_timings[T_PDEX],
1416*4882a593Smuzhiyun dst_clk_period);
1417*4882a593Smuzhiyun }
1418*4882a593Smuzhiyun
1419*4882a593Smuzhiyun emc_dbg(emc, INFO, "tZQCAL_lpddr4_fc_adj = %u\n", tZQCAL_lpddr4_fc_adj);
1420*4882a593Smuzhiyun emc_dbg(emc, INFO, "dst_clk_period = %u\n",
1421*4882a593Smuzhiyun dst_clk_period);
1422*4882a593Smuzhiyun emc_dbg(emc, INFO, "next->dram_timings[T_PDEX] = %u\n",
1423*4882a593Smuzhiyun next->dram_timings[T_PDEX]);
1424*4882a593Smuzhiyun emc_dbg(emc, INFO, "zq_latch_dvfs_wait_time = %d\n",
1425*4882a593Smuzhiyun max_t(s32, 0, zq_latch_dvfs_wait_time));
1426*4882a593Smuzhiyun
1427*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4 && opt_zcal_en_cc) {
1428*4882a593Smuzhiyun delay = div_o3(1000 * next->dram_timings[T_PDEX],
1429*4882a593Smuzhiyun dst_clk_period);
1430*4882a593Smuzhiyun
1431*4882a593Smuzhiyun if (emc->num_devices < 2) {
1432*4882a593Smuzhiyun if (dst_clk_period > zqcal_before_cc_cutoff)
1433*4882a593Smuzhiyun ccfifo_writel(emc,
1434*4882a593Smuzhiyun 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1435*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_CAL_CMD, EMC_ZQ_CAL,
1436*4882a593Smuzhiyun delay);
1437*4882a593Smuzhiyun
1438*4882a593Smuzhiyun value = (mr13_flip_fspop & 0xfffffff7) | 0x0c000000;
1439*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_MRW3, delay);
1440*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
1441*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_REF, 0);
1442*4882a593Smuzhiyun ccfifo_writel(emc, 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1443*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_LATCH_CMD,
1444*4882a593Smuzhiyun EMC_ZQ_CAL,
1445*4882a593Smuzhiyun max_t(s32, 0, zq_latch_dvfs_wait_time));
1446*4882a593Smuzhiyun } else if (shared_zq_resistor) {
1447*4882a593Smuzhiyun if (dst_clk_period > zqcal_before_cc_cutoff)
1448*4882a593Smuzhiyun ccfifo_writel(emc,
1449*4882a593Smuzhiyun 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1450*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_CAL_CMD, EMC_ZQ_CAL,
1451*4882a593Smuzhiyun delay);
1452*4882a593Smuzhiyun
1453*4882a593Smuzhiyun ccfifo_writel(emc, 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1454*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_LATCH_CMD, EMC_ZQ_CAL,
1455*4882a593Smuzhiyun max_t(s32, 0, zq_latch_dvfs_wait_time) +
1456*4882a593Smuzhiyun delay);
1457*4882a593Smuzhiyun ccfifo_writel(emc, 1UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1458*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_LATCH_CMD,
1459*4882a593Smuzhiyun EMC_ZQ_CAL, 0);
1460*4882a593Smuzhiyun
1461*4882a593Smuzhiyun value = (mr13_flip_fspop & 0xfffffff7) | 0x0c000000;
1462*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_MRW3, 0);
1463*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
1464*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_REF, 0);
1465*4882a593Smuzhiyun
1466*4882a593Smuzhiyun ccfifo_writel(emc, 1UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1467*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_LATCH_CMD, EMC_ZQ_CAL,
1468*4882a593Smuzhiyun tZQCAL_lpddr4 / dst_clk_period);
1469*4882a593Smuzhiyun } else {
1470*4882a593Smuzhiyun if (dst_clk_period > zqcal_before_cc_cutoff)
1471*4882a593Smuzhiyun ccfifo_writel(emc, EMC_ZQ_CAL_ZQ_CAL_CMD,
1472*4882a593Smuzhiyun EMC_ZQ_CAL, delay);
1473*4882a593Smuzhiyun
1474*4882a593Smuzhiyun value = (mr13_flip_fspop & 0xfffffff7) | 0x0c000000;
1475*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_MRW3, delay);
1476*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
1477*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_REF, 0);
1478*4882a593Smuzhiyun
1479*4882a593Smuzhiyun ccfifo_writel(emc, EMC_ZQ_CAL_ZQ_LATCH_CMD, EMC_ZQ_CAL,
1480*4882a593Smuzhiyun max_t(s32, 0, zq_latch_dvfs_wait_time));
1481*4882a593Smuzhiyun }
1482*4882a593Smuzhiyun }
1483*4882a593Smuzhiyun
1484*4882a593Smuzhiyun /* WAR: delay for zqlatch */
1485*4882a593Smuzhiyun ccfifo_writel(emc, 0, 0, 10);
1486*4882a593Smuzhiyun
1487*4882a593Smuzhiyun /*
1488*4882a593Smuzhiyun * Step 16:
1489*4882a593Smuzhiyun * LPDDR4 Conditional Training Kickoff. Removed.
1490*4882a593Smuzhiyun */
1491*4882a593Smuzhiyun
1492*4882a593Smuzhiyun /*
1493*4882a593Smuzhiyun * Step 17:
1494*4882a593Smuzhiyun * MANSR exit self refresh.
1495*4882a593Smuzhiyun */
1496*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 17\n");
1497*4882a593Smuzhiyun
1498*4882a593Smuzhiyun if (opt_dvfs_mode == MAN_SR && dram_type != DRAM_TYPE_LPDDR4)
1499*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
1500*4882a593Smuzhiyun
1501*4882a593Smuzhiyun /*
1502*4882a593Smuzhiyun * Step 18:
1503*4882a593Smuzhiyun * Send MRWs to LPDDR3/DDR3.
1504*4882a593Smuzhiyun */
1505*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 18\n");
1506*4882a593Smuzhiyun
1507*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR2) {
1508*4882a593Smuzhiyun ccfifo_writel(emc, next->emc_mrw2, EMC_MRW2, 0);
1509*4882a593Smuzhiyun ccfifo_writel(emc, next->emc_mrw, EMC_MRW, 0);
1510*4882a593Smuzhiyun if (is_lpddr3)
1511*4882a593Smuzhiyun ccfifo_writel(emc, next->emc_mrw4, EMC_MRW4, 0);
1512*4882a593Smuzhiyun } else if (dram_type == DRAM_TYPE_DDR3) {
1513*4882a593Smuzhiyun if (opt_dll_mode)
1514*4882a593Smuzhiyun ccfifo_writel(emc, next->emc_emrs &
1515*4882a593Smuzhiyun ~EMC_EMRS_USE_EMRS_LONG_CNT, EMC_EMRS, 0);
1516*4882a593Smuzhiyun ccfifo_writel(emc, next->emc_emrs2 &
1517*4882a593Smuzhiyun ~EMC_EMRS2_USE_EMRS2_LONG_CNT, EMC_EMRS2, 0);
1518*4882a593Smuzhiyun ccfifo_writel(emc, next->emc_mrs |
1519*4882a593Smuzhiyun EMC_EMRS_USE_EMRS_LONG_CNT, EMC_MRS, 0);
1520*4882a593Smuzhiyun }
1521*4882a593Smuzhiyun
1522*4882a593Smuzhiyun /*
1523*4882a593Smuzhiyun * Step 19:
1524*4882a593Smuzhiyun * ZQCAL for LPDDR3/DDR3
1525*4882a593Smuzhiyun */
1526*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 19\n");
1527*4882a593Smuzhiyun
1528*4882a593Smuzhiyun if (opt_zcal_en_cc) {
1529*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR2) {
1530*4882a593Smuzhiyun value = opt_cc_short_zcal ? 90000 : 360000;
1531*4882a593Smuzhiyun value = div_o3(value, dst_clk_period);
1532*4882a593Smuzhiyun value = value <<
1533*4882a593Smuzhiyun EMC_MRS_WAIT_CNT2_MRS_EXT2_WAIT_CNT_SHIFT |
1534*4882a593Smuzhiyun value <<
1535*4882a593Smuzhiyun EMC_MRS_WAIT_CNT2_MRS_EXT1_WAIT_CNT_SHIFT;
1536*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_MRS_WAIT_CNT2, 0);
1537*4882a593Smuzhiyun
1538*4882a593Smuzhiyun value = opt_cc_short_zcal ? 0x56 : 0xab;
1539*4882a593Smuzhiyun ccfifo_writel(emc, 2 << EMC_MRW_MRW_DEV_SELECTN_SHIFT |
1540*4882a593Smuzhiyun EMC_MRW_USE_MRW_EXT_CNT |
1541*4882a593Smuzhiyun 10 << EMC_MRW_MRW_MA_SHIFT |
1542*4882a593Smuzhiyun value << EMC_MRW_MRW_OP_SHIFT,
1543*4882a593Smuzhiyun EMC_MRW, 0);
1544*4882a593Smuzhiyun
1545*4882a593Smuzhiyun if (emc->num_devices > 1) {
1546*4882a593Smuzhiyun value = 1 << EMC_MRW_MRW_DEV_SELECTN_SHIFT |
1547*4882a593Smuzhiyun EMC_MRW_USE_MRW_EXT_CNT |
1548*4882a593Smuzhiyun 10 << EMC_MRW_MRW_MA_SHIFT |
1549*4882a593Smuzhiyun value << EMC_MRW_MRW_OP_SHIFT;
1550*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_MRW, 0);
1551*4882a593Smuzhiyun }
1552*4882a593Smuzhiyun } else if (dram_type == DRAM_TYPE_DDR3) {
1553*4882a593Smuzhiyun value = opt_cc_short_zcal ? 0 : EMC_ZQ_CAL_LONG;
1554*4882a593Smuzhiyun
1555*4882a593Smuzhiyun ccfifo_writel(emc, value |
1556*4882a593Smuzhiyun 2 << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1557*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_CAL_CMD, EMC_ZQ_CAL,
1558*4882a593Smuzhiyun 0);
1559*4882a593Smuzhiyun
1560*4882a593Smuzhiyun if (emc->num_devices > 1) {
1561*4882a593Smuzhiyun value = value | 1 << EMC_ZQ_CAL_DEV_SEL_SHIFT |
1562*4882a593Smuzhiyun EMC_ZQ_CAL_ZQ_CAL_CMD;
1563*4882a593Smuzhiyun ccfifo_writel(emc, value, EMC_ZQ_CAL, 0);
1564*4882a593Smuzhiyun }
1565*4882a593Smuzhiyun }
1566*4882a593Smuzhiyun }
1567*4882a593Smuzhiyun
1568*4882a593Smuzhiyun if (bg_reg_mode_change) {
1569*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
1570*4882a593Smuzhiyun
1571*4882a593Smuzhiyun if (ramp_up_wait <= 1250000)
1572*4882a593Smuzhiyun delay = (1250000 - ramp_up_wait) / dst_clk_period;
1573*4882a593Smuzhiyun else
1574*4882a593Smuzhiyun delay = 0;
1575*4882a593Smuzhiyun
1576*4882a593Smuzhiyun ccfifo_writel(emc,
1577*4882a593Smuzhiyun next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX],
1578*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0, delay);
1579*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
1580*4882a593Smuzhiyun }
1581*4882a593Smuzhiyun
1582*4882a593Smuzhiyun /*
1583*4882a593Smuzhiyun * Step 20:
1584*4882a593Smuzhiyun * Issue ref and optional QRST.
1585*4882a593Smuzhiyun */
1586*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 20\n");
1587*4882a593Smuzhiyun
1588*4882a593Smuzhiyun if (dram_type != DRAM_TYPE_LPDDR4)
1589*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_REF, 0);
1590*4882a593Smuzhiyun
1591*4882a593Smuzhiyun if (opt_do_sw_qrst) {
1592*4882a593Smuzhiyun ccfifo_writel(emc, 1, EMC_ISSUE_QRST, 0);
1593*4882a593Smuzhiyun ccfifo_writel(emc, 0, EMC_ISSUE_QRST, 2);
1594*4882a593Smuzhiyun }
1595*4882a593Smuzhiyun
1596*4882a593Smuzhiyun /*
1597*4882a593Smuzhiyun * Step 21:
1598*4882a593Smuzhiyun * Restore ZCAL and ZCAL interval.
1599*4882a593Smuzhiyun */
1600*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 21\n");
1601*4882a593Smuzhiyun
1602*4882a593Smuzhiyun if (save_restore_clkstop_pd || opt_zcal_en_cc) {
1603*4882a593Smuzhiyun ccfifo_writel(emc, emc_dbg | EMC_DBG_WRITE_MUX_ACTIVE,
1604*4882a593Smuzhiyun EMC_DBG, 0);
1605*4882a593Smuzhiyun if (opt_zcal_en_cc && dram_type != DRAM_TYPE_LPDDR4)
1606*4882a593Smuzhiyun ccfifo_writel(emc, next->burst_regs[EMC_ZCAL_INTERVAL_INDEX],
1607*4882a593Smuzhiyun EMC_ZCAL_INTERVAL, 0);
1608*4882a593Smuzhiyun
1609*4882a593Smuzhiyun if (save_restore_clkstop_pd)
1610*4882a593Smuzhiyun ccfifo_writel(emc, next->burst_regs[EMC_CFG_INDEX] &
1611*4882a593Smuzhiyun ~EMC_CFG_DYN_SELF_REF,
1612*4882a593Smuzhiyun EMC_CFG, 0);
1613*4882a593Smuzhiyun ccfifo_writel(emc, emc_dbg, EMC_DBG, 0);
1614*4882a593Smuzhiyun }
1615*4882a593Smuzhiyun
1616*4882a593Smuzhiyun /*
1617*4882a593Smuzhiyun * Step 22:
1618*4882a593Smuzhiyun * Restore EMC_CFG_PIPE_CLK.
1619*4882a593Smuzhiyun */
1620*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 22\n");
1621*4882a593Smuzhiyun
1622*4882a593Smuzhiyun ccfifo_writel(emc, emc_cfg_pipe_clk, EMC_CFG_PIPE_CLK, 0);
1623*4882a593Smuzhiyun
1624*4882a593Smuzhiyun if (bg_reg_mode_change) {
1625*4882a593Smuzhiyun if (enable_bg_reg)
1626*4882a593Smuzhiyun emc_writel(emc,
1627*4882a593Smuzhiyun next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
1628*4882a593Smuzhiyun ~EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD,
1629*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0);
1630*4882a593Smuzhiyun else
1631*4882a593Smuzhiyun emc_writel(emc,
1632*4882a593Smuzhiyun next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
1633*4882a593Smuzhiyun ~EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD,
1634*4882a593Smuzhiyun EMC_PMACRO_BG_BIAS_CTRL_0);
1635*4882a593Smuzhiyun }
1636*4882a593Smuzhiyun
1637*4882a593Smuzhiyun /*
1638*4882a593Smuzhiyun * Step 23:
1639*4882a593Smuzhiyun */
1640*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 23\n");
1641*4882a593Smuzhiyun
1642*4882a593Smuzhiyun value = emc_readl(emc, EMC_CFG_DIG_DLL);
1643*4882a593Smuzhiyun value |= EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_TRAFFIC;
1644*4882a593Smuzhiyun value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_RW_UNTIL_LOCK;
1645*4882a593Smuzhiyun value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_UNTIL_LOCK;
1646*4882a593Smuzhiyun value &= ~EMC_CFG_DIG_DLL_CFG_DLL_EN;
1647*4882a593Smuzhiyun value = (value & ~EMC_CFG_DIG_DLL_CFG_DLL_MODE_MASK) |
1648*4882a593Smuzhiyun (2 << EMC_CFG_DIG_DLL_CFG_DLL_MODE_SHIFT);
1649*4882a593Smuzhiyun emc_writel(emc, value, EMC_CFG_DIG_DLL);
1650*4882a593Smuzhiyun
1651*4882a593Smuzhiyun tegra210_emc_do_clock_change(emc, clksrc);
1652*4882a593Smuzhiyun
1653*4882a593Smuzhiyun /*
1654*4882a593Smuzhiyun * Step 24:
1655*4882a593Smuzhiyun * Save training results. Removed.
1656*4882a593Smuzhiyun */
1657*4882a593Smuzhiyun
1658*4882a593Smuzhiyun /*
1659*4882a593Smuzhiyun * Step 25:
1660*4882a593Smuzhiyun * Program MC updown registers.
1661*4882a593Smuzhiyun */
1662*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 25\n");
1663*4882a593Smuzhiyun
1664*4882a593Smuzhiyun if (next->rate > last->rate) {
1665*4882a593Smuzhiyun for (i = 0; i < next->num_up_down; i++)
1666*4882a593Smuzhiyun mc_writel(emc->mc, next->la_scale_regs[i],
1667*4882a593Smuzhiyun emc->offsets->la_scale[i]);
1668*4882a593Smuzhiyun
1669*4882a593Smuzhiyun tegra210_emc_timing_update(emc);
1670*4882a593Smuzhiyun }
1671*4882a593Smuzhiyun
1672*4882a593Smuzhiyun /*
1673*4882a593Smuzhiyun * Step 26:
1674*4882a593Smuzhiyun * Restore ZCAL registers.
1675*4882a593Smuzhiyun */
1676*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 26\n");
1677*4882a593Smuzhiyun
1678*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR4) {
1679*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
1680*4882a593Smuzhiyun emc_writel(emc, next->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX],
1681*4882a593Smuzhiyun EMC_ZCAL_WAIT_CNT);
1682*4882a593Smuzhiyun emc_writel(emc, next->burst_regs[EMC_ZCAL_INTERVAL_INDEX],
1683*4882a593Smuzhiyun EMC_ZCAL_INTERVAL);
1684*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
1685*4882a593Smuzhiyun }
1686*4882a593Smuzhiyun
1687*4882a593Smuzhiyun if (dram_type != DRAM_TYPE_LPDDR4 && opt_zcal_en_cc &&
1688*4882a593Smuzhiyun !opt_short_zcal && opt_cc_short_zcal) {
1689*4882a593Smuzhiyun udelay(2);
1690*4882a593Smuzhiyun
1691*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
1692*4882a593Smuzhiyun if (dram_type == DRAM_TYPE_LPDDR2)
1693*4882a593Smuzhiyun emc_writel(emc, next->burst_regs[EMC_MRS_WAIT_CNT_INDEX],
1694*4882a593Smuzhiyun EMC_MRS_WAIT_CNT);
1695*4882a593Smuzhiyun else if (dram_type == DRAM_TYPE_DDR3)
1696*4882a593Smuzhiyun emc_writel(emc, next->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX],
1697*4882a593Smuzhiyun EMC_ZCAL_WAIT_CNT);
1698*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
1699*4882a593Smuzhiyun }
1700*4882a593Smuzhiyun
1701*4882a593Smuzhiyun /*
1702*4882a593Smuzhiyun * Step 27:
1703*4882a593Smuzhiyun * Restore EMC_CFG, FDPD registers.
1704*4882a593Smuzhiyun */
1705*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 27\n");
1706*4882a593Smuzhiyun
1707*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
1708*4882a593Smuzhiyun emc_writel(emc, next->burst_regs[EMC_CFG_INDEX], EMC_CFG);
1709*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
1710*4882a593Smuzhiyun emc_writel(emc, next->emc_fdpd_ctrl_cmd_no_ramp,
1711*4882a593Smuzhiyun EMC_FDPD_CTRL_CMD_NO_RAMP);
1712*4882a593Smuzhiyun emc_writel(emc, next->emc_sel_dpd_ctrl, EMC_SEL_DPD_CTRL);
1713*4882a593Smuzhiyun
1714*4882a593Smuzhiyun /*
1715*4882a593Smuzhiyun * Step 28:
1716*4882a593Smuzhiyun * Training recover. Removed.
1717*4882a593Smuzhiyun */
1718*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 28\n");
1719*4882a593Smuzhiyun
1720*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ACTIVE);
1721*4882a593Smuzhiyun emc_writel(emc,
1722*4882a593Smuzhiyun next->burst_regs[EMC_PMACRO_AUTOCAL_CFG_COMMON_INDEX],
1723*4882a593Smuzhiyun EMC_PMACRO_AUTOCAL_CFG_COMMON);
1724*4882a593Smuzhiyun tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
1725*4882a593Smuzhiyun
1726*4882a593Smuzhiyun /*
1727*4882a593Smuzhiyun * Step 29:
1728*4882a593Smuzhiyun * Power fix WAR.
1729*4882a593Smuzhiyun */
1730*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 29\n");
1731*4882a593Smuzhiyun
1732*4882a593Smuzhiyun emc_writel(emc, EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE0 |
1733*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE1 |
1734*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE2 |
1735*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE3 |
1736*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE4 |
1737*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE5 |
1738*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE6 |
1739*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE7,
1740*4882a593Smuzhiyun EMC_PMACRO_CFG_PM_GLOBAL_0);
1741*4882a593Smuzhiyun emc_writel(emc, EMC_PMACRO_TRAINING_CTRL_0_CH0_TRAINING_E_WRPTR,
1742*4882a593Smuzhiyun EMC_PMACRO_TRAINING_CTRL_0);
1743*4882a593Smuzhiyun emc_writel(emc, EMC_PMACRO_TRAINING_CTRL_1_CH1_TRAINING_E_WRPTR,
1744*4882a593Smuzhiyun EMC_PMACRO_TRAINING_CTRL_1);
1745*4882a593Smuzhiyun emc_writel(emc, 0, EMC_PMACRO_CFG_PM_GLOBAL_0);
1746*4882a593Smuzhiyun
1747*4882a593Smuzhiyun /*
1748*4882a593Smuzhiyun * Step 30:
1749*4882a593Smuzhiyun * Re-enable autocal.
1750*4882a593Smuzhiyun */
1751*4882a593Smuzhiyun emc_dbg(emc, STEPS, "Step 30: Re-enable DLL and AUTOCAL\n");
1752*4882a593Smuzhiyun
1753*4882a593Smuzhiyun if (next->burst_regs[EMC_CFG_DIG_DLL_INDEX] & EMC_CFG_DIG_DLL_CFG_DLL_EN) {
1754*4882a593Smuzhiyun value = emc_readl(emc, EMC_CFG_DIG_DLL);
1755*4882a593Smuzhiyun value |= EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_TRAFFIC;
1756*4882a593Smuzhiyun value |= EMC_CFG_DIG_DLL_CFG_DLL_EN;
1757*4882a593Smuzhiyun value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_RW_UNTIL_LOCK;
1758*4882a593Smuzhiyun value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_UNTIL_LOCK;
1759*4882a593Smuzhiyun value = (value & ~EMC_CFG_DIG_DLL_CFG_DLL_MODE_MASK) |
1760*4882a593Smuzhiyun (2 << EMC_CFG_DIG_DLL_CFG_DLL_MODE_SHIFT);
1761*4882a593Smuzhiyun emc_writel(emc, value, EMC_CFG_DIG_DLL);
1762*4882a593Smuzhiyun tegra210_emc_timing_update(emc);
1763*4882a593Smuzhiyun }
1764*4882a593Smuzhiyun
1765*4882a593Smuzhiyun emc_writel(emc, next->emc_auto_cal_config, EMC_AUTO_CAL_CONFIG);
1766*4882a593Smuzhiyun
1767*4882a593Smuzhiyun /* Done! Yay. */
1768*4882a593Smuzhiyun }
1769*4882a593Smuzhiyun
1770*4882a593Smuzhiyun const struct tegra210_emc_sequence tegra210_emc_r21021 = {
1771*4882a593Smuzhiyun .revision = 0x7,
1772*4882a593Smuzhiyun .set_clock = tegra210_emc_r21021_set_clock,
1773*4882a593Smuzhiyun .periodic_compensation = tegra210_emc_r21021_periodic_compensation,
1774*4882a593Smuzhiyun };
1775