source: vendor/3.6.23/source3/libsmb/libsmb_file.c

Last change on this file was 740, checked in by Silvan Scherrer, 13 years ago

Samba Server: update vendor to 3.6.0

File size: 23.4 KB
Line 
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
10
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.
15
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.
20
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/>.
23*/
24
25#include "includes.h"
26#include "libsmb/libsmb.h"
27#include "libsmbclient.h"
28#include "libsmb_internal.h"
29
30
31/*
32 * Routine to open() a file ...
33 */
34
35SMBCFILE *
36SMBC_open_ctx(SMBCCTX *context,
37 const char *fname,
38 int flags,
39 mode_t mode)
40{
41 char *server = NULL;
42 char *share = NULL;
43 char *user = NULL;
44 char *password = NULL;
45 char *workgroup = NULL;
46 char *path = NULL;
47 char *targetpath = NULL;
48 struct cli_state *targetcli = NULL;
49 SMBCSRV *srv = NULL;
50 SMBCFILE *file = NULL;
51 uint16_t fd;
52 NTSTATUS status = NT_STATUS_OBJECT_PATH_INVALID;
53 TALLOC_CTX *frame = talloc_stackframe();
54
55 if (!context || !context->internal->initialized) {
56 errno = EINVAL; /* Best I can think of ... */
57 TALLOC_FREE(frame);
58 return NULL;
59 }
60
61 if (!fname) {
62 errno = EINVAL;
63 TALLOC_FREE(frame);
64 return NULL;
65 }
66
67 if (SMBC_parse_path(frame,
68 context,
69 fname,
70 &workgroup,
71 &server,
72 &share,
73 &path,
74 &user,
75 &password,
76 NULL)) {
77 errno = EINVAL;
78 TALLOC_FREE(frame);
79 return NULL;
80 }
81
82 if (!user || user[0] == (char)0) {
83 user = talloc_strdup(frame, smbc_getUser(context));
84 if (!user) {
85 errno = ENOMEM;
86 TALLOC_FREE(frame);
87 return NULL;
88 }
89 }
90
91 srv = SMBC_server(frame, context, True,
92 server, share, &workgroup, &user, &password);
93 if (!srv) {
94 if (errno == EPERM) errno = EACCES;
95 TALLOC_FREE(frame);
96 return NULL; /* SMBC_server sets errno */
97 }
98
99 /* Hmmm, the test for a directory is suspect here ... FIXME */
100
101 if (strlen(path) > 0 && path[strlen(path) - 1] == '\\') {
102 status = NT_STATUS_OBJECT_PATH_INVALID;
103 } else {
104 file = SMB_MALLOC_P(SMBCFILE);
105 if (!file) {
106 errno = ENOMEM;
107 TALLOC_FREE(frame);
108 return NULL;
109 }
110
111 ZERO_STRUCTP(file);
112
113 /*d_printf(">>>open: resolving %s\n", path);*/
114 if (!cli_resolve_path(frame, "", context->internal->auth_info,
115 srv->cli, path,
116 &targetcli, &targetpath)) {
117 d_printf("Could not resolve %s\n", path);
118 errno = ENOENT;
119 SAFE_FREE(file);
120 TALLOC_FREE(frame);
121 return NULL;
122 }
123 /*d_printf(">>>open: resolved %s as %s\n", path, targetpath);*/
124
125 status = cli_open(targetcli, targetpath, flags,
126 context->internal->share_mode, &fd);
127 if (!NT_STATUS_IS_OK(status)) {
128
129 /* Handle the error ... */
130
131 SAFE_FREE(file);
132 errno = SMBC_errno(context, targetcli);
133 TALLOC_FREE(frame);
134 return NULL;
135 }
136
137 /* Fill in file struct */
138
139 file->cli_fd = fd;
140 file->fname = SMB_STRDUP(fname);
141 file->srv = srv;
142 file->offset = 0;
143 file->file = True;
144
145 DLIST_ADD(context->internal->files, file);
146
147 /*
148 * If the file was opened in O_APPEND mode, all write
149 * operations should be appended to the file. To do that,
150 * though, using this protocol, would require a getattrE()
151 * call for each and every write, to determine where the end
152 * of the file is. (There does not appear to be an append flag
153 * in the protocol.) Rather than add all of that overhead of
154 * retrieving the current end-of-file offset prior to each
155 * write operation, we'll assume that most append operations
156 * will continuously write, so we'll just set the offset to
157 * the end of the file now and hope that's adequate.
158 *
159 * Note to self: If this proves inadequate, and O_APPEND
160 * should, in some cases, be forced for each write, add a
161 * field in the context options structure, for
162 * "strict_append_mode" which would select between the current
163 * behavior (if FALSE) or issuing a getattrE() prior to each
164 * write and forcing the write to the end of the file (if
165 * TRUE). Adding that capability will likely require adding
166 * an "append" flag into the _SMBCFILE structure to track
167 * whether a file was opened in O_APPEND mode. -- djl
168 */
169 if (flags & O_APPEND) {
170 if (SMBC_lseek_ctx(context, file, 0, SEEK_END) < 0) {
171 (void) SMBC_close_ctx(context, file);
172 errno = ENXIO;
173 TALLOC_FREE(frame);
174 return NULL;
175 }
176 }
177
178 TALLOC_FREE(frame);
179 return file;
180 }
181
182 /* Check if opendir needed ... */
183
184 if (!NT_STATUS_IS_OK(status)) {
185 int eno = 0;
186
187 eno = SMBC_errno(context, srv->cli);
188 file = smbc_getFunctionOpendir(context)(context, fname);
189 if (!file) errno = eno;
190 TALLOC_FREE(frame);
191 return file;
192 }
193
194 errno = EINVAL; /* FIXME, correct errno ? */
195 TALLOC_FREE(frame);
196 return NULL;
197}
198
199/*
200 * Routine to create a file
201 */
202
203SMBCFILE *
204SMBC_creat_ctx(SMBCCTX *context,
205 const char *path,
206 mode_t mode)
207{
208 if (!context || !context->internal->initialized) {
209 errno = EINVAL;
210 return NULL;
211 }
212
213 return SMBC_open_ctx(context, path,
214 O_WRONLY | O_CREAT | O_TRUNC, mode);
215}
216
217/*
218 * Routine to read() a file ...
219 */
220
221ssize_t
222SMBC_read_ctx(SMBCCTX *context,
223 SMBCFILE *file,
224 void *buf,
225 size_t count)
226{
227 int ret;
228 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
229 char *path = NULL;
230 char *targetpath = NULL;
231 struct cli_state *targetcli = NULL;
232 TALLOC_CTX *frame = talloc_stackframe();
233
234 /*
235 * offset:
236 *
237 * Compiler bug (possibly) -- gcc (GCC) 3.3.5 (Debian 1:3.3.5-2) --
238 * appears to pass file->offset (which is type off_t) differently than
239 * a local variable of type off_t. Using local variable "offset" in
240 * the call to cli_read() instead of file->offset fixes a problem
241 * retrieving data at an offset greater than 4GB.
242 */
243 off_t offset;
244
245 if (!context || !context->internal->initialized) {
246 errno = EINVAL;
247 TALLOC_FREE(frame);
248 return -1;
249 }
250
251 DEBUG(4, ("smbc_read(%p, %d)\n", file, (int)count));
252
253 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
254 errno = EBADF;
255 TALLOC_FREE(frame);
256 return -1;
257 }
258
259 offset = file->offset;
260
261 /* Check that the buffer exists ... */
262
263 if (buf == NULL) {
264 errno = EINVAL;
265 TALLOC_FREE(frame);
266 return -1;
267 }
268
269 /*d_printf(">>>read: parsing %s\n", file->fname);*/
270 if (SMBC_parse_path(frame,
271 context,
272 file->fname,
273 NULL,
274 &server,
275 &share,
276 &path,
277 &user,
278 &password,
279 NULL)) {
280 errno = EINVAL;
281 TALLOC_FREE(frame);
282 return -1;
283 }
284
285 /*d_printf(">>>read: resolving %s\n", path);*/
286 if (!cli_resolve_path(frame, "", context->internal->auth_info,
287 file->srv->cli, path,
288 &targetcli, &targetpath)) {
289 d_printf("Could not resolve %s\n", path);
290 errno = ENOENT;
291 TALLOC_FREE(frame);
292 return -1;
293 }
294 /*d_printf(">>>fstat: resolved path as %s\n", targetpath);*/
295
296 ret = cli_read(targetcli, file->cli_fd, (char *)buf, offset, count);
297
298 if (ret < 0) {
299 errno = SMBC_errno(context, targetcli);
300 TALLOC_FREE(frame);
301 return -1;
302 }
303
304 file->offset += ret;
305
306 DEBUG(4, (" --> %d\n", ret));
307
308 TALLOC_FREE(frame);
309 return ret; /* Success, ret bytes of data ... */
310}
311
312/*
313 * Routine to write() a file ...
314 */
315
316ssize_t
317SMBC_write_ctx(SMBCCTX *context,
318 SMBCFILE *file,
319 const void *buf,
320 size_t count)
321{
322 off_t offset;
323 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
324 char *path = NULL;
325 char *targetpath = NULL;
326 struct cli_state *targetcli = NULL;
327 TALLOC_CTX *frame = talloc_stackframe();
328 NTSTATUS status;
329
330 /* First check all pointers before dereferencing them */
331
332 if (!context || !context->internal->initialized) {
333 errno = EINVAL;
334 TALLOC_FREE(frame);
335 return -1;
336 }
337
338 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
339 errno = EBADF;
340 TALLOC_FREE(frame);
341 return -1;
342 }
343
344 /* Check that the buffer exists ... */
345
346 if (buf == NULL) {
347 errno = EINVAL;
348 TALLOC_FREE(frame);
349 return -1;
350 }
351
352 offset = file->offset; /* See "offset" comment in SMBC_read_ctx() */
353
354 /*d_printf(">>>write: parsing %s\n", file->fname);*/
355 if (SMBC_parse_path(frame,
356 context,
357 file->fname,
358 NULL,
359 &server,
360 &share,
361 &path,
362 &user,
363 &password,
364 NULL)) {
365 errno = EINVAL;
366 TALLOC_FREE(frame);
367 return -1;
368 }
369
370 /*d_printf(">>>write: resolving %s\n", path);*/
371 if (!cli_resolve_path(frame, "", context->internal->auth_info,
372 file->srv->cli, path,
373 &targetcli, &targetpath)) {
374 d_printf("Could not resolve %s\n", path);
375 errno = ENOENT;
376 TALLOC_FREE(frame);
377 return -1;
378 }
379 /*d_printf(">>>write: resolved path as %s\n", targetpath);*/
380
381 status = cli_writeall(targetcli, file->cli_fd,
382 0, (uint8_t *)buf, offset, count, NULL);
383 if (!NT_STATUS_IS_OK(status)) {
384 errno = map_errno_from_nt_status(status);
385 TALLOC_FREE(frame);
386 return -1;
387 }
388
389 file->offset += count;
390
391 TALLOC_FREE(frame);
392 return count; /* Success, 0 bytes of data ... */
393}
394
395/*
396 * Routine to close() a file ...
397 */
398
399int
400SMBC_close_ctx(SMBCCTX *context,
401 SMBCFILE *file)
402{
403 SMBCSRV *srv;
404 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
405 char *path = NULL;
406 char *targetpath = NULL;
407 struct cli_state *targetcli = NULL;
408 TALLOC_CTX *frame = talloc_stackframe();
409
410 if (!context || !context->internal->initialized) {
411 errno = EINVAL;
412 TALLOC_FREE(frame);
413 return -1;
414 }
415
416 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
417 errno = EBADF;
418 TALLOC_FREE(frame);
419 return -1;
420 }
421
422 /* IS a dir ... */
423 if (!file->file) {
424 TALLOC_FREE(frame);
425 return smbc_getFunctionClosedir(context)(context, file);
426 }
427
428 /*d_printf(">>>close: parsing %s\n", file->fname);*/
429 if (SMBC_parse_path(frame,
430 context,
431 file->fname,
432 NULL,
433 &server,
434 &share,
435 &path,
436 &user,
437 &password,
438 NULL)) {
439 errno = EINVAL;
440 TALLOC_FREE(frame);
441 return -1;
442 }
443
444 /*d_printf(">>>close: resolving %s\n", path);*/
445 if (!cli_resolve_path(frame, "", context->internal->auth_info,
446 file->srv->cli, path,
447 &targetcli, &targetpath)) {
448 d_printf("Could not resolve %s\n", path);
449 errno = ENOENT;
450 TALLOC_FREE(frame);
451 return -1;
452 }
453 /*d_printf(">>>close: resolved path as %s\n", targetpath);*/
454
455 if (!NT_STATUS_IS_OK(cli_close(targetcli, file->cli_fd))) {
456 DEBUG(3, ("cli_close failed on %s. purging server.\n",
457 file->fname));
458 /* Deallocate slot and remove the server
459 * from the server cache if unused */
460 errno = SMBC_errno(context, targetcli);
461 srv = file->srv;
462 DLIST_REMOVE(context->internal->files, file);
463 SAFE_FREE(file->fname);
464 SAFE_FREE(file);
465 smbc_getFunctionRemoveUnusedServer(context)(context, srv);
466 TALLOC_FREE(frame);
467 return -1;
468 }
469
470 DLIST_REMOVE(context->internal->files, file);
471 SAFE_FREE(file->fname);
472 SAFE_FREE(file);
473 TALLOC_FREE(frame);
474 return 0;
475}
476
477/*
478 * Get info from an SMB server on a file. Use a qpathinfo call first
479 * and if that fails, use getatr, as Win95 sometimes refuses qpathinfo
480 */
481bool
482SMBC_getatr(SMBCCTX * context,
483 SMBCSRV *srv,
484 char *path,
485 uint16 *mode,
486 SMB_OFF_T *size,
487 struct timespec *create_time_ts,
488 struct timespec *access_time_ts,
489 struct timespec *write_time_ts,
490 struct timespec *change_time_ts,
491 SMB_INO_T *ino)
492{
493 char *fixedpath = NULL;
494 char *targetpath = NULL;
495 struct cli_state *targetcli = NULL;
496 time_t write_time;
497 TALLOC_CTX *frame = talloc_stackframe();
498
499 if (!context || !context->internal->initialized) {
500 errno = EINVAL;
501 TALLOC_FREE(frame);
502 return False;
503 }
504
505 /* path fixup for . and .. */
506 if (strequal(path, ".") || strequal(path, "..")) {
507 fixedpath = talloc_strdup(frame, "\\");
508 if (!fixedpath) {
509 errno = ENOMEM;
510 TALLOC_FREE(frame);
511 return False;
512 }
513 } else {
514 fixedpath = talloc_strdup(frame, path);
515 if (!fixedpath) {
516 errno = ENOMEM;
517 TALLOC_FREE(frame);
518 return False;
519 }
520 trim_string(fixedpath, NULL, "\\..");
521 trim_string(fixedpath, NULL, "\\.");
522 }
523 DEBUG(4,("SMBC_getatr: sending qpathinfo\n"));
524
525 if (!cli_resolve_path(frame, "", context->internal->auth_info,
526 srv->cli, fixedpath,
527 &targetcli, &targetpath)) {
528 d_printf("Couldn't resolve %s\n", path);
529 errno = ENOENT;
530 TALLOC_FREE(frame);
531 return False;
532 }
533
534 if (!srv->no_pathinfo2 &&
535 NT_STATUS_IS_OK(cli_qpathinfo2(targetcli, targetpath,
536 create_time_ts,
537 access_time_ts,
538 write_time_ts,
539 change_time_ts,
540 size, mode, ino))) {
541 TALLOC_FREE(frame);
542 return True;
543 }
544
545 /* if this is NT then don't bother with the getatr */
546 if (targetcli->capabilities & CAP_NT_SMBS) {
547 errno = EPERM;
548 TALLOC_FREE(frame);
549 return False;
550 }
551
552 if (NT_STATUS_IS_OK(cli_getatr(targetcli, targetpath, mode, size, &write_time))) {
553 struct timespec w_time_ts;
554
555 w_time_ts = convert_time_t_to_timespec(write_time);
556 if (write_time_ts != NULL) {
557 *write_time_ts = w_time_ts;
558 }
559 if (create_time_ts != NULL) {
560 *create_time_ts = w_time_ts;
561 }
562 if (access_time_ts != NULL) {
563 *access_time_ts = w_time_ts;
564 }
565 if (change_time_ts != NULL) {
566 *change_time_ts = w_time_ts;
567 }
568 srv->no_pathinfo2 = True;
569 TALLOC_FREE(frame);
570 return True;
571 }
572
573 errno = EPERM;
574 TALLOC_FREE(frame);
575 return False;
576}
577
578/*
579 * Set file info on an SMB server. Use setpathinfo call first. If that
580 * fails, use setattrE..
581 *
582 * Access and modification time parameters are always used and must be
583 * provided. Create time, if zero, will be determined from the actual create
584 * time of the file. If non-zero, the create time will be set as well.
585 *
586 * "mode" (attributes) parameter may be set to -1 if it is not to be set.
587 */
588bool
589SMBC_setatr(SMBCCTX * context, SMBCSRV *srv, char *path,
590 time_t create_time,
591 time_t access_time,
592 time_t write_time,
593 time_t change_time,
594 uint16 mode)
595{
596 uint16_t fd;
597 int ret;
598 TALLOC_CTX *frame = talloc_stackframe();
599
600 /*
601 * First, try setpathinfo (if qpathinfo succeeded), for it is the
602 * modern function for "new code" to be using, and it works given a
603 * filename rather than requiring that the file be opened to have its
604 * attributes manipulated.
605 */
606 if (srv->no_pathinfo ||
607 !NT_STATUS_IS_OK(cli_setpathinfo_basic(srv->cli, path,
608 create_time,
609 access_time,
610 write_time,
611 change_time,
612 mode))) {
613
614 /*
615 * setpathinfo is not supported; go to plan B.
616 *
617 * cli_setatr() does not work on win98, and it also doesn't
618 * support setting the access time (only the modification
619 * time), so in all cases, we open the specified file and use
620 * cli_setattrE() which should work on all OS versions, and
621 * supports both times.
622 */
623
624 /* Don't try {q,set}pathinfo() again, with this server */
625 srv->no_pathinfo = True;
626
627 /* Open the file */
628 if (!NT_STATUS_IS_OK(cli_open(srv->cli, path, O_RDWR, DENY_NONE, &fd))) {
629 errno = SMBC_errno(context, srv->cli);
630 TALLOC_FREE(frame);
631 return -1;
632 }
633
634 /* Set the new attributes */
635 ret = NT_STATUS_IS_OK(cli_setattrE(srv->cli, fd,
636 change_time,
637 access_time,
638 write_time));
639
640 /* Close the file */
641 cli_close(srv->cli, fd);
642
643 /*
644 * Unfortunately, setattrE() doesn't have a provision for
645 * setting the access mode (attributes). We'll have to try
646 * cli_setatr() for that, and with only this parameter, it
647 * seems to work on win98.
648 */
649 if (ret && mode != (uint16) -1) {
650 ret = NT_STATUS_IS_OK(cli_setatr(srv->cli, path, mode, 0));
651 }
652
653 if (! ret) {
654 errno = SMBC_errno(context, srv->cli);
655 TALLOC_FREE(frame);
656 return False;
657 }
658 }
659
660 TALLOC_FREE(frame);
661 return True;
662}
663
664/*
665 * A routine to lseek() a file
666 */
667
668off_t
669SMBC_lseek_ctx(SMBCCTX *context,
670 SMBCFILE *file,
671 off_t offset,
672 int whence)
673{
674 SMB_OFF_T size;
675 char *server = NULL, *share = NULL, *user = NULL, *password = NULL;
676 char *path = NULL;
677 char *targetpath = NULL;
678 struct cli_state *targetcli = NULL;
679 TALLOC_CTX *frame = talloc_stackframe();
680
681 if (!context || !context->internal->initialized) {
682 errno = EINVAL;
683 TALLOC_FREE(frame);
684 return -1;
685 }
686
687 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
688 errno = EBADF;
689 TALLOC_FREE(frame);
690 return -1;
691 }
692
693 if (!file->file) {
694 errno = EINVAL;
695 TALLOC_FREE(frame);
696 return -1; /* Can't lseek a dir ... */
697 }
698
699 switch (whence) {
700 case SEEK_SET:
701 file->offset = offset;
702 break;
703 case SEEK_CUR:
704 file->offset += offset;
705 break;
706 case SEEK_END:
707 /*d_printf(">>>lseek: parsing %s\n", file->fname);*/
708 if (SMBC_parse_path(frame,
709 context,
710 file->fname,
711 NULL,
712 &server,
713 &share,
714 &path,
715 &user,
716 &password,
717 NULL)) {
718 errno = EINVAL;
719 TALLOC_FREE(frame);
720 return -1;
721 }
722
723 /*d_printf(">>>lseek: resolving %s\n", path);*/
724 if (!cli_resolve_path(frame, "", context->internal->auth_info,
725 file->srv->cli, path,
726 &targetcli, &targetpath)) {
727 d_printf("Could not resolve %s\n", path);
728 errno = ENOENT;
729 TALLOC_FREE(frame);
730 return -1;
731 }
732
733 /*d_printf(">>>lseek: resolved path as %s\n", targetpath);*/
734 if (!NT_STATUS_IS_OK(cli_qfileinfo_basic(
735 targetcli, file->cli_fd, NULL,
736 &size, NULL, NULL, NULL, NULL,
737 NULL))) {
738 SMB_OFF_T b_size = size;
739 if (!NT_STATUS_IS_OK(cli_getattrE(targetcli, file->cli_fd,
740 NULL, &b_size, NULL, NULL, NULL))) {
741 errno = EINVAL;
742 TALLOC_FREE(frame);
743 return -1;
744 } else
745 size = b_size;
746 }
747 file->offset = size + offset;
748 break;
749 default:
750 errno = EINVAL;
751 break;
752 }
753
754 TALLOC_FREE(frame);
755 return file->offset;
756}
757
758
759/*
760 * Routine to truncate a file given by its file descriptor, to a specified size
761 */
762
763int
764SMBC_ftruncate_ctx(SMBCCTX *context,
765 SMBCFILE *file,
766 off_t length)
767{
768 SMB_OFF_T size = length;
769 char *server = NULL;
770 char *share = NULL;
771 char *user = NULL;
772 char *password = NULL;
773 char *path = NULL;
774 char *targetpath = NULL;
775 struct cli_state *targetcli = NULL;
776 TALLOC_CTX *frame = talloc_stackframe();
777
778 if (!context || !context->internal->initialized) {
779 errno = EINVAL;
780 TALLOC_FREE(frame);
781 return -1;
782 }
783
784 if (!file || !SMBC_dlist_contains(context->internal->files, file)) {
785 errno = EBADF;
786 TALLOC_FREE(frame);
787 return -1;
788 }
789
790 if (!file->file) {
791 errno = EINVAL;
792 TALLOC_FREE(frame);
793 return -1;
794 }
795
796 /*d_printf(">>>fstat: parsing %s\n", file->fname);*/
797 if (SMBC_parse_path(frame,
798 context,
799 file->fname,
800 NULL,
801 &server,
802 &share,
803 &path,
804 &user,
805 &password,
806 NULL)) {
807 errno = EINVAL;
808 TALLOC_FREE(frame);
809 return -1;
810 }
811
812 /*d_printf(">>>fstat: resolving %s\n", path);*/
813 if (!cli_resolve_path(frame, "", context->internal->auth_info,
814 file->srv->cli, path,
815 &targetcli, &targetpath)) {
816 d_printf("Could not resolve %s\n", path);
817 errno = ENOENT;
818 TALLOC_FREE(frame);
819 return -1;
820 }
821 /*d_printf(">>>fstat: resolved path as %s\n", targetpath);*/
822
823 if (!NT_STATUS_IS_OK(cli_ftruncate(targetcli, file->cli_fd, (uint64_t)size))) {
824 errno = EINVAL;
825 TALLOC_FREE(frame);
826 return -1;
827 }
828
829 TALLOC_FREE(frame);
830 return 0;
831}
Note: See TracBrowser for help on using the repository browser.