xref: /titanic_50/usr/src/cmd/mdb/common/modules/ii/ii.c (revision 69ed0c8ece2346b34605e2c9567c9f7b0dad5dc8)
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 2008 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #include <sys/types.h>
27 #include <sys/mdb_modapi.h>
28 
29 #include <sys/nsctl/nsctl.h>
30 #include <sys/unistat/spcs_s.h>
31 #include <sys/unistat/spcs_s_k.h>
32 
33 
34 #include <sys/nsctl/dsw.h>
35 #include <sys/nsctl/dsw_dev.h>
36 
37 #include <sys/nsctl/nsvers.h>
38 
39 #define	offsetof(s, m)  ((size_t)(&((s *)0)->m))
40 
41 
42 const mdb_bitmask_t bi_flags_bits[] = {
43 	{ "DSW_GOLDEN", DSW_GOLDEN, DSW_GOLDEN },
44 	{ "DSW_COPYINGP", DSW_COPYINGP, DSW_COPYINGP },
45 	{ "DSW_COPYINGM", DSW_COPYINGM, DSW_COPYINGM },
46 	{ "DSW_COPYINGS", DSW_COPYINGS, DSW_COPYINGS },
47 	{ "DSW_COPYINGX", DSW_COPYINGX, DSW_COPYINGX },
48 	{ "DSW_BMPOFFLINE", DSW_BMPOFFLINE, DSW_BMPOFFLINE },
49 	{ "DSW_SHDOFFLINE", DSW_SHDOFFLINE, DSW_SHDOFFLINE },
50 	{ "DSW_MSTOFFLINE", DSW_MSTOFFLINE, DSW_MSTOFFLINE },
51 	{ "DSW_OVROFFLINE", DSW_OVROFFLINE, DSW_OVROFFLINE },
52 	{ "DSW_TREEMAP", DSW_TREEMAP, DSW_TREEMAP },
53 	{ "DSW_OVERFLOW", DSW_OVERFLOW, DSW_OVERFLOW },
54 	{ "DSW_SHDEXPORT", DSW_SHDEXPORT, DSW_SHDEXPORT },
55 	{ "DSW_SHDIMPORT", DSW_SHDIMPORT, DSW_SHDIMPORT },
56 	{ "DSW_VOVERFLOW", DSW_VOVERFLOW, DSW_VOVERFLOW },
57 	{ "DSW_HANGING", DSW_HANGING, DSW_HANGING },
58 	{ "DSW_CFGOFFLINE", DSW_CFGOFFLINE, DSW_CFGOFFLINE },
59 	{ "DSW_OVRHDRDRTY", DSW_OVRHDRDRTY, DSW_OVRHDRDRTY },
60 	{ "DSW_RESIZED", DSW_RESIZED, DSW_RESIZED },
61 	{ "DSW_FRECLAIM", DSW_FRECLAIM, DSW_FRECLAIM },
62 	{ NULL, 0, 0 }
63 };
64 
65 const mdb_bitmask_t bi_state_bits[] = {
66 	{ "DSW_IOCTL", DSW_IOCTL, DSW_IOCTL },
67 	{ "DSW_CLOSING", DSW_CLOSING, DSW_CLOSING },
68 	{ "DSW_MSTTARGET", DSW_MSTTARGET, DSW_MSTTARGET },
69 	{ "DSW_MULTIMST", DSW_MULTIMST, DSW_MULTIMST },
70 	{ NULL, 0, 0 }
71 };
72 static uintptr_t nextaddr;
73 /*
74  * Display a ii_fd_t
75  * Requires an address.
76  */
77 /*ARGSUSED*/
78 static int
79 ii_fd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
80 {
81 	ii_fd_t fd;
82 
83 	if (!(flags & DCMD_ADDRSPEC))
84 		return (DCMD_USAGE);
85 
86 	if (mdb_vread(&fd, sizeof (fd), addr) != sizeof (fd)) {
87 		mdb_warn("failed to read ii_fd_t at 0x%p", addr);
88 		return (DCMD_ERR);
89 	}
90 
91 	mdb_inc_indent(4);
92 	mdb_printf("ii_info: 0x%p ii_bmp: %d ii_shd: %d ii_ovr: %d ii_optr: "
93 	    "0x%p\nii_oflags: 0x%x\n", fd.ii_info, fd.ii_bmp, fd.ii_shd,
94 	    fd.ii_ovr, fd.ii_optr, fd.ii_oflags);
95 	mdb_dec_indent(4);
96 
97 	return (DCMD_OK);
98 }
99 
100 /*
101  * displays a ii_info_dev structure.
102  */
103 /*ARGSUSED*/
104 static int
105 ii_info_dev(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
106 {
107 	_ii_info_dev_t ipdev;
108 
109 	if (!(flags & DCMD_ADDRSPEC))
110 		return (DCMD_USAGE);
111 
112 	if (mdb_vread(&ipdev, sizeof (ipdev), addr) != sizeof (ipdev)) {
113 		mdb_warn("failed to read ii_info_dev_t at 0x%p", addr);
114 		return (DCMD_ERR);
115 	}
116 
117 	mdb_inc_indent(4);
118 	mdb_printf("bi_fd: 0x%p bi_iodev: 0x%p bi_tok: 0x%p\n",
119 	    ipdev.bi_fd, ipdev.bi_iodev, ipdev.bi_tok);
120 	mdb_printf("bi_ref: %d bi_rsrv: %d bi_orsrv: %d\n",
121 	    ipdev.bi_ref, ipdev.bi_rsrv, ipdev.bi_orsrv);
122 
123 	/*
124 	 * use nsc_fd to dump the fd details.... if present.
125 	 */
126 	if (ipdev.bi_fd) {
127 		mdb_printf("nsc_fd structure:\n");
128 		mdb_inc_indent(4);
129 		mdb_call_dcmd("nsc_fd", (uintptr_t)(ipdev.bi_fd),
130 		    flags, 0, NULL);
131 		mdb_dec_indent(4);
132 	}
133 	mdb_dec_indent(4);
134 	return (DCMD_OK);
135 }
136 
137 /*
138  * Displays an _ii_overflow structure
139  */
140 /*ARGSUSED*/
141 static int
142 ii_overflow(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
143 {
144 	_ii_overflow_t ii_overflow;
145 
146 	nextaddr = 0;
147 	if (!(flags & DCMD_ADDRSPEC))
148 		return (DCMD_USAGE);
149 
150 	if (mdb_vread(&ii_overflow, sizeof (ii_overflow), addr)
151 		!= sizeof (ii_overflow)) {
152 		mdb_warn("failed to read ii_overflow_t at 0x%p", addr);
153 		return (DCMD_ERR);
154 	}
155 
156 	mdb_inc_indent(4);
157 	mdb_printf("_ii_overflow at 0x%p\n", addr);
158 	mdb_printf("_ii_doverflow_t\n");
159 	mdb_inc_indent(4);
160 	mdb_printf("ii_dvolname: %s\n", ii_overflow.ii_volname);
161 	mdb_printf("ii_dhmagic: %x\n", ii_overflow.ii_hmagic);
162 	mdb_printf("ii_dhversion: %x\n", ii_overflow.ii_hversion);
163 	mdb_printf("ii_ddrefcnt: %x\n", ii_overflow.ii_drefcnt);
164 	mdb_printf("ii_dflags: %x\n", ii_overflow.ii_flags);
165 	mdb_printf("ii_dfreehead: %x\n", ii_overflow.ii_freehead);
166 	mdb_printf("ii_dnchunks: %x\n", ii_overflow.ii_nchunks);
167 	mdb_printf("ii_dunused: %x\n", ii_overflow.ii_unused);
168 	mdb_printf("ii_dused: %x\n", ii_overflow.ii_used);
169 	mdb_printf("ii_urefcnt: %x\n", ii_overflow.ii_urefcnt);
170 	mdb_dec_indent(4);
171 
172 	mdb_printf("ii_mutex: %x\n", ii_overflow.ii_mutex);
173 	mdb_printf("ii_kstat_mutex: %x\n", ii_overflow.ii_kstat_mutex);
174 	mdb_printf("ii_crefcnt: %d\n", ii_overflow.ii_crefcnt);
175 	mdb_printf("ii_detachcnt: %d\n", ii_overflow.ii_detachcnt);
176 	mdb_printf("ii_next: %x\n", ii_overflow.ii_next);
177 
178 	mdb_printf("Overflow volume:\n");
179 	if (ii_overflow.ii_dev)
180 		ii_info_dev((uintptr_t)ii_overflow.ii_dev, flags, 0, NULL);
181 
182 	mdb_printf("  ii_ioname: %s\n", &ii_overflow.ii_ioname);
183 	mdb_dec_indent(4);
184 
185 	nextaddr = (uintptr_t)ii_overflow.ii_next;
186 	return (DCMD_OK);
187 }
188 /*ARGSUSED*/
189 static int
190 ii_info(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
191 {
192 	_ii_info_t ii_info = {0};
193 	char string[DSW_NAMELEN];
194 
195 	nextaddr = 0;
196 	if (!(flags & DCMD_ADDRSPEC))
197 		return (DCMD_USAGE);
198 
199 	if (mdb_vread(&ii_info, sizeof (ii_info), addr) != sizeof (ii_info)) {
200 		mdb_warn("failed to read ii_info_t at 0x%p", addr);
201 		return (DCMD_ERR);
202 	}
203 
204 	mdb_printf(
205 		"bi_next: 0x%p\n"
206 		"bi_head: 0x%p\t"
207 		"bi_sibling: 0x%p\n"
208 		"bi_master: 0x%p\t"
209 		"bi_nextmst: 0x%p\n",
210 		ii_info.bi_next, ii_info.bi_head, ii_info.bi_sibling,
211 		ii_info.bi_master, ii_info.bi_nextmst);
212 
213 	mdb_printf("bi_mutex: 0x%p\n", ii_info.bi_mutex);
214 
215 	/*
216 	 * Print out all the fds by using ii_info_dev
217 	 */
218 	mdb_printf("Cache master:\n");
219 	if (ii_info.bi_mstdev)
220 		ii_info_dev((uintptr_t)ii_info.bi_mstdev, flags, 0, NULL);
221 
222 	mdb_printf("Raw master:\n");
223 	if (ii_info.bi_mstrdev)
224 		ii_info_dev((uintptr_t)ii_info.bi_mstrdev, flags, 0, NULL);
225 
226 	mdb_printf("Cache shadow:\n");
227 	ii_info_dev((uintptr_t)(addr + offsetof(_ii_info_t, bi_shddev)),
228 	    flags, 0, NULL);
229 
230 	mdb_printf("Raw shadow:\n");
231 	ii_info_dev((uintptr_t)(addr + offsetof(_ii_info_t, bi_shdrdev)),
232 	    flags, 0, NULL);
233 
234 	mdb_printf("Bitmap:\n");
235 	ii_info_dev((uintptr_t)(addr + offsetof(_ii_info_t, bi_bmpdev)),
236 	    flags, 0, NULL);
237 
238 	mdb_printf("bi_keyname: %-*s\n", DSW_NAMELEN, ii_info.bi_keyname);
239 	mdb_printf("bi_bitmap: 0x%p\n", ii_info.bi_bitmap);
240 
241 	if ((ii_info.bi_cluster == NULL) ||
242 	    (mdb_vread(&string, sizeof (string), (uintptr_t)ii_info.bi_cluster)
243 		!= sizeof (string)))
244 		string[0] = 0;
245 	mdb_printf("bi_cluster: %s\n", string);
246 
247 	if ((ii_info.bi_group == NULL) ||
248 	    (mdb_vread(&string, sizeof (string), (uintptr_t)ii_info.bi_group)
249 			!= sizeof (string)))
250 		string[0] = 0;
251 	mdb_printf("bi_group: %s\n", string);
252 
253 	mdb_printf("bi_busy: 0x%p\n", ii_info.bi_busy);
254 
255 	mdb_printf("bi_shdfba: %0x\t", ii_info.bi_shdfba);
256 	mdb_printf("bi_shdbits: %0x\n", ii_info.bi_shdbits);
257 	mdb_printf("bi_copyfba: %0x\t", ii_info.bi_copyfba);
258 	mdb_printf("bi_copybits: %0x\n", ii_info.bi_copybits);
259 
260 	mdb_printf("bi_size: %0x\n", ii_info.bi_size);
261 
262 	mdb_printf("bi_flags: 0x%x <%b>\n",
263 		ii_info.bi_flags, ii_info.bi_flags, bi_flags_bits);
264 
265 	mdb_printf("bi_state: 0x%x <%b>\n",
266 		ii_info.bi_state, ii_info.bi_state, bi_state_bits);
267 
268 	mdb_printf("bi_disabled: %d\n", ii_info.bi_disabled);
269 	mdb_printf("bi_ioctl: %d\n", ii_info.bi_ioctl);
270 	mdb_printf("bi_release: %d\t", ii_info.bi_release);
271 	mdb_printf("bi_rsrvcnt: %d\n", ii_info.bi_rsrvcnt);
272 
273 	mdb_printf("bi_copydonecv: %x\t", ii_info.bi_copydonecv);
274 	mdb_printf("bi_reservecv: %x\n", ii_info.bi_reservecv);
275 	mdb_printf("bi_releasecv: %x\t", ii_info.bi_releasecv);
276 	mdb_printf("bi_closingcv: %x\n", ii_info.bi_closingcv);
277 	mdb_printf("bi_ioctlcv: %x\t", ii_info.bi_ioctlcv);
278 	mdb_printf("bi_busycv: %x\n", ii_info.bi_busycv);
279 	mdb_call_dcmd("rwlock", (uintptr_t)(addr +
280 	    offsetof(_ii_info_t, bi_busyrw)), flags, 0, NULL);
281 	mdb_printf("bi_bitmap_ops: 0x%p\n", ii_info.bi_bitmap_ops);
282 
283 	mdb_printf("bi_rsrvmutex: %x\t", ii_info.bi_rsrvmutex);
284 	mdb_printf("bi_rlsemutex: %x\n", ii_info.bi_rlsemutex);
285 	mdb_printf("bi_bmpmutex: %x\n", ii_info.bi_bmpmutex);
286 
287 	mdb_printf("bi_mstchks: %d\t", ii_info.bi_mstchks);
288 	mdb_printf("bi_shdchks: %d\n", ii_info.bi_shdchks);
289 	mdb_printf("bi_shdchkused: %d\t", ii_info.bi_shdchkused);
290 	mdb_printf("bi_shdfchk: %d\n", ii_info.bi_shdfchk);
291 
292 	mdb_printf("bi_overflow\n");
293 	if (ii_info.bi_overflow)
294 		ii_overflow((uintptr_t)ii_info.bi_overflow, flags, 0, NULL);
295 
296 	mdb_printf("bi_iifd:\n");
297 	if (ii_info.bi_iifd)
298 		(void) ii_fd((uintptr_t)ii_info.bi_iifd, flags, 0, NULL);
299 
300 	mdb_printf("bi_throttle_unit: %d\t", ii_info.bi_throttle_unit);
301 	mdb_printf("bi_throttle_delay: %d\n", ii_info.bi_throttle_delay);
302 
303 	mdb_printf("bi_linkrw:\n");
304 	mdb_call_dcmd("rwlock", (uintptr_t)(addr +
305 	    offsetof(_ii_info_t, bi_linkrw)), flags, 0, NULL);
306 
307 	mdb_printf("bi_chksmutex: %x\n", ii_info.bi_chksmutex);
308 	mdb_printf("bi_locked_pid: %x\n", ii_info.bi_locked_pid);
309 	mdb_printf("bi_kstat: 0x%p\n", ii_info.bi_kstat);
310 	/* ii_kstat_info_t bi_kstat_io; */
311 
312 	nextaddr = (uintptr_t)ii_info.bi_next;
313 	return (DCMD_OK);
314 }
315 
316 /*
317  * This should be a walker surely.
318  */
319 /*ARGSUSED*/
320 static int
321 ii_info_all(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
322 {
323 	uintptr_t myaddr;
324 	/*
325 	 * we use the global address.
326 	 */
327 	if (flags & DCMD_ADDRSPEC)
328 		return (DCMD_USAGE);
329 
330 	if (mdb_readsym(&myaddr, sizeof (myaddr), "_ii_info_top") !=
331 	    sizeof (myaddr)) {
332 		return (DCMD_ERR);
333 	}
334 
335 	mdb_printf("_ii_info_top contains 0x%lx\n", myaddr);
336 
337 	while (myaddr) {
338 		ii_info(myaddr, DCMD_ADDRSPEC, 0, NULL);
339 		myaddr = nextaddr;
340 	}
341 	return (DCMD_OK);
342 }
343 
344 /*
345  * Display general ii module information.
346  */
347 
348 #define	ii_get_print(kvar, str, fmt, val)		\
349 	if (mdb_readvar(&(val), #kvar) == -1) {		\
350 		mdb_dec_indent(4);			\
351 		mdb_warn("unable to read '" #kvar "'");	\
352 		return (DCMD_ERR);			\
353 	}						\
354 	mdb_printf("%-20s" fmt "\n", str ":", val)
355 
356 /* ARGSUSED */
357 static int
358 ii(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
359 {
360 	int maj, min, mic, baseline, i;
361 
362 	if (argc != 0)
363 		return (DCMD_USAGE);
364 
365 	if (mdb_readvar(&maj, "dsw_major_rev") == -1) {
366 		mdb_warn("unable to read 'dsw_major_rev'");
367 		return (DCMD_ERR);
368 	}
369 
370 	if (mdb_readvar(&min, "dsw_minor_rev") == -1) {
371 		mdb_warn("unable to read 'dsw_minor_rev'");
372 		return (DCMD_ERR);
373 	}
374 
375 	if (mdb_readvar(&mic, "dsw_micro_rev") == -1) {
376 		mdb_warn("unable to read 'dsw_micro_rev'");
377 		return (DCMD_ERR);
378 	}
379 
380 	if (mdb_readvar(&baseline, "dsw_baseline_rev") == -1) {
381 		mdb_warn("unable to read 'dsw_baseline_rev'");
382 		return (DCMD_ERR);
383 	}
384 
385 	mdb_printf("Point-in-Time Copy module version: kernel %d.%d.%d.%d; "
386 	    "mdb %d.%d.%d.%d\n", maj, min, mic, baseline,
387 	    ISS_VERSION_MAJ, ISS_VERSION_MIN, ISS_VERSION_MIC, ISS_VERSION_NUM);
388 
389 	mdb_inc_indent(4);
390 	ii_get_print(ii_debug, "debug", "%d", i);
391 	ii_get_print(ii_bitmap, "bitmaps", "%d", i);
392 	mdb_dec_indent(4);
393 
394 	return (DCMD_OK);
395 }
396 
397 
398 /*
399  * MDB module linkage information:
400  */
401 
402 static const mdb_dcmd_t dcmds[] = {
403 { "ii", NULL, "display ii module info", ii },
404 { "ii_fd", NULL, "display ii_fd structure", ii_fd },
405 { "ii_info", NULL, "display ii_info structure", ii_info },
406 { "ii_info_all", NULL, "display all ii_info structures", ii_info_all },
407 { "ii_info_dev", NULL, "display ii_info_dev structure", ii_info_dev},
408 { "ii_overflow", NULL, "display ii_overflow structure", ii_overflow},
409 { NULL }
410 };
411 
412 
413 static const mdb_walker_t walkers[] = {
414 	{ NULL }
415 };
416 
417 
418 static const mdb_modinfo_t modinfo = {
419 	MDB_API_VERSION, dcmds, walkers
420 };
421 
422 
423 const mdb_modinfo_t *
424 _mdb_init(void)
425 {
426 	return (&modinfo);
427 }
428