Esempio n. 1
0
static int get_stat_data(struct cache_entry *ce,
                         const unsigned char **sha1p,
                         unsigned int *modep,
                         int cached, int match_missing,
                         unsigned *dirty_submodule, struct diff_options *diffopt)
{
    const unsigned char *sha1 = ce->sha1;
    unsigned int mode = ce->ce_mode;

    if (!cached && !ce_uptodate(ce)) {
        int changed;
        struct stat st;
        changed = check_removed(ce, &st);
        if (changed < 0)
            return -1;
        else if (changed) {
            if (match_missing) {
                *sha1p = sha1;
                *modep = mode;
                return 0;
            }
            return -1;
        }
        changed = match_stat_with_submodule(diffopt, ce, &st,
                                            0, dirty_submodule);
        if (changed) {
            mode = ce_mode_from_stat(ce, st.st_mode);
            sha1 = null_sha1;
        }
    }

    *sha1p = sha1;
    *modep = mode;
    return 0;
}
Esempio n. 2
0
static int get_stat_data(const struct cache_entry *ce,
			 const struct object_id **oidp,
			 unsigned int *modep,
			 int cached, int match_missing,
			 unsigned *dirty_submodule, struct diff_options *diffopt)
{
	const struct object_id *oid = &ce->oid;
	unsigned int mode = ce->ce_mode;

	if (!cached && !ce_uptodate(ce)) {
		int changed;
		struct stat st;
		changed = check_removed(ce, &st);
		if (changed < 0)
			return -1;
		else if (changed) {
			if (match_missing) {
				*oidp = oid;
				*modep = mode;
				return 0;
			}
			return -1;
		}
		changed = match_stat_with_submodule(diffopt, ce, &st,
						    0, dirty_submodule);
		if (changed) {
			mode = ce_mode_from_stat(ce, st.st_mode);
			oid = &null_oid;
		}
	}

	*oidp = oid;
	*modep = mode;
	return 0;
}
Esempio n. 3
0
int run_diff_files(struct rev_info *revs, unsigned int option)
{
    int entries, i;
    int diff_unmerged_stage = revs->max_count;
    int silent_on_removed = option & DIFF_SILENT_ON_REMOVED;
    unsigned ce_option = ((option & DIFF_RACY_IS_MODIFIED)
                          ? CE_MATCH_RACY_IS_DIRTY : 0);

    diff_set_mnemonic_prefix(&revs->diffopt, "i/", "w/");

    if (diff_unmerged_stage < 0)
        diff_unmerged_stage = 2;
    entries = active_nr;
    for (i = 0; i < entries; i++) {
        struct stat st;
        unsigned int oldmode, newmode;
        struct cache_entry *ce = active_cache[i];
        int changed;
        unsigned dirty_submodule = 0;

        if (diff_can_quit_early(&revs->diffopt))
            break;

        if (!ce_path_match(ce, &revs->prune_data))
            continue;

        if (ce_stage(ce)) {
            struct combine_diff_path *dpath;
            struct diff_filepair *pair;
            unsigned int wt_mode = 0;
            int num_compare_stages = 0;
            size_t path_len;

            path_len = ce_namelen(ce);

            dpath = xmalloc(combine_diff_path_size(5, path_len));
            dpath->path = (char *) &(dpath->parent[5]);

            dpath->next = NULL;
            dpath->len = path_len;
            memcpy(dpath->path, ce->name, path_len);
            dpath->path[path_len] = '\0';
            hashclr(dpath->sha1);
            memset(&(dpath->parent[0]), 0,
                   sizeof(struct combine_diff_parent)*5);

            changed = check_removed(ce, &st);
            if (!changed)
                wt_mode = ce_mode_from_stat(ce, st.st_mode);
            else {
                if (changed < 0) {
                    perror(ce->name);
                    continue;
                }
                if (silent_on_removed)
                    continue;
                wt_mode = 0;
            }
            dpath->mode = wt_mode;

            while (i < entries) {
                struct cache_entry *nce = active_cache[i];
                int stage;

                if (strcmp(ce->name, nce->name))
                    break;

                /* Stage #2 (ours) is the first parent,
                 * stage #3 (theirs) is the second.
                 */
                stage = ce_stage(nce);
                if (2 <= stage) {
                    int mode = nce->ce_mode;
                    num_compare_stages++;
                    hashcpy(dpath->parent[stage-2].sha1, nce->sha1);
                    dpath->parent[stage-2].mode = ce_mode_from_stat(nce, mode);
                    dpath->parent[stage-2].status =
                        DIFF_STATUS_MODIFIED;
                }

                /* diff against the proper unmerged stage */
                if (stage == diff_unmerged_stage)
                    ce = nce;
                i++;
            }
            /*
             * Compensate for loop update
             */
            i--;

            if (revs->combine_merges && num_compare_stages == 2) {
                show_combined_diff(dpath, 2,
                                   revs->dense_combined_merges,
                                   revs);
                free(dpath);
                continue;
            }
            free(dpath);
            dpath = NULL;

            /*
             * Show the diff for the 'ce' if we found the one
             * from the desired stage.
             */
            pair = diff_unmerge(&revs->diffopt, ce->name);
            if (wt_mode)
                pair->two->mode = wt_mode;
            if (ce_stage(ce) != diff_unmerged_stage)
                continue;
        }

        if (ce_uptodate(ce) || ce_skip_worktree(ce))
            continue;

        /* If CE_VALID is set, don't look at workdir for file removal */
        changed = (ce->ce_flags & CE_VALID) ? 0 : check_removed(ce, &st);
        if (changed) {
            if (changed < 0) {
                perror(ce->name);
                continue;
            }
            if (silent_on_removed)
                continue;
            diff_addremove(&revs->diffopt, '-', ce->ce_mode,
                           ce->sha1, ce->name, 0);
            continue;
        }
        changed = match_stat_with_submodule(&revs->diffopt, ce, &st,
                                            ce_option, &dirty_submodule);
        if (!changed && !dirty_submodule) {
            ce_mark_uptodate(ce);
            if (!DIFF_OPT_TST(&revs->diffopt, FIND_COPIES_HARDER))
                continue;
        }
        oldmode = ce->ce_mode;
        newmode = ce_mode_from_stat(ce, st.st_mode);
        diff_change(&revs->diffopt, oldmode, newmode,
                    ce->sha1, (changed ? null_sha1 : ce->sha1),
                    ce->name, 0, dirty_submodule);

    }
    diffcore_std(&revs->diffopt);
    diff_flush(&revs->diffopt);
    return 0;
}
Esempio n. 4
0
int run_diff_files(struct rev_info *revs, unsigned int option)
{
	int entries, i;
	int diff_unmerged_stage = revs->max_count;
	unsigned ce_option = ((option & DIFF_RACY_IS_MODIFIED)
			      ? CE_MATCH_RACY_IS_DIRTY : 0);
	uint64_t start = getnanotime();
	struct index_state *istate = revs->diffopt.repo->index;

	diff_set_mnemonic_prefix(&revs->diffopt, "i/", "w/");

	if (diff_unmerged_stage < 0)
		diff_unmerged_stage = 2;
	entries = istate->cache_nr;
	for (i = 0; i < entries; i++) {
		unsigned int oldmode, newmode;
		struct cache_entry *ce = istate->cache[i];
		int changed;
		unsigned dirty_submodule = 0;
		const struct object_id *old_oid, *new_oid;

		if (diff_can_quit_early(&revs->diffopt))
			break;

		if (!ce_path_match(istate, ce, &revs->prune_data, NULL))
			continue;

		if (ce_stage(ce)) {
			struct combine_diff_path *dpath;
			struct diff_filepair *pair;
			unsigned int wt_mode = 0;
			int num_compare_stages = 0;
			size_t path_len;
			struct stat st;

			path_len = ce_namelen(ce);

			dpath = xmalloc(combine_diff_path_size(5, path_len));
			dpath->path = (char *) &(dpath->parent[5]);

			dpath->next = NULL;
			memcpy(dpath->path, ce->name, path_len);
			dpath->path[path_len] = '\0';
			oidclr(&dpath->oid);
			memset(&(dpath->parent[0]), 0,
			       sizeof(struct combine_diff_parent)*5);

			changed = check_removed(ce, &st);
			if (!changed)
				wt_mode = ce_mode_from_stat(ce, st.st_mode);
			else {
				if (changed < 0) {
					perror(ce->name);
					continue;
				}
				wt_mode = 0;
			}
			dpath->mode = wt_mode;

			while (i < entries) {
				struct cache_entry *nce = istate->cache[i];
				int stage;

				if (strcmp(ce->name, nce->name))
					break;

				/* Stage #2 (ours) is the first parent,
				 * stage #3 (theirs) is the second.
				 */
				stage = ce_stage(nce);
				if (2 <= stage) {
					int mode = nce->ce_mode;
					num_compare_stages++;
					oidcpy(&dpath->parent[stage - 2].oid,
					       &nce->oid);
					dpath->parent[stage-2].mode = ce_mode_from_stat(nce, mode);
					dpath->parent[stage-2].status =
						DIFF_STATUS_MODIFIED;
				}

				/* diff against the proper unmerged stage */
				if (stage == diff_unmerged_stage)
					ce = nce;
				i++;
			}
			/*
			 * Compensate for loop update
			 */
			i--;

			if (revs->combine_merges && num_compare_stages == 2) {
				show_combined_diff(dpath, 2,
						   revs->dense_combined_merges,
						   revs);
				free(dpath);
				continue;
			}
			FREE_AND_NULL(dpath);

			/*
			 * Show the diff for the 'ce' if we found the one
			 * from the desired stage.
			 */
			pair = diff_unmerge(&revs->diffopt, ce->name);
			if (wt_mode)
				pair->two->mode = wt_mode;
			if (ce_stage(ce) != diff_unmerged_stage)
				continue;
		}

		if (ce_uptodate(ce) || ce_skip_worktree(ce))
			continue;

		/* If CE_VALID is set, don't look at workdir for file removal */
		if (ce->ce_flags & CE_VALID) {
			changed = 0;
			newmode = ce->ce_mode;
		} else {
			struct stat st;

			changed = check_removed(ce, &st);
			if (changed) {
				if (changed < 0) {
					perror(ce->name);
					continue;
				}
				diff_addremove(&revs->diffopt, '-', ce->ce_mode,
					       &ce->oid,
					       !is_null_oid(&ce->oid),
					       ce->name, 0);
				continue;
			} else if (revs->diffopt.ita_invisible_in_index &&
				   ce_intent_to_add(ce)) {
				diff_addremove(&revs->diffopt, '+', ce->ce_mode,
					       the_hash_algo->empty_tree, 0,
					       ce->name, 0);
				continue;
			}

			changed = match_stat_with_submodule(&revs->diffopt, ce, &st,
							    ce_option, &dirty_submodule);
			newmode = ce_mode_from_stat(ce, st.st_mode);
		}

		if (!changed && !dirty_submodule) {
			ce_mark_uptodate(ce);
			mark_fsmonitor_valid(ce);
			if (!revs->diffopt.flags.find_copies_harder)
				continue;
		}
		oldmode = ce->ce_mode;
		old_oid = &ce->oid;
		new_oid = changed ? &null_oid : &ce->oid;
		diff_change(&revs->diffopt, oldmode, newmode,
			    old_oid, new_oid,
			    !is_null_oid(old_oid),
			    !is_null_oid(new_oid),
			    ce->name, 0, dirty_submodule);

	}
	diffcore_std(&revs->diffopt);
	diff_flush(&revs->diffopt);
	trace_performance_since(start, "diff-files");
	return 0;
}