encoding.go 2.73 KB
Newer Older
1
// Copyright 2014 The go-ethereum Authors
2
// This file is part of the go-ethereum library.
3
//
4
// The go-ethereum library is free software: you can redistribute it and/or modify
5 6 7 8
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
9
// The go-ethereum library is distributed in the hope that it will be useful,
10
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 13 14
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
15
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
16

obscuren's avatar
obscuren committed
17
package trie
obscuren's avatar
obscuren committed
18

Felix Lange's avatar
Felix Lange committed
19 20
func compactEncode(hexSlice []byte) []byte {
	terminator := byte(0)
obscuren's avatar
obscuren committed
21 22 23 24
	if hexSlice[len(hexSlice)-1] == 16 {
		terminator = 1
		hexSlice = hexSlice[:len(hexSlice)-1]
	}
Felix Lange's avatar
Felix Lange committed
25 26 27 28 29 30 31 32 33
	var (
		odd    = byte(len(hexSlice) % 2)
		buflen = len(hexSlice)/2 + 1
		bi, hi = 0, 0    // indices
		hs     = byte(0) // shift: flips between 0 and 4
	)
	if odd == 0 {
		bi = 1
		hs = 4
obscuren's avatar
obscuren committed
34
	}
Felix Lange's avatar
Felix Lange committed
35 36 37 38 39 40 41 42
	buf := make([]byte, buflen)
	buf[0] = terminator<<5 | byte(odd)<<4
	for bi < len(buf) && hi < len(hexSlice) {
		buf[bi] |= hexSlice[hi] << hs
		if hs == 0 {
			bi++
		}
		hi, hs = hi+1, hs^(1<<2)
obscuren's avatar
obscuren committed
43
	}
44
	return buf
obscuren's avatar
obscuren committed
45 46
}

Felix Lange's avatar
Felix Lange committed
47 48
func compactDecode(str []byte) []byte {
	base := compactHexDecode(str)
obscuren's avatar
obscuren committed
49
	base = base[:len(base)-1]
obscuren's avatar
obscuren committed
50
	if base[0] >= 2 {
obscuren's avatar
obscuren committed
51 52 53 54 55 56 57 58 59 60
		base = append(base, 16)
	}
	if base[0]%2 == 1 {
		base = base[1:]
	} else {
		base = base[2:]
	}
	return base
}

Felix Lange's avatar
Felix Lange committed
61
func compactHexDecode(str []byte) []byte {
62 63 64 65 66
	l := len(str)*2 + 1
	var nibbles = make([]byte, l)
	for i, b := range str {
		nibbles[i*2] = b / 16
		nibbles[i*2+1] = b % 16
obscuren's avatar
obscuren committed
67
	}
68
	nibbles[l-1] = 16
69
	return nibbles
obscuren's avatar
obscuren committed
70
}
obscuren's avatar
obscuren committed
71

72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
// compactHexEncode encodes a series of nibbles into a byte array
func compactHexEncode(nibbles []byte) []byte {
	nl := len(nibbles)
	if nl == 0 {
		return nil
	}
	if nibbles[nl-1] == 16 {
		nl--
	}
	l := (nl + 1) / 2
	var str = make([]byte, l)
	for i, _ := range str {
		b := nibbles[i*2] * 16
		if nl > i*2 {
			b += nibbles[i*2+1]
		}
		str[i] = b
	}
	return str
}

Felix Lange's avatar
Felix Lange committed
93
func decodeCompact(key []byte) []byte {
94 95 96 97 98
	l := len(key) / 2
	var res = make([]byte, l)
	for i := 0; i < l; i++ {
		v1, v0 := key[2*i], key[2*i+1]
		res[i] = v1*16 + v0
obscuren's avatar
obscuren committed
99
	}
100
	return res
obscuren's avatar
obscuren committed
101
}
Felix Lange's avatar
Felix Lange committed
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128

// prefixLen returns the length of the common prefix of a and b.
func prefixLen(a, b []byte) int {
	var i, length = 0, len(a)
	if len(b) < length {
		length = len(b)
	}
	for ; i < length; i++ {
		if a[i] != b[i] {
			break
		}
	}
	return i
}

func hasTerm(s []byte) bool {
	return s[len(s)-1] == 16
}

func remTerm(s []byte) []byte {
	if hasTerm(s) {
		b := make([]byte, len(s)-1)
		copy(b, s)
		return b
	}
	return s
}