Commit | Line | Data |
---|---|---|
b2441318 | 1 | /* SPDX-License-Identifier: GPL-2.0 */ |
535c611d HC |
2 | /* |
3 | * String handling functions. | |
4 | * | |
5 | * Copyright IBM Corp. 2012 | |
6 | */ | |
7 | ||
b8c723f1 | 8 | #include <linux/export.h> |
535c611d | 9 | #include <linux/linkage.h> |
97489e06 MS |
10 | #include <asm/nospec-insn.h> |
11 | ||
12 | GEN_BR_THUNK %r14 | |
535c611d | 13 | |
b4623d4e HC |
14 | /* |
15 | * void *memmove(void *dest, const void *src, size_t n) | |
16 | */ | |
e48b6853 | 17 | SYM_FUNC_START(__memmove) |
b4623d4e HC |
18 | ltgr %r4,%r4 |
19 | lgr %r1,%r2 | |
5eda25b1 | 20 | jz .Lmemmove_exit |
551f4134 | 21 | aghi %r4,-1 |
b4623d4e HC |
22 | clgr %r2,%r3 |
23 | jnh .Lmemmove_forward | |
551f4134 | 24 | la %r5,1(%r4,%r3) |
b4623d4e HC |
25 | clgr %r2,%r5 |
26 | jl .Lmemmove_reverse | |
27 | .Lmemmove_forward: | |
b4623d4e HC |
28 | srlg %r0,%r4,8 |
29 | ltgr %r0,%r0 | |
551f4134 HC |
30 | jz .Lmemmove_forward_remainder |
31 | .Lmemmove_forward_loop: | |
b4623d4e HC |
32 | mvc 0(256,%r1),0(%r3) |
33 | la %r1,256(%r1) | |
34 | la %r3,256(%r3) | |
551f4134 HC |
35 | brctg %r0,.Lmemmove_forward_loop |
36 | .Lmemmove_forward_remainder: | |
807e39ed | 37 | exrl %r4,.Lmemmove_mvc |
5eda25b1 | 38 | .Lmemmove_exit: |
97489e06 | 39 | BR_EX %r14 |
b4623d4e | 40 | .Lmemmove_reverse: |
b4623d4e HC |
41 | ic %r0,0(%r4,%r3) |
42 | stc %r0,0(%r4,%r1) | |
551f4134 | 43 | brctg %r4,.Lmemmove_reverse |
b4623d4e HC |
44 | ic %r0,0(%r4,%r3) |
45 | stc %r0,0(%r4,%r1) | |
97489e06 | 46 | BR_EX %r14 |
b4623d4e HC |
47 | .Lmemmove_mvc: |
48 | mvc 0(1,%r1),0(%r3) | |
e48b6853 HC |
49 | SYM_FUNC_END(__memmove) |
50 | EXPORT_SYMBOL(__memmove) | |
51 | ||
52 | SYM_FUNC_ALIAS(memmove, __memmove) | |
b4623d4e HC |
53 | EXPORT_SYMBOL(memmove) |
54 | ||
535c611d HC |
55 | /* |
56 | * memset implementation | |
57 | * | |
58 | * This code corresponds to the C construct below. We do distinguish | |
59 | * between clearing (c == 0) and setting a memory array (c != 0) simply | |
60 | * because nearly all memset invocations in the kernel clear memory and | |
61 | * the xc instruction is preferred in such cases. | |
62 | * | |
63 | * void *memset(void *s, int c, size_t n) | |
64 | * { | |
65 | * if (likely(c == 0)) | |
66 | * return __builtin_memset(s, 0, n); | |
67 | * return __builtin_memset(s, c, n); | |
68 | * } | |
69 | */ | |
e48b6853 | 70 | SYM_FUNC_START(__memset) |
535c611d | 71 | ltgr %r4,%r4 |
5eda25b1 | 72 | jz .Lmemset_exit |
535c611d HC |
73 | ltgr %r3,%r3 |
74 | jnz .Lmemset_fill | |
75 | aghi %r4,-1 | |
76 | srlg %r3,%r4,8 | |
77 | ltgr %r3,%r3 | |
78 | lgr %r1,%r2 | |
551f4134 | 79 | jz .Lmemset_clear_remainder |
535c611d HC |
80 | .Lmemset_clear_loop: |
81 | xc 0(256,%r1),0(%r1) | |
82 | la %r1,256(%r1) | |
83 | brctg %r3,.Lmemset_clear_loop | |
551f4134 | 84 | .Lmemset_clear_remainder: |
807e39ed | 85 | exrl %r4,.Lmemset_xc |
5eda25b1 | 86 | .Lmemset_exit: |
97489e06 | 87 | BR_EX %r14 |
535c611d | 88 | .Lmemset_fill: |
535c611d HC |
89 | cghi %r4,1 |
90 | lgr %r1,%r2 | |
993fef95 | 91 | je .Lmemset_fill_exit |
535c611d | 92 | aghi %r4,-2 |
993fef95 HC |
93 | srlg %r5,%r4,8 |
94 | ltgr %r5,%r5 | |
551f4134 | 95 | jz .Lmemset_fill_remainder |
535c611d | 96 | .Lmemset_fill_loop: |
993fef95 HC |
97 | stc %r3,0(%r1) |
98 | mvc 1(255,%r1),0(%r1) | |
535c611d | 99 | la %r1,256(%r1) |
993fef95 | 100 | brctg %r5,.Lmemset_fill_loop |
551f4134 | 101 | .Lmemset_fill_remainder: |
993fef95 | 102 | stc %r3,0(%r1) |
807e39ed | 103 | exrl %r4,.Lmemset_mvc |
97489e06 | 104 | BR_EX %r14 |
993fef95 HC |
105 | .Lmemset_fill_exit: |
106 | stc %r3,0(%r1) | |
97489e06 | 107 | BR_EX %r14 |
535c611d HC |
108 | .Lmemset_xc: |
109 | xc 0(1,%r1),0(%r1) | |
110 | .Lmemset_mvc: | |
111 | mvc 1(1,%r1),0(%r1) | |
e48b6853 HC |
112 | SYM_FUNC_END(__memset) |
113 | EXPORT_SYMBOL(__memset) | |
114 | ||
115 | SYM_FUNC_ALIAS(memset, __memset) | |
711f5df7 | 116 | EXPORT_SYMBOL(memset) |
535c611d HC |
117 | |
118 | /* | |
119 | * memcpy implementation | |
120 | * | |
121 | * void *memcpy(void *dest, const void *src, size_t n) | |
122 | */ | |
e48b6853 | 123 | SYM_FUNC_START(__memcpy) |
535c611d | 124 | ltgr %r4,%r4 |
5eda25b1 | 125 | jz .Lmemcpy_exit |
535c611d HC |
126 | aghi %r4,-1 |
127 | srlg %r5,%r4,8 | |
128 | ltgr %r5,%r5 | |
129 | lgr %r1,%r2 | |
130 | jnz .Lmemcpy_loop | |
551f4134 | 131 | .Lmemcpy_remainder: |
807e39ed | 132 | exrl %r4,.Lmemcpy_mvc |
5eda25b1 | 133 | .Lmemcpy_exit: |
97489e06 | 134 | BR_EX %r14 |
535c611d HC |
135 | .Lmemcpy_loop: |
136 | mvc 0(256,%r1),0(%r3) | |
137 | la %r1,256(%r1) | |
138 | la %r3,256(%r3) | |
139 | brctg %r5,.Lmemcpy_loop | |
551f4134 | 140 | j .Lmemcpy_remainder |
535c611d HC |
141 | .Lmemcpy_mvc: |
142 | mvc 0(1,%r1),0(%r3) | |
e48b6853 HC |
143 | SYM_FUNC_END(__memcpy) |
144 | EXPORT_SYMBOL(__memcpy) | |
145 | ||
146 | SYM_FUNC_ALIAS(memcpy, __memcpy) | |
711f5df7 | 147 | EXPORT_SYMBOL(memcpy) |
0b77d670 HC |
148 | |
149 | /* | |
150 | * __memset16/32/64 | |
151 | * | |
152 | * void *__memset16(uint16_t *s, uint16_t v, size_t count) | |
153 | * void *__memset32(uint32_t *s, uint32_t v, size_t count) | |
154 | * void *__memset64(uint64_t *s, uint64_t v, size_t count) | |
155 | */ | |
156 | .macro __MEMSET bits,bytes,insn | |
45769052 | 157 | SYM_FUNC_START(__memset\bits) |
0b77d670 | 158 | ltgr %r4,%r4 |
5eda25b1 | 159 | jz .L__memset_exit\bits |
0b77d670 | 160 | cghi %r4,\bytes |
5eda25b1 | 161 | je .L__memset_store\bits |
0b77d670 HC |
162 | aghi %r4,-(\bytes+1) |
163 | srlg %r5,%r4,8 | |
164 | ltgr %r5,%r5 | |
165 | lgr %r1,%r2 | |
166 | jz .L__memset_remainder\bits | |
167 | .L__memset_loop\bits: | |
168 | \insn %r3,0(%r1) | |
169 | mvc \bytes(256-\bytes,%r1),0(%r1) | |
170 | la %r1,256(%r1) | |
171 | brctg %r5,.L__memset_loop\bits | |
172 | .L__memset_remainder\bits: | |
173 | \insn %r3,0(%r1) | |
807e39ed | 174 | exrl %r4,.L__memset_mvc\bits |
97489e06 | 175 | BR_EX %r14 |
5eda25b1 | 176 | .L__memset_store\bits: |
0b77d670 | 177 | \insn %r3,0(%r2) |
5eda25b1 | 178 | .L__memset_exit\bits: |
97489e06 | 179 | BR_EX %r14 |
0b77d670 HC |
180 | .L__memset_mvc\bits: |
181 | mvc \bytes(1,%r1),0(%r1) | |
45769052 | 182 | SYM_FUNC_END(__memset\bits) |
0b77d670 HC |
183 | .endm |
184 | ||
185 | __MEMSET 16,2,sth | |
186 | EXPORT_SYMBOL(__memset16) | |
187 | ||
188 | __MEMSET 32,4,st | |
189 | EXPORT_SYMBOL(__memset32) | |
190 | ||
191 | __MEMSET 64,8,stg | |
192 | EXPORT_SYMBOL(__memset64) |