Собрать и объяснить код (пример с сайта)

257
28 февраля 2018, 09:28

Имееться код hash-алгоритма. Так как его надо исследовать, необходимо его запустить (для начала). Но он не компилируется. Код был найден здесь.

#include <string.h>
#include <string>
#include <vector>
#include <stdexcept>
template <typename T, typename T_half, int size>
class double_uint
{
private:
    void fromstring(const std::string &s);
    std::string tostring(void) const;
public:
    T lo_, hi_;
    enum type_size
    {
        tsize = size,
        hsize = size / 2,
        qsize = size / 4
    };
    double_uint(const double_uint &u) : lo_(u.lo_), hi_(u.hi_) {}
    double_uint(const T_half &hi1, const T_half &hi2, const T_half &lo1, const T_half &lo2)
        : lo_(T(lo2) + (T(lo1) << qsize)),
          hi_(T(hi2) + (T(hi1) << qsize)) {}
    double_uint(const T &hi, const T &lo) : lo_(lo), hi_(hi) {}
    explicit double_uint(const T &lo) : lo_(lo), hi_() {}
    explicit double_uint(const std::string &s) {fromstring(s);};
    double_uint(void) : lo_(), hi_() {}
    /* Hack - basically we are a little-endian integer type */
    explicit double_uint(const unsigned char *c) {memcpy(this, c, sizeof(*this));}
    explicit double_uint(size_t len, const unsigned char *c)
    {
        if (len >= sizeof(*this))
        {
            memcpy(this, c, sizeof(*this));
        }
        else
        {
            memcpy(this, c, len);
            memset(reinterpret_cast<unsigned char *>(this) + len, 0, sizeof(*this) - len);
        }
    }
    friend std::ostream& operator<< (std::ostream &o, const double_uint &x)
    {
        return o << std::string(x);
    }
    double_uint& operator= (int x)
    {
        lo_ = x;
        hi_ = T();
        return *this;
    }
    double_uint& operator=(const double_uint &u)
    {
        lo_ = u.lo_;
        hi_ = u.hi_;
        return *this;
    }
    operator T() const {return lo_;}
    operator T() {return lo_;}
    operator const std::string() const {return tostring();}
    operator std::string() {return tostring();}
    double_uint operator+= (const double_uint &u)
    {
        T old = lo_;
        lo_ += u.lo_;
        if (lo_ < old) hi_++;
        hi_ += u.hi_;
        return *this;
    }
    double_uint operator-= (const double_uint &u)
    {
        T old = lo_;
        lo_ -= u.lo_;
        if (lo_ > old) hi_--;
        hi_ -= u.lo_;
        return *this;
    }
    double_uint operator+ (const double_uint &u) const
    {
        double_uint res(*this);
        return res += u;
    }
    double_uint operator- (const double_uint &u) const
    {
        double_uint res(*this);
        return res -= u;
    }
    double_uint operator++ (void)
    {
        return *this += double_uint(1);
    }
    double_uint operator++ (int)
    {
        double_uint res(*this);
        *this += double_uint(0, 1);
        return res;
    }
    double_uint operator- (void)
    {
        return double_uint(0) - *this;
    }
    double_uint operator+ (void) const
    {
        return *this;
    }
    double_uint operator<<= (int x)
    {
        x &= (tsize - 1);
        if (x >= hsize)
        {
            hi_ = lo_ << (x - hsize);
            lo_ = 0;
            return *this;
        }
        hi_ <<= x;
        hi_ += lo_ >> (hsize - x);
        lo_ <<= x;
        return *this;
    }
    double_uint operator>>= (int x)
    {
        x &= (tsize - 1);
        if (x >= hsize)
        {
            lo_ = hi_ >> (x - hsize);
            hi_ = 0;
            return *this;
        }
        lo_ >>= x;
        lo_ += hi_ << (hsize - x);
        hi_ >>= x;
        return *this;
    }
    double_uint operator<< (int x) const
    {
        double_uint res(*this);
        return res <<= x;
    }
    double_uint operator>> (int x) const
    {
        double_uint res(*this);
        return res >>= x;
    }
    static double_uint halfmul(T x, T y)
    {
        T_half x1, x2, y1, y2;
        x1 = x;
        x2 = x >> qsize;
        y1 = y;
        y2 = y >> qsize;
        T xx1(x1);
        T xx2(x2);
        T yy1(y1);
        T yy2(y2);
        double_uint p1, p2, p3, p4;
        p1.lo_ = xx1 * yy1;
        p2.lo_ = xx1 * yy2;
        p3.lo_ = xx2 * yy1;
        p4.lo_ = xx2 * yy2;
        p2 <<= qsize;
        p3 <<= qsize;
        p4 <<= hsize;
        return p1 + p2 + p3 + p4;
    }
    double_uint operator*= (const double_uint &u)
    {
        double_uint p1 = halfmul(lo_, u.lo_);
        double_uint p2 = halfmul(lo_, u.hi_);
        double_uint p3 = halfmul(hi_, u.lo_);
        p2 <<= hsize;
        p3 <<= hsize;
        return (*this = p1 + p2 + p3);
    }
    double_uint operator* (const double_uint &u) const
    {
        double_uint res(*this);
        return res *= u;
    }
    double_uint operator^= (const double_uint &u)
    {
        lo_ ^= u.lo_;
        hi_ ^= u.hi_;
        return *this;
    }
    double_uint operator^ (const double_uint &u) const
    {
        double_uint res(*this);
        return res ^= u;
    }
    /* Calculates (self * u, folded in half) xor u */
    double_uint fold(const double_uint &u) const
    {
        typedef double_uint<double_uint, T, tsize * 2> Tdouble;
        Tdouble x(Tdouble::halfmul(*this, u));
        return x.lo_ ^ x.hi_;
    }
};
template <class T, class T_half, int size>
    std::string double_uint<T, T_half, size>::tostring(void) const
{
    int num = size / 4;
    static const char hex_symb[] = "0123456789ABCDEF";
    /* Pad with zeros */
    std::string s(num, '0');
    double_uint<T, T_half, size> v(*this);
    for (int i = num - 1; i >= 0; i--)
    {
        s[i] = hex_symb[v.lo_ & 0xf];
        v >>= 4;
    }
    return s;
}
template <class T, class T_half, int size>
    void double_uint<T, T_half, size >::fromstring(const std::string &s)
{
    double_uint<T, T_half, size> temp;
    *this = 0;
    for (std::string::const_iterator i = s.begin(); i != s.end(); i++)
    {
        if ((*i >= '0') && (*i <= '9'))
        {
            temp = *i - '0';
        }
        else if ((*i >= 'A') && (*i <= 'F'))
        {
            temp = *i - 'A' + 10;
        }
        else if ((*i >= 'a') && (*i <= 'f'))
        {
            temp = *i - 'A' + 10;
        }
        else
        {
            throw std::runtime_error("Invalid hex character\n");
        }
        *this <<= 4;
        *this += temp;
    }
}
static std::string tostring(const __uint128_t &x)
{
    int num = 128 / 4;
    static const char hex_symb[] = "0123456789ABCDEF";
    /* Pad with zeros */
    std::string s(num, '0');
    __uint128_t v(x);
    for (int i = num - 1; i >= 0; i--)
    {
        s[i] = hex_symb[v & 0xf];
        v >>= 4;
    }
    return s;
}
std::ostream& operator<< (std::ostream &o, const __uint128_t &x)
{
    return o << tostring(x);
}
typedef double_uint<__uint128_t, unsigned long long, 256> u256;
void hash_step(const u256 &i1, const u256 &i2, u256 &o1, u256 &o2)
{
    /*
     * Cube roots of primes.
     * (Square roots don't work as well when folded with zero.
     *  The high part becomes nearly all one bits.)
     *
     * They are scaled so that the uppermost bit is set.
     * Then the lowest bit is also set, so that the constant is odd
     */
    /* cbrt(2) */
    static const u256 t1(0xa14517cc6b945711, 0x1eed5b8adf128686,
                         0x144788148b18fde0, 0x30c00661b7d16e9d);
    /* cbrt(3) */
    static const u256 t2(0xb89ba24891f7b2e6, 0xef3f8b62b71933e0,
                         0x50c4a6157ab766cc, 0xfa2ba143e9029653);
    /* cbrt(5) */
    static const u256 t3(0xdae07de7f6269d97, 0xed0ddb59924b141a,
                         0x0ae36687aa58c29f, 0xe8293af2918f493b);
    /* cbrt(7) */
    static const u256 t4(0xf4daedd2c0c4edde, 0x50536bb743875dac,
                         0xfdb214852ccf272e, 0x53a3540f5e5aa011);
    /* cbrt(11) */
    static const u256 t5(0x8e55b096fcd22d4e, 0x3c1e6d4936833117,
                         0x0ae1a0b51ea515b2, 0x6ef98efb6ebf35e3);
    /* cbrt(13) */
    static const u256 t6(0x967c447c6d817406, 0x7bc5196b06dc9887,
                         0x214ac2f50046dc65, 0x0f9bfa326367aeb7);
    /* cbrt(17) */
    static const u256 t7(0xa48fe0a92bc653e6, 0xec03c7ed7e59981b,
                         0x3e3a27d8d8e54797, 0xd607fe20b08d6175);
    /* cbrt(19) */
    static const u256 t8(0xaac717b5769b6046, 0x27896d0e27f2c11e,
                         0x281e73be041f0383, 0xa937169045fb3849);
    /* cbrt(23) */
    static const u256 t9(0xb601eaa628c0c090, 0xac51900eab494a5c,
                         0x236edd364b4df8c4, 0x5a0cdaed7df05aed);
    o1 = i1;
    o2 = i2;
    o1 += (o2^t1).fold(t1);
    o2 += (o1^t2).fold(t2);
    o1 += (o2^t3).fold(t3);
    o2 += (o1^t4).fold(t4);
    o1 += (o2^t5).fold(t5);
    o2 += (o1^t6).fold(t6);
    o1 += (o2^t7).fold(t7);
    o2 += (o1^t8).fold(t8);
    o1 += (o2^t9).fold(t9);
}

Можно ли вообще сократить код, к примеру, для демонстрации (на примере строки) его работы? То есть:

void hash {} // code of hash function 
  int main(){
  std::string strn;
  std::cout << "Enter your string ";
  getline (std::cin, strn);
  std::cout << "Hash: " << hash << "!\n";

}

Answer 1

добавьте определение перед использованием

typedef double_uint<__uint128_t, unsigned long long, 256> u256;
READ ALSO
Парсинг XSD С++

Парсинг XSD С++

Привет всемВозникла задача, на основе XML и XSD динамически подгружать интерфейс

199
Раскрытие макросов

Раскрытие макросов

Есть маленький вопрос по раскрытиям макросов

191
Перенос кода из C++ в C (СИ)

Перенос кода из C++ в C (СИ)

Пытаюсь переписать этот код, написанный на C++ в C (СИ) Код делает следующее: читает str2 если находит * читает str1 от позиции * до ближайшей < потом...

222