// Copyright (c) 2022 Klemens D. Morgenstern // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_PROCESS_V2_EXT_IMPL_ENV_IPP #define BOOST_PROCESS_V2_EXT_IMPL_ENV_IPP #include #include #include #include #include #if defined(BOOST_PROCESS_V2_WINDOWS) #include #else #include #endif #if (defined(__linux__) || defined(__ANDROID__)) #include #endif BOOST_PROCESS_V2_BEGIN_NAMESPACE namespace detail { namespace ext { #if defined(BOOST_PROCESS_V2_WINDOWS) void native_env_handle_deleter::operator()(native_env_handle_type h) const { delete [] h; } native_env_iterator next(native_env_iterator nh) { while (*nh != L'\0') nh ++; return ++nh ; } native_env_iterator find_end(native_env_iterator nh) { while (*nh - 1 != L'\0' && *nh != L'\0') nh ++; return nh ; } const environment::char_type * dereference(native_env_iterator iterator) { return iterator; } #elif (defined(__linux__) || defined(__ANDROID__)) //linux stores this as a blob with an EOF at the end void native_env_handle_deleter::operator()(native_env_handle_type h) const { delete [] h; } native_env_iterator next(native_env_iterator nh) { while (*nh != '\0') nh ++; return ++nh ; } native_env_iterator find_end(native_env_iterator nh) { while (*nh != EOF) nh ++; return nh ; } const environment::char_type * dereference(native_env_iterator iterator) { return iterator; } #elif (defined(__APPLE___) || defined(__MACH__)) void native_env_handle_deleter::operator()(native_env_handle_type h) const { delete [] h; } native_env_iterator next(native_env_iterator nh) { while (*nh != '\0') nh ++; return ++nh ; } native_env_iterator find_end(native_env_iterator nh) { while (*nh - 1 != '\0' && *nh != '\0') nh ++; return nh ; } const environment::char_type * dereference(native_env_iterator iterator) { return iterator; } #elif defined(__FreeBSD__) void native_env_handle_deleter::operator()(native_env_handle_type h) const { delete [] h; } native_env_iterator next(native_env_iterator nh) { return ++nh ; } native_env_iterator find_end(native_env_iterator nh) { while (*nh != nullptr) nh++; return nh ; } const environment::char_type * dereference(native_env_iterator iterator) { return *iterator; } #endif } } namespace ext { #if defined(BOOST_PROCESS_V2_WINDOWS) env_view env(HANDLE proc, boost::system::error_code & ec) { wchar_t *buffer = nullptr; PEB peb; SIZE_T nRead = 0; ULONG len = 0; PROCESS_BASIC_INFORMATION pbi; detail::ext::RTL_USER_PROCESS_PARAMETERS_EXTENDED upp; NTSTATUS status = 0; PVOID buf = nullptr; status = NtQueryInformationProcess(proc, ProcessBasicInformation, &pbi, sizeof(pbi), &len); ULONG error = RtlNtStatusToDosError(status); if (error) { BOOST_PROCESS_V2_ASSIGN_EC(ec, error, boost::system::system_category()) return {}; } if (!ReadProcessMemory(proc, pbi.PebBaseAddress, &peb, sizeof(peb), &nRead)) { BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) return {}; } if (!ReadProcessMemory(proc, peb.ProcessParameters, &upp, sizeof(upp), &nRead)) { BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) return {}; } env_view ev; buf = upp.Environment; len = (ULONG)upp.EnvironmentSize; ev.handle_.reset(new wchar_t[len / 2 + 1]()); if (!ReadProcessMemory(proc, buf, ev.handle_.get(), len, &nRead)) { BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) return {}; } ev.handle_.get()[len / 2] = L'\0'; return ev; } env_view env(HANDLE handle) { boost::system::error_code ec; auto res = env(handle, ec); if (ec) detail::throw_error(ec, "env"); return res; } env_view env(boost::process::v2::pid_type pid, boost::system::error_code & ec) { struct del { void operator()(HANDLE h) { ::CloseHandle(h); }; }; std::unique_ptr proc{detail::ext::open_process_with_debug_privilege(pid, ec)}; if (proc == nullptr) BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) else return env(proc.get(), ec); return {}; } #elif (defined(__APPLE___) || defined(__MACH__)) env_view env(boost::process::v2::pid_type pid, boost::system::error_code & ec) { int mib[3] = {CTL_KERN, KERN_ARGMAX, 0}; int argmax = 0; auto size = sizeof(argmax); if (sysctl(mib, 2, &argmax, &size, nullptr, 0) == -1) { BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) return {}; } std::string procargs; procargs.resize(argmax - 1); mib[1] = KERN_PROCARGS2; mib[2] = pid; size = argmax; if (sysctl(mib, 3, &*procargs.begin(), &size, nullptr, 0) != 0) { BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) return {}; } std::uint32_t nargs; memcpy(&nargs, &*procargs.begin(), sizeof(nargs)); char *cp = &*procargs.begin() + sizeof(nargs); for (; cp < &*procargs.end(); cp++) if (*cp == '\0') break; if (cp == &procargs[size]) return {}; for (; cp < &*procargs.end(); cp++) if (*cp != '\0') break; if (cp == &*procargs.end()) return {}; int i = 0; char *sp = cp; std::vector vec; while ((*sp != '\0' || i < nargs) && sp < &*procargs.end()) { if (i >= nargs) vec.push_back(*sp); sp += 1; } env_view ev; ev.handle_.reset(new char[vec.size()]()); std::copy(vec.begin(), vec.end(), ev.handle_.get()); return ev; } #elif (defined(__linux__) || defined(__ANDROID__)) env_view env(boost::process::v2::pid_type pid, boost::system::error_code & ec) { std::size_t size = 0; std::unique_ptr procargs{}; int f = ::open(("/proc/" + std::to_string(pid) + "/environ").c_str(), O_RDONLY); while (!procargs || procargs.get()[size - 1] != EOF) { std::unique_ptr buf{new char[size + 4096]}; if (size > 0) std::memmove(buf.get(), procargs.get(), size); auto r = ::read(f, buf.get() + size, 4096); if (r < 0) { BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) ::close(f); return {}; } procargs = std::move(buf); size += r; if (r < 4096) // done! { procargs.get()[size] = EOF; break; } } ::close(f); env_view ev; ev.handle_ = std::move(procargs); return ev; } #elif defined(__FreeBSD__) env_view env(boost::process::v2::pid_type pid, boost::system::error_code & ec) { env_view ev; unsigned cntp = 0; procstat *proc_stat = procstat_open_sysctl(); if (proc_stat != nullptr) { kinfo_proc *proc_info = procstat_getprocs(proc_stat, KERN_PROC_PID, pid, &cntp); if (proc_info != nullptr) { char **env = procstat_getenvv(proc_stat, proc_info, 0); if (env != nullptr) { auto e = env; std::size_t n = 0u, len = 0u; while (e && *e != nullptr) { n ++; len += std::strlen(*e); e++; } std::size_t mem_needed = // environ - nullptr - strlen + null terminators (n * sizeof(char*)) + sizeof(char*) + len + n; char * out = new (std::nothrow) char[mem_needed]; if (out != nullptr) { auto eno = reinterpret_cast(out); auto eeo = eno; auto str = out + (n * sizeof(char*)) + sizeof(char*); e = env; while (*e != nullptr) { auto len = std::strlen(*e) + 1u; std::memcpy(str, *e, len); *eno = str; str += len; eno ++; e++; } *eno = nullptr; ev.handle_.reset(eeo); } else BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) } procstat_freeprocs(proc_stat, proc_info); } else BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) procstat_close(proc_stat); } else BOOST_PROCESS_V2_ASSIGN_LAST_ERROR(ec) return ev; } #endif env_view env(boost::process::v2::pid_type pid) { boost::system::error_code ec; auto res = env(pid, ec); if (ec) detail::throw_error(ec, "env"); return res; } } BOOST_PROCESS_V2_END_NAMESPACE #endif //BOOST_PROCESS_V2_EXT_IMPL_ENV_IPP