blob: 6dfab4ec9e0c54c354cbc6584dcedc24c8e7e4f2 [file] [log] [blame]
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +09001/*
Yusuke Goda1a2334a2008-03-05 14:30:02 +09002 * Copyright (C) 2007,2008
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +09003 * Nobuhiro Iwamatsu <iwamatsu@nigauri.org>
4 *
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License as
7 * published by the Free Software Foundation; either version 2 of
8 * the License, or (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
18 * MA 02111-1307 USA
19 */
20
21#include <common.h>
22#include <command.h>
23#include <malloc.h>
24#include <devices.h>
25#include <version.h>
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090026#include <watchdog.h>
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +090027#include <net.h>
28#include <environment.h>
29
30extern void malloc_bin_reloc (void);
31extern int cpu_init(void);
32extern int board_init(void);
33extern int dram_init(void);
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +090034extern int timer_init(void);
35
36const char version_string[] = U_BOOT_VERSION" (" __DATE__ " - " __TIME__ ")";
37
38unsigned long monitor_flash_len = CFG_MONITOR_LEN;
39
40static unsigned long mem_malloc_start;
41static unsigned long mem_malloc_end;
42static unsigned long mem_malloc_brk;
43
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090044static void mem_malloc_init(void)
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +090045{
46
47 mem_malloc_start = (TEXT_BASE - CFG_GBL_DATA_SIZE - CFG_MALLOC_LEN);
48 mem_malloc_end = (mem_malloc_start + CFG_MALLOC_LEN - 16);
49 mem_malloc_brk = mem_malloc_start;
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090050 memset((void *) mem_malloc_start, 0,
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +090051 (mem_malloc_end - mem_malloc_start));
52}
53
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090054void *sbrk(ptrdiff_t increment)
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +090055{
56 unsigned long old = mem_malloc_brk;
57 unsigned long new = old + increment;
58
59 if ((new < mem_malloc_start) ||
60 (new > mem_malloc_end)) {
61 return NULL;
62 }
63
64 mem_malloc_brk = new;
65 return (void *) old;
66}
67
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +090068static int sh_flash_init(void)
69{
70 DECLARE_GLOBAL_DATA_PTR;
71
72 gd->bd->bi_flashsize = flash_init();
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090073 printf("FLASH: %ldMB\n", gd->bd->bi_flashsize / (1024*1024));
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +090074
75 return 0;
76}
77
78#if defined(CONFIG_CMD_NAND)
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090079# include <nand.h>
80# define INIT_FUNC_NAND_INIT nand_init,
81#else
82# define INIT_FUNC_NAND_INIT
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +090083#endif /* CONFIG_CMD_NAND */
84
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090085#if defined(CONFIG_WATCHDOG)
86extern int watchdog_init(void);
87extern int watchdog_disable(void);
88# define INIT_FUNC_WATCHDOG_INIT watchdog_init,
89# define WATCHDOG_DISABLE watchdog_disable
90#else
91# define INIT_FUNC_WATCHDOG_INIT
92# define WATCHDOG_DISABLE
93#endif /* CONFIG_WATCHDOG */
94
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +090095#if defined(CONFIG_CMD_IDE)
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +090096# include <ide.h>
97# define INIT_FUNC_IDE_INIT ide_init,
98#else
99# define INIT_FUNC_IDE_INIT
100#endif /* CONFIG_CMD_IDE */
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900101
Yusuke Goda1a2334a2008-03-05 14:30:02 +0900102#if defined(CONFIG_PCI)
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900103#include <pci.h>
Yusuke Goda1a2334a2008-03-05 14:30:02 +0900104static int sh_pci_init(void)
105{
106 pci_init();
107 return 0;
108}
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900109# define INIT_FUNC_PCI_INIT sh_pci_init,
110#else
111# define INIT_FUNC_PCI_INIT
Yusuke Goda1a2334a2008-03-05 14:30:02 +0900112#endif /* CONFIG_PCI */
113
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900114static int sh_mem_env_init(void)
115{
116 mem_malloc_init();
117 malloc_bin_reloc();
118 env_relocate();
119 jumptable_init();
120 return 0;
121}
122
Nobuhiro Iwamatsu9e23fe02008-07-08 12:03:24 +0900123#if defined(CONFIG_CMD_NET)
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900124static int sh_net_init(void)
125{
126 DECLARE_GLOBAL_DATA_PTR;
127 char *s, *e;
128 int i;
129
130 gd->bd->bi_ip_addr = getenv_IPaddr("ipaddr");
131 s = getenv("ethaddr");
132 for (i = 0; i < 6; ++i) {
133 gd->bd->bi_enetaddr[i] = s ? simple_strtoul(s, &e, 16) : 0;
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900134 if (s)
135 s = (*e) ? e + 1 : e;
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900136 }
137
138 return 0;
139}
Nobuhiro Iwamatsu9e23fe02008-07-08 12:03:24 +0900140#endif
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900141
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900142typedef int (init_fnc_t) (void);
143
144init_fnc_t *init_sequence[] =
145{
146 cpu_init, /* basic cpu dependent setup */
147 board_init, /* basic board dependent setup */
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900148 interrupt_init, /* set up exceptions */
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900149 env_init, /* event init */
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900150 serial_init, /* SCIF init */
151 INIT_FUNC_WATCHDOG_INIT /* watchdog init */
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900152 console_init_f,
153 display_options,
154 checkcpu,
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900155 checkboard, /* Check support board */
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900156 dram_init, /* SDRAM init */
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900157 timer_init, /* SuperH Timer (TCNT0 only) init */
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900158 sh_flash_init, /* Flash memory(NOR) init*/
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900159 sh_mem_env_init,
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900160 INIT_FUNC_NAND_INIT/* Flash memory (NAND) init */
161 INIT_FUNC_PCI_INIT /* PCI init */
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900162 devices_init,
163 console_init_r,
164 interrupt_init,
165#ifdef BOARD_LATE_INIT
166 board_late_init,
167#endif
168#if defined(CONFIG_CMD_NET)
169 sh_net_init, /* SH specific eth init */
170#endif
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900171 NULL /* Terminate this list */
172};
173
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900174void sh_generic_init(void)
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900175{
176 DECLARE_GLOBAL_DATA_PTR;
177
178 bd_t *bd;
179 init_fnc_t **init_fnc_ptr;
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900180
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900181 memset(gd, 0, CFG_GBL_DATA_SIZE);
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900182
183 gd->flags |= GD_FLG_RELOC; /* tell others: relocation done */
184
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900185 gd->bd = (bd_t *)(gd + 1); /* At end of global data */
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900186 gd->baudrate = CONFIG_BAUDRATE;
187
188 gd->cpu_clk = CONFIG_SYS_CLK_FREQ;
189
190 bd = gd->bd;
191 bd->bi_memstart = CFG_SDRAM_BASE;
192 bd->bi_memsize = CFG_SDRAM_SIZE;
193 bd->bi_flashstart = CFG_FLASH_BASE;
194#if defined(CFG_SRAM_BASE) && defined(CFG_SRAM_SIZE)
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900195 bd->bi_sramstart = CFG_SRAM_BASE;
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900196 bd->bi_sramsize = CFG_SRAM_SIZE;
197#endif
198 bd->bi_baudrate = CONFIG_BAUDRATE;
199
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900200 for (init_fnc_ptr = init_sequence; *init_fnc_ptr; ++init_fnc_ptr) {
201 WATCHDOG_RESET();
202 if ((*init_fnc_ptr) () != 0)
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900203 hang();
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900204 }
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900205
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900206#ifdef CONFIG_WATCHDOG
207 /* disable watchdog if environment is set */
208 {
209 char *s = getenv("watchdog");
210 if (s != NULL)
211 if (strncmp(s, "off", 3) == 0)
212 WATCHDOG_DISABLE();
213 }
214#endif /* CONFIG_WATCHDOG*/
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900215
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900216
217#if defined(CONFIG_CMD_NET)
218 {
219 char *s;
220 puts("Net: ");
221 eth_initialize(gd->bd);
222
223 s = getenv("bootfile");
224 if (s != NULL)
225 copy_filename(BootFile, s, sizeof(BootFile));
Nobuhiro Iwamatsub02bad12007-09-23 02:12:30 +0900226 }
227#endif /* CONFIG_CMD_NET */
228
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900229 while (1) {
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900230 WATCHDOG_RESET();
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900231 main_loop();
232 }
233}
234
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900235/***********************************************************************/
236
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900237void hang(void)
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900238{
Nobuhiro Iwamatsu4a065ab2008-09-18 19:04:26 +0900239 puts("Board ERROR\n");
240 for (;;)
241 ;
Nobuhiro Iwamatsu0b135cf2007-05-13 20:58:00 +0900242}