[PATCH] diff: mode bits fixes
The core GIT repository has trees that record regular file mode in 0664 instead of normalized 0644 pattern. Comparing such a tree with another tree that records the same file in 0644 pattern without content changes with git-diff-tree causes it to feed otherwise unmodified pairs to the diff_change() routine, which triggers a sanity check routine and barfs. This patch fixes the problem, along with the fix to another caller that uses unnormalized mode bits to call diff_change() routine in a similar way. Without this patch, you will see "fatal error" from diff-tree when you run git-deltafy-script on the core GIT repository itself. Signed-off-by: Junio C Hamano <junkio@cox.net> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:

committed by
Linus Torvalds

parent
81bb573ed8
commit
67574c403f
@ -88,7 +88,7 @@ int main(int argc, const char **argv)
|
|||||||
|
|
||||||
for (i = 0; i < entries; i++) {
|
for (i = 0; i < entries; i++) {
|
||||||
struct stat st;
|
struct stat st;
|
||||||
unsigned int oldmode, mode;
|
unsigned int oldmode;
|
||||||
struct cache_entry *ce = active_cache[i];
|
struct cache_entry *ce = active_cache[i];
|
||||||
int changed;
|
int changed;
|
||||||
|
|
||||||
@ -116,10 +116,8 @@ int main(int argc, const char **argv)
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
oldmode = ntohl(ce->ce_mode);
|
oldmode = ntohl(ce->ce_mode);
|
||||||
mode = (S_ISLNK(st.st_mode) ? S_IFLNK :
|
show_modified(oldmode, DIFF_FILE_CANON_MODE(st.st_mode),
|
||||||
S_IFREG | ce_permissions(st.st_mode));
|
ce->sha1, null_sha1,
|
||||||
|
|
||||||
show_modified(oldmode, mode, ce->sha1, null_sha1,
|
|
||||||
ce->name);
|
ce->name);
|
||||||
}
|
}
|
||||||
diffcore_std((1 < argc) ? argv + 1 : NULL,
|
diffcore_std((1 < argc) ? argv + 1 : NULL,
|
||||||
|
@ -44,10 +44,12 @@ static const unsigned char *extract(void *tree, unsigned long size, const char *
|
|||||||
int len = strlen(tree)+1;
|
int len = strlen(tree)+1;
|
||||||
const unsigned char *sha1 = tree + len;
|
const unsigned char *sha1 = tree + len;
|
||||||
const char *path = strchr(tree, ' ');
|
const char *path = strchr(tree, ' ');
|
||||||
|
unsigned int mode;
|
||||||
|
|
||||||
if (!path || size < len + 20 || sscanf(tree, "%o", modep) != 1)
|
if (!path || size < len + 20 || sscanf(tree, "%o", &mode) != 1)
|
||||||
die("corrupt tree file");
|
die("corrupt tree file");
|
||||||
*pathp = path+1;
|
*pathp = path+1;
|
||||||
|
*modep = DIFF_FILE_CANON_MODE(mode);
|
||||||
return sha1;
|
return sha1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
12
diff.c
12
diff.c
@ -854,12 +854,14 @@ static void diff_resolve_rename_copy(void)
|
|||||||
else if (memcmp(p->one->sha1, p->two->sha1, 20) ||
|
else if (memcmp(p->one->sha1, p->two->sha1, 20) ||
|
||||||
p->one->mode != p->two->mode)
|
p->one->mode != p->two->mode)
|
||||||
p->status = 'M';
|
p->status = 'M';
|
||||||
else
|
else {
|
||||||
/* this is a "no-change" entry.
|
/* This is a "no-change" entry and should not
|
||||||
* should not happen anymore.
|
* happen anymore, but prepare for broken callers.
|
||||||
* p->status = 'X';
|
|
||||||
*/
|
*/
|
||||||
die("internal error in diffcore: unmodified entry remains");
|
error("feeding unmodified %s to diffcore",
|
||||||
|
p->one->path);
|
||||||
|
p->status = 'X';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
diff_debug_queue("resolve-rename-copy done", q);
|
diff_debug_queue("resolve-rename-copy done", q);
|
||||||
}
|
}
|
||||||
|
4
diff.h
4
diff.h
@ -4,6 +4,10 @@
|
|||||||
#ifndef DIFF_H
|
#ifndef DIFF_H
|
||||||
#define DIFF_H
|
#define DIFF_H
|
||||||
|
|
||||||
|
#define DIFF_FILE_CANON_MODE(mode) \
|
||||||
|
(S_ISREG(mode) ? (S_IFREG | ce_permissions(mode)) : \
|
||||||
|
S_ISLNK(mode) ? S_IFLNK : S_IFDIR)
|
||||||
|
|
||||||
extern void diff_addremove(int addremove,
|
extern void diff_addremove(int addremove,
|
||||||
unsigned mode,
|
unsigned mode,
|
||||||
const unsigned char *sha1,
|
const unsigned char *sha1,
|
||||||
|
@ -59,10 +59,6 @@ struct diff_filepair {
|
|||||||
|
|
||||||
#define DIFF_PAIR_MODE_CHANGED(p) ((p)->one->mode != (p)->two->mode)
|
#define DIFF_PAIR_MODE_CHANGED(p) ((p)->one->mode != (p)->two->mode)
|
||||||
|
|
||||||
#define DIFF_FILE_CANON_MODE(mode) \
|
|
||||||
(S_ISREG(mode) ? (S_IFREG | ce_permissions(mode)) : \
|
|
||||||
S_ISLNK(mode) ? S_IFLNK : S_IFDIR)
|
|
||||||
|
|
||||||
extern void diff_free_filepair(struct diff_filepair *);
|
extern void diff_free_filepair(struct diff_filepair *);
|
||||||
|
|
||||||
extern int diff_unmodified_pair(struct diff_filepair *);
|
extern int diff_unmodified_pair(struct diff_filepair *);
|
||||||
|
Reference in New Issue
Block a user