aboutsummaryrefslogtreecommitdiff
path: root/toys/posix/mkdir.c
blob: 90329019edc16f31e7c5877dcf8b947c88bc6162 (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
/* vi: set sw=4 ts=4:
 *
 * mkdir.c - Make directories
 *
 * Copyright 2012 Georgi Chorbadzhiyski <georgi@unixsol.org>
 *
 * See http://pubs.opengroup.org/onlinepubs/009695399/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.
*/

#include "toys.h"

DEFINE_GLOBALS(
	long mode;
)

#define TT this.mkdir

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;
		}
	}
}