8322 nl: misleading-indentation
[unleashed/tickless.git] / usr / src / cmd / mdb / common / modules / smbsrv / smbsrv.c
bloba3e86379bec929dd54e47ec318f6d3583228efa1
1 /*
2 * CDDL HEADER START
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.
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.
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]
19 * CDDL HEADER END
23 * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
24 * Copyright 2015 Nexenta Systems, Inc. All rights reserved.
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>
35 #ifdef _KERNEL
36 #define SMBSRV_OBJNAME "smbsrv"
37 #else
38 #define SMBSRV_OBJNAME "libfksmbsrv.so.1"
39 #endif
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_}
46 #define SMB_MDB_MAX_OPTS 9
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
60 * Structure associating an ACE type to a string.
62 typedef struct {
63 uint8_t ace_type_value;
64 const char *ace_type_sting;
65 } ace_type_entry_t;
68 * Structure containing strings describing an SMB command.
70 typedef struct {
71 const char *smb_com;
72 const char *smb_andx;
73 } smb_com_entry_t;
76 * Structure describing an object to be expanded (displayed).
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;
86 * List of supported options. Ther order has the match the bits SMB_OPT_xxx.
88 typedef struct smb_mdb_opts {
89 char *o_name;
90 uint32_t o_value;
91 } smb_mdb_opts_t;
93 static smb_mdb_opts_t smb_opts[SMB_MDB_MAX_OPTS] =
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 }
106 static smb_com_entry_t smb_com[256] =
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, "?")
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"
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 *);
428 * MDB module linkage information:
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.
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 }
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 }
528 static const mdb_modinfo_t modinfo = {
529 MDB_API_VERSION, dcmds, walkers
532 const mdb_modinfo_t *
533 _mdb_init(void)
535 return (&modinfo);
539 * *****************************************************************************
540 * ****************************** Top level DCMD *******************************
541 * *****************************************************************************
544 static void
545 smb_dcmd_list_help(void)
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");
565 * ::smblist
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.
571 /*ARGSUSED*/
572 static int
573 smb_dcmd_list(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
575 GElf_Sym sym;
576 uint_t opts = 0;
577 int new_argc;
578 mdb_arg_t new_argv[SMB_MDB_MAX_OPTS];
580 if (smb_dcmd_getopt(&opts, argc, argv))
581 return (DCMD_USAGE);
583 if (!(opts & ~(SMB_OPT_WALK | SMB_OPT_VERBOSE)))
584 opts |= SMB_OPT_ALL_OBJ;
586 opts |= SMB_OPT_WALK;
588 new_argc = smb_dcmd_setopt(opts, SMB_MDB_MAX_OPTS, new_argv);
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);
595 addr = (uintptr_t)sym.st_value + OFFSETOF(smb_llist_t, ll_list);
597 if (mdb_pwalk_dcmd("list", "smbsrv", new_argc, new_argv, addr)) {
598 mdb_warn("cannot walk smb_server list");
599 return (DCMD_ERR);
601 return (DCMD_OK);
605 * *****************************************************************************
606 * ***************************** smb_server_t **********************************
607 * *****************************************************************************
610 static const char *smb_server_state[SMB_SERVER_STATE_SENTINEL] =
612 "CREATED",
613 "CONFIGURED",
614 "RUNNING",
615 "STOPPING",
616 "DELETING"
620 * List of objects that can be expanded under a server structure.
622 static const smb_exp_t smb_server_exp[] =
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 }
634 * ::smbsrv
636 * smbsrv dcmd - Print out smb_server structures.
638 /*ARGSUSED*/
639 static int
640 smb_dcmd_server(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
642 uint_t opts;
643 ulong_t indent = 0;
645 if (smb_dcmd_getopt(&opts, argc, argv))
646 return (DCMD_USAGE);
648 if (!(flags & DCMD_ADDRSPEC))
649 return (smb_obj_list("smb_server", opts | SMB_OPT_SERVER,
650 flags));
652 if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_SERVER)) ||
653 !(opts & SMB_OPT_WALK)) {
654 smb_server_t *sv;
655 const char *state;
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);
663 indent = SMB_DCMD_INDENT;
665 if (opts & SMB_OPT_VERBOSE) {
666 mdb_arg_t argv;
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");
681 if (sv->sv_state >= SMB_SERVER_STATE_SENTINEL)
682 state = "UNKNOWN";
683 else
684 state = smb_server_state[sv->sv_state];
686 mdb_printf("%-?p %-4d %-32s \n",
687 addr, sv->sv_zid, state);
690 if (smb_obj_expand(addr, opts, smb_server_exp, indent))
691 return (DCMD_ERR);
692 return (DCMD_OK);
696 * *****************************************************************************
697 * ***************************** smb_session_t *********************************
698 * *****************************************************************************
701 static const char *smb_session_state[SMB_SESSION_STATE_SENTINEL] =
703 "INITIALIZED",
704 "DISCONNECTED",
705 "CONNECTED",
706 "ESTABLISHED",
707 "NEGOTIATED",
708 "TERMINATED"
712 * List of objects that can be expanded under a session structure.
714 static const smb_exp_t smb_session_exp[] =
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}
728 static void
729 smb_dcmd_session_help(void)
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");
744 * ::smbsess
746 * smbsess dcmd - Print out the smb_session structure.
748 static int
749 smb_dcmd_session(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
751 uint_t opts;
752 ulong_t indent = 0;
754 if (smb_dcmd_getopt(&opts, argc, argv))
755 return (DCMD_USAGE);
757 if (!(flags & DCMD_ADDRSPEC)) {
758 opts |= SMB_OPT_SESSION;
759 opts &= ~SMB_OPT_SERVER;
760 return (smb_obj_list("smb_session", opts, flags));
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;
771 indent = SMB_DCMD_INDENT;
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);
778 if (se->s_state >= SMB_SESSION_STATE_SENTINEL)
779 state = "INVALID";
780 else
781 state = smb_session_state[se->s_state];
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");
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");
833 mdb_printf("%-?p %-*s %-8d %-8#x %s\n",
834 addr, ipaddrstrlen, cipaddr,
835 se->s_remote_port, se->dialect, state);
838 if (smb_obj_expand(addr, opts, smb_session_exp, indent))
839 return (DCMD_ERR);
840 return (DCMD_OK);
844 * *****************************************************************************
845 * **************************** smb_request_t **********************************
846 * *****************************************************************************
849 static const char *smb_request_state[SMB_REQ_STATE_SENTINEL] =
851 "FREE",
852 "INITIALIZING",
853 "SUBMITTED",
854 "ACTIVE",
855 "WAITING_EVENT",
856 "EVENT_OCCURRED",
857 "WAITING_LOCK",
858 "COMPLETED",
859 "CANCELED",
860 "CLEANED_UP"
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"
868 static int
869 smb_dcmd_request(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
871 uint_t opts;
873 if (smb_dcmd_getopt(&opts, argc, argv))
874 return (DCMD_USAGE);
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));
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;
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);
896 if (sr->sr_magic != SMB_REQ_MAGIC) {
897 mdb_warn("not an smb_request_t (%p)>", addr);
898 return (DCMD_ERR);
900 waiting = 0;
901 running = 0;
903 * Note: mdb_gethrtime() is only available in kmdb
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;
917 waiting /= NANOSEC;
918 running /= NANOSEC;
919 #endif /* _KERNEL */
921 if (sr->sr_state >= SMB_REQ_STATE_SENTINEL)
922 state = "INVALID";
923 else
924 state = smb_request_state[sr->sr_state];
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;
938 if (opts & SMB_OPT_VERBOSE) {
939 mdb_printf(
940 "%</b>%</u>SMB request information (%p):"
941 "%</u>%</b>\n\n", addr);
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);
951 mdb_printf(
952 "current SMB COM: %u (%s)\n",
953 cur_cmd_code, cur_cmd_name);
955 mdb_printf(
956 "state: %u (%s)\n",
957 sr->sr_state, state);
959 mdb_printf(
960 "TID(tree): %u (%p)\n",
961 sr->smb_tid, sr->tid_tree);
963 mdb_printf(
964 "UID(user): %u (%p)\n",
965 sr->smb_uid, sr->uid_user);
967 mdb_printf(
968 "FID(file): %u (%p)\n",
969 sr->smb_fid, sr->fid_ofile);
971 mdb_printf(
972 "PID: %u\n",
973 sr->smb_pid);
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);
985 mdb_printf(
986 "waiting time: %lld\n",
987 waiting);
989 mdb_printf(
990 "running time: %lld\n",
991 running);
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");
1008 mdb_printf(
1009 SMB_REQUEST_FORMAT,
1010 addr,
1011 sr->sr_worker,
1012 waiting,
1013 running,
1014 state,
1015 cur_cmd_name);
1018 return (DCMD_OK);
1022 * *****************************************************************************
1023 * ****************************** smb_user_t ***********************************
1024 * *****************************************************************************
1027 static const char *smb_user_state[SMB_USER_STATE_SENTINEL] =
1029 "LOGGING_ON",
1030 "LOGGED_ON",
1031 "LOGGING_OFF",
1032 "LOGGED_OFF"
1035 static void
1036 smb_dcmd_user_help(void)
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");
1047 static int
1048 smb_dcmd_user(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1050 uint_t opts;
1052 if (smb_dcmd_getopt(&opts, argc, argv))
1053 return (DCMD_USAGE);
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));
1061 if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_USER)) ||
1062 !(opts & SMB_OPT_WALK)) {
1063 smb_user_t *user;
1064 char *account;
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);
1071 account = mdb_zalloc(user->u_domain_len + user->u_name_len + 2,
1072 UM_SLEEP | UM_GC);
1074 if (user->u_domain_len)
1075 (void) mdb_vread(account, user->u_domain_len,
1076 (uintptr_t)user->u_domain);
1078 strcat(account, "\\");
1080 if (user->u_name_len)
1081 (void) mdb_vread(account + strlen(account),
1082 user->u_name_len, (uintptr_t)user->u_name);
1084 if (opts & SMB_OPT_VERBOSE) {
1085 const char *state;
1087 if (user->u_state >= SMB_USER_STATE_SENTINEL)
1088 state = "INVALID";
1089 else
1090 state = smb_user_state[user->u_state];
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");
1109 mdb_printf("%-?p %-5u %-32s\n", addr, user->u_uid,
1110 account);
1113 return (DCMD_OK);
1117 * *****************************************************************************
1118 * ****************************** smb_tree_t ***********************************
1119 * *****************************************************************************
1122 static const char *smb_tree_state[SMB_TREE_STATE_SENTINEL] =
1124 "CONNECTED",
1125 "DISCONNECTING",
1126 "DISCONNECTED"
1130 * List of objects that can be expanded under a tree structure.
1132 static const smb_exp_t smb_tree_exp[] =
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}
1143 static void
1144 smb_dcmd_tree_help(void)
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");
1157 static int
1158 smb_dcmd_tree(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1160 uint_t opts;
1161 ulong_t indent = 0;
1163 if (smb_dcmd_getopt(&opts, argc, argv))
1164 return (DCMD_USAGE);
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));
1173 if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_TREE)) ||
1174 !(opts & SMB_OPT_WALK)) {
1175 smb_tree_t *tree;
1177 indent = SMB_DCMD_INDENT;
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);
1184 if (opts & SMB_OPT_VERBOSE) {
1185 const char *state;
1187 if (tree->t_state >= SMB_TREE_STATE_SENTINEL)
1188 state = "INVALID";
1189 else
1190 state = smb_tree_state[tree->t_state];
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");
1210 mdb_printf("%-?p %-5u %-16s %-32s\n", addr,
1211 tree->t_tid, tree->t_sharename, tree->t_resource);
1214 if (smb_obj_expand(addr, opts, smb_tree_exp, indent))
1215 return (DCMD_ERR);
1216 return (DCMD_OK);
1220 * *****************************************************************************
1221 * ****************************** smb_odir_t ***********************************
1222 * *****************************************************************************
1225 static const char *smb_odir_state[SMB_ODIR_STATE_SENTINEL] =
1227 "OPEN",
1228 "IN_USE",
1229 "CLOSING",
1230 "CLOSED"
1233 static int
1234 smb_dcmd_odir(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1236 uint_t opts;
1238 if (smb_dcmd_getopt(&opts, argc, argv))
1239 return (DCMD_USAGE);
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));
1248 if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_ODIR)) ||
1249 !(opts & SMB_OPT_WALK)) {
1250 smb_odir_t *od;
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);
1257 if (opts & SMB_OPT_VERBOSE) {
1258 const char *state;
1260 if (od->d_state >= SMB_ODIR_STATE_SENTINEL)
1261 state = "INVALID";
1262 else
1263 state = smb_odir_state[od->d_state];
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");
1284 mdb_printf("%?p %-5u %-16p %s\n",
1285 addr, od->d_odid, od->d_dnode, od->d_pattern);
1288 return (DCMD_OK);
1292 * *****************************************************************************
1293 * ****************************** smb_ofile_t **********************************
1294 * *****************************************************************************
1297 static const char *smb_ofile_state[SMB_OFILE_STATE_SENTINEL] =
1299 "OPEN",
1300 "CLOSING",
1301 "CLOSED"
1304 static int
1305 smb_dcmd_ofile(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1307 uint_t opts;
1309 if (smb_dcmd_getopt(&opts, argc, argv))
1310 return (DCMD_USAGE);
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));
1319 if (((opts & SMB_OPT_WALK) && (opts & SMB_OPT_OFILE)) ||
1320 !(opts & SMB_OPT_WALK)) {
1321 smb_ofile_t *of;
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);
1328 if (opts & SMB_OPT_VERBOSE) {
1329 const char *state;
1331 if (of->f_state >= SMB_OFILE_STATE_SENTINEL)
1332 state = "INVALID";
1333 else
1334 state = smb_ofile_state[of->f_state];
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");
1359 mdb_printf("%?p %-5u %-p %p\n", addr,
1360 of->f_fid, of->f_node, of->f_cr);
1363 return (DCMD_OK);
1367 * *****************************************************************************
1368 * ******************************** smb_kshare_t *******************************
1369 * *****************************************************************************
1372 static int
1373 smb_kshare_cb(uintptr_t addr, const void *data, void *arg)
1375 uint_t *opts = arg;
1376 uintptr_t ta, sa;
1377 char name[32];
1378 char path[64];
1379 _NOTE(ARGUNUSED(data));
1381 if (*opts & SMB_OPT_VERBOSE) {
1382 mdb_arg_t argv;
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 {
1390 * Summary line for a kshare
1391 * Don't fail the walk if any of these fail.
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, "?");
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, "?");
1403 mdb_printf("%-?p ", addr); /* smb_kshare_t */
1404 mdb_printf("%-16s ", name);
1405 mdb_printf("%-s", path);
1406 mdb_printf("\n");
1409 return (WALK_NEXT);
1413 * ::smbshares
1415 * dcmd - Print out smb_kshare structures.
1416 * requires addr of an smb_server_t
1418 /*ARGSUSED*/
1419 static int
1420 smb_dcmd_kshare(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1422 uint_t opts = 0;
1424 if (mdb_getopts(argc, argv,
1425 'v', MDB_OPT_SETBITS, SMB_OPT_VERBOSE, &opts,
1426 NULL) != argc)
1427 return (DCMD_USAGE);
1429 if (!(flags & DCMD_ADDRSPEC))
1430 return (DCMD_USAGE);
1431 addr += OFFSETOF(smb_server_t, sv_export.e_share_avl.avl_tree);
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");
1443 if (mdb_pwalk("avl", smb_kshare_cb, &opts, addr) == -1) {
1444 mdb_warn("cannot walk smb_kshare avl");
1445 return (DCMD_ERR);
1448 return (DCMD_OK);
1452 * *****************************************************************************
1453 * ******************************** smb_vfs_t **********************************
1454 * *****************************************************************************
1458 * ::smbvfs
1460 * smbvfs dcmd - Prints out smb_vfs structures.
1462 /*ARGSUSED*/
1463 static int
1464 smb_dcmd_vfs(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1466 int verbose = FALSE;
1467 smb_vfs_t *sf;
1468 vnode_t *vn;
1469 char *path;
1471 if (mdb_getopts(argc, argv,
1472 'v', MDB_OPT_SETBITS, TRUE, &verbose,
1473 NULL) != argc)
1474 return (DCMD_USAGE);
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.
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);
1487 return (DCMD_OK);
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");
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);
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);
1515 path = mdb_zalloc(MAXPATHLEN, UM_SLEEP | UM_GC);
1516 (void) mdb_vread(path, MAXPATHLEN, (uintptr_t)vn->v_path);
1518 mdb_printf(
1519 "%-?p %-10d %-?p %-?p %-s\n", addr, sf->sv_refcnt,
1520 sf->sv_vfsp, sf->sv_rootvp, path);
1522 return (DCMD_OK);
1526 * Initialize the smb_vfs_t walker to point to the smb_export
1527 * in the specified smb_server_t instance. (no global walks)
1529 static int
1530 smb_vfs_walk_init(mdb_walk_state_t *wsp)
1533 if (wsp->walk_addr == NULL) {
1534 mdb_printf("require address of an smb_server_t\n");
1535 return (WALK_ERR);
1538 wsp->walk_addr +=
1539 OFFSETOF(smb_server_t, sv_export.e_vfs_list.ll_list);
1541 if (mdb_layered_walk("list", wsp) == -1) {
1542 mdb_warn("failed to walk list of VFS");
1543 return (WALK_ERR);
1546 return (WALK_NEXT);
1549 static int
1550 smb_vfs_walk_step(mdb_walk_state_t *wsp)
1552 return (wsp->walk_callback(wsp->walk_addr, wsp->walk_layer,
1553 wsp->walk_cbdata));
1557 * *****************************************************************************
1558 * ******************************* smb_node_t **********************************
1559 * *****************************************************************************
1562 static void
1563 smb_node_help(void)
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");
1578 * ::smbnode
1580 * smb_node dcmd - Print out smb_node structure.
1582 static int
1583 smb_dcmd_node(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
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;
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);
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.
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);
1613 return (DCMD_OK);
1617 * If this is the first invocation of the command, print a nice
1618 * header line for the output that will follow.
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");
1638 * For each smb_node, we just need to read the smb_node_t struct, read
1639 * and then print out the following fields.
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");
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");
1672 (void) mdb_dec_indent(SMB_DCMD_INDENT);
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);
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);
1693 if (print_full_path)
1694 mdb_printf("\t%s\n", path_name);
1696 if (stack_trace && node.n_audit_buf) {
1697 int ctr;
1698 smb_audit_buf_node_t *anb;
1700 anb = mdb_alloc(sizeof (smb_audit_buf_node_t),
1701 UM_SLEEP | UM_GC);
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);
1708 ctr = anb->anb_max_index + 1;
1709 anb->anb_index--;
1710 anb->anb_index &= anb->anb_max_index;
1712 while (ctr) {
1713 smb_audit_record_node_t *anr;
1715 anr = anb->anb_records + anb->anb_index;
1717 if (anr->anr_depth) {
1718 char c[MDB_SYM_NAMLEN];
1719 GElf_Sym sym;
1720 int i;
1722 mdb_printf("\nRefCnt: %u\t",
1723 anr->anr_refcnt);
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;
1735 mdb_printf("%s+0x%1x",
1737 anr->anr_stack[i] -
1738 (uintptr_t)sym.st_value);
1739 ++i;
1740 break;
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;
1752 mdb_printf("\n\t\t%s+0x%1x",
1754 anr->anr_stack[i] -
1755 (uintptr_t)sym.st_value);
1756 ++i;
1758 mdb_printf("\n");
1760 anb->anb_index--;
1761 anb->anb_index &= anb->anb_max_index;
1762 ctr--;
1765 } else {
1766 mdb_warn("failed to read struct smb_node at %p", addr);
1767 return (DCMD_ERR);
1770 return (DCMD_OK);
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.
1777 static int
1778 smb_node_walk_init(mdb_walk_state_t *wsp)
1780 GElf_Sym sym;
1781 int i;
1782 uintptr_t node_hash_table_addr;
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);
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);
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);
1805 return (WALK_NEXT);
1808 static int
1809 smb_node_walk_step(mdb_walk_state_t *wsp)
1811 return (wsp->walk_callback(wsp->walk_addr, wsp->walk_layer,
1812 wsp->walk_cbdata));
1816 * *****************************************************************************
1817 * ****************************** smb_lock_t ***********************************
1818 * *****************************************************************************
1821 static int
1822 smb_lock(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1824 smb_lock_t lock;
1825 int verbose = FALSE;
1826 uintptr_t list_addr;
1827 char *lock_type;
1829 if (mdb_getopts(argc, argv,
1830 'v', MDB_OPT_SETBITS, TRUE, &verbose,
1831 NULL) != argc)
1832 return (DCMD_USAGE);
1835 * An smb_lock_t address must be specified.
1837 if (!(flags & DCMD_ADDRSPEC))
1838 return (DCMD_USAGE);
1841 * If this is the first invocation of the command, print a nice
1842 * header line for the output that will follow.
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");
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;
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 ");
1892 (void) mdb_dec_indent(SMB_DCMD_INDENT);
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);
1904 } else {
1905 mdb_warn("failed to read struct smb_request at %p", addr);
1906 return (DCMD_ERR);
1909 return (DCMD_OK);
1913 * *****************************************************************************
1914 * ************************** smb_oplock_grant_t *******************************
1915 * *****************************************************************************
1917 /*ARGSUSED*/
1918 static int
1919 smb_oplock_grant(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1921 smb_oplock_grant_t grant;
1922 char *level;
1924 if (!(flags & DCMD_ADDRSPEC))
1925 return (DCMD_USAGE);
1928 * If this is the first invocation of the command, print a nice
1929 * header line for the output that will follow.
1931 if (DCMD_HDRSPEC(flags)) {
1932 mdb_printf("%<u>%-16s %-10s %-16s%</u>\n",
1933 "Grants:", "LEVEL", "OFILE");
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;
1952 mdb_printf("%-16p %-10s %-16p", addr, level, grant.og_ofile);
1954 return (DCMD_OK);
1958 * *****************************************************************************
1959 * ***************************** smb_oplock_t **********************************
1960 * *****************************************************************************
1962 /*ARGSUSED*/
1963 static int
1964 smb_oplock(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
1966 smb_oplock_t oplock;
1967 uintptr_t list_addr;
1969 if (!(flags & DCMD_ADDRSPEC))
1970 return (DCMD_USAGE);
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);
1977 if (oplock.ol_count == 0)
1978 return (DCMD_OK);
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;
1993 list_addr = addr + OFFSETOF(smb_oplock_t, ol_grants);
1995 if (mdb_pwalk_dcmd("list", "smboplockgrant",
1996 argc, argv, list_addr)) {
1997 mdb_warn("failed to walk oplock grants");
2000 (void) mdb_dec_indent(SMB_DCMD_INDENT);
2002 return (DCMD_OK);
2006 * ::smbstat
2008 * Prints SMB requests statistics.
2010 /*ARGSUSED*/
2011 static int
2012 smb_stats(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2014 smb_server_t *sv;
2016 if (!(flags & DCMD_ADDRSPEC))
2017 return (DCMD_USAGE);
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);
2024 if (sv->sv_magic != SMB_SERVER_MAGIC) {
2025 mdb_warn("not an smb_server_t (%p)>", addr);
2026 return (DCMD_ERR);
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);
2038 return (DCMD_OK);
2042 * *****************************************************************************
2043 * ******************************** smb_ace_t **********************************
2044 * *****************************************************************************
2046 static const ace_type_entry_t ace_types[ACE_TYPE_TABLEN] =
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)
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 }
2098 * ::smbace
2100 static int
2101 smb_ace(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2103 smb_ace_t ace;
2104 int verbose = FALSE;
2105 const char *ptr;
2106 int rc;
2108 if (mdb_getopts(argc, argv, 'v', MDB_OPT_SETBITS, TRUE, &verbose,
2109 NULL) != argc)
2110 return (DCMD_USAGE);
2113 * An smb_ace address is required.
2115 if (!(flags & DCMD_ADDRSPEC))
2116 return (DCMD_USAGE);
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);
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";
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);
2143 rc = smb_sid_print((uintptr_t)ace.se_sid);
2144 mdb_printf("\n");
2145 return (rc);
2148 static int
2149 smb_ace_walk_init(mdb_walk_state_t *wsp)
2151 if (wsp->walk_addr == 0) {
2152 mdb_printf("smb_ace walk only supports local walks\n");
2153 return (WALK_ERR);
2156 wsp->walk_addr += OFFSETOF(smb_acl_t, sl_sorted);
2158 if (mdb_layered_walk("list", wsp) == -1) {
2159 mdb_warn("failed to walk list of ACEs");
2160 return (WALK_ERR);
2163 return (WALK_NEXT);
2166 static int
2167 smb_ace_walk_step(mdb_walk_state_t *wsp)
2169 return (wsp->walk_callback(wsp->walk_addr, wsp->walk_layer,
2170 wsp->walk_cbdata));
2174 * *****************************************************************************
2175 * ******************************** smb_acl_t **********************************
2176 * *****************************************************************************
2180 * ::smbacl
2182 static int
2183 smb_acl(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2185 smb_acl_t acl;
2187 /* An smb_acl address is required. */
2188 if (!(flags & DCMD_ADDRSPEC))
2189 return (DCMD_USAGE);
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);
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);
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);
2206 (void) mdb_dec_indent(SMB_DCMD_INDENT);
2207 return (DCMD_OK);
2211 * *****************************************************************************
2212 * ********************************* smb_sd_t **********************************
2213 * *****************************************************************************
2217 * ::smbsd
2219 static int
2220 smb_sd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2222 smb_sd_t sd;
2223 int rc;
2226 * An smb_sid address is required.
2228 if (!(flags & DCMD_ADDRSPEC))
2229 return (DCMD_USAGE);
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);
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");
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");
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);
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);
2294 return (DCMD_OK);
2298 * *****************************************************************************
2299 * ********************************* smb_sid_t *********************************
2300 * *****************************************************************************
2304 * ::smbsid
2306 /*ARGSUSED*/
2307 static int
2308 smb_sid(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2311 * An smb_sid address is required.
2313 if (!(flags & DCMD_ADDRSPEC))
2314 return (DCMD_USAGE);
2316 return (smb_sid_print(addr));
2320 * smb_sid_print
2322 static int
2323 smb_sid_print(uintptr_t addr)
2325 smb_sid_t sid;
2326 smb_sid_t *psid;
2327 size_t sid_size;
2328 int i;
2329 uint64_t authority;
2331 sid_size = OFFSETOF(smb_sid_t, sid_subauth);
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);
2338 sid_size += sid.sid_subauthcnt * sizeof (sid.sid_subauth[0]);
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);
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);
2352 mdb_printf("-%ll", authority);
2354 for (i = 0; i < psid->sid_subauthcnt; i++)
2355 mdb_printf("-%d", psid->sid_subauth[i]);
2357 return (DCMD_OK);
2361 * *****************************************************************************
2362 * ********************************* smb_fssd_t ********************************
2363 * *****************************************************************************
2367 * ::smbfssd
2369 static int
2370 smb_fssd(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
2372 smb_fssd_t fssd;
2373 int rc;
2376 * An smb_fssd address is required.
2378 if (!(flags & DCMD_ADDRSPEC))
2379 return (DCMD_USAGE);
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);
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);
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);
2410 return (DCMD_OK);
2414 * *****************************************************************************
2415 * **************************** Utility Funcions *******************************
2416 * *****************************************************************************
2420 * smb_dcmd_getopt
2422 * This function analyzes the arguments passed in and sets the bit corresponding
2423 * to the options found in the opts variable.
2425 * Return Value
2427 * -1 An error occured during the decoding
2428 * 0 The decoding was successful
2430 static int
2431 smb_dcmd_getopt(uint_t *opts, int argc, const mdb_arg_t *argv)
2433 *opts = 0;
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);
2448 return (0);
2452 * smb_dcmd_setopt
2454 * This function set the arguments corresponding to the bits set in opts.
2456 * Return Value
2458 * Number of arguments set.
2460 static int
2461 smb_dcmd_setopt(uint_t opts, int max_argc, mdb_arg_t *argv)
2463 int i;
2464 int argc = 0;
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++;
2474 return (argc);
2478 * smb_obj_expand
2480 static int
2481 smb_obj_expand(uintptr_t addr, uint_t opts, const smb_exp_t *x, ulong_t indent)
2483 int rc = 0;
2484 int argc;
2485 mdb_arg_t argv[SMB_MDB_MAX_OPTS];
2487 argc = smb_dcmd_setopt(opts | SMB_OPT_WALK, SMB_MDB_MAX_OPTS, argv);
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);
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;
2501 x++;
2503 (void) mdb_dec_indent(indent);
2504 return (rc);
2508 * smb_obj_list
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).
2514 * Return Value
2516 * DCMD_OK
2517 * DCMD_ERR
2519 static int
2520 smb_obj_list(const char *name, uint_t opts, uint_t flags)
2522 int argc;
2523 mdb_arg_t argv[SMB_MDB_MAX_OPTS];
2525 argc = smb_dcmd_setopt(opts, SMB_MDB_MAX_OPTS, argv);
2527 if (mdb_call_dcmd("smblist", 0, flags, argc, argv)) {
2528 mdb_warn("failed to list %s", name);
2529 return (DCMD_ERR);
2531 return (DCMD_OK);
2534 static int
2535 smb_worker_findstack(uintptr_t addr)
2537 char cmd[80];
2538 mdb_arg_t cmdarg;
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);