1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286
|
/* -*- c-file-style: "java"; indent-tabs-mode: nil; tab-width: 4; fill-column: 78 -*-
*
* distcc -- A simple distributed compiler system
*
* Copyright (C) 2002, 2003, 2004 by Martin Pool <mbp@samba.org>
* Copyright 2007 Google Inc.
*
* This program 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 2
* of the License, or (at your option) any later version.
*
* This program 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 this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
* USA.
*/
#include <config.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include "distcc.h"
#include "trace.h"
#include "exec.h"
#include "rpc.h"
#include "exitcode.h"
#include "util.h"
#include "clinet.h"
#include "bulk.h"
#include "hosts.h"
#include "state.h"
#include "include_server_if.h"
#include "emaillog.h"
/**
* @file
*
* @brief Client-side RPC functions.
**/
/*
* Transmit header for whole request.
*/
int dcc_x_req_header(int fd,
enum dcc_protover protover)
{
return dcc_x_token_int(fd, "DIST", protover);
}
/**
* Transmit an argv array.
**/
int dcc_x_argv(int fd, char **argv)
{
int i;
int ret;
int argc;
argc = dcc_argv_len(argv);
if (dcc_x_token_int(fd, "ARGC", (unsigned) argc))
return EXIT_PROTOCOL_ERROR;
for (i = 0; i < argc; i++) {
if ((ret = dcc_x_token_string(fd, "ARGV", argv[i])))
return ret;
}
return 0;
}
/**
* Transmit the current working directory
*/
int dcc_x_cwd(int fd)
{
int ret;
char cwd[MAXPATHLEN + 1];
char * cwd_ret;
cwd_ret = getcwd(cwd, MAXPATHLEN);
if (cwd_ret == NULL) {
return 0;
}
ret = dcc_x_token_string(fd, "CDIR", cwd);
return ret;
}
/**
* Read the "DONE" token from the network that introduces a response.
**/
int dcc_r_result_header(int ifd,
enum dcc_protover expect_ver)
{
unsigned vers;
int ret;
if ((ret = dcc_r_token_int(ifd, "DONE", &vers)))
rs_log_error("server provided no answer. "
"Is the server configured to allow access from your IP"
" address? Does the server have the compiler installed?"
" Is the server configured to access the compiler?");
return ret;
if (vers != expect_ver) {
rs_log_error("got version %d not %d in response from server",
vers, expect_ver);
return EXIT_PROTOCOL_ERROR;
}
rs_trace("got response header");
return 0;
}
int dcc_r_cc_status(int ifd, int *status)
{
unsigned u_status;
int ret;
ret = dcc_r_token_int(ifd, "STAT", &u_status);
*status = u_status;
return ret;
}
/**
* The second half of the client protocol: retrieve all results from the server.
**/
int dcc_retrieve_results(int net_fd,
int *status,
const char *output_fname,
const char *deps_fname,
const char *server_stderr_fname,
struct dcc_hostdef *host)
{
unsigned len;
int ret;
unsigned o_len;
if ((ret = dcc_r_result_header(net_fd, host->protover)))
return ret;
/* We've started to see the response, so the server is done
* compiling. */
dcc_note_state(DCC_PHASE_RECEIVE, NULL, NULL, DCC_REMOTE);
if ((ret = dcc_r_cc_status(net_fd, status)))
return ret;
if ((ret = dcc_r_token_int(net_fd, "SERR", &len)))
return ret;
/* Save the server-side errors into a file. This way, we can
decide later whether we want to report them to the user
or not. We don't want to report them to the user if
we are going to redo the compilation locally, because then
the local errors are going to appear.
Always put the server-side errors in the email we will
send to the maintainers, though.
*/
if ((ret = dcc_r_file(net_fd, server_stderr_fname, len, host->compr)))
return ret;
if (dcc_add_file_to_log_email("server-side stderr", server_stderr_fname))
return ret;
if ((ret = dcc_r_token_int(net_fd, "SOUT", &len))
|| (ret = dcc_r_bulk(STDOUT_FILENO, net_fd, len, host->compr))
|| (ret = dcc_r_token_int(net_fd, "DOTO", &o_len)))
return ret;
/* If the compiler succeeded, then we always retrieve the result,
* even if it's 0 bytes. */
if (*status == 0) {
if ((ret = dcc_r_file_timed(net_fd, output_fname, o_len, host->compr)))
return ret;
if (host->cpp_where == DCC_CPP_ON_SERVER) {
if ((ret = dcc_r_token_int(net_fd, "DOTD", &len) == 0)
&& deps_fname != NULL) {
ret = dcc_r_file_timed(net_fd, deps_fname, len, host->compr);
return ret;
}
}
} else if (o_len != 0) {
rs_log_error("remote compiler failed but also returned output: "
"I don't know what to do");
}
return 0;
}
/* points_to must be at least MAXPATHLEN + 1 long */
int dcc_read_link(const char* fname, char *points_to)
{
int len;
if ((len = readlink(fname, points_to, MAXPATHLEN)) == -1) {
rs_log_error("readlink '%s' failed: %s", fname, strerror(errno));
return EXIT_IO_ERROR;
}
points_to[len] = '\0';
return 0;
}
int dcc_is_link(const char *fname, int *is_link)
{
struct stat buf;
if (lstat(fname, &buf) == -1) {
rs_log_error("stat '%s' failed: %s", fname, strerror(errno));
return EXIT_IO_ERROR;
}
*is_link = S_ISLNK(buf.st_mode);
return 0;
}
/* Send to @p ofd @p n_files whose names are in @p fnames.
* @fnames must be null-terminated.
* The names can be coming from the include server, so
* we consult dcc_get_original_fname to get the real names.
* Always uses lzo compression.
*/
/* TODO: This code is highly specific to DCC_VER_3; it assumes
lzo compression is on, and that the include server has
actually compressed the files. */
int dcc_x_many_files(int ofd,
unsigned int n_files,
char **fnames)
{
int ret;
char link_points_to[MAXPATHLEN + 1];
int is_link;
const char *fname;
char *original_fname;
dcc_x_token_int(ofd, "NFIL", n_files);
for (; *fnames != NULL; ++fnames) {
fname = *fnames;
ret = dcc_get_original_fname(fname, &original_fname);
if (ret) return ret;
if ((ret = dcc_is_link(fname, &is_link))) {
return ret;
}
if (is_link) {
if ((ret = dcc_read_link(fname, link_points_to)) ||
(ret = dcc_x_token_string(ofd, "NAME", original_fname)) ||
(ret = dcc_x_token_string(ofd, "LINK", link_points_to))) {
return ret;
}
} else {
ret = dcc_x_token_string(ofd, "NAME", original_fname);
if (ret) return ret;
/* File should be compressed already.
If we ever support non-compressed server-side-cpp,
we should have some checks here and then uncompress
the file if it is compressed. */
ret = dcc_x_file(ofd, fname, "FILE", DCC_COMPRESS_NONE,
NULL);
if (ret) return ret;
}
}
return 0;
}
|