Home | History | Annotate | Line # | Download | only in getconf
getconf.c revision 1.37
      1 /*	$NetBSD: getconf.c,v 1.37 2024/01/27 16:04:36 christos 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.37 2024/01/27 16:04:36 christos 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, *vn;
    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 	vn = varname;
    234 again:
    235 	for (cp = conf_table; cp->name != NULL; cp++) {
    236 		if (a_flag || strcmp(vn, cp->name) == 0) {
    237 			/*LINTED weird expression*/
    238 			if ((cp->type == PATHCONF) == (pathname != NULL)) {
    239 				printvar(cp, pathname);
    240 				found = 1;
    241 			} else if (!a_flag)
    242 				errx(EXIT_FAILURE,
    243 				    "%s: invalid variable type", cp->name);
    244 		}
    245 	}
    246 
    247 	if (!a_flag && !found) {
    248 		if (*vn++ == '_')
    249 			goto again;
    250 		errx(EXIT_FAILURE, "%s: unknown variable", varname);
    251 	}
    252 
    253 	(void)fflush(stdout);
    254 	return ferror(stdout) ? EXIT_FAILURE : EXIT_SUCCESS;
    255 }
    256 
    257 static void
    258 print_longvar(const char *name, long value)
    259 {
    260 	if (a_flag)
    261 		(void)printf("%s = %ld\n", name, value);
    262 	else
    263 		(void)printf("%ld\n", value);
    264 }
    265 
    266 static void
    267 print_strvar(const char *name, const char *sval)
    268 {
    269 	if (a_flag)
    270 		(void)printf("%s = %s\n", name, sval);
    271 	else
    272 		(void)printf("%s\n", sval);
    273 }
    274 
    275 static void
    276 printvar(const struct conf_variable *cp, const char *pathname)
    277 {
    278 	size_t slen;
    279 	char *sval;
    280 	long val;
    281 
    282 	switch (cp->type) {
    283 	case CONSTANT:
    284 		print_longvar(cp->name, cp->value);
    285 		break;
    286 
    287 	case CONFSTR:
    288 		errno = 0;
    289 		slen = confstr((int)cp->value, NULL, 0);
    290 		if (slen == 0) {
    291 			if (errno != 0)
    292 
    293 out:			 	err(EXIT_FAILURE, "confstr(%ld)", cp->value);
    294 			else
    295 				print_strvar(cp->name, "undefined");
    296 		}
    297 
    298 		if ((sval = malloc(slen)) == NULL)
    299 			err(EXIT_FAILURE, "Can't allocate %zu bytes", slen);
    300 
    301 		errno = 0;
    302 		if (confstr((int)cp->value, sval, slen) == 0) {
    303 			if (errno != 0)
    304 				goto out;
    305 			else
    306 				print_strvar(cp->name, "undefined");
    307 		} else
    308 			print_strvar(cp->name, sval);
    309 
    310 		free(sval);
    311 		break;
    312 
    313 	case SYSCONF:
    314 		errno = 0;
    315 		if ((val = sysconf((int)cp->value)) == -1) {
    316 			if (errno != 0)
    317 				err(EXIT_FAILURE, "sysconf(%ld)", cp->value);
    318 			print_strvar(cp->name, "undefined");
    319 		} else
    320 			print_longvar(cp->name, val);
    321 		break;
    322 
    323 	case PATHCONF:
    324 		errno = 0;
    325 		if ((val = pathconf(pathname, (int)cp->value)) == -1) {
    326 			if (errno != 0) {
    327 				if (a_flag && errno == EINVAL) {
    328 					/* Just skip invalid variables */
    329 					return;
    330 				}
    331 				err(EXIT_FAILURE, "pathconf(%s, %ld)",
    332 				    pathname, cp->value);
    333 				/* NOTREACHED */
    334 			}
    335 
    336 			print_strvar(cp->name, "undefined");
    337 		} else
    338 			print_longvar(cp->name, val);
    339 		break;
    340 	}
    341 }
    342 
    343 
    344 static void
    345 usage(void)
    346 {
    347 	const char *p = getprogname();
    348 	(void)fprintf(stderr, "Usage: %s system_var\n\t%s -a\n"
    349 	    "\t%s path_var pathname\n\t%s -a pathname\n", p, p, p, p);
    350 	exit(EXIT_FAILURE);
    351 }
    352