| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898 | #ifndef __OBLIVDS_TYPES_HPP__#define __OBLIVDS_TYPES_HPP__#include <tuple>#include <vector>#include <array>#include <cstdint>#include <x86intrin.h>  // SSE and AVX intrinsics#include <bsd/stdlib.h> // arc4random_buf#include "bitutils.hpp"// The number of bits in an MPC secret-shared memory word#ifndef VALUE_BITS#define VALUE_BITS 64#endif// Values in MPC secret-shared memory are of this type.// This is the type of the underlying shared value, not the types of the// shares themselves.#if VALUE_BITS == 64using value_t = uint64_t;#elif VALUE_BITS == 32using value_t = uint32_t;#else#error "Unsupported value of VALUE_BITS"#endif// Secret-shared bits are of this type.  Note that it is standards// compliant to treat a bool as an unsigned integer type with values 0// and 1.using bit_t = bool;// Counts of the number of bits in a value are of this type, which must// be large enough to store the _value_ VALUE_BITSusing nbits_t = uint8_t;// Convert a number of bits to the number of bytes required to store (or// more to the point, send) them.#define BITBYTES(nbits) (((nbits)+7)>>3)// A mask of this many bits; the test is to prevent 1<<nbits from// overflowing if nbits == VALUE_BITS#define MASKBITS(nbits) (((nbits) < VALUE_BITS) ? (value_t(1)<<(nbits))-1 : ~0)// The type of a register holding an additive share of a valuestruct RegAS {    value_t ashare;    RegAS() : ashare(0) {}    inline value_t share() const { return ashare; }    inline void set(value_t s) { ashare = s; }    // Set each side's share to a random value nbits bits long    inline void randomize(size_t nbits = VALUE_BITS) {        value_t mask = MASKBITS(nbits);        arc4random_buf(&ashare, sizeof(ashare));        ashare &= mask;    }    inline RegAS &operator+=(const RegAS &rhs) {        this->ashare += rhs.ashare;        return *this;    }    inline RegAS operator+(const RegAS &rhs) const {        RegAS res = *this;        res += rhs;        return res;    }    inline RegAS &operator-=(const RegAS &rhs) {        this->ashare -= rhs.ashare;        return *this;    }    inline RegAS operator-(const RegAS &rhs) const {        RegAS res = *this;        res -= rhs;        return res;    }    inline RegAS operator-() const {        RegAS res = *this;        res.ashare = -res.ashare;        return res;    }    inline RegAS &operator*=(value_t rhs) {        this->ashare *= rhs;        return *this;    }    inline RegAS operator*(value_t rhs) const {        RegAS res = *this;        res *= rhs;        return res;    }    // Multiply a scalar by a vector    template <size_t N>    inline std::array<RegAS,N> operator*(std::array<value_t,N> rhs) const {        std::array<RegAS,N> res;        for (size_t i=0;i<N;++i) {            res[i] = *this;            res[i] *= rhs[i];        }        return res;    }    inline RegAS &operator&=(value_t mask) {        this->ashare &= mask;        return *this;    }    inline RegAS operator&(value_t mask) const {        RegAS res = *this;        res &= mask;        return res;    }    // Multiply by the local share of the argument, not multiplcation of    // two shared values (two versions)    inline RegAS &mulshareeq(const RegAS &rhs) {        *this *= rhs.ashare;        return *this;    }    inline RegAS mulshare(const RegAS &rhs) const {        RegAS res = *this;        res *= rhs.ashare;        return res;    }    inline void dump() const {        printf("%016lx", ashare);    }};inline value_t combine(const RegAS &A, const RegAS &B,        nbits_t nbits = VALUE_BITS) {    value_t mask = ~0;    if (nbits < VALUE_BITS) {        mask = (value_t(1)<<nbits)-1;    }    return (A.ashare + B.ashare) & mask;}// The type of a register holding a bit sharestruct RegBS {    bit_t bshare;    RegBS() : bshare(0) {}    inline bit_t share() const { return bshare; }    inline void set(bit_t s) { bshare = s; }    // Set each side's share to a random bit    inline void randomize() {        unsigned char randb;        arc4random_buf(&randb, sizeof(randb));        bshare = randb & 1;    }    inline RegBS &operator^=(const RegBS &rhs) {        this->bshare ^= rhs.bshare;        return *this;    }    inline RegBS operator^(const RegBS &rhs) const {        RegBS res = *this;        res ^= rhs;        return res;    }    inline RegBS &operator^=(const bit_t &rhs) {        this->bshare ^= rhs;        return *this;    }    inline RegBS operator^(const bit_t &rhs) const {        RegBS res = *this;        res ^= rhs;        return res;    }};// The type of a register holding an XOR share of a valuestruct RegXS {    value_t xshare;    RegXS() : xshare(0) {}    RegXS(const RegBS &b) { xshare = b.bshare ? ~0 : 0; }    inline value_t share() const { return xshare; }    inline void set(value_t s) { xshare = s; }    // Set each side's share to a random value nbits bits long    inline void randomize(size_t nbits = VALUE_BITS) {        value_t mask = MASKBITS(nbits);        arc4random_buf(&xshare, sizeof(xshare));        xshare &= mask;    }    // For RegXS, + and * should be interpreted bitwise; that is, + is    // really XOR and * is really AND.  - is also XOR (the same as +).    // We also include actual XOR operators for convenience    inline RegXS &operator+=(const RegXS &rhs) {        this->xshare ^= rhs.xshare;        return *this;    }    inline RegXS operator+(const RegXS &rhs) const {        RegXS res = *this;        res += rhs;        return res;    }    inline RegXS &operator-=(const RegXS &rhs) {        this->xshare ^= rhs.xshare;        return *this;    }    inline RegXS operator-(const RegXS &rhs) const {        RegXS res = *this;        res -= rhs;        return res;    }    inline RegXS operator-() const {        RegXS res = *this;        return res;    }    inline RegXS &operator*=(value_t rhs) {        this->xshare &= rhs;        return *this;    }    inline RegXS operator*(value_t rhs) const {        RegXS res = *this;        res *= rhs;        return res;    }    // Multiply a scalar by a vector    template <size_t N>    inline std::array<RegXS,N> operator*(std::array<value_t,N> rhs) const {        std::array<RegXS,N> res;        for (size_t i=0;i<N;++i) {            res[i] = *this;            res[i] *= rhs[i];        }        return res;    }    inline RegXS &operator^=(const RegXS &rhs) {        this->xshare ^= rhs.xshare;        return *this;    }    inline RegXS operator^(const RegXS &rhs) const {        RegXS res = *this;        res ^= rhs;        return res;    }    inline RegXS &operator&=(value_t mask) {        this->xshare &= mask;        return *this;    }    inline RegXS operator&(value_t mask) const {        RegXS res = *this;        res &= mask;        return res;    }    // Bit shifting and bit extraction    inline RegXS &operator<<=(nbits_t shift) {        this->xshare <<= shift;        return *this;    }    inline RegXS operator<<(nbits_t shift) const {        RegXS res = *this;        res <<= shift;        return res;    }    inline RegXS &operator>>=(nbits_t shift) {        this->xshare >>= shift;        return *this;    }    inline RegXS operator>>(nbits_t shift) const {        RegXS res = *this;        res >>= shift;        return res;    }    inline RegBS bitat(nbits_t pos) const {        RegBS bs;        bs.set(!!(this->xshare & (value_t(1)<<pos)));        return bs;    }    // Multiply by the local share of the argument, not multiplcation of    // two shared values (two versions)    inline RegXS &mulshareeq(const RegXS &rhs) {        *this *= rhs.xshare;        return *this;    }    inline RegXS mulshare(const RegXS &rhs) const {        RegXS res = *this;        res *= rhs.xshare;        return res;    }    inline void dump() const {        printf("%016lx", xshare);    }    // Extract a bit share of bit bitnum of the XOR-shared register    inline RegBS bit(nbits_t bitnum) const {        RegBS bs;        bs.bshare = !!(xshare & (value_t(1)<<bitnum));        return bs;    }};inline value_t combine(const RegXS &A, const RegXS &B,        nbits_t nbits = VALUE_BITS) {    value_t mask = ~0;    if (nbits < VALUE_BITS) {        mask = (value_t(1)<<nbits)-1;    }    return (A.xshare ^ B.xshare) & mask;}// Enable templates to specialize on just the basic types RegAS and// RegXS.  Technique from// https://stackoverflow.com/questions/2430039/one-template-specialization-for-multiple-classestemplate <bool B> struct prac_template_bool_type {};using prac_template_true = prac_template_bool_type<true>;using prac_template_false = prac_template_bool_type<false>;template <typename T>struct prac_basic_Reg_S : prac_template_false{    static const bool value = false;};template<>struct prac_basic_Reg_S<RegAS>: prac_template_true{    static const bool value = true;};template<>struct prac_basic_Reg_S<RegXS>: prac_template_true{    static const bool value = true;};// Some useful operations on tuples, vectors, and arrays of the above// typestemplate <typename T>std::tuple<T,T> operator+=(std::tuple<T,T> &A,    const std::tuple<T,T> &B){    std::get<0>(A) += std::get<0>(B);    std::get<1>(A) += std::get<1>(B);    return A;}template <typename T>std::tuple<T,T> operator+=(const std::tuple<T&,T&> &A,    const std::tuple<T,T> &B){    std::get<0>(A) += std::get<0>(B);    std::get<1>(A) += std::get<1>(B);    return A;}template <typename T>std::tuple<T,T> operator+(const std::tuple<T,T> &A,    const std::tuple<T,T> &B){    auto res = A;    res += B;    return res;}template <typename T>std::tuple<T,T> operator-=(const std::tuple<T&,T&> &A,    const std::tuple<T,T> &B){    std::get<0>(A) -= std::get<0>(B);    std::get<1>(A) -= std::get<1>(B);    return A;}template <typename T>std::tuple<T,T> operator-=(std::tuple<T,T> &A,    const std::tuple<T,T> &B){    std::get<0>(A) -= std::get<0>(B);    std::get<1>(A) -= std::get<1>(B);    return A;}template <typename T>std::tuple<T,T> operator-(const std::tuple<T,T> &A,    const std::tuple<T,T> &B){    auto res = A;    res -= B;    return res;}template <typename T>std::tuple<T,T> operator*=(const std::tuple<T&,T&> &A,    const std::tuple<value_t,value_t> &B){    std::get<0>(A) *= std::get<0>(B);    std::get<1>(A) *= std::get<1>(B);    return A;}template <typename T>std::tuple<T,T> operator*=(std::tuple<T,T> &A,    const std::tuple<value_t,value_t> &B){    std::get<0>(A) *= std::get<0>(B);    std::get<1>(A) *= std::get<1>(B);    return A;}template <typename T>std::tuple<T,T> operator*(const std::tuple<T,T> &A,    const std::tuple<value_t,value_t> &B){    auto res = A;    res *= B;    return res;}template <typename T, size_t N>std::tuple<std::array<T,N>,std::array<T,N>> operator*(    const std::tuple<T,T> &A,    const std::tuple<std::array<value_t,N>,std::array<value_t,N>> &B){    std::tuple<std::array<T,N>,std::array<T,N>> res;    std::get<0>(res) = std::get<0>(A) * std::get<0>(B);    std::get<1>(res) = std::get<1>(A) * std::get<1>(B);    return res;}template <typename S, size_t N>inline std::array<value_t,N> combine(const std::array<S,N> &A,        const std::array<S,N> &B,        nbits_t nbits = VALUE_BITS) {    std::array<value_t,N> res;    for (size_t i=0;i<N;++i) {        res[i] = combine(A[i], B[i], nbits);    }    return res;}template <typename S, size_t N>inline std::tuple<std::array<value_t,N>,std::array<value_t,N>>    combine(const std::tuple<std::array<S,N>,std::array<S,N>> &A,        const std::tuple<std::array<S,N>,std::array<S,N>> &B,        nbits_t nbits = VALUE_BITS) {    return std::make_tuple(        combine(std::get<0>(A), std::get<0>(B), nbits),        combine(std::get<1>(A), std::get<1>(B), nbits));}template <typename T>std::tuple<T,T,T> operator+=(const std::tuple<T&,T&,T&> &A,    const std::tuple<T,T,T> &B){    std::get<0>(A) += std::get<0>(B);    std::get<1>(A) += std::get<1>(B);    std::get<2>(A) += std::get<2>(B);    return A;}template <typename T>std::tuple<T,T,T> operator+=(std::tuple<T,T,T> &A,    const std::tuple<T,T,T> &B){    std::get<0>(A) += std::get<0>(B);    std::get<1>(A) += std::get<1>(B);    std::get<2>(A) += std::get<2>(B);    return A;}template <typename T>std::tuple<T,T,T> operator+(const std::tuple<T,T,T> &A,    const std::tuple<T,T,T> &B){    auto res = A;    res += B;    return res;}template <typename T>std::tuple<T,T,T> operator-=(const std::tuple<T&,T&,T&> &A,    const std::tuple<T,T,T> &B){    std::get<0>(A) -= std::get<0>(B);    std::get<1>(A) -= std::get<1>(B);    std::get<2>(A) -= std::get<2>(B);    return A;}template <typename T>std::tuple<T,T,T> operator-=(std::tuple<T,T,T> &A,    const std::tuple<T,T,T> &B){    std::get<0>(A) -= std::get<0>(B);    std::get<1>(A) -= std::get<1>(B);    std::get<2>(A) -= std::get<2>(B);    return A;}template <typename T>std::tuple<T,T,T> operator-(const std::tuple<T,T,T> &A,    const std::tuple<T,T,T> &B){    auto res = A;    res -= B;    return res;}template <typename T>std::tuple<T,T,T> operator*=(const std::tuple<T&,T&,T&> &A,    const std::tuple<value_t,value_t,value_t> &B){    std::get<0>(A) *= std::get<0>(B);    std::get<1>(A) *= std::get<1>(B);    std::get<2>(A) *= std::get<2>(B);    return A;}template <typename T>std::tuple<T,T,T> operator*=(std::tuple<T,T,T> &A,    const std::tuple<value_t,value_t,value_t> &B){    std::get<0>(A) *= std::get<0>(B);    std::get<1>(A) *= std::get<1>(B);    std::get<2>(A) *= std::get<2>(B);    return A;}template <typename T>std::tuple<T,T,T> operator*(const std::tuple<T,T,T> &A,    const std::tuple<value_t,value_t,value_t> &B){    auto res = A;    res *= B;    return res;}template <typename T, size_t N>std::tuple<std::array<T,N>,std::array<T,N>,std::array<T,N>> operator*(    const std::tuple<T,T,T> &A,    const std::tuple<std::array<value_t,N>,std::array<value_t,N>,std::array<value_t,N>> &B){    std::tuple<std::array<T,N>,std::array<T,N>,std::array<T,N>> res;    std::get<0>(res) = std::get<0>(A) * std::get<0>(B);    std::get<1>(res) = std::get<1>(A) * std::get<1>(B);    std::get<2>(res) = std::get<2>(A) * std::get<2>(B);    return res;}inline std::vector<RegAS> operator-(const std::vector<RegAS> &A){    std::vector<RegAS> res;    for (const auto &v : A) {        res.push_back(-v);    }    return res;}inline std::vector<RegXS> operator-(const std::vector<RegXS> &A){    return A;}inline std::vector<RegBS> operator-(const std::vector<RegBS> &A){    return A;}template <size_t N>inline std::vector<RegAS> operator-(const std::array<RegAS,N> &A){    std::vector<RegAS> res;    for (const auto &v : A) {        res.push_back(-v);    }    return res;}template <size_t N>inline std::array<RegXS,N> operator-(const std::array<RegXS,N> &A){    return A;}template <size_t N>inline std::array<RegBS,N> operator-(const std::array<RegBS,N> &A){    return A;}template <typename S, size_t N>inline std::array<S,N> &operator+=(std::array<S,N> &A, const std::array<S,N> &B){    for (size_t i=0;i<N;++i) {        A[i] += B[i];    }    return A;}template <typename S, size_t N>inline std::array<S,N> &operator-=(std::array<S,N> &A, const std::array<S,N> &B){    for (size_t i=0;i<N;++i) {        A[i] -= B[i];    }    return A;}template <typename S, size_t N>inline std::array<S,N> &operator^=(std::array<S,N> &A, const std::array<S,N> &B){    for (size_t i=0;i<N;++i) {        A[i] ^= B[i];    }    return A;}// XOR the bit B into the low bit of Atemplate <typename S, size_t N>inline std::array<S,N> &xor_lsb(std::array<S,N> &A, bit_t B){    A[0] ^= lsb128_mask[B];    return A;}template <typename S, size_t N>inline std::tuple<std::array<value_t,N>,std::array<value_t,N>,std::array<value_t,N>>    combine(        const std::tuple<std::array<S,N>,std::array<S,N>,std::array<S,N>> &A,        const std::tuple<std::array<S,N>,std::array<S,N>,std::array<S,N>> &B,        nbits_t nbits = VALUE_BITS) {    return std::make_tuple(        combine(std::get<0>(A), std::get<0>(B), nbits),        combine(std::get<1>(A), std::get<1>(B), nbits),        combine(std::get<2>(A), std::get<2>(B), nbits));}// The _maximum_ number of bits in an MPC address; the actual size of// the memory will typically be set at runtime, but it cannot exceed// this value.  It is more efficient (in terms of communication) in some// places for this value to be at most 32.#ifndef ADDRESS_MAX_BITS#define ADDRESS_MAX_BITS 32#endif// Addresses of MPC secret-shared memory are of this type#if ADDRESS_MAX_BITS <= 32using address_t = uint32_t;#elif ADDRESS_MAX_BITS <= 64using address_t = uint64_t;#else#error "Unsupported value of ADDRESS_MAX_BITS"#endif#if ADDRESS_MAX_BITS > VALUE_BITS#error "VALUE_BITS must be at least as large as ADDRESS_MAX_BITS"#endif// A multiplication triple is a triple (X0,Y0,Z0) held by P0 (and// correspondingly (X1,Y1,Z1) held by P1), with all values random,// but subject to the relation that X0*Y1 + Y0*X1 = Z0+Z1using MultTriple = std::tuple<value_t, value_t, value_t>;// The *Name structs are a way to get strings representing the names of// the types as would be given to preprocessing to create them in// advance.struct MultTripleName { static constexpr const char *name = "m"; };// A half-triple is (X0,Z0) held by P0 (and correspondingly (Y1,Z1) held// by P1), with all values random, but subject to the relation that// X0*Y1 = Z0+Z1using HalfTriple = std::tuple<value_t, value_t>;struct HalfTripleName { static constexpr const char *name = "h"; };// An AND triple is a triple (X0,Y0,Z0) held by P0 (and correspondingly// (X1,Y1,Z1) held by P1), with all values random, but subject to the// relation that X0&Y1 ^ Y0&X1 = Z0^Z1using AndTriple = std::tuple<value_t, value_t, value_t>;struct AndTripleName { static constexpr const char *name = "a"; };// The type of nodes in a DPF.  This must be at least as many bits as// the security parameter, and at least twice as many bits as value_t.using DPFnode = __m128i;// XOR the bit B into the low bit of Ainline DPFnode &xor_lsb(DPFnode &A, bit_t B){    A ^= lsb128_mask[B];    return A;}// A Select triple for type V (V is DPFnode, value_t, or bit_t) is a// triple of (X0,Y0,Z0) where X0 is a bit and Y0 and Z0 are Vs held by// P0 (and correspondingly (X1,Y1,Z1) held by P1), with all values// random, but subject to the relation that (X0*Y1) ^ (Y0*X1) = Z0^Z1.// This is a struct instead of a tuple for alignment reasons.template <typename V>struct SelectTriple {    bit_t X;    V Y, Z;};// Of the three options for V, we only ever store V = value_tstruct ValSelectTripleName { static constexpr const char *name = "s"; };// These are defined in rdpf.hpp, but declared here to avoid cyclic// header dependencies.template <nbits_t WIDTH> struct RDPFPair;struct RDPFPairName { static constexpr const char *name = "r"; };// Incremental RDPFsstruct IRDPFPairName { static constexpr const char *name = "i"; };template <nbits_t WIDTH> struct RDPFTriple;struct RDPFTripleName { static constexpr const char *name = "r"; };// Incremental RDPFsstruct IRDPFTripleName { static constexpr const char *name = "i"; };struct CDPF;struct CDPFName { static constexpr const char *name = "c"; };// We want the I/O (using << and >>) for many classes// to just be a common thing: write out the bytes// straight from memory#define DEFAULT_IO(CLASSNAME)                  \    template <typename T>                      \    T& operator>>(T& is, CLASSNAME &x)         \    {                                          \        is.read((char *)&x, sizeof(x));        \        return is;                             \    }                                          \                                               \    template <typename T>                      \    T& operator<<(T& os, const CLASSNAME &x)   \    {                                          \        os.write((const char *)&x, sizeof(x)); \        return os;                             \    }// Default I/O for various typesDEFAULT_IO(DPFnode)DEFAULT_IO(RegBS)DEFAULT_IO(RegAS)DEFAULT_IO(RegXS)DEFAULT_IO(MultTriple)DEFAULT_IO(HalfTriple)// We don't need one for AndTriple because it's exactly the same type as// MultTriple// I/O for arraystemplate <typename T, typename S, size_t N>T& operator>>(T& is, std::array<S,N> &x){    for (size_t i=0;i<N;++i) {        is >> x[i];    }    return is;}template <typename T, typename S, size_t N>T& operator<<(T& os, const std::array<S,N> &x){    for (size_t i=0;i<N;++i) {        os << x[i];    }    return os;}// I/O for SelectTriplestemplate <typename T, typename V>T& operator>>(T& is, SelectTriple<V> &x){    is.read((char *)&x.X, sizeof(x.X));    is.read((char *)&x.Y, sizeof(x.Y));    is.read((char *)&x.Z, sizeof(x.Z));    return is;}template <typename T, typename V>T& operator<<(T& os, const SelectTriple<V> &x){    os.write((const char *)&x.X, sizeof(x.X));    os.write((const char *)&x.Y, sizeof(x.Y));    os.write((const char *)&x.Z, sizeof(x.Z));    return os;}// And for pairs and triples#define DEFAULT_TUPLE_IO(CLASSNAME)                                  \    template <typename T>                                            \    T& operator>>(T& is, std::tuple<CLASSNAME, CLASSNAME> &x)        \    {                                                                \        is >> std::get<0>(x) >> std::get<1>(x);                      \        return is;                                                   \    }                                                                \                                                                     \    template <typename T>                                            \    T& operator<<(T& os, const std::tuple<CLASSNAME, CLASSNAME> &x)  \    {                                                                \        os << std::get<0>(x) << std::get<1>(x);                      \        return os;                                                   \    }                                                                \                                                                     \    template <typename T>                                            \    T& operator>>(T& is, std::tuple<CLASSNAME, CLASSNAME, CLASSNAME> &x) \    {                                                                \        is >> std::get<0>(x) >> std::get<1>(x) >> std::get<2>(x);    \        return is;                                                   \    }                                                                \                                                                     \    template <typename T>                                            \    T& operator<<(T& os, const std::tuple<CLASSNAME, CLASSNAME, CLASSNAME> &x) \    {                                                                \        os << std::get<0>(x) << std::get<1>(x) << std::get<2>(x);    \        return os;                                                   \    }DEFAULT_TUPLE_IO(RegAS)DEFAULT_TUPLE_IO(RegXS)// And for pairs and triples of arraystemplate <typename T, typename S, size_t N>T& operator>>(T& is, std::tuple<std::array<S,N>, std::array<S,N>> &x){    is >> std::get<0>(x) >> std::get<1>(x);    return is;}template <typename T, typename S, size_t N>T& operator<<(T& os, const std::tuple<std::array<S,N>, std::array<S,N>> &x){    os << std::get<0>(x) << std::get<1>(x);    return os;}template <typename T, typename S, size_t N>T& operator>>(T& is, std::tuple<std::array<S,N>, std::array<S,N>, std::array<S,N>> &x){    is >> std::get<0>(x) >> std::get<1>(x) >> std::get<2>(x);    return is;}template <typename T, typename S, size_t N>T& operator<<(T& os, const std::tuple<std::array<S,N>, std::array<S,N>, std::array<S,N>> &x){    os << std::get<0>(x) << std::get<1>(x) << std::get<2>(x);    return os;}enum ProcessingMode {    MODE_ONLINE,        // Online mode, after preprocessing has been done    MODE_PREPROCESSING, // Preprocessing mode    MODE_ONLINEONLY     // Online-only mode, where all computations are};                      // done online#endif
 |