| // Copyright 2018 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| //go:build mips64 || mips64le |
| // +build mips64 mips64le |
| |
| #include "go_asm.h" |
| #include "textflag.h" |
| |
| #define REGCTXT R22 |
| |
| // memequal(a, b unsafe.Pointer, size uintptr) bool |
| TEXT runtime·memequal(SB),NOSPLIT|NOFRAME,$0-25 |
| MOVV a+0(FP), R1 |
| MOVV b+8(FP), R2 |
| BEQ R1, R2, eq |
| MOVV size+16(FP), R3 |
| ADDV R1, R3, R4 |
| |
| // chunk size is 16 |
| SGTU $16, R3, R8 |
| BEQ R0, R8, chunk_entry |
| |
| byte_loop: |
| BNE R1, R4, byte_test |
| MOVV $1, R1 |
| MOVB R1, ret+24(FP) |
| RET |
| byte_test: |
| MOVBU (R1), R6 |
| ADDV $1, R1 |
| MOVBU (R2), R7 |
| ADDV $1, R2 |
| BEQ R6, R7, byte_loop |
| JMP not_eq |
| |
| chunk_entry: |
| // make sure both a and b are aligned |
| OR R1, R2, R9 |
| AND $0x7, R9 |
| BNE R0, R9, byte_loop |
| JMP chunk_loop_1 |
| |
| chunk_loop: |
| // chunk size is 16 |
| SGTU $16, R3, R8 |
| BNE R0, R8, chunk_tail_8 |
| chunk_loop_1: |
| MOVV (R1), R6 |
| MOVV (R2), R7 |
| BNE R6, R7, not_eq |
| MOVV 8(R1), R12 |
| MOVV 8(R2), R13 |
| ADDV $16, R1 |
| ADDV $16, R2 |
| SUBV $16, R3 |
| BEQ R12, R13, chunk_loop |
| JMP not_eq |
| |
| chunk_tail_8: |
| AND $8, R3, R14 |
| BEQ R0, R14, chunk_tail_4 |
| MOVV (R1), R6 |
| MOVV (R2), R7 |
| BNE R6, R7, not_eq |
| ADDV $8, R1 |
| ADDV $8, R2 |
| |
| chunk_tail_4: |
| AND $4, R3, R14 |
| BEQ R0, R14, chunk_tail_2 |
| MOVWU (R1), R6 |
| MOVWU (R2), R7 |
| BNE R6, R7, not_eq |
| ADDV $4, R1 |
| ADDV $4, R2 |
| |
| chunk_tail_2: |
| AND $2, R3, R14 |
| BEQ R0, R14, chunk_tail_1 |
| MOVHU (R1), R6 |
| MOVHU (R2), R7 |
| BNE R6, R7, not_eq |
| ADDV $2, R1 |
| ADDV $2, R2 |
| |
| chunk_tail_1: |
| AND $1, R3, R14 |
| BEQ R0, R14, eq |
| MOVBU (R1), R6 |
| MOVBU (R2), R7 |
| BEQ R6, R7, eq |
| |
| not_eq: |
| MOVB R0, ret+24(FP) |
| RET |
| eq: |
| MOVV $1, R1 |
| MOVB R1, ret+24(FP) |
| RET |
| |
| // memequal_varlen(a, b unsafe.Pointer) bool |
| TEXT runtime·memequal_varlen(SB),NOSPLIT,$40-17 |
| MOVV a+0(FP), R1 |
| MOVV b+8(FP), R2 |
| BEQ R1, R2, eq |
| MOVV 8(REGCTXT), R3 // compiler stores size at offset 8 in the closure |
| MOVV R1, 8(R29) |
| MOVV R2, 16(R29) |
| MOVV R3, 24(R29) |
| JAL runtime·memequal(SB) |
| MOVBU 32(R29), R1 |
| MOVB R1, ret+16(FP) |
| RET |
| eq: |
| MOVV $1, R1 |
| MOVB R1, ret+16(FP) |
| RET |