blob: 83caf6bdbd780113862d8d80374412cc9ff3cef4 [file] [log] [blame]
wdenk591dda52002-11-18 00:14:45 +00001/*
2 * (C) Copyright 2002
3 * Sysgo Real-Time Solutions, GmbH <www.elinos.com>
4 * Marius Groeger <mgroeger@sysgo.de>
5 *
6 * Copyright (C) 2001 Erik Mouw (J.A.K.Mouw@its.tudelft.nl)
7 *
Graeme Russ45fc1d82011-04-13 19:43:26 +10008 * 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.
wdenk591dda52002-11-18 00:14:45 +000015 *
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
Graeme Russ45fc1d82011-04-13 19:43:26 +100023 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
24 * MA 02111-1307 USA
wdenk591dda52002-11-18 00:14:45 +000025 */
26
27#include <common.h>
28#include <command.h>
wdenk591dda52002-11-18 00:14:45 +000029#include <image.h>
Jean-Christophe PLAGNIOL-VILLARD6bb94492009-04-04 12:49:11 +020030#include <u-boot/zlib.h>
Gabe Black540c2622011-12-05 12:09:26 +000031#include <asm/bootparam.h>
wdenk591dda52002-11-18 00:14:45 +000032#include <asm/byteorder.h>
33#include <asm/zimage.h>
34
Gabe Black540c2622011-12-05 12:09:26 +000035#define COMMAND_LINE_OFFSET 0x9000
36
wdenk57b2d802003-06-27 21:31:46 +000037/*cmd_boot.c*/
Graeme Russ883c6032011-11-08 02:33:15 +000038int do_bootm_linux(int flag, int argc, char * const argv[],
39 bootm_headers_t *images)
wdenk591dda52002-11-18 00:14:45 +000040{
Gabe Black540c2622011-12-05 12:09:26 +000041 struct boot_params *base_ptr = NULL;
42 ulong os_data, os_len;
43 image_header_t *hdr;
44 void *load_address;
Graeme Russ2fe2a972008-09-07 07:08:42 +100045
Marian Balakowiczdf8ff332008-03-12 10:33:00 +010046#if defined(CONFIG_FIT)
47 const void *data;
48 size_t len;
49#endif
wdenk57b2d802003-06-27 21:31:46 +000050
Kumar Gala18178bc2008-10-21 17:25:45 -050051 if ((flag != 0) && (flag != BOOTM_STATE_OS_GO))
52 return 1;
53
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010054 if (images->legacy_hdr_valid) {
55 hdr = images->legacy_hdr_os;
Graeme Russ883c6032011-11-08 02:33:15 +000056 if (image_check_type(hdr, IH_TYPE_MULTI)) {
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010057 /* if multi-part image, we need to get first subimage */
Graeme Russ883c6032011-11-08 02:33:15 +000058 image_multi_getimg(hdr, 0, &os_data, &os_len);
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010059 } else {
60 /* otherwise get image data */
Graeme Russ883c6032011-11-08 02:33:15 +000061 os_data = image_get_data(hdr);
62 os_len = image_get_data_size(hdr);
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010063 }
64#if defined(CONFIG_FIT)
65 } else if (images->fit_uname_os) {
Graeme Russ883c6032011-11-08 02:33:15 +000066 ret = fit_image_get_data(images->fit_hdr_os,
Marian Balakowiczdf8ff332008-03-12 10:33:00 +010067 images->fit_noffset_os, &data, &len);
68 if (ret) {
Graeme Russ883c6032011-11-08 02:33:15 +000069 puts("Can't get image data/size!\n");
Marian Balakowiczdf8ff332008-03-12 10:33:00 +010070 goto error;
71 }
72 os_data = (ulong)data;
73 os_len = (ulong)len;
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010074#endif
Marian Balakowiczb4a12a92008-01-08 18:12:17 +010075 } else {
Graeme Russ883c6032011-11-08 02:33:15 +000076 puts("Could not find kernel image!\n");
Marian Balakowiczdf8ff332008-03-12 10:33:00 +010077 goto error;
Wolfgang Denk44cacdc2006-07-21 11:30:18 +020078 }
79
Graeme Russ19fc0d52011-11-08 02:33:20 +000080#ifdef CONFIG_CMD_ZBOOT
Gabe Black540c2622011-12-05 12:09:26 +000081 base_ptr = load_zimage((void *)os_data, os_len, &load_address);
Graeme Russ19fc0d52011-11-08 02:33:20 +000082#endif
wdenk591dda52002-11-18 00:14:45 +000083
84 if (NULL == base_ptr) {
Graeme Russ883c6032011-11-08 02:33:15 +000085 printf("## Kernel loading failed ...\n");
Marian Balakowiczdf8ff332008-03-12 10:33:00 +010086 goto error;
Gabe Black540c2622011-12-05 12:09:26 +000087 }
wdenk57b2d802003-06-27 21:31:46 +000088
Gabe Black540c2622011-12-05 12:09:26 +000089 if (setup_zimage(base_ptr, (char *)base_ptr + COMMAND_LINE_OFFSET,
90 0, images->rd_start,
91 images->rd_end - images->rd_start)) {
92 printf("## Setting up boot parameters failed ...\n");
93 goto error;
wdenk591dda52002-11-18 00:14:45 +000094 }
wdenk57b2d802003-06-27 21:31:46 +000095
wdenk591dda52002-11-18 00:14:45 +000096#ifdef DEBUG
Graeme Russ883c6032011-11-08 02:33:15 +000097 printf("## Transferring control to Linux (at address %08x) ...\n",
wdenk591dda52002-11-18 00:14:45 +000098 (u32)base_ptr);
99#endif
wdenk57b2d802003-06-27 21:31:46 +0000100
wdenk591dda52002-11-18 00:14:45 +0000101 /* we assume that the kernel is in place */
102 printf("\nStarting kernel ...\n\n");
wdenk591dda52002-11-18 00:14:45 +0000103
Gabe Blackc67a9482011-12-05 12:09:24 +0000104 boot_zimage(base_ptr, load_address);
Marian Balakowiczdf8ff332008-03-12 10:33:00 +0100105 /* does not return */
wdenk591dda52002-11-18 00:14:45 +0000106
Marian Balakowiczdf8ff332008-03-12 10:33:00 +0100107error:
Kumar Gala48626aa2008-08-15 08:24:45 -0500108 return 1;
wdenk57b2d802003-06-27 21:31:46 +0000109}