mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-26 14:05:51 +00:00
187 lines
4.3 KiB
C++
187 lines
4.3 KiB
C++
|
|
#include <stack>
|
|
|
|
#include "SHAMap.h"
|
|
|
|
void SHAMap::getMissingNodes(std::vector<SHAMapNode>& nodeIDs, std::vector<uint256>& hashes, int max)
|
|
{
|
|
boost::recursive_mutex::scoped_lock sl(mLock);
|
|
|
|
if(root->isFullBelow()) return;
|
|
|
|
std::stack<SHAMapInnerNode::pointer> stack;
|
|
stack.push(root);
|
|
while( (max>0) && (!stack.empty()) )
|
|
{
|
|
SHAMapInnerNode::pointer node=stack.top();
|
|
stack.pop();
|
|
bool all_leaves=true;
|
|
for(int i=0; i<32; i++)
|
|
{
|
|
if(!node->isEmptyBranch(i))
|
|
{
|
|
if(node->isChildLeaf())
|
|
{ // do we have this leaf node?
|
|
SHAMapLeafNode::pointer leaf=getLeaf(node->getChildNodeID(i), node->getChildHash(i), false);
|
|
if(!leaf)
|
|
{
|
|
if(max-->0)
|
|
{
|
|
nodeIDs.push_back(node->getChildNodeID(i));
|
|
hashes.push_back(node->getChildHash(i));
|
|
}
|
|
all_leaves=false;
|
|
}
|
|
}
|
|
else
|
|
{ // do we have this inner node?
|
|
SHAMapInnerNode::pointer desc=getInner(node->getChildNodeID(i), node->getChildHash(i), false);
|
|
if(!desc)
|
|
{
|
|
if(max-->0)
|
|
{
|
|
nodeIDs.push_back(node->getChildNodeID(i));
|
|
hashes.push_back(node->getChildHash(i));
|
|
}
|
|
all_leaves=false;
|
|
}
|
|
else if(!desc->isFullBelow()) stack.push(desc);
|
|
}
|
|
}
|
|
}
|
|
if(all_leaves) node->setFullBelow();
|
|
}
|
|
}
|
|
|
|
bool SHAMap::getNodeFat(const SHAMapNode& wanted, std::vector<SHAMapNode>& nodeIDs,
|
|
std::list<std::vector<unsigned char> >& rawNodes)
|
|
{
|
|
boost::recursive_mutex::scoped_lock sl(mLock);
|
|
if(wanted.isLeaf())
|
|
{ // no fat way to get a leaf
|
|
SHAMapLeafNode::pointer leaf=getLeafNode(wanted);
|
|
if(!leaf) return false;
|
|
nodeIDs.push_back(*leaf);
|
|
Serializer s;
|
|
leaf->addRaw(s);
|
|
rawNodes.push_back(s.peekData());
|
|
return true;
|
|
}
|
|
|
|
SHAMapInnerNode::pointer node=getInnerNode(wanted);
|
|
if(!node) return false;
|
|
|
|
bool ret=true;
|
|
for(int i=0; i<32; i++)
|
|
{
|
|
if(!node->isEmptyBranch(i))
|
|
{
|
|
if(node->isChildLeaf())
|
|
{
|
|
SHAMapLeafNode::pointer leaf=getLeaf(node->getChildNodeID(i), node->getChildHash(i), false);
|
|
if(!leaf) ret=false;
|
|
else
|
|
{
|
|
nodeIDs.push_back(*leaf);
|
|
Serializer s;
|
|
leaf->addRaw(s);
|
|
rawNodes.push_back(s.peekData());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SHAMapInnerNode::pointer ino=getInner(node->getChildNodeID(i), node->getChildHash(i), false);
|
|
if(!ino) ret=false;
|
|
else
|
|
{
|
|
nodeIDs.push_back(*ino);
|
|
Serializer s;
|
|
ino->addRaw(s);
|
|
rawNodes.push_back(s.peekData());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
bool SHAMap::addKnownNode(const SHAMapNode& node, const std::vector<unsigned char>& rawNode)
|
|
{
|
|
// WRITEME
|
|
return true;
|
|
}
|
|
|
|
bool SHAMap::deepCompare(SHAMap& other)
|
|
{ // Intended for debug/test only
|
|
std::stack<SHAMapInnerNode::pointer> stack;
|
|
SHAMapInnerNode::pointer node=root;
|
|
|
|
while(node)
|
|
{
|
|
SHAMapInnerNode::pointer node=stack.top();
|
|
stack.pop();
|
|
|
|
SHAMapInnerNode::pointer otherNode;
|
|
if(node->isRoot()) otherNode=other.root;
|
|
else otherNode=other.getInner(*node, node->getNodeHash(), false);
|
|
|
|
if(!otherNode)
|
|
{
|
|
std::cerr << "unable to fetch node" << std::endl;
|
|
return false;
|
|
}
|
|
else if(otherNode->getNodeHash()!=node->getNodeHash())
|
|
{
|
|
std::cerr << "node hash mismatch" << std::endl;
|
|
return false;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
std::cerr << "Comparing inner nodes " << node->getString() << std::endl;
|
|
#endif
|
|
|
|
for(int i=0; i<32; i++)
|
|
{
|
|
if(node->isEmptyBranch(i))
|
|
{
|
|
if(!otherNode->isEmptyBranch(i))
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
if(node->isChildLeaf())
|
|
{ //
|
|
SHAMapLeafNode::pointer leaf=getLeaf(node->getChildNodeID(i), node->getChildHash(i), false);
|
|
if(!leaf)
|
|
{
|
|
std::cerr << "unable to fetch leaf" << std::endl;
|
|
return false;
|
|
}
|
|
SHAMapLeafNode::pointer otherLeaf=other.getLeaf(*leaf, leaf->getNodeHash(), false);
|
|
if(!otherLeaf)
|
|
{
|
|
std::cerr << "unable to fetch other leaf" << std::endl;
|
|
return false;
|
|
}
|
|
if(leaf->getNodeHash()!=otherLeaf->getNodeHash())
|
|
{
|
|
std::cerr << "leaf hash mismatch" << std::endl;
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{ // do we have this inner node?
|
|
SHAMapInnerNode::pointer next=getInner(node->getChildNodeID(i), node->getChildHash(i), false);
|
|
if(!next)
|
|
{
|
|
std::cerr << "unable to fetch inner node" << std::endl;
|
|
return false;
|
|
}
|
|
stack.push(next);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|