| /* |
| * Copyright (C) 2013 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef ART_RUNTIME_ARCH_MIPS_ASM_SUPPORT_MIPS_S_ |
| #define ART_RUNTIME_ARCH_MIPS_ASM_SUPPORT_MIPS_S_ |
| |
| #include "asm_support_mips.h" |
| |
| // Define special registers. |
| |
| // Register holding suspend check count down. |
| #define rSUSPEND $s0 |
| // Register holding Thread::Current(). |
| #define rSELF $s1 |
| |
| // Declare a function called name, doesn't set up $gp. |
| .macro ENTRY_NO_GP_CUSTOM_CFA name, cfa_offset |
| .type \name, %function |
| .global \name |
| // Cache alignment for function entry. |
| .balign 16 |
| \name: |
| .cfi_startproc |
| // Ensure we get a sane starting CFA. |
| .cfi_def_cfa $sp, \cfa_offset |
| .endm |
| |
| // Declare a function called name, doesn't set up $gp. |
| .macro ENTRY_NO_GP name |
| ENTRY_NO_GP_CUSTOM_CFA \name, 0 |
| .endm |
| |
| // Declare a function called name, sets up $gp. |
| .macro ENTRY name |
| ENTRY_NO_GP \name |
| // Load $gp. We expect that ".set noreorder" is in effect. |
| .cpload $t9 |
| // Declare a local convenience label to be branched to when $gp is already set up. |
| .L\name\()_gp_set: |
| .endm |
| |
| .macro END name |
| .cfi_endproc |
| .size \name, .-\name |
| .endm |
| |
| .macro UNIMPLEMENTED name |
| ENTRY \name |
| break |
| break |
| END \name |
| .endm |
| |
| #if defined(__mips_isa_rev) && __mips_isa_rev > 2 |
| /* mips32r5 & mips32r6 have mthc1 op, and have 64-bit fp regs, |
| and in FPXX abi we avoid referring to odd-numbered fp regs */ |
| |
| /* LDu: Load 64-bit floating-point value to float reg feven, |
| from unaligned (mod-4-aligned) mem location disp(base) */ |
| .macro LDu feven,fodd,disp,base,temp |
| l.s \feven, \disp(\base) |
| lw \temp, \disp+4(\base) |
| mthc1 \temp, \feven |
| .endm |
| |
| /* SDu: Store 64-bit floating-point value from float reg feven, |
| to unaligned (mod-4-aligned) mem location disp(base) */ |
| .macro SDu feven,fodd,disp,base,temp |
| mfhc1 \temp, \feven |
| s.s \feven, \disp(\base) |
| sw \temp, \disp+4(\base) |
| .endm |
| |
| /* MTD: Move double, from general regpair (reven,rodd) |
| to float regpair (feven,fodd) */ |
| .macro MTD reven,rodd,feven,fodd |
| mtc1 \reven, \feven |
| mthc1 \rodd, \feven |
| .endm |
| |
| #else |
| /* mips32r1 has no mthc1 op; |
| mips32r1 and mips32r2 use 32-bit floating point register mode (FR=0), |
| and always hold doubles as (feven, fodd) fp reg pair */ |
| |
| .macro LDu feven,fodd,disp,base,temp |
| l.s \feven, \disp(\base) |
| l.s \fodd, \disp+4(\base) |
| .endm |
| |
| .macro SDu feven,fodd,disp,base,temp |
| s.s \feven, \disp(\base) |
| s.s \fodd, \disp+4(\base) |
| .endm |
| |
| .macro MTD reven,rodd,feven,fodd |
| mtc1 \reven, \feven |
| mtc1 \rodd, \fodd |
| .endm |
| |
| #endif /* mips_isa_rev */ |
| |
| // Macros to poison (negate) the reference for heap poisoning. |
| .macro POISON_HEAP_REF rRef |
| #ifdef USE_HEAP_POISONING |
| subu \rRef, $zero, \rRef |
| #endif // USE_HEAP_POISONING |
| .endm |
| |
| // Macros to unpoison (negate) the reference for heap poisoning. |
| .macro UNPOISON_HEAP_REF rRef |
| #ifdef USE_HEAP_POISONING |
| subu \rRef, $zero, \rRef |
| #endif // USE_HEAP_POISONING |
| .endm |
| |
| // Byte size of the instructions (un)poisoning heap references. |
| #ifdef USE_HEAP_POISONING |
| #define HEAP_POISON_INSTR_SIZE 4 |
| #else |
| #define HEAP_POISON_INSTR_SIZE 0 |
| #endif // USE_HEAP_POISONING |
| |
| // Based on contents of creg select the minimum integer |
| // At the end of the macro the original value of creg is lost |
| .macro MINint dreg,rreg,sreg,creg |
| .set push |
| .set noat |
| #if defined(_MIPS_ARCH_MIPS32R6) || defined(_MIPS_ARCH_MIPS64R6) |
| .ifc \dreg, \rreg |
| selnez \dreg, \rreg, \creg |
| seleqz \creg, \sreg, \creg |
| .else |
| seleqz \dreg, \sreg, \creg |
| selnez \creg, \rreg, \creg |
| .endif |
| or \dreg, \dreg, \creg |
| #else |
| movn \dreg, \rreg, \creg |
| movz \dreg, \sreg, \creg |
| #endif |
| .set pop |
| .endm |
| |
| // Find minimum of two signed registers |
| .macro MINs dreg,rreg,sreg |
| .set push |
| .set noat |
| slt $at, \rreg, \sreg |
| MINint \dreg, \rreg, \sreg, $at |
| .set pop |
| .endm |
| |
| // Find minimum of two unsigned registers |
| .macro MINu dreg,rreg,sreg |
| .set push |
| .set noat |
| sltu $at, \rreg, \sreg |
| MINint \dreg, \rreg, \sreg, $at |
| .set pop |
| .endm |
| |
| // This utility macro is used to check whether the address contained in |
| // a register is suitably aligned. Default usage is confirm that the |
| // address stored in $sp is a multiple of 16. It can be used for other |
| // alignments, and for other base address registers, if needed. |
| // |
| // Enable this macro by running the shell command: |
| // |
| // export ART_MIPS32_CHECK_ALIGNMENT=true |
| // |
| // NOTE: The value of alignment must be a power of 2, and must fit in an |
| // unsigned 15-bit integer. The macro won't behave as expected if these |
| // conditions aren't met. |
| // |
| .macro CHECK_ALIGNMENT ba=$sp, tmp=$at, alignment=16 |
| #ifdef ART_MIPS32_CHECK_ALIGNMENT |
| .set push |
| .set noat |
| .set noreorder |
| andi \tmp, \ba, \alignment-1 |
| beqz \tmp, .+12 # Skip break instruction if base address register (ba) is aligned |
| nop |
| break |
| .set pop |
| #endif |
| .endm |
| |
| #endif // ART_RUNTIME_ARCH_MIPS_ASM_SUPPORT_MIPS_S_ |