warp64.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. /*
  2. * This file is part of the UCB release of Plan 9. It is subject to the license
  3. * terms in the LICENSE file found in the top-level directory of this
  4. * distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No
  5. * part of the UCB release of Plan 9, including this file, may be copied,
  6. * modified, propagated, or distributed except according to the terms contained
  7. * in the LICENSE file.
  8. */
  9. #include "u.h"
  10. #include "lib.h"
  11. #include "mem.h"
  12. #include "dat.h"
  13. #include "fns.h"
  14. #include "io.h"
  15. #include "mmu64.h"
  16. #define Pml4 0x00108000
  17. typedef unsigned long long u64intptr;
  18. #define u64ptr2int(p) ((u64intptr)(p))
  19. #define u64int2ptr(i) ((void*)(i))
  20. static u64intptr
  21. mach0ptalloc(int l)
  22. {
  23. static u64intptr table = Pml4;
  24. static int ntable = 6;
  25. if(ntable <= 0)
  26. return ~0ULL;
  27. table += PGSIZE64;
  28. memset(KADDR(table), 0, PGSIZE64);
  29. ntable--;
  30. if(vflag)
  31. print("table[%d] used 0x%llux\n", l, table);
  32. return table;
  33. }
  34. PTE*
  35. mmuwalk64(PTE* pml4, u64intptr va, int level, u64intptr (*alloc)(int))
  36. {
  37. int l, idx;
  38. PTE *pte;
  39. u64intptr pa;
  40. idx = PTEX64(va, 4);
  41. pte = &pml4[idx];
  42. for(l = 4; l > 0; l--){
  43. if(vflag)
  44. print("mmuwalk64 0x%p 0x%llux %d: entry %d\n",
  45. pml4, va, l, idx);
  46. if(l == level)
  47. return pte;
  48. if(l == 2 && (*pte & PtePS))
  49. break;
  50. if(!(*pte & PteP)){
  51. if(alloc == nil)
  52. break;
  53. pa = alloc(l);
  54. if(pa == ~0ULL)
  55. break;
  56. *pte = pa|PteRW|PteP;
  57. if(vflag)
  58. print("mmuwalk64 0x%p 0x%llux %d: alloc 0x%llux\n",
  59. pml4, va, l, pa);
  60. }
  61. pte = u64int2ptr(KADDR(PPN64(*pte)));
  62. idx = PTEX64(va, l-1);
  63. pte += idx;
  64. }
  65. return nil;
  66. }
  67. void
  68. mkmach0pt(u64intptr kzero64)
  69. {
  70. //u32int r;
  71. u64intptr pa, va;
  72. PTE *pml4, *pte, *pte0;
  73. //uvlong uvlr;
  74. pml4 = u64int2ptr(KADDR(Pml4));
  75. if(vflag)
  76. print("pml4 = %p\n", pml4);
  77. memset(pml4, 0, PGSIZE64);
  78. va = kzero64;
  79. for(pa = 0; pa < 4*MiB; pa += 2*MiB){
  80. pte = mmuwalk64(pml4, va, 2, mach0ptalloc);
  81. *pte = (PPN64(pa))|PtePS|PteRW|PteP;
  82. if(vflag)
  83. print("va %#llux pte %#p *pte %#llux\n", va, pte, *pte);
  84. va += 2*MiB;
  85. }
  86. pte = mmuwalk64(pml4, kzero64, 4, 0);
  87. if(vflag)
  88. print("pte l4 %llux == 0x%llux\n", kzero64, *pte);
  89. pte0 = mmuwalk64(pml4, 0ULL, 2, mach0ptalloc);
  90. pte0 += PTEX64(0, 2);
  91. if(vflag)
  92. print("pte0 l2 @ 0x%p 0 == 0x%llux\n", pte0, *pte0);
  93. *pte0 = (PPN64(0))|PtePS|PteRW|PteP;
  94. if(vflag)
  95. print("pte0 l2 @ 0x%p 0 == 0x%llux\n", pte0, *pte0);
  96. /*
  97. * Have to do this with paging turned off. Bugger.
  98. r = getcr4();
  99. r |= Pae;
  100. putcr4(r);
  101. r = getcr3();
  102. putcr3(Pml4);
  103. rdmsr(Efer, &uvlr);
  104. uvlr |= Lme;
  105. wrmsr(Efer, uvlr);
  106. */
  107. }
  108. void
  109. warp64(uint64_t entry)
  110. {
  111. u64intptr kzero64;
  112. extern void _warp64(ulong);
  113. // kzero64 = KZERO64;
  114. // if(entry != 0xFFFFFFFF80110000ULL){
  115. // print("bad entry address %#llux\n", entry);
  116. // return;
  117. // }
  118. kzero64 = entry & ~0x000000000fffffffull;
  119. print("warp64(%#llux) %#llux %d\n", entry, kzero64, nmmap);
  120. if(vflag)
  121. print("mkmultiboot\n");
  122. mkmultiboot();
  123. if(vflag)
  124. print("mkmach0pt\n");
  125. mkmach0pt(kzero64);
  126. if(vflag)
  127. print("impulse\n");
  128. /*
  129. * No output after impulse().
  130. */
  131. if(vflag)
  132. print("_warp64\n");
  133. impulse();
  134. _warp64(Pml4);
  135. }