All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.bouncycastle.crypto.digests.ISAPDigest Maven / Gradle / Ivy

package org.bouncycastle.crypto.digests;

import java.io.ByteArrayOutputStream;

import org.bouncycastle.crypto.DataLengthException;
import org.bouncycastle.crypto.Digest;
import org.bouncycastle.crypto.OutputLengthException;
import org.bouncycastle.util.Pack;

/**
 * ISAP Hash v2, https://isap.iaik.tugraz.at/
 * https://csrc.nist.gov/CSRC/media/Projects/lightweight-cryptography/documents/finalist-round/updated-spec-doc/isap-spec-final.pdf
 * 

* ISAP Hash v2 with reference to C Reference Impl from: https://github.com/isap-lwc/isap-code-package *

*/ public class ISAPDigest implements Digest { private long x0, x1, x2, x3, x4; private long t0, t1, t2, t3, t4; private ByteArrayOutputStream buffer = new ByteArrayOutputStream(); private void ROUND(long C) { t0 = x0 ^ x1 ^ x2 ^ x3 ^ C ^ (x1 & (x0 ^ x2 ^ x4 ^ C)); t1 = x0 ^ x2 ^ x3 ^ x4 ^ C ^ ((x1 ^ x2 ^ C) & (x1 ^ x3)); t2 = x1 ^ x2 ^ x4 ^ C ^ (x3 & x4); t3 = x0 ^ x1 ^ x2 ^ C ^ ((~x0) & (x3 ^ x4)); t4 = x1 ^ x3 ^ x4 ^ ((x0 ^ x4) & x1); x0 = t0 ^ ROTR(t0, 19) ^ ROTR(t0, 28); x1 = t1 ^ ROTR(t1, 39) ^ ROTR(t1, 61); x2 = ~(t2 ^ ROTR(t2, 1) ^ ROTR(t2, 6)); x3 = t3 ^ ROTR(t3, 10) ^ ROTR(t3, 17); x4 = t4 ^ ROTR(t4, 7) ^ ROTR(t4, 41); } private void P12() { ROUND(0xf0); ROUND(0xe1); ROUND(0xd2); ROUND(0xc3); ROUND(0xb4); ROUND(0xa5); ROUND(0x96); ROUND(0x87); ROUND(0x78); ROUND(0x69); ROUND(0x5a); ROUND(0x4b); } private long ROTR(long x, long n) { return (x >>> n) | (x << (64 - n)); } protected long U64BIG(long x) { return ((ROTR(x, 8) & (0xFF000000FF000000L)) | (ROTR(x, 24) & (0x00FF000000FF0000L)) | (ROTR(x, 40) & (0x0000FF000000FF00L)) | (ROTR(x, 56) & (0x000000FF000000FFL))); } @Override public String getAlgorithmName() { return "ISAP Hash"; } @Override public int getDigestSize() { return 32; } @Override public void update(byte input) { buffer.write(input); } @Override public void update(byte[] input, int inOff, int len) { if ((inOff + len) > input.length) { throw new DataLengthException("input buffer too short"); } buffer.write(input, inOff, len); } @Override public int doFinal(byte[] out, int outOff) { if (32 + outOff > out.length) { throw new OutputLengthException("output buffer is too short"); } t0 = t1 = t2 = t3 = t4 = 0; /* init state */ x0 = -1255492011513352131L; x1 = -8380609354527731710L; x2 = -5437372128236807582L; x3 = 4834782570098516968L; x4 = 3787428097924915520L; /* absorb */ byte[] input = buffer.toByteArray(); int len = input.length; long[] in64 = new long[len >> 3]; Pack.littleEndianToLong(input, 0, in64, 0, in64.length); int idx = 0; while (len >= 8) { x0 ^= U64BIG(in64[idx++]); P12(); len -= 8; } /* absorb final input block */ x0 ^= 0x80L << ((7 - len) << 3); while (len > 0) { x0 ^= (input[(idx << 3) + --len] & 0xFFL) << ((7 - len) << 3); } P12(); // squeeze long[] out64 = new long[4]; for (idx = 0; idx < 3; ++idx) { out64[idx] = U64BIG(x0); P12(); } /* squeeze final output block */ out64[idx] = U64BIG(x0); Pack.longToLittleEndian(out64, out, outOff); buffer.reset(); return 32; } @Override public void reset() { buffer.reset(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy