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