257 lines
6.1 KiB
C
257 lines
6.1 KiB
C
/* $NetBSD: main.c,v 1.26 2021/04/07 14:45:28 simonb Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 2008 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* 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 <sys/cdefs.h>
|
|
#ifndef lint
|
|
__RCSID("$NetBSD: main.c,v 1.26 2021/04/07 14:45:28 simonb Exp $");
|
|
#endif /* !lint */
|
|
|
|
#include <sys/module.h>
|
|
#include <sys/param.h>
|
|
#include <sys/sysctl.h>
|
|
|
|
#include <err.h>
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdbool.h>
|
|
|
|
#include "prog_ops.h"
|
|
|
|
static void usage(void) __dead;
|
|
static int modstatcmp(const void *, const void *);
|
|
|
|
static const char *classes[] = {
|
|
"any",
|
|
"misc",
|
|
"vfs",
|
|
"driver",
|
|
"exec",
|
|
"secmodel",
|
|
"bufq"
|
|
};
|
|
const unsigned int class_max = __arraycount(classes);
|
|
|
|
static const char *sources[] = {
|
|
"builtin",
|
|
"boot",
|
|
"filesys",
|
|
};
|
|
const unsigned int source_max = __arraycount(sources);
|
|
|
|
static const char *modflags[] = {
|
|
"-", "f", "a", "af"
|
|
};
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
struct iovec iov;
|
|
modstat_t *ms;
|
|
size_t len;
|
|
const char *name;
|
|
char sbuf[32];
|
|
int ch, rc, modauto = 1;
|
|
size_t maxnamelen = 16, i, modautolen;
|
|
char loadable = '\0';
|
|
const char *reqoff, *req;
|
|
bool address = false;
|
|
|
|
name = NULL;
|
|
|
|
while ((ch = getopt(argc, argv, "Aaekn:")) != -1) {
|
|
switch (ch) {
|
|
case 'A': /* FALLTHROUGH */
|
|
case 'a': /* FALLTHROUGH */
|
|
case 'e':
|
|
loadable = (char)ch;
|
|
break;
|
|
case 'k':
|
|
address = true;
|
|
break;
|
|
case 'n':
|
|
name = optarg;
|
|
break;
|
|
default:
|
|
usage();
|
|
/* NOTREACHED */
|
|
}
|
|
}
|
|
|
|
argc -= optind;
|
|
argv += optind;
|
|
if (argc == 1 && name == NULL)
|
|
name = argv[0];
|
|
else if (argc != 0)
|
|
usage();
|
|
|
|
if (prog_init && prog_init() == -1)
|
|
err(1, "prog init failed");
|
|
|
|
if (loadable == 'A' || loadable == 'a') {
|
|
if (prog_modctl(MODCTL_EXISTS, (void *)(uintptr_t)1)) {
|
|
switch (errno) {
|
|
case ENOSYS:
|
|
errx(EXIT_FAILURE, "The kernel was compiled "
|
|
"without options MODULAR.");
|
|
break;
|
|
case EPERM:
|
|
errx(EXIT_FAILURE, "Modules can not be "
|
|
"autoloaded right now.");
|
|
break;
|
|
default:
|
|
err(EXIT_FAILURE, "modctl_exists for autoload");
|
|
break;
|
|
}
|
|
} else {
|
|
if (loadable == 'A') {
|
|
modautolen = sizeof(modauto);
|
|
rc = sysctlbyname("kern.module.autoload",
|
|
&modauto, &modautolen, NULL, 0);
|
|
if (rc != 0) {
|
|
err(EXIT_FAILURE, "sysctl "
|
|
"kern.module.autoload failed.");
|
|
}
|
|
}
|
|
errx(EXIT_SUCCESS, "Modules can be autoloaded%s.",
|
|
modauto ? "" : ", but kern.module.autoload = 0");
|
|
}
|
|
}
|
|
|
|
if (loadable == 'e') {
|
|
if (prog_modctl(MODCTL_EXISTS, (void *)(uintptr_t)0)) {
|
|
switch (errno) {
|
|
case ENOSYS:
|
|
errx(EXIT_FAILURE, "The kernel was compiled "
|
|
"without options MODULAR.");
|
|
break;
|
|
case EPERM:
|
|
errx(EXIT_FAILURE, "You are not allowed to "
|
|
"load modules right now.");
|
|
break;
|
|
default:
|
|
err(EXIT_FAILURE, "modctl_exists for autoload");
|
|
break;
|
|
}
|
|
} else {
|
|
errx(EXIT_SUCCESS, "You can load modules.");
|
|
}
|
|
}
|
|
|
|
for (len = 8192;;) {
|
|
iov.iov_base = malloc(len);
|
|
iov.iov_len = len;
|
|
if (prog_modctl(MODCTL_STAT, &iov)) {
|
|
err(EXIT_FAILURE, "modctl(MODCTL_STAT)");
|
|
}
|
|
if (len >= iov.iov_len) {
|
|
break;
|
|
}
|
|
free(iov.iov_base);
|
|
len = iov.iov_len;
|
|
}
|
|
|
|
len = *(int *)iov.iov_base;
|
|
ms = (modstat_t *)((char *)iov.iov_base + sizeof(int));
|
|
|
|
qsort(ms, len, sizeof(modstat_t), modstatcmp);
|
|
for (i = 0; i < len; i++, ms++) {
|
|
size_t namelen = strlen(ms->ms_name);
|
|
if (maxnamelen < namelen)
|
|
maxnamelen = namelen;
|
|
}
|
|
ms = (modstat_t *)((char *)iov.iov_base + sizeof(int));
|
|
reqoff = (char *)(&ms[len]);
|
|
|
|
printf("%-*s %-8s %-8s %-4s %5s ",
|
|
(int)maxnamelen, "NAME", "CLASS", "SOURCE", "FLAG", "REFS");
|
|
if (address)
|
|
printf("%-16s ", "ADDRESS");
|
|
printf("%7s %s \n", "SIZE", "REQUIRES");
|
|
|
|
for (; len != 0; ms++, len--) {
|
|
const char *class;
|
|
const char *source;
|
|
|
|
if (ms->ms_reqoffset == 0)
|
|
req = "-";
|
|
else {
|
|
req = &reqoff[ms->ms_reqoffset];
|
|
}
|
|
if (name != NULL && strcmp(ms->ms_name, name) != 0) {
|
|
continue;
|
|
}
|
|
if (ms->ms_size == 0) {
|
|
sbuf[0] = '-';
|
|
sbuf[1] = '\0';
|
|
} else {
|
|
snprintf(sbuf, sizeof(sbuf), "%u", ms->ms_size);
|
|
}
|
|
if (ms->ms_class <= class_max)
|
|
class = classes[ms->ms_class];
|
|
else
|
|
class = "UNKNOWN";
|
|
if (ms->ms_source < source_max)
|
|
source = sources[ms->ms_source];
|
|
else
|
|
source = "UNKNOWN";
|
|
|
|
printf("%-*s %-8s %-8s %-4s %5d ",
|
|
(int)maxnamelen, ms->ms_name, class, source,
|
|
modflags[ms->ms_flags & (__arraycount(modflags) - 1)],
|
|
ms->ms_refcnt);
|
|
if (address)
|
|
printf("%-16" PRIx64 " ", ms->ms_addr);
|
|
printf("%7s %s\n", sbuf, (req));
|
|
}
|
|
|
|
exit(EXIT_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
usage(void)
|
|
{
|
|
|
|
(void)fprintf(stderr, "Usage: %s [-Aaek] [-n name | name]\n",
|
|
getprogname());
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
static int
|
|
modstatcmp(const void *a, const void *b)
|
|
{
|
|
const modstat_t *msa, *msb;
|
|
|
|
msa = a;
|
|
msb = b;
|
|
|
|
return strcmp(msa->ms_name, msb->ms_name);
|
|
}
|