init_vpe1.S 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. /*
  2. * init_vpe1.S
  3. *
  4. * Initialize the second vpe and additional TCs
  5. */
  6. /*
  7. Copyright (c) 2007-2018, MIPS Tech, LLC and/or its affiliated group companies or licensors
  8. All rights reserved.
  9. Redistribution and use in source and binary forms, with or without modification, are
  10. permitted provided that the following conditions are met:
  11. 1. Redistributions of source code must retain the above copyright notice, this list of
  12. conditions and the following disclaimer.
  13. 2. Redistributions in binary form must reproduce the above copyright notice, this list
  14. of conditions and the following disclaimer in the documentation and/or other materials
  15. provided with the distribution.
  16. 3. Neither the name of the copyright holder nor the names of its contributors may be
  17. used to endorse or promote products derived from this software without specific prior
  18. written permission.
  19. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
  20. EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  21. OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
  22. SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  23. SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
  24. OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  25. HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  26. OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  27. SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. */
  29. #include <mips/asm.h>
  30. #include <boot.h>
  31. #include <mips/mt.h>
  32. #include <mips/regdef.h>
  33. #include <mips/m32c0.h>
  34. #define target_TC a3 // will hold the current TC being configured
  35. #define VPE_1 1
  36. .set noat
  37. /**************************************************************************************
  38. **************************************************************************************/
  39. LEAF(init_vpe1)
  40. // Each vpe will need to set up additional TC bound to it. (No rebinding.)
  41. beqz r21_more_tcs, done_init_vpe1 // If there is no more TCs then return
  42. beqz r20_more_vpes, done_init_vpe1 // If there is no vpe1 then return
  43. // This is executing on TC0 bound to VPE0. Therefore VPEConf0.MVP is already set.
  44. // Enable Virtual Processor Configuration to enter configuration mode
  45. mfc0 a0, C0_MVPCONTROL // read C0_MVPCtl
  46. or a0, (1 << 1) // set VPC to enable Virtual Processor Configuration
  47. mtc0 a0, C0_MVPCONTROL // write C0_MVPCtl
  48. ehb
  49. // Initialize target_TC, target_TC will be incremented at the
  50. // bottom of the loop if there are more TCs
  51. li target_TC, 1
  52. nexttc:
  53. // Set TargTC in the CP0 VPECONTROL register
  54. // TargTC Selects the TC number of the "other thread context" for any
  55. // Move to Thread Context or Move from Thread Context instructions
  56. // (any instructions that begin with mtt or mft)
  57. mfc0 a0, C0_VPECONTROL // read C0_VPECTL
  58. ins a0, target_TC, 0, 8 // insert TargTC
  59. mtc0 a0, C0_VPECONTROL // write C0_VPECTL
  60. ehb
  61. // Halt target_TC being configured
  62. li a0, 1 // set Halt bit
  63. mttc0 a0, C0_TCHALT // write C0_TCHALT
  64. ehb
  65. // Set up TCStatus register:
  66. // Disable Coprocessor Usable bits
  67. // Disable MDMX/DSP ASE
  68. // Clear Dirty target_TC
  69. // not dynamically allocatable
  70. // not allocated
  71. // Kernel mode
  72. // interrupt exempt
  73. // ASID 0
  74. // NOTE: Only bit that needs to be set is IXMT
  75. li a0, (1 << 10) // set IXMT
  76. mttc0 a0, C0_TCSTATUS // write C0_TCSTATUS of target TC
  77. // Initialize the target_TC's register file (NOTE: $2 is in the gpr of the tc executing this code)
  78. // NOTE: Good practice but not programmatically necessary
  79. li $2, 0xdeadbeef
  80. mttgpr $2, $1
  81. mttgpr $2, $2
  82. mttgpr $2, $3
  83. mttgpr $2, $4
  84. mttgpr $2, $5
  85. mttgpr $2, $6
  86. mttgpr $2, $7
  87. mttgpr $2, $8
  88. mttgpr $2, $9
  89. mttgpr $2, $10
  90. mttgpr $2, $11
  91. mttgpr $2, $12
  92. mttgpr $2, $13
  93. mttgpr $2, $14
  94. mttgpr $2, $15
  95. mttgpr $2, $16
  96. mttgpr $2, $17
  97. mttgpr $2, $18
  98. mttgpr $2, $19
  99. mttgpr $2, $20
  100. mttgpr $2, $21
  101. mttgpr $2, $22
  102. mttgpr $2, $23
  103. mttgpr $2, $24
  104. mttgpr $2, $25
  105. mttgpr $2, $26
  106. mttgpr $2, $27
  107. mttgpr $2, $28
  108. mttgpr $2, $29
  109. mttgpr $2, $30
  110. mttgpr $2, $31
  111. // Bind TC1 and all other remaining TCs to VPE1 context
  112. // All mftc0 and mttc0 instructions
  113. // will then operate on the VPE1 CP0 registers
  114. li a0, VPE_1
  115. mftc0 a1, C0_TCBIND // Read C0_TCBind
  116. ins a1, a0, 0, 4 // insert vpe 1 into CurVPE field
  117. mttc0 a1, C0_TCBIND // write C0_TCBind
  118. // Must only do next part up to check_for_more_TC label once
  119. bne a0, target_TC, check_for_more_TC // branch if not TC1 NOTE: a0 set to 1 above
  120. // Set XTC for target_TC (sets TC1 to be the only TC runnable on the VPE1)
  121. mftc0 a0, C0_VPECONF0 // read C0_VPECONF0
  122. ins a0, target_TC, 21, 8 // insert XTC
  123. mttc0 a0, C0_VPECONF0 // write C0_VPECONF0
  124. // Disable multi-threading for VPE1
  125. mftc0 a0, C0_VPECONTROL // read C0_VPECTL
  126. ins a0, zero, 15, 1 // clear TE (only tc1 can execute code)
  127. mttc0 a0, C0_VPECONTROL // write C0_VPECTL
  128. // for VPE1 (Just Clear VPA to prevent any TC bound to it from executing
  129. // and set master VPE so CP0 registers are writable
  130. mftc0 a0, C0_VPECONF0 // read C0_VPECONF0
  131. ins a0, zero, 0, 1 // clear VPA
  132. or a0, (1 << 1) // set MVP
  133. mttc0 a0, C0_VPECONF0 // write C0_VPECONF0
  134. mfc0 a0, C0_STATUS // read C0_STATUS
  135. mttc0 a0, C0_STATUS // write C0_Status
  136. li a0, 0x12345678
  137. mttc0 a0, C0_EPC // write C0_EPC
  138. mttc0 zero, C0_CAUSE // write C0_CAUSE
  139. mfc0 a0, C0_CONFIG // read VPE 0 C0_CONFIG
  140. mttc0 a0, C0_CONFIG // write it to VPE 1 C0_CONFIG
  141. mftc0 a0, C0_EBASE // read C0_EBASE
  142. ext a0, a0, 0, 10 // extract CPUNum
  143. mttgpr a0, r23_cpu_num // write CPUNum to GPR 23 of target TC
  144. // vpe1 of each core can execute cached as it's L1 I$ has already been initialized.
  145. // and the L2$ has been initialized or "disabled" via CCA override.
  146. la a1, __reset_vector // load boot code starting address
  147. #if !defined(EVA) && !defined(MPU)
  148. ins a1, zero, 29, 1 // Convert to cached kseg0 address in case we linked to kseg1.
  149. #endif
  150. mttc0 a1, C0_TCRESTART // write C0_TCRestart so TC 1 on VPE 1
  151. // will execute this boot code once EVP is set in C0_MVPCtl below
  152. // now set TC1 to take interrupts and set TC1 to active
  153. mftc0 a0, C0_TCSTATUS // read C0_TCSTATUS
  154. ins a0, zero, 10, 1 // insert IXMT to enable this TC to take interrupts
  155. li a1, 1
  156. ins a0, a1, 13, 1 // set A to Activate this TC
  157. mttc0 a0, C0_TCSTATUS // write C0_TCSTATUS
  158. // For this boot code only TC1 bound to VPE1 will be needed to execute (unhalted)
  159. // The other TCs if any will remain in a Halted state assuming they will later be
  160. // started by the OS.
  161. // Clear H in TCHalt to unhalt this TC
  162. // NOTE will not execute until EVP is set in C0_MVPCtl below
  163. mttc0 zero, C0_TCHALT
  164. check_for_more_TC: // Done initializing VPE 1 if there was one
  165. addu target_TC, 1 // advance TC number
  166. sltu a1, r21_more_tcs, target_TC // set a1 if TC number is less than total # of TC in system
  167. beqz a1, nexttc // go back and initialize another TC
  168. mftc0 a0, C0_VPECONF0 // read C0_VPECONF0
  169. ori a0, 1 // set VPA (Virtual Processor Activated)
  170. mttc0 a0, C0_VPECONF0 // write C0_VPECONF0 so this TC is able to execute once EVP is set in C0_MVPCtl below
  171. // Exit config mode
  172. mfc0 a0, C0_MVPCONTROL // read C0_MVPCtl
  173. ori a0, 1 // set EVP (Enable Virtual Processing) to enable execution by vpe1
  174. ins a0, zero, 1, 1 // Clear VPC (VPE Configuration State) bit
  175. mtc0 a0, C0_MVPCONTROL // write C0_MVPCtl
  176. ehb
  177. done_init_vpe1:
  178. jr ra
  179. END(init_vpe1)
  180. #undef target_TC
  181. #undef VPE_1