xref: /rk3399_rockchip-uboot/arch/sh/lib/bootm.c (revision de03f8bc4a3f7bbb09bccbf2e3a5f14b45d5ad64)
1ea0364f1SPeter Tyser /*
2ea0364f1SPeter Tyser  * (C) Copyright 2003
3ea0364f1SPeter Tyser  * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
4ea0364f1SPeter Tyser  *
5ea0364f1SPeter Tyser  * (c) Copyright 2008 Nobuhiro Iwamatsu <iwamatsu.nobuhiro@renesas.com>
6ea0364f1SPeter Tyser  * (c) Copyright 2008 Renesas Solutions Corp.
7ea0364f1SPeter Tyser  *
8ea0364f1SPeter Tyser  * See file CREDITS for list of people who contributed to this
9ea0364f1SPeter Tyser  * project.
10ea0364f1SPeter Tyser  *
11ea0364f1SPeter Tyser  * This program is free software; you can redistribute it and/or modify
12ea0364f1SPeter Tyser  * it under the terms of the GNU General Public License as published by
13ea0364f1SPeter Tyser  * the Free Software Foundation; either version 2 of the License, or
14ea0364f1SPeter Tyser  * (at your option) any later version.
15ea0364f1SPeter Tyser  *
16ea0364f1SPeter Tyser  * This program is distributed in the hope that it will be useful,
17ea0364f1SPeter Tyser  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18ea0364f1SPeter Tyser  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19ea0364f1SPeter Tyser  * GNU General Public License for more details.
20ea0364f1SPeter Tyser  *
21ea0364f1SPeter Tyser  * You should have received a copy of the GNU General Public License
22ea0364f1SPeter Tyser  * along with this program; if not, write to the Free Software
23ea0364f1SPeter Tyser  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
24ea0364f1SPeter Tyser  *
25ea0364f1SPeter Tyser  */
26ea0364f1SPeter Tyser 
27ea0364f1SPeter Tyser #include <common.h>
28ea0364f1SPeter Tyser #include <command.h>
29ea0364f1SPeter Tyser #include <asm/byteorder.h>
30ea0364f1SPeter Tyser 
31ea0364f1SPeter Tyser #ifdef CONFIG_SYS_DEBUG
32ea0364f1SPeter Tyser static void hexdump(unsigned char *buf, int len)
33ea0364f1SPeter Tyser {
34ea0364f1SPeter Tyser 	int i;
35ea0364f1SPeter Tyser 
36ea0364f1SPeter Tyser 	for (i = 0; i < len; i++) {
37ea0364f1SPeter Tyser 		if ((i % 16) == 0)
38ea0364f1SPeter Tyser 			printf("%s%08x: ", i ? "\n" : "",
39ea0364f1SPeter Tyser 							(unsigned int)&buf[i]);
40ea0364f1SPeter Tyser 		printf("%02x ", buf[i]);
41ea0364f1SPeter Tyser 	}
42ea0364f1SPeter Tyser 	printf("\n");
43ea0364f1SPeter Tyser }
44ea0364f1SPeter Tyser #endif
45ea0364f1SPeter Tyser 
46cf2c87d3SNobuhiro Iwamatsu #define MOUNT_ROOT_RDONLY	0x000
47cf2c87d3SNobuhiro Iwamatsu #define RAMDISK_FLAGS		0x004
48cf2c87d3SNobuhiro Iwamatsu #define ORIG_ROOT_DEV		0x008
49cf2c87d3SNobuhiro Iwamatsu #define LOADER_TYPE			0x00c
50cf2c87d3SNobuhiro Iwamatsu #define INITRD_START		0x010
51cf2c87d3SNobuhiro Iwamatsu #define INITRD_SIZE			0x014
52cf2c87d3SNobuhiro Iwamatsu #define COMMAND_LINE		0x100
53cf2c87d3SNobuhiro Iwamatsu 
54cf2c87d3SNobuhiro Iwamatsu #define RD_PROMPT	(1<<15)
55cf2c87d3SNobuhiro Iwamatsu #define RD_DOLOAD	(1<<14)
56cf2c87d3SNobuhiro Iwamatsu #define CMD_ARG_RD_PROMPT	"prompt_ramdisk="
57cf2c87d3SNobuhiro Iwamatsu #define CMD_ARG_RD_DOLOAD	"load_ramdisk="
58cf2c87d3SNobuhiro Iwamatsu 
59cf2c87d3SNobuhiro Iwamatsu #ifdef CONFIG_SH_SDRAM_OFFSET
60cf2c87d3SNobuhiro Iwamatsu #define GET_INITRD_START(initrd, linux) (initrd - linux + CONFIG_SH_SDRAM_OFFSET)
61cf2c87d3SNobuhiro Iwamatsu #else
62cf2c87d3SNobuhiro Iwamatsu #define GET_INITRD_START(initrd, linux) (initrd - linux)
63cf2c87d3SNobuhiro Iwamatsu #endif
64cf2c87d3SNobuhiro Iwamatsu 
65cf2c87d3SNobuhiro Iwamatsu static void set_sh_linux_param(unsigned long param_addr, unsigned long data)
66cf2c87d3SNobuhiro Iwamatsu {
67cf2c87d3SNobuhiro Iwamatsu 	*(unsigned long *)(param_addr) = data;
68cf2c87d3SNobuhiro Iwamatsu }
69cf2c87d3SNobuhiro Iwamatsu 
70cf2c87d3SNobuhiro Iwamatsu static unsigned long sh_check_cmd_arg(char *cmdline, char *key, int base)
71cf2c87d3SNobuhiro Iwamatsu {
72cf2c87d3SNobuhiro Iwamatsu 	unsigned long val = 0;
73cf2c87d3SNobuhiro Iwamatsu 	char *p = strstr(cmdline, key);
74cf2c87d3SNobuhiro Iwamatsu 	if (p) {
75cf2c87d3SNobuhiro Iwamatsu 		p += strlen(key);
76cf2c87d3SNobuhiro Iwamatsu 		val = simple_strtol(p, NULL, base);
77cf2c87d3SNobuhiro Iwamatsu 	}
78cf2c87d3SNobuhiro Iwamatsu 	return val;
79cf2c87d3SNobuhiro Iwamatsu }
80cf2c87d3SNobuhiro Iwamatsu 
8154841ab5SWolfgang Denk int do_bootm_linux(int flag, int argc, char * const argv[], bootm_headers_t *images)
82ea0364f1SPeter Tyser {
83ea0364f1SPeter Tyser 	/* Linux kernel load address */
84ea0364f1SPeter Tyser 	void (*kernel) (void) = (void (*)(void))images->ep;
85ea0364f1SPeter Tyser 	/* empty_zero_page */
86ea0364f1SPeter Tyser 	unsigned char *param
87ea0364f1SPeter Tyser 		= (unsigned char *)image_get_load(images->legacy_hdr_os);
88ea0364f1SPeter Tyser 	/* Linux kernel command line */
89cf2c87d3SNobuhiro Iwamatsu 	char *cmdline = (char *)param + COMMAND_LINE;
90ea0364f1SPeter Tyser 	/* PAGE_SIZE */
91ea0364f1SPeter Tyser 	unsigned long size = images->ep - (unsigned long)param;
92ea0364f1SPeter Tyser 	char *bootargs = getenv("bootargs");
93ea0364f1SPeter Tyser 
94ea0364f1SPeter Tyser 	if ((flag != 0) && (flag != BOOTM_STATE_OS_GO))
95ea0364f1SPeter Tyser 		return 1;
96ea0364f1SPeter Tyser 
97ea0364f1SPeter Tyser 	/* Setup parameters */
98ea0364f1SPeter Tyser 	memset(param, 0, size);	/* Clear zero page */
99cf2c87d3SNobuhiro Iwamatsu 
100cf2c87d3SNobuhiro Iwamatsu 	/* Set commandline */
101ea0364f1SPeter Tyser 	strcpy(cmdline, bootargs);
102ea0364f1SPeter Tyser 
103cf2c87d3SNobuhiro Iwamatsu 	sh_check_cmd_arg(bootargs, CMD_ARG_RD_DOLOAD, 10);
104cf2c87d3SNobuhiro Iwamatsu 	/* Initrd */
105cf2c87d3SNobuhiro Iwamatsu 	if (images->rd_start || images->rd_end) {
106*de03f8bcSNobuhiro Iwamatsu 		unsigned long ramdisk_flags = 0;
107cf2c87d3SNobuhiro Iwamatsu 		int val = sh_check_cmd_arg(bootargs, CMD_ARG_RD_PROMPT, 10);
108cf2c87d3SNobuhiro Iwamatsu 		if (val == 1)
109cf2c87d3SNobuhiro Iwamatsu 				ramdisk_flags |= RD_PROMPT;
110cf2c87d3SNobuhiro Iwamatsu 		else
111cf2c87d3SNobuhiro Iwamatsu 				ramdisk_flags &= ~RD_PROMPT;
112cf2c87d3SNobuhiro Iwamatsu 
113cf2c87d3SNobuhiro Iwamatsu 		val = sh_check_cmd_arg(bootargs, CMD_ARG_RD_DOLOAD, 10);
114cf2c87d3SNobuhiro Iwamatsu 		if (val == 1)
115cf2c87d3SNobuhiro Iwamatsu 				ramdisk_flags |= RD_DOLOAD;
116cf2c87d3SNobuhiro Iwamatsu 		else
117cf2c87d3SNobuhiro Iwamatsu 				ramdisk_flags &= ~RD_DOLOAD;
118cf2c87d3SNobuhiro Iwamatsu 
119cf2c87d3SNobuhiro Iwamatsu 		set_sh_linux_param((unsigned long)param + MOUNT_ROOT_RDONLY, 0x0001);
120cf2c87d3SNobuhiro Iwamatsu 		set_sh_linux_param((unsigned long)param + RAMDISK_FLAGS, ramdisk_flags);
121cf2c87d3SNobuhiro Iwamatsu 		set_sh_linux_param((unsigned long)param + ORIG_ROOT_DEV, 0x0200);
122cf2c87d3SNobuhiro Iwamatsu 		set_sh_linux_param((unsigned long)param + LOADER_TYPE, 0x0001);
123cf2c87d3SNobuhiro Iwamatsu 		set_sh_linux_param((unsigned long)param + INITRD_START,
124cf2c87d3SNobuhiro Iwamatsu 			GET_INITRD_START(images->rd_start, CONFIG_SYS_SDRAM_BASE));
125cf2c87d3SNobuhiro Iwamatsu 		set_sh_linux_param((unsigned long)param + INITRD_SIZE,
126cf2c87d3SNobuhiro Iwamatsu 			images->rd_end - images->rd_start);
127cf2c87d3SNobuhiro Iwamatsu 	}
128cf2c87d3SNobuhiro Iwamatsu 
129cf2c87d3SNobuhiro Iwamatsu 	/* Boot kernel */
130ea0364f1SPeter Tyser 	kernel();
131ea0364f1SPeter Tyser 	/* does not return */
132ea0364f1SPeter Tyser 
133ea0364f1SPeter Tyser 	return 1;
134ea0364f1SPeter Tyser }
135