Apply the changes to libsmbclient that derrell has contributed. Fix some
[Samba/id10ts.git] / source3 / libsmb / libsmb_compat.c
blob4c96c41c564bde7348be4a192189856da3db0ea8
1 /*
2 Unix SMB/CIFS implementation.
3 SMB client library implementation (Old interface compatibility)
4 Copyright (C) Andrew Tridgell 1998
5 Copyright (C) Richard Sharpe 2000
6 Copyright (C) John Terpstra 2000
7 Copyright (C) Tom Jansen (Ninja ISD) 2002
8 Copyright (C) Derrell Lipman 2003
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 2 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program; if not, write to the Free Software
22 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
26 #include "includes.h"
28 #include "../include/libsmb_internal.h"
30 struct smbc_compat_fdlist {
31 SMBCFILE * file;
32 int fd;
33 struct smbc_compat_fdlist *next, *prev;
36 static SMBCCTX * statcont = NULL;
37 static int smbc_compat_initialized = 0;
38 static int smbc_compat_nextfd = 0;
39 static struct smbc_compat_fdlist * smbc_compat_fd_in_use = NULL;
40 static struct smbc_compat_fdlist * smbc_compat_fd_avail = NULL;
42 /* Find an fd and return the SMBCFILE * or NULL on failure */
43 static SMBCFILE * find_fd(int fd)
45 struct smbc_compat_fdlist * f = smbc_compat_fd_in_use;
46 while (f) {
47 if (f->fd == fd)
48 return f->file;
49 f = f->next;
51 return NULL;
54 /* Add an fd, returns 0 on success, -1 on error with errno set */
55 static int add_fd(SMBCFILE * file)
57 struct smbc_compat_fdlist * f = smbc_compat_fd_avail;
59 if (f) {
60 /* We found one that's available */
61 DLIST_REMOVE(smbc_compat_fd_avail, f);
63 } else {
65 * None were available, so allocate one. Keep the number of
66 * file descriptors determinate. This allows the application
67 * to allocate bitmaps or mapping of file descriptors based on
68 * a known maximum number of file descriptors that will ever
69 * be returned.
71 if (smbc_compat_nextfd >= FD_SETSIZE) {
72 errno = EMFILE;
73 return -1;
76 f = malloc(sizeof(struct smbc_compat_fdlist));
77 if (!f) {
78 errno = ENOMEM;
79 return -1;
82 f->fd = SMBC_BASE_FD + smbc_compat_nextfd++;
85 f->file = file;
86 DLIST_ADD(smbc_compat_fd_in_use, f);
88 return f->fd;
93 /* Delete an fd, returns 0 on success */
94 static int del_fd(int fd)
96 struct smbc_compat_fdlist * f = smbc_compat_fd_in_use;
98 while (f) {
99 if (f->fd == fd)
100 break;
101 f = f->next;
104 if (f) {
105 /* found */
106 DLIST_REMOVE(smbc_compat_fd_in_use, f);
107 f->file = NULL;
108 DLIST_ADD(smbc_compat_fd_avail, f);
109 return 0;
111 return 1;
116 int smbc_init(smbc_get_auth_data_fn fn, int debug)
118 int i;
119 struct smbc_compat_fdlist * f;
121 if (!smbc_compat_initialized) {
122 statcont = smbc_new_context();
123 if (!statcont)
124 return -1;
126 statcont->debug = debug;
127 statcont->callbacks.auth_fn = fn;
129 if (!smbc_init_context(statcont)) {
130 smbc_free_context(statcont, False);
131 return -1;
134 smbc_compat_initialized = 1;
136 return 0;
138 return 0;
142 SMBCCTX *smbc_set_context(SMBCCTX * context)
144 SMBCCTX *old_context = statcont;
146 if (context) {
147 /* Save provided context. It must have been initialized! */
148 statcont = context;
150 /* You'd better know what you're doing. We won't help you. */
151 smbc_compat_initialized = 1;
154 return old_context;
158 int smbc_open(const char *furl, int flags, mode_t mode)
160 SMBCFILE * file;
161 int fd;
163 file = statcont->open(statcont, furl, flags, mode);
164 if (!file)
165 return -1;
167 fd = add_fd(file);
168 if (fd == -1)
169 statcont->close(statcont, file);
170 return fd;
174 int smbc_creat(const char *furl, mode_t mode)
176 SMBCFILE * file;
177 int fd;
179 file = statcont->creat(statcont, furl, mode);
180 if (!file)
181 return -1;
183 fd = add_fd(file);
184 if (fd == -1) {
185 /* Hmm... should we delete the file too ? I guess we could try */
186 statcont->close(statcont, file);
187 statcont->unlink(statcont, furl);
189 return fd;
193 ssize_t smbc_read(int fd, void *buf, size_t bufsize)
195 SMBCFILE * file = find_fd(fd);
196 return statcont->read(statcont, file, buf, bufsize);
199 ssize_t smbc_write(int fd, void *buf, size_t bufsize)
201 SMBCFILE * file = find_fd(fd);
202 return statcont->write(statcont, file, buf, bufsize);
205 off_t smbc_lseek(int fd, off_t offset, int whence)
207 SMBCFILE * file = find_fd(fd);
208 return statcont->lseek(statcont, file, offset, whence);
211 int smbc_close(int fd)
213 SMBCFILE * file = find_fd(fd);
214 del_fd(fd);
215 return statcont->close(statcont, file);
218 int smbc_unlink(const char *fname)
220 return statcont->unlink(statcont, fname);
223 int smbc_rename(const char *ourl, const char *nurl)
225 return statcont->rename(statcont, ourl, statcont, nurl);
228 int smbc_opendir(const char *durl)
230 SMBCFILE * file;
231 int fd;
233 file = statcont->opendir(statcont, durl);
234 if (!file)
235 return -1;
237 fd = add_fd(file);
238 if (fd == -1)
239 statcont->closedir(statcont, file);
241 return fd;
244 int smbc_closedir(int dh)
246 SMBCFILE * file = find_fd(dh);
247 del_fd(dh);
248 return statcont->closedir(statcont, file);
251 int smbc_getdents(unsigned int dh, struct smbc_dirent *dirp, int count)
253 SMBCFILE * file = find_fd(dh);
254 return statcont->getdents(statcont, file,dirp, count);
257 struct smbc_dirent* smbc_readdir(unsigned int dh)
259 SMBCFILE * file = find_fd(dh);
260 return statcont->readdir(statcont, file);
263 off_t smbc_telldir(int dh)
265 SMBCFILE * file = find_fd(dh);
266 return statcont->telldir(statcont, file);
269 int smbc_lseekdir(int fd, off_t offset)
271 SMBCFILE * file = find_fd(fd);
272 return statcont->lseekdir(statcont, file, offset);
275 int smbc_mkdir(const char *durl, mode_t mode)
277 return statcont->mkdir(statcont, durl, mode);
280 int smbc_rmdir(const char *durl)
282 return statcont->rmdir(statcont, durl);
285 int smbc_stat(const char *url, struct stat *st)
287 return statcont->stat(statcont, url, st);
290 int smbc_fstat(int fd, struct stat *st)
292 SMBCFILE * file = find_fd(fd);
293 return statcont->fstat(statcont, file, st);
296 int smbc_chmod(const char *url, mode_t mode)
298 return statcont->chmod(statcont, url, mode);
301 int smbc_utimes(const char *fname, struct timeval *tbuf)
303 return statcont->utimes(statcont, fname, tbuf);
306 #ifdef HAVE_UTIME_H
307 int smbc_utime(const char *fname, struct utimbuf *utbuf)
309 struct timeval tv;
311 if (utbuf == NULL)
312 return statcont->utimes(statcont, fname, NULL);
314 tv.tv_sec = utbuf->modtime;
315 tv.tv_usec = 0;
316 return statcont->utimes(statcont, fname, &tv);
318 #endif
320 int smbc_setxattr(const char *fname,
321 const char *name,
322 const void *value,
323 size_t size,
324 int flags)
326 return statcont->setxattr(statcont, fname, name, value, size, flags);
329 int smbc_lsetxattr(const char *fname,
330 const char *name,
331 const void *value,
332 size_t size,
333 int flags)
335 return statcont->setxattr(statcont, fname, name, value, size, flags);
338 int smbc_fsetxattr(int fd,
339 const char *name,
340 const void *value,
341 size_t size,
342 int flags)
344 SMBCFILE * file = find_fd(fd);
345 return statcont->setxattr(statcont, file->fname,
346 name, value, size, flags);
349 int smbc_getxattr(const char *fname,
350 const char *name,
351 const void *value,
352 size_t size)
354 return statcont->getxattr(statcont, fname, name, value, size);
357 int smbc_lgetxattr(const char *fname,
358 const char *name,
359 const void *value,
360 size_t size)
362 return statcont->getxattr(statcont, fname, name, value, size);
365 int smbc_fgetxattr(int fd,
366 const char *name,
367 const void *value,
368 size_t size)
370 SMBCFILE * file = find_fd(fd);
371 return statcont->getxattr(statcont, file->fname, name, value, size);
374 int smbc_removexattr(const char *fname,
375 const char *name)
377 return statcont->removexattr(statcont, fname, name);
380 int smbc_lremovexattr(const char *fname,
381 const char *name)
383 return statcont->removexattr(statcont, fname, name);
386 int smbc_fremovexattr(int fd,
387 const char *name)
389 SMBCFILE * file = find_fd(fd);
390 return statcont->removexattr(statcont, file->fname, name);
393 int smbc_listxattr(const char *fname,
394 char *list,
395 size_t size)
397 return statcont->listxattr(statcont, fname, list, size);
400 int smbc_llistxattr(const char *fname,
401 char *list,
402 size_t size)
404 return statcont->listxattr(statcont, fname, list, size);
407 int smbc_flistxattr(int fd,
408 char *list,
409 size_t size)
411 SMBCFILE * file = find_fd(fd);
412 return statcont->listxattr(statcont, file->fname, list, size);
415 int smbc_print_file(const char *fname, const char *printq)
417 return statcont->print_file(statcont, fname, statcont, printq);
420 int smbc_open_print_job(const char *fname)
422 SMBCFILE * file = statcont->open_print_job(statcont, fname);
423 if (!file) return -1;
424 return (int) file;
427 int smbc_list_print_jobs(const char *purl, smbc_list_print_job_fn fn)
429 return statcont->list_print_jobs(statcont, purl, fn);
432 int smbc_unlink_print_job(const char *purl, int id)
434 return statcont->unlink_print_job(statcont, purl, id);