QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
Loading...
Searching...
No Matches
quickcpplib::_xxx::algorithm::hash::fast_hash Class Reference

Fast very collision resistant uint128 hash. Currently SpookyHash @ 0.3 cycles/byte. More...

#include "hash.hpp"

Public Types

using result_type = uint128
 The result type of the hash.
 

Public Member Functions

constexpr fast_hash (const uint128 &seed=0) noexcept
 Initialise the hash with an optional seed.
 
template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
void add (const T *message, size_t length) noexcept
 Hash input.
 
uint128 finalise () noexcept
 Finalise and return hash.
 

Static Public Member Functions

template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
static uint128 hash (const T *message, size_t length, const uint128 &_ret=0) noexcept
 Single shot hash of a sequence of bytes.
 

Detailed Description

Fast very collision resistant uint128 hash. Currently SpookyHash @ 0.3 cycles/byte.

Member Typedef Documentation

◆ result_type

Constructor & Destructor Documentation

◆ fast_hash()

constexpr quickcpplib::_xxx::algorithm::hash::fast_hash::fast_hash ( const uint128 seed = 0)
inlineconstexprnoexcept

Initialise the hash with an optional seed.

459 : m_data{0}
460 , m_state{seed.as_longlongs[0], seed.as_longlongs[1]}
461 , m_length(0)
462 , m_remainder(0)
463 {
464 }
uint64 m_data[2 *sc_numVars]
Definition hash.hpp:335

Member Function Documentation

◆ add()

template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
void quickcpplib::_xxx::algorithm::hash::fast_hash::add ( const T *  message,
size_t  length 
)
inlinenoexcept

Hash input.

470 {
471 using namespace fash_hash_detail;
472 uint64 h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11;
473 size_t newLength = length + m_remainder;
474 uint8 remainder;
475 union
476 {
477 const uint8 *p8;
478 uint64 *p64;
479 size_t i;
480 } u;
481 const uint64 *end;
482
483 // Is this message fragment too short? If it is, stuff it away.
484 if(newLength < sc_bufSize)
485 {
486 memcpy(&((uint8 *) m_data)[m_remainder], message, length);
487 m_length = length + m_length;
488 m_remainder = (uint8) newLength;
489 return;
490 }
491
492 // init the variables
493 if(m_length < sc_bufSize)
494 {
495 h0 = h3 = h6 = h9 = m_state[0];
496 h1 = h4 = h7 = h10 = m_state[1];
497 h2 = h5 = h8 = h11 = sc_const;
498 }
499 else
500 {
501 h0 = m_state[0];
502 h1 = m_state[1];
503 h2 = m_state[2];
504 h3 = m_state[3];
505 h4 = m_state[4];
506 h5 = m_state[5];
507 h6 = m_state[6];
508 h7 = m_state[7];
509 h8 = m_state[8];
510 h9 = m_state[9];
511 h10 = m_state[10];
512 h11 = m_state[11];
513 }
514 m_length = length + m_length;
515
516 // if we've got anything stuffed away, use it now
517 if(m_remainder)
518 {
519 uint8 prefix = sc_bufSize - m_remainder;
520 memcpy(&(((uint8 *) m_data)[m_remainder]), message, prefix);
521 u.p64 = m_data;
522 Mix(u.p64, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
523 Mix(&u.p64[sc_numVars], h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
524 u.p8 = ((const uint8 *) message) + prefix;
525 length -= prefix;
526 }
527 else
528 {
529 u.p8 = (const uint8 *) message;
530 }
531
532 // handle all whole blocks of sc_blockSize bytes
533 end = u.p64 + (length / sc_blockSize) * sc_numVars;
534 remainder = (uint8) (length - ((const uint8 *) end - u.p8));
535 if(ALLOW_UNALIGNED_READS || (u.i & 0x7) == 0)
536 {
537 while(u.p64 < end)
538 {
539 Mix(u.p64, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
540 u.p64 += sc_numVars;
541 }
542 }
543 else
544 {
545 while(u.p64 < end)
546 {
547 memcpy(m_data, u.p8, sc_blockSize);
548 Mix(m_data, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
549 u.p64 += sc_numVars;
550 }
551 }
552
553 // stuff away the last few bytes
554 m_remainder = remainder;
555 memcpy(m_data, end, remainder);
556
557 // stuff away the variables
558 m_state[0] = h0;
559 m_state[1] = h1;
560 m_state[2] = h2;
561 m_state[3] = h3;
562 m_state[4] = h4;
563 m_state[5] = h5;
564 m_state[6] = h6;
565 m_state[7] = h7;
566 m_state[8] = h8;
567 m_state[9] = h9;
568 m_state[10] = h10;
569 m_state[11] = h11;
570 }

◆ finalise()

uint128 quickcpplib::_xxx::algorithm::hash::fast_hash::finalise ( )
inlinenoexcept

Finalise and return hash.

574 {
575 using namespace fash_hash_detail;
576 uint128 ret;
577 // init the variables
578 if(m_length < sc_bufSize)
579 {
580 ret.as_longlongs[0] = m_state[0];
581 ret.as_longlongs[1] = m_state[1];
582 short_(ret, m_data_, m_length);
583 return ret;
584 }
585
586 const uint64 *data = (const uint64 *) m_data;
587 uint8 remainder = m_remainder;
588
589 uint64 h0 = m_state[0];
590 uint64 h1 = m_state[1];
591 uint64 h2 = m_state[2];
592 uint64 h3 = m_state[3];
593 uint64 h4 = m_state[4];
594 uint64 h5 = m_state[5];
595 uint64 h6 = m_state[6];
596 uint64 h7 = m_state[7];
597 uint64 h8 = m_state[8];
598 uint64 h9 = m_state[9];
599 uint64 h10 = m_state[10];
600 uint64 h11 = m_state[11];
601
602 if(remainder >= sc_blockSize)
603 {
604 // m_data can contain two blocks; handle any whole first block
605 Mix(data, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
606 data += sc_numVars;
607 remainder -= sc_blockSize;
608 }
609
610 // mix in the last partial block, and the length mod sc_blockSize
611 memset(&((uint8 *) data)[remainder], 0, (sc_blockSize - remainder));
612
613 ((uint8 *) data)[sc_blockSize - 1] = remainder;
614
615 // do some final mixing
616 End(data, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
617
618 ret.as_longlongs[0] = h0;
619 ret.as_longlongs[1] = h1;
620 return ret;
621 }
uint8 m_data_[2 *sc_numVars *8]
Definition hash.hpp:334
uint64_t as_longlongs[2]
Definition uint128.hpp:54

◆ hash()

template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
static uint128 quickcpplib::_xxx::algorithm::hash::fast_hash::hash ( const T *  message,
size_t  length,
const uint128 _ret = 0 
)
inlinestaticnoexcept

Single shot hash of a sequence of bytes.

627 {
628 uint128 ret(_ret);
629 using namespace fash_hash_detail;
630 if(length < sc_bufSize)
631 {
632 short_(ret, message, length);
633 return ret;
634 }
635
636 uint64 h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11;
637 uint64 buf[sc_numVars];
638 uint64 *end;
639 union
640 {
641 const uint8 *p8;
642 uint64 *p64;
643 size_t i;
644 } u;
645 size_t remainder;
646
647 h0 = h3 = h6 = h9 = ret.as_longlongs[0];
648 h1 = h4 = h7 = h10 = ret.as_longlongs[1];
649 h2 = h5 = h8 = h11 = sc_const;
650
651 u.p8 = (const uint8 *) message;
652 end = u.p64 + (length / sc_blockSize) * sc_numVars;
653
654 // handle all whole sc_blockSize blocks of bytes
655 if(ALLOW_UNALIGNED_READS || ((u.i & 0x7) == 0))
656 {
657 while(u.p64 < end)
658 {
659 Mix(u.p64, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
660 u.p64 += sc_numVars;
661 }
662 }
663 else
664 {
665 while(u.p64 < end)
666 {
667 memcpy(buf, u.p64, sc_blockSize);
668 Mix(buf, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
669 u.p64 += sc_numVars;
670 }
671 }
672
673 // handle the last partial block of sc_blockSize bytes
674 remainder = (length - ((const uint8 *) end - (const uint8 *) message));
675 memcpy(buf, end, remainder);
676 memory::cmemset(((uint8 *) buf) + remainder, (uint8) 0, sc_blockSize - remainder);
677 ((uint8 *) buf)[sc_blockSize - 1] = (uint8) remainder;
678
679 // do some final mixing
680 End(buf, h0, h1, h2, h3, h4, h5, h6, h7, h8, h9, h10, h11);
681
682 ret.as_longlongs[0] = h0;
683 ret.as_longlongs[1] = h1;
684 return ret;
685 }
constexpr T * cmemset(T *dst, T value, size_t num)
Definition memory.hpp:89

Member Data Documentation

◆ m_data_

uint8 quickcpplib::_xxx::algorithm::hash::fast_hash::m_data_[2 *sc_numVars *8]

◆ m_data

uint64 quickcpplib::_xxx::algorithm::hash::fast_hash::m_data[2 *sc_numVars]

The documentation for this class was generated from the following file: