123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- // Copyright (C) 2012 Internet Systems Consortium, Inc. ("ISC")
- //
- // Permission to use, copy, modify, and/or distribute this software for any
- // purpose with or without fee is hereby granted, provided that the above
- // copyright notice and this permission notice appear in all copies.
- //
- // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
- // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
- // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
- // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
- // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
- // PERFORMANCE OF THIS SOFTWARE.
- #include <iostream>
- #include <stdlib.h>
- #include <string.h>
- #include <boost/lexical_cast.hpp>
- #include "benchmark.h"
- // The following headers are for getting precise time (clock_gettime on Linux or that mac os thingie)
- #include <time.h>
- #include <sys/time.h>
- #ifdef __MACH__
- #include <mach/clock.h>
- #include <mach/mach.h>
- #endif
- using namespace std;
- uBenchmark::uBenchmark(uint32_t iterations, const std::string& dbname,
- bool sync /*= false*/, bool verbose /*= true*/,
- const std::string& host /* = "" */,
- const std::string& user /* = "" */,
- const std::string& pass /* = "" */)
- :num_(iterations), sync_(sync), verbose_(verbose),
- hostname_(host), user_(user), passwd_(pass), dbname_(dbname),
- hitratio_(0.9f), compiled_stmt_(true)
- {
- /// @todo: make compiled statements a configurable parameter
- /// @todo: convert hitratio_ to user-configurable parameter
- memset(ts_, 0, sizeof(ts_));
- }
- void uBenchmark::usage() {
- cout << "This is a benchmark designed to measure expected performance" << endl;
- cout << "of several backends. This backend identifies itself as:" << endl;
- printInfo();
- cout << endl << "Possible command-line parameters:" << endl;
- cout << " -h - help (you are reading this)" << endl;
- cout << " -m hostname - specifies MySQL server to connect (MySQL backend only)" << endl;
- cout << " -u username - specifies MySQL user name (MySQL backend only)" << endl;
- cout << " -p password - specifies MySQL passwod (MySQL backend only)" << endl;
- cout << " -f name - database or filename (MySQL, SQLite and memfile)" << endl;
- cout << " -n integer - number of test iterations (MySQL, SQLite and memfile)" << endl;
- cout << " -s yes|no - synchronous/asynchronous operation (MySQL, SQLite and memfile)" << endl;
- cout << " -v yes|no - verbose mode (MySQL, SQLite and memfile)" << endl;
- cout << " -c yes|no - compiled statements (MySQL and SQLite)" << endl;
- exit(EXIT_FAILURE);
- }
- void uBenchmark::parseCmdline(int argc, char* const argv[]) {
- int ch;
- while ((ch = getopt(argc, argv, "hm:u:p:f:n:s:v:c:")) != -1) {
- switch (ch) {
- case 'h':
- usage();
- case 'm':
- hostname_ = string(optarg);
- break;
- case 'u':
- user_ = string(optarg);
- break;
- case 'p':
- passwd_ = string(optarg);
- break;
- case 'f':
- dbname_ = string(optarg);
- break;
- case 'n':
- try {
- num_ = boost::lexical_cast<unsigned int>(optarg);
- } catch (const boost::bad_lexical_cast &) {
- cerr << "Failed to parse number of iterations (-n option):"
- << optarg << endl;
- usage();
- }
- break;
- case 'c':
- compiled_stmt_ = !strcasecmp(optarg, "yes") || !strcmp(optarg, "1");
- break;
- case 's':
- sync_ = !strcasecmp(optarg, "yes") || !strcmp(optarg, "1");
- break;
- case 'v':
- verbose_ = !strcasecmp(optarg, "yes") || !strcmp(optarg, "1");
- break;
- default:
- usage();
- }
- }
- }
- void uBenchmark::failure(const char* operation) {
- cout << "Error during " << operation << endl;
- throw string(operation);
- }
- void uBenchmark::printClock(const std::string& operation, uint32_t num,
- const struct timespec& before,
- const struct timespec& after) {
- long int tv_sec = after.tv_sec - before.tv_sec;
- long int tv_nsec = after.tv_nsec - before.tv_nsec;
- if (tv_nsec < 0) {
- tv_sec--;
- tv_nsec += 1000000000; // 10^9
- }
- double oneoper = (tv_nsec/1000 + tv_sec*1000000)/num;
- cout << operation << " repeated " << num << " times took "
- << tv_sec << " s, " << tv_nsec/1000 << " us, 1 operation took "
- << oneoper << "us (or " << (1000000/oneoper) << " oper/sec)" << endl;
- }
- int uBenchmark::run() {
- cout << "Starting test. Parameters:" << endl
- << "Number of iterations : " << num_ << endl
- << "Sync/async : " << (sync_ ? "sync" : "async") << endl
- << "Verbose : " << (verbose_ ? "verbose" : "quiet") << endl
- << "Compiled statements : " << (compiled_stmt_ ? "yes": "no") << endl
- << "Database name : " << dbname_ << endl
- << "MySQL hostname : " << hostname_ << endl
- << "MySQL username : " << user_ << endl
- << "MySQL password : " << passwd_ << endl << endl;
- srandom(time(NULL));
- try {
- connect();
- ts_[0] = getTime();
- createLease4Test();
- ts_[1] = getTime();
- searchLease4Test();
- ts_[2] = getTime();
- updateLease4Test();
- ts_[3] = getTime();
- deleteLease4Test();
- ts_[4] = getTime();
- disconnect();
- } catch (const std::string& e) {
- cout << "Failed: " << e << endl;
- return (-1);
- }
- printClock("Create leases4", num_, ts_[0], ts_[1]);
- printClock("Search leases4", num_, ts_[1], ts_[2]);
- printClock("Update leases4", num_, ts_[2], ts_[3]);
- printClock("Delete leases4", num_, ts_[3], ts_[4]);
- return (0);
- }
- struct timespec uBenchmark::getTime() {
- struct timespec ts;
- #ifdef __MACH__ // OS X does not have clock_gettime, use clock_get_time
- clock_serv_t cclock;
- mach_timespec_t mts;
- host_get_clock_service(mach_host_self(), CALENDAR_CLOCK, &cclock);
- clock_get_time(cclock, &mts);
- mach_port_deallocate(mach_task_self(), cclock);
- ts.tv_sec = mts.tv_sec;
- ts.tv_nsec = mts.tv_nsec;
- #else
- clock_gettime(CLOCK_REALTIME, &ts);
- #endif
- return ts;
- }
|