* GPL HEADER END
*/
/*
- * Copyright (c) 2011, 2012 Whamcloud, Inc.
+ * Copyright (c) 2012, 2013, Intel Corporation.
* Use is subject to license terms.
*
*/
/*
* Author: Brian Behlendorf <behlendorf1@llnl.gov>
*/
-
+#include "mount_utils.h"
#include <stdio.h>
#include <string.h>
-#include <libzfs/libzfs.h>
+#include <libzfs.h>
#include <dlfcn.h>
-#include "mount_utils.h"
+/* Persistent mount data is stored in these user attributes */
+#define LDD_PREFIX "lustre:"
+#define LDD_VERSION_PROP LDD_PREFIX "version"
+#define LDD_FLAGS_PROP LDD_PREFIX "flags"
+#define LDD_INDEX_PROP LDD_PREFIX "index"
+#define LDD_FSNAME_PROP LDD_PREFIX "fsname"
+#define LDD_SVNAME_PROP LDD_PREFIX "svname"
+#define LDD_UUID_PROP LDD_PREFIX "uuid"
+#define LDD_USERDATA_PROP LDD_PREFIX "userdata"
+#define LDD_MOUNTOPTS_PROP LDD_PREFIX "mountopts"
+
+/* This structure is used to help bridge the gap between the ZFS
+ * properties Lustre uses and their corresponding internal LDD fields.
+ * It is meant to be used internally by the mount utility only. */
+struct zfs_ldd_prop_bridge {
+ /* Contains the publicly visible name for the property
+ * (i.e. what is shown when running "zfs get") */
+ char *zlpb_prop_name;
+ /* Contains the offset into the lustre_disk_data structure where
+ * the value of this property is or will be stored. (i.e. the
+ * property is read from and written to this offset within ldd) */
+ int zlpb_ldd_offset;
+ /* Function pointer responsible for reading in the @prop
+ * property from @zhp and storing it in @ldd_field */
+ int (*zlpb_get_prop_fn)(zfs_handle_t *zhp, char *prop, void *ldd_field);
+ /* Function pointer responsible for writing the value of @ldd_field
+ * into the @prop dataset property in @zhp */
+ int (*zlpb_set_prop_fn)(zfs_handle_t *zhp, char *prop, void *ldd_field);
+};
-/* Persistent mount data is stored in these user attributes */
-#define LDD_VERSION_PROP "lustre:version"
-#define LDD_FLAGS_PROP "lustre:flags"
-#define LDD_INDEX_PROP "lustre:index"
-#define LDD_FSNAME_PROP "lustre:fsname"
-#define LDD_SVNAME_PROP "lustre:svname"
-#define LDD_UUID_PROP "lustre:uuid"
-#define LDD_USERDATA_PROP "lustre:userdata"
-#define LDD_MOUNTOPTS_PROP "lustre:mountopts"
-#define LDD_MGSNODE_PROP "lustre:mgsnode"
-#define LDD_FAILNODE_PROP "lustre:failnode"
-#define LDD_FAILMODE_PROP "lustre:failmode"
-#define LDD_IDENTITY_UPCALL_PROP "lustre:identity_upcall"
+/* Forward declarations needed to initialize the ldd prop bridge list */
+static int zfs_get_prop_int(zfs_handle_t *, char *, void *);
+static int zfs_set_prop_int(zfs_handle_t *, char *, void *);
+static int zfs_get_prop_str(zfs_handle_t *, char *, void *);
+static int zfs_set_prop_str(zfs_handle_t *, char *, void *);
+
+/* Helper for initializing the entries in the special_ldd_prop_params list.
+ * - @name: stored directly in the zlpb_prop_name field
+ * (e.g. lustre:fsname, lustre:version, etc.)
+ * - @field: the field in the lustre_disk_data which directly maps to
+ * the @name property. (e.g. ldd_fsname, ldd_config_ver, etc.)
+ * - @type: The type of @field. Only "int" and "str" are supported.
+ */
+#define ZLB_INIT(name, field, type) \
+{ \
+ name, offsetof(struct lustre_disk_data, field), \
+ zfs_get_prop_ ## type, zfs_set_prop_ ## type \
+}
+
+/* These ldd properties are special because they all have their own
+ * individual fields in the lustre_disk_data structure, as opposed to
+ * being globbed into the ldd_params field. As such, these need special
+ * handling when reading/writing the ldd structure to/from persistent
+ * storage. */
+struct zfs_ldd_prop_bridge special_ldd_prop_params[] = {
+ ZLB_INIT(LDD_VERSION_PROP, ldd_config_ver, int),
+ ZLB_INIT(LDD_FLAGS_PROP, ldd_flags, int),
+ ZLB_INIT(LDD_INDEX_PROP, ldd_svindex, int),
+ ZLB_INIT(LDD_FSNAME_PROP, ldd_fsname, str),
+ ZLB_INIT(LDD_SVNAME_PROP, ldd_svname, str),
+ ZLB_INIT(LDD_UUID_PROP, ldd_uuid, str),
+ ZLB_INIT(LDD_USERDATA_PROP, ldd_userdata, str),
+ ZLB_INIT(LDD_MOUNTOPTS_PROP, ldd_mount_opts, str),
+ { NULL }
+};
/* indicate if the ZFS OSD has been successfully setup */
static int osd_zfs_setup = 0;
int (*zfs_name_valid)(const char *, zfs_type_t);
zpool_handle_t* (*zpool_open)(libzfs_handle_t *, const char *);
void (*zpool_close)(zpool_handle_t *zhp);
- int (*nvlist_lookup_string)(nvlist_t*, const char*, char**);
+ int (*nvlist_lookup_string)(nvlist_t*, const char*, char**);
int (*nvlist_lookup_nvlist)(nvlist_t *, const char *, nvlist_t **);
+ nvpair_t * (*nvlist_next_nvpair)(nvlist_t *, nvpair_t *);
+ char * (*nvpair_name)(nvpair_t *);
};
static struct zfs_symbols sym;
#define nvlist_lookup_string (*sym.nvlist_lookup_string)
DLSYM(handle_nvpair, nvlist_lookup_nvlist);
#define nvlist_lookup_nvlist (*sym.nvlist_lookup_nvlist)
+ DLSYM(handle_nvpair, nvlist_next_nvpair);
+#define nvlist_next_nvpair (*sym.nvlist_next_nvpair)
+ DLSYM(handle_nvpair, nvpair_name);
+#define nvpair_name (*sym.nvpair_name)
error = dlerror();
if (error != NULL) {
return 0;
}
-static int zfs_set_prop_int(zfs_handle_t *zhp, char *prop, __u32 val)
+static int zfs_set_prop_int(zfs_handle_t *zhp, char *prop, void *val)
{
char str[64];
int ret;
- (void) snprintf(str, sizeof (str), "%lu", (unsigned long)val);
+ (void) snprintf(str, sizeof (str), "%i", *(int *)val);
vprint(" %s=%s\n", prop, str);
ret = zfs_prop_set(zhp, prop, str);
* Write the zfs property string, note that properties with a NULL or
* zero-length value will not be written and 0 returned.
*/
-static int zfs_set_prop_str(zfs_handle_t *zhp, char *prop, char *val)
+static int zfs_set_prop_str(zfs_handle_t *zhp, char *prop, void *val)
{
int ret = 0;
if (val && strlen(val) > 0) {
- vprint(" %s=%s\n", prop, val);
- ret = zfs_prop_set(zhp, prop, val);
+ vprint(" %s=%s\n", prop, (char *)val);
+ ret = zfs_prop_set(zhp, prop, (char *)val);
}
return ret;
}
-static int zfs_set_prop_param(zfs_handle_t *zhp, struct lustre_disk_data *ldd,
- char *param, char *prop)
+/*
+ * Map '<key>=<value> ...' pairs in the passed string to dataset properties
+ * of the form 'lustre:<key>=<value>'. Malformed <key>=<value> pairs will
+ * be skipped.
+ */
+static int zfs_set_prop_params(zfs_handle_t *zhp, char *params)
{
- char *str;
+ char *params_dup, *token, *key, *value;
+ char *save_token = NULL;
+ char prop_name[ZFS_MAXNAMELEN];
int ret = 0;
- if (get_param(ldd->ldd_params, param, &str) == 0) {
- vprint(" %s=%s\n", prop, str);
- ret = zfs_prop_set(zhp, prop, str);
- free(str);
+ params_dup = strdup(params);
+ if (params_dup == NULL)
+ return ENOMEM;
+
+ token = strtok_r(params_dup, " ", &save_token);
+ while (token) {
+ key = strtok(token, "=");
+ if (key == NULL)
+ continue;
+
+ value = strtok(NULL, "=");
+ if (value == NULL)
+ continue;
+
+ sprintf(prop_name, "%s%s", LDD_PREFIX, key);
+ vprint(" %s=%s\n", prop_name, value);
+
+ ret = zfs_prop_set(zhp, prop_name, value);
+ if (ret)
+ break;
+
+ token = strtok_r(NULL, " ", &save_token);
}
+ free(params_dup);
+
return ret;
}
struct lustre_disk_data *ldd = &mop->mo_ldd;
char *ds = mop->mo_device;
zfs_handle_t *zhp;
- int ret = EINVAL;
+ struct zfs_ldd_prop_bridge *bridge;
+ int i, ret = EINVAL;
if (osd_check_zfs_setup() == 0)
return EINVAL;
vprint("Writing %s properties\n", ds);
- ret = zfs_set_prop_int(zhp, LDD_VERSION_PROP, ldd->ldd_config_ver);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_int(zhp, LDD_FLAGS_PROP, ldd->ldd_flags);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_int(zhp, LDD_INDEX_PROP, ldd->ldd_svindex);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_str(zhp, LDD_FSNAME_PROP, ldd->ldd_fsname);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_str(zhp, LDD_SVNAME_PROP, ldd->ldd_svname);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_str(zhp, LDD_UUID_PROP, (char *)ldd->ldd_uuid);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_str(zhp, LDD_USERDATA_PROP, ldd->ldd_userdata);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_str(zhp, LDD_MOUNTOPTS_PROP, ldd->ldd_mount_opts);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_param(zhp, ldd, PARAM_MGSNODE, LDD_MGSNODE_PROP);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_param(zhp, ldd, PARAM_FAILNODE, LDD_FAILNODE_PROP);
- if (ret)
- goto out_close;
-
- ret = zfs_set_prop_param(zhp, ldd, PARAM_FAILMODE, LDD_FAILMODE_PROP);
- if (ret)
- goto out_close;
+ for (i = 0; special_ldd_prop_params[i].zlpb_prop_name != NULL; i++) {
+ bridge = &special_ldd_prop_params[i];
+ ret = bridge->zlpb_set_prop_fn(zhp, bridge->zlpb_prop_name,
+ (void *)ldd + bridge->zlpb_ldd_offset);
+ if (ret)
+ goto out_close;
+ }
- ret = zfs_set_prop_param(zhp, ldd, PARAM_MDT PARAM_ID_UPCALL,
- LDD_IDENTITY_UPCALL_PROP);
- if (ret)
- goto out_close;
+ ret = zfs_set_prop_params(zhp, ldd->ldd_params);
out_close:
zfs_close(zhp);
return ret;
}
-static int zfs_get_prop_int(zfs_handle_t *zhp, char *prop, __u32 *val)
+static int zfs_get_prop_int(zfs_handle_t *zhp, char *prop, void *val)
{
nvlist_t *propval;
char *propstr;
return ret;
errno = 0;
- *val = strtoul(propstr, NULL, 10);
+ *(__u32 *)val = strtoul(propstr, NULL, 10);
if (errno)
return errno;
return ret;
}
-static int zfs_get_prop_str(zfs_handle_t *zhp, char *prop, char *val)
+static int zfs_get_prop_str(zfs_handle_t *zhp, char *prop, void *val)
{
nvlist_t *propval;
char *propstr;
return ret;
}
-static int zfs_get_prop_param(zfs_handle_t *zhp, struct lustre_disk_data *ldd,
- char *param, char *prop)
+static int zfs_is_special_ldd_prop_param(char *name)
{
- nvlist_t *propval;
- char *propstr;
- int ret;
+ int i;
- ret = nvlist_lookup_nvlist(zfs_get_user_props(zhp), prop, &propval);
- if (ret)
- return ret;
+ for (i = 0; special_ldd_prop_params[i].zlpb_prop_name != NULL; i++)
+ if (!strcmp(name, special_ldd_prop_params[i].zlpb_prop_name))
+ return 1;
- ret = nvlist_lookup_string(propval, ZPROP_VALUE, &propstr);
- if (ret)
- return ret;
+ return 0;
+}
- ret = add_param(ldd->ldd_params, param, propstr);
+static int zfs_get_prop_params(zfs_handle_t *zhp, char *param, int len)
+{
+ nvlist_t *props;
+ nvpair_t *nvp;
+ char key[ZFS_MAXNAMELEN];
+ char *value;
+ int ret = 0;
+
+ props = zfs_get_user_props(zhp);
+ if (props == NULL)
+ return ENOENT;
+
+ value = malloc(len);
+ if (value == NULL)
+ return ENOMEM;
+
+ nvp = NULL;
+ while (nvp = nvlist_next_nvpair(props, nvp), nvp) {
+ ret = zfs_get_prop_str(zhp, nvpair_name(nvp), value);
+ if (ret)
+ break;
+
+ if (strncmp(nvpair_name(nvp), LDD_PREFIX, strlen(LDD_PREFIX)))
+ continue;
+
+ if (zfs_is_special_ldd_prop_param(nvpair_name(nvp)))
+ continue;
+
+ sprintf(key, "%s=", nvpair_name(nvp) + strlen(LDD_PREFIX));
+
+ ret = add_param(param, key, value);
+ if (ret)
+ break;
+ }
+
+ free(value);
return ret;
}
int zfs_read_ldd(char *ds, struct lustre_disk_data *ldd)
{
zfs_handle_t *zhp;
- int ret = EINVAL;
+ struct zfs_ldd_prop_bridge *bridge;
+ int i, ret = EINVAL;
if (osd_check_zfs_setup() == 0)
return EINVAL;
if (zhp == NULL)
goto out;
- ret = zfs_get_prop_int(zhp, LDD_VERSION_PROP, &ldd->ldd_config_ver);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_int(zhp, LDD_FLAGS_PROP, &ldd->ldd_flags);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_int(zhp, LDD_INDEX_PROP, &ldd->ldd_svindex);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_str(zhp, LDD_FSNAME_PROP, ldd->ldd_fsname);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_str(zhp, LDD_SVNAME_PROP, ldd->ldd_svname);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_str(zhp, LDD_UUID_PROP, (char *)ldd->ldd_uuid);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_str(zhp, LDD_USERDATA_PROP, ldd->ldd_userdata);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_str(zhp, LDD_MOUNTOPTS_PROP, ldd->ldd_mount_opts);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_param(zhp, ldd, PARAM_MGSNODE, LDD_MGSNODE_PROP);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_param(zhp, ldd, PARAM_FAILNODE, LDD_FAILNODE_PROP);
- if (ret && (ret != ENOENT))
- goto out_close;
-
- ret = zfs_get_prop_param(zhp, ldd, PARAM_FAILMODE, LDD_FAILMODE_PROP);
- if (ret && (ret != ENOENT))
- goto out_close;
+ for (i = 0; special_ldd_prop_params[i].zlpb_prop_name != NULL; i++) {
+ bridge = &special_ldd_prop_params[i];
+ ret = bridge->zlpb_get_prop_fn(zhp, bridge->zlpb_prop_name,
+ (void *)ldd + bridge->zlpb_ldd_offset);
+ if (ret && (ret != ENOENT))
+ goto out_close;
+ }
- ret = zfs_get_prop_param(zhp, ldd, PARAM_MDT PARAM_ID_UPCALL,
- LDD_IDENTITY_UPCALL_PROP);
+ ret = zfs_get_prop_params(zhp, ldd->ldd_params, 4096);
if (ret && (ret != ENOENT))
goto out_close;
ret = zfs_read_ldd(ds, &tmp_ldd);
if ((ret == 0) && (tmp_ldd.ldd_config_ver > 0) &&
- (strlen(tmp_ldd.ldd_fsname) > 0) &&
(strlen(tmp_ldd.ldd_svname) > 0)) {
*mount_type = tmp_ldd.ldd_mount_type;
return 1;
goto out;
}
- /* Due to zfs_name_valid() check the '/' must exist */
+ /* Due to zfs_prepare_lustre() check the '/' must exist */
strchr(pool, '/')[0] = '\0';
/* If --reformat was given attempt to destroy the previous dataset */
char *default_mountopts, int default_len,
char *always_mountopts, int always_len)
{
- int ret;
-
if (osd_check_zfs_setup() == 0)
return EINVAL;
- ret = zfs_name_valid(mop->mo_device, ZFS_TYPE_FILESYSTEM);
- if (!ret) {
+ if (zfs_name_valid(mop->mo_device, ZFS_TYPE_FILESYSTEM) == 0) {
fatal();
fprintf(stderr, "Invalid filesystem name %s\n", mop->mo_device);
return EINVAL;
}
+ if (strchr(mop->mo_device, '/') == NULL) {
+ fatal();
+ fprintf(stderr, "Missing pool in filesystem name %s\n",
+ mop->mo_device);
+ return EINVAL;
+ }
+
return 0;
}
return 0;
}
+int zfs_label_lustre(struct mount_opts *mop)
+{
+ zfs_handle_t *zhp;
+ int ret;
+
+ if (osd_check_zfs_setup() == 0)
+ return EINVAL;
+
+ zhp = zfs_open(g_zfs, mop->mo_source, ZFS_TYPE_FILESYSTEM);
+ if (zhp == NULL)
+ return EINVAL;
+
+ ret = zfs_set_prop_str(zhp, LDD_SVNAME_PROP, mop->mo_ldd.ldd_svname);
+ zfs_close(zhp);
+
+ return ret;
+}
+
int zfs_init(void)
{
int ret = 0;
* spamming ldiskfs users. An error message will still be printed if
* someone tries to do some real work involving a ZFS backend */
- handle_libzfs = dlopen("libzfs.so", RTLD_LAZY);
+ handle_libzfs = dlopen("libzfs.so.1", RTLD_LAZY);
if (handle_libzfs == NULL)
return EINVAL;
- handle_nvpair = dlopen("libnvpair.so", RTLD_LAZY);
+ handle_nvpair = dlopen("libnvpair.so.1", RTLD_LAZY);
if (handle_nvpair == NULL) {
ret = EINVAL;
goto out;