Blame view

cmd/elf.c 14.3 KB
458ded34b   wdenk   Initial revision
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
  /*
   * Copyright (c) 2001 William L. Pitts
   * All rights reserved.
   *
   * Redistribution and use in source and binary forms are freely
   * permitted provided that the above copyright notice and this
   * paragraph and the following disclaimer are duplicated in all
   * such forms.
   *
   * This software is provided "AS IS" and without any express or
   * implied warranties, including, without limitation, the implied
   * warranties of merchantability and fitness for a particular
   * purpose.
   */
  
  #include <common.h>
  #include <command.h>
458ded34b   wdenk   Initial revision
18
  #include <elf.h>
9925f1dbc   Alex Kiernan   net: Move enetadd...
19
  #include <environment.h>
ebca3df78   Bin Meng   cmd: Clean up cmd...
20
  #include <net.h>
29a4c24de   Niklaus Giger   cmd_elf.c: Cleanu...
21
  #include <vxworks.h>
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
22
  #ifdef CONFIG_X86
447ae4f7a   Bin Meng   bootvx: x86: Make...
23
  #include <vbe.h>
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
24
  #include <asm/e820.h>
9aa1280a5   Bin Meng   cmd: bootvx: Add ...
25
  #include <linux/linkage.h>
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
26
  #endif
458ded34b   wdenk   Initial revision
27

9dffa52da   Bin Meng   cmd: elf: Reorder...
28
  /*
839c4e9c5   Bin Meng   elf: Add a very s...
29
   * A very simple ELF64 loader, assumes the image is valid, returns the
9dffa52da   Bin Meng   cmd: elf: Reorder...
30
   * entry point address.
839c4e9c5   Bin Meng   elf: Add a very s...
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
   *
   * Note if U-Boot is 32-bit, the loader assumes the to segment's
   * physical address and size is within the lower 32-bit address space.
   */
  static unsigned long load_elf64_image_phdr(unsigned long addr)
  {
  	Elf64_Ehdr *ehdr; /* Elf header structure pointer */
  	Elf64_Phdr *phdr; /* Program header structure pointer */
  	int i;
  
  	ehdr = (Elf64_Ehdr *)addr;
  	phdr = (Elf64_Phdr *)(addr + (ulong)ehdr->e_phoff);
  
  	/* Load each program header */
  	for (i = 0; i < ehdr->e_phnum; ++i) {
  		void *dst = (void *)(ulong)phdr->p_paddr;
  		void *src = (void *)addr + phdr->p_offset;
  
  		debug("Loading phdr %i to 0x%p (%lu bytes)
  ",
  		      i, dst, (ulong)phdr->p_filesz);
  		if (phdr->p_filesz)
  			memcpy(dst, src, phdr->p_filesz);
  		if (phdr->p_filesz != phdr->p_memsz)
  			memset(dst + phdr->p_filesz, 0x00,
  			       phdr->p_memsz - phdr->p_filesz);
  		flush_cache((unsigned long)dst, phdr->p_filesz);
  		++phdr;
  	}
2846ea81a   Rob Bracero   elf: Add support ...
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
  	if (ehdr->e_machine == EM_PPC64 && (ehdr->e_flags &
  					    EF_PPC64_ELFV1_ABI)) {
  		/*
  		 * For the 64-bit PowerPC ELF V1 ABI, e_entry is a function
  		 * descriptor pointer with the first double word being the
  		 * address of the entry point of the function.
  		 */
  		uintptr_t addr = ehdr->e_entry;
  
  		return *(Elf64_Addr *)addr;
  	}
  
  	return ehdr->e_entry;
  }
  
  static unsigned long load_elf64_image_shdr(unsigned long addr)
  {
  	Elf64_Ehdr *ehdr; /* Elf header structure pointer */
  	Elf64_Shdr *shdr; /* Section header structure pointer */
  	unsigned char *strtab = 0; /* String table pointer */
  	unsigned char *image; /* Binary image pointer */
  	int i; /* Loop counter */
  
  	ehdr = (Elf64_Ehdr *)addr;
  
  	/* Find the section header string table for output info */
  	shdr = (Elf64_Shdr *)(addr + (ulong)ehdr->e_shoff +
  			     (ehdr->e_shstrndx * sizeof(Elf64_Shdr)));
  
  	if (shdr->sh_type == SHT_STRTAB)
  		strtab = (unsigned char *)(addr + (ulong)shdr->sh_offset);
  
  	/* Load each appropriate section */
  	for (i = 0; i < ehdr->e_shnum; ++i) {
  		shdr = (Elf64_Shdr *)(addr + (ulong)ehdr->e_shoff +
  				     (i * sizeof(Elf64_Shdr)));
  
  		if (!(shdr->sh_flags & SHF_ALLOC) ||
  		    shdr->sh_addr == 0 || shdr->sh_size == 0) {
  			continue;
  		}
  
  		if (strtab) {
  			debug("%sing %s @ 0x%08lx (%ld bytes)
  ",
  			      (shdr->sh_type == SHT_NOBITS) ? "Clear" : "Load",
  			       &strtab[shdr->sh_name],
  			       (unsigned long)shdr->sh_addr,
  			       (long)shdr->sh_size);
  		}
  
  		if (shdr->sh_type == SHT_NOBITS) {
  			memset((void *)(uintptr_t)shdr->sh_addr, 0,
  			       shdr->sh_size);
  		} else {
  			image = (unsigned char *)addr + (ulong)shdr->sh_offset;
  			memcpy((void *)(uintptr_t)shdr->sh_addr,
  			       (const void *)image, shdr->sh_size);
  		}
8744d6c50   Neil Stainton   u-boot: align cac...
119
120
121
122
  		flush_cache(rounddown(shdr->sh_addr, ARCH_DMA_MINALIGN),
  			    roundup((shdr->sh_addr + shdr->sh_size),
  				     ARCH_DMA_MINALIGN) -
  			            rounddown(shdr->sh_addr, ARCH_DMA_MINALIGN));
2846ea81a   Rob Bracero   elf: Add support ...
123
124
125
126
127
128
129
130
131
132
133
134
135
  	}
  
  	if (ehdr->e_machine == EM_PPC64 && (ehdr->e_flags &
  					    EF_PPC64_ELFV1_ABI)) {
  		/*
  		 * For the 64-bit PowerPC ELF V1 ABI, e_entry is a function
  		 * descriptor pointer with the first double word being the
  		 * address of the entry point of the function.
  		 */
  		uintptr_t addr = ehdr->e_entry;
  
  		return *(Elf64_Addr *)addr;
  	}
839c4e9c5   Bin Meng   elf: Add a very s...
136
137
138
139
140
141
142
143
144
  	return ehdr->e_entry;
  }
  
  /*
   * A very simple ELF loader, assumes the image is valid, returns the
   * entry point address.
   *
   * The loader firstly reads the EFI class to see if it's a 64-bit image.
   * If yes, call the ELF64 loader. Otherwise continue with the ELF32 loader.
9dffa52da   Bin Meng   cmd: elf: Reorder...
145
146
147
148
149
150
151
152
   */
  static unsigned long load_elf_image_phdr(unsigned long addr)
  {
  	Elf32_Ehdr *ehdr; /* Elf header structure pointer */
  	Elf32_Phdr *phdr; /* Program header structure pointer */
  	int i;
  
  	ehdr = (Elf32_Ehdr *)addr;
839c4e9c5   Bin Meng   elf: Add a very s...
153
154
  	if (ehdr->e_ident[EI_CLASS] == ELFCLASS64)
  		return load_elf64_image_phdr(addr);
9dffa52da   Bin Meng   cmd: elf: Reorder...
155
156
157
158
159
160
  	phdr = (Elf32_Phdr *)(addr + ehdr->e_phoff);
  
  	/* Load each program header */
  	for (i = 0; i < ehdr->e_phnum; ++i) {
  		void *dst = (void *)(uintptr_t)phdr->p_paddr;
  		void *src = (void *)addr + phdr->p_offset;
839c4e9c5   Bin Meng   elf: Add a very s...
161

9dffa52da   Bin Meng   cmd: elf: Reorder...
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
  		debug("Loading phdr %i to 0x%p (%i bytes)
  ",
  		      i, dst, phdr->p_filesz);
  		if (phdr->p_filesz)
  			memcpy(dst, src, phdr->p_filesz);
  		if (phdr->p_filesz != phdr->p_memsz)
  			memset(dst + phdr->p_filesz, 0x00,
  			       phdr->p_memsz - phdr->p_filesz);
  		flush_cache((unsigned long)dst, phdr->p_filesz);
  		++phdr;
  	}
  
  	return ehdr->e_entry;
  }
  
  static unsigned long load_elf_image_shdr(unsigned long addr)
  {
  	Elf32_Ehdr *ehdr; /* Elf header structure pointer */
  	Elf32_Shdr *shdr; /* Section header structure pointer */
  	unsigned char *strtab = 0; /* String table pointer */
  	unsigned char *image; /* Binary image pointer */
  	int i; /* Loop counter */
  
  	ehdr = (Elf32_Ehdr *)addr;
2846ea81a   Rob Bracero   elf: Add support ...
186
187
  	if (ehdr->e_ident[EI_CLASS] == ELFCLASS64)
  		return load_elf64_image_shdr(addr);
9dffa52da   Bin Meng   cmd: elf: Reorder...
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
  
  	/* Find the section header string table for output info */
  	shdr = (Elf32_Shdr *)(addr + ehdr->e_shoff +
  			     (ehdr->e_shstrndx * sizeof(Elf32_Shdr)));
  
  	if (shdr->sh_type == SHT_STRTAB)
  		strtab = (unsigned char *)(addr + shdr->sh_offset);
  
  	/* Load each appropriate section */
  	for (i = 0; i < ehdr->e_shnum; ++i) {
  		shdr = (Elf32_Shdr *)(addr + ehdr->e_shoff +
  				     (i * sizeof(Elf32_Shdr)));
  
  		if (!(shdr->sh_flags & SHF_ALLOC) ||
  		    shdr->sh_addr == 0 || shdr->sh_size == 0) {
  			continue;
  		}
  
  		if (strtab) {
  			debug("%sing %s @ 0x%08lx (%ld bytes)
  ",
  			      (shdr->sh_type == SHT_NOBITS) ? "Clear" : "Load",
  			       &strtab[shdr->sh_name],
  			       (unsigned long)shdr->sh_addr,
  			       (long)shdr->sh_size);
  		}
  
  		if (shdr->sh_type == SHT_NOBITS) {
  			memset((void *)(uintptr_t)shdr->sh_addr, 0,
  			       shdr->sh_size);
  		} else {
  			image = (unsigned char *)addr + shdr->sh_offset;
  			memcpy((void *)(uintptr_t)shdr->sh_addr,
  			       (const void *)image, shdr->sh_size);
  		}
  		flush_cache(shdr->sh_addr, shdr->sh_size);
  	}
  
  	return ehdr->e_entry;
  }
017e9b792   Mike Frysinger   allow ports to ov...
228
229
  
  /* Allow ports to override the default behavior */
553d8c3a5   Jeroen Hofstee   common: cmd_elf: ...
230
  static unsigned long do_bootelf_exec(ulong (*entry)(int, char * const[]),
ebca3df78   Bin Meng   cmd: Clean up cmd...
231
  				     int argc, char * const argv[])
1f1d88dd4   Mike Frysinger   disable caches be...
232
  {
017e9b792   Mike Frysinger   allow ports to ov...
233
  	unsigned long ret;
1f1d88dd4   Mike Frysinger   disable caches be...
234
  	/*
017e9b792   Mike Frysinger   allow ports to ov...
235
236
237
  	 * pass address parameter as argv[0] (aka command name),
  	 * and all remaining args
  	 */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
238
  	ret = entry(argc, argv);
1f1d88dd4   Mike Frysinger   disable caches be...
239

017e9b792   Mike Frysinger   allow ports to ov...
240
241
  	return ret;
  }
458ded34b   wdenk   Initial revision
242

ebca3df78   Bin Meng   cmd: Clean up cmd...
243
  /*
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
244
   * Determine if a valid ELF image exists at the given memory location.
ebca3df78   Bin Meng   cmd: Clean up cmd...
245
246
247
   * First look at the ELF header magic field, then make sure that it is
   * executable.
   */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
248
249
  int valid_elf_image(unsigned long addr)
  {
ebca3df78   Bin Meng   cmd: Clean up cmd...
250
  	Elf32_Ehdr *ehdr; /* Elf header structure pointer */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
251

ebca3df78   Bin Meng   cmd: Clean up cmd...
252
  	ehdr = (Elf32_Ehdr *)addr;
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
253
254
255
256
257
258
259
260
261
262
263
264
  
  	if (!IS_ELF(*ehdr)) {
  		printf("## No elf image at address 0x%08lx
  ", addr);
  		return 0;
  	}
  
  	if (ehdr->e_type != ET_EXEC) {
  		printf("## Not a 32-bit elf image at address 0x%08lx
  ", addr);
  		return 0;
  	}
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
265
266
  	return 1;
  }
ebca3df78   Bin Meng   cmd: Clean up cmd...
267
  /* Interpreter command to boot an arbitrary ELF image from memory */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
268
  int do_bootelf(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
458ded34b   wdenk   Initial revision
269
  {
ebca3df78   Bin Meng   cmd: Clean up cmd...
270
271
  	unsigned long addr; /* Address of the ELF image */
  	unsigned long rc; /* Return value from user code */
be1b8679c   Tom Rini   cmd/elf.c: Suppor...
272
  	char *sload = NULL;
00caae6d4   Simon Glass   env: Rename geten...
273
  	const char *ep = env_get("autostart");
458ded34b   wdenk   Initial revision
274
  	int rcode = 0;
be1b8679c   Tom Rini   cmd/elf.c: Suppor...
275
276
  	/* Consume 'bootelf' */
  	argc--; argv++;
f44a928e7   Mike Frysinger   cmd_elf: add an o...
277

be1b8679c   Tom Rini   cmd/elf.c: Suppor...
278
279
280
281
282
283
284
285
286
287
288
289
  	/* Check for flag. */
  	if (argc >= 1 && (argv[0][0] == '-' && \
  				(argv[0][1] == 'p' || argv[0][1] == 's'))) {
  		sload = argv[0];
  		/* Consume flag. */
  		argc--; argv++;
  	}
  	/* Check for address. */
  	if (argc >= 1 && strict_strtoul(argv[0], 16, &addr) != -EINVAL) {
  		/* Consume address */
  		argc--; argv++;
  	} else
f44a928e7   Mike Frysinger   cmd_elf: add an o...
290
  		addr = load_addr;
458ded34b   wdenk   Initial revision
291

62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
292
  	if (!valid_elf_image(addr))
458ded34b   wdenk   Initial revision
293
  		return 1;
f44a928e7   Mike Frysinger   cmd_elf: add an o...
294
295
296
297
  	if (sload && sload[1] == 'p')
  		addr = load_elf_image_phdr(addr);
  	else
  		addr = load_elf_image_shdr(addr);
458ded34b   wdenk   Initial revision
298

44c8fd3ab   Siva Durga Prasad Paladugu   common: cmd_elf: ...
299
300
  	if (ep && !strcmp(ep, "no"))
  		return rcode;
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
301
302
  	printf("## Starting application at 0x%08lx ...
  ", addr);
458ded34b   wdenk   Initial revision
303

458ded34b   wdenk   Initial revision
304
305
306
307
  	/*
  	 * pass address parameter as argv[0] (aka command name),
  	 * and all remaining args
  	 */
be1b8679c   Tom Rini   cmd/elf.c: Suppor...
308
  	rc = do_bootelf_exec((void *)addr, argc, argv);
458ded34b   wdenk   Initial revision
309
310
  	if (rc != 0)
  		rcode = 1;
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
311
312
  	printf("## Application terminated, rc = 0x%lx
  ", rc);
ebca3df78   Bin Meng   cmd: Clean up cmd...
313

458ded34b   wdenk   Initial revision
314
315
  	return rcode;
  }
ebca3df78   Bin Meng   cmd: Clean up cmd...
316
  /*
458ded34b   wdenk   Initial revision
317
318
319
   * Interpreter command to boot VxWorks from a memory image.  The image can
   * be either an ELF image or a raw binary.  Will attempt to setup the
   * bootline and other parameters correctly.
ebca3df78   Bin Meng   cmd: Clean up cmd...
320
   */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
321
  int do_bootvx(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
458ded34b   wdenk   Initial revision
322
  {
ebca3df78   Bin Meng   cmd: Clean up cmd...
323
  	unsigned long addr; /* Address of image */
79c584e55   Bin Meng   bootvx: x86: Assi...
324
  	unsigned long bootaddr = 0; /* Address to put the bootline */
ebca3df78   Bin Meng   cmd: Clean up cmd...
325
326
327
  	char *bootline; /* Text of the bootline */
  	char *tmp; /* Temporary char pointer */
  	char build_buf[128]; /* Buffer for building the bootline */
7f0c3c51c   Bin Meng   cmd: bootvx: Avoi...
328
  	int ptr = 0;
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
329
  #ifdef CONFIG_X86
2902be86c   Bin Meng   bootvx: x86: Prep...
330
  	ulong base;
fa5e91f77   Bin Meng   vxworks: x86: Ren...
331
  	struct e820_info *info;
45519924a   Bin Meng   x86: Rename e820e...
332
  	struct e820_entry *data;
447ae4f7a   Bin Meng   bootvx: x86: Make...
333
334
  	struct efi_gop_info *gop;
  	struct vesa_mode_info *vesa = &mode_info.vesa;
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
335
  #endif
ebca3df78   Bin Meng   cmd: Clean up cmd...
336
337
  
  	/*
458ded34b   wdenk   Initial revision
338
339
340
  	 * Check the loadaddr variable.
  	 * If we don't know where the image is then we're done.
  	 */
07a1a0c93   Stany MARCEL   Correct vxWorks e...
341
  	if (argc < 2)
1bdd46832   Stefan Roese   [PATCH] common/cm...
342
343
  		addr = load_addr;
  	else
07a1a0c93   Stany MARCEL   Correct vxWorks e...
344
  		addr = simple_strtoul(argv[1], NULL, 16);
458ded34b   wdenk   Initial revision
345

baa26db41   Jon Loeliger   common/cmd_[af]*:...
346
  #if defined(CONFIG_CMD_NET)
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
347
  	/*
ebca3df78   Bin Meng   cmd: Clean up cmd...
348
349
  	 * Check to see if we need to tftp the image ourselves
  	 * before starting
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
350
  	 */
07a1a0c93   Stany MARCEL   Correct vxWorks e...
351
  	if ((argc == 2) && (strcmp(argv[1], "tftp") == 0)) {
bc0571fc1   Joe Hershberger   net: cosmetic: Fi...
352
  		if (net_loop(TFTPGET) <= 0)
458ded34b   wdenk   Initial revision
353
  			return 1;
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
354
355
356
  		printf("Automatic boot of VxWorks image at address 0x%08lx ...
  ",
  			addr);
458ded34b   wdenk   Initial revision
357
358
  	}
  #endif
ebca3df78   Bin Meng   cmd: Clean up cmd...
359
360
361
  	/*
  	 * This should equate to
  	 * NV_RAM_ADRS + NV_BOOT_OFFSET + NV_ENET_OFFSET
458ded34b   wdenk   Initial revision
362
363
364
  	 * from the VxWorks BSP header files.
  	 * This will vary from board to board
  	 */
8996975ff   Tuomas Tynkkynen   powerpc: Drop CON...
365
  #if defined(CONFIG_SYS_VXWORKS_MAC_PTR)
ebca3df78   Bin Meng   cmd: Clean up cmd...
366
  	tmp = (char *)CONFIG_SYS_VXWORKS_MAC_PTR;
35affd7a2   Simon Glass   env: Rename eth_g...
367
  	eth_env_get_enetaddr("ethaddr", (uchar *)build_buf);
62c93d92f   Mike Frysinger   bootvx: get mac a...
368
  	memcpy(tmp, build_buf, 6);
458ded34b   wdenk   Initial revision
369
  #else
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
370
371
  	puts("## Ethernet MAC address not copied to NV RAM
  ");
458ded34b   wdenk   Initial revision
372
  #endif
79c584e55   Bin Meng   bootvx: x86: Assi...
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
  #ifdef CONFIG_X86
  	/*
  	 * Get VxWorks's physical memory base address from environment,
  	 * if we don't specify it in the environment, use a default one.
  	 */
  	base = env_get_hex("vx_phys_mem_base", VXWORKS_PHYS_MEM_BASE);
  	data = (struct e820_entry *)(base + E820_DATA_OFFSET);
  	info = (struct e820_info *)(base + E820_INFO_OFFSET);
  
  	memset(info, 0, sizeof(struct e820_info));
  	info->sign = E820_SIGNATURE;
  	info->entries = install_e820_map(E820MAX, data);
  	info->addr = (info->entries - 1) * sizeof(struct e820_entry) +
  		     E820_DATA_OFFSET;
  
  	/*
  	 * Explicitly clear the bootloader image size otherwise if memory
  	 * at this offset happens to contain some garbage data, the final
  	 * available memory size for the kernel is insane.
  	 */
  	*(u32 *)(base + BOOT_IMAGE_SIZE_OFFSET) = 0;
  
  	/*
  	 * Prepare compatible framebuffer information block.
  	 * The VESA mode has to be 32-bit RGBA.
  	 */
  	if (vesa->x_resolution && vesa->y_resolution) {
  		gop = (struct efi_gop_info *)(base + EFI_GOP_INFO_OFFSET);
  		gop->magic = EFI_GOP_INFO_MAGIC;
  		gop->info.version = 0;
  		gop->info.width = vesa->x_resolution;
  		gop->info.height = vesa->y_resolution;
  		gop->info.pixel_format = EFI_GOT_RGBA8;
  		gop->info.pixels_per_scanline = vesa->bytes_per_scanline / 4;
  		gop->fb_base = vesa->phys_base_ptr;
  		gop->fb_size = vesa->bytes_per_scanline * vesa->y_resolution;
  	}
  #endif
8bde7f776   wdenk   * Code cleanup:
411
412
  	/*
  	 * Use bootaddr to find the location in memory that VxWorks
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
413
414
415
  	 * will look for the bootline string. The default value is
  	 * (LOCAL_MEM_LOCAL_ADRS + BOOT_LINE_OFFSET) as defined by
  	 * VxWorks BSP. For example, on PowerPC it defaults to 0x4200.
458ded34b   wdenk   Initial revision
416
  	 */
00caae6d4   Simon Glass   env: Rename geten...
417
  	tmp = env_get("bootaddr");
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
418
  	if (!tmp) {
79c584e55   Bin Meng   bootvx: x86: Assi...
419
420
421
  #ifdef CONFIG_X86
  		bootaddr = base + X86_BOOT_LINE_OFFSET;
  #else
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
422
423
  		printf("## VxWorks bootline address not specified
  ");
ced71a2f7   Bin Meng   bootvx: Exit if b...
424
  		return 1;
79c584e55   Bin Meng   bootvx: x86: Assi...
425
  #endif
ced71a2f7   Bin Meng   bootvx: Exit if b...
426
  	}
79c584e55   Bin Meng   bootvx: x86: Assi...
427
428
  	if (!bootaddr)
  		bootaddr = simple_strtoul(tmp, NULL, 16);
ced71a2f7   Bin Meng   bootvx: Exit if b...
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
  
  	/*
  	 * Check to see if the bootline is defined in the 'bootargs' parameter.
  	 * If it is not defined, we may be able to construct the info.
  	 */
  	bootline = env_get("bootargs");
  	if (!bootline) {
  		tmp = env_get("bootdev");
  		if (tmp) {
  			strcpy(build_buf, tmp);
  			ptr = strlen(tmp);
  		} else {
  			printf("## VxWorks boot device not specified
  ");
  		}
  
  		tmp = env_get("bootfile");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "host:%s ", tmp);
  		else
  			ptr += sprintf(build_buf + ptr, "host:vxWorks ");
458ded34b   wdenk   Initial revision
450

9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
451
  		/*
ced71a2f7   Bin Meng   bootvx: Exit if b...
452
453
  		 * The following parameters are only needed if 'bootdev'
  		 * is an ethernet device, otherwise they are optional.
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
454
  		 */
ced71a2f7   Bin Meng   bootvx: Exit if b...
455
456
457
458
  		tmp = env_get("ipaddr");
  		if (tmp) {
  			ptr += sprintf(build_buf + ptr, "e=%s", tmp);
  			tmp = env_get("netmask");
192bc6948   Ben Whitten   Fix GCC format-se...
459
  			if (tmp) {
ced71a2f7   Bin Meng   bootvx: Exit if b...
460
  				u32 mask = env_get_ip("netmask").s_addr;
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
461
  				ptr += sprintf(build_buf + ptr,
ced71a2f7   Bin Meng   bootvx: Exit if b...
462
463
464
  					       ":%08x ", ntohl(mask));
  			} else {
  				ptr += sprintf(build_buf + ptr, " ");
a4092dbd8   Bin Meng   cmd: bootvx: Pass...
465
  			}
ced71a2f7   Bin Meng   bootvx: Exit if b...
466
  		}
458ded34b   wdenk   Initial revision
467

ced71a2f7   Bin Meng   bootvx: Exit if b...
468
469
470
  		tmp = env_get("serverip");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "h=%s ", tmp);
a4092dbd8   Bin Meng   cmd: bootvx: Pass...
471

ced71a2f7   Bin Meng   bootvx: Exit if b...
472
473
474
  		tmp = env_get("gatewayip");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "g=%s ", tmp);
458ded34b   wdenk   Initial revision
475

ced71a2f7   Bin Meng   bootvx: Exit if b...
476
477
478
  		tmp = env_get("hostname");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "tn=%s ", tmp);
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
479

ced71a2f7   Bin Meng   bootvx: Exit if b...
480
481
482
483
  		tmp = env_get("othbootargs");
  		if (tmp) {
  			strcpy(build_buf + ptr, tmp);
  			ptr += strlen(tmp);
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
484
  		}
29a4c24de   Niklaus Giger   cmd_elf.c: Cleanu...
485

ced71a2f7   Bin Meng   bootvx: Exit if b...
486
  		bootline = build_buf;
458ded34b   wdenk   Initial revision
487
  	}
ced71a2f7   Bin Meng   bootvx: Exit if b...
488
489
490
491
  	memcpy((void *)bootaddr, bootline, max(strlen(bootline), (size_t)255));
  	flush_cache(bootaddr, max(strlen(bootline), (size_t)255));
  	printf("## Using bootline (@ 0x%lx): %s
  ", bootaddr, (char *)bootaddr);
8bde7f776   wdenk   * Code cleanup:
492
493
494
  	/*
  	 * If the data at the load address is an elf image, then
  	 * treat it like an elf image. Otherwise, assume that it is a
ebca3df78   Bin Meng   cmd: Clean up cmd...
495
  	 * binary image.
458ded34b   wdenk   Initial revision
496
  	 */
ebca3df78   Bin Meng   cmd: Clean up cmd...
497
  	if (valid_elf_image(addr))
476c2fcd2   Christian Gmeiner   bootvx: use progr...
498
  		addr = load_elf_image_phdr(addr);
ebca3df78   Bin Meng   cmd: Clean up cmd...
499
  	else
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
500
501
  		puts("## Not an ELF image, assuming binary
  ");
458ded34b   wdenk   Initial revision
502

62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
503
504
  	printf("## Starting vxWorks at 0x%08lx ...
  ", addr);
458ded34b   wdenk   Initial revision
505

6eee21daf   Reinhard Arlt   bootvx: Clear and...
506
  	dcache_disable();
3194daa10   Vasyl Vavrychuk   vxworks: fixed cp...
507
508
509
510
  #if defined(CONFIG_ARM64) && defined(CONFIG_ARMV8_PSCI)
  	armv8_setup_psci();
  	smp_kick_all_cpus();
  #endif
9aa1280a5   Bin Meng   cmd: bootvx: Add ...
511
512
513
514
  #ifdef CONFIG_X86
  	/* VxWorks on x86 uses stack to pass parameters */
  	((asmlinkage void (*)(int))addr)(0);
  #else
ebca3df78   Bin Meng   cmd: Clean up cmd...
515
  	((void (*)(int))addr)(0);
9aa1280a5   Bin Meng   cmd: bootvx: Add ...
516
  #endif
458ded34b   wdenk   Initial revision
517

62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
518
519
  	puts("## vxWorks terminated
  ");
ebca3df78   Bin Meng   cmd: Clean up cmd...
520

458ded34b   wdenk   Initial revision
521
522
  	return 1;
  }
0d4983930   wdenk   Patch by Kenneth ...
523
  U_BOOT_CMD(
be1b8679c   Tom Rini   cmd/elf.c: Suppor...
524
  	bootelf, CONFIG_SYS_MAXARGS, 0, do_bootelf,
2fb2604d5   Peter Tyser   Command usage cle...
525
  	"Boot from an ELF image in memory",
f44a928e7   Mike Frysinger   cmd_elf: add an o...
526
527
528
529
530
  	"[-p|-s] [address]
  "
  	"\t- load ELF image at [address] via program headers (-p)
  "
  	"\t  or via section headers (-s)"
8bde7f776   wdenk   * Code cleanup:
531
  );
0d4983930   wdenk   Patch by Kenneth ...
532
  U_BOOT_CMD(
ebca3df78   Bin Meng   cmd: Clean up cmd...
533
  	bootvx, 2, 0, do_bootvx,
2fb2604d5   Peter Tyser   Command usage cle...
534
  	"Boot vxWorks from an ELF image",
a89c33db9   Wolfgang Denk   General help mess...
535
  	" [address] - load address of vxWorks ELF image."
8bde7f776   wdenk   * Code cleanup:
536
  );