2003-06-19 04:00:59 +00:00
|
|
|
// -*- C++ -*-
|
|
|
|
// Testing performance utilities for the C++ library testsuite.
|
|
|
|
//
|
2005-01-18 16:44:51 +00:00
|
|
|
// Copyright (C) 2003, 2004, 2005 Free Software Foundation, Inc.
|
2003-06-19 04:00:59 +00:00
|
|
|
//
|
|
|
|
// This file is part of the GNU ISO C++ Library. This library is free
|
|
|
|
// software; you can redistribute it and/or modify it under the
|
|
|
|
// terms of the GNU General Public License as published by the
|
|
|
|
// Free Software Foundation; either version 2, or (at your option)
|
|
|
|
// any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License along
|
|
|
|
// with this library; see the file COPYING. If not, write to the Free
|
2005-08-17 02:28:44 +00:00
|
|
|
// Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
|
2003-06-19 04:00:59 +00:00
|
|
|
// USA.
|
|
|
|
//
|
|
|
|
// As a special exception, you may use this file as part of a free software
|
|
|
|
// library without restriction. Specifically, if other files instantiate
|
|
|
|
// templates or use macros or inline functions from this file, or you compile
|
|
|
|
// this file and link it with other files to produce an executable, this
|
|
|
|
// file does not by itself cause the resulting executable to be covered by
|
|
|
|
// the GNU General Public License. This exception does not however
|
|
|
|
// invalidate any other reasons why the executable file might be covered by
|
|
|
|
// the GNU General Public License.
|
|
|
|
|
2003-07-05 04:05:45 +00:00
|
|
|
#ifndef _GLIBCXX_PERFORMANCE_H
|
|
|
|
#define _GLIBCXX_PERFORMANCE_H
|
2003-06-19 04:00:59 +00:00
|
|
|
|
|
|
|
#include <sys/times.h>
|
|
|
|
#include <sys/resource.h>
|
2003-06-24 19:38:36 +00:00
|
|
|
#include <cstdlib>
|
2003-06-19 04:00:59 +00:00
|
|
|
#include <string>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iomanip>
|
|
|
|
|
2003-06-24 19:38:36 +00:00
|
|
|
#ifdef __linux__
|
|
|
|
#include <malloc.h>
|
2003-08-08 07:51:54 +00:00
|
|
|
#elif defined (__FreeBSD__)
|
|
|
|
extern "C"
|
|
|
|
{
|
2004-06-07 21:56:16 +00:00
|
|
|
struct mallinfo
|
|
|
|
{
|
|
|
|
int uordblks;
|
|
|
|
int hblkhd;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct mallinfo
|
|
|
|
mallinfo(void)
|
|
|
|
{
|
|
|
|
struct mallinfo m = { (((size_t) sbrk (0) + 1023) / 1024), 0 };
|
|
|
|
return m;
|
|
|
|
}
|
2003-08-08 07:51:54 +00:00
|
|
|
}
|
2004-12-05 23:28:22 +00:00
|
|
|
#elif !defined (__hpux__)
|
2003-06-24 19:38:36 +00:00
|
|
|
extern "C"
|
|
|
|
{
|
2004-06-07 21:56:16 +00:00
|
|
|
struct mallinfo
|
|
|
|
{
|
|
|
|
int uordblks;
|
|
|
|
int hblkhd;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct mallinfo empty = { 0, 0 };
|
|
|
|
|
|
|
|
struct mallinfo
|
|
|
|
mallinfo(void)
|
|
|
|
{ return empty; }
|
2003-06-24 19:38:36 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-07-24 21:08:03 +00:00
|
|
|
namespace __gnu_test
|
2003-06-19 04:00:59 +00:00
|
|
|
{
|
|
|
|
class time_counter
|
|
|
|
{
|
2005-01-18 16:44:51 +00:00
|
|
|
private:
|
2003-06-19 04:00:59 +00:00
|
|
|
clock_t elapsed_begin;
|
|
|
|
clock_t elapsed_end;
|
|
|
|
tms tms_begin;
|
|
|
|
tms tms_end;
|
2005-01-18 16:44:51 +00:00
|
|
|
|
2003-06-19 04:00:59 +00:00
|
|
|
public:
|
2005-01-18 16:44:51 +00:00
|
|
|
explicit
|
|
|
|
time_counter() : elapsed_begin(), elapsed_end(), tms_begin(), tms_end()
|
|
|
|
{ }
|
2003-06-19 04:00:59 +00:00
|
|
|
|
|
|
|
void
|
2005-01-18 16:44:51 +00:00
|
|
|
clear() throw()
|
2003-06-19 04:00:59 +00:00
|
|
|
{
|
2005-01-18 16:44:51 +00:00
|
|
|
elapsed_begin = clock_t();
|
|
|
|
elapsed_end = clock_t();
|
|
|
|
tms_begin = tms();
|
|
|
|
tms_end = tms();
|
2003-06-19 04:00:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
start()
|
2005-01-18 16:44:51 +00:00
|
|
|
{
|
|
|
|
this->clear();
|
|
|
|
elapsed_begin = times(&tms_begin);
|
|
|
|
const clock_t err = clock_t(-1);
|
|
|
|
if (elapsed_begin == err)
|
|
|
|
std::__throw_runtime_error("time_counter::start");
|
|
|
|
}
|
2003-06-19 04:00:59 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
stop()
|
2005-01-18 16:44:51 +00:00
|
|
|
{
|
|
|
|
elapsed_end = times(&tms_end);
|
|
|
|
const clock_t err = clock_t(-1);
|
|
|
|
if (elapsed_end == err)
|
|
|
|
std::__throw_runtime_error("time_counter::stop");
|
|
|
|
}
|
2003-06-19 04:00:59 +00:00
|
|
|
|
|
|
|
size_t
|
|
|
|
real_time() const
|
|
|
|
{ return elapsed_end - elapsed_begin; }
|
|
|
|
|
|
|
|
size_t
|
|
|
|
user_time() const
|
|
|
|
{ return tms_end.tms_utime - tms_begin.tms_utime; }
|
|
|
|
|
|
|
|
size_t
|
|
|
|
system_time() const
|
|
|
|
{ return tms_end.tms_stime - tms_begin.tms_stime; }
|
|
|
|
};
|
|
|
|
|
|
|
|
class resource_counter
|
|
|
|
{
|
2004-06-07 21:56:16 +00:00
|
|
|
int who;
|
|
|
|
rusage rusage_begin;
|
|
|
|
rusage rusage_end;
|
2003-06-19 04:00:59 +00:00
|
|
|
struct mallinfo allocation_begin;
|
|
|
|
struct mallinfo allocation_end;
|
|
|
|
|
|
|
|
public:
|
|
|
|
resource_counter(int i = RUSAGE_SELF) : who(i)
|
|
|
|
{ this->clear(); }
|
|
|
|
|
|
|
|
void
|
2005-01-18 16:44:51 +00:00
|
|
|
clear() throw()
|
2003-06-19 04:00:59 +00:00
|
|
|
{
|
|
|
|
memset(&rusage_begin, 0, sizeof(rusage_begin));
|
|
|
|
memset(&rusage_end, 0, sizeof(rusage_end));
|
|
|
|
memset(&allocation_begin, 0, sizeof(allocation_begin));
|
|
|
|
memset(&allocation_end, 0, sizeof(allocation_end));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
start()
|
|
|
|
{
|
|
|
|
if (getrusage(who, &rusage_begin) != 0 )
|
|
|
|
memset(&rusage_begin, 0, sizeof(rusage_begin));
|
2003-06-24 19:38:36 +00:00
|
|
|
malloc(0); // Needed for some implementations.
|
|
|
|
allocation_begin = mallinfo();
|
2003-06-19 04:00:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
stop()
|
|
|
|
{
|
|
|
|
if (getrusage(who, &rusage_end) != 0 )
|
|
|
|
memset(&rusage_end, 0, sizeof(rusage_end));
|
2003-06-24 19:38:36 +00:00
|
|
|
allocation_end = mallinfo();
|
2003-06-19 04:00:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
allocated_memory() const
|
2004-06-07 21:56:16 +00:00
|
|
|
{ return ((allocation_end.uordblks - allocation_begin.uordblks)
|
|
|
|
+ (allocation_end.hblkhd - allocation_begin.hblkhd)); }
|
2003-06-19 04:00:59 +00:00
|
|
|
|
|
|
|
long
|
|
|
|
hard_page_fault() const
|
|
|
|
{ return rusage_end.ru_majflt - rusage_begin.ru_majflt; }
|
|
|
|
|
|
|
|
long
|
|
|
|
swapped() const
|
|
|
|
{ return rusage_end.ru_nswap - rusage_begin.ru_nswap; }
|
|
|
|
};
|
|
|
|
|
2005-01-18 16:44:51 +00:00
|
|
|
inline void
|
2003-06-19 04:00:59 +00:00
|
|
|
start_counters(time_counter& t, resource_counter& r)
|
|
|
|
{
|
|
|
|
t.start();
|
|
|
|
r.start();
|
|
|
|
}
|
|
|
|
|
2005-01-18 16:44:51 +00:00
|
|
|
inline void
|
2003-06-19 04:00:59 +00:00
|
|
|
stop_counters(time_counter& t, resource_counter& r)
|
|
|
|
{
|
|
|
|
t.stop();
|
|
|
|
r.stop();
|
|
|
|
}
|
|
|
|
|
2005-01-18 16:44:51 +00:00
|
|
|
inline void
|
2003-06-19 04:00:59 +00:00
|
|
|
clear_counters(time_counter& t, resource_counter& r)
|
|
|
|
{
|
|
|
|
t.clear();
|
|
|
|
r.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
report_performance(const std::string file, const std::string comment,
|
|
|
|
const time_counter& t, const resource_counter& r)
|
|
|
|
{
|
|
|
|
const char space = ' ';
|
|
|
|
const char tab = '\t';
|
2003-08-08 15:24:00 +00:00
|
|
|
const char* name = "libstdc++-performance.sum";
|
2003-06-19 04:00:59 +00:00
|
|
|
std::string::const_iterator i = file.begin() + file.find_last_of('/') + 1;
|
|
|
|
std::string testname(i, file.end());
|
|
|
|
|
|
|
|
std::ofstream out(name, std::ios_base::app);
|
|
|
|
|
2004-01-30 08:24:27 +00:00
|
|
|
#ifdef __GTHREADS
|
2005-01-18 16:44:51 +00:00
|
|
|
if (__gthread_active_p())
|
|
|
|
testname.append("-thread");
|
2004-01-30 08:24:27 +00:00
|
|
|
#endif
|
|
|
|
|
2003-06-19 04:00:59 +00:00
|
|
|
out.setf(std::ios_base::left);
|
|
|
|
out << std::setw(25) << testname << tab;
|
2003-12-17 10:16:03 +00:00
|
|
|
out << std::setw(25) << comment << tab;
|
2003-06-19 04:00:59 +00:00
|
|
|
|
|
|
|
out.setf(std::ios_base::right);
|
|
|
|
out << std::setw(4) << t.real_time() << "r" << space;
|
|
|
|
out << std::setw(4) << t.user_time() << "u" << space;
|
|
|
|
out << std::setw(4) << t.system_time() << "s" << space;
|
2003-12-17 10:16:03 +00:00
|
|
|
out << std::setw(8) << r.allocated_memory() << "mem" << space;
|
2003-06-19 04:00:59 +00:00
|
|
|
out << std::setw(4) << r.hard_page_fault() << "pf" << space;
|
|
|
|
|
|
|
|
out << std::endl;
|
|
|
|
out.close();
|
|
|
|
}
|
2003-12-16 22:57:57 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
report_header(const std::string file, const std::string header)
|
|
|
|
{
|
|
|
|
const char space = ' ';
|
|
|
|
const char tab = '\t';
|
|
|
|
const char* name = "libstdc++-performance.sum";
|
|
|
|
std::string::const_iterator i = file.begin() + file.find_last_of('/') + 1;
|
|
|
|
std::string testname(i, file.end());
|
|
|
|
|
|
|
|
std::ofstream out(name, std::ios_base::app);
|
|
|
|
|
2004-01-30 08:24:27 +00:00
|
|
|
#ifdef __GTHREADS
|
|
|
|
if (__gthread_active_p ())
|
2005-01-18 16:44:51 +00:00
|
|
|
testname.append("-thread");
|
2004-01-30 08:24:27 +00:00
|
|
|
#endif
|
|
|
|
|
2003-12-16 22:57:57 +00:00
|
|
|
out.setf(std::ios_base::left);
|
|
|
|
out << std::setw(25) << testname << tab;
|
2003-12-17 10:16:03 +00:00
|
|
|
out << std::setw(40) << header << tab;
|
2003-12-16 22:57:57 +00:00
|
|
|
|
|
|
|
out << std::endl;
|
|
|
|
out.close();
|
|
|
|
}
|
2003-07-24 21:08:03 +00:00
|
|
|
}; // namespace __gnu_test
|
2003-06-19 04:00:59 +00:00
|
|
|
|
2003-07-05 04:05:45 +00:00
|
|
|
#endif // _GLIBCXX_PERFORMANCE_H
|
2003-06-19 04:00:59 +00:00
|
|
|
|