1*fa9e4066Sahrens /* 2*fa9e4066Sahrens * CDDL HEADER START 3*fa9e4066Sahrens * 4*fa9e4066Sahrens * The contents of this file are subject to the terms of the 5*fa9e4066Sahrens * Common Development and Distribution License, Version 1.0 only 6*fa9e4066Sahrens * (the "License"). You may not use this file except in compliance 7*fa9e4066Sahrens * with the License. 8*fa9e4066Sahrens * 9*fa9e4066Sahrens * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10*fa9e4066Sahrens * or http://www.opensolaris.org/os/licensing. 11*fa9e4066Sahrens * See the License for the specific language governing permissions 12*fa9e4066Sahrens * and limitations under the License. 13*fa9e4066Sahrens * 14*fa9e4066Sahrens * When distributing Covered Code, include this CDDL HEADER in each 15*fa9e4066Sahrens * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16*fa9e4066Sahrens * If applicable, add the following below this CDDL HEADER, with the 17*fa9e4066Sahrens * fields enclosed by brackets "[]" replaced with your own identifying 18*fa9e4066Sahrens * information: Portions Copyright [yyyy] [name of copyright owner] 19*fa9e4066Sahrens * 20*fa9e4066Sahrens * CDDL HEADER END 21*fa9e4066Sahrens */ 22*fa9e4066Sahrens /* 23*fa9e4066Sahrens * Copyright 2005 Sun Microsystems, Inc. All rights reserved. 24*fa9e4066Sahrens * Use is subject to license terms. 25*fa9e4066Sahrens */ 26*fa9e4066Sahrens 27*fa9e4066Sahrens #pragma ident "%Z%%M% %I% %E% SMI" 28*fa9e4066Sahrens 29*fa9e4066Sahrens #include <regex.h> 30*fa9e4066Sahrens #include <devfsadm.h> 31*fa9e4066Sahrens #include <stdio.h> 32*fa9e4066Sahrens #include <strings.h> 33*fa9e4066Sahrens #include <stdlib.h> 34*fa9e4066Sahrens #include <limits.h> 35*fa9e4066Sahrens #include <sys/mkdev.h> 36*fa9e4066Sahrens #include <sys/fs/zfs.h> 37*fa9e4066Sahrens 38*fa9e4066Sahrens /* zfs and zvol name info */ 39*fa9e4066Sahrens 40*fa9e4066Sahrens #define ZVOL_LINK_RE_DEVICES "zvol/r?dsk/.*/.*$" 41*fa9e4066Sahrens 42*fa9e4066Sahrens static int zfs(di_minor_t minor, di_node_t node); 43*fa9e4066Sahrens 44*fa9e4066Sahrens /* 45*fa9e4066Sahrens * devfs create callback register 46*fa9e4066Sahrens */ 47*fa9e4066Sahrens static devfsadm_create_t zfs_create_cbt[] = { 48*fa9e4066Sahrens { "pseudo", "ddi_pseudo", ZFS_DRIVER, 49*fa9e4066Sahrens TYPE_EXACT | DRV_EXACT, ILEVEL_0, zfs, 50*fa9e4066Sahrens }, 51*fa9e4066Sahrens }; 52*fa9e4066Sahrens DEVFSADM_CREATE_INIT_V0(zfs_create_cbt); 53*fa9e4066Sahrens 54*fa9e4066Sahrens /* 55*fa9e4066Sahrens * devfs cleanup register 56*fa9e4066Sahrens */ 57*fa9e4066Sahrens static devfsadm_remove_t zfs_remove_cbt[] = { 58*fa9e4066Sahrens { "pseudo", ZVOL_LINK_RE_DEVICES, RM_HOT | RM_POST, 59*fa9e4066Sahrens ILEVEL_0, devfsadm_rm_all }, 60*fa9e4066Sahrens }; 61*fa9e4066Sahrens DEVFSADM_REMOVE_INIT_V0(zfs_remove_cbt); 62*fa9e4066Sahrens 63*fa9e4066Sahrens /* 64*fa9e4066Sahrens * For the zfs control node: 65*fa9e4066Sahrens * /dev/zfs -> /devices/pseudo/zfs@0:zfs 66*fa9e4066Sahrens * For zvols: 67*fa9e4066Sahrens * /dev/zvol/dsk/<pool>/<dataset> -> /devices/pseudo/zfs@0:1 68*fa9e4066Sahrens * /dev/zvol/rdsk/<pool>/<dataset> -> /devices/pseudo/zfs@0:1,raw 69*fa9e4066Sahrens */ 70*fa9e4066Sahrens static int 71*fa9e4066Sahrens zfs(di_minor_t minor, di_node_t node) 72*fa9e4066Sahrens { 73*fa9e4066Sahrens dev_t dev; 74*fa9e4066Sahrens int err; 75*fa9e4066Sahrens char mn[MAXNAMELEN + 1]; 76*fa9e4066Sahrens char blkname[MAXNAMELEN + 1]; 77*fa9e4066Sahrens char rawname[MAXNAMELEN + 1]; 78*fa9e4066Sahrens char path[PATH_MAX + 1]; 79*fa9e4066Sahrens char *name; 80*fa9e4066Sahrens 81*fa9e4066Sahrens (void) strcpy(mn, di_minor_name(minor)); 82*fa9e4066Sahrens 83*fa9e4066Sahrens if (strcmp(mn, ZFS_DRIVER) == 0) { 84*fa9e4066Sahrens (void) devfsadm_mklink(ZFS_DRIVER, node, minor, 0); 85*fa9e4066Sahrens } else { 86*fa9e4066Sahrens dev = di_minor_devt(minor); 87*fa9e4066Sahrens err = di_prop_lookup_strings(dev, node, ZVOL_PROP_NAME, &name); 88*fa9e4066Sahrens if (err < 0) { 89*fa9e4066Sahrens /* property not defined so can't do anything */ 90*fa9e4066Sahrens return (DEVFSADM_CONTINUE); 91*fa9e4066Sahrens } 92*fa9e4066Sahrens (void) snprintf(blkname, sizeof (blkname), "%dc", 93*fa9e4066Sahrens (int)minor(dev)); 94*fa9e4066Sahrens (void) snprintf(rawname, sizeof (rawname), "%dc,raw", 95*fa9e4066Sahrens (int)minor(dev)); 96*fa9e4066Sahrens 97*fa9e4066Sahrens /* 98*fa9e4066Sahrens * This is where the actual public name gets constructed. 99*fa9e4066Sahrens * Change the snprintf format to change the public 100*fa9e4066Sahrens * path that gets constructed. 101*fa9e4066Sahrens */ 102*fa9e4066Sahrens if (strcmp(mn, blkname) == 0) { 103*fa9e4066Sahrens (void) snprintf(path, sizeof (path), "%s/%s", 104*fa9e4066Sahrens ZVOL_DEV_DIR, name); 105*fa9e4066Sahrens } else if (strcmp(mn, rawname) == 0) { 106*fa9e4066Sahrens (void) snprintf(path, sizeof (path), "%s/%s", 107*fa9e4066Sahrens ZVOL_RDEV_DIR, name); 108*fa9e4066Sahrens } else { 109*fa9e4066Sahrens return (DEVFSADM_CONTINUE); 110*fa9e4066Sahrens } 111*fa9e4066Sahrens 112*fa9e4066Sahrens (void) devfsadm_mklink(path, node, minor, 0); 113*fa9e4066Sahrens } 114*fa9e4066Sahrens return (DEVFSADM_CONTINUE); 115*fa9e4066Sahrens } 116