mirror of
https://github.com/guanzhi/GmSSL.git
synced 2026-05-07 00:46:17 +08:00
257 lines
7.5 KiB
C
Executable File
257 lines
7.5 KiB
C
Executable File
/* ====================================================================
|
|
* Copyright (c) 2014 - 2017 The GmSSL Project. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
*
|
|
* 3. All advertising materials mentioning features or use of this
|
|
* software must display the following acknowledgment:
|
|
* "This product includes software developed by the GmSSL Project.
|
|
* (http://gmssl.org/)"
|
|
*
|
|
* 4. The name "GmSSL Project" must not be used to endorse or promote
|
|
* products derived from this software without prior written
|
|
* permission. For written permission, please contact
|
|
* guanzhi1980@gmail.com.
|
|
*
|
|
* 5. Products derived from this software may not be called "GmSSL"
|
|
* nor may "GmSSL" appear in their names without prior written
|
|
* permission of the GmSSL Project.
|
|
*
|
|
* 6. Redistributions of any form whatsoever must retain the following
|
|
* acknowledgment:
|
|
* "This product includes software developed by the GmSSL Project
|
|
* (http://gmssl.org/)"
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE GmSSL PROJECT ``AS IS'' AND ANY
|
|
* EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE GmSSL PROJECT OR
|
|
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
* ====================================================================
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <openssl/sm3.h>
|
|
#include "internal/rotate.h"
|
|
#include "internal/byteorder.h"
|
|
#include "modes_lcl.h"
|
|
|
|
void sm3_init(sm3_ctx_t *ctx)
|
|
{
|
|
memset(ctx, 0, sizeof(*ctx));
|
|
ctx->digest[0] = 0x7380166F;
|
|
ctx->digest[1] = 0x4914B2B9;
|
|
ctx->digest[2] = 0x172442D7;
|
|
ctx->digest[3] = 0xDA8A0600;
|
|
ctx->digest[4] = 0xA96F30BC;
|
|
ctx->digest[5] = 0x163138AA;
|
|
ctx->digest[6] = 0xE38DEE4D;
|
|
ctx->digest[7] = 0xB0FB0E4E;
|
|
}
|
|
|
|
void sm3_update(sm3_ctx_t *ctx, const unsigned char *data, size_t data_len)
|
|
{
|
|
if (ctx->num) {
|
|
unsigned int left = SM3_BLOCK_SIZE - ctx->num;
|
|
if (data_len < left) {
|
|
memcpy(ctx->block + ctx->num, data, data_len);
|
|
ctx->num += data_len;
|
|
return;
|
|
} else {
|
|
memcpy(ctx->block + ctx->num, data, left);
|
|
sm3_compress(ctx->digest, ctx->block);
|
|
ctx->nblocks++;
|
|
data += left;
|
|
data_len -= left;
|
|
}
|
|
}
|
|
while (data_len >= SM3_BLOCK_SIZE) {
|
|
sm3_compress(ctx->digest, data);
|
|
ctx->nblocks++;
|
|
data += SM3_BLOCK_SIZE;
|
|
data_len -= SM3_BLOCK_SIZE;
|
|
}
|
|
ctx->num = data_len;
|
|
if (data_len) {
|
|
memcpy(ctx->block, data, data_len);
|
|
}
|
|
}
|
|
|
|
void sm3_final(sm3_ctx_t *ctx, unsigned char *digest)
|
|
{
|
|
int i;
|
|
|
|
ctx->block[ctx->num] = 0x80;
|
|
|
|
if (ctx->num + 9 <= SM3_BLOCK_SIZE) {
|
|
memset(ctx->block + ctx->num + 1, 0, SM3_BLOCK_SIZE - ctx->num - 9);
|
|
} else {
|
|
memset(ctx->block + ctx->num + 1, 0, SM3_BLOCK_SIZE - ctx->num - 1);
|
|
sm3_compress(ctx->digest, ctx->block);
|
|
memset(ctx->block, 0, SM3_BLOCK_SIZE - 8);
|
|
}
|
|
PUTU32(ctx->block + 56, ctx->nblocks >> 23);
|
|
PUTU32(ctx->block + 60, (ctx->nblocks << 9) + (ctx->num << 3));
|
|
|
|
sm3_compress(ctx->digest, ctx->block);
|
|
for (i = 0; i < 8; i++) {
|
|
PUTU32(digest + i*4, ctx->digest[i]);
|
|
}
|
|
}
|
|
|
|
#define ROTL(x,n) (((x)<<(n)) | ((x)>>(32-(n))))
|
|
#define P0(x) ((x) ^ ROL32((x), 9) ^ ROL32((x),17))
|
|
#define P1(x) ((x) ^ ROL32((x),15) ^ ROL32((x),23))
|
|
|
|
#define FF00(x,y,z) ((x) ^ (y) ^ (z))
|
|
#define FF16(x,y,z) (((x)&(y)) | ((x)&(z)) | ((y)&(z)))
|
|
#define GG00(x,y,z) ((x) ^ (y) ^ (z))
|
|
#define GG16(x,y,z) ((((y)^(z)) & (x)) ^ (z))
|
|
|
|
#define T00 0x79cc4519U
|
|
#define T16 0x7a879d8aU
|
|
|
|
uint32_t K[64] = {
|
|
0x79cc4519U, 0xf3988a32U, 0xe7311465U, 0xce6228cbU,
|
|
0x9cc45197U, 0x3988a32fU, 0x7311465eU, 0xe6228cbcU,
|
|
0xcc451979U, 0x988a32f3U, 0x311465e7U, 0x6228cbceU,
|
|
0xc451979cU, 0x88a32f39U, 0x11465e73U, 0x228cbce6U,
|
|
0x9d8a7a87U, 0x3b14f50fU, 0x7629ea1eU, 0xec53d43cU,
|
|
0xd8a7a879U, 0xb14f50f3U, 0x629ea1e7U, 0xc53d43ceU,
|
|
0x8a7a879dU, 0x14f50f3bU, 0x29ea1e76U, 0x53d43cecU,
|
|
0xa7a879d8U, 0x4f50f3b1U, 0x9ea1e762U, 0x3d43cec5U,
|
|
0x7a879d8aU, 0xf50f3b14U, 0xea1e7629U, 0xd43cec53U,
|
|
0xa879d8a7U, 0x50f3b14fU, 0xa1e7629eU, 0x43cec53dU,
|
|
0x879d8a7aU, 0x0f3b14f5U, 0x1e7629eaU, 0x3cec53d4U,
|
|
0x79d8a7a8U, 0xf3b14f50U, 0xe7629ea1U, 0xcec53d43U,
|
|
0x9d8a7a87U, 0x3b14f50fU, 0x7629ea1eU, 0xec53d43cU,
|
|
0xd8a7a879U, 0xb14f50f3U, 0x629ea1e7U, 0xc53d43ceU,
|
|
0x8a7a879dU, 0x14f50f3bU, 0x29ea1e76U, 0x53d43cecU,
|
|
0xa7a879d8U, 0x4f50f3b1U, 0x9ea1e762U, 0x3d43cec5U,
|
|
};
|
|
|
|
void sm3_compress(uint32_t digest[8], const unsigned char block[64])
|
|
{
|
|
uint32_t A = digest[0];
|
|
uint32_t B = digest[1];
|
|
uint32_t C = digest[2];
|
|
uint32_t D = digest[3];
|
|
uint32_t E = digest[4];
|
|
uint32_t F = digest[5];
|
|
uint32_t G = digest[6];
|
|
uint32_t H = digest[7];
|
|
uint32_t W[68];
|
|
uint32_t SS1, SS2, TT1, TT2;
|
|
int j;
|
|
|
|
for (j = 0; j < 16; j++)
|
|
W[j] = GETU32(block + j*4);
|
|
|
|
for (; j < 68; j++)
|
|
W[j] = P1(W[j - 16] ^ W[j - 9] ^ ROL32(W[j - 3], 15))
|
|
^ ROL32(W[j - 13], 7) ^ W[j - 6];
|
|
|
|
j = 0;
|
|
|
|
#define FULL_UNROLL
|
|
#ifndef FULL_UNROLL
|
|
for (; j < 16; j++) {
|
|
SS1 = ROL32((ROL32(A, 12) + E + K[j]), 7);
|
|
SS2 = SS1 ^ ROL32(A, 12);
|
|
TT1 = FF00(A, B, C) + D + SS2 + (W[j] ^ W[j + 4]);
|
|
TT2 = GG00(E, F, G) + H + SS1 + W[j];
|
|
D = C;
|
|
C = ROL32(B, 9);
|
|
B = A;
|
|
A = TT1;
|
|
H = G;
|
|
G = ROL32(F, 19);
|
|
F = E;
|
|
E = P0(TT2);
|
|
}
|
|
|
|
for (; j < 64; j++) {
|
|
SS1 = ROL32((ROL32(A, 12) + E + K[j]), 7);
|
|
SS2 = SS1 ^ ROL32(A, 12);
|
|
TT1 = FF16(A, B, C) + D + SS2 + (W[j] ^ W[j + 4]);
|
|
TT2 = GG16(E, F, G) + H + SS1 + W[j];
|
|
D = C;
|
|
C = ROL32(B, 9);
|
|
B = A;
|
|
A = TT1;
|
|
H = G;
|
|
G = ROL32(F, 19);
|
|
F = E;
|
|
E = P0(TT2);
|
|
}
|
|
#else
|
|
# define R(A, B, C, D, E, F, G, H, xx) \
|
|
SS1 = ROL32((ROL32(A, 12) + E + K[j]), 7); \
|
|
SS2 = SS1 ^ ROL32(A, 12); \
|
|
TT1 = FF##xx(A, B, C) + D + SS2 + (W[j] ^ W[j + 4]); \
|
|
TT2 = GG##xx(E, F, G) + H + SS1 + W[j]; \
|
|
B = ROL32(B, 9); \
|
|
H = TT1; \
|
|
F = ROL32(F, 19); \
|
|
D = P0(TT2); \
|
|
j++
|
|
|
|
# define R8(A, B, C, D, E, F, G, H, xx) \
|
|
R(A, B, C, D, E, F, G, H, xx); \
|
|
R(H, A, B, C, D, E, F, G, xx); \
|
|
R(G, H, A, B, C, D, E, F, xx); \
|
|
R(F, G, H, A, B, C, D, E, xx); \
|
|
R(E, F, G, H, A, B, C, D, xx); \
|
|
R(D, E, F, G, H, A, B, C, xx); \
|
|
R(C, D, E, F, G, H, A, B, xx); \
|
|
R(B, C, D, E, F, G, H, A, xx)
|
|
|
|
R8(A, B, C, D, E, F, G, H, 00);
|
|
R8(A, B, C, D, E, F, G, H, 00);
|
|
R8(A, B, C, D, E, F, G, H, 16);
|
|
R8(A, B, C, D, E, F, G, H, 16);
|
|
R8(A, B, C, D, E, F, G, H, 16);
|
|
R8(A, B, C, D, E, F, G, H, 16);
|
|
R8(A, B, C, D, E, F, G, H, 16);
|
|
R8(A, B, C, D, E, F, G, H, 16);
|
|
#endif
|
|
|
|
digest[0] ^= A;
|
|
digest[1] ^= B;
|
|
digest[2] ^= C;
|
|
digest[3] ^= D;
|
|
digest[4] ^= E;
|
|
digest[5] ^= F;
|
|
digest[6] ^= G;
|
|
digest[7] ^= H;
|
|
}
|
|
|
|
void sm3(const unsigned char *msg, size_t msglen,
|
|
unsigned char dgst[SM3_DIGEST_LENGTH])
|
|
{
|
|
sm3_ctx_t ctx;
|
|
|
|
sm3_init(&ctx);
|
|
sm3_update(&ctx, msg, msglen);
|
|
sm3_final(&ctx, dgst);
|
|
|
|
memset(&ctx, 0, sizeof(sm3_ctx_t));
|
|
}
|