of.c 2.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. /*
  2. * Copyright (C) Paul Mackerras 1997.
  3. *
  4. * This program is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU General Public License
  6. * as published by the Free Software Foundation; either version
  7. * 2 of the License, or (at your option) any later version.
  8. */
  9. #include <stdarg.h>
  10. #include <stddef.h>
  11. #include "types.h"
  12. #include "elf.h"
  13. #include "string.h"
  14. #include "stdio.h"
  15. #include "page.h"
  16. #include "ops.h"
  17. #include "of.h"
  18. /* Value picked to match that used by yaboot */
  19. #define PROG_START 0x01400000 /* only used on 64-bit systems */
  20. #define RAM_END (512<<20) /* Fixme: use OF */
  21. #define ONE_MB 0x100000
  22. static unsigned long claim_base;
  23. void epapr_platform_init(unsigned long r3, unsigned long r4, unsigned long r5,
  24. unsigned long r6, unsigned long r7);
  25. static void *of_try_claim(unsigned long size)
  26. {
  27. unsigned long addr = 0;
  28. if (claim_base == 0)
  29. claim_base = _ALIGN_UP((unsigned long)_end, ONE_MB);
  30. for(; claim_base < RAM_END; claim_base += ONE_MB) {
  31. #ifdef DEBUG
  32. printf(" trying: 0x%08lx\n\r", claim_base);
  33. #endif
  34. addr = (unsigned long) of_claim(claim_base, size, 0);
  35. if (addr != PROM_ERROR)
  36. break;
  37. }
  38. if (addr == 0)
  39. return NULL;
  40. claim_base = PAGE_ALIGN(claim_base + size);
  41. return (void *)addr;
  42. }
  43. static void of_image_hdr(const void *hdr)
  44. {
  45. const Elf64_Ehdr *elf64 = hdr;
  46. if (elf64->e_ident[EI_CLASS] == ELFCLASS64) {
  47. /*
  48. * Maintain a "magic" minimum address. This keeps some older
  49. * firmware platforms running.
  50. */
  51. if (claim_base < PROG_START)
  52. claim_base = PROG_START;
  53. }
  54. }
  55. static void of_platform_init(unsigned long a1, unsigned long a2, void *promptr)
  56. {
  57. platform_ops.image_hdr = of_image_hdr;
  58. platform_ops.malloc = of_try_claim;
  59. platform_ops.exit = of_exit;
  60. platform_ops.vmlinux_alloc = of_vmlinux_alloc;
  61. dt_ops.finddevice = of_finddevice;
  62. dt_ops.getprop = of_getprop;
  63. dt_ops.setprop = of_setprop;
  64. of_console_init();
  65. of_init(promptr);
  66. loader_info.promptr = promptr;
  67. if (a1 && a2 && a2 != 0xdeadbeef) {
  68. loader_info.initrd_addr = a1;
  69. loader_info.initrd_size = a2;
  70. }
  71. }
  72. void platform_init(unsigned long r3, unsigned long r4, unsigned long r5,
  73. unsigned long r6, unsigned long r7)
  74. {
  75. /* Detect OF vs. ePAPR boot */
  76. if (r5)
  77. of_platform_init(r3, r4, (void *)r5);
  78. else
  79. epapr_platform_init(r3, r4, r5, r6, r7);
  80. }