comparison dep/animia/src/fd/xnu.cc @ 153:bd439dd6ffc5

*: make win stuff actually work, rename bsd.cc to xnu.cc It's been OS X only for ages, and these functions are different between most BSDs anyway
author Paper <mrpapersonic@gmail.com>
date Wed, 15 Nov 2023 13:28:18 -0500
parents
children d43d68408d3c
comparison
equal deleted inserted replaced
152:8700806c2cc2 153:bd439dd6ffc5
1 /*
2 ** fd/xnu.cpp
3 ** - provides support for XNU (part of Darwin)
4 */
5 #include "animia/fd/xnu.h"
6 #include "animia.h"
7
8 #include <unordered_map>
9 #include <vector>
10 #include <string>
11
12 #include <fcntl.h>
13 #include <sys/sysctl.h>
14 #include <sys/types.h>
15 #include <sys/user.h>
16 #include <libproc.h>
17
18 namespace animia::internal::xnu {
19
20 static bool GetProcessName(pid_t pid, std::string& result) {
21 struct proc_bsdinfo proc;
22
23 int st = proc_pidinfo(pid, PROC_PIDTBSDINFO, 0, &proc, PROC_PIDTBSDINFO_SIZE);
24 if (st != PROC_PIDTBSDINFO_SIZE)
25 return false;
26
27 /* fixme: is this right? pbi_comm is an alternative, but it reduces the string size to
28 16 chars. does pbi_name do the same, or is it different? */
29 result = proc.pbi_name;
30 return true;
31 }
32
33 /* this is a cleaned up version of a function from... Apple?
34 ...anyway, what it essentially does is gets the size and stuff from
35 sysctl() and reserves the space in a vector to store the PIDs */
36 bool XnuFdTools::EnumerateOpenProcesses(process_proc_t process_proc) {
37 struct kinfo_proc* result = NULL;
38 size_t length = 0;
39 static const int name[] = {CTL_KERN, KERN_PROC, KERN_PROC_ALL, 0};
40
41 /* get appropriate length from sysctl()
42 note: the reason this isn't checked is actually because this will
43 *always* return an error on OS X (or... maybe I'm doing it wrong :) ) */
44 sysctl((int*)name, (sizeof(name) / sizeof(*name)) - 1, NULL, &length, NULL, 0);
45
46 result = (struct kinfo_proc*)malloc(length);
47 if (result == NULL)
48 return false;
49
50 /* TODO: this might actually return ENOMEM if the amount of file handles changes between the
51 original sysctl() call and this one, which is technically possible */
52 if (sysctl((int*)name, (sizeof(name) / sizeof(*name)) - 1, result, &length, NULL, 0) == ENOMEM) {
53 assert(result != NULL);
54 free(result);
55 return false;
56 }
57
58 for (int i = 0; i < length / sizeof(*result); i++) {
59 const pid_t pid = result[i].kp_proc.p_pid;
60 if (!process_proc({pid, GetProcessName(pid)}))
61 return false;
62 }
63 }
64
65 bool XnuFdTools::EnumerateOpenFiles(const std::set<pid_t>& pids, open_file_proc_t open_file_proc) {
66 if (!open_file_proc)
67 return false;
68
69 for (const auto& pid : pids) {
70 int bufsz = proc_pidinfo(pid, PROC_PIDLISTFDS, 0, NULL, 0);
71 if (bufsz == -1)
72 return false;
73
74 struct proc_fdinfo* info = (struct proc_fdinfo*)malloc(bufsz);
75 if (!info)
76 return false;
77
78 proc_pidinfo(pid, PROC_PIDLISTFDS, 0, info, bufsz);
79
80 for (int i = 0; i < bufsz / sizeof(info[0]); i++) {
81 if (info[i].proc_fdtype == PROX_FDTYPE_VNODE) {
82 struct vnode_fdinfowithpath vnodeInfo;
83
84 int sz = proc_pidfdinfo(pid, info[i].proc_fd, PROC_PIDFDVNODEPATHINFO, &vnodeInfo, PROC_PIDFDVNODEPATHINFO_SIZE);
85 if (sz != PROC_PIDFDVNODEPATHINFO_SIZE)
86 return false;
87
88 /* this doesn't work!
89 if (vnodeInfo.pfi.fi_openflags & O_WRONLY || vnodeInfo.pfi.fi_openflags & O_RDWR)
90 continue;
91 */
92
93 if (!open_file_proc({pid, vnodeInfo.pvip.vip_path}))
94 return false;
95 }
96 }
97 }
98 return true;
99 }
100
101 } // namespace animia::internal::unix