📄 glob.c
字号:
/* glob 命令 --
glob params
在 params 中的 "*" 匹配正则表达式 ".*"
在 params 中的 "?" 匹配正则表达式 "."
在 params 中的 "[...]" 匹配字符类
在 params 中的 "[!...]" 匹配字符类的补集。
在 params 中的 "[...a-z...]" 匹配 a 到 z。
执行命令并带有按如下规则构造的参数列表:
如果 param 不包含 "*","[",或 "?",则按原样使用它
如果包含,则在当前目录中找到匹配这个 param 的所有文件,
排序并使用它们。
按需要为命令名字添加 "/bin" 或 "/usr/bin" 前缀。
*/
#include <stdlib.h>
#include <limits.h>
#include <errno.h>
#include <unistd.h>
#include <dirent.h>
#define QUOTE 0x80 /* 引用标志位 */
#define STRSIZ _POSIX_ARG_MAX
extern char *arg0; /* 调用 sh 的全路径 */
static char ab[STRSIZ]; /* 生成的字符串的存储空间 */
static char *string;
static char *ava[STRSIZ/2]; /* 生成的参数列表 */
static char **av;
static int ncoll;
int glob(int argc, char *argv[]);
static int gexec(char *afile, char **aarg);
static void expand(char *as);
static void sort(char **oav);
static void toolong(void);
static int match(char *s, char *p);
static int amatch(char *as, char *ap);
static int umatch(char *s, char *p);
static int compare(char *as1, char *as2);
static char* cat(char *as1, char *as2);
int glob(int argc, char *argv[])
{
char *cp;
int r;
string = ab;
av = &ava[1]; /* 有意留出一个空位 */
if (argc < 2) {
write(STDERR_FILENO, "Arg count\n", 10);
exit (1);
}
argv++;
*av++ = *argv; /* 指向第一个参数,它是要执行的文件的名字 */
while (--argc >= 2)
expand(*++argv); /* 展开余下的所有的参数 */
if ((r = gexec(ava[1], &ava[1])))
return r;
cp = cat("/usr/bin/", ava[1]);
if ((r = gexec(cp+4, &ava[1])))
return r;
if ((r = gexec(cp, &ava[1])))
return r;
write(STDERR_FILENO, "Command not found.\n", 19);
return 127;
}
void expand(char *as)
{
register char *s, *cs;
struct dirent * dirp;
DIR *dp;
char **oav;
ncoll = 0;
s = cs = as;
/* 把 cs 定位到第一个通配符 */
while (*cs!='*' && *cs!='?' && *cs!='[') {
if (*cs++ == 0) { /* 没有找到通配符 */
*av++ = cat(s, "");
return;
}
}
for (;;) {
if (cs==s) { /* 在通配符之前没有'/' */
dp = opendir(".");
s = "";
break;
}
if (*--cs == '/') {
*cs = 0; /* 把参数分开为目录和文件名两个字符串 */
dp = opendir((s==cs)? "/": s);
*cs++ = QUOTE; /* 做标记,在后面的 cat 操作中恢复成斜杠 */
break;
}
}
if (dp == NULL) {
write(STDERR_FILENO, "No directory\n", 13);
exit (1);
}
oav = av;
while ((dirp = readdir(dp)) != NULL) {
if (match(dirp->d_name, cs)) {
*av++ = cat(s, dirp->d_name);
ncoll++;
}
}
if (!ncoll) /* 没有匹配 */
*av++ = cat(s, cs); /* 保持参数为原状 */
else /* 排序匹配的文件名字 */
sort(oav);
closedir(dp);
}
/* 排序 */
void sort(char **oav)
{
register char **p1, **p2, *c;
p1 = oav;
while (++p1 < av) {
c = *p1;
p2 = p1;
while (--p2 >= oav && compare(*p2, c) > 0)
*(p2+1) = *p2;
*(p2+1) = c;
}
}
/* 执行一个命令 */
int gexec(char *afile, char **aarg)
{
register char *file, **arg;
arg = aarg;
file = afile;
execv(file, arg);
if (errno == EACCES) {
write(STDERR_FILENO, "Permission denied\n", 18);
return 126;
}
if (errno==ENOEXEC) { /* 不是二进制可执行文件 */
arg[0] = file;
*--arg = arg0;
execv(*arg, arg); /* 作为命令文件执行 */
}
if (errno==E2BIG) {/* 在参数列表太长的时候退出 */
toolong();
return E2BIG;
}
return 0;
}
/* 打印错误消息并退出 */
void toolong(void)
{
write(2, "Arg list too long\n", 18);
}
/*匹配判断例程 */
int match(char *s, char *p)
{
if (*s=='.' && *p!='.') /* 对 . 开头的文件的处理 */
return 0;
return amatch(s, p);
}
/* 对字符的匹配 */
int amatch(char *as, char *ap)
{
register char *s, *p;
register int scc;
int c, cc, ok, lc;
int neg = 0;
s = as;
p = ap;
if ((scc = *s++) != 0) /* 文件名未终结 */
/* 传递给 glob 的参数中的字符可能设置了引用位,
这些字符在 shell 命令行中位于" "或' ' 中 */
if ((scc &= ~QUOTE) == 0) /* 如果 scc 清除了引用位之后是 0 */
scc = QUOTE; /* 重新把它设置为 QUOTE */
switch (c = *p++) {
case '[': /* 处理字符类 */
if (*p=='!') { /* 下个字符是 ! */
neg = 1;
p++;
}
ok = 0;
lc = INT_MAX;
while ((cc = *p++) != 0) {
if (cc==']') {
if ((ok && !neg) || (!ok && neg))
return amatch(s, p); /* 如果匹配,继续比较后面的字符 */
else
return 0;
} else if (cc=='-') {
if (lc<=scc && scc<=(c = *p++)) /* 在范围内 */
ok++;
} else
if (scc == (lc=cc))
ok++;
}
return 0;
default:
if (c!=scc)
return 0;
case '?':
if (scc)
return amatch(s, p);
return 0;
case '*':
return umatch(--s, p);
case '\0':
return !scc;
}
}
/* 对闭包的匹配 */
int umatch(char *s, char *p)
{
if(*p==0) /* 模式以 * 结束 */
return 1;
while(*s) /* 文件名中有字符匹配模式中 * 后面的字符 */
if (amatch(s++,p))
return 1;
return 0;
}
/* 比较两个字符串 */
int compare(char *as1, char *as2)
{
register char *s1, *s2;
s1 = as1;
s2 = as2;
while (*s1++ == *s2)
if (*s2++ == 0)
return 0;
return (*--s1 - *s2);
}
/* 联接两个字符串到 string 指向的数组中并返回它,
* 副作用是 string 指针右移,指向的数组空间减小 */
char* cat(char *as1, char *as2)
{
register char *s1, *s2;
register int c;
s2 = string;
s1 = as1;
while ((c = *s1++) != 0) {
if (s2 > &ab[STRSIZ])
toolong();
c &= ~QUOTE; /* 清除引用位 */
if (c==0) { /* 原先是路径分隔符 */
*s2++ = '/';
break;
}
*s2++ = c;
}
s1 = as2;
do {
if (s2 > &ab[STRSIZ])
toolong();
*s2++ = c = *s1++;
} while (c);
s1 = string;
string = s2;
return s1;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -