Print this page
7127 remove -Wno-missing-braces from Makefile.uts
Split |
Close |
Expand all |
Collapse all |
--- old/usr/src/uts/common/os/sunmdi.c
+++ new/usr/src/uts/common/os/sunmdi.c
1 1 /*
2 2 * CDDL HEADER START
3 3 *
4 4 * The contents of this file are subject to the terms of the
5 5 * Common Development and Distribution License (the "License").
6 6 * You may not use this file except in compliance with the License.
7 7 *
8 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 9 * or http://www.opensolaris.org/os/licensing.
10 10 * See the License for the specific language governing permissions
11 11 * and limitations under the License.
12 12 *
13 13 * When distributing Covered Code, include this CDDL HEADER in each
14 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 15 * If applicable, add the following below this CDDL HEADER, with the
16 16 * fields enclosed by brackets "[]" replaced with your own identifying
17 17 * information: Portions Copyright [yyyy] [name of copyright owner]
18 18 *
19 19 * CDDL HEADER END
20 20 */
21 21 /*
22 22 * Copyright (c) 2000, 2010, Oracle and/or its affiliates. All rights reserved.
23 23 * Copyright (c) 2014 Nexenta Systems Inc. All rights reserved.
24 24 */
25 25
26 26 /*
27 27 * Multipath driver interface (MDI) implementation; see mdi_impldefs.h for a
28 28 * more detailed discussion of the overall mpxio architecture.
29 29 *
30 30 * Default locking order:
31 31 *
32 32 * _NOTE(LOCK_ORDER(mdi_mutex, mdi_vhci:vh_phci_mutex);
33 33 * _NOTE(LOCK_ORDER(mdi_mutex, mdi_vhci:vh_client_mutex);
34 34 * _NOTE(LOCK_ORDER(mdi_vhci:vh_phci_mutex, mdi_phci::ph_mutex);
35 35 * _NOTE(LOCK_ORDER(mdi_vhci:vh_client_mutex, mdi_client::ct_mutex);
36 36 * _NOTE(LOCK_ORDER(mdi_phci::ph_mutex mdi_pathinfo::pi_mutex))
37 37 * _NOTE(LOCK_ORDER(mdi_phci::ph_mutex mdi_client::ct_mutex))
38 38 * _NOTE(LOCK_ORDER(mdi_client::ct_mutex mdi_pathinfo::pi_mutex))
39 39 */
40 40
41 41 #include <sys/note.h>
42 42 #include <sys/types.h>
43 43 #include <sys/varargs.h>
44 44 #include <sys/param.h>
45 45 #include <sys/errno.h>
46 46 #include <sys/uio.h>
47 47 #include <sys/buf.h>
48 48 #include <sys/modctl.h>
49 49 #include <sys/open.h>
50 50 #include <sys/kmem.h>
51 51 #include <sys/poll.h>
52 52 #include <sys/conf.h>
53 53 #include <sys/bootconf.h>
54 54 #include <sys/cmn_err.h>
55 55 #include <sys/stat.h>
56 56 #include <sys/ddi.h>
57 57 #include <sys/sunddi.h>
58 58 #include <sys/ddipropdefs.h>
59 59 #include <sys/sunndi.h>
60 60 #include <sys/ndi_impldefs.h>
61 61 #include <sys/promif.h>
62 62 #include <sys/sunmdi.h>
63 63 #include <sys/mdi_impldefs.h>
64 64 #include <sys/taskq.h>
65 65 #include <sys/epm.h>
66 66 #include <sys/sunpm.h>
67 67 #include <sys/modhash.h>
68 68 #include <sys/disp.h>
69 69 #include <sys/autoconf.h>
70 70 #include <sys/sysmacros.h>
71 71
72 72 #ifdef DEBUG
73 73 #include <sys/debug.h>
74 74 int mdi_debug = 1;
75 75 int mdi_debug_logonly = 0;
76 76 #define MDI_DEBUG(dbglevel, pargs) if (mdi_debug >= (dbglevel)) i_mdi_log pargs
77 77 #define MDI_WARN CE_WARN, __func__
78 78 #define MDI_NOTE CE_NOTE, __func__
79 79 #define MDI_CONT CE_CONT, __func__
80 80 static void i_mdi_log(int, const char *, dev_info_t *, const char *, ...);
81 81 #else /* !DEBUG */
82 82 #define MDI_DEBUG(dbglevel, pargs)
83 83 #endif /* DEBUG */
84 84 int mdi_debug_consoleonly = 0;
85 85 int mdi_delay = 3;
86 86
87 87 extern pri_t minclsyspri;
88 88 extern int modrootloaded;
89 89
90 90 /*
91 91 * Global mutex:
92 92 * Protects vHCI list and structure members.
93 93 */
94 94 kmutex_t mdi_mutex;
95 95
96 96 /*
97 97 * Registered vHCI class driver lists
98 98 */
99 99 int mdi_vhci_count;
100 100 mdi_vhci_t *mdi_vhci_head;
101 101 mdi_vhci_t *mdi_vhci_tail;
102 102
103 103 /*
104 104 * Client Hash Table size
105 105 */
106 106 static int mdi_client_table_size = CLIENT_HASH_TABLE_SIZE;
107 107
108 108 /*
109 109 * taskq interface definitions
110 110 */
111 111 #define MDI_TASKQ_N_THREADS 8
112 112 #define MDI_TASKQ_PRI minclsyspri
113 113 #define MDI_TASKQ_MINALLOC (4*mdi_taskq_n_threads)
114 114 #define MDI_TASKQ_MAXALLOC (500*mdi_taskq_n_threads)
115 115
116 116 taskq_t *mdi_taskq;
117 117 static uint_t mdi_taskq_n_threads = MDI_TASKQ_N_THREADS;
118 118
119 119 #define TICKS_PER_SECOND (drv_usectohz(1000000))
120 120
121 121 /*
122 122 * The data should be "quiet" for this interval (in seconds) before the
123 123 * vhci cached data is flushed to the disk.
124 124 */
125 125 static int mdi_vhcache_flush_delay = 10;
126 126
127 127 /* number of seconds the vhcache flush daemon will sleep idle before exiting */
128 128 static int mdi_vhcache_flush_daemon_idle_time = 60;
129 129
130 130 /*
131 131 * MDI falls back to discovery of all paths when a bus_config_one fails.
132 132 * The following parameters can be used to tune this operation.
133 133 *
134 134 * mdi_path_discovery_boot
135 135 * Number of times path discovery will be attempted during early boot.
136 136 * Probably there is no reason to ever set this value to greater than one.
137 137 *
138 138 * mdi_path_discovery_postboot
139 139 * Number of times path discovery will be attempted after early boot.
140 140 * Set it to a minimum of two to allow for discovery of iscsi paths which
141 141 * may happen very late during booting.
142 142 *
143 143 * mdi_path_discovery_interval
144 144 * Minimum number of seconds MDI will wait between successive discovery
145 145 * of all paths. Set it to -1 to disable discovery of all paths.
146 146 */
147 147 static int mdi_path_discovery_boot = 1;
148 148 static int mdi_path_discovery_postboot = 2;
149 149 static int mdi_path_discovery_interval = 10;
150 150
151 151 /*
152 152 * number of seconds the asynchronous configuration thread will sleep idle
153 153 * before exiting.
154 154 */
155 155 static int mdi_async_config_idle_time = 600;
156 156
157 157 static int mdi_bus_config_cache_hash_size = 256;
158 158
159 159 /* turns off multithreaded configuration for certain operations */
160 160 static int mdi_mtc_off = 0;
161 161
162 162 /*
163 163 * The "path" to a pathinfo node is identical to the /devices path to a
164 164 * devinfo node had the device been enumerated under a pHCI instead of
165 165 * a vHCI. This pathinfo "path" is associated with a 'path_instance'.
166 166 * This association persists across create/delete of the pathinfo nodes,
167 167 * but not across reboot.
168 168 */
169 169 static uint_t mdi_pathmap_instance = 1; /* 0 -> any path */
170 170 static int mdi_pathmap_hash_size = 256;
171 171 static kmutex_t mdi_pathmap_mutex;
172 172 static mod_hash_t *mdi_pathmap_bypath; /* "path"->instance */
173 173 static mod_hash_t *mdi_pathmap_byinstance; /* instance->"path" */
174 174 static mod_hash_t *mdi_pathmap_sbyinstance; /* inst->shortpath */
175 175
176 176 /*
177 177 * MDI component property name/value string definitions
178 178 */
179 179 const char *mdi_component_prop = "mpxio-component";
180 180 const char *mdi_component_prop_vhci = "vhci";
181 181 const char *mdi_component_prop_phci = "phci";
182 182 const char *mdi_component_prop_client = "client";
183 183
184 184 /*
185 185 * MDI client global unique identifier property name
186 186 */
187 187 const char *mdi_client_guid_prop = "client-guid";
188 188
189 189 /*
190 190 * MDI client load balancing property name/value string definitions
191 191 */
192 192 const char *mdi_load_balance = "load-balance";
193 193 const char *mdi_load_balance_none = "none";
194 194 const char *mdi_load_balance_rr = "round-robin";
195 195 const char *mdi_load_balance_lba = "logical-block";
196 196
197 197 /*
198 198 * Obsolete vHCI class definition; to be removed after Leadville update
199 199 */
200 200 const char *mdi_vhci_class_scsi = MDI_HCI_CLASS_SCSI;
201 201
202 202 static char vhci_greeting[] =
203 203 "\tThere already exists one vHCI driver for class %s\n"
204 204 "\tOnly one vHCI driver for each class is allowed\n";
205 205
206 206 /*
207 207 * Static function prototypes
208 208 */
209 209 static int i_mdi_phci_offline(dev_info_t *, uint_t);
210 210 static int i_mdi_client_offline(dev_info_t *, uint_t);
211 211 static int i_mdi_phci_pre_detach(dev_info_t *, ddi_detach_cmd_t);
212 212 static void i_mdi_phci_post_detach(dev_info_t *,
213 213 ddi_detach_cmd_t, int);
214 214 static int i_mdi_client_pre_detach(dev_info_t *,
215 215 ddi_detach_cmd_t);
216 216 static void i_mdi_client_post_detach(dev_info_t *,
217 217 ddi_detach_cmd_t, int);
218 218 static void i_mdi_pm_hold_pip(mdi_pathinfo_t *);
219 219 static void i_mdi_pm_rele_pip(mdi_pathinfo_t *);
220 220 static int i_mdi_lba_lb(mdi_client_t *ct,
221 221 mdi_pathinfo_t **ret_pip, struct buf *buf);
222 222 static void i_mdi_pm_hold_client(mdi_client_t *, int);
223 223 static void i_mdi_pm_rele_client(mdi_client_t *, int);
224 224 static void i_mdi_pm_reset_client(mdi_client_t *);
225 225 static int i_mdi_power_all_phci(mdi_client_t *);
226 226 static void i_mdi_log_sysevent(dev_info_t *, char *, char *);
227 227
228 228
229 229 /*
230 230 * Internal mdi_pathinfo node functions
231 231 */
232 232 static void i_mdi_pi_kstat_destroy(mdi_pathinfo_t *);
233 233
234 234 static mdi_vhci_t *i_mdi_vhci_class2vhci(char *);
235 235 static mdi_vhci_t *i_devi_get_vhci(dev_info_t *);
236 236 static mdi_phci_t *i_devi_get_phci(dev_info_t *);
237 237 static void i_mdi_phci_lock(mdi_phci_t *, mdi_pathinfo_t *);
238 238 static void i_mdi_phci_unlock(mdi_phci_t *);
239 239 static mdi_pathinfo_t *i_mdi_pi_alloc(mdi_phci_t *, char *, mdi_client_t *);
240 240 static void i_mdi_phci_add_path(mdi_phci_t *, mdi_pathinfo_t *);
241 241 static void i_mdi_client_add_path(mdi_client_t *, mdi_pathinfo_t *);
242 242 static void i_mdi_pi_free(mdi_phci_t *ph, mdi_pathinfo_t *,
243 243 mdi_client_t *);
244 244 static void i_mdi_phci_remove_path(mdi_phci_t *, mdi_pathinfo_t *);
245 245 static void i_mdi_client_remove_path(mdi_client_t *,
246 246 mdi_pathinfo_t *);
247 247
248 248 static int i_mdi_pi_state_change(mdi_pathinfo_t *,
249 249 mdi_pathinfo_state_t, int);
250 250 static int i_mdi_pi_offline(mdi_pathinfo_t *, int);
251 251 static dev_info_t *i_mdi_devinfo_create(mdi_vhci_t *, char *, char *,
252 252 char **, int);
253 253 static dev_info_t *i_mdi_devinfo_find(mdi_vhci_t *, char *, char *);
254 254 static int i_mdi_devinfo_remove(dev_info_t *, dev_info_t *, int);
255 255 static int i_mdi_is_child_present(dev_info_t *, dev_info_t *);
256 256 static mdi_client_t *i_mdi_client_alloc(mdi_vhci_t *, char *, char *);
257 257 static void i_mdi_client_enlist_table(mdi_vhci_t *, mdi_client_t *);
258 258 static void i_mdi_client_delist_table(mdi_vhci_t *, mdi_client_t *);
259 259 static mdi_client_t *i_mdi_client_find(mdi_vhci_t *, char *, char *);
260 260 static void i_mdi_client_update_state(mdi_client_t *);
261 261 static int i_mdi_client_compute_state(mdi_client_t *,
262 262 mdi_phci_t *);
263 263 static void i_mdi_client_lock(mdi_client_t *, mdi_pathinfo_t *);
264 264 static void i_mdi_client_unlock(mdi_client_t *);
265 265 static int i_mdi_client_free(mdi_vhci_t *, mdi_client_t *);
266 266 static mdi_client_t *i_devi_get_client(dev_info_t *);
267 267 /*
268 268 * NOTE: this will be removed once the NWS files are changed to use the new
269 269 * mdi_{enable,disable}_path interfaces
270 270 */
271 271 static int i_mdi_pi_enable_disable(dev_info_t *, dev_info_t *,
272 272 int, int);
273 273 static mdi_pathinfo_t *i_mdi_enable_disable_path(mdi_pathinfo_t *pip,
274 274 mdi_vhci_t *vh, int flags, int op);
275 275 /*
276 276 * Failover related function prototypes
277 277 */
278 278 static int i_mdi_failover(void *);
279 279
280 280 /*
281 281 * misc internal functions
282 282 */
283 283 static int i_mdi_get_hash_key(char *);
284 284 static int i_map_nvlist_error_to_mdi(int);
285 285 static void i_mdi_report_path_state(mdi_client_t *,
286 286 mdi_pathinfo_t *);
287 287
288 288 static void setup_vhci_cache(mdi_vhci_t *);
289 289 static int destroy_vhci_cache(mdi_vhci_t *);
290 290 static int stop_vhcache_async_threads(mdi_vhci_config_t *);
291 291 static boolean_t stop_vhcache_flush_thread(void *, int);
292 292 static void free_string_array(char **, int);
293 293 static void free_vhcache_phci(mdi_vhcache_phci_t *);
294 294 static void free_vhcache_pathinfo(mdi_vhcache_pathinfo_t *);
295 295 static void free_vhcache_client(mdi_vhcache_client_t *);
296 296 static int mainnvl_to_vhcache(mdi_vhci_cache_t *, nvlist_t *);
297 297 static nvlist_t *vhcache_to_mainnvl(mdi_vhci_cache_t *);
298 298 static void vhcache_phci_add(mdi_vhci_config_t *, mdi_phci_t *);
299 299 static void vhcache_phci_remove(mdi_vhci_config_t *, mdi_phci_t *);
300 300 static void vhcache_pi_add(mdi_vhci_config_t *,
301 301 struct mdi_pathinfo *);
302 302 static void vhcache_pi_remove(mdi_vhci_config_t *,
303 303 struct mdi_pathinfo *);
304 304 static void free_phclient_path_list(mdi_phys_path_t *);
305 305 static void sort_vhcache_paths(mdi_vhcache_client_t *);
306 306 static int flush_vhcache(mdi_vhci_config_t *, int);
307 307 static void vhcache_dirty(mdi_vhci_config_t *);
308 308 static void free_async_client_config(mdi_async_client_config_t *);
309 309 static void single_threaded_vhconfig_enter(mdi_vhci_config_t *);
310 310 static void single_threaded_vhconfig_exit(mdi_vhci_config_t *);
311 311 static nvlist_t *read_on_disk_vhci_cache(char *);
312 312 extern int fread_nvlist(char *, nvlist_t **);
313 313 extern int fwrite_nvlist(char *, nvlist_t *);
314 314
315 315 /* called once when first vhci registers with mdi */
316 316 static void
317 317 i_mdi_init()
318 318 {
319 319 static int initialized = 0;
320 320
321 321 if (initialized)
322 322 return;
323 323 initialized = 1;
324 324
325 325 mutex_init(&mdi_mutex, NULL, MUTEX_DEFAULT, NULL);
326 326
327 327 /* Create our taskq resources */
328 328 mdi_taskq = taskq_create("mdi_taskq", mdi_taskq_n_threads,
329 329 MDI_TASKQ_PRI, MDI_TASKQ_MINALLOC, MDI_TASKQ_MAXALLOC,
330 330 TASKQ_PREPOPULATE | TASKQ_CPR_SAFE);
331 331 ASSERT(mdi_taskq != NULL); /* taskq_create never fails */
332 332
333 333 /* Allocate ['path_instance' <-> "path"] maps */
334 334 mutex_init(&mdi_pathmap_mutex, NULL, MUTEX_DRIVER, NULL);
335 335 mdi_pathmap_bypath = mod_hash_create_strhash(
336 336 "mdi_pathmap_bypath", mdi_pathmap_hash_size,
337 337 mod_hash_null_valdtor);
338 338 mdi_pathmap_byinstance = mod_hash_create_idhash(
339 339 "mdi_pathmap_byinstance", mdi_pathmap_hash_size,
340 340 mod_hash_null_valdtor);
341 341 mdi_pathmap_sbyinstance = mod_hash_create_idhash(
342 342 "mdi_pathmap_sbyinstance", mdi_pathmap_hash_size,
343 343 mod_hash_null_valdtor);
344 344 }
345 345
346 346 /*
347 347 * mdi_get_component_type():
348 348 * Return mpxio component type
349 349 * Return Values:
350 350 * MDI_COMPONENT_NONE
351 351 * MDI_COMPONENT_VHCI
352 352 * MDI_COMPONENT_PHCI
353 353 * MDI_COMPONENT_CLIENT
354 354 * XXX This doesn't work under multi-level MPxIO and should be
355 355 * removed when clients migrate mdi_component_is_*() interfaces.
356 356 */
357 357 int
358 358 mdi_get_component_type(dev_info_t *dip)
359 359 {
360 360 return (DEVI(dip)->devi_mdi_component);
361 361 }
362 362
363 363 /*
364 364 * mdi_vhci_register():
365 365 * Register a vHCI module with the mpxio framework
366 366 * mdi_vhci_register() is called by vHCI drivers to register the
367 367 * 'class_driver' vHCI driver and its MDI entrypoints with the
368 368 * mpxio framework. The vHCI driver must call this interface as
369 369 * part of its attach(9e) handler.
370 370 * Competing threads may try to attach mdi_vhci_register() as
371 371 * the vHCI drivers are loaded and attached as a result of pHCI
372 372 * driver instance registration (mdi_phci_register()) with the
373 373 * framework.
374 374 * Return Values:
375 375 * MDI_SUCCESS
376 376 * MDI_FAILURE
377 377 */
378 378 /*ARGSUSED*/
379 379 int
380 380 mdi_vhci_register(char *class, dev_info_t *vdip, mdi_vhci_ops_t *vops,
381 381 int flags)
382 382 {
383 383 mdi_vhci_t *vh = NULL;
384 384
385 385 /* Registrant can't be older */
386 386 ASSERT(vops->vo_revision <= MDI_VHCI_OPS_REV);
387 387
388 388 #ifdef DEBUG
389 389 /*
390 390 * IB nexus driver is loaded only when IB hardware is present.
391 391 * In order to be able to do this there is a need to drive the loading
392 392 * and attaching of the IB nexus driver (especially when an IB hardware
393 393 * is dynamically plugged in) when an IB HCA driver (PHCI)
394 394 * is being attached. Unfortunately this gets into the limitations
395 395 * of devfs as there seems to be no clean way to drive configuration
396 396 * of a subtree from another subtree of a devfs. Hence, do not ASSERT
397 397 * for IB.
398 398 */
399 399 if (strcmp(class, MDI_HCI_CLASS_IB) != 0)
400 400 ASSERT(DEVI_BUSY_OWNED(ddi_get_parent(vdip)));
401 401 #endif
402 402
403 403 i_mdi_init();
404 404
405 405 mutex_enter(&mdi_mutex);
406 406 /*
407 407 * Scan for already registered vhci
408 408 */
409 409 for (vh = mdi_vhci_head; vh != NULL; vh = vh->vh_next) {
410 410 if (strcmp(vh->vh_class, class) == 0) {
411 411 /*
412 412 * vHCI has already been created. Check for valid
413 413 * vHCI ops registration. We only support one vHCI
414 414 * module per class
415 415 */
416 416 if (vh->vh_ops != NULL) {
417 417 mutex_exit(&mdi_mutex);
418 418 cmn_err(CE_NOTE, vhci_greeting, class);
419 419 return (MDI_FAILURE);
420 420 }
421 421 break;
422 422 }
423 423 }
424 424
425 425 /*
426 426 * if not yet created, create the vHCI component
427 427 */
428 428 if (vh == NULL) {
429 429 struct client_hash *hash = NULL;
430 430 char *load_balance;
431 431
432 432 /*
433 433 * Allocate and initialize the mdi extensions
434 434 */
435 435 vh = kmem_zalloc(sizeof (mdi_vhci_t), KM_SLEEP);
436 436 hash = kmem_zalloc(mdi_client_table_size * sizeof (*hash),
437 437 KM_SLEEP);
438 438 vh->vh_client_table = hash;
439 439 vh->vh_class = kmem_zalloc(strlen(class) + 1, KM_SLEEP);
440 440 (void) strcpy(vh->vh_class, class);
441 441 vh->vh_lb = LOAD_BALANCE_RR;
442 442 if (ddi_prop_lookup_string(DDI_DEV_T_ANY, vdip,
443 443 0, LOAD_BALANCE_PROP, &load_balance) == DDI_SUCCESS) {
444 444 if (strcmp(load_balance, LOAD_BALANCE_PROP_NONE) == 0) {
445 445 vh->vh_lb = LOAD_BALANCE_NONE;
446 446 } else if (strcmp(load_balance, LOAD_BALANCE_PROP_LBA)
447 447 == 0) {
448 448 vh->vh_lb = LOAD_BALANCE_LBA;
449 449 }
450 450 ddi_prop_free(load_balance);
451 451 }
452 452
453 453 mutex_init(&vh->vh_phci_mutex, NULL, MUTEX_DEFAULT, NULL);
454 454 mutex_init(&vh->vh_client_mutex, NULL, MUTEX_DEFAULT, NULL);
455 455
456 456 /*
457 457 * Store the vHCI ops vectors
458 458 */
459 459 vh->vh_dip = vdip;
460 460 vh->vh_ops = vops;
461 461
462 462 setup_vhci_cache(vh);
463 463
464 464 if (mdi_vhci_head == NULL) {
465 465 mdi_vhci_head = vh;
466 466 }
467 467 if (mdi_vhci_tail) {
468 468 mdi_vhci_tail->vh_next = vh;
469 469 }
470 470 mdi_vhci_tail = vh;
471 471 mdi_vhci_count++;
472 472 }
473 473
474 474 /*
475 475 * Claim the devfs node as a vhci component
476 476 */
477 477 DEVI(vdip)->devi_mdi_component |= MDI_COMPONENT_VHCI;
478 478
479 479 /*
480 480 * Initialize our back reference from dev_info node
481 481 */
482 482 DEVI(vdip)->devi_mdi_xhci = (caddr_t)vh;
483 483 mutex_exit(&mdi_mutex);
484 484 return (MDI_SUCCESS);
485 485 }
486 486
487 487 /*
488 488 * mdi_vhci_unregister():
489 489 * Unregister a vHCI module from mpxio framework
490 490 * mdi_vhci_unregister() is called from the detach(9E) entrypoint
491 491 * of a vhci to unregister it from the framework.
492 492 * Return Values:
493 493 * MDI_SUCCESS
494 494 * MDI_FAILURE
495 495 */
496 496 /*ARGSUSED*/
497 497 int
498 498 mdi_vhci_unregister(dev_info_t *vdip, int flags)
499 499 {
500 500 mdi_vhci_t *found, *vh, *prev = NULL;
501 501
502 502 ASSERT(DEVI_BUSY_OWNED(ddi_get_parent(vdip)));
503 503
504 504 /*
505 505 * Check for invalid VHCI
506 506 */
507 507 if ((vh = i_devi_get_vhci(vdip)) == NULL)
508 508 return (MDI_FAILURE);
509 509
510 510 /*
511 511 * Scan the list of registered vHCIs for a match
512 512 */
513 513 mutex_enter(&mdi_mutex);
514 514 for (found = mdi_vhci_head; found != NULL; found = found->vh_next) {
515 515 if (found == vh)
516 516 break;
517 517 prev = found;
518 518 }
519 519
520 520 if (found == NULL) {
521 521 mutex_exit(&mdi_mutex);
522 522 return (MDI_FAILURE);
523 523 }
524 524
525 525 /*
526 526 * Check the vHCI, pHCI and client count. All the pHCIs and clients
527 527 * should have been unregistered, before a vHCI can be
528 528 * unregistered.
529 529 */
530 530 MDI_VHCI_PHCI_LOCK(vh);
531 531 if (vh->vh_refcnt || vh->vh_phci_count || vh->vh_client_count) {
532 532 MDI_VHCI_PHCI_UNLOCK(vh);
533 533 mutex_exit(&mdi_mutex);
534 534 return (MDI_FAILURE);
535 535 }
536 536 MDI_VHCI_PHCI_UNLOCK(vh);
537 537
538 538 if (destroy_vhci_cache(vh) != MDI_SUCCESS) {
539 539 mutex_exit(&mdi_mutex);
540 540 return (MDI_FAILURE);
541 541 }
542 542
543 543 /*
544 544 * Remove the vHCI from the global list
545 545 */
546 546 if (vh == mdi_vhci_head) {
547 547 mdi_vhci_head = vh->vh_next;
548 548 } else {
549 549 prev->vh_next = vh->vh_next;
550 550 }
551 551 if (vh == mdi_vhci_tail) {
552 552 mdi_vhci_tail = prev;
553 553 }
554 554 mdi_vhci_count--;
555 555 mutex_exit(&mdi_mutex);
556 556
557 557 vh->vh_ops = NULL;
558 558 DEVI(vdip)->devi_mdi_component &= ~MDI_COMPONENT_VHCI;
559 559 DEVI(vdip)->devi_mdi_xhci = NULL;
560 560 kmem_free(vh->vh_class, strlen(vh->vh_class)+1);
561 561 kmem_free(vh->vh_client_table,
562 562 mdi_client_table_size * sizeof (struct client_hash));
563 563 mutex_destroy(&vh->vh_phci_mutex);
564 564 mutex_destroy(&vh->vh_client_mutex);
565 565
566 566 kmem_free(vh, sizeof (mdi_vhci_t));
567 567 return (MDI_SUCCESS);
568 568 }
569 569
570 570 /*
571 571 * i_mdi_vhci_class2vhci():
572 572 * Look for a matching vHCI module given a vHCI class name
573 573 * Return Values:
574 574 * Handle to a vHCI component
575 575 * NULL
576 576 */
577 577 static mdi_vhci_t *
578 578 i_mdi_vhci_class2vhci(char *class)
579 579 {
580 580 mdi_vhci_t *vh = NULL;
581 581
582 582 ASSERT(!MUTEX_HELD(&mdi_mutex));
583 583
584 584 mutex_enter(&mdi_mutex);
585 585 for (vh = mdi_vhci_head; vh != NULL; vh = vh->vh_next) {
586 586 if (strcmp(vh->vh_class, class) == 0) {
587 587 break;
588 588 }
589 589 }
590 590 mutex_exit(&mdi_mutex);
591 591 return (vh);
592 592 }
593 593
594 594 /*
595 595 * i_devi_get_vhci():
596 596 * Utility function to get the handle to a vHCI component
597 597 * Return Values:
598 598 * Handle to a vHCI component
599 599 * NULL
600 600 */
601 601 mdi_vhci_t *
602 602 i_devi_get_vhci(dev_info_t *vdip)
603 603 {
604 604 mdi_vhci_t *vh = NULL;
605 605 if (MDI_VHCI(vdip)) {
606 606 vh = (mdi_vhci_t *)DEVI(vdip)->devi_mdi_xhci;
607 607 }
608 608 return (vh);
609 609 }
610 610
611 611 /*
612 612 * mdi_phci_register():
613 613 * Register a pHCI module with mpxio framework
614 614 * mdi_phci_register() is called by pHCI drivers to register with
615 615 * the mpxio framework and a specific 'class_driver' vHCI. The
616 616 * pHCI driver must call this interface as part of its attach(9e)
617 617 * handler.
618 618 * Return Values:
619 619 * MDI_SUCCESS
620 620 * MDI_FAILURE
621 621 */
622 622 /*ARGSUSED*/
623 623 int
624 624 mdi_phci_register(char *class, dev_info_t *pdip, int flags)
625 625 {
626 626 mdi_phci_t *ph;
627 627 mdi_vhci_t *vh;
628 628 char *data;
629 629
630 630 /*
631 631 * Some subsystems, like fcp, perform pHCI registration from a
632 632 * different thread than the one doing the pHCI attach(9E) - the
633 633 * driver attach code is waiting for this other thread to complete.
634 634 * This means we can only ASSERT DEVI_BUSY_CHANGING of parent
635 635 * (indicating that some thread has done an ndi_devi_enter of parent)
636 636 * not DEVI_BUSY_OWNED (which would indicate that we did the enter).
637 637 */
638 638 ASSERT(DEVI_BUSY_CHANGING(ddi_get_parent(pdip)));
639 639
640 640 /*
641 641 * Check for mpxio-disable property. Enable mpxio if the property is
642 642 * missing or not set to "yes".
643 643 * If the property is set to "yes" then emit a brief message.
644 644 */
645 645 if ((ddi_prop_lookup_string(DDI_DEV_T_ANY, pdip, 0, "mpxio-disable",
646 646 &data) == DDI_SUCCESS)) {
647 647 if (strcmp(data, "yes") == 0) {
648 648 MDI_DEBUG(1, (MDI_CONT, pdip,
649 649 "?multipath capabilities disabled via %s.conf.",
650 650 ddi_driver_name(pdip)));
651 651 ddi_prop_free(data);
652 652 return (MDI_FAILURE);
653 653 }
654 654 ddi_prop_free(data);
655 655 }
656 656
657 657 /*
658 658 * Search for a matching vHCI
659 659 */
660 660 vh = (mdi_vhci_t *)i_mdi_vhci_class2vhci(class);
661 661 if (vh == NULL) {
662 662 return (MDI_FAILURE);
663 663 }
664 664
665 665 ph = kmem_zalloc(sizeof (mdi_phci_t), KM_SLEEP);
666 666 mutex_init(&ph->ph_mutex, NULL, MUTEX_DEFAULT, NULL);
667 667 ph->ph_dip = pdip;
668 668 ph->ph_vhci = vh;
669 669 ph->ph_next = NULL;
670 670 ph->ph_unstable = 0;
671 671 ph->ph_vprivate = 0;
672 672 cv_init(&ph->ph_unstable_cv, NULL, CV_DRIVER, NULL);
673 673
674 674 MDI_PHCI_LOCK(ph);
675 675 MDI_PHCI_SET_POWER_UP(ph);
676 676 MDI_PHCI_UNLOCK(ph);
677 677 DEVI(pdip)->devi_mdi_component |= MDI_COMPONENT_PHCI;
678 678 DEVI(pdip)->devi_mdi_xhci = (caddr_t)ph;
679 679
680 680 vhcache_phci_add(vh->vh_config, ph);
681 681
682 682 MDI_VHCI_PHCI_LOCK(vh);
683 683 if (vh->vh_phci_head == NULL) {
684 684 vh->vh_phci_head = ph;
685 685 }
686 686 if (vh->vh_phci_tail) {
687 687 vh->vh_phci_tail->ph_next = ph;
688 688 }
689 689 vh->vh_phci_tail = ph;
690 690 vh->vh_phci_count++;
691 691 MDI_VHCI_PHCI_UNLOCK(vh);
692 692
693 693 i_mdi_log_sysevent(pdip, class, ESC_DDI_INITIATOR_REGISTER);
694 694 return (MDI_SUCCESS);
695 695 }
696 696
697 697 /*
698 698 * mdi_phci_unregister():
699 699 * Unregister a pHCI module from mpxio framework
700 700 * mdi_phci_unregister() is called by the pHCI drivers from their
701 701 * detach(9E) handler to unregister their instances from the
702 702 * framework.
703 703 * Return Values:
704 704 * MDI_SUCCESS
705 705 * MDI_FAILURE
706 706 */
707 707 /*ARGSUSED*/
708 708 int
709 709 mdi_phci_unregister(dev_info_t *pdip, int flags)
710 710 {
711 711 mdi_vhci_t *vh;
712 712 mdi_phci_t *ph;
713 713 mdi_phci_t *tmp;
714 714 mdi_phci_t *prev = NULL;
715 715 mdi_pathinfo_t *pip;
716 716
717 717 ASSERT(DEVI_BUSY_CHANGING(ddi_get_parent(pdip)));
718 718
719 719 ph = i_devi_get_phci(pdip);
720 720 if (ph == NULL) {
721 721 MDI_DEBUG(1, (MDI_WARN, pdip, "!not a valid pHCI"));
722 722 return (MDI_FAILURE);
723 723 }
724 724
725 725 vh = ph->ph_vhci;
726 726 ASSERT(vh != NULL);
727 727 if (vh == NULL) {
728 728 MDI_DEBUG(1, (MDI_WARN, pdip, "!not a valid vHCI"));
729 729 return (MDI_FAILURE);
730 730 }
731 731
732 732 MDI_VHCI_PHCI_LOCK(vh);
733 733 tmp = vh->vh_phci_head;
734 734 while (tmp) {
735 735 if (tmp == ph) {
736 736 break;
737 737 }
738 738 prev = tmp;
739 739 tmp = tmp->ph_next;
740 740 }
741 741
742 742 if (ph == vh->vh_phci_head) {
743 743 vh->vh_phci_head = ph->ph_next;
744 744 } else {
745 745 prev->ph_next = ph->ph_next;
746 746 }
747 747
748 748 if (ph == vh->vh_phci_tail) {
749 749 vh->vh_phci_tail = prev;
750 750 }
751 751
752 752 vh->vh_phci_count--;
753 753 MDI_VHCI_PHCI_UNLOCK(vh);
754 754
755 755 /* Walk remaining pathinfo nodes and disassociate them from pHCI */
756 756 MDI_PHCI_LOCK(ph);
757 757 for (pip = (mdi_pathinfo_t *)ph->ph_path_head; pip;
758 758 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link)
759 759 MDI_PI(pip)->pi_phci = NULL;
760 760 MDI_PHCI_UNLOCK(ph);
761 761
762 762 i_mdi_log_sysevent(pdip, ph->ph_vhci->vh_class,
763 763 ESC_DDI_INITIATOR_UNREGISTER);
764 764 vhcache_phci_remove(vh->vh_config, ph);
765 765 cv_destroy(&ph->ph_unstable_cv);
766 766 mutex_destroy(&ph->ph_mutex);
767 767 kmem_free(ph, sizeof (mdi_phci_t));
768 768 DEVI(pdip)->devi_mdi_component &= ~MDI_COMPONENT_PHCI;
769 769 DEVI(pdip)->devi_mdi_xhci = NULL;
770 770 return (MDI_SUCCESS);
771 771 }
772 772
773 773 /*
774 774 * i_devi_get_phci():
775 775 * Utility function to return the phci extensions.
776 776 */
777 777 static mdi_phci_t *
778 778 i_devi_get_phci(dev_info_t *pdip)
779 779 {
780 780 mdi_phci_t *ph = NULL;
781 781
782 782 if (MDI_PHCI(pdip)) {
783 783 ph = (mdi_phci_t *)DEVI(pdip)->devi_mdi_xhci;
784 784 }
785 785 return (ph);
786 786 }
787 787
788 788 /*
789 789 * Single thread mdi entry into devinfo node for modifying its children.
790 790 * If necessary we perform an ndi_devi_enter of the vHCI before doing
791 791 * an ndi_devi_enter of 'dip'. We maintain circular in two parts: one
792 792 * for the vHCI and one for the pHCI.
793 793 */
794 794 void
795 795 mdi_devi_enter(dev_info_t *phci_dip, int *circular)
796 796 {
797 797 dev_info_t *vdip;
798 798 int vcircular, pcircular;
799 799
800 800 /* Verify calling context */
801 801 ASSERT(MDI_PHCI(phci_dip));
802 802 vdip = mdi_devi_get_vdip(phci_dip);
803 803 ASSERT(vdip); /* A pHCI always has a vHCI */
804 804
805 805 /*
806 806 * If pHCI is detaching then the framework has already entered the
807 807 * vHCI on a threads that went down the code path leading to
808 808 * detach_node(). This framework enter of the vHCI during pHCI
809 809 * detach is done to avoid deadlock with vHCI power management
810 810 * operations which enter the vHCI and the enter down the path
811 811 * to the pHCI. If pHCI is detaching then we piggyback this calls
812 812 * enter of the vHCI on frameworks vHCI enter that has already
813 813 * occurred - this is OK because we know that the framework thread
814 814 * doing detach is waiting for our completion.
815 815 *
816 816 * We should DEVI_IS_DETACHING under an enter of the parent to avoid
817 817 * race with detach - but we can't do that because the framework has
818 818 * already entered the parent, so we have some complexity instead.
819 819 */
820 820 for (;;) {
821 821 if (ndi_devi_tryenter(vdip, &vcircular)) {
822 822 ASSERT(vcircular != -1);
823 823 if (DEVI_IS_DETACHING(phci_dip)) {
824 824 ndi_devi_exit(vdip, vcircular);
825 825 vcircular = -1;
826 826 }
827 827 break;
828 828 } else if (DEVI_IS_DETACHING(phci_dip)) {
829 829 vcircular = -1;
830 830 break;
831 831 } else if (servicing_interrupt()) {
832 832 /*
833 833 * Don't delay an interrupt (and ensure adaptive
834 834 * mutex inversion support).
835 835 */
836 836 ndi_devi_enter(vdip, &vcircular);
837 837 break;
838 838 } else {
839 839 delay_random(mdi_delay);
840 840 }
841 841 }
842 842
843 843 ndi_devi_enter(phci_dip, &pcircular);
844 844 *circular = (vcircular << 16) | (pcircular & 0xFFFF);
845 845 }
846 846
847 847 /*
848 848 * Attempt to mdi_devi_enter.
849 849 */
850 850 int
851 851 mdi_devi_tryenter(dev_info_t *phci_dip, int *circular)
852 852 {
853 853 dev_info_t *vdip;
854 854 int vcircular, pcircular;
855 855
856 856 /* Verify calling context */
857 857 ASSERT(MDI_PHCI(phci_dip));
858 858 vdip = mdi_devi_get_vdip(phci_dip);
859 859 ASSERT(vdip); /* A pHCI always has a vHCI */
860 860
861 861 if (ndi_devi_tryenter(vdip, &vcircular)) {
862 862 if (ndi_devi_tryenter(phci_dip, &pcircular)) {
863 863 *circular = (vcircular << 16) | (pcircular & 0xFFFF);
864 864 return (1); /* locked */
865 865 }
866 866 ndi_devi_exit(vdip, vcircular);
867 867 }
868 868 return (0); /* busy */
869 869 }
870 870
871 871 /*
872 872 * Release mdi_devi_enter or successful mdi_devi_tryenter.
873 873 */
874 874 void
875 875 mdi_devi_exit(dev_info_t *phci_dip, int circular)
876 876 {
877 877 dev_info_t *vdip;
878 878 int vcircular, pcircular;
879 879
880 880 /* Verify calling context */
881 881 ASSERT(MDI_PHCI(phci_dip));
882 882 vdip = mdi_devi_get_vdip(phci_dip);
883 883 ASSERT(vdip); /* A pHCI always has a vHCI */
884 884
885 885 /* extract two circular recursion values from single int */
886 886 pcircular = (short)(circular & 0xFFFF);
887 887 vcircular = (short)((circular >> 16) & 0xFFFF);
888 888
889 889 ndi_devi_exit(phci_dip, pcircular);
890 890 if (vcircular != -1)
891 891 ndi_devi_exit(vdip, vcircular);
892 892 }
893 893
894 894 /*
895 895 * The functions mdi_devi_exit_phci() and mdi_devi_enter_phci() are used
896 896 * around a pHCI drivers calls to mdi_pi_online/offline, after holding
897 897 * the pathinfo node via mdi_hold_path/mdi_rele_path, to avoid deadlock
898 898 * with vHCI power management code during path online/offline. Each
899 899 * mdi_devi_exit_phci must have a matching mdi_devi_enter_phci, and both must
900 900 * occur within the scope of an active mdi_devi_enter that establishes the
901 901 * circular value.
902 902 */
903 903 void
904 904 mdi_devi_exit_phci(dev_info_t *phci_dip, int circular)
905 905 {
906 906 int pcircular;
907 907
908 908 /* Verify calling context */
909 909 ASSERT(MDI_PHCI(phci_dip));
910 910
911 911 /* Keep hold on pHCI until we reenter in mdi_devi_enter_phci */
912 912 ndi_hold_devi(phci_dip);
913 913
914 914 pcircular = (short)(circular & 0xFFFF);
915 915 ndi_devi_exit(phci_dip, pcircular);
916 916 }
917 917
918 918 void
919 919 mdi_devi_enter_phci(dev_info_t *phci_dip, int *circular)
920 920 {
921 921 int pcircular;
922 922
923 923 /* Verify calling context */
924 924 ASSERT(MDI_PHCI(phci_dip));
925 925
926 926 ndi_devi_enter(phci_dip, &pcircular);
927 927
928 928 /* Drop hold from mdi_devi_exit_phci. */
929 929 ndi_rele_devi(phci_dip);
930 930
931 931 /* verify matching mdi_devi_exit_phci/mdi_devi_enter_phci use */
932 932 ASSERT(pcircular == ((short)(*circular & 0xFFFF)));
933 933 }
934 934
935 935 /*
936 936 * mdi_devi_get_vdip():
937 937 * given a pHCI dip return vHCI dip
938 938 */
939 939 dev_info_t *
940 940 mdi_devi_get_vdip(dev_info_t *pdip)
941 941 {
942 942 mdi_phci_t *ph;
943 943
944 944 ph = i_devi_get_phci(pdip);
945 945 if (ph && ph->ph_vhci)
946 946 return (ph->ph_vhci->vh_dip);
947 947 return (NULL);
948 948 }
949 949
950 950 /*
951 951 * mdi_devi_pdip_entered():
952 952 * Return 1 if we are vHCI and have done an ndi_devi_enter
953 953 * of a pHCI
954 954 */
955 955 int
956 956 mdi_devi_pdip_entered(dev_info_t *vdip)
957 957 {
958 958 mdi_vhci_t *vh;
959 959 mdi_phci_t *ph;
960 960
961 961 vh = i_devi_get_vhci(vdip);
962 962 if (vh == NULL)
963 963 return (0);
964 964
965 965 MDI_VHCI_PHCI_LOCK(vh);
966 966 ph = vh->vh_phci_head;
967 967 while (ph) {
968 968 if (ph->ph_dip && DEVI_BUSY_OWNED(ph->ph_dip)) {
969 969 MDI_VHCI_PHCI_UNLOCK(vh);
970 970 return (1);
971 971 }
972 972 ph = ph->ph_next;
973 973 }
974 974 MDI_VHCI_PHCI_UNLOCK(vh);
975 975 return (0);
976 976 }
977 977
978 978 /*
979 979 * mdi_phci_path2devinfo():
980 980 * Utility function to search for a valid phci device given
981 981 * the devfs pathname.
982 982 */
983 983 dev_info_t *
984 984 mdi_phci_path2devinfo(dev_info_t *vdip, caddr_t pathname)
985 985 {
986 986 char *temp_pathname;
987 987 mdi_vhci_t *vh;
988 988 mdi_phci_t *ph;
989 989 dev_info_t *pdip = NULL;
990 990
991 991 vh = i_devi_get_vhci(vdip);
992 992 ASSERT(vh != NULL);
993 993
994 994 if (vh == NULL) {
995 995 /*
996 996 * Invalid vHCI component, return failure
997 997 */
998 998 return (NULL);
999 999 }
1000 1000
1001 1001 temp_pathname = kmem_zalloc(MAXPATHLEN, KM_SLEEP);
1002 1002 MDI_VHCI_PHCI_LOCK(vh);
1003 1003 ph = vh->vh_phci_head;
1004 1004 while (ph != NULL) {
1005 1005 pdip = ph->ph_dip;
1006 1006 ASSERT(pdip != NULL);
1007 1007 *temp_pathname = '\0';
1008 1008 (void) ddi_pathname(pdip, temp_pathname);
1009 1009 if (strcmp(temp_pathname, pathname) == 0) {
1010 1010 break;
1011 1011 }
1012 1012 ph = ph->ph_next;
1013 1013 }
1014 1014 if (ph == NULL) {
1015 1015 pdip = NULL;
1016 1016 }
1017 1017 MDI_VHCI_PHCI_UNLOCK(vh);
1018 1018 kmem_free(temp_pathname, MAXPATHLEN);
1019 1019 return (pdip);
1020 1020 }
1021 1021
1022 1022 /*
1023 1023 * mdi_phci_get_path_count():
1024 1024 * get number of path information nodes associated with a given
1025 1025 * pHCI device.
1026 1026 */
1027 1027 int
1028 1028 mdi_phci_get_path_count(dev_info_t *pdip)
1029 1029 {
1030 1030 mdi_phci_t *ph;
1031 1031 int count = 0;
1032 1032
1033 1033 ph = i_devi_get_phci(pdip);
1034 1034 if (ph != NULL) {
1035 1035 count = ph->ph_path_count;
1036 1036 }
1037 1037 return (count);
1038 1038 }
1039 1039
1040 1040 /*
1041 1041 * i_mdi_phci_lock():
1042 1042 * Lock a pHCI device
1043 1043 * Return Values:
1044 1044 * None
1045 1045 * Note:
1046 1046 * The default locking order is:
1047 1047 * _NOTE(LOCK_ORDER(mdi_phci::ph_mutex mdi_pathinfo::pi_mutex))
1048 1048 * But there are number of situations where locks need to be
1049 1049 * grabbed in reverse order. This routine implements try and lock
1050 1050 * mechanism depending on the requested parameter option.
1051 1051 */
1052 1052 static void
1053 1053 i_mdi_phci_lock(mdi_phci_t *ph, mdi_pathinfo_t *pip)
1054 1054 {
1055 1055 if (pip) {
1056 1056 /* Reverse locking is requested. */
1057 1057 while (MDI_PHCI_TRYLOCK(ph) == 0) {
1058 1058 if (servicing_interrupt()) {
1059 1059 MDI_PI_HOLD(pip);
1060 1060 MDI_PI_UNLOCK(pip);
1061 1061 MDI_PHCI_LOCK(ph);
1062 1062 MDI_PI_LOCK(pip);
1063 1063 MDI_PI_RELE(pip);
1064 1064 break;
1065 1065 } else {
1066 1066 /*
1067 1067 * tryenter failed. Try to grab again
1068 1068 * after a small delay
1069 1069 */
1070 1070 MDI_PI_HOLD(pip);
1071 1071 MDI_PI_UNLOCK(pip);
1072 1072 delay_random(mdi_delay);
1073 1073 MDI_PI_LOCK(pip);
1074 1074 MDI_PI_RELE(pip);
1075 1075 }
1076 1076 }
1077 1077 } else {
1078 1078 MDI_PHCI_LOCK(ph);
1079 1079 }
1080 1080 }
1081 1081
1082 1082 /*
1083 1083 * i_mdi_phci_unlock():
1084 1084 * Unlock the pHCI component
1085 1085 */
1086 1086 static void
1087 1087 i_mdi_phci_unlock(mdi_phci_t *ph)
1088 1088 {
1089 1089 MDI_PHCI_UNLOCK(ph);
1090 1090 }
1091 1091
1092 1092 /*
1093 1093 * i_mdi_devinfo_create():
1094 1094 * create client device's devinfo node
1095 1095 * Return Values:
1096 1096 * dev_info
1097 1097 * NULL
1098 1098 * Notes:
1099 1099 */
1100 1100 static dev_info_t *
1101 1101 i_mdi_devinfo_create(mdi_vhci_t *vh, char *name, char *guid,
1102 1102 char **compatible, int ncompatible)
1103 1103 {
1104 1104 dev_info_t *cdip = NULL;
1105 1105
1106 1106 ASSERT(MDI_VHCI_CLIENT_LOCKED(vh));
1107 1107
1108 1108 /* Verify for duplicate entry */
1109 1109 cdip = i_mdi_devinfo_find(vh, name, guid);
1110 1110 ASSERT(cdip == NULL);
1111 1111 if (cdip) {
1112 1112 cmn_err(CE_WARN,
1113 1113 "i_mdi_devinfo_create: client %s@%s already exists",
1114 1114 name ? name : "", guid ? guid : "");
1115 1115 }
1116 1116
1117 1117 ndi_devi_alloc_sleep(vh->vh_dip, name, DEVI_SID_NODEID, &cdip);
1118 1118 if (cdip == NULL)
1119 1119 goto fail;
1120 1120
1121 1121 /*
1122 1122 * Create component type and Global unique identifier
1123 1123 * properties
1124 1124 */
1125 1125 if (ndi_prop_update_string(DDI_DEV_T_NONE, cdip,
1126 1126 MDI_CLIENT_GUID_PROP, guid) != DDI_PROP_SUCCESS) {
1127 1127 goto fail;
1128 1128 }
1129 1129
1130 1130 /* Decorate the node with compatible property */
1131 1131 if (compatible &&
1132 1132 (ndi_prop_update_string_array(DDI_DEV_T_NONE, cdip,
1133 1133 "compatible", compatible, ncompatible) != DDI_PROP_SUCCESS)) {
1134 1134 goto fail;
1135 1135 }
1136 1136
1137 1137 return (cdip);
1138 1138
1139 1139 fail:
1140 1140 if (cdip) {
1141 1141 (void) ndi_prop_remove_all(cdip);
1142 1142 (void) ndi_devi_free(cdip);
1143 1143 }
1144 1144 return (NULL);
1145 1145 }
1146 1146
1147 1147 /*
1148 1148 * i_mdi_devinfo_find():
1149 1149 * Find a matching devinfo node for given client node name
1150 1150 * and its guid.
1151 1151 * Return Values:
1152 1152 * Handle to a dev_info node or NULL
1153 1153 */
1154 1154 static dev_info_t *
1155 1155 i_mdi_devinfo_find(mdi_vhci_t *vh, caddr_t name, char *guid)
1156 1156 {
1157 1157 char *data;
1158 1158 dev_info_t *cdip = NULL;
1159 1159 dev_info_t *ndip = NULL;
1160 1160 int circular;
1161 1161
1162 1162 ndi_devi_enter(vh->vh_dip, &circular);
1163 1163 ndip = (dev_info_t *)DEVI(vh->vh_dip)->devi_child;
1164 1164 while ((cdip = ndip) != NULL) {
1165 1165 ndip = (dev_info_t *)DEVI(cdip)->devi_sibling;
1166 1166
1167 1167 if (strcmp(DEVI(cdip)->devi_node_name, name)) {
1168 1168 continue;
1169 1169 }
1170 1170
1171 1171 if (ddi_prop_lookup_string(DDI_DEV_T_ANY, cdip,
1172 1172 DDI_PROP_DONTPASS, MDI_CLIENT_GUID_PROP,
1173 1173 &data) != DDI_PROP_SUCCESS) {
1174 1174 continue;
1175 1175 }
1176 1176
1177 1177 if (strcmp(data, guid) != 0) {
1178 1178 ddi_prop_free(data);
1179 1179 continue;
1180 1180 }
1181 1181 ddi_prop_free(data);
1182 1182 break;
1183 1183 }
1184 1184 ndi_devi_exit(vh->vh_dip, circular);
1185 1185 return (cdip);
1186 1186 }
1187 1187
1188 1188 /*
1189 1189 * i_mdi_devinfo_remove():
1190 1190 * Remove a client device node
1191 1191 */
1192 1192 static int
1193 1193 i_mdi_devinfo_remove(dev_info_t *vdip, dev_info_t *cdip, int flags)
1194 1194 {
1195 1195 int rv = MDI_SUCCESS;
1196 1196
1197 1197 if (i_mdi_is_child_present(vdip, cdip) == MDI_SUCCESS ||
1198 1198 (flags & MDI_CLIENT_FLAGS_DEV_NOT_SUPPORTED)) {
1199 1199 rv = ndi_devi_offline(cdip, NDI_DEVFS_CLEAN | NDI_DEVI_REMOVE);
1200 1200 if (rv != NDI_SUCCESS) {
1201 1201 MDI_DEBUG(1, (MDI_NOTE, cdip,
1202 1202 "!failed: cdip %p", (void *)cdip));
1203 1203 }
1204 1204 /*
1205 1205 * Convert to MDI error code
1206 1206 */
1207 1207 switch (rv) {
1208 1208 case NDI_SUCCESS:
1209 1209 rv = MDI_SUCCESS;
1210 1210 break;
1211 1211 case NDI_BUSY:
1212 1212 rv = MDI_BUSY;
1213 1213 break;
1214 1214 default:
1215 1215 rv = MDI_FAILURE;
1216 1216 break;
1217 1217 }
1218 1218 }
1219 1219 return (rv);
1220 1220 }
1221 1221
1222 1222 /*
1223 1223 * i_devi_get_client()
1224 1224 * Utility function to get mpxio component extensions
1225 1225 */
1226 1226 static mdi_client_t *
1227 1227 i_devi_get_client(dev_info_t *cdip)
1228 1228 {
1229 1229 mdi_client_t *ct = NULL;
1230 1230
1231 1231 if (MDI_CLIENT(cdip)) {
1232 1232 ct = (mdi_client_t *)DEVI(cdip)->devi_mdi_client;
1233 1233 }
1234 1234 return (ct);
1235 1235 }
1236 1236
1237 1237 /*
1238 1238 * i_mdi_is_child_present():
1239 1239 * Search for the presence of client device dev_info node
1240 1240 */
1241 1241 static int
1242 1242 i_mdi_is_child_present(dev_info_t *vdip, dev_info_t *cdip)
1243 1243 {
1244 1244 int rv = MDI_FAILURE;
1245 1245 struct dev_info *dip;
1246 1246 int circular;
1247 1247
1248 1248 ndi_devi_enter(vdip, &circular);
1249 1249 dip = DEVI(vdip)->devi_child;
1250 1250 while (dip) {
1251 1251 if (dip == DEVI(cdip)) {
1252 1252 rv = MDI_SUCCESS;
1253 1253 break;
1254 1254 }
1255 1255 dip = dip->devi_sibling;
1256 1256 }
1257 1257 ndi_devi_exit(vdip, circular);
1258 1258 return (rv);
1259 1259 }
1260 1260
1261 1261
1262 1262 /*
1263 1263 * i_mdi_client_lock():
1264 1264 * Grab client component lock
1265 1265 * Return Values:
1266 1266 * None
1267 1267 * Note:
1268 1268 * The default locking order is:
1269 1269 * _NOTE(LOCK_ORDER(mdi_client::ct_mutex mdi_pathinfo::pi_mutex))
1270 1270 * But there are number of situations where locks need to be
1271 1271 * grabbed in reverse order. This routine implements try and lock
1272 1272 * mechanism depending on the requested parameter option.
1273 1273 */
1274 1274 static void
1275 1275 i_mdi_client_lock(mdi_client_t *ct, mdi_pathinfo_t *pip)
1276 1276 {
1277 1277 if (pip) {
1278 1278 /*
1279 1279 * Reverse locking is requested.
1280 1280 */
1281 1281 while (MDI_CLIENT_TRYLOCK(ct) == 0) {
1282 1282 if (servicing_interrupt()) {
1283 1283 MDI_PI_HOLD(pip);
1284 1284 MDI_PI_UNLOCK(pip);
1285 1285 MDI_CLIENT_LOCK(ct);
1286 1286 MDI_PI_LOCK(pip);
1287 1287 MDI_PI_RELE(pip);
1288 1288 break;
1289 1289 } else {
1290 1290 /*
1291 1291 * tryenter failed. Try to grab again
1292 1292 * after a small delay
1293 1293 */
1294 1294 MDI_PI_HOLD(pip);
1295 1295 MDI_PI_UNLOCK(pip);
1296 1296 delay_random(mdi_delay);
1297 1297 MDI_PI_LOCK(pip);
1298 1298 MDI_PI_RELE(pip);
1299 1299 }
1300 1300 }
1301 1301 } else {
1302 1302 MDI_CLIENT_LOCK(ct);
1303 1303 }
1304 1304 }
1305 1305
1306 1306 /*
1307 1307 * i_mdi_client_unlock():
1308 1308 * Unlock a client component
1309 1309 */
1310 1310 static void
1311 1311 i_mdi_client_unlock(mdi_client_t *ct)
1312 1312 {
1313 1313 MDI_CLIENT_UNLOCK(ct);
1314 1314 }
1315 1315
1316 1316 /*
1317 1317 * i_mdi_client_alloc():
1318 1318 * Allocate and initialize a client structure. Caller should
1319 1319 * hold the vhci client lock.
1320 1320 * Return Values:
1321 1321 * Handle to a client component
1322 1322 */
1323 1323 /*ARGSUSED*/
1324 1324 static mdi_client_t *
1325 1325 i_mdi_client_alloc(mdi_vhci_t *vh, char *name, char *lguid)
1326 1326 {
1327 1327 mdi_client_t *ct;
1328 1328
1329 1329 ASSERT(MDI_VHCI_CLIENT_LOCKED(vh));
1330 1330
1331 1331 /*
1332 1332 * Allocate and initialize a component structure.
1333 1333 */
1334 1334 ct = kmem_zalloc(sizeof (*ct), KM_SLEEP);
1335 1335 mutex_init(&ct->ct_mutex, NULL, MUTEX_DEFAULT, NULL);
1336 1336 ct->ct_hnext = NULL;
1337 1337 ct->ct_hprev = NULL;
1338 1338 ct->ct_dip = NULL;
1339 1339 ct->ct_vhci = vh;
1340 1340 ct->ct_drvname = kmem_alloc(strlen(name) + 1, KM_SLEEP);
1341 1341 (void) strcpy(ct->ct_drvname, name);
1342 1342 ct->ct_guid = kmem_alloc(strlen(lguid) + 1, KM_SLEEP);
1343 1343 (void) strcpy(ct->ct_guid, lguid);
1344 1344 ct->ct_cprivate = NULL;
1345 1345 ct->ct_vprivate = NULL;
1346 1346 ct->ct_flags = 0;
1347 1347 ct->ct_state = MDI_CLIENT_STATE_FAILED;
1348 1348 MDI_CLIENT_LOCK(ct);
1349 1349 MDI_CLIENT_SET_OFFLINE(ct);
1350 1350 MDI_CLIENT_SET_DETACH(ct);
1351 1351 MDI_CLIENT_SET_POWER_UP(ct);
1352 1352 MDI_CLIENT_UNLOCK(ct);
1353 1353 ct->ct_failover_flags = 0;
1354 1354 ct->ct_failover_status = 0;
1355 1355 cv_init(&ct->ct_failover_cv, NULL, CV_DRIVER, NULL);
1356 1356 ct->ct_unstable = 0;
1357 1357 cv_init(&ct->ct_unstable_cv, NULL, CV_DRIVER, NULL);
1358 1358 cv_init(&ct->ct_powerchange_cv, NULL, CV_DRIVER, NULL);
1359 1359 ct->ct_lb = vh->vh_lb;
1360 1360 ct->ct_lb_args = kmem_zalloc(sizeof (client_lb_args_t), KM_SLEEP);
1361 1361 ct->ct_lb_args->region_size = LOAD_BALANCE_DEFAULT_REGION_SIZE;
1362 1362 ct->ct_path_count = 0;
1363 1363 ct->ct_path_head = NULL;
1364 1364 ct->ct_path_tail = NULL;
1365 1365 ct->ct_path_last = NULL;
1366 1366
1367 1367 /*
1368 1368 * Add this client component to our client hash queue
1369 1369 */
1370 1370 i_mdi_client_enlist_table(vh, ct);
1371 1371 return (ct);
1372 1372 }
1373 1373
1374 1374 /*
1375 1375 * i_mdi_client_enlist_table():
1376 1376 * Attach the client device to the client hash table. Caller
1377 1377 * should hold the vhci client lock.
1378 1378 */
1379 1379 static void
1380 1380 i_mdi_client_enlist_table(mdi_vhci_t *vh, mdi_client_t *ct)
1381 1381 {
1382 1382 int index;
1383 1383 struct client_hash *head;
1384 1384
1385 1385 ASSERT(MDI_VHCI_CLIENT_LOCKED(vh));
1386 1386
1387 1387 index = i_mdi_get_hash_key(ct->ct_guid);
1388 1388 head = &vh->vh_client_table[index];
1389 1389 ct->ct_hnext = (mdi_client_t *)head->ct_hash_head;
1390 1390 head->ct_hash_head = ct;
1391 1391 head->ct_hash_count++;
1392 1392 vh->vh_client_count++;
1393 1393 }
1394 1394
1395 1395 /*
1396 1396 * i_mdi_client_delist_table():
1397 1397 * Attach the client device to the client hash table.
1398 1398 * Caller should hold the vhci client lock.
1399 1399 */
1400 1400 static void
1401 1401 i_mdi_client_delist_table(mdi_vhci_t *vh, mdi_client_t *ct)
1402 1402 {
1403 1403 int index;
1404 1404 char *guid;
1405 1405 struct client_hash *head;
1406 1406 mdi_client_t *next;
1407 1407 mdi_client_t *last;
1408 1408
1409 1409 ASSERT(MDI_VHCI_CLIENT_LOCKED(vh));
1410 1410
1411 1411 guid = ct->ct_guid;
1412 1412 index = i_mdi_get_hash_key(guid);
1413 1413 head = &vh->vh_client_table[index];
1414 1414
1415 1415 last = NULL;
1416 1416 next = (mdi_client_t *)head->ct_hash_head;
1417 1417 while (next != NULL) {
1418 1418 if (next == ct) {
1419 1419 break;
1420 1420 }
1421 1421 last = next;
1422 1422 next = next->ct_hnext;
1423 1423 }
1424 1424
1425 1425 if (next) {
1426 1426 head->ct_hash_count--;
1427 1427 if (last == NULL) {
1428 1428 head->ct_hash_head = ct->ct_hnext;
1429 1429 } else {
1430 1430 last->ct_hnext = ct->ct_hnext;
1431 1431 }
1432 1432 ct->ct_hnext = NULL;
1433 1433 vh->vh_client_count--;
1434 1434 }
1435 1435 }
1436 1436
1437 1437
1438 1438 /*
1439 1439 * i_mdi_client_free():
1440 1440 * Free a client component
1441 1441 */
1442 1442 static int
1443 1443 i_mdi_client_free(mdi_vhci_t *vh, mdi_client_t *ct)
1444 1444 {
1445 1445 int rv = MDI_SUCCESS;
1446 1446 int flags = ct->ct_flags;
1447 1447 dev_info_t *cdip;
1448 1448 dev_info_t *vdip;
1449 1449
1450 1450 ASSERT(MDI_VHCI_CLIENT_LOCKED(vh));
1451 1451
1452 1452 vdip = vh->vh_dip;
1453 1453 cdip = ct->ct_dip;
1454 1454
1455 1455 (void) ndi_prop_remove(DDI_DEV_T_NONE, cdip, MDI_CLIENT_GUID_PROP);
1456 1456 DEVI(cdip)->devi_mdi_component &= ~MDI_COMPONENT_CLIENT;
1457 1457 DEVI(cdip)->devi_mdi_client = NULL;
1458 1458
1459 1459 /*
1460 1460 * Clear out back ref. to dev_info_t node
1461 1461 */
1462 1462 ct->ct_dip = NULL;
1463 1463
1464 1464 /*
1465 1465 * Remove this client from our hash queue
1466 1466 */
1467 1467 i_mdi_client_delist_table(vh, ct);
1468 1468
1469 1469 /*
1470 1470 * Uninitialize and free the component
1471 1471 */
1472 1472 kmem_free(ct->ct_drvname, strlen(ct->ct_drvname) + 1);
1473 1473 kmem_free(ct->ct_guid, strlen(ct->ct_guid) + 1);
1474 1474 kmem_free(ct->ct_lb_args, sizeof (client_lb_args_t));
1475 1475 cv_destroy(&ct->ct_failover_cv);
1476 1476 cv_destroy(&ct->ct_unstable_cv);
1477 1477 cv_destroy(&ct->ct_powerchange_cv);
1478 1478 mutex_destroy(&ct->ct_mutex);
1479 1479 kmem_free(ct, sizeof (*ct));
1480 1480
1481 1481 if (cdip != NULL) {
1482 1482 MDI_VHCI_CLIENT_UNLOCK(vh);
1483 1483 (void) i_mdi_devinfo_remove(vdip, cdip, flags);
1484 1484 MDI_VHCI_CLIENT_LOCK(vh);
1485 1485 }
1486 1486 return (rv);
1487 1487 }
1488 1488
1489 1489 /*
1490 1490 * i_mdi_client_find():
1491 1491 * Find the client structure corresponding to a given guid
1492 1492 * Caller should hold the vhci client lock.
1493 1493 */
1494 1494 static mdi_client_t *
1495 1495 i_mdi_client_find(mdi_vhci_t *vh, char *cname, char *guid)
1496 1496 {
1497 1497 int index;
1498 1498 struct client_hash *head;
1499 1499 mdi_client_t *ct;
1500 1500
1501 1501 ASSERT(MDI_VHCI_CLIENT_LOCKED(vh));
1502 1502
1503 1503 index = i_mdi_get_hash_key(guid);
1504 1504 head = &vh->vh_client_table[index];
1505 1505
1506 1506 ct = head->ct_hash_head;
1507 1507 while (ct != NULL) {
1508 1508 if (strcmp(ct->ct_guid, guid) == 0 &&
1509 1509 (cname == NULL || strcmp(ct->ct_drvname, cname) == 0)) {
1510 1510 break;
1511 1511 }
1512 1512 ct = ct->ct_hnext;
1513 1513 }
1514 1514 return (ct);
1515 1515 }
1516 1516
1517 1517 /*
1518 1518 * i_mdi_client_update_state():
1519 1519 * Compute and update client device state
1520 1520 * Notes:
1521 1521 * A client device can be in any of three possible states:
1522 1522 *
1523 1523 * MDI_CLIENT_STATE_OPTIMAL - Client in optimal state with more
1524 1524 * one online/standby paths. Can tolerate failures.
1525 1525 * MDI_CLIENT_STATE_DEGRADED - Client device in degraded state with
1526 1526 * no alternate paths available as standby. A failure on the online
1527 1527 * would result in loss of access to device data.
1528 1528 * MDI_CLIENT_STATE_FAILED - Client device in failed state with
1529 1529 * no paths available to access the device.
1530 1530 */
1531 1531 static void
1532 1532 i_mdi_client_update_state(mdi_client_t *ct)
1533 1533 {
1534 1534 int state;
1535 1535
1536 1536 ASSERT(MDI_CLIENT_LOCKED(ct));
1537 1537 state = i_mdi_client_compute_state(ct, NULL);
1538 1538 MDI_CLIENT_SET_STATE(ct, state);
1539 1539 }
1540 1540
1541 1541 /*
1542 1542 * i_mdi_client_compute_state():
1543 1543 * Compute client device state
1544 1544 *
1545 1545 * mdi_phci_t * Pointer to pHCI structure which should
1546 1546 * while computing the new value. Used by
1547 1547 * i_mdi_phci_offline() to find the new
1548 1548 * client state after DR of a pHCI.
1549 1549 */
1550 1550 static int
1551 1551 i_mdi_client_compute_state(mdi_client_t *ct, mdi_phci_t *ph)
1552 1552 {
1553 1553 int state;
1554 1554 int online_count = 0;
1555 1555 int standby_count = 0;
1556 1556 mdi_pathinfo_t *pip, *next;
1557 1557
1558 1558 ASSERT(MDI_CLIENT_LOCKED(ct));
1559 1559 pip = ct->ct_path_head;
1560 1560 while (pip != NULL) {
1561 1561 MDI_PI_LOCK(pip);
1562 1562 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
1563 1563 if (MDI_PI(pip)->pi_phci == ph) {
1564 1564 MDI_PI_UNLOCK(pip);
1565 1565 pip = next;
1566 1566 continue;
1567 1567 }
1568 1568
1569 1569 if ((MDI_PI(pip)->pi_state & MDI_PATHINFO_STATE_MASK)
1570 1570 == MDI_PATHINFO_STATE_ONLINE)
1571 1571 online_count++;
1572 1572 else if ((MDI_PI(pip)->pi_state & MDI_PATHINFO_STATE_MASK)
1573 1573 == MDI_PATHINFO_STATE_STANDBY)
1574 1574 standby_count++;
1575 1575 MDI_PI_UNLOCK(pip);
1576 1576 pip = next;
1577 1577 }
1578 1578
1579 1579 if (online_count == 0) {
1580 1580 if (standby_count == 0) {
1581 1581 state = MDI_CLIENT_STATE_FAILED;
1582 1582 MDI_DEBUG(2, (MDI_NOTE, ct->ct_dip,
1583 1583 "client state failed: ct = %p", (void *)ct));
1584 1584 } else if (standby_count == 1) {
1585 1585 state = MDI_CLIENT_STATE_DEGRADED;
1586 1586 } else {
1587 1587 state = MDI_CLIENT_STATE_OPTIMAL;
1588 1588 }
1589 1589 } else if (online_count == 1) {
1590 1590 if (standby_count == 0) {
1591 1591 state = MDI_CLIENT_STATE_DEGRADED;
1592 1592 } else {
1593 1593 state = MDI_CLIENT_STATE_OPTIMAL;
1594 1594 }
1595 1595 } else {
1596 1596 state = MDI_CLIENT_STATE_OPTIMAL;
1597 1597 }
1598 1598 return (state);
1599 1599 }
1600 1600
1601 1601 /*
1602 1602 * i_mdi_client2devinfo():
1603 1603 * Utility function
1604 1604 */
1605 1605 dev_info_t *
1606 1606 i_mdi_client2devinfo(mdi_client_t *ct)
1607 1607 {
1608 1608 return (ct->ct_dip);
1609 1609 }
1610 1610
1611 1611 /*
1612 1612 * mdi_client_path2_devinfo():
1613 1613 * Given the parent devinfo and child devfs pathname, search for
1614 1614 * a valid devfs node handle.
1615 1615 */
1616 1616 dev_info_t *
1617 1617 mdi_client_path2devinfo(dev_info_t *vdip, char *pathname)
1618 1618 {
1619 1619 dev_info_t *cdip = NULL;
1620 1620 dev_info_t *ndip = NULL;
1621 1621 char *temp_pathname;
1622 1622 int circular;
1623 1623
1624 1624 /*
1625 1625 * Allocate temp buffer
1626 1626 */
1627 1627 temp_pathname = kmem_zalloc(MAXPATHLEN, KM_SLEEP);
1628 1628
1629 1629 /*
1630 1630 * Lock parent against changes
1631 1631 */
1632 1632 ndi_devi_enter(vdip, &circular);
1633 1633 ndip = (dev_info_t *)DEVI(vdip)->devi_child;
1634 1634 while ((cdip = ndip) != NULL) {
1635 1635 ndip = (dev_info_t *)DEVI(cdip)->devi_sibling;
1636 1636
1637 1637 *temp_pathname = '\0';
1638 1638 (void) ddi_pathname(cdip, temp_pathname);
1639 1639 if (strcmp(temp_pathname, pathname) == 0) {
1640 1640 break;
1641 1641 }
1642 1642 }
1643 1643 /*
1644 1644 * Release devinfo lock
1645 1645 */
1646 1646 ndi_devi_exit(vdip, circular);
1647 1647
1648 1648 /*
1649 1649 * Free the temp buffer
1650 1650 */
1651 1651 kmem_free(temp_pathname, MAXPATHLEN);
1652 1652 return (cdip);
1653 1653 }
1654 1654
1655 1655 /*
1656 1656 * mdi_client_get_path_count():
1657 1657 * Utility function to get number of path information nodes
1658 1658 * associated with a given client device.
1659 1659 */
1660 1660 int
1661 1661 mdi_client_get_path_count(dev_info_t *cdip)
1662 1662 {
1663 1663 mdi_client_t *ct;
1664 1664 int count = 0;
1665 1665
1666 1666 ct = i_devi_get_client(cdip);
1667 1667 if (ct != NULL) {
1668 1668 count = ct->ct_path_count;
1669 1669 }
1670 1670 return (count);
1671 1671 }
1672 1672
1673 1673
1674 1674 /*
1675 1675 * i_mdi_get_hash_key():
1676 1676 * Create a hash using strings as keys
1677 1677 *
1678 1678 */
1679 1679 static int
1680 1680 i_mdi_get_hash_key(char *str)
1681 1681 {
1682 1682 uint32_t g, hash = 0;
1683 1683 char *p;
1684 1684
1685 1685 for (p = str; *p != '\0'; p++) {
1686 1686 g = *p;
1687 1687 hash += g;
1688 1688 }
1689 1689 return (hash % (CLIENT_HASH_TABLE_SIZE - 1));
1690 1690 }
1691 1691
1692 1692 /*
1693 1693 * mdi_get_lb_policy():
1694 1694 * Get current load balancing policy for a given client device
1695 1695 */
1696 1696 client_lb_t
1697 1697 mdi_get_lb_policy(dev_info_t *cdip)
1698 1698 {
1699 1699 client_lb_t lb = LOAD_BALANCE_NONE;
1700 1700 mdi_client_t *ct;
1701 1701
1702 1702 ct = i_devi_get_client(cdip);
1703 1703 if (ct != NULL) {
1704 1704 lb = ct->ct_lb;
1705 1705 }
1706 1706 return (lb);
1707 1707 }
1708 1708
1709 1709 /*
1710 1710 * mdi_set_lb_region_size():
1711 1711 * Set current region size for the load-balance
1712 1712 */
1713 1713 int
1714 1714 mdi_set_lb_region_size(dev_info_t *cdip, int region_size)
1715 1715 {
1716 1716 mdi_client_t *ct;
1717 1717 int rv = MDI_FAILURE;
1718 1718
1719 1719 ct = i_devi_get_client(cdip);
1720 1720 if (ct != NULL && ct->ct_lb_args != NULL) {
1721 1721 ct->ct_lb_args->region_size = region_size;
1722 1722 rv = MDI_SUCCESS;
1723 1723 }
1724 1724 return (rv);
1725 1725 }
1726 1726
1727 1727 /*
1728 1728 * mdi_Set_lb_policy():
1729 1729 * Set current load balancing policy for a given client device
1730 1730 */
1731 1731 int
1732 1732 mdi_set_lb_policy(dev_info_t *cdip, client_lb_t lb)
1733 1733 {
1734 1734 mdi_client_t *ct;
1735 1735 int rv = MDI_FAILURE;
1736 1736
1737 1737 ct = i_devi_get_client(cdip);
1738 1738 if (ct != NULL) {
1739 1739 ct->ct_lb = lb;
1740 1740 rv = MDI_SUCCESS;
1741 1741 }
1742 1742 return (rv);
1743 1743 }
1744 1744
1745 1745 /*
1746 1746 * mdi_failover():
1747 1747 * failover function called by the vHCI drivers to initiate
1748 1748 * a failover operation. This is typically due to non-availability
1749 1749 * of online paths to route I/O requests. Failover can be
1750 1750 * triggered through user application also.
1751 1751 *
1752 1752 * The vHCI driver calls mdi_failover() to initiate a failover
1753 1753 * operation. mdi_failover() calls back into the vHCI driver's
1754 1754 * vo_failover() entry point to perform the actual failover
1755 1755 * operation. The reason for requiring the vHCI driver to
1756 1756 * initiate failover by calling mdi_failover(), instead of directly
1757 1757 * executing vo_failover() itself, is to ensure that the mdi
1758 1758 * framework can keep track of the client state properly.
1759 1759 * Additionally, mdi_failover() provides as a convenience the
1760 1760 * option of performing the failover operation synchronously or
1761 1761 * asynchronously
1762 1762 *
1763 1763 * Upon successful completion of the failover operation, the
1764 1764 * paths that were previously ONLINE will be in the STANDBY state,
1765 1765 * and the newly activated paths will be in the ONLINE state.
1766 1766 *
1767 1767 * The flags modifier determines whether the activation is done
1768 1768 * synchronously: MDI_FAILOVER_SYNC
1769 1769 * Return Values:
1770 1770 * MDI_SUCCESS
1771 1771 * MDI_FAILURE
1772 1772 * MDI_BUSY
1773 1773 */
1774 1774 /*ARGSUSED*/
1775 1775 int
1776 1776 mdi_failover(dev_info_t *vdip, dev_info_t *cdip, int flags)
1777 1777 {
1778 1778 int rv;
1779 1779 mdi_client_t *ct;
1780 1780
1781 1781 ct = i_devi_get_client(cdip);
1782 1782 ASSERT(ct != NULL);
1783 1783 if (ct == NULL) {
1784 1784 /* cdip is not a valid client device. Nothing more to do. */
1785 1785 return (MDI_FAILURE);
1786 1786 }
1787 1787
1788 1788 MDI_CLIENT_LOCK(ct);
1789 1789
1790 1790 if (MDI_CLIENT_IS_PATH_FREE_IN_PROGRESS(ct)) {
1791 1791 /* A path to the client is being freed */
1792 1792 MDI_CLIENT_UNLOCK(ct);
1793 1793 return (MDI_BUSY);
1794 1794 }
1795 1795
1796 1796
1797 1797 if (MDI_CLIENT_IS_FAILED(ct)) {
1798 1798 /*
1799 1799 * Client is in failed state. Nothing more to do.
1800 1800 */
1801 1801 MDI_CLIENT_UNLOCK(ct);
1802 1802 return (MDI_FAILURE);
1803 1803 }
1804 1804
1805 1805 if (MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct)) {
1806 1806 /*
1807 1807 * Failover is already in progress; return BUSY
1808 1808 */
1809 1809 MDI_CLIENT_UNLOCK(ct);
1810 1810 return (MDI_BUSY);
1811 1811 }
1812 1812 /*
1813 1813 * Make sure that mdi_pathinfo node state changes are processed.
1814 1814 * We do not allow failovers to progress while client path state
1815 1815 * changes are in progress
1816 1816 */
1817 1817 if (ct->ct_unstable) {
1818 1818 if (flags == MDI_FAILOVER_ASYNC) {
1819 1819 MDI_CLIENT_UNLOCK(ct);
1820 1820 return (MDI_BUSY);
1821 1821 } else {
1822 1822 while (ct->ct_unstable)
1823 1823 cv_wait(&ct->ct_unstable_cv, &ct->ct_mutex);
1824 1824 }
1825 1825 }
1826 1826
1827 1827 /*
1828 1828 * Client device is in stable state. Before proceeding, perform sanity
1829 1829 * checks again.
1830 1830 */
1831 1831 if ((MDI_CLIENT_IS_DETACHED(ct)) || (MDI_CLIENT_IS_FAILED(ct)) ||
1832 1832 (!i_ddi_devi_attached(cdip))) {
1833 1833 /*
1834 1834 * Client is in failed state. Nothing more to do.
1835 1835 */
1836 1836 MDI_CLIENT_UNLOCK(ct);
1837 1837 return (MDI_FAILURE);
1838 1838 }
1839 1839
1840 1840 /*
1841 1841 * Set the client state as failover in progress.
1842 1842 */
1843 1843 MDI_CLIENT_SET_FAILOVER_IN_PROGRESS(ct);
1844 1844 ct->ct_failover_flags = flags;
1845 1845 MDI_CLIENT_UNLOCK(ct);
1846 1846
1847 1847 if (flags == MDI_FAILOVER_ASYNC) {
1848 1848 /*
1849 1849 * Submit the initiate failover request via CPR safe
1850 1850 * taskq threads.
1851 1851 */
1852 1852 (void) taskq_dispatch(mdi_taskq, (task_func_t *)i_mdi_failover,
1853 1853 ct, KM_SLEEP);
1854 1854 return (MDI_ACCEPT);
1855 1855 } else {
1856 1856 /*
1857 1857 * Synchronous failover mode. Typically invoked from the user
1858 1858 * land.
1859 1859 */
1860 1860 rv = i_mdi_failover(ct);
1861 1861 }
1862 1862 return (rv);
1863 1863 }
1864 1864
1865 1865 /*
1866 1866 * i_mdi_failover():
1867 1867 * internal failover function. Invokes vHCI drivers failover
1868 1868 * callback function and process the failover status
1869 1869 * Return Values:
1870 1870 * None
1871 1871 *
1872 1872 * Note: A client device in failover state can not be detached or freed.
1873 1873 */
1874 1874 static int
1875 1875 i_mdi_failover(void *arg)
1876 1876 {
1877 1877 int rv = MDI_SUCCESS;
1878 1878 mdi_client_t *ct = (mdi_client_t *)arg;
1879 1879 mdi_vhci_t *vh = ct->ct_vhci;
1880 1880
1881 1881 ASSERT(!MDI_CLIENT_LOCKED(ct));
1882 1882
1883 1883 if (vh->vh_ops->vo_failover != NULL) {
1884 1884 /*
1885 1885 * Call vHCI drivers callback routine
1886 1886 */
1887 1887 rv = (*vh->vh_ops->vo_failover)(vh->vh_dip, ct->ct_dip,
1888 1888 ct->ct_failover_flags);
1889 1889 }
1890 1890
1891 1891 MDI_CLIENT_LOCK(ct);
1892 1892 MDI_CLIENT_CLEAR_FAILOVER_IN_PROGRESS(ct);
1893 1893
1894 1894 /*
1895 1895 * Save the failover return status
1896 1896 */
1897 1897 ct->ct_failover_status = rv;
1898 1898
1899 1899 /*
1900 1900 * As a result of failover, client status would have been changed.
1901 1901 * Update the client state and wake up anyone waiting on this client
1902 1902 * device.
1903 1903 */
1904 1904 i_mdi_client_update_state(ct);
1905 1905
1906 1906 cv_broadcast(&ct->ct_failover_cv);
1907 1907 MDI_CLIENT_UNLOCK(ct);
1908 1908 return (rv);
1909 1909 }
1910 1910
1911 1911 /*
1912 1912 * Load balancing is logical block.
1913 1913 * IOs within the range described by region_size
1914 1914 * would go on the same path. This would improve the
1915 1915 * performance by cache-hit on some of the RAID devices.
1916 1916 * Search only for online paths(At some point we
1917 1917 * may want to balance across target ports).
1918 1918 * If no paths are found then default to round-robin.
1919 1919 */
1920 1920 static int
1921 1921 i_mdi_lba_lb(mdi_client_t *ct, mdi_pathinfo_t **ret_pip, struct buf *bp)
1922 1922 {
1923 1923 int path_index = -1;
1924 1924 int online_path_count = 0;
1925 1925 int online_nonpref_path_count = 0;
1926 1926 int region_size = ct->ct_lb_args->region_size;
1927 1927 mdi_pathinfo_t *pip;
1928 1928 mdi_pathinfo_t *next;
1929 1929 int preferred, path_cnt;
1930 1930
1931 1931 pip = ct->ct_path_head;
1932 1932 while (pip) {
1933 1933 MDI_PI_LOCK(pip);
1934 1934 if (MDI_PI(pip)->pi_state ==
1935 1935 MDI_PATHINFO_STATE_ONLINE && MDI_PI(pip)->pi_preferred) {
1936 1936 online_path_count++;
1937 1937 } else if (MDI_PI(pip)->pi_state ==
1938 1938 MDI_PATHINFO_STATE_ONLINE && !MDI_PI(pip)->pi_preferred) {
1939 1939 online_nonpref_path_count++;
1940 1940 }
1941 1941 next = (mdi_pathinfo_t *)
1942 1942 MDI_PI(pip)->pi_client_link;
1943 1943 MDI_PI_UNLOCK(pip);
1944 1944 pip = next;
1945 1945 }
1946 1946 /* if found any online/preferred then use this type */
1947 1947 if (online_path_count > 0) {
1948 1948 path_cnt = online_path_count;
1949 1949 preferred = 1;
1950 1950 } else if (online_nonpref_path_count > 0) {
1951 1951 path_cnt = online_nonpref_path_count;
1952 1952 preferred = 0;
1953 1953 } else {
1954 1954 path_cnt = 0;
1955 1955 }
1956 1956 if (path_cnt) {
1957 1957 path_index = (bp->b_blkno >> region_size) % path_cnt;
1958 1958 pip = ct->ct_path_head;
1959 1959 while (pip && path_index != -1) {
1960 1960 MDI_PI_LOCK(pip);
1961 1961 if (path_index == 0 &&
1962 1962 (MDI_PI(pip)->pi_state ==
1963 1963 MDI_PATHINFO_STATE_ONLINE) &&
1964 1964 MDI_PI(pip)->pi_preferred == preferred) {
1965 1965 MDI_PI_HOLD(pip);
1966 1966 MDI_PI_UNLOCK(pip);
1967 1967 *ret_pip = pip;
1968 1968 return (MDI_SUCCESS);
1969 1969 }
1970 1970 path_index --;
1971 1971 next = (mdi_pathinfo_t *)
1972 1972 MDI_PI(pip)->pi_client_link;
1973 1973 MDI_PI_UNLOCK(pip);
1974 1974 pip = next;
1975 1975 }
1976 1976 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
1977 1977 "lba %llx: path %s %p",
1978 1978 bp->b_lblkno, mdi_pi_spathname(pip), (void *)pip));
1979 1979 }
1980 1980 return (MDI_FAILURE);
1981 1981 }
1982 1982
1983 1983 /*
1984 1984 * mdi_select_path():
1985 1985 * select a path to access a client device.
1986 1986 *
1987 1987 * mdi_select_path() function is called by the vHCI drivers to
1988 1988 * select a path to route the I/O request to. The caller passes
1989 1989 * the block I/O data transfer structure ("buf") as one of the
1990 1990 * parameters. The mpxio framework uses the buf structure
1991 1991 * contents to maintain per path statistics (total I/O size /
1992 1992 * count pending). If more than one online paths are available to
1993 1993 * select, the framework automatically selects a suitable path
1994 1994 * for routing I/O request. If a failover operation is active for
1995 1995 * this client device the call shall be failed with MDI_BUSY error
1996 1996 * code.
1997 1997 *
1998 1998 * By default this function returns a suitable path in online
1999 1999 * state based on the current load balancing policy. Currently
2000 2000 * we support LOAD_BALANCE_NONE (Previously selected online path
2001 2001 * will continue to be used till the path is usable) and
2002 2002 * LOAD_BALANCE_RR (Online paths will be selected in a round
2003 2003 * robin fashion), LOAD_BALANCE_LB(Online paths will be selected
2004 2004 * based on the logical block). The load balancing
2005 2005 * through vHCI drivers configuration file (driver.conf).
2006 2006 *
2007 2007 * vHCI drivers may override this default behavior by specifying
2008 2008 * appropriate flags. The meaning of the thrid argument depends
2009 2009 * on the flags specified. If MDI_SELECT_PATH_INSTANCE is set
2010 2010 * then the argument is the "path instance" of the path to select.
2011 2011 * If MDI_SELECT_PATH_INSTANCE is not set then the argument is
2012 2012 * "start_pip". A non NULL "start_pip" is the starting point to
2013 2013 * walk and find the next appropriate path. The following values
2014 2014 * are currently defined: MDI_SELECT_ONLINE_PATH (to select an
2015 2015 * ONLINE path) and/or MDI_SELECT_STANDBY_PATH (to select an
2016 2016 * STANDBY path).
2017 2017 *
2018 2018 * The non-standard behavior is used by the scsi_vhci driver,
2019 2019 * whenever it has to use a STANDBY/FAULTED path. Eg. during
2020 2020 * attach of client devices (to avoid an unnecessary failover
2021 2021 * when the STANDBY path comes up first), during failover
2022 2022 * (to activate a STANDBY path as ONLINE).
2023 2023 *
2024 2024 * The selected path is returned in a a mdi_hold_path() state
2025 2025 * (pi_ref_cnt). Caller should release the hold by calling
2026 2026 * mdi_rele_path().
2027 2027 *
2028 2028 * Return Values:
2029 2029 * MDI_SUCCESS - Completed successfully
2030 2030 * MDI_BUSY - Client device is busy failing over
2031 2031 * MDI_NOPATH - Client device is online, but no valid path are
2032 2032 * available to access this client device
2033 2033 * MDI_FAILURE - Invalid client device or state
2034 2034 * MDI_DEVI_ONLINING
2035 2035 * - Client device (struct dev_info state) is in
2036 2036 * onlining state.
2037 2037 */
2038 2038
2039 2039 /*ARGSUSED*/
2040 2040 int
2041 2041 mdi_select_path(dev_info_t *cdip, struct buf *bp, int flags,
2042 2042 void *arg, mdi_pathinfo_t **ret_pip)
2043 2043 {
2044 2044 mdi_client_t *ct;
2045 2045 mdi_pathinfo_t *pip;
2046 2046 mdi_pathinfo_t *next;
2047 2047 mdi_pathinfo_t *head;
2048 2048 mdi_pathinfo_t *start;
2049 2049 client_lb_t lbp; /* load balancing policy */
2050 2050 int sb = 1; /* standard behavior */
2051 2051 int preferred = 1; /* preferred path */
2052 2052 int cond, cont = 1;
2053 2053 int retry = 0;
2054 2054 mdi_pathinfo_t *start_pip; /* request starting pathinfo */
2055 2055 int path_instance; /* request specific path instance */
2056 2056
2057 2057 /* determine type of arg based on flags */
2058 2058 if (flags & MDI_SELECT_PATH_INSTANCE) {
2059 2059 path_instance = (int)(intptr_t)arg;
2060 2060 start_pip = NULL;
2061 2061 } else {
2062 2062 path_instance = 0;
2063 2063 start_pip = (mdi_pathinfo_t *)arg;
2064 2064 }
2065 2065
2066 2066 if (flags != 0) {
2067 2067 /*
2068 2068 * disable default behavior
2069 2069 */
2070 2070 sb = 0;
2071 2071 }
2072 2072
2073 2073 *ret_pip = NULL;
2074 2074 ct = i_devi_get_client(cdip);
2075 2075 if (ct == NULL) {
2076 2076 /* mdi extensions are NULL, Nothing more to do */
2077 2077 return (MDI_FAILURE);
2078 2078 }
2079 2079
2080 2080 MDI_CLIENT_LOCK(ct);
2081 2081
2082 2082 if (sb) {
2083 2083 if (MDI_CLIENT_IS_FAILED(ct)) {
2084 2084 /*
2085 2085 * Client is not ready to accept any I/O requests.
2086 2086 * Fail this request.
2087 2087 */
2088 2088 MDI_DEBUG(2, (MDI_NOTE, cdip,
2089 2089 "client state offline ct = %p", (void *)ct));
2090 2090 MDI_CLIENT_UNLOCK(ct);
2091 2091 return (MDI_FAILURE);
2092 2092 }
2093 2093
2094 2094 if (MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct)) {
2095 2095 /*
2096 2096 * Check for Failover is in progress. If so tell the
2097 2097 * caller that this device is busy.
2098 2098 */
2099 2099 MDI_DEBUG(2, (MDI_NOTE, cdip,
2100 2100 "client failover in progress ct = %p",
2101 2101 (void *)ct));
2102 2102 MDI_CLIENT_UNLOCK(ct);
2103 2103 return (MDI_BUSY);
2104 2104 }
2105 2105
2106 2106 /*
2107 2107 * Check to see whether the client device is attached.
2108 2108 * If not so, let the vHCI driver manually select a path
2109 2109 * (standby) and let the probe/attach process to continue.
2110 2110 */
2111 2111 if (MDI_CLIENT_IS_DETACHED(ct) || !i_ddi_devi_attached(cdip)) {
2112 2112 MDI_DEBUG(4, (MDI_NOTE, cdip,
2113 2113 "devi is onlining ct = %p", (void *)ct));
2114 2114 MDI_CLIENT_UNLOCK(ct);
2115 2115 return (MDI_DEVI_ONLINING);
2116 2116 }
2117 2117 }
2118 2118
2119 2119 /*
2120 2120 * Cache in the client list head. If head of the list is NULL
2121 2121 * return MDI_NOPATH
2122 2122 */
2123 2123 head = ct->ct_path_head;
2124 2124 if (head == NULL) {
2125 2125 MDI_CLIENT_UNLOCK(ct);
2126 2126 return (MDI_NOPATH);
2127 2127 }
2128 2128
2129 2129 /* Caller is specifying a specific pathinfo path by path_instance */
2130 2130 if (path_instance) {
2131 2131 /* search for pathinfo with correct path_instance */
2132 2132 for (pip = head;
2133 2133 pip && (mdi_pi_get_path_instance(pip) != path_instance);
2134 2134 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link)
2135 2135 ;
2136 2136
2137 2137 /* If path can't be selected then MDI_NOPATH is returned. */
2138 2138 if (pip == NULL) {
2139 2139 MDI_CLIENT_UNLOCK(ct);
2140 2140 return (MDI_NOPATH);
2141 2141 }
2142 2142
2143 2143 /*
2144 2144 * Verify state of path. When asked to select a specific
2145 2145 * path_instance, we select the requested path in any
2146 2146 * state (ONLINE, OFFLINE, STANDBY, FAULT) other than INIT.
2147 2147 * We don't however select paths where the pHCI has detached.
2148 2148 * NOTE: last pathinfo node of an opened client device may
2149 2149 * exist in an OFFLINE state after the pHCI associated with
2150 2150 * that path has detached (but pi_phci will be NULL if that
2151 2151 * has occurred).
2152 2152 */
2153 2153 MDI_PI_LOCK(pip);
2154 2154 if ((MDI_PI(pip)->pi_state == MDI_PATHINFO_STATE_INIT) ||
2155 2155 (MDI_PI(pip)->pi_phci == NULL)) {
2156 2156 MDI_PI_UNLOCK(pip);
2157 2157 MDI_CLIENT_UNLOCK(ct);
2158 2158 return (MDI_FAILURE);
2159 2159 }
2160 2160
2161 2161 /* Return MDI_BUSY if we have a transient condition */
2162 2162 if (MDI_PI_IS_TRANSIENT(pip)) {
2163 2163 MDI_PI_UNLOCK(pip);
2164 2164 MDI_CLIENT_UNLOCK(ct);
2165 2165 return (MDI_BUSY);
2166 2166 }
2167 2167
2168 2168 /*
2169 2169 * Return the path in hold state. Caller should release the
2170 2170 * lock by calling mdi_rele_path()
2171 2171 */
2172 2172 MDI_PI_HOLD(pip);
2173 2173 MDI_PI_UNLOCK(pip);
2174 2174 *ret_pip = pip;
2175 2175 MDI_CLIENT_UNLOCK(ct);
2176 2176 return (MDI_SUCCESS);
2177 2177 }
2178 2178
2179 2179 /*
2180 2180 * for non default behavior, bypass current
2181 2181 * load balancing policy and always use LOAD_BALANCE_RR
2182 2182 * except that the start point will be adjusted based
2183 2183 * on the provided start_pip
2184 2184 */
2185 2185 lbp = sb ? ct->ct_lb : LOAD_BALANCE_RR;
2186 2186
2187 2187 switch (lbp) {
2188 2188 case LOAD_BALANCE_NONE:
2189 2189 /*
2190 2190 * Load balancing is None or Alternate path mode
2191 2191 * Start looking for a online mdi_pathinfo node starting from
2192 2192 * last known selected path
2193 2193 */
2194 2194 preferred = 1;
2195 2195 pip = (mdi_pathinfo_t *)ct->ct_path_last;
2196 2196 if (pip == NULL) {
2197 2197 pip = head;
2198 2198 }
2199 2199 start = pip;
2200 2200 do {
2201 2201 MDI_PI_LOCK(pip);
2202 2202 /*
2203 2203 * No need to explicitly check if the path is disabled.
2204 2204 * Since we are checking for state == ONLINE and the
2205 2205 * same variable is used for DISABLE/ENABLE information.
2206 2206 */
2207 2207 if ((MDI_PI(pip)->pi_state ==
2208 2208 MDI_PATHINFO_STATE_ONLINE) &&
2209 2209 preferred == MDI_PI(pip)->pi_preferred) {
2210 2210 /*
2211 2211 * Return the path in hold state. Caller should
2212 2212 * release the lock by calling mdi_rele_path()
2213 2213 */
2214 2214 MDI_PI_HOLD(pip);
2215 2215 MDI_PI_UNLOCK(pip);
2216 2216 ct->ct_path_last = pip;
2217 2217 *ret_pip = pip;
2218 2218 MDI_CLIENT_UNLOCK(ct);
2219 2219 return (MDI_SUCCESS);
2220 2220 }
2221 2221
2222 2222 /*
2223 2223 * Path is busy.
2224 2224 */
2225 2225 if (MDI_PI_IS_DRV_DISABLE_TRANSIENT(pip) ||
2226 2226 MDI_PI_IS_TRANSIENT(pip))
2227 2227 retry = 1;
2228 2228 /*
2229 2229 * Keep looking for a next available online path
2230 2230 */
2231 2231 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
2232 2232 if (next == NULL) {
2233 2233 next = head;
2234 2234 }
2235 2235 MDI_PI_UNLOCK(pip);
2236 2236 pip = next;
2237 2237 if (start == pip && preferred) {
2238 2238 preferred = 0;
2239 2239 } else if (start == pip && !preferred) {
2240 2240 cont = 0;
2241 2241 }
2242 2242 } while (cont);
2243 2243 break;
2244 2244
2245 2245 case LOAD_BALANCE_LBA:
2246 2246 /*
2247 2247 * Make sure we are looking
2248 2248 * for an online path. Otherwise, if it is for a STANDBY
2249 2249 * path request, it will go through and fetch an ONLINE
2250 2250 * path which is not desirable.
2251 2251 */
2252 2252 if ((ct->ct_lb_args != NULL) &&
2253 2253 (ct->ct_lb_args->region_size) && bp &&
2254 2254 (sb || (flags == MDI_SELECT_ONLINE_PATH))) {
2255 2255 if (i_mdi_lba_lb(ct, ret_pip, bp)
2256 2256 == MDI_SUCCESS) {
2257 2257 MDI_CLIENT_UNLOCK(ct);
2258 2258 return (MDI_SUCCESS);
2259 2259 }
2260 2260 }
2261 2261 /* FALLTHROUGH */
2262 2262 case LOAD_BALANCE_RR:
2263 2263 /*
2264 2264 * Load balancing is Round Robin. Start looking for a online
2265 2265 * mdi_pathinfo node starting from last known selected path
2266 2266 * as the start point. If override flags are specified,
2267 2267 * process accordingly.
2268 2268 * If the search is already in effect(start_pip not null),
2269 2269 * then lets just use the same path preference to continue the
2270 2270 * traversal.
2271 2271 */
2272 2272
2273 2273 if (start_pip != NULL) {
2274 2274 preferred = MDI_PI(start_pip)->pi_preferred;
2275 2275 } else {
2276 2276 preferred = 1;
2277 2277 }
2278 2278
2279 2279 start = sb ? (mdi_pathinfo_t *)ct->ct_path_last : start_pip;
2280 2280 if (start == NULL) {
2281 2281 pip = head;
2282 2282 } else {
2283 2283 pip = (mdi_pathinfo_t *)MDI_PI(start)->pi_client_link;
2284 2284 if (pip == NULL) {
2285 2285 if ( flags & MDI_SELECT_NO_PREFERRED) {
2286 2286 /*
2287 2287 * Return since we hit the end of list
2288 2288 */
2289 2289 MDI_CLIENT_UNLOCK(ct);
2290 2290 return (MDI_NOPATH);
2291 2291 }
2292 2292
2293 2293 if (!sb) {
2294 2294 if (preferred == 0) {
2295 2295 /*
2296 2296 * Looks like we have completed
2297 2297 * the traversal as preferred
2298 2298 * value is 0. Time to bail out.
2299 2299 */
2300 2300 *ret_pip = NULL;
2301 2301 MDI_CLIENT_UNLOCK(ct);
2302 2302 return (MDI_NOPATH);
2303 2303 } else {
2304 2304 /*
2305 2305 * Looks like we reached the
2306 2306 * end of the list. Lets enable
2307 2307 * traversal of non preferred
2308 2308 * paths.
2309 2309 */
2310 2310 preferred = 0;
2311 2311 }
2312 2312 }
2313 2313 pip = head;
2314 2314 }
2315 2315 }
2316 2316 start = pip;
2317 2317 do {
2318 2318 MDI_PI_LOCK(pip);
2319 2319 if (sb) {
2320 2320 cond = ((MDI_PI(pip)->pi_state ==
2321 2321 MDI_PATHINFO_STATE_ONLINE &&
2322 2322 MDI_PI(pip)->pi_preferred ==
2323 2323 preferred) ? 1 : 0);
2324 2324 } else {
2325 2325 if (flags == MDI_SELECT_ONLINE_PATH) {
2326 2326 cond = ((MDI_PI(pip)->pi_state ==
2327 2327 MDI_PATHINFO_STATE_ONLINE &&
2328 2328 MDI_PI(pip)->pi_preferred ==
2329 2329 preferred) ? 1 : 0);
2330 2330 } else if (flags == MDI_SELECT_STANDBY_PATH) {
2331 2331 cond = ((MDI_PI(pip)->pi_state ==
2332 2332 MDI_PATHINFO_STATE_STANDBY &&
2333 2333 MDI_PI(pip)->pi_preferred ==
2334 2334 preferred) ? 1 : 0);
2335 2335 } else if (flags == (MDI_SELECT_ONLINE_PATH |
2336 2336 MDI_SELECT_STANDBY_PATH)) {
2337 2337 cond = (((MDI_PI(pip)->pi_state ==
2338 2338 MDI_PATHINFO_STATE_ONLINE ||
2339 2339 (MDI_PI(pip)->pi_state ==
2340 2340 MDI_PATHINFO_STATE_STANDBY)) &&
2341 2341 MDI_PI(pip)->pi_preferred ==
2342 2342 preferred) ? 1 : 0);
2343 2343 } else if (flags ==
2344 2344 (MDI_SELECT_STANDBY_PATH |
2345 2345 MDI_SELECT_ONLINE_PATH |
2346 2346 MDI_SELECT_USER_DISABLE_PATH)) {
2347 2347 cond = (((MDI_PI(pip)->pi_state ==
2348 2348 MDI_PATHINFO_STATE_ONLINE ||
2349 2349 (MDI_PI(pip)->pi_state ==
2350 2350 MDI_PATHINFO_STATE_STANDBY) ||
2351 2351 (MDI_PI(pip)->pi_state ==
2352 2352 (MDI_PATHINFO_STATE_ONLINE|
2353 2353 MDI_PATHINFO_STATE_USER_DISABLE)) ||
2354 2354 (MDI_PI(pip)->pi_state ==
2355 2355 (MDI_PATHINFO_STATE_STANDBY |
2356 2356 MDI_PATHINFO_STATE_USER_DISABLE)))&&
2357 2357 MDI_PI(pip)->pi_preferred ==
2358 2358 preferred) ? 1 : 0);
2359 2359 } else if (flags ==
2360 2360 (MDI_SELECT_STANDBY_PATH |
2361 2361 MDI_SELECT_ONLINE_PATH |
2362 2362 MDI_SELECT_NO_PREFERRED)) {
2363 2363 cond = (((MDI_PI(pip)->pi_state ==
2364 2364 MDI_PATHINFO_STATE_ONLINE) ||
2365 2365 (MDI_PI(pip)->pi_state ==
2366 2366 MDI_PATHINFO_STATE_STANDBY))
2367 2367 ? 1 : 0);
2368 2368 } else {
2369 2369 cond = 0;
2370 2370 }
2371 2371 }
2372 2372 /*
2373 2373 * No need to explicitly check if the path is disabled.
2374 2374 * Since we are checking for state == ONLINE and the
2375 2375 * same variable is used for DISABLE/ENABLE information.
2376 2376 */
2377 2377 if (cond) {
2378 2378 /*
2379 2379 * Return the path in hold state. Caller should
2380 2380 * release the lock by calling mdi_rele_path()
2381 2381 */
2382 2382 MDI_PI_HOLD(pip);
2383 2383 MDI_PI_UNLOCK(pip);
2384 2384 if (sb)
2385 2385 ct->ct_path_last = pip;
2386 2386 *ret_pip = pip;
2387 2387 MDI_CLIENT_UNLOCK(ct);
2388 2388 return (MDI_SUCCESS);
2389 2389 }
2390 2390 /*
2391 2391 * Path is busy.
2392 2392 */
2393 2393 if (MDI_PI_IS_DRV_DISABLE_TRANSIENT(pip) ||
2394 2394 MDI_PI_IS_TRANSIENT(pip))
2395 2395 retry = 1;
2396 2396
2397 2397 /*
2398 2398 * Keep looking for a next available online path
2399 2399 */
2400 2400 do_again:
2401 2401 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
2402 2402 if (next == NULL) {
2403 2403 if ( flags & MDI_SELECT_NO_PREFERRED) {
2404 2404 /*
2405 2405 * Bail out since we hit the end of list
2406 2406 */
2407 2407 MDI_PI_UNLOCK(pip);
2408 2408 break;
2409 2409 }
2410 2410
2411 2411 if (!sb) {
2412 2412 if (preferred == 1) {
2413 2413 /*
2414 2414 * Looks like we reached the
2415 2415 * end of the list. Lets enable
2416 2416 * traversal of non preferred
2417 2417 * paths.
2418 2418 */
2419 2419 preferred = 0;
2420 2420 next = head;
2421 2421 } else {
2422 2422 /*
2423 2423 * We have done both the passes
2424 2424 * Preferred as well as for
2425 2425 * Non-preferred. Bail out now.
2426 2426 */
2427 2427 cont = 0;
2428 2428 }
2429 2429 } else {
2430 2430 /*
2431 2431 * Standard behavior case.
2432 2432 */
2433 2433 next = head;
2434 2434 }
2435 2435 }
2436 2436 MDI_PI_UNLOCK(pip);
2437 2437 if (cont == 0) {
2438 2438 break;
2439 2439 }
2440 2440 pip = next;
2441 2441
2442 2442 if (!sb) {
2443 2443 /*
2444 2444 * We need to handle the selection of
2445 2445 * non-preferred path in the following
2446 2446 * case:
2447 2447 *
2448 2448 * +------+ +------+ +------+ +-----+
2449 2449 * | A : 1| - | B : 1| - | C : 0| - |NULL |
2450 2450 * +------+ +------+ +------+ +-----+
2451 2451 *
2452 2452 * If we start the search with B, we need to
2453 2453 * skip beyond B to pick C which is non -
2454 2454 * preferred in the second pass. The following
2455 2455 * test, if true, will allow us to skip over
2456 2456 * the 'start'(B in the example) to select
2457 2457 * other non preferred elements.
2458 2458 */
2459 2459 if ((start_pip != NULL) && (start_pip == pip) &&
2460 2460 (MDI_PI(start_pip)->pi_preferred
2461 2461 != preferred)) {
2462 2462 /*
2463 2463 * try again after going past the start
2464 2464 * pip
2465 2465 */
2466 2466 MDI_PI_LOCK(pip);
2467 2467 goto do_again;
2468 2468 }
2469 2469 } else {
2470 2470 /*
2471 2471 * Standard behavior case
2472 2472 */
2473 2473 if (start == pip && preferred) {
2474 2474 /* look for nonpreferred paths */
2475 2475 preferred = 0;
2476 2476 } else if (start == pip && !preferred) {
2477 2477 /*
2478 2478 * Exit condition
2479 2479 */
2480 2480 cont = 0;
2481 2481 }
2482 2482 }
2483 2483 } while (cont);
2484 2484 break;
2485 2485 }
2486 2486
2487 2487 MDI_CLIENT_UNLOCK(ct);
2488 2488 if (retry == 1) {
2489 2489 return (MDI_BUSY);
2490 2490 } else {
2491 2491 return (MDI_NOPATH);
2492 2492 }
2493 2493 }
2494 2494
2495 2495 /*
2496 2496 * For a client, return the next available path to any phci
2497 2497 *
2498 2498 * Note:
2499 2499 * Caller should hold the branch's devinfo node to get a consistent
2500 2500 * snap shot of the mdi_pathinfo nodes.
2501 2501 *
2502 2502 * Please note that even the list is stable the mdi_pathinfo
2503 2503 * node state and properties are volatile. The caller should lock
2504 2504 * and unlock the nodes by calling mdi_pi_lock() and
2505 2505 * mdi_pi_unlock() functions to get a stable properties.
2506 2506 *
2507 2507 * If there is a need to use the nodes beyond the hold of the
2508 2508 * devinfo node period (For ex. I/O), then mdi_pathinfo node
2509 2509 * need to be held against unexpected removal by calling
2510 2510 * mdi_hold_path() and should be released by calling
2511 2511 * mdi_rele_path() on completion.
2512 2512 */
2513 2513 mdi_pathinfo_t *
2514 2514 mdi_get_next_phci_path(dev_info_t *ct_dip, mdi_pathinfo_t *pip)
2515 2515 {
2516 2516 mdi_client_t *ct;
2517 2517
2518 2518 if (!MDI_CLIENT(ct_dip))
2519 2519 return (NULL);
2520 2520
2521 2521 /*
2522 2522 * Walk through client link
2523 2523 */
2524 2524 ct = (mdi_client_t *)DEVI(ct_dip)->devi_mdi_client;
2525 2525 ASSERT(ct != NULL);
2526 2526
2527 2527 if (pip == NULL)
2528 2528 return ((mdi_pathinfo_t *)ct->ct_path_head);
2529 2529
2530 2530 return ((mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link);
2531 2531 }
2532 2532
2533 2533 /*
2534 2534 * For a phci, return the next available path to any client
2535 2535 * Note: ditto mdi_get_next_phci_path()
2536 2536 */
2537 2537 mdi_pathinfo_t *
2538 2538 mdi_get_next_client_path(dev_info_t *ph_dip, mdi_pathinfo_t *pip)
2539 2539 {
2540 2540 mdi_phci_t *ph;
2541 2541
2542 2542 if (!MDI_PHCI(ph_dip))
2543 2543 return (NULL);
2544 2544
2545 2545 /*
2546 2546 * Walk through pHCI link
2547 2547 */
2548 2548 ph = (mdi_phci_t *)DEVI(ph_dip)->devi_mdi_xhci;
2549 2549 ASSERT(ph != NULL);
2550 2550
2551 2551 if (pip == NULL)
2552 2552 return ((mdi_pathinfo_t *)ph->ph_path_head);
2553 2553
2554 2554 return ((mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link);
2555 2555 }
2556 2556
2557 2557 /*
2558 2558 * mdi_hold_path():
2559 2559 * Hold the mdi_pathinfo node against unwanted unexpected free.
2560 2560 * Return Values:
2561 2561 * None
2562 2562 */
2563 2563 void
2564 2564 mdi_hold_path(mdi_pathinfo_t *pip)
2565 2565 {
2566 2566 if (pip) {
2567 2567 MDI_PI_LOCK(pip);
2568 2568 MDI_PI_HOLD(pip);
2569 2569 MDI_PI_UNLOCK(pip);
2570 2570 }
2571 2571 }
2572 2572
2573 2573
2574 2574 /*
2575 2575 * mdi_rele_path():
2576 2576 * Release the mdi_pathinfo node which was selected
2577 2577 * through mdi_select_path() mechanism or manually held by
2578 2578 * calling mdi_hold_path().
2579 2579 * Return Values:
2580 2580 * None
2581 2581 */
2582 2582 void
2583 2583 mdi_rele_path(mdi_pathinfo_t *pip)
2584 2584 {
2585 2585 if (pip) {
2586 2586 MDI_PI_LOCK(pip);
2587 2587 MDI_PI_RELE(pip);
2588 2588 if (MDI_PI(pip)->pi_ref_cnt == 0) {
2589 2589 cv_broadcast(&MDI_PI(pip)->pi_ref_cv);
2590 2590 }
2591 2591 MDI_PI_UNLOCK(pip);
2592 2592 }
2593 2593 }
2594 2594
2595 2595 /*
2596 2596 * mdi_pi_lock():
2597 2597 * Lock the mdi_pathinfo node.
2598 2598 * Note:
2599 2599 * The caller should release the lock by calling mdi_pi_unlock()
2600 2600 */
2601 2601 void
2602 2602 mdi_pi_lock(mdi_pathinfo_t *pip)
2603 2603 {
2604 2604 ASSERT(pip != NULL);
2605 2605 if (pip) {
2606 2606 MDI_PI_LOCK(pip);
2607 2607 }
2608 2608 }
2609 2609
2610 2610
2611 2611 /*
2612 2612 * mdi_pi_unlock():
2613 2613 * Unlock the mdi_pathinfo node.
2614 2614 * Note:
2615 2615 * The mdi_pathinfo node should have been locked with mdi_pi_lock()
2616 2616 */
2617 2617 void
2618 2618 mdi_pi_unlock(mdi_pathinfo_t *pip)
2619 2619 {
2620 2620 ASSERT(pip != NULL);
2621 2621 if (pip) {
2622 2622 MDI_PI_UNLOCK(pip);
2623 2623 }
2624 2624 }
2625 2625
2626 2626 /*
2627 2627 * mdi_pi_find():
2628 2628 * Search the list of mdi_pathinfo nodes attached to the
2629 2629 * pHCI/Client device node whose path address matches "paddr".
2630 2630 * Returns a pointer to the mdi_pathinfo node if a matching node is
2631 2631 * found.
2632 2632 * Return Values:
2633 2633 * mdi_pathinfo node handle
2634 2634 * NULL
2635 2635 * Notes:
2636 2636 * Caller need not hold any locks to call this function.
2637 2637 */
2638 2638 mdi_pathinfo_t *
2639 2639 mdi_pi_find(dev_info_t *pdip, char *caddr, char *paddr)
2640 2640 {
2641 2641 mdi_phci_t *ph;
2642 2642 mdi_vhci_t *vh;
2643 2643 mdi_client_t *ct;
2644 2644 mdi_pathinfo_t *pip = NULL;
2645 2645
2646 2646 MDI_DEBUG(2, (MDI_NOTE, pdip,
2647 2647 "caddr@%s paddr@%s", caddr ? caddr : "", paddr ? paddr : ""));
2648 2648 if ((pdip == NULL) || (paddr == NULL)) {
2649 2649 return (NULL);
2650 2650 }
2651 2651 ph = i_devi_get_phci(pdip);
2652 2652 if (ph == NULL) {
2653 2653 /*
2654 2654 * Invalid pHCI device, Nothing more to do.
2655 2655 */
2656 2656 MDI_DEBUG(2, (MDI_WARN, pdip, "invalid phci"));
2657 2657 return (NULL);
2658 2658 }
2659 2659
2660 2660 vh = ph->ph_vhci;
2661 2661 if (vh == NULL) {
2662 2662 /*
2663 2663 * Invalid vHCI device, Nothing more to do.
2664 2664 */
2665 2665 MDI_DEBUG(2, (MDI_WARN, pdip, "invalid vhci"));
2666 2666 return (NULL);
2667 2667 }
2668 2668
2669 2669 /*
2670 2670 * Look for pathinfo node identified by paddr.
2671 2671 */
2672 2672 if (caddr == NULL) {
2673 2673 /*
2674 2674 * Find a mdi_pathinfo node under pHCI list for a matching
2675 2675 * unit address.
2676 2676 */
2677 2677 MDI_PHCI_LOCK(ph);
2678 2678 if (MDI_PHCI_IS_OFFLINE(ph)) {
2679 2679 MDI_DEBUG(2, (MDI_WARN, pdip,
2680 2680 "offline phci %p", (void *)ph));
2681 2681 MDI_PHCI_UNLOCK(ph);
2682 2682 return (NULL);
2683 2683 }
2684 2684 pip = (mdi_pathinfo_t *)ph->ph_path_head;
2685 2685
2686 2686 while (pip != NULL) {
2687 2687 if (strcmp(MDI_PI(pip)->pi_addr, paddr) == 0) {
2688 2688 break;
2689 2689 }
2690 2690 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
2691 2691 }
2692 2692 MDI_PHCI_UNLOCK(ph);
2693 2693 MDI_DEBUG(2, (MDI_NOTE, pdip,
2694 2694 "found %s %p", mdi_pi_spathname(pip), (void *)pip));
2695 2695 return (pip);
2696 2696 }
2697 2697
2698 2698 /*
2699 2699 * XXX - Is the rest of the code in this function really necessary?
2700 2700 * The consumers of mdi_pi_find() can search for the desired pathinfo
2701 2701 * node by calling mdi_pi_find(pdip, NULL, paddr). Irrespective of
2702 2702 * whether the search is based on the pathinfo nodes attached to
2703 2703 * the pHCI or the client node, the result will be the same.
2704 2704 */
2705 2705
2706 2706 /*
2707 2707 * Find the client device corresponding to 'caddr'
2708 2708 */
2709 2709 MDI_VHCI_CLIENT_LOCK(vh);
2710 2710
2711 2711 /*
2712 2712 * XXX - Passing NULL to the following function works as long as the
2713 2713 * the client addresses (caddr) are unique per vhci basis.
2714 2714 */
2715 2715 ct = i_mdi_client_find(vh, NULL, caddr);
2716 2716 if (ct == NULL) {
2717 2717 /*
2718 2718 * Client not found, Obviously mdi_pathinfo node has not been
2719 2719 * created yet.
2720 2720 */
2721 2721 MDI_VHCI_CLIENT_UNLOCK(vh);
2722 2722 MDI_DEBUG(2, (MDI_NOTE, pdip,
2723 2723 "client not found for caddr @%s", caddr ? caddr : ""));
2724 2724 return (NULL);
2725 2725 }
2726 2726
2727 2727 /*
2728 2728 * Hold the client lock and look for a mdi_pathinfo node with matching
2729 2729 * pHCI and paddr
2730 2730 */
2731 2731 MDI_CLIENT_LOCK(ct);
2732 2732
2733 2733 /*
2734 2734 * Release the global mutex as it is no more needed. Note: We always
2735 2735 * respect the locking order while acquiring.
2736 2736 */
2737 2737 MDI_VHCI_CLIENT_UNLOCK(vh);
2738 2738
2739 2739 pip = (mdi_pathinfo_t *)ct->ct_path_head;
2740 2740 while (pip != NULL) {
2741 2741 /*
2742 2742 * Compare the unit address
2743 2743 */
2744 2744 if ((MDI_PI(pip)->pi_phci == ph) &&
2745 2745 strcmp(MDI_PI(pip)->pi_addr, paddr) == 0) {
2746 2746 break;
2747 2747 }
2748 2748 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
2749 2749 }
2750 2750 MDI_CLIENT_UNLOCK(ct);
2751 2751 MDI_DEBUG(2, (MDI_NOTE, pdip,
2752 2752 "found: %s %p", mdi_pi_spathname(pip), (void *)pip));
2753 2753 return (pip);
2754 2754 }
2755 2755
2756 2756 /*
2757 2757 * mdi_pi_alloc():
2758 2758 * Allocate and initialize a new instance of a mdi_pathinfo node.
2759 2759 * The mdi_pathinfo node returned by this function identifies a
2760 2760 * unique device path is capable of having properties attached
2761 2761 * and passed to mdi_pi_online() to fully attach and online the
2762 2762 * path and client device node.
2763 2763 * The mdi_pathinfo node returned by this function must be
2764 2764 * destroyed using mdi_pi_free() if the path is no longer
2765 2765 * operational or if the caller fails to attach a client device
2766 2766 * node when calling mdi_pi_online(). The framework will not free
2767 2767 * the resources allocated.
2768 2768 * This function can be called from both interrupt and kernel
2769 2769 * contexts. DDI_NOSLEEP flag should be used while calling
2770 2770 * from interrupt contexts.
2771 2771 * Return Values:
2772 2772 * MDI_SUCCESS
2773 2773 * MDI_FAILURE
2774 2774 * MDI_NOMEM
2775 2775 */
2776 2776 /*ARGSUSED*/
2777 2777 int
2778 2778 mdi_pi_alloc_compatible(dev_info_t *pdip, char *cname, char *caddr, char *paddr,
2779 2779 char **compatible, int ncompatible, int flags, mdi_pathinfo_t **ret_pip)
2780 2780 {
2781 2781 mdi_vhci_t *vh;
2782 2782 mdi_phci_t *ph;
2783 2783 mdi_client_t *ct;
2784 2784 mdi_pathinfo_t *pip = NULL;
2785 2785 dev_info_t *cdip;
2786 2786 int rv = MDI_NOMEM;
2787 2787 int path_allocated = 0;
2788 2788
2789 2789 MDI_DEBUG(2, (MDI_NOTE, pdip,
2790 2790 "cname %s: caddr@%s paddr@%s",
2791 2791 cname ? cname : "", caddr ? caddr : "", paddr ? paddr : ""));
2792 2792
2793 2793 if (pdip == NULL || cname == NULL || caddr == NULL || paddr == NULL ||
2794 2794 ret_pip == NULL) {
2795 2795 /* Nothing more to do */
2796 2796 return (MDI_FAILURE);
2797 2797 }
2798 2798
2799 2799 *ret_pip = NULL;
2800 2800
2801 2801 /* No allocations on detaching pHCI */
2802 2802 if (DEVI_IS_DETACHING(pdip)) {
2803 2803 /* Invalid pHCI device, return failure */
2804 2804 MDI_DEBUG(1, (MDI_WARN, pdip,
2805 2805 "!detaching pHCI=%p", (void *)pdip));
2806 2806 return (MDI_FAILURE);
2807 2807 }
2808 2808
2809 2809 ph = i_devi_get_phci(pdip);
2810 2810 ASSERT(ph != NULL);
2811 2811 if (ph == NULL) {
2812 2812 /* Invalid pHCI device, return failure */
2813 2813 MDI_DEBUG(1, (MDI_WARN, pdip,
2814 2814 "!invalid pHCI=%p", (void *)pdip));
2815 2815 return (MDI_FAILURE);
2816 2816 }
2817 2817
2818 2818 MDI_PHCI_LOCK(ph);
2819 2819 vh = ph->ph_vhci;
2820 2820 if (vh == NULL) {
2821 2821 /* Invalid vHCI device, return failure */
2822 2822 MDI_DEBUG(1, (MDI_WARN, pdip,
2823 2823 "!invalid vHCI=%p", (void *)pdip));
2824 2824 MDI_PHCI_UNLOCK(ph);
2825 2825 return (MDI_FAILURE);
2826 2826 }
2827 2827
2828 2828 if (MDI_PHCI_IS_READY(ph) == 0) {
2829 2829 /*
2830 2830 * Do not allow new node creation when pHCI is in
2831 2831 * offline/suspended states
2832 2832 */
2833 2833 MDI_DEBUG(1, (MDI_WARN, pdip,
2834 2834 "pHCI=%p is not ready", (void *)ph));
2835 2835 MDI_PHCI_UNLOCK(ph);
2836 2836 return (MDI_BUSY);
2837 2837 }
2838 2838 MDI_PHCI_UNSTABLE(ph);
2839 2839 MDI_PHCI_UNLOCK(ph);
2840 2840
2841 2841 /* look for a matching client, create one if not found */
2842 2842 MDI_VHCI_CLIENT_LOCK(vh);
2843 2843 ct = i_mdi_client_find(vh, cname, caddr);
2844 2844 if (ct == NULL) {
2845 2845 ct = i_mdi_client_alloc(vh, cname, caddr);
2846 2846 ASSERT(ct != NULL);
2847 2847 }
2848 2848
2849 2849 if (ct->ct_dip == NULL) {
2850 2850 /*
2851 2851 * Allocate a devinfo node
2852 2852 */
2853 2853 ct->ct_dip = i_mdi_devinfo_create(vh, cname, caddr,
2854 2854 compatible, ncompatible);
2855 2855 if (ct->ct_dip == NULL) {
2856 2856 (void) i_mdi_client_free(vh, ct);
2857 2857 goto fail;
2858 2858 }
2859 2859 }
2860 2860 cdip = ct->ct_dip;
2861 2861
2862 2862 DEVI(cdip)->devi_mdi_component |= MDI_COMPONENT_CLIENT;
2863 2863 DEVI(cdip)->devi_mdi_client = (caddr_t)ct;
2864 2864
2865 2865 MDI_CLIENT_LOCK(ct);
2866 2866 pip = (mdi_pathinfo_t *)ct->ct_path_head;
2867 2867 while (pip != NULL) {
2868 2868 /*
2869 2869 * Compare the unit address
2870 2870 */
2871 2871 if ((MDI_PI(pip)->pi_phci == ph) &&
2872 2872 strcmp(MDI_PI(pip)->pi_addr, paddr) == 0) {
2873 2873 break;
2874 2874 }
2875 2875 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
2876 2876 }
2877 2877 MDI_CLIENT_UNLOCK(ct);
2878 2878
2879 2879 if (pip == NULL) {
2880 2880 /*
2881 2881 * This is a new path for this client device. Allocate and
2882 2882 * initialize a new pathinfo node
2883 2883 */
2884 2884 pip = i_mdi_pi_alloc(ph, paddr, ct);
2885 2885 ASSERT(pip != NULL);
2886 2886 path_allocated = 1;
2887 2887 }
2888 2888 rv = MDI_SUCCESS;
2889 2889
2890 2890 fail:
2891 2891 /*
2892 2892 * Release the global mutex.
2893 2893 */
2894 2894 MDI_VHCI_CLIENT_UNLOCK(vh);
2895 2895
2896 2896 /*
2897 2897 * Mark the pHCI as stable
2898 2898 */
2899 2899 MDI_PHCI_LOCK(ph);
2900 2900 MDI_PHCI_STABLE(ph);
2901 2901 MDI_PHCI_UNLOCK(ph);
2902 2902 *ret_pip = pip;
2903 2903
2904 2904 MDI_DEBUG(2, (MDI_NOTE, pdip,
2905 2905 "alloc %s %p", mdi_pi_spathname(pip), (void *)pip));
2906 2906
2907 2907 if (path_allocated)
2908 2908 vhcache_pi_add(vh->vh_config, MDI_PI(pip));
2909 2909
2910 2910 return (rv);
2911 2911 }
2912 2912
2913 2913 /*ARGSUSED*/
2914 2914 int
2915 2915 mdi_pi_alloc(dev_info_t *pdip, char *cname, char *caddr, char *paddr,
2916 2916 int flags, mdi_pathinfo_t **ret_pip)
2917 2917 {
2918 2918 return (mdi_pi_alloc_compatible(pdip, cname, caddr, paddr, NULL, 0,
2919 2919 flags, ret_pip));
2920 2920 }
2921 2921
2922 2922 /*
2923 2923 * i_mdi_pi_alloc():
2924 2924 * Allocate a mdi_pathinfo node and add to the pHCI path list
2925 2925 * Return Values:
2926 2926 * mdi_pathinfo
2927 2927 */
2928 2928 /*ARGSUSED*/
2929 2929 static mdi_pathinfo_t *
2930 2930 i_mdi_pi_alloc(mdi_phci_t *ph, char *paddr, mdi_client_t *ct)
2931 2931 {
2932 2932 mdi_pathinfo_t *pip;
2933 2933 int ct_circular;
2934 2934 int ph_circular;
2935 2935 static char path[MAXPATHLEN]; /* mdi_pathmap_mutex protects */
2936 2936 char *path_persistent;
2937 2937 int path_instance;
2938 2938 mod_hash_val_t hv;
2939 2939
2940 2940 ASSERT(MDI_VHCI_CLIENT_LOCKED(ph->ph_vhci));
2941 2941
2942 2942 pip = kmem_zalloc(sizeof (struct mdi_pathinfo), KM_SLEEP);
2943 2943 mutex_init(&MDI_PI(pip)->pi_mutex, NULL, MUTEX_DEFAULT, NULL);
2944 2944 MDI_PI(pip)->pi_state = MDI_PATHINFO_STATE_INIT |
2945 2945 MDI_PATHINFO_STATE_TRANSIENT;
2946 2946
2947 2947 if (MDI_PHCI_IS_USER_DISABLED(ph))
2948 2948 MDI_PI_SET_USER_DISABLE(pip);
2949 2949
2950 2950 if (MDI_PHCI_IS_DRV_DISABLED_TRANSIENT(ph))
2951 2951 MDI_PI_SET_DRV_DISABLE_TRANS(pip);
2952 2952
2953 2953 if (MDI_PHCI_IS_DRV_DISABLED(ph))
2954 2954 MDI_PI_SET_DRV_DISABLE(pip);
2955 2955
2956 2956 MDI_PI(pip)->pi_old_state = MDI_PATHINFO_STATE_INIT;
2957 2957 cv_init(&MDI_PI(pip)->pi_state_cv, NULL, CV_DEFAULT, NULL);
2958 2958 MDI_PI(pip)->pi_client = ct;
2959 2959 MDI_PI(pip)->pi_phci = ph;
2960 2960 MDI_PI(pip)->pi_addr = kmem_alloc(strlen(paddr) + 1, KM_SLEEP);
2961 2961 (void) strcpy(MDI_PI(pip)->pi_addr, paddr);
2962 2962
2963 2963 /*
2964 2964 * We form the "path" to the pathinfo node, and see if we have
2965 2965 * already allocated a 'path_instance' for that "path". If so,
2966 2966 * we use the already allocated 'path_instance'. If not, we
2967 2967 * allocate a new 'path_instance' and associate it with a copy of
2968 2968 * the "path" string (which is never freed). The association
2969 2969 * between a 'path_instance' this "path" string persists until
2970 2970 * reboot.
2971 2971 */
2972 2972 mutex_enter(&mdi_pathmap_mutex);
2973 2973 (void) ddi_pathname(ph->ph_dip, path);
2974 2974 (void) sprintf(path + strlen(path), "/%s@%s",
2975 2975 mdi_pi_get_node_name(pip), mdi_pi_get_addr(pip));
2976 2976 if (mod_hash_find(mdi_pathmap_bypath, (mod_hash_key_t)path, &hv) == 0) {
2977 2977 path_instance = (uint_t)(intptr_t)hv;
2978 2978 } else {
2979 2979 /* allocate a new 'path_instance' and persistent "path" */
2980 2980 path_instance = mdi_pathmap_instance++;
2981 2981 path_persistent = i_ddi_strdup(path, KM_SLEEP);
2982 2982 (void) mod_hash_insert(mdi_pathmap_bypath,
2983 2983 (mod_hash_key_t)path_persistent,
2984 2984 (mod_hash_val_t)(intptr_t)path_instance);
2985 2985 (void) mod_hash_insert(mdi_pathmap_byinstance,
2986 2986 (mod_hash_key_t)(intptr_t)path_instance,
2987 2987 (mod_hash_val_t)path_persistent);
2988 2988
2989 2989 /* create shortpath name */
2990 2990 (void) snprintf(path, sizeof(path), "%s%d/%s@%s",
2991 2991 ddi_driver_name(ph->ph_dip), ddi_get_instance(ph->ph_dip),
2992 2992 mdi_pi_get_node_name(pip), mdi_pi_get_addr(pip));
2993 2993 path_persistent = i_ddi_strdup(path, KM_SLEEP);
2994 2994 (void) mod_hash_insert(mdi_pathmap_sbyinstance,
2995 2995 (mod_hash_key_t)(intptr_t)path_instance,
2996 2996 (mod_hash_val_t)path_persistent);
2997 2997 }
2998 2998 mutex_exit(&mdi_pathmap_mutex);
2999 2999 MDI_PI(pip)->pi_path_instance = path_instance;
3000 3000
3001 3001 (void) nvlist_alloc(&MDI_PI(pip)->pi_prop, NV_UNIQUE_NAME, KM_SLEEP);
3002 3002 ASSERT(MDI_PI(pip)->pi_prop != NULL);
3003 3003 MDI_PI(pip)->pi_pprivate = NULL;
3004 3004 MDI_PI(pip)->pi_cprivate = NULL;
3005 3005 MDI_PI(pip)->pi_vprivate = NULL;
3006 3006 MDI_PI(pip)->pi_client_link = NULL;
3007 3007 MDI_PI(pip)->pi_phci_link = NULL;
3008 3008 MDI_PI(pip)->pi_ref_cnt = 0;
3009 3009 MDI_PI(pip)->pi_kstats = NULL;
3010 3010 MDI_PI(pip)->pi_preferred = 1;
3011 3011 cv_init(&MDI_PI(pip)->pi_ref_cv, NULL, CV_DEFAULT, NULL);
3012 3012
3013 3013 /*
3014 3014 * Lock both dev_info nodes against changes in parallel.
3015 3015 *
3016 3016 * The ndi_devi_enter(Client), is atypical since the client is a leaf.
3017 3017 * This atypical operation is done to synchronize pathinfo nodes
3018 3018 * during devinfo snapshot (see di_register_pip) by 'pretending' that
3019 3019 * the pathinfo nodes are children of the Client.
3020 3020 */
3021 3021 ndi_devi_enter(ct->ct_dip, &ct_circular);
3022 3022 ndi_devi_enter(ph->ph_dip, &ph_circular);
3023 3023
3024 3024 i_mdi_phci_add_path(ph, pip);
3025 3025 i_mdi_client_add_path(ct, pip);
3026 3026
3027 3027 ndi_devi_exit(ph->ph_dip, ph_circular);
3028 3028 ndi_devi_exit(ct->ct_dip, ct_circular);
3029 3029
3030 3030 return (pip);
3031 3031 }
3032 3032
3033 3033 /*
3034 3034 * mdi_pi_pathname_by_instance():
3035 3035 * Lookup of "path" by 'path_instance'. Return "path".
3036 3036 * NOTE: returned "path" remains valid forever (until reboot).
3037 3037 */
3038 3038 char *
3039 3039 mdi_pi_pathname_by_instance(int path_instance)
3040 3040 {
3041 3041 char *path;
3042 3042 mod_hash_val_t hv;
3043 3043
3044 3044 /* mdi_pathmap lookup of "path" by 'path_instance' */
3045 3045 mutex_enter(&mdi_pathmap_mutex);
3046 3046 if (mod_hash_find(mdi_pathmap_byinstance,
3047 3047 (mod_hash_key_t)(intptr_t)path_instance, &hv) == 0)
3048 3048 path = (char *)hv;
3049 3049 else
3050 3050 path = NULL;
3051 3051 mutex_exit(&mdi_pathmap_mutex);
3052 3052 return (path);
3053 3053 }
3054 3054
3055 3055 /*
3056 3056 * mdi_pi_spathname_by_instance():
3057 3057 * Lookup of "shortpath" by 'path_instance'. Return "shortpath".
3058 3058 * NOTE: returned "shortpath" remains valid forever (until reboot).
3059 3059 */
3060 3060 char *
3061 3061 mdi_pi_spathname_by_instance(int path_instance)
3062 3062 {
3063 3063 char *path;
3064 3064 mod_hash_val_t hv;
3065 3065
3066 3066 /* mdi_pathmap lookup of "path" by 'path_instance' */
3067 3067 mutex_enter(&mdi_pathmap_mutex);
3068 3068 if (mod_hash_find(mdi_pathmap_sbyinstance,
3069 3069 (mod_hash_key_t)(intptr_t)path_instance, &hv) == 0)
3070 3070 path = (char *)hv;
3071 3071 else
3072 3072 path = NULL;
3073 3073 mutex_exit(&mdi_pathmap_mutex);
3074 3074 return (path);
3075 3075 }
3076 3076
3077 3077
3078 3078 /*
3079 3079 * i_mdi_phci_add_path():
3080 3080 * Add a mdi_pathinfo node to pHCI list.
3081 3081 * Notes:
3082 3082 * Caller should per-pHCI mutex
3083 3083 */
3084 3084 static void
3085 3085 i_mdi_phci_add_path(mdi_phci_t *ph, mdi_pathinfo_t *pip)
3086 3086 {
3087 3087 ASSERT(DEVI_BUSY_OWNED(ph->ph_dip));
3088 3088
3089 3089 MDI_PHCI_LOCK(ph);
3090 3090 if (ph->ph_path_head == NULL) {
3091 3091 ph->ph_path_head = pip;
3092 3092 } else {
3093 3093 MDI_PI(ph->ph_path_tail)->pi_phci_link = MDI_PI(pip);
3094 3094 }
3095 3095 ph->ph_path_tail = pip;
3096 3096 ph->ph_path_count++;
3097 3097 MDI_PHCI_UNLOCK(ph);
3098 3098 }
3099 3099
3100 3100 /*
3101 3101 * i_mdi_client_add_path():
3102 3102 * Add mdi_pathinfo node to client list
3103 3103 */
3104 3104 static void
3105 3105 i_mdi_client_add_path(mdi_client_t *ct, mdi_pathinfo_t *pip)
3106 3106 {
3107 3107 ASSERT(DEVI_BUSY_OWNED(ct->ct_dip));
3108 3108
3109 3109 MDI_CLIENT_LOCK(ct);
3110 3110 if (ct->ct_path_head == NULL) {
3111 3111 ct->ct_path_head = pip;
3112 3112 } else {
3113 3113 MDI_PI(ct->ct_path_tail)->pi_client_link = MDI_PI(pip);
3114 3114 }
3115 3115 ct->ct_path_tail = pip;
3116 3116 ct->ct_path_count++;
3117 3117 MDI_CLIENT_UNLOCK(ct);
3118 3118 }
3119 3119
3120 3120 /*
3121 3121 * mdi_pi_free():
3122 3122 * Free the mdi_pathinfo node and also client device node if this
3123 3123 * is the last path to the device
3124 3124 * Return Values:
3125 3125 * MDI_SUCCESS
3126 3126 * MDI_FAILURE
3127 3127 * MDI_BUSY
3128 3128 */
3129 3129 /*ARGSUSED*/
3130 3130 int
3131 3131 mdi_pi_free(mdi_pathinfo_t *pip, int flags)
3132 3132 {
3133 3133 int rv;
3134 3134 mdi_vhci_t *vh;
3135 3135 mdi_phci_t *ph;
3136 3136 mdi_client_t *ct;
3137 3137 int (*f)();
3138 3138 int client_held = 0;
3139 3139
3140 3140 MDI_PI_LOCK(pip);
3141 3141 ph = MDI_PI(pip)->pi_phci;
3142 3142 ASSERT(ph != NULL);
3143 3143 if (ph == NULL) {
3144 3144 /*
3145 3145 * Invalid pHCI device, return failure
3146 3146 */
3147 3147 MDI_DEBUG(1, (MDI_WARN, NULL,
3148 3148 "!invalid pHCI: pip %s %p",
3149 3149 mdi_pi_spathname(pip), (void *)pip));
3150 3150 MDI_PI_UNLOCK(pip);
3151 3151 return (MDI_FAILURE);
3152 3152 }
3153 3153
3154 3154 vh = ph->ph_vhci;
3155 3155 ASSERT(vh != NULL);
3156 3156 if (vh == NULL) {
3157 3157 /* Invalid pHCI device, return failure */
3158 3158 MDI_DEBUG(1, (MDI_WARN, ph->ph_dip,
3159 3159 "!invalid vHCI: pip %s %p",
3160 3160 mdi_pi_spathname(pip), (void *)pip));
3161 3161 MDI_PI_UNLOCK(pip);
3162 3162 return (MDI_FAILURE);
3163 3163 }
3164 3164
3165 3165 ct = MDI_PI(pip)->pi_client;
3166 3166 ASSERT(ct != NULL);
3167 3167 if (ct == NULL) {
3168 3168 /*
3169 3169 * Invalid Client device, return failure
3170 3170 */
3171 3171 MDI_DEBUG(1, (MDI_WARN, ph->ph_dip,
3172 3172 "!invalid client: pip %s %p",
3173 3173 mdi_pi_spathname(pip), (void *)pip));
3174 3174 MDI_PI_UNLOCK(pip);
3175 3175 return (MDI_FAILURE);
3176 3176 }
3177 3177
3178 3178 /*
3179 3179 * Check to see for busy condition. A mdi_pathinfo can only be freed
3180 3180 * if the node state is either offline or init and the reference count
3181 3181 * is zero.
3182 3182 */
3183 3183 if (!(MDI_PI_IS_OFFLINE(pip) || MDI_PI_IS_INIT(pip) ||
3184 3184 MDI_PI_IS_INITING(pip))) {
3185 3185 /*
3186 3186 * Node is busy
3187 3187 */
3188 3188 MDI_DEBUG(1, (MDI_WARN, ct->ct_dip,
3189 3189 "!busy: pip %s %p", mdi_pi_spathname(pip), (void *)pip));
3190 3190 MDI_PI_UNLOCK(pip);
3191 3191 return (MDI_BUSY);
3192 3192 }
3193 3193
3194 3194 while (MDI_PI(pip)->pi_ref_cnt != 0) {
3195 3195 /*
3196 3196 * Give a chance for pending I/Os to complete.
3197 3197 */
3198 3198 MDI_DEBUG(1, (MDI_NOTE, ct->ct_dip,
3199 3199 "!%d cmds still pending on path: %s %p",
3200 3200 MDI_PI(pip)->pi_ref_cnt,
3201 3201 mdi_pi_spathname(pip), (void *)pip));
3202 3202 if (cv_reltimedwait(&MDI_PI(pip)->pi_ref_cv,
3203 3203 &MDI_PI(pip)->pi_mutex, drv_usectohz(60 * 1000000),
3204 3204 TR_CLOCK_TICK) == -1) {
3205 3205 /*
3206 3206 * The timeout time reached without ref_cnt being zero
3207 3207 * being signaled.
3208 3208 */
3209 3209 MDI_DEBUG(1, (MDI_NOTE, ct->ct_dip,
3210 3210 "!Timeout reached on path %s %p without the cond",
3211 3211 mdi_pi_spathname(pip), (void *)pip));
3212 3212 MDI_DEBUG(1, (MDI_NOTE, ct->ct_dip,
3213 3213 "!%d cmds still pending on path %s %p",
3214 3214 MDI_PI(pip)->pi_ref_cnt,
3215 3215 mdi_pi_spathname(pip), (void *)pip));
3216 3216 MDI_PI_UNLOCK(pip);
3217 3217 return (MDI_BUSY);
3218 3218 }
3219 3219 }
3220 3220 if (MDI_PI(pip)->pi_pm_held) {
3221 3221 client_held = 1;
3222 3222 }
3223 3223 MDI_PI_UNLOCK(pip);
3224 3224
3225 3225 vhcache_pi_remove(vh->vh_config, MDI_PI(pip));
3226 3226
3227 3227 MDI_CLIENT_LOCK(ct);
3228 3228
3229 3229 /* Prevent further failovers till MDI_VHCI_CLIENT_LOCK is held */
3230 3230 MDI_CLIENT_SET_PATH_FREE_IN_PROGRESS(ct);
3231 3231
3232 3232 /*
3233 3233 * Wait till failover is complete before removing this node.
3234 3234 */
3235 3235 while (MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct))
3236 3236 cv_wait(&ct->ct_failover_cv, &ct->ct_mutex);
3237 3237
3238 3238 MDI_CLIENT_UNLOCK(ct);
3239 3239 MDI_VHCI_CLIENT_LOCK(vh);
3240 3240 MDI_CLIENT_LOCK(ct);
3241 3241 MDI_CLIENT_CLEAR_PATH_FREE_IN_PROGRESS(ct);
3242 3242
3243 3243 if (!MDI_PI_IS_INITING(pip)) {
3244 3244 f = vh->vh_ops->vo_pi_uninit;
3245 3245 if (f != NULL) {
3246 3246 rv = (*f)(vh->vh_dip, pip, 0);
3247 3247 }
3248 3248 } else
3249 3249 rv = MDI_SUCCESS;
3250 3250
3251 3251 /*
3252 3252 * If vo_pi_uninit() completed successfully.
3253 3253 */
3254 3254 if (rv == MDI_SUCCESS) {
3255 3255 if (client_held) {
3256 3256 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
3257 3257 "i_mdi_pm_rele_client\n"));
3258 3258 i_mdi_pm_rele_client(ct, 1);
3259 3259 }
3260 3260 i_mdi_pi_free(ph, pip, ct);
3261 3261 if (ct->ct_path_count == 0) {
3262 3262 /*
3263 3263 * Client lost its last path.
3264 3264 * Clean up the client device
3265 3265 */
3266 3266 MDI_CLIENT_UNLOCK(ct);
3267 3267 (void) i_mdi_client_free(ct->ct_vhci, ct);
3268 3268 MDI_VHCI_CLIENT_UNLOCK(vh);
3269 3269 return (rv);
3270 3270 }
3271 3271 }
3272 3272 MDI_CLIENT_UNLOCK(ct);
3273 3273 MDI_VHCI_CLIENT_UNLOCK(vh);
3274 3274
3275 3275 if (rv == MDI_FAILURE)
3276 3276 vhcache_pi_add(vh->vh_config, MDI_PI(pip));
3277 3277
3278 3278 return (rv);
3279 3279 }
3280 3280
3281 3281 /*
3282 3282 * i_mdi_pi_free():
3283 3283 * Free the mdi_pathinfo node
3284 3284 */
3285 3285 static void
3286 3286 i_mdi_pi_free(mdi_phci_t *ph, mdi_pathinfo_t *pip, mdi_client_t *ct)
3287 3287 {
3288 3288 int ct_circular;
3289 3289 int ph_circular;
3290 3290
3291 3291 ASSERT(MDI_CLIENT_LOCKED(ct));
3292 3292
3293 3293 /*
3294 3294 * remove any per-path kstats
3295 3295 */
3296 3296 i_mdi_pi_kstat_destroy(pip);
3297 3297
3298 3298 /* See comments in i_mdi_pi_alloc() */
3299 3299 ndi_devi_enter(ct->ct_dip, &ct_circular);
3300 3300 ndi_devi_enter(ph->ph_dip, &ph_circular);
3301 3301
3302 3302 i_mdi_client_remove_path(ct, pip);
3303 3303 i_mdi_phci_remove_path(ph, pip);
3304 3304
3305 3305 ndi_devi_exit(ph->ph_dip, ph_circular);
3306 3306 ndi_devi_exit(ct->ct_dip, ct_circular);
3307 3307
3308 3308 mutex_destroy(&MDI_PI(pip)->pi_mutex);
3309 3309 cv_destroy(&MDI_PI(pip)->pi_state_cv);
3310 3310 cv_destroy(&MDI_PI(pip)->pi_ref_cv);
3311 3311 if (MDI_PI(pip)->pi_addr) {
3312 3312 kmem_free(MDI_PI(pip)->pi_addr,
3313 3313 strlen(MDI_PI(pip)->pi_addr) + 1);
3314 3314 MDI_PI(pip)->pi_addr = NULL;
3315 3315 }
3316 3316
3317 3317 if (MDI_PI(pip)->pi_prop) {
3318 3318 (void) nvlist_free(MDI_PI(pip)->pi_prop);
3319 3319 MDI_PI(pip)->pi_prop = NULL;
3320 3320 }
3321 3321 kmem_free(pip, sizeof (struct mdi_pathinfo));
3322 3322 }
3323 3323
3324 3324
3325 3325 /*
3326 3326 * i_mdi_phci_remove_path():
3327 3327 * Remove a mdi_pathinfo node from pHCI list.
3328 3328 * Notes:
3329 3329 * Caller should hold per-pHCI mutex
3330 3330 */
3331 3331 static void
3332 3332 i_mdi_phci_remove_path(mdi_phci_t *ph, mdi_pathinfo_t *pip)
3333 3333 {
3334 3334 mdi_pathinfo_t *prev = NULL;
3335 3335 mdi_pathinfo_t *path = NULL;
3336 3336
3337 3337 ASSERT(DEVI_BUSY_OWNED(ph->ph_dip));
3338 3338
3339 3339 MDI_PHCI_LOCK(ph);
3340 3340 path = ph->ph_path_head;
3341 3341 while (path != NULL) {
3342 3342 if (path == pip) {
3343 3343 break;
3344 3344 }
3345 3345 prev = path;
3346 3346 path = (mdi_pathinfo_t *)MDI_PI(path)->pi_phci_link;
3347 3347 }
3348 3348
3349 3349 if (path) {
3350 3350 ph->ph_path_count--;
3351 3351 if (prev) {
3352 3352 MDI_PI(prev)->pi_phci_link = MDI_PI(path)->pi_phci_link;
3353 3353 } else {
3354 3354 ph->ph_path_head =
3355 3355 (mdi_pathinfo_t *)MDI_PI(path)->pi_phci_link;
3356 3356 }
3357 3357 if (ph->ph_path_tail == path) {
3358 3358 ph->ph_path_tail = prev;
3359 3359 }
3360 3360 }
3361 3361
3362 3362 /*
3363 3363 * Clear the pHCI link
3364 3364 */
3365 3365 MDI_PI(pip)->pi_phci_link = NULL;
3366 3366 MDI_PI(pip)->pi_phci = NULL;
3367 3367 MDI_PHCI_UNLOCK(ph);
3368 3368 }
3369 3369
3370 3370 /*
3371 3371 * i_mdi_client_remove_path():
3372 3372 * Remove a mdi_pathinfo node from client path list.
3373 3373 */
3374 3374 static void
3375 3375 i_mdi_client_remove_path(mdi_client_t *ct, mdi_pathinfo_t *pip)
3376 3376 {
3377 3377 mdi_pathinfo_t *prev = NULL;
3378 3378 mdi_pathinfo_t *path;
3379 3379
3380 3380 ASSERT(DEVI_BUSY_OWNED(ct->ct_dip));
3381 3381
3382 3382 ASSERT(MDI_CLIENT_LOCKED(ct));
3383 3383 path = ct->ct_path_head;
3384 3384 while (path != NULL) {
3385 3385 if (path == pip) {
3386 3386 break;
3387 3387 }
3388 3388 prev = path;
3389 3389 path = (mdi_pathinfo_t *)MDI_PI(path)->pi_client_link;
3390 3390 }
3391 3391
3392 3392 if (path) {
3393 3393 ct->ct_path_count--;
3394 3394 if (prev) {
3395 3395 MDI_PI(prev)->pi_client_link =
3396 3396 MDI_PI(path)->pi_client_link;
3397 3397 } else {
3398 3398 ct->ct_path_head =
3399 3399 (mdi_pathinfo_t *)MDI_PI(path)->pi_client_link;
3400 3400 }
3401 3401 if (ct->ct_path_tail == path) {
3402 3402 ct->ct_path_tail = prev;
3403 3403 }
3404 3404 if (ct->ct_path_last == path) {
3405 3405 ct->ct_path_last = ct->ct_path_head;
3406 3406 }
3407 3407 }
3408 3408 MDI_PI(pip)->pi_client_link = NULL;
3409 3409 MDI_PI(pip)->pi_client = NULL;
3410 3410 }
3411 3411
3412 3412 /*
3413 3413 * i_mdi_pi_state_change():
3414 3414 * online a mdi_pathinfo node
3415 3415 *
3416 3416 * Return Values:
3417 3417 * MDI_SUCCESS
3418 3418 * MDI_FAILURE
3419 3419 */
3420 3420 /*ARGSUSED*/
3421 3421 static int
3422 3422 i_mdi_pi_state_change(mdi_pathinfo_t *pip, mdi_pathinfo_state_t state, int flag)
3423 3423 {
3424 3424 int rv = MDI_SUCCESS;
3425 3425 mdi_vhci_t *vh;
3426 3426 mdi_phci_t *ph;
3427 3427 mdi_client_t *ct;
3428 3428 int (*f)();
3429 3429 dev_info_t *cdip;
3430 3430
3431 3431 MDI_PI_LOCK(pip);
3432 3432
3433 3433 ph = MDI_PI(pip)->pi_phci;
3434 3434 ASSERT(ph);
3435 3435 if (ph == NULL) {
3436 3436 /*
3437 3437 * Invalid pHCI device, fail the request
3438 3438 */
3439 3439 MDI_PI_UNLOCK(pip);
3440 3440 MDI_DEBUG(1, (MDI_WARN, NULL,
3441 3441 "!invalid phci: pip %s %p",
3442 3442 mdi_pi_spathname(pip), (void *)pip));
3443 3443 return (MDI_FAILURE);
3444 3444 }
3445 3445
3446 3446 vh = ph->ph_vhci;
3447 3447 ASSERT(vh);
3448 3448 if (vh == NULL) {
3449 3449 /*
3450 3450 * Invalid vHCI device, fail the request
3451 3451 */
3452 3452 MDI_PI_UNLOCK(pip);
3453 3453 MDI_DEBUG(1, (MDI_WARN, ph->ph_dip,
3454 3454 "!invalid vhci: pip %s %p",
3455 3455 mdi_pi_spathname(pip), (void *)pip));
3456 3456 return (MDI_FAILURE);
3457 3457 }
3458 3458
3459 3459 ct = MDI_PI(pip)->pi_client;
3460 3460 ASSERT(ct != NULL);
3461 3461 if (ct == NULL) {
3462 3462 /*
3463 3463 * Invalid client device, fail the request
3464 3464 */
3465 3465 MDI_PI_UNLOCK(pip);
3466 3466 MDI_DEBUG(1, (MDI_WARN, ph->ph_dip,
3467 3467 "!invalid client: pip %s %p",
3468 3468 mdi_pi_spathname(pip), (void *)pip));
3469 3469 return (MDI_FAILURE);
3470 3470 }
3471 3471
3472 3472 /*
3473 3473 * If this path has not been initialized yet, Callback vHCI driver's
3474 3474 * pathinfo node initialize entry point
3475 3475 */
3476 3476
3477 3477 if (MDI_PI_IS_INITING(pip)) {
3478 3478 MDI_PI_UNLOCK(pip);
3479 3479 f = vh->vh_ops->vo_pi_init;
3480 3480 if (f != NULL) {
3481 3481 rv = (*f)(vh->vh_dip, pip, 0);
3482 3482 if (rv != MDI_SUCCESS) {
3483 3483 MDI_DEBUG(1, (MDI_WARN, ct->ct_dip,
3484 3484 "!vo_pi_init failed: vHCI %p, pip %s %p",
3485 3485 (void *)vh, mdi_pi_spathname(pip),
3486 3486 (void *)pip));
3487 3487 return (MDI_FAILURE);
3488 3488 }
3489 3489 }
3490 3490 MDI_PI_LOCK(pip);
3491 3491 MDI_PI_CLEAR_TRANSIENT(pip);
3492 3492 }
3493 3493
3494 3494 /*
3495 3495 * Do not allow state transition when pHCI is in offline/suspended
3496 3496 * states
3497 3497 */
3498 3498 i_mdi_phci_lock(ph, pip);
3499 3499 if (MDI_PHCI_IS_READY(ph) == 0) {
3500 3500 MDI_DEBUG(1, (MDI_WARN, ct->ct_dip,
3501 3501 "!pHCI not ready, pHCI=%p", (void *)ph));
3502 3502 MDI_PI_UNLOCK(pip);
3503 3503 i_mdi_phci_unlock(ph);
3504 3504 return (MDI_BUSY);
3505 3505 }
3506 3506 MDI_PHCI_UNSTABLE(ph);
3507 3507 i_mdi_phci_unlock(ph);
3508 3508
3509 3509 /*
3510 3510 * Check if mdi_pathinfo state is in transient state.
3511 3511 * If yes, offlining is in progress and wait till transient state is
3512 3512 * cleared.
3513 3513 */
3514 3514 if (MDI_PI_IS_TRANSIENT(pip)) {
3515 3515 while (MDI_PI_IS_TRANSIENT(pip)) {
3516 3516 cv_wait(&MDI_PI(pip)->pi_state_cv,
3517 3517 &MDI_PI(pip)->pi_mutex);
3518 3518 }
3519 3519 }
3520 3520
3521 3521 /*
3522 3522 * Grab the client lock in reverse order sequence and release the
3523 3523 * mdi_pathinfo mutex.
3524 3524 */
3525 3525 i_mdi_client_lock(ct, pip);
3526 3526 MDI_PI_UNLOCK(pip);
3527 3527
3528 3528 /*
3529 3529 * Wait till failover state is cleared
3530 3530 */
3531 3531 while (MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct))
3532 3532 cv_wait(&ct->ct_failover_cv, &ct->ct_mutex);
3533 3533
3534 3534 /*
3535 3535 * Mark the mdi_pathinfo node state as transient
3536 3536 */
3537 3537 MDI_PI_LOCK(pip);
3538 3538 switch (state) {
3539 3539 case MDI_PATHINFO_STATE_ONLINE:
3540 3540 MDI_PI_SET_ONLINING(pip);
3541 3541 break;
3542 3542
3543 3543 case MDI_PATHINFO_STATE_STANDBY:
3544 3544 MDI_PI_SET_STANDBYING(pip);
3545 3545 break;
3546 3546
3547 3547 case MDI_PATHINFO_STATE_FAULT:
3548 3548 /*
3549 3549 * Mark the pathinfo state as FAULTED
3550 3550 */
3551 3551 MDI_PI_SET_FAULTING(pip);
3552 3552 MDI_PI_ERRSTAT(pip, MDI_PI_HARDERR);
3553 3553 break;
3554 3554
3555 3555 case MDI_PATHINFO_STATE_OFFLINE:
3556 3556 /*
3557 3557 * ndi_devi_offline() cannot hold pip or ct locks.
3558 3558 */
3559 3559 MDI_PI_UNLOCK(pip);
3560 3560
3561 3561 /*
3562 3562 * If this is a user initiated path online->offline operation
3563 3563 * who's success would transition a client from DEGRADED to
3564 3564 * FAILED then only proceed if we can offline the client first.
3565 3565 */
3566 3566 cdip = ct->ct_dip;
3567 3567 if ((flag & NDI_USER_REQ) &&
3568 3568 MDI_PI_IS_ONLINE(pip) &&
3569 3569 (MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_DEGRADED)) {
3570 3570 i_mdi_client_unlock(ct);
3571 3571 rv = ndi_devi_offline(cdip, NDI_DEVFS_CLEAN);
3572 3572 if (rv != NDI_SUCCESS) {
3573 3573 /*
3574 3574 * Convert to MDI error code
3575 3575 */
3576 3576 switch (rv) {
3577 3577 case NDI_BUSY:
3578 3578 rv = MDI_BUSY;
3579 3579 break;
3580 3580 default:
3581 3581 rv = MDI_FAILURE;
3582 3582 break;
3583 3583 }
3584 3584 goto state_change_exit;
3585 3585 } else {
3586 3586 i_mdi_client_lock(ct, NULL);
3587 3587 }
3588 3588 }
3589 3589 /*
3590 3590 * Mark the mdi_pathinfo node state as transient
3591 3591 */
3592 3592 MDI_PI_LOCK(pip);
3593 3593 MDI_PI_SET_OFFLINING(pip);
3594 3594 break;
3595 3595 }
3596 3596 MDI_PI_UNLOCK(pip);
3597 3597 MDI_CLIENT_UNSTABLE(ct);
3598 3598 i_mdi_client_unlock(ct);
3599 3599
3600 3600 f = vh->vh_ops->vo_pi_state_change;
3601 3601 if (f != NULL)
3602 3602 rv = (*f)(vh->vh_dip, pip, state, 0, flag);
3603 3603
3604 3604 MDI_CLIENT_LOCK(ct);
3605 3605 MDI_PI_LOCK(pip);
3606 3606 if (rv == MDI_NOT_SUPPORTED) {
3607 3607 MDI_CLIENT_SET_DEV_NOT_SUPPORTED(ct);
3608 3608 }
3609 3609 if (rv != MDI_SUCCESS) {
3610 3610 MDI_DEBUG(2, (MDI_WARN, ct->ct_dip,
3611 3611 "vo_pi_state_change failed: rv %x", rv));
3612 3612 }
3613 3613 if (MDI_PI_IS_TRANSIENT(pip)) {
3614 3614 if (rv == MDI_SUCCESS) {
3615 3615 MDI_PI_CLEAR_TRANSIENT(pip);
3616 3616 } else {
3617 3617 MDI_PI(pip)->pi_state = MDI_PI_OLD_STATE(pip);
3618 3618 }
3619 3619 }
3620 3620
3621 3621 /*
3622 3622 * Wake anyone waiting for this mdi_pathinfo node
3623 3623 */
3624 3624 cv_broadcast(&MDI_PI(pip)->pi_state_cv);
3625 3625 MDI_PI_UNLOCK(pip);
3626 3626
3627 3627 /*
3628 3628 * Mark the client device as stable
3629 3629 */
3630 3630 MDI_CLIENT_STABLE(ct);
3631 3631 if (rv == MDI_SUCCESS) {
3632 3632 if (ct->ct_unstable == 0) {
3633 3633 cdip = ct->ct_dip;
3634 3634
3635 3635 /*
3636 3636 * Onlining the mdi_pathinfo node will impact the
3637 3637 * client state Update the client and dev_info node
3638 3638 * state accordingly
3639 3639 */
3640 3640 rv = NDI_SUCCESS;
3641 3641 i_mdi_client_update_state(ct);
3642 3642 switch (MDI_CLIENT_STATE(ct)) {
3643 3643 case MDI_CLIENT_STATE_OPTIMAL:
3644 3644 case MDI_CLIENT_STATE_DEGRADED:
3645 3645 if (cdip && !i_ddi_devi_attached(cdip) &&
3646 3646 ((state == MDI_PATHINFO_STATE_ONLINE) ||
3647 3647 (state == MDI_PATHINFO_STATE_STANDBY))) {
3648 3648
3649 3649 /*
3650 3650 * Must do ndi_devi_online() through
3651 3651 * hotplug thread for deferred
3652 3652 * attach mechanism to work
3653 3653 */
3654 3654 MDI_CLIENT_UNLOCK(ct);
3655 3655 rv = ndi_devi_online(cdip, 0);
3656 3656 MDI_CLIENT_LOCK(ct);
3657 3657 if ((rv != NDI_SUCCESS) &&
3658 3658 (MDI_CLIENT_STATE(ct) ==
3659 3659 MDI_CLIENT_STATE_DEGRADED)) {
3660 3660 MDI_DEBUG(1, (MDI_WARN, cdip,
3661 3661 "!ndi_devi_online failed "
3662 3662 "error %x", rv));
3663 3663 }
3664 3664 rv = NDI_SUCCESS;
3665 3665 }
3666 3666 break;
3667 3667
3668 3668 case MDI_CLIENT_STATE_FAILED:
3669 3669 /*
3670 3670 * This is the last path case for
3671 3671 * non-user initiated events.
3672 3672 */
3673 3673 if (((flag & NDI_USER_REQ) == 0) &&
3674 3674 cdip && (i_ddi_node_state(cdip) >=
3675 3675 DS_INITIALIZED)) {
3676 3676 MDI_CLIENT_UNLOCK(ct);
3677 3677 rv = ndi_devi_offline(cdip,
3678 3678 NDI_DEVFS_CLEAN);
3679 3679 MDI_CLIENT_LOCK(ct);
3680 3680
3681 3681 if (rv != NDI_SUCCESS) {
3682 3682 /*
3683 3683 * ndi_devi_offline failed.
3684 3684 * Reset client flags to
3685 3685 * online as the path could not
3686 3686 * be offlined.
3687 3687 */
3688 3688 MDI_DEBUG(1, (MDI_WARN, cdip,
3689 3689 "!ndi_devi_offline failed: "
3690 3690 "error %x", rv));
3691 3691 MDI_CLIENT_SET_ONLINE(ct);
3692 3692 }
3693 3693 }
3694 3694 break;
3695 3695 }
3696 3696 /*
3697 3697 * Convert to MDI error code
3698 3698 */
3699 3699 switch (rv) {
3700 3700 case NDI_SUCCESS:
3701 3701 MDI_CLIENT_SET_REPORT_DEV_NEEDED(ct);
3702 3702 i_mdi_report_path_state(ct, pip);
3703 3703 rv = MDI_SUCCESS;
3704 3704 break;
3705 3705 case NDI_BUSY:
3706 3706 rv = MDI_BUSY;
3707 3707 break;
3708 3708 default:
3709 3709 rv = MDI_FAILURE;
3710 3710 break;
3711 3711 }
3712 3712 }
3713 3713 }
3714 3714 MDI_CLIENT_UNLOCK(ct);
3715 3715
3716 3716 state_change_exit:
3717 3717 /*
3718 3718 * Mark the pHCI as stable again.
3719 3719 */
3720 3720 MDI_PHCI_LOCK(ph);
3721 3721 MDI_PHCI_STABLE(ph);
3722 3722 MDI_PHCI_UNLOCK(ph);
3723 3723 return (rv);
3724 3724 }
3725 3725
3726 3726 /*
3727 3727 * mdi_pi_online():
3728 3728 * Place the path_info node in the online state. The path is
3729 3729 * now available to be selected by mdi_select_path() for
3730 3730 * transporting I/O requests to client devices.
3731 3731 * Return Values:
3732 3732 * MDI_SUCCESS
3733 3733 * MDI_FAILURE
3734 3734 */
3735 3735 int
3736 3736 mdi_pi_online(mdi_pathinfo_t *pip, int flags)
3737 3737 {
3738 3738 mdi_client_t *ct = MDI_PI(pip)->pi_client;
3739 3739 int client_held = 0;
3740 3740 int rv;
3741 3741
3742 3742 ASSERT(ct != NULL);
3743 3743 rv = i_mdi_pi_state_change(pip, MDI_PATHINFO_STATE_ONLINE, flags);
3744 3744 if (rv != MDI_SUCCESS)
3745 3745 return (rv);
3746 3746
3747 3747 MDI_PI_LOCK(pip);
3748 3748 if (MDI_PI(pip)->pi_pm_held == 0) {
3749 3749 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
3750 3750 "i_mdi_pm_hold_pip %p", (void *)pip));
3751 3751 i_mdi_pm_hold_pip(pip);
3752 3752 client_held = 1;
3753 3753 }
3754 3754 MDI_PI_UNLOCK(pip);
3755 3755
3756 3756 if (client_held) {
3757 3757 MDI_CLIENT_LOCK(ct);
3758 3758 if (ct->ct_power_cnt == 0) {
3759 3759 rv = i_mdi_power_all_phci(ct);
3760 3760 }
3761 3761
3762 3762 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
3763 3763 "i_mdi_pm_hold_client %p", (void *)ct));
3764 3764 i_mdi_pm_hold_client(ct, 1);
3765 3765 MDI_CLIENT_UNLOCK(ct);
3766 3766 }
3767 3767
3768 3768 return (rv);
3769 3769 }
3770 3770
3771 3771 /*
3772 3772 * mdi_pi_standby():
3773 3773 * Place the mdi_pathinfo node in standby state
3774 3774 *
3775 3775 * Return Values:
3776 3776 * MDI_SUCCESS
3777 3777 * MDI_FAILURE
3778 3778 */
3779 3779 int
3780 3780 mdi_pi_standby(mdi_pathinfo_t *pip, int flags)
3781 3781 {
3782 3782 return (i_mdi_pi_state_change(pip, MDI_PATHINFO_STATE_STANDBY, flags));
3783 3783 }
3784 3784
3785 3785 /*
3786 3786 * mdi_pi_fault():
3787 3787 * Place the mdi_pathinfo node in fault'ed state
3788 3788 * Return Values:
3789 3789 * MDI_SUCCESS
3790 3790 * MDI_FAILURE
3791 3791 */
3792 3792 int
3793 3793 mdi_pi_fault(mdi_pathinfo_t *pip, int flags)
3794 3794 {
3795 3795 return (i_mdi_pi_state_change(pip, MDI_PATHINFO_STATE_FAULT, flags));
3796 3796 }
3797 3797
3798 3798 /*
3799 3799 * mdi_pi_offline():
3800 3800 * Offline a mdi_pathinfo node.
3801 3801 * Return Values:
3802 3802 * MDI_SUCCESS
3803 3803 * MDI_FAILURE
3804 3804 */
3805 3805 int
3806 3806 mdi_pi_offline(mdi_pathinfo_t *pip, int flags)
3807 3807 {
3808 3808 int ret, client_held = 0;
3809 3809 mdi_client_t *ct;
3810 3810
3811 3811 /*
3812 3812 * Original code overloaded NDI_DEVI_REMOVE to this interface, and
3813 3813 * used it to mean "user initiated operation" (i.e. devctl). Callers
3814 3814 * should now just use NDI_USER_REQ.
3815 3815 */
3816 3816 if (flags & NDI_DEVI_REMOVE) {
3817 3817 flags &= ~NDI_DEVI_REMOVE;
3818 3818 flags |= NDI_USER_REQ;
3819 3819 }
3820 3820
3821 3821 ret = i_mdi_pi_state_change(pip, MDI_PATHINFO_STATE_OFFLINE, flags);
3822 3822
3823 3823 if (ret == MDI_SUCCESS) {
3824 3824 MDI_PI_LOCK(pip);
3825 3825 if (MDI_PI(pip)->pi_pm_held) {
3826 3826 client_held = 1;
3827 3827 }
3828 3828 MDI_PI_UNLOCK(pip);
3829 3829
3830 3830 if (client_held) {
3831 3831 ct = MDI_PI(pip)->pi_client;
3832 3832 MDI_CLIENT_LOCK(ct);
3833 3833 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
3834 3834 "i_mdi_pm_rele_client\n"));
3835 3835 i_mdi_pm_rele_client(ct, 1);
3836 3836 MDI_CLIENT_UNLOCK(ct);
3837 3837 }
3838 3838 }
3839 3839
3840 3840 return (ret);
3841 3841 }
3842 3842
3843 3843 /*
3844 3844 * i_mdi_pi_offline():
3845 3845 * Offline a mdi_pathinfo node and call the vHCI driver's callback
3846 3846 */
3847 3847 static int
3848 3848 i_mdi_pi_offline(mdi_pathinfo_t *pip, int flags)
3849 3849 {
3850 3850 dev_info_t *vdip = NULL;
3851 3851 mdi_vhci_t *vh = NULL;
3852 3852 mdi_client_t *ct = NULL;
3853 3853 int (*f)();
3854 3854 int rv;
3855 3855
3856 3856 MDI_PI_LOCK(pip);
3857 3857 ct = MDI_PI(pip)->pi_client;
3858 3858 ASSERT(ct != NULL);
3859 3859
3860 3860 while (MDI_PI(pip)->pi_ref_cnt != 0) {
3861 3861 /*
3862 3862 * Give a chance for pending I/Os to complete.
3863 3863 */
3864 3864 MDI_DEBUG(1, (MDI_NOTE, ct->ct_dip,
3865 3865 "!%d cmds still pending on path %s %p",
3866 3866 MDI_PI(pip)->pi_ref_cnt, mdi_pi_spathname(pip),
3867 3867 (void *)pip));
3868 3868 if (cv_reltimedwait(&MDI_PI(pip)->pi_ref_cv,
3869 3869 &MDI_PI(pip)->pi_mutex, drv_usectohz(60 * 1000000),
3870 3870 TR_CLOCK_TICK) == -1) {
3871 3871 /*
3872 3872 * The timeout time reached without ref_cnt being zero
3873 3873 * being signaled.
3874 3874 */
3875 3875 MDI_DEBUG(1, (MDI_NOTE, ct->ct_dip,
3876 3876 "!Timeout reached on path %s %p without the cond",
3877 3877 mdi_pi_spathname(pip), (void *)pip));
3878 3878 MDI_DEBUG(1, (MDI_NOTE, ct->ct_dip,
3879 3879 "!%d cmds still pending on path %s %p",
3880 3880 MDI_PI(pip)->pi_ref_cnt,
3881 3881 mdi_pi_spathname(pip), (void *)pip));
3882 3882 }
3883 3883 }
3884 3884 vh = ct->ct_vhci;
3885 3885 vdip = vh->vh_dip;
3886 3886
3887 3887 /*
3888 3888 * Notify vHCI that has registered this event
3889 3889 */
3890 3890 ASSERT(vh->vh_ops);
3891 3891 f = vh->vh_ops->vo_pi_state_change;
3892 3892
3893 3893 if (f != NULL) {
3894 3894 MDI_PI_UNLOCK(pip);
3895 3895 if ((rv = (*f)(vdip, pip, MDI_PATHINFO_STATE_OFFLINE, 0,
3896 3896 flags)) != MDI_SUCCESS) {
3897 3897 MDI_DEBUG(1, (MDI_WARN, ct->ct_dip,
3898 3898 "!vo_path_offline failed: vdip %s%d %p: path %s %p",
3899 3899 ddi_driver_name(vdip), ddi_get_instance(vdip),
3900 3900 (void *)vdip, mdi_pi_spathname(pip), (void *)pip));
3901 3901 }
3902 3902 MDI_PI_LOCK(pip);
3903 3903 }
3904 3904
3905 3905 /*
3906 3906 * Set the mdi_pathinfo node state and clear the transient condition
3907 3907 */
3908 3908 MDI_PI_SET_OFFLINE(pip);
3909 3909 cv_broadcast(&MDI_PI(pip)->pi_state_cv);
3910 3910 MDI_PI_UNLOCK(pip);
3911 3911
3912 3912 MDI_CLIENT_LOCK(ct);
3913 3913 if (rv == MDI_SUCCESS) {
3914 3914 if (ct->ct_unstable == 0) {
3915 3915 dev_info_t *cdip = ct->ct_dip;
3916 3916
3917 3917 /*
3918 3918 * Onlining the mdi_pathinfo node will impact the
3919 3919 * client state Update the client and dev_info node
3920 3920 * state accordingly
3921 3921 */
3922 3922 i_mdi_client_update_state(ct);
3923 3923 rv = NDI_SUCCESS;
3924 3924 if (MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_FAILED) {
3925 3925 if (cdip &&
3926 3926 (i_ddi_node_state(cdip) >=
3927 3927 DS_INITIALIZED)) {
3928 3928 MDI_CLIENT_UNLOCK(ct);
3929 3929 rv = ndi_devi_offline(cdip,
3930 3930 NDI_DEVFS_CLEAN);
3931 3931 MDI_CLIENT_LOCK(ct);
3932 3932 if (rv != NDI_SUCCESS) {
3933 3933 /*
3934 3934 * ndi_devi_offline failed.
3935 3935 * Reset client flags to
3936 3936 * online.
3937 3937 */
3938 3938 MDI_DEBUG(4, (MDI_WARN, cdip,
3939 3939 "ndi_devi_offline failed: "
3940 3940 "error %x", rv));
3941 3941 MDI_CLIENT_SET_ONLINE(ct);
3942 3942 }
3943 3943 }
3944 3944 }
3945 3945 /*
3946 3946 * Convert to MDI error code
3947 3947 */
3948 3948 switch (rv) {
3949 3949 case NDI_SUCCESS:
3950 3950 rv = MDI_SUCCESS;
3951 3951 break;
3952 3952 case NDI_BUSY:
3953 3953 rv = MDI_BUSY;
3954 3954 break;
3955 3955 default:
3956 3956 rv = MDI_FAILURE;
3957 3957 break;
3958 3958 }
3959 3959 }
3960 3960 MDI_CLIENT_SET_REPORT_DEV_NEEDED(ct);
3961 3961 i_mdi_report_path_state(ct, pip);
3962 3962 }
3963 3963
3964 3964 MDI_CLIENT_UNLOCK(ct);
3965 3965
3966 3966 /*
3967 3967 * Change in the mdi_pathinfo node state will impact the client state
3968 3968 */
3969 3969 MDI_DEBUG(2, (MDI_NOTE, ct->ct_dip,
3970 3970 "ct = %p pip = %p", (void *)ct, (void *)pip));
3971 3971 return (rv);
3972 3972 }
3973 3973
3974 3974 /*
3975 3975 * i_mdi_pi_online():
3976 3976 * Online a mdi_pathinfo node and call the vHCI driver's callback
3977 3977 */
3978 3978 static int
3979 3979 i_mdi_pi_online(mdi_pathinfo_t *pip, int flags)
3980 3980 {
3981 3981 mdi_vhci_t *vh = NULL;
3982 3982 mdi_client_t *ct = NULL;
3983 3983 mdi_phci_t *ph;
3984 3984 int (*f)();
3985 3985 int rv;
3986 3986
3987 3987 MDI_PI_LOCK(pip);
3988 3988 ph = MDI_PI(pip)->pi_phci;
3989 3989 vh = ph->ph_vhci;
3990 3990 ct = MDI_PI(pip)->pi_client;
3991 3991 MDI_PI_SET_ONLINING(pip)
3992 3992 MDI_PI_UNLOCK(pip);
3993 3993 f = vh->vh_ops->vo_pi_state_change;
3994 3994 if (f != NULL)
3995 3995 rv = (*f)(vh->vh_dip, pip, MDI_PATHINFO_STATE_ONLINE, 0,
3996 3996 flags);
3997 3997 MDI_CLIENT_LOCK(ct);
3998 3998 MDI_PI_LOCK(pip);
3999 3999 cv_broadcast(&MDI_PI(pip)->pi_state_cv);
4000 4000 MDI_PI_UNLOCK(pip);
4001 4001 if (rv == MDI_SUCCESS) {
4002 4002 dev_info_t *cdip = ct->ct_dip;
4003 4003
4004 4004 rv = MDI_SUCCESS;
4005 4005 i_mdi_client_update_state(ct);
4006 4006 if (MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_OPTIMAL ||
4007 4007 MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_DEGRADED) {
4008 4008 if (cdip && !i_ddi_devi_attached(cdip)) {
4009 4009 MDI_CLIENT_UNLOCK(ct);
4010 4010 rv = ndi_devi_online(cdip, 0);
4011 4011 MDI_CLIENT_LOCK(ct);
4012 4012 if ((rv != NDI_SUCCESS) &&
4013 4013 (MDI_CLIENT_STATE(ct) ==
4014 4014 MDI_CLIENT_STATE_DEGRADED)) {
4015 4015 MDI_CLIENT_SET_OFFLINE(ct);
4016 4016 }
4017 4017 if (rv != NDI_SUCCESS) {
4018 4018 /* Reset the path state */
4019 4019 MDI_PI_LOCK(pip);
4020 4020 MDI_PI(pip)->pi_state =
4021 4021 MDI_PI_OLD_STATE(pip);
4022 4022 MDI_PI_UNLOCK(pip);
4023 4023 }
4024 4024 }
4025 4025 }
4026 4026 switch (rv) {
4027 4027 case NDI_SUCCESS:
4028 4028 MDI_CLIENT_SET_REPORT_DEV_NEEDED(ct);
4029 4029 i_mdi_report_path_state(ct, pip);
4030 4030 rv = MDI_SUCCESS;
4031 4031 break;
4032 4032 case NDI_BUSY:
4033 4033 rv = MDI_BUSY;
4034 4034 break;
4035 4035 default:
4036 4036 rv = MDI_FAILURE;
4037 4037 break;
4038 4038 }
4039 4039 } else {
4040 4040 /* Reset the path state */
4041 4041 MDI_PI_LOCK(pip);
4042 4042 MDI_PI(pip)->pi_state = MDI_PI_OLD_STATE(pip);
4043 4043 MDI_PI_UNLOCK(pip);
4044 4044 }
4045 4045 MDI_CLIENT_UNLOCK(ct);
4046 4046 return (rv);
4047 4047 }
4048 4048
4049 4049 /*
4050 4050 * mdi_pi_get_node_name():
4051 4051 * Get the name associated with a mdi_pathinfo node.
4052 4052 * Since pathinfo nodes are not directly named, we
4053 4053 * return the node_name of the client.
4054 4054 *
4055 4055 * Return Values:
4056 4056 * char *
4057 4057 */
4058 4058 char *
4059 4059 mdi_pi_get_node_name(mdi_pathinfo_t *pip)
4060 4060 {
4061 4061 mdi_client_t *ct;
4062 4062
4063 4063 if (pip == NULL)
4064 4064 return (NULL);
4065 4065 ct = MDI_PI(pip)->pi_client;
4066 4066 if ((ct == NULL) || (ct->ct_dip == NULL))
4067 4067 return (NULL);
4068 4068 return (ddi_node_name(ct->ct_dip));
4069 4069 }
4070 4070
4071 4071 /*
4072 4072 * mdi_pi_get_addr():
4073 4073 * Get the unit address associated with a mdi_pathinfo node
4074 4074 *
4075 4075 * Return Values:
4076 4076 * char *
4077 4077 */
4078 4078 char *
4079 4079 mdi_pi_get_addr(mdi_pathinfo_t *pip)
4080 4080 {
4081 4081 if (pip == NULL)
4082 4082 return (NULL);
4083 4083
4084 4084 return (MDI_PI(pip)->pi_addr);
4085 4085 }
4086 4086
4087 4087 /*
4088 4088 * mdi_pi_get_path_instance():
4089 4089 * Get the 'path_instance' of a mdi_pathinfo node
4090 4090 *
4091 4091 * Return Values:
4092 4092 * path_instance
4093 4093 */
4094 4094 int
4095 4095 mdi_pi_get_path_instance(mdi_pathinfo_t *pip)
4096 4096 {
4097 4097 if (pip == NULL)
4098 4098 return (0);
4099 4099
4100 4100 return (MDI_PI(pip)->pi_path_instance);
4101 4101 }
4102 4102
4103 4103 /*
4104 4104 * mdi_pi_pathname():
4105 4105 * Return pointer to path to pathinfo node.
4106 4106 */
4107 4107 char *
4108 4108 mdi_pi_pathname(mdi_pathinfo_t *pip)
4109 4109 {
4110 4110 if (pip == NULL)
4111 4111 return (NULL);
4112 4112 return (mdi_pi_pathname_by_instance(mdi_pi_get_path_instance(pip)));
4113 4113 }
4114 4114
4115 4115 /*
4116 4116 * mdi_pi_spathname():
4117 4117 * Return pointer to shortpath to pathinfo node. Used for debug
4118 4118 * messages, so return "" instead of NULL when unknown.
4119 4119 */
4120 4120 char *
4121 4121 mdi_pi_spathname(mdi_pathinfo_t *pip)
4122 4122 {
4123 4123 char *spath = "";
4124 4124
4125 4125 if (pip) {
4126 4126 spath = mdi_pi_spathname_by_instance(
4127 4127 mdi_pi_get_path_instance(pip));
4128 4128 if (spath == NULL)
4129 4129 spath = "";
4130 4130 }
4131 4131 return (spath);
4132 4132 }
4133 4133
4134 4134 char *
4135 4135 mdi_pi_pathname_obp(mdi_pathinfo_t *pip, char *path)
4136 4136 {
4137 4137 char *obp_path = NULL;
4138 4138 if ((pip == NULL) || (path == NULL))
4139 4139 return (NULL);
4140 4140
4141 4141 if (mdi_prop_lookup_string(pip, "obp-path", &obp_path) == MDI_SUCCESS) {
4142 4142 (void) strcpy(path, obp_path);
4143 4143 (void) mdi_prop_free(obp_path);
4144 4144 } else {
4145 4145 path = NULL;
4146 4146 }
4147 4147 return (path);
4148 4148 }
4149 4149
4150 4150 int
4151 4151 mdi_pi_pathname_obp_set(mdi_pathinfo_t *pip, char *component)
4152 4152 {
4153 4153 dev_info_t *pdip;
4154 4154 char *obp_path = NULL;
4155 4155 int rc = MDI_FAILURE;
4156 4156
4157 4157 if (pip == NULL)
4158 4158 return (MDI_FAILURE);
4159 4159
4160 4160 pdip = mdi_pi_get_phci(pip);
4161 4161 if (pdip == NULL)
4162 4162 return (MDI_FAILURE);
4163 4163
4164 4164 obp_path = kmem_zalloc(MAXPATHLEN, KM_SLEEP);
4165 4165
4166 4166 if (ddi_pathname_obp(pdip, obp_path) == NULL) {
4167 4167 (void) ddi_pathname(pdip, obp_path);
4168 4168 }
4169 4169
4170 4170 if (component) {
4171 4171 (void) strncat(obp_path, "/", MAXPATHLEN);
4172 4172 (void) strncat(obp_path, component, MAXPATHLEN);
4173 4173 }
4174 4174 rc = mdi_prop_update_string(pip, "obp-path", obp_path);
4175 4175
4176 4176 if (obp_path)
4177 4177 kmem_free(obp_path, MAXPATHLEN);
4178 4178 return (rc);
4179 4179 }
4180 4180
4181 4181 /*
4182 4182 * mdi_pi_get_client():
4183 4183 * Get the client devinfo associated with a mdi_pathinfo node
4184 4184 *
4185 4185 * Return Values:
4186 4186 * Handle to client device dev_info node
4187 4187 */
4188 4188 dev_info_t *
4189 4189 mdi_pi_get_client(mdi_pathinfo_t *pip)
4190 4190 {
4191 4191 dev_info_t *dip = NULL;
4192 4192 if (pip) {
4193 4193 dip = MDI_PI(pip)->pi_client->ct_dip;
4194 4194 }
4195 4195 return (dip);
4196 4196 }
4197 4197
4198 4198 /*
4199 4199 * mdi_pi_get_phci():
4200 4200 * Get the pHCI devinfo associated with the mdi_pathinfo node
4201 4201 * Return Values:
4202 4202 * Handle to dev_info node
4203 4203 */
4204 4204 dev_info_t *
4205 4205 mdi_pi_get_phci(mdi_pathinfo_t *pip)
4206 4206 {
4207 4207 dev_info_t *dip = NULL;
4208 4208 mdi_phci_t *ph;
4209 4209
4210 4210 if (pip) {
4211 4211 ph = MDI_PI(pip)->pi_phci;
4212 4212 if (ph)
4213 4213 dip = ph->ph_dip;
4214 4214 }
4215 4215 return (dip);
4216 4216 }
4217 4217
4218 4218 /*
4219 4219 * mdi_pi_get_client_private():
4220 4220 * Get the client private information associated with the
4221 4221 * mdi_pathinfo node
4222 4222 */
4223 4223 void *
4224 4224 mdi_pi_get_client_private(mdi_pathinfo_t *pip)
4225 4225 {
4226 4226 void *cprivate = NULL;
4227 4227 if (pip) {
4228 4228 cprivate = MDI_PI(pip)->pi_cprivate;
4229 4229 }
4230 4230 return (cprivate);
4231 4231 }
4232 4232
4233 4233 /*
4234 4234 * mdi_pi_set_client_private():
4235 4235 * Set the client private information in the mdi_pathinfo node
4236 4236 */
4237 4237 void
4238 4238 mdi_pi_set_client_private(mdi_pathinfo_t *pip, void *priv)
4239 4239 {
4240 4240 if (pip) {
4241 4241 MDI_PI(pip)->pi_cprivate = priv;
4242 4242 }
4243 4243 }
4244 4244
4245 4245 /*
4246 4246 * mdi_pi_get_phci_private():
4247 4247 * Get the pHCI private information associated with the
4248 4248 * mdi_pathinfo node
4249 4249 */
4250 4250 caddr_t
4251 4251 mdi_pi_get_phci_private(mdi_pathinfo_t *pip)
4252 4252 {
4253 4253 caddr_t pprivate = NULL;
4254 4254
4255 4255 if (pip) {
4256 4256 pprivate = MDI_PI(pip)->pi_pprivate;
4257 4257 }
4258 4258 return (pprivate);
4259 4259 }
4260 4260
4261 4261 /*
4262 4262 * mdi_pi_set_phci_private():
4263 4263 * Set the pHCI private information in the mdi_pathinfo node
4264 4264 */
4265 4265 void
4266 4266 mdi_pi_set_phci_private(mdi_pathinfo_t *pip, caddr_t priv)
4267 4267 {
4268 4268 if (pip) {
4269 4269 MDI_PI(pip)->pi_pprivate = priv;
4270 4270 }
4271 4271 }
4272 4272
4273 4273 /*
4274 4274 * mdi_pi_get_state():
4275 4275 * Get the mdi_pathinfo node state. Transient states are internal
4276 4276 * and not provided to the users
4277 4277 */
4278 4278 mdi_pathinfo_state_t
4279 4279 mdi_pi_get_state(mdi_pathinfo_t *pip)
4280 4280 {
4281 4281 mdi_pathinfo_state_t state = MDI_PATHINFO_STATE_INIT;
4282 4282
4283 4283 if (pip) {
4284 4284 if (MDI_PI_IS_TRANSIENT(pip)) {
4285 4285 /*
4286 4286 * mdi_pathinfo is in state transition. Return the
4287 4287 * last good state.
4288 4288 */
4289 4289 state = MDI_PI_OLD_STATE(pip);
4290 4290 } else {
4291 4291 state = MDI_PI_STATE(pip);
4292 4292 }
4293 4293 }
4294 4294 return (state);
4295 4295 }
4296 4296
4297 4297 /*
4298 4298 * mdi_pi_get_flags():
4299 4299 * Get the mdi_pathinfo node flags.
4300 4300 */
4301 4301 uint_t
4302 4302 mdi_pi_get_flags(mdi_pathinfo_t *pip)
4303 4303 {
4304 4304 return (pip ? MDI_PI(pip)->pi_flags : 0);
4305 4305 }
4306 4306
4307 4307 /*
4308 4308 * Note that the following function needs to be the new interface for
4309 4309 * mdi_pi_get_state when mpxio gets integrated to ON.
4310 4310 */
4311 4311 int
4312 4312 mdi_pi_get_state2(mdi_pathinfo_t *pip, mdi_pathinfo_state_t *state,
4313 4313 uint32_t *ext_state)
4314 4314 {
4315 4315 *state = MDI_PATHINFO_STATE_INIT;
4316 4316
4317 4317 if (pip) {
4318 4318 if (MDI_PI_IS_TRANSIENT(pip)) {
4319 4319 /*
4320 4320 * mdi_pathinfo is in state transition. Return the
4321 4321 * last good state.
4322 4322 */
4323 4323 *state = MDI_PI_OLD_STATE(pip);
4324 4324 *ext_state = MDI_PI_OLD_EXT_STATE(pip);
4325 4325 } else {
4326 4326 *state = MDI_PI_STATE(pip);
4327 4327 *ext_state = MDI_PI_EXT_STATE(pip);
4328 4328 }
4329 4329 }
4330 4330 return (MDI_SUCCESS);
4331 4331 }
4332 4332
4333 4333 /*
4334 4334 * mdi_pi_get_preferred:
4335 4335 * Get the preferred path flag
4336 4336 */
4337 4337 int
4338 4338 mdi_pi_get_preferred(mdi_pathinfo_t *pip)
4339 4339 {
4340 4340 if (pip) {
4341 4341 return (MDI_PI(pip)->pi_preferred);
4342 4342 }
4343 4343 return (0);
4344 4344 }
4345 4345
4346 4346 /*
4347 4347 * mdi_pi_set_preferred:
4348 4348 * Set the preferred path flag
4349 4349 */
4350 4350 void
4351 4351 mdi_pi_set_preferred(mdi_pathinfo_t *pip, int preferred)
4352 4352 {
4353 4353 if (pip) {
4354 4354 MDI_PI(pip)->pi_preferred = preferred;
4355 4355 }
4356 4356 }
4357 4357
4358 4358 /*
4359 4359 * mdi_pi_set_state():
4360 4360 * Set the mdi_pathinfo node state
4361 4361 */
4362 4362 void
4363 4363 mdi_pi_set_state(mdi_pathinfo_t *pip, mdi_pathinfo_state_t state)
4364 4364 {
4365 4365 uint32_t ext_state;
4366 4366
4367 4367 if (pip) {
4368 4368 ext_state = MDI_PI(pip)->pi_state & MDI_PATHINFO_EXT_STATE_MASK;
4369 4369 MDI_PI(pip)->pi_state = state;
4370 4370 MDI_PI(pip)->pi_state |= ext_state;
4371 4371
4372 4372 /* Path has changed state, invalidate DINFOCACHE snap shot. */
4373 4373 i_ddi_di_cache_invalidate();
4374 4374 }
4375 4375 }
4376 4376
4377 4377 /*
4378 4378 * Property functions:
4379 4379 */
4380 4380 int
4381 4381 i_map_nvlist_error_to_mdi(int val)
4382 4382 {
4383 4383 int rv;
4384 4384
4385 4385 switch (val) {
4386 4386 case 0:
4387 4387 rv = DDI_PROP_SUCCESS;
4388 4388 break;
4389 4389 case EINVAL:
4390 4390 case ENOTSUP:
4391 4391 rv = DDI_PROP_INVAL_ARG;
4392 4392 break;
4393 4393 case ENOMEM:
4394 4394 rv = DDI_PROP_NO_MEMORY;
4395 4395 break;
4396 4396 default:
4397 4397 rv = DDI_PROP_NOT_FOUND;
4398 4398 break;
4399 4399 }
4400 4400 return (rv);
4401 4401 }
4402 4402
4403 4403 /*
4404 4404 * mdi_pi_get_next_prop():
4405 4405 * Property walk function. The caller should hold mdi_pi_lock()
4406 4406 * and release by calling mdi_pi_unlock() at the end of walk to
4407 4407 * get a consistent value.
4408 4408 */
4409 4409 nvpair_t *
4410 4410 mdi_pi_get_next_prop(mdi_pathinfo_t *pip, nvpair_t *prev)
4411 4411 {
4412 4412 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4413 4413 return (NULL);
4414 4414 }
4415 4415 ASSERT(MDI_PI_LOCKED(pip));
4416 4416 return (nvlist_next_nvpair(MDI_PI(pip)->pi_prop, prev));
4417 4417 }
4418 4418
4419 4419 /*
4420 4420 * mdi_prop_remove():
4421 4421 * Remove the named property from the named list.
4422 4422 */
4423 4423 int
4424 4424 mdi_prop_remove(mdi_pathinfo_t *pip, char *name)
4425 4425 {
4426 4426 if (pip == NULL) {
4427 4427 return (DDI_PROP_NOT_FOUND);
4428 4428 }
4429 4429 ASSERT(!MDI_PI_LOCKED(pip));
4430 4430 MDI_PI_LOCK(pip);
4431 4431 if (MDI_PI(pip)->pi_prop == NULL) {
4432 4432 MDI_PI_UNLOCK(pip);
4433 4433 return (DDI_PROP_NOT_FOUND);
4434 4434 }
4435 4435 if (name) {
4436 4436 (void) nvlist_remove_all(MDI_PI(pip)->pi_prop, name);
4437 4437 } else {
4438 4438 char nvp_name[MAXNAMELEN];
4439 4439 nvpair_t *nvp;
4440 4440 nvp = nvlist_next_nvpair(MDI_PI(pip)->pi_prop, NULL);
4441 4441 while (nvp) {
4442 4442 nvpair_t *next;
4443 4443 next = nvlist_next_nvpair(MDI_PI(pip)->pi_prop, nvp);
4444 4444 (void) snprintf(nvp_name, sizeof(nvp_name), "%s",
4445 4445 nvpair_name(nvp));
4446 4446 (void) nvlist_remove_all(MDI_PI(pip)->pi_prop,
4447 4447 nvp_name);
4448 4448 nvp = next;
4449 4449 }
4450 4450 }
4451 4451 MDI_PI_UNLOCK(pip);
4452 4452 return (DDI_PROP_SUCCESS);
4453 4453 }
4454 4454
4455 4455 /*
4456 4456 * mdi_prop_size():
4457 4457 * Get buffer size needed to pack the property data.
4458 4458 * Caller should hold the mdi_pathinfo_t lock to get a consistent
4459 4459 * buffer size.
4460 4460 */
4461 4461 int
4462 4462 mdi_prop_size(mdi_pathinfo_t *pip, size_t *buflenp)
4463 4463 {
4464 4464 int rv;
4465 4465 size_t bufsize;
4466 4466
4467 4467 *buflenp = 0;
4468 4468 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4469 4469 return (DDI_PROP_NOT_FOUND);
4470 4470 }
4471 4471 ASSERT(MDI_PI_LOCKED(pip));
4472 4472 rv = nvlist_size(MDI_PI(pip)->pi_prop,
4473 4473 &bufsize, NV_ENCODE_NATIVE);
4474 4474 *buflenp = bufsize;
4475 4475 return (i_map_nvlist_error_to_mdi(rv));
4476 4476 }
4477 4477
4478 4478 /*
4479 4479 * mdi_prop_pack():
4480 4480 * pack the property list. The caller should hold the
4481 4481 * mdi_pathinfo_t node to get a consistent data
4482 4482 */
4483 4483 int
4484 4484 mdi_prop_pack(mdi_pathinfo_t *pip, char **bufp, uint_t buflen)
4485 4485 {
4486 4486 int rv;
4487 4487 size_t bufsize;
4488 4488
4489 4489 if ((pip == NULL) || MDI_PI(pip)->pi_prop == NULL) {
4490 4490 return (DDI_PROP_NOT_FOUND);
4491 4491 }
4492 4492
4493 4493 ASSERT(MDI_PI_LOCKED(pip));
4494 4494
4495 4495 bufsize = buflen;
4496 4496 rv = nvlist_pack(MDI_PI(pip)->pi_prop, bufp, (size_t *)&bufsize,
4497 4497 NV_ENCODE_NATIVE, KM_SLEEP);
4498 4498
4499 4499 return (i_map_nvlist_error_to_mdi(rv));
4500 4500 }
4501 4501
4502 4502 /*
4503 4503 * mdi_prop_update_byte():
4504 4504 * Create/Update a byte property
4505 4505 */
4506 4506 int
4507 4507 mdi_prop_update_byte(mdi_pathinfo_t *pip, char *name, uchar_t data)
4508 4508 {
4509 4509 int rv;
4510 4510
4511 4511 if (pip == NULL) {
4512 4512 return (DDI_PROP_INVAL_ARG);
4513 4513 }
4514 4514 ASSERT(!MDI_PI_LOCKED(pip));
4515 4515 MDI_PI_LOCK(pip);
4516 4516 if (MDI_PI(pip)->pi_prop == NULL) {
4517 4517 MDI_PI_UNLOCK(pip);
4518 4518 return (DDI_PROP_NOT_FOUND);
4519 4519 }
4520 4520 rv = nvlist_add_byte(MDI_PI(pip)->pi_prop, name, data);
4521 4521 MDI_PI_UNLOCK(pip);
4522 4522 return (i_map_nvlist_error_to_mdi(rv));
4523 4523 }
4524 4524
4525 4525 /*
4526 4526 * mdi_prop_update_byte_array():
4527 4527 * Create/Update a byte array property
4528 4528 */
4529 4529 int
4530 4530 mdi_prop_update_byte_array(mdi_pathinfo_t *pip, char *name, uchar_t *data,
4531 4531 uint_t nelements)
4532 4532 {
4533 4533 int rv;
4534 4534
4535 4535 if (pip == NULL) {
4536 4536 return (DDI_PROP_INVAL_ARG);
4537 4537 }
4538 4538 ASSERT(!MDI_PI_LOCKED(pip));
4539 4539 MDI_PI_LOCK(pip);
4540 4540 if (MDI_PI(pip)->pi_prop == NULL) {
4541 4541 MDI_PI_UNLOCK(pip);
4542 4542 return (DDI_PROP_NOT_FOUND);
4543 4543 }
4544 4544 rv = nvlist_add_byte_array(MDI_PI(pip)->pi_prop, name, data, nelements);
4545 4545 MDI_PI_UNLOCK(pip);
4546 4546 return (i_map_nvlist_error_to_mdi(rv));
4547 4547 }
4548 4548
4549 4549 /*
4550 4550 * mdi_prop_update_int():
4551 4551 * Create/Update a 32 bit integer property
4552 4552 */
4553 4553 int
4554 4554 mdi_prop_update_int(mdi_pathinfo_t *pip, char *name, int data)
4555 4555 {
4556 4556 int rv;
4557 4557
4558 4558 if (pip == NULL) {
4559 4559 return (DDI_PROP_INVAL_ARG);
4560 4560 }
4561 4561 ASSERT(!MDI_PI_LOCKED(pip));
4562 4562 MDI_PI_LOCK(pip);
4563 4563 if (MDI_PI(pip)->pi_prop == NULL) {
4564 4564 MDI_PI_UNLOCK(pip);
4565 4565 return (DDI_PROP_NOT_FOUND);
4566 4566 }
4567 4567 rv = nvlist_add_int32(MDI_PI(pip)->pi_prop, name, (int32_t)data);
4568 4568 MDI_PI_UNLOCK(pip);
4569 4569 return (i_map_nvlist_error_to_mdi(rv));
4570 4570 }
4571 4571
4572 4572 /*
4573 4573 * mdi_prop_update_int64():
4574 4574 * Create/Update a 64 bit integer property
4575 4575 */
4576 4576 int
4577 4577 mdi_prop_update_int64(mdi_pathinfo_t *pip, char *name, int64_t data)
4578 4578 {
4579 4579 int rv;
4580 4580
4581 4581 if (pip == NULL) {
4582 4582 return (DDI_PROP_INVAL_ARG);
4583 4583 }
4584 4584 ASSERT(!MDI_PI_LOCKED(pip));
4585 4585 MDI_PI_LOCK(pip);
4586 4586 if (MDI_PI(pip)->pi_prop == NULL) {
4587 4587 MDI_PI_UNLOCK(pip);
4588 4588 return (DDI_PROP_NOT_FOUND);
4589 4589 }
4590 4590 rv = nvlist_add_int64(MDI_PI(pip)->pi_prop, name, data);
4591 4591 MDI_PI_UNLOCK(pip);
4592 4592 return (i_map_nvlist_error_to_mdi(rv));
4593 4593 }
4594 4594
4595 4595 /*
4596 4596 * mdi_prop_update_int_array():
4597 4597 * Create/Update a int array property
4598 4598 */
4599 4599 int
4600 4600 mdi_prop_update_int_array(mdi_pathinfo_t *pip, char *name, int *data,
4601 4601 uint_t nelements)
4602 4602 {
4603 4603 int rv;
4604 4604
4605 4605 if (pip == NULL) {
4606 4606 return (DDI_PROP_INVAL_ARG);
4607 4607 }
4608 4608 ASSERT(!MDI_PI_LOCKED(pip));
4609 4609 MDI_PI_LOCK(pip);
4610 4610 if (MDI_PI(pip)->pi_prop == NULL) {
4611 4611 MDI_PI_UNLOCK(pip);
4612 4612 return (DDI_PROP_NOT_FOUND);
4613 4613 }
4614 4614 rv = nvlist_add_int32_array(MDI_PI(pip)->pi_prop, name, (int32_t *)data,
4615 4615 nelements);
4616 4616 MDI_PI_UNLOCK(pip);
4617 4617 return (i_map_nvlist_error_to_mdi(rv));
4618 4618 }
4619 4619
4620 4620 /*
4621 4621 * mdi_prop_update_string():
4622 4622 * Create/Update a string property
4623 4623 */
4624 4624 int
4625 4625 mdi_prop_update_string(mdi_pathinfo_t *pip, char *name, char *data)
4626 4626 {
4627 4627 int rv;
4628 4628
4629 4629 if (pip == NULL) {
4630 4630 return (DDI_PROP_INVAL_ARG);
4631 4631 }
4632 4632 ASSERT(!MDI_PI_LOCKED(pip));
4633 4633 MDI_PI_LOCK(pip);
4634 4634 if (MDI_PI(pip)->pi_prop == NULL) {
4635 4635 MDI_PI_UNLOCK(pip);
4636 4636 return (DDI_PROP_NOT_FOUND);
4637 4637 }
4638 4638 rv = nvlist_add_string(MDI_PI(pip)->pi_prop, name, data);
4639 4639 MDI_PI_UNLOCK(pip);
4640 4640 return (i_map_nvlist_error_to_mdi(rv));
4641 4641 }
4642 4642
4643 4643 /*
4644 4644 * mdi_prop_update_string_array():
4645 4645 * Create/Update a string array property
4646 4646 */
4647 4647 int
4648 4648 mdi_prop_update_string_array(mdi_pathinfo_t *pip, char *name, char **data,
4649 4649 uint_t nelements)
4650 4650 {
4651 4651 int rv;
4652 4652
4653 4653 if (pip == NULL) {
4654 4654 return (DDI_PROP_INVAL_ARG);
4655 4655 }
4656 4656 ASSERT(!MDI_PI_LOCKED(pip));
4657 4657 MDI_PI_LOCK(pip);
4658 4658 if (MDI_PI(pip)->pi_prop == NULL) {
4659 4659 MDI_PI_UNLOCK(pip);
4660 4660 return (DDI_PROP_NOT_FOUND);
4661 4661 }
4662 4662 rv = nvlist_add_string_array(MDI_PI(pip)->pi_prop, name, data,
4663 4663 nelements);
4664 4664 MDI_PI_UNLOCK(pip);
4665 4665 return (i_map_nvlist_error_to_mdi(rv));
4666 4666 }
4667 4667
4668 4668 /*
4669 4669 * mdi_prop_lookup_byte():
4670 4670 * Look for byte property identified by name. The data returned
4671 4671 * is the actual property and valid as long as mdi_pathinfo_t node
4672 4672 * is alive.
4673 4673 */
4674 4674 int
4675 4675 mdi_prop_lookup_byte(mdi_pathinfo_t *pip, char *name, uchar_t *data)
4676 4676 {
4677 4677 int rv;
4678 4678
4679 4679 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4680 4680 return (DDI_PROP_NOT_FOUND);
4681 4681 }
4682 4682 rv = nvlist_lookup_byte(MDI_PI(pip)->pi_prop, name, data);
4683 4683 return (i_map_nvlist_error_to_mdi(rv));
4684 4684 }
4685 4685
4686 4686
4687 4687 /*
4688 4688 * mdi_prop_lookup_byte_array():
4689 4689 * Look for byte array property identified by name. The data
4690 4690 * returned is the actual property and valid as long as
4691 4691 * mdi_pathinfo_t node is alive.
4692 4692 */
4693 4693 int
4694 4694 mdi_prop_lookup_byte_array(mdi_pathinfo_t *pip, char *name, uchar_t **data,
4695 4695 uint_t *nelements)
4696 4696 {
4697 4697 int rv;
4698 4698
4699 4699 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4700 4700 return (DDI_PROP_NOT_FOUND);
4701 4701 }
4702 4702 rv = nvlist_lookup_byte_array(MDI_PI(pip)->pi_prop, name, data,
4703 4703 nelements);
4704 4704 return (i_map_nvlist_error_to_mdi(rv));
4705 4705 }
4706 4706
4707 4707 /*
4708 4708 * mdi_prop_lookup_int():
4709 4709 * Look for int property identified by name. The data returned
4710 4710 * is the actual property and valid as long as mdi_pathinfo_t
4711 4711 * node is alive.
4712 4712 */
4713 4713 int
4714 4714 mdi_prop_lookup_int(mdi_pathinfo_t *pip, char *name, int *data)
4715 4715 {
4716 4716 int rv;
4717 4717
4718 4718 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4719 4719 return (DDI_PROP_NOT_FOUND);
4720 4720 }
4721 4721 rv = nvlist_lookup_int32(MDI_PI(pip)->pi_prop, name, (int32_t *)data);
4722 4722 return (i_map_nvlist_error_to_mdi(rv));
4723 4723 }
4724 4724
4725 4725 /*
4726 4726 * mdi_prop_lookup_int64():
4727 4727 * Look for int64 property identified by name. The data returned
4728 4728 * is the actual property and valid as long as mdi_pathinfo_t node
4729 4729 * is alive.
4730 4730 */
4731 4731 int
4732 4732 mdi_prop_lookup_int64(mdi_pathinfo_t *pip, char *name, int64_t *data)
4733 4733 {
4734 4734 int rv;
4735 4735 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4736 4736 return (DDI_PROP_NOT_FOUND);
4737 4737 }
4738 4738 rv = nvlist_lookup_int64(MDI_PI(pip)->pi_prop, name, data);
4739 4739 return (i_map_nvlist_error_to_mdi(rv));
4740 4740 }
4741 4741
4742 4742 /*
4743 4743 * mdi_prop_lookup_int_array():
4744 4744 * Look for int array property identified by name. The data
4745 4745 * returned is the actual property and valid as long as
4746 4746 * mdi_pathinfo_t node is alive.
4747 4747 */
4748 4748 int
4749 4749 mdi_prop_lookup_int_array(mdi_pathinfo_t *pip, char *name, int **data,
4750 4750 uint_t *nelements)
4751 4751 {
4752 4752 int rv;
4753 4753
4754 4754 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4755 4755 return (DDI_PROP_NOT_FOUND);
4756 4756 }
4757 4757 rv = nvlist_lookup_int32_array(MDI_PI(pip)->pi_prop, name,
4758 4758 (int32_t **)data, nelements);
4759 4759 return (i_map_nvlist_error_to_mdi(rv));
4760 4760 }
4761 4761
4762 4762 /*
4763 4763 * mdi_prop_lookup_string():
4764 4764 * Look for string property identified by name. The data
4765 4765 * returned is the actual property and valid as long as
4766 4766 * mdi_pathinfo_t node is alive.
4767 4767 */
4768 4768 int
4769 4769 mdi_prop_lookup_string(mdi_pathinfo_t *pip, char *name, char **data)
4770 4770 {
4771 4771 int rv;
4772 4772
4773 4773 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4774 4774 return (DDI_PROP_NOT_FOUND);
4775 4775 }
4776 4776 rv = nvlist_lookup_string(MDI_PI(pip)->pi_prop, name, data);
4777 4777 return (i_map_nvlist_error_to_mdi(rv));
4778 4778 }
4779 4779
4780 4780 /*
4781 4781 * mdi_prop_lookup_string_array():
4782 4782 * Look for string array property identified by name. The data
4783 4783 * returned is the actual property and valid as long as
4784 4784 * mdi_pathinfo_t node is alive.
4785 4785 */
4786 4786 int
4787 4787 mdi_prop_lookup_string_array(mdi_pathinfo_t *pip, char *name, char ***data,
4788 4788 uint_t *nelements)
4789 4789 {
4790 4790 int rv;
4791 4791
4792 4792 if ((pip == NULL) || (MDI_PI(pip)->pi_prop == NULL)) {
4793 4793 return (DDI_PROP_NOT_FOUND);
4794 4794 }
4795 4795 rv = nvlist_lookup_string_array(MDI_PI(pip)->pi_prop, name, data,
4796 4796 nelements);
4797 4797 return (i_map_nvlist_error_to_mdi(rv));
4798 4798 }
4799 4799
4800 4800 /*
4801 4801 * mdi_prop_free():
4802 4802 * Symmetrical function to ddi_prop_free(). nvlist_lookup_xx()
4803 4803 * functions return the pointer to actual property data and not a
4804 4804 * copy of it. So the data returned is valid as long as
4805 4805 * mdi_pathinfo_t node is valid.
4806 4806 */
4807 4807 /*ARGSUSED*/
4808 4808 int
4809 4809 mdi_prop_free(void *data)
4810 4810 {
4811 4811 return (DDI_PROP_SUCCESS);
4812 4812 }
4813 4813
4814 4814 /*ARGSUSED*/
4815 4815 static void
4816 4816 i_mdi_report_path_state(mdi_client_t *ct, mdi_pathinfo_t *pip)
4817 4817 {
4818 4818 char *ct_path;
4819 4819 char *ct_status;
4820 4820 char *status;
4821 4821 dev_info_t *cdip = ct->ct_dip;
4822 4822 char lb_buf[64];
4823 4823 int report_lb_c = 0, report_lb_p = 0;
4824 4824
4825 4825 ASSERT(MDI_CLIENT_LOCKED(ct));
4826 4826 if ((cdip == NULL) || (ddi_get_instance(cdip) == -1) ||
4827 4827 (MDI_CLIENT_IS_REPORT_DEV_NEEDED(ct) == 0)) {
4828 4828 return;
4829 4829 }
4830 4830 if (MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_OPTIMAL) {
4831 4831 ct_status = "optimal";
4832 4832 report_lb_c = 1;
4833 4833 } else if (MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_DEGRADED) {
4834 4834 ct_status = "degraded";
4835 4835 } else if (MDI_CLIENT_STATE(ct) == MDI_CLIENT_STATE_FAILED) {
4836 4836 ct_status = "failed";
4837 4837 } else {
4838 4838 ct_status = "unknown";
4839 4839 }
4840 4840
4841 4841 lb_buf[0] = 0; /* not interested in load balancing config */
4842 4842
4843 4843 if (MDI_PI_FLAGS_IS_DEVICE_REMOVED(pip)) {
4844 4844 status = "removed";
4845 4845 } else if (MDI_PI_IS_OFFLINE(pip)) {
4846 4846 status = "offline";
4847 4847 } else if (MDI_PI_IS_ONLINE(pip)) {
4848 4848 status = "online";
4849 4849 report_lb_p = 1;
4850 4850 } else if (MDI_PI_IS_STANDBY(pip)) {
4851 4851 status = "standby";
4852 4852 } else if (MDI_PI_IS_FAULT(pip)) {
4853 4853 status = "faulted";
4854 4854 } else {
4855 4855 status = "unknown";
4856 4856 }
4857 4857
4858 4858 if (cdip) {
4859 4859 ct_path = kmem_alloc(MAXPATHLEN, KM_SLEEP);
4860 4860
4861 4861 /*
4862 4862 * NOTE: Keeping "multipath status: %s" and
4863 4863 * "Load balancing: %s" format unchanged in case someone
4864 4864 * scrubs /var/adm/messages looking for these messages.
4865 4865 */
4866 4866 if (report_lb_c && report_lb_p) {
4867 4867 if (ct->ct_lb == LOAD_BALANCE_LBA) {
4868 4868 (void) snprintf(lb_buf, sizeof (lb_buf),
4869 4869 "%s, region-size: %d", mdi_load_balance_lba,
4870 4870 ct->ct_lb_args->region_size);
4871 4871 } else if (ct->ct_lb == LOAD_BALANCE_NONE) {
4872 4872 (void) snprintf(lb_buf, sizeof (lb_buf),
4873 4873 "%s", mdi_load_balance_none);
4874 4874 } else {
4875 4875 (void) snprintf(lb_buf, sizeof (lb_buf), "%s",
4876 4876 mdi_load_balance_rr);
4877 4877 }
4878 4878
4879 4879 cmn_err(mdi_debug_consoleonly ? CE_NOTE : CE_CONT,
4880 4880 "?%s (%s%d) multipath status: %s: "
4881 4881 "path %d %s is %s: Load balancing: %s\n",
4882 4882 ddi_pathname(cdip, ct_path), ddi_driver_name(cdip),
4883 4883 ddi_get_instance(cdip), ct_status,
4884 4884 mdi_pi_get_path_instance(pip),
4885 4885 mdi_pi_spathname(pip), status, lb_buf);
4886 4886 } else {
4887 4887 cmn_err(mdi_debug_consoleonly ? CE_NOTE : CE_CONT,
4888 4888 "?%s (%s%d) multipath status: %s: "
4889 4889 "path %d %s is %s\n",
4890 4890 ddi_pathname(cdip, ct_path), ddi_driver_name(cdip),
4891 4891 ddi_get_instance(cdip), ct_status,
4892 4892 mdi_pi_get_path_instance(pip),
4893 4893 mdi_pi_spathname(pip), status);
4894 4894 }
4895 4895
4896 4896 kmem_free(ct_path, MAXPATHLEN);
4897 4897 MDI_CLIENT_CLEAR_REPORT_DEV_NEEDED(ct);
4898 4898 }
4899 4899 }
4900 4900
4901 4901 #ifdef DEBUG
4902 4902 /*
4903 4903 * i_mdi_log():
4904 4904 * Utility function for error message management
4905 4905 *
4906 4906 * NOTE: Implementation takes care of trailing \n for cmn_err,
4907 4907 * MDI_DEBUG should not terminate fmt strings with \n.
4908 4908 *
4909 4909 * NOTE: If the level is >= 2, and there is no leading !?^
4910 4910 * then a leading ! is implied (but can be overriden via
4911 4911 * mdi_debug_consoleonly). If you are using kmdb on the console,
4912 4912 * consider setting mdi_debug_consoleonly to 1 as an aid.
4913 4913 */
4914 4914 /*PRINTFLIKE4*/
4915 4915 static void
4916 4916 i_mdi_log(int level, const char *func, dev_info_t *dip, const char *fmt, ...)
4917 4917 {
4918 4918 char name[MAXNAMELEN];
4919 4919 char buf[512];
4920 4920 char *bp;
4921 4921 va_list ap;
4922 4922 int log_only = 0;
4923 4923 int boot_only = 0;
4924 4924 int console_only = 0;
4925 4925
4926 4926 if (dip) {
4927 4927 (void) snprintf(name, sizeof(name), "%s%d: ",
4928 4928 ddi_driver_name(dip), ddi_get_instance(dip));
4929 4929 } else {
4930 4930 name[0] = 0;
4931 4931 }
4932 4932
4933 4933 va_start(ap, fmt);
4934 4934 (void) vsnprintf(buf, sizeof(buf), fmt, ap);
4935 4935 va_end(ap);
4936 4936
4937 4937 switch (buf[0]) {
4938 4938 case '!':
4939 4939 bp = &buf[1];
4940 4940 log_only = 1;
4941 4941 break;
4942 4942 case '?':
4943 4943 bp = &buf[1];
4944 4944 boot_only = 1;
4945 4945 break;
4946 4946 case '^':
4947 4947 bp = &buf[1];
4948 4948 console_only = 1;
4949 4949 break;
4950 4950 default:
4951 4951 if (level >= 2)
4952 4952 log_only = 1; /* ! implied */
4953 4953 bp = buf;
4954 4954 break;
4955 4955 }
4956 4956 if (mdi_debug_logonly) {
4957 4957 log_only = 1;
4958 4958 boot_only = 0;
4959 4959 console_only = 0;
4960 4960 }
4961 4961 if (mdi_debug_consoleonly) {
4962 4962 log_only = 0;
4963 4963 boot_only = 0;
4964 4964 console_only = 1;
4965 4965 level = CE_NOTE;
4966 4966 goto console;
4967 4967 }
4968 4968
4969 4969 switch (level) {
4970 4970 case CE_NOTE:
4971 4971 level = CE_CONT;
4972 4972 /* FALLTHROUGH */
4973 4973 case CE_CONT:
4974 4974 if (boot_only) {
4975 4975 cmn_err(level, "?mdi: %s%s: %s\n", name, func, bp);
4976 4976 } else if (console_only) {
4977 4977 cmn_err(level, "^mdi: %s%s: %s\n", name, func, bp);
4978 4978 } else if (log_only) {
4979 4979 cmn_err(level, "!mdi: %s%s: %s\n", name, func, bp);
4980 4980 } else {
4981 4981 cmn_err(level, "mdi: %s%s: %s\n", name, func, bp);
4982 4982 }
4983 4983 break;
4984 4984
4985 4985 case CE_WARN:
4986 4986 case CE_PANIC:
4987 4987 console:
4988 4988 if (boot_only) {
4989 4989 cmn_err(level, "?mdi: %s%s: %s", name, func, bp);
4990 4990 } else if (console_only) {
4991 4991 cmn_err(level, "^mdi: %s%s: %s", name, func, bp);
4992 4992 } else if (log_only) {
4993 4993 cmn_err(level, "!mdi: %s%s: %s", name, func, bp);
4994 4994 } else {
4995 4995 cmn_err(level, "mdi: %s%s: %s", name, func, bp);
4996 4996 }
4997 4997 break;
4998 4998 default:
4999 4999 cmn_err(level, "mdi: %s%s", name, bp);
5000 5000 break;
5001 5001 }
5002 5002 }
5003 5003 #endif /* DEBUG */
5004 5004
5005 5005 void
5006 5006 i_mdi_client_online(dev_info_t *ct_dip)
5007 5007 {
5008 5008 mdi_client_t *ct;
5009 5009
5010 5010 /*
5011 5011 * Client online notification. Mark client state as online
5012 5012 * restore our binding with dev_info node
5013 5013 */
5014 5014 ct = i_devi_get_client(ct_dip);
5015 5015 ASSERT(ct != NULL);
5016 5016 MDI_CLIENT_LOCK(ct);
5017 5017 MDI_CLIENT_SET_ONLINE(ct);
5018 5018 /* catch for any memory leaks */
5019 5019 ASSERT((ct->ct_dip == NULL) || (ct->ct_dip == ct_dip));
5020 5020 ct->ct_dip = ct_dip;
5021 5021
5022 5022 if (ct->ct_power_cnt == 0)
5023 5023 (void) i_mdi_power_all_phci(ct);
5024 5024
5025 5025 MDI_DEBUG(4, (MDI_NOTE, ct_dip,
5026 5026 "i_mdi_pm_hold_client %p", (void *)ct));
5027 5027 i_mdi_pm_hold_client(ct, 1);
5028 5028
5029 5029 MDI_CLIENT_UNLOCK(ct);
5030 5030 }
5031 5031
5032 5032 void
5033 5033 i_mdi_phci_online(dev_info_t *ph_dip)
5034 5034 {
5035 5035 mdi_phci_t *ph;
5036 5036
5037 5037 /* pHCI online notification. Mark state accordingly */
5038 5038 ph = i_devi_get_phci(ph_dip);
5039 5039 ASSERT(ph != NULL);
5040 5040 MDI_PHCI_LOCK(ph);
5041 5041 MDI_PHCI_SET_ONLINE(ph);
5042 5042 MDI_PHCI_UNLOCK(ph);
5043 5043 }
5044 5044
5045 5045 /*
5046 5046 * mdi_devi_online():
5047 5047 * Online notification from NDI framework on pHCI/client
5048 5048 * device online.
5049 5049 * Return Values:
5050 5050 * NDI_SUCCESS
5051 5051 * MDI_FAILURE
5052 5052 */
5053 5053 /*ARGSUSED*/
5054 5054 int
5055 5055 mdi_devi_online(dev_info_t *dip, uint_t flags)
5056 5056 {
5057 5057 if (MDI_PHCI(dip)) {
5058 5058 i_mdi_phci_online(dip);
5059 5059 }
5060 5060
5061 5061 if (MDI_CLIENT(dip)) {
5062 5062 i_mdi_client_online(dip);
5063 5063 }
5064 5064 return (NDI_SUCCESS);
5065 5065 }
5066 5066
5067 5067 /*
5068 5068 * mdi_devi_offline():
5069 5069 * Offline notification from NDI framework on pHCI/Client device
5070 5070 * offline.
5071 5071 *
5072 5072 * Return Values:
5073 5073 * NDI_SUCCESS
5074 5074 * NDI_FAILURE
5075 5075 */
5076 5076 /*ARGSUSED*/
5077 5077 int
5078 5078 mdi_devi_offline(dev_info_t *dip, uint_t flags)
5079 5079 {
5080 5080 int rv = NDI_SUCCESS;
5081 5081
5082 5082 if (MDI_CLIENT(dip)) {
5083 5083 rv = i_mdi_client_offline(dip, flags);
5084 5084 if (rv != NDI_SUCCESS)
5085 5085 return (rv);
5086 5086 }
5087 5087
5088 5088 if (MDI_PHCI(dip)) {
5089 5089 rv = i_mdi_phci_offline(dip, flags);
5090 5090
5091 5091 if ((rv != NDI_SUCCESS) && MDI_CLIENT(dip)) {
5092 5092 /* set client back online */
5093 5093 i_mdi_client_online(dip);
5094 5094 }
5095 5095 }
5096 5096
5097 5097 return (rv);
5098 5098 }
5099 5099
5100 5100 /*ARGSUSED*/
5101 5101 static int
5102 5102 i_mdi_phci_offline(dev_info_t *dip, uint_t flags)
5103 5103 {
5104 5104 int rv = NDI_SUCCESS;
5105 5105 mdi_phci_t *ph;
5106 5106 mdi_client_t *ct;
5107 5107 mdi_pathinfo_t *pip;
5108 5108 mdi_pathinfo_t *next;
5109 5109 mdi_pathinfo_t *failed_pip = NULL;
5110 5110 dev_info_t *cdip;
5111 5111
5112 5112 /*
5113 5113 * pHCI component offline notification
5114 5114 * Make sure that this pHCI instance is free to be offlined.
5115 5115 * If it is OK to proceed, Offline and remove all the child
5116 5116 * mdi_pathinfo nodes. This process automatically offlines
5117 5117 * corresponding client devices, for which this pHCI provides
5118 5118 * critical services.
5119 5119 */
5120 5120 ph = i_devi_get_phci(dip);
5121 5121 MDI_DEBUG(2, (MDI_NOTE, dip,
5122 5122 "called %p %p", (void *)dip, (void *)ph));
5123 5123 if (ph == NULL) {
5124 5124 return (rv);
5125 5125 }
5126 5126
5127 5127 MDI_PHCI_LOCK(ph);
5128 5128
5129 5129 if (MDI_PHCI_IS_OFFLINE(ph)) {
5130 5130 MDI_DEBUG(1, (MDI_WARN, dip,
5131 5131 "!pHCI already offlined: %p", (void *)dip));
5132 5132 MDI_PHCI_UNLOCK(ph);
5133 5133 return (NDI_SUCCESS);
5134 5134 }
5135 5135
5136 5136 /*
5137 5137 * Check to see if the pHCI can be offlined
5138 5138 */
5139 5139 if (ph->ph_unstable) {
5140 5140 MDI_DEBUG(1, (MDI_WARN, dip,
5141 5141 "!One or more target devices are in transient state. "
5142 5142 "This device can not be removed at this moment. "
5143 5143 "Please try again later."));
5144 5144 MDI_PHCI_UNLOCK(ph);
5145 5145 return (NDI_BUSY);
5146 5146 }
5147 5147
5148 5148 pip = ph->ph_path_head;
5149 5149 while (pip != NULL) {
5150 5150 MDI_PI_LOCK(pip);
5151 5151 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5152 5152
5153 5153 /*
5154 5154 * The mdi_pathinfo state is OK. Check the client state.
5155 5155 * If failover in progress fail the pHCI from offlining
5156 5156 */
5157 5157 ct = MDI_PI(pip)->pi_client;
5158 5158 i_mdi_client_lock(ct, pip);
5159 5159 if ((MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct)) ||
5160 5160 (ct->ct_unstable)) {
5161 5161 /*
5162 5162 * Failover is in progress, Fail the DR
5163 5163 */
5164 5164 MDI_DEBUG(1, (MDI_WARN, dip,
5165 5165 "!pHCI device is busy. "
5166 5166 "This device can not be removed at this moment. "
5167 5167 "Please try again later."));
5168 5168 MDI_PI_UNLOCK(pip);
5169 5169 i_mdi_client_unlock(ct);
5170 5170 MDI_PHCI_UNLOCK(ph);
5171 5171 return (NDI_BUSY);
5172 5172 }
5173 5173 MDI_PI_UNLOCK(pip);
5174 5174
5175 5175 /*
5176 5176 * Check to see of we are removing the last path of this
5177 5177 * client device...
5178 5178 */
5179 5179 cdip = ct->ct_dip;
5180 5180 if (cdip && (i_ddi_node_state(cdip) >= DS_INITIALIZED) &&
5181 5181 (i_mdi_client_compute_state(ct, ph) ==
5182 5182 MDI_CLIENT_STATE_FAILED)) {
5183 5183 i_mdi_client_unlock(ct);
5184 5184 MDI_PHCI_UNLOCK(ph);
5185 5185 if (ndi_devi_offline(cdip,
5186 5186 NDI_DEVFS_CLEAN) != NDI_SUCCESS) {
5187 5187 /*
5188 5188 * ndi_devi_offline() failed.
5189 5189 * This pHCI provides the critical path
5190 5190 * to one or more client devices.
5191 5191 * Return busy.
5192 5192 */
5193 5193 MDI_PHCI_LOCK(ph);
5194 5194 MDI_DEBUG(1, (MDI_WARN, dip,
5195 5195 "!pHCI device is busy. "
5196 5196 "This device can not be removed at this "
5197 5197 "moment. Please try again later."));
5198 5198 failed_pip = pip;
5199 5199 break;
5200 5200 } else {
5201 5201 MDI_PHCI_LOCK(ph);
5202 5202 pip = next;
5203 5203 }
5204 5204 } else {
5205 5205 i_mdi_client_unlock(ct);
5206 5206 pip = next;
5207 5207 }
5208 5208 }
5209 5209
5210 5210 if (failed_pip) {
5211 5211 pip = ph->ph_path_head;
5212 5212 while (pip != failed_pip) {
5213 5213 MDI_PI_LOCK(pip);
5214 5214 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5215 5215 ct = MDI_PI(pip)->pi_client;
5216 5216 i_mdi_client_lock(ct, pip);
5217 5217 cdip = ct->ct_dip;
5218 5218 switch (MDI_CLIENT_STATE(ct)) {
5219 5219 case MDI_CLIENT_STATE_OPTIMAL:
5220 5220 case MDI_CLIENT_STATE_DEGRADED:
5221 5221 if (cdip) {
5222 5222 MDI_PI_UNLOCK(pip);
5223 5223 i_mdi_client_unlock(ct);
5224 5224 MDI_PHCI_UNLOCK(ph);
5225 5225 (void) ndi_devi_online(cdip, 0);
5226 5226 MDI_PHCI_LOCK(ph);
5227 5227 pip = next;
5228 5228 continue;
5229 5229 }
5230 5230 break;
5231 5231
5232 5232 case MDI_CLIENT_STATE_FAILED:
5233 5233 if (cdip) {
5234 5234 MDI_PI_UNLOCK(pip);
5235 5235 i_mdi_client_unlock(ct);
5236 5236 MDI_PHCI_UNLOCK(ph);
5237 5237 (void) ndi_devi_offline(cdip,
5238 5238 NDI_DEVFS_CLEAN);
5239 5239 MDI_PHCI_LOCK(ph);
5240 5240 pip = next;
5241 5241 continue;
5242 5242 }
5243 5243 break;
5244 5244 }
5245 5245 MDI_PI_UNLOCK(pip);
5246 5246 i_mdi_client_unlock(ct);
5247 5247 pip = next;
5248 5248 }
5249 5249 MDI_PHCI_UNLOCK(ph);
5250 5250 return (NDI_BUSY);
5251 5251 }
5252 5252
5253 5253 /*
5254 5254 * Mark the pHCI as offline
5255 5255 */
5256 5256 MDI_PHCI_SET_OFFLINE(ph);
5257 5257
5258 5258 /*
5259 5259 * Mark the child mdi_pathinfo nodes as transient
5260 5260 */
5261 5261 pip = ph->ph_path_head;
5262 5262 while (pip != NULL) {
5263 5263 MDI_PI_LOCK(pip);
5264 5264 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5265 5265 MDI_PI_SET_OFFLINING(pip);
5266 5266 MDI_PI_UNLOCK(pip);
5267 5267 pip = next;
5268 5268 }
5269 5269 MDI_PHCI_UNLOCK(ph);
5270 5270 /*
5271 5271 * Give a chance for any pending commands to execute
5272 5272 */
5273 5273 delay_random(mdi_delay);
5274 5274 MDI_PHCI_LOCK(ph);
5275 5275 pip = ph->ph_path_head;
5276 5276 while (pip != NULL) {
5277 5277 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5278 5278 (void) i_mdi_pi_offline(pip, flags);
5279 5279 MDI_PI_LOCK(pip);
5280 5280 ct = MDI_PI(pip)->pi_client;
5281 5281 if (!MDI_PI_IS_OFFLINE(pip)) {
5282 5282 MDI_DEBUG(1, (MDI_WARN, dip,
5283 5283 "!pHCI device is busy. "
5284 5284 "This device can not be removed at this moment. "
5285 5285 "Please try again later."));
5286 5286 MDI_PI_UNLOCK(pip);
5287 5287 MDI_PHCI_SET_ONLINE(ph);
5288 5288 MDI_PHCI_UNLOCK(ph);
5289 5289 return (NDI_BUSY);
5290 5290 }
5291 5291 MDI_PI_UNLOCK(pip);
5292 5292 pip = next;
5293 5293 }
5294 5294 MDI_PHCI_UNLOCK(ph);
5295 5295
5296 5296 return (rv);
5297 5297 }
5298 5298
5299 5299 void
5300 5300 mdi_phci_mark_retiring(dev_info_t *dip, char **cons_array)
5301 5301 {
5302 5302 mdi_phci_t *ph;
5303 5303 mdi_client_t *ct;
5304 5304 mdi_pathinfo_t *pip;
5305 5305 mdi_pathinfo_t *next;
5306 5306 dev_info_t *cdip;
5307 5307
5308 5308 if (!MDI_PHCI(dip))
5309 5309 return;
5310 5310
5311 5311 ph = i_devi_get_phci(dip);
5312 5312 if (ph == NULL) {
5313 5313 return;
5314 5314 }
5315 5315
5316 5316 MDI_PHCI_LOCK(ph);
5317 5317
5318 5318 if (MDI_PHCI_IS_OFFLINE(ph)) {
5319 5319 /* has no last path */
5320 5320 MDI_PHCI_UNLOCK(ph);
5321 5321 return;
5322 5322 }
5323 5323
5324 5324 pip = ph->ph_path_head;
5325 5325 while (pip != NULL) {
5326 5326 MDI_PI_LOCK(pip);
5327 5327 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5328 5328
5329 5329 ct = MDI_PI(pip)->pi_client;
5330 5330 i_mdi_client_lock(ct, pip);
5331 5331 MDI_PI_UNLOCK(pip);
5332 5332
5333 5333 cdip = ct->ct_dip;
5334 5334 if (cdip && (i_ddi_node_state(cdip) >= DS_INITIALIZED) &&
5335 5335 (i_mdi_client_compute_state(ct, ph) ==
5336 5336 MDI_CLIENT_STATE_FAILED)) {
5337 5337 /* Last path. Mark client dip as retiring */
5338 5338 i_mdi_client_unlock(ct);
5339 5339 MDI_PHCI_UNLOCK(ph);
5340 5340 (void) e_ddi_mark_retiring(cdip, cons_array);
5341 5341 MDI_PHCI_LOCK(ph);
5342 5342 pip = next;
5343 5343 } else {
5344 5344 i_mdi_client_unlock(ct);
5345 5345 pip = next;
5346 5346 }
5347 5347 }
5348 5348
5349 5349 MDI_PHCI_UNLOCK(ph);
5350 5350
5351 5351 return;
5352 5352 }
5353 5353
5354 5354 void
5355 5355 mdi_phci_retire_notify(dev_info_t *dip, int *constraint)
5356 5356 {
5357 5357 mdi_phci_t *ph;
5358 5358 mdi_client_t *ct;
5359 5359 mdi_pathinfo_t *pip;
5360 5360 mdi_pathinfo_t *next;
5361 5361 dev_info_t *cdip;
5362 5362
5363 5363 if (!MDI_PHCI(dip))
5364 5364 return;
5365 5365
5366 5366 ph = i_devi_get_phci(dip);
5367 5367 if (ph == NULL)
5368 5368 return;
5369 5369
5370 5370 MDI_PHCI_LOCK(ph);
5371 5371
5372 5372 if (MDI_PHCI_IS_OFFLINE(ph)) {
5373 5373 MDI_PHCI_UNLOCK(ph);
5374 5374 /* not last path */
5375 5375 return;
5376 5376 }
5377 5377
5378 5378 if (ph->ph_unstable) {
5379 5379 MDI_PHCI_UNLOCK(ph);
5380 5380 /* can't check for constraints */
5381 5381 *constraint = 0;
5382 5382 return;
5383 5383 }
5384 5384
5385 5385 pip = ph->ph_path_head;
5386 5386 while (pip != NULL) {
5387 5387 MDI_PI_LOCK(pip);
5388 5388 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5389 5389
5390 5390 /*
5391 5391 * The mdi_pathinfo state is OK. Check the client state.
5392 5392 * If failover in progress fail the pHCI from offlining
5393 5393 */
5394 5394 ct = MDI_PI(pip)->pi_client;
5395 5395 i_mdi_client_lock(ct, pip);
5396 5396 if ((MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct)) ||
5397 5397 (ct->ct_unstable)) {
5398 5398 /*
5399 5399 * Failover is in progress, can't check for constraints
5400 5400 */
5401 5401 MDI_PI_UNLOCK(pip);
5402 5402 i_mdi_client_unlock(ct);
5403 5403 MDI_PHCI_UNLOCK(ph);
5404 5404 *constraint = 0;
5405 5405 return;
5406 5406 }
5407 5407 MDI_PI_UNLOCK(pip);
5408 5408
5409 5409 /*
5410 5410 * Check to see of we are retiring the last path of this
5411 5411 * client device...
5412 5412 */
5413 5413 cdip = ct->ct_dip;
5414 5414 if (cdip && (i_ddi_node_state(cdip) >= DS_INITIALIZED) &&
5415 5415 (i_mdi_client_compute_state(ct, ph) ==
5416 5416 MDI_CLIENT_STATE_FAILED)) {
5417 5417 i_mdi_client_unlock(ct);
5418 5418 MDI_PHCI_UNLOCK(ph);
5419 5419 (void) e_ddi_retire_notify(cdip, constraint);
5420 5420 MDI_PHCI_LOCK(ph);
5421 5421 pip = next;
5422 5422 } else {
5423 5423 i_mdi_client_unlock(ct);
5424 5424 pip = next;
5425 5425 }
5426 5426 }
5427 5427
5428 5428 MDI_PHCI_UNLOCK(ph);
5429 5429
5430 5430 return;
5431 5431 }
5432 5432
5433 5433 /*
5434 5434 * offline the path(s) hanging off the pHCI. If the
5435 5435 * last path to any client, check that constraints
5436 5436 * have been applied.
5437 5437 *
5438 5438 * If constraint is 0, we aren't going to retire the
5439 5439 * pHCI. However we still need to go through the paths
5440 5440 * calling e_ddi_retire_finalize() to clear their
5441 5441 * contract barriers.
5442 5442 */
5443 5443 void
5444 5444 mdi_phci_retire_finalize(dev_info_t *dip, int phci_only, void *constraint)
5445 5445 {
5446 5446 mdi_phci_t *ph;
5447 5447 mdi_client_t *ct;
5448 5448 mdi_pathinfo_t *pip;
5449 5449 mdi_pathinfo_t *next;
5450 5450 dev_info_t *cdip;
5451 5451 int unstable = 0;
5452 5452 int tmp_constraint;
5453 5453
5454 5454 if (!MDI_PHCI(dip))
5455 5455 return;
5456 5456
5457 5457 ph = i_devi_get_phci(dip);
5458 5458 if (ph == NULL) {
5459 5459 /* no last path and no pips */
5460 5460 return;
5461 5461 }
5462 5462
5463 5463 MDI_PHCI_LOCK(ph);
5464 5464
5465 5465 if (MDI_PHCI_IS_OFFLINE(ph)) {
5466 5466 MDI_PHCI_UNLOCK(ph);
5467 5467 /* no last path and no pips */
5468 5468 return;
5469 5469 }
5470 5470
5471 5471 /*
5472 5472 * Check to see if the pHCI can be offlined
5473 5473 */
5474 5474 if (ph->ph_unstable) {
5475 5475 unstable = 1;
5476 5476 }
5477 5477
5478 5478 pip = ph->ph_path_head;
5479 5479 while (pip != NULL) {
5480 5480 MDI_PI_LOCK(pip);
5481 5481 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5482 5482
5483 5483 /*
5484 5484 * if failover in progress fail the pHCI from offlining
5485 5485 */
5486 5486 ct = MDI_PI(pip)->pi_client;
5487 5487 i_mdi_client_lock(ct, pip);
5488 5488 if ((MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct)) ||
5489 5489 (ct->ct_unstable)) {
5490 5490 unstable = 1;
5491 5491 }
5492 5492 MDI_PI_UNLOCK(pip);
5493 5493
5494 5494 /*
5495 5495 * Check to see of we are removing the last path of this
5496 5496 * client device...
5497 5497 */
5498 5498 cdip = ct->ct_dip;
5499 5499 if (!phci_only && cdip &&
5500 5500 (i_ddi_node_state(cdip) >= DS_INITIALIZED) &&
5501 5501 (i_mdi_client_compute_state(ct, ph) ==
5502 5502 MDI_CLIENT_STATE_FAILED)) {
5503 5503 i_mdi_client_unlock(ct);
5504 5504 MDI_PHCI_UNLOCK(ph);
5505 5505 /*
5506 5506 * This is the last path to this client.
5507 5507 *
5508 5508 * Constraint will only be set to 1 if this client can
5509 5509 * be retired (as already determined by
5510 5510 * mdi_phci_retire_notify). However we don't actually
5511 5511 * need to retire the client (we just retire the last
5512 5512 * path - MPXIO will then fail all I/Os to the client).
5513 5513 * But we still need to call e_ddi_retire_finalize so
5514 5514 * the contract barriers can be cleared. Therefore we
5515 5515 * temporarily set constraint = 0 so that the client
5516 5516 * dip is not retired.
5517 5517 */
5518 5518 tmp_constraint = 0;
5519 5519 (void) e_ddi_retire_finalize(cdip, &tmp_constraint);
5520 5520 MDI_PHCI_LOCK(ph);
5521 5521 pip = next;
5522 5522 } else {
5523 5523 i_mdi_client_unlock(ct);
5524 5524 pip = next;
5525 5525 }
5526 5526 }
5527 5527
5528 5528 if (!phci_only && *((int *)constraint) == 0) {
5529 5529 MDI_PHCI_UNLOCK(ph);
5530 5530 return;
5531 5531 }
5532 5532
5533 5533 /*
5534 5534 * Cannot offline pip(s)
5535 5535 */
5536 5536 if (unstable) {
5537 5537 cmn_err(CE_WARN, "%s%d: mdi_phci_retire_finalize: "
5538 5538 "pHCI in transient state, cannot retire",
5539 5539 ddi_driver_name(dip), ddi_get_instance(dip));
5540 5540 MDI_PHCI_UNLOCK(ph);
5541 5541 return;
5542 5542 }
5543 5543
5544 5544 /*
5545 5545 * Mark the pHCI as offline
5546 5546 */
5547 5547 MDI_PHCI_SET_OFFLINE(ph);
5548 5548
5549 5549 /*
5550 5550 * Mark the child mdi_pathinfo nodes as transient
5551 5551 */
5552 5552 pip = ph->ph_path_head;
5553 5553 while (pip != NULL) {
5554 5554 MDI_PI_LOCK(pip);
5555 5555 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5556 5556 MDI_PI_SET_OFFLINING(pip);
5557 5557 MDI_PI_UNLOCK(pip);
5558 5558 pip = next;
5559 5559 }
5560 5560 MDI_PHCI_UNLOCK(ph);
5561 5561 /*
5562 5562 * Give a chance for any pending commands to execute
5563 5563 */
5564 5564 delay_random(mdi_delay);
5565 5565 MDI_PHCI_LOCK(ph);
5566 5566 pip = ph->ph_path_head;
5567 5567 while (pip != NULL) {
5568 5568 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5569 5569 (void) i_mdi_pi_offline(pip, 0);
5570 5570 MDI_PI_LOCK(pip);
5571 5571 ct = MDI_PI(pip)->pi_client;
5572 5572 if (!MDI_PI_IS_OFFLINE(pip)) {
5573 5573 cmn_err(CE_WARN, "mdi_phci_retire_finalize: "
5574 5574 "path %d %s busy, cannot offline",
5575 5575 mdi_pi_get_path_instance(pip),
5576 5576 mdi_pi_spathname(pip));
5577 5577 MDI_PI_UNLOCK(pip);
5578 5578 MDI_PHCI_SET_ONLINE(ph);
5579 5579 MDI_PHCI_UNLOCK(ph);
5580 5580 return;
5581 5581 }
5582 5582 MDI_PI_UNLOCK(pip);
5583 5583 pip = next;
5584 5584 }
5585 5585 MDI_PHCI_UNLOCK(ph);
5586 5586
5587 5587 return;
5588 5588 }
5589 5589
5590 5590 void
5591 5591 mdi_phci_unretire(dev_info_t *dip)
5592 5592 {
5593 5593 mdi_phci_t *ph;
5594 5594 mdi_pathinfo_t *pip;
5595 5595 mdi_pathinfo_t *next;
5596 5596
5597 5597 ASSERT(MDI_PHCI(dip));
5598 5598
5599 5599 /*
5600 5600 * Online the phci
5601 5601 */
5602 5602 i_mdi_phci_online(dip);
5603 5603
5604 5604 ph = i_devi_get_phci(dip);
5605 5605 MDI_PHCI_LOCK(ph);
5606 5606 pip = ph->ph_path_head;
5607 5607 while (pip != NULL) {
5608 5608 MDI_PI_LOCK(pip);
5609 5609 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5610 5610 MDI_PI_UNLOCK(pip);
5611 5611 (void) i_mdi_pi_online(pip, 0);
5612 5612 pip = next;
5613 5613 }
5614 5614 MDI_PHCI_UNLOCK(ph);
5615 5615 }
5616 5616
5617 5617 /*ARGSUSED*/
5618 5618 static int
5619 5619 i_mdi_client_offline(dev_info_t *dip, uint_t flags)
5620 5620 {
5621 5621 int rv = NDI_SUCCESS;
5622 5622 mdi_client_t *ct;
5623 5623
5624 5624 /*
5625 5625 * Client component to go offline. Make sure that we are
5626 5626 * not in failing over state and update client state
5627 5627 * accordingly
5628 5628 */
5629 5629 ct = i_devi_get_client(dip);
5630 5630 MDI_DEBUG(2, (MDI_NOTE, dip,
5631 5631 "called %p %p", (void *)dip, (void *)ct));
5632 5632 if (ct != NULL) {
5633 5633 MDI_CLIENT_LOCK(ct);
5634 5634 if (ct->ct_unstable) {
5635 5635 /*
5636 5636 * One or more paths are in transient state,
5637 5637 * Dont allow offline of a client device
5638 5638 */
5639 5639 MDI_DEBUG(1, (MDI_WARN, dip,
5640 5640 "!One or more paths to "
5641 5641 "this device are in transient state. "
5642 5642 "This device can not be removed at this moment. "
5643 5643 "Please try again later."));
5644 5644 MDI_CLIENT_UNLOCK(ct);
5645 5645 return (NDI_BUSY);
5646 5646 }
5647 5647 if (MDI_CLIENT_IS_FAILOVER_IN_PROGRESS(ct)) {
5648 5648 /*
5649 5649 * Failover is in progress, Dont allow DR of
5650 5650 * a client device
5651 5651 */
5652 5652 MDI_DEBUG(1, (MDI_WARN, dip,
5653 5653 "!Client device is Busy. "
5654 5654 "This device can not be removed at this moment. "
5655 5655 "Please try again later."));
5656 5656 MDI_CLIENT_UNLOCK(ct);
5657 5657 return (NDI_BUSY);
5658 5658 }
5659 5659 MDI_CLIENT_SET_OFFLINE(ct);
5660 5660
5661 5661 /*
5662 5662 * Unbind our relationship with the dev_info node
5663 5663 */
5664 5664 if (flags & NDI_DEVI_REMOVE) {
5665 5665 ct->ct_dip = NULL;
5666 5666 }
5667 5667 MDI_CLIENT_UNLOCK(ct);
5668 5668 }
5669 5669 return (rv);
5670 5670 }
5671 5671
5672 5672 /*
5673 5673 * mdi_pre_attach():
5674 5674 * Pre attach() notification handler
5675 5675 */
5676 5676 /*ARGSUSED*/
5677 5677 int
5678 5678 mdi_pre_attach(dev_info_t *dip, ddi_attach_cmd_t cmd)
5679 5679 {
5680 5680 /* don't support old DDI_PM_RESUME */
5681 5681 if ((DEVI(dip)->devi_mdi_component != MDI_COMPONENT_NONE) &&
5682 5682 (cmd == DDI_PM_RESUME))
5683 5683 return (DDI_FAILURE);
5684 5684
5685 5685 return (DDI_SUCCESS);
5686 5686 }
5687 5687
5688 5688 /*
5689 5689 * mdi_post_attach():
5690 5690 * Post attach() notification handler
5691 5691 */
5692 5692 /*ARGSUSED*/
5693 5693 void
5694 5694 mdi_post_attach(dev_info_t *dip, ddi_attach_cmd_t cmd, int error)
5695 5695 {
5696 5696 mdi_phci_t *ph;
5697 5697 mdi_client_t *ct;
5698 5698 mdi_vhci_t *vh;
5699 5699
5700 5700 if (MDI_PHCI(dip)) {
5701 5701 ph = i_devi_get_phci(dip);
5702 5702 ASSERT(ph != NULL);
5703 5703
5704 5704 MDI_PHCI_LOCK(ph);
5705 5705 switch (cmd) {
5706 5706 case DDI_ATTACH:
5707 5707 MDI_DEBUG(2, (MDI_NOTE, dip,
5708 5708 "phci post_attach called %p", (void *)ph));
5709 5709 if (error == DDI_SUCCESS) {
5710 5710 MDI_PHCI_SET_ATTACH(ph);
5711 5711 } else {
5712 5712 MDI_DEBUG(1, (MDI_NOTE, dip,
5713 5713 "!pHCI post_attach failed: error %d",
5714 5714 error));
5715 5715 MDI_PHCI_SET_DETACH(ph);
5716 5716 }
5717 5717 break;
5718 5718
5719 5719 case DDI_RESUME:
5720 5720 MDI_DEBUG(2, (MDI_NOTE, dip,
5721 5721 "pHCI post_resume: called %p", (void *)ph));
5722 5722 if (error == DDI_SUCCESS) {
5723 5723 MDI_PHCI_SET_RESUME(ph);
5724 5724 } else {
5725 5725 MDI_DEBUG(1, (MDI_NOTE, dip,
5726 5726 "!pHCI post_resume failed: error %d",
5727 5727 error));
5728 5728 MDI_PHCI_SET_SUSPEND(ph);
5729 5729 }
5730 5730 break;
5731 5731 }
5732 5732 MDI_PHCI_UNLOCK(ph);
5733 5733 }
5734 5734
5735 5735 if (MDI_CLIENT(dip)) {
5736 5736 ct = i_devi_get_client(dip);
5737 5737 ASSERT(ct != NULL);
5738 5738
5739 5739 MDI_CLIENT_LOCK(ct);
5740 5740 switch (cmd) {
5741 5741 case DDI_ATTACH:
5742 5742 MDI_DEBUG(2, (MDI_NOTE, dip,
5743 5743 "client post_attach called %p", (void *)ct));
5744 5744 if (error != DDI_SUCCESS) {
5745 5745 MDI_DEBUG(1, (MDI_NOTE, dip,
5746 5746 "!client post_attach failed: error %d",
5747 5747 error));
5748 5748 MDI_CLIENT_SET_DETACH(ct);
5749 5749 MDI_DEBUG(4, (MDI_WARN, dip,
5750 5750 "i_mdi_pm_reset_client"));
5751 5751 i_mdi_pm_reset_client(ct);
5752 5752 break;
5753 5753 }
5754 5754
5755 5755 /*
5756 5756 * Client device has successfully attached, inform
5757 5757 * the vhci.
5758 5758 */
5759 5759 vh = ct->ct_vhci;
5760 5760 if (vh->vh_ops->vo_client_attached)
5761 5761 (*vh->vh_ops->vo_client_attached)(dip);
5762 5762
5763 5763 MDI_CLIENT_SET_ATTACH(ct);
5764 5764 break;
5765 5765
5766 5766 case DDI_RESUME:
5767 5767 MDI_DEBUG(2, (MDI_NOTE, dip,
5768 5768 "client post_attach: called %p", (void *)ct));
5769 5769 if (error == DDI_SUCCESS) {
5770 5770 MDI_CLIENT_SET_RESUME(ct);
5771 5771 } else {
5772 5772 MDI_DEBUG(1, (MDI_NOTE, dip,
5773 5773 "!client post_resume failed: error %d",
5774 5774 error));
5775 5775 MDI_CLIENT_SET_SUSPEND(ct);
5776 5776 }
5777 5777 break;
5778 5778 }
5779 5779 MDI_CLIENT_UNLOCK(ct);
5780 5780 }
5781 5781 }
5782 5782
5783 5783 /*
5784 5784 * mdi_pre_detach():
5785 5785 * Pre detach notification handler
5786 5786 */
5787 5787 /*ARGSUSED*/
5788 5788 int
5789 5789 mdi_pre_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
5790 5790 {
5791 5791 int rv = DDI_SUCCESS;
5792 5792
5793 5793 if (MDI_CLIENT(dip)) {
5794 5794 (void) i_mdi_client_pre_detach(dip, cmd);
5795 5795 }
5796 5796
5797 5797 if (MDI_PHCI(dip)) {
5798 5798 rv = i_mdi_phci_pre_detach(dip, cmd);
5799 5799 }
5800 5800
5801 5801 return (rv);
5802 5802 }
5803 5803
5804 5804 /*ARGSUSED*/
5805 5805 static int
5806 5806 i_mdi_phci_pre_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
5807 5807 {
5808 5808 int rv = DDI_SUCCESS;
5809 5809 mdi_phci_t *ph;
5810 5810 mdi_client_t *ct;
5811 5811 mdi_pathinfo_t *pip;
5812 5812 mdi_pathinfo_t *failed_pip = NULL;
5813 5813 mdi_pathinfo_t *next;
5814 5814
5815 5815 ph = i_devi_get_phci(dip);
5816 5816 if (ph == NULL) {
5817 5817 return (rv);
5818 5818 }
5819 5819
5820 5820 MDI_PHCI_LOCK(ph);
5821 5821 switch (cmd) {
5822 5822 case DDI_DETACH:
5823 5823 MDI_DEBUG(2, (MDI_NOTE, dip,
5824 5824 "pHCI pre_detach: called %p", (void *)ph));
5825 5825 if (!MDI_PHCI_IS_OFFLINE(ph)) {
5826 5826 /*
5827 5827 * mdi_pathinfo nodes are still attached to
5828 5828 * this pHCI. Fail the detach for this pHCI.
5829 5829 */
5830 5830 MDI_DEBUG(2, (MDI_WARN, dip,
5831 5831 "pHCI pre_detach: paths are still attached %p",
5832 5832 (void *)ph));
5833 5833 rv = DDI_FAILURE;
5834 5834 break;
5835 5835 }
5836 5836 MDI_PHCI_SET_DETACH(ph);
5837 5837 break;
5838 5838
5839 5839 case DDI_SUSPEND:
5840 5840 /*
5841 5841 * pHCI is getting suspended. Since mpxio client
5842 5842 * devices may not be suspended at this point, to avoid
5843 5843 * a potential stack overflow, it is important to suspend
5844 5844 * client devices before pHCI can be suspended.
5845 5845 */
5846 5846
5847 5847 MDI_DEBUG(2, (MDI_NOTE, dip,
5848 5848 "pHCI pre_suspend: called %p", (void *)ph));
5849 5849 /*
5850 5850 * Suspend all the client devices accessible through this pHCI
5851 5851 */
5852 5852 pip = ph->ph_path_head;
5853 5853 while (pip != NULL && rv == DDI_SUCCESS) {
5854 5854 dev_info_t *cdip;
5855 5855 MDI_PI_LOCK(pip);
5856 5856 next =
5857 5857 (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5858 5858 ct = MDI_PI(pip)->pi_client;
5859 5859 i_mdi_client_lock(ct, pip);
5860 5860 cdip = ct->ct_dip;
5861 5861 MDI_PI_UNLOCK(pip);
5862 5862 if ((MDI_CLIENT_IS_DETACHED(ct) == 0) &&
5863 5863 MDI_CLIENT_IS_SUSPENDED(ct) == 0) {
5864 5864 i_mdi_client_unlock(ct);
5865 5865 if ((rv = devi_detach(cdip, DDI_SUSPEND)) !=
5866 5866 DDI_SUCCESS) {
5867 5867 /*
5868 5868 * Suspend of one of the client
5869 5869 * device has failed.
5870 5870 */
5871 5871 MDI_DEBUG(1, (MDI_WARN, dip,
5872 5872 "!suspend of device (%s%d) failed.",
5873 5873 ddi_driver_name(cdip),
5874 5874 ddi_get_instance(cdip)));
5875 5875 failed_pip = pip;
5876 5876 break;
5877 5877 }
5878 5878 } else {
5879 5879 i_mdi_client_unlock(ct);
5880 5880 }
5881 5881 pip = next;
5882 5882 }
5883 5883
5884 5884 if (rv == DDI_SUCCESS) {
5885 5885 /*
5886 5886 * Suspend of client devices is complete. Proceed
5887 5887 * with pHCI suspend.
5888 5888 */
5889 5889 MDI_PHCI_SET_SUSPEND(ph);
5890 5890 } else {
5891 5891 /*
5892 5892 * Revert back all the suspended client device states
5893 5893 * to converse.
5894 5894 */
5895 5895 pip = ph->ph_path_head;
5896 5896 while (pip != failed_pip) {
5897 5897 dev_info_t *cdip;
5898 5898 MDI_PI_LOCK(pip);
5899 5899 next =
5900 5900 (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
5901 5901 ct = MDI_PI(pip)->pi_client;
5902 5902 i_mdi_client_lock(ct, pip);
5903 5903 cdip = ct->ct_dip;
5904 5904 MDI_PI_UNLOCK(pip);
5905 5905 if (MDI_CLIENT_IS_SUSPENDED(ct)) {
5906 5906 i_mdi_client_unlock(ct);
5907 5907 (void) devi_attach(cdip, DDI_RESUME);
5908 5908 } else {
5909 5909 i_mdi_client_unlock(ct);
5910 5910 }
5911 5911 pip = next;
5912 5912 }
5913 5913 }
5914 5914 break;
5915 5915
5916 5916 default:
5917 5917 rv = DDI_FAILURE;
5918 5918 break;
5919 5919 }
5920 5920 MDI_PHCI_UNLOCK(ph);
5921 5921 return (rv);
5922 5922 }
5923 5923
5924 5924 /*ARGSUSED*/
5925 5925 static int
5926 5926 i_mdi_client_pre_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
5927 5927 {
5928 5928 int rv = DDI_SUCCESS;
5929 5929 mdi_client_t *ct;
5930 5930
5931 5931 ct = i_devi_get_client(dip);
5932 5932 if (ct == NULL) {
5933 5933 return (rv);
5934 5934 }
5935 5935
5936 5936 MDI_CLIENT_LOCK(ct);
5937 5937 switch (cmd) {
5938 5938 case DDI_DETACH:
5939 5939 MDI_DEBUG(2, (MDI_NOTE, dip,
5940 5940 "client pre_detach: called %p",
5941 5941 (void *)ct));
5942 5942 MDI_CLIENT_SET_DETACH(ct);
5943 5943 break;
5944 5944
5945 5945 case DDI_SUSPEND:
5946 5946 MDI_DEBUG(2, (MDI_NOTE, dip,
5947 5947 "client pre_suspend: called %p",
5948 5948 (void *)ct));
5949 5949 MDI_CLIENT_SET_SUSPEND(ct);
5950 5950 break;
5951 5951
5952 5952 default:
5953 5953 rv = DDI_FAILURE;
5954 5954 break;
5955 5955 }
5956 5956 MDI_CLIENT_UNLOCK(ct);
5957 5957 return (rv);
5958 5958 }
5959 5959
5960 5960 /*
5961 5961 * mdi_post_detach():
5962 5962 * Post detach notification handler
5963 5963 */
5964 5964 /*ARGSUSED*/
5965 5965 void
5966 5966 mdi_post_detach(dev_info_t *dip, ddi_detach_cmd_t cmd, int error)
5967 5967 {
5968 5968 /*
5969 5969 * Detach/Suspend of mpxio component failed. Update our state
5970 5970 * too
5971 5971 */
5972 5972 if (MDI_PHCI(dip))
5973 5973 i_mdi_phci_post_detach(dip, cmd, error);
5974 5974
5975 5975 if (MDI_CLIENT(dip))
5976 5976 i_mdi_client_post_detach(dip, cmd, error);
5977 5977 }
5978 5978
5979 5979 /*ARGSUSED*/
5980 5980 static void
5981 5981 i_mdi_phci_post_detach(dev_info_t *dip, ddi_detach_cmd_t cmd, int error)
5982 5982 {
5983 5983 mdi_phci_t *ph;
5984 5984
5985 5985 /*
5986 5986 * Detach/Suspend of phci component failed. Update our state
5987 5987 * too
5988 5988 */
5989 5989 ph = i_devi_get_phci(dip);
5990 5990 if (ph == NULL) {
5991 5991 return;
5992 5992 }
5993 5993
5994 5994 MDI_PHCI_LOCK(ph);
5995 5995 /*
5996 5996 * Detach of pHCI failed. Restore back converse
5997 5997 * state
5998 5998 */
5999 5999 switch (cmd) {
6000 6000 case DDI_DETACH:
6001 6001 MDI_DEBUG(2, (MDI_NOTE, dip,
6002 6002 "pHCI post_detach: called %p",
6003 6003 (void *)ph));
6004 6004 if (error != DDI_SUCCESS)
6005 6005 MDI_PHCI_SET_ATTACH(ph);
6006 6006 break;
6007 6007
6008 6008 case DDI_SUSPEND:
6009 6009 MDI_DEBUG(2, (MDI_NOTE, dip,
6010 6010 "pHCI post_suspend: called %p",
6011 6011 (void *)ph));
6012 6012 if (error != DDI_SUCCESS)
6013 6013 MDI_PHCI_SET_RESUME(ph);
6014 6014 break;
6015 6015 }
6016 6016 MDI_PHCI_UNLOCK(ph);
6017 6017 }
6018 6018
6019 6019 /*ARGSUSED*/
6020 6020 static void
6021 6021 i_mdi_client_post_detach(dev_info_t *dip, ddi_detach_cmd_t cmd, int error)
6022 6022 {
6023 6023 mdi_client_t *ct;
6024 6024
6025 6025 ct = i_devi_get_client(dip);
6026 6026 if (ct == NULL) {
6027 6027 return;
6028 6028 }
6029 6029 MDI_CLIENT_LOCK(ct);
6030 6030 /*
6031 6031 * Detach of Client failed. Restore back converse
6032 6032 * state
6033 6033 */
6034 6034 switch (cmd) {
6035 6035 case DDI_DETACH:
6036 6036 MDI_DEBUG(2, (MDI_NOTE, dip,
6037 6037 "client post_detach: called %p", (void *)ct));
6038 6038 if (DEVI_IS_ATTACHING(dip)) {
6039 6039 MDI_DEBUG(4, (MDI_NOTE, dip,
6040 6040 "i_mdi_pm_rele_client\n"));
6041 6041 i_mdi_pm_rele_client(ct, ct->ct_path_count);
6042 6042 } else {
6043 6043 MDI_DEBUG(4, (MDI_NOTE, dip,
6044 6044 "i_mdi_pm_reset_client\n"));
6045 6045 i_mdi_pm_reset_client(ct);
6046 6046 }
6047 6047 if (error != DDI_SUCCESS)
6048 6048 MDI_CLIENT_SET_ATTACH(ct);
6049 6049 break;
6050 6050
6051 6051 case DDI_SUSPEND:
6052 6052 MDI_DEBUG(2, (MDI_NOTE, dip,
6053 6053 "called %p", (void *)ct));
6054 6054 if (error != DDI_SUCCESS)
6055 6055 MDI_CLIENT_SET_RESUME(ct);
6056 6056 break;
6057 6057 }
6058 6058 MDI_CLIENT_UNLOCK(ct);
6059 6059 }
6060 6060
6061 6061 int
6062 6062 mdi_pi_kstat_exists(mdi_pathinfo_t *pip)
6063 6063 {
6064 6064 return (MDI_PI(pip)->pi_kstats ? 1 : 0);
6065 6065 }
6066 6066
6067 6067 /*
6068 6068 * create and install per-path (client - pHCI) statistics
6069 6069 * I/O stats supported: nread, nwritten, reads, and writes
6070 6070 * Error stats - hard errors, soft errors, & transport errors
6071 6071 */
6072 6072 int
6073 6073 mdi_pi_kstat_create(mdi_pathinfo_t *pip, char *ksname)
6074 6074 {
6075 6075 kstat_t *kiosp, *kerrsp;
6076 6076 struct pi_errs *nsp;
6077 6077 struct mdi_pi_kstats *mdi_statp;
6078 6078
6079 6079 if (MDI_PI(pip)->pi_kstats != NULL)
6080 6080 return (MDI_SUCCESS);
6081 6081
6082 6082 if ((kiosp = kstat_create("mdi", 0, ksname, "iopath",
6083 6083 KSTAT_TYPE_IO, 1, KSTAT_FLAG_PERSISTENT)) == NULL) {
6084 6084 return (MDI_FAILURE);
6085 6085 }
6086 6086
6087 6087 (void) strcat(ksname, ",err");
6088 6088 kerrsp = kstat_create("mdi", 0, ksname, "iopath_errors",
6089 6089 KSTAT_TYPE_NAMED,
6090 6090 sizeof (struct pi_errs) / sizeof (kstat_named_t), 0);
6091 6091 if (kerrsp == NULL) {
6092 6092 kstat_delete(kiosp);
6093 6093 return (MDI_FAILURE);
6094 6094 }
6095 6095
6096 6096 nsp = (struct pi_errs *)kerrsp->ks_data;
6097 6097 kstat_named_init(&nsp->pi_softerrs, "Soft Errors", KSTAT_DATA_UINT32);
6098 6098 kstat_named_init(&nsp->pi_harderrs, "Hard Errors", KSTAT_DATA_UINT32);
6099 6099 kstat_named_init(&nsp->pi_transerrs, "Transport Errors",
6100 6100 KSTAT_DATA_UINT32);
6101 6101 kstat_named_init(&nsp->pi_icnt_busy, "Interconnect Busy",
6102 6102 KSTAT_DATA_UINT32);
6103 6103 kstat_named_init(&nsp->pi_icnt_errors, "Interconnect Errors",
6104 6104 KSTAT_DATA_UINT32);
6105 6105 kstat_named_init(&nsp->pi_phci_rsrc, "pHCI No Resources",
6106 6106 KSTAT_DATA_UINT32);
6107 6107 kstat_named_init(&nsp->pi_phci_localerr, "pHCI Local Errors",
6108 6108 KSTAT_DATA_UINT32);
6109 6109 kstat_named_init(&nsp->pi_phci_invstate, "pHCI Invalid State",
6110 6110 KSTAT_DATA_UINT32);
6111 6111 kstat_named_init(&nsp->pi_failedfrom, "Failed From",
6112 6112 KSTAT_DATA_UINT32);
6113 6113 kstat_named_init(&nsp->pi_failedto, "Failed To", KSTAT_DATA_UINT32);
6114 6114
6115 6115 mdi_statp = kmem_alloc(sizeof (*mdi_statp), KM_SLEEP);
6116 6116 mdi_statp->pi_kstat_ref = 1;
6117 6117 mdi_statp->pi_kstat_iostats = kiosp;
6118 6118 mdi_statp->pi_kstat_errstats = kerrsp;
6119 6119 kstat_install(kiosp);
6120 6120 kstat_install(kerrsp);
6121 6121 MDI_PI(pip)->pi_kstats = mdi_statp;
6122 6122 return (MDI_SUCCESS);
6123 6123 }
6124 6124
6125 6125 /*
6126 6126 * destroy per-path properties
6127 6127 */
6128 6128 static void
6129 6129 i_mdi_pi_kstat_destroy(mdi_pathinfo_t *pip)
6130 6130 {
6131 6131
6132 6132 struct mdi_pi_kstats *mdi_statp;
6133 6133
6134 6134 if (MDI_PI(pip)->pi_kstats == NULL)
6135 6135 return;
6136 6136 if ((mdi_statp = MDI_PI(pip)->pi_kstats) == NULL)
6137 6137 return;
6138 6138
6139 6139 MDI_PI(pip)->pi_kstats = NULL;
6140 6140
6141 6141 /*
6142 6142 * the kstat may be shared between multiple pathinfo nodes
6143 6143 * decrement this pathinfo's usage, removing the kstats
6144 6144 * themselves when the last pathinfo reference is removed.
6145 6145 */
6146 6146 ASSERT(mdi_statp->pi_kstat_ref > 0);
6147 6147 if (--mdi_statp->pi_kstat_ref != 0)
6148 6148 return;
6149 6149
6150 6150 kstat_delete(mdi_statp->pi_kstat_iostats);
6151 6151 kstat_delete(mdi_statp->pi_kstat_errstats);
6152 6152 kmem_free(mdi_statp, sizeof (*mdi_statp));
6153 6153 }
6154 6154
6155 6155 /*
6156 6156 * update I/O paths KSTATS
6157 6157 */
6158 6158 void
6159 6159 mdi_pi_kstat_iosupdate(mdi_pathinfo_t *pip, struct buf *bp)
6160 6160 {
6161 6161 kstat_t *iostatp;
6162 6162 size_t xfer_cnt;
6163 6163
6164 6164 ASSERT(pip != NULL);
6165 6165
6166 6166 /*
6167 6167 * I/O can be driven across a path prior to having path
6168 6168 * statistics available, i.e. probe(9e).
6169 6169 */
6170 6170 if (bp != NULL && MDI_PI(pip)->pi_kstats != NULL) {
6171 6171 iostatp = MDI_PI(pip)->pi_kstats->pi_kstat_iostats;
6172 6172 xfer_cnt = bp->b_bcount - bp->b_resid;
6173 6173 if (bp->b_flags & B_READ) {
6174 6174 KSTAT_IO_PTR(iostatp)->reads++;
6175 6175 KSTAT_IO_PTR(iostatp)->nread += xfer_cnt;
6176 6176 } else {
6177 6177 KSTAT_IO_PTR(iostatp)->writes++;
6178 6178 KSTAT_IO_PTR(iostatp)->nwritten += xfer_cnt;
6179 6179 }
6180 6180 }
6181 6181 }
6182 6182
6183 6183 /*
6184 6184 * Enable the path(specific client/target/initiator)
6185 6185 * Enabling a path means that MPxIO may select the enabled path for routing
6186 6186 * future I/O requests, subject to other path state constraints.
6187 6187 */
6188 6188 int
6189 6189 mdi_pi_enable_path(mdi_pathinfo_t *pip, int flags)
6190 6190 {
6191 6191 mdi_phci_t *ph;
6192 6192
6193 6193 ph = MDI_PI(pip)->pi_phci;
6194 6194 if (ph == NULL) {
6195 6195 MDI_DEBUG(1, (MDI_NOTE, mdi_pi_get_phci(pip),
6196 6196 "!failed: path %s %p: NULL ph",
6197 6197 mdi_pi_spathname(pip), (void *)pip));
6198 6198 return (MDI_FAILURE);
6199 6199 }
6200 6200
6201 6201 (void) i_mdi_enable_disable_path(pip, ph->ph_vhci, flags,
6202 6202 MDI_ENABLE_OP);
6203 6203 MDI_DEBUG(5, (MDI_NOTE, ph->ph_dip,
6204 6204 "!returning success pip = %p. ph = %p",
6205 6205 (void *)pip, (void *)ph));
6206 6206 return (MDI_SUCCESS);
6207 6207
6208 6208 }
6209 6209
6210 6210 /*
6211 6211 * Disable the path (specific client/target/initiator)
6212 6212 * Disabling a path means that MPxIO will not select the disabled path for
6213 6213 * routing any new I/O requests.
6214 6214 */
6215 6215 int
6216 6216 mdi_pi_disable_path(mdi_pathinfo_t *pip, int flags)
6217 6217 {
6218 6218 mdi_phci_t *ph;
6219 6219
6220 6220 ph = MDI_PI(pip)->pi_phci;
6221 6221 if (ph == NULL) {
6222 6222 MDI_DEBUG(1, (MDI_NOTE, mdi_pi_get_phci(pip),
6223 6223 "!failed: path %s %p: NULL ph",
6224 6224 mdi_pi_spathname(pip), (void *)pip));
6225 6225 return (MDI_FAILURE);
6226 6226 }
6227 6227
6228 6228 (void) i_mdi_enable_disable_path(pip,
6229 6229 ph->ph_vhci, flags, MDI_DISABLE_OP);
6230 6230 MDI_DEBUG(5, (MDI_NOTE, ph->ph_dip,
6231 6231 "!returning success pip = %p. ph = %p",
6232 6232 (void *)pip, (void *)ph));
6233 6233 return (MDI_SUCCESS);
6234 6234 }
6235 6235
6236 6236 /*
6237 6237 * disable the path to a particular pHCI (pHCI specified in the phci_path
6238 6238 * argument) for a particular client (specified in the client_path argument).
6239 6239 * Disabling a path means that MPxIO will not select the disabled path for
6240 6240 * routing any new I/O requests.
6241 6241 * NOTE: this will be removed once the NWS files are changed to use the new
6242 6242 * mdi_{enable,disable}_path interfaces
6243 6243 */
6244 6244 int
6245 6245 mdi_pi_disable(dev_info_t *cdip, dev_info_t *pdip, int flags)
6246 6246 {
6247 6247 return (i_mdi_pi_enable_disable(cdip, pdip, flags, MDI_DISABLE_OP));
6248 6248 }
6249 6249
6250 6250 /*
6251 6251 * Enable the path to a particular pHCI (pHCI specified in the phci_path
6252 6252 * argument) for a particular client (specified in the client_path argument).
6253 6253 * Enabling a path means that MPxIO may select the enabled path for routing
6254 6254 * future I/O requests, subject to other path state constraints.
6255 6255 * NOTE: this will be removed once the NWS files are changed to use the new
6256 6256 * mdi_{enable,disable}_path interfaces
6257 6257 */
6258 6258
6259 6259 int
6260 6260 mdi_pi_enable(dev_info_t *cdip, dev_info_t *pdip, int flags)
6261 6261 {
6262 6262 return (i_mdi_pi_enable_disable(cdip, pdip, flags, MDI_ENABLE_OP));
6263 6263 }
6264 6264
6265 6265 /*
6266 6266 * Common routine for doing enable/disable.
6267 6267 */
6268 6268 static mdi_pathinfo_t *
6269 6269 i_mdi_enable_disable_path(mdi_pathinfo_t *pip, mdi_vhci_t *vh, int flags,
6270 6270 int op)
6271 6271 {
6272 6272 int sync_flag = 0;
6273 6273 int rv;
6274 6274 mdi_pathinfo_t *next;
6275 6275 int (*f)() = NULL;
6276 6276
6277 6277 /*
6278 6278 * Check to make sure the path is not already in the
6279 6279 * requested state. If it is just return the next path
6280 6280 * as we have nothing to do here.
6281 6281 */
6282 6282 if ((MDI_PI_IS_DISABLE(pip) && op == MDI_DISABLE_OP) ||
6283 6283 (!MDI_PI_IS_DISABLE(pip) && op == MDI_ENABLE_OP)) {
6284 6284 MDI_PI_LOCK(pip);
6285 6285 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
6286 6286 MDI_PI_UNLOCK(pip);
6287 6287 return (next);
6288 6288 }
6289 6289
6290 6290 f = vh->vh_ops->vo_pi_state_change;
6291 6291
6292 6292 sync_flag = (flags << 8) & 0xf00;
6293 6293
6294 6294 /*
6295 6295 * Do a callback into the mdi consumer to let it
6296 6296 * know that path is about to get enabled/disabled.
6297 6297 */
6298 6298 if (f != NULL) {
6299 6299 rv = (*f)(vh->vh_dip, pip, 0,
6300 6300 MDI_PI_EXT_STATE(pip),
6301 6301 MDI_EXT_STATE_CHANGE | sync_flag |
6302 6302 op | MDI_BEFORE_STATE_CHANGE);
6303 6303 if (rv != MDI_SUCCESS) {
6304 6304 MDI_DEBUG(2, (MDI_WARN, vh->vh_dip,
6305 6305 "vo_pi_state_change: failed rv = %x", rv));
6306 6306 }
6307 6307 }
6308 6308 MDI_PI_LOCK(pip);
6309 6309 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_phci_link;
6310 6310
6311 6311 switch (flags) {
6312 6312 case USER_DISABLE:
6313 6313 if (op == MDI_DISABLE_OP) {
6314 6314 MDI_PI_SET_USER_DISABLE(pip);
6315 6315 } else {
6316 6316 MDI_PI_SET_USER_ENABLE(pip);
6317 6317 }
6318 6318 break;
6319 6319 case DRIVER_DISABLE:
6320 6320 if (op == MDI_DISABLE_OP) {
6321 6321 MDI_PI_SET_DRV_DISABLE(pip);
6322 6322 } else {
6323 6323 MDI_PI_SET_DRV_ENABLE(pip);
6324 6324 }
6325 6325 break;
6326 6326 case DRIVER_DISABLE_TRANSIENT:
6327 6327 if (op == MDI_DISABLE_OP && rv == MDI_SUCCESS) {
6328 6328 MDI_PI_SET_DRV_DISABLE_TRANS(pip);
6329 6329 } else {
6330 6330 MDI_PI_SET_DRV_ENABLE_TRANS(pip);
6331 6331 }
6332 6332 break;
6333 6333 }
6334 6334 MDI_PI_UNLOCK(pip);
6335 6335 /*
6336 6336 * Do a callback into the mdi consumer to let it
6337 6337 * know that path is now enabled/disabled.
6338 6338 */
6339 6339 if (f != NULL) {
6340 6340 rv = (*f)(vh->vh_dip, pip, 0,
6341 6341 MDI_PI_EXT_STATE(pip),
6342 6342 MDI_EXT_STATE_CHANGE | sync_flag |
6343 6343 op | MDI_AFTER_STATE_CHANGE);
6344 6344 if (rv != MDI_SUCCESS) {
6345 6345 MDI_DEBUG(2, (MDI_WARN, vh->vh_dip,
6346 6346 "vo_pi_state_change failed: rv = %x", rv));
6347 6347 }
6348 6348 }
6349 6349 return (next);
6350 6350 }
6351 6351
6352 6352 /*
6353 6353 * Common routine for doing enable/disable.
6354 6354 * NOTE: this will be removed once the NWS files are changed to use the new
6355 6355 * mdi_{enable,disable}_path has been putback
6356 6356 */
6357 6357 int
6358 6358 i_mdi_pi_enable_disable(dev_info_t *cdip, dev_info_t *pdip, int flags, int op)
6359 6359 {
6360 6360
6361 6361 mdi_phci_t *ph;
6362 6362 mdi_vhci_t *vh = NULL;
6363 6363 mdi_client_t *ct;
6364 6364 mdi_pathinfo_t *next, *pip;
6365 6365 int found_it;
6366 6366
6367 6367 ph = i_devi_get_phci(pdip);
6368 6368 MDI_DEBUG(5, (MDI_NOTE, cdip ? cdip : pdip,
6369 6369 "!op = %d pdip = %p cdip = %p", op, (void *)pdip,
6370 6370 (void *)cdip));
6371 6371 if (ph == NULL) {
6372 6372 MDI_DEBUG(1, (MDI_NOTE, cdip ? cdip : pdip,
6373 6373 "!failed: operation %d: NULL ph", op));
6374 6374 return (MDI_FAILURE);
6375 6375 }
6376 6376
6377 6377 if ((op != MDI_ENABLE_OP) && (op != MDI_DISABLE_OP)) {
6378 6378 MDI_DEBUG(1, (MDI_NOTE, cdip ? cdip : pdip,
6379 6379 "!failed: invalid operation %d", op));
6380 6380 return (MDI_FAILURE);
6381 6381 }
6382 6382
6383 6383 vh = ph->ph_vhci;
6384 6384
6385 6385 if (cdip == NULL) {
6386 6386 /*
6387 6387 * Need to mark the Phci as enabled/disabled.
6388 6388 */
6389 6389 MDI_DEBUG(4, (MDI_NOTE, cdip ? cdip : pdip,
6390 6390 "op %d for the phci", op));
6391 6391 MDI_PHCI_LOCK(ph);
6392 6392 switch (flags) {
6393 6393 case USER_DISABLE:
6394 6394 if (op == MDI_DISABLE_OP) {
6395 6395 MDI_PHCI_SET_USER_DISABLE(ph);
6396 6396 } else {
6397 6397 MDI_PHCI_SET_USER_ENABLE(ph);
6398 6398 }
6399 6399 break;
6400 6400 case DRIVER_DISABLE:
6401 6401 if (op == MDI_DISABLE_OP) {
6402 6402 MDI_PHCI_SET_DRV_DISABLE(ph);
6403 6403 } else {
6404 6404 MDI_PHCI_SET_DRV_ENABLE(ph);
6405 6405 }
6406 6406 break;
6407 6407 case DRIVER_DISABLE_TRANSIENT:
6408 6408 if (op == MDI_DISABLE_OP) {
6409 6409 MDI_PHCI_SET_DRV_DISABLE_TRANSIENT(ph);
6410 6410 } else {
6411 6411 MDI_PHCI_SET_DRV_ENABLE_TRANSIENT(ph);
6412 6412 }
6413 6413 break;
6414 6414 default:
6415 6415 MDI_PHCI_UNLOCK(ph);
6416 6416 MDI_DEBUG(1, (MDI_NOTE, cdip ? cdip : pdip,
6417 6417 "!invalid flag argument= %d", flags));
6418 6418 }
6419 6419
6420 6420 /*
6421 6421 * Phci has been disabled. Now try to enable/disable
6422 6422 * path info's to each client.
6423 6423 */
6424 6424 pip = ph->ph_path_head;
6425 6425 while (pip != NULL) {
6426 6426 pip = i_mdi_enable_disable_path(pip, vh, flags, op);
6427 6427 }
6428 6428 MDI_PHCI_UNLOCK(ph);
6429 6429 } else {
6430 6430
6431 6431 /*
6432 6432 * Disable a specific client.
6433 6433 */
6434 6434 ct = i_devi_get_client(cdip);
6435 6435 if (ct == NULL) {
6436 6436 MDI_DEBUG(1, (MDI_NOTE, cdip ? cdip : pdip,
6437 6437 "!failed: operation = %d: NULL ct", op));
6438 6438 return (MDI_FAILURE);
6439 6439 }
6440 6440
6441 6441 MDI_CLIENT_LOCK(ct);
6442 6442 pip = ct->ct_path_head;
6443 6443 found_it = 0;
6444 6444 while (pip != NULL) {
6445 6445 MDI_PI_LOCK(pip);
6446 6446 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
6447 6447 if (MDI_PI(pip)->pi_phci == ph) {
6448 6448 MDI_PI_UNLOCK(pip);
6449 6449 found_it = 1;
6450 6450 break;
6451 6451 }
6452 6452 MDI_PI_UNLOCK(pip);
6453 6453 pip = next;
6454 6454 }
6455 6455
6456 6456
6457 6457 MDI_CLIENT_UNLOCK(ct);
6458 6458 if (found_it == 0) {
6459 6459 MDI_DEBUG(1, (MDI_NOTE, cdip ? cdip : pdip,
6460 6460 "!failed. Could not find corresponding pip\n"));
6461 6461 return (MDI_FAILURE);
6462 6462 }
6463 6463
6464 6464 (void) i_mdi_enable_disable_path(pip, vh, flags, op);
6465 6465 }
6466 6466
6467 6467 MDI_DEBUG(5, (MDI_NOTE, cdip ? cdip : pdip,
6468 6468 "!op %d returning success pdip = %p cdip = %p",
6469 6469 op, (void *)pdip, (void *)cdip));
6470 6470 return (MDI_SUCCESS);
6471 6471 }
6472 6472
6473 6473 /*
6474 6474 * Ensure phci powered up
6475 6475 */
6476 6476 static void
6477 6477 i_mdi_pm_hold_pip(mdi_pathinfo_t *pip)
6478 6478 {
6479 6479 dev_info_t *ph_dip;
6480 6480
6481 6481 ASSERT(pip != NULL);
6482 6482 ASSERT(MDI_PI_LOCKED(pip));
6483 6483
6484 6484 if (MDI_PI(pip)->pi_pm_held) {
6485 6485 return;
6486 6486 }
6487 6487
6488 6488 ph_dip = mdi_pi_get_phci(pip);
6489 6489 MDI_DEBUG(4, (MDI_NOTE, ph_dip,
6490 6490 "%s %p", mdi_pi_spathname(pip), (void *)pip));
6491 6491 if (ph_dip == NULL) {
6492 6492 return;
6493 6493 }
6494 6494
6495 6495 MDI_PI_UNLOCK(pip);
6496 6496 MDI_DEBUG(4, (MDI_NOTE, ph_dip, "kidsupcnt was %d",
6497 6497 DEVI(ph_dip)->devi_pm_kidsupcnt));
6498 6498 pm_hold_power(ph_dip);
6499 6499 MDI_DEBUG(4, (MDI_NOTE, ph_dip, "kidsupcnt is %d",
6500 6500 DEVI(ph_dip)->devi_pm_kidsupcnt));
6501 6501 MDI_PI_LOCK(pip);
6502 6502
6503 6503 /* If PM_GET_PM_INFO is NULL the pm_hold_power above was a noop */
6504 6504 if (DEVI(ph_dip)->devi_pm_info)
6505 6505 MDI_PI(pip)->pi_pm_held = 1;
6506 6506 }
6507 6507
6508 6508 /*
6509 6509 * Allow phci powered down
6510 6510 */
6511 6511 static void
6512 6512 i_mdi_pm_rele_pip(mdi_pathinfo_t *pip)
6513 6513 {
6514 6514 dev_info_t *ph_dip = NULL;
6515 6515
6516 6516 ASSERT(pip != NULL);
6517 6517 ASSERT(MDI_PI_LOCKED(pip));
6518 6518
6519 6519 if (MDI_PI(pip)->pi_pm_held == 0) {
6520 6520 return;
6521 6521 }
6522 6522
6523 6523 ph_dip = mdi_pi_get_phci(pip);
6524 6524 ASSERT(ph_dip != NULL);
6525 6525
6526 6526 MDI_DEBUG(4, (MDI_NOTE, ph_dip,
6527 6527 "%s %p", mdi_pi_spathname(pip), (void *)pip));
6528 6528
6529 6529 MDI_PI_UNLOCK(pip);
6530 6530 MDI_DEBUG(4, (MDI_NOTE, ph_dip,
6531 6531 "kidsupcnt was %d", DEVI(ph_dip)->devi_pm_kidsupcnt));
6532 6532 pm_rele_power(ph_dip);
6533 6533 MDI_DEBUG(4, (MDI_NOTE, ph_dip,
6534 6534 "kidsupcnt is %d", DEVI(ph_dip)->devi_pm_kidsupcnt));
6535 6535 MDI_PI_LOCK(pip);
6536 6536
6537 6537 MDI_PI(pip)->pi_pm_held = 0;
6538 6538 }
6539 6539
6540 6540 static void
6541 6541 i_mdi_pm_hold_client(mdi_client_t *ct, int incr)
6542 6542 {
6543 6543 ASSERT(MDI_CLIENT_LOCKED(ct));
6544 6544
6545 6545 ct->ct_power_cnt += incr;
6546 6546 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
6547 6547 "%p ct_power_cnt = %d incr = %d",
6548 6548 (void *)ct, ct->ct_power_cnt, incr));
6549 6549 ASSERT(ct->ct_power_cnt >= 0);
6550 6550 }
6551 6551
6552 6552 static void
6553 6553 i_mdi_rele_all_phci(mdi_client_t *ct)
6554 6554 {
6555 6555 mdi_pathinfo_t *pip;
6556 6556
6557 6557 ASSERT(MDI_CLIENT_LOCKED(ct));
6558 6558 pip = (mdi_pathinfo_t *)ct->ct_path_head;
6559 6559 while (pip != NULL) {
6560 6560 mdi_hold_path(pip);
6561 6561 MDI_PI_LOCK(pip);
6562 6562 i_mdi_pm_rele_pip(pip);
6563 6563 MDI_PI_UNLOCK(pip);
6564 6564 mdi_rele_path(pip);
6565 6565 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
6566 6566 }
6567 6567 }
6568 6568
6569 6569 static void
6570 6570 i_mdi_pm_rele_client(mdi_client_t *ct, int decr)
6571 6571 {
6572 6572 ASSERT(MDI_CLIENT_LOCKED(ct));
6573 6573
6574 6574 if (i_ddi_devi_attached(ct->ct_dip)) {
6575 6575 ct->ct_power_cnt -= decr;
6576 6576 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
6577 6577 "%p ct_power_cnt = %d decr = %d",
6578 6578 (void *)ct, ct->ct_power_cnt, decr));
6579 6579 }
6580 6580
6581 6581 ASSERT(ct->ct_power_cnt >= 0);
6582 6582 if (ct->ct_power_cnt == 0) {
6583 6583 i_mdi_rele_all_phci(ct);
6584 6584 return;
6585 6585 }
6586 6586 }
6587 6587
6588 6588 static void
6589 6589 i_mdi_pm_reset_client(mdi_client_t *ct)
6590 6590 {
6591 6591 MDI_DEBUG(4, (MDI_NOTE, ct->ct_dip,
6592 6592 "%p ct_power_cnt = %d", (void *)ct, ct->ct_power_cnt));
6593 6593 ASSERT(MDI_CLIENT_LOCKED(ct));
6594 6594 ct->ct_power_cnt = 0;
6595 6595 i_mdi_rele_all_phci(ct);
6596 6596 ct->ct_powercnt_config = 0;
6597 6597 ct->ct_powercnt_unconfig = 0;
6598 6598 ct->ct_powercnt_reset = 1;
6599 6599 }
6600 6600
6601 6601 static int
6602 6602 i_mdi_power_one_phci(mdi_pathinfo_t *pip)
6603 6603 {
6604 6604 int ret;
6605 6605 dev_info_t *ph_dip;
6606 6606
6607 6607 MDI_PI_LOCK(pip);
6608 6608 i_mdi_pm_hold_pip(pip);
6609 6609
6610 6610 ph_dip = mdi_pi_get_phci(pip);
6611 6611 MDI_PI_UNLOCK(pip);
6612 6612
6613 6613 /* bring all components of phci to full power */
6614 6614 MDI_DEBUG(4, (MDI_NOTE, ph_dip,
6615 6615 "pm_powerup for %s%d %p", ddi_driver_name(ph_dip),
6616 6616 ddi_get_instance(ph_dip), (void *)pip));
6617 6617
6618 6618 ret = pm_powerup(ph_dip);
6619 6619
6620 6620 if (ret == DDI_FAILURE) {
6621 6621 MDI_DEBUG(4, (MDI_NOTE, ph_dip,
6622 6622 "pm_powerup FAILED for %s%d %p",
6623 6623 ddi_driver_name(ph_dip), ddi_get_instance(ph_dip),
6624 6624 (void *)pip));
6625 6625
6626 6626 MDI_PI_LOCK(pip);
6627 6627 i_mdi_pm_rele_pip(pip);
6628 6628 MDI_PI_UNLOCK(pip);
6629 6629 return (MDI_FAILURE);
6630 6630 }
6631 6631
6632 6632 return (MDI_SUCCESS);
6633 6633 }
6634 6634
6635 6635 static int
6636 6636 i_mdi_power_all_phci(mdi_client_t *ct)
6637 6637 {
6638 6638 mdi_pathinfo_t *pip;
6639 6639 int succeeded = 0;
6640 6640
6641 6641 ASSERT(MDI_CLIENT_LOCKED(ct));
6642 6642 pip = (mdi_pathinfo_t *)ct->ct_path_head;
6643 6643 while (pip != NULL) {
6644 6644 /*
6645 6645 * Don't power if MDI_PATHINFO_STATE_FAULT
6646 6646 * or MDI_PATHINFO_STATE_OFFLINE.
6647 6647 */
6648 6648 if (MDI_PI_IS_INIT(pip) ||
6649 6649 MDI_PI_IS_ONLINE(pip) || MDI_PI_IS_STANDBY(pip)) {
6650 6650 mdi_hold_path(pip);
6651 6651 MDI_CLIENT_UNLOCK(ct);
6652 6652 if (i_mdi_power_one_phci(pip) == MDI_SUCCESS)
6653 6653 succeeded = 1;
6654 6654
6655 6655 ASSERT(ct == MDI_PI(pip)->pi_client);
6656 6656 MDI_CLIENT_LOCK(ct);
6657 6657 mdi_rele_path(pip);
6658 6658 }
6659 6659 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
6660 6660 }
6661 6661
6662 6662 return (succeeded ? MDI_SUCCESS : MDI_FAILURE);
6663 6663 }
6664 6664
6665 6665 /*
6666 6666 * mdi_bus_power():
6667 6667 * 1. Place the phci(s) into powered up state so that
6668 6668 * client can do power management
6669 6669 * 2. Ensure phci powered up as client power managing
6670 6670 * Return Values:
6671 6671 * MDI_SUCCESS
6672 6672 * MDI_FAILURE
6673 6673 */
6674 6674 int
6675 6675 mdi_bus_power(dev_info_t *parent, void *impl_arg, pm_bus_power_op_t op,
6676 6676 void *arg, void *result)
6677 6677 {
6678 6678 int ret = MDI_SUCCESS;
6679 6679 pm_bp_child_pwrchg_t *bpc;
6680 6680 mdi_client_t *ct;
6681 6681 dev_info_t *cdip;
6682 6682 pm_bp_has_changed_t *bphc;
6683 6683
6684 6684 /*
6685 6685 * BUS_POWER_NOINVOL not supported
6686 6686 */
6687 6687 if (op == BUS_POWER_NOINVOL)
6688 6688 return (MDI_FAILURE);
6689 6689
6690 6690 /*
6691 6691 * ignore other OPs.
6692 6692 * return quickly to save cou cycles on the ct processing
6693 6693 */
6694 6694 switch (op) {
6695 6695 case BUS_POWER_PRE_NOTIFICATION:
6696 6696 case BUS_POWER_POST_NOTIFICATION:
6697 6697 bpc = (pm_bp_child_pwrchg_t *)arg;
6698 6698 cdip = bpc->bpc_dip;
6699 6699 break;
6700 6700 case BUS_POWER_HAS_CHANGED:
6701 6701 bphc = (pm_bp_has_changed_t *)arg;
6702 6702 cdip = bphc->bphc_dip;
6703 6703 break;
6704 6704 default:
6705 6705 return (pm_busop_bus_power(parent, impl_arg, op, arg, result));
6706 6706 }
6707 6707
6708 6708 ASSERT(MDI_CLIENT(cdip));
6709 6709
6710 6710 ct = i_devi_get_client(cdip);
6711 6711 if (ct == NULL)
6712 6712 return (MDI_FAILURE);
6713 6713
6714 6714 /*
6715 6715 * wait till the mdi_pathinfo node state change are processed
6716 6716 */
6717 6717 MDI_CLIENT_LOCK(ct);
6718 6718 switch (op) {
6719 6719 case BUS_POWER_PRE_NOTIFICATION:
6720 6720 MDI_DEBUG(4, (MDI_NOTE, bpc->bpc_dip,
6721 6721 "BUS_POWER_PRE_NOTIFICATION:"
6722 6722 "%s@%s, olevel=%d, nlevel=%d, comp=%d",
6723 6723 ddi_node_name(bpc->bpc_dip), PM_ADDR(bpc->bpc_dip),
6724 6724 bpc->bpc_olevel, bpc->bpc_nlevel, bpc->bpc_comp));
6725 6725
6726 6726 /* serialize power level change per client */
6727 6727 while (MDI_CLIENT_IS_POWER_TRANSITION(ct))
6728 6728 cv_wait(&ct->ct_powerchange_cv, &ct->ct_mutex);
6729 6729
6730 6730 MDI_CLIENT_SET_POWER_TRANSITION(ct);
6731 6731
6732 6732 if (ct->ct_power_cnt == 0) {
6733 6733 ret = i_mdi_power_all_phci(ct);
6734 6734 }
6735 6735
6736 6736 /*
6737 6737 * if new_level > 0:
6738 6738 * - hold phci(s)
6739 6739 * - power up phci(s) if not already
6740 6740 * ignore power down
6741 6741 */
6742 6742 if (bpc->bpc_nlevel > 0) {
6743 6743 if (!DEVI_IS_ATTACHING(ct->ct_dip)) {
6744 6744 MDI_DEBUG(4, (MDI_NOTE, bpc->bpc_dip,
6745 6745 "i_mdi_pm_hold_client\n"));
6746 6746 i_mdi_pm_hold_client(ct, ct->ct_path_count);
6747 6747 }
6748 6748 }
6749 6749 break;
6750 6750 case BUS_POWER_POST_NOTIFICATION:
6751 6751 MDI_DEBUG(4, (MDI_NOTE, bpc->bpc_dip,
6752 6752 "BUS_POWER_POST_NOTIFICATION:"
6753 6753 "%s@%s, olevel=%d, nlevel=%d, comp=%d result=%d",
6754 6754 ddi_node_name(bpc->bpc_dip), PM_ADDR(bpc->bpc_dip),
6755 6755 bpc->bpc_olevel, bpc->bpc_nlevel, bpc->bpc_comp,
6756 6756 *(int *)result));
6757 6757
6758 6758 if (*(int *)result == DDI_SUCCESS) {
6759 6759 if (bpc->bpc_nlevel > 0) {
6760 6760 MDI_CLIENT_SET_POWER_UP(ct);
6761 6761 } else {
6762 6762 MDI_CLIENT_SET_POWER_DOWN(ct);
6763 6763 }
6764 6764 }
6765 6765
6766 6766 /* release the hold we did in pre-notification */
6767 6767 if (bpc->bpc_nlevel > 0 && (*(int *)result != DDI_SUCCESS) &&
6768 6768 !DEVI_IS_ATTACHING(ct->ct_dip)) {
6769 6769 MDI_DEBUG(4, (MDI_NOTE, bpc->bpc_dip,
6770 6770 "i_mdi_pm_rele_client\n"));
6771 6771 i_mdi_pm_rele_client(ct, ct->ct_path_count);
6772 6772 }
6773 6773
6774 6774 if (bpc->bpc_nlevel == 0 && (*(int *)result == DDI_SUCCESS)) {
6775 6775 /* another thread might started attaching */
6776 6776 if (DEVI_IS_ATTACHING(ct->ct_dip)) {
6777 6777 MDI_DEBUG(4, (MDI_NOTE, bpc->bpc_dip,
6778 6778 "i_mdi_pm_rele_client\n"));
6779 6779 i_mdi_pm_rele_client(ct, ct->ct_path_count);
6780 6780 /* detaching has been taken care in pm_post_unconfig */
6781 6781 } else if (!DEVI_IS_DETACHING(ct->ct_dip)) {
6782 6782 MDI_DEBUG(4, (MDI_NOTE, bpc->bpc_dip,
6783 6783 "i_mdi_pm_reset_client\n"));
6784 6784 i_mdi_pm_reset_client(ct);
6785 6785 }
6786 6786 }
6787 6787
6788 6788 MDI_CLIENT_CLEAR_POWER_TRANSITION(ct);
6789 6789 cv_broadcast(&ct->ct_powerchange_cv);
6790 6790
6791 6791 break;
6792 6792
6793 6793 /* need to do more */
6794 6794 case BUS_POWER_HAS_CHANGED:
6795 6795 MDI_DEBUG(4, (MDI_NOTE, bphc->bphc_dip,
6796 6796 "BUS_POWER_HAS_CHANGED:"
6797 6797 "%s@%s, olevel=%d, nlevel=%d, comp=%d",
6798 6798 ddi_node_name(bphc->bphc_dip), PM_ADDR(bphc->bphc_dip),
6799 6799 bphc->bphc_olevel, bphc->bphc_nlevel, bphc->bphc_comp));
6800 6800
6801 6801 if (bphc->bphc_nlevel > 0 &&
6802 6802 bphc->bphc_nlevel > bphc->bphc_olevel) {
6803 6803 if (ct->ct_power_cnt == 0) {
6804 6804 ret = i_mdi_power_all_phci(ct);
6805 6805 }
6806 6806 MDI_DEBUG(4, (MDI_NOTE, bphc->bphc_dip,
6807 6807 "i_mdi_pm_hold_client\n"));
6808 6808 i_mdi_pm_hold_client(ct, ct->ct_path_count);
6809 6809 }
6810 6810
6811 6811 if (bphc->bphc_nlevel == 0 && bphc->bphc_olevel != -1) {
6812 6812 MDI_DEBUG(4, (MDI_NOTE, bphc->bphc_dip,
6813 6813 "i_mdi_pm_rele_client\n"));
6814 6814 i_mdi_pm_rele_client(ct, ct->ct_path_count);
6815 6815 }
6816 6816 break;
6817 6817 }
6818 6818
6819 6819 MDI_CLIENT_UNLOCK(ct);
6820 6820 return (ret);
6821 6821 }
6822 6822
6823 6823 static int
6824 6824 i_mdi_pm_pre_config_one(dev_info_t *child)
6825 6825 {
6826 6826 int ret = MDI_SUCCESS;
6827 6827 mdi_client_t *ct;
6828 6828
6829 6829 ct = i_devi_get_client(child);
6830 6830 if (ct == NULL)
6831 6831 return (MDI_FAILURE);
6832 6832
6833 6833 MDI_CLIENT_LOCK(ct);
6834 6834 while (MDI_CLIENT_IS_POWER_TRANSITION(ct))
6835 6835 cv_wait(&ct->ct_powerchange_cv, &ct->ct_mutex);
6836 6836
6837 6837 if (!MDI_CLIENT_IS_FAILED(ct)) {
6838 6838 MDI_CLIENT_UNLOCK(ct);
6839 6839 MDI_DEBUG(4, (MDI_NOTE, child, "already configured\n"));
6840 6840 return (MDI_SUCCESS);
6841 6841 }
6842 6842
6843 6843 if (ct->ct_powercnt_config) {
6844 6844 MDI_CLIENT_UNLOCK(ct);
6845 6845 MDI_DEBUG(4, (MDI_NOTE, child, "already held\n"));
6846 6846 return (MDI_SUCCESS);
6847 6847 }
6848 6848
6849 6849 if (ct->ct_power_cnt == 0) {
6850 6850 ret = i_mdi_power_all_phci(ct);
6851 6851 }
6852 6852 MDI_DEBUG(4, (MDI_NOTE, child, "i_mdi_pm_hold_client\n"));
6853 6853 i_mdi_pm_hold_client(ct, ct->ct_path_count);
6854 6854 ct->ct_powercnt_config = 1;
6855 6855 ct->ct_powercnt_reset = 0;
6856 6856 MDI_CLIENT_UNLOCK(ct);
6857 6857 return (ret);
6858 6858 }
6859 6859
6860 6860 static int
6861 6861 i_mdi_pm_pre_config(dev_info_t *vdip, dev_info_t *child)
6862 6862 {
6863 6863 int ret = MDI_SUCCESS;
6864 6864 dev_info_t *cdip;
6865 6865 int circ;
6866 6866
6867 6867 ASSERT(MDI_VHCI(vdip));
6868 6868
6869 6869 /* ndi_devi_config_one */
6870 6870 if (child) {
6871 6871 ASSERT(DEVI_BUSY_OWNED(vdip));
6872 6872 return (i_mdi_pm_pre_config_one(child));
6873 6873 }
6874 6874
6875 6875 /* devi_config_common */
6876 6876 ndi_devi_enter(vdip, &circ);
6877 6877 cdip = ddi_get_child(vdip);
6878 6878 while (cdip) {
6879 6879 dev_info_t *next = ddi_get_next_sibling(cdip);
6880 6880
6881 6881 ret = i_mdi_pm_pre_config_one(cdip);
6882 6882 if (ret != MDI_SUCCESS)
6883 6883 break;
6884 6884 cdip = next;
6885 6885 }
6886 6886 ndi_devi_exit(vdip, circ);
6887 6887 return (ret);
6888 6888 }
6889 6889
6890 6890 static int
6891 6891 i_mdi_pm_pre_unconfig_one(dev_info_t *child, int *held, int flags)
6892 6892 {
6893 6893 int ret = MDI_SUCCESS;
6894 6894 mdi_client_t *ct;
6895 6895
6896 6896 ct = i_devi_get_client(child);
6897 6897 if (ct == NULL)
6898 6898 return (MDI_FAILURE);
6899 6899
6900 6900 MDI_CLIENT_LOCK(ct);
6901 6901 while (MDI_CLIENT_IS_POWER_TRANSITION(ct))
6902 6902 cv_wait(&ct->ct_powerchange_cv, &ct->ct_mutex);
6903 6903
6904 6904 if (!i_ddi_devi_attached(child)) {
6905 6905 MDI_DEBUG(4, (MDI_NOTE, child, "node detached already\n"));
6906 6906 MDI_CLIENT_UNLOCK(ct);
6907 6907 return (MDI_SUCCESS);
6908 6908 }
6909 6909
6910 6910 if (MDI_CLIENT_IS_POWERED_DOWN(ct) &&
6911 6911 (flags & NDI_AUTODETACH)) {
6912 6912 MDI_DEBUG(4, (MDI_NOTE, child, "auto-modunload\n"));
6913 6913 MDI_CLIENT_UNLOCK(ct);
6914 6914 return (MDI_FAILURE);
6915 6915 }
6916 6916
6917 6917 if (ct->ct_powercnt_unconfig) {
6918 6918 MDI_DEBUG(4, (MDI_NOTE, child, "ct_powercnt_held\n"));
6919 6919 MDI_CLIENT_UNLOCK(ct);
6920 6920 *held = 1;
6921 6921 return (MDI_SUCCESS);
6922 6922 }
6923 6923
6924 6924 if (ct->ct_power_cnt == 0) {
6925 6925 ret = i_mdi_power_all_phci(ct);
6926 6926 }
6927 6927 MDI_DEBUG(4, (MDI_NOTE, child, "i_mdi_pm_hold_client\n"));
6928 6928 i_mdi_pm_hold_client(ct, ct->ct_path_count);
6929 6929 ct->ct_powercnt_unconfig = 1;
6930 6930 ct->ct_powercnt_reset = 0;
6931 6931 MDI_CLIENT_UNLOCK(ct);
6932 6932 if (ret == MDI_SUCCESS)
6933 6933 *held = 1;
6934 6934 return (ret);
6935 6935 }
6936 6936
6937 6937 static int
6938 6938 i_mdi_pm_pre_unconfig(dev_info_t *vdip, dev_info_t *child, int *held,
6939 6939 int flags)
6940 6940 {
6941 6941 int ret = MDI_SUCCESS;
6942 6942 dev_info_t *cdip;
6943 6943 int circ;
6944 6944
6945 6945 ASSERT(MDI_VHCI(vdip));
6946 6946 *held = 0;
6947 6947
6948 6948 /* ndi_devi_unconfig_one */
6949 6949 if (child) {
6950 6950 ASSERT(DEVI_BUSY_OWNED(vdip));
6951 6951 return (i_mdi_pm_pre_unconfig_one(child, held, flags));
6952 6952 }
6953 6953
6954 6954 /* devi_unconfig_common */
6955 6955 ndi_devi_enter(vdip, &circ);
6956 6956 cdip = ddi_get_child(vdip);
6957 6957 while (cdip) {
6958 6958 dev_info_t *next = ddi_get_next_sibling(cdip);
6959 6959
6960 6960 ret = i_mdi_pm_pre_unconfig_one(cdip, held, flags);
6961 6961 cdip = next;
6962 6962 }
6963 6963 ndi_devi_exit(vdip, circ);
6964 6964
6965 6965 if (*held)
6966 6966 ret = MDI_SUCCESS;
6967 6967
6968 6968 return (ret);
6969 6969 }
6970 6970
6971 6971 static void
6972 6972 i_mdi_pm_post_config_one(dev_info_t *child)
6973 6973 {
6974 6974 mdi_client_t *ct;
6975 6975
6976 6976 ct = i_devi_get_client(child);
6977 6977 if (ct == NULL)
6978 6978 return;
6979 6979
6980 6980 MDI_CLIENT_LOCK(ct);
6981 6981 while (MDI_CLIENT_IS_POWER_TRANSITION(ct))
6982 6982 cv_wait(&ct->ct_powerchange_cv, &ct->ct_mutex);
6983 6983
6984 6984 if (ct->ct_powercnt_reset || !ct->ct_powercnt_config) {
6985 6985 MDI_DEBUG(4, (MDI_NOTE, child, "not configured\n"));
6986 6986 MDI_CLIENT_UNLOCK(ct);
6987 6987 return;
6988 6988 }
6989 6989
6990 6990 /* client has not been updated */
6991 6991 if (MDI_CLIENT_IS_FAILED(ct)) {
6992 6992 MDI_DEBUG(4, (MDI_NOTE, child, "client failed\n"));
6993 6993 MDI_CLIENT_UNLOCK(ct);
6994 6994 return;
6995 6995 }
6996 6996
6997 6997 /* another thread might have powered it down or detached it */
6998 6998 if ((MDI_CLIENT_IS_POWERED_DOWN(ct) &&
6999 6999 !DEVI_IS_ATTACHING(child)) ||
7000 7000 (!i_ddi_devi_attached(child) &&
7001 7001 !DEVI_IS_ATTACHING(child))) {
7002 7002 MDI_DEBUG(4, (MDI_NOTE, child, "i_mdi_pm_reset_client\n"));
7003 7003 i_mdi_pm_reset_client(ct);
7004 7004 } else {
7005 7005 mdi_pathinfo_t *pip, *next;
7006 7006 int valid_path_count = 0;
7007 7007
7008 7008 MDI_DEBUG(4, (MDI_NOTE, child, "i_mdi_pm_rele_client\n"));
7009 7009 pip = ct->ct_path_head;
7010 7010 while (pip != NULL) {
7011 7011 MDI_PI_LOCK(pip);
7012 7012 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
7013 7013 if (MDI_PI_IS_ONLINE(pip) || MDI_PI_IS_STANDBY(pip))
7014 7014 valid_path_count ++;
7015 7015 MDI_PI_UNLOCK(pip);
7016 7016 pip = next;
7017 7017 }
7018 7018 i_mdi_pm_rele_client(ct, valid_path_count);
7019 7019 }
7020 7020 ct->ct_powercnt_config = 0;
7021 7021 MDI_CLIENT_UNLOCK(ct);
7022 7022 }
7023 7023
7024 7024 static void
7025 7025 i_mdi_pm_post_config(dev_info_t *vdip, dev_info_t *child)
7026 7026 {
7027 7027 int circ;
7028 7028 dev_info_t *cdip;
7029 7029
7030 7030 ASSERT(MDI_VHCI(vdip));
7031 7031
7032 7032 /* ndi_devi_config_one */
7033 7033 if (child) {
7034 7034 ASSERT(DEVI_BUSY_OWNED(vdip));
7035 7035 i_mdi_pm_post_config_one(child);
7036 7036 return;
7037 7037 }
7038 7038
7039 7039 /* devi_config_common */
7040 7040 ndi_devi_enter(vdip, &circ);
7041 7041 cdip = ddi_get_child(vdip);
7042 7042 while (cdip) {
7043 7043 dev_info_t *next = ddi_get_next_sibling(cdip);
7044 7044
7045 7045 i_mdi_pm_post_config_one(cdip);
7046 7046 cdip = next;
7047 7047 }
7048 7048 ndi_devi_exit(vdip, circ);
7049 7049 }
7050 7050
7051 7051 static void
7052 7052 i_mdi_pm_post_unconfig_one(dev_info_t *child)
7053 7053 {
7054 7054 mdi_client_t *ct;
7055 7055
7056 7056 ct = i_devi_get_client(child);
7057 7057 if (ct == NULL)
7058 7058 return;
7059 7059
7060 7060 MDI_CLIENT_LOCK(ct);
7061 7061 while (MDI_CLIENT_IS_POWER_TRANSITION(ct))
7062 7062 cv_wait(&ct->ct_powerchange_cv, &ct->ct_mutex);
7063 7063
7064 7064 if (!ct->ct_powercnt_unconfig || ct->ct_powercnt_reset) {
7065 7065 MDI_DEBUG(4, (MDI_NOTE, child, "not held\n"));
7066 7066 MDI_CLIENT_UNLOCK(ct);
7067 7067 return;
7068 7068 }
7069 7069
7070 7070 /* failure detaching or another thread just attached it */
7071 7071 if ((MDI_CLIENT_IS_POWERED_DOWN(ct) &&
7072 7072 i_ddi_devi_attached(child)) ||
7073 7073 (!i_ddi_devi_attached(child) &&
7074 7074 !DEVI_IS_ATTACHING(child))) {
7075 7075 MDI_DEBUG(4, (MDI_NOTE, child, "i_mdi_pm_reset_client\n"));
7076 7076 i_mdi_pm_reset_client(ct);
7077 7077 } else {
7078 7078 mdi_pathinfo_t *pip, *next;
7079 7079 int valid_path_count = 0;
7080 7080
7081 7081 MDI_DEBUG(4, (MDI_NOTE, child, "i_mdi_pm_rele_client\n"));
7082 7082 pip = ct->ct_path_head;
7083 7083 while (pip != NULL) {
7084 7084 MDI_PI_LOCK(pip);
7085 7085 next = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link;
7086 7086 if (MDI_PI_IS_ONLINE(pip) || MDI_PI_IS_STANDBY(pip))
7087 7087 valid_path_count ++;
7088 7088 MDI_PI_UNLOCK(pip);
7089 7089 pip = next;
7090 7090 }
7091 7091 i_mdi_pm_rele_client(ct, valid_path_count);
7092 7092 ct->ct_powercnt_unconfig = 0;
7093 7093 }
7094 7094
7095 7095 MDI_CLIENT_UNLOCK(ct);
7096 7096 }
7097 7097
7098 7098 static void
7099 7099 i_mdi_pm_post_unconfig(dev_info_t *vdip, dev_info_t *child, int held)
7100 7100 {
7101 7101 int circ;
7102 7102 dev_info_t *cdip;
7103 7103
7104 7104 ASSERT(MDI_VHCI(vdip));
7105 7105
7106 7106 if (!held) {
7107 7107 MDI_DEBUG(4, (MDI_NOTE, vdip, "held = %d", held));
7108 7108 return;
7109 7109 }
7110 7110
7111 7111 if (child) {
7112 7112 ASSERT(DEVI_BUSY_OWNED(vdip));
7113 7113 i_mdi_pm_post_unconfig_one(child);
7114 7114 return;
7115 7115 }
7116 7116
7117 7117 ndi_devi_enter(vdip, &circ);
7118 7118 cdip = ddi_get_child(vdip);
7119 7119 while (cdip) {
7120 7120 dev_info_t *next = ddi_get_next_sibling(cdip);
7121 7121
7122 7122 i_mdi_pm_post_unconfig_one(cdip);
7123 7123 cdip = next;
7124 7124 }
7125 7125 ndi_devi_exit(vdip, circ);
7126 7126 }
7127 7127
7128 7128 int
7129 7129 mdi_power(dev_info_t *vdip, mdi_pm_op_t op, void *args, char *devnm, int flags)
7130 7130 {
7131 7131 int circ, ret = MDI_SUCCESS;
7132 7132 dev_info_t *client_dip = NULL;
7133 7133 mdi_client_t *ct;
7134 7134
7135 7135 /*
7136 7136 * Handling ndi_devi_config_one and ndi_devi_unconfig_one.
7137 7137 * Power up pHCI for the named client device.
7138 7138 * Note: Before the client is enumerated under vhci by phci,
7139 7139 * client_dip can be NULL. Then proceed to power up all the
7140 7140 * pHCIs.
7141 7141 */
7142 7142 if (devnm != NULL) {
7143 7143 ndi_devi_enter(vdip, &circ);
7144 7144 client_dip = ndi_devi_findchild(vdip, devnm);
7145 7145 }
7146 7146
7147 7147 MDI_DEBUG(4, (MDI_NOTE, vdip,
7148 7148 "op = %d %s %p", op, devnm ? devnm : "", (void *)client_dip));
7149 7149
7150 7150 switch (op) {
7151 7151 case MDI_PM_PRE_CONFIG:
7152 7152 ret = i_mdi_pm_pre_config(vdip, client_dip);
7153 7153 break;
7154 7154
7155 7155 case MDI_PM_PRE_UNCONFIG:
7156 7156 ret = i_mdi_pm_pre_unconfig(vdip, client_dip, (int *)args,
7157 7157 flags);
7158 7158 break;
7159 7159
7160 7160 case MDI_PM_POST_CONFIG:
7161 7161 i_mdi_pm_post_config(vdip, client_dip);
7162 7162 break;
7163 7163
7164 7164 case MDI_PM_POST_UNCONFIG:
7165 7165 i_mdi_pm_post_unconfig(vdip, client_dip, *(int *)args);
7166 7166 break;
7167 7167
7168 7168 case MDI_PM_HOLD_POWER:
7169 7169 case MDI_PM_RELE_POWER:
7170 7170 ASSERT(args);
7171 7171
7172 7172 client_dip = (dev_info_t *)args;
7173 7173 ASSERT(MDI_CLIENT(client_dip));
7174 7174
7175 7175 ct = i_devi_get_client(client_dip);
7176 7176 MDI_CLIENT_LOCK(ct);
7177 7177
7178 7178 if (op == MDI_PM_HOLD_POWER) {
7179 7179 if (ct->ct_power_cnt == 0) {
7180 7180 (void) i_mdi_power_all_phci(ct);
7181 7181 MDI_DEBUG(4, (MDI_NOTE, client_dip,
7182 7182 "i_mdi_pm_hold_client\n"));
7183 7183 i_mdi_pm_hold_client(ct, ct->ct_path_count);
7184 7184 }
7185 7185 } else {
7186 7186 if (DEVI_IS_ATTACHING(client_dip)) {
7187 7187 MDI_DEBUG(4, (MDI_NOTE, client_dip,
7188 7188 "i_mdi_pm_rele_client\n"));
7189 7189 i_mdi_pm_rele_client(ct, ct->ct_path_count);
7190 7190 } else {
7191 7191 MDI_DEBUG(4, (MDI_NOTE, client_dip,
7192 7192 "i_mdi_pm_reset_client\n"));
7193 7193 i_mdi_pm_reset_client(ct);
7194 7194 }
7195 7195 }
7196 7196
7197 7197 MDI_CLIENT_UNLOCK(ct);
7198 7198 break;
7199 7199
7200 7200 default:
7201 7201 break;
7202 7202 }
7203 7203
7204 7204 if (devnm)
7205 7205 ndi_devi_exit(vdip, circ);
7206 7206
7207 7207 return (ret);
7208 7208 }
7209 7209
7210 7210 int
7211 7211 mdi_component_is_vhci(dev_info_t *dip, const char **mdi_class)
7212 7212 {
7213 7213 mdi_vhci_t *vhci;
7214 7214
7215 7215 if (!MDI_VHCI(dip))
7216 7216 return (MDI_FAILURE);
7217 7217
7218 7218 if (mdi_class) {
7219 7219 vhci = DEVI(dip)->devi_mdi_xhci;
7220 7220 ASSERT(vhci);
7221 7221 *mdi_class = vhci->vh_class;
7222 7222 }
7223 7223
7224 7224 return (MDI_SUCCESS);
7225 7225 }
7226 7226
7227 7227 int
7228 7228 mdi_component_is_phci(dev_info_t *dip, const char **mdi_class)
7229 7229 {
7230 7230 mdi_phci_t *phci;
7231 7231
7232 7232 if (!MDI_PHCI(dip))
7233 7233 return (MDI_FAILURE);
7234 7234
7235 7235 if (mdi_class) {
7236 7236 phci = DEVI(dip)->devi_mdi_xhci;
7237 7237 ASSERT(phci);
7238 7238 *mdi_class = phci->ph_vhci->vh_class;
7239 7239 }
7240 7240
7241 7241 return (MDI_SUCCESS);
7242 7242 }
7243 7243
7244 7244 int
7245 7245 mdi_component_is_client(dev_info_t *dip, const char **mdi_class)
7246 7246 {
7247 7247 mdi_client_t *client;
7248 7248
7249 7249 if (!MDI_CLIENT(dip))
7250 7250 return (MDI_FAILURE);
7251 7251
7252 7252 if (mdi_class) {
7253 7253 client = DEVI(dip)->devi_mdi_client;
7254 7254 ASSERT(client);
7255 7255 *mdi_class = client->ct_vhci->vh_class;
7256 7256 }
7257 7257
7258 7258 return (MDI_SUCCESS);
7259 7259 }
7260 7260
7261 7261 void *
7262 7262 mdi_client_get_vhci_private(dev_info_t *dip)
7263 7263 {
7264 7264 ASSERT(mdi_component_is_client(dip, NULL) == MDI_SUCCESS);
7265 7265 if (mdi_component_is_client(dip, NULL) == MDI_SUCCESS) {
7266 7266 mdi_client_t *ct;
7267 7267 ct = i_devi_get_client(dip);
7268 7268 return (ct->ct_vprivate);
7269 7269 }
7270 7270 return (NULL);
7271 7271 }
7272 7272
7273 7273 void
7274 7274 mdi_client_set_vhci_private(dev_info_t *dip, void *data)
7275 7275 {
7276 7276 ASSERT(mdi_component_is_client(dip, NULL) == MDI_SUCCESS);
7277 7277 if (mdi_component_is_client(dip, NULL) == MDI_SUCCESS) {
7278 7278 mdi_client_t *ct;
7279 7279 ct = i_devi_get_client(dip);
7280 7280 ct->ct_vprivate = data;
7281 7281 }
7282 7282 }
7283 7283 /*
7284 7284 * mdi_pi_get_vhci_private():
7285 7285 * Get the vhci private information associated with the
7286 7286 * mdi_pathinfo node
7287 7287 */
7288 7288 void *
7289 7289 mdi_pi_get_vhci_private(mdi_pathinfo_t *pip)
7290 7290 {
7291 7291 caddr_t vprivate = NULL;
7292 7292 if (pip) {
7293 7293 vprivate = MDI_PI(pip)->pi_vprivate;
7294 7294 }
7295 7295 return (vprivate);
7296 7296 }
7297 7297
7298 7298 /*
7299 7299 * mdi_pi_set_vhci_private():
7300 7300 * Set the vhci private information in the mdi_pathinfo node
7301 7301 */
7302 7302 void
7303 7303 mdi_pi_set_vhci_private(mdi_pathinfo_t *pip, void *priv)
7304 7304 {
7305 7305 if (pip) {
7306 7306 MDI_PI(pip)->pi_vprivate = priv;
7307 7307 }
7308 7308 }
7309 7309
7310 7310 /*
7311 7311 * mdi_phci_get_vhci_private():
7312 7312 * Get the vhci private information associated with the
7313 7313 * mdi_phci node
7314 7314 */
7315 7315 void *
7316 7316 mdi_phci_get_vhci_private(dev_info_t *dip)
7317 7317 {
7318 7318 ASSERT(mdi_component_is_phci(dip, NULL) == MDI_SUCCESS);
7319 7319 if (mdi_component_is_phci(dip, NULL) == MDI_SUCCESS) {
7320 7320 mdi_phci_t *ph;
7321 7321 ph = i_devi_get_phci(dip);
7322 7322 return (ph->ph_vprivate);
7323 7323 }
7324 7324 return (NULL);
7325 7325 }
7326 7326
7327 7327 /*
7328 7328 * mdi_phci_set_vhci_private():
7329 7329 * Set the vhci private information in the mdi_phci node
7330 7330 */
7331 7331 void
7332 7332 mdi_phci_set_vhci_private(dev_info_t *dip, void *priv)
7333 7333 {
7334 7334 ASSERT(mdi_component_is_phci(dip, NULL) == MDI_SUCCESS);
7335 7335 if (mdi_component_is_phci(dip, NULL) == MDI_SUCCESS) {
7336 7336 mdi_phci_t *ph;
7337 7337 ph = i_devi_get_phci(dip);
7338 7338 ph->ph_vprivate = priv;
7339 7339 }
7340 7340 }
7341 7341
7342 7342 int
7343 7343 mdi_pi_ishidden(mdi_pathinfo_t *pip)
7344 7344 {
7345 7345 return (MDI_PI_FLAGS_IS_HIDDEN(pip));
7346 7346 }
7347 7347
7348 7348 int
7349 7349 mdi_pi_device_isremoved(mdi_pathinfo_t *pip)
7350 7350 {
7351 7351 return (MDI_PI_FLAGS_IS_DEVICE_REMOVED(pip));
7352 7352 }
7353 7353
7354 7354 /* Return 1 if all client paths are device_removed */
7355 7355 static int
7356 7356 i_mdi_client_all_devices_removed(mdi_client_t *ct)
7357 7357 {
7358 7358 mdi_pathinfo_t *pip;
7359 7359 int all_devices_removed = 1;
7360 7360
7361 7361 MDI_CLIENT_LOCK(ct);
7362 7362 for (pip = ct->ct_path_head; pip;
7363 7363 pip = (mdi_pathinfo_t *)MDI_PI(pip)->pi_client_link) {
7364 7364 if (!mdi_pi_device_isremoved(pip)) {
7365 7365 all_devices_removed = 0;
7366 7366 break;
7367 7367 }
7368 7368 }
7369 7369 MDI_CLIENT_UNLOCK(ct);
7370 7370 return (all_devices_removed);
7371 7371 }
7372 7372
7373 7373 /*
7374 7374 * When processing path hotunplug, represent device removal.
7375 7375 */
7376 7376 int
7377 7377 mdi_pi_device_remove(mdi_pathinfo_t *pip)
7378 7378 {
7379 7379 mdi_client_t *ct;
7380 7380
7381 7381 MDI_PI_LOCK(pip);
7382 7382 if (mdi_pi_device_isremoved(pip)) {
7383 7383 MDI_PI_UNLOCK(pip);
7384 7384 return (0);
7385 7385 }
7386 7386 MDI_PI_FLAGS_SET_DEVICE_REMOVED(pip);
7387 7387 MDI_PI_FLAGS_SET_HIDDEN(pip);
7388 7388 MDI_PI_UNLOCK(pip);
7389 7389
7390 7390 /*
7391 7391 * If all paths associated with the client are now DEVICE_REMOVED,
7392 7392 * reflect DEVICE_REMOVED in the client.
7393 7393 */
7394 7394 ct = MDI_PI(pip)->pi_client;
7395 7395 if (ct && ct->ct_dip && i_mdi_client_all_devices_removed(ct))
7396 7396 (void) ndi_devi_device_remove(ct->ct_dip);
7397 7397 else
7398 7398 i_ddi_di_cache_invalidate();
7399 7399
7400 7400 return (1);
7401 7401 }
7402 7402
7403 7403 /*
7404 7404 * When processing hotplug, if a path marked mdi_pi_device_isremoved()
7405 7405 * is now accessible then this interfaces is used to represent device insertion.
7406 7406 */
7407 7407 int
7408 7408 mdi_pi_device_insert(mdi_pathinfo_t *pip)
7409 7409 {
7410 7410 MDI_PI_LOCK(pip);
7411 7411 if (!mdi_pi_device_isremoved(pip)) {
7412 7412 MDI_PI_UNLOCK(pip);
7413 7413 return (0);
7414 7414 }
7415 7415 MDI_PI_FLAGS_CLR_DEVICE_REMOVED(pip);
7416 7416 MDI_PI_FLAGS_CLR_HIDDEN(pip);
7417 7417 MDI_PI_UNLOCK(pip);
7418 7418
7419 7419 i_ddi_di_cache_invalidate();
7420 7420
7421 7421 return (1);
7422 7422 }
7423 7423
7424 7424 /*
7425 7425 * List of vhci class names:
7426 7426 * A vhci class name must be in this list only if the corresponding vhci
7427 7427 * driver intends to use the mdi provided bus config implementation
7428 7428 * (i.e., mdi_vhci_bus_config()).
7429 7429 */
7430 7430 static char *vhci_class_list[] = { MDI_HCI_CLASS_SCSI, MDI_HCI_CLASS_IB };
7431 7431 #define N_VHCI_CLASSES (sizeof (vhci_class_list) / sizeof (char *))
7432 7432
7433 7433 /*
7434 7434 * During boot time, the on-disk vhci cache for every vhci class is read
7435 7435 * in the form of an nvlist and stored here.
7436 7436 */
7437 7437 static nvlist_t *vhcache_nvl[N_VHCI_CLASSES];
7438 7438
7439 7439 /* nvpair names in vhci cache nvlist */
7440 7440 #define MDI_VHCI_CACHE_VERSION 1
7441 7441 #define MDI_NVPNAME_VERSION "version"
7442 7442 #define MDI_NVPNAME_PHCIS "phcis"
7443 7443 #define MDI_NVPNAME_CTADDRMAP "clientaddrmap"
7444 7444
7445 7445 /*
7446 7446 * Given vhci class name, return its on-disk vhci cache filename.
7447 7447 * Memory for the returned filename which includes the full path is allocated
7448 7448 * by this function.
7449 7449 */
7450 7450 static char *
7451 7451 vhclass2vhcache_filename(char *vhclass)
7452 7452 {
7453 7453 char *filename;
7454 7454 int len;
7455 7455 static char *fmt = "/etc/devices/mdi_%s_cache";
7456 7456
7457 7457 /*
7458 7458 * fmt contains the on-disk vhci cache file name format;
7459 7459 * for scsi_vhci the filename is "/etc/devices/mdi_scsi_vhci_cache".
7460 7460 */
7461 7461
7462 7462 /* the -1 below is to account for "%s" in the format string */
7463 7463 len = strlen(fmt) + strlen(vhclass) - 1;
7464 7464 filename = kmem_alloc(len, KM_SLEEP);
7465 7465 (void) snprintf(filename, len, fmt, vhclass);
7466 7466 ASSERT(len == (strlen(filename) + 1));
7467 7467 return (filename);
7468 7468 }
7469 7469
7470 7470 /*
7471 7471 * initialize the vhci cache related data structures and read the on-disk
7472 7472 * vhci cached data into memory.
7473 7473 */
7474 7474 static void
7475 7475 setup_vhci_cache(mdi_vhci_t *vh)
7476 7476 {
7477 7477 mdi_vhci_config_t *vhc;
7478 7478 mdi_vhci_cache_t *vhcache;
7479 7479 int i;
7480 7480 nvlist_t *nvl = NULL;
7481 7481
7482 7482 vhc = kmem_zalloc(sizeof (mdi_vhci_config_t), KM_SLEEP);
7483 7483 vh->vh_config = vhc;
7484 7484 vhcache = &vhc->vhc_vhcache;
7485 7485
7486 7486 vhc->vhc_vhcache_filename = vhclass2vhcache_filename(vh->vh_class);
7487 7487
7488 7488 mutex_init(&vhc->vhc_lock, NULL, MUTEX_DEFAULT, NULL);
7489 7489 cv_init(&vhc->vhc_cv, NULL, CV_DRIVER, NULL);
7490 7490
7491 7491 rw_init(&vhcache->vhcache_lock, NULL, RW_DRIVER, NULL);
7492 7492
7493 7493 /*
7494 7494 * Create string hash; same as mod_hash_create_strhash() except that
7495 7495 * we use NULL key destructor.
7496 7496 */
7497 7497 vhcache->vhcache_client_hash = mod_hash_create_extended(vh->vh_class,
7498 7498 mdi_bus_config_cache_hash_size,
7499 7499 mod_hash_null_keydtor, mod_hash_null_valdtor,
7500 7500 mod_hash_bystr, NULL, mod_hash_strkey_cmp, KM_SLEEP);
7501 7501
7502 7502 /*
7503 7503 * The on-disk vhci cache is read during booting prior to the
7504 7504 * lights-out period by mdi_read_devices_files().
7505 7505 */
7506 7506 for (i = 0; i < N_VHCI_CLASSES; i++) {
7507 7507 if (strcmp(vhci_class_list[i], vh->vh_class) == 0) {
7508 7508 nvl = vhcache_nvl[i];
7509 7509 vhcache_nvl[i] = NULL;
7510 7510 break;
7511 7511 }
7512 7512 }
7513 7513
7514 7514 /*
7515 7515 * this is to cover the case of some one manually causing unloading
7516 7516 * (or detaching) and reloading (or attaching) of a vhci driver.
7517 7517 */
7518 7518 if (nvl == NULL && modrootloaded)
7519 7519 nvl = read_on_disk_vhci_cache(vh->vh_class);
7520 7520
7521 7521 if (nvl != NULL) {
7522 7522 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
7523 7523 if (mainnvl_to_vhcache(vhcache, nvl) == MDI_SUCCESS)
7524 7524 vhcache->vhcache_flags |= MDI_VHCI_CACHE_SETUP_DONE;
7525 7525 else {
7526 7526 cmn_err(CE_WARN,
7527 7527 "%s: data file corrupted, will recreate",
7528 7528 vhc->vhc_vhcache_filename);
7529 7529 }
7530 7530 rw_exit(&vhcache->vhcache_lock);
7531 7531 nvlist_free(nvl);
7532 7532 }
7533 7533
7534 7534 vhc->vhc_cbid = callb_add(stop_vhcache_flush_thread, vhc,
7535 7535 CB_CL_UADMIN_PRE_VFS, "mdi_vhcache_flush");
7536 7536
7537 7537 vhc->vhc_path_discovery_boot = mdi_path_discovery_boot;
7538 7538 vhc->vhc_path_discovery_postboot = mdi_path_discovery_postboot;
7539 7539 }
7540 7540
7541 7541 /*
7542 7542 * free all vhci cache related resources
7543 7543 */
7544 7544 static int
7545 7545 destroy_vhci_cache(mdi_vhci_t *vh)
7546 7546 {
7547 7547 mdi_vhci_config_t *vhc = vh->vh_config;
7548 7548 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
7549 7549 mdi_vhcache_phci_t *cphci, *cphci_next;
7550 7550 mdi_vhcache_client_t *cct, *cct_next;
7551 7551 mdi_vhcache_pathinfo_t *cpi, *cpi_next;
7552 7552
7553 7553 if (stop_vhcache_async_threads(vhc) != MDI_SUCCESS)
7554 7554 return (MDI_FAILURE);
7555 7555
7556 7556 kmem_free(vhc->vhc_vhcache_filename,
7557 7557 strlen(vhc->vhc_vhcache_filename) + 1);
7558 7558
7559 7559 mod_hash_destroy_strhash(vhcache->vhcache_client_hash);
7560 7560
7561 7561 for (cphci = vhcache->vhcache_phci_head; cphci != NULL;
7562 7562 cphci = cphci_next) {
7563 7563 cphci_next = cphci->cphci_next;
7564 7564 free_vhcache_phci(cphci);
7565 7565 }
7566 7566
7567 7567 for (cct = vhcache->vhcache_client_head; cct != NULL; cct = cct_next) {
7568 7568 cct_next = cct->cct_next;
7569 7569 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi_next) {
7570 7570 cpi_next = cpi->cpi_next;
7571 7571 free_vhcache_pathinfo(cpi);
7572 7572 }
7573 7573 free_vhcache_client(cct);
7574 7574 }
7575 7575
7576 7576 rw_destroy(&vhcache->vhcache_lock);
7577 7577
7578 7578 mutex_destroy(&vhc->vhc_lock);
7579 7579 cv_destroy(&vhc->vhc_cv);
7580 7580 kmem_free(vhc, sizeof (mdi_vhci_config_t));
7581 7581 return (MDI_SUCCESS);
7582 7582 }
7583 7583
7584 7584 /*
7585 7585 * Stop all vhci cache related async threads and free their resources.
7586 7586 */
7587 7587 static int
7588 7588 stop_vhcache_async_threads(mdi_vhci_config_t *vhc)
7589 7589 {
7590 7590 mdi_async_client_config_t *acc, *acc_next;
7591 7591
7592 7592 mutex_enter(&vhc->vhc_lock);
7593 7593 vhc->vhc_flags |= MDI_VHC_EXIT;
7594 7594 ASSERT(vhc->vhc_acc_thrcount >= 0);
7595 7595 cv_broadcast(&vhc->vhc_cv);
7596 7596
7597 7597 while ((vhc->vhc_flags & MDI_VHC_VHCACHE_FLUSH_THREAD) ||
7598 7598 vhc->vhc_acc_thrcount != 0) {
7599 7599 mutex_exit(&vhc->vhc_lock);
7600 7600 delay_random(mdi_delay);
7601 7601 mutex_enter(&vhc->vhc_lock);
7602 7602 }
7603 7603
7604 7604 vhc->vhc_flags &= ~MDI_VHC_EXIT;
7605 7605
7606 7606 for (acc = vhc->vhc_acc_list_head; acc != NULL; acc = acc_next) {
7607 7607 acc_next = acc->acc_next;
7608 7608 free_async_client_config(acc);
7609 7609 }
7610 7610 vhc->vhc_acc_list_head = NULL;
7611 7611 vhc->vhc_acc_list_tail = NULL;
7612 7612 vhc->vhc_acc_count = 0;
7613 7613
7614 7614 if (vhc->vhc_flags & MDI_VHC_VHCACHE_DIRTY) {
7615 7615 vhc->vhc_flags &= ~MDI_VHC_VHCACHE_DIRTY;
7616 7616 mutex_exit(&vhc->vhc_lock);
7617 7617 if (flush_vhcache(vhc, 0) != MDI_SUCCESS) {
7618 7618 vhcache_dirty(vhc);
7619 7619 return (MDI_FAILURE);
7620 7620 }
7621 7621 } else
7622 7622 mutex_exit(&vhc->vhc_lock);
7623 7623
7624 7624 if (callb_delete(vhc->vhc_cbid) != 0)
7625 7625 return (MDI_FAILURE);
7626 7626
7627 7627 return (MDI_SUCCESS);
7628 7628 }
7629 7629
7630 7630 /*
7631 7631 * Stop vhci cache flush thread
7632 7632 */
7633 7633 /* ARGSUSED */
7634 7634 static boolean_t
7635 7635 stop_vhcache_flush_thread(void *arg, int code)
7636 7636 {
7637 7637 mdi_vhci_config_t *vhc = (mdi_vhci_config_t *)arg;
7638 7638
7639 7639 mutex_enter(&vhc->vhc_lock);
7640 7640 vhc->vhc_flags |= MDI_VHC_EXIT;
7641 7641 cv_broadcast(&vhc->vhc_cv);
7642 7642
7643 7643 while (vhc->vhc_flags & MDI_VHC_VHCACHE_FLUSH_THREAD) {
7644 7644 mutex_exit(&vhc->vhc_lock);
7645 7645 delay_random(mdi_delay);
7646 7646 mutex_enter(&vhc->vhc_lock);
7647 7647 }
7648 7648
7649 7649 if (vhc->vhc_flags & MDI_VHC_VHCACHE_DIRTY) {
7650 7650 vhc->vhc_flags &= ~MDI_VHC_VHCACHE_DIRTY;
7651 7651 mutex_exit(&vhc->vhc_lock);
7652 7652 (void) flush_vhcache(vhc, 1);
7653 7653 } else
7654 7654 mutex_exit(&vhc->vhc_lock);
7655 7655
7656 7656 return (B_TRUE);
7657 7657 }
7658 7658
7659 7659 /*
7660 7660 * Enqueue the vhcache phci (cphci) at the tail of the list
7661 7661 */
7662 7662 static void
7663 7663 enqueue_vhcache_phci(mdi_vhci_cache_t *vhcache, mdi_vhcache_phci_t *cphci)
7664 7664 {
7665 7665 cphci->cphci_next = NULL;
7666 7666 if (vhcache->vhcache_phci_head == NULL)
7667 7667 vhcache->vhcache_phci_head = cphci;
7668 7668 else
7669 7669 vhcache->vhcache_phci_tail->cphci_next = cphci;
7670 7670 vhcache->vhcache_phci_tail = cphci;
7671 7671 }
7672 7672
7673 7673 /*
7674 7674 * Enqueue the vhcache pathinfo (cpi) at the tail of the list
7675 7675 */
7676 7676 static void
7677 7677 enqueue_tail_vhcache_pathinfo(mdi_vhcache_client_t *cct,
7678 7678 mdi_vhcache_pathinfo_t *cpi)
7679 7679 {
7680 7680 cpi->cpi_next = NULL;
7681 7681 if (cct->cct_cpi_head == NULL)
7682 7682 cct->cct_cpi_head = cpi;
7683 7683 else
7684 7684 cct->cct_cpi_tail->cpi_next = cpi;
7685 7685 cct->cct_cpi_tail = cpi;
7686 7686 }
7687 7687
7688 7688 /*
7689 7689 * Enqueue the vhcache pathinfo (cpi) at the correct location in the
7690 7690 * ordered list. All cpis which do not have MDI_CPI_HINT_PATH_DOES_NOT_EXIST
7691 7691 * flag set come at the beginning of the list. All cpis which have this
7692 7692 * flag set come at the end of the list.
7693 7693 */
7694 7694 static void
7695 7695 enqueue_vhcache_pathinfo(mdi_vhcache_client_t *cct,
7696 7696 mdi_vhcache_pathinfo_t *newcpi)
7697 7697 {
7698 7698 mdi_vhcache_pathinfo_t *cpi, *prev_cpi;
7699 7699
7700 7700 if (cct->cct_cpi_head == NULL ||
7701 7701 (newcpi->cpi_flags & MDI_CPI_HINT_PATH_DOES_NOT_EXIST))
7702 7702 enqueue_tail_vhcache_pathinfo(cct, newcpi);
7703 7703 else {
7704 7704 for (cpi = cct->cct_cpi_head, prev_cpi = NULL; cpi != NULL &&
7705 7705 !(cpi->cpi_flags & MDI_CPI_HINT_PATH_DOES_NOT_EXIST);
7706 7706 prev_cpi = cpi, cpi = cpi->cpi_next)
7707 7707 ;
7708 7708
7709 7709 if (prev_cpi == NULL)
7710 7710 cct->cct_cpi_head = newcpi;
7711 7711 else
7712 7712 prev_cpi->cpi_next = newcpi;
7713 7713
7714 7714 newcpi->cpi_next = cpi;
7715 7715
7716 7716 if (cpi == NULL)
7717 7717 cct->cct_cpi_tail = newcpi;
7718 7718 }
7719 7719 }
7720 7720
7721 7721 /*
7722 7722 * Enqueue the vhcache client (cct) at the tail of the list
7723 7723 */
7724 7724 static void
7725 7725 enqueue_vhcache_client(mdi_vhci_cache_t *vhcache,
7726 7726 mdi_vhcache_client_t *cct)
7727 7727 {
7728 7728 cct->cct_next = NULL;
7729 7729 if (vhcache->vhcache_client_head == NULL)
7730 7730 vhcache->vhcache_client_head = cct;
7731 7731 else
7732 7732 vhcache->vhcache_client_tail->cct_next = cct;
7733 7733 vhcache->vhcache_client_tail = cct;
7734 7734 }
7735 7735
7736 7736 static void
7737 7737 free_string_array(char **str, int nelem)
7738 7738 {
7739 7739 int i;
7740 7740
7741 7741 if (str) {
7742 7742 for (i = 0; i < nelem; i++) {
7743 7743 if (str[i])
7744 7744 kmem_free(str[i], strlen(str[i]) + 1);
7745 7745 }
7746 7746 kmem_free(str, sizeof (char *) * nelem);
7747 7747 }
7748 7748 }
7749 7749
7750 7750 static void
7751 7751 free_vhcache_phci(mdi_vhcache_phci_t *cphci)
7752 7752 {
7753 7753 kmem_free(cphci->cphci_path, strlen(cphci->cphci_path) + 1);
7754 7754 kmem_free(cphci, sizeof (*cphci));
7755 7755 }
7756 7756
7757 7757 static void
7758 7758 free_vhcache_pathinfo(mdi_vhcache_pathinfo_t *cpi)
7759 7759 {
7760 7760 kmem_free(cpi->cpi_addr, strlen(cpi->cpi_addr) + 1);
7761 7761 kmem_free(cpi, sizeof (*cpi));
7762 7762 }
7763 7763
7764 7764 static void
7765 7765 free_vhcache_client(mdi_vhcache_client_t *cct)
7766 7766 {
7767 7767 kmem_free(cct->cct_name_addr, strlen(cct->cct_name_addr) + 1);
7768 7768 kmem_free(cct, sizeof (*cct));
7769 7769 }
7770 7770
7771 7771 static char *
7772 7772 vhcache_mknameaddr(char *ct_name, char *ct_addr, int *ret_len)
7773 7773 {
7774 7774 char *name_addr;
7775 7775 int len;
7776 7776
7777 7777 len = strlen(ct_name) + strlen(ct_addr) + 2;
7778 7778 name_addr = kmem_alloc(len, KM_SLEEP);
7779 7779 (void) snprintf(name_addr, len, "%s@%s", ct_name, ct_addr);
7780 7780
7781 7781 if (ret_len)
7782 7782 *ret_len = len;
7783 7783 return (name_addr);
7784 7784 }
7785 7785
7786 7786 /*
7787 7787 * Copy the contents of paddrnvl to vhci cache.
7788 7788 * paddrnvl nvlist contains path information for a vhci client.
7789 7789 * See the comment in mainnvl_to_vhcache() for the format of this nvlist.
7790 7790 */
7791 7791 static void
7792 7792 paddrnvl_to_vhcache(nvlist_t *nvl, mdi_vhcache_phci_t *cphci_list[],
7793 7793 mdi_vhcache_client_t *cct)
7794 7794 {
7795 7795 nvpair_t *nvp = NULL;
7796 7796 mdi_vhcache_pathinfo_t *cpi;
7797 7797 uint_t nelem;
7798 7798 uint32_t *val;
7799 7799
7800 7800 while ((nvp = nvlist_next_nvpair(nvl, nvp)) != NULL) {
7801 7801 ASSERT(nvpair_type(nvp) == DATA_TYPE_UINT32_ARRAY);
7802 7802 cpi = kmem_zalloc(sizeof (*cpi), KM_SLEEP);
7803 7803 cpi->cpi_addr = i_ddi_strdup(nvpair_name(nvp), KM_SLEEP);
7804 7804 (void) nvpair_value_uint32_array(nvp, &val, &nelem);
7805 7805 ASSERT(nelem == 2);
7806 7806 cpi->cpi_cphci = cphci_list[val[0]];
7807 7807 cpi->cpi_flags = val[1];
7808 7808 enqueue_tail_vhcache_pathinfo(cct, cpi);
7809 7809 }
7810 7810 }
7811 7811
7812 7812 /*
7813 7813 * Copy the contents of caddrmapnvl to vhci cache.
7814 7814 * caddrmapnvl nvlist contains vhci client address to phci client address
7815 7815 * mappings. See the comment in mainnvl_to_vhcache() for the format of
7816 7816 * this nvlist.
7817 7817 */
7818 7818 static void
7819 7819 caddrmapnvl_to_vhcache(mdi_vhci_cache_t *vhcache, nvlist_t *nvl,
7820 7820 mdi_vhcache_phci_t *cphci_list[])
7821 7821 {
7822 7822 nvpair_t *nvp = NULL;
7823 7823 nvlist_t *paddrnvl;
7824 7824 mdi_vhcache_client_t *cct;
7825 7825
7826 7826 while ((nvp = nvlist_next_nvpair(nvl, nvp)) != NULL) {
7827 7827 ASSERT(nvpair_type(nvp) == DATA_TYPE_NVLIST);
7828 7828 cct = kmem_zalloc(sizeof (*cct), KM_SLEEP);
7829 7829 cct->cct_name_addr = i_ddi_strdup(nvpair_name(nvp), KM_SLEEP);
7830 7830 (void) nvpair_value_nvlist(nvp, &paddrnvl);
7831 7831 paddrnvl_to_vhcache(paddrnvl, cphci_list, cct);
7832 7832 /* the client must contain at least one path */
7833 7833 ASSERT(cct->cct_cpi_head != NULL);
7834 7834
7835 7835 enqueue_vhcache_client(vhcache, cct);
7836 7836 (void) mod_hash_insert(vhcache->vhcache_client_hash,
7837 7837 (mod_hash_key_t)cct->cct_name_addr, (mod_hash_val_t)cct);
7838 7838 }
7839 7839 }
7840 7840
7841 7841 /*
7842 7842 * Copy the contents of the main nvlist to vhci cache.
7843 7843 *
7844 7844 * VHCI busconfig cached data is stored in the form of a nvlist on the disk.
7845 7845 * The nvlist contains the mappings between the vhci client addresses and
7846 7846 * their corresponding phci client addresses.
7847 7847 *
7848 7848 * The structure of the nvlist is as follows:
7849 7849 *
7850 7850 * Main nvlist:
7851 7851 * NAME TYPE DATA
7852 7852 * version int32 version number
7853 7853 * phcis string array array of phci paths
7854 7854 * clientaddrmap nvlist_t c2paddrs_nvl (see below)
7855 7855 *
7856 7856 * structure of c2paddrs_nvl:
7857 7857 * NAME TYPE DATA
7858 7858 * caddr1 nvlist_t paddrs_nvl1
7859 7859 * caddr2 nvlist_t paddrs_nvl2
7860 7860 * ...
7861 7861 * where caddr1, caddr2, ... are vhci client name and addresses in the
7862 7862 * form of "<clientname>@<clientaddress>".
7863 7863 * (for example: "ssd@2000002037cd9f72");
7864 7864 * paddrs_nvl1, paddrs_nvl2, .. are nvlists that contain path information.
7865 7865 *
7866 7866 * structure of paddrs_nvl:
7867 7867 * NAME TYPE DATA
7868 7868 * pi_addr1 uint32_array (phci-id, cpi_flags)
7869 7869 * pi_addr2 uint32_array (phci-id, cpi_flags)
7870 7870 * ...
7871 7871 * where pi_addr1, pi_addr2, ... are bus specific addresses of pathinfo nodes
7872 7872 * (so called pi_addrs, for example: "w2100002037cd9f72,0");
7873 7873 * phci-ids are integers that identify pHCIs to which the
7874 7874 * the bus specific address belongs to. These integers are used as an index
7875 7875 * into to the phcis string array in the main nvlist to get the pHCI path.
7876 7876 */
7877 7877 static int
7878 7878 mainnvl_to_vhcache(mdi_vhci_cache_t *vhcache, nvlist_t *nvl)
7879 7879 {
7880 7880 char **phcis, **phci_namep;
7881 7881 uint_t nphcis;
7882 7882 mdi_vhcache_phci_t *cphci, **cphci_list;
7883 7883 nvlist_t *caddrmapnvl;
7884 7884 int32_t ver;
7885 7885 int i;
7886 7886 size_t cphci_list_size;
7887 7887
7888 7888 ASSERT(RW_WRITE_HELD(&vhcache->vhcache_lock));
7889 7889
7890 7890 if (nvlist_lookup_int32(nvl, MDI_NVPNAME_VERSION, &ver) != 0 ||
7891 7891 ver != MDI_VHCI_CACHE_VERSION)
7892 7892 return (MDI_FAILURE);
7893 7893
7894 7894 if (nvlist_lookup_string_array(nvl, MDI_NVPNAME_PHCIS, &phcis,
7895 7895 &nphcis) != 0)
7896 7896 return (MDI_SUCCESS);
7897 7897
7898 7898 ASSERT(nphcis > 0);
7899 7899
7900 7900 cphci_list_size = sizeof (mdi_vhcache_phci_t *) * nphcis;
7901 7901 cphci_list = kmem_alloc(cphci_list_size, KM_SLEEP);
7902 7902 for (i = 0, phci_namep = phcis; i < nphcis; i++, phci_namep++) {
7903 7903 cphci = kmem_zalloc(sizeof (mdi_vhcache_phci_t), KM_SLEEP);
7904 7904 cphci->cphci_path = i_ddi_strdup(*phci_namep, KM_SLEEP);
7905 7905 enqueue_vhcache_phci(vhcache, cphci);
7906 7906 cphci_list[i] = cphci;
7907 7907 }
7908 7908
7909 7909 ASSERT(vhcache->vhcache_phci_head != NULL);
7910 7910
7911 7911 if (nvlist_lookup_nvlist(nvl, MDI_NVPNAME_CTADDRMAP, &caddrmapnvl) == 0)
7912 7912 caddrmapnvl_to_vhcache(vhcache, caddrmapnvl, cphci_list);
7913 7913
7914 7914 kmem_free(cphci_list, cphci_list_size);
7915 7915 return (MDI_SUCCESS);
7916 7916 }
7917 7917
7918 7918 /*
7919 7919 * Build paddrnvl for the specified client using the information in the
7920 7920 * vhci cache and add it to the caddrmapnnvl.
7921 7921 * Returns 0 on success, errno on failure.
7922 7922 */
7923 7923 static int
7924 7924 vhcache_to_paddrnvl(mdi_vhci_cache_t *vhcache, mdi_vhcache_client_t *cct,
7925 7925 nvlist_t *caddrmapnvl)
7926 7926 {
7927 7927 mdi_vhcache_pathinfo_t *cpi;
7928 7928 nvlist_t *nvl;
7929 7929 int err;
7930 7930 uint32_t val[2];
7931 7931
7932 7932 ASSERT(RW_LOCK_HELD(&vhcache->vhcache_lock));
7933 7933
7934 7934 if ((err = nvlist_alloc(&nvl, 0, KM_SLEEP)) != 0)
7935 7935 return (err);
7936 7936
7937 7937 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi->cpi_next) {
7938 7938 val[0] = cpi->cpi_cphci->cphci_id;
7939 7939 val[1] = cpi->cpi_flags;
7940 7940 if ((err = nvlist_add_uint32_array(nvl, cpi->cpi_addr, val, 2))
7941 7941 != 0)
7942 7942 goto out;
7943 7943 }
7944 7944
7945 7945 err = nvlist_add_nvlist(caddrmapnvl, cct->cct_name_addr, nvl);
7946 7946 out:
7947 7947 nvlist_free(nvl);
7948 7948 return (err);
7949 7949 }
7950 7950
7951 7951 /*
7952 7952 * Build caddrmapnvl using the information in the vhci cache
7953 7953 * and add it to the mainnvl.
7954 7954 * Returns 0 on success, errno on failure.
7955 7955 */
7956 7956 static int
7957 7957 vhcache_to_caddrmapnvl(mdi_vhci_cache_t *vhcache, nvlist_t *mainnvl)
7958 7958 {
7959 7959 mdi_vhcache_client_t *cct;
7960 7960 nvlist_t *nvl;
7961 7961 int err;
7962 7962
7963 7963 ASSERT(RW_LOCK_HELD(&vhcache->vhcache_lock));
7964 7964
7965 7965 if ((err = nvlist_alloc(&nvl, NV_UNIQUE_NAME, KM_SLEEP)) != 0)
7966 7966 return (err);
7967 7967
7968 7968 for (cct = vhcache->vhcache_client_head; cct != NULL;
7969 7969 cct = cct->cct_next) {
7970 7970 if ((err = vhcache_to_paddrnvl(vhcache, cct, nvl)) != 0)
7971 7971 goto out;
7972 7972 }
7973 7973
7974 7974 err = nvlist_add_nvlist(mainnvl, MDI_NVPNAME_CTADDRMAP, nvl);
7975 7975 out:
7976 7976 nvlist_free(nvl);
7977 7977 return (err);
7978 7978 }
7979 7979
7980 7980 /*
7981 7981 * Build nvlist using the information in the vhci cache.
7982 7982 * See the comment in mainnvl_to_vhcache() for the format of the nvlist.
7983 7983 * Returns nvl on success, NULL on failure.
7984 7984 */
7985 7985 static nvlist_t *
7986 7986 vhcache_to_mainnvl(mdi_vhci_cache_t *vhcache)
7987 7987 {
7988 7988 mdi_vhcache_phci_t *cphci;
7989 7989 uint_t phci_count;
7990 7990 char **phcis;
7991 7991 nvlist_t *nvl;
7992 7992 int err, i;
7993 7993
7994 7994 if ((err = nvlist_alloc(&nvl, NV_UNIQUE_NAME, KM_SLEEP)) != 0) {
7995 7995 nvl = NULL;
7996 7996 goto out;
7997 7997 }
7998 7998
7999 7999 if ((err = nvlist_add_int32(nvl, MDI_NVPNAME_VERSION,
8000 8000 MDI_VHCI_CACHE_VERSION)) != 0)
8001 8001 goto out;
8002 8002
8003 8003 rw_enter(&vhcache->vhcache_lock, RW_READER);
8004 8004 if (vhcache->vhcache_phci_head == NULL) {
8005 8005 rw_exit(&vhcache->vhcache_lock);
8006 8006 return (nvl);
8007 8007 }
8008 8008
8009 8009 phci_count = 0;
8010 8010 for (cphci = vhcache->vhcache_phci_head; cphci != NULL;
8011 8011 cphci = cphci->cphci_next)
8012 8012 cphci->cphci_id = phci_count++;
8013 8013
8014 8014 /* build phci pathname list */
8015 8015 phcis = kmem_alloc(sizeof (char *) * phci_count, KM_SLEEP);
8016 8016 for (cphci = vhcache->vhcache_phci_head, i = 0; cphci != NULL;
8017 8017 cphci = cphci->cphci_next, i++)
8018 8018 phcis[i] = i_ddi_strdup(cphci->cphci_path, KM_SLEEP);
8019 8019
8020 8020 err = nvlist_add_string_array(nvl, MDI_NVPNAME_PHCIS, phcis,
8021 8021 phci_count);
8022 8022 free_string_array(phcis, phci_count);
8023 8023
8024 8024 if (err == 0 &&
8025 8025 (err = vhcache_to_caddrmapnvl(vhcache, nvl)) == 0) {
8026 8026 rw_exit(&vhcache->vhcache_lock);
8027 8027 return (nvl);
8028 8028 }
8029 8029
8030 8030 rw_exit(&vhcache->vhcache_lock);
8031 8031 out:
8032 8032 nvlist_free(nvl);
8033 8033 return (NULL);
8034 8034 }
8035 8035
8036 8036 /*
8037 8037 * Lookup vhcache phci structure for the specified phci path.
8038 8038 */
8039 8039 static mdi_vhcache_phci_t *
8040 8040 lookup_vhcache_phci_by_name(mdi_vhci_cache_t *vhcache, char *phci_path)
8041 8041 {
8042 8042 mdi_vhcache_phci_t *cphci;
8043 8043
8044 8044 ASSERT(RW_LOCK_HELD(&vhcache->vhcache_lock));
8045 8045
8046 8046 for (cphci = vhcache->vhcache_phci_head; cphci != NULL;
8047 8047 cphci = cphci->cphci_next) {
8048 8048 if (strcmp(cphci->cphci_path, phci_path) == 0)
8049 8049 return (cphci);
8050 8050 }
8051 8051
8052 8052 return (NULL);
8053 8053 }
8054 8054
8055 8055 /*
8056 8056 * Lookup vhcache phci structure for the specified phci.
8057 8057 */
8058 8058 static mdi_vhcache_phci_t *
8059 8059 lookup_vhcache_phci_by_addr(mdi_vhci_cache_t *vhcache, mdi_phci_t *ph)
8060 8060 {
8061 8061 mdi_vhcache_phci_t *cphci;
8062 8062
8063 8063 ASSERT(RW_LOCK_HELD(&vhcache->vhcache_lock));
8064 8064
8065 8065 for (cphci = vhcache->vhcache_phci_head; cphci != NULL;
8066 8066 cphci = cphci->cphci_next) {
8067 8067 if (cphci->cphci_phci == ph)
8068 8068 return (cphci);
8069 8069 }
8070 8070
8071 8071 return (NULL);
8072 8072 }
8073 8073
8074 8074 /*
8075 8075 * Add the specified phci to the vhci cache if not already present.
8076 8076 */
8077 8077 static void
8078 8078 vhcache_phci_add(mdi_vhci_config_t *vhc, mdi_phci_t *ph)
8079 8079 {
8080 8080 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8081 8081 mdi_vhcache_phci_t *cphci;
8082 8082 char *pathname;
8083 8083 int cache_updated;
8084 8084
8085 8085 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
8086 8086
8087 8087 pathname = kmem_alloc(MAXPATHLEN, KM_SLEEP);
8088 8088 (void) ddi_pathname(ph->ph_dip, pathname);
8089 8089 if ((cphci = lookup_vhcache_phci_by_name(vhcache, pathname))
8090 8090 != NULL) {
8091 8091 cphci->cphci_phci = ph;
8092 8092 cache_updated = 0;
8093 8093 } else {
8094 8094 cphci = kmem_zalloc(sizeof (*cphci), KM_SLEEP);
8095 8095 cphci->cphci_path = i_ddi_strdup(pathname, KM_SLEEP);
8096 8096 cphci->cphci_phci = ph;
8097 8097 enqueue_vhcache_phci(vhcache, cphci);
8098 8098 cache_updated = 1;
8099 8099 }
8100 8100
8101 8101 rw_exit(&vhcache->vhcache_lock);
8102 8102
8103 8103 /*
8104 8104 * Since a new phci has been added, reset
8105 8105 * vhc_path_discovery_cutoff_time to allow for discovery of paths
8106 8106 * during next vhcache_discover_paths().
8107 8107 */
8108 8108 mutex_enter(&vhc->vhc_lock);
8109 8109 vhc->vhc_path_discovery_cutoff_time = 0;
8110 8110 mutex_exit(&vhc->vhc_lock);
8111 8111
8112 8112 kmem_free(pathname, MAXPATHLEN);
8113 8113 if (cache_updated)
8114 8114 vhcache_dirty(vhc);
8115 8115 }
8116 8116
8117 8117 /*
8118 8118 * Remove the reference to the specified phci from the vhci cache.
8119 8119 */
8120 8120 static void
8121 8121 vhcache_phci_remove(mdi_vhci_config_t *vhc, mdi_phci_t *ph)
8122 8122 {
8123 8123 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8124 8124 mdi_vhcache_phci_t *cphci;
8125 8125
8126 8126 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
8127 8127 if ((cphci = lookup_vhcache_phci_by_addr(vhcache, ph)) != NULL) {
8128 8128 /* do not remove the actual mdi_vhcache_phci structure */
8129 8129 cphci->cphci_phci = NULL;
8130 8130 }
8131 8131 rw_exit(&vhcache->vhcache_lock);
8132 8132 }
8133 8133
8134 8134 static void
8135 8135 init_vhcache_lookup_token(mdi_vhcache_lookup_token_t *dst,
8136 8136 mdi_vhcache_lookup_token_t *src)
8137 8137 {
8138 8138 if (src == NULL) {
8139 8139 dst->lt_cct = NULL;
8140 8140 dst->lt_cct_lookup_time = 0;
8141 8141 } else {
8142 8142 dst->lt_cct = src->lt_cct;
8143 8143 dst->lt_cct_lookup_time = src->lt_cct_lookup_time;
8144 8144 }
8145 8145 }
8146 8146
8147 8147 /*
8148 8148 * Look up vhcache client for the specified client.
8149 8149 */
8150 8150 static mdi_vhcache_client_t *
8151 8151 lookup_vhcache_client(mdi_vhci_cache_t *vhcache, char *ct_name, char *ct_addr,
8152 8152 mdi_vhcache_lookup_token_t *token)
8153 8153 {
8154 8154 mod_hash_val_t hv;
8155 8155 char *name_addr;
8156 8156 int len;
8157 8157
8158 8158 ASSERT(RW_LOCK_HELD(&vhcache->vhcache_lock));
8159 8159
8160 8160 /*
8161 8161 * If no vhcache clean occurred since the last lookup, we can
8162 8162 * simply return the cct from the last lookup operation.
8163 8163 * It works because ccts are never freed except during the vhcache
8164 8164 * cleanup operation.
8165 8165 */
8166 8166 if (token != NULL &&
8167 8167 vhcache->vhcache_clean_time < token->lt_cct_lookup_time)
8168 8168 return (token->lt_cct);
8169 8169
8170 8170 name_addr = vhcache_mknameaddr(ct_name, ct_addr, &len);
8171 8171 if (mod_hash_find(vhcache->vhcache_client_hash,
8172 8172 (mod_hash_key_t)name_addr, &hv) == 0) {
8173 8173 if (token) {
8174 8174 token->lt_cct = (mdi_vhcache_client_t *)hv;
8175 8175 token->lt_cct_lookup_time = ddi_get_lbolt64();
8176 8176 }
8177 8177 } else {
8178 8178 if (token) {
8179 8179 token->lt_cct = NULL;
8180 8180 token->lt_cct_lookup_time = 0;
8181 8181 }
8182 8182 hv = NULL;
8183 8183 }
8184 8184 kmem_free(name_addr, len);
8185 8185 return ((mdi_vhcache_client_t *)hv);
8186 8186 }
8187 8187
8188 8188 /*
8189 8189 * Add the specified path to the vhci cache if not already present.
8190 8190 * Also add the vhcache client for the client corresponding to this path
8191 8191 * if it doesn't already exist.
8192 8192 */
8193 8193 static void
8194 8194 vhcache_pi_add(mdi_vhci_config_t *vhc, struct mdi_pathinfo *pip)
8195 8195 {
8196 8196 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8197 8197 mdi_vhcache_client_t *cct;
8198 8198 mdi_vhcache_pathinfo_t *cpi;
8199 8199 mdi_phci_t *ph = pip->pi_phci;
8200 8200 mdi_client_t *ct = pip->pi_client;
8201 8201 int cache_updated = 0;
8202 8202
8203 8203 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
8204 8204
8205 8205 /* if vhcache client for this pip doesn't already exist, add it */
8206 8206 if ((cct = lookup_vhcache_client(vhcache, ct->ct_drvname, ct->ct_guid,
8207 8207 NULL)) == NULL) {
8208 8208 cct = kmem_zalloc(sizeof (*cct), KM_SLEEP);
8209 8209 cct->cct_name_addr = vhcache_mknameaddr(ct->ct_drvname,
8210 8210 ct->ct_guid, NULL);
8211 8211 enqueue_vhcache_client(vhcache, cct);
8212 8212 (void) mod_hash_insert(vhcache->vhcache_client_hash,
8213 8213 (mod_hash_key_t)cct->cct_name_addr, (mod_hash_val_t)cct);
8214 8214 cache_updated = 1;
8215 8215 }
8216 8216
8217 8217 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi->cpi_next) {
8218 8218 if (cpi->cpi_cphci->cphci_phci == ph &&
8219 8219 strcmp(cpi->cpi_addr, pip->pi_addr) == 0) {
8220 8220 cpi->cpi_pip = pip;
8221 8221 if (cpi->cpi_flags & MDI_CPI_HINT_PATH_DOES_NOT_EXIST) {
8222 8222 cpi->cpi_flags &=
8223 8223 ~MDI_CPI_HINT_PATH_DOES_NOT_EXIST;
8224 8224 sort_vhcache_paths(cct);
8225 8225 cache_updated = 1;
8226 8226 }
8227 8227 break;
8228 8228 }
8229 8229 }
8230 8230
8231 8231 if (cpi == NULL) {
8232 8232 cpi = kmem_zalloc(sizeof (*cpi), KM_SLEEP);
8233 8233 cpi->cpi_addr = i_ddi_strdup(pip->pi_addr, KM_SLEEP);
8234 8234 cpi->cpi_cphci = lookup_vhcache_phci_by_addr(vhcache, ph);
8235 8235 ASSERT(cpi->cpi_cphci != NULL);
8236 8236 cpi->cpi_pip = pip;
8237 8237 enqueue_vhcache_pathinfo(cct, cpi);
8238 8238 cache_updated = 1;
8239 8239 }
8240 8240
8241 8241 rw_exit(&vhcache->vhcache_lock);
8242 8242
8243 8243 if (cache_updated)
8244 8244 vhcache_dirty(vhc);
8245 8245 }
8246 8246
8247 8247 /*
8248 8248 * Remove the reference to the specified path from the vhci cache.
8249 8249 */
8250 8250 static void
8251 8251 vhcache_pi_remove(mdi_vhci_config_t *vhc, struct mdi_pathinfo *pip)
8252 8252 {
8253 8253 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8254 8254 mdi_client_t *ct = pip->pi_client;
8255 8255 mdi_vhcache_client_t *cct;
8256 8256 mdi_vhcache_pathinfo_t *cpi;
8257 8257
8258 8258 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
8259 8259 if ((cct = lookup_vhcache_client(vhcache, ct->ct_drvname, ct->ct_guid,
8260 8260 NULL)) != NULL) {
8261 8261 for (cpi = cct->cct_cpi_head; cpi != NULL;
8262 8262 cpi = cpi->cpi_next) {
8263 8263 if (cpi->cpi_pip == pip) {
8264 8264 cpi->cpi_pip = NULL;
8265 8265 break;
8266 8266 }
8267 8267 }
8268 8268 }
8269 8269 rw_exit(&vhcache->vhcache_lock);
8270 8270 }
8271 8271
8272 8272 /*
8273 8273 * Flush the vhci cache to disk.
8274 8274 * Returns MDI_SUCCESS on success, MDI_FAILURE on failure.
8275 8275 */
8276 8276 static int
8277 8277 flush_vhcache(mdi_vhci_config_t *vhc, int force_flag)
8278 8278 {
8279 8279 nvlist_t *nvl;
8280 8280 int err;
8281 8281 int rv;
8282 8282
8283 8283 /*
8284 8284 * It is possible that the system may shutdown before
8285 8285 * i_ddi_io_initialized (during stmsboot for example). To allow for
8286 8286 * flushing the cache in this case do not check for
8287 8287 * i_ddi_io_initialized when force flag is set.
8288 8288 */
8289 8289 if (force_flag == 0 && !i_ddi_io_initialized())
8290 8290 return (MDI_FAILURE);
8291 8291
8292 8292 if ((nvl = vhcache_to_mainnvl(&vhc->vhc_vhcache)) != NULL) {
8293 8293 err = fwrite_nvlist(vhc->vhc_vhcache_filename, nvl);
8294 8294 nvlist_free(nvl);
8295 8295 } else
8296 8296 err = EFAULT;
8297 8297
8298 8298 rv = MDI_SUCCESS;
8299 8299 mutex_enter(&vhc->vhc_lock);
8300 8300 if (err != 0) {
8301 8301 if (err == EROFS) {
8302 8302 vhc->vhc_flags |= MDI_VHC_READONLY_FS;
8303 8303 vhc->vhc_flags &= ~(MDI_VHC_VHCACHE_FLUSH_ERROR |
8304 8304 MDI_VHC_VHCACHE_DIRTY);
8305 8305 } else {
8306 8306 if (!(vhc->vhc_flags & MDI_VHC_VHCACHE_FLUSH_ERROR)) {
8307 8307 cmn_err(CE_CONT, "%s: update failed\n",
8308 8308 vhc->vhc_vhcache_filename);
8309 8309 vhc->vhc_flags |= MDI_VHC_VHCACHE_FLUSH_ERROR;
8310 8310 }
8311 8311 rv = MDI_FAILURE;
8312 8312 }
8313 8313 } else if (vhc->vhc_flags & MDI_VHC_VHCACHE_FLUSH_ERROR) {
8314 8314 cmn_err(CE_CONT,
8315 8315 "%s: update now ok\n", vhc->vhc_vhcache_filename);
8316 8316 vhc->vhc_flags &= ~MDI_VHC_VHCACHE_FLUSH_ERROR;
8317 8317 }
8318 8318 mutex_exit(&vhc->vhc_lock);
8319 8319
8320 8320 return (rv);
8321 8321 }
8322 8322
8323 8323 /*
8324 8324 * Call flush_vhcache() to flush the vhci cache at the scheduled time.
8325 8325 * Exits itself if left idle for the idle timeout period.
8326 8326 */
8327 8327 static void
8328 8328 vhcache_flush_thread(void *arg)
8329 8329 {
8330 8330 mdi_vhci_config_t *vhc = (mdi_vhci_config_t *)arg;
8331 8331 clock_t idle_time, quit_at_ticks;
8332 8332 callb_cpr_t cprinfo;
8333 8333
8334 8334 /* number of seconds to sleep idle before exiting */
8335 8335 idle_time = mdi_vhcache_flush_daemon_idle_time * TICKS_PER_SECOND;
8336 8336
8337 8337 CALLB_CPR_INIT(&cprinfo, &vhc->vhc_lock, callb_generic_cpr,
8338 8338 "mdi_vhcache_flush");
8339 8339 mutex_enter(&vhc->vhc_lock);
8340 8340 for (; ; ) {
8341 8341 while (!(vhc->vhc_flags & MDI_VHC_EXIT) &&
8342 8342 (vhc->vhc_flags & MDI_VHC_VHCACHE_DIRTY)) {
8343 8343 if (ddi_get_lbolt() < vhc->vhc_flush_at_ticks) {
8344 8344 CALLB_CPR_SAFE_BEGIN(&cprinfo);
8345 8345 (void) cv_timedwait(&vhc->vhc_cv,
8346 8346 &vhc->vhc_lock, vhc->vhc_flush_at_ticks);
8347 8347 CALLB_CPR_SAFE_END(&cprinfo, &vhc->vhc_lock);
8348 8348 } else {
8349 8349 vhc->vhc_flags &= ~MDI_VHC_VHCACHE_DIRTY;
8350 8350 mutex_exit(&vhc->vhc_lock);
8351 8351
8352 8352 if (flush_vhcache(vhc, 0) != MDI_SUCCESS)
8353 8353 vhcache_dirty(vhc);
8354 8354
8355 8355 mutex_enter(&vhc->vhc_lock);
8356 8356 }
8357 8357 }
8358 8358
8359 8359 quit_at_ticks = ddi_get_lbolt() + idle_time;
8360 8360
8361 8361 while (!(vhc->vhc_flags & MDI_VHC_EXIT) &&
8362 8362 !(vhc->vhc_flags & MDI_VHC_VHCACHE_DIRTY) &&
8363 8363 ddi_get_lbolt() < quit_at_ticks) {
8364 8364 CALLB_CPR_SAFE_BEGIN(&cprinfo);
8365 8365 (void) cv_timedwait(&vhc->vhc_cv, &vhc->vhc_lock,
8366 8366 quit_at_ticks);
8367 8367 CALLB_CPR_SAFE_END(&cprinfo, &vhc->vhc_lock);
8368 8368 }
8369 8369
8370 8370 if ((vhc->vhc_flags & MDI_VHC_EXIT) ||
8371 8371 !(vhc->vhc_flags & MDI_VHC_VHCACHE_DIRTY))
8372 8372 goto out;
8373 8373 }
8374 8374
8375 8375 out:
8376 8376 vhc->vhc_flags &= ~MDI_VHC_VHCACHE_FLUSH_THREAD;
8377 8377 /* CALLB_CPR_EXIT releases the vhc->vhc_lock */
8378 8378 CALLB_CPR_EXIT(&cprinfo);
8379 8379 }
8380 8380
8381 8381 /*
8382 8382 * Make vhci cache dirty and schedule flushing by vhcache flush thread.
8383 8383 */
8384 8384 static void
8385 8385 vhcache_dirty(mdi_vhci_config_t *vhc)
8386 8386 {
8387 8387 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8388 8388 int create_thread;
8389 8389
8390 8390 rw_enter(&vhcache->vhcache_lock, RW_READER);
8391 8391 /* do not flush cache until the cache is fully built */
8392 8392 if (!(vhcache->vhcache_flags & MDI_VHCI_CACHE_SETUP_DONE)) {
8393 8393 rw_exit(&vhcache->vhcache_lock);
8394 8394 return;
8395 8395 }
8396 8396 rw_exit(&vhcache->vhcache_lock);
8397 8397
8398 8398 mutex_enter(&vhc->vhc_lock);
8399 8399 if (vhc->vhc_flags & MDI_VHC_READONLY_FS) {
8400 8400 mutex_exit(&vhc->vhc_lock);
8401 8401 return;
8402 8402 }
8403 8403
8404 8404 vhc->vhc_flags |= MDI_VHC_VHCACHE_DIRTY;
8405 8405 vhc->vhc_flush_at_ticks = ddi_get_lbolt() +
8406 8406 mdi_vhcache_flush_delay * TICKS_PER_SECOND;
8407 8407 if (vhc->vhc_flags & MDI_VHC_VHCACHE_FLUSH_THREAD) {
8408 8408 cv_broadcast(&vhc->vhc_cv);
8409 8409 create_thread = 0;
8410 8410 } else {
8411 8411 vhc->vhc_flags |= MDI_VHC_VHCACHE_FLUSH_THREAD;
8412 8412 create_thread = 1;
8413 8413 }
8414 8414 mutex_exit(&vhc->vhc_lock);
8415 8415
8416 8416 if (create_thread)
8417 8417 (void) thread_create(NULL, 0, vhcache_flush_thread, vhc,
8418 8418 0, &p0, TS_RUN, minclsyspri);
8419 8419 }
8420 8420
8421 8421 /*
8422 8422 * phci bus config structure - one for for each phci bus config operation that
8423 8423 * we initiate on behalf of a vhci.
8424 8424 */
8425 8425 typedef struct mdi_phci_bus_config_s {
8426 8426 char *phbc_phci_path;
8427 8427 struct mdi_vhci_bus_config_s *phbc_vhbusconfig; /* vhci bus config */
8428 8428 struct mdi_phci_bus_config_s *phbc_next;
8429 8429 } mdi_phci_bus_config_t;
8430 8430
8431 8431 /* vhci bus config structure - one for each vhci bus config operation */
8432 8432 typedef struct mdi_vhci_bus_config_s {
8433 8433 ddi_bus_config_op_t vhbc_op; /* bus config op */
8434 8434 major_t vhbc_op_major; /* bus config op major */
8435 8435 uint_t vhbc_op_flags; /* bus config op flags */
8436 8436 kmutex_t vhbc_lock;
8437 8437 kcondvar_t vhbc_cv;
8438 8438 int vhbc_thr_count;
8439 8439 } mdi_vhci_bus_config_t;
8440 8440
8441 8441 /*
8442 8442 * bus config the specified phci
8443 8443 */
8444 8444 static void
8445 8445 bus_config_phci(void *arg)
8446 8446 {
8447 8447 mdi_phci_bus_config_t *phbc = (mdi_phci_bus_config_t *)arg;
8448 8448 mdi_vhci_bus_config_t *vhbc = phbc->phbc_vhbusconfig;
8449 8449 dev_info_t *ph_dip;
8450 8450
8451 8451 /*
8452 8452 * first configure all path components upto phci and then configure
8453 8453 * the phci children.
8454 8454 */
8455 8455 if ((ph_dip = e_ddi_hold_devi_by_path(phbc->phbc_phci_path, 0))
8456 8456 != NULL) {
8457 8457 if (vhbc->vhbc_op == BUS_CONFIG_DRIVER ||
8458 8458 vhbc->vhbc_op == BUS_UNCONFIG_DRIVER) {
8459 8459 (void) ndi_devi_config_driver(ph_dip,
8460 8460 vhbc->vhbc_op_flags,
8461 8461 vhbc->vhbc_op_major);
8462 8462 } else
8463 8463 (void) ndi_devi_config(ph_dip,
8464 8464 vhbc->vhbc_op_flags);
8465 8465
8466 8466 /* release the hold that e_ddi_hold_devi_by_path() placed */
8467 8467 ndi_rele_devi(ph_dip);
8468 8468 }
8469 8469
8470 8470 kmem_free(phbc->phbc_phci_path, strlen(phbc->phbc_phci_path) + 1);
8471 8471 kmem_free(phbc, sizeof (*phbc));
8472 8472
8473 8473 mutex_enter(&vhbc->vhbc_lock);
8474 8474 vhbc->vhbc_thr_count--;
8475 8475 if (vhbc->vhbc_thr_count == 0)
8476 8476 cv_broadcast(&vhbc->vhbc_cv);
8477 8477 mutex_exit(&vhbc->vhbc_lock);
8478 8478 }
8479 8479
8480 8480 /*
8481 8481 * Bus config all phcis associated with the vhci in parallel.
8482 8482 * op must be BUS_CONFIG_DRIVER or BUS_CONFIG_ALL.
8483 8483 */
8484 8484 static void
8485 8485 bus_config_all_phcis(mdi_vhci_cache_t *vhcache, uint_t flags,
8486 8486 ddi_bus_config_op_t op, major_t maj)
8487 8487 {
8488 8488 mdi_phci_bus_config_t *phbc_head = NULL, *phbc, *phbc_next;
8489 8489 mdi_vhci_bus_config_t *vhbc;
8490 8490 mdi_vhcache_phci_t *cphci;
8491 8491
8492 8492 rw_enter(&vhcache->vhcache_lock, RW_READER);
8493 8493 if (vhcache->vhcache_phci_head == NULL) {
8494 8494 rw_exit(&vhcache->vhcache_lock);
8495 8495 return;
8496 8496 }
8497 8497
8498 8498 vhbc = kmem_zalloc(sizeof (*vhbc), KM_SLEEP);
8499 8499
8500 8500 for (cphci = vhcache->vhcache_phci_head; cphci != NULL;
8501 8501 cphci = cphci->cphci_next) {
8502 8502 /* skip phcis that haven't attached before root is available */
8503 8503 if (!modrootloaded && (cphci->cphci_phci == NULL))
8504 8504 continue;
8505 8505 phbc = kmem_zalloc(sizeof (*phbc), KM_SLEEP);
8506 8506 phbc->phbc_phci_path = i_ddi_strdup(cphci->cphci_path,
8507 8507 KM_SLEEP);
8508 8508 phbc->phbc_vhbusconfig = vhbc;
8509 8509 phbc->phbc_next = phbc_head;
8510 8510 phbc_head = phbc;
8511 8511 vhbc->vhbc_thr_count++;
8512 8512 }
8513 8513 rw_exit(&vhcache->vhcache_lock);
8514 8514
8515 8515 vhbc->vhbc_op = op;
8516 8516 vhbc->vhbc_op_major = maj;
8517 8517 vhbc->vhbc_op_flags = NDI_NO_EVENT |
8518 8518 (flags & (NDI_CONFIG_REPROBE | NDI_DRV_CONF_REPROBE));
8519 8519 mutex_init(&vhbc->vhbc_lock, NULL, MUTEX_DEFAULT, NULL);
8520 8520 cv_init(&vhbc->vhbc_cv, NULL, CV_DRIVER, NULL);
8521 8521
8522 8522 /* now create threads to initiate bus config on all phcis in parallel */
8523 8523 for (phbc = phbc_head; phbc != NULL; phbc = phbc_next) {
8524 8524 phbc_next = phbc->phbc_next;
8525 8525 if (mdi_mtc_off)
8526 8526 bus_config_phci((void *)phbc);
8527 8527 else
8528 8528 (void) thread_create(NULL, 0, bus_config_phci, phbc,
8529 8529 0, &p0, TS_RUN, minclsyspri);
8530 8530 }
8531 8531
8532 8532 mutex_enter(&vhbc->vhbc_lock);
8533 8533 /* wait until all threads exit */
8534 8534 while (vhbc->vhbc_thr_count > 0)
8535 8535 cv_wait(&vhbc->vhbc_cv, &vhbc->vhbc_lock);
8536 8536 mutex_exit(&vhbc->vhbc_lock);
8537 8537
8538 8538 mutex_destroy(&vhbc->vhbc_lock);
8539 8539 cv_destroy(&vhbc->vhbc_cv);
8540 8540 kmem_free(vhbc, sizeof (*vhbc));
8541 8541 }
8542 8542
8543 8543 /*
8544 8544 * Single threaded version of bus_config_all_phcis()
8545 8545 */
8546 8546 static void
8547 8547 st_bus_config_all_phcis(mdi_vhci_config_t *vhc, uint_t flags,
8548 8548 ddi_bus_config_op_t op, major_t maj)
8549 8549 {
8550 8550 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8551 8551
8552 8552 single_threaded_vhconfig_enter(vhc);
8553 8553 bus_config_all_phcis(vhcache, flags, op, maj);
8554 8554 single_threaded_vhconfig_exit(vhc);
8555 8555 }
8556 8556
8557 8557 /*
8558 8558 * Perform BUS_CONFIG_ONE on the specified child of the phci.
8559 8559 * The path includes the child component in addition to the phci path.
8560 8560 */
8561 8561 static int
8562 8562 bus_config_one_phci_child(char *path)
8563 8563 {
8564 8564 dev_info_t *ph_dip, *child;
8565 8565 char *devnm;
8566 8566 int rv = MDI_FAILURE;
8567 8567
8568 8568 /* extract the child component of the phci */
8569 8569 devnm = strrchr(path, '/');
8570 8570 *devnm++ = '\0';
8571 8571
8572 8572 /*
8573 8573 * first configure all path components upto phci and then
8574 8574 * configure the phci child.
8575 8575 */
8576 8576 if ((ph_dip = e_ddi_hold_devi_by_path(path, 0)) != NULL) {
8577 8577 if (ndi_devi_config_one(ph_dip, devnm, &child, NDI_NO_EVENT) ==
8578 8578 NDI_SUCCESS) {
8579 8579 /*
8580 8580 * release the hold that ndi_devi_config_one() placed
8581 8581 */
8582 8582 ndi_rele_devi(child);
8583 8583 rv = MDI_SUCCESS;
8584 8584 }
8585 8585
8586 8586 /* release the hold that e_ddi_hold_devi_by_path() placed */
8587 8587 ndi_rele_devi(ph_dip);
8588 8588 }
8589 8589
8590 8590 devnm--;
8591 8591 *devnm = '/';
8592 8592 return (rv);
8593 8593 }
8594 8594
8595 8595 /*
8596 8596 * Build a list of phci client paths for the specified vhci client.
8597 8597 * The list includes only those phci client paths which aren't configured yet.
8598 8598 */
8599 8599 static mdi_phys_path_t *
8600 8600 build_phclient_path_list(mdi_vhcache_client_t *cct, char *ct_name)
8601 8601 {
8602 8602 mdi_vhcache_pathinfo_t *cpi;
8603 8603 mdi_phys_path_t *pp_head = NULL, *pp_tail = NULL, *pp;
8604 8604 int config_path, len;
8605 8605
8606 8606 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi->cpi_next) {
8607 8607 /*
8608 8608 * include only those paths that aren't configured.
8609 8609 */
8610 8610 config_path = 0;
8611 8611 if (cpi->cpi_pip == NULL)
8612 8612 config_path = 1;
8613 8613 else {
8614 8614 MDI_PI_LOCK(cpi->cpi_pip);
8615 8615 if (MDI_PI_IS_INIT(cpi->cpi_pip))
8616 8616 config_path = 1;
8617 8617 MDI_PI_UNLOCK(cpi->cpi_pip);
8618 8618 }
8619 8619
8620 8620 if (config_path) {
8621 8621 pp = kmem_alloc(sizeof (*pp), KM_SLEEP);
8622 8622 len = strlen(cpi->cpi_cphci->cphci_path) +
8623 8623 strlen(ct_name) + strlen(cpi->cpi_addr) + 3;
8624 8624 pp->phys_path = kmem_alloc(len, KM_SLEEP);
8625 8625 (void) snprintf(pp->phys_path, len, "%s/%s@%s",
8626 8626 cpi->cpi_cphci->cphci_path, ct_name,
8627 8627 cpi->cpi_addr);
8628 8628 pp->phys_path_next = NULL;
8629 8629
8630 8630 if (pp_head == NULL)
8631 8631 pp_head = pp;
8632 8632 else
8633 8633 pp_tail->phys_path_next = pp;
8634 8634 pp_tail = pp;
8635 8635 }
8636 8636 }
8637 8637
8638 8638 return (pp_head);
8639 8639 }
8640 8640
8641 8641 /*
8642 8642 * Free the memory allocated for phci client path list.
8643 8643 */
8644 8644 static void
8645 8645 free_phclient_path_list(mdi_phys_path_t *pp_head)
8646 8646 {
8647 8647 mdi_phys_path_t *pp, *pp_next;
8648 8648
8649 8649 for (pp = pp_head; pp != NULL; pp = pp_next) {
8650 8650 pp_next = pp->phys_path_next;
8651 8651 kmem_free(pp->phys_path, strlen(pp->phys_path) + 1);
8652 8652 kmem_free(pp, sizeof (*pp));
8653 8653 }
8654 8654 }
8655 8655
8656 8656 /*
8657 8657 * Allocated async client structure and initialize with the specified values.
8658 8658 */
8659 8659 static mdi_async_client_config_t *
8660 8660 alloc_async_client_config(char *ct_name, char *ct_addr,
8661 8661 mdi_phys_path_t *pp_head, mdi_vhcache_lookup_token_t *tok)
8662 8662 {
8663 8663 mdi_async_client_config_t *acc;
8664 8664
8665 8665 acc = kmem_alloc(sizeof (*acc), KM_SLEEP);
8666 8666 acc->acc_ct_name = i_ddi_strdup(ct_name, KM_SLEEP);
8667 8667 acc->acc_ct_addr = i_ddi_strdup(ct_addr, KM_SLEEP);
8668 8668 acc->acc_phclient_path_list_head = pp_head;
8669 8669 init_vhcache_lookup_token(&acc->acc_token, tok);
8670 8670 acc->acc_next = NULL;
8671 8671 return (acc);
8672 8672 }
8673 8673
8674 8674 /*
8675 8675 * Free the memory allocated for the async client structure and their members.
8676 8676 */
8677 8677 static void
8678 8678 free_async_client_config(mdi_async_client_config_t *acc)
8679 8679 {
8680 8680 if (acc->acc_phclient_path_list_head)
8681 8681 free_phclient_path_list(acc->acc_phclient_path_list_head);
8682 8682 kmem_free(acc->acc_ct_name, strlen(acc->acc_ct_name) + 1);
8683 8683 kmem_free(acc->acc_ct_addr, strlen(acc->acc_ct_addr) + 1);
8684 8684 kmem_free(acc, sizeof (*acc));
8685 8685 }
8686 8686
8687 8687 /*
8688 8688 * Sort vhcache pathinfos (cpis) of the specified client.
8689 8689 * All cpis which do not have MDI_CPI_HINT_PATH_DOES_NOT_EXIST
8690 8690 * flag set come at the beginning of the list. All cpis which have this
8691 8691 * flag set come at the end of the list.
8692 8692 */
8693 8693 static void
8694 8694 sort_vhcache_paths(mdi_vhcache_client_t *cct)
8695 8695 {
8696 8696 mdi_vhcache_pathinfo_t *cpi, *cpi_next, *cpi_head;
8697 8697
8698 8698 cpi_head = cct->cct_cpi_head;
8699 8699 cct->cct_cpi_head = cct->cct_cpi_tail = NULL;
8700 8700 for (cpi = cpi_head; cpi != NULL; cpi = cpi_next) {
8701 8701 cpi_next = cpi->cpi_next;
8702 8702 enqueue_vhcache_pathinfo(cct, cpi);
8703 8703 }
8704 8704 }
8705 8705
8706 8706 /*
8707 8707 * Verify whether MDI_CPI_HINT_PATH_DOES_NOT_EXIST flag setting is correct for
8708 8708 * every vhcache pathinfo of the specified client. If not adjust the flag
8709 8709 * setting appropriately.
8710 8710 *
8711 8711 * Note that MDI_CPI_HINT_PATH_DOES_NOT_EXIST flag is persisted in the
8712 8712 * on-disk vhci cache. So every time this flag is updated the cache must be
8713 8713 * flushed.
8714 8714 */
8715 8715 static void
8716 8716 adjust_sort_vhcache_paths(mdi_vhci_config_t *vhc, char *ct_name, char *ct_addr,
8717 8717 mdi_vhcache_lookup_token_t *tok)
8718 8718 {
8719 8719 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8720 8720 mdi_vhcache_client_t *cct;
8721 8721 mdi_vhcache_pathinfo_t *cpi;
8722 8722
8723 8723 rw_enter(&vhcache->vhcache_lock, RW_READER);
8724 8724 if ((cct = lookup_vhcache_client(vhcache, ct_name, ct_addr, tok))
8725 8725 == NULL) {
8726 8726 rw_exit(&vhcache->vhcache_lock);
8727 8727 return;
8728 8728 }
8729 8729
8730 8730 /*
8731 8731 * to avoid unnecessary on-disk cache updates, first check if an
8732 8732 * update is really needed. If no update is needed simply return.
8733 8733 */
8734 8734 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi->cpi_next) {
8735 8735 if ((cpi->cpi_pip != NULL &&
8736 8736 (cpi->cpi_flags & MDI_CPI_HINT_PATH_DOES_NOT_EXIST)) ||
8737 8737 (cpi->cpi_pip == NULL &&
8738 8738 !(cpi->cpi_flags & MDI_CPI_HINT_PATH_DOES_NOT_EXIST))) {
8739 8739 break;
8740 8740 }
8741 8741 }
8742 8742 if (cpi == NULL) {
8743 8743 rw_exit(&vhcache->vhcache_lock);
8744 8744 return;
8745 8745 }
8746 8746
8747 8747 if (rw_tryupgrade(&vhcache->vhcache_lock) == 0) {
8748 8748 rw_exit(&vhcache->vhcache_lock);
8749 8749 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
8750 8750 if ((cct = lookup_vhcache_client(vhcache, ct_name, ct_addr,
8751 8751 tok)) == NULL) {
8752 8752 rw_exit(&vhcache->vhcache_lock);
8753 8753 return;
8754 8754 }
8755 8755 }
8756 8756
8757 8757 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi->cpi_next) {
8758 8758 if (cpi->cpi_pip != NULL)
8759 8759 cpi->cpi_flags &= ~MDI_CPI_HINT_PATH_DOES_NOT_EXIST;
8760 8760 else
8761 8761 cpi->cpi_flags |= MDI_CPI_HINT_PATH_DOES_NOT_EXIST;
8762 8762 }
8763 8763 sort_vhcache_paths(cct);
8764 8764
8765 8765 rw_exit(&vhcache->vhcache_lock);
8766 8766 vhcache_dirty(vhc);
8767 8767 }
8768 8768
8769 8769 /*
8770 8770 * Configure all specified paths of the client.
8771 8771 */
8772 8772 static void
8773 8773 config_client_paths_sync(mdi_vhci_config_t *vhc, char *ct_name, char *ct_addr,
8774 8774 mdi_phys_path_t *pp_head, mdi_vhcache_lookup_token_t *tok)
8775 8775 {
8776 8776 mdi_phys_path_t *pp;
8777 8777
8778 8778 for (pp = pp_head; pp != NULL; pp = pp->phys_path_next)
8779 8779 (void) bus_config_one_phci_child(pp->phys_path);
8780 8780 adjust_sort_vhcache_paths(vhc, ct_name, ct_addr, tok);
8781 8781 }
8782 8782
8783 8783 /*
8784 8784 * Dequeue elements from vhci async client config list and bus configure
8785 8785 * their corresponding phci clients.
8786 8786 */
8787 8787 static void
8788 8788 config_client_paths_thread(void *arg)
8789 8789 {
8790 8790 mdi_vhci_config_t *vhc = (mdi_vhci_config_t *)arg;
8791 8791 mdi_async_client_config_t *acc;
8792 8792 clock_t quit_at_ticks;
8793 8793 clock_t idle_time = mdi_async_config_idle_time * TICKS_PER_SECOND;
8794 8794 callb_cpr_t cprinfo;
8795 8795
8796 8796 CALLB_CPR_INIT(&cprinfo, &vhc->vhc_lock, callb_generic_cpr,
8797 8797 "mdi_config_client_paths");
8798 8798
8799 8799 for (; ; ) {
8800 8800 quit_at_ticks = ddi_get_lbolt() + idle_time;
8801 8801
8802 8802 mutex_enter(&vhc->vhc_lock);
8803 8803 while (!(vhc->vhc_flags & MDI_VHC_EXIT) &&
8804 8804 vhc->vhc_acc_list_head == NULL &&
8805 8805 ddi_get_lbolt() < quit_at_ticks) {
8806 8806 CALLB_CPR_SAFE_BEGIN(&cprinfo);
8807 8807 (void) cv_timedwait(&vhc->vhc_cv, &vhc->vhc_lock,
8808 8808 quit_at_ticks);
8809 8809 CALLB_CPR_SAFE_END(&cprinfo, &vhc->vhc_lock);
8810 8810 }
8811 8811
8812 8812 if ((vhc->vhc_flags & MDI_VHC_EXIT) ||
8813 8813 vhc->vhc_acc_list_head == NULL)
8814 8814 goto out;
8815 8815
8816 8816 acc = vhc->vhc_acc_list_head;
8817 8817 vhc->vhc_acc_list_head = acc->acc_next;
8818 8818 if (vhc->vhc_acc_list_head == NULL)
8819 8819 vhc->vhc_acc_list_tail = NULL;
8820 8820 vhc->vhc_acc_count--;
8821 8821 mutex_exit(&vhc->vhc_lock);
8822 8822
8823 8823 config_client_paths_sync(vhc, acc->acc_ct_name,
8824 8824 acc->acc_ct_addr, acc->acc_phclient_path_list_head,
8825 8825 &acc->acc_token);
8826 8826
8827 8827 free_async_client_config(acc);
8828 8828 }
8829 8829
8830 8830 out:
8831 8831 vhc->vhc_acc_thrcount--;
8832 8832 /* CALLB_CPR_EXIT releases the vhc->vhc_lock */
8833 8833 CALLB_CPR_EXIT(&cprinfo);
8834 8834 }
8835 8835
8836 8836 /*
8837 8837 * Arrange for all the phci client paths (pp_head) for the specified client
8838 8838 * to be bus configured asynchronously by a thread.
8839 8839 */
8840 8840 static void
8841 8841 config_client_paths_async(mdi_vhci_config_t *vhc, char *ct_name, char *ct_addr,
8842 8842 mdi_phys_path_t *pp_head, mdi_vhcache_lookup_token_t *tok)
8843 8843 {
8844 8844 mdi_async_client_config_t *acc, *newacc;
8845 8845 int create_thread;
8846 8846
8847 8847 if (pp_head == NULL)
8848 8848 return;
8849 8849
8850 8850 if (mdi_mtc_off) {
8851 8851 config_client_paths_sync(vhc, ct_name, ct_addr, pp_head, tok);
8852 8852 free_phclient_path_list(pp_head);
8853 8853 return;
8854 8854 }
8855 8855
8856 8856 newacc = alloc_async_client_config(ct_name, ct_addr, pp_head, tok);
8857 8857 ASSERT(newacc);
8858 8858
8859 8859 mutex_enter(&vhc->vhc_lock);
8860 8860 for (acc = vhc->vhc_acc_list_head; acc != NULL; acc = acc->acc_next) {
8861 8861 if (strcmp(ct_name, acc->acc_ct_name) == 0 &&
8862 8862 strcmp(ct_addr, acc->acc_ct_addr) == 0) {
8863 8863 free_async_client_config(newacc);
8864 8864 mutex_exit(&vhc->vhc_lock);
8865 8865 return;
8866 8866 }
8867 8867 }
8868 8868
8869 8869 if (vhc->vhc_acc_list_head == NULL)
8870 8870 vhc->vhc_acc_list_head = newacc;
8871 8871 else
8872 8872 vhc->vhc_acc_list_tail->acc_next = newacc;
8873 8873 vhc->vhc_acc_list_tail = newacc;
8874 8874 vhc->vhc_acc_count++;
8875 8875 if (vhc->vhc_acc_count <= vhc->vhc_acc_thrcount) {
8876 8876 cv_broadcast(&vhc->vhc_cv);
8877 8877 create_thread = 0;
8878 8878 } else {
8879 8879 vhc->vhc_acc_thrcount++;
8880 8880 create_thread = 1;
8881 8881 }
8882 8882 mutex_exit(&vhc->vhc_lock);
8883 8883
8884 8884 if (create_thread)
8885 8885 (void) thread_create(NULL, 0, config_client_paths_thread, vhc,
8886 8886 0, &p0, TS_RUN, minclsyspri);
8887 8887 }
8888 8888
8889 8889 /*
8890 8890 * Return number of online paths for the specified client.
8891 8891 */
8892 8892 static int
8893 8893 nonline_paths(mdi_vhcache_client_t *cct)
8894 8894 {
8895 8895 mdi_vhcache_pathinfo_t *cpi;
8896 8896 int online_count = 0;
8897 8897
8898 8898 for (cpi = cct->cct_cpi_head; cpi != NULL; cpi = cpi->cpi_next) {
8899 8899 if (cpi->cpi_pip != NULL) {
8900 8900 MDI_PI_LOCK(cpi->cpi_pip);
8901 8901 if (cpi->cpi_pip->pi_state == MDI_PATHINFO_STATE_ONLINE)
8902 8902 online_count++;
8903 8903 MDI_PI_UNLOCK(cpi->cpi_pip);
8904 8904 }
8905 8905 }
8906 8906
8907 8907 return (online_count);
8908 8908 }
8909 8909
8910 8910 /*
8911 8911 * Bus configure all paths for the specified vhci client.
8912 8912 * If at least one path for the client is already online, the remaining paths
8913 8913 * will be configured asynchronously. Otherwise, it synchronously configures
8914 8914 * the paths until at least one path is online and then rest of the paths
8915 8915 * will be configured asynchronously.
8916 8916 */
8917 8917 static void
8918 8918 config_client_paths(mdi_vhci_config_t *vhc, char *ct_name, char *ct_addr)
8919 8919 {
8920 8920 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
8921 8921 mdi_phys_path_t *pp_head, *pp;
8922 8922 mdi_vhcache_client_t *cct;
8923 8923 mdi_vhcache_lookup_token_t tok;
8924 8924
8925 8925 ASSERT(RW_LOCK_HELD(&vhcache->vhcache_lock));
8926 8926
8927 8927 init_vhcache_lookup_token(&tok, NULL);
8928 8928
8929 8929 if (ct_name == NULL || ct_addr == NULL ||
8930 8930 (cct = lookup_vhcache_client(vhcache, ct_name, ct_addr, &tok))
8931 8931 == NULL ||
8932 8932 (pp_head = build_phclient_path_list(cct, ct_name)) == NULL) {
8933 8933 rw_exit(&vhcache->vhcache_lock);
8934 8934 return;
8935 8935 }
8936 8936
8937 8937 /* if at least one path is online, configure the rest asynchronously */
8938 8938 if (nonline_paths(cct) > 0) {
8939 8939 rw_exit(&vhcache->vhcache_lock);
8940 8940 config_client_paths_async(vhc, ct_name, ct_addr, pp_head, &tok);
8941 8941 return;
8942 8942 }
8943 8943
8944 8944 rw_exit(&vhcache->vhcache_lock);
8945 8945
8946 8946 for (pp = pp_head; pp != NULL; pp = pp->phys_path_next) {
8947 8947 if (bus_config_one_phci_child(pp->phys_path) == MDI_SUCCESS) {
8948 8948 rw_enter(&vhcache->vhcache_lock, RW_READER);
8949 8949
8950 8950 if ((cct = lookup_vhcache_client(vhcache, ct_name,
8951 8951 ct_addr, &tok)) == NULL) {
8952 8952 rw_exit(&vhcache->vhcache_lock);
8953 8953 goto out;
8954 8954 }
8955 8955
8956 8956 if (nonline_paths(cct) > 0 &&
8957 8957 pp->phys_path_next != NULL) {
8958 8958 rw_exit(&vhcache->vhcache_lock);
8959 8959 config_client_paths_async(vhc, ct_name, ct_addr,
8960 8960 pp->phys_path_next, &tok);
8961 8961 pp->phys_path_next = NULL;
8962 8962 goto out;
8963 8963 }
8964 8964
8965 8965 rw_exit(&vhcache->vhcache_lock);
8966 8966 }
8967 8967 }
8968 8968
8969 8969 adjust_sort_vhcache_paths(vhc, ct_name, ct_addr, &tok);
8970 8970 out:
8971 8971 free_phclient_path_list(pp_head);
8972 8972 }
8973 8973
8974 8974 static void
8975 8975 single_threaded_vhconfig_enter(mdi_vhci_config_t *vhc)
8976 8976 {
8977 8977 mutex_enter(&vhc->vhc_lock);
8978 8978 while (vhc->vhc_flags & MDI_VHC_SINGLE_THREADED)
8979 8979 cv_wait(&vhc->vhc_cv, &vhc->vhc_lock);
8980 8980 vhc->vhc_flags |= MDI_VHC_SINGLE_THREADED;
8981 8981 mutex_exit(&vhc->vhc_lock);
8982 8982 }
8983 8983
8984 8984 static void
8985 8985 single_threaded_vhconfig_exit(mdi_vhci_config_t *vhc)
8986 8986 {
8987 8987 mutex_enter(&vhc->vhc_lock);
8988 8988 vhc->vhc_flags &= ~MDI_VHC_SINGLE_THREADED;
8989 8989 cv_broadcast(&vhc->vhc_cv);
8990 8990 mutex_exit(&vhc->vhc_lock);
8991 8991 }
8992 8992
8993 8993 typedef struct mdi_phci_driver_info {
8994 8994 char *phdriver_name; /* name of the phci driver */
8995 8995
8996 8996 /* set to non zero if the phci driver supports root device */
8997 8997 int phdriver_root_support;
8998 8998 } mdi_phci_driver_info_t;
8999 8999
9000 9000 /*
9001 9001 * vhci class and root support capability of a phci driver can be
9002 9002 * specified using ddi-vhci-class and ddi-no-root-support properties in the
9003 9003 * phci driver.conf file. The built-in tables below contain this information
↓ open down ↓ |
9003 lines elided |
↑ open up ↑ |
9004 9004 * for those phci drivers whose driver.conf files don't yet contain this info.
9005 9005 *
9006 9006 * All phci drivers expect iscsi have root device support.
9007 9007 */
9008 9008 static mdi_phci_driver_info_t scsi_phci_driver_list[] = {
9009 9009 { "fp", 1 },
9010 9010 { "iscsi", 0 },
9011 9011 { "ibsrp", 1 }
9012 9012 };
9013 9013
9014 -static mdi_phci_driver_info_t ib_phci_driver_list[] = { "tavor", 1 };
9014 +static mdi_phci_driver_info_t ib_phci_driver_list[] = {{ "tavor", 1 }};
9015 9015
9016 9016 static void *
9017 9017 mdi_realloc(void *old_ptr, size_t old_size, size_t new_size)
9018 9018 {
9019 9019 void *new_ptr;
9020 9020
9021 9021 new_ptr = kmem_zalloc(new_size, KM_SLEEP);
9022 9022 if (old_ptr) {
9023 9023 bcopy(old_ptr, new_ptr, MIN(old_size, new_size));
9024 9024 kmem_free(old_ptr, old_size);
9025 9025 }
9026 9026 return (new_ptr);
9027 9027 }
9028 9028
9029 9029 static void
9030 9030 add_to_phci_list(char ***driver_list, int **root_support_list,
9031 9031 int *cur_elements, int *max_elements, char *driver_name, int root_support)
9032 9032 {
9033 9033 ASSERT(*cur_elements <= *max_elements);
9034 9034 if (*cur_elements == *max_elements) {
9035 9035 *max_elements += 10;
9036 9036 *driver_list = mdi_realloc(*driver_list,
9037 9037 sizeof (char *) * (*cur_elements),
9038 9038 sizeof (char *) * (*max_elements));
9039 9039 *root_support_list = mdi_realloc(*root_support_list,
9040 9040 sizeof (int) * (*cur_elements),
9041 9041 sizeof (int) * (*max_elements));
9042 9042 }
9043 9043 (*driver_list)[*cur_elements] = i_ddi_strdup(driver_name, KM_SLEEP);
9044 9044 (*root_support_list)[*cur_elements] = root_support;
9045 9045 (*cur_elements)++;
9046 9046 }
9047 9047
9048 9048 static void
9049 9049 get_phci_driver_list(char *vhci_class, char ***driver_list,
9050 9050 int **root_support_list, int *cur_elements, int *max_elements)
9051 9051 {
9052 9052 mdi_phci_driver_info_t *st_driver_list, *p;
9053 9053 int st_ndrivers, root_support, i, j, driver_conf_count;
9054 9054 major_t m;
9055 9055 struct devnames *dnp;
9056 9056 ddi_prop_t *propp;
9057 9057
9058 9058 *driver_list = NULL;
9059 9059 *root_support_list = NULL;
9060 9060 *cur_elements = 0;
9061 9061 *max_elements = 0;
9062 9062
9063 9063 /* add the phci drivers derived from the phci driver.conf files */
9064 9064 for (m = 0; m < devcnt; m++) {
9065 9065 dnp = &devnamesp[m];
9066 9066
9067 9067 if (dnp->dn_flags & DN_PHCI_DRIVER) {
9068 9068 LOCK_DEV_OPS(&dnp->dn_lock);
9069 9069 if (dnp->dn_global_prop_ptr != NULL &&
9070 9070 (propp = i_ddi_prop_search(DDI_DEV_T_ANY,
9071 9071 DDI_VHCI_CLASS, DDI_PROP_TYPE_STRING,
9072 9072 &dnp->dn_global_prop_ptr->prop_list)) != NULL &&
9073 9073 strcmp(propp->prop_val, vhci_class) == 0) {
9074 9074
9075 9075 root_support = (i_ddi_prop_search(DDI_DEV_T_ANY,
9076 9076 DDI_NO_ROOT_SUPPORT, DDI_PROP_TYPE_INT,
9077 9077 &dnp->dn_global_prop_ptr->prop_list)
9078 9078 == NULL) ? 1 : 0;
9079 9079
9080 9080 add_to_phci_list(driver_list, root_support_list,
9081 9081 cur_elements, max_elements, dnp->dn_name,
9082 9082 root_support);
9083 9083
9084 9084 UNLOCK_DEV_OPS(&dnp->dn_lock);
9085 9085 } else
9086 9086 UNLOCK_DEV_OPS(&dnp->dn_lock);
9087 9087 }
9088 9088 }
9089 9089
9090 9090 driver_conf_count = *cur_elements;
9091 9091
9092 9092 /* add the phci drivers specified in the built-in tables */
9093 9093 if (strcmp(vhci_class, MDI_HCI_CLASS_SCSI) == 0) {
9094 9094 st_driver_list = scsi_phci_driver_list;
9095 9095 st_ndrivers = sizeof (scsi_phci_driver_list) /
9096 9096 sizeof (mdi_phci_driver_info_t);
9097 9097 } else if (strcmp(vhci_class, MDI_HCI_CLASS_IB) == 0) {
9098 9098 st_driver_list = ib_phci_driver_list;
9099 9099 st_ndrivers = sizeof (ib_phci_driver_list) /
9100 9100 sizeof (mdi_phci_driver_info_t);
9101 9101 } else {
9102 9102 st_driver_list = NULL;
9103 9103 st_ndrivers = 0;
9104 9104 }
9105 9105
9106 9106 for (i = 0, p = st_driver_list; i < st_ndrivers; i++, p++) {
9107 9107 /* add this phci driver if not already added before */
9108 9108 for (j = 0; j < driver_conf_count; j++) {
9109 9109 if (strcmp((*driver_list)[j], p->phdriver_name) == 0)
9110 9110 break;
9111 9111 }
9112 9112 if (j == driver_conf_count) {
9113 9113 add_to_phci_list(driver_list, root_support_list,
9114 9114 cur_elements, max_elements, p->phdriver_name,
9115 9115 p->phdriver_root_support);
9116 9116 }
9117 9117 }
9118 9118 }
9119 9119
9120 9120 /*
9121 9121 * Attach the phci driver instances associated with the specified vhci class.
9122 9122 * If root is mounted attach all phci driver instances.
9123 9123 * If root is not mounted, attach the instances of only those phci
9124 9124 * drivers that have the root support.
9125 9125 */
9126 9126 static void
9127 9127 attach_phci_drivers(char *vhci_class)
9128 9128 {
9129 9129 char **driver_list, **p;
9130 9130 int *root_support_list;
9131 9131 int cur_elements, max_elements, i;
9132 9132 major_t m;
9133 9133
9134 9134 get_phci_driver_list(vhci_class, &driver_list, &root_support_list,
9135 9135 &cur_elements, &max_elements);
9136 9136
9137 9137 for (i = 0; i < cur_elements; i++) {
9138 9138 if (modrootloaded || root_support_list[i]) {
9139 9139 m = ddi_name_to_major(driver_list[i]);
9140 9140 if (m != DDI_MAJOR_T_NONE &&
9141 9141 ddi_hold_installed_driver(m))
9142 9142 ddi_rele_driver(m);
9143 9143 }
9144 9144 }
9145 9145
9146 9146 if (driver_list) {
9147 9147 for (i = 0, p = driver_list; i < cur_elements; i++, p++)
9148 9148 kmem_free(*p, strlen(*p) + 1);
9149 9149 kmem_free(driver_list, sizeof (char *) * max_elements);
9150 9150 kmem_free(root_support_list, sizeof (int) * max_elements);
9151 9151 }
9152 9152 }
9153 9153
9154 9154 /*
9155 9155 * Build vhci cache:
9156 9156 *
9157 9157 * Attach phci driver instances and then drive BUS_CONFIG_ALL on
9158 9158 * the phci driver instances. During this process the cache gets built.
9159 9159 *
9160 9160 * Cache is built fully if the root is mounted.
9161 9161 * If the root is not mounted, phci drivers that do not have root support
9162 9162 * are not attached. As a result the cache is built partially. The entries
9163 9163 * in the cache reflect only those phci drivers that have root support.
9164 9164 */
9165 9165 static int
9166 9166 build_vhci_cache(mdi_vhci_t *vh)
9167 9167 {
9168 9168 mdi_vhci_config_t *vhc = vh->vh_config;
9169 9169 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
9170 9170
9171 9171 single_threaded_vhconfig_enter(vhc);
9172 9172
9173 9173 rw_enter(&vhcache->vhcache_lock, RW_READER);
9174 9174 if (vhcache->vhcache_flags & MDI_VHCI_CACHE_SETUP_DONE) {
9175 9175 rw_exit(&vhcache->vhcache_lock);
9176 9176 single_threaded_vhconfig_exit(vhc);
9177 9177 return (0);
9178 9178 }
9179 9179 rw_exit(&vhcache->vhcache_lock);
9180 9180
9181 9181 attach_phci_drivers(vh->vh_class);
9182 9182 bus_config_all_phcis(vhcache, NDI_DRV_CONF_REPROBE | NDI_NO_EVENT,
9183 9183 BUS_CONFIG_ALL, DDI_MAJOR_T_NONE);
9184 9184
9185 9185 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
9186 9186 vhcache->vhcache_flags |= MDI_VHCI_CACHE_SETUP_DONE;
9187 9187 rw_exit(&vhcache->vhcache_lock);
9188 9188
9189 9189 single_threaded_vhconfig_exit(vhc);
9190 9190 vhcache_dirty(vhc);
9191 9191 return (1);
9192 9192 }
9193 9193
9194 9194 /*
9195 9195 * Determine if discovery of paths is needed.
9196 9196 */
9197 9197 static int
9198 9198 vhcache_do_discovery(mdi_vhci_config_t *vhc)
9199 9199 {
9200 9200 int rv = 1;
9201 9201
9202 9202 mutex_enter(&vhc->vhc_lock);
9203 9203 if (i_ddi_io_initialized() == 0) {
9204 9204 if (vhc->vhc_path_discovery_boot > 0) {
9205 9205 vhc->vhc_path_discovery_boot--;
9206 9206 goto out;
9207 9207 }
9208 9208 } else {
9209 9209 if (vhc->vhc_path_discovery_postboot > 0) {
9210 9210 vhc->vhc_path_discovery_postboot--;
9211 9211 goto out;
9212 9212 }
9213 9213 }
9214 9214
9215 9215 /*
9216 9216 * Do full path discovery at most once per mdi_path_discovery_interval.
9217 9217 * This is to avoid a series of full path discoveries when opening
9218 9218 * stale /dev/[r]dsk links.
9219 9219 */
9220 9220 if (mdi_path_discovery_interval != -1 &&
9221 9221 ddi_get_lbolt64() >= vhc->vhc_path_discovery_cutoff_time)
9222 9222 goto out;
9223 9223
9224 9224 rv = 0;
9225 9225 out:
9226 9226 mutex_exit(&vhc->vhc_lock);
9227 9227 return (rv);
9228 9228 }
9229 9229
9230 9230 /*
9231 9231 * Discover all paths:
9232 9232 *
9233 9233 * Attach phci driver instances and then drive BUS_CONFIG_ALL on all the phci
9234 9234 * driver instances. During this process all paths will be discovered.
9235 9235 */
9236 9236 static int
9237 9237 vhcache_discover_paths(mdi_vhci_t *vh)
9238 9238 {
9239 9239 mdi_vhci_config_t *vhc = vh->vh_config;
9240 9240 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
9241 9241 int rv = 0;
9242 9242
9243 9243 single_threaded_vhconfig_enter(vhc);
9244 9244
9245 9245 if (vhcache_do_discovery(vhc)) {
9246 9246 attach_phci_drivers(vh->vh_class);
9247 9247 bus_config_all_phcis(vhcache, NDI_DRV_CONF_REPROBE |
9248 9248 NDI_NO_EVENT, BUS_CONFIG_ALL, DDI_MAJOR_T_NONE);
9249 9249
9250 9250 mutex_enter(&vhc->vhc_lock);
9251 9251 vhc->vhc_path_discovery_cutoff_time = ddi_get_lbolt64() +
9252 9252 mdi_path_discovery_interval * TICKS_PER_SECOND;
9253 9253 mutex_exit(&vhc->vhc_lock);
9254 9254 rv = 1;
9255 9255 }
9256 9256
9257 9257 single_threaded_vhconfig_exit(vhc);
9258 9258 return (rv);
9259 9259 }
9260 9260
9261 9261 /*
9262 9262 * Generic vhci bus config implementation:
9263 9263 *
9264 9264 * Parameters
9265 9265 * vdip vhci dip
9266 9266 * flags bus config flags
9267 9267 * op bus config operation
9268 9268 * The remaining parameters are bus config operation specific
9269 9269 *
9270 9270 * for BUS_CONFIG_ONE
9271 9271 * arg pointer to name@addr
9272 9272 * child upon successful return from this function, *child will be
9273 9273 * set to the configured and held devinfo child node of vdip.
9274 9274 * ct_addr pointer to client address (i.e. GUID)
9275 9275 *
9276 9276 * for BUS_CONFIG_DRIVER
9277 9277 * arg major number of the driver
9278 9278 * child and ct_addr parameters are ignored
9279 9279 *
9280 9280 * for BUS_CONFIG_ALL
9281 9281 * arg, child, and ct_addr parameters are ignored
9282 9282 *
9283 9283 * Note that for the rest of the bus config operations, this function simply
9284 9284 * calls the framework provided default bus config routine.
9285 9285 */
9286 9286 int
9287 9287 mdi_vhci_bus_config(dev_info_t *vdip, uint_t flags, ddi_bus_config_op_t op,
9288 9288 void *arg, dev_info_t **child, char *ct_addr)
9289 9289 {
9290 9290 mdi_vhci_t *vh = i_devi_get_vhci(vdip);
9291 9291 mdi_vhci_config_t *vhc = vh->vh_config;
9292 9292 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
9293 9293 int rv = 0;
9294 9294 int params_valid = 0;
9295 9295 char *cp;
9296 9296
9297 9297 /*
9298 9298 * To bus config vhcis we relay operation, possibly using another
9299 9299 * thread, to phcis. The phci driver then interacts with MDI to cause
9300 9300 * vhci child nodes to be enumerated under the vhci node. Adding a
9301 9301 * vhci child requires an ndi_devi_enter of the vhci. Since another
9302 9302 * thread may be adding the child, to avoid deadlock we can't wait
9303 9303 * for the relayed operations to complete if we have already entered
9304 9304 * the vhci node.
9305 9305 */
9306 9306 if (DEVI_BUSY_OWNED(vdip)) {
9307 9307 MDI_DEBUG(2, (MDI_NOTE, vdip,
9308 9308 "vhci dip is busy owned %p", (void *)vdip));
9309 9309 goto default_bus_config;
9310 9310 }
9311 9311
9312 9312 rw_enter(&vhcache->vhcache_lock, RW_READER);
9313 9313 if (!(vhcache->vhcache_flags & MDI_VHCI_CACHE_SETUP_DONE)) {
9314 9314 rw_exit(&vhcache->vhcache_lock);
9315 9315 rv = build_vhci_cache(vh);
9316 9316 rw_enter(&vhcache->vhcache_lock, RW_READER);
9317 9317 }
9318 9318
9319 9319 switch (op) {
9320 9320 case BUS_CONFIG_ONE:
9321 9321 if (arg != NULL && ct_addr != NULL) {
9322 9322 /* extract node name */
9323 9323 cp = (char *)arg;
9324 9324 while (*cp != '\0' && *cp != '@')
9325 9325 cp++;
9326 9326 if (*cp == '@') {
9327 9327 params_valid = 1;
9328 9328 *cp = '\0';
9329 9329 config_client_paths(vhc, (char *)arg, ct_addr);
9330 9330 /* config_client_paths() releases cache_lock */
9331 9331 *cp = '@';
9332 9332 break;
9333 9333 }
9334 9334 }
9335 9335
9336 9336 rw_exit(&vhcache->vhcache_lock);
9337 9337 break;
9338 9338
9339 9339 case BUS_CONFIG_DRIVER:
9340 9340 rw_exit(&vhcache->vhcache_lock);
9341 9341 if (rv == 0)
9342 9342 st_bus_config_all_phcis(vhc, flags, op,
9343 9343 (major_t)(uintptr_t)arg);
9344 9344 break;
9345 9345
9346 9346 case BUS_CONFIG_ALL:
9347 9347 rw_exit(&vhcache->vhcache_lock);
9348 9348 if (rv == 0)
9349 9349 st_bus_config_all_phcis(vhc, flags, op, -1);
9350 9350 break;
9351 9351
9352 9352 default:
9353 9353 rw_exit(&vhcache->vhcache_lock);
9354 9354 break;
9355 9355 }
9356 9356
9357 9357
9358 9358 default_bus_config:
9359 9359 /*
9360 9360 * All requested child nodes are enumerated under the vhci.
9361 9361 * Now configure them.
9362 9362 */
9363 9363 if (ndi_busop_bus_config(vdip, flags, op, arg, child, 0) ==
9364 9364 NDI_SUCCESS) {
9365 9365 return (MDI_SUCCESS);
9366 9366 } else if (op == BUS_CONFIG_ONE && rv == 0 && params_valid) {
9367 9367 /* discover all paths and try configuring again */
9368 9368 if (vhcache_discover_paths(vh) &&
9369 9369 ndi_busop_bus_config(vdip, flags, op, arg, child, 0) ==
9370 9370 NDI_SUCCESS)
9371 9371 return (MDI_SUCCESS);
9372 9372 }
9373 9373
9374 9374 return (MDI_FAILURE);
9375 9375 }
9376 9376
9377 9377 /*
9378 9378 * Read the on-disk vhci cache into an nvlist for the specified vhci class.
9379 9379 */
9380 9380 static nvlist_t *
9381 9381 read_on_disk_vhci_cache(char *vhci_class)
9382 9382 {
9383 9383 nvlist_t *nvl;
9384 9384 int err;
9385 9385 char *filename;
9386 9386
9387 9387 filename = vhclass2vhcache_filename(vhci_class);
9388 9388
9389 9389 if ((err = fread_nvlist(filename, &nvl)) == 0) {
9390 9390 kmem_free(filename, strlen(filename) + 1);
9391 9391 return (nvl);
9392 9392 } else if (err == EIO)
9393 9393 cmn_err(CE_WARN, "%s: I/O error, will recreate", filename);
9394 9394 else if (err == EINVAL)
9395 9395 cmn_err(CE_WARN,
9396 9396 "%s: data file corrupted, will recreate", filename);
9397 9397
9398 9398 kmem_free(filename, strlen(filename) + 1);
9399 9399 return (NULL);
9400 9400 }
9401 9401
9402 9402 /*
9403 9403 * Read on-disk vhci cache into nvlists for all vhci classes.
9404 9404 * Called during booting by i_ddi_read_devices_files().
9405 9405 */
9406 9406 void
9407 9407 mdi_read_devices_files(void)
9408 9408 {
9409 9409 int i;
9410 9410
9411 9411 for (i = 0; i < N_VHCI_CLASSES; i++)
9412 9412 vhcache_nvl[i] = read_on_disk_vhci_cache(vhci_class_list[i]);
9413 9413 }
9414 9414
9415 9415 /*
9416 9416 * Remove all stale entries from vhci cache.
9417 9417 */
9418 9418 static void
9419 9419 clean_vhcache(mdi_vhci_config_t *vhc)
9420 9420 {
9421 9421 mdi_vhci_cache_t *vhcache = &vhc->vhc_vhcache;
9422 9422 mdi_vhcache_phci_t *phci, *nxt_phci;
9423 9423 mdi_vhcache_client_t *client, *nxt_client;
9424 9424 mdi_vhcache_pathinfo_t *path, *nxt_path;
9425 9425
9426 9426 rw_enter(&vhcache->vhcache_lock, RW_WRITER);
9427 9427
9428 9428 client = vhcache->vhcache_client_head;
9429 9429 vhcache->vhcache_client_head = vhcache->vhcache_client_tail = NULL;
9430 9430 for ( ; client != NULL; client = nxt_client) {
9431 9431 nxt_client = client->cct_next;
9432 9432
9433 9433 path = client->cct_cpi_head;
9434 9434 client->cct_cpi_head = client->cct_cpi_tail = NULL;
9435 9435 for ( ; path != NULL; path = nxt_path) {
9436 9436 nxt_path = path->cpi_next;
9437 9437 if ((path->cpi_cphci->cphci_phci != NULL) &&
9438 9438 (path->cpi_pip != NULL)) {
9439 9439 enqueue_tail_vhcache_pathinfo(client, path);
9440 9440 } else if (path->cpi_pip != NULL) {
9441 9441 /* Not valid to have a path without a phci. */
9442 9442 free_vhcache_pathinfo(path);
9443 9443 }
9444 9444 }
9445 9445
9446 9446 if (client->cct_cpi_head != NULL)
9447 9447 enqueue_vhcache_client(vhcache, client);
9448 9448 else {
9449 9449 (void) mod_hash_destroy(vhcache->vhcache_client_hash,
9450 9450 (mod_hash_key_t)client->cct_name_addr);
9451 9451 free_vhcache_client(client);
9452 9452 }
9453 9453 }
9454 9454
9455 9455 phci = vhcache->vhcache_phci_head;
9456 9456 vhcache->vhcache_phci_head = vhcache->vhcache_phci_tail = NULL;
9457 9457 for ( ; phci != NULL; phci = nxt_phci) {
9458 9458
9459 9459 nxt_phci = phci->cphci_next;
9460 9460 if (phci->cphci_phci != NULL)
9461 9461 enqueue_vhcache_phci(vhcache, phci);
9462 9462 else
9463 9463 free_vhcache_phci(phci);
9464 9464 }
9465 9465
9466 9466 vhcache->vhcache_clean_time = ddi_get_lbolt64();
9467 9467 rw_exit(&vhcache->vhcache_lock);
9468 9468 vhcache_dirty(vhc);
9469 9469 }
9470 9470
9471 9471 /*
9472 9472 * Remove all stale entries from vhci cache.
9473 9473 * Called by i_ddi_clean_devices_files() during the execution of devfsadm -C
9474 9474 */
9475 9475 void
9476 9476 mdi_clean_vhcache(void)
9477 9477 {
9478 9478 mdi_vhci_t *vh;
9479 9479
9480 9480 mutex_enter(&mdi_mutex);
9481 9481 for (vh = mdi_vhci_head; vh != NULL; vh = vh->vh_next) {
9482 9482 vh->vh_refcnt++;
9483 9483 mutex_exit(&mdi_mutex);
9484 9484 clean_vhcache(vh->vh_config);
9485 9485 mutex_enter(&mdi_mutex);
9486 9486 vh->vh_refcnt--;
9487 9487 }
9488 9488 mutex_exit(&mdi_mutex);
9489 9489 }
9490 9490
9491 9491 /*
9492 9492 * mdi_vhci_walk_clients():
9493 9493 * Walker routine to traverse client dev_info nodes
9494 9494 * ddi_walk_devs(ddi_get_child(vdip), f, arg) returns the entire tree
9495 9495 * below the client, including nexus devices, which we dont want.
9496 9496 * So we just traverse the immediate siblings, starting from 1st client.
9497 9497 */
9498 9498 void
9499 9499 mdi_vhci_walk_clients(dev_info_t *vdip,
9500 9500 int (*f)(dev_info_t *, void *), void *arg)
9501 9501 {
9502 9502 mdi_vhci_t *vh = i_devi_get_vhci(vdip);
9503 9503 dev_info_t *cdip;
9504 9504 mdi_client_t *ct;
9505 9505
9506 9506 MDI_VHCI_CLIENT_LOCK(vh);
9507 9507 cdip = ddi_get_child(vdip);
9508 9508 while (cdip) {
9509 9509 ct = i_devi_get_client(cdip);
9510 9510 MDI_CLIENT_LOCK(ct);
9511 9511
9512 9512 if (((*f)(cdip, arg)) == DDI_WALK_CONTINUE)
9513 9513 cdip = ddi_get_next_sibling(cdip);
9514 9514 else
9515 9515 cdip = NULL;
9516 9516
9517 9517 MDI_CLIENT_UNLOCK(ct);
9518 9518 }
9519 9519 MDI_VHCI_CLIENT_UNLOCK(vh);
9520 9520 }
9521 9521
9522 9522 /*
9523 9523 * mdi_vhci_walk_phcis():
9524 9524 * Walker routine to traverse phci dev_info nodes
9525 9525 */
9526 9526 void
9527 9527 mdi_vhci_walk_phcis(dev_info_t *vdip,
9528 9528 int (*f)(dev_info_t *, void *), void *arg)
9529 9529 {
9530 9530 mdi_vhci_t *vh = i_devi_get_vhci(vdip);
9531 9531 mdi_phci_t *ph, *next;
9532 9532
9533 9533 MDI_VHCI_PHCI_LOCK(vh);
9534 9534 ph = vh->vh_phci_head;
9535 9535 while (ph) {
9536 9536 MDI_PHCI_LOCK(ph);
9537 9537
9538 9538 if (((*f)(ph->ph_dip, arg)) == DDI_WALK_CONTINUE)
9539 9539 next = ph->ph_next;
9540 9540 else
9541 9541 next = NULL;
9542 9542
9543 9543 MDI_PHCI_UNLOCK(ph);
9544 9544 ph = next;
9545 9545 }
9546 9546 MDI_VHCI_PHCI_UNLOCK(vh);
9547 9547 }
9548 9548
9549 9549
9550 9550 /*
9551 9551 * mdi_walk_vhcis():
9552 9552 * Walker routine to traverse vhci dev_info nodes
9553 9553 */
9554 9554 void
9555 9555 mdi_walk_vhcis(int (*f)(dev_info_t *, void *), void *arg)
9556 9556 {
9557 9557 mdi_vhci_t *vh = NULL;
9558 9558
9559 9559 mutex_enter(&mdi_mutex);
9560 9560 /*
9561 9561 * Scan for already registered vhci
9562 9562 */
9563 9563 for (vh = mdi_vhci_head; vh != NULL; vh = vh->vh_next) {
9564 9564 vh->vh_refcnt++;
9565 9565 mutex_exit(&mdi_mutex);
9566 9566 if (((*f)(vh->vh_dip, arg)) != DDI_WALK_CONTINUE) {
9567 9567 mutex_enter(&mdi_mutex);
9568 9568 vh->vh_refcnt--;
9569 9569 break;
9570 9570 } else {
9571 9571 mutex_enter(&mdi_mutex);
9572 9572 vh->vh_refcnt--;
9573 9573 }
9574 9574 }
9575 9575
9576 9576 mutex_exit(&mdi_mutex);
9577 9577 }
9578 9578
9579 9579 /*
9580 9580 * i_mdi_log_sysevent():
9581 9581 * Logs events for pickup by syseventd
9582 9582 */
9583 9583 static void
9584 9584 i_mdi_log_sysevent(dev_info_t *dip, char *ph_vh_class, char *subclass)
9585 9585 {
9586 9586 char *path_name;
9587 9587 nvlist_t *attr_list;
9588 9588
9589 9589 if (nvlist_alloc(&attr_list, NV_UNIQUE_NAME_TYPE,
9590 9590 KM_SLEEP) != DDI_SUCCESS) {
9591 9591 goto alloc_failed;
9592 9592 }
9593 9593
9594 9594 path_name = kmem_zalloc(MAXPATHLEN, KM_SLEEP);
9595 9595 (void) ddi_pathname(dip, path_name);
9596 9596
9597 9597 if (nvlist_add_string(attr_list, DDI_DRIVER_NAME,
9598 9598 ddi_driver_name(dip)) != DDI_SUCCESS) {
9599 9599 goto error;
9600 9600 }
9601 9601
9602 9602 if (nvlist_add_int32(attr_list, DDI_DRIVER_MAJOR,
9603 9603 (int32_t)ddi_driver_major(dip)) != DDI_SUCCESS) {
9604 9604 goto error;
9605 9605 }
9606 9606
9607 9607 if (nvlist_add_int32(attr_list, DDI_INSTANCE,
9608 9608 (int32_t)ddi_get_instance(dip)) != DDI_SUCCESS) {
9609 9609 goto error;
9610 9610 }
9611 9611
9612 9612 if (nvlist_add_string(attr_list, DDI_PATHNAME,
9613 9613 path_name) != DDI_SUCCESS) {
9614 9614 goto error;
9615 9615 }
9616 9616
9617 9617 if (nvlist_add_string(attr_list, DDI_CLASS,
9618 9618 ph_vh_class) != DDI_SUCCESS) {
9619 9619 goto error;
9620 9620 }
9621 9621
9622 9622 (void) ddi_log_sysevent(dip, DDI_VENDOR_SUNW, EC_DDI, subclass,
9623 9623 attr_list, NULL, DDI_SLEEP);
9624 9624
9625 9625 error:
9626 9626 kmem_free(path_name, MAXPATHLEN);
9627 9627 nvlist_free(attr_list);
9628 9628 return;
9629 9629
9630 9630 alloc_failed:
9631 9631 MDI_DEBUG(1, (MDI_WARN, dip, "!unable to send sysevent"));
9632 9632 }
9633 9633
9634 9634 char **
9635 9635 mdi_get_phci_driver_list(char *vhci_class, int *ndrivers)
9636 9636 {
9637 9637 char **driver_list, **ret_driver_list = NULL;
9638 9638 int *root_support_list;
9639 9639 int cur_elements, max_elements;
9640 9640
9641 9641 get_phci_driver_list(vhci_class, &driver_list, &root_support_list,
9642 9642 &cur_elements, &max_elements);
9643 9643
9644 9644
9645 9645 if (driver_list) {
9646 9646 kmem_free(root_support_list, sizeof (int) * max_elements);
9647 9647 ret_driver_list = mdi_realloc(driver_list, sizeof (char *)
9648 9648 * max_elements, sizeof (char *) * cur_elements);
9649 9649 }
9650 9650 *ndrivers = cur_elements;
9651 9651
9652 9652 return (ret_driver_list);
9653 9653
9654 9654 }
9655 9655
9656 9656 void
9657 9657 mdi_free_phci_driver_list(char **driver_list, int ndrivers)
9658 9658 {
9659 9659 char **p;
9660 9660 int i;
9661 9661
9662 9662 if (driver_list) {
9663 9663 for (i = 0, p = driver_list; i < ndrivers; i++, p++)
9664 9664 kmem_free(*p, strlen(*p) + 1);
9665 9665 kmem_free(driver_list, sizeof (char *) * ndrivers);
9666 9666 }
9667 9667 }
9668 9668
9669 9669 /*
9670 9670 * mdi_is_dev_supported():
9671 9671 * function called by pHCI bus config operation to determine if a
9672 9672 * device should be represented as a child of the vHCI or the
9673 9673 * pHCI. This decision is made by the vHCI, using cinfo idenity
9674 9674 * information passed by the pHCI - specifics of the cinfo
9675 9675 * representation are by agreement between the pHCI and vHCI.
9676 9676 * Return Values:
9677 9677 * MDI_SUCCESS
9678 9678 * MDI_FAILURE
9679 9679 */
9680 9680 int
9681 9681 mdi_is_dev_supported(char *class, dev_info_t *pdip, void *cinfo)
9682 9682 {
9683 9683 mdi_vhci_t *vh;
9684 9684
9685 9685 ASSERT(class && pdip);
9686 9686
9687 9687 /*
9688 9688 * For dev_supported, mdi_phci_register() must have established pdip as
9689 9689 * a pHCI.
9690 9690 *
9691 9691 * NOTE: mdi_phci_register() does "mpxio-disable" processing, and
9692 9692 * MDI_PHCI(pdip) will return false if mpxio is disabled.
9693 9693 */
9694 9694 if (!MDI_PHCI(pdip))
9695 9695 return (MDI_FAILURE);
9696 9696
9697 9697 /* Return MDI_FAILURE if vHCI does not support asking the question. */
9698 9698 vh = (mdi_vhci_t *)i_mdi_vhci_class2vhci(class);
9699 9699 if ((vh == NULL) || (vh->vh_ops->vo_is_dev_supported == NULL)) {
9700 9700 return (MDI_FAILURE);
9701 9701 }
9702 9702
9703 9703 /* Return vHCI answer */
9704 9704 return (vh->vh_ops->vo_is_dev_supported(vh->vh_dip, pdip, cinfo));
9705 9705 }
9706 9706
9707 9707 int
9708 9708 mdi_dc_return_dev_state(mdi_pathinfo_t *pip, struct devctl_iocdata *dcp)
9709 9709 {
9710 9710 uint_t devstate = 0;
9711 9711 dev_info_t *cdip;
9712 9712
9713 9713 if ((pip == NULL) || (dcp == NULL))
9714 9714 return (MDI_FAILURE);
9715 9715
9716 9716 cdip = mdi_pi_get_client(pip);
9717 9717
9718 9718 switch (mdi_pi_get_state(pip)) {
9719 9719 case MDI_PATHINFO_STATE_INIT:
9720 9720 devstate = DEVICE_DOWN;
9721 9721 break;
9722 9722 case MDI_PATHINFO_STATE_ONLINE:
9723 9723 devstate = DEVICE_ONLINE;
9724 9724 if ((cdip) && (devi_stillreferenced(cdip) == DEVI_REFERENCED))
9725 9725 devstate |= DEVICE_BUSY;
9726 9726 break;
9727 9727 case MDI_PATHINFO_STATE_STANDBY:
9728 9728 devstate = DEVICE_ONLINE;
9729 9729 break;
9730 9730 case MDI_PATHINFO_STATE_FAULT:
9731 9731 devstate = DEVICE_DOWN;
9732 9732 break;
9733 9733 case MDI_PATHINFO_STATE_OFFLINE:
9734 9734 devstate = DEVICE_OFFLINE;
9735 9735 break;
9736 9736 default:
9737 9737 ASSERT(MDI_PI(pip)->pi_state);
9738 9738 }
9739 9739
9740 9740 if (copyout(&devstate, dcp->cpyout_buf, sizeof (uint_t)) != 0)
9741 9741 return (MDI_FAILURE);
9742 9742
9743 9743 return (MDI_SUCCESS);
9744 9744 }
↓ open down ↓ |
720 lines elided |
↑ open up ↑ |
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX