Magisk/native/jni/utils/misc.hpp

174 lines
5.2 KiB
C++
Raw Normal View History

2019-07-01 22:58:19 -07:00
#pragma once
2019-12-13 00:37:06 -05:00
#include <pthread.h>
2019-07-01 22:58:19 -07:00
#include <string>
#include <functional>
#include <string_view>
2022-01-21 04:43:27 -08:00
#include <bitset>
2019-07-01 22:58:19 -07:00
2019-11-19 02:04:47 -05:00
#define UID_ROOT 0
#define UID_SHELL 2000
#define DISALLOW_COPY_AND_MOVE(clazz) \
clazz(const clazz &) = delete; \
clazz(clazz &&) = delete;
2021-09-17 02:07:32 -07:00
#define to_app_id(uid) (uid % 100000)
#define to_user_id(uid) (uid / 100000)
2019-09-25 23:55:39 -04:00
class mutex_guard {
DISALLOW_COPY_AND_MOVE(mutex_guard)
2019-07-01 22:58:19 -07:00
public:
explicit mutex_guard(pthread_mutex_t &m): mutex(&m) {
pthread_mutex_lock(mutex);
}
2021-01-12 00:07:48 -08:00
void unlock() {
pthread_mutex_unlock(mutex);
mutex = nullptr;
}
~mutex_guard() {
2021-01-12 00:07:48 -08:00
if (mutex) pthread_mutex_unlock(mutex);
}
2019-07-01 22:58:19 -07:00
private:
pthread_mutex_t *mutex;
2019-07-01 22:58:19 -07:00
};
2020-04-30 01:27:36 -07:00
template <class Func>
2019-09-25 23:55:39 -04:00
class run_finally {
DISALLOW_COPY_AND_MOVE(run_finally)
2019-07-01 22:58:19 -07:00
public:
2021-08-19 01:54:12 -07:00
explicit run_finally(Func &&fn) : fn(std::move(fn)) {}
~run_finally() { fn(); }
2019-07-01 22:58:19 -07:00
private:
2021-08-19 01:54:12 -07:00
Func fn;
2019-07-01 22:58:19 -07:00
};
template <typename T>
class reversed_container {
public:
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(); }
private:
T &base;
};
template <typename T>
reversed_container<T> reversed(T &base) {
return reversed_container<T>(base);
}
template<class T>
static inline void default_new(T *&p) { p = new T(); }
template<typename T, typename Impl>
class stateless_allocator {
public:
using value_type = T;
T *allocate(size_t num) { return static_cast<T*>(Impl::allocate(sizeof(T) * num)); }
void deallocate(T *ptr, size_t num) { Impl::deallocate(ptr, sizeof(T) * num); }
stateless_allocator() = default;
stateless_allocator(const stateless_allocator&) = default;
stateless_allocator(stateless_allocator&&) = default;
template <typename U>
stateless_allocator(const stateless_allocator<U, Impl>&) {}
bool operator==(const stateless_allocator&) { return true; }
bool operator!=(const stateless_allocator&) { return false; }
};
2022-01-21 04:43:27 -08:00
class dynamic_bitset {
private:
using long_bits = std::bitset<sizeof(unsigned long)>;
std::vector<long_bits> bits;
public:
constexpr static int slot_size = sizeof(unsigned long);
long_bits::reference operator[] (size_t pos) {
size_t slot = pos / slot_size;
size_t index = pos % slot_size;
if (bits.size() <= slot) {
bits.resize(slot + 1);
}
return bits[slot][index];
}
size_t slots() const { return bits.size(); }
unsigned long to_ulong(size_t slot) const { return bits[slot].to_ulong(); }
};
2021-10-23 14:38:30 -07:00
int parse_int(std::string_view s);
2019-07-01 22:58:19 -07:00
2020-04-30 01:27:36 -07:00
using thread_entry = void *(*)(void *);
int new_daemon_thread(thread_entry entry, void *arg = nullptr);
2019-07-01 22:58:19 -07:00
static inline bool str_contains(std::string_view s, std::string_view ss) {
return s.find(ss) != std::string::npos;
}
static inline bool str_starts(std::string_view s, std::string_view ss) {
return s.size() >= ss.size() && s.compare(0, ss.size(), ss) == 0;
}
static inline bool str_ends(std::string_view s, std::string_view ss) {
return s.size() >= ss.size() && s.compare(s.size() - ss.size(), std::string::npos, ss) == 0;
}
static inline std::string ltrim(std::string &&s) {
s.erase(s.begin(), std::find_if(s.begin(), s.end(), [](unsigned char ch) {
return !std::isspace(ch);
}));
return std::move(s);
}
static inline std::string rtrim(std::string &&s) {
s.erase(std::find_if(s.rbegin(), s.rend(), [](unsigned char ch) {
return !std::isspace(ch) && ch != '\0';
}).base(), s.end());
return std::move(s);
}
2020-11-07 14:36:13 -08:00
int fork_dont_care();
int fork_no_orphan();
2020-11-07 14:36:13 -08: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);
std::vector<std::string> split(const std::string& s, const std::string& delimiters);
2020-11-07 14:36:13 -08:00
2019-07-01 22:58:19 -07:00
struct exec_t {
bool err = false;
int fd = -2;
void (*pre_exec)() = nullptr;
int (*fork)() = xfork;
const char **argv = nullptr;
2019-07-01 22:58:19 -07:00
};
int exec_command(exec_t &exec);
template <class ...Args>
int exec_command(exec_t &exec, Args &&...args) {
const char *argv[] = {args..., nullptr};
exec.argv = argv;
return exec_command(exec);
2019-07-01 22:58:19 -07:00
}
int exec_command_sync(exec_t &exec);
template <class ...Args>
int exec_command_sync(exec_t &exec, Args &&...args) {
const char *argv[] = {args..., nullptr};
exec.argv = argv;
return exec_command_sync(exec);
2019-07-01 22:58:19 -07:00
}
template <class ...Args>
int exec_command_sync(Args &&...args) {
2021-08-26 03:09:56 -07:00
exec_t exec;
return exec_command_sync(exec, args...);
2019-07-01 22:58:19 -07:00
}
2020-11-07 14:36:13 -08:00
template <class ...Args>
void exec_command_async(Args &&...args) {
const char *argv[] = {args..., nullptr};
exec_t exec {
.argv = argv,
.fork = fork_dont_care
};
exec_command(exec);
2020-11-07 14:36:13 -08:00
}