/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

#include <sys/types.h>
#include <sys/ksynch.h>
#include <sys/kmem.h>
#include <sys/errno.h>
#include <sys/ddi.h>

#include <sys/mdb_modapi.h>

#define	__NSC_GEN__
#include <sys/nsc_thread.h>
#include <sys/nsctl/nsc_dev.h>
#include <sys/nsctl/nsc_gen.h>
#include <sys/nsctl/nsc_mem.h>
#include <sys/nsctl/nsctl.h>
#include <sys/nsctl/nsc_disk.h>


/*
 * Data struct for the complex walks.
 */

struct complex_args {
	int		argc;
	mdb_arg_t	*argv;
};


/*
 * Bit definitions
 */

#define	NSC_RW_BITS	\
	{ "NSC_READ", NSC_READ, NSC_READ },	\
	{ "NSC_WRITE", NSC_WRITE, NSC_WRITE }


static const mdb_bitmask_t nsc_bhflag_bits[] = {
	NSC_RW_BITS,
	{ "NSC_PINNABLE", NSC_PINNABLE, NSC_PINNABLE },
	{ "NSC_NOBLOCK", NSC_NOBLOCK, NSC_NOBLOCK },
	{ "NSC_HALLOCATED", NSC_HALLOCATED, NSC_HALLOCATED },
	{ "NSC_HACTIVE", NSC_HACTIVE, NSC_HACTIVE },
	{ "NSC_BCOPY", NSC_BCOPY, NSC_BCOPY },
	{ "NSC_PAGEIO", NSC_PAGEIO, NSC_PAGEIO },
	{ "NSC_ABUF", NSC_ABUF, NSC_ABUF },
	{ "NSC_MIXED", NSC_MIXED, NSC_MIXED },
	{ "NSC_WRTHRU", NSC_WRTHRU, NSC_WRTHRU },
	{ "NSC_FORCED_WRTHRU", NSC_FORCED_WRTHRU, NSC_FORCED_WRTHRU },
	{ "NSC_NOCACHE", NSC_NOCACHE, NSC_NOCACHE },
	{ "NSC_QUEUE", NSC_QUEUE, NSC_QUEUE },
	{ "NSC_RDAHEAD", NSC_RDAHEAD, NSC_RDAHEAD },
	{ "NSC_NO_FORCED_WRTHRU", NSC_NO_FORCED_WRTHRU, NSC_NO_FORCED_WRTHRU },
	{ "NSC_METADATA", NSC_METADATA, NSC_METADATA },
	{ "NSC_SEQ_IO", NSC_SEQ_IO, NSC_SEQ_IO },
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nsc_fdflag_bits[] = {
	NSC_RW_BITS,
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nsc_fdmode_bits[] = {
	{ "NSC_MULTI", NSC_MULTI, NSC_MULTI },
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nsc_type_bits[] = {
	/* types */
	{ "NSC_NULL", NSC_NULL, NSC_NULL },
	{ "NSC_DEVICE", NSC_DEVICE, NSC_DEVICE },
	{ "NSC_FILE", NSC_FILE, NSC_FILE },
	{ "NSC_CACHE", NSC_CACHE, NSC_CACHE },
	{ "NSC_VCHR", NSC_VCHR, NSC_VCHR },
	{ "NSC_NCALL", NSC_NCALL, NSC_NCALL },

	/* type flags */
	{ "NSC_ANON", NSC_ANON, NSC_ANON },

	/* ids */
	{ "NSC_RAW_ID", NSC_RAW_ID, NSC_RAW_ID },
	{ "NSC_FILE_ID", NSC_FILE_ID, NSC_FILE_ID },
	{ "NSC_FREEZE_ID", NSC_FREEZE_ID, NSC_FREEZE_ID },
	{ "NSC_VCHR_ID", NSC_VCHR_ID, NSC_VCHR_ID },
	{ "NSC_NCALL_ID", NSC_NCALL_ID, NSC_NCALL_ID },
	{ "NSC_SDBC_ID", NSC_SDBC_ID, NSC_SDBC_ID },
	{ "NSC_RDCLR_ID", NSC_RDCLR_ID, NSC_RDCLR_ID },
	{ "NSC_RDCL_ID", NSC_RDCL_ID, NSC_RDCL_ID },
	{ "NSC_IIR_ID", NSC_IIR_ID, NSC_IIR_ID },
	{ "NSC_II_ID", NSC_II_ID, NSC_II_ID },
	{ "NSC_RDCHR_ID", NSC_RDCHR_ID, NSC_RDCHR_ID },
	{ "NSC_RDCH_ID", NSC_RDCH_ID, NSC_RDCH_ID },
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nsc_availpend_bits[] = {
	NSC_RW_BITS,
	{ "_NSC_OPEN", _NSC_OPEN, _NSC_OPEN },
	{ "_NSC_CLOSE", _NSC_CLOSE, _NSC_CLOSE },
	{ "_NSC_PINNED", _NSC_PINNED, _NSC_PINNED },
	{ "_NSC_ATTACH", _NSC_ATTACH, _NSC_ATTACH },
	{ "_NSC_DETACH", _NSC_DETACH, _NSC_DETACH },
	{ "_NSC_OWNER", _NSC_OWNER, _NSC_OWNER },
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nsc_ioflag_bits[] = {
	{ "NSC_REFCNT", NSC_REFCNT, NSC_REFCNT },
	{ "NSC_FILTER", NSC_FILTER, NSC_FILTER },
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nstset_flag_bits[] = {
	{ "NST_SF_KILL", NST_SF_KILL, NST_SF_KILL },
	{ NULL, 0, 0 }
};


static const mdb_bitmask_t nst_flag_bits[] = {
	{ "NST_TF_INUSE", NST_TF_INUSE, NST_TF_INUSE },
	{ "NST_TF_ACTIVE", NST_TF_ACTIVE, NST_TF_ACTIVE },
	{ "NST_TF_PENDING", NST_TF_PENDING, NST_TF_PENDING },
	{ "NST_TF_DESTROY", NST_TF_DESTROY, NST_TF_DESTROY },
	{ "NST_TF_KILL", NST_TF_KILL, NST_TF_KILL },
	{ NULL, 0, 0 }
};


/*
 * Global data.
 */

static nsc_mem_t type_mem[20];
static int complex_walk;
static int complex_hdr;


/* ---------------------------------------------------------------------- */

/*
 * Walker for an nsc_io chain.
 * A global walk is assumed to start at _nsc_io_top.
 */

static int
nsc_io_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL &&
	    mdb_readvar(&wsp->walk_addr, "_nsc_io_top") == -1) {
		mdb_warn("unable to read '_nsc_io_top'");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_io_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t next;
	int status;

	if (wsp->walk_addr == NULL)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	next = wsp->walk_addr + OFFSETOF(nsc_io_t, next);

	if (mdb_vread(&wsp->walk_addr, sizeof (uintptr_t), next) == -1) {
		mdb_warn("failed to read nsc_io_t.next at %p", next);
		return (WALK_DONE);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for an nsc_dev chain.
 * A global walk is assumed to start at _nsc_dev_top.
 */

static int
nsc_dev_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL &&
	    mdb_readvar(&wsp->walk_addr, "_nsc_dev_top") == -1) {
		mdb_warn("unable to read '_nsc_dev_top'");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_dev_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t next;
	int status;

	if (wsp->walk_addr == NULL)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	next = wsp->walk_addr + OFFSETOF(nsc_dev_t, nsc_next);

	if (mdb_vread(&wsp->walk_addr, sizeof (uintptr_t), next) == -1) {
		mdb_warn("failed to read nsc_dev_t.nsc_next at %p", next);
		return (WALK_DONE);
	}

	return (status);
}


/* ARGSUSED */

static void
nsc_dev_wfini(mdb_walk_state_t *wsp)
{
	complex_walk = 0;
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_devval_t structures.
 * Global walks start from _nsc_devval_top;
 */

static int
nsc_devval_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL &&
	    mdb_readvar(&wsp->walk_addr, "_nsc_devval_top") == -1) {
		mdb_warn("unable to read '_nsc_devval_top'");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_devval_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t devval = wsp->walk_addr;
	int status;

	if (!devval)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next devval */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    devval + OFFSETOF(nsc_devval_t, dv_next)) == -1) {
		mdb_warn("failed to read nsc_devval_t.dv_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_fd_t structures.
 * No global walks.
 */

static int
nsc_fd_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL) {
		mdb_warn("nsc_fd doesn't support global walks");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_fd_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t fd = wsp->walk_addr;
	int status;

	if (!fd)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next fd */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    fd + OFFSETOF(nsc_fd_t, sf_next)) == -1) {
		mdb_warn("failed to read nsc_fd_t.sf_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_iodev_t structures.
 * No global walks.
 */

static int
nsc_iodev_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL) {
		mdb_warn("nsc_iodev doesn't support global walks");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_iodev_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t iodev = wsp->walk_addr;
	int status;

	if (!iodev)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    iodev + OFFSETOF(nsc_iodev_t, si_next)) == -1) {
		mdb_warn("failed to read nsc_iodev_t.si_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_service_t structures.
 * Global walks start at _nsc_services.
 */

static int
nsc_service_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL &&
	    mdb_readvar(&wsp->walk_addr, "_nsc_services") == -1) {
		mdb_warn("unable to read '_nsc_services'");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_service_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t service = wsp->walk_addr;
	int status;

	if (!service)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next service */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    service + OFFSETOF(nsc_service_t, s_next)) == -1) {
		mdb_warn("failed to read nsc_service_t.s_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_svc_t structures.
 * No global walks.
 */

static int
nsc_svc_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL) {
		mdb_warn("nsc_svc does not support global walks");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_svc_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t svc = wsp->walk_addr;
	int status;

	if (!svc)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next svc */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    svc + OFFSETOF(nsc_svc_t, svc_next)) == -1) {
		mdb_warn("failed to read nsc_svc_t.svc_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_val_t structures.
 * No global walks.
 */

static int
nsc_val_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL) {
		mdb_warn("nsc_val doesn't support global walks");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_val_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t val = wsp->walk_addr;
	int status;

	if (!val)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next val */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    val + OFFSETOF(nsc_val_t, sv_next)) == -1) {
		mdb_warn("failed to read nsc_val_t.sv_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nstset_t structures.
 * Global walks start at _nst_sets.
 */

static int
nstset_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL &&
	    mdb_readvar(&wsp->walk_addr, "nst_sets") == -1) {
		mdb_warn("unable to read 'nst_sets'");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nstset_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t set = wsp->walk_addr;
	int status;

	if (!set)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next set */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    set + OFFSETOF(nstset_t, set_next)) == -1) {
		mdb_warn("failed to read nstset_t.set_next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsthread_t structures.
 * No global walks.
 */

static int
nsthread_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL) {
		mdb_warn("nsthread does not support global walks");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsthread_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t thread = wsp->walk_addr;
	int status;

	if (!thread)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next iodev */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    thread + OFFSETOF(nsthread_t, tp_chain)) == -1) {
		mdb_warn("failed to read nsthread_t.tp_chain");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for nsthread_t free/reuse chain.
 * No global walks.
 */

static int
nst_free_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL) {
		mdb_warn("nst_free does not support global walks");
		return (WALK_ERR);
	}

	/* store starting address */

	wsp->walk_data = (void *)wsp->walk_addr;

	/* move on to next thread */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    wsp->walk_addr + OFFSETOF(nsthread_t, tp_link.q_forw)) == -1) {
		mdb_warn("failed to read nsthread_t.tp_link.q_forw");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nst_free_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t thread = wsp->walk_addr;
	int status;

	if (!thread)
		return (WALK_DONE);

	if (thread == (uintptr_t)wsp->walk_data)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next thread */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    thread + OFFSETOF(nsthread_t, tp_link.q_forw)) == -1) {
		mdb_warn("failed to read nsthread_t.tp_link.q_forw");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

/*
 * Walker for a chain of nsc_mem_t structures.
 * Global walks start at _nsc_mem_top.
 */

static int
nsc_mem_winit(mdb_walk_state_t *wsp)
{
	if (wsp->walk_addr == NULL &&
	    mdb_readvar(&wsp->walk_addr, "_nsc_mem_top") == -1) {
		mdb_warn("unable to read '_nsc_mem_top'");
		return (WALK_ERR);
	}

	return (WALK_NEXT);
}


static int
nsc_mem_wstep(mdb_walk_state_t *wsp)
{
	uintptr_t mem = wsp->walk_addr;
	int status;

	if (!mem)
		return (WALK_DONE);

	status = wsp->walk_callback(wsp->walk_addr, wsp->walk_data,
	    wsp->walk_cbdata);

	/* move on to next mem */

	if (mdb_vread(&wsp->walk_addr, sizeof (wsp->walk_addr),
	    mem + OFFSETOF(nsc_mem_t, next)) == -1) {
		mdb_warn("failed to read nsc_mem_t.next");
		return (WALK_ERR);
	}

	return (status);
}


/* ---------------------------------------------------------------------- */

struct {
	char	*name;
	int	id;
} io_ids[] = {
	{ "NSC_RAW_ID", NSC_RAW_ID },
	{ "NSC_FILE_ID", NSC_FILE_ID },
	{ "NSC_FREEZE_ID", NSC_FREEZE_ID },
	{ "NSC_SDBC_ID", NSC_SDBC_ID },
	{ "NSC_RDCLR_ID", NSC_RDCLR_ID },
	{ "NSC_RDCL_ID", NSC_RDCL_ID },
	{ "NSC_IIR_ID", NSC_IIR_ID },
	{ "NSC_II_ID", NSC_II_ID },
	{ "NSC_RDCHR_ID", NSC_RDCHR_ID },
	{ "NSC_RDCH_ID", NSC_RDCH_ID },
	{ NULL, 0 }
};


static char *
nsc_io_id(const int id)
{
	int i;

	for (i = 0; io_ids[i].name != NULL; i++) {
		if (io_ids[i].id == id) {
			return (io_ids[i].name);
		}
	}

	return ("unknown");
}


/*
 * Display a single nsc_io_t structure.
 * If called with no address, performs a global walk of all nsc_ios.
 */
static int
nsc_io(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	char io_name[128];
	nsc_io_t *io;
	int v_opt;

	v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		if (mdb_walk_dcmd("nsctl`nsc_io",
		    "nsctl`nsc_io", argc, argv) == -1) {
			mdb_warn("failed to walk 'nsc_io'");
			return (DCMD_ERR);
		}

		return (DCMD_OK);
	}

	io = mdb_zalloc(sizeof (*io), UM_SLEEP | UM_GC);
	memset(io_name, 0, sizeof (io_name));

	if (mdb_vread(io, sizeof (*io), addr) != sizeof (*io)) {
		mdb_warn("failed to read nsc_io at %p", addr);
		return (DCMD_ERR);
	}

	if (io->name) {
		if (mdb_readstr(io_name, sizeof (io_name),
		    (uintptr_t)io->name) == -1) {
			mdb_warn("failed to read nsc_io_t.name");
			return (DCMD_ERR);
		}
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8Tid       fl  ref abuf name\n", "io");
	}

	mdb_printf("%0?p  %8T%08x %2x %4d %4d %s\n",
	    addr, io->id, io->flag, io->refcnt, io->abufcnt, io_name);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("id: %08x <%s>\n", io->id, nsc_io_id(io->id));

	mdb_printf("provide: %08x <%b>\n", io->provide,
	    io->provide, nsc_type_bits);

	mdb_printf("flag: %08x <%b>\n", io->flag, io->flag, nsc_ioflag_bits);

	mdb_printf("pend: %d\n", io->pend);

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Display a single nsc_dev_t structure.
 * If called with no address, performs a global walk of all nsc_devs.
 */
static int
nsc_dev(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	char path[NSC_MAXPATH+1];
	nsc_devval_t *dv;
	nsc_dev_t *dev;
	uintptr_t dev_pend;
	int a_opt, v_opt;

	a_opt = v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'a', MDB_OPT_SETBITS, TRUE, &a_opt,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		mdb_printf("Active device structures:\n");

		if (mdb_walk_dcmd("nsctl`nsc_dev",
		    "nsctl`nsc_dev", argc, argv) == -1) {
			mdb_warn("failed to walk 'nsc_dev'");
			return (DCMD_ERR);
		}

		if (a_opt) {
			if (mdb_readvar(&dev_pend, "_nsc_dev_pend") == -1) {
				mdb_warn("failed to read _nsc_dev_pend");
				return (DCMD_ERR);
			}

			mdb_printf("\nPending device structures:");

			if (dev_pend) {
				mdb_printf("\n");

				if (mdb_pwalk_dcmd("nsctl`nsc_dev",
				    "nsctl`nsc_dev", argc, argv,
				    dev_pend) == -1) {
					mdb_warn("failed to walk "
					    "pending dev structs");
					return (DCMD_ERR);
				}
			} else {
				mdb_printf(" none\n");
			}
		}

		return (DCMD_OK);
	}

	memset(path, 0, sizeof (path));
	dev = mdb_zalloc(sizeof (*dev), UM_SLEEP | UM_GC);

	if (mdb_vread(dev, sizeof (*dev), addr) != sizeof (*dev)) {
		mdb_warn("failed to read nsc_dev at %p", addr);
		return (DCMD_ERR);
	}

	if (mdb_readstr(path, sizeof (path), (uintptr_t)dev->nsc_path) == -1) {
		mdb_warn("failed to read nsc_path at %p", dev->nsc_path);
		return (DCMD_ERR);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8Tref pend rpnd wait path\n", "dev");
	}

	mdb_printf("%0?p  %8T%3d %4d %4d %4d %s\n",
	    addr, dev->nsc_refcnt, dev->nsc_pend, dev->nsc_rpend,
	    dev->nsc_wait, path);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("next: %0?p  %8Tclose: %0?p\n",
	    dev->nsc_next, dev->nsc_close);

	mdb_printf("list: %0?p  %8Tlock: %0?p\n",
	    dev->nsc_list, addr + OFFSETOF(nsc_dev_t, nsc_lock));

	mdb_printf("cv: %0?p  %8Tpath: %0?p  %8Tphash: %016llx\n",
	    addr + OFFSETOF(nsc_dev_t, nsc_cv),
	    dev->nsc_path, dev->nsc_phash);

	mdb_printf("drop: %d  %8Treopen: %d\n",
	    dev->nsc_drop, dev->nsc_reopen);

	if (dev->nsc_values) {
		dv = mdb_zalloc(sizeof (*dv), UM_SLEEP | UM_GC);
		if (mdb_vread(dv, sizeof (*dv), (uintptr_t)dev->nsc_values) !=
		    sizeof (*dv)) {
			mdb_warn("unable to read nsc_dev_t.nsc_values");
			mdb_dec_indent(4);
			return (DCMD_ERR);
		}

		if (dv->dv_values) {
			mdb_printf("device/values: (nsc_devval: %0?p)\n",
			    dev->nsc_values);

			mdb_inc_indent(4);

			if (mdb_pwalk_dcmd("nsctl`nsc_val", "nsctl`nsc_val",
			    0, NULL, (uintptr_t)dv->dv_values) == -1) {
				mdb_dec_indent(8);
				return (DCMD_ERR);
			}

			mdb_dec_indent(4);
		}
	}

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Display a single nsc_devval_t structure.
 * If called with no address, performs a global walk of all nsc_devs.
 */
static int
nsc_devval(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_devval_t *dv;
	int a_opt;

	a_opt = 0;

	if (mdb_getopts(argc, argv,
	    'a', MDB_OPT_SETBITS, TRUE, &a_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		if (mdb_walk_dcmd("nsctl`nsc_devval",
		    "nsctl`nsc_devval", argc, argv) == -1) {
			mdb_warn("failed to walk 'nsc_devval'");
			return (DCMD_ERR);
		}

		return (DCMD_OK);
	}

	dv = mdb_zalloc(sizeof (*dv), UM_SLEEP | UM_GC);

	if (mdb_vread(dv, sizeof (*dv), addr) != sizeof (*dv)) {
		mdb_warn("failed to read nsc_devval at %p", addr);
		return (DCMD_ERR);
	}

	if (!a_opt && !dv->dv_values) {
		return (DCMD_OK);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8T%?-s  %8Tpath\n", "devval", "phash");
	}

	mdb_printf("%0?p  %8T%016llx  %8T%s\n", addr,
	    dv->dv_phash, dv->dv_path);

	mdb_inc_indent(4);

	if (dv->dv_values) {
		if (mdb_pwalk_dcmd("nsctl`nsc_val", "nsctl`nsc_val",
		    0, NULL, (uintptr_t)dv->dv_values) == -1) {
			return (DCMD_ERR);
		}
	} else {
		mdb_printf("No values\n");
	}

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Part 2 callback for the all devices and fds walk.  Called per iodev.
 */
/* ARGSUSED */
static int
nsc_fd_iodev(uintptr_t addr, const void *data, void *cbdata)
{
	struct complex_args *fdall = cbdata;
	struct nsc_fd_t *fd;

	if (mdb_vread(&fd, sizeof (fd),
	    addr + OFFSETOF(nsc_iodev_t, si_open)) == -1) {
		mdb_warn("unable to read nsc_iodev_t.si_open");
		return (WALK_ERR);
	}

	if (fd != NULL) {
		if (mdb_pwalk_dcmd("nsctl`nsc_fd", "nsctl`nsc_fd",
		    fdall->argc, fdall->argv, (uintptr_t)fd) == -1)
			return (WALK_ERR);
	}

	return (WALK_NEXT);
}


/*
 * Part 1 callback for the all devices and fds walk.  Called per device.
 */
/* ARGSUSED */
static int
nsc_fd_dev(uintptr_t addr, const void *data, void *cbdata)
{
	struct complex_args *fdall = cbdata;
	nsc_iodev_t *iodev;
	nsc_fd_t *fd;

	if (mdb_vread(&iodev, sizeof (iodev),
	    addr + OFFSETOF(nsc_dev_t, nsc_list)) == -1) {
		mdb_warn("unable to read nsc_dev_t.nsc_list at %p", addr);
		return (WALK_ERR);
	}

	/* walk iodev chains */

	if (iodev != NULL) {
		if (mdb_pwalk("nsctl`nsc_iodev",
		    nsc_fd_iodev, fdall, (uintptr_t)iodev) == -1)
			return (WALK_ERR);
	}

	/* walk nsc_close (closing fds) chains */

	if (mdb_vread(&fd, sizeof (fd),
	    addr + OFFSETOF(nsc_dev_t, nsc_close)) == -1) {
		mdb_warn("unable to read nsc_dev_t.nsc_close at %p", addr);
		return (WALK_ERR);
	}

	if (fd != NULL) {
		if (mdb_pwalk_dcmd("nsctl`nsc_fd", "nsctl`nsc_fd",
		    fdall->argc, fdall->argv, (uintptr_t)fd) == -1)
			return (WALK_ERR);
	}

	return (WALK_NEXT);
}


/*
 * Walk all devices and fds in the system.
 */
static int
nsc_fd_all(int argc, const mdb_arg_t *argv)
{
	struct complex_args fdall;

	fdall.argc = argc;
	fdall.argv = (mdb_arg_t *)argv;

	complex_walk = 1;
	complex_hdr = 0;

	if (mdb_walk("nsctl`nsc_dev", nsc_fd_dev, &fdall) == -1) {
		return (DCMD_ERR);
	}

	return (DCMD_OK);
}



/*
 * Display an nsd_fd_t structure, or walk all devices and fds in the system.
 */
static int
nsc_fd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	char io_name[128], *io_namep;
	char path[NSC_MAXPATH+1];
	uintptr_t pathp;
	nsc_fd_t *fd;
	nsc_io_t *io;
	int v_opt;
	int hdr;

	v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		return (nsc_fd_all(argc, argv));
	}

	memset(path, 0, sizeof (path));
	fd = mdb_zalloc(sizeof (*fd), UM_SLEEP | UM_GC);
	memset(io_name, 0, sizeof (io_name));

	if (mdb_vread(fd, sizeof (*fd), addr) != sizeof (*fd)) {
		mdb_warn("failed to read nsc_fd at %p", addr);
		return (DCMD_ERR);
	}

	if (mdb_vread(&pathp, sizeof (pathp),
	    (uintptr_t)fd->sf_dev + OFFSETOF(nsc_dev_t, nsc_path)) !=
	    sizeof (pathp)) {
		mdb_warn("failed to read nsc_dev.nsc_path");
		return (DCMD_ERR);
	}

	if (mdb_readstr(path, sizeof (path), pathp) == -1) {
		mdb_warn("failed to read nsc_path");
		return (DCMD_ERR);
	}

	if (fd->sf_iodev) {
		if (mdb_vread(&io, sizeof (io),
		    (uintptr_t)fd->sf_iodev + OFFSETOF(nsc_iodev_t, si_io)) !=
		    sizeof (io)) {
			mdb_warn("failed to read nsc_iodev.si_io");
			return (DCMD_ERR);
		}

		if (mdb_vread(&io_namep, sizeof (io_namep),
		    (uintptr_t)io + OFFSETOF(nsc_io_t, name)) !=
		    sizeof (io_namep)) {
			mdb_warn("failed to read nsc_io_t.name");
			return (DCMD_ERR);
		}

		if (mdb_readstr(io_name, sizeof (io_name),
		    (uintptr_t)io_namep) == -1) {
			mdb_warn("failed to read nsc_io_t.name string");
			return (DCMD_ERR);
		}
	}

	hdr = 0;
	if (complex_walk) {
		if (!complex_hdr) {
			complex_hdr = 1;
			hdr = 1;
		}
	} else if (DCMD_HDRSPEC(flags)) {
		hdr = 1;
	}

	if (hdr) {
		mdb_printf("%-?s  %8T%-?s  %8T%-8s  %-?s\n",
		    "fd", "dev", "io", "cd");
		mdb_printf("    %-?s  %8Trv pend av path\n", "arg");
	}

	mdb_printf("%0?p  %8T%0?p  %8T%-8s  %p\n",
	    addr, fd->sf_dev, io_name, fd->sf_cd);
	mdb_printf("    %0?p  %8T%2d %4x %2x %s\n",
	    fd->sf_arg, fd->sf_reserve, fd->sf_pend,
	    fd->sf_avail, path);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("open type: %08x <%b>\n", fd->sf_type,
	    fd->sf_type, nsc_type_bits);

	mdb_printf("avail: %08x <%b>\n", fd->sf_avail,
	    fd->sf_avail, nsc_availpend_bits);

	mdb_printf("flag: %08x <%b>\n", fd->sf_flag,
	    fd->sf_flag, nsc_fdflag_bits);

	mdb_printf("rsrv mode: %08x <%b>\n", fd->sf_mode,
	    fd->sf_mode, nsc_fdmode_bits);

	mdb_printf("open lbolt: %?x  %8Treopen: %d\n", fd->sf_lbolt,
	    fd->sf_reopen);

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Callback for the all devices and iodevs walk.  Called per device.
 */
/* ARGSUSED */
static int
nsc_iodev_dev(uintptr_t addr, const void *data, void *cbdata)
{
	struct complex_args *iodevall = cbdata;
	uintptr_t iodev;

	if (mdb_vread(&iodev, sizeof (iodev),
	    addr + OFFSETOF(nsc_dev_t, nsc_list)) == -1) {
		mdb_warn("unable to read nsc_dev_t.nsc_list at %p", addr);
		return (WALK_ERR);
	}

	/* walk iodev chains */

	if (iodev != NULL) {
		if (mdb_pwalk_dcmd("nsctl`nsc_iodev", "nsctl`nsc_iodev",
		    iodevall->argc, iodevall->argv, iodev) == -1)
			return (WALK_ERR);
	}

	return (WALK_NEXT);
}


/*
 * Walk all devices and iodevs in the system.
 */
static int
nsc_iodev_all(int argc, const mdb_arg_t *argv)
{
	struct complex_args iodevall;

	iodevall.argc = argc;
	iodevall.argv = (mdb_arg_t *)argv;

	complex_walk = 1;
	complex_hdr = 0;

	if (mdb_walk("nsctl`nsc_dev", nsc_iodev_dev, &iodevall) == -1) {
		return (DCMD_ERR);
	}

	return (DCMD_OK);
}


/*
 * Display an nsc_iodev_t structure, or walk all devices and
 * iodevs in the system.
 */
static int
nsc_iodev(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	char io_name[128], *io_namep;
	char path[NSC_MAXPATH+1];
	nsc_iodev_t *iodev;
	uintptr_t pathp;
	int v_opt;
	int hdr;

	v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		return (nsc_iodev_all(argc, argv));
	}

	memset(path, 0, sizeof (path));
	iodev = mdb_zalloc(sizeof (*iodev), UM_SLEEP | UM_GC);
	memset(io_name, 0, sizeof (io_name));

	if (mdb_vread(iodev, sizeof (*iodev), addr) != sizeof (*iodev)) {
		mdb_warn("failed to read nsc_iodev at %p", addr);
		return (DCMD_ERR);
	}

	if (mdb_vread(&pathp, sizeof (pathp),
	    (uintptr_t)iodev->si_dev + OFFSETOF(nsc_dev_t, nsc_path)) !=
	    sizeof (pathp)) {
		mdb_warn("failed to read nsc_dev.nsc_path");
		return (DCMD_ERR);
	}

	if (mdb_readstr(path, sizeof (path), pathp) == -1) {
		mdb_warn("failed to read nsc_path");
		return (DCMD_ERR);
	}

	if (mdb_vread(&io_namep, sizeof (io_namep),
	    (uintptr_t)iodev->si_io + OFFSETOF(nsc_io_t, name)) !=
	    sizeof (io_namep)) {
		mdb_warn("failed to read nsc_io_t.name");
		return (DCMD_ERR);
	}

	if (mdb_readstr(io_name, sizeof (io_name),
	    (uintptr_t)io_namep) == -1) {
		mdb_warn("failed to read nsc_io_t.name string");
		return (DCMD_ERR);
	}

	hdr = 0;
	if (complex_walk) {
		if (!complex_hdr) {
			complex_hdr = 1;
			hdr = 1;
		}
	} else if (DCMD_HDRSPEC(flags)) {
		hdr = 1;
	}

	if (hdr) {
		mdb_printf("%-?s  %8T%-?s  ref %-8s path\n",
		    "iodev", "dev", "io");
	}

	mdb_printf("%0?p  %8T%0?p  %3d %-8s %s\n",
	    addr, iodev->si_dev, iodev->si_refcnt, io_name, path);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("open fds: %?p  %8Tactive ios: %?p\n",
	    iodev->si_open, iodev->si_active);

	mdb_printf("busy: %d  %8Trsrv pend: %d\n",
	    iodev->si_busy, iodev->si_rpend);

	mdb_printf("pend: %08x <%b>\n", iodev->si_pend,
	    iodev->si_pend, nsc_availpend_bits);

	mdb_printf("avail: %08x <%b>\n", iodev->si_avail,
	    iodev->si_avail, nsc_availpend_bits);

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Display an nsc_service_t structure, or walk all services.
 */
static int
nsc_service(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_service_t *service;
	char s_name[32];
	int v_opt;

	v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		if (mdb_walk_dcmd("nsctl`nsc_service",
		    "nsctl`nsc_service", argc, argv) == -1) {
			mdb_warn("failed to walk 'nsc_service'");
			return (DCMD_ERR);
		}

		return (DCMD_OK);
	}

	service = mdb_zalloc(sizeof (*service), UM_SLEEP | UM_GC);

	if (mdb_vread(service, sizeof (*service), addr) != sizeof (*service)) {
		mdb_warn("failed to read nsc_service at %p", addr);
		return (DCMD_ERR);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8Tname\n", "service");
	}

	memset(s_name, 0, sizeof (s_name));
	if (service->s_name) {
		if (mdb_readstr(s_name, sizeof (s_name),
		    (uintptr_t)service->s_name) == -1) {
			mdb_warn("failed to read nsc_io_t.name");
			return (DCMD_ERR);
		}
	}

	mdb_printf("%0?p  %8T%s\n", addr, s_name);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("servers:\n");
	if (service->s_servers == NULL) {
		mdb_printf("<none>\n");
	} else {
		mdb_inc_indent(4);
		if (mdb_pwalk_dcmd("nsctl`nsc_svc", "nsctl`nsc_svc",
		    argc, argv, (uintptr_t)service->s_servers) == -1) {
			mdb_dec_indent(8);
			return (DCMD_ERR);
		}
		mdb_dec_indent(4);
	}

	mdb_printf("clients:\n");
	if (service->s_clients == NULL) {
		mdb_printf("<none>\n");
	} else {
		mdb_inc_indent(4);
		if (mdb_pwalk_dcmd("nsctl`nsc_svc", "nsctl`nsc_svc",
		    argc, argv, (uintptr_t)service->s_clients) == -1) {
			mdb_dec_indent(8);
			return (DCMD_ERR);
		}
		mdb_dec_indent(4);
	}

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Display an nsc_svc_t structure.
 */
/*ARGSUSED*/
static int
nsc_svc(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_svc_t *svc;

	if (!(flags & DCMD_ADDRSPEC))
		return (DCMD_USAGE);

	svc = mdb_zalloc(sizeof (*svc), UM_SLEEP | UM_GC);

	if (mdb_vread(svc, sizeof (*svc), addr) != sizeof (*svc)) {
		mdb_warn("failed to read nsc_svc at %p", addr);
		return (DCMD_ERR);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8T%-?s  %8Tfunc\n", "svc", "service");
	}

	mdb_printf("%0?p  %8T%0?p  %8T%a\n", addr, svc->svc_svc, svc->svc_fn);
	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Display a single nsc_val_t structure.
 * If called with no address, performs a global walk of all nsc_devs.
 */
/* ARGSUSED3 */
static int
nsc_val(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_val_t *vp;

	if (argc != 0)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		mdb_warn("nsc_val requires an address");
		return (DCMD_ERR);
	}

	vp = mdb_zalloc(sizeof (*vp), UM_SLEEP | UM_GC);

	if (mdb_vread(vp, sizeof (*vp), addr) != sizeof (*vp)) {
		mdb_warn("failed to read nsc_val at %p", addr);
		return (DCMD_ERR);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8T%8-s  %8Tname\n", "val", "value");
	}

	mdb_printf("%0?p  %8T%08x  %8T%s\n", addr, vp->sv_value, vp->sv_name);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Display an nstset_t structure, or walk all sets.
 */

static int
nstset(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nstset_t *set;
	int f_opt, r_opt, t_opt, v_opt;

	f_opt = r_opt = t_opt = v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'f', MDB_OPT_SETBITS, TRUE, &f_opt,		/* free list */
	    'r', MDB_OPT_SETBITS, TRUE, &r_opt,		/* reuse list */
	    't', MDB_OPT_SETBITS, TRUE, &t_opt,		/* all threads */
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	/* displaying threads implies verbose */
	if (f_opt || r_opt || t_opt)
		v_opt = 1;

	if (!(flags & DCMD_ADDRSPEC)) {
		if (mdb_walk_dcmd("nsctl`nstset",
		    "nsctl`nstset", argc, argv) == -1) {
			mdb_warn("failed to walk 'nstset'");
			return (DCMD_ERR);
		}

		return (DCMD_OK);
	}

	set = mdb_zalloc(sizeof (*set), UM_SLEEP | UM_GC);

	if (mdb_vread(set, sizeof (*set), addr) != sizeof (*set)) {
		mdb_warn("failed to read nstset at %p", addr);
		return (DCMD_ERR);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8T  live   nthr flag name\n", "set");
	}

	mdb_printf("%0?p  %8T%6d %6d %4x %s\n", addr,
	    set->set_nlive, set->set_nthread, set->set_flag, set->set_name);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("chain: %0?p  %8Tpending: %4d  res_cnt: %4d\n",
	    set->set_chain, set->set_pending, set->set_res_cnt);

	if (set->set_reuse.q_forw == set->set_reuse.q_back &&
	    (uintptr_t)set->set_reuse.q_forw ==
	    (addr + OFFSETOF(nstset_t, set_reuse))) {
		mdb_printf("reuse.forw: %-?s  %8Treuse.back: %s\n",
		    "empty", "empty");
	} else {
		mdb_printf("reuse.forw: %0?p  %8Treuse.back: %0?p\n",
		    set->set_reuse.q_forw, set->set_reuse.q_back);

		/* display all threads in reuse list */
		if (r_opt &&
		    mdb_pwalk_dcmd("nsctl`nst_free", "nsctl`nsthread",
		    0, (const mdb_arg_t *)NULL,
		    (addr + OFFSETOF(nstset_t, set_reuse))) == -1) {
			mdb_dec_indent(4);
			return (DCMD_ERR);
		}
	}

	if (set->set_free.q_forw == set->set_free.q_back &&
	    (uintptr_t)set->set_free.q_forw ==
	    (addr + OFFSETOF(nstset_t, set_free))) {
		mdb_printf("free.forw:  %-?s  %8Tfree.back:  %s\n",
		    "empty", "empty");
	} else {
		mdb_printf("free.forw:  %0?p  %8Tfree.back:  %0?p\n",
		    set->set_free.q_forw, set->set_free.q_back);

		/* display all threads in free list */
		if (f_opt &&
		    mdb_pwalk_dcmd("nsctl`nst_free", "nsctl`nsthread",
		    0, (const mdb_arg_t *)NULL,
		    (addr + OFFSETOF(nstset_t, set_free))) == -1) {
			mdb_dec_indent(4);
			return (DCMD_ERR);
		}
	}

	mdb_printf("flag: %08x <%b>\n",
	    set->set_flag, set->set_flag, nstset_flag_bits);

	/* display all threads in set */
	if (t_opt) {
		mdb_printf("all threads in set:\n");
		if (mdb_pwalk_dcmd("nsctl`nsthread", "nsctl`nsthread",
		    0, (const mdb_arg_t *)NULL,
		    (uintptr_t)set->set_chain) == -1) {
			mdb_dec_indent(4);
			return (DCMD_ERR);
		}
	}

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

/*
 * Callback for the all nstsets and threads walk.  Called per set.
 */
/* ARGSUSED */
static int
nst_thr_set(uintptr_t addr, const void *data, void *cbdata)
{
	struct complex_args *thrall = cbdata;
	char set_name[48];
	uintptr_t tp;

	if (mdb_vread(&tp, sizeof (tp),
	    addr + OFFSETOF(nstset_t, set_chain)) == -1) {
		mdb_warn("unable to read nstset_t.set_chain at %p", addr);
		return (WALK_ERR);
	}

	memset(set_name, 0, sizeof (set_name));

	if (mdb_readstr(set_name, sizeof (set_name),
	    addr + OFFSETOF(nstset_t, set_name)) == -1) {
		mdb_warn("unable to read nstset_t.set_name at %p", addr);
	}

	mdb_printf("nstset: %0?p (%s)\n", addr, set_name);

	/* walk thread chains */

	if (tp != NULL) {
		if (mdb_pwalk_dcmd("nsctl`nsthread", "nsctl`nsthread",
		    thrall->argc, thrall->argv, tp) == -1)
			return (WALK_ERR);
	} else
		mdb_printf("    no threads\n");

	mdb_printf("\n");

	return (WALK_NEXT);
}


/*
 * Walk all nstsets and threads in the system.
 */
static int
nst_thr_all(int argc, const mdb_arg_t *argv)
{
	struct complex_args thrall;

	thrall.argc = argc;
	thrall.argv = (mdb_arg_t *)argv;

	if (mdb_walk("nsctl`nstset", nst_thr_set, &thrall) == -1)
		return (DCMD_ERR);

	return (DCMD_OK);
}


/*
 * Display an nsthread_t structure, or walk all threads.
 */

static int
nsthread(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	uintptr_t thrpend;
	nsthread_t *tp;
	int a_opt, v_opt;
	int rc;

	a_opt = v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'a', MDB_OPT_SETBITS, TRUE, &a_opt,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		if ((rc = nst_thr_all(argc, argv)) != DCMD_OK)
			return (rc);

		if (a_opt) {
			if (mdb_readvar(&thrpend, "nst_pending") == -1) {
				mdb_warn("unable to read 'nst_pending'");
				return (DCMD_ERR);
			}

			if (thrpend) {
				mdb_printf("\nPending threads:\n");

				if (mdb_pwalk_dcmd("nsctl`nsthread",
				    "nsctl`nsthread", argc, argv,
				    thrpend) == -1) {
					mdb_warn("failed to walk 'nsthread'");
					return (DCMD_ERR);
				}
			}
		}

		return (DCMD_OK);
	}

	tp = mdb_zalloc(sizeof (*tp), UM_SLEEP | UM_GC);

	if (mdb_vread(tp, sizeof (*tp), addr) != sizeof (*tp)) {
		mdb_warn("failed to read nsthread at %p", addr);
		return (DCMD_ERR);
	}

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("%-?s  %8Tflag %-?s  %8Tfunc\n", "thread", "arg");
	}

	mdb_printf("%0?p  %8T%4x %0?p  %8T%a\n",
	    addr, tp->tp_flag, tp->tp_arg, tp->tp_func);

	if (!v_opt)
		return (DCMD_OK);

	mdb_inc_indent(4);

	mdb_printf("set: %0?p  %8Tchain: %0?p\n",
	    tp->tp_set, tp->tp_chain);

	mdb_printf("link.forw: %0?p  %8Tlink.back: %0?p\n",
	    tp->tp_link.q_forw, tp->tp_link.q_back);

	mdb_printf("flag: %08x <%b>\n",
	    tp->tp_flag, tp->tp_flag, nst_flag_bits);

	mdb_dec_indent(4);

	return (DCMD_OK);
}


/* ---------------------------------------------------------------------- */

static void
nsc_rmap(char *name)
{
	nsc_rmmap_t slot;
	uintptr_t addr;
	int nslot;
	char *cp;

	if (mdb_readvar(&addr, name) == -1) {
		mdb_warn("unable to read rmap '%s'", name);
		return;
	}

	if (mdb_vread(&slot, sizeof (slot), addr) != sizeof (slot)) {
		mdb_warn("unable to read rmap '%s' slot 0", name);
		return;
	}

	mdb_printf("\nmap name		offset      size    nslot\n");
	mdb_printf("%16s     %9d %9d    %5d\n",
	    slot.name, slot.offset, slot.size, slot.inuse);

	nslot = slot.inuse;
	mdb_printf("\nslot name	       offset      size    inuse\n");

	while (--nslot) {
		addr += sizeof (slot);

		if (mdb_vread(&slot, sizeof (slot), addr) != sizeof (slot)) {
			mdb_warn("unable to read rmap '%s' slot @ %p",
			    name, addr);
			return;
		}

		if (!slot.inuse || !slot.size)
			continue;

		for (cp = slot.name; *cp; cp++)
			if (*cp == ':')
				*cp = ' ';

		mdb_printf("%16s     %9d %9d %08x\n",
		    slot.name, slot.offset, slot.size, slot.inuse);
	}
}


static void
nsc_rmhdr(void)
{
	nsc_rmhdr_t *rmhdr = mdb_zalloc(sizeof (*rmhdr), UM_SLEEP | UM_GC);
	uintptr_t addr;

	if (mdb_readvar(&addr, "_nsc_rmhdr_ptr") == -1) {
		mdb_warn("unable to read _nsc_rmhdr_ptr");
		return;
	}

	if (!addr) {
		mdb_printf("\n\nGlobal header not initialised\n");
		return;
	}

	if (mdb_vread(rmhdr, sizeof (*rmhdr), addr) != sizeof (*rmhdr)) {
		mdb_warn("unable to read global header at %p", addr);
		return;
	}

	mdb_printf("\n\nglobal header    (magic %08x, version %d, size %d)\n",
	    rmhdr->magic, rmhdr->ver, rmhdr->size);

	nsc_rmap("_nsc_global_map");
}


static nsc_mem_t *
memptr(int type, int flag)
{
	int i;

	type &= NSC_MEM_GLOBAL;

	if (type)
		flag = 0;

	if (!type && !flag)
		return (&type_mem[0]);

	for (i = 1; i < (sizeof (type_mem) / sizeof (nsc_mem_t)); i++) {
		if (!type_mem[i].flag && !type_mem[i].type) {
			type_mem[i].flag = flag;
			type_mem[i].type = type;
			return (&type_mem[i]);
		}

		if (type_mem[i].flag == flag && type_mem[i].type == type)
			return (&type_mem[i]);
	}

	return (&type_mem[i]);
}


#define	typename(t)	\
		(((t) & NSC_MEM_GLOBAL) ? "gbl" : " - ")

#define	memname(t)	\
		(((t) & NSC_MEM_GLOBAL) ? "nsc_global" : "system kmem")

static void
nsc_mem_type(const int first, nsc_mem_t *mp)
{
	char *type, *name;

	if (first) {
		mdb_printf("\nregion	   typ  f      ");
		mdb_printf("used       hwm    pgs  alloc  free\n");
	}

	type = typename(mp->type);
	name = memname(mp->type);

	mdb_printf("%16s %s %2x %9d %9d %6d  %5d %5d\n",
	    name, type, mp->flag, mp->used, mp->hwm, mp->pagehwm,
	    mp->nalloc, mp->nfree);
}


static int
nsc_mem_all(int argc, const mdb_arg_t *argv, int v_opt)
{
	int first;
	int i;

	memset(type_mem, 0, sizeof (type_mem));

	if (mdb_walk_dcmd("nsctl`nsc_mem",
	    "nsctl`nsc_mem", argc, argv) == -1) {
		mdb_warn("unable to walk 'nsc_mem'");
		return (DCMD_ERR);
	}

	for (first = 1, i = 0;
	    i < (sizeof (type_mem) / sizeof (nsc_mem_t)); first = 0, i++) {
		if (type_mem[i].nalloc || type_mem[i].hwm) {
			nsc_mem_type(first, &type_mem[i]);
		}
	}

	if (v_opt)
		nsc_rmhdr();

	return (DCMD_OK);
}


static int
nsc_mem(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	char name[16], *type, *cp;
	nsc_mem_t mem, *mp;
	int v_opt;

	v_opt = 0;

	if (mdb_getopts(argc, argv,
	    'v', MDB_OPT_SETBITS, TRUE, &v_opt) != argc)
		return (DCMD_USAGE);

	if (!(flags & DCMD_ADDRSPEC)) {
		return (nsc_mem_all(argc, argv, v_opt));
	}

	if (mdb_vread(&mem, sizeof (mem), addr) != sizeof (mem)) {
		mdb_warn("failed to read nsc_mem_t at %p", addr);
		return (DCMD_ERR);
	}

	if (mdb_readstr(name, sizeof (name), (uintptr_t)mem.name) == -1) {
		mdb_warn("failed to read nsc_mem_t.name at %p", addr);
		return (DCMD_ERR);
	}

	if (!mem.nalloc && !mem.hwm && !v_opt)
		return (DCMD_OK);

	if (DCMD_HDRSPEC(flags)) {
		mdb_printf("name	     typ  f      ");
		mdb_printf("used       hwm   pgs alloc  free     base\n");
	}

	type = typename(mem.type);
	mp = memptr(mem.type, mem.flag);

	for (cp = name; *cp; cp++)
		if (*cp == ':')
			*cp = ' ';

	mdb_printf("%-16s %s %2x %9d %9d %5d %5d %5d %0?p\n",
	    name, type, mem.flag, mem.used, mem.hwm, mem.pagehwm,
	    mem.nalloc, mem.nfree, mem.base);

	mp->used += mem.used;
	mp->hwm += mem.hwm;
	mp->pagehwm += mem.pagehwm;
	mp->nalloc += mem.nalloc;
	mp->nfree += mem.nfree;

	return (DCMD_OK);
}

/*ARGSUSED*/
static int
nsc_vec(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_vec_t *vec;

	vec = mdb_zalloc(sizeof (*vec), UM_SLEEP | UM_GC);
	if (mdb_vread(vec, sizeof (*vec), addr) != sizeof (*vec)) {
		mdb_warn("failed to read nsc_vec at %p", addr);
		return (DCMD_ERR);
	}
	mdb_printf("nsc_vec_t @ 0x%p = {\n", addr);
	mdb_inc_indent(4);
	mdb_printf("sv_addr: %p\n", vec->sv_addr);
	mdb_printf("sv_vme:  %lu\n", vec->sv_vme);
	mdb_printf("sv_len:  %d\n", vec->sv_len);
	mdb_dec_indent(4);
	mdb_printf("};\n");
	if (vec->sv_addr)
		return (DCMD_OK);
	else
		return (DCMD_ERR);
}

/* ---------------------------------------------------------------------- */
/*
 * Display an nsc_buf_t structure.
 */

#ifdef NSC_MULTI_TERABYTE
#define	STRCONV	"ll"
#else
#define	STRCONV	""
#endif

/* ARGSUSED */
static int
nsc_buf(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_buf_t *bh;
	nsc_vec_t *v;

	if (!(flags & DCMD_ADDRSPEC))
		return (DCMD_USAGE);

	bh = mdb_zalloc(sizeof (*bh), UM_SLEEP | UM_GC);

	if (mdb_vread(bh, sizeof (*bh), addr) != sizeof (*bh)) {
		mdb_warn("failed to read nsc_buf at %p", addr);
		return (DCMD_ERR);
	}

	mdb_printf("nsc_buf_t @ 0x%p = {\n", addr);
	mdb_inc_indent(4);
	mdb_printf("sb_fd:    0x%p\n", bh->sb_fd);
	mdb_printf("sb_pos:   0x%" STRCONV "x\n", bh->sb_pos);
	mdb_printf("sb_len:   0x%" STRCONV "x\n", bh->sb_len);
	mdb_printf("sb_flag:  0x%08x <%b>\n", bh->sb_flag,
	    bh->sb_flag, nsc_bhflag_bits);
	mdb_printf("sb_error: %d\n", bh->sb_error);
#ifdef NSC_MULTI_TERABYTE
	mdb_printf("sb_user:  0x%p\n", bh->sb_user);
#else
	mdb_printf("sb_user:  0x%x\n", bh->sb_user);
#endif
	mdb_printf("sb_vec:   0x%p\n", bh->sb_vec);
	v = bh->sb_vec++;
	while (nsc_vec((uintptr_t)v, flags, argc, argv) == DCMD_OK)
		v++;

	mdb_dec_indent(4);
	mdb_printf("};\n");

	return (DCMD_OK);
}

/* ---------------------------------------------------------------------- */

/* ARGSUSED */
static int
nsc_dbuf(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
	nsc_dbuf_t *bh;

	if (!(flags & DCMD_ADDRSPEC))
		return (DCMD_USAGE);

	bh = mdb_zalloc(sizeof (*bh), UM_SLEEP | UM_GC);

	if (mdb_vread(bh, sizeof (*bh), addr) != sizeof (*bh)) {
		mdb_warn("failed to read nsc_dbuf at %p", addr);
		return (DCMD_ERR);
	}

	mdb_printf("nsc_dbuf_t @ 0x%p = {\n", addr);
	mdb_inc_indent(4);
	mdb_printf("db_disc:    0x%p\n", bh->db_disc);
	mdb_printf("db_addr:    0x%p\n", bh->db_addr);
	mdb_printf("db_next:    0x%p\n", bh->db_next);
	mdb_printf("db_maxfbas: 0x%d\n", bh->db_maxfbas);


	mdb_dec_indent(4);
	mdb_printf("};\n");

	return (DCMD_OK);
}
/* ---------------------------------------------------------------------- */

/*
 * MDB module linkage information:
 */

static const mdb_dcmd_t dcmds[] = {
#if 0
	{ "nsctl", NULL, "display nsctl module info", nsctl },
#endif
	{ "nsc_buf", ":", "list nsc_buf structure", nsc_buf },
	{ "nsc_dbuf", ":", "list nsc_dbuf structure", nsc_dbuf },
	{ "nsc_dev", "?[-av]", "list nsc_dev structure", nsc_dev },
	{ "nsc_devval", "?[-a]", "list nsc_devval structure", nsc_devval },
	{ "nsc_fd", "?[-v]", "list nsc_fd structure", nsc_fd },
	{ "nsc_iodev", "?[-v]", "list nsc_iodev structure", nsc_iodev },
	{ "nsc_io", "?[-v]", "list nsc_io structure", nsc_io },
	{ "nsc_mem", "?[-v]", "list nsc_mem structure", nsc_mem },
	{ "nsc_svc", ":", "list nsc_svc structure", nsc_svc },
	{ "nsc_service", "?[-v]", "list nsc_service structure", nsc_service },
	{ "nsc_val", ":", "list nsc_val structure", nsc_val },
	{ "nstset", "?[-frtv]", "list nstset structure", nstset },
	{ "nsthread", "?[-av]", "list nsthread structure", nsthread },
	{ NULL }
};


static const mdb_walker_t walkers[] = {
	{ "nsc_dev", "walk nsc_dev chain",
	    nsc_dev_winit, nsc_dev_wstep, nsc_dev_wfini, NULL },
	{ "nsc_devval", "walk nsc_devval chain",
	    nsc_devval_winit, nsc_devval_wstep, NULL, NULL },
	{ "nsc_fd", "walk nsc_fd chain",
	    nsc_fd_winit, nsc_fd_wstep, NULL, NULL },
	{ "nsc_io", "walk nsc_io chain",
	    nsc_io_winit, nsc_io_wstep, NULL, NULL },
	{ "nsc_iodev", "walk nsc_iodev chain",
	    nsc_iodev_winit, nsc_iodev_wstep, NULL, NULL },
	{ "nsc_mem", "walk nsc_mem chain",
	    nsc_mem_winit, nsc_mem_wstep, NULL, NULL },
	{ "nsc_service", "walk nsc_service chain",
	    nsc_service_winit, nsc_service_wstep, NULL, NULL },
	{ "nsc_svc", "walk nsc_svc chain",
	    nsc_svc_winit, nsc_svc_wstep, NULL, NULL },
	{ "nsc_val", "walk nsc_val chain",
	    nsc_val_winit, nsc_val_wstep, NULL, NULL },
	{ "nstset", "walk nstset chain",
	    nstset_winit, nstset_wstep, NULL, NULL },
	{ "nsthread", "walk nsthread chain",
	    nsthread_winit, nsthread_wstep, NULL, NULL },
	{ "nst_free", "walk nsthread free/reuse list",
	    nst_free_winit, nst_free_wstep, NULL, NULL },
	{ NULL }
};


static const mdb_modinfo_t modinfo = {
	MDB_API_VERSION, dcmds, walkers
};


const mdb_modinfo_t *
_mdb_init(void)
{
	return (&modinfo);
}