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
| package main
import ( "encoding/binary" "fmt" )
func wikiSha256(message []byte) [32]byte { h0 := uint32(0x6a09e667) h1 := uint32(0xbb67ae85) h2 := uint32(0x3c6ef372) h3 := uint32(0xa54ff53a) h4 := uint32(0x510e527f) h5 := uint32(0x9b05688c) h6 := uint32(0x1f83d9ab) h7 := uint32(0x5be0cd19)
k := [64]uint32{ 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2}
padded := append(message, 0x80) if len(padded)%64 < 56 { suffix := make([]byte, 56-(len(padded)%64)) padded = append(padded, suffix...) } else { suffix := make([]byte, 64+56-(len(padded)%64)) padded = append(padded, suffix...) } msgLen := len(message) * 8 bs := make([]byte, 8) binary.BigEndian.PutUint64(bs, uint64(msgLen)) padded = append(padded, bs...)
broken := [][]byte{}
for i := 0; i < len(padded)/64; i++ { broken = append(broken, padded[i*64:i*64+63]) }
for _, chunk := range broken { w := []uint32{}
for i := 0; i < 16; i++ { w = append(w, binary.BigEndian.Uint32(chunk[i*4:i*4+4])) } w = append(w, make([]uint32, 48)...)
for i := 16; i < 64; i++ { s0 := rightRotate(w[i-15], 7) ^ rightRotate(w[i-15], 18) ^ (w[i-15] >> 3) s1 := rightRotate(w[i-2], 17) ^ rightRotate(w[i-2], 19) ^ (w[i-2] >> 10) w[i] = w[i-16] + s0 + w[i-7] + s1 }
a := h0 b := h1 c := h2 d := h3 e := h4 f := h5 g := h6 h := h7
for i := 0; i < 64; i++ { S1 := rightRotate(e, 6) ^ rightRotate(e, 11) ^ rightRotate(e, 25) ch := (e & f) ^ ((^e) & g) temp1 := h + S1 + ch + k[i] + w[i] S0 := rightRotate(a, 2) ^ rightRotate(a, 13) ^ rightRotate(a, 22) maj := (a & b) ^ (a & c) ^ (b & c) temp2 := S0 + maj
h = g g = f f = e e = d + temp1 d = c c = b b = a a = temp1 + temp2
}
h0 = h0 + a
h1 = h1 + b h2 = h2 + c h3 = h3 + d h4 = h4 + e h5 = h5 + f h6 = h6 + g h7 = h7 + h } hashBytes := [][]byte{iToB(h0), iToB(h1), iToB(h2), iToB(h3), iToB(h4), iToB(h5), iToB(h6), iToB(h7)} hash := []byte{} hashArray := [32]byte{} for i := 0; i < 8; i++ { hash = append(hash, hashBytes[i]...) } copy(hashArray[:], hash[0:32]) return hashArray }
func iToB(i uint32) []byte { bs := make([]byte, 4) binary.BigEndian.PutUint32(bs, i) return bs }
func rightRotate(n uint32, d uint) uint32 { return (n >> d) | (n << (32 - d)) }
func main() { s := "" o := wikiSha256([]byte(s))
fmt.Printf("%x", o)
}
|