blob: 2c9a0a9af9103c40f96a6da294444e282d6fc4aa [file] [log] [blame]
Erik Andersene49d5ec2000-02-08 19:58:47 +00001/* vi: set sw=4 ts=4: */
Eric Andersen3d925622005-06-09 10:16:02 +00002
3#include <stdio.h>
4#include <stdlib.h>
5#include <string.h>
6#include <fcntl.h>
7#include <getopt.h>
8#include <time.h>
9#include <unistd.h>
10#include <sys/types.h>
11#include <sys/sysmacros.h> /* major() and minor() */
12#include "busybox.h"
13
14#ifdef CONFIG_FEATURE_MAKEDEVS_LEAF
Eric Andersencc8ed391999-10-05 16:24:54 +000015/*
16 * public domain -- Dave 'Kill a Cop' Cinege <dcinege@psychosis.com>
Eric Andersenc7bda1c2004-03-15 08:29:22 +000017 *
Eric Andersencc8ed391999-10-05 16:24:54 +000018 * makedevs
Eric Andersenc7bda1c2004-03-15 08:29:22 +000019 * Make ranges of device files quickly.
Eric Andersencc8ed391999-10-05 16:24:54 +000020 * known bugs: can't deal with alpha ranges
21 */
Erik Andersene49d5ec2000-02-08 19:58:47 +000022int makedevs_main(int argc, char **argv)
Eric Andersencc8ed391999-10-05 16:24:54 +000023{
Eric Andersen36006022002-06-08 12:44:17 +000024 mode_t mode;
25 char *basedev, *type, *nodname, buf[255];
Eric Andersen4f807a82004-07-26 09:11:12 +000026 int Smajor, Sminor, S, E;
Eric Andersencc8ed391999-10-05 16:24:54 +000027
Erik Andersen5e1189e2000-04-15 16:34:54 +000028 if (argc < 7 || *argv[1]=='-')
Manuel Novoa III cad53642003-03-19 09:13:01 +000029 bb_show_usage();
Erik Andersen5e1189e2000-04-15 16:34:54 +000030
Eric Andersen19732c62002-04-13 14:26:44 +000031 basedev = argv[1];
32 type = argv[2];
Eric Andersencbcdbc42004-07-30 17:48:21 +000033 Smajor = atoi(argv[3]);
34 Sminor = atoi(argv[4]);
Eric Andersen19732c62002-04-13 14:26:44 +000035 S = atoi(argv[5]);
36 E = atoi(argv[6]);
Eric Andersen36006022002-06-08 12:44:17 +000037 nodname = argc == 8 ? basedev : buf;
38
39 mode = 0660;
Eric Andersen19732c62002-04-13 14:26:44 +000040
Eric Andersencc8ed391999-10-05 16:24:54 +000041 switch (type[0]) {
Erik Andersene49d5ec2000-02-08 19:58:47 +000042 case 'c':
Eric Andersen36006022002-06-08 12:44:17 +000043 mode |= S_IFCHR;
Erik Andersene49d5ec2000-02-08 19:58:47 +000044 break;
45 case 'b':
Eric Andersen36006022002-06-08 12:44:17 +000046 mode |= S_IFBLK;
Erik Andersene49d5ec2000-02-08 19:58:47 +000047 break;
48 case 'f':
Eric Andersen36006022002-06-08 12:44:17 +000049 mode |= S_IFIFO;
Erik Andersene49d5ec2000-02-08 19:58:47 +000050 break;
51 default:
Manuel Novoa III cad53642003-03-19 09:13:01 +000052 bb_show_usage();
Erik Andersene49d5ec2000-02-08 19:58:47 +000053 }
Eric Andersencc8ed391999-10-05 16:24:54 +000054
Erik Andersene49d5ec2000-02-08 19:58:47 +000055 while (S <= E) {
Eric Andersen36006022002-06-08 12:44:17 +000056 int sz;
Erik Andersene49d5ec2000-02-08 19:58:47 +000057
Eric Andersen36006022002-06-08 12:44:17 +000058 sz = snprintf(buf, sizeof(buf), "%s%d", basedev, S);
59 if(sz<0 || sz>=sizeof(buf)) /* libc different */
Manuel Novoa III cad53642003-03-19 09:13:01 +000060 bb_error_msg_and_die("%s too large", basedev);
Erik Andersene49d5ec2000-02-08 19:58:47 +000061
Eric Andersen36006022002-06-08 12:44:17 +000062 /* if mode != S_IFCHR and != S_IFBLK third param in mknod() ignored */
Erik Andersene49d5ec2000-02-08 19:58:47 +000063
Eric Andersene71e7602004-07-30 14:36:37 +000064 if (mknod(nodname, mode, makedev(Smajor, Sminor)))
Manuel Novoa III cad53642003-03-19 09:13:01 +000065 bb_error_msg("Failed to create: %s", nodname);
Erik Andersene49d5ec2000-02-08 19:58:47 +000066
Eric Andersen36006022002-06-08 12:44:17 +000067 if (nodname == basedev) /* ex. /dev/hda - to /dev/hda1 ... */
68 nodname = buf;
Erik Andersene49d5ec2000-02-08 19:58:47 +000069 S++;
70 Sminor++;
Eric Andersencc8ed391999-10-05 16:24:54 +000071 }
72
Erik Andersene49d5ec2000-02-08 19:58:47 +000073 return 0;
Eric Andersencc8ed391999-10-05 16:24:54 +000074}
75
Eric Andersen3d925622005-06-09 10:16:02 +000076#elif defined CONFIG_FEATURE_MAKEDEVS_TABLE
77
Eric Andersencc8ed391999-10-05 16:24:54 +000078/*
Eric Andersen3d925622005-06-09 10:16:02 +000079 * This program is free software; you can redistribute it and/or modify
80 * it under the terms of the GNU General Public License version 2 as
81 * published by the Free Software Foundation.
82 *
83 * This program is distributed in the hope that it will be useful,
84 * but WITHOUT ANY WARRANTY; without even the implied warranty of
85 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
86 * GNU Library General Public License for more details.
87 *
88 * You should have received a copy of the GNU General Public License
89 * along with this program; if not, write to the Free Software
90 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
91 *
92 */
Eric Andersencc8ed391999-10-05 16:24:54 +000093
Eric Andersen3d925622005-06-09 10:16:02 +000094static const struct option makedevs_long_options[] = {
95 {"root", 1, NULL, 'r'},
96 {0, 0, 0, 0}
97};
Eric Andersencc8ed391999-10-05 16:24:54 +000098
Eric Andersen3d925622005-06-09 10:16:02 +000099extern int makedevs_main(int argc, char **argv)
100{
101 FILE *table;
102 int opt;
103 char *rootdir = "./";
104 char *line;
105 int ret = EXIT_SUCCESS;
Eric Andersencc8ed391999-10-05 16:24:54 +0000106
Eric Andersen3d925622005-06-09 10:16:02 +0000107 bb_opt_complementaly = "d~r";
108 bb_applet_long_options = makedevs_long_options;
109 opt = bb_getopt_ulflags(argc, argv, "d:r:", &rootdir, &rootdir);
110
111 if (optind + 1 == argc) {
112 table = bb_xfopen(argv[optind], "r");
113 } else {
114 table = stdin;
115 }
116
117 if (chdir(rootdir) == -1) {
118 bb_perror_msg_and_die("Couldnt chdor to %s", rootdir);
119 }
120
121 umask(0);
122
123 while ((line = bb_get_chomped_line_from_file(table))) {
124 char type;
125 unsigned int mode = 0755;
126 unsigned int major = 0;
127 unsigned int minor = 0;
128 unsigned int count = 0;
129 unsigned int increment = 0;
130 unsigned int start = 0;
131 char name[41];
132 char user[41];
133 char group[41];
134 char *full_name;
135 uid_t uid;
136 gid_t gid;
137
138 if ((2 > sscanf(line, "%40s %c %o %40s %40s %u %u %u %u %u", name,
139 &type, &mode, user, group, &major,
140 &minor, &start, &increment, &count)) ||
141 ((major | minor | start | count | increment) > 255)) {
142 bb_error_msg("Ignoring invalid line\n%s\n", line);
143 ret = EXIT_FAILURE;
144 continue;
145 }
146 if (name[0] == '#') {
147 continue;
148 }
149 if (group) {
150 gid = get_ug_id(group, my_getgrnam);
151 } else {
152 gid = getgid();
153 }
154 if (user) {
155 uid = get_ug_id(user, my_getpwnam);
156 } else {
157 uid = getuid();
158 }
159 full_name = concat_path_file(rootdir, name);
160
161 if (type == 'd') {
162 bb_make_directory(full_name, mode | S_IFDIR, 0);
163 if (chown(full_name, uid, gid) == -1) {
164 bb_perror_msg("chown failed for %s", full_name);
165 ret = EXIT_FAILURE;
166 goto loop;
167 }
168 } else {
169 dev_t rdev;
170
171 if (type == 'p') {
172 mode |= S_IFIFO;
173 }
174 else if (type == 'c') {
175 mode |= S_IFCHR;
176 }
177 else if (type == 'b') {
178 mode |= S_IFBLK;
179 } else {
180 bb_error_msg("Unsupported file type %c", type);
181 ret = EXIT_FAILURE;
182 goto loop;
183 }
184
185 if (count > 0) {
186 int i;
187 char *full_name_inc;
188
189 full_name_inc = xmalloc(strlen(full_name) + 4);
190 for (i = start; i < count; i++) {
191 sprintf(full_name_inc, "%s%d", full_name, i);
192 rdev = (major << 8) + minor + (i * increment - start);
193 if (mknod(full_name_inc, mode, rdev) == -1) {
194 bb_perror_msg("Couldnt create node %s", full_name_inc);
195 ret = EXIT_FAILURE;
196 }
197 else if (chown(full_name_inc, uid, gid) == -1) {
198 bb_perror_msg("chown failed for %s", full_name_inc);
199 ret = EXIT_FAILURE;
200 }
201 }
202 free(full_name_inc);
203 } else {
204 rdev = (major << 8) + minor;
205 if (mknod(full_name, mode, rdev) == -1) {
206 bb_perror_msg("Couldnt create node %s", full_name);
207 ret = EXIT_FAILURE;
208 }
209 else if (chown(full_name, uid, gid) == -1) {
210 bb_perror_msg("chown failed for %s", full_name);
211 ret = EXIT_FAILURE;
212 }
213 }
214 }
215loop:
216 free(line);
217 free(full_name);
218 }
219 fclose(table);
220
221 return 0;
Eric Andersencc8ed391999-10-05 16:24:54 +0000222}
Eric Andersen3d925622005-06-09 10:16:02 +0000223#else
224# error makdedevs configuration error, either leaf or table must be selected
225#endif