2018-01-07 23:55:52 +00:00
|
|
|
// Copyright (c) 2017-2018, The Monero Project
|
2017-11-30 19:59:10 +00:00
|
|
|
//
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without modification, are
|
|
|
|
// permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
|
|
// conditions and the following disclaimer.
|
|
|
|
//
|
|
|
|
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
|
|
// of conditions and the following disclaimer in the documentation and/or other
|
|
|
|
// materials provided with the distribution.
|
|
|
|
//
|
|
|
|
// 3. Neither the name of the copyright holder nor the names of its contributors may be
|
|
|
|
// used to endorse or promote products derived from this software without specific
|
|
|
|
// prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
|
|
|
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
|
|
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
|
|
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
//
|
|
|
|
// Adapted from Java code by Sarang Noether
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <openssl/ssl.h>
|
2018-08-21 15:28:06 +00:00
|
|
|
#include <openssl/bn.h>
|
2017-11-30 19:59:10 +00:00
|
|
|
#include <boost/thread/mutex.hpp>
|
|
|
|
#include "misc_log_ex.h"
|
|
|
|
#include "common/perf_timer.h"
|
2018-03-30 19:44:51 +00:00
|
|
|
#include "cryptonote_config.h"
|
2017-11-30 19:59:10 +00:00
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
#include "crypto/crypto-ops.h"
|
|
|
|
}
|
|
|
|
#include "rctOps.h"
|
2018-01-09 13:51:17 +00:00
|
|
|
#include "multiexp.h"
|
2017-11-30 19:59:10 +00:00
|
|
|
#include "bulletproofs.h"
|
|
|
|
|
|
|
|
#undef MONERO_DEFAULT_LOG_CATEGORY
|
|
|
|
#define MONERO_DEFAULT_LOG_CATEGORY "bulletproofs"
|
|
|
|
|
|
|
|
//#define DEBUG_BP
|
|
|
|
|
|
|
|
#define PERF_TIMER_START_BP(x) PERF_TIMER_START_UNIT(x, 1000000)
|
|
|
|
|
2018-05-29 11:39:00 +00:00
|
|
|
#define STRAUS_SIZE_LIMIT 128
|
|
|
|
#define PIPPENGER_SIZE_LIMIT 0
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
namespace rct
|
|
|
|
{
|
|
|
|
|
|
|
|
static rct::key vector_exponent(const rct::keyV &a, const rct::keyV &b);
|
2018-01-03 21:37:18 +00:00
|
|
|
static rct::keyV vector_powers(const rct::key &x, size_t n);
|
|
|
|
static rct::keyV vector_dup(const rct::key &x, size_t n);
|
2017-11-30 19:59:10 +00:00
|
|
|
static rct::key inner_product(const rct::keyV &a, const rct::keyV &b);
|
|
|
|
|
|
|
|
static constexpr size_t maxN = 64;
|
2018-03-30 19:44:51 +00:00
|
|
|
static constexpr size_t maxM = BULLETPROOF_MAX_OUTPUTS;
|
2018-01-03 21:37:18 +00:00
|
|
|
static rct::key Hi[maxN*maxM], Gi[maxN*maxM];
|
2018-01-09 13:51:17 +00:00
|
|
|
static ge_p3 Hi_p3[maxN*maxM], Gi_p3[maxN*maxM];
|
2018-05-29 11:39:00 +00:00
|
|
|
static std::shared_ptr<straus_cached_data> straus_HiGi_cache;
|
|
|
|
static std::shared_ptr<pippenger_cached_data> pippenger_HiGi_cache;
|
2017-11-30 19:59:10 +00:00
|
|
|
static const rct::key TWO = { {0x02, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 , 0x00, 0x00, 0x00,0x00 } };
|
2018-08-06 11:05:20 +00:00
|
|
|
static const rct::key MINUS_ONE = { { 0xec, 0xd3, 0xf5, 0x5c, 0x1a, 0x63, 0x12, 0x58, 0xd6, 0x9c, 0xf7, 0xa2, 0xde, 0xf9, 0xde, 0x14, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10 } };
|
|
|
|
static const rct::key MINUS_INV_EIGHT = { { 0x74, 0xa4, 0x19, 0x7a, 0xf0, 0x7d, 0x0b, 0xf7, 0x05, 0xc2, 0xda, 0x25, 0x2b, 0x5c, 0x0b, 0x0d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0a } };
|
2018-01-03 21:37:18 +00:00
|
|
|
static const rct::keyV oneN = vector_dup(rct::identity(), maxN);
|
2017-11-30 19:59:10 +00:00
|
|
|
static const rct::keyV twoN = vector_powers(TWO, maxN);
|
|
|
|
static const rct::key ip12 = inner_product(oneN, twoN);
|
|
|
|
static boost::mutex init_mutex;
|
|
|
|
|
2018-01-14 23:06:55 +00:00
|
|
|
static inline rct::key multiexp(const std::vector<MultiexpData> &data, bool HiGi)
|
|
|
|
{
|
2018-05-27 23:27:54 +00:00
|
|
|
if (HiGi)
|
2018-05-29 11:39:00 +00:00
|
|
|
{
|
|
|
|
static_assert(128 <= STRAUS_SIZE_LIMIT, "Straus in precalc mode can only be calculated till STRAUS_SIZE_LIMIT");
|
2018-07-16 13:40:51 +00:00
|
|
|
return data.size() <= 128 ? straus(data, straus_HiGi_cache, 0) : pippenger(data, pippenger_HiGi_cache, get_pippenger_c(data.size()));
|
2018-05-29 11:39:00 +00:00
|
|
|
}
|
2018-01-14 23:06:55 +00:00
|
|
|
else
|
2018-07-16 13:40:51 +00:00
|
|
|
return data.size() <= 64 ? straus(data, NULL, 0) : pippenger(data, NULL, get_pippenger_c(data.size()));
|
|
|
|
}
|
|
|
|
|
2018-08-08 21:20:50 +00:00
|
|
|
static inline bool is_reduced(const rct::key &scalar)
|
2018-07-16 13:40:51 +00:00
|
|
|
{
|
2018-08-08 21:20:50 +00:00
|
|
|
return sc_check(scalar.bytes) == 0;
|
2018-01-14 23:06:55 +00:00
|
|
|
}
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
static rct::key get_exponent(const rct::key &base, size_t idx)
|
|
|
|
{
|
|
|
|
static const std::string salt("bulletproof");
|
|
|
|
std::string hashed = std::string((const char*)base.bytes, sizeof(base)) + salt + tools::get_varint_data(idx);
|
2018-07-25 09:10:46 +00:00
|
|
|
const rct::key e = rct::hashToPoint(rct::hash2rct(crypto::cn_fast_hash(hashed.data(), hashed.size())));
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(!(e == rct::identity()), "Exponent is point at infinity");
|
|
|
|
return e;
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void init_exponents()
|
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> lock(init_mutex);
|
|
|
|
|
|
|
|
static bool init_done = false;
|
|
|
|
if (init_done)
|
|
|
|
return;
|
2018-01-18 12:01:45 +00:00
|
|
|
std::vector<MultiexpData> data;
|
2018-01-03 21:37:18 +00:00
|
|
|
for (size_t i = 0; i < maxN*maxM; ++i)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
|
|
|
Hi[i] = get_exponent(rct::H, i * 2);
|
2018-01-09 13:51:17 +00:00
|
|
|
CHECK_AND_ASSERT_THROW_MES(ge_frombytes_vartime(&Hi_p3[i], Hi[i].bytes) == 0, "ge_frombytes_vartime failed");
|
2017-11-30 19:59:10 +00:00
|
|
|
Gi[i] = get_exponent(rct::H, i * 2 + 1);
|
2018-01-09 13:51:17 +00:00
|
|
|
CHECK_AND_ASSERT_THROW_MES(ge_frombytes_vartime(&Gi_p3[i], Gi[i].bytes) == 0, "ge_frombytes_vartime failed");
|
2018-01-18 12:01:45 +00:00
|
|
|
|
2018-08-08 21:20:50 +00:00
|
|
|
data.push_back({rct::zero(), Gi_p3[i]});
|
|
|
|
data.push_back({rct::zero(), Hi_p3[i]});
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
2018-05-29 11:39:00 +00:00
|
|
|
|
|
|
|
straus_HiGi_cache = straus_init_cache(data, STRAUS_SIZE_LIMIT);
|
|
|
|
pippenger_HiGi_cache = pippenger_init_cache(data, PIPPENGER_SIZE_LIMIT);
|
|
|
|
|
|
|
|
MINFO("Hi/Gi cache size: " << (sizeof(Hi)+sizeof(Gi))/1024 << " kB");
|
|
|
|
MINFO("Hi_p3/Gi_p3 cache size: " << (sizeof(Hi_p3)+sizeof(Gi_p3))/1024 << " kB");
|
|
|
|
MINFO("Straus cache size: " << straus_get_cache_size(straus_HiGi_cache)/1024 << " kB");
|
|
|
|
MINFO("Pippenger cache size: " << pippenger_get_cache_size(pippenger_HiGi_cache)/1024 << " kB");
|
|
|
|
size_t cache_size = (sizeof(Hi)+sizeof(Hi_p3))*2 + straus_get_cache_size(straus_HiGi_cache) + pippenger_get_cache_size(pippenger_HiGi_cache);
|
|
|
|
MINFO("Total cache size: " << cache_size/1024 << "kB");
|
2017-11-30 19:59:10 +00:00
|
|
|
init_done = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Given two scalar arrays, construct a vector commitment */
|
|
|
|
static rct::key vector_exponent(const rct::keyV &a, const rct::keyV &b)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(a.size() == b.size(), "Incompatible sizes of a and b");
|
2018-01-03 21:37:18 +00:00
|
|
|
CHECK_AND_ASSERT_THROW_MES(a.size() <= maxN*maxM, "Incompatible sizes of a and maxN");
|
2018-01-14 23:06:55 +00:00
|
|
|
|
2018-01-09 13:51:17 +00:00
|
|
|
std::vector<MultiexpData> multiexp_data;
|
|
|
|
multiexp_data.reserve(a.size()*2);
|
|
|
|
for (size_t i = 0; i < a.size(); ++i)
|
|
|
|
{
|
2018-01-14 23:06:55 +00:00
|
|
|
multiexp_data.emplace_back(a[i], Gi_p3[i]);
|
|
|
|
multiexp_data.emplace_back(b[i], Hi_p3[i]);
|
2018-01-09 13:51:17 +00:00
|
|
|
}
|
2018-01-14 23:06:55 +00:00
|
|
|
return multiexp(multiexp_data, true);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Compute a custom vector-scalar commitment */
|
2018-08-07 08:02:42 +00:00
|
|
|
static rct::key cross_vector_exponent8(size_t size, const std::vector<ge_p3> &A, size_t Ao, const std::vector<ge_p3> &B, size_t Bo, const rct::keyV &a, size_t ao, const rct::keyV &b, size_t bo, const ge_p3 *extra_point, const rct::key *extra_scalar)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-08-07 08:02:42 +00:00
|
|
|
CHECK_AND_ASSERT_THROW_MES(size + Ao <= A.size(), "Incompatible size for A");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(size + Bo <= B.size(), "Incompatible size for B");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(size + ao <= a.size(), "Incompatible size for a");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(size + bo <= b.size(), "Incompatible size for b");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(size <= maxN*maxM, "size is too large");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(!!extra_point == !!extra_scalar, "only one of extra point/scalar present");
|
2018-01-14 23:06:55 +00:00
|
|
|
|
2018-01-09 13:51:17 +00:00
|
|
|
std::vector<MultiexpData> multiexp_data;
|
2018-08-07 08:02:42 +00:00
|
|
|
multiexp_data.resize(size*2 + (!!extra_point));
|
|
|
|
for (size_t i = 0; i < size; ++i)
|
2018-01-09 13:51:17 +00:00
|
|
|
{
|
2018-08-07 08:02:42 +00:00
|
|
|
sc_mul(multiexp_data[i*2].scalar.bytes, a[ao+i].bytes, INV_EIGHT.bytes);;
|
|
|
|
multiexp_data[i*2].point = A[Ao+i];
|
|
|
|
sc_mul(multiexp_data[i*2+1].scalar.bytes, b[bo+i].bytes, INV_EIGHT.bytes);
|
|
|
|
multiexp_data[i*2+1].point = B[Bo+i];
|
|
|
|
}
|
|
|
|
if (extra_point)
|
|
|
|
{
|
|
|
|
sc_mul(multiexp_data.back().scalar.bytes, extra_scalar->bytes, INV_EIGHT.bytes);
|
|
|
|
multiexp_data.back().point = *extra_point;
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
2018-01-14 23:06:55 +00:00
|
|
|
return multiexp(multiexp_data, false);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Given a scalar, construct a vector of powers */
|
2018-01-03 21:37:18 +00:00
|
|
|
static rct::keyV vector_powers(const rct::key &x, size_t n)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
|
|
|
rct::keyV res(n);
|
|
|
|
if (n == 0)
|
|
|
|
return res;
|
|
|
|
res[0] = rct::identity();
|
|
|
|
if (n == 1)
|
|
|
|
return res;
|
|
|
|
res[1] = x;
|
|
|
|
for (size_t i = 2; i < n; ++i)
|
|
|
|
{
|
|
|
|
sc_mul(res[i].bytes, res[i-1].bytes, x.bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-01-05 22:39:59 +00:00
|
|
|
/* Given a scalar, return the sum of its powers from 0 to n-1 */
|
|
|
|
static rct::key vector_power_sum(const rct::key &x, size_t n)
|
|
|
|
{
|
|
|
|
if (n == 0)
|
|
|
|
return rct::zero();
|
|
|
|
rct::key res = rct::identity();
|
|
|
|
if (n == 1)
|
|
|
|
return res;
|
|
|
|
rct::key prev = x;
|
|
|
|
for (size_t i = 1; i < n; ++i)
|
|
|
|
{
|
|
|
|
if (i > 1)
|
|
|
|
sc_mul(prev.bytes, prev.bytes, x.bytes);
|
|
|
|
sc_add(res.bytes, res.bytes, prev.bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
/* Given two scalar arrays, construct the inner product */
|
|
|
|
static rct::key inner_product(const rct::keyV &a, const rct::keyV &b)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(a.size() == b.size(), "Incompatible sizes of a and b");
|
|
|
|
rct::key res = rct::zero();
|
|
|
|
for (size_t i = 0; i < a.size(); ++i)
|
|
|
|
{
|
|
|
|
sc_muladd(res.bytes, a[i].bytes, b[i].bytes, res.bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Given two scalar arrays, construct the Hadamard product */
|
|
|
|
static rct::keyV hadamard(const rct::keyV &a, const rct::keyV &b)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(a.size() == b.size(), "Incompatible sizes of a and b");
|
|
|
|
rct::keyV res(a.size());
|
|
|
|
for (size_t i = 0; i < a.size(); ++i)
|
|
|
|
{
|
|
|
|
sc_mul(res[i].bytes, a[i].bytes, b[i].bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-08-07 08:02:42 +00:00
|
|
|
/* folds a curvepoint array using a two way scaled Hadamard product */
|
|
|
|
static void hadamard_fold(std::vector<ge_p3> &v, const rct::key &a, const rct::key &b)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-08-07 08:02:42 +00:00
|
|
|
CHECK_AND_ASSERT_THROW_MES((v.size() & 1) == 0, "Vector size should be even");
|
|
|
|
const size_t sz = v.size() / 2;
|
|
|
|
for (size_t n = 0; n < sz; ++n)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-08-07 08:02:42 +00:00
|
|
|
ge_dsmp c[2];
|
|
|
|
ge_dsm_precomp(c[0], &v[n]);
|
|
|
|
ge_dsm_precomp(c[1], &v[sz + n]);
|
|
|
|
ge_double_scalarmult_precomp_vartime2_p3(&v[n], a.bytes, c[0], b.bytes, c[1]);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
2018-08-07 08:02:42 +00:00
|
|
|
v.resize(sz);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Add two vectors */
|
|
|
|
static rct::keyV vector_add(const rct::keyV &a, const rct::keyV &b)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(a.size() == b.size(), "Incompatible sizes of a and b");
|
|
|
|
rct::keyV res(a.size());
|
|
|
|
for (size_t i = 0; i < a.size(); ++i)
|
|
|
|
{
|
|
|
|
sc_add(res[i].bytes, a[i].bytes, b[i].bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Subtract two vectors */
|
|
|
|
static rct::keyV vector_subtract(const rct::keyV &a, const rct::keyV &b)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(a.size() == b.size(), "Incompatible sizes of a and b");
|
|
|
|
rct::keyV res(a.size());
|
|
|
|
for (size_t i = 0; i < a.size(); ++i)
|
|
|
|
{
|
|
|
|
sc_sub(res[i].bytes, a[i].bytes, b[i].bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Multiply a scalar and a vector */
|
|
|
|
static rct::keyV vector_scalar(const rct::keyV &a, const rct::key &x)
|
|
|
|
{
|
|
|
|
rct::keyV res(a.size());
|
|
|
|
for (size_t i = 0; i < a.size(); ++i)
|
|
|
|
{
|
|
|
|
sc_mul(res[i].bytes, a[i].bytes, x.bytes);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-01-03 21:37:18 +00:00
|
|
|
/* Create a vector from copies of a single value */
|
|
|
|
static rct::keyV vector_dup(const rct::key &x, size_t N)
|
|
|
|
{
|
|
|
|
return rct::keyV(N, x);
|
|
|
|
}
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
static rct::key switch_endianness(rct::key k)
|
|
|
|
{
|
|
|
|
std::reverse(k.bytes, k.bytes + sizeof(k));
|
|
|
|
return k;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compute the inverse of a scalar, the stupid way */
|
|
|
|
static rct::key invert(const rct::key &x)
|
|
|
|
{
|
|
|
|
rct::key inv;
|
|
|
|
|
|
|
|
BN_CTX *ctx = BN_CTX_new();
|
|
|
|
BIGNUM *X = BN_new();
|
|
|
|
BIGNUM *L = BN_new();
|
|
|
|
BIGNUM *I = BN_new();
|
|
|
|
|
|
|
|
BN_bin2bn(switch_endianness(x).bytes, sizeof(rct::key), X);
|
|
|
|
BN_bin2bn(switch_endianness(rct::curveOrder()).bytes, sizeof(rct::key), L);
|
|
|
|
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(BN_mod_inverse(I, X, L, ctx), "Failed to invert");
|
|
|
|
|
|
|
|
const int len = BN_num_bytes(I);
|
|
|
|
CHECK_AND_ASSERT_THROW_MES((size_t)len <= sizeof(rct::key), "Invalid number length");
|
|
|
|
inv = rct::zero();
|
|
|
|
BN_bn2bin(I, inv.bytes);
|
|
|
|
std::reverse(inv.bytes, inv.bytes + len);
|
|
|
|
|
|
|
|
BN_free(I);
|
|
|
|
BN_free(L);
|
|
|
|
BN_free(X);
|
|
|
|
BN_CTX_free(ctx);
|
|
|
|
|
|
|
|
#ifdef DEBUG_BP
|
|
|
|
rct::key tmp;
|
|
|
|
sc_mul(tmp.bytes, inv.bytes, x.bytes);
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(tmp == rct::identity(), "invert failed");
|
|
|
|
#endif
|
|
|
|
return inv;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compute the slice of a vector */
|
|
|
|
static rct::keyV slice(const rct::keyV &a, size_t start, size_t stop)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(start < a.size(), "Invalid start index");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(stop <= a.size(), "Invalid stop index");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(start < stop, "Invalid start/stop indices");
|
|
|
|
rct::keyV res(stop - start);
|
|
|
|
for (size_t i = start; i < stop; ++i)
|
|
|
|
{
|
|
|
|
res[i - start] = a[i];
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-12-18 21:58:45 +00:00
|
|
|
static rct::key hash_cache_mash(rct::key &hash_cache, const rct::key &mash0, const rct::key &mash1)
|
|
|
|
{
|
|
|
|
rct::keyV data;
|
|
|
|
data.reserve(3);
|
|
|
|
data.push_back(hash_cache);
|
|
|
|
data.push_back(mash0);
|
|
|
|
data.push_back(mash1);
|
|
|
|
return hash_cache = rct::hash_to_scalar(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static rct::key hash_cache_mash(rct::key &hash_cache, const rct::key &mash0, const rct::key &mash1, const rct::key &mash2)
|
|
|
|
{
|
|
|
|
rct::keyV data;
|
|
|
|
data.reserve(4);
|
|
|
|
data.push_back(hash_cache);
|
|
|
|
data.push_back(mash0);
|
|
|
|
data.push_back(mash1);
|
|
|
|
data.push_back(mash2);
|
|
|
|
return hash_cache = rct::hash_to_scalar(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static rct::key hash_cache_mash(rct::key &hash_cache, const rct::key &mash0, const rct::key &mash1, const rct::key &mash2, const rct::key &mash3)
|
|
|
|
{
|
|
|
|
rct::keyV data;
|
|
|
|
data.reserve(5);
|
|
|
|
data.push_back(hash_cache);
|
|
|
|
data.push_back(mash0);
|
|
|
|
data.push_back(mash1);
|
|
|
|
data.push_back(mash2);
|
|
|
|
data.push_back(mash3);
|
|
|
|
return hash_cache = rct::hash_to_scalar(data);
|
|
|
|
}
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
/* Given a value v (0..2^N-1) and a mask gamma, construct a range proof */
|
|
|
|
Bulletproof bulletproof_PROVE(const rct::key &sv, const rct::key &gamma)
|
|
|
|
{
|
2018-08-08 18:39:31 +00:00
|
|
|
return bulletproof_PROVE(rct::keyV(1, sv), rct::keyV(1, gamma));
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Bulletproof bulletproof_PROVE(uint64_t v, const rct::key &gamma)
|
|
|
|
{
|
2018-08-08 18:39:31 +00:00
|
|
|
return bulletproof_PROVE(std::vector<uint64_t>(1, v), rct::keyV(1, gamma));
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
|
2018-01-03 21:37:18 +00:00
|
|
|
/* Given a set of values v (0..2^N-1) and masks gamma, construct a range proof */
|
|
|
|
Bulletproof bulletproof_PROVE(const rct::keyV &sv, const rct::keyV &gamma)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(sv.size() == gamma.size(), "Incompatible sizes of sv and gamma");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(!sv.empty(), "sv is empty");
|
2018-07-16 13:40:51 +00:00
|
|
|
for (const rct::key &sve: sv)
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(is_reduced(sve), "Invalid sv input");
|
|
|
|
for (const rct::key &g: gamma)
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(is_reduced(g), "Invalid gamma input");
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
init_exponents();
|
|
|
|
|
|
|
|
PERF_TIMER_UNIT(PROVE, 1000000);
|
|
|
|
|
|
|
|
constexpr size_t logN = 6; // log2(64)
|
|
|
|
constexpr size_t N = 1<<logN;
|
|
|
|
size_t M, logM;
|
|
|
|
for (logM = 0; (M = 1<<logM) <= maxM && M < sv.size(); ++logM);
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(M <= maxM, "sv/gamma are too large");
|
|
|
|
const size_t logMN = logM + logN;
|
|
|
|
const size_t MN = M * N;
|
|
|
|
|
|
|
|
rct::keyV V(sv.size());
|
|
|
|
rct::keyV aL(MN), aR(MN);
|
2018-08-08 15:01:41 +00:00
|
|
|
rct::keyV aL8(MN), aR8(MN);
|
2018-08-08 12:14:13 +00:00
|
|
|
rct::key tmp, tmp2;
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
PERF_TIMER_START_BP(PROVE_v);
|
|
|
|
for (size_t i = 0; i < sv.size(); ++i)
|
2018-08-06 11:05:20 +00:00
|
|
|
{
|
2018-08-08 12:14:13 +00:00
|
|
|
rct::key gamma8, sv8;
|
|
|
|
sc_mul(gamma8.bytes, gamma[i].bytes, INV_EIGHT.bytes);
|
|
|
|
sc_mul(sv8.bytes, sv[i].bytes, INV_EIGHT.bytes);
|
|
|
|
rct::addKeys2(V[i], gamma8, sv8, rct::H);
|
2018-08-06 11:05:20 +00:00
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
PERF_TIMER_STOP(PROVE_v);
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(PROVE_aLaR);
|
|
|
|
for (size_t j = 0; j < M; ++j)
|
|
|
|
{
|
|
|
|
for (size_t i = N; i-- > 0; )
|
|
|
|
{
|
2018-08-08 15:01:41 +00:00
|
|
|
if (j < sv.size() && (sv[j][i/8] & (((uint64_t)1)<<(i%8))))
|
2018-01-03 21:37:18 +00:00
|
|
|
{
|
|
|
|
aL[j*N+i] = rct::identity();
|
2018-08-08 15:01:41 +00:00
|
|
|
aL8[j*N+i] = INV_EIGHT;
|
|
|
|
aR[j*N+i] = aR8[j*N+i] = rct::zero();
|
2018-01-03 21:37:18 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-08-08 15:01:41 +00:00
|
|
|
aL[j*N+i] = aL8[j*N+i] = rct::zero();
|
|
|
|
aR[j*N+i] = MINUS_ONE;
|
|
|
|
aR8[j*N+i] = MINUS_INV_EIGHT;
|
2018-01-03 21:37:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PERF_TIMER_STOP(PROVE_aLaR);
|
|
|
|
|
|
|
|
// DEBUG: Test to ensure this recovers the value
|
|
|
|
#ifdef DEBUG_BP
|
|
|
|
for (size_t j = 0; j < M; ++j)
|
|
|
|
{
|
|
|
|
uint64_t test_aL = 0, test_aR = 0;
|
|
|
|
for (size_t i = 0; i < N; ++i)
|
|
|
|
{
|
|
|
|
if (aL[j*N+i] == rct::identity())
|
|
|
|
test_aL += ((uint64_t)1)<<i;
|
|
|
|
if (aR[j*N+i] == rct::zero())
|
|
|
|
test_aR += ((uint64_t)1)<<i;
|
|
|
|
}
|
|
|
|
uint64_t v_test = 0;
|
|
|
|
if (j < sv.size())
|
|
|
|
for (int n = 0; n < 8; ++n) v_test |= (((uint64_t)sv[j][n]) << (8*n));
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(test_aL == v_test, "test_aL failed");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(test_aR == v_test, "test_aR failed");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-07-24 18:56:48 +00:00
|
|
|
try_again:
|
|
|
|
rct::key hash_cache = rct::hash_to_scalar(V);
|
|
|
|
|
2018-01-03 21:37:18 +00:00
|
|
|
PERF_TIMER_START_BP(PROVE_step1);
|
|
|
|
// PAPER LINES 38-39
|
|
|
|
rct::key alpha = rct::skGen();
|
2018-08-08 15:01:41 +00:00
|
|
|
rct::key ve = vector_exponent(aL8, aR8);
|
2018-01-03 21:37:18 +00:00
|
|
|
rct::key A;
|
2018-08-08 15:01:41 +00:00
|
|
|
sc_mul(tmp.bytes, alpha.bytes, INV_EIGHT.bytes);
|
|
|
|
rct::addKeys(A, ve, rct::scalarmultBase(tmp));
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 40-42
|
|
|
|
rct::keyV sL = rct::skvGen(MN), sR = rct::skvGen(MN);
|
|
|
|
rct::key rho = rct::skGen();
|
|
|
|
ve = vector_exponent(sL, sR);
|
|
|
|
rct::key S;
|
|
|
|
rct::addKeys(S, ve, rct::scalarmultBase(rho));
|
2018-08-06 11:05:20 +00:00
|
|
|
S = rct::scalarmultKey(S, INV_EIGHT);
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 43-45
|
|
|
|
rct::key y = hash_cache_mash(hash_cache, A, S);
|
2018-07-24 18:56:48 +00:00
|
|
|
if (y == rct::zero())
|
|
|
|
{
|
|
|
|
PERF_TIMER_STOP(PROVE_step1);
|
|
|
|
MINFO("y is 0, trying again");
|
|
|
|
goto try_again;
|
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
rct::key z = hash_cache = rct::hash_to_scalar(y);
|
2018-07-24 18:56:48 +00:00
|
|
|
if (z == rct::zero())
|
|
|
|
{
|
|
|
|
PERF_TIMER_STOP(PROVE_step1);
|
|
|
|
MINFO("z is 0, trying again");
|
|
|
|
goto try_again;
|
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// Polynomial construction by coefficients
|
|
|
|
const auto zMN = vector_dup(z, MN);
|
|
|
|
rct::keyV l0 = vector_subtract(aL, zMN);
|
|
|
|
const rct::keyV &l1 = sL;
|
|
|
|
|
|
|
|
// This computes the ugly sum/concatenation from PAPER LINE 65
|
|
|
|
rct::keyV zero_twos(MN);
|
|
|
|
const rct::keyV zpow = vector_powers(z, M+2);
|
|
|
|
for (size_t i = 0; i < MN; ++i)
|
|
|
|
{
|
|
|
|
zero_twos[i] = rct::zero();
|
|
|
|
for (size_t j = 1; j <= M; ++j)
|
|
|
|
{
|
|
|
|
if (i >= (j-1)*N && i < j*N)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(1+j < zpow.size(), "invalid zpow index");
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(i-(j-1)*N < twoN.size(), "invalid twoN index");
|
|
|
|
sc_muladd(zero_twos[i].bytes, zpow[1+j].bytes, twoN[i-(j-1)*N].bytes, zero_twos[i].bytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rct::keyV r0 = vector_add(aR, zMN);
|
|
|
|
const auto yMN = vector_powers(y, MN);
|
|
|
|
r0 = hadamard(r0, yMN);
|
|
|
|
r0 = vector_add(r0, zero_twos);
|
|
|
|
rct::keyV r1 = hadamard(yMN, sR);
|
|
|
|
|
|
|
|
// Polynomial construction before PAPER LINE 46
|
|
|
|
rct::key t1_1 = inner_product(l0, r1);
|
|
|
|
rct::key t1_2 = inner_product(l1, r0);
|
|
|
|
rct::key t1;
|
|
|
|
sc_add(t1.bytes, t1_1.bytes, t1_2.bytes);
|
|
|
|
rct::key t2 = inner_product(l1, r1);
|
|
|
|
|
|
|
|
PERF_TIMER_STOP(PROVE_step1);
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(PROVE_step2);
|
|
|
|
// PAPER LINES 47-48
|
|
|
|
rct::key tau1 = rct::skGen(), tau2 = rct::skGen();
|
|
|
|
|
2018-08-08 12:14:13 +00:00
|
|
|
rct::key T1, T2;
|
|
|
|
ge_p3 p3;
|
|
|
|
sc_mul(tmp.bytes, t1.bytes, INV_EIGHT.bytes);
|
|
|
|
sc_mul(tmp2.bytes, tau1.bytes, INV_EIGHT.bytes);
|
|
|
|
ge_double_scalarmult_base_vartime_p3(&p3, tmp.bytes, &ge_p3_H, tmp2.bytes);
|
|
|
|
ge_p3_tobytes(T1.bytes, &p3);
|
|
|
|
sc_mul(tmp.bytes, t2.bytes, INV_EIGHT.bytes);
|
|
|
|
sc_mul(tmp2.bytes, tau2.bytes, INV_EIGHT.bytes);
|
|
|
|
ge_double_scalarmult_base_vartime_p3(&p3, tmp.bytes, &ge_p3_H, tmp2.bytes);
|
|
|
|
ge_p3_tobytes(T2.bytes, &p3);
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 49-51
|
|
|
|
rct::key x = hash_cache_mash(hash_cache, z, T1, T2);
|
2018-07-24 18:56:48 +00:00
|
|
|
if (x == rct::zero())
|
|
|
|
{
|
|
|
|
PERF_TIMER_STOP(PROVE_step2);
|
|
|
|
MINFO("x is 0, trying again");
|
|
|
|
goto try_again;
|
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 52-53
|
|
|
|
rct::key taux;
|
|
|
|
sc_mul(taux.bytes, tau1.bytes, x.bytes);
|
|
|
|
rct::key xsq;
|
|
|
|
sc_mul(xsq.bytes, x.bytes, x.bytes);
|
|
|
|
sc_muladd(taux.bytes, tau2.bytes, xsq.bytes, taux.bytes);
|
|
|
|
for (size_t j = 1; j <= sv.size(); ++j)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(j+1 < zpow.size(), "invalid zpow index");
|
|
|
|
sc_muladd(taux.bytes, zpow[j+1].bytes, gamma[j-1].bytes, taux.bytes);
|
|
|
|
}
|
|
|
|
rct::key mu;
|
|
|
|
sc_muladd(mu.bytes, x.bytes, rho.bytes, alpha.bytes);
|
|
|
|
|
|
|
|
// PAPER LINES 54-57
|
|
|
|
rct::keyV l = l0;
|
|
|
|
l = vector_add(l, vector_scalar(l1, x));
|
|
|
|
rct::keyV r = r0;
|
|
|
|
r = vector_add(r, vector_scalar(r1, x));
|
|
|
|
PERF_TIMER_STOP(PROVE_step2);
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(PROVE_step3);
|
|
|
|
rct::key t = inner_product(l, r);
|
|
|
|
|
|
|
|
// DEBUG: Test if the l and r vectors match the polynomial forms
|
|
|
|
#ifdef DEBUG_BP
|
|
|
|
rct::key test_t;
|
|
|
|
const rct::key t0 = inner_product(l0, r0);
|
|
|
|
sc_muladd(test_t.bytes, t1.bytes, x.bytes, t0.bytes);
|
|
|
|
sc_muladd(test_t.bytes, t2.bytes, xsq.bytes, test_t.bytes);
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(test_t == t, "test_t check failed");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// PAPER LINES 32-33
|
|
|
|
rct::key x_ip = hash_cache_mash(hash_cache, x, taux, mu, t);
|
2018-07-24 18:56:48 +00:00
|
|
|
if (x_ip == rct::zero())
|
|
|
|
{
|
|
|
|
PERF_TIMER_STOP(PROVE_step3);
|
|
|
|
MINFO("x_ip is 0, trying again");
|
|
|
|
goto try_again;
|
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// These are used in the inner product rounds
|
|
|
|
size_t nprime = MN;
|
2018-08-07 08:02:42 +00:00
|
|
|
std::vector<ge_p3> Gprime(MN);
|
|
|
|
std::vector<ge_p3> Hprime(MN);
|
2018-01-03 21:37:18 +00:00
|
|
|
rct::keyV aprime(MN);
|
|
|
|
rct::keyV bprime(MN);
|
|
|
|
const rct::key yinv = invert(y);
|
|
|
|
rct::key yinvpow = rct::identity();
|
|
|
|
for (size_t i = 0; i < MN; ++i)
|
|
|
|
{
|
2018-08-07 08:02:42 +00:00
|
|
|
Gprime[i] = Gi_p3[i];
|
|
|
|
ge_scalarmult_p3(&Hprime[i], yinvpow.bytes, &Hi_p3[i]);
|
2018-01-03 21:37:18 +00:00
|
|
|
sc_mul(yinvpow.bytes, yinvpow.bytes, yinv.bytes);
|
|
|
|
aprime[i] = l[i];
|
|
|
|
bprime[i] = r[i];
|
|
|
|
}
|
|
|
|
rct::keyV L(logMN);
|
|
|
|
rct::keyV R(logMN);
|
|
|
|
int round = 0;
|
|
|
|
rct::keyV w(logMN); // this is the challenge x in the inner product protocol
|
|
|
|
PERF_TIMER_STOP(PROVE_step3);
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(PROVE_step4);
|
|
|
|
// PAPER LINE 13
|
|
|
|
while (nprime > 1)
|
|
|
|
{
|
|
|
|
// PAPER LINE 15
|
|
|
|
nprime /= 2;
|
|
|
|
|
|
|
|
// PAPER LINES 16-17
|
2018-08-07 08:02:42 +00:00
|
|
|
PERF_TIMER_START_BP(PROVE_inner_product);
|
2018-01-03 21:37:18 +00:00
|
|
|
rct::key cL = inner_product(slice(aprime, 0, nprime), slice(bprime, nprime, bprime.size()));
|
|
|
|
rct::key cR = inner_product(slice(aprime, nprime, aprime.size()), slice(bprime, 0, nprime));
|
2018-08-07 08:02:42 +00:00
|
|
|
PERF_TIMER_STOP(PROVE_inner_product);
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 18-19
|
2018-08-07 08:02:42 +00:00
|
|
|
PERF_TIMER_START_BP(PROVE_LR);
|
2018-01-03 21:37:18 +00:00
|
|
|
sc_mul(tmp.bytes, cL.bytes, x_ip.bytes);
|
2018-08-07 08:02:42 +00:00
|
|
|
L[round] = cross_vector_exponent8(nprime, Gprime, nprime, Hprime, 0, aprime, 0, bprime, nprime, &ge_p3_H, &tmp);
|
2018-01-03 21:37:18 +00:00
|
|
|
sc_mul(tmp.bytes, cR.bytes, x_ip.bytes);
|
2018-08-07 08:02:42 +00:00
|
|
|
R[round] = cross_vector_exponent8(nprime, Gprime, 0, Hprime, nprime, aprime, nprime, bprime, 0, &ge_p3_H, &tmp);
|
|
|
|
PERF_TIMER_STOP(PROVE_LR);
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 21-22
|
|
|
|
w[round] = hash_cache_mash(hash_cache, L[round], R[round]);
|
2018-07-24 18:56:48 +00:00
|
|
|
if (w[round] == rct::zero())
|
|
|
|
{
|
|
|
|
PERF_TIMER_STOP(PROVE_step4);
|
|
|
|
MINFO("w[round] is 0, trying again");
|
|
|
|
goto try_again;
|
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 24-25
|
|
|
|
const rct::key winv = invert(w[round]);
|
2018-08-07 08:02:42 +00:00
|
|
|
if (nprime > 1)
|
|
|
|
{
|
|
|
|
PERF_TIMER_START_BP(PROVE_hadamard2);
|
|
|
|
hadamard_fold(Gprime, winv, w[round]);
|
|
|
|
hadamard_fold(Hprime, w[round], winv);
|
|
|
|
PERF_TIMER_STOP(PROVE_hadamard2);
|
|
|
|
}
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
// PAPER LINES 28-29
|
2018-08-07 08:02:42 +00:00
|
|
|
PERF_TIMER_START_BP(PROVE_prime);
|
2018-01-03 21:37:18 +00:00
|
|
|
aprime = vector_add(vector_scalar(slice(aprime, 0, nprime), w[round]), vector_scalar(slice(aprime, nprime, aprime.size()), winv));
|
|
|
|
bprime = vector_add(vector_scalar(slice(bprime, 0, nprime), winv), vector_scalar(slice(bprime, nprime, bprime.size()), w[round]));
|
2018-08-07 08:02:42 +00:00
|
|
|
PERF_TIMER_STOP(PROVE_prime);
|
2018-01-03 21:37:18 +00:00
|
|
|
|
|
|
|
++round;
|
|
|
|
}
|
|
|
|
PERF_TIMER_STOP(PROVE_step4);
|
|
|
|
|
|
|
|
// PAPER LINE 58 (with inclusions from PAPER LINE 8 and PAPER LINE 20)
|
2018-08-08 12:14:13 +00:00
|
|
|
return Bulletproof(std::move(V), A, S, T1, T2, taux, mu, std::move(L), std::move(R), aprime[0], bprime[0], t);
|
2018-01-03 21:37:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Bulletproof bulletproof_PROVE(const std::vector<uint64_t> &v, const rct::keyV &gamma)
|
|
|
|
{
|
|
|
|
CHECK_AND_ASSERT_THROW_MES(v.size() == gamma.size(), "Incompatible sizes of v and gamma");
|
|
|
|
|
|
|
|
// vG + gammaH
|
|
|
|
PERF_TIMER_START_BP(PROVE_v);
|
|
|
|
rct::keyV sv(v.size());
|
|
|
|
for (size_t i = 0; i < v.size(); ++i)
|
|
|
|
{
|
|
|
|
sv[i] = rct::zero();
|
|
|
|
sv[i].bytes[0] = v[i] & 255;
|
|
|
|
sv[i].bytes[1] = (v[i] >> 8) & 255;
|
|
|
|
sv[i].bytes[2] = (v[i] >> 16) & 255;
|
|
|
|
sv[i].bytes[3] = (v[i] >> 24) & 255;
|
|
|
|
sv[i].bytes[4] = (v[i] >> 32) & 255;
|
|
|
|
sv[i].bytes[5] = (v[i] >> 40) & 255;
|
|
|
|
sv[i].bytes[6] = (v[i] >> 48) & 255;
|
|
|
|
sv[i].bytes[7] = (v[i] >> 56) & 255;
|
|
|
|
}
|
|
|
|
PERF_TIMER_STOP(PROVE_v);
|
|
|
|
return bulletproof_PROVE(sv, gamma);
|
|
|
|
}
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
/* Given a range proof, determine if it is valid */
|
2018-02-03 14:36:29 +00:00
|
|
|
bool bulletproof_VERIFY(const std::vector<const Bulletproof*> &proofs)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
|
|
|
init_exponents();
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(VERIFY);
|
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
// sanity and figure out which proof is longest
|
|
|
|
size_t max_length = 0;
|
2018-08-23 18:52:05 +00:00
|
|
|
size_t nV = 0;
|
2018-02-03 14:36:29 +00:00
|
|
|
for (const Bulletproof *p: proofs)
|
2018-01-03 21:37:18 +00:00
|
|
|
{
|
2018-02-03 14:36:29 +00:00
|
|
|
const Bulletproof &proof = *p;
|
2018-06-29 14:03:00 +00:00
|
|
|
|
2018-07-16 13:40:51 +00:00
|
|
|
// check scalar range
|
|
|
|
CHECK_AND_ASSERT_MES(is_reduced(proof.taux), false, "Input scalar not in range");
|
|
|
|
CHECK_AND_ASSERT_MES(is_reduced(proof.mu), false, "Input scalar not in range");
|
|
|
|
CHECK_AND_ASSERT_MES(is_reduced(proof.a), false, "Input scalar not in range");
|
|
|
|
CHECK_AND_ASSERT_MES(is_reduced(proof.b), false, "Input scalar not in range");
|
|
|
|
CHECK_AND_ASSERT_MES(is_reduced(proof.t), false, "Input scalar not in range");
|
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
CHECK_AND_ASSERT_MES(proof.V.size() >= 1, false, "V does not have at least one element");
|
|
|
|
CHECK_AND_ASSERT_MES(proof.L.size() == proof.R.size(), false, "Mismatched L and R sizes");
|
|
|
|
CHECK_AND_ASSERT_MES(proof.L.size() > 0, false, "Empty proof");
|
|
|
|
|
|
|
|
max_length = std::max(max_length, proof.L.size());
|
2018-08-23 18:52:05 +00:00
|
|
|
nV += proof.V.size();
|
2018-01-03 21:37:18 +00:00
|
|
|
}
|
2018-02-03 14:36:29 +00:00
|
|
|
CHECK_AND_ASSERT_MES(max_length < 32, false, "At least one proof is too large");
|
|
|
|
size_t maxMN = 1u << max_length;
|
2017-11-30 19:59:10 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
const size_t logN = 6;
|
|
|
|
const size_t N = 1 << logN;
|
|
|
|
rct::key tmp;
|
|
|
|
|
2018-08-23 18:52:05 +00:00
|
|
|
std::vector<MultiexpData> multiexp_data;
|
|
|
|
multiexp_data.reserve(nV + (2 * (10/*logM*/ + logN) + 4) * proofs.size() + 2 * maxMN);
|
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
// setup weighted aggregates
|
|
|
|
rct::key z1 = rct::zero();
|
|
|
|
rct::key z3 = rct::zero();
|
|
|
|
rct::keyV z4(maxMN, rct::zero()), z5(maxMN, rct::zero());
|
2018-08-06 10:09:46 +00:00
|
|
|
rct::key y0 = rct::zero(), y1 = rct::zero();
|
2018-02-03 14:36:29 +00:00
|
|
|
for (const Bulletproof *p: proofs)
|
2018-01-05 22:39:59 +00:00
|
|
|
{
|
2018-02-03 14:36:29 +00:00
|
|
|
const Bulletproof &proof = *p;
|
|
|
|
|
|
|
|
size_t M, logM;
|
|
|
|
for (logM = 0; (M = 1<<logM) <= maxM && M < proof.V.size(); ++logM);
|
|
|
|
CHECK_AND_ASSERT_MES(proof.L.size() == 6+logM, false, "Proof is not the expected size");
|
|
|
|
const size_t MN = M*N;
|
2018-08-23 18:52:05 +00:00
|
|
|
const rct::key weight_y = rct::skGen();
|
|
|
|
const rct::key weight_z = rct::skGen();
|
2018-02-03 14:36:29 +00:00
|
|
|
|
|
|
|
// Reconstruct the challenges
|
|
|
|
PERF_TIMER_START_BP(VERIFY_start);
|
|
|
|
rct::key hash_cache = rct::hash_to_scalar(proof.V);
|
|
|
|
rct::key y = hash_cache_mash(hash_cache, proof.A, proof.S);
|
2018-07-24 18:56:48 +00:00
|
|
|
CHECK_AND_ASSERT_MES(!(y == rct::zero()), false, "y == 0");
|
2018-02-03 14:36:29 +00:00
|
|
|
rct::key z = hash_cache = rct::hash_to_scalar(y);
|
2018-07-24 18:56:48 +00:00
|
|
|
CHECK_AND_ASSERT_MES(!(z == rct::zero()), false, "z == 0");
|
2018-02-03 14:36:29 +00:00
|
|
|
rct::key x = hash_cache_mash(hash_cache, z, proof.T1, proof.T2);
|
2018-07-24 18:56:48 +00:00
|
|
|
CHECK_AND_ASSERT_MES(!(x == rct::zero()), false, "x == 0");
|
2018-02-03 14:36:29 +00:00
|
|
|
rct::key x_ip = hash_cache_mash(hash_cache, x, proof.taux, proof.mu, proof.t);
|
2018-07-24 18:56:48 +00:00
|
|
|
CHECK_AND_ASSERT_MES(!(x_ip == rct::zero()), false, "x_ip == 0");
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_start);
|
|
|
|
|
2018-09-24 14:01:33 +00:00
|
|
|
// pre-multiply some points by 8
|
|
|
|
rct::keyV proof8_V = proof.V; for (rct::key &k: proof8_V) k = rct::scalarmult8(k);
|
|
|
|
rct::keyV proof8_L = proof.L; for (rct::key &k: proof8_L) k = rct::scalarmult8(k);
|
|
|
|
rct::keyV proof8_R = proof.R; for (rct::key &k: proof8_R) k = rct::scalarmult8(k);
|
|
|
|
rct::key proof8_T1 = rct::scalarmult8(proof.T1);
|
|
|
|
rct::key proof8_T2 = rct::scalarmult8(proof.T2);
|
|
|
|
rct::key proof8_S = rct::scalarmult8(proof.S);
|
2018-08-23 18:52:05 +00:00
|
|
|
rct::key proof8_A = rct::scalarmult8(proof.A);
|
2018-09-24 14:01:33 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_START_BP(VERIFY_line_61);
|
|
|
|
// PAPER LINE 61
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_muladd(y0.bytes, proof.taux.bytes, weight_y.bytes, y0.bytes);
|
2018-02-03 14:36:29 +00:00
|
|
|
|
|
|
|
const rct::keyV zpow = vector_powers(z, M+3);
|
|
|
|
|
|
|
|
rct::key k;
|
|
|
|
const rct::key ip1y = vector_power_sum(y, MN);
|
|
|
|
sc_mulsub(k.bytes, zpow[2].bytes, ip1y.bytes, rct::zero().bytes);
|
|
|
|
for (size_t j = 1; j <= M; ++j)
|
2018-01-09 13:51:17 +00:00
|
|
|
{
|
2018-02-03 14:36:29 +00:00
|
|
|
CHECK_AND_ASSERT_MES(j+2 < zpow.size(), false, "invalid zpow index");
|
|
|
|
sc_mulsub(k.bytes, zpow[j+2].bytes, ip12.bytes, k.bytes);
|
2018-01-09 13:51:17 +00:00
|
|
|
}
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_line_61);
|
|
|
|
|
2018-08-06 10:09:46 +00:00
|
|
|
PERF_TIMER_START_BP(VERIFY_line_61rl_new);
|
|
|
|
sc_muladd(tmp.bytes, z.bytes, ip1y.bytes, k.bytes);
|
|
|
|
sc_sub(tmp.bytes, proof.t.bytes, tmp.bytes);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_muladd(y1.bytes, tmp.bytes, weight_y.bytes, y1.bytes);
|
2018-09-24 14:01:33 +00:00
|
|
|
for (size_t j = 0; j < proof8_V.size(); j++)
|
2018-01-09 13:51:17 +00:00
|
|
|
{
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_mul(tmp.bytes, zpow[j+2].bytes, weight_y.bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, proof8_V[j]);
|
2018-02-03 14:36:29 +00:00
|
|
|
}
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_mul(tmp.bytes, x.bytes, weight_y.bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, proof8_T1);
|
2018-08-06 10:09:46 +00:00
|
|
|
rct::key xsq;
|
|
|
|
sc_mul(xsq.bytes, x.bytes, x.bytes);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_mul(tmp.bytes, xsq.bytes, weight_y.bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, proof8_T2);
|
2018-08-06 10:09:46 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_line_61rl_new);
|
2017-11-30 19:59:10 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_START_BP(VERIFY_line_62);
|
|
|
|
// PAPER LINE 62
|
2018-08-23 18:52:05 +00:00
|
|
|
multiexp_data.emplace_back(weight_z, proof8_A);
|
|
|
|
sc_mul(tmp.bytes, x.bytes, weight_z.bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, proof8_S);
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_line_62);
|
2017-11-30 19:59:10 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
// Compute the number of rounds for the inner product
|
|
|
|
const size_t rounds = logM+logN;
|
|
|
|
CHECK_AND_ASSERT_MES(rounds > 0, false, "Zero rounds");
|
2017-11-30 19:59:10 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_START_BP(VERIFY_line_21_22);
|
|
|
|
// PAPER LINES 21-22
|
|
|
|
// The inner product challenges are computed per round
|
|
|
|
rct::keyV w(rounds);
|
|
|
|
for (size_t i = 0; i < rounds; ++i)
|
|
|
|
{
|
|
|
|
w[i] = hash_cache_mash(hash_cache, proof.L[i], proof.R[i]);
|
2018-07-24 18:56:48 +00:00
|
|
|
CHECK_AND_ASSERT_MES(!(w[i] == rct::zero()), false, "w[i] == 0");
|
2018-02-03 14:36:29 +00:00
|
|
|
}
|
|
|
|
PERF_TIMER_STOP(VERIFY_line_21_22);
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(VERIFY_line_24_25);
|
|
|
|
// Basically PAPER LINES 24-25
|
|
|
|
// Compute the curvepoints from G[i] and H[i]
|
|
|
|
rct::key yinvpow = rct::identity();
|
|
|
|
rct::key ypow = rct::identity();
|
|
|
|
|
|
|
|
PERF_TIMER_START_BP(VERIFY_line_24_25_invert);
|
|
|
|
const rct::key yinv = invert(y);
|
|
|
|
rct::keyV winv(rounds);
|
|
|
|
for (size_t i = 0; i < rounds; ++i)
|
|
|
|
winv[i] = invert(w[i]);
|
|
|
|
PERF_TIMER_STOP(VERIFY_line_24_25_invert);
|
|
|
|
|
2018-08-07 09:59:14 +00:00
|
|
|
// precalc
|
|
|
|
PERF_TIMER_START_BP(VERIFY_line_24_25_precalc);
|
|
|
|
rct::keyV w_cache(1<<rounds);
|
|
|
|
w_cache[0] = winv[0];
|
|
|
|
w_cache[1] = w[0];
|
|
|
|
for (size_t j = 1; j < rounds; ++j)
|
|
|
|
{
|
|
|
|
const size_t slots = 1<<(j+1);
|
|
|
|
for (size_t s = slots; s-- > 0; --s)
|
|
|
|
{
|
|
|
|
sc_mul(w_cache[s].bytes, w_cache[s/2].bytes, w[j].bytes);
|
|
|
|
sc_mul(w_cache[s-1].bytes, w_cache[s/2].bytes, winv[j].bytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PERF_TIMER_STOP(VERIFY_line_24_25_precalc);
|
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
for (size_t i = 0; i < MN; ++i)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-02-03 14:36:29 +00:00
|
|
|
rct::key g_scalar = proof.a;
|
|
|
|
rct::key h_scalar;
|
2018-08-07 09:59:14 +00:00
|
|
|
if (i == 0)
|
|
|
|
h_scalar = proof.b;
|
|
|
|
else
|
|
|
|
sc_mul(h_scalar.bytes, proof.b.bytes, yinvpow.bytes);
|
2017-11-30 19:59:10 +00:00
|
|
|
|
2018-08-07 09:59:14 +00:00
|
|
|
// Convert the index to binary IN REVERSE and construct the scalar exponent
|
|
|
|
sc_mul(g_scalar.bytes, g_scalar.bytes, w_cache[i].bytes);
|
|
|
|
sc_mul(h_scalar.bytes, h_scalar.bytes, w_cache[(~i) & (MN-1)].bytes);
|
2018-02-03 14:36:29 +00:00
|
|
|
|
|
|
|
// Adjust the scalars using the exponents from PAPER LINE 62
|
|
|
|
sc_add(g_scalar.bytes, g_scalar.bytes, z.bytes);
|
|
|
|
CHECK_AND_ASSERT_MES(2+i/N < zpow.size(), false, "invalid zpow index");
|
|
|
|
CHECK_AND_ASSERT_MES(i%N < twoN.size(), false, "invalid twoN index");
|
|
|
|
sc_mul(tmp.bytes, zpow[2+i/N].bytes, twoN[i%N].bytes);
|
2018-08-07 09:59:14 +00:00
|
|
|
if (i == 0)
|
|
|
|
{
|
|
|
|
sc_add(tmp.bytes, tmp.bytes, z.bytes);
|
|
|
|
sc_sub(h_scalar.bytes, h_scalar.bytes, tmp.bytes);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sc_muladd(tmp.bytes, z.bytes, ypow.bytes, tmp.bytes);
|
|
|
|
sc_mulsub(h_scalar.bytes, tmp.bytes, yinvpow.bytes, h_scalar.bytes);
|
|
|
|
}
|
2018-02-03 14:36:29 +00:00
|
|
|
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_muladd(z4[i].bytes, g_scalar.bytes, weight_z.bytes, z4[i].bytes);
|
|
|
|
sc_muladd(z5[i].bytes, h_scalar.bytes, weight_z.bytes, z5[i].bytes);
|
2018-02-03 14:36:29 +00:00
|
|
|
|
2018-08-07 09:59:14 +00:00
|
|
|
if (i == 0)
|
|
|
|
{
|
|
|
|
yinvpow = yinv;
|
|
|
|
ypow = y;
|
|
|
|
}
|
|
|
|
else if (i != MN-1)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-02-03 14:36:29 +00:00
|
|
|
sc_mul(yinvpow.bytes, yinvpow.bytes, yinv.bytes);
|
|
|
|
sc_mul(ypow.bytes, ypow.bytes, y.bytes);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_line_24_25);
|
2017-11-30 19:59:10 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
// PAPER LINE 26
|
|
|
|
PERF_TIMER_START_BP(VERIFY_line_26_new);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_muladd(z1.bytes, proof.mu.bytes, weight_z.bytes, z1.bytes);
|
2018-02-03 14:36:29 +00:00
|
|
|
for (size_t i = 0; i < rounds; ++i)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-02-03 14:36:29 +00:00
|
|
|
sc_mul(tmp.bytes, w[i].bytes, w[i].bytes);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_mul(tmp.bytes, tmp.bytes, weight_z.bytes);
|
2018-09-24 14:01:33 +00:00
|
|
|
multiexp_data.emplace_back(tmp, proof8_L[i]);
|
2018-02-03 14:36:29 +00:00
|
|
|
sc_mul(tmp.bytes, winv[i].bytes, winv[i].bytes);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_mul(tmp.bytes, tmp.bytes, weight_z.bytes);
|
2018-09-24 14:01:33 +00:00
|
|
|
multiexp_data.emplace_back(tmp, proof8_R[i]);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
2018-02-03 14:36:29 +00:00
|
|
|
sc_mulsub(tmp.bytes, proof.a.bytes, proof.b.bytes, proof.t.bytes);
|
|
|
|
sc_mul(tmp.bytes, tmp.bytes, x_ip.bytes);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_muladd(z3.bytes, tmp.bytes, weight_z.bytes, z3.bytes);
|
2018-02-03 14:36:29 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_line_26_new);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
2018-01-09 13:51:17 +00:00
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
// now check all proofs at once
|
|
|
|
PERF_TIMER_START_BP(VERIFY_step2_check);
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_sub(tmp.bytes, rct::zero().bytes, y0.bytes);
|
|
|
|
sc_sub(tmp.bytes, tmp.bytes, z1.bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, rct::G);
|
|
|
|
sc_sub(tmp.bytes, z3.bytes, y1.bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, rct::H);
|
2018-02-03 14:36:29 +00:00
|
|
|
for (size_t i = 0; i < maxMN; ++i)
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-08-23 18:52:05 +00:00
|
|
|
sc_sub(tmp.bytes, rct::zero().bytes, z4[i].bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, Gi_p3[i]);
|
|
|
|
sc_sub(tmp.bytes, rct::zero().bytes, z5[i].bytes);
|
|
|
|
multiexp_data.emplace_back(tmp, Hi_p3[i]);
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|
2018-08-23 18:52:05 +00:00
|
|
|
if (!(multiexp(multiexp_data, false) == rct::identity()))
|
2017-11-30 19:59:10 +00:00
|
|
|
{
|
2018-08-23 18:52:05 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_step2_check);
|
|
|
|
MERROR("Verification failure");
|
2017-11-30 19:59:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
2018-08-23 18:52:05 +00:00
|
|
|
PERF_TIMER_STOP(VERIFY_step2_check);
|
2017-11-30 19:59:10 +00:00
|
|
|
|
|
|
|
PERF_TIMER_STOP(VERIFY);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-03 14:36:29 +00:00
|
|
|
bool bulletproof_VERIFY(const std::vector<Bulletproof> &proofs)
|
|
|
|
{
|
|
|
|
std::vector<const Bulletproof*> proof_pointers;
|
|
|
|
for (const Bulletproof &proof: proofs)
|
|
|
|
proof_pointers.push_back(&proof);
|
|
|
|
return bulletproof_VERIFY(proof_pointers);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool bulletproof_VERIFY(const Bulletproof &proof)
|
|
|
|
{
|
|
|
|
std::vector<const Bulletproof*> proofs;
|
|
|
|
proofs.push_back(&proof);
|
|
|
|
return bulletproof_VERIFY(proofs);
|
|
|
|
}
|
|
|
|
|
2017-11-30 19:59:10 +00:00
|
|
|
}
|