main.c revision 1.12 1 1.12 nonaka /* $NetBSD: main.c,v 1.12 2011/06/03 15:34:46 nonaka Exp $ */
2 1.1 ad
3 1.1 ad /*-
4 1.1 ad * Copyright (c) 2008 The NetBSD Foundation, Inc.
5 1.1 ad * All rights reserved.
6 1.1 ad *
7 1.1 ad * Redistribution and use in source and binary forms, with or without
8 1.1 ad * modification, are permitted provided that the following conditions
9 1.1 ad * are met:
10 1.1 ad * 1. Redistributions of source code must retain the above copyright
11 1.1 ad * notice, this list of conditions and the following disclaimer.
12 1.1 ad * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 ad * notice, this list of conditions and the following disclaimer in the
14 1.1 ad * documentation and/or other materials provided with the distribution.
15 1.1 ad *
16 1.1 ad * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
17 1.1 ad * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
18 1.1 ad * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19 1.1 ad * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
20 1.1 ad * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21 1.1 ad * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22 1.1 ad * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23 1.1 ad * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24 1.1 ad * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25 1.1 ad * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26 1.1 ad * POSSIBILITY OF SUCH DAMAGE.
27 1.1 ad */
28 1.1 ad
29 1.1 ad #include <sys/cdefs.h>
30 1.1 ad #ifndef lint
31 1.12 nonaka __RCSID("$NetBSD: main.c,v 1.12 2011/06/03 15:34:46 nonaka Exp $");
32 1.1 ad #endif /* !lint */
33 1.1 ad
34 1.1 ad #include <sys/module.h>
35 1.1 ad
36 1.1 ad #include <stdio.h>
37 1.1 ad #include <stdlib.h>
38 1.1 ad #include <string.h>
39 1.1 ad #include <unistd.h>
40 1.1 ad #include <err.h>
41 1.1 ad
42 1.11 pooka #include "prog_ops.h"
43 1.11 pooka
44 1.1 ad int main(int, char **);
45 1.1 ad static void usage(void) __dead;
46 1.6 ad static int modstatcmp(const void *, const void *);
47 1.1 ad
48 1.1 ad static const char *classes[] = {
49 1.1 ad "any",
50 1.1 ad "misc",
51 1.1 ad "vfs",
52 1.1 ad "driver",
53 1.1 ad "exec",
54 1.7 elad "secmodel",
55 1.1 ad };
56 1.10 pooka const unsigned int class_max = __arraycount(classes);
57 1.1 ad
58 1.1 ad static const char *sources[] = {
59 1.5 ad "builtin",
60 1.1 ad "boot",
61 1.1 ad "filesys",
62 1.1 ad };
63 1.10 pooka const unsigned int source_max = __arraycount(sources);
64 1.1 ad
65 1.1 ad int
66 1.1 ad main(int argc, char **argv)
67 1.1 ad {
68 1.1 ad struct iovec iov;
69 1.1 ad modstat_t *ms;
70 1.1 ad size_t len;
71 1.1 ad const char *name;
72 1.1 ad char sbuf[32];
73 1.1 ad int ch;
74 1.12 nonaka size_t maxnamelen = 16, i;
75 1.1 ad
76 1.1 ad name = NULL;
77 1.1 ad
78 1.1 ad while ((ch = getopt(argc, argv, "n:")) != -1) {
79 1.1 ad switch (ch) {
80 1.1 ad case 'n':
81 1.1 ad name = optarg;
82 1.1 ad break;
83 1.1 ad default:
84 1.1 ad usage();
85 1.1 ad /* NOTREACHED */
86 1.1 ad }
87 1.1 ad }
88 1.1 ad
89 1.1 ad argc -= optind;
90 1.1 ad argv += optind;
91 1.1 ad if (argc != 0)
92 1.1 ad usage();
93 1.1 ad
94 1.11 pooka if (prog_init && prog_init() == -1)
95 1.11 pooka err(1, "prog init failed");
96 1.11 pooka
97 1.9 pooka for (len = 8192;;) {
98 1.1 ad iov.iov_base = malloc(len);
99 1.1 ad iov.iov_len = len;
100 1.11 pooka if (prog_modctl(MODCTL_STAT, &iov)) {
101 1.1 ad err(EXIT_FAILURE, "modctl(MODCTL_STAT)");
102 1.1 ad }
103 1.1 ad if (len >= iov.iov_len) {
104 1.1 ad break;
105 1.1 ad }
106 1.1 ad free(iov.iov_base);
107 1.2 ad len = iov.iov_len;
108 1.1 ad }
109 1.1 ad
110 1.1 ad len = iov.iov_len / sizeof(modstat_t);
111 1.6 ad qsort(iov.iov_base, len, sizeof(modstat_t), modstatcmp);
112 1.12 nonaka for (i = 0, ms = iov.iov_base; i < len; i++, ms++) {
113 1.12 nonaka size_t namelen = strlen(ms->ms_name);
114 1.12 nonaka if (maxnamelen < namelen)
115 1.12 nonaka maxnamelen = namelen;
116 1.12 nonaka }
117 1.12 nonaka printf("%-*s %-10s %-10s %-5s %-8s %s\n",
118 1.12 nonaka maxnamelen, "NAME", "CLASS", "SOURCE", "REFS", "SIZE", "REQUIRES");
119 1.1 ad for (ms = iov.iov_base; len != 0; ms++, len--) {
120 1.10 pooka const char *class;
121 1.10 pooka const char *source;
122 1.10 pooka
123 1.1 ad if (name != NULL && strcmp(ms->ms_name, name) != 0) {
124 1.1 ad continue;
125 1.1 ad }
126 1.1 ad if (ms->ms_required[0] == '\0') {
127 1.1 ad ms->ms_required[0] = '-';
128 1.1 ad ms->ms_required[1] = '\0';
129 1.1 ad }
130 1.1 ad if (ms->ms_size == 0) {
131 1.1 ad sbuf[0] = '-';
132 1.1 ad sbuf[1] = '\0';
133 1.1 ad } else {
134 1.1 ad snprintf(sbuf, sizeof(sbuf), "%u", ms->ms_size);
135 1.1 ad }
136 1.10 pooka if (ms->ms_class <= class_max)
137 1.10 pooka class = classes[ms->ms_class];
138 1.10 pooka else
139 1.10 pooka class = "UNKNOWN";
140 1.10 pooka if (ms->ms_source < source_max)
141 1.10 pooka source = sources[ms->ms_source];
142 1.10 pooka else
143 1.10 pooka source = "UNKNOWN";
144 1.10 pooka
145 1.12 nonaka printf("%-*s %-10s %-10s %-5d %-8s %s\n",
146 1.12 nonaka maxnamelen, ms->ms_name, class, source, ms->ms_refcnt, sbuf,
147 1.10 pooka ms->ms_required);
148 1.1 ad }
149 1.1 ad
150 1.1 ad exit(EXIT_SUCCESS);
151 1.1 ad }
152 1.1 ad
153 1.1 ad static void
154 1.1 ad usage(void)
155 1.1 ad {
156 1.1 ad
157 1.4 hira (void)fprintf(stderr, "Usage: %s [-n name]\n", getprogname());
158 1.1 ad exit(EXIT_FAILURE);
159 1.1 ad }
160 1.6 ad
161 1.6 ad static int
162 1.6 ad modstatcmp(const void *a, const void *b)
163 1.6 ad {
164 1.6 ad const modstat_t *msa, *msb;
165 1.6 ad
166 1.6 ad msa = a;
167 1.6 ad msb = b;
168 1.6 ad
169 1.6 ad return strcmp(msa->ms_name, msb->ms_name);
170 1.6 ad }
171