9#include <botan/internal/monty_exp.h> 
   11#include <botan/mem_ops.h> 
   12#include <botan/internal/ct_utils.h> 
   13#include <botan/internal/monty.h> 
   14#include <botan/internal/rounding.h> 
   18class Montgomery_Exponentation_State final {
 
   20      Montgomery_Exponentation_State(
const Montgomery_Int& g, 
size_t window_bits, 
bool const_time);
 
   22      Montgomery_Int exponentiation(
const BigInt& k, 
size_t max_k_bits) 
const;
 
   24      Montgomery_Int exponentiation_vartime(
const BigInt& k) 
const;
 
   27      Montgomery_Params m_params;
 
   28      std::vector<Montgomery_Int> m_g;
 
   32Montgomery_Exponentation_State::Montgomery_Exponentation_State(
const Montgomery_Int& g,
 
   35      m_params(g._params()), m_window_bits(window_bits == 0 ? 4 : window_bits) {
 
   36   if(m_window_bits < 1 || m_window_bits > 12) {  
 
   37      throw Invalid_Argument(
"Invalid window bits for Montgomery exponentiation");
 
   40   const size_t window_size = (
static_cast<size_t>(1) << m_window_bits);
 
   42   m_g.reserve(window_size);
 
   44   m_g.push_back(Montgomery_Int::one(m_params));
 
   48   secure_vector<word> ws(2 * m_params.p_words());
 
   50   for(
size_t i = 2; i != window_size; ++i) {
 
   51      m_g.push_back(m_g[1].mul(m_g[i - 1], ws));
 
   55      CT::poison_range(m_g);
 
   61void const_time_lookup(secure_vector<word>& output, 
const std::vector<Montgomery_Int>& g, 
size_t nibble) {
 
   64   const size_t words = output.size();
 
   68   for(
size_t i = 0; i != g.size(); i += 2) {
 
   69      const secure_vector<word>& vec_0 = g[i].repr();
 
   70      const secure_vector<word>& vec_1 = g[i + 1].repr();
 
   74      const auto mask_0 = CT::Mask<word>::is_equal(nibble, i);
 
   75      const auto mask_1 = CT::Mask<word>::is_equal(nibble, i + 1);
 
   77      for(
size_t w = 0; w != words; ++w) {
 
   78         output[w] |= mask_0.if_set_return(vec_0[w]);
 
   79         output[w] |= mask_1.if_set_return(vec_1[w]);
 
   86Montgomery_Int Montgomery_Exponentation_State::exponentiation(
const BigInt& scalar, 
size_t max_k_bits)
 const {
 
   90   const size_t exp_nibbles = (max_k_bits + m_window_bits - 1) / m_window_bits;
 
   92   if(exp_nibbles == 0) {
 
   93      return Montgomery_Int::one(m_params);
 
   96   secure_vector<word> e_bits(m_params.p_words());
 
   97   secure_vector<word> ws(2 * m_params.p_words());
 
   99   const_time_lookup(e_bits, m_g, scalar.get_substring(m_window_bits * (exp_nibbles - 1), m_window_bits));
 
  100   Montgomery_Int x(m_params, std::span{e_bits});
 
  102   for(
size_t i = exp_nibbles - 1; i > 0; --i) {
 
  103      x.square_this_n_times(ws, m_window_bits);
 
  104      const_time_lookup(e_bits, m_g, scalar.get_substring(m_window_bits * (i - 1), m_window_bits));
 
  105      x.mul_by(e_bits, ws);
 
  112Montgomery_Int Montgomery_Exponentation_State::exponentiation_vartime(
const BigInt& scalar)
 const {
 
  113   const size_t exp_nibbles = (scalar.bits() + m_window_bits - 1) / m_window_bits;
 
  115   secure_vector<word> ws(2 * m_params.p_words());
 
  117   if(exp_nibbles == 0) {
 
  118      return Montgomery_Int::one(m_params);
 
  121   Montgomery_Int x = m_g[scalar.get_substring(m_window_bits * (exp_nibbles - 1), m_window_bits)];
 
  123   for(
size_t i = exp_nibbles - 1; i > 0; --i) {
 
  124      x.square_this_n_times(ws, m_window_bits);
 
  126      const uint32_t nibble = scalar.get_substring(m_window_bits * (i - 1), m_window_bits);
 
  128         x.mul_by(m_g[nibble], ws);
 
  139   return std::make_shared<const Montgomery_Exponentation_State>(g, window_bits, const_time);
 
 
  154   return precomputed_state.exponentiation(k, max_k_bits);
 
 
  158   return precomputed_state.exponentiation_vartime(k);
 
 
  212   for(
size_t i = 0; i != z_bits; i += 2) {
 
  220      const uint32_t z12 = (4 * z2_b) + z1_b;
 
 
#define BOTAN_ASSERT_NOMSG(expr)
 
#define BOTAN_DEBUG_ASSERT(expr)
 
#define BOTAN_ARG_CHECK(expr, msg)
 
uint32_t get_substring(size_t offset, size_t length) const
 
Montgomery_Int square(secure_vector< word > &ws) const
 
static Montgomery_Int one(const Montgomery_Params ¶ms)
 
Montgomery_Int & square_this_n_times(secure_vector< word > &ws, size_t n)
 
Montgomery_Int mul(const Montgomery_Int &other, secure_vector< word > &ws) const
 
Montgomery_Int & mul_by(const Montgomery_Int &other, secure_vector< word > &ws)
 
Montgomery_Int monty_execute_vartime(const Montgomery_Exponentation_State &precomputed_state, const BigInt &k)
 
constexpr size_t round_up(size_t n, size_t align_to)
 
Montgomery_Int monty_multi_exp(const Montgomery_Params ¶ms_p, const BigInt &x_bn, const BigInt &z1, const BigInt &y_bn, const BigInt &z2)
 
Montgomery_Int monty_execute(const Montgomery_Exponentation_State &precomputed_state, const BigInt &k, size_t max_k_bits)
 
std::vector< T, secure_allocator< T > > secure_vector
 
std::shared_ptr< const Montgomery_Exponentation_State > monty_precompute(const Montgomery_Int &g, size_t window_bits, bool const_time)
 
constexpr void clear_mem(T *ptr, size_t n)