xref: /freebsd-src/contrib/llvm-project/lldb/source/Host/netbsd/HostNetBSD.cpp (revision e8d8bef961a50d4dc22501cde4fb9fb0be1b2532)
15ffd83dbSDimitry Andric //===-- source/Host/netbsd/HostNetBSD.cpp ---------------------------------===//
25ffd83dbSDimitry Andric //
35ffd83dbSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
45ffd83dbSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
55ffd83dbSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65ffd83dbSDimitry Andric //
75ffd83dbSDimitry Andric //===----------------------------------------------------------------------===//
85ffd83dbSDimitry Andric 
95ffd83dbSDimitry Andric #include <dlfcn.h>
105ffd83dbSDimitry Andric #include <execinfo.h>
115ffd83dbSDimitry Andric #include <stdio.h>
125ffd83dbSDimitry Andric #include <sys/proc.h>
135ffd83dbSDimitry Andric #include <sys/sysctl.h>
145ffd83dbSDimitry Andric #include <sys/types.h>
155ffd83dbSDimitry Andric 
165ffd83dbSDimitry Andric #include <limits.h>
175ffd83dbSDimitry Andric 
185ffd83dbSDimitry Andric #include <kvm.h>
195ffd83dbSDimitry Andric #include <sys/exec.h>
205ffd83dbSDimitry Andric #include <sys/ptrace.h>
215ffd83dbSDimitry Andric 
225ffd83dbSDimitry Andric #include "lldb/Host/FileSystem.h"
235ffd83dbSDimitry Andric #include "lldb/Host/Host.h"
245ffd83dbSDimitry Andric #include "lldb/Host/HostInfo.h"
255ffd83dbSDimitry Andric #include "lldb/Utility/DataBufferHeap.h"
265ffd83dbSDimitry Andric #include "lldb/Utility/DataExtractor.h"
275ffd83dbSDimitry Andric #include "lldb/Utility/Endian.h"
285ffd83dbSDimitry Andric #include "lldb/Utility/Log.h"
295ffd83dbSDimitry Andric #include "lldb/Utility/NameMatches.h"
305ffd83dbSDimitry Andric #include "lldb/Utility/ProcessInfo.h"
315ffd83dbSDimitry Andric #include "lldb/Utility/Status.h"
325ffd83dbSDimitry Andric #include "lldb/Utility/StreamString.h"
335ffd83dbSDimitry Andric 
345ffd83dbSDimitry Andric #include "llvm/Object/ELF.h"
355ffd83dbSDimitry Andric #include "llvm/Support/Host.h"
365ffd83dbSDimitry Andric 
375ffd83dbSDimitry Andric extern "C" {
385ffd83dbSDimitry Andric extern char **environ;
395ffd83dbSDimitry Andric }
405ffd83dbSDimitry Andric 
415ffd83dbSDimitry Andric using namespace lldb;
425ffd83dbSDimitry Andric using namespace lldb_private;
435ffd83dbSDimitry Andric 
445ffd83dbSDimitry Andric namespace lldb_private {
455ffd83dbSDimitry Andric class ProcessLaunchInfo;
465ffd83dbSDimitry Andric }
475ffd83dbSDimitry Andric 
485ffd83dbSDimitry Andric Environment Host::GetEnvironment() { return Environment(environ); }
495ffd83dbSDimitry Andric 
505ffd83dbSDimitry Andric static bool GetNetBSDProcessArgs(const ProcessInstanceInfoMatch *match_info_ptr,
515ffd83dbSDimitry Andric                                  ProcessInstanceInfo &process_info) {
525ffd83dbSDimitry Andric   if (!process_info.ProcessIDIsValid())
535ffd83dbSDimitry Andric     return false;
545ffd83dbSDimitry Andric 
555ffd83dbSDimitry Andric   int pid = process_info.GetProcessID();
565ffd83dbSDimitry Andric 
575ffd83dbSDimitry Andric   int mib[4] = {CTL_KERN, KERN_PROC_ARGS, pid, KERN_PROC_ARGV};
585ffd83dbSDimitry Andric 
595ffd83dbSDimitry Andric   char arg_data[8192];
605ffd83dbSDimitry Andric   size_t arg_data_size = sizeof(arg_data);
615ffd83dbSDimitry Andric   if (::sysctl(mib, 4, arg_data, &arg_data_size, NULL, 0) != 0)
625ffd83dbSDimitry Andric     return false;
635ffd83dbSDimitry Andric 
645ffd83dbSDimitry Andric   DataExtractor data(arg_data, arg_data_size, endian::InlHostByteOrder(),
655ffd83dbSDimitry Andric                      sizeof(void *));
665ffd83dbSDimitry Andric   lldb::offset_t offset = 0;
675ffd83dbSDimitry Andric   const char *cstr;
685ffd83dbSDimitry Andric 
695ffd83dbSDimitry Andric   cstr = data.GetCStr(&offset);
705ffd83dbSDimitry Andric   if (!cstr)
715ffd83dbSDimitry Andric     return false;
725ffd83dbSDimitry Andric 
735ffd83dbSDimitry Andric   process_info.GetExecutableFile().SetFile(cstr,
745ffd83dbSDimitry Andric                                            FileSpec::Style::native);
755ffd83dbSDimitry Andric 
765ffd83dbSDimitry Andric   if (!(match_info_ptr == NULL ||
775ffd83dbSDimitry Andric         NameMatches(process_info.GetExecutableFile().GetFilename().GetCString(),
785ffd83dbSDimitry Andric                     match_info_ptr->GetNameMatchType(),
795ffd83dbSDimitry Andric                     match_info_ptr->GetProcessInfo().GetName())))
805ffd83dbSDimitry Andric     return false;
815ffd83dbSDimitry Andric 
825ffd83dbSDimitry Andric   process_info.SetArg0(cstr);
835ffd83dbSDimitry Andric   Args &proc_args = process_info.GetArguments();
845ffd83dbSDimitry Andric   while (1) {
855ffd83dbSDimitry Andric     const uint8_t *p = data.PeekData(offset, 1);
865ffd83dbSDimitry Andric     while ((p != NULL) && (*p == '\0') && offset < arg_data_size) {
875ffd83dbSDimitry Andric       ++offset;
885ffd83dbSDimitry Andric       p = data.PeekData(offset, 1);
895ffd83dbSDimitry Andric     }
905ffd83dbSDimitry Andric     if (p == NULL || offset >= arg_data_size)
915ffd83dbSDimitry Andric       break;
925ffd83dbSDimitry Andric 
935ffd83dbSDimitry Andric     cstr = data.GetCStr(&offset);
945ffd83dbSDimitry Andric     if (!cstr)
955ffd83dbSDimitry Andric       break;
965ffd83dbSDimitry Andric 
975ffd83dbSDimitry Andric     proc_args.AppendArgument(llvm::StringRef(cstr));
985ffd83dbSDimitry Andric   }
995ffd83dbSDimitry Andric 
1005ffd83dbSDimitry Andric   return true;
1015ffd83dbSDimitry Andric }
1025ffd83dbSDimitry Andric 
1035ffd83dbSDimitry Andric static bool GetNetBSDProcessCPUType(ProcessInstanceInfo &process_info) {
1045ffd83dbSDimitry Andric   Log *log = GetLogIfAllCategoriesSet(LIBLLDB_LOG_HOST);
1055ffd83dbSDimitry Andric 
1065ffd83dbSDimitry Andric   if (process_info.ProcessIDIsValid()) {
1075ffd83dbSDimitry Andric     auto buffer_sp = FileSystem::Instance().CreateDataBuffer(
1085ffd83dbSDimitry Andric         process_info.GetExecutableFile(), 0x20, 0);
1095ffd83dbSDimitry Andric     if (buffer_sp) {
1105ffd83dbSDimitry Andric       uint8_t exe_class =
1115ffd83dbSDimitry Andric           llvm::object::getElfArchType(
1125ffd83dbSDimitry Andric               {buffer_sp->GetChars(), size_t(buffer_sp->GetByteSize())})
1135ffd83dbSDimitry Andric               .first;
1145ffd83dbSDimitry Andric 
1155ffd83dbSDimitry Andric       switch (exe_class) {
1165ffd83dbSDimitry Andric       case llvm::ELF::ELFCLASS32:
1175ffd83dbSDimitry Andric         process_info.GetArchitecture() =
1185ffd83dbSDimitry Andric             HostInfo::GetArchitecture(HostInfo::eArchKind32);
1195ffd83dbSDimitry Andric         return true;
1205ffd83dbSDimitry Andric       case llvm::ELF::ELFCLASS64:
1215ffd83dbSDimitry Andric         process_info.GetArchitecture() =
1225ffd83dbSDimitry Andric             HostInfo::GetArchitecture(HostInfo::eArchKind64);
1235ffd83dbSDimitry Andric         return true;
1245ffd83dbSDimitry Andric       default:
1255ffd83dbSDimitry Andric         LLDB_LOG(log, "Unknown elf class ({0}) in file {1}", exe_class,
1265ffd83dbSDimitry Andric                  process_info.GetExecutableFile());
1275ffd83dbSDimitry Andric       }
1285ffd83dbSDimitry Andric     }
1295ffd83dbSDimitry Andric   }
1305ffd83dbSDimitry Andric   process_info.GetArchitecture().Clear();
1315ffd83dbSDimitry Andric   return false;
1325ffd83dbSDimitry Andric }
1335ffd83dbSDimitry Andric 
1345ffd83dbSDimitry Andric static bool GetNetBSDProcessUserAndGroup(ProcessInstanceInfo &process_info) {
1355ffd83dbSDimitry Andric   ::kvm_t *kdp;
1365ffd83dbSDimitry Andric   char errbuf[_POSIX2_LINE_MAX]; /* XXX: error string unused */
1375ffd83dbSDimitry Andric 
1385ffd83dbSDimitry Andric   struct ::kinfo_proc2 *proc_kinfo;
1395ffd83dbSDimitry Andric   const int pid = process_info.GetProcessID();
1405ffd83dbSDimitry Andric   int nproc;
1415ffd83dbSDimitry Andric 
1425ffd83dbSDimitry Andric   if (!process_info.ProcessIDIsValid())
1435ffd83dbSDimitry Andric     goto error;
1445ffd83dbSDimitry Andric 
1455ffd83dbSDimitry Andric   if ((kdp = ::kvm_openfiles(NULL, NULL, NULL, KVM_NO_FILES, errbuf)) == NULL)
1465ffd83dbSDimitry Andric     goto error;
1475ffd83dbSDimitry Andric 
1485ffd83dbSDimitry Andric   if ((proc_kinfo = ::kvm_getproc2(kdp, KERN_PROC_PID, pid,
1495ffd83dbSDimitry Andric                                    sizeof(struct ::kinfo_proc2), &nproc)) ==
1505ffd83dbSDimitry Andric       NULL) {
1515ffd83dbSDimitry Andric     ::kvm_close(kdp);
1525ffd83dbSDimitry Andric     goto error;
1535ffd83dbSDimitry Andric   }
1545ffd83dbSDimitry Andric 
1555ffd83dbSDimitry Andric   if (nproc < 1) {
1565ffd83dbSDimitry Andric     ::kvm_close(kdp); /* XXX: we don't check for error here */
1575ffd83dbSDimitry Andric     goto error;
1585ffd83dbSDimitry Andric   }
1595ffd83dbSDimitry Andric 
1605ffd83dbSDimitry Andric   process_info.SetParentProcessID(proc_kinfo->p_ppid);
1615ffd83dbSDimitry Andric   process_info.SetUserID(proc_kinfo->p_ruid);
1625ffd83dbSDimitry Andric   process_info.SetGroupID(proc_kinfo->p_rgid);
1635ffd83dbSDimitry Andric   process_info.SetEffectiveUserID(proc_kinfo->p_uid);
1645ffd83dbSDimitry Andric   process_info.SetEffectiveGroupID(proc_kinfo->p_gid);
1655ffd83dbSDimitry Andric 
1665ffd83dbSDimitry Andric   ::kvm_close(kdp); /* XXX: we don't check for error here */
1675ffd83dbSDimitry Andric 
1685ffd83dbSDimitry Andric   return true;
1695ffd83dbSDimitry Andric 
1705ffd83dbSDimitry Andric error:
1715ffd83dbSDimitry Andric   process_info.SetParentProcessID(LLDB_INVALID_PROCESS_ID);
1725ffd83dbSDimitry Andric   process_info.SetUserID(UINT32_MAX);
1735ffd83dbSDimitry Andric   process_info.SetGroupID(UINT32_MAX);
1745ffd83dbSDimitry Andric   process_info.SetEffectiveUserID(UINT32_MAX);
1755ffd83dbSDimitry Andric   process_info.SetEffectiveGroupID(UINT32_MAX);
1765ffd83dbSDimitry Andric   return false;
1775ffd83dbSDimitry Andric }
1785ffd83dbSDimitry Andric 
1795ffd83dbSDimitry Andric uint32_t Host::FindProcessesImpl(const ProcessInstanceInfoMatch &match_info,
1805ffd83dbSDimitry Andric                                  ProcessInstanceInfoList &process_infos) {
1815ffd83dbSDimitry Andric   const ::pid_t our_pid = ::getpid();
1825ffd83dbSDimitry Andric   const ::uid_t our_uid = ::getuid();
1835ffd83dbSDimitry Andric 
1845ffd83dbSDimitry Andric   const bool all_users =
1855ffd83dbSDimitry Andric       match_info.GetMatchAllUsers() ||
1865ffd83dbSDimitry Andric       // Special case, if lldb is being run as root we can attach to anything
1875ffd83dbSDimitry Andric       (our_uid == 0);
1885ffd83dbSDimitry Andric 
1895ffd83dbSDimitry Andric   kvm_t *kdp;
1905ffd83dbSDimitry Andric   char errbuf[_POSIX2_LINE_MAX]; /* XXX: error string unused */
1915ffd83dbSDimitry Andric   if ((kdp = ::kvm_openfiles(NULL, NULL, NULL, KVM_NO_FILES, errbuf)) == NULL)
1925ffd83dbSDimitry Andric     return 0;
1935ffd83dbSDimitry Andric 
1945ffd83dbSDimitry Andric   struct ::kinfo_proc2 *proc_kinfo;
1955ffd83dbSDimitry Andric   int nproc;
1965ffd83dbSDimitry Andric   if ((proc_kinfo = ::kvm_getproc2(kdp, KERN_PROC_ALL, 0,
1975ffd83dbSDimitry Andric                                    sizeof(struct ::kinfo_proc2), &nproc)) ==
1985ffd83dbSDimitry Andric       NULL) {
1995ffd83dbSDimitry Andric     ::kvm_close(kdp);
2005ffd83dbSDimitry Andric     return 0;
2015ffd83dbSDimitry Andric   }
2025ffd83dbSDimitry Andric 
203*e8d8bef9SDimitry Andric   ProcessInstanceInfoMatch match_info_noname{match_info};
204*e8d8bef9SDimitry Andric   match_info_noname.SetNameMatchType(NameMatch::Ignore);
205*e8d8bef9SDimitry Andric 
2065ffd83dbSDimitry Andric   for (int i = 0; i < nproc; i++) {
2075ffd83dbSDimitry Andric     if (proc_kinfo[i].p_pid < 1)
2085ffd83dbSDimitry Andric       continue; /* not valid */
2095ffd83dbSDimitry Andric     /* Make sure the user is acceptable */
2105ffd83dbSDimitry Andric     if (!all_users && proc_kinfo[i].p_ruid != our_uid)
2115ffd83dbSDimitry Andric       continue;
2125ffd83dbSDimitry Andric 
2135ffd83dbSDimitry Andric     if (proc_kinfo[i].p_pid == our_pid ||  // Skip this process
2145ffd83dbSDimitry Andric         proc_kinfo[i].p_pid == 0 ||        // Skip kernel (kernel pid is 0)
2155ffd83dbSDimitry Andric         proc_kinfo[i].p_stat == LSZOMB ||  // Zombies are bad
2165ffd83dbSDimitry Andric         proc_kinfo[i].p_flag & P_TRACED || // Being debugged?
2175ffd83dbSDimitry Andric         proc_kinfo[i].p_flag & P_WEXIT)    // Working on exiting
2185ffd83dbSDimitry Andric       continue;
2195ffd83dbSDimitry Andric 
2205ffd83dbSDimitry Andric     // Every thread is a process in NetBSD, but all the threads of a single
2215ffd83dbSDimitry Andric     // process have the same pid. Do not store the process info in the result
2225ffd83dbSDimitry Andric     // list if a process with given identifier is already registered there.
2235ffd83dbSDimitry Andric     if (proc_kinfo[i].p_nlwps > 1) {
2245ffd83dbSDimitry Andric       bool already_registered = false;
2255ffd83dbSDimitry Andric       for (size_t pi = 0; pi < process_infos.size(); pi++) {
226*e8d8bef9SDimitry Andric         if ((::pid_t)process_infos[pi].GetProcessID() == proc_kinfo[i].p_pid) {
2275ffd83dbSDimitry Andric           already_registered = true;
2285ffd83dbSDimitry Andric           break;
2295ffd83dbSDimitry Andric         }
2305ffd83dbSDimitry Andric       }
2315ffd83dbSDimitry Andric 
2325ffd83dbSDimitry Andric       if (already_registered)
2335ffd83dbSDimitry Andric         continue;
2345ffd83dbSDimitry Andric     }
2355ffd83dbSDimitry Andric     ProcessInstanceInfo process_info;
2365ffd83dbSDimitry Andric     process_info.SetProcessID(proc_kinfo[i].p_pid);
2375ffd83dbSDimitry Andric     process_info.SetParentProcessID(proc_kinfo[i].p_ppid);
2385ffd83dbSDimitry Andric     process_info.SetUserID(proc_kinfo[i].p_ruid);
2395ffd83dbSDimitry Andric     process_info.SetGroupID(proc_kinfo[i].p_rgid);
2405ffd83dbSDimitry Andric     process_info.SetEffectiveUserID(proc_kinfo[i].p_uid);
2415ffd83dbSDimitry Andric     process_info.SetEffectiveGroupID(proc_kinfo[i].p_gid);
2425ffd83dbSDimitry Andric     // Make sure our info matches before we go fetch the name and cpu type
243*e8d8bef9SDimitry Andric     if (match_info_noname.Matches(process_info) &&
2445ffd83dbSDimitry Andric         GetNetBSDProcessArgs(&match_info, process_info)) {
2455ffd83dbSDimitry Andric       GetNetBSDProcessCPUType(process_info);
2465ffd83dbSDimitry Andric       if (match_info.Matches(process_info))
2475ffd83dbSDimitry Andric         process_infos.push_back(process_info);
2485ffd83dbSDimitry Andric     }
2495ffd83dbSDimitry Andric   }
2505ffd83dbSDimitry Andric 
2515ffd83dbSDimitry Andric   kvm_close(kdp); /* XXX: we don't check for error here */
2525ffd83dbSDimitry Andric 
2535ffd83dbSDimitry Andric   return process_infos.size();
2545ffd83dbSDimitry Andric }
2555ffd83dbSDimitry Andric 
2565ffd83dbSDimitry Andric bool Host::GetProcessInfo(lldb::pid_t pid, ProcessInstanceInfo &process_info) {
2575ffd83dbSDimitry Andric   process_info.SetProcessID(pid);
2585ffd83dbSDimitry Andric 
2595ffd83dbSDimitry Andric   if (GetNetBSDProcessArgs(NULL, process_info)) {
2605ffd83dbSDimitry Andric     GetNetBSDProcessCPUType(process_info);
2615ffd83dbSDimitry Andric     GetNetBSDProcessUserAndGroup(process_info);
2625ffd83dbSDimitry Andric     return true;
2635ffd83dbSDimitry Andric   }
2645ffd83dbSDimitry Andric 
2655ffd83dbSDimitry Andric   process_info.Clear();
2665ffd83dbSDimitry Andric   return false;
2675ffd83dbSDimitry Andric }
2685ffd83dbSDimitry Andric 
2695ffd83dbSDimitry Andric Status Host::ShellExpandArguments(ProcessLaunchInfo &launch_info) {
2705ffd83dbSDimitry Andric   return Status("unimplemented");
2715ffd83dbSDimitry Andric }
272