#ifndef BloomFilter_h
#define BloomFilter_h
#include "wtf/Compiler.h"
#include "wtf/text/AtomicString.h"
namespace WTF {
template <unsigned keyBits>
class BloomFilter {
public:
COMPILE_ASSERT(keyBits <= 16, bloom_filter_key_size);
static const size_t tableSize = 1 << keyBits;
static const unsigned keyMask = (1 << keyBits) - 1;
static uint8_t maximumCount() { return std::numeric_limits<uint8_t>::max(); }
BloomFilter() { clear(); }
void add(unsigned hash);
void remove(unsigned hash);
bool mayContain(unsigned hash) const { return firstSlot(hash) && secondSlot(hash); }
void clear();
void add(const AtomicString& string) { add(string.impl()->existingHash()); }
void add(const String& string) { add(string.impl()->hash()); }
void remove(const AtomicString& string) { remove(string.impl()->existingHash()); }
void remove(const String& string) { remove(string.impl()->hash()); }
bool mayContain(const AtomicString& string) const { return mayContain(string.impl()->existingHash()); }
bool mayContain(const String& string) const { return mayContain(string.impl()->hash()); }
#if ASSERT_ENABLED
bool likelyEmpty() const;
bool isClear() const;
#endif
private:
uint8_t& firstSlot(unsigned hash) { return m_table[hash & keyMask]; }
uint8_t& secondSlot(unsigned hash) { return m_table[(hash >> 16) & keyMask]; }
const uint8_t& firstSlot(unsigned hash) const { return m_table[hash & keyMask]; }
const uint8_t& secondSlot(unsigned hash) const { return m_table[(hash >> 16) & keyMask]; }
uint8_t m_table[tableSize];
};
template <unsigned keyBits>
inline void BloomFilter<keyBits>::add(unsigned hash)
{
uint8_t& first = firstSlot(hash);
uint8_t& second = secondSlot(hash);
if (LIKELY(first < maximumCount()))
++first;
if (LIKELY(second < maximumCount()))
++second;
}
template <unsigned keyBits>
inline void BloomFilter<keyBits>::remove(unsigned hash)
{
uint8_t& first = firstSlot(hash);
uint8_t& second = secondSlot(hash);
ASSERT(first);
ASSERT(second);
if (LIKELY(first < maximumCount()))
--first;
if (LIKELY(second < maximumCount()))
--second;
}
template <unsigned keyBits>
inline void BloomFilter<keyBits>::clear()
{
memset(m_table, 0, tableSize);
}
#if ASSERT_ENABLED
template <unsigned keyBits>
bool BloomFilter<keyBits>::likelyEmpty() const
{
for (size_t n = 0; n < tableSize; ++n) {
if (m_table[n] && m_table[n] != maximumCount())
return false;
}
return true;
}
template <unsigned keyBits>
bool BloomFilter<keyBits>::isClear() const
{
for (size_t n = 0; n < tableSize; ++n) {
if (m_table[n])
return false;
}
return true;
}
#endif
}
using WTF::BloomFilter;
#endif