Commit 93136576 authored by Carlo Alberto Ferraris's avatar Carlo Alberto Ferraris Committed by Emmanuel Odeke

math/rand: devirtualize interface in lockedSource

Avoid interface calls, enable inlining, and store the rngSource close to the
Mutex to exploit better memory locality.

Also add a benchmark to properly measure the threadsafe nature of globalRand.

On a linux/amd64 VM:

name                       old time/op  new time/op  delta
Int63Threadsafe-4          36.4ns ±12%  20.6ns ±11%  -43.52%  (p=0.000 n=30+30)
Int63ThreadsafeParallel-4  79.3ns ± 5%  56.5ns ± 5%  -28.69%  (p=0.000 n=29+30)

Change-Id: I6ab912c1a1e9afc7bacd8e72c82d4d50d546a510
Reviewed-on: https://go-review.googlesource.com/c/go/+/191538Reviewed-by: default avatarEmmanuel Odeke <emm.odeke@gmail.com>
Run-TryBot: Emmanuel Odeke <emm.odeke@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent 57662b15
...@@ -148,6 +148,10 @@ func TestIntendedInlining(t *testing.T) { ...@@ -148,6 +148,10 @@ func TestIntendedInlining(t *testing.T) {
"addVW", "addVW",
"subVW", "subVW",
}, },
"math/rand": {
"(*rngSource).Int63",
"(*rngSource).Uint64",
},
} }
if runtime.GOARCH != "386" && runtime.GOARCH != "mips64" && runtime.GOARCH != "mips64le" { if runtime.GOARCH != "386" && runtime.GOARCH != "mips64" && runtime.GOARCH != "mips64le" {
......
...@@ -285,7 +285,10 @@ func read(p []byte, int63 func() int64, readVal *int64, readPos *int8) (n int, e ...@@ -285,7 +285,10 @@ func read(p []byte, int63 func() int64, readVal *int64, readPos *int8) (n int, e
* Top-level convenience functions * Top-level convenience functions
*/ */
var globalRand = New(&lockedSource{src: NewSource(1).(Source64)}) var globalRand = New(&lockedSource{src: NewSource(1).(*rngSource)})
// Type assert that globalRand's source is a lockedSource whose src is a *rngSource.
var _ *rngSource = globalRand.src.(*lockedSource).src
// Seed uses the provided seed value to initialize the default Source to a // Seed uses the provided seed value to initialize the default Source to a
// deterministic state. If Seed is not called, the generator behaves as // deterministic state. If Seed is not called, the generator behaves as
...@@ -373,7 +376,7 @@ func ExpFloat64() float64 { return globalRand.ExpFloat64() } ...@@ -373,7 +376,7 @@ func ExpFloat64() float64 { return globalRand.ExpFloat64() }
type lockedSource struct { type lockedSource struct {
lk sync.Mutex lk sync.Mutex
src Source64 src *rngSource
} }
func (r *lockedSource) Int63() (n int64) { func (r *lockedSource) Int63() (n int64) {
......
...@@ -565,6 +565,14 @@ func BenchmarkInt63Threadsafe(b *testing.B) { ...@@ -565,6 +565,14 @@ func BenchmarkInt63Threadsafe(b *testing.B) {
} }
} }
func BenchmarkInt63ThreadsafeParallel(b *testing.B) {
b.RunParallel(func(pb *testing.PB) {
for pb.Next() {
Int63()
}
})
}
func BenchmarkInt63Unthreadsafe(b *testing.B) { func BenchmarkInt63Unthreadsafe(b *testing.B) {
r := New(NewSource(1)) r := New(NewSource(1))
for n := b.N; n > 0; n-- { for n := b.N; n > 0; n-- {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment