1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 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 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
|
// Copyright 2012-2013 Huan Truong. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// copied from: https://github.com/htruong/go-md2/blob/master/md2.go
// Package md2 implements the MD2 hash algorithm as defined in RFC 1319.
package md2
import (
"hash"
)
func init() {
}
// The size of an MD2 checksum in bytes.
const Size = 16
// The blocksize of MD2 in bytes.
const BlockSize = 16
const _Chunk = 16
var PI_SUBST = []uint8{
41, 46, 67, 201, 162, 216, 124, 1, 61, 54, 84, 161, 236, 240, 6,
19, 98, 167, 5, 243, 192, 199, 115, 140, 152, 147, 43, 217, 188,
76, 130, 202, 30, 155, 87, 60, 253, 212, 224, 22, 103, 66, 111, 24,
138, 23, 229, 18, 190, 78, 196, 214, 218, 158, 222, 73, 160, 251,
245, 142, 187, 47, 238, 122, 169, 104, 121, 145, 21, 178, 7, 63,
148, 194, 16, 137, 11, 34, 95, 33, 128, 127, 93, 154, 90, 144, 50,
39, 53, 62, 204, 231, 191, 247, 151, 3, 255, 25, 48, 179, 72, 165,
181, 209, 215, 94, 146, 42, 172, 86, 170, 198, 79, 184, 56, 210,
150, 164, 125, 182, 118, 252, 107, 226, 156, 116, 4, 241, 69, 157,
112, 89, 100, 113, 135, 32, 134, 91, 207, 101, 230, 45, 168, 2, 27,
96, 37, 173, 174, 176, 185, 246, 28, 70, 97, 105, 52, 64, 126, 15,
85, 71, 163, 35, 221, 81, 175, 58, 195, 92, 249, 206, 186, 197,
234, 38, 44, 83, 13, 110, 133, 40, 132, 9, 211, 223, 205, 244, 65,
129, 77, 82, 106, 220, 55, 200, 108, 193, 171, 250, 36, 225, 123,
8, 12, 189, 177, 74, 120, 136, 149, 139, 227, 99, 232, 109, 233,
203, 213, 254, 59, 0, 29, 57, 242, 239, 183, 14, 102, 88, 208, 228,
166, 119, 114, 248, 235, 117, 75, 10, 49, 68, 80, 180, 143, 237,
31, 26, 219, 153, 141, 51, 159, 17, 131, 20,
}
// digest represents the partial evaluation of a checksum.
type digest struct {
digest [Size]byte // the digest, Size
state [48]byte // state, 48 ints
x [_Chunk]byte // temp storage buffer, 16 bytes, _Chunk
nx uint8 // how many bytes are there in the buffer
}
func (d *digest) Reset() {
for i := range d.digest {
d.digest[i] = 0
}
for i := range d.state {
d.state[i] = 0
}
for i := range d.x {
d.x[i] = 0
}
d.nx = 0
}
// New returns a new hash.Hash computing the MD2 checksum.
func New() hash.Hash {
d := new(digest)
d.Reset()
return d
}
func (d *digest) Size() int { return Size }
func (d *digest) BlockSize() int { return BlockSize }
// Write is the interface for IO Writer
func (d *digest) Write(p []byte) (nn int, err error) {
nn = len(p)
//d.len += uint64(nn)
// If we have something left in the buffer
if d.nx > 0 {
n := uint8(len(p))
var i uint8
// try to copy the rest n bytes free of the buffer into the buffer
// then hash the buffer
if (n + d.nx) > _Chunk {
n = _Chunk - d.nx
}
for i = 0; i < n; i++ {
// copy n bytes to the buffer
d.x[d.nx+i] = p[i]
}
d.nx += n
// if we have exactly 1 block in the buffer then hash that block
if d.nx == _Chunk {
block(d, d.x[0:_Chunk])
d.nx = 0
}
p = p[n:]
}
imax := len(p) / _Chunk
// For the rest, try hashing by the blocksize
for i := 0; i < imax; i++ {
block(d, p[:_Chunk])
p = p[_Chunk:]
}
// Then stuff the rest that doesn't add up to a block to the buffer
if len(p) > 0 {
d.nx = uint8(copy(d.x[:], p))
}
return
}
func (d0 *digest) Sum(in []byte) []byte {
// Make a copy of d0 so that caller can keep writing and summing.
d := *d0
// Padding.
var tmp [_Chunk]byte
len := uint8(d.nx)
for i := range tmp {
tmp[i] = _Chunk - len
}
d.Write(tmp[0 : _Chunk-len])
// At this state we should have nothing left in buffer
if d.nx != 0 {
panic("d.nx != 0")
}
d.Write(d.digest[0:16])
// At this state we should have nothing left in buffer
if d.nx != 0 {
panic("d.nx != 0")
}
return append(in, d.state[0:16]...)
}
func block(dig *digest, p []byte) {
var t, i, j uint8
t = 0
for i = 0; i < 16; i++ {
dig.state[i+16] = p[i]
dig.state[i+32] = byte(p[i] ^ dig.state[i])
}
for i = 0; i < 18; i++ {
for j = 0; j < 48; j++ {
dig.state[j] = byte(dig.state[j] ^ PI_SUBST[t])
t = dig.state[j]
}
t = byte(t + i)
}
t = dig.digest[15]
for i = 0; i < 16; i++ {
dig.digest[i] = byte(dig.digest[i] ^ PI_SUBST[p[i]^t])
t = dig.digest[i]
}
}
|