2019-07-02 07:58:19 +02:00
|
|
|
#pragma once
|
|
|
|
|
2019-12-13 06:37:06 +01:00
|
|
|
#include <pthread.h>
|
2019-07-02 07:58:19 +02:00
|
|
|
#include <string>
|
|
|
|
#include <functional>
|
|
|
|
#include <string_view>
|
|
|
|
|
2019-11-19 08:04:47 +01:00
|
|
|
#define UID_ROOT 0
|
|
|
|
#define UID_SHELL 2000
|
2019-07-15 02:41:51 +02:00
|
|
|
|
2021-03-06 22:55:30 +01:00
|
|
|
#define DISALLOW_COPY_AND_MOVE(clazz) \
|
|
|
|
clazz(const clazz &) = delete; \
|
|
|
|
clazz(clazz &&) = delete;
|
|
|
|
|
2019-09-26 05:55:39 +02:00
|
|
|
class mutex_guard {
|
2021-03-06 22:55:30 +01:00
|
|
|
DISALLOW_COPY_AND_MOVE(mutex_guard)
|
2019-07-02 07:58:19 +02:00
|
|
|
public:
|
2020-12-31 07:11:24 +01:00
|
|
|
explicit mutex_guard(pthread_mutex_t &m): mutex(&m) {
|
|
|
|
pthread_mutex_lock(mutex);
|
|
|
|
}
|
2021-01-12 09:07:48 +01:00
|
|
|
void unlock() {
|
|
|
|
pthread_mutex_unlock(mutex);
|
|
|
|
mutex = nullptr;
|
2020-12-31 07:11:24 +01:00
|
|
|
}
|
|
|
|
~mutex_guard() {
|
2021-01-12 09:07:48 +01:00
|
|
|
if (mutex) pthread_mutex_unlock(mutex);
|
2020-12-31 07:11:24 +01:00
|
|
|
}
|
2019-07-02 07:58:19 +02:00
|
|
|
private:
|
2020-12-31 07:11:24 +01:00
|
|
|
pthread_mutex_t *mutex;
|
2019-07-02 07:58:19 +02:00
|
|
|
};
|
|
|
|
|
2020-04-30 10:27:36 +02:00
|
|
|
template <class Func>
|
2019-09-26 05:55:39 +02:00
|
|
|
class run_finally {
|
2021-03-06 22:55:30 +01:00
|
|
|
DISALLOW_COPY_AND_MOVE(run_finally)
|
2019-07-02 07:58:19 +02:00
|
|
|
public:
|
2020-12-31 07:11:24 +01:00
|
|
|
explicit run_finally(const Func &fn) : fn(fn) {}
|
|
|
|
~run_finally() { fn(); }
|
2019-07-02 07:58:19 +02:00
|
|
|
private:
|
2020-12-31 07:11:24 +01:00
|
|
|
const Func &fn;
|
2019-07-02 07:58:19 +02:00
|
|
|
};
|
|
|
|
|
2019-11-19 06:16:20 +01:00
|
|
|
template <typename T>
|
|
|
|
class reversed_container {
|
|
|
|
public:
|
2020-12-31 07:11:24 +01:00
|
|
|
reversed_container(T &base) : base(base) {}
|
|
|
|
decltype(std::declval<T>().rbegin()) begin() { return base.rbegin(); }
|
|
|
|
decltype(std::declval<T>().crbegin()) begin() const { return base.crbegin(); }
|
|
|
|
decltype(std::declval<T>().crbegin()) cbegin() const { return base.crbegin(); }
|
|
|
|
decltype(std::declval<T>().rend()) end() { return base.rend(); }
|
|
|
|
decltype(std::declval<T>().crend()) end() const { return base.crend(); }
|
|
|
|
decltype(std::declval<T>().crend()) cend() const { return base.crend(); }
|
2019-11-19 06:16:20 +01:00
|
|
|
private:
|
2020-12-31 07:11:24 +01:00
|
|
|
T &base;
|
2019-11-19 06:16:20 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
reversed_container<T> reversed(T &base) {
|
2020-12-31 07:11:24 +01:00
|
|
|
return reversed_container<T>(base);
|
2019-11-19 06:16:20 +01:00
|
|
|
}
|
|
|
|
|
2019-11-19 08:04:47 +01:00
|
|
|
int parse_int(const char *s);
|
2020-11-07 23:36:13 +01:00
|
|
|
static inline int parse_int(const std::string &s) { return parse_int(s.data()); }
|
2019-07-02 07:58:19 +02:00
|
|
|
static inline int parse_int(std::string_view s) { return parse_int(s.data()); }
|
|
|
|
|
2020-04-30 10:27:36 +02:00
|
|
|
using thread_entry = void *(*)(void *);
|
2021-01-07 07:21:17 +01:00
|
|
|
int new_daemon_thread(thread_entry entry, void *arg = nullptr);
|
2021-01-11 02:11:00 +01:00
|
|
|
int new_daemon_thread(void(*entry)());
|
2020-04-30 10:27:36 +02:00
|
|
|
int new_daemon_thread(std::function<void()> &&entry);
|
2019-07-02 07:58:19 +02:00
|
|
|
|
2020-12-31 00:55:53 +01:00
|
|
|
static inline bool str_contains(std::string_view s, std::string_view ss) {
|
2020-12-31 07:11:24 +01:00
|
|
|
return s.find(ss) != std::string::npos;
|
2020-12-31 00:55:53 +01:00
|
|
|
}
|
|
|
|
static inline bool str_starts(std::string_view s, std::string_view ss) {
|
2021-03-18 06:36:42 +01:00
|
|
|
return s.size() >= ss.size() && s.compare(0, ss.size(), ss) == 0;
|
2020-12-31 00:55:53 +01:00
|
|
|
}
|
|
|
|
static inline bool str_ends(std::string_view s, std::string_view ss) {
|
2020-12-31 07:11:24 +01:00
|
|
|
return s.size() >= ss.size() && s.compare(s.size() - ss.size(), std::string::npos, ss) == 0;
|
2020-12-31 00:55:53 +01:00
|
|
|
}
|
|
|
|
|
2020-11-07 23:36:13 +01:00
|
|
|
int fork_dont_care();
|
2020-12-18 01:54:53 +01:00
|
|
|
int fork_no_orphan();
|
2020-11-07 23:36:13 +01:00
|
|
|
void init_argv0(int argc, char **argv);
|
|
|
|
void set_nice_name(const char *name);
|
|
|
|
uint32_t binary_gcd(uint32_t u, uint32_t v);
|
|
|
|
int switch_mnt_ns(int pid);
|
|
|
|
int gen_rand_str(char *buf, int len, bool varlen = true);
|
|
|
|
std::string &replace_all(std::string &str, std::string_view from, std::string_view to);
|
|
|
|
|
2019-07-02 07:58:19 +02:00
|
|
|
struct exec_t {
|
2020-12-31 07:11:24 +01:00
|
|
|
bool err = false;
|
|
|
|
int fd = -2;
|
|
|
|
void (*pre_exec)() = nullptr;
|
|
|
|
int (*fork)() = xfork;
|
|
|
|
const char **argv = nullptr;
|
2019-07-02 07:58:19 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
int exec_command(exec_t &exec);
|
|
|
|
template <class ...Args>
|
|
|
|
int exec_command(exec_t &exec, Args &&...args) {
|
2020-12-31 07:11:24 +01:00
|
|
|
const char *argv[] = {args..., nullptr};
|
|
|
|
exec.argv = argv;
|
|
|
|
return exec_command(exec);
|
2019-07-02 07:58:19 +02:00
|
|
|
}
|
|
|
|
int exec_command_sync(exec_t &exec);
|
|
|
|
template <class ...Args>
|
|
|
|
int exec_command_sync(exec_t &exec, Args &&...args) {
|
2020-12-31 07:11:24 +01:00
|
|
|
const char *argv[] = {args..., nullptr};
|
|
|
|
exec.argv = argv;
|
|
|
|
return exec_command_sync(exec);
|
2019-07-02 07:58:19 +02:00
|
|
|
}
|
|
|
|
template <class ...Args>
|
|
|
|
int exec_command_sync(Args &&...args) {
|
2020-12-31 07:11:24 +01:00
|
|
|
exec_t exec{};
|
|
|
|
return exec_command_sync(exec, args...);
|
2019-07-02 07:58:19 +02:00
|
|
|
}
|
2020-11-07 23:36:13 +01:00
|
|
|
template <class ...Args>
|
|
|
|
void exec_command_async(Args &&...args) {
|
2020-12-31 07:11:24 +01:00
|
|
|
const char *argv[] = {args..., nullptr};
|
|
|
|
exec_t exec {
|
|
|
|
.argv = argv,
|
|
|
|
.fork = fork_dont_care
|
|
|
|
};
|
|
|
|
exec_command(exec);
|
2020-11-07 23:36:13 +01:00
|
|
|
}
|