xref: /OK3568_Linux_fs/u-boot/arch/arm/mach-rockchip/sdram.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright (C) 2017 Rockchip Electronics Co., Ltd.
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * SPDX-License-Identifier:     GPL-2.0+
5*4882a593Smuzhiyun  */
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun #include <common.h>
8*4882a593Smuzhiyun #include <bidram.h>
9*4882a593Smuzhiyun #include <dm.h>
10*4882a593Smuzhiyun #include <ram.h>
11*4882a593Smuzhiyun #include <asm/io.h>
12*4882a593Smuzhiyun #include <asm/arch/param.h>
13*4882a593Smuzhiyun #include <asm/arch/rk_atags.h>
14*4882a593Smuzhiyun #include <asm/arch/sdram.h>
15*4882a593Smuzhiyun #include <dm/uclass-internal.h>
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun DECLARE_GLOBAL_DATA_PTR;
18*4882a593Smuzhiyun #define PARAM_DRAM_INFO_OFFSET 0x2000000
19*4882a593Smuzhiyun #define TRUST_PARAMETER_OFFSET    (34 * 1024 * 1024)
20*4882a593Smuzhiyun 
rockchip_sdram_size(phys_addr_t reg)21*4882a593Smuzhiyun size_t rockchip_sdram_size(phys_addr_t reg)
22*4882a593Smuzhiyun {
23*4882a593Smuzhiyun 	u32 rank, cs0_col, bk, cs0_row, cs1_row, bw, row_3_4;
24*4882a593Smuzhiyun 	size_t chipsize_mb = 0;
25*4882a593Smuzhiyun 	size_t size_mb = 0;
26*4882a593Smuzhiyun 	u32 ch;
27*4882a593Smuzhiyun 	u32 cs1_col = 0;
28*4882a593Smuzhiyun 	u32 bg = 0;
29*4882a593Smuzhiyun 	u32 dbw, dram_type;
30*4882a593Smuzhiyun 	u32 sys_reg = readl(reg);
31*4882a593Smuzhiyun 	u32 sys_reg1 = readl(reg + 4);
32*4882a593Smuzhiyun 	u32 ch_num = 1 + ((sys_reg >> SYS_REG_NUM_CH_SHIFT)
33*4882a593Smuzhiyun 		       & SYS_REG_NUM_CH_MASK);
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun 	dram_type = (sys_reg >> SYS_REG_DDRTYPE_SHIFT) & SYS_REG_DDRTYPE_MASK;
36*4882a593Smuzhiyun 	debug("%s %x %x\n", __func__, (u32)reg, sys_reg);
37*4882a593Smuzhiyun 	for (ch = 0; ch < ch_num; ch++) {
38*4882a593Smuzhiyun 		rank = 1 + (sys_reg >> SYS_REG_RANK_SHIFT(ch) &
39*4882a593Smuzhiyun 			SYS_REG_RANK_MASK);
40*4882a593Smuzhiyun 		cs0_col = 9 + (sys_reg >> SYS_REG_COL_SHIFT(ch) & SYS_REG_COL_MASK);
41*4882a593Smuzhiyun 		cs1_col = cs0_col;
42*4882a593Smuzhiyun 		bk = 3 - ((sys_reg >> SYS_REG_BK_SHIFT(ch)) & SYS_REG_BK_MASK);
43*4882a593Smuzhiyun 		if ((sys_reg1 >> SYS_REG1_VERSION_SHIFT &
44*4882a593Smuzhiyun 		     SYS_REG1_VERSION_MASK) == 0x2) {
45*4882a593Smuzhiyun 			cs1_col = 9 + (sys_reg1 >> SYS_REG1_CS1_COL_SHIFT(ch) &
46*4882a593Smuzhiyun 				  SYS_REG1_CS1_COL_MASK);
47*4882a593Smuzhiyun 			if (((sys_reg1 >> SYS_REG1_EXTEND_CS0_ROW_SHIFT(ch) &
48*4882a593Smuzhiyun 			    SYS_REG1_EXTEND_CS0_ROW_MASK) << 2) + (sys_reg >>
49*4882a593Smuzhiyun 			    SYS_REG_CS0_ROW_SHIFT(ch) &
50*4882a593Smuzhiyun 			    SYS_REG_CS0_ROW_MASK) == 7)
51*4882a593Smuzhiyun 				cs0_row = 12;
52*4882a593Smuzhiyun 			else
53*4882a593Smuzhiyun 				cs0_row = 13 + (sys_reg >>
54*4882a593Smuzhiyun 					  SYS_REG_CS0_ROW_SHIFT(ch) &
55*4882a593Smuzhiyun 					  SYS_REG_CS0_ROW_MASK) +
56*4882a593Smuzhiyun 					  ((sys_reg1 >>
57*4882a593Smuzhiyun 					  SYS_REG1_EXTEND_CS0_ROW_SHIFT(ch) &
58*4882a593Smuzhiyun 					  SYS_REG1_EXTEND_CS0_ROW_MASK) << 2);
59*4882a593Smuzhiyun 			if (((sys_reg1 >> SYS_REG1_EXTEND_CS1_ROW_SHIFT(ch) &
60*4882a593Smuzhiyun 			    SYS_REG1_EXTEND_CS1_ROW_MASK) << 2) + (sys_reg >>
61*4882a593Smuzhiyun 			    SYS_REG_CS1_ROW_SHIFT(ch) &
62*4882a593Smuzhiyun 			    SYS_REG_CS1_ROW_MASK) == 7)
63*4882a593Smuzhiyun 				cs1_row = 12;
64*4882a593Smuzhiyun 			else
65*4882a593Smuzhiyun 				cs1_row = 13 + (sys_reg >>
66*4882a593Smuzhiyun 					  SYS_REG_CS1_ROW_SHIFT(ch) &
67*4882a593Smuzhiyun 					  SYS_REG_CS1_ROW_MASK) +
68*4882a593Smuzhiyun 					  ((sys_reg1 >>
69*4882a593Smuzhiyun 					  SYS_REG1_EXTEND_CS1_ROW_SHIFT(ch) &
70*4882a593Smuzhiyun 					  SYS_REG1_EXTEND_CS1_ROW_MASK) << 2);
71*4882a593Smuzhiyun 		}
72*4882a593Smuzhiyun 		else {
73*4882a593Smuzhiyun 			cs0_row = 13 + (sys_reg >> SYS_REG_CS0_ROW_SHIFT(ch) &
74*4882a593Smuzhiyun 				SYS_REG_CS0_ROW_MASK);
75*4882a593Smuzhiyun 			cs1_row = 13 + (sys_reg >> SYS_REG_CS1_ROW_SHIFT(ch) &
76*4882a593Smuzhiyun 				SYS_REG_CS1_ROW_MASK);
77*4882a593Smuzhiyun 		}
78*4882a593Smuzhiyun 		bw = (2 >> ((sys_reg >> SYS_REG_BW_SHIFT(ch)) &
79*4882a593Smuzhiyun 			SYS_REG_BW_MASK));
80*4882a593Smuzhiyun 		row_3_4 = sys_reg >> SYS_REG_ROW_3_4_SHIFT(ch) &
81*4882a593Smuzhiyun 			SYS_REG_ROW_3_4_MASK;
82*4882a593Smuzhiyun 		if (dram_type == DDR4) {
83*4882a593Smuzhiyun 			dbw = (sys_reg >> SYS_REG_DBW_SHIFT(ch)) &
84*4882a593Smuzhiyun 				SYS_REG_DBW_MASK;
85*4882a593Smuzhiyun 			bg = (dbw == 2) ? 2 : 1;
86*4882a593Smuzhiyun 		}
87*4882a593Smuzhiyun 		chipsize_mb = (1 << (cs0_row + cs0_col + bk + bg + bw - 20));
88*4882a593Smuzhiyun 
89*4882a593Smuzhiyun 		if (rank > 1)
90*4882a593Smuzhiyun 			chipsize_mb += chipsize_mb >> ((cs0_row - cs1_row) +
91*4882a593Smuzhiyun 				       (cs0_col - cs1_col));
92*4882a593Smuzhiyun 		if (row_3_4)
93*4882a593Smuzhiyun 			chipsize_mb = chipsize_mb * 3 / 4;
94*4882a593Smuzhiyun 		size_mb += chipsize_mb;
95*4882a593Smuzhiyun 		if (rank > 1)
96*4882a593Smuzhiyun 			debug("rank %d cs0_col %d cs1_col %d bk %d cs0_row %d\
97*4882a593Smuzhiyun 			       cs1_row %d bw %d row_3_4 %d\n",
98*4882a593Smuzhiyun 			       rank, cs0_col, cs1_col, bk, cs0_row,
99*4882a593Smuzhiyun 			       cs1_row, bw, row_3_4);
100*4882a593Smuzhiyun 		else
101*4882a593Smuzhiyun 			debug("rank %d cs0_col %d bk %d cs0_row %d\
102*4882a593Smuzhiyun                                bw %d row_3_4 %d\n",
103*4882a593Smuzhiyun                                rank, cs0_col, bk, cs0_row,
104*4882a593Smuzhiyun                                bw, row_3_4);
105*4882a593Smuzhiyun 	}
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun 	/* Handle 4GB size, or else size will be 0 after <<20 in 32bit system */
108*4882a593Smuzhiyun 	if (size_mb > (SDRAM_MAX_SIZE >> 20))
109*4882a593Smuzhiyun 		size_mb = (SDRAM_MAX_SIZE >> 20);
110*4882a593Smuzhiyun 
111*4882a593Smuzhiyun 	return (size_t)size_mb << 20;
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun 
get_ddr_os_reg(void)114*4882a593Smuzhiyun static unsigned int get_ddr_os_reg(void)
115*4882a593Smuzhiyun {
116*4882a593Smuzhiyun 	u32 os_reg = 0;
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun #if defined(CONFIG_ROCKCHIP_PX30)
119*4882a593Smuzhiyun 	os_reg = readl(0xff010208);
120*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK3328)
121*4882a593Smuzhiyun 	os_reg = readl(0xff1005d0);
122*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK3399)
123*4882a593Smuzhiyun 	os_reg = readl(0xff320308);
124*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK322X)
125*4882a593Smuzhiyun 	os_reg = readl(0x110005d0);
126*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK3368)
127*4882a593Smuzhiyun 	os_reg = readl(0xff738208);
128*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK3288)
129*4882a593Smuzhiyun 	os_reg = readl(0x20004048);
130*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK3036)
131*4882a593Smuzhiyun 	os_reg = readl(0x200081cc);
132*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK3308)
133*4882a593Smuzhiyun 	os_reg = readl(0xff000508);
134*4882a593Smuzhiyun #elif defined(CONFIG_ROCKCHIP_RK1808)
135*4882a593Smuzhiyun 	os_reg = readl(0xfe020208);
136*4882a593Smuzhiyun #else
137*4882a593Smuzhiyun 	printf("unsupported chip type, get page size fail\n");
138*4882a593Smuzhiyun #endif
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun 	return os_reg;
141*4882a593Smuzhiyun }
142*4882a593Smuzhiyun 
get_page_size(void)143*4882a593Smuzhiyun unsigned int get_page_size(void)
144*4882a593Smuzhiyun {
145*4882a593Smuzhiyun 	u32 os_reg;
146*4882a593Smuzhiyun 	u32 col, bw;
147*4882a593Smuzhiyun 	int page_size;
148*4882a593Smuzhiyun 
149*4882a593Smuzhiyun 	os_reg = get_ddr_os_reg();
150*4882a593Smuzhiyun 	if (!os_reg)
151*4882a593Smuzhiyun 		return 0;
152*4882a593Smuzhiyun 
153*4882a593Smuzhiyun 	col = 9 + (os_reg >> SYS_REG_COL_SHIFT(0) & SYS_REG_COL_MASK);
154*4882a593Smuzhiyun 	bw = (2 >> ((os_reg >> SYS_REG_BW_SHIFT(0)) & SYS_REG_BW_MASK));
155*4882a593Smuzhiyun 	page_size = 1u << (col + bw);
156*4882a593Smuzhiyun 
157*4882a593Smuzhiyun 	return page_size;
158*4882a593Smuzhiyun }
159*4882a593Smuzhiyun 
get_ddr_bw(void)160*4882a593Smuzhiyun unsigned int get_ddr_bw(void)
161*4882a593Smuzhiyun {
162*4882a593Smuzhiyun 	u32 os_reg;
163*4882a593Smuzhiyun 	u32 bw = 2;
164*4882a593Smuzhiyun 
165*4882a593Smuzhiyun 	os_reg = get_ddr_os_reg();
166*4882a593Smuzhiyun 	if (os_reg)
167*4882a593Smuzhiyun 		bw = 2 >> ((os_reg >> SYS_REG_BW_SHIFT(0)) & SYS_REG_BW_MASK);
168*4882a593Smuzhiyun 	return bw;
169*4882a593Smuzhiyun }
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun #if defined(CONFIG_SPL_FRAMEWORK) || !defined(CONFIG_SPL_OF_PLATDATA)
dram_init_banksize(void)172*4882a593Smuzhiyun int dram_init_banksize(void)
173*4882a593Smuzhiyun {
174*4882a593Smuzhiyun #ifdef CONFIG_BIDRAM
175*4882a593Smuzhiyun 	bidram_gen_gd_bi_dram();
176*4882a593Smuzhiyun #else
177*4882a593Smuzhiyun 	param_simple_parse_ddr_mem(1);
178*4882a593Smuzhiyun #endif
179*4882a593Smuzhiyun 	return 0;
180*4882a593Smuzhiyun }
181*4882a593Smuzhiyun 
dram_init(void)182*4882a593Smuzhiyun int dram_init(void)
183*4882a593Smuzhiyun {
184*4882a593Smuzhiyun #ifdef CONFIG_BIDRAM
185*4882a593Smuzhiyun 	gd->ram_size = bidram_get_ram_size();
186*4882a593Smuzhiyun #else
187*4882a593Smuzhiyun 	gd->ram_size = param_simple_parse_ddr_mem(0);
188*4882a593Smuzhiyun #endif
189*4882a593Smuzhiyun 	if (!gd->ram_size)
190*4882a593Smuzhiyun 		return -ENOMEM;
191*4882a593Smuzhiyun 
192*4882a593Smuzhiyun 	return 0;
193*4882a593Smuzhiyun }
194*4882a593Smuzhiyun #endif
195*4882a593Smuzhiyun 
board_get_usable_ram_top(ulong total_size)196*4882a593Smuzhiyun ulong board_get_usable_ram_top(ulong total_size)
197*4882a593Smuzhiyun {
198*4882a593Smuzhiyun 	unsigned long top = CONFIG_SYS_SDRAM_BASE + SDRAM_MAX_SIZE;
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun 	return (gd->ram_top > top) ? top : gd->ram_top;
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun 
rockchip_setup_ddr_param(struct ddr_param * info)203*4882a593Smuzhiyun int rockchip_setup_ddr_param(struct ddr_param *info)
204*4882a593Smuzhiyun {
205*4882a593Smuzhiyun 	u32 i;
206*4882a593Smuzhiyun 	struct ddr_param *dinfo = (struct ddr_param *)(CONFIG_SYS_SDRAM_BASE +
207*4882a593Smuzhiyun 					PARAM_DRAM_INFO_OFFSET);
208*4882a593Smuzhiyun 
209*4882a593Smuzhiyun 	dinfo->count = info->count;
210*4882a593Smuzhiyun 	for (i = 0; i < (info->count * 2); i++)
211*4882a593Smuzhiyun 		dinfo->para[i] = info->para[i];
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	return 0;
214*4882a593Smuzhiyun }
215