TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
hash_tables.hpp
1#ifndef LIBTEDDY_DETAILS_HASH_TABLES_HPP
2#define LIBTEDDY_DETAILS_HASH_TABLES_HPP
3
4#include <libteddy/details/config.hpp>
5#include <libteddy/details/debug.hpp>
6#include <libteddy/details/node.hpp>
7#include <libteddy/details/tools.hpp>
8
9#include <cstddef>
10#include <cstdlib>
11#include <cstring>
12
13namespace teddy
14{
19{
20public:
21 static auto get_gte_capacity (int64 capacity) -> int64;
22
23private:
24 static constexpr int64 Capacities[] {
25 307, 617, 1'237, 2'477, 4'957,
26 9'923, 19'853, 39'709, 79'423, 158'849,
27 317'701, 635'413, 1'270'849, 2'541'701, 5'083'423,
28 10'166'857, 20'333'759, 40'667'527, 81'335'063, 162'670'129,
29 325'340'273, 650'680'571, 1'301'361'143, 2'602'722'289};
30};
31
35template<class Data, class Degree>
37{
38public:
40
41public:
42 unique_table_iterator(node_t** firstBucket, node_t** lastBucket);
43 unique_table_iterator(node_t** bucket, node_t** lastBucket, node_t* node);
44
45public:
46 auto operator++ () -> unique_table_iterator&;
47 auto operator++ (int) -> unique_table_iterator;
48 auto operator* () const -> node_t*;
49 auto operator== (unique_table_iterator const& other) const -> bool;
50 auto operator!= (unique_table_iterator const& other) const -> bool;
51 auto get_bucket () const -> node_t**;
52
53private:
58 auto move_to_next_bucket () -> node_t*;
59
60private:
61 node_t** bucket_;
62 node_t** lastBucket_;
63 node_t* node_;
64};
65
69template<class Data, class Degree>
71{
72public:
74 using son_container = typename node_t::son_container;
76
77public:
79 {
80 node_t* node_;
81 std::size_t hash_;
82 };
83
84public:
90 unique_table(int64 capacity, int32 domain);
91
96
100 unique_table(unique_table&& other) noexcept;
101
106
107 auto operator= (unique_table const&) = delete;
108 auto operator= (unique_table&&) = delete;
109
110public:
117 [[nodiscard]] auto find (son_container const& sons) const -> result_of_find;
118
125 auto merge (unique_table other) -> void;
126
132 auto insert (node_t* node, std::size_t hash) -> void;
133
139 auto erase (iterator nodeIt) -> iterator;
140
146 auto erase (node_t* node) -> iterator;
147
151 auto adjust_capacity () -> void;
152
156 [[nodiscard]] auto get_size () const -> int64;
157
161 auto clear () -> void;
162
166 [[nodiscard]] auto begin () -> iterator;
167
171 [[nodiscard]] auto end () -> iterator;
172
176 [[nodiscard]] auto begin () const -> iterator;
177
181 [[nodiscard]] auto end () const -> iterator;
182
183private:
189 auto rehash (int64 newCapacity) -> void;
190
194 [[nodiscard]] auto get_load_factor () const -> double;
195
202 auto insert_impl (node_t* node, std::size_t hash) -> node_t*;
203
210 auto erase_impl (node_t** bucket, node_t* node) -> iterator;
211
217 [[nodiscard]] auto node_hash (son_container const& sons) const
218 -> std::size_t;
219
227 [[nodiscard]] auto node_equals (node_t* node, son_container const& sons)
228 const -> bool;
229
233 [[nodiscard]] auto callocate_buckets (int64 count) -> node_t**;
234
238 [[nodiscard]] auto mallocate_buckets (int64 count) -> node_t**;
239
240private:
241 static constexpr double LOAD_THRESHOLD = 0.75;
242
243private:
244 int32 domain_;
245 int64 size_;
246 int64 capacity_;
247 node_t** buckets_;
248};
249
253template<class Data, class Degree>
255{
256public:
258
259public:
261 {
262 int32 opId_;
263 node_t* lhs_;
264 node_t* rhs_;
265 node_t* result_;
266 };
267
268public:
269 apply_cache(int64 capacity);
270 apply_cache(apply_cache&& other) noexcept;
271 ~apply_cache();
272
273 apply_cache(apply_cache const&) = delete;
274 auto operator= (apply_cache const&) = delete;
275 auto operator= (apply_cache&&) = delete;
276
277public:
285 auto find (int32 opId, node_t* lhs, node_t* rhs) -> node_t*;
286
295 auto put (int32 opId, node_t* result, node_t* lhs, node_t* rhs) -> void;
296
302 auto grow_capacity (int64 aproxCapacity) -> void;
303
307 auto remove_unused () -> void;
308
312 auto clear () -> void;
313
314private:
318 [[nodiscard]] auto get_load_factor () const -> double;
319
325 auto rehash (int64 newCapacity) -> void;
326
330 [[nodiscard]] static auto callocate_entries (int64 count) -> cache_entry*;
331
332private:
333 int64 size_;
334 int64 capacity_;
335 cache_entry* entries_;
336};
337
338// table_base definitions:
339
340inline auto table_base::get_gte_capacity(int64 const desiredCapacity) -> int64
341{
342 for (int64 const tableCapacity : Capacities)
343 {
344 if (tableCapacity > desiredCapacity)
345 {
346 return tableCapacity;
347 }
348 }
349 return Capacities[std::size(Capacities) - 1];
350}
351
352// unique_table_iterator definitions:
353
354template<class Data, class Degree>
356 node_t** const firstBucket,
357 node_t** const lastBucket
358) :
359 bucket_(firstBucket),
360 lastBucket_(lastBucket),
361 node_(this->move_to_next_bucket())
362{
363}
364
365template<class Data, class Degree>
366unique_table_iterator<Data, Degree>::unique_table_iterator(
367 node_t** const bucket,
368 node_t** const lastBucket,
369 node_t* const node
370) :
371 bucket_(bucket),
372 lastBucket_(lastBucket),
373 node_(node)
374{
375}
376
377template<class Data, class Degree>
378auto unique_table_iterator<Data, Degree>::operator++ ()
379 -> unique_table_iterator&
380{
381 node_ = node_->get_next();
382 if (not node_)
383 {
384 ++bucket_;
385 node_ = this->move_to_next_bucket();
386 }
387 return *this;
388}
389
390template<class Data, class Degree>
391auto unique_table_iterator<Data, Degree>::operator++ (int)
392 -> unique_table_iterator
393{
394 auto const tmp = *this;
395 ++(*this);
396 return tmp;
397}
398
399template<class Data, class Degree>
400auto unique_table_iterator<Data, Degree>::operator* () const -> node_t*
401{
402 return node_;
403}
404
405template<class Data, class Degree>
406auto unique_table_iterator<Data, Degree>::operator== (
407 unique_table_iterator const& other
408) const -> bool
409{
410 return bucket_ == other.bucket_ && node_ == other.node_;
411}
412
413template<class Data, class Degree>
414auto unique_table_iterator<Data, Degree>::operator!= (
415 unique_table_iterator const& other
416) const -> bool
417{
418 return not (*this == other);
419}
420
421template<class Data, class Degree>
422auto unique_table_iterator<Data, Degree>::get_bucket() const -> node_t**
423{
424 return bucket_;
425}
426
427template<class Data, class Degree>
428auto unique_table_iterator<Data, Degree>::move_to_next_bucket() -> node_t*
429{
430 while (bucket_ != lastBucket_ && not *bucket_)
431 {
432 ++bucket_;
433 }
434 return bucket_ != lastBucket_ ? *bucket_ : nullptr;
435}
436
437// unique_table definitions:
438
439template<class Data, class Degree>
441 int64 const capacity,
442 int32 const domain
443) :
444 domain_(domain),
445 size_(0),
446 capacity_(table_base::get_gte_capacity(capacity)),
447 buckets_(callocate_buckets(capacity_))
448{
449}
450
451template<class Data, class Degree>
453 domain_(other.domain_),
454 size_(other.size_),
455 capacity_(other.capacity_),
456 buckets_(mallocate_buckets(other.capacity_))
457{
458 std::memcpy(
459 buckets_,
460 other.buckets_,
461 static_cast<std::size_t>(capacity_) * sizeof(node_t*)
462 );
463}
464
465template<class Data, class Degree>
467 domain_(other.domain_),
468 size_(utils::exchange(other.size_, 0)),
469 capacity_(other.capacity_),
470 buckets_(utils::exchange(other.buckets_, nullptr))
471{
472}
473
474template<class Data, class Degree>
476{
477 std::free(buckets_);
478}
479
480template<class Data, class Degree>
481auto unique_table<Data, Degree>::find(son_container const& sons) const
483{
484 std::size_t const hash = this->node_hash(sons);
485 int64 const index
486 = static_cast<int64>(hash % static_cast<std::size_t>(capacity_));
487 node_t* current = buckets_[index];
488 while (current)
489 {
490 if (this->node_equals(current, sons))
491 {
492 return {current, hash};
493 }
494 current = current->get_next();
495 }
496 return {nullptr, hash};
497}
498
499template<class Data, class Degree>
501{
502 size_ += other.size_;
503 this->adjust_capacity();
504 auto it = other.begin();
505 auto const end = other.end();
506 while (it != end)
507 {
508 node_t* const otherNode = *it;
509 ++it;
510 otherNode->set_next(nullptr);
511 std::size_t const hash = this->node_hash(otherNode->get_sons());
512 this->insert_impl(otherNode, hash);
513 }
514}
515
516template<class Data, class Degree>
518 node_t* const node,
519 std::size_t const hash
520) -> void
521{
522 this->insert_impl(node, hash);
523 ++size_;
524}
525
526template<class Data, class Degree>
528{
529 node_t** const bucket = nodeIt.get_bucket();
530 node_t* const node = *nodeIt;
531 return this->erase_impl(bucket, node);
532}
533
534template<class Data, class Degree>
536{
537 std::size_t const hash = this->node_hash(node->get_sons());
538 int64 const index
539 = static_cast<int64>(hash % static_cast<std::size_t>(capacity_));
540 return this->erase_impl(buckets_ + index, node);
541}
542
543template<class Data, class Degree>
545{
546 int64 const aproxCapacity
547 = static_cast<int64>(static_cast<double>(size_) / LOAD_THRESHOLD);
548 int64 const newCapacity = table_base::get_gte_capacity(aproxCapacity);
549 if (newCapacity > capacity_)
550 {
551 this->rehash(newCapacity);
552 }
553}
554
555template<class Data, class Degree>
557{
558 return size_;
559}
560
561template<class Data, class Degree>
563{
564 size_ = 0;
565 std::memset(
566 buckets_,
567 0,
568 static_cast<std::size_t>(capacity_) * sizeof(node_t*)
569 );
570}
571
572template<class Data, class Degree>
574{
575 return iterator(buckets_, buckets_ + capacity_);
576}
577
578template<class Data, class Degree>
580{
581 return iterator(buckets_ + capacity_, buckets_ + capacity_);
582}
583
584template<class Data, class Degree>
586{
587 return iterator(buckets_, buckets_ + capacity_);
588}
589
590template<class Data, class Degree>
592{
593 return iterator(buckets_ + capacity_, buckets_ + capacity_);
594}
595
596template<class Data, class Degree>
597auto unique_table<Data, Degree>::rehash(int64 const newCapacity) -> void
598{
599#ifdef LIBTEDDY_VERBOSE
600 debug::out(
601 " unique_table::rehash\tload before ",
602 this->get_load_factor(),
603 " capacity is ",
604 capacity_,
605 " should be ",
606 newCapacity
607 );
608#endif
609
610 node_t** const oldBuckets = buckets_;
611 int64 const oldCapacity = capacity_;
612 buckets_ = callocate_buckets(newCapacity);
613 capacity_ = newCapacity;
614 for (int64 i = 0; i < oldCapacity; ++i)
615 {
616 node_t* node = oldBuckets[i];
617 while (node)
618 {
619 node_t* const next = node->get_next();
620 std::size_t const hash = this->node_hash(node->get_sons());
621 node->set_next(nullptr);
622 this->insert_impl(node, hash);
623 node = next;
624 }
625 };
626
627#ifdef LIBTEDDY_VERBOSE
628 debug::out(", load after ", this->get_load_factor(), "\n");
629#endif
630}
631
632template<class Data, class Degree>
633auto unique_table<Data, Degree>::get_load_factor() const -> double
634{
635 return static_cast<double>(size_) / static_cast<double>(capacity_);
636}
637
638template<class Data, class Degree>
639auto unique_table<Data, Degree>::insert_impl(
640 node_t* const node,
641 std::size_t const hash
642) -> node_t*
643{
644 std::size_t const index = hash % static_cast<std::size_t>(capacity_);
645 node_t* const bucket = buckets_[index];
646 if (bucket)
647 {
648 node->set_next(bucket);
649 }
650 buckets_[index] = node;
651 return node;
652}
653
654template<class Data, class Degree>
655auto unique_table<Data, Degree>::erase_impl(
656 node_t** const bucket,
657 node_t* const node
658) -> iterator
659{
660 iterator retIt(bucket, buckets_ + capacity_, node);
661 ++retIt;
662 --size_;
663
664 if (*bucket == node)
665 {
666 *bucket = node->get_next();
667 node->set_next(nullptr);
668 return retIt;
669 }
670
671 node_t* prev = *bucket;
672 while (prev->get_next() != node)
673 {
674 prev = prev->get_next();
675 }
676 prev->set_next(node->get_next());
677 node->set_next(nullptr);
678 return retIt;
679}
680
681template<class Data, class Degree>
682auto unique_table<Data, Degree>::node_hash(son_container const& sons) const
683 -> std::size_t
684{
685 std::size_t result = 0;
686 for (int32 k = 0; k < domain_; ++k)
687 {
688 utils::add_hash(result, sons[as_uindex(k)]);
689 }
690 return result;
691}
692
693template<class Data, class Degree>
694auto unique_table<Data, Degree>::node_equals(
695 node_t* const node,
696 son_container const& sons
697) const -> bool
698{
699 for (int32 k = 0; k < domain_; ++k)
700 {
701 if (node->get_son(k) != sons[as_uindex(k)])
702 {
703 return false;
704 }
705 }
706 return true;
707}
708
709template<class Data, class Degree>
710auto unique_table<Data, Degree>::callocate_buckets(int64 const count)
711 -> node_t**
712{
713 return static_cast<node_t**>(
714 std::calloc(static_cast<std::size_t>(count), sizeof(node_t*))
715 );
716}
717
718template<class Data, class Degree>
719auto unique_table<Data, Degree>::mallocate_buckets(int64 const count)
720 -> node_t**
721{
722 return static_cast<node_t**>(
723 std::malloc(static_cast<std::size_t>(count) * sizeof(node_t*))
724 );
725}
726
727// apply_cache definitions:
728
729template<class Data, class Degree>
730apply_cache<Data, Degree>::apply_cache(int64 const capacity) :
731 size_(0),
732 capacity_(table_base::get_gte_capacity(capacity)),
733 entries_(callocate_entries(capacity_))
734{
735}
736
737template<class Data, class Degree>
738apply_cache<Data, Degree>::apply_cache(apply_cache&& other) noexcept :
739 size_(utils::exchange(other.size_, 0)),
740 capacity_(other.capacity_),
741 entries_(utils::exchange(other.entries_, nullptr))
742{
743}
744
745template<class Data, class Degree>
746apply_cache<Data, Degree>::~apply_cache()
747{
748 std::free(entries_);
749}
750
751template<class Data, class Degree>
753 int32 const opId,
754 node_t* const lhs,
755 node_t* const rhs
756) -> node_t*
757{
758 std::size_t const hash = utils::pack_hash(opId, lhs, rhs);
759 std::size_t const index = hash % static_cast<std::size_t>(capacity_);
760 cache_entry& entry = entries_[index];
761 bool const matches
762 = entry.opId_ == opId && entry.lhs_ == lhs && entry.rhs_ == rhs;
763 return matches ? entry.result_ : nullptr;
764}
765
766template<class Data, class Degree>
768 int32 const opId,
769 node_t* const result,
770 node_t* const lhs,
771 node_t* const rhs
772) -> void
773{
774 std::size_t const hash = utils::pack_hash(opId, lhs, rhs);
775 std::size_t const index = hash % static_cast<std::size_t>(capacity_);
776 cache_entry& entry = entries_[index];
777 if (not entry.result_)
778 {
779 ++size_;
780 }
781 entry.opId_ = opId;
782 entry.lhs_ = lhs;
783 entry.rhs_ = rhs;
784 entry.result_ = result;
785}
786
787template<class Data, class Degree>
788auto apply_cache<Data, Degree>::grow_capacity(int64 const aproxCapacity) -> void
789{
790 int64 const newCapacity = table_base::get_gte_capacity(aproxCapacity);
791 if (newCapacity > capacity_)
792 {
793 this->rehash(newCapacity);
794 }
795}
796
797template<class Data, class Degree>
799{
800 for (int64 i = 0; i < capacity_; ++i)
801 {
802 cache_entry& entry = entries_[i];
803 if (entry.result_)
804 {
805 bool const isUsed = entry.lhs_->is_used() && entry.rhs_->is_used()
806 && entry.result_->is_used();
807 if (not isUsed)
808 {
809 entry = cache_entry {};
810 --size_;
811 }
812 }
813 }
814}
815
816template<class Data, class Degree>
818{
819 size_ = 0;
820 std::memset(
821 entries_,
822 0,
823 static_cast<std::size_t>(capacity_) * sizeof(cache_entry)
824 );
825}
826
827template<class Data, class Degree>
829{
830 return static_cast<double>(size_) / static_cast<double>(capacity_);
831}
832
833template<class Data, class Degree>
834auto apply_cache<Data, Degree>::rehash(int64 const newCapacity) -> void
835{
836#ifdef LIBTEDDY_VERBOSE
837 debug::out(
838 "apply_cache::rehash\tload is ",
839 this->get_load_factor(),
840 ", capacity is ",
841 capacity_,
842 " should be ",
843 newCapacity
844 );
845#endif
846
847 cache_entry* const oldEntries = entries_;
848 int64 const oldCapacity = capacity_;
849 entries_ = callocate_entries(newCapacity);
850 capacity_ = newCapacity;
851 size_ = 0;
852 for (int64 i = 0; i < oldCapacity; ++i)
853 {
854 cache_entry const& entry = oldEntries[i];
855 if (entry.result_)
856 {
857 this->put(entry.opId_, entry.result_, entry.lhs_, entry.rhs_);
858 }
859 }
860
861#ifdef LIBTEDDY_VERBOSE
862 debug::out(" new load is ", this->get_load_factor(), "\n");
863#endif
864}
865
866template<class Data, class Degree>
867auto apply_cache<Data, Degree>::callocate_entries(int64 const count)
868 -> cache_entry*
869{
870 return static_cast<cache_entry*>(
871 std::calloc(static_cast<std::size_t>(count), sizeof(cache_entry))
872 );
873}
874
875} // namespace teddy
876
877#endif
Cache for the apply opertaion.
Definition hash_tables.hpp:255
auto find(int32 opId, node_t *lhs, node_t *rhs) -> node_t *
Looks up result of an operation.
Definition hash_tables.hpp:752
auto clear() -> void
Clears all entries.
Definition hash_tables.hpp:817
auto grow_capacity(int64 aproxCapacity) -> void
Increases the capacity so that it is close to aproxCapacity Never lowers the capacity!
Definition hash_tables.hpp:788
auto remove_unused() -> void
Removes entries pointing to unused nodes.
Definition hash_tables.hpp:798
auto put(int32 opId, node_t *result, node_t *lhs, node_t *rhs) -> void
Puts the result into the cache possibly overwriting old value.
Definition hash_tables.hpp:767
Definition hash_tables.hpp:261
Definition node.hpp:91
Base class for hash tables.
Definition hash_tables.hpp:19
Iterator for the unique table.
Definition hash_tables.hpp:37
Table of unique nodes.
Definition hash_tables.hpp:71
~unique_table()
Destructor.
Definition hash_tables.hpp:475
auto clear() -> void
Clears the table.
Definition hash_tables.hpp:562
auto insert(node_t *node, std::size_t hash) -> void
Inserts node using pre-computed hash.
Definition hash_tables.hpp:517
auto adjust_capacity() -> void
Adjusts capacity of the table (number of buckets)
Definition hash_tables.hpp:544
auto erase(iterator nodeIt) -> iterator
Erases node pointed to by it.
Definition hash_tables.hpp:527
auto merge(unique_table other) -> void
Adds all nodes from other into this table Adjusts capacity if necessary.
Definition hash_tables.hpp:500
auto find(son_container const &sons) const -> result_of_find
Tries to find an internal node.
Definition hash_tables.hpp:481
unique_table(int64 capacity, int32 domain)
Initializes empty table.
Definition hash_tables.hpp:440
auto end() -> iterator
Definition hash_tables.hpp:579
auto get_size() const -> int64
Definition hash_tables.hpp:556
auto begin() -> iterator
Definition hash_tables.hpp:573
Definition hash_tables.hpp:79