1
0
mirror of https://github.com/DNSCrypt/dnscrypt-proxy.git synced 2025-01-19 03:00:29 +01:00
dnscrypt-proxy/vendor/golang.org/x/crypto/poly1305/sum_s390x.go

77 lines
2.0 KiB
Go
Raw Normal View History

2018-05-19 01:50:26 +02:00
// 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.
2021-02-20 18:55:58 +01:00
//go:build gc && !purego
2020-12-10 19:00:57 +01:00
// +build gc,!purego
2018-05-19 01:50:26 +02:00
package poly1305
2019-03-01 18:44:37 +01:00
import (
"golang.org/x/sys/cpu"
)
2018-05-19 01:50:26 +02:00
2020-06-02 13:56:05 +02:00
// updateVX is an assembly implementation of Poly1305 that uses vector
2018-05-19 01:50:26 +02:00
// instructions. It must only be called if the vector facility (vx) is
// available.
//go:noescape
2020-06-02 13:56:05 +02:00
func updateVX(state *macState, msg []byte)
2018-05-19 01:50:26 +02:00
2020-06-02 13:56:05 +02:00
// mac is a replacement for macGeneric that uses a larger buffer and redirects
// calls that would have gone to updateGeneric to updateVX if the vector
// facility is installed.
//
// A larger buffer is required for good performance because the vector
// implementation has a higher fixed cost per call than the generic
// implementation.
type mac struct {
macState
buffer [16 * TagSize]byte // size must be a multiple of block size (16)
offset int
}
2018-05-19 01:50:26 +02:00
2020-06-02 13:56:05 +02:00
func (h *mac) Write(p []byte) (int, error) {
nn := len(p)
if h.offset > 0 {
n := copy(h.buffer[h.offset:], p)
if h.offset+n < len(h.buffer) {
h.offset += n
return nn, nil
2018-05-19 01:50:26 +02:00
}
2020-06-02 13:56:05 +02:00
p = p[n:]
h.offset = 0
if cpu.S390X.HasVX {
updateVX(&h.macState, h.buffer[:])
2018-05-19 01:50:26 +02:00
} else {
2020-06-02 13:56:05 +02:00
updateGeneric(&h.macState, h.buffer[:])
2018-05-19 01:50:26 +02:00
}
}
2020-06-02 13:56:05 +02:00
tail := len(p) % len(h.buffer) // number of bytes to copy into buffer
body := len(p) - tail // number of bytes to process now
if body > 0 {
if cpu.S390X.HasVX {
updateVX(&h.macState, p[:body])
} else {
updateGeneric(&h.macState, p[:body])
}
}
h.offset = copy(h.buffer[:], p[body:]) // copy tail bytes - can be 0
return nn, nil
}
func (h *mac) Sum(out *[TagSize]byte) {
state := h.macState
remainder := h.buffer[:h.offset]
// Use the generic implementation if we have 2 or fewer blocks left
// to sum. The vector implementation has a higher startup time.
if cpu.S390X.HasVX && len(remainder) > 2*TagSize {
updateVX(&state, remainder)
} else if len(remainder) > 0 {
updateGeneric(&state, remainder)
}
finalize(out, &state.h, &state.s)
2018-05-19 01:50:26 +02:00
}