xaizek / zograscope (License: AGPLv3 only) (since 2018-12-07)
Mainly a syntax-aware diff that also provides a number of additional tools.
<root> / src / utils / time.cpp (6b4b35a9045971f7412d402607dc9eaca2f9ed3a) (2,620B) (mode 100644) [raw]
// Copyright (C) 2018 xaizek <xaizek@posteo.net>
//
// This file is part of zograscope.
//
// zograscope is free software: you can redistribute it and/or modify
// it under the terms of version 3 of the GNU Affero General Public License as
// published by the Free Software Foundation.
//
// zograscope 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with zograscope.  If not, see <http://www.gnu.org/licenses/>.

#include "utils/time.hpp"

#include <boost/scope_exit.hpp>

#include <iomanip>
#include <ostream>

std::ostream &
operator<<(std::ostream &os, const TimeReport &tr)
{
    using Measure = TimeReport::Measure;

    struct MeasureTraits
    {
        static unsigned int size(const Measure *node)
        {
            return node->children.size();
        }

        static const Measure * getChild(const Measure *node, unsigned int i)
        {
            return &node->children[i];
        }
    };

    using msf = std::chrono::duration<float, std::milli>;

    auto osState = os.rdstate();
    BOOST_SCOPE_EXIT_ALL(&os, &osState) { os.setstate(osState); };

    os << std::fixed << std::setprecision(3);

    trees::printSetTraits<MeasureTraits>(os, &tr.root,
                 [](std::ostream &os, const Measure *m) {
                     msf duration = m->end - m->start;
                     if (m->foreign) {
                         os << "+ ";
                     }
                     os << m->stage << " -- " << duration.count() << "ms";

                     if (m->children.empty()) {
                         os << '\n';
                         return;
                     }

                     msf accounted = {};
                     msf foreign = {};
                     for (const Measure &child : m->children) {
                         if (child.foreign) {
                             foreign += child.end - child.start;
                         } else {
                             accounted += child.end - child.start;
                         }
                     }
                     msf unaccounted = duration - accounted;
                     os << " (-" << unaccounted.count() << "ms";
                     if (foreign.count()) {
                        os << ", +" << foreign.count() << "ms";
                     }
                     os << ")\n";
                 });

    return os;
}
Hints

Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://code.reversed.top/user/xaizek/zograscope

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@code.reversed.top/user/xaizek/zograscope

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a pull request:
... clone the repository ...
... make some changes and some commits ...
git push origin master