mirror of
https://github.com/moparisthebest/mailiverse
synced 2025-01-05 18:48:08 -05:00
255 lines
6.8 KiB
C++
255 lines
6.8 KiB
C++
/*
|
|
* (C) 2009 Jack Lloyd
|
|
*
|
|
* Distributed under the terms of the Botan license
|
|
*/
|
|
|
|
/*
|
|
* Test Driver for Botan
|
|
*/
|
|
|
|
#include <vector>
|
|
#include <string>
|
|
|
|
#include <iostream>
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include <exception>
|
|
#include <limits>
|
|
#include <memory>
|
|
|
|
#include <botan/botan.h>
|
|
#include <botan/libstate.h>
|
|
|
|
#if defined(BOTAN_HAS_DYNAMICALLY_LOADED_ENGINE)
|
|
#include <botan/dyn_engine.h>
|
|
#endif
|
|
|
|
using namespace Botan;
|
|
|
|
#include "getopt.h"
|
|
#include "bench.h"
|
|
#include "validate.h"
|
|
#include "common.h"
|
|
|
|
const std::string VALIDATION_FILE = "checks/validate.dat";
|
|
const std::string BIGINT_VALIDATION_FILE = "checks/mp_valid.dat";
|
|
const std::string PK_VALIDATION_FILE = "checks/pk_valid.dat";
|
|
const std::string EXPECTED_FAIL_FILE = "checks/fail.dat";
|
|
|
|
int run_test_suite(RandomNumberGenerator& rng);
|
|
|
|
namespace {
|
|
|
|
template<typename T>
|
|
bool test(const char* type, int digits, bool is_signed)
|
|
{
|
|
if(std::numeric_limits<T>::is_specialized == false)
|
|
{
|
|
std::cout << "Warning: Could not check parameters of " << type
|
|
<< " in std::numeric_limits" << std::endl;
|
|
|
|
// assume it's OK (full tests will catch it later)
|
|
return true;
|
|
}
|
|
|
|
// continue checking after failures
|
|
bool passed = true;
|
|
|
|
if(std::numeric_limits<T>::is_integer == false)
|
|
{
|
|
std::cout << "Warning: std::numeric_limits<> says " << type
|
|
<< " is not an integer" << std::endl;
|
|
passed = false;
|
|
}
|
|
|
|
if(std::numeric_limits<T>::is_signed != is_signed)
|
|
{
|
|
std::cout << "Warning: numeric_limits<" << type << ">::is_signed == "
|
|
<< std::boolalpha << std::numeric_limits<T>::is_signed
|
|
<< std::endl;
|
|
passed = false;
|
|
}
|
|
|
|
if(std::numeric_limits<T>::digits != digits && digits != 0)
|
|
{
|
|
std::cout << "Warning: numeric_limits<" << type << ">::digits == "
|
|
<< std::numeric_limits<T>::digits
|
|
<< " expected " << digits << std::endl;
|
|
passed = false;
|
|
}
|
|
|
|
return passed;
|
|
}
|
|
|
|
void test_types()
|
|
{
|
|
bool passed = true;
|
|
|
|
passed = passed && test<Botan::byte >("byte", 8, false);
|
|
passed = passed && test<Botan::u16bit>("u16bit", 16, false);
|
|
passed = passed && test<Botan::u32bit>("u32bit", 32, false);
|
|
passed = passed && test<Botan::u64bit>("u64bit", 64, false);
|
|
passed = passed && test<Botan::s32bit>("s32bit", 31, true);
|
|
|
|
if(!passed)
|
|
std::cout << "Typedefs in include/types.h may be incorrect!\n";
|
|
}
|
|
|
|
}
|
|
|
|
int main(int argc, char* argv[])
|
|
{
|
|
if(BOTAN_VERSION_MAJOR != version_major() ||
|
|
BOTAN_VERSION_MINOR != version_minor() ||
|
|
BOTAN_VERSION_PATCH != version_patch())
|
|
{
|
|
std::cout << "Warning: linked version ("
|
|
<< version_major() << '.'
|
|
<< version_minor() << '.'
|
|
<< version_patch()
|
|
<< ") does not match version built against ("
|
|
<< BOTAN_VERSION_MAJOR << '.'
|
|
<< BOTAN_VERSION_MINOR << '.'
|
|
<< BOTAN_VERSION_PATCH << ")\n";
|
|
}
|
|
|
|
try
|
|
{
|
|
OptionParser opts("help|test|validate|dyn-load=|"
|
|
"benchmark|bench-type=|bench-algo=|"
|
|
"seconds=|buf-size=");
|
|
opts.parse(argv);
|
|
|
|
test_types(); // do this always
|
|
|
|
Botan::LibraryInitializer init("thread_safe=no");
|
|
|
|
if(opts.is_set("dyn-load"))
|
|
{
|
|
const std::string lib = opts.value("dyn-load");
|
|
|
|
#if defined(BOTAN_HAS_DYNAMICALLY_LOADED_ENGINE)
|
|
global_state().algorithm_factory().add_engine(
|
|
new Dynamically_Loaded_Engine(lib));
|
|
#else
|
|
std::cout << "Can't load " << lib
|
|
<< "; DLL engines not supported in build\n";
|
|
#endif
|
|
}
|
|
|
|
if(opts.is_set("help") || argc <= 1)
|
|
{
|
|
std::cerr << "Test driver for "
|
|
<< Botan::version_string() << "\n"
|
|
<< "Options:\n"
|
|
<< " --test || --validate: Run tests (do this at least once)\n"
|
|
<< " --benchmark: Benchmark everything\n"
|
|
<< " --seconds=n: Benchmark for n seconds\n"
|
|
<< " --init=<str>: Pass <str> to the library\n"
|
|
<< " --help: Print this message\n";
|
|
return 1;
|
|
}
|
|
|
|
Botan::AutoSeeded_RNG rng;
|
|
|
|
if(opts.is_set("validate") || opts.is_set("test"))
|
|
{
|
|
return run_test_suite(rng);
|
|
}
|
|
if(opts.is_set("bench-algo") ||
|
|
opts.is_set("benchmark") ||
|
|
opts.is_set("bench-type"))
|
|
{
|
|
double seconds = 5;
|
|
|
|
u32bit buf_size = 16;
|
|
|
|
if(opts.is_set("seconds"))
|
|
{
|
|
seconds = std::atof(opts.value("seconds").c_str());
|
|
if(seconds < 0.1 || seconds > (5 * 60))
|
|
{
|
|
std::cout << "Invalid argument to --seconds\n";
|
|
return 2;
|
|
}
|
|
}
|
|
|
|
if(opts.is_set("buf-size"))
|
|
{
|
|
buf_size = std::atoi(opts.value("buf-size").c_str());
|
|
if(buf_size == 0 || buf_size > 1024)
|
|
{
|
|
std::cout << "Invalid argument to --buf-size\n";
|
|
return 2;
|
|
}
|
|
}
|
|
|
|
if(opts.is_set("benchmark"))
|
|
{
|
|
benchmark(rng, seconds, buf_size);
|
|
}
|
|
else if(opts.is_set("bench-algo"))
|
|
{
|
|
std::vector<std::string> algs =
|
|
Botan::split_on(opts.value("bench-algo"), ',');
|
|
|
|
for(u32bit j = 0; j != algs.size(); j++)
|
|
{
|
|
const std::string alg = algs[j];
|
|
if(!bench_algo(alg, rng, seconds, buf_size))
|
|
bench_pk(rng, alg, seconds);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch(std::exception& e)
|
|
{
|
|
std::cerr << "Exception: " << e.what() << std::endl;
|
|
return 1;
|
|
}
|
|
catch(...)
|
|
{
|
|
std::cerr << "Unknown (...) exception caught" << std::endl;
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int run_test_suite(RandomNumberGenerator& rng)
|
|
{
|
|
std::cout << "Beginning tests..." << std::endl;
|
|
|
|
u32bit errors = 0;
|
|
try
|
|
{
|
|
errors += do_validation_tests(VALIDATION_FILE, rng);
|
|
errors += do_validation_tests(EXPECTED_FAIL_FILE, rng, false);
|
|
errors += do_bigint_tests(BIGINT_VALIDATION_FILE, rng);
|
|
errors += do_pk_validation_tests(PK_VALIDATION_FILE, rng);
|
|
do_x509_tests(rng);
|
|
//errors += do_cvc_tests(rng);
|
|
}
|
|
catch(std::exception& e)
|
|
{
|
|
std::cout << "Exception: " << e.what() << std::endl;
|
|
return 1;
|
|
}
|
|
catch(...)
|
|
{
|
|
std::cout << "Unknown exception caught." << std::endl;
|
|
return 1;
|
|
}
|
|
|
|
if(errors)
|
|
{
|
|
std::cout << errors << " test" << ((errors == 1) ? "" : "s")
|
|
<< " failed." << std::endl;
|
|
return 1;
|
|
}
|
|
|
|
std::cout << "All tests passed!" << std::endl;
|
|
return 0;
|
|
}
|