blob: af9dbc146a3c7beef6f4adec33828d42cb270e3a [file] [log] [blame]
Graeme Russd47ab0e2011-12-23 16:51:29 +11001/*
2 * (C) Copyright 2011
3 * Graeme Russ, <graeme.russ@gmail.com>
4 *
5 * See file CREDITS for list of people who contributed to this
6 * project.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License as
10 * published by the Free Software Foundation; either version 2 of
11 * the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
21 * MA 02111-1307 USA
22 */
23#include <common.h>
24#include <command.h>
Simon Glassf697d522013-02-28 19:26:15 +000025#include <fdtdec.h>
Graeme Russd47ab0e2011-12-23 16:51:29 +110026#include <stdio_dev.h>
27#include <version.h>
28#include <malloc.h>
29#include <net.h>
30#include <ide.h>
31#include <serial.h>
Gabe Black83133152012-11-03 11:41:23 +000032#include <spi.h>
Graeme Russd47ab0e2011-12-23 16:51:29 +110033#include <status_led.h>
Graeme Russa1d57b72011-12-23 21:14:22 +110034#include <asm/processor.h>
Simon Glass86cfb6b2013-03-05 14:39:54 +000035#include <asm/sections.h>
Graeme Russd47ab0e2011-12-23 16:51:29 +110036#include <asm/u-boot-x86.h>
Gabe Blackd65297b2012-11-03 11:41:26 +000037#include <linux/compiler.h>
Graeme Russd47ab0e2011-12-23 16:51:29 +110038
39#include <asm/init_helpers.h>
40
41DECLARE_GLOBAL_DATA_PTR;
42
43/************************************************************************
44 * Init Utilities *
45 ************************************************************************
46 * Some of this code should be moved into the core functions,
47 * or dropped completely,
48 * but let's get it working (again) first...
49 */
50
51int display_banner(void)
52{
53 printf("\n\n%s\n\n", version_string);
54
55 return 0;
56}
57
58int display_dram_config(void)
59{
60 int i;
61
62 puts("DRAM Configuration:\n");
63
64 for (i = 0; i < CONFIG_NR_DRAM_BANKS; i++) {
65 printf("Bank #%d: %08lx ", i, gd->bd->bi_dram[i].start);
66 print_size(gd->bd->bi_dram[i].size, "\n");
67 }
68
69 return 0;
70}
71
72int init_baudrate_f(void)
73{
74 gd->baudrate = getenv_ulong("baudrate", 10, CONFIG_BAUDRATE);
75 return 0;
76}
77
Simon Glass5e989472013-02-28 19:26:10 +000078/* Get the top of usable RAM */
79__weak ulong board_get_usable_ram_top(ulong total_size)
Graeme Russa1d57b72011-12-23 21:14:22 +110080{
Simon Glass5e989472013-02-28 19:26:10 +000081 return gd->ram_size;
82}
83
84int calculate_relocation_address(void)
85{
86 const ulong uboot_size = (uintptr_t)&__bss_end -
87 (uintptr_t)&__text_start;
88 ulong total_size;
Graeme Russa1d57b72011-12-23 21:14:22 +110089 ulong dest_addr;
Simon Glassf697d522013-02-28 19:26:15 +000090 ulong fdt_size = 0;
Graeme Russa1d57b72011-12-23 21:14:22 +110091
Simon Glassf697d522013-02-28 19:26:15 +000092#if defined(CONFIG_OF_SEPARATE) && defined(CONFIG_OF_CONTROL)
93 if (gd->fdt_blob)
94 fdt_size = ALIGN(fdt_totalsize(gd->fdt_blob) + 0x1000, 32);
95#endif
Simon Glass5e989472013-02-28 19:26:10 +000096 total_size = ALIGN(uboot_size, 1 << 12) + CONFIG_SYS_MALLOC_LEN +
Simon Glassf697d522013-02-28 19:26:15 +000097 CONFIG_SYS_STACK_SIZE + fdt_size;
Simon Glass5e989472013-02-28 19:26:10 +000098
Simon Glassf697d522013-02-28 19:26:15 +000099 dest_addr = board_get_usable_ram_top(total_size);
Graeme Russa1d57b72011-12-23 21:14:22 +1100100 /*
101 * NOTE: All destination address are rounded down to 16-byte
102 * boundary to satisfy various worst-case alignment
103 * requirements
104 */
Simon Glassf697d522013-02-28 19:26:15 +0000105 dest_addr &= ~15;
Graeme Russa1d57b72011-12-23 21:14:22 +1100106
Simon Glassf697d522013-02-28 19:26:15 +0000107#if defined(CONFIG_OF_SEPARATE) && defined(CONFIG_OF_CONTROL)
108 /*
109 * If the device tree is sitting immediate above our image then we
110 * must relocate it. If it is embedded in the data section, then it
111 * will be relocated with other data.
112 */
113 if (gd->fdt_blob) {
114 dest_addr -= fdt_size;
Simon Glass1938f4a2013-03-11 06:49:53 +0000115 gd->new_fdt = (void *)dest_addr;
Simon Glassf697d522013-02-28 19:26:15 +0000116 dest_addr &= ~15;
117 }
118#endif
Simon Glass5e989472013-02-28 19:26:10 +0000119 /* U-Boot is below the FDT */
120 dest_addr -= uboot_size;
121 dest_addr &= ~((1 << 12) - 1);
Graeme Russa1d57b72011-12-23 21:14:22 +1100122 gd->relocaddr = dest_addr;
Simon Glass5e989472013-02-28 19:26:10 +0000123 gd->reloc_off = dest_addr - (uintptr_t)&__text_start;
Graeme Russa1d57b72011-12-23 21:14:22 +1100124
Gabe Black32f98732012-11-03 11:41:24 +0000125 /* Stack is at the bottom, so it can grow down */
126 gd->start_addr_sp = dest_addr - CONFIG_SYS_MALLOC_LEN;
127
Graeme Russa1d57b72011-12-23 21:14:22 +1100128 return 0;
129}
130
Graeme Russa1d57b72011-12-23 21:14:22 +1100131int init_cache_f_r(void)
132{
133 /* Initialise the CPU cache(s) */
134 return init_cache();
135}
136
137int set_reloc_flag_r(void)
138{
139 gd->flags = GD_FLG_RELOC;
140
141 return 0;
142}
143
Graeme Russd47ab0e2011-12-23 16:51:29 +1100144int mem_malloc_init_r(void)
145{
146 mem_malloc_init(((gd->relocaddr - CONFIG_SYS_MALLOC_LEN)+3)&~3,
147 CONFIG_SYS_MALLOC_LEN);
148
149 return 0;
150}
151
152bd_t bd_data;
153
154int init_bd_struct_r(void)
155{
156 gd->bd = &bd_data;
157 memset(gd->bd, 0, sizeof(bd_t));
158
159 return 0;
160}
161
162#ifndef CONFIG_SYS_NO_FLASH
163int flash_init_r(void)
164{
165 ulong size;
166
167 puts("Flash: ");
168
169 /* configure available FLASH banks */
170 size = flash_init();
171
172 print_size(size, "\n");
173
174 return 0;
175}
176#endif
177
Graeme Russd47ab0e2011-12-23 16:51:29 +1100178#ifdef CONFIG_STATUS_LED
179int status_led_set_r(void)
180{
181 status_led_set(STATUS_LED_BOOT, STATUS_LED_BLINKING);
182
183 return 0;
184}
185#endif
186
Graeme Russd47ab0e2011-12-23 16:51:29 +1100187int set_load_addr_r(void)
188{
189 /* Initialize from environment */
190 load_addr = getenv_ulong("loadaddr", 16, load_addr);
191
192 return 0;
193}
Gabe Black83133152012-11-03 11:41:23 +0000194
195int init_func_spi(void)
196{
197 puts("SPI: ");
198 spi_init();
199 puts("ready\n");
200 return 0;
201}
Gabe Blackb208c7f2012-11-03 11:41:30 +0000202
203#ifdef CONFIG_OF_CONTROL
204int find_fdt(void)
205{
206#ifdef CONFIG_OF_EMBED
207 /* Get a pointer to the FDT */
208 gd->fdt_blob = _binary_dt_dtb_start;
209#elif defined CONFIG_OF_SEPARATE
210 /* FDT is at end of image */
Simon Glass4b491b82013-02-28 19:26:13 +0000211 gd->fdt_blob = (ulong *)&_end;
Gabe Blackb208c7f2012-11-03 11:41:30 +0000212#endif
213 /* Allow the early environment to override the fdt address */
214 gd->fdt_blob = (void *)getenv_ulong("fdtcontroladdr", 16,
215 (uintptr_t)gd->fdt_blob);
216
217 return 0;
218}
219
220int prepare_fdt(void)
221{
222 /* For now, put this check after the console is ready */
223 if (fdtdec_prepare_fdt()) {
224 panic("** CONFIG_OF_CONTROL defined but no FDT - please see "
225 "doc/README.fdt-control");
226 }
227
228 return 0;
229}
230#endif