cache.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. /*
  2. * File : cache.c
  3. * This file is part of RT-Thread RTOS
  4. * COPYRIGHT (C) 2006 - 2011, RT-Thread Development Team
  5. *
  6. * The license and distribution terms for this file may be
  7. * found in the file LICENSE in this distribution or at
  8. * http://www.rt-thread.org/license/LICENSE
  9. *
  10. * Change Logs:
  11. * Date Author Notes
  12. * 2010-07-09 Bernard first version
  13. * 2011-08-08 lgnq modified for LS1B
  14. */
  15. #include "../common/mipsregs.h"
  16. #define K0BASE 0x80000000
  17. #define PRID_LS1B 0x4220
  18. extern void Clear_TagLo (void);
  19. extern void Invalidate_Icache_Ls1b(unsigned int);
  20. extern void Invalidate_Dcache_ClearTag_Ls1b(unsigned int);
  21. extern void Invalidate_Dcache_Fill_Ls1b(unsigned int);
  22. extern void Writeback_Invalidate_Dcache(unsigned int);
  23. typedef struct cacheinfo_t
  24. {
  25. unsigned int icache_size;
  26. unsigned int dcache_size;
  27. unsigned int icacheline_size;
  28. unsigned int dcacheline_size;
  29. } cacheinfo_t ;
  30. typedef struct cacheop_t
  31. {
  32. void (*Clear_TagLo) (void);
  33. void (*Invalidate_Icache) (unsigned int);
  34. void (*Invalidate_Dcache_Fill) (unsigned int);
  35. void (*Invalidate_Dcache_ClearTag) (unsigned int);
  36. void (*Init_Cache)(void);
  37. } cacheop_t ;
  38. static cacheop_t cacheop, *pcacheop;
  39. static cacheinfo_t cacheinfo, *pcacheinfo;
  40. int identify_cpu(void)
  41. {
  42. unsigned int cpu_id;
  43. pcacheop = &cacheop;
  44. pcacheinfo = &cacheinfo;
  45. rt_kprintf("CPU configure: 0x%08x\n", read_c0_config());
  46. cpu_id = read_c0_prid();
  47. switch (cpu_id)
  48. {
  49. case PRID_LS1B:
  50. rt_kprintf("CPU:LS1B\n");
  51. pcacheop->Clear_TagLo = Clear_TagLo;
  52. pcacheop->Invalidate_Icache = Invalidate_Icache_Ls1b;
  53. pcacheop->Invalidate_Dcache_Fill = Invalidate_Dcache_Fill_Ls1b;
  54. pcacheop->Invalidate_Dcache_ClearTag = Invalidate_Dcache_ClearTag_Ls1b;
  55. break;
  56. default:
  57. rt_kprintf("Unknown CPU type, system halted!\n");
  58. while (1)
  59. {
  60. ;
  61. }
  62. break;
  63. }
  64. return 0;
  65. }
  66. void probe_cache(void)
  67. {
  68. unsigned int config1 = read_c0_config1();
  69. unsigned int icache_size, icache_line_size, icache_sets, icache_ways;
  70. unsigned int dcache_size, dcache_line_size, dcache_sets, dcache_ways;
  71. if ((icache_line_size = ((config1 >> 19) & 7)))
  72. icache_line_size = 2 << icache_line_size;
  73. else
  74. icache_line_size = icache_line_size;
  75. icache_sets = 64 << ((config1 >> 22) & 7);
  76. icache_ways = 1 + ((config1 >> 16) & 7);
  77. icache_size = icache_sets * icache_ways * icache_line_size;
  78. if ((dcache_line_size = ((config1 >> 10) & 7)))
  79. dcache_line_size = 2 << dcache_line_size;
  80. else
  81. dcache_line_size = dcache_line_size;
  82. dcache_sets = 64 << ((config1 >> 13) & 7);
  83. dcache_ways = 1 + ((config1 >> 7) & 7);
  84. dcache_size = dcache_sets * dcache_ways * dcache_line_size;
  85. rt_kprintf("DCache %2dkb, linesize %d bytes.\n", dcache_size >> 10, dcache_line_size);
  86. rt_kprintf("ICache %2dkb, linesize %d bytes.\n", icache_size >> 10, icache_line_size);
  87. pcacheinfo->icache_size = icache_size;
  88. pcacheinfo->dcache_size = dcache_size;
  89. pcacheinfo->icacheline_size = icache_line_size;
  90. pcacheinfo->dcacheline_size = dcache_line_size;
  91. return ;
  92. }
  93. void invalidate_writeback_dcache_all(void)
  94. {
  95. unsigned int start = K0BASE;
  96. unsigned int end = (start + pcacheinfo->dcache_size);
  97. while (start < end)
  98. {
  99. Writeback_Invalidate_Dcache(start); //hit writeback invalidate
  100. start += pcacheinfo->dcacheline_size;
  101. }
  102. }
  103. void invalidate_writeback_dcache(unsigned long addr, int size)
  104. {
  105. unsigned long start, end;
  106. start = (addr +pcacheinfo->dcacheline_size -1) & (- pcacheinfo->dcacheline_size);
  107. end = (end + size + pcacheinfo->dcacheline_size -1) & ( -pcacheinfo->dcacheline_size);
  108. while (start <end)
  109. {
  110. Writeback_Invalidate_Dcache(start);
  111. start += pcacheinfo->dcacheline_size;
  112. }
  113. }
  114. void invalidate_icache_all(void)
  115. {
  116. unsigned int start = K0BASE;
  117. unsigned int end = (start + pcacheinfo->icache_size);
  118. while (start < end)
  119. {
  120. pcacheop->Invalidate_Icache(start);
  121. start += pcacheinfo->icacheline_size;
  122. }
  123. }
  124. void invalidate_dcache_all(void)
  125. {
  126. unsigned int start = K0BASE;
  127. unsigned int end = (start + pcacheinfo->dcache_size);
  128. while (start <end)
  129. {
  130. Invalidate_Dcache_Fill_Gc3210I(start);
  131. start += pcacheinfo->icacheline_size;
  132. }
  133. }
  134. //with cache disabled
  135. void init_dcache(void)
  136. {
  137. unsigned int start = K0BASE;
  138. unsigned int end = (start + pcacheinfo->dcache_size);
  139. while (start < end)
  140. {
  141. pcacheop->Invalidate_Dcache_ClearTag(start);
  142. start += pcacheinfo->dcacheline_size;
  143. }
  144. }
  145. void rt_hw_cache_init(void)
  146. {
  147. unsigned int start, end;
  148. /* 1. identify cpu and probe cache */
  149. identify_cpu();
  150. probe_cache();
  151. start = K0BASE;
  152. end = (start + pcacheinfo->icache_size);
  153. /*
  154. * 2. clear CP0 taglo/taghi register;
  155. */
  156. pcacheop->Clear_TagLo();
  157. /*
  158. * 3. invalidate instruction cache;
  159. */
  160. while (start < end)
  161. {
  162. pcacheop->Invalidate_Icache(start); //index invalidate icache
  163. start += pcacheinfo->icacheline_size;
  164. }
  165. /*
  166. * 4. invalidate data cache;
  167. */
  168. start = K0BASE;
  169. end = (start + pcacheinfo->dcache_size);
  170. while(start < end)
  171. {
  172. pcacheop->Invalidate_Dcache_ClearTag(start);
  173. start += pcacheinfo->dcacheline_size;
  174. }
  175. start = K0BASE;
  176. while(start < end)
  177. {
  178. pcacheop->Invalidate_Dcache_Fill(start); //index invalidate dcache
  179. start += pcacheinfo->dcacheline_size;
  180. }
  181. start = K0BASE;
  182. while(start < end)
  183. {
  184. pcacheop->Invalidate_Dcache_ClearTag(start);
  185. start += pcacheinfo->dcacheline_size;
  186. }
  187. /* enable cache */
  188. enable_cpu_cache();
  189. rt_kprintf("enable cpu cache done\n");
  190. return ;
  191. }