mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 12:47:36 +03:00
move the random number generator to the utils package
This commit is contained in:
parent
c72f05aa41
commit
3c0726e132
3 changed files with 62 additions and 26 deletions
|
@ -1,9 +1,6 @@
|
||||||
package ackhandler
|
package ackhandler
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/rand"
|
|
||||||
"encoding/binary"
|
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/internal/protocol"
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
||||||
"github.com/lucas-clemente/quic-go/internal/utils"
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
||||||
)
|
)
|
||||||
|
@ -33,28 +30,6 @@ func (p *sequentialPacketNumberGenerator) Pop() protocol.PacketNumber {
|
||||||
return next
|
return next
|
||||||
}
|
}
|
||||||
|
|
||||||
type rng struct {
|
|
||||||
buf [4]byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *rng) Int31() int32 {
|
|
||||||
rand.Read(r.buf[:])
|
|
||||||
return int32(binary.BigEndian.Uint32(r.buf[:]) & ^uint32(1<<31))
|
|
||||||
}
|
|
||||||
|
|
||||||
// copied from the standard library math/rand implementation of Int63n
|
|
||||||
func (r *rng) Int31n(n int32) int32 {
|
|
||||||
if n&(n-1) == 0 { // n is power of two, can mask
|
|
||||||
return r.Int31() & (n - 1)
|
|
||||||
}
|
|
||||||
max := int32((1 << 31) - 1 - (1<<31)%uint32(n))
|
|
||||||
v := r.Int31()
|
|
||||||
for v > max {
|
|
||||||
v = r.Int31()
|
|
||||||
}
|
|
||||||
return v % n
|
|
||||||
}
|
|
||||||
|
|
||||||
// The skippingPacketNumberGenerator generates the packet number for the next packet
|
// The skippingPacketNumberGenerator generates the packet number for the next packet
|
||||||
// it randomly skips a packet number every averagePeriod packets (on average).
|
// it randomly skips a packet number every averagePeriod packets (on average).
|
||||||
// It is guaranteed to never skip two consecutive packet numbers.
|
// It is guaranteed to never skip two consecutive packet numbers.
|
||||||
|
@ -65,7 +40,7 @@ type skippingPacketNumberGenerator struct {
|
||||||
next protocol.PacketNumber
|
next protocol.PacketNumber
|
||||||
nextToSkip protocol.PacketNumber
|
nextToSkip protocol.PacketNumber
|
||||||
|
|
||||||
rng rng
|
rng utils.Rand
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ packetNumberGenerator = &skippingPacketNumberGenerator{}
|
var _ packetNumberGenerator = &skippingPacketNumberGenerator{}
|
||||||
|
|
29
internal/utils/rand.go
Normal file
29
internal/utils/rand.go
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/rand"
|
||||||
|
"encoding/binary"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Rand is a wrapper around crypto/rand that adds some convenience functions known from math/rand.
|
||||||
|
type Rand struct {
|
||||||
|
buf [4]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Rand) Int31() int32 {
|
||||||
|
rand.Read(r.buf[:])
|
||||||
|
return int32(binary.BigEndian.Uint32(r.buf[:]) & ^uint32(1<<31))
|
||||||
|
}
|
||||||
|
|
||||||
|
// copied from the standard library math/rand implementation of Int63n
|
||||||
|
func (r *Rand) Int31n(n int32) int32 {
|
||||||
|
if n&(n-1) == 0 { // n is power of two, can mask
|
||||||
|
return r.Int31() & (n - 1)
|
||||||
|
}
|
||||||
|
max := int32((1 << 31) - 1 - (1<<31)%uint32(n))
|
||||||
|
v := r.Int31()
|
||||||
|
for v > max {
|
||||||
|
v = r.Int31()
|
||||||
|
}
|
||||||
|
return v % n
|
||||||
|
}
|
32
internal/utils/rand_test.go
Normal file
32
internal/utils/rand_test.go
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
. "github.com/onsi/ginkgo"
|
||||||
|
. "github.com/onsi/gomega"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ = Describe("Rand", func() {
|
||||||
|
It("generates random numbers", func() {
|
||||||
|
const (
|
||||||
|
num = 1000
|
||||||
|
max = 123456
|
||||||
|
)
|
||||||
|
|
||||||
|
var values [num]int32
|
||||||
|
var r Rand
|
||||||
|
for i := 0; i < num; i++ {
|
||||||
|
v := r.Int31n(max)
|
||||||
|
Expect(v).To(And(
|
||||||
|
BeNumerically(">=", 0),
|
||||||
|
BeNumerically("<", max),
|
||||||
|
))
|
||||||
|
values[i] = v
|
||||||
|
}
|
||||||
|
|
||||||
|
var sum uint64
|
||||||
|
for _, n := range values {
|
||||||
|
sum += uint64(n)
|
||||||
|
}
|
||||||
|
Expect(float64(sum) / num).To(BeNumerically("~", max/2, max/25))
|
||||||
|
})
|
||||||
|
})
|
Loading…
Add table
Add a link
Reference in a new issue