2002-03-26 15:46:43 +01:00
|
|
|
/*
|
|
|
|
a re-implementation of the compilercache scripts in C
|
|
|
|
|
|
|
|
The idea is based on the shell-script compilercache by Erik Thiele <erikyyy@erikyyy.de>
|
2002-03-27 00:58:31 +01:00
|
|
|
|
|
|
|
Copyright (C) Andrew Tridgell 2002
|
|
|
|
|
|
|
|
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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
2002-03-26 15:46:43 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ccache.h"
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
/* the base cache directory */
|
2002-03-30 11:43:26 +01:00
|
|
|
char *cache_dir = NULL;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the debug logfile name, if set */
|
2002-03-27 00:58:31 +01:00
|
|
|
char *cache_logfile = NULL;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the argument list after processing */
|
2002-03-26 15:46:43 +01:00
|
|
|
static ARGS *stripped_args;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the original argument list */
|
2002-03-26 15:46:43 +01:00
|
|
|
static ARGS *orig_args;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the output filename being compiled to */
|
2002-03-26 15:46:43 +01:00
|
|
|
static char *output_file;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the source file */
|
2002-04-02 06:09:56 +02:00
|
|
|
static char *input_file;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the name of the file containing the cached object code */
|
2002-03-26 15:46:43 +01:00
|
|
|
static char *hashname;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the extension of the file after pre-processing */
|
2003-01-06 03:10:15 +01:00
|
|
|
static const char *i_extension;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* the name of the temporary pre-processor file */
|
|
|
|
static char *i_tmpfile;
|
|
|
|
|
2002-05-12 11:51:36 +02:00
|
|
|
/* the name of the cpp stderr file */
|
|
|
|
static char *cpp_stderr;
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
/* the name of the statistics file */
|
2002-03-30 11:43:26 +01:00
|
|
|
char *stats_file = NULL;
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
/* did we find a -g option? */
|
2002-03-26 15:46:43 +01:00
|
|
|
static int found_debug;
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
/* a list of supported file extensions, and the equivalent
|
|
|
|
extension for code that has been through the pre-processor
|
|
|
|
*/
|
|
|
|
static struct {
|
|
|
|
char *extension;
|
|
|
|
char *i_extension;
|
|
|
|
} extensions[] = {
|
|
|
|
{"c", "i"},
|
|
|
|
{"C", "ii"},
|
|
|
|
{"m", "mi"},
|
|
|
|
{"cc", "ii"},
|
|
|
|
{"CC", "ii"},
|
|
|
|
{"cpp", "ii"},
|
|
|
|
{"CPP", "ii"},
|
|
|
|
{"cxx", "ii"},
|
|
|
|
{"CXX", "ii"},
|
2002-04-24 15:13:05 +02:00
|
|
|
{"c++", "ii"},
|
|
|
|
{"C++", "ii"},
|
2002-04-07 15:20:53 +02:00
|
|
|
{NULL, NULL}};
|
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/*
|
|
|
|
something went badly wrong - just execute the real compiler
|
|
|
|
*/
|
2002-03-26 15:46:43 +01:00
|
|
|
static void failed(void)
|
|
|
|
{
|
2003-02-16 03:28:35 +01:00
|
|
|
char *e;
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
/* delete intermediate pre-processor file if needed */
|
|
|
|
if (i_tmpfile) {
|
|
|
|
unlink(i_tmpfile);
|
|
|
|
free(i_tmpfile);
|
|
|
|
i_tmpfile = NULL;
|
|
|
|
}
|
2002-05-12 11:51:36 +02:00
|
|
|
|
|
|
|
/* delete the cpp stderr file if necessary */
|
|
|
|
if (cpp_stderr) {
|
|
|
|
unlink(cpp_stderr);
|
|
|
|
free(cpp_stderr);
|
|
|
|
cpp_stderr = NULL;
|
|
|
|
}
|
|
|
|
|
2003-02-08 06:49:22 +01:00
|
|
|
/* strip any local args */
|
|
|
|
args_strip(orig_args, "--ccache-");
|
|
|
|
|
2003-02-16 03:28:35 +01:00
|
|
|
if ((e=getenv("CCACHE_PREFIX"))) {
|
|
|
|
char *p = find_executable(e, MYNAME);
|
|
|
|
if (!p) {
|
|
|
|
perror(e);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
args_add_prefix(orig_args, p);
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
execv(orig_args->argv[0], orig_args->argv);
|
|
|
|
cc_log("execv returned (%s)!\n", strerror(errno));
|
2002-04-03 16:45:51 +02:00
|
|
|
perror(orig_args->argv[0]);
|
2002-03-26 15:46:43 +01:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2002-05-29 14:22:21 +02:00
|
|
|
|
|
|
|
/* return a string to be used to distinguish temporary files
|
|
|
|
this also tries to cope with NFS by adding the local hostname
|
|
|
|
*/
|
|
|
|
static const char *tmp_string(void)
|
|
|
|
{
|
|
|
|
static char *ret;
|
|
|
|
|
|
|
|
if (!ret) {
|
|
|
|
char hostname[200];
|
|
|
|
strcpy(hostname, "unknown");
|
2003-03-07 13:09:19 +01:00
|
|
|
#if HAVE_GETHOSTNAME
|
2002-05-29 14:22:21 +02:00
|
|
|
gethostname(hostname, sizeof(hostname)-1);
|
2003-03-07 13:09:19 +01:00
|
|
|
#endif
|
2002-05-29 14:22:21 +02:00
|
|
|
hostname[sizeof(hostname)-1] = 0;
|
|
|
|
asprintf(&ret, "%s.%u", hostname, (unsigned)getpid());
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
/* run the real compiler and put the result in cache */
|
|
|
|
static void to_cache(ARGS *args)
|
|
|
|
{
|
2002-03-27 01:39:06 +01:00
|
|
|
char *path_stderr;
|
|
|
|
char *tmp_stdout, *tmp_stderr, *tmp_hashname;
|
2002-03-30 11:43:26 +01:00
|
|
|
struct stat st1, st2;
|
2002-03-27 01:39:06 +01:00
|
|
|
int status;
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-05-29 14:22:21 +02:00
|
|
|
x_asprintf(&tmp_stdout, "%s/tmp.stdout.%s", cache_dir, tmp_string());
|
|
|
|
x_asprintf(&tmp_stderr, "%s/tmp.stderr.%s", cache_dir, tmp_string());
|
|
|
|
x_asprintf(&tmp_hashname, "%s/tmp.hash.%s.o", cache_dir, tmp_string());
|
2002-03-26 15:46:43 +01:00
|
|
|
|
|
|
|
args_add(args, "-o");
|
2002-03-27 01:39:06 +01:00
|
|
|
args_add(args, tmp_hashname);
|
2002-04-07 15:20:53 +02:00
|
|
|
|
|
|
|
if (getenv("CCACHE_CPP2")) {
|
|
|
|
args_add(args, input_file);
|
|
|
|
} else {
|
|
|
|
args_add(args, i_tmpfile);
|
|
|
|
}
|
2002-03-27 01:39:06 +01:00
|
|
|
status = execute(args->argv, tmp_stdout, tmp_stderr);
|
2002-04-07 15:20:53 +02:00
|
|
|
args_pop(args, 3);
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-03-30 11:43:26 +01:00
|
|
|
if (stat(tmp_stdout, &st1) != 0 || st1.st_size != 0) {
|
2002-03-27 02:29:53 +01:00
|
|
|
cc_log("compiler produced stdout for %s\n", output_file);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_STDOUT);
|
2002-03-27 01:39:06 +01:00
|
|
|
unlink(tmp_stdout);
|
|
|
|
unlink(tmp_stderr);
|
|
|
|
unlink(tmp_hashname);
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-27 02:29:53 +01:00
|
|
|
unlink(tmp_stdout);
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-03-27 02:29:53 +01:00
|
|
|
if (status != 0) {
|
|
|
|
int fd;
|
|
|
|
cc_log("compile of %s gave status = %d\n", output_file, status);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_STATUS);
|
2002-03-27 02:29:53 +01:00
|
|
|
|
|
|
|
fd = open(tmp_stderr, O_RDONLY);
|
2002-04-08 05:53:14 +02:00
|
|
|
if (fd != -1) {
|
|
|
|
if (strcmp(output_file, "/dev/null") == 0 ||
|
|
|
|
rename(tmp_hashname, output_file) == 0 || errno == ENOENT) {
|
2002-05-12 11:51:36 +02:00
|
|
|
if (cpp_stderr) {
|
|
|
|
/* we might have some stderr from cpp */
|
|
|
|
int fd2 = open(cpp_stderr, O_RDONLY);
|
|
|
|
if (fd2 != -1) {
|
|
|
|
copy_fd(fd2, 2);
|
|
|
|
close(fd2);
|
|
|
|
unlink(cpp_stderr);
|
|
|
|
cpp_stderr = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we can use a quick method of
|
|
|
|
getting the failed output */
|
2002-04-08 05:53:14 +02:00
|
|
|
copy_fd(fd, 2);
|
|
|
|
close(fd);
|
|
|
|
unlink(tmp_stderr);
|
|
|
|
if (i_tmpfile) {
|
|
|
|
unlink(i_tmpfile);
|
|
|
|
}
|
|
|
|
exit(status);
|
2002-04-07 16:03:15 +02:00
|
|
|
}
|
2002-03-27 02:29:53 +01:00
|
|
|
}
|
|
|
|
|
2002-03-27 01:39:06 +01:00
|
|
|
unlink(tmp_stderr);
|
|
|
|
unlink(tmp_hashname);
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-03-27 01:39:06 +01:00
|
|
|
x_asprintf(&path_stderr, "%s.stderr", hashname);
|
|
|
|
|
2002-03-30 11:43:26 +01:00
|
|
|
if (stat(tmp_stderr, &st1) != 0 ||
|
|
|
|
stat(tmp_hashname, &st2) != 0 ||
|
|
|
|
rename(tmp_hashname, hashname) != 0 ||
|
2002-03-27 01:39:06 +01:00
|
|
|
rename(tmp_stderr, path_stderr) != 0) {
|
2002-06-03 05:22:17 +02:00
|
|
|
cc_log("failed to rename tmp files - %s\n", strerror(errno));
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_ERROR);
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
cc_log("Placed %s into cache\n", output_file);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_tocache(file_size(&st1) + file_size(&st2));
|
2002-03-27 01:39:06 +01:00
|
|
|
|
|
|
|
free(tmp_hashname);
|
|
|
|
free(tmp_stderr);
|
|
|
|
free(tmp_stdout);
|
|
|
|
free(path_stderr);
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* find the hash for a command. The hash includes all argument lists,
|
|
|
|
plus the output from running the compiler with -E */
|
|
|
|
static void find_hash(ARGS *args)
|
|
|
|
{
|
|
|
|
int i;
|
2002-03-27 01:39:06 +01:00
|
|
|
char *path_stdout, *path_stderr;
|
2002-03-26 16:11:48 +01:00
|
|
|
char *hash_dir;
|
|
|
|
char *s;
|
2002-03-26 23:25:14 +01:00
|
|
|
struct stat st;
|
2002-03-27 01:39:06 +01:00
|
|
|
int status;
|
2002-04-03 16:45:51 +02:00
|
|
|
int nlevels = 2;
|
2003-09-24 08:39:27 +02:00
|
|
|
char *input_base;
|
|
|
|
char *tmp;
|
|
|
|
|
2002-04-03 16:45:51 +02:00
|
|
|
if ((s = getenv("CCACHE_NLEVELS"))) {
|
|
|
|
nlevels = atoi(s);
|
|
|
|
if (nlevels < 1) nlevels = 1;
|
|
|
|
if (nlevels > 8) nlevels = 8;
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
|
|
|
|
hash_start();
|
|
|
|
|
2002-04-23 04:10:22 +02:00
|
|
|
/* when we are doing the unifying tricks we need to include
|
|
|
|
the input file name in the hash to get the warnings right */
|
|
|
|
if (!found_debug) {
|
|
|
|
hash_string(input_file);
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
/* first the arguments */
|
2002-04-02 05:29:05 +02:00
|
|
|
for (i=1;i<args->argc;i++) {
|
2002-03-27 07:30:31 +01:00
|
|
|
/* some arguments don't contribute to the hash. The
|
|
|
|
theory is that these arguments will change the
|
|
|
|
output of -E if they are going to have any effect
|
|
|
|
at all, or they only affect linking */
|
|
|
|
if (i < args->argc-1) {
|
|
|
|
if (strcmp(args->argv[i], "-I") == 0 ||
|
|
|
|
strcmp(args->argv[i], "-include") == 0 ||
|
|
|
|
strcmp(args->argv[i], "-L") == 0 ||
|
|
|
|
strcmp(args->argv[i], "-D") == 0 ||
|
|
|
|
strcmp(args->argv[i], "-isystem") == 0) {
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
2002-04-08 03:32:07 +02:00
|
|
|
}
|
|
|
|
if (strncmp(args->argv[i], "-I", 2) == 0 ||
|
|
|
|
strncmp(args->argv[i], "-L", 2) == 0 ||
|
|
|
|
strncmp(args->argv[i], "-D", 2) == 0 ||
|
|
|
|
strncmp(args->argv[i], "-isystem", 8) == 0) {
|
|
|
|
continue;
|
2002-03-27 07:30:31 +01:00
|
|
|
}
|
2002-04-09 05:39:20 +02:00
|
|
|
|
|
|
|
if (strncmp(args->argv[i], "--specs=", 8) == 0 &&
|
|
|
|
stat(args->argv[i]+8, &st) == 0) {
|
|
|
|
/* if given a explicit specs file, then hash that file, but
|
|
|
|
don't include the path to it in the hash */
|
|
|
|
hash_file(args->argv[i]+8);
|
|
|
|
continue;
|
|
|
|
}
|
2002-04-09 05:47:29 +02:00
|
|
|
|
|
|
|
/* all other arguments are included in the hash */
|
2002-03-26 15:46:43 +01:00
|
|
|
hash_string(args->argv[i]);
|
|
|
|
}
|
|
|
|
|
2002-03-26 23:25:14 +01:00
|
|
|
/* the compiler driver size and date. This is a simple minded way
|
|
|
|
to try and detect compiler upgrades. It is not 100% reliable */
|
|
|
|
if (stat(args->argv[0], &st) != 0) {
|
2003-02-16 03:28:35 +01:00
|
|
|
cc_log("Couldn't stat the compiler!? (argv[0]='%s')\n", args->argv[0]);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_COMPILER);
|
2002-03-26 23:25:14 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
hash_int(st.st_size);
|
|
|
|
hash_int(st.st_mtime);
|
|
|
|
|
2003-01-07 06:19:00 +01:00
|
|
|
/* possibly hash the current working directory */
|
|
|
|
if (getenv("CCACHE_HASHDIR")) {
|
|
|
|
char *cwd = gnu_getcwd();
|
|
|
|
if (cwd) {
|
|
|
|
hash_string(cwd);
|
|
|
|
free(cwd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-09-24 08:39:27 +02:00
|
|
|
/* ~/hello.c -> tmp.hello.123.i
|
|
|
|
limit the basename to 10
|
|
|
|
characters in order to cope with filesystem with small
|
|
|
|
maximum filename length limits */
|
|
|
|
input_base = basename(input_file);
|
|
|
|
tmp = strchr(input_base, '.');
|
|
|
|
if (tmp != NULL) {
|
|
|
|
*tmp = 0;
|
|
|
|
}
|
|
|
|
if (strlen(input_base) > 10) {
|
|
|
|
input_base[10] = 0;
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
/* now the run */
|
2003-09-24 08:39:27 +02:00
|
|
|
x_asprintf(&path_stdout, "%s/%s.tmp.%s.%s", cache_dir,
|
|
|
|
input_base, tmp_string(),
|
2002-04-07 15:20:53 +02:00
|
|
|
i_extension);
|
2002-05-29 14:22:21 +02:00
|
|
|
x_asprintf(&path_stderr, "%s/tmp.cpp_stderr.%s", cache_dir, tmp_string());
|
2002-03-26 15:46:43 +01:00
|
|
|
|
|
|
|
args_add(args, "-E");
|
2002-04-07 15:20:53 +02:00
|
|
|
args_add(args, input_file);
|
2002-03-27 01:39:06 +01:00
|
|
|
status = execute(args->argv, path_stdout, path_stderr);
|
2002-04-07 15:20:53 +02:00
|
|
|
args_pop(args, 2);
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-03-27 01:39:06 +01:00
|
|
|
if (status != 0) {
|
|
|
|
unlink(path_stdout);
|
|
|
|
unlink(path_stderr);
|
|
|
|
cc_log("the preprocessor gave %d\n", status);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_PREPROCESSOR);
|
2002-03-27 01:39:06 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 23:25:14 +01:00
|
|
|
/* if the compilation is with -g then we have to inlcude the whole of the
|
|
|
|
preprocessor output, which means we are sensitive to line number
|
|
|
|
information. Otherwise we can discard line number info, which makes
|
|
|
|
us less sensitive to reformatting changes
|
2002-04-29 11:12:46 +02:00
|
|
|
|
|
|
|
Note! I have now disabled the unification code by default
|
|
|
|
as it gives the wrong line numbers for warnings. Pity.
|
2002-03-26 23:25:14 +01:00
|
|
|
*/
|
2002-04-29 11:12:46 +02:00
|
|
|
if (found_debug || !getenv("CCACHE_UNIFY")) {
|
2002-03-26 15:46:43 +01:00
|
|
|
hash_file(path_stdout);
|
|
|
|
} else {
|
2002-04-01 02:23:31 +02:00
|
|
|
if (unify_hash(path_stdout) != 0) {
|
2002-04-18 12:26:47 +02:00
|
|
|
stats_update(STATS_ERROR);
|
2002-04-01 02:23:31 +02:00
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
hash_file(path_stderr);
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
i_tmpfile = path_stdout;
|
2002-05-12 11:51:36 +02:00
|
|
|
|
|
|
|
if (!getenv("CCACHE_CPP2")) {
|
|
|
|
/* if we are using the CPP trick then we need to remember this stderr
|
|
|
|
data and output it just before the main stderr from the compiler
|
|
|
|
pass */
|
|
|
|
cpp_stderr = path_stderr;
|
|
|
|
} else {
|
|
|
|
unlink(path_stderr);
|
|
|
|
free(path_stderr);
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-04-03 16:45:51 +02:00
|
|
|
/* we use a N level subdir for the cache path to reduce the impact
|
2002-03-27 00:58:31 +01:00
|
|
|
on filesystems which are slow for large directories
|
|
|
|
*/
|
2002-03-26 16:11:48 +01:00
|
|
|
s = hash_result();
|
2002-04-03 16:45:51 +02:00
|
|
|
x_asprintf(&hash_dir, "%s/%c", cache_dir, s[0]);
|
|
|
|
x_asprintf(&stats_file, "%s/stats", hash_dir);
|
|
|
|
for (i=1; i<nlevels; i++) {
|
|
|
|
char *p;
|
|
|
|
if (create_dir(hash_dir) != 0) {
|
|
|
|
cc_log("failed to create %s\n", hash_dir);
|
|
|
|
failed();
|
|
|
|
}
|
|
|
|
x_asprintf(&p, "%s/%c", hash_dir, s[i]);
|
|
|
|
free(hash_dir);
|
|
|
|
hash_dir = p;
|
2002-04-02 07:04:49 +02:00
|
|
|
}
|
2002-03-26 23:25:14 +01:00
|
|
|
if (create_dir(hash_dir) != 0) {
|
2002-04-02 07:04:49 +02:00
|
|
|
cc_log("failed to create %s\n", hash_dir);
|
2002-03-26 23:25:14 +01:00
|
|
|
failed();
|
|
|
|
}
|
2002-04-03 16:45:51 +02:00
|
|
|
x_asprintf(&hashname, "%s/%s", hash_dir, s+nlevels);
|
2002-03-26 16:11:48 +01:00
|
|
|
free(hash_dir);
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
try to return the compile result from cache. If we can return from
|
|
|
|
cache then this function exits with the correct status code,
|
|
|
|
otherwise it returns */
|
|
|
|
static void from_cache(int first)
|
|
|
|
{
|
2002-05-12 11:51:36 +02:00
|
|
|
int fd_stderr, fd_cpp_stderr;
|
2002-03-28 05:22:40 +01:00
|
|
|
char *stderr_file;
|
2002-03-27 01:39:06 +01:00
|
|
|
int ret;
|
2002-03-30 11:43:26 +01:00
|
|
|
struct stat st;
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-03-28 05:22:40 +01:00
|
|
|
x_asprintf(&stderr_file, "%s.stderr", hashname);
|
|
|
|
fd_stderr = open(stderr_file, O_RDONLY);
|
2002-03-26 15:46:43 +01:00
|
|
|
if (fd_stderr == -1) {
|
2002-03-27 01:39:06 +01:00
|
|
|
/* it isn't in cache ... */
|
2002-03-28 05:22:40 +01:00
|
|
|
free(stderr_file);
|
2002-03-26 15:46:43 +01:00
|
|
|
return;
|
|
|
|
}
|
2002-03-30 11:43:26 +01:00
|
|
|
|
|
|
|
/* make sure the output is there too */
|
|
|
|
if (stat(hashname, &st) != 0) {
|
|
|
|
close(fd_stderr);
|
|
|
|
unlink(stderr_file);
|
|
|
|
free(stderr_file);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-01-06 01:20:54 +01:00
|
|
|
/* the user might be disabling cache hits */
|
|
|
|
if (first && getenv("CCACHE_RECACHE")) {
|
|
|
|
close(fd_stderr);
|
|
|
|
unlink(stderr_file);
|
|
|
|
free(stderr_file);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2002-03-28 05:22:40 +01:00
|
|
|
utime(stderr_file, NULL);
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-04-08 05:53:14 +02:00
|
|
|
if (strcmp(output_file, "/dev/null") == 0) {
|
|
|
|
ret = 0;
|
2002-04-01 08:58:31 +02:00
|
|
|
} else {
|
2002-04-08 05:53:14 +02:00
|
|
|
unlink(output_file);
|
2003-02-17 01:46:15 +01:00
|
|
|
if (getenv("CCACHE_HARDLINK")) {
|
2002-04-08 05:53:14 +02:00
|
|
|
ret = link(hashname, output_file);
|
2003-02-17 01:46:15 +01:00
|
|
|
} else {
|
|
|
|
ret = copy_file(hashname, output_file);
|
2002-04-08 05:53:14 +02:00
|
|
|
}
|
2002-04-01 08:58:31 +02:00
|
|
|
}
|
2002-03-28 06:48:39 +01:00
|
|
|
|
|
|
|
/* the hash file might have been deleted by some external process */
|
|
|
|
if (ret == -1 && errno == ENOENT) {
|
|
|
|
cc_log("hashfile missing for %s\n", output_file);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_MISSING);
|
2002-03-28 06:48:39 +01:00
|
|
|
close(fd_stderr);
|
|
|
|
unlink(stderr_file);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
free(stderr_file);
|
|
|
|
|
2002-03-28 22:38:41 +01:00
|
|
|
if (ret == -1) {
|
2002-03-27 06:40:28 +01:00
|
|
|
ret = copy_file(hashname, output_file);
|
2002-03-28 22:38:41 +01:00
|
|
|
if (ret == -1) {
|
2002-03-27 06:40:28 +01:00
|
|
|
cc_log("failed to copy %s -> %s (%s)\n",
|
|
|
|
hashname, output_file, strerror(errno));
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_ERROR);
|
2002-03-27 06:40:28 +01:00
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 23:25:14 +01:00
|
|
|
}
|
|
|
|
if (ret == 0) {
|
2002-03-27 10:46:40 +01:00
|
|
|
/* update the mtime on the file so that make doesn't get confused */
|
2002-03-26 23:25:14 +01:00
|
|
|
utime(output_file, NULL);
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
/* get rid of the intermediate preprocessor file */
|
|
|
|
if (i_tmpfile) {
|
|
|
|
unlink(i_tmpfile);
|
|
|
|
free(i_tmpfile);
|
|
|
|
i_tmpfile = NULL;
|
|
|
|
}
|
|
|
|
|
2002-05-12 11:51:36 +02:00
|
|
|
/* send the cpp stderr, if applicable */
|
|
|
|
fd_cpp_stderr = open(cpp_stderr, O_RDONLY);
|
|
|
|
if (fd_cpp_stderr != -1) {
|
|
|
|
copy_fd(fd_cpp_stderr, 2);
|
|
|
|
close(fd_cpp_stderr);
|
|
|
|
unlink(cpp_stderr);
|
|
|
|
free(cpp_stderr);
|
|
|
|
cpp_stderr = NULL;
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
/* send the stderr */
|
|
|
|
copy_fd(fd_stderr, 2);
|
|
|
|
close(fd_stderr);
|
|
|
|
|
|
|
|
/* and exit with the right status code */
|
|
|
|
if (first) {
|
2002-03-27 01:39:06 +01:00
|
|
|
cc_log("got cached result for %s\n", output_file);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_CACHED);
|
2002-03-26 16:11:48 +01:00
|
|
|
}
|
|
|
|
|
2002-03-27 01:39:06 +01:00
|
|
|
exit(0);
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/* find the real compiler. We just search the PATH to find a executable of the
|
|
|
|
same name that isn't a link to ourselves */
|
2002-03-30 01:24:23 +01:00
|
|
|
static void find_compiler(int argc, char **argv)
|
2002-03-26 15:46:43 +01:00
|
|
|
{
|
|
|
|
char *base;
|
2003-02-16 03:28:35 +01:00
|
|
|
char *path;
|
2002-03-26 23:39:25 +01:00
|
|
|
|
2003-02-16 03:28:35 +01:00
|
|
|
orig_args = args_init(argc, argv);
|
2002-03-27 02:29:53 +01:00
|
|
|
|
2002-03-30 01:24:23 +01:00
|
|
|
base = basename(argv[0]);
|
|
|
|
|
|
|
|
/* we might be being invoked like "ccache gcc -c foo.c" */
|
|
|
|
if (strcmp(base, MYNAME) == 0) {
|
2003-02-16 03:28:35 +01:00
|
|
|
args_remove_first(orig_args);
|
2002-03-30 01:24:23 +01:00
|
|
|
free(base);
|
2002-04-03 16:45:51 +02:00
|
|
|
if (strchr(argv[1],'/')) {
|
|
|
|
/* a full path was given */
|
|
|
|
return;
|
|
|
|
}
|
2002-03-30 01:24:23 +01:00
|
|
|
base = basename(argv[1]);
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
2003-01-06 05:25:13 +01:00
|
|
|
/* support user override of the compiler */
|
|
|
|
if ((path=getenv("CCACHE_CC"))) {
|
|
|
|
base = strdup(path);
|
|
|
|
}
|
|
|
|
|
2003-02-16 03:28:35 +01:00
|
|
|
orig_args->argv[0] = find_executable(base, MYNAME);
|
2002-03-30 01:24:23 +01:00
|
|
|
|
|
|
|
/* can't find the compiler! */
|
2003-02-16 14:31:13 +01:00
|
|
|
if (!orig_args->argv[0]) {
|
|
|
|
stats_update(STATS_COMPILER);
|
2003-02-16 03:28:35 +01:00
|
|
|
perror(base);
|
|
|
|
exit(1);
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
/* check a filename for C/C++ extension. Return the pre-processor
|
|
|
|
extension */
|
2003-01-06 03:10:15 +01:00
|
|
|
static const char *check_extension(const char *fname)
|
2002-03-31 16:15:06 +02:00
|
|
|
{
|
|
|
|
int i;
|
2003-01-06 03:10:15 +01:00
|
|
|
const char *p;
|
2002-03-31 16:15:06 +02:00
|
|
|
|
|
|
|
p = strrchr(fname, '.');
|
2002-04-07 15:20:53 +02:00
|
|
|
if (!p) return NULL;
|
2002-03-31 16:15:06 +02:00
|
|
|
p++;
|
2002-04-07 15:20:53 +02:00
|
|
|
for (i=0; extensions[i].extension; i++) {
|
|
|
|
if (strcmp(p, extensions[i].extension) == 0) {
|
2002-06-03 05:51:17 +02:00
|
|
|
p = getenv("CCACHE_EXTENSION");
|
|
|
|
if (p) return p;
|
2002-04-07 15:20:53 +02:00
|
|
|
return extensions[i].i_extension;
|
|
|
|
}
|
2002-03-31 16:15:06 +02:00
|
|
|
}
|
2002-04-07 15:20:53 +02:00
|
|
|
return NULL;
|
2002-03-31 16:15:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/*
|
|
|
|
process the compiler options to form the correct set of options
|
|
|
|
for obtaining the preprocessor output
|
|
|
|
*/
|
2002-03-26 15:46:43 +01:00
|
|
|
static void process_args(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int found_c_opt = 0;
|
2002-03-27 05:53:05 +01:00
|
|
|
int found_S_opt = 0;
|
2002-03-28 07:21:20 +01:00
|
|
|
struct stat st;
|
2003-02-16 03:28:35 +01:00
|
|
|
char *e;
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2003-02-16 03:28:35 +01:00
|
|
|
stripped_args = args_init(0, NULL);
|
2002-03-26 15:46:43 +01:00
|
|
|
|
|
|
|
args_add(stripped_args, argv[0]);
|
|
|
|
|
|
|
|
for (i=1; i<argc; i++) {
|
2002-03-27 08:31:55 +01:00
|
|
|
/* some options will never work ... */
|
2003-01-06 04:45:06 +01:00
|
|
|
if (strcmp(argv[i], "-E") == 0) {
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2003-02-17 02:09:12 +01:00
|
|
|
/* these are too hard */
|
2003-09-24 07:57:54 +02:00
|
|
|
if (strcmp(argv[i], "-fbranch-probabilities")==0 ||
|
2003-02-17 02:09:12 +01:00
|
|
|
strcmp(argv[i], "-M") == 0 ||
|
|
|
|
strcmp(argv[i], "-x") == 0) {
|
2002-04-24 14:07:37 +02:00
|
|
|
cc_log("argument %s is unsupported\n", argv[i]);
|
|
|
|
stats_update(STATS_UNSUPPORTED);
|
|
|
|
failed();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/* we must have -c */
|
2002-03-26 15:46:43 +01:00
|
|
|
if (strcmp(argv[i], "-c") == 0) {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
found_c_opt = 1;
|
|
|
|
continue;
|
|
|
|
}
|
2002-03-27 05:53:05 +01:00
|
|
|
|
|
|
|
/* -S changes the default extension */
|
|
|
|
if (strcmp(argv[i], "-S") == 0) {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
found_S_opt = 1;
|
|
|
|
continue;
|
|
|
|
}
|
2002-03-27 00:58:31 +01:00
|
|
|
|
|
|
|
/* we need to work out where the output was meant to go */
|
2002-03-26 15:46:43 +01:00
|
|
|
if (strcmp(argv[i], "-o") == 0) {
|
|
|
|
if (i == argc-1) {
|
|
|
|
cc_log("missing argument to %s\n", argv[i]);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_ARGS);
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
output_file = argv[i+1];
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
2002-04-10 09:22:59 +02:00
|
|
|
|
|
|
|
/* alternate form of -o, with no space */
|
|
|
|
if (strncmp(argv[i], "-o", 2) == 0) {
|
|
|
|
output_file = &argv[i][2];
|
|
|
|
continue;
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/* debugging is handled specially, so that we know if we
|
|
|
|
can strip line number info
|
|
|
|
*/
|
2002-03-26 15:46:43 +01:00
|
|
|
if (strncmp(argv[i], "-g", 2) == 0) {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
if (strcmp(argv[i], "-g0") != 0) {
|
|
|
|
found_debug = 1;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2003-02-08 05:27:02 +01:00
|
|
|
/* The user knows best: just swallow the next arg */
|
|
|
|
if (strcmp(argv[i], "--ccache-skip") == 0) {
|
|
|
|
i++;
|
|
|
|
if (i == argc) {
|
|
|
|
failed();
|
|
|
|
}
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
/* options that take an argument */
|
2002-06-03 06:34:56 +02:00
|
|
|
{
|
2003-01-09 22:40:05 +01:00
|
|
|
const char *opts[] = {"-I", "-include", "-imacros", "-iprefix",
|
2002-06-03 06:34:56 +02:00
|
|
|
"-iwithprefix", "-iwithprefixbefore",
|
2003-01-06 01:20:54 +01:00
|
|
|
"-L", "-D", "-U", "-x", "-MF",
|
2002-06-03 06:34:56 +02:00
|
|
|
"-MT", "-MQ", "-isystem", "-aux-info",
|
|
|
|
"--param", "-A", "-Xlinker", "-u",
|
|
|
|
NULL};
|
|
|
|
int j;
|
|
|
|
for (j=0;opts[j];j++) {
|
|
|
|
if (strcmp(argv[i], opts[j]) == 0) {
|
|
|
|
if (i == argc-1) {
|
|
|
|
cc_log("missing argument to %s\n",
|
|
|
|
argv[i]);
|
|
|
|
stats_update(STATS_ARGS);
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
|
2002-06-03 06:34:56 +02:00
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
args_add(stripped_args, argv[i+1]);
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (opts[j]) continue;
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
2002-03-27 00:58:31 +01:00
|
|
|
|
|
|
|
/* other options */
|
2002-03-26 15:46:43 +01:00
|
|
|
if (argv[i][0] == '-') {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2002-03-28 07:21:20 +01:00
|
|
|
/* if an argument isn't a plain file then assume its
|
|
|
|
an option, not an input file. This allows us to
|
|
|
|
cope better with unusual compiler options */
|
|
|
|
if (stat(argv[i], &st) != 0 || !S_ISREG(st.st_mode)) {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
if (input_file) {
|
2002-04-08 02:43:58 +02:00
|
|
|
if (check_extension(argv[i])) {
|
|
|
|
cc_log("multiple input files (%s and %s)\n",
|
|
|
|
input_file, argv[i]);
|
|
|
|
stats_update(STATS_MULTIPLE);
|
2002-04-09 14:21:19 +02:00
|
|
|
} else if (!found_c_opt) {
|
|
|
|
cc_log("called for link with %s\n", argv[i]);
|
2002-04-10 10:31:12 +02:00
|
|
|
if (strstr(argv[i], "conftest.")) {
|
|
|
|
stats_update(STATS_CONFTEST);
|
|
|
|
} else {
|
|
|
|
stats_update(STATS_LINK);
|
|
|
|
}
|
2002-04-08 02:43:58 +02:00
|
|
|
} else {
|
|
|
|
cc_log("non C/C++ file %s\n", argv[i]);
|
|
|
|
stats_update(STATS_NOTC);
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
input_file = argv[i];
|
|
|
|
}
|
|
|
|
|
2002-03-26 16:11:48 +01:00
|
|
|
if (!input_file) {
|
|
|
|
cc_log("No input file found\n");
|
2002-04-01 04:59:06 +02:00
|
|
|
stats_update(STATS_NOINPUT);
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-04-07 15:20:53 +02:00
|
|
|
i_extension = check_extension(input_file);
|
|
|
|
if (i_extension == NULL) {
|
2002-03-31 16:15:06 +02:00
|
|
|
cc_log("Not a C/C++ file - %s\n", input_file);
|
2002-04-01 04:59:06 +02:00
|
|
|
stats_update(STATS_NOTC);
|
2002-03-31 16:15:06 +02:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 16:11:48 +01:00
|
|
|
if (!found_c_opt) {
|
|
|
|
cc_log("No -c option found for %s\n", input_file);
|
2002-04-10 10:02:34 +02:00
|
|
|
/* I find that having a separate statistic for autoconf tests is useful,
|
|
|
|
as they are the dominant form of "called for link" in many cases */
|
2002-04-10 09:52:52 +02:00
|
|
|
if (strstr(input_file, "conftest.")) {
|
|
|
|
stats_update(STATS_CONFTEST);
|
|
|
|
} else {
|
|
|
|
stats_update(STATS_LINK);
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2003-01-07 06:06:07 +01:00
|
|
|
|
|
|
|
/* don't try to second guess the compilers heuristics for stdout handling */
|
2003-01-07 22:16:50 +01:00
|
|
|
if (output_file && strcmp(output_file, "-") == 0) {
|
2003-01-07 06:06:07 +01:00
|
|
|
stats_update(STATS_OUTSTDOUT);
|
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
if (!output_file) {
|
|
|
|
char *p;
|
2002-03-27 00:58:31 +01:00
|
|
|
output_file = x_strdup(input_file);
|
2002-03-26 15:46:43 +01:00
|
|
|
if ((p = strrchr(output_file, '/'))) {
|
|
|
|
output_file = p+1;
|
|
|
|
}
|
|
|
|
p = strrchr(output_file, '.');
|
|
|
|
if (!p || !p[1]) {
|
|
|
|
cc_log("badly formed output_file %s\n", output_file);
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_ARGS);
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
2002-03-27 05:53:05 +01:00
|
|
|
p[1] = found_S_opt ? 's' : 'o';
|
2002-03-26 15:46:43 +01:00
|
|
|
p[2] = 0;
|
|
|
|
}
|
2002-03-31 04:57:44 +02:00
|
|
|
|
|
|
|
/* cope with -o /dev/null */
|
2002-04-08 05:53:14 +02:00
|
|
|
if (strcmp(output_file,"/dev/null") != 0 && stat(output_file, &st) == 0 && !S_ISREG(st.st_mode)) {
|
2002-03-31 04:57:44 +02:00
|
|
|
cc_log("Not a regular file %s\n", output_file);
|
2002-04-01 04:59:06 +02:00
|
|
|
stats_update(STATS_DEVICE);
|
2002-03-31 04:57:44 +02:00
|
|
|
failed();
|
|
|
|
}
|
2003-02-16 03:28:35 +01:00
|
|
|
|
|
|
|
if ((e=getenv("CCACHE_PREFIX"))) {
|
|
|
|
char *p = find_executable(e, MYNAME);
|
|
|
|
if (!p) {
|
|
|
|
perror(e);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
args_add_prefix(stripped_args, p);
|
|
|
|
}
|
2002-03-26 15:46:43 +01:00
|
|
|
}
|
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/* the main ccache driver function */
|
2002-03-26 15:46:43 +01:00
|
|
|
static void ccache(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
/* find the real compiler */
|
2002-03-30 01:24:23 +01:00
|
|
|
find_compiler(argc, argv);
|
|
|
|
|
|
|
|
/* we might be disabled */
|
2002-03-27 07:15:32 +01:00
|
|
|
if (getenv("CCACHE_DISABLE")) {
|
|
|
|
cc_log("ccache is disabled\n");
|
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
/* process argument list, returning a new set of arguments for pre-processing */
|
2002-03-30 01:24:23 +01:00
|
|
|
process_args(orig_args->argc, orig_args->argv);
|
2002-03-26 15:46:43 +01:00
|
|
|
|
|
|
|
/* run with -E to find the hash */
|
|
|
|
find_hash(stripped_args);
|
|
|
|
|
|
|
|
/* if we can return from cache at this point then do */
|
|
|
|
from_cache(1);
|
|
|
|
|
2002-05-12 11:51:36 +02:00
|
|
|
/* run real compiler, sending output to cache */
|
2002-03-26 15:46:43 +01:00
|
|
|
to_cache(stripped_args);
|
|
|
|
|
|
|
|
/* return from cache */
|
|
|
|
from_cache(0);
|
|
|
|
|
|
|
|
/* oh oh! */
|
|
|
|
cc_log("secondary from_cache failed!\n");
|
2002-03-30 11:43:26 +01:00
|
|
|
stats_update(STATS_ERROR);
|
2002-03-26 15:46:43 +01:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-03-30 01:24:23 +01:00
|
|
|
static void usage(void)
|
|
|
|
{
|
2002-03-30 13:35:26 +01:00
|
|
|
printf("ccache, a compiler cache. Version %s\n", CCACHE_VERSION);
|
2002-03-30 11:43:26 +01:00
|
|
|
printf("Copyright Andrew Tridgell, 2002\n\n");
|
|
|
|
|
|
|
|
printf("Usage:\n");
|
|
|
|
printf("\tccache [options]\n");
|
|
|
|
printf("\tccache compiler [compile options]\n");
|
|
|
|
printf("\tcompiler [compile options] (via symbolic link)\n");
|
|
|
|
printf("\nOptions:\n");
|
|
|
|
|
|
|
|
printf("-s show statistics summary\n");
|
2002-03-30 14:26:42 +01:00
|
|
|
printf("-z zero statistics\n");
|
2002-03-30 11:43:26 +01:00
|
|
|
printf("-c run a cache cleanup\n");
|
2002-06-03 05:22:17 +02:00
|
|
|
printf("-C clear the cache completely\n");
|
2002-03-30 11:43:26 +01:00
|
|
|
printf("-F <maxfiles> set maximum files in cache\n");
|
|
|
|
printf("-M <maxsize> set maximum size of cache (use G, M or K)\n");
|
2002-03-30 13:35:26 +01:00
|
|
|
printf("-h this help page\n");
|
|
|
|
printf("-V print version number\n");
|
2002-03-30 01:24:23 +01:00
|
|
|
}
|
|
|
|
|
2002-03-30 11:43:26 +01:00
|
|
|
/* the main program when not doing a compile */
|
2002-03-30 01:24:23 +01:00
|
|
|
static int ccache_main(int argc, char *argv[])
|
|
|
|
{
|
2002-03-30 11:43:26 +01:00
|
|
|
extern int optind;
|
|
|
|
int c;
|
|
|
|
size_t v;
|
|
|
|
|
2002-06-03 05:22:17 +02:00
|
|
|
while ((c = getopt(argc, argv, "hszcCF:M:V")) != -1) {
|
2002-03-30 11:43:26 +01:00
|
|
|
switch (c) {
|
2002-03-30 13:35:26 +01:00
|
|
|
case 'V':
|
|
|
|
printf("ccache version %s\n", CCACHE_VERSION);
|
2002-03-31 11:30:27 +02:00
|
|
|
printf("Copyright Andrew Tridgell 2002\n");
|
|
|
|
printf("Released under the GNU GPL v2 or later\n");
|
2002-03-30 13:35:26 +01:00
|
|
|
exit(0);
|
|
|
|
|
2002-03-30 11:43:26 +01:00
|
|
|
case 'h':
|
|
|
|
usage();
|
|
|
|
exit(0);
|
|
|
|
|
|
|
|
case 's':
|
|
|
|
stats_summary();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'c':
|
|
|
|
cleanup_all(cache_dir);
|
2003-02-16 14:31:13 +01:00
|
|
|
printf("Cleaned cache\n");
|
2002-03-30 11:43:26 +01:00
|
|
|
break;
|
|
|
|
|
2002-06-03 05:22:17 +02:00
|
|
|
case 'C':
|
|
|
|
wipe_all(cache_dir);
|
|
|
|
printf("Cleared cache\n");
|
|
|
|
break;
|
|
|
|
|
2002-03-30 11:43:26 +01:00
|
|
|
case 'z':
|
|
|
|
stats_zero();
|
|
|
|
printf("Statistics cleared\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'F':
|
|
|
|
v = atoi(optarg);
|
|
|
|
stats_set_limits(v, -1);
|
|
|
|
printf("Set cache file limit to %u\n", (unsigned)v);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'M':
|
|
|
|
v = value_units(optarg);
|
|
|
|
stats_set_limits(-1, v);
|
|
|
|
printf("Set cache size limit to %uk\n", (unsigned)v);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2002-03-30 01:24:23 +01:00
|
|
|
}
|
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2002-03-27 00:58:31 +01:00
|
|
|
cache_dir = getenv("CCACHE_DIR");
|
2002-03-27 04:53:22 +01:00
|
|
|
if (!cache_dir) {
|
|
|
|
x_asprintf(&cache_dir, "%s/.ccache", getenv("HOME"));
|
|
|
|
}
|
2002-03-27 00:58:31 +01:00
|
|
|
|
|
|
|
cache_logfile = getenv("CCACHE_LOGFILE");
|
|
|
|
|
2002-03-30 01:24:23 +01:00
|
|
|
/* check if we are being invoked as "ccache" */
|
|
|
|
if (strlen(argv[0]) >= strlen(MYNAME) &&
|
|
|
|
strcmp(argv[0] + strlen(argv[0]) - strlen(MYNAME), MYNAME) == 0) {
|
|
|
|
if (argc < 2) {
|
|
|
|
usage();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
/* if the first argument isn't an option, then assume we are
|
|
|
|
being passed a compiler name and options */
|
|
|
|
if (argv[1][0] == '-') {
|
|
|
|
return ccache_main(argc, argv);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-27 00:58:31 +01:00
|
|
|
/* make sure the cache dir exists */
|
2002-03-26 23:25:14 +01:00
|
|
|
if (create_dir(cache_dir) != 0) {
|
|
|
|
fprintf(stderr,"ccache: failed to create %s (%s)\n",
|
2002-03-26 15:46:43 +01:00
|
|
|
cache_dir, strerror(errno));
|
|
|
|
exit(1);
|
|
|
|
}
|
2002-03-27 00:58:31 +01:00
|
|
|
|
2002-03-26 15:46:43 +01:00
|
|
|
ccache(argc, argv);
|
|
|
|
return 1;
|
|
|
|
}
|