aboutsummaryrefslogtreecommitdiff
path: root/toys/modinfo.c
blob: c878fc4e5960b5383c40326ec320ee9896d2b0e0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/* vi: set sw=4 ts=4:
 *
 * modinfo.c - Display module info
 *
 * Copyright 2012 Andre Renaud <andre@bluewatersys.com>
 *

USE_MODINFO(NEWTOY(modinfo, "<1F:0", TOYFLAG_BIN))

config MODINFO
	bool "modinfo"
	default y
	help
	  usage: modinfo [-0] [-F field] [modulename...]
*/

#include "toys.h"

#define FLAG_0  (1 << 0)

DEFINE_GLOBALS(
    char *field;
)
#define TT this.modinfo

static const char *modinfo_tags[] = {
    "alias", "license", "description", "author", "vermagic",
    "srcversion", "intree", "parm", "depends",
};

static void output_field(const char *field, const char *value)
{
    int len;

    if (TT.field && strcmp(TT.field, field) != 0)
        return;

    len = strlen(field);

    if (TT.field)
        xprintf("%s", value);
    else
        xprintf("%s:%*s%s",
                field, 15 - len, "", value);
    if (toys.optflags & FLAG_0)
        xwrite(fileno(stdout), "\0", 1);
    else
        xputs("");
}


static void modinfo_file(struct dirtree *dir)
{
    int fd, len, i;
    char *buf, *pos;
    char *full_name;

    full_name = dirtree_path(dir, NULL);

    output_field("filename", full_name);
    fd = xopen(full_name, O_RDONLY);
    len = fdlength(fd);
    buf = xmalloc(len);
    xreadall(fd, buf, len);

    for (pos = buf; pos < buf + len + 10; pos++) {
        if (*pos)
            continue;

        for (i = 0; i < sizeof(modinfo_tags) / sizeof(modinfo_tags[0]); i++) {
            const char *str = modinfo_tags[i];
            int len = strlen(str); 
            if (strncmp(pos + 1, str, len) == 0 && pos[len + 1] == '=')
                output_field(str, &pos[len + 2]);
        }
    }

    free(full_name);
    free(buf);
    close(fd);
}

static int check_module(struct dirtree *new)
{
    if (S_ISREG(new->st.st_mode)) {
        char **s;
        for (s = toys.optargs; *s; s++) {
            int len = strlen(*s);
            if (!strncmp(*s, new->name, len) && !strcmp(new->name+len, ".ko"))
                modinfo_file(new);
        }
    }

    return dirtree_notdotdot(new);
}

void modinfo_main(void)
{
    struct utsname uts;
    if (uname(&uts) < 0) perror_exit("bad uname");
    sprintf(toybuf, "/lib/modules/%s", uts.release);
    dirtree_read(toybuf, check_module);
}