livekit-protocol 0.7.5

Livekit protocol and utilities for the Rust SDK
Documentation
// 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 "math/bits"

type bitmapNumber interface {
	uint8 | uint16 | uint32 | uint64
}

type Bitmap[T bitmapNumber] struct {
	bits []uint64
}

func NewBitmap[T bitmapNumber](size int) *Bitmap[T] {
	return &Bitmap[T]{
		bits: make([]uint64, 1<<bits.Len64(uint64(size+63)/64)),
	}
}

func (b *Bitmap[T]) Len() int {
	return len(b.bits) << 6
}

func (b *Bitmap[T]) Set(val T) {
	sm, s, o := b.getSlotAndOffset(val)
	b.bits[s&sm] |= 1 << o
}

func (b *Bitmap[T]) GetAndSet(val T) bool {
	sm, s, o := b.getSlotAndOffset(val)
	prev := b.bits[s&sm]&(1<<o) != 0
	b.bits[s&sm] |= 1 << o
	return prev
}

func (b *Bitmap[T]) SetRange(min, max T) {
	if max < min {
		return
	}

	sm, ls, rs, lo, ro := b.getSlotsAndOffsets(min, max)
	if ls == rs {
		b.bits[ls&sm] |= (((1 << (ro - lo + 1)) - 1) << lo)
	} else {
		b.bits[ls&sm] |= ^((1 << lo) - 1)
		for i := ls + 1; i < rs; i++ {
			b.bits[i&sm] = ^uint64(0)
		}
		b.bits[rs&sm] |= (1 << (ro + 1)) - 1
	}
}

func (b *Bitmap[T]) Clear(val T) {
	sm, s, o := b.getSlotAndOffset(val)
	b.bits[s&sm] &= ^(1 << o)
}

func (b *Bitmap[T]) ClearRange(min, max T) {
	if max < min {
		return
	}

	sm, ls, rs, lo, ro := b.getSlotsAndOffsets(min, max)
	if ls == rs {
		b.bits[ls&sm] &= ^(((1 << (ro - lo + 1)) - 1) << lo)
	} else {
		b.bits[ls&sm] &= ^uint64(0) >> (64 - lo)
		for i := ls + 1; i < rs; i++ {
			b.bits[i&sm] = 0
		}
		b.bits[rs&sm] &= ^uint64(0) << (ro + 1)
	}
}

func (b *Bitmap[T]) IsSet(val T) bool {
	sm, s, o := b.getSlotAndOffset(val)
	return b.bits[s&sm]&(1<<o) != 0
}

func (b *Bitmap[T]) getSlotAndOffset(val T) (sm, s, o int) {
	sm = len(b.bits) - 1 // slot mask
	s = int(val >> 6)    // slot
	o = int(val & 0x3f)  // offset
	return
}

func (b *Bitmap[T]) getSlotsAndOffsets(min, max T) (sm, ls, rs, lo, ro int) {
	sm = len(b.bits) - 1 // slot mask

	ls = int(min >> 6) // left slot
	rs = int(max >> 6) // right slot

	if rs-ls > len(b.bits) {
		rs = ls + len(b.bits)
		return
	}

	lo = int(min & 0x3f) // left offset
	ro = int(max & 0x3f) // right offset
	return
}