view toys/posix/mkdir.c @ 689:c29e69a0e85e

On 32 bit platforms %ld doesn't match uint64_t, so do long long and %lld (rather than deal with verbose PRIu64 nonsense).
author Rob Landley <rob@landley.net>
date Sat, 10 Nov 2012 18:24:14 -0600
parents 7e846e281e38
children 786841fdb1e0
line wrap: on
line source

/* vi: set sw=4 ts=4:
 *
 * 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;
		}
	}
}