mirror of
https://github.com/stoatchat/livekit-protocol.git
synced 2026-07-01 22:04:52 -04:00
c9e9ae555f
* Add license headers & NOTICE * generated protobuf --------- Co-authored-by: github-actions <41898282+github-actions[bot]@users.noreply.github.com>
170 lines
3.7 KiB
Go
170 lines
3.7 KiB
Go
// Copyright 2023 LiveKit, Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package utils
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
"time"
|
|
|
|
"go.uber.org/atomic"
|
|
|
|
"github.com/livekit/protocol/livekit"
|
|
)
|
|
|
|
const tickBits uint64 = 13
|
|
const tickMask uint64 = (1 << tickBits) - 1
|
|
|
|
var epoch = time.Date(2000, 0, 0, 0, 0, 0, 0, time.UTC).UnixMicro()
|
|
|
|
type TimedVersionGenerator interface {
|
|
New() *TimedVersion
|
|
Next() TimedVersion
|
|
}
|
|
|
|
func timedVersionComponents(v uint64) (ts int64, ticks int32) {
|
|
return int64(v>>tickBits) + epoch, int32(v & tickMask)
|
|
}
|
|
|
|
func timedVersionFromComponents(ts int64, ticks int32) TimedVersion {
|
|
if ts < epoch {
|
|
ts = epoch
|
|
}
|
|
return TimedVersion{v: *atomic.NewUint64((uint64(ts-epoch) << tickBits) | uint64(ticks))}
|
|
}
|
|
|
|
type timedVersionGenerator struct {
|
|
mu sync.Mutex
|
|
ts int64
|
|
ticks uint64
|
|
}
|
|
|
|
func NewDefaultTimedVersionGenerator() TimedVersionGenerator {
|
|
return new(timedVersionGenerator)
|
|
}
|
|
|
|
func (g *timedVersionGenerator) New() *TimedVersion {
|
|
v := g.Next()
|
|
return &v
|
|
}
|
|
|
|
func (g *timedVersionGenerator) Next() TimedVersion {
|
|
ts := time.Now().UnixMicro()
|
|
|
|
g.mu.Lock()
|
|
defer g.mu.Unlock()
|
|
|
|
for {
|
|
if ts < g.ts {
|
|
ts = g.ts
|
|
}
|
|
if g.ts == ts {
|
|
// if incrementing the ticks would overflow the version sleep for a
|
|
// microsecond then try again.
|
|
if g.ticks == tickMask {
|
|
time.Sleep(time.Microsecond)
|
|
ts = time.Now().UnixMicro()
|
|
continue
|
|
}
|
|
g.ticks++
|
|
} else {
|
|
g.ts = ts
|
|
g.ticks = 0
|
|
}
|
|
return timedVersionFromComponents(g.ts, int32(g.ticks))
|
|
}
|
|
}
|
|
|
|
type TimedVersion struct {
|
|
v atomic.Uint64
|
|
}
|
|
|
|
func NewTimedVersionFromProto(proto *livekit.TimedVersion) *TimedVersion {
|
|
v := timedVersionFromComponents(proto.GetUnixMicro(), proto.GetTicks())
|
|
return &v
|
|
}
|
|
|
|
func NewTimedVersionFromTime(t time.Time) *TimedVersion {
|
|
v := timedVersionFromComponents(t.UnixMicro(), 0)
|
|
return &v
|
|
}
|
|
|
|
func TimedVersionFromProto(proto *livekit.TimedVersion) TimedVersion {
|
|
return timedVersionFromComponents(proto.GetUnixMicro(), proto.GetTicks())
|
|
}
|
|
|
|
func TimedVersionFromTime(t time.Time) TimedVersion {
|
|
return timedVersionFromComponents(t.UnixMicro(), 0)
|
|
}
|
|
|
|
func (t *TimedVersion) Update(other *TimedVersion) bool {
|
|
ov := other.v.Load()
|
|
for {
|
|
prev := t.v.Load()
|
|
if ov <= prev {
|
|
return false
|
|
}
|
|
if t.v.CompareAndSwap(prev, ov) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
func (t *TimedVersion) Store(other *TimedVersion) {
|
|
t.v.Store(other.v.Load())
|
|
}
|
|
|
|
func (t *TimedVersion) Load() TimedVersion {
|
|
return TimedVersion{v: *atomic.NewUint64(t.v.Load())}
|
|
}
|
|
|
|
func (t *TimedVersion) After(other *TimedVersion) bool {
|
|
return t.v.Load() > other.v.Load()
|
|
}
|
|
|
|
func (t *TimedVersion) Compare(other *TimedVersion) int {
|
|
ov := other.v.Load()
|
|
v := t.v.Load()
|
|
if v < ov {
|
|
return -1
|
|
}
|
|
if v == ov {
|
|
return 0
|
|
}
|
|
return 1
|
|
}
|
|
|
|
func (t *TimedVersion) IsZero() bool {
|
|
return t.v.Load() == 0
|
|
}
|
|
|
|
func (t *TimedVersion) ToProto() *livekit.TimedVersion {
|
|
ts, ticks := timedVersionComponents(t.v.Load())
|
|
return &livekit.TimedVersion{
|
|
UnixMicro: ts,
|
|
Ticks: ticks,
|
|
}
|
|
}
|
|
|
|
func (t *TimedVersion) Time() time.Time {
|
|
ts, _ := timedVersionComponents(t.v.Load())
|
|
return time.UnixMicro(ts)
|
|
}
|
|
|
|
func (t *TimedVersion) String() string {
|
|
ts, ticks := timedVersionComponents(t.v.Load())
|
|
return fmt.Sprintf("%d.%d", ts, ticks)
|
|
}
|