123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- /*
- * init_vpe1.S
- *
- * Initialize the second vpe and additional TCs
- */
- /*
- Copyright (c) 2007-2018, MIPS Tech, LLC and/or its affiliated group companies or licensors
- All rights reserved.
- Redistribution and use in source and binary forms, with or without modification, are
- permitted provided that the following conditions are met:
- 1. Redistributions of source code must retain the above copyright notice, this list of
- conditions and the following disclaimer.
- 2. Redistributions in binary form must reproduce the above copyright notice, this list
- of conditions and the following disclaimer in the documentation and/or other materials
- provided with the distribution.
- 3. Neither the name of the copyright holder nor the names of its contributors may be
- used to endorse or promote products derived from this software without specific prior
- written permission.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
- EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
- SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
- OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
- OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
- SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #include <mips/asm.h>
- #include <boot.h>
- #include <mips/mt.h>
- #include <mips/regdef.h>
- #include <mips/m32c0.h>
- #define target_TC a3 // will hold the current TC being configured
- #define VPE_1 1
- .set noat
- /**************************************************************************************
- **************************************************************************************/
- LEAF(init_vpe1)
- // Each vpe will need to set up additional TC bound to it. (No rebinding.)
- beqz r21_more_tcs, done_init_vpe1 // If there is no more TCs then return
- beqz r20_more_vpes, done_init_vpe1 // If there is no vpe1 then return
- // This is executing on TC0 bound to VPE0. Therefore VPEConf0.MVP is already set.
- // Enable Virtual Processor Configuration to enter configuration mode
- mfc0 a0, C0_MVPCONTROL // read C0_MVPCtl
- or a0, (1 << 1) // set VPC to enable Virtual Processor Configuration
- mtc0 a0, C0_MVPCONTROL // write C0_MVPCtl
- ehb
- // Initialize target_TC, target_TC will be incremented at the
- // bottom of the loop if there are more TCs
- li target_TC, 1
- nexttc:
- // Set TargTC in the CP0 VPECONTROL register
- // TargTC Selects the TC number of the "other thread context" for any
- // Move to Thread Context or Move from Thread Context instructions
- // (any instructions that begin with mtt or mft)
- mfc0 a0, C0_VPECONTROL // read C0_VPECTL
- ins a0, target_TC, 0, 8 // insert TargTC
- mtc0 a0, C0_VPECONTROL // write C0_VPECTL
- ehb
- // Halt target_TC being configured
- li a0, 1 // set Halt bit
- mttc0 a0, C0_TCHALT // write C0_TCHALT
- ehb
- // Set up TCStatus register:
- // Disable Coprocessor Usable bits
- // Disable MDMX/DSP ASE
- // Clear Dirty target_TC
- // not dynamically allocatable
- // not allocated
- // Kernel mode
- // interrupt exempt
- // ASID 0
- // NOTE: Only bit that needs to be set is IXMT
- li a0, (1 << 10) // set IXMT
- mttc0 a0, C0_TCSTATUS // write C0_TCSTATUS of target TC
- // Initialize the target_TC's register file (NOTE: $2 is in the gpr of the tc executing this code)
- // NOTE: Good practice but not programmatically necessary
- li $2, 0xdeadbeef
- mttgpr $2, $1
- mttgpr $2, $2
- mttgpr $2, $3
- mttgpr $2, $4
- mttgpr $2, $5
- mttgpr $2, $6
- mttgpr $2, $7
- mttgpr $2, $8
- mttgpr $2, $9
- mttgpr $2, $10
- mttgpr $2, $11
- mttgpr $2, $12
- mttgpr $2, $13
- mttgpr $2, $14
- mttgpr $2, $15
- mttgpr $2, $16
- mttgpr $2, $17
- mttgpr $2, $18
- mttgpr $2, $19
- mttgpr $2, $20
- mttgpr $2, $21
- mttgpr $2, $22
- mttgpr $2, $23
- mttgpr $2, $24
- mttgpr $2, $25
- mttgpr $2, $26
- mttgpr $2, $27
- mttgpr $2, $28
- mttgpr $2, $29
- mttgpr $2, $30
- mttgpr $2, $31
- // Bind TC1 and all other remaining TCs to VPE1 context
- // All mftc0 and mttc0 instructions
- // will then operate on the VPE1 CP0 registers
- li a0, VPE_1
- mftc0 a1, C0_TCBIND // Read C0_TCBind
- ins a1, a0, 0, 4 // insert vpe 1 into CurVPE field
- mttc0 a1, C0_TCBIND // write C0_TCBind
- // Must only do next part up to check_for_more_TC label once
- bne a0, target_TC, check_for_more_TC // branch if not TC1 NOTE: a0 set to 1 above
- // Set XTC for target_TC (sets TC1 to be the only TC runnable on the VPE1)
- mftc0 a0, C0_VPECONF0 // read C0_VPECONF0
- ins a0, target_TC, 21, 8 // insert XTC
- mttc0 a0, C0_VPECONF0 // write C0_VPECONF0
- // Disable multi-threading for VPE1
- mftc0 a0, C0_VPECONTROL // read C0_VPECTL
- ins a0, zero, 15, 1 // clear TE (only tc1 can execute code)
- mttc0 a0, C0_VPECONTROL // write C0_VPECTL
- // for VPE1 (Just Clear VPA to prevent any TC bound to it from executing
- // and set master VPE so CP0 registers are writable
- mftc0 a0, C0_VPECONF0 // read C0_VPECONF0
- ins a0, zero, 0, 1 // clear VPA
- or a0, (1 << 1) // set MVP
- mttc0 a0, C0_VPECONF0 // write C0_VPECONF0
- mfc0 a0, C0_STATUS // read C0_STATUS
- mttc0 a0, C0_STATUS // write C0_Status
- li a0, 0x12345678
- mttc0 a0, C0_EPC // write C0_EPC
- mttc0 zero, C0_CAUSE // write C0_CAUSE
- mfc0 a0, C0_CONFIG // read VPE 0 C0_CONFIG
- mttc0 a0, C0_CONFIG // write it to VPE 1 C0_CONFIG
- mftc0 a0, C0_EBASE // read C0_EBASE
- ext a0, a0, 0, 10 // extract CPUNum
- mttgpr a0, r23_cpu_num // write CPUNum to GPR 23 of target TC
- // vpe1 of each core can execute cached as it's L1 I$ has already been initialized.
- // and the L2$ has been initialized or "disabled" via CCA override.
- la a1, __reset_vector // load boot code starting address
- #if !defined(EVA) && !defined(MPU)
- ins a1, zero, 29, 1 // Convert to cached kseg0 address in case we linked to kseg1.
- #endif
- mttc0 a1, C0_TCRESTART // write C0_TCRestart so TC 1 on VPE 1
- // will execute this boot code once EVP is set in C0_MVPCtl below
- // now set TC1 to take interrupts and set TC1 to active
- mftc0 a0, C0_TCSTATUS // read C0_TCSTATUS
- ins a0, zero, 10, 1 // insert IXMT to enable this TC to take interrupts
- li a1, 1
- ins a0, a1, 13, 1 // set A to Activate this TC
- mttc0 a0, C0_TCSTATUS // write C0_TCSTATUS
- // For this boot code only TC1 bound to VPE1 will be needed to execute (unhalted)
- // The other TCs if any will remain in a Halted state assuming they will later be
- // started by the OS.
- // Clear H in TCHalt to unhalt this TC
- // NOTE will not execute until EVP is set in C0_MVPCtl below
- mttc0 zero, C0_TCHALT
- check_for_more_TC: // Done initializing VPE 1 if there was one
- addu target_TC, 1 // advance TC number
- sltu a1, r21_more_tcs, target_TC // set a1 if TC number is less than total # of TC in system
- beqz a1, nexttc // go back and initialize another TC
- mftc0 a0, C0_VPECONF0 // read C0_VPECONF0
- ori a0, 1 // set VPA (Virtual Processor Activated)
- mttc0 a0, C0_VPECONF0 // write C0_VPECONF0 so this TC is able to execute once EVP is set in C0_MVPCtl below
- // Exit config mode
- mfc0 a0, C0_MVPCONTROL // read C0_MVPCtl
- ori a0, 1 // set EVP (Enable Virtual Processing) to enable execution by vpe1
- ins a0, zero, 1, 1 // Clear VPC (VPE Configuration State) bit
- mtc0 a0, C0_MVPCONTROL // write C0_MVPCtl
- ehb
- done_init_vpe1:
- jr ra
- END(init_vpe1)
- #undef target_TC
- #undef VPE_1
|