Subversion Repositories zxusbnet

Rev

Rev 47 | Blame | Compare with Previous | Last modification | View Log | Download | RSS feed | ?url?

  1. /**
  2.  * support MD5 for PPPoE CHAP mode
  3.  *
  4.  * taken from RFC-1321/Appendix A.3
  5.  * MD5C.C - RSA Data Security, Inc., MD5 message-digest algorithm
  6.  */
  7.  
  8. /*
  9. ********************************************************************************
  10. File Include Section
  11. ********************************************************************************
  12. */
  13.  
  14. #include <string.h>
  15. #include "md5.h"
  16.  
  17. // Constants for Transform routine.
  18. #define S11    7
  19. #define S12   12
  20. #define S13   17
  21. #define S14   22
  22. #define S21    5
  23. #define S22    9
  24. #define S23   14
  25. #define S24   20
  26. #define S31    4
  27. #define S32   11
  28. #define S33   16
  29. #define S34   23
  30. #define S41    6
  31. #define S42   10
  32. #define S43   15
  33. #define S44   21
  34.  
  35. void md5_transform (uint32[4], uint8 [64]);
  36. void md5_encode    (uint8 *, uint32 *, uint32);
  37. void md5_decode    (uint32 *, uint8 *, uint32);
  38.  
  39. uint8 padding[64] = {
  40.         0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  41.         0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  42.         0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  43.         0, 0
  44. };
  45.  
  46. // F, G, H and I are basic md5 functions.
  47. #define F(x, y, z) (((x) & (y)) | ((~x) & (z)))
  48. #define G(x, y, z) (((x) & (z)) | ((y) & (~z)))
  49. #define H(x, y, z) ((x) ^ (y) ^ (z))
  50. #define I(x, y, z) ((y) ^ ((x) | (~z)))
  51.  
  52. // ROTATE_LEFT rotates x left n bits.
  53. #define ROTATE_LEFT(x, n) (((x) << (n)) | ((x) >> (32-(n))))
  54.        
  55.  
  56.         uint32 FF(uint32 a, uint32 b, uint32 c, uint32 d, uint32 x, uint32 s, uint32 ac)
  57.         {
  58.                 a += F (b, c, d) + x + (uint32)(ac);
  59.                 a = ROTATE_LEFT (a, s);
  60.                 a += b;
  61.                 return a;
  62.         }
  63.         uint32 GG(uint32 a, uint32 b, uint32 c, uint32 d, uint32 x, uint32 s, uint32 ac)
  64.         {
  65.                 a += G (b, c, d) + x + (uint32)(ac);
  66.                 a = ROTATE_LEFT (a, s);
  67.                 a += b;
  68.                 return a;
  69.         }
  70.        
  71.         uint32 HH(uint32 a, uint32 b, uint32 c, uint32 d, uint32 x, uint32 s, uint32 ac)
  72.         {
  73.                 a += H (b, c, d) + x + (uint32)(ac);
  74.                 a = ROTATE_LEFT (a, s);
  75.                 a += b;
  76.                 return a;
  77.         }
  78.  
  79.         uint32 II(uint32 a, uint32 b, uint32 c, uint32 d, uint32 x, uint32 s, uint32 ac)
  80.         {
  81.                 a += I (b, c, d) + x + (uint32)(ac);
  82.                 a = ROTATE_LEFT (a, s);
  83.                 a += b;
  84.                 return a;
  85.         }
  86.  
  87. // md5 initialization. Begins an md5 operation, writing a new context.
  88. void md5_init(md5_ctx *context)
  89. {
  90.         context->count[0] = context->count[1] = 0;
  91.        
  92.         // Load magic initialization constants.
  93.         context->state[0] = 0x67452301;
  94.         context->state[1] = 0xefcdab89;
  95.         context->state[2] = 0x98badcfe;
  96.         context->state[3] = 0x10325476;
  97. }
  98.  
  99. // md5 block update operation. Continues an md5 message-digest operation,
  100. // processing another message block, and updating the context.
  101. void md5_update(md5_ctx * context, uint8 *input, uint32 inputLen)
  102. {
  103.         uint32 i, index, partLen;
  104.        
  105.         // Compute number of bytes mod 64
  106.         index = (uint32)((context->count[0] >> 3) & 0x3F);
  107.        
  108.         // Update number of bits
  109.         if ((context->count[0] += ((uint32)inputLen << 3)) < ((uint32)inputLen << 3))
  110.                 context->count[1]++;
  111.         context->count[1] += ((uint32)inputLen >> 29);
  112.        
  113.         partLen = 64 - index;
  114.        
  115.         // md5_Transform as many times as possible.
  116.         if (inputLen >= partLen)
  117.         {
  118.                 memcpy(&context->buffer[index], input, partLen);
  119.                 md5_transform(context->state, context->buffer);
  120.        
  121.                 for (i = partLen; i + 63 < inputLen; i += 64)
  122.                         md5_transform(context->state, &input[i]);
  123.                 index = 0;
  124.         }
  125.         else
  126.                 i = 0;
  127.        
  128.         // Buffer remaining input
  129.         memcpy(&context->buffer[index], &input[i], inputLen - i);
  130. }
  131.  
  132. // md5 finalization. Ends an md5 message-digest operation, writing the
  133. // message digest and zeroizing the context.
  134. void md5_final(uint8 digest[16], md5_ctx *context)
  135. {
  136.         uint8  bits[8];
  137.         uint32 index, padLen;
  138.        
  139.         // Save number of bits
  140.         md5_encode(bits, context->count, 8);
  141.        
  142.         // Pad out to 56 mod 64.
  143.         index  = (uint32)((context->count[0] >> 3) & 0x3f);
  144.         padLen = (index < 56) ? (56 - index) : (120 - index);
  145.         md5_update(context, padding, padLen);
  146.        
  147.         // Append length (before padding)
  148.         md5_update(context, bits, 8);
  149.         // Store state in digest
  150.         md5_encode(digest, context->state, 16);
  151.        
  152.         // Zeroize sensitive information.
  153.         memset((void*)context,0,sizeof(*context));
  154. }
  155.  
  156. // md5 basic transformation. Transforms state based on block.
  157. void md5_transform(uint32 state[4], uint8 block[64])
  158. {
  159.         uint32 a = state[0];
  160.         uint32 b = state[1];
  161.         uint32 c = state[2];
  162.         uint32 d = state[3];
  163.         uint32 x[16];
  164.        
  165.         md5_decode(x, block, 64);
  166.  
  167.         // Round 1
  168.         a = FF(a, b, c, d, x[0],  S11, 0xd76aa478);     // 1
  169.         d = FF(d, a, b, c, x[1],  S12, 0xe8c7b756); // 2
  170.         c = FF(c, d, a, b, x[2],  S13, 0x242070db); // 3
  171.         b = FF(b, c, d, a, x[3],  S14, 0xc1bdceee); // 4
  172.         a = FF(a, b, c, d, x[4],  S11, 0xf57c0faf); // 5
  173.         d = FF(d, a, b, c, x[5],  S12, 0x4787c62a); // 6
  174.         c = FF(c, d, a, b, x[6],  S13, 0xa8304613); // 7
  175.         b = FF(b, c, d, a, x[7],  S14, 0xfd469501); // 8
  176.         a = FF(a, b, c, d, x[8],  S11, 0x698098d8); // 9
  177.         d = FF(d, a, b, c, x[9],  S12, 0x8b44f7af); // 10
  178.         c = FF(c, d, a, b, x[10], S13, 0xffff5bb1); // 11
  179.         b = FF(b, c, d, a, x[11], S14, 0x895cd7be); // 12
  180.         a = FF(a, b, c, d, x[12], S11, 0x6b901122); // 13
  181.         d = FF(d, a, b, c, x[13], S12, 0xfd987193); // 14
  182.         c = FF(c, d, a, b, x[14], S13, 0xa679438e); // 15
  183.         b = FF(b, c, d, a, x[15], S14, 0x49b40821); // 16
  184.  
  185.         // Round 2
  186.         a = GG(a, b, c, d, x[1],  S21, 0xf61e2562); // 17
  187.         d = GG(d, a, b, c, x[6],  S22, 0xc040b340); // 18
  188.         c = GG(c, d, a, b, x[11], S23, 0x265e5a51); // 19
  189.         b = GG(b, c, d, a, x[0],  S24, 0xe9b6c7aa); // 20
  190.         a = GG(a, b, c, d, x[5],  S21, 0xd62f105d); // 21
  191.         d = GG(d, a, b, c, x[10], S22, 0x2441453);  // 22
  192.         c = GG(c, d, a, b, x[15], S23, 0xd8a1e681); // 23
  193.         b = GG(b, c, d, a, x[4],  S24, 0xe7d3fbc8); // 24
  194.         a = GG(a, b, c, d, x[9],  S21, 0x21e1cde6); // 25
  195.         d = GG(d, a, b, c, x[14], S22, 0xc33707d6); // 26
  196.         c = GG(c, d, a, b, x[3],  S23, 0xf4d50d87); // 27
  197.         b = GG(b, c, d, a, x[8],  S24, 0x455a14ed); // 28
  198.         a = GG(a, b, c, d, x[13], S21, 0xa9e3e905); // 29
  199.         d = GG(d, a, b, c, x[2],  S22, 0xfcefa3f8); // 30
  200.         c = GG(c, d, a, b, x[7],  S23, 0x676f02d9); // 31
  201.         b = GG(b, c, d, a, x[12], S24, 0x8d2a4c8a); // 32
  202.        
  203.         // Round 3
  204.         a = HH(a, b, c, d, x[5],  S31, 0xfffa3942); // 33
  205.         d = HH(d, a, b, c, x[8],  S32, 0x8771f681); // 34
  206.         c = HH(c, d, a, b, x[11], S33, 0x6d9d6122); // 35
  207.         b = HH(b, c, d, a, x[14], S34, 0xfde5380c); // 36
  208.         a = HH(a, b, c, d, x[1],  S31, 0xa4beea44); // 37
  209.         d = HH(d, a, b, c, x[4],  S32, 0x4bdecfa9); // 38
  210.         c = HH(c, d, a, b, x[7],  S33, 0xf6bb4b60); // 39
  211.         b = HH(b, c, d, a, x[10], S34, 0xbebfbc70); // 40
  212.         a = HH(a, b, c, d, x[13], S31, 0x289b7ec6); // 41
  213.         d = HH(d, a, b, c, x[0],  S32, 0xeaa127fa); // 42
  214.         c = HH(c, d, a, b, x[3],  S33, 0xd4ef3085); // 43
  215.         b = HH(b, c, d, a, x[6],  S34, 0x4881d05);  // 44
  216.         a = HH(a, b, c, d, x[9],  S31, 0xd9d4d039); // 45
  217.         d = HH(d, a, b, c, x[12], S32, 0xe6db99e5); // 46
  218.         c = HH(c, d, a, b, x[15], S33, 0x1fa27cf8); // 47
  219.         b = HH(b, c, d, a, x[2],  S34, 0xc4ac5665); // 48
  220.  
  221.         // Round 4
  222.         a = II(a, b, c, d, x[0],  S41, 0xf4292244); // 49
  223.         d = II(d, a, b, c, x[7],  S42, 0x432aff97); // 50
  224.         c = II(c, d, a, b, x[14], S43, 0xab9423a7); // 51
  225.         b = II(b, c, d, a, x[5],  S44, 0xfc93a039); // 52
  226.         a = II(a, b, c, d, x[12], S41, 0x655b59c3); // 53
  227.         d = II(d, a, b, c, x[3],  S42, 0x8f0ccc92); // 54
  228.         c = II(c, d, a, b, x[10], S43, 0xffeff47d); // 55
  229.         b = II(b, c, d, a, x[1],  S44, 0x85845dd1); // 56
  230.         a = II(a, b, c, d, x[8],  S41, 0x6fa87e4f); // 57
  231.         d = II(d, a, b, c, x[15], S42, 0xfe2ce6e0); // 58
  232.         c = II(c, d, a, b, x[6],  S43, 0xa3014314); // 59
  233.         b = II(b, c, d, a, x[13], S44, 0x4e0811a1); // 60
  234.         a = II(a, b, c, d, x[4],  S41, 0xf7537e82); // 61
  235.         d = II(d, a, b, c, x[11], S42, 0xbd3af235); // 62
  236.         c = II(c, d, a, b, x[2],  S43, 0x2ad7d2bb); // 63
  237.         b = II(b, c, d, a, x[9],  S44, 0xeb86d391); // 64
  238.  
  239.         state[0] += a;
  240.         state[1] += b;
  241.         state[2] += c;
  242.         state[3] += d;
  243.        
  244.         // Zeroize sensitive information.
  245.         memset(&x,0,sizeof(x));
  246. }
  247.  
  248. // Encodes input (uint32) into output (uint8). Assumes len is a
  249. // multiple of 4.
  250. void md5_encode(uint8 *output, uint32 *input, uint32 len)
  251. {
  252.         uint32 i, j;
  253.        
  254.         for (i = 0, j = 0; j < len; i++, j += 4)
  255.         {
  256.                 output[j]     = (uint8)(input[i] & 0xff);
  257.                 output[j + 1] = (uint8)((input[i] >> 8) & 0xff);
  258.                 output[j + 2] = (uint8)((input[i] >> 16) & 0xff);
  259.                 output[j + 3] = (uint8)((input[i] >> 24) & 0xff);
  260.         }
  261. }
  262.  
  263. // Decodes input (uint8) into output (uint32). Assumes len is a
  264. // multiple of 4.
  265. void md5_decode(uint32 *output, uint8 *input, uint32 len)
  266. {
  267.         uint32 i, j;
  268.         for (i = 0, j = 0; j < len; i++, j += 4)
  269.                 output[i] = ((uint32) input[j]) | (((uint32) input[j + 1]) << 8) |
  270.                                 (((uint32)input[j + 2]) << 16) | (((uint32)input[j + 3]) << 24);
  271. }
  272.  
  273.