xref: /onnv-gate/usr/src/cmd/mdb/common/modules/mr_sas/mr_sas.c (revision 9795:152ae1212035)
1*9795SYu.Wu@Sun.COM /*
2*9795SYu.Wu@Sun.COM  * CDDL HEADER START
3*9795SYu.Wu@Sun.COM  *
4*9795SYu.Wu@Sun.COM  * The contents of this file are subject to the terms of the
5*9795SYu.Wu@Sun.COM  * Common Development and Distribution License (the "License").
6*9795SYu.Wu@Sun.COM  * You may not use this file except in compliance with the License.
7*9795SYu.Wu@Sun.COM  *
8*9795SYu.Wu@Sun.COM  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*9795SYu.Wu@Sun.COM  * or http://www.opensolaris.org/os/licensing.
10*9795SYu.Wu@Sun.COM  * See the License for the specific language governing permissions
11*9795SYu.Wu@Sun.COM  * and limitations under the License.
12*9795SYu.Wu@Sun.COM  *
13*9795SYu.Wu@Sun.COM  * When distributing Covered Code, include this CDDL HEADER in each
14*9795SYu.Wu@Sun.COM  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*9795SYu.Wu@Sun.COM  * If applicable, add the following below this CDDL HEADER, with the
16*9795SYu.Wu@Sun.COM  * fields enclosed by brackets "[]" replaced with your own identifying
17*9795SYu.Wu@Sun.COM  * information: Portions Copyright [yyyy] [name of copyright owner]
18*9795SYu.Wu@Sun.COM  *
19*9795SYu.Wu@Sun.COM  * CDDL HEADER END
20*9795SYu.Wu@Sun.COM  */
21*9795SYu.Wu@Sun.COM /*
22*9795SYu.Wu@Sun.COM  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
23*9795SYu.Wu@Sun.COM  * Use is subject to license terms.
24*9795SYu.Wu@Sun.COM  */
25*9795SYu.Wu@Sun.COM 
26*9795SYu.Wu@Sun.COM #include <limits.h>
27*9795SYu.Wu@Sun.COM #include <sys/mdb_modapi.h>
28*9795SYu.Wu@Sun.COM #include <sys/sysinfo.h>
29*9795SYu.Wu@Sun.COM #include <sys/sunmdi.h>
30*9795SYu.Wu@Sun.COM #include <sys/scsi/scsi.h>
31*9795SYu.Wu@Sun.COM #include "mr_sas.h"
32*9795SYu.Wu@Sun.COM 
33*9795SYu.Wu@Sun.COM int
construct_path(uintptr_t addr,char * result)34*9795SYu.Wu@Sun.COM construct_path(uintptr_t addr, char *result)
35*9795SYu.Wu@Sun.COM {
36*9795SYu.Wu@Sun.COM 	struct	dev_info	d;
37*9795SYu.Wu@Sun.COM 	char	devi_node[PATH_MAX];
38*9795SYu.Wu@Sun.COM 	char	devi_addr[PATH_MAX];
39*9795SYu.Wu@Sun.COM 
40*9795SYu.Wu@Sun.COM 	if (mdb_vread(&d, sizeof (d), addr) == -1) {
41*9795SYu.Wu@Sun.COM 		mdb_warn("couldn't read dev_info");
42*9795SYu.Wu@Sun.COM 		return (DCMD_ERR);
43*9795SYu.Wu@Sun.COM 	}
44*9795SYu.Wu@Sun.COM 
45*9795SYu.Wu@Sun.COM 	if (d.devi_parent) {
46*9795SYu.Wu@Sun.COM 		construct_path((uintptr_t)d.devi_parent, result);
47*9795SYu.Wu@Sun.COM 		mdb_readstr(devi_node, sizeof (devi_node),
48*9795SYu.Wu@Sun.COM 		    (uintptr_t)d.devi_node_name);
49*9795SYu.Wu@Sun.COM 		mdb_readstr(devi_addr, sizeof (devi_addr),
50*9795SYu.Wu@Sun.COM 		    (uintptr_t)d.devi_addr);
51*9795SYu.Wu@Sun.COM 		mdb_snprintf(result+strlen(result),
52*9795SYu.Wu@Sun.COM 		    PATH_MAX-strlen(result),
53*9795SYu.Wu@Sun.COM 		    "/%s%s%s", devi_node, (*devi_addr ? "@" : ""),
54*9795SYu.Wu@Sun.COM 		    devi_addr);
55*9795SYu.Wu@Sun.COM 	}
56*9795SYu.Wu@Sun.COM 	return (DCMD_OK);
57*9795SYu.Wu@Sun.COM }
58*9795SYu.Wu@Sun.COM 
59*9795SYu.Wu@Sun.COM void
display_targets(struct mrsas_instance m,int verbose)60*9795SYu.Wu@Sun.COM display_targets(struct mrsas_instance m, int verbose)
61*9795SYu.Wu@Sun.COM {
62*9795SYu.Wu@Sun.COM 	int	tgt;
63*9795SYu.Wu@Sun.COM 	struct mrsas_ld *mr_ldp;
64*9795SYu.Wu@Sun.COM 	char	device_path[PATH_MAX];
65*9795SYu.Wu@Sun.COM 
66*9795SYu.Wu@Sun.COM 	if (verbose) {
67*9795SYu.Wu@Sun.COM 		*device_path = 0;
68*9795SYu.Wu@Sun.COM 		if (construct_path((uintptr_t)m.dip, device_path) != DCMD_OK) {
69*9795SYu.Wu@Sun.COM 			strcpy(device_path, "couldn't determine device path");
70*9795SYu.Wu@Sun.COM 		}
71*9795SYu.Wu@Sun.COM 	}
72*9795SYu.Wu@Sun.COM 
73*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
74*9795SYu.Wu@Sun.COM 	if (verbose)
75*9795SYu.Wu@Sun.COM 		mdb_printf("%s\n", device_path);
76*9795SYu.Wu@Sun.COM 	mdb_printf("dev_type target\n");
77*9795SYu.Wu@Sun.COM 	mdb_printf("----------");
78*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
79*9795SYu.Wu@Sun.COM 	for (tgt = 0; tgt < MRDRV_MAX_LD; tgt++) {
80*9795SYu.Wu@Sun.COM 		mr_ldp = (struct mrsas_ld *)&m.mr_ld_list[tgt];
81*9795SYu.Wu@Sun.COM 		if ((mr_ldp != NULL) && (mr_ldp->dip != NULL) &&
82*9795SYu.Wu@Sun.COM 		    (mr_ldp->lun_type == MRSAS_LD_LUN)) {
83*9795SYu.Wu@Sun.COM 			mdb_printf("sd %d", tgt);
84*9795SYu.Wu@Sun.COM 			mdb_printf("\n");
85*9795SYu.Wu@Sun.COM 		}
86*9795SYu.Wu@Sun.COM 	}
87*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
88*9795SYu.Wu@Sun.COM }
89*9795SYu.Wu@Sun.COM 
90*9795SYu.Wu@Sun.COM void
display_deviceinfo(struct mrsas_instance m)91*9795SYu.Wu@Sun.COM display_deviceinfo(struct mrsas_instance m)
92*9795SYu.Wu@Sun.COM {
93*9795SYu.Wu@Sun.COM 	uint16_t vid, did, svid, sid;
94*9795SYu.Wu@Sun.COM 
95*9795SYu.Wu@Sun.COM 	vid = m.vendor_id;
96*9795SYu.Wu@Sun.COM 	did = m.device_id;
97*9795SYu.Wu@Sun.COM 	svid = m.subsysvid;
98*9795SYu.Wu@Sun.COM 	sid = m.subsysid;
99*9795SYu.Wu@Sun.COM 
100*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
101*9795SYu.Wu@Sun.COM 	mdb_printf("vendor_id device_id subsysvid subsysid");
102*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
103*9795SYu.Wu@Sun.COM 	mdb_printf("--------------------------------------");
104*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
105*9795SYu.Wu@Sun.COM 	mdb_printf("    0x%x   0x%x    0x%x    0x%x",
106*9795SYu.Wu@Sun.COM 	    vid, did, svid, sid);
107*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
108*9795SYu.Wu@Sun.COM }
109*9795SYu.Wu@Sun.COM 
110*9795SYu.Wu@Sun.COM static int
mr_sas_dcmd(uintptr_t addr,uint_t flags,int argc,const mdb_arg_t * argv)111*9795SYu.Wu@Sun.COM mr_sas_dcmd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
112*9795SYu.Wu@Sun.COM {
113*9795SYu.Wu@Sun.COM 	struct mrsas_instance m;
114*9795SYu.Wu@Sun.COM 
115*9795SYu.Wu@Sun.COM 	int	instance;
116*9795SYu.Wu@Sun.COM 	uint16_t ncmds;
117*9795SYu.Wu@Sun.COM 	uint_t	verbose = FALSE;
118*9795SYu.Wu@Sun.COM 	uint_t	device_info = FALSE;
119*9795SYu.Wu@Sun.COM 	uint_t	target_info = FALSE;
120*9795SYu.Wu@Sun.COM 	int	rv = DCMD_OK;
121*9795SYu.Wu@Sun.COM 	void	*mrsas_state;
122*9795SYu.Wu@Sun.COM 
123*9795SYu.Wu@Sun.COM 	if (!(flags & DCMD_ADDRSPEC)) {
124*9795SYu.Wu@Sun.COM 		mrsas_state = NULL;
125*9795SYu.Wu@Sun.COM 		if (mdb_readvar(&mrsas_state, "mrsas_state") == -1) {
126*9795SYu.Wu@Sun.COM 			mdb_warn("can't read mrsas_state");
127*9795SYu.Wu@Sun.COM 			return (DCMD_ERR);
128*9795SYu.Wu@Sun.COM 		}
129*9795SYu.Wu@Sun.COM 		if (mdb_pwalk_dcmd("genunix`softstate", "mr_sas`mr_sas",
130*9795SYu.Wu@Sun.COM 		    argc, argv, (uintptr_t)mrsas_state) == -1) {
131*9795SYu.Wu@Sun.COM 			mdb_warn("mdb_pwalk_dcmd failed");
132*9795SYu.Wu@Sun.COM 			return (DCMD_ERR);
133*9795SYu.Wu@Sun.COM 		}
134*9795SYu.Wu@Sun.COM 		return (DCMD_OK);
135*9795SYu.Wu@Sun.COM 	}
136*9795SYu.Wu@Sun.COM 
137*9795SYu.Wu@Sun.COM 	if (mdb_getopts(argc, argv,
138*9795SYu.Wu@Sun.COM 	    'd', MDB_OPT_SETBITS, TRUE, &device_info,
139*9795SYu.Wu@Sun.COM 	    't', MDB_OPT_SETBITS, TRUE, &target_info,
140*9795SYu.Wu@Sun.COM 	    'v', MDB_OPT_SETBITS, TRUE, &verbose,
141*9795SYu.Wu@Sun.COM 	    NULL) != argc)
142*9795SYu.Wu@Sun.COM 		return (DCMD_USAGE);
143*9795SYu.Wu@Sun.COM 
144*9795SYu.Wu@Sun.COM 	if (mdb_vread(&m, sizeof (m), addr) == -1) {
145*9795SYu.Wu@Sun.COM 		mdb_warn("couldn't read mrsas_instance struct at 0x%p", addr);
146*9795SYu.Wu@Sun.COM 		return (DCMD_ERR);
147*9795SYu.Wu@Sun.COM 	}
148*9795SYu.Wu@Sun.COM 	instance = m.instance;
149*9795SYu.Wu@Sun.COM 
150*9795SYu.Wu@Sun.COM 	/* cmd slot info */
151*9795SYu.Wu@Sun.COM 	ncmds = m.max_fw_cmds;
152*9795SYu.Wu@Sun.COM 
153*9795SYu.Wu@Sun.COM 	/* processing completed */
154*9795SYu.Wu@Sun.COM 	if (((flags & DCMD_ADDRSPEC) && !(flags & DCMD_LOOP)) ||
155*9795SYu.Wu@Sun.COM 	    (flags & DCMD_LOOPFIRST)) {
156*9795SYu.Wu@Sun.COM 		if ((flags & DCMD_LOOP) && !(flags & DCMD_LOOPFIRST))
157*9795SYu.Wu@Sun.COM 			mdb_printf("\n");
158*9795SYu.Wu@Sun.COM 		mdb_printf("         mrsas_t inst max_fw_cmds intr_type");
159*9795SYu.Wu@Sun.COM 		mdb_printf("\n");
160*9795SYu.Wu@Sun.COM 		mdb_printf("===========================================");
161*9795SYu.Wu@Sun.COM 		mdb_printf("\n");
162*9795SYu.Wu@Sun.COM 	}
163*9795SYu.Wu@Sun.COM 
164*9795SYu.Wu@Sun.COM 	mdb_printf("%16p %4d      %4d    ", addr, instance, ncmds);
165*9795SYu.Wu@Sun.COM 	switch (m.intr_type) {
166*9795SYu.Wu@Sun.COM 		case DDI_INTR_TYPE_MSIX:
167*9795SYu.Wu@Sun.COM 			mdb_printf("MSI-X");
168*9795SYu.Wu@Sun.COM 			break;
169*9795SYu.Wu@Sun.COM 		case DDI_INTR_TYPE_MSI:
170*9795SYu.Wu@Sun.COM 			mdb_printf("MSI");
171*9795SYu.Wu@Sun.COM 			break;
172*9795SYu.Wu@Sun.COM 		case DDI_INTR_TYPE_FIXED:
173*9795SYu.Wu@Sun.COM 			mdb_printf("FIXED");
174*9795SYu.Wu@Sun.COM 			break;
175*9795SYu.Wu@Sun.COM 		default:
176*9795SYu.Wu@Sun.COM 			mdb_printf("INVALD");
177*9795SYu.Wu@Sun.COM 	}
178*9795SYu.Wu@Sun.COM 	mdb_printf("\n");
179*9795SYu.Wu@Sun.COM 
180*9795SYu.Wu@Sun.COM 	if (target_info)
181*9795SYu.Wu@Sun.COM 		display_targets(m, verbose);
182*9795SYu.Wu@Sun.COM 
183*9795SYu.Wu@Sun.COM 	if (device_info)
184*9795SYu.Wu@Sun.COM 		display_deviceinfo(m);
185*9795SYu.Wu@Sun.COM 
186*9795SYu.Wu@Sun.COM 	return (rv);
187*9795SYu.Wu@Sun.COM }
188*9795SYu.Wu@Sun.COM 
189*9795SYu.Wu@Sun.COM void
mr_sas_help(void)190*9795SYu.Wu@Sun.COM mr_sas_help(void)
191*9795SYu.Wu@Sun.COM {
192*9795SYu.Wu@Sun.COM 	mdb_printf("Prints summary information about each mr_sas instance, "
193*9795SYu.Wu@Sun.COM 	    "Without the address of a \"struct mrsas_instance\", prints every "
194*9795SYu.Wu@Sun.COM 	    "instance.\n\n"
195*9795SYu.Wu@Sun.COM 	    "Switches:\n"
196*9795SYu.Wu@Sun.COM 	    "  -t   includes information about targets\n"
197*9795SYu.Wu@Sun.COM 	    "  -d   includes information about the hardware\n"
198*9795SYu.Wu@Sun.COM 	    "  -v   displays extra information for some options\n");
199*9795SYu.Wu@Sun.COM }
200*9795SYu.Wu@Sun.COM 
201*9795SYu.Wu@Sun.COM static const mdb_dcmd_t dcmds[] = {
202*9795SYu.Wu@Sun.COM 	{ "mr_sas", "?[-tdv]", "print mr_sas information", mr_sas_dcmd,
203*9795SYu.Wu@Sun.COM 	    mr_sas_help },
204*9795SYu.Wu@Sun.COM 	{ NULL }
205*9795SYu.Wu@Sun.COM };
206*9795SYu.Wu@Sun.COM 
207*9795SYu.Wu@Sun.COM static const mdb_modinfo_t modinfo = {
208*9795SYu.Wu@Sun.COM 	MDB_API_VERSION, dcmds, NULL
209*9795SYu.Wu@Sun.COM };
210*9795SYu.Wu@Sun.COM 
211*9795SYu.Wu@Sun.COM const mdb_modinfo_t *
_mdb_init(void)212*9795SYu.Wu@Sun.COM _mdb_init(void)
213*9795SYu.Wu@Sun.COM {
214*9795SYu.Wu@Sun.COM 	return (&modinfo);
215*9795SYu.Wu@Sun.COM }
216