xref: /titanic_51/usr/src/uts/common/avs/ns/nsctl/nsctl.c (revision c9f77c52c0735e65aa2534394c5151cdb963cbef)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  * Copyright 2012 Milan Jurik. All rights reserved.
25  */
26 
27 #include <sys/types.h>
28 #include <sys/ksynch.h>
29 #include <sys/kmem.h>
30 #include <sys/file.h>
31 #include <sys/errno.h>
32 #include <sys/open.h>
33 #include <sys/cred.h>
34 #include <sys/conf.h>
35 #include <sys/uio.h>
36 #include <sys/cmn_err.h>
37 #include <sys/modctl.h>
38 #include <sys/ddi.h>
39 
40 #define	__NSC_GEN__
41 #include <sys/nsctl/nsc_dev.h>
42 #include <sys/nsctl/nsc_gen.h>
43 #include <sys/nsctl/nsc_ioctl.h>
44 #include <sys/nsctl/nsc_power.h>
45 #include <sys/nsctl/nsc_mem.h>
46 #include "../nsctl.h"
47 
48 #include <sys/nsctl/nsvers.h>
49 
50 #ifdef DS_DDICT
51 #include "../contract.h"
52 #endif
53 
54 extern void nscsetup();
55 extern int _nsc_init_raw(int);
56 extern void _nsc_deinit_raw();
57 extern void _nsc_init_start();
58 extern void _nsc_init_os(), _nsc_deinit_os();
59 extern void _nsc_init_dev(), _nsc_init_mem();
60 extern void _nsc_init_gen(), _nsc_init_rmlock();
61 extern void _nsc_init_resv(), _nsc_deinit_resv();
62 extern void _nsc_init_frz(), _nsc_deinit_frz();
63 extern void _nsc_init_ncio(), _nsc_deinit_ncio();
64 extern void _nsc_deinit_mem(), _nsc_deinit_rmlock();
65 extern void _nsc_deinit_dev();
66 
67 extern int _nsc_frz_start(char *, int *);
68 extern int _nsc_frz_stop(char *, int *);
69 extern int _nsc_frz_isfrozen(char *, int *);
70 
71 extern nsc_mem_t *_nsc_local_mem;
72 extern nsc_rmhdr_t *_nsc_rmhdr_ptr;
73 extern nsc_def_t _nsc_raw_def[];
74 extern int _nsc_raw_flags;
75 
76 int nsc_devflag = D_MP;
77 
78 int _nsc_init_done = 0;
79 
80 kmutex_t _nsc_drv_lock;
81 nsc_io_t *_nsc_file_io;
82 nsc_io_t *_nsc_vchr_io;
83 nsc_io_t *_nsc_raw_io;
84 
85 nsc_fd_t **_nsc_minor_fd;
86 kmutex_t **_nsc_minor_slp;
87 
88 
89 /* Maximum number of devices - tunable in nsctl.conf */
90 static int _nsc_max_devices;
91 
92 /* Internal version of _nsc_max_devices */
93 int _nsc_maxdev;
94 
95 extern void _nsc_global_setup(void);
96 
97 static int nsc_load(), nsc_unload();
98 static void nscteardown();
99 
100 /*
101  * Solaris specific driver module interface code.
102  */
103 
104 extern int nscopen(dev_t *, int, int, cred_t *);
105 extern int nscioctl(dev_t, int, intptr_t, int, cred_t *, int *);
106 extern int nscclose(dev_t, int, int, cred_t *);
107 extern int nscread(dev_t, uio_t *, cred_t *);
108 extern int nscwrite(dev_t, uio_t *, cred_t *);
109 
110 static dev_info_t *nsctl_dip;		/* Single DIP for driver */
111 
112 static int _nsctl_print(dev_t, char *);
113 
114 static	struct	cb_ops nsctl_cb_ops = {
115 	nscopen,		/* open */
116 	nscclose,	/* close */
117 	nodev,		/* not a block driver, strategy not an entry point */
118 	_nsctl_print,	/* no print routine */
119 	nodev,		/* no dump routine */
120 	nscread,		/* read */
121 	nscwrite,	/* write */
122 	(int (*)()) nscioctl,	/* ioctl */
123 	nodev,		/* no devmap routine */
124 	nodev,		/* no mmap routine */
125 	nodev,		/* no segmap routine */
126 	nochpoll,	/* no chpoll routine */
127 	ddi_prop_op,
128 	0,		/* not a STREAMS driver, no cb_str routine */
129 	D_NEW | D_MP | D_64BIT,	/* safe for multi-thread/multi-processor */
130 	CB_REV,
131 	nodev,		/* aread */
132 	nodev,		/* awrite */
133 };
134 
135 static int _nsctl_getinfo(dev_info_t *, ddi_info_cmd_t, void *, void **);
136 static int _nsctl_attach(dev_info_t *, ddi_attach_cmd_t);
137 static int _nsctl_detach(dev_info_t *, ddi_detach_cmd_t);
138 
139 static	struct	dev_ops nsctl_ops = {
140 	DEVO_REV,			/* Driver build version */
141 	0,				/* device reference count */
142 	_nsctl_getinfo,
143 	nulldev,			/* Identify */
144 	nulldev,			/* Probe */
145 	_nsctl_attach,
146 	_nsctl_detach,
147 	nodev,				/* Reset */
148 	&nsctl_cb_ops,
149 	(struct bus_ops *)0
150 };
151 
152 static struct modldrv nsctl_ldrv = {
153 	&mod_driverops,
154 	"nws:Control:" ISS_VERSION_STR,
155 	&nsctl_ops
156 };
157 
158 static	struct modlinkage nsctl_modlinkage = {
159 	MODREV_1,
160 	&nsctl_ldrv,
161 	NULL
162 };
163 
164 /*
165  * Solaris module load time code
166  */
167 
168 int nsc_min_nodeid;
169 int nsc_max_nodeid;
170 
171 int
172 _init(void)
173 {
174 	int err;
175 
176 	err = nsc_load();
177 
178 	if (!err)
179 		err = mod_install(&nsctl_modlinkage);
180 
181 	if (err) {
182 		(void) nsc_unload();
183 		cmn_err(CE_NOTE, "!nsctl_init: err %d", err);
184 	}
185 
186 	return (err);
187 
188 }
189 
190 /*
191  * Solaris module unload time code
192  */
193 
194 int
195 _fini(void)
196 {
197 	int err;
198 
199 	if ((err = mod_remove(&nsctl_modlinkage)) == 0) {
200 		err = nsc_unload();
201 	}
202 	return (err);
203 }
204 
205 /*
206  * Solaris module info code
207  */
208 int
209 _info(struct modinfo *modinfop)
210 {
211 	return (mod_info(&nsctl_modlinkage, modinfop));
212 }
213 
214 /*
215  * Attach an instance of the device. This happens before an open
216  * can succeed.
217  */
218 static int
219 _nsctl_attach(dev_info_t *dip, ddi_attach_cmd_t cmd)
220 {
221 	int rc;
222 
223 	if (cmd == DDI_ATTACH) {
224 		nsctl_dip = dip;
225 
226 		/* Announce presence of the device */
227 		ddi_report_dev(dip);
228 
229 		/*
230 		 * Get the node parameters now that we can look up.
231 		 */
232 		nsc_min_nodeid = ddi_prop_get_int(DDI_DEV_T_ANY, dip,
233 		    DDI_PROP_DONTPASS | DDI_PROP_NOTPROM,
234 		    "nsc_min_nodeid", 0);
235 
236 		nsc_max_nodeid = ddi_prop_get_int(DDI_DEV_T_ANY, dip,
237 		    DDI_PROP_DONTPASS | DDI_PROP_NOTPROM,
238 		    "nsc_max_nodeid", 5);
239 
240 		_nsc_max_devices = ddi_prop_get_int(DDI_DEV_T_ANY, dip,
241 		    DDI_PROP_DONTPASS | DDI_PROP_NOTPROM,
242 		    "nsc_max_devices", 128);
243 
244 		_nsc_maxdev = _nsc_max_devices;
245 		nscsetup();
246 
247 		/*
248 		 * Init raw requires the _nsc_max_devices value and so
249 		 * cannot be done before the nsc_max_devices property has
250 		 * been read which can only be done after the module is
251 		 * attached and we have a dip.
252 		 */
253 
254 		if ((rc = _nsc_init_raw(_nsc_max_devices)) != 0) {
255 			cmn_err(CE_WARN,
256 			    "!nsctl: unable to initialize raw io provider: %d",
257 			    rc);
258 			return (DDI_FAILURE);
259 		}
260 
261 		/*
262 		 * Init rest of soft state structure
263 		 */
264 
265 		rc = ddi_create_minor_node(dip, "c,nsctl", S_IFCHR, 0,
266 		    DDI_PSEUDO, 0);
267 		if (rc != DDI_SUCCESS) {
268 			/* free anything we allocated here */
269 			cmn_err(CE_WARN,
270 			    "!_nsctl_attach: ddi_create_minor_node failed %d",
271 			    rc);
272 			return (DDI_FAILURE);
273 		}
274 
275 		/* Announce presence of the device */
276 		ddi_report_dev(dip);
277 
278 		/* mark the device as attached, opens may proceed */
279 		return (DDI_SUCCESS);
280 	} else
281 		return (DDI_FAILURE);
282 }
283 
284 static int
285 _nsctl_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
286 {
287 	if (cmd == DDI_DETACH) {
288 		nscteardown();
289 		_nsc_deinit_raw();
290 
291 		ddi_remove_minor_node(dip, NULL);
292 		nsctl_dip = NULL;
293 
294 		return (DDI_SUCCESS);
295 	}
296 	else
297 		return (DDI_FAILURE);
298 }
299 
300 
301 /* ARGSUSED */
302 static int
303 _nsctl_getinfo(dev_info_t *dip, ddi_info_cmd_t cmd, void *arg, void **result)
304 {
305 	dev_t dev;
306 	int rc;
307 
308 	switch (cmd) {
309 		case DDI_INFO_DEVT2INSTANCE:
310 			/* The "instance" number is the minor number */
311 			dev = (dev_t)arg;
312 			*result = (void *)(unsigned long)getminor(dev);
313 			rc = DDI_SUCCESS;
314 			break;
315 
316 		case DDI_INFO_DEVT2DEVINFO:
317 			*result = nsctl_dip;
318 			rc = DDI_SUCCESS;
319 			break;
320 
321 		default:
322 			rc = DDI_FAILURE;
323 			break;
324 	}
325 
326 	return (rc);
327 }
328 
329 
330 /* ARGSUSED */
331 static int
332 _nsctl_print(dev_t dev, char *s)
333 {
334 	cmn_err(CE_WARN, "!nsctl:%s", s);
335 	return (0);
336 }
337 
338 
339 void
340 nsc_init()
341 {
342 	if (_nsc_init_done)
343 		return;
344 
345 	_nsc_init_start();
346 	_nsc_init_gen();
347 	_nsc_init_svc();
348 	_nsc_init_mem();
349 	_nsc_init_dev();
350 	_nsc_init_rmlock();
351 	_nsc_init_resv();
352 	_nsc_init_os();
353 	(void) _nsc_init_power();
354 
355 	/*
356 	 * When using mc, nscsetup is done through mc callback to global_init.
357 	 */
358 	nscsetup();
359 
360 	mutex_init(&_nsc_drv_lock, NULL, MUTEX_DRIVER, NULL);
361 
362 	_nsc_raw_io = nsc_register_io("raw",
363 	    NSC_RAW_ID | _nsc_raw_flags, _nsc_raw_def);
364 
365 	if (!_nsc_raw_io)
366 		cmn_err(CE_WARN, "!_nsc_init: register io failed - raw");
367 
368 	_nsc_init_ncio();
369 	_nsc_init_frz();
370 
371 	_nsc_init_done = 1;
372 }
373 
374 
375 /*
376  * Called after the mc refresh is complete (SEG_INIT callbacks have
377  * been received) and module _attach() is done.  Only does any real
378  * work when all of the above conditions have been met.
379  */
380 void
381 nscsetup()
382 {
383 	if (nsc_max_devices() == 0 || _nsc_minor_fd != NULL)
384 		return;
385 
386 	_nsc_minor_fd = nsc_kmem_zalloc(sizeof (nsc_fd_t *)*_nsc_maxdev,
387 	    0, _nsc_local_mem);
388 
389 	if (!_nsc_minor_fd) {
390 		cmn_err(CE_WARN, "!nscsetup - alloc failed");
391 		return;
392 	}
393 
394 	_nsc_minor_slp = nsc_kmem_zalloc(sizeof (kmutex_t *)*_nsc_maxdev,
395 	    0, _nsc_local_mem);
396 
397 	if (!_nsc_minor_slp)  {
398 		cmn_err(CE_WARN, "!nscsetup - alloc failed");
399 		nsc_kmem_free(_nsc_minor_fd, sizeof (nsc_fd_t *) * _nsc_maxdev);
400 		_nsc_minor_fd = (nsc_fd_t **)NULL;
401 	}
402 }
403 
404 static void
405 nscteardown()
406 {
407 	int i;
408 
409 	if (_nsc_minor_fd == NULL)
410 		return;
411 
412 #ifdef DEBUG
413 	/* Check all devices were closed.  Index 0 is the prototype dev. */
414 	for (i = 1; i < _nsc_maxdev; i++) {
415 		ASSERT(_nsc_minor_slp[i] == NULL);
416 		ASSERT(_nsc_minor_fd[i] == NULL);
417 	}
418 #endif /* DEBUG */
419 
420 	nsc_kmem_free(_nsc_minor_fd, sizeof (nsc_fd_t *) * _nsc_maxdev);
421 	nsc_kmem_free(_nsc_minor_slp, sizeof (kmutex_t *) * _nsc_maxdev);
422 
423 	_nsc_minor_fd = (nsc_fd_t **)NULL;
424 	_nsc_minor_slp = (kmutex_t **)NULL;
425 }
426 
427 int
428 nsc_load()
429 {
430 	nsc_init();
431 	return (0);
432 }
433 
434 
435 int
436 nsc_unload()
437 {
438 	if (!_nsc_init_done) {
439 		return (0);
440 	}
441 
442 	nscteardown();
443 
444 	(void) _nsc_deinit_power();
445 	_nsc_deinit_resv();
446 	_nsc_deinit_mem();
447 	_nsc_deinit_rmlock();
448 	_nsc_deinit_svc();
449 	_nsc_deinit_frz();
450 	_nsc_deinit_ncio();
451 
452 	if (_nsc_vchr_io)
453 		(void) nsc_unregister_io(_nsc_vchr_io, 0);
454 
455 	if (_nsc_file_io)
456 		(void) nsc_unregister_io(_nsc_file_io, 0);
457 
458 	_nsc_vchr_io = NULL;
459 	_nsc_file_io = NULL;
460 
461 	if (_nsc_raw_io)
462 		(void) nsc_unregister_io(_nsc_raw_io, 0);
463 
464 	_nsc_raw_io = NULL;
465 
466 	_nsc_deinit_dev();
467 	_nsc_deinit_os();
468 
469 	_nsc_init_done = 0;
470 	return (0);
471 }
472 
473 
474 /* ARGSUSED */
475 
476 int
477 nscopen(dev_t *devp, int flag, int otyp, cred_t *crp)
478 {
479 	kmutex_t *slp;
480 	int i, error;
481 
482 	if (error = drv_priv(crp))
483 		return (error);
484 
485 	if (!_nsc_minor_fd || !_nsc_minor_slp)
486 		return (ENXIO);
487 
488 	if (getminor(*devp) != 0)
489 		return (ENXIO);
490 
491 	slp = nsc_kmem_alloc(sizeof (kmutex_t), 0, _nsc_local_mem);
492 	mutex_init(slp, NULL, MUTEX_DRIVER, NULL);
493 
494 	mutex_enter(&_nsc_drv_lock);
495 
496 	for (i = 1; i < _nsc_maxdev; i++) {
497 		if (_nsc_minor_slp[i] == NULL) {
498 			_nsc_minor_slp[i] = slp;
499 			break;
500 		}
501 	}
502 
503 	mutex_exit(&_nsc_drv_lock);
504 
505 	if (i >= _nsc_maxdev) {
506 		mutex_destroy(slp);
507 		nsc_kmem_free(slp, sizeof (kmutex_t));
508 		return (EAGAIN);
509 	}
510 
511 	*devp = makedevice(getmajor(*devp), i);
512 
513 	return (0);
514 }
515 
516 
517 int
518 _nscopen(dev_t dev, intptr_t arg, int mode, int *rvp)
519 {
520 	minor_t mindev = getminor(dev);
521 	struct nscioc_open *op;
522 	nsc_fd_t *fd;
523 	int rc;
524 
525 	op = nsc_kmem_alloc(sizeof (*op), KM_SLEEP, _nsc_local_mem);
526 	if (op == NULL) {
527 		return (ENOMEM);
528 	}
529 
530 	if (ddi_copyin((void *)arg, op, sizeof (*op), mode) < 0) {
531 		nsc_kmem_free(op, sizeof (*op));
532 		return (EFAULT);
533 	}
534 
535 	mutex_enter(_nsc_minor_slp[mindev]);
536 
537 	if (_nsc_minor_fd[mindev]) {
538 		mutex_exit(_nsc_minor_slp[mindev]);
539 		nsc_kmem_free(op, sizeof (*op));
540 		return (EBUSY);
541 	}
542 
543 	op->path[sizeof (op->path)-1] = 0;
544 
545 	fd = nsc_open(op->path, (op->flag & NSC_TYPES), 0, 0, &rc);
546 
547 	if (fd == NULL) {
548 		mutex_exit(_nsc_minor_slp[mindev]);
549 		nsc_kmem_free(op, sizeof (*op));
550 		return (rc);
551 	}
552 
553 	mode |= (op->mode - FOPEN);
554 
555 	if (mode & (FWRITE|FEXCL)) {
556 		if ((rc = nsc_reserve(fd, NSC_PCATCH)) != 0) {
557 			mutex_exit(_nsc_minor_slp[mindev]);
558 			(void) nsc_close(fd);
559 			nsc_kmem_free(op, sizeof (*op));
560 			return (rc);
561 		}
562 	}
563 
564 	*rvp = 0;
565 	_nsc_minor_fd[mindev] = fd;
566 
567 	mutex_exit(_nsc_minor_slp[mindev]);
568 	nsc_kmem_free(op, sizeof (*op));
569 	return (0);
570 }
571 
572 
573 /* ARGSUSED */
574 
575 int
576 nscclose(dev_t dev, int flag, int otyp, cred_t *crp)
577 {
578 	minor_t mindev = getminor(dev);
579 	kmutex_t *slp;
580 	nsc_fd_t *fd;
581 
582 	if (!_nsc_minor_fd || !_nsc_minor_slp)
583 		return (0);
584 
585 	if ((slp = _nsc_minor_slp[mindev]) == 0)
586 		return (0);
587 
588 	if ((fd = _nsc_minor_fd[mindev]) != NULL)
589 		(void) nsc_close(fd);
590 
591 	_nsc_minor_fd[mindev] = NULL;
592 	_nsc_minor_slp[mindev] = NULL;
593 
594 	mutex_destroy(slp);
595 	nsc_kmem_free(slp, sizeof (kmutex_t));
596 	return (0);
597 }
598 
599 
600 /* ARGSUSED */
601 
602 int
603 nscread(dev_t dev, uio_t *uiop, cred_t *crp)
604 {
605 	minor_t mindev = getminor(dev);
606 	int rc, resv;
607 	nsc_fd_t *fd;
608 
609 	if ((fd = _nsc_minor_fd[mindev]) == 0)
610 		return (EIO);
611 
612 	mutex_enter(_nsc_minor_slp[mindev]);
613 
614 	resv = (nsc_held(fd) == 0);
615 
616 	if (resv && (rc = nsc_reserve(fd, NSC_PCATCH)) != 0) {
617 		mutex_exit(_nsc_minor_slp[mindev]);
618 		return (rc);
619 	}
620 
621 	rc = nsc_uread(fd, uiop, crp);
622 
623 	if (resv)
624 		nsc_release(fd);
625 
626 	mutex_exit(_nsc_minor_slp[mindev]);
627 	return (rc);
628 }
629 
630 
631 /* ARGSUSED */
632 
633 int
634 nscwrite(dev_t dev, uio_t *uiop, cred_t *crp)
635 {
636 	minor_t mindev = getminor(dev);
637 	int rc, resv;
638 	nsc_fd_t *fd;
639 
640 	if ((fd = _nsc_minor_fd[mindev]) == 0)
641 		return (EIO);
642 
643 	mutex_enter(_nsc_minor_slp[mindev]);
644 
645 	resv = (nsc_held(fd) == 0);
646 
647 	if (resv && (rc = nsc_reserve(fd, NSC_PCATCH)) != 0) {
648 		mutex_exit(_nsc_minor_slp[mindev]);
649 		return (rc);
650 	}
651 
652 	rc = nsc_uwrite(fd, uiop, crp);
653 
654 	if (resv)
655 		nsc_release(fd);
656 
657 	mutex_exit(_nsc_minor_slp[mindev]);
658 	return (rc);
659 }
660 
661 
662 int
663 _nscreserve(dev_t dev, int *rvp)
664 {
665 	minor_t mindev = getminor(dev);
666 	nsc_fd_t *fd;
667 	int rc;
668 
669 	if ((fd = _nsc_minor_fd[mindev]) == 0)
670 		return (EIO);
671 
672 	mutex_enter(_nsc_minor_slp[mindev]);
673 
674 	if (nsc_held(fd)) {
675 		mutex_exit(_nsc_minor_slp[mindev]);
676 		return (EBUSY);
677 	}
678 
679 	if ((rc = nsc_reserve(fd, NSC_PCATCH)) != 0) {
680 		mutex_exit(_nsc_minor_slp[mindev]);
681 		return (rc);
682 	}
683 
684 	*rvp = 0;
685 
686 	mutex_exit(_nsc_minor_slp[mindev]);
687 	return (0);
688 }
689 
690 
691 int
692 _nscrelease(dev_t dev, int *rvp)
693 {
694 	minor_t mindev = getminor(dev);
695 	nsc_fd_t *fd;
696 
697 	if ((fd = _nsc_minor_fd[mindev]) == 0)
698 		return (EIO);
699 
700 	mutex_enter(_nsc_minor_slp[mindev]);
701 
702 	if (!nsc_held(fd)) {
703 		mutex_exit(_nsc_minor_slp[mindev]);
704 		return (EINVAL);
705 	}
706 
707 	nsc_release(fd);
708 
709 	*rvp = 0;
710 
711 	mutex_exit(_nsc_minor_slp[mindev]);
712 	return (0);
713 }
714 
715 
716 int
717 _nscpartsize(dev_t dev, intptr_t arg, int mode)
718 {
719 	struct nscioc_partsize partsize;
720 	minor_t mindev = getminor(dev);
721 	nsc_size_t size;
722 	int rc, resv;
723 	nsc_fd_t *fd;
724 
725 	if ((fd = _nsc_minor_fd[mindev]) == 0)
726 		return (EIO);
727 
728 	mutex_enter(_nsc_minor_slp[mindev]);
729 
730 	resv = (nsc_held(fd) == 0);
731 
732 	if (resv && (rc = nsc_reserve(fd, NSC_PCATCH)) != 0) {
733 		mutex_exit(_nsc_minor_slp[mindev]);
734 		return (rc);
735 	}
736 
737 	rc = nsc_partsize(fd, &size);
738 	partsize.partsize = (uint64_t)size;
739 
740 	if (resv)
741 		nsc_release(fd);
742 
743 	mutex_exit(_nsc_minor_slp[mindev]);
744 
745 	if (ddi_copyout((void *)&partsize, (void *)arg,
746 	    sizeof (partsize), mode) < 0) {
747 		return (EFAULT);
748 	}
749 
750 	return (rc);
751 }
752 
753 
754 /* ARGSUSED */
755 
756 int
757 nscioctl(dev_t dev, int cmd, intptr_t arg, int mode, cred_t *crp, int *rvp)
758 {
759 	struct nscioc_bsize *bsize = NULL;
760 	char *path = NULL;
761 	int rc = 0;
762 
763 	*rvp = 0;
764 
765 	switch (cmd) {
766 	case NSCIOC_OPEN:
767 		rc = _nscopen(dev, arg, mode, rvp);
768 		break;
769 
770 	case NSCIOC_RESERVE:
771 		rc = _nscreserve(dev, rvp);
772 		break;
773 
774 	case NSCIOC_RELEASE:
775 		rc = _nscrelease(dev, rvp);
776 		break;
777 
778 	case NSCIOC_PARTSIZE:
779 		rc = _nscpartsize(dev, arg, mode);
780 		break;
781 
782 	case NSCIOC_FREEZE:
783 		path = nsc_kmem_alloc(NSC_MAXPATH, KM_SLEEP, _nsc_local_mem);
784 		if (path == NULL) {
785 			rc = ENOMEM;
786 			break;
787 		}
788 		if (ddi_copyin((void *)arg, path, NSC_MAXPATH, mode) < 0)
789 			rc = EFAULT;
790 		else {
791 			path[NSC_MAXPATH-1] = 0;
792 			rc = _nsc_frz_start(path, rvp);
793 		}
794 		break;
795 
796 	case NSCIOC_UNFREEZE:
797 		path = nsc_kmem_alloc(NSC_MAXPATH, KM_SLEEP, _nsc_local_mem);
798 		if (path == NULL) {
799 			rc = ENOMEM;
800 			break;
801 		}
802 		if (ddi_copyin((void *)arg, path, NSC_MAXPATH, mode) < 0)
803 			rc = EFAULT;
804 		else {
805 			path[NSC_MAXPATH-1] = 0;
806 			rc = _nsc_frz_stop(path, rvp);
807 		}
808 		break;
809 
810 	case NSCIOC_ISFROZEN:
811 		path = nsc_kmem_alloc(NSC_MAXPATH, KM_SLEEP, _nsc_local_mem);
812 		if (path == NULL) {
813 			rc = ENOMEM;
814 			break;
815 		}
816 		if (ddi_copyin((void *)arg, path, NSC_MAXPATH, mode) < 0)
817 			rc = EFAULT;
818 		else {
819 			path[NSC_MAXPATH-1] = 0;
820 			rc = _nsc_frz_isfrozen(path, rvp);
821 		}
822 		break;
823 
824 #ifdef ENABLE_POWER_MSG
825 	case NSCIOC_POWERMSG:
826 		rc = _nsc_power((void *)arg, rvp);
827 		break;
828 #endif
829 
830 	case NSCIOC_NSKERND:
831 		rc = nskernd_command(arg, mode, rvp);
832 		break;
833 
834 	/* return sizes of global memory segments */
835 	case NSCIOC_GLOBAL_SIZES:
836 		if (!_nsc_init_done) {
837 			rc = EINVAL;
838 			break;
839 		}
840 
841 		rc = _nsc_get_global_sizes((void *)arg, rvp);
842 
843 		break;
844 
845 	/* return contents of global segments */
846 	case NSCIOC_GLOBAL_DATA:
847 		if (!_nsc_init_done) {
848 			rc = EINVAL;
849 			break;
850 		}
851 
852 		rc = _nsc_get_global_data((void *)arg, rvp);
853 		break;
854 
855 	/*
856 	 * nvmem systems:
857 	 * clear the hdr dirty bit to prevent loading from nvme on reboot
858 	 */
859 	case NSCIOC_NVMEM_CLEANF:
860 		rc = _nsc_clear_dirty(1);	/* dont be nice about it */
861 		break;
862 	case NSCIOC_NVMEM_CLEAN:
863 		rc = _nsc_clear_dirty(0);
864 		break;
865 
866 	case NSCIOC_BSIZE:
867 		bsize = nsc_kmem_alloc(sizeof (*bsize), KM_SLEEP,
868 		    _nsc_local_mem);
869 		if (bsize == NULL) {
870 			rc = ENOMEM;
871 			break;
872 		}
873 
874 		if (ddi_copyin((void *)arg, bsize, sizeof (*bsize), mode) < 0) {
875 			rc = EFAULT;
876 			break;
877 		}
878 
879 		rc = nskern_bsize(bsize, rvp);
880 		if (rc == 0) {
881 			if (ddi_copyout(bsize, (void *)arg,
882 			    sizeof (*bsize), mode) < 0) {
883 				rc = EFAULT;
884 				break;
885 			}
886 		}
887 
888 		break;
889 
890 	default:
891 		return (ENOTTY);
892 	}
893 
894 	if (bsize != NULL) {
895 		nsc_kmem_free(bsize, sizeof (*bsize));
896 		bsize = NULL;
897 	}
898 	if (path != NULL) {
899 		nsc_kmem_free(path, NSC_MAXPATH);
900 		path = NULL;
901 	}
902 	return (rc);
903 }
904 
905 
906 int
907 nsc_max_devices(void)
908 {
909 	return (_nsc_max_devices);
910 }
911 
912 
913 /*
914  * Used by _nsc_global_setup() in case nvram is dirty and has saved a different
915  * value for nsc_max_devices. We need to use the saved value, not the new
916  * one configured by the user.
917  */
918 void
919 _nsc_set_max_devices(int maxdev)
920 {
921 	_nsc_max_devices = maxdev;
922 	_nsc_maxdev = _nsc_max_devices;
923 }
924