1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
|
// 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
#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
|