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

USE_MKDIR(NEWTOY(mkdir, "<1p", TOYFLAG_BIN))

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

    -p	make parent directories as needed.
*/

#define FOR_mkdir
#include "toys.h"

GLOBALS(
  long mode;
)

static int do_mkdir(char *dir)
{
  struct stat buf;
  char *s;

  // 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) {
      save = *s;
      *s = 0;
    } else if (*s) continue;

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

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

  return 0;
}

void mkdir_main(void)
{
  char **s;

  TT.mode = 0777;

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