xref: /linux/drivers/net/wireless/intel/iwlwifi/pcie/gen1_2/trans-gen2.c (revision 8be4d31cb8aaeea27bde4b7ddb26e28a89062ebf)
1c8a00a6eSMiri Korenblit // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2c8a00a6eSMiri Korenblit /*
3c8a00a6eSMiri Korenblit  * Copyright (C) 2017 Intel Deutschland GmbH
4c8a00a6eSMiri Korenblit  * Copyright (C) 2018-2025 Intel Corporation
5c8a00a6eSMiri Korenblit  */
6c8a00a6eSMiri Korenblit #include "iwl-trans.h"
7c8a00a6eSMiri Korenblit #include "iwl-prph.h"
8c8a00a6eSMiri Korenblit #include "pcie/iwl-context-info.h"
9c8a00a6eSMiri Korenblit #include "pcie/iwl-context-info-v2.h"
10c8a00a6eSMiri Korenblit #include "internal.h"
11c8a00a6eSMiri Korenblit #include "fw/dbg.h"
12c8a00a6eSMiri Korenblit 
13c8a00a6eSMiri Korenblit #define FW_RESET_TIMEOUT (HZ / 5)
14c8a00a6eSMiri Korenblit 
15c8a00a6eSMiri Korenblit /*
16c8a00a6eSMiri Korenblit  * Start up NIC's basic functionality after it has been reset
17c8a00a6eSMiri Korenblit  * (e.g. after platform boot, or shutdown via iwl_pcie_apm_stop())
18c8a00a6eSMiri Korenblit  * NOTE:  This does not load uCode nor start the embedded processor
19c8a00a6eSMiri Korenblit  */
iwl_pcie_gen2_apm_init(struct iwl_trans * trans)20c8a00a6eSMiri Korenblit int iwl_pcie_gen2_apm_init(struct iwl_trans *trans)
21c8a00a6eSMiri Korenblit {
22c8a00a6eSMiri Korenblit 	int ret = 0;
23c8a00a6eSMiri Korenblit 
24c8a00a6eSMiri Korenblit 	IWL_DEBUG_INFO(trans, "Init card's basic functions\n");
25c8a00a6eSMiri Korenblit 
26c8a00a6eSMiri Korenblit 	/*
27c8a00a6eSMiri Korenblit 	 * Use "set_bit" below rather than "write", to preserve any hardware
28c8a00a6eSMiri Korenblit 	 * bits already set by default after reset.
29c8a00a6eSMiri Korenblit 	 */
30c8a00a6eSMiri Korenblit 
31c8a00a6eSMiri Korenblit 	/*
32c8a00a6eSMiri Korenblit 	 * Disable L0s without affecting L1;
33c8a00a6eSMiri Korenblit 	 * don't wait for ICH L0s (ICH bug W/A)
34c8a00a6eSMiri Korenblit 	 */
35c8a00a6eSMiri Korenblit 	iwl_set_bit(trans, CSR_GIO_CHICKEN_BITS,
36c8a00a6eSMiri Korenblit 		    CSR_GIO_CHICKEN_BITS_REG_BIT_L1A_NO_L0S_RX);
37c8a00a6eSMiri Korenblit 
38c8a00a6eSMiri Korenblit 	/* Set FH wait threshold to maximum (HW error during stress W/A) */
39c8a00a6eSMiri Korenblit 	iwl_set_bit(trans, CSR_DBG_HPET_MEM_REG, CSR_DBG_HPET_MEM_REG_VAL);
40c8a00a6eSMiri Korenblit 
41c8a00a6eSMiri Korenblit 	/*
42c8a00a6eSMiri Korenblit 	 * Enable HAP INTA (interrupt from management bus) to
43c8a00a6eSMiri Korenblit 	 * wake device's PCI Express link L1a -> L0s
44c8a00a6eSMiri Korenblit 	 */
45c8a00a6eSMiri Korenblit 	iwl_set_bit(trans, CSR_HW_IF_CONFIG_REG,
46c8a00a6eSMiri Korenblit 		    CSR_HW_IF_CONFIG_REG_HAP_WAKE);
47c8a00a6eSMiri Korenblit 
48c8a00a6eSMiri Korenblit 	iwl_pcie_apm_config(trans);
49c8a00a6eSMiri Korenblit 
50c8a00a6eSMiri Korenblit 	ret = iwl_finish_nic_init(trans);
51c8a00a6eSMiri Korenblit 	if (ret)
52c8a00a6eSMiri Korenblit 		return ret;
53c8a00a6eSMiri Korenblit 
54c8a00a6eSMiri Korenblit 	set_bit(STATUS_DEVICE_ENABLED, &trans->status);
55c8a00a6eSMiri Korenblit 
56c8a00a6eSMiri Korenblit 	return 0;
57c8a00a6eSMiri Korenblit }
58c8a00a6eSMiri Korenblit 
iwl_pcie_gen2_apm_stop(struct iwl_trans * trans,bool op_mode_leave)59c8a00a6eSMiri Korenblit static void iwl_pcie_gen2_apm_stop(struct iwl_trans *trans, bool op_mode_leave)
60c8a00a6eSMiri Korenblit {
61c8a00a6eSMiri Korenblit 	IWL_DEBUG_INFO(trans, "Stop card, put in low power state\n");
62c8a00a6eSMiri Korenblit 
63c8a00a6eSMiri Korenblit 	if (op_mode_leave) {
64c8a00a6eSMiri Korenblit 		if (!test_bit(STATUS_DEVICE_ENABLED, &trans->status))
65c8a00a6eSMiri Korenblit 			iwl_pcie_gen2_apm_init(trans);
66c8a00a6eSMiri Korenblit 
67c8a00a6eSMiri Korenblit 		/* inform ME that we are leaving */
68c8a00a6eSMiri Korenblit 		iwl_set_bit(trans, CSR_DBG_LINK_PWR_MGMT_REG,
69c8a00a6eSMiri Korenblit 			    CSR_RESET_LINK_PWR_MGMT_DISABLED);
70c8a00a6eSMiri Korenblit 		iwl_set_bit(trans, CSR_HW_IF_CONFIG_REG,
71c8a00a6eSMiri Korenblit 			    CSR_HW_IF_CONFIG_REG_WAKE_ME |
72c8a00a6eSMiri Korenblit 			    CSR_HW_IF_CONFIG_REG_WAKE_ME_PCIE_OWNER_EN);
73c8a00a6eSMiri Korenblit 		mdelay(1);
74c8a00a6eSMiri Korenblit 		iwl_clear_bit(trans, CSR_DBG_LINK_PWR_MGMT_REG,
75c8a00a6eSMiri Korenblit 			      CSR_RESET_LINK_PWR_MGMT_DISABLED);
76c8a00a6eSMiri Korenblit 		mdelay(5);
77c8a00a6eSMiri Korenblit 	}
78c8a00a6eSMiri Korenblit 
79c8a00a6eSMiri Korenblit 	clear_bit(STATUS_DEVICE_ENABLED, &trans->status);
80c8a00a6eSMiri Korenblit 
81c8a00a6eSMiri Korenblit 	/* Stop device's DMA activity */
82c8a00a6eSMiri Korenblit 	iwl_pcie_apm_stop_master(trans);
83c8a00a6eSMiri Korenblit 
84c8a00a6eSMiri Korenblit 	iwl_trans_pcie_sw_reset(trans, false);
85c8a00a6eSMiri Korenblit 
86c8a00a6eSMiri Korenblit 	/*
87c8a00a6eSMiri Korenblit 	 * Clear "initialization complete" bit to move adapter from
88c8a00a6eSMiri Korenblit 	 * D0A* (powered-up Active) --> D0U* (Uninitialized) state.
89c8a00a6eSMiri Korenblit 	 */
90c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_BZ)
91c8a00a6eSMiri Korenblit 		iwl_clear_bit(trans, CSR_GP_CNTRL,
92c8a00a6eSMiri Korenblit 			      CSR_GP_CNTRL_REG_FLAG_MAC_INIT);
93c8a00a6eSMiri Korenblit 	else
94c8a00a6eSMiri Korenblit 		iwl_clear_bit(trans, CSR_GP_CNTRL,
95c8a00a6eSMiri Korenblit 			      CSR_GP_CNTRL_REG_FLAG_INIT_DONE);
96c8a00a6eSMiri Korenblit }
97c8a00a6eSMiri Korenblit 
iwl_trans_pcie_fw_reset_handshake(struct iwl_trans * trans)98c8a00a6eSMiri Korenblit void iwl_trans_pcie_fw_reset_handshake(struct iwl_trans *trans)
99c8a00a6eSMiri Korenblit {
100c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
101c8a00a6eSMiri Korenblit 	int ret;
102c8a00a6eSMiri Korenblit 
103c8a00a6eSMiri Korenblit 	trans_pcie->fw_reset_state = FW_RESET_REQUESTED;
104c8a00a6eSMiri Korenblit 
105c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family < IWL_DEVICE_FAMILY_AX210)
106c8a00a6eSMiri Korenblit 		iwl_write_umac_prph(trans, UREG_NIC_SET_NMI_DRIVER,
107c8a00a6eSMiri Korenblit 				    UREG_NIC_SET_NMI_DRIVER_RESET_HANDSHAKE);
108c8a00a6eSMiri Korenblit 	else if (trans->mac_cfg->device_family == IWL_DEVICE_FAMILY_AX210)
109c8a00a6eSMiri Korenblit 		iwl_write_umac_prph(trans, UREG_DOORBELL_TO_ISR6,
110c8a00a6eSMiri Korenblit 				    UREG_DOORBELL_TO_ISR6_RESET_HANDSHAKE);
111c8a00a6eSMiri Korenblit 	else
112c8a00a6eSMiri Korenblit 		iwl_write32(trans, CSR_DOORBELL_VECTOR,
113c8a00a6eSMiri Korenblit 			    UREG_DOORBELL_TO_ISR6_RESET_HANDSHAKE);
114c8a00a6eSMiri Korenblit 
115c8a00a6eSMiri Korenblit 	/* wait 200ms */
116c8a00a6eSMiri Korenblit 	ret = wait_event_timeout(trans_pcie->fw_reset_waitq,
117c8a00a6eSMiri Korenblit 				 trans_pcie->fw_reset_state != FW_RESET_REQUESTED,
118c8a00a6eSMiri Korenblit 				 FW_RESET_TIMEOUT);
119c8a00a6eSMiri Korenblit 	if (!ret || trans_pcie->fw_reset_state == FW_RESET_ERROR) {
120c8a00a6eSMiri Korenblit 		bool reset_done;
121c8a00a6eSMiri Korenblit 		u32 inta_hw;
122c8a00a6eSMiri Korenblit 
123c8a00a6eSMiri Korenblit 		if (trans_pcie->msix_enabled) {
124c8a00a6eSMiri Korenblit 			inta_hw = iwl_read32(trans, CSR_MSIX_HW_INT_CAUSES_AD);
125c8a00a6eSMiri Korenblit 			reset_done =
126c8a00a6eSMiri Korenblit 				inta_hw & MSIX_HW_INT_CAUSES_REG_RESET_DONE;
127c8a00a6eSMiri Korenblit 		} else {
12843049a3cSJohannes Berg 			inta_hw = iwl_read32(trans, CSR_INT);
129c8a00a6eSMiri Korenblit 			reset_done = inta_hw & CSR_INT_BIT_RESET_DONE;
130c8a00a6eSMiri Korenblit 		}
131c8a00a6eSMiri Korenblit 
132c8a00a6eSMiri Korenblit 		IWL_ERR(trans,
133c8a00a6eSMiri Korenblit 			"timeout waiting for FW reset ACK (inta_hw=0x%x, reset_done %d)\n",
134c8a00a6eSMiri Korenblit 			inta_hw, reset_done);
135c8a00a6eSMiri Korenblit 
136c8a00a6eSMiri Korenblit 		if (!reset_done) {
137c8a00a6eSMiri Korenblit 			struct iwl_fw_error_dump_mode mode = {
138c8a00a6eSMiri Korenblit 				.type = IWL_ERR_TYPE_RESET_HS_TIMEOUT,
139c8a00a6eSMiri Korenblit 				.context = IWL_ERR_CONTEXT_FROM_OPMODE,
140c8a00a6eSMiri Korenblit 			};
141c8a00a6eSMiri Korenblit 			iwl_op_mode_nic_error(trans->op_mode,
142c8a00a6eSMiri Korenblit 					      IWL_ERR_TYPE_RESET_HS_TIMEOUT);
143c8a00a6eSMiri Korenblit 			iwl_op_mode_dump_error(trans->op_mode, &mode);
144c8a00a6eSMiri Korenblit 		}
145c8a00a6eSMiri Korenblit 	}
146c8a00a6eSMiri Korenblit 
147c8a00a6eSMiri Korenblit 	trans_pcie->fw_reset_state = FW_RESET_IDLE;
148c8a00a6eSMiri Korenblit }
149c8a00a6eSMiri Korenblit 
_iwl_trans_pcie_gen2_stop_device(struct iwl_trans * trans)150c8a00a6eSMiri Korenblit static void _iwl_trans_pcie_gen2_stop_device(struct iwl_trans *trans)
151c8a00a6eSMiri Korenblit {
152c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
153c8a00a6eSMiri Korenblit 
154c8a00a6eSMiri Korenblit 	lockdep_assert_held(&trans_pcie->mutex);
155c8a00a6eSMiri Korenblit 
156c8a00a6eSMiri Korenblit 	if (trans_pcie->is_down)
157c8a00a6eSMiri Korenblit 		return;
158c8a00a6eSMiri Korenblit 
159c8a00a6eSMiri Korenblit 	if (trans->state >= IWL_TRANS_FW_STARTED &&
160c8a00a6eSMiri Korenblit 	    trans->conf.fw_reset_handshake) {
161c8a00a6eSMiri Korenblit 		/*
162c8a00a6eSMiri Korenblit 		 * Reset handshake can dump firmware on timeout, but that
163c8a00a6eSMiri Korenblit 		 * should assume that the firmware is already dead.
164c8a00a6eSMiri Korenblit 		 */
165c8a00a6eSMiri Korenblit 		trans->state = IWL_TRANS_NO_FW;
166c8a00a6eSMiri Korenblit 		iwl_trans_pcie_fw_reset_handshake(trans);
167c8a00a6eSMiri Korenblit 	}
168c8a00a6eSMiri Korenblit 
169c8a00a6eSMiri Korenblit 	trans_pcie->is_down = true;
170c8a00a6eSMiri Korenblit 
171c8a00a6eSMiri Korenblit 	/* tell the device to stop sending interrupts */
172c8a00a6eSMiri Korenblit 	iwl_disable_interrupts(trans);
173c8a00a6eSMiri Korenblit 
174c8a00a6eSMiri Korenblit 	/* device going down, Stop using ICT table */
175c8a00a6eSMiri Korenblit 	iwl_pcie_disable_ict(trans);
176c8a00a6eSMiri Korenblit 
177c8a00a6eSMiri Korenblit 	/*
178c8a00a6eSMiri Korenblit 	 * If a HW restart happens during firmware loading,
179c8a00a6eSMiri Korenblit 	 * then the firmware loading might call this function
180c8a00a6eSMiri Korenblit 	 * and later it might be called again due to the
181c8a00a6eSMiri Korenblit 	 * restart. So don't process again if the device is
182c8a00a6eSMiri Korenblit 	 * already dead.
183c8a00a6eSMiri Korenblit 	 */
184c8a00a6eSMiri Korenblit 	if (test_and_clear_bit(STATUS_DEVICE_ENABLED, &trans->status)) {
185c8a00a6eSMiri Korenblit 		IWL_DEBUG_INFO(trans,
186c8a00a6eSMiri Korenblit 			       "DEVICE_ENABLED bit was set and is now cleared\n");
187c8a00a6eSMiri Korenblit 		iwl_pcie_synchronize_irqs(trans);
188c8a00a6eSMiri Korenblit 		iwl_pcie_rx_napi_sync(trans);
189c8a00a6eSMiri Korenblit 		iwl_txq_gen2_tx_free(trans);
190c8a00a6eSMiri Korenblit 		iwl_pcie_rx_stop(trans);
191c8a00a6eSMiri Korenblit 	}
192c8a00a6eSMiri Korenblit 
193c8a00a6eSMiri Korenblit 	iwl_pcie_ctxt_info_free_paging(trans);
194c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_AX210)
195c8a00a6eSMiri Korenblit 		iwl_pcie_ctxt_info_v2_free(trans, false);
196c8a00a6eSMiri Korenblit 	else
197c8a00a6eSMiri Korenblit 		iwl_pcie_ctxt_info_free(trans);
198c8a00a6eSMiri Korenblit 
199c8a00a6eSMiri Korenblit 	/* Stop the device, and put it in low power state */
200c8a00a6eSMiri Korenblit 	iwl_pcie_gen2_apm_stop(trans, false);
201c8a00a6eSMiri Korenblit 
202c8a00a6eSMiri Korenblit 	/* re-take ownership to prevent other users from stealing the device */
203c8a00a6eSMiri Korenblit 	iwl_trans_pcie_sw_reset(trans, true);
204c8a00a6eSMiri Korenblit 
205c8a00a6eSMiri Korenblit 	/*
206c8a00a6eSMiri Korenblit 	 * Upon stop, the IVAR table gets erased, so msi-x won't
207c8a00a6eSMiri Korenblit 	 * work. This causes a bug in RF-KILL flows, since the interrupt
208c8a00a6eSMiri Korenblit 	 * that enables radio won't fire on the correct irq, and the
209c8a00a6eSMiri Korenblit 	 * driver won't be able to handle the interrupt.
210c8a00a6eSMiri Korenblit 	 * Configure the IVAR table again after reset.
211c8a00a6eSMiri Korenblit 	 */
212c8a00a6eSMiri Korenblit 	iwl_pcie_conf_msix_hw(trans_pcie);
213c8a00a6eSMiri Korenblit 
214c8a00a6eSMiri Korenblit 	/*
215c8a00a6eSMiri Korenblit 	 * Upon stop, the APM issues an interrupt if HW RF kill is set.
216c8a00a6eSMiri Korenblit 	 * This is a bug in certain verions of the hardware.
217c8a00a6eSMiri Korenblit 	 * Certain devices also keep sending HW RF kill interrupt all
218c8a00a6eSMiri Korenblit 	 * the time, unless the interrupt is ACKed even if the interrupt
219c8a00a6eSMiri Korenblit 	 * should be masked. Re-ACK all the interrupts here.
220c8a00a6eSMiri Korenblit 	 */
221c8a00a6eSMiri Korenblit 	iwl_disable_interrupts(trans);
222c8a00a6eSMiri Korenblit 
223c8a00a6eSMiri Korenblit 	/* clear all status bits */
224c8a00a6eSMiri Korenblit 	clear_bit(STATUS_SYNC_HCMD_ACTIVE, &trans->status);
225c8a00a6eSMiri Korenblit 	clear_bit(STATUS_INT_ENABLED, &trans->status);
226c8a00a6eSMiri Korenblit 	clear_bit(STATUS_TPOWER_PMI, &trans->status);
227c8a00a6eSMiri Korenblit 
228c8a00a6eSMiri Korenblit 	/*
229c8a00a6eSMiri Korenblit 	 * Even if we stop the HW, we still want the RF kill
230c8a00a6eSMiri Korenblit 	 * interrupt
231c8a00a6eSMiri Korenblit 	 */
232c8a00a6eSMiri Korenblit 	iwl_enable_rfkill_int(trans);
233c8a00a6eSMiri Korenblit }
234c8a00a6eSMiri Korenblit 
iwl_trans_pcie_gen2_stop_device(struct iwl_trans * trans)235c8a00a6eSMiri Korenblit void iwl_trans_pcie_gen2_stop_device(struct iwl_trans *trans)
236c8a00a6eSMiri Korenblit {
237c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
238c8a00a6eSMiri Korenblit 	bool was_in_rfkill;
239c8a00a6eSMiri Korenblit 
240c8a00a6eSMiri Korenblit 	iwl_op_mode_time_point(trans->op_mode,
241c8a00a6eSMiri Korenblit 			       IWL_FW_INI_TIME_POINT_HOST_DEVICE_DISABLE,
242c8a00a6eSMiri Korenblit 			       NULL);
243c8a00a6eSMiri Korenblit 
244c8a00a6eSMiri Korenblit 	mutex_lock(&trans_pcie->mutex);
245c8a00a6eSMiri Korenblit 	trans_pcie->opmode_down = true;
246c8a00a6eSMiri Korenblit 	was_in_rfkill = test_bit(STATUS_RFKILL_OPMODE, &trans->status);
247c8a00a6eSMiri Korenblit 	_iwl_trans_pcie_gen2_stop_device(trans);
248c8a00a6eSMiri Korenblit 	iwl_trans_pcie_handle_stop_rfkill(trans, was_in_rfkill);
249c8a00a6eSMiri Korenblit 	mutex_unlock(&trans_pcie->mutex);
250c8a00a6eSMiri Korenblit }
251c8a00a6eSMiri Korenblit 
iwl_pcie_gen2_nic_init(struct iwl_trans * trans)252c8a00a6eSMiri Korenblit static int iwl_pcie_gen2_nic_init(struct iwl_trans *trans)
253c8a00a6eSMiri Korenblit {
254c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
255c8a00a6eSMiri Korenblit 	int queue_size = max_t(u32, IWL_CMD_QUEUE_SIZE,
256c8a00a6eSMiri Korenblit 			       trans->mac_cfg->base->min_txq_size);
257c8a00a6eSMiri Korenblit 	int ret;
258c8a00a6eSMiri Korenblit 
259c8a00a6eSMiri Korenblit 	/* TODO: most of the logic can be removed in A0 - but not in Z0 */
260c8a00a6eSMiri Korenblit 	spin_lock_bh(&trans_pcie->irq_lock);
261c8a00a6eSMiri Korenblit 	ret = iwl_pcie_gen2_apm_init(trans);
262c8a00a6eSMiri Korenblit 	spin_unlock_bh(&trans_pcie->irq_lock);
263c8a00a6eSMiri Korenblit 	if (ret)
264c8a00a6eSMiri Korenblit 		return ret;
265c8a00a6eSMiri Korenblit 
266c8a00a6eSMiri Korenblit 	iwl_op_mode_nic_config(trans->op_mode);
267c8a00a6eSMiri Korenblit 
268c8a00a6eSMiri Korenblit 	/* Allocate the RX queue, or reset if it is already allocated */
269c8a00a6eSMiri Korenblit 	if (iwl_pcie_gen2_rx_init(trans))
270c8a00a6eSMiri Korenblit 		return -ENOMEM;
271c8a00a6eSMiri Korenblit 
272c8a00a6eSMiri Korenblit 	/* Allocate or reset and init all Tx and Command queues */
273c8a00a6eSMiri Korenblit 	if (iwl_txq_gen2_init(trans, trans->conf.cmd_queue, queue_size))
274c8a00a6eSMiri Korenblit 		return -ENOMEM;
275c8a00a6eSMiri Korenblit 
276c8a00a6eSMiri Korenblit 	/* enable shadow regs in HW */
277c8a00a6eSMiri Korenblit 	iwl_set_bit(trans, CSR_MAC_SHADOW_REG_CTRL, 0x800FFFFF);
278c8a00a6eSMiri Korenblit 	IWL_DEBUG_INFO(trans, "Enabling shadow registers in device\n");
279c8a00a6eSMiri Korenblit 
280c8a00a6eSMiri Korenblit 	return 0;
281c8a00a6eSMiri Korenblit }
282c8a00a6eSMiri Korenblit 
iwl_pcie_get_rf_name(struct iwl_trans * trans)283c8a00a6eSMiri Korenblit static void iwl_pcie_get_rf_name(struct iwl_trans *trans)
284c8a00a6eSMiri Korenblit {
285c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
286c8a00a6eSMiri Korenblit 	char *buf = trans_pcie->rf_name;
287c8a00a6eSMiri Korenblit 	size_t buflen = sizeof(trans_pcie->rf_name);
288c8a00a6eSMiri Korenblit 	size_t pos;
289c8a00a6eSMiri Korenblit 	u32 version;
290c8a00a6eSMiri Korenblit 
291c8a00a6eSMiri Korenblit 	if (buf[0])
292c8a00a6eSMiri Korenblit 		return;
293c8a00a6eSMiri Korenblit 
294c8a00a6eSMiri Korenblit 	switch (CSR_HW_RFID_TYPE(trans->info.hw_rf_id)) {
295c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_JF):
296c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "JF");
297c8a00a6eSMiri Korenblit 		break;
298c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_GF):
299c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "GF");
300c8a00a6eSMiri Korenblit 		break;
301c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_GF4):
302c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "GF4");
303c8a00a6eSMiri Korenblit 		break;
304c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_HR):
305c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "HR");
306c8a00a6eSMiri Korenblit 		break;
307c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_HR1):
308c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "HR1");
309c8a00a6eSMiri Korenblit 		break;
310c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_HRCDB):
311c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "HRCDB");
312c8a00a6eSMiri Korenblit 		break;
313c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_FM):
314c8a00a6eSMiri Korenblit 		pos = scnprintf(buf, buflen, "FM");
315c8a00a6eSMiri Korenblit 		break;
316c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_WP):
317c8a00a6eSMiri Korenblit 		if (SILICON_Z_STEP ==
318c8a00a6eSMiri Korenblit 		    CSR_HW_RFID_STEP(trans->info.hw_rf_id))
319c8a00a6eSMiri Korenblit 			pos = scnprintf(buf, buflen, "WHTC");
320c8a00a6eSMiri Korenblit 		else
321c8a00a6eSMiri Korenblit 			pos = scnprintf(buf, buflen, "WH");
322c8a00a6eSMiri Korenblit 		break;
323c8a00a6eSMiri Korenblit 	default:
324c8a00a6eSMiri Korenblit 		return;
325c8a00a6eSMiri Korenblit 	}
326c8a00a6eSMiri Korenblit 
327c8a00a6eSMiri Korenblit 	switch (CSR_HW_RFID_TYPE(trans->info.hw_rf_id)) {
328c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_HR):
329c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_HR1):
330c8a00a6eSMiri Korenblit 	case CSR_HW_RFID_TYPE(CSR_HW_RF_ID_TYPE_HRCDB):
331c8a00a6eSMiri Korenblit 		version = iwl_read_prph(trans, CNVI_MBOX_C);
332c8a00a6eSMiri Korenblit 		switch (version) {
333c8a00a6eSMiri Korenblit 		case 0x20000:
334c8a00a6eSMiri Korenblit 			pos += scnprintf(buf + pos, buflen - pos, " B3");
335c8a00a6eSMiri Korenblit 			break;
336c8a00a6eSMiri Korenblit 		case 0x120000:
337c8a00a6eSMiri Korenblit 			pos += scnprintf(buf + pos, buflen - pos, " B5");
338c8a00a6eSMiri Korenblit 			break;
339c8a00a6eSMiri Korenblit 		default:
340c8a00a6eSMiri Korenblit 			pos += scnprintf(buf + pos, buflen - pos,
341c8a00a6eSMiri Korenblit 					 " (0x%x)", version);
342c8a00a6eSMiri Korenblit 			break;
343c8a00a6eSMiri Korenblit 		}
344c8a00a6eSMiri Korenblit 		break;
345c8a00a6eSMiri Korenblit 	default:
346c8a00a6eSMiri Korenblit 		break;
347c8a00a6eSMiri Korenblit 	}
348c8a00a6eSMiri Korenblit 
349c8a00a6eSMiri Korenblit 	pos += scnprintf(buf + pos, buflen - pos, ", rfid=0x%x",
350c8a00a6eSMiri Korenblit 			 trans->info.hw_rf_id);
351c8a00a6eSMiri Korenblit 
352c8a00a6eSMiri Korenblit 	IWL_INFO(trans, "Detected RF %s\n", buf);
353c8a00a6eSMiri Korenblit 
354c8a00a6eSMiri Korenblit 	/*
355c8a00a6eSMiri Korenblit 	 * also add a \n for debugfs - need to do it after printing
356c8a00a6eSMiri Korenblit 	 * since our IWL_INFO machinery wants to see a static \n at
357c8a00a6eSMiri Korenblit 	 * the end of the string
358c8a00a6eSMiri Korenblit 	 */
359c8a00a6eSMiri Korenblit 	pos += scnprintf(buf + pos, buflen - pos, "\n");
360c8a00a6eSMiri Korenblit }
361c8a00a6eSMiri Korenblit 
iwl_trans_pcie_gen2_fw_alive(struct iwl_trans * trans)362c8a00a6eSMiri Korenblit void iwl_trans_pcie_gen2_fw_alive(struct iwl_trans *trans)
363c8a00a6eSMiri Korenblit {
364c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
365c8a00a6eSMiri Korenblit 
366c8a00a6eSMiri Korenblit 	iwl_pcie_reset_ict(trans);
367c8a00a6eSMiri Korenblit 
368c8a00a6eSMiri Korenblit 	/* make sure all queue are not stopped/used */
369c8a00a6eSMiri Korenblit 	memset(trans_pcie->txqs.queue_stopped, 0,
370c8a00a6eSMiri Korenblit 	       sizeof(trans_pcie->txqs.queue_stopped));
371c8a00a6eSMiri Korenblit 	memset(trans_pcie->txqs.queue_used, 0,
372c8a00a6eSMiri Korenblit 	       sizeof(trans_pcie->txqs.queue_used));
373c8a00a6eSMiri Korenblit 
374c8a00a6eSMiri Korenblit 	/* now that we got alive we can free the fw image & the context info.
375c8a00a6eSMiri Korenblit 	 * paging memory cannot be freed included since FW will still use it
376c8a00a6eSMiri Korenblit 	 */
377c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_AX210)
378c8a00a6eSMiri Korenblit 		iwl_pcie_ctxt_info_v2_free(trans, true);
379c8a00a6eSMiri Korenblit 	else
380c8a00a6eSMiri Korenblit 		iwl_pcie_ctxt_info_free(trans);
381c8a00a6eSMiri Korenblit 
382c8a00a6eSMiri Korenblit 	/*
383c8a00a6eSMiri Korenblit 	 * Re-enable all the interrupts, including the RF-Kill one, now that
384c8a00a6eSMiri Korenblit 	 * the firmware is alive.
385c8a00a6eSMiri Korenblit 	 */
386c8a00a6eSMiri Korenblit 	iwl_enable_interrupts(trans);
387c8a00a6eSMiri Korenblit 	mutex_lock(&trans_pcie->mutex);
388c8a00a6eSMiri Korenblit 	iwl_pcie_check_hw_rf_kill(trans);
389c8a00a6eSMiri Korenblit 
390c8a00a6eSMiri Korenblit 	iwl_pcie_get_rf_name(trans);
391c8a00a6eSMiri Korenblit 	mutex_unlock(&trans_pcie->mutex);
392c8a00a6eSMiri Korenblit 
393c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_BZ)
394c8a00a6eSMiri Korenblit 		trans->step_urm = !!(iwl_read_umac_prph(trans,
395c8a00a6eSMiri Korenblit 							CNVI_PMU_STEP_FLOW) &
396c8a00a6eSMiri Korenblit 					CNVI_PMU_STEP_FLOW_FORCE_URM);
397c8a00a6eSMiri Korenblit }
398c8a00a6eSMiri Korenblit 
iwl_pcie_set_ltr(struct iwl_trans * trans)399c8a00a6eSMiri Korenblit static bool iwl_pcie_set_ltr(struct iwl_trans *trans)
400c8a00a6eSMiri Korenblit {
401c8a00a6eSMiri Korenblit 	u32 ltr_val = CSR_LTR_LONG_VAL_AD_NO_SNOOP_REQ |
402c8a00a6eSMiri Korenblit 		      u32_encode_bits(CSR_LTR_LONG_VAL_AD_SCALE_USEC,
403c8a00a6eSMiri Korenblit 				      CSR_LTR_LONG_VAL_AD_NO_SNOOP_SCALE) |
404c8a00a6eSMiri Korenblit 		      u32_encode_bits(250,
405c8a00a6eSMiri Korenblit 				      CSR_LTR_LONG_VAL_AD_NO_SNOOP_VAL) |
406c8a00a6eSMiri Korenblit 		      CSR_LTR_LONG_VAL_AD_SNOOP_REQ |
407c8a00a6eSMiri Korenblit 		      u32_encode_bits(CSR_LTR_LONG_VAL_AD_SCALE_USEC,
408c8a00a6eSMiri Korenblit 				      CSR_LTR_LONG_VAL_AD_SNOOP_SCALE) |
409c8a00a6eSMiri Korenblit 		      u32_encode_bits(250, CSR_LTR_LONG_VAL_AD_SNOOP_VAL);
410c8a00a6eSMiri Korenblit 
411c8a00a6eSMiri Korenblit 	/*
412c8a00a6eSMiri Korenblit 	 * To workaround hardware latency issues during the boot process,
413c8a00a6eSMiri Korenblit 	 * initialize the LTR to ~250 usec (see ltr_val above).
414c8a00a6eSMiri Korenblit 	 * The firmware initializes this again later (to a smaller value).
415c8a00a6eSMiri Korenblit 	 */
416c8a00a6eSMiri Korenblit 	if ((trans->mac_cfg->device_family == IWL_DEVICE_FAMILY_AX210 ||
417c8a00a6eSMiri Korenblit 	     trans->mac_cfg->device_family == IWL_DEVICE_FAMILY_22000) &&
418c8a00a6eSMiri Korenblit 	    !trans->mac_cfg->integrated) {
419c8a00a6eSMiri Korenblit 		iwl_write32(trans, CSR_LTR_LONG_VAL_AD, ltr_val);
420c8a00a6eSMiri Korenblit 		return true;
421c8a00a6eSMiri Korenblit 	}
422c8a00a6eSMiri Korenblit 
423c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->integrated &&
424c8a00a6eSMiri Korenblit 	    trans->mac_cfg->device_family == IWL_DEVICE_FAMILY_22000) {
425c8a00a6eSMiri Korenblit 		iwl_write_prph(trans, HPM_MAC_LTR_CSR, HPM_MAC_LRT_ENABLE_ALL);
426c8a00a6eSMiri Korenblit 		iwl_write_prph(trans, HPM_UMAC_LTR, ltr_val);
427c8a00a6eSMiri Korenblit 		return true;
428c8a00a6eSMiri Korenblit 	}
429c8a00a6eSMiri Korenblit 
430c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family == IWL_DEVICE_FAMILY_AX210) {
431c8a00a6eSMiri Korenblit 		/* First clear the interrupt, just in case */
432c8a00a6eSMiri Korenblit 		iwl_write32(trans, CSR_MSIX_HW_INT_CAUSES_AD,
433c8a00a6eSMiri Korenblit 			    MSIX_HW_INT_CAUSES_REG_IML);
434c8a00a6eSMiri Korenblit 		/* In this case, unfortunately the same ROM bug exists in the
435c8a00a6eSMiri Korenblit 		 * device (not setting LTR correctly), but we don't have control
436c8a00a6eSMiri Korenblit 		 * over the settings from the host due to some hardware security
437c8a00a6eSMiri Korenblit 		 * features. The only workaround we've been able to come up with
438c8a00a6eSMiri Korenblit 		 * so far is to try to keep the CPU and device busy by polling
439c8a00a6eSMiri Korenblit 		 * it and the IML (image loader) completed interrupt.
440c8a00a6eSMiri Korenblit 		 */
441c8a00a6eSMiri Korenblit 		return false;
442c8a00a6eSMiri Korenblit 	}
443c8a00a6eSMiri Korenblit 
444c8a00a6eSMiri Korenblit 	/* nothing needs to be done on other devices */
445c8a00a6eSMiri Korenblit 	return true;
446c8a00a6eSMiri Korenblit }
447c8a00a6eSMiri Korenblit 
iwl_pcie_spin_for_iml(struct iwl_trans * trans)448c8a00a6eSMiri Korenblit static void iwl_pcie_spin_for_iml(struct iwl_trans *trans)
449c8a00a6eSMiri Korenblit {
450c8a00a6eSMiri Korenblit /* in practice, this seems to complete in around 20-30ms at most, wait 100 */
451c8a00a6eSMiri Korenblit #define IML_WAIT_TIMEOUT	(HZ / 10)
452c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
453c8a00a6eSMiri Korenblit 	unsigned long end_time = jiffies + IML_WAIT_TIMEOUT;
454c8a00a6eSMiri Korenblit 	u32 value, loops = 0;
455c8a00a6eSMiri Korenblit 	bool irq = false;
456c8a00a6eSMiri Korenblit 
457c8a00a6eSMiri Korenblit 	if (WARN_ON(!trans_pcie->iml))
458c8a00a6eSMiri Korenblit 		return;
459c8a00a6eSMiri Korenblit 
460c8a00a6eSMiri Korenblit 	value = iwl_read32(trans, CSR_LTR_LAST_MSG);
461c8a00a6eSMiri Korenblit 	IWL_DEBUG_INFO(trans, "Polling for IML load - CSR_LTR_LAST_MSG=0x%x\n",
462c8a00a6eSMiri Korenblit 		       value);
463c8a00a6eSMiri Korenblit 
464c8a00a6eSMiri Korenblit 	while (time_before(jiffies, end_time)) {
465c8a00a6eSMiri Korenblit 		if (iwl_read32(trans, CSR_MSIX_HW_INT_CAUSES_AD) &
466c8a00a6eSMiri Korenblit 				MSIX_HW_INT_CAUSES_REG_IML) {
467c8a00a6eSMiri Korenblit 			irq = true;
468c8a00a6eSMiri Korenblit 			break;
469c8a00a6eSMiri Korenblit 		}
470c8a00a6eSMiri Korenblit 		/* Keep the CPU and device busy. */
471c8a00a6eSMiri Korenblit 		value = iwl_read32(trans, CSR_LTR_LAST_MSG);
472c8a00a6eSMiri Korenblit 		loops++;
473c8a00a6eSMiri Korenblit 	}
474c8a00a6eSMiri Korenblit 
475c8a00a6eSMiri Korenblit 	IWL_DEBUG_INFO(trans,
476c8a00a6eSMiri Korenblit 		       "Polled for IML load: irq=%d, loops=%d, CSR_LTR_LAST_MSG=0x%x\n",
477c8a00a6eSMiri Korenblit 		       irq, loops, value);
478c8a00a6eSMiri Korenblit 
479c8a00a6eSMiri Korenblit 	/* We don't fail here even if we timed out - maybe we get lucky and the
480c8a00a6eSMiri Korenblit 	 * interrupt comes in later (and we get alive from firmware) and then
481c8a00a6eSMiri Korenblit 	 * we're all happy - but if not we'll fail on alive timeout or get some
482c8a00a6eSMiri Korenblit 	 * other error out.
483c8a00a6eSMiri Korenblit 	 */
484c8a00a6eSMiri Korenblit }
485c8a00a6eSMiri Korenblit 
iwl_trans_pcie_gen2_start_fw(struct iwl_trans * trans,const struct iwl_fw * fw,const struct fw_img * img,bool run_in_rfkill)486c8a00a6eSMiri Korenblit int iwl_trans_pcie_gen2_start_fw(struct iwl_trans *trans,
487c8a00a6eSMiri Korenblit 				 const struct iwl_fw *fw,
488c8a00a6eSMiri Korenblit 				 const struct fw_img *img,
489c8a00a6eSMiri Korenblit 				 bool run_in_rfkill)
490c8a00a6eSMiri Korenblit {
491c8a00a6eSMiri Korenblit 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
492c8a00a6eSMiri Korenblit 	bool hw_rfkill, keep_ram_busy;
493c8a00a6eSMiri Korenblit 	bool top_reset_done = false;
494c8a00a6eSMiri Korenblit 	int ret;
495c8a00a6eSMiri Korenblit 
496c8a00a6eSMiri Korenblit 	mutex_lock(&trans_pcie->mutex);
497c8a00a6eSMiri Korenblit again:
498c8a00a6eSMiri Korenblit 	/* This may fail if AMT took ownership of the device */
499c8a00a6eSMiri Korenblit 	if (iwl_pcie_prepare_card_hw(trans)) {
500c8a00a6eSMiri Korenblit 		IWL_WARN(trans, "Exit HW not ready\n");
501c8a00a6eSMiri Korenblit 		ret = -EIO;
502c8a00a6eSMiri Korenblit 		goto out;
503c8a00a6eSMiri Korenblit 	}
504c8a00a6eSMiri Korenblit 
505c8a00a6eSMiri Korenblit 	iwl_enable_rfkill_int(trans);
506c8a00a6eSMiri Korenblit 
507c8a00a6eSMiri Korenblit 	iwl_write32(trans, CSR_INT, 0xFFFFFFFF);
508c8a00a6eSMiri Korenblit 
509c8a00a6eSMiri Korenblit 	/*
510c8a00a6eSMiri Korenblit 	 * We enabled the RF-Kill interrupt and the handler may very
511c8a00a6eSMiri Korenblit 	 * well be running. Disable the interrupts to make sure no other
512c8a00a6eSMiri Korenblit 	 * interrupt can be fired.
513c8a00a6eSMiri Korenblit 	 */
514c8a00a6eSMiri Korenblit 	iwl_disable_interrupts(trans);
515c8a00a6eSMiri Korenblit 
516c8a00a6eSMiri Korenblit 	/* Make sure it finished running */
517c8a00a6eSMiri Korenblit 	iwl_pcie_synchronize_irqs(trans);
518c8a00a6eSMiri Korenblit 
519c8a00a6eSMiri Korenblit 	/* If platform's RF_KILL switch is NOT set to KILL */
520c8a00a6eSMiri Korenblit 	hw_rfkill = iwl_pcie_check_hw_rf_kill(trans);
521c8a00a6eSMiri Korenblit 	if (hw_rfkill && !run_in_rfkill) {
522c8a00a6eSMiri Korenblit 		ret = -ERFKILL;
523c8a00a6eSMiri Korenblit 		goto out;
524c8a00a6eSMiri Korenblit 	}
525c8a00a6eSMiri Korenblit 
526c8a00a6eSMiri Korenblit 	/* Someone called stop_device, don't try to start_fw */
527c8a00a6eSMiri Korenblit 	if (trans_pcie->is_down) {
528c8a00a6eSMiri Korenblit 		IWL_WARN(trans,
529c8a00a6eSMiri Korenblit 			 "Can't start_fw since the HW hasn't been started\n");
530c8a00a6eSMiri Korenblit 		ret = -EIO;
531c8a00a6eSMiri Korenblit 		goto out;
532c8a00a6eSMiri Korenblit 	}
533c8a00a6eSMiri Korenblit 
534c8a00a6eSMiri Korenblit 	/* make sure rfkill handshake bits are cleared */
535c8a00a6eSMiri Korenblit 	iwl_write32(trans, CSR_UCODE_DRV_GP1_CLR, CSR_UCODE_SW_BIT_RFKILL);
536c8a00a6eSMiri Korenblit 	iwl_write32(trans, CSR_UCODE_DRV_GP1_CLR,
537c8a00a6eSMiri Korenblit 		    CSR_UCODE_DRV_GP1_BIT_CMD_BLOCKED);
538c8a00a6eSMiri Korenblit 
539c8a00a6eSMiri Korenblit 	/* clear (again), then enable host interrupts */
540c8a00a6eSMiri Korenblit 	iwl_write32(trans, CSR_INT, 0xFFFFFFFF);
541c8a00a6eSMiri Korenblit 
542c8a00a6eSMiri Korenblit 	ret = iwl_pcie_gen2_nic_init(trans);
543c8a00a6eSMiri Korenblit 	if (ret) {
544c8a00a6eSMiri Korenblit 		IWL_ERR(trans, "Unable to init nic\n");
545c8a00a6eSMiri Korenblit 		goto out;
546c8a00a6eSMiri Korenblit 	}
547c8a00a6eSMiri Korenblit 
548c8a00a6eSMiri Korenblit 	if (WARN_ON(trans->do_top_reset &&
549*af2d6148SJakub Kicinski 		    trans->mac_cfg->device_family < IWL_DEVICE_FAMILY_SC)) {
550*af2d6148SJakub Kicinski 		ret = -EINVAL;
551*af2d6148SJakub Kicinski 		goto out;
552*af2d6148SJakub Kicinski 	}
553c8a00a6eSMiri Korenblit 
554c8a00a6eSMiri Korenblit 	/* we need to wait later - set state */
555c8a00a6eSMiri Korenblit 	if (trans->do_top_reset)
556c8a00a6eSMiri Korenblit 		trans_pcie->fw_reset_state = FW_RESET_TOP_REQUESTED;
557c8a00a6eSMiri Korenblit 
558c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_AX210) {
559c8a00a6eSMiri Korenblit 		if (!top_reset_done) {
560c8a00a6eSMiri Korenblit 			ret = iwl_pcie_ctxt_info_v2_alloc(trans, fw, img);
561c8a00a6eSMiri Korenblit 			if (ret)
562c8a00a6eSMiri Korenblit 				goto out;
563c8a00a6eSMiri Korenblit 		}
564c8a00a6eSMiri Korenblit 
565c8a00a6eSMiri Korenblit 		iwl_pcie_ctxt_info_v2_kick(trans);
566c8a00a6eSMiri Korenblit 	} else {
567c8a00a6eSMiri Korenblit 		ret = iwl_pcie_ctxt_info_init(trans, img);
568c8a00a6eSMiri Korenblit 		if (ret)
569c8a00a6eSMiri Korenblit 			goto out;
570c8a00a6eSMiri Korenblit 	}
571c8a00a6eSMiri Korenblit 
572c8a00a6eSMiri Korenblit 	keep_ram_busy = !iwl_pcie_set_ltr(trans);
573c8a00a6eSMiri Korenblit 
574c8a00a6eSMiri Korenblit 	if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) {
575c8a00a6eSMiri Korenblit 		IWL_DEBUG_POWER(trans, "function scratch register value is 0x%08x\n",
576c8a00a6eSMiri Korenblit 				iwl_read32(trans, CSR_FUNC_SCRATCH));
577c8a00a6eSMiri Korenblit 		iwl_write32(trans, CSR_FUNC_SCRATCH, CSR_FUNC_SCRATCH_INIT_VALUE);
578c8a00a6eSMiri Korenblit 		iwl_set_bit(trans, CSR_GP_CNTRL,
579c8a00a6eSMiri Korenblit 			    CSR_GP_CNTRL_REG_FLAG_ROM_START);
580c8a00a6eSMiri Korenblit 	} else if (trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_AX210) {
581c8a00a6eSMiri Korenblit 		iwl_write_umac_prph(trans, UREG_CPU_INIT_RUN, 1);
582c8a00a6eSMiri Korenblit 	} else {
583c8a00a6eSMiri Korenblit 		iwl_write_prph(trans, UREG_CPU_INIT_RUN, 1);
584c8a00a6eSMiri Korenblit 	}
585c8a00a6eSMiri Korenblit 
586c8a00a6eSMiri Korenblit 	if (keep_ram_busy)
587c8a00a6eSMiri Korenblit 		iwl_pcie_spin_for_iml(trans);
588c8a00a6eSMiri Korenblit 
589c8a00a6eSMiri Korenblit 	if (trans->do_top_reset) {
590c8a00a6eSMiri Korenblit 		trans->do_top_reset = 0;
591c8a00a6eSMiri Korenblit 
592c8a00a6eSMiri Korenblit #define FW_TOP_RESET_TIMEOUT	(HZ / 4)
593c8a00a6eSMiri Korenblit 		ret = wait_event_timeout(trans_pcie->fw_reset_waitq,
594c8a00a6eSMiri Korenblit 					 trans_pcie->fw_reset_state != FW_RESET_TOP_REQUESTED,
595c8a00a6eSMiri Korenblit 					 FW_TOP_RESET_TIMEOUT);
596c8a00a6eSMiri Korenblit 
597c8a00a6eSMiri Korenblit 		if (trans_pcie->fw_reset_state != FW_RESET_OK) {
598c8a00a6eSMiri Korenblit 			if (trans_pcie->fw_reset_state != FW_RESET_TOP_REQUESTED)
599c8a00a6eSMiri Korenblit 				IWL_ERR(trans,
600c8a00a6eSMiri Korenblit 					"TOP reset interrupted by error (state %d)!\n",
601c8a00a6eSMiri Korenblit 					trans_pcie->fw_reset_state);
602c8a00a6eSMiri Korenblit 			else
603c8a00a6eSMiri Korenblit 				IWL_ERR(trans, "TOP reset timed out!\n");
604c8a00a6eSMiri Korenblit 			iwl_op_mode_nic_error(trans->op_mode,
605c8a00a6eSMiri Korenblit 					      IWL_ERR_TYPE_TOP_RESET_FAILED);
606c8a00a6eSMiri Korenblit 			iwl_trans_schedule_reset(trans,
607c8a00a6eSMiri Korenblit 						 IWL_ERR_TYPE_TOP_RESET_FAILED);
608c8a00a6eSMiri Korenblit 			ret = -EIO;
609c8a00a6eSMiri Korenblit 			goto out;
610c8a00a6eSMiri Korenblit 		}
611c8a00a6eSMiri Korenblit 
612c8a00a6eSMiri Korenblit 		msleep(10);
613c8a00a6eSMiri Korenblit 		IWL_INFO(trans, "TOP reset successful, reinit now\n");
614c8a00a6eSMiri Korenblit 		/* now load the firmware again properly */
615eda36f51SJohannes Berg 		ret = _iwl_trans_pcie_start_hw(trans);
616eda36f51SJohannes Berg 		if (ret) {
617eda36f51SJohannes Berg 			IWL_ERR(trans, "failed to start HW after TOP reset\n");
618eda36f51SJohannes Berg 			goto out;
619eda36f51SJohannes Berg 		}
620c8a00a6eSMiri Korenblit 		trans_pcie->prph_scratch->ctrl_cfg.control.control_flags &=
621c8a00a6eSMiri Korenblit 			~cpu_to_le32(IWL_PRPH_SCRATCH_TOP_RESET);
622c8a00a6eSMiri Korenblit 		top_reset_done = true;
623c8a00a6eSMiri Korenblit 		goto again;
624c8a00a6eSMiri Korenblit 	}
625c8a00a6eSMiri Korenblit 
626c8a00a6eSMiri Korenblit 	/* re-check RF-Kill state since we may have missed the interrupt */
627c8a00a6eSMiri Korenblit 	hw_rfkill = iwl_pcie_check_hw_rf_kill(trans);
628c8a00a6eSMiri Korenblit 	if (hw_rfkill && !run_in_rfkill)
629c8a00a6eSMiri Korenblit 		ret = -ERFKILL;
630c8a00a6eSMiri Korenblit 
631c8a00a6eSMiri Korenblit out:
632c8a00a6eSMiri Korenblit 	mutex_unlock(&trans_pcie->mutex);
633c8a00a6eSMiri Korenblit 	return ret;
634c8a00a6eSMiri Korenblit }
635be27286fSItamar Shalev 
iwl_trans_pcie_gen2_op_mode_leave(struct iwl_trans * trans)636be27286fSItamar Shalev void iwl_trans_pcie_gen2_op_mode_leave(struct iwl_trans *trans)
637be27286fSItamar Shalev {
638be27286fSItamar Shalev 	struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
639be27286fSItamar Shalev 
640be27286fSItamar Shalev 	mutex_lock(&trans_pcie->mutex);
641be27286fSItamar Shalev 
642be27286fSItamar Shalev 	/* disable interrupts - don't enable HW RF kill interrupt */
643be27286fSItamar Shalev 	iwl_disable_interrupts(trans);
644be27286fSItamar Shalev 
645be27286fSItamar Shalev 	iwl_pcie_gen2_apm_stop(trans, true);
646be27286fSItamar Shalev 
647be27286fSItamar Shalev 	iwl_disable_interrupts(trans);
648be27286fSItamar Shalev 
649be27286fSItamar Shalev 	iwl_pcie_disable_ict(trans);
650be27286fSItamar Shalev 
651be27286fSItamar Shalev 	mutex_unlock(&trans_pcie->mutex);
652be27286fSItamar Shalev 
653be27286fSItamar Shalev 	iwl_pcie_synchronize_irqs(trans);
654be27286fSItamar Shalev }
655