1
Fork 0
galene/rtptime/rtptime.go

76 lines
1.9 KiB
Go

// Package rtptime manipulates RTP and NTP time
package rtptime
import (
"math/bits"
"time"
)
// epoch is the origin of NTP time
var epoch = time.Now()
// FromDuration converts a time.Duration into units of 1/hz.
func FromDuration(d time.Duration, hz uint32) int64 {
if d < 0 {
return -FromDuration(d.Abs(), hz)
}
hi, lo := bits.Mul64(uint64(d), uint64(hz))
q, _ := bits.Div64(hi, lo, uint64(time.Second))
return int64(q)
}
// ToDuration converts units of 1/hz into a time.Duration.
func ToDuration(tm int64, hz uint32) time.Duration {
if tm < 0 {
return -ToDuration(-tm, hz)
}
hi, lo := bits.Mul64(uint64(tm), uint64(time.Second))
q, _ := bits.Div64(hi, lo, uint64(hz))
return time.Duration(q)
}
// Now returns the current time in units of 1/hz from an arbitrary origin.
func Now(hz uint32) uint64 {
return uint64(FromDuration(time.Since(epoch), hz))
}
// Microseconds is like Now, but uses microseconds.
func Microseconds() uint64 {
return Now(1000000)
}
// JiffiesPerSec is the number of jiffies in a second. This is the LCM of
// 48000, 96000 and 65536.
const JiffiesPerSec = 24576000
// Jiffies returns the current time in jiffies.
func Jiffies() uint64 {
return Now(JiffiesPerSec)
}
// TimeToJiffies converts a time.Time into jiffies.
func TimeToJiffies(tm time.Time) uint64 {
return uint64(FromDuration(tm.Sub(epoch), JiffiesPerSec))
}
// The origin of NTP time.
var ntpEpoch = time.Date(1900, 1, 1, 0, 0, 0, 0, time.UTC)
// NTPToTime converts an NTP time into a time.Time.
func NTPToTime(ntp uint64) time.Time {
sec := uint32(ntp >> 32)
frac := uint32(ntp & 0xFFFFFFFF)
return ntpEpoch.Add(
time.Duration(sec)*time.Second +
((time.Duration(frac) * time.Second) >> 32),
)
}
// TimeToNTP converts a time.Time into an NTP time.
func TimeToNTP(tm time.Time) uint64 {
d := tm.Sub(ntpEpoch)
sec := uint32(d / time.Second)
frac := uint32(d % time.Second)
return (uint64(sec) << 32) + (uint64(frac)<<32)/uint64(time.Second)
}