mirror of
https://github.com/bitcoin/bitcoin.git
synced 2026-01-20 15:19:07 +01:00
158 lines
5.4 KiB
C++
158 lines
5.4 KiB
C++
// Copyright (c) 2021-2022 The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include <ipc/process.h>
|
|
#include <ipc/protocol.h>
|
|
#include <logging.h>
|
|
#include <mp/util.h>
|
|
#include <tinyformat.h>
|
|
#include <util/fs.h>
|
|
#include <util/strencodings.h>
|
|
#include <util/syserror.h>
|
|
|
|
#include <cstdint>
|
|
#include <cstdlib>
|
|
#include <errno.h>
|
|
#include <exception>
|
|
#include <iostream>
|
|
#include <stdexcept>
|
|
#include <string.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
#include <unistd.h>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
using util::RemovePrefixView;
|
|
|
|
namespace ipc {
|
|
namespace {
|
|
class ProcessImpl : public Process
|
|
{
|
|
public:
|
|
int spawn(const std::string& new_exe_name, const fs::path& argv0_path, int& pid) override
|
|
{
|
|
return mp::SpawnProcess(pid, [&](int fd) {
|
|
fs::path path = argv0_path;
|
|
path.remove_filename();
|
|
path /= fs::PathFromString(new_exe_name);
|
|
return std::vector<std::string>{fs::PathToString(path), "-ipcfd", strprintf("%i", fd)};
|
|
});
|
|
}
|
|
int waitSpawned(int pid) override { return mp::WaitProcess(pid); }
|
|
bool checkSpawned(int argc, char* argv[], int& fd) override
|
|
{
|
|
// If this process was not started with a single -ipcfd argument, it is
|
|
// not a process spawned by the spawn() call above, so return false and
|
|
// do not try to serve requests.
|
|
if (argc != 3 || strcmp(argv[1], "-ipcfd") != 0) {
|
|
return false;
|
|
}
|
|
// If a single -ipcfd argument was provided, return true and get the
|
|
// file descriptor so Protocol::serve() can be called to handle
|
|
// requests from the parent process. The -ipcfd argument is not valid
|
|
// in combination with other arguments because the parent process
|
|
// should be able to control the child process through the IPC protocol
|
|
// without passing information out of band.
|
|
const auto maybe_fd{ToIntegral<int32_t>(argv[2])};
|
|
if (!maybe_fd) {
|
|
throw std::runtime_error(strprintf("Invalid -ipcfd number '%s'", argv[2]));
|
|
}
|
|
fd = *maybe_fd;
|
|
return true;
|
|
}
|
|
int connect(const fs::path& data_dir,
|
|
const std::string& dest_exe_name,
|
|
std::string& address) override;
|
|
int bind(const fs::path& data_dir, const std::string& exe_name, std::string& address) override;
|
|
};
|
|
|
|
static bool ParseAddress(std::string& address,
|
|
const fs::path& data_dir,
|
|
const std::string& dest_exe_name,
|
|
struct sockaddr_un& addr,
|
|
std::string& error)
|
|
{
|
|
if (address == "unix" || address.starts_with("unix:")) {
|
|
fs::path path;
|
|
if (address.size() <= 5) {
|
|
path = data_dir / fs::PathFromString(strprintf("%s.sock", RemovePrefixView(dest_exe_name, "bitcoin-")));
|
|
} else {
|
|
path = data_dir / fs::PathFromString(address.substr(5));
|
|
}
|
|
std::string path_str = fs::PathToString(path);
|
|
address = strprintf("unix:%s", path_str);
|
|
if (path_str.size() >= sizeof(addr.sun_path)) {
|
|
error = strprintf("Unix address path %s exceeded maximum socket path length", fs::quoted(fs::PathToString(path)));
|
|
return false;
|
|
}
|
|
memset(&addr, 0, sizeof(addr));
|
|
addr.sun_family = AF_UNIX;
|
|
strncpy(addr.sun_path, path_str.c_str(), sizeof(addr.sun_path)-1);
|
|
return true;
|
|
}
|
|
|
|
error = strprintf("Unrecognized address '%s'", address);
|
|
return false;
|
|
}
|
|
|
|
int ProcessImpl::connect(const fs::path& data_dir,
|
|
const std::string& dest_exe_name,
|
|
std::string& address)
|
|
{
|
|
struct sockaddr_un addr;
|
|
std::string error;
|
|
if (!ParseAddress(address, data_dir, dest_exe_name, addr, error)) {
|
|
throw std::invalid_argument(error);
|
|
}
|
|
|
|
int fd;
|
|
if ((fd = ::socket(addr.sun_family, SOCK_STREAM, 0)) == -1) {
|
|
throw std::system_error(errno, std::system_category());
|
|
}
|
|
if (::connect(fd, (struct sockaddr*)&addr, sizeof(addr)) == 0) {
|
|
return fd;
|
|
}
|
|
int connect_error = errno;
|
|
if (::close(fd) != 0) {
|
|
LogPrintf("Error closing file descriptor %i '%s': %s\n", fd, address, SysErrorString(errno));
|
|
}
|
|
throw std::system_error(connect_error, std::system_category());
|
|
}
|
|
|
|
int ProcessImpl::bind(const fs::path& data_dir, const std::string& exe_name, std::string& address)
|
|
{
|
|
struct sockaddr_un addr;
|
|
std::string error;
|
|
if (!ParseAddress(address, data_dir, exe_name, addr, error)) {
|
|
throw std::invalid_argument(error);
|
|
}
|
|
|
|
if (addr.sun_family == AF_UNIX) {
|
|
fs::path path = addr.sun_path;
|
|
if (path.has_parent_path()) fs::create_directories(path.parent_path());
|
|
if (fs::symlink_status(path).type() == fs::file_type::socket) {
|
|
fs::remove(path);
|
|
}
|
|
}
|
|
|
|
int fd;
|
|
if ((fd = ::socket(addr.sun_family, SOCK_STREAM, 0)) == -1) {
|
|
throw std::system_error(errno, std::system_category());
|
|
}
|
|
|
|
if (::bind(fd, (struct sockaddr*)&addr, sizeof(addr)) == 0) {
|
|
return fd;
|
|
}
|
|
int bind_error = errno;
|
|
if (::close(fd) != 0) {
|
|
LogPrintf("Error closing file descriptor %i: %s\n", fd, SysErrorString(errno));
|
|
}
|
|
throw std::system_error(bind_error, std::system_category());
|
|
}
|
|
} // namespace
|
|
|
|
std::unique_ptr<Process> MakeProcess() { return std::make_unique<ProcessImpl>(); }
|
|
} // namespace ipc
|