mirror of
https://github.com/data61/MP-SPDZ.git
synced 2026-01-10 05:57:57 -05:00
261 lines
7.5 KiB
C++
261 lines
7.5 KiB
C++
/*
|
|
* TripleMachine.cpp
|
|
*
|
|
*/
|
|
|
|
#include <OT/TripleMachine.h>
|
|
#include "OT/NPartyTripleGenerator.h"
|
|
#include "OT/OTTripleSetup.h"
|
|
#include "Math/gf2n.h"
|
|
#include "Math/Setup.h"
|
|
#include "Protocols/Spdz2kShare.h"
|
|
#include "Tools/ezOptionParser.h"
|
|
#include "Math/Setup.h"
|
|
#include "Protocols/fake-stuff.h"
|
|
#include "Math/BitVec.h"
|
|
|
|
#include "Protocols/fake-stuff.hpp"
|
|
#include "Math/Z2k.hpp"
|
|
#include "OT/NPartyTripleGenerator.hpp"
|
|
|
|
#include <iostream>
|
|
#include <fstream>
|
|
using namespace std;
|
|
|
|
void* run_ngenerator_thread(void* ptr)
|
|
{
|
|
((GeneratorThread*)ptr)->generate();
|
|
return 0;
|
|
}
|
|
|
|
TripleMachine::TripleMachine(int argc, const char** argv) :
|
|
nConnections(1), bonding(0)
|
|
{
|
|
opt.add(
|
|
"1", // Default.
|
|
0, // Required?
|
|
1, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"Number of loops (default: 1).", // Help description.
|
|
"-l", // Flag token.
|
|
"--nloops" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
0, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"Generate MACs (implies -a).", // Help description.
|
|
"-m", // Flag token.
|
|
"--macs" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
0, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"Amplify triples.", // Help description.
|
|
"-a", // Flag token.
|
|
"--amplify" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
0, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"Check triples (implies -m).", // Help description.
|
|
"-c", // Flag token.
|
|
"--check" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
0, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"GF(p) items", // Help description.
|
|
"-P", // Flag token.
|
|
"--prime-field" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
0, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"Channel bonding", // Help description.
|
|
"-b", // Flag token.
|
|
"--bonding" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
0, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"Generate bits", // Help description.
|
|
"-B", // Flag token.
|
|
"--bits" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
1, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"SPDZ2k parameter, e.g., 64", // Help description.
|
|
"-Z", // Flag token.
|
|
"--Z2k" // Flag token.
|
|
);
|
|
opt.add(
|
|
"64", // Default.
|
|
0, // Required?
|
|
1, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"SPDZ2k security parameter (default: 64)", // Help description.
|
|
"-S", // Flag token.
|
|
"--security" // Flag token.
|
|
);
|
|
|
|
parse_options(argc, argv);
|
|
|
|
opt.get("-l")->getInt(nloops);
|
|
generateBits = opt.get("-B")->isSet;
|
|
check = opt.get("-c")->isSet || generateBits;
|
|
generateMACs = opt.get("-m")->isSet || check;
|
|
amplify = opt.get("-a")->isSet || generateMACs;
|
|
primeField = opt.get("-P")->isSet;
|
|
bonding = opt.get("-b")->isSet;
|
|
opt.get("-Z")->getInt(z2k);
|
|
check |= z2k;
|
|
z2s = z2k;
|
|
if (opt.isSet("-S"))
|
|
opt.get("-S")->getInt(z2s);
|
|
|
|
bigint p;
|
|
if (output)
|
|
{
|
|
prep_data_dir = get_prep_dir(nplayers, 128, 128);
|
|
ofstream outf;
|
|
generate_online_setup(outf, prep_data_dir, p, 128, 128);
|
|
}
|
|
else
|
|
{
|
|
int idx, m;
|
|
SPDZ_Data_Setup_Primes(p, 128, idx, m);
|
|
}
|
|
|
|
// doesn't work with Montgomery multiplication
|
|
gfp1::init_field(p, false);
|
|
gf2n_long::init_field(128);
|
|
|
|
PRNG G;
|
|
G.ReSeed();
|
|
mac_key2l.randomize(G);
|
|
mac_key2s.randomize(G);
|
|
mac_keyp.randomize(G);
|
|
mac_keyz.randomize(G);
|
|
}
|
|
|
|
template<class T>
|
|
GeneratorThread* TripleMachine::new_generator(OTTripleSetup& setup, int i)
|
|
{
|
|
return new typename T::TripleGenerator(setup, N[i % nConnections], i,
|
|
nTriplesPerThread, nloops, *this);
|
|
}
|
|
|
|
void TripleMachine::run()
|
|
{
|
|
cout << "my_num: " << my_num << endl;
|
|
N[0].init(my_num, 10000, "HOSTS", nplayers);
|
|
nConnections = 1;
|
|
if (bonding)
|
|
{
|
|
N[1].init(my_num, 11000, "HOSTS2", nplayers);
|
|
nConnections = 2;
|
|
}
|
|
// do the base OTs
|
|
PlainPlayer P(N[0], 0xF000);
|
|
OTTripleSetup setup(P, true);
|
|
|
|
vector<GeneratorThread*> generators(nthreads);
|
|
vector<pthread_t> threads(nthreads);
|
|
|
|
for (int i = 0; i < nthreads; i++)
|
|
{
|
|
if (primeField)
|
|
generators[i] = new_generator<Share<gfp>>(setup, i);
|
|
else if (z2k)
|
|
{
|
|
if (z2k == 32 and z2s == 32)
|
|
generators[i] = new_generator<Spdz2kShare<32, 32>>(setup, i);
|
|
else if (z2k == 64 and z2s == 64)
|
|
generators[i] = new_generator<Spdz2kShare<64, 64>>(setup, i);
|
|
else if (z2k == 64 and z2s == 48)
|
|
generators[i] = new_generator<Spdz2kShare<64, 48>>(setup, i);
|
|
else if (z2k == 66 and z2s == 64)
|
|
generators[i] = new_generator<Spdz2kShare<66, 64>>(setup, i);
|
|
else if (z2k == 66 and z2s == 48)
|
|
generators[i] = new_generator<Spdz2kShare<66, 48>>(setup, i);
|
|
else
|
|
throw runtime_error("not compiled for k=" + to_string(z2k) + " and s=" + to_string(z2s));
|
|
}
|
|
else
|
|
generators[i] = new_generator<Share<gf2n>>(setup, i);
|
|
}
|
|
ntriples = generators[0]->nTriples * nthreads;
|
|
cout <<"Setup generators\n";
|
|
for (int i = 0; i < nthreads; i++)
|
|
{
|
|
// lock before starting thread to avoid race condition
|
|
generators[i]->lock();
|
|
pthread_create(&threads[i], 0, run_ngenerator_thread, generators[i]);
|
|
}
|
|
|
|
// wait for initialization, then start clock and computation
|
|
for (int i = 0; i < nthreads; i++)
|
|
generators[i]->wait();
|
|
cout << "Starting computation" << endl;
|
|
gettimeofday(&start, 0);
|
|
for (int i = 0; i < nthreads; i++)
|
|
{
|
|
generators[i]->signal();
|
|
generators[i]->unlock();
|
|
}
|
|
|
|
// wait for threads to finish
|
|
for (int i = 0; i < nthreads; i++)
|
|
{
|
|
pthread_join(threads[i],NULL);
|
|
cout << "thread " << i+1 << " finished\n" << flush;
|
|
}
|
|
|
|
map<string,Timer>& timers = generators[0]->timers;
|
|
for (map<string,Timer>::iterator it = timers.begin(); it != timers.end(); it++)
|
|
{
|
|
double sum = 0;
|
|
for (size_t i = 0; i < generators.size(); i++)
|
|
sum += generators[i]->timers[it->first].elapsed();
|
|
cout << it->first << " on average took time "
|
|
<< sum / generators.size() << endl;
|
|
}
|
|
|
|
gettimeofday(&stop, 0);
|
|
double time = timeval_diff_in_seconds(&start, &stop);
|
|
cout << "Time: " << time << endl;
|
|
cout << "Throughput: " << ntriples / time << endl;
|
|
|
|
for (size_t i = 0; i < generators.size(); i++)
|
|
delete generators[i];
|
|
|
|
if (output)
|
|
output_mac_keys();
|
|
}
|
|
|
|
void TripleMachine::output_mac_keys()
|
|
{
|
|
if (z2k) {
|
|
write_mac_keys(prep_data_dir, my_num, nplayers, mac_keyz, mac_key2l);
|
|
}
|
|
else if (gf2n::degree() > 64)
|
|
write_mac_keys(prep_data_dir, my_num, nplayers, mac_keyp, mac_key2l);
|
|
else
|
|
write_mac_keys(prep_data_dir, my_num, nplayers, mac_keyp, mac_key2s);
|
|
}
|