Mercurial > minori
view dep/animone/src/fd/kvm.cc @ 259:0362f3c4534c
widgets/graph: improve drawing code
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Mon, 01 Apr 2024 18:11:15 -0400 |
parents | 862d0d8619f6 |
children |
line wrap: on
line source
/* kvm.cc: provides support for *BSD. */ #include "animone/fd/kvm.h" #include "animone.h" #include "animone/fd.h" #include <sys/file.h> #include <sys/filedesc.h> #include <sys/param.h> #include <sys/queue.h> #include <sys/sysctl.h> #include <sys/types.h> #include <sys/user.h> #include <sys/vnode.h> #include <kvm.h> #ifdef LIBUTIL # include <libutil.h> #endif #include <string> namespace animone::internal::kvm { bool GetProcessName(pid_t pid, std::string& name) { char errbuf[_POSIX2_LINE_MAX]; kvm_t* kernel = kvm_openfiles(NULL, NULL, NULL, O_RDONLY, errbuf); if (!kernel) return false; int entries = 0; struct kinfo_proc* kinfo = kvm_getprocs(kernel, KERN_PROC_PID, pid, &entries); if (!kinfo) { kvm_close(kernel); return false; } if (entries < 1) { kvm_close(kernel); return false; } name = kinfo[0].ki_paddr->p_comm; return true; } /* Most of the BSDs share the common kvm library, * so accessing this information can be trivial. */ bool EnumerateOpenProcesses(process_proc_t process_proc) { char errbuf[_POSIX2_LINE_MAX]; kvm_t* kernel = kvm_openfiles(NULL, NULL, NULL, O_RDONLY, errbuf); if (!kernel) return false; int entries = 0; struct kinfo_proc* kinfo = kvm_getprocs(kernel, KERN_PROC_ALL, 0, &entries); if (!kinfo) { kvm_close(kernel); return false; } for (int i = 0; i < entries; i++) { if (!process_proc({kinfo[i].ki_paddr->p_pid, kinfo[i].ki_paddr->p_comm})) { kvm_close(kernel); return false; } } kvm_close(kernel); return true; } bool EnumerateOpenFiles(std::set<pid_t>& pids, open_file_proc_t open_file_proc) { #ifdef __OpenBSD__ char errbuf[_POSIX2_LINE_MAX]; kvm_t* kernel = kvm_openfiles(NULL, NULL, NULL, O_RDONLY, errbuf); if (!kernel) return false; for (const auto& pid : pids) { int cnt; struct kinfo_file* kfile = kvm_getfiles(kernel, KERN_FILE_BYPID, pid, &cnt); if (!kfile) { kvm_close(kernel); return false; } for (int i = 0; i < cnt; i++) { if (!open_file_proc({pid, kfile[i].kf_path})) { kvm_close(kernel); return false; } } } kvm_close(kernel); return true; #elif defined(LIBUTIL) /* does this code even work? */ for (const auto& pid : pids) { int cnt; std::unique_ptr<struct kinfo_file[]> files(kinfo_getfile(pid, &cnt)); if (!files) return false; for (int i = 0; i < cnt; i++) { const struct kinfo_file& current = files[i]; if (current.kf_vnode_type != KF_VTYPE_VREG) continue; if (!open_file_proc({pid, current.kf_path})) return false; } } return true; #elif defined(__NetBSD__) for (const auto& pid : pids) { int mib[6] = {CTL_KERN, KERN_FILE2, KERN_FILE_BYPID, pid, sizeof(struct kinfo_file), 0}; size_t len = 0; if (sysctl(mib, sizeof(mib) / sizeof(mib[0]), NULL, &len, NULL, 0) == -1) return false; mib[5] = len / sizeof(struct kinfo_file); std::unique_ptr<struct kinfo_file[]> buf(new struct kinfo_file[mib[5]]); if (!buf) return false; if (sysctl(mib, sizeof(mib) / sizeof(mib[0]), buf.get(), &len, NULL, 0) == -1) return false; /* TODO: check kfile[i].ki_ofileflags */ for (size_t i = 0; i < mib[5]; i++) if (!open_file_proc({pid, kfile[i].kf_path})) return false; } return true; #else return false; #endif } } // namespace animone::internal::kvm