mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-15 07:34:33 +00:00
27ce8feb4a
This enables IO error reports in both the child and server processes. The scheme still isn't entirely satisfactory and output is jumbled but it beats having no output at all. This will hopefully unblock ARM support (PR18057). git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@200017 91177308-0d34-0410-b5e6-96231b3b80d8
124 lines
3.2 KiB
C++
124 lines
3.2 KiB
C++
//=- RPCChannel.inc - LLVM out-of-process JIT execution for Unix --=//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Implementation of the Unix-specific parts of the RPCChannel class
|
|
// which executes JITed code in a separate process from where it was built.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/Support/Errno.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/wait.h>
|
|
#include <unistd.h>
|
|
|
|
namespace {
|
|
|
|
struct ConnectionData_t {
|
|
int InputPipe;
|
|
int OutputPipe;
|
|
|
|
ConnectionData_t(int in, int out) : InputPipe(in), OutputPipe(out) {}
|
|
};
|
|
|
|
} // namespace
|
|
|
|
namespace llvm {
|
|
|
|
bool RPCChannel::createServer() {
|
|
int PipeFD[2][2];
|
|
pid_t ChildPID;
|
|
|
|
// Create two pipes.
|
|
if (pipe(PipeFD[0]) != 0 || pipe(PipeFD[1]) != 0)
|
|
perror("Error creating pipe: ");
|
|
|
|
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.
|
|
char *args[1] = { NULL };
|
|
int rc = execv(ChildName.c_str(), args);
|
|
if (rc != 0)
|
|
perror("Error executing child process: ");
|
|
} else {
|
|
// 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
|
|
ConnectionData = (void *)new ConnectionData_t(PipeFD[1][0], PipeFD[0][1]);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool RPCChannel::createClient() {
|
|
// Store the parent ends of the pipes
|
|
ConnectionData = (void *)new ConnectionData_t(STDIN_FILENO, STDOUT_FILENO);
|
|
return true;
|
|
}
|
|
|
|
void RPCChannel::Wait() { wait(NULL); }
|
|
|
|
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;
|
|
char Number[10] = { 0 };
|
|
ErrorMsg += "Expecting ";
|
|
sprintf(Number, "%d", (uint32_t)Size);
|
|
ErrorMsg += Number;
|
|
ErrorMsg += " bytes, Got ";
|
|
sprintf(Number, "%d", rc);
|
|
ErrorMsg += Number;
|
|
llvm::errs() << "RPC Error: " << Desc << ": " << ErrorMsg << '\n';
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool RPCChannel::WriteBytes(const void *Data, size_t Size) {
|
|
int rc = write(((ConnectionData_t *)ConnectionData)->OutputPipe, Data, Size);
|
|
return CheckError(rc, Size, "WriteBytes");
|
|
}
|
|
|
|
bool RPCChannel::ReadBytes(void *Data, size_t Size) {
|
|
int rc = read(((ConnectionData_t *)ConnectionData)->InputPipe, Data, Size);
|
|
return CheckError(rc, Size, "ReadBytes");
|
|
}
|
|
|
|
RPCChannel::~RPCChannel() {
|
|
delete static_cast<ConnectionData_t *>(ConnectionData);
|
|
}
|
|
|
|
} // namespace llvm
|