cache.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  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. void invalidate_cache (void);
  44. pcacheop = &cacheop;
  45. pcacheinfo = &cacheinfo;
  46. rt_kprintf("CPU configure: 0x%08x\n", read_c0_config());
  47. cpu_id = read_c0_prid();
  48. switch (cpu_id)
  49. {
  50. case PRID_LS1B:
  51. rt_kprintf("CPU:LS1B\n");
  52. pcacheop->Clear_TagLo = Clear_TagLo;
  53. pcacheop->Invalidate_Icache = Invalidate_Icache_Ls1b;
  54. pcacheop->Invalidate_Dcache_Fill = Invalidate_Dcache_Fill_Ls1b;
  55. pcacheop->Invalidate_Dcache_ClearTag = Invalidate_Dcache_ClearTag_Ls1b;
  56. break;
  57. default:
  58. rt_kprintf("Unknown CPU type, system halted!\n");
  59. while (1)
  60. {
  61. ;
  62. }
  63. break;
  64. }
  65. return 0;
  66. }
  67. void probe_cache(void)
  68. {
  69. unsigned int config1 = read_c0_config1();
  70. unsigned int icache_size, icache_line_size, icache_sets, icache_ways;
  71. unsigned int dcache_size, dcache_line_size, dcache_sets, dcache_ways;
  72. if ((icache_line_size = ((config1 >> 19) & 7)))
  73. icache_line_size = 2 << icache_line_size;
  74. else
  75. icache_line_size = icache_line_size;
  76. icache_sets = 64 << ((config1 >> 22) & 7);
  77. icache_ways = 1 + ((config1 >> 16) & 7);
  78. icache_size = icache_sets * icache_ways * icache_line_size;
  79. if ((dcache_line_size = ((config1 >> 10) & 7)))
  80. dcache_line_size = 2 << dcache_line_size;
  81. else
  82. dcache_line_size = dcache_line_size;
  83. dcache_sets = 64 << ((config1 >> 13) & 7);
  84. dcache_ways = 1 + ((config1 >> 7) & 7);
  85. dcache_size = dcache_sets * dcache_ways * dcache_line_size;
  86. rt_kprintf("DCache %2dkb, linesize %d bytes.\n", dcache_size >> 10, dcache_line_size);
  87. rt_kprintf("ICache %2dkb, linesize %d bytes.\n", icache_size >> 10, icache_line_size);
  88. pcacheinfo->icache_size = icache_size;
  89. pcacheinfo->dcache_size = dcache_size;
  90. pcacheinfo->icacheline_size = icache_line_size;
  91. pcacheinfo->dcacheline_size = dcache_line_size;
  92. return ;
  93. }
  94. void invalidate_writeback_dcache_all(void)
  95. {
  96. unsigned int start = K0BASE;
  97. unsigned int end = (start + pcacheinfo->dcache_size);
  98. start = K0BASE;
  99. while(start < end)
  100. {
  101. Writeback_Invalidate_Dcache(start); //hit writeback invalidate
  102. start += pcacheinfo->dcacheline_size;
  103. }
  104. }
  105. void invalidate_writeback_dcache(unsigned long addr, int size)
  106. {
  107. unsigned long start, end;
  108. start = (addr +pcacheinfo->dcacheline_size -1) & (- pcacheinfo->dcacheline_size);
  109. end = (end + size + pcacheinfo->dcacheline_size -1) & ( -pcacheinfo->dcacheline_size);
  110. while(start <end)
  111. {
  112. Writeback_Invalidate_Dcache(start);
  113. start += pcacheinfo->dcacheline_size;
  114. }
  115. }
  116. void invalidate_icache_all(void)
  117. {
  118. unsigned int start = K0BASE;
  119. unsigned int end = (start + pcacheinfo->icache_size);
  120. while(start < end)
  121. {
  122. pcacheop->Invalidate_Icache(start);
  123. start += pcacheinfo->icacheline_size;
  124. }
  125. }
  126. void invalidate_dcache_all()
  127. {
  128. unsigned int start = K0BASE;
  129. unsigned int end = (start + pcacheinfo->dcache_size);
  130. while(start <end)
  131. {
  132. Invalidate_Dcache_Fill_Gc3210I(start);
  133. start += pcacheinfo->icacheline_size;
  134. }
  135. }
  136. //with cache disabled
  137. void init_dcache(void)
  138. {
  139. unsigned int start = K0BASE;
  140. unsigned int end = (start + pcacheinfo->dcache_size);
  141. while(start < end)
  142. {
  143. pcacheop->Invalidate_Dcache_ClearTag(start);
  144. start += pcacheinfo->dcacheline_size;
  145. }
  146. }
  147. void rt_hw_cache_init(void)
  148. {
  149. unsigned int start, end;
  150. /* 1. identify cpu and probe cache */
  151. identify_cpu();
  152. probe_cache();
  153. start = K0BASE;
  154. end = (start + pcacheinfo->icache_size);
  155. /*
  156. * 2. clear CP0 taglo/taghi register;
  157. */
  158. pcacheop->Clear_TagLo();
  159. /*
  160. * 3. invalidate instruction cache;
  161. */
  162. while(start < end)
  163. {
  164. pcacheop->Invalidate_Icache(start); //index invalidate icache
  165. start += pcacheinfo->icacheline_size;
  166. }
  167. /*
  168. * 4. invalidate data cache;
  169. */
  170. start = K0BASE;
  171. end = (start + pcacheinfo->dcache_size);
  172. while(start < end)
  173. {
  174. pcacheop->Invalidate_Dcache_ClearTag(start);
  175. start += pcacheinfo->dcacheline_size;
  176. }
  177. start = K0BASE;
  178. while(start < end)
  179. {
  180. pcacheop->Invalidate_Dcache_Fill(start); //index invalidate dcache
  181. start += pcacheinfo->dcacheline_size;
  182. }
  183. start = K0BASE;
  184. while(start < end)
  185. {
  186. pcacheop->Invalidate_Dcache_ClearTag(start);
  187. start += pcacheinfo->dcacheline_size;
  188. }
  189. /* enable cache */
  190. enable_cpu_cache();
  191. rt_kprintf("enable cpu cache done\n");
  192. return ;
  193. }