summaryrefslogtreecommitdiff
path: root/modutils/depmod.c
blob: 9131dc1abd2abe93c37c27b76cd4b0196c057769 (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
104
105
106
107
108
109
110
111
112
113
114
115
116
/* vi: set sw=4 ts=4: */
/*
 * depmod - generate modules.dep
 * Copyright (c) 2008 Bernhard Fischer
 *
 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
 */

#undef _GNU_SOURCE
#define _GNU_SOURCE
#include <libbb.h>
#include <sys/utsname.h> /* uname() */

struct globals {
	llist_t *lst;
};
#define G (*(struct globals*)&bb_common_bufsiz1)
/* We have to zero it out because of NOEXEC */
#define INIT_G() memset(&G, 0, sizeof(G))

static int fill_lst(const char *modulename, struct stat ATTRIBUTE_UNUSED *sb,
					void ATTRIBUTE_UNUSED *data, int ATTRIBUTE_UNUSED depth)
{
	llist_add_to_end(&G.lst, strdup(modulename));
	return TRUE;
}

static int fileAction(const char *fname, struct stat ATTRIBUTE_UNUSED *sb,
					void *data, int ATTRIBUTE_UNUSED depth)
{
	size_t seen = 0;
	size_t len = MAXINT(ssize_t);
	void *the_module = xmalloc_open_read_close(fname, &len), *ptr = the_module;
	const char *deps;
	RESERVE_CONFIG_BUFFER(depends, 512);
	RESERVE_CONFIG_BUFFER(buf1, 512);

	memset(buf1, 0, sizeof(buf1));
	memset(depends, 0, sizeof(depends));
	sprintf(buf1, "\n%s:", fname);

	if (last_char_is(fname, 'o') == NULL) /* not a module */
		goto done;
	write((int)data, buf1, strlen(buf1));
//bb_info_msg("[%d] fname='%s'", (int)data, fname);
	do {
		/* search for a 'd' */
		ptr = memchr(ptr, 'd', len - seen);
		if (ptr == NULL) /* no d left, done */
			break;
		if (sscanf(ptr, "depends=%s", depends) == 1)
			break;
		seen = ++ptr - the_module;
	} while (1);
//bb_info_msg(" depends='%s'", depends);
	deps = depends;
	while (*deps) {
		llist_t * _lst = G.lst;
		ptr = memchr(deps, ',', strlen(deps));
		if (ptr != NULL)
			*(char*)ptr = '\0';
		/* remember the length of the current dependency plus eventual 0 byte */
		len = strlen(deps) + (ptr != NULL);
		sprintf(buf1, "/%s.", deps); /* make sure we match the correct file */
		while (_lst) {
			ptr = strstr(_lst->data, buf1);
			if (ptr != NULL)
				break; /* found it */
			_lst = _lst->link;
		}
		if (_lst && _lst->data) {
			const char separator = ' ';
//bb_info_msg("[%s] -> '%s'", deps, _lst->data);
			write((int)data, &separator, 1);
			write((int)data, _lst->data, strlen(_lst->data));

			deps += len;
		}
	}
done:
	RELEASE_CONFIG_BUFFER(depends);
	RELEASE_CONFIG_BUFFER(buf1);
	free(the_module);
	return TRUE;
}

int depmod_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int depmod_main(int ATTRIBUTE_UNUSED argc, char **argv)
{
	int retval = EXIT_SUCCESS;
//	static const char moddir_base[] ALIGN1 = "/lib/modules/%s";

	int fd = xopen3("/tmp/modules.dep", O_CREAT|O_WRONLY|O_TRUNC,
					S_IWUSR|S_IRUSR|S_IRGRP|S_IROTH);
	argv++;
	do {
		if (!recursive_action(*argv,
				ACTION_RECURSE, /* flags */
				fill_lst, /* file action */
				NULL, /* dir action */
				NULL, /* user data */
				0) || /* depth */
			!recursive_action(*argv,
				ACTION_RECURSE, /* flags */
				fileAction, /* file action */
				NULL, /* dir action */
				(void*)fd, /* user data */
				0)) { /* depth */
			retval = EXIT_FAILURE;
		}
	} while (*++argv);

	if (ENABLE_FEATURE_CLEAN_UP)
		close(fd);
	return retval;
}