mirror of
https://github.com/safing/portmaster
synced 2025-04-20 10:59:10 +00:00
* Move portbase into monorepo * Add new simple module mgr * [WIP] Switch to new simple module mgr * Add StateMgr and more worker variants * [WIP] Switch more modules * [WIP] Switch more modules * [WIP] swtich more modules * [WIP] switch all SPN modules * [WIP] switch all service modules * [WIP] Convert all workers to the new module system * [WIP] add new task system to module manager * [WIP] Add second take for scheduling workers * [WIP] Add FIXME for bugs in new scheduler * [WIP] Add minor improvements to scheduler * [WIP] Add new worker scheduler * [WIP] Fix more bug related to new module system * [WIP] Fix start handing of the new module system * [WIP] Improve startup process * [WIP] Fix minor issues * [WIP] Fix missing subsystem in settings * [WIP] Initialize managers in constructor * [WIP] Move module event initialization to constrictors * [WIP] Fix setting for enabling and disabling the SPN module * [WIP] Move API registeration into module construction * [WIP] Update states mgr for all modules * [WIP] Add CmdLine operation support * Add state helper methods to module group and instance * Add notification and module status handling to status package * Fix starting issues * Remove pilot widget and update security lock to new status data * Remove debug logs * Improve http server shutdown * Add workaround for cleanly shutting down firewall+netquery * Improve logging * Add syncing states with notifications for new module system * Improve starting, stopping, shutdown; resolve FIXMEs/TODOs * [WIP] Fix most unit tests * Review new module system and fix minor issues * Push shutdown and restart events again via API * Set sleep mode via interface * Update example/template module * [WIP] Fix spn/cabin unit test * Remove deprecated UI elements * Make log output more similar for the logging transition phase * Switch spn hub and observer cmds to new module system * Fix log sources * Make worker mgr less error prone * Fix tests and minor issues * Fix observation hub * Improve shutdown and restart handling * Split up big connection.go source file * Move varint and dsd packages to structures repo * Improve expansion test * Fix linter warnings * Fix interception module on windows * Fix linter errors --------- Co-authored-by: Vladimir Stoilov <vladimir@safing.io>
94 lines
1.8 KiB
Go
94 lines
1.8 KiB
Go
package rng
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"errors"
|
|
"io"
|
|
"math"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
reseedAfterSeconds = 600 // ten minutes
|
|
reseedAfterBytes = 1048576 // one megabyte
|
|
)
|
|
|
|
var (
|
|
// Reader provides a global instance to read from the RNG.
|
|
Reader io.Reader
|
|
|
|
rngBytesRead uint64
|
|
rngLastFeed = time.Now()
|
|
)
|
|
|
|
// reader provides an io.Reader interface.
|
|
type reader struct{}
|
|
|
|
func init() {
|
|
Reader = reader{}
|
|
}
|
|
|
|
func checkEntropy() (err error) {
|
|
if !rngReady {
|
|
return errors.New("RNG is not ready yet")
|
|
}
|
|
if rngBytesRead > reseedAfterBytes ||
|
|
int(time.Since(rngLastFeed).Seconds()) > reseedAfterSeconds {
|
|
select {
|
|
case r := <-rngFeeder:
|
|
rng.Reseed(r)
|
|
rngBytesRead = 0
|
|
rngLastFeed = time.Now()
|
|
case <-time.After(1 * time.Second):
|
|
return errors.New("failed to get new entropy")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Read reads random bytes into the supplied byte slice.
|
|
func Read(b []byte) (n int, err error) {
|
|
rngLock.Lock()
|
|
defer rngLock.Unlock()
|
|
|
|
if err := checkEntropy(); err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return copy(b, rng.PseudoRandomData(uint(len(b)))), nil
|
|
}
|
|
|
|
// Read implements the io.Reader interface.
|
|
func (r reader) Read(b []byte) (n int, err error) {
|
|
return Read(b)
|
|
}
|
|
|
|
// Bytes allocates a new byte slice of given length and fills it with random data.
|
|
func Bytes(n int) ([]byte, error) {
|
|
rngLock.Lock()
|
|
defer rngLock.Unlock()
|
|
|
|
if err := checkEntropy(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return rng.PseudoRandomData(uint(n)), nil
|
|
}
|
|
|
|
// Number returns a random number from 0 to (incl.) max.
|
|
func Number(max uint64) (uint64, error) {
|
|
secureLimit := math.MaxUint64 - (math.MaxUint64 % max)
|
|
max++
|
|
|
|
for {
|
|
randomBytes, err := Bytes(8)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
candidate := binary.LittleEndian.Uint64(randomBytes)
|
|
if candidate < secureLimit {
|
|
return candidate % max, nil
|
|
}
|
|
}
|
|
}
|