mirror of
https://github.com/AGWA/git-crypt.git
synced 2026-04-28 11:53:12 -07:00
Add Coprocess class
It provides a convenient way to spawn a process and read from/write to its stdin/stdout.
This commit is contained in:
4
Makefile
4
Makefile
@@ -10,7 +10,9 @@ OBJFILES = \
|
|||||||
gpg.o \
|
gpg.o \
|
||||||
key.o \
|
key.o \
|
||||||
util.o \
|
util.o \
|
||||||
parse_options.o
|
parse_options.o \
|
||||||
|
coprocess.o \
|
||||||
|
fhstream.o
|
||||||
|
|
||||||
OBJFILES += crypto-openssl.o
|
OBJFILES += crypto-openssl.o
|
||||||
LDFLAGS += -lcrypto
|
LDFLAGS += -lcrypto
|
||||||
|
|||||||
186
coprocess-unix.cpp
Normal file
186
coprocess-unix.cpp
Normal file
@@ -0,0 +1,186 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 Andrew Ayer
|
||||||
|
*
|
||||||
|
* This file is part of git-crypt.
|
||||||
|
*
|
||||||
|
* git-crypt is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* git-crypt is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with git-crypt. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
* Additional permission under GNU GPL version 3 section 7:
|
||||||
|
*
|
||||||
|
* If you modify the Program, or any covered work, by linking or
|
||||||
|
* combining it with the OpenSSL project's OpenSSL library (or a
|
||||||
|
* modified version of that library), containing parts covered by the
|
||||||
|
* terms of the OpenSSL or SSLeay licenses, the licensors of the Program
|
||||||
|
* grant you additional permission to convey the resulting work.
|
||||||
|
* Corresponding Source for a non-source form of such a combination
|
||||||
|
* shall include the source code for the parts of OpenSSL used as well
|
||||||
|
* as that of the covered work.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "coprocess.hpp"
|
||||||
|
#include "util.hpp"
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/wait.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
static int execvp (const std::string& file, const std::vector<std::string>& args)
|
||||||
|
{
|
||||||
|
std::vector<const char*> args_c_str;
|
||||||
|
args_c_str.reserve(args.size());
|
||||||
|
for (std::vector<std::string>::const_iterator arg(args.begin()); arg != args.end(); ++arg) {
|
||||||
|
args_c_str.push_back(arg->c_str());
|
||||||
|
}
|
||||||
|
args_c_str.push_back(NULL);
|
||||||
|
return execvp(file.c_str(), const_cast<char**>(&args_c_str[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
Coprocess::Coprocess ()
|
||||||
|
{
|
||||||
|
pid = -1;
|
||||||
|
stdin_pipe_reader = -1;
|
||||||
|
stdin_pipe_writer = -1;
|
||||||
|
stdin_pipe_ostream = NULL;
|
||||||
|
stdout_pipe_reader = -1;
|
||||||
|
stdout_pipe_writer = -1;
|
||||||
|
stdout_pipe_istream = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
Coprocess::~Coprocess ()
|
||||||
|
{
|
||||||
|
close_stdin();
|
||||||
|
close_stdout();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostream* Coprocess::stdin_pipe ()
|
||||||
|
{
|
||||||
|
if (!stdin_pipe_ostream) {
|
||||||
|
int fds[2];
|
||||||
|
if (pipe(fds) == -1) {
|
||||||
|
throw System_error("pipe", "", errno);
|
||||||
|
}
|
||||||
|
stdin_pipe_reader = fds[0];
|
||||||
|
stdin_pipe_writer = fds[1];
|
||||||
|
stdin_pipe_ostream = new ofhstream(this, write_stdin);
|
||||||
|
}
|
||||||
|
return stdin_pipe_ostream;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Coprocess::close_stdin ()
|
||||||
|
{
|
||||||
|
delete stdin_pipe_ostream;
|
||||||
|
stdin_pipe_ostream = NULL;
|
||||||
|
if (stdin_pipe_writer != -1) {
|
||||||
|
close(stdin_pipe_writer);
|
||||||
|
stdin_pipe_writer = -1;
|
||||||
|
}
|
||||||
|
if (stdin_pipe_reader != -1) {
|
||||||
|
close(stdin_pipe_reader);
|
||||||
|
stdin_pipe_reader = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::istream* Coprocess::stdout_pipe ()
|
||||||
|
{
|
||||||
|
if (!stdout_pipe_istream) {
|
||||||
|
int fds[2];
|
||||||
|
if (pipe(fds) == -1) {
|
||||||
|
throw System_error("pipe", "", errno);
|
||||||
|
}
|
||||||
|
stdout_pipe_reader = fds[0];
|
||||||
|
stdout_pipe_writer = fds[1];
|
||||||
|
stdout_pipe_istream = new ifhstream(this, read_stdout);
|
||||||
|
}
|
||||||
|
return stdout_pipe_istream;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Coprocess::close_stdout ()
|
||||||
|
{
|
||||||
|
delete stdout_pipe_istream;
|
||||||
|
stdout_pipe_istream = NULL;
|
||||||
|
if (stdout_pipe_writer != -1) {
|
||||||
|
close(stdout_pipe_writer);
|
||||||
|
stdout_pipe_writer = -1;
|
||||||
|
}
|
||||||
|
if (stdout_pipe_reader != -1) {
|
||||||
|
close(stdout_pipe_reader);
|
||||||
|
stdout_pipe_reader = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Coprocess::spawn (const std::vector<std::string>& args)
|
||||||
|
{
|
||||||
|
pid = fork();
|
||||||
|
if (pid == -1) {
|
||||||
|
throw System_error("fork", "", errno);
|
||||||
|
}
|
||||||
|
if (pid == 0) {
|
||||||
|
if (stdin_pipe_writer != -1) {
|
||||||
|
close(stdin_pipe_writer);
|
||||||
|
}
|
||||||
|
if (stdout_pipe_reader != -1) {
|
||||||
|
close(stdout_pipe_reader);
|
||||||
|
}
|
||||||
|
if (stdin_pipe_reader != -1) {
|
||||||
|
dup2(stdin_pipe_reader, 0);
|
||||||
|
close(stdin_pipe_reader);
|
||||||
|
}
|
||||||
|
if (stdout_pipe_writer != -1) {
|
||||||
|
dup2(stdout_pipe_writer, 1);
|
||||||
|
close(stdout_pipe_writer);
|
||||||
|
}
|
||||||
|
|
||||||
|
execvp(args[0], args);
|
||||||
|
perror(args[0].c_str());
|
||||||
|
_exit(-1);
|
||||||
|
}
|
||||||
|
if (stdin_pipe_reader != -1) {
|
||||||
|
close(stdin_pipe_reader);
|
||||||
|
stdin_pipe_reader = -1;
|
||||||
|
}
|
||||||
|
if (stdout_pipe_writer != -1) {
|
||||||
|
close(stdout_pipe_writer);
|
||||||
|
stdout_pipe_writer = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int Coprocess::wait ()
|
||||||
|
{
|
||||||
|
int status = 0;
|
||||||
|
if (waitpid(pid, &status, 0) == -1) {
|
||||||
|
throw System_error("waitpid", "", errno);
|
||||||
|
}
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t Coprocess::write_stdin (void* handle, const void* buf, size_t count)
|
||||||
|
{
|
||||||
|
const int fd = static_cast<Coprocess*>(handle)->stdin_pipe_writer;
|
||||||
|
ssize_t ret;
|
||||||
|
while ((ret = write(fd, buf, count)) == -1 && errno == EINTR); // restart if interrupted
|
||||||
|
if (ret < 0) {
|
||||||
|
throw System_error("write", "", errno);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t Coprocess::read_stdout (void* handle, void* buf, size_t count)
|
||||||
|
{
|
||||||
|
const int fd = static_cast<Coprocess*>(handle)->stdout_pipe_reader;
|
||||||
|
ssize_t ret;
|
||||||
|
while ((ret = read(fd, buf, count)) == -1 && errno == EINTR); // restart if interrupted
|
||||||
|
if (ret < 0) {
|
||||||
|
throw System_error("read", "", errno);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
68
coprocess-unix.hpp
Normal file
68
coprocess-unix.hpp
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 Andrew Ayer
|
||||||
|
*
|
||||||
|
* This file is part of git-crypt.
|
||||||
|
*
|
||||||
|
* git-crypt is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* git-crypt is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with git-crypt. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
* Additional permission under GNU GPL version 3 section 7:
|
||||||
|
*
|
||||||
|
* If you modify the Program, or any covered work, by linking or
|
||||||
|
* combining it with the OpenSSL project's OpenSSL library (or a
|
||||||
|
* modified version of that library), containing parts covered by the
|
||||||
|
* terms of the OpenSSL or SSLeay licenses, the licensors of the Program
|
||||||
|
* grant you additional permission to convey the resulting work.
|
||||||
|
* Corresponding Source for a non-source form of such a combination
|
||||||
|
* shall include the source code for the parts of OpenSSL used as well
|
||||||
|
* as that of the covered work.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GIT_CRYPT_COPROCESS_HPP
|
||||||
|
#define GIT_CRYPT_COPROCESS_HPP
|
||||||
|
|
||||||
|
#include "fhstream.hpp"
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
class Coprocess {
|
||||||
|
pid_t pid;
|
||||||
|
|
||||||
|
int stdin_pipe_reader;
|
||||||
|
int stdin_pipe_writer;
|
||||||
|
ofhstream* stdin_pipe_ostream;
|
||||||
|
static size_t write_stdin (void*, const void*, size_t);
|
||||||
|
|
||||||
|
int stdout_pipe_reader;
|
||||||
|
int stdout_pipe_writer;
|
||||||
|
ifhstream* stdout_pipe_istream;
|
||||||
|
static size_t read_stdout (void*, void*, size_t);
|
||||||
|
|
||||||
|
Coprocess (const Coprocess&); // Disallow copy
|
||||||
|
Coprocess& operator= (const Coprocess&); // Disallow assignment
|
||||||
|
public:
|
||||||
|
Coprocess ();
|
||||||
|
~Coprocess ();
|
||||||
|
|
||||||
|
std::ostream* stdin_pipe ();
|
||||||
|
void close_stdin ();
|
||||||
|
|
||||||
|
std::istream* stdout_pipe ();
|
||||||
|
void close_stdout ();
|
||||||
|
|
||||||
|
void spawn (const std::vector<std::string>&);
|
||||||
|
|
||||||
|
int wait ();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
269
coprocess-win32.cpp
Normal file
269
coprocess-win32.cpp
Normal file
@@ -0,0 +1,269 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 Andrew Ayer
|
||||||
|
*
|
||||||
|
* This file is part of git-crypt.
|
||||||
|
*
|
||||||
|
* git-crypt is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* git-crypt is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with git-crypt. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
* Additional permission under GNU GPL version 3 section 7:
|
||||||
|
*
|
||||||
|
* If you modify the Program, or any covered work, by linking or
|
||||||
|
* combining it with the OpenSSL project's OpenSSL library (or a
|
||||||
|
* modified version of that library), containing parts covered by the
|
||||||
|
* terms of the OpenSSL or SSLeay licenses, the licensors of the Program
|
||||||
|
* grant you additional permission to convey the resulting work.
|
||||||
|
* Corresponding Source for a non-source form of such a combination
|
||||||
|
* shall include the source code for the parts of OpenSSL used as well
|
||||||
|
* as that of the covered work.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "coprocess-win32.hpp"
|
||||||
|
#include "util.hpp"
|
||||||
|
|
||||||
|
|
||||||
|
static void escape_cmdline_argument (std::string& cmdline, const std::string& arg)
|
||||||
|
{
|
||||||
|
// For an explanation of Win32's arcane argument quoting rules, see:
|
||||||
|
// http://msdn.microsoft.com/en-us/library/17w5ykft%28v=vs.85%29.aspx
|
||||||
|
// http://msdn.microsoft.com/en-us/library/bb776391%28v=vs.85%29.aspx
|
||||||
|
// http://blogs.msdn.com/b/twistylittlepassagesallalike/archive/2011/04/23/everyone-quotes-arguments-the-wrong-way.aspx
|
||||||
|
// http://blogs.msdn.com/b/oldnewthing/archive/2010/09/17/10063629.aspx
|
||||||
|
cmdline.push_back('"');
|
||||||
|
|
||||||
|
std::string::const_iterator p(arg.begin());
|
||||||
|
while (p != arg.end()) {
|
||||||
|
if (*p == '"') {
|
||||||
|
cmdline.push_back('\\');
|
||||||
|
cmdline.push_back('"');
|
||||||
|
++p;
|
||||||
|
} else if (*p == '\\') {
|
||||||
|
unsigned int num_backslashes = 0;
|
||||||
|
while (p != arg.end() && *p == '\\') {
|
||||||
|
++num_backslashes;
|
||||||
|
++p;
|
||||||
|
}
|
||||||
|
if (p == arg.end() || *p == '"') {
|
||||||
|
// Backslashes need to be escaped
|
||||||
|
num_backslashes *= 2;
|
||||||
|
}
|
||||||
|
while (num_backslashes--) {
|
||||||
|
cmdline.push_back('\\');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
cmdline.push_back(*p++);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmdline.push_back('"');
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string format_cmdline (const std::vector<std::string>& command)
|
||||||
|
{
|
||||||
|
std::string cmdline;
|
||||||
|
for (std::vector<std::string>::const_iterator arg(command.begin()); arg != command.end(); ++arg) {
|
||||||
|
if (arg != command.begin()) {
|
||||||
|
cmdline.push_back(' ');
|
||||||
|
}
|
||||||
|
escape_cmdline_argument(cmdline, *arg);
|
||||||
|
}
|
||||||
|
return cmdline;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HANDLE spawn_command (const std::vector<std::string>& command, HANDLE stdin_handle, HANDLE stdout_handle, HANDLE stderr_handle)
|
||||||
|
{
|
||||||
|
PROCESS_INFORMATION proc_info;
|
||||||
|
ZeroMemory(&proc_info, sizeof(proc_info));
|
||||||
|
|
||||||
|
STARTUPINFO start_info;
|
||||||
|
ZeroMemory(&start_info, sizeof(start_info));
|
||||||
|
|
||||||
|
start_info.cb = sizeof(STARTUPINFO);
|
||||||
|
start_info.hStdInput = stdin_handle ? stdin_handle : GetStdHandle(STD_INPUT_HANDLE);
|
||||||
|
start_info.hStdOutput = stdout_handle ? stdout_handle : GetStdHandle(STD_OUTPUT_HANDLE);
|
||||||
|
start_info.hStdError = stderr_handle ? stderr_handle : GetStdHandle(STD_ERROR_HANDLE);
|
||||||
|
start_info.dwFlags |= STARTF_USESTDHANDLES;
|
||||||
|
|
||||||
|
std::string cmdline(format_cmdline(command));
|
||||||
|
|
||||||
|
if (!CreateProcessA(NULL, // application name (NULL to use command line)
|
||||||
|
const_cast<char*>(cmdline.c_str()),
|
||||||
|
NULL, // process security attributes
|
||||||
|
NULL, // primary thread security attributes
|
||||||
|
TRUE, // handles are inherited
|
||||||
|
0, // creation flags
|
||||||
|
NULL, // use parent's environment
|
||||||
|
NULL, // use parent's current directory
|
||||||
|
&start_info,
|
||||||
|
&proc_info)) {
|
||||||
|
throw System_error("CreateProcess", cmdline, GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
CloseHandle(proc_info.hThread);
|
||||||
|
|
||||||
|
return proc_info.hProcess;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Coprocess::Coprocess ()
|
||||||
|
{
|
||||||
|
proc_handle = NULL;
|
||||||
|
stdin_pipe_reader = NULL;
|
||||||
|
stdin_pipe_writer = NULL;
|
||||||
|
stdin_pipe_ostream = NULL;
|
||||||
|
stdout_pipe_reader = NULL;
|
||||||
|
stdout_pipe_writer = NULL;
|
||||||
|
stdout_pipe_istream = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
Coprocess::~Coprocess ()
|
||||||
|
{
|
||||||
|
close_stdin();
|
||||||
|
close_stdout();
|
||||||
|
if (proc_handle) {
|
||||||
|
CloseHandle(proc_handle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostream* Coprocess::stdin_pipe ()
|
||||||
|
{
|
||||||
|
if (!stdin_pipe_ostream) {
|
||||||
|
SECURITY_ATTRIBUTES sec_attr;
|
||||||
|
|
||||||
|
// Set the bInheritHandle flag so pipe handles are inherited.
|
||||||
|
sec_attr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
||||||
|
sec_attr.bInheritHandle = TRUE;
|
||||||
|
sec_attr.lpSecurityDescriptor = NULL;
|
||||||
|
|
||||||
|
// Create a pipe for the child process's STDIN.
|
||||||
|
if (!CreatePipe(&stdin_pipe_reader, &stdin_pipe_writer, &sec_attr, 0)) {
|
||||||
|
throw System_error("CreatePipe", "", GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure the write handle to the pipe for STDIN is not inherited.
|
||||||
|
if (!SetHandleInformation(stdin_pipe_writer, HANDLE_FLAG_INHERIT, 0)) {
|
||||||
|
throw System_error("SetHandleInformation", "", GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
stdin_pipe_ostream = new ofhstream(this, write_stdin);
|
||||||
|
}
|
||||||
|
return stdin_pipe_ostream;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Coprocess::close_stdin ()
|
||||||
|
{
|
||||||
|
delete stdin_pipe_ostream;
|
||||||
|
stdin_pipe_ostream = NULL;
|
||||||
|
if (stdin_pipe_writer) {
|
||||||
|
CloseHandle(stdin_pipe_writer);
|
||||||
|
stdin_pipe_writer = NULL;
|
||||||
|
}
|
||||||
|
if (stdin_pipe_reader) {
|
||||||
|
CloseHandle(stdin_pipe_reader);
|
||||||
|
stdin_pipe_reader = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::istream* Coprocess::stdout_pipe ()
|
||||||
|
{
|
||||||
|
if (!stdout_pipe_istream) {
|
||||||
|
SECURITY_ATTRIBUTES sec_attr;
|
||||||
|
|
||||||
|
// Set the bInheritHandle flag so pipe handles are inherited.
|
||||||
|
sec_attr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
||||||
|
sec_attr.bInheritHandle = TRUE;
|
||||||
|
sec_attr.lpSecurityDescriptor = NULL;
|
||||||
|
|
||||||
|
// Create a pipe for the child process's STDOUT.
|
||||||
|
if (!CreatePipe(&stdout_pipe_reader, &stdout_pipe_writer, &sec_attr, 0)) {
|
||||||
|
throw System_error("CreatePipe", "", GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure the read handle to the pipe for STDOUT is not inherited.
|
||||||
|
if (!SetHandleInformation(stdout_pipe_reader, HANDLE_FLAG_INHERIT, 0)) {
|
||||||
|
throw System_error("SetHandleInformation", "", GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
stdout_pipe_istream = new ifhstream(this, read_stdout);
|
||||||
|
}
|
||||||
|
return stdout_pipe_istream;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Coprocess::close_stdout ()
|
||||||
|
{
|
||||||
|
delete stdout_pipe_istream;
|
||||||
|
stdout_pipe_istream = NULL;
|
||||||
|
if (stdout_pipe_writer) {
|
||||||
|
CloseHandle(stdout_pipe_writer);
|
||||||
|
stdout_pipe_writer = NULL;
|
||||||
|
}
|
||||||
|
if (stdout_pipe_reader) {
|
||||||
|
CloseHandle(stdout_pipe_reader);
|
||||||
|
stdout_pipe_reader = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Coprocess::spawn (const std::vector<std::string>& args)
|
||||||
|
{
|
||||||
|
proc_handle = spawn_command(args, stdin_pipe_reader, stdout_pipe_writer, NULL);
|
||||||
|
if (stdin_pipe_reader) {
|
||||||
|
CloseHandle(stdin_pipe_reader);
|
||||||
|
stdin_pipe_reader = NULL;
|
||||||
|
}
|
||||||
|
if (stdout_pipe_writer) {
|
||||||
|
CloseHandle(stdout_pipe_writer);
|
||||||
|
stdout_pipe_writer = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int Coprocess::wait ()
|
||||||
|
{
|
||||||
|
if (WaitForSingleObject(proc_handle, INFINITE) == WAIT_FAILED) {
|
||||||
|
throw System_error("WaitForSingleObject", "", GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
DWORD exit_code;
|
||||||
|
if (!GetExitCodeProcess(proc_handle, &exit_code)) {
|
||||||
|
throw System_error("GetExitCodeProcess", "", GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
return exit_code;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t Coprocess::write_stdin (void* handle, const void* buf, size_t count)
|
||||||
|
{
|
||||||
|
DWORD bytes_written;
|
||||||
|
if (!WriteFile(static_cast<Coprocess*>(handle)->stdin_pipe_writer, buf, count, &bytes_written, NULL)) {
|
||||||
|
throw System_error("WriteFile", "", GetLastError());
|
||||||
|
}
|
||||||
|
return bytes_written;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t Coprocess::read_stdout (void* handle, void* buf, size_t count)
|
||||||
|
{
|
||||||
|
// Note that ReadFile on a pipe may return with bytes_read==0 if the other
|
||||||
|
// end of the pipe writes zero bytes, so retry when this happens.
|
||||||
|
// When the other end of the pipe actually closes, ReadFile
|
||||||
|
// fails with ERROR_BROKEN_PIPE.
|
||||||
|
DWORD bytes_read;
|
||||||
|
do {
|
||||||
|
if (!ReadFile(static_cast<Coprocess*>(handle)->stdout_pipe_reader, buf, count, &bytes_read, NULL)) {
|
||||||
|
const DWORD read_error = GetLastError();
|
||||||
|
if (read_error != ERROR_BROKEN_PIPE) {
|
||||||
|
throw System_error("ReadFile", "", read_error);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
} while (bytes_read == 0);
|
||||||
|
return bytes_read;
|
||||||
|
}
|
||||||
68
coprocess-win32.hpp
Normal file
68
coprocess-win32.hpp
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 Andrew Ayer
|
||||||
|
*
|
||||||
|
* This file is part of git-crypt.
|
||||||
|
*
|
||||||
|
* git-crypt is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* git-crypt is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with git-crypt. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
* Additional permission under GNU GPL version 3 section 7:
|
||||||
|
*
|
||||||
|
* If you modify the Program, or any covered work, by linking or
|
||||||
|
* combining it with the OpenSSL project's OpenSSL library (or a
|
||||||
|
* modified version of that library), containing parts covered by the
|
||||||
|
* terms of the OpenSSL or SSLeay licenses, the licensors of the Program
|
||||||
|
* grant you additional permission to convey the resulting work.
|
||||||
|
* Corresponding Source for a non-source form of such a combination
|
||||||
|
* shall include the source code for the parts of OpenSSL used as well
|
||||||
|
* as that of the covered work.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GIT_CRYPT_COPROCESS_HPP
|
||||||
|
#define GIT_CRYPT_COPROCESS_HPP
|
||||||
|
|
||||||
|
#include "fhstream.hpp"
|
||||||
|
#include <windows.h>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
class Coprocess {
|
||||||
|
HANDLE proc_handle;
|
||||||
|
|
||||||
|
HANDLE stdin_pipe_reader;
|
||||||
|
HANDLE stdin_pipe_writer;
|
||||||
|
ofhstream* stdin_pipe_ostream;
|
||||||
|
static size_t write_stdin (void*, const void*, size_t);
|
||||||
|
|
||||||
|
HANDLE stdout_pipe_reader;
|
||||||
|
HANDLE stdout_pipe_writer;
|
||||||
|
ifhstream* stdout_pipe_istream;
|
||||||
|
static size_t read_stdout (void*, void*, size_t);
|
||||||
|
|
||||||
|
Coprocess (const Coprocess&); // Disallow copy
|
||||||
|
Coprocess& operator= (const Coprocess&); // Disallow assignment
|
||||||
|
public:
|
||||||
|
Coprocess ();
|
||||||
|
~Coprocess ();
|
||||||
|
|
||||||
|
std::ostream* stdin_pipe ();
|
||||||
|
void close_stdin ();
|
||||||
|
|
||||||
|
std::istream* stdout_pipe ();
|
||||||
|
void close_stdout ();
|
||||||
|
|
||||||
|
void spawn (const std::vector<std::string>&);
|
||||||
|
|
||||||
|
int wait ();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
5
coprocess.cpp
Normal file
5
coprocess.cpp
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
#ifdef _WIN32
|
||||||
|
#include "coprocess-win32.cpp"
|
||||||
|
#else
|
||||||
|
#include "coprocess-unix.cpp"
|
||||||
|
#endif
|
||||||
5
coprocess.hpp
Normal file
5
coprocess.hpp
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
#ifdef _WIN32
|
||||||
|
#include "coprocess-win32.hpp"
|
||||||
|
#else
|
||||||
|
#include "coprocess-unix.hpp"
|
||||||
|
#endif
|
||||||
227
fhstream.cpp
Normal file
227
fhstream.cpp
Normal file
@@ -0,0 +1,227 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2012, 2015 Andrew Ayer
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||||||
|
* copy of this software and associated documentation files (the "Software"),
|
||||||
|
* to deal in the Software without restriction, including without limitation
|
||||||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||||||
|
* Software is furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included
|
||||||
|
* in all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||||
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
||||||
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||||||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||||||
|
* OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*
|
||||||
|
* Except as contained in this notice, the name(s) of the above copyright
|
||||||
|
* holders shall not be used in advertising or otherwise to promote the
|
||||||
|
* sale, use or other dealings in this Software without prior written
|
||||||
|
* authorization.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <cstring>
|
||||||
|
#include <algorithm> // for std::min
|
||||||
|
|
||||||
|
#include "fhstream.hpp"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ofhstream
|
||||||
|
*/
|
||||||
|
|
||||||
|
ofhbuf::ofhbuf (void* arg_handle, size_t (*arg_write_fun)(void*, const void*, size_t))
|
||||||
|
: handle(arg_handle),
|
||||||
|
write_fun(arg_write_fun),
|
||||||
|
buffer(new char[default_buffer_size]),
|
||||||
|
buffer_size(default_buffer_size)
|
||||||
|
{
|
||||||
|
reset_buffer();
|
||||||
|
}
|
||||||
|
|
||||||
|
ofhbuf::~ofhbuf ()
|
||||||
|
{
|
||||||
|
if (handle) {
|
||||||
|
try {
|
||||||
|
sync();
|
||||||
|
} catch (...) {
|
||||||
|
// Ignore exception since we're in the destructor.
|
||||||
|
// To catch write errors, call sync() explicitly.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete[] buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
ofhbuf::int_type ofhbuf::overflow (ofhbuf::int_type c)
|
||||||
|
{
|
||||||
|
const char* p = pbase();
|
||||||
|
std::streamsize bytes_to_write = pptr() - p;
|
||||||
|
|
||||||
|
if (!is_eof(c)) {
|
||||||
|
*pptr() = c;
|
||||||
|
++bytes_to_write;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (bytes_to_write > 0) {
|
||||||
|
const size_t bytes_written = write_fun(handle, p, bytes_to_write);
|
||||||
|
bytes_to_write -= bytes_written;
|
||||||
|
p += bytes_written;
|
||||||
|
}
|
||||||
|
|
||||||
|
reset_buffer();
|
||||||
|
|
||||||
|
return traits_type::to_int_type(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int ofhbuf::sync ()
|
||||||
|
{
|
||||||
|
return !is_eof(overflow(traits_type::eof())) ? 0 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::streamsize ofhbuf::xsputn (const char* s, std::streamsize n)
|
||||||
|
{
|
||||||
|
// Use heuristic to decide whether to write directly or just use buffer
|
||||||
|
// Write directly only if n >= MIN(4096, available buffer capacity)
|
||||||
|
// (this is similar to what basic_filebuf does)
|
||||||
|
|
||||||
|
if (n < std::min<std::streamsize>(4096, epptr() - pptr())) {
|
||||||
|
// Not worth it to do a direct write
|
||||||
|
return std::streambuf::xsputn(s, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Before we can do a direct write of this string, we need to flush
|
||||||
|
// out the current contents of the buffer.
|
||||||
|
if (pbase() != pptr()) {
|
||||||
|
overflow(traits_type::eof()); // throws an exception or it succeeds
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now we can go ahead and write out the string.
|
||||||
|
size_t bytes_to_write = n;
|
||||||
|
|
||||||
|
while (bytes_to_write > 0) {
|
||||||
|
const size_t bytes_written = write_fun(handle, s, bytes_to_write);
|
||||||
|
bytes_to_write -= bytes_written;
|
||||||
|
s += bytes_written;
|
||||||
|
}
|
||||||
|
|
||||||
|
return n; // Return the total bytes written
|
||||||
|
}
|
||||||
|
|
||||||
|
std::streambuf* ofhbuf::setbuf (char* s, std::streamsize n)
|
||||||
|
{
|
||||||
|
if (s == 0 && n == 0) {
|
||||||
|
// Switch to unbuffered
|
||||||
|
// This won't take effect until the next overflow or sync
|
||||||
|
// (We defer it taking effect so that write errors can be properly reported)
|
||||||
|
// To cause it to take effect as soon as possible, we artificially reduce the
|
||||||
|
// size of the buffer so it has no space left. This will trigger an overflow
|
||||||
|
// on the next put.
|
||||||
|
std::streambuf::setp(pbase(), pptr());
|
||||||
|
std::streambuf::pbump(pptr() - pbase());
|
||||||
|
buffer_size = 1;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ifhstream
|
||||||
|
*/
|
||||||
|
|
||||||
|
ifhbuf::ifhbuf (void* arg_handle, size_t (*arg_read_fun)(void*, void*, size_t))
|
||||||
|
: handle(arg_handle),
|
||||||
|
read_fun(arg_read_fun),
|
||||||
|
buffer(new char[default_buffer_size + putback_size]),
|
||||||
|
buffer_size(default_buffer_size)
|
||||||
|
{
|
||||||
|
reset_buffer(0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
ifhbuf::~ifhbuf ()
|
||||||
|
{
|
||||||
|
delete[] buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
ifhbuf::int_type ifhbuf::underflow ()
|
||||||
|
{
|
||||||
|
if (gptr() >= egptr()) { // A true underflow (no bytes in buffer left to read)
|
||||||
|
|
||||||
|
// Move the putback_size most-recently-read characters into the putback area
|
||||||
|
size_t nputback = std::min<size_t>(gptr() - eback(), putback_size);
|
||||||
|
std::memmove(buffer + (putback_size - nputback), gptr() - nputback, nputback);
|
||||||
|
|
||||||
|
// Now read new characters from the file descriptor
|
||||||
|
const size_t nread = read_fun(handle, buffer + putback_size, buffer_size);
|
||||||
|
if (nread == 0) {
|
||||||
|
// EOF
|
||||||
|
return traits_type::eof();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset the buffer
|
||||||
|
reset_buffer(nputback, nread);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the next character
|
||||||
|
return traits_type::to_int_type(*gptr());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::streamsize ifhbuf::xsgetn (char* s, std::streamsize n)
|
||||||
|
{
|
||||||
|
// Use heuristic to decide whether to read directly
|
||||||
|
// Read directly only if n >= bytes_available + 4096
|
||||||
|
|
||||||
|
std::streamsize bytes_available = egptr() - gptr();
|
||||||
|
|
||||||
|
if (n < bytes_available + 4096) {
|
||||||
|
// Not worth it to do a direct read
|
||||||
|
return std::streambuf::xsgetn(s, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::streamsize total_bytes_read = 0;
|
||||||
|
|
||||||
|
// First, copy out the bytes currently in the buffer
|
||||||
|
std::memcpy(s, gptr(), bytes_available);
|
||||||
|
|
||||||
|
s += bytes_available;
|
||||||
|
n -= bytes_available;
|
||||||
|
total_bytes_read += bytes_available;
|
||||||
|
|
||||||
|
// Now do the direct read
|
||||||
|
while (n > 0) {
|
||||||
|
const size_t bytes_read = read_fun(handle, s, n);
|
||||||
|
if (bytes_read == 0) {
|
||||||
|
// EOF
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
s += bytes_read;
|
||||||
|
n -= bytes_read;
|
||||||
|
total_bytes_read += bytes_read;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fill up the putback area with the most recently read characters
|
||||||
|
size_t nputback = std::min<size_t>(total_bytes_read, putback_size);
|
||||||
|
std::memcpy(buffer + (putback_size - nputback), s - nputback, nputback);
|
||||||
|
|
||||||
|
// Reset the buffer with no bytes available for reading, but with some putback characters
|
||||||
|
reset_buffer(nputback, 0);
|
||||||
|
|
||||||
|
// Return the total number of bytes read
|
||||||
|
return total_bytes_read;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::streambuf* ifhbuf::setbuf (char* s, std::streamsize n)
|
||||||
|
{
|
||||||
|
if (s == 0 && n == 0) {
|
||||||
|
// Switch to unbuffered
|
||||||
|
// This won't take effect until the next underflow (we don't want to
|
||||||
|
// lose what's currently in the buffer!)
|
||||||
|
buffer_size = 1;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
134
fhstream.hpp
Normal file
134
fhstream.hpp
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2012, 2015 Andrew Ayer
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||||||
|
* copy of this software and associated documentation files (the "Software"),
|
||||||
|
* to deal in the Software without restriction, including without limitation
|
||||||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||||||
|
* Software is furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included
|
||||||
|
* in all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||||
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
||||||
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||||||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||||||
|
* OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*
|
||||||
|
* Except as contained in this notice, the name(s) of the above copyright
|
||||||
|
* holders shall not be used in advertising or otherwise to promote the
|
||||||
|
* sale, use or other dealings in this Software without prior written
|
||||||
|
* authorization.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GIT_CRYPT_FHSTREAM_HPP
|
||||||
|
#define GIT_CRYPT_FHSTREAM_HPP
|
||||||
|
|
||||||
|
#include <ostream>
|
||||||
|
#include <istream>
|
||||||
|
#include <streambuf>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ofhstream
|
||||||
|
*/
|
||||||
|
class ofhbuf : public std::streambuf {
|
||||||
|
enum { default_buffer_size = 8192 };
|
||||||
|
|
||||||
|
void* handle;
|
||||||
|
size_t (*write_fun)(void*, const void*, size_t);
|
||||||
|
char* buffer;
|
||||||
|
size_t buffer_size;
|
||||||
|
|
||||||
|
inline void reset_buffer ()
|
||||||
|
{
|
||||||
|
std::streambuf::setp(buffer, buffer + buffer_size - 1);
|
||||||
|
}
|
||||||
|
static inline bool is_eof (int_type ch) { return traits_type::eq_int_type(ch, traits_type::eof()); }
|
||||||
|
|
||||||
|
// Disallow copy
|
||||||
|
#if __cplusplus >= 201103L /* C++11 */
|
||||||
|
ofhbuf (const ofhbuf&) = delete;
|
||||||
|
ofhbuf& operator= (const ofhbuf&) = delete;
|
||||||
|
#else
|
||||||
|
ofhbuf (const ofhbuf&);
|
||||||
|
ofhbuf& operator= (const ofhbuf&);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual int_type overflow (int_type ch =traits_type::eof());
|
||||||
|
virtual int sync ();
|
||||||
|
virtual std::streamsize xsputn (const char*, std::streamsize);
|
||||||
|
virtual std::streambuf* setbuf (char*, std::streamsize);
|
||||||
|
|
||||||
|
public:
|
||||||
|
ofhbuf (void*, size_t (*)(void*, const void*, size_t));
|
||||||
|
~ofhbuf (); // WARNING: calls sync() and ignores exceptions
|
||||||
|
};
|
||||||
|
|
||||||
|
class ofhstream : public std::ostream {
|
||||||
|
mutable ofhbuf buf;
|
||||||
|
public:
|
||||||
|
ofhstream (void* handle, size_t (*write_fun)(void*, const void*, size_t))
|
||||||
|
: std::ostream(0), buf(handle, write_fun)
|
||||||
|
{
|
||||||
|
std::ostream::rdbuf(&buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
ofhbuf* rdbuf () const { return &buf; }
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ifhstream
|
||||||
|
*/
|
||||||
|
class ifhbuf : public std::streambuf {
|
||||||
|
enum {
|
||||||
|
default_buffer_size = 8192,
|
||||||
|
putback_size = 4
|
||||||
|
};
|
||||||
|
|
||||||
|
void* handle;
|
||||||
|
size_t (*read_fun)(void*, void*, size_t);
|
||||||
|
char* buffer;
|
||||||
|
size_t buffer_size;
|
||||||
|
|
||||||
|
inline void reset_buffer (size_t nputback, size_t nread)
|
||||||
|
{
|
||||||
|
std::streambuf::setg(buffer + (putback_size - nputback), buffer + putback_size, buffer + putback_size + nread);
|
||||||
|
}
|
||||||
|
// Disallow copy
|
||||||
|
#if __cplusplus >= 201103L /* C++11 */
|
||||||
|
ifhbuf (const ifhbuf&) = delete;
|
||||||
|
ifhbuf& operator= (const ifhbuf&) = delete;
|
||||||
|
#else
|
||||||
|
ifhbuf (const ifhbuf&);
|
||||||
|
ifhbuf& operator= (const ifhbuf&);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual int_type underflow ();
|
||||||
|
virtual std::streamsize xsgetn (char*, std::streamsize);
|
||||||
|
virtual std::streambuf* setbuf (char*, std::streamsize);
|
||||||
|
|
||||||
|
public:
|
||||||
|
ifhbuf (void*, size_t (*)(void*, void*, size_t));
|
||||||
|
~ifhbuf (); // Can't fail
|
||||||
|
};
|
||||||
|
|
||||||
|
class ifhstream : public std::istream {
|
||||||
|
mutable ifhbuf buf;
|
||||||
|
public:
|
||||||
|
explicit ifhstream (void* handle, size_t (*read_fun)(void*, void*, size_t))
|
||||||
|
: std::istream(0), buf(handle, read_fun)
|
||||||
|
{
|
||||||
|
std::istream::rdbuf(&buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
ifhbuf* rdbuf () const { return &buf; }
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
113
util-unix.cpp
113
util-unix.cpp
@@ -162,119 +162,6 @@ std::string our_exe_path ()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int execvp (const std::string& file, const std::vector<std::string>& args)
|
|
||||||
{
|
|
||||||
std::vector<const char*> args_c_str;
|
|
||||||
args_c_str.reserve(args.size());
|
|
||||||
for (std::vector<std::string>::const_iterator arg(args.begin()); arg != args.end(); ++arg) {
|
|
||||||
args_c_str.push_back(arg->c_str());
|
|
||||||
}
|
|
||||||
args_c_str.push_back(NULL);
|
|
||||||
return execvp(file.c_str(), const_cast<char**>(&args_c_str[0]));
|
|
||||||
}
|
|
||||||
|
|
||||||
int exec_command (const std::vector<std::string>& command)
|
|
||||||
{
|
|
||||||
pid_t child = fork();
|
|
||||||
if (child == -1) {
|
|
||||||
throw System_error("fork", "", errno);
|
|
||||||
}
|
|
||||||
if (child == 0) {
|
|
||||||
execvp(command[0], command);
|
|
||||||
perror(command[0].c_str());
|
|
||||||
_exit(-1);
|
|
||||||
}
|
|
||||||
int status = 0;
|
|
||||||
if (waitpid(child, &status, 0) == -1) {
|
|
||||||
throw System_error("waitpid", "", errno);
|
|
||||||
}
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exec_command (const std::vector<std::string>& command, std::ostream& output)
|
|
||||||
{
|
|
||||||
int pipefd[2];
|
|
||||||
if (pipe(pipefd) == -1) {
|
|
||||||
throw System_error("pipe", "", errno);
|
|
||||||
}
|
|
||||||
pid_t child = fork();
|
|
||||||
if (child == -1) {
|
|
||||||
int fork_errno = errno;
|
|
||||||
close(pipefd[0]);
|
|
||||||
close(pipefd[1]);
|
|
||||||
throw System_error("fork", "", fork_errno);
|
|
||||||
}
|
|
||||||
if (child == 0) {
|
|
||||||
close(pipefd[0]);
|
|
||||||
if (pipefd[1] != 1) {
|
|
||||||
dup2(pipefd[1], 1);
|
|
||||||
close(pipefd[1]);
|
|
||||||
}
|
|
||||||
execvp(command[0], command);
|
|
||||||
perror(command[0].c_str());
|
|
||||||
_exit(-1);
|
|
||||||
}
|
|
||||||
close(pipefd[1]);
|
|
||||||
char buffer[1024];
|
|
||||||
ssize_t bytes_read;
|
|
||||||
while ((bytes_read = read(pipefd[0], buffer, sizeof(buffer))) > 0) {
|
|
||||||
output.write(buffer, bytes_read);
|
|
||||||
}
|
|
||||||
if (bytes_read == -1) {
|
|
||||||
int read_errno = errno;
|
|
||||||
close(pipefd[0]);
|
|
||||||
throw System_error("read", "", read_errno);
|
|
||||||
}
|
|
||||||
close(pipefd[0]);
|
|
||||||
int status = 0;
|
|
||||||
if (waitpid(child, &status, 0) == -1) {
|
|
||||||
throw System_error("waitpid", "", errno);
|
|
||||||
}
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exec_command_with_input (const std::vector<std::string>& command, const char* p, size_t len)
|
|
||||||
{
|
|
||||||
int pipefd[2];
|
|
||||||
if (pipe(pipefd) == -1) {
|
|
||||||
throw System_error("pipe", "", errno);
|
|
||||||
}
|
|
||||||
pid_t child = fork();
|
|
||||||
if (child == -1) {
|
|
||||||
int fork_errno = errno;
|
|
||||||
close(pipefd[0]);
|
|
||||||
close(pipefd[1]);
|
|
||||||
throw System_error("fork", "", fork_errno);
|
|
||||||
}
|
|
||||||
if (child == 0) {
|
|
||||||
close(pipefd[1]);
|
|
||||||
if (pipefd[0] != 0) {
|
|
||||||
dup2(pipefd[0], 0);
|
|
||||||
close(pipefd[0]);
|
|
||||||
}
|
|
||||||
execvp(command[0], command);
|
|
||||||
perror(command[0].c_str());
|
|
||||||
_exit(-1);
|
|
||||||
}
|
|
||||||
close(pipefd[0]);
|
|
||||||
while (len > 0) {
|
|
||||||
ssize_t bytes_written = write(pipefd[1], p, len);
|
|
||||||
if (bytes_written == -1) {
|
|
||||||
int write_errno = errno;
|
|
||||||
close(pipefd[1]);
|
|
||||||
throw System_error("write", "", write_errno);
|
|
||||||
}
|
|
||||||
p += bytes_written;
|
|
||||||
len -= bytes_written;
|
|
||||||
}
|
|
||||||
close(pipefd[1]);
|
|
||||||
int status = 0;
|
|
||||||
if (waitpid(child, &status, 0) == -1) {
|
|
||||||
throw System_error("waitpid", "", errno);
|
|
||||||
}
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exit_status (int wait_status)
|
int exit_status (int wait_status)
|
||||||
{
|
{
|
||||||
return wait_status != -1 && WIFEXITED(wait_status) ? WEXITSTATUS(wait_status) : -1;
|
return wait_status != -1 && WIFEXITED(wait_status) ? WEXITSTATUS(wait_status) : -1;
|
||||||
|
|||||||
191
util-win32.cpp
191
util-win32.cpp
@@ -125,197 +125,6 @@ std::string our_exe_path ()
|
|||||||
return std::string(buffer.begin(), buffer.begin() + len);
|
return std::string(buffer.begin(), buffer.begin() + len);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void escape_cmdline_argument (std::string& cmdline, const std::string& arg)
|
|
||||||
{
|
|
||||||
// For an explanation of Win32's arcane argument quoting rules, see:
|
|
||||||
// http://msdn.microsoft.com/en-us/library/17w5ykft%28v=vs.85%29.aspx
|
|
||||||
// http://msdn.microsoft.com/en-us/library/bb776391%28v=vs.85%29.aspx
|
|
||||||
// http://blogs.msdn.com/b/twistylittlepassagesallalike/archive/2011/04/23/everyone-quotes-arguments-the-wrong-way.aspx
|
|
||||||
// http://blogs.msdn.com/b/oldnewthing/archive/2010/09/17/10063629.aspx
|
|
||||||
cmdline.push_back('"');
|
|
||||||
|
|
||||||
std::string::const_iterator p(arg.begin());
|
|
||||||
while (p != arg.end()) {
|
|
||||||
if (*p == '"') {
|
|
||||||
cmdline.push_back('\\');
|
|
||||||
cmdline.push_back('"');
|
|
||||||
++p;
|
|
||||||
} else if (*p == '\\') {
|
|
||||||
unsigned int num_backslashes = 0;
|
|
||||||
while (p != arg.end() && *p == '\\') {
|
|
||||||
++num_backslashes;
|
|
||||||
++p;
|
|
||||||
}
|
|
||||||
if (p == arg.end() || *p == '"') {
|
|
||||||
// Backslashes need to be escaped
|
|
||||||
num_backslashes *= 2;
|
|
||||||
}
|
|
||||||
while (num_backslashes--) {
|
|
||||||
cmdline.push_back('\\');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
cmdline.push_back(*p++);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
cmdline.push_back('"');
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::string format_cmdline (const std::vector<std::string>& command)
|
|
||||||
{
|
|
||||||
std::string cmdline;
|
|
||||||
for (std::vector<std::string>::const_iterator arg(command.begin()); arg != command.end(); ++arg) {
|
|
||||||
if (arg != command.begin()) {
|
|
||||||
cmdline.push_back(' ');
|
|
||||||
}
|
|
||||||
escape_cmdline_argument(cmdline, *arg);
|
|
||||||
}
|
|
||||||
return cmdline;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int wait_for_child (HANDLE child_handle)
|
|
||||||
{
|
|
||||||
if (WaitForSingleObject(child_handle, INFINITE) == WAIT_FAILED) {
|
|
||||||
throw System_error("WaitForSingleObject", "", GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
DWORD exit_code;
|
|
||||||
if (!GetExitCodeProcess(child_handle, &exit_code)) {
|
|
||||||
throw System_error("GetExitCodeProcess", "", GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
return exit_code;
|
|
||||||
}
|
|
||||||
|
|
||||||
static HANDLE spawn_command (const std::vector<std::string>& command, HANDLE stdin_handle, HANDLE stdout_handle, HANDLE stderr_handle)
|
|
||||||
{
|
|
||||||
PROCESS_INFORMATION proc_info;
|
|
||||||
ZeroMemory(&proc_info, sizeof(proc_info));
|
|
||||||
|
|
||||||
STARTUPINFO start_info;
|
|
||||||
ZeroMemory(&start_info, sizeof(start_info));
|
|
||||||
|
|
||||||
start_info.cb = sizeof(STARTUPINFO);
|
|
||||||
start_info.hStdInput = stdin_handle ? stdin_handle : GetStdHandle(STD_INPUT_HANDLE);
|
|
||||||
start_info.hStdOutput = stdout_handle ? stdout_handle : GetStdHandle(STD_OUTPUT_HANDLE);
|
|
||||||
start_info.hStdError = stderr_handle ? stderr_handle : GetStdHandle(STD_ERROR_HANDLE);
|
|
||||||
start_info.dwFlags |= STARTF_USESTDHANDLES;
|
|
||||||
|
|
||||||
std::string cmdline(format_cmdline(command));
|
|
||||||
|
|
||||||
if (!CreateProcessA(NULL, // application name (NULL to use command line)
|
|
||||||
const_cast<char*>(cmdline.c_str()),
|
|
||||||
NULL, // process security attributes
|
|
||||||
NULL, // primary thread security attributes
|
|
||||||
TRUE, // handles are inherited
|
|
||||||
0, // creation flags
|
|
||||||
NULL, // use parent's environment
|
|
||||||
NULL, // use parent's current directory
|
|
||||||
&start_info,
|
|
||||||
&proc_info)) {
|
|
||||||
throw System_error("CreateProcess", cmdline, GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
CloseHandle(proc_info.hThread);
|
|
||||||
|
|
||||||
return proc_info.hProcess;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exec_command (const std::vector<std::string>& command)
|
|
||||||
{
|
|
||||||
HANDLE child_handle = spawn_command(command, NULL, NULL, NULL);
|
|
||||||
int exit_code = wait_for_child(child_handle);
|
|
||||||
CloseHandle(child_handle);
|
|
||||||
return exit_code;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exec_command (const std::vector<std::string>& command, std::ostream& output)
|
|
||||||
{
|
|
||||||
HANDLE stdout_pipe_reader = NULL;
|
|
||||||
HANDLE stdout_pipe_writer = NULL;
|
|
||||||
SECURITY_ATTRIBUTES sec_attr;
|
|
||||||
|
|
||||||
// Set the bInheritHandle flag so pipe handles are inherited.
|
|
||||||
sec_attr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
|
||||||
sec_attr.bInheritHandle = TRUE;
|
|
||||||
sec_attr.lpSecurityDescriptor = NULL;
|
|
||||||
|
|
||||||
// Create a pipe for the child process's STDOUT.
|
|
||||||
if (!CreatePipe(&stdout_pipe_reader, &stdout_pipe_writer, &sec_attr, 0)) {
|
|
||||||
throw System_error("CreatePipe", "", GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure the read handle to the pipe for STDOUT is not inherited.
|
|
||||||
if (!SetHandleInformation(stdout_pipe_reader, HANDLE_FLAG_INHERIT, 0)) {
|
|
||||||
throw System_error("SetHandleInformation", "", GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
HANDLE child_handle = spawn_command(command, NULL, stdout_pipe_writer, NULL);
|
|
||||||
CloseHandle(stdout_pipe_writer);
|
|
||||||
|
|
||||||
// Read from stdout_pipe_reader.
|
|
||||||
// Note that ReadFile on a pipe may return with bytes_read==0 if the other
|
|
||||||
// end of the pipe writes zero bytes, so don't break out of the read loop
|
|
||||||
// when this happens. When the other end of the pipe closes, ReadFile
|
|
||||||
// fails with ERROR_BROKEN_PIPE.
|
|
||||||
char buffer[1024];
|
|
||||||
DWORD bytes_read;
|
|
||||||
while (ReadFile(stdout_pipe_reader, buffer, sizeof(buffer), &bytes_read, NULL)) {
|
|
||||||
output.write(buffer, bytes_read);
|
|
||||||
}
|
|
||||||
const DWORD read_error = GetLastError();
|
|
||||||
if (read_error != ERROR_BROKEN_PIPE) {
|
|
||||||
throw System_error("ReadFile", "", read_error);
|
|
||||||
}
|
|
||||||
|
|
||||||
CloseHandle(stdout_pipe_reader);
|
|
||||||
|
|
||||||
int exit_code = wait_for_child(child_handle);
|
|
||||||
CloseHandle(child_handle);
|
|
||||||
return exit_code;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exec_command_with_input (const std::vector<std::string>& command, const char* p, size_t len)
|
|
||||||
{
|
|
||||||
HANDLE stdin_pipe_reader = NULL;
|
|
||||||
HANDLE stdin_pipe_writer = NULL;
|
|
||||||
SECURITY_ATTRIBUTES sec_attr;
|
|
||||||
|
|
||||||
// Set the bInheritHandle flag so pipe handles are inherited.
|
|
||||||
sec_attr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
|
||||||
sec_attr.bInheritHandle = TRUE;
|
|
||||||
sec_attr.lpSecurityDescriptor = NULL;
|
|
||||||
|
|
||||||
// Create a pipe for the child process's STDIN.
|
|
||||||
if (!CreatePipe(&stdin_pipe_reader, &stdin_pipe_writer, &sec_attr, 0)) {
|
|
||||||
throw System_error("CreatePipe", "", GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure the write handle to the pipe for STDIN is not inherited.
|
|
||||||
if (!SetHandleInformation(stdin_pipe_writer, HANDLE_FLAG_INHERIT, 0)) {
|
|
||||||
throw System_error("SetHandleInformation", "", GetLastError());
|
|
||||||
}
|
|
||||||
|
|
||||||
HANDLE child_handle = spawn_command(command, stdin_pipe_reader, NULL, NULL);
|
|
||||||
CloseHandle(stdin_pipe_reader);
|
|
||||||
|
|
||||||
// Write to stdin_pipe_writer.
|
|
||||||
while (len > 0) {
|
|
||||||
DWORD bytes_written;
|
|
||||||
if (!WriteFile(stdin_pipe_writer, p, len, &bytes_written, NULL)) {
|
|
||||||
throw System_error("WriteFile", "", GetLastError());
|
|
||||||
}
|
|
||||||
p += bytes_written;
|
|
||||||
len -= bytes_written;
|
|
||||||
}
|
|
||||||
|
|
||||||
CloseHandle(stdin_pipe_writer);
|
|
||||||
|
|
||||||
int exit_code = wait_for_child(child_handle);
|
|
||||||
CloseHandle(child_handle);
|
|
||||||
return exit_code;
|
|
||||||
}
|
|
||||||
|
|
||||||
int exit_status (int status)
|
int exit_status (int status)
|
||||||
{
|
{
|
||||||
return status;
|
return status;
|
||||||
|
|||||||
27
util.cpp
27
util.cpp
@@ -30,9 +30,36 @@
|
|||||||
|
|
||||||
#include "git-crypt.hpp"
|
#include "git-crypt.hpp"
|
||||||
#include "util.hpp"
|
#include "util.hpp"
|
||||||
|
#include "coprocess.hpp"
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
|
int exec_command (const std::vector<std::string>& args)
|
||||||
|
{
|
||||||
|
Coprocess proc;
|
||||||
|
proc.spawn(args);
|
||||||
|
return proc.wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
int exec_command (const std::vector<std::string>& args, std::ostream& output)
|
||||||
|
{
|
||||||
|
Coprocess proc;
|
||||||
|
std::istream* proc_stdout = proc.stdout_pipe();
|
||||||
|
proc.spawn(args);
|
||||||
|
output << proc_stdout->rdbuf();
|
||||||
|
return proc.wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
int exec_command_with_input (const std::vector<std::string>& args, const char* p, size_t len)
|
||||||
|
{
|
||||||
|
Coprocess proc;
|
||||||
|
std::ostream* proc_stdin = proc.stdin_pipe();
|
||||||
|
proc.spawn(args);
|
||||||
|
proc_stdin->write(p, len);
|
||||||
|
proc.close_stdin();
|
||||||
|
return proc.wait();
|
||||||
|
}
|
||||||
|
|
||||||
std::string escape_shell_arg (const std::string& str)
|
std::string escape_shell_arg (const std::string& str)
|
||||||
{
|
{
|
||||||
std::string new_str;
|
std::string new_str;
|
||||||
|
|||||||
Reference in New Issue
Block a user