Merge branch 'cleanups' into next
[platform/kernel/u-boot.git] / lib_microblaze / board.c
1 /*
2  * (C) Copyright 2007 Michal Simek
3  * (C) Copyright 2004 Atmark Techno, Inc.
4  *
5  * Michal  SIMEK <monstr@monstr.eu>
6  * Yasushi SHOJI <yashi@atmark-techno.com>
7  *
8  * See file CREDITS for list of people who contributed to this
9  * project.
10  *
11  * This program is free software; you can redistribute it and/or
12  * modify it under the terms of the GNU General Public License as
13  * published by the Free Software Foundation; either version 2 of
14  * the License, or (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
24  * MA 02111-1307 USA
25  */
26
27 #include <common.h>
28 #include <command.h>
29 #include <malloc.h>
30 #include <timestamp.h>
31 #include <version.h>
32 #include <watchdog.h>
33
34 DECLARE_GLOBAL_DATA_PTR;
35
36 const char version_string[] = U_BOOT_VERSION " ("U_BOOT_DATE" - "U_BOOT_TIME")";
37
38 #ifdef CONFIG_SYS_GPIO_0
39 extern int gpio_init (void);
40 #endif
41 #ifdef CONFIG_SYS_INTC_0
42 extern int interrupts_init (void);
43 #endif
44 #if defined(CONFIG_CMD_NET)
45 extern int eth_init (bd_t * bis);
46 extern int getenv_IPaddr (char *);
47 #endif
48
49 /*
50  * Begin and End of memory area for malloc(), and current "brk"
51  */
52 static ulong mem_malloc_start;
53 static ulong mem_malloc_end;
54 static ulong mem_malloc_brk;
55
56 /*
57  * The Malloc area is immediately below the monitor copy in DRAM
58  * aka CONFIG_SYS_MONITOR_BASE - Note there is no need for reloc_off
59  * as our monitory code is run from SDRAM
60  */
61 static void mem_malloc_init (void)
62 {
63         mem_malloc_end = (CONFIG_SYS_MALLOC_BASE + CONFIG_SYS_MALLOC_LEN);
64         mem_malloc_start = CONFIG_SYS_MALLOC_BASE;
65         mem_malloc_brk = mem_malloc_start;
66         memset ((void *)mem_malloc_start, 0, mem_malloc_end - mem_malloc_start);
67 }
68
69 void *sbrk (ptrdiff_t increment)
70 {
71         ulong old = mem_malloc_brk;
72         ulong new = old + increment;
73
74         if ((new < mem_malloc_start) || (new > mem_malloc_end)) {
75                 return (NULL);
76         }
77         mem_malloc_brk = new;
78         return ((void *)old);
79 }
80
81 /*
82  * All attempts to come up with a "common" initialization sequence
83  * that works for all boards and architectures failed: some of the
84  * requirements are just _too_ different. To get rid of the resulting
85  * mess of board dependend #ifdef'ed code we now make the whole
86  * initialization sequence configurable to the user.
87  *
88  * The requirements for any new initalization function is simple: it
89  * receives a pointer to the "global data" structure as it's only
90  * argument, and returns an integer return code, where 0 means
91  * "continue" and != 0 means "fatal error, hang the system".
92  */
93 typedef int (init_fnc_t) (void);
94
95 init_fnc_t *init_sequence[] = {
96         env_init,
97         serial_init,
98 #ifdef CONFIG_SYS_GPIO_0
99         gpio_init,
100 #endif
101 #ifdef CONFIG_SYS_INTC_0
102         interrupts_init,
103 #endif
104         NULL,
105 };
106
107 void board_init (void)
108 {
109         bd_t *bd;
110         init_fnc_t **init_fnc_ptr;
111         gd = (gd_t *) CONFIG_SYS_GBL_DATA_OFFSET;
112 #if defined(CONFIG_CMD_FLASH)
113         ulong flash_size = 0;
114 #endif
115 #if defined(CONFIG_CMD_NET)
116         char *s, *e;
117         int i;
118 #endif
119         asm ("nop");    /* FIXME gd is not initialize - wait */
120         memset ((void *)gd, 0, CONFIG_SYS_GBL_DATA_SIZE);
121         gd->bd = (bd_t *) (gd + 1);     /* At end of global data */
122         gd->baudrate = CONFIG_BAUDRATE;
123         bd = gd->bd;
124         bd->bi_baudrate = CONFIG_BAUDRATE;
125         bd->bi_memstart = CONFIG_SYS_SDRAM_BASE;
126         bd->bi_memsize = CONFIG_SYS_SDRAM_SIZE;
127         gd->flags |= GD_FLG_RELOC;      /* tell others: relocation done */
128
129         /* Initialise malloc() area */
130         mem_malloc_init ();
131
132         for (init_fnc_ptr = init_sequence; *init_fnc_ptr; ++init_fnc_ptr) {
133                 WATCHDOG_RESET ();
134                 if ((*init_fnc_ptr) () != 0) {
135                         hang ();
136                 }
137         }
138
139         puts ("SDRAM :\n");
140         printf ("\t\tIcache:%s\n", icache_status() ? "OK" : "FAIL");
141         printf ("\t\tDcache:%s\n", dcache_status() ? "OK" : "FAIL");
142         printf ("\tU-Boot Start:0x%08x\n", TEXT_BASE);
143
144 #if defined(CONFIG_CMD_FLASH)
145         puts ("FLASH: ");
146         bd->bi_flashstart = CONFIG_SYS_FLASH_BASE;
147         if (0 < (flash_size = flash_init ())) {
148                 bd->bi_flashsize = flash_size;
149                 bd->bi_flashoffset = CONFIG_SYS_FLASH_BASE + flash_size;
150 # ifdef CONFIG_SYS_FLASH_CHECKSUM
151                 print_size (flash_size, "");
152                 /*
153                  * Compute and print flash CRC if flashchecksum is set to 'y'
154                  *
155                  * NOTE: Maybe we should add some WATCHDOG_RESET()? XXX
156                  */
157                 s = getenv ("flashchecksum");
158                 if (s && (*s == 'y')) {
159                         printf ("  CRC: %08X",
160                                 crc32 (0, (const unsigned char *) CONFIG_SYS_FLASH_BASE, flash_size)
161                         );
162                 }
163                 putc ('\n');
164 # else  /* !CONFIG_SYS_FLASH_CHECKSUM */
165                 print_size (flash_size, "\n");
166 # endif /* CONFIG_SYS_FLASH_CHECKSUM */
167         } else {
168                 puts ("Flash init FAILED");
169                 bd->bi_flashstart = 0;
170                 bd->bi_flashsize = 0;
171                 bd->bi_flashoffset = 0;
172         }
173 #endif
174
175 #if defined(CONFIG_CMD_NET)
176         /* board MAC address */
177         s = getenv ("ethaddr");
178         printf ("MAC:%s\n",s);
179         for (i = 0; i < 6; ++i) {
180                 bd->bi_enetaddr[i] = s ? simple_strtoul (s, &e, 16) : 0;
181                 if (s)
182                         s = (*e) ? e + 1 : e;
183         }
184         /* IP Address */
185         bd->bi_ip_addr = getenv_IPaddr ("ipaddr");
186         eth_init (bd);
187 #endif
188
189         /* relocate environment function pointers etc. */
190         env_relocate ();
191
192         /* main_loop */
193         for (;;) {
194                 WATCHDOG_RESET ();
195                 main_loop ();
196         }
197 }
198
199 void hang (void)
200 {
201         puts ("### ERROR ### Please RESET the board ###\n");
202         for (;;) ;
203 }