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