/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ #ifndef _HLLSKETCHIMPLFACTORY_HPP_ #define _HLLSKETCHIMPLFACTORY_HPP_ #include #include "HllUtil.hpp" #include "HllSketchImpl.hpp" #include "CouponList.hpp" #include "CouponHashSet.hpp" #include "HllArray.hpp" #include "Hll4Array.hpp" #include "Hll6Array.hpp" #include "Hll8Array.hpp" namespace datasketches { template class HllSketchImplFactory final { public: static HllSketchImpl* deserialize(std::istream& os, const A& allocator); static HllSketchImpl* deserialize(const void* bytes, size_t len, const A& allocator); static CouponHashSet* promoteListToSet(const CouponList& list); static HllArray* promoteListOrSetToHll(const CouponList& list); static HllArray* newHll(uint8_t lgConfigK, target_hll_type tgtHllType, bool startFullSize, const A& allocator); // resets the input impl, deleting the input pointer and returning a new pointer static HllSketchImpl* reset(HllSketchImpl* impl, bool startFullSize); static Hll4Array* convertToHll4(const HllArray& srcHllArr); static Hll6Array* convertToHll6(const HllArray& srcHllArr); static Hll8Array* convertToHll8(const HllArray& srcHllArr); }; template CouponHashSet* HllSketchImplFactory::promoteListToSet(const CouponList& list) { using ChsAlloc = typename std::allocator_traits::template rebind_alloc>; CouponHashSet* chSet = new (ChsAlloc(list.getAllocator()).allocate(1)) CouponHashSet(list.getLgConfigK(), list.getTgtHllType(), list.getAllocator()); for (const auto coupon: list) { chSet->couponUpdate(coupon); } return chSet; } template HllArray* HllSketchImplFactory::promoteListOrSetToHll(const CouponList& src) { HllArray* tgtHllArr = HllSketchImplFactory::newHll(src.getLgConfigK(), src.getTgtHllType(), false, src.getAllocator()); tgtHllArr->putKxQ0(1 << src.getLgConfigK()); for (const auto coupon: src) { tgtHllArr->couponUpdate(coupon); } tgtHllArr->putHipAccum(src.getEstimate()); tgtHllArr->putOutOfOrderFlag(false); return tgtHllArr; } template HllSketchImpl* HllSketchImplFactory::deserialize(std::istream& is, const A& allocator) { // we'll hand off the sketch based on PreInts so we don't need // to move the stream pointer back and forth -- perhaps somewhat fragile? const uint8_t preInts = static_cast(is.peek()); if (preInts == hll_constants::HLL_PREINTS) { return HllArray::newHll(is, allocator); } else if (preInts == hll_constants::HASH_SET_PREINTS) { return CouponHashSet::newSet(is, allocator); } else if (preInts == hll_constants::LIST_PREINTS) { return CouponList::newList(is, allocator); } else { throw std::invalid_argument("Attempt to deserialize unknown object type"); } } template HllSketchImpl* HllSketchImplFactory::deserialize(const void* bytes, size_t len, const A& allocator) { // read current mode directly const uint8_t preInts = static_cast(bytes)[0]; if (preInts == hll_constants::HLL_PREINTS) { return HllArray::newHll(bytes, len, allocator); } else if (preInts == hll_constants::HASH_SET_PREINTS) { return CouponHashSet::newSet(bytes, len, allocator); } else if (preInts == hll_constants::LIST_PREINTS) { return CouponList::newList(bytes, len, allocator); } else { throw std::invalid_argument("Attempt to deserialize unknown object type"); } } template HllArray* HllSketchImplFactory::newHll(uint8_t lgConfigK, target_hll_type tgtHllType, bool startFullSize, const A& allocator) { switch (tgtHllType) { case HLL_8: using Hll8Alloc = typename std::allocator_traits::template rebind_alloc>; return new (Hll8Alloc(allocator).allocate(1)) Hll8Array(lgConfigK, startFullSize, allocator); case HLL_6: using Hll6Alloc = typename std::allocator_traits::template rebind_alloc>; return new (Hll6Alloc(allocator).allocate(1)) Hll6Array(lgConfigK, startFullSize, allocator); case HLL_4: using Hll4Alloc = typename std::allocator_traits::template rebind_alloc>; return new (Hll4Alloc(allocator).allocate(1)) Hll4Array(lgConfigK, startFullSize, allocator); } throw std::logic_error("Invalid target_hll_type"); } template HllSketchImpl* HllSketchImplFactory::reset(HllSketchImpl* impl, bool startFullSize) { if (startFullSize) { HllArray* hll = newHll(impl->getLgConfigK(), impl->getTgtHllType(), startFullSize, impl->getAllocator()); impl->get_deleter()(impl); return hll; } else { using ClAlloc = typename std::allocator_traits::template rebind_alloc>; CouponList* cl = new (ClAlloc(impl->getAllocator()).allocate(1)) CouponList(impl->getLgConfigK(), impl->getTgtHllType(), hll_mode::LIST, impl->getAllocator()); impl->get_deleter()(impl); return cl; } } template Hll4Array* HllSketchImplFactory::convertToHll4(const HllArray& srcHllArr) { using Hll4Alloc = typename std::allocator_traits::template rebind_alloc>; return new (Hll4Alloc(srcHllArr.getAllocator()).allocate(1)) Hll4Array(srcHllArr); } template Hll6Array* HllSketchImplFactory::convertToHll6(const HllArray& srcHllArr) { using Hll6Alloc = typename std::allocator_traits::template rebind_alloc>; return new (Hll6Alloc(srcHllArr.getAllocator()).allocate(1)) Hll6Array(srcHllArr); } template Hll8Array* HllSketchImplFactory::convertToHll8(const HllArray& srcHllArr) { using Hll8Alloc = typename std::allocator_traits::template rebind_alloc>; return new (Hll8Alloc(srcHllArr.getAllocator()).allocate(1)) Hll8Array(srcHllArr); } } #endif /* _HLLSKETCHIMPLFACTORY_HPP_ */