123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225 |
- /*
- * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
- *
- * Licensed under the Apache License, Version 2.0 (the "License").
- * You may not use this file except in compliance with the License.
- * A copy of the License is located at
- *
- * http://aws.amazon.com/apache2.0
- *
- * or in the "license" file accompanying this file. This file 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.
- */
- #include "utils/s2n_safety.h"
- #include <stdint.h>
- #include <stdio.h>
- #include "utils/s2n_annotations.h"
- /**
- * Given arrays "a" and "b" of length "len", determine whether they
- * hold equal contents.
- *
- * The execution time of this function is independent of the values
- * stored in the arrays.
- *
- * Timing may depend on the length of the arrays, and on the location
- * of the arrays in memory (e.g. if a buffer has been paged out, this
- * will affect the timing of this function).
- *
- * Returns:
- * Whether all bytes in arrays "a" and "b" are identical
- */
- bool s2n_constant_time_equals(const uint8_t *a, const uint8_t *b, const uint32_t len)
- {
- S2N_PUBLIC_INPUT(a);
- S2N_PUBLIC_INPUT(b);
- S2N_PUBLIC_INPUT(len);
- /* if len is 0, they're always going to be equal */
- if (len == 0) {
- return true;
- }
- /* check if a and b are readable - if so, allow them to increment their pointer */
- uint8_t a_inc = S2N_MEM_IS_READABLE(a, len) ? 1 : 0;
- uint8_t b_inc = S2N_MEM_IS_READABLE(b, len) ? 1 : 0;
- /* reserve a stand-in pointer to replace NULL pointers */
- static uint8_t standin = 0;
- /* if the pointers can increment their values, then use the
- * original value; otherwise use the stand-in */
- const uint8_t *a_ptr = a_inc ? a : &standin;
- const uint8_t *b_ptr = b_inc ? b : &standin;
- /* start by assuming they are equal only if both increment their pointer */
- uint8_t xor = !((a_inc == 1) & (b_inc == 1));
- /* iterate over each byte in the slices */
- for (size_t i = 0; i < len; i++) {
- /* Invariants must hold for each execution of the loop
- * and at loop exit, hence the <= */
- S2N_INVARIANT(i <= len);
- /* mix the current cursor values in to the result */
- xor |= *a_ptr ^ *b_ptr;
- /* increment the pointers by their "inc" values */
- a_ptr += a_inc;
- b_ptr += b_inc;
- }
- /* finally check to make sure xor is still 0 */
- return (xor == 0);
- }
- /**
- * Given arrays "dest" and "src" of length "len", conditionally copy "src" to "dest"
- * The execution time of this function is independent of the values
- * stored in the arrays, and of whether the copy occurs.
- *
- * Timing may depend on the length of the arrays, and on the location
- * of the arrays in memory (e.g. if a buffer has been paged out, this
- * will affect the timing of this function).
- *
- */
- int s2n_constant_time_copy_or_dont(uint8_t *dest, const uint8_t *src, uint32_t len, uint8_t dont)
- {
- S2N_PUBLIC_INPUT(dest);
- S2N_PUBLIC_INPUT(src);
- S2N_PUBLIC_INPUT(len);
- uint8_t mask = (((0xFFFF & dont) - 1) >> 8) & 0xFF;
- /* dont = 0 : mask = 0xff */
- /* dont > 0 : mask = 0x00 */
- for (size_t i = 0; i < len; i++) {
- uint8_t old = dest[i];
- uint8_t diff = (old ^ src[i]) & mask;
- dest[i] = old ^ diff;
- }
- return 0;
- }
- /* If src contains valid PKCS#1 v1.5 padding of exactly expectlen bytes, decode
- * it into dst, otherwise leave dst alone. Execution time is independent of the
- * content of src, but may depend on srclen/expectlen.
- *
- * Normally, one would fill dst with random bytes before calling this function.
- */
- int s2n_constant_time_pkcs1_unpad_or_dont(uint8_t *dst, const uint8_t *src, uint32_t srclen, uint32_t expectlen)
- {
- S2N_PUBLIC_INPUT(dst);
- S2N_PUBLIC_INPUT(src);
- S2N_PUBLIC_INPUT(srclen);
- S2N_PUBLIC_INPUT(expectlen);
- /* Before doing anything else, some basic sanity checks on input lengths */
- if (srclen < expectlen + 3) {
- /* Not enough room for PKCS#1v1.5 padding, so treat it as bad padding */
- return 0;
- }
- /* First, determine (in constant time) whether the padding is valid.
- * If the padding is valid we expect that:
- * Bytes 0 and 1 will equal 0x00 and 0x02
- * Bytes (srclen-expectlen-1) will be zero
- * Bytes 2 through (srclen-expectlen-1) will be nonzero
- */
- uint8_t dont_copy = 0;
- const uint8_t *start_of_data = src + srclen - expectlen;
- dont_copy |= src[0] ^ 0x00;
- dont_copy |= src[1] ^ 0x02;
- dont_copy |= *(start_of_data - 1) ^ 0x00;
- for (size_t i = 2; i < srclen - expectlen - 1; i++) {
- /* Note! We avoid using logical NOT (!) here; while in practice
- * many compilers will use constant-time sequences for this operator,
- * at least on x86 (e.g. cmp -> setcc, or vectorized pcmpeq), this is
- * not guaranteed to hold, and some architectures might not have a
- * convenient mechanism for generating a branchless logical not. */
- uint8_t mask = (((0xFFFF & src[i]) - 1) >> 8) & 0xFF;
- /* src[i] = 0 : mask = 0xff */
- /* src[i] > 0 : mask = 0x00 */
- dont_copy |= mask;
- }
- s2n_constant_time_copy_or_dont(dst, start_of_data, expectlen, dont_copy);
- return 0;
- }
- static bool s2n_in_unit_test_value = false;
- static bool s2n_in_integ_test_value = false;
- int s2n_in_unit_test_set(bool is_unit)
- {
- s2n_in_unit_test_value = is_unit;
- return S2N_SUCCESS;
- }
- int s2n_in_integ_test_set(bool is_integ)
- {
- s2n_in_integ_test_value = is_integ;
- return S2N_SUCCESS;
- }
- bool s2n_in_unit_test()
- {
- return s2n_in_unit_test_value;
- }
- bool s2n_in_test()
- {
- return s2n_in_unit_test_value || s2n_in_integ_test_value;
- }
- int s2n_align_to(uint32_t initial, uint32_t alignment, uint32_t *out)
- {
- POSIX_ENSURE_REF(out);
- POSIX_ENSURE(alignment != 0, S2N_ERR_SAFETY);
- if (initial == 0) {
- *out = 0;
- return S2N_SUCCESS;
- }
- const uint64_t i = initial;
- const uint64_t a = alignment;
- const uint64_t result = a * (((i - 1) / a) + 1);
- POSIX_ENSURE(result <= UINT32_MAX, S2N_ERR_INTEGER_OVERFLOW);
- *out = (uint32_t) result;
- return S2N_SUCCESS;
- }
- int s2n_mul_overflow(uint32_t a, uint32_t b, uint32_t *out)
- {
- POSIX_ENSURE_REF(out);
- const uint64_t result = ((uint64_t) a) * ((uint64_t) b);
- POSIX_ENSURE(result <= UINT32_MAX, S2N_ERR_INTEGER_OVERFLOW);
- *out = (uint32_t) result;
- return S2N_SUCCESS;
- }
- int s2n_add_overflow(uint32_t a, uint32_t b, uint32_t *out)
- {
- POSIX_ENSURE_REF(out);
- uint64_t result = ((uint64_t) a) + ((uint64_t) b);
- POSIX_ENSURE(result <= UINT32_MAX, S2N_ERR_INTEGER_OVERFLOW);
- *out = (uint32_t) result;
- return S2N_SUCCESS;
- }
- int s2n_sub_overflow(uint32_t a, uint32_t b, uint32_t *out)
- {
- POSIX_ENSURE_REF(out);
- POSIX_ENSURE(a >= b, S2N_ERR_INTEGER_OVERFLOW);
- *out = a - b;
- return S2N_SUCCESS;
- }
|