xref: /openbmc/linux/drivers/memory/tegra/tegra210-emc-cc-r21021.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
19b9d8632SJoseph Lo // SPDX-License-Identifier: GPL-2.0
29b9d8632SJoseph Lo /*
39b9d8632SJoseph Lo  * Copyright (c) 2014-2020, NVIDIA CORPORATION.  All rights reserved.
49b9d8632SJoseph Lo  */
59b9d8632SJoseph Lo 
69b9d8632SJoseph Lo #include <linux/kernel.h>
79b9d8632SJoseph Lo #include <linux/io.h>
89b9d8632SJoseph Lo #include <linux/clk.h>
99b9d8632SJoseph Lo #include <linux/delay.h>
109b9d8632SJoseph Lo #include <linux/of.h>
119b9d8632SJoseph Lo 
129b9d8632SJoseph Lo #include <soc/tegra/mc.h>
139b9d8632SJoseph Lo 
149b9d8632SJoseph Lo #include "tegra210-emc.h"
159b9d8632SJoseph Lo #include "tegra210-mc.h"
169b9d8632SJoseph Lo 
179b9d8632SJoseph Lo /*
189b9d8632SJoseph Lo  * Enable flags for specifying verbosity.
199b9d8632SJoseph Lo  */
209b9d8632SJoseph Lo #define INFO            (1 << 0)
219b9d8632SJoseph Lo #define STEPS           (1 << 1)
229b9d8632SJoseph Lo #define SUB_STEPS       (1 << 2)
239b9d8632SJoseph Lo #define PRELOCK         (1 << 3)
249b9d8632SJoseph Lo #define PRELOCK_STEPS   (1 << 4)
259b9d8632SJoseph Lo #define ACTIVE_EN       (1 << 5)
269b9d8632SJoseph Lo #define PRAMP_UP        (1 << 6)
279b9d8632SJoseph Lo #define PRAMP_DN        (1 << 7)
289b9d8632SJoseph Lo #define EMA_WRITES      (1 << 10)
299b9d8632SJoseph Lo #define EMA_UPDATES     (1 << 11)
309b9d8632SJoseph Lo #define PER_TRAIN       (1 << 16)
319b9d8632SJoseph Lo #define CC_PRINT        (1 << 17)
329b9d8632SJoseph Lo #define CCFIFO          (1 << 29)
339b9d8632SJoseph Lo #define REGS            (1 << 30)
349b9d8632SJoseph Lo #define REG_LISTS       (1 << 31)
359b9d8632SJoseph Lo 
369b9d8632SJoseph Lo #define emc_dbg(emc, flags, ...) dev_dbg(emc->dev, __VA_ARGS__)
379b9d8632SJoseph Lo 
389b9d8632SJoseph Lo #define DVFS_CLOCK_CHANGE_VERSION	21021
399b9d8632SJoseph Lo #define EMC_PRELOCK_VERSION		2101
409b9d8632SJoseph Lo 
419b9d8632SJoseph Lo enum {
429b9d8632SJoseph Lo 	DVFS_SEQUENCE = 1,
439b9d8632SJoseph Lo 	WRITE_TRAINING_SEQUENCE = 2,
449b9d8632SJoseph Lo 	PERIODIC_TRAINING_SEQUENCE = 3,
459b9d8632SJoseph Lo 	DVFS_PT1 = 10,
469b9d8632SJoseph Lo 	DVFS_UPDATE = 11,
479b9d8632SJoseph Lo 	TRAINING_PT1 = 12,
489b9d8632SJoseph Lo 	TRAINING_UPDATE = 13,
499b9d8632SJoseph Lo 	PERIODIC_TRAINING_UPDATE = 14
509b9d8632SJoseph Lo };
519b9d8632SJoseph Lo 
529b9d8632SJoseph Lo /*
539b9d8632SJoseph Lo  * PTFV defines - basically just indexes into the per table PTFV array.
549b9d8632SJoseph Lo  */
559b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C0D0U0_INDEX		0
569b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C0D0U1_INDEX		1
579b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C0D1U0_INDEX		2
589b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C0D1U1_INDEX		3
599b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C1D0U0_INDEX		4
609b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C1D0U1_INDEX		5
619b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C1D1U0_INDEX		6
629b9d8632SJoseph Lo #define PTFV_DQSOSC_MOVAVG_C1D1U1_INDEX		7
639b9d8632SJoseph Lo #define PTFV_DVFS_SAMPLES_INDEX			9
649b9d8632SJoseph Lo #define PTFV_MOVAVG_WEIGHT_INDEX		10
659b9d8632SJoseph Lo #define PTFV_CONFIG_CTRL_INDEX			11
669b9d8632SJoseph Lo 
679b9d8632SJoseph Lo #define PTFV_CONFIG_CTRL_USE_PREVIOUS_EMA	(1 << 0)
689b9d8632SJoseph Lo 
699b9d8632SJoseph Lo /*
709b9d8632SJoseph Lo  * Do arithmetic in fixed point.
719b9d8632SJoseph Lo  */
729b9d8632SJoseph Lo #define MOVAVG_PRECISION_FACTOR		100
739b9d8632SJoseph Lo 
749b9d8632SJoseph Lo /*
759b9d8632SJoseph Lo  * The division portion of the average operation.
769b9d8632SJoseph Lo  */
779b9d8632SJoseph Lo #define __AVERAGE_PTFV(dev)						\
789b9d8632SJoseph Lo 	({ next->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] =	\
799b9d8632SJoseph Lo 	   next->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] /	\
809b9d8632SJoseph Lo 	   next->ptfv_list[PTFV_DVFS_SAMPLES_INDEX]; })
819b9d8632SJoseph Lo 
829b9d8632SJoseph Lo /*
839b9d8632SJoseph Lo  * Convert val to fixed point and add it to the temporary average.
849b9d8632SJoseph Lo  */
859b9d8632SJoseph Lo #define __INCREMENT_PTFV(dev, val)					\
869b9d8632SJoseph Lo 	({ next->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] +=	\
879b9d8632SJoseph Lo 	   ((val) * MOVAVG_PRECISION_FACTOR); })
889b9d8632SJoseph Lo 
899b9d8632SJoseph Lo /*
909b9d8632SJoseph Lo  * Convert a moving average back to integral form and return the value.
919b9d8632SJoseph Lo  */
929b9d8632SJoseph Lo #define __MOVAVG_AC(timing, dev)					\
939b9d8632SJoseph Lo 	((timing)->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX] /	\
949b9d8632SJoseph Lo 	 MOVAVG_PRECISION_FACTOR)
959b9d8632SJoseph Lo 
969b9d8632SJoseph Lo /* Weighted update. */
979b9d8632SJoseph Lo #define __WEIGHTED_UPDATE_PTFV(dev, nval)				\
989b9d8632SJoseph Lo 	do {								\
999b9d8632SJoseph Lo 		int w = PTFV_MOVAVG_WEIGHT_INDEX;			\
1009b9d8632SJoseph Lo 		int dqs = PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX;		\
1019b9d8632SJoseph Lo 									\
1029b9d8632SJoseph Lo 		next->ptfv_list[dqs] =					\
1039b9d8632SJoseph Lo 			((nval * MOVAVG_PRECISION_FACTOR) +		\
1049b9d8632SJoseph Lo 			 (next->ptfv_list[dqs] *			\
1059b9d8632SJoseph Lo 			  next->ptfv_list[w])) /			\
1069b9d8632SJoseph Lo 			(next->ptfv_list[w] + 1);			\
1079b9d8632SJoseph Lo 									\
1089b9d8632SJoseph Lo 		emc_dbg(emc, EMA_UPDATES, "%s: (s=%lu) EMA: %u\n",	\
1099b9d8632SJoseph Lo 			__stringify(dev), nval, next->ptfv_list[dqs]);	\
1109b9d8632SJoseph Lo 	} while (0)
1119b9d8632SJoseph Lo 
1129b9d8632SJoseph Lo /* Access a particular average. */
1139b9d8632SJoseph Lo #define __MOVAVG(timing, dev)                      \
1149b9d8632SJoseph Lo 	((timing)->ptfv_list[PTFV_DQSOSC_MOVAVG_ ## dev ## _INDEX])
1159b9d8632SJoseph Lo 
update_clock_tree_delay(struct tegra210_emc * emc,int type)1169b9d8632SJoseph Lo static u32 update_clock_tree_delay(struct tegra210_emc *emc, int type)
1179b9d8632SJoseph Lo {
1189b9d8632SJoseph Lo 	bool periodic_training_update = type == PERIODIC_TRAINING_UPDATE;
1199b9d8632SJoseph Lo 	struct tegra210_emc_timing *last = emc->last;
1209b9d8632SJoseph Lo 	struct tegra210_emc_timing *next = emc->next;
1219b9d8632SJoseph Lo 	u32 last_timing_rate_mhz = last->rate / 1000;
1229b9d8632SJoseph Lo 	u32 next_timing_rate_mhz = next->rate / 1000;
1239b9d8632SJoseph Lo 	bool dvfs_update = type == DVFS_UPDATE;
1249b9d8632SJoseph Lo 	s32 tdel = 0, tmdel = 0, adel = 0;
1259b9d8632SJoseph Lo 	bool dvfs_pt1 = type == DVFS_PT1;
1269b9d8632SJoseph Lo 	unsigned long cval = 0;
1279b9d8632SJoseph Lo 	u32 temp[2][2], value;
1289b9d8632SJoseph Lo 	unsigned int i;
1299b9d8632SJoseph Lo 
1309b9d8632SJoseph Lo 	/*
1319b9d8632SJoseph Lo 	 * Dev0 MSB.
1329b9d8632SJoseph Lo 	 */
1339b9d8632SJoseph Lo 	if (dvfs_pt1 || periodic_training_update) {
1349b9d8632SJoseph Lo 		value = tegra210_emc_mrr_read(emc, 2, 19);
1359b9d8632SJoseph Lo 
1369b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++) {
1379b9d8632SJoseph Lo 			temp[i][0] = (value & 0x00ff) << 8;
1389b9d8632SJoseph Lo 			temp[i][1] = (value & 0xff00) << 0;
1399b9d8632SJoseph Lo 			value >>= 16;
1409b9d8632SJoseph Lo 		}
1419b9d8632SJoseph Lo 
1429b9d8632SJoseph Lo 		/*
1439b9d8632SJoseph Lo 		 * Dev0 LSB.
1449b9d8632SJoseph Lo 		 */
1459b9d8632SJoseph Lo 		value = tegra210_emc_mrr_read(emc, 2, 18);
1469b9d8632SJoseph Lo 
1479b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++) {
1489b9d8632SJoseph Lo 			temp[i][0] |= (value & 0x00ff) >> 0;
1499b9d8632SJoseph Lo 			temp[i][1] |= (value & 0xff00) >> 8;
1509b9d8632SJoseph Lo 			value >>= 16;
1519b9d8632SJoseph Lo 		}
1529b9d8632SJoseph Lo 	}
1539b9d8632SJoseph Lo 
1549b9d8632SJoseph Lo 	if (dvfs_pt1 || periodic_training_update) {
1559b9d8632SJoseph Lo 		cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
1569b9d8632SJoseph Lo 		cval *= 1000000;
1579b9d8632SJoseph Lo 		cval /= last_timing_rate_mhz * 2 * temp[0][0];
1589b9d8632SJoseph Lo 	}
1599b9d8632SJoseph Lo 
1609b9d8632SJoseph Lo 	if (dvfs_pt1)
1619b9d8632SJoseph Lo 		__INCREMENT_PTFV(C0D0U0, cval);
1629b9d8632SJoseph Lo 	else if (dvfs_update)
1639b9d8632SJoseph Lo 		__AVERAGE_PTFV(C0D0U0);
1649b9d8632SJoseph Lo 	else if (periodic_training_update)
1659b9d8632SJoseph Lo 		__WEIGHTED_UPDATE_PTFV(C0D0U0, cval);
1669b9d8632SJoseph Lo 
1679b9d8632SJoseph Lo 	if (dvfs_update || periodic_training_update) {
1689b9d8632SJoseph Lo 		tdel = next->current_dram_clktree[C0D0U0] -
1699b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D0U0);
1709b9d8632SJoseph Lo 		tmdel = (tdel < 0) ? -1 * tdel : tdel;
1719b9d8632SJoseph Lo 		adel = tmdel;
1729b9d8632SJoseph Lo 
1739b9d8632SJoseph Lo 		if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
1749b9d8632SJoseph Lo 		    next->tree_margin)
1759b9d8632SJoseph Lo 			next->current_dram_clktree[C0D0U0] =
1769b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D0U0);
1779b9d8632SJoseph Lo 	}
1789b9d8632SJoseph Lo 
1799b9d8632SJoseph Lo 	if (dvfs_pt1 || periodic_training_update) {
1809b9d8632SJoseph Lo 		cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
1819b9d8632SJoseph Lo 		cval *= 1000000;
1829b9d8632SJoseph Lo 		cval /= last_timing_rate_mhz * 2 * temp[0][1];
1839b9d8632SJoseph Lo 	}
1849b9d8632SJoseph Lo 
1859b9d8632SJoseph Lo 	if (dvfs_pt1)
1869b9d8632SJoseph Lo 		__INCREMENT_PTFV(C0D0U1, cval);
1879b9d8632SJoseph Lo 	else if (dvfs_update)
1889b9d8632SJoseph Lo 		__AVERAGE_PTFV(C0D0U1);
1899b9d8632SJoseph Lo 	else if (periodic_training_update)
1909b9d8632SJoseph Lo 		__WEIGHTED_UPDATE_PTFV(C0D0U1, cval);
1919b9d8632SJoseph Lo 
1929b9d8632SJoseph Lo 	if (dvfs_update || periodic_training_update) {
1939b9d8632SJoseph Lo 		tdel = next->current_dram_clktree[C0D0U1] -
1949b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D0U1);
1959b9d8632SJoseph Lo 		tmdel = (tdel < 0) ? -1 * tdel : tdel;
1969b9d8632SJoseph Lo 
1979b9d8632SJoseph Lo 		if (tmdel > adel)
1989b9d8632SJoseph Lo 			adel = tmdel;
1999b9d8632SJoseph Lo 
2009b9d8632SJoseph Lo 		if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
2019b9d8632SJoseph Lo 		    next->tree_margin)
2029b9d8632SJoseph Lo 			next->current_dram_clktree[C0D0U1] =
2039b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D0U1);
2049b9d8632SJoseph Lo 	}
2059b9d8632SJoseph Lo 
2069b9d8632SJoseph Lo 	if (emc->num_channels > 1) {
2079b9d8632SJoseph Lo 		if (dvfs_pt1 || periodic_training_update) {
2089b9d8632SJoseph Lo 			cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
2099b9d8632SJoseph Lo 			cval *= 1000000;
2109b9d8632SJoseph Lo 			cval /= last_timing_rate_mhz * 2 * temp[1][0];
2119b9d8632SJoseph Lo 		}
2129b9d8632SJoseph Lo 
2139b9d8632SJoseph Lo 		if (dvfs_pt1)
2149b9d8632SJoseph Lo 			__INCREMENT_PTFV(C1D0U0, cval);
2159b9d8632SJoseph Lo 		else if (dvfs_update)
2169b9d8632SJoseph Lo 			__AVERAGE_PTFV(C1D0U0);
2179b9d8632SJoseph Lo 		else if (periodic_training_update)
2189b9d8632SJoseph Lo 			__WEIGHTED_UPDATE_PTFV(C1D0U0, cval);
2199b9d8632SJoseph Lo 
2209b9d8632SJoseph Lo 		if (dvfs_update || periodic_training_update) {
2219b9d8632SJoseph Lo 			tdel = next->current_dram_clktree[C1D0U0] -
2229b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D0U0);
2239b9d8632SJoseph Lo 			tmdel = (tdel < 0) ? -1 * tdel : tdel;
2249b9d8632SJoseph Lo 
2259b9d8632SJoseph Lo 			if (tmdel > adel)
2269b9d8632SJoseph Lo 				adel = tmdel;
2279b9d8632SJoseph Lo 
2289b9d8632SJoseph Lo 			if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
2299b9d8632SJoseph Lo 			    next->tree_margin)
2309b9d8632SJoseph Lo 				next->current_dram_clktree[C1D0U0] =
2319b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D0U0);
2329b9d8632SJoseph Lo 		}
2339b9d8632SJoseph Lo 
2349b9d8632SJoseph Lo 		if (dvfs_pt1 || periodic_training_update) {
2359b9d8632SJoseph Lo 			cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
2369b9d8632SJoseph Lo 			cval *= 1000000;
2379b9d8632SJoseph Lo 			cval /= last_timing_rate_mhz * 2 * temp[1][1];
2389b9d8632SJoseph Lo 		}
2399b9d8632SJoseph Lo 
2409b9d8632SJoseph Lo 		if (dvfs_pt1)
2419b9d8632SJoseph Lo 			__INCREMENT_PTFV(C1D0U1, cval);
2429b9d8632SJoseph Lo 		else if (dvfs_update)
2439b9d8632SJoseph Lo 			__AVERAGE_PTFV(C1D0U1);
2449b9d8632SJoseph Lo 		else if (periodic_training_update)
2459b9d8632SJoseph Lo 			__WEIGHTED_UPDATE_PTFV(C1D0U1, cval);
2469b9d8632SJoseph Lo 
2479b9d8632SJoseph Lo 		if (dvfs_update || periodic_training_update) {
2489b9d8632SJoseph Lo 			tdel = next->current_dram_clktree[C1D0U1] -
2499b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D0U1);
2509b9d8632SJoseph Lo 			tmdel = (tdel < 0) ? -1 * tdel : tdel;
2519b9d8632SJoseph Lo 
2529b9d8632SJoseph Lo 			if (tmdel > adel)
2539b9d8632SJoseph Lo 				adel = tmdel;
2549b9d8632SJoseph Lo 
2559b9d8632SJoseph Lo 			if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
2569b9d8632SJoseph Lo 			    next->tree_margin)
2579b9d8632SJoseph Lo 				next->current_dram_clktree[C1D0U1] =
2589b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D0U1);
2599b9d8632SJoseph Lo 		}
2609b9d8632SJoseph Lo 	}
2619b9d8632SJoseph Lo 
2629b9d8632SJoseph Lo 	if (emc->num_devices < 2)
2639b9d8632SJoseph Lo 		goto done;
2649b9d8632SJoseph Lo 
2659b9d8632SJoseph Lo 	/*
2669b9d8632SJoseph Lo 	 * Dev1 MSB.
2679b9d8632SJoseph Lo 	 */
2689b9d8632SJoseph Lo 	if (dvfs_pt1 || periodic_training_update) {
2699b9d8632SJoseph Lo 		value = tegra210_emc_mrr_read(emc, 1, 19);
2709b9d8632SJoseph Lo 
2719b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++) {
2729b9d8632SJoseph Lo 			temp[i][0] = (value & 0x00ff) << 8;
2739b9d8632SJoseph Lo 			temp[i][1] = (value & 0xff00) << 0;
2749b9d8632SJoseph Lo 			value >>= 16;
2759b9d8632SJoseph Lo 		}
2769b9d8632SJoseph Lo 
2779b9d8632SJoseph Lo 		/*
2789b9d8632SJoseph Lo 		 * Dev1 LSB.
2799b9d8632SJoseph Lo 		 */
280*be4c5c6eSDiogo Ivo 		value = tegra210_emc_mrr_read(emc, 1, 18);
2819b9d8632SJoseph Lo 
2829b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++) {
2839b9d8632SJoseph Lo 			temp[i][0] |= (value & 0x00ff) >> 0;
2849b9d8632SJoseph Lo 			temp[i][1] |= (value & 0xff00) >> 8;
2859b9d8632SJoseph Lo 			value >>= 16;
2869b9d8632SJoseph Lo 		}
2879b9d8632SJoseph Lo 	}
2889b9d8632SJoseph Lo 
2899b9d8632SJoseph Lo 	if (dvfs_pt1 || periodic_training_update) {
2909b9d8632SJoseph Lo 		cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
2919b9d8632SJoseph Lo 		cval *= 1000000;
2929b9d8632SJoseph Lo 		cval /= last_timing_rate_mhz * 2 * temp[0][0];
2939b9d8632SJoseph Lo 	}
2949b9d8632SJoseph Lo 
2959b9d8632SJoseph Lo 	if (dvfs_pt1)
2969b9d8632SJoseph Lo 		__INCREMENT_PTFV(C0D1U0, cval);
2979b9d8632SJoseph Lo 	else if (dvfs_update)
2989b9d8632SJoseph Lo 		__AVERAGE_PTFV(C0D1U0);
2999b9d8632SJoseph Lo 	else if (periodic_training_update)
3009b9d8632SJoseph Lo 		__WEIGHTED_UPDATE_PTFV(C0D1U0, cval);
3019b9d8632SJoseph Lo 
3029b9d8632SJoseph Lo 	if (dvfs_update || periodic_training_update) {
3039b9d8632SJoseph Lo 		tdel = next->current_dram_clktree[C0D1U0] -
3049b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D1U0);
3059b9d8632SJoseph Lo 		tmdel = (tdel < 0) ? -1 * tdel : tdel;
3069b9d8632SJoseph Lo 
3079b9d8632SJoseph Lo 		if (tmdel > adel)
3089b9d8632SJoseph Lo 			adel = tmdel;
3099b9d8632SJoseph Lo 
3109b9d8632SJoseph Lo 		if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
3119b9d8632SJoseph Lo 		    next->tree_margin)
3129b9d8632SJoseph Lo 			next->current_dram_clktree[C0D1U0] =
3139b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D1U0);
3149b9d8632SJoseph Lo 	}
3159b9d8632SJoseph Lo 
3169b9d8632SJoseph Lo 	if (dvfs_pt1 || periodic_training_update) {
3179b9d8632SJoseph Lo 		cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
3189b9d8632SJoseph Lo 		cval *= 1000000;
3199b9d8632SJoseph Lo 		cval /= last_timing_rate_mhz * 2 * temp[0][1];
3209b9d8632SJoseph Lo 	}
3219b9d8632SJoseph Lo 
3229b9d8632SJoseph Lo 	if (dvfs_pt1)
3239b9d8632SJoseph Lo 		__INCREMENT_PTFV(C0D1U1, cval);
3249b9d8632SJoseph Lo 	else if (dvfs_update)
3259b9d8632SJoseph Lo 		__AVERAGE_PTFV(C0D1U1);
3269b9d8632SJoseph Lo 	else if (periodic_training_update)
3279b9d8632SJoseph Lo 		__WEIGHTED_UPDATE_PTFV(C0D1U1, cval);
3289b9d8632SJoseph Lo 
3299b9d8632SJoseph Lo 	if (dvfs_update || periodic_training_update) {
3309b9d8632SJoseph Lo 		tdel = next->current_dram_clktree[C0D1U1] -
3319b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D1U1);
3329b9d8632SJoseph Lo 		tmdel = (tdel < 0) ? -1 * tdel : tdel;
3339b9d8632SJoseph Lo 
3349b9d8632SJoseph Lo 		if (tmdel > adel)
3359b9d8632SJoseph Lo 			adel = tmdel;
3369b9d8632SJoseph Lo 
3379b9d8632SJoseph Lo 		if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
3389b9d8632SJoseph Lo 		    next->tree_margin)
3399b9d8632SJoseph Lo 			next->current_dram_clktree[C0D1U1] =
3409b9d8632SJoseph Lo 				__MOVAVG_AC(next, C0D1U1);
3419b9d8632SJoseph Lo 	}
3429b9d8632SJoseph Lo 
3439b9d8632SJoseph Lo 	if (emc->num_channels > 1) {
3449b9d8632SJoseph Lo 		if (dvfs_pt1 || periodic_training_update) {
3459b9d8632SJoseph Lo 			cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
3469b9d8632SJoseph Lo 			cval *= 1000000;
3479b9d8632SJoseph Lo 			cval /= last_timing_rate_mhz * 2 * temp[1][0];
3489b9d8632SJoseph Lo 		}
3499b9d8632SJoseph Lo 
3509b9d8632SJoseph Lo 		if (dvfs_pt1)
3519b9d8632SJoseph Lo 			__INCREMENT_PTFV(C1D1U0, cval);
3529b9d8632SJoseph Lo 		else if (dvfs_update)
3539b9d8632SJoseph Lo 			__AVERAGE_PTFV(C1D1U0);
3549b9d8632SJoseph Lo 		else if (periodic_training_update)
3559b9d8632SJoseph Lo 			__WEIGHTED_UPDATE_PTFV(C1D1U0, cval);
3569b9d8632SJoseph Lo 
3579b9d8632SJoseph Lo 		if (dvfs_update || periodic_training_update) {
3589b9d8632SJoseph Lo 			tdel = next->current_dram_clktree[C1D1U0] -
3599b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D1U0);
3609b9d8632SJoseph Lo 			tmdel = (tdel < 0) ? -1 * tdel : tdel;
3619b9d8632SJoseph Lo 
3629b9d8632SJoseph Lo 			if (tmdel > adel)
3639b9d8632SJoseph Lo 				adel = tmdel;
3649b9d8632SJoseph Lo 
3659b9d8632SJoseph Lo 			if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
3669b9d8632SJoseph Lo 			    next->tree_margin)
3679b9d8632SJoseph Lo 				next->current_dram_clktree[C1D1U0] =
3689b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D1U0);
3699b9d8632SJoseph Lo 		}
3709b9d8632SJoseph Lo 
3719b9d8632SJoseph Lo 		if (dvfs_pt1 || periodic_training_update) {
3729b9d8632SJoseph Lo 			cval = tegra210_emc_actual_osc_clocks(last->run_clocks);
3739b9d8632SJoseph Lo 			cval *= 1000000;
3749b9d8632SJoseph Lo 			cval /= last_timing_rate_mhz * 2 * temp[1][1];
3759b9d8632SJoseph Lo 		}
3769b9d8632SJoseph Lo 
3779b9d8632SJoseph Lo 		if (dvfs_pt1)
3789b9d8632SJoseph Lo 			__INCREMENT_PTFV(C1D1U1, cval);
3799b9d8632SJoseph Lo 		else if (dvfs_update)
3809b9d8632SJoseph Lo 			__AVERAGE_PTFV(C1D1U1);
3819b9d8632SJoseph Lo 		else if (periodic_training_update)
3829b9d8632SJoseph Lo 			__WEIGHTED_UPDATE_PTFV(C1D1U1, cval);
3839b9d8632SJoseph Lo 
3849b9d8632SJoseph Lo 		if (dvfs_update || periodic_training_update) {
3859b9d8632SJoseph Lo 			tdel = next->current_dram_clktree[C1D1U1] -
3869b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D1U1);
3879b9d8632SJoseph Lo 			tmdel = (tdel < 0) ? -1 * tdel : tdel;
3889b9d8632SJoseph Lo 
3899b9d8632SJoseph Lo 			if (tmdel > adel)
3909b9d8632SJoseph Lo 				adel = tmdel;
3919b9d8632SJoseph Lo 
3929b9d8632SJoseph Lo 			if (tmdel * 128 * next_timing_rate_mhz / 1000000 >
3939b9d8632SJoseph Lo 			    next->tree_margin)
3949b9d8632SJoseph Lo 				next->current_dram_clktree[C1D1U1] =
3959b9d8632SJoseph Lo 					__MOVAVG_AC(next, C1D1U1);
3969b9d8632SJoseph Lo 		}
3979b9d8632SJoseph Lo 	}
3989b9d8632SJoseph Lo 
3999b9d8632SJoseph Lo done:
4009b9d8632SJoseph Lo 	return adel;
4019b9d8632SJoseph Lo }
4029b9d8632SJoseph Lo 
periodic_compensation_handler(struct tegra210_emc * emc,u32 type,struct tegra210_emc_timing * last,struct tegra210_emc_timing * next)4039b9d8632SJoseph Lo static u32 periodic_compensation_handler(struct tegra210_emc *emc, u32 type,
4049b9d8632SJoseph Lo 					 struct tegra210_emc_timing *last,
4059b9d8632SJoseph Lo 					 struct tegra210_emc_timing *next)
4069b9d8632SJoseph Lo {
4079b9d8632SJoseph Lo #define __COPY_EMA(nt, lt, dev)						\
4089b9d8632SJoseph Lo 	({ __MOVAVG(nt, dev) = __MOVAVG(lt, dev) *			\
4099b9d8632SJoseph Lo 	   (nt)->ptfv_list[PTFV_DVFS_SAMPLES_INDEX]; })
4109b9d8632SJoseph Lo 
4119b9d8632SJoseph Lo 	u32 i, adel = 0, samples = next->ptfv_list[PTFV_DVFS_SAMPLES_INDEX];
4129b9d8632SJoseph Lo 	u32 delay;
4139b9d8632SJoseph Lo 
4149b9d8632SJoseph Lo 	delay = tegra210_emc_actual_osc_clocks(last->run_clocks);
4159b9d8632SJoseph Lo 	delay *= 1000;
4169b9d8632SJoseph Lo 	delay = 2 + (delay / last->rate);
4179b9d8632SJoseph Lo 
4189b9d8632SJoseph Lo 	if (!next->periodic_training)
4199b9d8632SJoseph Lo 		return 0;
4209b9d8632SJoseph Lo 
4219b9d8632SJoseph Lo 	if (type == DVFS_SEQUENCE) {
4229b9d8632SJoseph Lo 		if (last->periodic_training &&
4239b9d8632SJoseph Lo 		    (next->ptfv_list[PTFV_CONFIG_CTRL_INDEX] &
4249b9d8632SJoseph Lo 		     PTFV_CONFIG_CTRL_USE_PREVIOUS_EMA)) {
4259b9d8632SJoseph Lo 			/*
4269b9d8632SJoseph Lo 			 * If the previous frequency was using periodic
4279b9d8632SJoseph Lo 			 * calibration then we can reuse the previous
4289b9d8632SJoseph Lo 			 * frequencies EMA data.
4299b9d8632SJoseph Lo 			 */
4309b9d8632SJoseph Lo 			__COPY_EMA(next, last, C0D0U0);
4319b9d8632SJoseph Lo 			__COPY_EMA(next, last, C0D0U1);
4329b9d8632SJoseph Lo 			__COPY_EMA(next, last, C1D0U0);
4339b9d8632SJoseph Lo 			__COPY_EMA(next, last, C1D0U1);
4349b9d8632SJoseph Lo 			__COPY_EMA(next, last, C0D1U0);
4359b9d8632SJoseph Lo 			__COPY_EMA(next, last, C0D1U1);
4369b9d8632SJoseph Lo 			__COPY_EMA(next, last, C1D1U0);
4379b9d8632SJoseph Lo 			__COPY_EMA(next, last, C1D1U1);
4389b9d8632SJoseph Lo 		} else {
4399b9d8632SJoseph Lo 			/* Reset the EMA.*/
4409b9d8632SJoseph Lo 			__MOVAVG(next, C0D0U0) = 0;
4419b9d8632SJoseph Lo 			__MOVAVG(next, C0D0U1) = 0;
4429b9d8632SJoseph Lo 			__MOVAVG(next, C1D0U0) = 0;
4439b9d8632SJoseph Lo 			__MOVAVG(next, C1D0U1) = 0;
4449b9d8632SJoseph Lo 			__MOVAVG(next, C0D1U0) = 0;
4459b9d8632SJoseph Lo 			__MOVAVG(next, C0D1U1) = 0;
4469b9d8632SJoseph Lo 			__MOVAVG(next, C1D1U0) = 0;
4479b9d8632SJoseph Lo 			__MOVAVG(next, C1D1U1) = 0;
4489b9d8632SJoseph Lo 
4499b9d8632SJoseph Lo 			for (i = 0; i < samples; i++) {
4509b9d8632SJoseph Lo 				tegra210_emc_start_periodic_compensation(emc);
4519b9d8632SJoseph Lo 				udelay(delay);
4529b9d8632SJoseph Lo 
4539b9d8632SJoseph Lo 				/*
4549b9d8632SJoseph Lo 				 * Generate next sample of data.
4559b9d8632SJoseph Lo 				 */
4569b9d8632SJoseph Lo 				adel = update_clock_tree_delay(emc, DVFS_PT1);
4579b9d8632SJoseph Lo 			}
4589b9d8632SJoseph Lo 		}
4599b9d8632SJoseph Lo 
4609b9d8632SJoseph Lo 		/*
4619b9d8632SJoseph Lo 		 * Seems like it should be part of the
4629b9d8632SJoseph Lo 		 * 'if (last_timing->periodic_training)' conditional
4639b9d8632SJoseph Lo 		 * since is already done for the else clause.
4649b9d8632SJoseph Lo 		 */
4659b9d8632SJoseph Lo 		adel = update_clock_tree_delay(emc, DVFS_UPDATE);
4669b9d8632SJoseph Lo 	}
4679b9d8632SJoseph Lo 
4689b9d8632SJoseph Lo 	if (type == PERIODIC_TRAINING_SEQUENCE) {
4699b9d8632SJoseph Lo 		tegra210_emc_start_periodic_compensation(emc);
4709b9d8632SJoseph Lo 		udelay(delay);
4719b9d8632SJoseph Lo 
4729b9d8632SJoseph Lo 		adel = update_clock_tree_delay(emc, PERIODIC_TRAINING_UPDATE);
4739b9d8632SJoseph Lo 	}
4749b9d8632SJoseph Lo 
4759b9d8632SJoseph Lo 	return adel;
4769b9d8632SJoseph Lo }
4779b9d8632SJoseph Lo 
tegra210_emc_r21021_periodic_compensation(struct tegra210_emc * emc)4789b9d8632SJoseph Lo static u32 tegra210_emc_r21021_periodic_compensation(struct tegra210_emc *emc)
4799b9d8632SJoseph Lo {
4809b9d8632SJoseph Lo 	u32 emc_cfg, emc_cfg_o, emc_cfg_update, del, value;
481d71b90e3SColin Ian King 	static const u32 list[] = {
4829b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_0,
4839b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_1,
4849b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_2,
4859b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_3,
4869b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_0,
4879b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_1,
4889b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_2,
4899b9d8632SJoseph Lo 		EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_3,
4909b9d8632SJoseph Lo 		EMC_DATA_BRLSHFT_0,
4919b9d8632SJoseph Lo 		EMC_DATA_BRLSHFT_1
4929b9d8632SJoseph Lo 	};
4939b9d8632SJoseph Lo 	struct tegra210_emc_timing *last = emc->last;
4949b9d8632SJoseph Lo 	unsigned int items = ARRAY_SIZE(list), i;
4959b9d8632SJoseph Lo 	unsigned long delay;
4969b9d8632SJoseph Lo 
4979b9d8632SJoseph Lo 	if (last->periodic_training) {
4989b9d8632SJoseph Lo 		emc_dbg(emc, PER_TRAIN, "Periodic training starting\n");
4999b9d8632SJoseph Lo 
5009b9d8632SJoseph Lo 		value = emc_readl(emc, EMC_DBG);
5019b9d8632SJoseph Lo 		emc_cfg_o = emc_readl(emc, EMC_CFG);
5029b9d8632SJoseph Lo 		emc_cfg = emc_cfg_o & ~(EMC_CFG_DYN_SELF_REF |
5039b9d8632SJoseph Lo 					EMC_CFG_DRAM_ACPD |
5049b9d8632SJoseph Lo 					EMC_CFG_DRAM_CLKSTOP_PD);
5059b9d8632SJoseph Lo 
5069b9d8632SJoseph Lo 
5079b9d8632SJoseph Lo 		/*
5089b9d8632SJoseph Lo 		 * 1. Power optimizations should be off.
5099b9d8632SJoseph Lo 		 */
5109b9d8632SJoseph Lo 		emc_writel(emc, emc_cfg, EMC_CFG);
5119b9d8632SJoseph Lo 
5129b9d8632SJoseph Lo 		/* Does emc_timing_update() for above changes. */
5139b9d8632SJoseph Lo 		tegra210_emc_dll_disable(emc);
5149b9d8632SJoseph Lo 
5159b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++)
5169b9d8632SJoseph Lo 			tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
5179b9d8632SJoseph Lo 						     EMC_EMC_STATUS_DRAM_IN_POWERDOWN_MASK,
5189b9d8632SJoseph Lo 						     0);
5199b9d8632SJoseph Lo 
5209b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++)
5219b9d8632SJoseph Lo 			tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
5229b9d8632SJoseph Lo 						     EMC_EMC_STATUS_DRAM_IN_SELF_REFRESH_MASK,
5239b9d8632SJoseph Lo 						     0);
5249b9d8632SJoseph Lo 
5259b9d8632SJoseph Lo 		emc_cfg_update = value = emc_readl(emc, EMC_CFG_UPDATE);
5269b9d8632SJoseph Lo 		value &= ~EMC_CFG_UPDATE_UPDATE_DLL_IN_UPDATE_MASK;
5279b9d8632SJoseph Lo 		value |= (2 << EMC_CFG_UPDATE_UPDATE_DLL_IN_UPDATE_SHIFT);
5289b9d8632SJoseph Lo 		emc_writel(emc, value, EMC_CFG_UPDATE);
5299b9d8632SJoseph Lo 
5309b9d8632SJoseph Lo 		/*
5319b9d8632SJoseph Lo 		 * 2. osc kick off - this assumes training and dvfs have set
5329b9d8632SJoseph Lo 		 *    correct MR23.
5339b9d8632SJoseph Lo 		 */
5349b9d8632SJoseph Lo 		tegra210_emc_start_periodic_compensation(emc);
5359b9d8632SJoseph Lo 
5369b9d8632SJoseph Lo 		/*
5379b9d8632SJoseph Lo 		 * 3. Let dram capture its clock tree delays.
5389b9d8632SJoseph Lo 		 */
5399b9d8632SJoseph Lo 		delay = tegra210_emc_actual_osc_clocks(last->run_clocks);
5409b9d8632SJoseph Lo 		delay *= 1000;
5419b9d8632SJoseph Lo 		delay /= last->rate + 1;
5429b9d8632SJoseph Lo 		udelay(delay);
5439b9d8632SJoseph Lo 
5449b9d8632SJoseph Lo 		/*
5459b9d8632SJoseph Lo 		 * 4. Check delta wrt previous values (save value if margin
5469b9d8632SJoseph Lo 		 *    exceeds what is set in table).
5479b9d8632SJoseph Lo 		 */
5489b9d8632SJoseph Lo 		del = periodic_compensation_handler(emc,
5499b9d8632SJoseph Lo 						    PERIODIC_TRAINING_SEQUENCE,
5509b9d8632SJoseph Lo 						    last, last);
5519b9d8632SJoseph Lo 
5529b9d8632SJoseph Lo 		/*
5539b9d8632SJoseph Lo 		 * 5. Apply compensation w.r.t. trained values (if clock tree
5549b9d8632SJoseph Lo 		 *    has drifted more than the set margin).
5559b9d8632SJoseph Lo 		 */
5569b9d8632SJoseph Lo 		if (last->tree_margin < ((del * 128 * (last->rate / 1000)) / 1000000)) {
5579b9d8632SJoseph Lo 			for (i = 0; i < items; i++) {
5589b9d8632SJoseph Lo 				value = tegra210_emc_compensate(last, list[i]);
5599b9d8632SJoseph Lo 				emc_dbg(emc, EMA_WRITES, "0x%08x <= 0x%08x\n",
5609b9d8632SJoseph Lo 					list[i], value);
5619b9d8632SJoseph Lo 				emc_writel(emc, value, list[i]);
5629b9d8632SJoseph Lo 			}
5639b9d8632SJoseph Lo 		}
5649b9d8632SJoseph Lo 
5659b9d8632SJoseph Lo 		emc_writel(emc, emc_cfg_o, EMC_CFG);
5669b9d8632SJoseph Lo 
5679b9d8632SJoseph Lo 		/*
5689b9d8632SJoseph Lo 		 * 6. Timing update actally applies the new trimmers.
5699b9d8632SJoseph Lo 		 */
5709b9d8632SJoseph Lo 		tegra210_emc_timing_update(emc);
5719b9d8632SJoseph Lo 
5729b9d8632SJoseph Lo 		/* 6.1. Restore the UPDATE_DLL_IN_UPDATE field. */
5739b9d8632SJoseph Lo 		emc_writel(emc, emc_cfg_update, EMC_CFG_UPDATE);
5749b9d8632SJoseph Lo 
5759b9d8632SJoseph Lo 		/* 6.2. Restore the DLL. */
5769b9d8632SJoseph Lo 		tegra210_emc_dll_enable(emc);
5779b9d8632SJoseph Lo 	}
5789b9d8632SJoseph Lo 
5799b9d8632SJoseph Lo 	return 0;
5809b9d8632SJoseph Lo }
5819b9d8632SJoseph Lo 
5829b9d8632SJoseph Lo /*
5839b9d8632SJoseph Lo  * Do the clock change sequence.
5849b9d8632SJoseph Lo  */
tegra210_emc_r21021_set_clock(struct tegra210_emc * emc,u32 clksrc)5859b9d8632SJoseph Lo static void tegra210_emc_r21021_set_clock(struct tegra210_emc *emc, u32 clksrc)
5869b9d8632SJoseph Lo {
5879b9d8632SJoseph Lo 	/* state variables */
5889b9d8632SJoseph Lo 	static bool fsp_for_next_freq;
5899b9d8632SJoseph Lo 	/* constant configuration parameters */
5909b9d8632SJoseph Lo 	const bool save_restore_clkstop_pd = true;
5919b9d8632SJoseph Lo 	const u32 zqcal_before_cc_cutoff = 2400;
5929b9d8632SJoseph Lo 	const bool cya_allow_ref_cc = false;
5939b9d8632SJoseph Lo 	const bool cya_issue_pc_ref = false;
5949b9d8632SJoseph Lo 	const bool opt_cc_short_zcal = true;
5959b9d8632SJoseph Lo 	const bool ref_b4_sref_en = false;
5969b9d8632SJoseph Lo 	const u32 tZQCAL_lpddr4 = 1000000;
5979b9d8632SJoseph Lo 	const bool opt_short_zcal = true;
5989b9d8632SJoseph Lo 	const bool opt_do_sw_qrst = true;
5999b9d8632SJoseph Lo 	const u32 opt_dvfs_mode = MAN_SR;
6009b9d8632SJoseph Lo 	/*
6019b9d8632SJoseph Lo 	 * This is the timing table for the source frequency. It does _not_
6029b9d8632SJoseph Lo 	 * necessarily correspond to the actual timing values in the EMC at the
6039b9d8632SJoseph Lo 	 * moment. If the boot BCT differs from the table then this can happen.
6049b9d8632SJoseph Lo 	 * However, we need it for accessing the dram_timings (which are not
6059b9d8632SJoseph Lo 	 * really registers) array for the current frequency.
6069b9d8632SJoseph Lo 	 */
6079b9d8632SJoseph Lo 	struct tegra210_emc_timing *fake, *last = emc->last, *next = emc->next;
6089b9d8632SJoseph Lo 	u32 tRTM, RP_war, R2P_war, TRPab_war, deltaTWATM, W2P_war, tRPST;
6099b9d8632SJoseph Lo 	u32 mr13_flip_fspwr, mr13_flip_fspop, ramp_up_wait, ramp_down_wait;
6109b9d8632SJoseph Lo 	u32 zq_wait_long, zq_latch_dvfs_wait_time, tZQCAL_lpddr4_fc_adj;
6119b9d8632SJoseph Lo 	u32 emc_auto_cal_config, auto_cal_en, emc_cfg, emc_sel_dpd_ctrl;
6129b9d8632SJoseph Lo 	u32 tFC_lpddr4 = 1000 * next->dram_timings[T_FC_LPDDR4];
6139b9d8632SJoseph Lo 	u32 bg_reg_mode_change, enable_bglp_reg, enable_bg_reg;
6149b9d8632SJoseph Lo 	bool opt_zcal_en_cc = false, is_lpddr3 = false;
6159b9d8632SJoseph Lo 	bool compensate_trimmer_applicable = false;
6169b9d8632SJoseph Lo 	u32 emc_dbg, emc_cfg_pipe_clk, emc_pin;
6179b9d8632SJoseph Lo 	u32 src_clk_period, dst_clk_period; /* in picoseconds */
6189b9d8632SJoseph Lo 	bool shared_zq_resistor = false;
6199b9d8632SJoseph Lo 	u32 value, dram_type;
6209b9d8632SJoseph Lo 	u32 opt_dll_mode = 0;
6219b9d8632SJoseph Lo 	unsigned long delay;
6229b9d8632SJoseph Lo 	unsigned int i;
6239b9d8632SJoseph Lo 
6249b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "Running clock change.\n");
6259b9d8632SJoseph Lo 
6269b9d8632SJoseph Lo 	/* XXX fake == last */
6279b9d8632SJoseph Lo 	fake = tegra210_emc_find_timing(emc, last->rate * 1000UL);
6289b9d8632SJoseph Lo 	fsp_for_next_freq = !fsp_for_next_freq;
6299b9d8632SJoseph Lo 
6309b9d8632SJoseph Lo 	value = emc_readl(emc, EMC_FBIO_CFG5) & EMC_FBIO_CFG5_DRAM_TYPE_MASK;
6319b9d8632SJoseph Lo 	dram_type = value >> EMC_FBIO_CFG5_DRAM_TYPE_SHIFT;
6329b9d8632SJoseph Lo 
6339b9d8632SJoseph Lo 	if (last->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX] & BIT(31))
6349b9d8632SJoseph Lo 		shared_zq_resistor = true;
6359b9d8632SJoseph Lo 
6369b9d8632SJoseph Lo 	if ((next->burst_regs[EMC_ZCAL_INTERVAL_INDEX] != 0 &&
6379b9d8632SJoseph Lo 	     last->burst_regs[EMC_ZCAL_INTERVAL_INDEX] == 0) ||
6389b9d8632SJoseph Lo 	    dram_type == DRAM_TYPE_LPDDR4)
6399b9d8632SJoseph Lo 		opt_zcal_en_cc = true;
6409b9d8632SJoseph Lo 
6419b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_DDR3)
6429b9d8632SJoseph Lo 		opt_dll_mode = tegra210_emc_get_dll_state(next);
6439b9d8632SJoseph Lo 
6449b9d8632SJoseph Lo 	if ((next->burst_regs[EMC_FBIO_CFG5_INDEX] & BIT(25)) &&
6459b9d8632SJoseph Lo 	    (dram_type == DRAM_TYPE_LPDDR2))
6469b9d8632SJoseph Lo 		is_lpddr3 = true;
6479b9d8632SJoseph Lo 
6489b9d8632SJoseph Lo 	emc_readl(emc, EMC_CFG);
6499b9d8632SJoseph Lo 	emc_readl(emc, EMC_AUTO_CAL_CONFIG);
6509b9d8632SJoseph Lo 
6519b9d8632SJoseph Lo 	src_clk_period = 1000000000 / last->rate;
6529b9d8632SJoseph Lo 	dst_clk_period = 1000000000 / next->rate;
6539b9d8632SJoseph Lo 
6549b9d8632SJoseph Lo 	if (dst_clk_period <= zqcal_before_cc_cutoff)
6559b9d8632SJoseph Lo 		tZQCAL_lpddr4_fc_adj = tZQCAL_lpddr4 - tFC_lpddr4;
6569b9d8632SJoseph Lo 	else
6579b9d8632SJoseph Lo 		tZQCAL_lpddr4_fc_adj = tZQCAL_lpddr4;
6589b9d8632SJoseph Lo 
6599b9d8632SJoseph Lo 	tZQCAL_lpddr4_fc_adj /= dst_clk_period;
6609b9d8632SJoseph Lo 
6619b9d8632SJoseph Lo 	emc_dbg = emc_readl(emc, EMC_DBG);
6629b9d8632SJoseph Lo 	emc_pin = emc_readl(emc, EMC_PIN);
6639b9d8632SJoseph Lo 	emc_cfg_pipe_clk = emc_readl(emc, EMC_CFG_PIPE_CLK);
6649b9d8632SJoseph Lo 
6659b9d8632SJoseph Lo 	emc_cfg = next->burst_regs[EMC_CFG_INDEX];
6669b9d8632SJoseph Lo 	emc_cfg &= ~(EMC_CFG_DYN_SELF_REF | EMC_CFG_DRAM_ACPD |
6679b9d8632SJoseph Lo 		     EMC_CFG_DRAM_CLKSTOP_SR | EMC_CFG_DRAM_CLKSTOP_PD);
6689b9d8632SJoseph Lo 	emc_sel_dpd_ctrl = next->emc_sel_dpd_ctrl;
6699b9d8632SJoseph Lo 	emc_sel_dpd_ctrl &= ~(EMC_SEL_DPD_CTRL_CLK_SEL_DPD_EN |
6709b9d8632SJoseph Lo 			      EMC_SEL_DPD_CTRL_CA_SEL_DPD_EN |
6719b9d8632SJoseph Lo 			      EMC_SEL_DPD_CTRL_RESET_SEL_DPD_EN |
6729b9d8632SJoseph Lo 			      EMC_SEL_DPD_CTRL_ODT_SEL_DPD_EN |
6739b9d8632SJoseph Lo 			      EMC_SEL_DPD_CTRL_DATA_SEL_DPD_EN);
6749b9d8632SJoseph Lo 
6759b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "Clock change version: %d\n",
6769b9d8632SJoseph Lo 		DVFS_CLOCK_CHANGE_VERSION);
6779b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "DRAM type = %d\n", dram_type);
6789b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "DRAM dev #: %u\n", emc->num_devices);
6799b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "Next EMC clksrc: 0x%08x\n", clksrc);
6809b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "DLL clksrc:      0x%08x\n", next->dll_clk_src);
6819b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "last rate: %u, next rate %u\n", last->rate,
6829b9d8632SJoseph Lo 		next->rate);
6839b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "last period: %u, next period: %u\n",
6849b9d8632SJoseph Lo 		src_clk_period, dst_clk_period);
6859b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "  shared_zq_resistor: %d\n", !!shared_zq_resistor);
6869b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "  num_channels: %u\n", emc->num_channels);
6879b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "  opt_dll_mode: %d\n", opt_dll_mode);
6889b9d8632SJoseph Lo 
6899b9d8632SJoseph Lo 	/*
6909b9d8632SJoseph Lo 	 * Step 1:
6919b9d8632SJoseph Lo 	 *   Pre DVFS SW sequence.
6929b9d8632SJoseph Lo 	 */
6939b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 1\n");
6949b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 1.1: Disable DLL temporarily.\n");
6959b9d8632SJoseph Lo 
6969b9d8632SJoseph Lo 	value = emc_readl(emc, EMC_CFG_DIG_DLL);
6979b9d8632SJoseph Lo 	value &= ~EMC_CFG_DIG_DLL_CFG_DLL_EN;
6989b9d8632SJoseph Lo 	emc_writel(emc, value, EMC_CFG_DIG_DLL);
6999b9d8632SJoseph Lo 
7009b9d8632SJoseph Lo 	tegra210_emc_timing_update(emc);
7019b9d8632SJoseph Lo 
7029b9d8632SJoseph Lo 	for (i = 0; i < emc->num_channels; i++)
7039b9d8632SJoseph Lo 		tegra210_emc_wait_for_update(emc, i, EMC_CFG_DIG_DLL,
7049b9d8632SJoseph Lo 					     EMC_CFG_DIG_DLL_CFG_DLL_EN, 0);
7059b9d8632SJoseph Lo 
7069b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 1.2: Disable AUTOCAL temporarily.\n");
7079b9d8632SJoseph Lo 
7089b9d8632SJoseph Lo 	emc_auto_cal_config = next->emc_auto_cal_config;
7099b9d8632SJoseph Lo 	auto_cal_en = emc_auto_cal_config & EMC_AUTO_CAL_CONFIG_AUTO_CAL_ENABLE;
7109b9d8632SJoseph Lo 	emc_auto_cal_config &= ~EMC_AUTO_CAL_CONFIG_AUTO_CAL_START;
7119b9d8632SJoseph Lo 	emc_auto_cal_config |= EMC_AUTO_CAL_CONFIG_AUTO_CAL_MEASURE_STALL;
7129b9d8632SJoseph Lo 	emc_auto_cal_config |= EMC_AUTO_CAL_CONFIG_AUTO_CAL_UPDATE_STALL;
7139b9d8632SJoseph Lo 	emc_auto_cal_config |= auto_cal_en;
7149b9d8632SJoseph Lo 	emc_writel(emc, emc_auto_cal_config, EMC_AUTO_CAL_CONFIG);
7159b9d8632SJoseph Lo 	emc_readl(emc, EMC_AUTO_CAL_CONFIG); /* Flush write. */
7169b9d8632SJoseph Lo 
7179b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 1.3: Disable other power features.\n");
7189b9d8632SJoseph Lo 
7199b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ACTIVE);
7209b9d8632SJoseph Lo 	emc_writel(emc, emc_cfg, EMC_CFG);
7219b9d8632SJoseph Lo 	emc_writel(emc, emc_sel_dpd_ctrl, EMC_SEL_DPD_CTRL);
7229b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
7239b9d8632SJoseph Lo 
7249b9d8632SJoseph Lo 	if (next->periodic_training) {
7259b9d8632SJoseph Lo 		tegra210_emc_reset_dram_clktree_values(next);
7269b9d8632SJoseph Lo 
7279b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++)
7289b9d8632SJoseph Lo 			tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
7299b9d8632SJoseph Lo 						     EMC_EMC_STATUS_DRAM_IN_POWERDOWN_MASK,
7309b9d8632SJoseph Lo 						     0);
7319b9d8632SJoseph Lo 
7329b9d8632SJoseph Lo 		for (i = 0; i < emc->num_channels; i++)
7339b9d8632SJoseph Lo 			tegra210_emc_wait_for_update(emc, i, EMC_EMC_STATUS,
7349b9d8632SJoseph Lo 						     EMC_EMC_STATUS_DRAM_IN_SELF_REFRESH_MASK,
7359b9d8632SJoseph Lo 						     0);
7369b9d8632SJoseph Lo 
7379b9d8632SJoseph Lo 		tegra210_emc_start_periodic_compensation(emc);
7389b9d8632SJoseph Lo 
7399b9d8632SJoseph Lo 		delay = 1000 * tegra210_emc_actual_osc_clocks(last->run_clocks);
7409b9d8632SJoseph Lo 		udelay((delay / last->rate) + 2);
7419b9d8632SJoseph Lo 
7429b9d8632SJoseph Lo 		value = periodic_compensation_handler(emc, DVFS_SEQUENCE, fake,
7439b9d8632SJoseph Lo 						      next);
7449b9d8632SJoseph Lo 		value = (value * 128 * next->rate / 1000) / 1000000;
7459b9d8632SJoseph Lo 
7469b9d8632SJoseph Lo 		if (next->periodic_training && value > next->tree_margin)
7479b9d8632SJoseph Lo 			compensate_trimmer_applicable = true;
7489b9d8632SJoseph Lo 	}
7499b9d8632SJoseph Lo 
7509b9d8632SJoseph Lo 	emc_writel(emc, EMC_INTSTATUS_CLKCHANGE_COMPLETE, EMC_INTSTATUS);
7519b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ACTIVE);
7529b9d8632SJoseph Lo 	emc_writel(emc, emc_cfg, EMC_CFG);
7539b9d8632SJoseph Lo 	emc_writel(emc, emc_sel_dpd_ctrl, EMC_SEL_DPD_CTRL);
7549b9d8632SJoseph Lo 	emc_writel(emc, emc_cfg_pipe_clk | EMC_CFG_PIPE_CLK_CLK_ALWAYS_ON,
7559b9d8632SJoseph Lo 		   EMC_CFG_PIPE_CLK);
7569b9d8632SJoseph Lo 	emc_writel(emc, next->emc_fdpd_ctrl_cmd_no_ramp &
7579b9d8632SJoseph Lo 			~EMC_FDPD_CTRL_CMD_NO_RAMP_CMD_DPD_NO_RAMP_ENABLE,
7589b9d8632SJoseph Lo 		   EMC_FDPD_CTRL_CMD_NO_RAMP);
7599b9d8632SJoseph Lo 
7609b9d8632SJoseph Lo 	bg_reg_mode_change =
7619b9d8632SJoseph Lo 		((next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7629b9d8632SJoseph Lo 		  EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD) ^
7639b9d8632SJoseph Lo 		 (last->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7649b9d8632SJoseph Lo 		  EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD)) ||
7659b9d8632SJoseph Lo 		((next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7669b9d8632SJoseph Lo 		  EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD) ^
7679b9d8632SJoseph Lo 		 (last->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7689b9d8632SJoseph Lo 		  EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD));
7699b9d8632SJoseph Lo 	enable_bglp_reg =
7709b9d8632SJoseph Lo 		(next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7719b9d8632SJoseph Lo 		 EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD) == 0;
7729b9d8632SJoseph Lo 	enable_bg_reg =
7739b9d8632SJoseph Lo 		(next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7749b9d8632SJoseph Lo 		 EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD) == 0;
7759b9d8632SJoseph Lo 
7769b9d8632SJoseph Lo 	if (bg_reg_mode_change) {
7779b9d8632SJoseph Lo 		if (enable_bg_reg)
7789b9d8632SJoseph Lo 			emc_writel(emc, last->burst_regs
7799b9d8632SJoseph Lo 				   [EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7809b9d8632SJoseph Lo 				   ~EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD,
7819b9d8632SJoseph Lo 				   EMC_PMACRO_BG_BIAS_CTRL_0);
7829b9d8632SJoseph Lo 
7839b9d8632SJoseph Lo 		if (enable_bglp_reg)
7849b9d8632SJoseph Lo 			emc_writel(emc, last->burst_regs
7859b9d8632SJoseph Lo 				   [EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
7869b9d8632SJoseph Lo 				   ~EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD,
7879b9d8632SJoseph Lo 				   EMC_PMACRO_BG_BIAS_CTRL_0);
7889b9d8632SJoseph Lo 	}
7899b9d8632SJoseph Lo 
7909b9d8632SJoseph Lo 	/* Check if we need to turn on VREF generator. */
7919b9d8632SJoseph Lo 	if ((((last->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
7929b9d8632SJoseph Lo 	       EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF) == 0) &&
7939b9d8632SJoseph Lo 	     ((next->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
7949b9d8632SJoseph Lo 	       EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF) == 1)) ||
7959b9d8632SJoseph Lo 	    (((last->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
7969b9d8632SJoseph Lo 	       EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF) == 0) &&
7979b9d8632SJoseph Lo 	     ((next->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX] &
7989b9d8632SJoseph Lo 	       EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF) != 0))) {
7999b9d8632SJoseph Lo 		u32 pad_tx_ctrl =
8009b9d8632SJoseph Lo 		    next->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX];
8019b9d8632SJoseph Lo 		u32 last_pad_tx_ctrl =
8029b9d8632SJoseph Lo 		    last->burst_regs[EMC_PMACRO_DATA_PAD_TX_CTRL_INDEX];
8039b9d8632SJoseph Lo 		u32 next_dq_e_ivref, next_dqs_e_ivref;
8049b9d8632SJoseph Lo 
8059b9d8632SJoseph Lo 		next_dqs_e_ivref = pad_tx_ctrl &
8069b9d8632SJoseph Lo 				   EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF;
8079b9d8632SJoseph Lo 		next_dq_e_ivref = pad_tx_ctrl &
8089b9d8632SJoseph Lo 				  EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF;
8099b9d8632SJoseph Lo 		value = (last_pad_tx_ctrl &
8109b9d8632SJoseph Lo 				~EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_E_IVREF &
8119b9d8632SJoseph Lo 				~EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQS_E_IVREF) |
8129b9d8632SJoseph Lo 			next_dq_e_ivref | next_dqs_e_ivref;
8139b9d8632SJoseph Lo 		emc_writel(emc, value, EMC_PMACRO_DATA_PAD_TX_CTRL);
8149b9d8632SJoseph Lo 		udelay(1);
8159b9d8632SJoseph Lo 	} else if (bg_reg_mode_change) {
8169b9d8632SJoseph Lo 		udelay(1);
8179b9d8632SJoseph Lo 	}
8189b9d8632SJoseph Lo 
8199b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
8209b9d8632SJoseph Lo 
8219b9d8632SJoseph Lo 	/*
8229b9d8632SJoseph Lo 	 * Step 2:
8239b9d8632SJoseph Lo 	 *   Prelock the DLL.
8249b9d8632SJoseph Lo 	 */
8259b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 2\n");
8269b9d8632SJoseph Lo 
8279b9d8632SJoseph Lo 	if (next->burst_regs[EMC_CFG_DIG_DLL_INDEX] &
8289b9d8632SJoseph Lo 	    EMC_CFG_DIG_DLL_CFG_DLL_EN) {
8299b9d8632SJoseph Lo 		emc_dbg(emc, INFO, "Prelock enabled for target frequency.\n");
8309b9d8632SJoseph Lo 		value = tegra210_emc_dll_prelock(emc, clksrc);
8319b9d8632SJoseph Lo 		emc_dbg(emc, INFO, "DLL out: 0x%03x\n", value);
8329b9d8632SJoseph Lo 	} else {
8339b9d8632SJoseph Lo 		emc_dbg(emc, INFO, "Disabling DLL for target frequency.\n");
8349b9d8632SJoseph Lo 		tegra210_emc_dll_disable(emc);
8359b9d8632SJoseph Lo 	}
8369b9d8632SJoseph Lo 
8379b9d8632SJoseph Lo 	/*
8389b9d8632SJoseph Lo 	 * Step 3:
8399b9d8632SJoseph Lo 	 *   Prepare autocal for the clock change.
8409b9d8632SJoseph Lo 	 */
8419b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 3\n");
8429b9d8632SJoseph Lo 
8439b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ACTIVE);
8449b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config2, EMC_AUTO_CAL_CONFIG2);
8459b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config3, EMC_AUTO_CAL_CONFIG3);
8469b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config4, EMC_AUTO_CAL_CONFIG4);
8479b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config5, EMC_AUTO_CAL_CONFIG5);
8489b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config6, EMC_AUTO_CAL_CONFIG6);
8499b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config7, EMC_AUTO_CAL_CONFIG7);
8509b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config8, EMC_AUTO_CAL_CONFIG8);
8519b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
8529b9d8632SJoseph Lo 
8539b9d8632SJoseph Lo 	emc_auto_cal_config |= (EMC_AUTO_CAL_CONFIG_AUTO_CAL_COMPUTE_START |
8549b9d8632SJoseph Lo 				auto_cal_en);
8559b9d8632SJoseph Lo 	emc_writel(emc, emc_auto_cal_config, EMC_AUTO_CAL_CONFIG);
8569b9d8632SJoseph Lo 
8579b9d8632SJoseph Lo 	/*
8589b9d8632SJoseph Lo 	 * Step 4:
8599b9d8632SJoseph Lo 	 *   Update EMC_CFG. (??)
8609b9d8632SJoseph Lo 	 */
8619b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 4\n");
8629b9d8632SJoseph Lo 
8639b9d8632SJoseph Lo 	if (src_clk_period > 50000 && dram_type == DRAM_TYPE_LPDDR4)
8649b9d8632SJoseph Lo 		ccfifo_writel(emc, 1, EMC_SELF_REF, 0);
8659b9d8632SJoseph Lo 	else
8669b9d8632SJoseph Lo 		emc_writel(emc, next->emc_cfg_2, EMC_CFG_2);
8679b9d8632SJoseph Lo 
8689b9d8632SJoseph Lo 	/*
8699b9d8632SJoseph Lo 	 * Step 5:
8709b9d8632SJoseph Lo 	 *   Prepare reference variables for ZQCAL regs.
8719b9d8632SJoseph Lo 	 */
8729b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 5\n");
8739b9d8632SJoseph Lo 
8749b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4)
8759b9d8632SJoseph Lo 		zq_wait_long = max((u32)1, div_o3(1000000, dst_clk_period));
8769b9d8632SJoseph Lo 	else if (dram_type == DRAM_TYPE_LPDDR2 || is_lpddr3)
8779b9d8632SJoseph Lo 		zq_wait_long = max(next->min_mrs_wait,
8789b9d8632SJoseph Lo 				   div_o3(360000, dst_clk_period)) + 4;
8799b9d8632SJoseph Lo 	else if (dram_type == DRAM_TYPE_DDR3)
8809b9d8632SJoseph Lo 		zq_wait_long = max((u32)256,
8819b9d8632SJoseph Lo 				   div_o3(320000, dst_clk_period) + 2);
8829b9d8632SJoseph Lo 	else
8839b9d8632SJoseph Lo 		zq_wait_long = 0;
8849b9d8632SJoseph Lo 
8859b9d8632SJoseph Lo 	/*
8869b9d8632SJoseph Lo 	 * Step 6:
8879b9d8632SJoseph Lo 	 *   Training code - removed.
8889b9d8632SJoseph Lo 	 */
8899b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 6\n");
8909b9d8632SJoseph Lo 
8919b9d8632SJoseph Lo 	/*
8929b9d8632SJoseph Lo 	 * Step 7:
8939b9d8632SJoseph Lo 	 *   Program FSP reference registers and send MRWs to new FSPWR.
8949b9d8632SJoseph Lo 	 */
8959b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 7\n");
8969b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Step 7.1: Bug 200024907 - Patch RP R2P");
8979b9d8632SJoseph Lo 
8989b9d8632SJoseph Lo 	/* WAR 200024907 */
8999b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
9009b9d8632SJoseph Lo 		u32 nRTP = 16;
9019b9d8632SJoseph Lo 
9029b9d8632SJoseph Lo 		if (src_clk_period >= 1000000 / 1866) /* 535.91 ps */
9039b9d8632SJoseph Lo 			nRTP = 14;
9049b9d8632SJoseph Lo 
9059b9d8632SJoseph Lo 		if (src_clk_period >= 1000000 / 1600) /* 625.00 ps */
9069b9d8632SJoseph Lo 			nRTP = 12;
9079b9d8632SJoseph Lo 
9089b9d8632SJoseph Lo 		if (src_clk_period >= 1000000 / 1333) /* 750.19 ps */
9099b9d8632SJoseph Lo 			nRTP = 10;
9109b9d8632SJoseph Lo 
9119b9d8632SJoseph Lo 		if (src_clk_period >= 1000000 / 1066) /* 938.09 ps */
9129b9d8632SJoseph Lo 			nRTP = 8;
9139b9d8632SJoseph Lo 
9149b9d8632SJoseph Lo 		deltaTWATM = max_t(u32, div_o3(7500, src_clk_period), 8);
9159b9d8632SJoseph Lo 
9169b9d8632SJoseph Lo 		/*
9179b9d8632SJoseph Lo 		 * Originally there was a + .5 in the tRPST calculation.
9189b9d8632SJoseph Lo 		 * However since we can't do FP in the kernel and the tRTM
9199b9d8632SJoseph Lo 		 * computation was in a floating point ceiling function, adding
9209b9d8632SJoseph Lo 		 * one to tRTP should be ok. There is no other source of non
9219b9d8632SJoseph Lo 		 * integer values, so the result was always going to be
9229b9d8632SJoseph Lo 		 * something for the form: f_ceil(N + .5) = N + 1;
9239b9d8632SJoseph Lo 		 */
9249b9d8632SJoseph Lo 		tRPST = (last->emc_mrw & 0x80) >> 7;
9259b9d8632SJoseph Lo 		tRTM = fake->dram_timings[RL] + div_o3(3600, src_clk_period) +
9269b9d8632SJoseph Lo 			max_t(u32, div_o3(7500, src_clk_period), 8) + tRPST +
9279b9d8632SJoseph Lo 			1 + nRTP;
9289b9d8632SJoseph Lo 
9299b9d8632SJoseph Lo 		emc_dbg(emc, INFO, "tRTM = %u, EMC_RP = %u\n", tRTM,
9309b9d8632SJoseph Lo 			next->burst_regs[EMC_RP_INDEX]);
9319b9d8632SJoseph Lo 
9329b9d8632SJoseph Lo 		if (last->burst_regs[EMC_RP_INDEX] < tRTM) {
9339b9d8632SJoseph Lo 			if (tRTM > (last->burst_regs[EMC_R2P_INDEX] +
9349b9d8632SJoseph Lo 				    last->burst_regs[EMC_RP_INDEX])) {
9359b9d8632SJoseph Lo 				R2P_war = tRTM - last->burst_regs[EMC_RP_INDEX];
9369b9d8632SJoseph Lo 				RP_war = last->burst_regs[EMC_RP_INDEX];
9379b9d8632SJoseph Lo 				TRPab_war = last->burst_regs[EMC_TRPAB_INDEX];
9389b9d8632SJoseph Lo 
9399b9d8632SJoseph Lo 				if (R2P_war > 63) {
9409b9d8632SJoseph Lo 					RP_war = R2P_war +
9419b9d8632SJoseph Lo 						 last->burst_regs[EMC_RP_INDEX] - 63;
9429b9d8632SJoseph Lo 
9439b9d8632SJoseph Lo 					if (TRPab_war < RP_war)
9449b9d8632SJoseph Lo 						TRPab_war = RP_war;
9459b9d8632SJoseph Lo 
9469b9d8632SJoseph Lo 					R2P_war = 63;
9479b9d8632SJoseph Lo 				}
9489b9d8632SJoseph Lo 			} else {
9499b9d8632SJoseph Lo 				R2P_war = last->burst_regs[EMC_R2P_INDEX];
9509b9d8632SJoseph Lo 				RP_war = last->burst_regs[EMC_RP_INDEX];
9519b9d8632SJoseph Lo 				TRPab_war = last->burst_regs[EMC_TRPAB_INDEX];
9529b9d8632SJoseph Lo 			}
9539b9d8632SJoseph Lo 
9549b9d8632SJoseph Lo 			if (RP_war < deltaTWATM) {
9559b9d8632SJoseph Lo 				W2P_war = last->burst_regs[EMC_W2P_INDEX]
9569b9d8632SJoseph Lo 					  + deltaTWATM - RP_war;
9579b9d8632SJoseph Lo 				if (W2P_war > 63) {
9589b9d8632SJoseph Lo 					RP_war = RP_war + W2P_war - 63;
9599b9d8632SJoseph Lo 					if (TRPab_war < RP_war)
9609b9d8632SJoseph Lo 						TRPab_war = RP_war;
9619b9d8632SJoseph Lo 					W2P_war = 63;
9629b9d8632SJoseph Lo 				}
9639b9d8632SJoseph Lo 			} else {
9649b9d8632SJoseph Lo 				W2P_war = last->burst_regs[
9659b9d8632SJoseph Lo 					  EMC_W2P_INDEX];
9669b9d8632SJoseph Lo 			}
9679b9d8632SJoseph Lo 
9689b9d8632SJoseph Lo 			if ((last->burst_regs[EMC_W2P_INDEX] ^ W2P_war) ||
9699b9d8632SJoseph Lo 			    (last->burst_regs[EMC_R2P_INDEX] ^ R2P_war) ||
9709b9d8632SJoseph Lo 			    (last->burst_regs[EMC_RP_INDEX] ^ RP_war) ||
9719b9d8632SJoseph Lo 			    (last->burst_regs[EMC_TRPAB_INDEX] ^ TRPab_war)) {
9729b9d8632SJoseph Lo 				emc_writel(emc, RP_war, EMC_RP);
9739b9d8632SJoseph Lo 				emc_writel(emc, R2P_war, EMC_R2P);
9749b9d8632SJoseph Lo 				emc_writel(emc, W2P_war, EMC_W2P);
9759b9d8632SJoseph Lo 				emc_writel(emc, TRPab_war, EMC_TRPAB);
9769b9d8632SJoseph Lo 			}
9779b9d8632SJoseph Lo 
9789b9d8632SJoseph Lo 			tegra210_emc_timing_update(emc);
9799b9d8632SJoseph Lo 		} else {
9809b9d8632SJoseph Lo 			emc_dbg(emc, INFO, "Skipped WAR\n");
9819b9d8632SJoseph Lo 		}
9829b9d8632SJoseph Lo 	}
9839b9d8632SJoseph Lo 
9849b9d8632SJoseph Lo 	if (!fsp_for_next_freq) {
9859b9d8632SJoseph Lo 		mr13_flip_fspwr = (next->emc_mrw3 & 0xffffff3f) | 0x80;
9869b9d8632SJoseph Lo 		mr13_flip_fspop = (next->emc_mrw3 & 0xffffff3f) | 0x00;
9879b9d8632SJoseph Lo 	} else {
9889b9d8632SJoseph Lo 		mr13_flip_fspwr = (next->emc_mrw3 & 0xffffff3f) | 0x40;
9899b9d8632SJoseph Lo 		mr13_flip_fspop = (next->emc_mrw3 & 0xffffff3f) | 0xc0;
9909b9d8632SJoseph Lo 	}
9919b9d8632SJoseph Lo 
9929b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
9939b9d8632SJoseph Lo 		emc_writel(emc, mr13_flip_fspwr, EMC_MRW3);
9949b9d8632SJoseph Lo 		emc_writel(emc, next->emc_mrw, EMC_MRW);
9959b9d8632SJoseph Lo 		emc_writel(emc, next->emc_mrw2, EMC_MRW2);
9969b9d8632SJoseph Lo 	}
9979b9d8632SJoseph Lo 
9989b9d8632SJoseph Lo 	/*
9999b9d8632SJoseph Lo 	 * Step 8:
10009b9d8632SJoseph Lo 	 *   Program the shadow registers.
10019b9d8632SJoseph Lo 	 */
10029b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 8\n");
10039b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Writing burst_regs\n");
10049b9d8632SJoseph Lo 
10059b9d8632SJoseph Lo 	for (i = 0; i < next->num_burst; i++) {
10069b9d8632SJoseph Lo 		const u16 *offsets = emc->offsets->burst;
10079b9d8632SJoseph Lo 		u16 offset;
10089b9d8632SJoseph Lo 
10099b9d8632SJoseph Lo 		if (!offsets[i])
10109b9d8632SJoseph Lo 			continue;
10119b9d8632SJoseph Lo 
10129b9d8632SJoseph Lo 		value = next->burst_regs[i];
10139b9d8632SJoseph Lo 		offset = offsets[i];
10149b9d8632SJoseph Lo 
10159b9d8632SJoseph Lo 		if (dram_type != DRAM_TYPE_LPDDR4 &&
10169b9d8632SJoseph Lo 		    (offset == EMC_MRW6 || offset == EMC_MRW7 ||
10179b9d8632SJoseph Lo 		     offset == EMC_MRW8 || offset == EMC_MRW9 ||
10189b9d8632SJoseph Lo 		     offset == EMC_MRW10 || offset == EMC_MRW11 ||
10199b9d8632SJoseph Lo 		     offset == EMC_MRW12 || offset == EMC_MRW13 ||
10209b9d8632SJoseph Lo 		     offset == EMC_MRW14 || offset == EMC_MRW15 ||
10219b9d8632SJoseph Lo 		     offset == EMC_TRAINING_CTRL))
10229b9d8632SJoseph Lo 			continue;
10239b9d8632SJoseph Lo 
10249b9d8632SJoseph Lo 		/* Pain... And suffering. */
10259b9d8632SJoseph Lo 		if (offset == EMC_CFG) {
10269b9d8632SJoseph Lo 			value &= ~EMC_CFG_DRAM_ACPD;
10279b9d8632SJoseph Lo 			value &= ~EMC_CFG_DYN_SELF_REF;
10289b9d8632SJoseph Lo 
10299b9d8632SJoseph Lo 			if (dram_type == DRAM_TYPE_LPDDR4) {
10309b9d8632SJoseph Lo 				value &= ~EMC_CFG_DRAM_CLKSTOP_SR;
10319b9d8632SJoseph Lo 				value &= ~EMC_CFG_DRAM_CLKSTOP_PD;
10329b9d8632SJoseph Lo 			}
10339b9d8632SJoseph Lo 		} else if (offset == EMC_MRS_WAIT_CNT &&
10349b9d8632SJoseph Lo 			   dram_type == DRAM_TYPE_LPDDR2 &&
10359b9d8632SJoseph Lo 			   opt_zcal_en_cc && !opt_cc_short_zcal &&
10369b9d8632SJoseph Lo 			   opt_short_zcal) {
10379b9d8632SJoseph Lo 			value = (value & ~(EMC_MRS_WAIT_CNT_SHORT_WAIT_MASK <<
10389b9d8632SJoseph Lo 					   EMC_MRS_WAIT_CNT_SHORT_WAIT_SHIFT)) |
10399b9d8632SJoseph Lo 				((zq_wait_long & EMC_MRS_WAIT_CNT_SHORT_WAIT_MASK) <<
10409b9d8632SJoseph Lo 						 EMC_MRS_WAIT_CNT_SHORT_WAIT_SHIFT);
10419b9d8632SJoseph Lo 		} else if (offset == EMC_ZCAL_WAIT_CNT &&
10429b9d8632SJoseph Lo 			   dram_type == DRAM_TYPE_DDR3 && opt_zcal_en_cc &&
10439b9d8632SJoseph Lo 			   !opt_cc_short_zcal && opt_short_zcal) {
10449b9d8632SJoseph Lo 			value = (value & ~(EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_MASK <<
10459b9d8632SJoseph Lo 					   EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_SHIFT)) |
10469b9d8632SJoseph Lo 				((zq_wait_long & EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_MASK) <<
10479b9d8632SJoseph Lo 						 EMC_MRS_WAIT_CNT_SHORT_WAIT_SHIFT);
10489b9d8632SJoseph Lo 		} else if (offset == EMC_ZCAL_INTERVAL && opt_zcal_en_cc) {
10499b9d8632SJoseph Lo 			value = 0; /* EMC_ZCAL_INTERVAL reset value. */
10509b9d8632SJoseph Lo 		} else if (offset == EMC_PMACRO_AUTOCAL_CFG_COMMON) {
10519b9d8632SJoseph Lo 			value |= EMC_PMACRO_AUTOCAL_CFG_COMMON_E_CAL_BYPASS_DVFS;
10529b9d8632SJoseph Lo 		} else if (offset == EMC_PMACRO_DATA_PAD_TX_CTRL) {
10539b9d8632SJoseph Lo 			value &= ~(EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQSP_TX_E_DCC |
10549b9d8632SJoseph Lo 				   EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQSN_TX_E_DCC |
10559b9d8632SJoseph Lo 				   EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_DQ_TX_E_DCC |
10569b9d8632SJoseph Lo 				   EMC_PMACRO_DATA_PAD_TX_CTRL_DATA_CMD_TX_E_DCC);
10579b9d8632SJoseph Lo 		} else if (offset == EMC_PMACRO_CMD_PAD_TX_CTRL) {
10589b9d8632SJoseph Lo 			value |= EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQ_TX_DRVFORCEON;
10599b9d8632SJoseph Lo 			value &= ~(EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQSP_TX_E_DCC |
10609b9d8632SJoseph Lo 				   EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQSN_TX_E_DCC |
10619b9d8632SJoseph Lo 				   EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_DQ_TX_E_DCC |
10629b9d8632SJoseph Lo 				   EMC_PMACRO_CMD_PAD_TX_CTRL_CMD_CMD_TX_E_DCC);
10639b9d8632SJoseph Lo 		} else if (offset == EMC_PMACRO_BRICK_CTRL_RFU1) {
10649b9d8632SJoseph Lo 			value &= 0xf800f800;
10659b9d8632SJoseph Lo 		} else if (offset == EMC_PMACRO_COMMON_PAD_TX_CTRL) {
10669b9d8632SJoseph Lo 			value &= 0xfffffff0;
10679b9d8632SJoseph Lo 		}
10689b9d8632SJoseph Lo 
10699b9d8632SJoseph Lo 		emc_writel(emc, value, offset);
10709b9d8632SJoseph Lo 	}
10719b9d8632SJoseph Lo 
10720553d7b2SThierry Reding 	/* SW addition: do EMC refresh adjustment here. */
10730553d7b2SThierry Reding 	tegra210_emc_adjust_timing(emc, next);
10740553d7b2SThierry Reding 
10759b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
10769b9d8632SJoseph Lo 		value = (23 << EMC_MRW_MRW_MA_SHIFT) |
10779b9d8632SJoseph Lo 			(next->run_clocks & EMC_MRW_MRW_OP_MASK);
10789b9d8632SJoseph Lo 		emc_writel(emc, value, EMC_MRW);
10799b9d8632SJoseph Lo 	}
10809b9d8632SJoseph Lo 
10819b9d8632SJoseph Lo 	/* Per channel burst registers. */
10829b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Writing burst_regs_per_ch\n");
10839b9d8632SJoseph Lo 
10849b9d8632SJoseph Lo 	for (i = 0; i < next->num_burst_per_ch; i++) {
10859b9d8632SJoseph Lo 		const struct tegra210_emc_per_channel_regs *burst =
10869b9d8632SJoseph Lo 				emc->offsets->burst_per_channel;
10879b9d8632SJoseph Lo 
10889b9d8632SJoseph Lo 		if (!burst[i].offset)
10899b9d8632SJoseph Lo 			continue;
10909b9d8632SJoseph Lo 
10919b9d8632SJoseph Lo 		if (dram_type != DRAM_TYPE_LPDDR4 &&
10929b9d8632SJoseph Lo 		    (burst[i].offset == EMC_MRW6 ||
10939b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW7 ||
10949b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW8 ||
10959b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW9 ||
10969b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW10 ||
10979b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW11 ||
10989b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW12 ||
10999b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW13 ||
11009b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW14 ||
11019b9d8632SJoseph Lo 		     burst[i].offset == EMC_MRW15))
11029b9d8632SJoseph Lo 			continue;
11039b9d8632SJoseph Lo 
11049b9d8632SJoseph Lo 		/* Filter out second channel if not in DUAL_CHANNEL mode. */
11059b9d8632SJoseph Lo 		if (emc->num_channels < 2 && burst[i].bank >= 1)
11069b9d8632SJoseph Lo 			continue;
11079b9d8632SJoseph Lo 
11089b9d8632SJoseph Lo 		emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
11099b9d8632SJoseph Lo 			next->burst_reg_per_ch[i], burst[i].offset);
11109b9d8632SJoseph Lo 		emc_channel_writel(emc, burst[i].bank,
11119b9d8632SJoseph Lo 				   next->burst_reg_per_ch[i],
11129b9d8632SJoseph Lo 				   burst[i].offset);
11139b9d8632SJoseph Lo 	}
11149b9d8632SJoseph Lo 
11159b9d8632SJoseph Lo 	/* Vref regs. */
11169b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Writing vref_regs\n");
11179b9d8632SJoseph Lo 
11189b9d8632SJoseph Lo 	for (i = 0; i < next->vref_num; i++) {
11199b9d8632SJoseph Lo 		const struct tegra210_emc_per_channel_regs *vref =
11209b9d8632SJoseph Lo 					emc->offsets->vref_per_channel;
11219b9d8632SJoseph Lo 
11229b9d8632SJoseph Lo 		if (!vref[i].offset)
11239b9d8632SJoseph Lo 			continue;
11249b9d8632SJoseph Lo 
11259b9d8632SJoseph Lo 		if (emc->num_channels < 2 && vref[i].bank >= 1)
11269b9d8632SJoseph Lo 			continue;
11279b9d8632SJoseph Lo 
11289b9d8632SJoseph Lo 		emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
11299b9d8632SJoseph Lo 			next->vref_perch_regs[i], vref[i].offset);
11309b9d8632SJoseph Lo 		emc_channel_writel(emc, vref[i].bank, next->vref_perch_regs[i],
11319b9d8632SJoseph Lo 				   vref[i].offset);
11329b9d8632SJoseph Lo 	}
11339b9d8632SJoseph Lo 
11349b9d8632SJoseph Lo 	/* Trimmers. */
11359b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Writing trim_regs\n");
11369b9d8632SJoseph Lo 
11379b9d8632SJoseph Lo 	for (i = 0; i < next->num_trim; i++) {
11389b9d8632SJoseph Lo 		const u16 *offsets = emc->offsets->trim;
11399b9d8632SJoseph Lo 
11409b9d8632SJoseph Lo 		if (!offsets[i])
11419b9d8632SJoseph Lo 			continue;
11429b9d8632SJoseph Lo 
11439b9d8632SJoseph Lo 		if (compensate_trimmer_applicable &&
11449b9d8632SJoseph Lo 		    (offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_0 ||
11459b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_1 ||
11469b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_2 ||
11479b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_3 ||
11489b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_0 ||
11499b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_1 ||
11509b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_2 ||
11519b9d8632SJoseph Lo 		     offsets[i] == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_3 ||
11529b9d8632SJoseph Lo 		     offsets[i] == EMC_DATA_BRLSHFT_0 ||
11539b9d8632SJoseph Lo 		     offsets[i] == EMC_DATA_BRLSHFT_1)) {
11549b9d8632SJoseph Lo 			value = tegra210_emc_compensate(next, offsets[i]);
11559b9d8632SJoseph Lo 			emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
11569b9d8632SJoseph Lo 				value, offsets[i]);
11579b9d8632SJoseph Lo 			emc_dbg(emc, EMA_WRITES, "0x%08x <= 0x%08x\n",
11589b9d8632SJoseph Lo 				(u32)(u64)offsets[i], value);
11599b9d8632SJoseph Lo 			emc_writel(emc, value, offsets[i]);
11609b9d8632SJoseph Lo 		} else {
11619b9d8632SJoseph Lo 			emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
11629b9d8632SJoseph Lo 				next->trim_regs[i], offsets[i]);
11639b9d8632SJoseph Lo 			emc_writel(emc, next->trim_regs[i], offsets[i]);
11649b9d8632SJoseph Lo 		}
11659b9d8632SJoseph Lo 	}
11669b9d8632SJoseph Lo 
11679b9d8632SJoseph Lo 	/* Per channel trimmers. */
11689b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Writing trim_regs_per_ch\n");
11699b9d8632SJoseph Lo 
11709b9d8632SJoseph Lo 	for (i = 0; i < next->num_trim_per_ch; i++) {
11719b9d8632SJoseph Lo 		const struct tegra210_emc_per_channel_regs *trim =
11729b9d8632SJoseph Lo 				&emc->offsets->trim_per_channel[0];
11739b9d8632SJoseph Lo 		unsigned int offset;
11749b9d8632SJoseph Lo 
11759b9d8632SJoseph Lo 		if (!trim[i].offset)
11769b9d8632SJoseph Lo 			continue;
11779b9d8632SJoseph Lo 
11789b9d8632SJoseph Lo 		if (emc->num_channels < 2 && trim[i].bank >= 1)
11799b9d8632SJoseph Lo 			continue;
11809b9d8632SJoseph Lo 
11819b9d8632SJoseph Lo 		offset = trim[i].offset;
11829b9d8632SJoseph Lo 
11839b9d8632SJoseph Lo 		if (compensate_trimmer_applicable &&
11849b9d8632SJoseph Lo 		    (offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_0 ||
11859b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_1 ||
11869b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_2 ||
11879b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK0_3 ||
11889b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_0 ||
11899b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_1 ||
11909b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_2 ||
11919b9d8632SJoseph Lo 		     offset == EMC_PMACRO_OB_DDLL_LONG_DQ_RANK1_3 ||
11929b9d8632SJoseph Lo 		     offset == EMC_DATA_BRLSHFT_0 ||
11939b9d8632SJoseph Lo 		     offset == EMC_DATA_BRLSHFT_1)) {
11949b9d8632SJoseph Lo 			value = tegra210_emc_compensate(next, offset);
11959b9d8632SJoseph Lo 			emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
11969b9d8632SJoseph Lo 				value, offset);
11979b9d8632SJoseph Lo 			emc_dbg(emc, EMA_WRITES, "0x%08x <= 0x%08x\n", offset,
11989b9d8632SJoseph Lo 				value);
11999b9d8632SJoseph Lo 			emc_channel_writel(emc, trim[i].bank, value, offset);
12009b9d8632SJoseph Lo 		} else {
12019b9d8632SJoseph Lo 			emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
12029b9d8632SJoseph Lo 				next->trim_perch_regs[i], offset);
12039b9d8632SJoseph Lo 			emc_channel_writel(emc, trim[i].bank,
12049b9d8632SJoseph Lo 					   next->trim_perch_regs[i], offset);
12059b9d8632SJoseph Lo 		}
12069b9d8632SJoseph Lo 	}
12079b9d8632SJoseph Lo 
12089b9d8632SJoseph Lo 	emc_dbg(emc, SUB_STEPS, "Writing burst_mc_regs\n");
12099b9d8632SJoseph Lo 
12109b9d8632SJoseph Lo 	for (i = 0; i < next->num_mc_regs; i++) {
12119b9d8632SJoseph Lo 		const u16 *offsets = emc->offsets->burst_mc;
12129b9d8632SJoseph Lo 		u32 *values = next->burst_mc_regs;
12139b9d8632SJoseph Lo 
12149b9d8632SJoseph Lo 		emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
12159b9d8632SJoseph Lo 			values[i], offsets[i]);
12169b9d8632SJoseph Lo 		mc_writel(emc->mc, values[i], offsets[i]);
12179b9d8632SJoseph Lo 	}
12189b9d8632SJoseph Lo 
12199b9d8632SJoseph Lo 	/* Registers to be programmed on the faster clock. */
12209b9d8632SJoseph Lo 	if (next->rate < last->rate) {
12219b9d8632SJoseph Lo 		const u16 *la = emc->offsets->la_scale;
12229b9d8632SJoseph Lo 
12239b9d8632SJoseph Lo 		emc_dbg(emc, SUB_STEPS, "Writing la_scale_regs\n");
12249b9d8632SJoseph Lo 
12259b9d8632SJoseph Lo 		for (i = 0; i < next->num_up_down; i++) {
12269b9d8632SJoseph Lo 			emc_dbg(emc, REG_LISTS, "(%u) 0x%08x => 0x%08x\n", i,
12279b9d8632SJoseph Lo 				next->la_scale_regs[i], la[i]);
12289b9d8632SJoseph Lo 			mc_writel(emc->mc, next->la_scale_regs[i], la[i]);
12299b9d8632SJoseph Lo 		}
12309b9d8632SJoseph Lo 	}
12319b9d8632SJoseph Lo 
12329b9d8632SJoseph Lo 	/* Flush all the burst register writes. */
12339b9d8632SJoseph Lo 	mc_readl(emc->mc, MC_EMEM_ADR_CFG);
12349b9d8632SJoseph Lo 
12359b9d8632SJoseph Lo 	/*
12369b9d8632SJoseph Lo 	 * Step 9:
12379b9d8632SJoseph Lo 	 *   LPDDR4 section A.
12389b9d8632SJoseph Lo 	 */
12399b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 9\n");
12409b9d8632SJoseph Lo 
12419b9d8632SJoseph Lo 	value = next->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX];
12429b9d8632SJoseph Lo 	value &= ~EMC_ZCAL_WAIT_CNT_ZCAL_WAIT_CNT_MASK;
12439b9d8632SJoseph Lo 
12449b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
12459b9d8632SJoseph Lo 		emc_writel(emc, 0, EMC_ZCAL_INTERVAL);
12469b9d8632SJoseph Lo 		emc_writel(emc, value, EMC_ZCAL_WAIT_CNT);
12479b9d8632SJoseph Lo 
12489b9d8632SJoseph Lo 		value = emc_dbg | (EMC_DBG_WRITE_MUX_ACTIVE |
12499b9d8632SJoseph Lo 				   EMC_DBG_WRITE_ACTIVE_ONLY);
12509b9d8632SJoseph Lo 
12519b9d8632SJoseph Lo 		emc_writel(emc, value, EMC_DBG);
12529b9d8632SJoseph Lo 		emc_writel(emc, 0, EMC_ZCAL_INTERVAL);
12539b9d8632SJoseph Lo 		emc_writel(emc, emc_dbg, EMC_DBG);
12549b9d8632SJoseph Lo 	}
12559b9d8632SJoseph Lo 
12569b9d8632SJoseph Lo 	/*
12579b9d8632SJoseph Lo 	 * Step 10:
12589b9d8632SJoseph Lo 	 *   LPDDR4 and DDR3 common section.
12599b9d8632SJoseph Lo 	 */
12609b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 10\n");
12619b9d8632SJoseph Lo 
12629b9d8632SJoseph Lo 	if (opt_dvfs_mode == MAN_SR || dram_type == DRAM_TYPE_LPDDR4) {
12639b9d8632SJoseph Lo 		if (dram_type == DRAM_TYPE_LPDDR4)
12649b9d8632SJoseph Lo 			ccfifo_writel(emc, 0x101, EMC_SELF_REF, 0);
12659b9d8632SJoseph Lo 		else
12669b9d8632SJoseph Lo 			ccfifo_writel(emc, 0x1, EMC_SELF_REF, 0);
12679b9d8632SJoseph Lo 
12689b9d8632SJoseph Lo 		if (dram_type == DRAM_TYPE_LPDDR4 &&
12699b9d8632SJoseph Lo 		    dst_clk_period <= zqcal_before_cc_cutoff) {
12709b9d8632SJoseph Lo 			ccfifo_writel(emc, mr13_flip_fspwr ^ 0x40, EMC_MRW3, 0);
12719b9d8632SJoseph Lo 			ccfifo_writel(emc, (next->burst_regs[EMC_MRW6_INDEX] &
12729b9d8632SJoseph Lo 						0xFFFF3F3F) |
12739b9d8632SJoseph Lo 					   (last->burst_regs[EMC_MRW6_INDEX] &
12749b9d8632SJoseph Lo 						0x0000C0C0), EMC_MRW6, 0);
12759b9d8632SJoseph Lo 			ccfifo_writel(emc, (next->burst_regs[EMC_MRW14_INDEX] &
12769b9d8632SJoseph Lo 						0xFFFF0707) |
12779b9d8632SJoseph Lo 					   (last->burst_regs[EMC_MRW14_INDEX] &
12789b9d8632SJoseph Lo 						0x00003838), EMC_MRW14, 0);
12799b9d8632SJoseph Lo 
12809b9d8632SJoseph Lo 			if (emc->num_devices > 1) {
12819b9d8632SJoseph Lo 				ccfifo_writel(emc,
12829b9d8632SJoseph Lo 				      (next->burst_regs[EMC_MRW7_INDEX] &
12839b9d8632SJoseph Lo 				       0xFFFF3F3F) |
12849b9d8632SJoseph Lo 				      (last->burst_regs[EMC_MRW7_INDEX] &
12859b9d8632SJoseph Lo 				       0x0000C0C0), EMC_MRW7, 0);
12869b9d8632SJoseph Lo 				ccfifo_writel(emc,
12879b9d8632SJoseph Lo 				     (next->burst_regs[EMC_MRW15_INDEX] &
12889b9d8632SJoseph Lo 				      0xFFFF0707) |
12899b9d8632SJoseph Lo 				     (last->burst_regs[EMC_MRW15_INDEX] &
12909b9d8632SJoseph Lo 				      0x00003838), EMC_MRW15, 0);
12919b9d8632SJoseph Lo 			}
12929b9d8632SJoseph Lo 
12939b9d8632SJoseph Lo 			if (opt_zcal_en_cc) {
12949b9d8632SJoseph Lo 				if (emc->num_devices < 2)
12959b9d8632SJoseph Lo 					ccfifo_writel(emc,
12969b9d8632SJoseph Lo 						2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT
12979b9d8632SJoseph Lo 						| EMC_ZQ_CAL_ZQ_CAL_CMD,
12989b9d8632SJoseph Lo 						EMC_ZQ_CAL, 0);
12999b9d8632SJoseph Lo 				else if (shared_zq_resistor)
13009b9d8632SJoseph Lo 					ccfifo_writel(emc,
13019b9d8632SJoseph Lo 						2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT
13029b9d8632SJoseph Lo 						| EMC_ZQ_CAL_ZQ_CAL_CMD,
13039b9d8632SJoseph Lo 						EMC_ZQ_CAL, 0);
13049b9d8632SJoseph Lo 				else
13059b9d8632SJoseph Lo 					ccfifo_writel(emc,
13069b9d8632SJoseph Lo 						      EMC_ZQ_CAL_ZQ_CAL_CMD,
13079b9d8632SJoseph Lo 						      EMC_ZQ_CAL, 0);
13089b9d8632SJoseph Lo 			}
13099b9d8632SJoseph Lo 		}
13109b9d8632SJoseph Lo 	}
13119b9d8632SJoseph Lo 
13129b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
13139b9d8632SJoseph Lo 		value = (1000 * fake->dram_timings[T_RP]) / src_clk_period;
13149b9d8632SJoseph Lo 		ccfifo_writel(emc, mr13_flip_fspop | 0x8, EMC_MRW3, value);
13159b9d8632SJoseph Lo 		ccfifo_writel(emc, 0, 0, tFC_lpddr4 / src_clk_period);
13169b9d8632SJoseph Lo 	}
13179b9d8632SJoseph Lo 
13189b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4 || opt_dvfs_mode != MAN_SR) {
13199b9d8632SJoseph Lo 		delay = 30;
13209b9d8632SJoseph Lo 
13219b9d8632SJoseph Lo 		if (cya_allow_ref_cc) {
13229b9d8632SJoseph Lo 			delay += (1000 * fake->dram_timings[T_RP]) /
13239b9d8632SJoseph Lo 					src_clk_period;
13249b9d8632SJoseph Lo 			delay += 4000 * fake->dram_timings[T_RFC];
13259b9d8632SJoseph Lo 		}
13269b9d8632SJoseph Lo 
13279b9d8632SJoseph Lo 		ccfifo_writel(emc, emc_pin & ~(EMC_PIN_PIN_CKE_PER_DEV |
13289b9d8632SJoseph Lo 					       EMC_PIN_PIN_CKEB |
13299b9d8632SJoseph Lo 					       EMC_PIN_PIN_CKE),
13309b9d8632SJoseph Lo 			      EMC_PIN, delay);
13319b9d8632SJoseph Lo 	}
13329b9d8632SJoseph Lo 
13339b9d8632SJoseph Lo 	/* calculate reference delay multiplier */
13349b9d8632SJoseph Lo 	value = 1;
13359b9d8632SJoseph Lo 
13369b9d8632SJoseph Lo 	if (ref_b4_sref_en)
13379b9d8632SJoseph Lo 		value++;
13389b9d8632SJoseph Lo 
13399b9d8632SJoseph Lo 	if (cya_allow_ref_cc)
13409b9d8632SJoseph Lo 		value++;
13419b9d8632SJoseph Lo 
13429b9d8632SJoseph Lo 	if (cya_issue_pc_ref)
13439b9d8632SJoseph Lo 		value++;
13449b9d8632SJoseph Lo 
13459b9d8632SJoseph Lo 	if (dram_type != DRAM_TYPE_LPDDR4) {
13469b9d8632SJoseph Lo 		delay = ((1000 * fake->dram_timings[T_RP] / src_clk_period) +
13479b9d8632SJoseph Lo 			 (1000 * fake->dram_timings[T_RFC] / src_clk_period));
13489b9d8632SJoseph Lo 		delay = value * delay + 20;
13499b9d8632SJoseph Lo 	} else {
13509b9d8632SJoseph Lo 		delay = 0;
13519b9d8632SJoseph Lo 	}
13529b9d8632SJoseph Lo 
13539b9d8632SJoseph Lo 	/*
13549b9d8632SJoseph Lo 	 * Step 11:
13559b9d8632SJoseph Lo 	 *   Ramp down.
13569b9d8632SJoseph Lo 	 */
13579b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 11\n");
13589b9d8632SJoseph Lo 
13599b9d8632SJoseph Lo 	ccfifo_writel(emc, 0x0, EMC_CFG_SYNC, delay);
13609b9d8632SJoseph Lo 
13619b9d8632SJoseph Lo 	value = emc_dbg | EMC_DBG_WRITE_MUX_ACTIVE | EMC_DBG_WRITE_ACTIVE_ONLY;
13629b9d8632SJoseph Lo 	ccfifo_writel(emc, value, EMC_DBG, 0);
13639b9d8632SJoseph Lo 
13649b9d8632SJoseph Lo 	ramp_down_wait = tegra210_emc_dvfs_power_ramp_down(emc, src_clk_period,
13659b9d8632SJoseph Lo 							   0);
13669b9d8632SJoseph Lo 
13679b9d8632SJoseph Lo 	/*
13689b9d8632SJoseph Lo 	 * Step 12:
13699b9d8632SJoseph Lo 	 *   And finally - trigger the clock change.
13709b9d8632SJoseph Lo 	 */
13719b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 12\n");
13729b9d8632SJoseph Lo 
13739b9d8632SJoseph Lo 	ccfifo_writel(emc, 1, EMC_STALL_THEN_EXE_AFTER_CLKCHANGE, 0);
13749b9d8632SJoseph Lo 	value &= ~EMC_DBG_WRITE_ACTIVE_ONLY;
13759b9d8632SJoseph Lo 	ccfifo_writel(emc, value, EMC_DBG, 0);
13769b9d8632SJoseph Lo 
13779b9d8632SJoseph Lo 	/*
13789b9d8632SJoseph Lo 	 * Step 13:
13799b9d8632SJoseph Lo 	 *   Ramp up.
13809b9d8632SJoseph Lo 	 */
13819b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 13\n");
13829b9d8632SJoseph Lo 
13839b9d8632SJoseph Lo 	ramp_up_wait = tegra210_emc_dvfs_power_ramp_up(emc, dst_clk_period, 0);
13849b9d8632SJoseph Lo 	ccfifo_writel(emc, emc_dbg, EMC_DBG, 0);
13859b9d8632SJoseph Lo 
13869b9d8632SJoseph Lo 	/*
13879b9d8632SJoseph Lo 	 * Step 14:
13889b9d8632SJoseph Lo 	 *   Bringup CKE pins.
13899b9d8632SJoseph Lo 	 */
13909b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 14\n");
13919b9d8632SJoseph Lo 
13929b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
13939b9d8632SJoseph Lo 		value = emc_pin | EMC_PIN_PIN_CKE;
13949b9d8632SJoseph Lo 
13959b9d8632SJoseph Lo 		if (emc->num_devices <= 1)
13969b9d8632SJoseph Lo 			value &= ~(EMC_PIN_PIN_CKEB | EMC_PIN_PIN_CKE_PER_DEV);
13979b9d8632SJoseph Lo 		else
13989b9d8632SJoseph Lo 			value |= EMC_PIN_PIN_CKEB | EMC_PIN_PIN_CKE_PER_DEV;
13999b9d8632SJoseph Lo 
14009b9d8632SJoseph Lo 		ccfifo_writel(emc, value, EMC_PIN, 0);
14019b9d8632SJoseph Lo 	}
14029b9d8632SJoseph Lo 
14039b9d8632SJoseph Lo 	/*
14049b9d8632SJoseph Lo 	 * Step 15: (two step 15s ??)
14059b9d8632SJoseph Lo 	 *   Calculate zqlatch wait time; has dependency on ramping times.
14069b9d8632SJoseph Lo 	 */
14079b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 15\n");
14089b9d8632SJoseph Lo 
14099b9d8632SJoseph Lo 	if (dst_clk_period <= zqcal_before_cc_cutoff) {
14109b9d8632SJoseph Lo 		s32 t = (s32)(ramp_up_wait + ramp_down_wait) /
14119b9d8632SJoseph Lo 			(s32)dst_clk_period;
14129b9d8632SJoseph Lo 		zq_latch_dvfs_wait_time = (s32)tZQCAL_lpddr4_fc_adj - t;
14139b9d8632SJoseph Lo 	} else {
14149b9d8632SJoseph Lo 		zq_latch_dvfs_wait_time = tZQCAL_lpddr4_fc_adj -
14159b9d8632SJoseph Lo 			div_o3(1000 * next->dram_timings[T_PDEX],
14169b9d8632SJoseph Lo 			       dst_clk_period);
14179b9d8632SJoseph Lo 	}
14189b9d8632SJoseph Lo 
14199b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "tZQCAL_lpddr4_fc_adj = %u\n", tZQCAL_lpddr4_fc_adj);
14209b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "dst_clk_period = %u\n",
14219b9d8632SJoseph Lo 		dst_clk_period);
14229b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "next->dram_timings[T_PDEX] = %u\n",
14239b9d8632SJoseph Lo 		next->dram_timings[T_PDEX]);
14249b9d8632SJoseph Lo 	emc_dbg(emc, INFO, "zq_latch_dvfs_wait_time = %d\n",
14259b9d8632SJoseph Lo 		max_t(s32, 0, zq_latch_dvfs_wait_time));
14269b9d8632SJoseph Lo 
14279b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4 && opt_zcal_en_cc) {
14289b9d8632SJoseph Lo 		delay = div_o3(1000 * next->dram_timings[T_PDEX],
14299b9d8632SJoseph Lo 			       dst_clk_period);
14309b9d8632SJoseph Lo 
14319b9d8632SJoseph Lo 		if (emc->num_devices < 2) {
14329b9d8632SJoseph Lo 			if (dst_clk_period > zqcal_before_cc_cutoff)
14339b9d8632SJoseph Lo 				ccfifo_writel(emc,
14349b9d8632SJoseph Lo 					      2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
14359b9d8632SJoseph Lo 					      EMC_ZQ_CAL_ZQ_CAL_CMD, EMC_ZQ_CAL,
14369b9d8632SJoseph Lo 					      delay);
14379b9d8632SJoseph Lo 
14389b9d8632SJoseph Lo 			value = (mr13_flip_fspop & 0xfffffff7) | 0x0c000000;
14399b9d8632SJoseph Lo 			ccfifo_writel(emc, value, EMC_MRW3, delay);
14409b9d8632SJoseph Lo 			ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
14419b9d8632SJoseph Lo 			ccfifo_writel(emc, 0, EMC_REF, 0);
14429b9d8632SJoseph Lo 			ccfifo_writel(emc, 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
14439b9d8632SJoseph Lo 				      EMC_ZQ_CAL_ZQ_LATCH_CMD,
14449b9d8632SJoseph Lo 				      EMC_ZQ_CAL,
14459b9d8632SJoseph Lo 				      max_t(s32, 0, zq_latch_dvfs_wait_time));
14469b9d8632SJoseph Lo 		} else if (shared_zq_resistor) {
14479b9d8632SJoseph Lo 			if (dst_clk_period > zqcal_before_cc_cutoff)
14489b9d8632SJoseph Lo 				ccfifo_writel(emc,
14499b9d8632SJoseph Lo 					      2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
14509b9d8632SJoseph Lo 					      EMC_ZQ_CAL_ZQ_CAL_CMD, EMC_ZQ_CAL,
14519b9d8632SJoseph Lo 					      delay);
14529b9d8632SJoseph Lo 
14539b9d8632SJoseph Lo 			ccfifo_writel(emc, 2UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
14549b9d8632SJoseph Lo 				      EMC_ZQ_CAL_ZQ_LATCH_CMD, EMC_ZQ_CAL,
14559b9d8632SJoseph Lo 				      max_t(s32, 0, zq_latch_dvfs_wait_time) +
14569b9d8632SJoseph Lo 					delay);
14579b9d8632SJoseph Lo 			ccfifo_writel(emc, 1UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
14589b9d8632SJoseph Lo 				      EMC_ZQ_CAL_ZQ_LATCH_CMD,
14599b9d8632SJoseph Lo 				      EMC_ZQ_CAL, 0);
14609b9d8632SJoseph Lo 
14619b9d8632SJoseph Lo 			value = (mr13_flip_fspop & 0xfffffff7) | 0x0c000000;
14629b9d8632SJoseph Lo 			ccfifo_writel(emc, value, EMC_MRW3, 0);
14639b9d8632SJoseph Lo 			ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
14649b9d8632SJoseph Lo 			ccfifo_writel(emc, 0, EMC_REF, 0);
14659b9d8632SJoseph Lo 
14669b9d8632SJoseph Lo 			ccfifo_writel(emc, 1UL << EMC_ZQ_CAL_DEV_SEL_SHIFT |
14679b9d8632SJoseph Lo 				      EMC_ZQ_CAL_ZQ_LATCH_CMD, EMC_ZQ_CAL,
14689b9d8632SJoseph Lo 				      tZQCAL_lpddr4 / dst_clk_period);
14699b9d8632SJoseph Lo 		} else {
14709b9d8632SJoseph Lo 			if (dst_clk_period > zqcal_before_cc_cutoff)
14719b9d8632SJoseph Lo 				ccfifo_writel(emc, EMC_ZQ_CAL_ZQ_CAL_CMD,
14729b9d8632SJoseph Lo 					      EMC_ZQ_CAL, delay);
14739b9d8632SJoseph Lo 
14749b9d8632SJoseph Lo 			value = (mr13_flip_fspop & 0xfffffff7) | 0x0c000000;
14759b9d8632SJoseph Lo 			ccfifo_writel(emc, value, EMC_MRW3, delay);
14769b9d8632SJoseph Lo 			ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
14779b9d8632SJoseph Lo 			ccfifo_writel(emc, 0, EMC_REF, 0);
14789b9d8632SJoseph Lo 
14799b9d8632SJoseph Lo 			ccfifo_writel(emc, EMC_ZQ_CAL_ZQ_LATCH_CMD, EMC_ZQ_CAL,
14809b9d8632SJoseph Lo 				      max_t(s32, 0, zq_latch_dvfs_wait_time));
14819b9d8632SJoseph Lo 		}
14829b9d8632SJoseph Lo 	}
14839b9d8632SJoseph Lo 
14849b9d8632SJoseph Lo 	/* WAR: delay for zqlatch */
14859b9d8632SJoseph Lo 	ccfifo_writel(emc, 0, 0, 10);
14869b9d8632SJoseph Lo 
14879b9d8632SJoseph Lo 	/*
14889b9d8632SJoseph Lo 	 * Step 16:
14899b9d8632SJoseph Lo 	 *   LPDDR4 Conditional Training Kickoff. Removed.
14909b9d8632SJoseph Lo 	 */
14919b9d8632SJoseph Lo 
14929b9d8632SJoseph Lo 	/*
14939b9d8632SJoseph Lo 	 * Step 17:
14949b9d8632SJoseph Lo 	 *   MANSR exit self refresh.
14959b9d8632SJoseph Lo 	 */
14969b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 17\n");
14979b9d8632SJoseph Lo 
14989b9d8632SJoseph Lo 	if (opt_dvfs_mode == MAN_SR && dram_type != DRAM_TYPE_LPDDR4)
14999b9d8632SJoseph Lo 		ccfifo_writel(emc, 0, EMC_SELF_REF, 0);
15009b9d8632SJoseph Lo 
15019b9d8632SJoseph Lo 	/*
15029b9d8632SJoseph Lo 	 * Step 18:
15039b9d8632SJoseph Lo 	 *   Send MRWs to LPDDR3/DDR3.
15049b9d8632SJoseph Lo 	 */
15059b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 18\n");
15069b9d8632SJoseph Lo 
15079b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR2) {
15089b9d8632SJoseph Lo 		ccfifo_writel(emc, next->emc_mrw2, EMC_MRW2, 0);
15099b9d8632SJoseph Lo 		ccfifo_writel(emc, next->emc_mrw,  EMC_MRW,  0);
15109b9d8632SJoseph Lo 		if (is_lpddr3)
15119b9d8632SJoseph Lo 			ccfifo_writel(emc, next->emc_mrw4, EMC_MRW4, 0);
15129b9d8632SJoseph Lo 	} else if (dram_type == DRAM_TYPE_DDR3) {
15139b9d8632SJoseph Lo 		if (opt_dll_mode)
15149b9d8632SJoseph Lo 			ccfifo_writel(emc, next->emc_emrs &
15159b9d8632SJoseph Lo 				      ~EMC_EMRS_USE_EMRS_LONG_CNT, EMC_EMRS, 0);
15169b9d8632SJoseph Lo 		ccfifo_writel(emc, next->emc_emrs2 &
15179b9d8632SJoseph Lo 			      ~EMC_EMRS2_USE_EMRS2_LONG_CNT, EMC_EMRS2, 0);
15189b9d8632SJoseph Lo 		ccfifo_writel(emc, next->emc_mrs |
15199b9d8632SJoseph Lo 			      EMC_EMRS_USE_EMRS_LONG_CNT, EMC_MRS, 0);
15209b9d8632SJoseph Lo 	}
15219b9d8632SJoseph Lo 
15229b9d8632SJoseph Lo 	/*
15239b9d8632SJoseph Lo 	 * Step 19:
15249b9d8632SJoseph Lo 	 *   ZQCAL for LPDDR3/DDR3
15259b9d8632SJoseph Lo 	 */
15269b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 19\n");
15279b9d8632SJoseph Lo 
15289b9d8632SJoseph Lo 	if (opt_zcal_en_cc) {
15299b9d8632SJoseph Lo 		if (dram_type == DRAM_TYPE_LPDDR2) {
15309b9d8632SJoseph Lo 			value = opt_cc_short_zcal ? 90000 : 360000;
15319b9d8632SJoseph Lo 			value = div_o3(value, dst_clk_period);
15329b9d8632SJoseph Lo 			value = value <<
15339b9d8632SJoseph Lo 				EMC_MRS_WAIT_CNT2_MRS_EXT2_WAIT_CNT_SHIFT |
15349b9d8632SJoseph Lo 				value <<
15359b9d8632SJoseph Lo 				EMC_MRS_WAIT_CNT2_MRS_EXT1_WAIT_CNT_SHIFT;
15369b9d8632SJoseph Lo 			ccfifo_writel(emc, value, EMC_MRS_WAIT_CNT2, 0);
15379b9d8632SJoseph Lo 
15389b9d8632SJoseph Lo 			value = opt_cc_short_zcal ? 0x56 : 0xab;
15399b9d8632SJoseph Lo 			ccfifo_writel(emc, 2 << EMC_MRW_MRW_DEV_SELECTN_SHIFT |
15409b9d8632SJoseph Lo 					   EMC_MRW_USE_MRW_EXT_CNT |
15419b9d8632SJoseph Lo 					   10 << EMC_MRW_MRW_MA_SHIFT |
15429b9d8632SJoseph Lo 					   value << EMC_MRW_MRW_OP_SHIFT,
15439b9d8632SJoseph Lo 				      EMC_MRW, 0);
15449b9d8632SJoseph Lo 
15459b9d8632SJoseph Lo 			if (emc->num_devices > 1) {
15469b9d8632SJoseph Lo 				value = 1 << EMC_MRW_MRW_DEV_SELECTN_SHIFT |
15479b9d8632SJoseph Lo 					EMC_MRW_USE_MRW_EXT_CNT |
15489b9d8632SJoseph Lo 					10 << EMC_MRW_MRW_MA_SHIFT |
15499b9d8632SJoseph Lo 					value << EMC_MRW_MRW_OP_SHIFT;
15509b9d8632SJoseph Lo 				ccfifo_writel(emc, value, EMC_MRW, 0);
15519b9d8632SJoseph Lo 			}
15529b9d8632SJoseph Lo 		} else if (dram_type == DRAM_TYPE_DDR3) {
15539b9d8632SJoseph Lo 			value = opt_cc_short_zcal ? 0 : EMC_ZQ_CAL_LONG;
15549b9d8632SJoseph Lo 
15559b9d8632SJoseph Lo 			ccfifo_writel(emc, value |
15569b9d8632SJoseph Lo 					   2 << EMC_ZQ_CAL_DEV_SEL_SHIFT |
15579b9d8632SJoseph Lo 					   EMC_ZQ_CAL_ZQ_CAL_CMD, EMC_ZQ_CAL,
15589b9d8632SJoseph Lo 					   0);
15599b9d8632SJoseph Lo 
15609b9d8632SJoseph Lo 			if (emc->num_devices > 1) {
15619b9d8632SJoseph Lo 				value = value | 1 << EMC_ZQ_CAL_DEV_SEL_SHIFT |
15629b9d8632SJoseph Lo 						EMC_ZQ_CAL_ZQ_CAL_CMD;
15639b9d8632SJoseph Lo 				ccfifo_writel(emc, value, EMC_ZQ_CAL, 0);
15649b9d8632SJoseph Lo 			}
15659b9d8632SJoseph Lo 		}
15669b9d8632SJoseph Lo 	}
15679b9d8632SJoseph Lo 
15689b9d8632SJoseph Lo 	if (bg_reg_mode_change) {
15699b9d8632SJoseph Lo 		tegra210_emc_set_shadow_bypass(emc, ACTIVE);
15709b9d8632SJoseph Lo 
15719b9d8632SJoseph Lo 		if (ramp_up_wait <= 1250000)
15729b9d8632SJoseph Lo 			delay = (1250000 - ramp_up_wait) / dst_clk_period;
15739b9d8632SJoseph Lo 		else
15749b9d8632SJoseph Lo 			delay = 0;
15759b9d8632SJoseph Lo 
15769b9d8632SJoseph Lo 		ccfifo_writel(emc,
15779b9d8632SJoseph Lo 			      next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX],
15789b9d8632SJoseph Lo 			      EMC_PMACRO_BG_BIAS_CTRL_0, delay);
15799b9d8632SJoseph Lo 		tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
15809b9d8632SJoseph Lo 	}
15819b9d8632SJoseph Lo 
15829b9d8632SJoseph Lo 	/*
15839b9d8632SJoseph Lo 	 * Step 20:
15849b9d8632SJoseph Lo 	 *   Issue ref and optional QRST.
15859b9d8632SJoseph Lo 	 */
15869b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 20\n");
15879b9d8632SJoseph Lo 
15889b9d8632SJoseph Lo 	if (dram_type != DRAM_TYPE_LPDDR4)
15899b9d8632SJoseph Lo 		ccfifo_writel(emc, 0, EMC_REF, 0);
15909b9d8632SJoseph Lo 
15919b9d8632SJoseph Lo 	if (opt_do_sw_qrst) {
15929b9d8632SJoseph Lo 		ccfifo_writel(emc, 1, EMC_ISSUE_QRST, 0);
15939b9d8632SJoseph Lo 		ccfifo_writel(emc, 0, EMC_ISSUE_QRST, 2);
15949b9d8632SJoseph Lo 	}
15959b9d8632SJoseph Lo 
15969b9d8632SJoseph Lo 	/*
15979b9d8632SJoseph Lo 	 * Step 21:
15989b9d8632SJoseph Lo 	 *   Restore ZCAL and ZCAL interval.
15999b9d8632SJoseph Lo 	 */
16009b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 21\n");
16019b9d8632SJoseph Lo 
16029b9d8632SJoseph Lo 	if (save_restore_clkstop_pd || opt_zcal_en_cc) {
16039b9d8632SJoseph Lo 		ccfifo_writel(emc, emc_dbg | EMC_DBG_WRITE_MUX_ACTIVE,
16049b9d8632SJoseph Lo 			      EMC_DBG, 0);
16059b9d8632SJoseph Lo 		if (opt_zcal_en_cc && dram_type != DRAM_TYPE_LPDDR4)
16069b9d8632SJoseph Lo 			ccfifo_writel(emc, next->burst_regs[EMC_ZCAL_INTERVAL_INDEX],
16079b9d8632SJoseph Lo 				      EMC_ZCAL_INTERVAL, 0);
16089b9d8632SJoseph Lo 
16099b9d8632SJoseph Lo 		if (save_restore_clkstop_pd)
16109b9d8632SJoseph Lo 			ccfifo_writel(emc, next->burst_regs[EMC_CFG_INDEX] &
16119b9d8632SJoseph Lo 						~EMC_CFG_DYN_SELF_REF,
16129b9d8632SJoseph Lo 				      EMC_CFG, 0);
16139b9d8632SJoseph Lo 		ccfifo_writel(emc, emc_dbg, EMC_DBG, 0);
16149b9d8632SJoseph Lo 	}
16159b9d8632SJoseph Lo 
16169b9d8632SJoseph Lo 	/*
16179b9d8632SJoseph Lo 	 * Step 22:
16189b9d8632SJoseph Lo 	 *   Restore EMC_CFG_PIPE_CLK.
16199b9d8632SJoseph Lo 	 */
16209b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 22\n");
16219b9d8632SJoseph Lo 
16229b9d8632SJoseph Lo 	ccfifo_writel(emc, emc_cfg_pipe_clk, EMC_CFG_PIPE_CLK, 0);
16239b9d8632SJoseph Lo 
16249b9d8632SJoseph Lo 	if (bg_reg_mode_change) {
16259b9d8632SJoseph Lo 		if (enable_bg_reg)
16269b9d8632SJoseph Lo 			emc_writel(emc,
16279b9d8632SJoseph Lo 				   next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
16289b9d8632SJoseph Lo 					~EMC_PMACRO_BG_BIAS_CTRL_0_BGLP_E_PWRD,
16299b9d8632SJoseph Lo 				   EMC_PMACRO_BG_BIAS_CTRL_0);
16309b9d8632SJoseph Lo 		else
16319b9d8632SJoseph Lo 			emc_writel(emc,
16329b9d8632SJoseph Lo 				   next->burst_regs[EMC_PMACRO_BG_BIAS_CTRL_0_INDEX] &
16339b9d8632SJoseph Lo 					~EMC_PMACRO_BG_BIAS_CTRL_0_BG_E_PWRD,
16349b9d8632SJoseph Lo 				   EMC_PMACRO_BG_BIAS_CTRL_0);
16359b9d8632SJoseph Lo 	}
16369b9d8632SJoseph Lo 
16379b9d8632SJoseph Lo 	/*
16389b9d8632SJoseph Lo 	 * Step 23:
16399b9d8632SJoseph Lo 	 */
16409b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 23\n");
16419b9d8632SJoseph Lo 
16429b9d8632SJoseph Lo 	value = emc_readl(emc, EMC_CFG_DIG_DLL);
16439b9d8632SJoseph Lo 	value |= EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_TRAFFIC;
16449b9d8632SJoseph Lo 	value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_RW_UNTIL_LOCK;
16459b9d8632SJoseph Lo 	value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_UNTIL_LOCK;
16469b9d8632SJoseph Lo 	value &= ~EMC_CFG_DIG_DLL_CFG_DLL_EN;
16479b9d8632SJoseph Lo 	value = (value & ~EMC_CFG_DIG_DLL_CFG_DLL_MODE_MASK) |
16489b9d8632SJoseph Lo 		(2 << EMC_CFG_DIG_DLL_CFG_DLL_MODE_SHIFT);
16499b9d8632SJoseph Lo 	emc_writel(emc, value, EMC_CFG_DIG_DLL);
16509b9d8632SJoseph Lo 
16519b9d8632SJoseph Lo 	tegra210_emc_do_clock_change(emc, clksrc);
16529b9d8632SJoseph Lo 
16539b9d8632SJoseph Lo 	/*
16549b9d8632SJoseph Lo 	 * Step 24:
16559b9d8632SJoseph Lo 	 *   Save training results. Removed.
16569b9d8632SJoseph Lo 	 */
16579b9d8632SJoseph Lo 
16589b9d8632SJoseph Lo 	/*
16599b9d8632SJoseph Lo 	 * Step 25:
16609b9d8632SJoseph Lo 	 *   Program MC updown registers.
16619b9d8632SJoseph Lo 	 */
16629b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 25\n");
16639b9d8632SJoseph Lo 
16649b9d8632SJoseph Lo 	if (next->rate > last->rate) {
16659b9d8632SJoseph Lo 		for (i = 0; i < next->num_up_down; i++)
16669b9d8632SJoseph Lo 			mc_writel(emc->mc, next->la_scale_regs[i],
16679b9d8632SJoseph Lo 				  emc->offsets->la_scale[i]);
16689b9d8632SJoseph Lo 
16699b9d8632SJoseph Lo 		tegra210_emc_timing_update(emc);
16709b9d8632SJoseph Lo 	}
16719b9d8632SJoseph Lo 
16729b9d8632SJoseph Lo 	/*
16739b9d8632SJoseph Lo 	 * Step 26:
16749b9d8632SJoseph Lo 	 *   Restore ZCAL registers.
16759b9d8632SJoseph Lo 	 */
16769b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 26\n");
16779b9d8632SJoseph Lo 
16789b9d8632SJoseph Lo 	if (dram_type == DRAM_TYPE_LPDDR4) {
16799b9d8632SJoseph Lo 		tegra210_emc_set_shadow_bypass(emc, ACTIVE);
16809b9d8632SJoseph Lo 		emc_writel(emc, next->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX],
16819b9d8632SJoseph Lo 			   EMC_ZCAL_WAIT_CNT);
16829b9d8632SJoseph Lo 		emc_writel(emc, next->burst_regs[EMC_ZCAL_INTERVAL_INDEX],
16839b9d8632SJoseph Lo 			   EMC_ZCAL_INTERVAL);
16849b9d8632SJoseph Lo 		tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
16859b9d8632SJoseph Lo 	}
16869b9d8632SJoseph Lo 
16879b9d8632SJoseph Lo 	if (dram_type != DRAM_TYPE_LPDDR4 && opt_zcal_en_cc &&
16889b9d8632SJoseph Lo 	    !opt_short_zcal && opt_cc_short_zcal) {
16899b9d8632SJoseph Lo 		udelay(2);
16909b9d8632SJoseph Lo 
16919b9d8632SJoseph Lo 		tegra210_emc_set_shadow_bypass(emc, ACTIVE);
16929b9d8632SJoseph Lo 		if (dram_type == DRAM_TYPE_LPDDR2)
16939b9d8632SJoseph Lo 			emc_writel(emc, next->burst_regs[EMC_MRS_WAIT_CNT_INDEX],
16949b9d8632SJoseph Lo 				   EMC_MRS_WAIT_CNT);
16959b9d8632SJoseph Lo 		else if (dram_type == DRAM_TYPE_DDR3)
16969b9d8632SJoseph Lo 			emc_writel(emc, next->burst_regs[EMC_ZCAL_WAIT_CNT_INDEX],
16979b9d8632SJoseph Lo 				   EMC_ZCAL_WAIT_CNT);
16989b9d8632SJoseph Lo 		tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
16999b9d8632SJoseph Lo 	}
17009b9d8632SJoseph Lo 
17019b9d8632SJoseph Lo 	/*
17029b9d8632SJoseph Lo 	 * Step 27:
17039b9d8632SJoseph Lo 	 *   Restore EMC_CFG, FDPD registers.
17049b9d8632SJoseph Lo 	 */
17059b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 27\n");
17069b9d8632SJoseph Lo 
17079b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ACTIVE);
17089b9d8632SJoseph Lo 	emc_writel(emc, next->burst_regs[EMC_CFG_INDEX], EMC_CFG);
17099b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
17109b9d8632SJoseph Lo 	emc_writel(emc, next->emc_fdpd_ctrl_cmd_no_ramp,
17119b9d8632SJoseph Lo 		   EMC_FDPD_CTRL_CMD_NO_RAMP);
17129b9d8632SJoseph Lo 	emc_writel(emc, next->emc_sel_dpd_ctrl, EMC_SEL_DPD_CTRL);
17139b9d8632SJoseph Lo 
17149b9d8632SJoseph Lo 	/*
17159b9d8632SJoseph Lo 	 * Step 28:
17169b9d8632SJoseph Lo 	 *   Training recover. Removed.
17179b9d8632SJoseph Lo 	 */
17189b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 28\n");
17199b9d8632SJoseph Lo 
17209b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ACTIVE);
17219b9d8632SJoseph Lo 	emc_writel(emc,
17229b9d8632SJoseph Lo 		   next->burst_regs[EMC_PMACRO_AUTOCAL_CFG_COMMON_INDEX],
17239b9d8632SJoseph Lo 		   EMC_PMACRO_AUTOCAL_CFG_COMMON);
17249b9d8632SJoseph Lo 	tegra210_emc_set_shadow_bypass(emc, ASSEMBLY);
17259b9d8632SJoseph Lo 
17269b9d8632SJoseph Lo 	/*
17279b9d8632SJoseph Lo 	 * Step 29:
17289b9d8632SJoseph Lo 	 *   Power fix WAR.
17299b9d8632SJoseph Lo 	 */
17309b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 29\n");
17319b9d8632SJoseph Lo 
17329b9d8632SJoseph Lo 	emc_writel(emc, EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE0 |
17339b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE1 |
17349b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE2 |
17359b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE3 |
17369b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE4 |
17379b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE5 |
17389b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE6 |
17399b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0_DISABLE_CFG_BYTE7,
17409b9d8632SJoseph Lo 		   EMC_PMACRO_CFG_PM_GLOBAL_0);
17419b9d8632SJoseph Lo 	emc_writel(emc, EMC_PMACRO_TRAINING_CTRL_0_CH0_TRAINING_E_WRPTR,
17429b9d8632SJoseph Lo 		   EMC_PMACRO_TRAINING_CTRL_0);
17439b9d8632SJoseph Lo 	emc_writel(emc, EMC_PMACRO_TRAINING_CTRL_1_CH1_TRAINING_E_WRPTR,
17449b9d8632SJoseph Lo 		   EMC_PMACRO_TRAINING_CTRL_1);
17459b9d8632SJoseph Lo 	emc_writel(emc, 0, EMC_PMACRO_CFG_PM_GLOBAL_0);
17469b9d8632SJoseph Lo 
17479b9d8632SJoseph Lo 	/*
17489b9d8632SJoseph Lo 	 * Step 30:
17499b9d8632SJoseph Lo 	 *   Re-enable autocal.
17509b9d8632SJoseph Lo 	 */
17519b9d8632SJoseph Lo 	emc_dbg(emc, STEPS, "Step 30: Re-enable DLL and AUTOCAL\n");
17529b9d8632SJoseph Lo 
17539b9d8632SJoseph Lo 	if (next->burst_regs[EMC_CFG_DIG_DLL_INDEX] & EMC_CFG_DIG_DLL_CFG_DLL_EN) {
17549b9d8632SJoseph Lo 		value = emc_readl(emc, EMC_CFG_DIG_DLL);
17559b9d8632SJoseph Lo 		value |=  EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_TRAFFIC;
17569b9d8632SJoseph Lo 		value |=  EMC_CFG_DIG_DLL_CFG_DLL_EN;
17579b9d8632SJoseph Lo 		value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_RW_UNTIL_LOCK;
17589b9d8632SJoseph Lo 		value &= ~EMC_CFG_DIG_DLL_CFG_DLL_STALL_ALL_UNTIL_LOCK;
17599b9d8632SJoseph Lo 		value = (value & ~EMC_CFG_DIG_DLL_CFG_DLL_MODE_MASK) |
17609b9d8632SJoseph Lo 			(2 << EMC_CFG_DIG_DLL_CFG_DLL_MODE_SHIFT);
17619b9d8632SJoseph Lo 		emc_writel(emc, value, EMC_CFG_DIG_DLL);
17629b9d8632SJoseph Lo 		tegra210_emc_timing_update(emc);
17639b9d8632SJoseph Lo 	}
17649b9d8632SJoseph Lo 
17659b9d8632SJoseph Lo 	emc_writel(emc, next->emc_auto_cal_config, EMC_AUTO_CAL_CONFIG);
17669b9d8632SJoseph Lo 
17679b9d8632SJoseph Lo 	/* Done! Yay. */
17689b9d8632SJoseph Lo }
17699b9d8632SJoseph Lo 
17709b9d8632SJoseph Lo const struct tegra210_emc_sequence tegra210_emc_r21021 = {
17719b9d8632SJoseph Lo 	.revision = 0x7,
17729b9d8632SJoseph Lo 	.set_clock = tegra210_emc_r21021_set_clock,
17739b9d8632SJoseph Lo 	.periodic_compensation = tegra210_emc_r21021_periodic_compensation,
17749b9d8632SJoseph Lo };
1775