[Bug 6228] SMBC_open_ctx failure due to path resolve failure doesn't set errno
[Samba/bb.git] / source / libsmb / libsmb_file.c
blob8741ed6c6eb7aa83d5008df98974dff602670974
1 /*
2 Unix SMB/Netbios implementation.
3 SMB client library implementation
4 Copyright (C) Andrew Tridgell 1998
5 Copyright (C) Richard Sharpe 2000, 2002
6 Copyright (C) John Terpstra 2000
7 Copyright (C) Tom Jansen (Ninja ISD) 2002
8 Copyright (C) Derrell Lipman 2003-2008
9 Copyright (C) Jeremy Allison 2007, 2008
11 This program is free software; you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation; either version 3 of the License, or
14 (at your option) any later version.
16 This program is distributed in the hope that it will be useful,
17 but WITHOUT ANY WARRANTY; without even the implied warranty of
18 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 GNU General Public License for more details.
21 You should have received a copy of the GNU General Public License
22 along with this program. If not, see <http://www.gnu.org/licenses/>.
25 #include "includes.h"
26 #include "libsmbclient.h"
27 #include "libsmb_internal.h"
31 * Routine to open() a file ...
34 SMBCFILE *
35 SMBC_open_ctx(SMBCCTX *context,
36 const char *fname,
37 int flags,
38 mode_t mode)
40 char *server = NULL;
41 char *share = NULL;
42 char *user = NULL;
43 char *password = NULL;
44 char *workgroup = NULL;
45 char *path = NULL;
46 char *targetpath = NULL;
47 struct cli_state *targetcli = NULL;
48 SMBCSRV *srv = NULL;
49 SMBCFILE *file = NULL;
50 int fd;
51 TALLOC_CTX *frame = talloc_stackframe();
53 if (!context || !context->internal->initialized) {
55 errno = EINVAL; /* Best I can think of ... */
56 TALLOC_FREE(frame);
57 return NULL;
61 if (!fname) {
63 errno = EINVAL;
64 TALLOC_FREE(frame);
65 return NULL;
69 if (SMBC_parse_path(frame,
70 context,
71 fname,
72 &workgroup,
73 &server,
74 &share,
75 &path,
76 &user,
77 &password,
78 NULL)) {
79 errno = EINVAL;
80 TALLOC_FREE(frame);
81 return NULL;
84 if (!user || user[0] == (char)0) {
85 user = talloc_strdup(frame, smbc_getUser(context));
86 if (!user) {
87 errno = ENOMEM;
88 TALLOC_FREE(frame);
89 return NULL;
93 srv = SMBC_server(frame, context, True,
94 server, share, &workgroup, &user, &password);
96 if (!srv) {
97 if (errno == EPERM) errno = EACCES;
98 TALLOC_FREE(frame);
99 return NULL; /* SMBC_server sets errno */
102 /* Hmmm, the test for a directory is suspect here ... FIXME */
104 if (strlen(path) > 0 && path[strlen(path) - 1] == '\\') {
105 fd = -1;
106 } else {
107 file = SMB_MALLOC_P(SMBCFILE);
109 if (!file) {
110 errno = ENOMEM;
111 TALLOC_FREE(frame);
112 return NULL;
115 ZERO_STRUCTP(file);
117 /*d_printf(">>>open: resolving %s\n", path);*/
118 if (!cli_resolve_path(frame, "", srv->cli, path,
119 &targetcli, &targetpath)) {
120 d_printf("Could not resolve %s\n", path);
121 errno = ENOENT;
122 SAFE_FREE(file);
123 TALLOC_FREE(frame);
124 return NULL;
126 /*d_printf(">>>open: resolved %s as %s\n", path, targetpath);*/
128 if ((fd = cli_open(targetcli, targetpath, flags,
129 context->internal->share_mode)) < 0) {
131 /* Handle the error ... */
133 SAFE_FREE(file);
134 errno = SMBC_errno(context, targetcli);
135 TALLOC_FREE(frame);
136 return NULL;
140 /* Fill in file struct */
142 file->cli_fd = fd;
143 file->fname = SMB_STRDUP(fname);
144 file->srv = srv;
145 file->offset = 0;
146 file->file = True;
148 DLIST_ADD(context->internal->files, file);
151 * If the file was opened in O_APPEND mode, all write
152 * operations should be appended to the file. To do that,
153 * though, using this protocol, would require a getattrE()
154 * call for each and every write, to determine where the end
155 * of the file is. (There does not appear to be an append flag
156 * in the protocol.) Rather than add all of that overhead of
157 * retrieving the current end-of-file offset prior to each
158 * write operation, we'll assume that most append operations
159 * will continuously write, so we'll just set the offset to
160 * the end of the file now and hope that's adequate.
162 * Note to self: If this proves inadequate, and O_APPEND
163 * should, in some cases, be forced for each write, add a
164 * field in the context options structure, for
165 * "strict_append_mode" which would select between the current
166 * behavior (if FALSE) or issuing a getattrE() prior to each
167 * write and forcing the write to the end of the file (if
168 * TRUE). Adding that capability will likely require adding
169 * an "append" flag into the _SMBCFILE structure to track
170 * whether a file was opened in O_APPEND mode. -- djl
172 if (flags & O_APPEND) {
173 if (SMBC_lseek_ctx(context, file, 0, SEEK_END) < 0) {
174 (void) SMBC_close_ctx(context, file);
175 errno = ENXIO;
176 TALLOC_FREE(frame);
177 return NULL;
181 TALLOC_FREE(frame);
182 return file;
186 /* Check if opendir needed ... */
188 if (fd == -1) {
189 int eno = 0;
191 eno = SMBC_errno(context, srv->cli);
192 file = smbc_getFunctionOpendir(context)(context, fname);
193 if (!file) errno = eno;
194 TALLOC_FREE(frame);
195 return file;
199 errno = EINVAL; /* FIXME, correct errno ? */
200 TALLOC_FREE(frame);
201 return NULL;
206 * Routine to create a file
209 SMBCFILE *
210 SMBC_creat_ctx(SMBCCTX *context,
211 const char *path,
212 mode_t mode)
215 if (!context || !context->internal->initialized) {
217 errno = EINVAL;
218 return NULL;
222 return SMBC_open_ctx(context, path,
223 O_WRONLY | O_CREAT | O_TRUNC, mode);
227 * Routine to read() a file ...
230 ssize_t
231 SMBC_read_ctx(SMBCCTX *context,
232 SMBCFILE *file,
233 void *buf,
234 size_t count)
236 int ret;
237 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
238 char *path = NULL;
239 char *targetpath = NULL;
240 struct cli_state *targetcli = NULL;
241 TALLOC_CTX *frame = talloc_stackframe();
244 * offset:
246 * Compiler bug (possibly) -- gcc (GCC) 3.3.5 (Debian 1:3.3.5-2) --
247 * appears to pass file->offset (which is type off_t) differently than
248 * a local variable of type off_t. Using local variable "offset" in
249 * the call to cli_read() instead of file->offset fixes a problem
250 * retrieving data at an offset greater than 4GB.
252 off_t offset;
254 if (!context || !context->internal->initialized) {
256 errno = EINVAL;
257 TALLOC_FREE(frame);
258 return -1;
262 DEBUG(4, ("smbc_read(%p, %d)\n", file, (int)count));
264 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
265 errno = EBADF;
266 TALLOC_FREE(frame);
267 return -1;
271 offset = file->offset;
273 /* Check that the buffer exists ... */
275 if (buf == NULL) {
276 errno = EINVAL;
277 TALLOC_FREE(frame);
278 return -1;
282 /*d_printf(">>>read: parsing %s\n", file->fname);*/
283 if (SMBC_parse_path(frame,
284 context,
285 file->fname,
286 NULL,
287 &server,
288 &share,
289 &path,
290 &user,
291 &password,
292 NULL)) {
293 errno = EINVAL;
294 TALLOC_FREE(frame);
295 return -1;
298 /*d_printf(">>>read: resolving %s\n", path);*/
299 if (!cli_resolve_path(frame, "", file->srv->cli, path,
300 &targetcli, &targetpath)) {
301 d_printf("Could not resolve %s\n", path);
302 errno = ENOENT;
303 TALLOC_FREE(frame);
304 return -1;
306 /*d_printf(">>>fstat: resolved path as %s\n", targetpath);*/
308 ret = cli_read(targetcli, file->cli_fd, (char *)buf, offset, count);
310 if (ret < 0) {
312 errno = SMBC_errno(context, targetcli);
313 TALLOC_FREE(frame);
314 return -1;
318 file->offset += ret;
320 DEBUG(4, (" --> %d\n", ret));
322 TALLOC_FREE(frame);
323 return ret; /* Success, ret bytes of data ... */
328 * Routine to write() a file ...
331 ssize_t
332 SMBC_write_ctx(SMBCCTX *context,
333 SMBCFILE *file,
334 const void *buf,
335 size_t count)
337 int ret;
338 off_t offset;
339 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
340 char *path = NULL;
341 char *targetpath = NULL;
342 struct cli_state *targetcli = NULL;
343 TALLOC_CTX *frame = talloc_stackframe();
345 /* First check all pointers before dereferencing them */
347 if (!context || !context->internal->initialized) {
349 errno = EINVAL;
350 TALLOC_FREE(frame);
351 return -1;
355 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
356 errno = EBADF;
357 TALLOC_FREE(frame);
358 return -1;
361 /* Check that the buffer exists ... */
363 if (buf == NULL) {
364 errno = EINVAL;
365 TALLOC_FREE(frame);
366 return -1;
370 offset = file->offset; /* See "offset" comment in SMBC_read_ctx() */
372 /*d_printf(">>>write: parsing %s\n", file->fname);*/
373 if (SMBC_parse_path(frame,
374 context,
375 file->fname,
376 NULL,
377 &server,
378 &share,
379 &path,
380 &user,
381 &password,
382 NULL)) {
383 errno = EINVAL;
384 TALLOC_FREE(frame);
385 return -1;
388 /*d_printf(">>>write: resolving %s\n", path);*/
389 if (!cli_resolve_path(frame, "", file->srv->cli, path,
390 &targetcli, &targetpath)) {
391 d_printf("Could not resolve %s\n", path);
392 errno = ENOENT;
393 TALLOC_FREE(frame);
394 return -1;
396 /*d_printf(">>>write: resolved path as %s\n", targetpath);*/
398 ret = cli_write(targetcli, file->cli_fd,
399 0, (char *)buf, offset, count);
401 if (ret <= 0) {
402 errno = SMBC_errno(context, targetcli);
403 TALLOC_FREE(frame);
404 return -1;
408 file->offset += ret;
410 TALLOC_FREE(frame);
411 return ret; /* Success, 0 bytes of data ... */
415 * Routine to close() a file ...
419 SMBC_close_ctx(SMBCCTX *context,
420 SMBCFILE *file)
422 SMBCSRV *srv;
423 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
424 char *path = NULL;
425 char *targetpath = NULL;
426 struct cli_state *targetcli = NULL;
427 TALLOC_CTX *frame = talloc_stackframe();
429 if (!context || !context->internal->initialized) {
431 errno = EINVAL;
432 TALLOC_FREE(frame);
433 return -1;
436 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
437 errno = EBADF;
438 TALLOC_FREE(frame);
439 return -1;
442 /* IS a dir ... */
443 if (!file->file) {
444 TALLOC_FREE(frame);
445 return smbc_getFunctionClosedir(context)(context, file);
448 /*d_printf(">>>close: parsing %s\n", file->fname);*/
449 if (SMBC_parse_path(frame,
450 context,
451 file->fname,
452 NULL,
453 &server,
454 &share,
455 &path,
456 &user,
457 &password,
458 NULL)) {
459 errno = EINVAL;
460 TALLOC_FREE(frame);
461 return -1;
464 /*d_printf(">>>close: resolving %s\n", path);*/
465 if (!cli_resolve_path(frame, "", file->srv->cli, path,
466 &targetcli, &targetpath)) {
467 d_printf("Could not resolve %s\n", path);
468 errno = ENOENT;
469 TALLOC_FREE(frame);
470 return -1;
472 /*d_printf(">>>close: resolved path as %s\n", targetpath);*/
474 if (!cli_close(targetcli, file->cli_fd)) {
476 DEBUG(3, ("cli_close failed on %s. purging server.\n",
477 file->fname));
478 /* Deallocate slot and remove the server
479 * from the server cache if unused */
480 errno = SMBC_errno(context, targetcli);
481 srv = file->srv;
482 DLIST_REMOVE(context->internal->files, file);
483 SAFE_FREE(file->fname);
484 SAFE_FREE(file);
485 smbc_getFunctionRemoveUnusedServer(context)(context, srv);
486 TALLOC_FREE(frame);
487 return -1;
491 DLIST_REMOVE(context->internal->files, file);
492 SAFE_FREE(file->fname);
493 SAFE_FREE(file);
494 TALLOC_FREE(frame);
496 return 0;
500 * Get info from an SMB server on a file. Use a qpathinfo call first
501 * and if that fails, use getatr, as Win95 sometimes refuses qpathinfo
503 bool
504 SMBC_getatr(SMBCCTX * context,
505 SMBCSRV *srv,
506 char *path,
507 uint16 *mode,
508 SMB_OFF_T *size,
509 struct timespec *create_time_ts,
510 struct timespec *access_time_ts,
511 struct timespec *write_time_ts,
512 struct timespec *change_time_ts,
513 SMB_INO_T *ino)
515 char *fixedpath = NULL;
516 char *targetpath = NULL;
517 struct cli_state *targetcli = NULL;
518 time_t write_time;
519 TALLOC_CTX *frame = talloc_stackframe();
521 if (!context || !context->internal->initialized) {
523 errno = EINVAL;
524 TALLOC_FREE(frame);
525 return false;
528 /* path fixup for . and .. */
529 if (strequal(path, ".") || strequal(path, "..")) {
530 fixedpath = talloc_strdup(frame, "\\");
531 if (!fixedpath) {
532 errno = ENOMEM;
533 TALLOC_FREE(frame);
534 return false;
536 } else {
537 fixedpath = talloc_strdup(frame, path);
538 if (!fixedpath) {
539 errno = ENOMEM;
540 TALLOC_FREE(frame);
541 return false;
543 trim_string(fixedpath, NULL, "\\..");
544 trim_string(fixedpath, NULL, "\\.");
546 DEBUG(4,("SMBC_getatr: sending qpathinfo\n"));
548 if (!cli_resolve_path(frame, "", srv->cli, fixedpath,
549 &targetcli, &targetpath)) {
550 d_printf("Couldn't resolve %s\n", path);
551 errno = ENOENT;
552 TALLOC_FREE(frame);
553 return False;
556 if (!srv->no_pathinfo2 &&
557 cli_qpathinfo2(targetcli, targetpath,
558 create_time_ts,
559 access_time_ts,
560 write_time_ts,
561 change_time_ts,
562 size, mode, ino)) {
563 TALLOC_FREE(frame);
564 return True;
567 /* if this is NT then don't bother with the getatr */
568 if (targetcli->capabilities & CAP_NT_SMBS) {
569 errno = EPERM;
570 TALLOC_FREE(frame);
571 return False;
574 if (cli_getatr(targetcli, targetpath, mode, size, &write_time)) {
576 struct timespec w_time_ts;
578 w_time_ts = convert_time_t_to_timespec(write_time);
580 if (write_time_ts != NULL) {
581 *write_time_ts = w_time_ts;
584 if (create_time_ts != NULL) {
585 *create_time_ts = w_time_ts;
588 if (access_time_ts != NULL) {
589 *access_time_ts = w_time_ts;
592 if (change_time_ts != NULL) {
593 *change_time_ts = w_time_ts;
596 srv->no_pathinfo2 = True;
597 TALLOC_FREE(frame);
598 return True;
601 errno = EPERM;
602 TALLOC_FREE(frame);
603 return False;
608 * Set file info on an SMB server. Use setpathinfo call first. If that
609 * fails, use setattrE..
611 * Access and modification time parameters are always used and must be
612 * provided. Create time, if zero, will be determined from the actual create
613 * time of the file. If non-zero, the create time will be set as well.
615 * "mode" (attributes) parameter may be set to -1 if it is not to be set.
617 bool
618 SMBC_setatr(SMBCCTX * context, SMBCSRV *srv, char *path,
619 time_t create_time,
620 time_t access_time,
621 time_t write_time,
622 time_t change_time,
623 uint16 mode)
625 int fd;
626 int ret;
627 TALLOC_CTX *frame = talloc_stackframe();
630 * First, try setpathinfo (if qpathinfo succeeded), for it is the
631 * modern function for "new code" to be using, and it works given a
632 * filename rather than requiring that the file be opened to have its
633 * attributes manipulated.
635 if (srv->no_pathinfo ||
636 ! cli_setpathinfo(srv->cli, path,
637 create_time,
638 access_time,
639 write_time,
640 change_time,
641 mode)) {
644 * setpathinfo is not supported; go to plan B.
646 * cli_setatr() does not work on win98, and it also doesn't
647 * support setting the access time (only the modification
648 * time), so in all cases, we open the specified file and use
649 * cli_setattrE() which should work on all OS versions, and
650 * supports both times.
653 /* Don't try {q,set}pathinfo() again, with this server */
654 srv->no_pathinfo = True;
656 /* Open the file */
657 if ((fd = cli_open(srv->cli, path, O_RDWR, DENY_NONE)) < 0) {
659 errno = SMBC_errno(context, srv->cli);
660 TALLOC_FREE(frame);
661 return -1;
664 /* Set the new attributes */
665 ret = cli_setattrE(srv->cli, fd,
666 change_time,
667 access_time,
668 write_time);
670 /* Close the file */
671 cli_close(srv->cli, fd);
674 * Unfortunately, setattrE() doesn't have a provision for
675 * setting the access mode (attributes). We'll have to try
676 * cli_setatr() for that, and with only this parameter, it
677 * seems to work on win98.
679 if (ret && mode != (uint16) -1) {
680 ret = cli_setatr(srv->cli, path, mode, 0);
683 if (! ret) {
684 errno = SMBC_errno(context, srv->cli);
685 TALLOC_FREE(frame);
686 return False;
690 TALLOC_FREE(frame);
691 return True;
695 * A routine to lseek() a file
698 off_t
699 SMBC_lseek_ctx(SMBCCTX *context,
700 SMBCFILE *file,
701 off_t offset,
702 int whence)
704 SMB_OFF_T size;
705 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
706 char *path = NULL;
707 char *targetpath = NULL;
708 struct cli_state *targetcli = NULL;
709 TALLOC_CTX *frame = talloc_stackframe();
711 if (!context || !context->internal->initialized) {
713 errno = EINVAL;
714 TALLOC_FREE(frame);
715 return -1;
718 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
720 errno = EBADF;
721 TALLOC_FREE(frame);
722 return -1;
726 if (!file->file) {
728 errno = EINVAL;
729 TALLOC_FREE(frame);
730 return -1; /* Can't lseek a dir ... */
734 switch (whence) {
735 case SEEK_SET:
736 file->offset = offset;
737 break;
739 case SEEK_CUR:
740 file->offset += offset;
741 break;
743 case SEEK_END:
744 /*d_printf(">>>lseek: parsing %s\n", file->fname);*/
745 if (SMBC_parse_path(frame,
746 context,
747 file->fname,
748 NULL,
749 &server,
750 &share,
751 &path,
752 &user,
753 &password,
754 NULL)) {
755 errno = EINVAL;
756 TALLOC_FREE(frame);
757 return -1;
760 /*d_printf(">>>lseek: resolving %s\n", path);*/
761 if (!cli_resolve_path(frame, "", file->srv->cli, path,
762 &targetcli, &targetpath)) {
763 d_printf("Could not resolve %s\n", path);
764 errno = ENOENT;
765 TALLOC_FREE(frame);
766 return -1;
768 /*d_printf(">>>lseek: resolved path as %s\n", targetpath);*/
770 if (!cli_qfileinfo(targetcli, file->cli_fd, NULL,
771 &size, NULL, NULL, NULL, NULL, NULL))
773 SMB_OFF_T b_size = size;
774 if (!cli_getattrE(targetcli, file->cli_fd,
775 NULL, &b_size, NULL, NULL, NULL))
777 errno = EINVAL;
778 TALLOC_FREE(frame);
779 return -1;
780 } else
781 size = b_size;
783 file->offset = size + offset;
784 break;
786 default:
787 errno = EINVAL;
788 break;
792 TALLOC_FREE(frame);
793 return file->offset;
799 * Routine to truncate a file given by its file descriptor, to a specified size
803 SMBC_ftruncate_ctx(SMBCCTX *context,
804 SMBCFILE *file,
805 off_t length)
807 SMB_OFF_T size = length;
808 char *server = NULL;
809 char *share = NULL;
810 char *user = NULL;
811 char *password = NULL;
812 char *path = NULL;
813 char *targetpath = NULL;
814 struct cli_state *targetcli = NULL;
815 TALLOC_CTX *frame = talloc_stackframe();
817 if (!context || !context->internal->initialized) {
819 errno = EINVAL;
820 TALLOC_FREE(frame);
821 return -1;
824 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
825 errno = EBADF;
826 TALLOC_FREE(frame);
827 return -1;
830 if (!file->file) {
831 errno = EINVAL;
832 TALLOC_FREE(frame);
833 return -1;
836 /*d_printf(">>>fstat: parsing %s\n", file->fname);*/
837 if (SMBC_parse_path(frame,
838 context,
839 file->fname,
840 NULL,
841 &server,
842 &share,
843 &path,
844 &user,
845 &password,
846 NULL)) {
847 errno = EINVAL;
848 TALLOC_FREE(frame);
849 return -1;
852 /*d_printf(">>>fstat: resolving %s\n", path);*/
853 if (!cli_resolve_path(frame, "", file->srv->cli, path,
854 &targetcli, &targetpath)) {
855 d_printf("Could not resolve %s\n", path);
856 errno = ENOENT;
857 TALLOC_FREE(frame);
858 return -1;
860 /*d_printf(">>>fstat: resolved path as %s\n", targetpath);*/
862 if (!cli_ftruncate(targetcli, file->cli_fd, size)) {
863 errno = EINVAL;
864 TALLOC_FREE(frame);
865 return -1;
868 TALLOC_FREE(frame);
869 return 0;