| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 1 | /** | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 2 | *  Core bignum functions | 
|  | 3 | * | 
| Janos Follath | af3f39c | 2022-08-22 09:06:32 +0100 | [diff] [blame] | 4 | *  This interface should only be used by the legacy bignum module (bignum.h) | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 5 | *  and the modular bignum modules (bignum_mod.c, bignum_mod_raw.c). All other | 
| Janos Follath | af3f39c | 2022-08-22 09:06:32 +0100 | [diff] [blame] | 6 | *  modules should use the high-level modular bignum interface (bignum_mod.h) | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 7 | *  or the legacy bignum interface (bignum.h). | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 8 | * | 
| Gilles Peskine | 7aab2fb | 2022-09-27 13:19:13 +0200 | [diff] [blame] | 9 | * This module is about processing non-negative integers with a fixed upper | 
| Gilles Peskine | 01af3dd | 2022-10-04 16:23:29 +0200 | [diff] [blame] | 10 | * bound that's of the form 2^n-1 where n is a multiple of #biL. | 
|  | 11 | * These can be thought of integers written in base 2^#biL with a fixed | 
|  | 12 | * number of digits. Digits in this base are called *limbs*. | 
|  | 13 | * Many operations treat these numbers as the principal representation of | 
|  | 14 | * a number modulo 2^n or a smaller bound. | 
| Gilles Peskine | 7aab2fb | 2022-09-27 13:19:13 +0200 | [diff] [blame] | 15 | * | 
| Gilles Peskine | 2926484 | 2022-09-27 13:19:50 +0200 | [diff] [blame] | 16 | * The functions in this module obey the following conventions unless | 
|  | 17 | * explicitly indicated otherwise: | 
|  | 18 | * | 
|  | 19 | * - **Overflow**: some functions indicate overflow from the range | 
| Gilles Peskine | 01af3dd | 2022-10-04 16:23:29 +0200 | [diff] [blame] | 20 | *   [0, 2^n-1] by returning carry parameters, while others operate | 
| Gilles Peskine | 2926484 | 2022-09-27 13:19:50 +0200 | [diff] [blame] | 21 | *   modulo and so cannot overflow. This should be clear from the function | 
|  | 22 | *   documentation. | 
|  | 23 | * - **Bignum parameters**: Bignums are passed as pointers to an array of | 
|  | 24 | *   limbs. A limb has the type #mbedtls_mpi_uint. Unless otherwise specified: | 
|  | 25 | *     - Bignum parameters called \p A, \p B, ... are inputs, and are | 
|  | 26 | *       not modified by the function. | 
|  | 27 | *     - For operations modulo some number, the modulus is called \p N | 
|  | 28 | *       and is input-only. | 
|  | 29 | *     - Bignum parameters called \p X, \p Y are outputs or input-output. | 
|  | 30 | *       The initial content of output-only parameters is ignored. | 
|  | 31 | *     - Some functions use different names that reflect traditional | 
|  | 32 | *       naming of operands of certain operations (e.g. | 
|  | 33 | *       divisor/dividend/quotient/remainder). | 
|  | 34 | *     - \p T is a temporary storage area. The initial content of such | 
|  | 35 | *       parameter is ignored and the final content is unspecified. | 
|  | 36 | * - **Bignum sizes**: bignum sizes are always expressed in limbs. | 
|  | 37 | *   Most functions work on bignums of a given size and take a single | 
|  | 38 | *   \p limbs parameter that applies to all parameters that are limb arrays. | 
|  | 39 | *   All bignum sizes must be at least 1 and must be significantly less than | 
|  | 40 | *   #SIZE_MAX. The behavior if a size is 0 is undefined. The behavior if the | 
|  | 41 | *   total size of all parameters overflows #SIZE_MAX is undefined. | 
|  | 42 | * - **Parameter ordering**: for bignum parameters, outputs come before inputs. | 
|  | 43 | *   Temporaries come last. | 
|  | 44 | * - **Aliasing**: in general, output bignums may be aliased to one or more | 
|  | 45 | *   inputs. As an exception, parameters that are documented as a modulus value | 
| Gilles Peskine | dcd1717 | 2022-10-14 17:14:20 +0200 | [diff] [blame] | 46 | *   may not be aliased to an output. Outputs may not be aliased to one another. | 
|  | 47 | *   Temporaries may not be aliased to any other parameter. | 
| Gilles Peskine | 2926484 | 2022-09-27 13:19:50 +0200 | [diff] [blame] | 48 | * - **Overlap**: apart from aliasing of limb array pointers (where two | 
|  | 49 | *   arguments are equal pointers), overlap is not supported and may result | 
|  | 50 | *   in undefined behavior. | 
|  | 51 | * - **Error handling**: This is a low-level module. Functions generally do not | 
|  | 52 | *   try to protect against invalid arguments such as nonsensical sizes or | 
|  | 53 | *   null pointers. Note that some functions that operate on bignums of | 
|  | 54 | *   different sizes have constraints about their size, and violating those | 
|  | 55 | *   constraints may lead to buffer overflows. | 
|  | 56 | * - **Modular representatives**: functions that operate modulo \p N expect | 
|  | 57 | *   all modular inputs to be in the range [0, \p N - 1] and guarantee outputs | 
|  | 58 | *   in the range [0, \p N - 1]. If an input is out of range, outputs are | 
|  | 59 | *   fully unspecified, though bignum values out of range should not cause | 
|  | 60 | *   buffer overflows (beware that this is not extensively tested). | 
| Gilles Peskine | 7f887bd | 2022-09-27 13:12:30 +0200 | [diff] [blame] | 61 | */ | 
|  | 62 |  | 
|  | 63 | /* | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 64 | *  Copyright The Mbed TLS Contributors | 
| Dave Rodgman | 16799db | 2023-11-02 19:47:20 +0000 | [diff] [blame] | 65 | *  SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 66 | */ | 
|  | 67 |  | 
|  | 68 | #ifndef MBEDTLS_BIGNUM_CORE_H | 
|  | 69 | #define MBEDTLS_BIGNUM_CORE_H | 
|  | 70 |  | 
|  | 71 | #include "common.h" | 
|  | 72 |  | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 73 | #include "mbedtls/bignum.h" | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 74 |  | 
| Dave Rodgman | cd2e38b | 2023-05-17 13:31:55 +0100 | [diff] [blame] | 75 | #include "constant_time_internal.h" | 
|  | 76 |  | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 77 | #define ciL    (sizeof(mbedtls_mpi_uint))     /** chars in limb  */ | 
|  | 78 | #define biL    (ciL << 3)                     /** bits  in limb  */ | 
|  | 79 | #define biH    (ciL << 2)                     /** half limb size */ | 
| Tom Cosgrove | 5eefc3d | 2022-08-31 17:16:50 +0100 | [diff] [blame] | 80 |  | 
|  | 81 | /* | 
|  | 82 | * Convert between bits/chars and number of limbs | 
|  | 83 | * Divide first in order to avoid potential overflows | 
|  | 84 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 85 | #define BITS_TO_LIMBS(i)  ((i) / biL + ((i) % biL != 0)) | 
|  | 86 | #define CHARS_TO_LIMBS(i) ((i) / ciL + ((i) % ciL != 0)) | 
| Tom Cosgrove | 5eefc3d | 2022-08-31 17:16:50 +0100 | [diff] [blame] | 87 | /* Get a specific byte, without range checks. */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 88 | #define GET_BYTE(X, i)                                \ | 
|  | 89 | (((X)[(i) / ciL] >> (((i) % ciL) * 8)) & 0xff) | 
| Tom Cosgrove | 5eefc3d | 2022-08-31 17:16:50 +0100 | [diff] [blame] | 90 |  | 
| Janos Follath | 90b4271 | 2024-08-12 19:32:45 +0100 | [diff] [blame] | 91 | /* Constants to identify whether a value is public or secret. If a parameter is marked as secret by | 
|  | 92 | * this constant, the function must be constant time with respect to the parameter. | 
|  | 93 | * | 
|  | 94 | * This is only needed for functions with the _optionally_safe postfix. All other functions have | 
|  | 95 | * fixed behavior that can't be changed at runtime and are constant time with respect to their | 
|  | 96 | * parameters as prescribed by their documentation or by conventions in their module's documentation. | 
|  | 97 | * | 
|  | 98 | * Parameters should be named X_public where X is the name of the | 
|  | 99 | * corresponding input parameter. | 
|  | 100 | * | 
|  | 101 | * Implementation should always check using | 
|  | 102 | *  if (X_public == MBEDTLS_MPI_IS_PUBLIC) { | 
|  | 103 | *      // unsafe path | 
|  | 104 | *  } else { | 
|  | 105 | *      // safe path | 
|  | 106 | *  } | 
| Manuel Pégourié-Gonnard | 15fa9ce | 2024-09-03 10:10:18 +0200 | [diff] [blame] | 107 | * not the other way round, in order to prevent misuse. (That is, if a value | 
| Janos Follath | 2f8ad59 | 2024-08-22 17:13:25 +0100 | [diff] [blame] | 108 | * other than the two below is passed, default to the safe path.) | 
|  | 109 | * | 
|  | 110 | * The value of MBEDTLS_MPI_IS_PUBLIC is chosen in a way that is unlikely to happen by accident, but | 
|  | 111 | * which can be used as an immediate value in a Thumb2 comparison (for code size). */ | 
| Janos Follath | 0c292b2 | 2024-08-12 19:55:02 +0100 | [diff] [blame] | 112 | #define MBEDTLS_MPI_IS_PUBLIC  0x2a2a2a2a | 
| Janos Follath | 90b4271 | 2024-08-12 19:32:45 +0100 | [diff] [blame] | 113 | #define MBEDTLS_MPI_IS_SECRET  0 | 
| Janos Follath | 42f72b3 | 2024-08-22 08:25:33 +0100 | [diff] [blame] | 114 | #if defined(MBEDTLS_TEST_HOOKS) && !defined(MBEDTLS_THREADING_C) | 
|  | 115 | // Default value for testing that is neither MBEDTLS_MPI_IS_PUBLIC nor MBEDTLS_MPI_IS_SECRET | 
|  | 116 | #define MBEDTLS_MPI_IS_TEST  1 | 
|  | 117 | #endif | 
| Janos Follath | 90b4271 | 2024-08-12 19:32:45 +0100 | [diff] [blame] | 118 |  | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 119 | /** Count leading zero bits in a given integer. | 
|  | 120 | * | 
| Dave Rodgman | 0f16d56 | 2023-04-24 12:53:29 +0100 | [diff] [blame] | 121 | * \warning     The result is undefined if \p a == 0 | 
|  | 122 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 123 | * \param a     Integer to count leading zero bits. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 124 | * | 
| Dave Rodgman | 0f16d56 | 2023-04-24 12:53:29 +0100 | [diff] [blame] | 125 | * \return      The number of leading zero bits in \p a, if \p a != 0. | 
|  | 126 | *              If \p a == 0, the result is undefined. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 127 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 128 | size_t mbedtls_mpi_core_clz(mbedtls_mpi_uint a); | 
| Janos Follath | 4670f88 | 2022-07-21 18:25:42 +0100 | [diff] [blame] | 129 |  | 
| Janos Follath | a95f204 | 2022-08-19 12:09:17 +0100 | [diff] [blame] | 130 | /** Return the minimum number of bits required to represent the value held | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 131 | * in the MPI. | 
|  | 132 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 133 | * \note This function returns 0 if all the limbs of \p A are 0. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 134 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 135 | * \param[in] A     The address of the MPI. | 
| Janos Follath | c459641 | 2022-08-22 10:01:27 +0100 | [diff] [blame] | 136 | * \param A_limbs   The number of limbs of \p A. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 137 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 138 | * \return      The number of bits in \p A. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 139 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 140 | size_t mbedtls_mpi_core_bitlen(const mbedtls_mpi_uint *A, size_t A_limbs); | 
| Janos Follath | 4670f88 | 2022-07-21 18:25:42 +0100 | [diff] [blame] | 141 |  | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 142 | /** Convert a big-endian byte array aligned to the size of mbedtls_mpi_uint | 
|  | 143 | * into the storage form used by mbedtls_mpi. | 
|  | 144 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 145 | * \param[in,out] A     The address of the MPI. | 
| Janos Follath | c459641 | 2022-08-22 10:01:27 +0100 | [diff] [blame] | 146 | * \param A_limbs       The number of limbs of \p A. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 147 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 148 | void mbedtls_mpi_core_bigendian_to_host(mbedtls_mpi_uint *A, | 
|  | 149 | size_t A_limbs); | 
| Janos Follath | 4670f88 | 2022-07-21 18:25:42 +0100 | [diff] [blame] | 150 |  | 
| Gilles Peskine | 6f949ea | 2022-09-20 18:38:35 +0200 | [diff] [blame] | 151 | /** \brief         Compare a machine integer with an MPI. | 
|  | 152 | * | 
|  | 153 | *                 This function operates in constant time with respect | 
|  | 154 | *                 to the values of \p min and \p A. | 
|  | 155 | * | 
|  | 156 | * \param min      A machine integer. | 
|  | 157 | * \param[in] A    An MPI. | 
|  | 158 | * \param A_limbs  The number of limbs of \p A. | 
|  | 159 | *                 This must be at least 1. | 
|  | 160 | * | 
| Dave Rodgman | fd7fab4 | 2023-05-17 14:00:39 +0100 | [diff] [blame] | 161 | * \return         MBEDTLS_CT_TRUE if \p min is less than or equal to \p A, otherwise MBEDTLS_CT_FALSE. | 
| Gilles Peskine | 6f949ea | 2022-09-20 18:38:35 +0200 | [diff] [blame] | 162 | */ | 
| Dave Rodgman | fd7fab4 | 2023-05-17 14:00:39 +0100 | [diff] [blame] | 163 | mbedtls_ct_condition_t mbedtls_mpi_core_uint_le_mpi(mbedtls_mpi_uint min, | 
|  | 164 | const mbedtls_mpi_uint *A, | 
|  | 165 | size_t A_limbs); | 
| Gilles Peskine | 6f949ea | 2022-09-20 18:38:35 +0200 | [diff] [blame] | 166 |  | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 167 | /** | 
| Dave Rodgman | 7d4f019 | 2023-05-09 14:01:05 +0100 | [diff] [blame] | 168 | * \brief          Check if one unsigned MPI is less than another in constant | 
|  | 169 | *                 time. | 
|  | 170 | * | 
|  | 171 | * \param A        The left-hand MPI. This must point to an array of limbs | 
|  | 172 | *                 with the same allocated length as \p B. | 
|  | 173 | * \param B        The right-hand MPI. This must point to an array of limbs | 
|  | 174 | *                 with the same allocated length as \p A. | 
|  | 175 | * \param limbs    The number of limbs in \p A and \p B. | 
|  | 176 | *                 This must not be 0. | 
|  | 177 | * | 
| Dave Rodgman | 8ac9a1d | 2023-05-17 15:16:22 +0100 | [diff] [blame] | 178 | * \return         MBEDTLS_CT_TRUE  if \p A is less than \p B. | 
|  | 179 | *                 MBEDTLS_CT_FALSE if \p A is greater than or equal to \p B. | 
| Dave Rodgman | 7d4f019 | 2023-05-09 14:01:05 +0100 | [diff] [blame] | 180 | */ | 
| Dave Rodgman | 8ac9a1d | 2023-05-17 15:16:22 +0100 | [diff] [blame] | 181 | mbedtls_ct_condition_t mbedtls_mpi_core_lt_ct(const mbedtls_mpi_uint *A, | 
|  | 182 | const mbedtls_mpi_uint *B, | 
|  | 183 | size_t limbs); | 
|  | 184 |  | 
| Dave Rodgman | 7d4f019 | 2023-05-09 14:01:05 +0100 | [diff] [blame] | 185 | /** | 
| Gabor Mezei | 4086de6 | 2022-10-14 16:29:42 +0200 | [diff] [blame] | 186 | * \brief   Perform a safe conditional copy of an MPI which doesn't reveal | 
|  | 187 | *          whether assignment was done or not. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 188 | * | 
| Gabor Mezei | 4086de6 | 2022-10-14 16:29:42 +0200 | [diff] [blame] | 189 | * \param[out] X        The address of the destination MPI. | 
|  | 190 | *                      This must be initialized. Must have enough limbs to | 
|  | 191 | *                      store the full value of \p A. | 
|  | 192 | * \param[in]  A        The address of the source MPI. This must be initialized. | 
| Gabor Mezei | 1c628d5 | 2022-09-27 12:13:51 +0200 | [diff] [blame] | 193 | * \param      limbs    The number of limbs of \p A. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 194 | * \param      assign   The condition deciding whether to perform the | 
| Dave Rodgman | fb1b851 | 2023-07-31 12:27:05 +0100 | [diff] [blame] | 195 | *                      assignment or not. Callers will need to use | 
|  | 196 | *                      the constant time interface (e.g. `mbedtls_ct_bool()`) | 
|  | 197 | *                      to construct this argument. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 198 | * | 
|  | 199 | * \note           This function avoids leaking any information about whether | 
|  | 200 | *                 the assignment was done or not. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 201 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 202 | void mbedtls_mpi_core_cond_assign(mbedtls_mpi_uint *X, | 
|  | 203 | const mbedtls_mpi_uint *A, | 
|  | 204 | size_t limbs, | 
| Dave Rodgman | cd2e38b | 2023-05-17 13:31:55 +0100 | [diff] [blame] | 205 | mbedtls_ct_condition_t assign); | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 206 |  | 
|  | 207 | /** | 
| Gabor Mezei | 4086de6 | 2022-10-14 16:29:42 +0200 | [diff] [blame] | 208 | * \brief   Perform a safe conditional swap of two MPIs which doesn't reveal | 
|  | 209 | *          whether the swap was done or not. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 210 | * | 
| Gabor Mezei | 86dfe38 | 2022-09-30 14:03:04 +0200 | [diff] [blame] | 211 | * \param[in,out] X         The address of the first MPI. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 212 | *                          This must be initialized. | 
| Gabor Mezei | 86dfe38 | 2022-09-30 14:03:04 +0200 | [diff] [blame] | 213 | * \param[in,out] Y         The address of the second MPI. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 214 | *                          This must be initialized. | 
| Gabor Mezei | e5b8585 | 2022-09-30 13:54:02 +0200 | [diff] [blame] | 215 | * \param         limbs     The number of limbs of \p X and \p Y. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 216 | * \param         swap      The condition deciding whether to perform | 
| Dave Rodgman | cd2e38b | 2023-05-17 13:31:55 +0100 | [diff] [blame] | 217 | *                          the swap or not. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 218 | * | 
|  | 219 | * \note           This function avoids leaking any information about whether | 
|  | 220 | *                 the swap was done or not. | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 221 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 222 | void mbedtls_mpi_core_cond_swap(mbedtls_mpi_uint *X, | 
|  | 223 | mbedtls_mpi_uint *Y, | 
|  | 224 | size_t limbs, | 
| Dave Rodgman | cd2e38b | 2023-05-17 13:31:55 +0100 | [diff] [blame] | 225 | mbedtls_ct_condition_t swap); | 
| Gabor Mezei | e1d31c4 | 2022-09-12 16:25:24 +0200 | [diff] [blame] | 226 |  | 
| Janos Follath | af3f39c | 2022-08-22 09:06:32 +0100 | [diff] [blame] | 227 | /** Import X from unsigned binary data, little-endian. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 228 | * | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 229 | * The MPI needs to have enough limbs to store the full value (including any | 
|  | 230 | * most significant zero bytes in the input). | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 231 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 232 | * \param[out] X         The address of the MPI. | 
|  | 233 | * \param X_limbs        The number of limbs of \p X. | 
|  | 234 | * \param[in] input      The input buffer to import from. | 
|  | 235 | * \param input_length   The length bytes of \p input. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 236 | * | 
|  | 237 | * \return       \c 0 if successful. | 
|  | 238 | * \return       #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if \p X isn't | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 239 | *               large enough to hold the value in \p input. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 240 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 241 | int mbedtls_mpi_core_read_le(mbedtls_mpi_uint *X, | 
|  | 242 | size_t X_limbs, | 
|  | 243 | const unsigned char *input, | 
|  | 244 | size_t input_length); | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 245 |  | 
| Janos Follath | af3f39c | 2022-08-22 09:06:32 +0100 | [diff] [blame] | 246 | /** Import X from unsigned binary data, big-endian. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 247 | * | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 248 | * The MPI needs to have enough limbs to store the full value (including any | 
|  | 249 | * most significant zero bytes in the input). | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 250 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 251 | * \param[out] X        The address of the MPI. | 
| Tom Cosgrove | 8a1f784 | 2023-02-01 08:43:54 +0000 | [diff] [blame] | 252 | *                      May only be #NULL if \p X_limbs is 0 and \p input_length | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 253 | *                      is 0. | 
|  | 254 | * \param X_limbs       The number of limbs of \p X. | 
|  | 255 | * \param[in] input     The input buffer to import from. | 
|  | 256 | *                      May only be #NULL if \p input_length is 0. | 
|  | 257 | * \param input_length  The length in bytes of \p input. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 258 | * | 
|  | 259 | * \return       \c 0 if successful. | 
|  | 260 | * \return       #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if \p X isn't | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 261 | *               large enough to hold the value in \p input. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 262 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 263 | int mbedtls_mpi_core_read_be(mbedtls_mpi_uint *X, | 
|  | 264 | size_t X_limbs, | 
|  | 265 | const unsigned char *input, | 
|  | 266 | size_t input_length); | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 267 |  | 
| Janos Follath | af3f39c | 2022-08-22 09:06:32 +0100 | [diff] [blame] | 268 | /** Export A into unsigned binary data, little-endian. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 269 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 270 | * \note If \p output is shorter than \p A the export is still successful if the | 
|  | 271 | *       value held in \p A fits in the buffer (that is, if enough of the most | 
|  | 272 | *       significant bytes of \p A are 0). | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 273 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 274 | * \param[in] A         The address of the MPI. | 
|  | 275 | * \param A_limbs       The number of limbs of \p A. | 
|  | 276 | * \param[out] output   The output buffer to export to. | 
|  | 277 | * \param output_length The length in bytes of \p output. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 278 | * | 
|  | 279 | * \return       \c 0 if successful. | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 280 | * \return       #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if \p output isn't | 
|  | 281 | *               large enough to hold the value of \p A. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 282 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 283 | int mbedtls_mpi_core_write_le(const mbedtls_mpi_uint *A, | 
|  | 284 | size_t A_limbs, | 
|  | 285 | unsigned char *output, | 
|  | 286 | size_t output_length); | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 287 |  | 
| Janos Follath | af3f39c | 2022-08-22 09:06:32 +0100 | [diff] [blame] | 288 | /** Export A into unsigned binary data, big-endian. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 289 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 290 | * \note If \p output is shorter than \p A the export is still successful if the | 
|  | 291 | *       value held in \p A fits in the buffer (that is, if enough of the most | 
|  | 292 | *       significant bytes of \p A are 0). | 
| Janos Follath | 6318468 | 2022-08-11 17:42:59 +0100 | [diff] [blame] | 293 | * | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 294 | * \param[in] A         The address of the MPI. | 
|  | 295 | * \param A_limbs       The number of limbs of \p A. | 
|  | 296 | * \param[out] output   The output buffer to export to. | 
|  | 297 | * \param output_length The length in bytes of \p output. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 298 | * | 
|  | 299 | * \return       \c 0 if successful. | 
| Janos Follath | b7a88ec | 2022-08-19 12:24:40 +0100 | [diff] [blame] | 300 | * \return       #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if \p output isn't | 
|  | 301 | *               large enough to hold the value of \p A. | 
| Gabor Mezei | 37b0636 | 2022-08-02 17:22:18 +0200 | [diff] [blame] | 302 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 303 | int mbedtls_mpi_core_write_be(const mbedtls_mpi_uint *A, | 
|  | 304 | size_t A_limbs, | 
|  | 305 | unsigned char *output, | 
|  | 306 | size_t output_length); | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 307 |  | 
| Minos Galanakis | ec09e25 | 2023-04-20 14:22:16 +0100 | [diff] [blame] | 308 | /** \brief              Shift an MPI in-place right by a number of bits. | 
| Gilles Peskine | 6641420 | 2022-09-21 15:36:16 +0200 | [diff] [blame] | 309 | * | 
|  | 310 | *                      Shifting by more bits than there are bit positions | 
|  | 311 | *                      in \p X is valid and results in setting \p X to 0. | 
|  | 312 | * | 
|  | 313 | *                      This function's execution time depends on the value | 
|  | 314 | *                      of \p count (and of course \p limbs). | 
|  | 315 | * | 
|  | 316 | * \param[in,out] X     The number to shift. | 
|  | 317 | * \param limbs         The number of limbs of \p X. This must be at least 1. | 
|  | 318 | * \param count         The number of bits to shift by. | 
|  | 319 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 320 | void mbedtls_mpi_core_shift_r(mbedtls_mpi_uint *X, size_t limbs, | 
|  | 321 | size_t count); | 
| Gilles Peskine | 6641420 | 2022-09-21 15:36:16 +0200 | [diff] [blame] | 322 |  | 
| Tom Cosgrove | 90c426b | 2022-08-23 16:15:19 +0100 | [diff] [blame] | 323 | /** | 
| Minos Galanakis | ec09e25 | 2023-04-20 14:22:16 +0100 | [diff] [blame] | 324 | * \brief               Shift an MPI in-place left by a number of bits. | 
| Minos Galanakis | ad808dd | 2023-04-20 12:18:41 +0100 | [diff] [blame] | 325 | * | 
| Minos Galanakis | ec09e25 | 2023-04-20 14:22:16 +0100 | [diff] [blame] | 326 | *                      Shifting by more bits than there are bit positions | 
| Minos Galanakis | b894403 | 2023-04-28 14:09:44 +0100 | [diff] [blame] | 327 | *                      in \p X will produce an unspecified result. | 
| Minos Galanakis | ad808dd | 2023-04-20 12:18:41 +0100 | [diff] [blame] | 328 | * | 
| Minos Galanakis | ec09e25 | 2023-04-20 14:22:16 +0100 | [diff] [blame] | 329 | *                      This function's execution time depends on the value | 
|  | 330 | *                      of \p count (and of course \p limbs). | 
|  | 331 | * \param[in,out] X     The number to shift. | 
|  | 332 | * \param limbs         The number of limbs of \p X. This must be at least 1. | 
|  | 333 | * \param count         The number of bits to shift by. | 
| Minos Galanakis | ad808dd | 2023-04-20 12:18:41 +0100 | [diff] [blame] | 334 | */ | 
| Minos Galanakis | ec09e25 | 2023-04-20 14:22:16 +0100 | [diff] [blame] | 335 | void mbedtls_mpi_core_shift_l(mbedtls_mpi_uint *X, size_t limbs, | 
|  | 336 | size_t count); | 
| Minos Galanakis | ad808dd | 2023-04-20 12:18:41 +0100 | [diff] [blame] | 337 |  | 
|  | 338 | /** | 
| Tom Cosgrove | af7d44b | 2022-08-24 14:05:26 +0100 | [diff] [blame] | 339 | * \brief Add two fixed-size large unsigned integers, returning the carry. | 
| Hanno Becker | c988713 | 2022-08-24 12:54:36 +0100 | [diff] [blame] | 340 | * | 
| Tom Cosgrove | af7d44b | 2022-08-24 14:05:26 +0100 | [diff] [blame] | 341 | * Calculates `A + B` where `A` and `B` have the same size. | 
|  | 342 | * | 
| Tom Cosgrove | 82f1310 | 2022-10-25 12:46:03 +0100 | [diff] [blame] | 343 | * This function operates modulo `2^(biL*limbs)` and returns the carry | 
| Hanno Becker | c988713 | 2022-08-24 12:54:36 +0100 | [diff] [blame] | 344 | * (1 if there was a wraparound, and 0 otherwise). | 
|  | 345 | * | 
| Tom Cosgrove | af7d44b | 2022-08-24 14:05:26 +0100 | [diff] [blame] | 346 | * \p X may be aliased to \p A or \p B. | 
| Hanno Becker | c988713 | 2022-08-24 12:54:36 +0100 | [diff] [blame] | 347 | * | 
| Tom Cosgrove | af7d44b | 2022-08-24 14:05:26 +0100 | [diff] [blame] | 348 | * \param[out] X    The result of the addition. | 
|  | 349 | * \param[in] A     Little-endian presentation of the left operand. | 
|  | 350 | * \param[in] B     Little-endian presentation of the right operand. | 
|  | 351 | * \param limbs     Number of limbs of \p X, \p A and \p B. | 
| Hanno Becker | c988713 | 2022-08-24 12:54:36 +0100 | [diff] [blame] | 352 | * | 
| Tom Cosgrove | af7d44b | 2022-08-24 14:05:26 +0100 | [diff] [blame] | 353 | * \return          1 if `A + B >= 2^(biL*limbs)`, 0 otherwise. | 
| Hanno Becker | c988713 | 2022-08-24 12:54:36 +0100 | [diff] [blame] | 354 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 355 | mbedtls_mpi_uint mbedtls_mpi_core_add(mbedtls_mpi_uint *X, | 
|  | 356 | const mbedtls_mpi_uint *A, | 
|  | 357 | const mbedtls_mpi_uint *B, | 
|  | 358 | size_t limbs); | 
| Hanno Becker | c988713 | 2022-08-24 12:54:36 +0100 | [diff] [blame] | 359 |  | 
| Tom Cosgrove | 90c426b | 2022-08-23 16:15:19 +0100 | [diff] [blame] | 360 | /** | 
| Tom Cosgrove | 5c0e810 | 2022-09-15 15:46:10 +0100 | [diff] [blame] | 361 | * \brief Conditional addition of two fixed-size large unsigned integers, | 
| Tom Cosgrove | d932de8 | 2022-08-25 16:43:43 +0100 | [diff] [blame] | 362 | *        returning the carry. | 
| Hanno Becker | 71f4b0d | 2022-08-23 12:09:35 +0100 | [diff] [blame] | 363 | * | 
|  | 364 | * Functionally equivalent to | 
|  | 365 | * | 
|  | 366 | * ``` | 
|  | 367 | * if( cond ) | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 368 | *    X += A; | 
| Hanno Becker | 71f4b0d | 2022-08-23 12:09:35 +0100 | [diff] [blame] | 369 | * return carry; | 
|  | 370 | * ``` | 
|  | 371 | * | 
| Tom Cosgrove | 4782823 | 2022-09-20 13:51:50 +0100 | [diff] [blame] | 372 | * This function operates modulo `2^(biL*limbs)`. | 
|  | 373 | * | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 374 | * \param[in,out] X  The pointer to the (little-endian) array | 
| Tom Cosgrove | 7259463 | 2022-08-24 11:51:58 +0100 | [diff] [blame] | 375 | *                   representing the bignum to accumulate onto. | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 376 | * \param[in] A      The pointer to the (little-endian) array | 
| Tom Cosgrove | 7259463 | 2022-08-24 11:51:58 +0100 | [diff] [blame] | 377 | *                   representing the bignum to conditionally add | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 378 | *                   to \p X. This may be aliased to \p X but may not | 
| Tom Cosgrove | ed43c6c | 2022-08-31 11:35:00 +0100 | [diff] [blame] | 379 | *                   overlap otherwise. | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 380 | * \param limbs      Number of limbs of \p X and \p A. | 
| Tom Cosgrove | 7259463 | 2022-08-24 11:51:58 +0100 | [diff] [blame] | 381 | * \param cond       Condition bit dictating whether addition should | 
|  | 382 | *                   happen or not. This must be \c 0 or \c 1. | 
| Hanno Becker | 71f4b0d | 2022-08-23 12:09:35 +0100 | [diff] [blame] | 383 | * | 
| Tom Cosgrove | ecbb124 | 2022-08-25 10:13:44 +0100 | [diff] [blame] | 384 | * \warning          If \p cond is neither 0 nor 1, the result of this function | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 385 | *                   is unspecified, and the resulting value in \p X might be | 
|  | 386 | *                   neither its original value nor \p X + \p A. | 
| Tom Cosgrove | 7259463 | 2022-08-24 11:51:58 +0100 | [diff] [blame] | 387 | * | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 388 | * \return           1 if `X + cond * A >= 2^(biL*limbs)`, 0 otherwise. | 
| Hanno Becker | 71f4b0d | 2022-08-23 12:09:35 +0100 | [diff] [blame] | 389 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 390 | mbedtls_mpi_uint mbedtls_mpi_core_add_if(mbedtls_mpi_uint *X, | 
|  | 391 | const mbedtls_mpi_uint *A, | 
|  | 392 | size_t limbs, | 
|  | 393 | unsigned cond); | 
| Hanno Becker | 71f4b0d | 2022-08-23 12:09:35 +0100 | [diff] [blame] | 394 |  | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 395 | /** | 
| Tom Cosgrove | 5c0e810 | 2022-09-15 15:46:10 +0100 | [diff] [blame] | 396 | * \brief Subtract two fixed-size large unsigned integers, returning the borrow. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 397 | * | 
| Tom Cosgrove | 5dd97e6 | 2022-08-30 14:31:49 +0100 | [diff] [blame] | 398 | * Calculate `A - B` where \p A and \p B have the same size. | 
| Tom Cosgrove | 630110a | 2022-08-31 17:09:29 +0100 | [diff] [blame] | 399 | * This function operates modulo `2^(biL*limbs)` and returns the carry | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 400 | * (1 if there was a wraparound, i.e. if `A < B`, and 0 otherwise). | 
|  | 401 | * | 
| Tom Cosgrove | 5dd97e6 | 2022-08-30 14:31:49 +0100 | [diff] [blame] | 402 | * \p X may be aliased to \p A or \p B, or even both, but may not overlap | 
|  | 403 | * either otherwise. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 404 | * | 
|  | 405 | * \param[out] X    The result of the subtraction. | 
|  | 406 | * \param[in] A     Little-endian presentation of left operand. | 
|  | 407 | * \param[in] B     Little-endian presentation of right operand. | 
|  | 408 | * \param limbs     Number of limbs of \p X, \p A and \p B. | 
|  | 409 | * | 
|  | 410 | * \return          1 if `A < B`. | 
|  | 411 | *                  0 if `A >= B`. | 
|  | 412 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 413 | mbedtls_mpi_uint mbedtls_mpi_core_sub(mbedtls_mpi_uint *X, | 
|  | 414 | const mbedtls_mpi_uint *A, | 
|  | 415 | const mbedtls_mpi_uint *B, | 
|  | 416 | size_t limbs); | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 417 |  | 
|  | 418 | /** | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 419 | * \brief Perform a fixed-size multiply accumulate operation: X += b * A | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 420 | * | 
| Tom Cosgrove | b0b77e1 | 2022-09-20 13:33:40 +0100 | [diff] [blame] | 421 | * \p X may be aliased to \p A (when \p X_limbs == \p A_limbs), but may not | 
|  | 422 | * otherwise overlap. | 
|  | 423 | * | 
| Tom Cosgrove | 4782823 | 2022-09-20 13:51:50 +0100 | [diff] [blame] | 424 | * This function operates modulo `2^(biL*X_limbs)`. | 
|  | 425 | * | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 426 | * \param[in,out] X  The pointer to the (little-endian) array | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 427 | *                   representing the bignum to accumulate onto. | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 428 | * \param X_limbs    The number of limbs of \p X. This must be | 
|  | 429 | *                   at least \p A_limbs. | 
|  | 430 | * \param[in] A      The pointer to the (little-endian) array | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 431 | *                   representing the bignum to multiply with. | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 432 | *                   This may be aliased to \p X but may not overlap | 
| Tom Cosgrove | ed43c6c | 2022-08-31 11:35:00 +0100 | [diff] [blame] | 433 | *                   otherwise. | 
| Tom Cosgrove | 3bd7bc3 | 2022-09-15 15:55:07 +0100 | [diff] [blame] | 434 | * \param A_limbs    The number of limbs of \p A. | 
|  | 435 | * \param b          X scalar to multiply with. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 436 | * | 
|  | 437 | * \return           The carry at the end of the operation. | 
|  | 438 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 439 | mbedtls_mpi_uint mbedtls_mpi_core_mla(mbedtls_mpi_uint *X, size_t X_limbs, | 
|  | 440 | const mbedtls_mpi_uint *A, size_t A_limbs, | 
|  | 441 | mbedtls_mpi_uint b); | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 442 |  | 
| Hanno Becker | 4ae890b | 2022-08-24 16:17:52 +0100 | [diff] [blame] | 443 | /** | 
|  | 444 | * \brief Perform a known-size multiplication | 
|  | 445 | * | 
| Gabor Mezei | d626051 | 2023-04-03 17:32:55 +0200 | [diff] [blame] | 446 | * \p X may not be aliased to any of the inputs for this function. | 
| Gabor Mezei | 6f182c3 | 2023-03-31 16:09:28 +0200 | [diff] [blame] | 447 | * \p A may be aliased to \p B. | 
|  | 448 | * | 
| Tom Cosgrove | 6af26f3 | 2022-08-24 16:40:55 +0100 | [diff] [blame] | 449 | * \param[out] X     The pointer to the (little-endian) array to receive | 
|  | 450 | *                   the product of \p A_limbs and \p B_limbs. | 
|  | 451 | *                   This must be of length \p A_limbs + \p B_limbs. | 
|  | 452 | * \param[in] A      The pointer to the (little-endian) array | 
|  | 453 | *                   representing the first factor. | 
|  | 454 | * \param A_limbs    The number of limbs in \p A. | 
|  | 455 | * \param[in] B      The pointer to the (little-endian) array | 
|  | 456 | *                   representing the second factor. | 
|  | 457 | * \param B_limbs    The number of limbs in \p B. | 
| Hanno Becker | 4ae890b | 2022-08-24 16:17:52 +0100 | [diff] [blame] | 458 | */ | 
| Tom Cosgrove | 6af26f3 | 2022-08-24 16:40:55 +0100 | [diff] [blame] | 459 | void mbedtls_mpi_core_mul(mbedtls_mpi_uint *X, | 
|  | 460 | const mbedtls_mpi_uint *A, size_t A_limbs, | 
|  | 461 | const mbedtls_mpi_uint *B, size_t B_limbs); | 
| Hanno Becker | 4ae890b | 2022-08-24 16:17:52 +0100 | [diff] [blame] | 462 |  | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 463 | /** | 
|  | 464 | * \brief Calculate initialisation value for fast Montgomery modular | 
|  | 465 | *        multiplication | 
|  | 466 | * | 
|  | 467 | * \param[in] N  Little-endian presentation of the modulus. This must have | 
|  | 468 | *               at least one limb. | 
|  | 469 | * | 
|  | 470 | * \return       The initialisation value for fast Montgomery modular multiplication | 
|  | 471 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 472 | mbedtls_mpi_uint mbedtls_mpi_core_montmul_init(const mbedtls_mpi_uint *N); | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 473 |  | 
|  | 474 | /** | 
| Tom Cosgrove | 4386ead | 2022-09-29 14:40:21 +0100 | [diff] [blame] | 475 | * \brief Montgomery multiplication: X = A * B * R^-1 mod N (HAC 14.36) | 
|  | 476 | * | 
|  | 477 | * \p A and \p B must be in canonical form. That is, < \p N. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 478 | * | 
| Tom Cosgrove | ea45c1d | 2022-09-20 13:17:51 +0100 | [diff] [blame] | 479 | * \p X may be aliased to \p A or \p N, or even \p B (if \p AN_limbs == | 
|  | 480 | * \p B_limbs) but may not overlap any parameters otherwise. | 
|  | 481 | * | 
| Tom Cosgrove | 4386ead | 2022-09-29 14:40:21 +0100 | [diff] [blame] | 482 | * \p A and \p B may alias each other, if \p AN_limbs == \p B_limbs. They may | 
|  | 483 | * not alias \p N (since they must be in canonical form, they cannot == \p N). | 
| Tom Cosgrove | ea45c1d | 2022-09-20 13:17:51 +0100 | [diff] [blame] | 484 | * | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 485 | * \param[out]    X         The destination MPI, as a little-endian array of | 
|  | 486 | *                          length \p AN_limbs. | 
|  | 487 | *                          On successful completion, X contains the result of | 
| Tom Cosgrove | 630110a | 2022-08-31 17:09:29 +0100 | [diff] [blame] | 488 | *                          the multiplication `A * B * R^-1` mod N where | 
|  | 489 | *                          `R = 2^(biL*AN_limbs)`. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 490 | * \param[in]     A         Little-endian presentation of first operand. | 
| Tom Cosgrove | 5dd97e6 | 2022-08-30 14:31:49 +0100 | [diff] [blame] | 491 | *                          Must have the same number of limbs as \p N. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 492 | * \param[in]     B         Little-endian presentation of second operand. | 
|  | 493 | * \param[in]     B_limbs   The number of limbs in \p B. | 
| Tom Cosgrove | 5dd97e6 | 2022-08-30 14:31:49 +0100 | [diff] [blame] | 494 | *                          Must be <= \p AN_limbs. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 495 | * \param[in]     N         Little-endian presentation of the modulus. | 
| Tom Cosgrove | 5dd97e6 | 2022-08-30 14:31:49 +0100 | [diff] [blame] | 496 | *                          This must be odd, and have exactly the same number | 
|  | 497 | *                          of limbs as \p A. | 
| Tom Cosgrove | 6da3a3b | 2022-09-29 17:20:18 +0100 | [diff] [blame] | 498 | *                          It may alias \p X, but must not alias or otherwise | 
|  | 499 | *                          overlap any of the other parameters. | 
| Tom Cosgrove | 5dd97e6 | 2022-08-30 14:31:49 +0100 | [diff] [blame] | 500 | * \param[in]     AN_limbs  The number of limbs in \p X, \p A and \p N. | 
| Tom Cosgrove | 630110a | 2022-08-31 17:09:29 +0100 | [diff] [blame] | 501 | * \param         mm        The Montgomery constant for \p N: -N^-1 mod 2^biL. | 
| Tom Cosgrove | b7438d1 | 2022-09-15 15:05:59 +0100 | [diff] [blame] | 502 | *                          This can be calculated by `mbedtls_mpi_core_montmul_init()`. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 503 | * \param[in,out] T         Temporary storage of size at least 2*AN_limbs+1 limbs. | 
|  | 504 | *                          Its initial content is unused and | 
|  | 505 | *                          its final content is indeterminate. | 
| Tom Cosgrove | 4386ead | 2022-09-29 14:40:21 +0100 | [diff] [blame] | 506 | *                          It must not alias or otherwise overlap any of the | 
|  | 507 | *                          other parameters. | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 508 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 509 | void mbedtls_mpi_core_montmul(mbedtls_mpi_uint *X, | 
|  | 510 | const mbedtls_mpi_uint *A, | 
|  | 511 | const mbedtls_mpi_uint *B, size_t B_limbs, | 
|  | 512 | const mbedtls_mpi_uint *N, size_t AN_limbs, | 
|  | 513 | mbedtls_mpi_uint mm, mbedtls_mpi_uint *T); | 
| Tom Cosgrove | b496486 | 2022-08-30 11:57:22 +0100 | [diff] [blame] | 514 |  | 
| Hanno Becker | ec440f2 | 2022-08-11 17:29:32 +0100 | [diff] [blame] | 515 | /** | 
| Minos Galanakis | b85506e | 2022-10-20 09:51:53 +0100 | [diff] [blame] | 516 | * \brief Calculate the square of the Montgomery constant. (Needed | 
|  | 517 | *        for conversion and operations in Montgomery form.) | 
| Hanno Becker | ec440f2 | 2022-08-11 17:29:32 +0100 | [diff] [blame] | 518 | * | 
|  | 519 | * \param[out] X  A pointer to the result of the calculation of | 
| Minos Galanakis | b85506e | 2022-10-20 09:51:53 +0100 | [diff] [blame] | 520 | *                the square of the Montgomery constant: | 
|  | 521 | *                2^{2*n*biL} mod N. | 
| Hanno Becker | ec440f2 | 2022-08-11 17:29:32 +0100 | [diff] [blame] | 522 | * \param[in]  N  Little-endian presentation of the modulus, which must be odd. | 
|  | 523 | * | 
|  | 524 | * \return        0 if successful. | 
|  | 525 | * \return        #MBEDTLS_ERR_MPI_ALLOC_FAILED if there is not enough space | 
|  | 526 | *                to store the value of Montgomery constant squared. | 
|  | 527 | * \return        #MBEDTLS_ERR_MPI_DIVISION_BY_ZERO if \p N modulus is zero. | 
|  | 528 | * \return        #MBEDTLS_ERR_MPI_NEGATIVE_VALUE if \p N modulus is negative. | 
| Hanno Becker | ec440f2 | 2022-08-11 17:29:32 +0100 | [diff] [blame] | 529 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 530 | int mbedtls_mpi_core_get_mont_r2_unsafe(mbedtls_mpi *X, | 
|  | 531 | const mbedtls_mpi *N); | 
| Hanno Becker | ec440f2 | 2022-08-11 17:29:32 +0100 | [diff] [blame] | 532 |  | 
| Janos Follath | 59cbd1b | 2022-10-28 18:13:43 +0100 | [diff] [blame] | 533 | #if defined(MBEDTLS_TEST_HOOKS) | 
| Janos Follath | e50f2f1 | 2022-10-26 15:14:33 +0100 | [diff] [blame] | 534 | /** | 
| Janos Follath | 8904a2d | 2022-10-31 15:32:28 +0000 | [diff] [blame] | 535 | * Copy an MPI from a table without leaking the index. | 
| Janos Follath | e50f2f1 | 2022-10-26 15:14:33 +0100 | [diff] [blame] | 536 | * | 
|  | 537 | * \param dest              The destination buffer. This must point to a writable | 
|  | 538 | *                          buffer of at least \p limbs limbs. | 
|  | 539 | * \param table             The address of the table. This must point to a readable | 
| Janos Follath | 8904a2d | 2022-10-31 15:32:28 +0000 | [diff] [blame] | 540 | *                          array of \p count elements of \p limbs limbs each. | 
|  | 541 | * \param limbs             The number of limbs in each table entry. | 
|  | 542 | * \param count             The number of entries in \p table. | 
|  | 543 | * \param index             The (secret) table index to look up. This must be in the | 
|  | 544 | *                          range `0 .. count-1`. | 
| Janos Follath | e50f2f1 | 2022-10-26 15:14:33 +0100 | [diff] [blame] | 545 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 546 | void mbedtls_mpi_core_ct_uint_table_lookup(mbedtls_mpi_uint *dest, | 
|  | 547 | const mbedtls_mpi_uint *table, | 
|  | 548 | size_t limbs, | 
|  | 549 | size_t count, | 
|  | 550 | size_t index); | 
| Janos Follath | 59cbd1b | 2022-10-28 18:13:43 +0100 | [diff] [blame] | 551 | #endif /* MBEDTLS_TEST_HOOKS */ | 
| Janos Follath | e50f2f1 | 2022-10-26 15:14:33 +0100 | [diff] [blame] | 552 |  | 
| Gilles Peskine | 909e03c | 2022-10-18 18:14:33 +0200 | [diff] [blame] | 553 | /** | 
|  | 554 | * \brief          Fill an integer with a number of random bytes. | 
|  | 555 | * | 
|  | 556 | * \param X        The destination MPI. | 
|  | 557 | * \param X_limbs  The number of limbs of \p X. | 
|  | 558 | * \param bytes    The number of random bytes to generate. | 
|  | 559 | * \param f_rng    The RNG function to use. This must not be \c NULL. | 
|  | 560 | * \param p_rng    The RNG parameter to be passed to \p f_rng. This may be | 
|  | 561 | *                 \c NULL if \p f_rng doesn't need a context argument. | 
|  | 562 | * | 
|  | 563 | * \return         \c 0 if successful. | 
|  | 564 | * \return         #MBEDTLS_ERR_MPI_BAD_INPUT_DATA if \p X does not have | 
|  | 565 | *                 enough room for \p bytes bytes. | 
|  | 566 | * \return         A negative error code on RNG failure. | 
|  | 567 | * | 
|  | 568 | * \note           The bytes obtained from the RNG are interpreted | 
|  | 569 | *                 as a big-endian representation of an MPI; this can | 
|  | 570 | *                 be relevant in applications like deterministic ECDSA. | 
|  | 571 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 572 | int mbedtls_mpi_core_fill_random(mbedtls_mpi_uint *X, size_t X_limbs, | 
|  | 573 | size_t bytes, | 
|  | 574 | int (*f_rng)(void *, unsigned char *, size_t), | 
|  | 575 | void *p_rng); | 
| Gilles Peskine | 909e03c | 2022-10-18 18:14:33 +0200 | [diff] [blame] | 576 |  | 
| Gilles Peskine | 4a8c5cd | 2022-10-18 18:15:01 +0200 | [diff] [blame] | 577 | /** Generate a random number uniformly in a range. | 
|  | 578 | * | 
|  | 579 | * This function generates a random number between \p min inclusive and | 
|  | 580 | * \p N exclusive. | 
|  | 581 | * | 
|  | 582 | * The procedure complies with RFC 6979 §3.3 (deterministic ECDSA) | 
|  | 583 | * when the RNG is a suitably parametrized instance of HMAC_DRBG | 
|  | 584 | * and \p min is \c 1. | 
|  | 585 | * | 
|  | 586 | * \note           There are `N - min` possible outputs. The lower bound | 
|  | 587 | *                 \p min can be reached, but the upper bound \p N cannot. | 
|  | 588 | * | 
|  | 589 | * \param X        The destination MPI, with \p limbs limbs. | 
|  | 590 | *                 It must not be aliased with \p N or otherwise overlap it. | 
|  | 591 | * \param min      The minimum value to return. | 
|  | 592 | * \param N        The upper bound of the range, exclusive, with \p limbs limbs. | 
|  | 593 | *                 In other words, this is one plus the maximum value to return. | 
|  | 594 | *                 \p N must be strictly larger than \p min. | 
|  | 595 | * \param limbs    The number of limbs of \p N and \p X. | 
|  | 596 | *                 This must not be 0. | 
|  | 597 | * \param f_rng    The RNG function to use. This must not be \c NULL. | 
|  | 598 | * \param p_rng    The RNG parameter to be passed to \p f_rng. | 
|  | 599 | * | 
|  | 600 | * \return         \c 0 if successful. | 
|  | 601 | * \return         #MBEDTLS_ERR_MPI_NOT_ACCEPTABLE if the implementation was | 
|  | 602 | *                 unable to find a suitable value within a limited number | 
|  | 603 | *                 of attempts. This has a negligible probability if \p N | 
|  | 604 | *                 is significantly larger than \p min, which is the case | 
|  | 605 | *                 for all usual cryptographic applications. | 
|  | 606 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 607 | int mbedtls_mpi_core_random(mbedtls_mpi_uint *X, | 
|  | 608 | mbedtls_mpi_uint min, | 
|  | 609 | const mbedtls_mpi_uint *N, | 
|  | 610 | size_t limbs, | 
|  | 611 | int (*f_rng)(void *, unsigned char *, size_t), | 
|  | 612 | void *p_rng); | 
| Gilles Peskine | 4a8c5cd | 2022-10-18 18:15:01 +0200 | [diff] [blame] | 613 |  | 
| Janos Follath | b6673f0 | 2022-09-30 14:13:14 +0100 | [diff] [blame] | 614 | /** | 
| Tom Cosgrove | 0a0dded | 2022-12-06 14:37:18 +0000 | [diff] [blame] | 615 | * \brief          Returns the number of limbs of working memory required for | 
|  | 616 | *                 a call to `mbedtls_mpi_core_exp_mod()`. | 
| Janos Follath | b6673f0 | 2022-09-30 14:13:14 +0100 | [diff] [blame] | 617 | * | 
| Tom Cosgrove | 28ff92c | 2022-12-12 17:06:27 +0000 | [diff] [blame] | 618 | * \note           This will always be at least | 
|  | 619 | *                 `mbedtls_mpi_core_montmul_working_limbs(AN_limbs)`, | 
|  | 620 | *                 i.e. sufficient for a call to `mbedtls_mpi_core_montmul()`. | 
|  | 621 | * | 
| Tom Cosgrove | 0a0dded | 2022-12-06 14:37:18 +0000 | [diff] [blame] | 622 | * \param AN_limbs The number of limbs in the input `A` and the modulus `N` | 
|  | 623 | *                 (they must be the same size) that will be given to | 
|  | 624 | *                 `mbedtls_mpi_core_exp_mod()`. | 
|  | 625 | * \param E_limbs  The number of limbs in the exponent `E` that will be given | 
|  | 626 | *                 to `mbedtls_mpi_core_exp_mod()`. | 
| Janos Follath | b6673f0 | 2022-09-30 14:13:14 +0100 | [diff] [blame] | 627 | * | 
| Tom Cosgrove | 0a0dded | 2022-12-06 14:37:18 +0000 | [diff] [blame] | 628 | * \return         The number of limbs of working memory required by | 
|  | 629 | *                 `mbedtls_mpi_core_exp_mod()`. | 
| Janos Follath | b6673f0 | 2022-09-30 14:13:14 +0100 | [diff] [blame] | 630 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 631 | size_t mbedtls_mpi_core_exp_mod_working_limbs(size_t AN_limbs, size_t E_limbs); | 
| Tom Cosgrove | 0a0dded | 2022-12-06 14:37:18 +0000 | [diff] [blame] | 632 |  | 
|  | 633 | /** | 
| Manuel Pégourié-Gonnard | 75ed587 | 2024-06-18 12:52:45 +0200 | [diff] [blame] | 634 | * \brief            Perform a modular exponentiation with public or secret exponent: | 
|  | 635 | *                   X = A^E mod N, where \p A is already in Montgomery form. | 
|  | 636 | * | 
| Janos Follath | 38ff70e | 2024-08-12 18:20:59 +0100 | [diff] [blame] | 637 | * \warning          This function is not constant time with respect to \p E (the exponent). | 
|  | 638 | * | 
| Manuel Pégourié-Gonnard | 75ed587 | 2024-06-18 12:52:45 +0200 | [diff] [blame] | 639 | * \p X may be aliased to \p A, but not to \p RR or \p E, even if \p E_limbs == | 
|  | 640 | * \p AN_limbs. | 
|  | 641 | * | 
|  | 642 | * \param[out] X     The destination MPI, as a little endian array of length | 
|  | 643 | *                   \p AN_limbs. | 
|  | 644 | * \param[in] A      The base MPI, as a little endian array of length \p AN_limbs. | 
|  | 645 | *                   Must be in Montgomery form. | 
|  | 646 | * \param[in] N      The modulus, as a little endian array of length \p AN_limbs. | 
|  | 647 | * \param AN_limbs   The number of limbs in \p X, \p A, \p N, \p RR. | 
|  | 648 | * \param[in] E      The exponent, as a little endian array of length \p E_limbs. | 
|  | 649 | * \param E_limbs    The number of limbs in \p E. | 
|  | 650 | * \param[in] RR     The precomputed residue of 2^{2*biL} modulo N, as a little | 
|  | 651 | *                   endian array of length \p AN_limbs. | 
|  | 652 | * \param[in,out] T  Temporary storage of at least the number of limbs returned | 
|  | 653 | *                   by `mbedtls_mpi_core_exp_mod_working_limbs()`. | 
|  | 654 | *                   Its initial content is unused and its final content is | 
|  | 655 | *                   indeterminate. | 
|  | 656 | *                   It must not alias or otherwise overlap any of the other | 
|  | 657 | *                   parameters. | 
|  | 658 | *                   It is up to the caller to zeroize \p T when it is no | 
|  | 659 | *                   longer needed, and before freeing it if it was dynamically | 
|  | 660 | *                   allocated. | 
| Manuel Pégourié-Gonnard | 75ed587 | 2024-06-18 12:52:45 +0200 | [diff] [blame] | 661 | */ | 
| Janos Follath | 38ff70e | 2024-08-12 18:20:59 +0100 | [diff] [blame] | 662 | void mbedtls_mpi_core_exp_mod_unsafe(mbedtls_mpi_uint *X, | 
|  | 663 | const mbedtls_mpi_uint *A, | 
|  | 664 | const mbedtls_mpi_uint *N, size_t AN_limbs, | 
|  | 665 | const mbedtls_mpi_uint *E, size_t E_limbs, | 
|  | 666 | const mbedtls_mpi_uint *RR, | 
|  | 667 | mbedtls_mpi_uint *T); | 
| Manuel Pégourié-Gonnard | 75ed587 | 2024-06-18 12:52:45 +0200 | [diff] [blame] | 668 |  | 
|  | 669 | /** | 
| Tom Cosgrove | 0a0dded | 2022-12-06 14:37:18 +0000 | [diff] [blame] | 670 | * \brief            Perform a modular exponentiation with secret exponent: | 
|  | 671 | *                   X = A^E mod N, where \p A is already in Montgomery form. | 
|  | 672 | * | 
| Tom Cosgrove | a7f0d7b | 2022-12-07 13:29:07 +0000 | [diff] [blame] | 673 | * \p X may be aliased to \p A, but not to \p RR or \p E, even if \p E_limbs == | 
|  | 674 | * \p AN_limbs. | 
|  | 675 | * | 
| Tom Cosgrove | 0a0dded | 2022-12-06 14:37:18 +0000 | [diff] [blame] | 676 | * \param[out] X     The destination MPI, as a little endian array of length | 
|  | 677 | *                   \p AN_limbs. | 
|  | 678 | * \param[in] A      The base MPI, as a little endian array of length \p AN_limbs. | 
|  | 679 | *                   Must be in Montgomery form. | 
|  | 680 | * \param[in] N      The modulus, as a little endian array of length \p AN_limbs. | 
|  | 681 | * \param AN_limbs   The number of limbs in \p X, \p A, \p N, \p RR. | 
|  | 682 | * \param[in] E      The exponent, as a little endian array of length \p E_limbs. | 
|  | 683 | * \param E_limbs    The number of limbs in \p E. | 
|  | 684 | * \param[in] RR     The precomputed residue of 2^{2*biL} modulo N, as a little | 
|  | 685 | *                   endian array of length \p AN_limbs. | 
|  | 686 | * \param[in,out] T  Temporary storage of at least the number of limbs returned | 
|  | 687 | *                   by `mbedtls_mpi_core_exp_mod_working_limbs()`. | 
|  | 688 | *                   Its initial content is unused and its final content is | 
|  | 689 | *                   indeterminate. | 
|  | 690 | *                   It must not alias or otherwise overlap any of the other | 
|  | 691 | *                   parameters. | 
|  | 692 | *                   It is up to the caller to zeroize \p T when it is no | 
|  | 693 | *                   longer needed, and before freeing it if it was dynamically | 
|  | 694 | *                   allocated. | 
|  | 695 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 696 | void mbedtls_mpi_core_exp_mod(mbedtls_mpi_uint *X, | 
|  | 697 | const mbedtls_mpi_uint *A, | 
|  | 698 | const mbedtls_mpi_uint *N, size_t AN_limbs, | 
|  | 699 | const mbedtls_mpi_uint *E, size_t E_limbs, | 
|  | 700 | const mbedtls_mpi_uint *RR, | 
|  | 701 | mbedtls_mpi_uint *T); | 
| Janos Follath | b6673f0 | 2022-09-30 14:13:14 +0100 | [diff] [blame] | 702 |  | 
| Hanno Becker | d9b2348 | 2022-08-25 08:25:19 +0100 | [diff] [blame] | 703 | /** | 
|  | 704 | * \brief Subtract unsigned integer from known-size large unsigned integers. | 
|  | 705 | *        Return the borrow. | 
|  | 706 | * | 
| Tom Cosgrove | f7ff4c9 | 2022-08-25 08:39:07 +0100 | [diff] [blame] | 707 | * \param[out] X    The result of the subtraction. | 
|  | 708 | * \param[in] A     The left operand. | 
|  | 709 | * \param b         The unsigned scalar to subtract. | 
|  | 710 | * \param limbs     Number of limbs of \p X and \p A. | 
| Hanno Becker | d9b2348 | 2022-08-25 08:25:19 +0100 | [diff] [blame] | 711 | * | 
| Tom Cosgrove | f7ff4c9 | 2022-08-25 08:39:07 +0100 | [diff] [blame] | 712 | * \return          1 if `A < b`. | 
|  | 713 | *                  0 if `A >= b`. | 
| Hanno Becker | d9b2348 | 2022-08-25 08:25:19 +0100 | [diff] [blame] | 714 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 715 | mbedtls_mpi_uint mbedtls_mpi_core_sub_int(mbedtls_mpi_uint *X, | 
|  | 716 | const mbedtls_mpi_uint *A, | 
|  | 717 | mbedtls_mpi_uint b, | 
|  | 718 | size_t limbs); | 
| Hanno Becker | d9b2348 | 2022-08-25 08:25:19 +0100 | [diff] [blame] | 719 |  | 
| Tom Cosgrove | 30f3b4d | 2022-12-12 16:54:57 +0000 | [diff] [blame] | 720 | /** | 
|  | 721 | * \brief Determine if a given MPI has the value \c 0 in constant time with | 
|  | 722 | *        respect to the value (but not with respect to the number of limbs). | 
|  | 723 | * | 
|  | 724 | * \param[in] A   The MPI to test. | 
|  | 725 | * \param limbs   Number of limbs in \p A. | 
|  | 726 | * | 
| Janos Follath | aec1a86 | 2024-02-21 11:24:20 +0000 | [diff] [blame] | 727 | * \return        MBEDTLS_CT_FALSE if `A == 0` | 
|  | 728 | *                MBEDTLS_CT_TRUE  if `A != 0`. | 
| Tom Cosgrove | 30f3b4d | 2022-12-12 16:54:57 +0000 | [diff] [blame] | 729 | */ | 
| Janos Follath | adb9d2d | 2024-03-11 10:03:05 +0000 | [diff] [blame] | 730 | mbedtls_ct_condition_t mbedtls_mpi_core_check_zero_ct(const mbedtls_mpi_uint *A, | 
| Janos Follath | aec1a86 | 2024-02-21 11:24:20 +0000 | [diff] [blame] | 731 | size_t limbs); | 
| Tom Cosgrove | 30f3b4d | 2022-12-12 16:54:57 +0000 | [diff] [blame] | 732 |  | 
| Tom Cosgrove | 28ff92c | 2022-12-12 17:06:27 +0000 | [diff] [blame] | 733 | /** | 
|  | 734 | * \brief          Returns the number of limbs of working memory required for | 
|  | 735 | *                 a call to `mbedtls_mpi_core_montmul()`. | 
|  | 736 | * | 
|  | 737 | * \param AN_limbs The number of limbs in the input `A` and the modulus `N` | 
|  | 738 | *                 (they must be the same size) that will be given to | 
|  | 739 | *                 `mbedtls_mpi_core_montmul()` or one of the other functions | 
|  | 740 | *                 that specifies this as the amount of working memory needed. | 
|  | 741 | * | 
|  | 742 | * \return         The number of limbs of working memory required by | 
|  | 743 | *                 `mbedtls_mpi_core_montmul()` (or other similar function). | 
|  | 744 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 745 | static inline size_t mbedtls_mpi_core_montmul_working_limbs(size_t AN_limbs) | 
| Tom Cosgrove | 28ff92c | 2022-12-12 17:06:27 +0000 | [diff] [blame] | 746 | { | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 747 | return 2 * AN_limbs + 1; | 
| Tom Cosgrove | 28ff92c | 2022-12-12 17:06:27 +0000 | [diff] [blame] | 748 | } | 
|  | 749 |  | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 750 | /** Convert an MPI into Montgomery form. | 
|  | 751 | * | 
|  | 752 | * \p X may be aliased to \p A, but may not otherwise overlap it. | 
|  | 753 | * | 
| Tom Cosgrove | 5c8505f | 2023-03-07 11:39:52 +0000 | [diff] [blame] | 754 | * \p X may not alias \p N (it is in canonical form, so must be strictly less | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 755 | * than \p N). Nor may it alias or overlap \p rr (this is unlikely to be | 
|  | 756 | * required in practice.) | 
|  | 757 | * | 
|  | 758 | * This function is a thin wrapper around `mbedtls_mpi_core_montmul()` that is | 
|  | 759 | * an alternative to calling `mbedtls_mpi_mod_raw_to_mont_rep()` when we | 
|  | 760 | * don't want to allocate memory. | 
|  | 761 | * | 
|  | 762 | * \param[out]    X         The result of the conversion. | 
|  | 763 | *                          Must have the same number of limbs as \p A. | 
|  | 764 | * \param[in]     A         The MPI to convert into Montgomery form. | 
|  | 765 | *                          Must have the same number of limbs as the modulus. | 
|  | 766 | * \param[in]     N         The address of the modulus, which gives the size of | 
| Tom Cosgrove | f723754 | 2022-12-16 16:10:36 +0000 | [diff] [blame] | 767 | *                          the base `R` = 2^(biL*N->limbs). | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 768 | * \param[in]     AN_limbs  The number of limbs in \p X, \p A, \p N and \p rr. | 
|  | 769 | * \param         mm        The Montgomery constant for \p N: -N^-1 mod 2^biL. | 
| Tom Cosgrove | b38c2ed | 2022-12-14 13:11:46 +0000 | [diff] [blame] | 770 | *                          This can be determined by calling | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 771 | *                          `mbedtls_mpi_core_montmul_init()`. | 
|  | 772 | * \param[in]     rr        The residue for `2^{2*n*biL} mod N`. | 
|  | 773 | * \param[in,out] T         Temporary storage of size at least | 
|  | 774 | *                          `mbedtls_mpi_core_montmul_working_limbs(AN_limbs)` | 
|  | 775 | *                          limbs. | 
|  | 776 | *                          Its initial content is unused and | 
|  | 777 | *                          its final content is indeterminate. | 
|  | 778 | *                          It must not alias or otherwise overlap any of the | 
|  | 779 | *                          other parameters. | 
|  | 780 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 781 | void mbedtls_mpi_core_to_mont_rep(mbedtls_mpi_uint *X, | 
|  | 782 | const mbedtls_mpi_uint *A, | 
|  | 783 | const mbedtls_mpi_uint *N, | 
|  | 784 | size_t AN_limbs, | 
|  | 785 | mbedtls_mpi_uint mm, | 
|  | 786 | const mbedtls_mpi_uint *rr, | 
|  | 787 | mbedtls_mpi_uint *T); | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 788 |  | 
|  | 789 | /** Convert an MPI from Montgomery form. | 
|  | 790 | * | 
|  | 791 | * \p X may be aliased to \p A, but may not otherwise overlap it. | 
|  | 792 | * | 
| Tom Cosgrove | 5c8505f | 2023-03-07 11:39:52 +0000 | [diff] [blame] | 793 | * \p X may not alias \p N (it is in canonical form, so must be strictly less | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 794 | * than \p N). | 
|  | 795 | * | 
|  | 796 | * This function is a thin wrapper around `mbedtls_mpi_core_montmul()` that is | 
|  | 797 | * an alternative to calling `mbedtls_mpi_mod_raw_from_mont_rep()` when we | 
|  | 798 | * don't want to allocate memory. | 
|  | 799 | * | 
|  | 800 | * \param[out]    X         The result of the conversion. | 
|  | 801 | *                          Must have the same number of limbs as \p A. | 
|  | 802 | * \param[in]     A         The MPI to convert from Montgomery form. | 
|  | 803 | *                          Must have the same number of limbs as the modulus. | 
|  | 804 | * \param[in]     N         The address of the modulus, which gives the size of | 
| Tom Cosgrove | f723754 | 2022-12-16 16:10:36 +0000 | [diff] [blame] | 805 | *                          the base `R` = 2^(biL*N->limbs). | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 806 | * \param[in]     AN_limbs  The number of limbs in \p X, \p A and \p N. | 
|  | 807 | * \param         mm        The Montgomery constant for \p N: -N^-1 mod 2^biL. | 
| Tom Cosgrove | b38c2ed | 2022-12-14 13:11:46 +0000 | [diff] [blame] | 808 | *                          This can be determined by calling | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 809 | *                          `mbedtls_mpi_core_montmul_init()`. | 
|  | 810 | * \param[in,out] T         Temporary storage of size at least | 
|  | 811 | *                          `mbedtls_mpi_core_montmul_working_limbs(AN_limbs)` | 
|  | 812 | *                          limbs. | 
|  | 813 | *                          Its initial content is unused and | 
|  | 814 | *                          its final content is indeterminate. | 
|  | 815 | *                          It must not alias or otherwise overlap any of the | 
|  | 816 | *                          other parameters. | 
|  | 817 | */ | 
| Gilles Peskine | 449bd83 | 2023-01-11 14:50:10 +0100 | [diff] [blame] | 818 | void mbedtls_mpi_core_from_mont_rep(mbedtls_mpi_uint *X, | 
|  | 819 | const mbedtls_mpi_uint *A, | 
|  | 820 | const mbedtls_mpi_uint *N, | 
|  | 821 | size_t AN_limbs, | 
|  | 822 | mbedtls_mpi_uint mm, | 
|  | 823 | mbedtls_mpi_uint *T); | 
| Tom Cosgrove | 786848b | 2022-12-13 10:45:19 +0000 | [diff] [blame] | 824 |  | 
| Manuel Pégourié-Gonnard | 07a0577 | 2025-07-09 10:42:28 +0200 | [diff] [blame] | 825 | /** Compute GCD(A, N) and optionally the inverse of A mod N if it exists. | 
|  | 826 | * | 
|  | 827 | * Requires N to be odd, and 0 <= A <= N. | 
| Manuel Pégourié-Gonnard | 5972096 | 2025-07-17 09:39:55 +0200 | [diff] [blame] | 828 | * When I != NULL, N (the modulus) must not be 1. | 
| Manuel Pégourié-Gonnard | 07a0577 | 2025-07-09 10:42:28 +0200 | [diff] [blame] | 829 | * | 
| Manuel Pégourié-Gonnard | 5972096 | 2025-07-17 09:39:55 +0200 | [diff] [blame] | 830 | * A and N may not alias each other. | 
| Manuel Pégourié-Gonnard | de5eeb5 | 2025-07-15 12:12:36 +0200 | [diff] [blame] | 831 | * When I == NULL (computing only the GCD), G may alias A or N. | 
|  | 832 | * When I != NULL (computing the modular inverse), G or I may alias A | 
|  | 833 | * but none of them may alias N (the modulus). | 
| Manuel Pégourié-Gonnard | 07a0577 | 2025-07-09 10:42:28 +0200 | [diff] [blame] | 834 | * | 
|  | 835 | * \param[out]    G       The GCD of \p A and \p N. | 
|  | 836 | *                        Must have the same number of limbs as \p N. | 
|  | 837 | * \param[out]    I       The inverse of \p A modulo \p N if it exists (that is, | 
|  | 838 | *                        if \p G above is 1 on exit); indeterminate otherwise. | 
|  | 839 | *                        This must either be NULL (to only compute the GCD), | 
|  | 840 | *                        or have the same number of limbs as \p N. | 
|  | 841 | * \param[in]     A       The 1st operand of GCD and number to invert. | 
|  | 842 | *                        This value must be less than or equal to \p N. | 
|  | 843 | * \param         A_limbs The number of limbs of \p A. | 
|  | 844 | *                        Must be less than or equal to \p N_limbs. | 
|  | 845 | * \param[in]     N       The 2nd operand of GCD and modulus for inversion. | 
|  | 846 | *                        Must be odd or the results are indeterminate. | 
|  | 847 | * \param         N_limbs The number of limbs of \p N. | 
|  | 848 | * \param[in,out] T       Temporary storage of size at least 5 * N_limbs limbs, | 
|  | 849 | *                        or 4 * N_limbs if \p I is NULL (GCD only). | 
|  | 850 | *                        Its initial content is unused and | 
|  | 851 | *                        its final content is indeterminate. | 
|  | 852 | *                        It must not alias or otherwise overlap any of the | 
|  | 853 | *                        other parameters. | 
|  | 854 | */ | 
|  | 855 | void mbedtls_mpi_core_gcd_modinv_odd(mbedtls_mpi_uint *G, | 
|  | 856 | mbedtls_mpi_uint *I, | 
|  | 857 | const mbedtls_mpi_uint *A, | 
|  | 858 | size_t A_limbs, | 
|  | 859 | const mbedtls_mpi_uint *N, | 
|  | 860 | size_t N_limbs, | 
|  | 861 | mbedtls_mpi_uint *T); | 
|  | 862 |  | 
| Gabor Mezei | b903070 | 2022-07-18 23:09:45 +0200 | [diff] [blame] | 863 | #endif /* MBEDTLS_BIGNUM_CORE_H */ |