mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
* dir.c: merge tuning from H.Yamamoto <ocean@m2.ccsnet.ne.jp>.
[ruby-dev:22761] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@5577 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
a84fc19203
commit
7a987b9bcd
5 changed files with 363 additions and 232 deletions
|
@ -1,3 +1,8 @@
|
||||||
|
Thu Jan 29 17:03:49 2004 Yukihiro Matsumoto <matz@ruby-lang.org>
|
||||||
|
|
||||||
|
* dir.c: merge tuning from H.Yamamoto <ocean@m2.ccsnet.ne.jp>.
|
||||||
|
[ruby-dev:22761]
|
||||||
|
|
||||||
Thu Jan 29 11:32:14 2004 NAKAMURA, Hiroshi <nakahiro@sarion.co.jp>
|
Thu Jan 29 11:32:14 2004 NAKAMURA, Hiroshi <nakahiro@sarion.co.jp>
|
||||||
|
|
||||||
* test/rss/test_*: do $: trick while searching a module in the current
|
* test/rss/test_*: do $: trick while searching a module in the current
|
||||||
|
|
585
dir.c
585
dir.c
|
@ -104,6 +104,7 @@ emx_mblen(p)
|
||||||
#else /* multi byte environment */
|
#else /* multi byte environment */
|
||||||
# define Inc(p) ((p) = Next(p))
|
# define Inc(p) ((p) = Next(p))
|
||||||
# define Compare(p1, p2) (CompareImpl(p1, p2, nocase))
|
# define Compare(p1, p2) (CompareImpl(p1, p2, nocase))
|
||||||
|
# ifndef _WIN32
|
||||||
static int
|
static int
|
||||||
CompareImpl(p1, p2, nocase)
|
CompareImpl(p1, p2, nocase)
|
||||||
const char *p1;
|
const char *p1;
|
||||||
|
@ -113,6 +114,10 @@ CompareImpl(p1, p2, nocase)
|
||||||
const int len1 = Next(p1) - p1;
|
const int len1 = Next(p1) - p1;
|
||||||
const int len2 = Next(p2) - p2;
|
const int len2 = Next(p2) - p2;
|
||||||
|
|
||||||
|
if (len1 < 0 || len2 < 0) {
|
||||||
|
rb_fatal("No-win32 CompareImpl: negative len");
|
||||||
|
}
|
||||||
|
|
||||||
if (len1 == 0) return len2;
|
if (len1 == 0) return len2;
|
||||||
if (len2 == 0) return -len1;
|
if (len2 == 0) return -len1;
|
||||||
|
|
||||||
|
@ -133,6 +138,40 @@ CompareImpl(p1, p2, nocase)
|
||||||
return ret ? ret : len1 - len2;
|
return ret ? ret : len1 - len2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
# else
|
||||||
|
static int
|
||||||
|
CompareImpl(p1, p2, nocase)
|
||||||
|
const char *p1;
|
||||||
|
const char *p2;
|
||||||
|
int nocase;
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
const int len1 = Next(p1) - p1;
|
||||||
|
const int len2 = Next(p2) - p2;
|
||||||
|
|
||||||
|
char buf1[10], buf2[10]; /* large enough? */
|
||||||
|
|
||||||
|
if (len1 < 0 || len2 < 0) {
|
||||||
|
rb_fatal("Win32 CompareImpl: negative len");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (len1 >= sizeof(buf1) || len2 >= sizeof(buf2)) {
|
||||||
|
rb_fatal("Win32 CompareImpl: too large len");
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(buf1, p1, len1); buf1[len1] = '\0';
|
||||||
|
memcpy(buf2, p2, len2); buf2[len2] = '\0';
|
||||||
|
|
||||||
|
if (nocase) {
|
||||||
|
CharLower(buf1);
|
||||||
|
CharLower(buf2);
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = memcmp(buf1, buf2, len1 < len2 ? len1 : len2);
|
||||||
|
return ret ? ret : len1 - len2;
|
||||||
|
}
|
||||||
|
# endif
|
||||||
#endif /* environment */
|
#endif /* environment */
|
||||||
|
|
||||||
#if defined DOSISH
|
#if defined DOSISH
|
||||||
|
@ -836,100 +875,7 @@ dir_s_rmdir(obj, dir)
|
||||||
return INT2FIX(0);
|
return INT2FIX(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return nonzero if S has any special globbing chars in it. */
|
/* System call with warning */
|
||||||
static int
|
|
||||||
has_magic(p, m, flags)
|
|
||||||
register char *p;
|
|
||||||
char **m;
|
|
||||||
int flags;
|
|
||||||
{
|
|
||||||
register char c;
|
|
||||||
int open = 0;
|
|
||||||
int escape = !(flags & FNM_NOESCAPE);
|
|
||||||
|
|
||||||
while (c = *p++, c != '\0' && c != '/') {
|
|
||||||
switch (c) {
|
|
||||||
case '?':
|
|
||||||
case '*':
|
|
||||||
goto found;
|
|
||||||
|
|
||||||
case '[': /* Only accept an open brace if there is a close */
|
|
||||||
open++; /* brace to match it. Bracket expressions must be */
|
|
||||||
continue; /* complete, according to Posix.2 */
|
|
||||||
case ']':
|
|
||||||
if (open)
|
|
||||||
goto found;
|
|
||||||
continue;
|
|
||||||
|
|
||||||
case '\\':
|
|
||||||
if (escape && (c = *p++, c == '\0' || c == '/'))
|
|
||||||
goto miss;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
p = Next(p-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
miss:
|
|
||||||
*m = p-1;
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
found:
|
|
||||||
while (*p != '\0' && *p != '/')
|
|
||||||
Inc(p);
|
|
||||||
*m = p;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
remove_backslashes(p, pend)
|
|
||||||
char *p;
|
|
||||||
char *pend;
|
|
||||||
{
|
|
||||||
char *t = p;
|
|
||||||
char *s = p;
|
|
||||||
int n = 0;
|
|
||||||
|
|
||||||
while (*p && p < pend) {
|
|
||||||
if (*p == '\\') {
|
|
||||||
if (t != s) {
|
|
||||||
memmove(t, s, p - s);
|
|
||||||
n++;
|
|
||||||
}
|
|
||||||
t += p - s;
|
|
||||||
s = ++p;
|
|
||||||
if (!(*p && p < pend)) break;
|
|
||||||
}
|
|
||||||
Inc(p);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (*p++);
|
|
||||||
|
|
||||||
if (t != s) {
|
|
||||||
memmove(t, s, p - s); /* move '\0' too */
|
|
||||||
n++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return n;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
do_fnmatch(p, pend, string, flags)
|
|
||||||
char *p;
|
|
||||||
char *pend;
|
|
||||||
const char *string;
|
|
||||||
int flags;
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
char c;
|
|
||||||
|
|
||||||
c = *pend;
|
|
||||||
*pend = '\0'; /* should I allocate new string? */
|
|
||||||
ret = fnmatch(p, string, flags);
|
|
||||||
*pend = c;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
do_stat(path, pst)
|
do_stat(path, pst)
|
||||||
const char *path;
|
const char *path;
|
||||||
|
@ -962,6 +908,160 @@ do_opendir(path)
|
||||||
return dirp;
|
return dirp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Return nonzero if S has any special globbing chars in it. */
|
||||||
|
static int
|
||||||
|
has_magic(s, flags)
|
||||||
|
char *s;
|
||||||
|
int flags;
|
||||||
|
{
|
||||||
|
register char *p = s;
|
||||||
|
register char c;
|
||||||
|
int open = 0;
|
||||||
|
int escape = !(flags & FNM_NOESCAPE);
|
||||||
|
|
||||||
|
while (c = *p++) {
|
||||||
|
switch (c) {
|
||||||
|
case '?':
|
||||||
|
case '*':
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
case '[': /* Only accept an open brace if there is a close */
|
||||||
|
open++; /* brace to match it. Bracket expressions must be */
|
||||||
|
continue; /* complete, according to Posix.2 */
|
||||||
|
case ']':
|
||||||
|
if (open)
|
||||||
|
return 1;
|
||||||
|
continue;
|
||||||
|
|
||||||
|
case '\\':
|
||||||
|
if (escape && !(c = *p++))
|
||||||
|
return 0;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
p = Next(p-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove escaping baskclashes */
|
||||||
|
static void
|
||||||
|
remove_backslashes(p)
|
||||||
|
char *p;
|
||||||
|
{
|
||||||
|
char *t = p;
|
||||||
|
char *s = p;
|
||||||
|
|
||||||
|
while (*p) {
|
||||||
|
if (*p == '\\') {
|
||||||
|
if (t != s)
|
||||||
|
memmove(t, s, p - s);
|
||||||
|
t += p - s;
|
||||||
|
s = ++p;
|
||||||
|
if (!*p) break;
|
||||||
|
}
|
||||||
|
Inc(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (*p++);
|
||||||
|
|
||||||
|
if (t != s)
|
||||||
|
memmove(t, s, p - s); /* move '\0' too */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Globing pattern */
|
||||||
|
enum glob_pattern_type { PLAIN, MAGICAL, RECURSIVE, MATCH_ALL, MATCH_DIR };
|
||||||
|
|
||||||
|
struct glob_pattern {
|
||||||
|
char *str;
|
||||||
|
enum glob_pattern_type type;
|
||||||
|
struct glob_pattern *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct glob_pattern *
|
||||||
|
glob_make_pattern(p, flags)
|
||||||
|
const char *p;
|
||||||
|
int flags;
|
||||||
|
{
|
||||||
|
char *buf;
|
||||||
|
int dirsep = 0; /* pattern terminates with '/' */
|
||||||
|
struct glob_pattern *list, *tmp, **tail = &list;
|
||||||
|
|
||||||
|
while (*p) {
|
||||||
|
tmp = ALLOC(struct glob_pattern);
|
||||||
|
if (p[0] == '*' && p[1] == '*' && p[2] == '/') {
|
||||||
|
do { p += 3; } while (p[0] == '*' && p[1] == '*' && p[2] == '/'); /* fold continuous RECURSIVEs */
|
||||||
|
tmp->type = RECURSIVE;
|
||||||
|
tmp->str = 0;
|
||||||
|
dirsep = 1;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const char *m;
|
||||||
|
for (m = p; *m && *m != '/'; Inc(m));
|
||||||
|
buf = ALLOC_N(char, m-p+1);
|
||||||
|
memcpy(buf, p, m-p);
|
||||||
|
buf[m-p] = '\0';
|
||||||
|
tmp->type = has_magic(buf, flags) ? MAGICAL : PLAIN;
|
||||||
|
tmp->str = buf;
|
||||||
|
if (*m) {
|
||||||
|
dirsep = 1;
|
||||||
|
p = m + 1;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
dirsep = 0;
|
||||||
|
p = m;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*tail = tmp;
|
||||||
|
tail = &tmp->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
tmp = ALLOC(struct glob_pattern);
|
||||||
|
tmp->type = dirsep ? MATCH_DIR : MATCH_ALL;
|
||||||
|
tmp->str = 0;
|
||||||
|
*tail = tmp;
|
||||||
|
tmp->next = 0;
|
||||||
|
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
glob_free_pattern(list)
|
||||||
|
struct glob_pattern *list;
|
||||||
|
{
|
||||||
|
while (list) {
|
||||||
|
struct glob_pattern *tmp = list;
|
||||||
|
list = list->next;
|
||||||
|
if (tmp->str)
|
||||||
|
free(tmp->str);
|
||||||
|
free(tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
|
join_path(path, dirsep, name)
|
||||||
|
const char *path;
|
||||||
|
int dirsep;
|
||||||
|
const char *name;
|
||||||
|
{
|
||||||
|
const int len = strlen(path);
|
||||||
|
|
||||||
|
char *buf = ALLOC_N(char, len+1+strlen(name)+1);
|
||||||
|
strcpy(buf, path);
|
||||||
|
if (dirsep) {
|
||||||
|
strcpy(buf+len, "/");
|
||||||
|
strcpy(buf+len+1, name);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
strcpy(buf+len, name);
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum answer { YES, NO, UNKNOWN };
|
||||||
|
|
||||||
#ifndef S_ISDIR
|
#ifndef S_ISDIR
|
||||||
# define S_ISDIR(m) ((m & S_IFMT) == S_IFDIR)
|
# define S_ISDIR(m) ((m & S_IFMT) == S_IFDIR)
|
||||||
#endif
|
#endif
|
||||||
|
@ -1009,160 +1109,177 @@ glob_call_func(func, path, arg)
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
glob_helper(path, sub, separator, flags, func, arg) /* if separator p[-1] is removable '/' */
|
glob_helper(path, dirsep, exist, isdir, beg, end, flags, func, arg)
|
||||||
char *path;
|
const char *path;
|
||||||
char *sub;
|
int dirsep; /* '/' should be placed before appending child entry's name to 'path'. */
|
||||||
int separator;
|
enum answer exist; /* exist? */
|
||||||
|
enum answer isdir; /* a directory or a symlink to a directory? */
|
||||||
|
struct glob_pattern **beg;
|
||||||
|
struct glob_pattern **end;
|
||||||
int flags;
|
int flags;
|
||||||
void (*func) _((const char*, VALUE));
|
void (*func) _((const char*, VALUE));
|
||||||
VALUE arg;
|
VALUE arg;
|
||||||
{
|
{
|
||||||
struct stat st;
|
struct stat st;
|
||||||
int status = 0;
|
int status = 0;
|
||||||
char *p = sub, *m, *buf;
|
struct glob_pattern **cur, **new_beg, **new_end;
|
||||||
DIR *dirp;
|
int recursive = 0, need_readdir = 0, need_plain = 0, match_all = 0, match_dir = 0;
|
||||||
struct dirent *dp;
|
const int escape = !(flags & FNM_NOESCAPE);
|
||||||
int recursive = 0;
|
|
||||||
int magical = 1;
|
|
||||||
|
|
||||||
struct d_link {
|
for (cur = beg; cur < end; ++cur) {
|
||||||
char *name;
|
struct glob_pattern *p = *cur;
|
||||||
struct d_link *next;
|
if (p->type == RECURSIVE) {
|
||||||
} *tmp, *link, **tail = &link;
|
recursive = 1;
|
||||||
|
need_readdir = 1;
|
||||||
|
p = p->next;
|
||||||
|
}
|
||||||
|
switch (p->type) {
|
||||||
|
case PLAIN:
|
||||||
|
need_plain = 1; /* ignored if need_readdir is set */
|
||||||
|
break;
|
||||||
|
case MAGICAL:
|
||||||
|
need_readdir = 1;
|
||||||
|
break;
|
||||||
|
case MATCH_ALL:
|
||||||
|
match_all = 1;
|
||||||
|
break;
|
||||||
|
case MATCH_DIR:
|
||||||
|
match_dir = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
while (*p && !has_magic(p, &m, flags)) {
|
if (match_all && exist == UNKNOWN) {
|
||||||
if (*m == '/') {
|
if (do_lstat(path, &st) == 0) {
|
||||||
separator = 1;
|
exist = YES;
|
||||||
p = m + 1;
|
isdir = S_ISDIR(st.st_mode) ? YES : S_ISLNK(st.st_mode) ? UNKNOWN : NO;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
separator = 0;
|
exist = NO;
|
||||||
p = m;
|
isdir = NO;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(flags & FNM_NOESCAPE)) {
|
if (match_dir && isdir == UNKNOWN) {
|
||||||
int n = remove_backslashes(sub, p);
|
if (do_stat(path, &st) == 0) {
|
||||||
p -= n;
|
exist = YES;
|
||||||
m -= n;
|
isdir = S_ISDIR(st.st_mode) ? YES : NO;
|
||||||
}
|
|
||||||
|
|
||||||
if (*p == '\0') { /* magic not found */
|
|
||||||
if (separator) {
|
|
||||||
p[-1] = '\0';
|
|
||||||
if (do_stat(path, &st) == 0 && S_ISDIR(st.st_mode)) {
|
|
||||||
p[-1] = '/';
|
|
||||||
return glob_call_func(func, path, arg);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
p[-1] = '/';
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (do_lstat(path, &st) == 0)
|
exist = NO;
|
||||||
return glob_call_func(func, path, arg);
|
isdir = NO;
|
||||||
}
|
}
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p[0] == '*' && p[1] == '*' && p[2] == '/') {
|
if (match_all && exist == YES) {
|
||||||
char *t = p + 3;
|
status = glob_call_func(func, path, arg);
|
||||||
while (t[0] == '*' && t[1] == '*' && t[2] == '/') t += 3;
|
|
||||||
memmove(p, t, strlen(t)+1); /* move '\0' too */
|
|
||||||
magical = has_magic(p, &m, flags); /* next element */
|
|
||||||
recursive = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (path == p) {
|
if (match_dir && isdir == YES) {
|
||||||
dirp = do_opendir(".");
|
char *buf = join_path(path, dirsep, "");
|
||||||
|
status = glob_call_func(func, buf, arg);
|
||||||
|
free(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status) return status;
|
||||||
|
|
||||||
|
if (exist == NO || isdir == NO) return 0;
|
||||||
|
|
||||||
|
if (need_readdir) {
|
||||||
|
|
||||||
|
struct dirent *dp;
|
||||||
|
|
||||||
|
DIR *dirp = do_opendir(*path ? path : ".");
|
||||||
if (dirp == NULL) return 0;
|
if (dirp == NULL) return 0;
|
||||||
}
|
|
||||||
else {
|
|
||||||
char *t = separator ? p - 1 : p;
|
|
||||||
char c = *t;
|
|
||||||
*t = '\0';
|
|
||||||
dirp = do_opendir(path);
|
|
||||||
*t = c;
|
|
||||||
if (dirp == NULL) return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (dp = readdir(dirp); dp != NULL; dp = readdir(dirp)) {
|
for (dp = readdir(dirp); dp != NULL; dp = readdir(dirp)) {
|
||||||
const int n1 = p - path;
|
|
||||||
const int n2 = n1 + NAMLEN(dp);
|
char *buf = join_path(path, dirsep, dp->d_name);
|
||||||
const int ok = 0;
|
|
||||||
const int ln = 1;
|
enum answer new_isdir = UNKNOWN;
|
||||||
const int no = 2;
|
|
||||||
int is_dir = -1; /* not checked yet */
|
if (recursive && strcmp(dp->d_name, ".") != 0 && strcmp(dp->d_name, "..") != 0) {
|
||||||
#ifdef _WIN32
|
|
||||||
is_dir = dp->d_isdir ? (dp->d_isrep ? ln : ok) : no;
|
|
||||||
#endif
|
|
||||||
if (recursive && strcmp(".", dp->d_name) != 0 && strcmp("..", dp->d_name) != 0) {
|
|
||||||
buf = ALLOC_N(char, n2+4+strlen(p)+1);
|
|
||||||
memcpy(buf, path, n1);
|
|
||||||
strcpy(buf+n1, dp->d_name);
|
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
is_dir = no;
|
if (do_lstat(buf, &st) == 0)
|
||||||
if (do_lstat(buf, &st) == 0) {
|
new_isdir = S_ISDIR(st.st_mode) ? YES : S_ISLNK(st.st_mode) ? UNKNOWN : NO;
|
||||||
if (S_ISDIR(st.st_mode))
|
else
|
||||||
is_dir = ok;
|
new_isdir = NO;
|
||||||
else if (S_ISLNK(st.st_mode) && do_stat(buf, &st) == 0 && S_ISDIR(st.st_mode))
|
#else
|
||||||
is_dir = ln;
|
new_isdir = dp->d_isdir ? YES : dp->d_isrep ? UNKNOWN : NO;
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
if (is_dir == ok) {
|
|
||||||
strcpy(buf+n2, "/**/");
|
|
||||||
strcpy(buf+n2+4, p);
|
|
||||||
status = glob_helper(buf, buf+n2+1, 1, flags, func, arg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
new_beg = new_end = ALLOC_N(struct glob_pattern *, (end - beg) * 2);
|
||||||
|
|
||||||
|
for (cur = beg; cur < end; ++cur) {
|
||||||
|
struct glob_pattern *p = *cur;
|
||||||
|
|
||||||
|
if (p->type == RECURSIVE) {
|
||||||
|
if (new_isdir == YES) /* not symlink but real directory */
|
||||||
|
*new_end++ = p; /* append recursive pattern */
|
||||||
|
p = p->next; /* 0 times recursion */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->type == PLAIN || p->type == MAGICAL) {
|
||||||
|
if (fnmatch(p->str, dp->d_name, flags) == 0)
|
||||||
|
*new_end++ = p->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
status = glob_helper(buf, 1, YES, new_isdir, new_beg, new_end, flags, func, arg);
|
||||||
|
|
||||||
|
free(new_beg);
|
||||||
free(buf);
|
free(buf);
|
||||||
|
|
||||||
if (status) break;
|
if (status) break;
|
||||||
}
|
}
|
||||||
if (is_dir == no && *m == '/') {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (magical && do_fnmatch(p, m, dp->d_name, flags) == 0) {
|
|
||||||
buf = ALLOC_N(char, n2+1+1);
|
|
||||||
memcpy(buf, path, n1);
|
|
||||||
strcpy(buf+n1, dp->d_name);
|
|
||||||
if (*m == '\0') {
|
|
||||||
status = glob_call_func(func, buf, arg);
|
|
||||||
}
|
|
||||||
else if (m[1] == '\0' && (is_dir == ok || is_dir == ln)) { /* *m == '/' */
|
|
||||||
strcpy(buf+n2, "/");
|
|
||||||
status = glob_call_func(func, buf, arg);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
tmp = ALLOC(struct d_link);
|
|
||||||
tmp->name = ALLOC_N(char, NAMLEN(dp)+1);
|
|
||||||
strcpy(tmp->name, dp->d_name);
|
|
||||||
*tail = tmp;
|
|
||||||
tail = &tmp->next;
|
|
||||||
}
|
|
||||||
free(buf);
|
|
||||||
if (status) break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
closedir(dirp);
|
closedir(dirp);
|
||||||
*tail = 0;
|
|
||||||
while (link) {
|
|
||||||
if (status == 0) {
|
|
||||||
const int n1 = p - path;
|
|
||||||
const int n2 = n1 + strlen(link->name);
|
|
||||||
buf = ALLOC_N(char, n2+strlen(m)+1);
|
|
||||||
memcpy(buf, path, n1);
|
|
||||||
strcpy(buf+n1, link->name);
|
|
||||||
strcpy(buf+n2, m);
|
|
||||||
status = glob_helper(buf, buf+n2+1, 1, flags, func, arg);
|
|
||||||
free(buf);
|
|
||||||
}
|
|
||||||
tmp = link;
|
|
||||||
link = link->next;
|
|
||||||
free(tmp->name);
|
|
||||||
free(tmp);
|
|
||||||
}
|
}
|
||||||
|
else if (need_plain) {
|
||||||
|
|
||||||
if (status == 0 && !magical) {
|
struct glob_pattern **copy_beg, **copy_end, **cur2;
|
||||||
status = glob_helper(path, p, separator, flags, func, arg);
|
|
||||||
|
copy_beg = copy_end = ALLOC_N(struct glob_pattern *, end - beg);
|
||||||
|
|
||||||
|
for (cur = beg; cur < end; ++cur)
|
||||||
|
*copy_end++ = (*cur)->type == PLAIN ? *cur : 0;
|
||||||
|
|
||||||
|
for (cur = copy_beg; cur < copy_end; ++cur) {
|
||||||
|
|
||||||
|
if (*cur) {
|
||||||
|
|
||||||
|
char *buf, *name;
|
||||||
|
|
||||||
|
name = ALLOC_N(char, strlen((*cur)->str) + 1);
|
||||||
|
strcpy(name, (*cur)->str);
|
||||||
|
if (escape) remove_backslashes(name);
|
||||||
|
|
||||||
|
new_beg = new_end = ALLOC_N(struct glob_pattern *, end - beg);
|
||||||
|
|
||||||
|
*new_end++ = (*cur)->next;
|
||||||
|
|
||||||
|
for (cur2 = cur + 1; cur2 < copy_end; ++cur2) {
|
||||||
|
if (*cur2 && fnmatch((*cur2)->str, name, flags) == 0) {
|
||||||
|
*new_end++ = (*cur2)->next;
|
||||||
|
*cur2 = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
buf = join_path(path, dirsep, name);
|
||||||
|
|
||||||
|
status = glob_helper(buf, 1, UNKNOWN, UNKNOWN, new_beg, new_end, flags, func, arg);
|
||||||
|
|
||||||
|
free(buf);
|
||||||
|
free(new_beg);
|
||||||
|
free(name);
|
||||||
|
|
||||||
|
if (status) break;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
free(copy_beg);
|
||||||
}
|
}
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
|
@ -1170,22 +1287,41 @@ glob_helper(path, sub, separator, flags, func, arg) /* if separator p[-1] is rem
|
||||||
|
|
||||||
static void
|
static void
|
||||||
rb_glob2(path, flags, func, arg)
|
rb_glob2(path, flags, func, arg)
|
||||||
char *path;
|
const char *path;
|
||||||
int flags;
|
int flags;
|
||||||
void (*func) _((const char*, VALUE));
|
void (*func) _((const char*, VALUE));
|
||||||
VALUE arg;
|
VALUE arg;
|
||||||
{
|
{
|
||||||
char *root = path;
|
struct glob_pattern *list;
|
||||||
|
const char *root = path;
|
||||||
|
char *buf;
|
||||||
|
int n;
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
|
if (flags & FNM_CASEFOLD) {
|
||||||
|
rb_warn("Dir.glob() ignores File::FNM_CASEFOLD");
|
||||||
|
}
|
||||||
|
|
||||||
#if defined DOSISH
|
#if defined DOSISH
|
||||||
flags |= FNM_CASEFOLD;
|
flags |= FNM_CASEFOLD;
|
||||||
root = rb_path_skip_prefix(root);
|
root = rb_path_skip_prefix(root);
|
||||||
|
#else
|
||||||
|
flags &= ~FNM_CASEFOLD;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (*root == '/') root++;
|
if (*root == '/') root++;
|
||||||
|
|
||||||
status = glob_helper(path, root, 0, flags, func, arg);
|
n = root - path;
|
||||||
|
buf = ALLOC_N(char, n+1);
|
||||||
|
memcpy(buf, path, n);
|
||||||
|
buf[n] = '\0';
|
||||||
|
|
||||||
|
list = glob_make_pattern(root, flags);
|
||||||
|
status = glob_helper(buf, 0, UNKNOWN, UNKNOWN, &list, &list + 1, flags, func, arg);
|
||||||
|
glob_free_pattern(list);
|
||||||
|
|
||||||
|
free(buf);
|
||||||
|
|
||||||
if (status) rb_jump_tag(status);
|
if (status) rb_jump_tag(status);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1198,15 +1334,6 @@ rb_glob(path, func, arg)
|
||||||
rb_glob2(path, 0, func, arg);
|
rb_glob2(path, 0, func, arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
rb_globi(path, func, arg)
|
|
||||||
char *path;
|
|
||||||
void (*func) _((const char*, VALUE));
|
|
||||||
VALUE arg;
|
|
||||||
{
|
|
||||||
rb_glob2(path, FNM_CASEFOLD, func, arg);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
push_pattern(path, ary)
|
push_pattern(path, ary)
|
||||||
const char *path;
|
const char *path;
|
||||||
|
|
1
ruby.h
1
ruby.h
|
@ -469,7 +469,6 @@ struct RBignum {
|
||||||
void rb_obj_infect _((VALUE,VALUE));
|
void rb_obj_infect _((VALUE,VALUE));
|
||||||
|
|
||||||
void rb_glob _((char*,void(*)(const char*,VALUE),VALUE));
|
void rb_glob _((char*,void(*)(const char*,VALUE),VALUE));
|
||||||
void rb_globi _((char*,void(*)(const char*,VALUE),VALUE));
|
|
||||||
|
|
||||||
VALUE rb_define_class _((const char*,VALUE));
|
VALUE rb_define_class _((const char*,VALUE));
|
||||||
VALUE rb_define_module _((const char*));
|
VALUE rb_define_module _((const char*));
|
||||||
|
|
2
util.c
2
util.c
|
@ -354,7 +354,7 @@ __crt0_glob_function(char *path)
|
||||||
info.count = 0;
|
info.count = 0;
|
||||||
info.head = 0;
|
info.head = 0;
|
||||||
|
|
||||||
rb_globi(buf, push_element, (VALUE)&info);
|
rb_glob(buf, push_element, (VALUE)&info);
|
||||||
|
|
||||||
if (buf != path_buffer)
|
if (buf != path_buffer)
|
||||||
ruby_xfree(buf);
|
ruby_xfree(buf);
|
||||||
|
|
|
@ -1019,7 +1019,7 @@ cmdglob(NtCmdLineElement *patt, NtCmdLineElement **tail)
|
||||||
for (p = buf; *p; p = CharNext(p))
|
for (p = buf; *p; p = CharNext(p))
|
||||||
if (*p == '\\')
|
if (*p == '\\')
|
||||||
*p = '/';
|
*p = '/';
|
||||||
rb_globi(buf, insert, (VALUE)&tail);
|
rb_glob(buf, insert, (VALUE)&tail);
|
||||||
if (buf != buffer)
|
if (buf != buffer)
|
||||||
free(buf);
|
free(buf);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue