risc_tlb.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477
  1. /*
  2. * Copyright (C) 2017-2018 MIPS Tech, LLC
  3. *
  4. * Redistribution and use in source and binary forms, with or without
  5. * modification, are permitted provided that the following conditions are met:
  6. *
  7. * 1. Redistributions of source code must retain the above copyright notice,
  8. * this list of conditions and the following disclaimer.
  9. * 2. Redistributions in binary form must reproduce the above copyright notice,
  10. * this list of conditions and the following disclaimer in the documentation
  11. * and/or other materials provided with the distribution.
  12. * 3. Neither the name of the copyright holder nor the names of its
  13. * contributors may be used to endorse or promote products derived from this
  14. * software without specific prior written permission.
  15. *
  16. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  17. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  18. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  19. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
  20. * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  21. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  22. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  23. * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  24. * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  25. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  26. * POSSIBILITY OF SUCH DAMAGE.
  27. */
  28. #include "cache.h"
  29. #include <mips/hal.h>
  30. #include <mips/m32tlb.h>
  31. /* Writes hi, lo0, lo1 and mask into the TLB entry specified by index */
  32. void __attribute__ ((use_hazard_barrier_return)) _MIPS_HAL_NOMIPS16
  33. mips_tlbwi2 (tlbhi_t hi, tlblo_t lo0, tlblo_t lo1, unsigned int mask,
  34. int index)
  35. {
  36. mips32_setentryhi (hi);
  37. mips32_setentrylo0 (lo0);
  38. mips32_setentrylo1 (lo1);
  39. mips32_setpagemask (mask);
  40. mips32_setindex (index);
  41. mips_ehb ();
  42. mips_tlbwi ();
  43. return;
  44. }
  45. /*
  46. * Writes hi, lo0, lo1 and mask into the TLB entry specified by the
  47. * random register.
  48. */
  49. void __attribute__ ((use_hazard_barrier_return)) _MIPS_HAL_NOMIPS16
  50. mips_tlbwr2 (tlbhi_t hi, tlblo_t lo0, tlblo_t lo1, unsigned int mask)
  51. {
  52. mips32_setentryhi (hi);
  53. mips32_setentrylo0 (lo0);
  54. mips32_setentrylo1 (lo1);
  55. mips32_setpagemask (mask);
  56. mips_ehb ();
  57. mips_tlbwr ();
  58. return;
  59. }
  60. /*
  61. * Probes the TLB for an entry matching hi and if present rewrites that entry,
  62. * otherwise updates a random entry. A safe way to update the TLB.
  63. */
  64. int __attribute__ ((use_hazard_barrier_return)) _MIPS_HAL_NOMIPS16
  65. mips_tlbrwr2 (tlbhi_t hi, tlblo_t lo0, tlblo_t lo1, unsigned int mask)
  66. {
  67. int index;
  68. tlbhi_t prev_hi;
  69. prev_hi = mips32_getentryhi ();
  70. mips32_setentryhi (hi);
  71. mips_ehb (); /* mtc0 hazard on tlbp */
  72. mips_tlbp ();
  73. mips_ehb (); /* tlbp hazard on mfc0 */
  74. index = mips32_getindex ();
  75. mips32_setentrylo0 (lo0);
  76. mips32_setentrylo1 (lo1);
  77. mips32_setpagemask (mask);
  78. mips_ehb (); /* mtc0 hazard on tlbwi/tlbwr */
  79. /* Check if entry matches */
  80. if (index >= 0)
  81. mips_tlbwi ();
  82. else
  83. mips_tlbwr ();
  84. mips32_setentryhi (prev_hi);
  85. return index;
  86. }
  87. /*
  88. * Reads the TLB entry specified by index, and returns the EntryHi,
  89. * EntryLo0, EntryLo1 and PageMask parts in *phi, *plo0, *plo1 and *pmask
  90. * respectively.
  91. */
  92. void _MIPS_HAL_NOMIPS16
  93. mips_tlbri2 (tlbhi_t *phi, tlblo_t *plo0, tlblo_t *plo1,
  94. unsigned int *pmask, int index)
  95. {
  96. mips32_setindex (index);
  97. mips_ehb (); /* mtc0 hazard on tlbr */
  98. mips_tlbr ();
  99. mips_ehb (); /* tlbr hazard on mfc0 */
  100. *phi = mips32_getentryhi ();
  101. *plo0 = mips32_getentrylo0 ();
  102. *plo1 = mips32_getentrylo1 ();
  103. *pmask = mips32_getpagemask ();
  104. return;
  105. }
  106. /*
  107. * Probes the TLB for an entry matching hi and return its index, or -1 if
  108. * not found. If found, the EntryLo0, EntryLo1 and PageMask parts of the
  109. * entry are also returned in *plo0, *plo1 and *pmask respectively.
  110. */
  111. int __attribute__ ((use_hazard_barrier_return)) _MIPS_HAL_NOMIPS16
  112. mips_tlbprobe2 (tlbhi_t hi, tlblo_t *plo0, tlblo_t *plo1,
  113. unsigned int *pmask)
  114. {
  115. int index;
  116. tlbhi_t prev_hi;
  117. prev_hi = mips32_getentryhi ();
  118. mips32_setentryhi (hi);
  119. mips_ehb (); /* mtc0 hazard on tlbp */
  120. mips_tlbp ();
  121. mips_ehb (); /* tlbp hazard on mfc0 */
  122. index = mips32_getindex ();
  123. if (index >= 0)
  124. {
  125. mips_tlbr ();
  126. mips_ehb (); /* tlbr hazard on mfc0 */
  127. *plo0 = mips32_getentrylo0 ();
  128. *plo1 = mips32_getentrylo1 ();
  129. *pmask = mips32_getpagemask ();
  130. }
  131. else
  132. index = -1;
  133. mips32_setentryhi (prev_hi); /* restore EntryHi */
  134. return index;
  135. }
  136. /* Probes the TLB for an entry matching hi, and if present invalidates it */
  137. void __attribute__ ((use_hazard_barrier_return)) _MIPS_HAL_NOMIPS16
  138. mips_tlbinval (tlbhi_t hi)
  139. {
  140. int index, tmp_idx;
  141. tlbhi_t prev_hi, tmp_hi = 0;
  142. register const int zero = 0;
  143. unsigned int cfg;
  144. prev_hi = mips32_getentryhi ();
  145. mips32_setentryhi (hi);
  146. mips_ehb (); /* mtc0 hazard on tlbp */
  147. mips_tlbp ();
  148. mips_ehb (); /* tlbp hazard on mfc0 */
  149. index = mips32_getindex ();
  150. if (index < 0)
  151. goto restore;
  152. mips32_setentrylo0 (zero);
  153. mips32_setentrylo1 (zero);
  154. /* Check if Config4 is implemented */
  155. cfg = mips32_getconfig3 ();
  156. if ((cfg & CFG3_M) != 0)
  157. {
  158. cfg = mips32_getconfig4 ();
  159. if ((cfg & CFG4_IE_MASK) != 0)
  160. {
  161. tmp_hi = C0_ENTRYHI_EHINV_MASK;
  162. goto do_tlbwi;
  163. }
  164. }
  165. tmp_hi = (tlbhi_t) ((unsigned long) KSEG0_BASE - 0x4000);
  166. do
  167. {
  168. tmp_hi = tmp_hi + 0x4000;
  169. mips32_setentryhi (tmp_hi);
  170. mips_ehb (); /* mtc0 hazard on tlbp */
  171. mips_tlbp ();
  172. mips_ehb (); /* tlbp hazard on mfc0 */
  173. tmp_idx = mips32_getindex ();
  174. }
  175. while (tmp_idx >= 0);
  176. mips32_setindex (index); /* restore Index */
  177. do_tlbwi:
  178. mips32_setentryhi (tmp_hi);
  179. mips_ehb (); /* mtc0 hazard on tlbp */
  180. mips_tlbwi ();
  181. mips_ehb (); /* tlbwi hazard on mfc0 */
  182. restore:
  183. mips32_setentryhi (prev_hi); /* restore EntryHi */
  184. return;
  185. }
  186. /*
  187. * Return number of entries and sets in TLB.
  188. * Return number of entries in *pentries and
  189. * sets in *psets
  190. */
  191. static void _MIPS_HAL_NOMIPS16
  192. mips_tlb_entries_sets (int *pentries, int *psets)
  193. {
  194. int entries = 0, sets = 0, ways = 0;
  195. unsigned int cfg, cfg1, tcfg, tmp;
  196. cfg = mips32_getconfig ();
  197. cfg = (cfg & CFG0_MT_MASK) >> CFG0_MT_SHIFT;
  198. if ((cfg == 0) /* No MMU */
  199. || (cfg == (CFG0_MT_FIXED >> CFG0_MT_SHIFT)) /* fixed address translation */
  200. || (cfg == (CFG0_MT_BAT >> CFG0_MT_SHIFT)) /* block address translator */
  201. || ((cfg & ((CFG0_MT_TLB | CFG0_MT_DUAL) >> CFG0_MT_SHIFT)) == 0)) /* presence of TLB */
  202. {
  203. *pentries = 0;
  204. *psets = 0;
  205. return;
  206. }
  207. cfg1 = mips32_getconfig1 ();
  208. /*
  209. * As per PRA, field holds number of entries - 1
  210. * Standard TLBs and dual TLBs have extension fields.
  211. */
  212. entries = ((cfg1 & CFG1_MMUS_MASK) >> CFG1_MMUSSHIFT) + 1;
  213. tcfg = mips32_getconfig3 ();
  214. if ((tcfg & CFG3_M) == 0)
  215. goto doReturn;
  216. tcfg = mips32_getconfig4 ();
  217. #if (__mips_isa_rev < 6)
  218. tmp = (tcfg & CFG4_MMUED) >> CFG4_MMUED_SHIFT;
  219. /* MMU Extension Definition */
  220. if (tmp == (CFG4_MMUED_FTLBVEXT >> CFG4_MMUED_SHIFT))
  221. goto doFTLBVTLB;
  222. /* MMUSizeExt */
  223. if (tmp == (CFG4_MMUED_SIZEEXT >> CFG4_MMUED_SHIFT))
  224. goto doSizeExt;
  225. if (tmp == 0)
  226. goto doReturn;
  227. goto doFTLBSize;
  228. doSizeExt:
  229. entries += ((tcfg & CFG4_MMUSE_MASK) >> CFG4_MMUSE_SHIFT)
  230. << CFG1_MMUS_BITS;
  231. goto doReturn;
  232. #endif
  233. doFTLBVTLB:
  234. entries += ((tcfg & CFG4_VTLBSEXT_MASK) >> CFG4_VTLBSEXT_SHIFT)
  235. << CFG1_MMUS_BITS;
  236. doFTLBSize:
  237. /* Skip FTLB size calculations if Config:MT != 4 */
  238. if (cfg != (CFG0_MT_DUAL >> CFG0_MT_SHIFT))
  239. goto doReturn;
  240. /* Ways */
  241. ways = 2 + ((tcfg & CFG4_FTLBW_MASK) >> CFG4_FTLBW_SHIFT);
  242. /* Sets per way */
  243. tmp = ((tcfg & CFG4_FTLBS_MASK) >> CFG4_FTLBS_SHIFT);
  244. sets = 1 << tmp;
  245. /* Total sets */
  246. entries += ways << tmp;
  247. doReturn:
  248. *pentries = entries;
  249. *psets = sets;
  250. return;
  251. }
  252. /*
  253. * Return number of entries in TLB
  254. * This function is used for both mips64 and mips32
  255. */
  256. int _MIPS_HAL_NOMIPS16
  257. mips_tlb_size (void)
  258. {
  259. int entries = 0, sets = 0;
  260. mips_tlb_entries_sets (&entries, &sets);
  261. (void) sets;
  262. return entries;
  263. }
  264. /*
  265. * Invalidate the whole TLB.
  266. * This function is used for both mips64 and mips32
  267. */
  268. void __attribute__ ((use_hazard_barrier_return)) _MIPS_HAL_NOMIPS16
  269. mips_tlbinvalall (void)
  270. {
  271. unsigned int cfg0, cfg;
  272. unsigned long tmp_hi, tmp_hi2;
  273. int entries = 0, sets = 0, tlb_stride = 0;
  274. int end_ptr = 0, index = 0;
  275. register const unsigned long zero = 0;
  276. extern void *__tlb_stride_length;
  277. cfg0 = mips32_getconfig ();
  278. cfg0 = (cfg0 & CFG0_MT_MASK) >> CFG0_MT_SHIFT;
  279. if ((cfg0 == 0) /* No MMU */
  280. || (cfg0 == (CFG0_MT_FIXED >> CFG0_MT_SHIFT)) /* fixed address translation */
  281. || (cfg0 == (CFG0_MT_BAT >> CFG0_MT_SHIFT))) /* block address translator */
  282. goto doReturn;
  283. mips_setentrylo0 (zero);
  284. mips_setentrylo1 (zero);
  285. mips_setpagemask (zero);
  286. /* Fetch size & number of sets */
  287. mips_tlb_entries_sets (&entries, &sets);
  288. cfg = mips32_getconfig3 ();
  289. if ((cfg & CFG3_M) == 0)
  290. goto doBasicInval;
  291. cfg = mips32_getconfig4 ();
  292. cfg = (cfg & CFG4_IE_MASK) >> CFG4_IE_SHIFT;
  293. /* If Config4[IE] = 0, use old method for invalidation */
  294. if (cfg == 0)
  295. goto doBasicInval;
  296. /* If Config4[IE] = 1, EHINV loop */
  297. if (cfg == (CFG4_IE_EHINV >> CFG4_IE_SHIFT))
  298. goto doEHINV;
  299. /* If Config[MT] = 1, one instruction required */
  300. if (cfg0 == (CFG0_MT_TLB >> CFG0_MT_SHIFT)
  301. || cfg == (CFG4_IE_INVALL >> CFG4_IE_SHIFT))
  302. {
  303. /* TLB walk done by hardware, Config4[IE] = 3 or Config[MT] = 1 */
  304. mips32_setindex (zero);
  305. mips_ehb ();
  306. mips_eva_tlbinvf ();
  307. goto doReturn;
  308. }
  309. /*
  310. * TLB walk done by software, Config4[IE] = 2, Config[MT] = 4
  311. *
  312. * One TLBINVF is executed with an index in VTLB range to
  313. * invalidate all VTLB entries.
  314. *
  315. * One TLBINVF is executed per FTLB set.
  316. *
  317. * We'll clean out the TLB by computing the Size of the VTLB
  318. * but not add the 1. This will give us a finger that points
  319. * at the last VTLB entry.
  320. */
  321. /* Clear VTLB */
  322. mips32_setindex (zero);
  323. mips_ehb ();
  324. mips_eva_tlbinvf ();
  325. tlb_stride = (int) ((unsigned long) &__tlb_stride_length);
  326. sets = sets * tlb_stride;
  327. end_ptr = entries - sets;
  328. do
  329. {
  330. entries = entries - tlb_stride;
  331. mips32_setindex (entries);
  332. mips_ehb ();
  333. mips_eva_tlbinvf ();
  334. }
  335. while (entries != end_ptr);
  336. goto doReturn;
  337. doEHINV:
  338. /*
  339. * Config4[IE] = 1. EHINV supported, but not tlbinvf.
  340. *
  341. * Invalidate the TLB for R3 onwards by loading EHINV and writing to all
  342. * TLB entries.
  343. */
  344. index = 0;
  345. tmp_hi = C0_ENTRYHI_EHINV_MASK;
  346. mips_setentryhi (tmp_hi);
  347. do
  348. {
  349. mips32_setindex (index);
  350. mips_ehb (); /* mtc0 hazard on tlbwi */
  351. mips_tlbwi ();
  352. index++;
  353. }
  354. while (entries != index);
  355. goto doReturn;
  356. doBasicInval:
  357. /*
  358. * Perform a basic invalidation of the TLB for R1 onwards by loading
  359. * 0x(FFFFFFFF)KSEG0_BASE into EntryHi and writing it into index 0
  360. * incrementing by a pagesize, writing into index 1, etc.
  361. * If large physical addressing is enabled, load 0xFFFFFFFF
  362. * into the top half of EntryHi.
  363. */
  364. tmp_hi = 0;
  365. cfg = mips32_getconfig3 ();
  366. /* If XPA is present */
  367. if ((cfg & CFG3_LPA) != 0)
  368. {
  369. cfg = mips32_getpagegrain ();
  370. if ((cfg & PAGEGRAIN_ELPA) == 0)
  371. tmp_hi = 0xFFFFFFFF;
  372. }
  373. tmp_hi2 = (unsigned long) KSEG0_BASE - 0x4000;
  374. index = 0;
  375. do
  376. {
  377. tmp_hi2 += 0x4000;
  378. mips_setentryhi (tmp_hi2);
  379. if (tmp_hi != 0)
  380. mips32_sethientryhi (tmp_hi);
  381. mips_ehb (); /* mtc0 hazard on tlbp */
  382. mips_tlbp ();
  383. mips_ehb (); /* tlbp hazard on mfc0 */
  384. if (mips32_getindex () == 0)
  385. continue;
  386. mips32_setindex (index);
  387. mips_ehb (); /* mtc0 hazard on tlbwi */
  388. mips_tlbwi ();
  389. index++;
  390. }
  391. while (entries != index);
  392. doReturn:
  393. /*
  394. * Clear EntryHI. The upper half is cleared
  395. * autmatically as mtc0 writes zeroes.
  396. */
  397. mips_setentryhi (zero);
  398. return;
  399. }
  400. _MIPS_HAL_NOMIPS16
  401. int m64_tlb_size (void) __attribute__ ((alias ("mips_tlb_size")));
  402. _MIPS_HAL_NOMIPS16
  403. void m64_tlbinvalall (void) __attribute__ ((alias ("mips_tlbinvalall")));