17c478bd9Sstevel@tonic-gate /* 27c478bd9Sstevel@tonic-gate * CDDL HEADER START 37c478bd9Sstevel@tonic-gate * 47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the 5facf4a8dSllai1 * Common Development and Distribution License (the "License"). 6facf4a8dSllai1 * You may not use this file except in compliance with the License. 77c478bd9Sstevel@tonic-gate * 87c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 97c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 107c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions 117c478bd9Sstevel@tonic-gate * and limitations under the License. 127c478bd9Sstevel@tonic-gate * 137c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 147c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 157c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 167c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 177c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 187c478bd9Sstevel@tonic-gate * 197c478bd9Sstevel@tonic-gate * CDDL HEADER END 207c478bd9Sstevel@tonic-gate */ 217aec1d6eScindi 227c478bd9Sstevel@tonic-gate /* 23*bf002425SStephen Hanson * Copyright (c) 1990, 2010, Oracle and/or its affiliates. All rights reserved. 247c478bd9Sstevel@tonic-gate */ 257c478bd9Sstevel@tonic-gate 267c478bd9Sstevel@tonic-gate /* 277c478bd9Sstevel@tonic-gate * modctl system call for loadable module support. 287c478bd9Sstevel@tonic-gate */ 297c478bd9Sstevel@tonic-gate 307c478bd9Sstevel@tonic-gate #include <sys/param.h> 317c478bd9Sstevel@tonic-gate #include <sys/user.h> 327c478bd9Sstevel@tonic-gate #include <sys/systm.h> 337c478bd9Sstevel@tonic-gate #include <sys/exec.h> 347c478bd9Sstevel@tonic-gate #include <sys/file.h> 357c478bd9Sstevel@tonic-gate #include <sys/stat.h> 367c478bd9Sstevel@tonic-gate #include <sys/conf.h> 377c478bd9Sstevel@tonic-gate #include <sys/time.h> 387c478bd9Sstevel@tonic-gate #include <sys/reboot.h> 397c478bd9Sstevel@tonic-gate #include <sys/fs/ufs_fsdir.h> 407c478bd9Sstevel@tonic-gate #include <sys/kmem.h> 417c478bd9Sstevel@tonic-gate #include <sys/sysconf.h> 427c478bd9Sstevel@tonic-gate #include <sys/cmn_err.h> 437c478bd9Sstevel@tonic-gate #include <sys/ddi.h> 447c478bd9Sstevel@tonic-gate #include <sys/sunddi.h> 457c478bd9Sstevel@tonic-gate #include <sys/sunndi.h> 467c478bd9Sstevel@tonic-gate #include <sys/ndi_impldefs.h> 477c478bd9Sstevel@tonic-gate #include <sys/ddi_impldefs.h> 487c478bd9Sstevel@tonic-gate #include <sys/ddi_implfuncs.h> 497c478bd9Sstevel@tonic-gate #include <sys/bootconf.h> 507c478bd9Sstevel@tonic-gate #include <sys/dc_ki.h> 517c478bd9Sstevel@tonic-gate #include <sys/cladm.h> 527c478bd9Sstevel@tonic-gate #include <sys/dtrace.h> 537c478bd9Sstevel@tonic-gate #include <sys/kdi.h> 547c478bd9Sstevel@tonic-gate 557c478bd9Sstevel@tonic-gate #include <sys/devpolicy.h> 567c478bd9Sstevel@tonic-gate #include <sys/modctl.h> 577c478bd9Sstevel@tonic-gate #include <sys/kobj.h> 587c478bd9Sstevel@tonic-gate #include <sys/devops.h> 597c478bd9Sstevel@tonic-gate #include <sys/autoconf.h> 607c478bd9Sstevel@tonic-gate #include <sys/hwconf.h> 617c478bd9Sstevel@tonic-gate #include <sys/callb.h> 627c478bd9Sstevel@tonic-gate #include <sys/debug.h> 637c478bd9Sstevel@tonic-gate #include <sys/cpuvar.h> 647c478bd9Sstevel@tonic-gate #include <sys/sysmacros.h> 657c478bd9Sstevel@tonic-gate #include <sys/sysevent.h> 667c478bd9Sstevel@tonic-gate #include <sys/sysevent_impl.h> 677c478bd9Sstevel@tonic-gate #include <sys/instance.h> 687c478bd9Sstevel@tonic-gate #include <sys/modhash.h> 697c478bd9Sstevel@tonic-gate #include <sys/modhash_impl.h> 707c478bd9Sstevel@tonic-gate #include <sys/dacf_impl.h> 717c478bd9Sstevel@tonic-gate #include <sys/vfs.h> 727c478bd9Sstevel@tonic-gate #include <sys/pathname.h> 737c478bd9Sstevel@tonic-gate #include <sys/console.h> 747c478bd9Sstevel@tonic-gate #include <sys/policy.h> 757c478bd9Sstevel@tonic-gate #include <ipp/ipp_impl.h> 767c478bd9Sstevel@tonic-gate #include <sys/fs/dv_node.h> 777c478bd9Sstevel@tonic-gate #include <sys/strsubr.h> 783c5e027bSEric Taylor #include <sys/fs/sdev_impl.h> 797c478bd9Sstevel@tonic-gate 807c478bd9Sstevel@tonic-gate static int mod_circdep(struct modctl *); 817c478bd9Sstevel@tonic-gate static int modinfo(modid_t, struct modinfo *); 827c478bd9Sstevel@tonic-gate 837c478bd9Sstevel@tonic-gate static void mod_uninstall_all(void); 847c478bd9Sstevel@tonic-gate static int mod_getinfo(struct modctl *, struct modinfo *); 857aec1d6eScindi static struct modctl *allocate_modp(const char *, const char *); 867c478bd9Sstevel@tonic-gate 877c478bd9Sstevel@tonic-gate static int mod_load(struct modctl *, int); 887c478bd9Sstevel@tonic-gate static void mod_unload(struct modctl *); 897c478bd9Sstevel@tonic-gate static int modinstall(struct modctl *); 907c478bd9Sstevel@tonic-gate static int moduninstall(struct modctl *); 917c478bd9Sstevel@tonic-gate 927aec1d6eScindi static struct modctl *mod_hold_by_name_common(struct modctl *, const char *); 937c478bd9Sstevel@tonic-gate static struct modctl *mod_hold_next_by_id(modid_t); 947c478bd9Sstevel@tonic-gate static struct modctl *mod_hold_loaded_mod(struct modctl *, char *, int *); 9520c794b3Sgavinm static struct modctl *mod_hold_installed_mod(char *, int, int, int *); 967c478bd9Sstevel@tonic-gate 977c478bd9Sstevel@tonic-gate static void mod_release(struct modctl *); 987c478bd9Sstevel@tonic-gate static void mod_make_requisite(struct modctl *, struct modctl *); 997c478bd9Sstevel@tonic-gate static int mod_install_requisites(struct modctl *); 1007c478bd9Sstevel@tonic-gate static void check_esc_sequences(char *, char *); 1017c478bd9Sstevel@tonic-gate static struct modctl *mod_hold_by_name_requisite(struct modctl *, char *); 1027c478bd9Sstevel@tonic-gate 1037c478bd9Sstevel@tonic-gate /* 1047c478bd9Sstevel@tonic-gate * module loading thread control structure. Calls to kobj_load_module()() are 1057c478bd9Sstevel@tonic-gate * handled off to a separate thead using this structure. 1067c478bd9Sstevel@tonic-gate */ 1077c478bd9Sstevel@tonic-gate struct loadmt { 1087c478bd9Sstevel@tonic-gate ksema_t sema; 1097c478bd9Sstevel@tonic-gate struct modctl *mp; 1107c478bd9Sstevel@tonic-gate int usepath; 1117c478bd9Sstevel@tonic-gate kthread_t *owner; 1127c478bd9Sstevel@tonic-gate int retval; 1137c478bd9Sstevel@tonic-gate }; 1147c478bd9Sstevel@tonic-gate 1157c478bd9Sstevel@tonic-gate static void modload_thread(struct loadmt *); 1167c478bd9Sstevel@tonic-gate 1177c478bd9Sstevel@tonic-gate kcondvar_t mod_cv; 1187c478bd9Sstevel@tonic-gate kcondvar_t mod_uninstall_cv; /* Communication between swapper */ 1197c478bd9Sstevel@tonic-gate /* and the uninstall daemon. */ 1207c478bd9Sstevel@tonic-gate kmutex_t mod_lock; /* protects &modules insert linkage, */ 1217c478bd9Sstevel@tonic-gate /* mod_busy, mod_want, and mod_ref. */ 1227c478bd9Sstevel@tonic-gate /* blocking operations while holding */ 1237c478bd9Sstevel@tonic-gate /* mod_lock should be avoided */ 1247c478bd9Sstevel@tonic-gate kmutex_t mod_uninstall_lock; /* protects mod_uninstall_cv */ 1257c478bd9Sstevel@tonic-gate kthread_id_t mod_aul_thread; 1267c478bd9Sstevel@tonic-gate 127a7aa4df7Scth int modunload_wait; 128a7aa4df7Scth kmutex_t modunload_wait_mutex; 129a7aa4df7Scth kcondvar_t modunload_wait_cv; 130a7aa4df7Scth int modunload_active_count; 131a7aa4df7Scth int modunload_disable_count; 132a7aa4df7Scth 1337c478bd9Sstevel@tonic-gate int isminiroot; /* set if running as miniroot */ 1347c478bd9Sstevel@tonic-gate int modrootloaded; /* set after root driver and fs are loaded */ 1357c478bd9Sstevel@tonic-gate int moddebug = 0x0; /* debug flags for module writers */ 1367c478bd9Sstevel@tonic-gate int swaploaded; /* set after swap driver and fs are loaded */ 1377c478bd9Sstevel@tonic-gate int bop_io_quiesced = 0; /* set when BOP I/O can no longer be used */ 1387c478bd9Sstevel@tonic-gate int last_module_id; 1397c478bd9Sstevel@tonic-gate clock_t mod_uninstall_interval = 0; 1407c478bd9Sstevel@tonic-gate int ddi_modclose_unload = 1; /* 0 -> just decrement reference */ 1417c478bd9Sstevel@tonic-gate 14251215d02SJerry Gilliam int devcnt_incr = 256; /* allow for additional drivers */ 14351215d02SJerry Gilliam int devcnt_min = 512; /* and always at least this number */ 14451215d02SJerry Gilliam 1457c478bd9Sstevel@tonic-gate struct devnames *devnamesp; 1467c478bd9Sstevel@tonic-gate struct devnames orphanlist; 1477c478bd9Sstevel@tonic-gate 1487c478bd9Sstevel@tonic-gate krwlock_t devinfo_tree_lock; /* obsolete, to be removed */ 1497c478bd9Sstevel@tonic-gate 1507c478bd9Sstevel@tonic-gate #define MAJBINDFILE "/etc/name_to_major" 1517c478bd9Sstevel@tonic-gate #define SYSBINDFILE "/etc/name_to_sysnum" 1527c478bd9Sstevel@tonic-gate 1537c478bd9Sstevel@tonic-gate static char majbind[] = MAJBINDFILE; 1547c478bd9Sstevel@tonic-gate static char sysbind[] = SYSBINDFILE; 1557c478bd9Sstevel@tonic-gate static uint_t mod_autounload_key; /* for module autounload detection */ 1567c478bd9Sstevel@tonic-gate 1577c478bd9Sstevel@tonic-gate extern int obpdebug; 1587c478bd9Sstevel@tonic-gate 1597c478bd9Sstevel@tonic-gate #define DEBUGGER_PRESENT ((boothowto & RB_DEBUG) || (obpdebug != 0)) 1607c478bd9Sstevel@tonic-gate 1617c478bd9Sstevel@tonic-gate static int minorperm_loaded = 0; 1627c478bd9Sstevel@tonic-gate 1637c478bd9Sstevel@tonic-gate void 1647c478bd9Sstevel@tonic-gate mod_setup(void) 1657c478bd9Sstevel@tonic-gate { 1667c478bd9Sstevel@tonic-gate struct sysent *callp; 1677c478bd9Sstevel@tonic-gate int callnum, exectype; 1687c478bd9Sstevel@tonic-gate int num_devs; 1697c478bd9Sstevel@tonic-gate int i; 1707c478bd9Sstevel@tonic-gate 1717c478bd9Sstevel@tonic-gate /* 1727c478bd9Sstevel@tonic-gate * Initialize the list of loaded driver dev_ops. 1737c478bd9Sstevel@tonic-gate * XXX - This must be done before reading the system file so that 1747c478bd9Sstevel@tonic-gate * forceloads of drivers will work. 1757c478bd9Sstevel@tonic-gate */ 1767c478bd9Sstevel@tonic-gate num_devs = read_binding_file(majbind, mb_hashtab, make_mbind); 1777c478bd9Sstevel@tonic-gate /* 1787c478bd9Sstevel@tonic-gate * Since read_binding_file is common code, it doesn't enforce that all 1797c478bd9Sstevel@tonic-gate * of the binding file entries have major numbers <= MAXMAJ32. Thus, 1807c478bd9Sstevel@tonic-gate * ensure that we don't allocate some massive amount of space due to a 1817c478bd9Sstevel@tonic-gate * bad entry. We can't have major numbers bigger than MAXMAJ32 1827c478bd9Sstevel@tonic-gate * until file system support for larger major numbers exists. 1837c478bd9Sstevel@tonic-gate */ 1847c478bd9Sstevel@tonic-gate 1857c478bd9Sstevel@tonic-gate /* 1867c478bd9Sstevel@tonic-gate * Leave space for expansion, but not more than L_MAXMAJ32 1877c478bd9Sstevel@tonic-gate */ 18851215d02SJerry Gilliam devcnt = MIN(num_devs + devcnt_incr, L_MAXMAJ32); 18951215d02SJerry Gilliam devcnt = MAX(devcnt, devcnt_min); 1907c478bd9Sstevel@tonic-gate devopsp = kmem_alloc(devcnt * sizeof (struct dev_ops *), KM_SLEEP); 1917c478bd9Sstevel@tonic-gate for (i = 0; i < devcnt; i++) 1927c478bd9Sstevel@tonic-gate devopsp[i] = &mod_nodev_ops; 1937c478bd9Sstevel@tonic-gate 1947c478bd9Sstevel@tonic-gate init_devnamesp(devcnt); 1957c478bd9Sstevel@tonic-gate 1967c478bd9Sstevel@tonic-gate /* 1977c478bd9Sstevel@tonic-gate * Sync up with the work that the stand-alone linker has already done. 1987c478bd9Sstevel@tonic-gate */ 1997c478bd9Sstevel@tonic-gate (void) kobj_sync(); 2007c478bd9Sstevel@tonic-gate 2017c478bd9Sstevel@tonic-gate if (boothowto & RB_DEBUG) 2027c478bd9Sstevel@tonic-gate kdi_dvec_modavail(); 2037c478bd9Sstevel@tonic-gate 2047c478bd9Sstevel@tonic-gate make_aliases(mb_hashtab); 2057c478bd9Sstevel@tonic-gate 2067c478bd9Sstevel@tonic-gate /* 2077c478bd9Sstevel@tonic-gate * Initialize streams device implementation structures. 2087c478bd9Sstevel@tonic-gate */ 2097c478bd9Sstevel@tonic-gate devimpl = kmem_zalloc(devcnt * sizeof (cdevsw_impl_t), KM_SLEEP); 2107c478bd9Sstevel@tonic-gate 2117c478bd9Sstevel@tonic-gate /* 2127c478bd9Sstevel@tonic-gate * If the cl_bootstrap module is present, 2137c478bd9Sstevel@tonic-gate * we should be configured as a cluster. Loading this module 2147c478bd9Sstevel@tonic-gate * will set "cluster_bootflags" to non-zero. 2157c478bd9Sstevel@tonic-gate */ 2167c478bd9Sstevel@tonic-gate (void) modload("misc", "cl_bootstrap"); 2177c478bd9Sstevel@tonic-gate 2187c478bd9Sstevel@tonic-gate (void) read_binding_file(sysbind, sb_hashtab, make_mbind); 2197c478bd9Sstevel@tonic-gate init_syscallnames(NSYSCALL); 2207c478bd9Sstevel@tonic-gate 2217c478bd9Sstevel@tonic-gate /* 2227c478bd9Sstevel@tonic-gate * Start up dynamic autoconfiguration framework (dacf). 2237c478bd9Sstevel@tonic-gate */ 2247c478bd9Sstevel@tonic-gate mod_hash_init(); 2257c478bd9Sstevel@tonic-gate dacf_init(); 2267c478bd9Sstevel@tonic-gate 2277c478bd9Sstevel@tonic-gate /* 2287c478bd9Sstevel@tonic-gate * Start up IP policy framework (ipp). 2297c478bd9Sstevel@tonic-gate */ 2307c478bd9Sstevel@tonic-gate ipp_init(); 2317c478bd9Sstevel@tonic-gate 2327c478bd9Sstevel@tonic-gate /* 2337c478bd9Sstevel@tonic-gate * Allocate loadable native system call locks. 2347c478bd9Sstevel@tonic-gate */ 2357c478bd9Sstevel@tonic-gate for (callnum = 0, callp = sysent; callnum < NSYSCALL; 2367c478bd9Sstevel@tonic-gate callnum++, callp++) { 2377c478bd9Sstevel@tonic-gate if (LOADABLE_SYSCALL(callp)) { 2387c478bd9Sstevel@tonic-gate if (mod_getsysname(callnum) != NULL) { 2397c478bd9Sstevel@tonic-gate callp->sy_lock = 2407c478bd9Sstevel@tonic-gate kobj_zalloc(sizeof (krwlock_t), KM_SLEEP); 2417c478bd9Sstevel@tonic-gate rw_init(callp->sy_lock, NULL, RW_DEFAULT, NULL); 2427c478bd9Sstevel@tonic-gate } else { 2437c478bd9Sstevel@tonic-gate callp->sy_flags &= ~SE_LOADABLE; 2447c478bd9Sstevel@tonic-gate callp->sy_callc = nosys; 2457c478bd9Sstevel@tonic-gate } 2467c478bd9Sstevel@tonic-gate #ifdef DEBUG 2477c478bd9Sstevel@tonic-gate } else { 2487c478bd9Sstevel@tonic-gate /* 2497c478bd9Sstevel@tonic-gate * Do some sanity checks on the sysent table 2507c478bd9Sstevel@tonic-gate */ 2517c478bd9Sstevel@tonic-gate switch (callp->sy_flags & SE_RVAL_MASK) { 2527c478bd9Sstevel@tonic-gate case SE_32RVAL1: 2537c478bd9Sstevel@tonic-gate /* only r_val1 returned */ 2547c478bd9Sstevel@tonic-gate case SE_32RVAL1 | SE_32RVAL2: 2557c478bd9Sstevel@tonic-gate /* r_val1 and r_val2 returned */ 2567c478bd9Sstevel@tonic-gate case SE_64RVAL: 2577c478bd9Sstevel@tonic-gate /* 64-bit rval returned */ 2587c478bd9Sstevel@tonic-gate break; 2597c478bd9Sstevel@tonic-gate default: 2607c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "sysent[%d]: bad flags %x", 2617c478bd9Sstevel@tonic-gate callnum, callp->sy_flags); 2627c478bd9Sstevel@tonic-gate } 2637c478bd9Sstevel@tonic-gate #endif 2647c478bd9Sstevel@tonic-gate } 2657c478bd9Sstevel@tonic-gate } 2667c478bd9Sstevel@tonic-gate 2677c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 2687c478bd9Sstevel@tonic-gate /* 2697c478bd9Sstevel@tonic-gate * Allocate loadable system call locks for 32-bit compat syscalls 2707c478bd9Sstevel@tonic-gate */ 2717c478bd9Sstevel@tonic-gate for (callnum = 0, callp = sysent32; callnum < NSYSCALL; 2727c478bd9Sstevel@tonic-gate callnum++, callp++) { 2737c478bd9Sstevel@tonic-gate if (LOADABLE_SYSCALL(callp)) { 2747c478bd9Sstevel@tonic-gate if (mod_getsysname(callnum) != NULL) { 2757c478bd9Sstevel@tonic-gate callp->sy_lock = 2767c478bd9Sstevel@tonic-gate kobj_zalloc(sizeof (krwlock_t), KM_SLEEP); 2777c478bd9Sstevel@tonic-gate rw_init(callp->sy_lock, NULL, RW_DEFAULT, NULL); 2787c478bd9Sstevel@tonic-gate } else { 2797c478bd9Sstevel@tonic-gate callp->sy_flags &= ~SE_LOADABLE; 2807c478bd9Sstevel@tonic-gate callp->sy_callc = nosys; 2817c478bd9Sstevel@tonic-gate } 2827c478bd9Sstevel@tonic-gate #ifdef DEBUG 2837c478bd9Sstevel@tonic-gate } else { 2847c478bd9Sstevel@tonic-gate /* 2857c478bd9Sstevel@tonic-gate * Do some sanity checks on the sysent table 2867c478bd9Sstevel@tonic-gate */ 2877c478bd9Sstevel@tonic-gate switch (callp->sy_flags & SE_RVAL_MASK) { 2887c478bd9Sstevel@tonic-gate case SE_32RVAL1: 2897c478bd9Sstevel@tonic-gate /* only r_val1 returned */ 2907c478bd9Sstevel@tonic-gate case SE_32RVAL1 | SE_32RVAL2: 2917c478bd9Sstevel@tonic-gate /* r_val1 and r_val2 returned */ 2927c478bd9Sstevel@tonic-gate case SE_64RVAL: 2937c478bd9Sstevel@tonic-gate /* 64-bit rval returned */ 2947c478bd9Sstevel@tonic-gate break; 2957c478bd9Sstevel@tonic-gate default: 2967c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "sysent32[%d]: bad flags %x", 2977c478bd9Sstevel@tonic-gate callnum, callp->sy_flags); 2987c478bd9Sstevel@tonic-gate goto skip; 2997c478bd9Sstevel@tonic-gate } 3007c478bd9Sstevel@tonic-gate 3017c478bd9Sstevel@tonic-gate /* 3027c478bd9Sstevel@tonic-gate * Cross-check the native and compatibility tables. 3037c478bd9Sstevel@tonic-gate */ 3047c478bd9Sstevel@tonic-gate if (callp->sy_callc == nosys || 3057c478bd9Sstevel@tonic-gate sysent[callnum].sy_callc == nosys) 3067c478bd9Sstevel@tonic-gate continue; 3077c478bd9Sstevel@tonic-gate /* 3087c478bd9Sstevel@tonic-gate * If only one or the other slot is loadable, then 3097c478bd9Sstevel@tonic-gate * there's an error -- they should match! 3107c478bd9Sstevel@tonic-gate */ 3117c478bd9Sstevel@tonic-gate if ((callp->sy_callc == loadable_syscall) ^ 3127c478bd9Sstevel@tonic-gate (sysent[callnum].sy_callc == loadable_syscall)) { 3137c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "sysent[%d] loadable?", 3147c478bd9Sstevel@tonic-gate callnum); 3157c478bd9Sstevel@tonic-gate } 3167c478bd9Sstevel@tonic-gate /* 3177c478bd9Sstevel@tonic-gate * This is more of a heuristic test -- if the 3187c478bd9Sstevel@tonic-gate * system call returns two values in the 32-bit 3197c478bd9Sstevel@tonic-gate * world, it should probably return two 32-bit 3207c478bd9Sstevel@tonic-gate * values in the 64-bit world too. 3217c478bd9Sstevel@tonic-gate */ 3227c478bd9Sstevel@tonic-gate if (((callp->sy_flags & SE_32RVAL2) == 0) ^ 3237c478bd9Sstevel@tonic-gate ((sysent[callnum].sy_flags & SE_32RVAL2) == 0)) { 3247c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "sysent[%d] rval2 mismatch!", 3257c478bd9Sstevel@tonic-gate callnum); 3267c478bd9Sstevel@tonic-gate } 3277c478bd9Sstevel@tonic-gate skip:; 3287c478bd9Sstevel@tonic-gate #endif /* DEBUG */ 3297c478bd9Sstevel@tonic-gate } 3307c478bd9Sstevel@tonic-gate } 3317c478bd9Sstevel@tonic-gate #endif /* _SYSCALL32_IMPL */ 3327c478bd9Sstevel@tonic-gate 3337c478bd9Sstevel@tonic-gate /* 3347c478bd9Sstevel@tonic-gate * Allocate loadable exec locks. (Assumes all execs are loadable) 3357c478bd9Sstevel@tonic-gate */ 3367c478bd9Sstevel@tonic-gate for (exectype = 0; exectype < nexectype; exectype++) { 3377c478bd9Sstevel@tonic-gate execsw[exectype].exec_lock = 3387c478bd9Sstevel@tonic-gate kobj_zalloc(sizeof (krwlock_t), KM_SLEEP); 3397c478bd9Sstevel@tonic-gate rw_init(execsw[exectype].exec_lock, NULL, RW_DEFAULT, NULL); 3407c478bd9Sstevel@tonic-gate } 3417c478bd9Sstevel@tonic-gate 3427c478bd9Sstevel@tonic-gate read_class_file(); 3437c478bd9Sstevel@tonic-gate 3447c478bd9Sstevel@tonic-gate /* init thread specific structure for mod_uninstall_all */ 3457c478bd9Sstevel@tonic-gate tsd_create(&mod_autounload_key, NULL); 3467c478bd9Sstevel@tonic-gate } 3477c478bd9Sstevel@tonic-gate 3487c478bd9Sstevel@tonic-gate static int 3497c478bd9Sstevel@tonic-gate modctl_modload(int use_path, char *filename, int *rvp) 3507c478bd9Sstevel@tonic-gate { 3517c478bd9Sstevel@tonic-gate struct modctl *modp; 3527c478bd9Sstevel@tonic-gate int retval = 0; 3537c478bd9Sstevel@tonic-gate char *filenamep; 3547c478bd9Sstevel@tonic-gate int modid; 3557c478bd9Sstevel@tonic-gate 3567c478bd9Sstevel@tonic-gate filenamep = kmem_zalloc(MOD_MAXPATH, KM_SLEEP); 3577c478bd9Sstevel@tonic-gate 3587c478bd9Sstevel@tonic-gate if (copyinstr(filename, filenamep, MOD_MAXPATH, 0)) { 3597c478bd9Sstevel@tonic-gate retval = EFAULT; 3607c478bd9Sstevel@tonic-gate goto out; 3617c478bd9Sstevel@tonic-gate } 3627c478bd9Sstevel@tonic-gate 3637c478bd9Sstevel@tonic-gate filenamep[MOD_MAXPATH - 1] = 0; 36420c794b3Sgavinm modp = mod_hold_installed_mod(filenamep, use_path, 0, &retval); 3657c478bd9Sstevel@tonic-gate 3667c478bd9Sstevel@tonic-gate if (modp == NULL) 3677c478bd9Sstevel@tonic-gate goto out; 3687c478bd9Sstevel@tonic-gate 3697c478bd9Sstevel@tonic-gate modp->mod_loadflags |= MOD_NOAUTOUNLOAD; 3707c478bd9Sstevel@tonic-gate modid = modp->mod_id; 3717c478bd9Sstevel@tonic-gate mod_release_mod(modp); 3720b38a8bdSahl CPU_STATS_ADDQ(CPU, sys, modload, 1); 3737c478bd9Sstevel@tonic-gate if (rvp != NULL && copyout(&modid, rvp, sizeof (modid)) != 0) 3747c478bd9Sstevel@tonic-gate retval = EFAULT; 3757c478bd9Sstevel@tonic-gate out: 3767c478bd9Sstevel@tonic-gate kmem_free(filenamep, MOD_MAXPATH); 3777c478bd9Sstevel@tonic-gate 3787c478bd9Sstevel@tonic-gate return (retval); 3797c478bd9Sstevel@tonic-gate } 3807c478bd9Sstevel@tonic-gate 3817c478bd9Sstevel@tonic-gate static int 3827c478bd9Sstevel@tonic-gate modctl_modunload(modid_t id) 3837c478bd9Sstevel@tonic-gate { 3847c478bd9Sstevel@tonic-gate int rval = 0; 3857c478bd9Sstevel@tonic-gate 3867c478bd9Sstevel@tonic-gate if (id == 0) { 3877c478bd9Sstevel@tonic-gate #ifdef DEBUG 3887c478bd9Sstevel@tonic-gate /* 3897c478bd9Sstevel@tonic-gate * Turn on mod_uninstall_daemon 3907c478bd9Sstevel@tonic-gate */ 3917c478bd9Sstevel@tonic-gate if (mod_uninstall_interval == 0) { 3927c478bd9Sstevel@tonic-gate mod_uninstall_interval = 60; 3937c478bd9Sstevel@tonic-gate modreap(); 3947c478bd9Sstevel@tonic-gate return (rval); 3957c478bd9Sstevel@tonic-gate } 3967c478bd9Sstevel@tonic-gate #endif 3977c478bd9Sstevel@tonic-gate mod_uninstall_all(); 3987c478bd9Sstevel@tonic-gate } else { 3997c478bd9Sstevel@tonic-gate rval = modunload(id); 4007c478bd9Sstevel@tonic-gate } 4017c478bd9Sstevel@tonic-gate return (rval); 4027c478bd9Sstevel@tonic-gate } 4037c478bd9Sstevel@tonic-gate 4047c478bd9Sstevel@tonic-gate static int 4057c478bd9Sstevel@tonic-gate modctl_modinfo(modid_t id, struct modinfo *umodi) 4067c478bd9Sstevel@tonic-gate { 4077c478bd9Sstevel@tonic-gate int retval; 4087c478bd9Sstevel@tonic-gate struct modinfo modi; 4097c478bd9Sstevel@tonic-gate #if defined(_SYSCALL32_IMPL) 4107c478bd9Sstevel@tonic-gate int nobase; 4117c478bd9Sstevel@tonic-gate struct modinfo32 modi32; 4127c478bd9Sstevel@tonic-gate #endif 4137c478bd9Sstevel@tonic-gate 4147c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 4157c478bd9Sstevel@tonic-gate if (copyin(umodi, &modi, sizeof (struct modinfo)) != 0) 4167c478bd9Sstevel@tonic-gate return (EFAULT); 4177c478bd9Sstevel@tonic-gate } 4187c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 4197c478bd9Sstevel@tonic-gate else { 4207c478bd9Sstevel@tonic-gate bzero(&modi, sizeof (modi)); 4217c478bd9Sstevel@tonic-gate if (copyin(umodi, &modi32, sizeof (struct modinfo32)) != 0) 4227c478bd9Sstevel@tonic-gate return (EFAULT); 4237c478bd9Sstevel@tonic-gate modi.mi_info = modi32.mi_info; 4247c478bd9Sstevel@tonic-gate modi.mi_id = modi32.mi_id; 4257c478bd9Sstevel@tonic-gate modi.mi_nextid = modi32.mi_nextid; 4267c478bd9Sstevel@tonic-gate nobase = modi.mi_info & MI_INFO_NOBASE; 4277c478bd9Sstevel@tonic-gate } 4287c478bd9Sstevel@tonic-gate #endif 4297c478bd9Sstevel@tonic-gate /* 4307c478bd9Sstevel@tonic-gate * This flag is -only- for the kernels use. 4317c478bd9Sstevel@tonic-gate */ 4327c478bd9Sstevel@tonic-gate modi.mi_info &= ~MI_INFO_LINKAGE; 4337c478bd9Sstevel@tonic-gate 4347c478bd9Sstevel@tonic-gate retval = modinfo(id, &modi); 4357c478bd9Sstevel@tonic-gate if (retval) 4367c478bd9Sstevel@tonic-gate return (retval); 4377c478bd9Sstevel@tonic-gate 4387c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 4397c478bd9Sstevel@tonic-gate if (copyout(&modi, umodi, sizeof (struct modinfo)) != 0) 4407c478bd9Sstevel@tonic-gate retval = EFAULT; 4417c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 4427c478bd9Sstevel@tonic-gate } else { 4437c478bd9Sstevel@tonic-gate int i; 4447c478bd9Sstevel@tonic-gate 4457c478bd9Sstevel@tonic-gate if (!nobase && (uintptr_t)modi.mi_base > UINT32_MAX) 4467c478bd9Sstevel@tonic-gate return (EOVERFLOW); 4477c478bd9Sstevel@tonic-gate 4487c478bd9Sstevel@tonic-gate modi32.mi_info = modi.mi_info; 4497c478bd9Sstevel@tonic-gate modi32.mi_state = modi.mi_state; 4507c478bd9Sstevel@tonic-gate modi32.mi_id = modi.mi_id; 4517c478bd9Sstevel@tonic-gate modi32.mi_nextid = modi.mi_nextid; 4527c478bd9Sstevel@tonic-gate modi32.mi_base = (caddr32_t)(uintptr_t)modi.mi_base; 4537c478bd9Sstevel@tonic-gate modi32.mi_size = modi.mi_size; 4547c478bd9Sstevel@tonic-gate modi32.mi_rev = modi.mi_rev; 4557c478bd9Sstevel@tonic-gate modi32.mi_loadcnt = modi.mi_loadcnt; 4567c478bd9Sstevel@tonic-gate bcopy(modi.mi_name, modi32.mi_name, sizeof (modi32.mi_name)); 4577c478bd9Sstevel@tonic-gate for (i = 0; i < MODMAXLINK32; i++) { 4587c478bd9Sstevel@tonic-gate modi32.mi_msinfo[i].msi_p0 = modi.mi_msinfo[i].msi_p0; 4597c478bd9Sstevel@tonic-gate bcopy(modi.mi_msinfo[i].msi_linkinfo, 4607c478bd9Sstevel@tonic-gate modi32.mi_msinfo[i].msi_linkinfo, 4617c478bd9Sstevel@tonic-gate sizeof (modi32.mi_msinfo[0].msi_linkinfo)); 4627c478bd9Sstevel@tonic-gate } 4637c478bd9Sstevel@tonic-gate if (copyout(&modi32, umodi, sizeof (struct modinfo32)) != 0) 4647c478bd9Sstevel@tonic-gate retval = EFAULT; 4657c478bd9Sstevel@tonic-gate #endif 4667c478bd9Sstevel@tonic-gate } 4677c478bd9Sstevel@tonic-gate 4687c478bd9Sstevel@tonic-gate return (retval); 4697c478bd9Sstevel@tonic-gate } 4707c478bd9Sstevel@tonic-gate 4717c478bd9Sstevel@tonic-gate /* 4727c478bd9Sstevel@tonic-gate * Return the last major number in the range of permissible major numbers. 4737c478bd9Sstevel@tonic-gate */ 4747c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 4757c478bd9Sstevel@tonic-gate static int 4767c478bd9Sstevel@tonic-gate modctl_modreserve(modid_t id, int *data) 4777c478bd9Sstevel@tonic-gate { 4787c478bd9Sstevel@tonic-gate if (copyout(&devcnt, data, sizeof (devcnt)) != 0) 4797c478bd9Sstevel@tonic-gate return (EFAULT); 4807c478bd9Sstevel@tonic-gate return (0); 4817c478bd9Sstevel@tonic-gate } 4827c478bd9Sstevel@tonic-gate 4836532b960SJerry Gilliam /* Add/Remove driver and binding aliases */ 4847c478bd9Sstevel@tonic-gate static int 4856532b960SJerry Gilliam modctl_update_driver_aliases(int add, int *data) 4867c478bd9Sstevel@tonic-gate { 4877c478bd9Sstevel@tonic-gate struct modconfig mc; 4886532b960SJerry Gilliam int i, n, rv = 0; 4897c478bd9Sstevel@tonic-gate struct aliases alias; 4907c478bd9Sstevel@tonic-gate struct aliases *ap; 4917c478bd9Sstevel@tonic-gate char name[MAXMODCONFNAME]; 4927c478bd9Sstevel@tonic-gate char cname[MAXMODCONFNAME]; 4937c478bd9Sstevel@tonic-gate char *drvname; 4946532b960SJerry Gilliam int resid; 4956532b960SJerry Gilliam struct alias_info { 4966532b960SJerry Gilliam char *alias_name; 4976532b960SJerry Gilliam int alias_resid; 4986532b960SJerry Gilliam } *aliases, *aip; 4997c478bd9Sstevel@tonic-gate 5007c478bd9Sstevel@tonic-gate bzero(&mc, sizeof (struct modconfig)); 5017c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 5027c478bd9Sstevel@tonic-gate if (copyin(data, &mc, sizeof (struct modconfig)) != 0) 5037c478bd9Sstevel@tonic-gate return (EFAULT); 5047c478bd9Sstevel@tonic-gate } 5057c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 5067c478bd9Sstevel@tonic-gate else { 5077c478bd9Sstevel@tonic-gate struct modconfig32 modc32; 5087c478bd9Sstevel@tonic-gate if (copyin(data, &modc32, sizeof (struct modconfig32)) != 0) 5097c478bd9Sstevel@tonic-gate return (EFAULT); 5107c478bd9Sstevel@tonic-gate else { 5117c478bd9Sstevel@tonic-gate bcopy(modc32.drvname, mc.drvname, 5127c478bd9Sstevel@tonic-gate sizeof (modc32.drvname)); 5137c478bd9Sstevel@tonic-gate bcopy(modc32.drvclass, mc.drvclass, 5147c478bd9Sstevel@tonic-gate sizeof (modc32.drvclass)); 5157c478bd9Sstevel@tonic-gate mc.major = modc32.major; 5166532b960SJerry Gilliam mc.flags = modc32.flags; 5177c478bd9Sstevel@tonic-gate mc.num_aliases = modc32.num_aliases; 5187c478bd9Sstevel@tonic-gate mc.ap = (struct aliases *)(uintptr_t)modc32.ap; 5197c478bd9Sstevel@tonic-gate } 5207c478bd9Sstevel@tonic-gate } 5217c478bd9Sstevel@tonic-gate #endif 5227c478bd9Sstevel@tonic-gate 5237c478bd9Sstevel@tonic-gate /* 5247c478bd9Sstevel@tonic-gate * If the driver is already in the mb_hashtab, and the name given 5257c478bd9Sstevel@tonic-gate * doesn't match that driver's name, fail. Otherwise, pass, since 5267c478bd9Sstevel@tonic-gate * we may be adding aliases. 5277c478bd9Sstevel@tonic-gate */ 5286532b960SJerry Gilliam drvname = mod_major_to_name(mc.major); 5296532b960SJerry Gilliam if ((drvname != NULL) && strcmp(drvname, mc.drvname) != 0) 5307c478bd9Sstevel@tonic-gate return (EINVAL); 5317c478bd9Sstevel@tonic-gate 5327c478bd9Sstevel@tonic-gate /* 5336532b960SJerry Gilliam * Precede alias removal by unbinding as many devices as possible. 5346532b960SJerry Gilliam */ 5356532b960SJerry Gilliam if (add == 0) { 5366532b960SJerry Gilliam (void) i_ddi_unload_drvconf(mc.major); 5376532b960SJerry Gilliam i_ddi_unbind_devs(mc.major); 5386532b960SJerry Gilliam } 5396532b960SJerry Gilliam 5406532b960SJerry Gilliam /* 5416532b960SJerry Gilliam * Add/remove each supplied driver alias to/from mb_hashtab 5427c478bd9Sstevel@tonic-gate */ 5437c478bd9Sstevel@tonic-gate ap = mc.ap; 5446532b960SJerry Gilliam if (mc.num_aliases > 0) 5456532b960SJerry Gilliam aliases = kmem_zalloc( 5466532b960SJerry Gilliam mc.num_aliases * sizeof (struct alias_info), KM_SLEEP); 5476532b960SJerry Gilliam aip = aliases; 5487c478bd9Sstevel@tonic-gate for (i = 0; i < mc.num_aliases; i++) { 5497c478bd9Sstevel@tonic-gate bzero(&alias, sizeof (struct aliases)); 5507c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 5516532b960SJerry Gilliam if (copyin(ap, &alias, sizeof (struct aliases)) != 0) { 5526532b960SJerry Gilliam rv = EFAULT; 5536532b960SJerry Gilliam goto error; 5546532b960SJerry Gilliam } 5556532b960SJerry Gilliam if (alias.a_len > MAXMODCONFNAME) { 5566532b960SJerry Gilliam rv = EINVAL; 5576532b960SJerry Gilliam goto error; 5586532b960SJerry Gilliam } 5596532b960SJerry Gilliam if (copyin(alias.a_name, name, alias.a_len) != 0) { 5606532b960SJerry Gilliam rv = EFAULT; 5616532b960SJerry Gilliam goto error; 5626532b960SJerry Gilliam } 5636532b960SJerry Gilliam if (name[alias.a_len - 1] != '\0') { 5646532b960SJerry Gilliam rv = EINVAL; 5656532b960SJerry Gilliam goto error; 5666532b960SJerry Gilliam } 5677c478bd9Sstevel@tonic-gate } 5687c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 5697c478bd9Sstevel@tonic-gate else { 5707c478bd9Sstevel@tonic-gate struct aliases32 al32; 5717c478bd9Sstevel@tonic-gate bzero(&al32, sizeof (struct aliases32)); 5726532b960SJerry Gilliam if (copyin(ap, &al32, sizeof (struct aliases32)) != 0) { 5736532b960SJerry Gilliam rv = EFAULT; 5746532b960SJerry Gilliam goto error; 5756532b960SJerry Gilliam } 5766532b960SJerry Gilliam if (al32.a_len > MAXMODCONFNAME) { 5776532b960SJerry Gilliam rv = EINVAL; 5786532b960SJerry Gilliam goto error; 5796532b960SJerry Gilliam } 5807c478bd9Sstevel@tonic-gate if (copyin((void *)(uintptr_t)al32.a_name, 5816532b960SJerry Gilliam name, al32.a_len) != 0) { 5826532b960SJerry Gilliam rv = EFAULT; 5836532b960SJerry Gilliam goto error; 5846532b960SJerry Gilliam } 5856532b960SJerry Gilliam if (name[al32.a_len - 1] != '\0') { 5866532b960SJerry Gilliam rv = EINVAL; 5876532b960SJerry Gilliam goto error; 5886532b960SJerry Gilliam } 5897c478bd9Sstevel@tonic-gate alias.a_next = (void *)(uintptr_t)al32.a_next; 5907c478bd9Sstevel@tonic-gate } 5917c478bd9Sstevel@tonic-gate #endif 5927c478bd9Sstevel@tonic-gate check_esc_sequences(name, cname); 5937f0b8309SEdward Pilatowicz aip->alias_name = strdup(cname); 5947c478bd9Sstevel@tonic-gate ap = alias.a_next; 5956532b960SJerry Gilliam aip++; 5966532b960SJerry Gilliam } 5976532b960SJerry Gilliam 5986532b960SJerry Gilliam if (add == 0) { 5996532b960SJerry Gilliam ap = mc.ap; 6006532b960SJerry Gilliam resid = 0; 6016532b960SJerry Gilliam aip = aliases; 6026532b960SJerry Gilliam /* attempt to unbind all devices bound to each alias */ 6036532b960SJerry Gilliam for (i = 0; i < mc.num_aliases; i++) { 6046532b960SJerry Gilliam n = i_ddi_unbind_devs_by_alias( 6056532b960SJerry Gilliam mc.major, aip->alias_name); 6066532b960SJerry Gilliam resid += n; 6076532b960SJerry Gilliam aip->alias_resid = n; 6087c478bd9Sstevel@tonic-gate } 6097c478bd9Sstevel@tonic-gate 6107c478bd9Sstevel@tonic-gate /* 6116532b960SJerry Gilliam * If some device bound to an alias remains in use, 6126532b960SJerry Gilliam * and override wasn't specified, no change is made to 6136532b960SJerry Gilliam * the binding state and we fail the operation. 6146532b960SJerry Gilliam */ 6156532b960SJerry Gilliam if (resid > 0 && ((mc.flags & MOD_UNBIND_OVERRIDE) == 0)) { 6166532b960SJerry Gilliam rv = EBUSY; 6176532b960SJerry Gilliam goto error; 6186532b960SJerry Gilliam } 6196532b960SJerry Gilliam 6206532b960SJerry Gilliam /* 6216532b960SJerry Gilliam * No device remains bound of any of the aliases, 6226532b960SJerry Gilliam * or force was requested. Mark each alias as 6236532b960SJerry Gilliam * inactive via delete_mbind so no future binds 6246532b960SJerry Gilliam * to this alias take place and that a new 6256532b960SJerry Gilliam * binding can be established. 6266532b960SJerry Gilliam */ 6276532b960SJerry Gilliam aip = aliases; 6286532b960SJerry Gilliam for (i = 0; i < mc.num_aliases; i++) { 6296532b960SJerry Gilliam if (moddebug & MODDEBUG_BINDING) 6306532b960SJerry Gilliam cmn_err(CE_CONT, "Removing binding for %s " 6316532b960SJerry Gilliam "(%d active references)\n", 6326532b960SJerry Gilliam aip->alias_name, aip->alias_resid); 6336532b960SJerry Gilliam delete_mbind(aip->alias_name, mb_hashtab); 6346532b960SJerry Gilliam aip++; 6356532b960SJerry Gilliam } 6366532b960SJerry Gilliam rv = 0; 6376532b960SJerry Gilliam } else { 6386532b960SJerry Gilliam aip = aliases; 6396532b960SJerry Gilliam for (i = 0; i < mc.num_aliases; i++) { 6406532b960SJerry Gilliam if (moddebug & MODDEBUG_BINDING) 6416532b960SJerry Gilliam cmn_err(CE_NOTE, "Adding binding for '%s'\n", 6426532b960SJerry Gilliam aip->alias_name); 6436532b960SJerry Gilliam (void) make_mbind(aip->alias_name, 6446532b960SJerry Gilliam mc.major, NULL, mb_hashtab); 6456532b960SJerry Gilliam aip++; 6466532b960SJerry Gilliam } 6476532b960SJerry Gilliam /* 6486532b960SJerry Gilliam * Try to establish an mbinding for mc.drvname, and add it to 6496532b960SJerry Gilliam * devnames. Add class if any after establishing the major 6506532b960SJerry Gilliam * number. 6517c478bd9Sstevel@tonic-gate */ 6527c478bd9Sstevel@tonic-gate (void) make_mbind(mc.drvname, mc.major, NULL, mb_hashtab); 653c9cc1492SJerry Gilliam if ((rv = make_devname(mc.drvname, mc.major, 654c9cc1492SJerry Gilliam (mc.flags & MOD_ADDMAJBIND_UPDATE) ? 655c9cc1492SJerry Gilliam DN_DRIVER_INACTIVE : 0)) != 0) { 6566532b960SJerry Gilliam goto error; 657c9cc1492SJerry Gilliam } 6587c478bd9Sstevel@tonic-gate 6597c478bd9Sstevel@tonic-gate if (mc.drvclass[0] != '\0') 6607c478bd9Sstevel@tonic-gate add_class(mc.drvname, mc.drvclass); 661c9cc1492SJerry Gilliam if ((mc.flags & MOD_ADDMAJBIND_UPDATE) == 0) { 6627c478bd9Sstevel@tonic-gate (void) i_ddi_load_drvconf(mc.major); 6636532b960SJerry Gilliam } 664c9cc1492SJerry Gilliam } 6656532b960SJerry Gilliam 6666532b960SJerry Gilliam /* 6676532b960SJerry Gilliam * Ensure that all nodes are bound to the most appropriate driver 6686532b960SJerry Gilliam * possible, attempting demotion and rebind when a more appropriate 669c9cc1492SJerry Gilliam * driver now exists. But not when adding a driver update-only. 6706532b960SJerry Gilliam */ 671c9cc1492SJerry Gilliam if ((add == 0) || ((mc.flags & MOD_ADDMAJBIND_UPDATE) == 0)) { 6727c478bd9Sstevel@tonic-gate i_ddi_bind_devs(); 6734c06356bSdh142964 i_ddi_di_cache_invalidate(); 674c9cc1492SJerry Gilliam } 6756532b960SJerry Gilliam 6766532b960SJerry Gilliam error: 6776532b960SJerry Gilliam if (mc.num_aliases > 0) { 6786532b960SJerry Gilliam aip = aliases; 6796532b960SJerry Gilliam for (i = 0; i < mc.num_aliases; i++) { 6806532b960SJerry Gilliam if (aip->alias_name != NULL) 6816532b960SJerry Gilliam strfree(aip->alias_name); 6826532b960SJerry Gilliam aip++; 6836532b960SJerry Gilliam } 6846532b960SJerry Gilliam kmem_free(aliases, mc.num_aliases * sizeof (struct alias_info)); 6857c478bd9Sstevel@tonic-gate } 6867c478bd9Sstevel@tonic-gate return (rv); 6877c478bd9Sstevel@tonic-gate } 6887c478bd9Sstevel@tonic-gate 6897c478bd9Sstevel@tonic-gate static int 6906532b960SJerry Gilliam modctl_add_driver_aliases(int *data) 6916532b960SJerry Gilliam { 6926532b960SJerry Gilliam return (modctl_update_driver_aliases(1, data)); 6936532b960SJerry Gilliam } 6946532b960SJerry Gilliam 6956532b960SJerry Gilliam static int 6966532b960SJerry Gilliam modctl_remove_driver_aliases(int *data) 6976532b960SJerry Gilliam { 6986532b960SJerry Gilliam return (modctl_update_driver_aliases(0, data)); 6996532b960SJerry Gilliam } 7006532b960SJerry Gilliam 7016532b960SJerry Gilliam static int 7027c478bd9Sstevel@tonic-gate modctl_rem_major(major_t major) 7037c478bd9Sstevel@tonic-gate { 7047c478bd9Sstevel@tonic-gate struct devnames *dnp; 7057c478bd9Sstevel@tonic-gate 7067c478bd9Sstevel@tonic-gate if (major >= devcnt) 7077c478bd9Sstevel@tonic-gate return (EINVAL); 7087c478bd9Sstevel@tonic-gate 7097c478bd9Sstevel@tonic-gate /* mark devnames as removed */ 7107c478bd9Sstevel@tonic-gate dnp = &devnamesp[major]; 7117c478bd9Sstevel@tonic-gate LOCK_DEV_OPS(&dnp->dn_lock); 7127c478bd9Sstevel@tonic-gate if (dnp->dn_name == NULL || 7137c478bd9Sstevel@tonic-gate (dnp->dn_flags & (DN_DRIVER_REMOVED | DN_TAKEN_GETUDEV))) { 7147c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 7157c478bd9Sstevel@tonic-gate return (EINVAL); 7167c478bd9Sstevel@tonic-gate } 7177c478bd9Sstevel@tonic-gate dnp->dn_flags |= DN_DRIVER_REMOVED; 7187c478bd9Sstevel@tonic-gate pm_driver_removed(major); 7197c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 7207c478bd9Sstevel@tonic-gate 7217c478bd9Sstevel@tonic-gate (void) i_ddi_unload_drvconf(major); 7227c478bd9Sstevel@tonic-gate i_ddi_unbind_devs(major); 7236532b960SJerry Gilliam i_ddi_bind_devs(); 7244c06356bSdh142964 i_ddi_di_cache_invalidate(); 7256532b960SJerry Gilliam 7266532b960SJerry Gilliam /* purge all the bindings to this driver */ 7276532b960SJerry Gilliam purge_mbind(major, mb_hashtab); 7287c478bd9Sstevel@tonic-gate return (0); 7297c478bd9Sstevel@tonic-gate } 7307c478bd9Sstevel@tonic-gate 7317c478bd9Sstevel@tonic-gate static struct vfs * 7327c478bd9Sstevel@tonic-gate path_to_vfs(char *name) 7337c478bd9Sstevel@tonic-gate { 7347c478bd9Sstevel@tonic-gate vnode_t *vp; 7357c478bd9Sstevel@tonic-gate struct vfs *vfsp; 7367c478bd9Sstevel@tonic-gate 7377c478bd9Sstevel@tonic-gate if (lookupname(name, UIO_SYSSPACE, FOLLOW, NULLVPP, &vp)) 7387c478bd9Sstevel@tonic-gate return (NULL); 7397c478bd9Sstevel@tonic-gate 7407c478bd9Sstevel@tonic-gate vfsp = vp->v_vfsp; 7417c478bd9Sstevel@tonic-gate VN_RELE(vp); 7427c478bd9Sstevel@tonic-gate return (vfsp); 7437c478bd9Sstevel@tonic-gate } 7447c478bd9Sstevel@tonic-gate 7457c478bd9Sstevel@tonic-gate static int 7467c478bd9Sstevel@tonic-gate new_vfs_in_modpath() 7477c478bd9Sstevel@tonic-gate { 7487c478bd9Sstevel@tonic-gate static int n_modpath = 0; 7497c478bd9Sstevel@tonic-gate static char *modpath_copy; 7507c478bd9Sstevel@tonic-gate static struct pathvfs { 7517c478bd9Sstevel@tonic-gate char *path; 7527c478bd9Sstevel@tonic-gate struct vfs *vfsp; 7537c478bd9Sstevel@tonic-gate } *pathvfs; 7547c478bd9Sstevel@tonic-gate 7557c478bd9Sstevel@tonic-gate int i, new_vfs = 0; 7567c478bd9Sstevel@tonic-gate char *tmp, *tmp1; 7577c478bd9Sstevel@tonic-gate struct vfs *vfsp; 7587c478bd9Sstevel@tonic-gate 7597c478bd9Sstevel@tonic-gate if (n_modpath != 0) { 7607c478bd9Sstevel@tonic-gate for (i = 0; i < n_modpath; i++) { 7617c478bd9Sstevel@tonic-gate vfsp = path_to_vfs(pathvfs[i].path); 7627c478bd9Sstevel@tonic-gate if (vfsp != pathvfs[i].vfsp) { 7637c478bd9Sstevel@tonic-gate pathvfs[i].vfsp = vfsp; 7647c478bd9Sstevel@tonic-gate if (vfsp) 7657c478bd9Sstevel@tonic-gate new_vfs = 1; 7667c478bd9Sstevel@tonic-gate } 7677c478bd9Sstevel@tonic-gate } 7687c478bd9Sstevel@tonic-gate return (new_vfs); 7697c478bd9Sstevel@tonic-gate } 7707c478bd9Sstevel@tonic-gate 7717c478bd9Sstevel@tonic-gate /* 7727c478bd9Sstevel@tonic-gate * First call, initialize the pathvfs structure 7737c478bd9Sstevel@tonic-gate */ 7747c478bd9Sstevel@tonic-gate modpath_copy = i_ddi_strdup(default_path, KM_SLEEP); 7757c478bd9Sstevel@tonic-gate tmp = modpath_copy; 7767c478bd9Sstevel@tonic-gate n_modpath = 1; 7777c478bd9Sstevel@tonic-gate tmp1 = strchr(tmp, ' '); 7787c478bd9Sstevel@tonic-gate while (tmp1) { 7797c478bd9Sstevel@tonic-gate *tmp1 = '\0'; 7807c478bd9Sstevel@tonic-gate n_modpath++; 7817c478bd9Sstevel@tonic-gate tmp = tmp1 + 1; 7827c478bd9Sstevel@tonic-gate tmp1 = strchr(tmp, ' '); 7837c478bd9Sstevel@tonic-gate } 7847c478bd9Sstevel@tonic-gate 7857c478bd9Sstevel@tonic-gate pathvfs = kmem_zalloc(n_modpath * sizeof (struct pathvfs), KM_SLEEP); 7867c478bd9Sstevel@tonic-gate tmp = modpath_copy; 7877c478bd9Sstevel@tonic-gate for (i = 0; i < n_modpath; i++) { 7887c478bd9Sstevel@tonic-gate pathvfs[i].path = tmp; 7897c478bd9Sstevel@tonic-gate vfsp = path_to_vfs(tmp); 7907c478bd9Sstevel@tonic-gate pathvfs[i].vfsp = vfsp; 7917c478bd9Sstevel@tonic-gate tmp += strlen(tmp) + 1; 7927c478bd9Sstevel@tonic-gate } 7937c478bd9Sstevel@tonic-gate return (1); /* always reread driver.conf the first time */ 7947c478bd9Sstevel@tonic-gate } 7957c478bd9Sstevel@tonic-gate 796facf4a8dSllai1 static int 797c9cc1492SJerry Gilliam modctl_load_drvconf(major_t major, int flags) 7987c478bd9Sstevel@tonic-gate { 7997c478bd9Sstevel@tonic-gate int ret; 8007c478bd9Sstevel@tonic-gate 801c9cc1492SJerry Gilliam /* 802c9cc1492SJerry Gilliam * devfsadm -u - read all new driver.conf files 803c9cc1492SJerry Gilliam * and bind and configure devices for new drivers. 804c9cc1492SJerry Gilliam */ 805c9cc1492SJerry Gilliam if (flags & MOD_LOADDRVCONF_RECONF) { 806c9cc1492SJerry Gilliam (void) i_ddi_load_drvconf(DDI_MAJOR_T_NONE); 807c9cc1492SJerry Gilliam i_ddi_bind_devs(); 808c9cc1492SJerry Gilliam i_ddi_di_cache_invalidate(); 809c9cc1492SJerry Gilliam return (0); 810c9cc1492SJerry Gilliam } 811c9cc1492SJerry Gilliam 812c9cc1492SJerry Gilliam /* 813c9cc1492SJerry Gilliam * update_drv <drv> - reload driver.conf for the specified driver 814c9cc1492SJerry Gilliam */ 815a204de77Scth if (major != DDI_MAJOR_T_NONE) { 8167c478bd9Sstevel@tonic-gate ret = i_ddi_load_drvconf(major); 8177c478bd9Sstevel@tonic-gate if (ret == 0) 8187c478bd9Sstevel@tonic-gate i_ddi_bind_devs(); 8197c478bd9Sstevel@tonic-gate return (ret); 8207c478bd9Sstevel@tonic-gate } 8217c478bd9Sstevel@tonic-gate 8227c478bd9Sstevel@tonic-gate /* 8237c478bd9Sstevel@tonic-gate * We are invoked to rescan new driver.conf files. It is 8247c478bd9Sstevel@tonic-gate * only necessary if a new file system was mounted in the 8257c478bd9Sstevel@tonic-gate * module_path. Because rescanning driver.conf files can 8267c478bd9Sstevel@tonic-gate * take some time on older platforms (sun4m), the following 8277c478bd9Sstevel@tonic-gate * code skips unnecessary driver.conf rescans to optimize 8287c478bd9Sstevel@tonic-gate * boot performance. 8297c478bd9Sstevel@tonic-gate */ 8307c478bd9Sstevel@tonic-gate if (new_vfs_in_modpath()) { 831a204de77Scth (void) i_ddi_load_drvconf(DDI_MAJOR_T_NONE); 8327c478bd9Sstevel@tonic-gate /* 8337c478bd9Sstevel@tonic-gate * If we are still initializing io subsystem, 8347c478bd9Sstevel@tonic-gate * load drivers with ddi-forceattach property 8357c478bd9Sstevel@tonic-gate */ 8367c478bd9Sstevel@tonic-gate if (!i_ddi_io_initialized()) 8377c478bd9Sstevel@tonic-gate i_ddi_forceattach_drivers(); 8387c478bd9Sstevel@tonic-gate } 8397c478bd9Sstevel@tonic-gate return (0); 8407c478bd9Sstevel@tonic-gate } 8417c478bd9Sstevel@tonic-gate 8426532b960SJerry Gilliam /* 8436532b960SJerry Gilliam * Unload driver.conf file and follow up by attempting 8446532b960SJerry Gilliam * to rebind devices to more appropriate driver. 8456532b960SJerry Gilliam */ 8467c478bd9Sstevel@tonic-gate static int 8477c478bd9Sstevel@tonic-gate modctl_unload_drvconf(major_t major) 8487c478bd9Sstevel@tonic-gate { 8497c478bd9Sstevel@tonic-gate int ret; 8507c478bd9Sstevel@tonic-gate 8517c478bd9Sstevel@tonic-gate if (major >= devcnt) 8527c478bd9Sstevel@tonic-gate return (EINVAL); 8537c478bd9Sstevel@tonic-gate 8547c478bd9Sstevel@tonic-gate ret = i_ddi_unload_drvconf(major); 8557c478bd9Sstevel@tonic-gate if (ret != 0) 8567c478bd9Sstevel@tonic-gate return (ret); 8577c478bd9Sstevel@tonic-gate (void) i_ddi_unbind_devs(major); 8586532b960SJerry Gilliam i_ddi_bind_devs(); 8597c478bd9Sstevel@tonic-gate 8607c478bd9Sstevel@tonic-gate return (0); 8617c478bd9Sstevel@tonic-gate } 8627c478bd9Sstevel@tonic-gate 8637c478bd9Sstevel@tonic-gate static void 8647c478bd9Sstevel@tonic-gate check_esc_sequences(char *str, char *cstr) 8657c478bd9Sstevel@tonic-gate { 8667c478bd9Sstevel@tonic-gate int i; 8677c478bd9Sstevel@tonic-gate size_t len; 8687c478bd9Sstevel@tonic-gate char *p; 8697c478bd9Sstevel@tonic-gate 8707c478bd9Sstevel@tonic-gate len = strlen(str); 8717c478bd9Sstevel@tonic-gate for (i = 0; i < len; i++, str++, cstr++) { 8727c478bd9Sstevel@tonic-gate if (*str != '\\') { 8737c478bd9Sstevel@tonic-gate *cstr = *str; 8747c478bd9Sstevel@tonic-gate } else { 8757c478bd9Sstevel@tonic-gate p = str + 1; 8767c478bd9Sstevel@tonic-gate /* 8777c478bd9Sstevel@tonic-gate * we only handle octal escape sequences for SPACE 8787c478bd9Sstevel@tonic-gate */ 8797c478bd9Sstevel@tonic-gate if (*p++ == '0' && *p++ == '4' && *p == '0') { 8807c478bd9Sstevel@tonic-gate *cstr = ' '; 8817c478bd9Sstevel@tonic-gate str += 3; 8827c478bd9Sstevel@tonic-gate } else { 8837c478bd9Sstevel@tonic-gate *cstr = *str; 8847c478bd9Sstevel@tonic-gate } 8857c478bd9Sstevel@tonic-gate } 8867c478bd9Sstevel@tonic-gate } 8877c478bd9Sstevel@tonic-gate *cstr = 0; 8887c478bd9Sstevel@tonic-gate } 8897c478bd9Sstevel@tonic-gate 8907c478bd9Sstevel@tonic-gate static int 8917c478bd9Sstevel@tonic-gate modctl_getmodpathlen(int *data) 8927c478bd9Sstevel@tonic-gate { 8937c478bd9Sstevel@tonic-gate int len; 8947c478bd9Sstevel@tonic-gate len = strlen(default_path); 8957c478bd9Sstevel@tonic-gate if (copyout(&len, data, sizeof (len)) != 0) 8967c478bd9Sstevel@tonic-gate return (EFAULT); 8977c478bd9Sstevel@tonic-gate return (0); 8987c478bd9Sstevel@tonic-gate } 8997c478bd9Sstevel@tonic-gate 9007c478bd9Sstevel@tonic-gate static int 9017c478bd9Sstevel@tonic-gate modctl_getmodpath(char *data) 9027c478bd9Sstevel@tonic-gate { 9037c478bd9Sstevel@tonic-gate if (copyout(default_path, data, strlen(default_path) + 1) != 0) 9047c478bd9Sstevel@tonic-gate return (EFAULT); 9057c478bd9Sstevel@tonic-gate return (0); 9067c478bd9Sstevel@tonic-gate } 9077c478bd9Sstevel@tonic-gate 9087c478bd9Sstevel@tonic-gate static int 9097c478bd9Sstevel@tonic-gate modctl_read_sysbinding_file(void) 9107c478bd9Sstevel@tonic-gate { 9117c478bd9Sstevel@tonic-gate (void) read_binding_file(sysbind, sb_hashtab, make_mbind); 9127c478bd9Sstevel@tonic-gate return (0); 9137c478bd9Sstevel@tonic-gate } 9147c478bd9Sstevel@tonic-gate 9157c478bd9Sstevel@tonic-gate static int 9167c478bd9Sstevel@tonic-gate modctl_getmaj(char *uname, uint_t ulen, int *umajorp) 9177c478bd9Sstevel@tonic-gate { 9187c478bd9Sstevel@tonic-gate char name[256]; 9197c478bd9Sstevel@tonic-gate int retval; 9207c478bd9Sstevel@tonic-gate major_t major; 9217c478bd9Sstevel@tonic-gate 922facf4a8dSllai1 if (ulen == 0) 923facf4a8dSllai1 return (EINVAL); 9247c478bd9Sstevel@tonic-gate if ((retval = copyinstr(uname, name, 9257c478bd9Sstevel@tonic-gate (ulen < 256) ? ulen : 256, 0)) != 0) 9267c478bd9Sstevel@tonic-gate return (retval); 927a204de77Scth if ((major = mod_name_to_major(name)) == DDI_MAJOR_T_NONE) 9287c478bd9Sstevel@tonic-gate return (ENODEV); 9297c478bd9Sstevel@tonic-gate if (copyout(&major, umajorp, sizeof (major_t)) != 0) 9307c478bd9Sstevel@tonic-gate return (EFAULT); 9317c478bd9Sstevel@tonic-gate return (0); 9327c478bd9Sstevel@tonic-gate } 9337c478bd9Sstevel@tonic-gate 93425e8c5aaSvikram static char ** 93525e8c5aaSvikram convert_constraint_string(char *constraints, size_t len) 93625e8c5aaSvikram { 93725e8c5aaSvikram int i; 93825e8c5aaSvikram int n; 93925e8c5aaSvikram char *p; 94025e8c5aaSvikram char **array; 94125e8c5aaSvikram 94225e8c5aaSvikram ASSERT(constraints != NULL); 94325e8c5aaSvikram ASSERT(len > 0); 94425e8c5aaSvikram 9451e1ddd6cScth for (i = 0, p = constraints; strlen(p) > 0; i++, p += strlen(p) + 1) 9461e1ddd6cScth ; 94725e8c5aaSvikram 94825e8c5aaSvikram n = i; 94925e8c5aaSvikram 95025e8c5aaSvikram if (n == 0) { 95125e8c5aaSvikram kmem_free(constraints, len); 95225e8c5aaSvikram return (NULL); 95325e8c5aaSvikram } 95425e8c5aaSvikram 95525e8c5aaSvikram array = kmem_alloc((n + 1) * sizeof (char *), KM_SLEEP); 95625e8c5aaSvikram 95725e8c5aaSvikram for (i = 0, p = constraints; i < n; i++, p += strlen(p) + 1) { 95825e8c5aaSvikram array[i] = i_ddi_strdup(p, KM_SLEEP); 95925e8c5aaSvikram } 96025e8c5aaSvikram array[n] = NULL; 96125e8c5aaSvikram 96225e8c5aaSvikram kmem_free(constraints, len); 96325e8c5aaSvikram 96425e8c5aaSvikram return (array); 96525e8c5aaSvikram } 96625e8c5aaSvikram /*ARGSUSED*/ 96725e8c5aaSvikram static int 96825e8c5aaSvikram modctl_retire(char *path, char *uconstraints, size_t ulen) 96925e8c5aaSvikram { 97025e8c5aaSvikram char *pathbuf; 97125e8c5aaSvikram char *devpath; 97225e8c5aaSvikram size_t pathsz; 97325e8c5aaSvikram int retval; 97425e8c5aaSvikram char *constraints; 97525e8c5aaSvikram char **cons_array; 97625e8c5aaSvikram 97725e8c5aaSvikram if (path == NULL) 97825e8c5aaSvikram return (EINVAL); 97925e8c5aaSvikram 98025e8c5aaSvikram if ((uconstraints == NULL) ^ (ulen == 0)) 98125e8c5aaSvikram return (EINVAL); 98225e8c5aaSvikram 98325e8c5aaSvikram pathbuf = kmem_alloc(MAXPATHLEN, KM_SLEEP); 98425e8c5aaSvikram retval = copyinstr(path, pathbuf, MAXPATHLEN, &pathsz); 98525e8c5aaSvikram if (retval != 0) { 98625e8c5aaSvikram kmem_free(pathbuf, MAXPATHLEN); 98725e8c5aaSvikram return (retval); 98825e8c5aaSvikram } 98925e8c5aaSvikram devpath = i_ddi_strdup(pathbuf, KM_SLEEP); 99025e8c5aaSvikram kmem_free(pathbuf, MAXPATHLEN); 99125e8c5aaSvikram 99225e8c5aaSvikram /* 99325e8c5aaSvikram * First check if the device is already retired. 994*bf002425SStephen Hanson * If it is, then persist the retire anyway, just in case the retire 995*bf002425SStephen Hanson * store has got out of sync with the boot archive. 99625e8c5aaSvikram */ 99725e8c5aaSvikram if (e_ddi_device_retired(devpath)) { 99825e8c5aaSvikram cmn_err(CE_NOTE, "Device: already retired: %s", devpath); 999*bf002425SStephen Hanson (void) e_ddi_retire_persist(devpath); 100025e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 100125e8c5aaSvikram return (0); 100225e8c5aaSvikram } 100325e8c5aaSvikram 100425e8c5aaSvikram cons_array = NULL; 100525e8c5aaSvikram if (uconstraints) { 100625e8c5aaSvikram constraints = kmem_alloc(ulen, KM_SLEEP); 100725e8c5aaSvikram if (copyin(uconstraints, constraints, ulen)) { 100825e8c5aaSvikram kmem_free(constraints, ulen); 100925e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 101025e8c5aaSvikram return (EFAULT); 101125e8c5aaSvikram } 101225e8c5aaSvikram cons_array = convert_constraint_string(constraints, ulen); 101325e8c5aaSvikram } 101425e8c5aaSvikram 101525e8c5aaSvikram /* 101625e8c5aaSvikram * Try to retire the device first. The following 101725e8c5aaSvikram * routine will return an error only if the device 101825e8c5aaSvikram * is not retireable i.e. retire constraints forbid 101925e8c5aaSvikram * a retire. A return of success from this routine 102025e8c5aaSvikram * indicates that device is retireable. 102125e8c5aaSvikram */ 102225e8c5aaSvikram retval = e_ddi_retire_device(devpath, cons_array); 102325e8c5aaSvikram if (retval != DDI_SUCCESS) { 102425e8c5aaSvikram cmn_err(CE_WARN, "constraints forbid retire: %s", devpath); 102525e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 102625e8c5aaSvikram return (ENOTSUP); 102725e8c5aaSvikram } 102825e8c5aaSvikram 102925e8c5aaSvikram /* 103025e8c5aaSvikram * Ok, the retire succeeded. Persist the retire. 103125e8c5aaSvikram * If retiring a nexus, we need to only persist the 103225e8c5aaSvikram * nexus retire. Any children of a retired nexus 103325e8c5aaSvikram * are automatically covered by the retire store 103425e8c5aaSvikram * code. 103525e8c5aaSvikram */ 103625e8c5aaSvikram retval = e_ddi_retire_persist(devpath); 103725e8c5aaSvikram if (retval != 0) { 103825e8c5aaSvikram cmn_err(CE_WARN, "Failed to persist device retire: error %d: " 103925e8c5aaSvikram "%s", retval, devpath); 104025e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 104125e8c5aaSvikram return (retval); 104225e8c5aaSvikram } 104325e8c5aaSvikram if (moddebug & MODDEBUG_RETIRE) 104425e8c5aaSvikram cmn_err(CE_NOTE, "Persisted retire of device: %s", devpath); 104525e8c5aaSvikram 104625e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 104725e8c5aaSvikram return (0); 104825e8c5aaSvikram } 104925e8c5aaSvikram 105025e8c5aaSvikram static int 105125e8c5aaSvikram modctl_is_retired(char *path, int *statep) 105225e8c5aaSvikram { 105325e8c5aaSvikram char *pathbuf; 105425e8c5aaSvikram char *devpath; 105525e8c5aaSvikram size_t pathsz; 105625e8c5aaSvikram int error; 105725e8c5aaSvikram int status; 105825e8c5aaSvikram 105925e8c5aaSvikram if (path == NULL || statep == NULL) 106025e8c5aaSvikram return (EINVAL); 106125e8c5aaSvikram 106225e8c5aaSvikram pathbuf = kmem_alloc(MAXPATHLEN, KM_SLEEP); 106325e8c5aaSvikram error = copyinstr(path, pathbuf, MAXPATHLEN, &pathsz); 106425e8c5aaSvikram if (error != 0) { 106525e8c5aaSvikram kmem_free(pathbuf, MAXPATHLEN); 106625e8c5aaSvikram return (error); 106725e8c5aaSvikram } 106825e8c5aaSvikram devpath = i_ddi_strdup(pathbuf, KM_SLEEP); 106925e8c5aaSvikram kmem_free(pathbuf, MAXPATHLEN); 107025e8c5aaSvikram 107125e8c5aaSvikram if (e_ddi_device_retired(devpath)) 107225e8c5aaSvikram status = 1; 107325e8c5aaSvikram else 107425e8c5aaSvikram status = 0; 107525e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 107625e8c5aaSvikram 107725e8c5aaSvikram return (copyout(&status, statep, sizeof (status)) ? EFAULT : 0); 107825e8c5aaSvikram } 107925e8c5aaSvikram 108025e8c5aaSvikram static int 108125e8c5aaSvikram modctl_unretire(char *path) 108225e8c5aaSvikram { 108325e8c5aaSvikram char *pathbuf; 108425e8c5aaSvikram char *devpath; 108525e8c5aaSvikram size_t pathsz; 108625e8c5aaSvikram int retired; 108725e8c5aaSvikram int retval; 108825e8c5aaSvikram 108925e8c5aaSvikram if (path == NULL) 109025e8c5aaSvikram return (EINVAL); 109125e8c5aaSvikram 109225e8c5aaSvikram pathbuf = kmem_alloc(MAXPATHLEN, KM_SLEEP); 109325e8c5aaSvikram retval = copyinstr(path, pathbuf, MAXPATHLEN, &pathsz); 109425e8c5aaSvikram if (retval != 0) { 109525e8c5aaSvikram kmem_free(pathbuf, MAXPATHLEN); 109625e8c5aaSvikram return (retval); 109725e8c5aaSvikram } 109825e8c5aaSvikram devpath = i_ddi_strdup(pathbuf, KM_SLEEP); 109925e8c5aaSvikram kmem_free(pathbuf, MAXPATHLEN); 110025e8c5aaSvikram 110125e8c5aaSvikram /* 110225e8c5aaSvikram * We check if a device is retired (first) before 110325e8c5aaSvikram * unpersisting the retire, because we use the 110425e8c5aaSvikram * retire store to determine if a device is retired. 110525e8c5aaSvikram * If we unpersist first, the device will always appear 110625e8c5aaSvikram * to be unretired. For the rationale behind unpersisting 110725e8c5aaSvikram * a device that is not retired, see the next comment. 110825e8c5aaSvikram */ 110925e8c5aaSvikram retired = e_ddi_device_retired(devpath); 111025e8c5aaSvikram 111125e8c5aaSvikram /* 111225e8c5aaSvikram * We call unpersist unconditionally because the lookup 111325e8c5aaSvikram * for retired devices (e_ddi_device_retired()), skips "bypassed" 111425e8c5aaSvikram * devices. We still want to be able remove "bypassed" entries 111525e8c5aaSvikram * from the persistent store, so we unpersist unconditionally 111625e8c5aaSvikram * i.e. whether or not the entry is found on a lookup. 111725e8c5aaSvikram * 111825e8c5aaSvikram * e_ddi_retire_unpersist() returns 1 if it found and cleared 111925e8c5aaSvikram * an entry from the retire store or 0 otherwise. 112025e8c5aaSvikram */ 112125e8c5aaSvikram if (e_ddi_retire_unpersist(devpath)) 112225e8c5aaSvikram if (moddebug & MODDEBUG_RETIRE) { 112325e8c5aaSvikram cmn_err(CE_NOTE, "Unpersisted retire of device: %s", 112425e8c5aaSvikram devpath); 112525e8c5aaSvikram } 112625e8c5aaSvikram 112725e8c5aaSvikram /* 112825e8c5aaSvikram * Check if the device is already unretired. If so, 112925e8c5aaSvikram * the unretire becomes a NOP 113025e8c5aaSvikram */ 113125e8c5aaSvikram if (!retired) { 113225e8c5aaSvikram cmn_err(CE_NOTE, "Not retired: %s", devpath); 113325e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 113425e8c5aaSvikram return (0); 113525e8c5aaSvikram } 113625e8c5aaSvikram 113725e8c5aaSvikram retval = e_ddi_unretire_device(devpath); 113825e8c5aaSvikram if (retval != 0) { 113925e8c5aaSvikram cmn_err(CE_WARN, "cannot unretire device: error %d, path %s\n", 114025e8c5aaSvikram retval, devpath); 114125e8c5aaSvikram } 114225e8c5aaSvikram 114325e8c5aaSvikram kmem_free(devpath, strlen(devpath) + 1); 114425e8c5aaSvikram 114525e8c5aaSvikram return (retval); 114625e8c5aaSvikram } 114725e8c5aaSvikram 11487c478bd9Sstevel@tonic-gate static int 11497c478bd9Sstevel@tonic-gate modctl_getname(char *uname, uint_t ulen, int *umajorp) 11507c478bd9Sstevel@tonic-gate { 11517c478bd9Sstevel@tonic-gate char *name; 11527c478bd9Sstevel@tonic-gate major_t major; 11537c478bd9Sstevel@tonic-gate 11547c478bd9Sstevel@tonic-gate if (copyin(umajorp, &major, sizeof (major)) != 0) 11557c478bd9Sstevel@tonic-gate return (EFAULT); 11567c478bd9Sstevel@tonic-gate if ((name = mod_major_to_name(major)) == NULL) 11577c478bd9Sstevel@tonic-gate return (ENODEV); 11587c478bd9Sstevel@tonic-gate if ((strlen(name) + 1) > ulen) 11597c478bd9Sstevel@tonic-gate return (ENOSPC); 11607c478bd9Sstevel@tonic-gate return (copyoutstr(name, uname, ulen, NULL)); 11617c478bd9Sstevel@tonic-gate } 11627c478bd9Sstevel@tonic-gate 11637c478bd9Sstevel@tonic-gate static int 11647c478bd9Sstevel@tonic-gate modctl_devt2instance(dev_t dev, int *uinstancep) 11657c478bd9Sstevel@tonic-gate { 11667c478bd9Sstevel@tonic-gate int instance; 11677c478bd9Sstevel@tonic-gate 11687c478bd9Sstevel@tonic-gate if ((instance = dev_to_instance(dev)) == -1) 11697c478bd9Sstevel@tonic-gate return (EINVAL); 11707c478bd9Sstevel@tonic-gate 11717c478bd9Sstevel@tonic-gate return (copyout(&instance, uinstancep, sizeof (int))); 11727c478bd9Sstevel@tonic-gate } 11737c478bd9Sstevel@tonic-gate 11747c478bd9Sstevel@tonic-gate /* 11757c478bd9Sstevel@tonic-gate * Return the sizeof of the device id. 11767c478bd9Sstevel@tonic-gate */ 11777c478bd9Sstevel@tonic-gate static int 11787c478bd9Sstevel@tonic-gate modctl_sizeof_devid(dev_t dev, uint_t *len) 11797c478bd9Sstevel@tonic-gate { 11807c478bd9Sstevel@tonic-gate uint_t sz; 11817c478bd9Sstevel@tonic-gate ddi_devid_t devid; 11827c478bd9Sstevel@tonic-gate 11837c478bd9Sstevel@tonic-gate /* get device id */ 11847c478bd9Sstevel@tonic-gate if (ddi_lyr_get_devid(dev, &devid) == DDI_FAILURE) 11857c478bd9Sstevel@tonic-gate return (EINVAL); 11867c478bd9Sstevel@tonic-gate 11877c478bd9Sstevel@tonic-gate sz = ddi_devid_sizeof(devid); 11887c478bd9Sstevel@tonic-gate ddi_devid_free(devid); 11897c478bd9Sstevel@tonic-gate 11907c478bd9Sstevel@tonic-gate /* copyout device id size */ 11917c478bd9Sstevel@tonic-gate if (copyout(&sz, len, sizeof (sz)) != 0) 11927c478bd9Sstevel@tonic-gate return (EFAULT); 11937c478bd9Sstevel@tonic-gate 11947c478bd9Sstevel@tonic-gate return (0); 11957c478bd9Sstevel@tonic-gate } 11967c478bd9Sstevel@tonic-gate 11977c478bd9Sstevel@tonic-gate /* 11987c478bd9Sstevel@tonic-gate * Return a copy of the device id. 11997c478bd9Sstevel@tonic-gate */ 12007c478bd9Sstevel@tonic-gate static int 12017c478bd9Sstevel@tonic-gate modctl_get_devid(dev_t dev, uint_t len, ddi_devid_t udevid) 12027c478bd9Sstevel@tonic-gate { 12037c478bd9Sstevel@tonic-gate uint_t sz; 12047c478bd9Sstevel@tonic-gate ddi_devid_t devid; 12057c478bd9Sstevel@tonic-gate int err = 0; 12067c478bd9Sstevel@tonic-gate 12077c478bd9Sstevel@tonic-gate /* get device id */ 12087c478bd9Sstevel@tonic-gate if (ddi_lyr_get_devid(dev, &devid) == DDI_FAILURE) 12097c478bd9Sstevel@tonic-gate return (EINVAL); 12107c478bd9Sstevel@tonic-gate 12117c478bd9Sstevel@tonic-gate sz = ddi_devid_sizeof(devid); 12127c478bd9Sstevel@tonic-gate 12137c478bd9Sstevel@tonic-gate /* Error if device id is larger than space allocated */ 12147c478bd9Sstevel@tonic-gate if (sz > len) { 12157c478bd9Sstevel@tonic-gate ddi_devid_free(devid); 12167c478bd9Sstevel@tonic-gate return (ENOSPC); 12177c478bd9Sstevel@tonic-gate } 12187c478bd9Sstevel@tonic-gate 12197c478bd9Sstevel@tonic-gate /* copy out device id */ 12207c478bd9Sstevel@tonic-gate if (copyout(devid, udevid, sz) != 0) 12217c478bd9Sstevel@tonic-gate err = EFAULT; 12227c478bd9Sstevel@tonic-gate ddi_devid_free(devid); 12237c478bd9Sstevel@tonic-gate return (err); 12247c478bd9Sstevel@tonic-gate } 12257c478bd9Sstevel@tonic-gate 12267c478bd9Sstevel@tonic-gate /* 12277c478bd9Sstevel@tonic-gate * return the /devices paths associated with the specified devid and 12287c478bd9Sstevel@tonic-gate * minor name. 12297c478bd9Sstevel@tonic-gate */ 12307c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 12317c478bd9Sstevel@tonic-gate static int 12327c478bd9Sstevel@tonic-gate modctl_devid2paths(ddi_devid_t udevid, char *uminor_name, uint_t flag, 12337c478bd9Sstevel@tonic-gate size_t *ulensp, char *upaths) 12347c478bd9Sstevel@tonic-gate { 12357c478bd9Sstevel@tonic-gate ddi_devid_t devid = NULL; 12367c478bd9Sstevel@tonic-gate int devid_len; 12377c478bd9Sstevel@tonic-gate char *minor_name = NULL; 12387c478bd9Sstevel@tonic-gate dev_info_t *dip = NULL; 1239b9ccdc5aScth int circ; 12407c478bd9Sstevel@tonic-gate struct ddi_minor_data *dmdp; 12417c478bd9Sstevel@tonic-gate char *path = NULL; 12427c478bd9Sstevel@tonic-gate int ulens; 12437c478bd9Sstevel@tonic-gate int lens; 12447c478bd9Sstevel@tonic-gate int len; 12457c478bd9Sstevel@tonic-gate dev_t *devlist = NULL; 12467c478bd9Sstevel@tonic-gate int ndevs; 12477c478bd9Sstevel@tonic-gate int i; 12487c478bd9Sstevel@tonic-gate int ret = 0; 12497c478bd9Sstevel@tonic-gate 12507c478bd9Sstevel@tonic-gate /* 12517c478bd9Sstevel@tonic-gate * If upaths is NULL then we are only computing the amount of space 12527c478bd9Sstevel@tonic-gate * needed to hold the paths and returning the value in *ulensp. If we 12537c478bd9Sstevel@tonic-gate * are copying out paths then we get the amount of space allocated by 12547c478bd9Sstevel@tonic-gate * the caller. If the actual space needed for paths is larger, or 12557c478bd9Sstevel@tonic-gate * things are changing out from under us, then we return EAGAIN. 12567c478bd9Sstevel@tonic-gate */ 12577c478bd9Sstevel@tonic-gate if (upaths) { 12587c478bd9Sstevel@tonic-gate if (ulensp == NULL) 12597c478bd9Sstevel@tonic-gate return (EINVAL); 12607c478bd9Sstevel@tonic-gate if (copyin(ulensp, &ulens, sizeof (ulens)) != 0) 12617c478bd9Sstevel@tonic-gate return (EFAULT); 12627c478bd9Sstevel@tonic-gate } 12637c478bd9Sstevel@tonic-gate 12647c478bd9Sstevel@tonic-gate /* 12657c478bd9Sstevel@tonic-gate * copyin enough of the devid to determine the length then 12667c478bd9Sstevel@tonic-gate * reallocate and copy in the entire devid. 12677c478bd9Sstevel@tonic-gate */ 12687c478bd9Sstevel@tonic-gate devid_len = ddi_devid_sizeof(NULL); 12697c478bd9Sstevel@tonic-gate devid = kmem_alloc(devid_len, KM_SLEEP); 12707c478bd9Sstevel@tonic-gate if (copyin(udevid, devid, devid_len)) { 12717c478bd9Sstevel@tonic-gate ret = EFAULT; 12727c478bd9Sstevel@tonic-gate goto out; 12737c478bd9Sstevel@tonic-gate } 12747c478bd9Sstevel@tonic-gate len = devid_len; 12757c478bd9Sstevel@tonic-gate devid_len = ddi_devid_sizeof(devid); 12767c478bd9Sstevel@tonic-gate kmem_free(devid, len); 12777c478bd9Sstevel@tonic-gate devid = kmem_alloc(devid_len, KM_SLEEP); 12787c478bd9Sstevel@tonic-gate if (copyin(udevid, devid, devid_len)) { 12797c478bd9Sstevel@tonic-gate ret = EFAULT; 12807c478bd9Sstevel@tonic-gate goto out; 12817c478bd9Sstevel@tonic-gate } 12827c478bd9Sstevel@tonic-gate 12837c478bd9Sstevel@tonic-gate /* copyin the minor name if specified. */ 12847c478bd9Sstevel@tonic-gate minor_name = uminor_name; 12857c478bd9Sstevel@tonic-gate if ((minor_name != DEVID_MINOR_NAME_ALL) && 12867c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL_CHR) && 12877c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL_BLK)) { 12887c478bd9Sstevel@tonic-gate minor_name = kmem_alloc(MAXPATHLEN, KM_SLEEP); 12897c478bd9Sstevel@tonic-gate if (copyinstr(uminor_name, minor_name, MAXPATHLEN, 0)) { 12907c478bd9Sstevel@tonic-gate ret = EFAULT; 12917c478bd9Sstevel@tonic-gate goto out; 12927c478bd9Sstevel@tonic-gate } 12937c478bd9Sstevel@tonic-gate } 12947c478bd9Sstevel@tonic-gate 12957c478bd9Sstevel@tonic-gate /* 12967c478bd9Sstevel@tonic-gate * Use existing function to resolve the devid into a devlist. 12977c478bd9Sstevel@tonic-gate * 12987c478bd9Sstevel@tonic-gate * NOTE: there is a loss of spectype information in the current 12997c478bd9Sstevel@tonic-gate * ddi_lyr_devid_to_devlist implementation. We work around this by not 13007c478bd9Sstevel@tonic-gate * passing down DEVID_MINOR_NAME_ALL here, but reproducing all minor 13017c478bd9Sstevel@tonic-gate * node forms in the loop processing the devlist below. It would be 13027c478bd9Sstevel@tonic-gate * best if at some point the use of this interface here was replaced 13037c478bd9Sstevel@tonic-gate * with a path oriented call. 13047c478bd9Sstevel@tonic-gate */ 13057c478bd9Sstevel@tonic-gate if (ddi_lyr_devid_to_devlist(devid, 13067c478bd9Sstevel@tonic-gate (minor_name == DEVID_MINOR_NAME_ALL) ? 13077c478bd9Sstevel@tonic-gate DEVID_MINOR_NAME_ALL_CHR : minor_name, 13087c478bd9Sstevel@tonic-gate &ndevs, &devlist) != DDI_SUCCESS) { 13097c478bd9Sstevel@tonic-gate ret = EINVAL; 13107c478bd9Sstevel@tonic-gate goto out; 13117c478bd9Sstevel@tonic-gate } 13127c478bd9Sstevel@tonic-gate 13137c478bd9Sstevel@tonic-gate /* 13147c478bd9Sstevel@tonic-gate * loop over the devlist, converting each devt to a path and doing 13157c478bd9Sstevel@tonic-gate * a copyout of the path and computation of the amount of space 13167c478bd9Sstevel@tonic-gate * needed to hold all the paths 13177c478bd9Sstevel@tonic-gate */ 13187c478bd9Sstevel@tonic-gate path = kmem_alloc(MAXPATHLEN, KM_SLEEP); 13197c478bd9Sstevel@tonic-gate for (i = 0, lens = 0; i < ndevs; i++) { 13207c478bd9Sstevel@tonic-gate 13217c478bd9Sstevel@tonic-gate /* find the dip associated with the dev_t */ 13227c478bd9Sstevel@tonic-gate if ((dip = e_ddi_hold_devi_by_dev(devlist[i], 0)) == NULL) 13237c478bd9Sstevel@tonic-gate continue; 13247c478bd9Sstevel@tonic-gate 13257c478bd9Sstevel@tonic-gate /* loop over all the minor nodes, skipping ones we don't want */ 1326b9ccdc5aScth ndi_devi_enter(dip, &circ); 13277c478bd9Sstevel@tonic-gate for (dmdp = DEVI(dip)->devi_minor; dmdp; dmdp = dmdp->next) { 13287c478bd9Sstevel@tonic-gate if ((dmdp->ddm_dev != devlist[i]) || 13297c478bd9Sstevel@tonic-gate (dmdp->type != DDM_MINOR)) 13307c478bd9Sstevel@tonic-gate continue; 13317c478bd9Sstevel@tonic-gate 13327c478bd9Sstevel@tonic-gate if ((minor_name != DEVID_MINOR_NAME_ALL) && 13337c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL_CHR) && 13347c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL_BLK) && 13357c478bd9Sstevel@tonic-gate strcmp(minor_name, dmdp->ddm_name)) 13367c478bd9Sstevel@tonic-gate continue; 13377c478bd9Sstevel@tonic-gate else { 13387c478bd9Sstevel@tonic-gate if ((minor_name == DEVID_MINOR_NAME_ALL_CHR) && 13397c478bd9Sstevel@tonic-gate (dmdp->ddm_spec_type != S_IFCHR)) 13407c478bd9Sstevel@tonic-gate continue; 13417c478bd9Sstevel@tonic-gate if ((minor_name == DEVID_MINOR_NAME_ALL_BLK) && 13427c478bd9Sstevel@tonic-gate (dmdp->ddm_spec_type != S_IFBLK)) 13437c478bd9Sstevel@tonic-gate continue; 13447c478bd9Sstevel@tonic-gate } 13457c478bd9Sstevel@tonic-gate 1346f9722deaSChris Horne (void) ddi_pathname_minor(dmdp, path); 13477c478bd9Sstevel@tonic-gate len = strlen(path) + 1; 13487c478bd9Sstevel@tonic-gate *(path + len) = '\0'; /* set double termination */ 13497c478bd9Sstevel@tonic-gate lens += len; 13507c478bd9Sstevel@tonic-gate 13517c478bd9Sstevel@tonic-gate /* copyout the path with double terminations */ 13527c478bd9Sstevel@tonic-gate if (upaths) { 13537c478bd9Sstevel@tonic-gate if (lens > ulens) { 13547c478bd9Sstevel@tonic-gate ret = EAGAIN; 13557c478bd9Sstevel@tonic-gate goto out; 13567c478bd9Sstevel@tonic-gate } 13577c478bd9Sstevel@tonic-gate if (copyout(path, upaths, len + 1)) { 13587c478bd9Sstevel@tonic-gate ret = EFAULT; 13597c478bd9Sstevel@tonic-gate goto out; 13607c478bd9Sstevel@tonic-gate } 13617c478bd9Sstevel@tonic-gate upaths += len; 13627c478bd9Sstevel@tonic-gate } 13637c478bd9Sstevel@tonic-gate } 1364b9ccdc5aScth ndi_devi_exit(dip, circ); 13657c478bd9Sstevel@tonic-gate ddi_release_devi(dip); 13667c478bd9Sstevel@tonic-gate dip = NULL; 13677c478bd9Sstevel@tonic-gate } 13687c478bd9Sstevel@tonic-gate lens++; /* add one for double termination */ 13697c478bd9Sstevel@tonic-gate 13707c478bd9Sstevel@tonic-gate /* copy out the amount of space needed to hold the paths */ 13717c478bd9Sstevel@tonic-gate if (ulensp && copyout(&lens, ulensp, sizeof (lens))) { 13727c478bd9Sstevel@tonic-gate ret = EFAULT; 13737c478bd9Sstevel@tonic-gate goto out; 13747c478bd9Sstevel@tonic-gate } 13757c478bd9Sstevel@tonic-gate ret = 0; 13767c478bd9Sstevel@tonic-gate 1377b9ccdc5aScth out: if (dip) { 1378b9ccdc5aScth ndi_devi_exit(dip, circ); 13797c478bd9Sstevel@tonic-gate ddi_release_devi(dip); 1380b9ccdc5aScth } 13817c478bd9Sstevel@tonic-gate if (path) 13827c478bd9Sstevel@tonic-gate kmem_free(path, MAXPATHLEN); 13837c478bd9Sstevel@tonic-gate if (devlist) 13847c478bd9Sstevel@tonic-gate ddi_lyr_free_devlist(devlist, ndevs); 13857c478bd9Sstevel@tonic-gate if (minor_name && 13867c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL) && 13877c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL_CHR) && 13887c478bd9Sstevel@tonic-gate (minor_name != DEVID_MINOR_NAME_ALL_BLK)) 13897c478bd9Sstevel@tonic-gate kmem_free(minor_name, MAXPATHLEN); 13907c478bd9Sstevel@tonic-gate if (devid) 13917c478bd9Sstevel@tonic-gate kmem_free(devid, devid_len); 13927c478bd9Sstevel@tonic-gate return (ret); 13937c478bd9Sstevel@tonic-gate } 13947c478bd9Sstevel@tonic-gate 13957c478bd9Sstevel@tonic-gate /* 13967c478bd9Sstevel@tonic-gate * Return the size of the minor name. 13977c478bd9Sstevel@tonic-gate */ 13987c478bd9Sstevel@tonic-gate static int 13997c478bd9Sstevel@tonic-gate modctl_sizeof_minorname(dev_t dev, int spectype, uint_t *len) 14007c478bd9Sstevel@tonic-gate { 14017c478bd9Sstevel@tonic-gate uint_t sz; 14027c478bd9Sstevel@tonic-gate char *name; 14037c478bd9Sstevel@tonic-gate 14047c478bd9Sstevel@tonic-gate /* get the minor name */ 14057c478bd9Sstevel@tonic-gate if (ddi_lyr_get_minor_name(dev, spectype, &name) == DDI_FAILURE) 14067c478bd9Sstevel@tonic-gate return (EINVAL); 14077c478bd9Sstevel@tonic-gate 14087c478bd9Sstevel@tonic-gate sz = strlen(name) + 1; 14097c478bd9Sstevel@tonic-gate kmem_free(name, sz); 14107c478bd9Sstevel@tonic-gate 14117c478bd9Sstevel@tonic-gate /* copy out the size of the minor name */ 14127c478bd9Sstevel@tonic-gate if (copyout(&sz, len, sizeof (sz)) != 0) 14137c478bd9Sstevel@tonic-gate return (EFAULT); 14147c478bd9Sstevel@tonic-gate 14157c478bd9Sstevel@tonic-gate return (0); 14167c478bd9Sstevel@tonic-gate } 14177c478bd9Sstevel@tonic-gate 14187c478bd9Sstevel@tonic-gate /* 14197c478bd9Sstevel@tonic-gate * Return the minor name. 14207c478bd9Sstevel@tonic-gate */ 14217c478bd9Sstevel@tonic-gate static int 14227c478bd9Sstevel@tonic-gate modctl_get_minorname(dev_t dev, int spectype, uint_t len, char *uname) 14237c478bd9Sstevel@tonic-gate { 14247c478bd9Sstevel@tonic-gate uint_t sz; 14257c478bd9Sstevel@tonic-gate char *name; 14267c478bd9Sstevel@tonic-gate int err = 0; 14277c478bd9Sstevel@tonic-gate 14287c478bd9Sstevel@tonic-gate /* get the minor name */ 14297c478bd9Sstevel@tonic-gate if (ddi_lyr_get_minor_name(dev, spectype, &name) == DDI_FAILURE) 14307c478bd9Sstevel@tonic-gate return (EINVAL); 14317c478bd9Sstevel@tonic-gate 14327c478bd9Sstevel@tonic-gate sz = strlen(name) + 1; 14337c478bd9Sstevel@tonic-gate 14347c478bd9Sstevel@tonic-gate /* Error if the minor name is larger than the space allocated */ 14357c478bd9Sstevel@tonic-gate if (sz > len) { 14367c478bd9Sstevel@tonic-gate kmem_free(name, sz); 14377c478bd9Sstevel@tonic-gate return (ENOSPC); 14387c478bd9Sstevel@tonic-gate } 14397c478bd9Sstevel@tonic-gate 14407c478bd9Sstevel@tonic-gate /* copy out the minor name */ 14417c478bd9Sstevel@tonic-gate if (copyout(name, uname, sz) != 0) 14427c478bd9Sstevel@tonic-gate err = EFAULT; 14437c478bd9Sstevel@tonic-gate kmem_free(name, sz); 14447c478bd9Sstevel@tonic-gate return (err); 14457c478bd9Sstevel@tonic-gate } 14467c478bd9Sstevel@tonic-gate 14477c478bd9Sstevel@tonic-gate /* 1448a08731ecScth * Return the size of the (dev_t,spectype) devfspath name. 14497c478bd9Sstevel@tonic-gate */ 14507c478bd9Sstevel@tonic-gate static int 14517c478bd9Sstevel@tonic-gate modctl_devfspath_len(dev_t dev, int spectype, uint_t *len) 14527c478bd9Sstevel@tonic-gate { 14537c478bd9Sstevel@tonic-gate uint_t sz; 14547c478bd9Sstevel@tonic-gate char *name; 14557c478bd9Sstevel@tonic-gate 14567c478bd9Sstevel@tonic-gate /* get the path name */ 14577c478bd9Sstevel@tonic-gate name = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 14587c478bd9Sstevel@tonic-gate if (ddi_dev_pathname(dev, spectype, name) == DDI_FAILURE) { 14597c478bd9Sstevel@tonic-gate kmem_free(name, MAXPATHLEN); 14607c478bd9Sstevel@tonic-gate return (EINVAL); 14617c478bd9Sstevel@tonic-gate } 14627c478bd9Sstevel@tonic-gate 14637c478bd9Sstevel@tonic-gate sz = strlen(name) + 1; 14647c478bd9Sstevel@tonic-gate kmem_free(name, MAXPATHLEN); 14657c478bd9Sstevel@tonic-gate 14667c478bd9Sstevel@tonic-gate /* copy out the size of the path name */ 14677c478bd9Sstevel@tonic-gate if (copyout(&sz, len, sizeof (sz)) != 0) 14687c478bd9Sstevel@tonic-gate return (EFAULT); 14697c478bd9Sstevel@tonic-gate 14707c478bd9Sstevel@tonic-gate return (0); 14717c478bd9Sstevel@tonic-gate } 14727c478bd9Sstevel@tonic-gate 14737c478bd9Sstevel@tonic-gate /* 1474a08731ecScth * Return the (dev_t,spectype) devfspath name. 14757c478bd9Sstevel@tonic-gate */ 14767c478bd9Sstevel@tonic-gate static int 14777c478bd9Sstevel@tonic-gate modctl_devfspath(dev_t dev, int spectype, uint_t len, char *uname) 14787c478bd9Sstevel@tonic-gate { 14797c478bd9Sstevel@tonic-gate uint_t sz; 14807c478bd9Sstevel@tonic-gate char *name; 14817c478bd9Sstevel@tonic-gate int err = 0; 14827c478bd9Sstevel@tonic-gate 14837c478bd9Sstevel@tonic-gate /* get the path name */ 14847c478bd9Sstevel@tonic-gate name = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 14857c478bd9Sstevel@tonic-gate if (ddi_dev_pathname(dev, spectype, name) == DDI_FAILURE) { 14867c478bd9Sstevel@tonic-gate kmem_free(name, MAXPATHLEN); 14877c478bd9Sstevel@tonic-gate return (EINVAL); 14887c478bd9Sstevel@tonic-gate } 14897c478bd9Sstevel@tonic-gate 14907c478bd9Sstevel@tonic-gate sz = strlen(name) + 1; 14917c478bd9Sstevel@tonic-gate 14927c478bd9Sstevel@tonic-gate /* Error if the path name is larger than the space allocated */ 14937c478bd9Sstevel@tonic-gate if (sz > len) { 14947c478bd9Sstevel@tonic-gate kmem_free(name, MAXPATHLEN); 14957c478bd9Sstevel@tonic-gate return (ENOSPC); 14967c478bd9Sstevel@tonic-gate } 14977c478bd9Sstevel@tonic-gate 14987c478bd9Sstevel@tonic-gate /* copy out the path name */ 14997c478bd9Sstevel@tonic-gate if (copyout(name, uname, sz) != 0) 15007c478bd9Sstevel@tonic-gate err = EFAULT; 15017c478bd9Sstevel@tonic-gate kmem_free(name, MAXPATHLEN); 15027c478bd9Sstevel@tonic-gate return (err); 15037c478bd9Sstevel@tonic-gate } 15047c478bd9Sstevel@tonic-gate 1505a08731ecScth /* 1506a08731ecScth * Return the size of the (major,instance) devfspath name. 1507a08731ecScth */ 1508a08731ecScth static int 1509a08731ecScth modctl_devfspath_mi_len(major_t major, int instance, uint_t *len) 1510a08731ecScth { 1511a08731ecScth uint_t sz; 1512a08731ecScth char *name; 1513a08731ecScth 1514a08731ecScth /* get the path name */ 1515a08731ecScth name = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 1516a08731ecScth if (e_ddi_majorinstance_to_path(major, instance, name) != DDI_SUCCESS) { 1517a08731ecScth kmem_free(name, MAXPATHLEN); 1518a08731ecScth return (EINVAL); 1519a08731ecScth } 1520a08731ecScth 1521a08731ecScth sz = strlen(name) + 1; 1522a08731ecScth kmem_free(name, MAXPATHLEN); 1523a08731ecScth 1524a08731ecScth /* copy out the size of the path name */ 1525a08731ecScth if (copyout(&sz, len, sizeof (sz)) != 0) 1526a08731ecScth return (EFAULT); 1527a08731ecScth 1528a08731ecScth return (0); 1529a08731ecScth } 1530a08731ecScth 1531a08731ecScth /* 1532a08731ecScth * Return the (major_instance) devfspath name. 1533a08731ecScth * NOTE: e_ddi_majorinstance_to_path does not require the device to attach to 1534a08731ecScth * return a path - it uses the instance tree. 1535a08731ecScth */ 1536a08731ecScth static int 1537a08731ecScth modctl_devfspath_mi(major_t major, int instance, uint_t len, char *uname) 1538a08731ecScth { 1539a08731ecScth uint_t sz; 1540a08731ecScth char *name; 1541a08731ecScth int err = 0; 1542a08731ecScth 1543a08731ecScth /* get the path name */ 1544a08731ecScth name = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 1545a08731ecScth if (e_ddi_majorinstance_to_path(major, instance, name) != DDI_SUCCESS) { 1546a08731ecScth kmem_free(name, MAXPATHLEN); 1547a08731ecScth return (EINVAL); 1548a08731ecScth } 1549a08731ecScth 1550a08731ecScth sz = strlen(name) + 1; 1551a08731ecScth 1552a08731ecScth /* Error if the path name is larger than the space allocated */ 1553a08731ecScth if (sz > len) { 1554a08731ecScth kmem_free(name, MAXPATHLEN); 1555a08731ecScth return (ENOSPC); 1556a08731ecScth } 1557a08731ecScth 1558a08731ecScth /* copy out the path name */ 1559a08731ecScth if (copyout(name, uname, sz) != 0) 1560a08731ecScth err = EFAULT; 1561a08731ecScth kmem_free(name, MAXPATHLEN); 1562a08731ecScth return (err); 1563a08731ecScth } 1564a08731ecScth 15657c478bd9Sstevel@tonic-gate static int 15667c478bd9Sstevel@tonic-gate modctl_get_fbname(char *path) 15677c478bd9Sstevel@tonic-gate { 15687c478bd9Sstevel@tonic-gate extern dev_t fbdev; 15697c478bd9Sstevel@tonic-gate char *pathname = NULL; 15707c478bd9Sstevel@tonic-gate int rval = 0; 15717c478bd9Sstevel@tonic-gate 15727c478bd9Sstevel@tonic-gate /* make sure fbdev is set before we plunge in */ 15737c478bd9Sstevel@tonic-gate if (fbdev == NODEV) 15747c478bd9Sstevel@tonic-gate return (ENODEV); 15757c478bd9Sstevel@tonic-gate 15767c478bd9Sstevel@tonic-gate pathname = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 15777c478bd9Sstevel@tonic-gate if ((rval = ddi_dev_pathname(fbdev, S_IFCHR, 15787c478bd9Sstevel@tonic-gate pathname)) == DDI_SUCCESS) { 15797c478bd9Sstevel@tonic-gate if (copyout(pathname, path, strlen(pathname)+1) != 0) { 15807c478bd9Sstevel@tonic-gate rval = EFAULT; 15817c478bd9Sstevel@tonic-gate } 15827c478bd9Sstevel@tonic-gate } 15837c478bd9Sstevel@tonic-gate kmem_free(pathname, MAXPATHLEN); 15847c478bd9Sstevel@tonic-gate return (rval); 15857c478bd9Sstevel@tonic-gate } 15867c478bd9Sstevel@tonic-gate 15877c478bd9Sstevel@tonic-gate /* 15887c478bd9Sstevel@tonic-gate * modctl_reread_dacf() 15897c478bd9Sstevel@tonic-gate * Reread the dacf rules database from the named binding file. 15907c478bd9Sstevel@tonic-gate * If NULL is specified, pass along the NULL, it means 'use the default'. 15917c478bd9Sstevel@tonic-gate */ 15927c478bd9Sstevel@tonic-gate static int 15937c478bd9Sstevel@tonic-gate modctl_reread_dacf(char *path) 15947c478bd9Sstevel@tonic-gate { 15957c478bd9Sstevel@tonic-gate int rval = 0; 15967c478bd9Sstevel@tonic-gate char *filename, *filenamep; 15977c478bd9Sstevel@tonic-gate 15987c478bd9Sstevel@tonic-gate filename = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 15997c478bd9Sstevel@tonic-gate 16007c478bd9Sstevel@tonic-gate if (path == NULL) { 16017c478bd9Sstevel@tonic-gate filenamep = NULL; 16027c478bd9Sstevel@tonic-gate } else { 16037c478bd9Sstevel@tonic-gate if (copyinstr(path, filename, MAXPATHLEN, 0) != 0) { 16047c478bd9Sstevel@tonic-gate rval = EFAULT; 16057c478bd9Sstevel@tonic-gate goto out; 16067c478bd9Sstevel@tonic-gate } 16077c478bd9Sstevel@tonic-gate filenamep = filename; 16087c478bd9Sstevel@tonic-gate filenamep[MAXPATHLEN - 1] = '\0'; 16097c478bd9Sstevel@tonic-gate } 16107c478bd9Sstevel@tonic-gate 16117c478bd9Sstevel@tonic-gate rval = read_dacf_binding_file(filenamep); 16127c478bd9Sstevel@tonic-gate out: 16137c478bd9Sstevel@tonic-gate kmem_free(filename, MAXPATHLEN); 16147c478bd9Sstevel@tonic-gate return (rval); 16157c478bd9Sstevel@tonic-gate } 16167c478bd9Sstevel@tonic-gate 16177c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 16187c478bd9Sstevel@tonic-gate static int 16197c478bd9Sstevel@tonic-gate modctl_modevents(int subcmd, uintptr_t a2, uintptr_t a3, uintptr_t a4, 16207c478bd9Sstevel@tonic-gate uint_t flag) 16217c478bd9Sstevel@tonic-gate { 16227c478bd9Sstevel@tonic-gate int error = 0; 16237c478bd9Sstevel@tonic-gate char *filenamep; 16247c478bd9Sstevel@tonic-gate 16257c478bd9Sstevel@tonic-gate switch (subcmd) { 16267c478bd9Sstevel@tonic-gate 16277c478bd9Sstevel@tonic-gate case MODEVENTS_FLUSH: 16287c478bd9Sstevel@tonic-gate /* flush all currently queued events */ 16297c478bd9Sstevel@tonic-gate log_sysevent_flushq(subcmd, flag); 16307c478bd9Sstevel@tonic-gate break; 16317c478bd9Sstevel@tonic-gate 16327c478bd9Sstevel@tonic-gate case MODEVENTS_SET_DOOR_UPCALL_FILENAME: 16337c478bd9Sstevel@tonic-gate /* 16347c478bd9Sstevel@tonic-gate * bind door_upcall to filename 16357c478bd9Sstevel@tonic-gate * this should only be done once per invocation 16367c478bd9Sstevel@tonic-gate * of the event daemon. 16377c478bd9Sstevel@tonic-gate */ 16387c478bd9Sstevel@tonic-gate 16397c478bd9Sstevel@tonic-gate filenamep = kmem_zalloc(MOD_MAXPATH, KM_SLEEP); 16407c478bd9Sstevel@tonic-gate 16417c478bd9Sstevel@tonic-gate if (copyinstr((char *)a2, filenamep, MOD_MAXPATH, 0)) { 16427c478bd9Sstevel@tonic-gate error = EFAULT; 16437c478bd9Sstevel@tonic-gate } else { 16447c478bd9Sstevel@tonic-gate error = log_sysevent_filename(filenamep); 16457c478bd9Sstevel@tonic-gate } 16467c478bd9Sstevel@tonic-gate kmem_free(filenamep, MOD_MAXPATH); 16477c478bd9Sstevel@tonic-gate break; 16487c478bd9Sstevel@tonic-gate 16497c478bd9Sstevel@tonic-gate case MODEVENTS_GETDATA: 16507c478bd9Sstevel@tonic-gate error = log_sysevent_copyout_data((sysevent_id_t *)a2, 16517c478bd9Sstevel@tonic-gate (size_t)a3, (caddr_t)a4); 16527c478bd9Sstevel@tonic-gate break; 16537c478bd9Sstevel@tonic-gate 16547c478bd9Sstevel@tonic-gate case MODEVENTS_FREEDATA: 16557c478bd9Sstevel@tonic-gate error = log_sysevent_free_data((sysevent_id_t *)a2); 16567c478bd9Sstevel@tonic-gate break; 16577c478bd9Sstevel@tonic-gate case MODEVENTS_POST_EVENT: 16587c478bd9Sstevel@tonic-gate error = log_usr_sysevent((sysevent_t *)a2, (uint32_t)a3, 16597c478bd9Sstevel@tonic-gate (sysevent_id_t *)a4); 16607c478bd9Sstevel@tonic-gate break; 16617c478bd9Sstevel@tonic-gate case MODEVENTS_REGISTER_EVENT: 16627c478bd9Sstevel@tonic-gate error = log_sysevent_register((char *)a2, (char *)a3, 16637c478bd9Sstevel@tonic-gate (se_pubsub_t *)a4); 16647c478bd9Sstevel@tonic-gate break; 16657c478bd9Sstevel@tonic-gate default: 16667c478bd9Sstevel@tonic-gate error = EINVAL; 16677c478bd9Sstevel@tonic-gate } 16687c478bd9Sstevel@tonic-gate 16697c478bd9Sstevel@tonic-gate return (error); 16707c478bd9Sstevel@tonic-gate } 16717c478bd9Sstevel@tonic-gate 16727c478bd9Sstevel@tonic-gate static void 16737c478bd9Sstevel@tonic-gate free_mperm(mperm_t *mp) 16747c478bd9Sstevel@tonic-gate { 16757c478bd9Sstevel@tonic-gate int len; 16767c478bd9Sstevel@tonic-gate 16777c478bd9Sstevel@tonic-gate if (mp->mp_minorname) { 16787c478bd9Sstevel@tonic-gate len = strlen(mp->mp_minorname) + 1; 16797c478bd9Sstevel@tonic-gate kmem_free(mp->mp_minorname, len); 16807c478bd9Sstevel@tonic-gate } 16817c478bd9Sstevel@tonic-gate kmem_free(mp, sizeof (mperm_t)); 16827c478bd9Sstevel@tonic-gate } 16837c478bd9Sstevel@tonic-gate 16847c478bd9Sstevel@tonic-gate #define MP_NO_DRV_ERR \ 16857c478bd9Sstevel@tonic-gate "/etc/minor_perm: no driver for %s\n" 16867c478bd9Sstevel@tonic-gate 16877c478bd9Sstevel@tonic-gate #define MP_EMPTY_MINOR \ 16887c478bd9Sstevel@tonic-gate "/etc/minor_perm: empty minor name for driver %s\n" 16897c478bd9Sstevel@tonic-gate 16907c478bd9Sstevel@tonic-gate #define MP_NO_MINOR \ 16917c478bd9Sstevel@tonic-gate "/etc/minor_perm: no minor matching %s for driver %s\n" 16927c478bd9Sstevel@tonic-gate 16937c478bd9Sstevel@tonic-gate /* 16947c478bd9Sstevel@tonic-gate * Remove mperm entry with matching minorname 16957c478bd9Sstevel@tonic-gate */ 16967c478bd9Sstevel@tonic-gate static void 16977c478bd9Sstevel@tonic-gate rem_minorperm(major_t major, char *drvname, mperm_t *mp, int is_clone) 16987c478bd9Sstevel@tonic-gate { 16997c478bd9Sstevel@tonic-gate mperm_t **mp_head; 17007c478bd9Sstevel@tonic-gate mperm_t *freemp = NULL; 17017c478bd9Sstevel@tonic-gate struct devnames *dnp = &devnamesp[major]; 17027c478bd9Sstevel@tonic-gate mperm_t **wildmp; 17037c478bd9Sstevel@tonic-gate 17047c478bd9Sstevel@tonic-gate ASSERT(mp->mp_minorname && strlen(mp->mp_minorname) > 0); 17057c478bd9Sstevel@tonic-gate 17067c478bd9Sstevel@tonic-gate LOCK_DEV_OPS(&dnp->dn_lock); 17077c478bd9Sstevel@tonic-gate if (strcmp(mp->mp_minorname, "*") == 0) { 17087c478bd9Sstevel@tonic-gate wildmp = ((is_clone == 0) ? 17097c478bd9Sstevel@tonic-gate &dnp->dn_mperm_wild : &dnp->dn_mperm_clone); 17107c478bd9Sstevel@tonic-gate if (*wildmp) 17117c478bd9Sstevel@tonic-gate freemp = *wildmp; 17127c478bd9Sstevel@tonic-gate *wildmp = NULL; 17137c478bd9Sstevel@tonic-gate } else { 17147c478bd9Sstevel@tonic-gate mp_head = &dnp->dn_mperm; 17157c478bd9Sstevel@tonic-gate while (*mp_head) { 17167c478bd9Sstevel@tonic-gate if (strcmp((*mp_head)->mp_minorname, 17177c478bd9Sstevel@tonic-gate mp->mp_minorname) != 0) { 17187c478bd9Sstevel@tonic-gate mp_head = &(*mp_head)->mp_next; 17197c478bd9Sstevel@tonic-gate continue; 17207c478bd9Sstevel@tonic-gate } 17217c478bd9Sstevel@tonic-gate /* remove the entry */ 17227c478bd9Sstevel@tonic-gate freemp = *mp_head; 17237c478bd9Sstevel@tonic-gate *mp_head = freemp->mp_next; 17247c478bd9Sstevel@tonic-gate break; 17257c478bd9Sstevel@tonic-gate } 17267c478bd9Sstevel@tonic-gate } 17277c478bd9Sstevel@tonic-gate if (freemp) { 17287c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 17297c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, "< %s %s 0%o %d %d\n", 17307c478bd9Sstevel@tonic-gate drvname, freemp->mp_minorname, 17317c478bd9Sstevel@tonic-gate freemp->mp_mode & 0777, 17327c478bd9Sstevel@tonic-gate freemp->mp_uid, freemp->mp_gid); 17337c478bd9Sstevel@tonic-gate } 17347c478bd9Sstevel@tonic-gate free_mperm(freemp); 17357c478bd9Sstevel@tonic-gate } else { 17367c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 17377c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, MP_NO_MINOR, 17387c478bd9Sstevel@tonic-gate drvname, mp->mp_minorname); 17397c478bd9Sstevel@tonic-gate } 17407c478bd9Sstevel@tonic-gate } 17417c478bd9Sstevel@tonic-gate 17427c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 17437c478bd9Sstevel@tonic-gate } 17447c478bd9Sstevel@tonic-gate 17457c478bd9Sstevel@tonic-gate /* 17467c478bd9Sstevel@tonic-gate * Add minor perm entry 17477c478bd9Sstevel@tonic-gate */ 17487c478bd9Sstevel@tonic-gate static void 17497c478bd9Sstevel@tonic-gate add_minorperm(major_t major, char *drvname, mperm_t *mp, int is_clone) 17507c478bd9Sstevel@tonic-gate { 17517c478bd9Sstevel@tonic-gate mperm_t **mp_head; 17527c478bd9Sstevel@tonic-gate mperm_t *freemp = NULL; 17537c478bd9Sstevel@tonic-gate struct devnames *dnp = &devnamesp[major]; 17547c478bd9Sstevel@tonic-gate mperm_t **wildmp; 17557c478bd9Sstevel@tonic-gate 17567c478bd9Sstevel@tonic-gate ASSERT(mp->mp_minorname && strlen(mp->mp_minorname) > 0); 17577c478bd9Sstevel@tonic-gate 17587c478bd9Sstevel@tonic-gate /* 17597c478bd9Sstevel@tonic-gate * Note that update_drv replace semantics require 17607c478bd9Sstevel@tonic-gate * replacing matching entries with the new permissions. 17617c478bd9Sstevel@tonic-gate */ 17627c478bd9Sstevel@tonic-gate LOCK_DEV_OPS(&dnp->dn_lock); 17637c478bd9Sstevel@tonic-gate if (strcmp(mp->mp_minorname, "*") == 0) { 17647c478bd9Sstevel@tonic-gate wildmp = ((is_clone == 0) ? 17657c478bd9Sstevel@tonic-gate &dnp->dn_mperm_wild : &dnp->dn_mperm_clone); 17667c478bd9Sstevel@tonic-gate if (*wildmp) 17677c478bd9Sstevel@tonic-gate freemp = *wildmp; 17687c478bd9Sstevel@tonic-gate *wildmp = mp; 17697c478bd9Sstevel@tonic-gate } else { 17707c478bd9Sstevel@tonic-gate mperm_t *p, *v = NULL; 17717c478bd9Sstevel@tonic-gate for (p = dnp->dn_mperm; p; v = p, p = p->mp_next) { 17727c478bd9Sstevel@tonic-gate if (strcmp(p->mp_minorname, mp->mp_minorname) == 0) { 17737c478bd9Sstevel@tonic-gate if (v == NULL) 17747c478bd9Sstevel@tonic-gate dnp->dn_mperm = mp; 17757c478bd9Sstevel@tonic-gate else 17767c478bd9Sstevel@tonic-gate v->mp_next = mp; 17777c478bd9Sstevel@tonic-gate mp->mp_next = p->mp_next; 17787c478bd9Sstevel@tonic-gate freemp = p; 17797c478bd9Sstevel@tonic-gate goto replaced; 17807c478bd9Sstevel@tonic-gate } 17817c478bd9Sstevel@tonic-gate } 17827c478bd9Sstevel@tonic-gate if (p == NULL) { 17837c478bd9Sstevel@tonic-gate mp_head = &dnp->dn_mperm; 17847c478bd9Sstevel@tonic-gate if (*mp_head == NULL) { 17857c478bd9Sstevel@tonic-gate *mp_head = mp; 17867c478bd9Sstevel@tonic-gate } else { 17877c478bd9Sstevel@tonic-gate mp->mp_next = *mp_head; 17887c478bd9Sstevel@tonic-gate *mp_head = mp; 17897c478bd9Sstevel@tonic-gate } 17907c478bd9Sstevel@tonic-gate } 17917c478bd9Sstevel@tonic-gate } 17927c478bd9Sstevel@tonic-gate replaced: 17937c478bd9Sstevel@tonic-gate if (freemp) { 17947c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 17957c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, "< %s %s 0%o %d %d\n", 17967c478bd9Sstevel@tonic-gate drvname, freemp->mp_minorname, 17977c478bd9Sstevel@tonic-gate freemp->mp_mode & 0777, 17987c478bd9Sstevel@tonic-gate freemp->mp_uid, freemp->mp_gid); 17997c478bd9Sstevel@tonic-gate } 18007c478bd9Sstevel@tonic-gate free_mperm(freemp); 18017c478bd9Sstevel@tonic-gate } 18027c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 18037c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, "> %s %s 0%o %d %d\n", 18047c478bd9Sstevel@tonic-gate drvname, mp->mp_minorname, mp->mp_mode & 0777, 18057c478bd9Sstevel@tonic-gate mp->mp_uid, mp->mp_gid); 18067c478bd9Sstevel@tonic-gate } 18077c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 18087c478bd9Sstevel@tonic-gate } 18097c478bd9Sstevel@tonic-gate 18107c478bd9Sstevel@tonic-gate 18117c478bd9Sstevel@tonic-gate static int 18127c478bd9Sstevel@tonic-gate process_minorperm(int cmd, nvlist_t *nvl) 18137c478bd9Sstevel@tonic-gate { 18147c478bd9Sstevel@tonic-gate char *minor; 18157c478bd9Sstevel@tonic-gate major_t major; 18167c478bd9Sstevel@tonic-gate mperm_t *mp; 18177c478bd9Sstevel@tonic-gate nvpair_t *nvp; 18187c478bd9Sstevel@tonic-gate char *name; 18197c478bd9Sstevel@tonic-gate int is_clone; 18207c478bd9Sstevel@tonic-gate major_t minmaj; 18217c478bd9Sstevel@tonic-gate 18227c478bd9Sstevel@tonic-gate ASSERT(cmd == MODLOADMINORPERM || 18237c478bd9Sstevel@tonic-gate cmd == MODADDMINORPERM || cmd == MODREMMINORPERM); 18247c478bd9Sstevel@tonic-gate 18257c478bd9Sstevel@tonic-gate nvp = NULL; 18267c478bd9Sstevel@tonic-gate while ((nvp = nvlist_next_nvpair(nvl, nvp)) != NULL) { 18277c478bd9Sstevel@tonic-gate name = nvpair_name(nvp); 18287c478bd9Sstevel@tonic-gate 18297c478bd9Sstevel@tonic-gate is_clone = 0; 18307c478bd9Sstevel@tonic-gate (void) nvpair_value_string(nvp, &minor); 18317c478bd9Sstevel@tonic-gate major = ddi_name_to_major(name); 1832a204de77Scth if (major != DDI_MAJOR_T_NONE) { 18337c478bd9Sstevel@tonic-gate mp = kmem_zalloc(sizeof (*mp), KM_SLEEP); 18347c478bd9Sstevel@tonic-gate if (minor == NULL || strlen(minor) == 0) { 18357c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 18367c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, MP_EMPTY_MINOR, name); 18377c478bd9Sstevel@tonic-gate } 18387c478bd9Sstevel@tonic-gate minor = "*"; 18397c478bd9Sstevel@tonic-gate } 18407c478bd9Sstevel@tonic-gate 18417c478bd9Sstevel@tonic-gate /* 18427c478bd9Sstevel@tonic-gate * The minor name of a node using the clone 18437c478bd9Sstevel@tonic-gate * driver must be the driver name. To avoid 18447c478bd9Sstevel@tonic-gate * multiple searches, we map entries in the form 18457c478bd9Sstevel@tonic-gate * clone:<driver> to <driver>:*. This also allows us 18467c478bd9Sstevel@tonic-gate * to filter out some of the litter in /etc/minor_perm. 18477c478bd9Sstevel@tonic-gate * Minor perm alias entries where the name is not 18487c478bd9Sstevel@tonic-gate * the driver kept on the clone list itself. 18497c478bd9Sstevel@tonic-gate * This all seems very fragile as a driver could 18507c478bd9Sstevel@tonic-gate * be introduced with an existing alias name. 18517c478bd9Sstevel@tonic-gate */ 18527c478bd9Sstevel@tonic-gate if (strcmp(name, "clone") == 0) { 18537c478bd9Sstevel@tonic-gate minmaj = ddi_name_to_major(minor); 1854a204de77Scth if (minmaj != DDI_MAJOR_T_NONE) { 18557c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 18567c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, 18577c478bd9Sstevel@tonic-gate "mapping %s:%s to %s:*\n", 18587c478bd9Sstevel@tonic-gate name, minor, minor); 18597c478bd9Sstevel@tonic-gate } 18607c478bd9Sstevel@tonic-gate major = minmaj; 18617c478bd9Sstevel@tonic-gate name = minor; 18627c478bd9Sstevel@tonic-gate minor = "*"; 18637c478bd9Sstevel@tonic-gate is_clone = 1; 18647c478bd9Sstevel@tonic-gate } 18657c478bd9Sstevel@tonic-gate } 18667c478bd9Sstevel@tonic-gate 18677c478bd9Sstevel@tonic-gate if (mp) { 18687c478bd9Sstevel@tonic-gate mp->mp_minorname = 18697c478bd9Sstevel@tonic-gate i_ddi_strdup(minor, KM_SLEEP); 18707c478bd9Sstevel@tonic-gate } 18717c478bd9Sstevel@tonic-gate } else { 18727c478bd9Sstevel@tonic-gate mp = NULL; 18737c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) { 18747c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, MP_NO_DRV_ERR, name); 18757c478bd9Sstevel@tonic-gate } 18767c478bd9Sstevel@tonic-gate } 18777c478bd9Sstevel@tonic-gate 18787c478bd9Sstevel@tonic-gate /* mode */ 18797c478bd9Sstevel@tonic-gate nvp = nvlist_next_nvpair(nvl, nvp); 18807c478bd9Sstevel@tonic-gate ASSERT(strcmp(nvpair_name(nvp), "mode") == 0); 18817c478bd9Sstevel@tonic-gate if (mp) 18827c478bd9Sstevel@tonic-gate (void) nvpair_value_int32(nvp, (int *)&mp->mp_mode); 18837c478bd9Sstevel@tonic-gate /* uid */ 18847c478bd9Sstevel@tonic-gate nvp = nvlist_next_nvpair(nvl, nvp); 18857c478bd9Sstevel@tonic-gate ASSERT(strcmp(nvpair_name(nvp), "uid") == 0); 18867c478bd9Sstevel@tonic-gate if (mp) 1887f48205beScasper (void) nvpair_value_uint32(nvp, &mp->mp_uid); 18887c478bd9Sstevel@tonic-gate /* gid */ 18897c478bd9Sstevel@tonic-gate nvp = nvlist_next_nvpair(nvl, nvp); 18907c478bd9Sstevel@tonic-gate ASSERT(strcmp(nvpair_name(nvp), "gid") == 0); 18917c478bd9Sstevel@tonic-gate if (mp) { 1892f48205beScasper (void) nvpair_value_uint32(nvp, &mp->mp_gid); 18937c478bd9Sstevel@tonic-gate 18947c478bd9Sstevel@tonic-gate if (cmd == MODREMMINORPERM) { 18957c478bd9Sstevel@tonic-gate rem_minorperm(major, name, mp, is_clone); 18967c478bd9Sstevel@tonic-gate free_mperm(mp); 18977c478bd9Sstevel@tonic-gate } else { 18987c478bd9Sstevel@tonic-gate add_minorperm(major, name, mp, is_clone); 18997c478bd9Sstevel@tonic-gate } 19007c478bd9Sstevel@tonic-gate } 19017c478bd9Sstevel@tonic-gate } 19027c478bd9Sstevel@tonic-gate 19037c478bd9Sstevel@tonic-gate if (cmd == MODLOADMINORPERM) 19047c478bd9Sstevel@tonic-gate minorperm_loaded = 1; 19057c478bd9Sstevel@tonic-gate 19067c478bd9Sstevel@tonic-gate /* 19077c478bd9Sstevel@tonic-gate * Reset permissions of cached dv_nodes 19087c478bd9Sstevel@tonic-gate */ 19097c478bd9Sstevel@tonic-gate (void) devfs_reset_perm(DV_RESET_PERM); 19107c478bd9Sstevel@tonic-gate 19117c478bd9Sstevel@tonic-gate return (0); 19127c478bd9Sstevel@tonic-gate } 19137c478bd9Sstevel@tonic-gate 19147c478bd9Sstevel@tonic-gate static int 19157c478bd9Sstevel@tonic-gate modctl_minorperm(int cmd, char *usrbuf, size_t buflen) 19167c478bd9Sstevel@tonic-gate { 19177c478bd9Sstevel@tonic-gate int error; 19187c478bd9Sstevel@tonic-gate nvlist_t *nvl; 19197c478bd9Sstevel@tonic-gate char *buf = kmem_alloc(buflen, KM_SLEEP); 19207c478bd9Sstevel@tonic-gate 19217c478bd9Sstevel@tonic-gate if ((error = ddi_copyin(usrbuf, buf, buflen, 0)) != 0) { 19227c478bd9Sstevel@tonic-gate kmem_free(buf, buflen); 19237c478bd9Sstevel@tonic-gate return (error); 19247c478bd9Sstevel@tonic-gate } 19257c478bd9Sstevel@tonic-gate 19267c478bd9Sstevel@tonic-gate error = nvlist_unpack(buf, buflen, &nvl, KM_SLEEP); 19277c478bd9Sstevel@tonic-gate kmem_free(buf, buflen); 19287c478bd9Sstevel@tonic-gate if (error) 19297c478bd9Sstevel@tonic-gate return (error); 19307c478bd9Sstevel@tonic-gate 19317c478bd9Sstevel@tonic-gate error = process_minorperm(cmd, nvl); 19327c478bd9Sstevel@tonic-gate nvlist_free(nvl); 19337c478bd9Sstevel@tonic-gate return (error); 19347c478bd9Sstevel@tonic-gate } 19357c478bd9Sstevel@tonic-gate 19367c478bd9Sstevel@tonic-gate struct walk_args { 19377c478bd9Sstevel@tonic-gate char *wa_drvname; 19387c478bd9Sstevel@tonic-gate list_t wa_pathlist; 19397c478bd9Sstevel@tonic-gate }; 19407c478bd9Sstevel@tonic-gate 19417c478bd9Sstevel@tonic-gate struct path_elem { 19427c478bd9Sstevel@tonic-gate char *pe_dir; 19437c478bd9Sstevel@tonic-gate char *pe_nodename; 19447c478bd9Sstevel@tonic-gate list_node_t pe_node; 19457c478bd9Sstevel@tonic-gate int pe_dirlen; 19467c478bd9Sstevel@tonic-gate }; 19477c478bd9Sstevel@tonic-gate 19487c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 19497c478bd9Sstevel@tonic-gate static int 19507c478bd9Sstevel@tonic-gate modctl_inst_walker(const char *path, in_node_t *np, in_drv_t *dp, void *arg) 19517c478bd9Sstevel@tonic-gate { 19527c478bd9Sstevel@tonic-gate struct walk_args *wargs = (struct walk_args *)arg; 19537c478bd9Sstevel@tonic-gate struct path_elem *pe; 19547c478bd9Sstevel@tonic-gate char *nodename; 19557c478bd9Sstevel@tonic-gate 19561aef0e11Sjg /* 19571aef0e11Sjg * Search may be restricted to a single driver in the case of rem_drv 19581aef0e11Sjg */ 19591aef0e11Sjg if (wargs->wa_drvname && 19601aef0e11Sjg strcmp(dp->ind_driver_name, wargs->wa_drvname) != 0) 19617c478bd9Sstevel@tonic-gate return (INST_WALK_CONTINUE); 19627c478bd9Sstevel@tonic-gate 19637c478bd9Sstevel@tonic-gate pe = kmem_zalloc(sizeof (*pe), KM_SLEEP); 19647c478bd9Sstevel@tonic-gate pe->pe_dir = i_ddi_strdup((char *)path, KM_SLEEP); 19657c478bd9Sstevel@tonic-gate pe->pe_dirlen = strlen(pe->pe_dir) + 1; 19667c478bd9Sstevel@tonic-gate ASSERT(strrchr(pe->pe_dir, '/') != NULL); 19677c478bd9Sstevel@tonic-gate nodename = strrchr(pe->pe_dir, '/'); 19687c478bd9Sstevel@tonic-gate *nodename++ = 0; 19697c478bd9Sstevel@tonic-gate pe->pe_nodename = nodename; 19707c478bd9Sstevel@tonic-gate list_insert_tail(&wargs->wa_pathlist, pe); 19717c478bd9Sstevel@tonic-gate 19727c478bd9Sstevel@tonic-gate return (INST_WALK_CONTINUE); 19737c478bd9Sstevel@tonic-gate } 19747c478bd9Sstevel@tonic-gate 19751aef0e11Sjg /* 19761aef0e11Sjg * /devices attribute nodes clean-up optionally performed 19771aef0e11Sjg * when removing a driver (rem_drv -C). 19781aef0e11Sjg * 19791aef0e11Sjg * Removing attribute nodes allows a machine to be reprovisioned 19801aef0e11Sjg * without the side-effect of inadvertently picking up stale 19811aef0e11Sjg * device node ownership or permissions. 19821aef0e11Sjg * 19831aef0e11Sjg * Preserving attributes (not performing cleanup) allows devices 19841aef0e11Sjg * attribute changes to be preserved across upgrades, as 19851aef0e11Sjg * upgrade rather heavy-handedly does a rem_drv/add_drv cycle. 19861aef0e11Sjg */ 19877c478bd9Sstevel@tonic-gate static int 19887c478bd9Sstevel@tonic-gate modctl_remdrv_cleanup(const char *u_drvname) 19897c478bd9Sstevel@tonic-gate { 19907c478bd9Sstevel@tonic-gate struct walk_args *wargs; 19917c478bd9Sstevel@tonic-gate struct path_elem *pe; 19927c478bd9Sstevel@tonic-gate char *drvname; 19937c478bd9Sstevel@tonic-gate int err, rval = 0; 19947c478bd9Sstevel@tonic-gate 19957c478bd9Sstevel@tonic-gate drvname = kmem_alloc(MAXMODCONFNAME, KM_SLEEP); 19967c478bd9Sstevel@tonic-gate if ((err = copyinstr(u_drvname, drvname, MAXMODCONFNAME, 0))) { 19977c478bd9Sstevel@tonic-gate kmem_free(drvname, MAXMODCONFNAME); 19987c478bd9Sstevel@tonic-gate return (err); 19997c478bd9Sstevel@tonic-gate } 20007c478bd9Sstevel@tonic-gate 20017c478bd9Sstevel@tonic-gate /* 20027c478bd9Sstevel@tonic-gate * First go through the instance database. For each 20037c478bd9Sstevel@tonic-gate * instance of a device bound to the driver being 20047c478bd9Sstevel@tonic-gate * removed, remove any underlying devfs attribute nodes. 20057c478bd9Sstevel@tonic-gate * 20067c478bd9Sstevel@tonic-gate * This is a two-step process. First we go through 20077c478bd9Sstevel@tonic-gate * the instance data itself, constructing a list of 20087c478bd9Sstevel@tonic-gate * the nodes discovered. The second step is then 20097c478bd9Sstevel@tonic-gate * to find and remove any devfs attribute nodes 20107c478bd9Sstevel@tonic-gate * for the instances discovered in the first step. 20117c478bd9Sstevel@tonic-gate * The two-step process avoids any difficulties 20127c478bd9Sstevel@tonic-gate * which could arise by holding the instance data 20137c478bd9Sstevel@tonic-gate * lock with simultaneous devfs operations. 20147c478bd9Sstevel@tonic-gate */ 20157c478bd9Sstevel@tonic-gate wargs = kmem_zalloc(sizeof (*wargs), KM_SLEEP); 20167c478bd9Sstevel@tonic-gate 20177c478bd9Sstevel@tonic-gate wargs->wa_drvname = drvname; 20187c478bd9Sstevel@tonic-gate list_create(&wargs->wa_pathlist, 20197c478bd9Sstevel@tonic-gate sizeof (struct path_elem), offsetof(struct path_elem, pe_node)); 20207c478bd9Sstevel@tonic-gate 20217c478bd9Sstevel@tonic-gate (void) e_ddi_walk_instances(modctl_inst_walker, (void *)wargs); 20227c478bd9Sstevel@tonic-gate 20237c478bd9Sstevel@tonic-gate for (pe = list_head(&wargs->wa_pathlist); pe != NULL; 20247c478bd9Sstevel@tonic-gate pe = list_next(&wargs->wa_pathlist, pe)) { 20257c478bd9Sstevel@tonic-gate err = devfs_remdrv_cleanup((const char *)pe->pe_dir, 20267c478bd9Sstevel@tonic-gate (const char *)pe->pe_nodename); 20277c478bd9Sstevel@tonic-gate if (rval == 0) 20287c478bd9Sstevel@tonic-gate rval = err; 20297c478bd9Sstevel@tonic-gate } 20307c478bd9Sstevel@tonic-gate 20317c478bd9Sstevel@tonic-gate while ((pe = list_head(&wargs->wa_pathlist)) != NULL) { 20327c478bd9Sstevel@tonic-gate list_remove(&wargs->wa_pathlist, pe); 20337c478bd9Sstevel@tonic-gate kmem_free(pe->pe_dir, pe->pe_dirlen); 20347c478bd9Sstevel@tonic-gate kmem_free(pe, sizeof (*pe)); 20357c478bd9Sstevel@tonic-gate } 20367c478bd9Sstevel@tonic-gate kmem_free(wargs, sizeof (*wargs)); 20377c478bd9Sstevel@tonic-gate 20387c478bd9Sstevel@tonic-gate /* 20397c478bd9Sstevel@tonic-gate * Pseudo nodes aren't recorded in the instance database 20407c478bd9Sstevel@tonic-gate * so any such nodes need to be handled separately. 20417c478bd9Sstevel@tonic-gate */ 20427c478bd9Sstevel@tonic-gate err = devfs_remdrv_cleanup("pseudo", (const char *)drvname); 20437c478bd9Sstevel@tonic-gate if (rval == 0) 20447c478bd9Sstevel@tonic-gate rval = err; 20457c478bd9Sstevel@tonic-gate 20467c478bd9Sstevel@tonic-gate kmem_free(drvname, MAXMODCONFNAME); 20477c478bd9Sstevel@tonic-gate return (rval); 20487c478bd9Sstevel@tonic-gate } 20497c478bd9Sstevel@tonic-gate 20501aef0e11Sjg /* 20511aef0e11Sjg * Perform a cleanup of non-existent /devices attribute nodes, 20521aef0e11Sjg * similar to rem_drv -C, but for all drivers/devices. 20531aef0e11Sjg * This is also optional, performed as part of devfsadm -C. 20541aef0e11Sjg */ 20551aef0e11Sjg void 20561aef0e11Sjg dev_devices_cleanup() 20571aef0e11Sjg { 20581aef0e11Sjg struct walk_args *wargs; 20591aef0e11Sjg struct path_elem *pe; 20601aef0e11Sjg dev_info_t *devi; 20611aef0e11Sjg char *path; 20621aef0e11Sjg int err; 20631aef0e11Sjg 20641aef0e11Sjg /* 20651aef0e11Sjg * It's expected that all drivers have been loaded and 20661aef0e11Sjg * module unloading disabled while performing cleanup. 20671aef0e11Sjg */ 20681aef0e11Sjg ASSERT(modunload_disable_count > 0); 20691aef0e11Sjg 20701aef0e11Sjg wargs = kmem_zalloc(sizeof (*wargs), KM_SLEEP); 20711aef0e11Sjg wargs->wa_drvname = NULL; 20721aef0e11Sjg list_create(&wargs->wa_pathlist, 20731aef0e11Sjg sizeof (struct path_elem), offsetof(struct path_elem, pe_node)); 20741aef0e11Sjg 20751aef0e11Sjg (void) e_ddi_walk_instances(modctl_inst_walker, (void *)wargs); 20761aef0e11Sjg 20771aef0e11Sjg path = kmem_alloc(MAXPATHLEN, KM_SLEEP); 20781aef0e11Sjg 20791aef0e11Sjg for (pe = list_head(&wargs->wa_pathlist); pe != NULL; 20801aef0e11Sjg pe = list_next(&wargs->wa_pathlist, pe)) { 20811aef0e11Sjg (void) snprintf(path, MAXPATHLEN, "%s/%s", 20821aef0e11Sjg pe->pe_dir, pe->pe_nodename); 20831aef0e11Sjg devi = e_ddi_hold_devi_by_path(path, 0); 20841aef0e11Sjg if (devi != NULL) { 20851aef0e11Sjg ddi_release_devi(devi); 20861aef0e11Sjg } else { 20871aef0e11Sjg err = devfs_remdrv_cleanup((const char *)pe->pe_dir, 20881aef0e11Sjg (const char *)pe->pe_nodename); 20891aef0e11Sjg if (err) { 20901aef0e11Sjg cmn_err(CE_CONT, 20911aef0e11Sjg "devfs: %s: clean-up error %d\n", 20921aef0e11Sjg path, err); 20931aef0e11Sjg } 20941aef0e11Sjg } 20951aef0e11Sjg } 20961aef0e11Sjg 20971aef0e11Sjg while ((pe = list_head(&wargs->wa_pathlist)) != NULL) { 20981aef0e11Sjg list_remove(&wargs->wa_pathlist, pe); 20991aef0e11Sjg kmem_free(pe->pe_dir, pe->pe_dirlen); 21001aef0e11Sjg kmem_free(pe, sizeof (*pe)); 21011aef0e11Sjg } 21021aef0e11Sjg kmem_free(wargs, sizeof (*wargs)); 21031aef0e11Sjg kmem_free(path, MAXPATHLEN); 21041aef0e11Sjg } 21051aef0e11Sjg 21067c478bd9Sstevel@tonic-gate static int 21077c478bd9Sstevel@tonic-gate modctl_allocpriv(const char *name) 21087c478bd9Sstevel@tonic-gate { 21097c478bd9Sstevel@tonic-gate char *pstr = kmem_alloc(PRIVNAME_MAX, KM_SLEEP); 21107c478bd9Sstevel@tonic-gate int error; 21117c478bd9Sstevel@tonic-gate 21127c478bd9Sstevel@tonic-gate if ((error = copyinstr(name, pstr, PRIVNAME_MAX, 0))) { 21137c478bd9Sstevel@tonic-gate kmem_free(pstr, PRIVNAME_MAX); 21147c478bd9Sstevel@tonic-gate return (error); 21157c478bd9Sstevel@tonic-gate } 21167c478bd9Sstevel@tonic-gate error = priv_getbyname(pstr, PRIV_ALLOC); 21177c478bd9Sstevel@tonic-gate if (error < 0) 21187c478bd9Sstevel@tonic-gate error = -error; 21197c478bd9Sstevel@tonic-gate else 21207c478bd9Sstevel@tonic-gate error = 0; 21217c478bd9Sstevel@tonic-gate kmem_free(pstr, PRIVNAME_MAX); 21227c478bd9Sstevel@tonic-gate return (error); 21237c478bd9Sstevel@tonic-gate } 21247c478bd9Sstevel@tonic-gate 2125facf4a8dSllai1 static int 2126facf4a8dSllai1 modctl_devexists(const char *upath, int pathlen) 2127facf4a8dSllai1 { 2128facf4a8dSllai1 char *path; 2129facf4a8dSllai1 int ret; 2130facf4a8dSllai1 2131facf4a8dSllai1 /* 2132facf4a8dSllai1 * copy in the path, including the terminating null 2133facf4a8dSllai1 */ 2134facf4a8dSllai1 pathlen++; 2135facf4a8dSllai1 if (pathlen <= 1 || pathlen > MAXPATHLEN) 2136facf4a8dSllai1 return (EINVAL); 2137facf4a8dSllai1 path = kmem_zalloc(pathlen + 1, KM_SLEEP); 2138facf4a8dSllai1 if ((ret = copyinstr(upath, path, pathlen, NULL)) == 0) { 2139facf4a8dSllai1 ret = sdev_modctl_devexists(path); 2140facf4a8dSllai1 } 2141facf4a8dSllai1 2142facf4a8dSllai1 kmem_free(path, pathlen + 1); 2143facf4a8dSllai1 return (ret); 2144facf4a8dSllai1 } 2145facf4a8dSllai1 2146facf4a8dSllai1 static int 2147facf4a8dSllai1 modctl_devreaddir(const char *udir, int udirlen, 2148facf4a8dSllai1 char *upaths, int64_t *ulensp) 2149facf4a8dSllai1 { 2150facf4a8dSllai1 char *paths = NULL; 2151facf4a8dSllai1 char **dirlist = NULL; 2152facf4a8dSllai1 char *dir; 2153facf4a8dSllai1 int64_t ulens; 2154facf4a8dSllai1 int64_t lens; 2155facf4a8dSllai1 int i, n; 2156facf4a8dSllai1 int ret = 0; 2157facf4a8dSllai1 char *p; 2158facf4a8dSllai1 int npaths; 2159facf4a8dSllai1 int npaths_alloc; 2160facf4a8dSllai1 2161facf4a8dSllai1 /* 2162facf4a8dSllai1 * If upaths is NULL then we are only computing the amount of space 2163facf4a8dSllai1 * needed to return the paths, with the value returned in *ulensp. If we 2164facf4a8dSllai1 * are copying out paths then we get the amount of space allocated by 2165facf4a8dSllai1 * the caller. If the actual space needed for paths is larger, or 2166facf4a8dSllai1 * things are changing out from under us, then we return EAGAIN. 2167facf4a8dSllai1 */ 2168facf4a8dSllai1 if (upaths) { 2169facf4a8dSllai1 if (ulensp == NULL) 2170facf4a8dSllai1 return (EINVAL); 2171facf4a8dSllai1 if (copyin(ulensp, &ulens, sizeof (ulens)) != 0) 2172facf4a8dSllai1 return (EFAULT); 2173facf4a8dSllai1 } 2174facf4a8dSllai1 2175facf4a8dSllai1 /* 2176facf4a8dSllai1 * copyin the /dev path including terminating null 2177facf4a8dSllai1 */ 2178facf4a8dSllai1 udirlen++; 2179facf4a8dSllai1 if (udirlen <= 1 || udirlen > MAXPATHLEN) 2180facf4a8dSllai1 return (EINVAL); 2181facf4a8dSllai1 dir = kmem_zalloc(udirlen + 1, KM_SLEEP); 2182facf4a8dSllai1 if ((ret = copyinstr(udir, dir, udirlen, NULL)) != 0) 2183facf4a8dSllai1 goto err; 2184facf4a8dSllai1 2185facf4a8dSllai1 if ((ret = sdev_modctl_readdir(dir, &dirlist, 2186e37c6c37Scth &npaths, &npaths_alloc, 0)) != 0) { 2187facf4a8dSllai1 ASSERT(dirlist == NULL); 2188facf4a8dSllai1 goto err; 2189facf4a8dSllai1 } 2190facf4a8dSllai1 2191facf4a8dSllai1 lens = 0; 2192facf4a8dSllai1 for (i = 0; i < npaths; i++) { 2193facf4a8dSllai1 lens += strlen(dirlist[i]) + 1; 2194facf4a8dSllai1 } 2195facf4a8dSllai1 lens++; /* add one for double termination */ 2196facf4a8dSllai1 2197facf4a8dSllai1 if (upaths) { 2198facf4a8dSllai1 if (lens > ulens) { 2199facf4a8dSllai1 ret = EAGAIN; 2200facf4a8dSllai1 goto out; 2201facf4a8dSllai1 } 2202facf4a8dSllai1 2203facf4a8dSllai1 paths = kmem_alloc(lens, KM_SLEEP); 2204facf4a8dSllai1 2205facf4a8dSllai1 p = paths; 2206facf4a8dSllai1 for (i = 0; i < npaths; i++) { 2207facf4a8dSllai1 n = strlen(dirlist[i]) + 1; 2208facf4a8dSllai1 bcopy(dirlist[i], p, n); 2209facf4a8dSllai1 p += n; 2210facf4a8dSllai1 } 2211facf4a8dSllai1 *p = 0; 2212facf4a8dSllai1 2213facf4a8dSllai1 if (copyout(paths, upaths, lens)) { 2214facf4a8dSllai1 ret = EFAULT; 2215facf4a8dSllai1 goto err; 2216facf4a8dSllai1 } 2217facf4a8dSllai1 } 2218facf4a8dSllai1 2219facf4a8dSllai1 out: 2220facf4a8dSllai1 /* copy out the amount of space needed to hold the paths */ 2221facf4a8dSllai1 if (copyout(&lens, ulensp, sizeof (lens))) 2222facf4a8dSllai1 ret = EFAULT; 2223facf4a8dSllai1 2224facf4a8dSllai1 err: 2225facf4a8dSllai1 if (dirlist) 2226facf4a8dSllai1 sdev_modctl_readdir_free(dirlist, npaths, npaths_alloc); 2227facf4a8dSllai1 if (paths) 2228facf4a8dSllai1 kmem_free(paths, lens); 2229facf4a8dSllai1 kmem_free(dir, udirlen + 1); 2230facf4a8dSllai1 return (ret); 2231facf4a8dSllai1 } 2232facf4a8dSllai1 2233e37c6c37Scth static int 2234e37c6c37Scth modctl_devemptydir(const char *udir, int udirlen, int *uempty) 2235e37c6c37Scth { 2236e37c6c37Scth char *dir; 2237e37c6c37Scth int ret; 2238e37c6c37Scth char **dirlist = NULL; 2239e37c6c37Scth int npaths; 2240e37c6c37Scth int npaths_alloc; 2241e37c6c37Scth int empty; 2242e37c6c37Scth 2243e37c6c37Scth /* 2244e37c6c37Scth * copyin the /dev path including terminating null 2245e37c6c37Scth */ 2246e37c6c37Scth udirlen++; 2247e37c6c37Scth if (udirlen <= 1 || udirlen > MAXPATHLEN) 2248e37c6c37Scth return (EINVAL); 2249e37c6c37Scth dir = kmem_zalloc(udirlen + 1, KM_SLEEP); 2250e37c6c37Scth if ((ret = copyinstr(udir, dir, udirlen, NULL)) != 0) 2251e37c6c37Scth goto err; 2252e37c6c37Scth 2253e37c6c37Scth if ((ret = sdev_modctl_readdir(dir, &dirlist, 2254e37c6c37Scth &npaths, &npaths_alloc, 1)) != 0) { 2255e37c6c37Scth goto err; 2256e37c6c37Scth } 2257e37c6c37Scth 2258e37c6c37Scth empty = npaths ? 0 : 1; 2259e37c6c37Scth if (copyout(&empty, uempty, sizeof (empty))) 2260e37c6c37Scth ret = EFAULT; 2261e37c6c37Scth 2262e37c6c37Scth err: 2263e37c6c37Scth if (dirlist) 2264e37c6c37Scth sdev_modctl_readdir_free(dirlist, npaths, npaths_alloc); 2265e37c6c37Scth kmem_free(dir, udirlen + 1); 2266e37c6c37Scth return (ret); 2267e37c6c37Scth } 2268e37c6c37Scth 226926947304SEvan Yan static int 227026947304SEvan Yan modctl_hp(int subcmd, const char *path, char *cn_name, uintptr_t arg, 227126947304SEvan Yan uintptr_t rval) 227226947304SEvan Yan { 227326947304SEvan Yan int error = 0; 227426947304SEvan Yan size_t pathsz, namesz; 227526947304SEvan Yan char *devpath, *cn_name_str; 227626947304SEvan Yan 227726947304SEvan Yan if (path == NULL) 227826947304SEvan Yan return (EINVAL); 227926947304SEvan Yan 228026947304SEvan Yan devpath = kmem_zalloc(MAXPATHLEN, KM_SLEEP); 228126947304SEvan Yan error = copyinstr(path, devpath, MAXPATHLEN, &pathsz); 228226947304SEvan Yan if (error != 0) { 228326947304SEvan Yan kmem_free(devpath, MAXPATHLEN); 228426947304SEvan Yan return (EFAULT); 228526947304SEvan Yan } 228626947304SEvan Yan 228726947304SEvan Yan cn_name_str = kmem_zalloc(MAXNAMELEN, KM_SLEEP); 228826947304SEvan Yan error = copyinstr(cn_name, cn_name_str, MAXNAMELEN, &namesz); 228926947304SEvan Yan if (error != 0) { 229026947304SEvan Yan kmem_free(devpath, MAXPATHLEN); 229126947304SEvan Yan kmem_free(cn_name_str, MAXNAMELEN); 229226947304SEvan Yan 229326947304SEvan Yan return (EFAULT); 229426947304SEvan Yan } 229526947304SEvan Yan 229626947304SEvan Yan switch (subcmd) { 229726947304SEvan Yan case MODHPOPS_CHANGE_STATE: 229826947304SEvan Yan error = ddihp_modctl(DDI_HPOP_CN_CHANGE_STATE, devpath, 229926947304SEvan Yan cn_name_str, arg, NULL); 230026947304SEvan Yan break; 230126947304SEvan Yan case MODHPOPS_CREATE_PORT: 230226947304SEvan Yan /* Create an empty PORT */ 230326947304SEvan Yan error = ddihp_modctl(DDI_HPOP_CN_CREATE_PORT, devpath, 230426947304SEvan Yan cn_name_str, NULL, NULL); 230526947304SEvan Yan break; 230626947304SEvan Yan case MODHPOPS_REMOVE_PORT: 230726947304SEvan Yan /* Remove an empty PORT */ 230826947304SEvan Yan error = ddihp_modctl(DDI_HPOP_CN_REMOVE_PORT, devpath, 230926947304SEvan Yan cn_name_str, NULL, NULL); 231026947304SEvan Yan break; 231126947304SEvan Yan case MODHPOPS_BUS_GET: 231226947304SEvan Yan error = ddihp_modctl(DDI_HPOP_CN_GET_PROPERTY, devpath, 231326947304SEvan Yan cn_name_str, arg, rval); 231426947304SEvan Yan break; 231526947304SEvan Yan case MODHPOPS_BUS_SET: 231626947304SEvan Yan error = ddihp_modctl(DDI_HPOP_CN_SET_PROPERTY, devpath, 231726947304SEvan Yan cn_name_str, arg, rval); 231826947304SEvan Yan break; 231926947304SEvan Yan default: 232026947304SEvan Yan error = ENOTSUP; 232126947304SEvan Yan break; 232226947304SEvan Yan } 232326947304SEvan Yan 232426947304SEvan Yan kmem_free(devpath, MAXPATHLEN); 232526947304SEvan Yan kmem_free(cn_name_str, MAXNAMELEN); 232626947304SEvan Yan 232726947304SEvan Yan return (error); 232826947304SEvan Yan } 232926947304SEvan Yan 2330facf4a8dSllai1 int 2331facf4a8dSllai1 modctl_moddevname(int subcmd, uintptr_t a1, uintptr_t a2) 2332facf4a8dSllai1 { 2333facf4a8dSllai1 int error = 0; 2334facf4a8dSllai1 2335facf4a8dSllai1 switch (subcmd) { 2336facf4a8dSllai1 case MODDEVNAME_LOOKUPDOOR: 23373c5e027bSEric Taylor error = devname_filename_register((char *)a1); 2338facf4a8dSllai1 break; 2339facf4a8dSllai1 case MODDEVNAME_PROFILE: 2340facf4a8dSllai1 error = devname_profile_update((char *)a1, (size_t)a2); 2341facf4a8dSllai1 break; 2342facf4a8dSllai1 case MODDEVNAME_RECONFIG: 2343facf4a8dSllai1 i_ddi_set_reconfig(); 2344facf4a8dSllai1 break; 2345facf4a8dSllai1 case MODDEVNAME_SYSAVAIL: 2346facf4a8dSllai1 i_ddi_set_sysavail(); 2347facf4a8dSllai1 break; 2348facf4a8dSllai1 default: 2349facf4a8dSllai1 error = EINVAL; 2350facf4a8dSllai1 break; 2351facf4a8dSllai1 } 2352facf4a8dSllai1 2353facf4a8dSllai1 return (error); 2354facf4a8dSllai1 } 2355facf4a8dSllai1 23567c478bd9Sstevel@tonic-gate /*ARGSUSED5*/ 23577c478bd9Sstevel@tonic-gate int 23587c478bd9Sstevel@tonic-gate modctl(int cmd, uintptr_t a1, uintptr_t a2, uintptr_t a3, uintptr_t a4, 23597c478bd9Sstevel@tonic-gate uintptr_t a5) 23607c478bd9Sstevel@tonic-gate { 23617c478bd9Sstevel@tonic-gate int error = EINVAL; 23627c478bd9Sstevel@tonic-gate dev_t dev; 23637c478bd9Sstevel@tonic-gate 23647c478bd9Sstevel@tonic-gate if (secpolicy_modctl(CRED(), cmd) != 0) 23657c478bd9Sstevel@tonic-gate return (set_errno(EPERM)); 23667c478bd9Sstevel@tonic-gate 23677c478bd9Sstevel@tonic-gate switch (cmd) { 23687c478bd9Sstevel@tonic-gate case MODLOAD: /* load a module */ 23697c478bd9Sstevel@tonic-gate error = modctl_modload((int)a1, (char *)a2, (int *)a3); 23707c478bd9Sstevel@tonic-gate break; 23717c478bd9Sstevel@tonic-gate 23727c478bd9Sstevel@tonic-gate case MODUNLOAD: /* unload a module */ 23737c478bd9Sstevel@tonic-gate error = modctl_modunload((modid_t)a1); 23747c478bd9Sstevel@tonic-gate break; 23757c478bd9Sstevel@tonic-gate 23767c478bd9Sstevel@tonic-gate case MODINFO: /* get module status */ 23777c478bd9Sstevel@tonic-gate error = modctl_modinfo((modid_t)a1, (struct modinfo *)a2); 23787c478bd9Sstevel@tonic-gate break; 23797c478bd9Sstevel@tonic-gate 23807c478bd9Sstevel@tonic-gate case MODRESERVED: /* get last major number in range */ 23817c478bd9Sstevel@tonic-gate error = modctl_modreserve((modid_t)a1, (int *)a2); 23827c478bd9Sstevel@tonic-gate break; 23837c478bd9Sstevel@tonic-gate 23847c478bd9Sstevel@tonic-gate case MODSETMINIROOT: /* we are running in miniroot */ 23857c478bd9Sstevel@tonic-gate isminiroot = 1; 23867c478bd9Sstevel@tonic-gate error = 0; 23877c478bd9Sstevel@tonic-gate break; 23887c478bd9Sstevel@tonic-gate 23896532b960SJerry Gilliam case MODADDMAJBIND: /* add major / driver alias bindings */ 23906532b960SJerry Gilliam error = modctl_add_driver_aliases((int *)a2); 23917c478bd9Sstevel@tonic-gate break; 23927c478bd9Sstevel@tonic-gate 23937c478bd9Sstevel@tonic-gate case MODGETPATHLEN: /* get modpath length */ 23947c478bd9Sstevel@tonic-gate error = modctl_getmodpathlen((int *)a2); 23957c478bd9Sstevel@tonic-gate break; 23967c478bd9Sstevel@tonic-gate 23977c478bd9Sstevel@tonic-gate case MODGETPATH: /* get modpath */ 23987c478bd9Sstevel@tonic-gate error = modctl_getmodpath((char *)a2); 23997c478bd9Sstevel@tonic-gate break; 24007c478bd9Sstevel@tonic-gate 24017c478bd9Sstevel@tonic-gate case MODREADSYSBIND: /* read system call binding file */ 24027c478bd9Sstevel@tonic-gate error = modctl_read_sysbinding_file(); 24037c478bd9Sstevel@tonic-gate break; 24047c478bd9Sstevel@tonic-gate 24057c478bd9Sstevel@tonic-gate case MODGETMAJBIND: /* get major number for named device */ 24067c478bd9Sstevel@tonic-gate error = modctl_getmaj((char *)a1, (uint_t)a2, (int *)a3); 24077c478bd9Sstevel@tonic-gate break; 24087c478bd9Sstevel@tonic-gate 24097c478bd9Sstevel@tonic-gate case MODGETNAME: /* get name of device given major number */ 24107c478bd9Sstevel@tonic-gate error = modctl_getname((char *)a1, (uint_t)a2, (int *)a3); 24117c478bd9Sstevel@tonic-gate break; 24127c478bd9Sstevel@tonic-gate 24137c478bd9Sstevel@tonic-gate case MODDEVT2INSTANCE: 24147c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24157c478bd9Sstevel@tonic-gate dev = (dev_t)a1; 24167c478bd9Sstevel@tonic-gate } 24177c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24187c478bd9Sstevel@tonic-gate else { 24197c478bd9Sstevel@tonic-gate dev = expldev(a1); 24207c478bd9Sstevel@tonic-gate } 24217c478bd9Sstevel@tonic-gate #endif 24227c478bd9Sstevel@tonic-gate error = modctl_devt2instance(dev, (int *)a2); 24237c478bd9Sstevel@tonic-gate break; 24247c478bd9Sstevel@tonic-gate 24257c478bd9Sstevel@tonic-gate case MODSIZEOF_DEVID: /* sizeof device id of device given dev_t */ 24267c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24277c478bd9Sstevel@tonic-gate dev = (dev_t)a1; 24287c478bd9Sstevel@tonic-gate } 24297c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24307c478bd9Sstevel@tonic-gate else { 24317c478bd9Sstevel@tonic-gate dev = expldev(a1); 24327c478bd9Sstevel@tonic-gate } 24337c478bd9Sstevel@tonic-gate #endif 24347c478bd9Sstevel@tonic-gate error = modctl_sizeof_devid(dev, (uint_t *)a2); 24357c478bd9Sstevel@tonic-gate break; 24367c478bd9Sstevel@tonic-gate 24377c478bd9Sstevel@tonic-gate case MODGETDEVID: /* get device id of device given dev_t */ 24387c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24397c478bd9Sstevel@tonic-gate dev = (dev_t)a1; 24407c478bd9Sstevel@tonic-gate } 24417c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24427c478bd9Sstevel@tonic-gate else { 24437c478bd9Sstevel@tonic-gate dev = expldev(a1); 24447c478bd9Sstevel@tonic-gate } 24457c478bd9Sstevel@tonic-gate #endif 24467c478bd9Sstevel@tonic-gate error = modctl_get_devid(dev, (uint_t)a2, (ddi_devid_t)a3); 24477c478bd9Sstevel@tonic-gate break; 24487c478bd9Sstevel@tonic-gate 2449a08731ecScth case MODSIZEOF_MINORNAME: /* sizeof minor nm (dev_t,spectype) */ 24507c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24517c478bd9Sstevel@tonic-gate error = modctl_sizeof_minorname((dev_t)a1, (int)a2, 24527c478bd9Sstevel@tonic-gate (uint_t *)a3); 24537c478bd9Sstevel@tonic-gate } 24547c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24557c478bd9Sstevel@tonic-gate else { 24567c478bd9Sstevel@tonic-gate error = modctl_sizeof_minorname(expldev(a1), (int)a2, 24577c478bd9Sstevel@tonic-gate (uint_t *)a3); 24587c478bd9Sstevel@tonic-gate } 24597c478bd9Sstevel@tonic-gate 24607c478bd9Sstevel@tonic-gate #endif 24617c478bd9Sstevel@tonic-gate break; 24627c478bd9Sstevel@tonic-gate 2463a08731ecScth case MODGETMINORNAME: /* get minor name of (dev_t,spectype) */ 24647c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24657c478bd9Sstevel@tonic-gate error = modctl_get_minorname((dev_t)a1, (int)a2, 24667c478bd9Sstevel@tonic-gate (uint_t)a3, (char *)a4); 24677c478bd9Sstevel@tonic-gate } 24687c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24697c478bd9Sstevel@tonic-gate else { 24707c478bd9Sstevel@tonic-gate error = modctl_get_minorname(expldev(a1), (int)a2, 24717c478bd9Sstevel@tonic-gate (uint_t)a3, (char *)a4); 24727c478bd9Sstevel@tonic-gate } 24737c478bd9Sstevel@tonic-gate #endif 24747c478bd9Sstevel@tonic-gate break; 24757c478bd9Sstevel@tonic-gate 2476a08731ecScth case MODGETDEVFSPATH_LEN: /* sizeof path nm of (dev_t,spectype) */ 24777c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24787c478bd9Sstevel@tonic-gate error = modctl_devfspath_len((dev_t)a1, (int)a2, 24797c478bd9Sstevel@tonic-gate (uint_t *)a3); 24807c478bd9Sstevel@tonic-gate } 24817c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24827c478bd9Sstevel@tonic-gate else { 24837c478bd9Sstevel@tonic-gate error = modctl_devfspath_len(expldev(a1), (int)a2, 24847c478bd9Sstevel@tonic-gate (uint_t *)a3); 24857c478bd9Sstevel@tonic-gate } 24867c478bd9Sstevel@tonic-gate 24877c478bd9Sstevel@tonic-gate #endif 24887c478bd9Sstevel@tonic-gate break; 24897c478bd9Sstevel@tonic-gate 2490a08731ecScth case MODGETDEVFSPATH: /* get path name of (dev_t,spec) type */ 24917c478bd9Sstevel@tonic-gate if (get_udatamodel() == DATAMODEL_NATIVE) { 24927c478bd9Sstevel@tonic-gate error = modctl_devfspath((dev_t)a1, (int)a2, 24937c478bd9Sstevel@tonic-gate (uint_t)a3, (char *)a4); 24947c478bd9Sstevel@tonic-gate } 24957c478bd9Sstevel@tonic-gate #ifdef _SYSCALL32_IMPL 24967c478bd9Sstevel@tonic-gate else { 24977c478bd9Sstevel@tonic-gate error = modctl_devfspath(expldev(a1), (int)a2, 24987c478bd9Sstevel@tonic-gate (uint_t)a3, (char *)a4); 24997c478bd9Sstevel@tonic-gate } 25007c478bd9Sstevel@tonic-gate #endif 25017c478bd9Sstevel@tonic-gate break; 25027c478bd9Sstevel@tonic-gate 2503a08731ecScth case MODGETDEVFSPATH_MI_LEN: /* sizeof path nm of (major,instance) */ 2504a08731ecScth error = modctl_devfspath_mi_len((major_t)a1, (int)a2, 2505a08731ecScth (uint_t *)a3); 2506a08731ecScth break; 2507a08731ecScth 2508a08731ecScth case MODGETDEVFSPATH_MI: /* get path name of (major,instance) */ 2509a08731ecScth error = modctl_devfspath_mi((major_t)a1, (int)a2, 2510a08731ecScth (uint_t)a3, (char *)a4); 2511a08731ecScth break; 2512a08731ecScth 25137c478bd9Sstevel@tonic-gate 25147c478bd9Sstevel@tonic-gate case MODEVENTS: 25157c478bd9Sstevel@tonic-gate error = modctl_modevents((int)a1, a2, a3, a4, (uint_t)a5); 25167c478bd9Sstevel@tonic-gate break; 25177c478bd9Sstevel@tonic-gate 25187c478bd9Sstevel@tonic-gate case MODGETFBNAME: /* get the framebuffer name */ 25197c478bd9Sstevel@tonic-gate error = modctl_get_fbname((char *)a1); 25207c478bd9Sstevel@tonic-gate break; 25217c478bd9Sstevel@tonic-gate 25227c478bd9Sstevel@tonic-gate case MODREREADDACF: /* reread dacf rule database from given file */ 25237c478bd9Sstevel@tonic-gate error = modctl_reread_dacf((char *)a1); 25247c478bd9Sstevel@tonic-gate break; 25257c478bd9Sstevel@tonic-gate 25267c478bd9Sstevel@tonic-gate case MODLOADDRVCONF: /* load driver.conf file for major */ 2527c9cc1492SJerry Gilliam error = modctl_load_drvconf((major_t)a1, (int)a2); 25287c478bd9Sstevel@tonic-gate break; 25297c478bd9Sstevel@tonic-gate 25307c478bd9Sstevel@tonic-gate case MODUNLOADDRVCONF: /* unload driver.conf file for major */ 25317c478bd9Sstevel@tonic-gate error = modctl_unload_drvconf((major_t)a1); 25327c478bd9Sstevel@tonic-gate break; 25337c478bd9Sstevel@tonic-gate 25347c478bd9Sstevel@tonic-gate case MODREMMAJBIND: /* remove a major binding */ 25357c478bd9Sstevel@tonic-gate error = modctl_rem_major((major_t)a1); 25367c478bd9Sstevel@tonic-gate break; 25377c478bd9Sstevel@tonic-gate 25386532b960SJerry Gilliam case MODREMDRVALIAS: /* remove a major/alias binding */ 25396532b960SJerry Gilliam error = modctl_remove_driver_aliases((int *)a2); 25406532b960SJerry Gilliam break; 25416532b960SJerry Gilliam 25427c478bd9Sstevel@tonic-gate case MODDEVID2PATHS: /* get paths given devid */ 25437c478bd9Sstevel@tonic-gate error = modctl_devid2paths((ddi_devid_t)a1, (char *)a2, 25447c478bd9Sstevel@tonic-gate (uint_t)a3, (size_t *)a4, (char *)a5); 25457c478bd9Sstevel@tonic-gate break; 25467c478bd9Sstevel@tonic-gate 25477c478bd9Sstevel@tonic-gate case MODSETDEVPOLICY: /* establish device policy */ 25487c478bd9Sstevel@tonic-gate error = devpolicy_load((int)a1, (size_t)a2, (devplcysys_t *)a3); 25497c478bd9Sstevel@tonic-gate break; 25507c478bd9Sstevel@tonic-gate 25517c478bd9Sstevel@tonic-gate case MODGETDEVPOLICY: /* get device policy */ 25527c478bd9Sstevel@tonic-gate error = devpolicy_get((int *)a1, (size_t)a2, 25537c478bd9Sstevel@tonic-gate (devplcysys_t *)a3); 25547c478bd9Sstevel@tonic-gate break; 25557c478bd9Sstevel@tonic-gate 25567c478bd9Sstevel@tonic-gate case MODALLOCPRIV: 25577c478bd9Sstevel@tonic-gate error = modctl_allocpriv((const char *)a1); 25587c478bd9Sstevel@tonic-gate break; 25597c478bd9Sstevel@tonic-gate 25607c478bd9Sstevel@tonic-gate case MODGETDEVPOLICYBYNAME: 25617c478bd9Sstevel@tonic-gate error = devpolicy_getbyname((size_t)a1, 25627c478bd9Sstevel@tonic-gate (devplcysys_t *)a2, (char *)a3); 25637c478bd9Sstevel@tonic-gate break; 25647c478bd9Sstevel@tonic-gate 25657c478bd9Sstevel@tonic-gate case MODLOADMINORPERM: 25667c478bd9Sstevel@tonic-gate case MODADDMINORPERM: 25677c478bd9Sstevel@tonic-gate case MODREMMINORPERM: 25687c478bd9Sstevel@tonic-gate error = modctl_minorperm(cmd, (char *)a1, (size_t)a2); 25697c478bd9Sstevel@tonic-gate break; 25707c478bd9Sstevel@tonic-gate 25717c478bd9Sstevel@tonic-gate case MODREMDRVCLEANUP: 25727c478bd9Sstevel@tonic-gate error = modctl_remdrv_cleanup((const char *)a1); 25737c478bd9Sstevel@tonic-gate break; 25747c478bd9Sstevel@tonic-gate 2575facf4a8dSllai1 case MODDEVEXISTS: /* non-reconfiguring /dev lookup */ 2576facf4a8dSllai1 error = modctl_devexists((const char *)a1, (size_t)a2); 2577facf4a8dSllai1 break; 2578facf4a8dSllai1 2579facf4a8dSllai1 case MODDEVREADDIR: /* non-reconfiguring /dev readdir */ 2580facf4a8dSllai1 error = modctl_devreaddir((const char *)a1, (size_t)a2, 2581facf4a8dSllai1 (char *)a3, (int64_t *)a4); 2582facf4a8dSllai1 break; 2583facf4a8dSllai1 2584e37c6c37Scth case MODDEVEMPTYDIR: /* non-reconfiguring /dev emptydir */ 2585e37c6c37Scth error = modctl_devemptydir((const char *)a1, (size_t)a2, 2586e37c6c37Scth (int *)a3); 2587e37c6c37Scth break; 2588e37c6c37Scth 2589facf4a8dSllai1 case MODDEVNAME: 2590facf4a8dSllai1 error = modctl_moddevname((int)a1, a2, a3); 2591facf4a8dSllai1 break; 2592facf4a8dSllai1 259325e8c5aaSvikram case MODRETIRE: /* retire device named by physpath a1 */ 259425e8c5aaSvikram error = modctl_retire((char *)a1, (char *)a2, (size_t)a3); 259525e8c5aaSvikram break; 259625e8c5aaSvikram 259725e8c5aaSvikram case MODISRETIRED: /* check if a device is retired. */ 259825e8c5aaSvikram error = modctl_is_retired((char *)a1, (int *)a2); 259925e8c5aaSvikram break; 260025e8c5aaSvikram 260125e8c5aaSvikram case MODUNRETIRE: /* unretire device named by physpath a1 */ 260225e8c5aaSvikram error = modctl_unretire((char *)a1); 260325e8c5aaSvikram break; 260425e8c5aaSvikram 260526947304SEvan Yan case MODHPOPS: /* hotplug operations */ 260626947304SEvan Yan /* device named by physpath a2 and Connection name a3 */ 260726947304SEvan Yan error = modctl_hp((int)a1, (char *)a2, (char *)a3, a4, a5); 260826947304SEvan Yan break; 260926947304SEvan Yan 26107c478bd9Sstevel@tonic-gate default: 26117c478bd9Sstevel@tonic-gate error = EINVAL; 26127c478bd9Sstevel@tonic-gate break; 26137c478bd9Sstevel@tonic-gate } 26147c478bd9Sstevel@tonic-gate 26157c478bd9Sstevel@tonic-gate return (error ? set_errno(error) : 0); 26167c478bd9Sstevel@tonic-gate } 26177c478bd9Sstevel@tonic-gate 26187c478bd9Sstevel@tonic-gate /* 26197c478bd9Sstevel@tonic-gate * Calls to kobj_load_module()() are handled off to this routine in a 26207c478bd9Sstevel@tonic-gate * separate thread. 26217c478bd9Sstevel@tonic-gate */ 26227c478bd9Sstevel@tonic-gate static void 26237c478bd9Sstevel@tonic-gate modload_thread(struct loadmt *ltp) 26247c478bd9Sstevel@tonic-gate { 26257c478bd9Sstevel@tonic-gate /* load the module and signal the creator of this thread */ 26267c478bd9Sstevel@tonic-gate kmutex_t cpr_lk; 26277c478bd9Sstevel@tonic-gate callb_cpr_t cpr_i; 26287c478bd9Sstevel@tonic-gate 26297c478bd9Sstevel@tonic-gate mutex_init(&cpr_lk, NULL, MUTEX_DEFAULT, NULL); 26307c478bd9Sstevel@tonic-gate CALLB_CPR_INIT(&cpr_i, &cpr_lk, callb_generic_cpr, "modload"); 26317c478bd9Sstevel@tonic-gate /* borrow the devi lock from thread which invoked us */ 26327c478bd9Sstevel@tonic-gate pm_borrow_lock(ltp->owner); 26337c478bd9Sstevel@tonic-gate ltp->retval = kobj_load_module(ltp->mp, ltp->usepath); 26347c478bd9Sstevel@tonic-gate pm_return_lock(); 26357c478bd9Sstevel@tonic-gate sema_v(<p->sema); 26367c478bd9Sstevel@tonic-gate mutex_enter(&cpr_lk); 26377c478bd9Sstevel@tonic-gate CALLB_CPR_EXIT(&cpr_i); 26387c478bd9Sstevel@tonic-gate mutex_destroy(&cpr_lk); 26397c478bd9Sstevel@tonic-gate thread_exit(); 26407c478bd9Sstevel@tonic-gate } 26417c478bd9Sstevel@tonic-gate 26427c478bd9Sstevel@tonic-gate /* 26437c478bd9Sstevel@tonic-gate * load a module, adding a reference if caller specifies rmodp. If rmodp 26447c478bd9Sstevel@tonic-gate * is specified then an errno is returned, otherwise a module index is 26457c478bd9Sstevel@tonic-gate * returned (-1 on error). 26467c478bd9Sstevel@tonic-gate */ 26477c478bd9Sstevel@tonic-gate static int 264843d5cd3dSjohnlev modrload(const char *subdir, const char *filename, struct modctl **rmodp) 26497c478bd9Sstevel@tonic-gate { 26507c478bd9Sstevel@tonic-gate struct modctl *modp; 26517c478bd9Sstevel@tonic-gate size_t size; 26527c478bd9Sstevel@tonic-gate char *fullname; 26537c478bd9Sstevel@tonic-gate int retval = EINVAL; 26547c478bd9Sstevel@tonic-gate int id = -1; 26557c478bd9Sstevel@tonic-gate 26567c478bd9Sstevel@tonic-gate if (rmodp) 26577c478bd9Sstevel@tonic-gate *rmodp = NULL; /* avoid garbage */ 26587c478bd9Sstevel@tonic-gate 26597c478bd9Sstevel@tonic-gate if (subdir != NULL) { 26607c478bd9Sstevel@tonic-gate /* 26617c478bd9Sstevel@tonic-gate * refuse / in filename to prevent "../" escapes. 26627c478bd9Sstevel@tonic-gate */ 26637c478bd9Sstevel@tonic-gate if (strchr(filename, '/') != NULL) 26647c478bd9Sstevel@tonic-gate return (rmodp ? retval : id); 26657c478bd9Sstevel@tonic-gate 26667c478bd9Sstevel@tonic-gate /* 26677c478bd9Sstevel@tonic-gate * allocate enough space for <subdir>/<filename><NULL> 26687c478bd9Sstevel@tonic-gate */ 26697c478bd9Sstevel@tonic-gate size = strlen(subdir) + strlen(filename) + 2; 26707c478bd9Sstevel@tonic-gate fullname = kmem_zalloc(size, KM_SLEEP); 26717c478bd9Sstevel@tonic-gate (void) sprintf(fullname, "%s/%s", subdir, filename); 26727c478bd9Sstevel@tonic-gate } else { 267343d5cd3dSjohnlev fullname = (char *)filename; 26747c478bd9Sstevel@tonic-gate } 26757c478bd9Sstevel@tonic-gate 267620c794b3Sgavinm modp = mod_hold_installed_mod(fullname, 1, 0, &retval); 26777c478bd9Sstevel@tonic-gate if (modp != NULL) { 26787c478bd9Sstevel@tonic-gate id = modp->mod_id; 26797c478bd9Sstevel@tonic-gate if (rmodp) { 26807c478bd9Sstevel@tonic-gate /* add mod_ref and return *rmodp */ 26817c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 26827c478bd9Sstevel@tonic-gate modp->mod_ref++; 26837c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 26847c478bd9Sstevel@tonic-gate *rmodp = modp; 26857c478bd9Sstevel@tonic-gate } 26867c478bd9Sstevel@tonic-gate mod_release_mod(modp); 26877c478bd9Sstevel@tonic-gate CPU_STATS_ADDQ(CPU, sys, modload, 1); 26887c478bd9Sstevel@tonic-gate } 26897c478bd9Sstevel@tonic-gate 26907c478bd9Sstevel@tonic-gate done: if (subdir != NULL) 26917c478bd9Sstevel@tonic-gate kmem_free(fullname, size); 26927c478bd9Sstevel@tonic-gate return (rmodp ? retval : id); 26937c478bd9Sstevel@tonic-gate } 26947c478bd9Sstevel@tonic-gate 26957c478bd9Sstevel@tonic-gate /* 26967c478bd9Sstevel@tonic-gate * This is the primary kernel interface to load a module. It loads and 26977c478bd9Sstevel@tonic-gate * installs the named module. It does not hold mod_ref of the module, so 26987c478bd9Sstevel@tonic-gate * a module unload attempt can occur at any time - it is up to the 26997c478bd9Sstevel@tonic-gate * _fini/mod_remove implementation to determine if unload will succeed. 27007c478bd9Sstevel@tonic-gate */ 27017c478bd9Sstevel@tonic-gate int 270243d5cd3dSjohnlev modload(const char *subdir, const char *filename) 27037c478bd9Sstevel@tonic-gate { 27047c478bd9Sstevel@tonic-gate return (modrload(subdir, filename, NULL)); 27057c478bd9Sstevel@tonic-gate } 27067c478bd9Sstevel@tonic-gate 27077c478bd9Sstevel@tonic-gate /* 27087aec1d6eScindi * Load a module using a series of qualified names from most specific to least 27097aec1d6eScindi * specific, e.g. for subdir "foo", p1 "bar", p2 "baz", we might try: 271020c794b3Sgavinm * Value returned in *chosen 271120c794b3Sgavinm * foo/bar.baz.1.2.3 3 271220c794b3Sgavinm * foo/bar.baz.1.2 2 271320c794b3Sgavinm * foo/bar.baz.1 1 271420c794b3Sgavinm * foo/bar.baz 0 27157aec1d6eScindi * 271620c794b3Sgavinm * Return the module ID on success; -1 if no module was loaded. On success 271720c794b3Sgavinm * and if 'chosen' is not NULL we also return the number of suffices that 271820c794b3Sgavinm * were in the module we chose to load. 27197aec1d6eScindi */ 27207aec1d6eScindi int 27217aec1d6eScindi modload_qualified(const char *subdir, const char *p1, 272220c794b3Sgavinm const char *p2, const char *delim, uint_t suffv[], int suffc, int *chosen) 27237aec1d6eScindi { 27247aec1d6eScindi char path[MOD_MAXPATH]; 27257aec1d6eScindi size_t n, resid = sizeof (path); 27267aec1d6eScindi char *p = path; 27277aec1d6eScindi 27287aec1d6eScindi char **dotv; 27297aec1d6eScindi int i, rc, id; 27307aec1d6eScindi modctl_t *mp; 27317aec1d6eScindi 27327aec1d6eScindi if (p2 != NULL) 27337aec1d6eScindi n = snprintf(p, resid, "%s/%s%s%s", subdir, p1, delim, p2); 27347aec1d6eScindi else 27357aec1d6eScindi n = snprintf(p, resid, "%s/%s", subdir, p1); 27367aec1d6eScindi 27377aec1d6eScindi if (n >= resid) 27387aec1d6eScindi return (-1); 27397aec1d6eScindi 27407aec1d6eScindi p += n; 27417aec1d6eScindi resid -= n; 27427aec1d6eScindi dotv = kmem_alloc(sizeof (char *) * (suffc + 1), KM_SLEEP); 27437aec1d6eScindi 27447aec1d6eScindi for (i = 0; i < suffc; i++) { 27457aec1d6eScindi dotv[i] = p; 27467aec1d6eScindi n = snprintf(p, resid, "%s%u", delim, suffv[i]); 27477aec1d6eScindi 27487aec1d6eScindi if (n >= resid) { 27497aec1d6eScindi kmem_free(dotv, sizeof (char *) * (suffc + 1)); 27507aec1d6eScindi return (-1); 27517aec1d6eScindi } 27527aec1d6eScindi 27537aec1d6eScindi p += n; 27547aec1d6eScindi resid -= n; 27557aec1d6eScindi } 27567aec1d6eScindi 27577aec1d6eScindi dotv[suffc] = p; 27587aec1d6eScindi 27597aec1d6eScindi for (i = suffc; i >= 0; i--) { 27607aec1d6eScindi dotv[i][0] = '\0'; 276120c794b3Sgavinm mp = mod_hold_installed_mod(path, 1, 1, &rc); 27627aec1d6eScindi 27637aec1d6eScindi if (mp != NULL) { 27647aec1d6eScindi kmem_free(dotv, sizeof (char *) * (suffc + 1)); 27657aec1d6eScindi id = mp->mod_id; 27667aec1d6eScindi mod_release_mod(mp); 276720c794b3Sgavinm if (chosen != NULL) 276820c794b3Sgavinm *chosen = i; 27697aec1d6eScindi return (id); 27707aec1d6eScindi } 27717aec1d6eScindi } 27727aec1d6eScindi 27737aec1d6eScindi kmem_free(dotv, sizeof (char *) * (suffc + 1)); 27747aec1d6eScindi return (-1); 27757aec1d6eScindi } 27767aec1d6eScindi 27777aec1d6eScindi /* 27787c478bd9Sstevel@tonic-gate * Load a module. 27797c478bd9Sstevel@tonic-gate */ 27807c478bd9Sstevel@tonic-gate int 278143d5cd3dSjohnlev modloadonly(const char *subdir, const char *filename) 27827c478bd9Sstevel@tonic-gate { 27837c478bd9Sstevel@tonic-gate struct modctl *modp; 27847c478bd9Sstevel@tonic-gate char *fullname; 27857c478bd9Sstevel@tonic-gate size_t size; 27867c478bd9Sstevel@tonic-gate int id, retval; 27877c478bd9Sstevel@tonic-gate 27887c478bd9Sstevel@tonic-gate if (subdir != NULL) { 27897c478bd9Sstevel@tonic-gate /* 27907c478bd9Sstevel@tonic-gate * allocate enough space for <subdir>/<filename><NULL> 27917c478bd9Sstevel@tonic-gate */ 27927c478bd9Sstevel@tonic-gate size = strlen(subdir) + strlen(filename) + 2; 27937c478bd9Sstevel@tonic-gate fullname = kmem_zalloc(size, KM_SLEEP); 27947c478bd9Sstevel@tonic-gate (void) sprintf(fullname, "%s/%s", subdir, filename); 27957c478bd9Sstevel@tonic-gate } else { 279643d5cd3dSjohnlev fullname = (char *)filename; 27977c478bd9Sstevel@tonic-gate } 27987c478bd9Sstevel@tonic-gate 27997c478bd9Sstevel@tonic-gate modp = mod_hold_loaded_mod(NULL, fullname, &retval); 28007c478bd9Sstevel@tonic-gate if (modp) { 28017c478bd9Sstevel@tonic-gate id = modp->mod_id; 28027c478bd9Sstevel@tonic-gate mod_release_mod(modp); 28037c478bd9Sstevel@tonic-gate } 28047c478bd9Sstevel@tonic-gate 28057c478bd9Sstevel@tonic-gate if (subdir != NULL) 28067c478bd9Sstevel@tonic-gate kmem_free(fullname, size); 28077c478bd9Sstevel@tonic-gate 28087c478bd9Sstevel@tonic-gate if (retval == 0) 28097c478bd9Sstevel@tonic-gate return (id); 28107c478bd9Sstevel@tonic-gate return (-1); 28117c478bd9Sstevel@tonic-gate } 28127c478bd9Sstevel@tonic-gate 28137c478bd9Sstevel@tonic-gate /* 28147c478bd9Sstevel@tonic-gate * Try to uninstall and unload a module, removing a reference if caller 28157c478bd9Sstevel@tonic-gate * specifies rmodp. 28167c478bd9Sstevel@tonic-gate */ 28177c478bd9Sstevel@tonic-gate static int 28187c478bd9Sstevel@tonic-gate modunrload(modid_t id, struct modctl **rmodp, int unload) 28197c478bd9Sstevel@tonic-gate { 28207c478bd9Sstevel@tonic-gate struct modctl *modp; 28217c478bd9Sstevel@tonic-gate int retval; 28227c478bd9Sstevel@tonic-gate 28237c478bd9Sstevel@tonic-gate if (rmodp) 28247c478bd9Sstevel@tonic-gate *rmodp = NULL; /* avoid garbage */ 28257c478bd9Sstevel@tonic-gate 28267c478bd9Sstevel@tonic-gate if ((modp = mod_hold_by_id((modid_t)id)) == NULL) 28277c478bd9Sstevel@tonic-gate return (EINVAL); 28287c478bd9Sstevel@tonic-gate 28297c478bd9Sstevel@tonic-gate if (rmodp) { 28307c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 28317c478bd9Sstevel@tonic-gate modp->mod_ref--; 28327c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 28337c478bd9Sstevel@tonic-gate *rmodp = modp; 28347c478bd9Sstevel@tonic-gate } 28357c478bd9Sstevel@tonic-gate 28367c478bd9Sstevel@tonic-gate if (unload) { 28377c478bd9Sstevel@tonic-gate retval = moduninstall(modp); 28387c478bd9Sstevel@tonic-gate if (retval == 0) { 28397c478bd9Sstevel@tonic-gate mod_unload(modp); 28407c478bd9Sstevel@tonic-gate CPU_STATS_ADDQ(CPU, sys, modunload, 1); 28417c478bd9Sstevel@tonic-gate } else if (retval == EALREADY) 28427c478bd9Sstevel@tonic-gate retval = 0; /* already unloaded, not an error */ 28437c478bd9Sstevel@tonic-gate } else 28447c478bd9Sstevel@tonic-gate retval = 0; 28457c478bd9Sstevel@tonic-gate 28467c478bd9Sstevel@tonic-gate mod_release_mod(modp); 28477c478bd9Sstevel@tonic-gate return (retval); 28487c478bd9Sstevel@tonic-gate } 28497c478bd9Sstevel@tonic-gate 28507c478bd9Sstevel@tonic-gate /* 28517c478bd9Sstevel@tonic-gate * Uninstall and unload a module. 28527c478bd9Sstevel@tonic-gate */ 28537c478bd9Sstevel@tonic-gate int 28547c478bd9Sstevel@tonic-gate modunload(modid_t id) 28557c478bd9Sstevel@tonic-gate { 2856a7aa4df7Scth int retval; 2857a7aa4df7Scth 2858a7aa4df7Scth /* synchronize with any active modunload_disable() */ 2859a7aa4df7Scth modunload_begin(); 2860a7aa4df7Scth if (ddi_root_node()) 2861a7aa4df7Scth (void) devfs_clean(ddi_root_node(), NULL, 0); 2862a7aa4df7Scth retval = modunrload(id, NULL, 1); 2863a7aa4df7Scth modunload_end(); 2864a7aa4df7Scth return (retval); 28657c478bd9Sstevel@tonic-gate } 28667c478bd9Sstevel@tonic-gate 28677c478bd9Sstevel@tonic-gate /* 28687c478bd9Sstevel@tonic-gate * Return status of a loaded module. 28697c478bd9Sstevel@tonic-gate */ 28707c478bd9Sstevel@tonic-gate static int 28717c478bd9Sstevel@tonic-gate modinfo(modid_t id, struct modinfo *modinfop) 28727c478bd9Sstevel@tonic-gate { 28737c478bd9Sstevel@tonic-gate struct modctl *modp; 28747c478bd9Sstevel@tonic-gate modid_t mid; 28757c478bd9Sstevel@tonic-gate int i; 28767c478bd9Sstevel@tonic-gate 28777c478bd9Sstevel@tonic-gate mid = modinfop->mi_id; 28787c478bd9Sstevel@tonic-gate if (modinfop->mi_info & MI_INFO_ALL) { 28797c478bd9Sstevel@tonic-gate while ((modp = mod_hold_next_by_id(mid++)) != NULL) { 28807c478bd9Sstevel@tonic-gate if ((modinfop->mi_info & MI_INFO_CNT) || 28817c478bd9Sstevel@tonic-gate modp->mod_installed) 28827c478bd9Sstevel@tonic-gate break; 28837c478bd9Sstevel@tonic-gate mod_release_mod(modp); 28847c478bd9Sstevel@tonic-gate } 28857c478bd9Sstevel@tonic-gate if (modp == NULL) 28867c478bd9Sstevel@tonic-gate return (EINVAL); 28877c478bd9Sstevel@tonic-gate } else { 28887c478bd9Sstevel@tonic-gate modp = mod_hold_by_id(id); 28897c478bd9Sstevel@tonic-gate if (modp == NULL) 28907c478bd9Sstevel@tonic-gate return (EINVAL); 28917c478bd9Sstevel@tonic-gate if (!(modinfop->mi_info & MI_INFO_CNT) && 28927c478bd9Sstevel@tonic-gate (modp->mod_installed == 0)) { 28937c478bd9Sstevel@tonic-gate mod_release_mod(modp); 28947c478bd9Sstevel@tonic-gate return (EINVAL); 28957c478bd9Sstevel@tonic-gate } 28967c478bd9Sstevel@tonic-gate } 28977c478bd9Sstevel@tonic-gate 28987c478bd9Sstevel@tonic-gate modinfop->mi_rev = 0; 28997c478bd9Sstevel@tonic-gate modinfop->mi_state = 0; 29007c478bd9Sstevel@tonic-gate for (i = 0; i < MODMAXLINK; i++) { 29017c478bd9Sstevel@tonic-gate modinfop->mi_msinfo[i].msi_p0 = -1; 29027c478bd9Sstevel@tonic-gate modinfop->mi_msinfo[i].msi_linkinfo[0] = 0; 29037c478bd9Sstevel@tonic-gate } 29047c478bd9Sstevel@tonic-gate if (modp->mod_loaded) { 29057c478bd9Sstevel@tonic-gate modinfop->mi_state = MI_LOADED; 29067c478bd9Sstevel@tonic-gate kobj_getmodinfo(modp->mod_mp, modinfop); 29077c478bd9Sstevel@tonic-gate } 29087c478bd9Sstevel@tonic-gate if (modp->mod_installed) { 29097c478bd9Sstevel@tonic-gate modinfop->mi_state |= MI_INSTALLED; 29107c478bd9Sstevel@tonic-gate 29117c478bd9Sstevel@tonic-gate (void) mod_getinfo(modp, modinfop); 29127c478bd9Sstevel@tonic-gate } 29137c478bd9Sstevel@tonic-gate 29147c478bd9Sstevel@tonic-gate modinfop->mi_id = modp->mod_id; 29157c478bd9Sstevel@tonic-gate modinfop->mi_loadcnt = modp->mod_loadcnt; 29167c478bd9Sstevel@tonic-gate (void) strcpy(modinfop->mi_name, modp->mod_modname); 29177c478bd9Sstevel@tonic-gate 29187c478bd9Sstevel@tonic-gate mod_release_mod(modp); 29197c478bd9Sstevel@tonic-gate return (0); 29207c478bd9Sstevel@tonic-gate } 29217c478bd9Sstevel@tonic-gate 29227c478bd9Sstevel@tonic-gate static char mod_stub_err[] = "mod_hold_stub: Couldn't load stub module %s"; 29237c478bd9Sstevel@tonic-gate static char no_err[] = "No error function for weak stub %s"; 29247c478bd9Sstevel@tonic-gate 29257c478bd9Sstevel@tonic-gate /* 29267c478bd9Sstevel@tonic-gate * used by the stubs themselves to load and hold a module. 29277c478bd9Sstevel@tonic-gate * Returns 0 if the module is successfully held; 29287c478bd9Sstevel@tonic-gate * the stub needs to call mod_release_stub(). 29297c478bd9Sstevel@tonic-gate * -1 if the stub should just call the err_fcn. 29307c478bd9Sstevel@tonic-gate * Note that this code is stretched out so that we avoid subroutine calls 29317c478bd9Sstevel@tonic-gate * and optimize for the most likely case. That is, the case where the 29327c478bd9Sstevel@tonic-gate * module is loaded and installed and not held. In that case we just inc 29337c478bd9Sstevel@tonic-gate * the mod_ref count and continue. 29347c478bd9Sstevel@tonic-gate */ 29357c478bd9Sstevel@tonic-gate int 29367c478bd9Sstevel@tonic-gate mod_hold_stub(struct mod_stub_info *stub) 29377c478bd9Sstevel@tonic-gate { 29387c478bd9Sstevel@tonic-gate struct modctl *mp; 29397c478bd9Sstevel@tonic-gate struct mod_modinfo *mip; 29407c478bd9Sstevel@tonic-gate 29417c478bd9Sstevel@tonic-gate mip = stub->mods_modinfo; 29427c478bd9Sstevel@tonic-gate 29437c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 29447c478bd9Sstevel@tonic-gate 29457c478bd9Sstevel@tonic-gate /* we do mod_hold_by_modctl inline for speed */ 29467c478bd9Sstevel@tonic-gate 29477c478bd9Sstevel@tonic-gate mod_check_again: 29487c478bd9Sstevel@tonic-gate if ((mp = mip->mp) != NULL) { 29497c478bd9Sstevel@tonic-gate if (mp->mod_busy == 0) { 29507c478bd9Sstevel@tonic-gate if (mp->mod_installed) { 29517c478bd9Sstevel@tonic-gate /* increment the reference count */ 29527c478bd9Sstevel@tonic-gate mp->mod_ref++; 29537c478bd9Sstevel@tonic-gate ASSERT(mp->mod_ref && mp->mod_installed); 29547c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 29557c478bd9Sstevel@tonic-gate return (0); 29567c478bd9Sstevel@tonic-gate } else { 29577c478bd9Sstevel@tonic-gate mp->mod_busy = 1; 29587c478bd9Sstevel@tonic-gate mp->mod_inprogress_thread = 29597c478bd9Sstevel@tonic-gate (curthread == NULL ? 29607c478bd9Sstevel@tonic-gate (kthread_id_t)-1 : curthread); 29617c478bd9Sstevel@tonic-gate } 29627c478bd9Sstevel@tonic-gate } else { 29637c478bd9Sstevel@tonic-gate /* 29647c478bd9Sstevel@tonic-gate * wait one time and then go see if someone 29657c478bd9Sstevel@tonic-gate * else has resolved the stub (set mip->mp). 29667c478bd9Sstevel@tonic-gate */ 29677c478bd9Sstevel@tonic-gate if (mod_hold_by_modctl(mp, 29687c478bd9Sstevel@tonic-gate MOD_WAIT_ONCE | MOD_LOCK_HELD)) 29697c478bd9Sstevel@tonic-gate goto mod_check_again; 29707c478bd9Sstevel@tonic-gate 29717c478bd9Sstevel@tonic-gate /* 29727c478bd9Sstevel@tonic-gate * what we have now may have been unloaded!, in 29737c478bd9Sstevel@tonic-gate * that case, mip->mp will be NULL, we'll hit this 29747c478bd9Sstevel@tonic-gate * module and load again.. 29757c478bd9Sstevel@tonic-gate */ 29767c478bd9Sstevel@tonic-gate cmn_err(CE_PANIC, "mod_hold_stub should have blocked"); 29777c478bd9Sstevel@tonic-gate } 29787c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 29797c478bd9Sstevel@tonic-gate } else { 29807c478bd9Sstevel@tonic-gate /* first time we've hit this module */ 29817c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 29827c478bd9Sstevel@tonic-gate mp = mod_hold_by_name(mip->modm_module_name); 29837c478bd9Sstevel@tonic-gate mip->mp = mp; 29847c478bd9Sstevel@tonic-gate } 29857c478bd9Sstevel@tonic-gate 29867c478bd9Sstevel@tonic-gate /* 29877c478bd9Sstevel@tonic-gate * If we are here, it means that the following conditions 29887c478bd9Sstevel@tonic-gate * are satisfied. 29897c478bd9Sstevel@tonic-gate * 29907c478bd9Sstevel@tonic-gate * mip->mp != NULL 29917c478bd9Sstevel@tonic-gate * this thread has set the mp->mod_busy = 1 29927c478bd9Sstevel@tonic-gate * mp->mod_installed = 0 29937c478bd9Sstevel@tonic-gate * 29947c478bd9Sstevel@tonic-gate */ 29957c478bd9Sstevel@tonic-gate ASSERT(mp != NULL); 29967c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy == 1); 29977c478bd9Sstevel@tonic-gate 29987c478bd9Sstevel@tonic-gate if (mp->mod_installed == 0) { 29997c478bd9Sstevel@tonic-gate /* Module not loaded, if weak stub don't load it */ 30007c478bd9Sstevel@tonic-gate if (stub->mods_flag & MODS_WEAK) { 30017c478bd9Sstevel@tonic-gate if (stub->mods_errfcn == NULL) { 30027c478bd9Sstevel@tonic-gate mod_release_mod(mp); 30037c478bd9Sstevel@tonic-gate cmn_err(CE_PANIC, no_err, 30047c478bd9Sstevel@tonic-gate mip->modm_module_name); 30057c478bd9Sstevel@tonic-gate } 30067c478bd9Sstevel@tonic-gate } else { 30077c478bd9Sstevel@tonic-gate /* Not a weak stub so load the module */ 30087c478bd9Sstevel@tonic-gate 30097c478bd9Sstevel@tonic-gate if (mod_load(mp, 1) != 0 || modinstall(mp) != 0) { 30107c478bd9Sstevel@tonic-gate /* 30117c478bd9Sstevel@tonic-gate * If mod_load() was successful 30127c478bd9Sstevel@tonic-gate * and modinstall() failed, then 30137c478bd9Sstevel@tonic-gate * unload the module. 30147c478bd9Sstevel@tonic-gate */ 30157c478bd9Sstevel@tonic-gate if (mp->mod_loaded) 30167c478bd9Sstevel@tonic-gate mod_unload(mp); 30177c478bd9Sstevel@tonic-gate 30187c478bd9Sstevel@tonic-gate mod_release_mod(mp); 30197c478bd9Sstevel@tonic-gate if (stub->mods_errfcn == NULL) { 30207c478bd9Sstevel@tonic-gate cmn_err(CE_PANIC, mod_stub_err, 30217c478bd9Sstevel@tonic-gate mip->modm_module_name); 30227c478bd9Sstevel@tonic-gate } else { 30237c478bd9Sstevel@tonic-gate return (-1); 30247c478bd9Sstevel@tonic-gate } 30257c478bd9Sstevel@tonic-gate } 30267c478bd9Sstevel@tonic-gate } 30277c478bd9Sstevel@tonic-gate } 30287c478bd9Sstevel@tonic-gate 30297c478bd9Sstevel@tonic-gate /* 30307c478bd9Sstevel@tonic-gate * At this point module is held and loaded. Release 30317c478bd9Sstevel@tonic-gate * the mod_busy and mod_inprogress_thread before 30327c478bd9Sstevel@tonic-gate * returning. We actually call mod_release() here so 30337c478bd9Sstevel@tonic-gate * that if another stub wants to access this module, 30347c478bd9Sstevel@tonic-gate * it can do so. mod_ref is incremented before mod_release() 30357c478bd9Sstevel@tonic-gate * is called to prevent someone else from snatching the 30367c478bd9Sstevel@tonic-gate * module from this thread. 30377c478bd9Sstevel@tonic-gate */ 30387c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 30397c478bd9Sstevel@tonic-gate mp->mod_ref++; 30407c478bd9Sstevel@tonic-gate ASSERT(mp->mod_ref && 30417c478bd9Sstevel@tonic-gate (mp->mod_loaded || (stub->mods_flag & MODS_WEAK))); 30427c478bd9Sstevel@tonic-gate mod_release(mp); 30437c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 30447c478bd9Sstevel@tonic-gate return (0); 30457c478bd9Sstevel@tonic-gate } 30467c478bd9Sstevel@tonic-gate 30477c478bd9Sstevel@tonic-gate void 30487c478bd9Sstevel@tonic-gate mod_release_stub(struct mod_stub_info *stub) 30497c478bd9Sstevel@tonic-gate { 30507c478bd9Sstevel@tonic-gate struct modctl *mp = stub->mods_modinfo->mp; 30517c478bd9Sstevel@tonic-gate 30527c478bd9Sstevel@tonic-gate /* inline mod_release_mod */ 30537c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 30547c478bd9Sstevel@tonic-gate ASSERT(mp->mod_ref && 30557c478bd9Sstevel@tonic-gate (mp->mod_loaded || (stub->mods_flag & MODS_WEAK))); 30567c478bd9Sstevel@tonic-gate mp->mod_ref--; 30577c478bd9Sstevel@tonic-gate if (mp->mod_want) { 30587c478bd9Sstevel@tonic-gate mp->mod_want = 0; 30597c478bd9Sstevel@tonic-gate cv_broadcast(&mod_cv); 30607c478bd9Sstevel@tonic-gate } 30617c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 30627c478bd9Sstevel@tonic-gate } 30637c478bd9Sstevel@tonic-gate 30647c478bd9Sstevel@tonic-gate static struct modctl * 30657c478bd9Sstevel@tonic-gate mod_hold_loaded_mod(struct modctl *dep, char *filename, int *status) 30667c478bd9Sstevel@tonic-gate { 30677c478bd9Sstevel@tonic-gate struct modctl *modp; 30687c478bd9Sstevel@tonic-gate int retval; 30697c478bd9Sstevel@tonic-gate 30707c478bd9Sstevel@tonic-gate /* 30717c478bd9Sstevel@tonic-gate * Hold the module. 30727c478bd9Sstevel@tonic-gate */ 30737c478bd9Sstevel@tonic-gate modp = mod_hold_by_name_requisite(dep, filename); 30747c478bd9Sstevel@tonic-gate if (modp) { 30757c478bd9Sstevel@tonic-gate retval = mod_load(modp, 1); 30767c478bd9Sstevel@tonic-gate if (retval != 0) { 30777c478bd9Sstevel@tonic-gate mod_release_mod(modp); 30787c478bd9Sstevel@tonic-gate modp = NULL; 30797c478bd9Sstevel@tonic-gate } 30807c478bd9Sstevel@tonic-gate *status = retval; 30817c478bd9Sstevel@tonic-gate } else { 30827c478bd9Sstevel@tonic-gate *status = ENOSPC; 30837c478bd9Sstevel@tonic-gate } 30847c478bd9Sstevel@tonic-gate 30857c478bd9Sstevel@tonic-gate /* 30867c478bd9Sstevel@tonic-gate * if dep is not NULL, clear the module dependency information. 30877c478bd9Sstevel@tonic-gate * This information is set in mod_hold_by_name_common(). 30887c478bd9Sstevel@tonic-gate */ 30897c478bd9Sstevel@tonic-gate if (dep != NULL && dep->mod_requisite_loading != NULL) { 30907c478bd9Sstevel@tonic-gate ASSERT(dep->mod_busy); 30917c478bd9Sstevel@tonic-gate dep->mod_requisite_loading = NULL; 30927c478bd9Sstevel@tonic-gate } 30937c478bd9Sstevel@tonic-gate 30947c478bd9Sstevel@tonic-gate return (modp); 30957c478bd9Sstevel@tonic-gate } 30967c478bd9Sstevel@tonic-gate 30977c478bd9Sstevel@tonic-gate /* 30987c478bd9Sstevel@tonic-gate * hold, load, and install the named module 30997c478bd9Sstevel@tonic-gate */ 31007c478bd9Sstevel@tonic-gate static struct modctl * 310120c794b3Sgavinm mod_hold_installed_mod(char *name, int usepath, int forcecheck, int *r) 31027c478bd9Sstevel@tonic-gate { 31037c478bd9Sstevel@tonic-gate struct modctl *modp; 31047c478bd9Sstevel@tonic-gate int retval; 31052bac1547Scth 31062bac1547Scth /* 31072bac1547Scth * Verify that that module in question actually exists on disk 31082bac1547Scth * before allocation of module structure by mod_hold_by_name. 31092bac1547Scth */ 311020c794b3Sgavinm if (modrootloaded && swaploaded || forcecheck) { 31115c311300Scth if (!kobj_path_exists(name, usepath)) { 31122bac1547Scth *r = ENOENT; 31132bac1547Scth return (NULL); 31142bac1547Scth } 31152bac1547Scth } 31167c478bd9Sstevel@tonic-gate 31177c478bd9Sstevel@tonic-gate /* 31187c478bd9Sstevel@tonic-gate * Hold the module. 31197c478bd9Sstevel@tonic-gate */ 31207c478bd9Sstevel@tonic-gate modp = mod_hold_by_name(name); 31217c478bd9Sstevel@tonic-gate if (modp) { 31227c478bd9Sstevel@tonic-gate retval = mod_load(modp, usepath); 31237c478bd9Sstevel@tonic-gate if (retval != 0) { 31247c478bd9Sstevel@tonic-gate mod_release_mod(modp); 31257c478bd9Sstevel@tonic-gate modp = NULL; 31267c478bd9Sstevel@tonic-gate *r = retval; 31277c478bd9Sstevel@tonic-gate } else { 31287c478bd9Sstevel@tonic-gate if ((*r = modinstall(modp)) != 0) { 31297c478bd9Sstevel@tonic-gate /* 31307c478bd9Sstevel@tonic-gate * We loaded it, but failed to _init() it. 31317c478bd9Sstevel@tonic-gate * Be kind to developers -- force it 31327c478bd9Sstevel@tonic-gate * out of memory now so that the next 31337c478bd9Sstevel@tonic-gate * attempt to use the module will cause 31347c478bd9Sstevel@tonic-gate * a reload. See 1093793. 31357c478bd9Sstevel@tonic-gate */ 31367c478bd9Sstevel@tonic-gate mod_unload(modp); 31377c478bd9Sstevel@tonic-gate mod_release_mod(modp); 31387c478bd9Sstevel@tonic-gate modp = NULL; 31397c478bd9Sstevel@tonic-gate } 31407c478bd9Sstevel@tonic-gate } 31417c478bd9Sstevel@tonic-gate } else { 31427c478bd9Sstevel@tonic-gate *r = ENOSPC; 31437c478bd9Sstevel@tonic-gate } 31447c478bd9Sstevel@tonic-gate return (modp); 31457c478bd9Sstevel@tonic-gate } 31467c478bd9Sstevel@tonic-gate 31477c478bd9Sstevel@tonic-gate static char mod_excl_msg[] = 31487c478bd9Sstevel@tonic-gate "module %s(%s) is EXCLUDED and will not be loaded\n"; 31497c478bd9Sstevel@tonic-gate static char mod_init_msg[] = "loadmodule:%s(%s): _init() error %d\n"; 31507c478bd9Sstevel@tonic-gate 31517c478bd9Sstevel@tonic-gate /* 31527c478bd9Sstevel@tonic-gate * This routine is needed for dependencies. Users specify dependencies 31537c478bd9Sstevel@tonic-gate * by declaring a character array initialized to filenames of dependents. 31547c478bd9Sstevel@tonic-gate * So the code that handles dependents deals with filenames (and not 31557c478bd9Sstevel@tonic-gate * module names) because that's all it has. We load by filename and once 31567c478bd9Sstevel@tonic-gate * we've loaded a file we can get the module name. 31577c478bd9Sstevel@tonic-gate * Unfortunately there isn't a single unified filename/modulename namespace. 31587c478bd9Sstevel@tonic-gate * C'est la vie. 31597c478bd9Sstevel@tonic-gate * 31607c478bd9Sstevel@tonic-gate * We allow the name being looked up to be prepended by an optional 31617c478bd9Sstevel@tonic-gate * subdirectory e.g. we can lookup (NULL, "fs/ufs") or ("fs", "ufs") 31627c478bd9Sstevel@tonic-gate */ 31637c478bd9Sstevel@tonic-gate struct modctl * 31647c478bd9Sstevel@tonic-gate mod_find_by_filename(char *subdir, char *filename) 31657c478bd9Sstevel@tonic-gate { 31667c478bd9Sstevel@tonic-gate struct modctl *mp; 31677c478bd9Sstevel@tonic-gate size_t sublen; 31687c478bd9Sstevel@tonic-gate 31697c478bd9Sstevel@tonic-gate ASSERT(!MUTEX_HELD(&mod_lock)); 31707c478bd9Sstevel@tonic-gate if (subdir != NULL) 31717c478bd9Sstevel@tonic-gate sublen = strlen(subdir); 31727c478bd9Sstevel@tonic-gate else 31737c478bd9Sstevel@tonic-gate sublen = 0; 31747c478bd9Sstevel@tonic-gate 31757c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 31767c478bd9Sstevel@tonic-gate mp = &modules; 31777c478bd9Sstevel@tonic-gate do { 31787c478bd9Sstevel@tonic-gate if (sublen) { 31797c478bd9Sstevel@tonic-gate char *mod_filename = mp->mod_filename; 31807c478bd9Sstevel@tonic-gate 31817c478bd9Sstevel@tonic-gate if (strncmp(subdir, mod_filename, sublen) == 0 && 31827c478bd9Sstevel@tonic-gate mod_filename[sublen] == '/' && 31837c478bd9Sstevel@tonic-gate strcmp(filename, &mod_filename[sublen + 1]) == 0) { 31847c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 31857c478bd9Sstevel@tonic-gate return (mp); 31867c478bd9Sstevel@tonic-gate } 31877c478bd9Sstevel@tonic-gate } else if (strcmp(filename, mp->mod_filename) == 0) { 31887c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 31897c478bd9Sstevel@tonic-gate return (mp); 31907c478bd9Sstevel@tonic-gate } 31917c478bd9Sstevel@tonic-gate } while ((mp = mp->mod_next) != &modules); 31927c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 31937c478bd9Sstevel@tonic-gate return (NULL); 31947c478bd9Sstevel@tonic-gate } 31957c478bd9Sstevel@tonic-gate 31967c478bd9Sstevel@tonic-gate /* 31977c478bd9Sstevel@tonic-gate * Check for circular dependencies. This is called from do_dependents() 31987c478bd9Sstevel@tonic-gate * in kobj.c. If we are the thread already loading this module, then 31997c478bd9Sstevel@tonic-gate * we're trying to load a dependent that we're already loading which 32007c478bd9Sstevel@tonic-gate * means the user specified circular dependencies. 32017c478bd9Sstevel@tonic-gate */ 32027c478bd9Sstevel@tonic-gate static int 32037c478bd9Sstevel@tonic-gate mod_circdep(struct modctl *modp) 32047c478bd9Sstevel@tonic-gate { 32057c478bd9Sstevel@tonic-gate struct modctl *rmod; 32067c478bd9Sstevel@tonic-gate 32077c478bd9Sstevel@tonic-gate ASSERT(MUTEX_HELD(&mod_lock)); 32087c478bd9Sstevel@tonic-gate 32097c478bd9Sstevel@tonic-gate /* 32107c478bd9Sstevel@tonic-gate * Check the mod_inprogress_thread first. 32117c478bd9Sstevel@tonic-gate * mod_inprogress_thread is used in mod_hold_stub() 32127c478bd9Sstevel@tonic-gate * directly to improve performance. 32137c478bd9Sstevel@tonic-gate */ 32147c478bd9Sstevel@tonic-gate if (modp->mod_inprogress_thread == curthread) 32157c478bd9Sstevel@tonic-gate return (1); 32167c478bd9Sstevel@tonic-gate 32177c478bd9Sstevel@tonic-gate /* 32187c478bd9Sstevel@tonic-gate * Check the module circular dependencies. 32197c478bd9Sstevel@tonic-gate */ 32207c478bd9Sstevel@tonic-gate for (rmod = modp; rmod != NULL; rmod = rmod->mod_requisite_loading) { 32217c478bd9Sstevel@tonic-gate /* 32227c478bd9Sstevel@tonic-gate * Check if there is a module circular dependency. 32237c478bd9Sstevel@tonic-gate */ 32247c478bd9Sstevel@tonic-gate if (rmod->mod_requisite_loading == modp) 32257c478bd9Sstevel@tonic-gate return (1); 32267c478bd9Sstevel@tonic-gate } 32277c478bd9Sstevel@tonic-gate return (0); 32287c478bd9Sstevel@tonic-gate } 32297c478bd9Sstevel@tonic-gate 32307c478bd9Sstevel@tonic-gate static int 32317c478bd9Sstevel@tonic-gate mod_getinfo(struct modctl *modp, struct modinfo *modinfop) 32327c478bd9Sstevel@tonic-gate { 32337c478bd9Sstevel@tonic-gate int (*func)(struct modinfo *); 32347c478bd9Sstevel@tonic-gate int retval; 32357c478bd9Sstevel@tonic-gate 32367c478bd9Sstevel@tonic-gate ASSERT(modp->mod_busy); 32377c478bd9Sstevel@tonic-gate 32387c478bd9Sstevel@tonic-gate /* primary modules don't do getinfo */ 32397c478bd9Sstevel@tonic-gate if (modp->mod_prim) 32407c478bd9Sstevel@tonic-gate return (0); 32417c478bd9Sstevel@tonic-gate 32427c478bd9Sstevel@tonic-gate func = (int (*)(struct modinfo *))kobj_lookup(modp->mod_mp, "_info"); 32437c478bd9Sstevel@tonic-gate 32447c478bd9Sstevel@tonic-gate if (kobj_addrcheck(modp->mod_mp, (caddr_t)func)) { 32457c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "_info() not defined properly in %s", 32467c478bd9Sstevel@tonic-gate modp->mod_filename); 32477c478bd9Sstevel@tonic-gate /* 32487c478bd9Sstevel@tonic-gate * The semantics of mod_info(9F) are that 0 is failure 32497c478bd9Sstevel@tonic-gate * and non-zero is success. 32507c478bd9Sstevel@tonic-gate */ 32517c478bd9Sstevel@tonic-gate retval = 0; 32527c478bd9Sstevel@tonic-gate } else 32537c478bd9Sstevel@tonic-gate retval = (*func)(modinfop); /* call _info() function */ 32547c478bd9Sstevel@tonic-gate 32557c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_USERDEBUG) 32567c478bd9Sstevel@tonic-gate printf("Returned from _info, retval = %x\n", retval); 32577c478bd9Sstevel@tonic-gate 32587c478bd9Sstevel@tonic-gate return (retval); 32597c478bd9Sstevel@tonic-gate } 32607c478bd9Sstevel@tonic-gate 32617c478bd9Sstevel@tonic-gate static void 32627c478bd9Sstevel@tonic-gate modadd(struct modctl *mp) 32637c478bd9Sstevel@tonic-gate { 32647c478bd9Sstevel@tonic-gate ASSERT(MUTEX_HELD(&mod_lock)); 32657c478bd9Sstevel@tonic-gate 32667c478bd9Sstevel@tonic-gate mp->mod_id = last_module_id++; 32677c478bd9Sstevel@tonic-gate mp->mod_next = &modules; 32687c478bd9Sstevel@tonic-gate mp->mod_prev = modules.mod_prev; 32697c478bd9Sstevel@tonic-gate modules.mod_prev->mod_next = mp; 32707c478bd9Sstevel@tonic-gate modules.mod_prev = mp; 32717c478bd9Sstevel@tonic-gate } 32727c478bd9Sstevel@tonic-gate 32737c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 32747c478bd9Sstevel@tonic-gate static struct modctl * 32757aec1d6eScindi allocate_modp(const char *filename, const char *modname) 32767c478bd9Sstevel@tonic-gate { 32777c478bd9Sstevel@tonic-gate struct modctl *mp; 32787c478bd9Sstevel@tonic-gate 32797c478bd9Sstevel@tonic-gate mp = kobj_zalloc(sizeof (*mp), KM_SLEEP); 32807c478bd9Sstevel@tonic-gate mp->mod_modname = kobj_zalloc(strlen(modname) + 1, KM_SLEEP); 32817c478bd9Sstevel@tonic-gate (void) strcpy(mp->mod_modname, modname); 32827c478bd9Sstevel@tonic-gate return (mp); 32837c478bd9Sstevel@tonic-gate } 32847c478bd9Sstevel@tonic-gate 32857c478bd9Sstevel@tonic-gate /* 32867c478bd9Sstevel@tonic-gate * Get the value of a symbol. This is a wrapper routine that 32877c478bd9Sstevel@tonic-gate * calls kobj_getsymvalue(). kobj_getsymvalue() may go away but this 32887c478bd9Sstevel@tonic-gate * wrapper will prevent callers from noticing. 32897c478bd9Sstevel@tonic-gate */ 32907c478bd9Sstevel@tonic-gate uintptr_t 32917c478bd9Sstevel@tonic-gate modgetsymvalue(char *name, int kernelonly) 32927c478bd9Sstevel@tonic-gate { 32937c478bd9Sstevel@tonic-gate return (kobj_getsymvalue(name, kernelonly)); 32947c478bd9Sstevel@tonic-gate } 32957c478bd9Sstevel@tonic-gate 32967c478bd9Sstevel@tonic-gate /* 32977c478bd9Sstevel@tonic-gate * Get the symbol nearest an address. This is a wrapper routine that 32987c478bd9Sstevel@tonic-gate * calls kobj_getsymname(). kobj_getsymname() may go away but this 32997c478bd9Sstevel@tonic-gate * wrapper will prevent callers from noticing. 33007c478bd9Sstevel@tonic-gate */ 33017c478bd9Sstevel@tonic-gate char * 33027c478bd9Sstevel@tonic-gate modgetsymname(uintptr_t value, ulong_t *offset) 33037c478bd9Sstevel@tonic-gate { 33047c478bd9Sstevel@tonic-gate return (kobj_getsymname(value, offset)); 33057c478bd9Sstevel@tonic-gate } 33067c478bd9Sstevel@tonic-gate 33077c478bd9Sstevel@tonic-gate /* 33087aec1d6eScindi * Lookup a symbol in a specified module. These are wrapper routines that 33097aec1d6eScindi * call kobj_lookup(). kobj_lookup() may go away but these wrappers will 33107aec1d6eScindi * prevent callers from noticing. 33117c478bd9Sstevel@tonic-gate */ 33127c478bd9Sstevel@tonic-gate uintptr_t 33137aec1d6eScindi modlookup(const char *modname, const char *symname) 33147c478bd9Sstevel@tonic-gate { 33157c478bd9Sstevel@tonic-gate struct modctl *modp; 33167c478bd9Sstevel@tonic-gate uintptr_t val; 33177c478bd9Sstevel@tonic-gate 33187c478bd9Sstevel@tonic-gate if ((modp = mod_hold_by_name(modname)) == NULL) 33197c478bd9Sstevel@tonic-gate return (0); 33207c478bd9Sstevel@tonic-gate val = kobj_lookup(modp->mod_mp, symname); 33217c478bd9Sstevel@tonic-gate mod_release_mod(modp); 33227c478bd9Sstevel@tonic-gate return (val); 33237c478bd9Sstevel@tonic-gate } 33247c478bd9Sstevel@tonic-gate 33257aec1d6eScindi uintptr_t 33267aec1d6eScindi modlookup_by_modctl(modctl_t *modp, const char *symname) 33277aec1d6eScindi { 33287aec1d6eScindi ASSERT(modp->mod_ref > 0 || modp->mod_busy); 33297aec1d6eScindi 33307aec1d6eScindi return (kobj_lookup(modp->mod_mp, symname)); 33317aec1d6eScindi } 33327aec1d6eScindi 33337c478bd9Sstevel@tonic-gate /* 33347c478bd9Sstevel@tonic-gate * Ask the user for the name of the system file and the default path 33357c478bd9Sstevel@tonic-gate * for modules. 33367c478bd9Sstevel@tonic-gate */ 33377c478bd9Sstevel@tonic-gate void 33387c478bd9Sstevel@tonic-gate mod_askparams() 33397c478bd9Sstevel@tonic-gate { 33407c478bd9Sstevel@tonic-gate static char s0[64]; 33417c478bd9Sstevel@tonic-gate intptr_t fd; 33427c478bd9Sstevel@tonic-gate 33437c478bd9Sstevel@tonic-gate if ((fd = kobj_open(systemfile)) != -1L) 33447c478bd9Sstevel@tonic-gate kobj_close(fd); 33457c478bd9Sstevel@tonic-gate else 33467c478bd9Sstevel@tonic-gate systemfile = NULL; 33477c478bd9Sstevel@tonic-gate 33487c478bd9Sstevel@tonic-gate /*CONSTANTCONDITION*/ 33497c478bd9Sstevel@tonic-gate while (1) { 33507c478bd9Sstevel@tonic-gate printf("Name of system file [%s]: ", 33517c478bd9Sstevel@tonic-gate systemfile ? systemfile : "/dev/null"); 33527c478bd9Sstevel@tonic-gate 33537c478bd9Sstevel@tonic-gate console_gets(s0, sizeof (s0)); 33547c478bd9Sstevel@tonic-gate 33557c478bd9Sstevel@tonic-gate if (s0[0] == '\0') 33567c478bd9Sstevel@tonic-gate break; 33577c478bd9Sstevel@tonic-gate else if (strcmp(s0, "/dev/null") == 0) { 33587c478bd9Sstevel@tonic-gate systemfile = NULL; 33597c478bd9Sstevel@tonic-gate break; 33607c478bd9Sstevel@tonic-gate } else { 33617c478bd9Sstevel@tonic-gate if ((fd = kobj_open(s0)) != -1L) { 33627c478bd9Sstevel@tonic-gate kobj_close(fd); 33637c478bd9Sstevel@tonic-gate systemfile = s0; 33647c478bd9Sstevel@tonic-gate break; 33657c478bd9Sstevel@tonic-gate } 33667c478bd9Sstevel@tonic-gate } 33677c478bd9Sstevel@tonic-gate printf("can't find file %s\n", s0); 33687c478bd9Sstevel@tonic-gate } 33697c478bd9Sstevel@tonic-gate } 33707c478bd9Sstevel@tonic-gate 33717c478bd9Sstevel@tonic-gate static char loading_msg[] = "loading '%s' id %d\n"; 33727c478bd9Sstevel@tonic-gate static char load_msg[] = "load '%s' id %d loaded @ 0x%p/0x%p size %d/%d\n"; 33737c478bd9Sstevel@tonic-gate 33747c478bd9Sstevel@tonic-gate /* 33757c478bd9Sstevel@tonic-gate * Common code for loading a module (but not installing it). 3376da6c28aaSamw * Handoff the task of module loading to a separate thread 33777c478bd9Sstevel@tonic-gate * with a large stack if possible, since this code may recurse a few times. 33787c478bd9Sstevel@tonic-gate * Return zero if there are no errors or an errno value. 33797c478bd9Sstevel@tonic-gate */ 33807c478bd9Sstevel@tonic-gate static int 33817c478bd9Sstevel@tonic-gate mod_load(struct modctl *mp, int usepath) 33827c478bd9Sstevel@tonic-gate { 33837c478bd9Sstevel@tonic-gate int retval; 33847c478bd9Sstevel@tonic-gate struct modinfo *modinfop = NULL; 33857c478bd9Sstevel@tonic-gate struct loadmt lt; 33867c478bd9Sstevel@tonic-gate 33877c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 33887c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy); 33897c478bd9Sstevel@tonic-gate 33907c478bd9Sstevel@tonic-gate if (mp->mod_loaded) 33917c478bd9Sstevel@tonic-gate return (0); 33927c478bd9Sstevel@tonic-gate 33937c478bd9Sstevel@tonic-gate if (mod_sysctl(SYS_CHECK_EXCLUDE, mp->mod_modname) != 0 || 33947c478bd9Sstevel@tonic-gate mod_sysctl(SYS_CHECK_EXCLUDE, mp->mod_filename) != 0) { 33957c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG) { 33967c478bd9Sstevel@tonic-gate printf(mod_excl_msg, mp->mod_filename, 33977c478bd9Sstevel@tonic-gate mp->mod_modname); 33987c478bd9Sstevel@tonic-gate } 33997c478bd9Sstevel@tonic-gate return (ENXIO); 34007c478bd9Sstevel@tonic-gate } 34017c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG2) 34027c478bd9Sstevel@tonic-gate printf(loading_msg, mp->mod_filename, mp->mod_id); 34037c478bd9Sstevel@tonic-gate 34047c478bd9Sstevel@tonic-gate if (curthread != &t0) { 34057c478bd9Sstevel@tonic-gate lt.mp = mp; 34067c478bd9Sstevel@tonic-gate lt.usepath = usepath; 34077c478bd9Sstevel@tonic-gate lt.owner = curthread; 34087c478bd9Sstevel@tonic-gate sema_init(<.sema, 0, NULL, SEMA_DEFAULT, NULL); 34097c478bd9Sstevel@tonic-gate 34107c478bd9Sstevel@tonic-gate /* create thread to hand of call to */ 34117c478bd9Sstevel@tonic-gate (void) thread_create(NULL, DEFAULTSTKSZ * 2, 34127c478bd9Sstevel@tonic-gate modload_thread, <, 0, &p0, TS_RUN, maxclsyspri); 34137c478bd9Sstevel@tonic-gate 34147c478bd9Sstevel@tonic-gate /* wait for thread to complete kobj_load_module */ 34157c478bd9Sstevel@tonic-gate sema_p(<.sema); 34167c478bd9Sstevel@tonic-gate 34177c478bd9Sstevel@tonic-gate sema_destroy(<.sema); 34187c478bd9Sstevel@tonic-gate retval = lt.retval; 34197c478bd9Sstevel@tonic-gate } else 34207c478bd9Sstevel@tonic-gate retval = kobj_load_module(mp, usepath); 34217c478bd9Sstevel@tonic-gate 34227c478bd9Sstevel@tonic-gate if (mp->mod_mp) { 34237c478bd9Sstevel@tonic-gate ASSERT(retval == 0); 34247c478bd9Sstevel@tonic-gate mp->mod_loaded = 1; 34257c478bd9Sstevel@tonic-gate mp->mod_loadcnt++; 34267c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG) { 34277c478bd9Sstevel@tonic-gate printf(load_msg, mp->mod_filename, mp->mod_id, 34287c478bd9Sstevel@tonic-gate (void *)((struct module *)mp->mod_mp)->text, 34297c478bd9Sstevel@tonic-gate (void *)((struct module *)mp->mod_mp)->data, 34307c478bd9Sstevel@tonic-gate ((struct module *)mp->mod_mp)->text_size, 34317c478bd9Sstevel@tonic-gate ((struct module *)mp->mod_mp)->data_size); 34327c478bd9Sstevel@tonic-gate } 34337c478bd9Sstevel@tonic-gate 34347c478bd9Sstevel@tonic-gate /* 34357c478bd9Sstevel@tonic-gate * XXX - There should be a better way to get this. 34367c478bd9Sstevel@tonic-gate */ 34377c478bd9Sstevel@tonic-gate modinfop = kmem_zalloc(sizeof (struct modinfo), KM_SLEEP); 34387c478bd9Sstevel@tonic-gate modinfop->mi_info = MI_INFO_LINKAGE; 34397c478bd9Sstevel@tonic-gate if (mod_getinfo(mp, modinfop) == 0) 34407c478bd9Sstevel@tonic-gate mp->mod_linkage = NULL; 34417c478bd9Sstevel@tonic-gate else { 34427c478bd9Sstevel@tonic-gate mp->mod_linkage = (void *)modinfop->mi_base; 34437c478bd9Sstevel@tonic-gate ASSERT(mp->mod_linkage->ml_rev == MODREV_1); 34447c478bd9Sstevel@tonic-gate } 34457c478bd9Sstevel@tonic-gate 34467c478bd9Sstevel@tonic-gate /* 34477c478bd9Sstevel@tonic-gate * DCS: bootstrapping code. If the driver is loaded 34487c478bd9Sstevel@tonic-gate * before root mount, it is assumed that the driver 34497c478bd9Sstevel@tonic-gate * may be used before mounting root. In order to 34507c478bd9Sstevel@tonic-gate * access mappings of global to local minor no.'s 34517c478bd9Sstevel@tonic-gate * during installation/open of the driver, we load 34527c478bd9Sstevel@tonic-gate * them into memory here while the BOP_interfaces 34537c478bd9Sstevel@tonic-gate * are still up. 34547c478bd9Sstevel@tonic-gate */ 34557c478bd9Sstevel@tonic-gate if ((cluster_bootflags & CLUSTER_BOOTED) && !modrootloaded) { 34567c478bd9Sstevel@tonic-gate retval = clboot_modload(mp); 34577c478bd9Sstevel@tonic-gate } 34587c478bd9Sstevel@tonic-gate 34597c478bd9Sstevel@tonic-gate kmem_free(modinfop, sizeof (struct modinfo)); 34607c478bd9Sstevel@tonic-gate (void) mod_sysctl(SYS_SET_MVAR, (void *)mp); 34617c478bd9Sstevel@tonic-gate retval = install_stubs_by_name(mp, mp->mod_modname); 34627c478bd9Sstevel@tonic-gate 34637c478bd9Sstevel@tonic-gate /* 34647c478bd9Sstevel@tonic-gate * Now that the module is loaded, we need to give DTrace 34657c478bd9Sstevel@tonic-gate * a chance to notify its providers. This is done via 34667c478bd9Sstevel@tonic-gate * the dtrace_modload function pointer. 34677c478bd9Sstevel@tonic-gate */ 34687c478bd9Sstevel@tonic-gate if (strcmp(mp->mod_modname, "dtrace") != 0) { 34697c478bd9Sstevel@tonic-gate struct modctl *dmp = mod_hold_by_name("dtrace"); 34707c478bd9Sstevel@tonic-gate 34717c478bd9Sstevel@tonic-gate if (dmp != NULL && dtrace_modload != NULL) 34727c478bd9Sstevel@tonic-gate (*dtrace_modload)(mp); 34737c478bd9Sstevel@tonic-gate 34747c478bd9Sstevel@tonic-gate mod_release_mod(dmp); 34757c478bd9Sstevel@tonic-gate } 34767c478bd9Sstevel@tonic-gate 34777c478bd9Sstevel@tonic-gate } else { 34787c478bd9Sstevel@tonic-gate /* 34797c478bd9Sstevel@tonic-gate * If load failed then we need to release any requisites 34807c478bd9Sstevel@tonic-gate * that we had established. 34817c478bd9Sstevel@tonic-gate */ 34827c478bd9Sstevel@tonic-gate ASSERT(retval); 34837c478bd9Sstevel@tonic-gate mod_release_requisites(mp); 34847c478bd9Sstevel@tonic-gate 34857c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_ERRMSG) 34867c478bd9Sstevel@tonic-gate printf("error loading '%s', error %d\n", 34877c478bd9Sstevel@tonic-gate mp->mod_filename, retval); 34887c478bd9Sstevel@tonic-gate } 34897c478bd9Sstevel@tonic-gate return (retval); 34907c478bd9Sstevel@tonic-gate } 34917c478bd9Sstevel@tonic-gate 34927c478bd9Sstevel@tonic-gate static char unload_msg[] = "unloading %s, module id %d, loadcnt %d.\n"; 34937c478bd9Sstevel@tonic-gate 34947c478bd9Sstevel@tonic-gate static void 34957c478bd9Sstevel@tonic-gate mod_unload(struct modctl *mp) 34967c478bd9Sstevel@tonic-gate { 34977c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 34987c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy); 34997c478bd9Sstevel@tonic-gate ASSERT((mp->mod_loaded && (mp->mod_installed == 0)) && 35007c478bd9Sstevel@tonic-gate ((mp->mod_prim == 0) && (mp->mod_ref >= 0))); 35017c478bd9Sstevel@tonic-gate 35027c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG) 35037c478bd9Sstevel@tonic-gate printf(unload_msg, mp->mod_modname, 35047c478bd9Sstevel@tonic-gate mp->mod_id, mp->mod_loadcnt); 35057c478bd9Sstevel@tonic-gate 35067c478bd9Sstevel@tonic-gate /* 35077c478bd9Sstevel@tonic-gate * If mod_ref is not zero, it means some modules might still refer 35087c478bd9Sstevel@tonic-gate * to this module. Then you can't unload this module right now. 35097c478bd9Sstevel@tonic-gate * Instead, set 1 to mod_delay_unload to notify the system of 35107c478bd9Sstevel@tonic-gate * unloading this module later when it's not required any more. 35117c478bd9Sstevel@tonic-gate */ 35127c478bd9Sstevel@tonic-gate if (mp->mod_ref > 0) { 35137c478bd9Sstevel@tonic-gate mp->mod_delay_unload = 1; 35147c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG2) { 35157c478bd9Sstevel@tonic-gate printf("module %s not unloaded," 35167c478bd9Sstevel@tonic-gate " non-zero reference count (%d)", 35177c478bd9Sstevel@tonic-gate mp->mod_modname, mp->mod_ref); 35187c478bd9Sstevel@tonic-gate } 35197c478bd9Sstevel@tonic-gate return; 35207c478bd9Sstevel@tonic-gate } 35217c478bd9Sstevel@tonic-gate 35227c478bd9Sstevel@tonic-gate if (((mp->mod_loaded == 0) || mp->mod_installed) || 35237c478bd9Sstevel@tonic-gate (mp->mod_ref || mp->mod_prim)) { 35247c478bd9Sstevel@tonic-gate /* 35257c478bd9Sstevel@tonic-gate * A DEBUG kernel would ASSERT panic above, the code is broken 35267c478bd9Sstevel@tonic-gate * if we get this warning. 35277c478bd9Sstevel@tonic-gate */ 35287c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "mod_unload: %s in incorrect state: %d %d %d", 35297c478bd9Sstevel@tonic-gate mp->mod_filename, mp->mod_installed, mp->mod_loaded, 35307c478bd9Sstevel@tonic-gate mp->mod_ref); 35317c478bd9Sstevel@tonic-gate return; 35327c478bd9Sstevel@tonic-gate } 35337c478bd9Sstevel@tonic-gate 35347c478bd9Sstevel@tonic-gate /* reset stub functions to call the binder again */ 35357c478bd9Sstevel@tonic-gate reset_stubs(mp); 35367c478bd9Sstevel@tonic-gate 35377c478bd9Sstevel@tonic-gate /* 35387c478bd9Sstevel@tonic-gate * mark module as unloaded before the modctl structure is freed. 35397c478bd9Sstevel@tonic-gate * This is required not to reuse the modctl structure before 35407c478bd9Sstevel@tonic-gate * the module is marked as unloaded. 35417c478bd9Sstevel@tonic-gate */ 35427c478bd9Sstevel@tonic-gate mp->mod_loaded = 0; 35437c478bd9Sstevel@tonic-gate mp->mod_linkage = NULL; 35447c478bd9Sstevel@tonic-gate 35457c478bd9Sstevel@tonic-gate /* free the memory */ 35467c478bd9Sstevel@tonic-gate kobj_unload_module(mp); 35477c478bd9Sstevel@tonic-gate 35487c478bd9Sstevel@tonic-gate if (mp->mod_delay_unload) { 35497c478bd9Sstevel@tonic-gate mp->mod_delay_unload = 0; 35507c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG2) { 35517c478bd9Sstevel@tonic-gate printf("deferred unload of module %s" 35527c478bd9Sstevel@tonic-gate " (id %d) successful", 35537c478bd9Sstevel@tonic-gate mp->mod_modname, mp->mod_id); 35547c478bd9Sstevel@tonic-gate } 35557c478bd9Sstevel@tonic-gate } 35567c478bd9Sstevel@tonic-gate 35577c478bd9Sstevel@tonic-gate /* release hold on requisites */ 35587c478bd9Sstevel@tonic-gate mod_release_requisites(mp); 35597c478bd9Sstevel@tonic-gate 35607c478bd9Sstevel@tonic-gate /* 35617c478bd9Sstevel@tonic-gate * Now that the module is gone, we need to give DTrace a chance to 35627c478bd9Sstevel@tonic-gate * remove any probes that it may have had in the module. This is 35637c478bd9Sstevel@tonic-gate * done via the dtrace_modunload function pointer. 35647c478bd9Sstevel@tonic-gate */ 35657c478bd9Sstevel@tonic-gate if (strcmp(mp->mod_modname, "dtrace") != 0) { 35667c478bd9Sstevel@tonic-gate struct modctl *dmp = mod_hold_by_name("dtrace"); 35677c478bd9Sstevel@tonic-gate 35687c478bd9Sstevel@tonic-gate if (dmp != NULL && dtrace_modunload != NULL) 35697c478bd9Sstevel@tonic-gate (*dtrace_modunload)(mp); 35707c478bd9Sstevel@tonic-gate 35717c478bd9Sstevel@tonic-gate mod_release_mod(dmp); 35727c478bd9Sstevel@tonic-gate } 35737c478bd9Sstevel@tonic-gate } 35747c478bd9Sstevel@tonic-gate 35757c478bd9Sstevel@tonic-gate static int 35767c478bd9Sstevel@tonic-gate modinstall(struct modctl *mp) 35777c478bd9Sstevel@tonic-gate { 35787c478bd9Sstevel@tonic-gate int val; 35797c478bd9Sstevel@tonic-gate int (*func)(void); 35807c478bd9Sstevel@tonic-gate 35817c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 35827c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy && mp->mod_loaded); 35837c478bd9Sstevel@tonic-gate 35847c478bd9Sstevel@tonic-gate if (mp->mod_installed) 35857c478bd9Sstevel@tonic-gate return (0); 35867c478bd9Sstevel@tonic-gate /* 35877c478bd9Sstevel@tonic-gate * If mod_delay_unload is on, it means the system chose the deferred 35887c478bd9Sstevel@tonic-gate * unload for this module. Then you can't install this module until 35897c478bd9Sstevel@tonic-gate * it's unloaded from the system. 35907c478bd9Sstevel@tonic-gate */ 35917c478bd9Sstevel@tonic-gate if (mp->mod_delay_unload) 35927c478bd9Sstevel@tonic-gate return (ENXIO); 35937c478bd9Sstevel@tonic-gate 35947c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG) 35957c478bd9Sstevel@tonic-gate printf("installing %s, module id %d.\n", 35967c478bd9Sstevel@tonic-gate mp->mod_modname, mp->mod_id); 35977c478bd9Sstevel@tonic-gate 35987c478bd9Sstevel@tonic-gate ASSERT(mp->mod_mp != NULL); 35997c478bd9Sstevel@tonic-gate if (mod_install_requisites(mp) != 0) { 36007c478bd9Sstevel@tonic-gate /* 36017c478bd9Sstevel@tonic-gate * Note that we can't call mod_unload(mp) here since 36027c478bd9Sstevel@tonic-gate * if modinstall() was called by mod_install_requisites(), 36037c478bd9Sstevel@tonic-gate * we won't be able to hold the dependent modules 36047c478bd9Sstevel@tonic-gate * (otherwise there would be a deadlock). 36057c478bd9Sstevel@tonic-gate */ 36067c478bd9Sstevel@tonic-gate return (ENXIO); 36077c478bd9Sstevel@tonic-gate } 36087c478bd9Sstevel@tonic-gate 36097c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_ERRMSG) { 36107c478bd9Sstevel@tonic-gate printf("init '%s' id %d loaded @ 0x%p/0x%p size %lu/%lu\n", 36117c478bd9Sstevel@tonic-gate mp->mod_filename, mp->mod_id, 36127c478bd9Sstevel@tonic-gate (void *)((struct module *)mp->mod_mp)->text, 36137c478bd9Sstevel@tonic-gate (void *)((struct module *)mp->mod_mp)->data, 36147c478bd9Sstevel@tonic-gate ((struct module *)mp->mod_mp)->text_size, 36157c478bd9Sstevel@tonic-gate ((struct module *)mp->mod_mp)->data_size); 36167c478bd9Sstevel@tonic-gate } 36177c478bd9Sstevel@tonic-gate 36187c478bd9Sstevel@tonic-gate func = (int (*)())kobj_lookup(mp->mod_mp, "_init"); 36197c478bd9Sstevel@tonic-gate 36207c478bd9Sstevel@tonic-gate if (kobj_addrcheck(mp->mod_mp, (caddr_t)func)) { 36217c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "_init() not defined properly in %s", 36227c478bd9Sstevel@tonic-gate mp->mod_filename); 36237c478bd9Sstevel@tonic-gate return (EFAULT); 36247c478bd9Sstevel@tonic-gate } 36257c478bd9Sstevel@tonic-gate 36267c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_USERDEBUG) { 36277c478bd9Sstevel@tonic-gate printf("breakpoint before calling %s:_init()\n", 36287c478bd9Sstevel@tonic-gate mp->mod_modname); 36297c478bd9Sstevel@tonic-gate if (DEBUGGER_PRESENT) 36307c478bd9Sstevel@tonic-gate debug_enter("_init"); 36317c478bd9Sstevel@tonic-gate } 36327c478bd9Sstevel@tonic-gate 36337c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 36347c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy && mp->mod_loaded); 36357c478bd9Sstevel@tonic-gate val = (*func)(); /* call _init */ 36367c478bd9Sstevel@tonic-gate 36377c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_USERDEBUG) 36387c478bd9Sstevel@tonic-gate printf("Returned from _init, val = %x\n", val); 36397c478bd9Sstevel@tonic-gate 36407c478bd9Sstevel@tonic-gate if (val == 0) { 36417c478bd9Sstevel@tonic-gate /* 36427c478bd9Sstevel@tonic-gate * Set the MODS_INSTALLED flag to enable this module 36437c478bd9Sstevel@tonic-gate * being called now. 36447c478bd9Sstevel@tonic-gate */ 36457c478bd9Sstevel@tonic-gate install_stubs(mp); 36467c478bd9Sstevel@tonic-gate mp->mod_installed = 1; 36477c478bd9Sstevel@tonic-gate } else if (moddebug & MODDEBUG_ERRMSG) 36487c478bd9Sstevel@tonic-gate printf(mod_init_msg, mp->mod_filename, mp->mod_modname, val); 36497c478bd9Sstevel@tonic-gate 36507c478bd9Sstevel@tonic-gate return (val); 36517c478bd9Sstevel@tonic-gate } 36527c478bd9Sstevel@tonic-gate 3653a7aa4df7Scth int detach_driver_unconfig = 0; 3654a7aa4df7Scth 36557c478bd9Sstevel@tonic-gate static int 36567c478bd9Sstevel@tonic-gate detach_driver(char *name) 36577c478bd9Sstevel@tonic-gate { 36587c478bd9Sstevel@tonic-gate major_t major; 36597c478bd9Sstevel@tonic-gate int error; 36607c478bd9Sstevel@tonic-gate 36617c478bd9Sstevel@tonic-gate /* 36627c478bd9Sstevel@tonic-gate * If being called from mod_uninstall_all() then the appropriate 36637c478bd9Sstevel@tonic-gate * driver detaches (leaf only) have already been done. 36647c478bd9Sstevel@tonic-gate */ 36657c478bd9Sstevel@tonic-gate if (mod_in_autounload()) 36667c478bd9Sstevel@tonic-gate return (0); 36677c478bd9Sstevel@tonic-gate 36687c478bd9Sstevel@tonic-gate major = ddi_name_to_major(name); 3669a204de77Scth if (major == DDI_MAJOR_T_NONE) 36707c478bd9Sstevel@tonic-gate return (0); 36717c478bd9Sstevel@tonic-gate 36727c478bd9Sstevel@tonic-gate error = ndi_devi_unconfig_driver(ddi_root_node(), 3673a7aa4df7Scth NDI_DETACH_DRIVER | detach_driver_unconfig, major); 36747c478bd9Sstevel@tonic-gate return (error == NDI_SUCCESS ? 0 : -1); 36757c478bd9Sstevel@tonic-gate } 36767c478bd9Sstevel@tonic-gate 36777c478bd9Sstevel@tonic-gate static char finiret_msg[] = "Returned from _fini for %s, status = %x\n"; 36787c478bd9Sstevel@tonic-gate 36797c478bd9Sstevel@tonic-gate static int 36807c478bd9Sstevel@tonic-gate moduninstall(struct modctl *mp) 36817c478bd9Sstevel@tonic-gate { 36827c478bd9Sstevel@tonic-gate int status = 0; 36837c478bd9Sstevel@tonic-gate int (*func)(void); 36847c478bd9Sstevel@tonic-gate 36857c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 36867c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy); 36877c478bd9Sstevel@tonic-gate 36887c478bd9Sstevel@tonic-gate /* 36897c478bd9Sstevel@tonic-gate * Verify that we need to do something and can uninstall the module. 36907c478bd9Sstevel@tonic-gate * 36917c478bd9Sstevel@tonic-gate * If we should not uninstall the module or if the module is not in 36927c478bd9Sstevel@tonic-gate * the correct state to start an uninstall we return EBUSY to prevent 36937c478bd9Sstevel@tonic-gate * us from progressing to mod_unload. If the module has already been 36947c478bd9Sstevel@tonic-gate * uninstalled and unloaded we return EALREADY. 36957c478bd9Sstevel@tonic-gate */ 36967c478bd9Sstevel@tonic-gate if (mp->mod_prim || mp->mod_ref || mp->mod_nenabled != 0) 36977c478bd9Sstevel@tonic-gate return (EBUSY); 36987c478bd9Sstevel@tonic-gate if ((mp->mod_installed == 0) || (mp->mod_loaded == 0)) 36997c478bd9Sstevel@tonic-gate return (EALREADY); 37007c478bd9Sstevel@tonic-gate 37017c478bd9Sstevel@tonic-gate /* 37027c478bd9Sstevel@tonic-gate * To avoid devinfo / module deadlock we must release this module 37037c478bd9Sstevel@tonic-gate * prior to initiating the detach_driver, otherwise the detach_driver 37047c478bd9Sstevel@tonic-gate * might deadlock on a devinfo node held by another thread 37057c478bd9Sstevel@tonic-gate * coming top down and involving the module we have locked. 37067c478bd9Sstevel@tonic-gate * 37077c478bd9Sstevel@tonic-gate * When we regrab the module we must reverify that it is OK 37087c478bd9Sstevel@tonic-gate * to proceed with the uninstall operation. 37097c478bd9Sstevel@tonic-gate */ 37107c478bd9Sstevel@tonic-gate mod_release_mod(mp); 37117c478bd9Sstevel@tonic-gate status = detach_driver(mp->mod_modname); 37127c478bd9Sstevel@tonic-gate (void) mod_hold_by_modctl(mp, MOD_WAIT_FOREVER | MOD_LOCK_NOT_HELD); 37137c478bd9Sstevel@tonic-gate 37147c478bd9Sstevel@tonic-gate /* check detach status and reverify state with lock */ 37157c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 37167c478bd9Sstevel@tonic-gate if ((status != 0) || mp->mod_prim || mp->mod_ref) { 37177c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 37187c478bd9Sstevel@tonic-gate return (EBUSY); 37197c478bd9Sstevel@tonic-gate } 37207c478bd9Sstevel@tonic-gate if ((mp->mod_installed == 0) || (mp->mod_loaded == 0)) { 37217c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 37227c478bd9Sstevel@tonic-gate return (EALREADY); 37237c478bd9Sstevel@tonic-gate } 37247c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 37257c478bd9Sstevel@tonic-gate 37267c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG2) 37277c478bd9Sstevel@tonic-gate printf("uninstalling %s\n", mp->mod_modname); 37287c478bd9Sstevel@tonic-gate 37297c478bd9Sstevel@tonic-gate /* 37307c478bd9Sstevel@tonic-gate * lookup _fini, return EBUSY if not defined. 37317c478bd9Sstevel@tonic-gate * 37327c478bd9Sstevel@tonic-gate * The MODDEBUG_FINI_EBUSY is usefull in resolving leaks in 37337c478bd9Sstevel@tonic-gate * detach(9E) - it allows bufctl addresses to be resolved. 37347c478bd9Sstevel@tonic-gate */ 37357c478bd9Sstevel@tonic-gate func = (int (*)())kobj_lookup(mp->mod_mp, "_fini"); 37367c478bd9Sstevel@tonic-gate if ((func == NULL) || (mp->mod_loadflags & MOD_NOUNLOAD) || 37377c478bd9Sstevel@tonic-gate (moddebug & MODDEBUG_FINI_EBUSY)) 37387c478bd9Sstevel@tonic-gate return (EBUSY); 37397c478bd9Sstevel@tonic-gate 37407c478bd9Sstevel@tonic-gate /* verify that _fini is in this module */ 37417c478bd9Sstevel@tonic-gate if (kobj_addrcheck(mp->mod_mp, (caddr_t)func)) { 37427c478bd9Sstevel@tonic-gate cmn_err(CE_WARN, "_fini() not defined properly in %s", 37437c478bd9Sstevel@tonic-gate mp->mod_filename); 37447c478bd9Sstevel@tonic-gate return (EFAULT); 37457c478bd9Sstevel@tonic-gate } 37467c478bd9Sstevel@tonic-gate 37477c478bd9Sstevel@tonic-gate /* call _fini() */ 37487c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 37497c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy && mp->mod_loaded && mp->mod_installed); 37507c478bd9Sstevel@tonic-gate 37517c478bd9Sstevel@tonic-gate status = (*func)(); 37527c478bd9Sstevel@tonic-gate 37537c478bd9Sstevel@tonic-gate if (status == 0) { 37547c478bd9Sstevel@tonic-gate /* _fini returned success, the module is no longer installed */ 37557c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG) 37567c478bd9Sstevel@tonic-gate printf("uninstalled %s\n", mp->mod_modname); 37577c478bd9Sstevel@tonic-gate 37587c478bd9Sstevel@tonic-gate /* 37597c478bd9Sstevel@tonic-gate * Even though we only set mod_installed to zero here, a zero 3760da6c28aaSamw * return value means we are committed to a code path were 37617c478bd9Sstevel@tonic-gate * mod_loaded will also end up as zero - we have no other 37627c478bd9Sstevel@tonic-gate * way to get the module data and bss back to the pre _init 37637c478bd9Sstevel@tonic-gate * state except a reload. To ensure this, after return, 37647c478bd9Sstevel@tonic-gate * mod_busy must stay set until mod_loaded is cleared. 37657c478bd9Sstevel@tonic-gate */ 37667c478bd9Sstevel@tonic-gate mp->mod_installed = 0; 37677c478bd9Sstevel@tonic-gate 37687c478bd9Sstevel@tonic-gate /* 37697c478bd9Sstevel@tonic-gate * Clear the MODS_INSTALLED flag not to call functions 37707c478bd9Sstevel@tonic-gate * in the module directly from now on. 37717c478bd9Sstevel@tonic-gate */ 37727c478bd9Sstevel@tonic-gate uninstall_stubs(mp); 37737c478bd9Sstevel@tonic-gate } else { 37747c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_USERDEBUG) 37757c478bd9Sstevel@tonic-gate printf(finiret_msg, mp->mod_filename, status); 37767c478bd9Sstevel@tonic-gate /* 37777c478bd9Sstevel@tonic-gate * By definition _fini is only allowed to return EBUSY or the 37787c478bd9Sstevel@tonic-gate * result of mod_remove (EBUSY or EINVAL). In the off chance 37797c478bd9Sstevel@tonic-gate * that a driver returns EALREADY we convert this to EINVAL 37807c478bd9Sstevel@tonic-gate * since to our caller EALREADY means module was already 37817c478bd9Sstevel@tonic-gate * removed. 37827c478bd9Sstevel@tonic-gate */ 37837c478bd9Sstevel@tonic-gate if (status == EALREADY) 37847c478bd9Sstevel@tonic-gate status = EINVAL; 37857c478bd9Sstevel@tonic-gate } 37867c478bd9Sstevel@tonic-gate 37877c478bd9Sstevel@tonic-gate return (status); 37887c478bd9Sstevel@tonic-gate } 37897c478bd9Sstevel@tonic-gate 37907c478bd9Sstevel@tonic-gate /* 37917c478bd9Sstevel@tonic-gate * Uninstall all modules. 37927c478bd9Sstevel@tonic-gate */ 37937c478bd9Sstevel@tonic-gate static void 37947c478bd9Sstevel@tonic-gate mod_uninstall_all(void) 37957c478bd9Sstevel@tonic-gate { 37967c478bd9Sstevel@tonic-gate struct modctl *mp; 37977c478bd9Sstevel@tonic-gate modid_t modid = 0; 37987c478bd9Sstevel@tonic-gate 3799a7aa4df7Scth /* synchronize with any active modunload_disable() */ 3800a7aa4df7Scth modunload_begin(); 3801a7aa4df7Scth 38027c478bd9Sstevel@tonic-gate /* mark this thread as doing autounloading */ 38037c478bd9Sstevel@tonic-gate (void) tsd_set(mod_autounload_key, (void *)1); 38047c478bd9Sstevel@tonic-gate 38057c478bd9Sstevel@tonic-gate (void) devfs_clean(ddi_root_node(), NULL, 0); 38067c478bd9Sstevel@tonic-gate (void) ndi_devi_unconfig(ddi_root_node(), NDI_AUTODETACH); 38077c478bd9Sstevel@tonic-gate 38087c478bd9Sstevel@tonic-gate while ((mp = mod_hold_next_by_id(modid)) != NULL) { 38097c478bd9Sstevel@tonic-gate modid = mp->mod_id; 38107c478bd9Sstevel@tonic-gate /* 38117c478bd9Sstevel@tonic-gate * Skip modules with the MOD_NOAUTOUNLOAD flag set 38127c478bd9Sstevel@tonic-gate */ 38137c478bd9Sstevel@tonic-gate if (mp->mod_loadflags & MOD_NOAUTOUNLOAD) { 38147c478bd9Sstevel@tonic-gate mod_release_mod(mp); 38157c478bd9Sstevel@tonic-gate continue; 38167c478bd9Sstevel@tonic-gate } 38177c478bd9Sstevel@tonic-gate 38180b38a8bdSahl if (moduninstall(mp) == 0) { 38197c478bd9Sstevel@tonic-gate mod_unload(mp); 38200b38a8bdSahl CPU_STATS_ADDQ(CPU, sys, modunload, 1); 38210b38a8bdSahl } 38227c478bd9Sstevel@tonic-gate mod_release_mod(mp); 38237c478bd9Sstevel@tonic-gate } 38247c478bd9Sstevel@tonic-gate 38257c478bd9Sstevel@tonic-gate (void) tsd_set(mod_autounload_key, NULL); 3826a7aa4df7Scth modunload_end(); 38277c478bd9Sstevel@tonic-gate } 38287c478bd9Sstevel@tonic-gate 3829a7aa4df7Scth /* wait for unloads that have begun before registering disable */ 38307c478bd9Sstevel@tonic-gate void 38317c478bd9Sstevel@tonic-gate modunload_disable(void) 38327c478bd9Sstevel@tonic-gate { 3833a7aa4df7Scth mutex_enter(&modunload_wait_mutex); 3834a7aa4df7Scth while (modunload_active_count) { 3835a7aa4df7Scth modunload_wait++; 3836a7aa4df7Scth cv_wait(&modunload_wait_cv, &modunload_wait_mutex); 3837a7aa4df7Scth modunload_wait--; 3838a7aa4df7Scth } 3839a7aa4df7Scth modunload_disable_count++; 3840a7aa4df7Scth mutex_exit(&modunload_wait_mutex); 38417c478bd9Sstevel@tonic-gate } 38427c478bd9Sstevel@tonic-gate 3843a7aa4df7Scth /* mark end of disable and signal waiters */ 38447c478bd9Sstevel@tonic-gate void 38457c478bd9Sstevel@tonic-gate modunload_enable(void) 38467c478bd9Sstevel@tonic-gate { 3847a7aa4df7Scth mutex_enter(&modunload_wait_mutex); 3848a7aa4df7Scth modunload_disable_count--; 3849a7aa4df7Scth if ((modunload_disable_count == 0) && modunload_wait) 3850a7aa4df7Scth cv_broadcast(&modunload_wait_cv); 3851a7aa4df7Scth mutex_exit(&modunload_wait_mutex); 3852a7aa4df7Scth } 3853a7aa4df7Scth 3854a7aa4df7Scth /* wait for disables to complete before begining unload */ 3855a7aa4df7Scth void 3856a7aa4df7Scth modunload_begin() 3857a7aa4df7Scth { 3858a7aa4df7Scth mutex_enter(&modunload_wait_mutex); 3859a7aa4df7Scth while (modunload_disable_count) { 3860a7aa4df7Scth modunload_wait++; 3861a7aa4df7Scth cv_wait(&modunload_wait_cv, &modunload_wait_mutex); 3862a7aa4df7Scth modunload_wait--; 3863a7aa4df7Scth } 3864a7aa4df7Scth modunload_active_count++; 3865a7aa4df7Scth mutex_exit(&modunload_wait_mutex); 3866a7aa4df7Scth } 3867a7aa4df7Scth 3868a7aa4df7Scth /* mark end of unload and signal waiters */ 3869a7aa4df7Scth void 3870a7aa4df7Scth modunload_end() 3871a7aa4df7Scth { 3872a7aa4df7Scth mutex_enter(&modunload_wait_mutex); 3873a7aa4df7Scth modunload_active_count--; 3874a7aa4df7Scth if ((modunload_active_count == 0) && modunload_wait) 3875a7aa4df7Scth cv_broadcast(&modunload_wait_cv); 3876a7aa4df7Scth mutex_exit(&modunload_wait_mutex); 38777c478bd9Sstevel@tonic-gate } 38787c478bd9Sstevel@tonic-gate 38797c478bd9Sstevel@tonic-gate void 38807c478bd9Sstevel@tonic-gate mod_uninstall_daemon(void) 38817c478bd9Sstevel@tonic-gate { 38827c478bd9Sstevel@tonic-gate callb_cpr_t cprinfo; 3883d3d50737SRafael Vanoni clock_t ticks; 38847c478bd9Sstevel@tonic-gate 38857c478bd9Sstevel@tonic-gate mod_aul_thread = curthread; 38867c478bd9Sstevel@tonic-gate 38877c478bd9Sstevel@tonic-gate CALLB_CPR_INIT(&cprinfo, &mod_uninstall_lock, callb_generic_cpr, "mud"); 38887c478bd9Sstevel@tonic-gate for (;;) { 38897c478bd9Sstevel@tonic-gate mutex_enter(&mod_uninstall_lock); 38907c478bd9Sstevel@tonic-gate CALLB_CPR_SAFE_BEGIN(&cprinfo); 38917c478bd9Sstevel@tonic-gate /* 38927c478bd9Sstevel@tonic-gate * In DEBUG kernels, unheld drivers are uninstalled periodically 38937c478bd9Sstevel@tonic-gate * every mod_uninstall_interval seconds. Periodic uninstall can 38947c478bd9Sstevel@tonic-gate * be disabled by setting mod_uninstall_interval to 0 which is 38957c478bd9Sstevel@tonic-gate * the default for a non-DEBUG kernel. 38967c478bd9Sstevel@tonic-gate */ 38977c478bd9Sstevel@tonic-gate if (mod_uninstall_interval) { 3898d3d50737SRafael Vanoni ticks = drv_usectohz(mod_uninstall_interval * 1000000); 3899d3d50737SRafael Vanoni (void) cv_reltimedwait(&mod_uninstall_cv, 3900d3d50737SRafael Vanoni &mod_uninstall_lock, ticks, TR_CLOCK_TICK); 39017c478bd9Sstevel@tonic-gate } else { 39027c478bd9Sstevel@tonic-gate cv_wait(&mod_uninstall_cv, &mod_uninstall_lock); 39037c478bd9Sstevel@tonic-gate } 39047c478bd9Sstevel@tonic-gate /* 39057c478bd9Sstevel@tonic-gate * The whole daemon is safe for CPR except we don't want 39067c478bd9Sstevel@tonic-gate * the daemon to run if FREEZE is issued and this daemon 39077c478bd9Sstevel@tonic-gate * wakes up from the cv_wait above. In this case, it'll be 39087c478bd9Sstevel@tonic-gate * blocked in CALLB_CPR_SAFE_END until THAW is issued. 39097c478bd9Sstevel@tonic-gate * 39107c478bd9Sstevel@tonic-gate * The reason of calling CALLB_CPR_SAFE_BEGIN twice is that 39117c478bd9Sstevel@tonic-gate * mod_uninstall_lock is used to protect cprinfo and 39127c478bd9Sstevel@tonic-gate * CALLB_CPR_SAFE_BEGIN assumes that this lock is held when 39137c478bd9Sstevel@tonic-gate * called. 39147c478bd9Sstevel@tonic-gate */ 39157c478bd9Sstevel@tonic-gate CALLB_CPR_SAFE_END(&cprinfo, &mod_uninstall_lock); 39167c478bd9Sstevel@tonic-gate CALLB_CPR_SAFE_BEGIN(&cprinfo); 39177c478bd9Sstevel@tonic-gate mutex_exit(&mod_uninstall_lock); 39187c478bd9Sstevel@tonic-gate if ((modunload_disable_count == 0) && 39197c478bd9Sstevel@tonic-gate ((moddebug & MODDEBUG_NOAUTOUNLOAD) == 0)) { 39207c478bd9Sstevel@tonic-gate mod_uninstall_all(); 39217c478bd9Sstevel@tonic-gate } 39227c478bd9Sstevel@tonic-gate } 39237c478bd9Sstevel@tonic-gate } 39247c478bd9Sstevel@tonic-gate 39257c478bd9Sstevel@tonic-gate /* 39267c478bd9Sstevel@tonic-gate * Unload all uninstalled modules. 39277c478bd9Sstevel@tonic-gate */ 39287c478bd9Sstevel@tonic-gate void 39297c478bd9Sstevel@tonic-gate modreap(void) 39307c478bd9Sstevel@tonic-gate { 39317c478bd9Sstevel@tonic-gate mutex_enter(&mod_uninstall_lock); 39327c478bd9Sstevel@tonic-gate cv_broadcast(&mod_uninstall_cv); 39337c478bd9Sstevel@tonic-gate mutex_exit(&mod_uninstall_lock); 39347c478bd9Sstevel@tonic-gate } 39357c478bd9Sstevel@tonic-gate 39367c478bd9Sstevel@tonic-gate /* 39377c478bd9Sstevel@tonic-gate * Hold the specified module. This is the module holding primitive. 39387c478bd9Sstevel@tonic-gate * 39397c478bd9Sstevel@tonic-gate * If MOD_LOCK_HELD then the caller already holds the mod_lock. 39407c478bd9Sstevel@tonic-gate * 39417c478bd9Sstevel@tonic-gate * Return values: 39427c478bd9Sstevel@tonic-gate * 0 ==> the module is held 39437c478bd9Sstevel@tonic-gate * 1 ==> the module is not held and the MOD_WAIT_ONCE caller needs 39447c478bd9Sstevel@tonic-gate * to determine how to retry. 39457c478bd9Sstevel@tonic-gate */ 39467c478bd9Sstevel@tonic-gate int 39477c478bd9Sstevel@tonic-gate mod_hold_by_modctl(struct modctl *mp, int f) 39487c478bd9Sstevel@tonic-gate { 39497c478bd9Sstevel@tonic-gate ASSERT((f & (MOD_WAIT_ONCE | MOD_WAIT_FOREVER)) && 39507c478bd9Sstevel@tonic-gate ((f & (MOD_WAIT_ONCE | MOD_WAIT_FOREVER)) != 39517c478bd9Sstevel@tonic-gate (MOD_WAIT_ONCE | MOD_WAIT_FOREVER))); 39527c478bd9Sstevel@tonic-gate ASSERT((f & (MOD_LOCK_HELD | MOD_LOCK_NOT_HELD)) && 39537c478bd9Sstevel@tonic-gate ((f & (MOD_LOCK_HELD | MOD_LOCK_NOT_HELD)) != 39547c478bd9Sstevel@tonic-gate (MOD_LOCK_HELD | MOD_LOCK_NOT_HELD))); 39557c478bd9Sstevel@tonic-gate ASSERT((f & MOD_LOCK_NOT_HELD) || MUTEX_HELD(&mod_lock)); 39567c478bd9Sstevel@tonic-gate 39577c478bd9Sstevel@tonic-gate if (f & MOD_LOCK_NOT_HELD) 39587c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 39597c478bd9Sstevel@tonic-gate 39607c478bd9Sstevel@tonic-gate while (mp->mod_busy) { 39617c478bd9Sstevel@tonic-gate mp->mod_want = 1; 39627c478bd9Sstevel@tonic-gate cv_wait(&mod_cv, &mod_lock); 39637c478bd9Sstevel@tonic-gate /* 39647c478bd9Sstevel@tonic-gate * Module may be unloaded by daemon. 39657c478bd9Sstevel@tonic-gate * Nevertheless, modctl structure is still in linked list 39667c478bd9Sstevel@tonic-gate * (i.e., off &modules), not freed! 39677c478bd9Sstevel@tonic-gate * Caller is not supposed to assume "mp" is valid, but there 39687c478bd9Sstevel@tonic-gate * is no reasonable way to detect this but using 39697c478bd9Sstevel@tonic-gate * mp->mod_modinfo->mp == NULL check (follow the back pointer) 39707c478bd9Sstevel@tonic-gate * (or similar check depending on calling context) 39717c478bd9Sstevel@tonic-gate * DON'T free modctl structure, it will be very very 39727c478bd9Sstevel@tonic-gate * problematic. 39737c478bd9Sstevel@tonic-gate */ 39747c478bd9Sstevel@tonic-gate if (f & MOD_WAIT_ONCE) { 39757c478bd9Sstevel@tonic-gate if (f & MOD_LOCK_NOT_HELD) 39767c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 39777c478bd9Sstevel@tonic-gate return (1); /* caller decides how to retry */ 39787c478bd9Sstevel@tonic-gate } 39797c478bd9Sstevel@tonic-gate } 39807c478bd9Sstevel@tonic-gate 39817c478bd9Sstevel@tonic-gate mp->mod_busy = 1; 39827c478bd9Sstevel@tonic-gate mp->mod_inprogress_thread = 39837c478bd9Sstevel@tonic-gate (curthread == NULL ? (kthread_id_t)-1 : curthread); 39847c478bd9Sstevel@tonic-gate 39857c478bd9Sstevel@tonic-gate if (f & MOD_LOCK_NOT_HELD) 39867c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 39877c478bd9Sstevel@tonic-gate return (0); 39887c478bd9Sstevel@tonic-gate } 39897c478bd9Sstevel@tonic-gate 39907c478bd9Sstevel@tonic-gate static struct modctl * 39917aec1d6eScindi mod_hold_by_name_common(struct modctl *dep, const char *filename) 39927c478bd9Sstevel@tonic-gate { 39937aec1d6eScindi const char *modname; 39947c478bd9Sstevel@tonic-gate struct modctl *mp; 39957c478bd9Sstevel@tonic-gate char *curname, *newname; 39967c478bd9Sstevel@tonic-gate int found = 0; 39977c478bd9Sstevel@tonic-gate 39987c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 39997c478bd9Sstevel@tonic-gate 40007c478bd9Sstevel@tonic-gate if ((modname = strrchr(filename, '/')) == NULL) 40017c478bd9Sstevel@tonic-gate modname = filename; 40027c478bd9Sstevel@tonic-gate else 40037c478bd9Sstevel@tonic-gate modname++; 40047c478bd9Sstevel@tonic-gate 40057c478bd9Sstevel@tonic-gate mp = &modules; 40067c478bd9Sstevel@tonic-gate do { 40077c478bd9Sstevel@tonic-gate if (strcmp(modname, mp->mod_modname) == 0) { 40087c478bd9Sstevel@tonic-gate found = 1; 40097c478bd9Sstevel@tonic-gate break; 40107c478bd9Sstevel@tonic-gate } 40117c478bd9Sstevel@tonic-gate } while ((mp = mp->mod_next) != &modules); 40127c478bd9Sstevel@tonic-gate 40137c478bd9Sstevel@tonic-gate if (found == 0) { 40147c478bd9Sstevel@tonic-gate mp = allocate_modp(filename, modname); 40157c478bd9Sstevel@tonic-gate modadd(mp); 40167c478bd9Sstevel@tonic-gate } 40177c478bd9Sstevel@tonic-gate 40187c478bd9Sstevel@tonic-gate /* 40197c478bd9Sstevel@tonic-gate * if dep is not NULL, set the mp in mod_requisite_loading for 40207c478bd9Sstevel@tonic-gate * the module circular dependency check. This field is used in 40217c478bd9Sstevel@tonic-gate * mod_circdep(), but it's cleard in mod_hold_loaded_mod(). 40227c478bd9Sstevel@tonic-gate */ 40237c478bd9Sstevel@tonic-gate if (dep != NULL) { 40247c478bd9Sstevel@tonic-gate ASSERT(dep->mod_busy && dep->mod_requisite_loading == NULL); 40257c478bd9Sstevel@tonic-gate dep->mod_requisite_loading = mp; 40267c478bd9Sstevel@tonic-gate } 40277c478bd9Sstevel@tonic-gate 40287c478bd9Sstevel@tonic-gate /* 40297c478bd9Sstevel@tonic-gate * If the module was held, then it must be us who has it held. 40307c478bd9Sstevel@tonic-gate */ 40317c478bd9Sstevel@tonic-gate if (mod_circdep(mp)) 40327c478bd9Sstevel@tonic-gate mp = NULL; 40337c478bd9Sstevel@tonic-gate else { 40347c478bd9Sstevel@tonic-gate (void) mod_hold_by_modctl(mp, MOD_WAIT_FOREVER | MOD_LOCK_HELD); 40357c478bd9Sstevel@tonic-gate 40367c478bd9Sstevel@tonic-gate /* 40377c478bd9Sstevel@tonic-gate * If the name hadn't been set or has changed, allocate 40387c478bd9Sstevel@tonic-gate * space and set it. Free space used by previous name. 40397c478bd9Sstevel@tonic-gate * 40407c478bd9Sstevel@tonic-gate * Do not change the name of primary modules, for primary 40417c478bd9Sstevel@tonic-gate * modules the mod_filename was allocated in standalone mode: 40427c478bd9Sstevel@tonic-gate * it is illegal to kobj_alloc in standalone mode and kobj_free 40437c478bd9Sstevel@tonic-gate * in non-standalone mode. 40447c478bd9Sstevel@tonic-gate */ 40457c478bd9Sstevel@tonic-gate curname = mp->mod_filename; 40467c478bd9Sstevel@tonic-gate if (curname == NULL || 40477c478bd9Sstevel@tonic-gate ((mp->mod_prim == 0) && 40487c478bd9Sstevel@tonic-gate (curname != filename) && 40497c478bd9Sstevel@tonic-gate (modname != filename) && 40507c478bd9Sstevel@tonic-gate (strcmp(curname, filename) != 0))) { 40517c478bd9Sstevel@tonic-gate newname = kobj_zalloc(strlen(filename) + 1, KM_SLEEP); 40527c478bd9Sstevel@tonic-gate (void) strcpy(newname, filename); 40537c478bd9Sstevel@tonic-gate mp->mod_filename = newname; 40547c478bd9Sstevel@tonic-gate if (curname != NULL) 40557c478bd9Sstevel@tonic-gate kobj_free(curname, strlen(curname) + 1); 40567c478bd9Sstevel@tonic-gate } 40577c478bd9Sstevel@tonic-gate } 40587c478bd9Sstevel@tonic-gate 40597c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 40607c478bd9Sstevel@tonic-gate if (mp && moddebug & MODDEBUG_LOADMSG2) 40617c478bd9Sstevel@tonic-gate printf("Holding %s\n", mp->mod_filename); 40627c478bd9Sstevel@tonic-gate if (mp == NULL && moddebug & MODDEBUG_LOADMSG2) 40637c478bd9Sstevel@tonic-gate printf("circular dependency loading %s\n", filename); 40647c478bd9Sstevel@tonic-gate return (mp); 40657c478bd9Sstevel@tonic-gate } 40667c478bd9Sstevel@tonic-gate 40677c478bd9Sstevel@tonic-gate static struct modctl * 40687c478bd9Sstevel@tonic-gate mod_hold_by_name_requisite(struct modctl *dep, char *filename) 40697c478bd9Sstevel@tonic-gate { 40707c478bd9Sstevel@tonic-gate return (mod_hold_by_name_common(dep, filename)); 40717c478bd9Sstevel@tonic-gate } 40727c478bd9Sstevel@tonic-gate 40737c478bd9Sstevel@tonic-gate struct modctl * 40747aec1d6eScindi mod_hold_by_name(const char *filename) 40757c478bd9Sstevel@tonic-gate { 40767c478bd9Sstevel@tonic-gate return (mod_hold_by_name_common(NULL, filename)); 40777c478bd9Sstevel@tonic-gate } 40787c478bd9Sstevel@tonic-gate 40797aec1d6eScindi struct modctl * 40807c478bd9Sstevel@tonic-gate mod_hold_by_id(modid_t modid) 40817c478bd9Sstevel@tonic-gate { 40827c478bd9Sstevel@tonic-gate struct modctl *mp; 40837c478bd9Sstevel@tonic-gate int found = 0; 40847c478bd9Sstevel@tonic-gate 40857c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 40867c478bd9Sstevel@tonic-gate mp = &modules; 40877c478bd9Sstevel@tonic-gate do { 40887c478bd9Sstevel@tonic-gate if (mp->mod_id == modid) { 40897c478bd9Sstevel@tonic-gate found = 1; 40907c478bd9Sstevel@tonic-gate break; 40917c478bd9Sstevel@tonic-gate } 40927c478bd9Sstevel@tonic-gate } while ((mp = mp->mod_next) != &modules); 40937c478bd9Sstevel@tonic-gate 40947c478bd9Sstevel@tonic-gate if ((found == 0) || mod_circdep(mp)) 40957c478bd9Sstevel@tonic-gate mp = NULL; 40967c478bd9Sstevel@tonic-gate else 40977c478bd9Sstevel@tonic-gate (void) mod_hold_by_modctl(mp, MOD_WAIT_FOREVER | MOD_LOCK_HELD); 40987c478bd9Sstevel@tonic-gate 40997c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 41007c478bd9Sstevel@tonic-gate return (mp); 41017c478bd9Sstevel@tonic-gate } 41027c478bd9Sstevel@tonic-gate 41037c478bd9Sstevel@tonic-gate static struct modctl * 41047c478bd9Sstevel@tonic-gate mod_hold_next_by_id(modid_t modid) 41057c478bd9Sstevel@tonic-gate { 41067c478bd9Sstevel@tonic-gate struct modctl *mp; 41077c478bd9Sstevel@tonic-gate int found = 0; 41087c478bd9Sstevel@tonic-gate 41097c478bd9Sstevel@tonic-gate if (modid < -1) 41107c478bd9Sstevel@tonic-gate return (NULL); 41117c478bd9Sstevel@tonic-gate 41127c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 41137c478bd9Sstevel@tonic-gate 41147c478bd9Sstevel@tonic-gate mp = &modules; 41157c478bd9Sstevel@tonic-gate do { 41167c478bd9Sstevel@tonic-gate if (mp->mod_id > modid) { 41177c478bd9Sstevel@tonic-gate found = 1; 41187c478bd9Sstevel@tonic-gate break; 41197c478bd9Sstevel@tonic-gate } 41207c478bd9Sstevel@tonic-gate } while ((mp = mp->mod_next) != &modules); 41217c478bd9Sstevel@tonic-gate 41227c478bd9Sstevel@tonic-gate if ((found == 0) || mod_circdep(mp)) 41237c478bd9Sstevel@tonic-gate mp = NULL; 41247c478bd9Sstevel@tonic-gate else 41257c478bd9Sstevel@tonic-gate (void) mod_hold_by_modctl(mp, MOD_WAIT_FOREVER | MOD_LOCK_HELD); 41267c478bd9Sstevel@tonic-gate 41277c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 41287c478bd9Sstevel@tonic-gate return (mp); 41297c478bd9Sstevel@tonic-gate } 41307c478bd9Sstevel@tonic-gate 41317c478bd9Sstevel@tonic-gate static void 41327c478bd9Sstevel@tonic-gate mod_release(struct modctl *mp) 41337c478bd9Sstevel@tonic-gate { 41347c478bd9Sstevel@tonic-gate ASSERT(MUTEX_HELD(&mod_lock)); 41357c478bd9Sstevel@tonic-gate ASSERT(mp->mod_busy); 41367c478bd9Sstevel@tonic-gate 41377c478bd9Sstevel@tonic-gate mp->mod_busy = 0; 41387c478bd9Sstevel@tonic-gate mp->mod_inprogress_thread = NULL; 41397c478bd9Sstevel@tonic-gate if (mp->mod_want) { 41407c478bd9Sstevel@tonic-gate mp->mod_want = 0; 41417c478bd9Sstevel@tonic-gate cv_broadcast(&mod_cv); 41427c478bd9Sstevel@tonic-gate } 41437c478bd9Sstevel@tonic-gate } 41447c478bd9Sstevel@tonic-gate 41457c478bd9Sstevel@tonic-gate void 41467c478bd9Sstevel@tonic-gate mod_release_mod(struct modctl *mp) 41477c478bd9Sstevel@tonic-gate { 41487c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_LOADMSG2) 41497c478bd9Sstevel@tonic-gate printf("Releasing %s\n", mp->mod_filename); 41507c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 41517c478bd9Sstevel@tonic-gate mod_release(mp); 41527c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 41537c478bd9Sstevel@tonic-gate } 41547c478bd9Sstevel@tonic-gate 41557c478bd9Sstevel@tonic-gate modid_t 41567c478bd9Sstevel@tonic-gate mod_name_to_modid(char *filename) 41577c478bd9Sstevel@tonic-gate { 41587c478bd9Sstevel@tonic-gate char *modname; 41597c478bd9Sstevel@tonic-gate struct modctl *mp; 41607c478bd9Sstevel@tonic-gate 41617c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 41627c478bd9Sstevel@tonic-gate 41637c478bd9Sstevel@tonic-gate if ((modname = strrchr(filename, '/')) == NULL) 41647c478bd9Sstevel@tonic-gate modname = filename; 41657c478bd9Sstevel@tonic-gate else 41667c478bd9Sstevel@tonic-gate modname++; 41677c478bd9Sstevel@tonic-gate 41687c478bd9Sstevel@tonic-gate mp = &modules; 41697c478bd9Sstevel@tonic-gate do { 41707c478bd9Sstevel@tonic-gate if (strcmp(modname, mp->mod_modname) == 0) { 41717c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 41727c478bd9Sstevel@tonic-gate return (mp->mod_id); 41737c478bd9Sstevel@tonic-gate } 41747c478bd9Sstevel@tonic-gate } while ((mp = mp->mod_next) != &modules); 41757c478bd9Sstevel@tonic-gate 41767c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 41777c478bd9Sstevel@tonic-gate return (-1); 41787c478bd9Sstevel@tonic-gate } 41797c478bd9Sstevel@tonic-gate 41807c478bd9Sstevel@tonic-gate 41817c478bd9Sstevel@tonic-gate int 41827c478bd9Sstevel@tonic-gate mod_remove_by_name(char *name) 41837c478bd9Sstevel@tonic-gate { 41847c478bd9Sstevel@tonic-gate struct modctl *mp; 41857c478bd9Sstevel@tonic-gate int retval; 41867c478bd9Sstevel@tonic-gate 41877c478bd9Sstevel@tonic-gate mp = mod_hold_by_name(name); 41887c478bd9Sstevel@tonic-gate 41897c478bd9Sstevel@tonic-gate if (mp == NULL) 41907c478bd9Sstevel@tonic-gate return (EINVAL); 41917c478bd9Sstevel@tonic-gate 41927c478bd9Sstevel@tonic-gate if (mp->mod_loadflags & MOD_NOAUTOUNLOAD) { 41937c478bd9Sstevel@tonic-gate /* 41947c478bd9Sstevel@tonic-gate * Do not unload forceloaded modules 41957c478bd9Sstevel@tonic-gate */ 41967c478bd9Sstevel@tonic-gate mod_release_mod(mp); 41977c478bd9Sstevel@tonic-gate return (0); 41987c478bd9Sstevel@tonic-gate } 41997c478bd9Sstevel@tonic-gate 42000b38a8bdSahl if ((retval = moduninstall(mp)) == 0) { 42017c478bd9Sstevel@tonic-gate mod_unload(mp); 42020b38a8bdSahl CPU_STATS_ADDQ(CPU, sys, modunload, 1); 42030b38a8bdSahl } else if (retval == EALREADY) 42047c478bd9Sstevel@tonic-gate retval = 0; /* already unloaded, not an error */ 42057c478bd9Sstevel@tonic-gate mod_release_mod(mp); 42067c478bd9Sstevel@tonic-gate return (retval); 42077c478bd9Sstevel@tonic-gate } 42087c478bd9Sstevel@tonic-gate 42097c478bd9Sstevel@tonic-gate /* 42107c478bd9Sstevel@tonic-gate * Record that module "dep" is dependent on module "on_mod." 42117c478bd9Sstevel@tonic-gate */ 42127c478bd9Sstevel@tonic-gate static void 42137c478bd9Sstevel@tonic-gate mod_make_requisite(struct modctl *dependent, struct modctl *on_mod) 42147c478bd9Sstevel@tonic-gate { 42157c478bd9Sstevel@tonic-gate struct modctl_list **pmlnp; /* previous next pointer */ 42167c478bd9Sstevel@tonic-gate struct modctl_list *mlp; 42177c478bd9Sstevel@tonic-gate struct modctl_list *new; 42187c478bd9Sstevel@tonic-gate 42197c478bd9Sstevel@tonic-gate ASSERT(dependent->mod_busy && on_mod->mod_busy); 42207c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); 42217c478bd9Sstevel@tonic-gate 42227c478bd9Sstevel@tonic-gate /* 42237c478bd9Sstevel@tonic-gate * Search dependent's requisite list to see if on_mod is recorded. 42247c478bd9Sstevel@tonic-gate * List is ordered by id. 42257c478bd9Sstevel@tonic-gate */ 42267c478bd9Sstevel@tonic-gate for (pmlnp = &dependent->mod_requisites, mlp = *pmlnp; 42277c478bd9Sstevel@tonic-gate mlp; pmlnp = &mlp->modl_next, mlp = *pmlnp) 42287c478bd9Sstevel@tonic-gate if (mlp->modl_modp->mod_id >= on_mod->mod_id) 42297c478bd9Sstevel@tonic-gate break; 42307c478bd9Sstevel@tonic-gate 42317c478bd9Sstevel@tonic-gate /* Create and insert if not already recorded */ 42327c478bd9Sstevel@tonic-gate if ((mlp == NULL) || (mlp->modl_modp->mod_id != on_mod->mod_id)) { 42337c478bd9Sstevel@tonic-gate new = kobj_zalloc(sizeof (*new), KM_SLEEP); 42347c478bd9Sstevel@tonic-gate new->modl_modp = on_mod; 42357c478bd9Sstevel@tonic-gate new->modl_next = mlp; 42367c478bd9Sstevel@tonic-gate *pmlnp = new; 42377c478bd9Sstevel@tonic-gate 42387c478bd9Sstevel@tonic-gate /* 42397c478bd9Sstevel@tonic-gate * Increment the mod_ref count in our new requisite module. 42407c478bd9Sstevel@tonic-gate * This is what keeps a module that has other modules 42417c478bd9Sstevel@tonic-gate * which are dependent on it from being uninstalled and 42427c478bd9Sstevel@tonic-gate * unloaded. "on_mod"'s mod_ref count decremented in 42437c478bd9Sstevel@tonic-gate * mod_release_requisites when the "dependent" module 42447c478bd9Sstevel@tonic-gate * unload is complete. "on_mod" must be loaded, but may not 42457c478bd9Sstevel@tonic-gate * yet be installed. 42467c478bd9Sstevel@tonic-gate */ 42477c478bd9Sstevel@tonic-gate on_mod->mod_ref++; 42487c478bd9Sstevel@tonic-gate ASSERT(on_mod->mod_ref && on_mod->mod_loaded); 42497c478bd9Sstevel@tonic-gate } 42507c478bd9Sstevel@tonic-gate 42517c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 42527c478bd9Sstevel@tonic-gate } 42537c478bd9Sstevel@tonic-gate 42547c478bd9Sstevel@tonic-gate /* 42557c478bd9Sstevel@tonic-gate * release the hold associated with mod_make_requisite mod_ref++ 42567c478bd9Sstevel@tonic-gate * as part of unload. 42577c478bd9Sstevel@tonic-gate */ 42587c478bd9Sstevel@tonic-gate void 42597c478bd9Sstevel@tonic-gate mod_release_requisites(struct modctl *modp) 42607c478bd9Sstevel@tonic-gate { 42617c478bd9Sstevel@tonic-gate struct modctl_list *modl; 42627c478bd9Sstevel@tonic-gate struct modctl_list *next; 42637c478bd9Sstevel@tonic-gate struct modctl *req; 42647c478bd9Sstevel@tonic-gate struct modctl_list *start = NULL, *mod_garbage; 42657c478bd9Sstevel@tonic-gate 42668a3feaaaSJerry Gilliam ASSERT(!quiesce_active); 42677c478bd9Sstevel@tonic-gate ASSERT(modp->mod_busy); 42688a3feaaaSJerry Gilliam ASSERT(MUTEX_NOT_HELD(&mod_lock)); 42697c478bd9Sstevel@tonic-gate 42707c478bd9Sstevel@tonic-gate mutex_enter(&mod_lock); /* needed for manipulation of req */ 42717c478bd9Sstevel@tonic-gate for (modl = modp->mod_requisites; modl; modl = next) { 42727c478bd9Sstevel@tonic-gate next = modl->modl_next; 42737c478bd9Sstevel@tonic-gate req = modl->modl_modp; 42747c478bd9Sstevel@tonic-gate ASSERT(req->mod_ref >= 1 && req->mod_loaded); 42757c478bd9Sstevel@tonic-gate req->mod_ref--; 42767c478bd9Sstevel@tonic-gate 42777c478bd9Sstevel@tonic-gate /* 42787c478bd9Sstevel@tonic-gate * Check if the module has to be unloaded or not. 42797c478bd9Sstevel@tonic-gate */ 42807c478bd9Sstevel@tonic-gate if (req->mod_ref == 0 && req->mod_delay_unload) { 42817c478bd9Sstevel@tonic-gate struct modctl_list *new; 42827c478bd9Sstevel@tonic-gate /* 42837c478bd9Sstevel@tonic-gate * Allocate the modclt_list holding the garbage 42847c478bd9Sstevel@tonic-gate * module which should be unloaded later. 42857c478bd9Sstevel@tonic-gate */ 42867c478bd9Sstevel@tonic-gate new = kobj_zalloc(sizeof (struct modctl_list), 42877c478bd9Sstevel@tonic-gate KM_SLEEP); 42887c478bd9Sstevel@tonic-gate new->modl_modp = req; 42897c478bd9Sstevel@tonic-gate 42907c478bd9Sstevel@tonic-gate if (start == NULL) 42917c478bd9Sstevel@tonic-gate mod_garbage = start = new; 42927c478bd9Sstevel@tonic-gate else { 42937c478bd9Sstevel@tonic-gate mod_garbage->modl_next = new; 42947c478bd9Sstevel@tonic-gate mod_garbage = new; 42957c478bd9Sstevel@tonic-gate } 42967c478bd9Sstevel@tonic-gate } 42977c478bd9Sstevel@tonic-gate 42987c478bd9Sstevel@tonic-gate /* free the list as we go */ 42997c478bd9Sstevel@tonic-gate kobj_free(modl, sizeof (*modl)); 43007c478bd9Sstevel@tonic-gate } 43017c478bd9Sstevel@tonic-gate modp->mod_requisites = NULL; 43027c478bd9Sstevel@tonic-gate mutex_exit(&mod_lock); 43037c478bd9Sstevel@tonic-gate 43047c478bd9Sstevel@tonic-gate /* 43057c478bd9Sstevel@tonic-gate * Unload the garbage modules. 43067c478bd9Sstevel@tonic-gate */ 43077c478bd9Sstevel@tonic-gate for (mod_garbage = start; mod_garbage != NULL; /* nothing */) { 43087c478bd9Sstevel@tonic-gate struct modctl_list *old = mod_garbage; 43097c478bd9Sstevel@tonic-gate struct modctl *mp = mod_garbage->modl_modp; 43107c478bd9Sstevel@tonic-gate ASSERT(mp != NULL); 43117c478bd9Sstevel@tonic-gate 43127c478bd9Sstevel@tonic-gate /* 43137c478bd9Sstevel@tonic-gate * Hold this module until it's unloaded completely. 43147c478bd9Sstevel@tonic-gate */ 43157c478bd9Sstevel@tonic-gate (void) mod_hold_by_modctl(mp, 43167c478bd9Sstevel@tonic-gate MOD_WAIT_FOREVER | MOD_LOCK_NOT_HELD); 43177c478bd9Sstevel@tonic-gate /* 43187c478bd9Sstevel@tonic-gate * Check if the module is not unloaded yet and nobody requires 43197c478bd9Sstevel@tonic-gate * the module. If it's unloaded already or somebody still 43207c478bd9Sstevel@tonic-gate * requires the module, don't unload it now. 43217c478bd9Sstevel@tonic-gate */ 43227c478bd9Sstevel@tonic-gate if (mp->mod_loaded && mp->mod_ref == 0) 43237c478bd9Sstevel@tonic-gate mod_unload(mp); 43247c478bd9Sstevel@tonic-gate ASSERT((mp->mod_loaded == 0 && mp->mod_delay_unload == 0) || 43257c478bd9Sstevel@tonic-gate (mp->mod_ref > 0)); 43267c478bd9Sstevel@tonic-gate mod_release_mod(mp); 43277c478bd9Sstevel@tonic-gate 43287c478bd9Sstevel@tonic-gate mod_garbage = mod_garbage->modl_next; 43297c478bd9Sstevel@tonic-gate kobj_free(old, sizeof (struct modctl_list)); 43307c478bd9Sstevel@tonic-gate } 43317c478bd9Sstevel@tonic-gate } 43327c478bd9Sstevel@tonic-gate 43337c478bd9Sstevel@tonic-gate /* 43347c478bd9Sstevel@tonic-gate * Process dependency of the module represented by "dep" on the 43357c478bd9Sstevel@tonic-gate * module named by "on." 43367c478bd9Sstevel@tonic-gate * 43377c478bd9Sstevel@tonic-gate * Called from kobj_do_dependents() to load a module "on" on which 43387c478bd9Sstevel@tonic-gate * "dep" depends. 43397c478bd9Sstevel@tonic-gate */ 43407c478bd9Sstevel@tonic-gate struct modctl * 43417c478bd9Sstevel@tonic-gate mod_load_requisite(struct modctl *dep, char *on) 43427c478bd9Sstevel@tonic-gate { 43437c478bd9Sstevel@tonic-gate struct modctl *on_mod; 43447c478bd9Sstevel@tonic-gate int retval; 43457c478bd9Sstevel@tonic-gate 43467c478bd9Sstevel@tonic-gate if ((on_mod = mod_hold_loaded_mod(dep, on, &retval)) != NULL) { 43477c478bd9Sstevel@tonic-gate mod_make_requisite(dep, on_mod); 43487c478bd9Sstevel@tonic-gate } else if (moddebug & MODDEBUG_ERRMSG) { 43497c478bd9Sstevel@tonic-gate printf("error processing %s on which module %s depends\n", 43507c478bd9Sstevel@tonic-gate on, dep->mod_modname); 43517c478bd9Sstevel@tonic-gate } 43527c478bd9Sstevel@tonic-gate return (on_mod); 43537c478bd9Sstevel@tonic-gate } 43547c478bd9Sstevel@tonic-gate 43557c478bd9Sstevel@tonic-gate static int 43567c478bd9Sstevel@tonic-gate mod_install_requisites(struct modctl *modp) 43577c478bd9Sstevel@tonic-gate { 43587c478bd9Sstevel@tonic-gate struct modctl_list *modl; 43597c478bd9Sstevel@tonic-gate struct modctl *req; 43607c478bd9Sstevel@tonic-gate int status = 0; 43617c478bd9Sstevel@tonic-gate 43627c478bd9Sstevel@tonic-gate ASSERT(MUTEX_NOT_HELD(&mod_lock)); 43637c478bd9Sstevel@tonic-gate ASSERT(modp->mod_busy); 43647c478bd9Sstevel@tonic-gate 43657c478bd9Sstevel@tonic-gate for (modl = modp->mod_requisites; modl; modl = modl->modl_next) { 43667c478bd9Sstevel@tonic-gate req = modl->modl_modp; 43677c478bd9Sstevel@tonic-gate (void) mod_hold_by_modctl(req, 43687c478bd9Sstevel@tonic-gate MOD_WAIT_FOREVER | MOD_LOCK_NOT_HELD); 43697c478bd9Sstevel@tonic-gate status = modinstall(req); 43707c478bd9Sstevel@tonic-gate mod_release_mod(req); 43717c478bd9Sstevel@tonic-gate 43727c478bd9Sstevel@tonic-gate if (status != 0) 43737c478bd9Sstevel@tonic-gate break; 43747c478bd9Sstevel@tonic-gate } 43757c478bd9Sstevel@tonic-gate return (status); 43767c478bd9Sstevel@tonic-gate } 43777c478bd9Sstevel@tonic-gate 43787c478bd9Sstevel@tonic-gate /* 43797c478bd9Sstevel@tonic-gate * returns 1 if this thread is doing autounload, 0 otherwise. 43807c478bd9Sstevel@tonic-gate * see mod_uninstall_all. 43817c478bd9Sstevel@tonic-gate */ 43827c478bd9Sstevel@tonic-gate int 43837c478bd9Sstevel@tonic-gate mod_in_autounload() 43847c478bd9Sstevel@tonic-gate { 43857c478bd9Sstevel@tonic-gate return ((int)(uintptr_t)tsd_get(mod_autounload_key)); 43867c478bd9Sstevel@tonic-gate } 43877c478bd9Sstevel@tonic-gate 43887c478bd9Sstevel@tonic-gate /* 43897c478bd9Sstevel@tonic-gate * gmatch adapted from libc, stripping the wchar stuff 43907c478bd9Sstevel@tonic-gate */ 43911aef0e11Sjg #define popchar(p, c) { \ 43927c478bd9Sstevel@tonic-gate c = *p++; \ 43931aef0e11Sjg if (c == 0) { \ 43941aef0e11Sjg return (0); \ 43951aef0e11Sjg } \ 43961aef0e11Sjg } 43977c478bd9Sstevel@tonic-gate 4398facf4a8dSllai1 int 43997c478bd9Sstevel@tonic-gate gmatch(const char *s, const char *p) 44007c478bd9Sstevel@tonic-gate { 44017c478bd9Sstevel@tonic-gate int c, sc; 44027c478bd9Sstevel@tonic-gate int ok, lc, notflag; 44037c478bd9Sstevel@tonic-gate 44047c478bd9Sstevel@tonic-gate sc = *s++; 44057c478bd9Sstevel@tonic-gate c = *p++; 44067c478bd9Sstevel@tonic-gate if (c == 0) 44077c478bd9Sstevel@tonic-gate return (sc == c); /* nothing matches nothing */ 44087c478bd9Sstevel@tonic-gate 44097c478bd9Sstevel@tonic-gate switch (c) { 44107c478bd9Sstevel@tonic-gate case '\\': 44117c478bd9Sstevel@tonic-gate /* skip to quoted character */ 44121e1ddd6cScth popchar(p, c); 44137c478bd9Sstevel@tonic-gate /*FALLTHRU*/ 44147c478bd9Sstevel@tonic-gate 44157c478bd9Sstevel@tonic-gate default: 44167c478bd9Sstevel@tonic-gate /* straight comparison */ 44177c478bd9Sstevel@tonic-gate if (c != sc) 44187c478bd9Sstevel@tonic-gate return (0); 44197c478bd9Sstevel@tonic-gate /*FALLTHRU*/ 44207c478bd9Sstevel@tonic-gate 44217c478bd9Sstevel@tonic-gate case '?': 44227c478bd9Sstevel@tonic-gate /* first char matches, move to remainder */ 44237c478bd9Sstevel@tonic-gate return (sc != '\0' ? gmatch(s, p) : 0); 44247c478bd9Sstevel@tonic-gate 44257c478bd9Sstevel@tonic-gate 44267c478bd9Sstevel@tonic-gate case '*': 44277c478bd9Sstevel@tonic-gate while (*p == '*') 44287c478bd9Sstevel@tonic-gate p++; 44297c478bd9Sstevel@tonic-gate 44307c478bd9Sstevel@tonic-gate /* * matches everything */ 44317c478bd9Sstevel@tonic-gate if (*p == 0) 44327c478bd9Sstevel@tonic-gate return (1); 44337c478bd9Sstevel@tonic-gate 44347c478bd9Sstevel@tonic-gate /* undo skip at the beginning & iterate over substrings */ 44357c478bd9Sstevel@tonic-gate --s; 44367c478bd9Sstevel@tonic-gate while (*s) { 44377c478bd9Sstevel@tonic-gate if (gmatch(s, p)) 44387c478bd9Sstevel@tonic-gate return (1); 44397c478bd9Sstevel@tonic-gate s++; 44407c478bd9Sstevel@tonic-gate } 44417c478bd9Sstevel@tonic-gate return (0); 44427c478bd9Sstevel@tonic-gate 44437c478bd9Sstevel@tonic-gate case '[': 44447c478bd9Sstevel@tonic-gate /* match any char within [] */ 44457c478bd9Sstevel@tonic-gate if (sc == 0) 44467c478bd9Sstevel@tonic-gate return (0); 44477c478bd9Sstevel@tonic-gate 44487c478bd9Sstevel@tonic-gate ok = lc = notflag = 0; 44497c478bd9Sstevel@tonic-gate 44507c478bd9Sstevel@tonic-gate if (*p == '!') { 44517c478bd9Sstevel@tonic-gate notflag = 1; 44527c478bd9Sstevel@tonic-gate p++; 44537c478bd9Sstevel@tonic-gate } 44541e1ddd6cScth popchar(p, c); 44557c478bd9Sstevel@tonic-gate 44567c478bd9Sstevel@tonic-gate do { 44577c478bd9Sstevel@tonic-gate if (c == '-' && lc && *p != ']') { 44587c478bd9Sstevel@tonic-gate /* test sc against range [c1-c2] */ 44591e1ddd6cScth popchar(p, c); 44607c478bd9Sstevel@tonic-gate if (c == '\\') { 44611e1ddd6cScth popchar(p, c); 44627c478bd9Sstevel@tonic-gate } 44637c478bd9Sstevel@tonic-gate 44647c478bd9Sstevel@tonic-gate if (notflag) { 44657c478bd9Sstevel@tonic-gate /* return 0 on mismatch */ 44667c478bd9Sstevel@tonic-gate if (lc <= sc && sc <= c) 44677c478bd9Sstevel@tonic-gate return (0); 44687c478bd9Sstevel@tonic-gate ok++; 44697c478bd9Sstevel@tonic-gate } else if (lc <= sc && sc <= c) { 44707c478bd9Sstevel@tonic-gate ok++; 44717c478bd9Sstevel@tonic-gate } 44727c478bd9Sstevel@tonic-gate /* keep going, may get a match next */ 44737c478bd9Sstevel@tonic-gate } else if (c == '\\') { 44747c478bd9Sstevel@tonic-gate /* skip to quoted character */ 44751e1ddd6cScth popchar(p, c); 44767c478bd9Sstevel@tonic-gate } 44777c478bd9Sstevel@tonic-gate lc = c; 44787c478bd9Sstevel@tonic-gate if (notflag) { 44797c478bd9Sstevel@tonic-gate if (sc == lc) 44807c478bd9Sstevel@tonic-gate return (0); 44817c478bd9Sstevel@tonic-gate ok++; 44827c478bd9Sstevel@tonic-gate } else if (sc == lc) { 44837c478bd9Sstevel@tonic-gate ok++; 44847c478bd9Sstevel@tonic-gate } 44851e1ddd6cScth popchar(p, c); 44867c478bd9Sstevel@tonic-gate } while (c != ']'); 44877c478bd9Sstevel@tonic-gate 44887c478bd9Sstevel@tonic-gate /* recurse on remainder of string */ 44897c478bd9Sstevel@tonic-gate return (ok ? gmatch(s, p) : 0); 44907c478bd9Sstevel@tonic-gate } 44917c478bd9Sstevel@tonic-gate /*NOTREACHED*/ 44927c478bd9Sstevel@tonic-gate } 44937c478bd9Sstevel@tonic-gate 44947c478bd9Sstevel@tonic-gate 44957c478bd9Sstevel@tonic-gate /* 44967c478bd9Sstevel@tonic-gate * Get default perm for device from /etc/minor_perm. Return 0 if match found. 44977c478bd9Sstevel@tonic-gate * 44987c478bd9Sstevel@tonic-gate * Pure wild-carded patterns are handled separately so the ordering of 44997c478bd9Sstevel@tonic-gate * these patterns doesn't matter. We're still dependent on ordering 45007c478bd9Sstevel@tonic-gate * however as the first matching entry is the one returned. 45017c478bd9Sstevel@tonic-gate * Not ideal but all existing examples and usage do imply this 45027c478bd9Sstevel@tonic-gate * ordering implicitly. 45037c478bd9Sstevel@tonic-gate * 45047c478bd9Sstevel@tonic-gate * Drivers using the clone driver are always good for some entertainment. 45057c478bd9Sstevel@tonic-gate * Clone nodes under pseudo have the form clone@0:<driver>. Some minor 45067c478bd9Sstevel@tonic-gate * perm entries have the form clone:<driver>, others use <driver>:* 45077c478bd9Sstevel@tonic-gate * Examples are clone:llc1 vs. llc2:*, for example. 45087c478bd9Sstevel@tonic-gate * 45097c478bd9Sstevel@tonic-gate * Minor perms in the clone:<driver> form are mapped to the drivers's 45107c478bd9Sstevel@tonic-gate * mperm list, not the clone driver, as wildcard entries for clone 45117c478bd9Sstevel@tonic-gate * reference only. In other words, a clone wildcard will match 45127c478bd9Sstevel@tonic-gate * references for clone@0:<driver> but never <driver>@<minor>. 45137c478bd9Sstevel@tonic-gate * 45147c478bd9Sstevel@tonic-gate * Additional minor perms in the standard form are also supported, 45157c478bd9Sstevel@tonic-gate * for mixed usage, ie a node with an entry clone:<driver> could 45167c478bd9Sstevel@tonic-gate * provide further entries <driver>:<minor>. 45177c478bd9Sstevel@tonic-gate * 45187c478bd9Sstevel@tonic-gate * Finally, some uses of clone use an alias as the minor name rather 45197c478bd9Sstevel@tonic-gate * than the driver name, with the alias as the minor perm entry. 45207c478bd9Sstevel@tonic-gate * This case is handled by attaching the driver to bring its 45217c478bd9Sstevel@tonic-gate * minor list into existence, then discover the alias via DDI_ALIAS. 45227c478bd9Sstevel@tonic-gate * The clone device's minor perm list can then be searched for 45237c478bd9Sstevel@tonic-gate * that alias. 45247c478bd9Sstevel@tonic-gate */ 45257c478bd9Sstevel@tonic-gate 45267c478bd9Sstevel@tonic-gate static int 45277c478bd9Sstevel@tonic-gate dev_alias_minorperm(dev_info_t *dip, char *minor_name, mperm_t *rmp) 45287c478bd9Sstevel@tonic-gate { 45297c478bd9Sstevel@tonic-gate major_t major; 45307c478bd9Sstevel@tonic-gate struct devnames *dnp; 45317c478bd9Sstevel@tonic-gate mperm_t *mp; 45327c478bd9Sstevel@tonic-gate char *alias = NULL; 45337c478bd9Sstevel@tonic-gate dev_info_t *cdevi; 4534b9ccdc5aScth int circ; 45357c478bd9Sstevel@tonic-gate struct ddi_minor_data *dmd; 45367c478bd9Sstevel@tonic-gate 45377c478bd9Sstevel@tonic-gate major = ddi_name_to_major(minor_name); 45387c478bd9Sstevel@tonic-gate 45397c478bd9Sstevel@tonic-gate ASSERT(dip == clone_dip); 4540a204de77Scth ASSERT(major != DDI_MAJOR_T_NONE); 45417c478bd9Sstevel@tonic-gate 45427c478bd9Sstevel@tonic-gate /* 45437c478bd9Sstevel@tonic-gate * Attach the driver named by the minor node, then 45447c478bd9Sstevel@tonic-gate * search its first instance's minor list for an 45457c478bd9Sstevel@tonic-gate * alias node. 45467c478bd9Sstevel@tonic-gate */ 45477c478bd9Sstevel@tonic-gate if (ddi_hold_installed_driver(major) == NULL) 45487c478bd9Sstevel@tonic-gate return (1); 45497c478bd9Sstevel@tonic-gate 45507c478bd9Sstevel@tonic-gate dnp = &devnamesp[major]; 45517c478bd9Sstevel@tonic-gate LOCK_DEV_OPS(&dnp->dn_lock); 45527c478bd9Sstevel@tonic-gate 45537c478bd9Sstevel@tonic-gate if ((cdevi = dnp->dn_head) != NULL) { 4554b9ccdc5aScth ndi_devi_enter(cdevi, &circ); 45557c478bd9Sstevel@tonic-gate for (dmd = DEVI(cdevi)->devi_minor; dmd; dmd = dmd->next) { 45567c478bd9Sstevel@tonic-gate if (dmd->type == DDM_ALIAS) { 45577c478bd9Sstevel@tonic-gate alias = i_ddi_strdup(dmd->ddm_name, KM_SLEEP); 45587c478bd9Sstevel@tonic-gate break; 45597c478bd9Sstevel@tonic-gate } 45607c478bd9Sstevel@tonic-gate } 4561b9ccdc5aScth ndi_devi_exit(cdevi, circ); 45627c478bd9Sstevel@tonic-gate } 45637c478bd9Sstevel@tonic-gate 45647c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 45657c478bd9Sstevel@tonic-gate ddi_rele_driver(major); 45667c478bd9Sstevel@tonic-gate 45677c478bd9Sstevel@tonic-gate if (alias == NULL) { 45687c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) 45697c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, "dev_minorperm: " 45707c478bd9Sstevel@tonic-gate "no alias for %s\n", minor_name); 45717c478bd9Sstevel@tonic-gate return (1); 45727c478bd9Sstevel@tonic-gate } 45737c478bd9Sstevel@tonic-gate 45747c478bd9Sstevel@tonic-gate major = ddi_driver_major(clone_dip); 45757c478bd9Sstevel@tonic-gate dnp = &devnamesp[major]; 45767c478bd9Sstevel@tonic-gate LOCK_DEV_OPS(&dnp->dn_lock); 45777c478bd9Sstevel@tonic-gate 45787c478bd9Sstevel@tonic-gate /* 45797c478bd9Sstevel@tonic-gate * Go through the clone driver's mperm list looking 45807c478bd9Sstevel@tonic-gate * for a match for the specified alias. 45817c478bd9Sstevel@tonic-gate */ 45827c478bd9Sstevel@tonic-gate for (mp = dnp->dn_mperm; mp; mp = mp->mp_next) { 45837c478bd9Sstevel@tonic-gate if (strcmp(alias, mp->mp_minorname) == 0) { 45847c478bd9Sstevel@tonic-gate break; 45857c478bd9Sstevel@tonic-gate } 45867c478bd9Sstevel@tonic-gate } 45877c478bd9Sstevel@tonic-gate 45887c478bd9Sstevel@tonic-gate if (mp) { 45897c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MP_MATCH) { 45907c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, 45917c478bd9Sstevel@tonic-gate "minor perm defaults: %s %s 0%o %d %d (aliased)\n", 45927c478bd9Sstevel@tonic-gate minor_name, alias, mp->mp_mode, 45937c478bd9Sstevel@tonic-gate mp->mp_uid, mp->mp_gid); 45947c478bd9Sstevel@tonic-gate } 45957c478bd9Sstevel@tonic-gate rmp->mp_uid = mp->mp_uid; 45967c478bd9Sstevel@tonic-gate rmp->mp_gid = mp->mp_gid; 45977c478bd9Sstevel@tonic-gate rmp->mp_mode = mp->mp_mode; 45987c478bd9Sstevel@tonic-gate } 45997c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 46007c478bd9Sstevel@tonic-gate 46017c478bd9Sstevel@tonic-gate kmem_free(alias, strlen(alias)+1); 46027c478bd9Sstevel@tonic-gate 46037c478bd9Sstevel@tonic-gate return (mp == NULL); 46047c478bd9Sstevel@tonic-gate } 46057c478bd9Sstevel@tonic-gate 46067c478bd9Sstevel@tonic-gate int 46077c478bd9Sstevel@tonic-gate dev_minorperm(dev_info_t *dip, char *name, mperm_t *rmp) 46087c478bd9Sstevel@tonic-gate { 46097c478bd9Sstevel@tonic-gate major_t major; 46107c478bd9Sstevel@tonic-gate char *minor_name; 46117c478bd9Sstevel@tonic-gate struct devnames *dnp; 46127c478bd9Sstevel@tonic-gate mperm_t *mp; 46137c478bd9Sstevel@tonic-gate int is_clone = 0; 46147c478bd9Sstevel@tonic-gate 46157c478bd9Sstevel@tonic-gate if (!minorperm_loaded) { 46167c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) 46177c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, 46187c478bd9Sstevel@tonic-gate "%s: minor perm not yet loaded\n", name); 46197c478bd9Sstevel@tonic-gate return (1); 46207c478bd9Sstevel@tonic-gate } 46217c478bd9Sstevel@tonic-gate 46227c478bd9Sstevel@tonic-gate minor_name = strchr(name, ':'); 46237c478bd9Sstevel@tonic-gate if (minor_name == NULL) 46247c478bd9Sstevel@tonic-gate return (1); 46257c478bd9Sstevel@tonic-gate minor_name++; 46267c478bd9Sstevel@tonic-gate 46277c478bd9Sstevel@tonic-gate /* 46287c478bd9Sstevel@tonic-gate * If it's the clone driver, search the driver as named 46297c478bd9Sstevel@tonic-gate * by the minor. All clone minor perm entries other than 46307c478bd9Sstevel@tonic-gate * alias nodes are actually installed on the real driver's list. 46317c478bd9Sstevel@tonic-gate */ 46327c478bd9Sstevel@tonic-gate if (dip == clone_dip) { 46337c478bd9Sstevel@tonic-gate major = ddi_name_to_major(minor_name); 4634a204de77Scth if (major == DDI_MAJOR_T_NONE) { 46357c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MINORPERM) 46367c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, "dev_minorperm: " 46377c478bd9Sstevel@tonic-gate "%s: no such driver\n", minor_name); 46387c478bd9Sstevel@tonic-gate return (1); 46397c478bd9Sstevel@tonic-gate } 46407c478bd9Sstevel@tonic-gate is_clone = 1; 46417c478bd9Sstevel@tonic-gate } else { 46427c478bd9Sstevel@tonic-gate major = ddi_driver_major(dip); 4643a204de77Scth ASSERT(major != DDI_MAJOR_T_NONE); 46447c478bd9Sstevel@tonic-gate } 46457c478bd9Sstevel@tonic-gate 46467c478bd9Sstevel@tonic-gate dnp = &devnamesp[major]; 46477c478bd9Sstevel@tonic-gate LOCK_DEV_OPS(&dnp->dn_lock); 46487c478bd9Sstevel@tonic-gate 46497c478bd9Sstevel@tonic-gate /* 46507c478bd9Sstevel@tonic-gate * Go through the driver's mperm list looking for 46517c478bd9Sstevel@tonic-gate * a match for the specified minor. If there's 46527c478bd9Sstevel@tonic-gate * no matching pattern, use the wild card. 46537c478bd9Sstevel@tonic-gate * Defer to the clone wild for clone if specified, 46547c478bd9Sstevel@tonic-gate * otherwise fall back to the normal form. 46557c478bd9Sstevel@tonic-gate */ 46567c478bd9Sstevel@tonic-gate for (mp = dnp->dn_mperm; mp; mp = mp->mp_next) { 46577c478bd9Sstevel@tonic-gate if (gmatch(minor_name, mp->mp_minorname) != 0) { 46587c478bd9Sstevel@tonic-gate break; 46597c478bd9Sstevel@tonic-gate } 46607c478bd9Sstevel@tonic-gate } 46617c478bd9Sstevel@tonic-gate if (mp == NULL) { 46627c478bd9Sstevel@tonic-gate if (is_clone) 46637c478bd9Sstevel@tonic-gate mp = dnp->dn_mperm_clone; 46647c478bd9Sstevel@tonic-gate if (mp == NULL) 46657c478bd9Sstevel@tonic-gate mp = dnp->dn_mperm_wild; 46667c478bd9Sstevel@tonic-gate } 46677c478bd9Sstevel@tonic-gate 46687c478bd9Sstevel@tonic-gate if (mp) { 46697c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_MP_MATCH) { 46707c478bd9Sstevel@tonic-gate cmn_err(CE_CONT, 46717c478bd9Sstevel@tonic-gate "minor perm defaults: %s %s 0%o %d %d\n", 46727c478bd9Sstevel@tonic-gate name, mp->mp_minorname, mp->mp_mode, 46737c478bd9Sstevel@tonic-gate mp->mp_uid, mp->mp_gid); 46747c478bd9Sstevel@tonic-gate } 46757c478bd9Sstevel@tonic-gate rmp->mp_uid = mp->mp_uid; 46767c478bd9Sstevel@tonic-gate rmp->mp_gid = mp->mp_gid; 46777c478bd9Sstevel@tonic-gate rmp->mp_mode = mp->mp_mode; 46787c478bd9Sstevel@tonic-gate } 46797c478bd9Sstevel@tonic-gate UNLOCK_DEV_OPS(&dnp->dn_lock); 46807c478bd9Sstevel@tonic-gate 46817c478bd9Sstevel@tonic-gate /* 46827c478bd9Sstevel@tonic-gate * If no match can be found for a clone node, 46837c478bd9Sstevel@tonic-gate * search for a possible match for an alias. 46847c478bd9Sstevel@tonic-gate * One such example is /dev/ptmx -> /devices/pseudo/clone@0:ptm, 46857c478bd9Sstevel@tonic-gate * with minor perm entry clone:ptmx. 46867c478bd9Sstevel@tonic-gate */ 46877c478bd9Sstevel@tonic-gate if (mp == NULL && is_clone) { 46887c478bd9Sstevel@tonic-gate return (dev_alias_minorperm(dip, minor_name, rmp)); 46897c478bd9Sstevel@tonic-gate } 46907c478bd9Sstevel@tonic-gate 46917c478bd9Sstevel@tonic-gate return (mp == NULL); 46927c478bd9Sstevel@tonic-gate } 46937c478bd9Sstevel@tonic-gate 46947c478bd9Sstevel@tonic-gate /* 46957c478bd9Sstevel@tonic-gate * dynamicaly reference load a dl module/library, returning handle 46967c478bd9Sstevel@tonic-gate */ 46977c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 46987c478bd9Sstevel@tonic-gate ddi_modhandle_t 46997c478bd9Sstevel@tonic-gate ddi_modopen(const char *modname, int mode, int *errnop) 47007c478bd9Sstevel@tonic-gate { 47017c478bd9Sstevel@tonic-gate char *subdir; 47027c478bd9Sstevel@tonic-gate char *mod; 47037c478bd9Sstevel@tonic-gate int subdirlen; 47047c478bd9Sstevel@tonic-gate struct modctl *hmodp = NULL; 47057c478bd9Sstevel@tonic-gate int retval = EINVAL; 47067c478bd9Sstevel@tonic-gate 47077c478bd9Sstevel@tonic-gate ASSERT(modname && (mode == KRTLD_MODE_FIRST)); 47087c478bd9Sstevel@tonic-gate if ((modname == NULL) || (mode != KRTLD_MODE_FIRST)) 47097c478bd9Sstevel@tonic-gate goto out; 47107c478bd9Sstevel@tonic-gate 47111e1ddd6cScth /* find last '/' in modname */ 47121e1ddd6cScth mod = strrchr(modname, '/'); 47137c478bd9Sstevel@tonic-gate 47147c478bd9Sstevel@tonic-gate if (mod) { 47157c478bd9Sstevel@tonic-gate /* for subdir string without modification to argument */ 47167c478bd9Sstevel@tonic-gate mod++; 47177c478bd9Sstevel@tonic-gate subdirlen = mod - modname; 47187c478bd9Sstevel@tonic-gate subdir = kmem_alloc(subdirlen, KM_SLEEP); 47197c478bd9Sstevel@tonic-gate (void) strlcpy(subdir, modname, subdirlen); 47207c478bd9Sstevel@tonic-gate } else { 47217c478bd9Sstevel@tonic-gate subdirlen = 0; 47227c478bd9Sstevel@tonic-gate subdir = "misc"; 47237c478bd9Sstevel@tonic-gate mod = (char *)modname; 47247c478bd9Sstevel@tonic-gate } 47257c478bd9Sstevel@tonic-gate 47267c478bd9Sstevel@tonic-gate /* reference load with errno return value */ 47277c478bd9Sstevel@tonic-gate retval = modrload(subdir, mod, &hmodp); 47287c478bd9Sstevel@tonic-gate 47297c478bd9Sstevel@tonic-gate if (subdirlen) 47307c478bd9Sstevel@tonic-gate kmem_free(subdir, subdirlen); 47317c478bd9Sstevel@tonic-gate 47327c478bd9Sstevel@tonic-gate out: if (errnop) 47337c478bd9Sstevel@tonic-gate *errnop = retval; 47347c478bd9Sstevel@tonic-gate 47357c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_DDI_MOD) 47367c478bd9Sstevel@tonic-gate printf("ddi_modopen %s mode %x: %s %p %d\n", 47377c478bd9Sstevel@tonic-gate modname ? modname : "<unknown>", mode, 47387c478bd9Sstevel@tonic-gate hmodp ? hmodp->mod_filename : "<unknown>", 47397c478bd9Sstevel@tonic-gate (void *)hmodp, retval); 47407c478bd9Sstevel@tonic-gate 47417c478bd9Sstevel@tonic-gate return ((ddi_modhandle_t)hmodp); 47427c478bd9Sstevel@tonic-gate } 47437c478bd9Sstevel@tonic-gate 47447c478bd9Sstevel@tonic-gate /* lookup "name" in open dl module/library */ 47457c478bd9Sstevel@tonic-gate void * 47467c478bd9Sstevel@tonic-gate ddi_modsym(ddi_modhandle_t h, const char *name, int *errnop) 47477c478bd9Sstevel@tonic-gate { 47487c478bd9Sstevel@tonic-gate struct modctl *hmodp = (struct modctl *)h; 47497c478bd9Sstevel@tonic-gate void *f; 47507c478bd9Sstevel@tonic-gate int retval; 47517c478bd9Sstevel@tonic-gate 47527c478bd9Sstevel@tonic-gate ASSERT(hmodp && name && hmodp->mod_installed && (hmodp->mod_ref >= 1)); 47537c478bd9Sstevel@tonic-gate if ((hmodp == NULL) || (name == NULL) || 47547c478bd9Sstevel@tonic-gate (hmodp->mod_installed == 0) || (hmodp->mod_ref < 1)) { 47557c478bd9Sstevel@tonic-gate f = NULL; 47567c478bd9Sstevel@tonic-gate retval = EINVAL; 47577c478bd9Sstevel@tonic-gate } else { 47587c478bd9Sstevel@tonic-gate f = (void *)kobj_lookup(hmodp->mod_mp, (char *)name); 47597c478bd9Sstevel@tonic-gate if (f) 47607c478bd9Sstevel@tonic-gate retval = 0; 47617c478bd9Sstevel@tonic-gate else 47627c478bd9Sstevel@tonic-gate retval = ENOTSUP; 47637c478bd9Sstevel@tonic-gate } 47647c478bd9Sstevel@tonic-gate 47657c478bd9Sstevel@tonic-gate if (moddebug & MODDEBUG_DDI_MOD) 47667c478bd9Sstevel@tonic-gate printf("ddi_modsym in %s of %s: %d %p\n", 47677c478bd9Sstevel@tonic-gate hmodp ? hmodp->mod_modname : "<unknown>", 47687c478bd9Sstevel@tonic-gate name ? name : "<unknown>", retval, f); 47697c478bd9Sstevel@tonic-gate 47707c478bd9Sstevel@tonic-gate if (errnop) 47717c478bd9Sstevel@tonic-gate *errnop = retval; 47727c478bd9Sstevel@tonic-gate return (f); 47737c478bd9Sstevel@tonic-gate } 47747c478bd9Sstevel@tonic-gate 47757c478bd9Sstevel@tonic-gate /* dynamic (un)reference unload of an open dl module/library */ 47767c478bd9Sstevel@tonic-gate int 47777c478bd9Sstevel@tonic-gate ddi_modclose(ddi_modhandle_t h) 47787c478bd9Sstevel@tonic-gate { 47797c478bd9Sstevel@tonic-gate struct modctl *hmodp = (struct modctl *)h; 47807c478bd9Sstevel@tonic-gate struct modctl *modp = NULL; 47817c478bd9Sstevel@tonic-gate int retval; 47827c478bd9Sstevel@tonic-gate 47837c478bd9Sstevel@tonic-gate ASSERT(hmodp && hmodp->mod_installed && (hmodp->mod_ref >= 1)); 47847c478bd9Sstevel@tonic-gate if ((hmodp == NULL) || 47857c478bd9Sstevel@tonic-gate (hmodp->mod_installed == 0) || (hmodp->mod_ref < 1)) { 47867c478bd9Sstevel@tonic-gate retval = EINVAL; 47877c478bd9Sstevel@tonic-gate goto out; 47887c478bd9Sstevel@tonic-gate } 47897c478bd9Sstevel@tonic-gate 47907c478bd9Sstevel@tonic-gate retval = modunrload(hmodp->mod_id, &modp, ddi_modclose_unload); 47917c478bd9Sstevel@tonic-gate if (retval == EBUSY) 47927c478bd9Sstevel@tonic-gate retval = 0; /* EBUSY is not an error */ 47937c478bd9Sstevel@tonic-gate 47947c478bd9Sstevel@tonic-gate if (retval == 0) { 47957c478bd9Sstevel@tonic-gate ASSERT(hmodp == modp); 47967c478bd9Sstevel@tonic-gate if (hmodp != modp) 47977c478bd9Sstevel@tonic-gate retval = EINVAL; 47987c478bd9Sstevel@tonic-gate } 47997c478bd9Sstevel@tonic-gate 48007c478bd9Sstevel@tonic-gate out: if (moddebug & MODDEBUG_DDI_MOD) 48017c478bd9Sstevel@tonic-gate printf("ddi_modclose %s: %d\n", 48027c478bd9Sstevel@tonic-gate hmodp ? hmodp->mod_modname : "<unknown>", retval); 48037c478bd9Sstevel@tonic-gate 48047c478bd9Sstevel@tonic-gate return (retval); 48057c478bd9Sstevel@tonic-gate } 4806