logo

utils-std

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

chmod.c (9482B)


  1. // utils-std: Collection of commonly available Unix tools
  2. // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
  3. // SPDX-License-Identifier: MPL-2.0
  4. #define _POSIX_C_SOURCE 200809L
  5. // NetBSD <10 hides fdopendir behind _NETBSD_SOURCE
  6. #if __NetBSD_Version__ < 1000000000
  7. #define _NETBSD_SOURCE
  8. #endif
  9. #include "../config.h" // HAS_*
  10. #include "../lib/getopt_nolong.h"
  11. #include "../lib/mode.h"
  12. #include <dirent.h> // fdopendir, readdir, closedir
  13. #include <errno.h>
  14. #include <fcntl.h> // AT_FDCWD
  15. #include <limits.h> // PATH_MAX
  16. #include <stdbool.h>
  17. #include <stdio.h> // fprintf
  18. #include <string.h> // strerror
  19. #include <sys/stat.h> // chmod, fstatat, S_ISDIR
  20. #include <unistd.h> // getopt
  21. #ifdef HAS_GETOPT_LONG
  22. #include <getopt.h>
  23. #endif
  24. const char *argv0 = "chmod";
  25. bool opt_c = false, opt_v = false;
  26. static int
  27. do_fchmodat(int fd, char *mode_arg, char *name, char *acc_path, bool recursive)
  28. {
  29. struct stat stats;
  30. int err = 0;
  31. if(fstatat(fd, name, &stats, AT_SYMLINK_NOFOLLOW) != 0)
  32. {
  33. fprintf(stderr,
  34. "%s: error: Failed getting status for '%s': %s\n",
  35. argv0,
  36. acc_path,
  37. strerror(errno));
  38. errno = 0;
  39. return 1;
  40. }
  41. const char *errstr = NULL;
  42. mode_t mode = new_mode(mode_arg, stats.st_mode, &errstr);
  43. if(errstr != NULL)
  44. {
  45. fprintf(stderr, "%s: error: Failed parsing mode '%s': %s\n", argv0, mode_arg, errstr);
  46. return 1;
  47. }
  48. char mode_from[11] = "";
  49. symbolize_mode(stats.st_mode, mode_from);
  50. if(mode != stats.st_mode)
  51. {
  52. if(fchmodat(fd, name, mode, 0) != 0)
  53. {
  54. fprintf(stderr,
  55. "%s: error: Failed setting permissions to 0%04o for '%s': %s\n",
  56. argv0,
  57. mode,
  58. acc_path,
  59. strerror(errno));
  60. errno = 0;
  61. return 1;
  62. }
  63. if(opt_c || opt_v)
  64. {
  65. char mode_to[11] = "";
  66. symbolize_mode(mode, mode_to);
  67. printf("%s: Permissions changed from 0%04o/%s to 0%04o/%s for '%s'\n",
  68. argv0,
  69. stats.st_mode & 07777,
  70. mode_from,
  71. mode & 07777,
  72. mode_to,
  73. acc_path);
  74. }
  75. }
  76. else if(opt_v)
  77. printf("%s: Permissions already set to 0%04o/%s for '%s'\n",
  78. argv0,
  79. stats.st_mode & 07777,
  80. mode_from,
  81. acc_path);
  82. if(recursive && S_ISDIR(stats.st_mode))
  83. {
  84. int dir = openat(fd, name, O_RDONLY | O_DIRECTORY | O_CLOEXEC);
  85. if(dir == -1)
  86. {
  87. fprintf(stderr,
  88. "%s: error: Couldn't open '%s' as directory: %s\n",
  89. argv0,
  90. acc_path,
  91. strerror(errno));
  92. errno = 0;
  93. return 1;
  94. }
  95. DIR *dirp = fdopendir(dir);
  96. if(dirp == NULL)
  97. {
  98. fprintf(stderr,
  99. "%s: error: Couldn't get DIR entry for opened '%s': %s\n",
  100. argv0,
  101. acc_path,
  102. strerror(errno));
  103. errno = 0;
  104. return 1;
  105. }
  106. while(true)
  107. {
  108. struct dirent *dp = readdir(dirp);
  109. if(dp == NULL)
  110. {
  111. if(errno == 0) break;
  112. fprintf(stderr,
  113. "%s: error: Failed reading directory '%s': %s\n",
  114. argv0,
  115. acc_path,
  116. strerror(errno));
  117. closedir(dirp); // FIXME: unhandled error
  118. errno = 0;
  119. return 1;
  120. }
  121. if(strcmp(dp->d_name, ".") == 0) continue;
  122. if(strcmp(dp->d_name, "..") == 0) continue;
  123. char new_path[PATH_MAX] = "";
  124. if(snprintf(new_path, PATH_MAX, "%s/%s", acc_path, dp->d_name) < 0)
  125. {
  126. fprintf(
  127. stderr,
  128. "%s: error: Couldn't concatenate '%s' into parent '%s', skipping to next entry: %s\n",
  129. argv0,
  130. name,
  131. acc_path,
  132. strerror(errno));
  133. err++;
  134. errno = 0;
  135. continue;
  136. }
  137. // No depth counter for now, unlikely to be a problem
  138. int ret = do_fchmodat(dir, mode_arg, dp->d_name, new_path, true);
  139. if(ret != 0) return ret;
  140. }
  141. // fdopendir allocates memory for DIR, needs closedir
  142. if(closedir(dirp) != 0)
  143. {
  144. fprintf(stderr,
  145. "%s: error: Deallocating directory entry for '%s' failed: %s\n",
  146. argv0,
  147. acc_path,
  148. strerror(errno));
  149. errno = 0;
  150. return 1;
  151. }
  152. }
  153. return err;
  154. }
  155. static int
  156. copy_mode(int fd, mode_t mode, char *name, char *acc_path, bool recursive)
  157. {
  158. struct stat stats;
  159. int err = 0;
  160. if(fstatat(fd, name, &stats, AT_SYMLINK_NOFOLLOW) != 0)
  161. {
  162. fprintf(stderr,
  163. "%s: error: Failed getting status for '%s': %s\n",
  164. argv0,
  165. acc_path,
  166. strerror(errno));
  167. errno = 0;
  168. return 1;
  169. }
  170. char mode_from[11] = "";
  171. symbolize_mode(stats.st_mode, mode_from);
  172. if(mode != stats.st_mode)
  173. {
  174. if(fchmodat(fd, name, mode, 0) != 0)
  175. {
  176. fprintf(stderr,
  177. "%s: error: Failed setting permissions to 0%04o for '%s': %s\n",
  178. argv0,
  179. mode,
  180. acc_path,
  181. strerror(errno));
  182. errno = 0;
  183. return 1;
  184. }
  185. if(opt_c || opt_v)
  186. {
  187. char mode_to[11] = "";
  188. symbolize_mode(mode, mode_to);
  189. printf("%s: Permissions changed from 0%04o/%s to 0%04o/%s for '%s'\n",
  190. argv0,
  191. stats.st_mode & 07777,
  192. mode_from,
  193. mode & 07777,
  194. mode_to,
  195. acc_path);
  196. }
  197. }
  198. else if(opt_v)
  199. printf("%s: Permissions already set to 0%04o/%s for '%s'\n",
  200. argv0,
  201. stats.st_mode & 07777,
  202. mode_from,
  203. acc_path);
  204. if(recursive && S_ISDIR(stats.st_mode))
  205. {
  206. int dir = openat(fd, name, O_RDONLY | O_DIRECTORY | O_CLOEXEC);
  207. if(dir == -1)
  208. {
  209. fprintf(stderr,
  210. "%s: error: Couldn't open '%s' as directory: %s\n",
  211. argv0,
  212. acc_path,
  213. strerror(errno));
  214. errno = 0;
  215. return 1;
  216. }
  217. DIR *dirp = fdopendir(dir);
  218. if(dirp == NULL)
  219. {
  220. fprintf(stderr,
  221. "%s: error: Couldn't get DIR entry for opened '%s': %s\n",
  222. argv0,
  223. acc_path,
  224. strerror(errno));
  225. errno = 0;
  226. return 1;
  227. }
  228. while(true)
  229. {
  230. struct dirent *dp = readdir(dirp);
  231. if(dp == NULL)
  232. {
  233. if(errno == 0) break;
  234. fprintf(stderr,
  235. "%s: error: Failed reading directory '%s': %s\n",
  236. argv0,
  237. acc_path,
  238. strerror(errno));
  239. closedir(dirp); // FIXME: unhandled error
  240. errno = 0;
  241. return 1;
  242. }
  243. if(strcmp(dp->d_name, ".") == 0) continue;
  244. if(strcmp(dp->d_name, "..") == 0) continue;
  245. char new_path[PATH_MAX] = "";
  246. if(snprintf(new_path, PATH_MAX, "%s/%s", acc_path, dp->d_name) < 0)
  247. {
  248. fprintf(
  249. stderr,
  250. "%s: error: Couldn't concatenate '%s' into parent '%s', skipping to next entry: %s\n",
  251. argv0,
  252. name,
  253. acc_path,
  254. strerror(errno));
  255. err++;
  256. errno = 0;
  257. continue;
  258. }
  259. // No depth counter for now, unlikely to be a problem
  260. int ret = copy_mode(dir, mode, dp->d_name, new_path, true);
  261. if(ret != 0) return ret;
  262. }
  263. // fdopendir allocates memory for DIR, needs closedir
  264. if(closedir(dirp) != 0)
  265. {
  266. fprintf(stderr,
  267. "%s: error: Deallocating directory entry for '%s' failed: %s\n",
  268. argv0,
  269. acc_path,
  270. strerror(errno));
  271. errno = 0;
  272. return 1;
  273. }
  274. }
  275. return err;
  276. }
  277. static void
  278. usage(void)
  279. {
  280. fprintf(stderr, "\
  281. Usage: chmod [-cRv] <mode> <file...>\n\
  282. chmod [-cRv] -F <ref_file> <file...>\n");
  283. }
  284. int
  285. main(int argc, char *argv[])
  286. {
  287. bool opt_R = false;
  288. char *ref_file = NULL;
  289. #ifdef HAS_GETOPT_LONG
  290. // clang-format off
  291. enum long_opt_vals {
  292. OPT_REFERENCE = 1,
  293. };
  294. static struct option opts[] = {
  295. {"changes", no_argument, 0, 'c'},
  296. {"recursive", no_argument, 0, 'R'},
  297. {"reference", required_argument, 0, OPT_REFERENCE},
  298. {"verbose", no_argument, 0, 'v'},
  299. {0, 0, 0, 0},
  300. };
  301. // clang-format on
  302. #endif
  303. for(int c = -1;;)
  304. {
  305. if(optind >= argc || !argv[optind]) break;
  306. if(argv[optind][0] == '-' && strchr("rwxugoa", argv[optind][1]))
  307. {
  308. fprintf(stderr,
  309. "%s: warning: (portability) Pass -- before a mode with a leading dash(-) to "
  310. "separate it from options\n",
  311. argv0);
  312. break;
  313. }
  314. #ifdef HAS_GETOPT_LONG
  315. // Need + as first character to get POSIX-style option parsing
  316. c = getopt_long(argc, argv, "+:cRv", opts, NULL);
  317. #else
  318. c = getopt_nolong(argc, argv, ":cRv");
  319. #endif
  320. if(c == -1) break;
  321. switch(c)
  322. {
  323. case 'c': // GNU
  324. opt_c = true;
  325. break;
  326. case OPT_REFERENCE: // GNU & NetBSD
  327. ref_file = optarg;
  328. break;
  329. case 'R': // POSIX
  330. opt_R = true;
  331. break;
  332. case 'v': // GNU
  333. opt_v = true;
  334. break;
  335. case ':':
  336. fprintf(stderr, "%s: error: Missing operand for option: '-%c'\n", argv0, optopt);
  337. usage();
  338. return 1;
  339. case '?': // GNU
  340. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  341. usage();
  342. return 1;
  343. }
  344. }
  345. argc -= optind;
  346. argv += optind;
  347. if(ref_file != NULL)
  348. {
  349. if(argc < 1)
  350. {
  351. fprintf(stderr, "%s: error: Expected >=1 arguments, %d given\n", argv0, argc);
  352. usage();
  353. return 1;
  354. }
  355. struct stat ref_stat;
  356. if(stat(ref_file, &ref_stat) != 0)
  357. {
  358. fprintf(stderr,
  359. "%s: error: Failed to get status from reference file '%s': %s\n",
  360. argv0,
  361. ref_file,
  362. strerror(errno));
  363. return 1;
  364. }
  365. for(int i = 0; i < argc; i++)
  366. {
  367. int ret = copy_mode(AT_FDCWD, ref_stat.st_mode, argv[i], argv[i], opt_R);
  368. if(ret != 0) return ret;
  369. }
  370. return 0;
  371. }
  372. if(argc < 2)
  373. {
  374. fprintf(stderr, "%s: error: Expected >=2 arguments, %d given\n", argv0, argc);
  375. usage();
  376. return 1;
  377. }
  378. for(int i = 1; i < argc; i++)
  379. {
  380. int ret = do_fchmodat(AT_FDCWD, argv[0], argv[i], argv[i], opt_R);
  381. if(ret != 0) return ret;
  382. }
  383. return 0;
  384. }