s4-netlogon: implement dcesrv_netr_DsRAddressToSitenamesExW
[Samba/aatanasov.git] / source3 / modules / vfs_shadow_copy2.c
blobeac83fca7499e01d7be883a0ecbe953919aee2f7
1 /*
2 * implementation of an Shadow Copy module - version 2
4 * Copyright (C) Andrew Tridgell 2007
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21 #include "includes.h"
25 This is a 2nd implemetation of a shadow copy module for exposing
26 snapshots to windows clients as shadow copies. This version has the
27 following features:
29 1) you don't need to populate your shares with symlinks to the
30 snapshots. This can be very important when you have thousands of
31 shares, or use [homes]
33 2) the inode number of the files is altered so it is different
34 from the original. This allows the 'restore' button to work
35 without a sharing violation
37 Module options:
39 shadow:snapdir = <directory where snapshots are kept>
41 This is the directory containing the @GMT-* snapshot directories. If it is an absolute
42 path it is used as-is. If it is a relative path, then it is taken relative to the mount
43 point of the filesystem that the root of this share is on
45 shadow:basedir = <base directory that snapshots are from>
47 This is an optional parameter that specifies the directory that
48 the snapshots are relative to. It defaults to the filesystem
49 mount point
51 shadow:fixinodes = yes/no
53 If you enable shadow:fixinodes then this module will modify the
54 apparent inode number of files in the snapshot directories using
55 a hash of the files path. This is needed for snapshot systems
56 where the snapshots have the same device:inode number as the
57 original files (such as happens with GPFS snapshots). If you
58 don't set this option then the 'restore' button in the shadow
59 copy UI will fail with a sharing violation.
61 Note that the directory names in the snapshot directory must take the form
62 @GMT-YYYY.MM.DD-HH.MM.SS
64 The following command would generate a correctly formatted directory name:
65 date -u +@GMT-%Y.%m.%d-%H.%M.%S
69 static int vfs_shadow_copy2_debug_level = DBGC_VFS;
71 #undef DBGC_CLASS
72 #define DBGC_CLASS vfs_shadow_copy2_debug_level
74 #define GMT_NAME_LEN 24 /* length of a @GMT- name */
77 make very sure it is one of our special names
79 static inline bool shadow_copy2_match_name(const char *name)
81 unsigned year, month, day, hr, min, sec;
82 if (name[0] != '@') return False;
83 if (strncmp(name, "@GMT-", 5) != 0) return False;
84 if (sscanf(name, "@GMT-%04u.%02u.%02u-%02u.%02u.%02u", &year, &month,
85 &day, &hr, &min, &sec) != 6) {
86 return False;
88 if (name[24] != 0 && name[24] != '/') {
89 return False;
91 return True;
95 convert a name to the shadow directory
98 #define _SHADOW2_NEXT(op, args, rtype, eret, extra) do { \
99 const char *name = fname; \
100 if (shadow_copy2_match_name(fname)) { \
101 char *name2; \
102 rtype ret; \
103 name2 = convert_shadow2_name(handle, fname); \
104 if (name2 == NULL) { \
105 errno = EINVAL; \
106 return eret; \
108 name = name2; \
109 ret = SMB_VFS_NEXT_ ## op args; \
110 talloc_free(name2); \
111 if (ret != eret) extra; \
112 return ret; \
113 } else { \
114 return SMB_VFS_NEXT_ ## op args; \
116 } while (0)
118 #define _SHADOW2_NEXT_SMB_FNAME(op, args, rtype, eret, extra) do { \
119 if (shadow_copy2_match_name(smb_fname->base_name)) { \
120 char *name2; \
121 char *smb_base_name_tmp = NULL; \
122 rtype ret; \
123 name2 = convert_shadow2_name(handle, smb_fname->base_name); \
124 if (name2 == NULL) { \
125 errno = EINVAL; \
126 return eret; \
128 smb_base_name_tmp = smb_fname->base_name; \
129 smb_fname->base_name = name2; \
130 ret = SMB_VFS_NEXT_ ## op args; \
131 smb_fname->base_name = smb_base_name_tmp; \
132 talloc_free(name2); \
133 if (ret != eret) extra; \
134 return ret; \
135 } else { \
136 return SMB_VFS_NEXT_ ## op args; \
138 } while (0)
141 convert a name to the shadow directory: NTSTATUS-specific handling
144 #define _SHADOW2_NTSTATUS_NEXT(op, args, eret, extra) do { \
145 const char *name = fname; \
146 if (shadow_copy2_match_name(fname)) { \
147 char *name2; \
148 NTSTATUS ret; \
149 name2 = convert_shadow2_name(handle, fname); \
150 if (name2 == NULL) { \
151 errno = EINVAL; \
152 return eret; \
154 name = name2; \
155 ret = SMB_VFS_NEXT_ ## op args; \
156 talloc_free(name2); \
157 if (!NT_STATUS_EQUAL(ret, eret)) extra; \
158 return ret; \
159 } else { \
160 return SMB_VFS_NEXT_ ## op args; \
162 } while (0)
164 #define SHADOW2_NTSTATUS_NEXT(op, args, eret) _SHADOW2_NTSTATUS_NEXT(op, args, eret, )
166 #define SHADOW2_NEXT(op, args, rtype, eret) _SHADOW2_NEXT(op, args, rtype, eret, )
168 #define SHADOW2_NEXT_SMB_FNAME(op, args, rtype, eret) _SHADOW2_NEXT_SMB_FNAME(op, args, rtype, eret, )
170 #define SHADOW2_NEXT2(op, args) do { \
171 if (shadow_copy2_match_name(oldname) || shadow_copy2_match_name(newname)) { \
172 errno = EROFS; \
173 return -1; \
174 } else { \
175 return SMB_VFS_NEXT_ ## op args; \
177 } while (0)
179 #define SHADOW2_NEXT2_SMB_FNAME(op, args) do { \
180 if (shadow_copy2_match_name(smb_fname_src->base_name) || \
181 shadow_copy2_match_name(smb_fname_dst->base_name)) { \
182 errno = EROFS; \
183 return -1; \
184 } else { \
185 return SMB_VFS_NEXT_ ## op args; \
187 } while (0)
191 find the mount point of a filesystem
193 static char *find_mount_point(TALLOC_CTX *mem_ctx, vfs_handle_struct *handle)
195 char *path = talloc_strdup(mem_ctx, handle->conn->connectpath);
196 dev_t dev;
197 struct stat st;
198 char *p;
200 if (stat(path, &st) != 0) {
201 talloc_free(path);
202 return NULL;
205 dev = st.st_dev;
207 while ((p = strrchr(path, '/')) && p > path) {
208 *p = 0;
209 if (stat(path, &st) != 0) {
210 talloc_free(path);
211 return NULL;
213 if (st.st_dev != dev) {
214 *p = '/';
215 break;
219 return path;
223 work out the location of the snapshot for this share
225 static const char *shadow_copy2_find_snapdir(TALLOC_CTX *mem_ctx, vfs_handle_struct *handle)
227 const char *snapdir;
228 char *mount_point;
229 const char *ret;
231 snapdir = lp_parm_const_string(SNUM(handle->conn), "shadow", "snapdir", NULL);
232 if (snapdir == NULL) {
233 return NULL;
235 /* if its an absolute path, we're done */
236 if (*snapdir == '/') {
237 return snapdir;
240 /* other its relative to the filesystem mount point */
241 mount_point = find_mount_point(mem_ctx, handle);
242 if (mount_point == NULL) {
243 return NULL;
246 ret = talloc_asprintf(mem_ctx, "%s/%s", mount_point, snapdir);
247 talloc_free(mount_point);
248 return ret;
252 work out the location of the base directory for snapshots of this share
254 static const char *shadow_copy2_find_basedir(TALLOC_CTX *mem_ctx, vfs_handle_struct *handle)
256 const char *basedir = lp_parm_const_string(SNUM(handle->conn), "shadow", "basedir", NULL);
258 /* other its the filesystem mount point */
259 if (basedir == NULL) {
260 basedir = find_mount_point(mem_ctx, handle);
263 return basedir;
267 convert a filename from a share relative path, to a path in the
268 snapshot directory
270 static char *convert_shadow2_name(vfs_handle_struct *handle, const char *fname)
272 TALLOC_CTX *tmp_ctx = talloc_new(handle->data);
273 const char *snapdir, *relpath, *baseoffset, *basedir;
274 size_t baselen;
275 char *ret;
277 snapdir = shadow_copy2_find_snapdir(tmp_ctx, handle);
278 if (snapdir == NULL) {
279 DEBUG(2,("no snapdir found for share at %s\n", handle->conn->connectpath));
280 talloc_free(tmp_ctx);
281 return NULL;
284 basedir = shadow_copy2_find_basedir(tmp_ctx, handle);
285 if (basedir == NULL) {
286 DEBUG(2,("no basedir found for share at %s\n", handle->conn->connectpath));
287 talloc_free(tmp_ctx);
288 return NULL;
291 relpath = fname + GMT_NAME_LEN;
292 baselen = strlen(basedir);
293 baseoffset = handle->conn->connectpath + baselen;
295 /* some sanity checks */
296 if (strncmp(basedir, handle->conn->connectpath, baselen) != 0 ||
297 (handle->conn->connectpath[baselen] != 0 && handle->conn->connectpath[baselen] != '/')) {
298 DEBUG(0,("convert_shadow2_name: basedir %s is not a parent of %s\n",
299 basedir, handle->conn->connectpath));
300 talloc_free(tmp_ctx);
301 return NULL;
304 if (*relpath == '/') relpath++;
305 if (*baseoffset == '/') baseoffset++;
307 ret = talloc_asprintf(handle->data, "%s/%.*s/%s/%s",
308 snapdir,
309 GMT_NAME_LEN, fname,
310 baseoffset,
311 relpath);
312 DEBUG(6,("convert_shadow2_name: '%s' -> '%s'\n", fname, ret));
313 talloc_free(tmp_ctx);
314 return ret;
319 simple string hash
321 static uint32 string_hash(const char *s)
323 uint32 n = 0;
324 while (*s) {
325 n = ((n << 5) + n) ^ (uint32)(*s++);
327 return n;
331 modify a sbuf return to ensure that inodes in the shadow directory
332 are different from those in the main directory
334 static void convert_sbuf(vfs_handle_struct *handle, const char *fname, SMB_STRUCT_STAT *sbuf)
336 if (lp_parm_bool(SNUM(handle->conn), "shadow", "fixinodes", False)) {
337 /* some snapshot systems, like GPFS, return the name
338 device:inode for the snapshot files as the current
339 files. That breaks the 'restore' button in the shadow copy
340 GUI, as the client gets a sharing violation.
342 This is a crude way of allowing both files to be
343 open at once. It has a slight chance of inode
344 number collision, but I can't see a better approach
345 without significant VFS changes
347 uint32_t shash = string_hash(fname) & 0xFF000000;
348 if (shash == 0) {
349 shash = 1;
351 sbuf->st_ex_ino ^= shash;
355 static int shadow_copy2_rename(vfs_handle_struct *handle,
356 const struct smb_filename *smb_fname_src,
357 const struct smb_filename *smb_fname_dst)
359 SHADOW2_NEXT2_SMB_FNAME(RENAME,
360 (handle, smb_fname_src, smb_fname_dst));
363 static int shadow_copy2_symlink(vfs_handle_struct *handle,
364 const char *oldname, const char *newname)
366 SHADOW2_NEXT2(SYMLINK, (handle, oldname, newname));
369 static int shadow_copy2_link(vfs_handle_struct *handle,
370 const char *oldname, const char *newname)
372 SHADOW2_NEXT2(LINK, (handle, oldname, newname));
375 static int shadow_copy2_open(vfs_handle_struct *handle,
376 struct smb_filename *smb_fname, files_struct *fsp,
377 int flags, mode_t mode)
379 SHADOW2_NEXT_SMB_FNAME(OPEN,
380 (handle, smb_fname, fsp, flags, mode),
381 int, -1);
384 static SMB_STRUCT_DIR *shadow_copy2_opendir(vfs_handle_struct *handle,
385 const char *fname, const char *mask, uint32 attr)
387 SHADOW2_NEXT(OPENDIR, (handle, name, mask, attr), SMB_STRUCT_DIR *, NULL);
390 static int shadow_copy2_stat(vfs_handle_struct *handle,
391 struct smb_filename *smb_fname)
393 _SHADOW2_NEXT_SMB_FNAME(STAT, (handle, smb_fname), int, -1,
394 convert_sbuf(handle, smb_fname->base_name,
395 &smb_fname->st));
398 static int shadow_copy2_lstat(vfs_handle_struct *handle,
399 struct smb_filename *smb_fname)
401 _SHADOW2_NEXT_SMB_FNAME(LSTAT, (handle, smb_fname), int, -1,
402 convert_sbuf(handle, smb_fname->base_name,
403 &smb_fname->st));
406 static int shadow_copy2_fstat(vfs_handle_struct *handle, files_struct *fsp, SMB_STRUCT_STAT *sbuf)
408 int ret = SMB_VFS_NEXT_FSTAT(handle, fsp, sbuf);
409 if (ret == 0 && shadow_copy2_match_name(fsp->fsp_name->base_name)) {
410 convert_sbuf(handle, fsp->fsp_name->base_name, sbuf);
412 return ret;
415 static int shadow_copy2_unlink(vfs_handle_struct *handle,
416 const struct smb_filename *smb_fname_in)
418 struct smb_filename *smb_fname = NULL;
419 NTSTATUS status;
421 status = copy_smb_filename(talloc_tos(), smb_fname_in, &smb_fname);
422 if (!NT_STATUS_IS_OK(status)) {
423 errno = map_errno_from_nt_status(status);
424 return -1;
427 SHADOW2_NEXT_SMB_FNAME(UNLINK, (handle, smb_fname), int, -1);
430 static int shadow_copy2_chmod(vfs_handle_struct *handle,
431 const char *fname, mode_t mode)
433 SHADOW2_NEXT(CHMOD, (handle, name, mode), int, -1);
436 static int shadow_copy2_chown(vfs_handle_struct *handle,
437 const char *fname, uid_t uid, gid_t gid)
439 SHADOW2_NEXT(CHOWN, (handle, name, uid, gid), int, -1);
442 static int shadow_copy2_chdir(vfs_handle_struct *handle,
443 const char *fname)
445 SHADOW2_NEXT(CHDIR, (handle, name), int, -1);
448 static int shadow_copy2_ntimes(vfs_handle_struct *handle,
449 const struct smb_filename *smb_fname_in,
450 struct smb_file_time *ft)
452 struct smb_filename *smb_fname = NULL;
453 NTSTATUS status;
455 status = copy_smb_filename(talloc_tos(), smb_fname_in, &smb_fname);
456 if (!NT_STATUS_IS_OK(status)) {
457 errno = map_errno_from_nt_status(status);
458 return -1;
461 SHADOW2_NEXT_SMB_FNAME(NTIMES, (handle, smb_fname, ft), int, -1);
464 static int shadow_copy2_readlink(vfs_handle_struct *handle,
465 const char *fname, char *buf, size_t bufsiz)
467 SHADOW2_NEXT(READLINK, (handle, name, buf, bufsiz), int, -1);
470 static int shadow_copy2_mknod(vfs_handle_struct *handle,
471 const char *fname, mode_t mode, SMB_DEV_T dev)
473 SHADOW2_NEXT(MKNOD, (handle, name, mode, dev), int, -1);
476 static char *shadow_copy2_realpath(vfs_handle_struct *handle,
477 const char *fname, char *resolved_path)
479 if (shadow_copy2_match_name(fname) && (fname[GMT_NAME_LEN] == '\0')) {
480 return SMB_VFS_NEXT_REALPATH(handle, ".", resolved_path);
482 SHADOW2_NEXT(REALPATH, (handle, name, resolved_path), char *, NULL);
485 static const char *shadow_copy2_connectpath(struct vfs_handle_struct *handle,
486 const char *fname)
488 TALLOC_CTX *tmp_ctx = talloc_stackframe();
489 const char *snapdir, *baseoffset, *basedir;
490 size_t baselen;
491 char *ret;
493 if (!shadow_copy2_match_name(fname)) {
494 return handle->conn->connectpath;
497 snapdir = shadow_copy2_find_snapdir(tmp_ctx, handle);
498 if (snapdir == NULL) {
499 DEBUG(2,("no snapdir found for share at %s\n",
500 handle->conn->connectpath));
501 TALLOC_FREE(tmp_ctx);
502 return NULL;
505 basedir = shadow_copy2_find_basedir(tmp_ctx, handle);
506 if (basedir == NULL) {
507 DEBUG(2,("no basedir found for share at %s\n",
508 handle->conn->connectpath));
509 TALLOC_FREE(tmp_ctx);
510 return NULL;
513 baselen = strlen(basedir);
514 baseoffset = handle->conn->connectpath + baselen;
516 /* some sanity checks */
517 if (strncmp(basedir, handle->conn->connectpath, baselen) != 0 ||
518 (handle->conn->connectpath[baselen] != 0
519 && handle->conn->connectpath[baselen] != '/')) {
520 DEBUG(0,("shadow_copy2_connectpath: basedir %s is not a "
521 "parent of %s\n", basedir,
522 handle->conn->connectpath));
523 TALLOC_FREE(tmp_ctx);
524 return NULL;
527 if (*baseoffset == '/') baseoffset++;
529 ret = talloc_asprintf(talloc_tos(), "%s/%.*s/%s",
530 snapdir,
531 GMT_NAME_LEN, fname,
532 baseoffset);
533 DEBUG(6,("shadow_copy2_connectpath: '%s' -> '%s'\n", fname, ret));
534 TALLOC_FREE(tmp_ctx);
535 return ret;
538 static NTSTATUS shadow_copy2_get_nt_acl(vfs_handle_struct *handle,
539 const char *fname, uint32 security_info,
540 struct security_descriptor **ppdesc)
542 SHADOW2_NTSTATUS_NEXT(GET_NT_ACL, (handle, name, security_info, ppdesc), NT_STATUS_ACCESS_DENIED);
545 static int shadow_copy2_mkdir(vfs_handle_struct *handle, const char *fname, mode_t mode)
547 SHADOW2_NEXT(MKDIR, (handle, name, mode), int, -1);
550 static int shadow_copy2_rmdir(vfs_handle_struct *handle, const char *fname)
552 SHADOW2_NEXT(RMDIR, (handle, name), int, -1);
555 static int shadow_copy2_chflags(vfs_handle_struct *handle, const char *fname,
556 unsigned int flags)
558 SHADOW2_NEXT(CHFLAGS, (handle, name, flags), int, -1);
561 static ssize_t shadow_copy2_getxattr(vfs_handle_struct *handle,
562 const char *fname, const char *aname, void *value, size_t size)
564 SHADOW2_NEXT(GETXATTR, (handle, name, aname, value, size), ssize_t, -1);
567 static ssize_t shadow_copy2_lgetxattr(vfs_handle_struct *handle,
568 const char *fname, const char *aname, void *value, size_t size)
570 SHADOW2_NEXT(LGETXATTR, (handle, name, aname, value, size), ssize_t, -1);
573 static ssize_t shadow_copy2_listxattr(struct vfs_handle_struct *handle, const char *fname,
574 char *list, size_t size)
576 SHADOW2_NEXT(LISTXATTR, (handle, name, list, size), ssize_t, -1);
579 static int shadow_copy2_removexattr(struct vfs_handle_struct *handle, const char *fname,
580 const char *aname)
582 SHADOW2_NEXT(REMOVEXATTR, (handle, name, aname), int, -1);
585 static int shadow_copy2_lremovexattr(struct vfs_handle_struct *handle, const char *fname,
586 const char *aname)
588 SHADOW2_NEXT(LREMOVEXATTR, (handle, name, aname), int, -1);
591 static int shadow_copy2_setxattr(struct vfs_handle_struct *handle, const char *fname,
592 const char *aname, const void *value, size_t size, int flags)
594 SHADOW2_NEXT(SETXATTR, (handle, name, aname, value, size, flags), int, -1);
597 static int shadow_copy2_lsetxattr(struct vfs_handle_struct *handle, const char *fname,
598 const char *aname, const void *value, size_t size, int flags)
600 SHADOW2_NEXT(LSETXATTR, (handle, name, aname, value, size, flags), int, -1);
603 static int shadow_copy2_chmod_acl(vfs_handle_struct *handle,
604 const char *fname, mode_t mode)
606 SHADOW2_NEXT(CHMOD_ACL, (handle, name, mode), int, -1);
609 static int shadow_copy2_get_shadow_copy2_data(vfs_handle_struct *handle,
610 files_struct *fsp,
611 SHADOW_COPY_DATA *shadow_copy2_data,
612 bool labels)
614 SMB_STRUCT_DIR *p;
615 const char *snapdir;
616 SMB_STRUCT_DIRENT *d;
617 TALLOC_CTX *tmp_ctx = talloc_new(handle->data);
619 snapdir = shadow_copy2_find_snapdir(tmp_ctx, handle);
620 if (snapdir == NULL) {
621 DEBUG(0,("shadow:snapdir not found for %s in get_shadow_copy_data\n",
622 handle->conn->connectpath));
623 errno = EINVAL;
624 talloc_free(tmp_ctx);
625 return -1;
628 p = SMB_VFS_NEXT_OPENDIR(handle, snapdir, NULL, 0);
630 if (!p) {
631 DEBUG(2,("shadow_copy2: SMB_VFS_NEXT_OPENDIR() failed for '%s'"
632 " - %s\n", snapdir, strerror(errno)));
633 talloc_free(tmp_ctx);
634 errno = ENOSYS;
635 return -1;
638 talloc_free(tmp_ctx);
640 shadow_copy2_data->num_volumes = 0;
641 shadow_copy2_data->labels = NULL;
643 while ((d = SMB_VFS_NEXT_READDIR(handle, p, NULL))) {
644 SHADOW_COPY_LABEL *tlabels;
646 /* ignore names not of the right form in the snapshot directory */
647 if (!shadow_copy2_match_name(d->d_name)) {
648 continue;
651 if (!labels) {
652 /* the caller doesn't want the labels */
653 shadow_copy2_data->num_volumes++;
654 continue;
657 tlabels = talloc_realloc(shadow_copy2_data->mem_ctx,
658 shadow_copy2_data->labels,
659 SHADOW_COPY_LABEL, shadow_copy2_data->num_volumes+1);
660 if (tlabels == NULL) {
661 DEBUG(0,("shadow_copy2: out of memory\n"));
662 SMB_VFS_NEXT_CLOSEDIR(handle, p);
663 return -1;
666 strlcpy(tlabels[shadow_copy2_data->num_volumes], d->d_name, sizeof(*tlabels));
667 shadow_copy2_data->num_volumes++;
668 shadow_copy2_data->labels = tlabels;
671 SMB_VFS_NEXT_CLOSEDIR(handle,p);
672 return 0;
675 static struct vfs_fn_pointers vfs_shadow_copy2_fns = {
676 .opendir = shadow_copy2_opendir,
677 .mkdir = shadow_copy2_mkdir,
678 .rmdir = shadow_copy2_rmdir,
679 .chflags = shadow_copy2_chflags,
680 .getxattr = shadow_copy2_getxattr,
681 .lgetxattr = shadow_copy2_lgetxattr,
682 .listxattr = shadow_copy2_listxattr,
683 .removexattr = shadow_copy2_removexattr,
684 .lremovexattr = shadow_copy2_lremovexattr,
685 .setxattr = shadow_copy2_setxattr,
686 .lsetxattr = shadow_copy2_lsetxattr,
687 .open = shadow_copy2_open,
688 .rename = shadow_copy2_rename,
689 .stat = shadow_copy2_stat,
690 .lstat = shadow_copy2_lstat,
691 .fstat = shadow_copy2_fstat,
692 .unlink = shadow_copy2_unlink,
693 .chmod = shadow_copy2_chmod,
694 .chown = shadow_copy2_chown,
695 .chdir = shadow_copy2_chdir,
696 .ntimes = shadow_copy2_ntimes,
697 .symlink = shadow_copy2_symlink,
698 .vfs_readlink = shadow_copy2_readlink,
699 .link = shadow_copy2_link,
700 .mknod = shadow_copy2_mknod,
701 .realpath = shadow_copy2_realpath,
702 .connectpath = shadow_copy2_connectpath,
703 .get_nt_acl = shadow_copy2_get_nt_acl,
704 .chmod_acl = shadow_copy2_chmod_acl,
705 .get_shadow_copy_data = shadow_copy2_get_shadow_copy2_data,
708 NTSTATUS vfs_shadow_copy2_init(void);
709 NTSTATUS vfs_shadow_copy2_init(void)
711 NTSTATUS ret;
713 ret = smb_register_vfs(SMB_VFS_INTERFACE_VERSION, "shadow_copy2",
714 &vfs_shadow_copy2_fns);
716 if (!NT_STATUS_IS_OK(ret))
717 return ret;
719 vfs_shadow_copy2_debug_level = debug_add_class("shadow_copy2");
720 if (vfs_shadow_copy2_debug_level == -1) {
721 vfs_shadow_copy2_debug_level = DBGC_VFS;
722 DEBUG(0, ("%s: Couldn't register custom debugging class!\n",
723 "vfs_shadow_copy2_init"));
724 } else {
725 DEBUG(10, ("%s: Debug class number of '%s': %d\n",
726 "vfs_shadow_copy2_init","shadow_copy2",vfs_shadow_copy2_debug_level));
729 return ret;