Fichiers
Rawat, Swati edb51fc861 update copyright date to 2025 (#102)
* Update LICENSE

* Update conf.py

* Update copyright year

* [fix] Update copyright year

* Update copyright year "ROCm Developer Tools"

* Add license headers to c++ files

* Add license to *.py

* Update licenses in rocdecode sources

---------

Co-authored-by: srawat <120587655+SwRaw@users.noreply.github.com>
Co-authored-by: Mythreya <mythreya.kuricheti@amd.com>
Co-authored-by: Jonathan R. Madsen <jonathanrmadsen@gmail.com>

[ROCm/rocprofiler-sdk commit: 97b7a6315d]
2025-01-22 19:11:20 -06:00

352 lignes
9.7 KiB
C++

// MIT License
//
// Copyright (c) 2023-2025 Advanced Micro Devices, Inc. All rights reserved.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
#pragma once
#include "domain_type.hpp"
#include "lib/common/logging.hpp"
#include "lib/common/mpl.hpp"
#include <rocprofiler-sdk/cxx/serialization.hpp>
#include <cmath>
#include <cstdint>
#include <fstream>
#include <functional>
#include <iomanip>
#include <iostream>
#include <limits>
#include <map>
#include <type_traits>
#include <vector>
namespace rocprofiler
{
namespace tool
{
/// \struct statistics
/// \tparam Tp data type for statistical accumulation
/// \tparam Fp floating point data type to use for division
/// \brief A generic class for statistical accumulation.
///
template <typename Tp, typename Fp = double>
struct statistics
{
public:
using value_type = Tp;
using float_type = Fp;
using this_type = statistics<Tp, Fp>;
static_assert(std::is_arithmetic<Tp>::value, "only supports arithmetic types");
public:
statistics() = default;
~statistics() = default;
statistics(const statistics&) = default;
statistics(statistics&&) noexcept = default;
statistics& operator=(const statistics&) = default;
statistics& operator=(statistics&&) noexcept = default;
explicit statistics(value_type val)
: m_cnt(1)
, m_sum(val)
, m_sqr(val * val)
, m_min(val)
, m_max(val)
{}
statistics& operator=(value_type val)
{
m_cnt = 1;
m_sum = val;
m_min = val;
m_max = val;
m_sqr = (val * val);
return *this;
}
public:
// Accumulated values
int64_t get_count() const { return m_cnt; }
value_type get_min() const { return m_min; }
value_type get_max() const { return m_max; }
value_type get_sum() const { return m_sum; }
value_type get_sqr() const { return m_sqr; }
float_type get_mean() const { return static_cast<float_type>(m_sum) / m_cnt; }
float_type get_variance() const
{
if(m_cnt < 2) return (m_sum - m_sum);
auto _sum_of_squared_samples = m_sqr;
auto _sum_squared_mean = (m_sum * m_sum) / static_cast<float_type>(m_cnt);
return (_sum_of_squared_samples - _sum_squared_mean) / static_cast<float_type>(m_cnt - 1);
}
float_type get_stddev() const { return ::std::sqrt(::std::abs(get_variance())); }
float_type get_percent(float_type _total) const;
float_type get_percent(const this_type&) const;
// Modifications
void reset()
{
m_cnt = 0;
m_sum = value_type{};
m_sqr = value_type{};
m_min = value_type{};
m_max = value_type{};
}
public:
// Operators (value_type)
statistics& operator+=(value_type val)
{
if(m_cnt == 0)
{
m_sum = val;
m_sqr = (val * val);
m_min = val;
m_max = val;
}
else
{
m_sum += val;
m_sqr += (val * val);
m_min = ::std::min(m_min, val);
m_max = ::std::max(m_max, val);
}
++m_cnt;
return *this;
}
statistics& operator-=(value_type val)
{
if(m_cnt > 1) --m_cnt;
m_sum -= val;
m_sqr -= (val * val);
m_min -= val;
m_max -= val;
return *this;
}
statistics& operator*=(value_type val)
{
m_sum *= val;
m_sqr *= (val * val);
m_min *= val;
m_max *= val;
return *this;
}
statistics& operator/=(value_type val)
{
m_sum /= val;
m_sqr /= (val * val);
m_min /= val;
m_max /= val;
return *this;
}
public:
// Operators (this_type)
statistics& operator+=(const statistics& rhs)
{
if(m_cnt == 0)
{
m_sum = rhs.m_sum;
m_sqr = rhs.m_sqr;
m_min = rhs.m_min;
m_max = rhs.m_max;
}
else
{
m_sum += rhs.m_sum;
m_sqr += rhs.m_sqr;
m_min = ::std::min(m_min, rhs.m_min);
m_max = ::std::max(m_max, rhs.m_max);
}
m_cnt += rhs.m_cnt;
return *this;
}
// Operators (this_type)
statistics& operator-=(const statistics& rhs)
{
if(m_cnt > 0)
{
m_sum -= rhs.m_sum;
m_sqr -= rhs.m_sqr;
m_min = ::std::min(m_min, rhs.m_min);
m_max = ::std::max(m_max, rhs.m_max);
}
return *this;
}
private:
// summation of each history^1
int64_t m_cnt = 0;
value_type m_sum = value_type{};
value_type m_sqr = value_type{};
value_type m_min = value_type{};
value_type m_max = value_type{};
public:
// friend operator for addition
friend statistics operator+(const statistics& lhs, const statistics& rhs)
{
return statistics(lhs) += rhs;
}
friend statistics operator-(const statistics& lhs, const statistics& rhs)
{
return statistics(lhs) -= rhs;
}
template <typename ArchiveT>
void serialize(ArchiveT& ar, const unsigned int) const
{
ar(cereal::make_nvp("count", m_cnt));
ar(cereal::make_nvp("sum", m_sum));
ar(cereal::make_nvp("sqr", m_sqr));
ar(cereal::make_nvp("min", m_min));
ar(cereal::make_nvp("max", m_max));
ar(cereal::make_nvp("mean", get_mean()));
ar(cereal::make_nvp("stddev", get_stddev()));
ar(cereal::make_nvp("variance", get_variance()));
}
};
template <typename Tp, typename Fp>
typename statistics<Tp, Fp>::float_type
statistics<Tp, Fp>::get_percent(float_type _total) const
{
constexpr float_type one_hundred = 100.0;
const float_type _sum = get_sum();
ROCP_WARNING_IF(static_cast<int64_t>(_sum) > static_cast<int64_t>(_total))
<< "percentage calculation > 100%. sum=" << _sum << " > total=" << _total;
return (_sum / _total) * one_hundred;
}
template <typename Tp, typename Fp>
typename statistics<Tp, Fp>::float_type
statistics<Tp, Fp>::get_percent(const statistics<Tp, Fp>& _rhs) const
{
return get_percent(_rhs.get_sum());
}
using float_type = double;
using stats_data_t = statistics<uint64_t, float_type>;
using stats_map_t = std::map<std::string_view, stats_data_t>;
using stats_pair_t = std::pair<std::string_view, stats_data_t>;
using stats_entry_vec_t = std::vector<stats_pair_t>;
inline bool
default_stats_sorter(const stats_pair_t& lhs, const stats_pair_t& rhs)
{
return (lhs.second.get_sum() > rhs.second.get_sum());
}
struct stats_entry_t
{
using sort_predicate_t = bool (*)(const stats_pair_t&, const stats_pair_t&);
stats_entry_t() = default;
~stats_entry_t() = default;
stats_entry_t(const stats_entry_t&) = default;
stats_entry_t(stats_entry_t&&) noexcept = default;
stats_entry_t& operator=(const stats_entry_t&) = default;
stats_entry_t& operator=(stats_entry_t&&) noexcept = default;
template <typename FuncT = sort_predicate_t>
stats_entry_t& sort(FuncT&& _predicate = default_stats_sorter);
explicit operator bool() const { return (total.get_count() > 0 && !entries.empty()); }
stats_data_t total = {};
stats_entry_vec_t entries = {};
template <typename ArchiveT>
void serialize(ArchiveT& ar, const unsigned int) const
{
total.serialize(ar, 0);
auto _entries_map = std::map<std::string, stats_data_t>{};
for(const auto& itr : entries)
_entries_map.emplace(std::string{itr.first}, itr.second);
ar(cereal::make_nvp("operations", _entries_map));
}
};
template <typename FuncT>
stats_entry_t&
stats_entry_t::sort(FuncT&& _predicate)
{
std::sort(entries.begin(), entries.end(), std::forward<FuncT>(_predicate));
return *this;
}
using domain_stats_t = std::pair<domain_type, stats_entry_t>;
using domain_stats_vec_t = std::vector<domain_stats_t>;
struct stats_formatter
{
template <typename Tp>
std::ostream& operator()(std::ostream& ofs, const Tp& _val) const;
};
struct percentage
{
float_type value = {};
friend std::ostream& operator<<(std::ostream& os, percentage val)
{
return (stats_formatter{}(os, val) << val.value);
}
};
} // namespace tool
} // namespace rocprofiler
namespace std
{
template <typename Tp>
::rocprofiler::tool::statistics<Tp>
max(::rocprofiler::tool::statistics<Tp> lhs, const Tp& rhs)
{
return lhs.get_max(rhs);
}
template <typename Tp>
::rocprofiler::tool::statistics<Tp>
min(::rocprofiler::tool::statistics<Tp> lhs, const Tp& rhs)
{
return lhs.get_min(rhs);
}
inline std::string
to_string(::rocprofiler::tool::percentage val)
{
auto _ss = std::stringstream{};
_ss << val;
return _ss.str();
}
} // namespace std