blob: b67ae877900064d06750390fb4f16bcb1b4f2c5c [file] [log] [blame]
Fangrui Songb0e54cb2019-05-13 04:42:32 +00001//===-- ProcessInfo.cpp -----------------------------------------*- C++ -*-===//
Zachary Turner805e7102019-03-04 21:51:03 +00002//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#include "lldb/Utility/ProcessInfo.h"
10
11#include "lldb/Utility/ArchSpec.h"
12#include "lldb/Utility/Stream.h"
13#include "lldb/Utility/StreamString.h"
14#include "lldb/Utility/UserIDResolver.h"
15#include "llvm/ADT/SmallString.h"
16
17#include <climits>
18
19using namespace lldb;
20using namespace lldb_private;
21
22ProcessInfo::ProcessInfo()
23 : m_executable(), m_arguments(), m_environment(), m_uid(UINT32_MAX),
24 m_gid(UINT32_MAX), m_arch(), m_pid(LLDB_INVALID_PROCESS_ID) {}
25
26ProcessInfo::ProcessInfo(const char *name, const ArchSpec &arch,
27 lldb::pid_t pid)
28 : m_executable(name), m_arguments(), m_environment(), m_uid(UINT32_MAX),
29 m_gid(UINT32_MAX), m_arch(arch), m_pid(pid) {}
30
31void ProcessInfo::Clear() {
32 m_executable.Clear();
33 m_arguments.Clear();
34 m_environment.clear();
35 m_uid = UINT32_MAX;
36 m_gid = UINT32_MAX;
37 m_arch.Clear();
38 m_pid = LLDB_INVALID_PROCESS_ID;
39}
40
41const char *ProcessInfo::GetName() const {
42 return m_executable.GetFilename().GetCString();
43}
44
45size_t ProcessInfo::GetNameLength() const {
46 return m_executable.GetFilename().GetLength();
47}
48
49void ProcessInfo::Dump(Stream &s, Platform *platform) const {
50 s << "Executable: " << GetName() << "\n";
51 s << "Triple: ";
52 m_arch.DumpTriple(s);
53 s << "\n";
54
55 s << "Arguments:\n";
56 m_arguments.Dump(s);
57
58 s.Format("Environment:\n{0}", m_environment);
59}
60
61void ProcessInfo::SetExecutableFile(const FileSpec &exe_file,
62 bool add_exe_file_as_first_arg) {
63 if (exe_file) {
64 m_executable = exe_file;
65 if (add_exe_file_as_first_arg) {
66 llvm::SmallString<128> filename;
67 exe_file.GetPath(filename);
68 if (!filename.empty())
69 m_arguments.InsertArgumentAtIndex(0, filename);
70 }
71 } else {
72 m_executable.Clear();
73 }
74}
75
76llvm::StringRef ProcessInfo::GetArg0() const { return m_arg0; }
77
78void ProcessInfo::SetArg0(llvm::StringRef arg) { m_arg0 = arg; }
79
80void ProcessInfo::SetArguments(char const **argv,
81 bool first_arg_is_executable) {
82 m_arguments.SetArguments(argv);
83
84 // Is the first argument the executable?
85 if (first_arg_is_executable) {
86 const char *first_arg = m_arguments.GetArgumentAtIndex(0);
87 if (first_arg) {
88 // Yes the first argument is an executable, set it as the executable in
89 // the launch options. Don't resolve the file path as the path could be a
90 // remote platform path
91 m_executable.SetFile(first_arg, FileSpec::Style::native);
92 }
93 }
94}
95
96void ProcessInfo::SetArguments(const Args &args, bool first_arg_is_executable) {
97 // Copy all arguments
98 m_arguments = args;
99
100 // Is the first argument the executable?
101 if (first_arg_is_executable) {
102 const char *first_arg = m_arguments.GetArgumentAtIndex(0);
103 if (first_arg) {
104 // Yes the first argument is an executable, set it as the executable in
105 // the launch options. Don't resolve the file path as the path could be a
106 // remote platform path
107 m_executable.SetFile(first_arg, FileSpec::Style::native);
108 }
109 }
110}
111
112void ProcessInstanceInfo::Dump(Stream &s, UserIDResolver &resolver) const {
113 if (m_pid != LLDB_INVALID_PROCESS_ID)
114 s.Printf(" pid = %" PRIu64 "\n", m_pid);
115
116 if (m_parent_pid != LLDB_INVALID_PROCESS_ID)
117 s.Printf(" parent = %" PRIu64 "\n", m_parent_pid);
118
119 if (m_executable) {
120 s.Printf(" name = %s\n", m_executable.GetFilename().GetCString());
121 s.PutCString(" file = ");
122 m_executable.Dump(&s);
123 s.EOL();
124 }
125 const uint32_t argc = m_arguments.GetArgumentCount();
126 if (argc > 0) {
127 for (uint32_t i = 0; i < argc; i++) {
128 const char *arg = m_arguments.GetArgumentAtIndex(i);
129 if (i < 10)
130 s.Printf(" arg[%u] = %s\n", i, arg);
131 else
132 s.Printf("arg[%u] = %s\n", i, arg);
133 }
134 }
135
136 s.Format("{0}", m_environment);
137
138 if (m_arch.IsValid()) {
139 s.Printf(" arch = ");
140 m_arch.DumpTriple(s);
141 s.EOL();
142 }
143
144 if (UserIDIsValid()) {
145 s.Format(" uid = {0,-5} ({1})\n", GetUserID(),
146 resolver.GetUserName(GetUserID()).getValueOr(""));
147 }
148 if (GroupIDIsValid()) {
149 s.Format(" gid = {0,-5} ({1})\n", GetGroupID(),
150 resolver.GetGroupName(GetGroupID()).getValueOr(""));
151 }
152 if (EffectiveUserIDIsValid()) {
153 s.Format(" euid = {0,-5} ({1})\n", GetEffectiveUserID(),
154 resolver.GetUserName(GetEffectiveUserID()).getValueOr(""));
155 }
156 if (EffectiveGroupIDIsValid()) {
157 s.Format(" egid = {0,-5} ({1})\n", GetEffectiveGroupID(),
158 resolver.GetGroupName(GetEffectiveGroupID()).getValueOr(""));
159 }
160}
161
162void ProcessInstanceInfo::DumpTableHeader(Stream &s, bool show_args,
163 bool verbose) {
164 const char *label;
165 if (show_args || verbose)
166 label = "ARGUMENTS";
167 else
168 label = "NAME";
169
170 if (verbose) {
171 s.Printf("PID PARENT USER GROUP EFF USER EFF GROUP TRIPLE "
172 " %s\n",
173 label);
174 s.PutCString("====== ====== ========== ========== ========== ========== "
175 "======================== ============================\n");
176 } else {
177 s.Printf("PID PARENT USER TRIPLE %s\n", label);
178 s.PutCString("====== ====== ========== ======================== "
179 "============================\n");
180 }
181}
182
183void ProcessInstanceInfo::DumpAsTableRow(Stream &s, UserIDResolver &resolver,
184 bool show_args, bool verbose) const {
185 if (m_pid != LLDB_INVALID_PROCESS_ID) {
186 s.Printf("%-6" PRIu64 " %-6" PRIu64 " ", m_pid, m_parent_pid);
187
188 StreamString arch_strm;
189 if (m_arch.IsValid())
190 m_arch.DumpTriple(arch_strm);
191
192 auto print = [&](UserIDResolver::id_t id,
193 llvm::Optional<llvm::StringRef> (UserIDResolver::*get)(
194 UserIDResolver::id_t id)) {
195 if (auto name = (resolver.*get)(id))
196 s.Format("{0,-10} ", *name);
197 else
198 s.Format("{0,-10} ", id);
199 };
200 if (verbose) {
201 print(m_uid, &UserIDResolver::GetUserName);
202 print(m_gid, &UserIDResolver::GetGroupName);
203 print(m_euid, &UserIDResolver::GetUserName);
204 print(m_egid, &UserIDResolver::GetGroupName);
205
206 s.Printf("%-24s ", arch_strm.GetData());
207 } else {
208 print(m_euid, &UserIDResolver::GetUserName);
209 s.Printf(" %-24s ", arch_strm.GetData());
210 }
211
212 if (verbose || show_args) {
213 const uint32_t argc = m_arguments.GetArgumentCount();
214 if (argc > 0) {
215 for (uint32_t i = 0; i < argc; i++) {
216 if (i > 0)
217 s.PutChar(' ');
218 s.PutCString(m_arguments.GetArgumentAtIndex(i));
219 }
220 }
221 } else {
222 s.PutCString(GetName());
223 }
224
225 s.EOL();
226 }
227}
228
229bool ProcessInstanceInfoMatch::NameMatches(const char *process_name) const {
230 if (m_name_match_type == NameMatch::Ignore || process_name == nullptr)
231 return true;
232 const char *match_name = m_match_info.GetName();
233 if (!match_name)
234 return true;
235
236 return lldb_private::NameMatches(process_name, m_name_match_type, match_name);
237}
238
239bool ProcessInstanceInfoMatch::Matches(
240 const ProcessInstanceInfo &proc_info) const {
241 if (!NameMatches(proc_info.GetName()))
242 return false;
243
244 if (m_match_info.ProcessIDIsValid() &&
245 m_match_info.GetProcessID() != proc_info.GetProcessID())
246 return false;
247
248 if (m_match_info.ParentProcessIDIsValid() &&
249 m_match_info.GetParentProcessID() != proc_info.GetParentProcessID())
250 return false;
251
252 if (m_match_info.UserIDIsValid() &&
253 m_match_info.GetUserID() != proc_info.GetUserID())
254 return false;
255
256 if (m_match_info.GroupIDIsValid() &&
257 m_match_info.GetGroupID() != proc_info.GetGroupID())
258 return false;
259
260 if (m_match_info.EffectiveUserIDIsValid() &&
261 m_match_info.GetEffectiveUserID() != proc_info.GetEffectiveUserID())
262 return false;
263
264 if (m_match_info.EffectiveGroupIDIsValid() &&
265 m_match_info.GetEffectiveGroupID() != proc_info.GetEffectiveGroupID())
266 return false;
267
268 if (m_match_info.GetArchitecture().IsValid() &&
269 !m_match_info.GetArchitecture().IsCompatibleMatch(
270 proc_info.GetArchitecture()))
271 return false;
272 return true;
273}
274
275bool ProcessInstanceInfoMatch::MatchAllProcesses() const {
276 if (m_name_match_type != NameMatch::Ignore)
277 return false;
278
279 if (m_match_info.ProcessIDIsValid())
280 return false;
281
282 if (m_match_info.ParentProcessIDIsValid())
283 return false;
284
285 if (m_match_info.UserIDIsValid())
286 return false;
287
288 if (m_match_info.GroupIDIsValid())
289 return false;
290
291 if (m_match_info.EffectiveUserIDIsValid())
292 return false;
293
294 if (m_match_info.EffectiveGroupIDIsValid())
295 return false;
296
297 if (m_match_info.GetArchitecture().IsValid())
298 return false;
299
300 if (m_match_all_users)
301 return false;
302
303 return true;
304}
305
306void ProcessInstanceInfoMatch::Clear() {
307 m_match_info.Clear();
308 m_name_match_type = NameMatch::Ignore;
309 m_match_all_users = false;
310}