SM3工具类

import lombok.extern.slf4j.Slf4j;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.Arrays;

@Slf4j
public class Sm3Util {

private final static char[] CHARS = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F'};
public static final byte[] IV = {0x73, (byte) 0x80, 0x16, 0x6f, 0x49, 0x14, (byte) 0xb2, (byte) 0xb9, 0x17, 0x24, 0x42,
        (byte) 0xd7, (byte) 0xda, (byte) 0x8a, 0x06, 0x00, (byte) 0xa9, 0x6f, 0x30, (byte) 0xbc, (byte) 0x16, 0x31,
        0x38, (byte) 0xaa, (byte) 0xe3, (byte) 0x8d, (byte) 0xee, 0x4d, (byte) 0xb0, (byte) 0xfb, 0x0e, 0x4e};
private static final Integer TJ_15 = Integer.valueOf("79cc4519", 16);
private static final Integer TJ_63 = Integer.valueOf("7a879d8a", 16);
private static final byte[] FIRST_PADDING = {(byte) 0x80};
private static final byte[] ZERO_PADDING = {(byte) 0x00};

public static String sign(String source) throws IOException {
    byte[] b = sign(source.getBytes());
    return byteToHexString(b);
}

public static byte[] sign(byte[] source) throws IOException {
    byte[] m1 = padding(source);
    int n = m1.length / (512 / 8);
    if (log.isDebugEnabled()) {
        log.debug("n = " + n);
    }
    byte[] b;
    byte[] vi = IV.clone();
    byte[] vi1 = null;
    for (int i = 0; i < n; i++) {
        b = Arrays.copyOfRange(m1, i * 64, (i + 1) * 64);
        vi1 = CF(vi, b);
        vi = vi1;
    }
    return vi1;
}

private static int T(int j) {
    if (j >= 0 && j <= 15) {
        return TJ_15;
    } else if (j >= 16 && j <= 63) {
        return TJ_63;
    } else {
        throw new RuntimeException("data invalid");
    }
}

private static Integer FF(Integer x, Integer y, Integer z, int j) {
    if (j >= 0 && j <= 15) {
        return x ^ y ^ z;
    } else if (j >= 16 && j <= 63) {
        return (x & y) | (x & z) | (y & z);
    } else {
        throw new RuntimeException("data invalid");
    }
}

private static Integer GG(Integer x, Integer y, Integer z, int j) {
    if (j >= 0 && j <= 15) {
        return x ^ y ^ z;
    } else if (j >= 16 && j <= 63) {
        return (x & y) | (~x & z);
    } else {
        throw new RuntimeException("data invalid");
    }
}

private static Integer P0(Integer x) {
    return x ^ Integer.rotateLeft(x, 9) ^ Integer.rotateLeft(x, 17);
}

private static Integer P1(Integer x) {
    return x ^ Integer.rotateLeft(x, 15) ^ Integer.rotateLeft(x, 23);
}

private static byte[] padding(byte[] source) throws IOException {
    long l = source.length * 8L;
    long k = 448 - (l + 1) % 512;
    if (k < 0) {
        k = k + 512;
    }
    if (log.isDebugEnabled()) {
        log.debug("k = " + k);
    }
    try (ByteArrayOutputStream baos = new ByteArrayOutputStream();) {
        baos.write(source);
        baos.write(FIRST_PADDING);
        long i = k - 7;
        while (i > 0) {
            baos.write(ZERO_PADDING);
            i -= 8;
        }
        baos.write(long2bytes(l));
        if (log.isDebugEnabled()) {
            log.debug("paded size = " + baos.size());
        }
        return baos.toByteArray();
    }
}

private static byte[] long2bytes(long l) {
    byte[] bytes = new byte[8];
    for (int i = 0; i < 8; i++) {
        bytes[i] = (byte) (l >>> ((7 - i) * 8));
    }
    return bytes;
}

private static byte[] CF(byte[] vi, byte[] bi) throws IOException {
    int a, b, c, d, e, f, g, h;
    a = toInteger(vi, 0);
    b = toInteger(vi, 1);
    c = toInteger(vi, 2);
    d = toInteger(vi, 3);
    e = toInteger(vi, 4);
    f = toInteger(vi, 5);
    g = toInteger(vi, 6);
    h = toInteger(vi, 7);

    int[] w = new int[68];
    int[] w1 = new int[64];
    for (int i = 0; i < 16; i++) {
        w[i] = toInteger(bi, i);
    }
    for (int j = 16; j < 68; j++) {
        w[j] = P1(w[j - 16] ^ w[j - 9] ^ Integer.rotateLeft(w[j - 3], 15)) ^ Integer.rotateLeft(w[j - 13], 7)
                ^ w[j - 6];
    }
    for (int j = 0; j < 64; j++) {
        w1[j] = w[j] ^ w[j + 4];
    }
    int ss1, ss2, tt1, tt2;
    for (int j = 0; j < 64; j++) {
        ss1 = Integer.rotateLeft(Integer.rotateLeft(a, 12) + e + Integer.rotateLeft(T(j), j), 7);
        ss2 = ss1 ^ Integer.rotateLeft(a, 12);
        tt1 = FF(a, b, c, j) + d + ss2 + w1[j];
        tt2 = GG(e, f, g, j) + h + ss1 + w[j];
        d = c;
        c = Integer.rotateLeft(b, 9);
        b = a;
        a = tt1;
        h = g;
        g = Integer.rotateLeft(f, 19);
        f = e;
        e = P0(tt2);
    }
    byte[] v = toByteArray(a, b, c, d, e, f, g, h);
    for (int i = 0; i < v.length; i++) {
        v[i] = (byte) (v[i] ^ vi[i]);
    }
    return v;
}

private static int toInteger(byte[] source, int index) {
    StringBuilder valueStr = new StringBuilder();
    for (int i = 0; i < 4; i++) {
        valueStr.append(CHARS[(byte) ((source[index * 4 + i] & 0xF0) >> 4)]);
        valueStr.append(CHARS[(byte) (source[index * 4 + i] & 0x0F)]);
    }
    return Long.valueOf(valueStr.toString(), 16).intValue();
}

private static byte[] toByteArray(int a, int b, int c, int d, int e, int f, int g, int h) throws IOException {
    try (ByteArrayOutputStream baos = new ByteArrayOutputStream(32);) {
        baos.write(toByteArray(a));
        baos.write(toByteArray(b));
        baos.write(toByteArray(c));
        baos.write(toByteArray(d));
        baos.write(toByteArray(e));
        baos.write(toByteArray(f));
        baos.write(toByteArray(g));
        baos.write(toByteArray(h));
        return baos.toByteArray();
    }
}

private static byte[] toByteArray(int i) {
    byte[] byteArray = new byte[4];
    byteArray[0] = (byte) (i >>> 24);
    byteArray[1] = (byte) ((i & 0xFFFFFF) >>> 16);
    byteArray[2] = (byte) ((i & 0xFFFF) >>> 8);
    byteArray[3] = (byte) (i & 0xFF);
    return byteArray;
}

private static String byteToHexString(byte[] bytes) {
    StringBuilder resultHexString = new StringBuilder();
    String tempStr;
    for (byte b : bytes) {
        tempStr = Integer.toHexString(b & 0xff);
        if (tempStr.length() == 1) {
            resultHexString.append(0).append(tempStr);
        } else {
            resultHexString.append(tempStr);
        }
    }
    return resultHexString.toString();
}

}

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值