diff options
-rw-r--r-- | source3/modules/vfs_shadow_copy2.c | 62 |
1 files changed, 62 insertions, 0 deletions
diff --git a/source3/modules/vfs_shadow_copy2.c b/source3/modules/vfs_shadow_copy2.c index 61f71b7bbc..eb2e4fc70b 100644 --- a/source3/modules/vfs_shadow_copy2.c +++ b/source3/modules/vfs_shadow_copy2.c @@ -2,6 +2,7 @@ * implementation of an Shadow Copy module - version 2 * * Copyright (C) Andrew Tridgell 2007 + * Copyright (C) Ed Plese 2009 * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -34,6 +35,10 @@ from the original. This allows the 'restore' button to work without a sharing violation + 3) shadow copy results can be sorted before being sent to the + client. This is beneficial for filesystems that don't read + directories alphabetically (the default unix). + Module options: shadow:snapdir = <directory where snapshots are kept> @@ -58,6 +63,14 @@ don't set this option then the 'restore' button in the shadow copy UI will fail with a sharing violation. + shadow:sort = asc/desc, or not specified for unsorted (default) + + This is an optional parameter that specifies that the shadow + copy directories should be sorted before sending them to the + client. This can be beneficial as unix filesystems are usually + not listed alphabetically sorted. If enabled, you typically + want to specify descending order. + Note that the directory names in the snapshot directory must take the form @GMT-YYYY.MM.DD-HH.MM.SS @@ -73,6 +86,8 @@ static int vfs_shadow_copy2_debug_level = DBGC_VFS; #define GMT_NAME_LEN 24 /* length of a @GMT- name */ +#define SHADOW_COPY2_DEFAULT_SORT NULL + /* make very sure it is one of our special names */ @@ -711,6 +726,50 @@ static int shadow_copy2_chmod_acl(vfs_handle_struct *handle, SHADOW2_NEXT(CHMOD_ACL, (handle, name, mode), int, -1); } +static int shadow_copy2_label_cmp_asc(const void *x, const void *y) +{ + return strncmp((char *)x, (char *)y, sizeof(SHADOW_COPY_LABEL)); +} + +static int shadow_copy2_label_cmp_desc(const void *x, const void *y) +{ + return -strncmp((char *)x, (char *)y, sizeof(SHADOW_COPY_LABEL)); +} + +/* + sort the shadow copy data in ascending or descending order + */ +static void shadow_copy2_sort_data(vfs_handle_struct *handle, + SHADOW_COPY_DATA *shadow_copy2_data) +{ + int (*cmpfunc)(const void *, const void *); + const char *sort; + + sort = lp_parm_const_string(SNUM(handle->conn), "shadow", + "sort", SHADOW_COPY2_DEFAULT_SORT); + if (sort == NULL) { + return; + } + + if (strcmp(sort, "asc") == 0) { + cmpfunc = shadow_copy2_label_cmp_asc; + } else if (strcmp(sort, "desc") == 0) { + cmpfunc = shadow_copy2_label_cmp_desc; + } else { + return; + } + + if (shadow_copy2_data && shadow_copy2_data->num_volumes > 0 && + shadow_copy2_data->labels) + { + qsort(shadow_copy2_data->labels, + shadow_copy2_data->num_volumes, + sizeof(SHADOW_COPY_LABEL), cmpfunc); + } + + return; +} + static int shadow_copy2_get_shadow_copy2_data(vfs_handle_struct *handle, files_struct *fsp, SHADOW_COPY_DATA *shadow_copy2_data, @@ -774,6 +833,9 @@ static int shadow_copy2_get_shadow_copy2_data(vfs_handle_struct *handle, } SMB_VFS_NEXT_CLOSEDIR(handle,p); + + shadow_copy2_sort_data(handle, shadow_copy2_data); + return 0; } |