getconf.c revision 1.38 1 1.38 rillig /* $NetBSD: getconf.c,v 1.38 2024/07/22 21:03:17 rillig Exp $ */
2 1.4 tls
3 1.3 jtc /*-
4 1.8 kleink * Copyright (c) 1996, 1998 The NetBSD Foundation, Inc.
5 1.1 jtc * All rights reserved.
6 1.1 jtc *
7 1.3 jtc * This code is derived from software contributed to The NetBSD Foundation
8 1.3 jtc * by J.T. Conklin.
9 1.3 jtc *
10 1.1 jtc * Redistribution and use in source and binary forms, with or without
11 1.1 jtc * modification, are permitted provided that the following conditions
12 1.1 jtc * are met:
13 1.1 jtc * 1. Redistributions of source code must retain the above copyright
14 1.1 jtc * notice, this list of conditions and the following disclaimer.
15 1.1 jtc * 2. Redistributions in binary form must reproduce the above copyright
16 1.1 jtc * notice, this list of conditions and the following disclaimer in the
17 1.1 jtc * documentation and/or other materials provided with the distribution.
18 1.1 jtc *
19 1.3 jtc * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 1.3 jtc * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 1.3 jtc * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 1.5 jtc * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 1.5 jtc * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 1.3 jtc * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 1.3 jtc * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 1.3 jtc * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 1.3 jtc * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 1.3 jtc * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 1.3 jtc * POSSIBILITY OF SUCH DAMAGE.
30 1.1 jtc */
31 1.1 jtc
32 1.6 lukem #include <sys/cdefs.h>
33 1.38 rillig __RCSID("$NetBSD: getconf.c,v 1.38 2024/07/22 21:03:17 rillig Exp $");
34 1.1 jtc
35 1.6 lukem #include <err.h>
36 1.6 lukem #include <errno.h>
37 1.6 lukem #include <limits.h>
38 1.6 lukem #include <locale.h>
39 1.1 jtc #include <stdio.h>
40 1.1 jtc #include <stdlib.h>
41 1.1 jtc #include <unistd.h>
42 1.13 matt #include <string.h>
43 1.1 jtc
44 1.1 jtc struct conf_variable
45 1.1 jtc {
46 1.1 jtc const char *name;
47 1.1 jtc enum { SYSCONF, CONFSTR, PATHCONF, CONSTANT } type;
48 1.1 jtc long value;
49 1.1 jtc };
50 1.1 jtc
51 1.23 elad static void print_longvar(const char *, long);
52 1.23 elad static void print_strvar(const char *, const char *);
53 1.23 elad static void printvar(const struct conf_variable *, const char *);
54 1.30 perry static void usage(void) __dead;
55 1.23 elad
56 1.23 elad static const struct conf_variable conf_table[] =
57 1.1 jtc {
58 1.1 jtc { "PATH", CONFSTR, _CS_PATH },
59 1.1 jtc
60 1.1 jtc /* Utility Limit Minimum Values */
61 1.1 jtc { "POSIX2_BC_BASE_MAX", CONSTANT, _POSIX2_BC_BASE_MAX },
62 1.1 jtc { "POSIX2_BC_DIM_MAX", CONSTANT, _POSIX2_BC_DIM_MAX },
63 1.1 jtc { "POSIX2_BC_SCALE_MAX", CONSTANT, _POSIX2_BC_SCALE_MAX },
64 1.1 jtc { "POSIX2_BC_STRING_MAX", CONSTANT, _POSIX2_BC_STRING_MAX },
65 1.1 jtc { "POSIX2_COLL_WEIGHTS_MAX", CONSTANT, _POSIX2_COLL_WEIGHTS_MAX },
66 1.1 jtc { "POSIX2_EXPR_NEST_MAX", CONSTANT, _POSIX2_EXPR_NEST_MAX },
67 1.1 jtc { "POSIX2_LINE_MAX", CONSTANT, _POSIX2_LINE_MAX },
68 1.1 jtc { "POSIX2_RE_DUP_MAX", CONSTANT, _POSIX2_RE_DUP_MAX },
69 1.1 jtc { "POSIX2_VERSION", CONSTANT, _POSIX2_VERSION },
70 1.1 jtc
71 1.1 jtc /* POSIX.1 Minimum Values */
72 1.27 rmind { "_POSIX_AIO_LISTIO_MAX", CONSTANT, _POSIX_AIO_LISTIO_MAX },
73 1.27 rmind { "_POSIX_AIO_MAX", CONSTANT, _POSIX_AIO_MAX },
74 1.1 jtc { "_POSIX_ARG_MAX", CONSTANT, _POSIX_ARG_MAX },
75 1.1 jtc { "_POSIX_CHILD_MAX", CONSTANT, _POSIX_CHILD_MAX },
76 1.16 soren { "_POSIX_LINK_MAX", CONSTANT, _POSIX_LINK_MAX },
77 1.1 jtc { "_POSIX_MAX_CANON", CONSTANT, _POSIX_MAX_CANON },
78 1.1 jtc { "_POSIX_MAX_INPUT", CONSTANT, _POSIX_MAX_INPUT },
79 1.28 rmind { "_POSIX_MQ_OPEN_MAX", CONSTANT, _POSIX_MQ_OPEN_MAX },
80 1.28 rmind { "_POSIX_MQ_PRIO_MAX", CONSTANT, _POSIX_MQ_PRIO_MAX },
81 1.16 soren { "_POSIX_NAME_MAX", CONSTANT, _POSIX_NAME_MAX },
82 1.1 jtc { "_POSIX_NGROUPS_MAX", CONSTANT, _POSIX_NGROUPS_MAX },
83 1.1 jtc { "_POSIX_OPEN_MAX", CONSTANT, _POSIX_OPEN_MAX },
84 1.18 tron { "_POSIX_PATH_MAX", CONSTANT, _POSIX_PATH_MAX },
85 1.1 jtc { "_POSIX_PIPE_BUF", CONSTANT, _POSIX_PIPE_BUF },
86 1.1 jtc { "_POSIX_SSIZE_MAX", CONSTANT, _POSIX_SSIZE_MAX },
87 1.1 jtc { "_POSIX_STREAM_MAX", CONSTANT, _POSIX_STREAM_MAX },
88 1.1 jtc { "_POSIX_TZNAME_MAX", CONSTANT, _POSIX_TZNAME_MAX },
89 1.1 jtc
90 1.1 jtc /* Symbolic Utility Limits */
91 1.1 jtc { "BC_BASE_MAX", SYSCONF, _SC_BC_BASE_MAX },
92 1.1 jtc { "BC_DIM_MAX", SYSCONF, _SC_BC_DIM_MAX },
93 1.1 jtc { "BC_SCALE_MAX", SYSCONF, _SC_BC_SCALE_MAX },
94 1.1 jtc { "BC_STRING_MAX", SYSCONF, _SC_BC_STRING_MAX },
95 1.1 jtc { "COLL_WEIGHTS_MAX", SYSCONF, _SC_COLL_WEIGHTS_MAX },
96 1.1 jtc { "EXPR_NEST_MAX", SYSCONF, _SC_EXPR_NEST_MAX },
97 1.1 jtc { "LINE_MAX", SYSCONF, _SC_LINE_MAX },
98 1.1 jtc { "RE_DUP_MAX", SYSCONF, _SC_RE_DUP_MAX },
99 1.1 jtc
100 1.1 jtc /* Optional Facility Configuration Values */
101 1.34 christos { "_POSIX2_C_BIND", SYSCONF, _SC_2_C_BIND },
102 1.1 jtc { "POSIX2_C_DEV", SYSCONF, _SC_2_C_DEV },
103 1.1 jtc { "POSIX2_CHAR_TERM", SYSCONF, _SC_2_CHAR_TERM },
104 1.1 jtc { "POSIX2_FORT_DEV", SYSCONF, _SC_2_FORT_DEV },
105 1.1 jtc { "POSIX2_FORT_RUN", SYSCONF, _SC_2_FORT_RUN },
106 1.1 jtc { "POSIX2_LOCALEDEF", SYSCONF, _SC_2_LOCALEDEF },
107 1.1 jtc { "POSIX2_SW_DEV", SYSCONF, _SC_2_SW_DEV },
108 1.1 jtc { "POSIX2_UPE", SYSCONF, _SC_2_UPE },
109 1.1 jtc
110 1.1 jtc /* POSIX.1 Configurable System Variables */
111 1.27 rmind { "AIO_LISTIO_MAX", SYSCONF, _SC_AIO_LISTIO_MAX },
112 1.28 rmind { "AIO_MAX", SYSCONF, _SC_AIO_MAX },
113 1.1 jtc { "ARG_MAX", SYSCONF, _SC_ARG_MAX },
114 1.1 jtc { "CHILD_MAX", SYSCONF, _SC_CHILD_MAX },
115 1.1 jtc { "CLK_TCK", SYSCONF, _SC_CLK_TCK },
116 1.28 rmind { "MQ_OPEN_MAX", SYSCONF, _SC_MQ_OPEN_MAX },
117 1.28 rmind { "MQ_PRIO_MAX", SYSCONF, _SC_MQ_PRIO_MAX },
118 1.1 jtc { "NGROUPS_MAX", SYSCONF, _SC_NGROUPS_MAX },
119 1.1 jtc { "OPEN_MAX", SYSCONF, _SC_OPEN_MAX },
120 1.1 jtc { "STREAM_MAX", SYSCONF, _SC_STREAM_MAX },
121 1.1 jtc { "TZNAME_MAX", SYSCONF, _SC_TZNAME_MAX },
122 1.1 jtc { "_POSIX_JOB_CONTROL", SYSCONF, _SC_JOB_CONTROL },
123 1.1 jtc { "_POSIX_SAVED_IDS", SYSCONF, _SC_SAVED_IDS },
124 1.1 jtc { "_POSIX_VERSION", SYSCONF, _SC_VERSION },
125 1.1 jtc
126 1.1 jtc { "LINK_MAX", PATHCONF, _PC_LINK_MAX },
127 1.1 jtc { "MAX_CANON", PATHCONF, _PC_MAX_CANON },
128 1.1 jtc { "MAX_INPUT", PATHCONF, _PC_MAX_INPUT },
129 1.1 jtc { "NAME_MAX", PATHCONF, _PC_NAME_MAX },
130 1.1 jtc { "PATH_MAX", PATHCONF, _PC_PATH_MAX },
131 1.1 jtc { "PIPE_BUF", PATHCONF, _PC_PIPE_BUF },
132 1.1 jtc { "_POSIX_CHOWN_RESTRICTED", PATHCONF, _PC_CHOWN_RESTRICTED },
133 1.1 jtc { "_POSIX_NO_TRUNC", PATHCONF, _PC_NO_TRUNC },
134 1.1 jtc { "_POSIX_VDISABLE", PATHCONF, _PC_VDISABLE },
135 1.1 jtc
136 1.8 kleink /* POSIX.1b Configurable System Variables */
137 1.8 kleink { "PAGESIZE", SYSCONF, _SC_PAGESIZE },
138 1.27 rmind { "_POSIX_ASYNCHRONOUS_IO", SYSCONF, _SC_ASYNCHRONOUS_IO },
139 1.7 kleink { "_POSIX_FSYNC", SYSCONF, _SC_FSYNC },
140 1.10 kleink { "_POSIX_MAPPED_FILES", SYSCONF, _SC_MAPPED_FILES },
141 1.10 kleink { "_POSIX_MEMLOCK", SYSCONF, _SC_MEMLOCK },
142 1.10 kleink { "_POSIX_MEMLOCK_RANGE", SYSCONF, _SC_MEMLOCK_RANGE },
143 1.10 kleink { "_POSIX_MEMORY_PROTECTION", SYSCONF, _SC_MEMORY_PROTECTION },
144 1.32 njoly { "_POSIX_MESSAGE_PASSING", SYSCONF, _SC_MESSAGE_PASSING },
145 1.14 kleink { "_POSIX_MONOTONIC_CLOCK", SYSCONF, _SC_MONOTONIC_CLOCK },
146 1.31 rmind { "_POSIX_PRIORITY_SCHEDULING", SYSCONF, _SC_PRIORITY_SCHEDULING },
147 1.20 kleink { "_POSIX_SEMAPHORES", SYSCONF, _SC_SEMAPHORES },
148 1.35 rmind { "_POSIX_SHARED_MEMORY_OBJECTS", SYSCONF, _SC_SHARED_MEMORY_OBJECTS },
149 1.8 kleink { "_POSIX_SYNCHRONIZED_IO", SYSCONF, _SC_SYNCHRONIZED_IO },
150 1.20 kleink { "_POSIX_TIMERS", SYSCONF, _SC_TIMERS },
151 1.8 kleink
152 1.8 kleink { "_POSIX_SYNC_IO", PATHCONF, _PC_SYNC_IO },
153 1.11 kleink
154 1.11 kleink /* POSIX.1c Configurable System Variables */
155 1.11 kleink { "LOGIN_NAME_MAX", SYSCONF, _SC_LOGIN_NAME_MAX },
156 1.20 kleink { "_POSIX_THREADS", SYSCONF, _SC_THREADS },
157 1.20 kleink
158 1.20 kleink /* POSIX.1j Configurable System Variables */
159 1.20 kleink { "_POSIX_BARRIERS", SYSCONF, _SC_BARRIERS },
160 1.20 kleink { "_POSIX_READER_WRITER_LOCKS", SYSCONF, _SC_READER_WRITER_LOCKS },
161 1.20 kleink { "_POSIX_SPIN_LOCKS", SYSCONF, _SC_SPIN_LOCKS },
162 1.7 kleink
163 1.7 kleink /* XPG4.2 Configurable System Variables */
164 1.8 kleink { "IOV_MAX", SYSCONF, _SC_IOV_MAX },
165 1.8 kleink { "PAGE_SIZE", SYSCONF, _SC_PAGE_SIZE },
166 1.7 kleink { "_XOPEN_SHM", SYSCONF, _SC_XOPEN_SHM },
167 1.9 kleink
168 1.9 kleink /* X/Open CAE Spec. Issue 5 Version 2 Configurable System Variables */
169 1.9 kleink { "FILESIZEBITS", PATHCONF, _PC_FILESIZEBITS },
170 1.19 kleink
171 1.19 kleink /* POSIX.1-2001 XSI Option Group Configurable System Variables */
172 1.19 kleink { "ATEXIT_MAX", SYSCONF, _SC_ATEXIT_MAX },
173 1.22 lukem
174 1.22 lukem /* POSIX.1-2001 TSF Configurable System Variables */
175 1.22 lukem { "GETGR_R_SIZE_MAX", SYSCONF, _SC_GETGR_R_SIZE_MAX },
176 1.22 lukem { "GETPW_R_SIZE_MAX", SYSCONF, _SC_GETPW_R_SIZE_MAX },
177 1.22 lukem
178 1.36 simonb /* Extensions found in Solaris and Linux. */
179 1.36 simonb { "SC_PHYS_PAGES", SYSCONF, _SC_PHYS_PAGES },
180 1.36 simonb { "SC_AVPHYS_PAGES", SYSCONF, _SC_AVPHYS_PAGES },
181 1.36 simonb
182 1.29 ad #ifdef _NETBSD_SOURCE
183 1.29 ad /* Commonly provided extensions */
184 1.29 ad { "NPROCESSORS_CONF", SYSCONF, _SC_NPROCESSORS_CONF },
185 1.29 ad { "NPROCESSORS_ONLN", SYSCONF, _SC_NPROCESSORS_ONLN },
186 1.29 ad #endif /* _NETBSD_SOURCE */
187 1.29 ad
188 1.23 elad { NULL, CONSTANT, 0L }
189 1.1 jtc };
190 1.1 jtc
191 1.23 elad static int a_flag = 0; /* list all variables */
192 1.1 jtc
193 1.1 jtc int
194 1.23 elad main(int argc, char **argv)
195 1.1 jtc {
196 1.1 jtc int ch;
197 1.1 jtc const struct conf_variable *cp;
198 1.37 christos const char *varname, *pathname, *vn;
199 1.23 elad int found;
200 1.1 jtc
201 1.24 christos setprogname(argv[0]);
202 1.24 christos (void)setlocale(LC_ALL, "");
203 1.1 jtc
204 1.23 elad while ((ch = getopt(argc, argv, "a")) != -1) {
205 1.1 jtc switch (ch) {
206 1.23 elad case 'a':
207 1.23 elad a_flag = 1;
208 1.23 elad break;
209 1.1 jtc case '?':
210 1.1 jtc default:
211 1.1 jtc usage();
212 1.1 jtc }
213 1.1 jtc }
214 1.1 jtc argc -= optind;
215 1.1 jtc argv += optind;
216 1.1 jtc
217 1.24 christos if (!a_flag) {
218 1.24 christos if (argc == 0)
219 1.24 christos usage();
220 1.23 elad varname = argv[0];
221 1.23 elad argc--;
222 1.23 elad argv++;
223 1.24 christos } else
224 1.24 christos varname = NULL;
225 1.24 christos
226 1.24 christos if (argc > 1)
227 1.1 jtc usage();
228 1.23 elad pathname = argv[0]; /* may be NULL */
229 1.1 jtc
230 1.23 elad found = 0;
231 1.37 christos vn = varname;
232 1.37 christos again:
233 1.1 jtc for (cp = conf_table; cp->name != NULL; cp++) {
234 1.37 christos if (a_flag || strcmp(vn, cp->name) == 0) {
235 1.23 elad if ((cp->type == PATHCONF) == (pathname != NULL)) {
236 1.23 elad printvar(cp, pathname);
237 1.23 elad found = 1;
238 1.24 christos } else if (!a_flag)
239 1.24 christos errx(EXIT_FAILURE,
240 1.24 christos "%s: invalid variable type", cp->name);
241 1.23 elad }
242 1.1 jtc }
243 1.23 elad
244 1.37 christos if (!a_flag && !found) {
245 1.37 christos if (*vn++ == '_')
246 1.37 christos goto again;
247 1.24 christos errx(EXIT_FAILURE, "%s: unknown variable", varname);
248 1.37 christos }
249 1.1 jtc
250 1.23 elad (void)fflush(stdout);
251 1.23 elad return ferror(stdout) ? EXIT_FAILURE : EXIT_SUCCESS;
252 1.23 elad }
253 1.23 elad
254 1.23 elad static void
255 1.23 elad print_longvar(const char *name, long value)
256 1.23 elad {
257 1.23 elad if (a_flag)
258 1.23 elad (void)printf("%s = %ld\n", name, value);
259 1.23 elad else
260 1.24 christos (void)printf("%ld\n", value);
261 1.23 elad }
262 1.23 elad
263 1.23 elad static void
264 1.23 elad print_strvar(const char *name, const char *sval)
265 1.23 elad {
266 1.23 elad if (a_flag)
267 1.23 elad (void)printf("%s = %s\n", name, sval);
268 1.23 elad else
269 1.23 elad (void)printf("%s\n", sval);
270 1.23 elad }
271 1.23 elad
272 1.23 elad static void
273 1.23 elad printvar(const struct conf_variable *cp, const char *pathname)
274 1.23 elad {
275 1.23 elad size_t slen;
276 1.23 elad char *sval;
277 1.23 elad long val;
278 1.1 jtc
279 1.1 jtc switch (cp->type) {
280 1.1 jtc case CONSTANT:
281 1.23 elad print_longvar(cp->name, cp->value);
282 1.1 jtc break;
283 1.1 jtc
284 1.1 jtc case CONFSTR:
285 1.26 mjf errno = 0;
286 1.24 christos slen = confstr((int)cp->value, NULL, 0);
287 1.26 mjf if (slen == 0) {
288 1.26 mjf if (errno != 0)
289 1.26 mjf
290 1.26 mjf out: err(EXIT_FAILURE, "confstr(%ld)", cp->value);
291 1.26 mjf else
292 1.26 mjf print_strvar(cp->name, "undefined");
293 1.26 mjf }
294 1.1 jtc
295 1.1 jtc if ((sval = malloc(slen)) == NULL)
296 1.24 christos err(EXIT_FAILURE, "Can't allocate %zu bytes", slen);
297 1.1 jtc
298 1.26 mjf errno = 0;
299 1.26 mjf if (confstr((int)cp->value, sval, slen) == 0) {
300 1.26 mjf if (errno != 0)
301 1.26 mjf goto out;
302 1.26 mjf else
303 1.26 mjf print_strvar(cp->name, "undefined");
304 1.26 mjf } else
305 1.26 mjf print_strvar(cp->name, sval);
306 1.26 mjf
307 1.24 christos free(sval);
308 1.1 jtc break;
309 1.1 jtc
310 1.1 jtc case SYSCONF:
311 1.1 jtc errno = 0;
312 1.24 christos if ((val = sysconf((int)cp->value)) == -1) {
313 1.24 christos if (errno != 0)
314 1.24 christos err(EXIT_FAILURE, "sysconf(%ld)", cp->value);
315 1.23 elad print_strvar(cp->name, "undefined");
316 1.24 christos } else
317 1.23 elad print_longvar(cp->name, val);
318 1.1 jtc break;
319 1.1 jtc
320 1.1 jtc case PATHCONF:
321 1.1 jtc errno = 0;
322 1.24 christos if ((val = pathconf(pathname, (int)cp->value)) == -1) {
323 1.1 jtc if (errno != 0) {
324 1.23 elad if (a_flag && errno == EINVAL) {
325 1.23 elad /* Just skip invalid variables */
326 1.23 elad return;
327 1.23 elad }
328 1.24 christos err(EXIT_FAILURE, "pathconf(%s, %ld)",
329 1.24 christos pathname, cp->value);
330 1.1 jtc /* NOTREACHED */
331 1.1 jtc }
332 1.1 jtc
333 1.23 elad print_strvar(cp->name, "undefined");
334 1.24 christos } else
335 1.23 elad print_longvar(cp->name, val);
336 1.1 jtc break;
337 1.1 jtc }
338 1.1 jtc }
339 1.1 jtc
340 1.1 jtc
341 1.1 jtc static void
342 1.23 elad usage(void)
343 1.1 jtc {
344 1.24 christos const char *p = getprogname();
345 1.24 christos (void)fprintf(stderr, "Usage: %s system_var\n\t%s -a\n"
346 1.24 christos "\t%s path_var pathname\n\t%s -a pathname\n", p, p, p, p);
347 1.23 elad exit(EXIT_FAILURE);
348 1.1 jtc }
349