blob: 0d7d72d8b330a89f393ec640232d8192a609f569 [file] [log] [blame]
Denis Vlasenkoba1315d2008-09-13 14:59:38 +00001/*
2 * Common modutils related functions for busybox
3 *
4 * Copyright (C) 2008 by Timo Teras <timo.teras@iki.fi>
5 *
6 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
7 */
8
9#include "modutils.h"
10
11#ifdef __UCLIBC__
12extern int init_module(void *module, unsigned long len, const char *options);
13extern int delete_module(const char *module, unsigned int flags);
14#else
15# include <sys/syscall.h>
16# define init_module(mod, len, opts) syscall(__NR_init_module, mod, len, opts)
17# define delete_module(mod, flags) syscall(__NR_delete_module, mod, flags)
18#endif
19
20USE_FEATURE_2_4_MODULES(char *insmod_outputname);
21
22/*
23 a libbb candidate from ice age!
24*/
25llist_t FAST_FUNC *llist_find(llist_t *first, const char *str)
26{
27 while (first != NULL) {
28 if (strcmp(first->data, str) == 0)
29 return first;
30 first = first->link;
31 }
32 return NULL;
33}
34
35void FAST_FUNC replace(char *s, char what, char with)
36{
37 while (*s) {
38 if (what == *s)
39 *s = with;
40 ++s;
41 }
42}
43
44char * FAST_FUNC replace_underscores(char *s)
45{
46 replace(s, '-', '_');
47 return s;
48}
49
50int FAST_FUNC string_to_llist(char *string, llist_t **llist, const char *delim)
51{
52 char *tok;
53 int len = 0;
54
55 while ((tok = strsep(&string, delim)) != NULL) {
56 if (tok[0] == '\0')
57 continue;
58 llist_add_to_end(llist, xstrdup(tok));
59 len += strlen(tok);
60 }
61 return len;
62}
63
64char * FAST_FUNC filename2modname(const char *filename, char *modname)
65{
66 int i;
67 char *from;
68
69 if (filename == NULL)
70 return NULL;
71 if (modname == NULL)
72 modname = xmalloc(MODULE_NAME_LEN);
73 from = bb_get_last_path_component_nostrip(filename);
74 for (i = 0; i < MODULE_NAME_LEN && from[i] != '\0' && from[i] != '.'; i++)
75 modname[i] = (from[i] == '-') ? '_' : from[i];
76 modname[i] = 0;
77
78 return modname;
79}
80
81const char * FAST_FUNC moderror(int err)
82{
83 switch (err) {
84 case -1:
85 return "no such module";
86 case ENOEXEC:
87 return "invalid module format";
88 case ENOENT:
89 return "unknown symbol in module, or unknown parameter";
90 case ESRCH:
91 return "module has wrong symbol version";
92 case ENOSYS:
93 return "kernel does not support requested operation";
94 default:
95 return strerror(err);
96 }
97}
98
99char * FAST_FUNC parse_cmdline_module_options(char **argv)
100{
101 char *options;
102 int optlen;
103
104 options = xzalloc(1);
105 optlen = 0;
106 while (*++argv) {
107 options = xrealloc(options, optlen + 2 + strlen(*argv) + 2);
108 /* Spaces handled by "" pairs, but no way of escaping quotes */
Denis Vlasenkobb26db42008-10-31 02:04:28 +0000109 optlen += sprintf(options + optlen, (strchr(*argv, ' ') ? "\"%s\" " : "%s "), *argv);
Denis Vlasenkoba1315d2008-09-13 14:59:38 +0000110 }
111 return options;
112}
113
114int FAST_FUNC bb_init_module(const char *filename, const char *options)
115{
116 size_t len = MAXINT(ssize_t);
117 char *image;
118 int rc = ENOENT;
119
120#if ENABLE_FEATURE_2_4_MODULES
121 if (get_linux_version_code() < KERNEL_VERSION(2,6,0))
122 return bb_init_module_24(filename, options);
123#endif
124
125 /* Use the 2.6 way */
Denis Vlasenkofc668922008-11-06 02:32:31 +0000126 image = xmalloc_open_zipped_read_close(filename, &len);
Denis Vlasenkoba1315d2008-09-13 14:59:38 +0000127 if (image) {
128 if (init_module(image, len, options) != 0)
129 rc = errno;
130 else
131 rc = 0;
132 free(image);
133 }
134
135 return rc;
136}
137
138int FAST_FUNC bb_delete_module(const char *module, unsigned int flags)
139{
140 return delete_module(module, flags);
141}