/* Copyright 2021 Aristocratos (jakob@qvantnet.com) Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. indent = tab tab-size = 4 */ #pragma once #include #include #include #include #include #include #include #include #include // From `man 3 getifaddrs`: must be included before // clang-format off #include #include // clang-format on #if defined(__FreeBSD__) || defined(__OpenBSD__) # include #endif using std::array; using std::atomic; using std::deque; using std::string; using std::tuple; using std::vector; using namespace std::literals; // for operator""s void term_resize(bool force=false); void banner_gen(); extern void clean_quit(int sig); namespace Global { extern const vector> Banner_src; extern const string Version; extern atomic quitting; extern string exit_error_msg; extern atomic thread_exception; extern string banner; extern atomic resized; extern string overlay; extern string clock; extern uid_t real_uid, set_uid; extern atomic init_conf; } namespace Runner { extern atomic active; extern atomic reading; extern atomic stopping; extern atomic redraw; extern atomic coreNum_reset; extern pthread_t runner_id; extern bool pause_output; extern string debug_bg; void run(const string& box="", bool no_update = false, bool force_redraw = false); void stop(); } namespace Tools { //* Platform specific function for system_uptime (seconds since last restart) double system_uptime(); } namespace Shared { //* Initialize platform specific needed variables and check for errors void init(); extern long coreCount, page_size, clk_tck; #if defined(__FreeBSD__) || defined(__OpenBSD__) struct KvmDeleter { void operator()(kvm_t* handle) { kvm_close(handle); } }; using KvmPtr = std::unique_ptr; #endif } namespace Gpu { #ifdef GPU_SUPPORT extern vector box; extern int width, height, min_width, min_height; extern vector x_vec, y_vec; extern vector redraw; extern int shown; extern vector shown_panels; extern vector gpu_names; extern vector gpu_b_height_offsets; extern long long gpu_pwr_total_max; extern std::unordered_map> shared_gpu_percent; // averages, power/vram total const array mem_names { "used"s, "free"s }; //* Container for process information // TODO /*struct proc_info { unsigned int pid; unsigned long long mem; };*/ //* Container for supported Gpu::*::collect() functions struct gpu_info_supported { bool gpu_utilization = true, mem_utilization = true, gpu_clock = true, mem_clock = true, pwr_usage = true, pwr_state = true, temp_info = true, mem_total = true, mem_used = true, pcie_txrx = true; }; //* Per-device container for GPU info struct gpu_info { std::unordered_map> gpu_percent = { {"gpu-totals", {}}, {"gpu-vram-totals", {}}, {"gpu-pwr-totals", {}}, }; unsigned int gpu_clock_speed; // MHz long long pwr_usage; // mW long long pwr_max_usage = 255000; long long pwr_state; deque temp = {0}; long long temp_max = 110; long long mem_total = 0; long long mem_used = 0; deque mem_utilization_percent = {0}; // TODO: properly handle GPUs that can't report some stats long long mem_clock_speed = 0; // MHz long long pcie_tx = 0; // KB/s long long pcie_rx = 0; gpu_info_supported supported_functions; // vector graphics_processes = {}; // TODO // vector compute_processes = {}; }; namespace Nvml { extern bool shutdown(); } namespace Rsmi { extern bool shutdown(); } //* Collect gpu stats and temperatures auto collect(bool no_update = false) -> vector&; //* Draw contents of gpu box using as source string draw(const gpu_info& gpu, unsigned long index, bool force_redraw, bool data_same); #else struct gpu_info { bool supported = false; }; #endif } namespace Cpu { extern string box; extern int x, y, width, height, min_width, min_height; extern bool shown, redraw, got_sensors, cpu_temp_only, has_battery; extern string cpuName, cpuHz; extern vector available_fields; extern vector available_sensors; extern tuple current_bat; struct cpu_info { std::unordered_map> cpu_percent = { {"total", {}}, {"user", {}}, {"nice", {}}, {"system", {}}, {"idle", {}}, {"iowait", {}}, {"irq", {}}, {"softirq", {}}, {"steal", {}}, {"guest", {}}, {"guest_nice", {}} }; vector> core_percent; vector> temp; long long temp_max = 0; array load_avg; }; //* Collect cpu stats and temperatures auto collect(bool no_update = false) -> cpu_info&; //* Draw contents of cpu box using as source string draw(const cpu_info& cpu, const vector& gpu, bool force_redraw = false, bool data_same = false); //* Parse /proc/cpu info for mapping of core ids auto get_core_mapping() -> std::unordered_map; extern std::unordered_map core_mapping; auto get_cpuHz() -> string; //* Get battery info from /sys auto get_battery() -> tuple; } namespace Mem { extern string box; extern int x, y, width, height, min_width, min_height; extern bool has_swap, shown, redraw; const array mem_names { "used"s, "available"s, "cached"s, "free"s }; const array swap_names { "swap_used"s, "swap_free"s }; extern int disk_ios; struct disk_info { std::filesystem::path dev; string name; string fstype{}; // defaults to "" std::filesystem::path stat{}; // defaults to "" int64_t total{}; int64_t used{}; int64_t free{}; int used_percent{}; int free_percent{}; array old_io = {0, 0, 0}; deque io_read = {}; deque io_write = {}; deque io_activity = {}; }; struct mem_info { std::unordered_map stats = {{"used", 0}, {"available", 0}, {"cached", 0}, {"free", 0}, {"swap_total", 0}, {"swap_used", 0}, {"swap_free", 0}}; std::unordered_map> percent = {{"used", {}}, {"available", {}}, {"cached", {}}, {"free", {}}, {"swap_total", {}}, {"swap_used", {}}, {"swap_free", {}}}; std::unordered_map disks; vector disks_order; }; //?* Get total system memory uint64_t get_totalMem(); //* Collect mem & disks stats auto collect(bool no_update = false) -> mem_info&; //* Draw contents of mem box using as source string draw(const mem_info& mem, bool force_redraw = false, bool data_same = false); } namespace Net { extern string box; extern int x, y, width, height, min_width, min_height; extern bool shown, redraw; extern string selected_iface; extern vector interfaces; extern bool rescale; extern std::unordered_map graph_max; struct net_stat { uint64_t speed{}; uint64_t top{}; uint64_t total{}; uint64_t last{}; uint64_t offset{}; uint64_t rollover{}; }; struct net_info { std::unordered_map> bandwidth = { {"download", {}}, {"upload", {}} }; std::unordered_map stat = { {"download", {}}, {"upload", {}} }; string ipv4{}; // defaults to "" string ipv6{}; // defaults to "" bool connected{}; }; class IfAddrsPtr { struct ifaddrs* ifaddr; int status; public: IfAddrsPtr() { status = getifaddrs(&ifaddr); } ~IfAddrsPtr() { freeifaddrs(ifaddr); } [[nodiscard]] constexpr auto operator()() -> struct ifaddrs* { return ifaddr; } [[nodiscard]] constexpr auto get() -> struct ifaddrs* { return ifaddr; } [[nodiscard]] constexpr auto get_status() const noexcept -> int { return status; }; }; extern std::unordered_map current_net; //* Collect net upload/download stats auto collect(bool no_update=false) -> net_info&; //* Draw contents of net box using as source string draw(const net_info& net, bool force_redraw = false, bool data_same = false); } namespace Proc { extern atomic numpids; extern string box; extern int x, y, width, height, min_width, min_height; extern bool shown, redraw; extern int select_max; extern atomic detailed_pid; extern int selected_pid, start, selected, collapse, expand, filter_found, selected_depth; extern string selected_name; //? Contains the valid sorting options for processes const vector sort_vector = { "pid", "name", "command", "threads", "user", "memory", "cpu direct", "cpu lazy", }; //? Translation from process state char to explanative string const std::unordered_map proc_states = { {'R', "Running"}, {'S', "Sleeping"}, {'D', "Waiting"}, {'Z', "Zombie"}, {'T', "Stopped"}, {'t', "Tracing"}, {'X', "Dead"}, {'x', "Dead"}, {'K', "Wakekill"}, {'W', "Unknown"}, {'P', "Parked"} }; //* Container for process information struct proc_info { size_t pid{}; string name{}; // defaults to "" string cmd{}; // defaults to "" string short_cmd{}; // defaults to "" size_t threads{}; int name_offset{}; string user{}; // defaults to "" uint64_t mem{}; double cpu_p{}; // defaults to = 0.0 double cpu_c{}; // defaults to = 0.0 char state = '0'; int64_t p_nice{}; uint64_t ppid{}; uint64_t cpu_s{}; uint64_t cpu_t{}; string prefix{}; // defaults to "" size_t depth{}; size_t tree_index{}; bool collapsed{}; bool filtered{}; }; //* Container for process info box struct detail_container { size_t last_pid{}; bool skip_smaps{}; proc_info entry; string elapsed, parent, status, io_read, io_write, memory; long long first_mem = -1; deque cpu_percent; deque mem_bytes; }; //? Contains all info for proc detailed box extern detail_container detailed; //* Collect and sort process information from /proc auto collect(bool no_update = false) -> vector&; //* Update current selection and view, returns -1 if no change otherwise the current selection int selection(const string& cmd_key); //* Draw contents of proc box using as data source string draw(const vector& plist, bool force_redraw = false, bool data_same = false); struct tree_proc { std::reference_wrapper entry; vector children; }; //* Sort vector of proc_info's void proc_sorter(vector& proc_vec, const string& sorting, bool reverse, bool tree = false); //* Recursive sort of process tree void tree_sort(vector& proc_vec, const string& sorting, bool reverse, int& c_index, const int index_max, bool collapsed = false); bool matches_filter(const proc_info& proc, const std::string& filter); //* Generate process tree list void _tree_gen(proc_info& cur_proc, vector& in_procs, vector& out_procs, int cur_depth, bool collapsed, const string& filter, bool found = false, bool no_update = false, bool should_filter = false); }