00001
00002
00003
00004
00005
00006
00007
00008 #include <cstdio>
00009 #include <cstring>
00010
00011 #include <boost/cstdint.hpp>
00012 using boost::uint32_t;
00013 using boost::uint8_t;
00014
00015 #include "libtorrent/config.hpp"
00016
00017 struct TORRENT_EXPORT SHA1_CTX
00018 {
00019 uint32_t state[5];
00020 uint32_t count[2];
00021 uint8_t buffer[64];
00022 };
00023
00024 TORRENT_EXPORT void SHA1Init(SHA1_CTX* context);
00025 TORRENT_EXPORT void SHA1Update(SHA1_CTX* context, uint8_t const* data, uint32_t len);
00026 TORRENT_EXPORT void SHA1Final(SHA1_CTX* context, uint8_t* digest);
00027
00028 namespace
00029 {
00030 union CHAR64LONG16
00031 {
00032 uint8_t c[64];
00033 uint32_t l[16];
00034 };
00035
00036 #define rol(value, bits) (((value) << (bits)) | ((value) >> (32 - (bits))))
00037
00038 struct little_endian_blk0
00039 {
00040 static uint32_t apply(CHAR64LONG16* block, int i)
00041 {
00042 return block->l[i] = (rol(block->l[i],24)&0xFF00FF00)
00043 | (rol(block->l[i],8)&0x00FF00FF);
00044 }
00045 };
00046
00047 struct big_endian_blk0
00048 {
00049 static uint32_t apply(CHAR64LONG16* block, int i)
00050 {
00051 return block->l[i];
00052 }
00053 };
00054
00055 #define blk(i) (block->l[i&15] = rol(block->l[(i+13)&15]^block->l[(i+8)&15] \
00056 ^block->l[(i+2)&15]^block->l[i&15],1))
00057
00058 #define R0(v,w,x,y,z,i) z+=((w&(x^y))^y)+BlkFun::apply(block, i)+0x5A827999+rol(v,5);w=rol(w,30);
00059 #define R1(v,w,x,y,z,i) z+=((w&(x^y))^y)+blk(i)+0x5A827999+rol(v,5);w=rol(w,30);
00060 #define R2(v,w,x,y,z,i) z+=(w^x^y)+blk(i)+0x6ED9EBA1+rol(v,5);w=rol(w,30);
00061 #define R3(v,w,x,y,z,i) z+=(((w|x)&y)|(w&x))+blk(i)+0x8F1BBCDC+rol(v,5);w=rol(w,30);
00062 #define R4(v,w,x,y,z,i) z+=(w^x^y)+blk(i)+0xCA62C1D6+rol(v,5);w=rol(w,30);
00063
00064
00065 template <class BlkFun>
00066 void SHA1Transform(uint32_t state[5], uint8_t const buffer[64])
00067 {
00068 using namespace std;
00069 uint32_t a, b, c, d, e;
00070
00071 CHAR64LONG16* block;
00072 uint8_t workspace[64];
00073 block = (CHAR64LONG16*)workspace;
00074 memcpy(block, buffer, 64);
00075
00076
00077 a = state[0];
00078 b = state[1];
00079 c = state[2];
00080 d = state[3];
00081 e = state[4];
00082
00083 R0(a,b,c,d,e, 0); R0(e,a,b,c,d, 1); R0(d,e,a,b,c, 2); R0(c,d,e,a,b, 3);
00084 R0(b,c,d,e,a, 4); R0(a,b,c,d,e, 5); R0(e,a,b,c,d, 6); R0(d,e,a,b,c, 7);
00085 R0(c,d,e,a,b, 8); R0(b,c,d,e,a, 9); R0(a,b,c,d,e,10); R0(e,a,b,c,d,11);
00086 R0(d,e,a,b,c,12); R0(c,d,e,a,b,13); R0(b,c,d,e,a,14); R0(a,b,c,d,e,15);
00087 R1(e,a,b,c,d,16); R1(d,e,a,b,c,17); R1(c,d,e,a,b,18); R1(b,c,d,e,a,19);
00088 R2(a,b,c,d,e,20); R2(e,a,b,c,d,21); R2(d,e,a,b,c,22); R2(c,d,e,a,b,23);
00089 R2(b,c,d,e,a,24); R2(a,b,c,d,e,25); R2(e,a,b,c,d,26); R2(d,e,a,b,c,27);
00090 R2(c,d,e,a,b,28); R2(b,c,d,e,a,29); R2(a,b,c,d,e,30); R2(e,a,b,c,d,31);
00091 R2(d,e,a,b,c,32); R2(c,d,e,a,b,33); R2(b,c,d,e,a,34); R2(a,b,c,d,e,35);
00092 R2(e,a,b,c,d,36); R2(d,e,a,b,c,37); R2(c,d,e,a,b,38); R2(b,c,d,e,a,39);
00093 R3(a,b,c,d,e,40); R3(e,a,b,c,d,41); R3(d,e,a,b,c,42); R3(c,d,e,a,b,43);
00094 R3(b,c,d,e,a,44); R3(a,b,c,d,e,45); R3(e,a,b,c,d,46); R3(d,e,a,b,c,47);
00095 R3(c,d,e,a,b,48); R3(b,c,d,e,a,49); R3(a,b,c,d,e,50); R3(e,a,b,c,d,51);
00096 R3(d,e,a,b,c,52); R3(c,d,e,a,b,53); R3(b,c,d,e,a,54); R3(a,b,c,d,e,55);
00097 R3(e,a,b,c,d,56); R3(d,e,a,b,c,57); R3(c,d,e,a,b,58); R3(b,c,d,e,a,59);
00098 R4(a,b,c,d,e,60); R4(e,a,b,c,d,61); R4(d,e,a,b,c,62); R4(c,d,e,a,b,63);
00099 R4(b,c,d,e,a,64); R4(a,b,c,d,e,65); R4(e,a,b,c,d,66); R4(d,e,a,b,c,67);
00100 R4(c,d,e,a,b,68); R4(b,c,d,e,a,69); R4(a,b,c,d,e,70); R4(e,a,b,c,d,71);
00101 R4(d,e,a,b,c,72); R4(c,d,e,a,b,73); R4(b,c,d,e,a,74); R4(a,b,c,d,e,75);
00102 R4(e,a,b,c,d,76); R4(d,e,a,b,c,77); R4(c,d,e,a,b,78); R4(b,c,d,e,a,79);
00103
00104 state[0] += a;
00105 state[1] += b;
00106 state[2] += c;
00107 state[3] += d;
00108 state[4] += e;
00109
00110 a = b = c = d = e = 0;
00111 }
00112
00113 void SHAPrintContext(SHA1_CTX *context, char *msg)
00114 {
00115 using namespace std;
00116 printf("%s (%d,%d) %x %x %x %x %x\n"
00117 , msg, context->count[0], context->count[1]
00118 , context->state[0], context->state[1]
00119 , context->state[2], context->state[3]
00120 , context->state[4]);
00121 }
00122
00123 template <class BlkFun>
00124 void internal_update(SHA1_CTX* context, uint8_t const* data, uint32_t len)
00125 {
00126 using namespace std;
00127 uint32_t i, j;
00128
00129 #ifdef VERBOSE
00130 SHAPrintContext(context, "before");
00131 #endif
00132 j = (context->count[0] >> 3) & 63;
00133 if ((context->count[0] += len << 3) < (len << 3)) context->count[1]++;
00134 context->count[1] += (len >> 29);
00135 if ((j + len) > 63)
00136 {
00137 memcpy(&context->buffer[j], data, (i = 64-j));
00138 SHA1Transform<BlkFun>(context->state, context->buffer);
00139 for ( ; i + 63 < len; i += 64)
00140 {
00141 SHA1Transform<BlkFun>(context->state, &data[i]);
00142 }
00143 j = 0;
00144 }
00145 else
00146 {
00147 i = 0;
00148 }
00149 memcpy(&context->buffer[j], &data[i], len - i);
00150 #ifdef VERBOSE
00151 SHAPrintContext(context, "after ");
00152 #endif
00153 }
00154
00155 bool is_big_endian()
00156 {
00157 uint32_t test = 1;
00158 return *reinterpret_cast<uint8_t*>(&test) == 0;
00159 }
00160 }
00161
00162 void SHA1Init(SHA1_CTX* context)
00163 {
00164
00165 context->state[0] = 0x67452301;
00166 context->state[1] = 0xEFCDAB89;
00167 context->state[2] = 0x98BADCFE;
00168 context->state[3] = 0x10325476;
00169 context->state[4] = 0xC3D2E1F0;
00170 context->count[0] = context->count[1] = 0;
00171 }
00172
00173 void SHA1Update(SHA1_CTX* context, uint8_t const* data, uint32_t len)
00174 {
00175 #if defined __BIG_ENDIAN__
00176 internal_update<big_endian_blk0>(context, data, len);
00177 #elif defined LITTLE_ENDIAN
00178 internal_update<little_endian_blk0>(context, data, len);
00179 #else
00180
00181
00182 if (is_big_endian())
00183 internal_update<big_endian_blk0>(context, data, len);
00184 else
00185 internal_update<little_endian_blk0>(context, data, len);
00186 #endif
00187 }
00188
00189 void SHA1Final(SHA1_CTX* context, uint8_t* digest)
00190 {
00191 uint8_t finalcount[8];
00192
00193 for (uint32_t i = 0; i < 8; ++i)
00194 {
00195
00196 finalcount[i] = static_cast<uint8_t>(
00197 (context->count[(i >= 4 ? 0 : 1)]
00198 >> ((3-(i & 3)) * 8) ) & 255);
00199 }
00200
00201 SHA1Update(context, (uint8_t const*)"\200", 1);
00202 while ((context->count[0] & 504) != 448)
00203 SHA1Update(context, (uint8_t const*)"\0", 1);
00204 SHA1Update(context, finalcount, 8);
00205
00206 for (uint32_t i = 0; i < 20; ++i)
00207 {
00208 digest[i] = static_cast<unsigned char>(
00209 (context->state[i>>2] >> ((3-(i & 3)) * 8) ) & 255);
00210 }
00211 }
00212
00213