blob: fec67cd9757fbfe478b8d6736b901ae650800354 [file] [log] [blame]
Damjan Marion0f68c792017-04-26 13:05:05 +02001/*
2 * Copyright (c) 2015 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#ifndef __included_crc32_h__
17#define __included_crc32_h__
18
Neale Rannsb32fde52017-06-12 06:12:26 -070019#include <vppinfra/clib.h>
20
Damjan Marion0f68c792017-04-26 13:05:05 +020021#if __SSE4_2__
Christophe Fontaineb4bd28a2017-05-31 11:27:19 +020022#define clib_crc32c_uses_intrinsics
Damjan Marion0f68c792017-04-26 13:05:05 +020023#include <x86intrin.h>
24
Gabriel Ganne8e66b9b2017-12-14 16:20:37 +010025#define crc32_u64 _mm_crc32_u64
Steven0d883012018-05-11 11:06:23 -070026#define crc32_u32 _mm_crc32_u32
Gabriel Ganne8e66b9b2017-12-14 16:20:37 +010027
Damjan Marion0f68c792017-04-26 13:05:05 +020028static_always_inline u32
29clib_crc32c (u8 * s, int len)
30{
31 u32 v = 0;
32
David Johnsond9818dd2018-12-14 14:53:41 -050033#if defined(__x86_64__)
Damjan Marion0f68c792017-04-26 13:05:05 +020034 for (; len >= 8; len -= 8, s += 8)
35 v = _mm_crc32_u64 (v, *((u64 *) s));
36#else
37 /* workaround weird GCC bug when using _mm_crc32_u32
38 which happens with -O2 optimization */
Dave Barach9466c452018-08-24 17:21:14 -040039#if !defined (__i686__)
David Johnsond9818dd2018-12-14 14:53:41 -050040 asm volatile ("":::"memory");
Damjan Marion0f68c792017-04-26 13:05:05 +020041#endif
Dave Barach9466c452018-08-24 17:21:14 -040042#endif
Damjan Marion0f68c792017-04-26 13:05:05 +020043
44 for (; len >= 4; len -= 4, s += 4)
45 v = _mm_crc32_u32 (v, *((u32 *) s));
46
47 for (; len >= 2; len -= 2, s += 2)
48 v = _mm_crc32_u16 (v, *((u16 *) s));
49
50 for (; len >= 1; len -= 1, s += 1)
51 v = _mm_crc32_u8 (v, *((u16 *) s));
52
53 return v;
54}
55
Christophe Fontaineb4bd28a2017-05-31 11:27:19 +020056#elif __ARM_FEATURE_CRC32
Gabriel Ganne9e12d772017-11-14 17:33:51 +010057#define clib_crc32c_uses_intrinsics
Christophe Fontaineb4bd28a2017-05-31 11:27:19 +020058#include <arm_acle.h>
Damjan Marion0f68c792017-04-26 13:05:05 +020059
Gabriel Ganne8e66b9b2017-12-14 16:20:37 +010060
61#define crc32_u64 __crc32cd
Steven0d883012018-05-11 11:06:23 -070062#define crc32_u32 __crc32cw
Gabriel Ganne8e66b9b2017-12-14 16:20:37 +010063
Christophe Fontaineb4bd28a2017-05-31 11:27:19 +020064static_always_inline u32
65clib_crc32c (u8 * s, int len)
66{
67 u32 v = 0;
68
69 for (; len >= 8; len -= 8, s += 8)
70 v = __crc32cd (v, *((u64 *) s));
71
72 for (; len >= 4; len -= 4, s += 4)
73 v = __crc32cw (v, *((u32 *) s));
74
75 for (; len >= 2; len -= 2, s += 2)
76 v = __crc32ch (v, *((u16 *) s));
77
78 for (; len >= 1; len -= 1, s += 1)
79 v = __crc32cb (v, *((u8 *) s));
80
81 return v;
82}
83
84#endif
Damjan Marion0f68c792017-04-26 13:05:05 +020085#endif /* __included_crc32_h__ */
86
87/*
88 * fd.io coding-style-patch-verification: ON
89 *
90 * Local Variables:
91 * eval: (c-set-style "gnu")
92 * End:
93 */