/* Copyright (c) 2015-2017, The Linux Foundation. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 and * only version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ #define pr_fmt(fmt) "[drm:%s:%d] " fmt, __func__, __LINE__ #include #include #include #include "sde_irq.h" #include "sde_core_irq.h" irqreturn_t sde_irq(struct msm_kms *kms) { struct sde_kms *sde_kms = to_sde_kms(kms); u32 interrupts; sde_kms->hw_intr->ops.get_interrupt_sources(sde_kms->hw_intr, &interrupts); /* * Taking care of MDP interrupt */ if (interrupts & IRQ_SOURCE_MDP) { interrupts &= ~IRQ_SOURCE_MDP; sde_core_irq(sde_kms); } /* * Routing all other interrupts to external drivers */ while (interrupts) { irq_hw_number_t hwirq = fls(interrupts) - 1; generic_handle_irq(irq_find_mapping( sde_kms->irq_controller.domain, hwirq)); interrupts &= ~(1 << hwirq); } return IRQ_HANDLED; } void sde_irq_preinstall(struct msm_kms *kms) { struct sde_kms *sde_kms = to_sde_kms(kms); if (!sde_kms->dev || !sde_kms->dev->dev) { pr_err("invalid device handles\n"); return; } sde_core_irq_preinstall(sde_kms); } int sde_irq_postinstall(struct msm_kms *kms) { struct sde_kms *sde_kms = to_sde_kms(kms); int rc; if (!kms) { SDE_ERROR("invalid parameters\n"); return -EINVAL; } rc = sde_core_irq_postinstall(sde_kms); return rc; } void sde_irq_uninstall(struct msm_kms *kms) { struct sde_kms *sde_kms = to_sde_kms(kms); if (!kms) { SDE_ERROR("invalid parameters\n"); return; } sde_core_irq_uninstall(sde_kms); sde_core_irq_domain_fini(sde_kms); }