You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
MangoHud/src/file_utils.cpp

183 lines
4.5 KiB
C++

#include "file_utils.h"
#include "string_utils.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <dirent.h>
#include <limits.h>
#include <fstream>
#include <cstring>
#include <string>
#include <spdlog/spdlog.h>
std::string read_line(const std::string& filename)
{
std::string line;
std::ifstream file(filename);
std::getline(file, line);
return line;
}
bool find_folder(const char* root, const char* prefix, std::string& dest)
{
struct dirent* dp;
DIR* dirp = opendir(root);
if (!dirp) {
spdlog::error("Error opening directory '{}': {}", root, strerror(errno));
return false;
}
// XXX xfs/jfs need stat() for inode type
while ((dp = readdir(dirp))) {
if ((dp->d_type == DT_LNK || dp->d_type == DT_DIR) && starts_with(dp->d_name, prefix)) {
dest = dp->d_name;
closedir(dirp);
return true;
}
}
closedir(dirp);
return false;
}
bool find_folder(const std::string& root, const std::string& prefix, std::string& dest)
{
return find_folder(root.c_str(), prefix.c_str(), dest);
}
std::vector<std::string> ls(const char* root, const char* prefix, LS_FLAGS flags)
{
std::vector<std::string> list;
struct dirent* dp;
DIR* dirp = opendir(root);
if (!dirp) {
spdlog::error("Error opening directory '{}': {}", root, strerror(errno));
return list;
}
while ((dp = readdir(dirp))) {
if ((prefix && !starts_with(dp->d_name, prefix))
|| !strcmp(dp->d_name, ".")
|| !strcmp(dp->d_name, ".."))
continue;
if (dp->d_type == DT_LNK) {
struct stat s;
std::string path(root);
if (path.back() != '/')
path += "/";
path += dp->d_name;
if (stat(path.c_str(), &s))
continue;
if (((flags & LS_DIRS) && S_ISDIR(s.st_mode))
|| ((flags & LS_FILES) && !S_ISDIR(s.st_mode))) {
list.push_back(dp->d_name);
}
} else if (((flags & LS_DIRS) && dp->d_type == DT_DIR)
|| ((flags & LS_FILES) && dp->d_type == DT_REG)
) {
list.push_back(dp->d_name);
}
}
closedir(dirp);
return list;
}
bool file_exists(const std::string& path)
{
struct stat s;
return !stat(path.c_str(), &s) && !S_ISDIR(s.st_mode);
}
bool dir_exists(const std::string& path)
{
struct stat s;
return !stat(path.c_str(), &s) && S_ISDIR(s.st_mode);
}
std::string read_symlink(const char * link)
{
char result[PATH_MAX] {};
ssize_t count = readlink(link, result, PATH_MAX);
return std::string(result, (count > 0) ? count : 0);
}
std::string get_exe_path()
{
return read_symlink("/proc/self/exe");
}
std::string get_wine_exe_name(bool keep_ext)
{
const std::string exe_path = get_exe_path();
if (!ends_with(exe_path, "wine-preloader") && !ends_with(exe_path, "wine64-preloader")) {
return std::string();
}
std::string line = read_line("/proc/self/comm"); // max 16 characters though
if (ends_with(line, ".exe", true))
{
auto dot = keep_ext ? std::string::npos : line.find_last_of('.');
return line.substr(0, dot);
}
std::ifstream cmdline("/proc/self/cmdline");
// Iterate over arguments (separated by NUL byte).
while (std::getline(cmdline, line, '\0')) {
auto n = std::string::npos;
if (!line.empty()
&& ((n = line.find_last_of("/\\")) != std::string::npos)
&& n < line.size() - 1) // have at least one character
{
auto dot = keep_ext ? std::string::npos : line.find_last_of('.');
if (dot < n)
dot = line.size();
return line.substr(n + 1, dot - n - 1);
}
else if (ends_with(line, ".exe", true))
{
auto dot = keep_ext ? std::string::npos : line.find_last_of('.');
return line.substr(0, dot);
}
}
return std::string();
}
std::string get_home_dir()
{
std::string path;
const char* p = getenv("HOME");
if (p)
path = p;
return path;
}
std::string get_data_dir()
{
const char* p = getenv("XDG_DATA_HOME");
if (p)
return p;
std::string path = get_home_dir();
if (!path.empty())
path += "/.local/share";
return path;
}
std::string get_config_dir()
{
const char* p = getenv("XDG_CONFIG_HOME");
if (p)
return p;
std::string path = get_home_dir();
if (!path.empty())
path += "/.config";
return path;
}