2005-09-20 20:27:13 +04:00
|
|
|
/*
|
|
|
|
* SHA-1 implementation optimized for ARM
|
|
|
|
*
|
|
|
|
* Copyright: (C) 2005 by Nicolas Pitre <nico@cam.org>
|
|
|
|
* Created: September 17, 2005
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include "sha1.h"
|
|
|
|
|
2008-10-01 22:05:20 +04:00
|
|
|
extern void arm_sha_transform(uint32_t *hash, const unsigned char *data, uint32_t *W);
|
2005-09-20 20:27:13 +04:00
|
|
|
|
2008-10-01 22:05:20 +04:00
|
|
|
void arm_SHA1_Init(arm_SHA_CTX *c)
|
2005-09-20 20:27:13 +04:00
|
|
|
{
|
|
|
|
c->len = 0;
|
|
|
|
c->hash[0] = 0x67452301;
|
|
|
|
c->hash[1] = 0xefcdab89;
|
|
|
|
c->hash[2] = 0x98badcfe;
|
|
|
|
c->hash[3] = 0x10325476;
|
|
|
|
c->hash[4] = 0xc3d2e1f0;
|
|
|
|
}
|
|
|
|
|
2008-10-01 22:05:20 +04:00
|
|
|
void arm_SHA1_Update(arm_SHA_CTX *c, const void *p, unsigned long n)
|
2005-09-20 20:27:13 +04:00
|
|
|
{
|
|
|
|
uint32_t workspace[80];
|
|
|
|
unsigned int partial;
|
|
|
|
unsigned long done;
|
|
|
|
|
|
|
|
partial = c->len & 0x3f;
|
|
|
|
c->len += n;
|
|
|
|
if ((partial + n) >= 64) {
|
|
|
|
if (partial) {
|
|
|
|
done = 64 - partial;
|
|
|
|
memcpy(c->buffer + partial, p, done);
|
2008-10-01 22:05:20 +04:00
|
|
|
arm_sha_transform(c->hash, c->buffer, workspace);
|
2005-09-20 20:27:13 +04:00
|
|
|
partial = 0;
|
|
|
|
} else
|
|
|
|
done = 0;
|
|
|
|
while (n >= done + 64) {
|
2008-10-01 22:05:20 +04:00
|
|
|
arm_sha_transform(c->hash, p + done, workspace);
|
2005-09-20 20:27:13 +04:00
|
|
|
done += 64;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
done = 0;
|
|
|
|
if (n - done)
|
|
|
|
memcpy(c->buffer + partial, p + done, n - done);
|
|
|
|
}
|
|
|
|
|
2008-10-01 22:05:20 +04:00
|
|
|
void arm_SHA1_Final(unsigned char *hash, arm_SHA_CTX *c)
|
2005-09-20 20:27:13 +04:00
|
|
|
{
|
|
|
|
uint64_t bitlen;
|
2007-06-07 11:04:01 +04:00
|
|
|
uint32_t bitlen_hi, bitlen_lo;
|
2005-09-20 20:27:13 +04:00
|
|
|
unsigned int i, offset, padlen;
|
|
|
|
unsigned char bits[8];
|
|
|
|
static const unsigned char padding[64] = { 0x80, };
|
|
|
|
|
|
|
|
bitlen = c->len << 3;
|
|
|
|
offset = c->len & 0x3f;
|
|
|
|
padlen = ((offset < 56) ? 56 : (64 + 56)) - offset;
|
2008-10-01 22:05:20 +04:00
|
|
|
arm_SHA1_Update(c, padding, padlen);
|
2005-09-20 20:27:13 +04:00
|
|
|
|
|
|
|
bitlen_hi = bitlen >> 32;
|
|
|
|
bitlen_lo = bitlen & 0xffffffff;
|
|
|
|
bits[0] = bitlen_hi >> 24;
|
|
|
|
bits[1] = bitlen_hi >> 16;
|
|
|
|
bits[2] = bitlen_hi >> 8;
|
|
|
|
bits[3] = bitlen_hi;
|
|
|
|
bits[4] = bitlen_lo >> 24;
|
|
|
|
bits[5] = bitlen_lo >> 16;
|
|
|
|
bits[6] = bitlen_lo >> 8;
|
|
|
|
bits[7] = bitlen_lo;
|
2008-10-01 22:05:20 +04:00
|
|
|
arm_SHA1_Update(c, bits, 8);
|
2005-09-20 20:27:13 +04:00
|
|
|
|
|
|
|
for (i = 0; i < 5; i++) {
|
|
|
|
uint32_t v = c->hash[i];
|
|
|
|
hash[0] = v >> 24;
|
|
|
|
hash[1] = v >> 16;
|
|
|
|
hash[2] = v >> 8;
|
|
|
|
hash[3] = v;
|
|
|
|
hash += 4;
|
|
|
|
}
|
|
|
|
}
|