1103 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1103 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Copyright (C) 2021 Broadcom. All Rights Reserved. The term
 | |
|  * “Broadcom” refers to Broadcom Inc. and/or its subsidiaries.
 | |
|  */
 | |
| 
 | |
| #include "efc.h"
 | |
| 
 | |
| int
 | |
| efc_remote_node_cb(void *arg, int event, void *data)
 | |
| {
 | |
| 	struct efc *efc = arg;
 | |
| 	struct efc_remote_node *rnode = data;
 | |
| 	struct efc_node *node = rnode->node;
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	efc_node_post_event(node, event, NULL);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| struct efc_node *
 | |
| efc_node_find(struct efc_nport *nport, u32 port_id)
 | |
| {
 | |
| 	/* Find an FC node structure given the FC port ID */
 | |
| 	return xa_load(&nport->lookup, port_id);
 | |
| }
 | |
| 
 | |
| static void
 | |
| _efc_node_free(struct kref *arg)
 | |
| {
 | |
| 	struct efc_node *node = container_of(arg, struct efc_node, ref);
 | |
| 	struct efc *efc = node->efc;
 | |
| 	struct efc_dma *dma;
 | |
| 
 | |
| 	dma = &node->sparm_dma_buf;
 | |
| 	dma_pool_free(efc->node_dma_pool, dma->virt, dma->phys);
 | |
| 	memset(dma, 0, sizeof(struct efc_dma));
 | |
| 	mempool_free(node, efc->node_pool);
 | |
| }
 | |
| 
 | |
| struct efc_node *efc_node_alloc(struct efc_nport *nport,
 | |
| 				u32 port_id, bool init, bool targ)
 | |
| {
 | |
| 	int rc;
 | |
| 	struct efc_node *node = NULL;
 | |
| 	struct efc *efc = nport->efc;
 | |
| 	struct efc_dma *dma;
 | |
| 
 | |
| 	if (nport->shutting_down) {
 | |
| 		efc_log_debug(efc, "node allocation when shutting down %06x",
 | |
| 			      port_id);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	node = mempool_alloc(efc->node_pool, GFP_ATOMIC);
 | |
| 	if (!node) {
 | |
| 		efc_log_err(efc, "node allocation failed %06x", port_id);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	memset(node, 0, sizeof(*node));
 | |
| 
 | |
| 	dma = &node->sparm_dma_buf;
 | |
| 	dma->size = NODE_SPARAMS_SIZE;
 | |
| 	dma->virt = dma_pool_zalloc(efc->node_dma_pool, GFP_ATOMIC, &dma->phys);
 | |
| 	if (!dma->virt) {
 | |
| 		efc_log_err(efc, "node dma alloc failed\n");
 | |
| 		goto dma_fail;
 | |
| 	}
 | |
| 	node->rnode.indicator = U32_MAX;
 | |
| 	node->nport = nport;
 | |
| 
 | |
| 	node->efc = efc;
 | |
| 	node->init = init;
 | |
| 	node->targ = targ;
 | |
| 
 | |
| 	spin_lock_init(&node->pend_frames_lock);
 | |
| 	INIT_LIST_HEAD(&node->pend_frames);
 | |
| 	spin_lock_init(&node->els_ios_lock);
 | |
| 	INIT_LIST_HEAD(&node->els_ios_list);
 | |
| 	node->els_io_enabled = true;
 | |
| 
 | |
| 	rc = efc_cmd_node_alloc(efc, &node->rnode, port_id, nport);
 | |
| 	if (rc) {
 | |
| 		efc_log_err(efc, "efc_hw_node_alloc failed: %d\n", rc);
 | |
| 		goto hw_alloc_fail;
 | |
| 	}
 | |
| 
 | |
| 	node->rnode.node = node;
 | |
| 	node->sm.app = node;
 | |
| 	node->evtdepth = 0;
 | |
| 
 | |
| 	efc_node_update_display_name(node);
 | |
| 
 | |
| 	rc = xa_err(xa_store(&nport->lookup, port_id, node, GFP_ATOMIC));
 | |
| 	if (rc) {
 | |
| 		efc_log_err(efc, "Node lookup store failed: %d\n", rc);
 | |
| 		goto xa_fail;
 | |
| 	}
 | |
| 
 | |
| 	/* initialize refcount */
 | |
| 	kref_init(&node->ref);
 | |
| 	node->release = _efc_node_free;
 | |
| 	kref_get(&nport->ref);
 | |
| 
 | |
| 	return node;
 | |
| 
 | |
| xa_fail:
 | |
| 	efc_node_free_resources(efc, &node->rnode);
 | |
| hw_alloc_fail:
 | |
| 	dma_pool_free(efc->node_dma_pool, dma->virt, dma->phys);
 | |
| dma_fail:
 | |
| 	mempool_free(node, efc->node_pool);
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_free(struct efc_node *node)
 | |
| {
 | |
| 	struct efc_nport *nport;
 | |
| 	struct efc *efc;
 | |
| 	int rc = 0;
 | |
| 	struct efc_node *ns = NULL;
 | |
| 
 | |
| 	nport = node->nport;
 | |
| 	efc = node->efc;
 | |
| 
 | |
| 	node_printf(node, "Free'd\n");
 | |
| 
 | |
| 	if (node->refound) {
 | |
| 		/*
 | |
| 		 * Save the name server node. We will send fake RSCN event at
 | |
| 		 * the end to handle ignored RSCN event during node deletion
 | |
| 		 */
 | |
| 		ns = efc_node_find(node->nport, FC_FID_DIR_SERV);
 | |
| 	}
 | |
| 
 | |
| 	if (!node->nport) {
 | |
| 		efc_log_err(efc, "Node already Freed\n");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* Free HW resources */
 | |
| 	rc = efc_node_free_resources(efc, &node->rnode);
 | |
| 	if (rc < 0)
 | |
| 		efc_log_err(efc, "efc_hw_node_free failed: %d\n", rc);
 | |
| 
 | |
| 	/* if the gidpt_delay_timer is still running, then delete it */
 | |
| 	if (timer_pending(&node->gidpt_delay_timer))
 | |
| 		del_timer(&node->gidpt_delay_timer);
 | |
| 
 | |
| 	xa_erase(&nport->lookup, node->rnode.fc_id);
 | |
| 
 | |
| 	/*
 | |
| 	 * If the node_list is empty,
 | |
| 	 * then post a ALL_CHILD_NODES_FREE event to the nport,
 | |
| 	 * after the lock is released.
 | |
| 	 * The nport may be free'd as a result of the event.
 | |
| 	 */
 | |
| 	if (xa_empty(&nport->lookup))
 | |
| 		efc_sm_post_event(&nport->sm, EFC_EVT_ALL_CHILD_NODES_FREE,
 | |
| 				  NULL);
 | |
| 
 | |
| 	node->nport = NULL;
 | |
| 	node->sm.current_state = NULL;
 | |
| 
 | |
| 	kref_put(&nport->ref, nport->release);
 | |
| 	kref_put(&node->ref, node->release);
 | |
| 
 | |
| 	if (ns) {
 | |
| 		/* sending fake RSCN event to name server node */
 | |
| 		efc_node_post_event(ns, EFC_EVT_RSCN_RCVD, NULL);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void
 | |
| efc_dma_copy_in(struct efc_dma *dma, void *buffer, u32 buffer_length)
 | |
| {
 | |
| 	if (!dma || !buffer || !buffer_length)
 | |
| 		return;
 | |
| 
 | |
| 	if (buffer_length > dma->size)
 | |
| 		buffer_length = dma->size;
 | |
| 
 | |
| 	memcpy(dma->virt, buffer, buffer_length);
 | |
| 	dma->len = buffer_length;
 | |
| }
 | |
| 
 | |
| int
 | |
| efc_node_attach(struct efc_node *node)
 | |
| {
 | |
| 	int rc = 0;
 | |
| 	struct efc_nport *nport = node->nport;
 | |
| 	struct efc_domain *domain = nport->domain;
 | |
| 	struct efc *efc = node->efc;
 | |
| 
 | |
| 	if (!domain->attached) {
 | |
| 		efc_log_err(efc, "Warning: unattached domain\n");
 | |
| 		return -EIO;
 | |
| 	}
 | |
| 	/* Update node->wwpn/wwnn */
 | |
| 
 | |
| 	efc_node_build_eui_name(node->wwpn, sizeof(node->wwpn),
 | |
| 				efc_node_get_wwpn(node));
 | |
| 	efc_node_build_eui_name(node->wwnn, sizeof(node->wwnn),
 | |
| 				efc_node_get_wwnn(node));
 | |
| 
 | |
| 	efc_dma_copy_in(&node->sparm_dma_buf, node->service_params + 4,
 | |
| 			sizeof(node->service_params) - 4);
 | |
| 
 | |
| 	/* take lock to protect node->rnode.attached */
 | |
| 	rc = efc_cmd_node_attach(efc, &node->rnode, &node->sparm_dma_buf);
 | |
| 	if (rc < 0)
 | |
| 		efc_log_debug(efc, "efc_hw_node_attach failed: %d\n", rc);
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_fcid_display(u32 fc_id, char *buffer, u32 buffer_length)
 | |
| {
 | |
| 	switch (fc_id) {
 | |
| 	case FC_FID_FLOGI:
 | |
| 		snprintf(buffer, buffer_length, "fabric");
 | |
| 		break;
 | |
| 	case FC_FID_FCTRL:
 | |
| 		snprintf(buffer, buffer_length, "fabctl");
 | |
| 		break;
 | |
| 	case FC_FID_DIR_SERV:
 | |
| 		snprintf(buffer, buffer_length, "nserve");
 | |
| 		break;
 | |
| 	default:
 | |
| 		if (fc_id == FC_FID_DOM_MGR) {
 | |
| 			snprintf(buffer, buffer_length, "dctl%02x",
 | |
| 				 (fc_id & 0x0000ff));
 | |
| 		} else {
 | |
| 			snprintf(buffer, buffer_length, "%06x", fc_id);
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_update_display_name(struct efc_node *node)
 | |
| {
 | |
| 	u32 port_id = node->rnode.fc_id;
 | |
| 	struct efc_nport *nport = node->nport;
 | |
| 	char portid_display[16];
 | |
| 
 | |
| 	efc_node_fcid_display(port_id, portid_display, sizeof(portid_display));
 | |
| 
 | |
| 	snprintf(node->display_name, sizeof(node->display_name), "%s.%s",
 | |
| 		 nport->display_name, portid_display);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_send_ls_io_cleanup(struct efc_node *node)
 | |
| {
 | |
| 	if (node->send_ls_acc != EFC_NODE_SEND_LS_ACC_NONE) {
 | |
| 		efc_log_debug(node->efc, "[%s] cleaning up LS_ACC oxid=0x%x\n",
 | |
| 			      node->display_name, node->ls_acc_oxid);
 | |
| 
 | |
| 		node->send_ls_acc = EFC_NODE_SEND_LS_ACC_NONE;
 | |
| 		node->ls_acc_io = NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void efc_node_handle_implicit_logo(struct efc_node *node)
 | |
| {
 | |
| 	int rc;
 | |
| 
 | |
| 	/*
 | |
| 	 * currently, only case for implicit logo is PLOGI
 | |
| 	 * recvd. Thus, node's ELS IO pending list won't be
 | |
| 	 * empty (PLOGI will be on it)
 | |
| 	 */
 | |
| 	WARN_ON(node->send_ls_acc != EFC_NODE_SEND_LS_ACC_PLOGI);
 | |
| 	node_printf(node, "Reason: implicit logout, re-authenticate\n");
 | |
| 
 | |
| 	/* Re-attach node with the same HW node resources */
 | |
| 	node->req_free = false;
 | |
| 	rc = efc_node_attach(node);
 | |
| 	efc_node_transition(node, __efc_d_wait_node_attach, NULL);
 | |
| 	node->els_io_enabled = true;
 | |
| 
 | |
| 	if (rc < 0)
 | |
| 		efc_node_post_event(node, EFC_EVT_NODE_ATTACH_FAIL, NULL);
 | |
| }
 | |
| 
 | |
| static void efc_node_handle_explicit_logo(struct efc_node *node)
 | |
| {
 | |
| 	s8 pend_frames_empty;
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	/* cleanup any pending LS_ACC ELSs */
 | |
| 	efc_node_send_ls_io_cleanup(node);
 | |
| 
 | |
| 	spin_lock_irqsave(&node->pend_frames_lock, flags);
 | |
| 	pend_frames_empty = list_empty(&node->pend_frames);
 | |
| 	spin_unlock_irqrestore(&node->pend_frames_lock, flags);
 | |
| 
 | |
| 	/*
 | |
| 	 * there are two scenarios where we want to keep
 | |
| 	 * this node alive:
 | |
| 	 * 1. there are pending frames that need to be
 | |
| 	 *    processed or
 | |
| 	 * 2. we're an initiator and the remote node is
 | |
| 	 *    a target and we need to re-authenticate
 | |
| 	 */
 | |
| 	node_printf(node, "Shutdown: explicit logo pend=%d ", !pend_frames_empty);
 | |
| 	node_printf(node, "nport.ini=%d node.tgt=%d\n",
 | |
| 		    node->nport->enable_ini, node->targ);
 | |
| 	if (!pend_frames_empty || (node->nport->enable_ini && node->targ)) {
 | |
| 		u8 send_plogi = false;
 | |
| 
 | |
| 		if (node->nport->enable_ini && node->targ) {
 | |
| 			/*
 | |
| 			 * we're an initiator and
 | |
| 			 * node shutting down is a target;
 | |
| 			 * we'll need to re-authenticate in
 | |
| 			 * initial state
 | |
| 			 */
 | |
| 			send_plogi = true;
 | |
| 		}
 | |
| 
 | |
| 		/*
 | |
| 		 * transition to __efc_d_init
 | |
| 		 * (will retain HW node resources)
 | |
| 		 */
 | |
| 		node->els_io_enabled = true;
 | |
| 		node->req_free = false;
 | |
| 
 | |
| 		/*
 | |
| 		 * either pending frames exist or we are re-authenticating
 | |
| 		 * with PLOGI (or both); in either case, return to initial
 | |
| 		 * state
 | |
| 		 */
 | |
| 		efc_node_init_device(node, send_plogi);
 | |
| 	}
 | |
| 	/* else: let node shutdown occur */
 | |
| }
 | |
| 
 | |
| static void
 | |
| efc_node_purge_pending(struct efc_node *node)
 | |
| {
 | |
| 	struct efc *efc = node->efc;
 | |
| 	struct efc_hw_sequence *frame, *next;
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&node->pend_frames_lock, flags);
 | |
| 
 | |
| 	list_for_each_entry_safe(frame, next, &node->pend_frames, list_entry) {
 | |
| 		list_del(&frame->list_entry);
 | |
| 		efc->tt.hw_seq_free(efc, frame);
 | |
| 	}
 | |
| 
 | |
| 	spin_unlock_irqrestore(&node->pend_frames_lock, flags);
 | |
| }
 | |
| 
 | |
| void
 | |
| __efc_node_shutdown(struct efc_sm_ctx *ctx,
 | |
| 		    enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	struct efc_node *node = ctx->app;
 | |
| 
 | |
| 	efc_node_evt_set(ctx, evt, __func__);
 | |
| 
 | |
| 	node_sm_trace();
 | |
| 
 | |
| 	switch (evt) {
 | |
| 	case EFC_EVT_ENTER: {
 | |
| 		efc_node_hold_frames(node);
 | |
| 		WARN_ON(!efc_els_io_list_empty(node, &node->els_ios_list));
 | |
| 		/* by default, we will be freeing node after we unwind */
 | |
| 		node->req_free = true;
 | |
| 
 | |
| 		switch (node->shutdown_reason) {
 | |
| 		case EFC_NODE_SHUTDOWN_IMPLICIT_LOGO:
 | |
| 			/* Node shutdown b/c of PLOGI received when node
 | |
| 			 * already logged in. We have PLOGI service
 | |
| 			 * parameters, so submit node attach; we won't be
 | |
| 			 * freeing this node
 | |
| 			 */
 | |
| 
 | |
| 			efc_node_handle_implicit_logo(node);
 | |
| 			break;
 | |
| 
 | |
| 		case EFC_NODE_SHUTDOWN_EXPLICIT_LOGO:
 | |
| 			efc_node_handle_explicit_logo(node);
 | |
| 			break;
 | |
| 
 | |
| 		case EFC_NODE_SHUTDOWN_DEFAULT:
 | |
| 		default: {
 | |
| 			/*
 | |
| 			 * shutdown due to link down,
 | |
| 			 * node going away (xport event) or
 | |
| 			 * nport shutdown, purge pending and
 | |
| 			 * proceed to cleanup node
 | |
| 			 */
 | |
| 
 | |
| 			/* cleanup any pending LS_ACC ELSs */
 | |
| 			efc_node_send_ls_io_cleanup(node);
 | |
| 
 | |
| 			node_printf(node,
 | |
| 				    "Shutdown reason: default, purge pending\n");
 | |
| 			efc_node_purge_pending(node);
 | |
| 			break;
 | |
| 		}
 | |
| 		}
 | |
| 
 | |
| 		break;
 | |
| 	}
 | |
| 	case EFC_EVT_EXIT:
 | |
| 		efc_node_accept_frames(node);
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		__efc_node_common(__func__, ctx, evt, arg);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static bool
 | |
| efc_node_check_els_quiesced(struct efc_node *node)
 | |
| {
 | |
| 	/* check to see if ELS requests, completions are quiesced */
 | |
| 	if (node->els_req_cnt == 0 && node->els_cmpl_cnt == 0 &&
 | |
| 	    efc_els_io_list_empty(node, &node->els_ios_list)) {
 | |
| 		if (!node->attached) {
 | |
| 			/* hw node detach already completed, proceed */
 | |
| 			node_printf(node, "HW node not attached\n");
 | |
| 			efc_node_transition(node,
 | |
| 					    __efc_node_wait_ios_shutdown,
 | |
| 					     NULL);
 | |
| 		} else {
 | |
| 			/*
 | |
| 			 * hw node detach hasn't completed,
 | |
| 			 * transition and wait
 | |
| 			 */
 | |
| 			node_printf(node, "HW node still attached\n");
 | |
| 			efc_node_transition(node, __efc_node_wait_node_free,
 | |
| 					    NULL);
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_initiate_cleanup(struct efc_node *node)
 | |
| {
 | |
| 	/*
 | |
| 	 * if ELS's have already been quiesced, will move to next state
 | |
| 	 * if ELS's have not been quiesced, abort them
 | |
| 	 */
 | |
| 	if (!efc_node_check_els_quiesced(node)) {
 | |
| 		efc_node_hold_frames(node);
 | |
| 		efc_node_transition(node, __efc_node_wait_els_shutdown, NULL);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| __efc_node_wait_els_shutdown(struct efc_sm_ctx *ctx,
 | |
| 			     enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	bool check_quiesce = false;
 | |
| 	struct efc_node *node = ctx->app;
 | |
| 
 | |
| 	efc_node_evt_set(ctx, evt, __func__);
 | |
| 
 | |
| 	node_sm_trace();
 | |
| 	/* Node state machine: Wait for all ELSs to complete */
 | |
| 	switch (evt) {
 | |
| 	case EFC_EVT_ENTER:
 | |
| 		efc_node_hold_frames(node);
 | |
| 		if (efc_els_io_list_empty(node, &node->els_ios_list)) {
 | |
| 			node_printf(node, "All ELS IOs complete\n");
 | |
| 			check_quiesce = true;
 | |
| 		}
 | |
| 		break;
 | |
| 	case EFC_EVT_EXIT:
 | |
| 		efc_node_accept_frames(node);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_OK:
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_FAIL:
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_RJT:
 | |
| 	case EFC_EVT_ELS_REQ_ABORTED:
 | |
| 		if (WARN_ON(!node->els_req_cnt))
 | |
| 			break;
 | |
| 		node->els_req_cnt--;
 | |
| 		check_quiesce = true;
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_SRRS_ELS_CMPL_OK:
 | |
| 	case EFC_EVT_SRRS_ELS_CMPL_FAIL:
 | |
| 		if (WARN_ON(!node->els_cmpl_cnt))
 | |
| 			break;
 | |
| 		node->els_cmpl_cnt--;
 | |
| 		check_quiesce = true;
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_ALL_CHILD_NODES_FREE:
 | |
| 		/* all ELS IO's complete */
 | |
| 		node_printf(node, "All ELS IOs complete\n");
 | |
| 		WARN_ON(!efc_els_io_list_empty(node, &node->els_ios_list));
 | |
| 		check_quiesce = true;
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_NODE_ACTIVE_IO_LIST_EMPTY:
 | |
| 		check_quiesce = true;
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_DOMAIN_ATTACH_OK:
 | |
| 		/* don't care about domain_attach_ok */
 | |
| 		break;
 | |
| 
 | |
| 	/* ignore shutdown events as we're already in shutdown path */
 | |
| 	case EFC_EVT_SHUTDOWN:
 | |
| 		/* have default shutdown event take precedence */
 | |
| 		node->shutdown_reason = EFC_NODE_SHUTDOWN_DEFAULT;
 | |
| 		fallthrough;
 | |
| 
 | |
| 	case EFC_EVT_SHUTDOWN_EXPLICIT_LOGO:
 | |
| 	case EFC_EVT_SHUTDOWN_IMPLICIT_LOGO:
 | |
| 		node_printf(node, "%s received\n", efc_sm_event_name(evt));
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		__efc_node_common(__func__, ctx, evt, arg);
 | |
| 	}
 | |
| 
 | |
| 	if (check_quiesce)
 | |
| 		efc_node_check_els_quiesced(node);
 | |
| }
 | |
| 
 | |
| void
 | |
| __efc_node_wait_node_free(struct efc_sm_ctx *ctx,
 | |
| 			  enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	struct efc_node *node = ctx->app;
 | |
| 
 | |
| 	efc_node_evt_set(ctx, evt, __func__);
 | |
| 
 | |
| 	node_sm_trace();
 | |
| 
 | |
| 	switch (evt) {
 | |
| 	case EFC_EVT_ENTER:
 | |
| 		efc_node_hold_frames(node);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_EXIT:
 | |
| 		efc_node_accept_frames(node);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_NODE_FREE_OK:
 | |
| 		/* node is officially no longer attached */
 | |
| 		node->attached = false;
 | |
| 		efc_node_transition(node, __efc_node_wait_ios_shutdown, NULL);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_ALL_CHILD_NODES_FREE:
 | |
| 	case EFC_EVT_NODE_ACTIVE_IO_LIST_EMPTY:
 | |
| 		/* As IOs and ELS IO's complete we expect to get these events */
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_DOMAIN_ATTACH_OK:
 | |
| 		/* don't care about domain_attach_ok */
 | |
| 		break;
 | |
| 
 | |
| 	/* ignore shutdown events as we're already in shutdown path */
 | |
| 	case EFC_EVT_SHUTDOWN:
 | |
| 		/* have default shutdown event take precedence */
 | |
| 		node->shutdown_reason = EFC_NODE_SHUTDOWN_DEFAULT;
 | |
| 		fallthrough;
 | |
| 
 | |
| 	case EFC_EVT_SHUTDOWN_EXPLICIT_LOGO:
 | |
| 	case EFC_EVT_SHUTDOWN_IMPLICIT_LOGO:
 | |
| 		node_printf(node, "%s received\n", efc_sm_event_name(evt));
 | |
| 		break;
 | |
| 	default:
 | |
| 		__efc_node_common(__func__, ctx, evt, arg);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| __efc_node_wait_ios_shutdown(struct efc_sm_ctx *ctx,
 | |
| 			     enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	struct efc_node *node = ctx->app;
 | |
| 	struct efc *efc = node->efc;
 | |
| 
 | |
| 	efc_node_evt_set(ctx, evt, __func__);
 | |
| 
 | |
| 	node_sm_trace();
 | |
| 
 | |
| 	switch (evt) {
 | |
| 	case EFC_EVT_ENTER:
 | |
| 		efc_node_hold_frames(node);
 | |
| 
 | |
| 		/* first check to see if no ELS IOs are outstanding */
 | |
| 		if (efc_els_io_list_empty(node, &node->els_ios_list))
 | |
| 			/* If there are any active IOS, Free them. */
 | |
| 			efc_node_transition(node, __efc_node_shutdown, NULL);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_NODE_ACTIVE_IO_LIST_EMPTY:
 | |
| 	case EFC_EVT_ALL_CHILD_NODES_FREE:
 | |
| 		if (efc_els_io_list_empty(node, &node->els_ios_list))
 | |
| 			efc_node_transition(node, __efc_node_shutdown, NULL);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_EXIT:
 | |
| 		efc_node_accept_frames(node);
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_FAIL:
 | |
| 		/* Can happen as ELS IO IO's complete */
 | |
| 		if (WARN_ON(!node->els_req_cnt))
 | |
| 			break;
 | |
| 		node->els_req_cnt--;
 | |
| 		break;
 | |
| 
 | |
| 	/* ignore shutdown events as we're already in shutdown path */
 | |
| 	case EFC_EVT_SHUTDOWN:
 | |
| 		/* have default shutdown event take precedence */
 | |
| 		node->shutdown_reason = EFC_NODE_SHUTDOWN_DEFAULT;
 | |
| 		fallthrough;
 | |
| 
 | |
| 	case EFC_EVT_SHUTDOWN_EXPLICIT_LOGO:
 | |
| 	case EFC_EVT_SHUTDOWN_IMPLICIT_LOGO:
 | |
| 		efc_log_debug(efc, "[%s] %-20s\n", node->display_name,
 | |
| 			      efc_sm_event_name(evt));
 | |
| 		break;
 | |
| 	case EFC_EVT_DOMAIN_ATTACH_OK:
 | |
| 		/* don't care about domain_attach_ok */
 | |
| 		break;
 | |
| 	default:
 | |
| 		__efc_node_common(__func__, ctx, evt, arg);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| __efc_node_common(const char *funcname, struct efc_sm_ctx *ctx,
 | |
| 		  enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	struct efc_node *node = NULL;
 | |
| 	struct efc *efc = NULL;
 | |
| 	struct efc_node_cb *cbdata = arg;
 | |
| 
 | |
| 	node = ctx->app;
 | |
| 	efc = node->efc;
 | |
| 
 | |
| 	switch (evt) {
 | |
| 	case EFC_EVT_ENTER:
 | |
| 	case EFC_EVT_REENTER:
 | |
| 	case EFC_EVT_EXIT:
 | |
| 	case EFC_EVT_NPORT_TOPOLOGY_NOTIFY:
 | |
| 	case EFC_EVT_NODE_MISSING:
 | |
| 	case EFC_EVT_FCP_CMD_RCVD:
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_NODE_REFOUND:
 | |
| 		node->refound = true;
 | |
| 		break;
 | |
| 
 | |
| 	/*
 | |
| 	 * node->attached must be set appropriately
 | |
| 	 * for all node attach/detach events
 | |
| 	 */
 | |
| 	case EFC_EVT_NODE_ATTACH_OK:
 | |
| 		node->attached = true;
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_NODE_FREE_OK:
 | |
| 	case EFC_EVT_NODE_ATTACH_FAIL:
 | |
| 		node->attached = false;
 | |
| 		break;
 | |
| 
 | |
| 	/*
 | |
| 	 * handle any ELS completions that
 | |
| 	 * other states either didn't care about
 | |
| 	 * or forgot about
 | |
| 	 */
 | |
| 	case EFC_EVT_SRRS_ELS_CMPL_OK:
 | |
| 	case EFC_EVT_SRRS_ELS_CMPL_FAIL:
 | |
| 		if (WARN_ON(!node->els_cmpl_cnt))
 | |
| 			break;
 | |
| 		node->els_cmpl_cnt--;
 | |
| 		break;
 | |
| 
 | |
| 	/*
 | |
| 	 * handle any ELS request completions that
 | |
| 	 * other states either didn't care about
 | |
| 	 * or forgot about
 | |
| 	 */
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_OK:
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_FAIL:
 | |
| 	case EFC_EVT_SRRS_ELS_REQ_RJT:
 | |
| 	case EFC_EVT_ELS_REQ_ABORTED:
 | |
| 		if (WARN_ON(!node->els_req_cnt))
 | |
| 			break;
 | |
| 		node->els_req_cnt--;
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_ELS_RCVD: {
 | |
| 		struct fc_frame_header *hdr = cbdata->header->dma.virt;
 | |
| 
 | |
| 		/*
 | |
| 		 * Unsupported ELS was received,
 | |
| 		 * send LS_RJT, command not supported
 | |
| 		 */
 | |
| 		efc_log_debug(efc,
 | |
| 			      "[%s] (%s) ELS x%02x, LS_RJT not supported\n",
 | |
| 			      node->display_name, funcname,
 | |
| 			      ((u8 *)cbdata->payload->dma.virt)[0]);
 | |
| 
 | |
| 		efc_send_ls_rjt(node, be16_to_cpu(hdr->fh_ox_id),
 | |
| 				ELS_RJT_UNSUP, ELS_EXPL_NONE, 0);
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	case EFC_EVT_PLOGI_RCVD:
 | |
| 	case EFC_EVT_FLOGI_RCVD:
 | |
| 	case EFC_EVT_LOGO_RCVD:
 | |
| 	case EFC_EVT_PRLI_RCVD:
 | |
| 	case EFC_EVT_PRLO_RCVD:
 | |
| 	case EFC_EVT_PDISC_RCVD:
 | |
| 	case EFC_EVT_FDISC_RCVD:
 | |
| 	case EFC_EVT_ADISC_RCVD:
 | |
| 	case EFC_EVT_RSCN_RCVD:
 | |
| 	case EFC_EVT_SCR_RCVD: {
 | |
| 		struct fc_frame_header *hdr = cbdata->header->dma.virt;
 | |
| 
 | |
| 		/* sm: / send ELS_RJT */
 | |
| 		efc_log_debug(efc, "[%s] (%s) %s sending ELS_RJT\n",
 | |
| 			      node->display_name, funcname,
 | |
| 			      efc_sm_event_name(evt));
 | |
| 		/* if we didn't catch this in a state, send generic LS_RJT */
 | |
| 		efc_send_ls_rjt(node, be16_to_cpu(hdr->fh_ox_id),
 | |
| 				ELS_RJT_UNAB, ELS_EXPL_NONE, 0);
 | |
| 		break;
 | |
| 	}
 | |
| 	case EFC_EVT_ABTS_RCVD: {
 | |
| 		efc_log_debug(efc, "[%s] (%s) %s sending BA_ACC\n",
 | |
| 			      node->display_name, funcname,
 | |
| 			      efc_sm_event_name(evt));
 | |
| 
 | |
| 		/* sm: / send BA_ACC */
 | |
| 		efc_send_bls_acc(node, cbdata->header->dma.virt);
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	default:
 | |
| 		efc_log_debug(node->efc, "[%s] %-20s %-20s not handled\n",
 | |
| 			      node->display_name, funcname,
 | |
| 			      efc_sm_event_name(evt));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_save_sparms(struct efc_node *node, void *payload)
 | |
| {
 | |
| 	memcpy(node->service_params, payload, sizeof(node->service_params));
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_post_event(struct efc_node *node,
 | |
| 		    enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	bool free_node = false;
 | |
| 
 | |
| 	node->evtdepth++;
 | |
| 
 | |
| 	efc_sm_post_event(&node->sm, evt, arg);
 | |
| 
 | |
| 	/* If our event call depth is one and
 | |
| 	 * we're not holding frames
 | |
| 	 * then we can dispatch any pending frames.
 | |
| 	 * We don't want to allow the efc_process_node_pending()
 | |
| 	 * call to recurse.
 | |
| 	 */
 | |
| 	if (!node->hold_frames && node->evtdepth == 1)
 | |
| 		efc_process_node_pending(node);
 | |
| 
 | |
| 	node->evtdepth--;
 | |
| 
 | |
| 	/*
 | |
| 	 * Free the node object if so requested,
 | |
| 	 * and we're at an event call depth of zero
 | |
| 	 */
 | |
| 	if (node->evtdepth == 0 && node->req_free)
 | |
| 		free_node = true;
 | |
| 
 | |
| 	if (free_node)
 | |
| 		efc_node_free(node);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_transition(struct efc_node *node,
 | |
| 		    void (*state)(struct efc_sm_ctx *,
 | |
| 				  enum efc_sm_event, void *), void *data)
 | |
| {
 | |
| 	struct efc_sm_ctx *ctx = &node->sm;
 | |
| 
 | |
| 	if (ctx->current_state == state) {
 | |
| 		efc_node_post_event(node, EFC_EVT_REENTER, data);
 | |
| 	} else {
 | |
| 		efc_node_post_event(node, EFC_EVT_EXIT, data);
 | |
| 		ctx->current_state = state;
 | |
| 		efc_node_post_event(node, EFC_EVT_ENTER, data);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_build_eui_name(char *buf, u32 buf_len, uint64_t eui_name)
 | |
| {
 | |
| 	memset(buf, 0, buf_len);
 | |
| 
 | |
| 	snprintf(buf, buf_len, "eui.%016llX", (unsigned long long)eui_name);
 | |
| }
 | |
| 
 | |
| u64
 | |
| efc_node_get_wwpn(struct efc_node *node)
 | |
| {
 | |
| 	struct fc_els_flogi *sp =
 | |
| 			(struct fc_els_flogi *)node->service_params;
 | |
| 
 | |
| 	return be64_to_cpu(sp->fl_wwpn);
 | |
| }
 | |
| 
 | |
| u64
 | |
| efc_node_get_wwnn(struct efc_node *node)
 | |
| {
 | |
| 	struct fc_els_flogi *sp =
 | |
| 			(struct fc_els_flogi *)node->service_params;
 | |
| 
 | |
| 	return be64_to_cpu(sp->fl_wwnn);
 | |
| }
 | |
| 
 | |
| int
 | |
| efc_node_check_els_req(struct efc_sm_ctx *ctx, enum efc_sm_event evt, void *arg,
 | |
| 		u8 cmd, void (*efc_node_common_func)(const char *,
 | |
| 				struct efc_sm_ctx *, enum efc_sm_event, void *),
 | |
| 		const char *funcname)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int
 | |
| efc_node_check_ns_req(struct efc_sm_ctx *ctx, enum efc_sm_event evt, void *arg,
 | |
| 		u16 cmd, void (*efc_node_common_func)(const char *,
 | |
| 				struct efc_sm_ctx *, enum efc_sm_event, void *),
 | |
| 		const char *funcname)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int
 | |
| efc_els_io_list_empty(struct efc_node *node, struct list_head *list)
 | |
| {
 | |
| 	int empty;
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&node->els_ios_lock, flags);
 | |
| 	empty = list_empty(list);
 | |
| 	spin_unlock_irqrestore(&node->els_ios_lock, flags);
 | |
| 	return empty;
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_pause(struct efc_node *node,
 | |
| 	       void (*state)(struct efc_sm_ctx *,
 | |
| 			     enum efc_sm_event, void *))
 | |
| 
 | |
| {
 | |
| 	node->nodedb_state = state;
 | |
| 	efc_node_transition(node, __efc_node_paused, NULL);
 | |
| }
 | |
| 
 | |
| void
 | |
| __efc_node_paused(struct efc_sm_ctx *ctx,
 | |
| 		  enum efc_sm_event evt, void *arg)
 | |
| {
 | |
| 	struct efc_node *node = ctx->app;
 | |
| 
 | |
| 	efc_node_evt_set(ctx, evt, __func__);
 | |
| 
 | |
| 	node_sm_trace();
 | |
| 
 | |
| 	/*
 | |
| 	 * This state is entered when a state is "paused". When resumed, the
 | |
| 	 * node is transitioned to a previously saved state (node->ndoedb_state)
 | |
| 	 */
 | |
| 	switch (evt) {
 | |
| 	case EFC_EVT_ENTER:
 | |
| 		node_printf(node, "Paused\n");
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_RESUME: {
 | |
| 		void (*pf)(struct efc_sm_ctx *ctx,
 | |
| 			   enum efc_sm_event evt, void *arg);
 | |
| 
 | |
| 		pf = node->nodedb_state;
 | |
| 
 | |
| 		node->nodedb_state = NULL;
 | |
| 		efc_node_transition(node, pf, NULL);
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	case EFC_EVT_DOMAIN_ATTACH_OK:
 | |
| 		break;
 | |
| 
 | |
| 	case EFC_EVT_SHUTDOWN:
 | |
| 		node->req_free = true;
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		__efc_node_common(__func__, ctx, evt, arg);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_recv_els_frame(struct efc_node *node,
 | |
| 			struct efc_hw_sequence *seq)
 | |
| {
 | |
| 	u32 prli_size = sizeof(struct fc_els_prli) + sizeof(struct fc_els_spp);
 | |
| 	struct {
 | |
| 		u32 cmd;
 | |
| 		enum efc_sm_event evt;
 | |
| 		u32 payload_size;
 | |
| 	} els_cmd_list[] = {
 | |
| 		{ELS_PLOGI, EFC_EVT_PLOGI_RCVD,	sizeof(struct fc_els_flogi)},
 | |
| 		{ELS_FLOGI, EFC_EVT_FLOGI_RCVD,	sizeof(struct fc_els_flogi)},
 | |
| 		{ELS_LOGO, EFC_EVT_LOGO_RCVD, sizeof(struct fc_els_ls_acc)},
 | |
| 		{ELS_PRLI, EFC_EVT_PRLI_RCVD, prli_size},
 | |
| 		{ELS_PRLO, EFC_EVT_PRLO_RCVD, prli_size},
 | |
| 		{ELS_PDISC, EFC_EVT_PDISC_RCVD,	MAX_ACC_REJECT_PAYLOAD},
 | |
| 		{ELS_FDISC, EFC_EVT_FDISC_RCVD,	MAX_ACC_REJECT_PAYLOAD},
 | |
| 		{ELS_ADISC, EFC_EVT_ADISC_RCVD,	sizeof(struct fc_els_adisc)},
 | |
| 		{ELS_RSCN, EFC_EVT_RSCN_RCVD, MAX_ACC_REJECT_PAYLOAD},
 | |
| 		{ELS_SCR, EFC_EVT_SCR_RCVD, MAX_ACC_REJECT_PAYLOAD},
 | |
| 	};
 | |
| 	struct efc_node_cb cbdata;
 | |
| 	u8 *buf = seq->payload->dma.virt;
 | |
| 	enum efc_sm_event evt = EFC_EVT_ELS_RCVD;
 | |
| 	u32 i;
 | |
| 
 | |
| 	memset(&cbdata, 0, sizeof(cbdata));
 | |
| 	cbdata.header = seq->header;
 | |
| 	cbdata.payload = seq->payload;
 | |
| 
 | |
| 	/* find a matching event for the ELS command */
 | |
| 	for (i = 0; i < ARRAY_SIZE(els_cmd_list); i++) {
 | |
| 		if (els_cmd_list[i].cmd == buf[0]) {
 | |
| 			evt = els_cmd_list[i].evt;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	efc_node_post_event(node, evt, &cbdata);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_recv_ct_frame(struct efc_node *node,
 | |
| 		       struct efc_hw_sequence *seq)
 | |
| {
 | |
| 	struct fc_ct_hdr *iu = seq->payload->dma.virt;
 | |
| 	struct fc_frame_header *hdr = seq->header->dma.virt;
 | |
| 	struct efc *efc = node->efc;
 | |
| 	u16 gscmd = be16_to_cpu(iu->ct_cmd);
 | |
| 
 | |
| 	efc_log_err(efc, "[%s] Received cmd :%x sending CT_REJECT\n",
 | |
| 		    node->display_name, gscmd);
 | |
| 	efc_send_ct_rsp(efc, node, be16_to_cpu(hdr->fh_ox_id), iu,
 | |
| 			FC_FS_RJT, FC_FS_RJT_UNSUP, 0);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_node_recv_fcp_cmd(struct efc_node *node, struct efc_hw_sequence *seq)
 | |
| {
 | |
| 	struct efc_node_cb cbdata;
 | |
| 
 | |
| 	memset(&cbdata, 0, sizeof(cbdata));
 | |
| 	cbdata.header = seq->header;
 | |
| 	cbdata.payload = seq->payload;
 | |
| 
 | |
| 	efc_node_post_event(node, EFC_EVT_FCP_CMD_RCVD, &cbdata);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_process_node_pending(struct efc_node *node)
 | |
| {
 | |
| 	struct efc *efc = node->efc;
 | |
| 	struct efc_hw_sequence *seq = NULL;
 | |
| 	u32 pend_frames_processed = 0;
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	for (;;) {
 | |
| 		/* need to check for hold frames condition after each frame
 | |
| 		 * processed because any given frame could cause a transition
 | |
| 		 * to a state that holds frames
 | |
| 		 */
 | |
| 		if (node->hold_frames)
 | |
| 			break;
 | |
| 
 | |
| 		seq = NULL;
 | |
| 		/* Get next frame/sequence */
 | |
| 		spin_lock_irqsave(&node->pend_frames_lock, flags);
 | |
| 
 | |
| 		if (!list_empty(&node->pend_frames)) {
 | |
| 			seq = list_first_entry(&node->pend_frames,
 | |
| 					struct efc_hw_sequence, list_entry);
 | |
| 			list_del(&seq->list_entry);
 | |
| 		}
 | |
| 		spin_unlock_irqrestore(&node->pend_frames_lock, flags);
 | |
| 
 | |
| 		if (!seq) {
 | |
| 			pend_frames_processed =	node->pend_frames_processed;
 | |
| 			node->pend_frames_processed = 0;
 | |
| 			break;
 | |
| 		}
 | |
| 		node->pend_frames_processed++;
 | |
| 
 | |
| 		/* now dispatch frame(s) to dispatch function */
 | |
| 		efc_node_dispatch_frame(node, seq);
 | |
| 		efc->tt.hw_seq_free(efc, seq);
 | |
| 	}
 | |
| 
 | |
| 	if (pend_frames_processed != 0)
 | |
| 		efc_log_debug(efc, "%u node frames held and processed\n",
 | |
| 			      pend_frames_processed);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_scsi_sess_reg_complete(struct efc_node *node, u32 status)
 | |
| {
 | |
| 	unsigned long flags = 0;
 | |
| 	enum efc_sm_event evt = EFC_EVT_NODE_SESS_REG_OK;
 | |
| 	struct efc *efc = node->efc;
 | |
| 
 | |
| 	if (status)
 | |
| 		evt = EFC_EVT_NODE_SESS_REG_FAIL;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	/* Notify the node to resume */
 | |
| 	efc_node_post_event(node, evt, NULL);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_scsi_del_initiator_complete(struct efc *efc, struct efc_node *node)
 | |
| {
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	/* Notify the node to resume */
 | |
| 	efc_node_post_event(node, EFC_EVT_NODE_DEL_INI_COMPLETE, NULL);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_scsi_del_target_complete(struct efc *efc, struct efc_node *node)
 | |
| {
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	/* Notify the node to resume */
 | |
| 	efc_node_post_event(node, EFC_EVT_NODE_DEL_TGT_COMPLETE, NULL);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| }
 | |
| 
 | |
| void
 | |
| efc_scsi_io_list_empty(struct efc *efc, struct efc_node *node)
 | |
| {
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	efc_node_post_event(node, EFC_EVT_NODE_ACTIVE_IO_LIST_EMPTY, NULL);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| }
 | |
| 
 | |
| void efc_node_post_els_resp(struct efc_node *node, u32 evt, void *arg)
 | |
| {
 | |
| 	struct efc *efc = node->efc;
 | |
| 	unsigned long flags = 0;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	efc_node_post_event(node, evt, arg);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| }
 | |
| 
 | |
| void efc_node_post_shutdown(struct efc_node *node, void *arg)
 | |
| {
 | |
| 	unsigned long flags = 0;
 | |
| 	struct efc *efc = node->efc;
 | |
| 
 | |
| 	spin_lock_irqsave(&efc->lock, flags);
 | |
| 	efc_node_post_event(node, EFC_EVT_SHUTDOWN, arg);
 | |
| 	spin_unlock_irqrestore(&efc->lock, flags);
 | |
| }
 |