|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
|
|
/* |
|
* (C) Copyright IBM Corp. 2013 |
|
*/ |
|
|
|
package com.sun.crypto.provider; |
|
|
|
import java.nio.ByteBuffer; |
|
import java.nio.ByteOrder; |
|
import java.util.Arrays; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
final class GCTR extends CounterMode implements GCM { |
|
|
|
|
|
private static final int MAX_LEN = 1024; |
|
private byte[] block; |
|
|
|
GCTR(SymmetricCipher cipher, byte[] initialCounterBlk) { |
|
super(cipher); |
|
if (initialCounterBlk.length != blockSize) { |
|
throw new RuntimeException("length of initial counter block (" + |
|
initialCounterBlk.length + ") not equal to blockSize (" + |
|
blockSize + ")"); |
|
} |
|
|
|
iv = initialCounterBlk; |
|
reset(); |
|
} |
|
|
|
@Override |
|
String getFeedback() { |
|
return "GCTR"; |
|
} |
|
|
|
|
|
private long blocksUntilRollover() { |
|
ByteBuffer buf = ByteBuffer.wrap(counter, counter.length - 4, 4); |
|
buf.order(ByteOrder.BIG_ENDIAN); |
|
long ctr32 = 0xFFFFFFFFL & buf.getInt(); |
|
long blocksLeft = (1L << 32) - ctr32; |
|
return blocksLeft; |
|
} |
|
|
|
private void checkBlock() { |
|
if (block == null) { |
|
block = new byte[blockSize]; |
|
} else { |
|
Arrays.fill(block, (byte)0); |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
public int update(byte[] in, int inOfs, int inLen, byte[] out, int outOfs) { |
|
if (inLen == 0) { |
|
return 0; |
|
} |
|
|
|
if (inLen - inOfs > in.length) { |
|
throw new RuntimeException("input length out of bound"); |
|
} |
|
if (inLen < 0) { |
|
throw new RuntimeException("input length unsupported"); |
|
} |
|
if (out.length - outOfs < (inLen - (inLen % blockSize))) { |
|
throw new RuntimeException("output buffer too small"); |
|
} |
|
|
|
inLen -= inLen % blockSize; |
|
long blocksLeft = blocksUntilRollover(); |
|
int numOfCompleteBlocks = inLen / blockSize; |
|
if (numOfCompleteBlocks >= blocksLeft) { |
|
// Counter Mode encryption cannot be used because counter will |
|
|
|
checkBlock(); |
|
for (int i = 0; i < numOfCompleteBlocks; i++) { |
|
embeddedCipher.encryptBlock(counter, 0, block, 0); |
|
for (int n = 0; n < blockSize; n++) { |
|
int index = (i * blockSize + n); |
|
out[outOfs + index] = |
|
(byte) ((in[inOfs + index] ^ block[n])); |
|
} |
|
GaloisCounterMode.increment32(counter); |
|
} |
|
return inLen; |
|
} else { |
|
return encrypt(in, inOfs, inLen, out, outOfs); |
|
} |
|
} |
|
|
|
|
|
|
|
*/ |
|
public int update(byte[] in, int inOfs, int inLen, ByteBuffer dst) { |
|
// If the bytebuffer is backed by arrays, use that instead of |
|
|
|
if (!dst.isDirect()) { |
|
int len = update(in, inOfs, inLen, dst.array(), |
|
dst.arrayOffset() + dst.position()); |
|
dst.position(dst.position() + len); |
|
return len; |
|
} |
|
|
|
|
|
if (inLen - inOfs > in.length) { |
|
throw new RuntimeException("input length out of bound"); |
|
} |
|
if (inLen < 0) { |
|
throw new RuntimeException("input length unsupported"); |
|
} |
|
// See GaloisCounterMode. decryptFinal(bytebuffer, bytebuffer) for |
|
// details on the check for 'dst' having enough space for the result. |
|
|
|
long blocksLeft = blocksUntilRollover(); |
|
int numOfCompleteBlocks = inLen / blockSize; |
|
if (numOfCompleteBlocks >= blocksLeft) { |
|
// Counter Mode encryption cannot be used because counter will |
|
|
|
checkBlock(); |
|
for (int i = 0; i < numOfCompleteBlocks; i++) { |
|
embeddedCipher.encryptBlock(counter, 0, block, 0); |
|
for (int n = 0; n < blockSize; n++) { |
|
int index = (i * blockSize + n); |
|
dst.put((byte) ((in[inOfs + index] ^ block[n]))); |
|
} |
|
GaloisCounterMode.increment32(counter); |
|
} |
|
return inLen; |
|
} else { |
|
int len = inLen - inLen % blockSize; |
|
int processed = len; |
|
byte[] out = new byte[Math.min(MAX_LEN, len)]; |
|
int offset = inOfs; |
|
while (processed > MAX_LEN) { |
|
encrypt(in, offset, MAX_LEN, out, 0); |
|
dst.put(out, 0, MAX_LEN); |
|
processed -= MAX_LEN; |
|
offset += MAX_LEN; |
|
} |
|
encrypt(in, offset, processed, out, 0); |
|
// If dst is less than blocksize, insert only what it can. Extra |
|
// bytes would cause buffers with enough size to fail with a |
|
|
|
dst.put(out, 0, Math.min(dst.remaining(), processed)); |
|
return len; |
|
} |
|
} |
|
|
|
|
|
|
|
*/ |
|
public int update(ByteBuffer src, ByteBuffer dst) { |
|
int len; |
|
|
|
// If the bytebuffer is backed by arrays, use that instead of |
|
|
|
if (src.hasArray() && dst.hasArray()) { |
|
len = update(src.array(), src.arrayOffset() + src.position(), |
|
src.remaining() - (src.remaining() % blockSize), |
|
dst.array(), dst.arrayOffset() + dst.position()); |
|
src.position(src.position() + len); |
|
dst.position(dst.position() + len); |
|
return len; |
|
} |
|
|
|
|
|
long blocksLeft = blocksUntilRollover(); |
|
int numOfCompleteBlocks = src.remaining() / blockSize; |
|
if (numOfCompleteBlocks >= blocksLeft) { |
|
// Counter Mode encryption cannot be used because counter will |
|
|
|
checkBlock(); |
|
for (int i = 0; i < numOfCompleteBlocks; i++) { |
|
embeddedCipher.encryptBlock(counter, 0, block, 0); |
|
for (int n = 0; n < blockSize; n++) { |
|
dst.put((byte) (src.get() ^ block[n])); |
|
} |
|
GaloisCounterMode.increment32(counter); |
|
} |
|
return numOfCompleteBlocks * blockSize; |
|
} |
|
|
|
len = src.remaining() - (src.remaining() % blockSize); |
|
int processed = len; |
|
byte[] in = new byte[Math.min(MAX_LEN, len)]; |
|
while (processed > MAX_LEN) { |
|
src.get(in, 0, MAX_LEN); |
|
encrypt(in, 0, MAX_LEN, in, 0); |
|
dst.put(in, 0, MAX_LEN); |
|
processed -= MAX_LEN; |
|
} |
|
src.get(in, 0, processed); |
|
encrypt(in, 0, processed, in, 0); |
|
dst.put(in, 0, processed); |
|
return len; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
public int doFinal(byte[] in, int inOfs, int inLen, byte[] out, |
|
int outOfs) { |
|
if (inLen == 0) { |
|
return 0; |
|
} |
|
int lastBlockSize = inLen % blockSize; |
|
int completeBlkLen = inLen - lastBlockSize; |
|
|
|
update(in, inOfs, completeBlkLen, out, outOfs); |
|
if (lastBlockSize != 0) { |
|
|
|
checkBlock(); |
|
embeddedCipher.encryptBlock(counter, 0, block, 0); |
|
for (int n = 0; n < lastBlockSize; n++) { |
|
out[outOfs + completeBlkLen + n] = |
|
(byte) ((in[inOfs + completeBlkLen + n] ^ block[n])); |
|
} |
|
} |
|
return inLen; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
public int doFinal(ByteBuffer src, ByteBuffer dst) { |
|
// If the bytebuffer is backed by arrays, use that instead of |
|
|
|
if (src.hasArray() && dst.hasArray()) { |
|
int len = doFinal(src.array(), src.arrayOffset() + src.position(), |
|
src.remaining(), dst.array(), |
|
dst.arrayOffset() + dst.position()); |
|
src.position(src.position() + len); |
|
dst.position(dst.position() + len); |
|
return len; |
|
} |
|
|
|
int len = src.remaining(); |
|
int lastBlockSize = len % blockSize; |
|
update(src, dst); |
|
if (lastBlockSize != 0) { |
|
checkBlock(); |
|
|
|
embeddedCipher.encryptBlock(counter, 0, block, 0); |
|
for (int n = 0; n < lastBlockSize; n++) { |
|
dst.put((byte) (src.get() ^ block[n])); |
|
} |
|
} |
|
return len; |
|
} |
|
} |