; libcrash -- cryptographic hash functions ; ; Copyright (C) 2012-2013,2016,2019 Ivan Baravy (dunkaist) ; ; This program is free software: you can redistribute it and/or modify ; it under the terms of the GNU General Public License as published by ; the Free Software Foundation, either version 3 of the License, or ; (at your option) any later version. ; ; This program is distributed in the hope that it will be useful, ; but WITHOUT ANY WARRANTY; without even the implied warranty of ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ; GNU General Public License for more details. ; ; You should have received a copy of the GNU General Public License ; along with this program. If not, see <http://www.gnu.org/licenses/>. MD4_HASH_SIZE = 16 MD4_BLOCK_SIZE = 64 MD4_ALIGN = 4 MD4_ALIGN_MASK = MD4_ALIGN - 1 struct ctx_md4 hash rb MD4_HASH_SIZE block rb MD4_BLOCK_SIZE index rd 1 msglen_0 rd 1 msglen_1 rd 1 ends if defined sizeof.crash_ctx assert sizeof.crash_ctx >= sizeof.ctx_md4 end if macro md4._.f b, c, d { mov eax, c xor eax, d and eax, b xor eax, d } macro md4._.g b, c, d { push c d mov eax, b and eax, c and c, d and d, b or eax, c or eax, d pop d c } macro md4._.h b, c, d { mov eax, b xor eax, c xor eax, d } macro md4._.round func, a, b, c, d, index, shift, ac { func b, c, d add eax, [esi + index*4] lea a, [a + eax + ac] rol a, shift } proc md4.init _ctx mov ebx, [_ctx] lea edi, [ebx + ctx_md4.hash] mov esi, md4._.hash_init mov ecx, MD4_HASH_SIZE/4 rep movsd xor eax, eax mov [ebx + ctx_md4.index], eax mov [ebx + ctx_md4.msglen_0], eax mov [ebx + ctx_md4.msglen_1], eax ret endp proc md4._.block _hash mov eax, [_hash] mov edi, [eax + 0x0] mov ebx, [eax + 0x4] mov ecx, [eax + 0x8] mov edx, [eax + 0xc] md4._.round md4._.f, edi, ebx, ecx, edx, 0, 3, 0x00000000 md4._.round md4._.f, edx, edi, ebx, ecx, 1, 7, 0x00000000 md4._.round md4._.f, ecx, edx, edi, ebx, 2, 11, 0x00000000 md4._.round md4._.f, ebx, ecx, edx, edi, 3, 19, 0x00000000 md4._.round md4._.f, edi, ebx, ecx, edx, 4, 3, 0x00000000 md4._.round md4._.f, edx, edi, ebx, ecx, 5, 7, 0x00000000 md4._.round md4._.f, ecx, edx, edi, ebx, 6, 11, 0x00000000 md4._.round md4._.f, ebx, ecx, edx, edi, 7, 19, 0x00000000 md4._.round md4._.f, edi, ebx, ecx, edx, 8, 3, 0x00000000 md4._.round md4._.f, edx, edi, ebx, ecx, 9, 7, 0x00000000 md4._.round md4._.f, ecx, edx, edi, ebx, 10, 11, 0x00000000 md4._.round md4._.f, ebx, ecx, edx, edi, 11, 19, 0x00000000 md4._.round md4._.f, edi, ebx, ecx, edx, 12, 3, 0x00000000 md4._.round md4._.f, edx, edi, ebx, ecx, 13, 7, 0x00000000 md4._.round md4._.f, ecx, edx, edi, ebx, 14, 11, 0x00000000 md4._.round md4._.f, ebx, ecx, edx, edi, 15, 19, 0x00000000 md4._.round md4._.g, edi, ebx, ecx, edx, 0, 3, 0x5a827999 md4._.round md4._.g, edx, edi, ebx, ecx, 4, 5, 0x5a827999 md4._.round md4._.g, ecx, edx, edi, ebx, 8, 9, 0x5a827999 md4._.round md4._.g, ebx, ecx, edx, edi, 12, 13, 0x5a827999 md4._.round md4._.g, edi, ebx, ecx, edx, 1, 3, 0x5a827999 md4._.round md4._.g, edx, edi, ebx, ecx, 5, 5, 0x5a827999 md4._.round md4._.g, ecx, edx, edi, ebx, 9, 9, 0x5a827999 md4._.round md4._.g, ebx, ecx, edx, edi, 13, 13, 0x5a827999 md4._.round md4._.g, edi, ebx, ecx, edx, 2, 3, 0x5a827999 md4._.round md4._.g, edx, edi, ebx, ecx, 6, 5, 0x5a827999 md4._.round md4._.g, ecx, edx, edi, ebx, 10, 9, 0x5a827999 md4._.round md4._.g, ebx, ecx, edx, edi, 14, 13, 0x5a827999 md4._.round md4._.g, edi, ebx, ecx, edx, 3, 3, 0x5a827999 md4._.round md4._.g, edx, edi, ebx, ecx, 7, 5, 0x5a827999 md4._.round md4._.g, ecx, edx, edi, ebx, 11, 9, 0x5a827999 md4._.round md4._.g, ebx, ecx, edx, edi, 15, 13, 0x5a827999 md4._.round md4._.h, edi, ebx, ecx, edx, 0, 3, 0x6ed9eba1 md4._.round md4._.h, edx, edi, ebx, ecx, 8, 9, 0x6ed9eba1 md4._.round md4._.h, ecx, edx, edi, ebx, 4, 11, 0x6ed9eba1 md4._.round md4._.h, ebx, ecx, edx, edi, 12, 15, 0x6ed9eba1 md4._.round md4._.h, edi, ebx, ecx, edx, 2, 3, 0x6ed9eba1 md4._.round md4._.h, edx, edi, ebx, ecx, 10, 9, 0x6ed9eba1 md4._.round md4._.h, ecx, edx, edi, ebx, 6, 11, 0x6ed9eba1 md4._.round md4._.h, ebx, ecx, edx, edi, 14, 15, 0x6ed9eba1 md4._.round md4._.h, edi, ebx, ecx, edx, 1, 3, 0x6ed9eba1 md4._.round md4._.h, edx, edi, ebx, ecx, 9, 9, 0x6ed9eba1 md4._.round md4._.h, ecx, edx, edi, ebx, 5, 11, 0x6ed9eba1 md4._.round md4._.h, ebx, ecx, edx, edi, 13, 15, 0x6ed9eba1 md4._.round md4._.h, edi, ebx, ecx, edx, 3, 3, 0x6ed9eba1 md4._.round md4._.h, edx, edi, ebx, ecx, 11, 9, 0x6ed9eba1 md4._.round md4._.h, ecx, edx, edi, ebx, 7, 11, 0x6ed9eba1 md4._.round md4._.h, ebx, ecx, edx, edi, 15, 15, 0x6ed9eba1 mov eax, [_hash] add [eax + 0x0], edi add [eax + 0x4], ebx add [eax + 0x8], ecx add [eax + 0xc], edx ret endp proc md4.update _ctx, _msg, _size mov ebx, [_ctx] mov ecx, [_size] add [ebx + ctx_md4.msglen_0], ecx adc [ebx + ctx_md4.msglen_1], 0 .next_block: mov ebx, [_ctx] mov esi, [_msg] mov eax, [ebx + ctx_md4.index] and eax, MD4_BLOCK_SIZE-1 jnz .copy_to_buf test esi, MD4_ALIGN_MASK jnz .copy_to_buf .no_copy: ; data is aligned, hash it in place without copying mov ebx, [_ctx] cmp [_size], MD4_BLOCK_SIZE jb .copy_quit lea eax, [ebx + ctx_md4.hash] stdcall md4._.block, eax sub [_size], MD4_BLOCK_SIZE add esi, MD4_BLOCK_SIZE jmp .no_copy .copy_to_buf: lea edi, [ebx + ctx_md4.block] add edi, eax mov ecx, MD4_BLOCK_SIZE sub ecx, eax cmp [_size], ecx jb .copy_quit sub [_size], ecx add [_msg], ecx add [ebx + ctx_md4.index], ecx rep movsb lea eax, [ebx + ctx_md4.hash] lea esi, [ebx + ctx_md4.block] stdcall md4._.block, eax jmp .next_block .copy_quit: mov ebx, [_ctx] lea edi, [ebx + ctx_md4.block] mov eax, [ebx + ctx_md4.index] and eax, MD4_BLOCK_SIZE-1 add edi, eax mov ecx, [_size] add [ebx + ctx_md4.index], ecx rep movsb .quit: ret endp proc md4.final _ctx mov ebx, [_ctx] lea edi, [ebx + ctx_md4.block] mov ecx, [ebx + ctx_md4.msglen_0] and ecx, MD4_BLOCK_SIZE-1 add edi, ecx mov byte[edi], 0x80 inc edi neg ecx add ecx, MD4_BLOCK_SIZE cmp ecx, 8 ja .last dec ecx xor eax, eax rep stosb lea esi, [ebx + ctx_md4.block] lea eax, [ebx + ctx_md4.hash] stdcall md4._.block, eax mov ebx, [_ctx] lea edi, [ebx + ctx_md4.block] mov ecx, MD4_BLOCK_SIZE+1 .last: dec ecx sub ecx, 8 xor eax, eax rep stosb mov eax, [ebx + ctx_md4.msglen_0] mov edx, [ebx + ctx_md4.msglen_1] shld edx, eax, 3 shl eax, 3 mov dword[edi], eax mov dword[edi+4], edx lea esi, [ebx + ctx_md4.block] lea eax, [ebx + ctx_md4.hash] stdcall md4._.block, eax ret endp proc md4.oneshot _ctx, _data, _len stdcall md4.init, [_ctx] stdcall md4.update, [_ctx], [_data], [_len] stdcall md4.final, [_ctx] ret endp iglobal align MD4_ALIGN md4._.hash_init dd 0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0 endg