mirror of https://github.com/PurpleI2P/i2pd.git
I2P: End-to-End encrypted and anonymous Internet
https://i2pd.website/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
77 lines
1.5 KiB
77 lines
1.5 KiB
/* |
|
* Copyright (c) 2013-2020, The PurpleI2P Project |
|
* |
|
* This file is part of Purple i2pd project and licensed under BSD3 |
|
* |
|
* See full license text in LICENSE file at top of project tree |
|
*/ |
|
|
|
#include "BloomFilter.h" |
|
#include "I2PEndian.h" |
|
#include <array> |
|
#include <openssl/sha.h> |
|
|
|
namespace i2p |
|
{ |
|
namespace util |
|
{ |
|
|
|
/** @brief decaying bloom filter implementation */ |
|
class DecayingBloomFilter : public IBloomFilter |
|
{ |
|
public: |
|
|
|
DecayingBloomFilter(const std::size_t size) |
|
{ |
|
m_Size = size; |
|
m_Data = new uint8_t[size]; |
|
} |
|
|
|
/** @brief implements IBloomFilter::~IBloomFilter */ |
|
~DecayingBloomFilter() |
|
{ |
|
delete [] m_Data; |
|
} |
|
|
|
/** @brief implements IBloomFilter::Add */ |
|
bool Add(const uint8_t * data, std::size_t len) |
|
{ |
|
std::size_t idx; |
|
uint8_t mask; |
|
Get(data, len, idx, mask); |
|
if(m_Data[idx] & mask) return false; // filter hit |
|
m_Data[idx] |= mask; |
|
return true; |
|
} |
|
|
|
/** @brief implements IBloomFilter::Decay */ |
|
void Decay() |
|
{ |
|
// reset bloom filter buffer |
|
memset(m_Data, 0, m_Size); |
|
} |
|
|
|
private: |
|
/** @brief get bit index for for data */ |
|
void Get(const uint8_t * data, std::size_t len, std::size_t & idx, uint8_t & bm) |
|
{ |
|
bm = 1; |
|
uint8_t digest[32]; |
|
// TODO: use blake2 because it's faster |
|
SHA256(data, len, digest); |
|
uint64_t i = buf64toh(digest); |
|
idx = i % m_Size; |
|
bm <<= (i % 8); |
|
} |
|
|
|
uint8_t * m_Data; |
|
std::size_t m_Size; |
|
}; |
|
|
|
|
|
BloomFilterPtr BloomFilter(std::size_t capacity) |
|
{ |
|
return std::make_shared<DecayingBloomFilter>(capacity); |
|
} |
|
} |
|
}
|
|
|