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

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

package org.bouncycastle.crypto.digests;

import org.bouncycastle.crypto.DataLengthException;
import org.bouncycastle.crypto.Digest;
import org.bouncycastle.crypto.Xof;
import org.bouncycastle.util.Strings;

/**
 * TupleHash - a hash designed  to  simply  hash  a  tuple  of  input  strings,  any  or  all  of  which  may  be  empty  strings,
 *  in  an  unambiguous way with an optional XOF mode.
 * 

* From NIST Special Publication 800-185 - SHA-3 Derived Functions:cSHAKE, KMAC, TupleHash and ParallelHash *

*/ public class TupleHash implements Xof, Digest { private static final byte[] N_TUPLE_HASH = Strings.toByteArray("TupleHash"); private final CSHAKEDigest cshake; private final int bitLength; private final int outputLength; private boolean firstOutput; /** * Base constructor. * * @param bitLength bit length of the underlying SHAKE function, 128 or 256. * @param S the customization string - available for local use. */ public TupleHash(int bitLength, byte[] S) { this(bitLength, S, bitLength * 2); } public TupleHash(int bitLength, byte[] S, int outputSize) { this.cshake = new CSHAKEDigest(bitLength, N_TUPLE_HASH, S); this.bitLength = bitLength; this.outputLength = (outputSize + 7) / 8; reset(); } public TupleHash(TupleHash original) { this.cshake = new CSHAKEDigest(original.cshake); this.bitLength = cshake.fixedOutputLength; this.outputLength = bitLength * 2 / 8; this.firstOutput = original.firstOutput; } public String getAlgorithmName() { return "TupleHash" + cshake.getAlgorithmName().substring(6); } public int getByteLength() { return cshake.getByteLength(); } public int getDigestSize() { return outputLength; } public void update(byte in) throws IllegalStateException { byte[] bytes = XofUtils.encode(in); cshake.update(bytes, 0, bytes.length); } public void update(byte[] in, int inOff, int len) throws DataLengthException, IllegalStateException { byte[] bytes = XofUtils.encode(in, inOff, len); cshake.update(bytes, 0, bytes.length); } private void wrapUp(int outputSize) { byte[] encOut = XofUtils.rightEncode(outputSize * 8L); cshake.update(encOut, 0, encOut.length); firstOutput = false; } public int doFinal(byte[] out, int outOff) throws DataLengthException, IllegalStateException { if (firstOutput) { wrapUp(getDigestSize()); } int rv = cshake.doFinal(out, outOff, getDigestSize()); reset(); return rv; } public int doFinal(byte[] out, int outOff, int outLen) { if (firstOutput) { wrapUp(getDigestSize()); } int rv = cshake.doFinal(out, outOff, outLen); reset(); return rv; } public int doOutput(byte[] out, int outOff, int outLen) { if (firstOutput) { wrapUp(0); } return cshake.doOutput(out, outOff, outLen); } public void reset() { cshake.reset(); firstOutput = true; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy