2023-11-14 10:58:33 -06:00
|
|
|
// MIT License
|
|
|
|
|
//
|
2025-01-23 06:41:20 +05:30
|
|
|
// Copyright (c) 2023-2025 Advanced Micro Devices, Inc.
|
2023-08-08 18:39:01 -05:00
|
|
|
//
|
|
|
|
|
// 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
|
|
|
|
|
|
2024-05-20 15:38:18 -05:00
|
|
|
#include "lib/common/logging.hpp"
|
2023-08-08 18:39:01 -05:00
|
|
|
|
2023-08-24 19:19:48 -05:00
|
|
|
#include <unistd.h>
|
2023-08-08 18:39:01 -05:00
|
|
|
#include <string>
|
|
|
|
|
#include <string_view>
|
|
|
|
|
#include <type_traits>
|
|
|
|
|
|
|
|
|
|
namespace rocprofiler
|
|
|
|
|
{
|
|
|
|
|
namespace common
|
|
|
|
|
{
|
2023-08-30 22:31:36 -05:00
|
|
|
namespace impl
|
2023-08-08 18:39:01 -05:00
|
|
|
{
|
2024-01-02 16:12:02 -03:00
|
|
|
struct sfinae
|
|
|
|
|
{};
|
|
|
|
|
|
2023-08-30 22:31:36 -05:00
|
|
|
std::string get_env(std::string_view, std::string_view);
|
2023-08-08 18:39:01 -05:00
|
|
|
|
2023-08-30 22:31:36 -05:00
|
|
|
std::string
|
|
|
|
|
get_env(std::string_view, const char*);
|
2023-08-08 18:39:01 -05:00
|
|
|
|
2023-08-30 22:31:36 -05:00
|
|
|
bool
|
|
|
|
|
get_env(std::string_view, bool);
|
2024-01-02 16:12:02 -03:00
|
|
|
|
|
|
|
|
template <typename Tp>
|
|
|
|
|
Tp get_env(std::string_view, Tp, std::enable_if_t<std::is_integral<Tp>::value, sfinae> = {});
|
2024-03-07 22:21:26 -06:00
|
|
|
|
|
|
|
|
int
|
|
|
|
|
set_env(std::string_view, bool, int override = 0);
|
|
|
|
|
|
|
|
|
|
template <typename Tp>
|
|
|
|
|
int
|
|
|
|
|
set_env(std::string_view, Tp, int override = 0);
|
2023-08-30 22:31:36 -05:00
|
|
|
} // namespace impl
|
2023-08-08 18:39:01 -05:00
|
|
|
|
|
|
|
|
template <typename Tp>
|
|
|
|
|
inline auto
|
|
|
|
|
get_env(std::string_view env_id, Tp&& _default)
|
|
|
|
|
{
|
|
|
|
|
if constexpr(std::is_enum<Tp>::value)
|
|
|
|
|
{
|
|
|
|
|
using Up = std::underlying_type_t<Tp>;
|
|
|
|
|
// cast to underlying type -> get_env -> cast to enum type
|
2023-08-30 22:31:36 -05:00
|
|
|
return static_cast<Tp>(impl::get_env(env_id, static_cast<Up>(_default)));
|
2023-08-08 18:39:01 -05:00
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2023-08-30 22:31:36 -05:00
|
|
|
return impl::get_env(env_id, std::forward<Tp>(_default));
|
2023-08-08 18:39:01 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-07 22:21:26 -06:00
|
|
|
template <typename Tp>
|
|
|
|
|
inline auto
|
|
|
|
|
set_env(std::string_view env_id, Tp&& value, int override = 0)
|
|
|
|
|
{
|
|
|
|
|
return impl::set_env(env_id, std::forward<Tp>(value), override);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-08 18:39:01 -05:00
|
|
|
struct env_config
|
|
|
|
|
{
|
|
|
|
|
std::string env_name = {};
|
|
|
|
|
std::string env_value = {};
|
2023-08-30 22:31:36 -05:00
|
|
|
int overwrite = 0;
|
2023-08-08 18:39:01 -05:00
|
|
|
|
|
|
|
|
auto operator()(bool _verbose = false) const
|
|
|
|
|
{
|
2025-05-30 00:13:19 -05:00
|
|
|
if(env_name.empty())
|
|
|
|
|
return -1;
|
|
|
|
|
else if(_verbose)
|
|
|
|
|
{
|
|
|
|
|
ROCP_INFO << "[rocprofiler][set_env] setenv(\"" << env_name << "\", \"" << env_value
|
|
|
|
|
<< "\", " << overwrite << ")\n";
|
|
|
|
|
}
|
|
|
|
|
return (env_value.empty() && overwrite > 0)
|
|
|
|
|
? unsetenv(env_name.c_str())
|
|
|
|
|
: setenv(env_name.c_str(), env_value.c_str(), overwrite);
|
2023-08-08 18:39:01 -05:00
|
|
|
}
|
|
|
|
|
};
|
2025-05-30 00:13:19 -05:00
|
|
|
|
|
|
|
|
struct env_store
|
|
|
|
|
{
|
|
|
|
|
template <template <typename, typename...> class ContainerT, typename... TailT>
|
|
|
|
|
explicit env_store(ContainerT<env_config, TailT...>&& _container);
|
|
|
|
|
explicit env_store(std::initializer_list<env_config>&& _container);
|
|
|
|
|
|
|
|
|
|
~env_store();
|
|
|
|
|
env_store(const env_store&) = default;
|
|
|
|
|
env_store(env_store&&) noexcept = default;
|
|
|
|
|
env_store& operator=(const env_store&) = default;
|
|
|
|
|
env_store& operator=(env_store&&) noexcept = default;
|
|
|
|
|
|
|
|
|
|
bool push();
|
|
|
|
|
bool pop(bool unset_if_empty = true);
|
|
|
|
|
bool is_pushed() const { return m_pushed; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
bool m_pushed = false;
|
|
|
|
|
std::vector<env_config> m_original = {};
|
|
|
|
|
std::vector<env_config> m_modified = {};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <template <typename, typename...> class ContainerT, typename... TailT>
|
|
|
|
|
env_store::env_store(ContainerT<env_config, TailT...>&& _container)
|
|
|
|
|
{
|
|
|
|
|
for(const auto& itr : _container)
|
|
|
|
|
{
|
|
|
|
|
m_original.emplace_back(env_config{itr.env_name, get_env(itr.env_name, ""), 1});
|
|
|
|
|
m_modified.emplace_back(env_config{itr.env_name, itr.env_value, 1});
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-08-08 18:39:01 -05:00
|
|
|
} // namespace common
|
|
|
|
|
} // namespace rocprofiler
|