mirror of
https://github.com/data61/MP-SPDZ.git
synced 2026-01-09 13:37:58 -05:00
271 lines
8.1 KiB
C++
271 lines
8.1 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 "GC/TinierSecret.h"
|
|
|
|
#include "Protocols/Share.hpp"
|
|
#include "Protocols/fake-stuff.hpp"
|
|
#include "Protocols/MascotPrep.hpp"
|
|
#include "Math/Z2k.hpp"
|
|
#include "Math/gfp.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), player(0), bonding(0),
|
|
network_opts(opt, argc, argv, 2, true)
|
|
{
|
|
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; always done with SPDZ2k).", // Help description.
|
|
"-c", // Flag token.
|
|
"--check" // Flag token.
|
|
);
|
|
opt.add(
|
|
"", // Default.
|
|
0, // Required?
|
|
1, // Number of args expected.
|
|
0, // Delimiter if expecting multiple args.
|
|
"GF(p) items for chosen prime", // 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;
|
|
correlation_check = opt.get("-c")->isSet;
|
|
generateMACs = opt.get("-m")->isSet || check;
|
|
amplify = opt.get("-a")->isSet || generateMACs;
|
|
if (opt.isSet("-P"))
|
|
{
|
|
string tmp;
|
|
opt.get("-P")->getString(tmp);
|
|
prime = tmp;
|
|
}
|
|
bonding = opt.get("-b")->isSet;
|
|
opt.get("-Z")->getInt(z2k);
|
|
check |= z2k;
|
|
amplify &= not z2k;
|
|
z2s = z2k;
|
|
if (opt.isSet("-S"))
|
|
opt.get("-S")->getInt(z2s);
|
|
|
|
// doesn't work with Montgomery multiplication
|
|
if (prime)
|
|
gfpvar1::init_field(prime, false);
|
|
else
|
|
gfpvar1::init_default(128, false);
|
|
gf2n_long::init_field();
|
|
gf2n_short::init_field();
|
|
|
|
PRNG G;
|
|
G.ReSeed();
|
|
mac_key2.randomize(G);
|
|
mac_keyp.randomize(G);
|
|
mac_keyz.randomize(G);
|
|
}
|
|
|
|
template<class T>
|
|
GeneratorThread* TripleMachine::new_generator(OTTripleSetup& setup, int i,
|
|
typename T::mac_key_type mac_key)
|
|
{
|
|
if (i == 0)
|
|
T::MAC_Check::setup(*player);
|
|
|
|
if (output and i == 0)
|
|
{
|
|
prep_data_dir = get_prep_sub_dir<T>(PREP_DIR, nplayers);
|
|
T::clear::write_setup(prep_data_dir);
|
|
write_mac_key(prep_data_dir, my_num, nplayers, mac_key);
|
|
}
|
|
|
|
return new typename T::TripleGenerator(setup, N[i % nConnections], i,
|
|
nTriplesPerThread, nloops, *this, mac_key);
|
|
}
|
|
|
|
void TripleMachine::run()
|
|
{
|
|
cout << "my_num: " << my_num << endl;
|
|
network_opts.start_networking(N[0], my_num);
|
|
nConnections = 1;
|
|
if (bonding)
|
|
{
|
|
if (network_opts.ip_filename.empty())
|
|
{
|
|
cerr << "Bonding only works with --ip-file-name" << endl;
|
|
exit(1);
|
|
}
|
|
N[1].init(my_num, network_opts.portnum_base + 1000,
|
|
network_opts.ip_filename + "2", nplayers);
|
|
nConnections = 2;
|
|
}
|
|
// do the base OTs
|
|
PlainPlayer P(N[0], "base");
|
|
OTTripleSetup setup(P, true);
|
|
player = &P;
|
|
|
|
vector<GeneratorThread*> generators(nthreads);
|
|
vector<pthread_t> threads(nthreads);
|
|
|
|
for (int i = 0; i < nthreads; i++)
|
|
{
|
|
if (prime)
|
|
generators[i] = new_generator<Share<gfpvar1>>(setup, i, mac_keyp);
|
|
else if (z2k)
|
|
{
|
|
if (z2k == 32 and z2s == 32)
|
|
generators[i] = new_generator<Spdz2kShare<32, 32>>(setup, i, mac_keyz);
|
|
else if (z2k == 64 and z2s == 64)
|
|
generators[i] = new_generator<Spdz2kShare<64, 64>>(setup, i, mac_keyz);
|
|
else if (z2k == 64 and z2s == 48)
|
|
generators[i] = new_generator<Spdz2kShare<64, 48>>(setup, i, mac_keyz);
|
|
else if (z2k == 66 and z2s == 64)
|
|
generators[i] = new_generator<Spdz2kShare<66, 64>>(setup, i, mac_keyz);
|
|
else if (z2k == 66 and z2s == 48)
|
|
generators[i] = new_generator<Spdz2kShare<66, 48>>(setup, i, mac_keyz);
|
|
#ifdef RING_SIZE
|
|
else if (z2k == RING_SIZE and z2s == SPDZ2K_DEFAULT_SECURITY)
|
|
generators[i] = new_generator<Spdz2kShare<RING_SIZE, SPDZ2K_DEFAULT_SECURITY>>(setup, i, mac_keyz);
|
|
#endif
|
|
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, mac_key2);
|
|
}
|
|
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];
|
|
}
|