blob: c2f7d3997edb36c0d38ad5739c4f5b61455cd3aa [file] [log] [blame]
// 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