diff options
| author | kraai <kraai@69ca8d6d-28ef-0310-b511-8ec308f3f277> | 2001-06-21 19:41:37 +0000 |
|---|---|---|
| committer | kraai <kraai@69ca8d6d-28ef-0310-b511-8ec308f3f277> | 2001-06-21 19:41:37 +0000 |
| commit | 5f0f7d3428d4c8cc51fef8fcb4e7bf0ac616dab2 (patch) | |
| tree | 3cdbaddffecc92649215fdc71a43b4e8e86b7ea3 | |
| parent | ce5c25da381038fa6d2640a2f4aa77771ca2ee43 (diff) | |
| download | busybox-w32-5f0f7d3428d4c8cc51fef8fcb4e7bf0ac616dab2.tar.gz busybox-w32-5f0f7d3428d4c8cc51fef8fcb4e7bf0ac616dab2.tar.bz2 busybox-w32-5f0f7d3428d4c8cc51fef8fcb4e7bf0ac616dab2.zip | |
Rewrote mkdir (and touched lots of things in the process).
git-svn-id: svn://busybox.net/trunk/busybox@2873 69ca8d6d-28ef-0310-b511-8ec308f3f277
| -rw-r--r-- | Makefile | 5 | ||||
| -rw-r--r-- | archival/dpkg.c | 2 | ||||
| -rw-r--r-- | archival/tar.c | 7 | ||||
| -rw-r--r-- | coreutils/dirname.c | 14 | ||||
| -rw-r--r-- | coreutils/mkdir.c | 97 | ||||
| -rw-r--r-- | dirname.c | 14 | ||||
| -rw-r--r-- | dpkg.c | 2 | ||||
| -rw-r--r-- | editors/sed.c | 9 | ||||
| -rw-r--r-- | include/libbb.h | 4 | ||||
| -rw-r--r-- | libbb/create_path.c | 71 | ||||
| -rw-r--r-- | libbb/dirname.c | 48 | ||||
| -rw-r--r-- | libbb/libbb.h | 4 | ||||
| -rw-r--r-- | libbb/make_directory.c | 66 | ||||
| -rw-r--r-- | libbb/strdup_substr.c | 32 | ||||
| -rw-r--r-- | mkdir.c | 97 | ||||
| -rw-r--r-- | sed.c | 9 | ||||
| -rw-r--r-- | tar.c | 7 |
17 files changed, 233 insertions, 255 deletions
| @@ -237,7 +237,7 @@ endif | |||
| 237 | LIBBB = libbb | 237 | LIBBB = libbb |
| 238 | LIBBB_LIB = libbb.a | 238 | LIBBB_LIB = libbb.a |
| 239 | LIBBB_CSRC= ask_confirmation.c chomp.c concat_path_file.c copy_file.c \ | 239 | LIBBB_CSRC= ask_confirmation.c chomp.c concat_path_file.c copy_file.c \ |
| 240 | copy_file_chunk.c create_path.c daemon.c device_open.c error_msg.c \ | 240 | copy_file_chunk.c daemon.c device_open.c error_msg.c \ |
| 241 | error_msg_and_die.c fgets_str.c find_mount_point.c find_pid_by_name.c \ | 241 | error_msg_and_die.c fgets_str.c find_mount_point.c find_pid_by_name.c \ |
| 242 | find_root_device.c full_read.c full_write.c get_console.c \ | 242 | find_root_device.c full_read.c full_write.c get_console.c \ |
| 243 | get_last_path_component.c get_line_from_file.c gz_open.c human_readable.c \ | 243 | get_last_path_component.c get_line_from_file.c gz_open.c human_readable.c \ |
| @@ -248,7 +248,8 @@ print_file.c process_escape_sequence.c read_package_field.c recursive_action.c \ | |||
| 248 | safe_read.c safe_strncpy.c syscalls.c syslog_msg_with_name.c time_string.c \ | 248 | safe_read.c safe_strncpy.c syscalls.c syslog_msg_with_name.c time_string.c \ |
| 249 | trim.c unarchive.c unzip.c vdprintf.c verror_msg.c vperror_msg.c wfopen.c xfuncs.c \ | 249 | trim.c unarchive.c unzip.c vdprintf.c verror_msg.c vperror_msg.c wfopen.c xfuncs.c \ |
| 250 | xgetcwd.c xreadlink.c xregcomp.c interface.c remove_file.c last_char_is.c \ | 250 | xgetcwd.c xreadlink.c xregcomp.c interface.c remove_file.c last_char_is.c \ |
| 251 | copyfd.c vherror_msg.c herror_msg.c herror_msg_and_die.c xgethostbyname.c | 251 | copyfd.c vherror_msg.c herror_msg.c herror_msg_and_die.c xgethostbyname.c \ |
| 252 | dirname.c make_directory.c strdup_substr.c | ||
| 252 | LIBBB_OBJS=$(patsubst %.c,$(LIBBB)/%.o, $(LIBBB_CSRC)) | 253 | LIBBB_OBJS=$(patsubst %.c,$(LIBBB)/%.o, $(LIBBB_CSRC)) |
| 253 | LIBBB_CFLAGS = -I$(LIBBB) | 254 | LIBBB_CFLAGS = -I$(LIBBB) |
| 254 | ifneq ($(strip $(BB_SRC_DIR)),) | 255 | ifneq ($(strip $(BB_SRC_DIR)),) |
diff --git a/archival/dpkg.c b/archival/dpkg.c index 4224672ec..7dd46be0c 100644 --- a/archival/dpkg.c +++ b/archival/dpkg.c | |||
| @@ -837,7 +837,7 @@ extern int dpkg_main(int argc, char **argv) | |||
| 837 | optind++; | 837 | optind++; |
| 838 | } | 838 | } |
| 839 | 839 | ||
| 840 | create_path(infodir, S_IRWXU); | 840 | make_directory(infodir, S_IRWXU, FILEUTILS_RECUR); |
| 841 | 841 | ||
| 842 | status = status_read(); | 842 | status = status_read(); |
| 843 | 843 | ||
diff --git a/archival/tar.c b/archival/tar.c index 55fb12c2c..e68194ff7 100644 --- a/archival/tar.c +++ b/archival/tar.c | |||
| @@ -338,7 +338,9 @@ tarExtractRegularFile(TarInfo *header, int extractFlag, int tostdoutFlag) | |||
| 338 | if (extractFlag==TRUE && tostdoutFlag==FALSE) { | 338 | if (extractFlag==TRUE && tostdoutFlag==FALSE) { |
| 339 | /* Create the path to the file, just in case it isn't there... | 339 | /* Create the path to the file, just in case it isn't there... |
| 340 | * This should not screw up path permissions or anything. */ | 340 | * This should not screw up path permissions or anything. */ |
| 341 | create_path(header->name, 0777); | 341 | char *dir = dirname (header->name); |
| 342 | make_directory (dir, -1, FILEUTILS_RECUR); | ||
| 343 | free (dir); | ||
| 342 | if ((outFd=open(header->name, O_CREAT|O_TRUNC|O_WRONLY, | 344 | if ((outFd=open(header->name, O_CREAT|O_TRUNC|O_WRONLY, |
| 343 | header->mode & ~S_IFMT)) < 0) { | 345 | header->mode & ~S_IFMT)) < 0) { |
| 344 | error_msg(io_error, header->name, strerror(errno)); | 346 | error_msg(io_error, header->name, strerror(errno)); |
| @@ -397,8 +399,7 @@ tarExtractDirectory(TarInfo *header, int extractFlag, int tostdoutFlag) | |||
| 397 | if (extractFlag==FALSE || tostdoutFlag==TRUE) | 399 | if (extractFlag==FALSE || tostdoutFlag==TRUE) |
| 398 | return( TRUE); | 400 | return( TRUE); |
| 399 | 401 | ||
| 400 | if (create_path(header->name, header->mode) != TRUE) { | 402 | if (make_directory(header->name, header->mode, FILEUTILS_RECUR) < 0) { |
| 401 | perror_msg("%s: Cannot mkdir", header->name); | ||
| 402 | return( FALSE); | 403 | return( FALSE); |
| 403 | } | 404 | } |
| 404 | /* make the final component, just in case it was | 405 | /* make the final component, just in case it was |
diff --git a/coreutils/dirname.c b/coreutils/dirname.c index 935a8313c..b534e6950 100644 --- a/coreutils/dirname.c +++ b/coreutils/dirname.c | |||
| @@ -30,21 +30,11 @@ | |||
| 30 | 30 | ||
| 31 | extern int dirname_main(int argc, char **argv) | 31 | extern int dirname_main(int argc, char **argv) |
| 32 | { | 32 | { |
| 33 | char* s; | ||
| 34 | |||
| 35 | if ((argc < 2) || (**(argv + 1) == '-')) | 33 | if ((argc < 2) || (**(argv + 1) == '-')) |
| 36 | show_usage(); | 34 | show_usage(); |
| 37 | argv++; | 35 | argv++; |
| 38 | 36 | ||
| 39 | s=*argv+strlen(*argv)-1; | 37 | puts (dirname (argv[0])); |
| 40 | while (s > *argv && *s == '/') { | 38 | |
| 41 | *s-- = '\0'; | ||
| 42 | } | ||
| 43 | s = strrchr(*argv, '/'); | ||
| 44 | if (s != NULL && s == *argv) | ||
| 45 | s[1] = '\0'; | ||
| 46 | else if (s != NULL) | ||
| 47 | *s = '\0'; | ||
| 48 | puts(s ? *argv : "."); | ||
| 49 | return EXIT_SUCCESS; | 39 | return EXIT_SUCCESS; |
| 50 | } | 40 | } |
diff --git a/coreutils/mkdir.c b/coreutils/mkdir.c index d78f57e2b..03c49f098 100644 --- a/coreutils/mkdir.c +++ b/coreutils/mkdir.c | |||
| @@ -2,8 +2,7 @@ | |||
| 2 | /* | 2 | /* |
| 3 | * Mini mkdir implementation for busybox | 3 | * Mini mkdir implementation for busybox |
| 4 | * | 4 | * |
| 5 | * Copyright (C) 1999,2000,2001 by Lineo, inc. | 5 | * Copyright (C) 2001 Matt Kraai <kraai@alumni.carnegiemellon.edu> |
| 6 | * Written by Erik Andersen <andersen@lineo.com>, <andersee@debian.org> | ||
| 7 | * | 6 | * |
| 8 | * This program is free software; you can redistribute it and/or modify | 7 | * This program is free software; you can redistribute it and/or modify |
| 9 | * it under the terms of the GNU General Public License as published by | 8 | * it under the terms of the GNU General Public License as published by |
| @@ -21,79 +20,45 @@ | |||
| 21 | * | 20 | * |
| 22 | */ | 21 | */ |
| 23 | 22 | ||
| 24 | #include <stdio.h> | ||
| 25 | #include <errno.h> | 23 | #include <errno.h> |
| 26 | #include <string.h> | 24 | #include <getopt.h> |
| 25 | #include <sys/stat.h> | ||
| 26 | #include <sys/types.h> | ||
| 27 | #include <fcntl.h> | ||
| 28 | #include <unistd.h> | ||
| 27 | #include <stdlib.h> | 29 | #include <stdlib.h> |
| 28 | #include "busybox.h" | 30 | #include <string.h> |
| 29 | |||
| 30 | |||
| 31 | static int parentFlag = FALSE; | ||
| 32 | static mode_t mode = 0777; | ||
| 33 | 31 | ||
| 32 | #include "busybox.h" | ||
| 34 | 33 | ||
| 35 | extern int mkdir_main(int argc, char **argv) | 34 | extern int mkdir_main (int argc, char **argv) |
| 36 | { | 35 | { |
| 37 | int i = FALSE; | 36 | mode_t mode = -1; |
| 37 | int flags = 0; | ||
| 38 | int status = 0; | ||
| 39 | int i, opt; | ||
| 38 | 40 | ||
| 39 | argc--; | 41 | while ((opt = getopt (argc, argv, "m:p")) != -1) { |
| 40 | argv++; | 42 | switch (opt) { |
| 41 | 43 | case 'm': | |
| 42 | /* Parse any options */ | 44 | mode = 0777; |
| 43 | while (argc > 0 && **argv == '-') { | 45 | if (!parse_mode (optarg, &mode)) |
| 44 | while (i == FALSE && *++(*argv)) { | 46 | error_msg_and_die ("invalid mode `%s'", optarg); |
| 45 | switch (**argv) { | 47 | break; |
| 46 | case 'm': | 48 | case 'p': |
| 47 | if (--argc == 0) | 49 | flags |= FILEUTILS_RECUR; |
| 48 | show_usage(); | 50 | break; |
| 49 | /* Find the specified modes */ | 51 | default: |
| 50 | mode = 0; | 52 | show_usage (); |
| 51 | if (parse_mode(*(++argv), &mode) == FALSE) { | ||
| 52 | error_msg_and_die("Unknown mode: %s", *argv); | ||
| 53 | } | ||
| 54 | /* Set the umask for this process so it doesn't | ||
| 55 | * screw up whatever the user just entered. */ | ||
| 56 | umask(0); | ||
| 57 | i = TRUE; | ||
| 58 | break; | ||
| 59 | case 'p': | ||
| 60 | parentFlag = TRUE; | ||
| 61 | break; | ||
| 62 | default: | ||
| 63 | show_usage(); | ||
| 64 | } | ||
| 65 | } | 53 | } |
| 66 | argc--; | ||
| 67 | argv++; | ||
| 68 | } | 54 | } |
| 69 | 55 | ||
| 70 | if (argc < 1) { | 56 | if (optind == argc) |
| 71 | show_usage(); | 57 | show_usage (); |
| 72 | } | ||
| 73 | 58 | ||
| 74 | while (argc > 0) { | 59 | for (i = optind; i < argc; i++) |
| 75 | int status; | 60 | if (make_directory (argv[i], mode, flags) < 0) |
| 76 | struct stat statBuf; | 61 | status = 1; |
| 77 | char buf[BUFSIZ + 1]; | ||
| 78 | 62 | ||
| 79 | if (strlen(*argv) > BUFSIZ - 1) { | 63 | return status; |
| 80 | error_msg_and_die(name_too_long); | ||
| 81 | } | ||
| 82 | strcpy(buf, *argv); | ||
| 83 | status = stat(buf, &statBuf); | ||
| 84 | if (parentFlag == FALSE && status != -1 && errno != ENOENT) { | ||
| 85 | error_msg_and_die("%s: File exists", buf); | ||
| 86 | } | ||
| 87 | if (parentFlag == TRUE) { | ||
| 88 | strcat(buf, "/"); | ||
| 89 | create_path(buf, mode); | ||
| 90 | } else { | ||
| 91 | if (mkdir(buf, mode) != 0 && parentFlag == FALSE) { | ||
| 92 | perror_msg_and_die(buf); | ||
| 93 | } | ||
| 94 | } | ||
| 95 | argc--; | ||
| 96 | argv++; | ||
| 97 | } | ||
| 98 | return EXIT_SUCCESS; | ||
| 99 | } | 64 | } |
| @@ -30,21 +30,11 @@ | |||
| 30 | 30 | ||
| 31 | extern int dirname_main(int argc, char **argv) | 31 | extern int dirname_main(int argc, char **argv) |
| 32 | { | 32 | { |
| 33 | char* s; | ||
| 34 | |||
| 35 | if ((argc < 2) || (**(argv + 1) == '-')) | 33 | if ((argc < 2) || (**(argv + 1) == '-')) |
| 36 | show_usage(); | 34 | show_usage(); |
| 37 | argv++; | 35 | argv++; |
| 38 | 36 | ||
| 39 | s=*argv+strlen(*argv)-1; | 37 | puts (dirname (argv[0])); |
| 40 | while (s > *argv && *s == '/') { | 38 | |
| 41 | *s-- = '\0'; | ||
| 42 | } | ||
| 43 | s = strrchr(*argv, '/'); | ||
| 44 | if (s != NULL && s == *argv) | ||
| 45 | s[1] = '\0'; | ||
| 46 | else if (s != NULL) | ||
| 47 | *s = '\0'; | ||
| 48 | puts(s ? *argv : "."); | ||
| 49 | return EXIT_SUCCESS; | 39 | return EXIT_SUCCESS; |
| 50 | } | 40 | } |
| @@ -837,7 +837,7 @@ extern int dpkg_main(int argc, char **argv) | |||
| 837 | optind++; | 837 | optind++; |
| 838 | } | 838 | } |
| 839 | 839 | ||
| 840 | create_path(infodir, S_IRWXU); | 840 | make_directory(infodir, S_IRWXU, FILEUTILS_RECUR); |
| 841 | 841 | ||
| 842 | status = status_read(); | 842 | status = status_read(); |
| 843 | 843 | ||
diff --git a/editors/sed.c b/editors/sed.c index e7208b0b5..89b16682f 100644 --- a/editors/sed.c +++ b/editors/sed.c | |||
| @@ -195,15 +195,6 @@ static int get_address(struct sed_cmd *sed_cmd, const char *str, int *linenum, r | |||
| 195 | return idx; | 195 | return idx; |
| 196 | } | 196 | } |
| 197 | 197 | ||
| 198 | static char *strdup_substr(const char *str, int start, int end) | ||
| 199 | { | ||
| 200 | int size = end - start + 1; | ||
| 201 | char *newstr = xmalloc(size); | ||
| 202 | memcpy(newstr, str+start, size-1); | ||
| 203 | newstr[size-1] = '\0'; | ||
| 204 | return newstr; | ||
| 205 | } | ||
| 206 | |||
| 207 | static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr) | 198 | static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr) |
| 208 | { | 199 | { |
| 209 | int oldidx, cflags = REG_NEWLINE; | 200 | int oldidx, cflags = REG_NEWLINE; |
diff --git a/include/libbb.h b/include/libbb.h index e42ca9f2b..c83cb7e7c 100644 --- a/include/libbb.h +++ b/include/libbb.h | |||
| @@ -243,6 +243,10 @@ extern FILE *gz_open(FILE *compressed_file, int *pid); | |||
| 243 | 243 | ||
| 244 | extern struct hostent *xgethostbyname(const char *name); | 244 | extern struct hostent *xgethostbyname(const char *name); |
| 245 | 245 | ||
| 246 | char *dirname (const char *path); | ||
| 247 | char *strdup_substr (const char *s, int start, int end); | ||
| 248 | int make_directory (char *path, mode_t mode, int flags); | ||
| 249 | |||
| 246 | #define CT_AUTO 0 | 250 | #define CT_AUTO 0 |
| 247 | #define CT_UNIX2DOS 1 | 251 | #define CT_UNIX2DOS 1 |
| 248 | #define CT_DOS2UNIX 2 | 252 | #define CT_DOS2UNIX 2 |
diff --git a/libbb/create_path.c b/libbb/create_path.c deleted file mode 100644 index 328afc351..000000000 --- a/libbb/create_path.c +++ /dev/null | |||
| @@ -1,71 +0,0 @@ | |||
| 1 | /* vi: set sw=4 ts=4: */ | ||
| 2 | /* | ||
| 3 | * Utility routines. | ||
| 4 | * | ||
| 5 | * Copyright (C) tons of folks. Tracking down who wrote what | ||
| 6 | * isn't something I'm going to worry about... If you wrote something | ||
| 7 | * here, please feel free to acknowledge your work. | ||
| 8 | * | ||
| 9 | * This program is free software; you can redistribute it and/or modify | ||
| 10 | * it under the terms of the GNU General Public License as published by | ||
| 11 | * the Free Software Foundation; either version 2 of the License, or | ||
| 12 | * (at your option) any later version. | ||
| 13 | * | ||
| 14 | * This program is distributed in the hope that it will be useful, | ||
| 15 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 16 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 17 | * General Public License for more details. | ||
| 18 | * | ||
| 19 | * You should have received a copy of the GNU General Public License | ||
| 20 | * along with this program; if not, write to the Free Software | ||
| 21 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 22 | * | ||
| 23 | * Based in part on code from sash, Copyright (c) 1999 by David I. Bell | ||
| 24 | * Permission has been granted to redistribute this code under the GPL. | ||
| 25 | * | ||
| 26 | */ | ||
| 27 | |||
| 28 | #include <stdio.h> | ||
| 29 | #include <errno.h> | ||
| 30 | #include <string.h> | ||
| 31 | #include "libbb.h" | ||
| 32 | |||
| 33 | /* | ||
| 34 | * Attempt to create the directories along the specified path, except for | ||
| 35 | * the final component. The mode is given for the final directory only, | ||
| 36 | * while all previous ones get default protections. Errors are not reported | ||
| 37 | * here, as failures to restore files can be reported later. | ||
| 38 | */ | ||
| 39 | extern int create_path(const char *name, int mode) | ||
| 40 | { | ||
| 41 | char *cp; | ||
| 42 | char *cpOld; | ||
| 43 | char buf[BUFSIZ + 1]; | ||
| 44 | int retVal = 0; | ||
| 45 | |||
| 46 | strcpy(buf, name); | ||
| 47 | for (cp = buf; *cp == '/'; cp++); | ||
| 48 | cp = strchr(cp, '/'); | ||
| 49 | while (cp) { | ||
| 50 | cpOld = cp; | ||
| 51 | cp = strchr(cp + 1, '/'); | ||
| 52 | *cpOld = '\0'; | ||
| 53 | retVal = mkdir(buf, cp ? 0777 : mode); | ||
| 54 | if (retVal != 0 && errno != EEXIST) { | ||
| 55 | perror_msg("%s", buf); | ||
| 56 | return FALSE; | ||
| 57 | } | ||
| 58 | *cpOld = '/'; | ||
| 59 | } | ||
| 60 | return TRUE; | ||
| 61 | } | ||
| 62 | |||
| 63 | |||
| 64 | /* END CODE */ | ||
| 65 | /* | ||
| 66 | Local Variables: | ||
| 67 | c-file-style: "linux" | ||
| 68 | c-basic-offset: 4 | ||
| 69 | tab-width: 4 | ||
| 70 | End: | ||
| 71 | */ | ||
diff --git a/libbb/dirname.c b/libbb/dirname.c new file mode 100644 index 000000000..2e89fc17a --- /dev/null +++ b/libbb/dirname.c | |||
| @@ -0,0 +1,48 @@ | |||
| 1 | /* vi: set sw=4 ts=4: */ | ||
| 2 | /* | ||
| 3 | * Mini dirname function. | ||
| 4 | * | ||
| 5 | * Copyright (C) 2001 Matt Kraai. | ||
| 6 | * | ||
| 7 | * This program is free software; you can redistribute it and/or modify | ||
| 8 | * it under the terms of the GNU General Public License as published by | ||
| 9 | * the Free Software Foundation; either version 2 of the License, or | ||
| 10 | * (at your option) any later version. | ||
| 11 | * | ||
| 12 | * This program is distributed in the hope that it will be useful, | ||
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 15 | * General Public License for more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU General Public License | ||
| 18 | * along with this program; if not, write to the Free Software | ||
| 19 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 20 | */ | ||
| 21 | |||
| 22 | #include "libbb.h" | ||
| 23 | |||
| 24 | /* Return a string on the heap containing the directory component of PATH. */ | ||
| 25 | |||
| 26 | char *dirname(const char *path) | ||
| 27 | { | ||
| 28 | const char *s; | ||
| 29 | |||
| 30 | /* Go to the end of the string. */ | ||
| 31 | s = path + strlen(path) - 1; | ||
| 32 | |||
| 33 | /* Strip off trailing /s (unless it is also the leading /). */ | ||
| 34 | while (path < s && s[0] == '/') | ||
| 35 | s--; | ||
| 36 | |||
| 37 | /* Strip the last component. */ | ||
| 38 | while (path <= s && s[0] != '/') | ||
| 39 | s--; | ||
| 40 | |||
| 41 | while (path < s && s[0] == '/') | ||
| 42 | s--; | ||
| 43 | |||
| 44 | if (s < path) | ||
| 45 | return xstrdup ("."); | ||
| 46 | else | ||
| 47 | return strdup_substr (path, 0, s - path + 1); | ||
| 48 | } | ||
diff --git a/libbb/libbb.h b/libbb/libbb.h index e42ca9f2b..c83cb7e7c 100644 --- a/libbb/libbb.h +++ b/libbb/libbb.h | |||
| @@ -243,6 +243,10 @@ extern FILE *gz_open(FILE *compressed_file, int *pid); | |||
| 243 | 243 | ||
| 244 | extern struct hostent *xgethostbyname(const char *name); | 244 | extern struct hostent *xgethostbyname(const char *name); |
| 245 | 245 | ||
| 246 | char *dirname (const char *path); | ||
| 247 | char *strdup_substr (const char *s, int start, int end); | ||
| 248 | int make_directory (char *path, mode_t mode, int flags); | ||
| 249 | |||
| 246 | #define CT_AUTO 0 | 250 | #define CT_AUTO 0 |
| 247 | #define CT_UNIX2DOS 1 | 251 | #define CT_UNIX2DOS 1 |
| 248 | #define CT_DOS2UNIX 2 | 252 | #define CT_DOS2UNIX 2 |
diff --git a/libbb/make_directory.c b/libbb/make_directory.c new file mode 100644 index 000000000..e2e28a8cd --- /dev/null +++ b/libbb/make_directory.c | |||
| @@ -0,0 +1,66 @@ | |||
| 1 | /* vi: set sw=4 ts=4: */ | ||
| 2 | /* | ||
| 3 | * Mini make_directory implementation for busybox | ||
| 4 | * | ||
| 5 | * Copyright (C) 2001 Matt Kraai. | ||
| 6 | * | ||
| 7 | * This program is free software; you can redistribute it and/or modify | ||
| 8 | * it under the terms of the GNU General Public License as published by | ||
| 9 | * the Free Software Foundation; either version 2 of the License, or | ||
| 10 | * (at your option) any later version. | ||
| 11 | * | ||
| 12 | * This program is distributed in the hope that it will be useful, | ||
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 15 | * General Public License for more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU General Public License | ||
| 18 | * along with this program; if not, write to the Free Software | ||
| 19 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 20 | * | ||
| 21 | */ | ||
| 22 | |||
| 23 | #include <errno.h> | ||
| 24 | #include <fcntl.h> | ||
| 25 | #include <sys/stat.h> | ||
| 26 | #include <sys/types.h> | ||
| 27 | #include <unistd.h> | ||
| 28 | |||
| 29 | #include "libbb.h" | ||
| 30 | |||
| 31 | /* Create the directory PATH with mode MODE, or the default if MODE is -1. | ||
| 32 | * Also create parent directories as necessary if flags contains | ||
| 33 | * FILEUTILS_RECUR. */ | ||
| 34 | |||
| 35 | int make_directory (char *path, mode_t mode, int flags) | ||
| 36 | { | ||
| 37 | if (!(flags & FILEUTILS_RECUR)) { | ||
| 38 | if (mkdir (path, 0777) < 0) { | ||
| 39 | perror_msg ("Cannot create directory `%s'", path); | ||
| 40 | return -1; | ||
| 41 | } | ||
| 42 | |||
| 43 | if (mode != -1 && chmod (path, mode) < 0) { | ||
| 44 | perror_msg ("Cannot set permissions of directory `%s'", path); | ||
| 45 | return -1; | ||
| 46 | } | ||
| 47 | } else { | ||
| 48 | struct stat st; | ||
| 49 | |||
| 50 | if (stat (path, &st) < 0 && errno == ENOENT) { | ||
| 51 | char *parent = dirname (path); | ||
| 52 | mode_t mask = umask (0); | ||
| 53 | umask (mask); | ||
| 54 | |||
| 55 | if (make_directory (parent, (0777 & ~mask) | 0300, | ||
| 56 | FILEUTILS_RECUR) < 0) | ||
| 57 | return -1; | ||
| 58 | free (parent); | ||
| 59 | |||
| 60 | if (make_directory (path, mode, 0) < 0) | ||
| 61 | return -1; | ||
| 62 | } | ||
| 63 | } | ||
| 64 | |||
| 65 | return 0; | ||
| 66 | } | ||
diff --git a/libbb/strdup_substr.c b/libbb/strdup_substr.c new file mode 100644 index 000000000..4542d5fbe --- /dev/null +++ b/libbb/strdup_substr.c | |||
| @@ -0,0 +1,32 @@ | |||
| 1 | /* vi: set sw=4 ts=4: */ | ||
| 2 | /* | ||
| 3 | * Mini strdup_substr function. | ||
| 4 | * | ||
| 5 | * Copyright (C) 2001 Mark Whitley. | ||
| 6 | * | ||
| 7 | * This program is free software; you can redistribute it and/or modify | ||
| 8 | * it under the terms of the GNU General Public License as published by | ||
| 9 | * the Free Software Foundation; either version 2 of the License, or | ||
| 10 | * (at your option) any later version. | ||
| 11 | * | ||
| 12 | * This program is distributed in the hope that it will be useful, | ||
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 15 | * General Public License for more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU General Public License | ||
| 18 | * along with this program; if not, write to the Free Software | ||
| 19 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
| 20 | */ | ||
| 21 | |||
| 22 | /* Return a substring of STR, starting at index START and ending at END, | ||
| 23 | * allocated on the heap. */ | ||
| 24 | |||
| 25 | char *strdup_substr(const char *str, int start, int end) | ||
| 26 | { | ||
| 27 | int size = end - start + 1; | ||
| 28 | char *newstr = xmalloc(size); | ||
| 29 | memcpy(newstr, str+start, size-1); | ||
| 30 | newstr[size-1] = '\0'; | ||
| 31 | return newstr; | ||
| 32 | } | ||
| @@ -2,8 +2,7 @@ | |||
| 2 | /* | 2 | /* |
| 3 | * Mini mkdir implementation for busybox | 3 | * Mini mkdir implementation for busybox |
| 4 | * | 4 | * |
| 5 | * Copyright (C) 1999,2000,2001 by Lineo, inc. | 5 | * Copyright (C) 2001 Matt Kraai <kraai@alumni.carnegiemellon.edu> |
| 6 | * Written by Erik Andersen <andersen@lineo.com>, <andersee@debian.org> | ||
| 7 | * | 6 | * |
| 8 | * This program is free software; you can redistribute it and/or modify | 7 | * This program is free software; you can redistribute it and/or modify |
| 9 | * it under the terms of the GNU General Public License as published by | 8 | * it under the terms of the GNU General Public License as published by |
| @@ -21,79 +20,45 @@ | |||
| 21 | * | 20 | * |
| 22 | */ | 21 | */ |
| 23 | 22 | ||
| 24 | #include <stdio.h> | ||
| 25 | #include <errno.h> | 23 | #include <errno.h> |
| 26 | #include <string.h> | 24 | #include <getopt.h> |
| 25 | #include <sys/stat.h> | ||
| 26 | #include <sys/types.h> | ||
| 27 | #include <fcntl.h> | ||
| 28 | #include <unistd.h> | ||
| 27 | #include <stdlib.h> | 29 | #include <stdlib.h> |
| 28 | #include "busybox.h" | 30 | #include <string.h> |
| 29 | |||
| 30 | |||
| 31 | static int parentFlag = FALSE; | ||
| 32 | static mode_t mode = 0777; | ||
| 33 | 31 | ||
| 32 | #include "busybox.h" | ||
| 34 | 33 | ||
| 35 | extern int mkdir_main(int argc, char **argv) | 34 | extern int mkdir_main (int argc, char **argv) |
| 36 | { | 35 | { |
| 37 | int i = FALSE; | 36 | mode_t mode = -1; |
| 37 | int flags = 0; | ||
| 38 | int status = 0; | ||
| 39 | int i, opt; | ||
| 38 | 40 | ||
| 39 | argc--; | 41 | while ((opt = getopt (argc, argv, "m:p")) != -1) { |
| 40 | argv++; | 42 | switch (opt) { |
| 41 | 43 | case 'm': | |
| 42 | /* Parse any options */ | 44 | mode = 0777; |
| 43 | while (argc > 0 && **argv == '-') { | 45 | if (!parse_mode (optarg, &mode)) |
| 44 | while (i == FALSE && *++(*argv)) { | 46 | error_msg_and_die ("invalid mode `%s'", optarg); |
| 45 | switch (**argv) { | 47 | break; |
| 46 | case 'm': | 48 | case 'p': |
| 47 | if (--argc == 0) | 49 | flags |= FILEUTILS_RECUR; |
| 48 | show_usage(); | 50 | break; |
| 49 | /* Find the specified modes */ | 51 | default: |
| 50 | mode = 0; | 52 | show_usage (); |
| 51 | if (parse_mode(*(++argv), &mode) == FALSE) { | ||
| 52 | error_msg_and_die("Unknown mode: %s", *argv); | ||
| 53 | } | ||
| 54 | /* Set the umask for this process so it doesn't | ||
| 55 | * screw up whatever the user just entered. */ | ||
| 56 | umask(0); | ||
| 57 | i = TRUE; | ||
| 58 | break; | ||
| 59 | case 'p': | ||
| 60 | parentFlag = TRUE; | ||
| 61 | break; | ||
| 62 | default: | ||
| 63 | show_usage(); | ||
| 64 | } | ||
| 65 | } | 53 | } |
| 66 | argc--; | ||
| 67 | argv++; | ||
| 68 | } | 54 | } |
| 69 | 55 | ||
| 70 | if (argc < 1) { | 56 | if (optind == argc) |
| 71 | show_usage(); | 57 | show_usage (); |
| 72 | } | ||
| 73 | 58 | ||
| 74 | while (argc > 0) { | 59 | for (i = optind; i < argc; i++) |
| 75 | int status; | 60 | if (make_directory (argv[i], mode, flags) < 0) |
| 76 | struct stat statBuf; | 61 | status = 1; |
| 77 | char buf[BUFSIZ + 1]; | ||
| 78 | 62 | ||
| 79 | if (strlen(*argv) > BUFSIZ - 1) { | 63 | return status; |
| 80 | error_msg_and_die(name_too_long); | ||
| 81 | } | ||
| 82 | strcpy(buf, *argv); | ||
| 83 | status = stat(buf, &statBuf); | ||
| 84 | if (parentFlag == FALSE && status != -1 && errno != ENOENT) { | ||
| 85 | error_msg_and_die("%s: File exists", buf); | ||
| 86 | } | ||
| 87 | if (parentFlag == TRUE) { | ||
| 88 | strcat(buf, "/"); | ||
| 89 | create_path(buf, mode); | ||
| 90 | } else { | ||
| 91 | if (mkdir(buf, mode) != 0 && parentFlag == FALSE) { | ||
| 92 | perror_msg_and_die(buf); | ||
| 93 | } | ||
| 94 | } | ||
| 95 | argc--; | ||
| 96 | argv++; | ||
| 97 | } | ||
| 98 | return EXIT_SUCCESS; | ||
| 99 | } | 64 | } |
| @@ -195,15 +195,6 @@ static int get_address(struct sed_cmd *sed_cmd, const char *str, int *linenum, r | |||
| 195 | return idx; | 195 | return idx; |
| 196 | } | 196 | } |
| 197 | 197 | ||
| 198 | static char *strdup_substr(const char *str, int start, int end) | ||
| 199 | { | ||
| 200 | int size = end - start + 1; | ||
| 201 | char *newstr = xmalloc(size); | ||
| 202 | memcpy(newstr, str+start, size-1); | ||
| 203 | newstr[size-1] = '\0'; | ||
| 204 | return newstr; | ||
| 205 | } | ||
| 206 | |||
| 207 | static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr) | 198 | static int parse_subst_cmd(struct sed_cmd *sed_cmd, const char *substr) |
| 208 | { | 199 | { |
| 209 | int oldidx, cflags = REG_NEWLINE; | 200 | int oldidx, cflags = REG_NEWLINE; |
| @@ -338,7 +338,9 @@ tarExtractRegularFile(TarInfo *header, int extractFlag, int tostdoutFlag) | |||
| 338 | if (extractFlag==TRUE && tostdoutFlag==FALSE) { | 338 | if (extractFlag==TRUE && tostdoutFlag==FALSE) { |
| 339 | /* Create the path to the file, just in case it isn't there... | 339 | /* Create the path to the file, just in case it isn't there... |
| 340 | * This should not screw up path permissions or anything. */ | 340 | * This should not screw up path permissions or anything. */ |
| 341 | create_path(header->name, 0777); | 341 | char *dir = dirname (header->name); |
| 342 | make_directory (dir, -1, FILEUTILS_RECUR); | ||
| 343 | free (dir); | ||
| 342 | if ((outFd=open(header->name, O_CREAT|O_TRUNC|O_WRONLY, | 344 | if ((outFd=open(header->name, O_CREAT|O_TRUNC|O_WRONLY, |
| 343 | header->mode & ~S_IFMT)) < 0) { | 345 | header->mode & ~S_IFMT)) < 0) { |
| 344 | error_msg(io_error, header->name, strerror(errno)); | 346 | error_msg(io_error, header->name, strerror(errno)); |
| @@ -397,8 +399,7 @@ tarExtractDirectory(TarInfo *header, int extractFlag, int tostdoutFlag) | |||
| 397 | if (extractFlag==FALSE || tostdoutFlag==TRUE) | 399 | if (extractFlag==FALSE || tostdoutFlag==TRUE) |
| 398 | return( TRUE); | 400 | return( TRUE); |
| 399 | 401 | ||
| 400 | if (create_path(header->name, header->mode) != TRUE) { | 402 | if (make_directory(header->name, header->mode, FILEUTILS_RECUR) < 0) { |
| 401 | perror_msg("%s: Cannot mkdir", header->name); | ||
| 402 | return( FALSE); | 403 | return( FALSE); |
| 403 | } | 404 | } |
| 404 | /* make the final component, just in case it was | 405 | /* make the final component, just in case it was |
