uhd/host/lib/utils/prefs.cpp
Martin Braun 1383fde345 uhd: paths: Harmonize around XDG Base Directory specification
Up until now, we completely ignore the XDG specification.

This commit does the following to change that:

- It uses XDG_DATA_HOME and XDG_CONFIG_HOME for cal and config data,
  respectively.
- If config data is in ~/.uhd/uhd.conf, that is still accepted, but if
  it conflicts with $XDG_CONFIG_HOME/uhd.conf, it is ignored and a
  warning is displayed
- The default location for cal data is thus ${HOME}/.local/share/uhd/cal
  on Unix, and %LOCALAPPDATA%\uhd\cal on Windows. This is a change in
  location!
- The UHD_CONFIG_DIR environment variable was confusingly named and is
  now removed. It provided an alternative location than the home
  directory. The same purpose is now much better served by XDG_DATA_HOME
  and XDG_CONFIG_HOME.

The specification can be found here:
specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
2020-04-02 12:36:20 -05:00

136 lines
4.6 KiB
C++

//
// Copyright 2017-2018 Ettus Research, a National Instruments Company
//
// SPDX-License-Identifier: GPL-3.0+
//
#include <uhd/utils/log.hpp>
#include <uhd/utils/paths.hpp>
#include <uhdlib/utils/paths.hpp>
#include <uhdlib/utils/prefs.hpp>
#include <config.h>
#include <boost/filesystem.hpp>
#include <cstdlib>
using namespace uhd;
namespace {
constexpr char UHD_CONF_FILE_VAR[] = "UHD_CONFIG_FILE";
inline bool _update_conf_file(
const std::string& path, const std::string& config_type, config_parser& conf_file)
{
if (not path.empty()) {
UHD_LOG_TRACE("PREFS", "Trying to load " << path);
if (boost::filesystem::exists(path)) {
try {
conf_file.read_file(path);
UHD_LOG_DEBUG(
"PREFS", "Loaded " << config_type << " config file " << path);
return true;
} catch (...) {
UHD_LOG_DEBUG(
"PREFS", "Failed to load " << config_type << " config file " << path);
return false;
}
} else {
UHD_LOG_TRACE(
"PREFS", "No " << config_type << " config file found at " << path);
return false;
}
}
return false;
}
void update_from_key(
const std::string& key, const std::string& value, uhd::device_addr_t& user_args)
{
if (value.empty()) {
return;
}
const std::string key_str = key + "=" + value;
for (const auto& key : uhd::prefs::get_uhd_config().options(key_str)) {
user_args[key] = uhd::prefs::get_uhd_config().get<std::string>(key_str, key);
}
}
device_addr_t get_args(const uhd::device_addr_t& user_args,
const std::vector<std::string>& keys_to_update_from)
{
device_addr_t args;
for (const auto& key : keys_to_update_from) {
update_from_key(key, user_args.get(key, ""), args);
}
// Finally, copy over the original user args:
for (const auto& user_key : user_args.keys()) {
args[user_key] = user_args[user_key];
}
return args;
}
} // namespace
config_parser& uhd::prefs::get_uhd_config()
{
static config_parser _conf_files{};
static bool init_done = false;
if (not init_done) {
UHD_LOG_TRACE("CONF", "Initializing config file object...");
const std::string sys_conf_file = path_expandvars(UHD_SYS_CONF_FILE);
_update_conf_file(sys_conf_file, "system", _conf_files);
// prefer .config/uhd.conf
// otherwise ~/.uhd/uhd.conf
const std::string user_conf_file =
(get_xdg_config_home() / std::string(UHD_USER_CONF_FILE)).string();
const bool user_conf_loaded =
_update_conf_file(user_conf_file, "user", _conf_files);
// Config files can be in ~/.config/ or in ~/.uhd. The latter is
// considered deprecated. We load from there (if we have not already
// loaded from ~/.config), but we show a warning.
if (!user_conf_loaded
&& _update_conf_file(
(get_legacy_config_home() / std::string(UHD_USER_CONF_FILE)).string(),
"user",
_conf_files)) {
UHD_LOG_WARNING("PREFS",
"Loaded config from " << get_legacy_config_home().string()
<< ". This location is considered deprecated, "
"consider moving your config file to "
<< get_xdg_config_home().string() << " instead.");
}
std::string env_conf_file;
try { // getenv into std::string can fail
if (std::getenv(UHD_CONF_FILE_VAR) != NULL) {
env_conf_file = std::string(std::getenv(UHD_CONF_FILE_VAR));
}
_update_conf_file(env_conf_file, "ENV", _conf_files);
} catch (const std::exception&) {
// nop
}
init_done = true;
UHD_LOG_TRACE("PREFS", "Done initializing.");
}
return _conf_files;
}
device_addr_t uhd::prefs::get_usrp_args(const uhd::device_addr_t& user_args)
{
const std::vector<std::string> keys_to_update_from = {"type", "product", "serial"};
return get_args(user_args, keys_to_update_from);
}
device_addr_t uhd::prefs::get_dpdk_args(const uhd::device_addr_t& user_args)
{
const std::vector<std::string> keys_to_update_from = {"use_dpdk"};
return get_args(user_args, keys_to_update_from);
}
device_addr_t uhd::prefs::get_dpdk_nic_args(const uhd::device_addr_t& user_args)
{
const std::vector<std::string> keys_to_update_from = {"dpdk_mac"};
return get_args(user_args, keys_to_update_from);
}