/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2005 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" /* * ZFS volume emulation driver. * * Makes a DMU object look like a volume of arbitrary size, up to 2^64 bytes. * Volumes are accessed through the symbolic links named: * * /dev/zvol/dsk// * /dev/zvol/rdsk// * * These links are created by the ZFS-specific devfsadm link generator. * Volumes are persistent through reboot. No user command needs to be * run before opening and using a device. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "zfs_namecheck.h" #define ZVOL_OBJ 1ULL #define ZVOL_ZAP_OBJ 2ULL static void *zvol_state; /* * This lock protects the zvol_state structure from being modified * while it's being used, e.g. an open that comes in before a create * finishes. It also protects temporary opens of the dataset so that, * e.g., an open doesn't get a spurious EBUSY. */ static kmutex_t zvol_state_lock; static uint32_t zvol_minors; /* * The in-core state of each volume. */ typedef struct zvol_state { char zv_name[MAXPATHLEN]; /* pool/dd name */ uint64_t zv_volsize; /* amount of space we advertise */ minor_t zv_minor; /* minor number */ uint8_t zv_min_bs; /* minimum addressable block shift */ uint8_t zv_readonly; /* hard readonly; like write-protect */ objset_t *zv_objset; /* objset handle */ uint32_t zv_mode; /* DS_MODE_* flags at open time */ uint32_t zv_open_count[OTYPCNT]; /* open counts */ uint32_t zv_total_opens; /* total open count */ } zvol_state_t; static void zvol_size_changed(zvol_state_t *zv, dev_t dev) { dev = makedevice(getmajor(dev), zv->zv_minor); VERIFY(ddi_prop_update_int64(dev, zfs_dip, "Size", zv->zv_volsize) == DDI_SUCCESS); VERIFY(ddi_prop_update_int64(dev, zfs_dip, "Nblocks", lbtodb(zv->zv_volsize)) == DDI_SUCCESS); } int zvol_check_volsize(zfs_cmd_t *zc, uint64_t blocksize) { if (zc->zc_volsize == 0) return (EINVAL); if (zc->zc_volsize % blocksize != 0) return (EINVAL); #ifdef _ILP32 if (zc->zc_volsize - 1 > SPEC_MAXOFFSET_T) return (EOVERFLOW); #endif return (0); } int zvol_check_volblocksize(zfs_cmd_t *zc) { if (zc->zc_volblocksize < SPA_MINBLOCKSIZE || zc->zc_volblocksize > SPA_MAXBLOCKSIZE || !ISP2(zc->zc_volblocksize)) return (EDOM); return (0); } static void zvol_readonly_changed_cb(void *arg, uint64_t newval) { zvol_state_t *zv = arg; zv->zv_readonly = (uint8_t)newval; } int zvol_get_stats(zfs_cmd_t *zc, objset_t *os) { int error; dmu_object_info_t doi; error = zap_lookup(os, ZVOL_ZAP_OBJ, "size", 8, 1, &zc->zc_volsize); if (error) return (error); error = dmu_object_info(os, ZVOL_OBJ, &doi); if (error == 0) zc->zc_volblocksize = doi.doi_data_block_size; return (error); } /* * Find a free minor number. */ static minor_t zvol_minor_alloc(void) { minor_t minor; ASSERT(MUTEX_HELD(&zvol_state_lock)); for (minor = 1; minor <= ZVOL_MAX_MINOR; minor++) if (ddi_get_soft_state(zvol_state, minor) == NULL) return (minor); return (0); } static zvol_state_t * zvol_minor_lookup(char *name) { minor_t minor; zvol_state_t *zv; ASSERT(MUTEX_HELD(&zvol_state_lock)); for (minor = 1; minor <= ZVOL_MAX_MINOR; minor++) { zv = ddi_get_soft_state(zvol_state, minor); if (zv == NULL) continue; if (strcmp(zv->zv_name, name) == 0) break; } return (zv); } void zvol_create_cb(objset_t *os, void *arg, dmu_tx_t *tx) { zfs_cmd_t *zc = arg; int error; error = dmu_object_claim(os, ZVOL_OBJ, DMU_OT_ZVOL, zc->zc_volblocksize, DMU_OT_NONE, 0, tx); ASSERT(error == 0); error = zap_create_claim(os, ZVOL_ZAP_OBJ, DMU_OT_ZVOL_PROP, DMU_OT_NONE, 0, tx); ASSERT(error == 0); error = zap_update(os, ZVOL_ZAP_OBJ, "size", 8, 1, &zc->zc_volsize, tx); ASSERT(error == 0); } /* * Create a minor node for the specified volume. */ int zvol_create_minor(zfs_cmd_t *zc) { char *name = zc->zc_name; dev_t dev = zc->zc_dev; zvol_state_t *zv; objset_t *os; uint64_t volsize; minor_t minor = 0; struct pathname linkpath; int ds_mode = DS_MODE_PRIMARY; vnode_t *vp = NULL; char *devpath; size_t devpathlen = strlen(ZVOL_FULL_DEV_DIR) + 1 + strlen(name) + 1; char chrbuf[30], blkbuf[30]; int error; mutex_enter(&zvol_state_lock); if ((zv = zvol_minor_lookup(name)) != NULL) { mutex_exit(&zvol_state_lock); return (EEXIST); } if (strchr(name, '@') != 0) ds_mode |= DS_MODE_READONLY; error = dmu_objset_open(name, DMU_OST_ZVOL, ds_mode, &os); if (error) { mutex_exit(&zvol_state_lock); return (error); } error = zap_lookup(os, ZVOL_ZAP_OBJ, "size", 8, 1, &volsize); if (error) { dmu_objset_close(os); mutex_exit(&zvol_state_lock); return (error); } /* * If there's an existing /dev/zvol symlink, try to use the * same minor number we used last time. */ devpath = kmem_alloc(devpathlen, KM_SLEEP); (void) sprintf(devpath, "%s/%s", ZVOL_FULL_DEV_DIR, name); error = lookupname(devpath, UIO_SYSSPACE, NO_FOLLOW, NULL, &vp); kmem_free(devpath, devpathlen); if (error == 0 && vp->v_type != VLNK) error = EINVAL; if (error == 0) { pn_alloc(&linkpath); error = pn_getsymlink(vp, &linkpath, kcred); if (error == 0) { char *ms = strstr(linkpath.pn_path, ZVOL_PSEUDO_DEV); if (ms != NULL) { ms += strlen(ZVOL_PSEUDO_DEV); minor = stoi(&ms); } } pn_free(&linkpath); } if (vp != NULL) VN_RELE(vp); /* * If we found a minor but it's already in use, we must pick a new one. */ if (minor != 0 && ddi_get_soft_state(zvol_state, minor) != NULL) minor = 0; if (minor == 0) minor = zvol_minor_alloc(); if (minor == 0) { dmu_objset_close(os); mutex_exit(&zvol_state_lock); return (ENXIO); } if (ddi_soft_state_zalloc(zvol_state, minor) != DDI_SUCCESS) { dmu_objset_close(os); mutex_exit(&zvol_state_lock); return (EAGAIN); } (void) ddi_prop_update_string(minor, zfs_dip, ZVOL_PROP_NAME, name); (void) sprintf(chrbuf, "%uc,raw", minor); if (ddi_create_minor_node(zfs_dip, chrbuf, S_IFCHR, minor, DDI_PSEUDO, 0) == DDI_FAILURE) { ddi_soft_state_free(zvol_state, minor); dmu_objset_close(os); mutex_exit(&zvol_state_lock); return (EAGAIN); } (void) sprintf(blkbuf, "%uc", minor); if (ddi_create_minor_node(zfs_dip, blkbuf, S_IFBLK, minor, DDI_PSEUDO, 0) == DDI_FAILURE) { ddi_remove_minor_node(zfs_dip, chrbuf); ddi_soft_state_free(zvol_state, minor); dmu_objset_close(os); mutex_exit(&zvol_state_lock); return (EAGAIN); } zv = ddi_get_soft_state(zvol_state, minor); (void) strcpy(zv->zv_name, name); zv->zv_min_bs = DEV_BSHIFT; zv->zv_minor = minor; zv->zv_volsize = volsize; zv->zv_objset = os; zv->zv_mode = ds_mode; zvol_size_changed(zv, dev); VERIFY(dsl_prop_register(dmu_objset_ds(zv->zv_objset), "readonly", zvol_readonly_changed_cb, zv) == 0); zvol_minors++; mutex_exit(&zvol_state_lock); return (0); } /* * Remove minor node for the specified volume. */ int zvol_remove_minor(zfs_cmd_t *zc) { zvol_state_t *zv; char namebuf[30]; mutex_enter(&zvol_state_lock); if ((zv = zvol_minor_lookup(zc->zc_name)) == NULL) { mutex_exit(&zvol_state_lock); return (ENXIO); } if (zv->zv_total_opens != 0) { mutex_exit(&zvol_state_lock); return (EBUSY); } (void) sprintf(namebuf, "%uc,raw", zv->zv_minor); ddi_remove_minor_node(zfs_dip, namebuf); (void) sprintf(namebuf, "%uc", zv->zv_minor); ddi_remove_minor_node(zfs_dip, namebuf); VERIFY(dsl_prop_unregister(dmu_objset_ds(zv->zv_objset), "readonly", zvol_readonly_changed_cb, zv) == 0); dmu_objset_close(zv->zv_objset); zv->zv_objset = NULL; ddi_soft_state_free(zvol_state, zv->zv_minor); zvol_minors--; mutex_exit(&zvol_state_lock); return (0); } int zvol_set_volsize(zfs_cmd_t *zc) { zvol_state_t *zv; dev_t dev = zc->zc_dev; dmu_tx_t *tx; int error; dmu_object_info_t doi; mutex_enter(&zvol_state_lock); if ((zv = zvol_minor_lookup(zc->zc_name)) == NULL) { mutex_exit(&zvol_state_lock); return (ENXIO); } if ((error = dmu_object_info(zv->zv_objset, ZVOL_OBJ, &doi)) != 0 || (error = zvol_check_volsize(zc, doi.doi_data_block_size)) != 0) { mutex_exit(&zvol_state_lock); return (error); } if (zv->zv_readonly || (zv->zv_mode & DS_MODE_READONLY)) { mutex_exit(&zvol_state_lock); return (EROFS); } tx = dmu_tx_create(zv->zv_objset); dmu_tx_hold_zap(tx, ZVOL_ZAP_OBJ, 1); dmu_tx_hold_free(tx, ZVOL_OBJ, zc->zc_volsize, DMU_OBJECT_END); error = dmu_tx_assign(tx, TXG_WAIT); if (error) { dmu_tx_abort(tx); mutex_exit(&zvol_state_lock); return (error); } error = zap_update(zv->zv_objset, ZVOL_ZAP_OBJ, "size", 8, 1, &zc->zc_volsize, tx); if (error == 0) dmu_free_range(zv->zv_objset, ZVOL_OBJ, zc->zc_volsize, DMU_OBJECT_END, tx); dmu_tx_commit(tx); if (error == 0) { zv->zv_volsize = zc->zc_volsize; zvol_size_changed(zv, dev); } mutex_exit(&zvol_state_lock); return (error); } int zvol_set_volblocksize(zfs_cmd_t *zc) { zvol_state_t *zv; dmu_tx_t *tx; int error; mutex_enter(&zvol_state_lock); if ((zv = zvol_minor_lookup(zc->zc_name)) == NULL) { mutex_exit(&zvol_state_lock); return (ENXIO); } if (zv->zv_readonly || (zv->zv_mode & DS_MODE_READONLY)) { mutex_exit(&zvol_state_lock); return (EROFS); } tx = dmu_tx_create(zv->zv_objset); dmu_tx_hold_bonus(tx, ZVOL_OBJ); error = dmu_tx_assign(tx, TXG_WAIT); if (error) { dmu_tx_abort(tx); } else { error = dmu_object_set_blocksize(zv->zv_objset, ZVOL_OBJ, zc->zc_volblocksize, 0, tx); if (error == ENOTSUP) error = EBUSY; dmu_tx_commit(tx); } mutex_exit(&zvol_state_lock); return (error); } /*ARGSUSED*/ int zvol_open(dev_t *devp, int flag, int otyp, cred_t *cr) { minor_t minor = getminor(*devp); zvol_state_t *zv; if (minor == 0) /* This is the control device */ return (0); mutex_enter(&zvol_state_lock); zv = ddi_get_soft_state(zvol_state, minor); if (zv == NULL) { mutex_exit(&zvol_state_lock); return (ENXIO); } ASSERT(zv->zv_objset != NULL); if ((flag & FWRITE) && (zv->zv_readonly || (zv->zv_mode & DS_MODE_READONLY))) { mutex_exit(&zvol_state_lock); return (EROFS); } if (zv->zv_open_count[otyp] == 0 || otyp == OTYP_LYR) { zv->zv_open_count[otyp]++; zv->zv_total_opens++; } mutex_exit(&zvol_state_lock); return (0); } /*ARGSUSED*/ int zvol_close(dev_t dev, int flag, int otyp, cred_t *cr) { minor_t minor = getminor(dev); zvol_state_t *zv; if (minor == 0) /* This is the control device */ return (0); mutex_enter(&zvol_state_lock); zv = ddi_get_soft_state(zvol_state, minor); if (zv == NULL) { mutex_exit(&zvol_state_lock); return (ENXIO); } /* * The next statement is a workaround for the following DDI bug: * 6343604 specfs race: multiple "last-close" of the same device */ if (zv->zv_total_opens == 0) { mutex_exit(&zvol_state_lock); return (0); } /* * If the open count is zero, this is a spurious close. * That indicates a bug in the kernel / DDI framework. */ ASSERT(zv->zv_open_count[otyp] != 0); ASSERT(zv->zv_total_opens != 0); /* * You may get multiple opens, but only one close. */ zv->zv_open_count[otyp]--; zv->zv_total_opens--; mutex_exit(&zvol_state_lock); return (0); } int zvol_strategy(buf_t *bp) { zvol_state_t *zv = ddi_get_soft_state(zvol_state, getminor(bp->b_edev)); uint64_t off, volsize; size_t size, resid; char *addr; int error = 0; if (zv == NULL) { bioerror(bp, ENXIO); biodone(bp); return (0); } if (getminor(bp->b_edev) == 0) { bioerror(bp, EINVAL); biodone(bp); return (0); } if (zv->zv_readonly && !(bp->b_flags & B_READ)) { bioerror(bp, EROFS); biodone(bp); return (0); } off = ldbtob(bp->b_blkno); volsize = zv->zv_volsize; ASSERT(zv->zv_objset != NULL); bp_mapin(bp); addr = bp->b_un.b_addr; resid = bp->b_bcount; while (resid != 0 && off < volsize) { size = MIN(resid, 1UL << 20); /* cap at 1MB per tx */ if (size > volsize - off) /* don't write past the end */ size = volsize - off; if (bp->b_flags & B_READ) { error = dmu_read_canfail(zv->zv_objset, ZVOL_OBJ, off, size, addr); } else { dmu_tx_t *tx = dmu_tx_create(zv->zv_objset); dmu_tx_hold_write(tx, ZVOL_OBJ, off, size); error = dmu_tx_assign(tx, TXG_WAIT); if (error) { dmu_tx_abort(tx); } else { dmu_write(zv->zv_objset, ZVOL_OBJ, off, size, addr, tx); dmu_tx_commit(tx); } } if (error) break; off += size; addr += size; resid -= size; } if ((bp->b_resid = resid) == bp->b_bcount) bioerror(bp, off > volsize ? EINVAL : error); biodone(bp); return (0); } /*ARGSUSED*/ int zvol_read(dev_t dev, uio_t *uiop, cred_t *cr) { return (physio(zvol_strategy, NULL, dev, B_READ, minphys, uiop)); } /*ARGSUSED*/ int zvol_write(dev_t dev, uio_t *uiop, cred_t *cr) { return (physio(zvol_strategy, NULL, dev, B_WRITE, minphys, uiop)); } /*ARGSUSED*/ int zvol_aread(dev_t dev, struct aio_req *aio, cred_t *cr) { return (aphysio(zvol_strategy, anocancel, dev, B_READ, minphys, aio)); } /*ARGSUSED*/ int zvol_awrite(dev_t dev, struct aio_req *aio, cred_t *cr) { return (aphysio(zvol_strategy, anocancel, dev, B_WRITE, minphys, aio)); } /* * Dirtbag ioctls to support mkfs(1M) for UFS filesystems. See dkio(7I). */ /*ARGSUSED*/ int zvol_ioctl(dev_t dev, int cmd, intptr_t arg, int flag, cred_t *cr, int *rvalp) { zvol_state_t *zv; struct dk_cinfo dkc; struct dk_minfo dkm; dk_efi_t efi; efi_gpt_t gpt; efi_gpe_t gpe; struct uuid uuid = EFI_RESERVED; uint32_t crc; int error = 0; mutex_enter(&zvol_state_lock); zv = ddi_get_soft_state(zvol_state, getminor(dev)); if (zv == NULL) { mutex_exit(&zvol_state_lock); return (ENXIO); } switch (cmd) { case DKIOCINFO: bzero(&dkc, sizeof (dkc)); (void) strcpy(dkc.dki_cname, "zvol"); (void) strcpy(dkc.dki_dname, "zvol"); dkc.dki_ctype = DKC_UNKNOWN; dkc.dki_maxtransfer = 1 << 15; mutex_exit(&zvol_state_lock); if (ddi_copyout(&dkc, (void *)arg, sizeof (dkc), flag)) error = EFAULT; return (error); case DKIOCGMEDIAINFO: bzero(&dkm, sizeof (dkm)); dkm.dki_lbsize = 1U << zv->zv_min_bs; dkm.dki_capacity = zv->zv_volsize >> zv->zv_min_bs; dkm.dki_media_type = DK_UNKNOWN; mutex_exit(&zvol_state_lock); if (ddi_copyout(&dkm, (void *)arg, sizeof (dkm), flag)) error = EFAULT; return (error); case DKIOCGETEFI: if (ddi_copyin((void *)arg, &efi, sizeof (dk_efi_t), flag)) { mutex_exit(&zvol_state_lock); return (EFAULT); } bzero(&gpt, sizeof (gpt)); bzero(&gpe, sizeof (gpe)); efi.dki_data = (void *)(uintptr_t)efi.dki_data_64; if (efi.dki_length < sizeof (gpt) + sizeof (gpe)) { mutex_exit(&zvol_state_lock); return (EINVAL); } efi.dki_length = sizeof (gpt) + sizeof (gpe); gpt.efi_gpt_Signature = LE_64(EFI_SIGNATURE); gpt.efi_gpt_Revision = LE_32(EFI_VERSION_CURRENT); gpt.efi_gpt_HeaderSize = LE_32(sizeof (gpt)); gpt.efi_gpt_FirstUsableLBA = LE_64(0ULL); gpt.efi_gpt_LastUsableLBA = LE_64((zv->zv_volsize >> zv->zv_min_bs) - 1); gpt.efi_gpt_NumberOfPartitionEntries = LE_32(1); gpt.efi_gpt_SizeOfPartitionEntry = LE_32(sizeof (gpe)); UUID_LE_CONVERT(gpe.efi_gpe_PartitionTypeGUID, uuid); gpe.efi_gpe_StartingLBA = gpt.efi_gpt_FirstUsableLBA; gpe.efi_gpe_EndingLBA = gpt.efi_gpt_LastUsableLBA; CRC32(crc, &gpe, sizeof (gpe), -1U, crc32_table); gpt.efi_gpt_PartitionEntryArrayCRC32 = LE_32(~crc); CRC32(crc, &gpt, sizeof (gpt), -1U, crc32_table); gpt.efi_gpt_HeaderCRC32 = LE_32(~crc); mutex_exit(&zvol_state_lock); if (ddi_copyout(&gpt, efi.dki_data, sizeof (gpt), flag) || ddi_copyout(&gpe, efi.dki_data + 1, sizeof (gpe), flag)) error = EFAULT; return (error); default: error = ENOTSUP; break; } mutex_exit(&zvol_state_lock); return (error); } int zvol_busy(void) { return (zvol_minors != 0); } void zvol_init(void) { VERIFY(ddi_soft_state_init(&zvol_state, sizeof (zvol_state_t), 1) == 0); mutex_init(&zvol_state_lock, NULL, MUTEX_DEFAULT, NULL); } void zvol_fini(void) { mutex_destroy(&zvol_state_lock); ddi_soft_state_fini(&zvol_state); }