Browse Source

Merge branches 'jc/nostat' and 'jc/empty-commit'

* jc/nostat:
  ls-files: debugging aid for CE_VALID changes.
  "Assume unchanged" git: do not set CE_VALID with --refresh
  "Assume unchanged" git

* jc/empty-commit:
  t6000: fix a careless test library add-on.
  Do not allow empty name or email.
maint
Junio C Hamano 19 years ago
parent
commit
3c91b216ab
  1. 2
      apply.c
  2. 6
      cache.h
  3. 1
      checkout-index.c
  4. 5
      config.c
  5. 2
      diff-files.c
  6. 2
      diff-index.c
  7. 2
      diff.c
  8. 2
      entry.c
  9. 1
      environment.c
  10. 5
      ident.c
  11. 18
      ls-files.c
  12. 28
      read-cache.c
  13. 2
      read-tree.c
  14. 5
      t/t6000lib.sh
  15. 74
      update-index.c
  16. 2
      write-tree.c

2
apply.c

@ -1309,7 +1309,7 @@ static int check_patch(struct patch *patch)
return -1; return -1;
} }


changed = ce_match_stat(active_cache[pos], &st); changed = ce_match_stat(active_cache[pos], &st, 1);
if (changed) if (changed)
return error("%s: does not match index", return error("%s: does not match index",
old_name); old_name);

6
cache.h

@ -91,6 +91,7 @@ struct cache_entry {
#define CE_NAMEMASK (0x0fff) #define CE_NAMEMASK (0x0fff)
#define CE_STAGEMASK (0x3000) #define CE_STAGEMASK (0x3000)
#define CE_UPDATE (0x4000) #define CE_UPDATE (0x4000)
#define CE_VALID (0x8000)
#define CE_STAGESHIFT 12 #define CE_STAGESHIFT 12


#define create_ce_flags(len, stage) htons((len) | ((stage) << CE_STAGESHIFT)) #define create_ce_flags(len, stage) htons((len) | ((stage) << CE_STAGESHIFT))
@ -144,8 +145,8 @@ extern int add_cache_entry(struct cache_entry *ce, int option);
extern int remove_cache_entry_at(int pos); extern int remove_cache_entry_at(int pos);
extern int remove_file_from_cache(const char *path); extern int remove_file_from_cache(const char *path);
extern int ce_same_name(struct cache_entry *a, struct cache_entry *b); extern int ce_same_name(struct cache_entry *a, struct cache_entry *b);
extern int ce_match_stat(struct cache_entry *ce, struct stat *st); extern int ce_match_stat(struct cache_entry *ce, struct stat *st, int);
extern int ce_modified(struct cache_entry *ce, struct stat *st); extern int ce_modified(struct cache_entry *ce, struct stat *st, int);
extern int ce_path_match(const struct cache_entry *ce, const char **pathspec); extern int ce_path_match(const struct cache_entry *ce, const char **pathspec);
extern int index_fd(unsigned char *sha1, int fd, struct stat *st, int write_object, const char *type); extern int index_fd(unsigned char *sha1, int fd, struct stat *st, int write_object, const char *type);
extern int index_pipe(unsigned char *sha1, int fd, const char *type, int write_object); extern int index_pipe(unsigned char *sha1, int fd, const char *type, int write_object);
@ -161,6 +162,7 @@ extern int commit_index_file(struct cache_file *);
extern void rollback_index_file(struct cache_file *); extern void rollback_index_file(struct cache_file *);


extern int trust_executable_bit; extern int trust_executable_bit;
extern int assume_unchanged;
extern int only_use_symrefs; extern int only_use_symrefs;
extern int diff_rename_limit_default; extern int diff_rename_limit_default;
extern int shared_repository; extern int shared_repository;

1
checkout-index.c

@ -116,6 +116,7 @@ int main(int argc, char **argv)
int all = 0; int all = 0;


prefix = setup_git_directory(); prefix = setup_git_directory();
git_config(git_default_config);
prefix_length = prefix ? strlen(prefix) : 0; prefix_length = prefix ? strlen(prefix) : 0;


if (read_cache() < 0) { if (read_cache() < 0) {

5
config.c

@ -222,6 +222,11 @@ int git_default_config(const char *var, const char *value)
return 0; return 0;
} }


if (!strcmp(var, "core.ignorestat")) {
assume_unchanged = git_config_bool(var, value);
return 0;
}

if (!strcmp(var, "core.symrefsonly")) { if (!strcmp(var, "core.symrefsonly")) {
only_use_symrefs = git_config_bool(var, value); only_use_symrefs = git_config_bool(var, value);
return 0; return 0;

2
diff-files.c

@ -191,7 +191,7 @@ int main(int argc, const char **argv)
show_file('-', ce); show_file('-', ce);
continue; continue;
} }
changed = ce_match_stat(ce, &st); changed = ce_match_stat(ce, &st, 0);
if (!changed && !diff_options.find_copies_harder) if (!changed && !diff_options.find_copies_harder)
continue; continue;
oldmode = ntohl(ce->ce_mode); oldmode = ntohl(ce->ce_mode);

2
diff-index.c

@ -33,7 +33,7 @@ static int get_stat_data(struct cache_entry *ce,
} }
return -1; return -1;
} }
changed = ce_match_stat(ce, &st); changed = ce_match_stat(ce, &st, 0);
if (changed) { if (changed) {
mode = create_ce_mode(st.st_mode); mode = create_ce_mode(st.st_mode);
if (!trust_executable_bit && if (!trust_executable_bit &&

2
diff.c

@ -311,7 +311,7 @@ static int work_tree_matches(const char *name, const unsigned char *sha1)
ce = active_cache[pos]; ce = active_cache[pos];
if ((lstat(name, &st) < 0) || if ((lstat(name, &st) < 0) ||
!S_ISREG(st.st_mode) || /* careful! */ !S_ISREG(st.st_mode) || /* careful! */
ce_match_stat(ce, &st) || ce_match_stat(ce, &st, 0) ||
memcmp(sha1, ce->sha1, 20)) memcmp(sha1, ce->sha1, 20))
return 0; return 0;
/* we return 1 only when we can stat, it is a regular file, /* we return 1 only when we can stat, it is a regular file,

2
entry.c

@ -123,7 +123,7 @@ int checkout_entry(struct cache_entry *ce, struct checkout *state)
strcpy(path + len, ce->name); strcpy(path + len, ce->name);


if (!lstat(path, &st)) { if (!lstat(path, &st)) {
unsigned changed = ce_match_stat(ce, &st); unsigned changed = ce_match_stat(ce, &st, 1);
if (!changed) if (!changed)
return 0; return 0;
if (!state->force) { if (!state->force) {

1
environment.c

@ -12,6 +12,7 @@
char git_default_email[MAX_GITNAME]; char git_default_email[MAX_GITNAME];
char git_default_name[MAX_GITNAME]; char git_default_name[MAX_GITNAME];
int trust_executable_bit = 1; int trust_executable_bit = 1;
int assume_unchanged = 0;
int only_use_symrefs = 0; int only_use_symrefs = 0;
int repository_format_version = 0; int repository_format_version = 0;
char git_commit_encoding[MAX_ENCODING_LENGTH] = "utf-8"; char git_commit_encoding[MAX_ENCODING_LENGTH] = "utf-8";

5
ident.c

@ -167,6 +167,11 @@ static const char *get_ident(const char *name, const char *email,
name = git_default_name; name = git_default_name;
if (!email) if (!email)
email = git_default_email; email = git_default_email;

if (!*name || !*email)
die("empty ident %s <%s> not allowed",
name, email);

strcpy(date, git_default_date); strcpy(date, git_default_date);
if (date_str) if (date_str)
parse_date(date_str, date, sizeof(date)); parse_date(date_str, date, sizeof(date));

18
ls-files.c

@ -447,6 +447,22 @@ static void show_ce_entry(const char *tag, struct cache_entry *ce)
if (pathspec && !match(pathspec, ce->name, len)) if (pathspec && !match(pathspec, ce->name, len))
return; return;


if (tag && *tag && (ce->ce_flags & htons(CE_VALID))) {
static char alttag[4];
memcpy(alttag, tag, 3);
if (isalpha(tag[0]))
alttag[0] = tolower(tag[0]);
else if (tag[0] == '?')
alttag[0] = '!';
else {
alttag[0] = 'v';
alttag[1] = tag[0];
alttag[2] = ' ';
alttag[3] = 0;
}
tag = alttag;
}

if (!show_stage) { if (!show_stage) {
fputs(tag, stdout); fputs(tag, stdout);
write_name_quoted("", 0, ce->name + offset, write_name_quoted("", 0, ce->name + offset,
@ -503,7 +519,7 @@ static void show_files(void)
err = lstat(ce->name, &st); err = lstat(ce->name, &st);
if (show_deleted && err) if (show_deleted && err)
show_ce_entry(tag_removed, ce); show_ce_entry(tag_removed, ce);
if (show_modified && ce_modified(ce, &st)) if (show_modified && ce_modified(ce, &st, 0))
show_ce_entry(tag_modified, ce); show_ce_entry(tag_modified, ce);
} }
} }

28
read-cache.c

@ -27,6 +27,9 @@ void fill_stat_cache_info(struct cache_entry *ce, struct stat *st)
ce->ce_uid = htonl(st->st_uid); ce->ce_uid = htonl(st->st_uid);
ce->ce_gid = htonl(st->st_gid); ce->ce_gid = htonl(st->st_gid);
ce->ce_size = htonl(st->st_size); ce->ce_size = htonl(st->st_size);

if (assume_unchanged)
ce->ce_flags |= htons(CE_VALID);
} }


static int ce_compare_data(struct cache_entry *ce, struct stat *st) static int ce_compare_data(struct cache_entry *ce, struct stat *st)
@ -146,9 +149,18 @@ static int ce_match_stat_basic(struct cache_entry *ce, struct stat *st)
return changed; return changed;
} }


int ce_match_stat(struct cache_entry *ce, struct stat *st) int ce_match_stat(struct cache_entry *ce, struct stat *st, int ignore_valid)
{ {
unsigned int changed = ce_match_stat_basic(ce, st); unsigned int changed;

/*
* If it's marked as always valid in the index, it's
* valid whatever the checked-out copy says.
*/
if (!ignore_valid && (ce->ce_flags & htons(CE_VALID)))
return 0;

changed = ce_match_stat_basic(ce, st);


/* /*
* Within 1 second of this sequence: * Within 1 second of this sequence:
@ -164,7 +176,7 @@ int ce_match_stat(struct cache_entry *ce, struct stat *st)
* effectively mean we can make at most one commit per second, * effectively mean we can make at most one commit per second,
* which is not acceptable. Instead, we check cache entries * which is not acceptable. Instead, we check cache entries
* whose mtime are the same as the index file timestamp more * whose mtime are the same as the index file timestamp more
* careful than others. * carefully than others.
*/ */
if (!changed && if (!changed &&
index_file_timestamp && index_file_timestamp &&
@ -174,10 +186,10 @@ int ce_match_stat(struct cache_entry *ce, struct stat *st)
return changed; return changed;
} }


int ce_modified(struct cache_entry *ce, struct stat *st) int ce_modified(struct cache_entry *ce, struct stat *st, int really)
{ {
int changed, changed_fs; int changed, changed_fs;
changed = ce_match_stat(ce, st); changed = ce_match_stat(ce, st, really);
if (!changed) if (!changed)
return 0; return 0;
/* /*
@ -233,6 +245,11 @@ int cache_name_compare(const char *name1, int flags1, const char *name2, int fla
return -1; return -1;
if (len1 > len2) if (len1 > len2)
return 1; return 1;

/* Differences between "assume up-to-date" should not matter. */
flags1 &= ~CE_VALID;
flags2 &= ~CE_VALID;

if (flags1 < flags2) if (flags1 < flags2)
return -1; return -1;
if (flags1 > flags2) if (flags1 > flags2)
@ -430,6 +447,7 @@ int add_cache_entry(struct cache_entry *ce, int option)
int ok_to_add = option & ADD_CACHE_OK_TO_ADD; int ok_to_add = option & ADD_CACHE_OK_TO_ADD;
int ok_to_replace = option & ADD_CACHE_OK_TO_REPLACE; int ok_to_replace = option & ADD_CACHE_OK_TO_REPLACE;
int skip_df_check = option & ADD_CACHE_SKIP_DFCHECK; int skip_df_check = option & ADD_CACHE_SKIP_DFCHECK;

pos = cache_name_pos(ce->name, ntohs(ce->ce_flags)); pos = cache_name_pos(ce->name, ntohs(ce->ce_flags));


/* existing match? Just replace it. */ /* existing match? Just replace it. */

2
read-tree.c

@ -349,7 +349,7 @@ static void verify_uptodate(struct cache_entry *ce)
return; return;


if (!lstat(ce->name, &st)) { if (!lstat(ce->name, &st)) {
unsigned changed = ce_match_stat(ce, &st); unsigned changed = ce_match_stat(ce, &st, 1);
if (!changed) if (!changed)
return; return;
errno = 0; errno = 0;

5
t/t6000lib.sh

@ -51,7 +51,12 @@ as_author()


export GIT_AUTHOR_EMAIL="$_author" export GIT_AUTHOR_EMAIL="$_author"
"$@" "$@"
if test -z "$_save"
then
unset GIT_AUTHOR_EMAIL
else
export GIT_AUTHOR_EMAIL="$_save" export GIT_AUTHOR_EMAIL="$_save"
fi
} }


commit_date() commit_date()

74
update-index.c

@ -23,6 +23,10 @@ static int quiet; /* --refresh needing update is not error */
static int info_only; static int info_only;
static int force_remove; static int force_remove;
static int verbose; static int verbose;
static int mark_valid_only = 0;
#define MARK_VALID 1
#define UNMARK_VALID 2



/* Three functions to allow overloaded pointer return; see linux/err.h */ /* Three functions to allow overloaded pointer return; see linux/err.h */
static inline void *ERR_PTR(long error) static inline void *ERR_PTR(long error)
@ -53,6 +57,25 @@ static void report(const char *fmt, ...)
va_end(vp); va_end(vp);
} }


static int mark_valid(const char *path)
{
int namelen = strlen(path);
int pos = cache_name_pos(path, namelen);
if (0 <= pos) {
switch (mark_valid_only) {
case MARK_VALID:
active_cache[pos]->ce_flags |= htons(CE_VALID);
break;
case UNMARK_VALID:
active_cache[pos]->ce_flags &= ~htons(CE_VALID);
break;
}
active_cache_changed = 1;
return 0;
}
return -1;
}

static int add_file_to_cache(const char *path) static int add_file_to_cache(const char *path)
{ {
int size, namelen, option, status; int size, namelen, option, status;
@ -94,6 +117,7 @@ static int add_file_to_cache(const char *path)
ce = xmalloc(size); ce = xmalloc(size);
memset(ce, 0, size); memset(ce, 0, size);
memcpy(ce->name, path, namelen); memcpy(ce->name, path, namelen);
ce->ce_flags = htons(namelen);
fill_stat_cache_info(ce, &st); fill_stat_cache_info(ce, &st);


ce->ce_mode = create_ce_mode(st.st_mode); ce->ce_mode = create_ce_mode(st.st_mode);
@ -105,7 +129,6 @@ static int add_file_to_cache(const char *path)
if (0 <= pos) if (0 <= pos)
ce->ce_mode = active_cache[pos]->ce_mode; ce->ce_mode = active_cache[pos]->ce_mode;
} }
ce->ce_flags = htons(namelen);


if (index_path(ce->sha1, path, &st, !info_only)) if (index_path(ce->sha1, path, &st, !info_only))
return -1; return -1;
@ -128,7 +151,7 @@ static int add_file_to_cache(const char *path)
* For example, you'd want to do this after doing a "git-read-tree", * For example, you'd want to do this after doing a "git-read-tree",
* to link up the stat cache details with the proper files. * to link up the stat cache details with the proper files.
*/ */
static struct cache_entry *refresh_entry(struct cache_entry *ce) static struct cache_entry *refresh_entry(struct cache_entry *ce, int really)
{ {
struct stat st; struct stat st;
struct cache_entry *updated; struct cache_entry *updated;
@ -137,21 +160,31 @@ static struct cache_entry *refresh_entry(struct cache_entry *ce)
if (lstat(ce->name, &st) < 0) if (lstat(ce->name, &st) < 0)
return ERR_PTR(-errno); return ERR_PTR(-errno);


changed = ce_match_stat(ce, &st); changed = ce_match_stat(ce, &st, really);
if (!changed) if (!changed)
return NULL; return NULL;


if (ce_modified(ce, &st)) if (ce_modified(ce, &st, really))
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);


size = ce_size(ce); size = ce_size(ce);
updated = xmalloc(size); updated = xmalloc(size);
memcpy(updated, ce, size); memcpy(updated, ce, size);
fill_stat_cache_info(updated, &st); fill_stat_cache_info(updated, &st);

/* In this case, if really is not set, we should leave
* CE_VALID bit alone. Otherwise, paths marked with
* --no-assume-unchanged (i.e. things to be edited) will
* reacquire CE_VALID bit automatically, which is not
* really what we want.
*/
if (!really && assume_unchanged && !(ce->ce_flags & htons(CE_VALID)))
updated->ce_flags &= ~htons(CE_VALID);

return updated; return updated;
} }


static int refresh_cache(void) static int refresh_cache(int really)
{ {
int i; int i;
int has_errors = 0; int has_errors = 0;
@ -171,12 +204,19 @@ static int refresh_cache(void)
continue; continue;
} }


new = refresh_entry(ce); new = refresh_entry(ce, really);
if (!new) if (!new)
continue; continue;
if (IS_ERR(new)) { if (IS_ERR(new)) {
if (not_new && PTR_ERR(new) == -ENOENT) if (not_new && PTR_ERR(new) == -ENOENT)
continue; continue;
if (really && PTR_ERR(new) == -EINVAL) {
/* If we are doing --really-refresh that
* means the index is not valid anymore.
*/
ce->ce_flags &= ~htons(CE_VALID);
active_cache_changed = 1;
}
if (quiet) if (quiet)
continue; continue;
printf("%s: needs update\n", ce->name); printf("%s: needs update\n", ce->name);
@ -274,6 +314,8 @@ static int add_cacheinfo(unsigned int mode, const unsigned char *sha1,
memcpy(ce->name, path, len); memcpy(ce->name, path, len);
ce->ce_flags = create_ce_flags(len, stage); ce->ce_flags = create_ce_flags(len, stage);
ce->ce_mode = create_ce_mode(mode); ce->ce_mode = create_ce_mode(mode);
if (assume_unchanged)
ce->ce_flags |= htons(CE_VALID);
option = allow_add ? ADD_CACHE_OK_TO_ADD : 0; option = allow_add ? ADD_CACHE_OK_TO_ADD : 0;
option |= allow_replace ? ADD_CACHE_OK_TO_REPLACE : 0; option |= allow_replace ? ADD_CACHE_OK_TO_REPLACE : 0;
if (add_cache_entry(ce, option)) if (add_cache_entry(ce, option))
@ -317,6 +359,12 @@ static void update_one(const char *path, const char *prefix, int prefix_length)
fprintf(stderr, "Ignoring path %s\n", path); fprintf(stderr, "Ignoring path %s\n", path);
return; return;
} }
if (mark_valid_only) {
if (mark_valid(p))
die("Unable to mark file %s", path);
return;
}

if (force_remove) { if (force_remove) {
if (remove_file_from_cache(p)) if (remove_file_from_cache(p))
die("git-update-index: unable to remove %s", path); die("git-update-index: unable to remove %s", path);
@ -467,7 +515,11 @@ int main(int argc, const char **argv)
continue; continue;
} }
if (!strcmp(path, "--refresh")) { if (!strcmp(path, "--refresh")) {
has_errors |= refresh_cache(); has_errors |= refresh_cache(0);
continue;
}
if (!strcmp(path, "--really-refresh")) {
has_errors |= refresh_cache(1);
continue; continue;
} }
if (!strcmp(path, "--cacheinfo")) { if (!strcmp(path, "--cacheinfo")) {
@ -493,6 +545,14 @@ int main(int argc, const char **argv)
die("git-update-index: %s cannot chmod %s", path, argv[i]); die("git-update-index: %s cannot chmod %s", path, argv[i]);
continue; continue;
} }
if (!strcmp(path, "--assume-unchanged")) {
mark_valid_only = MARK_VALID;
continue;
}
if (!strcmp(path, "--no-assume-unchanged")) {
mark_valid_only = UNMARK_VALID;
continue;
}
if (!strcmp(path, "--info-only")) { if (!strcmp(path, "--info-only")) {
info_only = 1; info_only = 1;
continue; continue;

2
write-tree.c

@ -111,7 +111,7 @@ int main(int argc, char **argv)
funny = 0; funny = 0;
for (i = 0; i < entries; i++) { for (i = 0; i < entries; i++) {
struct cache_entry *ce = active_cache[i]; struct cache_entry *ce = active_cache[i];
if (ntohs(ce->ce_flags) & ~CE_NAMEMASK) { if (ce_stage(ce)) {
if (10 < ++funny) { if (10 < ++funny) {
fprintf(stderr, "...\n"); fprintf(stderr, "...\n");
break; break;

Loading…
Cancel
Save