[PATCH] Rewrite ls-tree to behave more like "/bin/ls -a"
[git/dscho.git] / diff-helper.c
blob5ad2273bf063fb431f095de2d58ebdfb78814d3b
1 /*
2 * Copyright (C) 2005 Junio C Hamano
3 */
4 #include "cache.h"
5 #include "strbuf.h"
6 #include "diff.h"
8 static const char *pickaxe = NULL;
9 static int pickaxe_opts = 0;
10 static int line_termination = '\n';
11 static int inter_name_termination = '\t';
13 static const char *diff_helper_usage =
14 "git-diff-helper [-z] [-S<string>] paths...";
16 int main(int ac, const char **av) {
17 struct strbuf sb;
18 const char *garbage_flush_format;
20 strbuf_init(&sb);
22 while (1 < ac && av[1][0] == '-') {
23 if (av[1][1] == 'z')
24 line_termination = inter_name_termination = 0;
25 else if (av[1][1] == 'S') {
26 pickaxe = av[1] + 2;
28 else if (!strcmp(av[1], "--pickaxe-all"))
29 pickaxe_opts = DIFF_PICKAXE_ALL;
30 else
31 usage(diff_helper_usage);
32 ac--; av++;
34 garbage_flush_format = (line_termination == 0) ? "%s" : "%s\n";
36 /* the remaining parameters are paths patterns */
38 diff_setup(0);
39 while (1) {
40 unsigned old_mode, new_mode;
41 unsigned char old_sha1[20], new_sha1[20];
42 char old_path[PATH_MAX];
43 int status, score, two_paths;
44 char new_path[PATH_MAX];
46 int ch;
47 char *cp, *ep;
49 read_line(&sb, stdin, line_termination);
50 if (sb.eof)
51 break;
52 switch (sb.buf[0]) {
53 case ':':
54 /* parse the first part up to the status */
55 cp = sb.buf + 1;
56 old_mode = new_mode = 0;
57 while ((ch = *cp) && ('0' <= ch && ch <= '7')) {
58 old_mode = (old_mode << 3) | (ch - '0');
59 cp++;
61 if (*cp++ != ' ')
62 break;
63 while ((ch = *cp) && ('0' <= ch && ch <= '7')) {
64 new_mode = (new_mode << 3) | (ch - '0');
65 cp++;
67 if (*cp++ != ' ')
68 break;
69 if (get_sha1_hex(cp, old_sha1))
70 break;
71 cp += 40;
72 if (*cp++ != ' ')
73 break;
74 if (get_sha1_hex(cp, new_sha1))
75 break;
76 cp += 40;
77 if (*cp++ != ' ')
78 break;
79 status = *cp++;
80 if (!strchr("MCRNDU", status))
81 break;
82 two_paths = score = 0;
83 if (status == 'R' || status == 'C') {
84 two_paths = 1;
85 sscanf(cp, "%d", &score);
86 if (line_termination) {
87 cp = strchr(cp,
88 inter_name_termination);
89 if (!cp)
90 break;
94 if (*cp++ != inter_name_termination)
95 break;
97 /* first pathname */
98 if (!line_termination) {
99 read_line(&sb, stdin, line_termination);
100 if (sb.eof)
101 break;
102 strcpy(old_path, sb.buf);
104 else if (!two_paths)
105 strcpy(old_path, cp);
106 else {
107 ep = strchr(cp, inter_name_termination);
108 if (!ep)
109 break;
110 strncpy(old_path, cp, ep-cp);
111 old_path[ep-cp] = 0;
112 cp = ep + 1;
115 /* second pathname */
116 if (!two_paths)
117 strcpy(new_path, old_path);
118 else {
119 if (!line_termination) {
120 read_line(&sb, stdin,
121 line_termination);
122 if (sb.eof)
123 break;
124 strcpy(new_path, sb.buf);
126 else
127 strcpy(new_path, cp);
129 diff_helper_input(old_mode, new_mode,
130 old_sha1, new_sha1,
131 old_path, status, score,
132 new_path);
133 continue;
135 if (1 < ac)
136 diffcore_pathspec(av + 1);
137 if (pickaxe)
138 diffcore_pickaxe(pickaxe, pickaxe_opts);
139 diff_flush(DIFF_FORMAT_PATCH, 0);
140 printf(garbage_flush_format, sb.buf);
142 if (1 < ac)
143 diffcore_pathspec(av + 1);
144 if (pickaxe)
145 diffcore_pickaxe(pickaxe, pickaxe_opts);
146 diff_flush(DIFF_FORMAT_PATCH, 0);
147 return 0;