ledisdb/ledis/t_bit.go

932 lines
21 KiB
Go
Raw Permalink Normal View History

2014-06-30 06:50:42 +04:00
package ledis
2015-03-03 04:21:00 +03:00
// import (
// "encoding/binary"
// "errors"
// "github.com/siddontang/go/log"
// "github.com/siddontang/go/num"
// "github.com/siddontang/ledisdb/store"
// "sort"
// "time"
// )
// /*
// We will not maintain bitmap anymore, and will add bit operations for kv type later.
// Use your own risk.
// */
// const (
// OPand uint8 = iota + 1
// OPor
// OPxor
// OPnot
// )
// type BitPair struct {
// Pos int32
// Val uint8
// }
// type segBitInfo struct {
// Seq uint32
// Off uint32
// Val uint8
// }
// type segBitInfoArray []segBitInfo
// const (
// // byte
// segByteWidth uint32 = 9
// segByteSize uint32 = 1 << segByteWidth
// // bit
// segBitWidth uint32 = segByteWidth + 3
// segBitSize uint32 = segByteSize << 3
// maxByteSize uint32 = 8 << 20
// maxSegCount uint32 = maxByteSize / segByteSize
// minSeq uint32 = 0
// maxSeq uint32 = uint32((maxByteSize << 3) - 1)
// )
// var fillBits = [...]uint8{1, 3, 7, 15, 31, 63, 127, 255}
// var emptySegment []byte = make([]byte, segByteSize, segByteSize)
// var fillSegment []byte = func() []byte {
// data := make([]byte, segByteSize, segByteSize)
// for i := uint32(0); i < segByteSize; i++ {
// data[i] = 0xff
// }
// return data
// }()
// var errBinKey = errors.New("invalid bin key")
// var errOffset = errors.New("invalid offset")
// var errDuplicatePos = errors.New("duplicate bit pos")
// func getBit(sz []byte, offset uint32) uint8 {
// index := offset >> 3
// if index >= uint32(len(sz)) {
// return 0 // error("overflow")
// }
// offset -= index << 3
// return sz[index] >> offset & 1
// }
// func setBit(sz []byte, offset uint32, val uint8) bool {
// if val != 1 && val != 0 {
// return false // error("invalid val")
// }
// index := offset >> 3
// if index >= uint32(len(sz)) {
// return false // error("overflow")
// }
// offset -= index << 3
// if sz[index]>>offset&1 != val {
// sz[index] ^= (1 << offset)
// }
// return true
// }
// func (datas segBitInfoArray) Len() int {
// return len(datas)
// }
// func (datas segBitInfoArray) Less(i, j int) bool {
// res := (datas)[i].Seq < (datas)[j].Seq
// if !res && (datas)[i].Seq == (datas)[j].Seq {
// res = (datas)[i].Off < (datas)[j].Off
// }
// return res
// }
// func (datas segBitInfoArray) Swap(i, j int) {
// datas[i], datas[j] = datas[j], datas[i]
// }
// func (db *DB) bEncodeMetaKey(key []byte) []byte {
// mk := make([]byte, len(key)+2)
// mk[0] = db.index
// mk[1] = BitMetaType
// copy(mk[2:], key)
// return mk
// }
// func (db *DB) bDecodeMetaKey(bkey []byte) ([]byte, error) {
// if len(bkey) < 2 || bkey[0] != db.index || bkey[1] != BitMetaType {
// return nil, errBinKey
// }
// return bkey[2:], nil
// }
// func (db *DB) bEncodeBinKey(key []byte, seq uint32) []byte {
// bk := make([]byte, len(key)+8)
// pos := 0
// bk[pos] = db.index
// pos++
// bk[pos] = BitType
// pos++
// binary.BigEndian.PutUint16(bk[pos:], uint16(len(key)))
// pos += 2
// copy(bk[pos:], key)
// pos += len(key)
// binary.BigEndian.PutUint32(bk[pos:], seq)
// return bk
// }
// func (db *DB) bDecodeBinKey(bkey []byte) (key []byte, seq uint32, err error) {
// if len(bkey) < 8 || bkey[0] != db.index {
// err = errBinKey
// return
// }
2014-06-30 06:50:42 +04:00
2015-03-03 04:21:00 +03:00
// keyLen := binary.BigEndian.Uint16(bkey[2:4])
// if int(keyLen+8) != len(bkey) {
// err = errBinKey
// return
// }
// key = bkey[4 : 4+keyLen]
// seq = uint32(binary.BigEndian.Uint32(bkey[4+keyLen:]))
// return
// }
// func (db *DB) bCapByteSize(seq uint32, off uint32) uint32 {
// var offByteSize uint32 = (off >> 3) + 1
// if offByteSize > segByteSize {
// offByteSize = segByteSize
// }
// return seq<<segByteWidth + offByteSize
// }
// func (db *DB) bParseOffset(key []byte, offset int32) (seq uint32, off uint32, err error) {
// if offset < 0 {
// if tailSeq, tailOff, e := db.bGetMeta(key); e != nil {
// err = e
// return
// } else if tailSeq >= 0 {
// offset += int32((uint32(tailSeq)<<segBitWidth | uint32(tailOff)) + 1)
// if offset < 0 {
// err = errOffset
// return
// }
// }
// }
// off = uint32(offset)
// seq = off >> segBitWidth
// off &= (segBitSize - 1)
// return
// }
// func (db *DB) bGetMeta(key []byte) (tailSeq int32, tailOff int32, err error) {
// var v []byte
// mk := db.bEncodeMetaKey(key)
// v, err = db.bucket.Get(mk)
// if err != nil {
// return
// }
// if v != nil {
// tailSeq = int32(binary.LittleEndian.Uint32(v[0:4]))
// tailOff = int32(binary.LittleEndian.Uint32(v[4:8]))
// } else {
// tailSeq = -1
// tailOff = -1
// }
// return
// }
// func (db *DB) bSetMeta(t *batch, key []byte, tailSeq uint32, tailOff uint32) {
// ek := db.bEncodeMetaKey(key)
// buf := make([]byte, 8)
// binary.LittleEndian.PutUint32(buf[0:4], tailSeq)
// binary.LittleEndian.PutUint32(buf[4:8], tailOff)
// t.Put(ek, buf)
// return
// }
// func (db *DB) bUpdateMeta(t *batch, key []byte, seq uint32, off uint32) (tailSeq uint32, tailOff uint32, err error) {
// var tseq, toff int32
// var update bool = false
// if tseq, toff, err = db.bGetMeta(key); err != nil {
// return
// } else if tseq < 0 {
// update = true
// } else {
// tailSeq = uint32(num.MaxInt32(tseq, 0))
// tailOff = uint32(num.MaxInt32(toff, 0))
// update = (seq > tailSeq || (seq == tailSeq && off > tailOff))
// }
// if update {
// db.bSetMeta(t, key, seq, off)
// tailSeq = seq
// tailOff = off
// }
2014-06-30 06:50:42 +04:00
// return
// }
2015-03-03 04:21:00 +03:00
// func (db *DB) bDelete(t *batch, key []byte) (drop int64) {
// mk := db.bEncodeMetaKey(key)
// t.Delete(mk)
// minKey := db.bEncodeBinKey(key, minSeq)
// maxKey := db.bEncodeBinKey(key, maxSeq)
// it := db.bucket.RangeIterator(minKey, maxKey, store.RangeClose)
// for ; it.Valid(); it.Next() {
// t.Delete(it.RawKey())
// drop++
// }
// it.Close()
// return drop
// }
// func (db *DB) bGetSegment(key []byte, seq uint32) ([]byte, []byte, error) {
// bk := db.bEncodeBinKey(key, seq)
// segment, err := db.bucket.Get(bk)
// if err != nil {
// return bk, nil, err
// }
// return bk, segment, nil
// }
// func (db *DB) bAllocateSegment(key []byte, seq uint32) ([]byte, []byte, error) {
// bk, segment, err := db.bGetSegment(key, seq)
// if err == nil && segment == nil {
// segment = make([]byte, segByteSize, segByteSize)
// }
// return bk, segment, err
// }
// func (db *DB) bIterator(key []byte) *store.RangeLimitIterator {
// sk := db.bEncodeBinKey(key, minSeq)
// ek := db.bEncodeBinKey(key, maxSeq)
// return db.bucket.RangeIterator(sk, ek, store.RangeClose)
// }
// func (db *DB) bSegAnd(a []byte, b []byte, res *[]byte) {
// if a == nil || b == nil {
// *res = nil
// return
// }
// data := *res
// if data == nil {
// data = make([]byte, segByteSize, segByteSize)
// *res = data
// }
// for i := uint32(0); i < segByteSize; i++ {
// data[i] = a[i] & b[i]
// }
// return
// }
// func (db *DB) bSegOr(a []byte, b []byte, res *[]byte) {
// if a == nil || b == nil {
// if a == nil && b == nil {
// *res = nil
// } else if a == nil {
// *res = b
// } else {
// *res = a
// }
// return
// }
// data := *res
// if data == nil {
// data = make([]byte, segByteSize, segByteSize)
// *res = data
// }
// for i := uint32(0); i < segByteSize; i++ {
// data[i] = a[i] | b[i]
// }
// return
// }
// func (db *DB) bSegXor(a []byte, b []byte, res *[]byte) {
// if a == nil && b == nil {
// *res = fillSegment
// return
// }
// if a == nil {
// a = emptySegment
// }
// if b == nil {
// b = emptySegment
// }
// data := *res
// if data == nil {
// data = make([]byte, segByteSize, segByteSize)
// *res = data
// }
// for i := uint32(0); i < segByteSize; i++ {
// data[i] = a[i] ^ b[i]
// }
// return
// }
// func (db *DB) bExpireAt(key []byte, when int64) (int64, error) {
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// if seq, _, err := db.bGetMeta(key); err != nil || seq < 0 {
// return 0, err
// } else {
// db.expireAt(t, BitType, key, when)
// if err := t.Commit(); err != nil {
// return 0, err
// }
// }
// return 1, nil
// }
// func (db *DB) bCountByte(val byte, soff uint32, eoff uint32) int32 {
// if soff > eoff {
// soff, eoff = eoff, soff
// }
// mask := uint8(0)
// if soff > 0 {
// mask |= fillBits[soff-1]
// }
// if eoff < 7 {
// mask |= (fillBits[7] ^ fillBits[eoff])
// }
// mask = fillBits[7] ^ mask
// return bitsInByte[val&mask]
// }
// func (db *DB) bCountSeg(key []byte, seq uint32, soff uint32, eoff uint32) (cnt int32, err error) {
// if soff >= segBitSize || soff < 0 ||
// eoff >= segBitSize || eoff < 0 {
// return
// }
// var segment []byte
// if _, segment, err = db.bGetSegment(key, seq); err != nil {
// return
// }
// if segment == nil {
// return
// }
// if soff > eoff {
// soff, eoff = eoff, soff
// }
// headIdx := int(soff >> 3)
// endIdx := int(eoff >> 3)
// sByteOff := soff - ((soff >> 3) << 3)
// eByteOff := eoff - ((eoff >> 3) << 3)
// if headIdx == endIdx {
// cnt = db.bCountByte(segment[headIdx], sByteOff, eByteOff)
// } else {
// cnt = db.bCountByte(segment[headIdx], sByteOff, 7) +
// db.bCountByte(segment[endIdx], 0, eByteOff)
// }
// // sum up following bytes
// for idx, end := headIdx+1, endIdx-1; idx <= end; idx += 1 {
// cnt += bitsInByte[segment[idx]]
// if idx == end {
// break
// }
// }
// return
// }
// func (db *DB) BGet(key []byte) (data []byte, err error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if err = checkKeySize(key); err != nil {
// return
// }
// var ts, to int32
// if ts, to, err = db.bGetMeta(key); err != nil || ts < 0 {
// return
// }
// var tailSeq, tailOff = uint32(ts), uint32(to)
// var capByteSize uint32 = db.bCapByteSize(tailSeq, tailOff)
// data = make([]byte, capByteSize, capByteSize)
// minKey := db.bEncodeBinKey(key, minSeq)
// maxKey := db.bEncodeBinKey(key, tailSeq)
// it := db.bucket.RangeIterator(minKey, maxKey, store.RangeClose)
// var seq, s, e uint32
// for ; it.Valid(); it.Next() {
// if _, seq, err = db.bDecodeBinKey(it.RawKey()); err != nil {
// data = nil
// break
// }
// s = seq << segByteWidth
// e = num.MinUint32(s+segByteSize, capByteSize)
// copy(data[s:e], it.RawValue())
// }
// it.Close()
// return
// }
// func (db *DB) BDelete(key []byte) (drop int64, err error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if err = checkKeySize(key); err != nil {
// return
// }
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// drop = db.bDelete(t, key)
// db.rmExpire(t, BitType, key)
// err = t.Commit()
// return
// }
// func (db *DB) BSetBit(key []byte, offset int32, val uint8) (ori uint8, err error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if err = checkKeySize(key); err != nil {
// return
// }
// // todo : check offset
// var seq, off uint32
// if seq, off, err = db.bParseOffset(key, offset); err != nil {
// return 0, err
// }
// var bk, segment []byte
// if bk, segment, err = db.bAllocateSegment(key, seq); err != nil {
// return 0, err
// }
// if segment != nil {
// ori = getBit(segment, off)
// if setBit(segment, off, val) {
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// t.Put(bk, segment)
// if _, _, e := db.bUpdateMeta(t, key, seq, off); e != nil {
// err = e
// return
// }
// err = t.Commit()
// }
// }
// return
// }
// func (db *DB) BMSetBit(key []byte, args ...BitPair) (place int64, err error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if err = checkKeySize(key); err != nil {
// return
// }
// // (ps : so as to aviod wasting memory copy while calling db.Get() and batch.Put(),
// // here we sequence the params by pos, so that we can merge the execution of
// // diff pos setting which targets on the same segment respectively. )
// // #1 : sequence request data
// var argCnt = len(args)
// var bitInfos segBitInfoArray = make(segBitInfoArray, argCnt)
// var seq, off uint32
// for i, info := range args {
// if seq, off, err = db.bParseOffset(key, info.Pos); err != nil {
// return
// }
// bitInfos[i].Seq = seq
// bitInfos[i].Off = off
// bitInfos[i].Val = info.Val
// }
// sort.Sort(bitInfos)
// for i := 1; i < argCnt; i++ {
// if bitInfos[i].Seq == bitInfos[i-1].Seq && bitInfos[i].Off == bitInfos[i-1].Off {
// return 0, errDuplicatePos
// }
// }
// // #2 : execute bit set in order
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// var curBinKey, curSeg []byte
// var curSeq, maxSeq, maxOff uint32
// for _, info := range bitInfos {
// if curSeg != nil && info.Seq != curSeq {
// t.Put(curBinKey, curSeg)
// curSeg = nil
// }
// if curSeg == nil {
// curSeq = info.Seq
// if curBinKey, curSeg, err = db.bAllocateSegment(key, info.Seq); err != nil {
// return
// }
// if curSeg == nil {
// continue
// }
// }
// if setBit(curSeg, info.Off, info.Val) {
// maxSeq = info.Seq
// maxOff = info.Off
// place++
// }
// }
// if curSeg != nil {
// t.Put(curBinKey, curSeg)
// }
// // finally, update meta
// if place > 0 {
// if _, _, err = db.bUpdateMeta(t, key, maxSeq, maxOff); err != nil {
// return
// }
// err = t.Commit()
// }
// return
// }
// func (db *DB) BGetBit(key []byte, offset int32) (uint8, error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if seq, off, err := db.bParseOffset(key, offset); err != nil {
// return 0, err
// } else {
// _, segment, err := db.bGetSegment(key, seq)
// if err != nil {
// return 0, err
// }
// if segment == nil {
// return 0, nil
// } else {
// return getBit(segment, off), nil
// }
// }
// }
// // func (db *DB) BGetRange(key []byte, start int32, end int32) ([]byte, error) {
// // section := make([]byte)
// // return
// // }
// func (db *DB) BCount(key []byte, start int32, end int32) (cnt int32, err error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// var sseq, soff uint32
// if sseq, soff, err = db.bParseOffset(key, start); err != nil {
// return
// }
// var eseq, eoff uint32
// if eseq, eoff, err = db.bParseOffset(key, end); err != nil {
// return
// }
// if sseq > eseq || (sseq == eseq && soff > eoff) {
// sseq, eseq = eseq, sseq
// soff, eoff = eoff, soff
// }
// var segCnt int32
// if eseq == sseq {
// if segCnt, err = db.bCountSeg(key, sseq, soff, eoff); err != nil {
// return 0, err
// }
// cnt = segCnt
// } else {
// if segCnt, err = db.bCountSeg(key, sseq, soff, segBitSize-1); err != nil {
// return 0, err
// } else {
// cnt += segCnt
// }
// if segCnt, err = db.bCountSeg(key, eseq, 0, eoff); err != nil {
// return 0, err
// } else {
// cnt += segCnt
// }
// }
// // middle segs
// var segment []byte
// skey := db.bEncodeBinKey(key, sseq)
// ekey := db.bEncodeBinKey(key, eseq)
// it := db.bucket.RangeIterator(skey, ekey, store.RangeOpen)
// for ; it.Valid(); it.Next() {
// segment = it.RawValue()
// for _, bt := range segment {
// cnt += bitsInByte[bt]
// }
// }
// it.Close()
// return
// }
// func (db *DB) BTail(key []byte) (int32, error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// // effective length of data, the highest bit-pos set in history
// tailSeq, tailOff, err := db.bGetMeta(key)
// if err != nil {
// return 0, err
// }
// tail := int32(-1)
// if tailSeq >= 0 {
// tail = int32(uint32(tailSeq)<<segBitWidth | uint32(tailOff))
// }
// return tail, nil
// }
// func (db *DB) BOperation(op uint8, dstkey []byte, srckeys ...[]byte) (blen int32, err error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// // blen -
// // the total bit size of data stored in destination key,
// // that is equal to the size of the longest input string.
// var exeOp func([]byte, []byte, *[]byte)
// switch op {
// case OPand:
// exeOp = db.bSegAnd
// case OPor:
// exeOp = db.bSegOr
// case OPxor, OPnot:
// exeOp = db.bSegXor
// default:
// return
// }
// if dstkey == nil || srckeys == nil {
// return
// }
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// var srcKseq, srcKoff int32
// var seq, off, maxDstSeq, maxDstOff uint32
// var keyNum int = len(srckeys)
// var validKeyNum int
// for i := 0; i < keyNum; i++ {
// if srcKseq, srcKoff, err = db.bGetMeta(srckeys[i]); err != nil {
// return
// } else if srcKseq < 0 {
// srckeys[i] = nil
// continue
// }
// validKeyNum++
// seq = uint32(srcKseq)
// off = uint32(srcKoff)
// if seq > maxDstSeq || (seq == maxDstSeq && off > maxDstOff) {
// maxDstSeq = seq
// maxDstOff = off
// }
// }
// if (op == OPnot && validKeyNum != 1) ||
// (op != OPnot && validKeyNum < 2) {
// return // with not enough existing source key
// }
// var srcIdx int
// for srcIdx = 0; srcIdx < keyNum; srcIdx++ {
// if srckeys[srcIdx] != nil {
// break
// }
// }
// // init - data
// var segments = make([][]byte, maxDstSeq+1)
// if op == OPnot {
// // ps :
// // ( ~num == num ^ 0x11111111 )
// // we init the result segments with all bit set,
// // then we can calculate through the way of 'xor'.
// // ahead segments bin format : 1111 ... 1111
// for i := uint32(0); i < maxDstSeq; i++ {
// segments[i] = fillSegment
// }
// // last segment bin format : 1111..1100..0000
// var tailSeg = make([]byte, segByteSize, segByteSize)
// var fillByte = fillBits[7]
// var tailSegLen = db.bCapByteSize(uint32(0), maxDstOff)
// for i := uint32(0); i < tailSegLen-1; i++ {
// tailSeg[i] = fillByte
// }
// tailSeg[tailSegLen-1] = fillBits[maxDstOff-(tailSegLen-1)<<3]
// segments[maxDstSeq] = tailSeg
// } else {
// // ps : init segments by data corresponding to the 1st valid source key
// it := db.bIterator(srckeys[srcIdx])
// for ; it.Valid(); it.Next() {
// if _, seq, err = db.bDecodeBinKey(it.RawKey()); err != nil {
// // to do ...
// it.Close()
// return
// }
// segments[seq] = it.Value()
// }
// it.Close()
// srcIdx++
// }
// // operation with following keys
// var res []byte
// for i := srcIdx; i < keyNum; i++ {
// if srckeys[i] == nil {
// continue
// }
// it := db.bIterator(srckeys[i])
// for idx, end := uint32(0), false; !end; it.Next() {
// end = !it.Valid()
// if !end {
// if _, seq, err = db.bDecodeBinKey(it.RawKey()); err != nil {
// // to do ...
// it.Close()
// return
// }
// } else {
// seq = maxDstSeq + 1
// }
// // todo :
// // operation 'and' can be optimize here :
// // if seq > max_segments_idx, this loop can be break,
// // which can avoid cost from Key() and bDecodeBinKey()
// for ; idx < seq; idx++ {
// res = nil
// exeOp(segments[idx], nil, &res)
// segments[idx] = res
// }
// if !end {
// res = it.Value()
// exeOp(segments[seq], res, &res)
// segments[seq] = res
// idx++
// }
// }
// it.Close()
// }
// // clear the old data in case
// db.bDelete(t, dstkey)
// db.rmExpire(t, BitType, dstkey)
// // set data
// db.bSetMeta(t, dstkey, maxDstSeq, maxDstOff)
// var bk []byte
// for seq, segt := range segments {
// if segt != nil {
// bk = db.bEncodeBinKey(dstkey, uint32(seq))
// t.Put(bk, segt)
// }
// }
// err = t.Commit()
// if err == nil {
// // blen = int32(db.bCapByteSize(maxDstOff, maxDstOff))
// blen = int32(maxDstSeq<<segBitWidth | maxDstOff + 1)
// }
// return
// }
// func (db *DB) BExpire(key []byte, duration int64) (int64, error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if duration <= 0 {
// return 0, errExpireValue
// }
// if err := checkKeySize(key); err != nil {
// return -1, err
// }
// return db.bExpireAt(key, time.Now().Unix()+duration)
// }
// func (db *DB) BExpireAt(key []byte, when int64) (int64, error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if when <= time.Now().Unix() {
// return 0, errExpireValue
// }
// if err := checkKeySize(key); err != nil {
// return -1, err
// }
// return db.bExpireAt(key, when)
// }
// func (db *DB) BTTL(key []byte) (int64, error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if err := checkKeySize(key); err != nil {
// return -1, err
// }
// return db.ttl(BitType, key)
// }
// func (db *DB) BPersist(key []byte) (int64, error) {
// log.Error("bitmap type will be deprecated later, please use bit operations in kv type")
// if err := checkKeySize(key); err != nil {
// return 0, err
// }
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// n, err := db.rmExpire(t, BitType, key)
// if err != nil {
// return 0, err
// }
// err = t.Commit()
// return n, err
// }
// func (db *DB) bFlush() (drop int64, err error) {
// t := db.binBatch
// t.Lock()
// defer t.Unlock()
// return db.flushType(t, BitType)
// }