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

org.springframework.security.crypto.password.Md4 Maven / Gradle / Ivy

There is a newer version: 6.2.4
Show newest version
/*
 * Copyright 2002-2017 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.security.crypto.password;

/**
 * Implementation of the MD4 message digest derived from the RSA Data Security, Inc, MD4
 * Message-Digest Algorithm.
 *
 * @author Alan Stewart
 */
class Md4 {

	private static final int BLOCK_SIZE = 64;

	private static final int HASH_SIZE = 16;

	private final byte[] buffer = new byte[BLOCK_SIZE];

	private int bufferOffset;

	private long byteCount;

	private final int[] state = new int[4];

	private final int[] tmp = new int[16];

	Md4() {
		reset();
	}

	void reset() {
		this.bufferOffset = 0;
		this.byteCount = 0;
		this.state[0] = 0x67452301;
		this.state[1] = 0xEFCDAB89;
		this.state[2] = 0x98BADCFE;
		this.state[3] = 0x10325476;
	}

	byte[] digest() {
		byte[] resBuf = new byte[HASH_SIZE];
		digest(resBuf, 0, HASH_SIZE);
		return resBuf;
	}

	private void digest(byte[] buffer, int off) {
		for (int i = 0; i < 4; i++) {
			for (int j = 0; j < 4; j++) {
				buffer[off + (i * 4 + j)] = (byte) (this.state[i] >>> (8 * j));
			}
		}
	}

	private void digest(byte[] buffer, int offset, int len) {
		this.buffer[this.bufferOffset++] = (byte) 0x80;
		int lenOfBitLen = 8;
		int C = BLOCK_SIZE - lenOfBitLen;
		if (this.bufferOffset > C) {
			while (this.bufferOffset < BLOCK_SIZE) {
				this.buffer[this.bufferOffset++] = (byte) 0x00;
			}
			update(this.buffer, 0);
			this.bufferOffset = 0;
		}
		while (this.bufferOffset < C) {
			this.buffer[this.bufferOffset++] = (byte) 0x00;
		}
		long bitCount = this.byteCount * 8;
		for (int i = 0; i < 64; i += 8) {
			this.buffer[this.bufferOffset++] = (byte) (bitCount >>> (i));
		}
		update(this.buffer, 0);
		digest(buffer, offset);
	}

	void update(byte[] input, int offset, int length) {
		this.byteCount += length;
		int todo;
		while (length >= (todo = BLOCK_SIZE - this.bufferOffset)) {
			System.arraycopy(input, offset, this.buffer, this.bufferOffset, todo);
			update(this.buffer, 0);
			length -= todo;
			offset += todo;
			this.bufferOffset = 0;
		}

		System.arraycopy(input, offset, this.buffer, this.bufferOffset, length);
		this.bufferOffset += length;
	}

	private void update(byte[] block, int offset) {
		for (int i = 0; i < 16; i++) {
			this.tmp[i] = (block[offset++] & 0xFF) | (block[offset++] & 0xFF) << 8 | (block[offset++] & 0xFF) << 16
					| (block[offset++] & 0xFF) << 24;
		}

		int A = this.state[0];
		int B = this.state[1];
		int C = this.state[2];
		int D = this.state[3];

		A = FF(A, B, C, D, this.tmp[0], 3);
		D = FF(D, A, B, C, this.tmp[1], 7);
		C = FF(C, D, A, B, this.tmp[2], 11);
		B = FF(B, C, D, A, this.tmp[3], 19);
		A = FF(A, B, C, D, this.tmp[4], 3);
		D = FF(D, A, B, C, this.tmp[5], 7);
		C = FF(C, D, A, B, this.tmp[6], 11);
		B = FF(B, C, D, A, this.tmp[7], 19);
		A = FF(A, B, C, D, this.tmp[8], 3);
		D = FF(D, A, B, C, this.tmp[9], 7);
		C = FF(C, D, A, B, this.tmp[10], 11);
		B = FF(B, C, D, A, this.tmp[11], 19);
		A = FF(A, B, C, D, this.tmp[12], 3);
		D = FF(D, A, B, C, this.tmp[13], 7);
		C = FF(C, D, A, B, this.tmp[14], 11);
		B = FF(B, C, D, A, this.tmp[15], 19);

		A = GG(A, B, C, D, this.tmp[0], 3);
		D = GG(D, A, B, C, this.tmp[4], 5);
		C = GG(C, D, A, B, this.tmp[8], 9);
		B = GG(B, C, D, A, this.tmp[12], 13);
		A = GG(A, B, C, D, this.tmp[1], 3);
		D = GG(D, A, B, C, this.tmp[5], 5);
		C = GG(C, D, A, B, this.tmp[9], 9);
		B = GG(B, C, D, A, this.tmp[13], 13);
		A = GG(A, B, C, D, this.tmp[2], 3);
		D = GG(D, A, B, C, this.tmp[6], 5);
		C = GG(C, D, A, B, this.tmp[10], 9);
		B = GG(B, C, D, A, this.tmp[14], 13);
		A = GG(A, B, C, D, this.tmp[3], 3);
		D = GG(D, A, B, C, this.tmp[7], 5);
		C = GG(C, D, A, B, this.tmp[11], 9);
		B = GG(B, C, D, A, this.tmp[15], 13);

		A = HH(A, B, C, D, this.tmp[0], 3);
		D = HH(D, A, B, C, this.tmp[8], 9);
		C = HH(C, D, A, B, this.tmp[4], 11);
		B = HH(B, C, D, A, this.tmp[12], 15);
		A = HH(A, B, C, D, this.tmp[2], 3);
		D = HH(D, A, B, C, this.tmp[10], 9);
		C = HH(C, D, A, B, this.tmp[6], 11);
		B = HH(B, C, D, A, this.tmp[14], 15);
		A = HH(A, B, C, D, this.tmp[1], 3);
		D = HH(D, A, B, C, this.tmp[9], 9);
		C = HH(C, D, A, B, this.tmp[5], 11);
		B = HH(B, C, D, A, this.tmp[13], 15);
		A = HH(A, B, C, D, this.tmp[3], 3);
		D = HH(D, A, B, C, this.tmp[11], 9);
		C = HH(C, D, A, B, this.tmp[7], 11);
		B = HH(B, C, D, A, this.tmp[15], 15);

		this.state[0] += A;
		this.state[1] += B;
		this.state[2] += C;
		this.state[3] += D;
	}

	private int FF(int a, int b, int c, int d, int x, int s) {
		int t = a + ((b & c) | (~b & d)) + x;
		return t << s | t >>> (32 - s);
	}

	private int GG(int a, int b, int c, int d, int x, int s) {
		int t = a + ((b & (c | d)) | (c & d)) + x + 0x5A827999;
		return t << s | t >>> (32 - s);
	}

	private int HH(int a, int b, int c, int d, int x, int s) {
		int t = a + (b ^ c ^ d) + x + 0x6ED9EBA1;
		return t << s | t >>> (32 - s);
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy