1
0

mipscfg.c 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. /*
  2. * File : mipscfg.c
  3. * COPYRIGHT (C) 2008 - 2016, RT-Thread Development Team
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation; either version 2 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License along
  16. * with this program; if not, write to the Free Software Foundation, Inc.,
  17. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  18. *
  19. * Change Logs:
  20. * Date Author Notes
  21. * 2010-05-27 swkyer first version
  22. */
  23. #include <rtthread.h>
  24. #include "../common/mipsregs.h"
  25. #include "../common/mipscfg.h"
  26. mips32_core_cfg_t g_mips_core =
  27. {
  28. 16, /* icache_line_size */
  29. 256, /* icache_lines_per_way */
  30. 4, /* icache_ways */
  31. 16, /* dcache_line_size */
  32. 256, /* dcache_lines_per_way */
  33. 4, /* dcache_ways */
  34. 16, /* max_tlb_entries */
  35. };
  36. static rt_uint16_t m_pow(rt_uint16_t b, rt_uint16_t n)
  37. {
  38. rt_uint16_t rets = 1;
  39. while (n--)
  40. rets *= b;
  41. return rets;
  42. }
  43. /**
  44. * read core attribute
  45. */
  46. void mips32_cfg_init(void)
  47. {
  48. rt_uint16_t val;
  49. rt_uint32_t cp0_config1;
  50. cp0_config1 = read_c0_config();
  51. if (cp0_config1 & 0x80000000)
  52. {
  53. cp0_config1 = read_c0_config1();
  54. val = (cp0_config1 & (7<<22))>>22;
  55. g_mips_core.icache_lines_per_way = 64 * m_pow(2, val);
  56. val = (cp0_config1 & (7<<19))>>19;
  57. g_mips_core.icache_line_size = 2 * m_pow(2, val);
  58. val = (cp0_config1 & (7<<16))>>16;
  59. g_mips_core.icache_ways = val + 1;
  60. val = (cp0_config1 & (7<<13))>>13;
  61. g_mips_core.dcache_lines_per_way = 64 * m_pow(2, val);
  62. val = (cp0_config1 & (7<<10))>>10;
  63. g_mips_core.dcache_line_size = 2 * m_pow(2, val);
  64. val = (cp0_config1 & (7<<7))>>7;
  65. g_mips_core.dcache_ways = val + 1;
  66. val = (cp0_config1 & (0x3F<<25))>>25;
  67. g_mips_core.max_tlb_entries = val + 1;
  68. }
  69. }
  70. #ifdef RT_USING_FINSH
  71. #include <finsh.h>
  72. static void CP0_status_analyze(unsigned long value)
  73. {
  74. if(value & (1<<26))
  75. rt_kprintf(" FR");
  76. if(value & (1<<23))
  77. rt_kprintf(" PX");
  78. if(value & (1<<22))
  79. rt_kprintf(" BEV");
  80. if(value & (1<<20))
  81. rt_kprintf(" SR");
  82. if(value & (1<<19))
  83. rt_kprintf(" NMI");
  84. if(value & (1<<20))
  85. rt_kprintf(" SR");
  86. if(value & (0xFF<<8))
  87. rt_kprintf(" IM:0x%02X", (value >> 8) & 0xFF);
  88. if(value & (1<<7))
  89. rt_kprintf(" KX");
  90. if(value & (1<<6))
  91. rt_kprintf(" SX");
  92. if(value & (1<<5))
  93. rt_kprintf(" UX");
  94. if(value & (0x03<<3))
  95. rt_kprintf(" KSU:0x%02X", (value >> 3) & 0x03);
  96. if(value & (1<<2))
  97. rt_kprintf(" ERL");
  98. if(value & (1<<1))
  99. rt_kprintf(" EXL");
  100. if(value & (1<<0))
  101. rt_kprintf(" IE");
  102. }
  103. static void CP0_config0_analyze(unsigned long value)
  104. {
  105. /* [31] M */
  106. if(value & (1UL<<31))
  107. rt_kprintf(" M");
  108. /* [15] BE */
  109. if(value & (1<<15))
  110. rt_kprintf(" big-endian");
  111. else
  112. rt_kprintf(" little-endian");
  113. /* [14:13] AT */
  114. {
  115. int AT = (value >> 13) & 0x03;
  116. if(AT == 0)
  117. {
  118. rt_kprintf(" MIPS32");
  119. }
  120. else if(AT == 1)
  121. {
  122. rt_kprintf(" MIPS64/A32");
  123. }
  124. else if(AT == 2)
  125. {
  126. rt_kprintf(" MIPS64/A64");
  127. }
  128. else
  129. {
  130. rt_kprintf(" unkown");
  131. }
  132. }
  133. /* [12:10] AR */
  134. {
  135. int AR = (value >> 10) & 0x07;
  136. if(AR == 0)
  137. {
  138. rt_kprintf(" R1");
  139. }
  140. else if(AR == 1)
  141. {
  142. rt_kprintf(" R2");
  143. }
  144. else
  145. {
  146. rt_kprintf(" reserve");
  147. }
  148. }
  149. /* [3] VI */
  150. if(value & (1UL<<31))
  151. rt_kprintf(" VI");
  152. /* [2:0] K0 */
  153. {
  154. int K0 = value & 0x07;
  155. if(K0 == 2)
  156. {
  157. rt_kprintf(" uncached");
  158. }
  159. else if(K0 == 3)
  160. {
  161. rt_kprintf(" cacheable");
  162. }
  163. else
  164. {
  165. rt_kprintf(" K0:reserve");
  166. }
  167. }
  168. }
  169. static void CP0_config1_analyze(unsigned long value)
  170. {
  171. /* [31] M */
  172. if(value & (1UL<<31))
  173. rt_kprintf(" M");
  174. /* [30:25] MMU size */
  175. {
  176. int MMU_size = (value >> 25) & 0x3F;
  177. rt_kprintf(" TLB:%d", MMU_size + 1);
  178. }
  179. /* [24:22] IS, [21:19] IL, [18:16] IA */
  180. {
  181. int IS = (value >> 22) & 0x07;
  182. int IL = (value >> 19) & 0x07;
  183. int IA = (value >> 16) & 0x07;
  184. IA = IA + 1;
  185. IS = 64 << IS;
  186. IL = 2 << IL;
  187. rt_kprintf(" Icache-%dKB:%dway*%dset*%dbyte",
  188. (IA*IS*IL) >> 10, IA, IS, IL);
  189. }
  190. /* [15:13] DS, [12:10] DL, [9:7] DA */
  191. {
  192. int DS = (value >> 13) & 0x07;
  193. int DL = (value >> 10) & 0x07;
  194. int DA = (value >> 7) & 0x07;
  195. DA = DA + 1;
  196. DS = 64 << DS;
  197. DL = 2 << DL;
  198. rt_kprintf(" Dcache-%dKB:%dway*%dset*%dbyte",
  199. (DA*DS*DL) >> 10, DA, DS, DL);
  200. }
  201. /* [6] C2 */
  202. if(value & (1UL<<6))
  203. rt_kprintf(" CP2");
  204. /* [5] MD */
  205. if(value & (1UL<<5))
  206. rt_kprintf(" MDMX-ASE");
  207. /* [4] PC */
  208. if(value & (1UL<<4))
  209. rt_kprintf(" performa-count");
  210. /* [3] WR */
  211. if(value & (1UL<<3))
  212. rt_kprintf(" Watch");
  213. /* [2] CA */
  214. if(value & (1UL<<2))
  215. rt_kprintf(" MIPS16e");
  216. /* [1] EP */
  217. if(value & (1UL<<1))
  218. rt_kprintf(" EJTAG");
  219. /* [0] FP */
  220. if(value & (1UL<<0))
  221. rt_kprintf(" FPU");
  222. }
  223. static void CP0_config2_analyze(unsigned long value)
  224. {
  225. /* [31] M */
  226. if(value & (1UL<<31))
  227. rt_kprintf(" M");
  228. }
  229. static void CP0_config3_analyze(unsigned long value)
  230. {
  231. /* [31] M */
  232. if(value & (1UL<<31))
  233. rt_kprintf(" M");
  234. }
  235. static void list_mips(void)
  236. {
  237. unsigned long value;
  238. unsigned long num = 0;
  239. rt_kprintf("MIPS coprocessor register:\r\n");
  240. rt_kprintf("( 0,0) INDEX : 0x%08X\r\n", read_c0_index());
  241. rt_kprintf("( 1,0) RANDOM : 0x%08X\r\n", read_c0_random());
  242. rt_kprintf("( 2,0) ENTRYLO0 : 0x%08X\r\n", read_c0_entrylo0());
  243. rt_kprintf("( 3,0) ENTRYLO1 : 0x%08X\r\n", read_c0_entrylo1());
  244. rt_kprintf("( 4,0) CONTEXT : 0x%08X\r\n", read_c0_context());
  245. rt_kprintf("( 5,0) PAGEMASK : 0x%08X\r\n", read_c0_pagemask());
  246. rt_kprintf("( 6,0) WIRED : 0x%08X\r\n", read_c0_wired());
  247. rt_kprintf("( 7,0) INFO : 0x%08X\r\n", read_c0_info());
  248. rt_kprintf("( 8,0) BADVADDR : 0x%08X\r\n", read_c0_badvaddr());
  249. rt_kprintf("( 9,0) COUNT : 0x%08X\r\n", read_c0_count());
  250. rt_kprintf("(10,0) ENTRYHI : 0x%08X\r\n", read_c0_entryhi());
  251. rt_kprintf("(11,0) COMPARE : 0x%08X\r\n", read_c0_compare());
  252. value = read_c0_status();
  253. rt_kprintf("(12,0) STATUS : 0x%08X", value);
  254. CP0_status_analyze(value);
  255. rt_kprintf("\r\n");
  256. /*
  257. rt_kprintf("(12,1) INTCTL : 0x%08X\r\n", __read_32bit_c0_register(12, 1));
  258. rt_kprintf("(12,2) SRSCTL : 0x%08X\r\n", __read_32bit_c0_register(12, 2));
  259. */
  260. rt_kprintf("(13,0) CAUSE : 0x%08X\r\n", read_c0_cause());
  261. rt_kprintf("(14,0) EPC : 0x%08X\r\n", read_c0_epc());
  262. rt_kprintf("(15,0) PRID : 0x%08X\r\n", read_c0_prid());
  263. rt_kprintf("(15,1) EBASE : 0x%08X\r\n", read_c0_ebase());
  264. value = read_c0_config();
  265. rt_kprintf("(16,0) CONFIG : 0x%08X", value);
  266. CP0_config0_analyze(value);
  267. rt_kprintf("\r\n");
  268. if(value & (1UL << 31))
  269. {
  270. value = read_c0_config1();
  271. rt_kprintf("(16,1) CONFIG1 : 0x%08X", value);
  272. CP0_config1_analyze(value);
  273. rt_kprintf("\r\n");
  274. if(value & (1UL << 31))
  275. {
  276. value = read_c0_config2();
  277. rt_kprintf("(16,2) CONFIG2 : 0x%08X\r\n", value);
  278. CP0_config2_analyze(value);
  279. rt_kprintf("\r\n");
  280. if(value & (1UL << 31))
  281. {
  282. value = read_c0_config3();
  283. rt_kprintf("(16,3) CONFIG3 : 0x%08X\r\n", value);
  284. CP0_config3_analyze(value);
  285. rt_kprintf("\r\n");
  286. }
  287. }
  288. }
  289. rt_kprintf("(17,0) LLADDR : 0x%08X\r\n", __read_32bit_c0_register($17, 0));
  290. rt_kprintf("(18,0) WATCHLO : 0x%08X\r\n", __read_32bit_c0_register($18, 0));
  291. rt_kprintf("(19,0) WATCHHI : 0x%08X\r\n", __read_32bit_c0_register($19, 0));
  292. rt_kprintf("(20,0) XCONTEXT : 0x%08X\r\n", __read_32bit_c0_register($20, 0));
  293. rt_kprintf("(21,0) FRAMEMASK : 0x%08X\r\n", __read_32bit_c0_register($21, 0));
  294. rt_kprintf("(22,0) DIAGNOSTIC: 0x%08X\r\n", __read_32bit_c0_register($22, 0));
  295. rt_kprintf("(23,0) DEBUG : 0x%08X\r\n", __read_32bit_c0_register($23, 0));
  296. rt_kprintf("(24,0) DEPC : 0x%08X\r\n", __read_32bit_c0_register($24, 0));
  297. rt_kprintf("(25,0) PERFCTL0 : 0x%08X\r\n", __read_32bit_c0_register($25, 0));
  298. rt_kprintf("(26,0) ECC : 0x%08X\r\n", __read_32bit_c0_register($26, 0));
  299. rt_kprintf("(27,0) CACHEERR : 0x%08X\r\n", __read_32bit_c0_register($27, 0));
  300. rt_kprintf("(28,0) TAGLO : 0x%08X\r\n", __read_32bit_c0_register($28, 0));
  301. rt_kprintf("(29,0) TAGHI : 0x%08X\r\n", __read_32bit_c0_register($29, 0));
  302. /*
  303. rt_kprintf("(30,0) ERROREPC : 0x%08X\r\n", __read_32bit_c0_register($30, 0));
  304. rt_kprintf("(31,0) DESAVE : 0x%08X\r\n", __read_32bit_c0_register($31, 0));
  305. */
  306. rt_kprintf("\r\n");
  307. }
  308. FINSH_FUNCTION_EXPORT(list_mips, list CPU info)
  309. #endif /* RT_USING_FINSH */