1 /* vi: set sw=4 ts=4: */
3 * Sysctl 1.01 - A utility to read and manipulate the sysctl parameters
5 * Copyright 1999 George Staikos
7 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
11 * - added -p <preload> to preload values from a file
13 * - busybox applet aware by <solar@gentoo.org>
22 static int sysctl_read_setting(const char *setting, int output);
23 static int sysctl_write_setting(const char *setting, int output);
24 static int sysctl_preload_file(const char *filename, int output);
25 static int sysctl_display_all(const char *path, int output, int show_table);
30 static const char PROC_PATH[] ALIGN1 = "/proc/sys/";
31 static const char DEFAULT_PRELOAD[] ALIGN1 = "/etc/sysctl.conf";
34 static const char ERR_UNKNOWN_PARAMETER[] ALIGN1 =
35 "error: Unknown parameter '%s'\n";
36 static const char ERR_MALFORMED_SETTING[] ALIGN1 =
37 "error: Malformed setting '%s'\n";
38 static const char ERR_NO_EQUALS[] ALIGN1 =
39 "error: '%s' must be of the form name=value\n";
40 static const char ERR_INVALID_KEY[] ALIGN1 =
41 "error: '%s' is an unknown key\n";
42 static const char ERR_UNKNOWN_WRITING[] ALIGN1 =
43 "error: unknown error %d setting key '%s'\n";
44 static const char ERR_UNKNOWN_READING[] ALIGN1 =
45 "error: unknown error %d reading key '%s'\n";
46 static const char ERR_PERMISSION_DENIED[] ALIGN1 =
47 "error: permission denied on key '%s'\n";
48 static const char ERR_PRELOAD_FILE[] ALIGN1 =
49 "error: cannot open preload file '%s'\n";
50 static const char WARN_BAD_LINE[] ALIGN1 =
51 "warning: %s(%d): invalid syntax, continuing...\n";
54 static void dwrite_str(int fd, const char *buf)
56 write(fd, buf, strlen(buf));
62 int sysctl_main(int argc, char **argv);
63 int sysctl_main(int argc, char **argv)
68 int switches_allowed = 1;
75 for (; argv && *argv && **argv; argv++) {
76 if (switches_allowed && **argv == '-') { /* we have a switch */
88 sysctl_preload_file(((argv && *argv
90 DEFAULT_PRELOAD), output);
94 return sysctl_display_all(PROC_PATH, output,
95 ((*argv)[1] == 'a') ? 0 : 1);
100 bb_error_msg(ERR_UNKNOWN_PARAMETER, *argv);
104 switches_allowed = 0;
106 retval = sysctl_write_setting(*argv, output);
108 sysctl_read_setting(*argv, output);
112 } /* end sysctl_main() */
117 * sysctl_preload_file
118 * preload the sysctl's from a conf file
119 * - we parse the file and then reform it (strip out whitespace)
121 #define PRELOAD_BUF 256
123 int sysctl_preload_file(const char *filename, int output)
126 char oneline[PRELOAD_BUF];
127 char buffer[PRELOAD_BUF];
128 char *name, *value, *ptr;
131 if (!filename || ((fp = fopen(filename, "r")) == NULL)) {
132 bb_error_msg_and_die(ERR_PRELOAD_FILE, filename);
135 while (fgets(oneline, sizeof(oneline) - 1, fp)) {
136 oneline[sizeof(oneline) - 1] = '\0';
139 ptr = (char *) oneline;
141 if (*ptr == '#' || *ptr == ';')
147 name = strtok(ptr, "=");
148 if (!name || !*name) {
149 bb_error_msg(WARN_BAD_LINE, filename, lineno);
155 value = strtok(NULL, "\n\r");
156 if (!value || !*value) {
157 bb_error_msg(WARN_BAD_LINE, filename, lineno);
161 while ((*value == ' ' || *value == '\t') && *value != 0)
163 /* safe because sizeof(oneline) == sizeof(buffer) */
164 sprintf(buffer, "%s=%s", name, value);
165 sysctl_write_setting(buffer, output);
169 } /* end sysctl_preload_file() */
173 * Write a single sysctl setting
175 int sysctl_write_setting(const char *setting, int output)
178 const char *name = setting;
181 char *tmpname, *outname, *cptr;
184 if (!name) /* probably dont' want to display this err */
187 if (!(equals = strchr(setting, '='))) {
188 bb_error_msg(ERR_NO_EQUALS, setting);
192 value = equals + sizeof(char); /* point to the value in name=value */
194 if (!*name || !*value || name == equals) {
195 bb_error_msg(ERR_MALFORMED_SETTING, setting);
199 tmpname = xasprintf("%s%.*s", PROC_PATH, (int)(equals - name), name);
200 outname = xstrdup(tmpname + strlen(PROC_PATH));
202 while ((cptr = strchr(tmpname, '.')) != NULL)
205 while ((cptr = strchr(outname, '/')) != NULL)
208 fd = open(tmpname, O_WRONLY | O_CREAT | O_TRUNC, 0666);
212 bb_error_msg(ERR_INVALID_KEY, outname);
215 bb_perror_msg(ERR_PERMISSION_DENIED, outname);
218 bb_error_msg(ERR_UNKNOWN_WRITING, errno, outname);
223 dwrite_str(fd, value);
226 dwrite_str(STDOUT_FILENO, outname);
227 dwrite_str(STDOUT_FILENO, " = ");
229 dwrite_str(STDOUT_FILENO, value);
230 dwrite_str(STDOUT_FILENO, "\n");
237 } /* end sysctl_write_setting() */
241 * Read a sysctl setting
244 int sysctl_read_setting(const char *setting, int output)
247 char *tmpname, *outname, *cptr;
249 const char *name = setting;
252 if (!setting || !*setting)
253 bb_error_msg(ERR_INVALID_KEY, setting);
255 tmpname = concat_path_file(PROC_PATH, name);
256 outname = xstrdup(tmpname + strlen(PROC_PATH));
258 while ((cptr = strchr(tmpname, '.')) != NULL)
260 while ((cptr = strchr(outname, '/')) != NULL)
263 if ((fp = fopen(tmpname, "r")) == NULL) {
266 bb_error_msg(ERR_INVALID_KEY, outname);
269 bb_error_msg(ERR_PERMISSION_DENIED, outname);
272 bb_error_msg(ERR_UNKNOWN_READING, errno, outname);
277 while (fgets(inbuf, sizeof(inbuf) - 1, fp)) {
279 dwrite_str(STDOUT_FILENO, outname);
280 dwrite_str(STDOUT_FILENO, " = ");
282 dwrite_str(STDOUT_FILENO, inbuf);
290 } /* end sysctl_read_setting() */
295 * Display all the sysctl settings
298 int sysctl_display_all(const char *path, int output, int show_table)
311 while ((de = readdir(dp)) != NULL) {
312 tmpdir = concat_subpath_file(path, de->d_name);
315 retval2 = stat(tmpdir, &ts);
317 bb_perror_msg(tmpdir);
319 if (S_ISDIR(ts.st_mode)) {
320 sysctl_display_all(tmpdir, output, show_table);
323 sysctl_read_setting(tmpdir + strlen(PROC_PATH),
333 } /* end sysctl_display_all() */