2014-01-23 22:19:45 +00:00
|
|
|
//=- RPCChannel.inc - LLVM out-of-process JIT execution for Unix --=//
|
2013-10-02 17:12:36 +00:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2014-01-23 22:19:45 +00:00
|
|
|
// Implementation of the Unix-specific parts of the RPCChannel class
|
2013-10-02 17:12:36 +00:00
|
|
|
// which executes JITed code in a separate process from where it was built.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2014-01-24 17:18:52 +00:00
|
|
|
#include "llvm/Support/Errno.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2013-10-02 17:12:36 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/wait.h>
|
2014-01-13 08:04:33 +00:00
|
|
|
#include <unistd.h>
|
2013-10-02 17:12:36 +00:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
struct ConnectionData_t {
|
|
|
|
int InputPipe;
|
|
|
|
int OutputPipe;
|
|
|
|
|
|
|
|
ConnectionData_t(int in, int out) : InputPipe(in), OutputPipe(out) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2014-01-23 22:19:45 +00:00
|
|
|
bool RPCChannel::createServer() {
|
2013-10-02 17:12:36 +00:00
|
|
|
int PipeFD[2][2];
|
|
|
|
pid_t ChildPID;
|
|
|
|
|
2013-10-04 19:10:03 +00:00
|
|
|
// Create two pipes.
|
|
|
|
if (pipe(PipeFD[0]) != 0 || pipe(PipeFD[1]) != 0)
|
|
|
|
perror("Error creating pipe: ");
|
2013-10-02 17:12:36 +00:00
|
|
|
|
|
|
|
ChildPID = fork();
|
|
|
|
|
|
|
|
if (ChildPID == 0) {
|
|
|
|
// In the child...
|
|
|
|
|
|
|
|
// Close the parent ends of the pipes
|
|
|
|
close(PipeFD[0][1]);
|
|
|
|
close(PipeFD[1][0]);
|
|
|
|
|
|
|
|
// Use our pipes as stdin and stdout
|
|
|
|
if (PipeFD[0][0] != STDIN_FILENO) {
|
|
|
|
dup2(PipeFD[0][0], STDIN_FILENO);
|
|
|
|
close(PipeFD[0][0]);
|
|
|
|
}
|
|
|
|
if (PipeFD[1][1] != STDOUT_FILENO) {
|
|
|
|
dup2(PipeFD[1][1], STDOUT_FILENO);
|
|
|
|
close(PipeFD[1][1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Execute the child process.
|
2014-04-28 04:05:08 +00:00
|
|
|
char *args[1] = { nullptr };
|
2013-10-02 17:12:36 +00:00
|
|
|
int rc = execv(ChildName.c_str(), args);
|
|
|
|
if (rc != 0)
|
|
|
|
perror("Error executing child process: ");
|
2014-01-23 22:19:45 +00:00
|
|
|
} else {
|
2013-10-02 17:12:36 +00:00
|
|
|
// In the parent...
|
|
|
|
|
|
|
|
// Close the child ends of the pipes
|
|
|
|
close(PipeFD[0][0]);
|
|
|
|
close(PipeFD[1][1]);
|
|
|
|
|
|
|
|
// Store the parent ends of the pipes
|
2014-01-23 22:19:45 +00:00
|
|
|
ConnectionData = (void *)new ConnectionData_t(PipeFD[1][0], PipeFD[0][1]);
|
|
|
|
return true;
|
2014-01-14 22:43:43 +00:00
|
|
|
}
|
2014-01-23 22:19:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RPCChannel::createClient() {
|
|
|
|
// Store the parent ends of the pipes
|
|
|
|
ConnectionData = (void *)new ConnectionData_t(STDIN_FILENO, STDOUT_FILENO);
|
2014-01-14 22:43:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-04-28 04:05:08 +00:00
|
|
|
void RPCChannel::Wait() { wait(nullptr); }
|
2014-01-24 17:18:52 +00:00
|
|
|
|
|
|
|
static bool CheckError(int rc, size_t Size, const char *Desc) {
|
|
|
|
if (rc < 0) {
|
|
|
|
llvm::errs() << "IO Error: " << Desc << ": " << sys::StrError() << '\n';
|
|
|
|
return false;
|
|
|
|
} else if ((size_t)rc != Size) {
|
|
|
|
std::string ErrorMsg;
|
2014-01-14 22:43:43 +00:00
|
|
|
char Number[10] = { 0 };
|
|
|
|
ErrorMsg += "Expecting ";
|
|
|
|
sprintf(Number, "%d", (uint32_t)Size);
|
|
|
|
ErrorMsg += Number;
|
|
|
|
ErrorMsg += " bytes, Got ";
|
|
|
|
sprintf(Number, "%d", rc);
|
|
|
|
ErrorMsg += Number;
|
2014-01-24 17:18:52 +00:00
|
|
|
llvm::errs() << "RPC Error: " << Desc << ": " << ErrorMsg << '\n';
|
|
|
|
return false;
|
2013-10-02 17:12:36 +00:00
|
|
|
}
|
2014-01-24 17:18:52 +00:00
|
|
|
return true;
|
2013-10-02 17:12:36 +00:00
|
|
|
}
|
|
|
|
|
2014-01-24 17:18:52 +00:00
|
|
|
bool RPCChannel::WriteBytes(const void *Data, size_t Size) {
|
|
|
|
int rc = write(((ConnectionData_t *)ConnectionData)->OutputPipe, Data, Size);
|
|
|
|
return CheckError(rc, Size, "WriteBytes");
|
2013-10-02 17:12:36 +00:00
|
|
|
}
|
|
|
|
|
2014-01-24 17:18:52 +00:00
|
|
|
bool RPCChannel::ReadBytes(void *Data, size_t Size) {
|
|
|
|
int rc = read(((ConnectionData_t *)ConnectionData)->InputPipe, Data, Size);
|
|
|
|
return CheckError(rc, Size, "ReadBytes");
|
2013-10-02 17:12:36 +00:00
|
|
|
}
|
|
|
|
|
2014-01-23 22:19:45 +00:00
|
|
|
RPCChannel::~RPCChannel() {
|
2013-10-05 11:53:20 +00:00
|
|
|
delete static_cast<ConnectionData_t *>(ConnectionData);
|
|
|
|
}
|
|
|
|
|
2013-10-02 21:58:02 +00:00
|
|
|
} // namespace llvm
|