1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 #include <sys/conf.h> 27 #include <sys/file.h> 28 #include <sys/ddi.h> 29 #include <sys/sunddi.h> 30 #include <sys/modctl.h> 31 #include <sys/scsi/scsi.h> 32 #include <sys/scsi/impl/scsi_reset_notify.h> 33 #include <sys/disp.h> 34 #include <sys/byteorder.h> 35 #include <sys/atomic.h> 36 37 #include "stmf.h" 38 #include "lpif.h" 39 #include "portif.h" 40 #include "stmf_ioctl.h" 41 #include "stmf_impl.h" 42 #include "lun_map.h" 43 #include "stmf_state.h" 44 45 void stmf_update_sessions_per_ve(stmf_view_entry_t *ve, 46 stmf_lu_t *lu, int action); 47 void stmf_add_lus_to_session_per_vemap(stmf_i_local_port_t *ilport, 48 stmf_i_scsi_session_t *iss, stmf_lun_map_t *vemap); 49 stmf_id_data_t *stmf_lookup_group_for_host(uint8_t *ident, uint16_t ident_size); 50 stmf_status_t stmf_add_ent_to_map(stmf_lun_map_t *sm, void *ent, uint8_t *lun); 51 stmf_status_t stmf_remove_ent_from_map(stmf_lun_map_t *sm, uint8_t *lun); 52 uint16_t stmf_get_next_free_lun(stmf_lun_map_t *sm, uint8_t *lun); 53 stmf_status_t stmf_add_tg(uint8_t *tg_name, uint16_t tg_name_size, 54 int allow_special, uint32_t *err_detail); 55 stmf_status_t stmf_add_hg(uint8_t *hg_name, uint16_t hg_name_size, 56 int allow_special, uint32_t *err_detail); 57 stmf_i_local_port_t *stmf_targetident_to_ilport(uint8_t *target_ident, 58 uint16_t ident_size); 59 stmf_i_scsi_session_t *stmf_lookup_session_for_hostident( 60 stmf_i_local_port_t *ilport, uint8_t *host_ident, 61 uint16_t ident_size); 62 stmf_i_lu_t *stmf_luident_to_ilu(uint8_t *lu_ident); 63 stmf_lun_map_t *stmf_get_ve_map_per_ids(stmf_id_data_t *tgid, 64 stmf_id_data_t *hgid); 65 stmf_lun_map_t *stmf_duplicate_ve_map(stmf_lun_map_t *src); 66 int stmf_merge_ve_map(stmf_lun_map_t *src, stmf_lun_map_t *dst, 67 stmf_lun_map_t **pp_ret_map, stmf_merge_flags_t mf); 68 void stmf_destroy_ve_map(stmf_lun_map_t *dst); 69 void stmf_free_id(stmf_id_data_t *id); 70 71 72 /* 73 * Init the view 74 */ 75 void 76 stmf_view_init() 77 { 78 uint8_t grpname_forall = '*'; 79 (void) stmf_add_hg(&grpname_forall, 1, 1, NULL); 80 (void) stmf_add_tg(&grpname_forall, 1, 1, NULL); 81 } 82 83 /* 84 * Clear config database here 85 */ 86 void 87 stmf_view_clear_config() 88 { 89 stmf_id_data_t *idgrp, *idgrp_next, *idmemb, *idmemb_next; 90 stmf_ver_tg_t *vtg, *vtg_next; 91 stmf_ver_hg_t *vhg, *vhg_next; 92 stmf_view_entry_t *ve, *ve_next; 93 stmf_i_lu_t *ilu; 94 stmf_id_list_t *idlist; 95 stmf_i_local_port_t *ilport; 96 97 for (vtg = stmf_state.stmf_ver_tg_head; vtg; vtg = vtg_next) { 98 for (vhg = vtg->vert_verh_list; vhg; vhg = vhg_next) { 99 if (vhg->verh_ve_map.lm_nentries) { 100 kmem_free(vhg->verh_ve_map.lm_plus, 101 vhg->verh_ve_map.lm_nentries * 102 sizeof (void *)); 103 } 104 vhg_next = vhg->verh_next; 105 kmem_free(vhg, sizeof (stmf_ver_hg_t)); 106 } 107 vtg_next = vtg->vert_next; 108 kmem_free(vtg, sizeof (stmf_ver_tg_t)); 109 } 110 stmf_state.stmf_ver_tg_head = NULL; 111 112 if (stmf_state.stmf_luid_list.id_count) { 113 /* clear the views for lus */ 114 for (idmemb = stmf_state.stmf_luid_list.idl_head; 115 idmemb; idmemb = idmemb_next) { 116 for (ve = (stmf_view_entry_t *)idmemb->id_impl_specific; 117 ve; ve = ve_next) { 118 ve_next = ve->ve_next; 119 ve->ve_hg->id_refcnt--; 120 ve->ve_tg->id_refcnt--; 121 kmem_free(ve, sizeof (stmf_view_entry_t)); 122 } 123 if (idmemb->id_pt_to_object) { 124 ilu = (stmf_i_lu_t *)(idmemb->id_pt_to_object); 125 ilu->ilu_luid = NULL; 126 } 127 idmemb_next = idmemb->id_next; 128 stmf_free_id(idmemb); 129 } 130 stmf_state.stmf_luid_list.id_count = 0; 131 stmf_state.stmf_luid_list.idl_head = 132 stmf_state.stmf_luid_list.idl_tail = NULL; 133 } 134 135 if (stmf_state.stmf_hg_list.id_count) { 136 /* free all the host group */ 137 for (idgrp = stmf_state.stmf_hg_list.idl_head; 138 idgrp; idgrp = idgrp_next) { 139 idlist = (stmf_id_list_t *)(idgrp->id_impl_specific); 140 if (idlist->id_count) { 141 for (idmemb = idlist->idl_head; idmemb; 142 idmemb = idmemb_next) { 143 idmemb_next = idmemb->id_next; 144 stmf_free_id(idmemb); 145 } 146 } 147 idgrp_next = idgrp->id_next; 148 stmf_free_id(idgrp); 149 } 150 stmf_state.stmf_hg_list.id_count = 0; 151 stmf_state.stmf_hg_list.idl_head = 152 stmf_state.stmf_hg_list.idl_tail = NULL; 153 } 154 if (stmf_state.stmf_tg_list.id_count) { 155 /* free all the target group */ 156 for (idgrp = stmf_state.stmf_tg_list.idl_head; 157 idgrp; idgrp = idgrp_next) { 158 idlist = (stmf_id_list_t *)(idgrp->id_impl_specific); 159 if (idlist->id_count) { 160 for (idmemb = idlist->idl_head; idmemb; 161 idmemb = idmemb_next) { 162 idmemb_next = idmemb->id_next; 163 stmf_free_id(idmemb); 164 } 165 } 166 idgrp_next = idgrp->id_next; 167 stmf_free_id(idgrp); 168 } 169 stmf_state.stmf_tg_list.id_count = 0; 170 stmf_state.stmf_tg_list.idl_head = 171 stmf_state.stmf_tg_list.idl_tail = NULL; 172 } 173 174 for (ilport = stmf_state.stmf_ilportlist; ilport; 175 ilport = ilport->ilport_next) { 176 ilport->ilport_tg = NULL; 177 } 178 } 179 180 /* 181 * Create luns map for session based on the view 182 */ 183 stmf_status_t 184 stmf_session_create_lun_map(stmf_i_local_port_t *ilport, 185 stmf_i_scsi_session_t *iss) 186 { 187 stmf_id_data_t *tg; 188 stmf_id_data_t *hg; 189 stmf_ver_tg_t *vertg; 190 char *phg_data, *ptg_data; 191 stmf_ver_hg_t *verhg; 192 stmf_lun_map_t *ve_map; 193 194 if (iss->iss_sm != NULL) 195 cmn_err(CE_PANIC, "create lun map called with non NULL map"); 196 iss->iss_sm = (stmf_lun_map_t *)kmem_zalloc(sizeof (stmf_lun_map_t), 197 KM_SLEEP); 198 mutex_enter(&stmf_state.stmf_lock); 199 tg = ilport->ilport_tg; 200 hg = stmf_lookup_group_for_host(iss->iss_ss->ss_rport_id->ident, 201 iss->iss_ss->ss_rport_id->ident_length); 202 iss->iss_hg = hg; 203 204 /* 205 * get the view entry map, 206 * take all host/target group into consideration 207 */ 208 ve_map = stmf_duplicate_ve_map(0); 209 for (vertg = stmf_state.stmf_ver_tg_head; vertg != NULL; 210 vertg = vertg->vert_next) { 211 ptg_data = (char *)vertg->vert_tg_ref->id_data; 212 if ((ptg_data[0] != '*') && (!tg || 213 ((tg->id_data[0] != '*') && 214 (vertg->vert_tg_ref != tg)))) { 215 continue; 216 } 217 for (verhg = vertg->vert_verh_list; verhg != NULL; 218 verhg = verhg->verh_next) { 219 phg_data = (char *)verhg->verh_hg_ref->id_data; 220 if ((phg_data[0] != '*') && (!hg || 221 ((hg->id_data[0] != '*') && 222 (verhg->verh_hg_ref != hg)))) { 223 continue; 224 } 225 (void) stmf_merge_ve_map(&verhg->verh_ve_map, ve_map, 226 &ve_map, 0); 227 } 228 } 229 230 231 if (ve_map->lm_nluns) { 232 stmf_add_lus_to_session_per_vemap(ilport, iss, ve_map); 233 } 234 /* not configured, cannot access any luns for now */ 235 236 mutex_exit(&stmf_state.stmf_lock); 237 stmf_destroy_ve_map(ve_map); 238 239 return (STMF_SUCCESS); 240 } 241 242 /* 243 * destroy lun map for session 244 */ 245 /* ARGSUSED */ 246 stmf_status_t 247 stmf_session_destroy_lun_map(stmf_i_local_port_t *ilport, 248 stmf_i_scsi_session_t *iss) 249 { 250 stmf_lun_map_t *sm; 251 stmf_i_lu_t *ilu; 252 uint16_t n; 253 stmf_lun_map_ent_t *ent; 254 255 /* 256 * to avoid conflict with updating session's map, 257 * which only grab stmf_lock 258 */ 259 mutex_enter(&stmf_state.stmf_lock); 260 sm = iss->iss_sm; 261 iss->iss_sm = NULL; 262 iss->iss_hg = NULL; 263 mutex_exit(&stmf_state.stmf_lock); 264 if (sm->lm_nentries) { 265 for (n = 0; n < sm->lm_nentries; n++) { 266 if ((ent = (stmf_lun_map_ent_t *)sm->lm_plus[n]) 267 != NULL) { 268 if (ent->ent_itl_datap) { 269 stmf_do_itl_dereg(ent->ent_lu, 270 ent->ent_itl_datap, 271 STMF_ITL_REASON_IT_NEXUS_LOSS); 272 } 273 ilu = (stmf_i_lu_t *) 274 ent->ent_lu->lu_stmf_private; 275 atomic_add_32(&ilu->ilu_ref_cnt, -1); 276 kmem_free(sm->lm_plus[n], 277 sizeof (stmf_lun_map_ent_t)); 278 } 279 } 280 kmem_free(sm->lm_plus, 281 sizeof (stmf_lun_map_ent_t *) * sm->lm_nentries); 282 } 283 284 kmem_free(sm, sizeof (*sm)); 285 return (STMF_SUCCESS); 286 } 287 288 /* 289 * Expects the session lock to be held. 290 */ 291 stmf_xfer_data_t * 292 stmf_session_prepare_report_lun_data(stmf_lun_map_t *sm) 293 { 294 stmf_xfer_data_t *xd; 295 uint16_t nluns, ent; 296 uint32_t alloc_size, data_size; 297 int i; 298 299 nluns = sm->lm_nluns; 300 301 data_size = 8 + (((uint32_t)nluns) << 3); 302 if (nluns == 0) { 303 data_size += 8; 304 } 305 alloc_size = data_size + sizeof (stmf_xfer_data_t) - 4; 306 307 xd = (stmf_xfer_data_t *)kmem_zalloc(alloc_size, KM_NOSLEEP); 308 309 if (xd == NULL) 310 return (NULL); 311 312 xd->alloc_size = alloc_size; 313 xd->size_left = data_size; 314 315 *((uint32_t *)xd->buf) = BE_32(data_size - 8); 316 if (nluns == 0) { 317 return (xd); 318 } 319 320 ent = 0; 321 322 for (i = 0; ((i < sm->lm_nentries) && (ent < nluns)); i++) { 323 if (sm->lm_plus[i] == NULL) 324 continue; 325 /* Fill in the entry */ 326 xd->buf[8 + (ent << 3) + 1] = (uchar_t)i; 327 xd->buf[8 + (ent << 3) + 0] = ((uchar_t)(i >> 8)); 328 ent++; 329 } 330 331 ASSERT(ent == nluns); 332 333 return (xd); 334 } 335 336 /* 337 * Add a lu to active sessions based on LUN inventory. 338 * Only invoked when the lu is onlined 339 */ 340 void 341 stmf_add_lu_to_active_sessions(stmf_lu_t *lu) 342 { 343 stmf_id_data_t *luid; 344 stmf_view_entry_t *ve; 345 stmf_i_lu_t *ilu; 346 347 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 348 ilu = (stmf_i_lu_t *)lu->lu_stmf_private; 349 ASSERT(ilu->ilu_state == STMF_STATE_ONLINE); 350 351 luid = ((stmf_i_lu_t *)lu->lu_stmf_private)->ilu_luid; 352 353 if (!luid) { 354 /* we did not configure view for this lun, so just return */ 355 return; 356 } 357 358 for (ve = (stmf_view_entry_t *)luid->id_impl_specific; 359 ve; ve = ve->ve_next) { 360 stmf_update_sessions_per_ve(ve, lu, 1); 361 } 362 } 363 /* 364 * Unmap a lun from all sessions 365 */ 366 void 367 stmf_session_lu_unmapall(stmf_lu_t *lu) 368 { 369 stmf_i_lu_t *ilu; 370 stmf_id_data_t *luid; 371 stmf_view_entry_t *ve; 372 373 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 374 375 ilu = (stmf_i_lu_t *)lu->lu_stmf_private; 376 377 if (ilu->ilu_ref_cnt == 0) 378 return; 379 380 luid = ((stmf_i_lu_t *)lu->lu_stmf_private)->ilu_luid; 381 if (!luid) { 382 /* 383 * we did not configure view for this lun, this should be 384 * an error 385 */ 386 return; 387 } 388 389 for (ve = (stmf_view_entry_t *)luid->id_impl_specific; 390 ve; ve = ve->ve_next) { 391 stmf_update_sessions_per_ve(ve, lu, 0); 392 if (ilu->ilu_ref_cnt == 0) 393 break; 394 } 395 } 396 /* 397 * add lu to a session, stmf_lock is already held 398 */ 399 stmf_status_t 400 stmf_add_lu_to_session(stmf_i_local_port_t *ilport, 401 stmf_i_scsi_session_t *iss, 402 stmf_lu_t *lu, 403 uint8_t *lu_nbr) 404 { 405 stmf_lun_map_t *sm = iss->iss_sm; 406 stmf_status_t ret; 407 stmf_i_lu_t *ilu = (stmf_i_lu_t *)lu->lu_stmf_private; 408 stmf_lun_map_ent_t *lun_map_ent; 409 uint32_t new_flags = 0; 410 uint16_t luNbr = 411 ((uint16_t)lu_nbr[1] | (((uint16_t)(lu_nbr[0] & 0x3F)) << 8)); 412 413 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 414 ASSERT(!stmf_get_ent_from_map(sm, luNbr)); 415 416 if ((sm->lm_nluns == 0) && 417 ((iss->iss_flags & ISS_BEING_CREATED) == 0)) { 418 new_flags = ISS_GOT_INITIAL_LUNS; 419 atomic_or_32(&ilport->ilport_flags, ILPORT_SS_GOT_INITIAL_LUNS); 420 stmf_state.stmf_process_initial_luns = 1; 421 } 422 423 lun_map_ent = (stmf_lun_map_ent_t *) 424 kmem_zalloc(sizeof (stmf_lun_map_ent_t), KM_SLEEP); 425 lun_map_ent->ent_lu = lu; 426 ret = stmf_add_ent_to_map(sm, (void *)lun_map_ent, lu_nbr); 427 ASSERT(ret == STMF_SUCCESS); 428 atomic_add_32(&ilu->ilu_ref_cnt, 1); 429 new_flags |= ISS_LUN_INVENTORY_CHANGED; 430 atomic_or_32(&iss->iss_flags, new_flags); 431 return (STMF_SUCCESS); 432 } 433 434 /* 435 * remvoe lu from a session, stmf_lock is already held 436 */ 437 /* ARGSUSED */ 438 stmf_status_t 439 stmf_remove_lu_from_session(stmf_i_local_port_t *ilport, 440 stmf_i_scsi_session_t *iss, 441 stmf_lu_t *lu, 442 uint8_t *lu_nbr) 443 { 444 stmf_status_t ret; 445 stmf_i_lu_t *ilu; 446 stmf_lun_map_t *sm = iss->iss_sm; 447 stmf_lun_map_ent_t *lun_map_ent; 448 uint16_t luNbr = 449 ((uint16_t)lu_nbr[1] | (((uint16_t)(lu_nbr[0] & 0x3F)) << 8)); 450 451 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 452 lun_map_ent = stmf_get_ent_from_map(sm, luNbr); 453 ASSERT(lun_map_ent && lun_map_ent->ent_lu == lu); 454 455 ilu = (stmf_i_lu_t *)lu->lu_stmf_private; 456 457 ret = stmf_remove_ent_from_map(sm, lu_nbr); 458 ASSERT(ret == STMF_SUCCESS); 459 atomic_add_32(&ilu->ilu_ref_cnt, -1); 460 iss->iss_flags |= ISS_LUN_INVENTORY_CHANGED; 461 if (lun_map_ent->ent_itl_datap) { 462 stmf_do_itl_dereg(lu, lun_map_ent->ent_itl_datap, 463 STMF_ITL_REASON_USER_REQUEST); 464 } 465 kmem_free((void *)lun_map_ent, sizeof (stmf_lun_map_ent_t)); 466 return (STMF_SUCCESS); 467 } 468 469 /* 470 * add or remove lu from all related sessions based on view entry, 471 * action is 0 for delete, 1 for add 472 */ 473 void 474 stmf_update_sessions_per_ve(stmf_view_entry_t *ve, 475 stmf_lu_t *lu, int action) 476 { 477 stmf_i_lu_t *ilu_tmp; 478 stmf_lu_t *lu_to_add; 479 stmf_i_local_port_t *ilport; 480 stmf_i_scsi_session_t *iss; 481 stmf_id_list_t *hostlist; 482 stmf_id_list_t *targetlist; 483 int all_hg = 0, all_tg = 0; 484 485 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 486 487 if (!lu) { 488 ilu_tmp = (stmf_i_lu_t *)ve->ve_luid->id_pt_to_object; 489 if (!ilu_tmp) 490 return; 491 lu_to_add = ilu_tmp->ilu_lu; 492 } else { 493 lu_to_add = lu; 494 ilu_tmp = (stmf_i_lu_t *)lu->lu_stmf_private; 495 } 496 497 if (ve->ve_hg->id_data[0] == '*') 498 all_hg = 1; 499 if (ve->ve_tg->id_data[0] == '*') 500 all_tg = 1; 501 hostlist = (stmf_id_list_t *)ve->ve_hg->id_impl_specific; 502 targetlist = (stmf_id_list_t *)ve->ve_tg->id_impl_specific; 503 504 if ((!all_hg && !hostlist->idl_head) || 505 (!all_tg && !targetlist->idl_head)) 506 /* No sessions to be updated */ 507 return; 508 509 for (ilport = stmf_state.stmf_ilportlist; ilport != NULL; 510 ilport = ilport->ilport_next) { 511 if (!all_tg && ilport->ilport_tg != ve->ve_tg) 512 continue; 513 /* This ilport belongs to the target group */ 514 rw_enter(&ilport->ilport_lock, RW_WRITER); 515 for (iss = ilport->ilport_ss_list; iss != NULL; 516 iss = iss->iss_next) { 517 if (!all_hg && iss->iss_hg != ve->ve_hg) 518 continue; 519 /* This host belongs to the host group */ 520 if (action == 0) { /* to remove */ 521 (void) stmf_remove_lu_from_session(ilport, iss, 522 lu_to_add, ve->ve_lun); 523 if (ilu_tmp->ilu_ref_cnt == 0) { 524 rw_exit(&ilport->ilport_lock); 525 return; 526 } 527 } else { 528 (void) stmf_add_lu_to_session(ilport, iss, 529 lu_to_add, ve->ve_lun); 530 } 531 } 532 rw_exit(&ilport->ilport_lock); 533 } 534 } 535 536 /* 537 * add luns in view entry map to a session, 538 * and stmf_lock is already held 539 */ 540 void 541 stmf_add_lus_to_session_per_vemap(stmf_i_local_port_t *ilport, 542 stmf_i_scsi_session_t *iss, 543 stmf_lun_map_t *vemap) 544 { 545 stmf_lu_t *lu; 546 stmf_i_lu_t *ilu; 547 stmf_view_entry_t *ve; 548 uint32_t i; 549 550 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 551 552 for (i = 0; i < vemap->lm_nentries; i++) { 553 ve = (stmf_view_entry_t *)vemap->lm_plus[i]; 554 if (!ve) 555 continue; 556 ilu = (stmf_i_lu_t *)ve->ve_luid->id_pt_to_object; 557 if (ilu && ilu->ilu_state == STMF_STATE_ONLINE) { 558 lu = ilu->ilu_lu; 559 (void) stmf_add_lu_to_session(ilport, iss, lu, 560 ve->ve_lun); 561 } 562 } 563 } 564 /* remove luns in view entry map from a session */ 565 void 566 stmf_remove_lus_from_session_per_vemap(stmf_i_local_port_t *ilport, 567 stmf_i_scsi_session_t *iss, 568 stmf_lun_map_t *vemap) 569 { 570 stmf_lu_t *lu; 571 stmf_i_lu_t *ilu; 572 stmf_view_entry_t *ve; 573 uint32_t i; 574 575 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 576 577 for (i = 0; i < vemap->lm_nentries; i++) { 578 ve = (stmf_view_entry_t *)vemap->lm_plus[i]; 579 if (!ve) 580 continue; 581 ilu = (stmf_i_lu_t *)ve->ve_luid->id_pt_to_object; 582 if (ilu && ilu->ilu_state == STMF_STATE_ONLINE) { 583 lu = ilu->ilu_lu; 584 (void) stmf_remove_lu_from_session(ilport, iss, lu, 585 ve->ve_lun); 586 } 587 } 588 } 589 590 stmf_id_data_t * 591 stmf_alloc_id(uint16_t id_size, uint16_t type, uint8_t *id_data, 592 uint32_t additional_size) 593 { 594 stmf_id_data_t *id; 595 int struct_size, total_size, real_id_size; 596 597 real_id_size = ((uint32_t)id_size + 7) & (~7); 598 struct_size = (sizeof (*id) + 7) & (~7); 599 total_size = ((additional_size + 7) & (~7)) + struct_size + 600 real_id_size; 601 id = (stmf_id_data_t *)kmem_zalloc(total_size, KM_SLEEP); 602 id->id_type = type; 603 id->id_data_size = id_size; 604 id->id_data = ((uint8_t *)id) + struct_size; 605 id->id_total_alloc_size = total_size; 606 if (additional_size) { 607 id->id_impl_specific = ((uint8_t *)id) + struct_size + 608 real_id_size; 609 } 610 bcopy(id_data, id->id_data, id_size); 611 612 return (id); 613 } 614 615 void 616 stmf_free_id(stmf_id_data_t *id) 617 { 618 kmem_free(id, id->id_total_alloc_size); 619 } 620 621 622 stmf_id_data_t * 623 stmf_lookup_id(stmf_id_list_t *idlist, uint16_t id_size, uint8_t *data) 624 { 625 stmf_id_data_t *id; 626 627 for (id = idlist->idl_head; id != NULL; id = id->id_next) { 628 if ((id->id_data_size == id_size) && 629 (bcmp(id->id_data, data, id_size) == 0)) { 630 return (id); 631 } 632 } 633 634 return (NULL); 635 } 636 /* Return the target group which a target belong to */ 637 stmf_id_data_t * 638 stmf_lookup_group_for_target(uint8_t *ident, uint16_t ident_size) 639 { 640 stmf_id_data_t *tgid; 641 stmf_id_data_t *target; 642 643 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 644 645 for (tgid = stmf_state.stmf_tg_list.idl_head; tgid; 646 tgid = tgid->id_next) { 647 target = stmf_lookup_id( 648 (stmf_id_list_t *)tgid->id_impl_specific, 649 ident_size, ident); 650 if (target) 651 return (tgid); 652 } 653 return (NULL); 654 } 655 /* Return the host group which a host belong to */ 656 stmf_id_data_t * 657 stmf_lookup_group_for_host(uint8_t *ident, uint16_t ident_size) 658 { 659 stmf_id_data_t *hgid; 660 stmf_id_data_t *host; 661 662 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 663 664 for (hgid = stmf_state.stmf_hg_list.idl_head; hgid; 665 hgid = hgid->id_next) { 666 host = stmf_lookup_id( 667 (stmf_id_list_t *)hgid->id_impl_specific, 668 ident_size, ident); 669 if (host) 670 return (hgid); 671 } 672 return (NULL); 673 } 674 675 void 676 stmf_append_id(stmf_id_list_t *idlist, stmf_id_data_t *id) 677 { 678 id->id_next = NULL; 679 680 if ((id->id_prev = idlist->idl_tail) == NULL) { 681 idlist->idl_head = idlist->idl_tail = id; 682 } else { 683 idlist->idl_tail->id_next = id; 684 idlist->idl_tail = id; 685 } 686 atomic_add_32(&idlist->id_count, 1); 687 } 688 689 void 690 stmf_remove_id(stmf_id_list_t *idlist, stmf_id_data_t *id) 691 { 692 if (id->id_next) { 693 id->id_next->id_prev = id->id_prev; 694 } else { 695 idlist->idl_tail = id->id_prev; 696 } 697 698 if (id->id_prev) { 699 id->id_prev->id_next = id->id_next; 700 } else { 701 idlist->idl_head = id->id_next; 702 } 703 atomic_add_32(&idlist->id_count, -1); 704 } 705 706 707 /* 708 * The refcnts of objects in a view entry are updated when then entry 709 * is successfully added. ve_map is just another representation of the 710 * view enrtries in a LU. Duplicating or merging a ve map does not 711 * affect any refcnts. 712 */ 713 stmf_lun_map_t * 714 stmf_duplicate_ve_map(stmf_lun_map_t *src) 715 { 716 stmf_lun_map_t *dst; 717 int i; 718 719 dst = (stmf_lun_map_t *)kmem_zalloc(sizeof (*dst), KM_SLEEP); 720 721 if (src == NULL) 722 return (dst); 723 724 if (src->lm_nentries) { 725 dst->lm_plus = kmem_zalloc(dst->lm_nentries * 726 sizeof (void *), KM_SLEEP); 727 for (i = 0; i < dst->lm_nentries; i++) { 728 dst->lm_plus[i] = src->lm_plus[i]; 729 } 730 } 731 732 return (dst); 733 } 734 735 void 736 stmf_destroy_ve_map(stmf_lun_map_t *dst) 737 { 738 if (dst->lm_nentries) { 739 kmem_free(dst->lm_plus, dst->lm_nentries * sizeof (void *)); 740 } 741 kmem_free(dst, sizeof (*dst)); 742 } 743 744 int 745 stmf_merge_ve_map(stmf_lun_map_t *src, stmf_lun_map_t *dst, 746 stmf_lun_map_t **pp_ret_map, stmf_merge_flags_t mf) 747 { 748 int i; 749 int nentries; 750 int to_create_space = 0; 751 752 if (dst == NULL) { 753 *pp_ret_map = stmf_duplicate_ve_map(src); 754 return (1); 755 } 756 757 if (src == NULL || src->lm_nluns == 0) { 758 if (mf & MERGE_FLAG_RETURN_NEW_MAP) 759 *pp_ret_map = stmf_duplicate_ve_map(dst); 760 else 761 *pp_ret_map = dst; 762 return (1); 763 } 764 765 if (mf & MERGE_FLAG_RETURN_NEW_MAP) { 766 *pp_ret_map = stmf_duplicate_ve_map(NULL); 767 nentries = max(dst->lm_nentries, src->lm_nentries); 768 to_create_space = 1; 769 } else { 770 *pp_ret_map = dst; 771 /* If there is not enough space in dst map */ 772 if (dst->lm_nentries < src->lm_nentries) { 773 nentries = src->lm_nentries; 774 to_create_space = 1; 775 } 776 } 777 if (to_create_space) { 778 void **p; 779 p = (void **)kmem_zalloc(nentries * sizeof (void *), KM_SLEEP); 780 if (dst->lm_nentries) { 781 bcopy(dst->lm_plus, p, 782 dst->lm_nentries * sizeof (void *)); 783 } 784 if (mf & (MERGE_FLAG_RETURN_NEW_MAP == 0)) 785 kmem_free(dst->lm_plus, 786 dst->lm_nentries * sizeof (void *)); 787 (*pp_ret_map)->lm_plus = p; 788 (*pp_ret_map)->lm_nentries = nentries; 789 } 790 791 for (i = 0; i < src->lm_nentries; i++) { 792 if (src->lm_plus[i] == NULL) 793 continue; 794 if (dst->lm_plus[i] != NULL) { 795 if (mf & MERGE_FLAG_NO_DUPLICATE) { 796 if (mf & MERGE_FLAG_RETURN_NEW_MAP) { 797 stmf_destroy_ve_map(*pp_ret_map); 798 *pp_ret_map = NULL; 799 } 800 return (0); 801 } 802 } else { 803 dst->lm_plus[i] = src->lm_plus[i]; 804 dst->lm_nluns++; 805 } 806 } 807 808 return (1); 809 } 810 811 /* 812 * add host group, id_impl_specific point to a list of hosts, 813 * on return, if error happened, err_detail may be assigned if 814 * the pointer is not NULL 815 */ 816 stmf_status_t 817 stmf_add_hg(uint8_t *hg_name, uint16_t hg_name_size, 818 int allow_special, uint32_t *err_detail) 819 { 820 stmf_id_data_t *id; 821 822 if (!allow_special) { 823 if (hg_name[0] == '*') 824 return (STMF_INVALID_ARG); 825 } 826 827 if (stmf_lookup_id(&stmf_state.stmf_hg_list, 828 hg_name_size, (uint8_t *)hg_name)) { 829 if (err_detail) 830 *err_detail = STMF_IOCERR_HG_EXISTS; 831 return (STMF_ALREADY); 832 } 833 id = stmf_alloc_id(hg_name_size, STMF_ID_TYPE_HOST_GROUP, 834 (uint8_t *)hg_name, sizeof (stmf_id_list_t)); 835 stmf_append_id(&stmf_state.stmf_hg_list, id); 836 837 return (STMF_SUCCESS); 838 } 839 840 /* add target group */ 841 stmf_status_t 842 stmf_add_tg(uint8_t *tg_name, uint16_t tg_name_size, 843 int allow_special, uint32_t *err_detail) 844 { 845 stmf_id_data_t *id; 846 847 if (!allow_special) { 848 if (tg_name[0] == '*') 849 return (STMF_INVALID_ARG); 850 } 851 852 853 if (stmf_lookup_id(&stmf_state.stmf_tg_list, tg_name_size, 854 (uint8_t *)tg_name)) { 855 if (err_detail) 856 *err_detail = STMF_IOCERR_TG_EXISTS; 857 return (STMF_ALREADY); 858 } 859 id = stmf_alloc_id(tg_name_size, STMF_ID_TYPE_TARGET_GROUP, 860 (uint8_t *)tg_name, sizeof (stmf_id_list_t)); 861 stmf_append_id(&stmf_state.stmf_tg_list, id); 862 863 return (STMF_SUCCESS); 864 } 865 866 /* 867 * insert view entry into list for a luid, if ve->ve_id is 0xffffffff, 868 * pick up a smallest available veid for it, and return the veid in ve->ve_id. 869 * The view entries list is sorted based on veid. 870 */ 871 stmf_status_t 872 stmf_add_ve_to_luid(stmf_id_data_t *luid, stmf_view_entry_t *ve) 873 { 874 stmf_view_entry_t *ve_tmp = NULL; 875 stmf_view_entry_t *ve_prev = NULL; 876 877 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 878 879 ve_tmp = (stmf_view_entry_t *)luid->id_impl_specific; 880 881 if (ve->ve_id != 0xffffffff) { 882 for (; ve_tmp; ve_tmp = ve_tmp->ve_next) { 883 if (ve_tmp->ve_id > ve->ve_id) { 884 break; 885 } else if (ve_tmp->ve_id == ve->ve_id) { 886 return (STMF_ALREADY); 887 } 888 ve_prev = ve_tmp; 889 } 890 } else { 891 uint32_t veid = 0; 892 /* search the smallest available veid */ 893 for (; ve_tmp; ve_tmp = ve_tmp->ve_next) { 894 ASSERT(ve_tmp->ve_id >= veid); 895 if (ve_tmp->ve_id != veid) 896 break; 897 veid++; 898 if (veid == 0xffffffff) 899 return (STMF_NOT_SUPPORTED); 900 ve_prev = ve_tmp; 901 } 902 ve->ve_id = veid; 903 } 904 905 /* insert before ve_tmp if it exist */ 906 ve->ve_next = ve_tmp; 907 ve->ve_prev = ve_prev; 908 if (ve_tmp) { 909 ve_tmp->ve_prev = ve; 910 } 911 if (ve_prev) { 912 ve_prev->ve_next = ve; 913 } else { 914 luid->id_impl_specific = (void *)ve; 915 } 916 return (STMF_SUCCESS); 917 } 918 919 /* stmf_lock is already held, err_detail may be assigned if error happens */ 920 stmf_status_t 921 stmf_add_view_entry(stmf_id_data_t *hg, stmf_id_data_t *tg, 922 uint8_t *lu_guid, uint32_t *ve_id, uint8_t *lun, 923 stmf_view_entry_t **conflicting, uint32_t *err_detail) 924 { 925 stmf_id_data_t *luid; 926 stmf_view_entry_t *ve; 927 char *phg, *ptg; 928 stmf_lun_map_t *ve_map = NULL; 929 stmf_ver_hg_t *verhg = NULL, *verhg_ex = NULL; 930 stmf_ver_tg_t *vertg = NULL, *vertg_ex = NULL; 931 char luid_new; 932 uint16_t lun_num; 933 stmf_i_lu_t *ilu; 934 stmf_status_t ret; 935 936 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 937 938 lun_num = ((uint16_t)lun[1] | (((uint16_t)(lun[0] & 0x3F)) << 8)); 939 940 luid = stmf_lookup_id(&stmf_state.stmf_luid_list, 16, lu_guid); 941 if (luid == NULL) { 942 luid = stmf_alloc_id(16, STMF_ID_TYPE_LU_GUID, lu_guid, 0); 943 ilu = stmf_luident_to_ilu(lu_guid); 944 if (ilu) { 945 ilu->ilu_luid = luid; 946 luid->id_pt_to_object = (void *)ilu; 947 } 948 luid_new = 1; 949 } else { 950 luid_new = 0; 951 ilu = (stmf_i_lu_t *)luid->id_pt_to_object; 952 } 953 954 /* The view entry won't be added if there is any confilict */ 955 phg = (char *)hg->id_data; ptg = (char *)tg->id_data; 956 for (ve = (stmf_view_entry_t *)luid->id_impl_specific; ve != NULL; 957 ve = ve->ve_next) { 958 if (((phg[0] == '*') || (ve->ve_hg->id_data[0] == '*') || 959 (hg == ve->ve_hg)) && ((ptg[0] == '*') || 960 (ve->ve_tg->id_data[0] == '*') || (tg == ve->ve_tg))) { 961 *conflicting = ve; 962 *err_detail = STMF_IOCERR_VIEW_ENTRY_CONFLICT; 963 ret = STMF_ALREADY; 964 goto add_ve_err_ret; 965 } 966 } 967 968 ve_map = stmf_duplicate_ve_map(0); 969 for (vertg = stmf_state.stmf_ver_tg_head; vertg != NULL; 970 vertg = vertg->vert_next) { 971 ptg = (char *)vertg->vert_tg_ref->id_data; 972 if ((ptg[0] != '*') && (tg->id_data[0] != '*') && 973 (vertg->vert_tg_ref != tg)) { 974 continue; 975 } 976 if (vertg->vert_tg_ref == tg) 977 vertg_ex = vertg; 978 for (verhg = vertg->vert_verh_list; verhg != NULL; 979 verhg = verhg->verh_next) { 980 phg = (char *)verhg->verh_hg_ref->id_data; 981 if ((phg[0] != '*') && (hg->id_data[0] != '*') && 982 (verhg->verh_hg_ref != hg)) { 983 continue; 984 } 985 if ((vertg_ex == vertg) && (verhg->verh_hg_ref == hg)) 986 verhg_ex = verhg; 987 (void) stmf_merge_ve_map(&verhg->verh_ve_map, ve_map, 988 &ve_map, 0); 989 } 990 } 991 992 if (lun[2] == 0xFF) { 993 /* Pick a LUN number */ 994 lun_num = stmf_get_next_free_lun(ve_map, lun); 995 if (lun_num > 0x3FFF) { 996 stmf_destroy_ve_map(ve_map); 997 ret = STMF_NOT_SUPPORTED; 998 goto add_ve_err_ret; 999 } 1000 } else { 1001 if ((*conflicting = stmf_get_ent_from_map(ve_map, lun_num)) 1002 != NULL) { 1003 stmf_destroy_ve_map(ve_map); 1004 *err_detail = STMF_IOCERR_LU_NUMBER_IN_USE; 1005 ret = STMF_LUN_TAKEN; 1006 goto add_ve_err_ret; 1007 } 1008 } 1009 stmf_destroy_ve_map(ve_map); 1010 1011 /* All is well, do the actual addition now */ 1012 ve = (stmf_view_entry_t *)kmem_zalloc(sizeof (*ve), KM_SLEEP); 1013 ve->ve_id = *ve_id; 1014 ve->ve_lun[0] = lun[0]; 1015 ve->ve_lun[1] = lun[1]; 1016 1017 if ((ret = stmf_add_ve_to_luid(luid, ve)) != STMF_SUCCESS) { 1018 kmem_free(ve, sizeof (stmf_view_entry_t)); 1019 goto add_ve_err_ret; 1020 } 1021 ve->ve_hg = hg; hg->id_refcnt++; 1022 ve->ve_tg = tg; tg->id_refcnt++; 1023 ve->ve_luid = luid; luid->id_refcnt++; 1024 1025 *ve_id = ve->ve_id; 1026 1027 if (luid_new) { 1028 stmf_append_id(&stmf_state.stmf_luid_list, luid); 1029 } 1030 1031 if (vertg_ex == NULL) { 1032 vertg_ex = (stmf_ver_tg_t *)kmem_zalloc(sizeof (stmf_ver_tg_t), 1033 KM_SLEEP); 1034 vertg_ex->vert_next = stmf_state.stmf_ver_tg_head; 1035 stmf_state.stmf_ver_tg_head = vertg_ex; 1036 vertg_ex->vert_tg_ref = tg; 1037 verhg_ex = vertg_ex->vert_verh_list = 1038 (stmf_ver_hg_t *)kmem_zalloc(sizeof (stmf_ver_hg_t), 1039 KM_SLEEP); 1040 verhg_ex->verh_hg_ref = hg; 1041 } 1042 if (verhg_ex == NULL) { 1043 verhg_ex = (stmf_ver_hg_t *)kmem_zalloc(sizeof (stmf_ver_hg_t), 1044 KM_SLEEP); 1045 verhg_ex->verh_next = vertg_ex->vert_verh_list; 1046 vertg_ex->vert_verh_list = verhg_ex; 1047 verhg_ex->verh_hg_ref = hg; 1048 } 1049 ret = stmf_add_ent_to_map(&verhg_ex->verh_ve_map, ve, ve->ve_lun); 1050 ASSERT(ret == STMF_SUCCESS); 1051 1052 /* we need to update the affected session */ 1053 if (stmf_state.stmf_service_running) { 1054 if (ilu && ilu->ilu_state == STMF_STATE_ONLINE) 1055 stmf_update_sessions_per_ve(ve, ilu->ilu_lu, 1); 1056 } 1057 1058 return (STMF_SUCCESS); 1059 add_ve_err_ret: 1060 if (luid_new) { 1061 if (ilu) 1062 ilu->ilu_luid = NULL; 1063 stmf_free_id(luid); 1064 } 1065 return (ret); 1066 } 1067 1068 stmf_status_t 1069 stmf_add_ent_to_map(stmf_lun_map_t *lm, void *ent, uint8_t *lun) 1070 { 1071 uint16_t n; 1072 if (((lun[0] & 0xc0) >> 6) != 0) 1073 return (STMF_FAILURE); 1074 1075 n = (uint16_t)lun[1] | (((uint16_t)(lun[0] & 0x3F)) << 8); 1076 try_again_to_add: 1077 if (lm->lm_nentries && (n < lm->lm_nentries)) { 1078 if (lm->lm_plus[n] == NULL) { 1079 lm->lm_plus[n] = ent; 1080 lm->lm_nluns++; 1081 return (STMF_SUCCESS); 1082 } else { 1083 return (STMF_LUN_TAKEN); 1084 } 1085 } else { 1086 void **pplu; 1087 uint16_t m = n + 1; 1088 m = ((m + 7) & ~7) & 0x7FFF; 1089 pplu = (void **)kmem_zalloc(m * sizeof (void *), KM_SLEEP); 1090 bcopy(lm->lm_plus, pplu, 1091 lm->lm_nentries * sizeof (void *)); 1092 kmem_free(lm->lm_plus, lm->lm_nentries * sizeof (void *)); 1093 lm->lm_plus = pplu; 1094 lm->lm_nentries = m; 1095 goto try_again_to_add; 1096 } 1097 } 1098 1099 1100 stmf_status_t 1101 stmf_remove_ent_from_map(stmf_lun_map_t *lm, uint8_t *lun) 1102 { 1103 uint16_t n, i; 1104 uint8_t lutype = (lun[0] & 0xc0) >> 6; 1105 if (lutype != 0) 1106 return (STMF_FAILURE); 1107 1108 n = (uint16_t)lun[1] | (((uint16_t)(lun[0] & 0x3F)) << 8); 1109 1110 if (n >= lm->lm_nentries) 1111 return (STMF_NOT_FOUND); 1112 if (lm->lm_plus[n] == NULL) 1113 return (STMF_NOT_FOUND); 1114 1115 lm->lm_plus[n] = NULL; 1116 lm->lm_nluns--; 1117 1118 for (i = 0; i < lm->lm_nentries; i++) { 1119 if (lm->lm_plus[lm->lm_nentries - 1 - i] != NULL) 1120 break; 1121 } 1122 i &= ~15; 1123 if (i >= 16) { 1124 void **pplu; 1125 uint16_t m; 1126 m = lm->lm_nentries - i; 1127 pplu = (void **)kmem_zalloc(m * sizeof (void *), KM_SLEEP); 1128 bcopy(lm->lm_plus, pplu, m * sizeof (void *)); 1129 kmem_free(lm->lm_plus, lm->lm_nentries * sizeof (void *)); 1130 lm->lm_plus = pplu; 1131 lm->lm_nentries = m; 1132 } 1133 1134 return (STMF_SUCCESS); 1135 } 1136 1137 uint16_t 1138 stmf_get_next_free_lun(stmf_lun_map_t *sm, uint8_t *lun) 1139 { 1140 uint16_t luNbr; 1141 1142 1143 if (sm->lm_nluns < 0x4000) { 1144 for (luNbr = 0; luNbr < sm->lm_nentries; luNbr++) { 1145 if (sm->lm_plus[luNbr] == NULL) 1146 break; 1147 } 1148 } else { 1149 return (0xFFFF); 1150 } 1151 if (lun) { 1152 bzero(lun, 8); 1153 lun[1] = luNbr & 0xff; 1154 lun[0] = (luNbr >> 8) & 0xff; 1155 } 1156 1157 return (luNbr); 1158 } 1159 1160 void * 1161 stmf_get_ent_from_map(stmf_lun_map_t *sm, uint16_t lun_num) 1162 { 1163 if ((lun_num & 0xC000) == 0) { 1164 if (sm->lm_nentries > lun_num) 1165 return (sm->lm_plus[lun_num & 0x3FFF]); 1166 else 1167 return (NULL); 1168 } 1169 1170 return (NULL); 1171 } 1172 1173 int 1174 stmf_add_ve(uint8_t *hgname, uint16_t hgname_size, 1175 uint8_t *tgname, uint16_t tgname_size, 1176 uint8_t *lu_guid, uint32_t *ve_id, 1177 uint8_t *luNbr, uint32_t *err_detail) 1178 { 1179 stmf_id_data_t *hg; 1180 stmf_id_data_t *tg; 1181 stmf_view_entry_t *conflictve; 1182 stmf_status_t ret; 1183 1184 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1185 1186 hg = stmf_lookup_id(&stmf_state.stmf_hg_list, hgname_size, 1187 (uint8_t *)hgname); 1188 if (!hg) { 1189 *err_detail = STMF_IOCERR_INVALID_HG; 1190 return (ENOENT); /* could not find group */ 1191 } 1192 tg = stmf_lookup_id(&stmf_state.stmf_tg_list, tgname_size, 1193 (uint8_t *)tgname); 1194 if (!tg) { 1195 *err_detail = STMF_IOCERR_INVALID_TG; 1196 return (ENOENT); /* could not find group */ 1197 } 1198 ret = stmf_add_view_entry(hg, tg, lu_guid, ve_id, luNbr, 1199 &conflictve, err_detail); 1200 1201 if (ret == STMF_ALREADY) { 1202 return (EALREADY); 1203 } else if (ret == STMF_LUN_TAKEN) { 1204 return (EEXIST); 1205 } else if (ret == STMF_NOT_SUPPORTED) { 1206 return (E2BIG); 1207 } else if (ret != STMF_SUCCESS) { 1208 return (EINVAL); 1209 } 1210 return (0); 1211 } 1212 1213 int 1214 stmf_remove_ve_by_id(uint8_t *guid, uint32_t veid, uint32_t *err_detail) 1215 { 1216 stmf_id_data_t *luid; 1217 stmf_view_entry_t *ve; 1218 stmf_ver_tg_t *vtg; 1219 stmf_ver_hg_t *vhg; 1220 stmf_ver_tg_t *prev_vtg = NULL; 1221 stmf_ver_hg_t *prev_vhg = NULL; 1222 int found = 0; 1223 stmf_i_lu_t *ilu; 1224 1225 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1226 luid = stmf_lookup_id(&stmf_state.stmf_luid_list, 16, guid); 1227 if (luid == NULL) { 1228 *err_detail = STMF_IOCERR_INVALID_LU_ID; 1229 return (ENODEV); 1230 } 1231 ilu = (stmf_i_lu_t *)luid->id_pt_to_object; 1232 1233 for (ve = (stmf_view_entry_t *)luid->id_impl_specific; 1234 ve; ve = ve->ve_next) { 1235 if (ve->ve_id == veid) { 1236 break; 1237 } 1238 } 1239 if (!ve) { 1240 *err_detail = STMF_IOCERR_INVALID_VE_ID; 1241 return (ENODEV); 1242 } 1243 /* remove the ve */ 1244 if (ve->ve_next) 1245 ve->ve_next->ve_prev = ve->ve_prev; 1246 if (ve->ve_prev) 1247 ve->ve_prev->ve_next = ve->ve_next; 1248 else { 1249 luid->id_impl_specific = (void *)ve->ve_next; 1250 if (!luid->id_impl_specific) { 1251 /* don't have any view entries related to this lu */ 1252 stmf_remove_id(&stmf_state.stmf_luid_list, luid); 1253 if (ilu) 1254 ilu->ilu_luid = NULL; 1255 stmf_free_id(luid); 1256 } 1257 } 1258 1259 /* we need to update ver_hg->verh_ve_map */ 1260 for (vtg = stmf_state.stmf_ver_tg_head; vtg; vtg = vtg->vert_next) { 1261 if (vtg->vert_tg_ref == ve->ve_tg) { 1262 found = 1; 1263 break; 1264 } 1265 prev_vtg = vtg; 1266 } 1267 ASSERT(found); 1268 found = 0; 1269 for (vhg = vtg->vert_verh_list; vhg; vhg = vhg->verh_next) { 1270 if (vhg->verh_hg_ref == ve->ve_hg) { 1271 found = 1; 1272 break; 1273 } 1274 prev_vhg = vhg; 1275 } 1276 ASSERT(found); 1277 1278 (void) stmf_remove_ent_from_map(&vhg->verh_ve_map, ve->ve_lun); 1279 1280 /* free verhg if it don't have any ve entries related */ 1281 if (!vhg->verh_ve_map.lm_nluns) { 1282 /* we don't have any view entry related */ 1283 if (prev_vhg) 1284 prev_vhg->verh_next = vhg->verh_next; 1285 else 1286 vtg->vert_verh_list = vhg->verh_next; 1287 1288 /* Free entries in case the map still has memory */ 1289 if (vhg->verh_ve_map.lm_nentries) { 1290 kmem_free(vhg->verh_ve_map.lm_plus, 1291 vhg->verh_ve_map.lm_nentries * 1292 sizeof (void *)); 1293 } 1294 kmem_free(vhg, sizeof (stmf_ver_hg_t)); 1295 if (!vtg->vert_verh_list) { 1296 /* we don't have any ve related */ 1297 if (prev_vtg) 1298 prev_vtg->vert_next = vtg->vert_next; 1299 else 1300 stmf_state.stmf_ver_tg_head = vtg->vert_next; 1301 kmem_free(vtg, sizeof (stmf_ver_tg_t)); 1302 } 1303 } 1304 1305 if (stmf_state.stmf_service_running && ilu && 1306 ilu->ilu_state == STMF_STATE_ONLINE) { 1307 stmf_update_sessions_per_ve(ve, ilu->ilu_lu, 0); 1308 } 1309 1310 ve->ve_hg->id_refcnt--; 1311 ve->ve_tg->id_refcnt--; 1312 1313 kmem_free(ve, sizeof (stmf_view_entry_t)); 1314 return (0); 1315 } 1316 1317 int 1318 stmf_add_group(uint8_t *grpname, uint16_t grpname_size, 1319 stmf_id_type_t group_type, uint32_t *err_detail) 1320 { 1321 stmf_status_t status; 1322 1323 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1324 1325 if (group_type == STMF_ID_TYPE_HOST_GROUP) 1326 status = stmf_add_hg(grpname, grpname_size, 0, err_detail); 1327 else if (group_type == STMF_ID_TYPE_TARGET_GROUP) 1328 status = stmf_add_tg(grpname, grpname_size, 0, err_detail); 1329 else { 1330 return (EINVAL); 1331 } 1332 switch (status) { 1333 case STMF_SUCCESS: 1334 return (0); 1335 case STMF_INVALID_ARG: 1336 return (EINVAL); 1337 case STMF_ALREADY: 1338 return (EEXIST); 1339 default: 1340 return (EIO); 1341 } 1342 } 1343 1344 /* 1345 * Group can only be removed only when it does not have 1346 * any view entry related 1347 */ 1348 int 1349 stmf_remove_group(uint8_t *grpname, uint16_t grpname_size, 1350 stmf_id_type_t group_type, uint32_t *err_detail) 1351 { 1352 stmf_id_data_t *id; 1353 stmf_id_data_t *idmemb; 1354 stmf_id_list_t *grp_memblist; 1355 stmf_i_scsi_session_t *iss; 1356 stmf_i_local_port_t *ilport; 1357 1358 if (grpname[0] == '*') 1359 return (EINVAL); 1360 1361 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1362 1363 if (group_type == STMF_ID_TYPE_HOST_GROUP) 1364 id = stmf_lookup_id(&stmf_state.stmf_hg_list, 1365 grpname_size, grpname); 1366 else if (group_type == STMF_ID_TYPE_TARGET_GROUP) 1367 id = stmf_lookup_id(&stmf_state.stmf_tg_list, 1368 grpname_size, grpname); 1369 if (!id) { 1370 *err_detail = (group_type == STMF_ID_TYPE_HOST_GROUP)? 1371 STMF_IOCERR_INVALID_HG:STMF_IOCERR_INVALID_TG; 1372 return (ENODEV); /* no such grp */ 1373 } 1374 if (id->id_refcnt) { 1375 /* fail, still have viewentry related to it */ 1376 *err_detail = (group_type == STMF_ID_TYPE_HOST_GROUP)? 1377 STMF_IOCERR_HG_IN_USE:STMF_IOCERR_TG_IN_USE; 1378 return (EBUSY); 1379 } 1380 grp_memblist = (stmf_id_list_t *)id->id_impl_specific; 1381 while ((idmemb = grp_memblist->idl_head) != NULL) { 1382 stmf_remove_id(grp_memblist, idmemb); 1383 stmf_free_id(idmemb); 1384 } 1385 1386 ASSERT(!grp_memblist->id_count); 1387 if (id->id_type == STMF_ID_TYPE_TARGET_GROUP) { 1388 for (ilport = stmf_state.stmf_ilportlist; ilport; 1389 ilport = ilport->ilport_next) { 1390 if (ilport->ilport_tg == (void *)id) { 1391 ilport->ilport_tg = NULL; 1392 } 1393 } 1394 stmf_remove_id(&stmf_state.stmf_tg_list, id); 1395 } else { 1396 for (ilport = stmf_state.stmf_ilportlist; ilport; 1397 ilport = ilport->ilport_next) { 1398 for (iss = ilport->ilport_ss_list; iss; 1399 iss = iss->iss_next) { 1400 if (iss->iss_hg == (void *)id) 1401 iss->iss_hg = NULL; 1402 } 1403 } 1404 stmf_remove_id(&stmf_state.stmf_hg_list, id); 1405 } 1406 stmf_free_id(id); 1407 return (0); 1408 1409 } 1410 1411 int 1412 stmf_add_group_member(uint8_t *grpname, uint16_t grpname_size, 1413 uint8_t *entry_ident, uint16_t entry_size, 1414 stmf_id_type_t entry_type, uint32_t *err_detail) 1415 { 1416 stmf_id_data_t *id_grp, *id_alltgt; 1417 stmf_id_data_t *id_member; 1418 stmf_id_data_t *id_grp_tmp; 1419 stmf_i_scsi_session_t *iss; 1420 stmf_i_local_port_t *ilport; 1421 stmf_lun_map_t *vemap, *vemap_alltgt; 1422 uint8_t grpname_forall = '*'; 1423 1424 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1425 ASSERT(grpname[0] != '*'); 1426 1427 if (entry_type == STMF_ID_TYPE_HOST) { 1428 id_grp = stmf_lookup_id(&stmf_state.stmf_hg_list, 1429 grpname_size, grpname); 1430 id_grp_tmp = stmf_lookup_group_for_host(entry_ident, 1431 entry_size); 1432 } else { 1433 id_grp = stmf_lookup_id(&stmf_state.stmf_tg_list, 1434 grpname_size, grpname); 1435 id_grp_tmp = stmf_lookup_group_for_target(entry_ident, 1436 entry_size); 1437 } 1438 if (id_grp == NULL) { 1439 *err_detail = (entry_type == STMF_ID_TYPE_HOST)? 1440 STMF_IOCERR_INVALID_HG:STMF_IOCERR_INVALID_TG; 1441 return (ENODEV); /* not found */ 1442 } 1443 1444 /* Check whether this member already bound to a group */ 1445 if (id_grp_tmp) { 1446 if (id_grp_tmp != id_grp) { 1447 *err_detail = (entry_type == STMF_ID_TYPE_HOST)? 1448 STMF_IOCERR_HG_ENTRY_EXISTS: 1449 STMF_IOCERR_TG_ENTRY_EXISTS; 1450 return (EEXIST); /* already added into another grp */ 1451 } 1452 else 1453 return (0); 1454 } 1455 id_member = stmf_alloc_id(entry_size, entry_type, 1456 entry_ident, 0); 1457 stmf_append_id((stmf_id_list_t *)id_grp->id_impl_specific, id_member); 1458 1459 if (entry_type == STMF_ID_TYPE_TARGET) { 1460 ilport = stmf_targetident_to_ilport(entry_ident, entry_size); 1461 if (ilport) 1462 ilport->ilport_tg = (void *)id_grp; 1463 return (0); 1464 } 1465 /* For host group member, update the session if needed */ 1466 if (!stmf_state.stmf_service_running) 1467 return (0); 1468 /* Need to consider all target group + this host group */ 1469 id_alltgt = stmf_lookup_id(&stmf_state.stmf_tg_list, 1470 1, &grpname_forall); 1471 vemap_alltgt = stmf_get_ve_map_per_ids(id_alltgt, id_grp); 1472 1473 /* check whether there are sessions may be affected */ 1474 for (ilport = stmf_state.stmf_ilportlist; ilport; 1475 ilport = ilport->ilport_next) { 1476 if (ilport->ilport_state != STMF_STATE_ONLINE) 1477 continue; 1478 iss = stmf_lookup_session_for_hostident(ilport, 1479 entry_ident, entry_size); 1480 if (iss) { 1481 stmf_id_data_t *tgid; 1482 iss->iss_hg = (void *)id_grp; 1483 tgid = ilport->ilport_tg; 1484 if (tgid) { 1485 vemap = stmf_get_ve_map_per_ids(tgid, id_grp); 1486 if (vemap) 1487 stmf_add_lus_to_session_per_vemap( 1488 ilport, iss, vemap); 1489 } 1490 if (vemap_alltgt) 1491 stmf_add_lus_to_session_per_vemap(ilport, 1492 iss, vemap_alltgt); 1493 } 1494 } 1495 1496 return (0); 1497 } 1498 1499 int 1500 stmf_remove_group_member(uint8_t *grpname, uint16_t grpname_size, 1501 uint8_t *entry_ident, uint16_t entry_size, 1502 stmf_id_type_t entry_type, uint32_t *err_detail) 1503 { 1504 stmf_id_data_t *id_grp, *id_alltgt; 1505 stmf_id_data_t *id_member; 1506 stmf_lun_map_t *vemap, *vemap_alltgt; 1507 uint8_t grpname_forall = '*'; 1508 stmf_i_local_port_t *ilport; 1509 stmf_i_scsi_session_t *iss; 1510 1511 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1512 ASSERT(grpname[0] != '*'); 1513 1514 if (entry_type == STMF_ID_TYPE_HOST) { 1515 id_grp = stmf_lookup_id(&stmf_state.stmf_hg_list, 1516 grpname_size, grpname); 1517 } else { 1518 id_grp = stmf_lookup_id(&stmf_state.stmf_tg_list, 1519 grpname_size, grpname); 1520 } 1521 if (id_grp == NULL) { 1522 *err_detail = (entry_type == STMF_ID_TYPE_HOST)? 1523 STMF_IOCERR_INVALID_HG:STMF_IOCERR_INVALID_TG; 1524 return (ENODEV); /* no such group */ 1525 } 1526 id_member = stmf_lookup_id((stmf_id_list_t *)id_grp->id_impl_specific, 1527 entry_size, entry_ident); 1528 if (!id_member) { 1529 *err_detail = (entry_type == STMF_ID_TYPE_HOST)? 1530 STMF_IOCERR_INVALID_HG_ENTRY:STMF_IOCERR_INVALID_TG_ENTRY; 1531 return (ENODEV); /* no such member */ 1532 } 1533 /* verify target is offline */ 1534 if (entry_type == STMF_ID_TYPE_TARGET) { 1535 ilport = stmf_targetident_to_ilport(entry_ident, entry_size); 1536 if (ilport && ilport->ilport_state != STMF_STATE_OFFLINE) { 1537 *err_detail = STMF_IOCERR_TG_NEED_TG_OFFLINE; 1538 return (EBUSY); 1539 } 1540 } 1541 1542 stmf_remove_id((stmf_id_list_t *)id_grp->id_impl_specific, id_member); 1543 stmf_free_id(id_member); 1544 1545 if (entry_type == STMF_ID_TYPE_TARGET) { 1546 ilport = stmf_targetident_to_ilport(entry_ident, entry_size); 1547 if (ilport) 1548 ilport->ilport_tg = NULL; 1549 return (0); 1550 } 1551 /* For host group member, update the session */ 1552 if (!stmf_state.stmf_service_running) 1553 return (0); 1554 1555 /* Need to consider all target group + this host group */ 1556 id_alltgt = stmf_lookup_id(&stmf_state.stmf_tg_list, 1557 1, &grpname_forall); 1558 vemap_alltgt = stmf_get_ve_map_per_ids(id_alltgt, id_grp); 1559 1560 /* check if there are session related, if so, update it */ 1561 for (ilport = stmf_state.stmf_ilportlist; ilport; 1562 ilport = ilport->ilport_next) { 1563 if (ilport->ilport_state != STMF_STATE_ONLINE) 1564 continue; 1565 iss = stmf_lookup_session_for_hostident(ilport, 1566 entry_ident, entry_size); 1567 if (iss) { 1568 stmf_id_data_t *tgid; 1569 iss->iss_hg = NULL; 1570 tgid = ilport->ilport_tg; 1571 if (tgid) { 1572 vemap = stmf_get_ve_map_per_ids(tgid, id_grp); 1573 if (vemap) 1574 stmf_remove_lus_from_session_per_vemap( 1575 ilport, iss, vemap); 1576 } 1577 if (vemap_alltgt) 1578 stmf_remove_lus_from_session_per_vemap(ilport, 1579 iss, vemap_alltgt); 1580 } 1581 } 1582 1583 return (0); 1584 } 1585 1586 /* Assert stmf_lock is already held */ 1587 stmf_i_local_port_t * 1588 stmf_targetident_to_ilport(uint8_t *target_ident, uint16_t ident_size) 1589 { 1590 stmf_i_local_port_t *ilport; 1591 uint8_t *id; 1592 1593 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1594 1595 for (ilport = stmf_state.stmf_ilportlist; ilport; 1596 ilport = ilport->ilport_next) { 1597 id = (uint8_t *)ilport->ilport_lport->lport_id; 1598 if ((id[3] == ident_size) && 1599 bcmp(id + 4, target_ident, ident_size) == 0) { 1600 return (ilport); 1601 } 1602 } 1603 return (NULL); 1604 } 1605 1606 stmf_i_scsi_session_t * 1607 stmf_lookup_session_for_hostident(stmf_i_local_port_t *ilport, 1608 uint8_t *host_ident, uint16_t ident_size) 1609 { 1610 stmf_i_scsi_session_t *iss; 1611 uint8_t *id; 1612 1613 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1614 1615 for (iss = ilport->ilport_ss_list; iss; iss = iss->iss_next) { 1616 id = (uint8_t *)iss->iss_ss->ss_rport_id; 1617 if ((id[3] == ident_size) && 1618 bcmp(id + 4, host_ident, ident_size) == 0) { 1619 return (iss); 1620 } 1621 } 1622 return (NULL); 1623 } 1624 1625 stmf_i_lu_t * 1626 stmf_luident_to_ilu(uint8_t *lu_ident) 1627 { 1628 stmf_i_lu_t *ilu; 1629 1630 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1631 1632 for (ilu = stmf_state.stmf_ilulist; ilu; ilu = ilu->ilu_next) { 1633 if (bcmp(&ilu->ilu_lu->lu_id->ident[0], lu_ident, 16) == 0) 1634 return (ilu); 1635 } 1636 1637 return (NULL); 1638 } 1639 1640 /* 1641 * Assert stmf_lock is already held, 1642 * Just get the view map for the specific target group and host group 1643 * tgid and hgid can not be NULL 1644 */ 1645 stmf_lun_map_t * 1646 stmf_get_ve_map_per_ids(stmf_id_data_t *tgid, stmf_id_data_t *hgid) 1647 { 1648 int found = 0; 1649 stmf_ver_tg_t *vertg; 1650 stmf_ver_hg_t *verhg; 1651 1652 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1653 1654 for (vertg = stmf_state.stmf_ver_tg_head; 1655 vertg; vertg = vertg->vert_next) { 1656 if (vertg->vert_tg_ref == tgid) { 1657 found = 1; 1658 break; 1659 } 1660 } 1661 if (!found) 1662 return (NULL); 1663 1664 for (verhg = vertg->vert_verh_list; verhg; verhg = verhg->verh_next) { 1665 if (verhg->verh_hg_ref == hgid) { 1666 return (&verhg->verh_ve_map); 1667 } 1668 } 1669 return (NULL); 1670 } 1671 1672 stmf_status_t 1673 stmf_validate_lun_view_entry(stmf_id_data_t *hg, stmf_id_data_t *tg, 1674 uint8_t *lun, uint32_t *err_detail) 1675 { 1676 char *phg, *ptg; 1677 stmf_lun_map_t *ve_map = NULL; 1678 stmf_ver_hg_t *verhg = NULL; 1679 stmf_ver_tg_t *vertg = NULL; 1680 uint16_t lun_num; 1681 stmf_status_t ret = STMF_SUCCESS; 1682 1683 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1684 1685 ve_map = stmf_duplicate_ve_map(0); 1686 for (vertg = stmf_state.stmf_ver_tg_head; vertg != NULL; 1687 vertg = vertg->vert_next) { 1688 ptg = (char *)vertg->vert_tg_ref->id_data; 1689 if ((ptg[0] != '*') && (tg->id_data[0] != '*') && 1690 (vertg->vert_tg_ref != tg)) { 1691 continue; 1692 } 1693 for (verhg = vertg->vert_verh_list; verhg != NULL; 1694 verhg = verhg->verh_next) { 1695 phg = (char *)verhg->verh_hg_ref->id_data; 1696 if ((phg[0] != '*') && (hg->id_data[0] != '*') && 1697 (verhg->verh_hg_ref != hg)) { 1698 continue; 1699 } 1700 (void) stmf_merge_ve_map(&verhg->verh_ve_map, ve_map, 1701 &ve_map, 0); 1702 } 1703 } 1704 1705 ret = STMF_SUCCESS; 1706 /* Return an available lun number */ 1707 if (lun[2] == 0xFF) { 1708 /* Pick a LUN number */ 1709 lun_num = stmf_get_next_free_lun(ve_map, lun); 1710 if (lun_num > 0x3FFF) 1711 ret = STMF_NOT_SUPPORTED; 1712 } else { 1713 lun_num = (uint16_t)lun[1] | (((uint16_t)(lun[0] & 0x3F)) << 8); 1714 if (stmf_get_ent_from_map(ve_map, lun_num) != NULL) { 1715 *err_detail = STMF_IOCERR_LU_NUMBER_IN_USE; 1716 ret = STMF_LUN_TAKEN; 1717 } 1718 } 1719 stmf_destroy_ve_map(ve_map); 1720 1721 return (ret); 1722 } 1723 1724 int 1725 stmf_validate_lun_ve(uint8_t *hgname, uint16_t hgname_size, 1726 uint8_t *tgname, uint16_t tgname_size, 1727 uint8_t *luNbr, uint32_t *err_detail) 1728 { 1729 stmf_id_data_t *hg; 1730 stmf_id_data_t *tg; 1731 stmf_status_t ret; 1732 1733 ASSERT(mutex_owned(&stmf_state.stmf_lock)); 1734 1735 hg = stmf_lookup_id(&stmf_state.stmf_hg_list, hgname_size, 1736 (uint8_t *)hgname); 1737 if (!hg) { 1738 *err_detail = STMF_IOCERR_INVALID_HG; 1739 return (ENOENT); /* could not find group */ 1740 } 1741 tg = stmf_lookup_id(&stmf_state.stmf_tg_list, tgname_size, 1742 (uint8_t *)tgname); 1743 if (!tg) { 1744 *err_detail = STMF_IOCERR_INVALID_TG; 1745 return (ENOENT); /* could not find group */ 1746 } 1747 ret = stmf_validate_lun_view_entry(hg, tg, luNbr, err_detail); 1748 1749 if (ret == STMF_LUN_TAKEN) { 1750 return (EEXIST); 1751 } else if (ret == STMF_NOT_SUPPORTED) { 1752 return (E2BIG); 1753 } else if (ret != STMF_SUCCESS) { 1754 return (EINVAL); 1755 } 1756 return (0); 1757 } 1758