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