Browse Source

[PATCH] Introducing software archaeologist's tool "pickaxe".

This steals the "pickaxe" feature from JIT and make it available
to the bare Plumbing layer.  From the command line, the user
gives a string he is intersted in.

Using the diff-core infrastructure previously introduced, it
filters the differences to limit the output only to the diffs
between <src> and <dst> where the string appears only in one but
not in the other.  For example:

 $ ./git-rev-list HEAD | ./git-diff-tree -Sdiff-tree-helper --stdin -M

would show the diffs that touch the string "diff-tree-helper".

In real software-archaeologist application, you would typically
look for a few to several lines of code and see where that code
came from.

The "pickaxe" module runs after "rename/copy detection" module,
so it even crosses the file rename boundary, as the above
example demonstrates.

Signed-off-by: Junio C Hamano <junkio@cox.net>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
maint
Junio C Hamano 20 years ago committed by Linus Torvalds
parent
commit
52e9578985
  1. 6
      Documentation/git-diff-cache.txt
  2. 6
      Documentation/git-diff-files.txt
  3. 6
      Documentation/git-diff-helper.txt
  4. 5
      Documentation/git-diff-tree.txt
  5. 3
      Makefile
  6. 11
      diff-cache.c
  7. 9
      diff-files.c
  8. 10
      diff-helper.c
  9. 15
      diff-tree.c
  10. 23
      diff.c
  11. 1
      diff.h
  12. 56
      diffcore-pickaxe.c
  13. 29
      diffcore-rename.c
  14. 11
      diffcore.h

6
Documentation/git-diff-cache.txt

@ -9,7 +9,7 @@ git-diff-cache - Compares content and mode of blobs between the cache and reposi


SYNOPSIS SYNOPSIS
-------- --------
'git-diff-cache' [-p] [-r] [-z] [-m] [-M] [-R] [-C] [--cached] <tree-ish> 'git-diff-cache' [-p] [-r] [-z] [-m] [-M] [-R] [-C] [-S<string>] [--cached] <tree-ish>


DESCRIPTION DESCRIPTION
----------- -----------
@ -39,6 +39,10 @@ OPTIONS
-C:: -C::
Detect copies as well as renames; implies -p. Detect copies as well as renames; implies -p.


-S<string>::
Look for differences that contains the change in <string>.


-R:: -R::
Output diff in reverse. Output diff in reverse.



6
Documentation/git-diff-files.txt

@ -9,7 +9,7 @@ git-diff-files - Compares files in the working tree and the cache


SYNOPSIS SYNOPSIS
-------- --------
'git-diff-files' [-p] [-q] [-r] [-z] [-M] [-C] [-R] [<pattern>...] 'git-diff-files' [-p] [-q] [-r] [-z] [-M] [-C] [-R] [-S<string>] [<pattern>...]


DESCRIPTION DESCRIPTION
----------- -----------
@ -35,6 +35,10 @@ OPTIONS
-C:: -C::
Detect copies as well as renames; implies -p. Detect copies as well as renames; implies -p.


-S<string>::
Look for differences that contains the change in <string>.


-r:: -r::
This flag does not mean anything. It is there only to match This flag does not mean anything. It is there only to match
git-diff-tree. Unlike git-diff-tree, git-diff-files always looks git-diff-tree. Unlike git-diff-tree, git-diff-files always looks

6
Documentation/git-diff-helper.txt

@ -9,7 +9,7 @@ git-diff-helper - Generates patch format output for git-diff-*


SYNOPSIS SYNOPSIS
-------- --------
'git-diff-helper' [-z] [-R] [-M] [-C] 'git-diff-helper' [-z] [-R] [-M] [-C] [-S<string>]


DESCRIPTION DESCRIPTION
----------- -----------
@ -37,6 +37,10 @@ OPTIONS
-C:: -C::
Detect copies as well as renames. Detect copies as well as renames.


-S<string>::
Look for differences that contains the change in <string>.


See Also See Also
-------- --------
The section on generating patches in link:git-diff-cache.html[git-diff-cache] The section on generating patches in link:git-diff-cache.html[git-diff-cache]

5
Documentation/git-diff-tree.txt

@ -9,7 +9,7 @@ git-diff-tree - Compares the content and mode of blobs found via two tree object


SYNOPSIS SYNOPSIS
-------- --------
'git-diff-tree' [-p] [-r] [-z] [--stdin] [-M] [-R] [-C] [-m] [-s] [-v] <tree-ish> <tree-ish> [<pattern>]\* 'git-diff-tree' [-p] [-r] [-z] [--stdin] [-M] [-R] [-C] [-S<string>] [-m] [-s] [-v] <tree-ish> <tree-ish> [<pattern>]\*


DESCRIPTION DESCRIPTION
----------- -----------
@ -43,6 +43,9 @@ OPTIONS
-R:: -R::
Output diff in reverse. Output diff in reverse.


-S<string>::
Look for differences that contains the change in <string>.

-r:: -r::
recurse recurse



3
Makefile

@ -45,7 +45,7 @@ LIB_H += strbuf.h
LIB_OBJS += strbuf.o LIB_OBJS += strbuf.o


LIB_H += diff.h LIB_H += diff.h
LIB_OBJS += diff.o diffcore-rename.o LIB_OBJS += diff.o diffcore-rename.o diffcore-pickaxe.o


LIB_OBJS += gitenv.o LIB_OBJS += gitenv.o


@ -125,6 +125,7 @@ strbuf.o: $(LIB_H)
gitenv.o: $(LIB_H) gitenv.o: $(LIB_H)
diff.o: $(LIB_H) diff.o: $(LIB_H)
diffcore-rename.o : $(LIB_H) diffcore-rename.o : $(LIB_H)
diffcore-pickaxe.o : $(LIB_H)


test: all test: all
make -C t/ all make -C t/ all

11
diff-cache.c

@ -8,6 +8,7 @@ static int line_termination = '\n';
static int detect_rename = 0; static int detect_rename = 0;
static int reverse_diff = 0; static int reverse_diff = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
static char *pickaxe = 0;


/* A file entry went away or appeared */ /* A file entry went away or appeared */
static void show_file(const char *prefix, struct cache_entry *ce, unsigned char *sha1, unsigned int mode) static void show_file(const char *prefix, struct cache_entry *ce, unsigned char *sha1, unsigned int mode)
@ -153,7 +154,7 @@ static void mark_merge_entries(void)
} }


static char *diff_cache_usage = static char *diff_cache_usage =
"git-diff-cache [-p] [-r] [-z] [-m] [-M] [-C] [-R] [--cached] <tree-ish>"; "git-diff-cache [-p] [-r] [-z] [-m] [-M] [-C] [-R] [-S<string>] [--cached] <tree-ish>";


int main(int argc, char **argv) int main(int argc, char **argv)
{ {
@ -194,6 +195,10 @@ int main(int argc, char **argv)
reverse_diff = 1; reverse_diff = 1;
continue; continue;
} }
if (!strcmp(arg, "-S")) {
pickaxe = arg + 2;
continue;
}
if (!strcmp(arg, "-m")) { if (!strcmp(arg, "-m")) {
match_nonexisting = 1; match_nonexisting = 1;
continue; continue;
@ -208,8 +213,8 @@ int main(int argc, char **argv)
if (argc != 2 || get_sha1(argv[1], tree_sha1)) if (argc != 2 || get_sha1(argv[1], tree_sha1))
usage(diff_cache_usage); usage(diff_cache_usage);


diff_setup(detect_rename, diff_score_opt, reverse_diff, diff_setup(detect_rename, diff_score_opt, pickaxe,
(generate_patch ? -1 : line_termination), reverse_diff, (generate_patch ? -1 : line_termination),
NULL, 0); NULL, 0);


mark_merge_entries(); mark_merge_entries();

9
diff-files.c

@ -7,13 +7,14 @@
#include "diff.h" #include "diff.h"


static const char *diff_files_usage = static const char *diff_files_usage =
"git-diff-files [-p] [-q] [-r] [-z] [-M] [-C] [-R] [paths...]"; "git-diff-files [-p] [-q] [-r] [-z] [-M] [-C] [-R] [-S<string>] [paths...]";


static int generate_patch = 0; static int generate_patch = 0;
static int line_termination = '\n'; static int line_termination = '\n';
static int detect_rename = 0; static int detect_rename = 0;
static int reverse_diff = 0; static int reverse_diff = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
static char *pickaxe = 0;
static int silent = 0; static int silent = 0;


static int matches_pathspec(struct cache_entry *ce, char **spec, int cnt) static int matches_pathspec(struct cache_entry *ce, char **spec, int cnt)
@ -67,6 +68,8 @@ int main(int argc, char **argv)
line_termination = 0; line_termination = 0;
else if (!strcmp(argv[1], "-R")) else if (!strcmp(argv[1], "-R"))
reverse_diff = 1; reverse_diff = 1;
else if (!strcmp(argv[1], "-S"))
pickaxe = argv[1] + 2;
else if (!strncmp(argv[1], "-M", 2)) { else if (!strncmp(argv[1], "-M", 2)) {
diff_score_opt = diff_scoreopt_parse(argv[1]); diff_score_opt = diff_scoreopt_parse(argv[1]);
detect_rename = generate_patch = 1; detect_rename = generate_patch = 1;
@ -89,8 +92,8 @@ int main(int argc, char **argv)
exit(1); exit(1);
} }


diff_setup(detect_rename, diff_score_opt, reverse_diff, diff_setup(detect_rename, diff_score_opt, pickaxe,
(generate_patch ? -1 : line_termination), reverse_diff, (generate_patch ? -1 : line_termination),
NULL, 0); NULL, 0);


for (i = 0; i < entries; i++) { for (i = 0; i < entries; i++) {

10
diff-helper.c

@ -9,6 +9,7 @@
static int detect_rename = 0; static int detect_rename = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
static int generate_patch = 1; static int generate_patch = 1;
static char *pickaxe = 0;


static int parse_oneside_change(const char *cp, int *mode, static int parse_oneside_change(const char *cp, int *mode,
unsigned char *sha1, char *path) unsigned char *sha1, char *path)
@ -93,7 +94,7 @@ static int parse_diff_raw_output(const char *buf)
} }


static const char *diff_helper_usage = static const char *diff_helper_usage =
"git-diff-helper [-z] [-R] [-M] [-C] paths..."; "git-diff-helper [-z] [-R] [-M] [-C] [-S<string>] paths...";


int main(int ac, const char **av) { int main(int ac, const char **av) {
struct strbuf sb; struct strbuf sb;
@ -117,14 +118,17 @@ int main(int ac, const char **av) {
detect_rename = 2; detect_rename = 2;
diff_score_opt = diff_scoreopt_parse(av[1]); diff_score_opt = diff_scoreopt_parse(av[1]);
} }
else if (av[1][1] == 'S') {
pickaxe = av[1] + 2;
}
else else
usage(diff_helper_usage); usage(diff_helper_usage);
ac--; av++; ac--; av++;
} }
/* the remaining parameters are paths patterns */ /* the remaining parameters are paths patterns */


diff_setup(detect_rename, diff_score_opt, reverse, diff_setup(detect_rename, diff_score_opt, pickaxe,
(generate_patch ? -1 : line_termination), reverse, (generate_patch ? -1 : line_termination),
av+1, ac-1); av+1, ac-1);


while (1) { while (1) {

15
diff-tree.c

@ -13,6 +13,7 @@ static int generate_patch = 0;
static int detect_rename = 0; static int detect_rename = 0;
static int reverse_diff = 0; static int reverse_diff = 0;
static int diff_score_opt = 0; static int diff_score_opt = 0;
static char *pickaxe = 0;
static const char *header = NULL; static const char *header = NULL;
static const char *header_prefix = ""; static const char *header_prefix = "";


@ -271,8 +272,8 @@ static int diff_tree_sha1_top(const unsigned char *old,
{ {
int ret; int ret;


diff_setup(detect_rename, diff_score_opt, reverse_diff, diff_setup(detect_rename, diff_score_opt, pickaxe,
(generate_patch ? -1 : line_termination), reverse_diff, (generate_patch ? -1 : line_termination),
NULL, 0); NULL, 0);
ret = diff_tree_sha1(old, new, base); ret = diff_tree_sha1(old, new, base);
diff_flush(); diff_flush();
@ -285,8 +286,8 @@ static int diff_root_tree(const unsigned char *new, const char *base)
void *tree; void *tree;
unsigned long size; unsigned long size;


diff_setup(detect_rename, diff_score_opt, reverse_diff, diff_setup(detect_rename, diff_score_opt, pickaxe,
(generate_patch ? -1 : line_termination), reverse_diff, (generate_patch ? -1 : line_termination),
NULL, 0); NULL, 0);
tree = read_object_with_reference(new, "tree", &size, NULL); tree = read_object_with_reference(new, "tree", &size, NULL);
if (!tree) if (!tree)
@ -430,7 +431,7 @@ static int diff_tree_stdin(char *line)
} }


static char *diff_tree_usage = static char *diff_tree_usage =
"git-diff-tree [-p] [-r] [-z] [--stdin] [-M] [-C] [-R] [-m] [-s] [-v] <tree-ish> <tree-ish>"; "git-diff-tree [-p] [-r] [-z] [--stdin] [-M] [-C] [-R] [-S<string>] [-m] [-s] [-v] <tree-ish> <tree-ish>";


int main(int argc, char **argv) int main(int argc, char **argv)
{ {
@ -473,6 +474,10 @@ int main(int argc, char **argv)
recursive = generate_patch = 1; recursive = generate_patch = 1;
continue; continue;
} }
if (!strncmp(arg, "-S", 2)) {
pickaxe = arg + 2;
continue;
}
if (!strncmp(arg, "-M", 2)) { if (!strncmp(arg, "-M", 2)) {
detect_rename = recursive = generate_patch = 1; detect_rename = recursive = generate_patch = 1;
diff_score_opt = diff_scoreopt_parse(arg); diff_score_opt = diff_scoreopt_parse(arg);

23
diff.c

@ -17,6 +17,7 @@ static int reverse_diff;
static int diff_raw_output = -1; static int diff_raw_output = -1;
static const char **pathspec; static const char **pathspec;
static int speccnt; static int speccnt;
static const char *pickaxe;
static int minimum_score; static int minimum_score;


static const char *external_diff(void) static const char *external_diff(void)
@ -511,8 +512,9 @@ int diff_scoreopt_parse(const char *opt)
return MAX_SCORE * num / scale; return MAX_SCORE * num / scale;
} }


void diff_setup(int detect_rename_, int minimum_score_, int reverse_diff_, void diff_setup(int detect_rename_, int minimum_score_,
int diff_raw_output_, char *pickaxe_,
int reverse_diff_, int diff_raw_output_,
const char **pathspec_, int speccnt_) const char **pathspec_, int speccnt_)
{ {
detect_rename = detect_rename_; detect_rename = detect_rename_;
@ -521,15 +523,16 @@ void diff_setup(int detect_rename_, int minimum_score_, int reverse_diff_,
diff_raw_output = diff_raw_output_; diff_raw_output = diff_raw_output_;
speccnt = speccnt_; speccnt = speccnt_;
minimum_score = minimum_score_ ? : DEFAULT_MINIMUM_SCORE; minimum_score = minimum_score_ ? : DEFAULT_MINIMUM_SCORE;
pickaxe = pickaxe_;
} }


static struct diff_queue_struct queued_diff; static struct diff_queue_struct queued_diff;


struct diff_file_pair *diff_queue(struct diff_queue_struct *queue, struct diff_filepair *diff_queue(struct diff_queue_struct *queue,
struct diff_filespec *one, struct diff_filespec *one,
struct diff_filespec *two) struct diff_filespec *two)
{ {
struct diff_file_pair *dp = xmalloc(sizeof(*dp)); struct diff_filepair *dp = xmalloc(sizeof(*dp));
dp->one = one; dp->one = one;
dp->two = two; dp->two = two;
dp->xfrm_msg = 0; dp->xfrm_msg = 0;
@ -549,7 +552,7 @@ static const char *git_object_type(unsigned mode)
return S_ISDIR(mode) ? "tree" : "blob"; return S_ISDIR(mode) ? "tree" : "blob";
} }


static void diff_flush_raw(struct diff_file_pair *p) static void diff_flush_raw(struct diff_filepair *p)
{ {
struct diff_filespec *it; struct diff_filespec *it;
int addremove; int addremove;
@ -583,7 +586,7 @@ static void diff_flush_raw(struct diff_file_pair *p)
sha1_to_hex(it->sha1), it->path, diff_raw_output); sha1_to_hex(it->sha1), it->path, diff_raw_output);
} }


static void diff_flush_patch(struct diff_file_pair *p) static void diff_flush_patch(struct diff_filepair *p)
{ {
const char *name, *other; const char *name, *other;


@ -600,7 +603,7 @@ static int identical(struct diff_filespec *one, struct diff_filespec *two)
{ {
/* This function is written stricter than necessary to support /* This function is written stricter than necessary to support
* the currently implemented transformers, but the idea is to * the currently implemented transformers, but the idea is to
* let transformers to produce diff_file_pairs any way they want, * let transformers to produce diff_filepairs any way they want,
* and filter and clean them up here before producing the output. * and filter and clean them up here before producing the output.
*/ */


@ -623,7 +626,7 @@ static int identical(struct diff_filespec *one, struct diff_filespec *two)
return 0; return 0;
} }


static void diff_flush_one(struct diff_file_pair *p) static void diff_flush_one(struct diff_filepair *p)
{ {
if (identical(p->one, p->two)) if (identical(p->one, p->two))
return; return;
@ -640,11 +643,13 @@ void diff_flush(void)


if (detect_rename) if (detect_rename)
diff_detect_rename(q, detect_rename, minimum_score); diff_detect_rename(q, detect_rename, minimum_score);
if (pickaxe)
diff_pickaxe(q, pickaxe);
for (i = 0; i < q->nr; i++) for (i = 0; i < q->nr; i++)
diff_flush_one(q->queue[i]); diff_flush_one(q->queue[i]);


for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_file_pair *p = q->queue[i]; struct diff_filepair *p = q->queue[i];
diff_free_filespec_data(p->one); diff_free_filespec_data(p->one);
diff_free_filespec_data(p->two); diff_free_filespec_data(p->two);
free(p->xfrm_msg); free(p->xfrm_msg);

1
diff.h

@ -20,6 +20,7 @@ extern void diff_unmerge(const char *path);
extern int diff_scoreopt_parse(const char *opt); extern int diff_scoreopt_parse(const char *opt);


extern void diff_setup(int detect_rename, int minimum_score, extern void diff_setup(int detect_rename, int minimum_score,
char *pickaxe,
int reverse, int raw_output, int reverse, int raw_output,
const char **spec, int cnt); const char **spec, int cnt);



56
diffcore-pickaxe.c

@ -0,0 +1,56 @@
/*
* Copyright (C) 2005 Junio C Hamano
*/
#include "cache.h"
#include "diff.h"
#include "diffcore.h"
#include "delta.h"

static int contains(struct diff_filespec *one,
const char *needle, unsigned long len)
{
unsigned long offset, sz;
const char *data;
if (diff_populate_filespec(one))
return 0;
sz = one->size;
data = one->data;
for (offset = 0; offset + len <= sz; offset++)
if (!strncmp(needle, data + offset, len))
return 1;
return 0;
}

void diff_pickaxe(struct diff_queue_struct *q, const char *needle)
{
unsigned long len = strlen(needle);
int i;
struct diff_queue_struct outq;
outq.queue = NULL;
outq.nr = outq.alloc = 0;

for (i = 0; i < q->nr; i++) {
struct diff_filepair *p = q->queue[i];
if (!p->one->file_valid) {
if (!p->two->file_valid)
continue; /* ignore nonsense */
/* created */
if (contains(p->two, needle, len))
diff_queue(&outq, p->one, p->two);
}
else if (!p->two->file_valid) {
if (contains(p->one, needle, len))
diff_queue(&outq, p->one, p->two);
}
else if (contains(p->one, needle, len) !=
contains(p->two, needle, len))
diff_queue(&outq, p->one, p->two);
}
for (i = 0; i < q->nr; i++) {
struct diff_filepair *p = q->queue[i];
free(p);
}
free(q->queue);
*q = outq;
return;
}

29
diffcore-rename.c

@ -129,7 +129,7 @@ static void record_rename_pair(struct diff_queue_struct *outq,
* To achieve this sort order, we give xform_work the number * To achieve this sort order, we give xform_work the number
* above. * above.
*/ */
struct diff_file_pair *dp = diff_queue(outq, src, dst); struct diff_filepair *dp = diff_queue(outq, src, dst);
dp->xfrm_work = (rank * 2 + 1) | (score<<RENAME_SCORE_SHIFT); dp->xfrm_work = (rank * 2 + 1) | (score<<RENAME_SCORE_SHIFT);
dst->xfrm_flags |= RENAME_DST_MATCHED; dst->xfrm_flags |= RENAME_DST_MATCHED;
} }
@ -148,7 +148,7 @@ static void debug_filespec(struct diff_filespec *s, int x, const char *one)
s->size, s->xfrm_flags); s->size, s->xfrm_flags);
} }


static void debug_filepair(const struct diff_file_pair *p, int i) static void debug_filepair(const struct diff_filepair *p, int i)
{ {
debug_filespec(p->one, i, "one"); debug_filespec(p->one, i, "one");
debug_filespec(p->two, i, "two"); debug_filespec(p->two, i, "two");
@ -165,7 +165,7 @@ static void debug_queue(const char *msg, struct diff_queue_struct *q)
fprintf(stderr, "%s\n", msg); fprintf(stderr, "%s\n", msg);
fprintf(stderr, "q->nr = %d\n", q->nr); fprintf(stderr, "q->nr = %d\n", q->nr);
for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_file_pair *p = q->queue[i]; struct diff_filepair *p = q->queue[i];
debug_filepair(p, i); debug_filepair(p, i);
} }
} }
@ -180,8 +180,8 @@ static void debug_queue(const char *msg, struct diff_queue_struct *q)
*/ */
static int rank_compare(const void *a_, const void *b_) static int rank_compare(const void *a_, const void *b_)
{ {
const struct diff_file_pair *a = *(const struct diff_file_pair **)a_; const struct diff_filepair *a = *(const struct diff_filepair **)a_;
const struct diff_file_pair *b = *(const struct diff_file_pair **)b_; const struct diff_filepair *b = *(const struct diff_filepair **)b_;
int a_rank = a->xfrm_work & ((1<<RENAME_SCORE_SHIFT) - 1); int a_rank = a->xfrm_work & ((1<<RENAME_SCORE_SHIFT) - 1);
int b_rank = b->xfrm_work & ((1<<RENAME_SCORE_SHIFT) - 1); int b_rank = b->xfrm_work & ((1<<RENAME_SCORE_SHIFT) - 1);


@ -207,7 +207,7 @@ static int needs_to_stay(struct diff_queue_struct *q, int i,
* as the source of rename/copy), we need to copy, not rename. * as the source of rename/copy), we need to copy, not rename.
*/ */
while (i < q->nr) { while (i < q->nr) {
struct diff_file_pair *p = q->queue[i++]; struct diff_filepair *p = q->queue[i++];
if (!p->two->file_valid) if (!p->two->file_valid)
continue; /* removed is fine */ continue; /* removed is fine */
if (strcmp(p->one->path, it->path)) if (strcmp(p->one->path, it->path))
@ -243,15 +243,8 @@ void diff_detect_rename(struct diff_queue_struct *q,
srcs[0] = &deleted; srcs[0] = &deleted;
srcs[1] = &stay; srcs[1] = &stay;


/* NEEDSWORK:
* (1) make sure we properly ignore but pass trees.
*
* (2) make sure we do right thing on the same path deleted
* and created in the same patch.
*/

for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_file_pair *p = q->queue[i]; struct diff_filepair *p = q->queue[i];
if (!p->one->file_valid) if (!p->one->file_valid)
if (!p->two->file_valid) if (!p->two->file_valid)
continue; /* ignore nonsense */ continue; /* ignore nonsense */
@ -340,11 +333,11 @@ void diff_detect_rename(struct diff_queue_struct *q,
* See comments at the top of record_rename_pair for numbers used * See comments at the top of record_rename_pair for numbers used
* to assign xfrm_work. * to assign xfrm_work.
* *
* Note that we have not annotated the diff_file_pair with any comment * Note that we have not annotated the diff_filepair with any comment
* so there is nothing other than p to free. * so there is nothing other than p to free.
*/ */
for (i = 0; i < q->nr; i++) { for (i = 0; i < q->nr; i++) {
struct diff_file_pair *dp, *p = q->queue[i]; struct diff_filepair *dp, *p = q->queue[i];
if (!p->one->file_valid) { if (!p->one->file_valid) {
if (p->two->file_valid) { if (p->two->file_valid) {
/* creation */ /* creation */
@ -378,7 +371,7 @@ void diff_detect_rename(struct diff_queue_struct *q,


/* Copy it out to q, removing duplicates. */ /* Copy it out to q, removing duplicates. */
for (i = 0; i < outq.nr; i++) { for (i = 0; i < outq.nr; i++) {
struct diff_file_pair *p = outq.queue[i]; struct diff_filepair *p = outq.queue[i];
if (!p->one->file_valid) { if (!p->one->file_valid) {
/* created */ /* created */
if (p->two->xfrm_flags & RENAME_DST_MATCHED) if (p->two->xfrm_flags & RENAME_DST_MATCHED)
@ -395,7 +388,7 @@ void diff_detect_rename(struct diff_queue_struct *q,
} }
else if (strcmp(p->one->path, p->two->path)) { else if (strcmp(p->one->path, p->two->path)) {
/* rename or copy */ /* rename or copy */
struct diff_file_pair *dp = struct diff_filepair *dp =
diff_queue(q, p->one, p->two); diff_queue(q, p->one, p->two);
int msglen = (strlen(p->one->path) + int msglen = (strlen(p->one->path) +
strlen(p->two->path) + 100); strlen(p->two->path) + 100);

11
diffcore.h

@ -38,7 +38,7 @@ extern void fill_filespec(struct diff_filespec *, const unsigned char *,
extern int diff_populate_filespec(struct diff_filespec *); extern int diff_populate_filespec(struct diff_filespec *);
extern void diff_free_filespec_data(struct diff_filespec *); extern void diff_free_filespec_data(struct diff_filespec *);


struct diff_file_pair { struct diff_filepair {
struct diff_filespec *one; struct diff_filespec *one;
struct diff_filespec *two; struct diff_filespec *two;
char *xfrm_msg; char *xfrm_msg;
@ -47,14 +47,15 @@ struct diff_file_pair {
}; };


struct diff_queue_struct { struct diff_queue_struct {
struct diff_file_pair **queue; struct diff_filepair **queue;
int alloc; int alloc;
int nr; int nr;
}; };


extern struct diff_file_pair *diff_queue(struct diff_queue_struct *, extern struct diff_filepair *diff_queue(struct diff_queue_struct *,
struct diff_filespec *, struct diff_filespec *,
struct diff_filespec *); struct diff_filespec *);
extern void diff_detect_rename(struct diff_queue_struct *, int, int); extern void diff_detect_rename(struct diff_queue_struct *, int, int);
extern void diff_pickaxe(struct diff_queue_struct *, const char *);


#endif #endif

Loading…
Cancel
Save