2002-03-26 14:46:43 +00: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-26 23:58:31 +00: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 14:46:43 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ccache.h"
|
|
|
|
|
2002-03-30 10:43:26 +00:00
|
|
|
char *cache_dir = NULL;
|
2002-03-26 23:58:31 +00:00
|
|
|
char *cache_logfile = NULL;
|
2002-03-26 14:46:43 +00:00
|
|
|
static ARGS *stripped_args;
|
|
|
|
static ARGS *orig_args;
|
|
|
|
static char *output_file;
|
|
|
|
static char *hashname;
|
2002-03-30 10:43:26 +00:00
|
|
|
char *stats_file = NULL;
|
2002-03-26 14:46:43 +00:00
|
|
|
static int found_debug;
|
|
|
|
|
2002-03-26 23:58:31 +00:00
|
|
|
/*
|
|
|
|
something went badly wrong - just execute the real compiler
|
|
|
|
*/
|
2002-03-26 14:46:43 +00:00
|
|
|
static void failed(void)
|
|
|
|
{
|
|
|
|
execv(orig_args->argv[0], orig_args->argv);
|
|
|
|
cc_log("execv returned (%s)!\n", strerror(errno));
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* run the real compiler and put the result in cache */
|
|
|
|
static void to_cache(ARGS *args)
|
|
|
|
{
|
2002-03-27 00:39:06 +00:00
|
|
|
char *path_stderr;
|
|
|
|
char *tmp_stdout, *tmp_stderr, *tmp_hashname;
|
2002-03-30 10:43:26 +00:00
|
|
|
struct stat st1, st2;
|
2002-03-27 00:39:06 +00:00
|
|
|
int status;
|
2002-03-26 14:46:43 +00:00
|
|
|
|
2002-03-27 00:39:06 +00:00
|
|
|
x_asprintf(&tmp_stdout, "%s/tmp.stdout.%d", cache_dir, getpid());
|
|
|
|
x_asprintf(&tmp_stderr, "%s/tmp.stderr.%d", cache_dir, getpid());
|
2002-03-27 01:57:45 +00:00
|
|
|
x_asprintf(&tmp_hashname, "%s/tmp.hash.%d.o", cache_dir, getpid());
|
2002-03-26 14:46:43 +00:00
|
|
|
|
|
|
|
args_add(args, "-o");
|
2002-03-27 00:39:06 +00:00
|
|
|
args_add(args, tmp_hashname);
|
|
|
|
status = execute(args->argv, tmp_stdout, tmp_stderr);
|
2002-03-27 08:40:13 +00:00
|
|
|
args_pop(args, 2);
|
2002-03-26 14:46:43 +00:00
|
|
|
|
2002-03-30 10:43:26 +00:00
|
|
|
if (stat(tmp_stdout, &st1) != 0 || st1.st_size != 0) {
|
2002-03-27 01:29:53 +00:00
|
|
|
cc_log("compiler produced stdout for %s\n", output_file);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_STDOUT);
|
2002-03-27 00:39:06 +00:00
|
|
|
unlink(tmp_stdout);
|
|
|
|
unlink(tmp_stderr);
|
|
|
|
unlink(tmp_hashname);
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-27 01:29:53 +00:00
|
|
|
unlink(tmp_stdout);
|
2002-03-26 14:46:43 +00:00
|
|
|
|
2002-03-27 01:29:53 +00:00
|
|
|
if (status != 0) {
|
|
|
|
int fd;
|
|
|
|
cc_log("compile of %s gave status = %d\n", output_file, status);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_STATUS);
|
2002-03-27 01:29:53 +00:00
|
|
|
|
|
|
|
fd = open(tmp_stderr, O_RDONLY);
|
|
|
|
if (fd != -1 &&
|
|
|
|
(rename(tmp_hashname, output_file) == 0 || errno == ENOENT)) {
|
|
|
|
/* we can use a quick method of getting the failed output */
|
|
|
|
copy_fd(fd, 2);
|
|
|
|
close(fd);
|
|
|
|
unlink(tmp_stderr);
|
|
|
|
exit(status);
|
|
|
|
}
|
|
|
|
|
2002-03-27 00:39:06 +00:00
|
|
|
unlink(tmp_stderr);
|
|
|
|
unlink(tmp_hashname);
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 14:46:43 +00:00
|
|
|
|
2002-03-27 00:39:06 +00:00
|
|
|
x_asprintf(&path_stderr, "%s.stderr", hashname);
|
|
|
|
|
2002-03-30 10:43:26 +00:00
|
|
|
if (stat(tmp_stderr, &st1) != 0 ||
|
|
|
|
stat(tmp_hashname, &st2) != 0 ||
|
|
|
|
rename(tmp_hashname, hashname) != 0 ||
|
2002-03-27 00:39:06 +00:00
|
|
|
rename(tmp_stderr, path_stderr) != 0) {
|
|
|
|
cc_log("failed to rename tmp files\n");
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_ERROR);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
cc_log("Placed %s into cache\n", output_file);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_tocache(file_size(&st1) + file_size(&st2));
|
2002-03-27 00:39:06 +00:00
|
|
|
|
|
|
|
free(tmp_hashname);
|
|
|
|
free(tmp_stderr);
|
|
|
|
free(tmp_stdout);
|
|
|
|
free(path_stderr);
|
2002-03-26 14:46:43 +00: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 00:39:06 +00:00
|
|
|
char *path_stdout, *path_stderr;
|
2002-03-26 15:11:48 +00:00
|
|
|
char *hash_dir;
|
|
|
|
char *s;
|
2002-03-26 22:25:14 +00:00
|
|
|
struct stat st;
|
2002-03-27 00:39:06 +00:00
|
|
|
int status;
|
2002-03-26 14:46:43 +00:00
|
|
|
|
|
|
|
hash_start();
|
|
|
|
|
|
|
|
/* first the arguments */
|
|
|
|
for (i=0;i<args->argc;i++) {
|
2002-03-27 06:30:31 +00: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;
|
|
|
|
}
|
|
|
|
if (strncmp(args->argv[i], "-I", 2) == 0 ||
|
|
|
|
strncmp(args->argv[i], "-L", 2) == 0 ||
|
2002-03-27 08:02:24 +00:00
|
|
|
strncmp(args->argv[i], "-D", 2) == 0 ||
|
|
|
|
strncmp(args->argv[i], "-isystem", 8) == 0) {
|
2002-03-27 06:30:31 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-26 14:46:43 +00:00
|
|
|
hash_string(args->argv[i]);
|
|
|
|
}
|
|
|
|
|
2002-03-26 22:25:14 +00: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) {
|
|
|
|
cc_log("Couldn't stat the compiler!?\n");
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_COMPILER);
|
2002-03-26 22:25:14 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
hash_int(st.st_size);
|
|
|
|
hash_int(st.st_mtime);
|
|
|
|
|
2002-03-26 14:46:43 +00:00
|
|
|
/* now the run */
|
|
|
|
x_asprintf(&path_stdout, "%s/tmp.stdout.%d", cache_dir, getpid());
|
|
|
|
x_asprintf(&path_stderr, "%s/tmp.stderr.%d", cache_dir, getpid());
|
|
|
|
|
|
|
|
args_add(args, "-E");
|
2002-03-27 00:39:06 +00:00
|
|
|
status = execute(args->argv, path_stdout, path_stderr);
|
2002-03-27 08:40:13 +00:00
|
|
|
args_pop(args, 1);
|
2002-03-26 14:46:43 +00:00
|
|
|
|
2002-03-27 00:39:06 +00:00
|
|
|
if (status != 0) {
|
|
|
|
unlink(path_stdout);
|
|
|
|
unlink(path_stderr);
|
|
|
|
cc_log("the preprocessor gave %d\n", status);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_PREPROCESSOR);
|
2002-03-27 00:39:06 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 22:25:14 +00: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-01 00:23:31 +00:00
|
|
|
if (found_debug || getenv("CCACHE_NOUNIFY")) {
|
2002-03-26 14:46:43 +00:00
|
|
|
hash_file(path_stdout);
|
|
|
|
} else {
|
2002-04-01 00:23:31 +00:00
|
|
|
if (unify_hash(path_stdout) != 0) {
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
|
|
|
hash_file(path_stderr);
|
|
|
|
|
|
|
|
unlink(path_stdout);
|
|
|
|
unlink(path_stderr);
|
|
|
|
free(path_stdout);
|
|
|
|
free(path_stderr);
|
|
|
|
|
2002-03-26 23:58:31 +00:00
|
|
|
/* we use a single level subdir for the cache path to reduce the impact
|
|
|
|
on filesystems which are slow for large directories
|
|
|
|
*/
|
2002-03-26 15:11:48 +00:00
|
|
|
s = hash_result();
|
|
|
|
x_asprintf(&hash_dir, "%s/%c", cache_dir, *s);
|
2002-03-26 22:25:14 +00:00
|
|
|
if (create_dir(hash_dir) != 0) {
|
|
|
|
cc_log("failed to create %s\n", cache_dir);
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 15:11:48 +00:00
|
|
|
x_asprintf(&hashname, "%s/%s", hash_dir, s+1);
|
2002-03-30 10:43:26 +00:00
|
|
|
x_asprintf(&stats_file, "%s/stats", hash_dir);
|
2002-03-26 15:11:48 +00:00
|
|
|
free(hash_dir);
|
2002-03-26 14:46:43 +00: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-03-27 00:39:06 +00:00
|
|
|
int fd_stderr;
|
2002-03-28 04:22:40 +00:00
|
|
|
char *stderr_file;
|
2002-03-27 00:39:06 +00:00
|
|
|
int ret;
|
2002-03-30 10:43:26 +00:00
|
|
|
struct stat st;
|
2002-03-26 14:46:43 +00:00
|
|
|
|
2002-03-28 04:22:40 +00:00
|
|
|
x_asprintf(&stderr_file, "%s.stderr", hashname);
|
|
|
|
fd_stderr = open(stderr_file, O_RDONLY);
|
2002-03-26 14:46:43 +00:00
|
|
|
if (fd_stderr == -1) {
|
2002-03-27 00:39:06 +00:00
|
|
|
/* it isn't in cache ... */
|
2002-03-28 04:22:40 +00:00
|
|
|
free(stderr_file);
|
2002-03-26 14:46:43 +00:00
|
|
|
return;
|
|
|
|
}
|
2002-03-30 10:43:26 +00:00
|
|
|
|
|
|
|
/* make sure the output is there too */
|
|
|
|
if (stat(hashname, &st) != 0) {
|
|
|
|
close(fd_stderr);
|
|
|
|
unlink(stderr_file);
|
|
|
|
free(stderr_file);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2002-03-28 04:22:40 +00:00
|
|
|
utime(stderr_file, NULL);
|
2002-03-26 14:46:43 +00:00
|
|
|
|
|
|
|
unlink(output_file);
|
2002-04-01 06:58:31 +00:00
|
|
|
if (getenv("CCACHE_NOLINK")) {
|
|
|
|
ret = copy_file(hashname, output_file);
|
|
|
|
} else {
|
|
|
|
ret = link(hashname, output_file);
|
|
|
|
}
|
2002-03-28 05:48:39 +00: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 10:43:26 +00:00
|
|
|
stats_update(STATS_MISSING);
|
2002-03-28 05:48:39 +00:00
|
|
|
close(fd_stderr);
|
|
|
|
unlink(stderr_file);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
free(stderr_file);
|
|
|
|
|
2002-03-28 21:38:41 +00:00
|
|
|
if (ret == -1) {
|
2002-03-27 05:40:28 +00:00
|
|
|
ret = copy_file(hashname, output_file);
|
2002-03-28 21:38:41 +00:00
|
|
|
if (ret == -1) {
|
2002-03-27 05:40:28 +00:00
|
|
|
cc_log("failed to copy %s -> %s (%s)\n",
|
|
|
|
hashname, output_file, strerror(errno));
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_ERROR);
|
2002-03-27 05:40:28 +00:00
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 22:25:14 +00:00
|
|
|
}
|
|
|
|
if (ret == 0) {
|
2002-03-27 09:46:40 +00:00
|
|
|
/* update the mtime on the file so that make doesn't get confused */
|
2002-03-26 22:25:14 +00:00
|
|
|
utime(output_file, NULL);
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* send the stderr */
|
|
|
|
copy_fd(fd_stderr, 2);
|
|
|
|
close(fd_stderr);
|
|
|
|
|
|
|
|
/* and exit with the right status code */
|
|
|
|
if (first) {
|
2002-03-27 00:39:06 +00:00
|
|
|
cc_log("got cached result for %s\n", output_file);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_CACHED);
|
2002-03-26 15:11:48 +00:00
|
|
|
}
|
|
|
|
|
2002-03-27 00:39:06 +00:00
|
|
|
exit(0);
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
|
|
|
|
2002-03-26 23:58:31 +00: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 00:24:23 +00:00
|
|
|
static void find_compiler(int argc, char **argv)
|
2002-03-26 14:46:43 +00:00
|
|
|
{
|
|
|
|
char *base;
|
|
|
|
char *path, *tok;
|
|
|
|
struct stat st1, st2;
|
|
|
|
|
2002-03-30 00:24:23 +00:00
|
|
|
orig_args = args_init();
|
2002-04-01 03:58:57 +00:00
|
|
|
free(orig_args->argv);
|
2002-03-26 22:39:25 +00:00
|
|
|
|
2002-03-30 00:24:23 +00:00
|
|
|
orig_args->argv = argv;
|
|
|
|
orig_args->argc = argc;
|
2002-03-27 01:29:53 +00:00
|
|
|
|
2002-03-30 00:24:23 +00:00
|
|
|
base = basename(argv[0]);
|
|
|
|
|
|
|
|
/* we might be being invoked like "ccache gcc -c foo.c" */
|
|
|
|
if (strcmp(base, MYNAME) == 0) {
|
|
|
|
orig_args->argv++;
|
|
|
|
orig_args->argc--;
|
|
|
|
free(base);
|
|
|
|
base = basename(argv[1]);
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
|
|
|
|
2002-03-26 22:25:14 +00:00
|
|
|
path = getenv("CCACHE_PATH");
|
|
|
|
if (!path) {
|
|
|
|
path = getenv("PATH");
|
|
|
|
}
|
|
|
|
if (!path) {
|
|
|
|
cc_log("no PATH variable!?\n");
|
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 14:46:43 +00:00
|
|
|
|
|
|
|
path = x_strdup(path);
|
|
|
|
|
2002-03-30 00:24:23 +00:00
|
|
|
/* search the path looking for the first compiler of the right name
|
2002-03-26 14:46:43 +00:00
|
|
|
that isn't us */
|
|
|
|
for (tok=strtok(path,":"); tok; tok = strtok(NULL, ":")) {
|
|
|
|
char *fname;
|
|
|
|
x_asprintf(&fname, "%s/%s", tok, base);
|
2002-03-26 22:39:25 +00:00
|
|
|
/* look for a normal executable file */
|
|
|
|
if (access(fname, X_OK) == 0 &&
|
2002-03-30 00:24:23 +00:00
|
|
|
lstat(fname, &st1) == 0 &&
|
2002-03-26 22:39:25 +00:00
|
|
|
stat(fname, &st2) == 0 &&
|
|
|
|
S_ISREG(st2.st_mode)) {
|
2002-03-30 00:24:23 +00:00
|
|
|
/* if its a symlink then ensure it doesn't
|
|
|
|
point at something called "ccache" */
|
|
|
|
if (S_ISLNK(st1.st_mode)) {
|
2002-03-31 05:01:05 +00:00
|
|
|
char *buf = x_realpath(fname);
|
|
|
|
if (buf) {
|
|
|
|
char *p = basename(buf);
|
2002-03-30 00:24:23 +00:00
|
|
|
if (strcmp(p, MYNAME) == 0) {
|
|
|
|
/* its a link to "ccache" ! */
|
|
|
|
free(p);
|
2002-03-31 05:01:05 +00:00
|
|
|
free(buf);
|
2002-03-30 00:24:23 +00:00
|
|
|
continue;
|
|
|
|
}
|
2002-03-31 05:01:05 +00:00
|
|
|
free(buf);
|
2002-03-30 00:24:23 +00:00
|
|
|
free(p);
|
|
|
|
}
|
2002-03-27 01:29:53 +00:00
|
|
|
}
|
|
|
|
|
2002-03-30 00:24:23 +00:00
|
|
|
|
|
|
|
/* found it! */
|
|
|
|
free(path);
|
|
|
|
orig_args->argv[0] = fname;
|
|
|
|
free(base);
|
|
|
|
return;
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
2002-03-30 00:24:23 +00:00
|
|
|
free(fname);
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
2002-03-30 00:24:23 +00:00
|
|
|
|
|
|
|
/* can't find the compiler! */
|
|
|
|
perror(base);
|
|
|
|
exit(1);
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-03-31 14:15:06 +00:00
|
|
|
/* check a filename for C/C++ extension */
|
|
|
|
static int check_extension(const char *fname)
|
|
|
|
{
|
|
|
|
char *extensions[] = {"c", "C", "m", "cc", "CC", "cpp", "CPP", "cxx", "CXX", 0};
|
|
|
|
int i;
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
p = strrchr(fname, '.');
|
|
|
|
if (!p) return -1;
|
|
|
|
p++;
|
|
|
|
for (i=0; extensions[i]; i++) {
|
|
|
|
if (strcmp(p, extensions[i]) == 0) return 0;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-03-26 23:58:31 +00:00
|
|
|
/*
|
|
|
|
process the compiler options to form the correct set of options
|
|
|
|
for obtaining the preprocessor output
|
|
|
|
*/
|
2002-03-26 14:46:43 +00:00
|
|
|
static void process_args(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int found_c_opt = 0;
|
2002-03-27 04:53:05 +00:00
|
|
|
int found_S_opt = 0;
|
2002-03-26 14:46:43 +00:00
|
|
|
char *input_file = NULL;
|
2002-03-28 06:21:20 +00:00
|
|
|
struct stat st;
|
2002-03-26 14:46:43 +00:00
|
|
|
|
|
|
|
stripped_args = args_init();
|
|
|
|
|
|
|
|
args_add(stripped_args, argv[0]);
|
|
|
|
|
|
|
|
for (i=1; i<argc; i++) {
|
2002-03-27 07:31:55 +00:00
|
|
|
/* some options will never work ... */
|
|
|
|
if (strncmp(argv[i], "-E", 2) == 0 ||
|
|
|
|
strncmp(argv[i], "-M", 2) == 0) {
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 23:58:31 +00:00
|
|
|
/* we must have -c */
|
2002-03-26 14:46:43 +00:00
|
|
|
if (strcmp(argv[i], "-c") == 0) {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
found_c_opt = 1;
|
|
|
|
continue;
|
|
|
|
}
|
2002-03-27 04:53:05 +00: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-26 23:58:31 +00:00
|
|
|
|
|
|
|
/* we need to work out where the output was meant to go */
|
2002-03-26 14:46:43 +00:00
|
|
|
if (strcmp(argv[i], "-o") == 0) {
|
|
|
|
if (i == argc-1) {
|
|
|
|
cc_log("missing argument to %s\n", argv[i]);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_ARGS);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
output_file = argv[i+1];
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2002-03-26 23:58:31 +00:00
|
|
|
/* debugging is handled specially, so that we know if we
|
|
|
|
can strip line number info
|
|
|
|
*/
|
2002-03-26 14:46:43 +00:00
|
|
|
if (strncmp(argv[i], "-g", 2) == 0) {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
if (strcmp(argv[i], "-g0") != 0) {
|
|
|
|
found_debug = 1;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* options that take an argument */
|
|
|
|
if (strcmp(argv[i], "-I") == 0 ||
|
|
|
|
strcmp(argv[i], "-include") == 0 ||
|
|
|
|
strcmp(argv[i], "-L") == 0 ||
|
|
|
|
strcmp(argv[i], "-D") == 0 ||
|
|
|
|
strcmp(argv[i], "-isystem") == 0) {
|
|
|
|
if (i == argc-1) {
|
|
|
|
cc_log("missing argument to %s\n", argv[i]);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_ARGS);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
args_add(stripped_args, argv[i+1]);
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
2002-03-26 23:58:31 +00:00
|
|
|
|
|
|
|
/* other options */
|
2002-03-26 14:46:43 +00:00
|
|
|
if (argv[i][0] == '-') {
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2002-03-28 06:21:20 +00: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 14:46:43 +00:00
|
|
|
if (input_file) {
|
|
|
|
cc_log("multiple input files (%s and %s)\n",
|
|
|
|
input_file, argv[i]);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_LINK);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
input_file = argv[i];
|
|
|
|
args_add(stripped_args, argv[i]);
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:11:48 +00:00
|
|
|
if (!input_file) {
|
|
|
|
cc_log("No input file found\n");
|
2002-04-01 02:59:06 +00:00
|
|
|
stats_update(STATS_NOINPUT);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-31 14:15:06 +00:00
|
|
|
if (check_extension(input_file) != 0) {
|
|
|
|
cc_log("Not a C/C++ file - %s\n", input_file);
|
2002-04-01 02:59:06 +00:00
|
|
|
stats_update(STATS_NOTC);
|
2002-03-31 14:15:06 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 15:11:48 +00:00
|
|
|
if (!found_c_opt) {
|
|
|
|
cc_log("No -c option found for %s\n", input_file);
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_LINK);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!output_file) {
|
|
|
|
char *p;
|
2002-03-26 23:58:31 +00:00
|
|
|
output_file = x_strdup(input_file);
|
2002-03-26 14:46:43 +00: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 10:43:26 +00:00
|
|
|
stats_update(STATS_ARGS);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
2002-03-27 04:53:05 +00:00
|
|
|
p[1] = found_S_opt ? 's' : 'o';
|
2002-03-26 14:46:43 +00:00
|
|
|
p[2] = 0;
|
|
|
|
#if 0
|
|
|
|
cc_log("Formed output file %s from input_file %s\n",
|
|
|
|
output_file, input_file);
|
|
|
|
#endif
|
|
|
|
}
|
2002-03-31 02:57:44 +00:00
|
|
|
|
|
|
|
/* cope with -o /dev/null */
|
|
|
|
if (stat(output_file, &st) == 0 && !S_ISREG(st.st_mode)) {
|
|
|
|
cc_log("Not a regular file %s\n", output_file);
|
2002-04-01 02:59:06 +00:00
|
|
|
stats_update(STATS_DEVICE);
|
2002-03-31 02:57:44 +00:00
|
|
|
failed();
|
|
|
|
}
|
2002-03-26 14:46:43 +00:00
|
|
|
}
|
|
|
|
|
2002-03-26 23:58:31 +00:00
|
|
|
/* the main ccache driver function */
|
2002-03-26 14:46:43 +00:00
|
|
|
static void ccache(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
/* find the real compiler */
|
2002-03-30 00:24:23 +00:00
|
|
|
find_compiler(argc, argv);
|
|
|
|
|
|
|
|
/* we might be disabled */
|
2002-03-27 06:15:32 +00:00
|
|
|
if (getenv("CCACHE_DISABLE")) {
|
|
|
|
cc_log("ccache is disabled\n");
|
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
2002-03-26 14:46:43 +00:00
|
|
|
/* process argument list, returning a new set of arguments for pre-processing */
|
2002-03-30 00:24:23 +00:00
|
|
|
process_args(orig_args->argc, orig_args->argv);
|
2002-03-26 14:46:43 +00: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);
|
|
|
|
|
|
|
|
/* run real compiler, semding output to cache */
|
|
|
|
to_cache(stripped_args);
|
|
|
|
|
|
|
|
/* return from cache */
|
|
|
|
from_cache(0);
|
|
|
|
|
|
|
|
/* oh oh! */
|
|
|
|
cc_log("secondary from_cache failed!\n");
|
2002-03-30 10:43:26 +00:00
|
|
|
stats_update(STATS_ERROR);
|
2002-03-26 14:46:43 +00:00
|
|
|
failed();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-03-30 00:24:23 +00:00
|
|
|
static void usage(void)
|
|
|
|
{
|
2002-03-30 12:35:26 +00:00
|
|
|
printf("ccache, a compiler cache. Version %s\n", CCACHE_VERSION);
|
2002-03-30 10:43:26 +00: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 13:26:42 +00:00
|
|
|
printf("-z zero statistics\n");
|
2002-03-30 10:43:26 +00:00
|
|
|
printf("-c run a cache cleanup\n");
|
|
|
|
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 12:35:26 +00:00
|
|
|
printf("-h this help page\n");
|
|
|
|
printf("-V print version number\n");
|
2002-03-30 00:24:23 +00:00
|
|
|
}
|
|
|
|
|
2002-03-30 10:43:26 +00:00
|
|
|
/* the main program when not doing a compile */
|
2002-03-30 00:24:23 +00:00
|
|
|
static int ccache_main(int argc, char *argv[])
|
|
|
|
{
|
2002-03-30 10:43:26 +00:00
|
|
|
extern int optind;
|
|
|
|
int c;
|
|
|
|
size_t v;
|
|
|
|
|
2002-03-30 12:35:26 +00:00
|
|
|
while ((c = getopt(argc, argv, "hszcF:M:V")) != -1) {
|
2002-03-30 10:43:26 +00:00
|
|
|
switch (c) {
|
2002-03-30 12:35:26 +00:00
|
|
|
case 'V':
|
|
|
|
printf("ccache version %s\n", CCACHE_VERSION);
|
2002-03-31 09:30:27 +00:00
|
|
|
printf("Copyright Andrew Tridgell 2002\n");
|
|
|
|
printf("Released under the GNU GPL v2 or later\n");
|
2002-03-30 12:35:26 +00:00
|
|
|
exit(0);
|
|
|
|
|
2002-03-30 10:43:26 +00:00
|
|
|
case 'h':
|
|
|
|
usage();
|
|
|
|
exit(0);
|
|
|
|
|
|
|
|
case 's':
|
|
|
|
stats_summary();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'c':
|
|
|
|
cleanup_all(cache_dir);
|
|
|
|
printf("Cleaned cached\n");
|
|
|
|
break;
|
|
|
|
|
|
|
|
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 00:24:23 +00:00
|
|
|
}
|
|
|
|
|
2002-03-26 14:46:43 +00:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2002-03-26 23:58:31 +00:00
|
|
|
cache_dir = getenv("CCACHE_DIR");
|
2002-03-27 03:53:22 +00:00
|
|
|
if (!cache_dir) {
|
|
|
|
x_asprintf(&cache_dir, "%s/.ccache", getenv("HOME"));
|
|
|
|
}
|
2002-03-26 23:58:31 +00:00
|
|
|
|
|
|
|
cache_logfile = getenv("CCACHE_LOGFILE");
|
|
|
|
|
2002-03-30 00:24:23 +00: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-26 23:58:31 +00:00
|
|
|
/* make sure the cache dir exists */
|
2002-03-26 22:25:14 +00:00
|
|
|
if (create_dir(cache_dir) != 0) {
|
|
|
|
fprintf(stderr,"ccache: failed to create %s (%s)\n",
|
2002-03-26 14:46:43 +00:00
|
|
|
cache_dir, strerror(errno));
|
|
|
|
exit(1);
|
|
|
|
}
|
2002-03-26 23:58:31 +00:00
|
|
|
|
2002-03-26 14:46:43 +00:00
|
|
|
ccache(argc, argv);
|
|
|
|
return 1;
|
|
|
|
}
|