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