Blame view

cmd/elf.c 14.6 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
   *
   * 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);
957f51e86   Kurban Mallachiev   elf: fix cache fl...
57
58
  		flush_cache(rounddown((unsigned long)dst, ARCH_DMA_MINALIGN),
  			    roundup(phdr->p_memsz, ARCH_DMA_MINALIGN));
839c4e9c5   Bin Meng   elf: Add a very s...
59
60
  		++phdr;
  	}
2846ea81a   Rob Bracero   elf: Add support ...
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
119
  	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...
120
121
122
123
  		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 ...
124
125
126
127
128
129
130
131
132
133
134
135
136
  	}
  
  	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...
137
138
139
140
141
142
143
144
145
  	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...
146
147
148
149
150
151
152
153
   */
  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...
154
155
  	if (ehdr->e_ident[EI_CLASS] == ELFCLASS64)
  		return load_elf64_image_phdr(addr);
9dffa52da   Bin Meng   cmd: elf: Reorder...
156
157
158
159
160
161
  	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...
162

9dffa52da   Bin Meng   cmd: elf: Reorder...
163
164
165
166
167
168
169
170
  		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);
957f51e86   Kurban Mallachiev   elf: fix cache fl...
171
172
  		flush_cache(rounddown((unsigned long)dst, ARCH_DMA_MINALIGN),
  			    roundup(phdr->p_memsz, ARCH_DMA_MINALIGN));
9dffa52da   Bin Meng   cmd: elf: Reorder...
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
  		++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 ...
188
189
  	if (ehdr->e_ident[EI_CLASS] == ELFCLASS64)
  		return load_elf64_image_shdr(addr);
9dffa52da   Bin Meng   cmd: elf: Reorder...
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
  
  	/* 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);
  		}
18f201ea2   Neil Stainton   u-boot: align cac...
225
226
227
228
  		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));
9dffa52da   Bin Meng   cmd: elf: Reorder...
229
230
231
232
  	}
  
  	return ehdr->e_entry;
  }
017e9b792   Mike Frysinger   allow ports to ov...
233
234
  
  /* Allow ports to override the default behavior */
553d8c3a5   Jeroen Hofstee   common: cmd_elf: ...
235
  static unsigned long do_bootelf_exec(ulong (*entry)(int, char * const[]),
ebca3df78   Bin Meng   cmd: Clean up cmd...
236
  				     int argc, char * const argv[])
1f1d88dd4   Mike Frysinger   disable caches be...
237
  {
017e9b792   Mike Frysinger   allow ports to ov...
238
  	unsigned long ret;
1f1d88dd4   Mike Frysinger   disable caches be...
239
  	/*
017e9b792   Mike Frysinger   allow ports to ov...
240
241
242
  	 * pass address parameter as argv[0] (aka command name),
  	 * and all remaining args
  	 */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
243
  	ret = entry(argc, argv);
1f1d88dd4   Mike Frysinger   disable caches be...
244

017e9b792   Mike Frysinger   allow ports to ov...
245
246
  	return ret;
  }
458ded34b   wdenk   Initial revision
247

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

ebca3df78   Bin Meng   cmd: Clean up cmd...
257
  	ehdr = (Elf32_Ehdr *)addr;
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
258
259
260
261
262
263
264
265
266
267
268
269
  
  	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...
270
271
  	return 1;
  }
ebca3df78   Bin Meng   cmd: Clean up cmd...
272
  /* Interpreter command to boot an arbitrary ELF image from memory */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
273
  int do_bootelf(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
458ded34b   wdenk   Initial revision
274
  {
ebca3df78   Bin Meng   cmd: Clean up cmd...
275
276
  	unsigned long addr; /* Address of the ELF image */
  	unsigned long rc; /* Return value from user code */
be1b8679c   Tom Rini   cmd/elf.c: Suppor...
277
  	char *sload = NULL;
00caae6d4   Simon Glass   env: Rename geten...
278
  	const char *ep = env_get("autostart");
458ded34b   wdenk   Initial revision
279
  	int rcode = 0;
be1b8679c   Tom Rini   cmd/elf.c: Suppor...
280
281
  	/* Consume 'bootelf' */
  	argc--; argv++;
f44a928e7   Mike Frysinger   cmd_elf: add an o...
282

be1b8679c   Tom Rini   cmd/elf.c: Suppor...
283
284
285
286
287
288
289
290
291
292
293
294
  	/* 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...
295
  		addr = load_addr;
458ded34b   wdenk   Initial revision
296

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

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

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

458ded34b   wdenk   Initial revision
319
320
  	return rcode;
  }
ebca3df78   Bin Meng   cmd: Clean up cmd...
321
  /*
458ded34b   wdenk   Initial revision
322
323
324
   * 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...
325
   */
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
326
  int do_bootvx(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
458ded34b   wdenk   Initial revision
327
  {
ebca3df78   Bin Meng   cmd: Clean up cmd...
328
  	unsigned long addr; /* Address of image */
79c584e55   Bin Meng   bootvx: x86: Assi...
329
  	unsigned long bootaddr = 0; /* Address to put the bootline */
ebca3df78   Bin Meng   cmd: Clean up cmd...
330
331
332
  	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...
333
  	int ptr = 0;
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
334
  #ifdef CONFIG_X86
2902be86c   Bin Meng   bootvx: x86: Prep...
335
  	ulong base;
fa5e91f77   Bin Meng   vxworks: x86: Ren...
336
  	struct e820_info *info;
45519924a   Bin Meng   x86: Rename e820e...
337
  	struct e820_entry *data;
447ae4f7a   Bin Meng   bootvx: x86: Make...
338
339
  	struct efi_gop_info *gop;
  	struct vesa_mode_info *vesa = &mode_info.vesa;
b90ff0fda   Bin Meng   cmd: bootvx: Pass...
340
  #endif
ebca3df78   Bin Meng   cmd: Clean up cmd...
341
342
  
  	/*
458ded34b   wdenk   Initial revision
343
344
345
  	 * Check the loadaddr variable.
  	 * If we don't know where the image is then we're done.
  	 */
07a1a0c93   Stany MARCEL   Correct vxWorks e...
346
  	if (argc < 2)
1bdd46832   Stefan Roese   [PATCH] common/cm...
347
348
  		addr = load_addr;
  	else
07a1a0c93   Stany MARCEL   Correct vxWorks e...
349
  		addr = simple_strtoul(argv[1], NULL, 16);
458ded34b   wdenk   Initial revision
350

baa26db41   Jon Loeliger   common/cmd_[af]*:...
351
  #if defined(CONFIG_CMD_NET)
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
352
  	/*
ebca3df78   Bin Meng   cmd: Clean up cmd...
353
354
  	 * Check to see if we need to tftp the image ourselves
  	 * before starting
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
355
  	 */
07a1a0c93   Stany MARCEL   Correct vxWorks e...
356
  	if ((argc == 2) && (strcmp(argv[1], "tftp") == 0)) {
bc0571fc1   Joe Hershberger   net: cosmetic: Fi...
357
  		if (net_loop(TFTPGET) <= 0)
458ded34b   wdenk   Initial revision
358
  			return 1;
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
359
360
361
  		printf("Automatic boot of VxWorks image at address 0x%08lx ...
  ",
  			addr);
458ded34b   wdenk   Initial revision
362
363
  	}
  #endif
ebca3df78   Bin Meng   cmd: Clean up cmd...
364
365
366
  	/*
  	 * This should equate to
  	 * NV_RAM_ADRS + NV_BOOT_OFFSET + NV_ENET_OFFSET
458ded34b   wdenk   Initial revision
367
368
369
  	 * from the VxWorks BSP header files.
  	 * This will vary from board to board
  	 */
8996975ff   Tuomas Tynkkynen   powerpc: Drop CON...
370
  #if defined(CONFIG_SYS_VXWORKS_MAC_PTR)
ebca3df78   Bin Meng   cmd: Clean up cmd...
371
  	tmp = (char *)CONFIG_SYS_VXWORKS_MAC_PTR;
35affd7a2   Simon Glass   env: Rename eth_g...
372
  	eth_env_get_enetaddr("ethaddr", (uchar *)build_buf);
62c93d92f   Mike Frysinger   bootvx: get mac a...
373
  	memcpy(tmp, build_buf, 6);
458ded34b   wdenk   Initial revision
374
  #else
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
375
376
  	puts("## Ethernet MAC address not copied to NV RAM
  ");
458ded34b   wdenk   Initial revision
377
  #endif
79c584e55   Bin Meng   bootvx: x86: Assi...
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
411
412
413
414
415
  #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:
416
417
  	/*
  	 * Use bootaddr to find the location in memory that VxWorks
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
418
419
420
  	 * 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
421
  	 */
00caae6d4   Simon Glass   env: Rename geten...
422
  	tmp = env_get("bootaddr");
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
423
  	if (!tmp) {
79c584e55   Bin Meng   bootvx: x86: Assi...
424
425
426
  #ifdef CONFIG_X86
  		bootaddr = base + X86_BOOT_LINE_OFFSET;
  #else
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
427
428
  		printf("## VxWorks bootline address not specified
  ");
ced71a2f7   Bin Meng   bootvx: Exit if b...
429
  		return 1;
79c584e55   Bin Meng   bootvx: x86: Assi...
430
  #endif
ced71a2f7   Bin Meng   bootvx: Exit if b...
431
  	}
79c584e55   Bin Meng   bootvx: x86: Assi...
432
433
  	if (!bootaddr)
  		bootaddr = simple_strtoul(tmp, NULL, 16);
ced71a2f7   Bin Meng   bootvx: Exit if b...
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
  
  	/*
  	 * 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
455

9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
456
  		/*
ced71a2f7   Bin Meng   bootvx: Exit if b...
457
458
  		 * The following parameters are only needed if 'bootdev'
  		 * is an ethernet device, otherwise they are optional.
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
459
  		 */
ced71a2f7   Bin Meng   bootvx: Exit if b...
460
461
462
463
  		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...
464
  			if (tmp) {
ced71a2f7   Bin Meng   bootvx: Exit if b...
465
  				u32 mask = env_get_ip("netmask").s_addr;
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
466
  				ptr += sprintf(build_buf + ptr,
ced71a2f7   Bin Meng   bootvx: Exit if b...
467
468
469
  					       ":%08x ", ntohl(mask));
  			} else {
  				ptr += sprintf(build_buf + ptr, " ");
a4092dbd8   Bin Meng   cmd: bootvx: Pass...
470
  			}
ced71a2f7   Bin Meng   bootvx: Exit if b...
471
  		}
458ded34b   wdenk   Initial revision
472

ced71a2f7   Bin Meng   bootvx: Exit if b...
473
474
475
  		tmp = env_get("serverip");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "h=%s ", tmp);
a4092dbd8   Bin Meng   cmd: bootvx: Pass...
476

ced71a2f7   Bin Meng   bootvx: Exit if b...
477
478
479
  		tmp = env_get("gatewayip");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "g=%s ", tmp);
458ded34b   wdenk   Initial revision
480

ced71a2f7   Bin Meng   bootvx: Exit if b...
481
482
483
  		tmp = env_get("hostname");
  		if (tmp)
  			ptr += sprintf(build_buf + ptr, "tn=%s ", tmp);
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
484

ced71a2f7   Bin Meng   bootvx: Exit if b...
485
486
487
488
  		tmp = env_get("othbootargs");
  		if (tmp) {
  			strcpy(build_buf + ptr, tmp);
  			ptr += strlen(tmp);
9e98b7e3c   Bin Meng   cmd: bootvx: Alwa...
489
  		}
29a4c24de   Niklaus Giger   cmd_elf.c: Cleanu...
490

ced71a2f7   Bin Meng   bootvx: Exit if b...
491
  		bootline = build_buf;
458ded34b   wdenk   Initial revision
492
  	}
ced71a2f7   Bin Meng   bootvx: Exit if b...
493
494
495
496
  	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:
497
498
499
  	/*
  	 * 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...
500
  	 * binary image.
458ded34b   wdenk   Initial revision
501
  	 */
ebca3df78   Bin Meng   cmd: Clean up cmd...
502
  	if (valid_elf_image(addr))
476c2fcd2   Christian Gmeiner   bootvx: use progr...
503
  		addr = load_elf_image_phdr(addr);
ebca3df78   Bin Meng   cmd: Clean up cmd...
504
  	else
62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
505
506
  		puts("## Not an ELF image, assuming binary
  ");
458ded34b   wdenk   Initial revision
507

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

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

62e03d33c   Daniel Schwierzeck   common: cmd_elf.c...
523
524
  	puts("## vxWorks terminated
  ");
ebca3df78   Bin Meng   cmd: Clean up cmd...
525

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