98 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			98 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) Paul Mackerras 1997.
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version
 | |
|  * 2 of the License, or (at your option) any later version.
 | |
|  */
 | |
| #include <stdarg.h>
 | |
| #include <stddef.h>
 | |
| #include "types.h"
 | |
| #include "elf.h"
 | |
| #include "string.h"
 | |
| #include "stdio.h"
 | |
| #include "page.h"
 | |
| #include "ops.h"
 | |
| 
 | |
| #include "of.h"
 | |
| 
 | |
| /* Value picked to match that used by yaboot */
 | |
| #define PROG_START	0x01400000	/* only used on 64-bit systems */
 | |
| #define RAM_END		(512<<20)	/* Fixme: use OF */
 | |
| #define	ONE_MB		0x100000
 | |
| 
 | |
| 
 | |
| 
 | |
| static unsigned long claim_base;
 | |
| 
 | |
| void epapr_platform_init(unsigned long r3, unsigned long r4, unsigned long r5,
 | |
| 			 unsigned long r6, unsigned long r7);
 | |
| 
 | |
| static void *of_try_claim(unsigned long size)
 | |
| {
 | |
| 	unsigned long addr = 0;
 | |
| 
 | |
| 	if (claim_base == 0)
 | |
| 		claim_base = _ALIGN_UP((unsigned long)_end, ONE_MB);
 | |
| 
 | |
| 	for(; claim_base < RAM_END; claim_base += ONE_MB) {
 | |
| #ifdef DEBUG
 | |
| 		printf("    trying: 0x%08lx\n\r", claim_base);
 | |
| #endif
 | |
| 		addr = (unsigned long) of_claim(claim_base, size, 0);
 | |
| 		if (addr != PROM_ERROR)
 | |
| 			break;
 | |
| 	}
 | |
| 	if (addr == 0)
 | |
| 		return NULL;
 | |
| 	claim_base = PAGE_ALIGN(claim_base + size);
 | |
| 	return (void *)addr;
 | |
| }
 | |
| 
 | |
| static void of_image_hdr(const void *hdr)
 | |
| {
 | |
| 	const Elf64_Ehdr *elf64 = hdr;
 | |
| 
 | |
| 	if (elf64->e_ident[EI_CLASS] == ELFCLASS64) {
 | |
| 		/*
 | |
| 		 * Maintain a "magic" minimum address. This keeps some older
 | |
| 		 * firmware platforms running.
 | |
| 		 */
 | |
| 		if (claim_base < PROG_START)
 | |
| 			claim_base = PROG_START;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void of_platform_init(unsigned long a1, unsigned long a2, void *promptr)
 | |
| {
 | |
| 	platform_ops.image_hdr = of_image_hdr;
 | |
| 	platform_ops.malloc = of_try_claim;
 | |
| 	platform_ops.exit = of_exit;
 | |
| 	platform_ops.vmlinux_alloc = of_vmlinux_alloc;
 | |
| 
 | |
| 	dt_ops.finddevice = of_finddevice;
 | |
| 	dt_ops.getprop = of_getprop;
 | |
| 	dt_ops.setprop = of_setprop;
 | |
| 
 | |
| 	of_console_init();
 | |
| 
 | |
| 	of_init(promptr);
 | |
| 	loader_info.promptr = promptr;
 | |
| 	if (a1 && a2 && a2 != 0xdeadbeef) {
 | |
| 		loader_info.initrd_addr = a1;
 | |
| 		loader_info.initrd_size = a2;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void platform_init(unsigned long r3, unsigned long r4, unsigned long r5,
 | |
| 		   unsigned long r6, unsigned long r7)
 | |
| {
 | |
| 	/* Detect OF vs. ePAPR boot */
 | |
| 	if (r5)
 | |
| 		of_platform_init(r3, r4, (void *)r5);
 | |
| 	else
 | |
| 		epapr_platform_init(r3, r4, r5, r6, r7);
 | |
| }
 | |
| 
 |