libzfs_impl.h revision 247540
1/* 2 * CDDL HEADER SART 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22/* 23 * Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved. 24 * Copyright (c) 2011 Pawel Jakub Dawidek <pawel@dawidek.net>. 25 * All rights reserved. 26 * Copyright (c) 2011 by Delphix. All rights reserved. 27 * Copyright (c) 2013 Martin Matuska <mm@FreeBSD.org>. All rights reserved. 28 */ 29 30#ifndef _LIBFS_IMPL_H 31#define _LIBFS_IMPL_H 32 33#include <sys/dmu.h> 34#include <sys/fs/zfs.h> 35#include <sys/zfs_ioctl.h> 36#include <sys/spa.h> 37#include <sys/nvpair.h> 38 39#include <libshare.h> 40#include <libuutil.h> 41#include <libzfs.h> 42 43#include "zfs_ioctl_compat.h" 44 45#ifdef __cplusplus 46extern "C" { 47#endif 48 49#ifdef VERIFY 50#undef VERIFY 51#endif 52#define VERIFY verify 53 54typedef struct libzfs_fru { 55 char *zf_device; 56 char *zf_fru; 57 struct libzfs_fru *zf_chain; 58 struct libzfs_fru *zf_next; 59} libzfs_fru_t; 60 61struct libzfs_handle { 62 int libzfs_error; 63 int libzfs_fd; 64 FILE *libzfs_mnttab; 65 FILE *libzfs_sharetab; 66 zpool_handle_t *libzfs_pool_handles; 67 uu_avl_pool_t *libzfs_ns_avlpool; 68 uu_avl_t *libzfs_ns_avl; 69 uint64_t libzfs_ns_gen; 70 int libzfs_desc_active; 71 char libzfs_action[1024]; 72 char libzfs_desc[1024]; 73 char *libzfs_log_str; 74 int libzfs_printerr; 75 int libzfs_storeerr; /* stuff error messages into buffer */ 76 void *libzfs_sharehdl; /* libshare handle */ 77 uint_t libzfs_shareflags; 78 boolean_t libzfs_mnttab_enable; 79 avl_tree_t libzfs_mnttab_cache; 80 int libzfs_pool_iter; 81 libzfs_fru_t **libzfs_fru_hash; 82 libzfs_fru_t *libzfs_fru_list; 83 char libzfs_chassis_id[256]; 84}; 85 86#define ZFSSHARE_MISS 0x01 /* Didn't find entry in cache */ 87 88struct zfs_handle { 89 libzfs_handle_t *zfs_hdl; 90 zpool_handle_t *zpool_hdl; 91 char zfs_name[ZFS_MAXNAMELEN]; 92 zfs_type_t zfs_type; /* type including snapshot */ 93 zfs_type_t zfs_head_type; /* type excluding snapshot */ 94 dmu_objset_stats_t zfs_dmustats; 95 nvlist_t *zfs_props; 96 nvlist_t *zfs_user_props; 97 nvlist_t *zfs_recvd_props; 98 boolean_t zfs_mntcheck; 99 char *zfs_mntopts; 100 uint8_t *zfs_props_table; 101}; 102 103/* 104 * This is different from checking zfs_type, because it will also catch 105 * snapshots of volumes. 106 */ 107#define ZFS_IS_VOLUME(zhp) ((zhp)->zfs_head_type == ZFS_TYPE_VOLUME) 108 109struct zpool_handle { 110 libzfs_handle_t *zpool_hdl; 111 zpool_handle_t *zpool_next; 112 char zpool_name[ZPOOL_MAXNAMELEN]; 113 int zpool_state; 114 size_t zpool_config_size; 115 nvlist_t *zpool_config; 116 nvlist_t *zpool_old_config; 117 nvlist_t *zpool_props; 118 diskaddr_t zpool_start_block; 119}; 120 121typedef enum { 122 PROTO_NFS = 0, 123 PROTO_SMB = 1, 124 PROTO_END = 2 125} zfs_share_proto_t; 126 127/* 128 * The following can be used as a bitmask and any new values 129 * added must preserve that capability. 130 */ 131typedef enum { 132 SHARED_NOT_SHARED = 0x0, 133 SHARED_NFS = 0x2, 134 SHARED_SMB = 0x4 135} zfs_share_type_t; 136 137int zfs_error(libzfs_handle_t *, int, const char *); 138int zfs_error_fmt(libzfs_handle_t *, int, const char *, ...); 139void zfs_error_aux(libzfs_handle_t *, const char *, ...); 140void *zfs_alloc(libzfs_handle_t *, size_t); 141void *zfs_realloc(libzfs_handle_t *, void *, size_t, size_t); 142char *zfs_asprintf(libzfs_handle_t *, const char *, ...); 143char *zfs_strdup(libzfs_handle_t *, const char *); 144int no_memory(libzfs_handle_t *); 145 146int zfs_standard_error(libzfs_handle_t *, int, const char *); 147int zfs_standard_error_fmt(libzfs_handle_t *, int, const char *, ...); 148int zpool_standard_error(libzfs_handle_t *, int, const char *); 149int zpool_standard_error_fmt(libzfs_handle_t *, int, const char *, ...); 150 151int get_dependents(libzfs_handle_t *, boolean_t, const char *, char ***, 152 size_t *); 153zfs_handle_t *make_dataset_handle_zc(libzfs_handle_t *, zfs_cmd_t *); 154zfs_handle_t *make_dataset_simple_handle_zc(zfs_handle_t *, zfs_cmd_t *); 155 156int zprop_parse_value(libzfs_handle_t *, nvpair_t *, int, zfs_type_t, 157 nvlist_t *, char **, uint64_t *, const char *); 158int zprop_expand_list(libzfs_handle_t *hdl, zprop_list_t **plp, 159 zfs_type_t type); 160 161/* 162 * Use this changelist_gather() flag to force attempting mounts 163 * on each change node regardless of whether or not it is currently 164 * mounted. 165 */ 166#define CL_GATHER_MOUNT_ALWAYS 0x01 167/* 168 * Use this changelist_gather() flag to prevent unmounting of file systems. 169 */ 170#define CL_GATHER_DONT_UNMOUNT 0x02 171 172typedef struct prop_changelist prop_changelist_t; 173 174int zcmd_alloc_dst_nvlist(libzfs_handle_t *, zfs_cmd_t *, size_t); 175int zcmd_write_src_nvlist(libzfs_handle_t *, zfs_cmd_t *, nvlist_t *); 176int zcmd_write_conf_nvlist(libzfs_handle_t *, zfs_cmd_t *, nvlist_t *); 177int zcmd_expand_dst_nvlist(libzfs_handle_t *, zfs_cmd_t *); 178int zcmd_read_dst_nvlist(libzfs_handle_t *, zfs_cmd_t *, nvlist_t **); 179void zcmd_free_nvlists(zfs_cmd_t *); 180 181int changelist_prefix(prop_changelist_t *); 182int changelist_postfix(prop_changelist_t *); 183void changelist_rename(prop_changelist_t *, const char *, const char *); 184void changelist_remove(prop_changelist_t *, const char *); 185void changelist_free(prop_changelist_t *); 186prop_changelist_t *changelist_gather(zfs_handle_t *, zfs_prop_t, int, int); 187int changelist_unshare(prop_changelist_t *, zfs_share_proto_t *); 188int changelist_haszonedchild(prop_changelist_t *); 189 190void remove_mountpoint(zfs_handle_t *); 191int create_parents(libzfs_handle_t *, char *, int); 192boolean_t isa_child_of(const char *dataset, const char *parent); 193 194zfs_handle_t *make_dataset_handle(libzfs_handle_t *, const char *); 195 196int zpool_open_silent(libzfs_handle_t *, const char *, zpool_handle_t **); 197 198boolean_t zpool_name_valid(libzfs_handle_t *, boolean_t, const char *); 199 200int zfs_validate_name(libzfs_handle_t *hdl, const char *path, int type, 201 boolean_t modifying); 202 203void namespace_clear(libzfs_handle_t *); 204 205/* 206 * libshare (sharemgr) interfaces used internally. 207 */ 208 209extern int zfs_init_libshare(libzfs_handle_t *, int); 210extern void zfs_uninit_libshare(libzfs_handle_t *); 211extern int zfs_parse_options(char *, zfs_share_proto_t); 212 213extern int zfs_unshare_proto(zfs_handle_t *, 214 const char *, zfs_share_proto_t *); 215 216extern void libzfs_fru_clear(libzfs_handle_t *, boolean_t); 217 218#ifndef sun 219static int zfs_kernel_version = 0; 220static int zfs_ioctl_version = 0; 221 222/* 223 * This is FreeBSD version of ioctl, because Solaris' ioctl() updates 224 * zc_nvlist_dst_size even if an error is returned, on FreeBSD if an 225 * error is returned zc_nvlist_dst_size won't be updated. 226 */ 227static __inline int 228zcmd_ioctl(int fd, unsigned long cmd, zfs_cmd_t *zc) 229{ 230 size_t oldsize, zfs_kernel_version_size, zfs_ioctl_version_size; 231 int version, ret, cflag = ZFS_CMD_COMPAT_NONE; 232 233 zfs_ioctl_version_size = sizeof(zfs_ioctl_version); 234 if (zfs_ioctl_version == 0) { 235 sysctlbyname("vfs.zfs.version.ioctl", &zfs_ioctl_version, 236 &zfs_ioctl_version_size, NULL, 0); 237 } 238 239 /* 240 * If vfs.zfs.version.ioctl is not defined, assume we have v28 241 * compatible binaries and use vfs.zfs.version.spa to test for v15 242 */ 243 if (zfs_ioctl_version < ZFS_IOCVER_DEADMAN) { 244 cflag = ZFS_CMD_COMPAT_V28; 245 zfs_kernel_version_size = sizeof(zfs_kernel_version); 246 247 if (zfs_kernel_version == 0) { 248 sysctlbyname("vfs.zfs.version.spa", 249 &zfs_kernel_version, 250 &zfs_kernel_version_size, NULL, 0); 251 } 252 253 if (zfs_kernel_version == SPA_VERSION_15 || 254 zfs_kernel_version == SPA_VERSION_14 || 255 zfs_kernel_version == SPA_VERSION_13) 256 cflag = ZFS_CMD_COMPAT_V15; 257 } 258 259 oldsize = zc->zc_nvlist_dst_size; 260 ret = zcmd_ioctl_compat(fd, cmd, zc, cflag); 261 262 if (ret == 0 && oldsize < zc->zc_nvlist_dst_size) { 263 ret = -1; 264 errno = ENOMEM; 265 } 266 267 return (ret); 268} 269#define ioctl(fd, cmd, zc) zcmd_ioctl((fd), (cmd), (zc)) 270#endif /* !sun */ 271 272#ifdef __cplusplus 273} 274#endif 275 276#endif /* _LIBFS_IMPL_H */ 277