Files
MP-SPDZ/Tools/BitVector.cpp
Marcel Keller e07d9bf2a3 Maintenance.
2022-01-11 16:05:26 +11:00

169 lines
3.6 KiB
C++

#include "BitVector.h"
#include "Tools/random.h"
#include "Tools/octetStream.h"
#include "Math/gf2n.h"
#include "Math/gfp.h"
#include "Math/Z2k.h"
#include <fstream>
#include <assert.h>
void BitVector::assign(const BitVector& K)
{
if (nbits != K.nbits)
{
resize(K.nbits);
}
memcpy(bytes, K.bytes, nbytes);
}
void BitVector::resize_zero(size_t new_nbits)
{
size_t old_nbytes = nbytes;
resize(new_nbits);
if (old_nbytes < nbytes)
avx_memzero(bytes + old_nbytes, nbytes - old_nbytes);
}
const void* BitVector::get_ptr_to_byte(size_t i, size_t block_size) const
{
assert(i + block_size <= nbytes);
return bytes + i * block_size;
}
const void* BitVector::get_ptr_to_bit(size_t i, size_t block_size) const
{
assert((i * block_size) % 8 == 0);
assert((i + 1) * block_size <= nbits);
return bytes + block_size * i / 8;
}
BitVector BitVector::operator &(const BitVector& other) const
{
assert(size() == other.size());
BitVector res(size());
for (size_t i = 0; i < size_bytes(); i++)
res.bytes[i] = bytes[i] & other.bytes[i];
return res;
}
bool BitVector::parity() const
{
#if (defined(__SSE4_2__) or not defined(__clang__)) and defined(__x86_64__)
bool res = 0;
for (size_t i = 0; i < size_bytes() / 8; i++)
res ^= _popcnt64(((word*)bytes)[i]) & 1;
for (size_t i = size_bytes() / 8 * 8; i < size_bytes(); i++)
res ^= _popcnt32(bytes[i]) & 1;
return res;
#else
bool res = 0;
for (size_t i = 0; i < size_bytes() / 8; i++)
{
word x = ((word*)bytes)[i];
for (int i = 5; i >= 0; i--)
x ^= (x >> (1 << i));
res ^= (x & 1);
}
for (size_t i = size_bytes() / 8 * 8; i < size_bytes(); i++)
res ^= (*this)[i];
return res;
#endif
}
void BitVector::append(const BitVector& other, size_t length)
{
assert(nbits % 8 == 0);
assert(length % 8 == 0);
assert(length <= other.nbits);
auto old_nbytes = nbytes;
resize(nbits + length);
memcpy(bytes + old_nbytes, other.bytes, length / 8);
}
void BitVector::randomize(PRNG& G)
{
G.get_octets(bytes, nbytes);
}
void BitVector::randomize_at(int a, int nb, PRNG& G)
{
if (nb < 1)
throw invalid_length();
G.get_octets(bytes + a, nb);
}
/*
*/
void BitVector::output(ostream& s,bool human) const
{
if (human)
{
s << nbits << " " << hex;
for (unsigned int i = 0; i < nbytes; i++)
{
s << int(bytes[i]) << " ";
}
s << dec << endl;
}
else
{
int len = nbits;
s.write((char*) &len, sizeof(int));
s.write((char*) bytes, nbytes);
}
}
void BitVector::input(istream& s,bool human)
{
if (s.peek() == EOF)
{
if (s.tellg() == 0)
{
cout << "IO problem. Empty file?" << endl;
throw file_error("BitVector input");
}
throw end_of_file();
}
int len;
if (human)
{
s >> len >> hex;
resize(len);
for (size_t i = 0; i < nbytes; i++)
{
s >> bytes[i];
}
s >> dec;
}
else
{
s.read((char*) &len, sizeof(int));
resize(len);
s.read((char*) bytes, nbytes);
}
}
void BitVector::pack(octetStream& o) const
{
o.store_int(nbits, 8);
o.append((octet*)bytes, nbytes);
}
void BitVector::unpack(octetStream& o)
{
resize(o.get_int(8));
o.consume((octet*)bytes, nbytes);
}
BitVector& BitVector::operator =(const octetStream other)
{
resize(other.get_length() * 8);
memcpy(bytes, other.get_data(), nbytes);
return *this;
}