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