1 /*
2 * Xor Based Zero Run Length Encoding
3 *
4 * Copyright 2013 Red Hat, Inc. and/or its affiliates
5 *
6 * Authors:
7 * Orit Wasserman <owasserm@redhat.com>
8 *
9 * This work is licensed under the terms of the GNU GPL, version 2 or later.
10 * See the COPYING file in the top-level directory.
11 *
12 */
13 #include "qemu/osdep.h"
14 #include "qemu/cutils.h"
15 #include "qemu/host-utils.h"
16 #include "xbzrle.h"
17
18 #if defined(CONFIG_AVX512BW_OPT)
19 #include <immintrin.h>
20 #include "host/cpuinfo.h"
21
22 static int __attribute__((target("avx512bw")))
xbzrle_encode_buffer_avx512(uint8_t * old_buf,uint8_t * new_buf,int slen,uint8_t * dst,int dlen)23 xbzrle_encode_buffer_avx512(uint8_t *old_buf, uint8_t *new_buf, int slen,
24 uint8_t *dst, int dlen)
25 {
26 uint32_t zrun_len = 0, nzrun_len = 0;
27 int d = 0, i = 0, num = 0;
28 uint8_t *nzrun_start = NULL;
29 /* add 1 to include residual part in main loop */
30 uint32_t count512s = (slen >> 6) + 1;
31 /* countResidual is tail of data, i.e., countResidual = slen % 64 */
32 uint32_t count_residual = slen & 0b111111;
33 bool never_same = true;
34 uint64_t mask_residual = 1;
35 mask_residual <<= count_residual;
36 mask_residual -= 1;
37 __m512i r = _mm512_set1_epi32(0);
38
39 while (count512s) {
40 int bytes_to_check = 64;
41 uint64_t mask = 0xffffffffffffffff;
42 if (count512s == 1) {
43 bytes_to_check = count_residual;
44 mask = mask_residual;
45 }
46 __m512i old_data = _mm512_mask_loadu_epi8(r,
47 mask, old_buf + i);
48 __m512i new_data = _mm512_mask_loadu_epi8(r,
49 mask, new_buf + i);
50 uint64_t comp = _mm512_cmpeq_epi8_mask(old_data, new_data);
51 count512s--;
52
53 bool is_same = (comp & 0x1);
54 while (bytes_to_check) {
55 if (d + 2 > dlen) {
56 return -1;
57 }
58 if (is_same) {
59 if (nzrun_len) {
60 d += uleb128_encode_small(dst + d, nzrun_len);
61 if (d + nzrun_len > dlen) {
62 return -1;
63 }
64 nzrun_start = new_buf + i - nzrun_len;
65 memcpy(dst + d, nzrun_start, nzrun_len);
66 d += nzrun_len;
67 nzrun_len = 0;
68 }
69 /* 64 data at a time for speed */
70 if (count512s && (comp == 0xffffffffffffffff)) {
71 i += 64;
72 zrun_len += 64;
73 break;
74 }
75 never_same = false;
76 num = ctz64(~comp);
77 num = (num < bytes_to_check) ? num : bytes_to_check;
78 zrun_len += num;
79 bytes_to_check -= num;
80 comp >>= num;
81 i += num;
82 if (bytes_to_check) {
83 /* still has different data after same data */
84 d += uleb128_encode_small(dst + d, zrun_len);
85 zrun_len = 0;
86 } else {
87 break;
88 }
89 }
90 if (never_same || zrun_len) {
91 /*
92 * never_same only acts if
93 * data begins with diff in first count512s
94 */
95 d += uleb128_encode_small(dst + d, zrun_len);
96 zrun_len = 0;
97 never_same = false;
98 }
99 /* has diff, 64 data at a time for speed */
100 if ((bytes_to_check == 64) && (comp == 0x0)) {
101 i += 64;
102 nzrun_len += 64;
103 break;
104 }
105 num = ctz64(comp);
106 num = (num < bytes_to_check) ? num : bytes_to_check;
107 nzrun_len += num;
108 bytes_to_check -= num;
109 comp >>= num;
110 i += num;
111 if (bytes_to_check) {
112 /* mask like 111000 */
113 d += uleb128_encode_small(dst + d, nzrun_len);
114 /* overflow */
115 if (d + nzrun_len > dlen) {
116 return -1;
117 }
118 nzrun_start = new_buf + i - nzrun_len;
119 memcpy(dst + d, nzrun_start, nzrun_len);
120 d += nzrun_len;
121 nzrun_len = 0;
122 is_same = true;
123 }
124 }
125 }
126
127 if (nzrun_len != 0) {
128 d += uleb128_encode_small(dst + d, nzrun_len);
129 /* overflow */
130 if (d + nzrun_len > dlen) {
131 return -1;
132 }
133 nzrun_start = new_buf + i - nzrun_len;
134 memcpy(dst + d, nzrun_start, nzrun_len);
135 d += nzrun_len;
136 }
137 return d;
138 }
139
140 static int xbzrle_encode_buffer_int(uint8_t *old_buf, uint8_t *new_buf,
141 int slen, uint8_t *dst, int dlen);
142
143 static int (*accel_func)(uint8_t *, uint8_t *, int, uint8_t *, int);
144
init_accel(void)145 static void __attribute__((constructor)) init_accel(void)
146 {
147 unsigned info = cpuinfo_init();
148 if (info & CPUINFO_AVX512BW) {
149 accel_func = xbzrle_encode_buffer_avx512;
150 } else {
151 accel_func = xbzrle_encode_buffer_int;
152 }
153 }
154
xbzrle_encode_buffer(uint8_t * old_buf,uint8_t * new_buf,int slen,uint8_t * dst,int dlen)155 int xbzrle_encode_buffer(uint8_t *old_buf, uint8_t *new_buf, int slen,
156 uint8_t *dst, int dlen)
157 {
158 return accel_func(old_buf, new_buf, slen, dst, dlen);
159 }
160
161 #define xbzrle_encode_buffer xbzrle_encode_buffer_int
162 #endif
163
164 /*
165 page = zrun nzrun
166 | zrun nzrun page
167
168 zrun = length
169
170 nzrun = length byte...
171
172 length = uleb128 encoded integer
173 */
xbzrle_encode_buffer(uint8_t * old_buf,uint8_t * new_buf,int slen,uint8_t * dst,int dlen)174 int xbzrle_encode_buffer(uint8_t *old_buf, uint8_t *new_buf, int slen,
175 uint8_t *dst, int dlen)
176 {
177 uint32_t zrun_len = 0, nzrun_len = 0;
178 int d = 0, i = 0;
179 long res;
180 uint8_t *nzrun_start = NULL;
181
182 g_assert(!(((uintptr_t)old_buf | (uintptr_t)new_buf | slen) %
183 sizeof(long)));
184
185 while (i < slen) {
186 /* overflow */
187 if (d + 2 > dlen) {
188 return -1;
189 }
190
191 /* not aligned to sizeof(long) */
192 res = (slen - i) % sizeof(long);
193 while (res && old_buf[i] == new_buf[i]) {
194 zrun_len++;
195 i++;
196 res--;
197 }
198
199 /* word at a time for speed */
200 if (!res) {
201 while (i < slen &&
202 (*(long *)(old_buf + i)) == (*(long *)(new_buf + i))) {
203 i += sizeof(long);
204 zrun_len += sizeof(long);
205 }
206
207 /* go over the rest */
208 while (i < slen && old_buf[i] == new_buf[i]) {
209 zrun_len++;
210 i++;
211 }
212 }
213
214 /* buffer unchanged */
215 if (zrun_len == slen) {
216 return 0;
217 }
218
219 /* skip last zero run */
220 if (i == slen) {
221 return d;
222 }
223
224 d += uleb128_encode_small(dst + d, zrun_len);
225
226 zrun_len = 0;
227 nzrun_start = new_buf + i;
228
229 /* overflow */
230 if (d + 2 > dlen) {
231 return -1;
232 }
233 /* not aligned to sizeof(long) */
234 res = (slen - i) % sizeof(long);
235 while (res && old_buf[i] != new_buf[i]) {
236 i++;
237 nzrun_len++;
238 res--;
239 }
240
241 /* word at a time for speed, use of 32-bit long okay */
242 if (!res) {
243 /* truncation to 32-bit long okay */
244 unsigned long mask = (unsigned long)0x0101010101010101ULL;
245 while (i < slen) {
246 unsigned long xor;
247 xor = *(unsigned long *)(old_buf + i)
248 ^ *(unsigned long *)(new_buf + i);
249 if ((xor - mask) & ~xor & (mask << 7)) {
250 /* found the end of an nzrun within the current long */
251 while (old_buf[i] != new_buf[i]) {
252 nzrun_len++;
253 i++;
254 }
255 break;
256 } else {
257 i += sizeof(long);
258 nzrun_len += sizeof(long);
259 }
260 }
261 }
262
263 d += uleb128_encode_small(dst + d, nzrun_len);
264 /* overflow */
265 if (d + nzrun_len > dlen) {
266 return -1;
267 }
268 memcpy(dst + d, nzrun_start, nzrun_len);
269 d += nzrun_len;
270 nzrun_len = 0;
271 }
272
273 return d;
274 }
275
xbzrle_decode_buffer(uint8_t * src,int slen,uint8_t * dst,int dlen)276 int xbzrle_decode_buffer(uint8_t *src, int slen, uint8_t *dst, int dlen)
277 {
278 int i = 0, d = 0;
279 int ret;
280 uint32_t count = 0;
281
282 while (i < slen) {
283
284 /* zrun */
285 if ((slen - i) < 2) {
286 return -1;
287 }
288
289 ret = uleb128_decode_small(src + i, &count);
290 if (ret < 0 || (i && !count)) {
291 return -1;
292 }
293 i += ret;
294 d += count;
295
296 /* overflow */
297 if (d > dlen) {
298 return -1;
299 }
300
301 /* nzrun */
302 if ((slen - i) < 2) {
303 return -1;
304 }
305
306 ret = uleb128_decode_small(src + i, &count);
307 if (ret < 0 || !count) {
308 return -1;
309 }
310 i += ret;
311
312 /* overflow */
313 if (d + count > dlen || i + count > slen) {
314 return -1;
315 }
316
317 memcpy(dst + d, src + i, count);
318 d += count;
319 i += count;
320 }
321
322 return d;
323 }
324