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