Blame src/iterator.c

Packit Service 20376f
/*
Packit Service 20376f
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit Service 20376f
 *
Packit Service 20376f
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit Service 20376f
 * a Linking Exception. For full terms see the included COPYING file.
Packit Service 20376f
 */
Packit Service 20376f
Packit Service 20376f
#include "iterator.h"
Packit Service 20376f
#include "tree.h"
Packit Service 20376f
#include "index.h"
Packit Service 20376f
Packit Service 20376f
#define GIT_ITERATOR_FIRST_ACCESS   (1 << 15)
Packit Service 20376f
#define GIT_ITERATOR_HONOR_IGNORES  (1 << 16)
Packit Service 20376f
#define GIT_ITERATOR_IGNORE_DOT_GIT (1 << 17)
Packit Service 20376f
Packit Service 20376f
#define iterator__flag(I,F) ((((git_iterator *)(I))->flags & GIT_ITERATOR_ ## F) != 0)
Packit Service 20376f
#define iterator__ignore_case(I)       iterator__flag(I,IGNORE_CASE)
Packit Service 20376f
#define iterator__include_trees(I)     iterator__flag(I,INCLUDE_TREES)
Packit Service 20376f
#define iterator__dont_autoexpand(I)   iterator__flag(I,DONT_AUTOEXPAND)
Packit Service 20376f
#define iterator__do_autoexpand(I)    !iterator__flag(I,DONT_AUTOEXPAND)
Packit Service 20376f
#define iterator__include_conflicts(I) iterator__flag(I,INCLUDE_CONFLICTS)
Packit Service 20376f
#define iterator__has_been_accessed(I) iterator__flag(I,FIRST_ACCESS)
Packit Service 20376f
#define iterator__honor_ignores(I)     iterator__flag(I,HONOR_IGNORES)
Packit Service 20376f
#define iterator__ignore_dot_git(I)    iterator__flag(I,IGNORE_DOT_GIT)
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
static void iterator_set_ignore_case(git_iterator *iter, bool ignore_case)
Packit Service 20376f
{
Packit Service 20376f
	if (ignore_case)
Packit Service 20376f
		iter->flags |= GIT_ITERATOR_IGNORE_CASE;
Packit Service 20376f
	else
Packit Service 20376f
		iter->flags &= ~GIT_ITERATOR_IGNORE_CASE;
Packit Service 20376f
Packit Service 20376f
	iter->strcomp = ignore_case ? git__strcasecmp : git__strcmp;
Packit Service 20376f
	iter->strncomp = ignore_case ? git__strncasecmp : git__strncmp;
Packit Service 20376f
	iter->prefixcomp = ignore_case ? git__prefixcmp_icase : git__prefixcmp;
Packit Service 20376f
	iter->entry_srch = ignore_case ? git_index_entry_isrch : git_index_entry_srch;
Packit Service 20376f
Packit Service 20376f
	git_vector_set_cmp(&iter->pathlist, (git_vector_cmp)iter->strcomp);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int iterator_range_init(
Packit Service 20376f
	git_iterator *iter, const char *start, const char *end)
Packit Service 20376f
{
Packit Service 20376f
	if (start && *start) {
Packit Service 20376f
		iter->start = git__strdup(start);
Packit Service 20376f
		GITERR_CHECK_ALLOC(iter->start);
Packit Service 20376f
Packit Service 20376f
		iter->start_len = strlen(iter->start);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (end && *end) {
Packit Service 20376f
		iter->end = git__strdup(end);
Packit Service 20376f
		GITERR_CHECK_ALLOC(iter->end);
Packit Service 20376f
Packit Service 20376f
		iter->end_len = strlen(iter->end);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	iter->started = (iter->start == NULL);
Packit Service 20376f
	iter->ended = false;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void iterator_range_free(git_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	if (iter->start) {
Packit Service 20376f
		git__free(iter->start);
Packit Service 20376f
		iter->start = NULL;
Packit Service 20376f
		iter->start_len = 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (iter->end) {
Packit Service 20376f
		git__free(iter->end);
Packit Service 20376f
		iter->end = NULL;
Packit Service 20376f
		iter->end_len = 0;
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int iterator_reset_range(
Packit Service 20376f
	git_iterator *iter, const char *start, const char *end)
Packit Service 20376f
{
Packit Service 20376f
	iterator_range_free(iter);
Packit Service 20376f
	return iterator_range_init(iter, start, end);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int iterator_pathlist_init(git_iterator *iter, git_strarray *pathlist)
Packit Service 20376f
{
Packit Service 20376f
	size_t i;
Packit Service 20376f
Packit Service 20376f
	if (git_vector_init(&iter->pathlist, pathlist->count, NULL) < 0)
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	for (i = 0; i < pathlist->count; i++) {
Packit Service 20376f
		if (!pathlist->strings[i])
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		if (git_vector_insert(&iter->pathlist, pathlist->strings[i]) < 0)
Packit Service 20376f
			return -1;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int iterator_init_common(
Packit Service 20376f
	git_iterator *iter,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_index *index,
Packit Service 20376f
	git_iterator_options *given_opts)
Packit Service 20376f
{
Packit Service 20376f
	static git_iterator_options default_opts = GIT_ITERATOR_OPTIONS_INIT;
Packit Service 20376f
	git_iterator_options *options = given_opts ? given_opts : &default_opts;
Packit Service 20376f
	bool ignore_case;
Packit Service 20376f
	int precompose;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	iter->repo = repo;
Packit Service 20376f
	iter->index = index;
Packit Service 20376f
	iter->flags = options->flags;
Packit Service 20376f
Packit Service 20376f
	if ((iter->flags & GIT_ITERATOR_IGNORE_CASE) != 0) {
Packit Service 20376f
		ignore_case = true;
Packit Service 20376f
	} else if ((iter->flags & GIT_ITERATOR_DONT_IGNORE_CASE) != 0) {
Packit Service 20376f
		ignore_case = false;
Packit Service 20376f
	} else if (repo) {
Packit Service 20376f
		git_index *index;
Packit Service 20376f
Packit Service 20376f
		if ((error = git_repository_index__weakptr(&index, iter->repo)) < 0)
Packit Service 20376f
			return error;
Packit Service 20376f
Packit Service 20376f
		ignore_case = !!index->ignore_case;
Packit Service 20376f
Packit Service 20376f
		if (ignore_case == 1)
Packit Service 20376f
			iter->flags |= GIT_ITERATOR_IGNORE_CASE;
Packit Service 20376f
		else
Packit Service 20376f
			iter->flags |= GIT_ITERATOR_DONT_IGNORE_CASE;
Packit Service 20376f
	} else {
Packit Service 20376f
		ignore_case = false;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* try to look up precompose and set flag if appropriate */
Packit Service 20376f
	if (repo &&
Packit Service 20376f
		(iter->flags & GIT_ITERATOR_PRECOMPOSE_UNICODE) == 0 &&
Packit Service 20376f
		(iter->flags & GIT_ITERATOR_DONT_PRECOMPOSE_UNICODE) == 0) {
Packit Service 20376f
Packit Service 20376f
		if (git_repository__cvar(&precompose, repo, GIT_CVAR_PRECOMPOSE) < 0)
Packit Service 20376f
			giterr_clear();
Packit Service 20376f
		else if (precompose)
Packit Service 20376f
			iter->flags |= GIT_ITERATOR_PRECOMPOSE_UNICODE;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if ((iter->flags & GIT_ITERATOR_DONT_AUTOEXPAND))
Packit Service 20376f
		iter->flags |= GIT_ITERATOR_INCLUDE_TREES;
Packit Service 20376f
Packit Service 20376f
	if ((error = iterator_range_init(iter, options->start, options->end)) < 0 ||
Packit Service 20376f
		(error = iterator_pathlist_init(iter, &options->pathlist)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	iterator_set_ignore_case(iter, ignore_case);
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void iterator_clear(git_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	iter->started = false;
Packit Service 20376f
	iter->ended = false;
Packit Service 20376f
	iter->stat_calls = 0;
Packit Service 20376f
	iter->pathlist_walk_idx = 0;
Packit Service 20376f
	iter->flags &= ~GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(bool) iterator_has_started(
Packit Service 20376f
	git_iterator *iter, const char *path, bool is_submodule)
Packit Service 20376f
{
Packit Service 20376f
	size_t path_len;
Packit Service 20376f
Packit Service 20376f
	if (iter->start == NULL || iter->started == true)
Packit Service 20376f
		return true;
Packit Service 20376f
Packit Service 20376f
	/* the starting path is generally a prefix - we have started once we
Packit Service 20376f
	 * are prefixed by this path
Packit Service 20376f
	 */
Packit Service 20376f
	iter->started = (iter->prefixcomp(path, iter->start) >= 0);
Packit Service 20376f
Packit Service 20376f
	if (iter->started)
Packit Service 20376f
		return true;
Packit Service 20376f
Packit Service 20376f
	path_len = strlen(path);
Packit Service 20376f
Packit Service 20376f
	/* if, however, we are a submodule, then we support `start` being
Packit Service 20376f
	 * suffixed with a `/` for crazy legacy reasons.  match `submod`
Packit Service 20376f
	 * with a start path of `submod/`.
Packit Service 20376f
	 */
Packit Service 20376f
	if (is_submodule && iter->start_len && path_len == iter->start_len - 1 &&
Packit Service 20376f
		iter->start[iter->start_len-1] == '/')
Packit Service 20376f
		return true;
Packit Service 20376f
Packit Service 20376f
	/* if, however, our current path is a directory, and our starting path
Packit Service 20376f
	 * is _beneath_ that directory, then recurse into the directory (even
Packit Service 20376f
	 * though we have not yet "started")
Packit Service 20376f
	 */
Packit Service 20376f
	if (path_len > 0 && path[path_len-1] == '/' &&
Packit Service 20376f
		iter->strncomp(path, iter->start, path_len) == 0)
Packit Service 20376f
		return true;
Packit Service 20376f
Packit Service 20376f
	return false;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(bool) iterator_has_ended(git_iterator *iter, const char *path)
Packit Service 20376f
{
Packit Service 20376f
	if (iter->end == NULL)
Packit Service 20376f
		return false;
Packit Service 20376f
	else if (iter->ended)
Packit Service 20376f
		return true;
Packit Service 20376f
Packit Service 20376f
	iter->ended = (iter->prefixcomp(path, iter->end) > 0);
Packit Service 20376f
	return iter->ended;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/* walker for the index and tree iterator that allows it to walk the sorted
Packit Service 20376f
 * pathlist entries alongside sorted iterator entries.
Packit Service 20376f
 */
Packit Service 20376f
static bool iterator_pathlist_next_is(git_iterator *iter, const char *path)
Packit Service 20376f
{
Packit Service 20376f
	char *p;
Packit Service 20376f
	size_t path_len, p_len, cmp_len, i;
Packit Service 20376f
	int cmp;
Packit Service 20376f
Packit Service 20376f
	if (iter->pathlist.length == 0)
Packit Service 20376f
		return true;
Packit Service 20376f
Packit Service 20376f
	git_vector_sort(&iter->pathlist);
Packit Service 20376f
Packit Service 20376f
	path_len = strlen(path);
Packit Service 20376f
Packit Service 20376f
	/* for comparison, drop the trailing slash on the current '/' */
Packit Service 20376f
	if (path_len && path[path_len-1] == '/')
Packit Service 20376f
		path_len--;
Packit Service 20376f
Packit Service 20376f
	for (i = iter->pathlist_walk_idx; i < iter->pathlist.length; i++) {
Packit Service 20376f
		p = iter->pathlist.contents[i];
Packit Service 20376f
		p_len = strlen(p);
Packit Service 20376f
Packit Service 20376f
		if (p_len && p[p_len-1] == '/')
Packit Service 20376f
			p_len--;
Packit Service 20376f
Packit Service 20376f
		cmp_len = min(path_len, p_len);
Packit Service 20376f
Packit Service 20376f
		/* see if the pathlist entry is a prefix of this path */
Packit Service 20376f
		cmp = iter->strncomp(p, path, cmp_len);
Packit Service 20376f
Packit Service 20376f
		/* prefix match - see if there's an exact match, or if we were
Packit Service 20376f
		 * given a path that matches the directory
Packit Service 20376f
		 */
Packit Service 20376f
		if (cmp == 0) {
Packit Service 20376f
			/* if this pathlist entry is not suffixed with a '/' then
Packit Service 20376f
			 * it matches a path that is a file or a directory.
Packit Service 20376f
			 * (eg, pathlist = "foo" and path is "foo" or "foo/" or
Packit Service 20376f
			 * "foo/something")
Packit Service 20376f
			 */
Packit Service 20376f
			if (p[cmp_len] == '\0' &&
Packit Service 20376f
				(path[cmp_len] == '\0' || path[cmp_len] == '/'))
Packit Service 20376f
				return true;
Packit Service 20376f
Packit Service 20376f
			/* if this pathlist entry _is_ suffixed with a '/' then
Packit Service 20376f
			 * it matches only paths that are directories.
Packit Service 20376f
			 * (eg, pathlist = "foo/" and path is "foo/" or "foo/something")
Packit Service 20376f
			 */
Packit Service 20376f
			if (p[cmp_len] == '/' && path[cmp_len] == '/')
Packit Service 20376f
				return true;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* this pathlist entry sorts before the given path, try the next */
Packit Service 20376f
		else if (cmp < 0) {
Packit Service 20376f
			iter->pathlist_walk_idx++;
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* this pathlist sorts after the given path, no match. */
Packit Service 20376f
		else if (cmp > 0) {
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return false;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
typedef enum {
Packit Service 20376f
	ITERATOR_PATHLIST_NONE = 0,
Packit Service 20376f
	ITERATOR_PATHLIST_IS_FILE = 1,
Packit Service 20376f
	ITERATOR_PATHLIST_IS_DIR = 2,
Packit Service 20376f
	ITERATOR_PATHLIST_IS_PARENT = 3,
Packit Service 20376f
	ITERATOR_PATHLIST_FULL = 4,
Packit Service 20376f
} iterator_pathlist_search_t;
Packit Service 20376f
Packit Service 20376f
static iterator_pathlist_search_t iterator_pathlist_search(
Packit Service 20376f
	git_iterator *iter, const char *path, size_t path_len)
Packit Service 20376f
{
Packit Service 20376f
	const char *p;
Packit Service 20376f
	size_t idx;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if (iter->pathlist.length == 0)
Packit Service 20376f
		return ITERATOR_PATHLIST_FULL;
Packit Service 20376f
Packit Service 20376f
	git_vector_sort(&iter->pathlist);
Packit Service 20376f
Packit Service 20376f
	error = git_vector_bsearch2(&idx, &iter->pathlist,
Packit Service 20376f
		(git_vector_cmp)iter->strcomp, path);
Packit Service 20376f
Packit Service 20376f
	/* the given path was found in the pathlist.  since the pathlist only
Packit Service 20376f
	 * matches directories when they're suffixed with a '/', analyze the
Packit Service 20376f
	 * path string to determine whether it's a directory or not.
Packit Service 20376f
	 */
Packit Service 20376f
	if (error == 0) {
Packit Service 20376f
		if (path_len && path[path_len-1] == '/')
Packit Service 20376f
			return ITERATOR_PATHLIST_IS_DIR;
Packit Service 20376f
Packit Service 20376f
		return ITERATOR_PATHLIST_IS_FILE;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* at this point, the path we're examining may be a directory (though we
Packit Service 20376f
	 * don't know that yet, since we're avoiding a stat unless it's necessary)
Packit Service 20376f
	 * so walk the pathlist looking for the given path with a '/' after it,
Packit Service 20376f
	 */
Packit Service 20376f
	while ((p = git_vector_get(&iter->pathlist, idx)) != NULL) {
Packit Service 20376f
		if (iter->prefixcomp(p, path) != 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		/* an exact match would have been matched by the bsearch above */
Packit Service 20376f
		assert(p[path_len]);
Packit Service 20376f
Packit Service 20376f
		/* is this a literal directory entry (eg `foo/`) or a file beneath */
Packit Service 20376f
		if (p[path_len] == '/') {
Packit Service 20376f
			return (p[path_len+1] == '\0') ?
Packit Service 20376f
				ITERATOR_PATHLIST_IS_DIR :
Packit Service 20376f
				ITERATOR_PATHLIST_IS_PARENT;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (p[path_len] > '/')
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		idx++;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return ITERATOR_PATHLIST_NONE;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/* Empty iterator */
Packit Service 20376f
Packit Service 20376f
static int empty_iterator_noop(const git_index_entry **e, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(i);
Packit Service 20376f
Packit Service 20376f
	if (e)
Packit Service 20376f
		*e = NULL;
Packit Service 20376f
Packit Service 20376f
	return GIT_ITEROVER;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int empty_iterator_advance_over(
Packit Service 20376f
	const git_index_entry **e,
Packit Service 20376f
	git_iterator_status_t *s,
Packit Service 20376f
	git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	*s = GIT_ITERATOR_STATUS_EMPTY;
Packit Service 20376f
	return empty_iterator_noop(e, i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int empty_iterator_reset(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(i);
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void empty_iterator_free(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	GIT_UNUSED(i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_iterator base;
Packit Service 20376f
	git_iterator_callbacks cb;
Packit Service 20376f
} empty_iterator;
Packit Service 20376f
Packit Service 20376f
int git_iterator_for_nothing(
Packit Service 20376f
	git_iterator **out,
Packit Service 20376f
	git_iterator_options *options)
Packit Service 20376f
{
Packit Service 20376f
	empty_iterator *iter;
Packit Service 20376f
Packit Service 20376f
	static git_iterator_callbacks callbacks = {
Packit Service 20376f
		empty_iterator_noop,
Packit Service 20376f
		empty_iterator_noop,
Packit Service 20376f
		empty_iterator_noop,
Packit Service 20376f
		empty_iterator_advance_over,
Packit Service 20376f
		empty_iterator_reset,
Packit Service 20376f
		empty_iterator_free
Packit Service 20376f
	};
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	iter = git__calloc(1, sizeof(empty_iterator));
Packit Service 20376f
	GITERR_CHECK_ALLOC(iter);
Packit Service 20376f
Packit Service 20376f
	iter->base.type = GIT_ITERATOR_TYPE_EMPTY;
Packit Service 20376f
	iter->base.cb = &callbacks;
Packit Service 20376f
	iter->base.flags = options->flags;
Packit Service 20376f
Packit Service 20376f
	*out = &iter->base;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/* Tree iterator */
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_tree_entry *tree_entry;
Packit Service 20376f
	const char *parent_path;
Packit Service 20376f
} tree_iterator_entry;
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_tree *tree;
Packit Service 20376f
Packit Service 20376f
	/* path to this particular frame (folder) */
Packit Service 20376f
	git_buf path;
Packit Service 20376f
Packit Service 20376f
	/* a sorted list of the entries for this frame (folder), these are
Packit Service 20376f
	 * actually pointers to the iterator's entry pool.
Packit Service 20376f
	 */
Packit Service 20376f
	git_vector entries;
Packit Service 20376f
	tree_iterator_entry *current;
Packit Service 20376f
Packit Service 20376f
	size_t next_idx;
Packit Service 20376f
Packit Service 20376f
	/* on case insensitive iterations, we also have an array of other
Packit Service 20376f
	 * paths that were case insensitively equal to this one, and their
Packit Service 20376f
	 * tree objects.  we have coalesced the tree entries into this frame.
Packit Service 20376f
	 * a child `tree_iterator_entry` will contain a pointer to its actual
Packit Service 20376f
	 * parent path.
Packit Service 20376f
	 */
Packit Service 20376f
	git_vector similar_trees;
Packit Service 20376f
	git_array_t(git_buf) similar_paths;
Packit Service 20376f
} tree_iterator_frame;
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_iterator base;
Packit Service 20376f
	git_tree *root;
Packit Service 20376f
	git_array_t(tree_iterator_frame) frames;
Packit Service 20376f
Packit Service 20376f
	git_index_entry entry;
Packit Service 20376f
	git_buf entry_path;
Packit Service 20376f
Packit Service 20376f
	/* a pool of entries to reduce the number of allocations */
Packit Service 20376f
	git_pool entry_pool;
Packit Service 20376f
} tree_iterator;
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(tree_iterator_frame *) tree_iterator_parent_frame(
Packit Service 20376f
	tree_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	return iter->frames.size > 1 ?
Packit Service 20376f
		&iter->frames.ptr[iter->frames.size-2] : NULL;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(tree_iterator_frame *) tree_iterator_current_frame(
Packit Service 20376f
	tree_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	return iter->frames.size ? &iter->frames.ptr[iter->frames.size-1] : NULL;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(int) tree_entry_cmp(
Packit Service 20376f
	const git_tree_entry *a, const git_tree_entry *b, bool icase)
Packit Service 20376f
{
Packit Service 20376f
	return git_path_cmp(
Packit Service 20376f
		a->filename, a->filename_len, a->attr == GIT_FILEMODE_TREE,
Packit Service 20376f
		b->filename, b->filename_len, b->attr == GIT_FILEMODE_TREE,
Packit Service 20376f
		icase ? git__strncasecmp : git__strncmp);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(int) tree_iterator_entry_cmp(const void *ptr_a, const void *ptr_b)
Packit Service 20376f
{
Packit Service 20376f
	const tree_iterator_entry *a = (const tree_iterator_entry *)ptr_a;
Packit Service 20376f
	const tree_iterator_entry *b = (const tree_iterator_entry *)ptr_b;
Packit Service 20376f
Packit Service 20376f
	return tree_entry_cmp(a->tree_entry, b->tree_entry, false);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(int) tree_iterator_entry_cmp_icase(
Packit Service 20376f
	const void *ptr_a, const void *ptr_b)
Packit Service 20376f
{
Packit Service 20376f
	const tree_iterator_entry *a = (const tree_iterator_entry *)ptr_a;
Packit Service 20376f
	const tree_iterator_entry *b = (const tree_iterator_entry *)ptr_b;
Packit Service 20376f
Packit Service 20376f
	return tree_entry_cmp(a->tree_entry, b->tree_entry, true);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_entry_sort_icase(const void *ptr_a, const void *ptr_b)
Packit Service 20376f
{
Packit Service 20376f
	const tree_iterator_entry *a = (const tree_iterator_entry *)ptr_a;
Packit Service 20376f
	const tree_iterator_entry *b = (const tree_iterator_entry *)ptr_b;
Packit Service 20376f
Packit Service 20376f
	int c = tree_entry_cmp(a->tree_entry, b->tree_entry, true);
Packit Service 20376f
Packit Service 20376f
	/* stabilize the sort order for filenames that are (case insensitively)
Packit Service 20376f
	 * the same by examining the parent path (case sensitively) before
Packit Service 20376f
	 * falling back to a case sensitive sort of the filename.
Packit Service 20376f
	 */
Packit Service 20376f
	if (!c && a->parent_path != b->parent_path)
Packit Service 20376f
		c = git__strcmp(a->parent_path, b->parent_path);
Packit Service 20376f
Packit Service 20376f
	if (!c)
Packit Service 20376f
		c = tree_entry_cmp(a->tree_entry, b->tree_entry, false);
Packit Service 20376f
Packit Service 20376f
	return c;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_compute_path(
Packit Service 20376f
	git_buf *out,
Packit Service 20376f
	tree_iterator_entry *entry)
Packit Service 20376f
{
Packit Service 20376f
	git_buf_clear(out);
Packit Service 20376f
Packit Service 20376f
	if (entry->parent_path)
Packit Service 20376f
		git_buf_joinpath(out, entry->parent_path, entry->tree_entry->filename);
Packit Service 20376f
	else
Packit Service 20376f
		git_buf_puts(out, entry->tree_entry->filename);
Packit Service 20376f
Packit Service 20376f
	if (git_tree_entry__is_tree(entry->tree_entry))
Packit Service 20376f
		git_buf_putc(out, '/');
Packit Service 20376f
Packit Service 20376f
	if (git_buf_oom(out))
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_frame_init(
Packit Service 20376f
	tree_iterator *iter,
Packit Service 20376f
	git_tree *tree,
Packit Service 20376f
	tree_iterator_entry *frame_entry)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator_frame *new_frame = NULL;
Packit Service 20376f
	tree_iterator_entry *new_entry;
Packit Service 20376f
	git_tree *dup = NULL;
Packit Service 20376f
	git_tree_entry *tree_entry;
Packit Service 20376f
	git_vector_cmp cmp;
Packit Service 20376f
	size_t i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	new_frame = git_array_alloc(iter->frames);
Packit Service 20376f
	GITERR_CHECK_ALLOC(new_frame);
Packit Service 20376f
Packit Service 20376f
	memset(new_frame, 0, sizeof(tree_iterator_frame));
Packit Service 20376f
Packit Service 20376f
	if ((error = git_tree_dup(&dup, tree)) < 0)
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	memset(new_frame, 0x0, sizeof(tree_iterator_frame));
Packit Service 20376f
	new_frame->tree = dup;
Packit Service 20376f
Packit Service 20376f
	if (frame_entry &&
Packit Service 20376f
		(error = tree_iterator_compute_path(&new_frame->path, frame_entry)) < 0)
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	cmp = iterator__ignore_case(&iter->base) ?
Packit Service 20376f
		tree_iterator_entry_sort_icase : NULL;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_vector_init(
Packit Service 20376f
		&new_frame->entries, dup->entries.size, cmp)) < 0)
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	git_array_foreach(dup->entries, i, tree_entry) {
Packit Service 20376f
		new_entry = git_pool_malloc(&iter->entry_pool, 1);
Packit Service 20376f
		GITERR_CHECK_ALLOC(new_entry);
Packit Service 20376f
Packit Service 20376f
		new_entry->tree_entry = tree_entry;
Packit Service 20376f
		new_entry->parent_path = new_frame->path.ptr;
Packit Service 20376f
Packit Service 20376f
		if ((error = git_vector_insert(&new_frame->entries, new_entry)) < 0)
Packit Service 20376f
			goto done;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	git_vector_set_sorted(&new_frame->entries,
Packit Service 20376f
		!iterator__ignore_case(&iter->base));
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	if (error < 0) {
Packit Service 20376f
		git_tree_free(dup);
Packit Service 20376f
		git_array_pop(iter->frames);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(tree_iterator_entry *) tree_iterator_current_entry(
Packit Service 20376f
	tree_iterator_frame *frame)
Packit Service 20376f
{
Packit Service 20376f
	return frame->current;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(int) tree_iterator_frame_push_neighbors(
Packit Service 20376f
	tree_iterator *iter,
Packit Service 20376f
	tree_iterator_frame *parent_frame,
Packit Service 20376f
	tree_iterator_frame *frame,
Packit Service 20376f
	const char *filename)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator_entry *entry, *new_entry;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
	git_tree_entry *tree_entry;
Packit Service 20376f
	git_buf *path;
Packit Service 20376f
	size_t new_size, i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	while (parent_frame->next_idx < parent_frame->entries.length) {
Packit Service 20376f
		entry = parent_frame->entries.contents[parent_frame->next_idx];
Packit Service 20376f
Packit Service 20376f
		if (strcasecmp(filename, entry->tree_entry->filename) != 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		if ((error = git_tree_lookup(&tree,
Packit Service 20376f
			iter->base.repo, entry->tree_entry->oid)) < 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		if (git_vector_insert(&parent_frame->similar_trees, tree) < 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		path = git_array_alloc(parent_frame->similar_paths);
Packit Service 20376f
		GITERR_CHECK_ALLOC(path);
Packit Service 20376f
Packit Service 20376f
		memset(path, 0, sizeof(git_buf));
Packit Service 20376f
Packit Service 20376f
		if ((error = tree_iterator_compute_path(path, entry)) < 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		GITERR_CHECK_ALLOC_ADD(&new_size,
Packit Service 20376f
			frame->entries.length, tree->entries.size);
Packit Service 20376f
		git_vector_size_hint(&frame->entries, new_size);
Packit Service 20376f
Packit Service 20376f
		git_array_foreach(tree->entries, i, tree_entry) {
Packit Service 20376f
			new_entry = git_pool_malloc(&iter->entry_pool, 1);
Packit Service 20376f
			GITERR_CHECK_ALLOC(new_entry);
Packit Service 20376f
Packit Service 20376f
			new_entry->tree_entry = tree_entry;
Packit Service 20376f
			new_entry->parent_path = path->ptr;
Packit Service 20376f
Packit Service 20376f
			if ((error = git_vector_insert(&frame->entries, new_entry)) < 0)
Packit Service 20376f
				break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (error)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		parent_frame->next_idx++;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(int) tree_iterator_frame_push(
Packit Service 20376f
	tree_iterator *iter, tree_iterator_entry *entry)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator_frame *parent_frame, *frame;
Packit Service 20376f
	git_tree *tree = NULL;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_tree_lookup(&tree,
Packit Service 20376f
			iter->base.repo, entry->tree_entry->oid)) < 0 ||
Packit Service 20376f
		(error = tree_iterator_frame_init(iter, tree, entry)) < 0)
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	parent_frame = tree_iterator_parent_frame(iter);
Packit Service 20376f
	frame = tree_iterator_current_frame(iter);
Packit Service 20376f
Packit Service 20376f
	/* if we're case insensitive, then we may have another directory that
Packit Service 20376f
	 * is (case insensitively) equal to this one.  coalesce those children
Packit Service 20376f
	 * into this tree.
Packit Service 20376f
	 */
Packit Service 20376f
	if (iterator__ignore_case(&iter->base))
Packit Service 20376f
		error = tree_iterator_frame_push_neighbors(iter,
Packit Service 20376f
			parent_frame, frame, entry->tree_entry->filename);
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	git_tree_free(tree);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void tree_iterator_frame_pop(tree_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator_frame *frame;
Packit Service 20376f
	git_buf *buf = NULL;
Packit Service 20376f
	git_tree *tree;
Packit Service 20376f
	size_t i;
Packit Service 20376f
Packit Service 20376f
	assert(iter->frames.size);
Packit Service 20376f
Packit Service 20376f
	frame = git_array_pop(iter->frames);
Packit Service 20376f
Packit Service 20376f
	git_vector_free(&frame->entries);
Packit Service 20376f
	git_tree_free(frame->tree);
Packit Service 20376f
Packit Service 20376f
	do {
Packit Service 20376f
		buf = git_array_pop(frame->similar_paths);
Packit Service 20376f
		git_buf_free(buf);
Packit Service 20376f
	} while (buf != NULL);
Packit Service 20376f
Packit Service 20376f
	git_array_clear(frame->similar_paths);
Packit Service 20376f
Packit Service 20376f
	git_vector_foreach(&frame->similar_trees, i, tree)
Packit Service 20376f
		git_tree_free(tree);
Packit Service 20376f
Packit Service 20376f
	git_vector_free(&frame->similar_trees);
Packit Service 20376f
Packit Service 20376f
	git_buf_free(&frame->path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_current(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter = (tree_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	if (!iterator__has_been_accessed(i))
Packit Service 20376f
		return iter->base.cb->advance(out, i);
Packit Service 20376f
Packit Service 20376f
	if (!iter->frames.size) {
Packit Service 20376f
		*out = NULL;
Packit Service 20376f
		return GIT_ITEROVER;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	*out = &iter->entry;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void tree_iterator_set_current(
Packit Service 20376f
	tree_iterator *iter,
Packit Service 20376f
	tree_iterator_frame *frame,
Packit Service 20376f
	tree_iterator_entry *entry)
Packit Service 20376f
{
Packit Service 20376f
	git_tree_entry *tree_entry = entry->tree_entry;
Packit Service 20376f
Packit Service 20376f
	frame->current = entry;
Packit Service 20376f
Packit Service 20376f
	memset(&iter->entry, 0x0, sizeof(git_index_entry));
Packit Service 20376f
Packit Service 20376f
	iter->entry.mode = tree_entry->attr;
Packit Service 20376f
	iter->entry.path = iter->entry_path.ptr;
Packit Service 20376f
	git_oid_cpy(&iter->entry.id, tree_entry->oid);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_advance(const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter = (tree_iterator *)i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	iter->base.flags |= GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
Packit Service 20376f
	/* examine tree entries until we find the next one to return */
Packit Service 20376f
	while (true) {
Packit Service 20376f
		tree_iterator_entry *prev_entry, *entry;
Packit Service 20376f
		tree_iterator_frame *frame;
Packit Service 20376f
		bool is_tree;
Packit Service 20376f
Packit Service 20376f
		if ((frame = tree_iterator_current_frame(iter)) == NULL) {
Packit Service 20376f
			error = GIT_ITEROVER;
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* no more entries in this frame.  pop the frame out */
Packit Service 20376f
		if (frame->next_idx == frame->entries.length) {
Packit Service 20376f
			tree_iterator_frame_pop(iter);
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* we may have coalesced the contents of case-insensitively same-named
Packit Service 20376f
		 * directories, so do the sort now.
Packit Service 20376f
		 */
Packit Service 20376f
		if (frame->next_idx == 0 && !git_vector_is_sorted(&frame->entries))
Packit Service 20376f
			git_vector_sort(&frame->entries);
Packit Service 20376f
Packit Service 20376f
		/* we have more entries in the current frame, that's our next entry */
Packit Service 20376f
		prev_entry = tree_iterator_current_entry(frame);
Packit Service 20376f
		entry = frame->entries.contents[frame->next_idx];
Packit Service 20376f
		frame->next_idx++;
Packit Service 20376f
Packit Service 20376f
		/* we can have collisions when iterating case insensitively.  (eg,
Packit Service 20376f
		 * 'A/a' and 'a/A').  squash this one if it's already been seen.
Packit Service 20376f
		 */
Packit Service 20376f
		if (iterator__ignore_case(&iter->base) &&
Packit Service 20376f
			prev_entry &&
Packit Service 20376f
			tree_iterator_entry_cmp_icase(prev_entry, entry) == 0)
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		if ((error = tree_iterator_compute_path(&iter->entry_path, entry)) < 0)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		/* if this path is before our start, advance over this entry */
Packit Service 20376f
		if (!iterator_has_started(&iter->base, iter->entry_path.ptr, false))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		/* if this path is after our end, stop */
Packit Service 20376f
		if (iterator_has_ended(&iter->base, iter->entry_path.ptr)) {
Packit Service 20376f
			error = GIT_ITEROVER;
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* if we have a list of paths we're interested in, examine it */
Packit Service 20376f
		if (!iterator_pathlist_next_is(&iter->base, iter->entry_path.ptr))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		is_tree = git_tree_entry__is_tree(entry->tree_entry);
Packit Service 20376f
Packit Service 20376f
		/* if we are *not* including trees then advance over this entry */
Packit Service 20376f
		if (is_tree && !iterator__include_trees(iter)) {
Packit Service 20376f
Packit Service 20376f
			/* if we've found a tree (and are not returning it to the caller)
Packit Service 20376f
			 * and we are autoexpanding, then we want to return the first
Packit Service 20376f
			 * child.  push the new directory and advance.
Packit Service 20376f
			 */
Packit Service 20376f
			if (iterator__do_autoexpand(iter)) {
Packit Service 20376f
				if ((error = tree_iterator_frame_push(iter, entry)) < 0)
Packit Service 20376f
					break;
Packit Service 20376f
			}
Packit Service 20376f
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		tree_iterator_set_current(iter, frame, entry);
Packit Service 20376f
Packit Service 20376f
		/* if we are autoexpanding, then push this as a new frame, so that
Packit Service 20376f
		 * the next call to `advance` will dive into this directory.
Packit Service 20376f
		 */
Packit Service 20376f
		if (is_tree && iterator__do_autoexpand(iter))
Packit Service 20376f
			error = tree_iterator_frame_push(iter, entry);
Packit Service 20376f
Packit Service 20376f
		break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (out)
Packit Service 20376f
		*out = (error == 0) ? &iter->entry : NULL;
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_advance_into(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter = (tree_iterator *)i;
Packit Service 20376f
    tree_iterator_frame *frame;
Packit Service 20376f
	tree_iterator_entry *prev_entry;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if (out)
Packit Service 20376f
		*out = NULL;
Packit Service 20376f
Packit Service 20376f
	if ((frame = tree_iterator_current_frame(iter)) == NULL)
Packit Service 20376f
		return GIT_ITEROVER;
Packit Service 20376f
Packit Service 20376f
	/* get the last seen entry */
Packit Service 20376f
	prev_entry = tree_iterator_current_entry(frame);
Packit Service 20376f
Packit Service 20376f
	/* it's legal to call advance_into when auto-expand is on.  in this case,
Packit Service 20376f
	 * we will have pushed a new (empty) frame on to the stack for this
Packit Service 20376f
	 * new directory.  since it's empty, its current_entry should be null.
Packit Service 20376f
	 */
Packit Service 20376f
	assert(iterator__do_autoexpand(i) ^ (prev_entry != NULL));
Packit Service 20376f
Packit Service 20376f
	if (prev_entry) {
Packit Service 20376f
		if (!git_tree_entry__is_tree(prev_entry->tree_entry))
Packit Service 20376f
			return 0;
Packit Service 20376f
Packit Service 20376f
		if ((error = tree_iterator_frame_push(iter, prev_entry)) < 0)
Packit Service 20376f
			return error;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* we've advanced into the directory in question, let advance
Packit Service 20376f
	 * find the first entry
Packit Service 20376f
	 */
Packit Service 20376f
	return tree_iterator_advance(out, i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_advance_over(
Packit Service 20376f
	const git_index_entry **out,
Packit Service 20376f
	git_iterator_status_t *status,
Packit Service 20376f
	git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	*status = GIT_ITERATOR_STATUS_NORMAL;
Packit Service 20376f
	return git_iterator_advance(out, i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void tree_iterator_clear(tree_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	while (iter->frames.size)
Packit Service 20376f
		tree_iterator_frame_pop(iter);
Packit Service 20376f
Packit Service 20376f
	git_array_clear(iter->frames);
Packit Service 20376f
Packit Service 20376f
	git_pool_clear(&iter->entry_pool);
Packit Service 20376f
	git_buf_clear(&iter->entry_path);
Packit Service 20376f
Packit Service 20376f
	iterator_clear(&iter->base);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_init(tree_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	git_pool_init(&iter->entry_pool, sizeof(tree_iterator_entry));
Packit Service 20376f
Packit Service 20376f
	if ((error = tree_iterator_frame_init(iter, iter->root, NULL)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	iter->base.flags &= ~GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int tree_iterator_reset(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter = (tree_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	tree_iterator_clear(iter);
Packit Service 20376f
	return tree_iterator_init(iter);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void tree_iterator_free(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter = (tree_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	tree_iterator_clear(iter);
Packit Service 20376f
Packit Service 20376f
	git_tree_free(iter->root);
Packit Service 20376f
	git_buf_free(&iter->entry_path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_for_tree(
Packit Service 20376f
	git_iterator **out,
Packit Service 20376f
	git_tree *tree,
Packit Service 20376f
	git_iterator_options *options)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	static git_iterator_callbacks callbacks = {
Packit Service 20376f
		tree_iterator_current,
Packit Service 20376f
		tree_iterator_advance,
Packit Service 20376f
		tree_iterator_advance_into,
Packit Service 20376f
		tree_iterator_advance_over,
Packit Service 20376f
		tree_iterator_reset,
Packit Service 20376f
		tree_iterator_free
Packit Service 20376f
	};
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	if (tree == NULL)
Packit Service 20376f
		return git_iterator_for_nothing(out, options);
Packit Service 20376f
Packit Service 20376f
	iter = git__calloc(1, sizeof(tree_iterator));
Packit Service 20376f
	GITERR_CHECK_ALLOC(iter);
Packit Service 20376f
Packit Service 20376f
	iter->base.type = GIT_ITERATOR_TYPE_TREE;
Packit Service 20376f
	iter->base.cb = &callbacks;
Packit Service 20376f
Packit Service 20376f
	if ((error = iterator_init_common(&iter->base,
Packit Service 20376f
			git_tree_owner(tree), NULL, options)) < 0 ||
Packit Service 20376f
		(error = git_tree_dup(&iter->root, tree)) < 0 ||
Packit Service 20376f
		(error = tree_iterator_init(iter)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	*out = &iter->base;
Packit Service 20376f
	return 0;
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	git_iterator_free(&iter->base);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_current_tree_entry(
Packit Service 20376f
	const git_tree_entry **tree_entry, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter;
Packit Service 20376f
	tree_iterator_frame *frame;
Packit Service 20376f
	tree_iterator_entry *entry;
Packit Service 20376f
Packit Service 20376f
	assert(i->type == GIT_ITERATOR_TYPE_TREE);
Packit Service 20376f
Packit Service 20376f
	iter = (tree_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	frame = tree_iterator_current_frame(iter);
Packit Service 20376f
	entry = tree_iterator_current_entry(frame);
Packit Service 20376f
Packit Service 20376f
	*tree_entry = entry->tree_entry;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_current_parent_tree(
Packit Service 20376f
	const git_tree **parent_tree, git_iterator *i, size_t depth)
Packit Service 20376f
{
Packit Service 20376f
	tree_iterator *iter;
Packit Service 20376f
	tree_iterator_frame *frame;
Packit Service 20376f
Packit Service 20376f
	assert(i->type == GIT_ITERATOR_TYPE_TREE);
Packit Service 20376f
Packit Service 20376f
	iter = (tree_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	assert(depth < iter->frames.size);
Packit Service 20376f
	frame = &iter->frames.ptr[iter->frames.size-depth-1];
Packit Service 20376f
Packit Service 20376f
	*parent_tree = frame->tree;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/* Filesystem iterator */
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	struct stat st;
Packit Service 20376f
	size_t path_len;
Packit Service 20376f
	iterator_pathlist_search_t match;
Packit Service 20376f
	char path[GIT_FLEX_ARRAY];
Packit Service 20376f
} filesystem_iterator_entry;
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_vector entries;
Packit Service 20376f
	git_pool entry_pool;
Packit Service 20376f
	size_t next_idx;
Packit Service 20376f
Packit Service 20376f
	size_t path_len;
Packit Service 20376f
	int is_ignored;
Packit Service 20376f
} filesystem_iterator_frame;
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_iterator base;
Packit Service 20376f
	char *root;
Packit Service 20376f
	size_t root_len;
Packit Service 20376f
Packit Service 20376f
	unsigned int dirload_flags;
Packit Service 20376f
Packit Service 20376f
	git_tree *tree;
Packit Service 20376f
	git_index *index;
Packit Service 20376f
	git_vector index_snapshot;
Packit Service 20376f
Packit Service 20376f
	git_array_t(filesystem_iterator_frame) frames;
Packit Service 20376f
	git_ignores ignores;
Packit Service 20376f
Packit Service 20376f
	/* info about the current entry */
Packit Service 20376f
	git_index_entry entry;
Packit Service 20376f
	git_buf current_path;
Packit Service 20376f
	int current_is_ignored;
Packit Service 20376f
Packit Service 20376f
	/* temporary buffer for advance_over */
Packit Service 20376f
	git_buf tmp_buf;
Packit Service 20376f
} filesystem_iterator;
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(filesystem_iterator_frame *) filesystem_iterator_parent_frame(
Packit Service 20376f
	filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	return iter->frames.size > 1 ?
Packit Service 20376f
		&iter->frames.ptr[iter->frames.size-2] : NULL;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(filesystem_iterator_frame *) filesystem_iterator_current_frame(
Packit Service 20376f
	filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	return iter->frames.size ? &iter->frames.ptr[iter->frames.size-1] : NULL;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(filesystem_iterator_entry *) filesystem_iterator_current_entry(
Packit Service 20376f
	filesystem_iterator_frame *frame)
Packit Service 20376f
{
Packit Service 20376f
	return frame->next_idx == 0 ?
Packit Service 20376f
		NULL : frame->entries.contents[frame->next_idx-1];
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_entry_cmp(const void *_a, const void *_b)
Packit Service 20376f
{
Packit Service 20376f
	const filesystem_iterator_entry *a = (const filesystem_iterator_entry *)_a;
Packit Service 20376f
	const filesystem_iterator_entry *b = (const filesystem_iterator_entry *)_b;
Packit Service 20376f
Packit Service 20376f
	return git__strcmp(a->path, b->path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_entry_cmp_icase(const void *_a, const void *_b)
Packit Service 20376f
{
Packit Service 20376f
	const filesystem_iterator_entry *a = (const filesystem_iterator_entry *)_a;
Packit Service 20376f
	const filesystem_iterator_entry *b = (const filesystem_iterator_entry *)_b;
Packit Service 20376f
Packit Service 20376f
	return git__strcasecmp(a->path, b->path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
#define FILESYSTEM_MAX_DEPTH 100
Packit Service 20376f
Packit Service 20376f
/**
Packit Service 20376f
 * Figure out if an entry is a submodule.
Packit Service 20376f
 *
Packit Service 20376f
 * We consider it a submodule if the path is listed as a submodule in
Packit Service 20376f
 * either the tree or the index.
Packit Service 20376f
 */
Packit Service 20376f
static int filesystem_iterator_is_submodule(
Packit Service 20376f
	bool *out, filesystem_iterator *iter, const char *path, size_t path_len)
Packit Service 20376f
{
Packit Service 20376f
	bool is_submodule = false;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	*out = false;
Packit Service 20376f
Packit Service 20376f
	/* first see if this path is a submodule in HEAD */
Packit Service 20376f
	if (iter->tree) {
Packit Service 20376f
		git_tree_entry *entry;
Packit Service 20376f
Packit Service 20376f
		error = git_tree_entry_bypath(&entry, iter->tree, path);
Packit Service 20376f
Packit Service 20376f
		if (error < 0 && error != GIT_ENOTFOUND)
Packit Service 20376f
			return error;
Packit Service 20376f
Packit Service 20376f
		if (!error) {
Packit Service 20376f
			is_submodule = (entry->attr == GIT_FILEMODE_COMMIT);
Packit Service 20376f
			git_tree_entry_free(entry);
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (!is_submodule && iter->base.index) {
Packit Service 20376f
		size_t pos;
Packit Service 20376f
Packit Service 20376f
		error = git_index_snapshot_find(&pos,
Packit Service 20376f
			&iter->index_snapshot, iter->base.entry_srch, path, path_len, 0);
Packit Service 20376f
Packit Service 20376f
		if (error < 0 && error != GIT_ENOTFOUND)
Packit Service 20376f
			return error;
Packit Service 20376f
Packit Service 20376f
		if (!error) {
Packit Service 20376f
			git_index_entry *e = git_vector_get(&iter->index_snapshot, pos);
Packit Service 20376f
			is_submodule = (e->mode == GIT_FILEMODE_COMMIT);
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	*out = is_submodule;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void filesystem_iterator_frame_push_ignores(
Packit Service 20376f
	filesystem_iterator *iter,
Packit Service 20376f
	filesystem_iterator_entry *frame_entry,
Packit Service 20376f
	filesystem_iterator_frame *new_frame)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator_frame *previous_frame;
Packit Service 20376f
	const char *path = frame_entry ? frame_entry->path : "";
Packit Service 20376f
Packit Service 20376f
	if (!iterator__honor_ignores(&iter->base))
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	if (git_ignore__lookup(&new_frame->is_ignored,
Packit Service 20376f
			&iter->ignores, path, GIT_DIR_FLAG_TRUE) < 0) {
Packit Service 20376f
		giterr_clear();
Packit Service 20376f
		new_frame->is_ignored = GIT_IGNORE_NOTFOUND;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* if this is not the top level directory... */
Packit Service 20376f
	if (frame_entry) {
Packit Service 20376f
		const char *relative_path;
Packit Service 20376f
Packit Service 20376f
		previous_frame = filesystem_iterator_parent_frame(iter);
Packit Service 20376f
Packit Service 20376f
		/* push new ignores for files in this directory */
Packit Service 20376f
		relative_path = frame_entry->path + previous_frame->path_len;
Packit Service 20376f
Packit Service 20376f
		/* inherit ignored from parent if no rule specified */
Packit Service 20376f
		if (new_frame->is_ignored <= GIT_IGNORE_NOTFOUND)
Packit Service 20376f
			new_frame->is_ignored = previous_frame->is_ignored;
Packit Service 20376f
Packit Service 20376f
		git_ignore__push_dir(&iter->ignores, relative_path);
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void filesystem_iterator_frame_pop_ignores(
Packit Service 20376f
	filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	if (iterator__honor_ignores(&iter->base))
Packit Service 20376f
		git_ignore__pop_dir(&iter->ignores);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(bool) filesystem_iterator_examine_path(
Packit Service 20376f
	bool *is_dir_out,
Packit Service 20376f
	iterator_pathlist_search_t *match_out,
Packit Service 20376f
	filesystem_iterator *iter,
Packit Service 20376f
	filesystem_iterator_entry *frame_entry,
Packit Service 20376f
	const char *path,
Packit Service 20376f
	size_t path_len)
Packit Service 20376f
{
Packit Service 20376f
	bool is_dir = 0;
Packit Service 20376f
	iterator_pathlist_search_t match = ITERATOR_PATHLIST_FULL;
Packit Service 20376f
Packit Service 20376f
	*is_dir_out = false;
Packit Service 20376f
	*match_out = ITERATOR_PATHLIST_NONE;
Packit Service 20376f
Packit Service 20376f
	if (iter->base.start_len) {
Packit Service 20376f
		int cmp = iter->base.strncomp(path, iter->base.start, path_len);
Packit Service 20376f
Packit Service 20376f
		/* we haven't stat'ed `path` yet, so we don't yet know if it's a
Packit Service 20376f
		 * directory or not.  special case if the current path may be a
Packit Service 20376f
		 * directory that matches the start prefix.
Packit Service 20376f
		 */
Packit Service 20376f
		if (cmp == 0) {
Packit Service 20376f
			if (iter->base.start[path_len] == '/')
Packit Service 20376f
				is_dir = true;
Packit Service 20376f
Packit Service 20376f
			else if (iter->base.start[path_len] != '\0')
Packit Service 20376f
				cmp = -1;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (cmp < 0)
Packit Service 20376f
			return false;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (iter->base.end_len) {
Packit Service 20376f
		int cmp = iter->base.strncomp(path, iter->base.end, iter->base.end_len);
Packit Service 20376f
Packit Service 20376f
		if (cmp > 0)
Packit Service 20376f
			return false;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* if we have a pathlist that we're limiting to, examine this path now
Packit Service 20376f
	 * to avoid a `stat` if we're not interested in the path.
Packit Service 20376f
	 */
Packit Service 20376f
	if (iter->base.pathlist.length) {
Packit Service 20376f
		/* if our parent was explicitly included, so too are we */
Packit Service 20376f
		if (frame_entry && frame_entry->match != ITERATOR_PATHLIST_IS_PARENT)
Packit Service 20376f
			match = ITERATOR_PATHLIST_FULL;
Packit Service 20376f
		else
Packit Service 20376f
			match = iterator_pathlist_search(&iter->base, path, path_len);
Packit Service 20376f
Packit Service 20376f
		if (match == ITERATOR_PATHLIST_NONE)
Packit Service 20376f
			return false;
Packit Service 20376f
Packit Service 20376f
		/* Ensure that the pathlist entry lines up with what we expected */
Packit Service 20376f
		if (match == ITERATOR_PATHLIST_IS_DIR ||
Packit Service 20376f
			match == ITERATOR_PATHLIST_IS_PARENT)
Packit Service 20376f
			is_dir = true;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	*is_dir_out = is_dir;
Packit Service 20376f
	*match_out = match;
Packit Service 20376f
	return true;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(bool) filesystem_iterator_is_dot_git(
Packit Service 20376f
	filesystem_iterator *iter, const char *path, size_t path_len)
Packit Service 20376f
{
Packit Service 20376f
	size_t len;
Packit Service 20376f
Packit Service 20376f
	if (!iterator__ignore_dot_git(&iter->base))
Packit Service 20376f
		return false;
Packit Service 20376f
Packit Service 20376f
	if ((len = path_len) < 4)
Packit Service 20376f
		return false;
Packit Service 20376f
Packit Service 20376f
	if (path[len - 1] == '/')
Packit Service 20376f
		len--;
Packit Service 20376f
Packit Service 20376f
	if (git__tolower(path[len - 1]) != 't' ||
Packit Service 20376f
		git__tolower(path[len - 2]) != 'i' ||
Packit Service 20376f
		git__tolower(path[len - 3]) != 'g' ||
Packit Service 20376f
		git__tolower(path[len - 4]) != '.')
Packit Service 20376f
		return false;
Packit Service 20376f
Packit Service 20376f
	return (len == 4 || path[len - 5] == '/');
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static filesystem_iterator_entry *filesystem_iterator_entry_init(
Packit Service 20376f
	filesystem_iterator_frame *frame,
Packit Service 20376f
	const char *path,
Packit Service 20376f
	size_t path_len,
Packit Service 20376f
	struct stat *statbuf,
Packit Service 20376f
	iterator_pathlist_search_t pathlist_match)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator_entry *entry;
Packit Service 20376f
	size_t entry_size;
Packit Service 20376f
Packit Service 20376f
	/* Make sure to append two bytes, one for the path's null
Packit Service 20376f
	 * termination, one for a possible trailing '/' for folders.
Packit Service 20376f
	 */
Packit Service 20376f
	if (GIT_ADD_SIZET_OVERFLOW(&entry_size,
Packit Service 20376f
			sizeof(filesystem_iterator_entry), path_len) ||
Packit Service 20376f
		GIT_ADD_SIZET_OVERFLOW(&entry_size, entry_size, 2) ||
Packit Service 20376f
		(entry = git_pool_malloc(&frame->entry_pool, entry_size)) == NULL)
Packit Service 20376f
		return NULL;
Packit Service 20376f
Packit Service 20376f
	entry->path_len = path_len;
Packit Service 20376f
	entry->match = pathlist_match;
Packit Service 20376f
	memcpy(entry->path, path, path_len);
Packit Service 20376f
	memcpy(&entry->st, statbuf, sizeof(struct stat));
Packit Service 20376f
Packit Service 20376f
	/* Suffix directory paths with a '/' */
Packit Service 20376f
	if (S_ISDIR(entry->st.st_mode))
Packit Service 20376f
		entry->path[entry->path_len++] = '/';
Packit Service 20376f
Packit Service 20376f
	entry->path[entry->path_len] = '\0';
Packit Service 20376f
Packit Service 20376f
	return entry;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_frame_push(
Packit Service 20376f
	filesystem_iterator *iter,
Packit Service 20376f
	filesystem_iterator_entry *frame_entry)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator_frame *new_frame = NULL;
Packit Service 20376f
	git_path_diriter diriter = GIT_PATH_DIRITER_INIT;
Packit Service 20376f
	git_buf root = GIT_BUF_INIT;
Packit Service 20376f
	const char *path;
Packit Service 20376f
	filesystem_iterator_entry *entry;
Packit Service 20376f
	struct stat statbuf;
Packit Service 20376f
	size_t path_len;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if (iter->frames.size == FILESYSTEM_MAX_DEPTH) {
Packit Service 20376f
		giterr_set(GITERR_REPOSITORY,
Packit Service 20376f
			"directory nesting too deep (%"PRIuZ")", iter->frames.size);
Packit Service 20376f
		return -1;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	new_frame = git_array_alloc(iter->frames);
Packit Service 20376f
	GITERR_CHECK_ALLOC(new_frame);
Packit Service 20376f
Packit Service 20376f
	memset(new_frame, 0, sizeof(filesystem_iterator_frame));
Packit Service 20376f
Packit Service 20376f
	if (frame_entry)
Packit Service 20376f
		git_buf_joinpath(&root, iter->root, frame_entry->path);
Packit Service 20376f
	else
Packit Service 20376f
		git_buf_puts(&root, iter->root);
Packit Service 20376f
Packit Service 20376f
	if (git_buf_oom(&root)) {
Packit Service 20376f
		error = -1;
Packit Service 20376f
		goto done;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	new_frame->path_len = frame_entry ? frame_entry->path_len : 0;
Packit Service 20376f
Packit Service 20376f
	/* Any error here is equivalent to the dir not existing, skip over it */
Packit Service 20376f
	if ((error = git_path_diriter_init(
Packit Service 20376f
			&diriter, root.ptr, iter->dirload_flags)) < 0) {
Packit Service 20376f
		error = GIT_ENOTFOUND;
Packit Service 20376f
		goto done;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if ((error = git_vector_init(&new_frame->entries, 64,
Packit Service 20376f
			iterator__ignore_case(&iter->base) ?
Packit Service 20376f
			filesystem_iterator_entry_cmp_icase :
Packit Service 20376f
			filesystem_iterator_entry_cmp)) < 0)
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	git_pool_init(&new_frame->entry_pool, 1);
Packit Service 20376f
Packit Service 20376f
	/* check if this directory is ignored */
Packit Service 20376f
	filesystem_iterator_frame_push_ignores(iter, frame_entry, new_frame);
Packit Service 20376f
Packit Service 20376f
	while ((error = git_path_diriter_next(&diriter)) == 0) {
Packit Service 20376f
		iterator_pathlist_search_t pathlist_match = ITERATOR_PATHLIST_FULL;
Packit Service 20376f
		bool dir_expected = false;
Packit Service 20376f
Packit Service 20376f
		if ((error = git_path_diriter_fullpath(&path, &path_len, &diriter)) < 0)
Packit Service 20376f
			goto done;
Packit Service 20376f
Packit Service 20376f
		assert(path_len > iter->root_len);
Packit Service 20376f
Packit Service 20376f
		/* remove the prefix if requested */
Packit Service 20376f
		path += iter->root_len;
Packit Service 20376f
		path_len -= iter->root_len;
Packit Service 20376f
Packit Service 20376f
		/* examine start / end and the pathlist to see if this path is in it.
Packit Service 20376f
		 * note that since we haven't yet stat'ed the path, we cannot know
Packit Service 20376f
		 * whether it's a directory yet or not, so this can give us an
Packit Service 20376f
		 * expected type (S_IFDIR or S_IFREG) that we should examine)
Packit Service 20376f
		 */
Packit Service 20376f
		if (!filesystem_iterator_examine_path(&dir_expected, &pathlist_match,
Packit Service 20376f
			iter, frame_entry, path, path_len))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		/* TODO: don't need to stat if assume unchanged for this path and
Packit Service 20376f
		 * we have an index, we can just copy the data out of it.
Packit Service 20376f
		 */
Packit Service 20376f
Packit Service 20376f
		if ((error = git_path_diriter_stat(&statbuf, &diriter)) < 0) {
Packit Service 20376f
			/* file was removed between readdir and lstat */
Packit Service 20376f
			if (error == GIT_ENOTFOUND)
Packit Service 20376f
				continue;
Packit Service 20376f
Packit Service 20376f
			/* treat the file as unreadable */
Packit Service 20376f
			memset(&statbuf, 0, sizeof(statbuf));
Packit Service 20376f
			statbuf.st_mode = GIT_FILEMODE_UNREADABLE;
Packit Service 20376f
Packit Service 20376f
			error = 0;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		iter->base.stat_calls++;
Packit Service 20376f
Packit Service 20376f
		/* Ignore wacky things in the filesystem */
Packit Service 20376f
		if (!S_ISDIR(statbuf.st_mode) &&
Packit Service 20376f
			!S_ISREG(statbuf.st_mode) &&
Packit Service 20376f
			!S_ISLNK(statbuf.st_mode) &&
Packit Service 20376f
			statbuf.st_mode != GIT_FILEMODE_UNREADABLE)
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		if (filesystem_iterator_is_dot_git(iter, path, path_len))
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		/* convert submodules to GITLINK and remove trailing slashes */
Packit Service 20376f
		if (S_ISDIR(statbuf.st_mode)) {
Packit Service 20376f
			bool submodule = false;
Packit Service 20376f
Packit Service 20376f
			if ((error = filesystem_iterator_is_submodule(&submodule,
Packit Service 20376f
					iter, path, path_len)) < 0)
Packit Service 20376f
				goto done;
Packit Service 20376f
Packit Service 20376f
			if (submodule)
Packit Service 20376f
				statbuf.st_mode = GIT_FILEMODE_COMMIT;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* Ensure that the pathlist entry lines up with what we expected */
Packit Service 20376f
		else if (dir_expected)
Packit Service 20376f
			continue;
Packit Service 20376f
Packit Service 20376f
		entry = filesystem_iterator_entry_init(new_frame,
Packit Service 20376f
			path, path_len, &statbuf, pathlist_match);
Packit Service 20376f
		GITERR_CHECK_ALLOC(entry);
Packit Service 20376f
Packit Service 20376f
		git_vector_insert(&new_frame->entries, entry);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ITEROVER)
Packit Service 20376f
		error = 0;
Packit Service 20376f
Packit Service 20376f
	/* sort now that directory suffix is added */
Packit Service 20376f
	git_vector_sort(&new_frame->entries);
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	if (error < 0)
Packit Service 20376f
		git_array_pop(iter->frames);
Packit Service 20376f
Packit Service 20376f
	git_buf_free(&root);
Packit Service 20376f
	git_path_diriter_free(&diriter);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(void) filesystem_iterator_frame_pop(filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator_frame *frame;
Packit Service 20376f
Packit Service 20376f
	assert(iter->frames.size);
Packit Service 20376f
Packit Service 20376f
	frame = git_array_pop(iter->frames);
Packit Service 20376f
	filesystem_iterator_frame_pop_ignores(iter);
Packit Service 20376f
Packit Service 20376f
	git_pool_clear(&frame->entry_pool);
Packit Service 20376f
	git_vector_free(&frame->entries);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void filesystem_iterator_set_current(
Packit Service 20376f
	filesystem_iterator *iter,
Packit Service 20376f
	filesystem_iterator_entry *entry)
Packit Service 20376f
{
Packit Service 20376f
	iter->entry.ctime.seconds = entry->st.st_ctime;
Packit Service 20376f
	iter->entry.ctime.nanoseconds = entry->st.st_ctime_nsec;
Packit Service 20376f
Packit Service 20376f
	iter->entry.mtime.seconds = entry->st.st_mtime;
Packit Service 20376f
	iter->entry.mtime.nanoseconds = entry->st.st_mtime_nsec;
Packit Service 20376f
Packit Service 20376f
	iter->entry.dev = entry->st.st_dev;
Packit Service 20376f
	iter->entry.ino = entry->st.st_ino;
Packit Service 20376f
	iter->entry.mode = git_futils_canonical_mode(entry->st.st_mode);
Packit Service 20376f
	iter->entry.uid = entry->st.st_uid;
Packit Service 20376f
	iter->entry.gid = entry->st.st_gid;
Packit Service 20376f
	iter->entry.file_size = entry->st.st_size;
Packit Service 20376f
Packit Service 20376f
	iter->entry.path = entry->path;
Packit Service 20376f
Packit Service 20376f
	iter->current_is_ignored = GIT_IGNORE_UNCHECKED;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_current(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	if (!iterator__has_been_accessed(i))
Packit Service 20376f
		return iter->base.cb->advance(out, i);
Packit Service 20376f
Packit Service 20376f
	if (!iter->frames.size) {
Packit Service 20376f
		*out = NULL;
Packit Service 20376f
		return GIT_ITEROVER;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	*out = &iter->entry;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_advance(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	iter->base.flags |= GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
Packit Service 20376f
	/* examine filesystem entries until we find the next one to return */
Packit Service 20376f
	while (true) {
Packit Service 20376f
		filesystem_iterator_frame *frame;
Packit Service 20376f
		filesystem_iterator_entry *entry;
Packit Service 20376f
Packit Service 20376f
		if ((frame = filesystem_iterator_current_frame(iter)) == NULL) {
Packit Service 20376f
			error = GIT_ITEROVER;
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* no more entries in this frame.  pop the frame out */
Packit Service 20376f
		if (frame->next_idx == frame->entries.length) {
Packit Service 20376f
			filesystem_iterator_frame_pop(iter);
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* we have more entries in the current frame, that's our next entry */
Packit Service 20376f
		entry = frame->entries.contents[frame->next_idx];
Packit Service 20376f
		frame->next_idx++;
Packit Service 20376f
Packit Service 20376f
		if (S_ISDIR(entry->st.st_mode)) {
Packit Service 20376f
			if (iterator__do_autoexpand(iter)) {
Packit Service 20376f
				error = filesystem_iterator_frame_push(iter, entry);
Packit Service 20376f
Packit Service 20376f
				/* may get GIT_ENOTFOUND due to races or permission problems
Packit Service 20376f
				 * that we want to quietly swallow
Packit Service 20376f
				 */
Packit Service 20376f
				if (error == GIT_ENOTFOUND)
Packit Service 20376f
					continue;
Packit Service 20376f
				else if (error < 0)
Packit Service 20376f
					break;
Packit Service 20376f
			}
Packit Service 20376f
Packit Service 20376f
			if (!iterator__include_trees(iter))
Packit Service 20376f
				continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		filesystem_iterator_set_current(iter, entry);
Packit Service 20376f
		break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (out)
Packit Service 20376f
		*out = (error == 0) ? &iter->entry : NULL;
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_advance_into(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
	filesystem_iterator_frame *frame;
Packit Service 20376f
	filesystem_iterator_entry *prev_entry;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if (out)
Packit Service 20376f
		*out = NULL;
Packit Service 20376f
Packit Service 20376f
	if ((frame = filesystem_iterator_current_frame(iter)) == NULL)
Packit Service 20376f
		return GIT_ITEROVER;
Packit Service 20376f
Packit Service 20376f
	/* get the last seen entry */
Packit Service 20376f
	prev_entry = filesystem_iterator_current_entry(frame);
Packit Service 20376f
Packit Service 20376f
	/* it's legal to call advance_into when auto-expand is on.  in this case,
Packit Service 20376f
	 * we will have pushed a new (empty) frame on to the stack for this
Packit Service 20376f
	 * new directory.  since it's empty, its current_entry should be null.
Packit Service 20376f
	 */
Packit Service 20376f
	assert(iterator__do_autoexpand(i) ^ (prev_entry != NULL));
Packit Service 20376f
Packit Service 20376f
	if (prev_entry) {
Packit Service 20376f
		if (prev_entry->st.st_mode != GIT_FILEMODE_COMMIT &&
Packit Service 20376f
			!S_ISDIR(prev_entry->st.st_mode))
Packit Service 20376f
			return 0;
Packit Service 20376f
Packit Service 20376f
		if ((error = filesystem_iterator_frame_push(iter, prev_entry)) < 0)
Packit Service 20376f
			return error;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* we've advanced into the directory in question, let advance
Packit Service 20376f
	 * find the first entry
Packit Service 20376f
	 */
Packit Service 20376f
	return filesystem_iterator_advance(out, i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_current_workdir_path(git_buf **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
	const git_index_entry *entry;
Packit Service 20376f
Packit Service 20376f
	if (i->type != GIT_ITERATOR_TYPE_FS &&
Packit Service 20376f
		i->type != GIT_ITERATOR_TYPE_WORKDIR) {
Packit Service 20376f
		*out = NULL;
Packit Service 20376f
		return 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	git_buf_truncate(&iter->current_path, iter->root_len);
Packit Service 20376f
Packit Service 20376f
	if (git_iterator_current(&entry, i) < 0 ||
Packit Service 20376f
		git_buf_puts(&iter->current_path, entry->path) < 0)
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	*out = &iter->current_path;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(git_dir_flag) entry_dir_flag(git_index_entry *entry)
Packit Service 20376f
{
Packit Service 20376f
#if defined(GIT_WIN32) && !defined(__MINGW32__)
Packit Service 20376f
	return (entry && entry->mode) ?
Packit Service 20376f
		(S_ISDIR(entry->mode) ? GIT_DIR_FLAG_TRUE : GIT_DIR_FLAG_FALSE) :
Packit Service 20376f
		GIT_DIR_FLAG_UNKNOWN;
Packit Service 20376f
#else
Packit Service 20376f
	GIT_UNUSED(entry);
Packit Service 20376f
	return GIT_DIR_FLAG_UNKNOWN;
Packit Service 20376f
#endif
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void filesystem_iterator_update_ignored(filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator_frame *frame;
Packit Service 20376f
	git_dir_flag dir_flag = entry_dir_flag(&iter->entry);
Packit Service 20376f
Packit Service 20376f
	if (git_ignore__lookup(&iter->current_is_ignored,
Packit Service 20376f
			&iter->ignores, iter->entry.path, dir_flag) < 0) {
Packit Service 20376f
		giterr_clear();
Packit Service 20376f
		iter->current_is_ignored = GIT_IGNORE_NOTFOUND;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* use ignore from containing frame stack */
Packit Service 20376f
	if (iter->current_is_ignored <= GIT_IGNORE_NOTFOUND) {
Packit Service 20376f
		frame = filesystem_iterator_current_frame(iter);
Packit Service 20376f
		iter->current_is_ignored = frame->is_ignored;
Packit Service 20376f
	}
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(bool) filesystem_iterator_current_is_ignored(
Packit Service 20376f
	filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	if (iter->current_is_ignored == GIT_IGNORE_UNCHECKED)
Packit Service 20376f
		filesystem_iterator_update_ignored(iter);
Packit Service 20376f
Packit Service 20376f
	return (iter->current_is_ignored == GIT_IGNORE_TRUE);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
bool git_iterator_current_is_ignored(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	if (i->type != GIT_ITERATOR_TYPE_WORKDIR)
Packit Service 20376f
		return false;
Packit Service 20376f
Packit Service 20376f
	return filesystem_iterator_current_is_ignored((filesystem_iterator *)i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
bool git_iterator_current_tree_is_ignored(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
	filesystem_iterator_frame *frame;
Packit Service 20376f
Packit Service 20376f
	if (i->type != GIT_ITERATOR_TYPE_WORKDIR)
Packit Service 20376f
		return false;
Packit Service 20376f
Packit Service 20376f
	frame = filesystem_iterator_current_frame(iter);
Packit Service 20376f
	return (frame->is_ignored == GIT_IGNORE_TRUE);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_advance_over(
Packit Service 20376f
	const git_index_entry **out,
Packit Service 20376f
	git_iterator_status_t *status,
Packit Service 20376f
	git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
	filesystem_iterator_frame *current_frame;
Packit Service 20376f
	filesystem_iterator_entry *current_entry;
Packit Service 20376f
	const git_index_entry *entry = NULL;
Packit Service 20376f
	const char *base;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
	*status = GIT_ITERATOR_STATUS_NORMAL;
Packit Service 20376f
Packit Service 20376f
	assert(iterator__has_been_accessed(i));
Packit Service 20376f
Packit Service 20376f
	current_frame = filesystem_iterator_current_frame(iter);
Packit Service 20376f
	assert(current_frame);
Packit Service 20376f
	current_entry = filesystem_iterator_current_entry(current_frame);
Packit Service 20376f
	assert(current_entry);
Packit Service 20376f
Packit Service 20376f
	if ((error = git_iterator_current(&entry, i)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if (!S_ISDIR(entry->mode)) {
Packit Service 20376f
		if (filesystem_iterator_current_is_ignored(iter))
Packit Service 20376f
			*status = GIT_ITERATOR_STATUS_IGNORED;
Packit Service 20376f
Packit Service 20376f
		return filesystem_iterator_advance(out, i);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	git_buf_clear(&iter->tmp_buf);
Packit Service 20376f
	if ((error = git_buf_puts(&iter->tmp_buf, entry->path)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	base = iter->tmp_buf.ptr;
Packit Service 20376f
Packit Service 20376f
	/* scan inside the directory looking for files.  if we find nothing,
Packit Service 20376f
	 * we will remain EMPTY.  if we find any ignored item, upgrade EMPTY to
Packit Service 20376f
	 * IGNORED.  if we find a real actual item, upgrade all the way to NORMAL
Packit Service 20376f
	 * and then stop.
Packit Service 20376f
	 *
Packit Service 20376f
	 * however, if we're here looking for a pathlist item (but are not
Packit Service 20376f
	 * actually in the pathlist ourselves) then start at FILTERED instead of
Packit Service 20376f
	 * EMPTY.  callers then know that this path was not something they asked
Packit Service 20376f
	 * about.
Packit Service 20376f
	 */
Packit Service 20376f
	*status = current_entry->match == ITERATOR_PATHLIST_IS_PARENT ?
Packit Service 20376f
		GIT_ITERATOR_STATUS_FILTERED : GIT_ITERATOR_STATUS_EMPTY;
Packit Service 20376f
Packit Service 20376f
	while (entry && !iter->base.prefixcomp(entry->path, base)) {
Packit Service 20376f
		if (filesystem_iterator_current_is_ignored(iter)) {
Packit Service 20376f
			/* if we found an explicitly ignored item, then update from
Packit Service 20376f
			 * EMPTY to IGNORED
Packit Service 20376f
			 */
Packit Service 20376f
			*status = GIT_ITERATOR_STATUS_IGNORED;
Packit Service 20376f
		} else if (S_ISDIR(entry->mode)) {
Packit Service 20376f
			error = filesystem_iterator_advance_into(&entry, i);
Packit Service 20376f
Packit Service 20376f
			if (!error)
Packit Service 20376f
				continue;
Packit Service 20376f
Packit Service 20376f
			/* this directory disappeared, ignore it */
Packit Service 20376f
			else if (error == GIT_ENOTFOUND)
Packit Service 20376f
				error = 0;
Packit Service 20376f
Packit Service 20376f
			/* a real error occurred */
Packit Service 20376f
			else
Packit Service 20376f
				break;
Packit Service 20376f
		} else {
Packit Service 20376f
			/* we found a non-ignored item, treat parent as untracked */
Packit Service 20376f
			*status = GIT_ITERATOR_STATUS_NORMAL;
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if ((error = git_iterator_advance(&entry, i)) < 0)
Packit Service 20376f
			break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* wrap up scan back to base directory */
Packit Service 20376f
	while (entry && !iter->base.prefixcomp(entry->path, base)) {
Packit Service 20376f
		if ((error = git_iterator_advance(&entry, i)) < 0)
Packit Service 20376f
			break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (!error)
Packit Service 20376f
		*out = entry;
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void filesystem_iterator_clear(filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	while (iter->frames.size)
Packit Service 20376f
		filesystem_iterator_frame_pop(iter);
Packit Service 20376f
Packit Service 20376f
	git_array_clear(iter->frames);
Packit Service 20376f
	git_ignore__free(&iter->ignores);
Packit Service 20376f
Packit Service 20376f
	git_buf_free(&iter->tmp_buf);
Packit Service 20376f
Packit Service 20376f
	iterator_clear(&iter->base);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_init(filesystem_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if (iterator__honor_ignores(&iter->base) &&
Packit Service 20376f
		(error = git_ignore__for_path(iter->base.repo,
Packit Service 20376f
			".gitignore", &iter->ignores)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if ((error = filesystem_iterator_frame_push(iter, NULL)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	iter->base.flags &= ~GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int filesystem_iterator_reset(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	filesystem_iterator_clear(iter);
Packit Service 20376f
	return filesystem_iterator_init(iter);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void filesystem_iterator_free(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter = (filesystem_iterator *)i;
Packit Service 20376f
	git__free(iter->root);
Packit Service 20376f
	git_buf_free(&iter->current_path);
Packit Service 20376f
	git_tree_free(iter->tree);
Packit Service 20376f
	if (iter->index)
Packit Service 20376f
		git_index_snapshot_release(&iter->index_snapshot, iter->index);
Packit Service 20376f
	filesystem_iterator_clear(iter);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int iterator_for_filesystem(
Packit Service 20376f
	git_iterator **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const char *root,
Packit Service 20376f
	git_index *index,
Packit Service 20376f
	git_tree *tree,
Packit Service 20376f
	git_iterator_type_t type,
Packit Service 20376f
	git_iterator_options *options)
Packit Service 20376f
{
Packit Service 20376f
	filesystem_iterator *iter;
Packit Service 20376f
	size_t root_len;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	static git_iterator_callbacks callbacks = {
Packit Service 20376f
		filesystem_iterator_current,
Packit Service 20376f
		filesystem_iterator_advance,
Packit Service 20376f
		filesystem_iterator_advance_into,
Packit Service 20376f
		filesystem_iterator_advance_over,
Packit Service 20376f
		filesystem_iterator_reset,
Packit Service 20376f
		filesystem_iterator_free
Packit Service 20376f
	};
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	if (root == NULL)
Packit Service 20376f
		return git_iterator_for_nothing(out, options);
Packit Service 20376f
Packit Service 20376f
	iter = git__calloc(1, sizeof(filesystem_iterator));
Packit Service 20376f
	GITERR_CHECK_ALLOC(iter);
Packit Service 20376f
Packit Service 20376f
	iter->base.type = type;
Packit Service 20376f
	iter->base.cb = &callbacks;
Packit Service 20376f
Packit Service 20376f
	root_len = strlen(root);
Packit Service 20376f
Packit Service 20376f
	iter->root = git__malloc(root_len+2);
Packit Service 20376f
	GITERR_CHECK_ALLOC(iter->root);
Packit Service 20376f
Packit Service 20376f
	memcpy(iter->root, root, root_len);
Packit Service 20376f
Packit Service 20376f
	if (root_len == 0 || root[root_len-1] != '/') {
Packit Service 20376f
		iter->root[root_len] = '/';
Packit Service 20376f
		root_len++;
Packit Service 20376f
	}
Packit Service 20376f
	iter->root[root_len] = '\0';
Packit Service 20376f
	iter->root_len = root_len;
Packit Service 20376f
Packit Service 20376f
	if ((error = git_buf_puts(&iter->current_path, iter->root)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	if ((error = iterator_init_common(&iter->base, repo, index, options)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	if (tree && (error = git_tree_dup(&iter->tree, tree)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	if (index &&
Packit Service 20376f
		(error = git_index_snapshot_new(&iter->index_snapshot, index)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	iter->index = index;
Packit Service 20376f
	iter->dirload_flags =
Packit Service 20376f
		(iterator__ignore_case(&iter->base) ? GIT_PATH_DIR_IGNORE_CASE : 0) |
Packit Service 20376f
		(iterator__flag(&iter->base, PRECOMPOSE_UNICODE) ?
Packit Service 20376f
			 GIT_PATH_DIR_PRECOMPOSE_UNICODE : 0);
Packit Service 20376f
Packit Service 20376f
	if ((error = filesystem_iterator_init(iter)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	*out = &iter->base;
Packit Service 20376f
	return 0;
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	git_iterator_free(&iter->base);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_for_filesystem(
Packit Service 20376f
	git_iterator **out,
Packit Service 20376f
	const char *root,
Packit Service 20376f
	git_iterator_options *options)
Packit Service 20376f
{
Packit Service 20376f
	return iterator_for_filesystem(out,
Packit Service 20376f
		NULL, root, NULL, NULL, GIT_ITERATOR_TYPE_FS, options);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_for_workdir_ext(
Packit Service 20376f
	git_iterator **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	const char *repo_workdir,
Packit Service 20376f
	git_index *index,
Packit Service 20376f
	git_tree *tree,
Packit Service 20376f
	git_iterator_options *given_opts)
Packit Service 20376f
{
Packit Service 20376f
	git_iterator_options options = GIT_ITERATOR_OPTIONS_INIT;
Packit Service 20376f
Packit Service 20376f
	if (!repo_workdir) {
Packit Service 20376f
		if (git_repository__ensure_not_bare(repo, "scan working directory") < 0)
Packit Service 20376f
			return GIT_EBAREREPO;
Packit Service 20376f
Packit Service 20376f
		repo_workdir = git_repository_workdir(repo);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* upgrade to a workdir iterator, adding necessary internal flags */
Packit Service 20376f
	if (given_opts)
Packit Service 20376f
		memcpy(&options, given_opts, sizeof(git_iterator_options));
Packit Service 20376f
Packit Service 20376f
	options.flags |= GIT_ITERATOR_HONOR_IGNORES |
Packit Service 20376f
		GIT_ITERATOR_IGNORE_DOT_GIT;
Packit Service 20376f
Packit Service 20376f
	return iterator_for_filesystem(out,
Packit Service 20376f
		repo, repo_workdir, index, tree, GIT_ITERATOR_TYPE_WORKDIR, &options);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
/* Index iterator */
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
typedef struct {
Packit Service 20376f
	git_iterator base;
Packit Service 20376f
	git_vector entries;
Packit Service 20376f
	size_t next_idx;
Packit Service 20376f
Packit Service 20376f
	/* the pseudotree entry */
Packit Service 20376f
	git_index_entry tree_entry;
Packit Service 20376f
	git_buf tree_buf;
Packit Service 20376f
	bool skip_tree;
Packit Service 20376f
Packit Service 20376f
	const git_index_entry *entry;
Packit Service 20376f
} index_iterator;
Packit Service 20376f
Packit Service 20376f
static int index_iterator_current(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter = (index_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	if (!iterator__has_been_accessed(i))
Packit Service 20376f
		return iter->base.cb->advance(out, i);
Packit Service 20376f
Packit Service 20376f
	if (iter->entry == NULL) {
Packit Service 20376f
		*out = NULL;
Packit Service 20376f
		return GIT_ITEROVER;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	*out = iter->entry;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static bool index_iterator_create_pseudotree(
Packit Service 20376f
	const git_index_entry **out,
Packit Service 20376f
	index_iterator *iter,
Packit Service 20376f
	const char *path)
Packit Service 20376f
{
Packit Service 20376f
	const char *prev_path, *relative_path, *dirsep;
Packit Service 20376f
	size_t common_len;
Packit Service 20376f
Packit Service 20376f
	prev_path = iter->entry ? iter->entry->path : "";
Packit Service 20376f
Packit Service 20376f
	/* determine if the new path is in a different directory from the old */
Packit Service 20376f
	common_len = git_path_common_dirlen(prev_path, path);
Packit Service 20376f
	relative_path = path + common_len;
Packit Service 20376f
Packit Service 20376f
	if ((dirsep = strchr(relative_path, '/')) == NULL)
Packit Service 20376f
		return false;
Packit Service 20376f
Packit Service 20376f
	git_buf_clear(&iter->tree_buf);
Packit Service 20376f
	git_buf_put(&iter->tree_buf, path, (dirsep - path) + 1);
Packit Service 20376f
Packit Service 20376f
	iter->tree_entry.mode = GIT_FILEMODE_TREE;
Packit Service 20376f
	iter->tree_entry.path = iter->tree_buf.ptr;
Packit Service 20376f
Packit Service 20376f
	*out = &iter->tree_entry;
Packit Service 20376f
	return true;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int index_iterator_skip_pseudotree(index_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	assert(iterator__has_been_accessed(&iter->base));
Packit Service 20376f
	assert(S_ISDIR(iter->entry->mode));
Packit Service 20376f
Packit Service 20376f
	while (true) {
Packit Service 20376f
		const git_index_entry *next_entry = NULL;
Packit Service 20376f
Packit Service 20376f
		if (++iter->next_idx >= iter->entries.length)
Packit Service 20376f
			return GIT_ITEROVER;
Packit Service 20376f
Packit Service 20376f
		next_entry = iter->entries.contents[iter->next_idx];
Packit Service 20376f
Packit Service 20376f
		if (iter->base.strncomp(iter->tree_buf.ptr, next_entry->path,
Packit Service 20376f
			iter->tree_buf.size) != 0)
Packit Service 20376f
			break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	iter->skip_tree = false;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int index_iterator_advance(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter = (index_iterator *)i;
Packit Service 20376f
	const git_index_entry *entry = NULL;
Packit Service 20376f
	bool is_submodule;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	iter->base.flags |= GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
Packit Service 20376f
	while (true) {
Packit Service 20376f
		if (iter->next_idx >= iter->entries.length) {
Packit Service 20376f
			error = GIT_ITEROVER;
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* we were not asked to expand this pseudotree.  advance over it. */
Packit Service 20376f
		if (iter->skip_tree) {
Packit Service 20376f
			index_iterator_skip_pseudotree(iter);
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		entry = iter->entries.contents[iter->next_idx];
Packit Service 20376f
		is_submodule = S_ISGITLINK(entry->mode);
Packit Service 20376f
Packit Service 20376f
		if (!iterator_has_started(&iter->base, entry->path, is_submodule)) {
Packit Service 20376f
			iter->next_idx++;
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (iterator_has_ended(&iter->base, entry->path)) {
Packit Service 20376f
			error = GIT_ITEROVER;
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* if we have a list of paths we're interested in, examine it */
Packit Service 20376f
		if (!iterator_pathlist_next_is(&iter->base, entry->path)) {
Packit Service 20376f
			iter->next_idx++;
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* if this is a conflict, skip it unless we're including conflicts */
Packit Service 20376f
		if (git_index_entry_is_conflict(entry) &&
Packit Service 20376f
			!iterator__include_conflicts(&iter->base)) {
Packit Service 20376f
			iter->next_idx++;
Packit Service 20376f
			continue;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* we've found what will be our next _file_ entry.  but if we are
Packit Service 20376f
		 * returning trees entries, we may need to return a pseudotree
Packit Service 20376f
		 * entry that will contain this.  don't advance over this entry,
Packit Service 20376f
		 * though, we still need to return it on the next `advance`.
Packit Service 20376f
		 */
Packit Service 20376f
		if (iterator__include_trees(&iter->base) &&
Packit Service 20376f
			index_iterator_create_pseudotree(&entry, iter, entry->path)) {
Packit Service 20376f
Packit Service 20376f
			/* Note whether this pseudo tree should be expanded or not */
Packit Service 20376f
			iter->skip_tree = iterator__dont_autoexpand(&iter->base);
Packit Service 20376f
			break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		iter->next_idx++;
Packit Service 20376f
		break;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	iter->entry = (error == 0) ? entry : NULL;
Packit Service 20376f
Packit Service 20376f
	if (out)
Packit Service 20376f
		*out = iter->entry;
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int index_iterator_advance_into(
Packit Service 20376f
	const git_index_entry **out, git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter = (index_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	if (! S_ISDIR(iter->tree_entry.mode)) {
Packit Service 20376f
		if (out)
Packit Service 20376f
			*out = NULL;
Packit Service 20376f
Packit Service 20376f
		return 0;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	iter->skip_tree = false;
Packit Service 20376f
	return index_iterator_advance(out, i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int index_iterator_advance_over(
Packit Service 20376f
	const git_index_entry **out,
Packit Service 20376f
	git_iterator_status_t *status,
Packit Service 20376f
	git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter = (index_iterator *)i;
Packit Service 20376f
	const git_index_entry *entry;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if ((error = index_iterator_current(&entry, i)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if (S_ISDIR(entry->mode))
Packit Service 20376f
		index_iterator_skip_pseudotree(iter);
Packit Service 20376f
Packit Service 20376f
	*status = GIT_ITERATOR_STATUS_NORMAL;
Packit Service 20376f
	return index_iterator_advance(out, i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void index_iterator_clear(index_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	iterator_clear(&iter->base);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int index_iterator_init(index_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	iter->base.flags &= ~GIT_ITERATOR_FIRST_ACCESS;
Packit Service 20376f
	iter->next_idx = 0;
Packit Service 20376f
	iter->skip_tree = false;
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int index_iterator_reset(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter = (index_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	index_iterator_clear(iter);
Packit Service 20376f
	return index_iterator_init(iter);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static void index_iterator_free(git_iterator *i)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter = (index_iterator *)i;
Packit Service 20376f
Packit Service 20376f
	git_index_snapshot_release(&iter->entries, iter->base.index);
Packit Service 20376f
	git_buf_free(&iter->tree_buf);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_for_index(
Packit Service 20376f
	git_iterator **out,
Packit Service 20376f
	git_repository *repo,
Packit Service 20376f
	git_index  *index,
Packit Service 20376f
	git_iterator_options *options)
Packit Service 20376f
{
Packit Service 20376f
	index_iterator *iter;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	static git_iterator_callbacks callbacks = {
Packit Service 20376f
		index_iterator_current,
Packit Service 20376f
		index_iterator_advance,
Packit Service 20376f
		index_iterator_advance_into,
Packit Service 20376f
		index_iterator_advance_over,
Packit Service 20376f
		index_iterator_reset,
Packit Service 20376f
		index_iterator_free
Packit Service 20376f
	};
Packit Service 20376f
Packit Service 20376f
	*out = NULL;
Packit Service 20376f
Packit Service 20376f
	if (index == NULL)
Packit Service 20376f
		return git_iterator_for_nothing(out, options);
Packit Service 20376f
Packit Service 20376f
	iter = git__calloc(1, sizeof(index_iterator));
Packit Service 20376f
	GITERR_CHECK_ALLOC(iter);
Packit Service 20376f
Packit Service 20376f
	iter->base.type = GIT_ITERATOR_TYPE_INDEX;
Packit Service 20376f
	iter->base.cb = &callbacks;
Packit Service 20376f
Packit Service 20376f
	if ((error = iterator_init_common(&iter->base, repo, index, options)) < 0 ||
Packit Service 20376f
		(error = git_index_snapshot_new(&iter->entries, index)) < 0 ||
Packit Service 20376f
		(error = index_iterator_init(iter)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	git_vector_set_cmp(&iter->entries, iterator__ignore_case(&iter->base) ?
Packit Service 20376f
		git_index_entry_icmp : git_index_entry_cmp);
Packit Service 20376f
	git_vector_sort(&iter->entries);
Packit Service 20376f
Packit Service 20376f
	*out = &iter->base;
Packit Service 20376f
	return 0;
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	git_iterator_free(&iter->base);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
Packit Service 20376f
/* Iterator API */
Packit Service 20376f
Packit Service 20376f
int git_iterator_reset_range(
Packit Service 20376f
	git_iterator *i, const char *start, const char *end)
Packit Service 20376f
{
Packit Service 20376f
	if (iterator_reset_range(i, start, end) < 0)
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	return i->cb->reset(i);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void git_iterator_set_ignore_case(git_iterator *i, bool ignore_case)
Packit Service 20376f
{
Packit Service 20376f
	assert(!iterator__has_been_accessed(i));
Packit Service 20376f
	iterator_set_ignore_case(i, ignore_case);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
void git_iterator_free(git_iterator *iter)
Packit Service 20376f
{
Packit Service 20376f
	if (iter == NULL)
Packit Service 20376f
		return;
Packit Service 20376f
Packit Service 20376f
	iter->cb->free(iter);
Packit Service 20376f
Packit Service 20376f
	git_vector_free(&iter->pathlist);
Packit Service 20376f
	git__free(iter->start);
Packit Service 20376f
	git__free(iter->end);
Packit Service 20376f
Packit Service 20376f
	memset(iter, 0, sizeof(*iter));
Packit Service 20376f
Packit Service 20376f
	git__free(iter);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_iterator_walk(
Packit Service 20376f
	git_iterator **iterators,
Packit Service 20376f
	size_t cnt,
Packit Service 20376f
	git_iterator_walk_cb cb,
Packit Service 20376f
	void *data)
Packit Service 20376f
{
Packit Service 20376f
	const git_index_entry **iterator_item;	/* next in each iterator */
Packit Service 20376f
	const git_index_entry **cur_items;		/* current path in each iter */
Packit Service 20376f
	const git_index_entry *first_match;
Packit Service 20376f
	size_t i, j;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	iterator_item = git__calloc(cnt, sizeof(git_index_entry *));
Packit Service 20376f
	cur_items = git__calloc(cnt, sizeof(git_index_entry *));
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_ALLOC(iterator_item);
Packit Service 20376f
	GITERR_CHECK_ALLOC(cur_items);
Packit Service 20376f
Packit Service 20376f
	/* Set up the iterators */
Packit Service 20376f
	for (i = 0; i < cnt; i++) {
Packit Service 20376f
		error = git_iterator_current(&iterator_item[i], iterators[i]);
Packit Service 20376f
Packit Service 20376f
		if (error < 0 && error != GIT_ITEROVER)
Packit Service 20376f
			goto done;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	while (true) {
Packit Service 20376f
		for (i = 0; i < cnt; i++)
Packit Service 20376f
			cur_items[i] = NULL;
Packit Service 20376f
Packit Service 20376f
		first_match = NULL;
Packit Service 20376f
Packit Service 20376f
		/* Find the next path(s) to consume from each iterator */
Packit Service 20376f
		for (i = 0; i < cnt; i++) {
Packit Service 20376f
			if (iterator_item[i] == NULL)
Packit Service 20376f
				continue;
Packit Service 20376f
Packit Service 20376f
			if (first_match == NULL) {
Packit Service 20376f
				first_match = iterator_item[i];
Packit Service 20376f
				cur_items[i] = iterator_item[i];
Packit Service 20376f
			} else {
Packit Service 20376f
				int path_diff = git_index_entry_cmp(iterator_item[i], first_match);
Packit Service 20376f
Packit Service 20376f
				if (path_diff < 0) {
Packit Service 20376f
					/* Found an index entry that sorts before the one we're
Packit Service 20376f
					 * looking at.  Forget that we've seen the other and
Packit Service 20376f
					 * look at the other iterators for this path.
Packit Service 20376f
					 */
Packit Service 20376f
					for (j = 0; j < i; j++)
Packit Service 20376f
						cur_items[j] = NULL;
Packit Service 20376f
Packit Service 20376f
					first_match = iterator_item[i];
Packit Service 20376f
					cur_items[i] = iterator_item[i];
Packit Service 20376f
				} else if (path_diff == 0) {
Packit Service 20376f
					cur_items[i] = iterator_item[i];
Packit Service 20376f
				}
Packit Service 20376f
			}
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		if (first_match == NULL)
Packit Service 20376f
			break;
Packit Service 20376f
Packit Service 20376f
		if ((error = cb(cur_items, data)) != 0)
Packit Service 20376f
			goto done;
Packit Service 20376f
Packit Service 20376f
		/* Advance each iterator that participated */
Packit Service 20376f
		for (i = 0; i < cnt; i++) {
Packit Service 20376f
			if (cur_items[i] == NULL)
Packit Service 20376f
				continue;
Packit Service 20376f
Packit Service 20376f
			error = git_iterator_advance(&iterator_item[i], iterators[i]);
Packit Service 20376f
Packit Service 20376f
			if (error < 0 && error != GIT_ITEROVER)
Packit Service 20376f
				goto done;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	git__free((git_index_entry **)iterator_item);
Packit Service 20376f
	git__free((git_index_entry **)cur_items);
Packit Service 20376f
Packit Service 20376f
	if (error == GIT_ITEROVER)
Packit Service 20376f
		error = 0;
Packit Service 20376f
Packit Service 20376f
	return error;
Packit Service 20376f
}