This commit is contained in:
Aliaksandr Valialkin 2024-06-08 01:15:58 +02:00
parent dc99281295
commit 62fc9479b2
No known key found for this signature in database
GPG key ID: 52C003EE2BCDB9EB
3 changed files with 64 additions and 13 deletions

View file

@ -98,13 +98,6 @@ func (bm *bitmap) areAllBitsSet() bool {
return true return true
} }
func (bm *bitmap) isSetBit(i int) bool {
wordIdx := uint(i) / 64
wordOffset := uint(i) % 64
word := bm.a[wordIdx]
return (word & (1 << wordOffset)) != 0
}
func (bm *bitmap) andNot(x *bitmap) { func (bm *bitmap) andNot(x *bitmap) {
if bm.bitsLen != x.bitsLen { if bm.bitsLen != x.bitsLen {
logger.Panicf("BUG: cannot merge bitmaps with distinct lengths; %d vs %d", bm.bitsLen, x.bitsLen) logger.Panicf("BUG: cannot merge bitmaps with distinct lengths; %d vs %d", bm.bitsLen, x.bitsLen)
@ -127,6 +120,13 @@ func (bm *bitmap) or(x *bitmap) {
} }
} }
func (bm *bitmap) isSetBit(i int) bool {
wordIdx := uint(i) / 64
wordOffset := uint(i) % 64
word := bm.a[wordIdx]
return (word & (1 << wordOffset)) != 0
}
// forEachSetBit calls f for each set bit and clears that bit if f returns false // forEachSetBit calls f for each set bit and clears that bit if f returns false
func (bm *bitmap) forEachSetBit(f func(idx int) bool) { func (bm *bitmap) forEachSetBit(f func(idx int) bool) {
a := bm.a a := bm.a
@ -143,7 +143,7 @@ func (bm *bitmap) forEachSetBit(f func(idx int) bool) {
} }
idx := i*64 + j idx := i*64 + j
if idx >= bitsLen { if idx >= bitsLen {
break return
} }
if !f(idx) { if !f(idx) {
wordNew &= ^mask wordNew &= ^mask
@ -178,7 +178,7 @@ func (bm *bitmap) forEachSetBitReadonly(f func(idx int)) {
} }
idx := i*64 + j idx := i*64 + j
if idx >= bitsLen { if idx >= bitsLen {
break return
} }
f(idx) f(idx)
} }

View file

@ -6,6 +6,8 @@ import (
func TestBitmap(t *testing.T) { func TestBitmap(t *testing.T) {
for i := 0; i < 100; i++ { for i := 0; i < 100; i++ {
bitsLen := i
bm := getBitmap(i) bm := getBitmap(i)
if bm.bitsLen != i { if bm.bitsLen != i {
t.Fatalf("unexpected bits length: %d; want %d", bm.bitsLen, i) t.Fatalf("unexpected bits length: %d; want %d", bm.bitsLen, i)
@ -41,6 +43,9 @@ func TestBitmap(t *testing.T) {
} }
nextIdx++ nextIdx++
}) })
if nextIdx != bitsLen {
t.Fatalf("unexpected number of bits set; got %d; want %d", nextIdx, bitsLen)
}
if !bm.areAllBitsSet() { if !bm.areAllBitsSet() {
t.Fatalf("all bits must be set for bitmap with %d bits", i) t.Fatalf("all bits must be set for bitmap with %d bits", i)
@ -71,6 +76,9 @@ func TestBitmap(t *testing.T) {
} }
nextIdx += 2 nextIdx += 2
}) })
if nextIdx < bitsLen {
t.Fatalf("unexpected number of bits visited; got %d; want %d", nextIdx, bitsLen)
}
// Clear all the bits // Clear all the bits
bm.forEachSetBit(func(_ int) bool { bm.forEachSetBit(func(_ int) bool {

View file

@ -4,6 +4,35 @@ import (
"testing" "testing"
) )
func BenchmarkBitmapIsSetBit(b *testing.B) {
const bitsLen = 64 * 1024
b.Run("no-zero-bits", func(b *testing.B) {
bm := getBitmap(bitsLen)
bm.setBits()
benchmarkBitmapIsSetBit(b, bm)
putBitmap(bm)
})
b.Run("half-zero-bits", func(b *testing.B) {
bm := getBitmap(bitsLen)
bm.setBits()
bm.forEachSetBit(func(idx int) bool {
return idx%2 == 0
})
benchmarkBitmapIsSetBit(b, bm)
putBitmap(bm)
})
b.Run("one-set-bit", func(b *testing.B) {
bm := getBitmap(bitsLen)
bm.setBits()
bm.forEachSetBit(func(idx int) bool {
return idx == bitsLen/2
})
benchmarkBitmapIsSetBit(b, bm)
putBitmap(bm)
})
}
func BenchmarkBitmapForEachSetBitReadonly(b *testing.B) { func BenchmarkBitmapForEachSetBitReadonly(b *testing.B) {
const bitsLen = 64 * 1024 const bitsLen = 64 * 1024
@ -86,19 +115,33 @@ func BenchmarkBitmapForEachSetBit(b *testing.B) {
}) })
} }
func benchmarkBitmapIsSetBit(b *testing.B, bm *bitmap) {
bitsLen := bm.bitsLen
b.SetBytes(int64(bitsLen))
b.ReportAllocs()
b.RunParallel(func(pb *testing.PB) {
n := 0
for pb.Next() {
for i := 0; i < bitsLen; i++ {
if bm.isSetBit(i) {
n++
}
}
}
GlobalSink.Add(uint64(n))
})
}
func benchmarkBitmapForEachSetBitReadonly(b *testing.B, bm *bitmap) { func benchmarkBitmapForEachSetBitReadonly(b *testing.B, bm *bitmap) {
b.SetBytes(int64(bm.bitsLen)) b.SetBytes(int64(bm.bitsLen))
b.ReportAllocs() b.ReportAllocs()
b.RunParallel(func(pb *testing.PB) { b.RunParallel(func(pb *testing.PB) {
bmLocal := getBitmap(bm.bitsLen)
n := 0 n := 0
for pb.Next() { for pb.Next() {
bmLocal.copyFrom(bm) bm.forEachSetBitReadonly(func(_ int) {
bmLocal.forEachSetBitReadonly(func(_ int) {
n++ n++
}) })
} }
putBitmap(bmLocal)
GlobalSink.Add(uint64(n)) GlobalSink.Add(uint64(n))
}) })
} }