HEBench
Quickstart C++ Wrapper Backend Tutorial - Engine Initialization and Benchmark Description

Steps

1. Benchmark description initialization

The first step for our new benchmark is to fill out the benchmark description. TutorialEltwiseAddBenchmarkDescription defined in tutorial_eltwiseadd_benchmark_palisade.h inherits from hebench::cpp::BenchmarkDescription.

In the constructor we should initialize and set the inherited m_descriptor object. We currently have a description already in place from the original example to which we will make some changes for our PALISADE-based workload.

TutorialEltwiseAddBenchmarkDescription is declared in tutorial_eltwiseadd_benchmark_palisade.cpp.

First, we initialize the descriptor object memory:

28  // initialize the descriptor for this benchmark
29  std::memset(&m_descriptor, 0, sizeof(hebench::APIBridge::BenchmarkDescriptor));
Defines a benchmark test.
Definition: types.h:527

hebench::APIBridge::BenchmarkDescriptor.workload: specifies which workload this benchmark will perform. Workloads supported by your current HEBench version are defined under hebench::APIBridge::Workload in hebench/api_bridge/types_palisade.h. For our example we will set it to hebench::APIBridge::Workload::EltwiseAdd:

hebench::APIBridge::BenchmarkDescriptor.data_type: tells the front end what type of data the benchmark is expecting to recieve and return. The possible types are defined in hebench::APIBridge::DataType.

37  m_descriptor.data_type = hebench::APIBridge::DataType::Int64;
@ Int64
64 bits signed integers.
Definition: types.h:304

hebench::APIBridge::BenchmarkDescriptor.category: Specifies one of the benchmark categories defined under hebench::APIBridge::Category. For our first benchmark we are setting it to offline.

41  m_descriptor.category = hebench::APIBridge::Category::Offline;

hebench::APIBridge::BenchmarkDescriptor.cat_params.offline: These are parameters which are relevant to tests of the offline category. These specify the supported number of samples (or batch sizes) for each of the operation parameters. These are arbitrary and specific to the benchmark we want to create. See the documentation for more details.

Since our initial workflow forced sample sizes of 2 samples for the first operand and 5 for the second, we are setting them the same here. If our benchmark can support variable sample sizes for any operand, we set the corresponding value to 0. The specific size can be set in a configuration file passed when running this benchmark. For more information regarding benchmark configuration files see Benchmark Configuration File Reference .

45  m_descriptor.cat_params.offline.data_count[0] = 2; // 2 data samples for op parameter 0
46  m_descriptor.cat_params.offline.data_count[1] = 5; // 5 data samples for op parameter 1

hebench::APIBridge::BenchmarkDescriptor.cipher_param_mask: Specifies which operands will be plain text and which will be encrypted. This is a bit mask where the bit position represents the operation parameter, and its value represents whether it is plain text (0) or encrypted (1). For this tutorial we want first operand plain text and second encrypted.

50  m_descriptor.cipher_param_mask = 1 << 1;

Note that we are setting the mask to 1 << 1 or 0000 0010 in binary, indicating that operation parameter in position 0 is plain text, and postion 1 is encrypted. The rest of the bits are ignored if there are not corresponding operation parameters.

hebench::APIBridge::BenchmarkDescriptor.scheme: Specifies the scheme the test will use. It must be one of the schemes registered during engine initialization earlier.

55  m_descriptor.scheme = HEBENCH_HE_SCHEME_BFV;
#define HEBENCH_HE_SCHEME_BFV
Definition: types.h:69

hebench::APIBridge::BenchmarkDescriptor.security: Specifies what security level the benchmark uses. It must be one of the security levels registered during engine initialization earlier.

59  m_descriptor.security = TUTORIAL_HE_SECURITY_128;

hebench::APIBridge::BenchmarkDescriptor.other: This is an optional parameter which can be used for a backend defined purpose to differentiate benchmarks in additional backend-specific ways.

63  m_descriptor.other = 0; // no extras needed for our purpose:
64  // Other backends can use this field to differentiate between
65  // benchmarks for which internal parameters, not specified by
66  // other fields of this structure, differ.

Finally, a workload that requires flexible workload parameters must specify, at least, one set of default arguments. Workload parameters are flexible enough for backends to specify other parameters beyond the mandatory requirements, such as, but not limited to parameters that could be used for optimization like padding size, extra memory, etc.

According to Vector Element-wise Addition Workload , element-wise addition requires the number of elements for a vector. For this tutorial, the required parameter is enough, so, we do not need to create any extra workload parameters.

We create a single default set with a default value for the vector size:

70  // specify default arguments for this workload flexible parameters:
71  hebench::cpp::WorkloadParams::EltwiseAdd default_workload_params;
72  default_workload_params.n() = 400;
73  this->addDefaultParameters(default_workload_params);
std::uint64_t n() const
Number elements in a vector.

Since the value in our original workload was 400, we are setting it the same as default. If our implementation happens to support only vectors of this size, we can enforce it during benchmark initialization itself.

The complete implementation of our Benchmark description is provided below

25 TutorialEltwiseAddBenchmarkDescription::TutorialEltwiseAddBenchmarkDescription()
26 {
28  // initialize the descriptor for this benchmark
29  std::memset(&m_descriptor, 0, sizeof(hebench::APIBridge::BenchmarkDescriptor));
31 
33  m_descriptor.workload = hebench::APIBridge::Workload::EltwiseAdd;
35 
37  m_descriptor.data_type = hebench::APIBridge::DataType::Int64;
39 
41  m_descriptor.category = hebench::APIBridge::Category::Offline;
43 
45  m_descriptor.cat_params.offline.data_count[0] = 2; // 2 data samples for op parameter 0
46  m_descriptor.cat_params.offline.data_count[1] = 5; // 5 data samples for op parameter 1
48 
50  m_descriptor.cipher_param_mask = 1 << 1;
52 
53  //
55  m_descriptor.scheme = HEBENCH_HE_SCHEME_BFV;
57 
59  m_descriptor.security = TUTORIAL_HE_SECURITY_128;
61 
63  m_descriptor.other = 0; // no extras needed for our purpose:
64  // Other backends can use this field to differentiate between
65  // benchmarks for which internal parameters, not specified by
66  // other fields of this structure, differ.
68 
70  // specify default arguments for this workload flexible parameters:
71  hebench::cpp::WorkloadParams::EltwiseAdd default_workload_params;
72  default_workload_params.n() = 400;
73  this->addDefaultParameters(default_workload_params);
75 }

2. Benchmark description overrides

We must override methods to create and destroy our actual benchmark. These will be called by C++ wrapper's BaseEngine implementation when the benchmark needs to be instantiated to execute, and disposed of when completed.

Our create method follows:

105  const hebench::APIBridge::WorkloadParams *p_params)
106 {
107  if (!p_params)
108  throw hebench::cpp::HEBenchError(HEBERROR_MSG_CLASS("Invalid empty workload parameters. This workload requires flexible parameters."),
110 
111  TutorialEngine &ex_engine = dynamic_cast<TutorialEngine &>(engine);
112  return new TutorialEltwiseAddBenchmark(ex_engine, m_descriptor, *p_params);
113 }
Top level opaque benchmark class.
Definition: benchmark.hpp:143
Base class that encapsulates common behavior of backend engines.
Definition: engine.hpp:70
#define HEBERROR_MSG_CLASS(message)
ErrorCode createBenchmark(Handle h_engine, Handle h_bench_desc, const WorkloadParams *p_params, Handle *h_benchmark)
Instantiates a benchmark on the backend.
Specifies the parameters for a workload.
Definition: types.h:363
#define HEBENCH_ECODE_CRITICAL_ERROR
Specifies a critical error.
Definition: types.h:50

This is the destroy method:

117 void TutorialEltwiseAddBenchmarkDescription::destroyBenchmark(hebench::cpp::BaseBenchmark *p_bench)
118 {
119  if (p_bench)
120  delete p_bench;
121 }

There are other methods that we can override to modify the behavior of the description class. Check the documentation on hebench::cpp::BenchmarkDescription for more information.

In this tutorial we are overriding hebench::cpp::BenchmarkDescription::getBenchmarkDescription() to output extra information customized for our example.

84 std::string TutorialEltwiseAddBenchmarkDescription::getBenchmarkDescription(const hebench::APIBridge::WorkloadParams *p_w_params) const
85 {
86  std::stringstream ss;
87  ss << BenchmarkDescription::getBenchmarkDescription(p_w_params);
88  ss << ", Encryption parameters" << std::endl
89  << ", , HE Library, PALISADE v1.11.3" << std::endl
90  << ", , Poly modulus degree, " << PolyModulusDegree << std::endl
91  << ", , Coefficient Moduli, 60, 60";
92  if (p_w_params)
93  {
94  hebench::cpp::WorkloadParams::EltwiseAdd w_params(*p_w_params);
95  ss << std::endl
96  << ", Workload parameters" << std::endl
97  << ", , n, " << w_params.n();
98  } // end if
99  return ss.str();
100 }

3. Engine initialization

At this point, we can start filling out new information for our specific benchmark inside the engine. This is required to make our benchmark visible to the Test Harness since the engine is responsible for managing the benchmarks' lifespans. First we are going to add options to the TutorialEngine::init() method which implements the hebench::cpp::BaseEngine::init() virtual method. The order of initialization in the next sections can be any.

Inside of TutorialEngine::init() we will see the following code already populated from our default example:

void TutorialEngine::init()
{
// add any new error codes: use
// this->addErrorCode(code, "generic description");
// add supported schemes
addSchemeName(HEBENCH_HE_SCHEME_PLAIN, "Plain");
// add supported security
addSecurityName(HEBENCH_HE_SECURITY_NONE, "None");
// add the all benchmark descriptors
addBenchmarkDescription(std::make_shared<TutorialEltwiseAddBenchmarkDescription>());
}

For this tutorial, we will make a few changes as we plan to perform our operations homomorphically.

First, we add any new error codes which we want our engine to be capable of returning back to the frontend. For our example we define one new error as shown below. These errors are defined inside of tutorial_error_palisade.h.

70  // add any new error codes
71 
72  addErrorCode(TUTORIAL_ECODE_PALISADE_ERROR, "PALISADE error.");

Next, we add the scheme names we want to support. For our simple example we will just add BFV scheme which is predefined as part of hebench::APIBridge namespace. Any other schemes can be custom defined as well.

76  // add supported schemes
77 
78  //addSchemeName(HEBENCH_HE_SCHEME_PLAIN, "Plain");
79  addSchemeName(HEBENCH_HE_SCHEME_BFV, "BFV");

We then add our supported security levels. For our new backend we specify a flag for 128-bit security inside of tutorial_engine_palisade.h and then pass it here.

83  // add supported security
84 
85  //addSecurityName(HEBENCH_HE_SECURITY_NONE, "None");
86  addSecurityName(TUTORIAL_HE_SECURITY_128, "128 bits");

Last, in init() we add all of the supported benchmarks to the engine. For our example we will be adding just the single example benchmark but as we expand our backend and add more benchmarks, we would add them to this section in the same fashion as shown.

90  // add the all benchmark descriptors
91  addBenchmarkDescription(std::make_shared<TutorialEltwiseAddBenchmarkDescription>());

The full code for our engine init function looks like this:

67 void TutorialEngine::init()
68 {
70  // add any new error codes
71 
72  addErrorCode(TUTORIAL_ECODE_PALISADE_ERROR, "PALISADE error.");
74 
76  // add supported schemes
77 
78  //addSchemeName(HEBENCH_HE_SCHEME_PLAIN, "Plain");
79  addSchemeName(HEBENCH_HE_SCHEME_BFV, "BFV");
81 
83  // add supported security
84 
85  //addSecurityName(HEBENCH_HE_SECURITY_NONE, "None");
86  addSecurityName(TUTORIAL_HE_SECURITY_128, "128 bits");
88 
90  // add the all benchmark descriptors
91  addBenchmarkDescription(std::make_shared<TutorialEltwiseAddBenchmarkDescription>());
93 }

Tutorial steps

Tutorial Home
Preparation
Engine Initialization and Benchmark Description
Benchmark Implementation
File References