Mercurial > minori
diff dep/animia/src/fd/kvm.cc @ 169:e44b7c428d7c
dep/animia: add libkvm method (UNTESTED)
author | Paper <mrpapersonic@gmail.com> |
---|---|
date | Sun, 19 Nov 2023 17:30:38 -0500 (14 months ago) |
parents | |
children | 8f6f8dd2eb23 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/dep/animia/src/fd/kvm.cc Sun Nov 19 17:30:38 2023 -0500 @@ -0,0 +1,134 @@ +/* kvm.cc: provides support for libkvm in multiple BSDs +** +** this is really the only way to get a thing that works on +** OpenBSD AND NetBSD. +** +** Much of this file is taken from the fstat source code in +** NetBSD. +*/ + +#include <kvm.h> + +namespace animia::internal::kvm { + +static bool GetFilename(kvm_t* kvm, struct vnode *vp, std::string& name) { + struct vnode vn; + if (!kvm_read(kvm, vp, &vn, sizeof(*vn))) + return 0; + + struct filestat fst; + const char* type = vfilestat(vn, &fst); + if (type == dead) + return false; + + for (DEVS* d = devs; d != NULL; d = d->next) { + if (d->fsid == fst->fsid && d->ino == fst->fileid) { + name = d->name; + break; + } + } + + return true; +} + +static bool GetFilePath(kvm_t* kvm, fdfile_t* fp, std::string& path) { + struct file file; + fdfile_t fdfile; + + if (!kvm_read(kvm, fp, &fdfile, sizeof(fdfile))) + return false; + + if (!fdfile.ff_file) + return false; + + if (!kvm_read(fdfile.ff_file, &file, sizeof(file))) + return false; + + if (file.f_type != DTYPE_VNODE) + return false; + + return GetFilename(kvm, file.f_data, path); +} + +static bool OpenFiles(kvm_t* kvm, struct kinfo_proc* p, open_file_proc_t open_file_proc) { + if (p->proc->p_fd == 0 || p->proc->p_cwdi == 0) + return false; + + struct filedesc filed; + if (!kvm_read(kvm, p->proc->p_fd, &filed, sizeof(filed))) + return false; + + if (filed.fd_lastfile == -1) + return false; + + struct cwdinfo cwdi; + if (!kvm_read(kvm, p->proc->p_cwdi, &cwdi, sizeof(cwdi))) + return false; + + struct fdtab dt; + if (!kvm_read(kvm, filed.fd_dt, &dt, sizeof(dt))) + return false; + + /* check for corrupted files? */ + if ((unsigned)filed.fd_lastfile >= dt.dt_nfiles || filed.fd_freefile > filed.fd_lastfile + 1) + return false; + + /* open files */ + std::unique_ptr<fdfile_t*[]> ofiles = nullptr; + { + ofiles.reset(malloc((filed.fd_lastfile + 1) * sizeof(fdfile_t*))); + if (!ofiles.get()) + return false; + } + + if (!kvm_read(kvm, &filed.fd_dt->dt_ff, ofiles.get(), filed.fd_lastfile + 1 * (sizeof(fdfile_t*)))) + return false; + + for (int i = 0; i <= filed.fd_lastfile; i++) { + if (!ofiles[i]) + continue; + std::string name; + GetFilePath(kvm, ofiles[i], name); + if (!open_file_proc(p->proc->p_pid, name)) + return false; + } + + return true; +} + +bool EnumerateOpenProcesses(process_proc_t process_proc) { + char errbuf[_POSIX2_LINE_MAX]; + kvm_t* kernel = kvm_openfiles(NULL, NULL, NULL, KVM_NO_FILES, errbuf); + if (!kernel) + return false; + + int entries = 0; + struct kinfo_proc* kinfo = kvm_getprocs(kernel, KERN_PROC_ALL, 0, sizeof(struct kinfo_proc), &nentries); + if (!kinfo) + return false; + + for (int i = 0; i < entries; i++) + if (!process_proc({kinfo[i].p_pid, kinfo[i].p_comm})) + return false; + + return true; +} + +bool EnumerateOpenFiles(std::set<pid_t>& pids, open_file_proc_t open_file_proc) { + kvm_t* kernel = kvm_openfiles(NULL, NULL, NULL, KVM_NO_FILES, errbuf); + if (!kernel) + return false; + + for (const auto& pid : pids) { + int cnt; + struct kinfo_proc* kinfo = kvm_getprocs(kernel, KERN_PROC_PID, pid, &cnt); + if (!kinfo) + return false; + + for (int i = 0; i < cnt; i++) { + OpenFiles(kernel, kinfo, open_file_proc); + } + } +} + +}