Blob Blame History Raw
/*-
 * Copyright (c) 1996, 1998 The NetBSD Foundation, Inc.
 * All rights reserved.
 *
 * This code is derived from software contributed to The NetBSD Foundation
 * by J.T. Conklin.
 *
 * Mostly rewritten to be used in Alpine Linux (with musl c-library)
 * by Timo Teräs.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

#include <err.h>
#include <errno.h>
#include <values.h>
#include <limits.h>
#include <locale.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

struct conf_variable {
	const char *name;
	enum { SYSCONF, CONFSTR, PATHCONF, CONSTANT, UCONSTANT, NUM_TYPES } type;
	long value;
};

static const struct conf_variable conf_table[] = {
{ "PATH",			CONFSTR,	_CS_PATH		},

/* Utility Limit Minimum Values */
{ "POSIX2_BC_BASE_MAX",		CONSTANT,	_POSIX2_BC_BASE_MAX	},
{ "POSIX2_BC_DIM_MAX",		CONSTANT,	_POSIX2_BC_DIM_MAX	},
{ "POSIX2_BC_SCALE_MAX",	CONSTANT,	_POSIX2_BC_SCALE_MAX	},
{ "POSIX2_BC_STRING_MAX",	CONSTANT,	_POSIX2_BC_STRING_MAX	},
{ "POSIX2_COLL_WEIGHTS_MAX",	CONSTANT,	_POSIX2_COLL_WEIGHTS_MAX },
{ "POSIX2_EXPR_NEST_MAX",	CONSTANT,	_POSIX2_EXPR_NEST_MAX	},
{ "POSIX2_LINE_MAX",		CONSTANT,	_POSIX2_LINE_MAX	},
{ "POSIX2_RE_DUP_MAX",		CONSTANT,	_POSIX2_RE_DUP_MAX	},
{ "POSIX2_VERSION",		CONSTANT,	_POSIX2_VERSION		},

/* POSIX.1 Minimum Values */
{ "_POSIX_AIO_LISTIO_MAX",	CONSTANT,	_POSIX_AIO_LISTIO_MAX	},
{ "_POSIX_AIO_MAX",		CONSTANT,       _POSIX_AIO_MAX		},
{ "_POSIX_ARG_MAX",		CONSTANT,	_POSIX_ARG_MAX		},
{ "_POSIX_CHILD_MAX",		CONSTANT,	_POSIX_CHILD_MAX	},
{ "_POSIX_LINK_MAX",		CONSTANT,	_POSIX_LINK_MAX		},
{ "_POSIX_MAX_CANON",		CONSTANT,	_POSIX_MAX_CANON	},
{ "_POSIX_MAX_INPUT",		CONSTANT,	_POSIX_MAX_INPUT	},
{ "_POSIX_MQ_OPEN_MAX",		CONSTANT,	_POSIX_MQ_OPEN_MAX	},
{ "_POSIX_MQ_PRIO_MAX",		CONSTANT,	_POSIX_MQ_PRIO_MAX	},
{ "_POSIX_NAME_MAX",		CONSTANT,	_POSIX_NAME_MAX		},
{ "_POSIX_NGROUPS_MAX",		CONSTANT,	_POSIX_NGROUPS_MAX	},
{ "_POSIX_OPEN_MAX",		CONSTANT,	_POSIX_OPEN_MAX		},
{ "_POSIX_PATH_MAX",		CONSTANT,	_POSIX_PATH_MAX		},
{ "_POSIX_PIPE_BUF",		CONSTANT,	_POSIX_PIPE_BUF		},
{ "_POSIX_SSIZE_MAX",		CONSTANT,	_POSIX_SSIZE_MAX	},
{ "_POSIX_STREAM_MAX",		CONSTANT,	_POSIX_STREAM_MAX	},
{ "_POSIX_TZNAME_MAX",		CONSTANT,	_POSIX_TZNAME_MAX	},

/* Symbolic Utility Limits */
{ "BC_BASE_MAX",		SYSCONF,	_SC_BC_BASE_MAX		},
{ "BC_DIM_MAX",			SYSCONF,	_SC_BC_DIM_MAX		},
{ "BC_SCALE_MAX",		SYSCONF,	_SC_BC_SCALE_MAX	},
{ "BC_STRING_MAX",		SYSCONF,	_SC_BC_STRING_MAX	},
{ "COLL_WEIGHTS_MAX",		SYSCONF,	_SC_COLL_WEIGHTS_MAX	},
{ "EXPR_NEST_MAX",		SYSCONF,	_SC_EXPR_NEST_MAX	},
{ "LINE_MAX",			SYSCONF,	_SC_LINE_MAX		},
{ "RE_DUP_MAX",			SYSCONF,	_SC_RE_DUP_MAX		},

/* Optional Facility Configuration Values */
{ "_POSIX2_C_BIND",		SYSCONF,	_SC_2_C_BIND		},
{ "POSIX2_C_DEV",		SYSCONF,	_SC_2_C_DEV		},
{ "POSIX2_CHAR_TERM",		SYSCONF,	_SC_2_CHAR_TERM		},
{ "POSIX2_FORT_DEV",		SYSCONF,	_SC_2_FORT_DEV		},
{ "POSIX2_FORT_RUN",		SYSCONF,	_SC_2_FORT_RUN		},
{ "POSIX2_LOCALEDEF",		SYSCONF,	_SC_2_LOCALEDEF		},
{ "POSIX2_SW_DEV",		SYSCONF,	_SC_2_SW_DEV		},
{ "POSIX2_UPE",			SYSCONF,	_SC_2_UPE		},

/* POSIX.1 Configurable System Variables */
{ "AIO_LISTIO_MAX",		SYSCONF,	_SC_AIO_LISTIO_MAX	},
{ "AIO_MAX",			SYSCONF,	_SC_AIO_MAX		},
{ "ARG_MAX",			SYSCONF,	_SC_ARG_MAX 		},
{ "CHILD_MAX",			SYSCONF,	_SC_CHILD_MAX		},
{ "CLK_TCK",			SYSCONF,	_SC_CLK_TCK		},
{ "MQ_OPEN_MAX",		SYSCONF,	_SC_MQ_OPEN_MAX		},
{ "MQ_PRIO_MAX",		SYSCONF,	_SC_MQ_PRIO_MAX		},
{ "NGROUPS_MAX",		SYSCONF,	_SC_NGROUPS_MAX		},
{ "OPEN_MAX",			SYSCONF,	_SC_OPEN_MAX		},
{ "STREAM_MAX",			SYSCONF,	_SC_STREAM_MAX		},
{ "TZNAME_MAX",			SYSCONF,	_SC_TZNAME_MAX		},
{ "_POSIX_JOB_CONTROL",		SYSCONF,	_SC_JOB_CONTROL 	},
{ "_POSIX_SAVED_IDS",		SYSCONF,	_SC_SAVED_IDS		},
{ "_POSIX_VERSION",		SYSCONF,	_SC_VERSION		},

{ "LINK_MAX",			PATHCONF,	_PC_LINK_MAX		},
{ "MAX_CANON",			PATHCONF,	_PC_MAX_CANON		},
{ "MAX_INPUT",			PATHCONF,	_PC_MAX_INPUT		},
{ "NAME_MAX",			PATHCONF,	_PC_NAME_MAX		},
{ "PATH_MAX",			PATHCONF,	_PC_PATH_MAX		},
{ "PIPE_BUF",			PATHCONF,	_PC_PIPE_BUF		},
{ "_POSIX_CHOWN_RESTRICTED",	PATHCONF,	_PC_CHOWN_RESTRICTED	},
{ "_POSIX_NO_TRUNC",		PATHCONF,	_PC_NO_TRUNC		},
{ "_POSIX_VDISABLE",		PATHCONF,	_PC_VDISABLE		},

/* POSIX.1b Configurable System Variables */
{ "PAGESIZE",			SYSCONF,	_SC_PAGESIZE		},
{ "_POSIX_ASYNCHRONOUS_IO",	SYSCONF,	_SC_ASYNCHRONOUS_IO	},
{ "_POSIX_FSYNC",		SYSCONF,	_SC_FSYNC		},
{ "_POSIX_MAPPED_FILES",	SYSCONF,	_SC_MAPPED_FILES	},
{ "_POSIX_MEMLOCK",		SYSCONF,	_SC_MEMLOCK		},
{ "_POSIX_MEMLOCK_RANGE",	SYSCONF,	_SC_MEMLOCK_RANGE	},
{ "_POSIX_MEMORY_PROTECTION",	SYSCONF,	_SC_MEMORY_PROTECTION	},
{ "_POSIX_MESSAGE_PASSING",	SYSCONF,	_SC_MESSAGE_PASSING	},
{ "_POSIX_MONOTONIC_CLOCK",	SYSCONF,	_SC_MONOTONIC_CLOCK	},
{ "_POSIX_PRIORITY_SCHEDULING", SYSCONF,	_SC_PRIORITY_SCHEDULING },
{ "_POSIX_SEMAPHORES",		SYSCONF,	_SC_SEMAPHORES		},
{ "_POSIX_SHARED_MEMORY_OBJECTS", SYSCONF,	_SC_SHARED_MEMORY_OBJECTS },
{ "_POSIX_SYNCHRONIZED_IO",	SYSCONF,	_SC_SYNCHRONIZED_IO	},
{ "_POSIX_TIMERS",		SYSCONF,	_SC_TIMERS		},

{ "_POSIX_SYNC_IO",		PATHCONF,	_PC_SYNC_IO		},

/* POSIX.1c Configurable System Variables */
{ "LOGIN_NAME_MAX",		SYSCONF,	_SC_LOGIN_NAME_MAX	},
{ "_POSIX_THREADS",		SYSCONF,	_SC_THREADS		},

/* POSIX.1j Configurable System Variables */
{ "_POSIX_BARRIERS",		SYSCONF,	_SC_BARRIERS		},
{ "_POSIX_READER_WRITER_LOCKS", SYSCONF,	_SC_READER_WRITER_LOCKS	},
{ "_POSIX_SPIN_LOCKS",		SYSCONF,	_SC_SPIN_LOCKS		},

/* XPG4.2 Configurable System Variables */
{ "IOV_MAX",			SYSCONF,	_SC_IOV_MAX		},
{ "PAGE_SIZE",			SYSCONF,	_SC_PAGE_SIZE		},
{ "_XOPEN_SHM",			SYSCONF,	_SC_XOPEN_SHM		},

/* X/Open CAE Spec. Issue 5 Version 2 Configurable System Variables */
{ "FILESIZEBITS",		PATHCONF,	_PC_FILESIZEBITS	},

/* POSIX.1-2001 XSI Option Group Configurable System Variables */
{ "ATEXIT_MAX",			SYSCONF,	_SC_ATEXIT_MAX		},

/* POSIX.1-2001 TSF Configurable System Variables */
{ "GETGR_R_SIZE_MAX",		SYSCONF,	_SC_GETGR_R_SIZE_MAX	},
{ "GETPW_R_SIZE_MAX",		SYSCONF,	_SC_GETPW_R_SIZE_MAX	},

/* Commonly provided extensions */
{ "_PHYS_PAGES",		SYSCONF,	_SC_PHYS_PAGES		},
{ "_AVPHYS_PAGES",		SYSCONF,	_SC_AVPHYS_PAGES	},
{ "_NPROCESSORS_CONF",		SYSCONF,	_SC_NPROCESSORS_CONF	},
{ "_NPROCESSORS_ONLN",		SYSCONF,	_SC_NPROCESSORS_ONLN	},

/* Data type related extensions */
{ "CHAR_BIT",			CONSTANT,	CHAR_BIT		},
{ "CHAR_MAX",			CONSTANT,	CHAR_MAX		},
{ "CHAR_MIN",			CONSTANT,	CHAR_MIN		},
{ "INT_MAX",			CONSTANT,	INT_MAX			},
{ "INT_MIN",			CONSTANT,	INT_MIN			},
{ "LONG_BIT",			CONSTANT,	LONG_BIT		},
{ "LONG_MAX",			CONSTANT,	LONG_MAX		},
{ "LONG_MIN",			CONSTANT,	LONG_MIN		},
{ "SCHAR_MAX",			CONSTANT,	SCHAR_MAX		},
{ "SCHAR_MIN",			CONSTANT,	SCHAR_MIN		},
{ "SHRT_MAX",			CONSTANT,	SHRT_MAX		},
{ "SHRT_MIN",			CONSTANT,	SHRT_MIN		},
{ "SSIZE_MAX",			CONSTANT,	SSIZE_MAX		},
{ "UCHAR_MAX",			UCONSTANT,	(long) UCHAR_MAX	},
{ "UINT_MAX",			UCONSTANT,	(long) UINT_MAX		},
{ "ULONG_MAX",			UCONSTANT,	(long) ULONG_MAX	},
{ "USHRT_MAX",			UCONSTANT,	(long) USHRT_MAX	},
{ "WORD_BIT",			CONSTANT,	WORD_BIT		},

{ NULL, CONSTANT, 0L }
};

static int all = 0;

static void usage(const char *p)
{
	(void)fprintf(stderr, "Usage: %s system_var\n\t%s -a\n"
	    "\t%s path_var pathname\n\t%s -a pathname\n", p, p, p, p);
	exit(EXIT_FAILURE);
}

static void print_long(const char *name, long val)
{
	if (all) printf("%s = %ld\n", name, val);
	else printf("%ld\n", val);
}

static void print_ulong(const char *name, unsigned long val)
{
	if (all) printf("%s = %lu\n", name, val);
	else printf("%lu\n", val);
}

static void print_string(const char *name, const char *val)
{
	if (all) printf("%s = %s\n", name, val);
	else printf("%s\n", val);
}

static int print_constant(const struct conf_variable *cp, const char *pathname)
{
	print_long(cp->name, cp->value);
	return 0;
}

static int print_uconstant(const struct conf_variable *cp, const char *pathname)
{
	print_ulong(cp->name, (unsigned long) cp->value);
	return 0;
}

static int print_sysconf(const struct conf_variable *cp, const char *pathname)
{
	long val;

	errno = 0;
	if ((val = sysconf((int)cp->value)) == -1) {
		if (errno != 0) err(EXIT_FAILURE, "sysconf(%ld)", cp->value);
		return -1;
	}
	print_long(cp->name, val);
	return 0;
}

static int print_confstr(const struct conf_variable *cp, const char *pathname)
{
	size_t len;
	char *val;

	errno = 0;
	if ((len = confstr((int)cp->value, NULL, 0)) == 0) goto error;
	if ((val = malloc(len)) == NULL) err(EXIT_FAILURE, "Can't allocate %zu bytes", len);
	errno = 0;
	if (confstr((int)cp->value, val, len) == 0) goto error;
	print_string(cp->name, val);
	free(val);
	return 0;
error:
	if (errno != EINVAL) err(EXIT_FAILURE, "confstr(%ld)", cp->value);
	return -1;
}

static int print_pathconf(const struct conf_variable *cp, const char *pathname)
{
	long val;

	errno = 0;
	if ((val = pathconf(pathname, (int)cp->value)) == -1) {
		if (all && errno == EINVAL) return 0;
		if (errno != 0) err(EXIT_FAILURE, "pathconf(%s, %ld)", pathname, cp->value);
		return -1;
	}
	print_long(cp->name, val);
	return 0;
}

typedef int (*handler_t)(const struct conf_variable *cp, const char *pathname);
static const handler_t type_handlers[NUM_TYPES] = {
	[SYSCONF]	= print_sysconf,
	[CONFSTR]	= print_confstr,
	[PATHCONF]	= print_pathconf,
	[CONSTANT]	= print_constant,
	[UCONSTANT]	= print_uconstant,
};

int main(int argc, char **argv)
{
	const char *progname = argv[0];
	const struct conf_variable *cp;
	const char *varname, *pathname;
	int ch, found = 0;

	(void)setlocale(LC_ALL, "");
	while ((ch = getopt(argc, argv, "a")) != -1) {
		switch (ch) {
		case 'a':
			all = 1;
			break;
		case '?':
		default:
			usage(progname);
		}
	}
	argc -= optind;
	argv += optind;

	if (!all) {
		if (argc == 0)
			usage(progname);
		varname = argv[0];
		argc--;
		argv++;
	} else
		varname = NULL;

	if (argc > 1)
		usage(progname);
	pathname = argv[0];	/* may be NULL */

	for (cp = conf_table; cp->name != NULL; cp++) {
		if (!all && strcmp(varname, cp->name) != 0) continue;
		if ((cp->type == PATHCONF) == (pathname != NULL)) {
			if (type_handlers[cp->type](cp, pathname) < 0)
				print_string(cp->name, "undefined");
			found = 1;
		} else if (!all)
			errx(EXIT_FAILURE, "%s: invalid variable type", cp->name);
	}
	if (!all && !found) errx(EXIT_FAILURE, "%s: unknown variable", varname);
	(void)fflush(stdout);
	return ferror(stdout) ? EXIT_FAILURE : EXIT_SUCCESS;
}