/* * Copyright (C) 2005 Junio C Hamano */ #include "cache.h" #include "strbuf.h" #include "diff.h" static const char *pickaxe = NULL; static int pickaxe_opts = 0; static const char *orderfile = NULL; static int line_termination = '\n'; static int inter_name_termination = '\t'; static void flush_them(int ac, const char **av) { diffcore_std(av + 1, 0, 0, /* no renames */ pickaxe, pickaxe_opts, -1, /* no breaks */ orderfile); diff_flush(DIFF_FORMAT_PATCH, 0); } static const char *diff_helper_usage = "git-diff-helper [-z] [-S<string>] [-O<orderfile>] paths..."; int main(int ac, const char **av) { struct strbuf sb; const char *garbage_flush_format; strbuf_init(&sb); while (1 < ac && av[1][0] == '-') { if (av[1][1] == 'z') line_termination = inter_name_termination = 0; else if (av[1][1] == 'S') { pickaxe = av[1] + 2; } else if (!strcmp(av[1], "--pickaxe-all")) pickaxe_opts = DIFF_PICKAXE_ALL; else usage(diff_helper_usage); ac--; av++; } garbage_flush_format = (line_termination == 0) ? "%s" : "%s\n"; /* the remaining parameters are paths patterns */ diff_setup(0); while (1) { unsigned old_mode, new_mode; unsigned char old_sha1[20], new_sha1[20]; char old_path[PATH_MAX]; int status, score, two_paths; char new_path[PATH_MAX]; int ch; char *cp, *ep; read_line(&sb, stdin, line_termination); if (sb.eof) break; switch (sb.buf[0]) { case ':': /* parse the first part up to the status */ cp = sb.buf + 1; old_mode = new_mode = 0; while ((ch = *cp) && ('0' <= ch && ch <= '7')) { old_mode = (old_mode << 3) | (ch - '0'); cp++; } if (*cp++ != ' ') break; while ((ch = *cp) && ('0' <= ch && ch <= '7')) { new_mode = (new_mode << 3) | (ch - '0'); cp++; } if (*cp++ != ' ') break; if (get_sha1_hex(cp, old_sha1)) break; cp += 40; if (*cp++ != ' ') break; if (get_sha1_hex(cp, new_sha1)) break; cp += 40; if (*cp++ != ' ') break; status = *cp++; if (!strchr("MCRNDU", status)) break; two_paths = score = 0; if (status == 'R' || status == 'C') two_paths = 1; /* pick up score if exists */ if (sscanf(cp, "%d", &score) != 1) score = 0; cp = strchr(cp, inter_name_termination); if (!cp) break; if (*cp++ != inter_name_termination) break; /* first pathname */ if (!line_termination) { read_line(&sb, stdin, line_termination); if (sb.eof) break; strcpy(old_path, sb.buf); } else if (!two_paths) strcpy(old_path, cp); else { ep = strchr(cp, inter_name_termination); if (!ep) break; strncpy(old_path, cp, ep-cp); old_path[ep-cp] = 0; cp = ep + 1; } /* second pathname */ if (!two_paths) strcpy(new_path, old_path); else { if (!line_termination) { read_line(&sb, stdin, line_termination); if (sb.eof) break; strcpy(new_path, sb.buf); } else strcpy(new_path, cp); } diff_helper_input(old_mode, new_mode, old_sha1, new_sha1, old_path, status, score, new_path); continue; } flush_them(ac, av); printf(garbage_flush_format, sb.buf); } flush_them(ac, av); return 0; }