/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2013 CERN * @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch> * 2019 KiCad Developers, see AUTHORS.txt for contributors. * * This program 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 * of the License, or (at your option) any later version. * * This program 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 program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * or you may search the http://www.gnu.org website for the version 2 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ /** * @file profile.h: * @brief Simple profiling functions for measuring code execution time. */ #ifndef __TPROFILE_H #define __TPROFILE_H #include <chrono> #include <string> #include <iostream> #include <iomanip> #include <wx/log.h> /** * The class PROF_COUNTER is a small class to help profiling. * It allows the calculation of the elapsed time (in millisecondes) between * its creation (or the last call to Start() ) and the last call to Stop() */ class PROF_COUNTER { public: /** * Creates a PROF_COUNTER for measuring an elapsed time in milliseconds * @param aName = a string that will be printed in message. * @param aAutostart = true (default) to immediately start the timer */ PROF_COUNTER( const std::string& aName, bool aAutostart = true ) : m_name( aName ), m_running( false ) { if( aAutostart ) Start(); } /** * Creates a PROF_COUNTER for measuring an elapsed time in milliseconds * The counter is started and the string to print in message is left empty. */ PROF_COUNTER() { Start(); } /** * Starts or restarts the counter */ void Start() { m_running = true; m_starttime = std::chrono::high_resolution_clock::now(); m_lasttime = m_starttime; } /** * save the time when this function was called, and set the counter stane to stop */ void Stop() { if( !m_running ) return; m_stoptime = std::chrono::high_resolution_clock::now(); } /** * Print the elapsed time (in ms) to STDERR. */ void Show() { TIME_POINT display_stoptime = m_running ? std::chrono::high_resolution_clock::now() : m_stoptime; std::chrono::duration<double, std::milli> elapsed = display_stoptime - m_starttime; m_lasttime = display_stoptime; std::cerr << m_name << " took " << elapsed.count() << " ms." << std::endl; } /** * Print a message and the elapsed time (in ms) since the start time * and since the last call to STDERR. * @param aMessage is the message to print */ void Show( const std::string& aMessage ) { TIME_POINT display_stoptime = m_running ? std::chrono::high_resolution_clock::now() : m_stoptime; std::chrono::duration<double, std::milli> elapsed = display_stoptime - m_starttime; std::chrono::duration<double, std::milli> delta_time = display_stoptime - m_lasttime; std::cerr << m_name << " (" << aMessage << ") took " << elapsed.count() << " ms." << " delta " << delta_time.count() << " ms." << std::endl; m_lasttime = display_stoptime; } /** * Show the elapsed time (in ms) in a wxLogMessage window. */ void ShowDlg() { TIME_POINT display_stoptime = m_running ? std::chrono::high_resolution_clock::now() : m_stoptime; std::chrono::duration<double, std::milli> elapsed = display_stoptime - m_starttime; wxString msg; if( elapsed.count() < 1000.0 ) msg << m_name << " took " << elapsed.count() << " ms."; else msg << m_name << " took " << elapsed.count()/1000.0 << " sec."; wxLogMessage( msg ); } /** * @return the elapsed time in ms */ double msecs() const { TIME_POINT stoptime = m_running ? std::chrono::high_resolution_clock::now() : m_stoptime; std::chrono::duration<double, std::milli> elapsed = stoptime - m_starttime; return elapsed.count(); } private: std::string m_name; // a string printed in message bool m_running; typedef std::chrono::time_point<std::chrono::high_resolution_clock> TIME_POINT; TIME_POINT m_starttime, m_lasttime, m_stoptime; }; /** * Function GetRunningMicroSecs * An alternate way to calculate an elapset time (in microsecondes) to class PROF_COUNTER * @return an ever increasing indication of elapsed microseconds. * Use this by computing differences between two calls. * @author Dick Hollenbeck */ unsigned GetRunningMicroSecs(); #endif