logo

utils-std

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

getconf.c (7085B)


  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 202405L
  5. #define _XOPEN_SOURCE 800
  6. #include "../lib/getopt_nolong.h"
  7. #include <assert.h>
  8. #include <errno.h>
  9. #include <limits.h>
  10. #include <stdbool.h>
  11. #include <stdio.h> // fputs
  12. #include <stdlib.h> // exit
  13. #include <string.h> // strerror
  14. #include <unistd.h> // getopt, pathconf
  15. // need to be after system headers
  16. #include "./getconf_vars.h"
  17. const char *argv0 = "getconf";
  18. static int
  19. print_system_var(const char *var)
  20. {
  21. for(size_t i = 0; i < (sizeof(confstr_vars) / sizeof(*confstr_vars)); i++)
  22. {
  23. if(strcmp(confstr_vars[i].name, var) != 0) continue;
  24. errno = 0;
  25. size_t buflen = confstr(confstr_vars[i].value, (char *)NULL, 0);
  26. if(buflen == 0)
  27. {
  28. if(errno != 0)
  29. {
  30. fprintf(stderr,
  31. "getconf: error: confstr(%d /* \"%s\" */, NULL, 0): %s\n",
  32. confstr_vars[i].value,
  33. var,
  34. strerror(errno));
  35. return 1;
  36. }
  37. printf("undefined\n");
  38. return 0;
  39. }
  40. errno = 0;
  41. char *buf = malloc(buflen);
  42. if(!buf)
  43. {
  44. fprintf(stderr,
  45. "getconf: error: Failed to allocate %zd bytes buffer for confstr: %s\n",
  46. buflen,
  47. strerror(errno));
  48. return 1;
  49. }
  50. errno = 0;
  51. size_t ret = confstr(confstr_vars[i].value, buf, buflen);
  52. if(ret == 0)
  53. {
  54. if(errno != 0)
  55. {
  56. fprintf(stderr,
  57. "getconf: error: confstr(%d /* \"%s\" */, buf, %zd): %s\n",
  58. confstr_vars[i].value,
  59. var,
  60. buflen,
  61. strerror(errno));
  62. free(buf);
  63. return 1;
  64. }
  65. printf("undefined\n");
  66. free(buf);
  67. return 0;
  68. }
  69. fwrite(buf, buflen, 1, stdout);
  70. free(buf);
  71. return 0;
  72. }
  73. for(size_t i = 0; i < (sizeof(sysconf_vars) / sizeof(*sysconf_vars)); i++)
  74. {
  75. if(strcmp(sysconf_vars[i].name, var) != 0) continue;
  76. errno = 0;
  77. long ret = sysconf((int)(sysconf_vars[i].value));
  78. if(ret == -1 && errno != 0)
  79. {
  80. fprintf(stderr,
  81. "getconf: error: sysconf(%d /* \"%s\" */): %s\n",
  82. sysconf_vars[i].value,
  83. var,
  84. strerror(errno));
  85. return 1;
  86. }
  87. if(ret == -1) ret = sysconf_vars[i].limit_h;
  88. if(ret == -1)
  89. {
  90. printf("undefined\n");
  91. return 0;
  92. }
  93. printf("%ld\n", ret);
  94. return 0;
  95. }
  96. for(size_t i = 0; i < (sizeof(limits_vars) / sizeof(*limits_vars)); i++)
  97. {
  98. if(strcmp(limits_vars[i].name, var) != 0) continue;
  99. long ret = limits_vars[i].limit_h;
  100. if(ret == -1)
  101. {
  102. printf("undefined\n");
  103. return 0;
  104. }
  105. printf("%ld\n", ret);
  106. return 0;
  107. }
  108. fprintf(stderr, "getconf: error: unknown system_var \"%s\"\n", var);
  109. return 1;
  110. }
  111. int
  112. main(int argc, char *argv[])
  113. {
  114. bool o_all = false;
  115. int c = -1;
  116. while((c = getopt_nolong(argc, argv, ":av:")) != -1)
  117. {
  118. switch(c)
  119. {
  120. case 'a':
  121. o_all = true;
  122. break;
  123. case 'v':
  124. fputs("getconf: -v option is unsupported\n", stderr);
  125. return 1;
  126. }
  127. }
  128. argc -= optind;
  129. argv += optind;
  130. if(!o_all && argc == 1)
  131. {
  132. return print_system_var(argv[0]);
  133. }
  134. else if(!o_all && argc == 2)
  135. {
  136. for(size_t i = 0; i < (sizeof(pathconf_vars) / sizeof(*pathconf_vars)); i++)
  137. {
  138. if(strcmp(pathconf_vars[i].name, argv[0]) != 0) continue;
  139. errno = 0;
  140. long ret = pathconf(argv[1], (int)(pathconf_vars[i].value));
  141. if(ret == -1 && errno != 0)
  142. {
  143. fprintf(stderr,
  144. "getconf: error: pathconf(\"%s\", %d /* \"%s\" */): %s\n",
  145. argv[1],
  146. pathconf_vars[i].value,
  147. argv[0],
  148. strerror(errno));
  149. return 1;
  150. }
  151. if(ret == -1) ret = sysconf_vars[i].limit_h;
  152. printf("%ld\n", ret);
  153. return 0;
  154. }
  155. fprintf(stderr, "getconf: error: unknown path_var \"%s\"\n", argv[0]);
  156. return 1;
  157. }
  158. else if(o_all && (argc == 0 || argc == 1))
  159. {
  160. int err = 0;
  161. char *buf = NULL;
  162. size_t buflen = 0;
  163. for(size_t i = 0; i < (sizeof(confstr_vars) / sizeof(*confstr_vars)); i++)
  164. {
  165. errno = 0;
  166. size_t buflen_ret = confstr(confstr_vars[i].value, (char *)NULL, 0);
  167. if(buflen_ret == 0)
  168. {
  169. if(errno != 0)
  170. {
  171. fprintf(stderr,
  172. "getconf: error: confstr(%d /* \"%s\" */, NULL, 0): %s\n",
  173. confstr_vars[i].value,
  174. confstr_vars[i].name,
  175. strerror(errno));
  176. free(buf);
  177. err = 1;
  178. continue;
  179. }
  180. printf("%s: undefined\n", confstr_vars[i].name);
  181. continue;
  182. }
  183. if(buflen_ret > buflen)
  184. {
  185. errno = 0;
  186. char *buf_ret = realloc(buf, buflen_ret);
  187. if(!buf_ret)
  188. {
  189. fprintf(stderr,
  190. "getconf: error: Failed to allocate %zd bytes buffer for confstr: %s\n",
  191. buflen,
  192. strerror(errno));
  193. free(buf);
  194. err = 1;
  195. continue;
  196. }
  197. buflen = buflen_ret;
  198. buf = buf_ret;
  199. }
  200. errno = 0;
  201. size_t ret = confstr(confstr_vars[i].value, buf, buflen);
  202. if(ret == 0)
  203. {
  204. if(errno != 0)
  205. {
  206. fprintf(stderr,
  207. "getconf: error: confstr(%d /* \"%s\" */, buf, %zd): %s\n",
  208. confstr_vars[i].value,
  209. confstr_vars[i].name,
  210. buflen,
  211. strerror(errno));
  212. free(buf);
  213. err = 1;
  214. continue;
  215. }
  216. printf("%s: undefined\n", confstr_vars[i].name);
  217. continue;
  218. }
  219. printf("%s: ", confstr_vars[i].name);
  220. fwrite(buf, buflen_ret, 1, stdout);
  221. printf("\n");
  222. }
  223. free(buf);
  224. for(size_t i = 0; i < (sizeof(sysconf_vars) / sizeof(*sysconf_vars)); i++)
  225. {
  226. errno = 0;
  227. long ret = sysconf((int)(sysconf_vars[i].value));
  228. if(ret == -1 && errno != 0)
  229. {
  230. fprintf(stderr,
  231. "getconf: error: sysconf(%d /* \"%s\" */): %s\n",
  232. confstr_vars[i].value,
  233. confstr_vars[i].name,
  234. strerror(errno));
  235. err = 1;
  236. continue;
  237. }
  238. if(ret == -1) ret = sysconf_vars[i].limit_h;
  239. if(ret == -1)
  240. {
  241. printf("%s: undefined\n", sysconf_vars[i].name);
  242. continue;
  243. }
  244. printf("%s: %ld\n", sysconf_vars[i].name, ret);
  245. }
  246. for(size_t i = 0; i < (sizeof(limits_vars) / sizeof(*limits_vars)); i++)
  247. {
  248. long ret = limits_vars[i].limit_h;
  249. if(ret == -1)
  250. {
  251. printf("%s: undefined\n", limits_vars[i].name);
  252. continue;
  253. }
  254. printf("%s: %ld\n", limits_vars[i].name, ret);
  255. }
  256. const char *path = argc == 1 ? argv[0] : ".";
  257. for(size_t i = 0; i < (sizeof(pathconf_vars) / sizeof(*pathconf_vars)); i++)
  258. {
  259. errno = 0;
  260. long ret = pathconf(path, (int)(pathconf_vars[i].value));
  261. if(ret == -1 && errno != 0)
  262. {
  263. fprintf(stderr,
  264. "getconf: error: pathconf(\"%s\", %d /* \"%s\" */): %s\n",
  265. path,
  266. pathconf_vars[i].value,
  267. pathconf_vars[i].name,
  268. strerror(errno));
  269. err = 1;
  270. continue;
  271. }
  272. if(ret == -1) ret = sysconf_vars[i].limit_h;
  273. printf("%s: %ld\n", pathconf_vars[i].name, ret);
  274. }
  275. return err;
  276. }
  277. else
  278. {
  279. fprintf(stderr, "getconf: error: wrong number of arguments\n");
  280. fprintf(stderr, "\
  281. Usage: getconf <system_var>\n\
  282. getconf <path_var> <path>\n\
  283. getconf -a [path]\n\
  284. ");
  285. return 1;
  286. }
  287. }