2011-11-08 13:35:38 -05:00
|
|
|
/* See LICENSE file for copyright and license details. */
|
2015-02-14 15:02:41 -05:00
|
|
|
#include <string.h>
|
2011-11-08 13:35:38 -05:00
|
|
|
#include <unistd.h>
|
2015-03-04 20:28:18 -05:00
|
|
|
#include <sys/stat.h>
|
2014-11-13 12:29:30 -05:00
|
|
|
|
2011-11-08 13:35:38 -05:00
|
|
|
#include "util.h"
|
|
|
|
|
2015-02-09 16:21:23 -05:00
|
|
|
#define STOI(s) enstrtonum(2, s, LLONG_MIN, LLONG_MAX)
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2016-03-24 19:18:02 -04:00
|
|
|
static int
|
|
|
|
mtimecmp(struct stat *buf1, struct stat *buf2)
|
|
|
|
{
|
|
|
|
if (buf1->st_mtime < buf2->st_mtime) return -1;
|
|
|
|
if (buf1->st_mtime > buf2->st_mtime) return +1;
|
|
|
|
#ifdef st_mtime
|
|
|
|
if (buf1->st_mtim.tv_nsec < buf2->st_mtim.tv_nsec) return -1;
|
|
|
|
if (buf1->st_mtim.tv_nsec > buf2->st_mtim.tv_nsec) return +1;
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-11-13 15:24:47 -05:00
|
|
|
static int unary_b(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISBLK (buf.st_mode); }
|
|
|
|
static int unary_c(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISCHR (buf.st_mode); }
|
|
|
|
static int unary_d(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISDIR (buf.st_mode); }
|
|
|
|
static int unary_f(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISREG (buf.st_mode); }
|
|
|
|
static int unary_g(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISGID & buf.st_mode ; }
|
|
|
|
static int unary_h(char *s) { struct stat buf; if (lstat(s, &buf)) return 0; return S_ISLNK (buf.st_mode); }
|
2016-03-24 19:18:02 -04:00
|
|
|
static int unary_k(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISVTX & buf.st_mode ; }
|
2014-11-13 15:24:47 -05:00
|
|
|
static int unary_p(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISFIFO (buf.st_mode); }
|
|
|
|
static int unary_S(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISSOCK (buf.st_mode); }
|
|
|
|
static int unary_s(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return buf.st_size ; }
|
|
|
|
static int unary_u(char *s) { struct stat buf; if ( stat(s, &buf)) return 0; return S_ISUID & buf.st_mode ; }
|
|
|
|
|
2015-02-20 14:19:02 -05:00
|
|
|
static int unary_n(char *s) { return *s; }
|
|
|
|
static int unary_z(char *s) { return !*s; }
|
2014-11-13 15:24:47 -05:00
|
|
|
|
2015-03-13 19:44:18 -04:00
|
|
|
static int unary_e(char *s) { return !access(s, F_OK); }
|
|
|
|
static int unary_r(char *s) { return !access(s, R_OK); }
|
|
|
|
static int unary_w(char *s) { return !access(s, W_OK); }
|
|
|
|
static int unary_x(char *s) { return !access(s, X_OK); }
|
2014-11-13 15:24:47 -05:00
|
|
|
|
2015-02-09 16:21:23 -05:00
|
|
|
static int unary_t(char *s) { int fd = enstrtonum(2, s, 0, INT_MAX); return isatty(fd); }
|
2014-11-13 15:24:47 -05:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
static int binary_se(char *s1, char *s2) { return !strcmp(s1, s2); }
|
|
|
|
static int binary_sn(char *s1, char *s2) { return strcmp(s1, s2); }
|
2014-11-13 15:24:47 -05:00
|
|
|
|
2015-02-09 16:21:23 -05:00
|
|
|
static int binary_eq(char *s1, char *s2) { long long a = STOI(s1), b = STOI(s2); return a == b; }
|
|
|
|
static int binary_ne(char *s1, char *s2) { long long a = STOI(s1), b = STOI(s2); return a != b; }
|
|
|
|
static int binary_gt(char *s1, char *s2) { long long a = STOI(s1), b = STOI(s2); return a > b; }
|
|
|
|
static int binary_ge(char *s1, char *s2) { long long a = STOI(s1), b = STOI(s2); return a >= b; }
|
|
|
|
static int binary_lt(char *s1, char *s2) { long long a = STOI(s1), b = STOI(s2); return a < b; }
|
|
|
|
static int binary_le(char *s1, char *s2) { long long a = STOI(s1), b = STOI(s2); return a <= b; }
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2016-03-24 19:18:02 -04:00
|
|
|
static int
|
|
|
|
binary_ef(char *s1, char *s2)
|
|
|
|
{
|
|
|
|
struct stat buf1, buf2;
|
|
|
|
if (stat(s1, &buf1) || stat(s2, &buf2)) return 0;
|
|
|
|
return buf1.st_dev == buf2.st_dev && buf1.st_ino == buf2.st_ino;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
binary_ot(char *s1, char *s2)
|
|
|
|
{
|
|
|
|
struct stat buf1, buf2;
|
|
|
|
if (stat(s1, &buf1) || stat(s2, &buf2)) return 0;
|
|
|
|
return mtimecmp(&buf1, &buf2) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
binary_nt(char *s1, char *s2)
|
|
|
|
{
|
|
|
|
struct stat buf1, buf2;
|
|
|
|
if (stat(s1, &buf1) || stat(s2, &buf2)) return 0;
|
|
|
|
return mtimecmp(&buf1, &buf2) > 0;
|
|
|
|
}
|
|
|
|
|
2015-03-17 13:59:16 -04:00
|
|
|
struct test {
|
2014-10-17 10:44:29 -04:00
|
|
|
char *name;
|
2014-11-13 15:24:47 -05:00
|
|
|
int (*func)();
|
2015-03-17 13:59:16 -04:00
|
|
|
};
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2015-03-17 13:59:16 -04:00
|
|
|
static struct test unary[] = {
|
2014-10-17 10:44:29 -04:00
|
|
|
{ "-b", unary_b },
|
|
|
|
{ "-c", unary_c },
|
|
|
|
{ "-d", unary_d },
|
|
|
|
{ "-e", unary_e },
|
|
|
|
{ "-f", unary_f },
|
|
|
|
{ "-g", unary_g },
|
|
|
|
{ "-h", unary_h },
|
2016-03-24 19:18:02 -04:00
|
|
|
{ "-k", unary_k },
|
2014-10-17 10:44:29 -04:00
|
|
|
{ "-L", unary_h },
|
|
|
|
{ "-n", unary_n },
|
|
|
|
{ "-p", unary_p },
|
|
|
|
{ "-r", unary_r },
|
|
|
|
{ "-S", unary_S },
|
|
|
|
{ "-s", unary_s },
|
|
|
|
{ "-t", unary_t },
|
|
|
|
{ "-u", unary_u },
|
|
|
|
{ "-w", unary_w },
|
|
|
|
{ "-x", unary_x },
|
|
|
|
{ "-z", unary_z },
|
|
|
|
|
|
|
|
{ NULL, NULL },
|
2014-10-16 05:10:00 -04:00
|
|
|
};
|
|
|
|
|
2015-03-17 13:59:16 -04:00
|
|
|
static struct test binary[] = {
|
2014-10-17 10:44:29 -04:00
|
|
|
{ "=" , binary_se },
|
|
|
|
{ "!=" , binary_sn },
|
|
|
|
{ "-eq", binary_eq },
|
|
|
|
{ "-ne", binary_ne },
|
|
|
|
{ "-gt", binary_gt },
|
|
|
|
{ "-ge", binary_ge },
|
|
|
|
{ "-lt", binary_lt },
|
|
|
|
{ "-le", binary_le },
|
2016-03-24 19:18:02 -04:00
|
|
|
{ "-ef", binary_ef },
|
|
|
|
{ "-ot", binary_ot },
|
|
|
|
{ "-nt", binary_nt },
|
2014-04-22 09:44:16 -04:00
|
|
|
|
2014-10-17 10:44:29 -04:00
|
|
|
{ NULL, NULL },
|
|
|
|
};
|
|
|
|
|
2015-03-17 13:59:16 -04:00
|
|
|
static struct test *
|
|
|
|
find_test(struct test *tests, char *name)
|
2014-04-22 09:44:16 -04:00
|
|
|
{
|
2015-03-17 13:59:16 -04:00
|
|
|
struct test *t;
|
2014-04-22 09:44:16 -04:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
for (t = tests; t->name; t++)
|
|
|
|
if (!strcmp(t->name, name))
|
2014-10-17 10:44:29 -04:00
|
|
|
return t;
|
2015-03-17 18:35:11 -04:00
|
|
|
|
2014-10-17 10:44:29 -04:00
|
|
|
return NULL;
|
2014-04-22 09:44:16 -04:00
|
|
|
}
|
2011-11-08 13:35:38 -05:00
|
|
|
|
2014-11-13 15:24:47 -05:00
|
|
|
static int
|
2015-03-04 20:28:18 -05:00
|
|
|
noarg(char *argv[])
|
2014-10-17 10:44:29 -04:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-11-13 15:24:47 -05:00
|
|
|
static int
|
2015-03-04 20:28:18 -05:00
|
|
|
onearg(char *argv[])
|
2011-11-08 13:35:38 -05:00
|
|
|
{
|
2015-02-22 14:37:48 -05:00
|
|
|
return unary_n(argv[0]);
|
2011-11-08 13:35:38 -05:00
|
|
|
}
|
|
|
|
|
2014-11-13 15:24:47 -05:00
|
|
|
static int
|
2015-03-04 20:28:18 -05:00
|
|
|
twoarg(char *argv[])
|
2011-11-08 13:35:38 -05:00
|
|
|
{
|
2015-03-17 13:59:16 -04:00
|
|
|
struct test *t;
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
if (!strcmp(argv[0], "!"))
|
2014-10-17 10:44:29 -04:00
|
|
|
return !onearg(argv + 1);
|
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
if ((t = find_test(unary, *argv)))
|
2014-10-17 10:44:29 -04:00
|
|
|
return t->func(argv[1]);
|
|
|
|
|
2015-02-25 12:01:20 -05:00
|
|
|
enprintf(2, "bad unary test %s\n", argv[0]);
|
2015-03-17 18:35:11 -04:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
return 0; /* not reached */
|
2011-11-08 13:35:38 -05:00
|
|
|
}
|
|
|
|
|
2014-11-13 15:24:47 -05:00
|
|
|
static int
|
2015-03-04 20:28:18 -05:00
|
|
|
threearg(char *argv[])
|
2011-11-08 13:35:38 -05:00
|
|
|
{
|
2015-03-17 13:59:16 -04:00
|
|
|
struct test *t = find_test(binary, argv[1]);
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2014-11-13 12:29:30 -05:00
|
|
|
if (t)
|
2014-10-17 10:44:29 -04:00
|
|
|
return t->func(argv[0], argv[2]);
|
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
if (!strcmp(argv[0], "!"))
|
2014-10-17 10:44:29 -04:00
|
|
|
return !twoarg(argv + 1);
|
|
|
|
|
2015-02-25 12:01:20 -05:00
|
|
|
enprintf(2, "bad binary test %s\n", argv[1]);
|
2015-03-17 18:35:11 -04:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
return 0; /* not reached */
|
2014-10-17 10:44:29 -04:00
|
|
|
}
|
|
|
|
|
2014-11-13 15:24:47 -05:00
|
|
|
static int
|
2015-03-04 20:28:18 -05:00
|
|
|
fourarg(char *argv[])
|
2014-10-17 10:44:29 -04:00
|
|
|
{
|
2015-03-04 20:28:18 -05:00
|
|
|
if (!strcmp(argv[0], "!"))
|
2014-10-17 10:44:29 -04:00
|
|
|
return !threearg(argv + 1);
|
|
|
|
|
2015-02-25 12:01:20 -05:00
|
|
|
enprintf(2, "too many arguments\n");
|
2015-03-17 18:35:11 -04:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
return 0; /* not reached */
|
2014-10-17 10:44:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2015-03-04 20:28:18 -05:00
|
|
|
main(int argc, char *argv[])
|
2014-10-17 10:44:29 -04:00
|
|
|
{
|
2015-03-04 20:28:18 -05:00
|
|
|
int (*narg[])(char *[]) = { noarg, onearg, twoarg, threearg, fourarg };
|
|
|
|
size_t len;
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2015-03-13 18:43:13 -04:00
|
|
|
argv0 = argv[0], argc--, argv++;
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2015-03-04 20:28:18 -05:00
|
|
|
len = strlen(argv0);
|
|
|
|
if (len && argv0[--len] == '[' && (!len || argv0[--len] == '/') && strcmp(argv[--argc], "]"))
|
|
|
|
enprintf(2, "no matching ]\n");
|
2014-10-17 10:44:29 -04:00
|
|
|
|
2014-11-13 12:29:30 -05:00
|
|
|
if (argc > 4)
|
2014-10-17 10:44:29 -04:00
|
|
|
enprintf(2, "too many arguments\n");
|
|
|
|
|
|
|
|
return !narg[argc](argv);
|
2011-11-08 13:35:38 -05:00
|
|
|
}
|