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