| /* |
| * Copyright 2025 The OpenSSL Project Authors. All Rights Reserved. |
| * |
| * Licensed under the Apache License 2.0 (the "License"). You may not use |
| * this file except in compliance with the License. You can obtain a copy |
| * in the file LICENSE in the source distribution or at |
| * https://www.openssl.org/source/license.html |
| */ |
| |
| /* including crypto/sha.h requires this for SHA256_CTX */ |
| #include "internal/deprecated.h" |
| |
| /* |
| * NIST.SP.800-185 Encoding/Padding Methods used for SHA3 derived functions |
| * e.g. It is used by KMAC and cSHAKE |
| */ |
| |
| #include <string.h> /* memcpy */ |
| #include <openssl/err.h> |
| #include <openssl/proverr.h> |
| #include "crypto/sha.h" |
| #include "internal/common.h" /* ossl_assert */ |
| |
| /* Returns the number of bytes required to store 'bits' into a byte array */ |
| static unsigned int get_encode_size(size_t bits) |
| { |
| unsigned int cnt = 0, sz = sizeof(size_t); |
| |
| while (bits && (cnt < sz)) { |
| ++cnt; |
| bits >>= 8; |
| } |
| /* If bits is zero 1 byte is required */ |
| if (cnt == 0) |
| cnt = 1; |
| return cnt; |
| } |
| |
| /* |
| * Convert an integer into bytes. The number of bytes is appended |
| * to the end of the buffer. |
| * Returns an array of bytes 'out' of size *out_len. |
| * |
| * e.g if bits = 32, out[2] = { 0x20, 0x01 } |
| */ |
| int ossl_sp800_185_right_encode(unsigned char *out, |
| size_t out_max_len, size_t *out_len, |
| size_t bits) |
| { |
| unsigned int len = get_encode_size(bits); |
| int i; |
| |
| if (len >= out_max_len) { |
| ERR_raise(ERR_LIB_PROV, PROV_R_LENGTH_TOO_LARGE); |
| return 0; |
| } |
| |
| /* MSB's are at the start of the bytes array */ |
| for (i = len - 1; i >= 0; --i) { |
| out[i] = (unsigned char)(bits & 0xFF); |
| bits >>= 8; |
| } |
| /* Tack the length onto the end */ |
| out[len] = (unsigned char)len; |
| |
| /* The Returned length includes the tacked on byte */ |
| *out_len = len + 1; |
| return 1; |
| } |
| |
| /* |
| * Encodes a string with a left encoded length added. Note that the |
| * in_len is converted to bits (* 8). |
| * |
| * e.g- in="KMAC" gives out[6] = { 0x01, 0x20, 0x4B, 0x4D, 0x41, 0x43 } |
| * len bits K M A C |
| */ |
| int ossl_sp800_185_encode_string(unsigned char *out, |
| size_t out_max_len, size_t *out_len, |
| const unsigned char *in, size_t in_len) |
| { |
| if (in == NULL) { |
| *out_len = 0; |
| } else { |
| size_t i, bits, len, sz; |
| |
| bits = 8 * in_len; |
| len = get_encode_size(bits); |
| sz = 1 + len + in_len; |
| |
| if (sz > out_max_len) { |
| ERR_raise(ERR_LIB_PROV, PROV_R_LENGTH_TOO_LARGE); |
| return 0; |
| } |
| |
| out[0] = (unsigned char)len; |
| for (i = len; i > 0; --i) { |
| out[i] = (bits & 0xFF); |
| bits >>= 8; |
| } |
| memcpy(out + len + 1, in, in_len); |
| *out_len = sz; |
| } |
| return 1; |
| } |
| |
| /* |
| * Returns a zero padded encoding of the inputs in1 and an optional |
| * in2 (can be NULL). The padded output must be a multiple of the blocksize 'w'. |
| * The value of w is in bytes (< 256). |
| * |
| * The returned output is: |
| * zero_padded(multiple of w, (left_encode(w) || in1 [|| in2]) |
| */ |
| int ossl_sp800_185_bytepad(unsigned char *out, size_t out_len_max, size_t *out_len, |
| const unsigned char *in1, size_t in1_len, |
| const unsigned char *in2, size_t in2_len, |
| size_t w) |
| { |
| size_t len; |
| unsigned char *p = out; |
| size_t sz; |
| |
| if (!ossl_assert(w <= 255)) |
| return 0; |
| sz = (2 + in1_len + (in2 != NULL ? in2_len : 0) + w - 1) / w * w; |
| if (out_len_max != 0 && sz > out_len_max) |
| return 0; |
| |
| if (out == NULL) { |
| if (out_len == NULL) { |
| ERR_raise(ERR_LIB_PROV, ERR_R_PASSED_NULL_PARAMETER); |
| return 0; |
| } |
| *out_len = sz; |
| return 1; |
| } |
| |
| /* Left encoded w */ |
| *p++ = 1; |
| *p++ = (unsigned char)w; |
| /* || in1 */ |
| memcpy(p, in1, in1_len); |
| p += in1_len; |
| /* [ || in2 ] */ |
| if (in2 != NULL && in2_len > 0) { |
| memcpy(p, in2, in2_len); |
| p += in2_len; |
| } |
| /* Figure out the pad size (divisible by w) */ |
| len = p - out; |
| /* zero pad the end of the buffer */ |
| if (sz != len) |
| memset(p, 0, sz - len); |
| if (out_len != NULL) |
| *out_len = sz; |
| return 1; |
| } |