mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-05 12:57:38 +03:00
61 lines
961 B
Go
61 lines
961 B
Go
package random
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"encoding/binary"
|
|
"io"
|
|
"sync"
|
|
|
|
"github.com/sagernet/sing/common"
|
|
"lukechampine.com/blake3"
|
|
)
|
|
|
|
var System = &Source{rand.Reader}
|
|
|
|
var Default = Blake3KeyedHash()
|
|
|
|
func Blake3KeyedHash() *Source {
|
|
key := make([]byte, 32)
|
|
common.Must1(io.ReadFull(System, key))
|
|
h := blake3.New(1024, key)
|
|
return &Source{&SyncReader{Reader: h.XOF()}}
|
|
}
|
|
|
|
type SyncReader struct {
|
|
io.Reader
|
|
sync.Mutex
|
|
}
|
|
|
|
func (r *SyncReader) Read(p []byte) (n int, err error) {
|
|
r.Lock()
|
|
defer r.Unlock()
|
|
return r.Reader.Read(p)
|
|
}
|
|
|
|
const (
|
|
rngMax = 1 << 63
|
|
rngMask = rngMax - 1
|
|
)
|
|
|
|
type Source struct {
|
|
io.Reader
|
|
}
|
|
|
|
func (s Source) Int63() int64 {
|
|
return s.Int64() & rngMask
|
|
}
|
|
|
|
func (s Source) Int64() int64 {
|
|
var num int64
|
|
common.Must(binary.Read(s, binary.BigEndian, &num))
|
|
return num
|
|
}
|
|
|
|
func (s Source) Uint64() uint64 {
|
|
var num uint64
|
|
common.Must(binary.Read(s, binary.BigEndian, &num))
|
|
return num
|
|
}
|
|
|
|
func (s Source) Seed(int64) {
|
|
}
|