mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-18 15:54:13 +00:00
678ffa607e
* fix bitfield * test * fix reference * fix tests * remove test * fix test * add new helper * add test * fix tests * fix test * gaz * add continue
90 lines
2.2 KiB
Go
90 lines
2.2 KiB
Go
package bitutil
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/mathutil"
|
|
"github.com/steakknife/hamming"
|
|
)
|
|
|
|
// SetBitfield takes an index and returns bitfield with the index flipped.
|
|
func SetBitfield(index int, committeeLength int) ([]byte, error) {
|
|
if index >= committeeLength {
|
|
return nil, fmt.Errorf("invalid index, as index %d is more than"+
|
|
" or equal to committee length %d", index, committeeLength)
|
|
}
|
|
chunkLocation := index / 8
|
|
indexLocation := mathutil.PowerOf2(uint64(7 - (index % 8)))
|
|
var bitfield []byte
|
|
|
|
for i := 0; i < chunkLocation; i++ {
|
|
bitfield = append(bitfield, byte(0))
|
|
}
|
|
bitfield = append(bitfield, byte(indexLocation))
|
|
|
|
for len(bitfield) < mathutil.CeilDiv8(committeeLength) {
|
|
bitfield = append(bitfield, byte(0))
|
|
}
|
|
|
|
return bitfield, nil
|
|
}
|
|
|
|
// CheckBit checks if a bit in a bit field (small endian) is one.
|
|
func CheckBit(bitfield []byte, index int) (bool, error) {
|
|
chunkLocation := (index + 1) / 8
|
|
indexLocation := (index + 1) % 8
|
|
|
|
if indexLocation == 0 {
|
|
indexLocation = 8
|
|
} else {
|
|
chunkLocation++
|
|
}
|
|
if chunkLocation > len(bitfield) {
|
|
return false, fmt.Errorf("index out of range for bitfield: length: %d, position: %d ",
|
|
len(bitfield), chunkLocation-1)
|
|
}
|
|
|
|
field := bitfield[chunkLocation-1] >> (7 - uint(indexLocation-1))
|
|
return field%2 != 0, nil
|
|
}
|
|
|
|
// BitSetCount counts the number of 1s in a byte using Hamming weight.
|
|
// See: https://en.wikipedia.org/wiki/Hamming_weight
|
|
func BitSetCount(b []byte) int {
|
|
return hamming.CountBitsBytes(b)
|
|
}
|
|
|
|
// BitLength returns the length of the bitfield in bytes.
|
|
func BitLength(b int) int {
|
|
return (b + 7) / 8
|
|
}
|
|
|
|
// FillBitfield returns a bitfield of length `count`, all set to true.
|
|
func FillBitfield(count int) []byte {
|
|
numChunks := count/8 + 1
|
|
bitfield := make([]byte, numChunks)
|
|
for i := 0; i < numChunks; i++ {
|
|
if i+1 == numChunks {
|
|
bitfield[i] = fillNBits(uint64(count % 8))
|
|
} else {
|
|
bitfield[i] = byte(8)
|
|
}
|
|
}
|
|
|
|
return bitfield
|
|
}
|
|
|
|
func fillNBits(numBits uint64) byte {
|
|
result := byte(0)
|
|
for i := uint64(0); i < numBits; i++ {
|
|
result = fillBit(result, i)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func fillBit(target byte, index uint64) byte {
|
|
bitShift := 7 - index
|
|
return target ^ (1 << bitShift)
|
|
}
|