source: trunk/mindi-busybox/modutils/lsmod.c @ 956

Last change on this file since 956 was 904, checked in by bruno, 13 years ago

merge -r890:902 $SVN_M/branches/stable

File size: 4.8 KB
Line 
1/* vi: set sw=4 ts=4: */
2/*
3 * Mini lsmod implementation for busybox
4 *
5 * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
6 *
7 * Modified by Alcove, Julien Gaulmin <julien.gaulmin@alcove.fr> and
8 * Nicolas Ferre <nicolas.ferre@alcove.fr> to support pre 2.1 kernels
9 * (which lack the query_module() interface).
10 *
11 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
12 */
13
14#include "busybox.h"
15#include <stdlib.h>
16#include <stdio.h>
17#include <string.h>
18#include <stddef.h>
19#include <errno.h>
20#include <unistd.h>
21#include <dirent.h>
22#include <ctype.h>
23#include <assert.h>
24#include <getopt.h>
25#include <sys/utsname.h>
26#include <sys/file.h>
27
28
29#ifndef CONFIG_FEATURE_CHECK_TAINTED_MODULE
30static inline void check_tainted(void) { printf("\n"); }
31#else
32#define TAINT_FILENAME                  "/proc/sys/kernel/tainted"
33#define TAINT_PROPRIETORY_MODULE        (1<<0)
34#define TAINT_FORCED_MODULE             (1<<1)
35#define TAINT_UNSAFE_SMP                (1<<2)
36
37static void check_tainted(void)
38{
39    int tainted;
40    FILE *f;
41
42    tainted = 0;
43    if ((f = fopen(TAINT_FILENAME, "r"))) {
44        fscanf(f, "%d", &tainted);
45        fclose(f);
46    }
47    if (f && tainted) {
48        printf("    Tainted: %c%c%c\n",
49                tainted & TAINT_PROPRIETORY_MODULE      ? 'P' : 'G',
50                tainted & TAINT_FORCED_MODULE           ? 'F' : ' ',
51                tainted & TAINT_UNSAFE_SMP              ? 'S' : ' ');
52    }
53    else {
54        printf("    Not tainted\n");
55    }
56}
57#endif
58
59#ifdef CONFIG_FEATURE_QUERY_MODULE_INTERFACE
60
61struct module_info
62{
63    unsigned long addr;
64    unsigned long size;
65    unsigned long flags;
66    long usecount;
67};
68
69
70int query_module(const char *name, int which, void *buf, size_t bufsize, size_t *ret);
71
72enum {
73/* Values for query_module's which.  */
74    QM_MODULES = 1,
75    QM_DEPS = 2,
76    QM_REFS = 3,
77    QM_SYMBOLS = 4,
78    QM_INFO = 5,
79
80/* Bits of module.flags.  */
81    NEW_MOD_RUNNING = 1,
82    NEW_MOD_DELETED = 2,
83    NEW_MOD_AUTOCLEAN = 4,
84    NEW_MOD_VISITED = 8,
85    NEW_MOD_USED_ONCE = 16,
86    NEW_MOD_INITIALIZING = 64
87};
88
89int lsmod_main(int argc, char **argv)
90{
91    struct module_info info;
92    char *module_names, *mn, *deps, *dn;
93    size_t bufsize, depsize, nmod, count, i, j;
94
95    module_names = xmalloc(bufsize = 256);
96    if (my_query_module(NULL, QM_MODULES, &module_names, &bufsize, &nmod)) {
97        bb_perror_msg_and_die("QM_MODULES");
98    }
99
100    deps = xmalloc(depsize = 256);
101    printf("Module                  Size  Used by");
102    check_tainted();
103
104    for (i = 0, mn = module_names; i < nmod; mn += strlen(mn) + 1, i++) {
105        if (query_module(mn, QM_INFO, &info, sizeof(info), &count)) {
106            if (errno == ENOENT) {
107                /* The module was removed out from underneath us. */
108                continue;
109            }
110            /* else choke */
111            bb_perror_msg_and_die("module %s: QM_INFO", mn);
112        }
113        if (my_query_module(mn, QM_REFS, &deps, &depsize, &count)) {
114            if (errno == ENOENT) {
115                /* The module was removed out from underneath us. */
116                continue;
117            }
118            bb_perror_msg_and_die("module %s: QM_REFS", mn);
119        }
120        printf("%-20s%8lu%4ld", mn, info.size, info.usecount);
121        if (info.flags & NEW_MOD_DELETED)
122            printf(" (deleted)");
123        else if (info.flags & NEW_MOD_INITIALIZING)
124            printf(" (initializing)");
125        else if (!(info.flags & NEW_MOD_RUNNING))
126            printf(" (uninitialized)");
127        else {
128            if (info.flags & NEW_MOD_AUTOCLEAN)
129                printf(" (autoclean) ");
130            if (!(info.flags & NEW_MOD_USED_ONCE))
131                printf(" (unused)");
132        }
133        if (count) printf(" [");
134        for (j = 0, dn = deps; j < count; dn += strlen(dn) + 1, j++) {
135            printf("%s%s", dn, (j==count-1)? "":" ");
136        }
137        if (count) printf("]");
138
139        printf("\n");
140    }
141
142#ifdef CONFIG_FEATURE_CLEAN_UP
143    free(module_names);
144#endif
145
146    return( 0);
147}
148
149#else /* CONFIG_FEATURE_QUERY_MODULE_INTERFACE */
150
151int lsmod_main(int argc, char **argv)
152{
153    printf("Module                  Size  Used by");
154    check_tainted();
155#if defined(CONFIG_FEATURE_LSMOD_PRETTY_2_6_OUTPUT)
156    {
157      FILE *file;
158      char line[4096];
159
160      file = bb_xfopen("/proc/modules", "r");
161
162      while (fgets(line, sizeof(line), file)) {
163        char *tok;
164
165        tok = strtok(line, " \t");
166        printf("%-19s", tok);
167        tok = strtok(NULL, " \t\n");
168        printf(" %8s", tok);
169        tok = strtok(NULL, " \t\n");
170        /* Null if no module unloading support. */
171        if (tok) {
172          printf("  %s", tok);
173          tok = strtok(NULL, "\n");
174          if (!tok)
175        tok = "";
176          /* New-style has commas, or -.  If so,
177         truncate (other fields might follow). */
178          else if (strchr(tok, ',')) {
179        tok = strtok(tok, "\t ");
180        /* Strip trailing comma. */
181        if (tok[strlen(tok)-1] == ',')
182          tok[strlen(tok)-1] = '\0';
183          } else if (tok[0] == '-'
184             && (tok[1] == '\0' || isspace(tok[1])))
185        tok = "";
186          printf(" %s", tok);
187        }
188        printf("\n");
189      }
190      fclose(file);
191    }
192    return EXIT_SUCCESS;
193#else
194    if (bb_xprint_file_by_name("/proc/modules") == 0)
195        return EXIT_SUCCESS;
196#endif  /*  CONFIG_FEATURE_2_6_MODULES  */
197
198    return EXIT_FAILURE;
199}
200
201#endif /* CONFIG_FEATURE_QUERY_MODULE_INTERFACE */
Note: See TracBrowser for help on using the repository browser.