1*789Sahrens /* 2*789Sahrens * CDDL HEADER START 3*789Sahrens * 4*789Sahrens * The contents of this file are subject to the terms of the 5*789Sahrens * Common Development and Distribution License, Version 1.0 only 6*789Sahrens * (the "License"). You may not use this file except in compliance 7*789Sahrens * with the License. 8*789Sahrens * 9*789Sahrens * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10*789Sahrens * or http://www.opensolaris.org/os/licensing. 11*789Sahrens * See the License for the specific language governing permissions 12*789Sahrens * and limitations under the License. 13*789Sahrens * 14*789Sahrens * When distributing Covered Code, include this CDDL HEADER in each 15*789Sahrens * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16*789Sahrens * If applicable, add the following below this CDDL HEADER, with the 17*789Sahrens * fields enclosed by brackets "[]" replaced with your own identifying 18*789Sahrens * information: Portions Copyright [yyyy] [name of copyright owner] 19*789Sahrens * 20*789Sahrens * CDDL HEADER END 21*789Sahrens */ 22*789Sahrens /* 23*789Sahrens * Copyright 2005 Sun Microsystems, Inc. All rights reserved. 24*789Sahrens * Use is subject to license terms. 25*789Sahrens */ 26*789Sahrens 27*789Sahrens #pragma ident "%Z%%M% %I% %E% SMI" 28*789Sahrens 29*789Sahrens #include <sys/types.h> 30*789Sahrens #include <sys/param.h> 31*789Sahrens #include <sys/errno.h> 32*789Sahrens #include <sys/uio.h> 33*789Sahrens #include <sys/buf.h> 34*789Sahrens #include <sys/modctl.h> 35*789Sahrens #include <sys/open.h> 36*789Sahrens #include <sys/file.h> 37*789Sahrens #include <sys/kmem.h> 38*789Sahrens #include <sys/conf.h> 39*789Sahrens #include <sys/cmn_err.h> 40*789Sahrens #include <sys/stat.h> 41*789Sahrens #include <sys/zfs_ioctl.h> 42*789Sahrens #include <sys/zap.h> 43*789Sahrens #include <sys/spa.h> 44*789Sahrens #include <sys/vdev.h> 45*789Sahrens #include <sys/dmu.h> 46*789Sahrens #include <sys/dsl_dir.h> 47*789Sahrens #include <sys/dsl_dataset.h> 48*789Sahrens #include <sys/dsl_prop.h> 49*789Sahrens #include <sys/ddi.h> 50*789Sahrens #include <sys/sunddi.h> 51*789Sahrens #include <sys/sunldi.h> 52*789Sahrens #include <sys/policy.h> 53*789Sahrens #include <sys/zone.h> 54*789Sahrens #include <sys/nvpair.h> 55*789Sahrens #include <sys/pathname.h> 56*789Sahrens #include <sys/mount.h> 57*789Sahrens #include <sys/sdt.h> 58*789Sahrens #include <sys/fs/zfs.h> 59*789Sahrens #include <sys/zfs_ctldir.h> 60*789Sahrens 61*789Sahrens #include "zfs_namecheck.h" 62*789Sahrens 63*789Sahrens extern struct modlfs zfs_modlfs; 64*789Sahrens 65*789Sahrens extern void zfs_init(void); 66*789Sahrens extern void zfs_fini(void); 67*789Sahrens 68*789Sahrens ldi_ident_t zfs_li = NULL; 69*789Sahrens dev_info_t *zfs_dip; 70*789Sahrens 71*789Sahrens typedef int zfs_ioc_func_t(zfs_cmd_t *); 72*789Sahrens typedef int zfs_secpolicy_func_t(const char *, const char *, cred_t *); 73*789Sahrens 74*789Sahrens typedef struct zfs_ioc_vec { 75*789Sahrens zfs_ioc_func_t *zvec_func; 76*789Sahrens zfs_secpolicy_func_t *zvec_secpolicy; 77*789Sahrens enum { 78*789Sahrens no_name, 79*789Sahrens pool_name, 80*789Sahrens dataset_name 81*789Sahrens } zvec_namecheck; 82*789Sahrens } zfs_ioc_vec_t; 83*789Sahrens 84*789Sahrens /* _NOTE(PRINTFLIKE(4)) - this is printf-like, but lint is too whiney */ 85*789Sahrens void 86*789Sahrens __dprintf(const char *file, const char *func, int line, const char *fmt, ...) 87*789Sahrens { 88*789Sahrens const char *newfile; 89*789Sahrens char buf[256]; 90*789Sahrens va_list adx; 91*789Sahrens 92*789Sahrens /* 93*789Sahrens * Get rid of annoying "../common/" prefix to filename. 94*789Sahrens */ 95*789Sahrens newfile = strrchr(file, '/'); 96*789Sahrens if (newfile != NULL) { 97*789Sahrens newfile = newfile + 1; /* Get rid of leading / */ 98*789Sahrens } else { 99*789Sahrens newfile = file; 100*789Sahrens } 101*789Sahrens 102*789Sahrens va_start(adx, fmt); 103*789Sahrens (void) vsnprintf(buf, sizeof (buf), fmt, adx); 104*789Sahrens va_end(adx); 105*789Sahrens 106*789Sahrens /* 107*789Sahrens * To get this data, use the zfs-dprintf probe as so: 108*789Sahrens * dtrace -q -n 'zfs-dprintf \ 109*789Sahrens * /stringof(arg0) == "dbuf.c"/ \ 110*789Sahrens * {printf("%s: %s", stringof(arg1), stringof(arg3))}' 111*789Sahrens * arg0 = file name 112*789Sahrens * arg1 = function name 113*789Sahrens * arg2 = line number 114*789Sahrens * arg3 = message 115*789Sahrens */ 116*789Sahrens DTRACE_PROBE4(zfs__dprintf, 117*789Sahrens char *, newfile, char *, func, int, line, char *, buf); 118*789Sahrens } 119*789Sahrens 120*789Sahrens /* 121*789Sahrens * Policy for top-level read operations (list pools). Requires no privileges, 122*789Sahrens * and can be used in the local zone, as there is no associated dataset. 123*789Sahrens */ 124*789Sahrens /* ARGSUSED */ 125*789Sahrens static int 126*789Sahrens zfs_secpolicy_none(const char *unused1, const char *unused2, cred_t *cr) 127*789Sahrens { 128*789Sahrens return (0); 129*789Sahrens } 130*789Sahrens 131*789Sahrens /* 132*789Sahrens * Policy for dataset read operations (list children, get statistics). Requires 133*789Sahrens * no privileges, but must be visible in the local zone. 134*789Sahrens */ 135*789Sahrens /* ARGSUSED */ 136*789Sahrens static int 137*789Sahrens zfs_secpolicy_read(const char *dataset, const char *unused, cred_t *cr) 138*789Sahrens { 139*789Sahrens if (INGLOBALZONE(curproc) || 140*789Sahrens zone_dataset_visible(dataset, NULL)) 141*789Sahrens return (0); 142*789Sahrens 143*789Sahrens return (ENOENT); 144*789Sahrens } 145*789Sahrens 146*789Sahrens static int 147*789Sahrens zfs_dozonecheck(const char *dataset, cred_t *cr) 148*789Sahrens { 149*789Sahrens uint64_t zoned; 150*789Sahrens int writable = 1; 151*789Sahrens 152*789Sahrens /* 153*789Sahrens * The dataset must be visible by this zone -- check this first 154*789Sahrens * so they don't see EPERM on something they shouldn't know about. 155*789Sahrens */ 156*789Sahrens if (!INGLOBALZONE(curproc) && 157*789Sahrens !zone_dataset_visible(dataset, &writable)) 158*789Sahrens return (ENOENT); 159*789Sahrens 160*789Sahrens if (dsl_prop_get_integer(dataset, "zoned", &zoned, NULL)) 161*789Sahrens return (ENOENT); 162*789Sahrens 163*789Sahrens if (INGLOBALZONE(curproc)) { 164*789Sahrens /* 165*789Sahrens * If the fs is zoned, only root can access it from the 166*789Sahrens * global zone. 167*789Sahrens */ 168*789Sahrens if (secpolicy_zfs(cr) && zoned) 169*789Sahrens return (EPERM); 170*789Sahrens } else { 171*789Sahrens /* 172*789Sahrens * If we are in a local zone, the 'zoned' property must be set. 173*789Sahrens */ 174*789Sahrens if (!zoned) 175*789Sahrens return (EPERM); 176*789Sahrens 177*789Sahrens /* must be writable by this zone */ 178*789Sahrens if (!writable) 179*789Sahrens return (EPERM); 180*789Sahrens } 181*789Sahrens return (0); 182*789Sahrens } 183*789Sahrens 184*789Sahrens /* 185*789Sahrens * Policy for dataset write operations (create children, set properties, etc). 186*789Sahrens * Requires SYS_MOUNT privilege, and must be writable in the local zone. 187*789Sahrens */ 188*789Sahrens /* ARGSUSED */ 189*789Sahrens int 190*789Sahrens zfs_secpolicy_write(const char *dataset, const char *unused, cred_t *cr) 191*789Sahrens { 192*789Sahrens int error; 193*789Sahrens 194*789Sahrens if (error = zfs_dozonecheck(dataset, cr)) 195*789Sahrens return (error); 196*789Sahrens 197*789Sahrens return (secpolicy_zfs(cr)); 198*789Sahrens } 199*789Sahrens 200*789Sahrens /* 201*789Sahrens * Policy for operations that want to write a dataset's parent: 202*789Sahrens * create, destroy, snapshot, clone, restore. 203*789Sahrens */ 204*789Sahrens static int 205*789Sahrens zfs_secpolicy_parent(const char *dataset, const char *unused, cred_t *cr) 206*789Sahrens { 207*789Sahrens char parentname[MAXNAMELEN]; 208*789Sahrens char *cp; 209*789Sahrens 210*789Sahrens /* 211*789Sahrens * Remove the @bla or /bla from the end of the name to get the parent. 212*789Sahrens */ 213*789Sahrens (void) strncpy(parentname, dataset, sizeof (parentname)); 214*789Sahrens cp = strrchr(parentname, '@'); 215*789Sahrens if (cp != NULL) { 216*789Sahrens cp[0] = '\0'; 217*789Sahrens } else { 218*789Sahrens cp = strrchr(parentname, '/'); 219*789Sahrens if (cp == NULL) 220*789Sahrens return (ENOENT); 221*789Sahrens cp[0] = '\0'; 222*789Sahrens 223*789Sahrens } 224*789Sahrens 225*789Sahrens return (zfs_secpolicy_write(parentname, unused, cr)); 226*789Sahrens } 227*789Sahrens 228*789Sahrens /* 229*789Sahrens * Policy for dataset write operations (create children, set properties, etc). 230*789Sahrens * Requires SYS_MOUNT privilege, and must be writable in the local zone. 231*789Sahrens */ 232*789Sahrens static int 233*789Sahrens zfs_secpolicy_setprop(const char *dataset, const char *prop, cred_t *cr) 234*789Sahrens { 235*789Sahrens int error; 236*789Sahrens 237*789Sahrens if (error = zfs_dozonecheck(dataset, cr)) 238*789Sahrens return (error); 239*789Sahrens 240*789Sahrens if (strcmp(prop, "zoned") == 0) { 241*789Sahrens /* 242*789Sahrens * Disallow setting of 'zoned' from within a local zone. 243*789Sahrens */ 244*789Sahrens if (!INGLOBALZONE(curproc)) 245*789Sahrens return (EPERM); 246*789Sahrens } 247*789Sahrens 248*789Sahrens return (secpolicy_zfs(cr)); 249*789Sahrens } 250*789Sahrens 251*789Sahrens /* 252*789Sahrens * Security policy for setting the quota. This is the same as 253*789Sahrens * zfs_secpolicy_write, except that the local zone may not change the quota at 254*789Sahrens * the zone-property setpoint. 255*789Sahrens */ 256*789Sahrens /* ARGSUSED */ 257*789Sahrens static int 258*789Sahrens zfs_secpolicy_quota(const char *dataset, const char *unused, cred_t *cr) 259*789Sahrens { 260*789Sahrens int error; 261*789Sahrens 262*789Sahrens if (error = zfs_dozonecheck(dataset, cr)) 263*789Sahrens return (error); 264*789Sahrens 265*789Sahrens if (!INGLOBALZONE(curproc)) { 266*789Sahrens uint64_t zoned; 267*789Sahrens char setpoint[MAXNAMELEN]; 268*789Sahrens int dslen; 269*789Sahrens /* 270*789Sahrens * Unprivileged users are allowed to modify the quota 271*789Sahrens * on things *under* (ie. contained by) the thing they 272*789Sahrens * own. 273*789Sahrens */ 274*789Sahrens if (dsl_prop_get_integer(dataset, "zoned", &zoned, setpoint)) 275*789Sahrens return (EPERM); 276*789Sahrens if (!zoned) /* this shouldn't happen */ 277*789Sahrens return (EPERM); 278*789Sahrens dslen = strlen(dataset); 279*789Sahrens if (dslen <= strlen(setpoint)) 280*789Sahrens return (EPERM); 281*789Sahrens } 282*789Sahrens 283*789Sahrens return (secpolicy_zfs(cr)); 284*789Sahrens } 285*789Sahrens 286*789Sahrens /* 287*789Sahrens * Policy for pool operations - create/destroy pools, add vdevs, etc. Requires 288*789Sahrens * SYS_CONFIG privilege, which is not available in a local zone. 289*789Sahrens */ 290*789Sahrens /* ARGSUSED */ 291*789Sahrens static int 292*789Sahrens zfs_secpolicy_config(const char *unused, const char *unused2, cred_t *cr) 293*789Sahrens { 294*789Sahrens if (secpolicy_sys_config(cr, B_FALSE) != 0) 295*789Sahrens return (EPERM); 296*789Sahrens 297*789Sahrens return (0); 298*789Sahrens } 299*789Sahrens 300*789Sahrens /* 301*789Sahrens * Returns the nvlist as specified by the user in the zfs_cmd_t. 302*789Sahrens */ 303*789Sahrens static int 304*789Sahrens get_config(zfs_cmd_t *zc, nvlist_t **nvp) 305*789Sahrens { 306*789Sahrens char *packed; 307*789Sahrens size_t size; 308*789Sahrens int error; 309*789Sahrens nvlist_t *config = NULL; 310*789Sahrens 311*789Sahrens /* 312*789Sahrens * Read in and unpack the user-supplied nvlist. By this point, we know 313*789Sahrens * that the user has the SYS_CONFIG privilege, so allocating arbitrary 314*789Sahrens * sized regions of memory should not be a problem. 315*789Sahrens */ 316*789Sahrens if ((size = zc->zc_config_src_size) == 0) 317*789Sahrens return (EINVAL); 318*789Sahrens 319*789Sahrens packed = kmem_alloc(size, KM_SLEEP); 320*789Sahrens 321*789Sahrens if ((error = xcopyin((void *)(uintptr_t)zc->zc_config_src, packed, 322*789Sahrens size)) != 0) { 323*789Sahrens kmem_free(packed, size); 324*789Sahrens return (error); 325*789Sahrens } 326*789Sahrens 327*789Sahrens if ((error = nvlist_unpack(packed, size, &config, 0)) != 0) { 328*789Sahrens kmem_free(packed, size); 329*789Sahrens return (error); 330*789Sahrens } 331*789Sahrens 332*789Sahrens kmem_free(packed, size); 333*789Sahrens 334*789Sahrens *nvp = config; 335*789Sahrens return (0); 336*789Sahrens } 337*789Sahrens 338*789Sahrens static int 339*789Sahrens zfs_ioc_pool_create(zfs_cmd_t *zc) 340*789Sahrens { 341*789Sahrens int error; 342*789Sahrens nvlist_t *config; 343*789Sahrens 344*789Sahrens if ((error = get_config(zc, &config)) != 0) 345*789Sahrens return (error); 346*789Sahrens 347*789Sahrens error = spa_create(zc->zc_name, config, zc->zc_root[0] == '\0' ? 348*789Sahrens NULL : zc->zc_root); 349*789Sahrens 350*789Sahrens nvlist_free(config); 351*789Sahrens 352*789Sahrens return (error); 353*789Sahrens } 354*789Sahrens 355*789Sahrens static int 356*789Sahrens zfs_ioc_pool_destroy(zfs_cmd_t *zc) 357*789Sahrens { 358*789Sahrens return (spa_destroy(zc->zc_name)); 359*789Sahrens } 360*789Sahrens 361*789Sahrens static int 362*789Sahrens zfs_ioc_pool_import(zfs_cmd_t *zc) 363*789Sahrens { 364*789Sahrens int error; 365*789Sahrens nvlist_t *config; 366*789Sahrens uint64_t guid; 367*789Sahrens 368*789Sahrens if ((error = get_config(zc, &config)) != 0) 369*789Sahrens return (error); 370*789Sahrens 371*789Sahrens if (nvlist_lookup_uint64(config, ZPOOL_CONFIG_POOL_GUID, &guid) != 0 || 372*789Sahrens guid != zc->zc_pool_guid) 373*789Sahrens error = EINVAL; 374*789Sahrens else 375*789Sahrens error = spa_import(zc->zc_name, config, 376*789Sahrens zc->zc_root[0] == '\0' ? NULL : zc->zc_root); 377*789Sahrens 378*789Sahrens nvlist_free(config); 379*789Sahrens 380*789Sahrens return (error); 381*789Sahrens } 382*789Sahrens 383*789Sahrens static int 384*789Sahrens zfs_ioc_pool_export(zfs_cmd_t *zc) 385*789Sahrens { 386*789Sahrens return (spa_export(zc->zc_name)); 387*789Sahrens } 388*789Sahrens 389*789Sahrens static int 390*789Sahrens zfs_ioc_pool_configs(zfs_cmd_t *zc) 391*789Sahrens { 392*789Sahrens nvlist_t *configs; 393*789Sahrens char *packed = NULL; 394*789Sahrens size_t size = 0; 395*789Sahrens int error; 396*789Sahrens 397*789Sahrens if ((configs = spa_all_configs(&zc->zc_cookie)) == NULL) 398*789Sahrens return (EEXIST); 399*789Sahrens 400*789Sahrens VERIFY(nvlist_pack(configs, &packed, &size, NV_ENCODE_NATIVE, 0) == 0); 401*789Sahrens 402*789Sahrens if (size > zc->zc_config_dst_size) 403*789Sahrens error = ENOMEM; 404*789Sahrens else 405*789Sahrens error = xcopyout(packed, (void *)(uintptr_t)zc->zc_config_dst, 406*789Sahrens size); 407*789Sahrens 408*789Sahrens zc->zc_config_dst_size = size; 409*789Sahrens 410*789Sahrens kmem_free(packed, size); 411*789Sahrens nvlist_free(configs); 412*789Sahrens 413*789Sahrens return (error); 414*789Sahrens } 415*789Sahrens 416*789Sahrens static int 417*789Sahrens zfs_ioc_pool_guid(zfs_cmd_t *zc) 418*789Sahrens { 419*789Sahrens spa_t *spa; 420*789Sahrens int error; 421*789Sahrens 422*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 423*789Sahrens if (error == 0) { 424*789Sahrens zc->zc_pool_guid = spa_guid(spa); 425*789Sahrens spa_close(spa, FTAG); 426*789Sahrens } 427*789Sahrens return (error); 428*789Sahrens } 429*789Sahrens 430*789Sahrens static int 431*789Sahrens zfs_ioc_pool_stats(zfs_cmd_t *zc) 432*789Sahrens { 433*789Sahrens nvlist_t *config; 434*789Sahrens char *packed = NULL; 435*789Sahrens size_t size = 0; 436*789Sahrens int error; 437*789Sahrens 438*789Sahrens error = spa_get_stats(zc->zc_name, &config); 439*789Sahrens 440*789Sahrens if (config != NULL) { 441*789Sahrens VERIFY(nvlist_pack(config, &packed, &size, 442*789Sahrens NV_ENCODE_NATIVE, 0) == 0); 443*789Sahrens 444*789Sahrens if (size > zc->zc_config_dst_size) 445*789Sahrens error = ENOMEM; 446*789Sahrens else if (xcopyout(packed, (void *)(uintptr_t)zc->zc_config_dst, 447*789Sahrens size)) 448*789Sahrens error = EFAULT; 449*789Sahrens 450*789Sahrens zc->zc_config_dst_size = size; 451*789Sahrens 452*789Sahrens kmem_free(packed, size); 453*789Sahrens nvlist_free(config); 454*789Sahrens } else { 455*789Sahrens ASSERT(error != 0); 456*789Sahrens } 457*789Sahrens 458*789Sahrens return (error); 459*789Sahrens } 460*789Sahrens 461*789Sahrens /* 462*789Sahrens * Try to import the given pool, returning pool stats as appropriate so that 463*789Sahrens * user land knows which devices are available and overall pool health. 464*789Sahrens */ 465*789Sahrens static int 466*789Sahrens zfs_ioc_pool_tryimport(zfs_cmd_t *zc) 467*789Sahrens { 468*789Sahrens nvlist_t *tryconfig, *config; 469*789Sahrens char *packed = NULL; 470*789Sahrens size_t size = 0; 471*789Sahrens int error; 472*789Sahrens 473*789Sahrens if ((error = get_config(zc, &tryconfig)) != 0) 474*789Sahrens return (error); 475*789Sahrens 476*789Sahrens config = spa_tryimport(tryconfig); 477*789Sahrens 478*789Sahrens nvlist_free(tryconfig); 479*789Sahrens 480*789Sahrens if (config == NULL) 481*789Sahrens return (EINVAL); 482*789Sahrens 483*789Sahrens VERIFY(nvlist_pack(config, &packed, &size, NV_ENCODE_NATIVE, 0) == 0); 484*789Sahrens 485*789Sahrens if (size > zc->zc_config_dst_size) 486*789Sahrens error = ENOMEM; 487*789Sahrens else 488*789Sahrens error = xcopyout(packed, (void *)(uintptr_t)zc->zc_config_dst, 489*789Sahrens size); 490*789Sahrens 491*789Sahrens zc->zc_config_dst_size = size; 492*789Sahrens 493*789Sahrens kmem_free(packed, size); 494*789Sahrens nvlist_free(config); 495*789Sahrens 496*789Sahrens return (error); 497*789Sahrens } 498*789Sahrens 499*789Sahrens static int 500*789Sahrens zfs_ioc_pool_scrub(zfs_cmd_t *zc) 501*789Sahrens { 502*789Sahrens spa_t *spa; 503*789Sahrens int error; 504*789Sahrens 505*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 506*789Sahrens if (error == 0) { 507*789Sahrens error = spa_scrub(spa, zc->zc_cookie, B_FALSE); 508*789Sahrens spa_close(spa, FTAG); 509*789Sahrens } 510*789Sahrens return (error); 511*789Sahrens } 512*789Sahrens 513*789Sahrens static int 514*789Sahrens zfs_ioc_pool_freeze(zfs_cmd_t *zc) 515*789Sahrens { 516*789Sahrens spa_t *spa; 517*789Sahrens int error; 518*789Sahrens 519*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 520*789Sahrens if (error == 0) { 521*789Sahrens spa_freeze(spa); 522*789Sahrens spa_close(spa, FTAG); 523*789Sahrens } 524*789Sahrens return (error); 525*789Sahrens } 526*789Sahrens 527*789Sahrens static int 528*789Sahrens zfs_ioc_vdev_add(zfs_cmd_t *zc) 529*789Sahrens { 530*789Sahrens spa_t *spa; 531*789Sahrens int error; 532*789Sahrens nvlist_t *config; 533*789Sahrens 534*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 535*789Sahrens if (error != 0) 536*789Sahrens return (error); 537*789Sahrens 538*789Sahrens if ((error = get_config(zc, &config)) == 0) { 539*789Sahrens error = spa_vdev_add(spa, config); 540*789Sahrens nvlist_free(config); 541*789Sahrens } 542*789Sahrens 543*789Sahrens spa_close(spa, FTAG); 544*789Sahrens return (error); 545*789Sahrens } 546*789Sahrens 547*789Sahrens /* ARGSUSED */ 548*789Sahrens static int 549*789Sahrens zfs_ioc_vdev_remove(zfs_cmd_t *zc) 550*789Sahrens { 551*789Sahrens return (ENOTSUP); 552*789Sahrens } 553*789Sahrens 554*789Sahrens static int 555*789Sahrens zfs_ioc_vdev_online(zfs_cmd_t *zc) 556*789Sahrens { 557*789Sahrens spa_t *spa; 558*789Sahrens char *path = zc->zc_prop_value; 559*789Sahrens int error; 560*789Sahrens 561*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 562*789Sahrens if (error != 0) 563*789Sahrens return (error); 564*789Sahrens error = vdev_online(spa, path); 565*789Sahrens spa_close(spa, FTAG); 566*789Sahrens return (error); 567*789Sahrens } 568*789Sahrens 569*789Sahrens static int 570*789Sahrens zfs_ioc_vdev_offline(zfs_cmd_t *zc) 571*789Sahrens { 572*789Sahrens spa_t *spa; 573*789Sahrens char *path = zc->zc_prop_value; 574*789Sahrens int error; 575*789Sahrens 576*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 577*789Sahrens if (error != 0) 578*789Sahrens return (error); 579*789Sahrens error = vdev_offline(spa, path); 580*789Sahrens spa_close(spa, FTAG); 581*789Sahrens return (error); 582*789Sahrens } 583*789Sahrens 584*789Sahrens static int 585*789Sahrens zfs_ioc_vdev_attach(zfs_cmd_t *zc) 586*789Sahrens { 587*789Sahrens spa_t *spa; 588*789Sahrens char *path = zc->zc_prop_value; 589*789Sahrens int replacing = zc->zc_cookie; 590*789Sahrens nvlist_t *config; 591*789Sahrens int error; 592*789Sahrens 593*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 594*789Sahrens if (error != 0) 595*789Sahrens return (error); 596*789Sahrens 597*789Sahrens if ((error = get_config(zc, &config)) == 0) { 598*789Sahrens error = spa_vdev_attach(spa, path, config, replacing); 599*789Sahrens nvlist_free(config); 600*789Sahrens } 601*789Sahrens 602*789Sahrens spa_close(spa, FTAG); 603*789Sahrens return (error); 604*789Sahrens } 605*789Sahrens 606*789Sahrens static int 607*789Sahrens zfs_ioc_vdev_detach(zfs_cmd_t *zc) 608*789Sahrens { 609*789Sahrens spa_t *spa; 610*789Sahrens char *path = zc->zc_prop_value; 611*789Sahrens int error; 612*789Sahrens 613*789Sahrens error = spa_open(zc->zc_name, &spa, FTAG); 614*789Sahrens if (error != 0) 615*789Sahrens return (error); 616*789Sahrens 617*789Sahrens error = spa_vdev_detach(spa, path, 0, B_FALSE); 618*789Sahrens 619*789Sahrens spa_close(spa, FTAG); 620*789Sahrens return (error); 621*789Sahrens } 622*789Sahrens 623*789Sahrens static int 624*789Sahrens zfs_get_stats(zfs_cmd_t *zc) 625*789Sahrens { 626*789Sahrens char *name = zc->zc_name; 627*789Sahrens zfs_stats_t *zs = &zc->zc_zfs_stats; 628*789Sahrens int error; 629*789Sahrens 630*789Sahrens bzero(zs, sizeof (zfs_stats_t)); 631*789Sahrens 632*789Sahrens if ((error = dsl_prop_get_integer(name, "atime", 633*789Sahrens &zs->zs_atime, zs->zs_atime_setpoint)) != 0 || 634*789Sahrens (error = dsl_prop_get_integer(name, "recordsize", 635*789Sahrens &zs->zs_recordsize, zs->zs_recordsize_setpoint)) != 0 || 636*789Sahrens (error = dsl_prop_get_integer(name, "readonly", 637*789Sahrens &zs->zs_readonly, zs->zs_readonly_setpoint)) != 0 || 638*789Sahrens (error = dsl_prop_get_integer(name, "devices", 639*789Sahrens &zs->zs_devices, zs->zs_devices_setpoint)) != 0 || 640*789Sahrens (error = dsl_prop_get_integer(name, "setuid", 641*789Sahrens &zs->zs_setuid, zs->zs_setuid_setpoint)) != 0 || 642*789Sahrens (error = dsl_prop_get_integer(name, "exec", 643*789Sahrens &zs->zs_exec, zs->zs_exec_setpoint)) != 0 || 644*789Sahrens (error = dsl_prop_get_string(name, "mountpoint", zs->zs_mountpoint, 645*789Sahrens sizeof (zs->zs_mountpoint), zs->zs_mountpoint_setpoint)) != 0 || 646*789Sahrens (error = dsl_prop_get_string(name, "sharenfs", zs->zs_sharenfs, 647*789Sahrens sizeof (zs->zs_sharenfs), zs->zs_sharenfs_setpoint)) != 0 || 648*789Sahrens (error = dsl_prop_get_integer(name, "aclmode", 649*789Sahrens &zs->zs_acl_mode, zs->zs_acl_mode_setpoint)) != 0 || 650*789Sahrens (error = dsl_prop_get_integer(name, "snapdir", 651*789Sahrens &zs->zs_snapdir, zs->zs_snapdir_setpoint)) != 0 || 652*789Sahrens (error = dsl_prop_get_integer(name, "aclinherit", 653*789Sahrens &zs->zs_acl_inherit, zs->zs_acl_inherit_setpoint)) != 0) 654*789Sahrens return (error); 655*789Sahrens 656*789Sahrens return (0); 657*789Sahrens } 658*789Sahrens 659*789Sahrens static int 660*789Sahrens zfs_ioc_objset_stats(zfs_cmd_t *zc) 661*789Sahrens { 662*789Sahrens objset_t *os = NULL; 663*789Sahrens int error; 664*789Sahrens 665*789Sahrens retry: 666*789Sahrens error = dmu_objset_open(zc->zc_name, DMU_OST_ANY, 667*789Sahrens DS_MODE_STANDARD | DS_MODE_READONLY, &os); 668*789Sahrens if (error != 0) { 669*789Sahrens /* 670*789Sahrens * This is ugly: dmu_objset_open() can return EBUSY if 671*789Sahrens * the objset is held exclusively. Fortunately this hold is 672*789Sahrens * only for a short while, so we retry here. 673*789Sahrens * This avoids user code having to handle EBUSY, 674*789Sahrens * for example for a "zfs list". 675*789Sahrens */ 676*789Sahrens if (error == EBUSY) { 677*789Sahrens delay(1); 678*789Sahrens goto retry; 679*789Sahrens } 680*789Sahrens return (error); 681*789Sahrens } 682*789Sahrens 683*789Sahrens dmu_objset_stats(os, &zc->zc_objset_stats); 684*789Sahrens 685*789Sahrens switch (zc->zc_objset_stats.dds_type) { 686*789Sahrens 687*789Sahrens case DMU_OST_ZFS: 688*789Sahrens error = zfs_get_stats(zc); 689*789Sahrens break; 690*789Sahrens 691*789Sahrens case DMU_OST_ZVOL: 692*789Sahrens error = zvol_get_stats(zc, os); 693*789Sahrens break; 694*789Sahrens } 695*789Sahrens 696*789Sahrens dmu_objset_close(os); 697*789Sahrens return (error); 698*789Sahrens } 699*789Sahrens 700*789Sahrens static int 701*789Sahrens zfs_ioc_dataset_list_next(zfs_cmd_t *zc) 702*789Sahrens { 703*789Sahrens dsl_dir_t *dd; 704*789Sahrens zap_cursor_t cursor; 705*789Sahrens zap_attribute_t attr; 706*789Sahrens int error; 707*789Sahrens char *p; 708*789Sahrens 709*789Sahrens dd = dsl_dir_open(zc->zc_name, FTAG, NULL); 710*789Sahrens if (dd == NULL) 711*789Sahrens return (ESRCH); 712*789Sahrens 713*789Sahrens if (dd->dd_phys->dd_child_dir_zapobj == 0) { 714*789Sahrens dsl_dir_close(dd, FTAG); 715*789Sahrens return (ESRCH); 716*789Sahrens } 717*789Sahrens 718*789Sahrens p = strrchr(zc->zc_name, '/'); 719*789Sahrens if (p == NULL || p[1] != '\0') 720*789Sahrens (void) strlcat(zc->zc_name, "/", sizeof (zc->zc_name)); 721*789Sahrens p = zc->zc_name + strlen(zc->zc_name); 722*789Sahrens 723*789Sahrens do { 724*789Sahrens zap_cursor_init_serialized(&cursor, dd->dd_pool->dp_meta_objset, 725*789Sahrens dd->dd_phys->dd_child_dir_zapobj, zc->zc_cookie); 726*789Sahrens 727*789Sahrens error = zap_cursor_retrieve(&cursor, &attr); 728*789Sahrens if (error == ENOENT) 729*789Sahrens error = ESRCH; 730*789Sahrens if (error != 0) { 731*789Sahrens dsl_dir_close(dd, FTAG); 732*789Sahrens *p = '\0'; 733*789Sahrens return (error); 734*789Sahrens } 735*789Sahrens 736*789Sahrens (void) strlcpy(p, attr.za_name, sizeof (zc->zc_name) - 737*789Sahrens (p - zc->zc_name)); 738*789Sahrens 739*789Sahrens zap_cursor_advance(&cursor); 740*789Sahrens zc->zc_cookie = zap_cursor_serialize(&cursor); 741*789Sahrens 742*789Sahrens } while (!INGLOBALZONE(curproc) && 743*789Sahrens !zone_dataset_visible(zc->zc_name, NULL)); 744*789Sahrens 745*789Sahrens dsl_dir_close(dd, FTAG); 746*789Sahrens 747*789Sahrens /* 748*789Sahrens * If it's a hidden dataset, don't try to get stats for it. 749*789Sahrens * User land will skip over it. 750*789Sahrens */ 751*789Sahrens if (strchr(zc->zc_name, '$') != NULL) 752*789Sahrens return (0); 753*789Sahrens 754*789Sahrens error = zfs_ioc_objset_stats(zc); /* will just fill in the stats */ 755*789Sahrens return (error); 756*789Sahrens } 757*789Sahrens 758*789Sahrens static int 759*789Sahrens zfs_ioc_snapshot_list_next(zfs_cmd_t *zc) 760*789Sahrens { 761*789Sahrens zap_cursor_t cursor; 762*789Sahrens zap_attribute_t attr; 763*789Sahrens dsl_dataset_t *ds; 764*789Sahrens int error; 765*789Sahrens 766*789Sahrens retry: 767*789Sahrens error = dsl_dataset_open(zc->zc_name, 768*789Sahrens DS_MODE_STANDARD | DS_MODE_READONLY, FTAG, &ds); 769*789Sahrens if (error) { 770*789Sahrens /* 771*789Sahrens * This is ugly: dsl_dataset_open() can return EBUSY if 772*789Sahrens * the objset is held exclusively. Fortunately this hold is 773*789Sahrens * only for a short while, so we retry here. 774*789Sahrens * This avoids user code having to handle EBUSY, 775*789Sahrens * for example for a "zfs list -s". 776*789Sahrens */ 777*789Sahrens if (error == EBUSY) { 778*789Sahrens delay(1); 779*789Sahrens goto retry; 780*789Sahrens } 781*789Sahrens if (error == ENOENT) 782*789Sahrens return (ESRCH); 783*789Sahrens return (error); 784*789Sahrens } 785*789Sahrens 786*789Sahrens /* 787*789Sahrens * If ds_snapnames_zapobj is 0, someone is trying to iterate over 788*789Sahrens * snapshots of a snapshot. In this case, pretend that it has no 789*789Sahrens * snapshots; otherwise zap_cursor_retrieve() will blow up. 790*789Sahrens */ 791*789Sahrens if (ds->ds_phys->ds_snapnames_zapobj == 0) { 792*789Sahrens error = ESRCH; 793*789Sahrens goto out; 794*789Sahrens } 795*789Sahrens 796*789Sahrens zap_cursor_init_serialized(&cursor, 797*789Sahrens ds->ds_dir->dd_pool->dp_meta_objset, 798*789Sahrens ds->ds_phys->ds_snapnames_zapobj, zc->zc_cookie); 799*789Sahrens 800*789Sahrens error = zap_cursor_retrieve(&cursor, &attr); 801*789Sahrens if (error == ENOENT) 802*789Sahrens error = ESRCH; 803*789Sahrens if (error != 0) 804*789Sahrens goto out; 805*789Sahrens 806*789Sahrens if (strlcat(zc->zc_name, "@", sizeof (zc->zc_name)) >= 807*789Sahrens sizeof (zc->zc_name) || 808*789Sahrens strlcat(zc->zc_name, attr.za_name, sizeof (zc->zc_name)) >= 809*789Sahrens sizeof (zc->zc_name)) { 810*789Sahrens error = ENAMETOOLONG; 811*789Sahrens goto out; 812*789Sahrens } 813*789Sahrens 814*789Sahrens zap_cursor_advance(&cursor); 815*789Sahrens zc->zc_cookie = zap_cursor_serialize(&cursor); 816*789Sahrens 817*789Sahrens error = zfs_ioc_objset_stats(zc); /* will just fill in the stats */ 818*789Sahrens 819*789Sahrens out: 820*789Sahrens dsl_dataset_close(ds, DS_MODE_STANDARD, FTAG); 821*789Sahrens return (error); 822*789Sahrens } 823*789Sahrens 824*789Sahrens static int 825*789Sahrens zfs_ioc_set_prop(zfs_cmd_t *zc) 826*789Sahrens { 827*789Sahrens return (dsl_prop_set(zc->zc_name, zc->zc_prop_name, 828*789Sahrens zc->zc_intsz, zc->zc_numints, zc->zc_prop_value)); 829*789Sahrens } 830*789Sahrens 831*789Sahrens static int 832*789Sahrens zfs_ioc_set_quota(zfs_cmd_t *zc) 833*789Sahrens { 834*789Sahrens return (dsl_dir_set_quota(zc->zc_name, zc->zc_cookie)); 835*789Sahrens } 836*789Sahrens 837*789Sahrens static int 838*789Sahrens zfs_ioc_set_reservation(zfs_cmd_t *zc) 839*789Sahrens { 840*789Sahrens return (dsl_dir_set_reservation(zc->zc_name, zc->zc_cookie)); 841*789Sahrens } 842*789Sahrens 843*789Sahrens static int 844*789Sahrens zfs_ioc_set_volsize(zfs_cmd_t *zc) 845*789Sahrens { 846*789Sahrens return (zvol_set_volsize(zc)); 847*789Sahrens } 848*789Sahrens 849*789Sahrens static int 850*789Sahrens zfs_ioc_set_volblocksize(zfs_cmd_t *zc) 851*789Sahrens { 852*789Sahrens return (zvol_set_volblocksize(zc)); 853*789Sahrens } 854*789Sahrens 855*789Sahrens static int 856*789Sahrens zfs_ioc_create_minor(zfs_cmd_t *zc) 857*789Sahrens { 858*789Sahrens return (zvol_create_minor(zc)); 859*789Sahrens } 860*789Sahrens 861*789Sahrens static int 862*789Sahrens zfs_ioc_remove_minor(zfs_cmd_t *zc) 863*789Sahrens { 864*789Sahrens return (zvol_remove_minor(zc)); 865*789Sahrens } 866*789Sahrens 867*789Sahrens /* 868*789Sahrens * Search the vfs list for a specified resource. Returns a pointer to it 869*789Sahrens * or NULL if no suitable entry is found. The caller of this routine 870*789Sahrens * is responsible for releasing the returned vfs pointer. 871*789Sahrens */ 872*789Sahrens static vfs_t * 873*789Sahrens zfs_get_vfs(const char *resource) 874*789Sahrens { 875*789Sahrens struct vfs *vfsp; 876*789Sahrens struct vfs *vfs_found = NULL; 877*789Sahrens 878*789Sahrens vfs_list_read_lock(); 879*789Sahrens vfsp = rootvfs; 880*789Sahrens do { 881*789Sahrens if (strcmp(refstr_value(vfsp->vfs_resource), resource) == 0) { 882*789Sahrens VFS_HOLD(vfsp); 883*789Sahrens vfs_found = vfsp; 884*789Sahrens break; 885*789Sahrens } 886*789Sahrens vfsp = vfsp->vfs_next; 887*789Sahrens } while (vfsp != rootvfs); 888*789Sahrens vfs_list_unlock(); 889*789Sahrens return (vfs_found); 890*789Sahrens } 891*789Sahrens 892*789Sahrens static void 893*789Sahrens zfs_create_cb(objset_t *os, void *arg, dmu_tx_t *tx) 894*789Sahrens { 895*789Sahrens zfs_cmd_t *zc = arg; 896*789Sahrens zfs_create_fs(os, (cred_t *)(uintptr_t)zc->zc_cred, tx); 897*789Sahrens } 898*789Sahrens 899*789Sahrens static int 900*789Sahrens zfs_ioc_create(zfs_cmd_t *zc) 901*789Sahrens { 902*789Sahrens objset_t *clone; 903*789Sahrens int error = 0; 904*789Sahrens void (*cbfunc)(objset_t *os, void *arg, dmu_tx_t *tx); 905*789Sahrens dmu_objset_type_t type = zc->zc_objset_type; 906*789Sahrens 907*789Sahrens switch (type) { 908*789Sahrens 909*789Sahrens case DMU_OST_ZFS: 910*789Sahrens cbfunc = zfs_create_cb; 911*789Sahrens break; 912*789Sahrens 913*789Sahrens case DMU_OST_ZVOL: 914*789Sahrens cbfunc = zvol_create_cb; 915*789Sahrens break; 916*789Sahrens 917*789Sahrens default: 918*789Sahrens return (EINVAL); 919*789Sahrens } 920*789Sahrens 921*789Sahrens if (zc->zc_filename[0] != '\0') { 922*789Sahrens /* 923*789Sahrens * We're creating a clone of an existing snapshot. 924*789Sahrens */ 925*789Sahrens zc->zc_filename[sizeof (zc->zc_filename) - 1] = '\0'; 926*789Sahrens if (dataset_namecheck(zc->zc_filename, NULL, NULL) != 0) 927*789Sahrens return (EINVAL); 928*789Sahrens 929*789Sahrens error = dmu_objset_open(zc->zc_filename, type, 930*789Sahrens DS_MODE_STANDARD | DS_MODE_READONLY, &clone); 931*789Sahrens if (error) 932*789Sahrens return (error); 933*789Sahrens error = dmu_objset_create(zc->zc_name, type, clone, NULL, NULL); 934*789Sahrens dmu_objset_close(clone); 935*789Sahrens } else if (strchr(zc->zc_name, '@') != 0) { 936*789Sahrens /* 937*789Sahrens * We're taking a snapshot of an existing dataset. 938*789Sahrens */ 939*789Sahrens error = dmu_objset_create(zc->zc_name, type, NULL, NULL, NULL); 940*789Sahrens } else { 941*789Sahrens /* 942*789Sahrens * We're creating a new dataset. 943*789Sahrens */ 944*789Sahrens if (type == DMU_OST_ZVOL) { 945*789Sahrens if ((error = zvol_check_volsize(zc)) != 0) 946*789Sahrens return (error); 947*789Sahrens if ((error = zvol_check_volblocksize(zc)) != 0) 948*789Sahrens return (error); 949*789Sahrens } 950*789Sahrens error = dmu_objset_create(zc->zc_name, type, NULL, cbfunc, zc); 951*789Sahrens } 952*789Sahrens return (error); 953*789Sahrens } 954*789Sahrens 955*789Sahrens static int 956*789Sahrens zfs_ioc_destroy(zfs_cmd_t *zc) 957*789Sahrens { 958*789Sahrens if (strchr(zc->zc_name, '@') != NULL && 959*789Sahrens zc->zc_objset_type == DMU_OST_ZFS) { 960*789Sahrens vfs_t *vfsp; 961*789Sahrens int err; 962*789Sahrens 963*789Sahrens /* 964*789Sahrens * Snapshots under .zfs control must be unmounted 965*789Sahrens * before they can be destroyed. 966*789Sahrens */ 967*789Sahrens if ((vfsp = zfs_get_vfs(zc->zc_name)) != NULL) { 968*789Sahrens /* 969*789Sahrens * Always force the unmount for snapshots. 970*789Sahrens */ 971*789Sahrens int flag = MS_FORCE; 972*789Sahrens 973*789Sahrens if ((err = vn_vfswlock(vfsp->vfs_vnodecovered)) != 0) { 974*789Sahrens VFS_RELE(vfsp); 975*789Sahrens return (err); 976*789Sahrens } 977*789Sahrens VFS_RELE(vfsp); 978*789Sahrens if ((err = dounmount(vfsp, flag, kcred)) != 0) 979*789Sahrens return (err); 980*789Sahrens } 981*789Sahrens } 982*789Sahrens 983*789Sahrens return (dmu_objset_destroy(zc->zc_name)); 984*789Sahrens } 985*789Sahrens 986*789Sahrens static int 987*789Sahrens zfs_ioc_rollback(zfs_cmd_t *zc) 988*789Sahrens { 989*789Sahrens return (dmu_objset_rollback(zc->zc_name)); 990*789Sahrens } 991*789Sahrens 992*789Sahrens static int 993*789Sahrens zfs_ioc_rename(zfs_cmd_t *zc) 994*789Sahrens { 995*789Sahrens zc->zc_prop_value[sizeof (zc->zc_prop_value) - 1] = '\0'; 996*789Sahrens if (dataset_namecheck(zc->zc_prop_value, NULL, NULL) != 0) 997*789Sahrens return (EINVAL); 998*789Sahrens 999*789Sahrens if (strchr(zc->zc_name, '@') != NULL && 1000*789Sahrens zc->zc_objset_type == DMU_OST_ZFS) { 1001*789Sahrens vfs_t *vfsp; 1002*789Sahrens int err; 1003*789Sahrens 1004*789Sahrens /* 1005*789Sahrens * Snapshots under .zfs control must be unmounted 1006*789Sahrens * before they can be renamed. 1007*789Sahrens */ 1008*789Sahrens if ((vfsp = zfs_get_vfs(zc->zc_name)) != NULL) { 1009*789Sahrens /* 1010*789Sahrens * Always force the unmount for snapshots. 1011*789Sahrens */ 1012*789Sahrens int flag = MS_FORCE; 1013*789Sahrens 1014*789Sahrens if ((err = vn_vfswlock(vfsp->vfs_vnodecovered)) != 0) { 1015*789Sahrens VFS_RELE(vfsp); 1016*789Sahrens return (err); 1017*789Sahrens } 1018*789Sahrens VFS_RELE(vfsp); 1019*789Sahrens if ((err = dounmount(vfsp, flag, kcred)) != 0) 1020*789Sahrens return (err); 1021*789Sahrens } 1022*789Sahrens } 1023*789Sahrens 1024*789Sahrens return (dmu_objset_rename(zc->zc_name, zc->zc_prop_value)); 1025*789Sahrens } 1026*789Sahrens 1027*789Sahrens static int 1028*789Sahrens zfs_ioc_recvbackup(zfs_cmd_t *zc) 1029*789Sahrens { 1030*789Sahrens file_t *fp; 1031*789Sahrens int error, fd; 1032*789Sahrens 1033*789Sahrens fd = zc->zc_cookie; 1034*789Sahrens fp = getf(fd); 1035*789Sahrens if (fp == NULL) 1036*789Sahrens return (EBADF); 1037*789Sahrens error = dmu_recvbackup(&zc->zc_begin_record, &zc->zc_cookie, 1038*789Sahrens fp->f_vnode, fp->f_offset); 1039*789Sahrens releasef(fd); 1040*789Sahrens return (error); 1041*789Sahrens } 1042*789Sahrens 1043*789Sahrens static int 1044*789Sahrens zfs_ioc_sendbackup(zfs_cmd_t *zc) 1045*789Sahrens { 1046*789Sahrens objset_t *fromsnap = NULL; 1047*789Sahrens objset_t *tosnap; 1048*789Sahrens file_t *fp; 1049*789Sahrens int error; 1050*789Sahrens 1051*789Sahrens error = dmu_objset_open(zc->zc_name, DMU_OST_ANY, 1052*789Sahrens DS_MODE_STANDARD | DS_MODE_READONLY, &tosnap); 1053*789Sahrens if (error) 1054*789Sahrens return (error); 1055*789Sahrens 1056*789Sahrens if (zc->zc_prop_value[0] != '\0') { 1057*789Sahrens error = dmu_objset_open(zc->zc_prop_value, DMU_OST_ANY, 1058*789Sahrens DS_MODE_STANDARD | DS_MODE_READONLY, &fromsnap); 1059*789Sahrens if (error) { 1060*789Sahrens dmu_objset_close(tosnap); 1061*789Sahrens return (error); 1062*789Sahrens } 1063*789Sahrens } 1064*789Sahrens 1065*789Sahrens fp = getf(zc->zc_cookie); 1066*789Sahrens if (fp == NULL) { 1067*789Sahrens dmu_objset_close(tosnap); 1068*789Sahrens if (fromsnap) 1069*789Sahrens dmu_objset_close(fromsnap); 1070*789Sahrens return (EBADF); 1071*789Sahrens } 1072*789Sahrens 1073*789Sahrens error = dmu_sendbackup(tosnap, fromsnap, fp->f_vnode); 1074*789Sahrens 1075*789Sahrens releasef(zc->zc_cookie); 1076*789Sahrens if (fromsnap) 1077*789Sahrens dmu_objset_close(fromsnap); 1078*789Sahrens dmu_objset_close(tosnap); 1079*789Sahrens return (error); 1080*789Sahrens } 1081*789Sahrens 1082*789Sahrens static zfs_ioc_vec_t zfs_ioc_vec[] = { 1083*789Sahrens { zfs_ioc_pool_create, zfs_secpolicy_config, pool_name }, 1084*789Sahrens { zfs_ioc_pool_destroy, zfs_secpolicy_config, pool_name }, 1085*789Sahrens { zfs_ioc_pool_import, zfs_secpolicy_config, pool_name }, 1086*789Sahrens { zfs_ioc_pool_export, zfs_secpolicy_config, pool_name }, 1087*789Sahrens { zfs_ioc_pool_configs, zfs_secpolicy_none, no_name }, 1088*789Sahrens { zfs_ioc_pool_guid, zfs_secpolicy_read, pool_name }, 1089*789Sahrens { zfs_ioc_pool_stats, zfs_secpolicy_read, pool_name }, 1090*789Sahrens { zfs_ioc_pool_tryimport, zfs_secpolicy_config, no_name }, 1091*789Sahrens { zfs_ioc_pool_scrub, zfs_secpolicy_config, pool_name }, 1092*789Sahrens { zfs_ioc_pool_freeze, zfs_secpolicy_config, no_name }, 1093*789Sahrens { zfs_ioc_vdev_add, zfs_secpolicy_config, pool_name }, 1094*789Sahrens { zfs_ioc_vdev_remove, zfs_secpolicy_config, pool_name }, 1095*789Sahrens { zfs_ioc_vdev_online, zfs_secpolicy_config, pool_name }, 1096*789Sahrens { zfs_ioc_vdev_offline, zfs_secpolicy_config, pool_name }, 1097*789Sahrens { zfs_ioc_vdev_attach, zfs_secpolicy_config, pool_name }, 1098*789Sahrens { zfs_ioc_vdev_detach, zfs_secpolicy_config, pool_name }, 1099*789Sahrens { zfs_ioc_objset_stats, zfs_secpolicy_read, dataset_name }, 1100*789Sahrens { zfs_ioc_dataset_list_next, zfs_secpolicy_read, dataset_name }, 1101*789Sahrens { zfs_ioc_snapshot_list_next, zfs_secpolicy_read, dataset_name }, 1102*789Sahrens { zfs_ioc_set_prop, zfs_secpolicy_setprop, dataset_name }, 1103*789Sahrens { zfs_ioc_set_quota, zfs_secpolicy_quota, dataset_name }, 1104*789Sahrens { zfs_ioc_set_reservation, zfs_secpolicy_write, dataset_name }, 1105*789Sahrens { zfs_ioc_set_volsize, zfs_secpolicy_config, dataset_name }, 1106*789Sahrens { zfs_ioc_set_volblocksize, zfs_secpolicy_config, dataset_name }, 1107*789Sahrens { zfs_ioc_create_minor, zfs_secpolicy_config, dataset_name }, 1108*789Sahrens { zfs_ioc_remove_minor, zfs_secpolicy_config, dataset_name }, 1109*789Sahrens { zfs_ioc_create, zfs_secpolicy_parent, dataset_name }, 1110*789Sahrens { zfs_ioc_destroy, zfs_secpolicy_parent, dataset_name }, 1111*789Sahrens { zfs_ioc_rollback, zfs_secpolicy_write, dataset_name }, 1112*789Sahrens { zfs_ioc_rename, zfs_secpolicy_write, dataset_name }, 1113*789Sahrens { zfs_ioc_recvbackup, zfs_secpolicy_write, dataset_name }, 1114*789Sahrens { zfs_ioc_sendbackup, zfs_secpolicy_write, dataset_name }, 1115*789Sahrens }; 1116*789Sahrens 1117*789Sahrens static int 1118*789Sahrens zfsdev_ioctl(dev_t dev, int cmd, intptr_t arg, int flag, cred_t *cr, int *rvalp) 1119*789Sahrens { 1120*789Sahrens zfs_cmd_t *zc; 1121*789Sahrens uint_t vec; 1122*789Sahrens int error; 1123*789Sahrens 1124*789Sahrens if (getminor(dev) != 0) 1125*789Sahrens return (zvol_ioctl(dev, cmd, arg, flag, cr, rvalp)); 1126*789Sahrens 1127*789Sahrens vec = cmd - ZFS_IOC; 1128*789Sahrens 1129*789Sahrens if (vec >= sizeof (zfs_ioc_vec) / sizeof (zfs_ioc_vec[0])) 1130*789Sahrens return (EINVAL); 1131*789Sahrens 1132*789Sahrens zc = kmem_zalloc(sizeof (zfs_cmd_t), KM_SLEEP); 1133*789Sahrens 1134*789Sahrens error = xcopyin((void *)arg, zc, sizeof (zfs_cmd_t)); 1135*789Sahrens 1136*789Sahrens if (error == 0) { 1137*789Sahrens zc->zc_cred = (uintptr_t)cr; 1138*789Sahrens zc->zc_dev = dev; 1139*789Sahrens error = zfs_ioc_vec[vec].zvec_secpolicy(zc->zc_name, 1140*789Sahrens zc->zc_prop_name, cr); 1141*789Sahrens } 1142*789Sahrens 1143*789Sahrens /* 1144*789Sahrens * Ensure that all pool/dataset names are valid before we pass down to 1145*789Sahrens * the lower layers. 1146*789Sahrens */ 1147*789Sahrens if (error == 0) { 1148*789Sahrens zc->zc_name[sizeof (zc->zc_name) - 1] = '\0'; 1149*789Sahrens switch (zfs_ioc_vec[vec].zvec_namecheck) { 1150*789Sahrens case pool_name: 1151*789Sahrens if (pool_namecheck(zc->zc_name, NULL, NULL) != 0) 1152*789Sahrens error = EINVAL; 1153*789Sahrens break; 1154*789Sahrens 1155*789Sahrens case dataset_name: 1156*789Sahrens if (dataset_namecheck(zc->zc_name, NULL, NULL) != 0) 1157*789Sahrens error = EINVAL; 1158*789Sahrens break; 1159*789Sahrens } 1160*789Sahrens } 1161*789Sahrens 1162*789Sahrens if (error == 0) 1163*789Sahrens error = zfs_ioc_vec[vec].zvec_func(zc); 1164*789Sahrens 1165*789Sahrens if (error == 0 || error == ENOMEM) { 1166*789Sahrens int rc = xcopyout(zc, (void *)arg, sizeof (zfs_cmd_t)); 1167*789Sahrens if (error == 0) 1168*789Sahrens error = rc; 1169*789Sahrens } 1170*789Sahrens 1171*789Sahrens kmem_free(zc, sizeof (zfs_cmd_t)); 1172*789Sahrens return (error); 1173*789Sahrens } 1174*789Sahrens 1175*789Sahrens static int 1176*789Sahrens zfs_attach(dev_info_t *dip, ddi_attach_cmd_t cmd) 1177*789Sahrens { 1178*789Sahrens if (cmd != DDI_ATTACH) 1179*789Sahrens return (DDI_FAILURE); 1180*789Sahrens 1181*789Sahrens if (ddi_create_minor_node(dip, "zfs", S_IFCHR, 0, 1182*789Sahrens DDI_PSEUDO, 0) == DDI_FAILURE) 1183*789Sahrens return (DDI_FAILURE); 1184*789Sahrens 1185*789Sahrens zfs_dip = dip; 1186*789Sahrens 1187*789Sahrens ddi_report_dev(dip); 1188*789Sahrens 1189*789Sahrens return (DDI_SUCCESS); 1190*789Sahrens } 1191*789Sahrens 1192*789Sahrens static int 1193*789Sahrens zfs_detach(dev_info_t *dip, ddi_detach_cmd_t cmd) 1194*789Sahrens { 1195*789Sahrens if (spa_busy() || zfs_busy() || zvol_busy()) 1196*789Sahrens return (DDI_FAILURE); 1197*789Sahrens 1198*789Sahrens if (cmd != DDI_DETACH) 1199*789Sahrens return (DDI_FAILURE); 1200*789Sahrens 1201*789Sahrens zfs_dip = NULL; 1202*789Sahrens 1203*789Sahrens ddi_prop_remove_all(dip); 1204*789Sahrens ddi_remove_minor_node(dip, NULL); 1205*789Sahrens 1206*789Sahrens return (DDI_SUCCESS); 1207*789Sahrens } 1208*789Sahrens 1209*789Sahrens /*ARGSUSED*/ 1210*789Sahrens static int 1211*789Sahrens zfs_info(dev_info_t *dip, ddi_info_cmd_t infocmd, void *arg, void **result) 1212*789Sahrens { 1213*789Sahrens switch (infocmd) { 1214*789Sahrens case DDI_INFO_DEVT2DEVINFO: 1215*789Sahrens *result = zfs_dip; 1216*789Sahrens return (DDI_SUCCESS); 1217*789Sahrens 1218*789Sahrens case DDI_INFO_DEVT2INSTANCE: 1219*789Sahrens *result = (void *)(uintptr_t)getminor((dev_t)arg); 1220*789Sahrens return (DDI_SUCCESS); 1221*789Sahrens } 1222*789Sahrens 1223*789Sahrens return (DDI_FAILURE); 1224*789Sahrens } 1225*789Sahrens 1226*789Sahrens /* 1227*789Sahrens * OK, so this is a little weird. 1228*789Sahrens * 1229*789Sahrens * /dev/zfs is the control node, i.e. minor 0. 1230*789Sahrens * /dev/zvol/[r]dsk/pool/dataset are the zvols, minor > 0. 1231*789Sahrens * 1232*789Sahrens * /dev/zfs has basically nothing to do except serve up ioctls, 1233*789Sahrens * so most of the standard driver entry points are in zvol.c. 1234*789Sahrens */ 1235*789Sahrens static struct cb_ops zfs_cb_ops = { 1236*789Sahrens zvol_open, /* open */ 1237*789Sahrens zvol_close, /* close */ 1238*789Sahrens zvol_strategy, /* strategy */ 1239*789Sahrens nodev, /* print */ 1240*789Sahrens nodev, /* dump */ 1241*789Sahrens zvol_read, /* read */ 1242*789Sahrens zvol_write, /* write */ 1243*789Sahrens zfsdev_ioctl, /* ioctl */ 1244*789Sahrens nodev, /* devmap */ 1245*789Sahrens nodev, /* mmap */ 1246*789Sahrens nodev, /* segmap */ 1247*789Sahrens nochpoll, /* poll */ 1248*789Sahrens ddi_prop_op, /* prop_op */ 1249*789Sahrens NULL, /* streamtab */ 1250*789Sahrens D_NEW | D_MP | D_64BIT, /* Driver compatibility flag */ 1251*789Sahrens CB_REV, /* version */ 1252*789Sahrens zvol_aread, /* async read */ 1253*789Sahrens zvol_awrite, /* async write */ 1254*789Sahrens }; 1255*789Sahrens 1256*789Sahrens static struct dev_ops zfs_dev_ops = { 1257*789Sahrens DEVO_REV, /* version */ 1258*789Sahrens 0, /* refcnt */ 1259*789Sahrens zfs_info, /* info */ 1260*789Sahrens nulldev, /* identify */ 1261*789Sahrens nulldev, /* probe */ 1262*789Sahrens zfs_attach, /* attach */ 1263*789Sahrens zfs_detach, /* detach */ 1264*789Sahrens nodev, /* reset */ 1265*789Sahrens &zfs_cb_ops, /* driver operations */ 1266*789Sahrens NULL /* no bus operations */ 1267*789Sahrens }; 1268*789Sahrens 1269*789Sahrens static struct modldrv zfs_modldrv = { 1270*789Sahrens &mod_driverops, "ZFS storage pool version 1", &zfs_dev_ops 1271*789Sahrens }; 1272*789Sahrens 1273*789Sahrens static struct modlinkage modlinkage = { 1274*789Sahrens MODREV_1, 1275*789Sahrens (void *)&zfs_modlfs, 1276*789Sahrens (void *)&zfs_modldrv, 1277*789Sahrens NULL 1278*789Sahrens }; 1279*789Sahrens 1280*789Sahrens int 1281*789Sahrens _init(void) 1282*789Sahrens { 1283*789Sahrens int error; 1284*789Sahrens 1285*789Sahrens if ((error = mod_install(&modlinkage)) != 0) 1286*789Sahrens return (error); 1287*789Sahrens 1288*789Sahrens error = ldi_ident_from_mod(&modlinkage, &zfs_li); 1289*789Sahrens ASSERT(error == 0); 1290*789Sahrens 1291*789Sahrens spa_init(FREAD | FWRITE); 1292*789Sahrens zfs_init(); 1293*789Sahrens zvol_init(); 1294*789Sahrens 1295*789Sahrens return (0); 1296*789Sahrens } 1297*789Sahrens 1298*789Sahrens int 1299*789Sahrens _fini(void) 1300*789Sahrens { 1301*789Sahrens int error; 1302*789Sahrens 1303*789Sahrens if (spa_busy() || zfs_busy() || zvol_busy()) 1304*789Sahrens return (EBUSY); 1305*789Sahrens 1306*789Sahrens if ((error = mod_remove(&modlinkage)) != 0) 1307*789Sahrens return (error); 1308*789Sahrens 1309*789Sahrens zvol_fini(); 1310*789Sahrens zfs_fini(); 1311*789Sahrens spa_fini(); 1312*789Sahrens 1313*789Sahrens ldi_ident_release(zfs_li); 1314*789Sahrens zfs_li = NULL; 1315*789Sahrens 1316*789Sahrens return (error); 1317*789Sahrens } 1318*789Sahrens 1319*789Sahrens int 1320*789Sahrens _info(struct modinfo *modinfop) 1321*789Sahrens { 1322*789Sahrens return (mod_info(&modlinkage, modinfop)); 1323*789Sahrens } 1324