Dir.glob with FNM_EXTGLOB is optimized [Feature #13873]
The order of resulted array is changed in some cases. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@60253 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
a2d6b37f0c
commit
e2a92c54e6
2
NEWS
2
NEWS
@ -31,6 +31,8 @@ with all sufficient information, see the ChangeLog file or Redmine
|
|||||||
|
|
||||||
* Dir.glob provides new optional keyword argument, :base.
|
* Dir.glob provides new optional keyword argument, :base.
|
||||||
[Feature #13056]
|
[Feature #13056]
|
||||||
|
* Dir.glob with FNM_EXTGLOB is optimized [Feature #13873]
|
||||||
|
The order of resulted array is changed in some cases.
|
||||||
* Dir.children [Feature #11302]
|
* Dir.children [Feature #11302]
|
||||||
* Dir.each_child [Feature #11302]
|
* Dir.each_child [Feature #11302]
|
||||||
|
|
||||||
|
84
dir.c
84
dir.c
@ -291,6 +291,8 @@ bracket(
|
|||||||
#define UNESCAPE(p) (escape && *(p) == '\\' ? (p) + 1 : (p))
|
#define UNESCAPE(p) (escape && *(p) == '\\' ? (p) + 1 : (p))
|
||||||
#define ISEND(p) (!*(p) || (pathname && *(p) == '/'))
|
#define ISEND(p) (!*(p) || (pathname && *(p) == '/'))
|
||||||
#define RETURN(val) return *pcur = p, *scur = s, (val);
|
#define RETURN(val) return *pcur = p, *scur = s, (val);
|
||||||
|
#define FNMATCH_ALLOC_N(type, n) ((type *)malloc(sizeof(type) * (n)))
|
||||||
|
#define FNMATCH_FREE(ptr) free(ptr)
|
||||||
|
|
||||||
static int
|
static int
|
||||||
fnmatch_helper(
|
fnmatch_helper(
|
||||||
@ -349,6 +351,56 @@ fnmatch_helper(
|
|||||||
}
|
}
|
||||||
goto failed;
|
goto failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case '{': {
|
||||||
|
size_t len = pend - p;
|
||||||
|
char *buf = FNMATCH_ALLOC_N(char, len);
|
||||||
|
const char *rbrace = NULL;
|
||||||
|
while (p < pend) {
|
||||||
|
const char *t = ++p;
|
||||||
|
int nest = 0;
|
||||||
|
while (p < pend && !(*p == ',' && nest == 0)) {
|
||||||
|
if (*p == '{') nest++;
|
||||||
|
if (*p == '}') {
|
||||||
|
if (nest == 0) {
|
||||||
|
if (!rbrace) rbrace = p;
|
||||||
|
goto rest;
|
||||||
|
}
|
||||||
|
nest--;
|
||||||
|
}
|
||||||
|
if (*p == '\\' && escape) {
|
||||||
|
if (++p >= pend) break;
|
||||||
|
}
|
||||||
|
Inc(p, pend, enc);
|
||||||
|
}
|
||||||
|
if (!rbrace) {
|
||||||
|
rbrace = p;
|
||||||
|
while (rbrace < pend && !(*rbrace == '}' && nest == 0)) {
|
||||||
|
if (*rbrace == '{') nest++;
|
||||||
|
if (*rbrace == '}') nest--;
|
||||||
|
if (*rbrace == '\\' && escape) {
|
||||||
|
if (++p >= pend) break;
|
||||||
|
}
|
||||||
|
Inc(rbrace, pend, enc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
rest:
|
||||||
|
memcpy(buf, t, p-t);
|
||||||
|
buf[p-t]=0;
|
||||||
|
strlcpy(buf+(p-t), rbrace+1, len-(p-t));
|
||||||
|
{
|
||||||
|
const char *pp = buf, *ss = s;
|
||||||
|
r = fnmatch_helper((const char **)&pp, &ss, flags|FNM_DOTMATCH, enc);
|
||||||
|
}
|
||||||
|
if (r == 0) {
|
||||||
|
p = buf;
|
||||||
|
FNMATCH_FREE(buf);
|
||||||
|
RETURN(0);
|
||||||
|
}
|
||||||
|
if (p >= rbrace) break;
|
||||||
|
}
|
||||||
|
FNMATCH_FREE(buf);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ordinary */
|
/* ordinary */
|
||||||
@ -1429,6 +1481,12 @@ has_magic(const char *p, const char *pend, int flags, rb_encoding *enc)
|
|||||||
case '[':
|
case '[':
|
||||||
return MAGICAL;
|
return MAGICAL;
|
||||||
|
|
||||||
|
case '{':
|
||||||
|
if (flags & FNM_EXTGLOB) {
|
||||||
|
return MAGICAL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case '\\':
|
case '\\':
|
||||||
if (escape && p++ >= pend)
|
if (escape && p++ >= pend)
|
||||||
continue;
|
continue;
|
||||||
@ -2275,6 +2333,13 @@ push_pattern(const char *path, VALUE ary, void *enc)
|
|||||||
rb_ary_push(ary, name);
|
rb_ary_push(ary, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct push_glob_args {
|
||||||
|
struct glob_args glob;
|
||||||
|
int flags;
|
||||||
|
int fd;
|
||||||
|
};
|
||||||
|
static int push_caller(const char *path, VALUE val, void *enc);
|
||||||
|
|
||||||
static int
|
static int
|
||||||
ruby_brace_expand(const char *str, int flags, ruby_glob_func *func, VALUE arg,
|
ruby_brace_expand(const char *str, int flags, ruby_glob_func *func, VALUE arg,
|
||||||
rb_encoding *enc, VALUE var)
|
rb_encoding *enc, VALUE var)
|
||||||
@ -2283,7 +2348,7 @@ ruby_brace_expand(const char *str, int flags, ruby_glob_func *func, VALUE arg,
|
|||||||
const char *p = str;
|
const char *p = str;
|
||||||
const char *pend = p + strlen(p);
|
const char *pend = p + strlen(p);
|
||||||
const char *s = p;
|
const char *s = p;
|
||||||
const char *lbrace = 0, *rbrace = 0;
|
const char *lbrace = NULL, *rbrace = NULL;
|
||||||
int nest = 0, status = 0;
|
int nest = 0, status = 0;
|
||||||
|
|
||||||
while (*p) {
|
while (*p) {
|
||||||
@ -2302,9 +2367,18 @@ ruby_brace_expand(const char *str, int flags, ruby_glob_func *func, VALUE arg,
|
|||||||
|
|
||||||
if (lbrace && rbrace) {
|
if (lbrace && rbrace) {
|
||||||
size_t len = strlen(s) + 1;
|
size_t len = strlen(s) + 1;
|
||||||
char *buf = GLOB_ALLOC_N(char, len);
|
char *buf;
|
||||||
long shift;
|
long shift;
|
||||||
|
|
||||||
|
if (func == push_caller && !strchr(lbrace, '/')) {
|
||||||
|
/* Now it reaches file basename entry. */
|
||||||
|
/* Handle braces in glob_helper */
|
||||||
|
struct push_glob_args *a = (struct push_glob_args *)arg;
|
||||||
|
a->flags |= FNM_EXTGLOB;
|
||||||
|
return glob_call_func(func, s, arg, enc);
|
||||||
|
}
|
||||||
|
|
||||||
|
buf = GLOB_ALLOC_N(char, len);
|
||||||
if (!buf) return -1;
|
if (!buf) return -1;
|
||||||
memcpy(buf, s, lbrace-s);
|
memcpy(buf, s, lbrace-s);
|
||||||
shift = (lbrace-s);
|
shift = (lbrace-s);
|
||||||
@ -2368,12 +2442,6 @@ ruby_brace_glob(const char *str, int flags, ruby_glob_func *func, VALUE arg)
|
|||||||
return ruby_brace_glob_with_enc(str, flags, func, arg, rb_ascii8bit_encoding());
|
return ruby_brace_glob_with_enc(str, flags, func, arg, rb_ascii8bit_encoding());
|
||||||
}
|
}
|
||||||
|
|
||||||
struct push_glob_args {
|
|
||||||
struct glob_args glob;
|
|
||||||
int flags;
|
|
||||||
int fd;
|
|
||||||
};
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
push_caller(const char *path, VALUE val, void *enc)
|
push_caller(const char *path, VALUE val, void *enc)
|
||||||
{
|
{
|
||||||
|
@ -155,7 +155,7 @@ class TestDir < Test::Unit::TestCase
|
|||||||
open(File.join(@root, "}}{}"), "wb") {}
|
open(File.join(@root, "}}{}"), "wb") {}
|
||||||
open(File.join(@root, "}}a"), "wb") {}
|
open(File.join(@root, "}}a"), "wb") {}
|
||||||
assert_equal(%w(}}{} }}a).map {|f| File.join(@root, f)}, Dir.glob(File.join(@root, '}}{\{\},a}')))
|
assert_equal(%w(}}{} }}a).map {|f| File.join(@root, f)}, Dir.glob(File.join(@root, '}}{\{\},a}')))
|
||||||
assert_equal(%w(}}{} }}a b c).map {|f| File.join(@root, f)}, Dir.glob(File.join(@root, '{\}\}{\{\},a},b,c}')))
|
assert_equal(%w(}}{} }}a b c).map {|f| File.join(@root, f)}.sort, Dir.glob(File.join(@root, '{\}\}{\{\},a},b,c}')).sort)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_glob_recursive
|
def test_glob_recursive
|
||||||
|
Loading…
x
Reference in New Issue
Block a user