cache.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  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 <rtthread.h>
  16. #include "../common/mipsregs.h"
  17. #define K0BASE 0x80000000
  18. #define PRID_LS1B 0x4220
  19. extern void Clear_TagLo (void);
  20. extern void Invalidate_Icache_Ls1b(unsigned int);
  21. extern void Invalidate_Dcache_ClearTag_Ls1b(unsigned int);
  22. extern void Invalidate_Dcache_Fill_Ls1b(unsigned int);
  23. extern void Writeback_Invalidate_Dcache(unsigned int);
  24. extern void enable_cpu_cache(void);
  25. typedef struct cacheinfo_t
  26. {
  27. unsigned int icache_size;
  28. unsigned int dcache_size;
  29. unsigned int icacheline_size;
  30. unsigned int dcacheline_size;
  31. } cacheinfo_t ;
  32. typedef struct cacheop_t
  33. {
  34. void (*Clear_TagLo) (void);
  35. void (*Invalidate_Icache) (unsigned int);
  36. void (*Invalidate_Dcache_Fill) (unsigned int);
  37. void (*Invalidate_Dcache_ClearTag) (unsigned int);
  38. void (*Init_Cache)(void);
  39. } cacheop_t ;
  40. static cacheop_t cacheop, *pcacheop;
  41. static cacheinfo_t cacheinfo, *pcacheinfo;
  42. int identify_cpu(void)
  43. {
  44. unsigned int cpu_id;
  45. pcacheop = &cacheop;
  46. pcacheinfo = &cacheinfo;
  47. rt_kprintf("CPU configure: 0x%08x\n", read_c0_config());
  48. cpu_id = read_c0_prid();
  49. switch (cpu_id)
  50. {
  51. case PRID_LS1B:
  52. rt_kprintf("CPU:LS1B\n");
  53. pcacheop->Clear_TagLo = Clear_TagLo;
  54. pcacheop->Invalidate_Icache = Invalidate_Icache_Ls1b;
  55. pcacheop->Invalidate_Dcache_Fill = Invalidate_Dcache_Fill_Ls1b;
  56. pcacheop->Invalidate_Dcache_ClearTag = Invalidate_Dcache_ClearTag_Ls1b;
  57. break;
  58. default:
  59. rt_kprintf("Unknown CPU type, system halted!\n");
  60. while (1)
  61. {
  62. ;
  63. }
  64. break;
  65. }
  66. return 0;
  67. }
  68. void probe_cache(void)
  69. {
  70. unsigned int config1 = read_c0_config1();
  71. unsigned int icache_size, icache_line_size, icache_sets, icache_ways;
  72. unsigned int dcache_size, dcache_line_size, dcache_sets, dcache_ways;
  73. if ((icache_line_size = ((config1 >> 19) & 7)))
  74. icache_line_size = 2 << icache_line_size;
  75. else
  76. icache_line_size = icache_line_size;
  77. icache_sets = 64 << ((config1 >> 22) & 7);
  78. icache_ways = 1 + ((config1 >> 16) & 7);
  79. icache_size = icache_sets * icache_ways * icache_line_size;
  80. if ((dcache_line_size = ((config1 >> 10) & 7)))
  81. dcache_line_size = 2 << dcache_line_size;
  82. else
  83. dcache_line_size = dcache_line_size;
  84. dcache_sets = 64 << ((config1 >> 13) & 7);
  85. dcache_ways = 1 + ((config1 >> 7) & 7);
  86. dcache_size = dcache_sets * dcache_ways * dcache_line_size;
  87. rt_kprintf("DCache %2dkb, linesize %d bytes.\n", dcache_size >> 10, dcache_line_size);
  88. rt_kprintf("ICache %2dkb, linesize %d bytes.\n", icache_size >> 10, icache_line_size);
  89. pcacheinfo->icache_size = icache_size;
  90. pcacheinfo->dcache_size = dcache_size;
  91. pcacheinfo->icacheline_size = icache_line_size;
  92. pcacheinfo->dcacheline_size = dcache_line_size;
  93. return ;
  94. }
  95. void invalidate_writeback_dcache_all(void)
  96. {
  97. unsigned int start = K0BASE;
  98. unsigned int end = (start + pcacheinfo->dcache_size);
  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(void)
  127. {
  128. unsigned int start = K0BASE;
  129. unsigned int end = (start + pcacheinfo->dcache_size);
  130. while (start <end)
  131. {
  132. Invalidate_Dcache_Fill_Ls1b(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. }