logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/

test.c (11239B)


  1. // SPDX-License-Identifier: 0BSD
  2. // Copyright Erik Baalbergen, Eric Gisin, Arnold Robbins, J.T. Conklin
  3. /* $NetBSD: test.c,v 1.21 1999/04/05 09:48:38 kleink Exp $ */
  4. /*-
  5. * test(1); version 7-like -- author Erik Baalbergen
  6. * modified by Eric Gisin to be used as built-in.
  7. * modified by Arnold Robbins to add SVR3 compatibility
  8. * (-x -c -b -p -u -g -k) plus Korn's -L -nt -ot -ef and new -S (socket).
  9. * modified by J.T. Conklin for NetBSD.
  10. *
  11. * This program is in the Public Domain.
  12. */
  13. #define _POSIX_C_SOURCE 200809L
  14. #define _XOPEN_SOURCE 700 // S_ISVTX for -k option
  15. #include <ctype.h>
  16. #include <errno.h>
  17. #include <inttypes.h>
  18. #include <locale.h>
  19. #include <stdarg.h>
  20. #include <stdio.h>
  21. #include <stdlib.h>
  22. #include <string.h>
  23. #include <sys/stat.h>
  24. #include <sys/types.h>
  25. #include <unistd.h>
  26. const char *argv0 = "test";
  27. _Noreturn static void
  28. error(const char *msg, ...)
  29. {
  30. va_list ap;
  31. fprintf(stderr, "%s: error: ", argv0);
  32. va_start(ap, msg);
  33. vfprintf(stderr, msg, ap);
  34. va_end(ap);
  35. putc('\n', stderr);
  36. exit(2);
  37. }
  38. /* test(1) accepts the following grammar:
  39. oexpr ::= aexpr | aexpr "-o" oexpr ;
  40. aexpr ::= nexpr | nexpr "-a" aexpr ;
  41. nexpr ::= primary | "!" primary
  42. primary ::= unary-operator operand
  43. | operand binary-operator operand
  44. | operand
  45. | "(" oexpr ")"
  46. ;
  47. unary-operator ::= "-r"|"-w"|"-x"|"-f"|"-d"|"-c"|"-b"|"-p"|
  48. "-u"|"-g"|"-k"|"-s"|"-t"|"-z"|"-n"|"-o"|"-O"|"-G"|"-L"|"-S";
  49. binary-operator ::= "="|"!="|"-eq"|"-ne"|"-ge"|"-gt"|"-le"|"-lt"|
  50. "-nt"|"-ot"|"-ef";
  51. operand ::= <any legal UNIX file name>
  52. */
  53. enum token_types
  54. {
  55. UNOP = 0x100,
  56. BINOP = 0x200,
  57. BUNOP = 0x300,
  58. BBINOP = 0x400,
  59. PAREN = 0x500
  60. };
  61. enum token
  62. {
  63. EOI,
  64. OPERAND,
  65. FILRD = UNOP + 1,
  66. FILWR,
  67. FILEX,
  68. FILEXIST,
  69. FILREG,
  70. FILDIR,
  71. FILCDEV,
  72. FILBDEV,
  73. FILFIFO,
  74. FILSOCK,
  75. FILSYM,
  76. FILGZ,
  77. FILTT,
  78. FILSUID,
  79. FILSGID,
  80. FILSTCK,
  81. STREZ,
  82. STRNZ,
  83. FILUID,
  84. FILGID,
  85. FILNT = BINOP + 1,
  86. FILOT,
  87. FILEQ,
  88. STREQ,
  89. STRNE,
  90. STRLT,
  91. STRGT,
  92. INTEQ,
  93. INTNE,
  94. INTGE,
  95. INTGT,
  96. INTLE,
  97. INTLT,
  98. UNOT = BUNOP + 1,
  99. BAND = BBINOP + 1,
  100. BOR,
  101. LPAREN = PAREN + 1,
  102. RPAREN
  103. };
  104. #define TOKEN_TYPE(token) ((token) & 0xff00)
  105. // clang-format off
  106. static const struct t_op
  107. {
  108. char op_text[2];
  109. short op_num;
  110. } ops1[] =
  111. {
  112. {"=", STREQ},
  113. {"<", STRLT},
  114. {">", STRGT},
  115. {"!", UNOT},
  116. {"(", LPAREN},
  117. {")", RPAREN},
  118. },
  119. opsm1[] = {
  120. {"r", FILRD},
  121. {"w", FILWR},
  122. {"x", FILEX},
  123. {"e", FILEXIST},
  124. {"f", FILREG},
  125. {"d", FILDIR},
  126. {"c", FILCDEV},
  127. {"b", FILBDEV},
  128. {"p", FILFIFO},
  129. {"u", FILSUID},
  130. {"g", FILSGID},
  131. {"k", FILSTCK},
  132. {"s", FILGZ},
  133. {"t", FILTT},
  134. {"z", STREZ},
  135. {"n", STRNZ},
  136. {"h", FILSYM}, /* for backwards compat */
  137. {"O", FILUID},
  138. {"G", FILGID},
  139. {"L", FILSYM},
  140. {"S", FILSOCK},
  141. {"a", BAND},
  142. {"o", BOR},
  143. },
  144. ops2[] = {
  145. {"==", STREQ},
  146. {"!=", STRNE},
  147. },
  148. opsm2[] = {
  149. {"eq", INTEQ},
  150. {"ne", INTNE},
  151. {"ge", INTGE},
  152. {"gt", INTGT},
  153. {"le", INTLE},
  154. {"lt", INTLT},
  155. {"nt", FILNT},
  156. {"ot", FILOT},
  157. {"ef", FILEQ},
  158. };
  159. // clang-format on
  160. static int nargc;
  161. static char **t_wp;
  162. static int parenlevel;
  163. static int aexpr(enum token);
  164. static int binop(enum token);
  165. static int equalf(const char *, const char *);
  166. static int filstat(char *, enum token);
  167. static int getn(const char *);
  168. static intmax_t getq(const char *);
  169. static int intcmp(const char *, const char *);
  170. static int isunopoperand(void);
  171. static int islparenoperand(void);
  172. static int isrparenoperand(void);
  173. static int newerf(const char *, const char *);
  174. static int nexpr(enum token);
  175. static int oexpr(enum token);
  176. static int olderf(const char *, const char *);
  177. static int primary(enum token);
  178. static void syntax(const char *, const char *);
  179. static enum token t_lex(char *);
  180. int
  181. main(int argc, char **argv)
  182. {
  183. int res;
  184. char *p;
  185. if((p = strrchr(argv[0], '/')) == NULL)
  186. p = argv[0];
  187. else
  188. p++;
  189. if(strcmp(p, "[") == 0)
  190. {
  191. argv0 = "[";
  192. if(strcmp(argv[--argc], "]") != 0) error("missing ]");
  193. argv[argc] = NULL;
  194. }
  195. /* no expression => false */
  196. if(--argc <= 0) return 1;
  197. char *lc_ctype = setlocale(LC_CTYPE, "");
  198. if(lc_ctype == NULL)
  199. {
  200. fprintf(stderr,
  201. "%s: warning: Failed loading locales. setlocale(LC_CTYPE, \"\"): %s\n",
  202. argv0,
  203. strerror(errno));
  204. }
  205. errno = 0;
  206. nargc = argc;
  207. t_wp = &argv[1];
  208. parenlevel = 0;
  209. if(nargc == 4 && strcmp(*t_wp, "!") == 0)
  210. {
  211. /* Things like ! "" -o x do not fit in the normal grammar. */
  212. --nargc;
  213. ++t_wp;
  214. res = oexpr(t_lex(*t_wp));
  215. }
  216. else
  217. res = !oexpr(t_lex(*t_wp));
  218. if(--nargc > 0) syntax(*t_wp, "unexpected operator");
  219. return res;
  220. }
  221. static void
  222. syntax(const char *op, const char *msg)
  223. {
  224. if(op && *op)
  225. error("%s: %s", op, msg);
  226. else
  227. error("%s", msg);
  228. }
  229. static int
  230. oexpr(enum token n)
  231. {
  232. int res;
  233. res = aexpr(n);
  234. if(t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL) == BOR)
  235. return oexpr(t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL)) || res;
  236. t_wp--;
  237. nargc++;
  238. return res;
  239. }
  240. static int
  241. aexpr(enum token n)
  242. {
  243. int res;
  244. res = nexpr(n);
  245. if(t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL) == BAND)
  246. return aexpr(t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL)) && res;
  247. t_wp--;
  248. nargc++;
  249. return res;
  250. }
  251. static int
  252. nexpr(enum token n)
  253. {
  254. if(n == UNOT) return !nexpr(t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL));
  255. return primary(n);
  256. }
  257. static int
  258. primary(enum token n)
  259. {
  260. enum token nn;
  261. int res;
  262. if(n == EOI) return 0; /* missing expression */
  263. if(n == LPAREN)
  264. {
  265. parenlevel++;
  266. if((nn = t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL)) == RPAREN)
  267. {
  268. parenlevel--;
  269. return 0; /* missing expression */
  270. }
  271. res = oexpr(nn);
  272. if(t_lex(nargc > 0 ? (--nargc, *++t_wp) : NULL) != RPAREN)
  273. syntax(NULL, "closing paren expected");
  274. parenlevel--;
  275. return res;
  276. }
  277. if(TOKEN_TYPE(n) == UNOP)
  278. {
  279. /* unary expression */
  280. if(--nargc == 0) syntax(NULL, "argument expected"); /* impossible */
  281. switch(n)
  282. {
  283. case STREZ:
  284. return strlen(*++t_wp) == 0;
  285. case STRNZ:
  286. return strlen(*++t_wp) != 0;
  287. case FILTT:
  288. return isatty(getn(*++t_wp));
  289. default:
  290. return filstat(*++t_wp, n);
  291. }
  292. }
  293. nn = t_lex(nargc > 0 ? t_wp[1] : NULL);
  294. if(TOKEN_TYPE(nn) == BINOP) return binop(nn);
  295. return strlen(*t_wp) > 0;
  296. }
  297. static int
  298. binop(enum token n)
  299. {
  300. const char *opnd1, *op, *opnd2;
  301. opnd1 = *t_wp;
  302. op = nargc > 0 ? (--nargc, *++t_wp) : NULL;
  303. if((opnd2 = nargc > 0 ? (--nargc, *++t_wp) : NULL) == NULL)
  304. {
  305. syntax(op, "argument expected");
  306. return 0;
  307. }
  308. switch(n)
  309. {
  310. case STREQ:
  311. return strcmp(opnd1, opnd2) == 0;
  312. case STRNE:
  313. return strcmp(opnd1, opnd2) != 0;
  314. case STRLT:
  315. return strcmp(opnd1, opnd2) < 0;
  316. case STRGT:
  317. return strcmp(opnd1, opnd2) > 0;
  318. case INTEQ:
  319. return intcmp(opnd1, opnd2) == 0;
  320. case INTNE:
  321. return intcmp(opnd1, opnd2) != 0;
  322. case INTGE:
  323. return intcmp(opnd1, opnd2) >= 0;
  324. case INTGT:
  325. return intcmp(opnd1, opnd2) > 0;
  326. case INTLE:
  327. return intcmp(opnd1, opnd2) <= 0;
  328. case INTLT:
  329. return intcmp(opnd1, opnd2) < 0;
  330. case FILNT:
  331. return newerf(opnd1, opnd2);
  332. case FILOT:
  333. return olderf(opnd1, opnd2);
  334. case FILEQ:
  335. return equalf(opnd1, opnd2);
  336. default:
  337. abort();
  338. /* NOTREACHED */
  339. }
  340. }
  341. static int
  342. filstat(char *nm, enum token mode)
  343. {
  344. struct stat s;
  345. if(mode == FILSYM ? lstat(nm, &s) : stat(nm, &s)) return 0;
  346. switch(mode)
  347. {
  348. case FILRD:
  349. return (access(nm, R_OK) == 0);
  350. case FILWR:
  351. return (access(nm, W_OK) == 0);
  352. case FILEX:
  353. return (access(nm, X_OK) == 0);
  354. case FILEXIST:
  355. return (access(nm, F_OK) == 0);
  356. case FILREG:
  357. return S_ISREG(s.st_mode);
  358. case FILDIR:
  359. return S_ISDIR(s.st_mode);
  360. case FILCDEV:
  361. return S_ISCHR(s.st_mode);
  362. case FILBDEV:
  363. return S_ISBLK(s.st_mode);
  364. case FILFIFO:
  365. return S_ISFIFO(s.st_mode);
  366. case FILSOCK:
  367. return S_ISSOCK(s.st_mode);
  368. case FILSYM:
  369. return S_ISLNK(s.st_mode);
  370. case FILSUID:
  371. return (s.st_mode & S_ISUID) != 0;
  372. case FILSGID:
  373. return (s.st_mode & S_ISGID) != 0;
  374. case FILSTCK:
  375. return (s.st_mode & S_ISVTX) != 0;
  376. case FILGZ:
  377. return s.st_size > (off_t)0;
  378. case FILUID:
  379. return s.st_uid == geteuid();
  380. case FILGID:
  381. return s.st_gid == getegid();
  382. default:
  383. return 1;
  384. }
  385. }
  386. static int
  387. find_op_1char(const struct t_op *op, const struct t_op *end, const char *s)
  388. {
  389. char c;
  390. c = s[0];
  391. while(op != end)
  392. {
  393. if(c == *op->op_text) return op->op_num;
  394. op++;
  395. }
  396. return OPERAND;
  397. }
  398. static int
  399. find_op_2char(const struct t_op *op, const struct t_op *end, const char *s)
  400. {
  401. while(op != end)
  402. {
  403. if(s[0] == op->op_text[0] && s[1] == op->op_text[1]) return op->op_num;
  404. op++;
  405. }
  406. return OPERAND;
  407. }
  408. static int
  409. find_op(const char *s)
  410. {
  411. if(s[0] == '\0')
  412. return OPERAND;
  413. else if(s[1] == '\0')
  414. return find_op_1char(ops1, (&ops1)[1], s);
  415. else if(s[2] == '\0')
  416. return s[0] == '-' ? find_op_1char(opsm1, (&opsm1)[1], s + 1)
  417. : find_op_2char(ops2, (&ops2)[1], s);
  418. else if(s[3] == '\0')
  419. return s[0] == '-' ? find_op_2char(opsm2, (&opsm2)[1], s + 1) : OPERAND;
  420. else
  421. return OPERAND;
  422. }
  423. static enum token
  424. t_lex(char *s)
  425. {
  426. int num;
  427. if(s == NULL)
  428. {
  429. return EOI;
  430. }
  431. num = find_op(s);
  432. if(((TOKEN_TYPE(num) == UNOP || TOKEN_TYPE(num) == BUNOP) && isunopoperand()) ||
  433. (num == LPAREN && islparenoperand()) || (num == RPAREN && isrparenoperand()))
  434. return OPERAND;
  435. return num;
  436. }
  437. static int
  438. isunopoperand(void)
  439. {
  440. char *s;
  441. char *t;
  442. int num;
  443. if(nargc == 1) return 1;
  444. s = *(t_wp + 1);
  445. if(nargc == 2) return parenlevel == 1 && strcmp(s, ")") == 0;
  446. t = *(t_wp + 2);
  447. num = find_op(s);
  448. return TOKEN_TYPE(num) == BINOP && (parenlevel == 0 || t[0] != ')' || t[1] != '\0');
  449. }
  450. static int
  451. islparenoperand(void)
  452. {
  453. char *s;
  454. int num;
  455. if(nargc == 1) return 1;
  456. s = *(t_wp + 1);
  457. if(nargc == 2) return parenlevel == 1 && strcmp(s, ")") == 0;
  458. if(nargc != 3) return 0;
  459. num = find_op(s);
  460. return TOKEN_TYPE(num) == BINOP;
  461. }
  462. static int
  463. isrparenoperand(void)
  464. {
  465. char *s;
  466. if(nargc == 1) return 0;
  467. s = *(t_wp + 1);
  468. if(nargc == 2) return parenlevel == 1 && strcmp(s, ")") == 0;
  469. return 0;
  470. }
  471. /* atoi with error detection */
  472. static int
  473. getn(const char *s)
  474. {
  475. char *p;
  476. long r;
  477. errno = 0;
  478. r = strtol(s, &p, 10);
  479. if(s == p) error("%s: bad number", s);
  480. if(errno != 0) error((errno == EINVAL) ? "%s: bad number" : "%s: out of range", s);
  481. while(isspace((unsigned char)*p))
  482. p++;
  483. if(*p) error("%s: bad number", s);
  484. return (int)r;
  485. }
  486. /* atoi with error detection and 64 bit range */
  487. static intmax_t
  488. getq(const char *s)
  489. {
  490. char *p;
  491. intmax_t r;
  492. errno = 0;
  493. r = strtoimax(s, &p, 10);
  494. if(s == p) error("%s: bad number", s);
  495. if(errno != 0) error((errno == EINVAL) ? "%s: bad number" : "%s: out of range", s);
  496. while(isspace((unsigned char)*p))
  497. p++;
  498. if(*p) error("%s: bad number", s);
  499. return r;
  500. }
  501. static int
  502. intcmp(const char *s1, const char *s2)
  503. {
  504. intmax_t q1, q2;
  505. q1 = getq(s1);
  506. q2 = getq(s2);
  507. if(q1 > q2) return 1;
  508. if(q1 < q2) return -1;
  509. return 0;
  510. }
  511. static int
  512. newerf(const char *f1, const char *f2)
  513. {
  514. struct stat b1, b2;
  515. if(stat(f1, &b1) != 0 || stat(f2, &b2) != 0) return 0;
  516. if(b1.st_mtim.tv_sec > b2.st_mtim.tv_sec) return 1;
  517. if(b1.st_mtim.tv_sec < b2.st_mtim.tv_sec) return 0;
  518. return (b1.st_mtim.tv_nsec > b2.st_mtim.tv_nsec);
  519. }
  520. static int
  521. olderf(const char *f1, const char *f2)
  522. {
  523. return (newerf(f2, f1));
  524. }
  525. static int
  526. equalf(const char *f1, const char *f2)
  527. {
  528. struct stat b1, b2;
  529. return (stat(f1, &b1) == 0 && stat(f2, &b2) == 0 && b1.st_dev == b2.st_dev &&
  530. b1.st_ino == b2.st_ino);
  531. }