diff options
| author | Junio C Hamano <gitster@pobox.com> | 2013-09-09 14:36:15 -0700 | 
|---|---|---|
| committer | Junio C Hamano <gitster@pobox.com> | 2013-09-09 14:36:15 -0700 | 
| commit | b02f5aeda6f66ac3be9b2e35f9b237d4f1f80d73 (patch) | |
| tree | 8b85c36e79c9f3dffbc9ea9df9c812c5f5070514 /tree-walk.c | |
| parent | de9a25354aa22aa6796787f3ef3af276fba82339 (diff) | |
| parent | 95c16418f0375e2fc325f32c3d7578fba9cfd7ef (diff) | |
| download | git-b02f5aeda6f66ac3be9b2e35f9b237d4f1f80d73.tar.gz | |
Merge branch 'jl/submodule-mv'
"git mv A B" when moving a submodule A does "the right thing",
inclusing relocating its working tree and adjusting the paths in
the .gitmodules file.
* jl/submodule-mv: (53 commits)
  rm: delete .gitmodules entry of submodules removed from the work tree
  mv: update the path entry in .gitmodules for moved submodules
  submodule.c: add .gitmodules staging helper functions
  mv: move submodules using a gitfile
  mv: move submodules together with their work trees
  rm: do not set a variable twice without intermediate reading.
  t6131 - skip tests if on case-insensitive file system
  parse_pathspec: accept :(icase)path syntax
  pathspec: support :(glob) syntax
  pathspec: make --literal-pathspecs disable pathspec magic
  pathspec: support :(literal) syntax for noglob pathspec
  kill limit_pathspec_to_literal() as it's only used by parse_pathspec()
  parse_pathspec: preserve prefix length via PATHSPEC_PREFIX_ORIGIN
  parse_pathspec: make sure the prefix part is wildcard-free
  rename field "raw" to "_raw" in struct pathspec
  tree-diff: remove the use of pathspec's raw[] in follow-rename codepath
  remove match_pathspec() in favor of match_pathspec_depth()
  remove init_pathspec() in favor of parse_pathspec()
  remove diff_tree_{setup,release}_paths
  convert common_prefix() to use struct pathspec
  ...
Diffstat (limited to 'tree-walk.c')
| -rw-r--r-- | tree-walk.c | 78 | 
1 files changed, 62 insertions, 16 deletions
| diff --git a/tree-walk.c b/tree-walk.c index c626135234..5ece8c3477 100644 --- a/tree-walk.c +++ b/tree-walk.c @@ -3,6 +3,7 @@  #include "unpack-trees.h"  #include "dir.h"  #include "tree.h" +#include "pathspec.h"  static const char *get_mode(const char *str, unsigned int *modep)  { @@ -487,13 +488,25 @@ int get_tree_entry(const unsigned char *tree_sha1, const char *name, unsigned ch  	return retval;  } -static int match_entry(const struct name_entry *entry, int pathlen, +static int match_entry(const struct pathspec_item *item, +		       const struct name_entry *entry, int pathlen,  		       const char *match, int matchlen,  		       enum interesting *never_interesting)  {  	int m = -1; /* signals that we haven't called strncmp() */ -	if (*never_interesting != entry_not_interesting) { +	if (item->magic & PATHSPEC_ICASE) +		/* +		 * "Never interesting" trick requires exact +		 * matching. We could do something clever with inexact +		 * matching, but it's trickier (and not to forget that +		 * strcasecmp is locale-dependent, at least in +		 * glibc). Just disable it for now. It can't be worse +		 * than the wildcard's codepath of '[Tt][Hi][Is][Ss]' +		 * pattern. +		 */ +		*never_interesting = entry_not_interesting; +	else if (*never_interesting != entry_not_interesting) {  		/*  		 * We have not seen any match that sorts later  		 * than the current path. @@ -539,7 +552,7 @@ static int match_entry(const struct name_entry *entry, int pathlen,  		 * we cheated and did not do strncmp(), so we do  		 * that here.  		 */ -		m = strncmp(match, entry->path, pathlen); +		m = ps_strncmp(item, match, entry->path, pathlen);  	/*  	 * If common part matched earlier then it is a hit, @@ -547,15 +560,39 @@ static int match_entry(const struct name_entry *entry, int pathlen,  	 * leading directory and is shorter than match.  	 */  	if (!m) +		/* +		 * match_entry does not check if the prefix part is +		 * matched case-sensitively. If the entry is a +		 * directory and part of prefix, it'll be rematched +		 * eventually by basecmp with special treatment for +		 * the prefix. +		 */  		return 1;  	return 0;  } -static int match_dir_prefix(const char *base, +/* :(icase)-aware string compare */ +static int basecmp(const struct pathspec_item *item, +		   const char *base, const char *match, int len) +{ +	if (item->magic & PATHSPEC_ICASE) { +		int ret, n = len > item->prefix ? item->prefix : len; +		ret = strncmp(base, match, n); +		if (ret) +			return ret; +		base += n; +		match += n; +		len -= n; +	} +	return ps_strncmp(item, base, match, len); +} + +static int match_dir_prefix(const struct pathspec_item *item, +			    const char *base,  			    const char *match, int matchlen)  { -	if (strncmp(base, match, matchlen)) +	if (basecmp(item, base, match, matchlen))  		return 0;  	/* @@ -592,7 +629,7 @@ static int match_wildcard_base(const struct pathspec_item *item,  		 */  		if (baselen >= matchlen) {  			*matched = matchlen; -			return !strncmp(base, match, matchlen); +			return !basecmp(item, base, match, matchlen);  		}  		dirlen = matchlen; @@ -605,7 +642,7 @@ static int match_wildcard_base(const struct pathspec_item *item,  		 * base ends with '/' so we are sure it really matches  		 * directory  		 */ -		if (strncmp(base, match, baselen)) +		if (basecmp(item, base, match, baselen))  			return 0;  		*matched = baselen;  	} else @@ -634,8 +671,17 @@ enum interesting tree_entry_interesting(const struct name_entry *entry,  	enum interesting never_interesting = ps->has_wildcard ?  		entry_not_interesting : all_entries_not_interesting; +	GUARD_PATHSPEC(ps, +		       PATHSPEC_FROMTOP | +		       PATHSPEC_MAXDEPTH | +		       PATHSPEC_LITERAL | +		       PATHSPEC_GLOB | +		       PATHSPEC_ICASE); +  	if (!ps->nr) { -		if (!ps->recursive || ps->max_depth == -1) +		if (!ps->recursive || +		    !(ps->magic & PATHSPEC_MAXDEPTH) || +		    ps->max_depth == -1)  			return all_entries_interesting;  		return within_depth(base->buf + base_offset, baselen,  				    !!S_ISDIR(entry->mode), @@ -653,10 +699,12 @@ enum interesting tree_entry_interesting(const struct name_entry *entry,  		if (baselen >= matchlen) {  			/* If it doesn't match, move along... */ -			if (!match_dir_prefix(base_str, match, matchlen)) +			if (!match_dir_prefix(item, base_str, match, matchlen))  				goto match_wildcards; -			if (!ps->recursive || ps->max_depth == -1) +			if (!ps->recursive || +			    !(ps->magic & PATHSPEC_MAXDEPTH) || +			    ps->max_depth == -1)  				return all_entries_interesting;  			return within_depth(base_str + matchlen + 1, @@ -667,15 +715,14 @@ enum interesting tree_entry_interesting(const struct name_entry *entry,  		}  		/* Either there must be no base, or the base must match. */ -		if (baselen == 0 || !strncmp(base_str, match, baselen)) { -			if (match_entry(entry, pathlen, +		if (baselen == 0 || !basecmp(item, base_str, match, baselen)) { +			if (match_entry(item, entry, pathlen,  					match + baselen, matchlen - baselen,  					&never_interesting))  				return entry_interesting;  			if (item->nowildcard_len < item->len) { -				if (!git_fnmatch(match + baselen, entry->path, -						 item->flags & PATHSPEC_ONESTAR ? GFNM_ONESTAR : 0, +				if (!git_fnmatch(item, match + baselen, entry->path,  						 item->nowildcard_len - baselen))  					return entry_interesting; @@ -716,8 +763,7 @@ match_wildcards:  		strbuf_add(base, entry->path, pathlen); -		if (!git_fnmatch(match, base->buf + base_offset, -				 item->flags & PATHSPEC_ONESTAR ? GFNM_ONESTAR : 0, +		if (!git_fnmatch(item, match, base->buf + base_offset,  				 item->nowildcard_len)) {  			strbuf_setlen(base, base_offset + baselen);  			return entry_interesting; | 
