xref: /OK3568_Linux_fs/u-boot/arch/arm/mach-orion5x/lowlevel_init.S (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun/*
2*4882a593Smuzhiyun * Copyright (C) 2010 Albert ARIBAUD <albert.u.boot@aribaud.net>
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * (C) Copyright 2009
5*4882a593Smuzhiyun * Marvell Semiconductor <www.marvell.com>
6*4882a593Smuzhiyun * Written-by: Prafulla Wadaskar <prafulla@marvell.com>
7*4882a593Smuzhiyun *
8*4882a593Smuzhiyun * SPDX-License-Identifier:	GPL-2.0+
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun#include <config.h>
12*4882a593Smuzhiyun#include "asm/arch/orion5x.h"
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun/*
15*4882a593Smuzhiyun * Configuration values for SDRAM access setup
16*4882a593Smuzhiyun */
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun#define SDRAM_CONFIG			0x3148400
19*4882a593Smuzhiyun#define SDRAM_MODE			0x62
20*4882a593Smuzhiyun#define SDRAM_CONTROL			0x4041000
21*4882a593Smuzhiyun#define SDRAM_TIME_CTRL_LOW		0x11602220
22*4882a593Smuzhiyun#define SDRAM_TIME_CTRL_HI		0x40c
23*4882a593Smuzhiyun#define SDRAM_OPEN_PAGE_EN		0x0
24*4882a593Smuzhiyun/* DDR 1 2x 32M NANYA NT5DS16M16CS-6K ==> 64MB */
25*4882a593Smuzhiyun#define SDRAM_BANK0_SIZE		0x3ff0001
26*4882a593Smuzhiyun#define SDRAM_ADDR_CTRL			0x10
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun#define SDRAM_OP_NOP			0x05
29*4882a593Smuzhiyun#define SDRAM_OP_SETMODE		0x03
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun#define SDRAM_PAD_CTRL_WR_EN		0x80000000
32*4882a593Smuzhiyun#define SDRAM_PAD_CTRL_TUNE_EN		0x00010000
33*4882a593Smuzhiyun#define SDRAM_PAD_CTRL_DRVN_MASK	0x0000003f
34*4882a593Smuzhiyun#define SDRAM_PAD_CTRL_DRVP_MASK	0x00000fc0
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun/*
37*4882a593Smuzhiyun * For Guideline MEM-3 - Drive Strength value
38*4882a593Smuzhiyun */
39*4882a593Smuzhiyun
40*4882a593Smuzhiyun#define DDR1_PAD_STRENGTH_DEFAULT	0x00001000
41*4882a593Smuzhiyun#define SDRAM_PAD_CTRL_DRV_STR_MASK	0x00003000
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun/*
44*4882a593Smuzhiyun * For Guideline MEM-4 - DQS Reference Delay Tuning
45*4882a593Smuzhiyun */
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_MASK		0x000000f0
48*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_H_MASK		0x00000100
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_333_167		0x00000000
51*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_500_167		0x00000030
52*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_667_167		0x00000060
53*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_400_200_1	0x000001E0
54*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_400_200		0x00000010
55*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_600_200		0x00000050
56*4882a593Smuzhiyun#define MSAR_ARMDDRCLCK_800_200		0x00000070
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun#define FTDLL_DDR1_166MHZ		0x0047F001
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun#define FTDLL_DDR1_200MHZ		0x0044D001
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun/*
63*4882a593Smuzhiyun * Low-level init happens right after start.S has switched to SVC32,
64*4882a593Smuzhiyun * flushed and disabled caches and disabled MMU. We're still running
65*4882a593Smuzhiyun * from the boot chip select, so the first thing SPL should do is to
66*4882a593Smuzhiyun * set up the RAM to copy U-Boot into.
67*4882a593Smuzhiyun */
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun.globl lowlevel_init
70*4882a593Smuzhiyun
71*4882a593Smuzhiyunlowlevel_init:
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun#ifdef CONFIG_SPL_BUILD
74*4882a593Smuzhiyun
75*4882a593Smuzhiyun	/* Use 'r4 as the base for internal register accesses */
76*4882a593Smuzhiyun	ldr	r4, =ORION5X_REGS_PHY_BASE
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun	/* move internal registers from the default 0xD0000000
79*4882a593Smuzhiyun	 * to their intended location, defined by SoC */
80*4882a593Smuzhiyun	ldr	r3, =0xD0000000
81*4882a593Smuzhiyun	add	r3, r3, #0x20000
82*4882a593Smuzhiyun	str	r4, [r3, #0x80]
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun	/* Use R3 as the base for DRAM registers */
85*4882a593Smuzhiyun	add	r3, r4, #0x01000
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun	/*DDR SDRAM Initialization Control */
88*4882a593Smuzhiyun	ldr	r6, =0x00000001
89*4882a593Smuzhiyun	str	r6, [r3, #0x480]
90*4882a593Smuzhiyun
91*4882a593Smuzhiyun	/* Use R3 as the base for PCI registers */
92*4882a593Smuzhiyun	add	r3, r4, #0x31000
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun	/* Disable arbiter */
95*4882a593Smuzhiyun	ldr	r6, =0x00000030
96*4882a593Smuzhiyun	str	r6, [r3, #0xd00]
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun	/* Use R3 as the base for DRAM registers */
99*4882a593Smuzhiyun	add	r3, r4, #0x01000
100*4882a593Smuzhiyun
101*4882a593Smuzhiyun	/* set all dram windows to 0 */
102*4882a593Smuzhiyun	mov	r6, #0
103*4882a593Smuzhiyun	str	r6, [r3, #0x504]
104*4882a593Smuzhiyun	str	r6, [r3, #0x50C]
105*4882a593Smuzhiyun	str	r6, [r3, #0x514]
106*4882a593Smuzhiyun	str	r6, [r3, #0x51C]
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun	/* 1) Configure SDRAM  */
109*4882a593Smuzhiyun	ldr	r6, =SDRAM_CONFIG
110*4882a593Smuzhiyun	str	r6, [r3, #0x400]
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun	/* 2) Set SDRAM Control reg */
113*4882a593Smuzhiyun	ldr	r6, =SDRAM_CONTROL
114*4882a593Smuzhiyun	str	r6, [r3, #0x404]
115*4882a593Smuzhiyun
116*4882a593Smuzhiyun	/* 3) Write SDRAM address control register */
117*4882a593Smuzhiyun	ldr	r6, =SDRAM_ADDR_CTRL
118*4882a593Smuzhiyun	str	r6, [r3, #0x410]
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun	/* 4) Write SDRAM bank 0 size register */
121*4882a593Smuzhiyun	ldr	r6, =SDRAM_BANK0_SIZE
122*4882a593Smuzhiyun	str	r6, [r3, #0x504]
123*4882a593Smuzhiyun	/* keep other banks disabled */
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun	/* 5) Write SDRAM open pages control register */
126*4882a593Smuzhiyun	ldr	r6, =SDRAM_OPEN_PAGE_EN
127*4882a593Smuzhiyun	str	r6, [r3, #0x414]
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun	/* 6) Write SDRAM timing Low register */
130*4882a593Smuzhiyun	ldr	r6, =SDRAM_TIME_CTRL_LOW
131*4882a593Smuzhiyun	str	r6, [r3, #0x408]
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun	/* 7) Write SDRAM timing High register */
134*4882a593Smuzhiyun	ldr	r6, =SDRAM_TIME_CTRL_HI
135*4882a593Smuzhiyun	str	r6, [r3, #0x40C]
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun	/* 8) Write SDRAM mode register */
138*4882a593Smuzhiyun	/* The CPU must not attempt to change the SDRAM Mode register setting */
139*4882a593Smuzhiyun	/* prior to DRAM controller completion of the DRAM initialization     */
140*4882a593Smuzhiyun	/* sequence. To guarantee this restriction, it is recommended that    */
141*4882a593Smuzhiyun	/* the CPU sets the SDRAM Operation register to NOP command, performs */
142*4882a593Smuzhiyun	/* read polling until the register is back in Normal operation value, */
143*4882a593Smuzhiyun	/* and then sets SDRAM Mode register to its new value.		      */
144*4882a593Smuzhiyun
145*4882a593Smuzhiyun	/* 8.1 write 'nop' to SDRAM operation */
146*4882a593Smuzhiyun	ldr	r6, =SDRAM_OP_NOP
147*4882a593Smuzhiyun	str	r6, [r3, #0x418]
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun	/* 8.2 poll SDRAM operation until back in 'normal' mode.  */
150*4882a593Smuzhiyun1:
151*4882a593Smuzhiyun	ldr	r6, [r3, #0x418]
152*4882a593Smuzhiyun	cmp	r6, #0
153*4882a593Smuzhiyun	bne	1b
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun	/* 8.3 Now its safe to write new value to SDRAM Mode register	      */
156*4882a593Smuzhiyun	ldr	r6, =SDRAM_MODE
157*4882a593Smuzhiyun	str	r6, [r3, #0x41C]
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun	/* 8.4 Set new mode */
160*4882a593Smuzhiyun	ldr	r6, =SDRAM_OP_SETMODE
161*4882a593Smuzhiyun	str	r6, [r3, #0x418]
162*4882a593Smuzhiyun
163*4882a593Smuzhiyun	/* 8.5 poll SDRAM operation until back in 'normal' mode.  */
164*4882a593Smuzhiyun2:
165*4882a593Smuzhiyun	ldr	r6, [r3, #0x418]
166*4882a593Smuzhiyun	cmp	r6, #0
167*4882a593Smuzhiyun	bne	2b
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun	/* DDR SDRAM Address/Control Pads Calibration */
170*4882a593Smuzhiyun	ldr	r6, [r3, #0x4C0]
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun	/* Set Bit [31] to make the register writable			*/
173*4882a593Smuzhiyun	orr	r6, r6, #SDRAM_PAD_CTRL_WR_EN
174*4882a593Smuzhiyun	str	r6, [r3, #0x4C0]
175*4882a593Smuzhiyun
176*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_WR_EN
177*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_TUNE_EN
178*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_DRVN_MASK
179*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_DRVP_MASK
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun	/* Get the final N locked value of driving strength [22:17]	*/
182*4882a593Smuzhiyun	mov	r1, r6
183*4882a593Smuzhiyun	mov	r1, r1, LSL #9
184*4882a593Smuzhiyun	mov	r1, r1, LSR #26	 /* r1[5:0]<DrvN>  = r3[22:17]<LockN>	*/
185*4882a593Smuzhiyun	orr	r1, r1, r1, LSL #6 /* r1[11:6]<DrvP> = r1[5:0]<DrvN>	*/
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun	/* Write to both <DrvN> bits [5:0] and <DrvP> bits [11:6]	*/
188*4882a593Smuzhiyun	orr	r6, r6, r1
189*4882a593Smuzhiyun	str	r6, [r3, #0x4C0]
190*4882a593Smuzhiyun
191*4882a593Smuzhiyun	/* DDR SDRAM Data Pads Calibration				*/
192*4882a593Smuzhiyun	ldr	r6, [r3, #0x4C4]
193*4882a593Smuzhiyun
194*4882a593Smuzhiyun	/* Set Bit [31] to make the register writable			*/
195*4882a593Smuzhiyun	orr	r6, r6, #SDRAM_PAD_CTRL_WR_EN
196*4882a593Smuzhiyun	str	r6, [r3, #0x4C4]
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_WR_EN
199*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_TUNE_EN
200*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_DRVN_MASK
201*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_DRVP_MASK
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun	/* Get the final N locked value of driving strength [22:17]	*/
204*4882a593Smuzhiyun	mov	r1, r6
205*4882a593Smuzhiyun	mov	r1, r1, LSL #9
206*4882a593Smuzhiyun	mov	r1, r1, LSR #26
207*4882a593Smuzhiyun	orr	r1, r1, r1, LSL #6 /* r1[5:0] = r3[22:17]<LockN>	*/
208*4882a593Smuzhiyun
209*4882a593Smuzhiyun	/* Write to both <DrvN> bits [5:0] and <DrvP> bits [11:6]	*/
210*4882a593Smuzhiyun	orr	r6, r6, r1
211*4882a593Smuzhiyun
212*4882a593Smuzhiyun	str	r6, [r3, #0x4C4]
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun	/* Implement Guideline (GL# MEM-3) Drive Strength Value		*/
215*4882a593Smuzhiyun	/* Relevant for: 88F5181-A1/B0/B1 and 88F5281-A0/B0		*/
216*4882a593Smuzhiyun
217*4882a593Smuzhiyun	ldr	r1, =DDR1_PAD_STRENGTH_DEFAULT
218*4882a593Smuzhiyun
219*4882a593Smuzhiyun	/* Enable writes to DDR SDRAM Addr/Ctrl Pads Calibration register */
220*4882a593Smuzhiyun	ldr	r6, [r3, #0x4C0]
221*4882a593Smuzhiyun	orr	r6, r6, #SDRAM_PAD_CTRL_WR_EN
222*4882a593Smuzhiyun	str	r6, [r3, #0x4C0]
223*4882a593Smuzhiyun
224*4882a593Smuzhiyun	/* Correct strength and disable writes again */
225*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_WR_EN
226*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_DRV_STR_MASK
227*4882a593Smuzhiyun	orr	r6, r6, r1
228*4882a593Smuzhiyun	str	r6, [r3, #0x4C0]
229*4882a593Smuzhiyun
230*4882a593Smuzhiyun	/* Enable writes to DDR SDRAM Data Pads Calibration register */
231*4882a593Smuzhiyun	ldr	r6, [r3, #0x4C4]
232*4882a593Smuzhiyun	orr	r6, r6, #SDRAM_PAD_CTRL_WR_EN
233*4882a593Smuzhiyun	str	r6, [r3, #0x4C4]
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun	/* Correct strength and disable writes again */
236*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_DRV_STR_MASK
237*4882a593Smuzhiyun	bic	r6, r6, #SDRAM_PAD_CTRL_WR_EN
238*4882a593Smuzhiyun	orr	r6, r6, r1
239*4882a593Smuzhiyun	str	r6, [r3, #0x4C4]
240*4882a593Smuzhiyun
241*4882a593Smuzhiyun	/* Implement Guideline (GL# MEM-4) DQS Reference Delay Tuning	*/
242*4882a593Smuzhiyun	/* Relevant for: 88F5181-A1/B0/B1 and 88F5281-A0/B0		*/
243*4882a593Smuzhiyun
244*4882a593Smuzhiyun	/* Get the "sample on reset" register for the DDR frequancy	*/
245*4882a593Smuzhiyun	ldr	r3, =0x10000
246*4882a593Smuzhiyun	ldr	r6, [r3, #0x010]
247*4882a593Smuzhiyun	ldr	r1, =MSAR_ARMDDRCLCK_MASK
248*4882a593Smuzhiyun	and	r1, r6, r1
249*4882a593Smuzhiyun
250*4882a593Smuzhiyun	ldr	r6, =FTDLL_DDR1_166MHZ
251*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_333_167
252*4882a593Smuzhiyun	beq	3f
253*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_500_167
254*4882a593Smuzhiyun	beq	3f
255*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_667_167
256*4882a593Smuzhiyun	beq	3f
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun	ldr	r6, =FTDLL_DDR1_200MHZ
259*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_400_200_1
260*4882a593Smuzhiyun	beq	3f
261*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_400_200
262*4882a593Smuzhiyun	beq	3f
263*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_600_200
264*4882a593Smuzhiyun	beq	3f
265*4882a593Smuzhiyun	cmp	r1, #MSAR_ARMDDRCLCK_800_200
266*4882a593Smuzhiyun	beq	3f
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun	ldr	r6, =0
269*4882a593Smuzhiyun
270*4882a593Smuzhiyun3:
271*4882a593Smuzhiyun	/* Use R3 as the base for DRAM registers */
272*4882a593Smuzhiyun	add	r3, r4, #0x01000
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun	ldr	r2, [r3, #0x484]
275*4882a593Smuzhiyun	orr	r2, r2, r6
276*4882a593Smuzhiyun	str	r2, [r3, #0x484]
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun	/* enable for 2 GB DDR; detection should find out real amount */
279*4882a593Smuzhiyun	sub	r6, r6, r6
280*4882a593Smuzhiyun	str	r6, [r3, #0x500]
281*4882a593Smuzhiyun	ldr	r6, =0x7fff0001
282*4882a593Smuzhiyun	str	r6, [r3, #0x504]
283*4882a593Smuzhiyun
284*4882a593Smuzhiyun#endif /* CONFIG_SPL_BUILD */
285*4882a593Smuzhiyun
286*4882a593Smuzhiyun	/* Return to U-Boot via saved link register */
287*4882a593Smuzhiyun	mov	pc, lr
288