123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2014-2019, The Linux Foundation. All rights reserved.
- */
- #include <linux/interrupt.h>
- #include "ipa_i.h"
- #define INTERRUPT_WORKQUEUE_NAME "ipa_interrupt_wq"
- #define DIS_SUSPEND_INTERRUPT_TIMEOUT 5
- #define IPA_IRQ_NUM_MAX 32
- struct ipa3_interrupt_info {
- ipa_irq_handler_t handler;
- enum ipa_irq_type interrupt;
- void *private_data;
- bool deferred_flag;
- };
- struct ipa3_interrupt_work_wrap {
- struct work_struct interrupt_work;
- ipa_irq_handler_t handler;
- enum ipa_irq_type interrupt;
- void *private_data;
- void *interrupt_data;
- };
- static struct ipa3_interrupt_info ipa_interrupt_to_cb[IPA_IRQ_NUM_MAX];
- static struct workqueue_struct *ipa_interrupt_wq;
- static u32 ipa_ee;
- static void ipa3_tx_suspend_interrupt_wa(void);
- static void ipa3_enable_tx_suspend_wa(struct work_struct *work);
- static DECLARE_DELAYED_WORK(dwork_en_suspend_int,
- ipa3_enable_tx_suspend_wa);
- static spinlock_t suspend_wa_lock;
- static void ipa3_process_interrupts(bool isr_context);
- static int ipa3_irq_mapping[IPA_IRQ_MAX] = {
- [IPA_BAD_SNOC_ACCESS_IRQ] = 0,
- [IPA_UC_IRQ_0] = 2,
- [IPA_UC_IRQ_1] = 3,
- [IPA_UC_IRQ_2] = 4,
- [IPA_UC_IRQ_3] = 5,
- [IPA_UC_IN_Q_NOT_EMPTY_IRQ] = 6,
- [IPA_UC_RX_CMD_Q_NOT_FULL_IRQ] = 7,
- [IPA_PROC_TO_UC_ACK_Q_NOT_EMPTY_IRQ] = 8,
- [IPA_RX_ERR_IRQ] = 9,
- [IPA_DEAGGR_ERR_IRQ] = 10,
- [IPA_TX_ERR_IRQ] = 11,
- [IPA_STEP_MODE_IRQ] = 12,
- [IPA_PROC_ERR_IRQ] = 13,
- [IPA_TX_SUSPEND_IRQ] = 14,
- [IPA_TX_HOLB_DROP_IRQ] = 15,
- [IPA_BAM_GSI_IDLE_IRQ] = 16,
- [IPA_PIPE_YELLOW_MARKER_BELOW_IRQ] = 17,
- [IPA_PIPE_RED_MARKER_BELOW_IRQ] = 18,
- [IPA_PIPE_YELLOW_MARKER_ABOVE_IRQ] = 19,
- [IPA_PIPE_RED_MARKER_ABOVE_IRQ] = 20,
- [IPA_UCP_IRQ] = 21,
- [IPA_DCMP_IRQ] = 22,
- [IPA_GSI_EE_IRQ] = 23,
- [IPA_GSI_IPA_IF_TLV_RCVD_IRQ] = 24,
- [IPA_GSI_UC_IRQ] = 25,
- [IPA_TLV_LEN_MIN_DSM_IRQ] = 26,
- [IPA_DRBIP_PKT_EXCEED_MAX_SIZE_IRQ] = 27,
- [IPA_DRBIP_DATA_SCTR_CFG_ERROR_IRQ] = 28,
- [IPA_DRBIP_IMM_CMD_NO_FLSH_HZRD_IRQ] = 29,
- };
- static void ipa3_interrupt_defer(struct work_struct *work);
- static DECLARE_WORK(ipa3_interrupt_defer_work, ipa3_interrupt_defer);
- static void ipa3_deferred_interrupt_work(struct work_struct *work)
- {
- struct ipa3_interrupt_work_wrap *work_data =
- container_of(work,
- struct ipa3_interrupt_work_wrap,
- interrupt_work);
- IPADBG("call handler from workq for interrupt %d...\n",
- work_data->interrupt);
- work_data->handler(work_data->interrupt, work_data->private_data,
- work_data->interrupt_data);
- kfree(work_data->interrupt_data);
- kfree(work_data);
- }
- static bool ipa3_is_valid_ep(u32 ep_suspend_data)
- {
- u32 bmsk = 1;
- u32 i = 0;
- for (i = 0; i < ipa3_ctx->ipa_num_pipes; i++) {
- if ((ep_suspend_data & bmsk) && (ipa3_ctx->ep[i].valid))
- return true;
- bmsk = bmsk << 1;
- }
- return false;
- }
- static int ipa3_handle_interrupt(int irq_num, bool isr_context)
- {
- struct ipa3_interrupt_info interrupt_info;
- struct ipa3_interrupt_work_wrap *work_data;
- u32 suspend_data;
- void *interrupt_data = NULL;
- struct ipa_tx_suspend_irq_data *suspend_interrupt_data = NULL;
- int res;
- interrupt_info = ipa_interrupt_to_cb[irq_num];
- if (interrupt_info.handler == NULL) {
- IPAERR("A callback function wasn't set for interrupt num %d\n",
- irq_num);
- return -EINVAL;
- }
- switch (interrupt_info.interrupt) {
- case IPA_TX_SUSPEND_IRQ:
- IPADBG_LOW("processing TX_SUSPEND interrupt\n");
- ipa3_tx_suspend_interrupt_wa();
- suspend_data = ipahal_read_reg_n(IPA_SUSPEND_IRQ_INFO_EE_n,
- ipa_ee);
- IPADBG_LOW("get interrupt %d\n", suspend_data);
- if (ipa3_ctx->ipa_hw_type >= IPA_HW_v3_1) {
- /* Clearing L2 interrupts status */
- ipahal_write_reg_n(IPA_SUSPEND_IRQ_CLR_EE_n,
- ipa_ee, suspend_data);
- }
- if (!ipa3_is_valid_ep(suspend_data))
- return 0;
- suspend_interrupt_data =
- kzalloc(sizeof(*suspend_interrupt_data), GFP_ATOMIC);
- if (!suspend_interrupt_data) {
- IPAERR("failed allocating suspend_interrupt_data\n");
- return -ENOMEM;
- }
- suspend_interrupt_data->endpoints = suspend_data;
- interrupt_data = suspend_interrupt_data;
- break;
- default:
- break;
- }
- /* Force defer processing if in ISR context. */
- if (interrupt_info.deferred_flag || isr_context) {
- IPADBG_LOW("Defer handling interrupt %d\n",
- interrupt_info.interrupt);
- work_data = kzalloc(sizeof(struct ipa3_interrupt_work_wrap),
- GFP_ATOMIC);
- if (!work_data) {
- IPAERR("failed allocating ipa3_interrupt_work_wrap\n");
- res = -ENOMEM;
- goto fail_alloc_work;
- }
- INIT_WORK(&work_data->interrupt_work,
- ipa3_deferred_interrupt_work);
- work_data->handler = interrupt_info.handler;
- work_data->interrupt = interrupt_info.interrupt;
- work_data->private_data = interrupt_info.private_data;
- work_data->interrupt_data = interrupt_data;
- queue_work(ipa_interrupt_wq, &work_data->interrupt_work);
- } else {
- IPADBG_LOW("Handle interrupt %d\n", interrupt_info.interrupt);
- interrupt_info.handler(interrupt_info.interrupt,
- interrupt_info.private_data,
- interrupt_data);
- kfree(interrupt_data);
- }
- return 0;
- fail_alloc_work:
- kfree(interrupt_data);
- return res;
- }
- static void ipa3_enable_tx_suspend_wa(struct work_struct *work)
- {
- u32 en;
- u32 suspend_bmask;
- int irq_num;
- IPADBG_LOW("Enter\n");
- irq_num = ipa3_irq_mapping[IPA_TX_SUSPEND_IRQ];
- if (irq_num == -1) {
- WARN_ON(1);
- return;
- }
- /* make sure ipa hw is clocked on*/
- IPA_ACTIVE_CLIENTS_INC_SIMPLE();
- en = ipahal_read_reg_n(IPA_IRQ_EN_EE_n, ipa_ee);
- suspend_bmask = 1 << irq_num;
- /*enable TX_SUSPEND_IRQ*/
- en |= suspend_bmask;
- IPADBG("enable TX_SUSPEND_IRQ, IPA_IRQ_EN_EE reg, write val = %u\n"
- , en);
- ipahal_write_reg_n(IPA_IRQ_EN_EE_n, ipa_ee, en);
- ipa3_process_interrupts(false);
- IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
- IPADBG_LOW("Exit\n");
- }
- static void ipa3_tx_suspend_interrupt_wa(void)
- {
- u32 val;
- u32 suspend_bmask;
- int irq_num;
- int wa_delay;
- IPADBG_LOW("Enter\n");
- irq_num = ipa3_irq_mapping[IPA_TX_SUSPEND_IRQ];
- if (irq_num == -1) {
- WARN_ON(1);
- return;
- }
- /*disable TX_SUSPEND_IRQ*/
- val = ipahal_read_reg_n(IPA_IRQ_EN_EE_n, ipa_ee);
- suspend_bmask = 1 << irq_num;
- val &= ~suspend_bmask;
- IPADBG("Disabling TX_SUSPEND_IRQ, write val: %u to IPA_IRQ_EN_EE reg\n",
- val);
- ipahal_write_reg_n(IPA_IRQ_EN_EE_n, ipa_ee, val);
- IPADBG_LOW(" processing suspend interrupt work-around, delayed work\n");
- wa_delay = DIS_SUSPEND_INTERRUPT_TIMEOUT;
- if (ipa3_ctx->ipa3_hw_mode == IPA_HW_MODE_VIRTUAL ||
- ipa3_ctx->ipa3_hw_mode == IPA_HW_MODE_EMULATION) {
- wa_delay *= 400;
- }
- IPADBG_LOW("Delay period %d msec\n", wa_delay);
- queue_delayed_work(ipa_interrupt_wq, &dwork_en_suspend_int,
- msecs_to_jiffies(wa_delay));
- IPADBG_LOW("Exit\n");
- }
- static inline bool is_uc_irq(int irq_num)
- {
- if (ipa_interrupt_to_cb[irq_num].interrupt >= IPA_UC_IRQ_0 &&
- ipa_interrupt_to_cb[irq_num].interrupt <= IPA_UC_IRQ_3)
- return true;
- else
- return false;
- }
- static void ipa3_process_interrupts(bool isr_context)
- {
- u32 reg;
- u32 bmsk;
- u32 i = 0;
- u32 en;
- unsigned long flags;
- bool uc_irq;
- IPADBG_LOW("Enter isr_context=%d\n", isr_context);
- spin_lock_irqsave(&suspend_wa_lock, flags);
- en = ipahal_read_reg_n(IPA_IRQ_EN_EE_n, ipa_ee);
- reg = ipahal_read_reg_n(IPA_IRQ_STTS_EE_n, ipa_ee);
- while (en & reg) {
- IPADBG_LOW("en=0x%x reg=0x%x\n", en, reg);
- bmsk = 1;
- for (i = 0; i < IPA_IRQ_NUM_MAX; i++) {
- IPADBG_LOW("Check irq number %d\n", i);
- if (en & reg & bmsk) {
- IPADBG_LOW("Irq number %d asserted\n", i);
- uc_irq = is_uc_irq(i);
- /*
- * Clear uC interrupt before processing to avoid
- * clearing unhandled interrupts
- */
- if (uc_irq)
- ipahal_write_reg_n(IPA_IRQ_CLR_EE_n,
- ipa_ee, bmsk);
- /*
- * handle the interrupt with spin_lock
- * unlocked to avoid calling client in atomic
- * context. mutual exclusion still preserved
- * as the read/clr is done with spin_lock
- * locked.
- */
- spin_unlock_irqrestore(&suspend_wa_lock, flags);
- ipa3_handle_interrupt(i, isr_context);
- spin_lock_irqsave(&suspend_wa_lock, flags);
- /*
- * Clear non uC interrupt after processing
- * to avoid clearing interrupt data
- */
- if (!uc_irq)
- ipahal_write_reg_n(IPA_IRQ_CLR_EE_n,
- ipa_ee, bmsk);
- }
- bmsk = bmsk << 1;
- }
- reg = ipahal_read_reg_n(IPA_IRQ_STTS_EE_n, ipa_ee);
- /* since the suspend interrupt HW bug we must
- * read again the EN register, otherwise the while is endless
- */
- en = ipahal_read_reg_n(IPA_IRQ_EN_EE_n, ipa_ee);
- }
- spin_unlock_irqrestore(&suspend_wa_lock, flags);
- IPADBG_LOW("Exit\n");
- }
- static void ipa3_interrupt_defer(struct work_struct *work)
- {
- IPADBG("processing interrupts in wq\n");
- IPA_ACTIVE_CLIENTS_INC_SIMPLE();
- ipa3_process_interrupts(false);
- IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
- IPADBG("Done\n");
- }
- static irqreturn_t ipa3_isr(int irq, void *ctxt)
- {
- struct ipa_active_client_logging_info log_info;
- IPA_ACTIVE_CLIENTS_PREP_SIMPLE(log_info);
- IPADBG_LOW("Enter\n");
- /* defer interrupt handling in case IPA is not clocked on */
- if (ipa3_inc_client_enable_clks_no_block(&log_info)) {
- IPADBG("defer interrupt processing\n");
- queue_work(ipa3_ctx->power_mgmt_wq, &ipa3_interrupt_defer_work);
- return IRQ_HANDLED;
- }
- ipa3_process_interrupts(true);
- IPADBG_LOW("Exit\n");
- ipa3_dec_client_disable_clks_no_block(&log_info);
- return IRQ_HANDLED;
- }
- irq_handler_t ipa3_get_isr(void)
- {
- return ipa3_isr;
- }
- /**
- * ipa3_add_interrupt_handler() - Adds handler to an interrupt type
- * @interrupt: Interrupt type
- * @handler: The handler to be added
- * @deferred_flag: whether the handler processing should be deferred in
- * a workqueue
- * @private_data: the client's private data
- *
- * Adds handler to an interrupt type and enable the specific bit
- * in IRQ_EN register, associated interrupt in IRQ_STTS register will be enabled
- */
- int ipa3_add_interrupt_handler(enum ipa_irq_type interrupt,
- ipa_irq_handler_t handler,
- bool deferred_flag,
- void *private_data)
- {
- u32 val;
- u32 bmsk;
- int irq_num;
- int client_idx, ep_idx;
- IPADBG("interrupt_enum(%d)\n", interrupt);
- if (interrupt < IPA_BAD_SNOC_ACCESS_IRQ ||
- interrupt >= IPA_IRQ_MAX) {
- IPAERR("invalid interrupt number %d\n", interrupt);
- return -EINVAL;
- }
- irq_num = ipa3_irq_mapping[interrupt];
- if (irq_num < 0 || irq_num >= IPA_IRQ_NUM_MAX) {
- IPAERR("interrupt %d not supported\n", interrupt);
- WARN_ON(1);
- return -EFAULT;
- }
- IPADBG("ipa_interrupt_to_cb irq_num(%d)\n", irq_num);
- ipa_interrupt_to_cb[irq_num].deferred_flag = deferred_flag;
- ipa_interrupt_to_cb[irq_num].handler = handler;
- ipa_interrupt_to_cb[irq_num].private_data = private_data;
- ipa_interrupt_to_cb[irq_num].interrupt = interrupt;
- val = ipahal_read_reg_n(IPA_IRQ_EN_EE_n, ipa_ee);
- IPADBG("read IPA_IRQ_EN_EE_n register. reg = %d\n", val);
- bmsk = 1 << irq_num;
- val |= bmsk;
- ipahal_write_reg_n(IPA_IRQ_EN_EE_n, ipa_ee, val);
- IPADBG("wrote IPA_IRQ_EN_EE_n register. reg = %d\n", val);
- /* register SUSPEND_IRQ_EN_EE_n_ADDR for L2 interrupt*/
- if ((interrupt == IPA_TX_SUSPEND_IRQ) &&
- (ipa3_ctx->ipa_hw_type >= IPA_HW_v3_1)) {
- val = ~0;
- for (client_idx = 0; client_idx < IPA_CLIENT_MAX; client_idx++)
- if (IPA_CLIENT_IS_Q6_CONS(client_idx) ||
- IPA_CLIENT_IS_Q6_PROD(client_idx)) {
- ep_idx = ipa3_get_ep_mapping(client_idx);
- IPADBG("modem ep_idx(%d) client_idx = %d\n",
- ep_idx, client_idx);
- if (ep_idx == -1)
- IPADBG("Invalid IPA client\n");
- else
- val &= ~(1 << ep_idx);
- }
- ipahal_write_reg_n(IPA_SUSPEND_IRQ_EN_EE_n, ipa_ee, val);
- IPADBG("wrote IPA_SUSPEND_IRQ_EN_EE_n reg = %d\n", val);
- }
- return 0;
- }
- /**
- * ipa3_remove_interrupt_handler() - Removes handler to an interrupt type
- * @interrupt: Interrupt type
- *
- * Removes the handler and disable the specific bit in IRQ_EN register
- */
- int ipa3_remove_interrupt_handler(enum ipa_irq_type interrupt)
- {
- u32 val;
- u32 bmsk;
- int irq_num;
- if (interrupt < IPA_BAD_SNOC_ACCESS_IRQ ||
- interrupt >= IPA_IRQ_MAX) {
- IPAERR("invalid interrupt number %d\n", interrupt);
- return -EINVAL;
- }
- irq_num = ipa3_irq_mapping[interrupt];
- if (irq_num < 0 || irq_num >= IPA_IRQ_NUM_MAX) {
- IPAERR("interrupt %d not supported\n", interrupt);
- WARN_ON(1);
- return -EFAULT;
- }
- kfree(ipa_interrupt_to_cb[irq_num].private_data);
- ipa_interrupt_to_cb[irq_num].deferred_flag = false;
- ipa_interrupt_to_cb[irq_num].handler = NULL;
- ipa_interrupt_to_cb[irq_num].private_data = NULL;
- ipa_interrupt_to_cb[irq_num].interrupt = -1;
- /* clean SUSPEND_IRQ_EN_EE_n_ADDR for L2 interrupt */
- if ((interrupt == IPA_TX_SUSPEND_IRQ) &&
- (ipa3_ctx->ipa_hw_type >= IPA_HW_v3_1)) {
- ipahal_write_reg_n(IPA_SUSPEND_IRQ_EN_EE_n, ipa_ee, 0);
- IPADBG("wrote IPA_SUSPEND_IRQ_EN_EE_n reg = %d\n", 0);
- }
- val = ipahal_read_reg_n(IPA_IRQ_EN_EE_n, ipa_ee);
- bmsk = 1 << irq_num;
- val &= ~bmsk;
- ipahal_write_reg_n(IPA_IRQ_EN_EE_n, ipa_ee, val);
- return 0;
- }
- /**
- * ipa3_interrupts_init() - Initialize the IPA interrupts framework
- * @ipa_irq: The interrupt number to allocate
- * @ee: Execution environment
- * @ipa_dev: The basic device structure representing the IPA driver
- *
- * - Initialize the ipa_interrupt_to_cb array
- * - Clear interrupts status
- * - Register the ipa interrupt handler - ipa3_isr
- * - Enable apps processor wakeup by IPA interrupts
- */
- int ipa3_interrupts_init(u32 ipa_irq, u32 ee, struct device *ipa_dev)
- {
- int idx;
- int res = 0;
- ipa_ee = ee;
- for (idx = 0; idx < IPA_IRQ_NUM_MAX; idx++) {
- ipa_interrupt_to_cb[idx].deferred_flag = false;
- ipa_interrupt_to_cb[idx].handler = NULL;
- ipa_interrupt_to_cb[idx].private_data = NULL;
- ipa_interrupt_to_cb[idx].interrupt = -1;
- }
- ipa_interrupt_wq = create_singlethread_workqueue(
- INTERRUPT_WORKQUEUE_NAME);
- if (!ipa_interrupt_wq) {
- IPAERR("workqueue creation failed\n");
- return -ENOMEM;
- }
- /*
- * NOTE:
- *
- * We'll only register an isr on non-emulator (ie. real UE)
- * systems.
- *
- * On the emulator, emulator_soft_irq_isr() will be calling
- * ipa3_isr, so hence, no isr registration here, and instead,
- * we'll pass the address of ipa3_isr to the gsi layer where
- * emulator interrupts are handled...
- */
- if (ipa3_ctx->ipa3_hw_mode != IPA_HW_MODE_EMULATION) {
- res = request_irq(ipa_irq, (irq_handler_t) ipa3_isr,
- IRQF_TRIGGER_RISING, "ipa", ipa_dev);
- if (res) {
- IPAERR(
- "fail to register IPA IRQ handler irq=%d\n",
- ipa_irq);
- destroy_workqueue(ipa_interrupt_wq);
- ipa_interrupt_wq = NULL;
- return -ENODEV;
- }
- IPADBG("IPA IRQ handler irq=%d registered\n", ipa_irq);
- res = enable_irq_wake(ipa_irq);
- if (res)
- IPAERR("fail to enable IPA IRQ wakeup irq=%d res=%d\n",
- ipa_irq, res);
- else
- IPADBG("IPA IRQ wakeup enabled irq=%d\n", ipa_irq);
- }
- spin_lock_init(&suspend_wa_lock);
- return 0;
- }
- /**
- * ipa3_interrupts_destroy() - Destroy the IPA interrupts framework
- * @ipa_irq: The interrupt number to allocate
- * @ee: Execution environment
- * @ipa_dev: The basic device structure representing the IPA driver
- *
- * - Disable apps processor wakeup by IPA interrupts
- * - Unregister the ipa interrupt handler - ipa3_isr
- * - Destroy the interrupt workqueue
- */
- void ipa3_interrupts_destroy(u32 ipa_irq, struct device *ipa_dev)
- {
- if (ipa3_ctx->ipa3_hw_mode != IPA_HW_MODE_EMULATION) {
- disable_irq_wake(ipa_irq);
- free_irq(ipa_irq, ipa_dev);
- }
- destroy_workqueue(ipa_interrupt_wq);
- ipa_interrupt_wq = NULL;
- }
- /**
- * ipa3_suspend_active_aggr_wa() - Emulate suspend IRQ
- * @clnt_hndl: suspended client handle, IRQ is emulated for this pipe
- *
- * Emulate suspend IRQ to unsuspend client which was suspended with an open
- * aggregation frame in order to bypass HW bug of IRQ not generated when
- * endpoint is suspended during an open aggregation.
- */
- void ipa3_suspend_active_aggr_wa(u32 clnt_hdl)
- {
- struct ipa3_interrupt_info interrupt_info;
- struct ipa3_interrupt_work_wrap *work_data;
- struct ipa_tx_suspend_irq_data *suspend_interrupt_data;
- int irq_num;
- int aggr_active_bitmap = ipahal_read_reg(IPA_STATE_AGGR_ACTIVE);
- if (aggr_active_bitmap & (1 << clnt_hdl)) {
- /* force close aggregation */
- ipahal_write_reg(IPA_AGGR_FORCE_CLOSE, (1 << clnt_hdl));
- /* simulate suspend IRQ */
- irq_num = ipa3_irq_mapping[IPA_TX_SUSPEND_IRQ];
- interrupt_info = ipa_interrupt_to_cb[irq_num];
- if (interrupt_info.handler == NULL) {
- IPAERR("no CB function for IPA_TX_SUSPEND_IRQ\n");
- return;
- }
- suspend_interrupt_data = kzalloc(
- sizeof(*suspend_interrupt_data),
- GFP_ATOMIC);
- if (!suspend_interrupt_data) {
- IPAERR("failed allocating suspend_interrupt_data\n");
- return;
- }
- suspend_interrupt_data->endpoints = 1 << clnt_hdl;
- work_data = kzalloc(sizeof(struct ipa3_interrupt_work_wrap),
- GFP_ATOMIC);
- if (!work_data) {
- IPAERR("failed allocating ipa3_interrupt_work_wrap\n");
- goto fail_alloc_work;
- }
- INIT_WORK(&work_data->interrupt_work,
- ipa3_deferred_interrupt_work);
- work_data->handler = interrupt_info.handler;
- work_data->interrupt = IPA_TX_SUSPEND_IRQ;
- work_data->private_data = interrupt_info.private_data;
- work_data->interrupt_data = (void *)suspend_interrupt_data;
- queue_work(ipa_interrupt_wq, &work_data->interrupt_work);
- return;
- fail_alloc_work:
- kfree(suspend_interrupt_data);
- }
- }
|