2005-04-20 12:00:09 +08:00
|
|
|
#include "cache.h"
|
|
|
|
|
|
|
|
static int cached_only = 0;
|
|
|
|
static int line_termination = '\n';
|
|
|
|
|
|
|
|
/* A file entry went away or appeared */
|
2005-04-23 08:15:28 +08:00
|
|
|
static void show_file(const char *prefix, struct cache_entry *ce)
|
2005-04-20 12:00:09 +08:00
|
|
|
{
|
2005-04-23 08:15:28 +08:00
|
|
|
printf("%s%o\t%s\t%s\t%s%c", prefix, ntohl(ce->ce_mode), "blob",
|
|
|
|
sha1_to_hex(ce->sha1), ce->name, line_termination);
|
2005-04-20 12:00:09 +08:00
|
|
|
}
|
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
static int show_modified(struct cache_entry *old, struct cache_entry *new)
|
2005-04-20 12:00:09 +08:00
|
|
|
{
|
2005-04-23 08:15:28 +08:00
|
|
|
unsigned int mode = ntohl(new->ce_mode), oldmode;
|
|
|
|
unsigned char *sha1 = new->sha1;
|
|
|
|
unsigned char old_sha1_hex[60];
|
2005-04-20 12:00:09 +08:00
|
|
|
|
|
|
|
if (!cached_only) {
|
|
|
|
static unsigned char no_sha1[20];
|
2005-04-23 08:15:28 +08:00
|
|
|
int changed;
|
2005-04-20 12:00:09 +08:00
|
|
|
struct stat st;
|
2005-04-23 08:15:28 +08:00
|
|
|
if (stat(new->name, &st) < 0) {
|
|
|
|
show_file("-", old);
|
2005-04-20 12:00:09 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2005-04-23 08:15:28 +08:00
|
|
|
changed = cache_match_stat(new, &st);
|
2005-04-20 12:00:09 +08:00
|
|
|
if (changed) {
|
2005-04-23 08:15:28 +08:00
|
|
|
mode = st.st_mode;
|
|
|
|
sha1 = no_sha1;
|
2005-04-20 12:00:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
oldmode = ntohl(old->ce_mode);
|
|
|
|
if (mode == oldmode && !memcmp(sha1, old->sha1, 20))
|
2005-04-20 12:00:09 +08:00
|
|
|
return 0;
|
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
strcpy(old_sha1_hex, sha1_to_hex(old->sha1));
|
|
|
|
printf("*%o->%o\t%s\t%s->%s\t%s%c", oldmode, mode,
|
|
|
|
"blob",
|
|
|
|
old_sha1_hex, sha1_to_hex(sha1),
|
|
|
|
old->name, line_termination);
|
2005-04-20 12:00:09 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
static int diff_cache(struct cache_entry **ac, int entries)
|
2005-04-20 12:00:09 +08:00
|
|
|
{
|
2005-04-23 08:15:28 +08:00
|
|
|
while (entries) {
|
|
|
|
struct cache_entry *ce = *ac;
|
2005-04-20 12:00:09 +08:00
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
/* No matching 0-stage (current) entry? Show it as deleted */
|
|
|
|
if (ce_stage(ce)) {
|
|
|
|
show_file("-", ce);
|
2005-04-20 12:00:09 +08:00
|
|
|
ac++;
|
|
|
|
entries--;
|
|
|
|
continue;
|
|
|
|
}
|
2005-04-23 08:15:28 +08:00
|
|
|
/* No matching 1-stage (tree) entry? Show the current one as added */
|
|
|
|
if (entries == 1 || !same_name(ce, ac[1])) {
|
2005-04-24 09:05:07 +08:00
|
|
|
show_file("+", ce);
|
2005-04-20 12:00:09 +08:00
|
|
|
ac++;
|
|
|
|
entries--;
|
|
|
|
continue;
|
|
|
|
}
|
2005-04-23 08:15:28 +08:00
|
|
|
show_modified(ac[1], ce);
|
|
|
|
ac += 2;
|
|
|
|
entries -= 2;
|
|
|
|
continue;
|
2005-04-20 12:00:09 +08:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
static void remove_merge_entries(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < active_nr; i++) {
|
|
|
|
struct cache_entry *ce = active_cache[i];
|
|
|
|
if (!ce_stage(ce))
|
2005-04-27 01:13:31 +08:00
|
|
|
continue;
|
2005-04-27 08:48:57 +08:00
|
|
|
printf("U %s%c", ce->name, line_termination);
|
2005-04-23 08:15:28 +08:00
|
|
|
while (remove_entry_at(i)) {
|
|
|
|
if (!ce_stage(active_cache[i]))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-04-21 10:49:16 +08:00
|
|
|
static char *diff_cache_usage = "diff-cache [-r] [-z] [--cached] <tree sha1>";
|
|
|
|
|
2005-04-20 12:00:09 +08:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
unsigned char tree_sha1[20];
|
|
|
|
void *tree;
|
|
|
|
unsigned long size;
|
|
|
|
|
|
|
|
read_cache();
|
|
|
|
while (argc > 2) {
|
|
|
|
char *arg = argv[1];
|
|
|
|
argv++;
|
|
|
|
argc--;
|
|
|
|
if (!strcmp(arg, "-r")) {
|
2005-04-23 08:15:28 +08:00
|
|
|
/* We accept the -r flag just to look like diff-tree */
|
2005-04-20 12:00:09 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!strcmp(arg, "-z")) {
|
|
|
|
line_termination = '\0';
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!strcmp(arg, "--cached")) {
|
|
|
|
cached_only = 1;
|
|
|
|
continue;
|
|
|
|
}
|
2005-04-21 10:49:16 +08:00
|
|
|
usage(diff_cache_usage);
|
2005-04-20 12:00:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (argc != 2 || get_sha1_hex(argv[1], tree_sha1))
|
2005-04-21 10:49:16 +08:00
|
|
|
usage(diff_cache_usage);
|
2005-04-20 12:00:09 +08:00
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
remove_merge_entries();
|
|
|
|
|
2005-04-21 09:06:49 +08:00
|
|
|
tree = read_tree_with_tree_or_commit_sha1(tree_sha1, &size, 0);
|
2005-04-20 12:00:09 +08:00
|
|
|
if (!tree)
|
|
|
|
die("bad tree object %s", argv[1]);
|
2005-04-23 08:15:28 +08:00
|
|
|
if (read_tree(tree, size, 1))
|
|
|
|
die("unable to read tree object %s", argv[1]);
|
2005-04-20 12:00:09 +08:00
|
|
|
|
2005-04-23 08:15:28 +08:00
|
|
|
return diff_cache(active_cache, active_nr);
|
2005-04-20 12:00:09 +08:00
|
|
|
}
|