]> arthur.barton.de Git - netatalk.git/blob - libatalk/vfs/vfs.c
d79ac05e1142cb14d53bf226ba45e53722ef74ab
[netatalk.git] / libatalk / vfs / vfs.c
1 /*
2     Copyright (c) 2004 Didier Gautheron
3     Copyright (c) 2009 Frank Lahm
4  
5    This program is free software; you can redistribute it and/or modify
6    it under the terms of the GNU General Public License as published by
7    the Free Software Foundation; either version 2 of the License, or
8    (at your option) any later version.
9  
10    This program is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13    GNU General Public License for more details.
14  
15    You should have received a copy of the GNU General Public License
16    along with this program; if not, write to the Free Software
17    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18  
19 */
20 #ifdef HAVE_CONFIG_H
21 #include "config.h"
22 #endif /* HAVE_CONFIG_H */
23
24 #include <sys/types.h>
25 #include <sys/stat.h>
26 #include <unistd.h>
27 #include <string.h>
28 #include <stdio.h>
29 #include <stdlib.h>
30 #include <libgen.h>
31
32 #include <atalk/afp.h>    
33 #include <atalk/adouble.h>
34 #include <atalk/ea.h>
35 #include <atalk/acl.h>
36 #include <atalk/logger.h>
37 #include <atalk/util.h>
38 #include <atalk/volume.h>
39 #include <atalk/vfs.h>
40 #include <atalk/directory.h>
41 #include <atalk/unix.h>
42 #include <atalk/errchk.h>
43 #include <atalk/bstrlib.h>
44 #include <atalk/bstradd.h>
45 #include <atalk/compat.h>
46
47 struct perm {
48     uid_t uid;
49     gid_t gid;
50 };
51
52 typedef int (*rf_loop)(const struct vol *, struct dirent *, char *, void *, int);
53
54 /* ----------------------------- */
55 static int 
56 for_each_adouble(const char *from, const char *name, rf_loop fn, const struct vol *vol, void *data, int flag)
57 {
58     char            buf[ MAXPATHLEN + 1];
59     char            *m;
60     DIR             *dp;
61     struct dirent   *de;
62     int             ret;
63     
64
65     if (NULL == ( dp = opendir( name)) ) {
66         if (!flag) {
67             LOG(log_error, logtype_afpd, "%s: opendir %s: %s", from, fullpathname(name),strerror(errno) );
68             return -1;
69         }
70         return 0;
71     }
72     strlcpy( buf, name, sizeof(buf));
73     strlcat( buf, "/", sizeof(buf) );
74     m = strchr( buf, '\0' );
75     ret = 0;
76     while ((de = readdir(dp))) {
77         if (!strcmp(de->d_name, ".") || !strcmp(de->d_name, "..")) {
78                 continue;
79         }
80         
81         strlcat(buf, de->d_name, sizeof(buf));
82         if (fn && (ret = fn(vol, de, buf, data, flag))) {
83            closedir(dp);
84            return ret;
85         }
86         *m = 0;
87     }
88     closedir(dp);
89     return ret;
90 }
91
92 static int netatalk_name(const char *name)
93 {
94     return strcmp(name,".AppleDB") && strcmp(name,".AppleDesktop");        
95 }
96
97 /*******************************************************************************
98  * classic adouble format 
99  *******************************************************************************/
100
101 static int validupath_adouble(VFS_FUNC_ARGS_VALIDUPATH)
102 {
103     if (name[0] != '.')
104         return 1;
105     
106     return netatalk_name(name) && strcmp(name,".AppleDouble") && strcasecmp(name,".Parent");
107 }                                           
108
109 /* ----------------- */
110 static int RF_chown_adouble(VFS_FUNC_ARGS_CHOWN)
111 {
112     struct stat st;
113     const char *ad_p;
114
115     ad_p = vol->ad_path(path, ADFLAGS_HF );
116
117     if ( stat( ad_p, &st ) < 0 )
118         return 0; /* ignore */
119
120     return chown( ad_p, uid, gid );
121 }
122
123 /* ----------------- */
124 static int RF_renamedir_adouble(VFS_FUNC_ARGS_RENAMEDIR)
125 {
126     return 0;
127 }
128
129 /* ----------------- */
130 static int deletecurdir_adouble_loop(const struct vol *vol, struct dirent *de, char *name, void *data _U_, int flag _U_)
131 {
132     struct stat st;
133     int         err;
134     
135     /* bail if the file exists in the current directory.
136      * note: this will not fail with dangling symlinks */
137     
138     if (lstat(de->d_name, &st) == 0)
139         return AFPERR_DIRNEMPT;
140
141     if ((err = netatalk_unlink(name)))
142         return err;
143
144     return 0;
145 }
146
147 static int RF_deletecurdir_adouble(VFS_FUNC_ARGS_DELETECURDIR)
148 {
149     int err;
150
151     /* delete stray .AppleDouble files. this happens to get .Parent files
152        as well. */
153     if ((err = for_each_adouble("deletecurdir", ".AppleDouble", deletecurdir_adouble_loop, vol, NULL, 1))) 
154         return err;
155     return netatalk_rmdir(-1, ".AppleDouble" );
156 }
157
158 /* ----------------- */
159 static int adouble_setfilmode(const struct vol *vol, const char *name, mode_t mode, struct stat *st)
160 {
161     return setfilmode(vol, name, ad_hf_mode(mode), st);
162 }
163
164 static int RF_setfilmode_adouble(VFS_FUNC_ARGS_SETFILEMODE)
165 {
166     return adouble_setfilmode(vol, vol->ad_path(name, ADFLAGS_HF ), mode, st);
167 }
168
169 /* ----------------- */
170 static int RF_setdirunixmode_adouble(VFS_FUNC_ARGS_SETDIRUNIXMODE)
171 {
172     const char *adouble = vol->ad_path(name, ADFLAGS_DIR );
173
174     if (dir_rx_set(mode)) {
175         if (chmod_acl(ad_dir(adouble), (DIRBITS | mode) & ~vol->v_umask) < 0 ) 
176             return -1;
177     }
178
179     if (adouble_setfilmode(vol, vol->ad_path(name, ADFLAGS_DIR ), mode, st) < 0) 
180         return -1;
181
182     if (!dir_rx_set(mode)) {
183         if (chmod_acl(ad_dir(adouble), (DIRBITS | mode) & ~vol->v_umask) < 0 ) 
184             return  -1 ;
185     }
186     return 0;
187 }
188
189 /* ----------------- */
190 static int setdirmode_adouble_loop(const struct vol *vol, struct dirent *de _U_, char *name, void *data, int flag)
191 {
192     mode_t hf_mode = *(mode_t *)data;
193     struct stat st;
194
195     if (ostat(name, &st, vol_syml_opt(vol)) < 0 ) {
196         if (flag)
197             return 0;
198         LOG(log_error, logtype_afpd, "setdirmode: stat %s: %s", name, strerror(errno) );
199     }
200     else if (!S_ISDIR(st.st_mode)) {
201         if (setfilmode(vol, name, hf_mode, &st) < 0) {
202                /* FIXME what do we do then? */
203         }
204     }
205     return 0;
206 }
207
208 static int RF_setdirmode_adouble(VFS_FUNC_ARGS_SETDIRMODE)
209 {
210     mode_t hf_mode = ad_hf_mode(mode);
211     const char  *adouble = vol->ad_path(name, ADFLAGS_DIR );
212     const char  *adouble_p = ad_dir(adouble);
213
214     if (dir_rx_set(mode)) {
215         if (chmod_acl(ad_dir(adouble), (DIRBITS | mode) & ~vol->v_umask) < 0) 
216             return -1;
217     }
218
219     if (for_each_adouble("setdirmode", adouble_p, setdirmode_adouble_loop, vol, &hf_mode, 0))
220         return -1;
221
222     if (!dir_rx_set(mode)) {
223         if (chmod_acl(ad_dir(adouble), (DIRBITS | mode) & ~vol->v_umask) < 0) 
224             return  -1 ;
225     }
226     return 0;
227 }
228
229 static int RF_setdirowner_adouble(VFS_FUNC_ARGS_SETDIROWNER)
230 {
231     if (lchown(".AppleDouble", uid, gid) < 0 && errno != EPERM ) {
232         LOG(log_debug, logtype_afpd, "setdirowner: chown %d/%d %s: %s",
233             uid, gid,fullpathname(".AppleDouble"), strerror(errno));
234     }
235     return 0;
236 }
237
238 /* ----------------- */
239 static int RF_deletefile_adouble(VFS_FUNC_ARGS_DELETEFILE)
240 {
241         return netatalk_unlinkat(dirfd, vol->ad_path(file, ADFLAGS_HF));
242 }
243
244 /* ----------------- */
245 static int RF_renamefile_adouble(VFS_FUNC_ARGS_RENAMEFILE)
246 {
247     char  adsrc[ MAXPATHLEN + 1];
248     int   err = 0;
249
250     strcpy( adsrc, vol->ad_path(src, 0 ));
251     if (unix_rename(dirfd, adsrc, -1, vol->ad_path(dst, 0 )) < 0) {
252         struct stat st;
253
254         err = errno;
255         if (errno == ENOENT) {
256                 struct adouble    ad;
257
258             if (ostatat(dirfd, adsrc, &st, vol_syml_opt(vol))) /* source has no ressource fork, */
259                 return 0;
260
261             /* We are here  because :
262              * -there's no dest folder. 
263              * -there's no .AppleDouble in the dest folder.
264              * if we use the struct adouble passed in parameter it will not
265              * create .AppleDouble if the file is already opened, so we
266              * use a diff one, it's not a pb,ie it's not the same file, yet.
267              */
268             ad_init(&ad, vol); 
269             if (ad_open(&ad, dst, ADFLAGS_HF | ADFLAGS_RDWR | ADFLAGS_CREATE, 0666) == 0) {
270                 ad_close(&ad, ADFLAGS_HF);
271                 if (!unix_rename(dirfd, adsrc, -1, vol->ad_path(dst, 0 )) ) 
272                    err = 0;
273                 else 
274                    err = errno;
275             }
276             else { /* it's something else, bail out */
277                     err = errno;
278                 }
279             }
280         }
281         if (err) {
282                 errno = err;
283                 return -1;
284         }
285         return 0;
286 }
287
288 static int RF_copyfile_adouble(VFS_FUNC_ARGS_COPYFILE)
289 /* const struct vol *vol, int sfd, const char *src, const char *dst */
290 {
291     EC_INIT;
292     bstring s = NULL, d = NULL;
293     char *dup1 = NULL;
294     char *dup2 = NULL;
295     char *dup3 = NULL;
296     char *dup4 = NULL;
297     const char *name = NULL;
298     const char *dir = NULL;
299
300     struct stat st;
301     EC_ZERO(stat(dst, &st));
302
303     if (S_ISDIR(st.st_mode)) {
304         /* build src path to AppleDouble file*/
305         EC_NULL(s = bfromcstr(src));
306         EC_ZERO(bcatcstr(s, "/.AppleDouble/.Parent"));
307
308         /* build dst path to AppleDouble file*/
309         EC_NULL(d = bfromcstr(dst));
310         EC_ZERO(bcatcstr(d, "/.AppleDouble/.Parent"));
311     } else {
312         /* get basename */
313
314         /* build src path to AppleDouble file*/
315         EC_NULL(dup1 = strdup(src));
316         EC_NULL(name = basename(strdup(dup1)));
317
318         EC_NULL(dup2 = strdup(src));
319         EC_NULL(dir = dirname(dup2));
320         EC_NULL(s = bfromcstr(dir));
321         EC_ZERO(bcatcstr(s, "/.AppleDouble/"));
322         EC_ZERO(bcatcstr(s, name));
323
324         /* build dst path to AppleDouble file*/
325         EC_NULL(dup4 = strdup(dst));
326         EC_NULL(name = basename(strdup(dup4)));
327
328         EC_NULL(dup3 = strdup(dst));
329         EC_NULL(dir = dirname(dup3));
330         EC_NULL(d = bfromcstr(dir));
331         EC_ZERO(bcatcstr(d, "/.AppleDouble/"));
332         EC_ZERO(bcatcstr(d, name));
333     }
334
335     /* ignore errors */
336     if (copy_file(sfd, cfrombstr(s), cfrombstr(d), 0666) != 0)
337         if (errno != ENOENT)
338             EC_FAIL;
339
340 EC_CLEANUP:
341     bdestroy(s);
342     bdestroy(d);
343     if (dup1) free(dup1);
344     if (dup2) free(dup2);
345     if (dup3) free(dup3);
346     if (dup4) free(dup4);
347
348     EC_EXIT;
349 }
350
351 #ifdef HAVE_SOLARIS_ACLS
352 static int RF_solaris_acl(VFS_FUNC_ARGS_ACL)
353 {
354     static char buf[ MAXPATHLEN + 1];
355     struct stat st;
356     int len;
357
358     if ((stat(path, &st)) != 0) {
359         if (errno == ENOENT)
360             return AFP_OK;
361         return AFPERR_MISC;
362     }
363     if (!S_ISDIR(st.st_mode)) {
364         /* set ACL on ressource fork */
365         if ((acl(vol->ad_path(path, ADFLAGS_HF), cmd, count, aces)) != 0)
366             return AFPERR_MISC;
367     }
368     return AFP_OK;
369 }
370
371 static int RF_solaris_remove_acl(VFS_FUNC_ARGS_REMOVE_ACL)
372 {
373     int ret;
374     static char buf[ MAXPATHLEN + 1];
375     int len;
376
377     if (dir)
378         return AFP_OK;
379
380     /* remove ACL from ressource fork */
381     if ((ret = remove_acl_vfs(vol->ad_path(path, ADFLAGS_HF))) != AFP_OK) {
382         if (errno == ENOENT)
383             return AFP_OK;
384         return ret;
385     }
386
387     return AFP_OK;
388 }
389 #endif
390
391 #ifdef HAVE_POSIX_ACLS
392 static int RF_posix_acl(VFS_FUNC_ARGS_ACL)
393 {
394     EC_INIT;
395     struct stat st;
396
397     if (stat(path, &st) == -1)
398         EC_FAIL;
399
400     if (!S_ISDIR(st.st_mode)) {
401         /* set ACL on ressource fork */
402         EC_ZERO_ERR( acl_set_file(vol->ad_path(path, ADFLAGS_HF), type, acl), AFPERR_MISC );
403     }
404     
405 EC_CLEANUP:
406     if (errno == ENOENT)
407         EC_STATUS(AFP_OK);
408     EC_EXIT;
409 }
410
411 static int RF_posix_remove_acl(VFS_FUNC_ARGS_REMOVE_ACL)
412 {
413     EC_INIT;
414
415     if (dir)
416         EC_EXIT_STATUS(AFP_OK);
417
418     /* remove ACL from ressource fork */
419     EC_ZERO_ERR( remove_acl_vfs(vol->ad_path(path, ADFLAGS_HF)), AFPERR_MISC );
420
421 EC_CLEANUP:
422     if (errno == ENOENT)
423         EC_STATUS(AFP_OK);
424     EC_EXIT;
425 }
426 #endif
427
428 /*************************************************************************
429  * EA adouble format 
430  ************************************************************************/
431 static int validupath_ea(VFS_FUNC_ARGS_VALIDUPATH)
432 {
433     if (name[0] != '.')
434         return 1;
435     
436 #ifndef HAVE_EAFD
437     if (name[1] == '_')
438         return ad_valid_header_osx(name);
439 #endif
440     return netatalk_name(name);
441 }
442
443 /* ----------------- */
444 static int RF_chown_ea(VFS_FUNC_ARGS_CHOWN)
445 {
446 #ifndef HAVE_EAFD
447     return chown(vol->ad_path(path, ADFLAGS_HF ), uid, gid);
448 #endif
449     return 0;
450 }
451
452 /* ---------------- */
453 static int RF_renamedir_ea(VFS_FUNC_ARGS_RENAMEDIR)
454 {
455     return 0;
456 }
457
458 /* Returns 1 if the entry is NOT an ._ file */
459 static int deletecurdir_ea_osx_chkifempty_loop(const struct vol *vol, struct dirent *de, char *name, void *data _U_, int flag _U_)
460 {
461     if (de->d_name[0] != '.' || de->d_name[0] == '_')
462         return 1;
463
464     return 0;
465 }
466
467 static int deletecurdir_ea_osx_loop(const struct vol *vol, struct dirent *de, char *name, void *data _U_, int flag _U_)
468 {
469     int ret;
470     
471     if ((ret = netatalk_unlink(name)) != 0)
472         return ret;
473
474     return 0;
475 }
476
477 /* ---------------- */
478 static int RF_deletecurdir_ea(VFS_FUNC_ARGS_DELETECURDIR)
479 {
480 #ifndef HAVE_EAFD
481     int err;
482     /* delete stray ._AppleDouble files */
483
484     /* first check if there's really no other file besides files starting with ._ */
485     if ((err = for_each_adouble("deletecurdir_ea_osx", ".",
486                                 deletecurdir_ea_osx_chkifempty_loop,
487                                 vol, NULL, 0)) != 0) {
488         if (err == 1)
489             return AFPERR_DIRNEMPT;
490         return AFPERR_MISC;
491     }
492
493     /* Now delete orphaned ._ files */
494     if ((err = for_each_adouble("deletecurdir_ea_osx", ".",
495                                 deletecurdir_ea_osx_loop,
496                                 vol, NULL, 0)) != 0)
497         return err;
498
499 #endif
500     return 0;
501 }
502
503 /* ---------------- */
504 static int RF_setdirunixmode_ea(VFS_FUNC_ARGS_SETDIRUNIXMODE)
505 {
506 #ifndef HAVE_EAFD
507 #endif
508     return 0;
509 }
510
511 static int RF_setfilmode_ea(VFS_FUNC_ARGS_SETFILEMODE)
512 {
513 #ifndef HAVE_EAFD
514     return adouble_setfilmode(vol, vol->ad_path(name, ADFLAGS_HF ), mode, st);
515 #endif
516     return 0;
517 }
518
519 /* ---------------- */
520 static int RF_setdirmode_ea(VFS_FUNC_ARGS_SETDIRMODE)
521 {
522 #ifndef HAVE_EAFD
523 #endif
524     return 0;
525 }
526
527 /* ---------------- */
528 static int RF_setdirowner_ea(VFS_FUNC_ARGS_SETDIROWNER)
529 {
530 #ifndef HAVE_EAFD
531 #endif
532         return 0;
533 }
534
535 static int RF_deletefile_ea(VFS_FUNC_ARGS_DELETEFILE)
536 {
537 #ifndef HAVE_EAFD
538         return netatalk_unlinkat(dirfd, vol->ad_path(file, ADFLAGS_HF));
539 #endif
540     return 0;
541 }
542 static int RF_copyfile_ea(VFS_FUNC_ARGS_COPYFILE)
543 {
544 #ifdef HAVE_EAFD
545     /* the EA VFS module does this all for us */
546     return 0;
547 #endif
548
549     EC_INIT;
550     bstring s = NULL, d = NULL;
551     char *dup1 = NULL;
552     char *dup2 = NULL;
553     char *dup3 = NULL;
554     char *dup4 = NULL;
555     const char *name = NULL;
556     const char *dir = NULL;
557
558     /* get basename */
559
560     /* build src path to ._ file*/
561     EC_NULL(dup1 = strdup(src));
562     EC_NULL(name = basename(strdup(dup1)));
563
564     EC_NULL(dup2 = strdup(src));
565     EC_NULL(dir = dirname(dup2));
566     EC_NULL(s = bfromcstr(dir));
567     EC_ZERO(bcatcstr(s, "/._"));
568     EC_ZERO(bcatcstr(s, name));
569
570     /* build dst path to ._file*/
571     EC_NULL(dup4 = strdup(dst));
572     EC_NULL(name = basename(strdup(dup4)));
573
574     EC_NULL(dup3 = strdup(dst));
575     EC_NULL(dir = dirname(dup3));
576     EC_NULL(d = bfromcstr(dir));
577     EC_ZERO(bcatcstr(d, "/._"));
578     EC_ZERO(bcatcstr(d, name));
579
580     if (copy_file(sfd, cfrombstr(s), cfrombstr(d), 0666) != 0) {
581         switch (errno) {
582         case ENOENT:
583             break;
584         default:
585             LOG(log_error, logtype_afpd, "[VFS] copyfile(\"%s\" -> \"%s\"): %s",
586                 cfrombstr(s), cfrombstr(d), strerror(errno));
587             EC_FAIL;
588                     
589         }
590     }
591
592 EC_CLEANUP:
593     bdestroy(s);
594     bdestroy(d);
595     free(dup1);
596     free(dup2);
597     free(dup3);
598     free(dup4);
599     EC_EXIT;
600 }
601
602 /* ---------------- */
603 static int RF_renamefile_ea(VFS_FUNC_ARGS_RENAMEFILE)
604 {
605 #ifndef HAVE_EAFD
606     char  adsrc[ MAXPATHLEN + 1];
607     int   err = 0;
608
609     strcpy( adsrc, vol->ad_path(src, 0 ));
610
611     if (unix_rename(dirfd, adsrc, -1, vol->ad_path(dst, 0 )) < 0) {
612         struct stat st;
613
614         err = errno;
615         if (errno == ENOENT && ostatat(dirfd, adsrc, &st, vol_syml_opt(vol))) /* source has no ressource fork, */
616             return 0;
617         errno = err;
618         return -1;
619     }
620     return 0;
621 #endif
622     return 0;
623 }
624
625 /********************************************************************************************
626  * VFS chaining
627  ********************************************************************************************/
628
629 /* 
630  * Up until we really start stacking many VFS modules on top of one another or use
631  * dynamic module loading like we do for UAMs, up until then we just stack VFS modules
632  * via an fixed size array.
633  * All VFS funcs must return AFP_ERR codes. When a func in the chain returns an error
634  * this error code will be returned to the caller, BUT the chain in followed and all
635  * following funcs are called in order to give them a chance.
636  */
637
638 /* 
639  * Define most VFS funcs with macros as they all do the same.
640  * Only "ad_path" and "validupath" will NOT do stacking and only
641  * call the func from the first module.
642  */
643
644 #define VFS_MFUNC(name, args, vars) \
645     static int vfs_ ## name(args) \
646     { \
647         int i = 0, ret = AFP_OK, err; \
648         while (vol->vfs_modules[i]) { \
649             if (vol->vfs_modules[i]->vfs_ ## name) { \
650                 err = vol->vfs_modules[i]->vfs_ ## name (vars); \
651                 if ((ret == AFP_OK) && (err != AFP_OK)) \
652                     ret = err; \
653             } \
654             i ++; \
655         } \
656         return ret; \
657     }
658
659 VFS_MFUNC(chown, VFS_FUNC_ARGS_CHOWN, VFS_FUNC_VARS_CHOWN)
660 VFS_MFUNC(renamedir, VFS_FUNC_ARGS_RENAMEDIR, VFS_FUNC_VARS_RENAMEDIR) 
661 VFS_MFUNC(deletecurdir, VFS_FUNC_ARGS_DELETECURDIR, VFS_FUNC_VARS_DELETECURDIR)
662 VFS_MFUNC(setfilmode, VFS_FUNC_ARGS_SETFILEMODE, VFS_FUNC_VARS_SETFILEMODE)
663 VFS_MFUNC(setdirmode, VFS_FUNC_ARGS_SETDIRMODE, VFS_FUNC_VARS_SETDIRMODE)
664 VFS_MFUNC(setdirunixmode, VFS_FUNC_ARGS_SETDIRUNIXMODE, VFS_FUNC_VARS_SETDIRUNIXMODE)
665 VFS_MFUNC(setdirowner, VFS_FUNC_ARGS_SETDIROWNER, VFS_FUNC_VARS_SETDIROWNER)
666 VFS_MFUNC(deletefile, VFS_FUNC_ARGS_DELETEFILE, VFS_FUNC_VARS_DELETEFILE)
667 VFS_MFUNC(renamefile, VFS_FUNC_ARGS_RENAMEFILE, VFS_FUNC_VARS_RENAMEFILE)
668 VFS_MFUNC(copyfile, VFS_FUNC_ARGS_COPYFILE, VFS_FUNC_VARS_COPYFILE)
669 #ifdef HAVE_ACLS
670 VFS_MFUNC(acl, VFS_FUNC_ARGS_ACL, VFS_FUNC_VARS_ACL)
671 VFS_MFUNC(remove_acl, VFS_FUNC_ARGS_REMOVE_ACL, VFS_FUNC_VARS_REMOVE_ACL)
672 #endif
673 VFS_MFUNC(ea_getsize, VFS_FUNC_ARGS_EA_GETSIZE, VFS_FUNC_VARS_EA_GETSIZE)
674 VFS_MFUNC(ea_getcontent, VFS_FUNC_ARGS_EA_GETCONTENT, VFS_FUNC_VARS_EA_GETCONTENT)
675 VFS_MFUNC(ea_list, VFS_FUNC_ARGS_EA_LIST, VFS_FUNC_VARS_EA_LIST)
676 VFS_MFUNC(ea_set, VFS_FUNC_ARGS_EA_SET, VFS_FUNC_VARS_EA_SET)
677 VFS_MFUNC(ea_remove, VFS_FUNC_ARGS_EA_REMOVE, VFS_FUNC_VARS_EA_REMOVE)
678
679 static int vfs_validupath(VFS_FUNC_ARGS_VALIDUPATH)
680 {
681     return vol->vfs_modules[0]->vfs_validupath(VFS_FUNC_VARS_VALIDUPATH);
682 }
683
684 /*
685  * These function pointers get called from the lib users via vol->vfs->func.
686  * These funcs are defined via the macros above.
687  */
688 static struct vfs_ops vfs_master_funcs = {
689     vfs_validupath,
690     vfs_chown,
691     vfs_renamedir,
692     vfs_deletecurdir,
693     vfs_setfilmode,
694     vfs_setdirmode,
695     vfs_setdirunixmode,
696     vfs_setdirowner,
697     vfs_deletefile,
698     vfs_renamefile,
699     vfs_copyfile,
700 #ifdef HAVE_ACLS
701     vfs_acl,
702     vfs_remove_acl,
703 #endif
704     vfs_ea_getsize,
705     vfs_ea_getcontent,
706     vfs_ea_list,
707     vfs_ea_set,
708     vfs_ea_remove
709 };
710
711 /* 
712  * Primary adouble modules: v2, ea
713  */
714
715 static struct vfs_ops netatalk_adouble_v2 = {
716     /* vfs_validupath:    */ validupath_adouble,
717     /* vfs_chown:         */ RF_chown_adouble,
718     /* vfs_renamedir:     */ RF_renamedir_adouble,
719     /* vfs_deletecurdir:  */ RF_deletecurdir_adouble,
720     /* vfs_setfilmode:    */ RF_setfilmode_adouble,
721     /* vfs_setdirmode:    */ RF_setdirmode_adouble,
722     /* vfs_setdirunixmode:*/ RF_setdirunixmode_adouble,
723     /* vfs_setdirowner:   */ RF_setdirowner_adouble,
724     /* vfs_deletefile:    */ RF_deletefile_adouble,
725     /* vfs_renamefile:    */ RF_renamefile_adouble,
726     /* vfs_copyfile:      */ RF_copyfile_adouble,
727     NULL
728 };
729
730 static struct vfs_ops netatalk_adouble_ea = {
731     /* vfs_validupath:    */ validupath_ea,
732     /* vfs_chown:         */ RF_chown_ea,
733     /* vfs_renamedir:     */ RF_renamedir_ea,
734     /* vfs_deletecurdir:  */ RF_deletecurdir_ea,
735     /* vfs_setfilmode:    */ RF_setfilmode_ea,
736     /* vfs_setdirmode:    */ RF_setdirmode_ea,
737     /* vfs_setdirunixmode:*/ RF_setdirunixmode_ea,
738     /* vfs_setdirowner:   */ RF_setdirowner_ea,
739     /* vfs_deletefile:    */ RF_deletefile_ea,
740     /* vfs_renamefile:    */ RF_renamefile_ea,
741     /* vfs_copyfile:      */ RF_copyfile_ea,
742     NULL
743 };
744
745 /* 
746  * Secondary vfs modules for Extended Attributes
747  */
748
749 static struct vfs_ops netatalk_ea_adouble = {
750     /* vfs_validupath:    */ NULL,
751     /* vfs_chown:         */ ea_chown,
752     /* vfs_renamedir:     */ NULL, /* ok */
753     /* vfs_deletecurdir:  */ NULL, /* ok */
754     /* vfs_setfilmode:    */ ea_chmod_file,
755     /* vfs_setdirmode:    */ NULL, /* ok */
756     /* vfs_setdirunixmode:*/ ea_chmod_dir,
757     /* vfs_setdirowner:   */ NULL, /* ok */
758     /* vfs_deletefile:    */ ea_deletefile,
759     /* vfs_renamefile:    */ ea_renamefile,
760     /* vfs_copyfile       */ ea_copyfile,
761 #ifdef HAVE_ACLS
762     /* vfs_acl:           */ NULL,
763     /* vfs_remove_acl     */ NULL,
764 #endif
765     /* vfs_getsize        */ get_easize,
766     /* vfs_getcontent     */ get_eacontent,
767     /* vfs_list           */ list_eas,
768     /* vfs_set            */ set_ea,
769     /* vfs_remove         */ remove_ea
770 };
771
772 static struct vfs_ops netatalk_ea_sys = {
773     /* validupath:        */ NULL,
774     /* rf_chown:          */ NULL,
775     /* rf_renamedir:      */ NULL,
776     /* rf_deletecurdir:   */ NULL,
777     /* rf_setfilmode:     */ NULL,
778     /* rf_setdirmode:     */ NULL,
779     /* rf_setdirunixmode: */ NULL,
780     /* rf_setdirowner:    */ NULL,
781     /* rf_deletefile:     */ NULL,
782     /* rf_renamefile:     */ NULL,
783     /* vfs_copyfile:      */ sys_ea_copyfile,
784 #ifdef HAVE_ACLS
785     /* rf_acl:            */ NULL,
786     /* rf_remove_acl      */ NULL,
787 #endif
788     /* ea_getsize         */ sys_get_easize,
789     /* ea_getcontent      */ sys_get_eacontent,
790     /* ea_list            */ sys_list_eas,
791     /* ea_set             */ sys_set_ea,
792     /* ea_remove          */ sys_remove_ea
793 };
794
795 /* 
796  * Tertiary VFS modules for ACLs
797  */
798
799 #ifdef HAVE_SOLARIS_ACLS
800 static struct vfs_ops netatalk_solaris_acl_adouble = {
801     /* validupath:        */ NULL,
802     /* rf_chown:          */ NULL,
803     /* rf_renamedir:      */ NULL,
804     /* rf_deletecurdir:   */ NULL,
805     /* rf_setfilmode:     */ NULL,
806     /* rf_setdirmode:     */ NULL,
807     /* rf_setdirunixmode: */ NULL,
808     /* rf_setdirowner:    */ NULL,
809     /* rf_deletefile:     */ NULL,
810     /* rf_renamefile:     */ NULL,
811     /* vfs_copyfile       */ NULL,
812     /* rf_acl:            */ RF_solaris_acl,
813     /* rf_remove_acl      */ RF_solaris_remove_acl,
814     NULL
815 };
816 #endif
817
818 #ifdef HAVE_POSIX_ACLS
819 static struct vfs_ops netatalk_posix_acl_adouble = {
820     /* validupath:        */ NULL,
821     /* rf_chown:          */ NULL,
822     /* rf_renamedir:      */ NULL,
823     /* rf_deletecurdir:   */ NULL,
824     /* rf_setfilmode:     */ NULL,
825     /* rf_setdirmode:     */ NULL,
826     /* rf_setdirunixmode: */ NULL,
827     /* rf_setdirowner:    */ NULL,
828     /* rf_deletefile:     */ NULL,
829     /* rf_renamefile:     */ NULL,
830     /* vfs_copyfile       */ NULL,
831     /* rf_acl:            */ RF_posix_acl,
832     /* rf_remove_acl      */ RF_posix_remove_acl,
833     NULL
834 };
835 #endif
836
837 /* ---------------- */
838 void initvol_vfs(struct vol *vol)
839 {
840     vol->vfs = &vfs_master_funcs;
841
842     /* Default adouble stuff */
843     if (vol->v_adouble == AD_VERSION2) {
844         vol->vfs_modules[0] = &netatalk_adouble_v2;
845         vol->ad_path = ad_path;
846     } else {
847         vol->vfs_modules[0] = &netatalk_adouble_ea;
848 #ifdef HAVE_EAFD
849         vol->ad_path = ad_path_ea;
850 #else
851         vol->ad_path = ad_path_osx;
852 #endif
853     }
854
855     /* Extended Attributes */
856     if (vol->v_vfs_ea == AFPVOL_EA_SYS) {
857         LOG(log_debug, logtype_afpd, "initvol_vfs: enabling EA support with native EAs");
858         vol->vfs_modules[1] = &netatalk_ea_sys;
859     } else if (vol->v_vfs_ea == AFPVOL_EA_AD) {
860         LOG(log_debug, logtype_afpd, "initvol_vfs: enabling EA support with adouble files");
861         vol->vfs_modules[1] = &netatalk_ea_adouble;
862     } else {
863         LOG(log_debug, logtype_afpd, "initvol_vfs: volume without EA support");
864     }
865
866     /* ACLs */
867 #ifdef HAVE_SOLARIS_ACLS
868     vol->vfs_modules[2] = &netatalk_solaris_acl_adouble;
869 #endif
870 #ifdef HAVE_POSIX_ACLS
871     vol->vfs_modules[2] = &netatalk_posix_acl_adouble;
872 #endif
873
874 }