mirror of
https://github.com/SChernykh/p2pool.git
synced 2024-12-22 19:39:22 +00:00
Added merkle proofs
This commit is contained in:
parent
e51af9e6e8
commit
12a3eb862e
3 changed files with 86 additions and 0 deletions
|
@ -18,6 +18,7 @@
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "keccak.h"
|
#include "keccak.h"
|
||||||
#include "merkle.h"
|
#include "merkle.h"
|
||||||
|
#include "keccak.h"
|
||||||
|
|
||||||
namespace p2pool {
|
namespace p2pool {
|
||||||
|
|
||||||
|
@ -115,4 +116,81 @@ void merkle_hash_full_tree(const std::vector<hash>& hashes, std::vector<std::vec
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool get_merkle_proof(const std::vector<std::vector<hash>>& tree, const hash& h, std::vector<std::pair<bool, hash>>& proof)
|
||||||
|
{
|
||||||
|
if (tree.empty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<hash>& hashes = tree[0];
|
||||||
|
const size_t count = hashes.size();
|
||||||
|
|
||||||
|
size_t index = 0;
|
||||||
|
|
||||||
|
while ((index < count) && (hashes[index] != h)) {
|
||||||
|
++index;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (index >= count) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
proof.clear();
|
||||||
|
|
||||||
|
if (count == 1) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (count == 2) {
|
||||||
|
proof.emplace_back(index != 0, hashes[index ^ 1]);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
size_t cnt = 1;
|
||||||
|
do { cnt <<= 1; } while (cnt <= count);
|
||||||
|
cnt >>= 1;
|
||||||
|
|
||||||
|
const size_t k = cnt * 2 - count;
|
||||||
|
|
||||||
|
if (index >= k) {
|
||||||
|
index -= k;
|
||||||
|
const size_t j = (index ^ 1) + k;
|
||||||
|
if (j >= count) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
proof.emplace_back((index & 1) != 0, hashes[j]);
|
||||||
|
index = (index >> 1) + k;
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t n = tree.size();
|
||||||
|
|
||||||
|
for (size_t i = 1; cnt >= 2; ++i, index >>= 1, cnt >>= 1) {
|
||||||
|
const size_t j = index ^ 1;
|
||||||
|
if ((i >= n) || (j >= tree[i].size())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
proof.emplace_back((index & 1) != 0, tree[i][j]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool verify_merkle_proof(hash h, const std::vector<std::pair<bool, hash>>& proof, const hash& root)
|
||||||
|
{
|
||||||
|
hash tmp[2];
|
||||||
|
|
||||||
|
for (size_t i = 0, n = proof.size(); i < n; ++i) {
|
||||||
|
if (proof[i].first) {
|
||||||
|
tmp[0] = proof[i].second;
|
||||||
|
tmp[1] = h;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
tmp[0] = h;
|
||||||
|
tmp[1] = proof[i].second;
|
||||||
|
}
|
||||||
|
keccak(tmp[0].h, HASH_SIZE * 2, h.h);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (h == root);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace p2pool
|
} // namespace p2pool
|
||||||
|
|
|
@ -21,5 +21,7 @@ namespace p2pool {
|
||||||
|
|
||||||
void merkle_hash(const std::vector<hash>& hashes, hash& root);
|
void merkle_hash(const std::vector<hash>& hashes, hash& root);
|
||||||
void merkle_hash_full_tree(const std::vector<hash>& hashes, std::vector<std::vector<hash>>& tree);
|
void merkle_hash_full_tree(const std::vector<hash>& hashes, std::vector<std::vector<hash>>& tree);
|
||||||
|
bool get_merkle_proof(const std::vector<std::vector<hash>>& tree, const hash& h, std::vector<std::pair<bool, hash>>& proof);
|
||||||
|
bool verify_merkle_proof(hash h, const std::vector<std::pair<bool, hash>>& proof, const hash& root);
|
||||||
|
|
||||||
} // namespace p2pool
|
} // namespace p2pool
|
||||||
|
|
|
@ -69,6 +69,12 @@ TEST(merkle, root_hash)
|
||||||
ASSERT_EQ(tmp, tree[i][j]);
|
ASSERT_EQ(tmp, tree[i][j]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (const hash& h : hashes) {
|
||||||
|
std::vector<std::pair<bool, hash>> proof;
|
||||||
|
ASSERT_TRUE(get_merkle_proof(tree, h, proof));
|
||||||
|
ASSERT_TRUE(verify_merkle_proof(h, proof, root));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// 1 leaf
|
// 1 leaf
|
||||||
|
|
Loading…
Reference in a new issue