blob: 6a9ae14b599fae13588d1804bd8e57cae199f722 [file] [log] [blame]
Alp Toker632c6cd2014-01-23 22:19:45 +00001//=- RPCChannel.inc - LLVM out-of-process JIT execution for Unix --=//
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
Alp Toker632c6cd2014-01-23 22:19:45 +000010// Implementation of the Unix-specific parts of the RPCChannel class
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000011// which executes JITed code in a separate process from where it was built.
12//
13//===----------------------------------------------------------------------===//
14
Alp Tokerad6aa472014-01-24 17:18:52 +000015#include "llvm/Support/Errno.h"
16#include "llvm/Support/raw_ostream.h"
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000017#include <stdio.h>
18#include <stdlib.h>
19#include <sys/wait.h>
Chandler Carruth07baed52014-01-13 08:04:33 +000020#include <unistd.h>
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000021
22namespace {
23
24struct ConnectionData_t {
25 int InputPipe;
26 int OutputPipe;
27
28 ConnectionData_t(int in, int out) : InputPipe(in), OutputPipe(out) {}
29};
30
31} // namespace
32
33namespace llvm {
34
Alp Toker632c6cd2014-01-23 22:19:45 +000035bool RPCChannel::createServer() {
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000036 int PipeFD[2][2];
37 pid_t ChildPID;
38
Benjamin Kramer3cfafb82013-10-04 19:10:03 +000039 // Create two pipes.
40 if (pipe(PipeFD[0]) != 0 || pipe(PipeFD[1]) != 0)
41 perror("Error creating pipe: ");
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000042
43 ChildPID = fork();
44
45 if (ChildPID == 0) {
46 // In the child...
47
48 // Close the parent ends of the pipes
49 close(PipeFD[0][1]);
50 close(PipeFD[1][0]);
51
52 // Use our pipes as stdin and stdout
53 if (PipeFD[0][0] != STDIN_FILENO) {
54 dup2(PipeFD[0][0], STDIN_FILENO);
55 close(PipeFD[0][0]);
56 }
57 if (PipeFD[1][1] != STDOUT_FILENO) {
58 dup2(PipeFD[1][1], STDOUT_FILENO);
59 close(PipeFD[1][1]);
60 }
61
62 // Execute the child process.
Craig Toppere73658d2014-04-28 04:05:08 +000063 char *args[1] = { nullptr };
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000064 int rc = execv(ChildName.c_str(), args);
65 if (rc != 0)
66 perror("Error executing child process: ");
Alp Toker632c6cd2014-01-23 22:19:45 +000067 } else {
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +000068 // In the parent...
69
70 // Close the child ends of the pipes
71 close(PipeFD[0][0]);
72 close(PipeFD[1][1]);
73
74 // Store the parent ends of the pipes
Alp Toker632c6cd2014-01-23 22:19:45 +000075 ConnectionData = (void *)new ConnectionData_t(PipeFD[1][0], PipeFD[0][1]);
76 return true;
Renato Golin695895c2014-01-14 22:43:43 +000077 }
Alp Toker632c6cd2014-01-23 22:19:45 +000078 return false;
79}
80
81bool RPCChannel::createClient() {
82 // Store the parent ends of the pipes
83 ConnectionData = (void *)new ConnectionData_t(STDIN_FILENO, STDOUT_FILENO);
Renato Golin695895c2014-01-14 22:43:43 +000084 return true;
85}
86
Craig Toppere73658d2014-04-28 04:05:08 +000087void RPCChannel::Wait() { wait(nullptr); }
Alp Tokerad6aa472014-01-24 17:18:52 +000088
89static bool CheckError(int rc, size_t Size, const char *Desc) {
90 if (rc < 0) {
91 llvm::errs() << "IO Error: " << Desc << ": " << sys::StrError() << '\n';
92 return false;
93 } else if ((size_t)rc != Size) {
94 std::string ErrorMsg;
Renato Golin695895c2014-01-14 22:43:43 +000095 char Number[10] = { 0 };
96 ErrorMsg += "Expecting ";
97 sprintf(Number, "%d", (uint32_t)Size);
98 ErrorMsg += Number;
99 ErrorMsg += " bytes, Got ";
100 sprintf(Number, "%d", rc);
101 ErrorMsg += Number;
Alp Tokerad6aa472014-01-24 17:18:52 +0000102 llvm::errs() << "RPC Error: " << Desc << ": " << ErrorMsg << '\n';
103 return false;
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +0000104 }
Alp Tokerad6aa472014-01-24 17:18:52 +0000105 return true;
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +0000106}
107
Alp Tokerad6aa472014-01-24 17:18:52 +0000108bool RPCChannel::WriteBytes(const void *Data, size_t Size) {
109 int rc = write(((ConnectionData_t *)ConnectionData)->OutputPipe, Data, Size);
110 return CheckError(rc, Size, "WriteBytes");
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +0000111}
112
Alp Tokerad6aa472014-01-24 17:18:52 +0000113bool RPCChannel::ReadBytes(void *Data, size_t Size) {
114 int rc = read(((ConnectionData_t *)ConnectionData)->InputPipe, Data, Size);
115 return CheckError(rc, Size, "ReadBytes");
Andrew Kaylorc2ebf3f2013-10-02 17:12:36 +0000116}
117
Alp Toker632c6cd2014-01-23 22:19:45 +0000118RPCChannel::~RPCChannel() {
Benjamin Kramerc83946f2013-10-05 11:53:20 +0000119 delete static_cast<ConnectionData_t *>(ConnectionData);
120}
121
Benjamin Kramerc12c7d02013-10-02 21:58:02 +0000122} // namespace llvm