63 lines
1.6 KiB
Go
63 lines
1.6 KiB
Go
// Copyright 2011 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.
|
|
|
|
// +build !appengine,!gccgo
|
|
|
|
package crc32
|
|
|
|
// This file contains the code to call the SSE 4.2 version of the Castagnoli
|
|
// and IEEE CRC.
|
|
|
|
// haveSSE41/haveSSE42/haveCLMUL are defined in crc_amd64.s and uses
|
|
// CPUID to test for SSE 4.1, 4.2 and CLMUL support.
|
|
func haveSSE41() bool
|
|
func haveSSE42() bool
|
|
func haveCLMUL() bool
|
|
|
|
// castagnoliSSE42 is defined in crc_amd64.s and uses the SSE4.2 CRC32
|
|
// instruction.
|
|
// go:noescape
|
|
func castagnoliSSE42(crc uint32, p []byte) uint32
|
|
|
|
// ieeeCLMUL is defined in crc_amd64.s and uses the PCLMULQDQ
|
|
// instruction as well as SSE 4.1.
|
|
// go:noescape
|
|
func ieeeCLMUL(crc uint32, p []byte) uint32
|
|
|
|
var sse42 = haveSSE42()
|
|
var useFastIEEE = haveCLMUL() && haveSSE41()
|
|
|
|
func updateCastagnoli(crc uint32, p []byte) uint32 {
|
|
if sse42 {
|
|
return castagnoliSSE42(crc, p)
|
|
}
|
|
// only use slicing-by-8 when input is >= 16 Bytes
|
|
if len(p) >= 16 {
|
|
return updateSlicingBy8(crc, castagnoliTable8, p)
|
|
}
|
|
return update(crc, castagnoliTable, p)
|
|
}
|
|
|
|
func updateIEEE(crc uint32, p []byte) uint32 {
|
|
if useFastIEEE && len(p) >= 64 {
|
|
left := len(p) & 15
|
|
do := len(p) - left
|
|
crc := ^ieeeCLMUL(^crc, p[:do])
|
|
if left > 0 {
|
|
crc = update(crc, IEEETable, p[do:])
|
|
}
|
|
return crc
|
|
}
|
|
|
|
// only use slicing-by-8 when input is >= 16 Bytes
|
|
if len(p) >= 16 {
|
|
iEEETable8Once.Do(func() {
|
|
iEEETable8 = makeTable8(IEEE)
|
|
})
|
|
return updateSlicingBy8(crc, iEEETable8, p)
|
|
}
|
|
|
|
return update(crc, IEEETable, p)
|
|
}
|