188f3b62aSDan Williams /*
288f3b62aSDan Williams * This file is provided under a dual BSD/GPLv2 license. When using or
388f3b62aSDan Williams * redistributing this file, you may do so under either license.
488f3b62aSDan Williams *
588f3b62aSDan Williams * GPL LICENSE SUMMARY
688f3b62aSDan Williams *
788f3b62aSDan Williams * Copyright(c) 2008 - 2011 Intel Corporation. All rights reserved.
888f3b62aSDan Williams *
988f3b62aSDan Williams * This program is free software; you can redistribute it and/or modify
1088f3b62aSDan Williams * it under the terms of version 2 of the GNU General Public License as
1188f3b62aSDan Williams * published by the Free Software Foundation.
1288f3b62aSDan Williams *
1388f3b62aSDan Williams * This program is distributed in the hope that it will be useful, but
1488f3b62aSDan Williams * WITHOUT ANY WARRANTY; without even the implied warranty of
1588f3b62aSDan Williams * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1688f3b62aSDan Williams * General Public License for more details.
1788f3b62aSDan Williams *
1888f3b62aSDan Williams * You should have received a copy of the GNU General Public License
1988f3b62aSDan Williams * along with this program; if not, write to the Free Software
2088f3b62aSDan Williams * Foundation, Inc., 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
2188f3b62aSDan Williams * The full GNU General Public License is included in this distribution
2288f3b62aSDan Williams * in the file called LICENSE.GPL.
2388f3b62aSDan Williams *
2488f3b62aSDan Williams * BSD LICENSE
2588f3b62aSDan Williams *
2688f3b62aSDan Williams * Copyright(c) 2008 - 2011 Intel Corporation. All rights reserved.
2788f3b62aSDan Williams * All rights reserved.
2888f3b62aSDan Williams *
2988f3b62aSDan Williams * Redistribution and use in source and binary forms, with or without
3088f3b62aSDan Williams * modification, are permitted provided that the following conditions
3188f3b62aSDan Williams * are met:
3288f3b62aSDan Williams *
3388f3b62aSDan Williams * * Redistributions of source code must retain the above copyright
3488f3b62aSDan Williams * notice, this list of conditions and the following disclaimer.
3588f3b62aSDan Williams * * Redistributions in binary form must reproduce the above copyright
3688f3b62aSDan Williams * notice, this list of conditions and the following disclaimer in
3788f3b62aSDan Williams * the documentation and/or other materials provided with the
3888f3b62aSDan Williams * distribution.
3988f3b62aSDan Williams * * Neither the name of Intel Corporation nor the names of its
4088f3b62aSDan Williams * contributors may be used to endorse or promote products derived
4188f3b62aSDan Williams * from this software without specific prior written permission.
4288f3b62aSDan Williams *
4388f3b62aSDan Williams * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
4488f3b62aSDan Williams * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
4588f3b62aSDan Williams * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
4688f3b62aSDan Williams * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
4788f3b62aSDan Williams * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
4888f3b62aSDan Williams * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
4988f3b62aSDan Williams * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
5088f3b62aSDan Williams * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
5188f3b62aSDan Williams * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
5288f3b62aSDan Williams * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
5388f3b62aSDan Williams * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
5488f3b62aSDan Williams */
55c94fc1adSJeff Skirvin #include <scsi/sas_ata.h>
56cc9203bfSDan Williams #include "host.h"
5712ef6544SEdmund Nadolski #include "isci.h"
5888f3b62aSDan Williams #include "remote_device.h"
5988f3b62aSDan Williams #include "remote_node_context.h"
6088f3b62aSDan Williams #include "scu_event_codes.h"
6188f3b62aSDan Williams #include "scu_task_context.h"
6288f3b62aSDan Williams
63d7a0ccddSDan Williams #undef C
64d7a0ccddSDan Williams #define C(a) (#a)
rnc_state_name(enum scis_sds_remote_node_context_states state)65d7a0ccddSDan Williams const char *rnc_state_name(enum scis_sds_remote_node_context_states state)
66d7a0ccddSDan Williams {
67d7a0ccddSDan Williams static const char * const strings[] = RNC_STATES;
6888f3b62aSDan Williams
695cfa2a3cSArnd Bergmann if (state >= ARRAY_SIZE(strings))
705cfa2a3cSArnd Bergmann return "UNKNOWN";
715cfa2a3cSArnd Bergmann
72d7a0ccddSDan Williams return strings[state];
73d7a0ccddSDan Williams }
74d7a0ccddSDan Williams #undef C
7588f3b62aSDan Williams
7688f3b62aSDan Williams /**
77ad276048SLee Jones * sci_remote_node_context_is_ready()
7888f3b62aSDan Williams * @sci_rnc: The state of the remote node context object to check.
7988f3b62aSDan Williams *
8088f3b62aSDan Williams * This method will return true if the remote node context is in a READY state
8188f3b62aSDan Williams * otherwise it will return false bool true if the remote node context is in
8288f3b62aSDan Williams * the ready state. false if the remote node context is not in the ready state.
8388f3b62aSDan Williams */
sci_remote_node_context_is_ready(struct sci_remote_node_context * sci_rnc)8489a7301fSDan Williams bool sci_remote_node_context_is_ready(
8589a7301fSDan Williams struct sci_remote_node_context *sci_rnc)
8688f3b62aSDan Williams {
87e301370aSEdmund Nadolski u32 current_state = sci_rnc->sm.current_state_id;
8888f3b62aSDan Williams
89e301370aSEdmund Nadolski if (current_state == SCI_RNC_READY) {
9088f3b62aSDan Williams return true;
9188f3b62aSDan Williams }
9288f3b62aSDan Williams
9388f3b62aSDan Williams return false;
9488f3b62aSDan Williams }
9588f3b62aSDan Williams
sci_remote_node_context_is_suspended(struct sci_remote_node_context * sci_rnc)9631a38ef0SJeff Skirvin bool sci_remote_node_context_is_suspended(struct sci_remote_node_context *sci_rnc)
9731a38ef0SJeff Skirvin {
9831a38ef0SJeff Skirvin u32 current_state = sci_rnc->sm.current_state_id;
9931a38ef0SJeff Skirvin
10031a38ef0SJeff Skirvin if (current_state == SCI_RNC_TX_RX_SUSPENDED)
10131a38ef0SJeff Skirvin return true;
10231a38ef0SJeff Skirvin return false;
10331a38ef0SJeff Skirvin }
10431a38ef0SJeff Skirvin
sci_rnc_by_id(struct isci_host * ihost,u16 id)10589a7301fSDan Williams static union scu_remote_node_context *sci_rnc_by_id(struct isci_host *ihost, u16 id)
10689a7301fSDan Williams {
10789a7301fSDan Williams if (id < ihost->remote_node_entries &&
10889a7301fSDan Williams ihost->device_table[id])
10989a7301fSDan Williams return &ihost->remote_node_context_table[id];
11089a7301fSDan Williams
11189a7301fSDan Williams return NULL;
11289a7301fSDan Williams }
11389a7301fSDan Williams
sci_remote_node_context_construct_buffer(struct sci_remote_node_context * sci_rnc)11489a7301fSDan Williams static void sci_remote_node_context_construct_buffer(struct sci_remote_node_context *sci_rnc)
11588f3b62aSDan Williams {
11678a6f06eSDan Williams struct isci_remote_device *idev = rnc_to_dev(sci_rnc);
11778a6f06eSDan Williams struct domain_device *dev = idev->domain_dev;
1181f4fa1f9SDan Williams int rni = sci_rnc->remote_node_index;
119a1a113b0SDan Williams union scu_remote_node_context *rnc;
120d9dcb4baSDan Williams struct isci_host *ihost;
121a3d568f0SDan Williams __le64 sas_addr;
12288f3b62aSDan Williams
12334a99158SDan Williams ihost = idev->owning_port->owning_controller;
12489a7301fSDan Williams rnc = sci_rnc_by_id(ihost, rni);
12588f3b62aSDan Williams
12688f3b62aSDan Williams memset(rnc, 0, sizeof(union scu_remote_node_context)
12789a7301fSDan Williams * sci_remote_device_node_count(idev));
12888f3b62aSDan Williams
1291f4fa1f9SDan Williams rnc->ssp.remote_node_index = rni;
13078a6f06eSDan Williams rnc->ssp.remote_node_port_width = idev->device_port_width;
13178a6f06eSDan Williams rnc->ssp.logical_port_index = idev->owning_port->physical_port_index;
13288f3b62aSDan Williams
133a3d568f0SDan Williams /* sas address is __be64, context ram format is __le64 */
134a3d568f0SDan Williams sas_addr = cpu_to_le64(SAS_ADDR(dev->sas_addr));
135a3d568f0SDan Williams rnc->ssp.remote_sas_address_hi = upper_32_bits(sas_addr);
136a3d568f0SDan Williams rnc->ssp.remote_sas_address_lo = lower_32_bits(sas_addr);
13788f3b62aSDan Williams
13888f3b62aSDan Williams rnc->ssp.nexus_loss_timer_enable = true;
13988f3b62aSDan Williams rnc->ssp.check_bit = false;
14088f3b62aSDan Williams rnc->ssp.is_valid = false;
14188f3b62aSDan Williams rnc->ssp.is_remote_node_context = true;
14288f3b62aSDan Williams rnc->ssp.function_number = 0;
14388f3b62aSDan Williams
14488f3b62aSDan Williams rnc->ssp.arbitration_wait_time = 0;
14588f3b62aSDan Williams
14611cc5183SDan Williams if (dev_is_sata(dev)) {
14788f3b62aSDan Williams rnc->ssp.connection_occupancy_timeout =
14889a7301fSDan Williams ihost->user_parameters.stp_max_occupancy_timeout;
14988f3b62aSDan Williams rnc->ssp.connection_inactivity_timeout =
15089a7301fSDan Williams ihost->user_parameters.stp_inactivity_timeout;
15188f3b62aSDan Williams } else {
15288f3b62aSDan Williams rnc->ssp.connection_occupancy_timeout =
15389a7301fSDan Williams ihost->user_parameters.ssp_max_occupancy_timeout;
15488f3b62aSDan Williams rnc->ssp.connection_inactivity_timeout =
15589a7301fSDan Williams ihost->user_parameters.ssp_inactivity_timeout;
15688f3b62aSDan Williams }
15788f3b62aSDan Williams
15888f3b62aSDan Williams rnc->ssp.initial_arbitration_wait_time = 0;
15988f3b62aSDan Williams
16088f3b62aSDan Williams /* Open Address Frame Parameters */
16178a6f06eSDan Williams rnc->ssp.oaf_connection_rate = idev->connection_rate;
16288f3b62aSDan Williams rnc->ssp.oaf_features = 0;
16388f3b62aSDan Williams rnc->ssp.oaf_source_zone_group = 0;
16488f3b62aSDan Williams rnc->ssp.oaf_more_compatibility_features = 0;
16588f3b62aSDan Williams }
166ad276048SLee Jones /*
16788f3b62aSDan Williams * This method will setup the remote node context object so it will transition
16888f3b62aSDan Williams * to its ready state. If the remote node context is already setup to
16988f3b62aSDan Williams * transition to its final state then this function does nothing. none
17088f3b62aSDan Williams */
sci_remote_node_context_setup_to_resume(struct sci_remote_node_context * sci_rnc,scics_sds_remote_node_context_callback callback,void * callback_parameter,enum sci_remote_node_context_destination_state dest_param)17189a7301fSDan Williams static void sci_remote_node_context_setup_to_resume(
17289a7301fSDan Williams struct sci_remote_node_context *sci_rnc,
17388f3b62aSDan Williams scics_sds_remote_node_context_callback callback,
17459e35396SJeff Skirvin void *callback_parameter,
17559e35396SJeff Skirvin enum sci_remote_node_context_destination_state dest_param)
17688f3b62aSDan Williams {
17759e35396SJeff Skirvin if (sci_rnc->destination_state != RNC_DEST_FINAL) {
17859e35396SJeff Skirvin sci_rnc->destination_state = dest_param;
1796c6aacbbSJeff Skirvin if (callback != NULL) {
1806c6aacbbSJeff Skirvin sci_rnc->user_callback = callback;
1816c6aacbbSJeff Skirvin sci_rnc->user_cookie = callback_parameter;
1826c6aacbbSJeff Skirvin }
18388f3b62aSDan Williams }
18459e35396SJeff Skirvin }
18588f3b62aSDan Williams
sci_remote_node_context_setup_to_destroy(struct sci_remote_node_context * sci_rnc,scics_sds_remote_node_context_callback callback,void * callback_parameter)18659e35396SJeff Skirvin static void sci_remote_node_context_setup_to_destroy(
18789a7301fSDan Williams struct sci_remote_node_context *sci_rnc,
18888f3b62aSDan Williams scics_sds_remote_node_context_callback callback,
18988f3b62aSDan Williams void *callback_parameter)
19088f3b62aSDan Williams {
191de2eb4d5SJeff Skirvin struct isci_host *ihost = idev_to_ihost(rnc_to_dev(sci_rnc));
192de2eb4d5SJeff Skirvin
19359e35396SJeff Skirvin sci_rnc->destination_state = RNC_DEST_FINAL;
19488f3b62aSDan Williams sci_rnc->user_callback = callback;
19588f3b62aSDan Williams sci_rnc->user_cookie = callback_parameter;
196de2eb4d5SJeff Skirvin
197de2eb4d5SJeff Skirvin wake_up(&ihost->eventq);
19888f3b62aSDan Williams }
19988f3b62aSDan Williams
200*d2d480f1SLee Jones /*
20188f3b62aSDan Williams * This method just calls the user callback function and then resets the
20288f3b62aSDan Williams * callback.
20388f3b62aSDan Williams */
sci_remote_node_context_notify_user(struct sci_remote_node_context * rnc)20489a7301fSDan Williams static void sci_remote_node_context_notify_user(
20589a7301fSDan Williams struct sci_remote_node_context *rnc)
20688f3b62aSDan Williams {
20788f3b62aSDan Williams if (rnc->user_callback != NULL) {
20888f3b62aSDan Williams (*rnc->user_callback)(rnc->user_cookie);
20988f3b62aSDan Williams
21088f3b62aSDan Williams rnc->user_callback = NULL;
21188f3b62aSDan Williams rnc->user_cookie = NULL;
21288f3b62aSDan Williams }
21388f3b62aSDan Williams }
21488f3b62aSDan Williams
sci_remote_node_context_continue_state_transitions(struct sci_remote_node_context * rnc)21589a7301fSDan Williams static void sci_remote_node_context_continue_state_transitions(struct sci_remote_node_context *rnc)
21688f3b62aSDan Williams {
21779cbab89SJeff Skirvin switch (rnc->destination_state) {
21879cbab89SJeff Skirvin case RNC_DEST_READY:
21979cbab89SJeff Skirvin case RNC_DEST_SUSPENDED_RESUME:
22059e35396SJeff Skirvin rnc->destination_state = RNC_DEST_READY;
221df561f66SGustavo A. R. Silva fallthrough;
22279cbab89SJeff Skirvin case RNC_DEST_FINAL:
22389a7301fSDan Williams sci_remote_node_context_resume(rnc, rnc->user_callback,
224ed3efb77SDan Williams rnc->user_cookie);
22579cbab89SJeff Skirvin break;
22679cbab89SJeff Skirvin default:
22759e35396SJeff Skirvin rnc->destination_state = RNC_DEST_UNSPECIFIED;
22879cbab89SJeff Skirvin break;
22979cbab89SJeff Skirvin }
23088f3b62aSDan Williams }
23188f3b62aSDan Williams
sci_remote_node_context_validate_context_buffer(struct sci_remote_node_context * sci_rnc)23289a7301fSDan Williams static void sci_remote_node_context_validate_context_buffer(struct sci_remote_node_context *sci_rnc)
23388f3b62aSDan Williams {
23489a7301fSDan Williams union scu_remote_node_context *rnc_buffer;
23578a6f06eSDan Williams struct isci_remote_device *idev = rnc_to_dev(sci_rnc);
23678a6f06eSDan Williams struct domain_device *dev = idev->domain_dev;
23789a7301fSDan Williams struct isci_host *ihost = idev->owning_port->owning_controller;
23888f3b62aSDan Williams
23989a7301fSDan Williams rnc_buffer = sci_rnc_by_id(ihost, sci_rnc->remote_node_index);
24088f3b62aSDan Williams
24188f3b62aSDan Williams rnc_buffer->ssp.is_valid = true;
24288f3b62aSDan Williams
24311cc5183SDan Williams if (dev_is_sata(dev) && dev->parent) {
24489a7301fSDan Williams sci_remote_device_post_request(idev, SCU_CONTEXT_COMMAND_POST_RNC_96);
24588f3b62aSDan Williams } else {
24689a7301fSDan Williams sci_remote_device_post_request(idev, SCU_CONTEXT_COMMAND_POST_RNC_32);
24788f3b62aSDan Williams
24811cc5183SDan Williams if (!dev->parent)
24989a7301fSDan Williams sci_port_setup_transports(idev->owning_port,
25088f3b62aSDan Williams sci_rnc->remote_node_index);
25188f3b62aSDan Williams }
25288f3b62aSDan Williams }
25388f3b62aSDan Williams
sci_remote_node_context_invalidate_context_buffer(struct sci_remote_node_context * sci_rnc)25489a7301fSDan Williams static void sci_remote_node_context_invalidate_context_buffer(struct sci_remote_node_context *sci_rnc)
25588f3b62aSDan Williams {
25688f3b62aSDan Williams union scu_remote_node_context *rnc_buffer;
25789a7301fSDan Williams struct isci_remote_device *idev = rnc_to_dev(sci_rnc);
25889a7301fSDan Williams struct isci_host *ihost = idev->owning_port->owning_controller;
25988f3b62aSDan Williams
26089a7301fSDan Williams rnc_buffer = sci_rnc_by_id(ihost, sci_rnc->remote_node_index);
26188f3b62aSDan Williams
26288f3b62aSDan Williams rnc_buffer->ssp.is_valid = false;
26388f3b62aSDan Williams
26489a7301fSDan Williams sci_remote_device_post_request(rnc_to_dev(sci_rnc),
26588f3b62aSDan Williams SCU_CONTEXT_COMMAND_POST_RNC_INVALIDATE);
26688f3b62aSDan Williams }
26788f3b62aSDan Williams
sci_remote_node_context_initial_state_enter(struct sci_base_state_machine * sm)26889a7301fSDan Williams static void sci_remote_node_context_initial_state_enter(struct sci_base_state_machine *sm)
26988f3b62aSDan Williams {
27089a7301fSDan Williams struct sci_remote_node_context *rnc = container_of(sm, typeof(*rnc), sm);
2711f053889SJeff Skirvin struct isci_remote_device *idev = rnc_to_dev(rnc);
2721f053889SJeff Skirvin struct isci_host *ihost = idev->owning_port->owning_controller;
27388f3b62aSDan Williams
274f34d9e5dSDan Williams /* Check to see if we have gotten back to the initial state because
275f34d9e5dSDan Williams * someone requested to destroy the remote node context object.
276f34d9e5dSDan Williams */
277e301370aSEdmund Nadolski if (sm->previous_state_id == SCI_RNC_INVALIDATING) {
27859e35396SJeff Skirvin rnc->destination_state = RNC_DEST_UNSPECIFIED;
27989a7301fSDan Williams sci_remote_node_context_notify_user(rnc);
2801f053889SJeff Skirvin
2811f053889SJeff Skirvin smp_wmb();
2821f053889SJeff Skirvin wake_up(&ihost->eventq);
28388f3b62aSDan Williams }
28488f3b62aSDan Williams }
28588f3b62aSDan Williams
sci_remote_node_context_posting_state_enter(struct sci_base_state_machine * sm)28689a7301fSDan Williams static void sci_remote_node_context_posting_state_enter(struct sci_base_state_machine *sm)
28788f3b62aSDan Williams {
28889a7301fSDan Williams struct sci_remote_node_context *sci_rnc = container_of(sm, typeof(*sci_rnc), sm);
28988f3b62aSDan Williams
29089a7301fSDan Williams sci_remote_node_context_validate_context_buffer(sci_rnc);
29188f3b62aSDan Williams }
29288f3b62aSDan Williams
sci_remote_node_context_invalidating_state_enter(struct sci_base_state_machine * sm)29389a7301fSDan Williams static void sci_remote_node_context_invalidating_state_enter(struct sci_base_state_machine *sm)
29488f3b62aSDan Williams {
29589a7301fSDan Williams struct sci_remote_node_context *rnc = container_of(sm, typeof(*rnc), sm);
29688f3b62aSDan Williams
297aa20d934SJeff Skirvin /* Terminate all outstanding requests. */
298aa20d934SJeff Skirvin sci_remote_device_terminate_requests(rnc_to_dev(rnc));
29989a7301fSDan Williams sci_remote_node_context_invalidate_context_buffer(rnc);
30088f3b62aSDan Williams }
30188f3b62aSDan Williams
sci_remote_node_context_resuming_state_enter(struct sci_base_state_machine * sm)30289a7301fSDan Williams static void sci_remote_node_context_resuming_state_enter(struct sci_base_state_machine *sm)
30388f3b62aSDan Williams {
30489a7301fSDan Williams struct sci_remote_node_context *rnc = container_of(sm, typeof(*rnc), sm);
30578a6f06eSDan Williams struct isci_remote_device *idev;
306a1a113b0SDan Williams struct domain_device *dev;
30788f3b62aSDan Williams
30878a6f06eSDan Williams idev = rnc_to_dev(rnc);
30978a6f06eSDan Williams dev = idev->domain_dev;
31088f3b62aSDan Williams
31188f3b62aSDan Williams /*
31288f3b62aSDan Williams * For direct attached SATA devices we need to clear the TLCR
31388f3b62aSDan Williams * NCQ to TCi tag mapping on the phy and in cases where we
31488f3b62aSDan Williams * resume because of a target reset we also need to update
31588f3b62aSDan Williams * the STPTLDARNI register with the RNi of the device
31688f3b62aSDan Williams */
31711cc5183SDan Williams if (dev_is_sata(dev) && !dev->parent)
31811cc5183SDan Williams sci_port_setup_transports(idev->owning_port, rnc->remote_node_index);
31988f3b62aSDan Williams
32089a7301fSDan Williams sci_remote_device_post_request(idev, SCU_CONTEXT_COMMAND_POST_RNC_RESUME);
32188f3b62aSDan Williams }
32288f3b62aSDan Williams
sci_remote_node_context_ready_state_enter(struct sci_base_state_machine * sm)32389a7301fSDan Williams static void sci_remote_node_context_ready_state_enter(struct sci_base_state_machine *sm)
32488f3b62aSDan Williams {
32589a7301fSDan Williams struct sci_remote_node_context *rnc = container_of(sm, typeof(*rnc), sm);
32659e35396SJeff Skirvin enum sci_remote_node_context_destination_state dest_select;
32759e35396SJeff Skirvin int tell_user = 1;
32888f3b62aSDan Williams
32959e35396SJeff Skirvin dest_select = rnc->destination_state;
33059e35396SJeff Skirvin rnc->destination_state = RNC_DEST_UNSPECIFIED;
33188f3b62aSDan Williams
33259e35396SJeff Skirvin if ((dest_select == RNC_DEST_SUSPENDED) ||
33359e35396SJeff Skirvin (dest_select == RNC_DEST_SUSPENDED_RESUME)) {
33459e35396SJeff Skirvin sci_remote_node_context_suspend(
335447bfbceSJeff Skirvin rnc, rnc->suspend_reason,
336447bfbceSJeff Skirvin SCI_SOFTWARE_SUSPEND_EXPECTED_EVENT);
33759e35396SJeff Skirvin
338d76689e4SJeff Skirvin if (dest_select == RNC_DEST_SUSPENDED_RESUME)
33959e35396SJeff Skirvin tell_user = 0; /* Wait until ready again. */
34059e35396SJeff Skirvin }
341d76689e4SJeff Skirvin if (tell_user)
34289a7301fSDan Williams sci_remote_node_context_notify_user(rnc);
34388f3b62aSDan Williams }
34488f3b62aSDan Williams
sci_remote_node_context_tx_suspended_state_enter(struct sci_base_state_machine * sm)34589a7301fSDan Williams static void sci_remote_node_context_tx_suspended_state_enter(struct sci_base_state_machine *sm)
34688f3b62aSDan Williams {
34789a7301fSDan Williams struct sci_remote_node_context *rnc = container_of(sm, typeof(*rnc), sm);
34888f3b62aSDan Williams
34989a7301fSDan Williams sci_remote_node_context_continue_state_transitions(rnc);
35088f3b62aSDan Williams }
35188f3b62aSDan Williams
sci_remote_node_context_tx_rx_suspended_state_enter(struct sci_base_state_machine * sm)35289a7301fSDan Williams static void sci_remote_node_context_tx_rx_suspended_state_enter(struct sci_base_state_machine *sm)
35388f3b62aSDan Williams {
35489a7301fSDan Williams struct sci_remote_node_context *rnc = container_of(sm, typeof(*rnc), sm);
355726980d5SJeff Skirvin struct isci_remote_device *idev = rnc_to_dev(rnc);
356726980d5SJeff Skirvin struct isci_host *ihost = idev->owning_port->owning_controller;
35731a38ef0SJeff Skirvin u32 new_count = rnc->suspend_count + 1;
35831a38ef0SJeff Skirvin
35931a38ef0SJeff Skirvin if (new_count == 0)
36031a38ef0SJeff Skirvin rnc->suspend_count = 1;
36131a38ef0SJeff Skirvin else
36231a38ef0SJeff Skirvin rnc->suspend_count = new_count;
36331a38ef0SJeff Skirvin smp_wmb();
36488f3b62aSDan Williams
365726980d5SJeff Skirvin /* Terminate outstanding requests pending abort. */
366726980d5SJeff Skirvin sci_remote_device_abort_requests_pending_abort(idev);
367726980d5SJeff Skirvin
368726980d5SJeff Skirvin wake_up(&ihost->eventq);
36989a7301fSDan Williams sci_remote_node_context_continue_state_transitions(rnc);
37088f3b62aSDan Williams }
37188f3b62aSDan Williams
sci_remote_node_context_await_suspend_state_exit(struct sci_base_state_machine * sm)3726f48844eSJeff Skirvin static void sci_remote_node_context_await_suspend_state_exit(
3736f48844eSJeff Skirvin struct sci_base_state_machine *sm)
3746f48844eSJeff Skirvin {
3756f48844eSJeff Skirvin struct sci_remote_node_context *rnc
3766f48844eSJeff Skirvin = container_of(sm, typeof(*rnc), sm);
377c94fc1adSJeff Skirvin struct isci_remote_device *idev = rnc_to_dev(rnc);
3786f48844eSJeff Skirvin
379c94fc1adSJeff Skirvin if (dev_is_sata(idev->domain_dev))
380c94fc1adSJeff Skirvin isci_dev_set_hang_detection_timeout(idev, 0);
3816f48844eSJeff Skirvin }
3826f48844eSJeff Skirvin
38389a7301fSDan Williams static const struct sci_base_state sci_remote_node_context_state_table[] = {
384e301370aSEdmund Nadolski [SCI_RNC_INITIAL] = {
38589a7301fSDan Williams .enter_state = sci_remote_node_context_initial_state_enter,
38688f3b62aSDan Williams },
387e301370aSEdmund Nadolski [SCI_RNC_POSTING] = {
38889a7301fSDan Williams .enter_state = sci_remote_node_context_posting_state_enter,
38988f3b62aSDan Williams },
390e301370aSEdmund Nadolski [SCI_RNC_INVALIDATING] = {
39189a7301fSDan Williams .enter_state = sci_remote_node_context_invalidating_state_enter,
39288f3b62aSDan Williams },
393e301370aSEdmund Nadolski [SCI_RNC_RESUMING] = {
39489a7301fSDan Williams .enter_state = sci_remote_node_context_resuming_state_enter,
39588f3b62aSDan Williams },
396e301370aSEdmund Nadolski [SCI_RNC_READY] = {
39789a7301fSDan Williams .enter_state = sci_remote_node_context_ready_state_enter,
39888f3b62aSDan Williams },
399e301370aSEdmund Nadolski [SCI_RNC_TX_SUSPENDED] = {
40089a7301fSDan Williams .enter_state = sci_remote_node_context_tx_suspended_state_enter,
40188f3b62aSDan Williams },
402e301370aSEdmund Nadolski [SCI_RNC_TX_RX_SUSPENDED] = {
40389a7301fSDan Williams .enter_state = sci_remote_node_context_tx_rx_suspended_state_enter,
40488f3b62aSDan Williams },
4056f48844eSJeff Skirvin [SCI_RNC_AWAIT_SUSPENSION] = {
4066f48844eSJeff Skirvin .exit_state = sci_remote_node_context_await_suspend_state_exit,
4076f48844eSJeff Skirvin },
40888f3b62aSDan Williams };
40988f3b62aSDan Williams
sci_remote_node_context_construct(struct sci_remote_node_context * rnc,u16 remote_node_index)41089a7301fSDan Williams void sci_remote_node_context_construct(struct sci_remote_node_context *rnc,
41188f3b62aSDan Williams u16 remote_node_index)
41288f3b62aSDan Williams {
41389a7301fSDan Williams memset(rnc, 0, sizeof(struct sci_remote_node_context));
41488f3b62aSDan Williams
41588f3b62aSDan Williams rnc->remote_node_index = remote_node_index;
41659e35396SJeff Skirvin rnc->destination_state = RNC_DEST_UNSPECIFIED;
41788f3b62aSDan Williams
41889a7301fSDan Williams sci_init_sm(&rnc->sm, sci_remote_node_context_state_table, SCI_RNC_INITIAL);
41988f3b62aSDan Williams }
420338e386dSDan Williams
sci_remote_node_context_event_handler(struct sci_remote_node_context * sci_rnc,u32 event_code)42189a7301fSDan Williams enum sci_status sci_remote_node_context_event_handler(struct sci_remote_node_context *sci_rnc,
422338e386dSDan Williams u32 event_code)
423338e386dSDan Williams {
424338e386dSDan Williams enum scis_sds_remote_node_context_states state;
425ac78ed0fSJeff Skirvin u32 next_state;
426338e386dSDan Williams
427e301370aSEdmund Nadolski state = sci_rnc->sm.current_state_id;
428338e386dSDan Williams switch (state) {
429e301370aSEdmund Nadolski case SCI_RNC_POSTING:
430338e386dSDan Williams switch (scu_get_event_code(event_code)) {
431338e386dSDan Williams case SCU_EVENT_POST_RNC_COMPLETE:
432e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_READY);
433338e386dSDan Williams break;
434338e386dSDan Williams default:
435338e386dSDan Williams goto out;
436338e386dSDan Williams }
437338e386dSDan Williams break;
438e301370aSEdmund Nadolski case SCI_RNC_INVALIDATING:
439338e386dSDan Williams if (scu_get_event_code(event_code) == SCU_EVENT_POST_RNC_INVALIDATE_COMPLETE) {
44059e35396SJeff Skirvin if (sci_rnc->destination_state == RNC_DEST_FINAL)
44159e35396SJeff Skirvin next_state = SCI_RNC_INITIAL;
442338e386dSDan Williams else
44359e35396SJeff Skirvin next_state = SCI_RNC_POSTING;
44459e35396SJeff Skirvin sci_change_state(&sci_rnc->sm, next_state);
445338e386dSDan Williams } else {
446338e386dSDan Williams switch (scu_get_event_type(event_code)) {
447338e386dSDan Williams case SCU_EVENT_TYPE_RNC_SUSPEND_TX:
448338e386dSDan Williams case SCU_EVENT_TYPE_RNC_SUSPEND_TX_RX:
449338e386dSDan Williams /* We really dont care if the hardware is going to suspend
450338e386dSDan Williams * the device since it's being invalidated anyway */
4518c731888SJeff Skirvin dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
452338e386dSDan Williams "%s: SCIC Remote Node Context 0x%p was "
45399c7b6aeSColin Ian King "suspended by hardware while being "
454338e386dSDan Williams "invalidated.\n", __func__, sci_rnc);
455338e386dSDan Williams break;
456338e386dSDan Williams default:
457338e386dSDan Williams goto out;
458338e386dSDan Williams }
459338e386dSDan Williams }
460338e386dSDan Williams break;
461e301370aSEdmund Nadolski case SCI_RNC_RESUMING:
462338e386dSDan Williams if (scu_get_event_code(event_code) == SCU_EVENT_POST_RCN_RELEASE) {
463e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_READY);
464338e386dSDan Williams } else {
465338e386dSDan Williams switch (scu_get_event_type(event_code)) {
466338e386dSDan Williams case SCU_EVENT_TYPE_RNC_SUSPEND_TX:
467338e386dSDan Williams case SCU_EVENT_TYPE_RNC_SUSPEND_TX_RX:
468338e386dSDan Williams /* We really dont care if the hardware is going to suspend
469338e386dSDan Williams * the device since it's being resumed anyway */
4708c731888SJeff Skirvin dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
471338e386dSDan Williams "%s: SCIC Remote Node Context 0x%p was "
47299c7b6aeSColin Ian King "suspended by hardware while being resumed.\n",
473338e386dSDan Williams __func__, sci_rnc);
474338e386dSDan Williams break;
475338e386dSDan Williams default:
476338e386dSDan Williams goto out;
477338e386dSDan Williams }
478338e386dSDan Williams }
479338e386dSDan Williams break;
480e301370aSEdmund Nadolski case SCI_RNC_READY:
481338e386dSDan Williams switch (scu_get_event_type(event_code)) {
482338e386dSDan Williams case SCU_EVENT_TL_RNC_SUSPEND_TX:
483e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_TX_SUSPENDED);
484ac78ed0fSJeff Skirvin sci_rnc->suspend_type = scu_get_event_type(event_code);
485338e386dSDan Williams break;
486338e386dSDan Williams case SCU_EVENT_TL_RNC_SUSPEND_TX_RX:
487e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_TX_RX_SUSPENDED);
488ac78ed0fSJeff Skirvin sci_rnc->suspend_type = scu_get_event_type(event_code);
489338e386dSDan Williams break;
490338e386dSDan Williams default:
491338e386dSDan Williams goto out;
492338e386dSDan Williams }
493338e386dSDan Williams break;
494e301370aSEdmund Nadolski case SCI_RNC_AWAIT_SUSPENSION:
495338e386dSDan Williams switch (scu_get_event_type(event_code)) {
496338e386dSDan Williams case SCU_EVENT_TL_RNC_SUSPEND_TX:
497ac78ed0fSJeff Skirvin next_state = SCI_RNC_TX_SUSPENDED;
498338e386dSDan Williams break;
499338e386dSDan Williams case SCU_EVENT_TL_RNC_SUSPEND_TX_RX:
500ac78ed0fSJeff Skirvin next_state = SCI_RNC_TX_RX_SUSPENDED;
501338e386dSDan Williams break;
502338e386dSDan Williams default:
503338e386dSDan Williams goto out;
504338e386dSDan Williams }
505ac78ed0fSJeff Skirvin if (sci_rnc->suspend_type == scu_get_event_type(event_code))
506ac78ed0fSJeff Skirvin sci_change_state(&sci_rnc->sm, next_state);
507338e386dSDan Williams break;
508338e386dSDan Williams default:
509338e386dSDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
51014e99b4aSDan Williams "%s: invalid state: %s\n", __func__,
51114e99b4aSDan Williams rnc_state_name(state));
512338e386dSDan Williams return SCI_FAILURE_INVALID_STATE;
513338e386dSDan Williams }
514338e386dSDan Williams return SCI_SUCCESS;
515338e386dSDan Williams
516338e386dSDan Williams out:
517338e386dSDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
51814e99b4aSDan Williams "%s: code: %#x state: %s\n", __func__, event_code,
51914e99b4aSDan Williams rnc_state_name(state));
520338e386dSDan Williams return SCI_FAILURE;
521338e386dSDan Williams
522338e386dSDan Williams }
523c845ae96SDan Williams
sci_remote_node_context_destruct(struct sci_remote_node_context * sci_rnc,scics_sds_remote_node_context_callback cb_fn,void * cb_p)52489a7301fSDan Williams enum sci_status sci_remote_node_context_destruct(struct sci_remote_node_context *sci_rnc,
525c845ae96SDan Williams scics_sds_remote_node_context_callback cb_fn,
526c845ae96SDan Williams void *cb_p)
527c845ae96SDan Williams {
528c845ae96SDan Williams enum scis_sds_remote_node_context_states state;
529c845ae96SDan Williams
530e301370aSEdmund Nadolski state = sci_rnc->sm.current_state_id;
531c845ae96SDan Williams switch (state) {
532e301370aSEdmund Nadolski case SCI_RNC_INVALIDATING:
53359e35396SJeff Skirvin sci_remote_node_context_setup_to_destroy(sci_rnc, cb_fn, cb_p);
534c845ae96SDan Williams return SCI_SUCCESS;
535e301370aSEdmund Nadolski case SCI_RNC_POSTING:
536e301370aSEdmund Nadolski case SCI_RNC_RESUMING:
537e301370aSEdmund Nadolski case SCI_RNC_READY:
538e301370aSEdmund Nadolski case SCI_RNC_TX_SUSPENDED:
539e301370aSEdmund Nadolski case SCI_RNC_TX_RX_SUSPENDED:
54059e35396SJeff Skirvin sci_remote_node_context_setup_to_destroy(sci_rnc, cb_fn, cb_p);
541e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_INVALIDATING);
542c845ae96SDan Williams return SCI_SUCCESS;
54379cbab89SJeff Skirvin case SCI_RNC_AWAIT_SUSPENSION:
54479cbab89SJeff Skirvin sci_remote_node_context_setup_to_destroy(sci_rnc, cb_fn, cb_p);
54579cbab89SJeff Skirvin return SCI_SUCCESS;
546e301370aSEdmund Nadolski case SCI_RNC_INITIAL:
547c845ae96SDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
54814e99b4aSDan Williams "%s: invalid state: %s\n", __func__,
54914e99b4aSDan Williams rnc_state_name(state));
550c845ae96SDan Williams /* We have decided that the destruct request on the remote node context
551c845ae96SDan Williams * can not fail since it is either in the initial/destroyed state or is
552c845ae96SDan Williams * can be destroyed.
553c845ae96SDan Williams */
554c845ae96SDan Williams return SCI_SUCCESS;
555c845ae96SDan Williams default:
556c845ae96SDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
55714e99b4aSDan Williams "%s: invalid state %s\n", __func__,
55814e99b4aSDan Williams rnc_state_name(state));
559c845ae96SDan Williams return SCI_FAILURE_INVALID_STATE;
560c845ae96SDan Williams }
561c845ae96SDan Williams }
562ed3efb77SDan Williams
sci_remote_node_context_suspend(struct sci_remote_node_context * sci_rnc,enum sci_remote_node_suspension_reasons suspend_reason,u32 suspend_type)563ac78ed0fSJeff Skirvin enum sci_status sci_remote_node_context_suspend(
564ac78ed0fSJeff Skirvin struct sci_remote_node_context *sci_rnc,
565ac78ed0fSJeff Skirvin enum sci_remote_node_suspension_reasons suspend_reason,
566447bfbceSJeff Skirvin u32 suspend_type)
567ed3efb77SDan Williams {
568ac78ed0fSJeff Skirvin enum scis_sds_remote_node_context_states state
569ac78ed0fSJeff Skirvin = sci_rnc->sm.current_state_id;
570ac78ed0fSJeff Skirvin struct isci_remote_device *idev = rnc_to_dev(sci_rnc);
571ac78ed0fSJeff Skirvin enum sci_status status = SCI_FAILURE_INVALID_STATE;
57259e35396SJeff Skirvin enum sci_remote_node_context_destination_state dest_param =
57359e35396SJeff Skirvin RNC_DEST_UNSPECIFIED;
574ed3efb77SDan Williams
575726980d5SJeff Skirvin dev_dbg(scirdev_to_dev(idev),
5768c731888SJeff Skirvin "%s: current state %s, current suspend_type %x dest state %d,"
577726980d5SJeff Skirvin " arg suspend_reason %d, arg suspend_type %x",
5788c731888SJeff Skirvin __func__, rnc_state_name(state), sci_rnc->suspend_type,
579726980d5SJeff Skirvin sci_rnc->destination_state, suspend_reason,
580726980d5SJeff Skirvin suspend_type);
581726980d5SJeff Skirvin
582ac78ed0fSJeff Skirvin /* Disable automatic state continuations if explicitly suspending. */
583c94fc1adSJeff Skirvin if ((suspend_reason == SCI_HW_SUSPEND) ||
58459e35396SJeff Skirvin (sci_rnc->destination_state == RNC_DEST_FINAL))
58559e35396SJeff Skirvin dest_param = sci_rnc->destination_state;
58659e35396SJeff Skirvin
587ac78ed0fSJeff Skirvin switch (state) {
588ac78ed0fSJeff Skirvin case SCI_RNC_READY:
589ac78ed0fSJeff Skirvin break;
59059e35396SJeff Skirvin case SCI_RNC_INVALIDATING:
59159e35396SJeff Skirvin if (sci_rnc->destination_state == RNC_DEST_FINAL) {
59259e35396SJeff Skirvin dev_warn(scirdev_to_dev(idev),
59359e35396SJeff Skirvin "%s: already destroying %p\n",
59459e35396SJeff Skirvin __func__, sci_rnc);
59559e35396SJeff Skirvin return SCI_FAILURE_INVALID_STATE;
59659e35396SJeff Skirvin }
597df561f66SGustavo A. R. Silva fallthrough; /* and handle like SCI_RNC_POSTING */
598d76689e4SJeff Skirvin case SCI_RNC_RESUMING:
599df561f66SGustavo A. R. Silva fallthrough; /* and handle like SCI_RNC_POSTING */
60059e35396SJeff Skirvin case SCI_RNC_POSTING:
60159e35396SJeff Skirvin /* Set the destination state to AWAIT - this signals the
60259e35396SJeff Skirvin * entry into the SCI_RNC_READY state that a suspension
60359e35396SJeff Skirvin * needs to be done immediately.
60459e35396SJeff Skirvin */
6056c6aacbbSJeff Skirvin if (sci_rnc->destination_state != RNC_DEST_FINAL)
60659e35396SJeff Skirvin sci_rnc->destination_state = RNC_DEST_SUSPENDED;
607447bfbceSJeff Skirvin sci_rnc->suspend_type = suspend_type;
608447bfbceSJeff Skirvin sci_rnc->suspend_reason = suspend_reason;
60959e35396SJeff Skirvin return SCI_SUCCESS;
61059e35396SJeff Skirvin
611ac78ed0fSJeff Skirvin case SCI_RNC_TX_SUSPENDED:
612ac78ed0fSJeff Skirvin if (suspend_type == SCU_EVENT_TL_RNC_SUSPEND_TX)
613ac78ed0fSJeff Skirvin status = SCI_SUCCESS;
614ac78ed0fSJeff Skirvin break;
615ac78ed0fSJeff Skirvin case SCI_RNC_TX_RX_SUSPENDED:
616ac78ed0fSJeff Skirvin if (suspend_type == SCU_EVENT_TL_RNC_SUSPEND_TX_RX)
617ac78ed0fSJeff Skirvin status = SCI_SUCCESS;
618ac78ed0fSJeff Skirvin break;
619ac78ed0fSJeff Skirvin case SCI_RNC_AWAIT_SUSPENSION:
620ac78ed0fSJeff Skirvin if ((sci_rnc->suspend_type == SCU_EVENT_TL_RNC_SUSPEND_TX_RX)
621ac78ed0fSJeff Skirvin || (suspend_type == sci_rnc->suspend_type))
622ac78ed0fSJeff Skirvin return SCI_SUCCESS;
623ac78ed0fSJeff Skirvin break;
624ac78ed0fSJeff Skirvin default:
625ed3efb77SDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
62614e99b4aSDan Williams "%s: invalid state %s\n", __func__,
62714e99b4aSDan Williams rnc_state_name(state));
628ed3efb77SDan Williams return SCI_FAILURE_INVALID_STATE;
629ed3efb77SDan Williams }
63059e35396SJeff Skirvin sci_rnc->destination_state = dest_param;
631ac78ed0fSJeff Skirvin sci_rnc->suspend_type = suspend_type;
632447bfbceSJeff Skirvin sci_rnc->suspend_reason = suspend_reason;
633ed3efb77SDan Williams
634ac78ed0fSJeff Skirvin if (status == SCI_SUCCESS) { /* Already in the destination state? */
635726980d5SJeff Skirvin struct isci_host *ihost = idev->owning_port->owning_controller;
636726980d5SJeff Skirvin
637726980d5SJeff Skirvin wake_up_all(&ihost->eventq); /* Let observers look. */
638ac78ed0fSJeff Skirvin return SCI_SUCCESS;
639ed3efb77SDan Williams }
640c94fc1adSJeff Skirvin if ((suspend_reason == SCI_SW_SUSPEND_NORMAL) ||
641c94fc1adSJeff Skirvin (suspend_reason == SCI_SW_SUSPEND_LINKHANG_DETECT)) {
642c94fc1adSJeff Skirvin
6439608b640SJeff Skirvin if (suspend_reason == SCI_SW_SUSPEND_LINKHANG_DETECT)
644ac78ed0fSJeff Skirvin isci_dev_set_hang_detection_timeout(idev, 0x00000001);
645c94fc1adSJeff Skirvin
646ac78ed0fSJeff Skirvin sci_remote_device_post_request(
647ac78ed0fSJeff Skirvin idev, SCI_SOFTWARE_SUSPEND_CMD);
648ac78ed0fSJeff Skirvin }
649ac78ed0fSJeff Skirvin if (state != SCI_RNC_AWAIT_SUSPENSION)
650e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_AWAIT_SUSPENSION);
651ac78ed0fSJeff Skirvin
652ed3efb77SDan Williams return SCI_SUCCESS;
653ed3efb77SDan Williams }
654ed3efb77SDan Williams
sci_remote_node_context_resume(struct sci_remote_node_context * sci_rnc,scics_sds_remote_node_context_callback cb_fn,void * cb_p)65589a7301fSDan Williams enum sci_status sci_remote_node_context_resume(struct sci_remote_node_context *sci_rnc,
656ed3efb77SDan Williams scics_sds_remote_node_context_callback cb_fn,
657ed3efb77SDan Williams void *cb_p)
658ed3efb77SDan Williams {
659ed3efb77SDan Williams enum scis_sds_remote_node_context_states state;
6600c3ce38fSJeff Skirvin struct isci_remote_device *idev = rnc_to_dev(sci_rnc);
661ed3efb77SDan Williams
662e301370aSEdmund Nadolski state = sci_rnc->sm.current_state_id;
6630c3ce38fSJeff Skirvin dev_dbg(scirdev_to_dev(idev),
6640c3ce38fSJeff Skirvin "%s: state %s, cb_fn = %p, cb_p = %p; dest_state = %d; "
6650c3ce38fSJeff Skirvin "dev resume path %s\n",
66631a38ef0SJeff Skirvin __func__, rnc_state_name(state), cb_fn, cb_p,
6670c3ce38fSJeff Skirvin sci_rnc->destination_state,
6680c3ce38fSJeff Skirvin test_bit(IDEV_ABORT_PATH_ACTIVE, &idev->flags)
6690c3ce38fSJeff Skirvin ? "<abort active>" : "<normal>");
67031a38ef0SJeff Skirvin
671ed3efb77SDan Williams switch (state) {
672e301370aSEdmund Nadolski case SCI_RNC_INITIAL:
673ed3efb77SDan Williams if (sci_rnc->remote_node_index == SCIC_SDS_REMOTE_NODE_CONTEXT_INVALID_INDEX)
674ed3efb77SDan Williams return SCI_FAILURE_INVALID_STATE;
675ed3efb77SDan Williams
6766c6aacbbSJeff Skirvin sci_remote_node_context_setup_to_resume(sci_rnc, cb_fn, cb_p,
6776c6aacbbSJeff Skirvin RNC_DEST_READY);
6786c6aacbbSJeff Skirvin if (!test_bit(IDEV_ABORT_PATH_ACTIVE, &idev->flags)) {
67989a7301fSDan Williams sci_remote_node_context_construct_buffer(sci_rnc);
680e301370aSEdmund Nadolski sci_change_state(&sci_rnc->sm, SCI_RNC_POSTING);
6810c3ce38fSJeff Skirvin }
682ed3efb77SDan Williams return SCI_SUCCESS;
6830c3ce38fSJeff Skirvin
684e301370aSEdmund Nadolski case SCI_RNC_POSTING:
685e301370aSEdmund Nadolski case SCI_RNC_INVALIDATING:
686e301370aSEdmund Nadolski case SCI_RNC_RESUMING:
6870c3ce38fSJeff Skirvin /* We are still waiting to post when a resume was
6880c3ce38fSJeff Skirvin * requested.
6890c3ce38fSJeff Skirvin */
69059e35396SJeff Skirvin switch (sci_rnc->destination_state) {
69159e35396SJeff Skirvin case RNC_DEST_SUSPENDED:
69259e35396SJeff Skirvin case RNC_DEST_SUSPENDED_RESUME:
6930c3ce38fSJeff Skirvin /* Previously waiting to suspend after posting.
6940c3ce38fSJeff Skirvin * Now continue onto resumption.
69559e35396SJeff Skirvin */
69659e35396SJeff Skirvin sci_remote_node_context_setup_to_resume(
69759e35396SJeff Skirvin sci_rnc, cb_fn, cb_p,
69859e35396SJeff Skirvin RNC_DEST_SUSPENDED_RESUME);
69959e35396SJeff Skirvin break;
70059e35396SJeff Skirvin default:
70159e35396SJeff Skirvin sci_remote_node_context_setup_to_resume(
70259e35396SJeff Skirvin sci_rnc, cb_fn, cb_p,
70359e35396SJeff Skirvin RNC_DEST_READY);
70459e35396SJeff Skirvin break;
70559e35396SJeff Skirvin }
706ed3efb77SDan Williams return SCI_SUCCESS;
707ed3efb77SDan Williams
7080c3ce38fSJeff Skirvin case SCI_RNC_TX_SUSPENDED:
7090c3ce38fSJeff Skirvin case SCI_RNC_TX_RX_SUSPENDED:
7106c6aacbbSJeff Skirvin {
7110c3ce38fSJeff Skirvin struct domain_device *dev = idev->domain_dev;
71256d7c013SJeff Skirvin /* If this is an expander attached SATA device we must
7130c3ce38fSJeff Skirvin * invalidate and repost the RNC since this is the only
7140c3ce38fSJeff Skirvin * way to clear the TCi to NCQ tag mapping table for
7150c3ce38fSJeff Skirvin * the RNi. All other device types we can just resume.
71656d7c013SJeff Skirvin */
7170c3ce38fSJeff Skirvin sci_remote_node_context_setup_to_resume(
7180c3ce38fSJeff Skirvin sci_rnc, cb_fn, cb_p, RNC_DEST_READY);
719ed3efb77SDan Williams
7206c6aacbbSJeff Skirvin if (!test_bit(IDEV_ABORT_PATH_ACTIVE, &idev->flags)) {
7216c6aacbbSJeff Skirvin if ((dev_is_sata(dev) && dev->parent) ||
7226c6aacbbSJeff Skirvin (sci_rnc->destination_state == RNC_DEST_FINAL))
7230c3ce38fSJeff Skirvin sci_change_state(&sci_rnc->sm,
7240c3ce38fSJeff Skirvin SCI_RNC_INVALIDATING);
72511cc5183SDan Williams else
7260c3ce38fSJeff Skirvin sci_change_state(&sci_rnc->sm,
7270c3ce38fSJeff Skirvin SCI_RNC_RESUMING);
728ed3efb77SDan Williams }
7296c6aacbbSJeff Skirvin }
7300c3ce38fSJeff Skirvin return SCI_SUCCESS;
7310c3ce38fSJeff Skirvin
732e301370aSEdmund Nadolski case SCI_RNC_AWAIT_SUSPENSION:
73359e35396SJeff Skirvin sci_remote_node_context_setup_to_resume(
7346c6aacbbSJeff Skirvin sci_rnc, cb_fn, cb_p, RNC_DEST_SUSPENDED_RESUME);
735ed3efb77SDan Williams return SCI_SUCCESS;
736ed3efb77SDan Williams default:
737ed3efb77SDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
73814e99b4aSDan Williams "%s: invalid state %s\n", __func__,
73914e99b4aSDan Williams rnc_state_name(state));
740ed3efb77SDan Williams return SCI_FAILURE_INVALID_STATE;
741ed3efb77SDan Williams }
742ed3efb77SDan Williams }
743f34d9e5dSDan Williams
sci_remote_node_context_start_io(struct sci_remote_node_context * sci_rnc,struct isci_request * ireq)74489a7301fSDan Williams enum sci_status sci_remote_node_context_start_io(struct sci_remote_node_context *sci_rnc,
7455076a1a9SDan Williams struct isci_request *ireq)
746f34d9e5dSDan Williams {
747f34d9e5dSDan Williams enum scis_sds_remote_node_context_states state;
748f34d9e5dSDan Williams
749e301370aSEdmund Nadolski state = sci_rnc->sm.current_state_id;
750fd536601SJeff Skirvin
751fd536601SJeff Skirvin switch (state) {
752fd536601SJeff Skirvin case SCI_RNC_READY:
753fd536601SJeff Skirvin return SCI_SUCCESS;
754fd536601SJeff Skirvin case SCI_RNC_TX_SUSPENDED:
755fd536601SJeff Skirvin case SCI_RNC_TX_RX_SUSPENDED:
756fd536601SJeff Skirvin case SCI_RNC_AWAIT_SUSPENSION:
757f34d9e5dSDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
75814e99b4aSDan Williams "%s: invalid state %s\n", __func__,
75914e99b4aSDan Williams rnc_state_name(state));
760f34d9e5dSDan Williams return SCI_FAILURE_REMOTE_DEVICE_RESET_REQUIRED;
761fd536601SJeff Skirvin default:
762fd536601SJeff Skirvin dev_dbg(scirdev_to_dev(rnc_to_dev(sci_rnc)),
76314e99b4aSDan Williams "%s: invalid state %s\n", __func__,
76414e99b4aSDan Williams rnc_state_name(state));
765fd536601SJeff Skirvin return SCI_FAILURE_INVALID_STATE;
766f34d9e5dSDan Williams }
76714e99b4aSDan Williams }
768f34d9e5dSDan Williams
sci_remote_node_context_start_task(struct sci_remote_node_context * sci_rnc,struct isci_request * ireq,scics_sds_remote_node_context_callback cb_fn,void * cb_p)76914aaa9f0SJeff Skirvin enum sci_status sci_remote_node_context_start_task(
77014aaa9f0SJeff Skirvin struct sci_remote_node_context *sci_rnc,
77114aaa9f0SJeff Skirvin struct isci_request *ireq,
77214aaa9f0SJeff Skirvin scics_sds_remote_node_context_callback cb_fn,
77314aaa9f0SJeff Skirvin void *cb_p)
774f34d9e5dSDan Williams {
77559e35396SJeff Skirvin enum sci_status status = sci_remote_node_context_resume(sci_rnc,
77659e35396SJeff Skirvin cb_fn, cb_p);
77759e35396SJeff Skirvin if (status != SCI_SUCCESS)
778f34d9e5dSDan Williams dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
77959e35396SJeff Skirvin "%s: resume failed: %d\n", __func__, status);
78059e35396SJeff Skirvin return status;
781f34d9e5dSDan Williams }
782726980d5SJeff Skirvin
sci_remote_node_context_is_safe_to_abort(struct sci_remote_node_context * sci_rnc)783726980d5SJeff Skirvin int sci_remote_node_context_is_safe_to_abort(
784726980d5SJeff Skirvin struct sci_remote_node_context *sci_rnc)
785726980d5SJeff Skirvin {
786726980d5SJeff Skirvin enum scis_sds_remote_node_context_states state;
787726980d5SJeff Skirvin
788726980d5SJeff Skirvin state = sci_rnc->sm.current_state_id;
789726980d5SJeff Skirvin switch (state) {
790726980d5SJeff Skirvin case SCI_RNC_INVALIDATING:
791726980d5SJeff Skirvin case SCI_RNC_TX_RX_SUSPENDED:
792726980d5SJeff Skirvin return 1;
793726980d5SJeff Skirvin case SCI_RNC_POSTING:
794726980d5SJeff Skirvin case SCI_RNC_RESUMING:
795726980d5SJeff Skirvin case SCI_RNC_READY:
796726980d5SJeff Skirvin case SCI_RNC_TX_SUSPENDED:
797726980d5SJeff Skirvin case SCI_RNC_AWAIT_SUSPENSION:
798726980d5SJeff Skirvin case SCI_RNC_INITIAL:
799726980d5SJeff Skirvin return 0;
800726980d5SJeff Skirvin default:
801726980d5SJeff Skirvin dev_warn(scirdev_to_dev(rnc_to_dev(sci_rnc)),
802726980d5SJeff Skirvin "%s: invalid state %d\n", __func__, state);
803726980d5SJeff Skirvin return 0;
804726980d5SJeff Skirvin }
805726980d5SJeff Skirvin }
806