xref: /illumos-gate/usr/src/cmd/mdb/common/modules/smbsrv/smbsrv.c (revision e86c665f2e10cf29ac3b5f93b53c60bef762a690)
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 /*
23  * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
24  * Copyright 2015 Nexenta Systems, Inc. All rights reserved.
25  */
26 
27 #include <mdb/mdb_modapi.h>
28 #include <mdb/mdb_ks.h>
29 #include <sys/thread.h>
30 #include <sys/taskq.h>
31 #include <smbsrv/smb_vops.h>
32 #include <smbsrv/smb.h>
33 #include <smbsrv/smb_ktypes.h>
34 
35 #ifdef _KERNEL
36 #define	SMBSRV_OBJNAME	"smbsrv"
37 #else
38 #define	SMBSRV_OBJNAME	"libfksmbsrv.so.1"
39 #endif
40 
41 #define	SMB_DCMD_INDENT		2
42 #define	ACE_TYPE_TABLEN		(ACE_ALL_TYPES + 1)
43 #define	ACE_TYPE_ENTRY(_v_)	{_v_, #_v_}
44 #define	SMB_COM_ENTRY(_v_, _x_)	{#_v_, _x_}
45 
46 #define	SMB_MDB_MAX_OPTS	9
47 
48 #define	SMB_OPT_SERVER		0x00000001
49 #define	SMB_OPT_SESSION		0x00000002
50 #define	SMB_OPT_REQUEST		0x00000004
51 #define	SMB_OPT_USER		0x00000008
52 #define	SMB_OPT_TREE		0x00000010
53 #define	SMB_OPT_OFILE		0x00000020
54 #define	SMB_OPT_ODIR		0x00000040
55 #define	SMB_OPT_WALK		0x00000100
56 #define	SMB_OPT_VERBOSE		0x00000200
57 #define	SMB_OPT_ALL_OBJ		0x000000FF
58 
59 /*
60  * Structure associating an ACE type to a string.
61  */
62 typedef struct {
63 	uint8_t		ace_type_value;
64 	const char	*ace_type_sting;
65 } ace_type_entry_t;
66 
67 /*
68  * Structure containing strings describing an SMB command.
69  */
70 typedef struct {
71 	const char	*smb_com;
72 	const char	*smb_andx;
73 } smb_com_entry_t;
74 
75 /*
76  * Structure describing an object to be expanded (displayed).
77  */
78 typedef struct {
79 	uint_t		ex_mask;
80 	size_t		ex_offset;
81 	const char	*ex_dcmd;
82 	const char	*ex_name;
83 } smb_exp_t;
84 
85 /*
86  * List of supported options. Ther order has the match the bits SMB_OPT_xxx.
87  */
88 typedef struct smb_mdb_opts {
89 	char		*o_name;
90 	uint32_t	o_value;
91 } smb_mdb_opts_t;
92 
93 static smb_mdb_opts_t smb_opts[SMB_MDB_MAX_OPTS] =
94 {
95 	{ "-s", SMB_OPT_SERVER	},
96 	{ "-e", SMB_OPT_SESSION	},
97 	{ "-r", SMB_OPT_REQUEST	},
98 	{ "-u", SMB_OPT_USER	},
99 	{ "-t", SMB_OPT_TREE	},
100 	{ "-f", SMB_OPT_OFILE	},
101 	{ "-d", SMB_OPT_ODIR	},
102 	{ "-w", SMB_OPT_WALK	},
103 	{ "-v", SMB_OPT_VERBOSE	}
104 };
105 
106 static smb_com_entry_t	smb_com[256] =
107 {
108 	SMB_COM_ENTRY(SMB_COM_CREATE_DIRECTORY, "No"),
109 	SMB_COM_ENTRY(SMB_COM_DELETE_DIRECTORY, "No"),
110 	SMB_COM_ENTRY(SMB_COM_OPEN, "No"),
111 	SMB_COM_ENTRY(SMB_COM_CREATE, "No"),
112 	SMB_COM_ENTRY(SMB_COM_CLOSE, "No"),
113 	SMB_COM_ENTRY(SMB_COM_FLUSH, "No"),
114 	SMB_COM_ENTRY(SMB_COM_DELETE, "No"),
115 	SMB_COM_ENTRY(SMB_COM_RENAME, "No"),
116 	SMB_COM_ENTRY(SMB_COM_QUERY_INFORMATION, "No"),
117 	SMB_COM_ENTRY(SMB_COM_SET_INFORMATION, "No"),
118 	SMB_COM_ENTRY(SMB_COM_READ, "No"),
119 	SMB_COM_ENTRY(SMB_COM_WRITE, "No"),
120 	SMB_COM_ENTRY(SMB_COM_LOCK_BYTE_RANGE, "No"),
121 	SMB_COM_ENTRY(SMB_COM_UNLOCK_BYTE_RANGE, "No"),
122 	SMB_COM_ENTRY(SMB_COM_CREATE_TEMPORARY, "No"),
123 	SMB_COM_ENTRY(SMB_COM_CREATE_NEW, "No"),
124 	SMB_COM_ENTRY(SMB_COM_CHECK_DIRECTORY, "No"),
125 	SMB_COM_ENTRY(SMB_COM_PROCESS_EXIT, "No"),
126 	SMB_COM_ENTRY(SMB_COM_SEEK, "No"),
127 	SMB_COM_ENTRY(SMB_COM_LOCK_AND_READ, "No"),
128 	SMB_COM_ENTRY(SMB_COM_WRITE_AND_UNLOCK, "No"),
129 	SMB_COM_ENTRY(0x15, "?"),
130 	SMB_COM_ENTRY(0x16, "?"),
131 	SMB_COM_ENTRY(0x17, "?"),
132 	SMB_COM_ENTRY(0x18, "?"),
133 	SMB_COM_ENTRY(0x19, "?"),
134 	SMB_COM_ENTRY(SMB_COM_READ_RAW, "No"),
135 	SMB_COM_ENTRY(SMB_COM_READ_MPX, "No"),
136 	SMB_COM_ENTRY(SMB_COM_READ_MPX_SECONDARY, "No"),
137 	SMB_COM_ENTRY(SMB_COM_WRITE_RAW, "No"),
138 	SMB_COM_ENTRY(SMB_COM_WRITE_MPX, "No"),
139 	SMB_COM_ENTRY(SMB_COM_WRITE_MPX_SECONDARY, "No"),
140 	SMB_COM_ENTRY(SMB_COM_WRITE_COMPLETE, "No"),
141 	SMB_COM_ENTRY(SMB_COM_QUERY_SERVER, "No"),
142 	SMB_COM_ENTRY(SMB_COM_SET_INFORMATION2, "No"),
143 	SMB_COM_ENTRY(SMB_COM_QUERY_INFORMATION2, "No"),
144 	SMB_COM_ENTRY(SMB_COM_LOCKING_ANDX, "No"),
145 	SMB_COM_ENTRY(SMB_COM_TRANSACTION, "No"),
146 	SMB_COM_ENTRY(SMB_COM_TRANSACTION_SECONDARY, "No"),
147 	SMB_COM_ENTRY(SMB_COM_IOCTL, "No"),
148 	SMB_COM_ENTRY(SMB_COM_IOCTL_SECONDARY, "No"),
149 	SMB_COM_ENTRY(SMB_COM_COPY, "No"),
150 	SMB_COM_ENTRY(SMB_COM_MOVE, "No"),
151 	SMB_COM_ENTRY(SMB_COM_ECHO, "No"),
152 	SMB_COM_ENTRY(SMB_COM_WRITE_AND_CLOSE, "No"),
153 	SMB_COM_ENTRY(SMB_COM_OPEN_ANDX, "No"),
154 	SMB_COM_ENTRY(SMB_COM_READ_ANDX, "No"),
155 	SMB_COM_ENTRY(SMB_COM_WRITE_ANDX, "No"),
156 	SMB_COM_ENTRY(SMB_COM_NEW_FILE_SIZE, "No"),
157 	SMB_COM_ENTRY(SMB_COM_CLOSE_AND_TREE_DISC, "No"),
158 	SMB_COM_ENTRY(SMB_COM_TRANSACTION2, "No"),
159 	SMB_COM_ENTRY(SMB_COM_TRANSACTION2_SECONDARY, "No"),
160 	SMB_COM_ENTRY(SMB_COM_FIND_CLOSE2, "No"),
161 	SMB_COM_ENTRY(SMB_COM_FIND_NOTIFY_CLOSE, "No"),
162 	SMB_COM_ENTRY(0x36, "?"),
163 	SMB_COM_ENTRY(0x37, "?"),
164 	SMB_COM_ENTRY(0x38, "?"),
165 	SMB_COM_ENTRY(0x39, "?"),
166 	SMB_COM_ENTRY(0x3A, "?"),
167 	SMB_COM_ENTRY(0x3B, "?"),
168 	SMB_COM_ENTRY(0x3C, "?"),
169 	SMB_COM_ENTRY(0x3D, "?"),
170 	SMB_COM_ENTRY(0x3E, "?"),
171 	SMB_COM_ENTRY(0x3F, "?"),
172 	SMB_COM_ENTRY(0x40, "?"),
173 	SMB_COM_ENTRY(0x41, "?"),
174 	SMB_COM_ENTRY(0x42, "?"),
175 	SMB_COM_ENTRY(0x43, "?"),
176 	SMB_COM_ENTRY(0x44, "?"),
177 	SMB_COM_ENTRY(0x45, "?"),
178 	SMB_COM_ENTRY(0x46, "?"),
179 	SMB_COM_ENTRY(0x47, "?"),
180 	SMB_COM_ENTRY(0x48, "?"),
181 	SMB_COM_ENTRY(0x49, "?"),
182 	SMB_COM_ENTRY(0x4A, "?"),
183 	SMB_COM_ENTRY(0x4B, "?"),
184 	SMB_COM_ENTRY(0x4C, "?"),
185 	SMB_COM_ENTRY(0x4D, "?"),
186 	SMB_COM_ENTRY(0x4E, "?"),
187 	SMB_COM_ENTRY(0x4F, "?"),
188 	SMB_COM_ENTRY(0x50, "?"),
189 	SMB_COM_ENTRY(0x51, "?"),
190 	SMB_COM_ENTRY(0x52, "?"),
191 	SMB_COM_ENTRY(0x53, "?"),
192 	SMB_COM_ENTRY(0x54, "?"),
193 	SMB_COM_ENTRY(0x55, "?"),
194 	SMB_COM_ENTRY(0x56, "?"),
195 	SMB_COM_ENTRY(0x57, "?"),
196 	SMB_COM_ENTRY(0x58, "?"),
197 	SMB_COM_ENTRY(0x59, "?"),
198 	SMB_COM_ENTRY(0x5A, "?"),
199 	SMB_COM_ENTRY(0x5B, "?"),
200 	SMB_COM_ENTRY(0x5C, "?"),
201 	SMB_COM_ENTRY(0x5D, "?"),
202 	SMB_COM_ENTRY(0x5E, "?"),
203 	SMB_COM_ENTRY(0x5F, "?"),
204 	SMB_COM_ENTRY(0x60, "?"),
205 	SMB_COM_ENTRY(0x61, "?"),
206 	SMB_COM_ENTRY(0x62, "?"),
207 	SMB_COM_ENTRY(0x63, "?"),
208 	SMB_COM_ENTRY(0x64, "?"),
209 	SMB_COM_ENTRY(0x65, "?"),
210 	SMB_COM_ENTRY(0x66, "?"),
211 	SMB_COM_ENTRY(0x67, "?"),
212 	SMB_COM_ENTRY(0x68, "?"),
213 	SMB_COM_ENTRY(0x69, "?"),
214 	SMB_COM_ENTRY(0x6A, "?"),
215 	SMB_COM_ENTRY(0x6B, "?"),
216 	SMB_COM_ENTRY(0x6C, "?"),
217 	SMB_COM_ENTRY(0x6D, "?"),
218 	SMB_COM_ENTRY(0x6E, "?"),
219 	SMB_COM_ENTRY(0x6F, "?"),
220 	SMB_COM_ENTRY(SMB_COM_TREE_CONNECT, "No"),
221 	SMB_COM_ENTRY(SMB_COM_TREE_DISCONNECT, "No"),
222 	SMB_COM_ENTRY(SMB_COM_NEGOTIATE, "No"),
223 	SMB_COM_ENTRY(SMB_COM_SESSION_SETUP_ANDX, "No"),
224 	SMB_COM_ENTRY(SMB_COM_LOGOFF_ANDX, "No"),
225 	SMB_COM_ENTRY(SMB_COM_TREE_CONNECT_ANDX, "No"),
226 	SMB_COM_ENTRY(0x76, "?"),
227 	SMB_COM_ENTRY(0x77, "?"),
228 	SMB_COM_ENTRY(0x78, "?"),
229 	SMB_COM_ENTRY(0x79, "?"),
230 	SMB_COM_ENTRY(0x7A, "?"),
231 	SMB_COM_ENTRY(0x7B, "?"),
232 	SMB_COM_ENTRY(0x7C, "?"),
233 	SMB_COM_ENTRY(0x7D, "?"),
234 	SMB_COM_ENTRY(0x7E, "?"),
235 	SMB_COM_ENTRY(0x7F, "?"),
236 	SMB_COM_ENTRY(SMB_COM_QUERY_INFORMATION_DISK, "No"),
237 	SMB_COM_ENTRY(SMB_COM_SEARCH, "No"),
238 	SMB_COM_ENTRY(SMB_COM_FIND, "No"),
239 	SMB_COM_ENTRY(SMB_COM_FIND_UNIQUE, "No"),
240 	SMB_COM_ENTRY(SMB_COM_FIND_CLOSE, "No"),
241 	SMB_COM_ENTRY(0x85, "?"),
242 	SMB_COM_ENTRY(0x86, "?"),
243 	SMB_COM_ENTRY(0x87, "?"),
244 	SMB_COM_ENTRY(0x88, "?"),
245 	SMB_COM_ENTRY(0x89, "?"),
246 	SMB_COM_ENTRY(0x8A, "?"),
247 	SMB_COM_ENTRY(0x8B, "?"),
248 	SMB_COM_ENTRY(0x8C, "?"),
249 	SMB_COM_ENTRY(0x8D, "?"),
250 	SMB_COM_ENTRY(0x8E, "?"),
251 	SMB_COM_ENTRY(0x8F, "?"),
252 	SMB_COM_ENTRY(0x90, "?"),
253 	SMB_COM_ENTRY(0x91, "?"),
254 	SMB_COM_ENTRY(0x92, "?"),
255 	SMB_COM_ENTRY(0x93, "?"),
256 	SMB_COM_ENTRY(0x94, "?"),
257 	SMB_COM_ENTRY(0x95, "?"),
258 	SMB_COM_ENTRY(0x96, "?"),
259 	SMB_COM_ENTRY(0x97, "?"),
260 	SMB_COM_ENTRY(0x98, "?"),
261 	SMB_COM_ENTRY(0x99, "?"),
262 	SMB_COM_ENTRY(0x9A, "?"),
263 	SMB_COM_ENTRY(0x9B, "?"),
264 	SMB_COM_ENTRY(0x9C, "?"),
265 	SMB_COM_ENTRY(0x9D, "?"),
266 	SMB_COM_ENTRY(0x9E, "?"),
267 	SMB_COM_ENTRY(0x9F, "?"),
268 	SMB_COM_ENTRY(SMB_COM_NT_TRANSACT, "No"),
269 	SMB_COM_ENTRY(SMB_COM_NT_TRANSACT_SECONDARY, "No"),
270 	SMB_COM_ENTRY(SMB_COM_NT_CREATE_ANDX, "No"),
271 	SMB_COM_ENTRY(0xA3, "?"),
272 	SMB_COM_ENTRY(SMB_COM_NT_CANCEL, "No"),
273 	SMB_COM_ENTRY(SMB_COM_NT_RENAME, "No"),
274 	SMB_COM_ENTRY(0xA6, "?"),
275 	SMB_COM_ENTRY(0xA7, "?"),
276 	SMB_COM_ENTRY(0xA8, "?"),
277 	SMB_COM_ENTRY(0xA9, "?"),
278 	SMB_COM_ENTRY(0xAA, "?"),
279 	SMB_COM_ENTRY(0xAB, "?"),
280 	SMB_COM_ENTRY(0xAC, "?"),
281 	SMB_COM_ENTRY(0xAD, "?"),
282 	SMB_COM_ENTRY(0xAE, "?"),
283 	SMB_COM_ENTRY(0xAF, "?"),
284 	SMB_COM_ENTRY(0xB0, "?"),
285 	SMB_COM_ENTRY(0xB1, "?"),
286 	SMB_COM_ENTRY(0xB2, "?"),
287 	SMB_COM_ENTRY(0xB3, "?"),
288 	SMB_COM_ENTRY(0xB4, "?"),
289 	SMB_COM_ENTRY(0xB5, "?"),
290 	SMB_COM_ENTRY(0xB6, "?"),
291 	SMB_COM_ENTRY(0xB7, "?"),
292 	SMB_COM_ENTRY(0xB8, "?"),
293 	SMB_COM_ENTRY(0xB9, "?"),
294 	SMB_COM_ENTRY(0xBA, "?"),
295 	SMB_COM_ENTRY(0xBB, "?"),
296 	SMB_COM_ENTRY(0xBC, "?"),
297 	SMB_COM_ENTRY(0xBD, "?"),
298 	SMB_COM_ENTRY(0xBE, "?"),
299 	SMB_COM_ENTRY(0xBF, "?"),
300 	SMB_COM_ENTRY(SMB_COM_OPEN_PRINT_FILE, "No"),
301 	SMB_COM_ENTRY(SMB_COM_WRITE_PRINT_FILE, "No"),
302 	SMB_COM_ENTRY(SMB_COM_CLOSE_PRINT_FILE, "No"),
303 	SMB_COM_ENTRY(SMB_COM_GET_PRINT_QUEUE, "No"),
304 	SMB_COM_ENTRY(0xC4, "?"),
305 	SMB_COM_ENTRY(0xC5, "?"),
306 	SMB_COM_ENTRY(0xC6, "?"),
307 	SMB_COM_ENTRY(0xC7, "?"),
308 	SMB_COM_ENTRY(0xC8, "?"),
309 	SMB_COM_ENTRY(0xC9, "?"),
310 	SMB_COM_ENTRY(0xCA, "?"),
311 	SMB_COM_ENTRY(0xCB, "?"),
312 	SMB_COM_ENTRY(0xCC, "?"),
313 	SMB_COM_ENTRY(0xCD, "?"),
314 	SMB_COM_ENTRY(0xCE, "?"),
315 	SMB_COM_ENTRY(0xCF, "?"),
316 	SMB_COM_ENTRY(0xD0, "?"),
317 	SMB_COM_ENTRY(0xD1, "?"),
318 	SMB_COM_ENTRY(0xD2, "?"),
319 	SMB_COM_ENTRY(0xD3, "?"),
320 	SMB_COM_ENTRY(0xD4, "?"),
321 	SMB_COM_ENTRY(0xD5, "?"),
322 	SMB_COM_ENTRY(0xD6, "?"),
323 	SMB_COM_ENTRY(0xD7, "?"),
324 	SMB_COM_ENTRY(SMB_COM_READ_BULK, "No"),
325 	SMB_COM_ENTRY(SMB_COM_WRITE_BULK, "No"),
326 	SMB_COM_ENTRY(SMB_COM_WRITE_BULK_DATA, "No"),
327 	SMB_COM_ENTRY(0xDB, "?"),
328 	SMB_COM_ENTRY(0xDC, "?"),
329 	SMB_COM_ENTRY(0xDD, "?"),
330 	SMB_COM_ENTRY(0xDE, "?"),
331 	SMB_COM_ENTRY(0xDF, "?"),
332 	SMB_COM_ENTRY(0xE0, "?"),
333 	SMB_COM_ENTRY(0xE1, "?"),
334 	SMB_COM_ENTRY(0xE2, "?"),
335 	SMB_COM_ENTRY(0xE3, "?"),
336 	SMB_COM_ENTRY(0xE4, "?"),
337 	SMB_COM_ENTRY(0xE5, "?"),
338 	SMB_COM_ENTRY(0xE6, "?"),
339 	SMB_COM_ENTRY(0xE7, "?"),
340 	SMB_COM_ENTRY(0xE8, "?"),
341 	SMB_COM_ENTRY(0xE9, "?"),
342 	SMB_COM_ENTRY(0xEA, "?"),
343 	SMB_COM_ENTRY(0xEB, "?"),
344 	SMB_COM_ENTRY(0xEC, "?"),
345 	SMB_COM_ENTRY(0xED, "?"),
346 	SMB_COM_ENTRY(0xEE, "?"),
347 	SMB_COM_ENTRY(0xEF, "?"),
348 	SMB_COM_ENTRY(0xF0, "?"),
349 	SMB_COM_ENTRY(0xF1, "?"),
350 	SMB_COM_ENTRY(0xF2, "?"),
351 	SMB_COM_ENTRY(0xF3, "?"),
352 	SMB_COM_ENTRY(0xF4, "?"),
353 	SMB_COM_ENTRY(0xF5, "?"),
354 	SMB_COM_ENTRY(0xF6, "?"),
355 	SMB_COM_ENTRY(0xF7, "?"),
356 	SMB_COM_ENTRY(0xF8, "?"),
357 	SMB_COM_ENTRY(0xF9, "?"),
358 	SMB_COM_ENTRY(0xFA, "?"),
359 	SMB_COM_ENTRY(0xFB, "?"),
360 	SMB_COM_ENTRY(0xFC, "?"),
361 	SMB_COM_ENTRY(0xFD, "?"),
362 	SMB_COM_ENTRY(0xFE, "?"),
363 	SMB_COM_ENTRY(0xFF, "?")
364 };
365 
366 static const char *smb2_cmd_names[SMB2__NCMDS] = {
367 	"smb2_negotiate",
368 	"smb2_session_setup",
369 	"smb2_logoff",
370 	"smb2_tree_connect",
371 	"smb2_tree_disconn",
372 	"smb2_create",
373 	"smb2_close",
374 	"smb2_flush",
375 	"smb2_read",
376 	"smb2_write",
377 	"smb2_lock",
378 	"smb2_ioctl",
379 	"smb2_cancel",
380 	"smb2_echo",
381 	"smb2_query_dir",
382 	"smb2_change_notify",
383 	"smb2_query_info",
384 	"smb2_set_info",
385 	"smb2_oplock_break",
386 	"smb2_invalid_cmd"
387 };
388 
389 static int smb_dcmd_list(uintptr_t, uint_t, int, const mdb_arg_t *);
390 static void smb_dcmd_list_help(void);
391 static int smb_dcmd_server(uintptr_t, uint_t, int, const mdb_arg_t *);
392 static void smb_dcmd_session_help(void);
393 static int smb_dcmd_session(uintptr_t, uint_t, int, const mdb_arg_t *);
394 static int smb_dcmd_request(uintptr_t, uint_t, int, const mdb_arg_t *);
395 static void smb_dcmd_user_help(void);
396 static int smb_dcmd_user(uintptr_t, uint_t, int, const mdb_arg_t *);
397 static void smb_dcmd_tree_help(void);
398 static int smb_dcmd_tree(uintptr_t, uint_t, int, const mdb_arg_t *);
399 static int smb_dcmd_odir(uintptr_t, uint_t, int, const mdb_arg_t *);
400 static int smb_dcmd_ofile(uintptr_t, uint_t, int, const mdb_arg_t *);
401 static int smb_dcmd_kshare(uintptr_t, uint_t, int, const mdb_arg_t *);
402 static int smb_dcmd_vfs(uintptr_t, uint_t, int, const mdb_arg_t *);
403 static int smb_vfs_walk_init(mdb_walk_state_t *);
404 static int smb_vfs_walk_step(mdb_walk_state_t *);
405 static void smb_node_help(void);
406 static int smb_dcmd_node(uintptr_t, uint_t, int, const mdb_arg_t *);
407 static int smb_node_walk_init(mdb_walk_state_t *);
408 static int smb_node_walk_step(mdb_walk_state_t *);
409 static int smb_lock(uintptr_t, uint_t, int, const mdb_arg_t *);
410 static int smb_oplock(uintptr_t, uint_t, int, const mdb_arg_t *);
411 static int smb_oplock_grant(uintptr_t, uint_t, int, const mdb_arg_t *);
412 static int smb_ace(uintptr_t, uint_t, int, const mdb_arg_t *);
413 static int smb_ace_walk_init(mdb_walk_state_t *);
414 static int smb_ace_walk_step(mdb_walk_state_t *);
415 static int smb_acl(uintptr_t, uint_t, int, const mdb_arg_t *);
416 static int smb_sd(uintptr_t, uint_t, int, const mdb_arg_t *);
417 static int smb_sid(uintptr_t, uint_t, int, const mdb_arg_t *);
418 static int smb_sid_print(uintptr_t);
419 static int smb_fssd(uintptr_t, uint_t, int, const mdb_arg_t *);
420 static int smb_dcmd_getopt(uint_t *, int, const mdb_arg_t *);
421 static int smb_dcmd_setopt(uint_t, int, mdb_arg_t *);
422 static int smb_obj_expand(uintptr_t, uint_t, const smb_exp_t *, ulong_t);
423 static int smb_obj_list(const char *, uint_t, uint_t);
424 static int smb_worker_findstack(uintptr_t);
425 static int smb_stats(uintptr_t, uint_t, int, const mdb_arg_t *);
426 
427 /*
428  * MDB module linkage information:
429  *
430  * We declare a list of structures describing our dcmds, a list of structures
431  * describing our walkers and a function named _mdb_init to return a pointer
432  * to our module information.
433  */
434 static const mdb_dcmd_t dcmds[] = {
435 	{   "smblist",
436 	    "[-seutfdwv]",
437 	    "print tree of SMB objects",
438 	    smb_dcmd_list,
439 	    smb_dcmd_list_help },
440 	{   "smbsrv",
441 	    "[-seutfdwv]",
442 	    "print smb_server information",
443 	    smb_dcmd_server },
444 	{   "smbshares",
445 	    "[-v]",
446 	    "print smb_kshare_t information",
447 	    smb_dcmd_kshare },
448 	{   "smbvfs",
449 	    "[-v]",
450 	    "print smb_vfs information",
451 	    smb_dcmd_vfs },
452 	{   "smbnode",
453 	    "?[-vps]",
454 	    "print smb_node_t information",
455 	    smb_dcmd_node,
456 	    smb_node_help },
457 	{   "smbsess",
458 	    "[-utfdwv]",
459 	    "print smb_session_t information",
460 	    smb_dcmd_session,
461 	    smb_dcmd_session_help},
462 	{   "smbreq",
463 	    ":[-v]",
464 	    "print smb_request_t information",
465 	    smb_dcmd_request },
466 	{   "smblock", ":[-v]",
467 	    "print smb_lock_t information", smb_lock },
468 	{   "smbuser",
469 	    ":[-vdftq]",
470 	    "print smb_user_t information",
471 	    smb_dcmd_user,
472 	    smb_dcmd_user_help },
473 	{   "smbtree",
474 	    ":[-vdf]",
475 	    "print smb_tree_t information",
476 	    smb_dcmd_tree,
477 	    smb_dcmd_tree_help },
478 	{   "smbodir",
479 	    ":[-v]",
480 	    "print smb_odir_t information",
481 	    smb_dcmd_odir },
482 	{   "smbofile",
483 	    "[-v]",
484 	    "print smb_file_t information",
485 	    smb_dcmd_ofile },
486 	{   "smboplock", NULL,
487 	    "print smb_oplock_t information", smb_oplock },
488 	{   "smboplockgrant", NULL,
489 	    "print smb_oplock_grant_t information", smb_oplock_grant },
490 	{   "smbstat", NULL,
491 	    "print all smb dispatched requests statistics",
492 	    smb_stats },
493 	{   "smbace", "[-v]",
494 	    "print smb_ace_t information", smb_ace },
495 	{   "smbacl", "[-v]",
496 	    "print smb_acl_t information", smb_acl },
497 	{   "smbsid", "[-v]",
498 	    "print smb_sid_t information", smb_sid },
499 	{   "smbsd", "[-v]",
500 	    "print smb_sd_t information", smb_sd },
501 	{   "smbfssd", "[-v]",
502 	    "print smb_fssd_t information", smb_fssd },
503 	{ NULL }
504 };
505 
506 static const mdb_walker_t walkers[] = {
507 	{   "smbnode_walker",
508 	    "walk list of smb_node_t structures",
509 	    smb_node_walk_init,
510 	    smb_node_walk_step,
511 	    NULL,
512 	    NULL },
513 	{   "smbvfs_walker",
514 	    "walk list of smb_vfs_t structures",
515 	    smb_vfs_walk_init,
516 	    smb_vfs_walk_step,
517 	    NULL,
518 	    NULL },
519 	{   "smbace_walker",
520 	    "walk list of smb_ace_t structures",
521 	    smb_ace_walk_init,
522 	    smb_ace_walk_step,
523 	    NULL,
524 	    NULL },
525 	{ NULL }
526 };
527 
528 static const mdb_modinfo_t modinfo = {
529 	MDB_API_VERSION, dcmds, walkers
530 };
531 
532 const mdb_modinfo_t *
533 _mdb_init(void)
534 {
535 	return (&modinfo);
536 }
537 
538 /*
539  * *****************************************************************************
540  * ****************************** Top level DCMD *******************************
541  * *****************************************************************************
542  */
543 
544 static void
545 smb_dcmd_list_help(void)
546 {
547 	mdb_printf(
548 	    "Displays the list of objects using an indented tree format.\n"
549 	    "If no option is specified the entire tree is displayed\n\n");
550 	(void) mdb_dec_indent(2);
551 	mdb_printf("%<b>OPTIONS%</b>\n");
552 	(void) mdb_inc_indent(2);
553 	mdb_printf(
554 	    "-v\tDisplay verbose information\n"
555 	    "-s\tDisplay the list of servers\n"
556 	    "-e\tDisplay the list of sessions\n"
557 	    "-r\tDisplay the list of smb requests\n"
558 	    "-u\tDisplay the list of users\n"
559 	    "-t\tDisplay the list of trees\n"
560 	    "-f\tDisplay the list of open files\n"
561 	    "-d\tDisplay the list of open searches\n");
562 }
563 
564 /*
565  * ::smblist
566  *
567  * This function lists the objects specified on the command line. If no object
568  * is specified the entire tree (server through ofile and odir) is displayed.
569  *
570  */
571 /*ARGSUSED*/
572 static int
573 smb_dcmd_list(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
574 {
575 	GElf_Sym	sym;
576 	uint_t		opts = 0;
577 	int		new_argc;
578 	mdb_arg_t	new_argv[SMB_MDB_MAX_OPTS];
579 
580 	if (smb_dcmd_getopt(&opts, argc, argv))
581 		return (DCMD_USAGE);
582 
583 	if (!(opts & ~(SMB_OPT_WALK | SMB_OPT_VERBOSE)))
584 		opts |= SMB_OPT_ALL_OBJ;
585 
586 	opts |= SMB_OPT_WALK;
587 
588 	new_argc = smb_dcmd_setopt(opts, SMB_MDB_MAX_OPTS, new_argv);
589 
590 	if (mdb_lookup_by_obj(SMBSRV_OBJNAME, "smb_servers", &sym) == -1) {
591 		mdb_warn("failed to find symbol smb_servers");
592 		return (DCMD_ERR);
593 	}
594 
595 	addr = (uintptr_t)sym.st_value + OFFSETOF(smb_llist_t, ll_list);
596 
597 	if (mdb_pwalk_dcmd("list", "smbsrv", new_argc, new_argv, addr)) {
598 		mdb_warn("cannot walk smb_server list");
599 		return (DCMD_ERR);
600 	}
601 	return (DCMD_OK);
602 }
603 
604 /*
605  * *****************************************************************************
606  * ***************************** smb_server_t **********************************
607  * *****************************************************************************
608  */
609 
610 static const char *smb_server_state[SMB_SERVER_STATE_SENTINEL] =
611 {
612 	"CREATED",
613 	"CONFIGURED",
614 	"RUNNING",
615 	"STOPPING",
616 	"DELETING"
617 };
618 
619 /*
620  * List of objects that can be expanded under a server structure.
621  */
622 static const smb_exp_t smb_server_exp[] =
623 {
624 	{ SMB_OPT_ALL_OBJ,
625 	    OFFSETOF(smb_server_t, sv_nbt_daemon.ld_session_list.ll_list),
626 	    "smbsess", "smb_session"},
627 	{ SMB_OPT_ALL_OBJ,
628 	    OFFSETOF(smb_server_t, sv_tcp_daemon.ld_session_list.ll_list),
629 	    "smbsess", "smb_session"},
630 	{ 0, 0, NULL, NULL }
631 };
632 
633 /*
634  * ::smbsrv
635  *
636  * smbsrv dcmd - Print out smb_server structures.
637  */
638 /*ARGSUSED*/
639 static int
640 smb_dcmd_server(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
641 {
642 	uint_t		opts;
643 	ulong_t		indent = 0;
644 
645 	if (smb_dcmd_getopt(&opts, argc, argv))
646 		return (DCMD_USAGE);
647 
648 	if (!(flags & DCMD_ADDRSPEC))
649 		return (smb_obj_list("smb_server", opts | SMB_OPT_SERVER,
650 		    flags));
651 
652 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_SERVER)) ||
653 	    !(opts & SMB_OPT_WALK)) {
654 		smb_server_t	*sv;
655 		const char	*state;
656 
657 		sv = mdb_alloc(sizeof (smb_server_t), UM_SLEEP | UM_GC);
658 		if (mdb_vread(sv, sizeof (smb_server_t), addr) == -1) {
659 			mdb_warn("failed to read smb_server at %p", addr);
660 			return (DCMD_ERR);
661 		}
662 
663 		indent = SMB_DCMD_INDENT;
664 
665 		if (opts & SMB_OPT_VERBOSE) {
666 			mdb_arg_t	argv;
667 
668 			argv.a_type = MDB_TYPE_STRING;
669 			argv.a_un.a_str = "smb_server_t";
670 			if (mdb_call_dcmd("print", addr, flags, 1, &argv))
671 				return (DCMD_ERR);
672 		} else {
673 			if (DCMD_HDRSPEC(flags))
674 				mdb_printf(
675 				    "%<b>%<u>%-?s% "
676 				    "%-4s% "
677 				    "%-32s% "
678 				    "%</u>%</b>\n",
679 				    "SERVER", "ZONE", "STATE");
680 
681 			if (sv->sv_state >= SMB_SERVER_STATE_SENTINEL)
682 				state = "UNKNOWN";
683 			else
684 				state = smb_server_state[sv->sv_state];
685 
686 			mdb_printf("%-?p %-4d %-32s \n",
687 			    addr, sv->sv_zid, state);
688 		}
689 	}
690 	if (smb_obj_expand(addr, opts, smb_server_exp, indent))
691 		return (DCMD_ERR);
692 	return (DCMD_OK);
693 }
694 
695 /*
696  * *****************************************************************************
697  * ***************************** smb_session_t *********************************
698  * *****************************************************************************
699  */
700 
701 static const char *smb_session_state[SMB_SESSION_STATE_SENTINEL] =
702 {
703 	"INITIALIZED",
704 	"DISCONNECTED",
705 	"CONNECTED",
706 	"ESTABLISHED",
707 	"NEGOTIATED",
708 	"TERMINATED"
709 };
710 
711 /*
712  * List of objects that can be expanded under a session structure.
713  */
714 static const smb_exp_t smb_session_exp[] =
715 {
716 	{ SMB_OPT_REQUEST,
717 	    OFFSETOF(smb_session_t, s_req_list.sl_list),
718 	    "smbreq", "smb_request"},
719 	{ SMB_OPT_USER,
720 	    OFFSETOF(smb_session_t, s_user_list.ll_list),
721 	    "smbuser", "smb_user"},
722 	{ SMB_OPT_TREE | SMB_OPT_OFILE | SMB_OPT_ODIR,
723 	    OFFSETOF(smb_session_t, s_tree_list.ll_list),
724 	    "smbtree", "smb_tree"},
725 	{ 0, 0, NULL, NULL}
726 };
727 
728 static void
729 smb_dcmd_session_help(void)
730 {
731 	mdb_printf(
732 	    "Display the contents of smb_session_t, with optional"
733 	    " filtering.\n\n");
734 	(void) mdb_dec_indent(2);
735 	mdb_printf("%<b>OPTIONS%</b>\n");
736 	(void) mdb_inc_indent(2);
737 	mdb_printf(
738 	    "-v\tDisplay verbose smb_session information\n"
739 	    "-r\tDisplay the list of smb requests attached\n"
740 	    "-u\tDisplay the list of users attached\n");
741 }
742 
743 /*
744  * ::smbsess
745  *
746  * smbsess dcmd - Print out the smb_session structure.
747  */
748 static int
749 smb_dcmd_session(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
750 {
751 	uint_t		opts;
752 	ulong_t		indent = 0;
753 
754 	if (smb_dcmd_getopt(&opts, argc, argv))
755 		return (DCMD_USAGE);
756 
757 	if (!(flags & DCMD_ADDRSPEC)) {
758 		opts |= SMB_OPT_SESSION;
759 		opts &= ~SMB_OPT_SERVER;
760 		return (smb_obj_list("smb_session", opts, flags));
761 	}
762 
763 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_SESSION)) ||
764 	    !(opts & SMB_OPT_WALK)) {
765 		char	cipaddr[INET6_ADDRSTRLEN];
766 		char	lipaddr[INET6_ADDRSTRLEN];
767 		int	ipaddrstrlen;
768 		smb_session_t	*se;
769 		const char	*state;
770 
771 		indent = SMB_DCMD_INDENT;
772 
773 		se = mdb_alloc(sizeof (*se), UM_SLEEP | UM_GC);
774 		if (mdb_vread(se, sizeof (*se), addr) == -1) {
775 			mdb_warn("failed to read smb_session at %p", addr);
776 			return (DCMD_ERR);
777 		}
778 		if (se->s_state >= SMB_SESSION_STATE_SENTINEL)
779 			state = "INVALID";
780 		else
781 			state = smb_session_state[se->s_state];
782 
783 		switch (se->ipaddr.a_family) {
784 		case AF_INET:
785 			ipaddrstrlen = INET_ADDRSTRLEN;
786 			(void) mdb_snprintf(cipaddr, sizeof (cipaddr),
787 			    "%I", se->ipaddr.a_ipv4);
788 			(void) mdb_snprintf(lipaddr, sizeof (lipaddr),
789 			    "%I", se->local_ipaddr.a_ipv4);
790 			break;
791 		case AF_INET6:
792 			ipaddrstrlen = INET6_ADDRSTRLEN;
793 			(void) mdb_snprintf(cipaddr, sizeof (cipaddr),
794 			    "%N", &(se->ipaddr.a_ipv6));
795 			(void) mdb_snprintf(lipaddr, sizeof (lipaddr),
796 			    "%N", &(se->local_ipaddr.a_ipv6));
797 			break;
798 		default:
799 			ipaddrstrlen = INET_ADDRSTRLEN;
800 			(void) mdb_snprintf(cipaddr, sizeof (cipaddr),
801 			    "unknown");
802 			(void) mdb_snprintf(lipaddr, sizeof (lipaddr),
803 			    "unknown");
804 		}
805 
806 		if (opts & SMB_OPT_VERBOSE) {
807 			mdb_printf("%<b>%<u>SMB session information "
808 			    "(%p): %</u>%</b>\n", addr);
809 			mdb_printf("Client IP address: %s %d\n",
810 			    cipaddr, se->s_remote_port);
811 			mdb_printf("Local IP Address: %s %d\n",
812 			    lipaddr, se->s_local_port);
813 			mdb_printf("Session KID: %u\n", se->s_kid);
814 			mdb_printf("Workstation Name: %s\n",
815 			    se->workstation);
816 			mdb_printf("Session state: %u (%s)\n", se->s_state,
817 			    state);
818 			mdb_printf("Session dialect: %#x\n", se->dialect);
819 			mdb_printf("Number of Users: %u\n",
820 			    se->s_user_list.ll_count);
821 			mdb_printf("Number of Trees: %u\n", se->s_tree_cnt);
822 			mdb_printf("Number of Files: %u\n", se->s_file_cnt);
823 			mdb_printf("Number of Shares: %u\n", se->s_dir_cnt);
824 			mdb_printf("Number of active Transact.: %u\n\n",
825 			    se->s_xa_list.ll_count);
826 		} else {
827 			if (DCMD_HDRSPEC(flags)) {
828 				mdb_printf(
829 			"%<b>%<u>%-?s %-*s %-8s %-8s %-12s%</u>%</b>\n",
830 				    "SESSION", ipaddrstrlen, "IP_ADDR",
831 				    "PORT", "DIALECT", "STATE");
832 			}
833 			mdb_printf("%-?p %-*s %-8d %-8#x %s\n",
834 			    addr, ipaddrstrlen, cipaddr,
835 			    se->s_remote_port, se->dialect, state);
836 		}
837 	}
838 	if (smb_obj_expand(addr, opts, smb_session_exp, indent))
839 		return (DCMD_ERR);
840 	return (DCMD_OK);
841 }
842 
843 /*
844  * *****************************************************************************
845  * **************************** smb_request_t **********************************
846  * *****************************************************************************
847  */
848 
849 static const char *smb_request_state[SMB_REQ_STATE_SENTINEL] =
850 {
851 	"FREE",
852 	"INITIALIZING",
853 	"SUBMITTED",
854 	"ACTIVE",
855 	"WAITING_EVENT",
856 	"EVENT_OCCURRED",
857 	"WAITING_LOCK",
858 	"COMPLETED",
859 	"CANCELED",
860 	"CLEANED_UP"
861 };
862 
863 #define	SMB_REQUEST_BANNER	\
864 	"%<b>%<u>%-?s %-?s %-14s %-14s %-16s %-32s%</u>%</b>\n"
865 #define	SMB_REQUEST_FORMAT	\
866 	"%-?p %-?p %-14lld %-14lld %-16s %s\n"
867 
868 static int
869 smb_dcmd_request(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
870 {
871 	uint_t		opts;
872 
873 	if (smb_dcmd_getopt(&opts, argc, argv))
874 		return (DCMD_USAGE);
875 
876 	if (!(flags & DCMD_ADDRSPEC)) {
877 		opts |= SMB_OPT_REQUEST;
878 		opts &= ~(SMB_OPT_SERVER | SMB_OPT_SESSION | SMB_OPT_USER);
879 		return (smb_obj_list("smb_request", opts, flags));
880 	}
881 
882 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_REQUEST)) ||
883 	    !(opts & SMB_OPT_WALK)) {
884 		smb_request_t	*sr;
885 		const char	*state;
886 		const char	*cur_cmd_name;
887 		uint_t		cur_cmd_code;
888 		uint64_t	waiting;
889 		uint64_t	running;
890 
891 		sr = mdb_alloc(sizeof (*sr), UM_SLEEP | UM_GC);
892 		if (mdb_vread(sr, sizeof (*sr), addr) == -1) {
893 			mdb_warn("failed to read smb_request at %p", addr);
894 			return (DCMD_ERR);
895 		}
896 		if (sr->sr_magic != SMB_REQ_MAGIC) {
897 			mdb_warn("not an smb_request_t (%p)>", addr);
898 			return (DCMD_ERR);
899 		}
900 		waiting = 0;
901 		running = 0;
902 		/*
903 		 * Note: mdb_gethrtime() is only available in kmdb
904 		 */
905 #ifdef	_KERNEL
906 		if (sr->sr_time_submitted != 0) {
907 			if (sr->sr_time_active != 0) {
908 				waiting = sr->sr_time_active -
909 				    sr->sr_time_submitted;
910 				running = mdb_gethrtime() -
911 				    sr->sr_time_active;
912 			} else {
913 				waiting = mdb_gethrtime() -
914 				    sr->sr_time_submitted;
915 			}
916 		}
917 		waiting /= NANOSEC;
918 		running /= NANOSEC;
919 #endif	/* _KERNEL */
920 
921 		if (sr->sr_state >= SMB_REQ_STATE_SENTINEL)
922 			state = "INVALID";
923 		else
924 			state = smb_request_state[sr->sr_state];
925 
926 		if (sr->smb2_cmd_code != 0) {
927 			/* SMB2 request */
928 			cur_cmd_code = sr->smb2_cmd_code;
929 			if (cur_cmd_code > SMB2_INVALID_CMD)
930 				cur_cmd_code = SMB2_INVALID_CMD;
931 			cur_cmd_name = smb2_cmd_names[cur_cmd_code];
932 		} else {
933 			/* SMB1 request */
934 			cur_cmd_code = sr->smb_com & 0xFF;
935 			cur_cmd_name = smb_com[cur_cmd_code].smb_com;
936 		}
937 
938 		if (opts & SMB_OPT_VERBOSE) {
939 			mdb_printf(
940 			    "%</b>%</u>SMB request information (%p):"
941 			    "%</u>%</b>\n\n", addr);
942 
943 			if (sr->smb2_cmd_code == 0) {
944 				/* SMB1 request */
945 				mdb_printf(
946 				    "first SMB COM: %u (%s)\n",
947 				    sr->first_smb_com,
948 				    smb_com[sr->first_smb_com].smb_com);
949 			}
950 
951 			mdb_printf(
952 			    "current SMB COM: %u (%s)\n",
953 			    cur_cmd_code, cur_cmd_name);
954 
955 			mdb_printf(
956 			    "state: %u (%s)\n",
957 			    sr->sr_state, state);
958 
959 			mdb_printf(
960 			    "TID(tree): %u (%p)\n",
961 			    sr->smb_tid, sr->tid_tree);
962 
963 			mdb_printf(
964 			    "UID(user): %u (%p)\n",
965 			    sr->smb_uid, sr->uid_user);
966 
967 			mdb_printf(
968 			    "FID(file): %u (%p)\n",
969 			    sr->smb_fid, sr->fid_ofile);
970 
971 			mdb_printf(
972 			    "PID: %u\n",
973 			    sr->smb_pid);
974 
975 			if (sr->smb2_messageid != 0) {
976 				mdb_printf(
977 				    "MID: 0x%llx\n\n",
978 				    sr->smb2_messageid);
979 			} else {
980 				mdb_printf(
981 				    "MID: %u\n\n",
982 				    sr->smb_mid);
983 			}
984 
985 			mdb_printf(
986 			    "waiting time: %lld\n",
987 			    waiting);
988 
989 			mdb_printf(
990 			    "running time: %lld\n",
991 			    running);
992 
993 			mdb_printf(
994 			    "worker thread: %p\n",
995 			    sr->sr_worker);
996 			smb_worker_findstack((uintptr_t)sr->sr_worker);
997 		} else {
998 			if (DCMD_HDRSPEC(flags))
999 				mdb_printf(
1000 				    SMB_REQUEST_BANNER,
1001 				    "ADDR",
1002 				    "WORKER",
1003 				    "WAITING(s)",
1004 				    "RUNNING(s)",
1005 				    "STATE",
1006 				    "COMMAND");
1007 
1008 			mdb_printf(
1009 			    SMB_REQUEST_FORMAT,
1010 			    addr,
1011 			    sr->sr_worker,
1012 			    waiting,
1013 			    running,
1014 			    state,
1015 			    cur_cmd_name);
1016 		}
1017 	}
1018 	return (DCMD_OK);
1019 }
1020 
1021 /*
1022  * *****************************************************************************
1023  * ****************************** smb_user_t ***********************************
1024  * *****************************************************************************
1025  */
1026 
1027 static const char *smb_user_state[SMB_USER_STATE_SENTINEL] =
1028 {
1029 	"LOGGING_ON",
1030 	"LOGGED_ON",
1031 	"LOGGING_OFF",
1032 	"LOGGED_OFF"
1033 };
1034 
1035 static void
1036 smb_dcmd_user_help(void)
1037 {
1038 	mdb_printf(
1039 	    "Display the contents of smb_user_t, with optional filtering.\n\n");
1040 	(void) mdb_dec_indent(2);
1041 	mdb_printf("%<b>OPTIONS%</b>\n");
1042 	(void) mdb_inc_indent(2);
1043 	mdb_printf(
1044 	    "-v\tDisplay verbose smb_user information\n");
1045 }
1046 
1047 static int
1048 smb_dcmd_user(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1049 {
1050 	uint_t		opts;
1051 
1052 	if (smb_dcmd_getopt(&opts, argc, argv))
1053 		return (DCMD_USAGE);
1054 
1055 	if (!(flags & DCMD_ADDRSPEC)) {
1056 		opts |= SMB_OPT_USER;
1057 		opts &= ~(SMB_OPT_SERVER | SMB_OPT_SESSION | SMB_OPT_REQUEST);
1058 		return (smb_obj_list("smb_user", opts, flags));
1059 	}
1060 
1061 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_USER)) ||
1062 	    !(opts & SMB_OPT_WALK)) {
1063 		smb_user_t	*user;
1064 		char		*account;
1065 
1066 		user = mdb_alloc(sizeof (*user), UM_SLEEP | UM_GC);
1067 		if (mdb_vread(user, sizeof (*user), addr) == -1) {
1068 			mdb_warn("failed to read smb_user at %p", addr);
1069 			return (DCMD_ERR);
1070 		}
1071 		account = mdb_zalloc(user->u_domain_len + user->u_name_len + 2,
1072 		    UM_SLEEP | UM_GC);
1073 
1074 		if (user->u_domain_len)
1075 			(void) mdb_vread(account, user->u_domain_len,
1076 			    (uintptr_t)user->u_domain);
1077 
1078 		strcat(account, "\\");
1079 
1080 		if (user->u_name_len)
1081 			(void) mdb_vread(account + strlen(account),
1082 			    user->u_name_len, (uintptr_t)user->u_name);
1083 
1084 		if (opts & SMB_OPT_VERBOSE) {
1085 			const char	*state;
1086 
1087 			if (user->u_state >= SMB_USER_STATE_SENTINEL)
1088 				state = "INVALID";
1089 			else
1090 				state = smb_user_state[user->u_state];
1091 
1092 			mdb_printf("%<b>%<u>SMB user information (%p):"
1093 			    "%</u>%</b>\n", addr);
1094 			mdb_printf("UID: %u\n", user->u_uid);
1095 			mdb_printf("State: %d (%s)\n", user->u_state, state);
1096 			mdb_printf("Flags: 0x%08x\n", user->u_flags);
1097 			mdb_printf("Privileges: 0x%08x\n", user->u_privileges);
1098 			mdb_printf("Credential: %p\n", user->u_cred);
1099 			mdb_printf("Reference Count: %d\n", user->u_refcnt);
1100 			mdb_printf("User Account: %s\n\n", account);
1101 		} else {
1102 			if (DCMD_HDRSPEC(flags))
1103 				mdb_printf(
1104 				    "%<b>%<u>%?-s "
1105 				    "%-5s "
1106 				    "%-32s%</u>%</b>\n",
1107 				    "USER", "UID", "ACCOUNT");
1108 
1109 			mdb_printf("%-?p %-5u %-32s\n", addr, user->u_uid,
1110 			    account);
1111 		}
1112 	}
1113 	return (DCMD_OK);
1114 }
1115 
1116 /*
1117  * *****************************************************************************
1118  * ****************************** smb_tree_t ***********************************
1119  * *****************************************************************************
1120  */
1121 
1122 static const char *smb_tree_state[SMB_TREE_STATE_SENTINEL] =
1123 {
1124 	"CONNECTED",
1125 	"DISCONNECTING",
1126 	"DISCONNECTED"
1127 };
1128 
1129 /*
1130  * List of objects that can be expanded under a tree structure.
1131  */
1132 static const smb_exp_t smb_tree_exp[] =
1133 {
1134 	{ SMB_OPT_OFILE,
1135 	    OFFSETOF(smb_tree_t, t_ofile_list.ll_list),
1136 	    "smbofile", "smb_ofile"},
1137 	{ SMB_OPT_ODIR,
1138 	    OFFSETOF(smb_tree_t, t_odir_list.ll_list),
1139 	    "smbodir", "smb_odir"},
1140 	{ 0, 0, NULL, NULL}
1141 };
1142 
1143 static void
1144 smb_dcmd_tree_help(void)
1145 {
1146 	mdb_printf(
1147 	    "Display the contents of smb_tree_t, with optional filtering.\n\n");
1148 	(void) mdb_dec_indent(2);
1149 	mdb_printf("%<b>OPTIONS%</b>\n");
1150 	(void) mdb_inc_indent(2);
1151 	mdb_printf(
1152 	    "-v\tDisplay verbose smb_tree information\n"
1153 	    "-d\tDisplay the list of smb_odirs attached\n"
1154 	    "-f\tDisplay the list of smb_ofiles attached\n");
1155 }
1156 
1157 static int
1158 smb_dcmd_tree(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1159 {
1160 	uint_t		opts;
1161 	ulong_t		indent = 0;
1162 
1163 	if (smb_dcmd_getopt(&opts, argc, argv))
1164 		return (DCMD_USAGE);
1165 
1166 	if (!(flags & DCMD_ADDRSPEC)) {
1167 		opts |= SMB_OPT_TREE;
1168 		opts &= ~(SMB_OPT_SERVER | SMB_OPT_SESSION | SMB_OPT_REQUEST |
1169 		    SMB_OPT_USER);
1170 		return (smb_obj_list("smb_tree", opts, flags));
1171 	}
1172 
1173 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_TREE)) ||
1174 	    !(opts & SMB_OPT_WALK)) {
1175 		smb_tree_t	*tree;
1176 
1177 		indent = SMB_DCMD_INDENT;
1178 
1179 		tree = mdb_alloc(sizeof (*tree), UM_SLEEP | UM_GC);
1180 		if (mdb_vread(tree, sizeof (*tree), addr) == -1) {
1181 			mdb_warn("failed to read smb_tree at %p", addr);
1182 			return (DCMD_ERR);
1183 		}
1184 		if (opts & SMB_OPT_VERBOSE) {
1185 			const char	*state;
1186 
1187 			if (tree->t_state >= SMB_TREE_STATE_SENTINEL)
1188 				state = "INVALID";
1189 			else
1190 				state = smb_tree_state[tree->t_state];
1191 
1192 			mdb_printf("%<b>%<u>SMB tree information (%p):"
1193 			    "%</u>%</b>\n\n", addr);
1194 			mdb_printf("TID: %04x\n", tree->t_tid);
1195 			mdb_printf("State: %d (%s)\n", tree->t_state, state);
1196 			mdb_printf("Share: %s\n", tree->t_sharename);
1197 			mdb_printf("Resource: %s\n", tree->t_resource);
1198 			mdb_printf("Type: %s\n", tree->t_typename);
1199 			mdb_printf("Volume: %s\n", tree->t_volume);
1200 			mdb_printf("Umask: %04x\n", tree->t_umask);
1201 			mdb_printf("Flags: %08x\n", tree->t_flags);
1202 			mdb_printf("SMB Node: %llx\n", tree->t_snode);
1203 			mdb_printf("Reference Count: %d\n\n", tree->t_refcnt);
1204 		} else {
1205 			if (DCMD_HDRSPEC(flags))
1206 				mdb_printf(
1207 				    "%<b>%<u>%-?s %-5s %-16s %-32s%</u>%</b>\n",
1208 				    "TREE", "TID", "SHARE NAME", "RESOURCE");
1209 
1210 			mdb_printf("%-?p %-5u %-16s %-32s\n", addr,
1211 			    tree->t_tid, tree->t_sharename, tree->t_resource);
1212 		}
1213 	}
1214 	if (smb_obj_expand(addr, opts, smb_tree_exp, indent))
1215 		return (DCMD_ERR);
1216 	return (DCMD_OK);
1217 }
1218 
1219 /*
1220  * *****************************************************************************
1221  * ****************************** smb_odir_t ***********************************
1222  * *****************************************************************************
1223  */
1224 
1225 static const char *smb_odir_state[SMB_ODIR_STATE_SENTINEL] =
1226 {
1227 	"OPEN",
1228 	"IN_USE",
1229 	"CLOSING",
1230 	"CLOSED"
1231 };
1232 
1233 static int
1234 smb_dcmd_odir(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1235 {
1236 	uint_t		opts;
1237 
1238 	if (smb_dcmd_getopt(&opts, argc, argv))
1239 		return (DCMD_USAGE);
1240 
1241 	if (!(flags & DCMD_ADDRSPEC)) {
1242 		opts |= SMB_OPT_ODIR;
1243 		opts &= ~(SMB_OPT_SERVER | SMB_OPT_SESSION | SMB_OPT_REQUEST |
1244 		    SMB_OPT_USER | SMB_OPT_TREE | SMB_OPT_OFILE);
1245 		return (smb_obj_list("smb_odir", opts, flags));
1246 	}
1247 
1248 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_ODIR)) ||
1249 	    !(opts & SMB_OPT_WALK)) {
1250 		smb_odir_t	*od;
1251 
1252 		od = mdb_alloc(sizeof (*od), UM_SLEEP | UM_GC);
1253 		if (mdb_vread(od, sizeof (*od), addr) == -1) {
1254 			mdb_warn("failed to read smb_odir at %p", addr);
1255 			return (DCMD_ERR);
1256 		}
1257 		if (opts & SMB_OPT_VERBOSE) {
1258 			const char	*state;
1259 
1260 			if (od->d_state >= SMB_ODIR_STATE_SENTINEL)
1261 				state = "INVALID";
1262 			else
1263 				state = smb_odir_state[od->d_state];
1264 
1265 			mdb_printf(
1266 			    "%<b>%<u>SMB odir information (%p):%</u>%</b>\n\n",
1267 			    addr);
1268 			mdb_printf("State: %d (%s)\n", od->d_state, state);
1269 			mdb_printf("SID: %u\n", od->d_odid);
1270 			mdb_printf("User: %p\n", od->d_user);
1271 			mdb_printf("Tree: %p\n", od->d_tree);
1272 			mdb_printf("Reference Count: %d\n", od->d_refcnt);
1273 			mdb_printf("Pattern: %s\n", od->d_pattern);
1274 			mdb_printf("SMB Node: %p\n\n", od->d_dnode);
1275 		} else {
1276 			if (DCMD_HDRSPEC(flags))
1277 				mdb_printf(
1278 				    "%<b>%<u>%-?s "
1279 				    "%-5s "
1280 				    "%-?s "
1281 				    "%-16s%</u>%</b>\n",
1282 				    "ODIR", "SID", "VNODE", "PATTERN");
1283 
1284 			mdb_printf("%?p %-5u %-16p %s\n",
1285 			    addr, od->d_odid, od->d_dnode, od->d_pattern);
1286 		}
1287 	}
1288 	return (DCMD_OK);
1289 }
1290 
1291 /*
1292  * *****************************************************************************
1293  * ****************************** smb_ofile_t **********************************
1294  * *****************************************************************************
1295  */
1296 
1297 static const char *smb_ofile_state[SMB_OFILE_STATE_SENTINEL] =
1298 {
1299 	"OPEN",
1300 	"CLOSING",
1301 	"CLOSED"
1302 };
1303 
1304 static int
1305 smb_dcmd_ofile(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1306 {
1307 	uint_t		opts;
1308 
1309 	if (smb_dcmd_getopt(&opts, argc, argv))
1310 		return (DCMD_USAGE);
1311 
1312 	if (!(flags & DCMD_ADDRSPEC)) {
1313 		opts |= SMB_OPT_OFILE;
1314 		opts &= ~(SMB_OPT_SERVER | SMB_OPT_SESSION | SMB_OPT_REQUEST |
1315 		    SMB_OPT_USER | SMB_OPT_TREE | SMB_OPT_ODIR);
1316 		return (smb_obj_list("smb_ofile", opts, flags));
1317 	}
1318 
1319 	if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_OFILE)) ||
1320 	    !(opts & SMB_OPT_WALK)) {
1321 		smb_ofile_t	*of;
1322 
1323 		of = mdb_alloc(sizeof (*of), UM_SLEEP | UM_GC);
1324 		if (mdb_vread(of, sizeof (*of), addr) == -1) {
1325 			mdb_warn("failed to read smb_ofile at %p", addr);
1326 			return (DCMD_ERR);
1327 		}
1328 		if (opts & SMB_OPT_VERBOSE) {
1329 			const char	*state;
1330 
1331 			if (of->f_state >= SMB_OFILE_STATE_SENTINEL)
1332 				state = "INVALID";
1333 			else
1334 				state = smb_ofile_state[of->f_state];
1335 
1336 			mdb_printf(
1337 			    "%<b>%<u>SMB ofile information (%p):%</u>%</b>\n\n",
1338 			    addr);
1339 			mdb_printf("FID: %u\n", of->f_fid);
1340 			mdb_printf("State: %d (%s)\n", of->f_state, state);
1341 			mdb_printf("SMB Node: %p\n", of->f_node);
1342 			mdb_printf("LLF Offset: 0x%llx (%s)\n",
1343 			    of->f_llf_pos,
1344 			    ((of->f_flags & SMB_OFLAGS_LLF_POS_VALID) ?
1345 			    "Valid" : "Invalid"));
1346 			mdb_printf("Flags: 0x%08x\n", of->f_flags);
1347 			mdb_printf("User: %p\n", of->f_user);
1348 			mdb_printf("Tree: %p\n", of->f_tree);
1349 			mdb_printf("Credential: %p\n\n", of->f_cr);
1350 		} else {
1351 			if (DCMD_HDRSPEC(flags))
1352 				mdb_printf(
1353 				    "%<b>%<u>%-?s "
1354 				    "%-5s "
1355 				    "%-?s "
1356 				    "%-?s%</u>%</b>\n",
1357 				    "OFILE", "FID", "SMB NODE", "CRED");
1358 
1359 			mdb_printf("%?p %-5u %-p %p\n", addr,
1360 			    of->f_fid, of->f_node, of->f_cr);
1361 		}
1362 	}
1363 	return (DCMD_OK);
1364 }
1365 
1366 /*
1367  * *****************************************************************************
1368  * ******************************** smb_kshare_t *******************************
1369  * *****************************************************************************
1370  */
1371 
1372 static int
1373 smb_kshare_cb(uintptr_t addr, const void *data, void *arg)
1374 {
1375 	uint_t *opts = arg;
1376 	uintptr_t ta, sa;
1377 	char name[32];
1378 	char path[64];
1379 	_NOTE(ARGUNUSED(data));
1380 
1381 	if (*opts & SMB_OPT_VERBOSE) {
1382 		mdb_arg_t	argv;
1383 
1384 		argv.a_type = MDB_TYPE_STRING;
1385 		argv.a_un.a_str = "smb_kshare_t";
1386 		/* Don't fail the walk if this fails. */
1387 		mdb_call_dcmd("print", addr, 0, 1, &argv);
1388 	} else {
1389 		/*
1390 		 * Summary line for a kshare
1391 		 * Don't fail the walk if any of these fail.
1392 		 */
1393 		ta = addr + OFFSETOF(smb_kshare_t, shr_name);
1394 		if (mdb_vread(&sa, sizeof (sa), ta) < 0 ||
1395 		    mdb_readstr(name, sizeof (name), sa) <= 0)
1396 			strcpy(name, "?");
1397 
1398 		ta = addr + OFFSETOF(smb_kshare_t, shr_path);
1399 		if (mdb_vread(&sa, sizeof (sa), ta) < 0 ||
1400 		    mdb_readstr(path, sizeof (path), sa) <= 0)
1401 			strcpy(path, "?");
1402 
1403 		mdb_printf("%-?p ", addr);	/* smb_kshare_t */
1404 		mdb_printf("%-16s ", name);
1405 		mdb_printf("%-s", path);
1406 		mdb_printf("\n");
1407 	}
1408 
1409 	return (WALK_NEXT);
1410 }
1411 
1412 /*
1413  * ::smbshares
1414  *
1415  * dcmd - Print out smb_kshare structures.
1416  *	requires addr of an smb_server_t
1417  */
1418 /*ARGSUSED*/
1419 static int
1420 smb_dcmd_kshare(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1421 {
1422 	uint_t		opts = 0;
1423 
1424 	if (mdb_getopts(argc, argv,
1425 	    'v', MDB_OPT_SETBITS, SMB_OPT_VERBOSE, &opts,
1426 	    NULL) != argc)
1427 		return (DCMD_USAGE);
1428 
1429 	if (!(flags & DCMD_ADDRSPEC))
1430 		return (DCMD_USAGE);
1431 	addr += OFFSETOF(smb_server_t, sv_export.e_share_avl.avl_tree);
1432 
1433 	if (DCMD_HDRSPEC(flags)) {
1434 		mdb_printf(
1435 		    "%<b>%<u>"
1436 		    "%-?s "
1437 		    "%-16s "
1438 		    "%-s"
1439 		    "%</u>%</b>\n",
1440 		    "smb_kshare_t", "name", "path");
1441 	}
1442 
1443 	if (mdb_pwalk("avl", smb_kshare_cb, &opts, addr) == -1) {
1444 		mdb_warn("cannot walk smb_kshare avl");
1445 		return (DCMD_ERR);
1446 	}
1447 
1448 	return (DCMD_OK);
1449 }
1450 
1451 /*
1452  * *****************************************************************************
1453  * ******************************** smb_vfs_t **********************************
1454  * *****************************************************************************
1455  */
1456 
1457 /*
1458  * ::smbvfs
1459  *
1460  * smbvfs dcmd - Prints out smb_vfs structures.
1461  */
1462 /*ARGSUSED*/
1463 static int
1464 smb_dcmd_vfs(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1465 {
1466 	int		verbose = FALSE;
1467 	smb_vfs_t	*sf;
1468 	vnode_t		*vn;
1469 	char		*path;
1470 
1471 	if (mdb_getopts(argc, argv,
1472 	    'v', MDB_OPT_SETBITS, TRUE, &verbose,
1473 	    NULL) != argc)
1474 		return (DCMD_USAGE);
1475 
1476 	/*
1477 	 * If no smb_vfs address was specified on the command line, we can
1478 	 * print out all smb_vfs by invoking the smb_vfs walker, using
1479 	 * this dcmd itself as the callback.
1480 	 */
1481 	if (!(flags & DCMD_ADDRSPEC)) {
1482 		if (mdb_walk_dcmd("smbvfs_walker", "smbvfs",
1483 		    argc, argv) == -1) {
1484 			mdb_warn("failed to walk 'smb_vfs'");
1485 			return (DCMD_ERR);
1486 		}
1487 		return (DCMD_OK);
1488 	}
1489 
1490 	if (DCMD_HDRSPEC(flags)) {
1491 		mdb_printf(
1492 		    "%<b>%<u>"
1493 		    "%-?s "
1494 		    "%-10s "
1495 		    "%-16s "
1496 		    "%-16s"
1497 		    "%-s"
1498 		    "%</u>%</b>\n",
1499 		    "SMB_VFS", "REFCNT", "VFS", "VNODE", "ROOT");
1500 	}
1501 
1502 	sf = mdb_alloc(sizeof (*sf), UM_SLEEP | UM_GC);
1503 	if (mdb_vread(sf, sizeof (*sf), addr) == -1) {
1504 		mdb_warn("failed to read smb_vfs at %p", addr);
1505 		return (DCMD_ERR);
1506 	}
1507 
1508 	vn = mdb_alloc(sizeof (*vn), UM_SLEEP | UM_GC);
1509 	if (mdb_vread(vn, sizeof (*vn),
1510 	    (uintptr_t)sf->sv_rootvp) == -1) {
1511 		mdb_warn("failed to read vnode at %p", sf->sv_rootvp);
1512 		return (DCMD_ERR);
1513 	}
1514 
1515 	path = mdb_zalloc(MAXPATHLEN, UM_SLEEP | UM_GC);
1516 	(void) mdb_vread(path, MAXPATHLEN, (uintptr_t)vn->v_path);
1517 
1518 	mdb_printf(
1519 	    "%-?p %-10d %-?p %-?p %-s\n", addr, sf->sv_refcnt,
1520 	    sf->sv_vfsp, sf->sv_rootvp, path);
1521 
1522 	return (DCMD_OK);
1523 }
1524 
1525 /*
1526  * Initialize the smb_vfs_t walker to point to the smb_export
1527  * in the specified smb_server_t instance.  (no global walks)
1528  */
1529 static int
1530 smb_vfs_walk_init(mdb_walk_state_t *wsp)
1531 {
1532 
1533 	if (wsp->walk_addr == NULL) {
1534 		mdb_printf("require address of an smb_server_t\n");
1535 		return (WALK_ERR);
1536 	}
1537 
1538 	wsp->walk_addr +=
1539 	    OFFSETOF(smb_server_t, sv_export.e_vfs_list.ll_list);
1540 
1541 	if (mdb_layered_walk("list", wsp) == -1) {
1542 		mdb_warn("failed to walk list of VFS");
1543 		return (WALK_ERR);
1544 	}
1545 
1546 	return (WALK_NEXT);
1547 }
1548 
1549 static int
1550 smb_vfs_walk_step(mdb_walk_state_t *wsp)
1551 {
1552 	return (wsp->walk_callback(wsp->walk_addr, wsp->walk_layer,
1553 	    wsp->walk_cbdata));
1554 }
1555 
1556 /*
1557  * *****************************************************************************
1558  * ******************************* smb_node_t **********************************
1559  * *****************************************************************************
1560  */
1561 
1562 static void
1563 smb_node_help(void)
1564 {
1565 	mdb_printf(
1566 	    "Display the contents of smb_node_t, with optional filtering.\n\n");
1567 	(void) mdb_dec_indent(2);
1568 	mdb_printf("%<b>OPTIONS%</b>\n");
1569 	(void) mdb_inc_indent(2);
1570 	mdb_printf(
1571 	    "-v\tDisplay verbose smb_node information\n"
1572 	    "-p\tDisplay the full path of the vnode associated\n"
1573 	    "-s\tDisplay the stack of the last 16 calls that modified the "
1574 	    "reference\n\tcount\n");
1575 }
1576 
1577 /*
1578  * ::smbnode
1579  *
1580  * smb_node dcmd - Print out smb_node structure.
1581  */
1582 static int
1583 smb_dcmd_node(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1584 {
1585 	smb_node_t	node;
1586 	int		rc;
1587 	int		verbose = FALSE;
1588 	int		print_full_path = FALSE;
1589 	int		stack_trace = FALSE;
1590 	vnode_t		vnode;
1591 	char		od_name[MAXNAMELEN];
1592 	char		path_name[1024];
1593 	uintptr_t	list_addr, oplock_addr;
1594 
1595 	if (mdb_getopts(argc, argv,
1596 	    'v', MDB_OPT_SETBITS, TRUE, &verbose,
1597 	    'p', MDB_OPT_SETBITS, TRUE, &print_full_path,
1598 	    's', MDB_OPT_SETBITS, TRUE, &stack_trace,
1599 	    NULL) != argc)
1600 		return (DCMD_USAGE);
1601 
1602 	/*
1603 	 * If no smb_node address was specified on the command line, we can
1604 	 * print out all smb nodes by invoking the smb_node walker, using
1605 	 * this dcmd itself as the callback.
1606 	 */
1607 	if (!(flags & DCMD_ADDRSPEC)) {
1608 		if (mdb_walk_dcmd("smbnode_walker", "smbnode",
1609 		    argc, argv) == -1) {
1610 			mdb_warn("failed to walk 'smb_node'");
1611 			return (DCMD_ERR);
1612 		}
1613 		return (DCMD_OK);
1614 	}
1615 
1616 	/*
1617 	 * If this is the first invocation of the command, print a nice
1618 	 * header line for the output that will follow.
1619 	 */
1620 	if (DCMD_HDRSPEC(flags)) {
1621 		if (verbose) {
1622 			mdb_printf("%<b>%<u>SMB node information:%</u>%</b>\n");
1623 		} else {
1624 			mdb_printf(
1625 			    "%<b>%<u>%-?s "
1626 			    "%-?s "
1627 			    "%-18s "
1628 			    "%-6s "
1629 			    "%-6s "
1630 			    "%-8s "
1631 			    "%-6s%</u>%</b>\n",
1632 			    "ADDR", "VP", "NODE-NAME", "OFILES", "LOCKS",
1633 			    "OPLOCK", "REF");
1634 		}
1635 	}
1636 
1637 	/*
1638 	 * For each smb_node, we just need to read the smb_node_t struct, read
1639 	 * and then print out the following fields.
1640 	 */
1641 	if (mdb_vread(&node, sizeof (node), addr) == sizeof (node)) {
1642 		(void) mdb_snprintf(od_name, sizeof (od_name), "%s",
1643 		    node.od_name);
1644 		if (print_full_path) {
1645 			if (mdb_vread(&vnode, sizeof (vnode_t),
1646 			    (uintptr_t)node.vp) == sizeof (vnode_t)) {
1647 				if (mdb_readstr(path_name, sizeof (path_name),
1648 				    (uintptr_t)vnode.v_path) != 0) {
1649 					(void) mdb_snprintf(od_name,
1650 					    sizeof (od_name), "N/A");
1651 				}
1652 			}
1653 		}
1654 		if (verbose) {
1655 			mdb_printf("VP: %p\n", node.vp);
1656 			mdb_printf("Name: %s\n", od_name);
1657 			if (print_full_path)
1658 				mdb_printf("V-node Path: %s\n", path_name);
1659 			mdb_printf("Ofiles: %u\n", node.n_ofile_list.ll_count);
1660 			mdb_printf("Range Locks: %u\n",
1661 			    node.n_lock_list.ll_count);
1662 			if (node.n_lock_list.ll_count != 0) {
1663 				(void) mdb_inc_indent(SMB_DCMD_INDENT);
1664 				list_addr = addr +
1665 				    OFFSETOF(smb_node_t, n_lock_list) +
1666 				    OFFSETOF(smb_llist_t, ll_list);
1667 				if (mdb_pwalk_dcmd("list", "smblock", 0,
1668 				    NULL, list_addr)) {
1669 					mdb_warn("failed to walk node's active"
1670 					    " locks");
1671 				}
1672 				(void) mdb_dec_indent(SMB_DCMD_INDENT);
1673 			}
1674 			if (node.n_oplock.ol_count == 0) {
1675 				mdb_printf("Opportunistic Locks: 0\n");
1676 			} else {
1677 				oplock_addr =
1678 				    addr + OFFSETOF(smb_node_t, n_oplock);
1679 				mdb_printf("Opportunistic Lock: %p\n",
1680 				    oplock_addr);
1681 				rc = mdb_call_dcmd("smboplock", oplock_addr,
1682 				    flags, argc, argv);
1683 				if (rc != DCMD_OK)
1684 					return (rc);
1685 			}
1686 			mdb_printf("Reference Count: %u\n\n", node.n_refcnt);
1687 		} else {
1688 			mdb_printf("%-?p %-?p %-18s %-6d %-6d %-8d %-6d ",
1689 			    addr, node.vp, od_name, node.n_ofile_list.ll_count,
1690 			    node.n_lock_list.ll_count,
1691 			    node.n_oplock.ol_count, node.n_refcnt);
1692 
1693 			if (print_full_path)
1694 				mdb_printf("\t%s\n", path_name);
1695 		}
1696 		if (stack_trace && node.n_audit_buf) {
1697 			int ctr;
1698 			smb_audit_buf_node_t *anb;
1699 
1700 			anb = mdb_alloc(sizeof (smb_audit_buf_node_t),
1701 			    UM_SLEEP | UM_GC);
1702 
1703 			if (mdb_vread(anb, sizeof (*anb),
1704 			    (uintptr_t)node.n_audit_buf) != sizeof (*anb)) {
1705 				mdb_warn("failed to read audit buffer");
1706 				return (DCMD_ERR);
1707 			}
1708 			ctr = anb->anb_max_index + 1;
1709 			anb->anb_index--;
1710 			anb->anb_index &= anb->anb_max_index;
1711 
1712 			while (ctr) {
1713 				smb_audit_record_node_t	*anr;
1714 
1715 				anr = anb->anb_records + anb->anb_index;
1716 
1717 				if (anr->anr_depth) {
1718 					char c[MDB_SYM_NAMLEN];
1719 					GElf_Sym sym;
1720 					int i;
1721 
1722 					mdb_printf("\nRefCnt: %u\t",
1723 					    anr->anr_refcnt);
1724 
1725 					for (i = 0;
1726 					    i < anr->anr_depth;
1727 					    i++) {
1728 						if (mdb_lookup_by_addr(
1729 						    anr->anr_stack[i],
1730 						    MDB_SYM_FUZZY,
1731 						    c, sizeof (c),
1732 						    &sym) == -1) {
1733 							continue;
1734 						}
1735 						mdb_printf("%s+0x%1x",
1736 						    c,
1737 						    anr->anr_stack[i] -
1738 						    (uintptr_t)sym.st_value);
1739 						++i;
1740 						break;
1741 					}
1742 
1743 					while (i < anr->anr_depth) {
1744 						if (mdb_lookup_by_addr(
1745 						    anr->anr_stack[i],
1746 						    MDB_SYM_FUZZY,
1747 						    c, sizeof (c),
1748 						    &sym) == -1) {
1749 							++i;
1750 							continue;
1751 						}
1752 						mdb_printf("\n\t\t%s+0x%1x",
1753 						    c,
1754 						    anr->anr_stack[i] -
1755 						    (uintptr_t)sym.st_value);
1756 						++i;
1757 					}
1758 					mdb_printf("\n");
1759 				}
1760 				anb->anb_index--;
1761 				anb->anb_index &= anb->anb_max_index;
1762 				ctr--;
1763 			}
1764 		}
1765 	} else {
1766 		mdb_warn("failed to read struct smb_node at %p", addr);
1767 		return (DCMD_ERR);
1768 	}
1769 
1770 	return (DCMD_OK);
1771 }
1772 
1773 /*
1774  * Initialize the smb_node_t walker by reading the value of smb_node_hash_table
1775  * in the kernel's symbol table. Only global walk supported.
1776  */
1777 static int
1778 smb_node_walk_init(mdb_walk_state_t *wsp)
1779 {
1780 	GElf_Sym	sym;
1781 	int		i;
1782 	uintptr_t	node_hash_table_addr;
1783 
1784 	if (wsp->walk_addr == NULL) {
1785 		if (mdb_lookup_by_obj(SMBSRV_OBJNAME, "smb_node_hash_table",
1786 		    &sym) == -1) {
1787 			mdb_warn("failed to find 'smb_node_hash_table'");
1788 			return (WALK_ERR);
1789 		}
1790 		node_hash_table_addr = (uintptr_t)sym.st_value;
1791 	} else {
1792 		mdb_printf("smb_node walk only supports global walks\n");
1793 		return (WALK_ERR);
1794 	}
1795 
1796 	for (i = 0; i < SMBND_HASH_MASK + 1; i++) {
1797 		wsp->walk_addr = node_hash_table_addr +
1798 		    (i * sizeof (smb_llist_t)) + OFFSETOF(smb_llist_t, ll_list);
1799 		if (mdb_layered_walk("list", wsp) == -1) {
1800 			mdb_warn("failed to walk 'list'");
1801 			return (WALK_ERR);
1802 		}
1803 	}
1804 
1805 	return (WALK_NEXT);
1806 }
1807 
1808 static int
1809 smb_node_walk_step(mdb_walk_state_t *wsp)
1810 {
1811 	return (wsp->walk_callback(wsp->walk_addr, wsp->walk_layer,
1812 	    wsp->walk_cbdata));
1813 }
1814 
1815 /*
1816  * *****************************************************************************
1817  * ****************************** smb_lock_t ***********************************
1818  * *****************************************************************************
1819  */
1820 
1821 static int
1822 smb_lock(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1823 {
1824 	smb_lock_t	lock;
1825 	int		verbose = FALSE;
1826 	uintptr_t	list_addr;
1827 	char		*lock_type;
1828 
1829 	if (mdb_getopts(argc, argv,
1830 	    'v', MDB_OPT_SETBITS, TRUE, &verbose,
1831 	    NULL) != argc)
1832 		return (DCMD_USAGE);
1833 
1834 	/*
1835 	 * An smb_lock_t address must be specified.
1836 	 */
1837 	if (!(flags & DCMD_ADDRSPEC))
1838 		return (DCMD_USAGE);
1839 
1840 	/*
1841 	 * If this is the first invocation of the command, print a nice
1842 	 * header line for the output that will follow.
1843 	 */
1844 	if (DCMD_HDRSPEC(flags)) {
1845 		if (verbose)
1846 			mdb_printf("SMB lock information:\n\n");
1847 		else
1848 			mdb_printf("%<u>%-?s %4s %16s %8s %9s%</u>\n",
1849 			    "Locks: ", "TYPE", "START", "LENGTH",
1850 			    "CONFLICTS");
1851 	}
1852 
1853 	if (mdb_vread(&lock, sizeof (lock), addr) == sizeof (lock)) {
1854 		switch (lock.l_type) {
1855 		case SMB_LOCK_TYPE_READWRITE:
1856 			lock_type = "RW";
1857 			break;
1858 		case SMB_LOCK_TYPE_READONLY:
1859 			lock_type = "RO";
1860 			break;
1861 		default:
1862 			lock_type = "N/A";
1863 			break;
1864 		}
1865 		if (verbose) {
1866 			mdb_printf("Type             :\t%s (%u)\n",
1867 			    lock_type, lock.l_type);
1868 			mdb_printf("Start            :\t%llx\n",
1869 			    lock.l_start);
1870 			mdb_printf("Length           :\t%lx\n",
1871 			    lock.l_length);
1872 			mdb_printf("Session          :\t%p\n",
1873 			    lock.l_session);
1874 			mdb_printf("File             :\t%p\n",
1875 			    lock.l_file);
1876 			mdb_printf("User ID          :\t%u\n",
1877 			    lock.l_uid);
1878 			mdb_printf("Process ID       :\t%u\n",
1879 			    lock.l_pid);
1880 			mdb_printf("Conflicts        :\t%u\n",
1881 			    lock.l_conflict_list.sl_count);
1882 			if (lock.l_conflict_list.sl_count != 0) {
1883 				(void) mdb_inc_indent(SMB_DCMD_INDENT);
1884 				list_addr = addr +
1885 				    OFFSETOF(smb_lock_t, l_conflict_list) +
1886 				    OFFSETOF(smb_slist_t, sl_list);
1887 				if (mdb_pwalk_dcmd("list", "smb_lock",
1888 				    0, NULL, list_addr)) {
1889 					mdb_warn("failed to walk conflict "
1890 					    "locks ");
1891 				}
1892 				(void) mdb_dec_indent(SMB_DCMD_INDENT);
1893 			}
1894 			mdb_printf("Blocked by       :\t%p\n",
1895 			    lock.l_blocked_by);
1896 			mdb_printf("Flags            :\t0x%x\n",
1897 			    lock.l_flags);
1898 			mdb_printf("\n");
1899 		} else {
1900 			mdb_printf("%?p %4s %16llx %08lx %9x", addr,
1901 			    lock_type, lock.l_start, lock.l_length,
1902 			    lock.l_conflict_list.sl_count);
1903 		}
1904 	} else {
1905 		mdb_warn("failed to read struct smb_request at %p", addr);
1906 		return (DCMD_ERR);
1907 	}
1908 
1909 	return (DCMD_OK);
1910 }
1911 
1912 /*
1913  * *****************************************************************************
1914  * ************************** smb_oplock_grant_t *******************************
1915  * *****************************************************************************
1916  */
1917 /*ARGSUSED*/
1918 static int
1919 smb_oplock_grant(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1920 {
1921 	smb_oplock_grant_t	grant;
1922 	char			 *level;
1923 
1924 	if (!(flags & DCMD_ADDRSPEC))
1925 		return (DCMD_USAGE);
1926 
1927 	/*
1928 	 * If this is the first invocation of the command, print a nice
1929 	 * header line for the output that will follow.
1930 	 */
1931 	if (DCMD_HDRSPEC(flags)) {
1932 		mdb_printf("%<u>%-16s %-10s %-16s%</u>\n",
1933 		    "Grants:", "LEVEL", "OFILE");
1934 	}
1935 
1936 	if (mdb_vread(&grant, sizeof (grant), addr) == sizeof (grant)) {
1937 		switch (grant.og_level) {
1938 		case SMB_OPLOCK_EXCLUSIVE:
1939 			level = "EXCLUSIVE";
1940 			break;
1941 		case SMB_OPLOCK_BATCH:
1942 			level = "BATCH";
1943 			break;
1944 		case SMB_OPLOCK_LEVEL_II:
1945 			level = "LEVEL_II";
1946 			break;
1947 		default:
1948 			level = "UNKNOWN";
1949 			break;
1950 		}
1951 
1952 		mdb_printf("%-16p %-10s %-16p", addr, level, grant.og_ofile);
1953 	}
1954 	return (DCMD_OK);
1955 }
1956 
1957 /*
1958  * *****************************************************************************
1959  * ***************************** smb_oplock_t **********************************
1960  * *****************************************************************************
1961  */
1962 /*ARGSUSED*/
1963 static int
1964 smb_oplock(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1965 {
1966 	smb_oplock_t	oplock;
1967 	uintptr_t	list_addr;
1968 
1969 	if (!(flags & DCMD_ADDRSPEC))
1970 		return (DCMD_USAGE);
1971 
1972 	if (mdb_vread(&oplock, sizeof (oplock), addr) != sizeof (oplock)) {
1973 		mdb_warn("failed to read struct smb_oplock at %p", addr);
1974 		return (DCMD_ERR);
1975 	}
1976 
1977 	if (oplock.ol_count == 0)
1978 		return (DCMD_OK);
1979 
1980 	(void) mdb_inc_indent(SMB_DCMD_INDENT);
1981 	switch (oplock.ol_break) {
1982 	case SMB_OPLOCK_BREAK_TO_NONE:
1983 		mdb_printf("Break Pending: BREAK_TO_NONE\n");
1984 		break;
1985 	case SMB_OPLOCK_BREAK_TO_LEVEL_II:
1986 		mdb_printf(
1987 		    "Break Pending: BREAK_TO_LEVEL_II\n");
1988 		break;
1989 	default:
1990 		break;
1991 	}
1992 
1993 	list_addr = addr + OFFSETOF(smb_oplock_t, ol_grants);
1994 
1995 	if (mdb_pwalk_dcmd("list", "smboplockgrant",
1996 	    argc, argv, list_addr)) {
1997 		mdb_warn("failed to walk oplock grants");
1998 	}
1999 
2000 	(void) mdb_dec_indent(SMB_DCMD_INDENT);
2001 
2002 	return (DCMD_OK);
2003 }
2004 
2005 /*
2006  * ::smbstat
2007  *
2008  * Prints SMB requests statistics.
2009  */
2010 /*ARGSUSED*/
2011 static int
2012 smb_stats(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2013 {
2014 	smb_server_t	*sv;
2015 
2016 	if (!(flags & DCMD_ADDRSPEC))
2017 		return (DCMD_USAGE);
2018 
2019 	sv = mdb_alloc(sizeof (*sv), UM_SLEEP | UM_GC);
2020 	if (mdb_vread(sv, sizeof (*sv), addr) == -1) {
2021 		mdb_warn("failed to read server object at %p", addr);
2022 		return (DCMD_ERR);
2023 	}
2024 	if (sv->sv_magic != SMB_SERVER_MAGIC) {
2025 		mdb_warn("not an smb_server_t (%p)>", addr);
2026 		return (DCMD_ERR);
2027 	}
2028 	mdb_printf(
2029 	    "\n%<b>  nbt   tcp users trees files pipes%</b>\n"
2030 	    "%5d %5d %5d %5d %5d %5d\n",
2031 	    sv->sv_nbt_sess,
2032 	    sv->sv_tcp_sess,
2033 	    sv->sv_users,
2034 	    sv->sv_trees,
2035 	    sv->sv_files,
2036 	    sv->sv_pipes);
2037 
2038 	return (DCMD_OK);
2039 }
2040 
2041 /*
2042  * *****************************************************************************
2043  * ******************************** smb_ace_t **********************************
2044  * *****************************************************************************
2045  */
2046 static const ace_type_entry_t	ace_types[ACE_TYPE_TABLEN] =
2047 {
2048 	ACE_TYPE_ENTRY(ACE_ACCESS_ALLOWED_ACE_TYPE),
2049 	ACE_TYPE_ENTRY(ACE_ACCESS_DENIED_ACE_TYPE),
2050 	ACE_TYPE_ENTRY(ACE_SYSTEM_AUDIT_ACE_TYPE),
2051 	ACE_TYPE_ENTRY(ACE_SYSTEM_ALARM_ACE_TYPE),
2052 	ACE_TYPE_ENTRY(ACE_ACCESS_ALLOWED_COMPOUND_ACE_TYPE),
2053 	ACE_TYPE_ENTRY(ACE_ACCESS_ALLOWED_OBJECT_ACE_TYPE),
2054 	ACE_TYPE_ENTRY(ACE_ACCESS_DENIED_OBJECT_ACE_TYPE),
2055 	ACE_TYPE_ENTRY(ACE_SYSTEM_AUDIT_OBJECT_ACE_TYPE),
2056 	ACE_TYPE_ENTRY(ACE_SYSTEM_ALARM_OBJECT_ACE_TYPE),
2057 	ACE_TYPE_ENTRY(ACE_ACCESS_ALLOWED_CALLBACK_ACE_TYPE),
2058 	ACE_TYPE_ENTRY(ACE_ACCESS_DENIED_CALLBACK_ACE_TYPE),
2059 	ACE_TYPE_ENTRY(ACE_ACCESS_ALLOWED_CALLBACK_OBJECT_ACE_TYPE),
2060 	ACE_TYPE_ENTRY(ACE_ACCESS_DENIED_CALLBACK_OBJECT_ACE_TYPE),
2061 	ACE_TYPE_ENTRY(ACE_SYSTEM_AUDIT_CALLBACK_ACE_TYPE),
2062 	ACE_TYPE_ENTRY(ACE_SYSTEM_ALARM_CALLBACK_ACE_TYPE),
2063 	ACE_TYPE_ENTRY(ACE_SYSTEM_AUDIT_CALLBACK_OBJECT_ACE_TYPE),
2064 	ACE_TYPE_ENTRY(ACE_SYSTEM_ALARM_CALLBACK_OBJECT_ACE_TYPE),
2065 	ACE_TYPE_ENTRY(0x11),
2066 	ACE_TYPE_ENTRY(0x12),
2067 	ACE_TYPE_ENTRY(0x13),
2068 	ACE_TYPE_ENTRY(0x14),
2069 	ACE_TYPE_ENTRY(0x15),
2070 	ACE_TYPE_ENTRY(0x16),
2071 	ACE_TYPE_ENTRY(0x17),
2072 	ACE_TYPE_ENTRY(0x18),
2073 	ACE_TYPE_ENTRY(0x19),
2074 	ACE_TYPE_ENTRY(0x1A),
2075 	ACE_TYPE_ENTRY(0x1B),
2076 	ACE_TYPE_ENTRY(0x1C),
2077 	ACE_TYPE_ENTRY(0x1D),
2078 	ACE_TYPE_ENTRY(0x1E),
2079 	ACE_TYPE_ENTRY(0x1F)
2080 };
2081 
2082 static const mdb_bitmask_t ace_flag_bits[] = {
2083 	{ "OBJECT_INHERIT_ACE", OBJECT_INHERIT_ACE, OBJECT_INHERIT_ACE },
2084 	{ "CONTAINER_INHERIT_ACE", CONTAINER_INHERIT_ACE,
2085 	    CONTAINER_INHERIT_ACE },
2086 	{ "NO_PROPOGATE_INHERIT_ACE", NO_PROPOGATE_INHERIT_ACE,
2087 	    NO_PROPOGATE_INHERIT_ACE },
2088 	{ "INHERIT_ONLY_ACE", INHERIT_ONLY_ACE, INHERIT_ONLY_ACE },
2089 	{ "INHERITED_ACE", INHERITED_ACE, INHERITED_ACE },
2090 	{ "SUCCESSFUL_ACCESS_ACE_FLAG", SUCCESSFUL_ACCESS_ACE_FLAG,
2091 	    SUCCESSFUL_ACCESS_ACE_FLAG },
2092 	{ "FAILED_ACCESS_ACE_FLAG", FAILED_ACCESS_ACE_FLAG,
2093 	    FAILED_ACCESS_ACE_FLAG },
2094 	{ NULL, 0, 0 }
2095 };
2096 
2097 /*
2098  * ::smbace
2099  */
2100 static int
2101 smb_ace(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2102 {
2103 	smb_ace_t	ace;
2104 	int		verbose = FALSE;
2105 	const char	*ptr;
2106 	int		rc;
2107 
2108 	if (mdb_getopts(argc, argv, 'v', MDB_OPT_SETBITS, TRUE, &verbose,
2109 	    NULL) != argc)
2110 		return (DCMD_USAGE);
2111 
2112 	/*
2113 	 * An smb_ace address is required.
2114 	 */
2115 	if (!(flags & DCMD_ADDRSPEC))
2116 		return (DCMD_USAGE);
2117 
2118 	if (mdb_vread(&ace, sizeof (ace), addr) != sizeof (ace)) {
2119 		mdb_warn("failed to read struct smb_ace at %p", addr);
2120 		return (DCMD_ERR);
2121 	}
2122 
2123 	if (verbose) {
2124 		if (ace.se_hdr.se_type < ACE_TYPE_TABLEN)
2125 			ptr = ace_types[ace.se_hdr.se_type].ace_type_sting;
2126 		else
2127 			ptr = "Unknown";
2128 
2129 		mdb_printf("ACE Type: 0x%02x (%s)\n", ace.se_hdr.se_type, ptr);
2130 		mdb_printf("ACE Flags: %b\n", (int)ace.se_hdr.se_flags,
2131 		    ace_flag_bits);
2132 		mdb_printf("ACE Wire Size: 0x%04x\n", ace.se_hdr.se_bsize);
2133 		mdb_printf("ACE Mask: 0x%08x\n", ace.se_mask);
2134 		mdb_printf("ACE SID: ");
2135 	} else {
2136 		if (DCMD_HDRSPEC(flags))
2137 			mdb_printf(
2138 			    "%<b>%<u>%?-s %-4s %-4s %-8s %s%</u>%</b>\n",
2139 			    "ACE", "TYPE", "FLAGS", "MASK", "SID");
2140 		mdb_printf("%?p 0x%02x 0x%02x 0x%08x ", addr,
2141 		    ace.se_hdr.se_type, ace.se_hdr.se_flags, ace.se_mask);
2142 	}
2143 	rc = smb_sid_print((uintptr_t)ace.se_sid);
2144 	mdb_printf("\n");
2145 	return (rc);
2146 }
2147 
2148 static int
2149 smb_ace_walk_init(mdb_walk_state_t *wsp)
2150 {
2151 	if (wsp->walk_addr == 0) {
2152 		mdb_printf("smb_ace walk only supports local walks\n");
2153 		return (WALK_ERR);
2154 	}
2155 
2156 	wsp->walk_addr += OFFSETOF(smb_acl_t, sl_sorted);
2157 
2158 	if (mdb_layered_walk("list", wsp) == -1) {
2159 		mdb_warn("failed to walk list of ACEs");
2160 		return (WALK_ERR);
2161 	}
2162 
2163 	return (WALK_NEXT);
2164 }
2165 
2166 static int
2167 smb_ace_walk_step(mdb_walk_state_t *wsp)
2168 {
2169 	return (wsp->walk_callback(wsp->walk_addr, wsp->walk_layer,
2170 	    wsp->walk_cbdata));
2171 }
2172 
2173 /*
2174  * *****************************************************************************
2175  * ******************************** smb_acl_t **********************************
2176  * *****************************************************************************
2177  */
2178 
2179 /*
2180  * ::smbacl
2181  */
2182 static int
2183 smb_acl(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2184 {
2185 	smb_acl_t	acl;
2186 
2187 	/* An smb_acl address is required. */
2188 	if (!(flags & DCMD_ADDRSPEC))
2189 		return (DCMD_USAGE);
2190 
2191 	if (mdb_vread(&acl, sizeof (acl), addr) != sizeof (acl)) {
2192 		mdb_warn("failed to read struct smb_acl at %p", addr);
2193 		return (DCMD_ERR);
2194 	}
2195 
2196 	mdb_printf("ACL Revision: %d\n", acl.sl_revision);
2197 	mdb_printf("ACL Size on Wire: %d\n", acl.sl_bsize);
2198 	mdb_printf("ACL Number of ACEs: %d\n", acl.sl_acecnt);
2199 
2200 	(void) mdb_inc_indent(SMB_DCMD_INDENT);
2201 	if (mdb_pwalk_dcmd("smbace_walker", "smbace", argc, argv, addr)) {
2202 		(void) mdb_dec_indent(SMB_DCMD_INDENT);
2203 		mdb_warn("failed to walk list of ACEs for ACL %p", addr);
2204 		return (DCMD_ERR);
2205 	}
2206 	(void) mdb_dec_indent(SMB_DCMD_INDENT);
2207 	return (DCMD_OK);
2208 }
2209 
2210 /*
2211  * *****************************************************************************
2212  * ********************************* smb_sd_t **********************************
2213  * *****************************************************************************
2214  */
2215 
2216 /*
2217  * ::smbsd
2218  */
2219 static int
2220 smb_sd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2221 {
2222 	smb_sd_t	sd;
2223 	int		rc;
2224 
2225 	/*
2226 	 * An smb_sid address is required.
2227 	 */
2228 	if (!(flags & DCMD_ADDRSPEC))
2229 		return (DCMD_USAGE);
2230 
2231 	if (mdb_vread(&sd, sizeof (sd), addr) != sizeof (sd)) {
2232 		mdb_warn("failed to read struct smb_sd at %p", addr);
2233 		return (DCMD_ERR);
2234 	}
2235 
2236 	mdb_printf("SD Revision: %d\n", sd.sd_revision);
2237 	mdb_printf("SD Control: %04x\n", sd.sd_control);
2238 	if (sd.sd_control & SE_OWNER_DEFAULTED)
2239 		mdb_printf("\t    SE_OWNER_DEFAULTED\n");
2240 	if (sd.sd_control & SE_GROUP_DEFAULTED)
2241 		mdb_printf("\t    SE_GROUP_DEFAULTED\n");
2242 	if (sd.sd_control & SE_DACL_PRESENT)
2243 		mdb_printf("\t    SE_DACL_PRESENT\n");
2244 	if (sd.sd_control & SE_DACL_DEFAULTED)
2245 		mdb_printf("\t    SE_DACL_DEFAULTED\n");
2246 	if (sd.sd_control & SE_SACL_PRESENT)
2247 		mdb_printf("\t    SE_SACL_PRESENT\n");
2248 	if (sd.sd_control & SE_SACL_DEFAULTED)
2249 		mdb_printf("\t    SE_SACL_DEFAULTED\n");
2250 	if (sd.sd_control & SE_DACL_AUTO_INHERIT_REQ)
2251 		mdb_printf("\t    SE_DACL_AUTO_INHERIT_REQ\n");
2252 	if (sd.sd_control & SE_SACL_AUTO_INHERIT_REQ)
2253 		mdb_printf("\t    SE_SACL_AUTO_INHERIT_REQ\n");
2254 	if (sd.sd_control & SE_DACL_AUTO_INHERITED)
2255 		mdb_printf("\t    SE_DACL_AUTO_INHERITED\n");
2256 	if (sd.sd_control & SE_SACL_AUTO_INHERITED)
2257 		mdb_printf("\t    SE_SACL_AUTO_INHERITED\n");
2258 	if (sd.sd_control & SE_DACL_PROTECTED)
2259 		mdb_printf("\t    SE_DACL_PROTECTED\n");
2260 	if (sd.sd_control & SE_SACL_PROTECTED)
2261 		mdb_printf("\t    SE_SACL_PROTECTED\n");
2262 	if (sd.sd_control & SE_SELF_RELATIVE)
2263 		mdb_printf("\t    SE_SELF_RELATIVE\n");
2264 
2265 	mdb_printf("SID of Owner: ");
2266 	rc = smb_sid_print((uintptr_t)sd.sd_owner);
2267 	if (rc != DCMD_OK)
2268 		return (rc);
2269 	mdb_printf("\nSID of Group: ");
2270 	rc = smb_sid_print((uintptr_t)sd.sd_group);
2271 	if (rc != DCMD_OK)
2272 		return (rc);
2273 	mdb_printf("\n");
2274 
2275 	if (sd.sd_control & SE_SACL_PRESENT && sd.sd_sacl) {
2276 		mdb_printf("%<b>%<u>System ACL%</u>%</b>\n");
2277 		(void) mdb_inc_indent(SMB_DCMD_INDENT);
2278 		rc = mdb_call_dcmd("smbacl", (uintptr_t)sd.sd_sacl, flags,
2279 		    argc, argv);
2280 		(void) mdb_dec_indent(SMB_DCMD_INDENT);
2281 		if (rc != DCMD_OK)
2282 			return (rc);
2283 	}
2284 	if (sd.sd_control & SE_DACL_PRESENT && sd.sd_dacl) {
2285 		mdb_printf("%<b>%<u>Discretionary ACL%</u>%</b>\n");
2286 		(void) mdb_inc_indent(SMB_DCMD_INDENT);
2287 		rc = mdb_call_dcmd("smbacl", (uintptr_t)sd.sd_dacl, flags,
2288 		    argc, argv);
2289 		(void) mdb_dec_indent(SMB_DCMD_INDENT);
2290 		if (rc != DCMD_OK)
2291 			return (rc);
2292 	}
2293 
2294 	return (DCMD_OK);
2295 }
2296 
2297 /*
2298  * *****************************************************************************
2299  * ********************************* smb_sid_t *********************************
2300  * *****************************************************************************
2301  */
2302 
2303 /*
2304  * ::smbsid
2305  */
2306 /*ARGSUSED*/
2307 static int
2308 smb_sid(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2309 {
2310 	/*
2311 	 * An smb_sid address is required.
2312 	 */
2313 	if (!(flags & DCMD_ADDRSPEC))
2314 		return (DCMD_USAGE);
2315 
2316 	return (smb_sid_print(addr));
2317 }
2318 
2319 /*
2320  * smb_sid_print
2321  */
2322 static int
2323 smb_sid_print(uintptr_t addr)
2324 {
2325 	smb_sid_t	sid;
2326 	smb_sid_t	*psid;
2327 	size_t		sid_size;
2328 	int		i;
2329 	uint64_t	authority;
2330 
2331 	sid_size = OFFSETOF(smb_sid_t, sid_subauth);
2332 
2333 	if (mdb_vread(&sid, sid_size, addr) != sid_size) {
2334 		mdb_warn("failed to read struct smb_sid at %p", addr);
2335 		return (DCMD_ERR);
2336 	}
2337 
2338 	sid_size += sid.sid_subauthcnt * sizeof (sid.sid_subauth[0]);
2339 
2340 	psid = mdb_zalloc(sid_size, UM_SLEEP | UM_GC);
2341 	if (mdb_vread(psid, sid_size, addr) != sid_size) {
2342 		mdb_warn("failed to read struct smb_sid at %p", addr);
2343 		return (DCMD_ERR);
2344 	}
2345 
2346 	mdb_printf("S-%d", psid->sid_revision);
2347 	authority = 0;
2348 	for (i = 0; i < NT_SID_AUTH_MAX; i++) {
2349 		authority += ((uint64_t)psid->sid_authority[i]) <<
2350 		    (8 * (NT_SID_AUTH_MAX - 1) - i);
2351 	}
2352 	mdb_printf("-%ll", authority);
2353 
2354 	for (i = 0; i < psid->sid_subauthcnt; i++)
2355 		mdb_printf("-%d", psid->sid_subauth[i]);
2356 
2357 	return (DCMD_OK);
2358 }
2359 
2360 /*
2361  * *****************************************************************************
2362  * ********************************* smb_fssd_t ********************************
2363  * *****************************************************************************
2364  */
2365 
2366 /*
2367  * ::smbfssd
2368  */
2369 static int
2370 smb_fssd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2371 {
2372 	smb_fssd_t	fssd;
2373 	int		rc;
2374 
2375 	/*
2376 	 * An smb_fssd address is required.
2377 	 */
2378 	if (!(flags & DCMD_ADDRSPEC))
2379 		return (DCMD_USAGE);
2380 
2381 	if (mdb_vread(&fssd, sizeof (fssd), addr) != sizeof (fssd)) {
2382 		mdb_warn("failed to read struct smb_fssd at %p", addr);
2383 		return (DCMD_ERR);
2384 	}
2385 
2386 	mdb_printf("FSSD secinfo: 0x%x\n", fssd.sd_secinfo);
2387 	if (fssd.sd_secinfo & SMB_OWNER_SECINFO)
2388 		mdb_printf("FSSD uid: %d\n", fssd.sd_uid);
2389 	if (fssd.sd_secinfo & SMB_GROUP_SECINFO)
2390 		mdb_printf("FSSD gid: %d\n", fssd.sd_gid);
2391 	if (fssd.sd_secinfo & SMB_SACL_SECINFO && fssd.sd_zsacl) {
2392 		mdb_printf("%<b>%<u>System ACL%</u>%</b>\n");
2393 		(void) mdb_inc_indent(SMB_DCMD_INDENT);
2394 		rc = mdb_call_dcmd("smbacl", (uintptr_t)fssd.sd_zsacl, flags,
2395 		    argc, argv);
2396 		(void) mdb_dec_indent(SMB_DCMD_INDENT);
2397 		if (rc != DCMD_OK)
2398 			return (rc);
2399 	}
2400 	if (fssd.sd_secinfo & SMB_DACL_SECINFO && fssd.sd_zdacl) {
2401 		mdb_printf("%<b>%<u>Discretionary ACL%</u>%</b>\n");
2402 		(void) mdb_inc_indent(SMB_DCMD_INDENT);
2403 		rc = mdb_call_dcmd("smbacl", (uintptr_t)fssd.sd_zdacl, flags,
2404 		    argc, argv);
2405 		(void) mdb_dec_indent(SMB_DCMD_INDENT);
2406 		if (rc != DCMD_OK)
2407 			return (rc);
2408 	}
2409 
2410 	return (DCMD_OK);
2411 }
2412 
2413 /*
2414  * *****************************************************************************
2415  * **************************** Utility Funcions *******************************
2416  * *****************************************************************************
2417  */
2418 
2419 /*
2420  * smb_dcmd_getopt
2421  *
2422  * This function analyzes the arguments passed in and sets the bit corresponding
2423  * to the options found in the opts variable.
2424  *
2425  * Return Value
2426  *
2427  *	-1	An error occured during the decoding
2428  *	0	The decoding was successful
2429  */
2430 static int
2431 smb_dcmd_getopt(uint_t *opts, int argc, const mdb_arg_t *argv)
2432 {
2433 	*opts = 0;
2434 
2435 	if (mdb_getopts(argc, argv,
2436 	    's', MDB_OPT_SETBITS, SMB_OPT_SERVER, opts,
2437 	    'e', MDB_OPT_SETBITS, SMB_OPT_SESSION, opts,
2438 	    'r', MDB_OPT_SETBITS, SMB_OPT_REQUEST, opts,
2439 	    'u', MDB_OPT_SETBITS, SMB_OPT_USER, opts,
2440 	    't', MDB_OPT_SETBITS, SMB_OPT_TREE, opts,
2441 	    'f', MDB_OPT_SETBITS, SMB_OPT_OFILE, opts,
2442 	    'd', MDB_OPT_SETBITS, SMB_OPT_ODIR, opts,
2443 	    'w', MDB_OPT_SETBITS, SMB_OPT_WALK, opts,
2444 	    'v', MDB_OPT_SETBITS, SMB_OPT_VERBOSE, opts,
2445 	    NULL) != argc)
2446 		return (-1);
2447 
2448 	return (0);
2449 }
2450 
2451 /*
2452  * smb_dcmd_setopt
2453  *
2454  * This function set the arguments corresponding to the bits set in opts.
2455  *
2456  * Return Value
2457  *
2458  *	Number of arguments set.
2459  */
2460 static int
2461 smb_dcmd_setopt(uint_t opts, int max_argc, mdb_arg_t *argv)
2462 {
2463 	int	i;
2464 	int	argc = 0;
2465 
2466 	for (i = 0; i < SMB_MDB_MAX_OPTS; i++) {
2467 		if ((opts & smb_opts[i].o_value) && (argc < max_argc)) {
2468 			argv->a_type = MDB_TYPE_STRING;
2469 			argv->a_un.a_str = smb_opts[i].o_name;
2470 			argc++;
2471 			argv++;
2472 		}
2473 	}
2474 	return (argc);
2475 }
2476 
2477 /*
2478  * smb_obj_expand
2479  */
2480 static int
2481 smb_obj_expand(uintptr_t addr, uint_t opts, const smb_exp_t *x, ulong_t indent)
2482 {
2483 	int		rc = 0;
2484 	int		argc;
2485 	mdb_arg_t	argv[SMB_MDB_MAX_OPTS];
2486 
2487 	argc = smb_dcmd_setopt(opts | SMB_OPT_WALK, SMB_MDB_MAX_OPTS, argv);
2488 
2489 	(void) mdb_inc_indent(indent);
2490 	while (x->ex_dcmd) {
2491 		if (x->ex_mask & opts) {
2492 			rc = mdb_pwalk_dcmd("list", x->ex_dcmd, argc, argv,
2493 			    addr + x->ex_offset);
2494 
2495 			if (rc) {
2496 				mdb_warn("failed to walk the list of %s in %p",
2497 				    x->ex_name, addr + x->ex_offset);
2498 				break;
2499 			}
2500 		}
2501 		x++;
2502 	}
2503 	(void) mdb_dec_indent(indent);
2504 	return (rc);
2505 }
2506 
2507 /*
2508  * smb_obj_list
2509  *
2510  * Function called by the DCMDs when no address is provided. It expands the
2511  * tree under the object type associated with the calling DCMD (based on the
2512  * flags passed in).
2513  *
2514  * Return Value
2515  *
2516  *	DCMD_OK
2517  *	DCMD_ERR
2518  */
2519 static int
2520 smb_obj_list(const char *name, uint_t opts, uint_t flags)
2521 {
2522 	int		argc;
2523 	mdb_arg_t	argv[SMB_MDB_MAX_OPTS];
2524 
2525 	argc = smb_dcmd_setopt(opts, SMB_MDB_MAX_OPTS, argv);
2526 
2527 	if (mdb_call_dcmd("smblist", 0, flags, argc, argv)) {
2528 		mdb_warn("failed to list %s", name);
2529 		return (DCMD_ERR);
2530 	}
2531 	return (DCMD_OK);
2532 }
2533 
2534 static int
2535 smb_worker_findstack(uintptr_t addr)
2536 {
2537 	char		cmd[80];
2538 	mdb_arg_t	cmdarg;
2539 
2540 	mdb_inc_indent(2);
2541 	mdb_snprintf(cmd, sizeof (cmd), "<.$c%d", 16);
2542 	cmdarg.a_type = MDB_TYPE_STRING;
2543 	cmdarg.a_un.a_str = cmd;
2544 	(void) mdb_call_dcmd("findstack", addr, DCMD_ADDRSPEC, 1, &cmdarg);
2545 	mdb_dec_indent(2);
2546 	mdb_printf("\n");
2547 	return (DCMD_OK);
2548 }
2549