| Duncan P. N. Exon Smith | 91d3cfe | 2016-04-05 20:45:04 +0000 | [diff] [blame] | 1 | //======- SHA1.h - Private copy of the SHA1 implementation ---*- C++ -* ======// | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 9 | // | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 10 | // This code is taken from public domain | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 11 | // (http://oauth.googlecode.com/svn/code/c/liboauth/src/sha1.c and | 
|  | 12 | // http://cvsweb.netbsd.org/bsdweb.cgi/src/common/lib/libc/hash/sha1/sha1.c?rev=1.6) | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 13 | // and modified by wrapping it in a C++ interface for LLVM, | 
|  | 14 | // and removing unnecessary code. | 
|  | 15 | // | 
|  | 16 | //===----------------------------------------------------------------------===// | 
|  | 17 |  | 
| Mehdi Amini | 180441f | 2016-04-01 05:12:24 +0000 | [diff] [blame] | 18 | #include "llvm/Support/Host.h" | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 19 | #include "llvm/Support/SHA1.h" | 
| Mehdi Amini | b550cb1 | 2016-04-18 09:17:29 +0000 | [diff] [blame] | 20 | #include "llvm/ADT/ArrayRef.h" | 
| Eugene Zelenko | 1760dc2 | 2016-04-05 20:19:49 +0000 | [diff] [blame] | 21 | using namespace llvm; | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 22 |  | 
| Duncan P. N. Exon Smith | 91d3cfe | 2016-04-05 20:45:04 +0000 | [diff] [blame] | 23 | #include <stdint.h> | 
|  | 24 | #include <string.h> | 
|  | 25 |  | 
| Mehdi Amini | 180441f | 2016-04-01 05:12:24 +0000 | [diff] [blame] | 26 | #if defined(BYTE_ORDER) && defined(BIG_ENDIAN) && BYTE_ORDER == BIG_ENDIAN | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 27 | #define SHA_BIG_ENDIAN | 
|  | 28 | #endif | 
|  | 29 |  | 
| Rui Ueyama | 567d9c4 | 2016-11-20 01:15:56 +0000 | [diff] [blame] | 30 | static uint32_t rol(uint32_t Number, int Bits) { | 
|  | 31 | return (Number << Bits) | (Number >> (32 - Bits)); | 
| Rui Ueyama | 218072a9 | 2016-11-20 01:13:22 +0000 | [diff] [blame] | 32 | } | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 33 |  | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 34 | static uint32_t blk0(uint32_t *Buf, int I) { return Buf[I]; } | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 35 |  | 
|  | 36 | static uint32_t blk(uint32_t *Buf, int I) { | 
|  | 37 | Buf[I & 15] = rol(Buf[(I + 13) & 15] ^ Buf[(I + 8) & 15] ^ Buf[(I + 2) & 15] ^ | 
|  | 38 | Buf[I & 15], | 
|  | 39 | 1); | 
|  | 40 | return Buf[I & 15]; | 
|  | 41 | } | 
|  | 42 |  | 
|  | 43 | static void r0(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, | 
|  | 44 | int I, uint32_t *Buf) { | 
|  | 45 | E += ((B & (C ^ D)) ^ D) + blk0(Buf, I) + 0x5A827999 + rol(A, 5); | 
|  | 46 | B = rol(B, 30); | 
|  | 47 | } | 
|  | 48 |  | 
|  | 49 | static void r1(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, | 
|  | 50 | int I, uint32_t *Buf) { | 
|  | 51 | E += ((B & (C ^ D)) ^ D) + blk(Buf, I) + 0x5A827999 + rol(A, 5); | 
|  | 52 | B = rol(B, 30); | 
|  | 53 | } | 
|  | 54 |  | 
|  | 55 | static void r2(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, | 
|  | 56 | int I, uint32_t *Buf) { | 
|  | 57 | E += (B ^ C ^ D) + blk(Buf, I) + 0x6ED9EBA1 + rol(A, 5); | 
|  | 58 | B = rol(B, 30); | 
|  | 59 | } | 
|  | 60 |  | 
|  | 61 | static void r3(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, | 
|  | 62 | int I, uint32_t *Buf) { | 
|  | 63 | E += (((B | C) & D) | (B & C)) + blk(Buf, I) + 0x8F1BBCDC + rol(A, 5); | 
|  | 64 | B = rol(B, 30); | 
|  | 65 | } | 
|  | 66 |  | 
|  | 67 | static void r4(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, | 
|  | 68 | int I, uint32_t *Buf) { | 
|  | 69 | E += (B ^ C ^ D) + blk(Buf, I) + 0xCA62C1D6 + rol(A, 5); | 
|  | 70 | B = rol(B, 30); | 
|  | 71 | } | 
|  | 72 |  | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 73 | /* code */ | 
|  | 74 | #define SHA1_K0 0x5a827999 | 
|  | 75 | #define SHA1_K20 0x6ed9eba1 | 
|  | 76 | #define SHA1_K40 0x8f1bbcdc | 
|  | 77 | #define SHA1_K60 0xca62c1d6 | 
|  | 78 |  | 
|  | 79 | #define SEED_0 0x67452301 | 
|  | 80 | #define SEED_1 0xefcdab89 | 
|  | 81 | #define SEED_2 0x98badcfe | 
|  | 82 | #define SEED_3 0x10325476 | 
|  | 83 | #define SEED_4 0xc3d2e1f0 | 
|  | 84 |  | 
|  | 85 | void SHA1::init() { | 
|  | 86 | InternalState.State[0] = SEED_0; | 
|  | 87 | InternalState.State[1] = SEED_1; | 
|  | 88 | InternalState.State[2] = SEED_2; | 
|  | 89 | InternalState.State[3] = SEED_3; | 
|  | 90 | InternalState.State[4] = SEED_4; | 
|  | 91 | InternalState.ByteCount = 0; | 
|  | 92 | InternalState.BufferOffset = 0; | 
|  | 93 | } | 
|  | 94 |  | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 95 | void SHA1::hashBlock() { | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 96 | uint32_t A = InternalState.State[0]; | 
|  | 97 | uint32_t B = InternalState.State[1]; | 
|  | 98 | uint32_t C = InternalState.State[2]; | 
|  | 99 | uint32_t D = InternalState.State[3]; | 
|  | 100 | uint32_t E = InternalState.State[4]; | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 101 |  | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 102 | // 4 rounds of 20 operations each. Loop unrolled. | 
|  | 103 | r0(A, B, C, D, E, 0, InternalState.Buffer.L); | 
|  | 104 | r0(E, A, B, C, D, 1, InternalState.Buffer.L); | 
|  | 105 | r0(D, E, A, B, C, 2, InternalState.Buffer.L); | 
|  | 106 | r0(C, D, E, A, B, 3, InternalState.Buffer.L); | 
|  | 107 | r0(B, C, D, E, A, 4, InternalState.Buffer.L); | 
|  | 108 | r0(A, B, C, D, E, 5, InternalState.Buffer.L); | 
|  | 109 | r0(E, A, B, C, D, 6, InternalState.Buffer.L); | 
|  | 110 | r0(D, E, A, B, C, 7, InternalState.Buffer.L); | 
|  | 111 | r0(C, D, E, A, B, 8, InternalState.Buffer.L); | 
|  | 112 | r0(B, C, D, E, A, 9, InternalState.Buffer.L); | 
|  | 113 | r0(A, B, C, D, E, 10, InternalState.Buffer.L); | 
|  | 114 | r0(E, A, B, C, D, 11, InternalState.Buffer.L); | 
|  | 115 | r0(D, E, A, B, C, 12, InternalState.Buffer.L); | 
|  | 116 | r0(C, D, E, A, B, 13, InternalState.Buffer.L); | 
|  | 117 | r0(B, C, D, E, A, 14, InternalState.Buffer.L); | 
|  | 118 | r0(A, B, C, D, E, 15, InternalState.Buffer.L); | 
|  | 119 | r1(E, A, B, C, D, 16, InternalState.Buffer.L); | 
|  | 120 | r1(D, E, A, B, C, 17, InternalState.Buffer.L); | 
|  | 121 | r1(C, D, E, A, B, 18, InternalState.Buffer.L); | 
|  | 122 | r1(B, C, D, E, A, 19, InternalState.Buffer.L); | 
|  | 123 |  | 
|  | 124 | r2(A, B, C, D, E, 20, InternalState.Buffer.L); | 
|  | 125 | r2(E, A, B, C, D, 21, InternalState.Buffer.L); | 
|  | 126 | r2(D, E, A, B, C, 22, InternalState.Buffer.L); | 
|  | 127 | r2(C, D, E, A, B, 23, InternalState.Buffer.L); | 
|  | 128 | r2(B, C, D, E, A, 24, InternalState.Buffer.L); | 
|  | 129 | r2(A, B, C, D, E, 25, InternalState.Buffer.L); | 
|  | 130 | r2(E, A, B, C, D, 26, InternalState.Buffer.L); | 
|  | 131 | r2(D, E, A, B, C, 27, InternalState.Buffer.L); | 
|  | 132 | r2(C, D, E, A, B, 28, InternalState.Buffer.L); | 
|  | 133 | r2(B, C, D, E, A, 29, InternalState.Buffer.L); | 
|  | 134 | r2(A, B, C, D, E, 30, InternalState.Buffer.L); | 
|  | 135 | r2(E, A, B, C, D, 31, InternalState.Buffer.L); | 
|  | 136 | r2(D, E, A, B, C, 32, InternalState.Buffer.L); | 
|  | 137 | r2(C, D, E, A, B, 33, InternalState.Buffer.L); | 
|  | 138 | r2(B, C, D, E, A, 34, InternalState.Buffer.L); | 
|  | 139 | r2(A, B, C, D, E, 35, InternalState.Buffer.L); | 
|  | 140 | r2(E, A, B, C, D, 36, InternalState.Buffer.L); | 
|  | 141 | r2(D, E, A, B, C, 37, InternalState.Buffer.L); | 
|  | 142 | r2(C, D, E, A, B, 38, InternalState.Buffer.L); | 
|  | 143 | r2(B, C, D, E, A, 39, InternalState.Buffer.L); | 
|  | 144 |  | 
|  | 145 | r3(A, B, C, D, E, 40, InternalState.Buffer.L); | 
|  | 146 | r3(E, A, B, C, D, 41, InternalState.Buffer.L); | 
|  | 147 | r3(D, E, A, B, C, 42, InternalState.Buffer.L); | 
|  | 148 | r3(C, D, E, A, B, 43, InternalState.Buffer.L); | 
|  | 149 | r3(B, C, D, E, A, 44, InternalState.Buffer.L); | 
|  | 150 | r3(A, B, C, D, E, 45, InternalState.Buffer.L); | 
|  | 151 | r3(E, A, B, C, D, 46, InternalState.Buffer.L); | 
|  | 152 | r3(D, E, A, B, C, 47, InternalState.Buffer.L); | 
|  | 153 | r3(C, D, E, A, B, 48, InternalState.Buffer.L); | 
|  | 154 | r3(B, C, D, E, A, 49, InternalState.Buffer.L); | 
|  | 155 | r3(A, B, C, D, E, 50, InternalState.Buffer.L); | 
|  | 156 | r3(E, A, B, C, D, 51, InternalState.Buffer.L); | 
|  | 157 | r3(D, E, A, B, C, 52, InternalState.Buffer.L); | 
|  | 158 | r3(C, D, E, A, B, 53, InternalState.Buffer.L); | 
|  | 159 | r3(B, C, D, E, A, 54, InternalState.Buffer.L); | 
|  | 160 | r3(A, B, C, D, E, 55, InternalState.Buffer.L); | 
|  | 161 | r3(E, A, B, C, D, 56, InternalState.Buffer.L); | 
|  | 162 | r3(D, E, A, B, C, 57, InternalState.Buffer.L); | 
|  | 163 | r3(C, D, E, A, B, 58, InternalState.Buffer.L); | 
|  | 164 | r3(B, C, D, E, A, 59, InternalState.Buffer.L); | 
|  | 165 |  | 
|  | 166 | r4(A, B, C, D, E, 60, InternalState.Buffer.L); | 
|  | 167 | r4(E, A, B, C, D, 61, InternalState.Buffer.L); | 
|  | 168 | r4(D, E, A, B, C, 62, InternalState.Buffer.L); | 
|  | 169 | r4(C, D, E, A, B, 63, InternalState.Buffer.L); | 
|  | 170 | r4(B, C, D, E, A, 64, InternalState.Buffer.L); | 
|  | 171 | r4(A, B, C, D, E, 65, InternalState.Buffer.L); | 
|  | 172 | r4(E, A, B, C, D, 66, InternalState.Buffer.L); | 
|  | 173 | r4(D, E, A, B, C, 67, InternalState.Buffer.L); | 
|  | 174 | r4(C, D, E, A, B, 68, InternalState.Buffer.L); | 
|  | 175 | r4(B, C, D, E, A, 69, InternalState.Buffer.L); | 
|  | 176 | r4(A, B, C, D, E, 70, InternalState.Buffer.L); | 
|  | 177 | r4(E, A, B, C, D, 71, InternalState.Buffer.L); | 
|  | 178 | r4(D, E, A, B, C, 72, InternalState.Buffer.L); | 
|  | 179 | r4(C, D, E, A, B, 73, InternalState.Buffer.L); | 
|  | 180 | r4(B, C, D, E, A, 74, InternalState.Buffer.L); | 
|  | 181 | r4(A, B, C, D, E, 75, InternalState.Buffer.L); | 
|  | 182 | r4(E, A, B, C, D, 76, InternalState.Buffer.L); | 
|  | 183 | r4(D, E, A, B, C, 77, InternalState.Buffer.L); | 
|  | 184 | r4(C, D, E, A, B, 78, InternalState.Buffer.L); | 
|  | 185 | r4(B, C, D, E, A, 79, InternalState.Buffer.L); | 
|  | 186 |  | 
|  | 187 | InternalState.State[0] += A; | 
|  | 188 | InternalState.State[1] += B; | 
|  | 189 | InternalState.State[2] += C; | 
|  | 190 | InternalState.State[3] += D; | 
|  | 191 | InternalState.State[4] += E; | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 192 | } | 
|  | 193 |  | 
| Rui Ueyama | 567d9c4 | 2016-11-20 01:15:56 +0000 | [diff] [blame] | 194 | void SHA1::addUncounted(uint8_t Data) { | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 195 | #ifdef SHA_BIG_ENDIAN | 
| Rui Ueyama | 567d9c4 | 2016-11-20 01:15:56 +0000 | [diff] [blame] | 196 | InternalState.Buffer.C[InternalState.BufferOffset] = Data; | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 197 | #else | 
| Rui Ueyama | 567d9c4 | 2016-11-20 01:15:56 +0000 | [diff] [blame] | 198 | InternalState.Buffer.C[InternalState.BufferOffset ^ 3] = Data; | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 199 | #endif | 
| Rui Ueyama | fe33661 | 2016-11-20 01:03:22 +0000 | [diff] [blame] | 200 |  | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 201 | InternalState.BufferOffset++; | 
|  | 202 | if (InternalState.BufferOffset == BLOCK_LENGTH) { | 
|  | 203 | hashBlock(); | 
|  | 204 | InternalState.BufferOffset = 0; | 
|  | 205 | } | 
|  | 206 | } | 
|  | 207 |  | 
| Rui Ueyama | 567d9c4 | 2016-11-20 01:15:56 +0000 | [diff] [blame] | 208 | void SHA1::writebyte(uint8_t Data) { | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 209 | ++InternalState.ByteCount; | 
| Rui Ueyama | 567d9c4 | 2016-11-20 01:15:56 +0000 | [diff] [blame] | 210 | addUncounted(Data); | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 211 | } | 
|  | 212 |  | 
|  | 213 | void SHA1::update(ArrayRef<uint8_t> Data) { | 
|  | 214 | for (auto &C : Data) | 
|  | 215 | writebyte(C); | 
|  | 216 | } | 
|  | 217 |  | 
|  | 218 | void SHA1::pad() { | 
| Mehdi Amini | 180441f | 2016-04-01 05:12:24 +0000 | [diff] [blame] | 219 | // Implement SHA-1 padding (fips180-2 5.1.1) | 
| Mehdi Amini | 4cd5702 | 2016-04-01 04:30:16 +0000 | [diff] [blame] | 220 |  | 
|  | 221 | // Pad with 0x80 followed by 0x00 until the end of the block | 
|  | 222 | addUncounted(0x80); | 
|  | 223 | while (InternalState.BufferOffset != 56) | 
|  | 224 | addUncounted(0x00); | 
|  | 225 |  | 
|  | 226 | // Append length in the last 8 bytes | 
|  | 227 | addUncounted(0); // We're only using 32 bit lengths | 
|  | 228 | addUncounted(0); // But SHA-1 supports 64 bit lengths | 
|  | 229 | addUncounted(0); // So zero pad the top bits | 
|  | 230 | addUncounted(InternalState.ByteCount >> 29); // Shifting to multiply by 8 | 
|  | 231 | addUncounted(InternalState.ByteCount >> | 
|  | 232 | 21); // as SHA-1 supports bitstreams as well as | 
|  | 233 | addUncounted(InternalState.ByteCount >> 13); // byte. | 
|  | 234 | addUncounted(InternalState.ByteCount >> 5); | 
|  | 235 | addUncounted(InternalState.ByteCount << 3); | 
|  | 236 | } | 
|  | 237 |  | 
|  | 238 | StringRef SHA1::final() { | 
|  | 239 | // Pad to complete the last block | 
|  | 240 | pad(); | 
|  | 241 |  | 
|  | 242 | #ifdef SHA_BIG_ENDIAN | 
|  | 243 | // Just copy the current state | 
|  | 244 | for (int i = 0; i < 5; i++) { | 
|  | 245 | HashResult[i] = InternalState.State[i]; | 
|  | 246 | } | 
|  | 247 | #else | 
|  | 248 | // Swap byte order back | 
|  | 249 | for (int i = 0; i < 5; i++) { | 
|  | 250 | HashResult[i] = (((InternalState.State[i]) << 24) & 0xff000000) | | 
|  | 251 | (((InternalState.State[i]) << 8) & 0x00ff0000) | | 
|  | 252 | (((InternalState.State[i]) >> 8) & 0x0000ff00) | | 
|  | 253 | (((InternalState.State[i]) >> 24) & 0x000000ff); | 
|  | 254 | } | 
|  | 255 | #endif | 
|  | 256 |  | 
|  | 257 | // Return pointer to hash (20 characters) | 
|  | 258 | return StringRef((char *)HashResult, HASH_LENGTH); | 
|  | 259 | } | 
|  | 260 |  | 
|  | 261 | StringRef SHA1::result() { | 
|  | 262 | auto StateToRestore = InternalState; | 
|  | 263 |  | 
|  | 264 | auto Hash = final(); | 
|  | 265 |  | 
|  | 266 | // Restore the state | 
|  | 267 | InternalState = StateToRestore; | 
|  | 268 |  | 
|  | 269 | // Return pointer to hash (20 characters) | 
|  | 270 | return Hash; | 
|  | 271 | } | 
| Rui Ueyama | 877c26c | 2016-11-23 00:46:09 +0000 | [diff] [blame] | 272 |  | 
|  | 273 | std::array<uint8_t, 20> SHA1::hash(ArrayRef<uint8_t> Data) { | 
|  | 274 | SHA1 Hash; | 
|  | 275 | Hash.update(Data); | 
| Rui Ueyama | c464fad | 2016-11-23 03:58:12 +0000 | [diff] [blame] | 276 | StringRef S = Hash.final(); | 
| Rui Ueyama | 877c26c | 2016-11-23 00:46:09 +0000 | [diff] [blame] | 277 |  | 
|  | 278 | std::array<uint8_t, 20> Arr; | 
|  | 279 | memcpy(Arr.data(), S.data(), S.size()); | 
|  | 280 | return Arr; | 
|  | 281 | } |