lib/rand: Optimizations (#7964)

rand.secureSource.Uint64 no longer allocates. rand.String uses a
strings.Builder. Benchmark results on linux/amd64:

name            old time/op    new time/op    delta
SecureSource-8    69.1ns ± 3%    51.7ns ± 3%   -25.21%  (p=0.000 n=20+10)
String-8          2.66µs ± 2%    1.95µs ± 1%   -26.61%  (p=0.000 n=10+10)

name            old alloc/op   new alloc/op   delta
SecureSource-8     8.00B ± 0%     0.00B       -100.00%  (p=0.000 n=20+10)
String-8            288B ± 0%       32B ± 0%   -88.89%  (p=0.000 n=10+10)

name            old allocs/op  new allocs/op  delta
SecureSource-8      1.00 ± 0%      0.00       -100.00%  (p=0.000 n=20+10)
String-8            33.0 ± 0%       1.0 ± 0%   -96.97%  (p=0.000 n=10+10)
This commit is contained in:
greatroar 2021-09-26 12:15:39 +02:00 committed by GitHub
parent c5ec6cd7ef
commit 198028d627
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 19 additions and 11 deletions

View File

@ -12,6 +12,7 @@ import (
"io" "io"
mathRand "math/rand" mathRand "math/rand"
"reflect" "reflect"
"strings"
) )
// Reader is the standard crypto/rand.Reader with added buffering. // Reader is the standard crypto/rand.Reader with added buffering.
@ -37,11 +38,13 @@ var (
// (taken from randomCharset) of the specified length. The returned string // (taken from randomCharset) of the specified length. The returned string
// contains ~5.8 bits of entropy per character, due to the character set used. // contains ~5.8 bits of entropy per character, due to the character set used.
func String(l int) string { func String(l int) string {
bs := make([]byte, l) var sb strings.Builder
for i := range bs { sb.Grow(l)
bs[i] = randomCharset[defaultSecureRand.Intn(len(randomCharset))]
for i := 0; i < l; i++ {
sb.WriteByte(randomCharset[defaultSecureRand.Intn(len(randomCharset))])
} }
return string(bs) return sb.String()
} }
// Int63 returns a cryptographically secure random int63. // Int63 returns a cryptographically secure random int63.

View File

@ -44,3 +44,10 @@ func TestRandomUint64(t *testing.T) {
} }
} }
} }
func BenchmarkString(b *testing.B) {
b.ReportAllocs()
for i := 0; i < b.N; i++ {
String(32)
}
}

View File

@ -21,6 +21,7 @@ import (
type secureSource struct { type secureSource struct {
rd io.Reader rd io.Reader
mut sync.Mutex mut sync.Mutex
buf [8]byte
} }
func newSecureSource() *secureSource { func newSecureSource() *secureSource {
@ -47,18 +48,15 @@ func (s *secureSource) Read(p []byte) (int, error) {
} }
func (s *secureSource) Uint64() uint64 { func (s *secureSource) Uint64() uint64 {
var buf [8]byte
// Read eight bytes of entropy from the buffered, secure random number // Read eight bytes of entropy from the buffered, secure random number
// generator. The buffered reader isn't concurrency safe, so we lock // generator. The buffered reader isn't concurrency safe, so we lock
// around that. // around that.
s.mut.Lock() s.mut.Lock()
_, err := io.ReadFull(s.rd, buf[:]) defer s.mut.Unlock()
s.mut.Unlock()
_, err := io.ReadFull(s.rd, s.buf[:])
if err != nil { if err != nil {
panic("randomness failure: " + err.Error()) panic("randomness failure: " + err.Error())
} }
return binary.LittleEndian.Uint64(s.buf[:])
// Grab those bytes as an uint64
return binary.BigEndian.Uint64(buf[:])
} }