| 
     1 /*-
    2  * Copyright (c) 2017 Broadcom. All rights reserved.
    3  * The term "Broadcom" refers to Broadcom Limited and/or its subsidiaries.
    4  *
    5  * Redistribution and use in source and binary forms, with or without
    6  * modification, are permitted provided that the following conditions are met:
    7  *
    8  * 1. Redistributions of source code must retain the above copyright notice,
    9  *    this list of conditions and the following disclaimer.
   10  *
   11  * 2. Redistributions in binary form must reproduce the above copyright notice,
   12  *    this list of conditions and the following disclaimer in the documentation
   13  *    and/or other materials provided with the distribution.
   14  *
   15  * 3. Neither the name of the copyright holder nor the names of its contributors
   16  *    may be used to endorse or promote products derived from this software
   17  *    without specific prior written permission.
   18  *
   19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
   20  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
   21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
   22  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
   23  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
   24  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
   25  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
   26  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
   27  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
   28  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
   29  * POSSIBILITY OF SUCH DAMAGE.
   30  *
   31  * $FreeBSD$
   32  */
   33 
   34 /**
   35  * @file
   36  * Handles the domain object callback from the HW.
   37  */
   38 
   39 /*!
   40 @defgroup domain_sm Domain State Machine: States
   41 */
   42 
   43 #include "ocs.h"
   44 
   45 #include "ocs_fabric.h"
   46 #include "ocs_device.h"
   47 
   48 #define domain_sm_trace(domain)  \
   49         do { \
   50                 if (OCS_LOG_ENABLE_DOMAIN_SM_TRACE(domain->ocs)) \
   51                         ocs_log_info(domain->ocs, "[domain] %-20s %-20s\n", __func__, ocs_sm_event_name(evt)); \
   52         } while (0)
   53 
   54 #define domain_trace(domain, fmt, ...) \
   55         do { \
   56                 if (OCS_LOG_ENABLE_DOMAIN_SM_TRACE(domain ? domain->ocs : NULL)) \
   57                         ocs_log_info(domain ? domain->ocs : NULL, fmt, ##__VA_ARGS__); \
   58         } while (0)
   59 
   60 #define domain_printf(domain, fmt, ...) \
   61         do { \
   62                 ocs_log_info(domain ? domain->ocs : NULL, fmt, ##__VA_ARGS__); \
   63         } while (0)
   64 
   65 void ocs_mgmt_domain_list(ocs_textbuf_t *textbuf, void *domain);
   66 void ocs_mgmt_domain_get_all(ocs_textbuf_t *textbuf, void *domain);
   67 int ocs_mgmt_domain_get(ocs_textbuf_t *textbuf, char *parent, char *name, void *domain);
   68 int ocs_mgmt_domain_set(char *parent, char *name, char *value, void *domain);
   69 int ocs_mgmt_domain_exec(char *parent, char *action, void *arg_in, uint32_t arg_in_length,
   70                 void *arg_out, uint32_t arg_out_length, void *domain);
   71 
   72 static ocs_mgmt_functions_t domain_mgmt_functions = {
   73         .get_list_handler = ocs_mgmt_domain_list,
   74         .get_handler = ocs_mgmt_domain_get,
   75         .get_all_handler = ocs_mgmt_domain_get_all,
   76         .set_handler = ocs_mgmt_domain_set,
   77         .exec_handler = ocs_mgmt_domain_exec,
   78 };
   79 
   80 /**
   81  * @brief Accept domain callback events from the HW.
   82  *
   83  * <h3 class="desc">Description</h3>
   84  * HW calls this function with various domain-related events.
   85  *
   86  * @param arg Application-specified argument.
   87  * @param event Domain event.
   88  * @param data Event specific data.
   89  *
   90  * @return Returns 0 on success; or a negative error value on failure.
   91  */
   92 
   93 int32_t
   94 ocs_domain_cb(void *arg, ocs_hw_domain_event_e event, void *data)
   95 {
   96         ocs_t *ocs = arg;
   97         ocs_domain_t *domain = NULL;
   98         int32_t rc = 0;
   99 
  100         ocs_assert(data, -1);
  101 
  102         if (event != OCS_HW_DOMAIN_FOUND) {
  103                 domain = data;
  104         }
  105 
  106         switch (event) {
  107         case OCS_HW_DOMAIN_FOUND: {
  108                 uint64_t fcf_wwn = 0;
  109                 ocs_domain_record_t *drec = data;
  110                 ocs_assert(drec, -1);
  111 
  112                 /* extract the fcf_wwn */
  113                 fcf_wwn = ocs_be64toh(*((uint64_t*)drec->wwn));
  114 
  115                 /* lookup domain, or allocate a new one if one doesn't exist already */
  116                 domain = ocs_domain_find(ocs, fcf_wwn);
  117                 if (domain == NULL) {
  118                         domain = ocs_domain_alloc(ocs, fcf_wwn);
  119                         if (domain == NULL) {
  120                                 ocs_log_err(ocs, "ocs_domain_alloc() failed\n");
  121                                 rc = -1;
  122                                 break;
  123                         }
  124                         ocs_sm_transition(&domain->drvsm, __ocs_domain_init, NULL);
  125                 }
  126                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_FOUND, drec);
  127                 break;
  128         }
  129 
  130         case OCS_HW_DOMAIN_LOST:
  131                 domain_trace(domain, "OCS_HW_DOMAIN_LOST:\n");
  132                 ocs_domain_hold_frames(domain);
  133                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_LOST, NULL);
  134                 break;
  135 
  136         case OCS_HW_DOMAIN_ALLOC_OK: {
  137                 domain_trace(domain, "OCS_HW_DOMAIN_ALLOC_OK:\n");
  138                 domain->instance_index = 0;
  139                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_ALLOC_OK, NULL);
  140                 break;
  141         }
  142 
  143         case OCS_HW_DOMAIN_ALLOC_FAIL:
  144                 domain_trace(domain, "OCS_HW_DOMAIN_ALLOC_FAIL:\n");
  145                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_ALLOC_FAIL, NULL);
  146                 break;
  147 
  148         case OCS_HW_DOMAIN_ATTACH_OK:
  149                 domain_trace(domain, "OCS_HW_DOMAIN_ATTACH_OK:\n");
  150                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_ATTACH_OK, NULL);
  151                 break;
  152 
  153         case OCS_HW_DOMAIN_ATTACH_FAIL:
  154                 domain_trace(domain, "OCS_HW_DOMAIN_ATTACH_FAIL:\n");
  155                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_ATTACH_FAIL, NULL);
  156                 break;
  157 
  158         case OCS_HW_DOMAIN_FREE_OK:
  159                 domain_trace(domain, "OCS_HW_DOMAIN_FREE_OK:\n");
  160                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_FREE_OK, NULL);
  161                 break;
  162 
  163         case OCS_HW_DOMAIN_FREE_FAIL:
  164                 domain_trace(domain, "OCS_HW_DOMAIN_FREE_FAIL:\n");
  165                 ocs_domain_post_event(domain, OCS_EVT_DOMAIN_FREE_FAIL, NULL);
  166                 break;
  167 
  168         default:
  169                 ocs_log_warn(ocs, "unsupported event %#x\n", event);
  170         }
  171 
  172         return rc;
  173 }
  174 
  175 /**
  176  * @brief Find the domain, given its FCF_WWN.
  177  *
  178  * <h3 class="desc">Description</h3>
  179  * Search the domain_list to find a matching domain object.
  180  *
  181  * @param ocs Pointer to the OCS device.
  182  * @param fcf_wwn FCF WWN to find.
  183  *
  184  * @return Returns the pointer to the domain if found; or NULL otherwise.
  185  */
  186 
  187 ocs_domain_t *
  188 ocs_domain_find(ocs_t *ocs, uint64_t fcf_wwn)
  189 {
  190         ocs_domain_t *domain = NULL;
  191 
  192         /* Check to see if this domain is already allocated */
  193         ocs_device_lock(ocs);
  194                 ocs_list_foreach(&ocs->domain_list, domain) {
  195                         if (fcf_wwn == domain->fcf_wwn) {
  196                                 break;
  197                         }
  198                 }
  199         ocs_device_unlock(ocs);
  200         return domain;
  201 }
  202 
  203 /**
  204  * @brief Allocate a domain object.
  205  *
  206  * <h3 class="desc">Description</h3>
  207  * A domain object is allocated and initialized. It is associated with the
  208  * \c ocs argument.
  209  *
  210  * @param ocs Pointer to the OCS device.
  211  * @param fcf_wwn FCF WWN of the domain.
  212  *
  213  * @return Returns a pointer to the ocs_domain_t object; or NULL.
  214  */
  215 
  216 ocs_domain_t *
  217 ocs_domain_alloc(ocs_t *ocs, uint64_t fcf_wwn)
  218 {
  219         ocs_domain_t *domain;
  220 
  221         ocs_assert(ocs, NULL);
  222 
  223         domain = ocs_malloc(ocs, sizeof(*domain), OCS_M_NOWAIT | OCS_M_ZERO);
  224         if (domain) {
  225                 domain->ocs = ocs;
  226                 domain->instance_index = ocs->domain_instance_count++;
  227                 domain->drvsm.app = domain;
  228                 ocs_domain_lock_init(domain);
  229                 ocs_lock_init(ocs, &domain->lookup_lock, "Domain lookup[%d]", domain->instance_index);
  230 
  231                 /* Allocate a sparse vector for sport FC_ID's */
  232                 domain->lookup = spv_new(ocs);
  233                 if (domain->lookup == NULL) {
  234                         ocs_log_err(ocs, "spv_new() failed\n");
  235                         ocs_free(ocs, domain, sizeof(*domain));
  236                         return NULL;
  237                 }
  238 
  239                 ocs_list_init(&domain->sport_list, ocs_sport_t, link);
  240                 domain->fcf_wwn = fcf_wwn;
  241                 ocs_log_debug(ocs, "Domain allocated: wwn %016" PRIX64 "\n", domain->fcf_wwn);
  242                 domain->femul_enable = (ocs->ctrlmask & OCS_CTRLMASK_ENABLE_FABRIC_EMULATION) != 0;
  243 
  244                 ocs_device_lock(ocs);
  245                         /* if this is the first domain, then assign it as the "root" domain */
  246                         if (ocs_list_empty(&ocs->domain_list)) {
  247                                 ocs->domain = domain;
  248                         }
  249                         ocs_list_add_tail(&ocs->domain_list, domain);
  250                 ocs_device_unlock(ocs);
  251 
  252                 domain->mgmt_functions = &domain_mgmt_functions;
  253         } else {
  254                 ocs_log_err(ocs, "domain allocation failed\n");
  255         }
  256 
  257         return domain;
  258 }
  259 
  260 /**
  261  * @brief Free a domain object.
  262  *
  263  * <h3 class="desc">Description</h3>
  264  * The domain object is freed.
  265  *
  266  * @param domain Domain object to free.
  267  *
  268  * @return None.
  269  */
  270 
  271 void
  272 ocs_domain_free(ocs_domain_t *domain)
  273 {
  274         ocs_t *ocs;
  275 
  276         ocs_assert(domain);
  277         ocs_assert(domain->ocs);
  278 
  279         /* Hold frames to clear the domain pointer from the xport lookup */
  280         ocs_domain_hold_frames(domain);
  281 
  282         ocs = domain->ocs;
  283 
  284         ocs_log_debug(ocs, "Domain free: wwn %016" PRIX64 "\n", domain->fcf_wwn);
  285 
  286         spv_del(domain->lookup);
  287         domain->lookup = NULL;
  288 
  289         ocs_device_lock(ocs);
  290                 ocs_list_remove(&ocs->domain_list, domain);
  291                 if (domain == ocs->domain) {
  292                         /* set global domain to the new head */
  293                         ocs->domain = ocs_list_get_head(&ocs->domain_list);
  294                         if (ocs->domain) {
  295                                 ocs_log_debug(ocs, "setting new domain, old=%p new=%p\n",
  296                                                 domain, ocs->domain);
  297                         }
  298                 }
  299 
  300                 if (ocs_list_empty(&ocs->domain_list) && ocs->domain_list_empty_cb ) {
  301                         (*ocs->domain_list_empty_cb)(ocs, ocs->domain_list_empty_cb_arg);
  302                 }
  303         ocs_device_unlock(ocs);
  304 
  305         ocs_lock_free(&domain->lookup_lock);
  306 
  307         ocs_free(ocs, domain, sizeof(*domain));
  308 }
  309 
  310 /**
  311  * @brief Free memory resources of a domain object.
  312  *
  313  * <h3 class="desc">Description</h3>
  314  * After the domain object is freed, its child objects are also freed.
  315  *
  316  * @param domain Pointer to a domain object.
  317  *
  318  * @return None.
  319  */
  320 
  321 void
  322 ocs_domain_force_free(ocs_domain_t *domain)
  323 {
  324         ocs_sport_t *sport;
  325         ocs_sport_t *next;
  326 
  327         /* Shutdown domain sm */
  328         ocs_sm_disable(&domain->drvsm);
  329 
  330         ocs_scsi_notify_domain_force_free(domain);
  331 
  332         ocs_domain_lock(domain);
  333                 ocs_list_foreach_safe(&domain->sport_list, sport, next) {
  334                         ocs_sport_force_free(sport);
  335                 }
  336         ocs_domain_unlock(domain);
  337         ocs_hw_domain_force_free(&domain->ocs->hw, domain);
  338         ocs_domain_free(domain);
  339 }
  340 
  341 /**
  342  * @brief Register a callback when the domain_list goes empty.
  343  *
  344  * <h3 class="desc">Description</h3>
  345  * A function callback may be registered when the domain_list goes empty.
  346  *
  347  * @param ocs Pointer to a device object.
  348  * @param callback Callback function.
  349  * @param arg Callback argument.
  350  *
  351  * @return None.
  352  */
  353 
  354 void
  355 ocs_register_domain_list_empty_cb(ocs_t *ocs, void (*callback)(ocs_t *ocs, void *arg), void *arg)
  356 {
  357         ocs_device_lock(ocs);
  358                 ocs->domain_list_empty_cb = callback;
  359                 ocs->domain_list_empty_cb_arg = arg;
  360                 if (ocs_list_empty(&ocs->domain_list) && callback) {
  361                         (*callback)(ocs, arg);
  362                 }
  363         ocs_device_unlock(ocs);
  364 }
  365 
  366 /**
  367  * @brief Return a pointer to the domain, given the instance index.
  368  *
  369  * <h3 class="desc">Description</h3>
  370  * A pointer to the domain context, given by the index, is returned.
  371  *
  372  * @param ocs Pointer to the driver instance context.
  373  * @param index Instance index.
  374  *
  375  * @return Returns a pointer to the domain; or NULL.
  376  */
  377 
  378 ocs_domain_t *
  379 ocs_domain_get_instance(ocs_t *ocs, uint32_t index)
  380 {
  381         ocs_domain_t *domain = NULL;
  382 
  383         if (index >= OCS_MAX_DOMAINS) {
  384                 ocs_log_err(ocs, "invalid index: %d\n", index);
  385                 return NULL;
  386         }
  387         ocs_device_lock(ocs);
  388                 ocs_list_foreach(&ocs->domain_list, domain) {
  389                         if (domain->instance_index == index) {
  390                                 break;
  391                         }
  392                 }
  393         ocs_device_unlock(ocs);
  394         return domain;
  395 }
  396 
  397 /**
  398  * @ingroup domain_sm
  399  * @brief Domain state machine: Common event handler.
  400  *
  401  * <h3 class="desc">Description</h3>
  402  * Common/shared events are handled here for the domain state machine.
  403  *
  404  * @param funcname Function name text.
  405  * @param ctx Domain state machine context.
  406  * @param evt Event to process.
  407  * @param arg Per event optional argument.
  408  *
  409  * @return Returns NULL.
  410  */
  411 
  412 static void *
  413 __ocs_domain_common(const char *funcname, ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  414 {
  415         ocs_domain_t *domain = ctx->app;
  416 
  417         switch(evt) {
  418         case OCS_EVT_ENTER:
  419         case OCS_EVT_REENTER:
  420         case OCS_EVT_EXIT:
  421         case OCS_EVT_ALL_CHILD_NODES_FREE:
  422                 /* this can arise if an FLOGI fails on the SPORT, and the SPORT is shutdown */
  423                 break;
  424         default:
  425                 ocs_log_warn(domain->ocs, "%-20s %-20s not handled\n", funcname, ocs_sm_event_name(evt));
  426                 break;
  427         }
  428 
  429         return NULL;
  430 }
  431 
  432 /**
  433  * @ingroup domain_sm
  434  * @brief Domain state machine: Common shutdown.
  435  *
  436  * <h3 class="desc">Description</h3>
  437  * Handles common shutdown events.
  438  *
  439  * @param funcname Function name text.
  440  * @param ctx Remote node state machine context.
  441  * @param evt Event to process.
  442  * @param arg Per event optional argument.
  443  *
  444  * @return Returns NULL.
  445  */
  446 
  447 static void *
  448 __ocs_domain_common_shutdown(const char *funcname, ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  449 {
  450         ocs_domain_t *domain = ctx->app;
  451 
  452         switch(evt) {
  453         case OCS_EVT_ENTER:
  454         case OCS_EVT_REENTER:
  455         case OCS_EVT_EXIT:
  456                 break;
  457         case OCS_EVT_DOMAIN_FOUND:
  458                 ocs_assert(arg, NULL);
  459                 /* sm: / save drec, mark domain_found_pending */
  460                 ocs_memcpy(&domain->pending_drec, arg, sizeof(domain->pending_drec));
  461                 domain->domain_found_pending = TRUE;
  462                 break;
  463         case OCS_EVT_DOMAIN_LOST: 
  464                 /* clear drec available
  465                  * sm: unmark domain_found_pending */
  466                 domain->domain_found_pending = FALSE;
  467                 break;
  468 
  469         default:
  470                 ocs_log_warn(domain->ocs, "%-20s %-20s not handled\n", funcname, ocs_sm_event_name(evt));
  471                 break;
  472         }
  473 
  474         return NULL;
  475 }
  476 
  477 #define std_domain_state_decl(...) \
  478         ocs_domain_t *domain = NULL; \
  479         ocs_t *ocs = NULL; \
  480         \
  481         ocs_assert(ctx, NULL); \
  482         ocs_assert(ctx->app, NULL); \
  483         domain = ctx->app; \
  484         ocs_assert(domain->ocs, NULL); \
  485         ocs = domain->ocs; \
  486         ocs_assert(ocs->xport, NULL);
  487 
  488 /**
  489  * @ingroup domain_sm
  490  * @brief Domain state machine: Initial state.
  491  *
  492  * <h3 class="desc">Description</h3>
  493  * The initial state for a domain. Each domain is initialized to
  494  * this state at start of day (SOD).
  495  *
  496  * @param ctx Domain state machine context.
  497  * @param evt Event to process.
  498  * @param arg Per event optional argument.
  499  *
  500  * @return Returns NULL.
  501  */
  502 
  503 void *
  504 __ocs_domain_init(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  505 {
  506         std_domain_state_decl();
  507 
  508         domain_sm_trace(domain);
  509 
  510         switch(evt) {
  511         case OCS_EVT_ENTER:
  512                 domain->attached = 0;
  513                 break;
  514 
  515         case OCS_EVT_DOMAIN_FOUND: {
  516                 int32_t         vlan = 0;
  517                 uint32_t        i;
  518                 ocs_domain_record_t *drec = arg;
  519                 ocs_sport_t *sport;
  520 
  521                 uint64_t        my_wwnn = ocs->xport->req_wwnn;
  522                 uint64_t        my_wwpn = ocs->xport->req_wwpn;
  523                 uint64_t        be_wwpn;
  524 
  525                 /* For now, user must specify both port name and node name, or we let firmware
  526                  * pick both (same as for vports).
  527                  * TODO: do we want to allow setting only port name or only node name?
  528                  */
  529                 if ((my_wwpn == 0) || (my_wwnn == 0)) {
  530                         ocs_log_debug(ocs, "using default hardware WWN configuration \n");
  531                         my_wwpn = ocs_get_wwn(&ocs->hw, OCS_HW_WWN_PORT);
  532                         my_wwnn = ocs_get_wwn(&ocs->hw, OCS_HW_WWN_NODE);
  533                 }
  534 
  535                 ocs_log_debug(ocs, "Creating base sport using WWPN %016" PRIx64 " WWNN %016" PRIx64 "\n",
  536                         my_wwpn, my_wwnn);
  537 
  538                 /* Allocate a sport and transition to __ocs_sport_allocated */
  539                 sport = ocs_sport_alloc(domain, my_wwpn, my_wwnn, UINT32_MAX, ocs->enable_ini, ocs->enable_tgt);
  540 
  541                 if (sport == NULL) {
  542                         ocs_log_err(ocs, "ocs_sport_alloc() failed\n");
  543                         break;
  544                 }
  545                 ocs_sm_transition(&sport->sm, __ocs_sport_allocated, NULL);
  546 
  547                 /* If domain is ethernet, then fetch the vlan id value */
  548                 if (drec->is_ethernet) {
  549                         vlan = ocs_bitmap_search((void *)drec->map.vlan, TRUE, 512 * 8);
  550                         if (vlan < 0) {
  551                                 ocs_log_err(ocs, "no VLAN id available (FCF=%d)\n",
  552                                                 drec->index);
  553                                 break;
  554                         }
  555                 }
  556 
  557                 be_wwpn = ocs_htobe64(sport->wwpn);
  558 
  559                 /* allocate ocs_sli_port_t object for local port
  560                  * Note: drec->fc_id is ALPA from read_topology only if loop
  561                  */
  562                 if (ocs_hw_port_alloc(&ocs->hw, sport, NULL, (uint8_t *)&be_wwpn)) {
  563                         ocs_log_err(ocs, "Can't allocate port\n");
  564                         ocs_sport_free(sport);
  565                         break;
  566                 }
  567 
  568                 /* initialize domain object */
  569                 domain->is_loop = drec->is_loop;
  570                 domain->is_fc = drec->is_fc;
  571 
  572                 /*
  573                  * If the loop position map includes ALPA == 0, then we are in a public loop (NL_PORT)
  574                  * Note that the first element of the loopmap[] contains the count of elements, and if
  575                  * ALPA == 0 is present, it will occupy the first location after the count.
  576                  */
  577                 domain->is_nlport = drec->map.loop[1] == 0x00;
  578 
  579                 if (domain->is_loop) {
  580                         ocs_log_debug(ocs, "%s fc_id=%#x speed=%d\n",
  581                                         drec->is_loop ? (domain->is_nlport ? "public-loop" : "loop") : "other",
  582                                         drec->fc_id, drec->speed);
  583 
  584                         sport->fc_id = drec->fc_id;
  585                         sport->topology = OCS_SPORT_TOPOLOGY_LOOP;
  586                         ocs_snprintf(sport->display_name, sizeof(sport->display_name), "s%06x", drec->fc_id);
  587 
  588                         if (ocs->enable_ini) {
  589                                 uint32_t count = drec->map.loop[0];
  590                                 ocs_log_debug(ocs, "%d position map entries\n", count);
  591                                 for (i = 1; i <= count; i++) {
  592                                         if (drec->map.loop[i] != drec->fc_id) {
  593                                                 ocs_node_t *node;
  594 
  595                                                 ocs_log_debug(ocs, "%#x -> %#x\n",
  596                                                                 drec->fc_id, drec->map.loop[i]);
  597                                                 node = ocs_node_alloc(sport, drec->map.loop[i], FALSE, TRUE);
  598                                                 if (node == NULL) {
  599                                                         ocs_log_err(ocs, "ocs_node_alloc() failed\n");
  600                                                         break;
  601                                                 }
  602                                                 ocs_node_transition(node, __ocs_d_wait_loop, NULL);
  603                                         }
  604                                 }
  605                         }
  606                 }
  607 
  608                 /* Initiate HW domain alloc */
  609                 if (ocs_hw_domain_alloc(&ocs->hw, domain, drec->index, vlan)) {
  610                         ocs_log_err(ocs, "Failed to initiate HW domain allocation\n");
  611                         break;
  612                 }
  613                 ocs_sm_transition(ctx, __ocs_domain_wait_alloc, arg);
  614                 break;
  615         }
  616         default:
  617                 __ocs_domain_common(__func__, ctx, evt, arg);
  618                 return NULL;
  619         }
  620 
  621         return NULL;
  622 }
  623 
  624 /**
  625  * @ingroup domain_sm
  626  * @brief Domain state machine: Wait for the domain allocation to complete.
  627  *
  628  * <h3 class="desc">Description</h3>
  629  * Waits for the domain state to be allocated. After the HW domain
  630  * allocation process has been initiated, this state waits for
  631  * that process to complete (i.e. a domain-alloc-ok event).
  632  *
  633  * @param ctx Domain state machine context.
  634  * @param evt Event to process.
  635  * @param arg Per event optional argument.
  636  *
  637  * @return Returns NULL.
  638  */
  639 
  640 void *
  641 __ocs_domain_wait_alloc(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  642 {
  643         ocs_sport_t *sport;
  644         std_domain_state_decl();
  645 
  646         domain_sm_trace(domain);
  647 
  648         switch(evt) {
  649         case OCS_EVT_DOMAIN_ALLOC_OK: {
  650                 char prop_buf[32];
  651                 uint64_t wwn_bump = 0;
  652                 fc_plogi_payload_t *sp;
  653 
  654                 if (ocs_get_property("wwn_bump", prop_buf, sizeof(prop_buf)) == 0) {
  655                         wwn_bump = ocs_strtoull(prop_buf, 0, 0);
  656                 }
  657 
  658                 sport = domain->sport;
  659                 ocs_assert(sport, NULL);
  660                 sp = (fc_plogi_payload_t*) sport->service_params;
  661 
  662                 /* Save the domain service parameters */
  663                 ocs_memcpy(domain->service_params + 4, domain->dma.virt, sizeof(fc_plogi_payload_t) - 4);
  664                 ocs_memcpy(sport->service_params + 4, domain->dma.virt, sizeof(fc_plogi_payload_t) - 4);
  665 
  666                 /* If we're in fabric emulation mode, the flogi service parameters have not been setup yet,
  667                  * so we need some reasonable BB credit value
  668                  */
  669                 if (domain->femul_enable) {
  670                         ocs_memcpy(domain->flogi_service_params + 4, domain->service_params + 4, sizeof(fc_plogi_payload_t) - 4);
  671                 }
  672 
  673                 /* Update the sport's service parameters, user might have specified non-default names */
  674                 sp->port_name_hi = ocs_htobe32((uint32_t) (sport->wwpn >> 32ll));
  675                 sp->port_name_lo = ocs_htobe32((uint32_t) sport->wwpn);
  676                 sp->node_name_hi = ocs_htobe32((uint32_t) (sport->wwnn >> 32ll));
  677                 sp->node_name_lo = ocs_htobe32((uint32_t) sport->wwnn);
  678 
  679                 if (wwn_bump) {
  680                         sp->port_name_lo = ocs_htobe32(ocs_be32toh(sp->port_name_lo) ^ ((uint32_t)(wwn_bump)));
  681                         sp->port_name_hi = ocs_htobe32(ocs_be32toh(sp->port_name_hi) ^ ((uint32_t)(wwn_bump >> 32)));
  682                         sp->node_name_lo = ocs_htobe32(ocs_be32toh(sp->node_name_lo) ^ ((uint32_t)(wwn_bump)));
  683                         sp->node_name_hi = ocs_htobe32(ocs_be32toh(sp->node_name_hi) ^ ((uint32_t)(wwn_bump >> 32)));
  684                         ocs_log_info(ocs, "Overriding WWN\n");
  685                 }
  686 
  687                 /* Take the loop topology path, unless we are an NL_PORT (public loop) */
  688                 if (domain->is_loop && !domain->is_nlport) {
  689                         /*
  690                          * For loop, we already have our FC ID and don't need fabric login.
  691                          * Transition to the allocated state and post an event to attach to
  692                          * the domain. Note that this breaks the normal action/transition
  693                          * pattern here to avoid a race with the domain attach callback.
  694                          */
  695                         /* sm: is_loop / domain_attach */
  696                         ocs_sm_transition(ctx, __ocs_domain_allocated, NULL);
  697                         __ocs_domain_attach_internal(domain, sport->fc_id);
  698                         break;
  699                 } else {
  700                         ocs_node_t *node;
  701 
  702                         /* alloc fabric node, send FLOGI */
  703                         node = ocs_node_find(sport, FC_ADDR_FABRIC);
  704                         if (node) {
  705                                 ocs_log_err(ocs, "Hmmmm ... Fabric Controller node already exists\n");
  706                                 break;
  707                         }
  708                         node = ocs_node_alloc(sport, FC_ADDR_FABRIC, FALSE, FALSE);
  709                         if (!node) {
  710                                 ocs_log_err(ocs, "Error: ocs_node_alloc() failed\n");
  711                         } else {
  712                                 if (ocs->nodedb_mask & OCS_NODEDB_PAUSE_FABRIC_LOGIN) {
  713                                         ocs_node_pause(node, __ocs_fabric_init);
  714                                 } else {
  715                                         ocs_node_transition(node, __ocs_fabric_init, NULL);
  716                                 }
  717                         }
  718                         /* Accept frames */
  719                         domain->req_accept_frames = 1;
  720                 }
  721                 /* sm: start fabric logins */
  722                 ocs_sm_transition(ctx, __ocs_domain_allocated, NULL);
  723                 break;
  724         }
  725 
  726         case OCS_EVT_DOMAIN_ALLOC_FAIL:
  727                 /* TODO: hw/device reset */
  728                 ocs_log_err(ocs, "%s recv'd waiting for DOMAIN_ALLOC_OK; shutting down domain\n",
  729                         ocs_sm_event_name(evt));
  730                 domain->req_domain_free = 1;
  731                 break;
  732 
  733         case OCS_EVT_DOMAIN_FOUND:
  734                 /* Should not happen */
  735                 ocs_assert(evt, NULL);
  736                 break;
  737 
  738         case OCS_EVT_DOMAIN_LOST:
  739                 ocs_log_debug(ocs, "%s received while waiting for ocs_hw_domain_alloc() to complete\n", ocs_sm_event_name(evt));
  740                 ocs_sm_transition(ctx, __ocs_domain_wait_domain_lost, NULL);
  741                 break;
  742 
  743         default:
  744                 __ocs_domain_common(__func__, ctx, evt, arg);
  745                 return NULL;
  746         }
  747 
  748         return NULL;
  749 }
  750 
  751 /**
  752  * @ingroup domain_sm
  753  * @brief Domain state machine: Wait for the domain attach request.
  754  *
  755  * <h3 class="desc">Description</h3>
  756  * In this state, the domain has been allocated and is waiting for a domain attach request.
  757  * The attach request comes from a node instance completing the fabric login,
  758  * or from a point-to-point negotiation and login.
  759  *
  760  * @param ctx Remote node state machine context.
  761  * @param evt Event to process.
  762  * @param arg Per event optional argument.
  763  *
  764  * @return Returns NULL.
  765  */
  766 
  767 void *
  768 __ocs_domain_allocated(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  769 {
  770         int32_t rc = 0;
  771         std_domain_state_decl();
  772 
  773         domain_sm_trace(domain);
  774 
  775         switch(evt) {
  776         case OCS_EVT_DOMAIN_REQ_ATTACH: {
  777                 uint32_t fc_id;
  778 
  779                 ocs_assert(arg, NULL);
  780 
  781                 fc_id = *((uint32_t*)arg);
  782                 ocs_log_debug(ocs, "Requesting hw domain attach fc_id x%x\n", fc_id);
  783                 /* Update sport lookup */
  784                 ocs_lock(&domain->lookup_lock);
  785                         spv_set(domain->lookup, fc_id, domain->sport);
  786                 ocs_unlock(&domain->lookup_lock);
  787 
  788                 /* Update display name for the sport */
  789                 ocs_node_fcid_display(fc_id, domain->sport->display_name, sizeof(domain->sport->display_name));
  790 
  791                 /* Issue domain attach call */
  792                 rc = ocs_hw_domain_attach(&ocs->hw, domain, fc_id);
  793                 if (rc) {
  794                         ocs_log_err(ocs, "ocs_hw_domain_attach failed: %d\n", rc);
  795                         return NULL;
  796                 }
  797                 /* sm: / domain_attach */
  798                 ocs_sm_transition(ctx, __ocs_domain_wait_attach, NULL);
  799                 break;
  800         }
  801 
  802         case OCS_EVT_DOMAIN_FOUND:
  803                 /* Should not happen */
  804                 ocs_assert(evt, NULL);
  805                 break;
  806 
  807         case OCS_EVT_DOMAIN_LOST: {
  808                 int32_t rc;
  809                 ocs_log_debug(ocs, "%s received while waiting for OCS_EVT_DOMAIN_REQ_ATTACH\n",
  810                         ocs_sm_event_name(evt));
  811                 ocs_domain_lock(domain);
  812                 if (!ocs_list_empty(&domain->sport_list)) {
  813                         /* if there are sports, transition to wait state and
  814                          * send shutdown to each sport */
  815                         ocs_sport_t     *sport = NULL;
  816                         ocs_sport_t     *sport_next = NULL;
  817                         ocs_sm_transition(ctx, __ocs_domain_wait_sports_free, NULL);
  818                         ocs_list_foreach_safe(&domain->sport_list, sport, sport_next) {
  819                                 ocs_sm_post_event(&sport->sm, OCS_EVT_SHUTDOWN, NULL);
  820                         }
  821                         ocs_domain_unlock(domain);
  822                 } else {
  823                         ocs_domain_unlock(domain);
  824                         /* no sports exist, free domain */
  825                         ocs_sm_transition(ctx, __ocs_domain_wait_shutdown, NULL);
  826                         rc = ocs_hw_domain_free(&ocs->hw, domain);
  827                         if (rc) {
  828                                 ocs_log_err(ocs, "ocs_hw_domain_free() failed: %d\n", rc);
  829                                 /* TODO: hw/device reset needed */
  830                         }
  831                 }
  832 
  833                 break;
  834         }
  835 
  836         default:
  837                 __ocs_domain_common(__func__, ctx, evt, arg);
  838                 return NULL;
  839         }
  840 
  841         return NULL;
  842 }
  843 
  844 /**
  845  * @ingroup domain_sm
  846  * @brief Domain state machine: Wait for the HW domain attach to complete.
  847  *
  848  * <h3 class="desc">Description</h3>
  849  * Waits for the HW domain attach to complete. Forwards attach ok event to the
  850  * fabric node state machine.
  851  *
  852  * @param ctx Remote node state machine context.
  853  * @param evt Event to process.
  854  * @param arg Per event optional argument.
  855  *
  856  * @return Returns NULL.
  857  */
  858 
  859 void *
  860 __ocs_domain_wait_attach(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  861 {
  862         std_domain_state_decl();
  863 
  864         domain_sm_trace(domain);
  865 
  866         switch(evt) {
  867         case OCS_EVT_DOMAIN_ATTACH_OK: {
  868                 ocs_node_t *node = NULL;
  869                 ocs_node_t *next_node = NULL;
  870                 ocs_sport_t *sport;
  871                 ocs_sport_t *next_sport;
  872 
  873                 /* Mark as attached */
  874                 domain->attached = 1;
  875 
  876                 /* Register with SCSI API */
  877                 if (ocs->enable_tgt)
  878                         ocs_scsi_tgt_new_domain(domain);
  879                 if (ocs->enable_ini)
  880                         ocs_scsi_ini_new_domain(domain);
  881 
  882                 /* Transition to ready */
  883                 /* sm: / forward event to all sports and nodes */
  884                 ocs_sm_transition(ctx, __ocs_domain_ready, NULL);
  885 
  886                 /* We have an FCFI, so we can accept frames */
  887                 domain->req_accept_frames = 1;
  888                 /* Set domain notify pending state to avoid duplicate domain event post */
  889                 domain->domain_notify_pend = 1;
  890 
  891                 /* Notify all nodes that the domain attach request has completed
  892                  * Note: sport will have already received notification of sport attached
  893                  * as a result of the HW's port attach.
  894                  */
  895                 ocs_domain_lock(domain);
  896                         ocs_list_foreach_safe(&domain->sport_list, sport, next_sport) {
  897                                 ocs_sport_lock(sport);
  898                                         ocs_list_foreach_safe(&sport->node_list, node, next_node) {
  899                                                 ocs_node_post_event(node, OCS_EVT_DOMAIN_ATTACH_OK, NULL);
  900                                         }
  901                                 ocs_sport_unlock(sport);
  902                         }
  903                 ocs_domain_unlock(domain);
  904                 domain->domain_notify_pend = 0;
  905                 break;
  906         }
  907 
  908         case OCS_EVT_DOMAIN_ATTACH_FAIL:
  909                 ocs_log_debug(ocs, "%s received while waiting for hw attach to complete\n", ocs_sm_event_name(evt));
  910                 /* TODO: hw/device reset */
  911                 break;
  912 
  913         case OCS_EVT_DOMAIN_FOUND:
  914                 /* Should not happen */
  915                 ocs_assert(evt, NULL);
  916                 break;
  917 
  918         case OCS_EVT_DOMAIN_LOST:
  919                 /* Domain lost while waiting for an attach to complete, go to a state that waits for
  920                  * the domain attach to complete, then handle domain lost
  921                  */
  922                 ocs_sm_transition(ctx, __ocs_domain_wait_domain_lost, NULL);
  923                 break;
  924 
  925         case OCS_EVT_DOMAIN_REQ_ATTACH:
  926                 /* In P2P we can get an attach request from the other FLOGI path, so drop this one */
  927                 break;
  928 
  929         default:
  930                 __ocs_domain_common(__func__, ctx, evt, arg);
  931                 return NULL;
  932         }
  933 
  934         return NULL;
  935 }
  936 
  937 /**
  938  * @ingroup domain_sm
  939  * @brief Domain state machine: Ready state.
  940  *
  941  * <h3 class="desc">Description</h3>
  942  * This is a domain ready state. It waits for a domain-lost event, and initiates shutdown.
  943  *
  944  * @param ctx Remote node state machine context.
  945  * @param evt Event to process.
  946  * @param arg Per event optional argument.
  947  *
  948  * @return Returns NULL.
  949  */
  950 
  951 void *
  952 __ocs_domain_ready(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
  953 {
  954         std_domain_state_decl();
  955 
  956         domain_sm_trace(domain);
  957 
  958         switch(evt) {
  959         case OCS_EVT_ENTER: {
  960                 /* start any pending vports */
  961                 if (ocs_vport_start(domain)) {
  962                         ocs_log_debug(domain->ocs, "ocs_vport_start() did not start all vports\n");
  963                 }
  964                 break;
  965         }
  966         case OCS_EVT_DOMAIN_LOST: {
  967                 int32_t rc;
  968                 ocs_domain_lock(domain);
  969                 if (!ocs_list_empty(&domain->sport_list)) {
  970                         /* if there are sports, transition to wait state and send
  971                         * shutdown to each sport */
  972                         ocs_sport_t     *sport = NULL;
  973                         ocs_sport_t     *sport_next = NULL;
  974                         ocs_sm_transition(ctx, __ocs_domain_wait_sports_free, NULL);
  975                         ocs_list_foreach_safe(&domain->sport_list, sport, sport_next) {
  976                                 ocs_sm_post_event(&sport->sm, OCS_EVT_SHUTDOWN, NULL);
  977                         }
  978                         ocs_domain_unlock(domain);
  979                 } else {
  980                         ocs_domain_unlock(domain);
  981                         /* no sports exist, free domain */
  982                         ocs_sm_transition(ctx, __ocs_domain_wait_shutdown, NULL);
  983                         rc = ocs_hw_domain_free(&ocs->hw, domain);
  984                         if (rc) {
  985                                 ocs_log_err(ocs, "ocs_hw_domain_free() failed: %d\n", rc);
  986                                 /* TODO: hw/device reset needed */
  987                         }
  988                 }
  989                 break;
  990         }
  991 
  992         case OCS_EVT_DOMAIN_FOUND:
  993                 /* Should not happen */
  994                 ocs_assert(evt, NULL);
  995                 break;
  996 
  997         case OCS_EVT_DOMAIN_REQ_ATTACH: {
  998                 /* can happen during p2p */
  999                 uint32_t fc_id;
 1000 
 1001                 ocs_assert(arg, NULL);
 1002                 fc_id = *((uint32_t*)arg);
 1003 
 1004                 /* Assume that the domain is attached */
 1005                 ocs_assert(domain->attached, NULL);
 1006 
 1007                 /* Verify that the requested FC_ID is the same as the one we're working with */
 1008                 ocs_assert(domain->sport->fc_id == fc_id, NULL);
 1009                 break;
 1010         }
 1011 
 1012         default:
 1013                 __ocs_domain_common(__func__, ctx, evt, arg);
 1014                 return NULL;
 1015         }
 1016 
 1017         return NULL;
 1018 }
 1019 
 1020 /**
 1021  * @ingroup domain_sm
 1022  * @brief Domain state machine: Wait for nodes to free prior to the domain shutdown.
 1023  *
 1024  * <h3 class="desc">Description</h3>
 1025  * All nodes are freed, and ready for a domain shutdown.
 1026  *
 1027  * @param ctx Remote node sm context.
 1028  * @param evt Event to process.
 1029  * @param arg Per event optional argument.
 1030  *
 1031  * @return Returns NULL.
 1032  */
 1033 
 1034 void *
 1035 __ocs_domain_wait_sports_free(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
 1036 {
 1037         std_domain_state_decl();
 1038 
 1039         domain_sm_trace(domain);
 1040 
 1041         switch(evt) {
 1042         case OCS_EVT_ALL_CHILD_NODES_FREE: {
 1043                 int32_t rc;
 1044 
 1045                 /* sm: ocs_hw_domain_free */
 1046                 ocs_sm_transition(ctx, __ocs_domain_wait_shutdown, NULL);
 1047 
 1048                 /* Request ocs_hw_domain_free and wait for completion */
 1049                 rc = ocs_hw_domain_free(&ocs->hw, domain);
 1050                 if (rc) {
 1051                         ocs_log_err(ocs, "ocs_hw_domain_free() failed: %d\n", rc);
 1052                 }
 1053                 break;
 1054         }
 1055         default:
 1056                 __ocs_domain_common_shutdown(__func__, ctx, evt, arg);
 1057                 return NULL;
 1058         }
 1059 
 1060         return NULL;
 1061 }
 1062 
 1063 /**
 1064  * @ingroup domain_sm
 1065  * @brief Domain state machine: Complete the domain shutdown.
 1066  *
 1067  * <h3 class="desc">Description</h3>
 1068  * Waits for a HW domain free to complete.
 1069  *
 1070  * @param ctx Remote node state machine context.
 1071  * @param evt Event to process.
 1072  * @param arg Per event optional argument.
 1073  *
 1074  * @return Returns NULL.
 1075  */
 1076 
 1077 void *
 1078 __ocs_domain_wait_shutdown(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
 1079 {
 1080         std_domain_state_decl();
 1081 
 1082         domain_sm_trace(domain);
 1083 
 1084         switch(evt) {
 1085         case OCS_EVT_DOMAIN_FREE_OK: {
 1086                 if (ocs->enable_ini)
 1087                         ocs_scsi_ini_del_domain(domain);
 1088                 if (ocs->enable_tgt)
 1089                         ocs_scsi_tgt_del_domain(domain);
 1090 
 1091                 /* sm: domain_free */
 1092                 if (domain->domain_found_pending) {
 1093                         /* save fcf_wwn and drec from this domain, free current domain and allocate
 1094                          * a new one with the same fcf_wwn
 1095                          * TODO: could use a SLI-4 "re-register VPI" operation here
 1096                          */
 1097                         uint64_t fcf_wwn = domain->fcf_wwn;
 1098                         ocs_domain_record_t drec = domain->pending_drec;
 1099 
 1100                         ocs_log_debug(ocs, "Reallocating domain\n");
 1101                         domain->req_domain_free = 1;
 1102                         domain = ocs_domain_alloc(ocs, fcf_wwn);
 1103 
 1104                         if (domain == NULL) {
 1105                                 ocs_log_err(ocs, "ocs_domain_alloc() failed\n");
 1106                                 /* TODO: hw/device reset needed */
 1107                                 return NULL;
 1108                         }
 1109                         /*
 1110                          * got a new domain; at this point, there are at least two domains
 1111                          * once the req_domain_free flag is processed, the associated domain
 1112                          * will be removed.
 1113                          */
 1114                         ocs_sm_transition(&domain->drvsm, __ocs_domain_init, NULL);
 1115                         ocs_sm_post_event(&domain->drvsm, OCS_EVT_DOMAIN_FOUND, &drec);
 1116                 } else {
 1117                         domain->req_domain_free = 1;
 1118                 }
 1119                 break;
 1120         }
 1121 
 1122         default:
 1123                 __ocs_domain_common_shutdown(__func__, ctx, evt, arg);
 1124                 return NULL;
 1125         }
 1126 
 1127         return NULL;
 1128 }
 1129 
 1130 /**
 1131  * @ingroup domain_sm
 1132  * @brief Domain state machine: Wait for the domain alloc/attach completion
 1133  * after receiving a domain lost.
 1134  *
 1135  * <h3 class="desc">Description</h3>
 1136  * This state is entered when receiving a domain lost while waiting for a domain alloc
 1137  * or a domain attach to complete.
 1138  *
 1139  * @param ctx Remote node state machine context.
 1140  * @param evt Event to process.
 1141  * @param arg Per event optional argument.
 1142  *
 1143  * @return Returns NULL.
 1144  */
 1145 
 1146 void *
 1147 __ocs_domain_wait_domain_lost(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *arg)
 1148 {
 1149         std_domain_state_decl();
 1150 
 1151         domain_sm_trace(domain);
 1152 
 1153         switch(evt) {
 1154         case OCS_EVT_DOMAIN_ALLOC_OK:
 1155         case OCS_EVT_DOMAIN_ATTACH_OK: {
 1156                 int32_t rc;
 1157                 ocs_domain_lock(domain);
 1158                 if (!ocs_list_empty(&domain->sport_list)) {
 1159                         /* if there are sports, transition to wait state and send
 1160                         * shutdown to each sport */
 1161                         ocs_sport_t     *sport = NULL;
 1162                         ocs_sport_t     *sport_next = NULL;
 1163                         ocs_sm_transition(ctx, __ocs_domain_wait_sports_free, NULL);
 1164                         ocs_list_foreach_safe(&domain->sport_list, sport, sport_next) {
 1165                                 ocs_sm_post_event(&sport->sm, OCS_EVT_SHUTDOWN, NULL);
 1166                         }
 1167                         ocs_domain_unlock(domain);
 1168                 } else {
 1169                         ocs_domain_unlock(domain);
 1170                         /* no sports exist, free domain */
 1171                         ocs_sm_transition(ctx, __ocs_domain_wait_shutdown, NULL);
 1172                         rc = ocs_hw_domain_free(&ocs->hw, domain);
 1173                         if (rc) {
 1174                                 ocs_log_err(ocs, "ocs_hw_domain_free() failed: %d\n", rc);
 1175                                 /* TODO: hw/device reset needed */
 1176                         }
 1177                 }
 1178                 break;
 1179         }
 1180         case OCS_EVT_DOMAIN_ALLOC_FAIL:
 1181         case OCS_EVT_DOMAIN_ATTACH_FAIL:
 1182                 ocs_log_err(ocs, "[domain] %-20s: failed\n", ocs_sm_event_name(evt));
 1183                 /* TODO: hw/device reset needed */
 1184                 break;
 1185 
 1186         default:
 1187                 __ocs_domain_common_shutdown(__func__, ctx, evt, arg);
 1188                 return NULL;
 1189         }
 1190 
 1191         return NULL;
 1192 }
 1193 
 1194 /**
 1195  * @brief Save the port's service parameters.
 1196  *
 1197  * <h3 class="desc">Description</h3>
 1198  * Service parameters from the fabric FLOGI are saved in the domain's
 1199  * flogi_service_params array.
 1200  *
 1201  * @param domain Pointer to the domain.
 1202  * @param payload Service parameters to save.
 1203  *
 1204  * @return None.
 1205  */
 1206 
 1207 void
 1208 ocs_domain_save_sparms(ocs_domain_t *domain, void *payload)
 1209 {
 1210         ocs_memcpy(domain->flogi_service_params, payload, sizeof (fc_plogi_payload_t));
 1211 }
 1212 /**
 1213  * @brief Initiator domain attach. (internal call only)
 1214  *
 1215  * Assumes that the domain SM lock is already locked
 1216  *
 1217  * <h3 class="desc">Description</h3>
 1218  * The HW domain attach function is started.
 1219  *
 1220  * @param domain Pointer to the domain object.
 1221  * @param s_id FC_ID of which to register this domain.
 1222  *
 1223  * @return None.
 1224  */
 1225 
 1226 void
 1227 __ocs_domain_attach_internal(ocs_domain_t *domain, uint32_t s_id)
 1228 {
 1229         ocs_memcpy(domain->dma.virt, ((uint8_t*)domain->flogi_service_params)+4, sizeof (fc_plogi_payload_t) - 4);
 1230         (void)ocs_sm_post_event(&domain->drvsm, OCS_EVT_DOMAIN_REQ_ATTACH, &s_id);
 1231 }
 1232 
 1233 /**
 1234  * @brief Initiator domain attach.
 1235  *
 1236  * <h3 class="desc">Description</h3>
 1237  * The HW domain attach function is started.
 1238  *
 1239  * @param domain Pointer to the domain object.
 1240  * @param s_id FC_ID of which to register this domain.
 1241  *
 1242  * @return None.
 1243  */
 1244 
 1245 void
 1246 ocs_domain_attach(ocs_domain_t *domain, uint32_t s_id)
 1247 {
 1248         __ocs_domain_attach_internal(domain, s_id);
 1249 }
 1250 
 1251 int
 1252 ocs_domain_post_event(ocs_domain_t *domain, ocs_sm_event_t event, void *arg)
 1253 {
 1254         int rc;
 1255         int accept_frames;
 1256         int req_domain_free;
 1257 
 1258         rc = ocs_sm_post_event(&domain->drvsm, event, arg);
 1259 
 1260         req_domain_free = domain->req_domain_free;
 1261         domain->req_domain_free = 0;
 1262 
 1263         accept_frames = domain->req_accept_frames;
 1264         domain->req_accept_frames = 0;
 1265 
 1266         if (accept_frames) {
 1267                 ocs_domain_accept_frames(domain);
 1268         }
 1269 
 1270         if (req_domain_free) {
 1271                 ocs_domain_free(domain);
 1272         }
 1273 
 1274         return rc;
 1275 }
 1276 
 1277 /**
 1278  * @brief Return the WWN as a uint64_t.
 1279  *
 1280  * <h3 class="desc">Description</h3>
 1281  * Calls the HW property function for the WWNN or WWPN, and returns the value
 1282  * as a uint64_t.
 1283  *
 1284  * @param hw Pointer to the HW object.
 1285  * @param prop HW property.
 1286  *
 1287  * @return Returns uint64_t request value.
 1288  */
 1289 
 1290 uint64_t
 1291 ocs_get_wwn(ocs_hw_t *hw, ocs_hw_property_e prop)
 1292 {
 1293         uint8_t *p = ocs_hw_get_ptr(hw, prop);
 1294         uint64_t value = 0;
 1295 
 1296         if (p) {
 1297                 uint32_t i;
 1298                 for (i = 0; i < sizeof(value); i++) {
 1299                         value = (value << 8) | p[i];
 1300                 }
 1301         }
 1302         return value;
 1303 }
 1304 
 1305 /**
 1306  * @brief Generate a domain ddump.
 1307  *
 1308  * <h3 class="desc">Description</h3>
 1309  * Generates a domain ddump.
 1310  *
 1311  * @param textbuf Pointer to the text buffer.
 1312  * @param domain Pointer to the domain context.
 1313  *
 1314  * @return Returns 0 on success, or a negative value on failure.
 1315  */
 1316 
 1317 int
 1318 ocs_ddump_domain(ocs_textbuf_t *textbuf, ocs_domain_t *domain)
 1319 {
 1320         ocs_sport_t *sport;
 1321         int retval = 0;
 1322 
 1323         ocs_ddump_section(textbuf, "domain", domain->instance_index);
 1324         ocs_ddump_value(textbuf, "display_name", "%s", domain->display_name);
 1325 
 1326         ocs_ddump_value(textbuf, "fcf", "%#x", domain->fcf);
 1327         ocs_ddump_value(textbuf, "fcf_indicator", "%#x", domain->fcf_indicator);
 1328         ocs_ddump_value(textbuf, "vlan_id", "%#x", domain->vlan_id);
 1329         ocs_ddump_value(textbuf, "indicator", "%#x", domain->indicator);
 1330         ocs_ddump_value(textbuf, "attached", "%d", domain->attached);
 1331         ocs_ddump_value(textbuf, "is_loop", "%d", domain->is_loop);
 1332         ocs_ddump_value(textbuf, "is_nlport", "%d", domain->is_nlport);
 1333 
 1334         ocs_scsi_ini_ddump(textbuf, OCS_SCSI_DDUMP_DOMAIN, domain);
 1335         ocs_scsi_tgt_ddump(textbuf, OCS_SCSI_DDUMP_DOMAIN, domain);
 1336 
 1337         ocs_display_sparams(NULL, "domain_sparms", 1, textbuf, domain->dma.virt);
 1338 
 1339         if (ocs_domain_lock_try(domain) != TRUE) {
 1340                 /* Didn't get the lock */
 1341                 return -1;
 1342         }
 1343                 ocs_list_foreach(&domain->sport_list, sport) {
 1344                         retval = ocs_ddump_sport(textbuf, sport);
 1345                         if (retval != 0) {
 1346                                 break;
 1347                         }
 1348                 }
 1349 
 1350 #if defined(ENABLE_FABRIC_EMULATION)
 1351                 ocs_ddump_ns(textbuf, domain->ocs_ns);
 1352 #endif
 1353 
 1354         ocs_domain_unlock(domain);
 1355 
 1356         ocs_ddump_endsection(textbuf, "domain", domain->instance_index);
 1357 
 1358         return retval;
 1359 }
 1360 
 1361 void
 1362 ocs_mgmt_domain_list(ocs_textbuf_t *textbuf, void *object)
 1363 {
 1364         ocs_sport_t *sport;
 1365         ocs_domain_t *domain = (ocs_domain_t *)object;
 1366 
 1367         ocs_mgmt_start_section(textbuf, "domain", domain->instance_index);
 1368 
 1369         /* Add my status values to textbuf */
 1370         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "fcf");
 1371         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "fcf_indicator");
 1372         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "vlan_id");
 1373         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "indicator");
 1374         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "attached");
 1375         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "is_loop");
 1376         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "display_name");
 1377         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RD, "num_sports");
 1378 #if defined(ENABLE_FABRIC_EMULATION)
 1379         ocs_mgmt_emit_property_name(textbuf, MGMT_MODE_RW, "femul_enable");
 1380 #endif
 1381 
 1382         if (ocs_domain_lock_try(domain) == TRUE) {
 1383                 /* If we get here, then we are holding the domain lock */
 1384                 ocs_list_foreach(&domain->sport_list, sport) {
 1385                         if ((sport->mgmt_functions) && (sport->mgmt_functions->get_list_handler)) {
 1386                                 sport->mgmt_functions->get_list_handler(textbuf, sport);
 1387                         }
 1388                 }
 1389                 ocs_domain_unlock(domain);
 1390         }
 1391 
 1392         ocs_mgmt_end_section(textbuf, "domain", domain->instance_index);
 1393 }
 1394 
 1395 int
 1396 ocs_mgmt_domain_get(ocs_textbuf_t *textbuf, char *parent, char *name, void *object)
 1397 {
 1398         ocs_sport_t *sport;
 1399         ocs_domain_t *domain = (ocs_domain_t *)object;
 1400         char qualifier[80];
 1401         int retval = -1;
 1402 
 1403         ocs_mgmt_start_section(textbuf, "domain", domain->instance_index);
 1404 
 1405         snprintf(qualifier, sizeof(qualifier), "%s/domain[%d]", parent, domain->instance_index);
 1406 
 1407         /* If it doesn't start with my qualifier I don't know what to do with it */
 1408         if (ocs_strncmp(name, qualifier, strlen(qualifier)) == 0) {
 1409                 char *unqualified_name = name + strlen(qualifier) +1;
 1410 
 1411                 /* See if it's a value I can supply */
 1412                 if (ocs_strcmp(unqualified_name, "display_name") == 0) {
 1413                         ocs_mgmt_emit_string(textbuf, MGMT_MODE_RD, "display_name", domain->display_name);
 1414                         retval = 0;
 1415                 } else if (ocs_strcmp(unqualified_name, "fcf") == 0) {
 1416                         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "fcf", "%#x", domain->fcf);
 1417                         retval = 0;
 1418                 } else if (ocs_strcmp(unqualified_name, "fcf_indicator") == 0) {
 1419                         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "fcf_indicator", "%#x", domain->fcf_indicator);
 1420                         retval = 0;
 1421                 } else if (ocs_strcmp(unqualified_name, "vlan_id") == 0) {
 1422                         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "vlan_id", "%#x", domain->vlan_id);
 1423                         retval = 0;
 1424                 } else if (ocs_strcmp(unqualified_name, "indicator") == 0) {
 1425                         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "indicator", "%#x", domain->indicator);
 1426                         retval = 0;
 1427                 } else if (ocs_strcmp(unqualified_name, "attached") == 0) {
 1428                         ocs_mgmt_emit_boolean(textbuf, MGMT_MODE_RD, "attached", domain->attached);
 1429                         retval = 0;
 1430                 } else if (ocs_strcmp(unqualified_name, "is_loop") == 0) {
 1431                         ocs_mgmt_emit_boolean(textbuf, MGMT_MODE_RD, "is_loop",  domain->is_loop);
 1432                         retval = 0;
 1433                 } else if (ocs_strcmp(unqualified_name, "is_nlport") == 0) {
 1434                         ocs_mgmt_emit_boolean(textbuf, MGMT_MODE_RD, "is_nlport",  domain->is_nlport);
 1435                         retval = 0;
 1436                 } else if (ocs_strcmp(unqualified_name, "display_name") == 0) {
 1437                         ocs_mgmt_emit_string(textbuf, MGMT_MODE_RD, "display_name", domain->display_name);
 1438                         retval = 0;
 1439 #if defined(ENABLE_FABRIC_EMULATION)
 1440                 } else if (ocs_strcmp(unqualified_name, "femul_enable") == 0) {
 1441                         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RW, "femul_enable", "%d", domain->femul_enable);
 1442                         retval = 0;
 1443 #endif
 1444                 } else if (ocs_strcmp(unqualified_name, "num_sports") == 0) {
 1445                         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "num_sports", "%d", domain->sport_instance_count);
 1446                         retval = 0;
 1447                 } else {
 1448                         /* If I didn't know the value of this status pass the request to each of my children */
 1449 
 1450                         ocs_domain_lock(domain);
 1451                         ocs_list_foreach(&domain->sport_list, sport) {
 1452                                 if ((sport->mgmt_functions) && (sport->mgmt_functions->get_handler)) {
 1453                                         retval = sport->mgmt_functions->get_handler(textbuf, qualifier, name, sport);
 1454                                 }
 1455 
 1456                                 if (retval == 0) {
 1457                                         break;
 1458                                 }
 1459                         }
 1460                         ocs_domain_unlock(domain);
 1461                 }
 1462         }
 1463 
 1464         ocs_mgmt_end_section(textbuf, "domain", domain->instance_index);
 1465         return retval;
 1466 }
 1467 
 1468 void
 1469 ocs_mgmt_domain_get_all(ocs_textbuf_t *textbuf, void *object)
 1470 {
 1471         ocs_sport_t *sport;
 1472         ocs_domain_t *domain = (ocs_domain_t *)object;
 1473 
 1474         ocs_mgmt_start_section(textbuf, "domain", domain->instance_index);
 1475 
 1476         ocs_mgmt_emit_string(textbuf, MGMT_MODE_RD, "display_name", domain->display_name);
 1477         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "fcf", "%#x", domain->fcf);
 1478         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "fcf_indicator", "%#x", domain->fcf_indicator);
 1479         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "vlan_id", "%#x", domain->vlan_id);
 1480         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "indicator", "%#x", domain->indicator);
 1481         ocs_mgmt_emit_boolean(textbuf, MGMT_MODE_RD, "attached", domain->attached);
 1482         ocs_mgmt_emit_boolean(textbuf, MGMT_MODE_RD, "is_loop",  domain->is_loop);
 1483         ocs_mgmt_emit_boolean(textbuf, MGMT_MODE_RD, "is_nlport",  domain->is_nlport);
 1484 #if defined(ENABLE_FABRIC_EMULATION)
 1485         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RW, "femul_enable", "%d", domain->femul_enable);
 1486 #endif
 1487         ocs_mgmt_emit_int(textbuf, MGMT_MODE_RD, "num_sports",  "%d", domain->sport_instance_count);
 1488 
 1489         ocs_domain_lock(domain);
 1490         ocs_list_foreach(&domain->sport_list, sport) {
 1491                 if ((sport->mgmt_functions) && (sport->mgmt_functions->get_all_handler)) {
 1492                         sport->mgmt_functions->get_all_handler(textbuf, sport);
 1493                 }
 1494         }
 1495         ocs_domain_unlock(domain);
 1496 
 1497         ocs_mgmt_end_unnumbered_section(textbuf, "domain");
 1498 
 1499 }
 1500 
 1501 int
 1502 ocs_mgmt_domain_set(char *parent, char *name, char *value, void *object)
 1503 {
 1504         ocs_sport_t *sport;
 1505         ocs_domain_t *domain = (ocs_domain_t *)object;
 1506         char qualifier[80];
 1507         int retval = -1;
 1508 
 1509         snprintf(qualifier, sizeof(qualifier), "%s/domain[%d]", parent, domain->instance_index);
 1510 
 1511         /* If it doesn't start with my qualifier I don't know what to do with it */
 1512         if (ocs_strncmp(name, qualifier, strlen(qualifier)) == 0) {
 1513                 /* See if it's a value I can supply */
 1514 
 1515                 /* if (ocs_strcmp(unqualified_name, "display_name") == 0) {
 1516                 } else */
 1517                 {
 1518                         /* If I didn't know the value of this status pass the request to each of my children */
 1519                         ocs_domain_lock(domain);
 1520                         ocs_list_foreach(&domain->sport_list, sport) {
 1521                                 if ((sport->mgmt_functions) && (sport->mgmt_functions->set_handler)) {
 1522                                         retval = sport->mgmt_functions->set_handler(qualifier, name, value, sport);
 1523                                 }
 1524 
 1525                                 if (retval == 0) {
 1526                                         break;
 1527                                 }
 1528                         }
 1529                         ocs_domain_unlock(domain);
 1530                 }
 1531         }
 1532 
 1533         return retval;
 1534 }
 1535 
 1536 int
 1537 ocs_mgmt_domain_exec(char *parent, char *action, void *arg_in, uint32_t arg_in_length,
 1538                         void *arg_out, uint32_t arg_out_length, void *object)
 1539 {
 1540         ocs_sport_t *sport;
 1541         ocs_domain_t *domain = (ocs_domain_t *)object;
 1542         char qualifier[80];
 1543         int retval = -1;
 1544 
 1545         snprintf(qualifier, sizeof(qualifier), "%s.domain%d", parent, domain->instance_index);
 1546 
 1547         /* If it doesn't start with my qualifier I don't know what to do with it */
 1548         if (ocs_strncmp(action, qualifier, strlen(qualifier)) == 0) {
 1549                 {
 1550                         /* If I didn't know how to do this action pass the request to each of my children */
 1551                         ocs_domain_lock(domain);
 1552                         ocs_list_foreach(&domain->sport_list, sport) {
 1553                                 if ((sport->mgmt_functions) && (sport->mgmt_functions->exec_handler)) {
 1554                                         retval = sport->mgmt_functions->exec_handler(qualifier, action, arg_in, arg_in_length, arg_out, arg_out_length, sport);
 1555                                 }
 1556 
 1557                                 if (retval == 0) {
 1558                                         break;
 1559                                 }
 1560                         }
 1561                         ocs_domain_unlock(domain);
 1562                 }
 1563         }
 1564 
 1565         return retval;
 1566 }
Cache object: 4c93325a8a0c063ce78f2111a4d90400 
 
 |