Browse Source

[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>
maint
Junio C Hamano 20 years ago committed by Linus Torvalds
parent
commit
67574c403f
  1. 8
      diff-files.c
  2. 4
      diff-tree.c
  3. 12
      diff.c
  4. 4
      diff.h
  5. 4
      diffcore.h

8
diff-files.c

@ -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,

4
diff-tree.c

@ -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

@ -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,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,

4
diffcore.h

@ -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 *);

Loading…
Cancel
Save