#include "remoting/host/native_messaging/pipe_messaging_channel.h"
#include <utility>
#include "base/check.h"
#include "base/files/file.h"
#include "base/files/file_path.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/functional/callback_helpers.h"
#include "base/location.h"
#include "base/process/process_info.h"
#include "base/values.h"
#include "build/build_config.h"
#if BUILDFLAG(IS_POSIX)
#include <unistd.h>
#include "base/posix/eintr_wrapper.h"
#endif
namespace remoting {
namespace {
#if BUILDFLAG(IS_POSIX)
base::File DuplicateAndBlock(int fd, base::File::Flags read_write_flag) {
base::File original{fd};
base::File target_dup = original.Duplicate();
int original_fd = original.TakePlatformFile();
CHECK_EQ(original_fd, fd);
base::File dev_null = base::File(base::FilePath("/dev/null"),
base::File::FLAG_OPEN | read_write_flag);
int new_fd = HANDLE_EINTR(dup2(dev_null.GetPlatformFile(), fd));
PCHECK(new_fd != -1) << "Unexpected error when executing dup2 with fd " << fd;
DCHECK_EQ(new_fd, fd);
return target_dup;
}
#endif
}
PipeMessagingChannel::PipeMessagingChannel(base::File input, base::File output)
: native_messaging_reader_(std::move(input)),
native_messaging_writer_(new NativeMessagingWriter(std::move(output))),
event_handler_(nullptr) {
weak_ptr_ = weak_factory_.GetWeakPtr();
}
PipeMessagingChannel::~PipeMessagingChannel() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
}
#if BUILDFLAG(IS_POSIX)
void PipeMessagingChannel::OpenAndBlockStdio(base::File& stdin_file,
base::File& stdout_file) {
stdin_file = DuplicateAndBlock(STDIN_FILENO, base::File::FLAG_READ);
stdout_file = DuplicateAndBlock(STDOUT_FILENO, base::File::FLAG_WRITE);
}
#endif
void PipeMessagingChannel::Start(EventHandler* event_handler) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(!event_handler_);
event_handler_ = event_handler;
DCHECK(event_handler_);
native_messaging_reader_.Start(
base::BindRepeating(&PipeMessagingChannel::ProcessMessage, weak_ptr_),
base::BindOnce(&PipeMessagingChannel::Shutdown, weak_ptr_));
}
void PipeMessagingChannel::ProcessMessage(base::Value message) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (event_handler_) {
event_handler_->OnMessage(std::move(message));
}
}
void PipeMessagingChannel::SendMessage(std::optional<base::ValueView> message) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
bool success = message && native_messaging_writer_;
if (success) {
success = native_messaging_writer_->WriteMessage(*message);
}
if (!success) {
native_messaging_writer_.reset();
Shutdown();
}
}
void PipeMessagingChannel::Shutdown() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (event_handler_) {
EventHandler* handler = event_handler_;
event_handler_ = nullptr;
handler->OnDisconnect();
}
}
}