Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 :
4 : POSIX NTVFS backend - filename resolution
5 :
6 : Copyright (C) Andrew Tridgell 2004
7 :
8 : This program is free software; you can redistribute it and/or modify
9 : it under the terms of the GNU General Public License as published by
10 : the Free Software Foundation; either version 3 of the License, or
11 : (at your option) any later version.
12 :
13 : This program is distributed in the hope that it will be useful,
14 : but WITHOUT ANY WARRANTY; without even the implied warranty of
15 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 : GNU General Public License for more details.
17 :
18 : You should have received a copy of the GNU General Public License
19 : along with this program. If not, see <http://www.gnu.org/licenses/>.
20 : */
21 :
22 : /*
23 : this is the core code for converting a filename from the format as
24 : given by a client to a posix filename, including any case-matching
25 : required, and checks for legal characters
26 : */
27 :
28 :
29 : #include "includes.h"
30 : #include "vfs_posix.h"
31 : #include "system/dir.h"
32 : #include "param/param.h"
33 :
34 : /**
35 : compare two filename components. This is where the name mangling hook will go
36 : */
37 38106910 : static int component_compare(struct pvfs_state *pvfs, const char *comp, const char *name)
38 : {
39 : int ret;
40 :
41 38106910 : ret = strcasecmp_m(comp, name);
42 :
43 38106910 : if (ret != 0) {
44 38106593 : char *shortname = pvfs_short_name_component(pvfs, name);
45 38106593 : if (shortname) {
46 359777 : ret = strcasecmp_m(comp, shortname);
47 359777 : talloc_free(shortname);
48 : }
49 : }
50 :
51 38106910 : return ret;
52 : }
53 :
54 : /*
55 : search for a filename in a case insensitive fashion
56 :
57 : TODO: add a cache for previously resolved case-insensitive names
58 : TODO: add mangled name support
59 : */
60 172283 : static NTSTATUS pvfs_case_search(struct pvfs_state *pvfs,
61 : struct pvfs_filename *name,
62 : unsigned int flags)
63 : {
64 : /* break into a series of components */
65 : size_t num_components;
66 : char **components;
67 : char *p, *partial_name;
68 : size_t i;
69 :
70 : /* break up the full name info pathname components */
71 172283 : num_components=2;
72 172283 : p = name->full_name + strlen(pvfs->base_directory) + 1;
73 :
74 3414214 : for (;*p;p++) {
75 3241931 : if (*p == '/') {
76 290670 : num_components++;
77 : }
78 : }
79 :
80 172283 : components = talloc_array(name, char *, num_components);
81 172283 : p = name->full_name + strlen(pvfs->base_directory);
82 172283 : *p++ = 0;
83 :
84 172283 : components[0] = name->full_name;
85 :
86 635236 : for (i=1;i<num_components;i++) {
87 462953 : components[i] = p;
88 462953 : p = strchr(p, '/');
89 462953 : if (p) *p++ = 0;
90 462953 : if (pvfs_is_reserved_name(pvfs, components[i])) {
91 0 : return NT_STATUS_ACCESS_DENIED;
92 : }
93 : }
94 :
95 172283 : partial_name = talloc_strdup(name, components[0]);
96 172283 : if (!partial_name) {
97 0 : return NT_STATUS_NO_MEMORY;
98 : }
99 :
100 : /* for each component, check if it exists as-is, and if not then
101 : do a directory scan */
102 635100 : for (i=1;i<num_components;i++) {
103 : char *test_name;
104 : DIR *dir;
105 : struct dirent *de;
106 : char *long_component;
107 :
108 : /* possibly remap from the short name cache */
109 462868 : long_component = pvfs_mangled_lookup(pvfs, name, components[i]);
110 462868 : if (long_component) {
111 17 : components[i] = long_component;
112 : }
113 :
114 462868 : test_name = talloc_asprintf(name, "%s/%s", partial_name, components[i]);
115 462868 : if (!test_name) {
116 0 : return NT_STATUS_NO_MEMORY;
117 : }
118 :
119 : /* check if this component exists as-is */
120 462868 : if (stat(test_name, &name->st) == 0) {
121 290475 : if (i<num_components-1 && !S_ISDIR(name->st.st_mode)) {
122 5 : return NT_STATUS_OBJECT_PATH_NOT_FOUND;
123 : }
124 290470 : talloc_free(partial_name);
125 290470 : partial_name = test_name;
126 290470 : if (i == num_components - 1) {
127 158 : name->exists = true;
128 : }
129 290470 : continue;
130 : }
131 :
132 : /* the filesystem might be case insensitive, in which
133 : case a search is pointless unless the name is
134 : mangled */
135 172393 : if ((pvfs->flags & PVFS_FLAG_CI_FILESYSTEM) &&
136 0 : !pvfs_is_mangled_component(pvfs, components[i])) {
137 0 : if (i < num_components-1) {
138 0 : return NT_STATUS_OBJECT_PATH_NOT_FOUND;
139 : }
140 0 : partial_name = test_name;
141 0 : continue;
142 : }
143 :
144 172393 : dir = opendir(partial_name);
145 172393 : if (!dir) {
146 0 : return pvfs_map_errno(pvfs, errno);
147 : }
148 :
149 38451371 : while ((de = readdir(dir))) {
150 38106910 : if (component_compare(pvfs, components[i], de->d_name) == 0) {
151 317 : break;
152 : }
153 : }
154 :
155 172393 : if (!de) {
156 172076 : if (i < num_components-1) {
157 46 : closedir(dir);
158 46 : return NT_STATUS_OBJECT_PATH_NOT_FOUND;
159 : }
160 : } else {
161 317 : components[i] = talloc_strdup(name, de->d_name);
162 : }
163 172347 : test_name = talloc_asprintf(name, "%s/%s", partial_name, components[i]);
164 172347 : talloc_free(partial_name);
165 172347 : partial_name = test_name;
166 :
167 172347 : closedir(dir);
168 : }
169 :
170 172232 : if (!name->exists) {
171 172074 : if (stat(partial_name, &name->st) == 0) {
172 44 : name->exists = true;
173 : }
174 : }
175 :
176 172232 : talloc_free(name->full_name);
177 172232 : name->full_name = partial_name;
178 :
179 172232 : if (name->exists) {
180 202 : return pvfs_fill_dos_info(pvfs, name, flags, -1);
181 : }
182 :
183 172030 : return NT_STATUS_OK;
184 : }
185 :
186 : /*
187 : parse a alternate data stream name
188 : */
189 419 : static NTSTATUS parse_stream_name(struct pvfs_filename *name,
190 : const char *s)
191 : {
192 : char *p, *stream_name;
193 419 : if (s[1] == '\0') {
194 1 : return NT_STATUS_OBJECT_NAME_INVALID;
195 : }
196 418 : name->stream_name = stream_name = talloc_strdup(name, s+1);
197 418 : if (name->stream_name == NULL) {
198 0 : return NT_STATUS_NO_MEMORY;
199 : }
200 :
201 418 : p = stream_name;
202 :
203 5525 : while (*p) {
204 : size_t c_size;
205 4710 : codepoint_t c = next_codepoint(p, &c_size);
206 :
207 4710 : switch (c) {
208 4 : case '/':
209 : case '\\':
210 25 : return NT_STATUS_OBJECT_NAME_INVALID;
211 340 : case ':':
212 340 : *p= 0;
213 340 : p++;
214 340 : if (*p == '\0') {
215 6 : return NT_STATUS_OBJECT_NAME_INVALID;
216 : }
217 334 : if (strcasecmp_m(p, "$DATA") != 0) {
218 11 : if (strchr_m(p, ':')) {
219 2 : return NT_STATUS_OBJECT_NAME_INVALID;
220 : }
221 9 : return NT_STATUS_INVALID_PARAMETER;
222 : }
223 323 : c_size = 0;
224 323 : p--;
225 323 : break;
226 : }
227 :
228 4689 : p += c_size;
229 : }
230 :
231 397 : if (strcmp(name->stream_name, "") == 0) {
232 : /*
233 : * we don't set stream_name to NULL, here
234 : * as this would be wrong for directories
235 : *
236 : * pvfs_fill_dos_info() will set it to NULL
237 : * if it's not a directory.
238 : */
239 11 : name->stream_id = 0;
240 : } else {
241 386 : name->stream_id = pvfs_name_hash(name->stream_name,
242 386 : strlen(name->stream_name));
243 : }
244 :
245 397 : return NT_STATUS_OK;
246 : }
247 :
248 :
249 : /*
250 : convert a CIFS pathname to a unix pathname. Note that this does NOT
251 : take into account case insensitivity, and in fact does not access
252 : the filesystem at all. It is merely a reformatting and charset
253 : checking routine.
254 :
255 : errors are returned if the filename is illegal given the flags
256 : */
257 334843 : static NTSTATUS pvfs_unix_path(struct pvfs_state *pvfs, const char *cifs_name,
258 : unsigned int flags, struct pvfs_filename *name)
259 : {
260 : char *ret, *p, *p_start;
261 : NTSTATUS status;
262 :
263 334843 : name->original_name = talloc_strdup(name, cifs_name);
264 :
265 : /* remove any :$DATA */
266 334843 : p = strrchr(name->original_name, ':');
267 334843 : if (p && strcasecmp_m(p, ":$DATA") == 0) {
268 332 : if (p > name->original_name && p[-1] == ':') {
269 11 : p--;
270 : }
271 332 : *p = 0;
272 : }
273 :
274 334843 : name->stream_name = NULL;
275 334843 : name->stream_id = 0;
276 334843 : name->has_wildcard = false;
277 :
278 796064 : while (*cifs_name == '\\') {
279 126424 : cifs_name++;
280 : }
281 :
282 334843 : if (*cifs_name == 0) {
283 5340 : name->full_name = talloc_asprintf(name, "%s/.", pvfs->base_directory);
284 5340 : if (name->full_name == NULL) {
285 0 : return NT_STATUS_NO_MEMORY;
286 : }
287 5340 : return NT_STATUS_OK;
288 : }
289 :
290 329503 : ret = talloc_asprintf(name, "%s/%s", pvfs->base_directory, cifs_name);
291 329503 : if (ret == NULL) {
292 0 : return NT_STATUS_NO_MEMORY;
293 : }
294 :
295 329503 : p = ret + strlen(pvfs->base_directory) + 1;
296 :
297 : /* now do an in-place conversion of '\' to '/', checking
298 : for legal characters */
299 329503 : p_start = p;
300 :
301 6670695 : while (*p) {
302 : size_t c_size;
303 6012186 : codepoint_t c = next_codepoint(p, &c_size);
304 :
305 6012186 : if (c <= 0x1F) {
306 697 : return NT_STATUS_OBJECT_NAME_INVALID;
307 : }
308 :
309 6011938 : switch (c) {
310 473557 : case '\\':
311 473557 : if (name->has_wildcard) {
312 : /* wildcards are only allowed in the last part
313 : of a name */
314 0 : return NT_STATUS_OBJECT_NAME_INVALID;
315 : }
316 473557 : if (p > p_start && (p[1] == '\\' || p[1] == '\0')) {
317 : /* see if it is definately a "\\" or
318 : * a trailing "\". If it is then fail here,
319 : * and let the next layer up try again after
320 : * pvfs_reduce_name() if it wants to. This is
321 : * much more efficient on average than always
322 : * scanning for these separately
323 : */
324 61 : return NT_STATUS_OBJECT_PATH_SYNTAX_BAD;
325 : } else {
326 473496 : *p = '/';
327 : }
328 473510 : break;
329 428 : case ':':
330 428 : if (!(flags & PVFS_RESOLVE_STREAMS)) {
331 9 : return NT_STATUS_OBJECT_NAME_INVALID;
332 : }
333 419 : if (name->has_wildcard) {
334 0 : return NT_STATUS_OBJECT_NAME_INVALID;
335 : }
336 419 : status = parse_stream_name(name, p);
337 419 : if (!NT_STATUS_IS_OK(status)) {
338 22 : return status;
339 : }
340 397 : *p-- = 0;
341 397 : break;
342 6418 : case '*':
343 : case '>':
344 : case '<':
345 : case '?':
346 : case '"':
347 6418 : if (!(flags & PVFS_RESOLVE_WILDCARD)) {
348 16 : return NT_STATUS_OBJECT_NAME_INVALID;
349 : }
350 6402 : name->has_wildcard = true;
351 6402 : break;
352 9 : case '/':
353 : case '|':
354 9 : return NT_STATUS_OBJECT_NAME_INVALID;
355 33067 : case '.':
356 : /* see if it is definately a .. or
357 : . component. If it is then fail here, and
358 : let the next layer up try again after
359 : pvfs_reduce_name() if it wants to. This is
360 : much more efficient on average than always
361 : scanning for these separately */
362 33146 : if (p[1] == '.' &&
363 133 : (p[2] == 0 || p[2] == '\\') &&
364 26 : (p == p_start || p[-1] == '/')) {
365 27 : return NT_STATUS_OBJECT_PATH_SYNTAX_BAD;
366 : }
367 33040 : if ((p[1] == 0 || p[1] == '\\') &&
368 90 : (p == p_start || p[-1] == '/')) {
369 60 : return NT_STATUS_OBJECT_PATH_SYNTAX_BAD;
370 : }
371 32980 : break;
372 : }
373 :
374 6011734 : p += c_size;
375 : }
376 :
377 329051 : name->full_name = ret;
378 :
379 329051 : return NT_STATUS_OK;
380 : }
381 :
382 :
383 : /*
384 : reduce a name that contains .. components or repeated \ separators
385 : return NULL if it can't be reduced
386 : */
387 148 : static NTSTATUS pvfs_reduce_name(TALLOC_CTX *mem_ctx,
388 : const char **fname, unsigned int flags)
389 : {
390 : codepoint_t c;
391 : size_t c_size, len;
392 : size_t i, num_components, err_count;
393 : char **components;
394 : char *p, *s, *ret;
395 :
396 148 : s = talloc_strdup(mem_ctx, *fname);
397 148 : if (s == NULL) return NT_STATUS_NO_MEMORY;
398 :
399 2652 : for (num_components=1, p=s; *p; p += c_size) {
400 2504 : c = next_codepoint(p, &c_size);
401 2504 : if (c == '\\') num_components++;
402 : }
403 :
404 148 : components = talloc_array(s, char *, num_components+1);
405 148 : if (components == NULL) {
406 0 : talloc_free(s);
407 0 : return NT_STATUS_NO_MEMORY;
408 : }
409 :
410 148 : components[0] = s;
411 2652 : for (i=0, p=s; *p; p += c_size) {
412 2504 : c = next_codepoint(p, &c_size);
413 2504 : if (c == '\\') {
414 471 : *p = 0;
415 471 : components[++i] = p+1;
416 : }
417 : }
418 148 : components[i+1] = NULL;
419 :
420 : /*
421 : rather bizarre!
422 :
423 : '.' components are not allowed, but the rules for what error
424 : code to give don't seem to make sense. This is a close
425 : approximation.
426 : */
427 767 : for (err_count=i=0;components[i];i++) {
428 619 : if (strcmp(components[i], "") == 0) {
429 257 : continue;
430 : }
431 362 : if (ISDOT(components[i]) || err_count) {
432 134 : err_count++;
433 : }
434 : }
435 148 : if (err_count > 0) {
436 60 : if (flags & PVFS_RESOLVE_WILDCARD) err_count--;
437 :
438 60 : if (err_count==1) {
439 27 : return NT_STATUS_OBJECT_NAME_INVALID;
440 : } else {
441 33 : return NT_STATUS_OBJECT_PATH_NOT_FOUND;
442 : }
443 : }
444 :
445 : /* remove any null components */
446 415 : for (i=0;components[i];i++) {
447 339 : if (strcmp(components[i], "") == 0) {
448 141 : memmove(&components[i], &components[i+1],
449 141 : sizeof(char *)*(num_components-i));
450 141 : i--;
451 141 : continue;
452 : }
453 198 : if (ISDOTDOT(components[i])) {
454 36 : if (i < 1) return NT_STATUS_OBJECT_PATH_SYNTAX_BAD;
455 24 : memmove(&components[i-1], &components[i+1],
456 24 : sizeof(char *)*(num_components-i));
457 24 : i -= 2;
458 24 : continue;
459 : }
460 : }
461 :
462 76 : if (components[0] == NULL) {
463 3 : talloc_free(s);
464 3 : *fname = talloc_strdup(mem_ctx, "\\");
465 3 : return NT_STATUS_OK;
466 : }
467 :
468 211 : for (len=i=0;components[i];i++) {
469 138 : len += strlen(components[i]) + 1;
470 : }
471 :
472 : /* rebuild the name */
473 73 : ret = talloc_array(mem_ctx, char, len+1);
474 73 : if (ret == NULL) {
475 0 : talloc_free(s);
476 0 : return NT_STATUS_NO_MEMORY;
477 : }
478 :
479 211 : for (len=0,i=0;components[i];i++) {
480 138 : size_t len1 = strlen(components[i]);
481 138 : ret[len] = '\\';
482 138 : memcpy(ret+len+1, components[i], len1);
483 138 : len += len1 + 1;
484 : }
485 73 : ret[len] = 0;
486 :
487 73 : talloc_set_name_const(ret, ret);
488 :
489 73 : talloc_free(s);
490 :
491 73 : *fname = ret;
492 :
493 73 : return NT_STATUS_OK;
494 : }
495 :
496 :
497 : /*
498 : resolve a name from relative client format to a struct pvfs_filename
499 : the memory for the filename is made as a talloc child of 'name'
500 :
501 : flags include:
502 : PVFS_RESOLVE_NO_WILDCARD = wildcards are considered illegal characters
503 : PVFS_RESOLVE_STREAMS = stream names are allowed
504 :
505 : TODO: ../ collapsing, and outside share checking
506 : */
507 334771 : NTSTATUS pvfs_resolve_name(struct pvfs_state *pvfs,
508 : struct ntvfs_request *req,
509 : const char *cifs_name,
510 : unsigned int flags, struct pvfs_filename **name)
511 : {
512 : NTSTATUS status;
513 :
514 334771 : *name = talloc(req, struct pvfs_filename);
515 334771 : if (*name == NULL) {
516 0 : return NT_STATUS_NO_MEMORY;
517 : }
518 :
519 334771 : (*name)->exists = false;
520 334771 : (*name)->stream_exists = false;
521 334771 : (*name)->allow_override = false;
522 :
523 334771 : if (!(pvfs->fs_attribs & FS_ATTR_NAMED_STREAMS)) {
524 0 : flags &= ~PVFS_RESOLVE_STREAMS;
525 : }
526 :
527 : /* SMB2 doesn't allow a leading slash */
528 542693 : if (req->ctx->protocol >= PROTOCOL_SMB2_02 &&
529 207922 : *cifs_name == '\\') {
530 4 : return NT_STATUS_INVALID_PARAMETER;
531 : }
532 :
533 : /* do the basic conversion to a unix formatted path,
534 : also checking for allowable characters */
535 334767 : status = pvfs_unix_path(pvfs, cifs_name, flags, *name);
536 :
537 334767 : if (NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_SYNTAX_BAD)) {
538 : /* it might contain .. components which need to be reduced */
539 148 : status = pvfs_reduce_name(*name, &cifs_name, flags);
540 148 : if (!NT_STATUS_IS_OK(status)) {
541 72 : return status;
542 : }
543 76 : status = pvfs_unix_path(pvfs, cifs_name, flags, *name);
544 : }
545 :
546 334695 : if (!NT_STATUS_IS_OK(status)) {
547 304 : return status;
548 : }
549 :
550 : /* if it has a wildcard then no point doing a stat() of the
551 : full name. Instead We need check if the directory exists
552 : */
553 334391 : if ((*name)->has_wildcard) {
554 : const char *p;
555 : char *dir_name, *saved_name;
556 5432 : p = strrchr((*name)->full_name, '/');
557 5432 : if (p == NULL) {
558 : /* root directory wildcard is OK */
559 0 : return NT_STATUS_OK;
560 : }
561 5432 : dir_name = talloc_strndup(*name, (*name)->full_name, (p-(*name)->full_name));
562 5432 : if (stat(dir_name, &(*name)->st) == 0) {
563 5427 : talloc_free(dir_name);
564 5427 : return NT_STATUS_OK;
565 : }
566 : /* we need to search for a matching name */
567 5 : saved_name = (*name)->full_name;
568 5 : (*name)->full_name = dir_name;
569 5 : status = pvfs_case_search(pvfs, *name, flags);
570 5 : if (!NT_STATUS_IS_OK(status)) {
571 : /* the directory doesn't exist */
572 0 : (*name)->full_name = saved_name;
573 0 : return status;
574 : }
575 : /* it does exist, but might need a case change */
576 5 : if (dir_name != (*name)->full_name) {
577 5 : (*name)->full_name = talloc_asprintf(*name, "%s%s",
578 5 : (*name)->full_name, p);
579 5 : NT_STATUS_HAVE_NO_MEMORY((*name)->full_name);
580 : } else {
581 0 : (*name)->full_name = saved_name;
582 0 : talloc_free(dir_name);
583 : }
584 5 : return NT_STATUS_OK;
585 : }
586 :
587 : /* if we can stat() the full name now then we are done */
588 328959 : if (stat((*name)->full_name, &(*name)->st) == 0) {
589 156681 : (*name)->exists = true;
590 156681 : return pvfs_fill_dos_info(pvfs, *name, flags, -1);
591 : }
592 :
593 : /* search for a matching filename */
594 172278 : status = pvfs_case_search(pvfs, *name, flags);
595 :
596 172278 : return status;
597 : }
598 :
599 :
600 : /*
601 : do a partial resolve, returning a pvfs_filename structure given a
602 : base path and a relative component. It is an error if the file does
603 : not exist. No case-insensitive matching is done.
604 :
605 : this is used in places like directory searching where we need a pvfs_filename
606 : to pass to a function, but already know the unix base directory and component
607 : */
608 130545 : NTSTATUS pvfs_resolve_partial(struct pvfs_state *pvfs, TALLOC_CTX *mem_ctx,
609 : const char *unix_dir, const char *fname,
610 : unsigned int flags, struct pvfs_filename **name)
611 : {
612 : NTSTATUS status;
613 :
614 130545 : *name = talloc(mem_ctx, struct pvfs_filename);
615 130545 : if (*name == NULL) {
616 0 : return NT_STATUS_NO_MEMORY;
617 : }
618 :
619 130545 : (*name)->full_name = talloc_asprintf(*name, "%s/%s", unix_dir, fname);
620 130545 : if ((*name)->full_name == NULL) {
621 0 : return NT_STATUS_NO_MEMORY;
622 : }
623 :
624 130545 : if (stat((*name)->full_name, &(*name)->st) == -1) {
625 0 : return NT_STATUS_OBJECT_NAME_NOT_FOUND;
626 : }
627 :
628 130545 : (*name)->exists = true;
629 130545 : (*name)->stream_exists = true;
630 130545 : (*name)->has_wildcard = false;
631 130545 : (*name)->original_name = talloc_strdup(*name, fname);
632 130545 : (*name)->stream_name = NULL;
633 130545 : (*name)->stream_id = 0;
634 130545 : (*name)->allow_override = false;
635 :
636 130545 : status = pvfs_fill_dos_info(pvfs, *name, flags, -1);
637 :
638 130545 : return status;
639 : }
640 :
641 :
642 : /*
643 : fill in the pvfs_filename info for an open file, given the current
644 : info for a (possibly) non-open file. This is used by places that need
645 : to update the pvfs_filename stat information, and by pvfs_open()
646 : */
647 194277 : NTSTATUS pvfs_resolve_name_fd(struct pvfs_state *pvfs, int fd,
648 : struct pvfs_filename *name, unsigned int flags)
649 : {
650 194277 : dev_t device = (dev_t)0;
651 194277 : ino_t inode = 0;
652 :
653 194277 : if (name->exists) {
654 97238 : device = name->st.st_dev;
655 97238 : inode = name->st.st_ino;
656 : }
657 :
658 194277 : if (fd == -1) {
659 548 : if (stat(name->full_name, &name->st) == -1) {
660 0 : return NT_STATUS_INVALID_HANDLE;
661 : }
662 : } else {
663 193729 : if (fstat(fd, &name->st) == -1) {
664 0 : return NT_STATUS_INVALID_HANDLE;
665 : }
666 : }
667 :
668 291501 : if (name->exists &&
669 194462 : (device != name->st.st_dev || inode != name->st.st_ino)) {
670 : /* the file we are looking at has changed! this could
671 : be someone trying to exploit a race
672 : condition. Certainly we don't want to continue
673 : operating on this file */
674 0 : DEBUG(0,("pvfs: WARNING: file '%s' changed during resolve - failing\n",
675 : name->full_name));
676 0 : return NT_STATUS_UNEXPECTED_IO_ERROR;
677 : }
678 :
679 194277 : name->exists = true;
680 :
681 194277 : return pvfs_fill_dos_info(pvfs, name, flags, fd);
682 : }
683 :
684 : /*
685 : fill in the pvfs_filename info for an open file, given the current
686 : info for a (possibly) non-open file. This is used by places that need
687 : to update the pvfs_filename stat information, and the path
688 : after a possible rename on a different handle.
689 : */
690 8281 : NTSTATUS pvfs_resolve_name_handle(struct pvfs_state *pvfs,
691 : struct pvfs_file_handle *h)
692 : {
693 : NTSTATUS status;
694 :
695 8281 : if (h->have_opendb_entry) {
696 : struct odb_lock *lck;
697 8281 : const char *name = NULL;
698 :
699 8281 : lck = odb_lock(h, h->pvfs->odb_context, &h->odb_locking_key);
700 8281 : if (lck == NULL) {
701 0 : DEBUG(0,("%s: failed to lock file '%s' in opendb\n",
702 : __FUNCTION__, h->name->full_name));
703 : /* we were supposed to do a blocking lock, so something
704 : is badly wrong! */
705 0 : return NT_STATUS_INTERNAL_DB_CORRUPTION;
706 : }
707 :
708 8281 : status = odb_get_path(lck, &name);
709 8281 : if (NT_STATUS_IS_OK(status)) {
710 : /*
711 : * This relies an the fact that
712 : * renames of open files are only
713 : * allowed by setpathinfo() and setfileinfo()
714 : * and there're only renames within the same
715 : * directory supported
716 : */
717 8281 : if (strcmp(h->name->full_name, name) != 0) {
718 : const char *orig_dir;
719 : const char *new_file;
720 : char *new_orig;
721 : char *delim;
722 9 : char *full_name = discard_const_p(char, name);
723 :
724 9 : delim = strrchr(name, '/');
725 9 : if (!delim) {
726 0 : talloc_free(lck);
727 0 : return NT_STATUS_INTERNAL_ERROR;
728 : }
729 :
730 9 : new_file = delim + 1;
731 9 : delim = strrchr(h->name->original_name, '\\');
732 9 : if (delim) {
733 9 : delim[0] = '\0';
734 9 : orig_dir = h->name->original_name;
735 9 : new_orig = talloc_asprintf(h->name, "%s\\%s",
736 : orig_dir, new_file);
737 9 : if (!new_orig) {
738 0 : talloc_free(lck);
739 0 : return NT_STATUS_NO_MEMORY;
740 : }
741 : } else {
742 0 : new_orig = talloc_strdup(h->name, new_file);
743 0 : if (!new_orig) {
744 0 : talloc_free(lck);
745 0 : return NT_STATUS_NO_MEMORY;
746 : }
747 : }
748 :
749 9 : talloc_free(h->name->original_name);
750 9 : talloc_free(h->name->full_name);
751 9 : h->name->full_name = talloc_steal(h->name, full_name);
752 9 : h->name->original_name = new_orig;
753 : }
754 : }
755 :
756 8281 : talloc_free(lck);
757 : }
758 :
759 : /*
760 : * TODO: pass PVFS_RESOLVE_NO_OPENDB and get
761 : * the write time from odb_lock() above.
762 : */
763 8281 : status = pvfs_resolve_name_fd(pvfs, h->fd, h->name, 0);
764 8281 : NT_STATUS_NOT_OK_RETURN(status);
765 :
766 8281 : if (!null_nttime(h->write_time.close_time)) {
767 0 : h->name->dos.write_time = h->write_time.close_time;
768 : }
769 :
770 8281 : return NT_STATUS_OK;
771 : }
772 :
773 :
774 : /*
775 : resolve the parent of a given name
776 : */
777 324250 : NTSTATUS pvfs_resolve_parent(struct pvfs_state *pvfs, TALLOC_CTX *mem_ctx,
778 : const struct pvfs_filename *child,
779 : struct pvfs_filename **name)
780 : {
781 : NTSTATUS status;
782 : char *p;
783 :
784 324250 : *name = talloc(mem_ctx, struct pvfs_filename);
785 324250 : if (*name == NULL) {
786 0 : return NT_STATUS_NO_MEMORY;
787 : }
788 :
789 324250 : (*name)->full_name = talloc_strdup(*name, child->full_name);
790 324250 : if ((*name)->full_name == NULL) {
791 0 : return NT_STATUS_NO_MEMORY;
792 : }
793 :
794 324250 : p = strrchr_m((*name)->full_name, '/');
795 324250 : if (p == NULL) {
796 0 : return NT_STATUS_OBJECT_PATH_SYNTAX_BAD;
797 : }
798 :
799 : /* this handles the root directory */
800 324250 : if (p == (*name)->full_name) {
801 0 : p[1] = 0;
802 : } else {
803 324250 : p[0] = 0;
804 : }
805 :
806 324250 : if (stat((*name)->full_name, &(*name)->st) == -1) {
807 3 : return NT_STATUS_OBJECT_NAME_NOT_FOUND;
808 : }
809 :
810 324247 : (*name)->exists = true;
811 324247 : (*name)->stream_exists = true;
812 324247 : (*name)->has_wildcard = false;
813 : /* we can't get the correct 'original_name', but for the purposes
814 : of this call this is close enough */
815 324247 : (*name)->original_name = talloc_strdup(*name, child->original_name);
816 324247 : if ((*name)->original_name == NULL) {
817 0 : return NT_STATUS_NO_MEMORY;
818 : }
819 324247 : (*name)->stream_name = NULL;
820 324247 : (*name)->stream_id = 0;
821 324247 : (*name)->allow_override = false;
822 :
823 324247 : status = pvfs_fill_dos_info(pvfs, *name, PVFS_RESOLVE_NO_OPENDB, -1);
824 :
825 324247 : return status;
826 : }
|