source: mainline/uspace/app/bdsh/cmds/modules/mkdir/mkdir.c

Last change on this file was 28a5ebd, checked in by Martin Decky <martin@…>, 4 years ago

Use char32_t instead of wchat_t to represent UTF-32 strings

The intention of the native HelenOS string API has been always to
support Unicode in the UTF-8 and UTF-32 encodings as the sole character
representations and ignore the obsolete mess of older single-byte and
multibyte character encodings. Before C11, the wchar_t type has been
slightly misused for the purpose of the UTF-32 strings. The newer
char32_t type is obviously a much more suitable option. The standard
defines char32_t as uint_least32_t, thus we can take the liberty to fix
it to uint32_t.

To maintain compatilibity with the C Standard, the putwchar(wchar_t)
functions has been replaced by our custom putuchar(char32_t) functions
where appropriate.

  • Property mode set to 100644
File size: 6.0 KB
Line 
1/*
2 * Copyright (c) 2008 Tim Post
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29#include <stdbool.h>
30#include <stdio.h>
31#include <stdlib.h>
32#include <stdint.h>
33#include <dirent.h>
34#include <getopt.h>
35#include <stdarg.h>
36#include <str.h>
37#include <errno.h>
38#include <str_error.h>
39#include <vfs/vfs.h>
40
41#include "config.h"
42#include "errors.h"
43#include "util.h"
44#include "entry.h"
45#include "mkdir.h"
46#include "cmds.h"
47
48#define MKDIR_VERSION "0.0.1"
49
50static const char *cmdname = "mkdir";
51
52static struct option const long_options[] = {
53 { "parents", no_argument, 0, 'p' },
54 { "verbose", no_argument, 0, 'v' },
55 { "mode", required_argument, 0, 'm' },
56 { "help", no_argument, 0, 'h' },
57 { "version", no_argument, 0, 'V' },
58 { "follow", no_argument, 0, 'f' },
59 { 0, 0, 0, 0 }
60};
61
62void help_cmd_mkdir(unsigned int level)
63{
64 if (level == HELP_SHORT) {
65 printf("`%s' creates a new directory\n", cmdname);
66 } else {
67 help_cmd_mkdir(HELP_SHORT);
68 printf(
69 "Usage: %s [options] <path>\n"
70 "Options:\n"
71 " -h, --help A short option summary\n"
72 " -V, --version Print version information and exit\n"
73 " -p, --parents Create needed parents for <path>\n"
74 " -m, --mode Set permissions to [mode] (UNUSED)\n"
75 " -v, --verbose Be extremely noisy about what is happening\n"
76 " -f, --follow Go to the new directory once created\n"
77 "Currently, %s is under development, some options don't work.\n",
78 cmdname, cmdname);
79 }
80
81 return;
82}
83
84/* This is kind of clunky, but effective for now */
85static unsigned int
86create_directory(const char *user_path, bool create_parents)
87{
88 /* Ensure we would always work with absolute and canonified path. */
89 char *path = vfs_absolutize(user_path, NULL);
90 if (path == NULL) {
91 cli_error(CL_ENOMEM, "%s: path too big?", cmdname);
92 return 1;
93 }
94
95 int ret = 0;
96 errno_t rc;
97
98 if (!create_parents) {
99 rc = vfs_link_path(path, KIND_DIRECTORY, NULL);
100 if (rc != EOK) {
101 cli_error(CL_EFAIL, "%s: could not create %s (%s)",
102 cmdname, path, str_error(rc));
103 ret = 1;
104 }
105 } else {
106 /* Create the parent directories as well. */
107 size_t off = 0;
108 while (true) {
109 size_t prev_off = off;
110 char32_t cur_char = str_decode(path, &off, STR_NO_LIMIT);
111 if ((cur_char == 0) || (cur_char == U_SPECIAL)) {
112 break;
113 }
114 if (cur_char != '/') {
115 continue;
116 }
117 if (prev_off == 0) {
118 continue;
119 }
120 /*
121 * If we are here, it means that:
122 * - we found /
123 * - it is not the first / (no need to create root
124 * directory)
125 *
126 * We would now overwrite the / with 0 to terminate the
127 * string (that shall be okay because we are
128 * overwriting at the beginning of UTF sequence).
129 * That would allow us to create the directories
130 * in correct nesting order.
131 *
132 * Notice that we ignore EEXIST errors as some of
133 * the parent directories may already exist.
134 */
135 char slash_char = path[prev_off];
136 path[prev_off] = 0;
137
138 rc = vfs_link_path(path, KIND_DIRECTORY, NULL);
139 if (rc != EOK && rc != EEXIST) {
140 cli_error(CL_EFAIL, "%s: could not create %s (%s)",
141 cmdname, path, str_error(rc));
142 ret = 1;
143 goto leave;
144 }
145
146 path[prev_off] = slash_char;
147 }
148 /* Create the final directory. */
149 rc = vfs_link_path(path, KIND_DIRECTORY, NULL);
150 if (rc != EOK) {
151 cli_error(CL_EFAIL, "%s: could not create %s (%s)",
152 cmdname, path, str_error(rc));
153 ret = 1;
154 }
155 }
156
157leave:
158 free(path);
159 return ret;
160}
161
162int cmd_mkdir(char **argv)
163{
164 unsigned int argc, i, ret = 0;
165 bool create_parents = false, follow = false, verbose = false;
166 int c, opt_ind;
167
168 argc = cli_count_args(argv);
169
170 c = 0;
171 optreset = 1;
172 optind = 0;
173 opt_ind = 0;
174
175 while (c != -1) {
176 c = getopt_long(argc, argv, "pvhVfm:", long_options, &opt_ind);
177 switch (c) {
178 case 'p':
179 create_parents = true;
180 break;
181 case 'v':
182 verbose = true;
183 break;
184 case 'h':
185 help_cmd_mkdir(HELP_LONG);
186 return CMD_SUCCESS;
187 case 'V':
188 printf("%s\n", MKDIR_VERSION);
189 return CMD_SUCCESS;
190 case 'f':
191 follow = true;
192 break;
193 case 'm':
194 printf("%s: [W] Ignoring mode %s\n", cmdname, optarg);
195 break;
196 }
197 }
198
199 argc -= optind;
200
201 if (argc < 1) {
202 printf("%s - incorrect number of arguments. Try `%s --help'\n",
203 cmdname, cmdname);
204 return CMD_FAILURE;
205 }
206
207 for (i = optind; argv[i] != NULL; i++) {
208 if (verbose)
209 printf("%s: creating %s%s\n",
210 cmdname, argv[i],
211 create_parents ? " (and all parents)" : "");
212 ret += create_directory(argv[i], create_parents);
213 }
214
215 if (follow && (argv[optind] != NULL)) {
216 if (vfs_cwd_set(argv[optind]) != EOK)
217 printf("%s: Error switching to directory.", cmdname);
218 }
219
220 if (ret)
221 return CMD_FAILURE;
222 else
223 return CMD_SUCCESS;
224}
Note: See TracBrowser for help on using the repository browser.