aboutsummaryrefslogtreecommitdiff
path: root/toys/posix/mkdir.c
blob: 746a918ab66ca3812c29b2aa862a90804c8467f4 (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
/* mkdir.c - Make directories
 *
 * Copyright 2012 Georgi Chorbadzhiyski <georgi@unixsol.org>
 *
 * See http://opengroup.org/onlinepubs/9699919799/utilities/mkdir.html
 *

USE_MKDIR(NEWTOY(mkdir, "<1pm:", TOYFLAG_BIN))

config MKDIR
  bool "mkdir"
  default y
  help
    usage: mkdir [-p] [-m mode] [dirname...]
    Create one or more directories.

    -p	make parent directories as needed.
    -m  set permissions of directory to mode.
*/

#define FOR_mkdir
#include "toys.h"

GLOBALS(
  char *arg_mode;
  mode_t mode;
)

static int do_mkdir(char *dir)
{
  struct stat buf;
  char *s;
  mode_t mode = 0777;

  // mkdir -p one/two/three is not an error if the path already exists,
  // but is if "three" is a file.  The others we dereference and catch
  // not-a-directory along the way, but the last one we must explicitly
  // test for. Might as well do it up front.

  if (!stat(dir, &buf) && !S_ISDIR(buf.st_mode)) {
    errno = EEXIST;
    return 1;
  }

  for (s=dir; ; s++) {
    char save=0;

    // Skip leading / of absolute paths.
    if (s!=dir && *s == '/' && (toys.optflags&FLAG_p)) {
      save = *s;
      *s = 0;
    } else if (*s) continue;

    // Use the mode from the -m option only for the last directory.
    if ((toys.optflags&FLAG_m) && save != '/')
      mode = TT.mode;

    if (mkdir(dir, mode)<0 && ((toys.optflags&~FLAG_p) || errno != EEXIST))
      return 1;

    if (!(*s = save)) break;
  }

  return 0;
}

void mkdir_main(void)
{
  char **s;

  TT.mode = 0777;
  if(toys.optflags&FLAG_m)
    TT.mode = string_to_mode(TT.arg_mode, TT.mode);

  for (s=toys.optargs; *s; s++) {
    if (do_mkdir(*s)) {
      perror_msg("cannot create directory '%s'", *s);
      toys.exitval = 1;
    }
  }
}